PendingIntent не отправляет дополнительные сведения о намерениях

127

Мой MainActicity начинается RefreshServiceс a, у Intentкоторого есть booleanдополнительный вызов isNextWeek.

My RefreshServiceсоздает объект, Notificationкоторый запускает мой, MainActivityкогда пользователь нажимает на него.

это выглядит так:

    Log.d("Refresh", "RefreshService got: isNextWeek: " + String.valueOf(isNextWeek));

    Intent notificationIntent = new Intent(this, MainActivity.class);
    notificationIntent.putExtra(MainActivity.IS_NEXT_WEEK, isNextWeek);

    Log.d("Refresh", "RefreshService put in Intent: isNextWeek: " + String.valueOf(notificationIntent.getBooleanExtra(MainActivity.IS_NEXT_WEEK,false)));
    pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);

    builder = new NotificationCompat.Builder(this).setContentTitle("Title").setContentText("ContentText").setSmallIcon(R.drawable.ic_notification).setContentIntent(pendingIntent);
    notification = builder.build();
    // Hide the notification after its selected
    notification.flags |= Notification.FLAG_AUTO_CANCEL;
    notificationManager.notify(NOTIFICATION_REFRESH, notification);

Как видите, в notificationIntentфайле должен быть booleanдополнительный файл, IS_NEXT_WEEKзначение isNextWeekкоторого помещается в PendingIntent.

Когда я щелкаю сейчас, Notificationя всегда получаю falseзначениеisNextWeek

Вот как я получаю значение в MainActivity:

    isNextWeek = getIntent().getBooleanExtra(IS_NEXT_WEEK, false);

Журнал:

08-04 00:19:32.500  13367-13367/de.MayerhoferSimon.Vertretungsplan D/Refresh: MainActivity sent: isNextWeek: true
08-04 00:19:32.510  13367-13573/de.MayerhoferSimon.Vertretungsplan D/Refresh: RefreshService got: isNextWeek: true
08-04 00:19:32.510  13367-13573/de.MayerhoferSimon.Vertretungsplan D/Refresh: RefreshService put in Intent: isNextWeek: true
08-04 00:19:41.990  13367-13367/de.MayerhoferSimon.Vertretungsplan D/Refresh: MainActivity.onCreate got: isNextWeek: false

Когда я непосредственно начать MainActivityс Intentс ìsNextValue` , как это:

    Intent i = new Intent(this, MainActivity.class);
    i.putExtra(IS_NEXT_WEEK, isNextWeek);
    finish();
    startActivity(i);

все работает нормально и я получаю trueкогда isNextWeekесть true.

Что я ошибаюсь, что всегда есть falseценность?

ОБНОВИТЬ

это решает проблему: https://stackoverflow.com/a/18049676/2180161

Quote:

Я подозреваю, что, поскольку единственное, что меняется в намерении, - это дополнительные функции, PendingIntent.getActivity(...)фабричный метод просто повторно использует старое намерение в качестве оптимизации.

В RefreshService попробуйте:

PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, PendingIntent.FLAG_CANCEL_CURRENT);

Видеть:

http://developer.android.com/reference/android/app/PendingIntent.html#FLAG_CANCEL_CURRENT

ОБНОВЛЕНИЕ 2

См. Ответ ниже, почему его лучше использовать PendingIntent.FLAG_UPDATE_CURRENT.

maysi
источник
2
PendingIntent.FLAG_CANCEL_CURRENT у меня сработал, спасибо
Pelanes
1
сэкономил мне много часов работы. правильный ответ!
TharakaNirmana
у вас есть вопрос и решение: D отлично. Думаю, вам стоит добавить это как ответ на вопросы. +10 лучше, чем +5;)
Мухаммед Рефаат
Ссылаясь
FLAG_UPDATE_CURRENT в моем случае не хватило, так как тот же PendingIntent был повторно использован моим виджетом. В итоге я использовал FLAG_ONE_SHOT для действия, которое происходит редко, и оставил виджет PendingIntent без изменений.
Эйр

Ответы:

30

Использование PendingIntent.FLAG_CANCEL_CURRENT - не лучшее решение из-за неэффективного использования памяти. Вместо этого используйте PendingIntent.FLAG_UPDATE_CURRENT .

Используйте также Intent.FLAG_ACTIVITY_SINGLE_TOP (действие не будет запущено, если оно уже запущено в верхней части стека истории).

Intent resultIntent = new Intent(this, FragmentPagerSupportActivity.class).
                    addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);  
resultIntent.putExtra(FragmentPagerSupportActivity.PAGE_NUMBER_KEY, pageNumber);

PendingIntent resultPendingIntent =
                PendingIntent.getActivity(
                        this,
                        0,
                        resultIntent,
                        PendingIntent.FLAG_UPDATE_CURRENT
                );

Затем:

@Override
    protected void onCreate(Bundle savedInstanceState) {
        try {
            super.onCreate(savedInstanceState);

            int startPageNumber;
            if ( savedInstanceState != null)
            {
                startPageNumber = savedInstanceState.getInt(PAGE_NUMBER_KEY);
//so on

Теперь он должен работать.


Если вы по-прежнему не ожидаете поведения, попробуйте реализовать void onNewIntent(Intent intent)обработчик событий, таким образом вы сможете получить доступ к новому намерению, которое было вызвано для действия (что не то же самое, что просто вызов getIntent (), это всегда будет возвращать первое запущенное намерение ваша деятельность.

@Override
protected void onNewIntent(Intent intent) {
    int startPageNumber;

    if (intent != null) {
        startPageNumber = intent.getExtras().getInt(PAGE_NUMBER_KEY);
    } else {
        startPageNumber = 0;
    }
}
Юлия Ашомок
источник
21

Я думаю, вам нужно обновить, Intentкогда вы получите новый, переопределив onNewIntent(Intent)в своей деятельности. Добавьте в свою деятельность следующее:

@Override
public void onNewIntent(Intent newIntent) {
    this.setIntent(newIntent);

    // Now getIntent() returns the updated Intent
    isNextWeek = getIntent().getBooleanExtra(IS_NEXT_WEEK, false);        
}

Редактировать:

Это необходимо только в том случае, если ваша деятельность уже была запущена на момент получения намерения. Если ваша деятельность началась (а не просто возобновлена) по намерению, значит, проблема в другом месте, и мое предложение не поможет ее исправить.

Викрам
источник
это появляется также, если действие закрыто, а затем открылось с уведомлением.
майси
3

Следующий код должен работать: -

int icon = R.drawable.icon;
String message = "hello";
long when = System.currentTimeMillis();
NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
Notification notification = new Notification(icon, message, when);

Intent notificationIntent = new Intent(context, MainActivity.class);
notificationIntent.putExtra("isNexWeek", true);
notificationIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
PendingIntent pIntent = PendingIntent.getActivity(context, 0, notificationIntent, 0);
notification.setLatestEventInfo(context, title, message, pIntent);
notification.flags |= Notification.FLAG_AUTO_CANCEL;
notificationManager.notify(0, notification);

В MainActivity onCreate:

if (getIntent().getExtras() != null && getIntent().getExtras().containsKey("isNextWeek")) {
        boolean isNextWeek = getIntent().getExtras().getBoolean("isNextWeek");
}
Харис ур Рехман
источник
new Notification(icon, message, when);устарело
maysi 03
Да и вообще без CLEAR_TOP сделать?
Ричард