web-dev-qa-db-fra.com

Comment effectuer une notification-action (cliquez) sur l'écran de verrouillage?

TL; DR

Comment puis-je faire une notification qui fonctionne à partir de l'écran de verrouillage sans déverrouiller? Après avoir cliqué sur une action, un bouton sur la notification ou simplement la notification complète, je souhaite faire un appel API (sans taper mon code de déverrouillage)

Détails

Objectif

Sur la base de la réponse sur cette question J'ai essayé de faire une notification avec une action qui fonctionne sur l'écran de verrouillage sans déverrouiller l'appareil. L'action est quelque chose qui n'a besoin d'aucune autre interface ou interaction (pensez à "envoyer une demande d'API").

Statut

La notification et le clic fonctionnent avec un appareil déverrouillé. Cependant, une fois verrouillé, je dois d'abord saisir le code de déverrouillage, donc soit il y a quelque chose de nouveau, soit j'ai mal compris la façon dont il est censé fonctionner.

Si je comprends bien, je peux définir ma visibilité sur `` public '' pour afficher le contenu (cela fonctionne), et au lieu de définir une action (qui ne semble pas être publique), je peux gérer les clics sur la mise en page (désormais visible). J'ai essayé cela avec le code ci-dessous, mais évidemment cela ne fonctionne pas.

J'ai essayé d'envoyer l'intention à mon application et à un service, comme Florian l'a suggéré ci-dessous.

Code

C'est le code où je commence la notification (cela vit dans une activité, le code a été raccourci pour votre commodité)

private void startNotification() {

    NotificationCompat.Builder builder = 
            new NotificationCompat.Builder(this)
            .setVisibility(Notification.VISIBILITY_PUBLIC)
            .setOngoing(true)
            .setSmallIcon(R.drawable.abc_ic_menu_share_mtrl_alpha)
            .setContentTitle("title text")
            .setContentText("content text");

    Intent openIntent = new Intent(MyMainActivity.this, MyMainActivity.class);
    openIntent.setAction("some_string");
    PendingIntent pOpenIntent = PendingIntent.getActivity(this, 0, openIntent, 0);
    builder.setContentIntent(pOpenIntent);

    RemoteViews view = new RemoteViews(getPackageName(), R.layout.notification);
    builder.setContent(view);

    NotificationManager mNotificationManager =
            (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
    mNotificationManager.notify(id, builder.build());

}

Comme dit, j'ai également essayé le service comme l'a suggéré Florian, avec ceci comme un appel:

    Intent yepIntent = new Intent(this, MyIntentService.class);
    yepIntent.setAction("test");
    yepIntent.putExtra("foo", true);
    yepIntent.putExtra("bar", "more info");
    PendingIntent yepPendingIntent = PendingIntent.getService(this, notificationId, yepIntent, PendingIntent.FLAG_CANCEL_CURRENT);
    //builder.addAction(R.drawable.abc_ic_menu_share_mtrl_alpha, "My Action", yepPendingIntent);
    builder.setContentIntent(yepPendingIntent);

L'action n'apparaissait pas sur l'écran de verrouillage, donc je l'ai changé en setContentIntent que vous voyez ci-dessus. Le résultat est le même cependant, aucune action pour moi :(

25
Nanne

Essayez d'utiliser un IntentService . Remplacez votre cible d'intention par votre service d'intention:

    Intent yepIntent = new Intent(context, MyIntentService.class);
    yepIntent.putExtra("foo", true);
    yepIntent.putExtra("bar", "more info");
    PendingIntent yepPendingIntent = PendingIntent.getService(context, notificationId, yepIntent, PendingIntent.FLAG_CANCEL_CURRENT);
    notificationBuilder.addAction(R.drawable.icon_of_choice, "My Action", yepPendingIntent);

Enregistrez votre service dans le manifeste:

  <service
        Android:name="app.great.mypackage.MyIntentService"
        Android:exported="false"/>

Votre service pourrait ressembler à ceci:

public class MyIntentSerice extends IntentService {
    @Override
    protected void onHandleIntent(Intent intent) {
        Log.d("myapp", "I got this awesome intent and will now do stuff in the background!");
        // .... do what you like
    }
}

MISE À JOUR avec les commentaires de Nanne

L'astuce semble être de

  1. Utilisez un service
  2. Ajoutez l'intention non pas comme une action ou un contentIntent, mais avec la méthode RemoteViews.

Combiné, ce sera:

NotificationCompat.Builder builder = new NotificationCompat.Builder(this)
        .setVisibility(Notification.VISIBILITY_PUBLIC)
        .setOngoing(true)
        .setSmallIcon(R.drawable.abc_ic_menu_share_mtrl_alpha)
        .setContentTitle("My notification")
        .setContentText("Hello World!");

int notificationId = 1;
Intent yepIntent = new Intent(this, MyIntentService.class);
yepIntent.setAction("test");
yepIntent.putExtra("foo", true);
yepIntent.putExtra("bar", "more info");
PendingIntent yepPendingIntent = PendingIntent.getService(this, notificationId, yepIntent, PendingIntent.FLAG_CANCEL_CURRENT);

// doesn't show up on my lock-screen
//builder.addAction(R.drawable.abc_ic_menu_share_mtrl_alpha, "My Action", yepPendingIntent);

// asks for unlock code for some reason
//builder.setContentIntent(yepPendingIntent);

// Bingo
RemoteViews view = new RemoteViews(getPackageName(), R.layout.notification);
view.setOnClickPendingIntent(R.id.notification_closebtn_ib, yepPendingIntent);
builder.setContent(view);
16
Florian Barth

En combinant la réponse de la question que j'ai liée ( bouton d'action de notification non cliquable dans l'écran de verrouillage ) et celle donnée par @florian_barth ci-dessus, je l'ai fait fonctionner

L'astuce semble être de

  1. Utilisez un service
  2. Ajoutez l'intention non pas comme une action ou un contentIntent, mais avec la méthode RemoteViews.

Combiné, ce sera:

    NotificationCompat.Builder builder = new NotificationCompat.Builder(this)
            .setVisibility(Notification.VISIBILITY_PUBLIC)
            .setOngoing(true)
            .setSmallIcon(R.drawable.abc_ic_menu_share_mtrl_alpha)
            .setContentTitle("My notification")
            .setContentText("Hello World!");

    int notificationId = 1;
    Intent yepIntent = new Intent(this, MyIntentService.class);
    yepIntent.setAction("test");
    yepIntent.putExtra("foo", true);
    yepIntent.putExtra("bar", "more info");
    PendingIntent yepPendingIntent = PendingIntent.getService(this, notificationId, yepIntent, PendingIntent.FLAG_CANCEL_CURRENT);

    // doesn't show up on my lock-screen
    //builder.addAction(R.drawable.abc_ic_menu_share_mtrl_alpha, "My Action", yepPendingIntent);

    // asks for unlock code for some reason
    //builder.setContentIntent(yepPendingIntent);

    // Bingo
    RemoteViews view = new RemoteViews(getPackageName(), R.layout.notification);
    view.setOnClickPendingIntent(R.id.notification_closebtn_ib, yepPendingIntent);
    builder.setContent(view);
5
Nanne

Il fonctionne également avec le récepteur de diffusion et setAction

PendingIntent pendingIntent = PendingIntent.getBroadcast(..
builder.addAction(..
   .setVisibility(NotificationCompat.VISIBILITY_PUBLIC)

Sur l'écran de verrouillage, faites glisser la notification vers le bas pour l'agrandir et appuyez sur la zone d'action pour appeler le récepteur de diffusion sans déverrouiller le téléphone.

1
farid_z