Android开发(3) | 权限和内容提供器的应用——调用相机和相册


拍照并保存到 ImageView 控件

布局文件 notice_layout.xml

在这里插入图片描述

按钮 button_takePhoto 的点击操作

    public static final int TAKE_PHOTO = 1;
    private ImageView picture;
    private Uri imageUri;

    @Override
    protected 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);
            }

			// 指定开启系统相机的Action
            Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
            // 指定图片的输出地址为之前创建的Uri对象imageUri
            intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
            // 隐式Intent,startActivityForResult之后回调onActivityResult
            startActivityForResult(intent, TAKE_PHOTO);
        });
    }

流程

  1. 以 File 形式存储拍摄的照片: 存储拍摄后的照片到 getExternalCacheDir() 指定的应用关联缓存目录;(此时只有 output_image.jpg 这个文件名还没有与之对应的照片)
  2. 将 File 对象转为 Uri 对象: 7.0 版本后使用封装过的 Uri 来替换原来标识真实路径的 Uri,增强安全性;
  3. 将照片的输出地址与 Uri 对象绑定: 此时才完成了 通过 Intent 跳转到相机、通过 Uri对象 将拍摄好的照片与文件名 output_image.jpg 绑定 的代码逻辑。

为什么使用应用关联缓存目录存放图片?

首先明确该目录的路径是 /sdcard/Android/data/<packge name>/cache

从 Android 6.0 开始,读写 SD 卡被列为危险权限,如果将图片放在 SD 卡的任何其他目录,都要进行运行时权限处理,而使用应用关联缓存目录无需进行。

隐式 Intent 启动后的回调

	// startActivityForResult之后回调onActivityResult
    @Override
    protected 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结束后回调
    @Override
    public 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
    @Override
    protected 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进行封装,需要进一步解析才能得到真实Uri
                    if(Build.VERSION.SDK_INT >= 19){
    
    
                        handleImageOnKitKat(data);
                    }
                    // 4.4以下系统可以直接获得真实Uri
                    else{
    
    
                        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类型的Uri
        if(DocumentsContract.isDocumentUri(this, uri)){
    
    
            // 则通过document id处理
            String docId = DocumentsContract.getDocumentId(uri);
            // 如果authority是media格式,需要分割字符串得到真正的数字id
            if("com.android.providers.media.documents".equals(uri.getAuthority())){
    
    
                // 根据":"分割docId
                String id = docId.split(":")[1]; // 解析出数字格式的id
                String 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); // 根据图片路径显示图片
    }

进一步解析 documentcontent 类型初步解析得到的的 Uri

    // 进一步解析真实Uri
    private 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();
        }
    }

猜你喜欢

转载自blog.csdn.net/Jormungand_V/article/details/123285249