feat(notifications): Handle some edge cases

This commit is contained in:
Peter Vacho 2025-01-04 16:36:28 +01:00
parent 8c57542934
commit ef08d6ddd3
Signed by: school
GPG key ID: 8CFC3837052871B4
3 changed files with 74 additions and 20 deletions

View file

@ -25,6 +25,8 @@ import org.json.JSONException
import org.json.JSONObject import org.json.JSONObject
import retrofit2.HttpException import retrofit2.HttpException
// TODO: Localize all strings
class NotificationsActivity : AppCompatActivity() { class NotificationsActivity : AppCompatActivity() {
private lateinit var rvNotifications: RecyclerView private lateinit var rvNotifications: RecyclerView
private lateinit var btnBack: ImageButton private lateinit var btnBack: ImageButton
@ -90,7 +92,11 @@ class NotificationsActivity : AppCompatActivity() {
} }
private fun getInvitationData(invitationId: String, rvPosition: Int): InvitationResponse? { private fun getInvitationData(invitationId: String, rvPosition: Int): InvitationResponse? {
return invitations[invitationId] val ret = invitations[invitationId]
if (ret == null) {
Log.w("NotificationsActivity", "NotificationAdapter requested unknown invitation: $invitationId")
}
return ret
} }
private fun getUserData(userId: String, rvPosition: Int?): UserResponse? { private fun getUserData(userId: String, rvPosition: Int?): UserResponse? {
@ -139,25 +145,39 @@ class NotificationsActivity : AppCompatActivity() {
Toast.makeText(this@NotificationsActivity, "Invitation accepted", Toast.LENGTH_SHORT).show() Toast.makeText(this@NotificationsActivity, "Invitation accepted", Toast.LENGTH_SHORT).show()
// Also mark the notification as read after the interaction // Also mark the notification as read after the interaction
if (!notification.read) handleNotificationClick(notification, position, false) if (!notification.read) {
} handleNotificationClick(notification, position, false)
} else {
// If the notification was unread, handleNotificationClick will have triggered this
// but otherwise, we'll need to trigger an item update ourselves, to make sure that
// the accept/decline buttons are removed
rvNotifications.adapter!!.notifyItemChanged(position)
} }
} }
} }
NotificationAdapter.Action.DECLINE -> { NotificationAdapter.Action.DECLINE -> {
lifecycleScope.launch(Dispatchers.IO) { lifecycleScope.launch(Dispatchers.IO) {
val invitationId = notification.data val invitationId = notification.data
RetrofitClient.invitationService.declineInvitation(invitationId) invitations[invitationId] = RetrofitClient.invitationService.declineInvitation(invitationId)
withContext(Dispatchers.Main) { withContext(Dispatchers.Main) {
Toast.makeText(this@NotificationsActivity, "Invitation declined", Toast.LENGTH_SHORT).show() Toast.makeText(this@NotificationsActivity, "Invitation declined", Toast.LENGTH_SHORT).show()
// Also mark the notification as read after the interaction // Also mark the notification as read after the interaction
if (!notification.read) handleNotificationClick(notification, position, false) if (!notification.read) {
handleNotificationClick(notification, position, false)
} else {
// If the notification was unread, handleNotificationClick will have triggered this
// but otherwise, we'll need to trigger an item update ourselves, to make sure that
// the accept/decline buttons are removed
rvNotifications.adapter!!.notifyItemChanged(position)
}
} }
} }
} }
NotificationAdapter.Action.VIEW_EVENT -> { NotificationAdapter.Action.VIEW_EVENT -> {
// TODO: Handle viewing the event // TODO: Handle viewing the event
if (!notification.read) handleNotificationClick(notification, position, false)
} }
} }
} }

View file

