web-dev-qa-db-fra.com

Envoi d'une notification depuis un service dans Android

J'ai un service en cours d'exécution et j'aimerais envoyer une notification. Dommage, l'objet de notification nécessite un Context, comme un Activity, et non un Service.

Savez-vous un moyen de contourner cela? J'ai essayé de créer un Activity pour chaque notification, mais cela semble moche et je ne trouve pas le moyen de lancer un Activity sans aucun View.

98
e-satis

Activity et Service en fait extendContext afin que vous puissiez simplement utiliser this comme votre Context dans votre Service.

NotificationManager notificationManager =
    (NotificationManager) getSystemService(Service.NOTIFICATION_SERVICE);
Notification notification = new Notification(/* your notification */);
PendingIntent pendingIntent = /* your intent */;
notification.setLatestEventInfo(this, /* your content */, pendingIntent);
notificationManager.notify(/* id */, notification);
105
Josef Pfleger

Ce type de notification est déconseillé, tel qu'il apparaît dans les documents:

@Java.lang.Deprecated
public Notification(int icon, Java.lang.CharSequence tickerText, long when) { /* compiled code */ }

public Notification(Android.os.Parcel parcel) { /* compiled code */ }

@Java.lang.Deprecated
public void setLatestEventInfo(Android.content.Context context, Java.lang.CharSequence contentTitle, Java.lang.CharSequence contentText, Android.app.PendingIntent contentIntent) { /* compiled code */ }

Meilleure façon
Vous pouvez envoyer une notification comme celle-ci:

// prepare intent which is triggered if the
// notification is selected

Intent intent = new Intent(this, NotificationReceiver.class);
PendingIntent pIntent = PendingIntent.getActivity(this, 0, intent, 0);

// build notification
// the addAction re-use the same intent to keep the example short
Notification n  = new Notification.Builder(this)
        .setContentTitle("New mail from " + "[email protected]")
        .setContentText("Subject")
        .setSmallIcon(R.drawable.icon)
        .setContentIntent(pIntent)
        .setAutoCancel(true)
        .addAction(R.drawable.icon, "Call", pIntent)
        .addAction(R.drawable.icon, "More", pIntent)
        .addAction(R.drawable.icon, "And more", pIntent).build();


NotificationManager notificationManager = 
  (NotificationManager) getSystemService(NOTIFICATION_SERVICE);

notificationManager.notify(0, n); 

Meilleure façon
Le code ci-dessus requiert le niveau minimum d'API 11 (Android 3.0).
Si votre niveau minimal d’API est inférieur à 11, vous devriez utiliser la classe NotificationCompat de bibliothèque de support comme ceci.

Donc, si votre niveau minimum d'API cible est 4+ (Android 1.6+), utilisez ceci:

    import Android.support.v4.app.NotificationCompat;
    -------------
    NotificationCompat.Builder builder =
            new NotificationCompat.Builder(this)
                    .setSmallIcon(R.drawable.mylogo)
                    .setContentTitle("My Notification Title")
                    .setContentText("Something interesting happened");
    int NOTIFICATION_ID = 12345;

    Intent targetIntent = new Intent(this, MyFavoriteActivity.class);
    PendingIntent contentIntent = PendingIntent.getActivity(this, 0, targetIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    builder.setContentIntent(contentIntent);
    NotificationManager nManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    nManager.notify(NOTIFICATION_ID, builder.build());
75
trante
@TargetApi(Build.VERSION_CODES.JELLY_BEAN)
public void PushNotification()
{
    NotificationManager nm = (NotificationManager)context.getSystemService(NOTIFICATION_SERVICE);
    Notification.Builder builder = new Notification.Builder(context);
    Intent notificationIntent = new Intent(context, MainActivity.class);
    PendingIntent contentIntent = PendingIntent.getActivity(context,0,notificationIntent,0);

    //set
    builder.setContentIntent(contentIntent);
    builder.setSmallIcon(R.drawable.cal_icon);
    builder.setContentText("Contents");
    builder.setContentTitle("title");
    builder.setAutoCancel(true);
    builder.setDefaults(Notification.DEFAULT_ALL);

    Notification notification = builder.build();
    nm.notify((int)System.currentTimeMillis(),notification);
}
7
王怡飞

Eh bien, je ne suis pas sûr si ma solution est la meilleure pratique. En utilisant NotificationBuilder mon code ressemble à ça:

private void showNotification() {
    Intent notificationIntent = new Intent(this, MainActivity.class);

    PendingIntent contentIntent = PendingIntent.getActivity(
                this, 0, notificationIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    builder.setContentIntent(contentIntent);
    NotificationManager notificationManager =
            (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    notificationManager.notify(NOTIFICATION_ID, builder.build());
    }

Manifeste:

    <activity
        Android:name=".MainActivity"
        Android:launchMode="singleInstance"
    </activity>

et ici le service:

    <service
        Android:name=".services.ProtectionService"
        Android:launchMode="singleTask">
    </service>

Je ne sais pas s'il y a vraiment un singleTask à Service mais cela fonctionne correctement sur mon application ...

1
Martin Pfeffer