feat: edit community channels

Closes #2344.

Add ability to edit name, description, and private fields of a community channel.

feat: Display community channel description
Ensure the width of the description does not surpass the context menu and instead wraps to the next line.

feat: After channel is created, set it as the active channel
This commit is contained in:
Eric Mastro 2021-05-28 12:55:50 +10:00 committed by Eric Mastro
parent a564b48f61
commit 21e6affa98
22 changed files with 297 additions and 129 deletions

View File

@ -20,6 +20,29 @@ toc: true
### `createCommunity` ### `createCommunity`
### `createCommunityChannel`
Creates a community channel with the given name and description, belonging to the community specified by the `communityId`.
Returns a `Chat` object containing the created community channel.
Throws an`RpcException` if there is an error returned from status-go.
*Parameters*
| Name | Type | Description |
|---------------|----------|--------------|
| `communityId` | `string` | community id |
| `name` | `string` | community name |
| `description` | `string` | community description |
### `editCommunityChannel`
Edits a community channel, specified by `communityId` and `channelId`, with the given name and description.
Returns a `Chat` object with the edited community channel.
Throws an `RpcException` if there is an error returned from status-go.
*Parameters*
| Name | Type | Description |
|---------------|----------|--------------|
| `communityId` | `string` | community id |
| `channelId` | `string` | channel id |
| `name` | `string` | community name |
| `description` | `string` | community description |
### `createCommunityChat` ### `createCommunityChat`
### `createCommunityCategory` ### `createCommunityCategory`

View File

