700字范文,内容丰富有趣,生活中的好帮手!
700字范文 > android调用系统相机进行拍照 android调用系统相机拍照

android调用系统相机进行拍照 android调用系统相机拍照

时间:2018-11-01 09:05:08

相关推荐

android调用系统相机进行拍照 android调用系统相机拍照

获取缩略图

直接调取相机拍照,无需任何权限,但是只能获取到缩略图

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

if (takePictureIntent.resolveActivity(getPackageManager()) != null) {//判断是否有相机应用

startActivityForResult(takePictureIntent, REQ_THUMB);

}

@Override

protected void onActivityResult(int requestCode, int resultCode, Intent data) {

super.onActivityResult(requestCode, resultCode, data);

switch (requestCode) {

case REQ_THUMB://返回结果

if (resultCode != Activity.RESULT_OK) return;

Bundle extras = data.getExtras();

Bitmap imageBitmap = (Bitmap) extras.get("data");

mImageView.setImageBitmap(imageBitmap);

break;

}

}

图片很模糊。一般不采取此种方式

保存全尺寸照片

调取相机拍照保存一个全尺寸的照片,必须提供完整的文件名,相机应用自动保存照片。此时也无需任何权限,创建一个空的临时文件用来保存图片,使用日期时间戳新照片返回一个唯一的文件名

String mCurrentPhotoPath;

private File createImageFile() throws IOException {

// Create an image file name

String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.CHINA).format(new Date());

String imageFileName = "JPEG_" + timeStamp + "_";

//.getExternalFilesDir()方法可以获取到 SDCard/Android/data/你的应用的包名/files/ 目录,一般放一些长时间保存的数据

File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES);

//创建临时文件,文件前缀不能少于三个字符,后缀如果为空默认未".tmp"

File image = File.createTempFile(

imageFileName, /* 前缀 */

".jpg", /* 后缀 */

storageDir /* 文件夹 */

);

mCurrentPhotoPath = "file:" + image.getAbsolutePath();

return image;

}

每执行一次就会创建一个空的文件

利用上述方法创建的文件只能是自己的app访问,随着app的卸载,文件也会删除。对于Android N以下,文件直接Uri.fromFile(file)就可以直接使用,Audroid N 即编译app的版本compileSdkVersion 24时,此时会报出FileUriExposedException异常,解释如下:

对于面向 Android N 的应用,Android 框架执行的StrictMode,API 禁止向您的应用外公开 file://URI。

如果一项包含文件 URI 的 Intent 离开您的应用,应用失败,并出现 FileUriExposedException异常。

若要在应用间共享文件,您应发送一项 content://URI,并授予 URI 临时访问权限。

进行此授权的最简单方式是使用FileProvider类。 如需有关权限和共享文件的更多信息,

请参阅共享文件。

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

// Ensure that there's a camera activity to handle the intent

if (takePictureIntent.resolveActivity(getPackageManager()) != null) {//判断是否有相机应用

// Create the File where the photo should go

File photoFile = null;

try {

photoFile = createImageFile();//创建临时图片文件

} catch (IOException ex) {

ex.printStackTrace();

}

// Continue only if the File was successfully created

if (photoFile != null) {

//FileProvider 是一个特殊的 ContentProvider 的子类,

//它使用 content:// Uri 代替了 file:/// Uri. ,更便利而且安全的为另一个app分享文件

Uri photoURI = FileProvider.getUriForFile(this,

"com.example.android.fileprovider",

photoFile);

takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);

startActivityForResult(takePictureIntent, REQUEST_TAKE_PHOTO);

}

}

现在,需要配置FileProvider。在应用程序的清单,提供者添加到您的应用程序,authorities="applicationId.fileprovider",使用时

Uri photoURI = FileProvider.getUriForFile(context,"applicationId.fileprovider", photoFile)

清单中authorities 和 参数authority保持一致。

android:name="android.support.v4.content.FileProvider"

android:authorities="com.youga.capturingphotos.fileprovider"

android:exported="false"

android:grantUriPermissions="true">

android:name="android.support.FILE_PROVIDER_PATHS"

android:resource="@xml/file_paths"/>

@Override

protected void onActivityResult(int requestCode, int resultCode, Intent data) {

super.onActivityResult(requestCode, resultCode, data);

switch (requestCode) {

case REQUEST_TAKE_PHOTO://返回结果

if (resultCode != Activity.RESULT_OK) return;

mImageView.setImageBitmap(BitmapFactory.decodeFile(mCurrentPhotoPath));

break;

}

}

关于FileProvider

FileProvider 是 ContentProvider 的一个特殊的子类,它有利于安全地分享应用相关的文件,通过对一个文件创建content:// Uri而不是file:/// Uri。

由于FileProvider的默认功能包括文件的content URI的生成,你并不需要在代码中定义一个子类。相反,你可以在你的应用中包含一个FileProvider通过在XML文件中指定它。对于指定FileProvider,添加一个元素在你应用的清单文件中。设置android:name属性为android.support.v4.content.FileProvider。根据你控制的域名设置android:authorities属性为一个URI authority(authorities可以随意填写,但是要保证使用时与authority保持一致,推荐applicationId.fileprovider,以免定义重复)。设置android:exported属性为false;FileProvider不需要公开。设置android:grantUriPermissions属性为true,为了允许你进行临时访问文件的授权。

android:name="android.support.v4.content.FileProvider"

android:authorities="com.youga.fileprovider"

android:exported="false"

android:grantUriPermissions="true">

android:name="android.support.FILE_PROVIDER_PATHS"

android:resource="@xml/file_paths"/>

一个FileProvider只能生成一个content URI 对应你事先指定目录下的文件。对于指定一个目录,使用元素的子元素,在XML中指定它的存储区域和路径。例如,下面的paths元素告诉FileProvider你打算请求你的私有文件区域的images/子目录的content URIs

file-path表示你应用内部存储区域的文件的子目录。这个子目录和getFilesDir()的返回值一样。external-path表示你应用外部存储区域的文件的子目录。这个子目录和getExternalFilesDir()的返回值一样。cache-path表示你应用内部存储区域的缓存子目录。这个子目录的根目录和getCacheDir()的返回值一样。(如果你修改了provider和paths中的值,需要把应用卸载重装或者开关机一下才能看到变化。)

照片添加到图库

上述保存全尺寸图片时,创建图片临时文件在目录getExternalFilesDir()中,图库无法访问到,我们可以直接创建文件在Environment.getExternalStorageDirectory()目录下;然后发送一个广播让图库更新就好了。当然也可以直接创建文件在图库目录下(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM)),即DCIM。此时需要SD卡读写权限。

File path = Environment.getExternalStoragePublicDirectory(

Environment.DIRECTORY_DCIM);

// Create an image file name

Log.i(TAG, "path:" + path.getAbsolutePath());

String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.CHINA).format(new Date());

String imageFileName = "JPEG_" + timeStamp;

File image = File.createTempFile(

imageFileName, /* 前缀 */

".jpg", /* 后缀 */

path /* 文件夹 */

);

mPublicPhotoPath = image.getAbsolutePath();

同时发送广播

Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);

File f = new File(mPublicPhotoPath);

Uri contentUri = Uri.fromFile(f);

mediaScanIntent.setData(contentUri);

sendBroadcast(mediaScanIntent);

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。