java - 启动相机导致 Android 崩溃

标签 java android debugging camera crash

我关注了这个简单的应用 AndroidScannerDemo它有两个主要按钮打开相机和打开画廊。相机在我的手机 API 19 上运行良好,但当我尝试在其他设备或模拟器上启动相机时,应用程序崩溃了。

据我所知,这可能是由于许可

编辑:显然这是 asked不久前也在这里,但问题仍然存在

错误更新:根本问题来自createImageFile方法

我试着改变 //cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, tempFileUri);

cameraIntent.putExtra(ScanConstants.OPEN_INTENT_PREFERENCE, 偏好);

我可以启动相机,但拍照后立即崩溃

更新 2:我正在尝试关注此 article在我使用 fragment 的唯一问题下面提供了答案

那么我该如何改变这一行 tempFileUri = FileProvider.getUriForFile(getActivity().getApplicationContext(), "com.scanlibrary.provider",//如 list 中所定义 文件);

tempFileUri = FileProvider.getUriForFile(PickImageFragment.this, getString(R.string.file_provider_authority), 文件); 在 fragment 中!

Wrong First argument PickImageFragment

编辑:更改了 PickImageFragment 中的 openCamera() 方法

我错过了什么?

堆栈跟踪

2019-11-29 23:45:05.750 27993-27993/com.nabeeltech.capturedoc E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.nabeeltech.capturedoc, PID: 27993
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.nabeeltech.capturedoc/com.scanlibrary.ScanActivity}: java.lang.SecurityException: UID 10091 does not have permission to content://com.scanlibrary.provider/external_files/scanSample/IMG_20191129_224505.jpg [user 0]
 Caused by: java.lang.SecurityException: UID 10091 does not have permission to content://com.scanlibrary.provider/external_files/scanSample/IMG_20191129_224505.jpg [user 0]
    at com.scanlibrary.PickImageFragment.openCamera(PickImageFragment.java:131)
    at com.scanlibrary.PickImageFragment.handleIntentPreference(PickImageFragment.java:79)
    at com.scanlibrary.PickImageFragment.init(PickImageFragment.java:60)
    at com.scanlibrary.PickImageFragment.onCreateView(PickImageFragment.java:50)

PickImageFragment

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    view = inflater.inflate(R.layout.pick_image_fragment, null);
    init();
    return view;
}

private void init() {
    cameraButton = (ImageButton) view.findViewById(R.id.cameraButton);
    cameraButton.setOnClickListener(new CameraButtonClickListener());
    galleryButton = (ImageButton)
 view.findViewById(R.id.selectButton);
    galleryButton.setOnClickListener(new GalleryClickListener());
    if (isIntentPreferenceSet()) {
        handleIntentPreference();
    } else {
        getActivity().finish();
    }
}

private void handleIntentPreference() {
    int preference = getIntentPreference();
    if (preference == ScanConstants.OPEN_CAMERA) {
        openCamera();
    } else if (preference == ScanConstants.OPEN_MEDIA) {
        openMediaContent();
    }
}

public void openCamera() {
    Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    Uri tempFileUri = null;
        File file = createImageFile();
        boolean isDirectoryCreated = file.getParentFile().mkdirs();
        Log.d("", "openCamera: isDirectoryCreated: " + isDirectoryCreated);
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        tempFileUri = FileProvider.getUriForFile(getActivity().getApplicationContext(),
                "com.scanlibrary.provider", // As defined in Manifest
                file);
        cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, tempFileUri);
    } else {
        tempFileUri = Uri.fromFile(file);
        cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, tempFileUri);
    }
    if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.LOLLIPOP) {
        cameraIntent.setClipData(ClipData.newRawUri("", tempFileUri));
        cameraIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
    }
    startActivityForResult(cameraIntent, ScanConstants.START_CAMERA_REQUEST_CODE);

private File createImageFile() {
    clearTempImages();
    String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new
            Date());
    File file = new File(ScanConstants.IMAGE_PATH, "IMG_" + timeStamp +
            ".jpg");
    fileUri = Uri.fromFile(file);
    return file;
}

private void clearTempImages() {
    try {
        File tempFolder = new File(ScanConstants.IMAGE_PATH);
        for (File f : tempFolder.listFiles())
            f.delete();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

扫描常量

public class ScanConstants {

public final static int PICKFILE_REQUEST_CODE = 1;
public final static int START_CAMERA_REQUEST_CODE = 2;
public final static String OPEN_INTENT_PREFERENCE = "selectContent";
public final static String IMAGE_BASE_PATH_EXTRA = "ImageBasePath";
public final static int OPEN_CAMERA = 4;
public final static int OPEN_MEDIA = 5;
public final static String SCANNED_RESULT = "scannedResult";
public final static String IMAGE_PATH = Environment
        .getExternalStorageDirectory().getPath() + "/scanSample";

public final static String SELECTED_BITMAP = "selectedBitmap";
}

最佳答案

您已经编写了很好的代码 Fileprovider.getUriforFile,但是您是否声明了所需的权限。

解决这个问题的唯一方法是向所有可能需要它的包授予权限,如下所示:

List<ResolveInfo> resInfoList = context.getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
    String packageName = resolveInfo.activityInfo.packageName;
    context.grantUriPermission(packageName, uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}

如果以上没有解决问题,我建议引用这个 article由 Lorenzo Quiroli 解决了旧 Android 版本的这个问题。

他发现需要手动设置Intent的ClipData,并为它设置权限,像这样:

if ( Build.VERSION.SDK_INT <= Build.VERSION_CODES.LOLLIPOP ) {
    takePictureIntent.setClipData( ClipData.newRawUri( "", photoURI ) );
    takePictureIntent.addFlags( Intent.FLAG_GRANT_WRITE_URI_PERMISSION|Intent.FLAG_GRANT_READ_URI_PERMISSION );
}

关于java - 启动相机导致 Android 崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59111931/

相关文章:

java - 发送复杂的 JSON 对象

java - Android RSS 阅读器 - 教程错误

java - 安卓 : Firebase push notification not received even though onMessageReceived method is triggered

java - 如何在后台线程中正确执行SQL查询?

Go 版本 1.17.6 对于该版本的 Delve 来说太旧了(最低支持版本 1.18,使用 --check-go-version=false 抑制此错误)

java - 单击按钮打开 servlet 时应用程序名称从 url 中删除

java - XML 中定义的 boolean 值。如何在 Java 中引用?

android - 如何使用 AsyncStorage React Native 缓存 API 数据

http - Web 应用程序开发 - 如何让入站 http 请求命中多个环境

windows - nt!KiSystemCall64 中的断点设置不起作用