Можно использовать общий доступ к внутренним файлам (например, с помощью FileProvider) в пределах собственного расширения flex для других приложений.
В рамках приложения Adobe Flex я пишу собственное расширение для android, позволяющее мне отправлять электронные письма с вложенными файлами, используя файлы, созданные из кода flex. Однако я не хочу, чтобы файл был доступен для чтения во всем мире, поскольку он может содержать конфиденциальные данные. Поэтому я хочу отправить файл из внутренней области хранения / кэша моего приложения. Я использую Intents для связи с другими приложениями (такими как Gmail) для отправки электронной почты.
После проведения некоторых исследований я обнаружил, что функциональность FileProviders должна делать именно то, что я хочу. Однако статический метод FileProvider.getUriForFile () молча завершает работу / сбой собственного расширения. Собственное расширение останавливается и возвращает null без ошибок или вывода из LogCat.Если я вручную создаю Uri, анализируя строку, Gmail жалуется, что не может прикрепить файл к письму, и письмо отправляется без вложения.
Код в FREObject звоните():
//... (Deal with params from flex app) ...
//Setup Intent, and attach email data send from flex application:
Intent intent = new Intent(Intent.ACTION_SEND);
intent.setType("message/rfc822");
intent.putExtra(Intent.EXTRA_EMAIL, toArray);
intent.putExtra(Intent.EXTRA_CC, ccArray);
intent.putExtra(Intent.EXTRA_BCC, bccArray);
intent.putExtra(Intent.EXTRA_SUBJECT, subject);
intent.putExtra(Intent.EXTRA_TEXT, message);
Log.d("emaildebug", filePath);
//Check we can read our email attachment from flex app
//filePath is a temporary cache location.
File appFile = new File(filePath);
if (appFile.exists() && appFile.canRead()) {
Log.d("emaildebug", "file successfully read");
} else {
return null;
}
//Get a handle on the android Context instead of FREContext.
Context androidContext = (Context) context.getActivity();
//Get the location of the root files directory.
File attachPath = androidContext.getFilesDir();
File attachFile = new File(attachPath, "attachment.pdf");
//Copy file to root of files directory, so that our FileProvider can access it.
try {
copyFileUsingChannel(appFile, attachFile);
} catch (Exception e) {
Log.d("emaildebug", e.getMessage());
}
Log.d("emaildebug", "attachFile exists: " + attachFile.exists() );
Log.d("emaildebug", "attachFile path: " + attachFile.getAbsolutePath());
//This line will silently crash the native extension, instantly returning null, even in try catch.
//Uri contentUri = FileProvider.getUriForFile(androidContext, "com.example.androidextensiontest.provider", attachFile);
//Therefore manually create the Uri from a string.
Uri contentUri = Uri.parse("content://com.example.androidextensiontest.provider/files/attachment.pdf");
Log.d("emaildebug", "uri created");
Log.d("emaildebug", contentUri.toString());
//Grant permisions for all apps that can handle given intent
//Courtesy of: http://stackoverflow.com/questions/18249007/how-to-use-support-fileprovider-for-sharing-content-to-other-apps
List<ResolveInfo> resInfoList = androidContext.getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
String packageName = resolveInfo.activityInfo.packageName;
Log.d("emaildebug", "package: " + packageName);
androidContext.grantUriPermission(packageName, contentUri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.putExtra(Intent.EXTRA_STREAM, contentUri);
context.getActivity().startActivity(Intent.createChooser(intent, "Send mail using..."));
В Файле Манифеста:
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.example.androidextensiontest.provider"
android:exported="false"
android:grantUriPermissions="true" >
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/my_paths" />
</provider>
В my_paths.xml:
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<files-path name="files" path="." />
</paths>
Выход LogCat:
11-01 10:58:09.971: D/emaildebug(17013): /data/data/air.com.example.MyAppName.debug/cache/FlashTmp.V17013/attachment.pdf
11-01 10:58:09.971: D/emaildebug(17013): file successfully read
11-01 10:58:09.991: D/emaildebug(17013): attachFile exists: true
11-01 10:58:09.991: D/emaildebug(17013): attachFile path: /data/data/air.com.example.MyAppName.debug/files/attachment.pdf
11-01 10:58:09.991: D/emaildebug(17013): uri created
11-01 10:58:09.991: D/emaildebug(17013): content://com.example.androidextensiontest.provider/files/attachment.pdf
11-01 10:58:09.991: D/emaildebug(17013): package: com.android.email
11-01 10:58:09.991: D/emaildebug(17013): package: com.google.android.gm
Если я попытаюсь отправить файл из внешнего хранилища с помощью файла:// Uri, вложение файла будет работать идеально. Однако, как уже упоминалось ранее, вложение файла потенциально содержит конфиденциальные данные, поэтому я хочу избежать использования внешнего хранилища.
Мне кажется, я вижу, что файлообменники не работают внутри собственные расширения. Я попытался также использовать ContentProvider, однако ни конструктор, ни методы onCreate не были вызваны.
Манифест ContentProvider:
<provider
android:name="com.example.androidextensiontest.provider.MyContentProvider"
android:authorities="com.example.androidextensiontest.provider">
</provider>
MyContentProvider:
public class MyContentProvider extends ContentProvider implements PipeDataWriter<InputStream> {
public MyContentProvider() {
// Never Output
Log.d("emaildebug", "Constructor");
}
@Override
public boolean onCreate() {
// Never Output
Log.d("emaildebug", "OnCreate");
return false;
}
//.. Rest of class
}
Я делаю что-то не так или ContentProviders / FileProviders просто не поддерживаются в Flex native extensions? Кроме того, существуют ли какие-либо другие решения для прикрепления файла, находящегося во внутреннем хранилище, к электронной почте. Мой поиск на SO и google имеет не встречал никого, кто имел бы подобный опыт, особенно в отношении FileProvider.getUriForFile (). Любая помощь / комментарии приветствуются.
Извините за стену текста, я хотел записать все, что я пробовал до сих пор, и что работает, а что нет.Мы выбрали делать работы контентпровайдеров / поставщиками файла работа в основном с расширениями флекс? Существуют ли другие способы отправки внутренних файлов в виде вложений электронной почты?
1 ответ:
Итак, у меня наконец-то было время вернуться и посмотреть на этот вопрос. Я узнал немного больше о процессе сборки расширения flex и разработке android в целом.
Теперь у меня есть полностью работающий fileprovider в расширении adobe flex.
Первая часть решения состоит в том, что манифест в.ane игнорируется для расширений flex. Вместо какой-либо деятельности или поставщиков, должны быть предоставлены в имя_приложения-приложение.xml в flashbuilder под расширением манифеста:
<android> <colorDepth>16bit</colorDepth> <manifestAdditions><![CDATA[ <manifest android:installLocation="auto"> <application android:enabled="true" android:launchMode="singleInstance"> <provider android:name="android.support.v4.content.FileProvider" android:authorities="com.example.androidextensiontest.files" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider> </application> </manifest> ]]></manifestAdditions> </android>
Это заставит flashbuilder пожаловаться, что он не может найти ресурс @xml/file_paths.
Поэтому вам нужно скопировать всю папку res/ из расширения android в папку ANE build / android, где вы строите ane.
Итак, структура папок сборки выглядит следующим образом:
-BuildFolder -android -libs -res -otherfolders -xml file_paths.xml androidLib.jar library.swf -default -ios
Построить ane как обычно. После этого мой fileprovider начал работать, что означает, что вы можете использовать эту строку для создания uri fileprovider:
Uri contentUri = FileProvider.getUriForFile(androidContext, "com.example.androidextensiontest.files", attachFile);