Поделиться uri файла из ACTION PICK

im пытается сделать приложение, которое попросит пользователя выбрать файл изображения, а затем отправить его через намерение в другое приложение (Whatsapp в этом случае, но это должно работать с другими приложениями)

Im делает это, чтобы запросить файл:

 Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
    intent.setType("image/*");
    intent.addCategory(Intent.CATEGORY_OPENABLE);

    try {
        startActivityForResult(
                Intent.createChooser(intent, "Select a File to Upload"),
                FILE_SELECT_CODE);
    } catch (android.content.ActivityNotFoundException ex) {
        Toast.makeText(this, "Please install a File Manager.", Toast.LENGTH_SHORT).show();
    }

Это возвращает URI" content://", но при отправке его через intent:

 if (isPackageExisted(whatsAppPackage)) {
        Intent sendIntent = new Intent();
        sendIntent.setAction(Intent.ACTION_SEND);
        sendIntent.putExtra(Intent.EXTRA_TEXT, "Some text");
        sendIntent.putExtra(Intent.EXTRA_STREAM, imageUri);
        sendIntent.setType("image/*");
        sendIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        sendIntent.setPackage(whatsAppPackage);
        startActivity(sendIntent);
    }else{
        goToGooglePlay(whatsAppPackage);
    }

Whatsapp показывает тост, говоря, что формат файла не поддерживается, и я вижу stacktrace на чтение logcat

java.lang.ClassCastException: android.net.Uri$StringUri cannot be cast to java.util.ArrayList
                                         at android.os.Bundle.getParcelableArrayList(Bundle.java:838)
                                         at android.content.Intent.getParcelableArrayListExtra(Intent.java:5405)
                                         at com.whatsapp.ContactPicker.c(ContactPicker.java:12)
                                         at com.whatsapp.ContactPicker.onCreate(ContactPicker.java:526)
                                         at android.app.Activity.performCreate(Activity.java:6251)
                                         at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1107)
                                         at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2369)
                                         at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476)
                                         at android.app.ActivityThread.-wrap11(ActivityThread.java)
                                         at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344)
                                         at android.os.Handler.dispatchMessage(Handler.java:102)
                                         at android.os.Looper.loop(Looper.java:148)
                                         at android.app.ActivityThread.main(ActivityThread.java:5417)
                                         at java.lang.reflect.Method.invoke(Native Method)
                                         at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
                                         at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)

...

Я не уверен, могу ли я поделиться uri "picker app" с другим приложением, что-то делать с разрешениями?

что бы быть правильный способ сделать это? нужно ли мне сделать копию файла, чтобы мое приложение могло поделиться им?

EDIT:

это работает с другими приложениями, такими как Gmail, но не в whatsapp, поэтому можно иметь решение, которое работает со всеми или большинством приложений? мне все равно, если придется. отдельные методы для обмена, но WhatsApp является обязательным.

3 ответов


код отлично работает на реальном устройстве, он находится на симуляторе android, где он не работает.

Я не ожидал этого, я попробовал по телефону на всякий случай.


как журнал вашего кода говорит вам не удается привести uri к строке Arraylist.

так попробуй преобразование StringURI в ArrayListURI вот так

ArrayList<Uri> tempURI = new ArrayList<>();
tempURI.add(imageUri); //Your String URI

А Теперь Проходите tempURI с намерением.

sendIntent.putExtra(Intent.EXTRA_STREAM, tempURI);

попробуйте это и проверьте, работает ли он на тренажере или нет..

также прочитайте это

ACTION_SEND поддерживает EXTRA_STREAM, но только один Uri. ACTION_SEND_MULTIPLE поддерживает EXTRA_STREAM С ArrayList<Uri>.


этот код работает в эмуляторе и реальном устройстве

    private val FILE_SELECT_CODE = 101
    private val whatsAppPackage = "com.whatsapp"

       fun sharePicture() {
         val intent = Intent(Intent.ACTION_GET_CONTENT)
         intent.type = "image/*"
         intent.addCategory(Intent.CATEGORY_OPENABLE)
         if (intent.resolveActivity(packageManager) != null){
         startActivityForResult(
                Intent.createChooser(intent, "Select a File to Upload"),
                FILE_SELECT_CODE)
         }else{
             Toast.makeText(this, "Please install a File Manager.", 
             Toast.LENGTH_SHORT).show()
         }
      }


      override fun onActivityResult(requestCode: Int, resultCode: Int,                                       
      data: Intent?) {
         super.onActivityResult(requestCode, resultCode, data)

         if (resultCode == Activity.RESULT_OK) {
             when (requestCode) {
                 FILE_SELECT_CODE -> {
                        val sendIntent = Intent()
                        sendIntent.action = Intent.ACTION_SEND
                        sendIntent.putExtra(Intent.EXTRA_TEXT, "Some text")
                        sendIntent.putExtra(Intent.EXTRA_STREAM, data!!.data)
                        sendIntent.type = "image/*"
                        sendIntent.addFlags(
                                            Intent.FLAG_GRANT_READ_URI_PERMISSION)
                        sendIntent.`package` = whatsAppPackage
                     if (sendIntent.resolveActivity(packageManager) != null) {
                        startActivity(sendIntent)
                     } else {
                          goToGooglePlay(whatsAppPackage)
                     }
               }
            }
         } 
      }


       fun goToGooglePlay(appPackageName: String) {
         val playIntent = Intent(Intent.ACTION_VIEW, 
                          Uri.parse("market://details?id=$appPackageName"))

         if (playIntent.resolveActivity(packageManager) != null){
             startActivity(playIntent)
         }else
             startActivity(Intent(Intent.ACTION_VIEW, 
                        Uri.parse("https://play.google.com/store/apps/details? id=$appPackageName")))
        }


       fun isPackageExisted(packageName: String): Boolean {
           val pm = packageManager
           val isInstalled: Boolean
           isInstalled = try {
               pm.getPackageInfo(packageName, PackageManager.GET_ACTIVITIES)
               true
           } catch (e: PackageManager.NameNotFoundException) {
               false
           }

          return isInstalled
       }