Как отправлять уведомления на Android в определенное время в Android Oreo?

Я ищу способ создать preference в Settings для отправки уведомления в определенное время дня (установленное пользователем в настройках) в приложении для Android. Я просмотрел разные темы, такие как this, однако это не работает в Android Oreo.

Может ли кто-нибудь помочь мне с этим или указать на учебное пособие?


person Tricky Bay    schedule 14.07.2018    source источник
comment
Поскольку никто не ответил ... Действительно ли уведомление нужно отправлять в определенное время? Во многих случаях нет, но если да, то AlarmManager.setAlarmClock - единственная реальная локальная альтернатива. Другое решение - использовать FCM с высоким приоритетом: сообщение Firebase Cloud. Если допускаются задержки до ~ 10 минут, то может оказаться полезным что-то вроде AlarmManager.setExactAndAllowWhileIdle. В противном случае посмотрите, может ли ваше приложение использовать JobScheduler.   -  person Elletlar    schedule 16.07.2018
comment
Да, я хочу отправлять ежедневное уведомление в указанное пользователем время. В моем случае может работать Alarm Manager, но я ищу способ реализации. Учебник, который работает для Android Oreo, если быть точным.   -  person Tricky Bay    schedule 17.07.2018


Ответы (2)


После просмотра различных сообщений и некоторых исследований реализации AlarmManager, это то, что у меня сработало.

Основой для этого является этот пост и Расписание повторяющихся сигналов тревоги Android Документация.

Это моя текущая реализация:

У меня SwitchPreference и TimePicker реализация Settings

SwitchPreference, чтобы спросить, хочет ли пользователь включить повторяющиеся ежедневные уведомления.

TimePicker, чтобы установить время уведомления.

В методе OnCreate MainActivity или везде, где вы читаете SharedPreferences, сделайте следующее:

PreferenceManager.setDefaultValues(this, R.xml.preferences, false);
SharedPreferences sharedPref = PreferenceManager.getDefaultSharedPreferences(this);
Boolean dailyNotify = sharedPref.getBoolean(SettingsActivity.KEY_PREF_DAILY_NOTIFICATION, true);
PackageManager pm = this.getPackageManager();
ComponentName receiver = new ComponentName(this, DeviceBootReceiver.class);
Intent alarmIntent = new Intent(this, AlarmReceiver.class);
PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0, alarmIntent, 0);
AlarmManager manager = (AlarmManager) getSystemService(Context.ALARM_SERVICE);

// if user enabled daily notifications
if (dailyNotify) {
    //region Enable Daily Notifications
    Calendar calendar = Calendar.getInstance();
    calendar.setTimeInMillis(System.currentTimeMillis());
    calendar.set(Calendar.HOUR_OF_DAY, sharedPref.getInt("dailyNotificationHour", 7));
    calendar.set(Calendar.MINUTE, sharedPref.getInt("dailyNotificationMin", 15));
    calendar.set(Calendar.SECOND, 1);
    // if notification time is before selected time, send notification the next day
    if (calendar.before(Calendar.getInstance())) {
        calendar.add(Calendar.DATE, 1);
    }
    if (manager != null) {
        manager.setRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
                AlarmManager.INTERVAL_DAY, pendingIntent);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            manager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), pendingIntent);
        }
    }
    //To enable Boot Receiver class
    pm.setComponentEnabledSetting(receiver,
            PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
            PackageManager.DONT_KILL_APP);
    //endregion
} else { //Disable Daily Notifications
    if (PendingIntent.getBroadcast(this, 0, alarmIntent, 0) != null && manager != null) {
        manager.cancel(pendingIntent);
        //Toast.makeText(this,"Notifications were disabled",Toast.LENGTH_SHORT).show();
    }
    pm.setComponentEnabledSetting(receiver,
            PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
            PackageManager.DONT_KILL_APP);
}

Затем добавьте класс AlarmReceiver, который реализует BroadcastReceiver следующим образом:

public class AlarmReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {

    SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(Objects.requireNonNull(context));
    SharedPreferences.Editor sharedPrefEditor = prefs.edit();

    NotificationManager nm = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
    Intent notificationIntent = new Intent(context, MainActivity.class);

    notificationIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP
            | Intent.FLAG_ACTIVITY_SINGLE_TOP);

    PendingIntent pendingI = PendingIntent.getActivity(context, 0,
            notificationIntent, 0);
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
            NotificationChannel channel = new NotificationChannel("default",
                    "Daily Notification",
                    NotificationManager.IMPORTANCE_DEFAULT);
            channel.setDescription("Daily Notification");
            if (nm != null) {
                nm.createNotificationChannel(channel);
            }
        }
        NotificationCompat.Builder b = new NotificationCompat.Builder(context, "default");
        b.setAutoCancel(true)
                .setDefaults(NotificationCompat.DEFAULT_ALL)
                .setWhen(System.currentTimeMillis())
                .setSmallIcon(R.mipmap.ic_launcher_foreground)
                .setTicker("{Time to watch some cool stuff!}")
                .setContentTitle("My Cool App")
                .setContentText("Time to watch some cool stuff!")
                .setContentInfo("INFO")
                .setContentIntent(pendingI);

        if (nm != null) {
            nm.notify(1, b.build());
            Calendar nextNotifyTime = Calendar.getInstance();
            nextNotifyTime.add(Calendar.DATE, 1);
            sharedPrefEditor.putLong("nextNotifyTime", nextNotifyTime.getTimeInMillis());
            sharedPrefEditor.apply();
        }
    }
}

Система выключит AlarmManager, если Устройство выключено или перезагружается, поэтому перезапустите его снова, когда BOOT COMPLETE добавьте этот класс:

public class DeviceBootReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
    if (Objects.equals(intent.getAction(), "android.intent.action.BOOT_COMPLETED")) {
        // on device boot complete, reset the alarm
        Intent alarmIntent = new Intent(context, AlarmReceiver.class);
        PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, alarmIntent, 0);

        AlarmManager manager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
        final SharedPreferences sharedPref = PreferenceManager.getDefaultSharedPreferences(Objects.requireNonNull(context));

        Calendar calendar = Calendar.getInstance();
        calendar.setTimeInMillis(System.currentTimeMillis());
        calendar.set(Calendar.HOUR_OF_DAY, sharedPref.getInt("dailyNotificationHour", 7));
        calendar.set(Calendar.MINUTE, sharedPref.getInt("dailyNotificationMin", 15));
        calendar.set(Calendar.SECOND, 1);

        Calendar newC = new GregorianCalendar();
        newC.setTimeInMillis(sharedPref.getLong("nextNotifyTime", Calendar.getInstance().getTimeInMillis()));

        if (calendar.after(newC)) {
            calendar.add(Calendar.HOUR, 1);
        }

        if (manager != null) {
            manager.setRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
                    AlarmManager.INTERVAL_DAY, pendingIntent);
        }
    }
}
}

И, наконец, не забудьте добавить эти разрешения в AndroidManidest:

<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

и зарегистрируйте свои приемники в AndroidManifest

<application
    <!--YOUR APPLICATION STUFF-->

    <receiver android:name=".DeviceBootReceiver"
        android:enabled="false">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED" />
        </intent-filter>
    </receiver>
    <receiver android:name=".AlarmReceiver" />

Notification должен быть установлен этим в определенное время дня, указанное TimePicker, и если пользователь включил SwitchPreference.

person Tricky Bay    schedule 02.08.2018

В StackOverflow есть масса примеров того, как это сделать, но, к сожалению, ни один из них больше не работает, потому что Google изменил принцип работы AlarmManager.

  • Android 6: дремота
  • Android 7: Aggressive Doze [отключает ЦП вскоре после выключения экрана]
  • Android 8: дополнительные ограничения для фоновых служб

Параметр only AlarmManager, который позволяет разработчику активировать устройство в определенное время, - это AlarmManager.setAlarmClock.

Использование setAlarmClock

Помимо будильника, вы можете установить намерение, которое позволит вам запускать приложение при нажатии на будильник.

Устройство также можно разбудить в определенное время с помощью высокоприоритетного FCM, Firebase Cloud Message.

Но в целом другого варианта нет: setExact и связанные с ним методы больше не работают, как следует из названия.

person Elletlar    schedule 17.07.2018