Android开发在Activity外申请权限调用相机打开相册

问题描述:android

  最近在项目中遇到一个须要调用相册和打开相机的需求,可是,在Android 6.0之后,调用相册属于危险权限,须要开发者动态获取,这就意味着咱们申请权限是与Activity绑定的,但若是一个App中须要多个地方请求打开相册,那咱们要每一个地方都要写一遍打开相册的程序吗(固然你能够Ctrl c v),可是,这对于任何一个有追求的程序员来讲,都是不恰当的,因此咱们要定义一个公共接口,作到在任何一个须要调用打开相册的地方随时调用,增长了代码的复用性。好记性不如烂笔头,把这个过程记录下来,let's go!!程序员

解决思路:ide

  既然申请权限是与Activity绑定的,那么咱们就建立一个的Activity专门用于完成打开相册/相机,申请权限的任务,当其它Activity须要用到这个功能时,直接跳转到这个Activity,完成任务后,再返回照片的真实路径就好了,Ok,思路有了,话很少说,直接撸码。this

1. 首先建立一个Activity专门用于申请权限和打开相册相机的功能,这里我命名为 SelectImageActivityspa

打开相机的程序以下:第一个方法是打开相机,第二个方法是建立一个图片文件,这种写法是Android官方写法,这里定义了一个 currentPhotoPath 用来保存图片的路径,后面须要返回给调用它的ativitycode

// 调用相机
    private void selectImageFromCamera() {
        Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);  // 初始化打开相机的意图
        if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
            File photoFile = null;
            try {
                photoFile = createImageFile();
            } catch (IOException e) {
                e.printStackTrace();
            }
            if (photoFile != null) {
                Uri photoURI = FileProvider.getUriForFile(this, "com.example.encryption.fileProvider", photoFile);
                takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);
                startActivityForResult(takePictureIntent, SELECT_FROM_CAMERA);
            }
        }
    }


    // 建立图片文件
    private File createImageFile() throws IOException {
        String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());  // 文件名以时间命名
        String imageFileName = "JPEG_" + timeStamp + "_";  // 完善文件名
        File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES);  // 获取文件存储路径
        File image = File.createTempFile(imageFileName, ".jpg", storageDir);
        currentPhotoPath = image.getAbsolutePath();
        return image; // 返回文件
    }

接下来就是调用相册的代码了,前面提到过,调用相册须要动态申请权限,虽然参考Android官方提供的代码不申请也是能够正常打开的,可是获取不到图片的真实路径(多是我技术不到位),这里就申请一次吧orm

 // 询问用户是否受权
    private void selectImageFromAlbum() {
        // 询问用户是否授予权限
        if (ContextCompat.checkSelfPermission(SelectImageActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(SelectImageActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, 1);  // 请求权限
        } else {  // 已经授予权限
            openAlbum();  // 调用相册
        }

    }

    // 判断用户是否授予权限
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull @NotNull String[] permissions, @NonNull @NotNull int[] grantResults) {
        if (requestCode == 1) {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {  // 用户给予权限
                openAlbum();
            } else {
                Toast.makeText(SelectImageActivity.this, "请授予读取相册权限!", Toast.LENGTH_SHORT).show();
            }
        }
    }

    // 从相册选取图片处理
    private String handleImage(Intent data) {
        String imagePath = null;
        Uri uri = data.getData();
        if (DocumentsContract.isDocumentUri(this, uri)) {
            String docId = DocumentsContract.getDocumentId(uri);
            if ("com.android.providers.media.documents".equals(uri.getAuthority())) {
                String id = docId.split(":")[1];
                String selection = MediaStore.Images.Media._ID + "=" + id;
                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.parseLong(docId));
                imagePath = getImagePath(contentUri, null);
            }
        } else if ("content".equalsIgnoreCase(uri.getScheme())) {
            imagePath = getImagePath(uri, null);
        } else if ("file".equalsIgnoreCase(uri.getScheme())) {
            imagePath = uri.getPath();
        }
        return imagePath;  // 用户选择图片的真实地址
    }


    // 得到相册中图片的真实路径
    private String getImagePath(Uri externalContentUri, String selection) {
        String path = null;
        Cursor cursor = getContentResolver().query(externalContentUri, null, selection, null, null);
        if (cursor != null) {
            if (cursor.moveToNext()) {
                path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
            }
            cursor.close();
        }
        return path;
    }

    // 打开相册
    private void openAlbum() {
        Intent intent = new Intent("android.intent.action.GET_CONTENT");
        intent.setType("image/*");
        startActivityForResult(intent, SELECT_FROM_ALBUM); // 跳转意图
    }

