2020-06-17 15:18:31 -04:00
import QtQuick 2.13
import QtQuick . Controls 2.13
2020-11-20 13:39:14 -04:00
import QtQuick . Window 2.13
2020-06-17 15:18:31 -04:00
import QtQuick . Layouts 1.13
import QtQml . Models 2.13
2020-09-24 11:05:17 -04:00
import QtGraphicalEffects 1.13
2020-11-30 23:24:01 +02:00
import QtQuick . Dialogs 1.3
2020-05-27 18:59:17 -04:00
import "../../../../shared"
2021-05-26 13:36:24 -04:00
import "../../../../shared/status"
2020-05-27 18:59:17 -04:00
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"
2021-06-30 14:46:26 -04:00
import "../ContactsColumn"
2021-07-06 18:41:26 -04:00
import "../CommunityComponents"
2020-05-27 18:59:17 -04:00
2021-07-22 17:53:19 +03:00
Item {
2021-08-02 16:38:03 +03:00
id: root
2021-07-22 17:53:19 +03:00
anchors.fill: parent
2020-11-19 14:30:09 -04:00
property alias chatLogView: chatLogView
2021-06-10 15:20:43 -04:00
property alias scrollToMessage: chatLogView . scrollToMessage
2021-07-08 11:20:03 -04:00
2021-07-16 18:02:47 +03:00
property var messageContextMenuInst
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-09-24 11:05:17 -04:00
property int newMessages: 0
2021-08-02 16:38:03 +03:00
property int countOnStartUp: 0
2021-05-26 13:36:24 -04:00
2021-08-02 16:38:03 +03:00
ListView {
id: chatLogView
2021-07-22 17:53:19 +03:00
anchors.fill: parent
2021-08-02 16:38:03 +03:00
anchors.bottomMargin: Style . current . bigPadding
spacing: appSettings . useCompactMode ? 0 : 4
boundsBehavior: Flickable . StopAtBounds
clip: true
flickDeceleration: {
if ( utilsModel . getOs ( ) === Constants . windows ) {
return 5000
2020-09-24 11:05:17 -04:00
}
2021-08-02 16:38:03 +03:00
return 10000
}
verticalLayoutDirection: ListView . BottomToTop
// This header and Connections is to create an invisible padding so that the chat identifier is at the top
// The Connections is necessary, because doing the check inside the header created a binding loop (the contentHeight includes the header height
// If the content height is smaller than the full height, we "show" the padding so that the chat identifier is at the top, otherwise we disable the Connections
header: Item {
height: 0
width: chatLogView . width
}
2020-07-23 16:22:45 -04:00
2021-08-02 16:38:03 +03:00
function checkHeaderHeight ( ) {
if ( ! chatLogView . headerItem ) {
return
2020-07-23 16:22:45 -04:00
}
2020-09-24 11:05:17 -04:00
2021-08-02 16:38:03 +03:00
if ( chatLogView . contentItem . height - chatLogView . headerItem . height < chatLogView . height ) {
chatLogView . headerItem . height = chatLogView . height - ( chatLogView . contentItem . height - chatLogView . headerItem . height ) - 36
} else {
chatLogView . headerItem . height = 0
2020-07-23 16:22:45 -04:00
}
2021-08-02 16:38:03 +03:00
}
2020-09-24 11:05:17 -04:00
2021-08-02 16:38:03 +03:00
property var scrollToMessage: function ( messageId ) {
let item
for ( let i = 0 ; i < messageListDelegate . count ; i ++ ) {
item = messageListDelegate . items . get ( i )
if ( item . model . messageId === messageId ) {
chatLogView . positionViewAtIndex ( i , ListView . Center )
return
2021-06-30 14:46:26 -04:00
}
2020-09-25 15:44:40 -04:00
}
2021-08-02 16:38:03 +03:00
}
ScrollBar.vertical: ScrollBar {
visible: chatLogView . visibleArea . heightRatio < 1
}
2020-07-23 16:22:45 -04:00
2021-08-02 16:38:03 +03:00
Connections {
id: contentHeightConnection
enabled: true
target: chatLogView
onContentHeightChanged: {
chatLogView . checkHeaderHeight ( )
}
onHeightChanged: {
chatLogView . checkHeaderHeight ( )
2020-07-22 14:37:43 -04:00
}
2021-08-02 16:38:03 +03:00
}
2020-07-22 14:37:43 -04:00
2021-08-02 16:38:03 +03:00
Timer {
id: timer
}
2021-06-23 13:30:57 -04:00
2021-08-02 16:38:03 +03:00
Button {
readonly property int buttonPadding: 5
id: scrollDownButton
visible: false
height: 32
width: nbMessages . width + arrowImage . width + 2 * Style . current . halfPadding + ( nbMessages . visible ? scrollDownButton.buttonPadding : 0 )
anchors.bottom: parent . bottom
anchors.right: parent . right
anchors.rightMargin: Style . current . padding
background: Rectangle {
color: Style . current . buttonSecondaryColor
border.width: 0
radius: 16
}
onClicked: {
newMessages = 0
scrollDownButton . visible = false
chatLogView . scrollToBottom ( true )
}
2021-06-23 13:30:57 -04:00
2021-08-02 16:38:03 +03:00
StyledText {
id: nbMessages
visible: newMessages > 0
width: visible ? implicitWidth : 0
text: newMessages
anchors.verticalCenter: parent . verticalCenter
anchors.left: parent . left
color: Style . current . pillButtonTextColor
font.pixelSize: 15
anchors.leftMargin: Style . current . halfPadding
}
2020-06-08 15:25:46 -04:00
2021-08-02 16:38:03 +03:00
SVGImage {
id: arrowImage
width: 24
height: 24
anchors.verticalCenter: parent . verticalCenter
anchors.left: nbMessages . right
source: "../../../img/leave_chat.svg"
anchors.leftMargin: nbMessages . visible ? scrollDownButton.buttonPadding : 0
rotation: - 90
ColorOverlay {
2021-06-30 14:46:26 -04:00
anchors.fill: parent
2021-08-02 16:38:03 +03:00
source: parent
color: Style . current . pillButtonTextColor
2021-06-30 14:46:26 -04:00
}
2020-07-22 11:12:24 -04:00
}
2021-08-02 16:38:03 +03:00
MouseArea {
cursorShape: Qt . PointingHandCursor
anchors.fill: parent
onPressed: mouse . accepted = false
2020-11-30 23:24:01 +02:00
}
2021-08-02 16:38:03 +03:00
}
2020-11-30 23:24:01 +02:00
2021-08-02 16:38:03 +03:00
function scrollToBottom ( force , caller ) {
if ( ! force && ! chatLogView . atYEnd ) {
// User has scrolled up, we don't want to scroll back
return false
}
if ( caller && caller !== chatLogView . itemAtIndex ( chatLogView . count - 1 ) ) {
// If we have a caller, only accept its request if it's the last message
return false
2021-07-08 11:20:03 -04:00
}
2021-08-02 16:38:03 +03:00
// Call this twice and with a timer since the first scroll to bottom might have happened before some stuff loads
// meaning that the scroll will not actually be at the bottom on switch
// Add a small delay because images, even though they say they say they are loaed, they aren't shown yet
Qt . callLater ( chatLogView . positionViewAtBeginning )
timer . setTimeout ( function ( ) {
Qt . callLater ( chatLogView . positionViewAtBeginning )
} , 100 ) ;
return true
}
2021-07-08 11:20:03 -04:00
2021-08-02 16:38:03 +03:00
Connections {
target: chatsModel
2020-07-10 17:47:31 -04:00
2021-08-02 16:38:03 +03:00
onAppReady: {
chatLogView . scrollToBottom ( true )
}
}
2020-12-28 10:45:46 -05:00
2021-08-02 16:38:03 +03:00
Connections {
target: chatsModel . messageView
onMessagesLoaded: {
loadingMessages = false ;
}
2020-12-07 12:37:39 -05:00
2021-08-02 16:38:03 +03:00
onSendingMessage: {
chatLogView . scrollToBottom ( true )
2020-07-10 17:47:31 -04:00
}
2021-02-11 15:37:31 -05:00
2021-08-02 16:38:03 +03:00
onSendingMessageFailed: {
sendingMsgFailedPopup . open ( ) ;
}
2021-02-10 15:37:17 -05:00
2021-08-02 16:38:03 +03:00
onNewMessagePushed: {
if ( ! chatLogView . scrollToBottom ( ) ) {
newMessages ++
2021-06-30 14:46:26 -04:00
}
2021-02-10 15:37:17 -05:00
}
2021-08-02 16:38:03 +03:00
}
2020-06-08 15:25:46 -04:00
2021-08-02 16:38:03 +03:00
Connections {
target: chatsModel . communities
onMembershipRequestChanged: function ( communityId , communityName , accepted ) {
chatColumnLayout . currentNotificationChatId = null
chatColumnLayout . currentNotificationCommunityId = communityId
systemTray . showMessage ( "Status" ,
//% "You have been accepted into the ‘ %1’ community"
accepted ? qsTrId ( "you-have-been-accepted-into-the---1--community" ) . arg ( communityName ) :
//% "Your request to join the ‘ %1’ community was declined"
qsTrId ( "your-request-to-join-the---1--community-was-declined" ) . arg ( communityName ) ,
SystemTrayIcon . NoIcon ,
Constants . notificationPopupTTL )
2020-06-08 15:25:46 -04:00
}
2021-08-02 16:38:03 +03:00
onMembershipRequestPushed: function ( communityId , communityName , pubKey ) {
chatColumnLayout . currentNotificationChatId = null
chatColumnLayout . currentNotificationCommunityId = communityId
//% "New membership request"
systemTray . showMessage ( qsTrId ( "new-membership-request" ) ,
//% "%1 asks to join ‘ %2’ "
qsTrId ( "-1-asks-to-join---2-" ) . arg ( Utils . getDisplayName ( pubKey ) ) . arg ( communityName ) ,
SystemTrayIcon . NoIcon ,
Constants . notificationPopupTTL )
}
2021-06-30 14:46:26 -04:00
}
2021-06-22 14:30:51 -04:00
2021-08-02 16:38:03 +03:00
property var loadMsgs : Backpressure . oneInTime ( chatLogView , 500 , function ( ) {
if ( loadingMessages ) return ;
loadingMessages = true ;
chatsModel . messageView . loadMoreMessages ( ) ;
} ) ;
2021-06-30 14:46:26 -04:00
2021-08-02 16:38:03 +03:00
onContentYChanged: {
scrollDownButton . visible = ( contentHeight - ( scrollY + height ) > 400 )
if ( scrollY < 500 ) {
loadMsgs ( ) ;
2021-07-16 20:00:06 +03:00
}
}
2021-08-02 16:38:03 +03:00
model: messageListDelegate
section.property: "sectionIdentifier"
section.criteria: ViewSection . FullString
}
MessageDialog {
id: sendingMsgFailedPopup
standardButtons: StandardButton . Ok
//% "Failed to send message."
text: qsTrId ( "failed-to-send-message-" )
icon: StandardIcon . Critical
}
Timer {
id: modelLoadingDelayTimer
interval: 1000
onTriggered: {
root . countOnStartUp = messageListDelegate . count ;
}
}
DelegateModelGeneralized {
id: messageListDelegate
lessThan: [
function ( left , right ) { return left . clock > right . clock }
]
model: messageList
delegate: Message {
id: msgDelegate
fromAuthor: model . fromAuthor
chatId: model . chatId
userName: model . userName
alias: model . alias
localName: model . localName
message: model . message
plainText: model . plainText
identicon: model . identicon
isCurrentUser: model . isCurrentUser
timestamp: model . timestamp
sticker: model . sticker
contentType: model . contentType
replaces: model . replaces
isEdited: model . isEdited
outgoingStatus: model . outgoingStatus
responseTo: model . responseTo
authorCurrentMsg: msgDelegate . ListView . section
// The previous message is actually the nextSection since we reversed the list order
authorPrevMsg: msgDelegate . ListView . nextSection
imageClick: imagePopup . openPopup . bind ( imagePopup )
messageId: model . messageId
emojiReactions: model . emojiReactions
linkUrls: model . linkUrls
communityId: model . communityId
hasMention: model . hasMention
stickerPackId: model . stickerPackId
pinnedMessage: model . isPinned
pinnedBy: model . pinnedBy
gapFrom: model . gapFrom
gapTo: model . gapTo
Component.onCompleted: {
if ( ( root . countOnStartUp > 0 ) && ( root . countOnStartUp - 1 ) < index ) {
//new message, increment z order
z = index ;
2021-06-30 14:46:26 -04:00
}
2021-08-02 16:38:03 +03:00
}
messageContextMenu: root . messageContextMenuInst
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 < messageListDelegate . items . count - 1 ) {
return messageListDelegate . items . get ( msgDelegate . DelegateModel . itemsIndex + 1 ) . model . index
2021-06-22 14:30:51 -04:00
}
2021-08-02 16:38:03 +03:00
return - 1 ;
2021-06-22 14:30:51 -04:00
}
2021-08-02 16:38:03 +03:00
nextMessageIndex: {
if ( msgDelegate . DelegateModel . itemsIndex < 1 ) {
return - 1
}
return messageListDelegate . items . get ( msgDelegate . DelegateModel . itemsIndex - 1 ) . model . index
2021-07-16 20:00:06 +03:00
}
2021-08-02 16:38:03 +03:00
scrollToBottom: chatLogView . scrollToBottom
timeout: model . timeout
}
Component.onCompleted: {
modelLoadingDelayTimer . start ( ) ;
2021-06-30 14:46:26 -04:00
}
}
2020-05-27 18:59:17 -04:00
}