mirror of
https://github.com/status-im/status-desktop.git
synced 2025-01-10 22:36:24 +00:00
feat(chat): implement marking specific messages as seen
iterates: #9069
This commit is contained in:
parent
2e68c97c8e
commit
e822c37716
@ -224,6 +224,9 @@ proc unblockChat*(self: Controller) =
|
|||||||
proc markAllMessagesRead*(self: Controller) =
|
proc markAllMessagesRead*(self: Controller) =
|
||||||
self.messageService.markAllMessagesRead(self.chatId)
|
self.messageService.markAllMessagesRead(self.chatId)
|
||||||
|
|
||||||
|
proc markMessageRead*(self: Controller, msgID: string) =
|
||||||
|
self.messageService.markCertainMessagesRead(self.chatId, @[msgID])
|
||||||
|
|
||||||
proc clearChatHistory*(self: Controller) =
|
proc clearChatHistory*(self: Controller) =
|
||||||
self.chatService.clearChatHistory(self.chatId)
|
self.chatService.clearChatHistory(self.chatId)
|
||||||
|
|
||||||
|
@ -101,6 +101,9 @@ method unblockChat*(self: AccessInterface) {.base.} =
|
|||||||
method markAllMessagesRead*(self: AccessInterface) {.base.} =
|
method markAllMessagesRead*(self: AccessInterface) {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
|
method markMessageRead*(self: AccessInterface, msgID: string) {.base.} =
|
||||||
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method clearChatHistory*(self: AccessInterface) {.base.} =
|
method clearChatHistory*(self: AccessInterface) {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
|
@ -132,7 +132,10 @@ proc init*(self: Controller) =
|
|||||||
let args = MessagesMarkedAsReadArgs(e)
|
let args = MessagesMarkedAsReadArgs(e)
|
||||||
if(self.chatId != args.chatId):
|
if(self.chatId != args.chatId):
|
||||||
return
|
return
|
||||||
self.delegate.markAllMessagesRead()
|
if(args.allMessagesMarked):
|
||||||
|
self.delegate.markAllMessagesRead()
|
||||||
|
else:
|
||||||
|
self.delegate.markMessagesAsRead(args.messagesIds)
|
||||||
|
|
||||||
self.events.on(SIGNAL_CONTACT_NICKNAME_CHANGED) do(e: Args):
|
self.events.on(SIGNAL_CONTACT_NICKNAME_CHANGED) do(e: Args):
|
||||||
var args = ContactArgs(e)
|
var args = ContactArgs(e)
|
||||||
|
@ -164,6 +164,9 @@ method resetAndScrollToNewMessagesMarker*(self: AccessInterface) =
|
|||||||
method markAllMessagesRead*(self: AccessInterface) =
|
method markAllMessagesRead*(self: AccessInterface) =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
|
method markMessagesAsRead*(self: AccessInterface, messages: seq[string]) =
|
||||||
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method updateCommunityDetails*(self: AccessInterface, community: CommunityDto) =
|
method updateCommunityDetails*(self: AccessInterface, community: CommunityDto) =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
|
@ -712,6 +712,9 @@ method removeNewMessagesMarker*(self: Module) =
|
|||||||
method markAllMessagesRead*(self: Module) =
|
method markAllMessagesRead*(self: Module) =
|
||||||
self.view.model().markAllAsSeen()
|
self.view.model().markAllAsSeen()
|
||||||
|
|
||||||
|
method markMessagesAsRead*(self: Module, messages: seq[string]) =
|
||||||
|
self.view.model().markAsSeen(messages)
|
||||||
|
|
||||||
method updateCommunityDetails*(self: Module, community: CommunityDto) =
|
method updateCommunityDetails*(self: Module, community: CommunityDto) =
|
||||||
self.view.setAmIChatAdmin(community.admin)
|
self.view.setAmIChatAdmin(community.admin)
|
||||||
self.view.setIsPinMessageAllowedForMembers(community.adminSettings.pinMessageAllMembersEnabled)
|
self.view.setIsPinMessageAllowedForMembers(community.adminSettings.pinMessageAllMembersEnabled)
|
||||||
|
@ -274,6 +274,9 @@ method unblockChat*(self: Module) =
|
|||||||
method markAllMessagesRead*(self: Module) =
|
method markAllMessagesRead*(self: Module) =
|
||||||
self.controller.markAllMessagesRead()
|
self.controller.markAllMessagesRead()
|
||||||
|
|
||||||
|
method markMessageRead*(self: Module, msgID: string) =
|
||||||
|
self.controller.markMessageRead(msgID)
|
||||||
|
|
||||||
method clearChatHistory*(self: Module) =
|
method clearChatHistory*(self: Module) =
|
||||||
self.controller.clearChatHistory()
|
self.controller.clearChatHistory()
|
||||||
|
|
||||||
|
@ -86,6 +86,9 @@ QtObject:
|
|||||||
proc markAllMessagesRead*(self: View) {.slot.} =
|
proc markAllMessagesRead*(self: View) {.slot.} =
|
||||||
self.delegate.markAllMessagesRead()
|
self.delegate.markAllMessagesRead()
|
||||||
|
|
||||||
|
proc markMessageRead*(self: View, msgID: string) {.slot.} =
|
||||||
|
self.delegate.markMessageRead(msgID)
|
||||||
|
|
||||||
proc clearChatHistory*(self: View) {.slot.} =
|
proc clearChatHistory*(self: View) {.slot.} =
|
||||||
self.delegate.clearChatHistory()
|
self.delegate.clearChatHistory()
|
||||||
|
|
||||||
|
@ -147,14 +147,12 @@ proc init*(self: Controller) =
|
|||||||
self.events.on(message_service.SIGNAL_MESSAGES_MARKED_AS_READ) do(e: Args):
|
self.events.on(message_service.SIGNAL_MESSAGES_MARKED_AS_READ) do(e: Args):
|
||||||
let args = message_service.MessagesMarkedAsReadArgs(e)
|
let args = message_service.MessagesMarkedAsReadArgs(e)
|
||||||
# update chat entity in chat service
|
# update chat entity in chat service
|
||||||
var chat = self.chatService.getChatById(args.chatId)
|
let chat = self.chatService.getChatById(args.chatId)
|
||||||
if ((self.isCommunitySection and chat.communityId != self.sectionId) or
|
if ((self.isCommunitySection and chat.communityId != self.sectionId) or
|
||||||
(not self.isCommunitySection and chat.communityId != "")):
|
(not self.isCommunitySection and chat.communityId != "")):
|
||||||
return
|
return
|
||||||
chat.unviewedMessagesCount = 0
|
self.chatService.updateUnreadMessagesAndMentions(args.chatId, args.allMessagesMarked, args.messagesCount, args.messagesWithMentionsCount)
|
||||||
chat.unviewedMentionsCount = 0
|
self.delegate.updateUnreadMessagesAndMentions(args.chatId)
|
||||||
self.chatService.updateOrAddChat(chat)
|
|
||||||
self.delegate.onMarkAllMessagesRead(args.chatId)
|
|
||||||
|
|
||||||
self.events.on(chat_service.SIGNAL_CHAT_LEFT) do(e: Args):
|
self.events.on(chat_service.SIGNAL_CHAT_LEFT) do(e: Args):
|
||||||
let args = chat_service.ChatArgs(e)
|
let args = chat_service.ChatArgs(e)
|
||||||
|
@ -111,7 +111,7 @@ method onChatMuted*(self: AccessInterface, chatId: string) {.base.} =
|
|||||||
method onChatUnmuted*(self: AccessInterface, chatId: string) {.base.} =
|
method onChatUnmuted*(self: AccessInterface, chatId: string) {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method onMarkAllMessagesRead*(self: AccessInterface, chatId: string) {.base.} =
|
method updateUnreadMessagesAndMentions*(self: AccessInterface, chatId: string) {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method onContactAdded*(self: AccessInterface, publicKey: string) {.base.} =
|
method onContactAdded*(self: AccessInterface, publicKey: string) {.base.} =
|
||||||
|
@ -827,9 +827,12 @@ method onJoinedCommunity*(self: Module) =
|
|||||||
method onUserAuthenticated*(self: Module, pin: string, password: string, keyUid: string) =
|
method onUserAuthenticated*(self: Module, pin: string, password: string, keyUid: string) =
|
||||||
self.controller.requestToJoinCommunityAuthenticated(password)
|
self.controller.requestToJoinCommunityAuthenticated(password)
|
||||||
|
|
||||||
method onMarkAllMessagesRead*(self: Module, chatId: string) =
|
method updateUnreadMessagesAndMentions*(self: Module, chatId: string) =
|
||||||
self.updateBadgeNotifications(chatId, hasUnreadMessages=false, unviewedMentionsCount=0)
|
|
||||||
let chatDetails = self.controller.getChatDetails(chatId)
|
let chatDetails = self.controller.getChatDetails(chatId)
|
||||||
|
self.updateBadgeNotifications(
|
||||||
|
chatId=chatId,
|
||||||
|
hasUnreadMessages=chatDetails.unviewedMessagesCount > 0,
|
||||||
|
unviewedMentionsCount=chatDetails.unviewedMentionsCount)
|
||||||
if chatDetails.categoryId != "":
|
if chatDetails.categoryId != "":
|
||||||
let hasUnreadMessages = self.controller.chatsWithCategoryHaveUnreadMessages(chatDetails.communityId, chatDetails.categoryId)
|
let hasUnreadMessages = self.controller.chatsWithCategoryHaveUnreadMessages(chatDetails.communityId, chatDetails.categoryId)
|
||||||
self.view.chatsModel().setCategoryHasUnreadMessages(chatDetails.categoryId, hasUnreadMessages)
|
self.view.chatsModel().setCategoryHasUnreadMessages(chatDetails.categoryId, hasUnreadMessages)
|
||||||
|
@ -713,6 +713,21 @@ QtObject:
|
|||||||
let index = self.createIndex(i, 0, nil)
|
let index = self.createIndex(i, 0, nil)
|
||||||
self.dataChanged(index, index, @[ModelRole.Seen.int])
|
self.dataChanged(index, index, @[ModelRole.Seen.int])
|
||||||
|
|
||||||
|
proc markAsSeen*(self: Model, messages: seq[string]) =
|
||||||
|
var messagesSet = toHashSet(messages)
|
||||||
|
|
||||||
|
for i in 0 ..< self.items.len:
|
||||||
|
let currentItemID = self.items[i].id
|
||||||
|
|
||||||
|
if messagesSet.contains(currentItemID):
|
||||||
|
self.items[i].seen = true
|
||||||
|
let index = self.createIndex(i, 0, nil)
|
||||||
|
self.dataChanged(index, index, @[ModelRole.Seen.int])
|
||||||
|
messagesSet.excl(currentItemID)
|
||||||
|
|
||||||
|
if messagesSet.len == 0:
|
||||||
|
return
|
||||||
|
|
||||||
proc updateAlbumIfExists*(self: Model, albumId: string, messageImage: string, messageId: string): bool =
|
proc updateAlbumIfExists*(self: Model, albumId: string, messageImage: string, messageId: string): bool =
|
||||||
for i in 0 ..< self.items.len:
|
for i in 0 ..< self.items.len:
|
||||||
let item = self.items[i]
|
let item = self.items[i]
|
||||||
|
@ -725,3 +725,15 @@ QtObject:
|
|||||||
result.add(member)
|
result.add(member)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
error "error while getting members", msg = e.msg, communityID, chatId
|
error "error while getting members", msg = e.msg, communityID, chatId
|
||||||
|
|
||||||
|
proc updateUnreadMessagesAndMentions*(self: Service, chatID: string, markAllAsRead: bool, markAsReadCount: int, markAsReadMentionsCount: int) =
|
||||||
|
var chat = self.getChatById(chatID)
|
||||||
|
if chat.id == "":
|
||||||
|
return
|
||||||
|
if markAllAsRead:
|
||||||
|
chat.unviewedMessagesCount = 0
|
||||||
|
chat.unviewedMentionsCount = 0
|
||||||
|
else:
|
||||||
|
chat.unviewedMessagesCount = max(0, chat.unviewedMessagesCount - markAsReadCount)
|
||||||
|
chat.unviewedMentionsCount = max(0, chat.unviewedMentionsCount - markAsReadMentionsCount)
|
||||||
|
self.updateOrAddChat(chat)
|
||||||
|
@ -142,16 +142,21 @@ const asyncMarkCertainMessagesReadTask: Task = proc(argEncoded: string) {.gcsafe
|
|||||||
|
|
||||||
let response = status_go.markCertainMessagesFromChatWithIdAsRead(arg.chatId, arg.messagesIds)
|
let response = status_go.markCertainMessagesFromChatWithIdAsRead(arg.chatId, arg.messagesIds)
|
||||||
|
|
||||||
var numberOfAffectedMessages: int
|
var count: int
|
||||||
discard response.result.getProp("count", numberOfAffectedMessages)
|
discard response.result.getProp("count", count)
|
||||||
|
|
||||||
|
var countWithMentions: int
|
||||||
|
discard response.result.getProp("countWithMentions", countWithMentions)
|
||||||
|
|
||||||
var error = ""
|
var error = ""
|
||||||
if(numberOfAffectedMessages == 0):
|
if(count == 0):
|
||||||
error = "no message has updated"
|
error = "no message has updated"
|
||||||
|
|
||||||
let responseJson = %*{
|
let responseJson = %*{
|
||||||
"chatId": arg.chatId,
|
"chatId": arg.chatId,
|
||||||
"messagesIds": arg.messagesIds,
|
"messagesIds": arg.messagesIds,
|
||||||
|
"count": count,
|
||||||
|
"countWithMentions": countWithMentions,
|
||||||
"error": error
|
"error": error
|
||||||
}
|
}
|
||||||
arg.finish(responseJson)
|
arg.finish(responseJson)
|
||||||
|
@ -88,6 +88,8 @@ type
|
|||||||
chatId*: string
|
chatId*: string
|
||||||
allMessagesMarked*: bool
|
allMessagesMarked*: bool
|
||||||
messagesIds*: seq[string]
|
messagesIds*: seq[string]
|
||||||
|
messagesCount*: int
|
||||||
|
messagesWithMentionsCount*: int
|
||||||
|
|
||||||
MessageAddRemoveReactionArgs* = ref object of Args
|
MessageAddRemoveReactionArgs* = ref object of Args
|
||||||
chatId*: string
|
chatId*: string
|
||||||
@ -640,7 +642,7 @@ QtObject:
|
|||||||
var error: string
|
var error: string
|
||||||
discard responseObj.getProp("error", error)
|
discard responseObj.getProp("error", error)
|
||||||
if(error.len > 0):
|
if(error.len > 0):
|
||||||
error "error: ", procName="onMarkCertainMessagesRead", errDescription=error
|
error "error: ", procName="onMarkAllMessagesRead", errDescription=error
|
||||||
return
|
return
|
||||||
|
|
||||||
var chatId: string
|
var chatId: string
|
||||||
@ -681,7 +683,21 @@ QtObject:
|
|||||||
for id in messagesIdsArr:
|
for id in messagesIdsArr:
|
||||||
messagesIds.add(id.getStr)
|
messagesIds.add(id.getStr)
|
||||||
|
|
||||||
let data = MessagesMarkedAsReadArgs(chatId: chatId, allMessagesMarked: false, messagesIds: messagesIds)
|
var count: int
|
||||||
|
discard responseObj.getProp("count", count)
|
||||||
|
|
||||||
|
if count < len(messagesIds):
|
||||||
|
warn "warning: ", procName="onMarkCertainMessagesRead", errDescription="not all messages has been marked as read"
|
||||||
|
|
||||||
|
var countWithMentions: int
|
||||||
|
discard responseObj.getProp("countWithMentions", countWithMentions)
|
||||||
|
|
||||||
|
let data = MessagesMarkedAsReadArgs(
|
||||||
|
chatId: chatId,
|
||||||
|
allMessagesMarked: false,
|
||||||
|
messagesIds: messagesIds,
|
||||||
|
messagesCount: count,
|
||||||
|
messagesWithMentionsCount: countWithMentions)
|
||||||
self.events.emit(SIGNAL_MESSAGES_MARKED_AS_READ, data)
|
self.events.emit(SIGNAL_MESSAGES_MARKED_AS_READ, data)
|
||||||
|
|
||||||
proc markCertainMessagesRead*(self: Service, chatId: string, messagesIds: seq[string]) =
|
proc markCertainMessagesRead*(self: Service, chatId: string, messagesIds: seq[string]) =
|
||||||
|
@ -311,3 +311,38 @@ suite "simulations":
|
|||||||
check(model.items[5].id == message1.id)
|
check(model.items[5].id == message1.id)
|
||||||
check(model.items[6].id == message0_fetchMoreMessages.id)
|
check(model.items[6].id == message0_fetchMoreMessages.id)
|
||||||
check(model.items[7].id == message0_chatIdentifier.id)
|
check(model.items[7].id == message0_chatIdentifier.id)
|
||||||
|
|
||||||
|
|
||||||
|
suite "mark as seen":
|
||||||
|
setup:
|
||||||
|
let model = newModel()
|
||||||
|
|
||||||
|
var msg1 = createTestMessageItem("0xa", 1)
|
||||||
|
msg1.seen=false
|
||||||
|
let msg2 = createTestMessageItem("0xb", 2)
|
||||||
|
msg2.seen=false
|
||||||
|
let msg3 = createTestMessageItem("0xc", 3)
|
||||||
|
msg3.seen=true
|
||||||
|
|
||||||
|
model.insertItemsBasedOnClock(@[msg1, msg2, msg3])
|
||||||
|
require(model.items.len == 3)
|
||||||
|
check(model.items[0].seen == true)
|
||||||
|
check(model.items[1].seen == false)
|
||||||
|
check(model.items[2].seen == false)
|
||||||
|
|
||||||
|
test "mark all as seen":
|
||||||
|
model.markAllAsSeen()
|
||||||
|
check(model.items[0].seen == true)
|
||||||
|
check(model.items[1].seen == true)
|
||||||
|
check(model.items[2].seen == true)
|
||||||
|
|
||||||
|
test "mark some as seen":
|
||||||
|
model.markAsSeen(@["0xa"])
|
||||||
|
check(model.items[0].seen == true)
|
||||||
|
check(model.items[1].seen == false)
|
||||||
|
check(model.items[2].seen == true)
|
||||||
|
|
||||||
|
model.markAsSeen(@["0xb"])
|
||||||
|
check(model.items[0].seen == true)
|
||||||
|
check(model.items[1].seen == true)
|
||||||
|
check(model.items[2].seen == true)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user