Получение данных GPS из метаданных файла в Android 10

#java #android #photo-gallery

#java #Android #фотогалерея

Вопрос:

Итак, я должен получить данные GPS из метаданных файла. «Нет проблем», — сказал я, «Это займет всего час или два». Шесть часов спустя я просматриваю всю документацию по Android и вопросы по обмену стеками, которые я могу, но я все еще не могу заставить его работать.

Мой код прост:

  1. Я использую ACTION_PICK для выбора видео (ACTION_OPEN_DOCUMENT тоже не работает).
  2. Я уже запросил следующие разрешения и добавил их в манифест:
     <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.ACCESS_MEDIA_LOCATION" />
  

Ниже приведен код из прослушивателя кнопок

             @Override
            public void onClick(View v) {
                Intent intent = new Intent(Intent.ACTION_PICK , MediaStore.Video.Media.EXTERNAL_CONTENT_URI);
                //intent.setType("video/*");
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                intent.addFlags(Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION);
                startActivityForResult(Intent.createChooser(intent,"Select A Video"),REQUEST_VIDEO_CONTENT);
            }
        });
  

И из onActivityResult:

     protected void onActivityResult(int requestCode, int resultCode, Intent vIntent) {
        super.onActivityResult(requestCode, resultCode, vIntent);

        if ((requestCode == REQUEST_VIDEO_CAPTURE || requestCode == REQUEST_VIDEO_CONTENT) amp;amp; resultCode == RESULT_OK) {
            //Attempt to get Content Data
            vIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            vIntent.addFlags(Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION);
            Uri vUri = vIntent.getData();
            if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.Q) {
                vUri = MediaStore.setRequireOriginal(vUri);
            }
            try {
                InputStream vIS = getContentResolver().openInputStream(vUri);
                ExifInterface vExif =  new ExifInterface(vIS);

            } catch (SecurityException | IOException e) {
                e.printStackTrace();
            }
  

И, наконец, прекрасное исключение SecurityException:

 W/System.err: java.lang.SecurityException: Permission Denial: reading com.google.android.apps.photos.contentprovider.impl.MediaContentProvider uri content://com.google.android.apps.photos.contentprovider/0/2/content://media/external/video/media/198668/ORIGINAL/NONE/video/mp4/328149382?requireOriginal=1 from pid=6190, uid=10146 requires the provider be exported, or grantUriPermission()
W/System.err:     at android.os.Parcel.createException(Parcel.java:2071)
       at android.os.Parcel.readException(Parcel.java:2039)
W/System.err:     at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:188)
       at android.database.DatabaseUtils.readExceptionWithFileNotFoundExceptionFromParcel(DatabaseUtils.java:151)
W/System.err:     at android.content.ContentProviderProxy.openTypedAssetFile(ContentProviderNative.java:705)
       at android.content.ContentResolver.openTypedAssetFileDescriptor(ContentResolver.java:1710)
W/System.err:     at android.content.ContentResolver.openAssetFileDescriptor(ContentResolver.java:1526)
W/System.err:     at android.content.ContentResolver.openFileDescriptor(ContentResolver.java:1345)
       at android.content.ContentResolver.openFileDescriptor(ContentResolver.java:1293)
W/System.err:     at com.equinox.openeyes.VideoSubmission.onActivityResult(VideoSubmission.java:117)
       at android.app.Activity.dispatchActivityResult(Activity.java:8147)
       at android.app.ActivityThread.deliverResults(ActivityThread.java:4883)
W/System.err:     at android.app.ActivityThread.handleSendResult(ActivityThread.java:4931)
       at android.app.servertransaction.ActivityResultItem.execute(ActivityResultItem.java:51)
W/System.err:     at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
       at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2022)
W/System.err:     at android.os.Handler.dispatchMessage(Handler.java:107)
       at android.os.Looper.loop(Looper.java:235)
W/System.err:     at android.app.ActivityThread.main(ActivityThread.java:7441)
       at java.lang.reflect.Method.invoke(Native Method)
       at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)
W/System.err:     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:936)
  

Я понимаю, что ошибка заключается в том, что провайдер должен быть экспортирован или grantUriPermission, что звучит как простое решение, за исключением того, что я не могу найти способ реализовать то, что они просят. Во всех местах, которые я просматривал, предлагалось добавить флаги .. как вы можете видеть, это не сработало.

Пожалуйста, помогите:-(

Комментарии:

1. Трассировка стека не совсем соответствует вашему коду (используется ваш код openInputStream() , используется трассировка стека openFileDescriptor() ). Ни один из ваших addFlags() вызовов ничего не делает, поскольку разрешения определяются провайдером, а не вами. Если вы пропустите setRequireOriginal() вызов и пропустите ExifInterface бит, успешно ли ваш код работает с Uri ? IOW, у вас возникли проблемы с базовым ACTION_PICK результатом, или setRequireOriginal() вызов вызывает ваши проблемы?

2. Да, если я удалю setRequireOriginal, он будет запущен и даже предоставит некоторую информацию Exif, но не данные о местоположении, которые я ищу. Ошибка находится в этой строке: InputStream vIS = getContentResolver().openInputStream(vUri);

3. Хорошо, и для подтверждения вы запрашиваете ACCESS_MEDIA_LOCATION во время выполнения в дополнение к наличию его в манифесте? setRequireOriginal() был беспорядок, когда я пытался работать с ним в прошлом году. Однако в моих экспериментах мне это было не нужно ACTION_OPEN_DOCUMENT — я мог получить данные о местоположении через ExifInterface without setRequireOriginal() .

4. Тем не менее, мои тесты были с изображениями. ExifInterface не документирует поддержку форматов видеофайлов . Работает ли ваш код на старых устройствах?

5. 1. Да, я запрашиваю ActivityCompat.requestPermissions во время выполнения объявления (это, новая строка [] {Manifest.permission. ACCESS_MEDIA_LOCATION},ACCESS_MEDIA_LOCATION_REQUEST_CODE); `