From 7d3aafdd658a93c6df79f35babaed358672a2bd9 Mon Sep 17 00:00:00 2001 From: Konrad Pozniak Date: Fri, 23 Feb 2024 10:26:46 +0100 Subject: [PATCH] fix quick replies from notifications (#4250) While working on #4249 I noticed that quick replies also don't work as expected. The notification just stays in the sending state forever. There are actually 2 problems: - Notifications are sent in `NotificationFetcher` with the id of the Mastodon notification as tag and the current account id as id. The wrong notification id was forwarded to `SendStatusBroadcastReceiver` so it never had a chance of updating the notification. - Notifications containing an active remote input can't be cancelled (they just stop their animation when doing so). So instead I update the notification with info that the reply is being sent and have it dismiss automatically. I also tried replacing the original notification with the "sending" notification of `SendStatusService`, but that doesn't work because `Service.startForeground` doesn't have a tag parameter, only an id. --------- Co-authored-by: Willow --- .../notifications/NotificationHelper.java | 4 +- .../receiver/SendStatusBroadcastReceiver.kt | 47 +++++++++---------- app/src/main/res/values/strings.xml | 4 +- 3 files changed, 27 insertions(+), 28 deletions(-) diff --git a/app/src/main/java/com/keylesspalace/tusky/components/notifications/NotificationHelper.java b/app/src/main/java/com/keylesspalace/tusky/components/notifications/NotificationHelper.java index 14012c6e..bdbc9fa8 100644 --- a/app/src/main/java/com/keylesspalace/tusky/components/notifications/NotificationHelper.java +++ b/app/src/main/java/com/keylesspalace/tusky/components/notifications/NotificationHelper.java @@ -97,7 +97,7 @@ public class NotificationHelper { public static final String KEY_SENDER_ACCOUNT_FULL_NAME = "KEY_SENDER_ACCOUNT_FULL_NAME"; - public static final String KEY_NOTIFICATION_ID = "KEY_NOTIFICATION_ID"; + public static final String KEY_SERVER_NOTIFICATION_ID = "KEY_SERVER_NOTIFICATION_ID"; public static final String KEY_CITED_STATUS_ID = "KEY_CITED_STATUS_ID"; @@ -412,7 +412,7 @@ public class NotificationHelper { .putExtra(KEY_SENDER_ACCOUNT_ID, account.getId()) .putExtra(KEY_SENDER_ACCOUNT_IDENTIFIER, account.getIdentifier()) .putExtra(KEY_SENDER_ACCOUNT_FULL_NAME, account.getFullName()) - .putExtra(KEY_NOTIFICATION_ID, notificationId) + .putExtra(KEY_SERVER_NOTIFICATION_ID, body.getId()) .putExtra(KEY_CITED_STATUS_ID, inReplyToId) .putExtra(KEY_VISIBILITY, replyVisibility) .putExtra(KEY_SPOILER, contentWarning) diff --git a/app/src/main/java/com/keylesspalace/tusky/receiver/SendStatusBroadcastReceiver.kt b/app/src/main/java/com/keylesspalace/tusky/receiver/SendStatusBroadcastReceiver.kt index d175522b..083cdeca 100644 --- a/app/src/main/java/com/keylesspalace/tusky/receiver/SendStatusBroadcastReceiver.kt +++ b/app/src/main/java/com/keylesspalace/tusky/receiver/SendStatusBroadcastReceiver.kt @@ -45,7 +45,7 @@ class SendStatusBroadcastReceiver : BroadcastReceiver() { AndroidInjection.inject(this, context) if (intent.action == NotificationHelper.REPLY_ACTION) { - val notificationId = intent.getIntExtra(NotificationHelper.KEY_NOTIFICATION_ID, -1) + val serverNotificationId = intent.getStringExtra(NotificationHelper.KEY_SERVER_NOTIFICATION_ID) val senderId = intent.getLongExtra(NotificationHelper.KEY_SENDER_ACCOUNT_ID, -1) val senderIdentifier = intent.getStringExtra( NotificationHelper.KEY_SENDER_ACCOUNT_IDENTIFIER @@ -69,24 +69,23 @@ class SendStatusBroadcastReceiver : BroadcastReceiver() { if (account == null) { Log.w(TAG, "Account \"$senderId\" not found in database. Aborting quick reply!") - val builder = NotificationCompat.Builder( + val notification = NotificationCompat.Builder( context, NotificationHelper.CHANNEL_MENTION + senderIdentifier ) .setSmallIcon(R.drawable.ic_notify) .setColor(context.getColor(R.color.tusky_blue)) .setGroup(senderFullName) - .setDefaults(0) // So it doesn't ring twice, notify only in Target callback + .setDefaults(0) // We don't want this to make any sound or vibration + .setOnlyAlertOnce(true) + .setContentTitle(context.getString(R.string.error_generic)) + .setContentText(context.getString(R.string.error_sender_account_gone)) + .setSubText(senderFullName) + .setVisibility(NotificationCompat.VISIBILITY_PUBLIC) + .setCategory(NotificationCompat.CATEGORY_SOCIAL) + .build() - builder.setContentTitle(context.getString(R.string.error_generic)) - builder.setContentText(context.getString(R.string.error_sender_account_gone)) - - builder.setSubText(senderFullName) - builder.setVisibility(NotificationCompat.VISIBILITY_PUBLIC) - builder.setCategory(NotificationCompat.CATEGORY_SOCIAL) - builder.setOnlyAlertOnce(true) - - notificationManager.notify(notificationId, builder.build()) + notificationManager.notify(serverNotificationId, senderId.toInt(), notification) } else { val text = mentions.joinToString(" ", postfix = " ") { "@$it" } + message.toString() @@ -114,25 +113,25 @@ class SendStatusBroadcastReceiver : BroadcastReceiver() { context.startService(sendIntent) - val builder = NotificationCompat.Builder( + // Notifications with remote input active can't be cancelled, so let's replace it with another one that will dismiss automatically + val notification = NotificationCompat.Builder( context, NotificationHelper.CHANNEL_MENTION + senderIdentifier ) .setSmallIcon(R.drawable.ic_notify) .setColor(context.getColor(R.color.notification_color)) .setGroup(senderFullName) - .setDefaults(0) // So it doesn't ring twice, notify only in Target callback + .setDefaults(0) // We don't want this to make any sound or vibration + .setOnlyAlertOnce(true) + .setContentTitle(context.getString(R.string.reply_sending)) + .setContentText(context.getString(R.string.reply_sending_long)) + .setSubText(senderFullName) + .setVisibility(NotificationCompat.VISIBILITY_PUBLIC) + .setCategory(NotificationCompat.CATEGORY_SOCIAL) + .setTimeoutAfter(5000) + .build() - builder.setContentTitle(context.getString(R.string.post_sent)) - builder.setContentText(context.getString(R.string.post_sent_long)) - - builder.setSubText(senderFullName) - builder.setVisibility(NotificationCompat.VISIBILITY_PUBLIC) - builder.setCategory(NotificationCompat.CATEGORY_SOCIAL) - builder.setOnlyAlertOnce(true) - - // There is a separate "I am sending" notification, so simply remove the handled one. - notificationManager.cancel(notificationId) + notificationManager.notify(serverNotificationId, senderId.toInt(), notification) } } } diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 2503cf40..04d767b6 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -227,8 +227,8 @@ %s unhidden #%s unfollowed - Sent! - Reply sent successfully. + Sending… + Your reply is being sent. Which instance? What\'s happening?