2023-02-21 23:48:49 +01:00
|
|
|
// SPDX-FileCopyrightText: Nheko Contributors
|
2021-03-14 02:45:20 +01:00
|
|
|
//
|
2021-03-05 00:35:15 +01:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
2021-01-12 02:22:40 +01:00
|
|
|
import "./ui"
|
2021-12-30 05:46:30 +01:00
|
|
|
import QtQuick 2.15
|
|
|
|
import QtQuick.Controls 2.15
|
|
|
|
import QtQuick.Layouts 1.15
|
2020-11-20 01:22:36 +01:00
|
|
|
import im.nheko 1.0
|
|
|
|
|
2022-02-21 04:06:49 +01:00
|
|
|
Control {
|
2020-11-20 01:22:36 +01:00
|
|
|
id: popup
|
|
|
|
|
2021-12-30 05:46:30 +01:00
|
|
|
property alias currentIndex: listView.currentIndex
|
2022-05-27 16:31:54 +02:00
|
|
|
property string roomId
|
2020-11-20 01:22:36 +01:00
|
|
|
property string completerName
|
|
|
|
property var completer
|
2020-11-24 02:35:38 +01:00
|
|
|
property bool bottomToTop: true
|
2021-02-21 19:31:50 +01:00
|
|
|
property bool fullWidth: false
|
2021-02-24 09:08:01 +01:00
|
|
|
property bool centerRowContent: true
|
2021-02-23 17:06:21 +01:00
|
|
|
property int avatarHeight: 24
|
|
|
|
property int avatarWidth: 24
|
2021-02-24 15:02:13 +01:00
|
|
|
property int rowMargin: 0
|
2023-05-25 21:50:54 +02:00
|
|
|
property int rowSpacing: Nheko.paddingSmall
|
2021-01-27 19:19:21 +01:00
|
|
|
property alias count: listView.count
|
2020-11-20 01:22:36 +01:00
|
|
|
|
2020-11-24 17:32:45 +01:00
|
|
|
signal completionClicked(string completion)
|
2021-02-22 18:38:42 +01:00
|
|
|
signal completionSelected(string id)
|
2020-11-24 17:32:45 +01:00
|
|
|
|
2020-11-20 01:22:36 +01:00
|
|
|
function up() {
|
2020-11-24 02:35:38 +01:00
|
|
|
if (bottomToTop)
|
|
|
|
down_();
|
|
|
|
else
|
|
|
|
up_();
|
|
|
|
}
|
|
|
|
|
|
|
|
function down() {
|
|
|
|
if (bottomToTop)
|
|
|
|
up_();
|
|
|
|
else
|
|
|
|
down_();
|
|
|
|
}
|
|
|
|
|
|
|
|
function up_() {
|
2020-11-20 01:22:36 +01:00
|
|
|
currentIndex = currentIndex - 1;
|
|
|
|
if (currentIndex == -2)
|
2020-11-24 02:35:38 +01:00
|
|
|
currentIndex = listView.count - 1;
|
2020-11-20 01:22:36 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
function down_() {
|
2020-11-20 01:22:36 +01:00
|
|
|
currentIndex = currentIndex + 1;
|
2020-11-24 02:35:38 +01:00
|
|
|
if (currentIndex >= listView.count)
|
2020-11-20 01:22:36 +01:00
|
|
|
currentIndex = -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function currentCompletion() {
|
2020-11-24 02:35:38 +01:00
|
|
|
if (currentIndex > -1 && currentIndex < listView.count)
|
2020-11-20 01:22:36 +01:00
|
|
|
return completer.completionAt(currentIndex);
|
|
|
|
else
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2021-02-22 19:48:31 +01:00
|
|
|
function finishCompletion() {
|
2021-03-14 01:24:26 +01:00
|
|
|
if (popup.completerName == "room")
|
|
|
|
popup.completionSelected(listView.itemAtIndex(currentIndex).modelData.roomid);
|
2022-05-27 16:31:54 +02:00
|
|
|
else if (popup.completerName == "user")
|
|
|
|
popup.completionSelected(listView.itemAtIndex(currentIndex).modelData.userid);
|
2021-03-14 01:24:26 +01:00
|
|
|
|
2021-02-22 19:48:31 +01:00
|
|
|
}
|
|
|
|
|
2022-05-27 16:31:54 +02:00
|
|
|
function changeCompleter() {
|
2020-11-24 02:35:38 +01:00
|
|
|
if (completerName) {
|
2022-05-27 16:31:54 +02:00
|
|
|
completer = TimelineManager.completerFor(completerName, completerName == "room" ? "" : (popup.roomId != "" ? popup.roomId : room.roomId));
|
2020-11-24 02:35:38 +01:00
|
|
|
completer.setSearchString("");
|
|
|
|
} else {
|
2020-11-20 02:38:08 +01:00
|
|
|
completer = undefined;
|
2020-11-24 02:35:38 +01:00
|
|
|
}
|
2022-02-21 04:06:49 +01:00
|
|
|
currentIndex = -1
|
2020-11-20 01:22:36 +01:00
|
|
|
}
|
2022-05-27 16:31:54 +02:00
|
|
|
onCompleterNameChanged: changeCompleter()
|
|
|
|
onRoomIdChanged: changeCompleter()
|
2022-03-01 01:59:06 +01:00
|
|
|
|
|
|
|
bottomPadding: 1
|
|
|
|
leftPadding: 1
|
|
|
|
topPadding: 1
|
|
|
|
rightPadding: 1
|
2022-02-21 04:06:49 +01:00
|
|
|
|
|
|
|
contentItem: ListView {
|
2020-11-24 02:35:38 +01:00
|
|
|
id: listView
|
2020-11-20 01:22:36 +01:00
|
|
|
|
2022-02-21 04:06:49 +01:00
|
|
|
// If we have fewer than 7 items, just use the list view's content height.
|
|
|
|
// Otherwise, we want to show 7 items. Each item consists of row spacing between rows, row margins
|
|
|
|
// on each side of a row, 1px of padding above the first item and below the last item, and nominally
|
|
|
|
// some kind of content height. avatarHeight is used for just about every delegate, so we're using
|
|
|
|
// that until we find something better. Put is all together and you have the formula below!
|
|
|
|
implicitHeight: Math.min(contentHeight, 6*rowSpacing + 7*(popup.avatarHeight + 2*rowMargin))
|
2021-12-30 05:46:30 +01:00
|
|
|
clip: true
|
|
|
|
ScrollHelper {
|
|
|
|
flickable: parent
|
|
|
|
anchors.fill: parent
|
|
|
|
enabled: !Settings.mobileMode
|
|
|
|
}
|
|
|
|
|
|
|
|
Timer {
|
|
|
|
id: deadTimer
|
|
|
|
interval: 50
|
|
|
|
}
|
|
|
|
|
|
|
|
onContentYChanged: deadTimer.restart()
|
|
|
|
|
2022-06-12 17:49:20 +02:00
|
|
|
// Broken, see https://bugreports.qt.io/browse/QTBUG-102811
|
|
|
|
//reuseItems: true
|
2022-02-21 04:06:49 +01:00
|
|
|
implicitWidth: listView.contentItem.childrenRect.width
|
2020-11-24 02:35:38 +01:00
|
|
|
model: completer
|
|
|
|
verticalLayoutDirection: popup.bottomToTop ? ListView.BottomToTop : ListView.TopToBottom
|
2021-02-24 15:02:13 +01:00
|
|
|
spacing: rowSpacing
|
2021-03-06 19:48:24 +01:00
|
|
|
pixelAligned: true
|
2021-12-30 05:46:30 +01:00
|
|
|
highlightFollowsCurrentItem: true
|
2020-11-20 01:22:36 +01:00
|
|
|
|
2023-02-08 20:19:35 +01:00
|
|
|
displayMarginBeginning: height / 2
|
|
|
|
displayMarginEnd: height / 2
|
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
delegate: Rectangle {
|
2021-03-14 01:24:26 +01:00
|
|
|
property variant modelData: model
|
|
|
|
|
2023-03-25 22:31:18 +01:00
|
|
|
ListView.delayRemove: true
|
|
|
|
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlight : Nheko.colors.base
|
2021-12-30 05:46:30 +01:00
|
|
|
height: chooser.child.implicitHeight + 2 * popup.rowMargin
|
2022-03-01 01:59:06 +01:00
|
|
|
implicitWidth: fullWidth ? ListView.view.width : chooser.child.implicitWidth + 4
|
2020-11-20 01:22:36 +01:00
|
|
|
|
2020-11-24 17:32:45 +01:00
|
|
|
MouseArea {
|
2021-01-12 02:22:40 +01:00
|
|
|
id: mouseArea
|
|
|
|
|
2020-11-24 17:32:45 +01:00
|
|
|
anchors.fill: parent
|
|
|
|
hoverEnabled: true
|
2021-12-30 05:46:30 +01:00
|
|
|
onPositionChanged: if (!listView.moving && !deadTimer.running) popup.currentIndex = model.index
|
2021-02-22 18:38:42 +01:00
|
|
|
onClicked: {
|
2021-12-30 05:46:30 +01:00
|
|
|
popup.completionClicked(completer.completionAt(model.index));
|
|
|
|
if (popup.completerName == "room")
|
|
|
|
popup.completionSelected(model.roomid);
|
2022-05-27 16:31:54 +02:00
|
|
|
else if (popup.completerName == "user")
|
|
|
|
popup.completionSelected(model.userid);
|
2021-02-22 18:38:42 +01:00
|
|
|
}
|
2022-01-02 06:22:27 +01:00
|
|
|
}
|
|
|
|
Ripple {
|
|
|
|
color: Qt.rgba(Nheko.colors.base.r, Nheko.colors.base.g, Nheko.colors.base.b, 0.5)
|
2020-11-24 17:32:45 +01:00
|
|
|
}
|
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
DelegateChooser {
|
|
|
|
id: chooser
|
2020-11-20 01:22:36 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
roleValue: popup.completerName
|
2021-02-24 15:02:13 +01:00
|
|
|
anchors.fill: parent
|
|
|
|
anchors.margins: popup.rowMargin
|
2021-12-30 05:46:30 +01:00
|
|
|
enabled: false
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
DelegateChoice {
|
|
|
|
roleValue: "user"
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
RowLayout {
|
|
|
|
id: del
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2022-05-27 16:31:54 +02:00
|
|
|
anchors.centerIn: centerRowContent ? parent : undefined
|
2021-02-24 15:02:13 +01:00
|
|
|
spacing: rowSpacing
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
Avatar {
|
2021-02-23 17:06:21 +01:00
|
|
|
height: popup.avatarHeight
|
|
|
|
width: popup.avatarWidth
|
2020-11-24 02:35:38 +01:00
|
|
|
displayName: model.displayName
|
2021-09-01 03:53:12 +02:00
|
|
|
userid: model.userid
|
2020-11-24 02:35:38 +01:00
|
|
|
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
|
2022-05-27 16:31:54 +02:00
|
|
|
enabled: false
|
2020-11-24 02:35:38 +01:00
|
|
|
}
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
Label {
|
|
|
|
text: model.displayName
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
|
2020-11-20 04:33:11 +01:00
|
|
|
}
|
|
|
|
|
2020-11-24 19:06:31 +01:00
|
|
|
Label {
|
|
|
|
text: "(" + model.userid + ")"
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.buttonText
|
2020-11-24 19:06:31 +01:00
|
|
|
}
|
|
|
|
|
2020-11-20 01:22:36 +01:00
|
|
|
}
|
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
}
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
DelegateChoice {
|
|
|
|
roleValue: "emoji"
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
RowLayout {
|
|
|
|
id: del
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
anchors.centerIn: parent
|
2021-02-24 15:02:13 +01:00
|
|
|
spacing: rowSpacing
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2020-11-24 02:35:38 +01:00
|
|
|
Label {
|
2023-05-25 21:50:54 +02:00
|
|
|
visible: !!model.unicode
|
2020-11-24 02:35:38 +01:00
|
|
|
text: model.unicode
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
|
2020-11-24 02:35:38 +01:00
|
|
|
font: Settings.emojiFont
|
|
|
|
}
|
2020-11-20 04:33:11 +01:00
|
|
|
|
2023-05-25 21:50:54 +02:00
|
|
|
Avatar {
|
|
|
|
visible: !model.unicode
|
|
|
|
height: popup.avatarHeight
|
|
|
|
width: popup.avatarWidth
|
|
|
|
displayName: model.shortcode
|
|
|
|
//userid: model.shortcode
|
|
|
|
url: (model.url ? model.url : "").replace("mxc://", "image://MxcImage/")
|
|
|
|
enabled: false
|
|
|
|
crop: false
|
2020-11-20 04:33:11 +01:00
|
|
|
}
|
|
|
|
|
2022-11-09 04:58:19 +01:00
|
|
|
Label {
|
2023-05-25 21:50:54 +02:00
|
|
|
Layout.leftMargin: Nheko.paddingSmall
|
|
|
|
Layout.rightMargin: Nheko.paddingSmall
|
|
|
|
text: model.shortcode
|
2022-11-09 04:58:19 +01:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
|
|
|
|
}
|
|
|
|
|
|
|
|
Label {
|
2023-05-25 21:50:54 +02:00
|
|
|
text: "(" + model.packname + ")"
|
2022-11-09 04:58:19 +01:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.buttonText
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-12-27 06:23:36 +01:00
|
|
|
DelegateChoice {
|
2023-05-25 21:50:54 +02:00
|
|
|
roleValue: "command"
|
2021-12-27 06:23:36 +01:00
|
|
|
|
|
|
|
RowLayout {
|
|
|
|
id: del
|
|
|
|
|
|
|
|
anchors.centerIn: parent
|
|
|
|
spacing: rowSpacing
|
|
|
|
|
|
|
|
Label {
|
2023-05-25 21:50:54 +02:00
|
|
|
text: model.name
|
2021-12-27 06:23:36 +01:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
|
2023-05-25 21:50:54 +02:00
|
|
|
font.bold: true
|
2021-12-27 06:23:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Label {
|
2023-05-25 21:50:54 +02:00
|
|
|
text: model.description
|
2021-12-27 06:23:36 +01:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.buttonText
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-02-15 20:17:17 +01:00
|
|
|
DelegateChoice {
|
|
|
|
roleValue: "room"
|
|
|
|
|
|
|
|
RowLayout {
|
|
|
|
id: del
|
|
|
|
|
2021-02-24 15:02:13 +01:00
|
|
|
anchors.centerIn: centerRowContent ? parent : undefined
|
|
|
|
spacing: rowSpacing
|
2021-02-15 20:17:17 +01:00
|
|
|
|
|
|
|
Avatar {
|
2021-02-23 17:06:21 +01:00
|
|
|
height: popup.avatarHeight
|
|
|
|
width: popup.avatarWidth
|
2021-03-14 02:50:44 +01:00
|
|
|
displayName: model.roomName
|
2021-09-01 03:53:12 +02:00
|
|
|
roomid: model.roomid
|
2021-02-15 20:17:17 +01:00
|
|
|
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
|
2022-05-27 16:31:54 +02:00
|
|
|
enabled: false
|
2021-02-15 20:17:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Label {
|
2021-02-21 19:31:50 +01:00
|
|
|
text: model.roomName
|
2021-03-06 19:48:24 +01:00
|
|
|
font.pixelSize: popup.avatarHeight * 0.5
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
|
2023-05-06 19:30:29 +02:00
|
|
|
font.italic: model.isTombstoned
|
2021-06-05 23:20:23 +02:00
|
|
|
textFormat: Text.RichText
|
2021-02-21 19:31:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
DelegateChoice {
|
|
|
|
roleValue: "roomAliases"
|
|
|
|
|
2021-02-15 20:17:17 +01:00
|
|
|
RowLayout {
|
|
|
|
id: del
|
|
|
|
|
|
|
|
anchors.centerIn: parent
|
2021-02-24 15:02:13 +01:00
|
|
|
spacing: rowSpacing
|
2021-02-15 20:17:17 +01:00
|
|
|
|
|
|
|
Avatar {
|
2021-02-23 17:06:21 +01:00
|
|
|
height: popup.avatarHeight
|
|
|
|
width: popup.avatarWidth
|
2021-03-14 02:50:44 +01:00
|
|
|
displayName: model.roomName
|
2021-09-01 03:53:12 +02:00
|
|
|
roomid: model.roomid
|
2021-02-15 20:17:17 +01:00
|
|
|
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
|
2022-05-27 16:31:54 +02:00
|
|
|
enabled: false
|
2021-02-15 20:17:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Label {
|
|
|
|
text: model.roomName
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
|
2023-05-06 19:30:29 +02:00
|
|
|
font.italic: model.isTombstoned
|
2021-06-05 23:20:23 +02:00
|
|
|
textFormat: Text.RichText
|
2021-02-15 20:17:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Label {
|
|
|
|
text: "(" + model.roomAlias + ")"
|
2021-05-13 08:23:56 +02:00
|
|
|
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.buttonText
|
2021-06-05 23:20:23 +02:00
|
|
|
textFormat: Text.RichText
|
2021-02-15 20:17:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-11-20 01:22:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
background: Rectangle {
|
2021-05-13 08:23:56 +02:00
|
|
|
color: Nheko.colors.base
|
|
|
|
border.color: Nheko.colors.mid
|
2020-11-20 01:22:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|