@ -14,6 +14,8 @@ import java.time.Duration
import java.time.OffsetDateTime import java.time.OffsetDateTime
import java.time.format.DateTimeFormatter import java.time.format.DateTimeFormatter
// TODO: Localize all strings
class NotificationAdapter( class NotificationAdapter(
private val notifications: MutableList<NotificationResponse>, private val notifications: MutableList<NotificationResponse>,
private val onActionClick: (NotificationResponse, Action, Int) -> Unit, private val onActionClick: (NotificationResponse, Action, Int) -> Unit,
@ -88,33 +90,65 @@ class NotificationAdapter(
val user = getUserData(usernameId, position) val user = getUserData(usernameId, position)
val username = user?.username ?: "Unknown User" val username = user?.username ?: "Unknown User"
val statusSuffix = when (invitation.status) {
"accepted" -> " [already accepted]"
"declined" -> " [already declined]"
"pending" -> ""
else -> throw IllegalStateException("Unexpected invitation status: ${invitation.status} for invitation ID: ${invitation.id}")
}
return when (notification.message) { return when (notification.message) {
"new-invitation" -> "You have received an event invitation from @$username" "new-invitation" -> "You have received an event invitation from @$username$statusSuffix"
"invitation-accepted" -> "@$username has accepted your event invitation" "invitation-accepted" -> "@$username has accepted your event invitation$statusSuffix"
"invitation-declined" -> "@$username has declined your event invitation" "invitation-declined" -> "@$username has declined your event invitation$statusSuffix"
else -> throw IllegalArgumentException("Unexpected notification message: ${notification.message}") else -> throw IllegalArgumentException("Unexpected notification message: ${notification.message}")
} }
} }
private fun handleInvitationActions( private fun handleInvitationActions(
holder: NotificationViewHolder, holder: NotificationViewHolder,
notification: NotificationResponse, notification: NotificationResponse,
position: Int position: Int
) { ) {
if (notification.event_type == "invitation") { if (notification.event_type == "invitation") {
val invitation = getInvitationData(notification.data, position)
if (invitation == null) {
// No buttons for deleted invitations
holder.invitationActions.visibility = View.GONE
return
}
when (notification.message) { when (notification.message) {
"new-invitation" -> { "new-invitation" -> {
// Show Accept/Decline & View buttons when (invitation.status) {
holder.invitationActions.visibility = View.VISIBLE "pending" -> {
holder.acceptButton.visibility = View.VISIBLE // Show Accept/Decline & View buttons
holder.declineButton.visibility = View.VISIBLE holder.invitationActions.visibility = View.VISIBLE
holder.viewEventButton.visibility = View.VISIBLE holder.acceptButton.visibility = View.VISIBLE
holder.declineButton.visibility = View.VISIBLE
holder.viewEventButton.visibility = View.VISIBLE
holder.acceptButton.setOnClickListener { holder.acceptButton.setOnClickListener {
onActionClick(notification, Action.ACCEPT, position) onActionClick(notification, Action.ACCEPT, position)
} }
holder.declineButton.setOnClickListener { holder.declineButton.setOnClickListener {
onActionClick(notification, Action.DECLINE, position) onActionClick(notification, Action.DECLINE, position)
}
}
"accepted", "declined" -> {
// Show only View button
holder.invitationActions.visibility = View.VISIBLE
holder.acceptButton.visibility = View.GONE
holder.declineButton.visibility = View.GONE
holder.viewEventButton.visibility = View.VISIBLE
holder.viewEventButton.setOnClickListener {
onActionClick(notification, Action.VIEW_EVENT, position)
}
}
else -> throw IllegalStateException("Unexpected invitation status: ${invitation.status} for invite ID: ${invitation.id}")
} }
} }
"invitation-accepted", "invitation-declined" -> { "invitation-accepted", "invitation-declined" -> {

View file

@ -21,9 +21,9 @@ interface InvitationsService {
suspend fun getIncomingInvitations(@Path("user_id") userId: String): List<InvitationResponse> suspend fun getIncomingInvitations(@Path("user_id") userId: String): List<InvitationResponse>
@POST("invitations/{invitation_id}/accept") @POST("invitations/{invitation_id}/accept")
suspend fun acceptInvitation(@Path("invitation_id") invitationId: String): Unit suspend fun acceptInvitation(@Path("invitation_id") invitationId: String): InvitationResponse
@POST("invitations/{invitation_id}/decline") @POST("invitations/{invitation_id}/decline")
suspend fun declineInvitation(@Path("invitation_id") invitationId: String): Unit suspend fun declineInvitation(@Path("invitation_id") invitationId: String): InvitationResponse
} }