调用完相册和相机后须要返回到 SelectImageActivity进行处理,这里咱们重写 onActivityResult 方法处理结果,为何只须要处理相册的返回结果呢?由于相机拍摄图片的路径咱们已经知道了,保存在 currentPhotoPath 了,因此这里只处理从相册中选择图片的地址blog

    //Activity返回结果处理
    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable @org.jetbrains.annotations.Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        String imagePath = null;
        if (resultCode == RESULT_OK) {  // 返回码为真
            if (requestCode == SELECT_FROM_ALBUM) {
                assert data != null;
                currentPhotoPath = handleImage(data);
            }
            returnImagePath(currentPhotoPath);
        }else {
            finish();  // 若是返回值不是RESULT_OK, 这种状况多是用户放弃选择图片,此时直接结束该activity不然会显示当前activity
        }
    }

如今咱们已经能获取拍摄图片或者用户选择图片的真实地址了,那么咱们须要把这个地址返回给调用它的Activity,代码实现以下继承

 private void returnImagePath(String imgPath) {
        Intent intent = new Intent();
        intent.putExtra("imagePath", imgPath);
        setResult(RESULT_OK, intent);
        finish();
    }

好了,基本完成了,如今咱们能够获取到图片的真实地址并进行返回,可是还须要知道调用的Activity是要打开相册的仍是打开相机,所以咱们须要接收上一级Activity发送过来的信息,代码以下接口

 @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_select_image);

        int imageWhere = getIntent().getIntExtra("imageFromWhere", 0);
        switch (imageWhere) {
            case SELECT_FROM_CAMERA:  // 拍摄新图片
                selectImageFromCamera();
                break;
            case SELECT_FROM_ALBUM:  // 从相册中选取
                openAlbum();
                break;
        }
    }

到这里,咱们接口以及设置完毕了,但因为SelectImageActivity是个空白的Activity,跳转过程会有空白一闪而过,对于“高质量”程序员,这是不能忍受的,因此我这里把SelectImageActivity设置为透明,这样就不会有空白了,怎么设置呢?,只须要在AndroidManifest文件中把当前的Activity后面加上这句

<activity android:name="function.SelectImageActivity" android:theme="@android:style/Theme.Translucent"/>

注意,若是你使用这种方法,SelectImageActivity不能继承AppCompatActivity,而是向我同样继承Activity,不然会报错,到这里SelectImageActivity的配置完成,下面咱们须要调用一下,调用SelectImageActivity的Activity中的代码以下

 
Intent intent = new Intent(EncryptionActivity.this, SelectImageActivity.class);

case R.id.tv_image_select_from_album: // 从相册中选取
    intent.putExtra("imageFromWhere", SELECT_FROM_ALBUM);     startActivityForResult(intent, 1);
    break; case R.id.tv_image_select_from_camera: // 拍摄新图片
    intent.putExtra("imageFromWhere", SELECT_FROM_CAMERA);
    startActivityForResult(intent,
1);
    break;

ok,这里咱们在须要调用的Activity中发送两个整形数字,来告诉SelectImageActivity咱们是要调用相机仍是打开相册,当SelectImageActivity完成并返回后,咱们就能在onActivityResult中接收到图片的真实路径了

 @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable @org.jetbrains.annotations.Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode == RESULT_OK){
            Bitmap bitmap = BitmapFactory.decodeFile(data.getStringExtra("imagePath"));
            mIvSelectImage.setImageBitmap(bitmap);
        }
    }

这里,我直接把图片放在一个Image View中,所有过程完成!

Android开发新人,写的很差,大神勿喷,一个简单的案例,但愿可以帮助到你

相关文章
相关标签/搜索