文章目录
- 拍照并保存到 ImageView 控件
- 布局文件 notice_layout.xml
- 按钮 button_takePhoto 的点击操作
- 隐式 Intent 启动后的回调
- AndroidManifest.xml
- 从相册选取照片并在 ImageView 控件中显示
- 布局文件 notice_layout.xml
- 按钮 button_takePhoto 的点击操作
- 自定义打开相册的方法 openAlbum
- 隐式 Intent 启动后的回调
拍照并保存到 ImageView 控件
布局文件 notice_layout.xml
按钮 button_takePhoto 的点击操作
public static final int TAKE_PHOTO = 1;private ImageView picture;private Uri imageUri;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.notice_layout);picture = findViewById(R.id.picture);Button button_takePhoto = findViewById(R.id.button_takePhoto);button_takePhoto.setOnClickListener(v->{// 存储拍摄后的照片到 getExternalCacheDir() 指定的应用关联缓存目录File outputImage = new File(getExternalCacheDir(), "output_image.jpg");try {// 除首次拍照外,都需要删除原有存在的旧照片if(outputImage.exists()){outputImage.delete();}// 再创建新文件outputImage.createNewFile();} catch (IOException e) {e.printStackTrace();}// 7.0 版本后,直接使用标识本地真实路径的Uri会抛出 FileUriExposedException 异常if(Build.VERSION.SDK_INT >= 24){// getUriForFile三个参数:Context对象、任意唯一字符串、File对象// 参数二必须和AndroidManifest.xml中provider标签的authorities属性一致// 作用是将File对象转为封装过的Uri对象,提高安全性imageUri = FileProvider.getUriForFile(this,"com.example.activitytest.Activity.fileProvider", outputImage);}else{// fromFile将File对象转为标识图片本地真实路径的Uri对象imageUri = Uri.fromFile(outputImage);}// 指定开启系统相机的ActionIntent intent = new Intent("android.media.action.IMAGE_CAPTURE");// 指定图片的输出地址为之前创建的Uri对象imageUriintent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);// 隐式Intent,startActivityForResult之后回调onActivityResultstartActivityForResult(intent, TAKE_PHOTO);});}
流程
- 以 File 形式存储拍摄的照片: 存储拍摄后的照片到 getExternalCacheDir() 指定的应用关联缓存目录;(此时只有
output_image.jpg
这个文件名还没有与之对应的照片) - 将 File 对象转为 Uri 对象: 7.0 版本后使用封装过的 Uri 来替换原来标识真实路径的 Uri,增强安全性;
- 将照片的输出地址与 Uri 对象绑定: 此时才完成了 通过 Intent 跳转到相机、通过 Uri对象 将拍摄好的照片与文件名
output_image.jpg
绑定 的代码逻辑。
为什么使用应用关联缓存目录存放图片?
首先明确该目录的路径是 /sdcard/Android/data/<packge name>/cache
。
从 Android 6.0 开始,读写 SD 卡被列为危险权限,如果将图片放在 SD 卡的任何其他目录,都要进行运行时权限处理,而使用应用关联缓存目录无需进行。
隐式 Intent 启动后的回调
// startActivityForResult之后回调onActivityResult@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);switch (requestCode) {case TAKE_PHOTO:if (resultCode == RESULT_OK) {try {// 将output_image.jpg解析成Bitmap对象Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));// 设置到ImageView中picture.setImageBitmap(bitmap);} catch (FileNotFoundException e) {e.printStackTrace();}}break;default:break;}}
AndroidManifest.xml
为了兼容 4.4及之前 的系统,需要声明访问SD卡的权限:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
provider 标签:
name
属性的值是固定的authorities
属性的值必须和FileProvider.getUriForFile()
方法中参数二一致<meta-adta>
中的resource
属性的值是我们自创的文件
从相册选取照片并在 ImageView 控件中显示
布局文件 notice_layout.xml
按钮 button_takePhoto 的点击操作
public static final int CHOOSE_PHOTO = 2;protected void onCreate(Bundle savedInstanceState) {Button button_album = findViewById(R.id.button_album);button_album.setOnClickListener(v->{// WRITE_EXTERNAL_STORAGE:对SD卡读和写的权限// 相等说明用户已授权,不等说明未授权if(ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){ActivityCompat.requestPermissions(this,new String[] { Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);}else {openAlbum();}});}
请求授权 requestPermissions
的回调 onRequestPermissionsResult
:
// ActivityCompat.requestPermissions结束后回调@Overridepublic void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults){switch (requestCode) {case 1:if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {openAlbum();}break;default:}if(!ActivityCompat.shouldShowRequestPermissionRationale(this,Manifest.permission.WRITE_EXTERNAL_STORAGE)){AlertDialog.Builder dialog = new AlertDialog.Builder(this);dialog.setTitle("图库权限不可用").setMessage("请在-应用设置-权限中,允许APP使用图库权限。").setCancelable(false).setPositiveButton("立即设置", (dialog1, which) -> goToAppSetting()).setNegativeButton("取消", (dialog2, which) -> dialog2.dismiss()).show();}}
用户未授权却想授权时跳转到权限设置界面:
// 跳转到权限设置界面private void goToAppSetting() {Intent intent = new Intent();intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);Uri uri = Uri.fromParts("package", getPackageName(), null);intent.setData(uri);startActivity(intent);}
自定义打开相册的方法 openAlbum
// 打开相册private void openAlbum(){// 获取内容,具体调用哪那个程序由type属性决定Intent intent = new Intent("android.intent.action.GET_CONTENT");// 设置type属性,调用图库intent.setType("image/*");// 启动隐式Intent,跳转到相册startActivityForResult(intent, CHOOSE_PHOTO);}
隐式 Intent 启动后的回调
// startActivityForResult之后回调onActivityResult@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);switch (requestCode) {case CHOOSE_PHOTO:// 处理图片if(resultCode == RESULT_OK){// 4.4及以上系统会对Uri进行封装,需要进一步解析才能得到真实Uriif(Build.VERSION.SDK_INT >= 19){handleImageOnKitKat(data);}// 4.4以下系统可以直接获得真实Urielse{handleImageBeforeKitKat(data);}}default:break;}}
4.4以下系统可以直接获得 真实Uri:
private void handleImageBeforeKitKat(Intent data){Uri uri = data.getData();String imagePath = getImagePath(uri, null);displayImage(imagePath);}
4.4及以上系统需要进一步解析 封装的Uri 才能得到 真实Uri,想要读取视频只需要将MediaStore.Images
改为MediaStore.Video
即可:
private void handleImageOnKitKat(Intent data) {String imagePath = null;Uri uri = data.getData();// 如果是document类型的Uriif(DocumentsContract.isDocumentUri(this, uri)){// 则通过document id处理String docId = DocumentsContract.getDocumentId(uri);// 如果authority是media格式,需要分割字符串得到真正的数字idif("com.android.providers.media.documents".equals(uri.getAuthority())){// 根据":"分割docIdString id = docId.split(":")[1]; // 解析出数字格式的idString selection = MediaStore.Images.Media._ID + "=" + id;// EXTERNAL_CONTENT_URI:“主”外部存储卷的样式URI。imagePath = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);}else if("com.android.providers.downloads.documents".equals(uri.getAuthority())){Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads" +"/public_downloads"), Long.valueOf(docId));imagePath = getImagePath(contentUri, null);}// 如果是content类型的Uri} else if("content".equalsIgnoreCase(uri.getScheme())){// 则使用普通方式处理imagePath = getImagePath(uri, null);// 如果是file类型的Uri} else if("file".equalsIgnoreCase(uri.getScheme())){// 直接获取图片路径即可imagePath = uri.getPath();}displayImage(imagePath); // 根据图片路径显示图片}
进一步解析 document
、content
类型初步解析得到的的 Uri
:
// 进一步解析真实Uriprivate String getImagePath(Uri uri, String selection){String path = null;Cursor cursor = getContentResolver().query(uri, null, selection,null, null);if(cursor != null){if(cursor.moveToFirst()){path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));}cursor.close();}return path;}
在 ImageView
中显示图片:
// 在ImageView中显示图片private void displayImage(String imagePath){if(imagePath != null){Bitmap bitmap = BitmapFactory.decodeFile(imagePath);picture.setImageBitmap(bitmap);}else{Toast.makeText(this, "failed to get image", Toast.LENGTH_LONG).show();}}