我尝试使用firebase存储示例应用程序构建storage app,当我拍照时,我总是在onActivityResult(..)中获得RESUL_CANCELED。 这是代码:
@AfterPermissionGranted(RC_STORAGE_PERMS)
private void launchCamera() {
Log.d(LOG_TAG, "launchCamera");
// Check that we have permission to read images from external storage.
String perm = Manifest.permission.WRITE_EXTERNAL_STORAGE;
if (!EasyPermissions.hasPermissions(this, perm)) {
EasyPermissions.requestPermissions(this, getString(R.string.rationale_storage),
RC_STORAGE_PERMS, perm);
return;
}
// Choose file storage location, must be listed in res/xml/file_paths.xml
File externalDir = Environment.getExternalStorageDirectory();
File file = new File(externalDir, "photos/" + UUID.randomUUID().toString() + ".jpg");
// Create content:// URI for file, required since Android N
// See: https://developer.android.com/reference/android/support/v4/content/FileProvider.html
mFileUri = FileProvider.getUriForFile(this,
"com.google.firebase.quickstart.firebasestorage.fileprovider", file);
// Create and launch the intent
Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, mFileUri);
startActivityForResult(takePictureIntent, RC_TAKE_PICTURE);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
Log.d(LOG_TAG, "onActivityResult:" + requestCode + ":" + resultCode + ":" + data);
if (requestCode == RC_TAKE_PICTURE) {
if (resultCode == RESULT_OK) {
if (mFileUri != null) {
uploadFromUri(mFileUri);
} else {
Log.w(LOG_TAG, "File URI is null");
}
} else {
Toast.makeText(this, "Taking picture failed.", Toast.LENGTH_SHORT).show();
}
}
}
logs:onActivityResult:101:0:Intent {}
答案 0 :(得分:7)
所以这取决于Android的版本?
这取决于你的意思和错误是什么。
正如我在评论中指出的那样,ACTION_IMAGE_CAPTURE
实施有a history of bugs。存在大量此类错误,不一定与Android OS版本相关。
但是,您的代码采用了一种有趣的EXTRA_OUTPUT
方法:将FileProvider
与外部存储上的位置结合使用。长期,FileProvider
是一个很好的答案,Android N is starting to ban file:
Uri
values。但是,有许多相机应用会遇到content:
Uri
值的问题,因为这些相机应用的开发人员并不期望这样的值。例如,谷歌自己的相机应用程序在ACTION_VIDEO_CAPTURE
content:
值上Uri
失败,至少在几个月之前,尽管ACTION_IMAGE_CAPTURE
没问题。
更糟糕的是,应用程序无法宣传它支持的额外计划。使用Intent
的“数据”方面,<intent-filter>
可以通告支持的方案,但这不适用于额外的方案。因此,当您在ACTION_IMAGE_CAPTURE
中使用content:
Uri
调用EXTRA_OUTPUT
时,您不仅限于支持content:
Uri
的相机应用值。
您可以考虑暂时将targetSdkVersion
降至24以下,然后使用Uri.fromFile(file)
代替FileProvider.getUriForFile(...)
,看看会发生什么。如果您的原始相机应用程序工作正常,那么该应用程序不能正确支持content:
Uri
值,这就是导致原始问题的原因。
总的来说,我建议使用ACTION_IMAGE_CAPTURE
的开发人员将他们的targetSdkVersion
保持在24以下并使用file:
Uri
值几年,直到更高的百分比为止用户安装了支持content:
Uri
值的相机应用。或者,让您的用户在现有的相机应用程序(通过ACTION_IMAGE_CAPTURE
)和您构建到自己的应用程序中的某些相机功能(例如,使用my library)之间进行选择,以增加的可能性会起作用。
答案 1 :(得分:0)
添加一些代码,调用添加到@CommonsWare的平滑解决方法。 检查API和处理是否合适。
private void startCamera() {
Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
File photoFile = null;
try {
photoFile = ImgUtils.createTempImageFile(this);
} catch (IOException ex) {
ex.printStackTrace();
}
if (photoFile != null) {
mTempPhotoPath = photoFile.getAbsolutePath();
if(Build.VERSION.SDK_INT < 24) {
//create Uri with 'file://' prefix
tempImgUri = Uri.fromFile(photoFile);
}else{
//create Uri with 'content://' prefix
tempImgUri = FileProvider.getUriForFile(this,
FILE_PROVIDER_AUTHORITY,
photoFile);
}
mImgUriString = tempImgUri.toString();
// Add the URI so the camera can store the image
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, tempImgUri);
startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
}
}