@ -33,7 +33,7 @@ toc: true
- [`src/app/chat/view.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/app/chat/view.nim#L166) - [`src/app/chat/view.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/app/chat/view.nim#L166)
- [`src/app/chat/views/community_list.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/app/chat/views/community_list.nim#L40) - [`src/app/chat/views/community_list.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/app/chat/views/community_list.nim#L40)
- [`src/app/chat/views/communities.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/app/chat/views/communities.nim#L109) - [`src/app/chat/views/communities.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/app/chat/views/communities.nim#L109)
- [`src/status/chat.nim#](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/status/chat.nim#L441) - [`src/status/chat.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/status/chat.nim#L441)
- [`src/status/libstatus/chat.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/status/libstatus/chat.nim#L272) - [`src/status/libstatus/chat.nim`](https://github.com/status-im/status-desktop/blob/358091a8eb19f36c9843b42d61473e35ea87d05b/src/status/libstatus/chat.nim#L272)
#### selecting a community #### selecting a community
@ -78,6 +78,24 @@ toc: true
## Manage Community ## Manage Community
### Creating Channels ### Creating Channels
**Key source files**
- [`ui/app/AppLayouts/Chat/CommunityComponents/CreateChannelPopup.qml`](https://github.com/status-im/status-desktop/blob/4a407d920499e3c31244f019afc0ab20f2c8f5e3/ui/app/AppLayouts/Chat/CommunityComponents/CreateChannelPopup.qml)
- [`ui/app/AppLayouts/Chat/CommunityColumn.qml`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/ui/app/AppLayouts/Chat/CommunityColumn.qml#L84-L92)
- [`src/status/chat.nim`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/src/status/chat.nim#L478-L479)
- [`src/status/libstatus/chat.nim`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/src/status/libstatus/chat.nim#L337-L364)
- [`src/app/chat/view.nim#L978-L990`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/src/app/chat/view.nim#L978-L990)
### Editing Channels
**Key source files**
- [`ui/app/AppLayouts/Chat/CommunityComponents/CreateChannelPopup.qml`](https://github.com/status-im/status-desktop/blob/4a407d920499e3c31244f019afc0ab20f2c8f5e3/ui/app/AppLayouts/Chat/CommunityComponents/CreateChannelPopup.qml#L179-L182)
- [`ui/app/AppLayouts/Chat/components/ChannelContextMenu.qml`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/ui/app/AppLayouts/Chat/components/ChannelContextMenu.qml#L109-L116)
- [`ui/app/AppMain.qml`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/ui/app/AppMain.qml#L283-L291)
- [`src/status/chat.nim`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/src/status/chat.nim#L481-L482)
- [`src/status/libstatus/chat.nim`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/src/status/libstatus/chat.nim#L366-L394)
- [`src/app/chat/view.nim`](https://github.com/status-im/status-desktop/blob/1f585d159b9814198863b729ed26a218c09ea56d/src/app/chat/view.nim#L992-L1002)
**Notes**
Editing a channel reuses the same modal popup used to create a channel, the difference being that it's prefilled with information from the selected channel, and has the `isEdit` attribute set to true, which determines the UI behavior for editing the channel as well as knowing the right slot to call.
### Categories ### Categories
Channels within a community might be organized in categories. Only the community admin might create/edit/delete a category. Creating a channel in a category works by calling `wakuext_reorderCommunityChat` after the chat is created, then the `Chat` is immediatly assigned a `categoryId`. Channels within a community might be organized in categories. Only the community admin might create/edit/delete a category. Creating a channel in a category works by calling `wakuext_reorderCommunityChat` after the chat is created, then the `Chat` is immediatly assigned a `categoryId`.

View File

@ -61,6 +61,11 @@ toc: true
![url_bar](/images/communities/community_menu.png) ![url_bar](/images/communities/community_menu.png)
![url_bar](/images/communities/new_channel.png) ![url_bar](/images/communities/new_channel.png)
### Editing Channels
![url_bar](/images/communities/community_edit_menu.png)
![url_bar](/images/communities/edit_channel.png)
### Creating Categories ### Creating Categories

Binary file not shown.

After

Width:  |  Height:  |  Size: 44 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 75 KiB

View File

@ -44,7 +44,8 @@ proc handleChatEvents(self: ChatController) =
if (self.view.communities.activeCommunity.active and self.view.communities.activeCommunity.communityItem.id == community.id): if (self.view.communities.activeCommunity.active and self.view.communities.activeCommunity.communityItem.id == community.id):
if (self.view.activeChannel.chatItem != nil): if (self.view.activeChannel.chatItem != nil):
let communityChannel = self.view.communities.activeCommunity.chats.getChannelById(self.view.activeChannel.chatItem.id) let communityChannel = self.view.communities.activeCommunity.chats.getChannelById(self.view.activeChannel.chatItem.id)
self.view.activeChannel.chatItem.canPost = communityChannel.canPost if communityChannel != nil:
self.view.activeChannel.chatItem.canPost = communityChannel.canPost
self.view.activeChannelChanged() self.view.activeChannelChanged()
if (evArgs.communityMembershipRequests.len > 0): if (evArgs.communityMembershipRequests.len > 0):

View File

@ -17,6 +17,8 @@ import views/[channels_list, message_list, chat_item, suggestions_list, reaction
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
import ../../status/signals/types as signal_types
import ../../status/libstatus/types
logScope: logScope:
topics = "chats-view" topics = "chats-view"
@ -972,3 +974,29 @@ QtObject:
proc formatInputCode(self: ChatsView, inputText: string): string {.slot.} = proc formatInputCode(self: ChatsView, inputText: string): string {.slot.} =
let strikeThroughRegex = re"""(?<!\>)`(?!<span style=" font-family:'monospace';">)([^*]+)(?!<\/span>)`""" let strikeThroughRegex = re"""(?<!\>)`(?!<span style=" font-family:'monospace';">)([^*]+)(?!<\/span>)`"""
self.formatInputStuff(strikeThroughRegex, inputText) self.formatInputStuff(strikeThroughRegex, inputText)
proc createCommunityChannel*(self: ChatsView, communityId: string, name: string, description: string, categoryId: string): string {.slot.} =
try:
let chat = self.status.chat.createCommunityChannel(communityId, name, description)
if categoryId != "":
self.status.chat.reorderCommunityChannel(communityId, categoryId, chat.id.replace(communityId, ""), 0)
chat.categoryId = categoryId
self.communities.joinedCommunityList.addChannelToCommunity(communityId, chat)
self.communities.activeCommunity.addChatItemToList(chat)
self.setActiveChannel(chat.id)
except RpcException as e:
error "Error creating channel", msg=e.msg, name, description
result = StatusGoError(error: e.msg).toJson
proc editCommunityChannel*(self: ChatsView, communityId: string, channelId: string, name: string, description: string, categoryId: string): string {.slot.} =
try:
let chat = self.status.chat.editCommunityChannel(communityId, channelId, name, description)
chat.categoryId = categoryId
self.communities.joinedCommunityList.replaceChannelInCommunity(communityId, chat)
self.communities.activeCommunity.updateChatItemInList(chat)
self.setActiveChannel(chat.id)
except RpcException as e:
error "Error editing channel", msg=e.msg, channelId, name, description
result = StatusGoError(error: e.msg).toJson

View File

@ -39,6 +39,16 @@ QtObject:
QtProperty[string] id: QtProperty[string] id:
read = id read = id
proc description*(self: ChatItemView): string {.slot.} = result = ?.self.chatItem.description
QtProperty[string] description:
read = description
proc private*(self: ChatItemView): bool {.slot.} = result = ?.self.chatItem.private
QtProperty[bool] private:
read = private
proc contactsUpdated*(self: ChatItemView) {.signal} proc contactsUpdated*(self: ChatItemView) {.signal}
proc userNameOrAlias(self: ChatItemView, pubKey: string): string {.slot.} = proc userNameOrAlias(self: ChatItemView, pubKey: string): string {.slot.} =

View File

@ -245,23 +245,6 @@ QtObject:
error "Error editing the community", msg = e.msg error "Error editing the community", msg = e.msg
result = StatusGoError(error: e.msg).toJson result = StatusGoError(error: e.msg).toJson
proc createCommunityChannel*(self: CommunitiesView, communityId: string, name: string, description: string, categoryId: string): string {.slot.} =
result = ""
try:
let chat = self.status.chat.createCommunityChannel(communityId, name, description)
if (chat.id == ""):
return "Chat was not created. Please try again later"
if categoryId != "":
self.status.chat.reorderCommunityChannel(communityId, categoryId, chat.id.replace(communityId, ""), 0)
chat.categoryId = categoryId
self.joinedCommunityList.addChannelToCommunity(communityId, chat)
self.activeCommunity.addChatItemToList(chat)
except Exception as e:
error "Error creating the channel", msg = e.msg
result = fmt"Error creating the channel: {e.msg}"
proc createCommunityCategory*(self: CommunitiesView, communityId: string, name: string, channels: string): string {.slot.} = proc createCommunityCategory*(self: CommunitiesView, communityId: string, name: string, channels: string): string {.slot.} =
result = "" result = ""
@ -295,7 +278,6 @@ QtObject:
error "Error creating the category", msg = e.msg error "Error creating the category", msg = e.msg
result = fmt"Error creating the category: {e.msg}" result = fmt"Error creating the category: {e.msg}"
proc observedCommunityChanged*(self: CommunitiesView) {.signal.} proc observedCommunityChanged*(self: CommunitiesView) {.signal.}
proc setObservedCommunity*(self: CommunitiesView, communityId: string) {.slot.} = proc setObservedCommunity*(self: CommunitiesView, communityId: string) {.slot.} =

View File

@ -156,6 +156,10 @@ QtObject:
discard self.chats.addChatItemToList(chat) discard self.chats.addChatItemToList(chat)
self.chatsChanged() self.chatsChanged()
proc updateChatItemInList*(self: CommunityItemView, chat: Chat) =
self.chats.updateChat(chat)
self.chatsChanged()
proc addCategoryToList*(self: CommunityItemView, category: CommunityCategory) = proc addCategoryToList*(self: CommunityItemView, category: CommunityCategory) =
self.communityItem.categories.add(category) self.communityItem.categories.add(category)
discard self.categories.addCategoryToList(category) discard self.categories.addCategoryToList(category)

View File

@ -129,6 +129,13 @@ QtObject:
let index = self.communities.findIndexById(communityId) let index = self.communities.findIndexById(communityId)
self.communities[index] = community self.communities[index] = community
proc replaceChannelInCommunity*(self: CommunityList, communityId: string, channel: Chat) =
var community = self.getCommunityById(communityId)
if community.id != "":
let channelIdx = community.chats.findIndexById(channel.id)
if channelIdx > -1:
community.chats[channelIdx] = channel
proc addCategoryToCommunity*(self: CommunityList, communityId: string, category: CommunityCategory) = proc addCategoryToCommunity*(self: CommunityList, communityId: string, category: CommunityCategory) =
var community = self.getCommunityById(communityId) var community = self.getCommunityById(communityId)
community.categories.add(category) community.categories.add(category)

View File

@ -478,6 +478,9 @@ proc editCommunity*(self: ChatModel, id: string, name: string, description: stri
proc createCommunityChannel*(self: ChatModel, communityId: string, name: string, description: string): Chat = proc createCommunityChannel*(self: ChatModel, communityId: string, name: string, description: string): Chat =
result = status_chat.createCommunityChannel(communityId, name, description) result = status_chat.createCommunityChannel(communityId, name, description)
proc editCommunityChannel*(self: ChatModel, communityId: string, channelId: string, name: string, description: string): Chat =
result = status_chat.editCommunityChannel(communityId, channelId, name, description)
proc createCommunityCategory*(self: ChatModel, communityId: string, name: string, channels: seq[string]): CommunityCategory = proc createCommunityCategory*(self: ChatModel, communityId: string, name: string, channels: seq[string]): CommunityCategory =
result = status_chat.createCommunityCategory(communityId, name, channels) result = status_chat.createCommunityCategory(communityId, name, channels)

View File

@ -60,6 +60,7 @@ proc toJsonNode*(self: seq[ChatMembershipEvent]): seq[JsonNode] =
type Chat* = ref object type Chat* = ref object
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 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
communityId*: string communityId*: string
private*: bool
categoryId*: string categoryId*: string
name*: string name*: string
description*: string description*: string

View File

@ -356,7 +356,41 @@ proc createCommunityChannel*(communityId: string, name: string, description: str
} }
}]).parseJSON() }]).parseJSON()
if rpcResult{"result"}.kind != JNull: if rpcResult{"error"} != nil:
let error = Json.decode($rpcResult{"error"}, RpcError)
raise newException(RpcException, "Error creating community channel: " & error.message)
if rpcResult{"result"} != nil and rpcResult{"result"}.kind != JNull:
result = rpcResult["result"]["chats"][0].toChat()
proc editCommunityChannel*(communityId: string, channelId: string, name: string, description: string): Chat =
let rpcResult = callPrivateRPC("editCommunityChat".prefix, %*[
communityId,
channelId.replace(communityId, ""),
{
"permissions": {
"access": 1 # TODO get this from user selected privacy setting
},
"identity": {
"display_name": name,
"description": description#,
# "color": color#,
# TODO add images once it is supported by Status-Go
# "images": [
# {
# "payload": image,
# # TODO get that from an enum
# "image_type": 1 # 1 is a raw payload
# }
# ]
}
}]).parseJSON()
if rpcResult{"error"} != nil:
let error = Json.decode($rpcResult{"error"}, RpcError)
raise newException(RpcException, "Error editing community channel: " & error.message)
if rpcResult{"result"} != nil and rpcResult{"result"}.kind != JNull:
result = rpcResult["result"]["chats"][0].toChat() result = rpcResult["result"]["chats"][0].toChat()
proc createCommunityCategory*(communityId: string, name: string, channels: seq[string]): CommunityCategory = proc createCommunityCategory*(communityId: string, name: string, channels: seq[string]): CommunityCategory =

View File

@ -150,6 +150,7 @@ proc toChat*(jsonChat: JsonNode): Chat =
id: jsonChat{"id"}.getStr, id: jsonChat{"id"}.getStr,
communityId: jsonChat{"communityId"}.getStr, communityId: jsonChat{"communityId"}.getStr,
name: jsonChat{"name"}.getStr, name: jsonChat{"name"}.getStr,
description: jsonChat{"description"}.getStr,
identicon: "", identicon: "",
color: jsonChat{"color"}.getStr, color: jsonChat{"color"}.getStr,
isActive: jsonChat{"active"}.getBool, isActive: jsonChat{"active"}.getBool,
@ -161,7 +162,8 @@ proc toChat*(jsonChat: JsonNode): Chat =
hasMentions: false, hasMentions: false,
muted: false, muted: false,
ensName: "", ensName: "",
joined: 0 joined: 0,
private: jsonChat{"private"}.getBool
) )
if jsonChat.hasKey("muted") and jsonChat["muted"].kind != JNull: if jsonChat.hasKey("muted") and jsonChat["muted"].kind != JNull:
@ -223,8 +225,10 @@ proc toCommunity*(jsonCommunity: JsonNode): Community =
categoryId: chat{"categoryID"}.getStr(), categoryId: chat{"categoryID"}.getStr(),
communityId: result.id, communityId: result.id,
name: chat{"name"}.getStr, name: chat{"name"}.getStr,
description: chat{"description"}.getStr,
canPost: chat{"canPost"}.getBool, canPost: chat{"canPost"}.getBool,
chatType: ChatType.CommunityChat chatType: ChatType.CommunityChat,
private: chat{"permissions"}{"private"}.getBool
)) ))
if jsonCommunity.hasKey("categories") and jsonCommunity["categories"].kind != JNull: if jsonCommunity.hasKey("categories") and jsonCommunity["categories"].kind != JNull:

