nheko/resources/qml/Completer.qml

337 lines
11 KiB
QML
Raw Normal View History

// 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"
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
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
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
property int rowMargin: 0
property int rowSpacing: Nheko.paddingSmall
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)
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;
}
function finishCompletion() {
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);
}
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))
clip: true
ScrollHelper {
flickable: parent
anchors.fill: parent
enabled: !Settings.mobileMode
}
Timer {
id: deadTimer
interval: 50
}
onContentYChanged: deadTimer.restart()
// 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
spacing: rowSpacing
pixelAligned: true
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 {
property variant modelData: model
ListView.delayRemove: true
2021-05-13 08:23:56 +02:00
color: model.index == popup.currentIndex ? Nheko.colors.highlight : Nheko.colors.base
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
onPositionChanged: if (!listView.moving && !deadTimer.running) popup.currentIndex = model.index
onClicked: {
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);
}
}
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
anchors.fill: parent
anchors.margins: popup.rowMargin
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
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
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
spacing: rowSpacing
2020-11-20 04:33:11 +01:00
2020-11-24 02:35:38 +01:00
Label {
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
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
}
Label {
Layout.leftMargin: Nheko.paddingSmall
Layout.rightMargin: Nheko.paddingSmall
text: model.shortcode
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
}
Label {
text: "(" + model.packname + ")"
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.buttonText
}
}
}
DelegateChoice {
roleValue: "command"
RowLayout {
id: del
anchors.centerIn: parent
spacing: rowSpacing
Label {
text: model.name
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
font.bold: true
}
Label {
text: model.description
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.buttonText
}
}
}
DelegateChoice {
roleValue: "room"
RowLayout {
id: del
anchors.centerIn: centerRowContent ? parent : undefined
spacing: rowSpacing
Avatar {
2021-02-23 17:06:21 +01:00
height: popup.avatarHeight
width: popup.avatarWidth
displayName: model.roomName
roomid: model.roomid
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
2022-05-27 16:31:54 +02:00
enabled: false
}
Label {
text: model.roomName
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
font.italic: model.isTombstoned
2021-06-05 23:20:23 +02:00
textFormat: Text.RichText
}
}
}
DelegateChoice {
roleValue: "roomAliases"
RowLayout {
id: del
anchors.centerIn: parent
spacing: rowSpacing
Avatar {
2021-02-23 17:06:21 +01:00
height: popup.avatarHeight
width: popup.avatarWidth
displayName: model.roomName
roomid: model.roomid
url: model.avatarUrl.replace("mxc://", "image://MxcImage/")
2022-05-27 16:31:54 +02:00
enabled: false
}
Label {
text: model.roomName
2021-05-13 08:23:56 +02:00
color: model.index == popup.currentIndex ? Nheko.colors.highlightedText : Nheko.colors.text
font.italic: model.isTombstoned
2021-06-05 23:20:23 +02:00
textFormat: Text.RichText
}
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
}
}
}
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
}
}