2020-06-17 15:18:31 -04:00
|
|
|
import QtQuick 2.13
|
|
|
|
import QtQuick.Controls 2.13
|
|
|
|
import QtQuick.Layouts 1.13
|
|
|
|
import QtQml.Models 2.13
|
2020-05-27 18:59:17 -04:00
|
|
|
import "../../../../shared"
|
|
|
|
import "../../../../imports"
|
2020-06-17 17:43:26 -04:00
|
|
|
import "../components"
|
2020-05-28 13:34:54 -04:00
|
|
|
import "./samples/"
|
2020-07-22 14:37:43 -04:00
|
|
|
import "./MessageComponents"
|
2020-05-27 18:59:17 -04:00
|
|
|
|
2020-05-28 18:22:51 -04:00
|
|
|
ScrollView {
|
2020-06-04 19:42:11 -04:00
|
|
|
id: scrollView
|
|
|
|
|
2020-05-28 13:34:54 -04:00
|
|
|
property var messageList: MessagesData {}
|
2020-06-08 15:25:46 -04:00
|
|
|
property bool loadingMessages: false
|
2020-06-15 08:51:04 -04:00
|
|
|
property real scrollY: chatLogView.visibleArea.yPosition * chatLogView.contentHeight
|
2020-05-28 18:22:51 -04:00
|
|
|
|
2020-06-04 19:42:11 -04:00
|
|
|
contentItem: chatLogView
|
2020-05-27 18:59:17 -04:00
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.fillHeight: true
|
|
|
|
|
2020-06-08 18:34:41 -04:00
|
|
|
ScrollBar.vertical.policy: chatLogView.contentHeight > chatLogView.height ? ScrollBar.AlwaysOn : ScrollBar.AlwaysOff
|
2020-05-28 18:22:51 -04:00
|
|
|
ScrollBar.horizontal.policy: ScrollBar.AlwaysOff
|
|
|
|
|
2020-06-04 19:42:11 -04:00
|
|
|
ListView {
|
2020-07-10 11:37:23 -04:00
|
|
|
id: chatLogView
|
2020-06-04 19:42:11 -04:00
|
|
|
anchors.fill: parent
|
2020-07-15 17:15:01 -04:00
|
|
|
anchors.bottomMargin: Style.current.bigPadding
|
2020-06-04 19:42:11 -04:00
|
|
|
spacing: 4
|
2020-06-15 08:51:04 -04:00
|
|
|
boundsBehavior: Flickable.StopAtBounds
|
2020-06-04 19:42:11 -04:00
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.fillHeight: true
|
2020-06-08 15:25:46 -04:00
|
|
|
|
2020-07-22 13:46:17 -04:00
|
|
|
Timer {
|
|
|
|
id: timer
|
|
|
|
}
|
|
|
|
|
2020-07-23 16:22:45 -04:00
|
|
|
Rectangle {
|
|
|
|
id: newMessagesBox
|
2020-09-23 12:13:39 -04:00
|
|
|
visible: state !== "hidden"
|
2020-07-23 16:22:45 -04:00
|
|
|
color: Style.current.secondaryBackground
|
|
|
|
anchors.bottom: parent.bottom
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.rightMargin: Style.current.padding
|
2020-09-23 12:13:39 -04:00
|
|
|
height: childrenRect.height + 2 * Style.current.smallPadding
|
2020-07-23 16:22:45 -04:00
|
|
|
width: 200
|
|
|
|
radius: Style.current.radius
|
2020-09-23 12:13:39 -04:00
|
|
|
state: "hidden"
|
2020-07-23 16:22:45 -04:00
|
|
|
|
|
|
|
StyledText {
|
|
|
|
id: newMessagesText
|
2020-09-23 12:13:39 -04:00
|
|
|
text: newMessagesBox.state === "new-message" ?
|
|
|
|
//% "New message(s) received"
|
|
|
|
qsTrId("new-message-s--received") :
|
2020-09-23 16:49:15 -04:00
|
|
|
//% "Go back to bottom"
|
|
|
|
qsTrId("go-back-to-bottom")
|
2020-07-23 16:22:45 -04:00
|
|
|
horizontalAlignment: Text.AlignHCenter
|
|
|
|
wrapMode: Text.WordWrap
|
|
|
|
anchors.left: parent.left
|
|
|
|
anchors.leftMargin: Style.current.smallPadding
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.rightMargin: Style.current.smallPadding
|
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.topMargin: Style.current.smallPadding
|
|
|
|
font.pixelSize: 15
|
|
|
|
}
|
|
|
|
StyledText {
|
|
|
|
id: clickHereText
|
2020-09-23 12:13:39 -04:00
|
|
|
visible: newMessagesBox.state === "new-message"
|
|
|
|
height: visible ? implicitHeight : 0
|
2020-08-26 11:52:26 -04:00
|
|
|
//% "Click here to scroll back down"
|
|
|
|
text: qsTrId("click-here-to-scroll-back-down")
|
2020-07-23 16:22:45 -04:00
|
|
|
horizontalAlignment: Text.AlignHCenter
|
|
|
|
wrapMode: Text.WordWrap
|
|
|
|
anchors.left: parent.left
|
|
|
|
anchors.leftMargin: Style.current.smallPadding
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.rightMargin: Style.current.smallPadding
|
|
|
|
anchors.top: newMessagesText.bottom
|
|
|
|
anchors.topMargin: 0
|
|
|
|
font.pixelSize: 12
|
|
|
|
color: Style.current.darkGrey
|
|
|
|
}
|
2020-09-23 12:13:39 -04:00
|
|
|
}
|
|
|
|
MouseArea {
|
2020-09-25 15:44:40 -04:00
|
|
|
enabled: newMessagesBox.visible
|
|
|
|
cursorShape: enabled ? Qt.PointingHandCursor : undefined
|
|
|
|
anchors.fill: newMessagesBox
|
|
|
|
onClicked: {
|
|
|
|
newMessagesBox.state = "hidden"
|
|
|
|
chatLogView.scrollToBottom(true)
|
|
|
|
}
|
2020-07-23 16:22:45 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
onAtYEndChanged: {
|
|
|
|
if (chatLogView.atYEnd) {
|
2020-09-23 12:13:39 -04:00
|
|
|
newMessagesBox.state = "hidden"
|
|
|
|
} else {
|
|
|
|
newMessagesBox.state = "scrolled-up"
|
2020-07-23 16:22:45 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-22 14:37:43 -04:00
|
|
|
function scrollToBottom(force, caller) {
|
|
|
|
if (!force && !chatLogView.atYEnd) {
|
|
|
|
// User has scrolled up, we don't want to scroll back
|
2020-07-23 16:22:45 -04:00
|
|
|
return false
|
2020-07-22 14:37:43 -04:00
|
|
|
}
|
|
|
|
if (caller) {
|
|
|
|
if (caller !== chatLogView.itemAtIndex(chatLogView.count - 1)) {
|
|
|
|
// If we have a caller, only accept its request if it's the last message
|
2020-07-23 16:22:45 -04:00
|
|
|
return false
|
2020-07-22 14:37:43 -04:00
|
|
|
}
|
|
|
|
// Add a small delay because images, even though they say they say they are loaed, they aren't shown yet
|
|
|
|
timer.setTimeout(function() {
|
|
|
|
Qt.callLater(chatLogView.positionViewAtEnd)
|
|
|
|
}, 100);
|
2020-07-23 16:22:45 -04:00
|
|
|
return true
|
2020-07-22 14:37:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
Qt.callLater(chatLogView.positionViewAtEnd)
|
2020-07-23 16:22:45 -04:00
|
|
|
return true
|
2020-07-22 14:37:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-08 15:25:46 -04:00
|
|
|
Connections {
|
2020-07-22 13:46:17 -04:00
|
|
|
|
2020-06-08 15:25:46 -04:00
|
|
|
target: chatsModel
|
|
|
|
onMessagesLoaded: {
|
|
|
|
loadingMessages = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
onActiveChannelChanged: {
|
2020-08-05 14:18:07 -04:00
|
|
|
Qt.callLater(chatLogView.scrollToBottom.bind(this, true))
|
2020-06-05 18:20:45 -04:00
|
|
|
}
|
2020-06-08 15:25:46 -04:00
|
|
|
|
2020-07-22 11:12:24 -04:00
|
|
|
onSendingMessage: {
|
2020-07-23 12:23:20 -04:00
|
|
|
chatLogView.scrollToBottom(true)
|
2020-07-22 11:12:24 -04:00
|
|
|
}
|
|
|
|
|
2020-07-23 16:22:45 -04:00
|
|
|
onNewMessagePushed: {
|
|
|
|
if (!chatLogView.scrollToBottom()) {
|
2020-09-23 12:13:39 -04:00
|
|
|
newMessagesBox.state = "new-message"
|
2020-07-23 16:22:45 -04:00
|
|
|
}
|
2020-06-08 15:25:46 -04:00
|
|
|
}
|
2020-07-10 17:47:31 -04:00
|
|
|
|
2020-07-22 13:46:17 -04:00
|
|
|
onAppReady: {
|
|
|
|
// Add an additionnal delay, since the app can be "ready" just milliseconds before the UI updated to show the chat
|
|
|
|
timer.setTimeout(function() {
|
2020-07-22 14:37:43 -04:00
|
|
|
chatLogView.scrollToBottom(true)
|
2020-07-22 13:46:17 -04:00
|
|
|
}, 500);
|
|
|
|
}
|
|
|
|
|
2020-08-05 14:18:07 -04:00
|
|
|
onMessageNotificationPushed: function(chatId, msg, messageType, chatType, timestamp, identicon, username) {
|
|
|
|
notificationWindow.notifyUser(chatId, msg, messageType, chatType, timestamp, identicon, username)
|
2020-07-10 17:47:31 -04:00
|
|
|
}
|
2020-06-04 15:07:07 -04:00
|
|
|
}
|
2020-06-08 15:25:46 -04:00
|
|
|
|
2020-07-22 11:12:24 -04:00
|
|
|
property var loadMsgs : Backpressure.oneInTime(chatLogView, 500, function() {
|
|
|
|
if(loadingMessages) return;
|
|
|
|
loadingMessages = true;
|
|
|
|
chatsModel.loadMoreMessages();
|
|
|
|
});
|
|
|
|
|
2020-06-08 15:25:46 -04:00
|
|
|
onContentYChanged: {
|
2020-07-22 11:12:24 -04:00
|
|
|
if(scrollY < 500){
|
|
|
|
loadMsgs();
|
2020-06-08 15:25:46 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-04 19:42:11 -04:00
|
|
|
model: messageListDelegate
|
2020-06-10 14:23:18 -04:00
|
|
|
section.property: "sectionIdentifier"
|
2020-06-05 18:20:45 -04:00
|
|
|
section.criteria: ViewSection.FullString
|
2020-06-04 19:42:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
DelegateModel {
|
|
|
|
id: messageListDelegate
|
2020-06-05 18:20:45 -04:00
|
|
|
property var lessThan: [
|
2020-09-17 16:29:00 -04:00
|
|
|
function(left, right) { return left.clock < right.clock }
|
2020-06-05 18:20:45 -04:00
|
|
|
]
|
2020-05-28 18:22:51 -04:00
|
|
|
|
2020-06-05 18:20:45 -04:00
|
|
|
property int sortOrder: 0
|
|
|
|
onSortOrderChanged: items.setGroups(0, items.count, "unsorted")
|
2020-06-04 19:42:11 -04:00
|
|
|
|
|
|
|
function insertPosition(lessThan, item) {
|
|
|
|
var lower = 0
|
|
|
|
var upper = items.count
|
|
|
|
while (lower < upper) {
|
|
|
|
var middle = Math.floor(lower + (upper - lower) / 2)
|
|
|
|
var result = lessThan(item.model, items.get(middle).model);
|
|
|
|
if (result) {
|
|
|
|
upper = middle
|
|
|
|
} else {
|
|
|
|
lower = middle + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return lower
|
|
|
|
}
|
|
|
|
|
|
|
|
function sort(lessThan) {
|
|
|
|
while (unsortedItems.count > 0) {
|
|
|
|
var item = unsortedItems.get(0)
|
|
|
|
var index = insertPosition(lessThan, item)
|
|
|
|
item.groups = "items"
|
|
|
|
items.move(item.itemsIndex, index)
|
2020-05-28 18:22:51 -04:00
|
|
|
}
|
2020-06-04 19:42:11 -04:00
|
|
|
}
|
2020-05-27 18:59:17 -04:00
|
|
|
|
2020-06-04 19:42:11 -04:00
|
|
|
items.includeByDefault: false
|
|
|
|
groups: DelegateModelGroup {
|
|
|
|
id: unsortedItems
|
|
|
|
name: "unsorted"
|
|
|
|
includeByDefault: true
|
|
|
|
onChanged: {
|
2020-06-05 18:20:45 -04:00
|
|
|
if (messageListDelegate.sortOrder == messageListDelegate.lessThan.length)
|
|
|
|
setGroups(0, count, "items")
|
|
|
|
else {
|
|
|
|
messageListDelegate.sort(messageListDelegate.lessThan[messageListDelegate.sortOrder])
|
|
|
|
}
|
2020-05-28 18:22:51 -04:00
|
|
|
}
|
2020-05-27 18:59:17 -04:00
|
|
|
}
|
2020-06-05 18:20:45 -04:00
|
|
|
model: messageList
|
2020-06-08 13:29:28 -04:00
|
|
|
|
2020-07-11 21:03:39 -04:00
|
|
|
delegate: Message {
|
2020-06-05 18:20:45 -04:00
|
|
|
id: msgDelegate
|
2020-06-16 17:24:43 -04:00
|
|
|
fromAuthor: model.fromAuthor
|
2020-06-08 13:29:28 -04:00
|
|
|
chatId: model.chatId
|
2020-06-05 18:20:45 -04:00
|
|
|
userName: model.userName
|
|
|
|
message: model.message
|
2020-07-10 11:24:52 -04:00
|
|
|
plainText: model.plainText
|
2020-06-05 18:20:45 -04:00
|
|
|
identicon: model.identicon
|
|
|
|
isCurrentUser: model.isCurrentUser
|
|
|
|
timestamp: model.timestamp
|
|
|
|
sticker: model.sticker
|
|
|
|
contentType: model.contentType
|
2020-07-01 14:24:13 -04:00
|
|
|
outgoingStatus: model.outgoingStatus
|
2020-07-09 11:50:38 -04:00
|
|
|
responseTo: model.responseTo
|
2020-06-05 18:20:45 -04:00
|
|
|
authorCurrentMsg: msgDelegate.ListView.section
|
|
|
|
authorPrevMsg: msgDelegate.ListView.previousSection
|
2020-07-09 13:47:36 -04:00
|
|
|
profileClick: profilePopup.setPopupData.bind(profilePopup)
|
2020-09-12 20:22:07 +02:00
|
|
|
imageClick: imagePopup.openPopup.bind(imagePopup)
|
2020-07-09 13:47:36 -04:00
|
|
|
messageId: model.messageId
|
2020-08-12 11:01:03 -04:00
|
|
|
emojiReactions: model.emojiReactions
|
2020-07-11 21:03:39 -04:00
|
|
|
prevMessageIndex: {
|
|
|
|
// This is used in order to have access to the previous message and determine the timestamp
|
|
|
|
// we can't rely on the index because the sequence of messages is not ordered on the nim side
|
|
|
|
if(msgDelegate.DelegateModel.itemsIndex > 0){
|
|
|
|
return messageListDelegate.items.get(msgDelegate.DelegateModel.itemsIndex - 1).model.index
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
2020-07-22 14:37:43 -04:00
|
|
|
scrollToBottom: chatLogView.scrollToBottom
|
2020-07-14 11:35:21 -04:00
|
|
|
timeout: model.timeout
|
2020-06-05 18:20:45 -04:00
|
|
|
}
|
2020-05-27 18:59:17 -04:00
|
|
|
}
|
|
|
|
}
|
2020-05-28 11:55:52 -04:00
|
|
|
|
|
|
|
/*##^##
|
|
|
|
Designer {
|
|
|
|
D{i:0;autoSize:true;height:480;width:640}
|
|
|
|
}
|
|
|
|
##^##*/
|