Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to use PendingIntent to communicate from a Service to a client/Activity?

I have been reading the following text on the Android Developers Site, specifically under the Framework Topics -> Services -> Starting a Service.

There it states the following :

If the service does not also provide binding, the intent delivered with startService() is the only mode of communication between the application component and the service. However, if you want the service to send a result back, then the client that starts the service can create a PendingIntent for a broadcast (with getBroadcast()) and deliver it to the service in the Intent that starts the service. The service can then use the broadcast to deliver a result.

I have a couple of questions regarding this :

  1. Does this text both apply to Services and IntentServices ?
  2. How (codewise) should this be achieved from within the Service; The service can then use the broadcast to deliver a result. and also where would the mentioned broadcast deliver the result to the original client/activity? Is there some method that should be overwritten (like onActivityResult()) or something?
like image 987
TiGer Avatar asked May 23 '11 15:05

TiGer


People also ask

How do you communicate between service and activity?

This example demonstrates how do I communicate between Activity and Service in android. Step 1 − Create a new project in Android Studio, go to File ⇒ New Project and fill all required details to create a new project. Step 2 − Add the following code to res/layout/activity_main. xml.

Why would you use a PendingIntent?

A Pending Intent specifies an action to take in the future. It lets you pass a future Intent to another application and allow that application to execute that Intent as if it had the same permissions as your application, whether or not your application is still around when the Intent is eventually invoked.

Is it possible for a service to communicate with the activity if yes how?

Communication between service and Activity can be done using PendingIntent. For that we can use createPendingResult(). createPendingResult() creates a new PendingIntent object which you can hand to service to use and to send result data back to your activity inside onActivityResult(int, int, Intent) callback.


1 Answers


Question was asked few months ago, but in case anyone is still looking for answer I hope I can help.

In the example below we have local service, responsible for performing some time-consuming operations. Activity makes the requests to the service, but does not bind to it - just sends the intent with request. Additionally, Activity includes the information of BroadcastReceiver that should be called back when service is done with the requested task. The information is passed by PendingIntent. The service handles the task in background thread and when task is finished, service broadcasts the BroadcastReceiver with an answer.

1. Create BroadcastReceiver subclass:

public class DataBroadcastReceiver extends BroadcastReceiver {    static Logger log = LoggerFactory.getLogger(DataRequestService.class);       @Override    public void onReceive(Context context, Intent intent) {       log.info(" onReceive");    } } 

This broadcast receiver will be notified from service, when task is done.

2. Create Service

public class DataRequestService extends Service {     private final class ServiceHandler extends Handler {       public ServiceHandler(Looper looper) {          super(looper);       }        @Override       public void handleMessage(Message msg) {          log.info("handleMessage");          //... performing some time-consuming operation                   Bundle bundle = msg.getData();          PendingIntent receiver = bundle.getParcelable("receiver");          // Perform the operation associated with PendingIntent          try {                         //you can attach data from the operation in the intent.             Intent intent = new Intent();             Bundle b = new Bundle();             //b.putString("key", value);             intent.putExtras(b);             receiver.send(getApplicationContext(), status, intent);          } catch (CanceledException e) {                   e.printStackTrace();          }                }    }        @Override    public void onStart(Intent intent, int startId) {       Bundle bundle = intent.getExtras();       Message msg = mServiceHandler.obtainMessage();       msg.setData(bundle);       mServiceHandler.sendMessage(msg);    } 

Well, the most important part is in handleMessage() method. Service simply makes the broadcasts operation for delivering results to Broadcast Receiver.

3. You also need to register your broadcast receiver and service in Manifest.xml

<manifest xmlns:android="http://schemas.android.com/apk/res/android"     package="com.ramps.servicetest"     android:versionCode="1"     android:versionName="1.0" >    ....        <service android:name=".service.DataRequestService" android:exported="false"/>        <receiver android:name=".service.DataBroadcastReceiver"></receiver>     </application> </manifest><br> 

4. And finally, make request to your service from Activity:

Intent serviceIntent = new Intent(context, DataRequestService.class);       @Override    public void onClick(View v) {       //this is the intent that will be broadcasted by service.       Intent broadcastReceiverIntent = new Intent(context, DataBroadcastReceiver.class);             //create pending intent for broadcasting the DataBroadcastReceiver       PendingIntent pi = PendingIntent.getBroadcast(context, 0, broadcastReceiverIntent, 0);             Bundle bundle = new Bundle();                   bundle.putParcelable("receiver", pi);       //we want to start our service (for handling our time-consuming operation)       Intent serviceIntent = new Intent(context, DataRequestService.class);       serviceIntent.putExtras(bundle);       context.startService(serviceIntent);    } 



5. Delivering response to original client/activity.

You can have abstract activity from which all your activities will be extending. This abstrct activity can automatically register/deregister itself as a response listener in broadcast receiver. Not many options here actually, but it is important that if you keep static references to your activity then you must remove the refernece when activity is destroyed.

Regards,
Ramps

like image 129
Ramps Avatar answered Sep 20 '22 19:09

Ramps