2020-06-17 15:18:31 -04:00
|
|
|
import QtQuick 2.13
|
|
|
|
import QtQuick.Controls 2.13
|
|
|
|
import QtQuick.Layouts 1.13
|
2020-05-27 17:09:12 -04:00
|
|
|
import "../../../shared"
|
2020-05-25 16:34:26 -04:00
|
|
|
import "../../../imports"
|
2020-06-25 12:27:46 -04:00
|
|
|
import "./components"
|
2020-05-27 17:59:34 -04:00
|
|
|
import "./ChatColumn"
|
2020-07-20 15:29:57 -04:00
|
|
|
import "./data"
|
2020-05-25 16:34:26 -04:00
|
|
|
|
2020-05-26 14:16:07 -04:00
|
|
|
StackLayout {
|
2020-07-10 11:37:23 -04:00
|
|
|
id: chatColumnLayout
|
2020-05-26 14:16:07 -04:00
|
|
|
property int chatGroupsListViewCount: 0
|
2020-07-20 13:04:33 -04:00
|
|
|
|
2020-07-09 13:47:36 -04:00
|
|
|
property bool isReply: false
|
2020-07-20 13:04:33 -04:00
|
|
|
property bool isImage: false
|
|
|
|
|
|
|
|
property bool isExtendedInput: isReply || isImage
|
|
|
|
|
2020-07-10 11:37:23 -04:00
|
|
|
property var appSettings
|
2020-07-15 12:19:25 -04:00
|
|
|
property bool isConnected: false
|
2020-08-10 15:06:46 +02:00
|
|
|
property string contactToRemove: ""
|
|
|
|
|
2020-05-26 14:16:07 -04:00
|
|
|
Layout.fillHeight: true
|
|
|
|
Layout.fillWidth: true
|
2020-05-28 08:56:43 -04:00
|
|
|
Layout.minimumWidth: 300
|
2020-05-25 16:34:26 -04:00
|
|
|
|
2020-06-05 10:50:39 -04:00
|
|
|
currentIndex: chatsModel.activeChannelIndex > -1 && chatGroupsListViewCount > 0 ? 0 : 1
|
2020-05-26 11:55:32 -04:00
|
|
|
|
2020-07-20 13:04:33 -04:00
|
|
|
function showReplyArea(){
|
|
|
|
isReply = true;
|
|
|
|
isImage = false;
|
|
|
|
replyAreaContainer.setup()
|
|
|
|
}
|
|
|
|
|
|
|
|
function showImageArea(imagePath){
|
|
|
|
isImage = true;
|
|
|
|
isReply = false;
|
|
|
|
sendImageArea.image = imagePath[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
function hideExtendedArea(){
|
|
|
|
isImage = false;
|
|
|
|
isReply = false;
|
|
|
|
replyAreaContainer.setup();
|
|
|
|
sendImageArea.image = "";
|
|
|
|
}
|
|
|
|
|
2020-05-26 14:16:07 -04:00
|
|
|
ColumnLayout {
|
2020-07-09 13:47:36 -04:00
|
|
|
spacing: 0
|
2020-05-26 11:55:32 -04:00
|
|
|
|
2020-05-26 14:16:07 -04:00
|
|
|
RowLayout {
|
|
|
|
id: chatTopBar
|
|
|
|
Layout.alignment: Qt.AlignHCenter | Qt.AlignTop
|
2020-05-26 11:55:32 -04:00
|
|
|
Layout.fillWidth: true
|
2020-05-26 14:16:07 -04:00
|
|
|
z: 60
|
2020-07-09 13:47:36 -04:00
|
|
|
spacing: 0
|
2020-07-15 12:19:25 -04:00
|
|
|
TopBar {
|
|
|
|
id: topBar
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout {
|
|
|
|
Layout.alignment: Qt.AlignHCenter
|
|
|
|
Layout.fillWidth: true
|
|
|
|
z: 60
|
|
|
|
Rectangle {
|
2020-07-24 10:33:17 -04:00
|
|
|
Component.onCompleted: {
|
2020-07-24 10:33:17 -04:00
|
|
|
isConnected = chatsModel.isOnline
|
2020-07-24 10:33:17 -04:00
|
|
|
if(!isConnected){
|
|
|
|
connectedStatusRect.visible = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-15 12:19:25 -04:00
|
|
|
id: connectedStatusRect
|
|
|
|
Layout.fillWidth: true
|
|
|
|
height: 40;
|
|
|
|
color: isConnected ? Style.current.green : Style.current.darkGrey
|
|
|
|
visible: false
|
|
|
|
Text {
|
|
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
color: Style.current.white
|
|
|
|
id: connectedStatusLbl
|
|
|
|
text: isConnected ?
|
2020-08-26 11:52:26 -04:00
|
|
|
//% "Connected"
|
|
|
|
qsTrId("connected") :
|
|
|
|
//% "Disconnected"
|
|
|
|
qsTrId("disconnected")
|
2020-07-15 12:19:25 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Timer {
|
|
|
|
id: timer
|
|
|
|
}
|
|
|
|
|
|
|
|
Connections {
|
|
|
|
target: chatsModel
|
|
|
|
onOnlineStatusChanged: {
|
2020-07-24 10:33:17 -04:00
|
|
|
if (connected == isConnected) return;
|
|
|
|
isConnected = connected;
|
|
|
|
if(isConnected){
|
2020-07-15 12:19:25 -04:00
|
|
|
timer.setTimeout(function(){
|
|
|
|
connectedStatusRect.visible = false;
|
|
|
|
}, 5000);
|
|
|
|
} else {
|
|
|
|
connectedStatusRect.visible = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-05-25 16:34:26 -04:00
|
|
|
}
|
|
|
|
|
2020-05-26 14:16:07 -04:00
|
|
|
RowLayout {
|
|
|
|
id: chatContainer
|
2020-05-25 16:34:26 -04:00
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.fillHeight: true
|
2020-05-26 14:16:07 -04:00
|
|
|
Layout.alignment: Qt.AlignLeft | Qt.AlignTop
|
2020-07-09 13:47:36 -04:00
|
|
|
spacing: 0
|
2020-05-28 13:34:54 -04:00
|
|
|
ChatMessages {
|
2020-07-09 13:47:36 -04:00
|
|
|
id: chatMessages
|
2020-05-28 13:34:54 -04:00
|
|
|
messageList: chatsModel.messageList
|
2020-07-10 11:37:23 -04:00
|
|
|
appSettings: chatColumnLayout.appSettings
|
2020-05-28 13:34:54 -04:00
|
|
|
}
|
2020-05-27 18:59:17 -04:00
|
|
|
}
|
2020-05-25 16:34:26 -04:00
|
|
|
|
2020-06-25 12:27:46 -04:00
|
|
|
ProfilePopup {
|
|
|
|
id: profilePopup
|
2020-08-10 13:56:16 +02:00
|
|
|
onBlockButtonClicked: {
|
|
|
|
blockContactConfirmationDialog.contactName = name
|
2020-08-10 15:06:46 +02:00
|
|
|
chatColumnLayout.contact = address
|
2020-08-10 13:56:16 +02:00
|
|
|
blockContactConfirmationDialog.open()
|
|
|
|
}
|
2020-08-10 14:15:57 +02:00
|
|
|
onRemoveButtonClicked: {
|
2020-08-10 15:06:46 +02:00
|
|
|
chatColumnLayout.contactToRemove = address
|
2020-08-10 14:15:57 +02:00
|
|
|
removeContactConfirmationDialog.open()
|
|
|
|
}
|
2020-06-25 12:27:46 -04:00
|
|
|
}
|
2020-06-24 13:23:49 +10:00
|
|
|
|
2020-08-10 13:56:16 +02:00
|
|
|
BlockContactConfirmationDialog {
|
|
|
|
id: blockContactConfirmationDialog
|
|
|
|
onBlockButtonClicked: {
|
|
|
|
chatsModel.blockContact(blockContactConfirmationDialog.contactAddress)
|
|
|
|
blockContactConfirmationDialog.close()
|
|
|
|
profilePopup.close()
|
|
|
|
}
|
|
|
|
}
|
2020-07-20 15:29:57 -04:00
|
|
|
|
2020-08-10 15:06:46 +02:00
|
|
|
ConfirmationDialog {
|
|
|
|
id: removeContactConfirmationDialog
|
|
|
|
// % "Remove contact"
|
|
|
|
title: qsTrId("remove-contact")
|
2020-08-26 11:52:26 -04:00
|
|
|
//% "Are you sure you want to remove this contact?"
|
|
|
|
confirmationText: qsTrId("are-you-sure-you-want-to-remove-this-contact-")
|
2020-08-10 15:06:46 +02:00
|
|
|
onConfirmButtonClicked: {
|
|
|
|
if (profileModel.isAdded(chatColumnLayout.contactToRemove)) {
|
|
|
|
profileModel.removeContact(chatColumnLayout.contactToRemove)
|
|
|
|
}
|
|
|
|
removeContactConfirmationDialog.close()
|
|
|
|
}
|
2020-08-10 14:15:57 +02:00
|
|
|
}
|
|
|
|
|
2020-07-20 15:29:57 -04:00
|
|
|
EmojiReactions {
|
|
|
|
id: reactionModel
|
|
|
|
}
|
|
|
|
|
2020-08-03 13:17:03 -04:00
|
|
|
MessageContextMenu {
|
2020-07-09 13:47:36 -04:00
|
|
|
id: messageContextMenu
|
|
|
|
}
|
|
|
|
|
2020-07-14 13:40:58 +02:00
|
|
|
ListModel {
|
|
|
|
id: suggestions
|
|
|
|
}
|
|
|
|
|
|
|
|
Connections {
|
|
|
|
target: chatsModel
|
|
|
|
onActiveChannelChanged: {
|
|
|
|
suggestions.clear()
|
|
|
|
for (let i = 0; i < chatsModel.suggestionList.rowCount(); i++) {
|
|
|
|
suggestions.append({
|
|
|
|
alias: chatsModel.suggestionList.rowData(i, "alias"),
|
|
|
|
ensName: chatsModel.suggestionList.rowData(i, "ensName"),
|
|
|
|
address: chatsModel.suggestionList.rowData(i, "address"),
|
|
|
|
identicon: chatsModel.suggestionList.rowData(i, "identicon"),
|
|
|
|
ensVerified: chatsModel.suggestionList.rowData(i, "ensVerified")
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-24 13:23:49 +10:00
|
|
|
Rectangle {
|
2020-07-09 13:47:36 -04:00
|
|
|
id: inputArea
|
2020-07-13 14:45:54 -04:00
|
|
|
color: Style.current.background
|
2020-07-09 13:47:36 -04:00
|
|
|
border.width: 1
|
2020-07-13 14:45:54 -04:00
|
|
|
border.color: Style.current.border
|
2020-06-24 13:23:49 +10:00
|
|
|
Layout.alignment: Qt.AlignHCenter | Qt.AlignBottom
|
2020-05-25 16:34:26 -04:00
|
|
|
Layout.fillWidth: true
|
2020-06-24 13:23:49 +10:00
|
|
|
Layout.preferredWidth: parent.width
|
2020-07-20 13:04:33 -04:00
|
|
|
height: !isExtendedInput ? 70 : 140
|
2020-06-24 13:23:49 +10:00
|
|
|
Layout.preferredHeight: height
|
2020-07-09 13:47:36 -04:00
|
|
|
|
2020-07-20 11:38:24 -04:00
|
|
|
SuggestionBox {
|
|
|
|
id: suggestionsBox
|
|
|
|
model: suggestions
|
|
|
|
width: chatContainer.width
|
|
|
|
anchors.bottom: inputArea.top
|
|
|
|
anchors.left: inputArea.left
|
|
|
|
filter: chatInput.textInput.text
|
2020-07-20 18:48:23 +02:00
|
|
|
cursorPosition: chatInput.textInput.cursorPosition
|
2020-07-20 11:38:24 -04:00
|
|
|
property: "ensName, alias"
|
2020-07-20 18:48:23 +02:00
|
|
|
onItemSelected: function (item, lastAtPosition, lastCursorPosition) {
|
2020-08-27 19:43:06 +02:00
|
|
|
let hasEmoji = Emoji.hasEmoji(chatInput.textInput.text)
|
|
|
|
let currentText = hasEmoji ?
|
|
|
|
chatsModel.plainText(Emoji.deparse(chatInput.textInput.text)) :
|
|
|
|
chatsModel.plainText(chatInput.textInput.text);
|
|
|
|
|
2020-07-20 11:38:24 -04:00
|
|
|
let aliasName = item[suggestionsBox.property.split(",").map(p => p.trim()).find(p => !!item[p])]
|
2020-08-27 19:47:32 +02:00
|
|
|
aliasName = aliasName.replace(".stateofus.eth", "")
|
2020-07-20 11:38:24 -04:00
|
|
|
let nameLen = aliasName.length + 2 // We're doing a +2 here because of the `@` and the trailing whitespace
|
|
|
|
let position = 0;
|
|
|
|
let text = ""
|
|
|
|
|
2020-07-20 18:48:23 +02:00
|
|
|
if (currentText === "@") {
|
2020-07-20 11:38:24 -04:00
|
|
|
position = nameLen
|
|
|
|
text = "@" + aliasName + " "
|
|
|
|
} else {
|
2020-08-27 19:43:06 +02:00
|
|
|
let left = currentText.substring(0, lastAtPosition)
|
|
|
|
let right = currentText.substring(hasEmoji ? lastCursorPosition + 2 : lastCursorPosition)
|
2020-07-31 17:30:55 -04:00
|
|
|
text = `${left} @${aliasName} ${right}`
|
2020-07-20 11:38:24 -04:00
|
|
|
}
|
|
|
|
|
2020-08-27 19:43:06 +02:00
|
|
|
chatInput.textInput.text = hasEmoji ? Emoji.parse(text, "26x26") : text
|
|
|
|
chatInput.textInput.cursorPosition = lastAtPosition + aliasName.length + 2
|
2020-07-20 11:38:24 -04:00
|
|
|
suggestionsBox.suggestionsModel.clear()
|
|
|
|
}
|
|
|
|
}
|
2020-07-09 13:47:36 -04:00
|
|
|
|
|
|
|
ReplyArea {
|
2020-07-10 18:22:39 -04:00
|
|
|
id: replyAreaContainer
|
2020-07-09 13:47:36 -04:00
|
|
|
visible: isReply
|
|
|
|
}
|
2020-05-25 16:34:26 -04:00
|
|
|
|
2020-07-20 13:04:33 -04:00
|
|
|
SendImageArea {
|
|
|
|
id: sendImageArea
|
|
|
|
visible: isImage
|
|
|
|
}
|
|
|
|
|
2020-09-07 13:47:00 +02:00
|
|
|
Loader {
|
|
|
|
active: chatsModel.loadingMessages
|
|
|
|
sourceComponent: loadingIndicator
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.bottom: chatInput.top
|
|
|
|
anchors.rightMargin: Style.current.padding
|
|
|
|
anchors.bottomMargin: Style.current.padding
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: loadingIndicator
|
|
|
|
SVGImage {
|
|
|
|
id: loadingImg
|
|
|
|
source: "../../../app/img/loading.svg"
|
|
|
|
width: 25
|
|
|
|
height: 25
|
|
|
|
fillMode: Image.Stretch
|
|
|
|
RotationAnimator {
|
|
|
|
target: loadingImg;
|
|
|
|
from: 0;
|
|
|
|
to: 360;
|
|
|
|
duration: 1200
|
|
|
|
running: true
|
|
|
|
loops: Animation.Infinite
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-24 13:23:49 +10:00
|
|
|
ChatInput {
|
2020-07-14 13:40:58 +02:00
|
|
|
id: chatInput
|
2020-07-09 13:47:36 -04:00
|
|
|
height: 40
|
2020-07-20 13:04:33 -04:00
|
|
|
anchors.top: {
|
|
|
|
if(!isExtendedInput){
|
|
|
|
return inputArea.top;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(isReply){
|
|
|
|
return replyAreaContainer.bottom;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(isImage){
|
|
|
|
return sendImageArea.bottom;
|
|
|
|
}
|
|
|
|
}
|
2020-07-09 13:47:36 -04:00
|
|
|
anchors.topMargin: 4
|
|
|
|
anchors.left: parent.left
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.bottom: parent.bottom
|
|
|
|
}
|
2020-05-25 16:34:26 -04:00
|
|
|
}
|
2020-05-26 14:16:07 -04:00
|
|
|
}
|
|
|
|
|
2020-05-27 20:12:07 -04:00
|
|
|
EmptyChat {}
|
2020-05-25 16:34:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*##^##
|
|
|
|
Designer {
|
2020-05-27 13:26:21 -04:00
|
|
|
D{i:0;formeditorColor:"#ffffff";height:770;width:800}
|
2020-05-25 16:34:26 -04:00
|
|
|
}
|
|
|
|
##^##*/
|