29

设备:模拟器像素 3a - Android 11

代码:

    final List<Intent> cameraIntents = new ArrayList<Intent>();
    final Intent captureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    final List<ResolveInfo> listCam = 
    context.getPackageManager().queryIntentActivities(captureIntent, 0);

使用时:

targetSdkVersion 30
compileSdkVersion 30

listCam 大小为 0

当更改为:

compileSdkVersion 29

listCam 大小为 1 - 应该是。

使用以下代码:

    val captureIntent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
    baseActivity.startActivity(captureIntent)

工作正常并显示相机应用程序。

知道为什么 queryIntentActivities 没有返回相机意图吗?

谢谢!

4

4 回答 4

78

Android 11 改变了应用查询和与其他应用交互的方式。

文档

返回有关其他应用程序的结果的PackageManager方法,例如queryIntentActivities(),根据调用应用程序的<queries>声明进行过滤。

所以你需要<queries>在你的声明AndroidManifest.xml

<manifest package="com.example">
    <queries>
        <intent>
            <action android:name="android.media.action.IMAGE_CAPTURE" />
        </intent>
    </queries>
    ...
</manifest>
于 2020-08-04T12:21:48.193 回答
25

packageManager.queryIntentActivities(intent, 0)如果您的应用正在运行,将返回一个EMPTY列表targetSdkVersion 30

要解决此问题,您必须<queries>在清单中使用queryIntentActivities()根据调用应用程序的声明进行过滤。

修复图像捕获 + 图像上传以使用 Android“范围存储”

该问题可能与新的包可见性(https://developer.android.com/about/versions/11/privacy/package-visibility)有关。在所有更新(至少是 Android Studio 4.1)之后,尝试在您的清单中添加显示您的应用需要执行的操作。

就我而言,当我 为 CAMERA添加IMAGE_CAPTURE 、为 GALLERY 添加GET_CONTENT(如果您想要视频,则获取文件更改 mimeType)、为 GALLERY 添加PICK(如果您想要视频,应该更改 mimetype) 为 GALLERY 添加选择器(如果有人有其他图像),问题就消失了浏览器)

您还可以在 logcat 中检查您必须添加哪些查询(应包含“BLOCKED”或“无权限”。错误是因为ImagePickerModule当您在 Intent 中没有resolveActivity返回权限时null(您可以对其进行评论以检查更好的错误startActivityForResult

加入<query>_AndroidManifest.xml

<manifest>
.....
.....
<queries>
    <!-- Browser -->
    <intent>
        <action android:name="android.intent.action.VIEW" />
        <data android:scheme="http" />
    </intent>
    <!-- Camera -->
    <intent>
        <action android:name="android.media.action.IMAGE_CAPTURE" />
    </intent>
    <!-- Gallery -->
    <intent>
        <action android:name="android.intent.action.GET_CONTENT" />
        <data android:mimeType="image/*" />
    </intent>
    <intent>
        <action android:name="android.intent.action.PICK" />
        <data android:mimeType="image/*" />
    </intent>
    <intent>
        <action android:name="android.intent.action.CHOOSER" />
     </intent>
</queries>
.....
.....
</manifest>
于 2021-04-29T11:59:07.520 回答
7

我的 Android 11 解决方案,用于获取 ResolveInfo 列表。

  • 当我们仅通过 MediaStore.ACTION_IMAGE_CAPTURE 过滤器扫描时,我们将只得到一个!应用记录 - 系统默认相机应用。
  • 为了使用其他相机应用程序,我们需要通过包名称指定每个应用程序,并为其提供 setPackage() 调用 - 然后 queryIntentActivities 正常工作,即使在 Android R 中也是如此

完整的解决方案如下:

/**
 * Return all camera possible apps
 * @param context
 * @return
 */
public static List<ResolveInfo> getCameraAppsResolveInfo(Context context){
    List<ResolveInfo> resolveInfo = new ArrayList<>();
    if (Utils.isNull(context)){
        return resolveInfo;
    }
    final Intent capturePhoto = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    PackageManager pm = context.getPackageManager();
    resolveInfo = pm.queryIntentActivities(capturePhoto, 0);
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R){
        // For Android 11 we need to add specific camera apps
        // due them are not added during ACTION_IMAGE_CAPTURE scanning...
        resolveInfo.addAll(getCameraSpecificAppsInfo(context));
    }
    return resolveInfo;
}

/**
 * For Android 11
 * Return camera possible apps
 */
static final String[] CAMERA_SPECIFIC_APPS =  new String[]{
        "best.camera",
        "net.sourceforge.opencamera",
        "com.google.android.GoogleCamera",
        "tools.photo.hd.camera",
};
private static List<ResolveInfo> getCameraSpecificAppsInfo(Context context){
    List<ResolveInfo> resolveInfo = new ArrayList<>();
    if (Utils.isNull(context)){
        return resolveInfo;
    }
    PackageManager pm = context.getPackageManager();
    for (String packageName : CAMERA_SPECIFIC_APPS) {
        resolveInfo.addAll(getCameraSpecificAppInfo(packageName, pm));
    }
    return resolveInfo;
}
private static List<ResolveInfo> getCameraSpecificAppInfo(String packageName, PackageManager pm){
    Intent specificCameraApp = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    specificCameraApp.setPackage(packageName);
    return pm.queryIntentActivities(specificCameraApp, 0);
}

当然,在清单文件中,我们应该添加这些行(如接受的答案中所述)

<queries>
        <intent>
            <action android:name="android.media.action.IMAGE_CAPTURE" />
        </intent>
</queries>
于 2020-09-25T08:18:43.527 回答
1

@saurabh-thorat 关于查询是正确的。但我发现,即使你想要全部,你仍然需要添加一个数据标签以使其适用于所有 mime 类型(或者至少在我的 react-native 应用程序中是这样)。因此,对于 mime 类型的操作(例如:查看/发送/打开)应该如下所示:

<manifest package="com.example">
    <queries>
        <intent>
            <action android:name="android.intent.action.VIEW" />
            <data android:mimeType="*/*" />
        </intent>
    </queries>
    ...
</manifest>
于 2021-12-22T10:32:40.363 回答