matrixion/resources/qml/MessageInput.qml

356 lines
15 KiB
QML
Raw Permalink Normal View History

2021-03-05 02:35:15 +03:00
// SPDX-FileCopyrightText: 2021 Nheko Contributors
2021-03-14 04:45:20 +03:00
//
2021-03-05 02:35:15 +03:00
// SPDX-License-Identifier: GPL-3.0-or-later
2020-12-30 23:03:07 +03:00
import "./voip"
2021-02-21 04:11:50 +03:00
import QtQuick 2.12
import QtQuick.Controls 2.3
import QtQuick.Layouts 1.2
import QtQuick.Window 2.2
2020-11-01 01:24:07 +03:00
import im.nheko 1.0
Rectangle {
2021-02-21 04:11:50 +03:00
id: inputBar
color: colors.window
Layout.fillWidth: true
2021-02-21 04:11:50 +03:00
Layout.preferredHeight: row.implicitHeight
Layout.minimumHeight: 40
2020-12-18 20:49:24 +03:00
Component {
id: placeCallDialog
PlaceCall {
}
2021-01-12 01:51:39 +03:00
2020-12-18 20:49:24 +03:00
}
RowLayout {
2021-02-21 04:11:50 +03:00
id: row
anchors.fill: parent
ImageButton {
visible: CallManager.callsSupported
2021-01-12 01:51:39 +03:00
opacity: CallManager.haveCallInvite ? 0.3 : 1
Layout.alignment: Qt.AlignBottom
hoverEnabled: true
width: 22
height: 22
image: CallManager.isOnCall ? ":/icons/icons/ui/end-call.png" : ":/icons/icons/ui/place-call.png"
2020-11-16 01:14:47 +03:00
ToolTip.visible: hovered
ToolTip.text: CallManager.isOnCall ? qsTr("Hang up") : qsTr("Place a call")
2021-02-21 04:11:50 +03:00
Layout.margins: 8
2020-12-18 20:49:24 +03:00
onClicked: {
if (TimelineManager.timeline) {
if (CallManager.haveCallInvite) {
2021-01-12 01:51:39 +03:00
return ;
} else if (CallManager.isOnCall) {
2020-12-18 20:49:24 +03:00
CallManager.hangUp();
2021-01-12 01:51:39 +03:00
} else {
2020-12-18 20:49:24 +03:00
var dialog = placeCallDialog.createObject(timelineRoot);
2020-12-30 23:03:07 +03:00
dialog.open();
2020-12-18 20:49:24 +03:00
}
}
}
}
ImageButton {
Layout.alignment: Qt.AlignBottom
hoverEnabled: true
width: 22
height: 22
image: ":/icons/icons/ui/paper-clip-outline.png"
2021-02-21 04:11:50 +03:00
Layout.margins: 8
2020-11-15 06:52:49 +03:00
onClicked: TimelineManager.timeline.input.openFileSelection()
2020-11-16 01:14:47 +03:00
ToolTip.visible: hovered
ToolTip.text: qsTr("Send a file")
2020-11-15 06:52:49 +03:00
Rectangle {
anchors.fill: parent
color: colors.window
2021-01-19 05:25:56 +03:00
visible: TimelineManager.timeline && TimelineManager.timeline.input.uploading
2020-11-15 06:52:49 +03:00
NhekoBusyIndicator {
anchors.fill: parent
running: parent.visible
}
}
}
2021-02-21 04:11:50 +03:00
ScrollView {
id: textInput
2021-02-21 04:11:50 +03:00
Layout.alignment: Qt.AlignBottom // | Qt.AlignHCenter
Layout.maximumHeight: Window.height / 4
2021-01-17 06:05:02 +03:00
Layout.minimumHeight: Settings.fontSize
2021-02-21 04:11:50 +03:00
implicitWidth: inputBar.width - 4 * (22 + 16) - 24
TextArea {
id: messageInput
2020-11-01 01:24:07 +03:00
2020-11-20 03:22:36 +03:00
property int completerTriggeredAt: -1
2020-11-24 19:32:45 +03:00
function insertCompletion(completion) {
messageInput.remove(completerTriggeredAt, cursorPosition);
messageInput.insert(cursorPosition, completion);
2020-11-24 19:32:45 +03:00
}
2020-11-25 21:03:22 +03:00
function openCompleter(pos, type) {
completerTriggeredAt = pos;
popup.completerName = type;
popup.open();
popup.completer.setSearchString(messageInput.getText(completerTriggeredAt, cursorPosition));
2020-11-25 21:03:22 +03:00
}
function positionCursorAtEnd() {
cursorPosition = messageInput.length;
}
function positionCursorAtStart() {
cursorPosition = 0;
}
2020-11-30 14:09:24 +03:00
selectByMouse: true
placeholderText: qsTr("Write a message...")
2021-02-21 04:11:50 +03:00
placeholderTextColor: colors.buttonText
color: colors.text
2021-01-17 06:05:02 +03:00
width: textInput.width
wrapMode: TextEdit.Wrap
2021-02-21 04:11:50 +03:00
padding: 8
focus: true
2021-01-19 05:25:56 +03:00
onTextChanged: {
2021-01-20 01:58:25 +03:00
if (TimelineManager.timeline)
2021-01-19 05:25:56 +03:00
TimelineManager.timeline.input.updateState(selectionStart, selectionEnd, cursorPosition, text);
2021-01-20 01:58:25 +03:00
forceActiveFocus();
2021-01-19 05:25:56 +03:00
}
2020-11-20 03:22:36 +03:00
onCursorPositionChanged: {
2021-01-19 05:25:56 +03:00
if (!TimelineManager.timeline)
return ;
2020-11-20 03:22:36 +03:00
TimelineManager.timeline.input.updateState(selectionStart, selectionEnd, cursorPosition, text);
2020-11-20 04:38:08 +03:00
if (cursorPosition <= completerTriggeredAt) {
2020-11-20 03:22:36 +03:00
completerTriggeredAt = -1;
popup.close();
}
if (popup.opened)
popup.completer.setSearchString(messageInput.getText(completerTriggeredAt, cursorPosition));
2020-11-20 03:22:36 +03:00
}
2020-11-01 01:24:07 +03:00
onSelectionStartChanged: TimelineManager.timeline.input.updateState(selectionStart, selectionEnd, cursorPosition, text)
onSelectionEndChanged: TimelineManager.timeline.input.updateState(selectionStart, selectionEnd, cursorPosition, text)
2020-11-24 19:32:45 +03:00
// Ensure that we get escape key press events first.
2020-11-20 03:22:36 +03:00
Keys.onShortcutOverride: event.accepted = (completerTriggeredAt != -1 && (event.key === Qt.Key_Escape || event.key === Qt.Key_Tab || event.key === Qt.Key_Enter))
2020-11-01 01:24:07 +03:00
Keys.onPressed: {
if (event.matches(StandardKey.Paste)) {
2020-11-15 06:52:49 +03:00
TimelineManager.timeline.input.paste(false);
event.accepted = true;
} else if (event.key == Qt.Key_Space) {
// close popup if user enters space after colon
if (cursorPosition == completerTriggeredAt + 1)
popup.close();
if (popup.opened && popup.count <= 0)
popup.close();
2020-11-20 03:22:36 +03:00
} else if (event.modifiers == Qt.ControlModifier && event.key == Qt.Key_U) {
messageInput.clear();
2020-11-20 03:22:36 +03:00
} else if (event.modifiers == Qt.ControlModifier && event.key == Qt.Key_P) {
messageInput.text = TimelineManager.timeline.input.previousText();
2020-11-20 03:22:36 +03:00
} else if (event.modifiers == Qt.ControlModifier && event.key == Qt.Key_N) {
messageInput.text = TimelineManager.timeline.input.nextText();
2020-11-20 03:22:36 +03:00
} else if (event.key == Qt.Key_At) {
messageInput.openCompleter(cursorPosition, "user");
2020-11-20 03:22:36 +03:00
popup.open();
2020-11-20 06:33:11 +03:00
} else if (event.key == Qt.Key_Colon) {
messageInput.openCompleter(cursorPosition, "emoji");
2020-11-20 06:33:11 +03:00
popup.open();
} else if (event.key == Qt.Key_NumberSign) {
messageInput.openCompleter(cursorPosition, "roomAliases");
popup.open();
2020-11-20 03:22:36 +03:00
} else if (event.key == Qt.Key_Escape && popup.opened) {
completerTriggeredAt = -1;
2020-11-20 04:38:08 +03:00
popup.completerName = "";
2020-11-20 03:22:36 +03:00
event.accepted = true;
popup.close();
2020-11-25 04:20:42 +03:00
} else if (event.matches(StandardKey.InsertParagraphSeparator)) {
if (popup.opened) {
var currentCompletion = popup.currentCompletion();
popup.completerName = "";
popup.close();
if (currentCompletion) {
messageInput.insertCompletion(currentCompletion);
2020-11-25 04:20:42 +03:00
event.accepted = true;
return ;
}
2020-11-20 03:22:36 +03:00
}
2020-11-25 04:20:42 +03:00
TimelineManager.timeline.input.send();
event.accepted = true;
2020-11-25 21:03:22 +03:00
} else if (event.key == Qt.Key_Tab) {
2020-11-20 03:22:36 +03:00
event.accepted = true;
2020-11-25 21:03:22 +03:00
if (popup.opened) {
popup.up();
} else {
var pos = cursorPosition - 1;
while (pos > -1) {
var t = messageInput.getText(pos, pos + 1);
2020-11-25 21:03:22 +03:00
console.log('"' + t + '"');
if (t == '@') {
messageInput.openCompleter(pos, "user");
2020-11-25 21:03:22 +03:00
return ;
} else if (t == ' ' || t == '\t') {
messageInput.openCompleter(pos + 1, "user");
return ;
2020-11-25 21:03:22 +03:00
} else if (t == ':') {
messageInput.openCompleter(pos, "emoji");
2020-11-25 21:03:22 +03:00
return ;
}
pos = pos - 1;
}
// At start of input
messageInput.openCompleter(0, "user");
2020-11-25 21:03:22 +03:00
}
2020-11-20 03:22:36 +03:00
} else if (event.key == Qt.Key_Up && popup.opened) {
event.accepted = true;
popup.up();
} else if (event.key == Qt.Key_Down && popup.opened) {
event.accepted = true;
popup.down();
} else if (event.key == Qt.Key_Up && event.modifiers == Qt.NoModifier) {
2021-02-25 01:51:05 +03:00
if (cursorPosition == 0) {
event.accepted = true;
var idx = TimelineManager.timeline.edit ? TimelineManager.timeline.idToIndex(TimelineManager.timeline.edit) + 1 : 0;
while (true) {
var id = TimelineManager.timeline.indexToId(idx);
if (!id || TimelineManager.timeline.getDump(id, "").isEditable) {
TimelineManager.timeline.edit = id;
cursorPosition = 0;
Qt.callLater(positionCursorAtEnd);
2021-02-25 01:51:05 +03:00
break;
}
idx++;
}
} else if (positionAt(0, cursorRectangle.y) === 0) {
2021-02-25 01:51:05 +03:00
event.accepted = true;
positionCursorAtStart();
2021-02-25 01:51:05 +03:00
}
} else if (event.key == Qt.Key_Down && event.modifiers == Qt.NoModifier) {
if (cursorPosition == messageInput.length && TimelineManager.timeline.edit) {
2021-02-25 01:51:05 +03:00
event.accepted = true;
var idx = TimelineManager.timeline.idToIndex(TimelineManager.timeline.edit) - 1;
while (true) {
var id = TimelineManager.timeline.indexToId(idx);
if (!id || TimelineManager.timeline.getDump(id, "").isEditable) {
TimelineManager.timeline.edit = id;
Qt.callLater(positionCursorAtStart);
2021-02-25 01:51:05 +03:00
break;
}
idx--;
}
} else if (positionAt(width, cursorRectangle.y + 2) === messageInput.length) {
event.accepted = true;
positionCursorAtEnd();
2021-02-25 01:51:05 +03:00
}
2020-11-20 03:22:36 +03:00
}
}
2021-01-17 06:05:02 +03:00
background: null
2020-11-20 03:22:36 +03:00
2020-11-20 04:38:08 +03:00
Connections {
onActiveTimelineChanged: {
messageInput.clear();
messageInput.append(TimelineManager.timeline.input.text());
messageInput.completerTriggeredAt = -1;
2020-11-20 04:38:08 +03:00
popup.completerName = "";
messageInput.forceActiveFocus();
2020-11-20 04:38:08 +03:00
}
target: TimelineManager
}
2020-11-24 19:32:45 +03:00
Connections {
onCompletionClicked: messageInput.insertCompletion(completion)
2020-11-24 19:32:45 +03:00
target: popup
}
2020-11-20 04:38:08 +03:00
2020-11-20 03:22:36 +03:00
Completer {
id: popup
x: messageInput.completerTriggeredAt >= 0 ? messageInput.positionToRectangle(messageInput.completerTriggeredAt).x : 0
y: messageInput.completerTriggeredAt >= 0 ? messageInput.positionToRectangle(messageInput.completerTriggeredAt).y - height : 0
2020-11-15 06:52:49 +03:00
}
Connections {
2021-01-19 05:25:56 +03:00
ignoreUnknownSignals: true
onInsertText: {
2021-04-15 02:45:18 +03:00
messageInput.remove(messageInput.selectionStart, messageInput.selectionEnd);
messageInput.insert(messageInput.cursorPosition, text);
}
onTextChanged: {
messageInput.text = newText;
messageInput.cursorPosition = newText.length;
}
2021-01-19 05:25:56 +03:00
target: TimelineManager.timeline ? TimelineManager.timeline.input : null
2020-11-01 01:24:07 +03:00
}
Connections {
ignoreUnknownSignals: true
onReplyChanged: messageInput.forceActiveFocus()
onEditChanged: messageInput.forceActiveFocus()
target: TimelineManager.timeline
}
Connections {
target: TimelineManager
onFocusInput: messageInput.forceActiveFocus()
}
MouseArea {
// workaround for wrong cursor shape on some platforms
anchors.fill: parent
2020-11-01 01:24:07 +03:00
acceptedButtons: Qt.MiddleButton
cursorShape: Qt.IBeamCursor
2020-11-09 05:12:37 +03:00
onClicked: TimelineManager.timeline.input.paste(true)
}
2021-01-17 06:05:02 +03:00
}
}
ImageButton {
2020-11-16 20:41:29 +03:00
id: emojiButton
Layout.alignment: Qt.AlignRight | Qt.AlignBottom
2021-02-21 04:11:50 +03:00
Layout.margins: 8
hoverEnabled: true
width: 22
height: 22
image: ":/icons/icons/ui/smile.png"
2020-11-16 01:14:47 +03:00
ToolTip.visible: hovered
ToolTip.text: qsTr("Emoji")
2020-11-16 20:41:29 +03:00
onClicked: emojiPopup.visible ? emojiPopup.close() : emojiPopup.show(emojiButton, function(emoji) {
messageInput.insert(messageInput.cursorPosition, emoji);
TimelineManager.focusMessageInput();
2020-11-16 20:41:29 +03:00
})
}
ImageButton {
Layout.alignment: Qt.AlignRight | Qt.AlignBottom
2021-02-21 04:11:50 +03:00
Layout.margins: 8
hoverEnabled: true
width: 22
height: 22
image: ":/icons/icons/ui/cursor.png"
2021-01-17 06:05:02 +03:00
Layout.rightMargin: 8
2020-11-16 01:14:47 +03:00
ToolTip.visible: hovered
ToolTip.text: qsTr("Send")
2020-11-15 06:52:49 +03:00
onClicked: {
TimelineManager.timeline.input.send();
}
}
}
}