背景
类似于我最近在共享/打开 APK 文件时遇到的问题(here https://stackoverflow.com/q/41160353/878126, here https://stackoverflow.com/q/41085950/878126 and here https://stackoverflow.com/q/40941709/878126),我现在遇到了将图像文件(在资产、res/raw 甚至从 URL 中)发送到特定应用程序的问题:WhatsApp 和 Viber。
我需要能够将图像文件共享给所有应用程序,尤其是 WhatsApp 和 Viber 等流行应用程序。
问题
当我尝试在 Android 7.1.1 上共享图像文件时,WhatsApp 和 Viber 都出现问题。在其他应用程序以及以前版本的 Android 上,它运行良好。
在我尝试过的所有测试中,它们要么显示黑屏(无图像),要么自行关闭。
我尝试并发现了什么
1.我开始使用名为“的库”从应用程序的资产文件夹共享文件CWAC 提供商 https://github.com/commonsguy/cwac-provider“。除了 WhatsApp 和 Viber 之外,它适用于所有应用程序。
在 WhatsApp 上,我得到了这个日志(这与我在 Viber 上得到的日志非常相似):
02-06 17:05:04.379 24590-24590/com.whatsapp 附带捆绑包:钥匙
android.intent.extra.STREAM 期望 ArrayList 但值是
android.net.Uri$HierarchicalUri。返回默认值。 02-06
17:05:04.382 24590-24590/com.whatsapp 包含捆绑包:尝试投射
生成的内部异常:java.lang.ClassCastException:
android.net.Uri$HierarchicalUri 无法转换为 java.util.ArrayList
在 android.os.Bundle.getParcelableArrayList(Bundle.java:916) 处
android.content.Intent.getParcelableArrayListExtra(Intent.java:6357)
在 com.whatsapp.ContactPicker.k(ContactPicker.java:618) 处
com.whatsapp.ContactPicker.onCreate(ContactPicker.java:360) 在
android.app.Activity.performCreate(Activity.java:6688) 在
android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1118)
在
android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2633)
在
android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2741)
在 android.app.ActivityThread.-wrap12(ActivityThread.java) 处
android.app.ActivityThread$H.handleMessage(ActivityThread.java:1488)
在 android.os.Handler.dispatchMessage(Handler.java:102) 处
android.os.Looper.loop(Looper.java:154) 位于
android.app.ActivityThread.main(ActivityThread.java:6169) 在
java.lang.reflect.Method.invoke(本机方法) at
com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:888)
在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:778)
2.有人告诉我(here https://github.com/commonsguy/cwac-provider/issues/28#issuecomment-277853514)尝试通过放置一个来分享ArrayList<Uri>
进入EXTRA_STREAM
:
ArrayList<Uri> uriArrayList=new ArrayList<>();
uriArrayList.add(getUri());
share.putExtra(Intent.EXTRA_STREAM, uriArrayList);
没有成功,WhatsApp 的日志显示:
Caused by: java.lang.SecurityException: Permission Denial: opening provider
com.commonsware.cwac.provider.StreamProvider 来自
ProcessRecord{9405e93 12914:com.whatsapp/u0a210} (pid=12914,
uid=10210) 不是从 uid 10123 导出的
在 android.os.Parcel.readException(Parcel.java:1684)
在 android.os.Parcel.readException(Parcel.java:1637)
在 android.app.ActivityManagerProxy.getContentProvider(ActivityManagerNative.java:4213)
在 android.app.ActivityThread.acquireProvider(ActivityThread.java:5526)
在 android.app.ContextImpl$ApplicationContentResolver.acquireUnstableProvider(ContextImpl.java:2239)
在 android.content.ContentResolver.acquireUnstableProvider(ContentResolver.java:1517)
在 android.content.ContentResolver.openTypedAssetFileDescriptor(ContentResolver.java:1131)
在 android.content.ContentResolver.openAssetFileDescriptor(ContentResolver.java:984)
在 android.content.ContentResolver.openInputStream(ContentResolver.java:704)
在 com.whatsapp.util.ah.b(MediaFileUtils.java:1290)
在 com.whatsapp.util.ah.a(MediaFileUtils.java:1498)
在 com.whatsapp.util.ah.a(MediaFileUtils.java:1543)
在 com.whatsapp.gallerypicker.ImagePreview$b$1.a(ImagePreview.java:901)
在 com.whatsapp.gallerypicker.ImagePreview$b$1.doInBackground(ImagePreview.java:896)
在 android.os.AsyncTask$2.call(AsyncTask.java:305)
在 java.util.concurrent.FutureTask.run(FutureTask.java:237)
在 android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:243)
在java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:607)
at java.lang.Thread.run(Thread.java:761)
3.我还尝试将意图的动作更改为ACTION_SEND_MULTIPLE
:
Intent share = new Intent(Intent.ACTION_SEND_MULTIPLE);
...
ArrayList<Uri> uriArrayList=new ArrayList<>();
uriArrayList.add(getUri());
share.putExtra(Intent.EXTRA_STREAM, uriArrayList);
但它也没有帮助,显示了 Viber 的日志(看不到 WhatsApp 的任何特殊内容):
02-07 09:54:07.084 926-10718/system_process W/ActivityManager:
拒绝许可:开放提供商
com.commonsware.cwac.provider.StreamProvider 来自
ProcessRecord{adbb1ed 5565:com.viber.voip/u0a175} (pid=5565,
uid=10175) 不是从 uid 10123 02-07 09:54:07.087 导出的
926-10717/system_process W/ActivityManager:权限拒绝:打开
提供商 com.commonsware.cwac.provider.StreamProvider 来自
ProcessRecord{adbb1ed 5565:com.viber.voip/u0a175} (pid=5565,
uid=10175) 不是从 uid 10123 02-07 09:54:07.091 导出的
926-946/system_process W/ActivityManager:权限拒绝:打开
提供商 com.commonsware.cwac.provider.StreamProvider 来自
ProcessRecord{adbb1ed 5565:com.viber.voip/u0a175} (pid=5565,
uid=10175) 不是从 uid 10123 导出的
4.奇怪的是,对于 WhatsApp 来说,在上述所有尝试中,它都会请求存储权限,尽管它不应该这样做(因为应用程序无论如何都会自行提供内容)。
5.我发现的另一个奇怪的事情是即使图像来自服务器,也可以很好地与这些应用程序共享图像。它将文件下载到某个地方并共享它。但我看不到它在哪里下载文件。我认为它会位于应用程序的外部存储路径 ("/.../Android/data/com.google.android.apps.photos/...") 上,但它不在那里。
6.我尝试使用支持库的FileProvider来制作从外部存储共享文件的POC(因为我已经知道如何通过共享APK文件来使用):
manifest
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="${applicationId}.provider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths"/>
</provider>
res/xml/provider_paths.xml
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-path name="external_files" path="."/>
</paths>
code:
final File bitmapFile = new File(getExternalFilesDir(null), "test.jpg");
if (!bitmapFile.exists()) {
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), android.R.drawable.sym_def_app_icon);
bitmap.compress(CompressFormat.JPEG, 100, new FileOutputStream(bitmapFile));
}
Intent intent = new Intent(Intent.ACTION_SEND);
Uri fileUri = FileProvider.getUriForFile(this, getApplicationContext().getPackageName() + ".provider", bitmapFile);
intent.setType(MimeTypeMap.getSingleton().getMimeTypeFromExtension("jpg"));
intent.putExtra(Intent.EXTRA_STREAM, fileUri);
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
startActivity(intent);
但它只适用于 WhatsApp,不适用于 Viber,后者会显示日志
02-07 10:21:19.285 24043-24043/com.viber.voip 附带捆绑包:钥匙
android.intent.extra.STREAM 期望 ArrayList 但值是
android.net.Uri$HierarchicalUri。默认值为
回。 02-07 10:21:19.285 24043-24043/com.viber.voip 包含捆绑包:
尝试强制转换生成的内部异常:
java.lang.ClassCastException:android.net.Uri$HierarchicalUri 不能
被转换为 java.util.ArrayList
在 android.os.Bundle.getParcelableArrayList(Bundle.java:916)
在 android.content.Intent.getParcelableArrayListExtra(Intent.java:6357)
在 com.viber.voip.util.af.f(源文件:156)
在 com.viber.voip.util.af.a(源文件:106)
在 com.viber.voip.HomeActivity.i(源文件:487)
在 com.viber.voip.HomeActivity.onCreate(源文件:317)
在 android.app.Activity.performCreate(Activity.java:6688)
在 android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1118)
在 android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2633)
在 android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2741)
在 android.app.ActivityThread.-wrap12(ActivityThread.java)
在 android.app.ActivityThread$H.handleMessage(ActivityThread.java:1488)
在 android.os.Handler.dispatchMessage(Handler.java:102)
在 android.os.Looper.loop(Looper.java:154)
在 android.app.ActivityThread.main(ActivityThread.java:6169)
在 java.lang.reflect.Method.invoke(本机方法)
在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:888)
在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:778)
我尝试执行我在 #2 和 #3 上所做的操作,但它在 Viber 上仍然不起作用。
7.我尝试使用旧的方法从外部存储共享文件,这种方法应该被弃用并由 FileProvider 替换:
startActivityForResult(Intent.createChooser(prepareIntentToSharePhoto(bitmapFile.getAbsolutePath(), "title",
"body"), "choose"), 1);
public static Intent prepareIntentToSharePhoto(String imagePath, String title, String body) {
Intent sharingIntent = new Intent(Intent.ACTION_SEND).setType("image/*")
.putExtra(Intent.EXTRA_STREAM, Uri.parse("file://" + imagePath)).putExtra(android.content.Intent.EXTRA_SUBJECT, title)
.putExtra(android.content.Intent.EXTRA_TEXT, body);
return sharingIntent;
}
它适用于这两个应用程序,但仅当它们都被授予存储权限时才有效。对于 Viber,如果没有存储权限,它会显示黑色图像;对于 WhatsApp,它会要求用户授予存储权限。
问题
为什么以上任何一个都不起作用?
我应该如何正确地将图像文件共享给这些应用程序?即使通过 FileProvider 共享又有什么问题? Google Photos 应用为何运行良好?
有解决方法吗?
这是应用程序本身的问题,还是 Android 的问题?