123

到目前为止我有这种方法,但它出现的时候好像缺少了什么

例如,我有一个返回 false 的文件 /sdcard/sound.3ga (就像没有可以处理此类文件的活动一样),但是当我从文件管理器中打开它时,它会通过媒体播放器打开,没有问题

我认为这个意图并不完整,我需要做更多的事情来让我自己确保 handlerExists 变量只有在没有可以处理这个意图的活动时才会为假

PackageManager pm = getPackageManager();
Intent intent = new Intent(android.content.Intent.ACTION_VIEW);
String extension = android.webkit.MimeTypeMap.getFileExtensionFromUrl(Uri.fromFile(new File(uriString)).toString());
String mimetype = android.webkit.MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
intent.setDataAndType(Uri.fromFile(new File(uriString)),mimetype);
boolean handlerExists = intent.resolveActivity(pm) != null;
4

9 回答 9

155

edwardxu 的解决方案非常适合我。

只是为了澄清一点:

PackageManager packageManager = getActivity().getPackageManager();
if (intent.resolveActivity(packageManager) != null) {
    startActivity(intent);
} else {
    Log.d(TAG, "No Intent available to handle action");
}
于 2015-05-01T20:17:48.870 回答
86
PackageManager manager = context.getPackageManager();
List<ResolveInfo> infos = manager.queryIntentActivities(intent, 0);
if (infos.size() > 0) {
    //Then there is an Application(s) can handle your intent
} else {
    //No Application can handle your intent
}

你试过这个意图吗?

Intent intent = new Intent(Intent.ACTION_VIEW, Uri.fromFile(yourFileHere));
于 2013-03-14T11:15:21.273 回答
55
if (intent.resolveActivity(getPackageManager()) == null) {
    // No Activity found that can handle this intent. 
}
else{
    // There is an activity which can handle this intent. 
}
于 2015-02-16T02:51:41.390 回答
2

您可以使用:

public static boolean isAvailable(Context ctx, Intent intent) {
   final PackageManager mgr = ctx.getPackageManager();
   List<ResolveInfo> list =
      mgr.queryIntentActivities(intent, 
         PackageManager.MATCH_DEFAULT_ONLY);
   return list.size() > 0;
}
于 2015-10-16T15:55:03.943 回答
1

使用 Kotlin 如果您需要在意图不可用时做某事,

fun isIntentAvailable(context: Context, action: String?): Boolean {
    val packageManager = context.packageManager
    val intent = Intent(action)
    val resolveInfo: List<*> = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY)
    return resolveInfo.isNotEmpty()
}

将此方法实现为

private const val SAMPLE_INTENT = "com.realwear.barcodereader.intent.action.SCAN_BARCODE"

if(isIntentAvailable(this,SAMPLE_INTENT)){
    //Do Stuff
}

如果你无事可做,

Intent(SAMPLE_INTENT).also { barcodeReaderIntent ->
    barcodeReaderIntent.resolveActivity(packageManager)?.also {
        barcodeReaderIntent.putExtra(EXTRA_CODE_128, false)
        startActivityForResult(barcodeReaderIntent, BARCODE_REQUEST_CODE)
    }
}
于 2020-02-24T11:23:16.453 回答
1

解决这个问题的明确答案

if (intent.resolveActivity(getPackageManager()) != null) {
    startActivity(intent);
}
于 2020-10-14T15:36:14.260 回答
0

对于在 2021 年寻找类似东西的人来说是小更新 :) 由于 Android 11 不需要调用包管理器,因此需要额外的努力。那么为什么不将 startActivity() 包装在 try-catch 中呢?甚至更好 - 使用 Rx 漂亮的错误处理:

....
.flatMapCompletable { 
      doSomethingThrowable()
          .onErrorResumeNext { completableCallbackIfNotResolvable() }
}
.subscribe()
于 2021-03-08T14:48:30.270 回答
0

使用 kotlin 扩展的另一种方法

 fun Context.isIntentAvailable(intent: Intent): Boolean {
    val resolveInfo: List<*> = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY)
    return resolveInfo.isNotEmpty()
 }

片段内的使用

val available = requireContext().isIntentAvailable(this)

活动内部使用

val available = this.isIntentAvailable(this)
于 2020-03-25T23:03:08.643 回答
0

这是另一个 Kotlin 解决方案。它甚至适用于Android 30+。首先将其包含在您的应用清单中:

<queries>
    <intent>
        <action android:name="android.intent.action.VIEW" />
        <category android:name="android.intent.category.BROWSABLE" />
        <data android:scheme="choose-scheme-goes-here" />
    </intent>
</queries>

然后您可以检查是否可以使用此函数处理意图:

private fun isIntentAvailable(intent: Intent): Boolean {
    val manager = requireContext().packageManager
    val info = manager.queryIntentActivities(intent, 0)
    return info.size > 0
}
于 2022-01-20T09:28:14.837 回答