parent
c0013a0956
commit
414b39d7e0
|
@ -41,6 +41,8 @@ proc init*(self: ChatController) =
|
||||||
self.status.chat.init(pubKey, messagesFromContactsOnly)
|
self.status.chat.init(pubKey, messagesFromContactsOnly)
|
||||||
self.status.stickers.init()
|
self.status.stickers.init()
|
||||||
self.view.reactions.init()
|
self.view.reactions.init()
|
||||||
|
|
||||||
|
self.view.asyncActivityNotificationLoad()
|
||||||
|
|
||||||
let recentStickers = self.status.stickers.getRecentStickers()
|
let recentStickers = self.status.stickers.getRecentStickers()
|
||||||
for sticker in recentStickers:
|
for sticker in recentStickers:
|
||||||
|
|
|
@ -22,6 +22,9 @@ proc handleChatEvents(self: ChatController) =
|
||||||
self.status.events.on("pinnedMessagesLoaded") do(e:Args):
|
self.status.events.on("pinnedMessagesLoaded") do(e:Args):
|
||||||
self.view.pushPinnedMessages(MsgsLoadedArgs(e).messages)
|
self.view.pushPinnedMessages(MsgsLoadedArgs(e).messages)
|
||||||
|
|
||||||
|
self.status.events.on("activityCenterNotificationsLoaded") do(e:Args):
|
||||||
|
self.view.pushActivityCenterNotifications(ActivityCenterNotificationsArgs(e).activityCenterNotifications)
|
||||||
|
|
||||||
self.status.events.on("contactUpdate") do(e: Args):
|
self.status.events.on("contactUpdate") do(e: Args):
|
||||||
var evArgs = ContactUpdateArgs(e)
|
var evArgs = ContactUpdateArgs(e)
|
||||||
self.view.updateUsernames(evArgs.contacts)
|
self.view.updateUsernames(evArgs.contacts)
|
||||||
|
@ -56,6 +59,8 @@ proc handleChatEvents(self: ChatController) =
|
||||||
self.view.communities.addMembershipRequests(evArgs.communityMembershipRequests)
|
self.view.communities.addMembershipRequests(evArgs.communityMembershipRequests)
|
||||||
if (evArgs.pinnedMessages.len > 0):
|
if (evArgs.pinnedMessages.len > 0):
|
||||||
self.view.addPinnedMessages(evArgs.pinnedMessages)
|
self.view.addPinnedMessages(evArgs.pinnedMessages)
|
||||||
|
if (evArgs.activityCenterNotifications.len > 0):
|
||||||
|
self.view.addActivityCenterNotification(evArgs.activityCenterNotifications)
|
||||||
|
|
||||||
self.status.events.on("channelUpdate") do(e: Args):
|
self.status.events.on("channelUpdate") do(e: Args):
|
||||||
var evArgs = ChatUpdateArgs(e)
|
var evArgs = ChatUpdateArgs(e)
|
||||||
|
|
|
@ -4,7 +4,7 @@ import
|
||||||
proc handleSignals(self: ChatController) =
|
proc handleSignals(self: ChatController) =
|
||||||
self.status.events.on(SignalType.Message.event) do(e:Args):
|
self.status.events.on(SignalType.Message.event) do(e:Args):
|
||||||
var data = MessageSignal(e)
|
var data = MessageSignal(e)
|
||||||
self.status.chat.update(data.chats, data.messages, data.emojiReactions, data.communities, data.membershipRequests, data.pinnedMessages)
|
self.status.chat.update(data.chats, data.messages, data.emojiReactions, data.communities, data.membershipRequests, data.pinnedMessages, data.activityCenterNotification)
|
||||||
|
|
||||||
self.status.events.on(SignalType.DiscoverySummary.event) do(e:Args):
|
self.status.events.on(SignalType.DiscoverySummary.event) do(e:Args):
|
||||||
## Handle mailserver peers being added and removed
|
## Handle mailserver peers being added and removed
|
||||||
|
|
|
@ -13,7 +13,7 @@ import ../../status/ens as status_ens
|
||||||
import ../../status/chat/[chat, message]
|
import ../../status/chat/[chat, message]
|
||||||
import ../../status/profile/profile
|
import ../../status/profile/profile
|
||||||
import web3/[conversions, ethtypes]
|
import web3/[conversions, ethtypes]
|
||||||
import views/[channels_list, message_list, chat_item, suggestions_list, reactions, stickers, groups, transactions, communities, community_list, community_item]
|
import views/[channels_list, message_list, chat_item, suggestions_list, reactions, stickers, groups, transactions, communities, community_list, community_item, activity_notification_list]
|
||||||
import ../utils/image_utils
|
import ../utils/image_utils
|
||||||
import ../../status/tasks/[qt, task_runner_impl]
|
import ../../status/tasks/[qt, task_runner_impl]
|
||||||
import ../../status/tasks/marathon/mailserver/worker
|
import ../../status/tasks/marathon/mailserver/worker
|
||||||
|
@ -29,6 +29,7 @@ type
|
||||||
GetLinkPreviewDataTaskArg = ref object of QObjectTaskArg
|
GetLinkPreviewDataTaskArg = ref object of QObjectTaskArg
|
||||||
link: string
|
link: string
|
||||||
uuid: string
|
uuid: string
|
||||||
|
AsyncActivityNotificationLoadTaskArg = ref object of QObjectTaskArg
|
||||||
AsyncMessageLoadTaskArg = ref object of QObjectTaskArg
|
AsyncMessageLoadTaskArg = ref object of QObjectTaskArg
|
||||||
chatId: string
|
chatId: string
|
||||||
ResolveEnsTaskArg = ref object of QObjectTaskArg
|
ResolveEnsTaskArg = ref object of QObjectTaskArg
|
||||||
|
@ -81,6 +82,19 @@ const asyncMessageLoadTask: Task = proc(argEncoded: string) {.gcsafe, nimcall.}
|
||||||
}
|
}
|
||||||
arg.finish(responseJson)
|
arg.finish(responseJson)
|
||||||
|
|
||||||
|
const asyncActivityNotificationLoadTask: Task = proc(argEncoded: string) {.gcsafe, nimcall.} =
|
||||||
|
let arg = decode[AsyncActivityNotificationLoadTaskArg](argEncoded)
|
||||||
|
var activityNotifications: JsonNode
|
||||||
|
var activityNotificationsCallSuccess: bool
|
||||||
|
let activityNotificationsCallResult = rpcActivityCenterNotifications(newJString(""), 20, activityNotificationsCallSuccess)
|
||||||
|
if(activityNotificationsCallSuccess):
|
||||||
|
activityNotifications = activityNotificationsCallResult.parseJson()["result"]
|
||||||
|
|
||||||
|
let responseJson = %*{
|
||||||
|
"activityNotifications": activityNotifications
|
||||||
|
}
|
||||||
|
arg.finish(responseJson)
|
||||||
|
|
||||||
proc asyncMessageLoad[T](self: T, slot: string, chatId: string) =
|
proc asyncMessageLoad[T](self: T, slot: string, chatId: string) =
|
||||||
let arg = AsyncMessageLoadTaskArg(
|
let arg = AsyncMessageLoadTaskArg(
|
||||||
tptr: cast[ByteAddress](asyncMessageLoadTask),
|
tptr: cast[ByteAddress](asyncMessageLoadTask),
|
||||||
|
@ -90,6 +104,14 @@ proc asyncMessageLoad[T](self: T, slot: string, chatId: string) =
|
||||||
)
|
)
|
||||||
self.status.tasks.threadpool.start(arg)
|
self.status.tasks.threadpool.start(arg)
|
||||||
|
|
||||||
|
proc asyncActivityNotificationLoad[T](self: T, slot: string) =
|
||||||
|
let arg = AsyncActivityNotificationLoadTaskArg(
|
||||||
|
tptr: cast[ByteAddress](asyncActivityNotificationLoadTask),
|
||||||
|
vptr: cast[ByteAddress](self.vptr),
|
||||||
|
slot: slot
|
||||||
|
)
|
||||||
|
self.status.tasks.threadpool.start(arg)
|
||||||
|
|
||||||
const resolveEnsTask: Task = proc(argEncoded: string) {.gcsafe, nimcall.} =
|
const resolveEnsTask: Task = proc(argEncoded: string) {.gcsafe, nimcall.} =
|
||||||
let
|
let
|
||||||
arg = decode[ResolveEnsTaskArg](argEncoded)
|
arg = decode[ResolveEnsTaskArg](argEncoded)
|
||||||
|
@ -111,6 +133,7 @@ QtObject:
|
||||||
status: Status
|
status: Status
|
||||||
chats*: ChannelsList
|
chats*: ChannelsList
|
||||||
currentSuggestions*: SuggestionsList
|
currentSuggestions*: SuggestionsList
|
||||||
|
activityNotificationList*: ActivityNotificationList
|
||||||
callResult: string
|
callResult: string
|
||||||
messageList*: OrderedTable[string, ChatMessageList]
|
messageList*: OrderedTable[string, ChatMessageList]
|
||||||
pinnedMessagesList*: OrderedTable[string, ChatMessageList]
|
pinnedMessagesList*: OrderedTable[string, ChatMessageList]
|
||||||
|
@ -137,6 +160,7 @@ QtObject:
|
||||||
self.activeChannel.delete
|
self.activeChannel.delete
|
||||||
self.contextChannel.delete
|
self.contextChannel.delete
|
||||||
self.currentSuggestions.delete
|
self.currentSuggestions.delete
|
||||||
|
self.activityNotificationList.delete
|
||||||
for msg in self.messageList.values:
|
for msg in self.messageList.values:
|
||||||
msg.delete
|
msg.delete
|
||||||
for msg in self.pinnedMessagesList.values:
|
for msg in self.pinnedMessagesList.values:
|
||||||
|
@ -159,6 +183,7 @@ QtObject:
|
||||||
result.activeChannel = newChatItemView(status)
|
result.activeChannel = newChatItemView(status)
|
||||||
result.contextChannel = newChatItemView(status)
|
result.contextChannel = newChatItemView(status)
|
||||||
result.currentSuggestions = newSuggestionsList()
|
result.currentSuggestions = newSuggestionsList()
|
||||||
|
result.activityNotificationList = newActivityNotificationList(status)
|
||||||
result.messageList = initOrderedTable[string, ChatMessageList]()
|
result.messageList = initOrderedTable[string, ChatMessageList]()
|
||||||
result.pinnedMessagesList = initOrderedTable[string, ChatMessageList]()
|
result.pinnedMessagesList = initOrderedTable[string, ChatMessageList]()
|
||||||
result.reactions = newReactionView(status, result.messageList.addr, result.activeChannel)
|
result.reactions = newReactionView(status, result.messageList.addr, result.activeChannel)
|
||||||
|
@ -438,6 +463,15 @@ QtObject:
|
||||||
QtProperty[QVariant] suggestionList:
|
QtProperty[QVariant] suggestionList:
|
||||||
read = getCurrentSuggestions
|
read = getCurrentSuggestions
|
||||||
|
|
||||||
|
proc activityNotificationsChanged*(self: ChatsView) {.signal.}
|
||||||
|
|
||||||
|
proc getActivityNotificationList(self: ChatsView): QVariant {.slot.} =
|
||||||
|
return newQVariant(self.activityNotificationList)
|
||||||
|
|
||||||
|
QtProperty[QVariant] activityNotificationList:
|
||||||
|
read = getActivityNotificationList
|
||||||
|
notify = activityNotificationsChanged
|
||||||
|
|
||||||
proc upsertChannel(self: ChatsView, channel: string) =
|
proc upsertChannel(self: ChatsView, channel: string) =
|
||||||
var chat: Chat = nil
|
var chat: Chat = nil
|
||||||
if self.status.chat.channels.hasKey(channel):
|
if self.status.chat.channels.hasKey(channel):
|
||||||
|
@ -481,6 +515,15 @@ QtObject:
|
||||||
# put the message as pinned in the message list
|
# put the message as pinned in the message list
|
||||||
self.messageList[msg.chatId].changeMessagePinned(msg.id, true, msg.pinnedBy)
|
self.messageList[msg.chatId].changeMessagePinned(msg.id, true, msg.pinnedBy)
|
||||||
|
|
||||||
|
proc pushActivityCenterNotifications*(self:ChatsView, activityCenterNotifications: seq[ActivityCenterNotification]) =
|
||||||
|
self.activityNotificationList.setNewData(activityCenterNotifications)
|
||||||
|
self.activityNotificationsChanged()
|
||||||
|
|
||||||
|
proc addActivityCenterNotification*(self:ChatsView, activityCenterNotifications: seq[ActivityCenterNotification]) =
|
||||||
|
for activityCenterNotification in activityCenterNotifications:
|
||||||
|
self.activityNotificationList.addActivityNotificationItemToList(activityCenterNotification)
|
||||||
|
self.activityNotificationsChanged()
|
||||||
|
|
||||||
proc pushMessages*(self:ChatsView, messages: var seq[Message]) =
|
proc pushMessages*(self:ChatsView, messages: var seq[Message]) =
|
||||||
for msg in messages.mitems:
|
for msg in messages.mitems:
|
||||||
self.upsertChannel(msg.chatId)
|
self.upsertChannel(msg.chatId)
|
||||||
|
@ -626,6 +669,9 @@ QtObject:
|
||||||
proc asyncMessageLoad*(self: ChatsView, chatId: string) {.slot.} =
|
proc asyncMessageLoad*(self: ChatsView, chatId: string) {.slot.} =
|
||||||
self.asyncMessageLoad("asyncMessageLoaded", chatId)
|
self.asyncMessageLoad("asyncMessageLoaded", chatId)
|
||||||
|
|
||||||
|
proc asyncActivityNotificationLoad*(self: ChatsView) {.slot.} =
|
||||||
|
self.asyncActivityNotificationLoad("asyncActivityNotificationLoaded")
|
||||||
|
|
||||||
proc asyncMessageLoaded*(self: ChatsView, rpcResponse: string) {.slot.} =
|
proc asyncMessageLoaded*(self: ChatsView, rpcResponse: string) {.slot.} =
|
||||||
let
|
let
|
||||||
rpcResponseObj = rpcResponse.parseJson
|
rpcResponseObj = rpcResponse.parseJson
|
||||||
|
@ -636,7 +682,7 @@ QtObject:
|
||||||
|
|
||||||
let messages = rpcResponseObj{"messages"}
|
let messages = rpcResponseObj{"messages"}
|
||||||
if(messages != nil and messages.kind != JNull):
|
if(messages != nil and messages.kind != JNull):
|
||||||
let chatMessages = parseChatMessagesResponse(chatId, messages)
|
let chatMessages = parseChatMessagesResponse(messages)
|
||||||
self.status.chat.chatMessages(chatId, true, chatMessages[0], chatMessages[1])
|
self.status.chat.chatMessages(chatId, true, chatMessages[0], chatMessages[1])
|
||||||
|
|
||||||
let rxns = rpcResponseObj{"reactions"}
|
let rxns = rpcResponseObj{"reactions"}
|
||||||
|
@ -646,9 +692,16 @@ QtObject:
|
||||||
|
|
||||||
let pinnedMsgs = rpcResponseObj{"pinnedMessages"}
|
let pinnedMsgs = rpcResponseObj{"pinnedMessages"}
|
||||||
if(pinnedMsgs != nil and pinnedMsgs.kind != JNull):
|
if(pinnedMsgs != nil and pinnedMsgs.kind != JNull):
|
||||||
let pinnedMessages = parseChatPinnedMessagesResponse(chatId, pinnedMsgs)
|
let pinnedMessages = parseChatPinnedMessagesResponse(pinnedMsgs)
|
||||||
self.status.chat.pinnedMessagesByChatID(chatId, pinnedMessages[0], pinnedMessages[1])
|
self.status.chat.pinnedMessagesByChatID(chatId, pinnedMessages[0], pinnedMessages[1])
|
||||||
|
|
||||||
|
proc asyncActivityNotificationLoaded*(self: ChatsView, rpcResponse: string) {.slot.} =
|
||||||
|
let rpcResponseObj = rpcResponse.parseJson
|
||||||
|
|
||||||
|
if(rpcResponseObj["activityNotifications"].kind != JNull):
|
||||||
|
let activityNotifications = parseActivityCenterNotifications(rpcResponseObj["activityNotifications"])
|
||||||
|
self.status.chat.activityCenterNotifications(activityNotifications[0], activityNotifications[1])
|
||||||
|
|
||||||
proc hideLoadingIndicator*(self: ChatsView) {.slot.} =
|
proc hideLoadingIndicator*(self: ChatsView) {.slot.} =
|
||||||
self.loadingMessages = false
|
self.loadingMessages = false
|
||||||
self.loadingMessagesChanged(false)
|
self.loadingMessagesChanged(false)
|
||||||
|
|
|
@ -0,0 +1,149 @@
|
||||||
|
import NimQml, Tables, chronicles
|
||||||
|
import ../../../status/chat/chat
|
||||||
|
import ../../../status/status
|
||||||
|
import ../../../status/accounts
|
||||||
|
import strutils
|
||||||
|
import message_item
|
||||||
|
|
||||||
|
type ActivityCenterNotificationViewItem* = ref object of ActivityCenterNotification
|
||||||
|
messageItem*: MessageItem
|
||||||
|
|
||||||
|
type
|
||||||
|
NotifRoles {.pure.} = enum
|
||||||
|
Id = UserRole + 1
|
||||||
|
ChatId = UserRole + 2
|
||||||
|
Name = UserRole + 3
|
||||||
|
NotificationType = UserRole + 4
|
||||||
|
Message = UserRole + 5
|
||||||
|
Timestamp = UserRole + 6
|
||||||
|
Read = UserRole + 7
|
||||||
|
Dismissed = UserRole + 8
|
||||||
|
Accepted = UserRole + 9
|
||||||
|
|
||||||
|
QtObject:
|
||||||
|
type
|
||||||
|
ActivityNotificationList* = ref object of QAbstractListModel
|
||||||
|
activityCenterNotifications*: seq[ActivityCenterNotificationViewItem]
|
||||||
|
status: Status
|
||||||
|
nbUnreadNotifications*: int
|
||||||
|
|
||||||
|
proc setup(self: ActivityNotificationList) = self.QAbstractListModel.setup
|
||||||
|
|
||||||
|
proc delete(self: ActivityNotificationList) =
|
||||||
|
self.activityCenterNotifications = @[]
|
||||||
|
self.QAbstractListModel.delete
|
||||||
|
|
||||||
|
proc newActivityNotificationList*(status: Status): ActivityNotificationList =
|
||||||
|
new(result, delete)
|
||||||
|
result.activityCenterNotifications = @[]
|
||||||
|
result.status = status
|
||||||
|
result.setup()
|
||||||
|
|
||||||
|
proc unreadCountChanged*(self: ActivityNotificationList) {.signal.}
|
||||||
|
|
||||||
|
proc unreadCount*(self: ActivityNotificationList): int {.slot.} =
|
||||||
|
self.nbUnreadNotifications
|
||||||
|
|
||||||
|
QtProperty[int] unreadCount:
|
||||||
|
read = unreadCount
|
||||||
|
notify = unreadCountChanged
|
||||||
|
|
||||||
|
method rowCount*(self: ActivityNotificationList, index: QModelIndex = nil): int = self.activityCenterNotifications.len
|
||||||
|
|
||||||
|
method data(self: ActivityNotificationList, index: QModelIndex, role: int): QVariant =
|
||||||
|
if not index.isValid:
|
||||||
|
return
|
||||||
|
if index.row < 0 or index.row >= self.activityCenterNotifications.len:
|
||||||
|
return
|
||||||
|
|
||||||
|
let acitivityNotificationItem = self.activityCenterNotifications[index.row]
|
||||||
|
let communityItemRole = role.NotifRoles
|
||||||
|
case communityItemRole:
|
||||||
|
of NotifRoles.Id: result = newQVariant(acitivityNotificationItem.id)
|
||||||
|
of NotifRoles.ChatId: result = newQVariant(acitivityNotificationItem.chatId)
|
||||||
|
of NotifRoles.Name: result = newQVariant(acitivityNotificationItem.name)
|
||||||
|
of NotifRoles.NotificationType: result = newQVariant(acitivityNotificationItem.notificationType.int)
|
||||||
|
of NotifRoles.Message: result = newQVariant(acitivityNotificationItem.messageItem)
|
||||||
|
of NotifRoles.Timestamp: result = newQVariant(acitivityNotificationItem.timestamp)
|
||||||
|
of NotifRoles.Read: result = newQVariant(acitivityNotificationItem.read.bool)
|
||||||
|
of NotifRoles.Dismissed: result = newQVariant(acitivityNotificationItem.dismissed.bool)
|
||||||
|
of NotifRoles.Accepted: result = newQVariant(acitivityNotificationItem.accepted.bool)
|
||||||
|
|
||||||
|
proc getNotificationData(self: ActivityNotificationList, index: int, data: string): string {.slot.} =
|
||||||
|
if index < 0 or index >= self.activityCenterNotifications.len: return ("")
|
||||||
|
|
||||||
|
let notif = self.activityCenterNotifications[index]
|
||||||
|
case data:
|
||||||
|
of "id": result = notif.id
|
||||||
|
of "chatId": result = notif.chatId
|
||||||
|
of "name": result = notif.name
|
||||||
|
of "notificationType": result = $(notif.notificationType.int)
|
||||||
|
of "timestamp": result = $(notif.timestamp)
|
||||||
|
of "read": result = $(notif.read)
|
||||||
|
of "dismissed": result = $(notif.dismissed)
|
||||||
|
of "accepted": result = $(notif.accepted)
|
||||||
|
else: result = ("")
|
||||||
|
|
||||||
|
method roleNames(self: ActivityNotificationList): Table[int, string] =
|
||||||
|
{
|
||||||
|
NotifRoles.Id.int:"id",
|
||||||
|
NotifRoles.ChatId.int:"chatId",
|
||||||
|
NotifRoles.Name.int: "name",
|
||||||
|
NotifRoles.NotificationType.int: "notificationType",
|
||||||
|
NotifRoles.Message.int: "message",
|
||||||
|
NotifRoles.Timestamp.int: "timestamp",
|
||||||
|
NotifRoles.Read.int: "read",
|
||||||
|
NotifRoles.Dismissed.int: "dismissed",
|
||||||
|
NotifRoles.Accepted.int: "accepted"
|
||||||
|
}.toTable
|
||||||
|
|
||||||
|
proc markAllActivityCenterNotificationsRead(self: ActivityNotificationList): string {.slot.} =
|
||||||
|
let error = self.status.chat.markAllActivityCenterNotificationsRead()
|
||||||
|
if (error != ""):
|
||||||
|
return error
|
||||||
|
|
||||||
|
self.nbUnreadNotifications = 0
|
||||||
|
self.unreadCountChanged()
|
||||||
|
|
||||||
|
for activityCenterNotification in self.activityCenterNotifications:
|
||||||
|
activityCenterNotification.read = true
|
||||||
|
|
||||||
|
let topLeft = self.createIndex(0, 0, nil)
|
||||||
|
let bottomRight = self.createIndex(self.activityCenterNotifications.len - 1, 0, nil)
|
||||||
|
self.dataChanged(topLeft, bottomRight, @[NotifRoles.Read.int])
|
||||||
|
|
||||||
|
|
||||||
|
proc toActivityCenterNotificationViewItem*(self: ActivityNotificationList, activityCenterNotification: ActivityCenterNotification): ActivityCenterNotificationViewItem =
|
||||||
|
ActivityCenterNotificationViewItem(
|
||||||
|
id: activityCenterNotification.id,
|
||||||
|
chatId: activityCenterNotification.chatId,
|
||||||
|
name: activityCenterNotification.name,
|
||||||
|
notificationType: activityCenterNotification.notificationType,
|
||||||
|
timestamp: activityCenterNotification.timestamp,
|
||||||
|
read: activityCenterNotification.read,
|
||||||
|
dismissed: activityCenterNotification.dismissed,
|
||||||
|
accepted: activityCenterNotification.accepted,
|
||||||
|
messageItem: newMessageItem(self.status, activityCenterNotification.message)
|
||||||
|
)
|
||||||
|
|
||||||
|
proc setNewData*(self: ActivityNotificationList, activityCenterNotifications: seq[ActivityCenterNotification]) =
|
||||||
|
self.beginResetModel()
|
||||||
|
self.activityCenterNotifications = @[]
|
||||||
|
|
||||||
|
for activityCenterNotification in activityCenterNotifications:
|
||||||
|
self.activityCenterNotifications.add(self.toActivityCenterNotificationViewItem(activityCenterNotification))
|
||||||
|
|
||||||
|
self.endResetModel()
|
||||||
|
|
||||||
|
self.nbUnreadNotifications = self.status.chat.unreadActivityCenterNotificationsCount()
|
||||||
|
self.unreadCountChanged()
|
||||||
|
|
||||||
|
proc addActivityNotificationItemToList*(self: ActivityNotificationList, activityCenterNotification: ActivityCenterNotification) =
|
||||||
|
self.beginInsertRows(newQModelIndex(), self.activityCenterNotifications.len, self.activityCenterNotifications.len)
|
||||||
|
|
||||||
|
self.activityCenterNotifications.add(self.toActivityCenterNotificationViewItem(activityCenterNotification))
|
||||||
|
|
||||||
|
self.endInsertRows()
|
||||||
|
|
||||||
|
if (not activityCenterNotification.read):
|
||||||
|
self.nbUnreadNotifications = self.nbUnreadNotifications + 1
|
|
@ -146,6 +146,11 @@ QtObject:
|
||||||
if (channel == nil): return
|
if (channel == nil): return
|
||||||
return channel.color
|
return channel.color
|
||||||
|
|
||||||
|
proc getChannelType*(self: ChannelsList, id: string): int {.slot.} =
|
||||||
|
let channel = self.getChannelById(id)
|
||||||
|
if (channel == nil): return ChatType.Unknown.int
|
||||||
|
return channel.chatType.int
|
||||||
|
|
||||||
proc updateChat*(self: ChannelsList, channel: Chat) =
|
proc updateChat*(self: ChannelsList, channel: Chat) =
|
||||||
let idx = self.upsertChannel(channel)
|
let idx = self.upsertChannel(channel)
|
||||||
if idx == -1: return
|
if idx == -1: return
|
||||||
|
|
|
@ -1,22 +1,29 @@
|
||||||
import sequtils, re, strutils
|
import NimQml, Tables, sets, json, sugar, re
|
||||||
|
import ../../../status/status
|
||||||
|
import ../../../status/accounts
|
||||||
|
import ../../../status/chat
|
||||||
|
import ../../../status/chat/[message,stickers]
|
||||||
|
import ../../../status/profile/profile
|
||||||
|
import ../../../status/ens
|
||||||
|
import strformat, strutils, sequtils
|
||||||
|
|
||||||
let NEW_LINE = re"\n|\r"
|
let NEW_LINE = re"\n|\r"
|
||||||
|
|
||||||
proc sectionIdentifier(message: Message): string =
|
proc sectionIdentifier*(message: Message): string =
|
||||||
result = message.fromAuthor
|
result = message.fromAuthor
|
||||||
# Force section change, because group status messages are sent with the
|
# Force section change, because group status messages are sent with the
|
||||||
# same fromAuthor, and ends up causing the header to not be shown
|
# same fromAuthor, and ends up causing the header to not be shown
|
||||||
if message.contentType == ContentType.Group:
|
if message.contentType == ContentType.Group:
|
||||||
result = "GroupChatMessage"
|
result = "GroupChatMessage"
|
||||||
|
|
||||||
proc mention(self: ChatMessageList, pubKey: string): string =
|
proc mention*(pubKey: string, contacts: Table[string, Profile]): string =
|
||||||
if self.status.chat.contacts.hasKey(pubKey):
|
if contacts.hasKey(pubKey):
|
||||||
return ens.userNameOrAlias(self.status.chat.contacts[pubKey], true)
|
return ens.userNameOrAlias(contacts[pubKey], true)
|
||||||
generateAlias(pubKey)
|
generateAlias(pubKey)
|
||||||
|
|
||||||
|
|
||||||
# See render-inline in status-react/src/status_im/ui/screens/chat/message/message.cljs
|
# See render-inline in status-react/src/status_im/ui/screens/chat/message/message.cljs
|
||||||
proc renderInline(self: ChatMessageList, elem: TextItem): string =
|
proc renderInline*(elem: TextItem, contacts: Table[string, Profile]): string =
|
||||||
let value = escape_html(elem.literal).multiReplace(("\r\n", "<br/>")).multiReplace(("\n", "<br/>")).multiReplace((" ", " "))
|
let value = escape_html(elem.literal).multiReplace(("\r\n", "<br/>")).multiReplace(("\n", "<br/>")).multiReplace((" ", " "))
|
||||||
case elem.textType:
|
case elem.textType:
|
||||||
of "": result = value
|
of "": result = value
|
||||||
|
@ -25,19 +32,19 @@ proc renderInline(self: ChatMessageList, elem: TextItem): string =
|
||||||
of "strong": result = fmt("<strong>{value}</strong>")
|
of "strong": result = fmt("<strong>{value}</strong>")
|
||||||
of "strong-emph": result = fmt(" <strong><em>{value}</em></strong> ")
|
of "strong-emph": result = fmt(" <strong><em>{value}</em></strong> ")
|
||||||
of "link": result = fmt("{elem.destination}")
|
of "link": result = fmt("{elem.destination}")
|
||||||
of "mention": result = fmt("<a href=\"//{value}\" class=\"mention\">{self.mention(value)}</a>")
|
of "mention": result = fmt("<a href=\"//{value}\" class=\"mention\">{mention(value, contacts)}</a>")
|
||||||
of "status-tag": result = fmt("<a href=\"#{value}\" class=\"status-tag\">#{value}</a>")
|
of "status-tag": result = fmt("<a href=\"#{value}\" class=\"status-tag\">#{value}</a>")
|
||||||
of "del": result = fmt("<del>{value}</del>")
|
of "del": result = fmt("<del>{value}</del>")
|
||||||
else: result = fmt(" {value} ")
|
else: result = fmt(" {value} ")
|
||||||
|
|
||||||
# See render-block in status-react/src/status_im/ui/screens/chat/message/message.cljs
|
# See render-block in status-react/src/status_im/ui/screens/chat/message/message.cljs
|
||||||
proc renderBlock(self: ChatMessageList, message: Message): string =
|
proc renderBlock*(message: Message, contacts: Table[string, Profile]): string =
|
||||||
for pMsg in message.parsedText:
|
for pMsg in message.parsedText:
|
||||||
case pMsg.textType:
|
case pMsg.textType:
|
||||||
of "paragraph":
|
of "paragraph":
|
||||||
result = result & "<p>"
|
result = result & "<p>"
|
||||||
for children in pMsg.children:
|
for children in pMsg.children:
|
||||||
result = result & self.renderInline(children)
|
result = result & renderInline(children, contacts)
|
||||||
result = result & "</p>"
|
result = result & "</p>"
|
||||||
of "blockquote":
|
of "blockquote":
|
||||||
var
|
var
|
||||||
|
|
|
@ -0,0 +1,174 @@
|
||||||
|
import NimQml, std/wrapnils, chronicles
|
||||||
|
import ../../../status/status
|
||||||
|
import ../../../status/chat/message
|
||||||
|
import ../../../status/chat/stickers
|
||||||
|
import message_format
|
||||||
|
|
||||||
|
QtObject:
|
||||||
|
type MessageItem* = ref object of QObject
|
||||||
|
messageItem*: Message
|
||||||
|
status*: Status
|
||||||
|
|
||||||
|
proc setup(self: MessageItem) =
|
||||||
|
self.QObject.setup
|
||||||
|
|
||||||
|
proc delete*(self: MessageItem) =
|
||||||
|
self.QObject.delete
|
||||||
|
|
||||||
|
proc newMessageItem*(status: Status, message: Message): MessageItem =
|
||||||
|
new(result, delete)
|
||||||
|
result.messageItem = message
|
||||||
|
result.status = status
|
||||||
|
result.setup
|
||||||
|
|
||||||
|
proc setMessageItem*(self: MessageItem, messageItem: Message) =
|
||||||
|
self.messageItem = messageItem
|
||||||
|
|
||||||
|
proc alias*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.alias
|
||||||
|
QtProperty[string] alias:
|
||||||
|
read = alias
|
||||||
|
|
||||||
|
proc userName*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.userName
|
||||||
|
QtProperty[string] userName:
|
||||||
|
read = userName
|
||||||
|
|
||||||
|
proc message*(self: MessageItem): string {.slot.} = result = renderBlock(self.messageItem, self.status.chat.contacts)
|
||||||
|
QtProperty[string] message:
|
||||||
|
read = message
|
||||||
|
|
||||||
|
proc localName*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.localName
|
||||||
|
QtProperty[string] localName:
|
||||||
|
read = localName
|
||||||
|
|
||||||
|
proc chatId*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.chatId
|
||||||
|
QtProperty[string] chatId:
|
||||||
|
read = chatId
|
||||||
|
|
||||||
|
proc clock*(self: MessageItem): int {.slot.} = result = ?.self.messageItem.clock
|
||||||
|
QtProperty[int] clock:
|
||||||
|
read = clock
|
||||||
|
|
||||||
|
proc gapFrom*(self: MessageItem): int {.slot.} = result = ?.self.messageItem.gapFrom
|
||||||
|
QtProperty[int] gapFrom:
|
||||||
|
read = gapFrom
|
||||||
|
|
||||||
|
proc gapTo*(self: MessageItem): int {.slot.} = result = ?.self.messageItem.gapTo
|
||||||
|
QtProperty[int] gapTo:
|
||||||
|
read = gapTo
|
||||||
|
|
||||||
|
proc contentType*(self: MessageItem): int {.slot.} = result = self.messageItem.contentType.int
|
||||||
|
QtProperty[int] contentType:
|
||||||
|
read = contentType
|
||||||
|
|
||||||
|
proc ensName*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.ensName
|
||||||
|
QtProperty[string] ensName:
|
||||||
|
read = ensName
|
||||||
|
|
||||||
|
proc fromAuthor*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.fromAuthor
|
||||||
|
QtProperty[string] fromAuthor:
|
||||||
|
read = fromAuthor
|
||||||
|
|
||||||
|
proc messageId*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.id
|
||||||
|
QtProperty[string] messageId:
|
||||||
|
read = messageId
|
||||||
|
|
||||||
|
proc identicon*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.identicon
|
||||||
|
QtProperty[string] identicon:
|
||||||
|
read = identicon
|
||||||
|
|
||||||
|
proc lineCount*(self: MessageItem): int {.slot.} = result = ?.self.messageItem.lineCount
|
||||||
|
QtProperty[int] lineCount:
|
||||||
|
read = lineCount
|
||||||
|
|
||||||
|
proc localChatId*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.localChatId
|
||||||
|
QtProperty[string] localChatId:
|
||||||
|
read = localChatId
|
||||||
|
|
||||||
|
proc messageType*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.messageType
|
||||||
|
QtProperty[string] messageType:
|
||||||
|
read = messageType
|
||||||
|
|
||||||
|
proc replace*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.replace
|
||||||
|
QtProperty[string] replace:
|
||||||
|
read = replace
|
||||||
|
|
||||||
|
proc responseTo*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.responseTo
|
||||||
|
QtProperty[string] responseTo:
|
||||||
|
read = responseTo
|
||||||
|
|
||||||
|
proc rtl*(self: MessageItem): bool {.slot.} = result = ?.self.messageItem.rtl
|
||||||
|
QtProperty[bool] rtl:
|
||||||
|
read = rtl
|
||||||
|
|
||||||
|
proc seen*(self: MessageItem): bool {.slot.} = result = ?.self.messageItem.seen
|
||||||
|
QtProperty[bool] seen:
|
||||||
|
read = seen
|
||||||
|
|
||||||
|
proc sticker*(self: MessageItem): string {.slot.} = result = self.messageItem.stickerHash.decodeContentHash()
|
||||||
|
QtProperty[string] sticker:
|
||||||
|
read = sticker
|
||||||
|
|
||||||
|
proc sectionIdentifier*(self: MessageItem): string {.slot.} = result = sectionIdentifier(self.messageItem)
|
||||||
|
QtProperty[string] sectionIdentifier:
|
||||||
|
read = sectionIdentifier
|
||||||
|
|
||||||
|
proc stickerPackId*(self: MessageItem): int {.slot.} = result = ?.self.messageItem.stickerPackId
|
||||||
|
QtProperty[int] stickerPackId:
|
||||||
|
read = stickerPackId
|
||||||
|
|
||||||
|
proc plainText*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.text
|
||||||
|
QtProperty[string] plainText:
|
||||||
|
read = plainText
|
||||||
|
|
||||||
|
proc timestamp*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.timestamp
|
||||||
|
QtProperty[string] timestamp:
|
||||||
|
read = timestamp
|
||||||
|
|
||||||
|
proc whisperTimestamp*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.whisperTimestamp
|
||||||
|
QtProperty[string] whisperTimestamp:
|
||||||
|
read = whisperTimestamp
|
||||||
|
|
||||||
|
proc isCurrentUser*(self: MessageItem): bool {.slot.} = result = ?.self.messageItem.isCurrentUser
|
||||||
|
QtProperty[bool] isCurrentUser:
|
||||||
|
read = isCurrentUser
|
||||||
|
|
||||||
|
proc stickerHash*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.stickerHash
|
||||||
|
QtProperty[string] stickerHash:
|
||||||
|
read = stickerHash
|
||||||
|
|
||||||
|
proc outgoingStatus*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.outgoingStatus
|
||||||
|
QtProperty[string] outgoingStatus:
|
||||||
|
read = outgoingStatus
|
||||||
|
|
||||||
|
proc linkUrls*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.linkUrls
|
||||||
|
QtProperty[string] linkUrls:
|
||||||
|
read = linkUrls
|
||||||
|
|
||||||
|
proc image*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.image
|
||||||
|
QtProperty[string] image:
|
||||||
|
read = image
|
||||||
|
|
||||||
|
|
||||||
|
proc audio*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.audio
|
||||||
|
QtProperty[string] audio:
|
||||||
|
read = audio
|
||||||
|
|
||||||
|
proc communityId*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.communityId
|
||||||
|
QtProperty[string] communityId:
|
||||||
|
read = communityId
|
||||||
|
|
||||||
|
proc audioDurationMs*(self: MessageItem): int {.slot.} = result = ?.self.messageItem.audioDurationMs
|
||||||
|
QtProperty[int] audioDurationMs:
|
||||||
|
read = audioDurationMs
|
||||||
|
|
||||||
|
proc hasMention*(self: MessageItem): bool {.slot.} = result = ?.self.messageItem.hasMention
|
||||||
|
QtProperty[bool] hasMention:
|
||||||
|
read = hasMention
|
||||||
|
|
||||||
|
proc isPinned*(self: MessageItem): bool {.slot.} = result = ?.self.messageItem.isPinned
|
||||||
|
QtProperty[bool] isPinned:
|
||||||
|
read = isPinned
|
||||||
|
|
||||||
|
proc pinnedBy*(self: MessageItem): string {.slot.} = result = ?.self.messageItem.pinnedBy
|
||||||
|
QtProperty[string] pinnedBy:
|
||||||
|
read = pinnedBy
|
|
@ -1,4 +1,4 @@
|
||||||
import NimQml, Tables, sets, json, sugar, chronicles
|
import NimQml, Tables, sets, json, sugar, chronicles, sequtils
|
||||||
import ../../../status/status
|
import ../../../status/status
|
||||||
import ../../../status/accounts
|
import ../../../status/accounts
|
||||||
import ../../../status/chat
|
import ../../../status/chat
|
||||||
|
@ -6,6 +6,7 @@ import ../../../status/chat/[message,stickers]
|
||||||
import ../../../status/profile/profile
|
import ../../../status/profile/profile
|
||||||
import ../../../status/ens
|
import ../../../status/ens
|
||||||
import strformat, strutils
|
import strformat, strutils
|
||||||
|
import message_format
|
||||||
|
|
||||||
type
|
type
|
||||||
ChatMessageRoles {.pure.} = enum
|
ChatMessageRoles {.pure.} = enum
|
||||||
|
@ -61,8 +62,6 @@ QtObject:
|
||||||
proc setup(self: ChatMessageList) =
|
proc setup(self: ChatMessageList) =
|
||||||
self.QAbstractListModel.setup
|
self.QAbstractListModel.setup
|
||||||
|
|
||||||
include message_format
|
|
||||||
|
|
||||||
proc fetchMoreMessagesButton(self: ChatMessageList): Message =
|
proc fetchMoreMessagesButton(self: ChatMessageList): Message =
|
||||||
result = Message()
|
result = Message()
|
||||||
result.contentType = ContentType.FetchMoreMessagesButton;
|
result.contentType = ContentType.FetchMoreMessagesButton;
|
||||||
|
@ -148,7 +147,7 @@ QtObject:
|
||||||
let chatMessageRole = role.ChatMessageRoles
|
let chatMessageRole = role.ChatMessageRoles
|
||||||
case chatMessageRole:
|
case chatMessageRole:
|
||||||
of ChatMessageRoles.UserName: result = newQVariant(message.userName)
|
of ChatMessageRoles.UserName: result = newQVariant(message.userName)
|
||||||
of ChatMessageRoles.Message: result = newQVariant(self.renderBlock(message))
|
of ChatMessageRoles.Message: result = newQVariant(renderBlock(message, self.status.chat.contacts))
|
||||||
of ChatMessageRoles.PlainText: result = newQVariant(message.text)
|
of ChatMessageRoles.PlainText: result = newQVariant(message.text)
|
||||||
of ChatMessageRoles.Timestamp: result = newQVariant(message.timestamp)
|
of ChatMessageRoles.Timestamp: result = newQVariant(message.timestamp)
|
||||||
of ChatMessageRoles.Clock: result = newQVariant($message.clock)
|
of ChatMessageRoles.Clock: result = newQVariant($message.clock)
|
||||||
|
@ -236,13 +235,13 @@ QtObject:
|
||||||
|
|
||||||
let message = self.messages[index]
|
let message = self.messages[index]
|
||||||
case data:
|
case data:
|
||||||
of "userName": result = (message.userName)
|
of "userName": result = message.userName
|
||||||
of "publicKey": result = (message.fromAuthor)
|
of "publicKey": result = message.fromAuthor
|
||||||
of "alias": result = (message.alias)
|
of "alias": result = message.alias
|
||||||
of "localName": result = (message.localName)
|
of "localName": result = message.localName
|
||||||
of "ensName": result = (message.ensName)
|
of "ensName": result = message.ensName
|
||||||
of "message": result = (self.renderBlock(message))
|
of "message": result = (renderBlock(message, self.status.chat.contacts))
|
||||||
of "identicon": result = (message.identicon)
|
of "identicon": result = message.identicon
|
||||||
of "timestamp": result = $(message.timestamp)
|
of "timestamp": result = $(message.timestamp)
|
||||||
of "image": result = $(message.image)
|
of "image": result = $(message.image)
|
||||||
of "contentType": result = $(message.contentType.int)
|
of "contentType": result = $(message.contentType.int)
|
||||||
|
|
|
@ -29,6 +29,7 @@ type
|
||||||
emojiReactions*: seq[Reaction]
|
emojiReactions*: seq[Reaction]
|
||||||
communities*: seq[Community]
|
communities*: seq[Community]
|
||||||
communityMembershipRequests*: seq[CommunityMembershipRequest]
|
communityMembershipRequests*: seq[CommunityMembershipRequest]
|
||||||
|
activityCenterNotifications*: seq[ActivityCenterNotification]
|
||||||
|
|
||||||
ChatIdArg* = ref object of Args
|
ChatIdArg* = ref object of Args
|
||||||
chatId*: string
|
chatId*: string
|
||||||
|
@ -42,10 +43,12 @@ type
|
||||||
CommunityActiveChangedArgs* = ref object of Args
|
CommunityActiveChangedArgs* = ref object of Args
|
||||||
active*: bool
|
active*: bool
|
||||||
|
|
||||||
|
|
||||||
MsgsLoadedArgs* = ref object of Args
|
MsgsLoadedArgs* = ref object of Args
|
||||||
messages*: seq[Message]
|
messages*: seq[Message]
|
||||||
|
|
||||||
|
ActivityCenterNotificationsArgs* = ref object of Args
|
||||||
|
activityCenterNotifications*: seq[ActivityCenterNotification]
|
||||||
|
|
||||||
ReactionsLoadedArgs* = ref object of Args
|
ReactionsLoadedArgs* = ref object of Args
|
||||||
reactions*: seq[Reaction]
|
reactions*: seq[Reaction]
|
||||||
|
|
||||||
|
@ -105,7 +108,7 @@ proc cleanSpamChatGroups(self: ChatModel, chats: seq[Chat], contacts: seq[Profil
|
||||||
else:
|
else:
|
||||||
result.add(chat)
|
result.add(chat)
|
||||||
|
|
||||||
proc update*(self: ChatModel, chats: seq[Chat], messages: seq[Message], emojiReactions: seq[Reaction], communities: seq[Community], communityMembershipRequests: seq[CommunityMembershipRequest], pinnedMessages: seq[Message]) =
|
proc update*(self: ChatModel, chats: seq[Chat], messages: seq[Message], emojiReactions: seq[Reaction], communities: seq[Community], communityMembershipRequests: seq[CommunityMembershipRequest], pinnedMessages: seq[Message], activityCenterNotifications: seq[ActivityCenterNotification]) =
|
||||||
var contacts = getAddedContacts()
|
var contacts = getAddedContacts()
|
||||||
|
|
||||||
var chatList = chats
|
var chatList = chats
|
||||||
|
@ -126,7 +129,7 @@ proc update*(self: ChatModel, chats: seq[Chat], messages: seq[Message], emojiRea
|
||||||
if self.lastMessageTimestamps[chatId] > ts:
|
if self.lastMessageTimestamps[chatId] > ts:
|
||||||
self.lastMessageTimestamps[chatId] = ts
|
self.lastMessageTimestamps[chatId] = ts
|
||||||
|
|
||||||
self.events.emit("chatUpdate", ChatUpdateArgs(messages: messages, chats: chatList, contacts: @[], emojiReactions: emojiReactions, communities: communities, communityMembershipRequests: communityMembershipRequests, pinnedMessages: pinnedMessages))
|
self.events.emit("chatUpdate", ChatUpdateArgs(messages: messages,chats: chatList, contacts: @[], emojiReactions: emojiReactions, communities: communities, communityMembershipRequests: communityMembershipRequests, pinnedMessages: pinnedMessages, activityCenterNotifications: activityCenterNotifications))
|
||||||
|
|
||||||
proc hasChannel*(self: ChatModel, chatId: string): bool =
|
proc hasChannel*(self: ChatModel, chatId: string): bool =
|
||||||
self.channels.hasKey(chatId)
|
self.channels.hasKey(chatId)
|
||||||
|
@ -177,21 +180,6 @@ proc requestMissingCommunityInfos*(self: ChatModel) =
|
||||||
for communityId in self.communitiesToFetch:
|
for communityId in self.communitiesToFetch:
|
||||||
status_chat.requestCommunityInfo(communityId)
|
status_chat.requestCommunityInfo(communityId)
|
||||||
|
|
||||||
proc activityCenterNotification*(self: ChatModel, initialLoad:bool = true) =
|
|
||||||
# Notifications were already loaded, since cursor will
|
|
||||||
# be nil/empty if there are no more notifs
|
|
||||||
if(not initialLoad and self.activityCenterCursor == ""): return
|
|
||||||
|
|
||||||
status_chat.activityCenterNotification(self.activityCenterCursor)
|
|
||||||
# self.activityCenterCursor[chatId] = messageTuple[0];
|
|
||||||
|
|
||||||
# if messageTuple[1].len > 0:
|
|
||||||
# let lastMsgIndex = messageTuple[1].len - 1
|
|
||||||
# let ts = times.convert(Milliseconds, Seconds, messageTuple[1][lastMsgIndex].whisperTimestamp.parseInt())
|
|
||||||
# self.lastMessageTimestamps[chatId] = ts
|
|
||||||
|
|
||||||
# self.events.emit("messagesLoaded", MsgsLoadedArgs(messages: messageTuple[1]))
|
|
||||||
|
|
||||||
proc init*(self: ChatModel, pubKey: string, messagesFromContactsOnly: bool) =
|
proc init*(self: ChatModel, pubKey: string, messagesFromContactsOnly: bool) =
|
||||||
self.publicKey = pubKey
|
self.publicKey = pubKey
|
||||||
self.messagesFromContactsOnly = messagesFromContactsOnly
|
self.messagesFromContactsOnly = messagesFromContactsOnly
|
||||||
|
@ -202,8 +190,6 @@ proc init*(self: ChatModel, pubKey: string, messagesFromContactsOnly: bool) =
|
||||||
if (messagesFromContactsOnly):
|
if (messagesFromContactsOnly):
|
||||||
chatList = self.cleanSpamChatGroups(chatList, contacts)
|
chatList = self.cleanSpamChatGroups(chatList, contacts)
|
||||||
|
|
||||||
# self.activityCenterNotification()
|
|
||||||
|
|
||||||
let profileUpdatesChatIds = chatList.filter(c => c.chatType == ChatType.Profile).map(c => c.id)
|
let profileUpdatesChatIds = chatList.filter(c => c.chatType == ChatType.Profile).map(c => c.id)
|
||||||
|
|
||||||
if chatList.filter(c => c.chatType == ChatType.Timeline).len == 0:
|
if chatList.filter(c => c.chatType == ChatType.Timeline).len == 0:
|
||||||
|
@ -555,3 +541,31 @@ proc pinnedMessagesByChatID*(self: ChatModel, chatId: string, cursor: string = "
|
||||||
self.msgCursor[chatId] = cursor
|
self.msgCursor[chatId] = cursor
|
||||||
|
|
||||||
self.events.emit("pinnedMessagesLoaded", MsgsLoadedArgs(messages: pinnedMessages))
|
self.events.emit("pinnedMessagesLoaded", MsgsLoadedArgs(messages: pinnedMessages))
|
||||||
|
|
||||||
|
proc activityCenterNotifications*(self: ChatModel, initialLoad: bool = true) =
|
||||||
|
# Notifications were already loaded, since cursor will
|
||||||
|
# be nil/empty if there are no more notifs
|
||||||
|
if(not initialLoad and self.activityCenterCursor == ""): return
|
||||||
|
|
||||||
|
let activityCenterNotificationsTuple = status_chat.activityCenterNotification(self.activityCenterCursor)
|
||||||
|
self.activityCenterCursor = activityCenterNotificationsTuple[0];
|
||||||
|
|
||||||
|
self.events.emit("activityCenterNotificationsLoaded", ActivityCenterNotificationsArgs(activityCenterNotifications: activityCenterNotificationsTuple[1]))
|
||||||
|
|
||||||
|
|
||||||
|
proc activityCenterNotifications*(self: ChatModel, cursor: string = "", activityCenterNotifications: seq[ActivityCenterNotification]) =
|
||||||
|
self.activityCenterCursor = cursor
|
||||||
|
|
||||||
|
self.events.emit("activityCenterNotificationsLoaded", ActivityCenterNotificationsArgs(activityCenterNotifications: activityCenterNotifications))
|
||||||
|
|
||||||
|
proc markAllActivityCenterNotificationsRead*(self: ChatModel): string =
|
||||||
|
try:
|
||||||
|
status_chat.markAllActivityCenterNotificationsRead()
|
||||||
|
except Exception as e:
|
||||||
|
error "Error marking all as read", msg = e.msg
|
||||||
|
result = e.msg
|
||||||
|
|
||||||
|
|
||||||
|
proc unreadActivityCenterNotificationsCount*(self: ChatModel): int =
|
||||||
|
status_chat.unreadActivityCenterNotificationsCount()
|
||||||
|
|
|
@ -11,6 +11,12 @@ type ChatType* {.pure.}= enum
|
||||||
Timeline = 5
|
Timeline = 5
|
||||||
CommunityChat = 6
|
CommunityChat = 6
|
||||||
|
|
||||||
|
type ActivityCenterNotificationType* {.pure.}= enum
|
||||||
|
Unknown = 0,
|
||||||
|
NewOneToOne = 1,
|
||||||
|
NewPrivateGroupChat = 2,
|
||||||
|
Mention = 3
|
||||||
|
|
||||||
proc isOneToOne*(self: ChatType): bool = self == ChatType.OneToOne
|
proc isOneToOne*(self: ChatType): bool = self == ChatType.OneToOne
|
||||||
proc isTimeline*(self: ChatType): bool = self == ChatType.Timeline
|
proc isTimeline*(self: ChatType): bool = self == ChatType.Timeline
|
||||||
|
|
||||||
|
@ -122,6 +128,17 @@ type Community* = object
|
||||||
membershipRequests*: seq[CommunityMembershipRequest]
|
membershipRequests*: seq[CommunityMembershipRequest]
|
||||||
communityColor*: string
|
communityColor*: string
|
||||||
|
|
||||||
|
type ActivityCenterNotification* = ref object of RootObj
|
||||||
|
id*: string # ID is the id of the chat, for public chats it is the name e.g. status, for one-to-one is the hex encoded public key and for group chats is a random uuid appended with the hex encoded pk of the creator of the chat
|
||||||
|
chatId*: string
|
||||||
|
name*: string
|
||||||
|
notificationType*: ActivityCenterNotificationType
|
||||||
|
message*: Message
|
||||||
|
timestamp*: int64
|
||||||
|
read*: bool
|
||||||
|
dismissed*: bool
|
||||||
|
accepted*: bool
|
||||||
|
|
||||||
proc `$`*(self: Chat): string =
|
proc `$`*(self: Chat): string =
|
||||||
result = fmt"Chat(id:{self.id}, name:{self.name}, active:{self.isActive}, type:{self.chatType})"
|
result = fmt"Chat(id:{self.id}, name:{self.name}, active:{self.isActive}, type:{self.chatType})"
|
||||||
|
|
||||||
|
|
|
@ -73,7 +73,7 @@ proc loadChats*(): seq[Chat] =
|
||||||
result.add(chat)
|
result.add(chat)
|
||||||
result.sort(sortChats)
|
result.sort(sortChats)
|
||||||
|
|
||||||
proc parseChatMessagesResponse*(chatId: string, rpcResult: JsonNode): (string, seq[Message]) =
|
proc parseChatMessagesResponse*(rpcResult: JsonNode): (string, seq[Message]) =
|
||||||
var messages: seq[Message] = @[]
|
var messages: seq[Message] = @[]
|
||||||
let messagesObj = rpcResult{"messages"}
|
let messagesObj = rpcResult{"messages"}
|
||||||
if(messagesObj != nil and messagesObj.kind != JNull):
|
if(messagesObj != nil and messagesObj.kind != JNull):
|
||||||
|
@ -82,6 +82,15 @@ proc parseChatMessagesResponse*(chatId: string, rpcResult: JsonNode): (string, s
|
||||||
messages.add(jsonMsg.toMessage(pk))
|
messages.add(jsonMsg.toMessage(pk))
|
||||||
return (rpcResult{"cursor"}.getStr, messages)
|
return (rpcResult{"cursor"}.getStr, messages)
|
||||||
|
|
||||||
|
proc parseActivityCenterNotifications*(rpcResult: JsonNode): (string, seq[ActivityCenterNotification]) =
|
||||||
|
let pk = status_settings.getSetting[string](Setting.PublicKey, "0x0")
|
||||||
|
var notifs: seq[ActivityCenterNotification] = @[]
|
||||||
|
var msg: Message
|
||||||
|
if rpcResult{"notifications"}.kind != JNull:
|
||||||
|
for jsonMsg in rpcResult["notifications"]:
|
||||||
|
notifs.add(jsonMsg.toActivityCenterNotification(pk))
|
||||||
|
return (rpcResult{"cursor"}.getStr, notifs)
|
||||||
|
|
||||||
proc rpcChatMessages*(chatId: string, cursorVal: JsonNode, limit: int, success: var bool): string =
|
proc rpcChatMessages*(chatId: string, cursorVal: JsonNode, limit: int, success: var bool): string =
|
||||||
success = true
|
success = true
|
||||||
try:
|
try:
|
||||||
|
@ -101,7 +110,7 @@ proc chatMessages*(chatId: string, cursor: string = ""): (string, seq[Message])
|
||||||
var success: bool
|
var success: bool
|
||||||
let callResult = rpcChatMessages(chatId, cursorVal, 20, success)
|
let callResult = rpcChatMessages(chatId, cursorVal, 20, success)
|
||||||
if success:
|
if success:
|
||||||
result = parseChatMessagesResponse(chatId, callResult.parseJson()["result"])
|
result = parseChatMessagesResponse(callResult.parseJson()["result"])
|
||||||
|
|
||||||
proc parseReactionsResponse*(chatId: string, rpcResult: JsonNode): (string, seq[Reaction]) =
|
proc parseReactionsResponse*(chatId: string, rpcResult: JsonNode): (string, seq[Reaction]) =
|
||||||
var reactions: seq[Reaction] = @[]
|
var reactions: seq[Reaction] = @[]
|
||||||
|
@ -517,7 +526,7 @@ proc banUserFromCommunity*(pubKey: string, communityId: string): string =
|
||||||
}])
|
}])
|
||||||
|
|
||||||
|
|
||||||
proc parseChatPinnedMessagesResponse*(chatId: string, rpcResult: JsonNode): (string, seq[Message]) =
|
proc parseChatPinnedMessagesResponse*(rpcResult: JsonNode): (string, seq[Message]) =
|
||||||
var messages: seq[Message] = @[]
|
var messages: seq[Message] = @[]
|
||||||
let messagesObj = rpcResult{"pinnedMessages"}
|
let messagesObj = rpcResult{"pinnedMessages"}
|
||||||
if(messagesObj != nil and messagesObj.kind != JNull):
|
if(messagesObj != nil and messagesObj.kind != JNull):
|
||||||
|
@ -549,7 +558,7 @@ proc pinnedMessagesByChatID*(chatId: string, cursor: string): (string, seq[Messa
|
||||||
var success: bool
|
var success: bool
|
||||||
let callResult = rpcPinnedChatMessages(chatId, cursorVal, 20, success)
|
let callResult = rpcPinnedChatMessages(chatId, cursorVal, 20, success)
|
||||||
if success:
|
if success:
|
||||||
result = parseChatPinnedMessagesResponse(chatId, callResult.parseJson()["result"])
|
result = parseChatPinnedMessagesResponse(callResult.parseJson()["result"])
|
||||||
|
|
||||||
proc setPinMessage*(messageId: string, chatId: string, pinned: bool) =
|
proc setPinMessage*(messageId: string, chatId: string, pinned: bool) =
|
||||||
discard callPrivateRPC("sendPinMessage".prefix, %*[{
|
discard callPrivateRPC("sendPinMessage".prefix, %*[{
|
||||||
|
@ -566,7 +575,7 @@ proc rpcActivityCenterNotifications*(cursorVal: JsonNode, limit: int, success: v
|
||||||
success = false
|
success = false
|
||||||
result = e.msg
|
result = e.msg
|
||||||
|
|
||||||
proc activityCenterNotification*(cursor: string = "") =
|
proc activityCenterNotification*(cursor: string = ""): (string, seq[ActivityCenterNotification]) =
|
||||||
var cursorVal: JsonNode
|
var cursorVal: JsonNode
|
||||||
|
|
||||||
if cursor == "":
|
if cursor == "":
|
||||||
|
@ -576,6 +585,14 @@ proc activityCenterNotification*(cursor: string = "") =
|
||||||
|
|
||||||
var success: bool
|
var success: bool
|
||||||
let callResult = rpcActivityCenterNotifications(cursorVal, 20, success)
|
let callResult = rpcActivityCenterNotifications(cursorVal, 20, success)
|
||||||
debug "Activity center", callResult
|
if success:
|
||||||
# if success:
|
result = parseActivityCenterNotifications(callResult.parseJson()["result"])
|
||||||
# result = parseChatMessagesResponse(chatId, callResult.parseJson()["result"])
|
|
||||||
|
proc markAllActivityCenterNotificationsRead*() =
|
||||||
|
discard callPrivateRPC("markAllActivityCenterNotificationsRead".prefix, %*[])
|
||||||
|
|
||||||
|
proc unreadActivityCenterNotificationsCount*(): int =
|
||||||
|
let rpcResult = callPrivateRPC("unreadActivityCenterNotificationsCount".prefix, %*[]).parseJson
|
||||||
|
|
||||||
|
if rpcResult{"result"}.kind != JNull:
|
||||||
|
return rpcResult["result"].getInt
|
||||||
|
|
|
@ -23,6 +23,8 @@ proc toCommunity*(jsonCommunity: JsonNode): Community
|
||||||
|
|
||||||
proc toCommunityMembershipRequest*(jsonCommunityMembershipRequest: JsonNode): CommunityMembershipRequest
|
proc toCommunityMembershipRequest*(jsonCommunityMembershipRequest: JsonNode): CommunityMembershipRequest
|
||||||
|
|
||||||
|
proc toActivityCenterNotification*(jsonNotification: JsonNode, pk: string): ActivityCenterNotification
|
||||||
|
|
||||||
proc fromEvent*(event: JsonNode): Signal =
|
proc fromEvent*(event: JsonNode): Signal =
|
||||||
var signal:MessageSignal = MessageSignal()
|
var signal:MessageSignal = MessageSignal()
|
||||||
signal.messages = @[]
|
signal.messages = @[]
|
||||||
|
@ -66,6 +68,10 @@ proc fromEvent*(event: JsonNode): Signal =
|
||||||
for jsonCommunity in event["event"]["requestsToJoinCommunity"]:
|
for jsonCommunity in event["event"]["requestsToJoinCommunity"]:
|
||||||
signal.membershipRequests.add(jsonCommunity.toCommunityMembershipRequest)
|
signal.membershipRequests.add(jsonCommunity.toCommunityMembershipRequest)
|
||||||
|
|
||||||
|
if event["event"]{"activityCenterNotifications"} != nil:
|
||||||
|
for jsonNotification in event["event"]["activityCenterNotifications"]:
|
||||||
|
signal.activityCenterNotification.add(jsonNotification.toActivityCenterNotification(pk))
|
||||||
|
|
||||||
if event["event"]{"pinMessages"} != nil:
|
if event["event"]{"pinMessages"} != nil:
|
||||||
for jsonPinnedMessage in event["event"]["pinMessages"]:
|
for jsonPinnedMessage in event["event"]["pinMessages"]:
|
||||||
var contentType: ContentType
|
var contentType: ContentType
|
||||||
|
@ -365,3 +371,24 @@ proc toReaction*(jsonReaction: JsonNode): Reaction =
|
||||||
emojiId: jsonReaction{"emojiId"}.getInt,
|
emojiId: jsonReaction{"emojiId"}.getInt,
|
||||||
retracted: jsonReaction{"retracted"}.getBool
|
retracted: jsonReaction{"retracted"}.getBool
|
||||||
)
|
)
|
||||||
|
|
||||||
|
proc toActivityCenterNotification*(jsonNotification: JsonNode, pk: string): ActivityCenterNotification =
|
||||||
|
var activityCenterNotificationType: ActivityCenterNotificationType
|
||||||
|
try:
|
||||||
|
activityCenterNotificationType = ActivityCenterNotificationType(jsonNotification{"type"}.getInt)
|
||||||
|
except:
|
||||||
|
warn "Unknown notification type received", type = jsonNotification{"type"}.getInt
|
||||||
|
activityCenterNotificationType = ActivityCenterNotificationType.Unknown
|
||||||
|
result = ActivityCenterNotification(
|
||||||
|
id: jsonNotification{"id"}.getStr,
|
||||||
|
chatId: jsonNotification{"chatId"}.getStr,
|
||||||
|
name: jsonNotification{"name"}.getStr,
|
||||||
|
notificationType: activityCenterNotificationType,
|
||||||
|
timestamp: jsonNotification{"timestamp"}.getInt,
|
||||||
|
read: jsonNotification{"read"}.getBool,
|
||||||
|
dismissed: jsonNotification{"dismissed"}.getBool,
|
||||||
|
accepted: jsonNotification{"accepted"}.getBool
|
||||||
|
)
|
||||||
|
|
||||||
|
if jsonNotification.contains("message") and jsonNotification{"message"}.kind != JNull:
|
||||||
|
result.message = jsonNotification{"message"}.toMessage(pk)
|
||||||
|
|
|
@ -36,6 +36,7 @@ type MessageSignal* = ref object of Signal
|
||||||
emojiReactions*: seq[Reaction]
|
emojiReactions*: seq[Reaction]
|
||||||
communities*: seq[Community]
|
communities*: seq[Community]
|
||||||
membershipRequests*: seq[CommunityMembershipRequest]
|
membershipRequests*: seq[CommunityMembershipRequest]
|
||||||
|
activityCenterNotification*: seq[ActivityCenterNotification]
|
||||||
|
|
||||||
type CommunitySignal* = ref object of Signal
|
type CommunitySignal* = ref object of Signal
|
||||||
community*: Community
|
community*: Community
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
import QtQuick 2.13
|
import QtQuick 2.13
|
||||||
import QtQuick.Controls 2.13
|
import QtQuick.Controls 2.13
|
||||||
|
import QtQml.Models 2.13
|
||||||
import "../../../../shared"
|
import "../../../../shared"
|
||||||
import "../../../../shared/status"
|
import "../../../../shared/status"
|
||||||
import "../../../../imports"
|
import "../../../../imports"
|
||||||
import "./ChatComponents"
|
import "./ChatComponents"
|
||||||
import "../components"
|
import "../components"
|
||||||
|
import "./MessageComponents"
|
||||||
|
|
||||||
Popup {
|
Popup {
|
||||||
enum Filter {
|
enum Filter {
|
||||||
|
@ -43,104 +45,182 @@ Popup {
|
||||||
id: activityCenterTopBar
|
id: activityCenterTopBar
|
||||||
}
|
}
|
||||||
|
|
||||||
Column {
|
ScrollView {
|
||||||
id: notificationsContainer
|
id: scrollView
|
||||||
anchors.top: activityCenterTopBar.bottom
|
anchors.top: activityCenterTopBar.bottom
|
||||||
anchors.topMargin: 13
|
anchors.topMargin: 13
|
||||||
|
anchors.bottom: parent.bottom
|
||||||
|
anchors.bottomMargin: Style.current.smallPadding
|
||||||
width: parent.width
|
width: parent.width
|
||||||
|
clip: true
|
||||||
|
|
||||||
property Component profilePopupComponent: ProfilePopup {
|
ScrollBar.horizontal.policy: ScrollBar.AlwaysOff
|
||||||
id: profilePopup
|
|
||||||
onClosed: destroy()
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO remove this once it is handled by the activity center
|
Column {
|
||||||
Repeater {
|
id: notificationsContainer
|
||||||
id: contactList
|
|
||||||
model: profileModel.contacts.contactRequests
|
|
||||||
|
|
||||||
delegate: ContactRequest {
|
|
||||||
visible: activityCenter.currentFilter === ActivityCenter.Filter.All || activityCenter.currentFilter === ActivityCenter.Filter.ContactRequests
|
|
||||||
name: Utils.removeStatusEns(model.name)
|
|
||||||
address: model.address
|
|
||||||
localNickname: model.localNickname
|
|
||||||
identicon: model.thumbnailImage || model.identicon
|
|
||||||
// TODO set to transparent bg if the notif is read
|
|
||||||
color: Utils.setColorAlpha(Style.current.blue, 0.1)
|
|
||||||
radius: 0
|
|
||||||
profileClick: function (showFooter, userName, fromAuthor, identicon, textParam, nickName) {
|
|
||||||
var popup = profilePopupComponent.createObject(contactList);
|
|
||||||
popup.openPopup(showFooter, userName, fromAuthor, identicon, textParam, nickName);
|
|
||||||
}
|
|
||||||
onBlockContactActionTriggered: {
|
|
||||||
blockContactConfirmationDialog.contactName = name
|
|
||||||
blockContactConfirmationDialog.contactAddress = address
|
|
||||||
blockContactConfirmationDialog.open()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
StyledText {
|
|
||||||
text: "Today"
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: Style.current.padding
|
|
||||||
font.pixelSize: 15
|
|
||||||
bottomPadding: 4
|
|
||||||
topPadding: Style.current.halfPadding
|
|
||||||
color: Style.current.secondaryText
|
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
visible: activityCenter.currentFilter === ActivityCenter.Filter.All || activityCenter.currentFilter === ActivityCenter.Filter.Mentions
|
|
||||||
width: parent.width
|
width: parent.width
|
||||||
height: visible ? childrenRect.height + Style.current.smallPadding : 0
|
spacing: 0
|
||||||
color: Utils.setColorAlpha(Style.current.blue, 0.1)
|
|
||||||
|
|
||||||
Message {
|
property Component profilePopupComponent: ProfilePopup {
|
||||||
id: placeholderMessage
|
id: profilePopup
|
||||||
anchors.right: undefined
|
onClosed: destroy()
|
||||||
messageId: "placeholderMessage"
|
|
||||||
userName: "@vitalik"
|
|
||||||
identicon: ""
|
|
||||||
message: "@roger great question my dude"
|
|
||||||
contentType: Constants.messageType
|
|
||||||
placeholderMessage: true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
StatusIconButton {
|
// TODO remove this once it is handled by the activity center
|
||||||
id: markReadBtn
|
Repeater {
|
||||||
icon.name: "double-check"
|
id: contactList
|
||||||
iconColor: Style.current.primary
|
model: profileModel.contacts.contactRequests
|
||||||
icon.width: 24
|
|
||||||
icon.height: 24
|
|
||||||
width: 32
|
|
||||||
height: 32
|
|
||||||
onClicked: console.log('TODO mark read')
|
|
||||||
anchors.right: parent.right
|
|
||||||
anchors.rightMargin: 12
|
|
||||||
anchors.verticalCenter: placeholderMessage.verticalCenter
|
|
||||||
z: 52
|
|
||||||
|
|
||||||
StatusToolTip {
|
delegate: ContactRequest {
|
||||||
visible: markReadBtn.hovered
|
visible: activityCenter.currentFilter === ActivityCenter.Filter.All || activityCenter.currentFilter === ActivityCenter.Filter.ContactRequests
|
||||||
text: qsTr("Mark as Read")
|
name: Utils.removeStatusEns(model.name)
|
||||||
orientation: "left"
|
address: model.address
|
||||||
x: - width - Style.current.padding
|
localNickname: model.localNickname
|
||||||
y: markReadBtn.height / 2 - height / 2 + 4
|
identicon: model.thumbnailImage || model.identicon
|
||||||
|
// TODO set to transparent bg if the notif is read
|
||||||
|
color: Utils.setColorAlpha(Style.current.blue, 0.1)
|
||||||
|
radius: 0
|
||||||
|
profileClick: function (showFooter, userName, fromAuthor, identicon, textParam, nickName) {
|
||||||
|
var popup = profilePopupComponent.createObject(contactList);
|
||||||
|
popup.openPopup(showFooter, userName, fromAuthor, identicon, textParam, nickName);
|
||||||
|
}
|
||||||
|
onBlockContactActionTriggered: {
|
||||||
|
blockContactConfirmationDialog.contactName = name
|
||||||
|
blockContactConfirmationDialog.contactAddress = address
|
||||||
|
blockContactConfirmationDialog.open()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ActivityChannelBadge {
|
Repeater {
|
||||||
name: "status-desktop-ui"
|
model: notifDelegateList
|
||||||
chatType: Constants.chatTypePublic
|
|
||||||
chatId: "status-desktop-ui"
|
|
||||||
anchors.top: markReadBtn.bottom
|
|
||||||
anchors.topMargin: Style.current.halfPadding
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: 61 // TODO find a way to align with the text of the message
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// TODO add reply placeholder and chaeck if we can do the bubble under
|
DelegateModelGeneralized {
|
||||||
|
id: notifDelegateList
|
||||||
|
lessThan: [
|
||||||
|
function(left, right) { return left.timestamp > right.timestamp }
|
||||||
|
]
|
||||||
|
|
||||||
|
model: chatsModel.activityNotificationList
|
||||||
|
|
||||||
|
delegate: Item {
|
||||||
|
id: notificationDelegate
|
||||||
|
width: parent.width
|
||||||
|
height: notifLoader.active ? childrenRect.height : 0
|
||||||
|
|
||||||
|
property int idx: DelegateModel.itemsIndex
|
||||||
|
|
||||||
|
|
||||||
|
Loader {
|
||||||
|
id: notifLoader
|
||||||
|
anchors.top: parent.top
|
||||||
|
active: !!sourceComponent
|
||||||
|
width: parent.width
|
||||||
|
sourceComponent: {
|
||||||
|
switch (model.notificationType) {
|
||||||
|
// TODO add to constants (mention)
|
||||||
|
case 3: return messageNotificationComponent
|
||||||
|
default: return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: messageNotificationComponent
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
visible: activityCenter.currentFilter === ActivityCenter.Filter.All || activityCenter.currentFilter === ActivityCenter.Filter.Mentions
|
||||||
|
width: parent.width
|
||||||
|
height: childrenRect.height + Style.current.smallPadding
|
||||||
|
color: model.read ? Style.current.transparent : Utils.setColorAlpha(Style.current.blue, 0.1)
|
||||||
|
|
||||||
|
Message {
|
||||||
|
id: notificationMessage
|
||||||
|
anchors.right: undefined
|
||||||
|
fromAuthor: model.message.fromAuthor
|
||||||
|
chatId: model.message.chatId
|
||||||
|
userName: model.message.userName
|
||||||
|
alias: model.message.alias
|
||||||
|
localName: model.message.localName
|
||||||
|
message: model.message.message
|
||||||
|
plainText: model.message.plainText
|
||||||
|
identicon: model.message.identicon
|
||||||
|
isCurrentUser: model.message.isCurrentUser
|
||||||
|
timestamp: model.message.timestamp
|
||||||
|
sticker: model.message.sticker
|
||||||
|
contentType: model.message.contentType
|
||||||
|
outgoingStatus: model.message.outgoingStatus
|
||||||
|
responseTo: model.message.responseTo
|
||||||
|
imageClick: imagePopup.openPopup.bind(imagePopup)
|
||||||
|
messageId: model.message.messageId
|
||||||
|
linkUrls: model.message.linkUrls
|
||||||
|
communityId: model.message.communityId
|
||||||
|
hasMention: model.message.hasMention
|
||||||
|
stickerPackId: model.message.stickerPackId
|
||||||
|
pinnedBy: model.message.pinnedBy
|
||||||
|
pinnedMessage: model.message.isPinned
|
||||||
|
activityCenterMessage: true
|
||||||
|
prevMessageIndex: {
|
||||||
|
if (notificationDelegate.idx === 0) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is used in order to have access to the previous message and determine the timestamp
|
||||||
|
// we can't rely on the index because the sequence of messages is not ordered on the nim side
|
||||||
|
if (notificationDelegate.idx < notifDelegateList.items.count - 1) {
|
||||||
|
return notifDelegateList.items.get(notificationDelegate.idx - 1).model.index
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
prevMsgTimestamp: notificationDelegate.idx === 0 ? "" : chatsModel.activityNotificationList.getNotificationData(prevMessageIndex, "timestamp")
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO add this back when single MarkAsRead is available
|
||||||
|
// StatusIconButton {
|
||||||
|
// id: markReadBtn
|
||||||
|
// icon.name: "double-check"
|
||||||
|
// iconColor: Style.current.primary
|
||||||
|
// icon.width: 24
|
||||||
|
// icon.height: 24
|
||||||
|
// width: 32
|
||||||
|
// height: 32
|
||||||
|
// onClicked: console.log('TODO mark read')
|
||||||
|
// anchors.right: parent.right
|
||||||
|
// anchors.rightMargin: 12
|
||||||
|
// anchors.verticalCenter: notificationMessage.verticalCenter
|
||||||
|
// z: 52
|
||||||
|
|
||||||
|
// StatusToolTip {
|
||||||
|
// visible: markReadBtn.hovered
|
||||||
|
// text: qsTr("Mark as Read")
|
||||||
|
// orientation: "left"
|
||||||
|
// x: - width - Style.current.padding
|
||||||
|
// y: markReadBtn.height / 2 - height / 2 + 4
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
ActivityChannelBadge {
|
||||||
|
name: model.name
|
||||||
|
chatId: model.chatId
|
||||||
|
anchors.top: notificationMessage.bottom
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: 61 // TODO find a way to align with the text of the message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// StyledText {
|
||||||
|
// text: "Today"
|
||||||
|
// anchors.left: parent.left
|
||||||
|
// anchors.leftMargin: Style.current.padding
|
||||||
|
// font.pixelSize: 15
|
||||||
|
// bottomPadding: 4
|
||||||
|
// topPadding: Style.current.halfPadding
|
||||||
|
// color: Style.current.secondaryText
|
||||||
|
// }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -71,7 +71,9 @@ Item {
|
||||||
icon.height: 24
|
icon.height: 24
|
||||||
width: 32
|
width: 32
|
||||||
height: 32
|
height: 32
|
||||||
onClicked: console.log('TODO mark all as read')
|
onClicked: {
|
||||||
|
errorText.text = chatsModel.activityNotificationList.markAllActivityCenterNotificationsRead()
|
||||||
|
}
|
||||||
|
|
||||||
StatusToolTip {
|
StatusToolTip {
|
||||||
visible: markAllReadBtn.hovered
|
visible: markAllReadBtn.hovered
|
||||||
|
@ -109,4 +111,12 @@ Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
id: errorText
|
||||||
|
visible: !!text
|
||||||
|
anchors.top: filterButtons.bottom
|
||||||
|
anchors.topMargin: Style.current.smallPadding
|
||||||
|
color: Style.current.danger
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ Rectangle {
|
||||||
property string chatId: ""
|
property string chatId: ""
|
||||||
property string name: "channelName"
|
property string name: "channelName"
|
||||||
property string identicon
|
property string identicon
|
||||||
property int chatType: Constants.chatTypePublic
|
property int chatType: chatsModel.chats.getChannelType(chatId)
|
||||||
property int realChatType: {
|
property int realChatType: {
|
||||||
if (chatType === Constants.chatTypeCommunity) {
|
if (chatType === Constants.chatTypeCommunity) {
|
||||||
// TODO add a check for private community chats once it is created
|
// TODO add a check for private community chats once it is created
|
||||||
|
|
|
@ -29,6 +29,7 @@ Item {
|
||||||
property bool hasMention: false
|
property bool hasMention: false
|
||||||
property string linkUrls: ""
|
property string linkUrls: ""
|
||||||
property bool placeholderMessage: false
|
property bool placeholderMessage: false
|
||||||
|
property bool activityCenterMessage: false
|
||||||
property bool pinnedMessage: false
|
property bool pinnedMessage: false
|
||||||
property string pinnedBy
|
property string pinnedBy
|
||||||
property bool forceHoverHandler: false // Used to force the HoverHandler to be active (useful for messages in popups)
|
property bool forceHoverHandler: false // Used to force the HoverHandler to be active (useful for messages in popups)
|
||||||
|
@ -176,7 +177,7 @@ Item {
|
||||||
}
|
}
|
||||||
|
|
||||||
function clickMessage(isProfileClick, isSticker = false, isImage = false, image = null, emojiOnly = false, hideEmojiPicker = false) {
|
function clickMessage(isProfileClick, isSticker = false, isImage = false, image = null, emojiOnly = false, hideEmojiPicker = false) {
|
||||||
if (placeholderMessage) {
|
if (placeholderMessage || activityCenterMessage) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -300,7 +301,7 @@ Item {
|
||||||
height: 12
|
height: 12
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
StyledText {
|
StyledText {
|
||||||
id: fetchMoreButton
|
id: fetchMoreButton
|
||||||
font.weight: Font.Medium
|
font.weight: Font.Medium
|
||||||
|
|
|
@ -23,7 +23,7 @@ Item {
|
||||||
+ (dateGroupLbl.visible ? dateGroupLbl.height + dateGroupLbl.anchors.topMargin : 0)
|
+ (dateGroupLbl.visible ? dateGroupLbl.height + dateGroupLbl.anchors.topMargin : 0)
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
enabled: !placeholderMessage
|
enabled: !placeholderMessage && !activityCenterMessage
|
||||||
anchors.fill: messageContainer
|
anchors.fill: messageContainer
|
||||||
acceptedButtons: Qt.RightButton
|
acceptedButtons: Qt.RightButton
|
||||||
onClicked: messageMouseArea.clicked(mouse)
|
onClicked: messageMouseArea.clicked(mouse)
|
||||||
|
@ -53,6 +53,9 @@ Item {
|
||||||
|
|
||||||
DateGroup {
|
DateGroup {
|
||||||
id: dateGroupLbl
|
id: dateGroupLbl
|
||||||
|
previousMessageIndex: prevMessageIndex
|
||||||
|
previousMessageTimestamp: prevMsgTimestamp
|
||||||
|
messageTimestamp: timestamp
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
|
@ -60,7 +63,7 @@ Item {
|
||||||
|
|
||||||
id: messageContainer
|
id: messageContainer
|
||||||
anchors.top: dateGroupLbl.visible ? dateGroupLbl.bottom : parent.top
|
anchors.top: dateGroupLbl.visible ? dateGroupLbl.bottom : parent.top
|
||||||
anchors.topMargin: dateGroupLbl.visible ? Style.current.padding : 0
|
anchors.topMargin: dateGroupLbl.visible ? (activityCenterMessage ? 4 : Style.current.padding) : 0
|
||||||
height: childrenRect.height
|
height: childrenRect.height
|
||||||
+ (chatName.visible || emojiReactionLoader.active ? Style.current.halfPadding : 0)
|
+ (chatName.visible || emojiReactionLoader.active ? Style.current.halfPadding : 0)
|
||||||
+ (chatName.visible && emojiReactionLoader.active ? Style.current.padding : 0)
|
+ (chatName.visible && emojiReactionLoader.active ? Style.current.padding : 0)
|
||||||
|
@ -71,6 +74,10 @@ Item {
|
||||||
width: parent.width
|
width: parent.width
|
||||||
|
|
||||||
color: {
|
color: {
|
||||||
|
if (placeholderMessage || activityCenterMessage) {
|
||||||
|
return Style.current.transparent
|
||||||
|
}
|
||||||
|
|
||||||
if (pinnedMessage) {
|
if (pinnedMessage) {
|
||||||
return root.isHovered || isMessageActive ? Style.current.pinnedMessageBackgroundHovered : Style.current.pinnedMessageBackground
|
return root.isHovered || isMessageActive ? Style.current.pinnedMessageBackgroundHovered : Style.current.pinnedMessageBackground
|
||||||
}
|
}
|
||||||
|
@ -297,7 +304,7 @@ Item {
|
||||||
}
|
}
|
||||||
|
|
||||||
Loader {
|
Loader {
|
||||||
active: hasMention || pinnedMessage
|
active: !activityCenterMessage && (hasMention || pinnedMessage)
|
||||||
height: messageContainer.height
|
height: messageContainer.height
|
||||||
anchors.left: messageContainer.left
|
anchors.left: messageContainer.left
|
||||||
anchors.top: messageContainer.top
|
anchors.top: messageContainer.top
|
||||||
|
|
|
@ -3,20 +3,29 @@ import "../../../../../shared"
|
||||||
import "../../../../../imports"
|
import "../../../../../imports"
|
||||||
|
|
||||||
StyledText {
|
StyledText {
|
||||||
|
property int previousMessageIndex: -1
|
||||||
|
property string previousMessageTimestamp
|
||||||
|
property string messageTimestamp
|
||||||
|
|
||||||
id: dateGroupLbl
|
id: dateGroupLbl
|
||||||
font.pixelSize: 13
|
font.pixelSize: 13
|
||||||
color: Style.current.secondaryText
|
color: Style.current.secondaryText
|
||||||
horizontalAlignment: Text.AlignHCenter
|
horizontalAlignment: Text.AlignHCenter
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
anchors.horizontalCenter: activityCenterMessage ? undefined : parent.horizontalCenter
|
||||||
|
anchors.top: parent.top
|
||||||
|
anchors.topMargin: visible ? (activityCenterMessage ? Style.current.halfPadding : 20) : 0
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: activityCenterMessage ? Style.current.padding : 0
|
||||||
|
|
||||||
text: {
|
text: {
|
||||||
if (prevMessageIndex == -1) return ""; // identifier
|
if (previousMessageIndex === -1) return ""; // identifier
|
||||||
|
|
||||||
let now = new Date()
|
let now = new Date()
|
||||||
let yesterday = new Date()
|
let yesterday = new Date()
|
||||||
yesterday.setDate(now.getDate()-1)
|
yesterday.setDate(now.getDate()-1)
|
||||||
|
|
||||||
var currentMsgDate = new Date(parseInt(timestamp, 10));
|
var currentMsgDate = new Date(parseInt(messageTimestamp, 10));
|
||||||
var prevMsgDate = prevMsgTimestamp === "" ? new Date(0) : new Date(parseInt(prevMsgTimestamp, 10));
|
var prevMsgDate = previousMessageTimestamp === "" ? new Date(0) : new Date(parseInt(previousMessageTimestamp, 10));
|
||||||
|
|
||||||
if (currentMsgDate.getDay() === prevMsgDate.getDay()) {
|
if (currentMsgDate.getDay() === prevMsgDate.getDay()) {
|
||||||
return ""
|
return ""
|
||||||
|
@ -59,6 +68,4 @@ StyledText {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
visible: text !== ""
|
visible: text !== ""
|
||||||
anchors.top: parent.top
|
|
||||||
anchors.topMargin: this.visible ? 20 : 0
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ import "../../../../../shared"
|
||||||
import "../../../../../imports"
|
import "../../../../../imports"
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
enabled: !placeholderMessage
|
enabled: !placeholderMessage && !activityCenterMessage
|
||||||
cursorShape: chatText.hoveredLink ? Qt.PointingHandCursor : undefined
|
cursorShape: chatText.hoveredLink ? Qt.PointingHandCursor : undefined
|
||||||
acceptedButtons: Qt.RightButton | Qt.LeftButton
|
acceptedButtons: Qt.RightButton | Qt.LeftButton
|
||||||
z: 50
|
z: 50
|
||||||
|
|
|
@ -18,6 +18,9 @@ Item {
|
||||||
|
|
||||||
DateGroup {
|
DateGroup {
|
||||||
id: dateGroupLbl
|
id: dateGroupLbl
|
||||||
|
previousMessageIndex: prevMessageIndex
|
||||||
|
previousMessageTimestamp: prevMsgTimestamp
|
||||||
|
messageTimestamp: timestamp
|
||||||
}
|
}
|
||||||
|
|
||||||
UserImage {
|
UserImage {
|
||||||
|
|
|
@ -135,50 +135,50 @@ Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rectangle {
|
Rectangle {
|
||||||
// id: separator
|
id: separator
|
||||||
// width: 1
|
width: 1
|
||||||
// height: 24
|
height: 24
|
||||||
// color: Style.current.separator
|
color: Style.current.separator
|
||||||
// anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
// }
|
}
|
||||||
|
|
||||||
// StatusIconButton {
|
StatusIconButton {
|
||||||
// id: activityCenterBtn
|
id: activityCenterBtn
|
||||||
// icon.name: "bell"
|
icon.name: "bell"
|
||||||
// iconColor: Style.current.contextMenuButtonForegroundColor
|
iconColor: Style.current.contextMenuButtonForegroundColor
|
||||||
// hoveredIconColor: Style.current.contextMenuButtonForegroundColor
|
hoveredIconColor: Style.current.contextMenuButtonForegroundColor
|
||||||
// highlightedBackgroundColor: Style.current.contextMenuButtonBackgroundHoverColor
|
highlightedBackgroundColor: Style.current.contextMenuButtonBackgroundHoverColor
|
||||||
// anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
|
||||||
// onClicked: activityCenter.open()
|
onClicked: activityCenter.open()
|
||||||
|
|
||||||
// Rectangle {
|
Rectangle {
|
||||||
// // TODO unhardcode this
|
property int nbUnseenNotifs: chatsModel.activityNotificationList.unreadCount
|
||||||
// property int nbUnseenNotifs: 3
|
|
||||||
|
|
||||||
// id: badge
|
id: badge
|
||||||
// visible: nbUnseenNotifs > 0
|
visible: nbUnseenNotifs > 0
|
||||||
// anchors.top: parent.top
|
anchors.top: parent.top
|
||||||
// anchors.topMargin: -2
|
anchors.topMargin: -2
|
||||||
// anchors.left: parent.right
|
anchors.left: parent.right
|
||||||
// anchors.leftMargin: -17
|
anchors.leftMargin: -17
|
||||||
// radius: height / 2
|
radius: height / 2
|
||||||
// color: Style.current.blue
|
color: Style.current.blue
|
||||||
// border.color: activityCenterBtn.hovered ? Style.current.secondaryBackground : Style.current.background
|
border.color: activityCenterBtn.hovered ? Style.current.secondaryBackground : Style.current.background
|
||||||
// border.width: 2
|
border.width: 2
|
||||||
// width: badge.nbUnseenNotifs < 10 ? 18 : badge.width + 14
|
width: badge.nbUnseenNotifs < 10 ? 18 : badgeText.width + 14
|
||||||
// height: 18
|
height: 18
|
||||||
|
|
||||||
// Text {
|
Text {
|
||||||
// font.pixelSize: 12
|
id: badgeText
|
||||||
// color: Style.current.white
|
font.pixelSize: 12
|
||||||
// anchors.centerIn: parent
|
color: Style.current.white
|
||||||
// text: badge.nbUnseenNotifs
|
anchors.centerIn: parent
|
||||||
// }
|
text: badge.nbUnseenNotifs
|
||||||
// }
|
}
|
||||||
// }
|
}
|
||||||
// }
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ActivityCenter {
|
ActivityCenter {
|
||||||
id: activityCenter
|
id: activityCenter
|
||||||
|
|
Loading…
Reference in New Issue