2023-09-20 02:17:20 +02:00
// SPDX-FileCopyrightText: Nheko Contributors
//
// SPDX-License-Identifier: GPL-3.0-or-later
import QtQuick
import QtQuick . Controls
import QtQuick . Window
import im . nheko
TimelineEvent {
id: wrapper
ListView.delayRemove: true
width: chat . delegateMaxWidth
2024-01-08 00:21:42 +01:00
// We return a larger size for any item but the most bottom one, if it isn't initialized yet, since otherwise Qt will create way too many items.
// If we did that also for the first item, it would mess with the scroll location a bit, so we don't do it for that item.
height: Math . max ( ( section . item ? . height ? ? 0 ) + ( ( gridContainer . implicitHeight < 1 && index != 0 ) ? 100 : gridContainer . implicitHeight ) + reactionRow . implicitHeight + unreadRow . height , 10 )
2023-09-20 02:17:20 +02:00
anchors.horizontalCenter: ListView . view . contentItem . horizontalCenter
//room: chatRoot.roommodel
required property var day
required property bool isSender
required property int index
property var previousMessageDay: ( index + 1 ) >= chat . count ? 0 : chat . model . dataByIndex ( index + 1 , Room . Day )
property bool previousMessageIsStateEvent: ( index + 1 ) >= chat . count ? true : chat . model . dataByIndex ( index + 1 , Room . IsStateEvent )
property string previousMessageUserId: ( index + 1 ) >= chat . count ? "" : chat . model . dataByIndex ( index + 1 , Room . UserId )
required property date timestamp
required property string userId
required property string userName
required property string threadId
required property int userPowerlevel
required property bool isEdited
required property bool isEncrypted
required property var reactions
required property int status
required property int trustlevel
2023-09-20 02:38:06 +02:00
required property int notificationlevel
2023-09-20 02:17:20 +02:00
required property int type
required property bool isEditable
required property QtObject messageContextMenu
2023-10-10 00:00:17 +02:00
required property QtObject replyContextMenu
2023-09-20 02:17:20 +02:00
required property Item messageActions
property int avatarMargin: ( wrapper . isStateEvent || Settings . smallAvatars ? 0 : ( Nheko . avatarSize + 8 ) ) // align bubble with section header
property alias hovered: messageHover . hovered
2023-10-09 21:28:39 +02:00
mainInset: ( threadId ? ( 4 + Nheko . paddingSmall ) : 0 )
2023-10-08 23:52:23 +02:00
replyInset: mainInset + 4 + Nheko . paddingSmall
2023-10-08 20:14:13 +02:00
maxWidth: chat . delegateMaxWidth - avatarMargin - metadata . width
2023-09-20 02:17:20 +02:00
data: [
Loader {
id: section
active: wrapper . previousMessageUserId !== wrapper . userId || wrapper . previousMessageDay !== wrapper . day || wrapper . previousMessageIsStateEvent !== wrapper . isStateEvent
//asynchronous: true
sourceComponent: TimelineSectionHeader {
day: wrapper . day
isSender: wrapper . isSender
isStateEvent: wrapper . isStateEvent
parentWidth: wrapper . width
previousMessageDay: wrapper . previousMessageDay
previousMessageIsStateEvent: wrapper . previousMessageIsStateEvent
previousMessageUserId: wrapper . previousMessageUserId
timestamp: wrapper . timestamp
userId: wrapper . userId
userName: wrapper . userName
userPowerlevel: wrapper . userPowerlevel
}
visible: status == Loader . Ready
z: 4
} ,
Rectangle {
2024-03-10 22:21:40 +01:00
// this looks better without margins
2023-09-20 02:17:20 +02:00
anchors.fill: gridContainer
color: ( Settings . messageHoverHighlight && messageHover . hovered ) ? palette.alternateBase : "transparent"
2024-03-10 22:21:40 +01:00
// This is partially duplicated by a later handler, however we need this to handle the remaining events around the reply.
2023-09-20 02:17:20 +02:00
TapHandler {
acceptedButtons: Qt . RightButton
acceptedDevices: PointerDevice . Mouse | PointerDevice . Stylus | PointerDevice . TouchPad
gesturePolicy: TapHandler . ReleaseWithinBounds
2024-03-10 22:21:40 +01:00
onSingleTapped: ( event ) = > {
messageContextMenu . show ( wrapper . eventId , wrapper . threadId , wrapper . type , wrapper . isSender , wrapper . isEncrypted , wrapper . isEditable , wrapper . main . hoveredLink , wrapper . main . copyText ) ;
event . accepted = true ;
}
2023-09-20 02:17:20 +02:00
}
} ,
2023-09-20 02:38:06 +02:00
Rectangle {
id: scrollHighlight
anchors.fill: gridContainer
color: palette . highlight
enabled: false
opacity: 0
visible: true
z: 1
states: State {
name: "revealed"
when: wrapper . scrolledToThis
}
transitions: Transition {
from: ""
to: "revealed"
SequentialAnimation {
PropertyAnimation {
duration: 500
easing.type: Easing . InOutQuad
from: 0
properties: "opacity"
target: scrollHighlight
to: 1
}
PropertyAnimation {
duration: 500
easing.type: Easing . InOutQuad
from: 1
properties: "opacity"
target: scrollHighlight
to: 0
}
ScriptAction {
2024-05-27 21:57:26 +02:00
script: {
wrapper . room . eventShown ( ) ;
}
2023-09-20 02:38:06 +02:00
}
}
}
} ,
2023-10-08 20:14:13 +02:00
Row {
2023-09-20 02:17:20 +02:00
id: gridContainer
2023-10-08 20:14:13 +02:00
width: wrapper . width - wrapper . avatarMargin
x: wrapper . avatarMargin
2023-09-20 02:17:20 +02:00
y: section . visible && section . active ? section . y + section.height : 0
2023-10-08 20:14:13 +02:00
spacing: Nheko . paddingSmall
2023-09-20 02:17:20 +02:00
HoverHandler {
id: messageHover
blocking: false
onHoveredChanged: ( ) = > {
if ( ! Settings . mobileMode && hovered ) {
if ( ! messageActions . hovered ) {
messageActions . model = wrapper ;
messageActions . attached = wrapper ;
messageActions . anchors . bottomMargin = - gridContainer . y
2023-10-08 20:14:13 +02:00
messageActions . anchors . rightMargin = metadata . width
2023-09-20 02:17:20 +02:00
}
}
}
}
AbstractButton {
ToolTip.delay: Nheko . tooltipDelay
ToolTip.text: qsTr ( "Part of a thread" )
ToolTip.visible: hovered
2023-10-08 20:14:13 +02:00
height: contentColumn . height
2023-09-20 02:17:20 +02:00
visible: wrapper . threadId
2023-10-08 20:14:13 +02:00
width: 4
2023-09-20 02:17:20 +02:00
onClicked: wrapper . room . thread = wrapper . threadId
Rectangle {
id: threadLine
anchors.fill: parent
color: TimelineManager . userColor ( wrapper . threadId , palette . base )
}
}
2023-10-08 23:52:23 +02:00
Item {
2023-12-22 02:59:16 +01:00
id: stateEventSpacing
2023-10-08 23:52:23 +02:00
visible: wrapper . isStateEvent
width: ( wrapper . maxWidth - ( wrapper . main ? . width ? ? 0 ) ) / 2
height: 1
}
2023-10-08 20:14:13 +02:00
Column {
2023-09-20 02:17:20 +02:00
id: contentColumn
AbstractButton {
id: replyRow
visible: wrapper . reply
2023-10-08 23:52:23 +02:00
height: replyLine . height
2023-09-20 02:17:20 +02:00
property color userColor: TimelineManager . userColor ( wrapper . reply ? . userId ? ? '' , palette . base )
clip: true
NhekoCursorShape {
anchors.fill: parent
cursorShape: Qt . PointingHandCursor
}
2023-10-08 20:14:13 +02:00
contentItem: Row {
2023-09-20 02:17:20 +02:00
id: replyRowLay
2023-10-08 20:14:13 +02:00
spacing: Nheko . paddingSmall
2023-09-20 02:17:20 +02:00
Rectangle {
id: replyLine
2023-10-09 21:41:00 +02:00
height: Math . min ( wrapper . reply ? . height , timelineView . height / 10 ) + Nheko . paddingSmall + replyUserButton . height
2023-09-20 02:17:20 +02:00
color: replyRow . userColor
2023-10-08 20:14:13 +02:00
width: 4
2023-09-20 02:17:20 +02:00
}
2023-10-08 20:14:13 +02:00
Column {
2023-09-20 02:17:20 +02:00
spacing: 0
2023-10-08 20:14:13 +02:00
id: replyCol
2023-09-20 02:17:20 +02:00
AbstractButton {
id: replyUserButton
2023-10-08 20:14:13 +02:00
contentItem: Label {
2023-09-20 02:17:20 +02:00
id: userName_
2023-10-08 20:14:13 +02:00
text: wrapper . reply ? . userName ? ? ''
2023-09-20 02:17:20 +02:00
color: replyRow . userColor
textFormat: Text . RichText
2023-10-08 20:14:13 +02:00
width: wrapper . maxWidth
//elideWidth: wrapper.maxWidth
2023-09-20 02:17:20 +02:00
}
onClicked: wrapper . room . openUserProfile ( wrapper . reply ? . userId )
}
data: [
replyUserButton ,
wrapper . reply ,
]
}
}
background: Rectangle {
2023-10-08 20:14:13 +02:00
//width: replyRow.implicitContentWidth
2023-09-20 02:17:20 +02:00
color: Qt . tint ( palette . base , Qt . hsla ( replyRow . userColor . hslHue , 0.5 , replyRow . userColor . hslLightness , 0.1 ) )
}
onClicked: {
let link = wrapper . reply . hoveredLink
if ( link ) {
Nheko . openLink ( link )
} else {
console . log ( "Scrolling to " + wrapper . replyTo ) ;
wrapper . room . showEvent ( wrapper . replyTo )
}
}
2023-10-10 00:00:17 +02:00
onPressAndHold: wrapper . replyContextMenu . show ( wrapper . reply . copyText ? ? "" , wrapper . reply . linkAt ? wrapper . reply . linkAt ( pressX - replyLine . width - Nheko . paddingSmall , pressY - replyUserButton . implicitHeight ) : "" , wrapper . replyTo )
TapHandler {
acceptedButtons: Qt . RightButton
onSingleTapped: ( eventPoint ) = > wrapper . replyContextMenu . show ( wrapper . reply . copyText ? ? "" , wrapper . reply . linkAt ? wrapper . reply . linkAt ( eventPoint . position . x - replyLine . width - Nheko . paddingSmall , eventPoint . position . y - replyUserButton . implicitHeight ) : "" , wrapper . replyTo )
gesturePolicy: TapHandler . ReleaseWithinBounds
acceptedDevices: PointerDevice . Mouse | PointerDevice . Stylus | PointerDevice . TouchPad
}
2023-09-20 02:17:20 +02:00
}
data: [
replyRow , wrapper . main ,
]
}
2023-10-27 20:19:07 +02:00
DragHandler {
id: replyDragHandler
2023-12-31 20:15:38 +01:00
enabled: ! Settings . disableSwipe
2023-10-27 20:19:07 +02:00
yAxis.enabled: false
xAxis.enabled: true
xAxis.minimum: wrapper . avatarMargin - 100
xAxis.maximum: wrapper . avatarMargin
onActiveChanged: {
if ( ! replyDragHandler . active ) {
if ( replyDragHandler . xAxis . minimum <= replyDragHandler . xAxis . activeValue + 1 ) {
wrapper . room . reply = wrapper . eventId
}
gridContainer . x = wrapper . avatarMargin ;
}
}
}
TapHandler {
onDoubleTapped: wrapper . room . reply = wrapper . eventId
}
2024-03-08 18:43:59 +01:00
} ,
Rectangle {
anchors.top: gridContainer . top
anchors.left: gridContainer . left
anchors.topMargin: - 2
anchors.leftMargin: - 2 + ( stateEventSpacing . visible ? ( stateEventSpacing . width + gridContainer . spacing ) : 0 )
color: "transparent"
border.color: Nheko . theme . red
border.width: wrapper . notificationlevel == MtxEvent . Highlight ? 1 : 0
radius: 4
height: contentColumn . implicitHeight + 4
width: contentColumn . implicitWidth + 4 + ( wrapper . threadId ? ( 4 + gridContainer . spacing ) : 0 )
2023-10-08 20:14:13 +02:00
} ,
2023-10-09 21:28:39 +02:00
TimelineMetadata {
2023-09-20 02:17:20 +02:00
id: metadata
2023-10-09 21:28:39 +02:00
scaling: 1
2023-09-20 02:17:20 +02:00
2023-10-08 20:14:13 +02:00
anchors.right: parent . right
y: section . visible && section . active ? section . y + section.height : 0
2023-10-09 21:28:39 +02:00
visible: ! wrapper . isStateEvent
2023-09-20 02:17:20 +02:00
2023-10-09 21:28:39 +02:00
eventId: wrapper . eventId
status: wrapper . status
trustlevel: wrapper . trustlevel
isEdited: wrapper . isEdited
isEncrypted: wrapper . isEncrypted
threadId: wrapper . threadId
timestamp: wrapper . timestamp
room: wrapper . room
2023-10-08 20:14:13 +02:00
} ,
2024-03-10 22:21:40 +01:00
Item {
// We need this item to grab events, that otherwise would go to the TextArea in the main item. If we don't have this, it would trigger a right click menu on KDE...
// https://invent.kde.org/frameworks/qqc2-desktop-style/-/blob/9d71fe874186009f76d392e203d9fa25a49f8be7/org.kde.desktop/TextArea.qml#L55
anchors.fill: gridContainer
anchors.topMargin: replyRow . height
TapHandler {
acceptedButtons: Qt . RightButton
acceptedDevices: PointerDevice . Mouse | PointerDevice . Stylus | PointerDevice . TouchPad
gesturePolicy: TapHandler . ReleaseWithinBounds
onSingleTapped: ( event ) = > {
messageContextMenu . show ( wrapper . eventId , wrapper . threadId , wrapper . type , wrapper . isSender , wrapper . isEncrypted , wrapper . isEditable , wrapper . main . hoveredLink , wrapper . main . copyText ) ;
}
}
} ,
2023-09-20 02:17:20 +02:00
Reactions {
id: reactionRow
eventId: wrapper . eventId
reactions: wrapper . reactions
width: wrapper . width - wrapper . avatarMargin
x: wrapper . avatarMargin
anchors {
top: gridContainer . bottom
topMargin: - 4
}
} ,
Rectangle {
id: unreadRow
color: palette . highlight
height: visible ? 3 : 0
visible: ( wrapper . index > 0 && ( wrapper . room . fullyReadEventId == wrapper . eventId ) )
anchors {
left: parent . left
right: parent . right
top: reactionRow . bottom
topMargin: 5
}
}
]
}