Merge pull request #595 from LorenDB/qml-all-the-things
QML all the things, part 1
This commit is contained in:
commit
8091e25913
@ -281,11 +281,9 @@ set(SRC_FILES
|
|||||||
src/dialogs/CreateRoom.cpp
|
src/dialogs/CreateRoom.cpp
|
||||||
src/dialogs/FallbackAuth.cpp
|
src/dialogs/FallbackAuth.cpp
|
||||||
src/dialogs/ImageOverlay.cpp
|
src/dialogs/ImageOverlay.cpp
|
||||||
src/dialogs/InviteUsers.cpp
|
|
||||||
src/dialogs/JoinRoom.cpp
|
src/dialogs/JoinRoom.cpp
|
||||||
src/dialogs/LeaveRoom.cpp
|
src/dialogs/LeaveRoom.cpp
|
||||||
src/dialogs/Logout.cpp
|
src/dialogs/Logout.cpp
|
||||||
src/dialogs/MemberList.cpp
|
|
||||||
src/dialogs/PreviewUploadOverlay.cpp
|
src/dialogs/PreviewUploadOverlay.cpp
|
||||||
src/dialogs/ReCaptcha.cpp
|
src/dialogs/ReCaptcha.cpp
|
||||||
src/dialogs/ReadReceipts.cpp
|
src/dialogs/ReadReceipts.cpp
|
||||||
@ -346,11 +344,12 @@ set(SRC_FILES
|
|||||||
src/CompletionProxyModel.cpp
|
src/CompletionProxyModel.cpp
|
||||||
src/DeviceVerificationFlow.cpp
|
src/DeviceVerificationFlow.cpp
|
||||||
src/EventAccessors.cpp
|
src/EventAccessors.cpp
|
||||||
src/InviteeItem.cpp
|
src/InviteesModel.cpp
|
||||||
src/Logging.cpp
|
src/Logging.cpp
|
||||||
src/LoginPage.cpp
|
src/LoginPage.cpp
|
||||||
src/MainWindow.cpp
|
src/MainWindow.cpp
|
||||||
src/MatrixClient.cpp
|
src/MatrixClient.cpp
|
||||||
|
src/MemberList.cpp
|
||||||
src/MxcImageProvider.cpp
|
src/MxcImageProvider.cpp
|
||||||
src/Olm.cpp
|
src/Olm.cpp
|
||||||
src/RegisterPage.cpp
|
src/RegisterPage.cpp
|
||||||
@ -492,11 +491,9 @@ qt5_wrap_cpp(MOC_HEADERS
|
|||||||
src/dialogs/CreateRoom.h
|
src/dialogs/CreateRoom.h
|
||||||
src/dialogs/FallbackAuth.h
|
src/dialogs/FallbackAuth.h
|
||||||
src/dialogs/ImageOverlay.h
|
src/dialogs/ImageOverlay.h
|
||||||
src/dialogs/InviteUsers.h
|
|
||||||
src/dialogs/JoinRoom.h
|
src/dialogs/JoinRoom.h
|
||||||
src/dialogs/LeaveRoom.h
|
src/dialogs/LeaveRoom.h
|
||||||
src/dialogs/Logout.h
|
src/dialogs/Logout.h
|
||||||
src/dialogs/MemberList.h
|
|
||||||
src/dialogs/PreviewUploadOverlay.h
|
src/dialogs/PreviewUploadOverlay.h
|
||||||
src/dialogs/RawMessage.h
|
src/dialogs/RawMessage.h
|
||||||
src/dialogs/ReCaptcha.h
|
src/dialogs/ReCaptcha.h
|
||||||
@ -554,9 +551,10 @@ qt5_wrap_cpp(MOC_HEADERS
|
|||||||
src/Clipboard.h
|
src/Clipboard.h
|
||||||
src/CompletionProxyModel.h
|
src/CompletionProxyModel.h
|
||||||
src/DeviceVerificationFlow.h
|
src/DeviceVerificationFlow.h
|
||||||
src/InviteeItem.h
|
src/InviteesModel.h
|
||||||
src/LoginPage.h
|
src/LoginPage.h
|
||||||
src/MainWindow.h
|
src/MainWindow.h
|
||||||
|
src/MemberList.h
|
||||||
src/MxcImageProvider.h
|
src/MxcImageProvider.h
|
||||||
src/RegisterPage.h
|
src/RegisterPage.h
|
||||||
src/SSOHandler.h
|
src/SSOHandler.h
|
||||||
|
@ -70,7 +70,7 @@ Popup {
|
|||||||
onCompleterNameChanged: {
|
onCompleterNameChanged: {
|
||||||
if (completerName) {
|
if (completerName) {
|
||||||
if (completerName == "user")
|
if (completerName == "user")
|
||||||
completer = TimelineManager.completerFor(completerName, room.roomId());
|
completer = TimelineManager.completerFor(completerName, room.roomId);
|
||||||
else
|
else
|
||||||
completer = TimelineManager.completerFor(completerName);
|
completer = TimelineManager.completerFor(completerName);
|
||||||
completer.setSearchString("");
|
completer.setSearchString("");
|
||||||
|
158
resources/qml/InviteDialog.qml
Normal file
158
resources/qml/InviteDialog.qml
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
import QtQuick 2.12
|
||||||
|
import QtQuick.Controls 2.12
|
||||||
|
import QtQuick.Layouts 1.12
|
||||||
|
import im.nheko 1.0
|
||||||
|
|
||||||
|
ApplicationWindow {
|
||||||
|
id: inviteDialogRoot
|
||||||
|
|
||||||
|
property string roomId
|
||||||
|
property string plainRoomName
|
||||||
|
property InviteesModel invitees
|
||||||
|
|
||||||
|
function addInvite() {
|
||||||
|
if (inviteeEntry.isValidMxid) {
|
||||||
|
invitees.addUser(inviteeEntry.text);
|
||||||
|
inviteeEntry.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function cleanUpAndClose() {
|
||||||
|
if (inviteeEntry.isValidMxid)
|
||||||
|
addInvite();
|
||||||
|
|
||||||
|
invitees.accept();
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
|
||||||
|
title: qsTr("Invite users to ") + plainRoomName
|
||||||
|
x: MainWindow.x + (MainWindow.width / 2) - (width / 2)
|
||||||
|
y: MainWindow.y + (MainWindow.height / 2) - (height / 2)
|
||||||
|
height: 380
|
||||||
|
width: 340
|
||||||
|
palette: Nheko.colors
|
||||||
|
color: Nheko.colors.window
|
||||||
|
|
||||||
|
Shortcut {
|
||||||
|
sequence: "Ctrl+Enter"
|
||||||
|
onActivated: cleanUpAndClose()
|
||||||
|
}
|
||||||
|
|
||||||
|
Shortcut {
|
||||||
|
sequence: StandardKey.Cancel
|
||||||
|
onActivated: inviteDialogRoot.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
anchors.margins: Nheko.paddingMedium
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: qsTr("User ID to invite")
|
||||||
|
Layout.fillWidth: true
|
||||||
|
}
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
|
||||||
|
MatrixTextField {
|
||||||
|
id: inviteeEntry
|
||||||
|
|
||||||
|
property bool isValidMxid: text.match("@.+?:.{3,}")
|
||||||
|
|
||||||
|
backgroundColor: Nheko.colors.window
|
||||||
|
placeholderText: qsTr("@joe:matrix.org", "Example user id. The name 'joe' can be localized however you want.")
|
||||||
|
Layout.fillWidth: true
|
||||||
|
onAccepted: {
|
||||||
|
if (isValidMxid)
|
||||||
|
addInvite();
|
||||||
|
|
||||||
|
}
|
||||||
|
Component.onCompleted: forceActiveFocus()
|
||||||
|
Keys.onShortcutOverride: event.accepted = ((event.key === Qt.Key_Return || event.key === Qt.Key_Enter) && (event.modifiers & Qt.ControlModifier))
|
||||||
|
Keys.onPressed: {
|
||||||
|
if ((event.key === Qt.Key_Return || event.key === Qt.Key_Enter) && (event.modifiers === Qt.ControlModifier))
|
||||||
|
cleanUpAndClose();
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Button {
|
||||||
|
text: qsTr("Add")
|
||||||
|
enabled: inviteeEntry.isValidMxid
|
||||||
|
onClicked: addInvite()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
ListView {
|
||||||
|
id: inviteesList
|
||||||
|
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.fillHeight: true
|
||||||
|
model: invitees
|
||||||
|
|
||||||
|
delegate: RowLayout {
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
|
||||||
|
Avatar {
|
||||||
|
width: Nheko.avatarSize
|
||||||
|
height: Nheko.avatarSize
|
||||||
|
userid: model.mxid
|
||||||
|
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
|
||||||
|
displayName: model.displayName
|
||||||
|
onClicked: Rooms.currentRoom.openUserProfile(model.mxid)
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
spacing: Nheko.paddingSmall
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: model.displayName
|
||||||
|
color: TimelineManager.userColor(model ? model.mxid : "", Nheko.colors.window)
|
||||||
|
font.pointSize: fontMetrics.font.pointSize
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: model.mxid
|
||||||
|
color: Nheko.colors.buttonText
|
||||||
|
font.pointSize: fontMetrics.font.pointSize * 0.9
|
||||||
|
}
|
||||||
|
|
||||||
|
Item {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
footer: DialogButtonBox {
|
||||||
|
id: buttons
|
||||||
|
|
||||||
|
Button {
|
||||||
|
text: qsTr("Invite")
|
||||||
|
DialogButtonBox.buttonRole: DialogButtonBox.AcceptRole
|
||||||
|
enabled: invitees.count > 0
|
||||||
|
onClicked: cleanUpAndClose()
|
||||||
|
}
|
||||||
|
|
||||||
|
Button {
|
||||||
|
text: qsTr("Cancel")
|
||||||
|
DialogButtonBox.buttonRole: DialogButtonBox.DestructiveRole
|
||||||
|
onClicked: inviteDialogRoot.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -10,6 +10,8 @@ import im.nheko 1.0
|
|||||||
TextField {
|
TextField {
|
||||||
id: input
|
id: input
|
||||||
|
|
||||||
|
property alias backgroundColor: backgroundRect.color
|
||||||
|
|
||||||
palette: Nheko.colors
|
palette: Nheko.colors
|
||||||
color: Nheko.colors.text
|
color: Nheko.colors.text
|
||||||
|
|
||||||
@ -62,6 +64,8 @@ TextField {
|
|||||||
}
|
}
|
||||||
|
|
||||||
background: Rectangle {
|
background: Rectangle {
|
||||||
|
id: backgroundRect
|
||||||
|
|
||||||
color: Nheko.colors.base
|
color: Nheko.colors.base
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -331,7 +331,7 @@ Rectangle {
|
|||||||
image: ":/icons/icons/ui/sticky-note-solid.svg"
|
image: ":/icons/icons/ui/sticky-note-solid.svg"
|
||||||
ToolTip.visible: hovered
|
ToolTip.visible: hovered
|
||||||
ToolTip.text: qsTr("Stickers")
|
ToolTip.text: qsTr("Stickers")
|
||||||
onClicked: stickerPopup.visible ? stickerPopup.close() : stickerPopup.show(stickerButton, room.roomId(), function(row) {
|
onClicked: stickerPopup.visible ? stickerPopup.close() : stickerPopup.show(stickerButton, room.roomId, function(row) {
|
||||||
room.input.sticker(stickerPopup.model.sourceModel, row);
|
room.input.sticker(stickerPopup.model.sourceModel, row);
|
||||||
TimelineManager.focusMessageInput();
|
TimelineManager.focusMessageInput();
|
||||||
})
|
})
|
||||||
|
@ -33,8 +33,8 @@ Page {
|
|||||||
|
|
||||||
Connections {
|
Connections {
|
||||||
onActiveTimelineChanged: {
|
onActiveTimelineChanged: {
|
||||||
roomlist.positionViewAtIndex(Rooms.roomidToIndex(Rooms.currentRoom.roomId()), ListView.Contain);
|
roomlist.positionViewAtIndex(Rooms.roomidToIndex(Rooms.currentRoom.roomId), ListView.Contain);
|
||||||
console.log("Test" + Rooms.currentRoom.roomId() + " " + Rooms.roomidToIndex(Rooms.currentRoom.roomId()));
|
console.log("Test" + Rooms.currentRoom.roomId + " " + Rooms.roomidToIndex(Rooms.currentRoom.roomId));
|
||||||
}
|
}
|
||||||
target: TimelineManager
|
target: TimelineManager
|
||||||
}
|
}
|
||||||
@ -133,7 +133,7 @@ Page {
|
|||||||
states: [
|
states: [
|
||||||
State {
|
State {
|
||||||
name: "highlight"
|
name: "highlight"
|
||||||
when: hovered.hovered && !((Rooms.currentRoom && roomId == Rooms.currentRoom.roomId()) || Rooms.currentRoomPreview.roomid == roomId)
|
when: hovered.hovered && !((Rooms.currentRoom && roomId == Rooms.currentRoom.roomId) || Rooms.currentRoomPreview.roomid == roomId)
|
||||||
|
|
||||||
PropertyChanges {
|
PropertyChanges {
|
||||||
target: roomItem
|
target: roomItem
|
||||||
@ -147,7 +147,7 @@ Page {
|
|||||||
},
|
},
|
||||||
State {
|
State {
|
||||||
name: "selected"
|
name: "selected"
|
||||||
when: (Rooms.currentRoom && roomId == Rooms.currentRoom.roomId()) || Rooms.currentRoomPreview.roomid == roomId
|
when: (Rooms.currentRoom && roomId == Rooms.currentRoom.roomId) || Rooms.currentRoomPreview.roomid == roomId
|
||||||
|
|
||||||
PropertyChanges {
|
PropertyChanges {
|
||||||
target: roomItem
|
target: roomItem
|
||||||
|
149
resources/qml/RoomMembers.qml
Normal file
149
resources/qml/RoomMembers.qml
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
import "./ui"
|
||||||
|
import QtQuick 2.12
|
||||||
|
import QtQuick.Controls 2.12
|
||||||
|
import QtQuick.Layouts 1.12
|
||||||
|
import QtQuick.Window 2.12
|
||||||
|
import im.nheko 1.0
|
||||||
|
|
||||||
|
ApplicationWindow {
|
||||||
|
id: roomMembersRoot
|
||||||
|
|
||||||
|
property MemberList members
|
||||||
|
|
||||||
|
title: qsTr("Members of ") + members.roomName
|
||||||
|
x: MainWindow.x + (MainWindow.width / 2) - (width / 2)
|
||||||
|
y: MainWindow.y + (MainWindow.height / 2) - (height / 2)
|
||||||
|
height: 650
|
||||||
|
width: 420
|
||||||
|
minimumHeight: 420
|
||||||
|
palette: Nheko.colors
|
||||||
|
color: Nheko.colors.window
|
||||||
|
|
||||||
|
Shortcut {
|
||||||
|
sequence: StandardKey.Cancel
|
||||||
|
onActivated: roomMembersRoot.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
anchors.margins: Nheko.paddingMedium
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
|
||||||
|
Avatar {
|
||||||
|
id: roomAvatar
|
||||||
|
|
||||||
|
width: 130
|
||||||
|
height: width
|
||||||
|
displayName: members.roomName
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
url: members.avatarUrl.replace("mxc://", "image://MxcImage/")
|
||||||
|
onClicked: Rooms.currentRoom.openRoomSettings(members.roomId)
|
||||||
|
}
|
||||||
|
|
||||||
|
ElidedLabel {
|
||||||
|
font.pixelSize: fontMetrics.font.pixelSize * 2
|
||||||
|
fullText: members.memberCount + (members.memberCount === 1 ? qsTr(" person in ") : qsTr(" people in ")) + members.roomName
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
elideWidth: parent.width - Nheko.paddingMedium
|
||||||
|
}
|
||||||
|
|
||||||
|
ImageButton {
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
image: ":/icons/icons/ui/add-square-button.png"
|
||||||
|
hoverEnabled: true
|
||||||
|
ToolTip.visible: hovered
|
||||||
|
ToolTip.text: qsTr("Invite more people")
|
||||||
|
onClicked: Rooms.currentRoom.openInviteUsers()
|
||||||
|
}
|
||||||
|
|
||||||
|
ScrollView {
|
||||||
|
palette: Nheko.colors
|
||||||
|
padding: Nheko.paddingMedium
|
||||||
|
ScrollBar.horizontal.visible: false
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.minimumHeight: 200
|
||||||
|
Layout.fillWidth: true
|
||||||
|
|
||||||
|
ListView {
|
||||||
|
id: memberList
|
||||||
|
|
||||||
|
clip: true
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
boundsBehavior: Flickable.StopAtBounds
|
||||||
|
model: members
|
||||||
|
|
||||||
|
ScrollHelper {
|
||||||
|
flickable: parent
|
||||||
|
anchors.fill: parent
|
||||||
|
enabled: !Settings.mobileMode
|
||||||
|
}
|
||||||
|
|
||||||
|
delegate: RowLayout {
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
|
||||||
|
Avatar {
|
||||||
|
width: Nheko.avatarSize
|
||||||
|
height: Nheko.avatarSize
|
||||||
|
userid: model.mxid
|
||||||
|
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
|
||||||
|
displayName: model.displayName
|
||||||
|
onClicked: Rooms.currentRoom.openUserProfile(model.mxid)
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
spacing: Nheko.paddingSmall
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: model.displayName
|
||||||
|
color: TimelineManager.userColor(model ? model.mxid : "", Nheko.colors.window)
|
||||||
|
font.pointSize: fontMetrics.font.pointSize
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: model.mxid
|
||||||
|
color: Nheko.colors.buttonText
|
||||||
|
font.pointSize: fontMetrics.font.pointSize * 0.9
|
||||||
|
}
|
||||||
|
|
||||||
|
Item {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
footer: Item {
|
||||||
|
width: parent.width
|
||||||
|
visible: (members.numUsersLoaded < members.memberCount) && members.loadingMoreMembers
|
||||||
|
|
||||||
|
// use the default height if it's visible, otherwise no height at all
|
||||||
|
height: membersLoadingSpinner.height
|
||||||
|
anchors.margins: Nheko.paddingMedium
|
||||||
|
|
||||||
|
Spinner {
|
||||||
|
id: membersLoadingSpinner
|
||||||
|
|
||||||
|
anchors.centerIn: parent
|
||||||
|
height: visible ? 35 : 0
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
footer: DialogButtonBox {
|
||||||
|
standardButtons: DialogButtonBox.Ok
|
||||||
|
onAccepted: roomMembersRoot.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -98,7 +98,7 @@ ApplicationWindow {
|
|||||||
|
|
||||||
MatrixText {
|
MatrixText {
|
||||||
text: roomSettings.roomName
|
text: roomSettings.roomName
|
||||||
font.pixelSize: 24
|
font.pixelSize: fontMetrics.font.pixelSize * 2
|
||||||
Layout.alignment: Qt.AlignHCenter
|
Layout.alignment: Qt.AlignHCenter
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -264,7 +264,7 @@ ApplicationWindow {
|
|||||||
|
|
||||||
MatrixText {
|
MatrixText {
|
||||||
text: roomSettings.roomId
|
text: roomSettings.roomId
|
||||||
font.pixelSize: 14
|
font.pixelSize: fontMetrics.font.pixelSize * 1.2
|
||||||
Layout.alignment: Qt.AlignRight
|
Layout.alignment: Qt.AlignRight
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -274,16 +274,16 @@ ApplicationWindow {
|
|||||||
|
|
||||||
MatrixText {
|
MatrixText {
|
||||||
text: roomSettings.roomVersion
|
text: roomSettings.roomVersion
|
||||||
font.pixelSize: 14
|
font.pixelSize: fontMetrics.font.pixelSize * 1.2
|
||||||
Layout.alignment: Qt.AlignRight
|
Layout.alignment: Qt.AlignRight
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Button {
|
DialogButtonBox {
|
||||||
Layout.alignment: Qt.AlignRight
|
Layout.fillWidth: true
|
||||||
text: qsTr("OK")
|
standardButtons: DialogButtonBox.Ok
|
||||||
onClicked: close()
|
onAccepted: close()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -47,6 +47,14 @@ Page {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: roomMembersComponent
|
||||||
|
|
||||||
|
RoomMembers {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
Component {
|
Component {
|
||||||
id: mobileCallInviteDialog
|
id: mobileCallInviteDialog
|
||||||
|
|
||||||
@ -63,6 +71,22 @@ Page {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: deviceVerificationDialog
|
||||||
|
|
||||||
|
DeviceVerification {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: inviteDialog
|
||||||
|
|
||||||
|
InviteDialog {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
Shortcut {
|
Shortcut {
|
||||||
sequence: "Ctrl+K"
|
sequence: "Ctrl+K"
|
||||||
onActivated: {
|
onActivated: {
|
||||||
@ -82,14 +106,6 @@ Page {
|
|||||||
onActivated: Rooms.previousRoom()
|
onActivated: Rooms.previousRoom()
|
||||||
}
|
}
|
||||||
|
|
||||||
Component {
|
|
||||||
id: deviceVerificationDialog
|
|
||||||
|
|
||||||
DeviceVerification {
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
Connections {
|
Connections {
|
||||||
target: TimelineManager
|
target: TimelineManager
|
||||||
onNewDeviceVerificationRequest: {
|
onNewDeviceVerificationRequest: {
|
||||||
@ -116,6 +132,31 @@ Page {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
target: Rooms.currentRoom
|
||||||
|
onOpenRoomMembersDialog: {
|
||||||
|
var membersDialog = roomMembersComponent.createObject(timelineRoot, {
|
||||||
|
"members": members,
|
||||||
|
"roomName": Rooms.currentRoom.roomName
|
||||||
|
});
|
||||||
|
membersDialog.show();
|
||||||
|
}
|
||||||
|
onOpenRoomSettingsDialog: {
|
||||||
|
var roomSettings = roomSettingsComponent.createObject(timelineRoot, {
|
||||||
|
"roomSettings": settings
|
||||||
|
});
|
||||||
|
roomSettings.show();
|
||||||
|
}
|
||||||
|
onOpenInviteUsersDialog: {
|
||||||
|
var dialog = inviteDialog.createObject(timelineRoot, {
|
||||||
|
"roomId": Rooms.currentRoom.roomId,
|
||||||
|
"plainRoomName": Rooms.currentRoom.plainRoomName,
|
||||||
|
"invitees": invitees
|
||||||
|
});
|
||||||
|
dialog.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ChatPage {
|
ChatPage {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
}
|
}
|
||||||
|
@ -246,17 +246,7 @@ Item {
|
|||||||
|
|
||||||
NhekoDropArea {
|
NhekoDropArea {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
roomid: room ? room.roomId() : ""
|
roomid: room ? room.roomId : ""
|
||||||
}
|
|
||||||
|
|
||||||
Connections {
|
|
||||||
target: room
|
|
||||||
onOpenRoomSettingsDialog: {
|
|
||||||
var roomSettings = roomSettingsComponent.createObject(timelineRoot, {
|
|
||||||
"roomSettings": settings
|
|
||||||
});
|
|
||||||
roomSettings.show();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -111,17 +111,17 @@ Rectangle {
|
|||||||
Platform.MenuItem {
|
Platform.MenuItem {
|
||||||
visible: room ? room.permissions.canInvite() : false
|
visible: room ? room.permissions.canInvite() : false
|
||||||
text: qsTr("Invite users")
|
text: qsTr("Invite users")
|
||||||
onTriggered: TimelineManager.openInviteUsersDialog()
|
onTriggered: Rooms.currentRoom.openInviteUsers()
|
||||||
}
|
}
|
||||||
|
|
||||||
Platform.MenuItem {
|
Platform.MenuItem {
|
||||||
text: qsTr("Members")
|
text: qsTr("Members")
|
||||||
onTriggered: TimelineManager.openMemberListDialog(room.roomId())
|
onTriggered: Rooms.currentRoom.openRoomMembers()
|
||||||
}
|
}
|
||||||
|
|
||||||
Platform.MenuItem {
|
Platform.MenuItem {
|
||||||
text: qsTr("Leave room")
|
text: qsTr("Leave room")
|
||||||
onTriggered: TimelineManager.openLeaveRoomDialog(room.roomId())
|
onTriggered: TimelineManager.openLeaveRoomDialog(room.roomId)
|
||||||
}
|
}
|
||||||
|
|
||||||
Platform.MenuItem {
|
Platform.MenuItem {
|
||||||
|
@ -232,7 +232,7 @@ Item {
|
|||||||
body: formatted
|
body: formatted
|
||||||
isOnlyEmoji: false
|
isOnlyEmoji: false
|
||||||
isReply: d.isReply
|
isReply: d.isReply
|
||||||
formatted: qsTr("%1 created and configured room: %2").arg(d.userName).arg(room.roomId())
|
formatted: qsTr("%1 created and configured room: %2").arg(d.userName).arg(room.roomId)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -88,7 +88,7 @@ Popup {
|
|||||||
onClicked: {
|
onClicked: {
|
||||||
if (buttonLayout.validateMic()) {
|
if (buttonLayout.validateMic()) {
|
||||||
Settings.microphone = micCombo.currentText;
|
Settings.microphone = micCombo.currentText;
|
||||||
CallManager.sendInvite(room.roomId(), CallType.VOICE);
|
CallManager.sendInvite(room.roomId, CallType.VOICE);
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -102,7 +102,7 @@ Popup {
|
|||||||
if (buttonLayout.validateMic()) {
|
if (buttonLayout.validateMic()) {
|
||||||
Settings.microphone = micCombo.currentText;
|
Settings.microphone = micCombo.currentText;
|
||||||
Settings.camera = cameraCombo.currentText;
|
Settings.camera = cameraCombo.currentText;
|
||||||
CallManager.sendInvite(room.roomId(), CallType.VIDEO);
|
CallManager.sendInvite(room.roomId, CallType.VIDEO);
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -136,7 +136,7 @@ Popup {
|
|||||||
Settings.screenSharePiP = pipCheckBox.checked;
|
Settings.screenSharePiP = pipCheckBox.checked;
|
||||||
Settings.screenShareRemoteVideo = remoteVideoCheckBox.checked;
|
Settings.screenShareRemoteVideo = remoteVideoCheckBox.checked;
|
||||||
Settings.screenShareHideCursor = hideCursorCheckBox.checked;
|
Settings.screenShareHideCursor = hideCursorCheckBox.checked;
|
||||||
CallManager.sendInvite(room.roomId(), CallType.SCREEN, windowCombo.currentIndex);
|
CallManager.sendInvite(room.roomId, CallType.SCREEN, windowCombo.currentIndex);
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,7 +9,6 @@
|
|||||||
<file>icons/ui/do-not-disturb-rounded-sign@2x.png</file>
|
<file>icons/ui/do-not-disturb-rounded-sign@2x.png</file>
|
||||||
<file>icons/ui/round-remove-button.png</file>
|
<file>icons/ui/round-remove-button.png</file>
|
||||||
<file>icons/ui/round-remove-button@2x.png</file>
|
<file>icons/ui/round-remove-button@2x.png</file>
|
||||||
|
|
||||||
<file>icons/ui/double-tick-indicator.png</file>
|
<file>icons/ui/double-tick-indicator.png</file>
|
||||||
<file>icons/ui/double-tick-indicator@2x.png</file>
|
<file>icons/ui/double-tick-indicator@2x.png</file>
|
||||||
<file>icons/ui/lock.png</file>
|
<file>icons/ui/lock.png</file>
|
||||||
@ -55,22 +54,17 @@
|
|||||||
<file>icons/ui/pause-symbol@2x.png</file>
|
<file>icons/ui/pause-symbol@2x.png</file>
|
||||||
<file>icons/ui/remove-symbol.png</file>
|
<file>icons/ui/remove-symbol.png</file>
|
||||||
<file>icons/ui/remove-symbol@2x.png</file>
|
<file>icons/ui/remove-symbol@2x.png</file>
|
||||||
|
|
||||||
<file>icons/ui/world.png</file>
|
<file>icons/ui/world.png</file>
|
||||||
<file>icons/ui/world@2x.png</file>
|
<file>icons/ui/world@2x.png</file>
|
||||||
|
|
||||||
<file>icons/ui/tag.png</file>
|
<file>icons/ui/tag.png</file>
|
||||||
<file>icons/ui/tag@2x.png</file>
|
<file>icons/ui/tag@2x.png</file>
|
||||||
<file>icons/ui/star.png</file>
|
<file>icons/ui/star.png</file>
|
||||||
<file>icons/ui/star@2x.png</file>
|
<file>icons/ui/star@2x.png</file>
|
||||||
<file>icons/ui/lowprio.png</file>
|
<file>icons/ui/lowprio.png</file>
|
||||||
<file>icons/ui/lowprio@2x.png</file>
|
<file>icons/ui/lowprio@2x.png</file>
|
||||||
|
|
||||||
<file>icons/ui/edit.png</file>
|
<file>icons/ui/edit.png</file>
|
||||||
<file>icons/ui/edit@2x.png</file>
|
<file>icons/ui/edit@2x.png</file>
|
||||||
|
|
||||||
<file>icons/ui/mail-reply.png</file>
|
<file>icons/ui/mail-reply.png</file>
|
||||||
|
|
||||||
<file>icons/ui/place-call.png</file>
|
<file>icons/ui/place-call.png</file>
|
||||||
<file>icons/ui/end-call.png</file>
|
<file>icons/ui/end-call.png</file>
|
||||||
<file>icons/ui/microphone-mute.png</file>
|
<file>icons/ui/microphone-mute.png</file>
|
||||||
@ -78,7 +72,6 @@
|
|||||||
<file>icons/ui/screen-share.png</file>
|
<file>icons/ui/screen-share.png</file>
|
||||||
<file>icons/ui/toggle-camera-view.png</file>
|
<file>icons/ui/toggle-camera-view.png</file>
|
||||||
<file>icons/ui/video-call.png</file>
|
<file>icons/ui/video-call.png</file>
|
||||||
|
|
||||||
<file>icons/emoji-categories/people.png</file>
|
<file>icons/emoji-categories/people.png</file>
|
||||||
<file>icons/emoji-categories/people@2x.png</file>
|
<file>icons/emoji-categories/people@2x.png</file>
|
||||||
<file>icons/emoji-categories/nature.png</file>
|
<file>icons/emoji-categories/nature.png</file>
|
||||||
@ -99,16 +92,12 @@
|
|||||||
<qresource prefix="/logos">
|
<qresource prefix="/logos">
|
||||||
<file>nheko.png</file>
|
<file>nheko.png</file>
|
||||||
<file>nheko.svg</file>
|
<file>nheko.svg</file>
|
||||||
|
|
||||||
<file>splash.png</file>
|
<file>splash.png</file>
|
||||||
<file>splash@2x.png</file>
|
<file>splash@2x.png</file>
|
||||||
|
|
||||||
<file>register.png</file>
|
<file>register.png</file>
|
||||||
<file>register@2x.png</file>
|
<file>register@2x.png</file>
|
||||||
|
|
||||||
<file>login.png</file>
|
<file>login.png</file>
|
||||||
<file>login@2x.png</file>
|
<file>login@2x.png</file>
|
||||||
|
|
||||||
<file>nheko-512.png</file>
|
<file>nheko-512.png</file>
|
||||||
<file>nheko-256.png</file>
|
<file>nheko-256.png</file>
|
||||||
<file>nheko-128.png</file>
|
<file>nheko-128.png</file>
|
||||||
@ -185,6 +174,8 @@
|
|||||||
<file>qml/components/AdaptiveLayout.qml</file>
|
<file>qml/components/AdaptiveLayout.qml</file>
|
||||||
<file>qml/components/AdaptiveLayoutElement.qml</file>
|
<file>qml/components/AdaptiveLayoutElement.qml</file>
|
||||||
<file>qml/components/FlatButton.qml</file>
|
<file>qml/components/FlatButton.qml</file>
|
||||||
|
<file>qml/RoomMembers.qml</file>
|
||||||
|
<file>qml/InviteDialog.qml</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
<qresource prefix="/media">
|
<qresource prefix="/media">
|
||||||
<file>media/ring.ogg</file>
|
<file>media/ring.ogg</file>
|
||||||
|
@ -116,29 +116,31 @@ ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
|||||||
|
|
||||||
connect(this, &ChatPage::loggedOut, this, &ChatPage::logout);
|
connect(this, &ChatPage::loggedOut, this, &ChatPage::logout);
|
||||||
|
|
||||||
connect(view_manager_, &TimelineViewManager::inviteUsers, this, [this](QStringList users) {
|
connect(
|
||||||
const auto room_id = currentRoom().toStdString();
|
view_manager_,
|
||||||
|
&TimelineViewManager::inviteUsers,
|
||||||
|
this,
|
||||||
|
[this](QString roomId, QStringList users) {
|
||||||
|
for (int ii = 0; ii < users.size(); ++ii) {
|
||||||
|
QTimer::singleShot(ii * 500, this, [this, roomId, ii, users]() {
|
||||||
|
const auto user = users.at(ii);
|
||||||
|
|
||||||
for (int ii = 0; ii < users.size(); ++ii) {
|
http::client()->invite_user(
|
||||||
QTimer::singleShot(ii * 500, this, [this, room_id, ii, users]() {
|
roomId.toStdString(),
|
||||||
const auto user = users.at(ii);
|
user.toStdString(),
|
||||||
|
[this, user](const mtx::responses::RoomInvite &,
|
||||||
|
mtx::http::RequestErr err) {
|
||||||
|
if (err) {
|
||||||
|
emit showNotification(
|
||||||
|
tr("Failed to invite user: %1").arg(user));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
http::client()->invite_user(
|
emit showNotification(tr("Invited user: %1").arg(user));
|
||||||
room_id,
|
});
|
||||||
user.toStdString(),
|
});
|
||||||
[this, user](const mtx::responses::RoomInvite &,
|
}
|
||||||
mtx::http::RequestErr err) {
|
});
|
||||||
if (err) {
|
|
||||||
emit showNotification(
|
|
||||||
tr("Failed to invite user: %1").arg(user));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
emit showNotification(tr("Invited user: %1").arg(user));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
connect(this, &ChatPage::leftRoom, this, &ChatPage::removeRoom);
|
connect(this, &ChatPage::leftRoom, this, &ChatPage::removeRoom);
|
||||||
connect(this, &ChatPage::newRoom, this, &ChatPage::changeRoom, Qt::QueuedConnection);
|
connect(this, &ChatPage::newRoom, this, &ChatPage::changeRoom, Qt::QueuedConnection);
|
||||||
|
@ -1,28 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
#include <QHBoxLayout>
|
|
||||||
#include <QLabel>
|
|
||||||
#include <QPushButton>
|
|
||||||
|
|
||||||
#include "InviteeItem.h"
|
|
||||||
|
|
||||||
constexpr int SidePadding = 10;
|
|
||||||
|
|
||||||
InviteeItem::InviteeItem(mtx::identifiers::User user, QWidget *parent)
|
|
||||||
: QWidget{parent}
|
|
||||||
, user_{QString::fromStdString(user.to_string())}
|
|
||||||
{
|
|
||||||
auto topLayout_ = new QHBoxLayout(this);
|
|
||||||
topLayout_->setSpacing(0);
|
|
||||||
topLayout_->setContentsMargins(SidePadding, 0, 3 * SidePadding, 0);
|
|
||||||
|
|
||||||
name_ = new QLabel(user_, this);
|
|
||||||
removeUserBtn_ = new QPushButton(tr("Remove"), this);
|
|
||||||
|
|
||||||
topLayout_->addWidget(name_);
|
|
||||||
topLayout_->addWidget(removeUserBtn_, 0, Qt::AlignRight);
|
|
||||||
|
|
||||||
connect(removeUserBtn_, &QPushButton::clicked, this, &InviteeItem::removeItem);
|
|
||||||
}
|
|
@ -1,31 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <QWidget>
|
|
||||||
|
|
||||||
#include <mtx/identifiers.hpp>
|
|
||||||
|
|
||||||
class QPushButton;
|
|
||||||
class QLabel;
|
|
||||||
|
|
||||||
class InviteeItem : public QWidget
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
public:
|
|
||||||
InviteeItem(mtx::identifiers::User user, QWidget *parent = nullptr);
|
|
||||||
|
|
||||||
QString userID() { return user_; }
|
|
||||||
|
|
||||||
signals:
|
|
||||||
void removeItem();
|
|
||||||
|
|
||||||
private:
|
|
||||||
QString user_;
|
|
||||||
|
|
||||||
QLabel *name_;
|
|
||||||
QPushButton *removeUserBtn_;
|
|
||||||
};
|
|
84
src/InviteesModel.cpp
Normal file
84
src/InviteesModel.cpp
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#include "InviteesModel.h"
|
||||||
|
|
||||||
|
#include "Cache.h"
|
||||||
|
#include "Logging.h"
|
||||||
|
#include "MatrixClient.h"
|
||||||
|
#include "mtx/responses/profile.hpp"
|
||||||
|
|
||||||
|
InviteesModel::InviteesModel(QObject *parent)
|
||||||
|
: QAbstractListModel{parent}
|
||||||
|
{}
|
||||||
|
|
||||||
|
void
|
||||||
|
InviteesModel::addUser(QString mxid)
|
||||||
|
{
|
||||||
|
beginInsertRows(QModelIndex(), invitees_.count(), invitees_.count());
|
||||||
|
|
||||||
|
auto invitee = new Invitee{mxid, this};
|
||||||
|
auto indexOfInvitee = invitees_.count();
|
||||||
|
connect(invitee, &Invitee::userInfoLoaded, this, [this, indexOfInvitee]() {
|
||||||
|
emit dataChanged(index(indexOfInvitee), index(indexOfInvitee));
|
||||||
|
});
|
||||||
|
|
||||||
|
invitees_.push_back(invitee);
|
||||||
|
|
||||||
|
endInsertRows();
|
||||||
|
emit countChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
QHash<int, QByteArray>
|
||||||
|
InviteesModel::roleNames() const
|
||||||
|
{
|
||||||
|
return {{Mxid, "mxid"}, {DisplayName, "displayName"}, {AvatarUrl, "avatarUrl"}};
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant
|
||||||
|
InviteesModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid() || index.row() >= (int)invitees_.size() || index.row() < 0)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
switch (role) {
|
||||||
|
case Mxid:
|
||||||
|
return invitees_[index.row()]->mxid_;
|
||||||
|
case DisplayName:
|
||||||
|
return invitees_[index.row()]->displayName_;
|
||||||
|
case AvatarUrl:
|
||||||
|
return invitees_[index.row()]->avatarUrl_;
|
||||||
|
default:
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList
|
||||||
|
InviteesModel::mxids()
|
||||||
|
{
|
||||||
|
QStringList mxidList;
|
||||||
|
for (int i = 0; i < invitees_.length(); ++i)
|
||||||
|
mxidList.push_back(invitees_[i]->mxid_);
|
||||||
|
return mxidList;
|
||||||
|
}
|
||||||
|
|
||||||
|
Invitee::Invitee(const QString &mxid, QObject *parent)
|
||||||
|
: QObject{parent}
|
||||||
|
, mxid_{mxid}
|
||||||
|
{
|
||||||
|
http::client()->get_profile(
|
||||||
|
mxid_.toStdString(),
|
||||||
|
[this](const mtx::responses::Profile &res, mtx::http::RequestErr err) {
|
||||||
|
if (err) {
|
||||||
|
nhlog::net()->warn("failed to retrieve profile info");
|
||||||
|
emit userInfoLoaded();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
displayName_ = QString::fromStdString(res.display_name);
|
||||||
|
avatarUrl_ = QString::fromStdString(res.avatar_url);
|
||||||
|
|
||||||
|
emit userInfoLoaded();
|
||||||
|
});
|
||||||
|
}
|
63
src/InviteesModel.h
Normal file
63
src/InviteesModel.h
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#ifndef INVITEESMODEL_H
|
||||||
|
#define INVITEESMODEL_H
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
#include <QVector>
|
||||||
|
|
||||||
|
class Invitee : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
Invitee(const QString &mxid, QObject *parent = nullptr);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void userInfoLoaded();
|
||||||
|
|
||||||
|
private:
|
||||||
|
const QString mxid_;
|
||||||
|
QString displayName_;
|
||||||
|
QString avatarUrl_;
|
||||||
|
|
||||||
|
friend class InviteesModel;
|
||||||
|
};
|
||||||
|
|
||||||
|
class InviteesModel : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
Q_PROPERTY(int count READ rowCount NOTIFY countChanged)
|
||||||
|
|
||||||
|
public:
|
||||||
|
enum Roles
|
||||||
|
{
|
||||||
|
Mxid,
|
||||||
|
DisplayName,
|
||||||
|
AvatarUrl,
|
||||||
|
};
|
||||||
|
|
||||||
|
InviteesModel(QObject *parent = nullptr);
|
||||||
|
|
||||||
|
Q_INVOKABLE void addUser(QString mxid);
|
||||||
|
|
||||||
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
|
int rowCount(const QModelIndex & = QModelIndex()) const override
|
||||||
|
{
|
||||||
|
return (int)invitees_.size();
|
||||||
|
}
|
||||||
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
|
QStringList mxids();
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void accept();
|
||||||
|
void countChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QVector<Invitee *> invitees_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // INVITEESMODEL_H
|
@ -21,6 +21,7 @@
|
|||||||
#include "LoginPage.h"
|
#include "LoginPage.h"
|
||||||
#include "MainWindow.h"
|
#include "MainWindow.h"
|
||||||
#include "MatrixClient.h"
|
#include "MatrixClient.h"
|
||||||
|
#include "MemberList.h"
|
||||||
#include "RegisterPage.h"
|
#include "RegisterPage.h"
|
||||||
#include "TrayIcon.h"
|
#include "TrayIcon.h"
|
||||||
#include "UserSettingsPage.h"
|
#include "UserSettingsPage.h"
|
||||||
@ -32,11 +33,9 @@
|
|||||||
#include "ui/SnackBar.h"
|
#include "ui/SnackBar.h"
|
||||||
|
|
||||||
#include "dialogs/CreateRoom.h"
|
#include "dialogs/CreateRoom.h"
|
||||||
#include "dialogs/InviteUsers.h"
|
|
||||||
#include "dialogs/JoinRoom.h"
|
#include "dialogs/JoinRoom.h"
|
||||||
#include "dialogs/LeaveRoom.h"
|
#include "dialogs/LeaveRoom.h"
|
||||||
#include "dialogs/Logout.h"
|
#include "dialogs/Logout.h"
|
||||||
#include "dialogs/MemberList.h"
|
|
||||||
#include "dialogs/ReadReceipts.h"
|
#include "dialogs/ReadReceipts.h"
|
||||||
|
|
||||||
MainWindow *MainWindow::instance_ = nullptr;
|
MainWindow *MainWindow::instance_ = nullptr;
|
||||||
@ -310,14 +309,6 @@ MainWindow::hasActiveUser()
|
|||||||
settings.contains(prefix + "auth/user_id");
|
settings.contains(prefix + "auth/user_id");
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
MainWindow::openMemberListDialog(const QString &room_id)
|
|
||||||
{
|
|
||||||
auto dialog = new dialogs::MemberList(room_id, this);
|
|
||||||
|
|
||||||
showDialog(dialog);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::openLeaveRoomDialog(const QString &room_id)
|
MainWindow::openLeaveRoomDialog(const QString &room_id)
|
||||||
{
|
{
|
||||||
@ -341,18 +332,6 @@ MainWindow::showOverlayProgressBar()
|
|||||||
showSolidOverlayModal(spinner_);
|
showSolidOverlayModal(spinner_);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
MainWindow::openInviteUsersDialog(std::function<void(const QStringList &invitees)> callback)
|
|
||||||
{
|
|
||||||
auto dialog = new dialogs::InviteUsers(this);
|
|
||||||
connect(dialog, &dialogs::InviteUsers::sendInvites, this, [callback](QStringList invitees) {
|
|
||||||
if (!invitees.isEmpty())
|
|
||||||
callback(invitees);
|
|
||||||
});
|
|
||||||
|
|
||||||
showDialog(dialog);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::openJoinRoomDialog(std::function<void(const QString &room_id)> callback)
|
MainWindow::openJoinRoomDialog(std::function<void(const QString &room_id)> callback)
|
||||||
{
|
{
|
||||||
|
@ -65,7 +65,6 @@ public:
|
|||||||
std::function<void(const mtx::requests::CreateRoom &request)> callback);
|
std::function<void(const mtx::requests::CreateRoom &request)> callback);
|
||||||
void openJoinRoomDialog(std::function<void(const QString &room_id)> callback);
|
void openJoinRoomDialog(std::function<void(const QString &room_id)> callback);
|
||||||
void openLogoutDialog();
|
void openLogoutDialog();
|
||||||
void openMemberListDialog(const QString &room_id);
|
|
||||||
void openReadReceiptsDialog(const QString &event_id);
|
void openReadReceiptsDialog(const QString &event_id);
|
||||||
|
|
||||||
void hideOverlay();
|
void hideOverlay();
|
||||||
|
111
src/MemberList.cpp
Normal file
111
src/MemberList.cpp
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#include <QAbstractSlider>
|
||||||
|
#include <QLabel>
|
||||||
|
#include <QListWidgetItem>
|
||||||
|
#include <QPainter>
|
||||||
|
#include <QPushButton>
|
||||||
|
#include <QScrollBar>
|
||||||
|
#include <QShortcut>
|
||||||
|
#include <QStyleOption>
|
||||||
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
|
#include "MemberList.h"
|
||||||
|
|
||||||
|
#include "Cache.h"
|
||||||
|
#include "ChatPage.h"
|
||||||
|
#include "Config.h"
|
||||||
|
#include "Logging.h"
|
||||||
|
#include "Utils.h"
|
||||||
|
#include "timeline/TimelineViewManager.h"
|
||||||
|
#include "ui/Avatar.h"
|
||||||
|
|
||||||
|
MemberList::MemberList(const QString &room_id, QWidget *parent)
|
||||||
|
: QAbstractListModel{parent}
|
||||||
|
, room_id_{room_id}
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
info_ = cache::singleRoomInfo(room_id_.toStdString());
|
||||||
|
} catch (const lmdb::error &) {
|
||||||
|
nhlog::db()->warn("failed to retrieve room info from cache: {}",
|
||||||
|
room_id_.toStdString());
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto members = cache::getMembers(room_id_.toStdString());
|
||||||
|
addUsers(members);
|
||||||
|
numUsersLoaded_ = members.size();
|
||||||
|
} catch (const lmdb::error &e) {
|
||||||
|
nhlog::db()->critical("Failed to retrieve members from cache: {}", e.what());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
MemberList::addUsers(const std::vector<RoomMember> &members)
|
||||||
|
{
|
||||||
|
beginInsertRows(
|
||||||
|
QModelIndex{}, m_memberList.count(), m_memberList.count() + members.size() - 1);
|
||||||
|
|
||||||
|
for (const auto &member : members)
|
||||||
|
m_memberList.push_back(
|
||||||
|
{member,
|
||||||
|
ChatPage::instance()->timelineManager()->rooms()->currentRoom()->avatarUrl(
|
||||||
|
member.user_id)});
|
||||||
|
|
||||||
|
endInsertRows();
|
||||||
|
}
|
||||||
|
|
||||||
|
QHash<int, QByteArray>
|
||||||
|
MemberList::roleNames() const
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
{Mxid, "mxid"},
|
||||||
|
{DisplayName, "displayName"},
|
||||||
|
{AvatarUrl, "avatarUrl"},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant
|
||||||
|
MemberList::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid() || index.row() >= (int)m_memberList.size() || index.row() < 0)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
switch (role) {
|
||||||
|
case Mxid:
|
||||||
|
return m_memberList[index.row()].first.user_id;
|
||||||
|
case DisplayName:
|
||||||
|
return m_memberList[index.row()].first.display_name;
|
||||||
|
case AvatarUrl:
|
||||||
|
return m_memberList[index.row()].second;
|
||||||
|
default:
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
MemberList::canFetchMore(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
const size_t numMembers = rowCount();
|
||||||
|
if (numMembers > 1 && numMembers < info_.member_count)
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
MemberList::fetchMore(const QModelIndex &)
|
||||||
|
{
|
||||||
|
loadingMoreMembers_ = true;
|
||||||
|
emit loadingMoreMembersChanged();
|
||||||
|
|
||||||
|
auto members = cache::getMembers(room_id_.toStdString(), rowCount());
|
||||||
|
addUsers(members);
|
||||||
|
numUsersLoaded_ += members.size();
|
||||||
|
emit numUsersLoadedChanged();
|
||||||
|
|
||||||
|
loadingMoreMembers_ = false;
|
||||||
|
emit loadingMoreMembersChanged();
|
||||||
|
}
|
66
src/MemberList.h
Normal file
66
src/MemberList.h
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "CacheStructs.h"
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
|
||||||
|
class MemberList : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
Q_PROPERTY(QString roomName READ roomName NOTIFY roomNameChanged)
|
||||||
|
Q_PROPERTY(int memberCount READ memberCount NOTIFY memberCountChanged)
|
||||||
|
Q_PROPERTY(QString avatarUrl READ avatarUrl NOTIFY avatarUrlChanged)
|
||||||
|
Q_PROPERTY(QString roomId READ roomId NOTIFY roomIdChanged)
|
||||||
|
Q_PROPERTY(int numUsersLoaded READ numUsersLoaded NOTIFY numUsersLoadedChanged)
|
||||||
|
Q_PROPERTY(bool loadingMoreMembers READ loadingMoreMembers NOTIFY loadingMoreMembersChanged)
|
||||||
|
|
||||||
|
public:
|
||||||
|
enum Roles
|
||||||
|
{
|
||||||
|
Mxid,
|
||||||
|
DisplayName,
|
||||||
|
AvatarUrl,
|
||||||
|
};
|
||||||
|
MemberList(const QString &room_id, QWidget *parent = nullptr);
|
||||||
|
|
||||||
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override
|
||||||
|
{
|
||||||
|
Q_UNUSED(parent)
|
||||||
|
return static_cast<int>(m_memberList.size());
|
||||||
|
}
|
||||||
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
|
|
||||||
|
QString roomName() const { return QString::fromStdString(info_.name); }
|
||||||
|
int memberCount() const { return info_.member_count; }
|
||||||
|
QString avatarUrl() const { return QString::fromStdString(info_.avatar_url); }
|
||||||
|
QString roomId() const { return room_id_; }
|
||||||
|
int numUsersLoaded() const { return numUsersLoaded_; }
|
||||||
|
bool loadingMoreMembers() const { return loadingMoreMembers_; }
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void roomNameChanged();
|
||||||
|
void memberCountChanged();
|
||||||
|
void avatarUrlChanged();
|
||||||
|
void roomIdChanged();
|
||||||
|
void numUsersLoadedChanged();
|
||||||
|
void loadingMoreMembersChanged();
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void addUsers(const std::vector<RoomMember> &users);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
bool canFetchMore(const QModelIndex &) const override;
|
||||||
|
void fetchMore(const QModelIndex &) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QVector<QPair<RoomMember, QString>> m_memberList;
|
||||||
|
QString room_id_;
|
||||||
|
RoomInfo info_;
|
||||||
|
int numUsersLoaded_{0};
|
||||||
|
bool loadingMoreMembers_{false};
|
||||||
|
};
|
@ -1,158 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QIcon>
|
|
||||||
#include <QLabel>
|
|
||||||
#include <QListWidget>
|
|
||||||
#include <QListWidgetItem>
|
|
||||||
#include <QPushButton>
|
|
||||||
#include <QStyleOption>
|
|
||||||
#include <QTimer>
|
|
||||||
#include <QVBoxLayout>
|
|
||||||
|
|
||||||
#include "dialogs/InviteUsers.h"
|
|
||||||
|
|
||||||
#include "Config.h"
|
|
||||||
#include "InviteeItem.h"
|
|
||||||
#include "ui/TextField.h"
|
|
||||||
|
|
||||||
#include <mtx/identifiers.hpp>
|
|
||||||
|
|
||||||
using namespace dialogs;
|
|
||||||
|
|
||||||
InviteUsers::InviteUsers(QWidget *parent)
|
|
||||||
: QFrame(parent)
|
|
||||||
{
|
|
||||||
setAutoFillBackground(true);
|
|
||||||
setWindowFlags(Qt::Tool | Qt::WindowStaysOnTopHint);
|
|
||||||
setWindowModality(Qt::WindowModal);
|
|
||||||
setAttribute(Qt::WA_DeleteOnClose, true);
|
|
||||||
|
|
||||||
setMinimumWidth(conf::window::minModalWidth);
|
|
||||||
setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum);
|
|
||||||
|
|
||||||
auto layout = new QVBoxLayout(this);
|
|
||||||
layout->setSpacing(conf::modals::WIDGET_SPACING);
|
|
||||||
layout->setMargin(conf::modals::WIDGET_MARGIN);
|
|
||||||
|
|
||||||
auto buttonLayout = new QHBoxLayout();
|
|
||||||
buttonLayout->setSpacing(0);
|
|
||||||
buttonLayout->setMargin(0);
|
|
||||||
|
|
||||||
confirmBtn_ = new QPushButton("Invite", this);
|
|
||||||
confirmBtn_->setDefault(true);
|
|
||||||
cancelBtn_ = new QPushButton(tr("Cancel"), this);
|
|
||||||
|
|
||||||
buttonLayout->addStretch(1);
|
|
||||||
buttonLayout->setSpacing(15);
|
|
||||||
buttonLayout->addWidget(cancelBtn_);
|
|
||||||
buttonLayout->addWidget(confirmBtn_);
|
|
||||||
|
|
||||||
inviteeInput_ = new TextField(this);
|
|
||||||
inviteeInput_->setLabel(tr("User ID to invite"));
|
|
||||||
|
|
||||||
inviteeList_ = new QListWidget;
|
|
||||||
inviteeList_->setFrameStyle(QFrame::NoFrame);
|
|
||||||
inviteeList_->setSelectionMode(QAbstractItemView::NoSelection);
|
|
||||||
inviteeList_->setAttribute(Qt::WA_MacShowFocusRect, 0);
|
|
||||||
inviteeList_->setSpacing(5);
|
|
||||||
|
|
||||||
errorLabel_ = new QLabel(this);
|
|
||||||
errorLabel_->setAlignment(Qt::AlignCenter);
|
|
||||||
|
|
||||||
layout->addWidget(inviteeInput_);
|
|
||||||
layout->addWidget(errorLabel_);
|
|
||||||
layout->addWidget(inviteeList_);
|
|
||||||
layout->addLayout(buttonLayout);
|
|
||||||
|
|
||||||
connect(inviteeInput_, &TextField::returnPressed, this, &InviteUsers::addUser);
|
|
||||||
connect(confirmBtn_, &QPushButton::clicked, [this]() {
|
|
||||||
if (!inviteeInput_->text().trimmed().isEmpty()) {
|
|
||||||
addUser();
|
|
||||||
}
|
|
||||||
|
|
||||||
emit sendInvites(invitedUsers());
|
|
||||||
|
|
||||||
inviteeInput_->clear();
|
|
||||||
inviteeList_->clear();
|
|
||||||
errorLabel_->hide();
|
|
||||||
|
|
||||||
emit close();
|
|
||||||
});
|
|
||||||
|
|
||||||
connect(cancelBtn_, &QPushButton::clicked, [this]() {
|
|
||||||
inviteeInput_->clear();
|
|
||||||
inviteeList_->clear();
|
|
||||||
errorLabel_->hide();
|
|
||||||
|
|
||||||
emit close();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
InviteUsers::addUser()
|
|
||||||
{
|
|
||||||
auto user_id = inviteeInput_->text();
|
|
||||||
|
|
||||||
try {
|
|
||||||
namespace ids = mtx::identifiers;
|
|
||||||
auto user = ids::parse<ids::User>(user_id.toStdString());
|
|
||||||
|
|
||||||
auto item = new QListWidgetItem(inviteeList_);
|
|
||||||
auto invitee = new InviteeItem(user, this);
|
|
||||||
|
|
||||||
item->setSizeHint(invitee->minimumSizeHint());
|
|
||||||
item->setFlags(Qt::NoItemFlags);
|
|
||||||
item->setTextAlignment(Qt::AlignCenter);
|
|
||||||
|
|
||||||
inviteeList_->setItemWidget(item, invitee);
|
|
||||||
|
|
||||||
connect(invitee, &InviteeItem::removeItem, this, [this, item]() {
|
|
||||||
emit removeInvitee(item);
|
|
||||||
});
|
|
||||||
|
|
||||||
errorLabel_->hide();
|
|
||||||
inviteeInput_->clear();
|
|
||||||
} catch (std::exception &e) {
|
|
||||||
errorLabel_->setText(e.what());
|
|
||||||
errorLabel_->show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
InviteUsers::removeInvitee(QListWidgetItem *item)
|
|
||||||
{
|
|
||||||
int row = inviteeList_->row(item);
|
|
||||||
auto widget = inviteeList_->takeItem(row);
|
|
||||||
|
|
||||||
inviteeList_->removeItemWidget(widget);
|
|
||||||
}
|
|
||||||
|
|
||||||
QStringList
|
|
||||||
InviteUsers::invitedUsers() const
|
|
||||||
{
|
|
||||||
QStringList users;
|
|
||||||
|
|
||||||
for (int ii = 0; ii < inviteeList_->count(); ++ii) {
|
|
||||||
auto item = inviteeList_->item(ii);
|
|
||||||
auto widget = inviteeList_->itemWidget(item);
|
|
||||||
auto invitee = qobject_cast<InviteeItem *>(widget);
|
|
||||||
|
|
||||||
if (invitee)
|
|
||||||
users << invitee->userID();
|
|
||||||
else
|
|
||||||
qDebug() << "Cast InviteeItem failed";
|
|
||||||
}
|
|
||||||
|
|
||||||
return users;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
InviteUsers::showEvent(QShowEvent *event)
|
|
||||||
{
|
|
||||||
inviteeInput_->setFocus();
|
|
||||||
|
|
||||||
QFrame::showEvent(event);
|
|
||||||
}
|
|
@ -1,45 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <QFrame>
|
|
||||||
#include <QStringList>
|
|
||||||
|
|
||||||
class QPushButton;
|
|
||||||
class QLabel;
|
|
||||||
class TextField;
|
|
||||||
class QListWidget;
|
|
||||||
class QListWidgetItem;
|
|
||||||
|
|
||||||
namespace dialogs {
|
|
||||||
|
|
||||||
class InviteUsers : public QFrame
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
public:
|
|
||||||
explicit InviteUsers(QWidget *parent = nullptr);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void showEvent(QShowEvent *event) override;
|
|
||||||
|
|
||||||
signals:
|
|
||||||
void sendInvites(QStringList invitees);
|
|
||||||
|
|
||||||
private slots:
|
|
||||||
void removeInvitee(QListWidgetItem *item);
|
|
||||||
|
|
||||||
private:
|
|
||||||
void addUser();
|
|
||||||
QStringList invitedUsers() const;
|
|
||||||
|
|
||||||
QPushButton *confirmBtn_;
|
|
||||||
QPushButton *cancelBtn_;
|
|
||||||
|
|
||||||
TextField *inviteeInput_;
|
|
||||||
QLabel *errorLabel_;
|
|
||||||
|
|
||||||
QListWidget *inviteeList_;
|
|
||||||
};
|
|
||||||
} // dialogs
|
|
@ -1,146 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
#include <QAbstractSlider>
|
|
||||||
#include <QLabel>
|
|
||||||
#include <QListWidgetItem>
|
|
||||||
#include <QPainter>
|
|
||||||
#include <QPushButton>
|
|
||||||
#include <QScrollBar>
|
|
||||||
#include <QShortcut>
|
|
||||||
#include <QStyleOption>
|
|
||||||
#include <QVBoxLayout>
|
|
||||||
|
|
||||||
#include "dialogs/MemberList.h"
|
|
||||||
|
|
||||||
#include "Cache.h"
|
|
||||||
#include "ChatPage.h"
|
|
||||||
#include "Config.h"
|
|
||||||
#include "Logging.h"
|
|
||||||
#include "Utils.h"
|
|
||||||
#include "ui/Avatar.h"
|
|
||||||
|
|
||||||
using namespace dialogs;
|
|
||||||
|
|
||||||
MemberItem::MemberItem(const RoomMember &member, QWidget *parent)
|
|
||||||
: QWidget(parent)
|
|
||||||
{
|
|
||||||
topLayout_ = new QHBoxLayout(this);
|
|
||||||
topLayout_->setMargin(0);
|
|
||||||
|
|
||||||
textLayout_ = new QVBoxLayout;
|
|
||||||
textLayout_->setMargin(0);
|
|
||||||
textLayout_->setSpacing(0);
|
|
||||||
|
|
||||||
avatar_ = new Avatar(this, 44);
|
|
||||||
avatar_->setLetter(utils::firstChar(member.display_name));
|
|
||||||
|
|
||||||
avatar_->setImage(ChatPage::instance()->currentRoom(), member.user_id);
|
|
||||||
|
|
||||||
QFont nameFont;
|
|
||||||
nameFont.setPointSizeF(nameFont.pointSizeF() * 1.1);
|
|
||||||
|
|
||||||
userId_ = new QLabel(member.user_id, this);
|
|
||||||
userName_ = new QLabel(member.display_name, this);
|
|
||||||
userName_->setFont(nameFont);
|
|
||||||
|
|
||||||
textLayout_->addWidget(userName_);
|
|
||||||
textLayout_->addWidget(userId_);
|
|
||||||
|
|
||||||
topLayout_->addWidget(avatar_);
|
|
||||||
topLayout_->addLayout(textLayout_, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
MemberItem::paintEvent(QPaintEvent *)
|
|
||||||
{
|
|
||||||
QStyleOption opt;
|
|
||||||
opt.init(this);
|
|
||||||
QPainter p(this);
|
|
||||||
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
MemberList::MemberList(const QString &room_id, QWidget *parent)
|
|
||||||
: QFrame(parent)
|
|
||||||
, room_id_{room_id}
|
|
||||||
{
|
|
||||||
setAutoFillBackground(true);
|
|
||||||
setWindowFlags(Qt::Tool | Qt::WindowStaysOnTopHint);
|
|
||||||
setWindowModality(Qt::WindowModal);
|
|
||||||
setAttribute(Qt::WA_DeleteOnClose, true);
|
|
||||||
|
|
||||||
auto layout = new QVBoxLayout(this);
|
|
||||||
layout->setSpacing(conf::modals::WIDGET_SPACING);
|
|
||||||
layout->setMargin(conf::modals::WIDGET_MARGIN);
|
|
||||||
|
|
||||||
list_ = new QListWidget;
|
|
||||||
list_->setFrameStyle(QFrame::NoFrame);
|
|
||||||
list_->setSelectionMode(QAbstractItemView::NoSelection);
|
|
||||||
list_->setSpacing(5);
|
|
||||||
|
|
||||||
QFont largeFont;
|
|
||||||
largeFont.setPointSizeF(largeFont.pointSizeF() * 1.5);
|
|
||||||
|
|
||||||
setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum);
|
|
||||||
setMinimumHeight(list_->sizeHint().height() * 2);
|
|
||||||
setMinimumWidth(std::max(list_->sizeHint().width() + 4 * conf::modals::WIDGET_MARGIN,
|
|
||||||
QFontMetrics(largeFont).averageCharWidth() * 30 -
|
|
||||||
2 * conf::modals::WIDGET_MARGIN));
|
|
||||||
|
|
||||||
QFont font;
|
|
||||||
font.setPointSizeF(font.pointSizeF() * conf::modals::LABEL_MEDIUM_SIZE_RATIO);
|
|
||||||
|
|
||||||
topLabel_ = new QLabel(tr("Room members"), this);
|
|
||||||
topLabel_->setAlignment(Qt::AlignCenter);
|
|
||||||
topLabel_->setFont(font);
|
|
||||||
|
|
||||||
auto okBtn = new QPushButton(tr("OK"), this);
|
|
||||||
|
|
||||||
auto buttonLayout = new QHBoxLayout();
|
|
||||||
buttonLayout->setSpacing(15);
|
|
||||||
buttonLayout->addStretch(1);
|
|
||||||
buttonLayout->addWidget(okBtn);
|
|
||||||
|
|
||||||
layout->addWidget(topLabel_);
|
|
||||||
layout->addWidget(list_);
|
|
||||||
layout->addLayout(buttonLayout);
|
|
||||||
|
|
||||||
list_->clear();
|
|
||||||
|
|
||||||
connect(list_->verticalScrollBar(), &QAbstractSlider::valueChanged, this, [this](int pos) {
|
|
||||||
if (pos != list_->verticalScrollBar()->maximum())
|
|
||||||
return;
|
|
||||||
|
|
||||||
const size_t numMembers = list_->count() - 1;
|
|
||||||
|
|
||||||
if (numMembers > 0)
|
|
||||||
addUsers(cache::getMembers(room_id_.toStdString(), numMembers));
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
addUsers(cache::getMembers(room_id_.toStdString()));
|
|
||||||
} catch (const lmdb::error &e) {
|
|
||||||
nhlog::db()->critical("Failed to retrieve members from cache: {}", e.what());
|
|
||||||
}
|
|
||||||
|
|
||||||
auto closeShortcut = new QShortcut(QKeySequence(QKeySequence::Cancel), this);
|
|
||||||
connect(closeShortcut, &QShortcut::activated, this, &MemberList::close);
|
|
||||||
connect(okBtn, &QPushButton::clicked, this, &MemberList::close);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
MemberList::addUsers(const std::vector<RoomMember> &members)
|
|
||||||
{
|
|
||||||
for (const auto &member : members) {
|
|
||||||
auto user = new MemberItem(member, this);
|
|
||||||
auto item = new QListWidgetItem;
|
|
||||||
|
|
||||||
item->setSizeHint(user->minimumSizeHint());
|
|
||||||
item->setFlags(Qt::NoItemFlags);
|
|
||||||
item->setTextAlignment(Qt::AlignCenter);
|
|
||||||
|
|
||||||
list_->insertItem(list_->count() - 1, item);
|
|
||||||
list_->setItemWidget(item, user);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,57 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <QFrame>
|
|
||||||
#include <QListWidget>
|
|
||||||
|
|
||||||
class Avatar;
|
|
||||||
class QPushButton;
|
|
||||||
class QHBoxLayout;
|
|
||||||
class QLabel;
|
|
||||||
class QVBoxLayout;
|
|
||||||
|
|
||||||
struct RoomMember;
|
|
||||||
|
|
||||||
template<class T>
|
|
||||||
class QSharedPointer;
|
|
||||||
|
|
||||||
namespace dialogs {
|
|
||||||
|
|
||||||
class MemberItem : public QWidget
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
public:
|
|
||||||
MemberItem(const RoomMember &member, QWidget *parent);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void paintEvent(QPaintEvent *) override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
QHBoxLayout *topLayout_;
|
|
||||||
QVBoxLayout *textLayout_;
|
|
||||||
|
|
||||||
Avatar *avatar_;
|
|
||||||
|
|
||||||
QLabel *userName_;
|
|
||||||
QLabel *userId_;
|
|
||||||
};
|
|
||||||
|
|
||||||
class MemberList : public QFrame
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
public:
|
|
||||||
MemberList(const QString &room_id, QWidget *parent = nullptr);
|
|
||||||
|
|
||||||
public slots:
|
|
||||||
void addUsers(const std::vector<RoomMember> &users);
|
|
||||||
|
|
||||||
private:
|
|
||||||
QString room_id_;
|
|
||||||
QLabel *topLabel_;
|
|
||||||
QListWidget *list_;
|
|
||||||
};
|
|
||||||
} // dialogs
|
|
@ -8,9 +8,9 @@
|
|||||||
#include "MatrixClient.h"
|
#include "MatrixClient.h"
|
||||||
#include "TimelineModel.h"
|
#include "TimelineModel.h"
|
||||||
|
|
||||||
Permissions::Permissions(TimelineModel *parent)
|
Permissions::Permissions(QString roomId, QObject *parent)
|
||||||
: QObject(parent)
|
: QObject(parent)
|
||||||
, room(parent)
|
, roomId_(roomId)
|
||||||
{
|
{
|
||||||
invalidate();
|
invalidate();
|
||||||
}
|
}
|
||||||
@ -19,7 +19,7 @@ void
|
|||||||
Permissions::invalidate()
|
Permissions::invalidate()
|
||||||
{
|
{
|
||||||
pl = cache::client()
|
pl = cache::client()
|
||||||
->getStateEvent<mtx::events::state::PowerLevels>(room->roomId().toStdString())
|
->getStateEvent<mtx::events::state::PowerLevels>(roomId_.toStdString())
|
||||||
.value_or(mtx::events::StateEvent<mtx::events::state::PowerLevels>{})
|
.value_or(mtx::events::StateEvent<mtx::events::state::PowerLevels>{})
|
||||||
.content;
|
.content;
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,7 @@ class Permissions : public QObject
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Permissions(TimelineModel *parent);
|
Permissions(QString roomId, QObject *parent = nullptr);
|
||||||
|
|
||||||
Q_INVOKABLE bool canInvite();
|
Q_INVOKABLE bool canInvite();
|
||||||
Q_INVOKABLE bool canBan();
|
Q_INVOKABLE bool canBan();
|
||||||
@ -28,6 +28,6 @@ public:
|
|||||||
void invalidate();
|
void invalidate();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TimelineModel *room;
|
QString roomId_;
|
||||||
mtx::events::state::PowerLevels pl;
|
mtx::events::state::PowerLevels pl;
|
||||||
};
|
};
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
#include "Logging.h"
|
#include "Logging.h"
|
||||||
#include "MainWindow.h"
|
#include "MainWindow.h"
|
||||||
#include "MatrixClient.h"
|
#include "MatrixClient.h"
|
||||||
|
#include "MemberList.h"
|
||||||
#include "MxcImageProvider.h"
|
#include "MxcImageProvider.h"
|
||||||
#include "Olm.h"
|
#include "Olm.h"
|
||||||
#include "TimelineViewManager.h"
|
#include "TimelineViewManager.h"
|
||||||
@ -317,6 +318,7 @@ TimelineModel::TimelineModel(TimelineViewManager *manager, QString room_id, QObj
|
|||||||
, events(room_id.toStdString(), this)
|
, events(room_id.toStdString(), this)
|
||||||
, room_id_(room_id)
|
, room_id_(room_id)
|
||||||
, manager_(manager)
|
, manager_(manager)
|
||||||
|
, permissions_{room_id}
|
||||||
{
|
{
|
||||||
lastMessage_.timestamp = 0;
|
lastMessage_.timestamp = 0;
|
||||||
|
|
||||||
@ -325,6 +327,10 @@ TimelineModel::TimelineModel(TimelineViewManager *manager, QString room_id, QObj
|
|||||||
this->isSpace_ = create->content.type == mtx::events::state::room_type::space;
|
this->isSpace_ = create->content.type == mtx::events::state::room_type::space;
|
||||||
this->isEncrypted_ = cache::isRoomEncrypted(room_id_.toStdString());
|
this->isEncrypted_ = cache::isRoomEncrypted(room_id_.toStdString());
|
||||||
|
|
||||||
|
// this connection will simplify adding the plainRoomNameChanged() signal everywhere that it
|
||||||
|
// needs to be
|
||||||
|
connect(this, &TimelineModel::roomNameChanged, this, &TimelineModel::plainRoomNameChanged);
|
||||||
|
|
||||||
connect(
|
connect(
|
||||||
this,
|
this,
|
||||||
&TimelineModel::redactionFailed,
|
&TimelineModel::redactionFailed,
|
||||||
@ -1061,11 +1067,28 @@ TimelineModel::openUserProfile(QString userid)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
TimelineModel::openRoomSettings()
|
TimelineModel::openRoomMembers()
|
||||||
{
|
{
|
||||||
RoomSettings *settings = new RoomSettings(roomId(), this);
|
MemberList *memberList = new MemberList(roomId());
|
||||||
|
emit openRoomMembersDialog(memberList);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
TimelineModel::openRoomSettings(QString room_id)
|
||||||
|
{
|
||||||
|
RoomSettings *settings = new RoomSettings(room_id == QString() ? roomId() : room_id, this);
|
||||||
connect(this, &TimelineModel::roomAvatarUrlChanged, settings, &RoomSettings::avatarChanged);
|
connect(this, &TimelineModel::roomAvatarUrlChanged, settings, &RoomSettings::avatarChanged);
|
||||||
openRoomSettingsDialog(settings);
|
emit openRoomSettingsDialog(settings);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
TimelineModel::openInviteUsers(QString roomId)
|
||||||
|
{
|
||||||
|
InviteesModel *model = new InviteesModel{this};
|
||||||
|
connect(model, &InviteesModel::accept, this, [this, model, roomId]() {
|
||||||
|
emit manager_->inviteUsers(roomId == QString() ? room_id_ : roomId, model->mxids());
|
||||||
|
});
|
||||||
|
emit openInviteUsersDialog(model);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
#include "CacheStructs.h"
|
#include "CacheStructs.h"
|
||||||
#include "EventStore.h"
|
#include "EventStore.h"
|
||||||
#include "InputBar.h"
|
#include "InputBar.h"
|
||||||
|
#include "InviteesModel.h"
|
||||||
|
#include "MemberList.h"
|
||||||
#include "Permissions.h"
|
#include "Permissions.h"
|
||||||
#include "ui/RoomSettings.h"
|
#include "ui/RoomSettings.h"
|
||||||
#include "ui/UserProfile.h"
|
#include "ui/UserProfile.h"
|
||||||
@ -158,7 +160,9 @@ class TimelineModel : public QAbstractListModel
|
|||||||
Q_PROPERTY(QString edit READ edit WRITE setEdit NOTIFY editChanged RESET resetEdit)
|
Q_PROPERTY(QString edit READ edit WRITE setEdit NOTIFY editChanged RESET resetEdit)
|
||||||
Q_PROPERTY(
|
Q_PROPERTY(
|
||||||
bool paginationInProgress READ paginationInProgress NOTIFY paginationInProgressChanged)
|
bool paginationInProgress READ paginationInProgress NOTIFY paginationInProgressChanged)
|
||||||
|
Q_PROPERTY(QString roomId READ roomId CONSTANT)
|
||||||
Q_PROPERTY(QString roomName READ roomName NOTIFY roomNameChanged)
|
Q_PROPERTY(QString roomName READ roomName NOTIFY roomNameChanged)
|
||||||
|
Q_PROPERTY(QString plainRoomName READ plainRoomName NOTIFY plainRoomNameChanged)
|
||||||
Q_PROPERTY(QString roomAvatarUrl READ roomAvatarUrl NOTIFY roomAvatarUrlChanged)
|
Q_PROPERTY(QString roomAvatarUrl READ roomAvatarUrl NOTIFY roomAvatarUrlChanged)
|
||||||
Q_PROPERTY(QString roomTopic READ roomTopic NOTIFY roomTopicChanged)
|
Q_PROPERTY(QString roomTopic READ roomTopic NOTIFY roomTopicChanged)
|
||||||
Q_PROPERTY(int roomMemberCount READ roomMemberCount NOTIFY roomMemberCountChanged)
|
Q_PROPERTY(int roomMemberCount READ roomMemberCount NOTIFY roomMemberCountChanged)
|
||||||
@ -235,7 +239,9 @@ public:
|
|||||||
Q_INVOKABLE void forwardMessage(QString eventId, QString roomId);
|
Q_INVOKABLE void forwardMessage(QString eventId, QString roomId);
|
||||||
Q_INVOKABLE void viewDecryptedRawMessage(QString id) const;
|
Q_INVOKABLE void viewDecryptedRawMessage(QString id) const;
|
||||||
Q_INVOKABLE void openUserProfile(QString userid);
|
Q_INVOKABLE void openUserProfile(QString userid);
|
||||||
Q_INVOKABLE void openRoomSettings();
|
Q_INVOKABLE void openRoomMembers();
|
||||||
|
Q_INVOKABLE void openRoomSettings(QString room_id = QString());
|
||||||
|
Q_INVOKABLE void openInviteUsers(QString roomId = QString());
|
||||||
Q_INVOKABLE void editAction(QString id);
|
Q_INVOKABLE void editAction(QString id);
|
||||||
Q_INVOKABLE void replyAction(QString id);
|
Q_INVOKABLE void replyAction(QString id);
|
||||||
Q_INVOKABLE void readReceiptsAction(QString id) const;
|
Q_INVOKABLE void readReceiptsAction(QString id) const;
|
||||||
@ -352,7 +358,9 @@ signals:
|
|||||||
void lastMessageChanged();
|
void lastMessageChanged();
|
||||||
void notificationsChanged();
|
void notificationsChanged();
|
||||||
|
|
||||||
|
void openRoomMembersDialog(MemberList *members);
|
||||||
void openRoomSettingsDialog(RoomSettings *settings);
|
void openRoomSettingsDialog(RoomSettings *settings);
|
||||||
|
void openInviteUsersDialog(InviteesModel *invitees);
|
||||||
|
|
||||||
void newMessageToSend(mtx::events::collections::TimelineEvents event);
|
void newMessageToSend(mtx::events::collections::TimelineEvents event);
|
||||||
void addPendingMessageToStore(mtx::events::collections::TimelineEvents event);
|
void addPendingMessageToStore(mtx::events::collections::TimelineEvents event);
|
||||||
@ -360,6 +368,7 @@ signals:
|
|||||||
|
|
||||||
void encryptionChanged();
|
void encryptionChanged();
|
||||||
void roomNameChanged();
|
void roomNameChanged();
|
||||||
|
void plainRoomNameChanged();
|
||||||
void roomTopicChanged();
|
void roomTopicChanged();
|
||||||
void roomAvatarUrlChanged();
|
void roomAvatarUrlChanged();
|
||||||
void roomMemberCountChanged();
|
void roomMemberCountChanged();
|
||||||
@ -389,7 +398,7 @@ private:
|
|||||||
TimelineViewManager *manager_;
|
TimelineViewManager *manager_;
|
||||||
|
|
||||||
InputBar input_{this};
|
InputBar input_{this};
|
||||||
Permissions permissions_{this};
|
Permissions permissions_;
|
||||||
|
|
||||||
QTimer showEventTimer{this};
|
QTimer showEventTimer{this};
|
||||||
QString eventIdToShow;
|
QString eventIdToShow;
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
#include "DeviceVerificationFlow.h"
|
#include "DeviceVerificationFlow.h"
|
||||||
#include "EventAccessors.h"
|
#include "EventAccessors.h"
|
||||||
#include "ImagePackModel.h"
|
#include "ImagePackModel.h"
|
||||||
|
#include "InviteesModel.h"
|
||||||
#include "Logging.h"
|
#include "Logging.h"
|
||||||
#include "MainWindow.h"
|
#include "MainWindow.h"
|
||||||
#include "MatrixClient.h"
|
#include "MatrixClient.h"
|
||||||
@ -174,6 +175,8 @@ TimelineViewManager::TimelineViewManager(CallManager *callManager, ChatPage *par
|
|||||||
0,
|
0,
|
||||||
"UserProfileModel",
|
"UserProfileModel",
|
||||||
"UserProfile needs to be instantiated on the C++ side");
|
"UserProfile needs to be instantiated on the C++ side");
|
||||||
|
qmlRegisterUncreatableType<MemberList>(
|
||||||
|
"im.nheko", 1, 0, "MemberList", "MemberList needs to be instantiated on the C++ side");
|
||||||
qmlRegisterUncreatableType<RoomSettings>(
|
qmlRegisterUncreatableType<RoomSettings>(
|
||||||
"im.nheko",
|
"im.nheko",
|
||||||
1,
|
1,
|
||||||
@ -182,6 +185,12 @@ TimelineViewManager::TimelineViewManager(CallManager *callManager, ChatPage *par
|
|||||||
"Room Settings needs to be instantiated on the C++ side");
|
"Room Settings needs to be instantiated on the C++ side");
|
||||||
qmlRegisterUncreatableType<TimelineModel>(
|
qmlRegisterUncreatableType<TimelineModel>(
|
||||||
"im.nheko", 1, 0, "Room", "Room needs to be instantiated on the C++ side");
|
"im.nheko", 1, 0, "Room", "Room needs to be instantiated on the C++ side");
|
||||||
|
qmlRegisterUncreatableType<InviteesModel>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"InviteesModel",
|
||||||
|
"InviteesModel needs to be instantiated on the C++ side");
|
||||||
|
|
||||||
static auto self = this;
|
static auto self = this;
|
||||||
qmlRegisterSingletonType<MainWindow>(
|
qmlRegisterSingletonType<MainWindow>(
|
||||||
@ -421,17 +430,6 @@ TimelineViewManager::openImageOverlayInternal(QString eventId, QImage img)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
TimelineViewManager::openInviteUsersDialog()
|
|
||||||
{
|
|
||||||
MainWindow::instance()->openInviteUsersDialog(
|
|
||||||
[this](const QStringList &invitees) { emit inviteUsers(invitees); });
|
|
||||||
}
|
|
||||||
void
|
|
||||||
TimelineViewManager::openMemberListDialog(QString roomid) const
|
|
||||||
{
|
|
||||||
MainWindow::instance()->openMemberListDialog(roomid);
|
|
||||||
}
|
|
||||||
void
|
void
|
||||||
TimelineViewManager::openLeaveRoomDialog(QString roomid) const
|
TimelineViewManager::openLeaveRoomDialog(QString roomid) const
|
||||||
{
|
{
|
||||||
|
@ -65,8 +65,6 @@ public:
|
|||||||
Q_INVOKABLE QString userStatus(QString id) const;
|
Q_INVOKABLE QString userStatus(QString id) const;
|
||||||
|
|
||||||
Q_INVOKABLE void focusMessageInput();
|
Q_INVOKABLE void focusMessageInput();
|
||||||
Q_INVOKABLE void openInviteUsersDialog();
|
|
||||||
Q_INVOKABLE void openMemberListDialog(QString roomid) const;
|
|
||||||
Q_INVOKABLE void openLeaveRoomDialog(QString roomid) const;
|
Q_INVOKABLE void openLeaveRoomDialog(QString roomid) const;
|
||||||
Q_INVOKABLE void removeVerificationFlow(DeviceVerificationFlow *flow);
|
Q_INVOKABLE void removeVerificationFlow(DeviceVerificationFlow *flow);
|
||||||
|
|
||||||
@ -81,7 +79,9 @@ signals:
|
|||||||
void replyingEventChanged(QString replyingEvent);
|
void replyingEventChanged(QString replyingEvent);
|
||||||
void replyClosed();
|
void replyClosed();
|
||||||
void newDeviceVerificationRequest(DeviceVerificationFlow *flow);
|
void newDeviceVerificationRequest(DeviceVerificationFlow *flow);
|
||||||
void inviteUsers(QStringList users);
|
void inviteUsers(QString roomId, QStringList users);
|
||||||
|
void showRoomList();
|
||||||
|
void narrowViewChanged();
|
||||||
void focusChanged();
|
void focusChanged();
|
||||||
void focusInput();
|
void focusInput();
|
||||||
void openImageOverlayInternalCb(QString eventId, QImage img);
|
void openImageOverlayInternalCb(QString eventId, QImage img);
|
||||||
|
Loading…
Reference in New Issue
Block a user