2021-03-05 02:35:15 +03:00
|
|
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
|
|
// 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
|
2020-10-26 16:57:54 +03:00
|
|
|
import QtQuick.Controls 2.3
|
2021-01-12 17:03:39 +03:00
|
|
|
import QtQuick.Layouts 1.2
|
|
|
|
import QtQuick.Window 2.2
|
2020-11-01 01:24:07 +03:00
|
|
|
import im.nheko 1.0
|
|
|
|
|
2020-10-26 16:57:54 +03:00
|
|
|
Rectangle {
|
2021-02-21 04:11:50 +03:00
|
|
|
id: inputBar
|
|
|
|
|
2020-10-26 16:57:54 +03:00
|
|
|
color: colors.window
|
|
|
|
Layout.fillWidth: true
|
2021-02-21 04:11:50 +03:00
|
|
|
Layout.preferredHeight: row.implicitHeight
|
2020-10-26 16:57:54 +03:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2020-10-26 16:57:54 +03:00
|
|
|
RowLayout {
|
2021-02-21 04:11:50 +03:00
|
|
|
id: row
|
2020-10-26 16:57:54 +03:00
|
|
|
|
|
|
|
anchors.fill: parent
|
|
|
|
|
|
|
|
ImageButton {
|
2020-12-10 04:49:48 +03:00
|
|
|
visible: CallManager.callsSupported
|
2021-01-12 01:51:39 +03:00
|
|
|
opacity: CallManager.haveCallInvite ? 0.3 : 1
|
2020-10-26 16:57:54 +03:00
|
|
|
Layout.alignment: Qt.AlignBottom
|
|
|
|
hoverEnabled: true
|
|
|
|
width: 22
|
|
|
|
height: 22
|
2020-12-10 04:49:48 +03:00
|
|
|
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
|
2020-12-10 04:49:48 +03:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-10-26 16:57:54 +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
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-10-26 16:57:54 +03:00
|
|
|
}
|
|
|
|
|
2021-02-21 04:11:50 +03:00
|
|
|
ScrollView {
|
2020-10-26 16:57:54 +03:00
|
|
|
id: textInput
|
|
|
|
|
2021-02-21 04:11:50 +03:00
|
|
|
Layout.alignment: Qt.AlignBottom // | Qt.AlignHCenter
|
2020-10-26 16:57:54 +03:00
|
|
|
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
|
2020-10-26 16:57:54 +03:00
|
|
|
|
|
|
|
TextArea {
|
2021-01-25 18:17:14 +03:00
|
|
|
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) {
|
2021-01-25 18:17:14 +03:00
|
|
|
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();
|
2021-01-25 18:17:14 +03:00
|
|
|
popup.completer.setSearchString(messageInput.getText(completerTriggeredAt, cursorPosition));
|
2020-11-25 21:03:22 +03:00
|
|
|
}
|
|
|
|
|
2020-11-30 14:09:24 +03:00
|
|
|
selectByMouse: true
|
2020-10-26 16:57:54 +03:00
|
|
|
placeholderText: qsTr("Write a message...")
|
2021-02-21 04:11:50 +03:00
|
|
|
placeholderTextColor: colors.buttonText
|
2020-10-26 16:57:54 +03:00
|
|
|
color: colors.text
|
2021-01-17 06:05:02 +03:00
|
|
|
width: textInput.width
|
2020-10-26 16:57:54 +03:00
|
|
|
wrapMode: TextEdit.Wrap
|
2021-02-21 04:11:50 +03:00
|
|
|
padding: 8
|
2020-11-25 19:38:06 +03:00
|
|
|
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
|
|
|
|
2021-02-02 20:54:45 +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();
|
|
|
|
}
|
2020-11-24 05:06:24 +03:00
|
|
|
if (popup.opened)
|
2021-01-25 18:17:14 +03:00
|
|
|
popup.completer.setSearchString(messageInput.getText(completerTriggeredAt, cursorPosition));
|
2020-11-24 05:06:24 +03:00
|
|
|
|
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;
|
2021-01-27 21:19:21 +03:00
|
|
|
} else if (event.key == Qt.Key_Space) {
|
2021-02-05 19:22:49 +03:00
|
|
|
// close popup if user enters space after colon
|
2021-02-10 16:32:16 +03:00
|
|
|
if (cursorPosition == completerTriggeredAt + 1)
|
2021-02-05 19:22:49 +03:00
|
|
|
popup.close();
|
|
|
|
|
2021-01-27 21:19:21 +03:00
|
|
|
if (popup.opened && popup.count <= 0)
|
2021-01-27 21:36:53 +03:00
|
|
|
popup.close();
|
|
|
|
|
2020-11-20 03:22:36 +03:00
|
|
|
} else if (event.modifiers == Qt.ControlModifier && event.key == Qt.Key_U) {
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.clear();
|
2020-11-20 03:22:36 +03:00
|
|
|
} else if (event.modifiers == Qt.ControlModifier && event.key == Qt.Key_P) {
|
2021-01-25 18:17:14 +03:00
|
|
|
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) {
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.text = TimelineManager.timeline.input.nextText();
|
2020-11-20 03:22:36 +03:00
|
|
|
} else if (event.key == Qt.Key_At) {
|
2021-01-25 18:17:14 +03:00
|
|
|
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) {
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.openCompleter(cursorPosition, "emoji");
|
2020-11-20 06:33:11 +03:00
|
|
|
popup.open();
|
2021-02-15 22:17:17 +03:00
|
|
|
} else if (event.key == Qt.Key_NumberSign) {
|
|
|
|
messageInput.openCompleter(cursorPosition, "room");
|
|
|
|
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) {
|
2021-01-25 18:17:14 +03:00
|
|
|
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) {
|
2021-01-25 18:17:14 +03:00
|
|
|
var t = messageInput.getText(pos, pos + 1);
|
2020-11-25 21:03:22 +03:00
|
|
|
console.log('"' + t + '"');
|
2021-01-27 22:26:54 +03:00
|
|
|
if (t == '@') {
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.openCompleter(pos, "user");
|
2020-11-25 21:03:22 +03:00
|
|
|
return ;
|
2021-01-27 22:26:54 +03:00
|
|
|
} else if (t == ' ' || t == '\t') {
|
|
|
|
messageInput.openCompleter(pos + 1, "user");
|
|
|
|
return ;
|
2020-11-25 21:03:22 +03:00
|
|
|
} else if (t == ':') {
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.openCompleter(pos, "emoji");
|
2020-11-25 21:03:22 +03:00
|
|
|
return ;
|
|
|
|
}
|
|
|
|
pos = pos - 1;
|
|
|
|
}
|
|
|
|
// At start of input
|
2021-01-25 18:17:14 +03:00
|
|
|
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();
|
2021-02-25 16:54:50 +03:00
|
|
|
} 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;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
idx++;
|
|
|
|
}
|
|
|
|
} else if (cursorPosition == messageInput.length) {
|
|
|
|
event.accepted = true;
|
|
|
|
cursorPosition = 0;
|
|
|
|
}
|
2021-02-25 16:54:50 +03:00
|
|
|
} else if (event.key == Qt.Key_Down && event.modifiers == Qt.NoModifier) {
|
2021-02-25 01:51:05 +03:00
|
|
|
if (cursorPosition == 0) {
|
|
|
|
event.accepted = true;
|
|
|
|
cursorPosition = messageInput.length;
|
|
|
|
} else if (cursorPosition == messageInput.length && TimelineManager.timeline.edit) {
|
|
|
|
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;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
idx--;
|
|
|
|
}
|
|
|
|
}
|
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 {
|
2021-01-25 18:17:14 +03:00
|
|
|
onActiveTimelineChanged: {
|
|
|
|
messageInput.clear();
|
|
|
|
messageInput.append(TimelineManager.timeline.input.text());
|
|
|
|
messageInput.completerTriggeredAt = -1;
|
2020-11-20 04:38:08 +03:00
|
|
|
popup.completerName = "";
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.forceActiveFocus();
|
2020-11-20 04:38:08 +03:00
|
|
|
}
|
|
|
|
target: TimelineManager
|
|
|
|
}
|
2020-11-24 19:32:45 +03:00
|
|
|
|
|
|
|
Connections {
|
2021-01-25 18:17:14 +03:00
|
|
|
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
|
|
|
|
|
2021-01-25 18:17:14 +03:00
|
|
|
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
|
2021-02-02 20:54:45 +03:00
|
|
|
onInsertText: {
|
|
|
|
messageInput.insert(messageInput.cursorPosition, text);
|
|
|
|
}
|
2021-02-01 04:22:53 +03:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2021-01-25 18:17:14 +03:00
|
|
|
Connections {
|
|
|
|
ignoreUnknownSignals: true
|
2021-01-27 21:36:53 +03:00
|
|
|
onReplyChanged: messageInput.forceActiveFocus()
|
2021-02-02 20:54:45 +03:00
|
|
|
onEditChanged: messageInput.forceActiveFocus()
|
2021-01-25 18:17:14 +03:00
|
|
|
target: TimelineManager.timeline
|
|
|
|
}
|
|
|
|
|
2021-02-05 20:12:08 +03:00
|
|
|
Connections {
|
|
|
|
target: TimelineManager
|
|
|
|
onFocusInput: messageInput.forceActiveFocus()
|
|
|
|
}
|
|
|
|
|
2020-10-26 16:57:54 +03:00
|
|
|
MouseArea {
|
|
|
|
// workaround for wrong cursor shape on some platforms
|
|
|
|
anchors.fill: parent
|
2020-11-01 01:24:07 +03:00
|
|
|
acceptedButtons: Qt.MiddleButton
|
2020-10-26 16:57:54 +03:00
|
|
|
cursorShape: Qt.IBeamCursor
|
2020-11-09 05:12:37 +03:00
|
|
|
onClicked: TimelineManager.timeline.input.paste(true)
|
2020-10-26 16:57:54 +03:00
|
|
|
}
|
2021-02-03 05:12:08 +03:00
|
|
|
|
2021-01-17 06:05:02 +03:00
|
|
|
}
|
2020-10-26 16:57:54 +03:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
ImageButton {
|
2020-11-16 20:41:29 +03:00
|
|
|
id: emojiButton
|
|
|
|
|
2020-10-26 16:57:54 +03:00
|
|
|
Layout.alignment: Qt.AlignRight | Qt.AlignBottom
|
2021-02-21 04:11:50 +03:00
|
|
|
Layout.margins: 8
|
2020-10-26 16:57:54 +03:00
|
|
|
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) {
|
2021-01-25 18:17:14 +03:00
|
|
|
messageInput.insert(messageInput.cursorPosition, emoji);
|
2021-02-10 16:32:16 +03:00
|
|
|
TimelineManager.focusMessageInput();
|
2020-11-16 20:41:29 +03:00
|
|
|
})
|
2020-10-26 16:57:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ImageButton {
|
|
|
|
Layout.alignment: Qt.AlignRight | Qt.AlignBottom
|
2021-02-21 04:11:50 +03:00
|
|
|
Layout.margins: 8
|
2020-10-26 16:57:54 +03:00
|
|
|
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();
|
|
|
|
}
|
2020-10-26 16:57:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|