2022-08-09 15:08:39 +00:00
|
|
|
import QtQuick 2.14
|
|
|
|
import QtQuick.Controls 2.14
|
|
|
|
import QtQuick.Layouts 1.13
|
|
|
|
|
|
|
|
import StatusQ.Core 0.1
|
|
|
|
import StatusQ.Core.Theme 0.1
|
|
|
|
import StatusQ.Controls 0.1
|
2022-09-12 16:35:20 +00:00
|
|
|
import StatusQ.Core.Utils 0.1 as SQUtils
|
2022-08-09 15:08:39 +00:00
|
|
|
|
|
|
|
import utils 1.0
|
|
|
|
|
|
|
|
import "../panels"
|
2022-09-30 14:36:49 +00:00
|
|
|
import "../stores"
|
2022-08-09 15:08:39 +00:00
|
|
|
|
2022-10-06 09:58:58 +00:00
|
|
|
Item {
|
2022-08-09 15:08:39 +00:00
|
|
|
id: root
|
|
|
|
|
|
|
|
property alias menuButton: menuButton
|
|
|
|
property alias membersButton: membersButton
|
|
|
|
property alias searchButton: searchButton
|
|
|
|
|
|
|
|
property var rootStore
|
2022-12-13 10:45:26 +00:00
|
|
|
property var chatContentModule: root.rootStore.currentChatContentModule() || null
|
2022-09-14 10:38:25 +00:00
|
|
|
property var emojiPopup
|
2022-09-20 10:54:28 +00:00
|
|
|
property int padding: Style.current.halfPadding
|
2022-08-09 15:08:39 +00:00
|
|
|
|
|
|
|
signal searchButtonClicked()
|
|
|
|
|
2022-09-06 13:19:50 +00:00
|
|
|
QtObject {
|
|
|
|
id: d
|
|
|
|
|
|
|
|
readonly property string stateInfoButtonContent: ""
|
|
|
|
readonly property string stateMembersSelectorContent: "selectingMembers"
|
|
|
|
|
|
|
|
readonly property bool selectingMembers: root.state == stateMembersSelectorContent
|
|
|
|
}
|
|
|
|
|
2022-09-30 14:36:49 +00:00
|
|
|
MessageStore {
|
|
|
|
id: messageStore
|
|
|
|
messageModule: chatContentModule ? chatContentModule.messagesModule : null
|
|
|
|
chatSectionModule: root.rootStore.chatCommunitySectionModule
|
|
|
|
}
|
|
|
|
|
2022-08-09 15:08:39 +00:00
|
|
|
Loader {
|
|
|
|
id: loader
|
2022-10-06 09:58:58 +00:00
|
|
|
|
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.bottom: parent.bottom
|
|
|
|
anchors.left: parent.left
|
|
|
|
anchors.right: d.selectingMembers ? parent.right : undefined
|
|
|
|
|
|
|
|
sourceComponent: d.selectingMembers ? membersSelector : statusChatInfoButton
|
|
|
|
}
|
|
|
|
|
|
|
|
Rectangle {
|
|
|
|
anchors.fill: actionButtons
|
|
|
|
visible: actionButtons.visible
|
|
|
|
opacity: 0.8
|
|
|
|
color: Style.current.background
|
2022-08-09 15:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout {
|
|
|
|
id: actionButtons
|
2022-10-06 09:58:58 +00:00
|
|
|
|
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.bottom: parent.bottom
|
|
|
|
anchors.right: parent.right
|
|
|
|
|
2022-08-09 15:08:39 +00:00
|
|
|
spacing: 8
|
2022-09-06 13:19:50 +00:00
|
|
|
visible: !d.selectingMembers
|
2022-08-09 15:08:39 +00:00
|
|
|
|
|
|
|
StatusFlatRoundButton {
|
|
|
|
id: searchButton
|
|
|
|
icon.name: "search"
|
|
|
|
type: StatusFlatRoundButton.Type.Secondary
|
|
|
|
onClicked: root.searchButtonClicked()
|
|
|
|
|
|
|
|
// initializing the tooltip
|
|
|
|
tooltip.text: qsTr("Search")
|
|
|
|
tooltip.orientation: StatusToolTip.Orientation.Bottom
|
|
|
|
tooltip.y: parent.height + 12
|
|
|
|
}
|
|
|
|
|
|
|
|
StatusFlatRoundButton {
|
|
|
|
id: membersButton
|
|
|
|
visible: {
|
|
|
|
if(!chatContentModule || chatContentModule.chatDetails.type === Constants.chatType.publicChat)
|
|
|
|
return false
|
|
|
|
|
|
|
|
return localAccountSensitiveSettings.showOnlineUsers &&
|
|
|
|
chatContentModule.chatDetails.isUsersListAvailable
|
|
|
|
}
|
|
|
|
highlighted: localAccountSensitiveSettings.expandUsersList
|
|
|
|
icon.name: "group-chat"
|
|
|
|
type: StatusFlatRoundButton.Type.Secondary
|
|
|
|
onClicked: {
|
|
|
|
localAccountSensitiveSettings.expandUsersList = !localAccountSensitiveSettings.expandUsersList;
|
|
|
|
}
|
|
|
|
// initializing the tooltip
|
|
|
|
tooltip.text: qsTr("Members")
|
|
|
|
tooltip.orientation: StatusToolTip.Orientation.Bottom
|
|
|
|
tooltip.y: parent.height + 12
|
|
|
|
}
|
|
|
|
|
|
|
|
StatusFlatRoundButton {
|
|
|
|
id: menuButton
|
|
|
|
objectName: "chatToolbarMoreOptionsButton"
|
|
|
|
icon.name: "more"
|
|
|
|
type: StatusFlatRoundButton.Type.Secondary
|
|
|
|
|
|
|
|
// initializing the tooltip
|
|
|
|
tooltip.visible: !!tooltip.text && menuButton.hovered && !contextMenu.opened
|
|
|
|
tooltip.text: qsTr("More")
|
|
|
|
tooltip.orientation: StatusToolTip.Orientation.Bottom
|
|
|
|
tooltip.y: parent.height + 12
|
|
|
|
|
|
|
|
property bool showMoreMenu: false
|
|
|
|
onClicked: {
|
|
|
|
menuButton.highlighted = true
|
|
|
|
|
|
|
|
let originalOpenHandler = contextMenu.openHandler
|
|
|
|
let originalCloseHandler = contextMenu.closeHandler
|
|
|
|
|
|
|
|
contextMenu.openHandler = function () {
|
|
|
|
if (!!originalOpenHandler) {
|
|
|
|
originalOpenHandler()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
contextMenu.closeHandler = function () {
|
|
|
|
menuButton.highlighted = false
|
|
|
|
if (!!originalCloseHandler) {
|
|
|
|
originalCloseHandler()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
contextMenu.openHandler = originalOpenHandler
|
|
|
|
contextMenu.popup(-contextMenu.width + menuButton.width, menuButton.height + 4)
|
|
|
|
}
|
|
|
|
|
|
|
|
ChatContextMenuView {
|
|
|
|
id: contextMenu
|
|
|
|
objectName: "moreOptionsContextMenu"
|
|
|
|
emojiPopup: root.emojiPopup
|
|
|
|
openHandler: function () {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on open chat context menu handler - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
currentFleet = chatContentModule.getCurrentFleet()
|
|
|
|
isCommunityChat = chatContentModule.chatDetails.belongsToCommunity
|
|
|
|
amIChatAdmin = chatContentModule.amIChatAdmin()
|
|
|
|
chatId = chatContentModule.chatDetails.id
|
|
|
|
chatName = chatContentModule.chatDetails.name
|
|
|
|
chatDescription = chatContentModule.chatDetails.description
|
|
|
|
chatEmoji = chatContentModule.chatDetails.emoji
|
|
|
|
chatColor = chatContentModule.chatDetails.color
|
2022-09-07 10:03:51 +00:00
|
|
|
chatIcon = chatContentModule.chatDetails.icon
|
2022-08-09 15:08:39 +00:00
|
|
|
chatType = chatContentModule.chatDetails.type
|
|
|
|
chatMuted = chatContentModule.chatDetails.muted
|
|
|
|
channelPosition = chatContentModule.chatDetails.position
|
|
|
|
}
|
|
|
|
|
|
|
|
onMuteChat: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on mute chat from context menu - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.muteChat()
|
|
|
|
}
|
|
|
|
|
|
|
|
onUnmuteChat: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on unmute chat from context menu - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.unmuteChat()
|
|
|
|
}
|
|
|
|
|
|
|
|
onMarkAllMessagesRead: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on mark all messages read from context menu - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.markAllMessagesRead()
|
|
|
|
}
|
|
|
|
|
|
|
|
onClearChatHistory: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on clear chat history from context menu - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.clearChatHistory()
|
|
|
|
}
|
|
|
|
|
|
|
|
onRequestAllHistoricMessages: {
|
|
|
|
// Not Refactored Yet - Check in the `master` branch if this is applicable here.
|
|
|
|
}
|
|
|
|
|
|
|
|
onLeaveChat: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on leave chat from context menu - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.leaveChat()
|
|
|
|
}
|
|
|
|
|
|
|
|
onDeleteCommunityChat: root.rootStore.removeCommunityChat(chatId)
|
|
|
|
|
|
|
|
onDownloadMessages: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on leave chat from context menu - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.downloadMessages(file)
|
|
|
|
}
|
|
|
|
|
|
|
|
onDisplayProfilePopup: {
|
|
|
|
Global.openProfilePopup(publicKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
onEditCommunityChannel: {
|
|
|
|
root.rootStore.editCommunityChannel(
|
|
|
|
chatId,
|
|
|
|
newName,
|
|
|
|
newDescription,
|
|
|
|
newEmoji,
|
|
|
|
newColor,
|
|
|
|
newCategory,
|
|
|
|
channelPosition // TODO change this to the signal once it is modifiable
|
|
|
|
)
|
|
|
|
}
|
|
|
|
onAddRemoveGroupMember: {
|
2022-09-06 13:19:50 +00:00
|
|
|
root.state = d.stateMembersSelectorContent
|
2022-08-09 15:08:39 +00:00
|
|
|
}
|
|
|
|
onFetchMoreMessages: {
|
2022-09-30 14:36:49 +00:00
|
|
|
messageStore.requestMoreMessages();
|
2022-08-09 15:08:39 +00:00
|
|
|
}
|
|
|
|
onLeaveGroup: {
|
|
|
|
chatContentModule.leaveChat();
|
|
|
|
}
|
|
|
|
onUpdateGroupChatDetails: {
|
|
|
|
root.rootStore.chatCommunitySectionModule.updateGroupChatDetails(
|
|
|
|
chatId,
|
|
|
|
groupName,
|
|
|
|
groupColor,
|
|
|
|
groupImage
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Rectangle {
|
|
|
|
implicitWidth: 1
|
|
|
|
implicitHeight: 24
|
|
|
|
color: Theme.palette.directColor7
|
|
|
|
Layout.alignment: Qt.AlignVCenter
|
|
|
|
visible: (menuButton.visible || membersButton.visible || searchButton.visible)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Chat toolbar content option 1:
|
|
|
|
Component {
|
|
|
|
id: statusChatInfoButton
|
|
|
|
|
|
|
|
StatusChatInfoButton {
|
2022-10-24 18:53:42 +00:00
|
|
|
readonly property string emojiIcon: chatContentModule? chatContentModule.chatDetails.emoji : "" // Needed for test
|
|
|
|
|
2022-08-09 15:08:39 +00:00
|
|
|
objectName: "chatInfoBtnInHeader"
|
2022-11-17 09:55:49 +00:00
|
|
|
title: chatContentModule? chatContentModule.chatDetails.name : ""
|
2022-09-22 22:18:15 +00:00
|
|
|
|
2022-08-09 15:08:39 +00:00
|
|
|
subTitle: {
|
|
|
|
if(!chatContentModule)
|
|
|
|
return ""
|
|
|
|
|
|
|
|
// In some moment in future this should be part of the backend logic.
|
|
|
|
// (once we add transaltion on the backend side)
|
|
|
|
switch (chatContentModule.chatDetails.type) {
|
|
|
|
case Constants.chatType.publicChat:
|
|
|
|
return qsTr("Public chat")
|
|
|
|
case Constants.chatType.privateGroupChat:
|
2022-09-07 10:03:51 +00:00
|
|
|
return qsTr("%n member(s)", "", chatContentModule.usersModule.model.count)
|
2022-08-09 15:08:39 +00:00
|
|
|
case Constants.chatType.communityChat:
|
2022-09-12 16:35:20 +00:00
|
|
|
return SQUtils.Utils.linkifyAndXSS(chatContentModule.chatDetails.description).trim()
|
2022-08-09 15:08:39 +00:00
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
2022-08-11 11:55:08 +00:00
|
|
|
asset.name: chatContentModule? chatContentModule.chatDetails.icon : ""
|
2022-12-13 10:45:26 +00:00
|
|
|
asset.isImage: chatContentModule && chatContentModule.chatDetails.icon !== ""
|
|
|
|
asset.isLetterIdenticon: chatContentModule && chatContentModule.chatDetails.icon === ""
|
2022-08-09 15:08:39 +00:00
|
|
|
ringSettings.ringSpecModel: chatContentModule && chatContentModule.chatDetails.type === Constants.chatType.oneToOne ?
|
|
|
|
Utils.getColorHashAsJson(chatContentModule.chatDetails.id) : ""
|
2022-08-11 11:55:08 +00:00
|
|
|
asset.color: chatContentModule?
|
2022-08-09 15:08:39 +00:00
|
|
|
chatContentModule.chatDetails.type === Constants.chatType.oneToOne ?
|
|
|
|
Utils.colorForPubkey(chatContentModule.chatDetails.id)
|
|
|
|
: chatContentModule.chatDetails.color
|
|
|
|
: ""
|
2022-10-24 18:53:42 +00:00
|
|
|
asset.emoji: emojiIcon
|
2022-08-11 11:55:08 +00:00
|
|
|
asset.emojiSize: "24x24"
|
2022-08-09 15:08:39 +00:00
|
|
|
type: chatContentModule? chatContentModule.chatDetails.type : Constants.chatType.unknown
|
|
|
|
pinnedMessagesCount: chatContentModule? chatContentModule.pinnedMessagesModel.count : 0
|
|
|
|
muted: chatContentModule? chatContentModule.chatDetails.muted : false
|
|
|
|
|
|
|
|
onPinnedMessagesCountClicked: {
|
|
|
|
if(!chatContentModule) {
|
2022-09-19 08:40:02 +00:00
|
|
|
console.warn("error on open pinned messages - chat content module is not set")
|
2022-08-09 15:08:39 +00:00
|
|
|
return
|
|
|
|
}
|
2022-09-12 16:35:20 +00:00
|
|
|
Global.openPopup(Global.pinnedMessagesPopup, {
|
2022-08-09 15:08:39 +00:00
|
|
|
store: rootStore,
|
2022-09-30 14:36:49 +00:00
|
|
|
messageStore: messageStore,
|
2022-08-09 15:08:39 +00:00
|
|
|
pinnedMessagesModel: chatContentModule.pinnedMessagesModel,
|
|
|
|
messageToPin: ""
|
|
|
|
})
|
|
|
|
}
|
|
|
|
onUnmute: {
|
|
|
|
if(!chatContentModule) {
|
|
|
|
console.debug("error on unmute chat - chat content module is not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chatContentModule.unmuteChat()
|
|
|
|
}
|
|
|
|
|
2022-09-14 10:41:29 +00:00
|
|
|
hoverEnabled: {
|
2022-08-09 15:08:39 +00:00
|
|
|
if(!chatContentModule)
|
|
|
|
return false
|
|
|
|
|
|
|
|
return chatContentModule.chatDetails.type !== Constants.chatType.publicChat &&
|
2022-09-08 07:49:24 +00:00
|
|
|
chatContentModule.chatDetails.type !== Constants.chatType.communityChat &&
|
|
|
|
chatContentModule.chatDetails.type !== Constants.chatType.privateGroupChat
|
2022-08-09 15:08:39 +00:00
|
|
|
}
|
|
|
|
onClicked: {
|
|
|
|
switch (chatContentModule.chatDetails.type) {
|
|
|
|
case Constants.chatType.oneToOne:
|
|
|
|
Global.openProfilePopup(chatContentModule.chatDetails.id)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-01-04 00:23:29 +00:00
|
|
|
onLinkActivated: Global.openLink(link)
|
2022-08-09 15:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Chat toolbar content option 2:
|
|
|
|
Component {
|
2022-09-06 13:19:50 +00:00
|
|
|
id: membersSelector
|
|
|
|
|
|
|
|
MembersEditSelectorView {
|
2022-08-09 15:08:39 +00:00
|
|
|
rootStore: root.rootStore
|
2022-11-08 08:36:08 +00:00
|
|
|
usersStore: UsersStore {
|
|
|
|
usersModule: root.chatContentModule.usersModule
|
|
|
|
}
|
2022-09-06 13:19:50 +00:00
|
|
|
|
|
|
|
onConfirmed: root.state = d.stateInfoButtonContent
|
|
|
|
onRejected: root.state = d.stateInfoButtonContent
|
2022-08-09 15:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|