Android 11 (R) return empty list when querying intent for ACTION_IMAGE_CAPTURE
packageManager.queryIntentActivities(intent, 0)
will return an EMPTY list if your app is running on targetSdkVersion 30
To resolve this issue you have to use <queries>
in manifest as queryIntentActivities()
are filtered based on the calling app's declaration.
Fix image capture + image upload to work with Android "scoped storage"
The issue can be related to new package visibility (https://developer.android.com/about/versions/11/privacy/package-visibility). After all updates (at least Android Studio 4.1) try to add in your manifest the that show what action is required in your app.
In my case, the problem disappears when I add IMAGE_CAPTURE for CAMERA, GET_CONTENT for GALLERY (to get files change mimeType if you want video), PICK for GALLERY (should change mimetype if u want video) CHOOSER for GALLERY (if someone has other image browsers)
You can also check in logcat what queries you have to add (should contains "BLOCKED" or "no permission".
Error is because ImagePickerModule
when you don't have permission in Intent with resolveActivity
returns null
(you can comment it to check better errors in startActivityForResult
)
Add <query>
in 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>
My solution for Android 11, for getting ResolveInfo list.
- When we scan only by MediaStore.ACTION_IMAGE_CAPTURE filter then we will get only One! application record - System default camera app.
- For using additional camera apps we need to specify each application by package name, and provide it with setPackage() call - and then queryIntentActivities works correctly, even in Android R
The full solution is below:
/**
* 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);
}
Of course in manifest file we should add these lines (as described in accepted answer)
<queries>
<intent>
<action android:name="android.media.action.IMAGE_CAPTURE" />
</intent>
</queries>
Android 11 changes how apps can query and interact with other apps.
From the docs:
The
PackageManager
methods that return results about other apps, such asqueryIntentActivities()
, are filtered based on the calling app's<queries>
declaration.
So you need to declare <queries>
in your AndroidManifest.xml
:
<manifest package="com.example">
<queries>
<intent>
<action android:name="android.media.action.IMAGE_CAPTURE" />
</intent>
</queries>
...
</manifest>
@saurabh-thorat is correct about the queries. But I found out that even if you want all, you still need to add a data tag for it to work with all mime types (or at least that was my case in my react-native app). So it should be as below for actions on mime types (e.g.: view/send/open):
<manifest package="com.example">
<queries>
<intent>
<action android:name="android.intent.action.VIEW" />
<data android:mimeType="*/*" />
</intent>
</queries>
...
</manifest>