Можно использовать общий доступ к внутренним файлам (например, с помощью 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 2

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);