Android11 相机拍照权限,以及解决resolveActivity返回null

一、配置拍照和读写权限

<uses-permission android:name="android.permission.CAMERA"/>
<uses-feature android:name="android.hardware.camera" />

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE"/>

二、手动申请权限

        implementation 'com.permissionx.guolindev:permissionx:1.4.0'

        1、手动申请读写,拍照权限

PermissionX.init(this)
    .permissions(Manifest.permission.WRITE_EXTERNAL_STORAGE,
                    Manifest.permission.READ_EXTERNAL_STORAGE,
                    Manifest.permission.CAMERA)
            .request((allGranted, grantedList, deniedList) -> {
                if(allGranted){

                }
            });

        2、手动申请文件管理权限

if (Build.VERSION.SDK_INT >= 30 && !Environment.isExternalStorageManager()) {
    Intent intent = new Intent(Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION);
    intent.setData(Uri.parse("package:" + getPackageName()));
    startActivity(intent);
}

三、Manifest中配置queries(解决resolveActivity为null)

<application
    ...>
    ...
</application>

<queries>
    <intent>
        <action android:name="android.media.action.IMAGE_CAPTURE"/>
    </intent>
    <intent>
        <action android:name="android.media.action.VIDEO_CAPTURE"/>
    </intent>
</queries>

四、Manifest中配置provider

<application
    ...>
    <provider
        android:name="androidx.core.content.FileProvider"
        android:authorities="你的package包名.fileprovider"
        android:exported="false"
        android:grantUriPermissions="true"
        tools:replace="android:authorities">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            tools:replace="android:resource"
            android:resource="@xml/file_paths" />
    </provider>
</application>

五、配置file_paths文件

<?xml version="1.0" encoding="utf-8"?>
<paths>
    <!--目录为:Environment.getExternalStorageDirectory()-->
    <external-path
        name="download"
        path="Download" />

    <external-path name="external_files" path="." />
</paths>

六、调用相机(FileUtil类)

public class FileUtil {
    public static String takePhoto(Activity activity) { //照相
        //调用相机
        Intent takePhotoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//用来打开相机的Intent
        if(takePhotoIntent.resolveActivity(activity.getPackageManager())!=null){
            //Constant.PIC_DIR = Environment.getExternalStorageDirectory().getAbsolutePath()+"/Download"
            File file=new File(Constant.PIC_DIR);
            if(!file.exists())file.mkdir();
            // 保存路径
            file = getFile(Constant.PIC_DIR, System.currentTimeMillis()+ ".jpg");
            Uri uri;
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                //设置7.0以上共享文件,分享路径定义在xml/file_paths.xml
                uri = FileProvider.getUriForFile(activity, "你的package包名.fileprovider", file);
            } else {
                // 7.0以下,共享文件
                uri = Uri.fromFile(file);   // 将路径转换为Uri对象
            }
            takePhotoIntent.putExtra(MediaStore.EXTRA_OUTPUT, uri);    // 表示录制完后保存的录制,如果不写,则会保存到默认的路径,在onActivityResult()的回调,通过intent.getData中返回保存的路径
            //Constant.REQ_CODE为任意整数,用于回调
            activity.startActivityForResult(takePhotoIntent, Constant.REQ_CODE);  // 跳转
            return file.getAbsolutePath();
        }
        return null;
    }

    public static File getFile(String filePath, String fileName){
        File dir = new File(filePath);
        if (!dir.exists()){//如果不存在,进行创建
            dir.mkdirs();
        }
        File soundFile = new File(dir,fileName);
        if (!soundFile.exists()){
            try {
                soundFile.createNewFile();
            }catch (IOException e){
                e.printStackTrace();
            }
        }
        return soundFile;
    }

    //通过URI获取file
    public static File getFileFromContentUri(Uri contentUri, Context context) {
        if (contentUri == null) {
            return null;
        }
        File file = null;
        String filePath;
        String[] projection = { MediaStore.Files.FileColumns.DATA };
        ContentResolver contentResolver = context.getContentResolver();
        Cursor cursor = contentResolver.query(contentUri, projection, null,
                null, null);
        if (cursor != null && cursor.moveToFirst()) {
            filePath = cursor.getString(0);
            cursor.close();
            if (!filePath.isEmpty()) {
                file = new File(filePath);
            }
        }
        return file;
    }
}

七、处理拍照 / 本地图片回调(Activity类)

private String curSelectPhoto = "";

//启动相机
private void takePhoto(){
    curSelectPhoto = FileUtil.takePhoto(this);
}

//从本地获取图片
private void getPhotoFromLocal(){
    Intent i = new Intent();
    i.setAction(Intent.ACTION_PICK);
    i.setType("image/*");
    //Constant.PHOTO_LOCAL为任意整数,用于获取本地图片回调
    startActivityForResult(i, Constant.PHOTO_LOCAL);
}

@Override
public void onActivityResult(final int requestCode, int resultCode, final Intent data) {
    super.onActivityResult(requestCode,resultCode,data);
    if(resultCode==RESULT_OK){
        ThreadManager.getThreadPool().exeute(() -> {
            File file=null;
            if(requestCode==Constant.REQ_CODE){    //拍照
                file = new File(curSelectPhoto);
            }else{  //从本地上传
                Uri uri = data.getData();  //图片的保存路径
                file = FileUtil.getFileFromContentUri(uri,getApplicationContext());
            }
        });
    }
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android 平台上调用相机拍照可以通过使用 Camera API 或者更现代的 Camera2 API 来实现。下面是一个简单的示例代码,演示如何使用 Camera API 来调用相机拍照: 首先,在 AndroidManifest.xml 文件中添加相机权限: ```xml <uses-permission android:name="android.permission.CAMERA" /> ``` 然后,在你的活动(Activity)中,添加以下代码: ```java import android.Manifest; import android.content.Intent; import android.content.pm.PackageManager; import android.graphics.Bitmap; import android.os.Bundle; import android.provider.MediaStore; import android.view.View; import android.widget.Button; import android.widget.ImageView; import androidx.annotation.NonNull; import androidx.appcompat.app.AppCompatActivity; import androidx.core.app.ActivityCompat; import androidx.core.content.ContextCompat; public class MainActivity extends AppCompatActivity { private static final int REQUEST_CAMERA_PERMISSION = 200; private static final int REQUEST_IMAGE_CAPTURE = 1; private Button captureButton; private ImageView imageView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); captureButton = findViewById(R.id.capture_button); imageView = findViewById(R.id.image_view); captureButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.CAMERA}, REQUEST_CAMERA_PERMISSION); } else { openCamera(); } } }); } private void openCamera() { Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); if (takePictureIntent.resolveActivity(getPackageManager()) != null) { startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE); } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == RESULT_OK) { Bundle extras = data.getExtras(); Bitmap imageBitmap = (Bitmap) extras.get("data"); imageView.setImageBitmap(imageBitmap); } } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { if (requestCode == REQUEST_CAMERA_PERMISSION) { if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { openCamera(); } } } } ``` 上述代码中,我们首先检查相机权限是否已经被授予,如果没有,则请求相机权限。然后,当用户点击拍照按钮时,我们打开相机应用,并在拍照完成后将结果显示在 ImageView 中。 请注意,上述代码仅涵盖了基本的相机调用和权限请求,实际开发中可能还需要处理其他方面的逻辑和错误情况。此外,由于 Android 平台的版本差异,还可能需要根据目标设备的版本进行适当的调整。希望这个示例能帮助到你!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值