Order the roomEvents models from newest to oldest
Qt somehow handles scrolling on new messages on its own when the ListView direction is bottom to top. In normal top to bottom, manual scrolling is completly buggy.
This commit is contained in:
parent
7e62da8733
commit
f0dab1801a
@ -96,19 +96,17 @@ class SignalManager(QObject):
|
|||||||
date_time = QDateTime.fromMSecsSinceEpoch(edict["server_timestamp"])
|
date_time = QDateTime.fromMSecsSinceEpoch(edict["server_timestamp"])
|
||||||
new_event = RoomEvent(type=etype, date_time=date_time, dict=edict)
|
new_event = RoomEvent(type=etype, date_time=date_time, dict=edict)
|
||||||
|
|
||||||
# Insert event in model at the right position, based on timestamps
|
# Model is sorted from newest to oldest message
|
||||||
# to keep them sorted by date of arrival.
|
insert_at = None
|
||||||
# Iterate in reverse, since a new event is more likely to be appended,
|
for i, event in enumerate(model):
|
||||||
# but events can arrive out of order.
|
if new_event.date_time > event.date_time:
|
||||||
if not model or model[-1].date_time < new_event.date_time:
|
insert_at = i
|
||||||
|
break
|
||||||
|
|
||||||
|
if insert_at is None:
|
||||||
model.append(new_event)
|
model.append(new_event)
|
||||||
else:
|
else:
|
||||||
for i, event in enumerate(reversed(model)):
|
model.insert(insert_at, new_event)
|
||||||
if event.date_time < new_event.date_time:
|
|
||||||
model.insert(-i, new_event)
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
model.insert(0, new_event)
|
|
||||||
|
|
||||||
|
|
||||||
def onRoomTypingUsersUpdated(
|
def onRoomTypingUsersUpdated(
|
||||||
|
@ -5,8 +5,8 @@ Base.HLabel {
|
|||||||
text: date_time.toLocaleDateString()
|
text: date_time.toLocaleDateString()
|
||||||
width: rootCol.width
|
width: rootCol.width
|
||||||
horizontalAlignment: Text.AlignHCenter
|
horizontalAlignment: Text.AlignHCenter
|
||||||
topPadding: rootCol.isFirstMessage ? 0 : rootCol.verticalPadding * 4
|
topPadding: rootCol.isFirstMessage ? 0 : rootCol.standardSpacing
|
||||||
bottomPadding: rootCol.verticalPadding * 2
|
bottomPadding: rootCol.standardSpacing
|
||||||
font.pixelSize: normalSize * 1.1
|
font.pixelSize: normalSize * 1.1
|
||||||
color: "darkolivegreen"
|
color: "darkolivegreen"
|
||||||
}
|
}
|
||||||
|
@ -50,6 +50,7 @@ Row {
|
|||||||
|
|
||||||
leftPadding: horizontalPadding
|
leftPadding: horizontalPadding
|
||||||
rightPadding: horizontalPadding
|
rightPadding: horizontalPadding
|
||||||
|
topPadding: nameLabel.visible ? 0 : verticalPadding
|
||||||
bottomPadding: verticalPadding
|
bottomPadding: verticalPadding
|
||||||
|
|
||||||
Layout.minimumWidth: nameLabel.implicitWidth
|
Layout.minimumWidth: nameLabel.implicitWidth
|
||||||
|
@ -4,13 +4,20 @@ import QtQuick.Layouts 1.4
|
|||||||
import "../base" as Base
|
import "../base" as Base
|
||||||
|
|
||||||
Column {
|
Column {
|
||||||
id: rootCol
|
id: "rootCol"
|
||||||
|
|
||||||
function mins_between(date1, date2) {
|
function mins_between(date1, date2) {
|
||||||
|
console.log(Math.round((((date2 - date1) % 86400000) % 3600000) / 60000))
|
||||||
return Math.round((((date2 - date1) % 86400000) % 3600000) / 60000)
|
return Math.round((((date2 - date1) % 86400000) % 3600000) / 60000)
|
||||||
}
|
}
|
||||||
|
|
||||||
readonly property bool isMessage: type.startsWith("RoomMessage")
|
function is_message(type_) { return type_.startsWith("RoomMessage") }
|
||||||
|
|
||||||
|
readonly property var previousItem:
|
||||||
|
index < messageListView.model.count - 1 ?
|
||||||
|
messageListView.model.get(index + 1) : null
|
||||||
|
|
||||||
|
readonly property bool isMessage: is_message(type)
|
||||||
|
|
||||||
readonly property bool isUndecryptableEvent:
|
readonly property bool isUndecryptableEvent:
|
||||||
type === "OlmEvent" || type === "MegolmEvent"
|
type === "OlmEvent" || type === "MegolmEvent"
|
||||||
@ -21,37 +28,38 @@ Column {
|
|||||||
readonly property bool isOwn:
|
readonly property bool isOwn:
|
||||||
chatPage.user_id === dict.sender
|
chatPage.user_id === dict.sender
|
||||||
|
|
||||||
readonly property var previousData:
|
readonly property bool isFirstMessage: ! previousItem
|
||||||
index > 0 ? messageListView.model.get(index - 1) : null
|
|
||||||
|
|
||||||
readonly property bool isFirstMessage: ! previousData
|
|
||||||
|
|
||||||
readonly property bool combine:
|
readonly property bool combine:
|
||||||
! isFirstMessage &&
|
! isFirstMessage &&
|
||||||
previousData.isMessage === isMessage &&
|
! talkBreak &&
|
||||||
previousData.dict.sender === dict.sender &&
|
! dayBreak &&
|
||||||
mins_between(previousData.date_time, date_time) <= 5
|
is_message(previousItem.type) === isMessage &&
|
||||||
|
previousItem.dict.sender === dict.sender &&
|
||||||
|
mins_between(previousItem.date_time, date_time) <= 5
|
||||||
|
|
||||||
readonly property bool dayBreak:
|
readonly property bool dayBreak:
|
||||||
isFirstMessage ||
|
isFirstMessage ||
|
||||||
previousData.date_time.getDay() != date_time.getDay()
|
date_time.getDay() != previousItem.date_time.getDay()
|
||||||
|
|
||||||
readonly property bool talkBreak:
|
readonly property bool talkBreak:
|
||||||
! isFirstMessage &&
|
! isFirstMessage &&
|
||||||
! dayBreak &&
|
! dayBreak &&
|
||||||
mins_between(previousData.date_time, date_time) >= 20
|
mins_between(previousItem.date_time, date_time) >= 20
|
||||||
|
|
||||||
|
|
||||||
property int standardSpacing: 8
|
property int standardSpacing: 16
|
||||||
property int horizontalPadding: 7
|
property int horizontalPadding: 7
|
||||||
property int verticalPadding: 5
|
property int verticalPadding: 5
|
||||||
|
|
||||||
width: parent.width
|
width: parent.width
|
||||||
topPadding:
|
topPadding:
|
||||||
previousData === null ? 0 :
|
! previousItem ? 0 :
|
||||||
talkBreak ? standardSpacing * 6 :
|
talkBreak ? standardSpacing * 3 :
|
||||||
combine ? standardSpacing / 2 :
|
combine ? standardSpacing / 4 :
|
||||||
standardSpacing * 1.2
|
standardSpacing
|
||||||
|
|
||||||
|
//Text { text: rootCol.topPadding }
|
||||||
|
|
||||||
Daybreak { visible: dayBreak }
|
Daybreak { visible: dayBreak }
|
||||||
|
|
||||||
|
@ -13,18 +13,27 @@ Rectangle {
|
|||||||
ListView {
|
ListView {
|
||||||
id: messageListView
|
id: messageListView
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
model: Backend.models.roomEvents.get(chatPage.room.room_id)
|
|
||||||
delegate: MessageDelegate {}
|
delegate: MessageDelegate {}
|
||||||
|
model: Backend.models.roomEvents.get(chatPage.room.room_id)
|
||||||
//highlight: Rectangle {color: "lightsteelblue"; radius: 5}
|
//highlight: Rectangle {color: "lightsteelblue"; radius: 5}
|
||||||
|
|
||||||
clip: true
|
clip: true
|
||||||
topMargin: space
|
topMargin: space
|
||||||
bottomMargin: space
|
bottomMargin: space
|
||||||
|
verticalLayoutDirection: ListView.BottomToTop
|
||||||
|
|
||||||
// Keep x scroll pages cached, to limit images having to be
|
// Keep x scroll pages cached, to limit images having to be
|
||||||
// reloaded from network.
|
// reloaded from network.
|
||||||
cacheBuffer: height * 6
|
cacheBuffer: height * 6
|
||||||
|
|
||||||
//Component.onCompleted: positionViewAtEnd()
|
function goToEnd() {
|
||||||
|
messageListView.positionViewAtEnd()
|
||||||
|
//messageListView.flick(0, -messageListView.bottomMargin * 100)
|
||||||
|
}
|
||||||
|
|
||||||
|
//Connections {
|
||||||
|
//target: messageListView.model
|
||||||
|
//onChanged: goToEnd()
|
||||||
|
//}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
function get_last_room_event_text(room_id) {
|
function get_last_room_event_text(room_id) {
|
||||||
var eventsModel = Backend.models.roomEvents.get(room_id)
|
var eventsModel = Backend.models.roomEvents.get(room_id)
|
||||||
|
|
||||||
for (var i = -1; i >= -eventsModel.count; i--) {
|
for (var i = 0; i < eventsModel.count; i++) {
|
||||||
var ev = eventsModel.get(i)
|
var ev = eventsModel.get(i)
|
||||||
|
|
||||||
if (ev.type !== "RoomMemberEvent") {
|
if (ev.type !== "RoomMemberEvent") {
|
||||||
|
Loading…
Reference in New Issue
Block a user