Sunday, 20 January 2019

How to handle createChooser's IntentSender without having the component class of the receiver Intent

I am trying to handle the IntentSender of Intent.createChooser() to do something when a user selects an app to share an image on. Most the examples I've found here (posted below), require using a BroadcastReceiver as follows:

Intent receiver = new Intent(context, MyReceiver.class);
PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, receiver, PendingIntent.FLAG_UPDATE_CURRENT);
String type = "image/*";
Intent share = new Intent(Intent.ACTION_SEND);
share.setType(type);
share.putExtra(Intent.EXTRA_STREAM, awesome_photo_uri);
startActivity(Intent.createChooser(share, "some_title", pendingIntent.getIntentSender()));

My problem with this solution, is located in this line:

Intent receiver = new Intent(context, MyReceiver.class);

The Intent constructor used in these examples require me to make a static MyReceiver class, so I have a class to pass into the second argument of the constructor. But, this causes an issue because I'd like the BroadcastReceiver's onReceive to do stuff in my Fragment. Therefore, I would prefer to create a BroadcastReceiver dynamically in my Fragment.

To no avail, I attempted the following work-around:

Inside MyFragment.kt:

private val receiver: BroadcastReceiver = getBroadcastReceiver()
private val intentFilter = IntentFilter("com.my.app.CHOOSER_ACTION")

override fun onResume() {
    requireContext().registerReceiver(receiver, intentFilter)
    super.onResume()
}

override fun onPause() {
    requireContext().unregisterReceiver(receiver)
    super.onPause()
}

private fun shareImage(imageFile: File) {
    Intent().apply {
        flags = Intent.FLAG_ACTIVITY_NEW_TASK
        action = Intent.ACTION_SEND
        type = "image/*"
        putExtra(Intent.EXTRA_STREAM, FileProvider.getUriForFile(
            requireContext(),
            "${myPackageName}.fileprovider",
            imageFile
        ))

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP_MR1) {
            val receiver = Intent("com.my.app.CHOOSER_ACTION")
            val pendingIntent = PendingIntent.getBroadcast(requireContext(), 0, receiver, PendingIntent.FLAG_CANCEL_CURRENT)
            startActivity(Intent.createChooser(this, "Share image using", pendingIntent.intentSender))
        } else {
            startActivity(Intent.createChooser(this, "Share image using"))
        }
    }
}

private fun getBroadcastReceiver() : BroadcastReceiver {
    return object : BroadcastReceiver() {
        override fun onReceive(context: Context?, intent: Intent?) {
            intent?.extras?.keySet()?.forEach {
                Log.v("MyTest", "$it: ${intent.extras?.get(it)}")
            }
            doSomethingInMyFragment()
        }
    }
}

Inside AndroidManifest.xml:

<activity android:name="MyActivityThatHasMyFragment" />
    <intent-filter>
        <action android:name="com.my.app.CHOOSER_ACTION" />
    </intent-filter>
</activity>

Unfortunately, the dynamic BroadcastReceiver's onReceive() function is never called after the user presses on a selection. A few questions:

  1. Why does this not work? What am I missing? Am I somehow setting the Intent or IntentFilter incorrectly?
  2. Is it even possible to use a dynamic BroadcastReceiver for handling the IntentSender of createChooser? If not, how can I create a static BroadcastReceiver that triggers something to happen in MyFragment?

Resources:



from How to handle createChooser's IntentSender without having the component class of the receiver Intent

No comments:

Post a Comment