首页 > 解决方案 > 如何模仿选择器活动/共享菜单?

问题描述

我试图制作自己的选择器活动来替换 androids 共享到...弹出窗口。

我看了一下ChooserActivity extends ResolverActivity并试图复制代码。在我的清单中,我有

<intent-filter>
    <action android:name="android.intent.action.CHOOSER" />
    <category android:name="android.intent.category.DEFAULT" />
</intent-filter>

在 onCreate 我做

Intent intent = getIntent();
Parcelable parcelable = intent.getParcelableExtra("android.intent.extra.INTENT");

Intent shareIntent = (Intent) parcelable;
shareIntent.setComponent(new ComponentName(packageName, activityName));
startActivity(shareIntent);
finish();

这有效,但附加了文件的共享除外。在那里,我遇到了这个崩溃:

java.lang.SecurityException: UID 10098 does not have permission to content://com.android.chrome.FileProvider/images/screenshot/15305419271134395322470190280016.jpg [user 0]
    at android.os.Parcel.readException(Parcel.java:1942)
    at android.os.Parcel.readException(Parcel.java:1888)
    at android.app.IActivityManager$Stub$Proxy.startActivity(IActivityManager.java:4365)
    at android.app.Instrumentation.execStartActivity(Instrumentation.java:1610)
    at android.app.Activity.startActivityForResult(Activity.java:4472)
    at androidx.fragment.app.FragmentActivity.startActivityForResult(FragmentActivity.java:754)
    at android.app.Activity.startActivityForResult(Activity.java:4430)
    at androidx.fragment.app.FragmentActivity.startActivityForResult(FragmentActivity.java:741)
    at android.app.Activity.startActivity(Activity.java:4791)
    at android.app.Activity.startActivity(Activity.java:4759)

似乎我的活动无法执行原始意图,因为它无权访问这些文件。可以做些什么来解决这个问题?有没有办法获得许可?毕竟,android 也以某种方式做到了这一点。

我已经尝试了一些修复,但到目前为止没有任何效果。如果我不能让它工作,我想从 android 调用通常的共享菜单来处理这个,但我也不能让它工作。

addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION|Intent.FLAG_GRANT_WRITE_URI_PERMISSION)

不起作用,因为这些标志已经存在于意图中。另一方面,删除它们将导致共享工作,但文件不会被共享,它将是空的。

startActivity(Intent.createChooser(shareIntent, "test");

会抛出同样的异常。

startActivity(Intent.createChooser(intent, "test"));

我试图通过为原始意图制作一个选择器来以这种方式调用 android 共享菜单,但它只是给了我一个让我选择我的应用程序的窗口。

intent.setComponent(null);
startActivity(Intent.createChooser(intent, "test"));

另一方面,这很有效,我能够选择 android 共享菜单,但是当我尝试共享时没有任何反应。该应用程序的行为就像它没有收到任何东西一样。

intent.setComponent(new ComponentName("android", "com.android.internal.app.ChooserActivity"));
startActivity(intent);

正是我想要的,打开正常的 android 共享菜单。但是仍然存在一个问题:无论份额是多少,它都无济于事。我选择了一些东西,就是这样,它关闭了。没有其他的。为什么会这样?我基本上开始了与旧应用程序相同的意图,并将其重定向到 android。为什么它不再起作用了?我能做些什么?

我读到可能需要root权限才能使其工作,但也许还有一种解决方法?或者至少是一种将原始意图反馈到 android 共享菜单而不破坏它的方法?

从某些应用程序共享时,它可以工作,例如图库应用程序。那些没有问题的应用程序都使用以 content://media/ 开头的 URI...

此外,根据我的研究,它似乎与 FileProvider 有关,但似乎必须在共享给我的应用程序中完成,而不是我的应用程序,所以我无能为力那。

另外,请注意,我不是在创建共享 Intent,而是接收并将其发送到另一个应用程序。

这是来自谷歌的问题跟踪器

标签: androidandroid-sharingandroid-intent-chooser

解决方案


我认为您的应用程序需要 FileProvider。Lollipop中的文件处理发生了变化。首先,您在 AndroidManifest.xml 中创建一个提供程序

步骤1:

<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="com.mydomain.fileprovider"
    android:exported="false"
    android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/file_paths" />
</provider>

第二步:创建并定义file_paths.xml

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <files-path name="image" path="image/"/>
    <external-path name="externalPath" path="external/path"/>
    <cache-path name="cachePath" path="./" />
</paths>

取决于您将文件放在哪里共享,您必须将它们放在正确的属性中,files-path, external-path, cache-path. 在这里查看更多信息。

第三步:分享码。

if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){
    File imagePath = new File(Context.getFilesDir(), "images");
    File newFile = new File(imagePath, "default_image.jpg");
    Uri contentUri = getUriForFile(getContext(), "com.mydomain.fileprovider", newFile);
    intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
    intent.setFlags(FLAG_GRANT_READ_URI_PERMISSION | FLAG_GRANT_WRITE_URI_PERMISSION);
}else{
    intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(image));
}

startActivity(intent);   

推荐阅读