diff --git a/packages/android_intent_plus/README.md b/packages/android_intent_plus/README.md
index 2cfa7718b1..cd6ff9fba5 100644
--- a/packages/android_intent_plus/README.md
+++ b/packages/android_intent_plus/README.md
@@ -83,6 +83,25 @@ of integers or strings.
> ACTION_VIEW intents for Android, however this intent plugin also allows
> clients to set extra parameters for the intent.
+### Querying activities
+`canResolveActivity()` and `getResolvedActivity()` can be used to query whether an activity can handle an intent,
+or get the details of the activity that can handle the intent.
+
+```dart
+final intent = AndroidIntent(
+ action: 'action_view',
+ data: Uri.encodeFull('http://'),
+ );
+
+// can this intent be handled by an activity
+final canHandleIntent = await intent.canResolveActivity();
+
+// get the details of the activity that will handle this intent
+final details = await intent.getResolvedActivity();
+
+print(details.packageName); // prints com.google.chrome
+```
+
## Android 11 package visibility
Android 11 introduced new permissions for package visibility.
diff --git a/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/IntentSender.java b/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/IntentSender.java
index 2158b42024..ef2bd7a549 100644
--- a/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/IntentSender.java
+++ b/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/IntentSender.java
@@ -5,12 +5,15 @@
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
+import android.content.pm.ResolveInfo;
import android.net.Uri;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import androidx.annotation.Nullable;
import java.net.URISyntaxException;
+import java.util.HashMap;
+import java.util.Map;
/** Forms and launches intents. */
public final class IntentSender {
@@ -102,6 +105,41 @@ boolean canResolveActivity(Intent intent) {
return packageManager.resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY) != null;
}
+ /**
+ * Get the default activity that will resolve the intent
+ *
+ *
This will fail to create and send the intent if {@code applicationContext} hasn't been set *
+ * at the time of calling.
+ *
+ *
This currently only supports resolving activities.
+ *
+ * @param intent Fully built intent.
+ * @return Whether the package manager found {@link android.content.pm.ResolveInfo} using its
+ * {@link PackageManager#resolveActivity(Intent, int)} method.
+ * @see #buildIntent(String, Integer, String, Uri, Bundle, String, ComponentName, String)
+ */
+ @Nullable
+ Map getResolvedActivity(Intent intent) {
+ if (applicationContext == null) {
+ Log.wtf(TAG, "Trying to resolve an activity before the applicationContext was initialized.");
+ return null;
+ }
+
+ final PackageManager packageManager = applicationContext.getPackageManager();
+ ResolveInfo resolveInfo =
+ packageManager.resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY);
+
+ if (resolveInfo != null) {
+ Map resultMap = new HashMap<>();
+ resultMap.put("activityName", resolveInfo.activityInfo.name);
+ resultMap.put("packageName", resolveInfo.activityInfo.packageName);
+ resultMap.put("appName", resolveInfo.loadLabel(packageManager));
+ return resultMap;
+ }
+
+ return null;
+ }
+
/** Caches the given {@code activity} to use for {@link #send}. */
void setActivity(@Nullable Activity activity) {
this.activity = activity;
diff --git a/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/MethodCallHandlerImpl.java b/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/MethodCallHandlerImpl.java
index b56ff44845..0cd0a0765f 100644
--- a/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/MethodCallHandlerImpl.java
+++ b/packages/android_intent_plus/android/src/main/java/dev/fluttercommunity/plus/androidintent/MethodCallHandlerImpl.java
@@ -119,6 +119,8 @@ public void onMethodCall(@NonNull MethodCall call, @NonNull Result result) {
result.success(null);
} else if ("canResolveActivity".equalsIgnoreCase(call.method)) {
result.success(sender.canResolveActivity(intent));
+ } else if ("getResolvedActivity".equalsIgnoreCase(call.method)) {
+ result.success(sender.getResolvedActivity(intent));
} else {
result.notImplemented();
}
diff --git a/packages/android_intent_plus/example/android/app/src/main/AndroidManifest.xml b/packages/android_intent_plus/example/android/app/src/main/AndroidManifest.xml
index e033f9db7b..20f0667b3e 100644
--- a/packages/android_intent_plus/example/android/app/src/main/AndroidManifest.xml
+++ b/packages/android_intent_plus/example/android/app/src/main/AndroidManifest.xml
@@ -8,7 +8,18 @@
flutter needs it to communicate with the running application
to allow setting breakpoints, to provide hot reload, etc.
-->
-
+
+
+
+
+
+
+
+
+
+
+
+
_getResolvedActivity(context),
+ child: const Text(
+ 'Tap here to get default resolved activity',
+ ),
+ ),
+ const SizedBox(height: 16),
ElevatedButton(
onPressed: _openGmail,
child: const Text(
diff --git a/packages/android_intent_plus/lib/android_intent.dart b/packages/android_intent_plus/lib/android_intent.dart
index b680b962af..42494eb4aa 100644
--- a/packages/android_intent_plus/lib/android_intent.dart
+++ b/packages/android_intent_plus/lib/android_intent.dart
@@ -207,6 +207,31 @@ class AndroidIntent {
);
}
+ /// Get the default activity that will resolve the intent
+ ///
+ /// Note: ensure the calling app's AndroidManifest contains queries that match the intent.
+ /// See: https://developer.android.com/guide/topics/manifest/queries-element
+ Future getResolvedActivity() async {
+ if (!_platform.isAndroid) {
+ return null;
+ }
+
+ final result = await _channel.invokeMethod