WorkManager хранить уведомления после завершения работы - PullRequest
0 голосов
/ 15 марта 2020

Я хочу отображать уведомление, когда Worker работает в фоновом режиме. Я могу сделать это с помощью кода, приведенного ниже:

override suspend fun doWork(): Result {
    val manager = NotificationManagerCompat.from(applicationContext)
    val channel = "some_channel"
    val id = 15
    val builder = NotificationCompat.Builder(applicationContext, channel)

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O)
        manager.createNotificationChannel(NotificationChannel(channel, "DefaultName", NotificationManager.IMPORTANCE_LOW))

    builder
        .setOnlyAlertOnce(true)
        .setOngoing(true)
        .setAutoCancel(false)
        .setPriority(NotificationCompat.PRIORITY_HIGH)
        .setSmallIcon(android.R.drawable.ic_dialog_alert)
        .setTicker("Background Task")
        .setContentText("Starting")
        .setProgress(0, 0, true)

    setForeground(ForegroundInfo(id, builder.build()))

    delay(500)

    manager.notify(id, builder.setContentText("Progress A").setProgress(20, 0, false).build())
    for (i in 1..20) {
        delay(100)
        manager.notify(id, builder.setProgress(20, i, false).build())
    }

    delay(500)

    manager.notify(id, builder.setContentText("Progress B").setProgress(2, 0, false).build())
    delay(1000)
    manager.notify(id, builder.setProgress(2, 1, false).build())
    delay(1000)
    manager.notify(id, builder.setProgress(2, 2, false).build())

    delay(1000)

    manager.notify(id, builder.setContentText("Done!").setProgress(0, 0, false).build())

    delay(500)

    return Result.success()
}

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

Как сохранить уведомление активным? .setOngoing(true) и .setAutoCancel(false) не помогли.

Ответы [ 2 ]

0 голосов
/ 24 марта 2020

Я думаю, что гораздо более простой способ - это уведомить окончательное уведомление с другим идентификатором (использовать .setOngoing (false)) :

manager.notify(
    id + 1,
    builder.setContentText("Done!")
        .setAutoCancel(true)
        .setOngoing(false)
        .build()
)
return Result.success()

Или отправить его после небольшой задержки:

CoroutineScope(Main).launch {
    delay(200)
    manager.notify(
        id,
        builder.setContentText("Done!")
            .setAutoCancel(true)
            .setOngoing(false)
            .build()
    )
}
return Result.success()
0 голосов
/ 16 марта 2020

Обходной путь, который я нашел, должен иметь BroadcastReceiver, который принимает окончательное уведомление и показывает его. Это возможно, потому что Уведомление расширяет Parcelable. Примечание: пусть NotificationReceiver использует другой идентификатор уведомления, чем тот, который использует ваш работник, в противном случае работник закроет уведомление.

class NotificationReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        if (intent.action == ACTION_SEND_NOTIFICATION) {
            val notification = intent.getParcelableExtra<Notification>(EXTRA_NOTIFICATION)
            val id = intent.getIntExtra(EXTRA_ID, DEFAULT_ID)
            if (notification == null)
                Log.e(TAG, "EXTRA_NOTIFICATION ($EXTRA_NOTIFICATION) was not provided!")
            else
                NotificationManagerCompat.from(context).notify(id, notification)
        }
    }

    companion object {
        const val TAG = "NotificationReceiver"

        const val ACTION_SEND_NOTIFICATION = "intent.action.SEND_NOTIFICATION"

        const val EXTRA_NOTIFICATION = "notification_parcel"
        const val EXTRA_ID = "notification_id"
        const val DEFAULT_ID = 1010
    }

}
<receiver android:name=".NotificationReceiver" android:enabled="true" />
    // At the end of doWork, before returning
    applicationContext.sendBroadcast(
        Intent(applicationContext, NotificationReceiver::class.java).setAction(
            NotificationReceiver.ACTION_SEND_NOTIFICATION
        ).putExtra(NotificationReceiver.EXTRA_NOTIFICATION, builder.build())
    )
...