View File

@ -18,6 +18,8 @@ Item {
anchors.leftMargin: this.isGroupChatOrOneToOne ? Style.current.padding : Style.current.padding + 4 anchors.leftMargin: this.isGroupChatOrOneToOne ? Style.current.padding : Style.current.padding + 4
anchors.top: parent.top anchors.top: parent.top
anchors.topMargin: 4 anchors.topMargin: 4
anchors.right: moreActionsBtn.left
anchors.rightMargin: Style.current.padding + moreActionsBtn.width
sourceComponent: this.isGroupChatOrOneToOne ? chatInfoButton : chatInfo sourceComponent: this.isGroupChatOrOneToOne ? chatInfoButton : chatInfo
} }
@ -64,78 +66,74 @@ Item {
} }
} }
Row { StatusContextMenuButton {
height: parent.height id: moreActionsBtn
anchors.verticalCenter: parent.verticalCenter
anchors.right: parent.right anchors.right: parent.right
anchors.rightMargin: Style.current.smallPadding anchors.rightMargin: Style.current.smallPadding
spacing: 12
StatusContextMenuButton { onClicked: {
id: moreActionsBtn var menu = chatContextMenu;
anchors.verticalCenter: parent.verticalCenter var isPrivateGroupChat = chatsModel.activeChannel.chatType === Constants.chatTypePrivateGroupChat
if(isPrivateGroupChat){
onClicked: { menu = groupContextMenu
var menu = chatContextMenu;
var isPrivateGroupChat = chatsModel.activeChannel.chatType === Constants.chatTypePrivateGroupChat
if(isPrivateGroupChat){
menu = groupContextMenu
}
if (menu.opened) {
return menu.close()
}
if (isPrivateGroupChat) {
menu.popup(moreActionsBtn.x, moreActionsBtn.height)
} else {
menu.openMenu(chatsModel.activeChannel, chatsModel.getActiveChannelIdx(),
moreActionsBtn.x - chatContextMenu.width + moreActionsBtn.width + 4,
moreActionsBtn.height - 4)
}
} }
ChannelContextMenu { if (menu.opened) {
id: chatContextMenu return menu.close()
closePolicy: Popup.CloseOnEscape | Popup.CloseOnPressOutsideParent
} }
PopupMenu { if (isPrivateGroupChat) {
id: groupContextMenu menu.popup(moreActionsBtn.x, moreActionsBtn.height)
closePolicy: Popup.CloseOnEscape | Popup.CloseOnPressOutsideParent } else {
Action { menu.openMenu(chatsModel.activeChannel, chatsModel.getActiveChannelIdx(),
icon.source: "../../../img/group_chat.svg" moreActionsBtn.x - chatContextMenu.width + moreActionsBtn.width + 4,
icon.width: chatTopBarContent.iconSize moreActionsBtn.height - 4)
icon.height: chatTopBarContent.iconSize }
//% "Group Information" }
text: qsTrId("group-information")
onTriggered: openPopup(groupInfoPopupComponent, {channel: chatsModel.activeChannel}) ChannelContextMenu {
} id: chatContextMenu
Action { closePolicy: Popup.CloseOnEscape | Popup.CloseOnPressOutsideParent
icon.source: "../../../img/close.svg" }
icon.width: chatTopBarContent.iconSize
icon.height: chatTopBarContent.iconSize PopupMenu {
//% "Clear History" id: groupContextMenu
text: qsTrId("clear-history") closePolicy: Popup.CloseOnEscape | Popup.CloseOnPressOutsideParent
onTriggered: chatsModel.clearChatHistory(chatsModel.activeChannel.id) Action {
} icon.source: "../../../img/group_chat.svg"
Action { icon.width: chatTopBarContent.iconSize
icon.source: "../../../img/leave_chat.svg" icon.height: chatTopBarContent.iconSize
icon.width: chatTopBarContent.iconSize //% "Group Information"
icon.height: chatTopBarContent.iconSize text: qsTrId("group-information")
onTriggered: openPopup(groupInfoPopupComponent, {channel: chatsModel.activeChannel})
}
Action {
icon.source: "../../../img/close.svg"
icon.width: chatTopBarContent.iconSize
icon.height: chatTopBarContent.iconSize
//% "Clear History"
text: qsTrId("clear-history")
onTriggered: chatsModel.clearChatHistory(chatsModel.activeChannel.id)
}
Action {
icon.source: "../../../img/leave_chat.svg"
icon.width: chatTopBarContent.iconSize
icon.height: chatTopBarContent.iconSize
//% "Leave group"
text: qsTrId("leave-group")
onTriggered: {
//% "Leave group" //% "Leave group"
text: qsTrId("leave-group") deleteChatConfirmationDialog.title = qsTrId("leave-group")
onTriggered: { //% "Leave group"
//% "Leave group" deleteChatConfirmationDialog.confirmButtonLabel = qsTrId("leave-group")
deleteChatConfirmationDialog.title = qsTrId("leave-group") //% "Are you sure you want to leave this chat?"
//% "Leave group" deleteChatConfirmationDialog.confirmationText = qsTrId("are-you-sure-you-want-to-leave-this-chat-")
deleteChatConfirmationDialog.confirmButtonLabel = qsTrId("leave-group") deleteChatConfirmationDialog.open()
//% "Are you sure you want to leave this chat?"
deleteChatConfirmationDialog.confirmationText = qsTrId("are-you-sure-you-want-to-leave-this-chat-")
deleteChatConfirmationDialog.open()
}
} }
} }
} }
}
// Rectangle { // Rectangle {
// id: separator // id: separator
@ -180,7 +178,7 @@ Item {
// } // }
// } // }
// } // }
} // }
ActivityCenter { ActivityCenter {
id: activityCenter id: activityCenter

View File

@ -8,7 +8,7 @@ import "../components"
import "./" import "./"
Button { Button {
implicitWidth: Math.max(communityImage.width + communityName.width + Style.current.padding, 200) implicitWidth: Math.max(communityImage.width + communityName.width + (Style.current.halfPadding * 3), 200)
implicitHeight: communityImage.height + Style.current.padding implicitHeight: communityImage.height + Style.current.padding
background: Rectangle { background: Rectangle {

View File

@ -8,6 +8,8 @@ import "../../../../shared/status"
ModalPopup { ModalPopup {
property string communityId property string communityId
property QtObject channel
property bool isEdit: false
property string categoryId: "" property string categoryId: ""
readonly property int maxDescChars: 140 readonly property int maxDescChars: 140
property string nameValidationError: "" property string nameValidationError: ""
@ -16,10 +18,16 @@ ModalPopup {
descriptionTextArea.isValid descriptionTextArea.isValid
id: popup id: popup
height: 500 height: 400
onOpened: { onOpened: {
nameInput.text = ""; nameInput.text = "";
if (isEdit) {
nameInput.text = channel.name;
descriptionTextArea.text = channel.description;
// TODO: re-enable once private channels are implemented
// privateSwitch.checked = channel.private
}
nameInput.forceActiveFocus(Qt.MouseFocusReason) nameInput.forceActiveFocus(Qt.MouseFocusReason)
} }
onClosed: destroy() onClosed: destroy()
@ -59,7 +67,7 @@ ModalPopup {
placeholderText: qsTrId("a-cool-name") placeholderText: qsTrId("a-cool-name")
validationError: popup.nameValidationError validationError: popup.nameValidationError
property bool isValid: false property bool isValid: false || isEdit
onTextEdited: { onTextEdited: {
if (text.includes(" ")) { if (text.includes(" ")) {
@ -97,7 +105,7 @@ ModalPopup {
anchors.topMargin: Style.current.bigPadding anchors.topMargin: Style.current.bigPadding
customHeight: 88 customHeight: 88
property bool isValid: false property bool isValid: false || isEdit
onTextChanged: validate() onTextChanged: validate()
function resetValidation() { function resetValidation() {
@ -127,62 +135,80 @@ ModalPopup {
color: !descriptionTextArea.validationError ? Style.current.textColor : Style.current.danger color: !descriptionTextArea.validationError ? Style.current.textColor : Style.current.danger
} }
Separator { // Separator {
id: separator1 // id: separator1
anchors.top: charLimit.bottom // anchors.top: charLimit.bottom
anchors.topMargin: Style.current.bigPadding // anchors.topMargin: Style.current.bigPadding
} // }
Item { // TODO: use the switch below to enable private channels
id: privateSwitcher // Item {
height: privateSwitch.height // id: privateSwitcher
width: parent.width // height: privateSwitch.height
visible: false // width: parent.width
anchors.top: separator1.bottom // anchors.top: separator1.bottom
anchors.topMargin: Style.current.smallPadding * 2 // anchors.topMargin: Style.current.smallPadding * 2
StyledText { // StyledText {
//% "Private channel" // //% "Private channel"
text: qsTrId("private-channel") // text: qsTrId("private-channel")
anchors.verticalCenter: parent.verticalCenter // anchors.verticalCenter: parent.verticalCenter
} // }
StatusSwitch { // StatusSwitch {
id: privateSwitch // id: privateSwitch
anchors.right: parent.right // anchors.right: parent.right
} // }
} // }
StyledText { // StyledText {
id: privateExplanation // id: privateExplanation
anchors.top: separator1.bottom // anchors.top: privateSwitcher.bottom
color: Style.current.secondaryText // color: Style.current.secondaryText
wrapMode: Text.WordWrap // wrapMode: Text.WordWrap
anchors.topMargin: Style.current.smallPadding * 2 // anchors.topMargin: Style.current.smallPadding * 2
width: parent.width // width: parent.width
//% "By making a channel private, only members with selected permission will be able to access it" // //% "By making a channel private, only members with selected permission will be able to access it"
text: qsTrId("by-making-a-channel-private--only-members-with-selected-permission-will-be-able-to-access-it") // text: qsTrId("by-making-a-channel-private--only-members-with-selected-permission-will-be-able-to-access-it")
} // }
} }
} }
footer: StatusButton { footer: StatusButton {
enabled: popup.isValid enabled: popup.isValid
//% "Create" text: isEdit ?
text: qsTrId("create") qsTr("Save") :
//% "Create"
qsTrId("create")
anchors.right: parent.right anchors.right: parent.right
onClicked: { onClicked: {
if (!validate()) { if (!validate()) {
scrollView.scrollBackUp() scrollView.scrollBackUp()
return return
} }
const error = chatsModel.communities.createCommunityChannel(communityId, let error = "";
if (!isEdit) {
error = chatsModel.createCommunityChannel(communityId,
Utils.filterXSS(nameInput.text), Utils.filterXSS(nameInput.text),
Utils.filterXSS(descriptionTextArea.text), Utils.filterXSS(descriptionTextArea.text),
categoryId) categoryId)
// TODO: pass the private value when private channels
// are implemented
//privateSwitch.checked)
} else {
error = chatsModel.editCommunityChannel(communityId,
channel.id,
Utils.filterXSS(nameInput.text),
Utils.filterXSS(descriptionTextArea.text),
categoryId)
// TODO: pass the private value when private channels
// are implemented
//privateSwitch.checked)
}
if (error) { if (error) {
creatingError.text = error const errorJson = JSON.parse(error)
creatingError.text = errorJson.error
return creatingError.open() return creatingError.open()
} }

View File

@ -106,6 +106,14 @@ PopupMenu {
icon.height: 16 icon.height: 16
onTriggered: chatsModel.clearChatHistoryByIndex(channelContextMenu.channelIndex) onTriggered: chatsModel.clearChatHistoryByIndex(channelContextMenu.channelIndex)
} }
Action {
enabled: chatsModel.communities.activeCommunity.active && chatsModel.communities.activeCommunity.admin
text: qsTr("Edit Channel")
icon.source: "../../../img/edit.svg"
icon.width: 16
icon.height: 16
onTriggered: openPopup(editChannelPopup, {communityId: chatsModel.communities.activeCommunity.id, channel: chatsModel.activeChannel})
}
Separator { Separator {
visible: deleteAction.enabled visible: deleteAction.enabled

View File

@ -280,6 +280,16 @@ RowLayout {
} }
} }
Component {
id: editChannelPopup
CreateChannelPopup {
isEdit: true
onClosed: {
destroy()
}
}
}
ToastMessage { ToastMessage {
id: toastMessage id: toastMessage
} }

View File

@ -26,7 +26,6 @@ Item {
property string profileImage: realChatType === Constants.chatTypeOneToOne ? appMain.getProfileImage(chatId) || "" : "" property string profileImage: realChatType === Constants.chatTypeOneToOne ? appMain.getProfileImage(chatId) || "" : ""
height: 48 height: 48
width: nameAndInfo.width + chatIdenticon.width + Style.current.smallPadding
Connections { Connections {
enabled: realChatType === Constants.chatTypeOneToOne enabled: realChatType === Constants.chatTypeOneToOne
@ -52,7 +51,7 @@ Item {
Item { Item {
id: nameAndInfo id: nameAndInfo
height: chatName.height + chatInfo.height height: chatName.height + chatInfo.height
width: childrenRect.width width: parent.width
anchors.verticalCenter: parent.verticalCenter anchors.verticalCenter: parent.verticalCenter
anchors.left: chatIdenticon.right anchors.left: chatIdenticon.right
anchors.leftMargin: Style.current.smallPadding anchors.leftMargin: Style.current.smallPadding
@ -116,7 +115,13 @@ Item {
StyledText { StyledText {
id: chatInfo id: chatInfo
color: Style.current.secondaryText color: Style.current.secondaryText
wrapMode: Text.Wrap
width: parent.width
elide: Text.ElideRight
text: { text: {
if (root.chatType === Constants.chatTypeCommunity) {
return chatsModel.activeChannel.description
}
switch(root.realChatType){ switch(root.realChatType){
//% "Public chat" //% "Public chat"
case Constants.chatTypePublic: return qsTrId("public-chat") case Constants.chatTypePublic: return qsTrId("public-chat")
@ -139,6 +144,7 @@ Item {
} }
font.pixelSize: 12 font.pixelSize: 12
anchors.top: chatName.bottom anchors.top: chatName.bottom
anchors.topMargin: 2
} }
Item { Item {

2
vendor/status-go vendored

@ -1 +1 @@
Subproject commit 10becb490228d416ff361bdb8a965537b4803af5 Subproject commit b395144704f29c9e1f4fd11714d1031a10160ad1