Wysyłanie powiadomienia z usługi w systemie Android

105

Mam uruchomioną usługę i chcę wysłać powiadomienie. Szkoda, obiekt powiadomienia wymaga a Context, jak an Activity, a nie Service.

Czy wiesz, jak to ominąć? Próbowałem utworzyć Activitydla każdego powiadomienia, ale wydaje się brzydkie i nie mogę znaleźć sposobu na uruchomienie Activitybez żadnego View.

e-satis
źródło
14
Umm ... Usługa to kontekst!
Isaac Waller
19
Boże, jestem takim głupkiem. Ok, przepraszam za marnowanie czasu.
e-satis
28
W porządku - to dobre pytanie od Google.
Isaac Waller
Jak twój drugi komentarz: D: D
Faizan Mubasher
Ten post właśnie uratował mi dzień ...
Muhammad Faizan

Odpowiedzi:

109

Zarówno Activityi Servicefaktycznie extend Context, możesz po prostu używać thisjako Contextswojego 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);
Josef Pfleger
źródło
4
Pamiętaj, że będziesz mieć wiele problemów z wykonaniem powiadomienia z usługi. Jeśli masz problemy, spójrz na to groups.google.com/group/android-developers/browse_thread/thread/…
Karussell
1
jak możesz to zrobić za pomocą Notification.Builder? ponieważ setLatestEventInfo jest już przestarzałe.
Kairi San,
77

Ten typ powiadomienia jest przestarzały, co widać w dokumentach:

@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 */ }

Lepszy sposób
Możesz wysłać takie powiadomienie:

// 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); 

Najlepszy sposób, aby
powyższy kod wymagał minimalnego poziomu API 11 (Android 3.0).
Jeśli twój minimalny poziom API jest niższy niż 11, powinieneś użyć klasy NotificationCompat biblioteki obsługi w ten sposób.

Jeśli więc minimalny docelowy poziom interfejsu API to 4+ (Android 1.6+), użyj tego:

    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());
trante
źródło
4
to powinna być najlepsza odpowiedź, ponieważ zaakceptowana jest przestarzała
Marcel Krivek
4
@MarcelKrivek Wygląda na to, że „zapomniał” o zacytowaniu źródła. vogella.com/tutorials/AndroidNotifications/article.html
StarWind0
Co to jest „NotificationReceiver”?
user3690202
„NotificationReceiver” to działanie, które zostanie otwarte przez powiadomienie. Sprawdź link podany przez @ StarWind0.
George Theodorakis
NotificationCompat.Builder jest już przestarzałe. To już nie jest najlepsza odpowiedź
Devil's Dream,
7
@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);
}
王怡飞
źródło
Pytanie tematyczne pochodzi z usługi SERVICE not Activity
Duna,
1

Cóż, nie jestem pewien, czy moje rozwiązanie jest najlepszą praktyką. Korzystanie z NotificationBuildermojego kodu wygląda następująco:

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());
    }

Oczywisty:

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

a tutaj Serwis:

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

Nie wiem, czy naprawdę istnieje singleTaskat, Serviceale to działa poprawnie w mojej aplikacji ...

Martin Pfeffer
źródło
co to za budowniczy w tym?
Viswanath Lekshmanan
-2

Jeśli żadne z tych rozwiązań nie działa, spróbuj getBaseContext()zamiast contextlub this.

kurdist android
źródło
2
Nie powinieneś używać getBaseContext()tych scenariuszy.
Rahat Zaman