Fix layout - no more binding loops (hopefully)

This commit is contained in:
Malte E 2022-02-11 22:02:30 +01:00
parent 0e548b7d39
commit d0ac110cb7
10 changed files with 35 additions and 60 deletions

View File

@ -33,7 +33,7 @@ ScrollView {
//reuseItems: true //reuseItems: true
boundsBehavior: Flickable.StopAtBounds boundsBehavior: Flickable.StopAtBounds
pixelAligned: true pixelAligned: true
spacing: 4 spacing: 2
verticalLayoutDirection: ListView.BottomToTop verticalLayoutDirection: ListView.BottomToTop
onCountChanged: { onCountChanged: {
// Mark timeline as read // Mark timeline as read
@ -464,7 +464,7 @@ ScrollView {
property date timestamp: wrapper.timestamp property date timestamp: wrapper.timestamp
z: 4 z: 4
active: previousMessageUserId !== undefined && previousMessageUserId !== userId || previousMessageDay !== day active: previousMessageUserId !== undefined && previousMessageUserId !== userId || previousMessageDay !== day || previousMessageIsStateEvent !== isStateEvent
//asynchronous: true //asynchronous: true
sourceComponent: sectionHeader sourceComponent: sectionHeader
visible: status == Loader.Ready visible: status == Loader.Ready

View File

@ -47,6 +47,7 @@ Rectangle {
userId: modelData.userId ?? "" userId: modelData.userId ?? ""
userName: modelData.userName ?? "" userName: modelData.userName ?? ""
encryptionError: modelData.encryptionError ?? "" encryptionError: modelData.encryptionError ?? ""
width: parent.width
} }
ImageButton { ImageButton {

View File

@ -47,7 +47,7 @@ Item {
anchors.left: parent.left anchors.left: parent.left
anchors.right: parent.right anchors.right: parent.right
height: row.height+reactionRow.height+(Settings.bubbles? 8 : 4) height: row.height+reactionRow.height
Rectangle { Rectangle {
color: (Settings.messageHoverHighlight && hoverHandler.hovered) ? Nheko.colors.alternateBase : "transparent" color: (Settings.messageHoverHighlight && hoverHandler.hovered) ? Nheko.colors.alternateBase : "transparent"
@ -78,30 +78,19 @@ Item {
anchors.rightMargin: 1 anchors.rightMargin: 1
anchors.leftMargin: Nheko.avatarSize + 12 // align bubble with section header anchors.leftMargin: Nheko.avatarSize + 12 // align bubble with section header
anchors.left: parent.left anchors.left: parent.left
//anchors.right: parent.right
property int maxWidth: parent.width-anchors.leftMargin-anchors.rightMargin property int maxWidth: parent.width-anchors.leftMargin-anchors.rightMargin
width: Math.min(maxWidth,msg.implicitWidth) width: Settings.bubbles? Math.min(maxWidth,implicitWidth+metadata.width) : maxWidth
height: msg.height padding: isStateEvent? 0 : 3
topInset: -4
bottomInset: -4
leftInset: -4
rightInset: -4
background: Rectangle { background: Rectangle {
//anchors.fill: msg
property color userColor: TimelineManager.userColor(userId, Nheko.colors.base) property color userColor: TimelineManager.userColor(userId, Nheko.colors.base)
property color bgColor: Nheko.colors.base property color bgColor: Nheko.colors.base
color: Qt.rgba(userColor.r*0.1+bgColor.r*0.9,userColor.g*0.1+bgColor.g*0.9,userColor.b*0.1+bgColor.b*0.9,1) //TimelineManager.userColor(userId, Nheko.colors.base) color: Qt.tint(bgColor, Qt.rgba(userColor.r, userColor.g, userColor.b, 0.2))
radius: 4 radius: parent.padding*2
visible: Settings.bubbles && !isStateEvent visible: Settings.bubbles && !isStateEvent
} }
GridLayout { contentItem: GridLayout {
id: msg id: msg
anchors {
right: parent.right
left: parent.left
top: parent.top
}
property bool narrowLayout: Settings.bubbles //&& (timelineView.width < 500) // timelineView causes fewew binding loops than r. But maybe it shouldn't depend on width anyway property bool narrowLayout: Settings.bubbles //&& (timelineView.width < 500) // timelineView causes fewew binding loops than r. But maybe it shouldn't depend on width anyway
rowSpacing: 0 rowSpacing: 0
columnSpacing: 2 columnSpacing: 2
@ -114,6 +103,8 @@ Item {
Layout.column: 0 Layout.column: 0
Layout.fillWidth: true Layout.fillWidth: true
Layout.bottomMargin: visible? 2 : 0 Layout.bottomMargin: visible? 2 : 0
Layout.preferredHeight: height
Layout.maximumWidth: implicitWidth
id: reply id: reply
function fromModel(role) { function fromModel(role) {
@ -142,7 +133,6 @@ Item {
callType: r.relatedEventCacheBuster, fromModel(Room.CallType) ?? "" callType: r.relatedEventCacheBuster, fromModel(Room.CallType) ?? ""
encryptionError: r.relatedEventCacheBuster, fromModel(Room.EncryptionError) ?? "" encryptionError: r.relatedEventCacheBuster, fromModel(Room.EncryptionError) ?? ""
relatedEventCacheBuster: r.relatedEventCacheBuster, fromModel(Room.RelatedEventCacheBuster) ?? 0 relatedEventCacheBuster: r.relatedEventCacheBuster, fromModel(Room.RelatedEventCacheBuster) ?? 0
maxWidth: row.maxWidth
} }
// actual message content // actual message content
@ -150,6 +140,8 @@ Item {
Layout.row: 1 Layout.row: 1
Layout.column: 0 Layout.column: 0
Layout.fillWidth: true Layout.fillWidth: true
Layout.preferredHeight: height
Layout.maximumWidth: implicitWidth
id: contentItem id: contentItem
blurhash: r.blurhash blurhash: r.blurhash
@ -174,7 +166,6 @@ Item {
encryptionError: r.encryptionError encryptionError: r.encryptionError
relatedEventCacheBuster: r.relatedEventCacheBuster relatedEventCacheBuster: r.relatedEventCacheBuster
isReply: false isReply: false
maxWidth: row.maxWidth
} }
RowLayout { RowLayout {
@ -182,7 +173,7 @@ Item {
Layout.column: msg.narrowLayout? 0 : 1 Layout.column: msg.narrowLayout? 0 : 1
Layout.row: msg.narrowLayout? 2 : 0 Layout.row: msg.narrowLayout? 2 : 0
Layout.rowSpan: msg.narrowLayout? 1 : 2 Layout.rowSpan: msg.narrowLayout? 1 : 2
Layout.bottomMargin: msg.narrowLayout? -4 : 0 Layout.bottomMargin: -4
Layout.alignment: Qt.AlignTop | Qt.AlignRight Layout.alignment: Qt.AlignTop | Qt.AlignRight
Layout.preferredWidth: implicitWidth Layout.preferredWidth: implicitWidth
visible: !isStateEvent visible: !isStateEvent

View File

@ -11,11 +11,10 @@ Item {
required property string eventId required property string eventId
required property string filename required property string filename
required property string filesize required property string filesize
property int maxWidth
height: row.height + 24 height: row.height + 24
width: maxWidth width: parent.width
implicitWidth: maxWidth implicitWidth: row.implicitWidth
RowLayout { RowLayout {
id: row id: row

View File

@ -17,16 +17,11 @@ Item {
required property string filename required property string filename
required property bool isReply required property bool isReply
required property string eventId required property string eventId
property int maxWidth
property double tempWidth: Math.min(maxWidth, originalWidth < 1 ? 200 : originalWidth)
property double tempHeight: tempWidth * proportionalHeight
property double divisor: isReply ? 5 : 3 property double divisor: isReply ? 5 : 3
property bool tooHigh: tempHeight > timelineView.height / divisor
height: Math.round(tooHigh ? timelineView.height / divisor : tempHeight) implicitWidth: Math.round(originalWidth*Math.min((timelineView.height/divisor)/(originalWidth*proportionalHeight), 1))
width: Math.round(tooHigh ? (timelineView.height / divisor) / proportionalHeight : tempWidth) width: parent.width
implicitHeight: height height: width*proportionalHeight
implicitWidth: width
Image { Image {
id: blurhash_ id: blurhash_

View File

@ -35,9 +35,8 @@ Item {
required property string callType required property string callType
required property int encryptionError required property int encryptionError
required property int relatedEventCacheBuster required property int relatedEventCacheBuster
property int maxWidth
Layout.preferredHeight: chooser.child ? chooser.child.height : Nheko.paddingLarge height: chooser.child ? chooser.child.height : Nheko.paddingLarge
DelegateChooser { DelegateChooser {
id: chooser id: chooser
@ -46,8 +45,7 @@ Item {
roleValue: type roleValue: type
//anchors.fill: parent //anchors.fill: parent
anchors.left: parent.left width: parent.width? parent.width: 0 // this should get rid of "cannot read property 'width' of null"
anchors.right: parent.right
DelegateChoice { DelegateChoice {
roleValue: MtxEvent.UnknownMessage roleValue: MtxEvent.UnknownMessage
@ -111,7 +109,6 @@ Item {
filename: d.filename filename: d.filename
isReply: d.isReply isReply: d.isReply
eventId: d.eventId eventId: d.eventId
maxWidth: d.maxWidth
} }
} }
@ -129,7 +126,6 @@ Item {
filename: d.filename filename: d.filename
isReply: d.isReply isReply: d.isReply
eventId: d.eventId eventId: d.eventId
maxWidth: d.maxWidth
} }
} }
@ -141,7 +137,6 @@ Item {
eventId: d.eventId eventId: d.eventId
filename: d.filename filename: d.filename
filesize: d.filesize filesize: d.filesize
maxWidth: d.maxWidth
} }
} }
@ -158,7 +153,6 @@ Item {
url: d.url url: d.url
body: d.body body: d.body
filesize: d.filesize filesize: d.filesize
maxWidth: d.maxWidth
} }
} }
@ -175,7 +169,6 @@ Item {
url: d.url url: d.url
body: d.body body: d.body
filesize: d.filesize filesize: d.filesize
maxWidth: d.maxWidth
} }
} }
@ -184,7 +177,7 @@ Item {
roleValue: MtxEvent.Redacted roleValue: MtxEvent.Redacted
Redacted { Redacted {
delegateWidth: d.width //delegateWidth: d.width
} }
} }

View File

@ -11,8 +11,8 @@ Label {
property bool isStateEvent property bool isStateEvent
color: Nheko.colors.text color: Nheko.colors.text
horizontalAlignment: Text.AlignHCenter horizontalAlignment: Text.AlignHCenter
height: contentHeight * 1.2 //height: contentHeight * 1.2
width: contentWidth * 1.2 //width: contentWidth * 1.2
font.pointSize: isStateEvent? 0.75*fontMetrics.font.pointSize : 1*fontMetrics.font.pointSize font.pointSize: isStateEvent? 0.75*fontMetrics.font.pointSize : 1*fontMetrics.font.pointSize
background: Rectangle { background: Rectangle {

View File

@ -22,16 +22,11 @@ Item {
required property string url required property string url
required property string body required property string body
required property string filesize required property string filesize
property int maxWidth
property double tempWidth: Math.min(maxWidth, originalWidth < 1 ? 400 : originalWidth)
property double tempHeight: tempWidth * proportionalHeight
property double divisor: isReply ? 4 : 2 property double divisor: isReply ? 4 : 2
property bool tooHigh: tempHeight > timelineRoot.height / divisor implicitWidth: type == MtxEvent.VideoMessage ? Math.round(originalWidth*Math.min((timelineView.height/divisor)/(originalWidth*proportionalHeight), 1)) : 500
width: parent.width
height: (type == MtxEvent.VideoMessage ? tooHigh ? timelineRoot.height / divisor : tempHeight : 80) + fileInfoLabel.height height: (type == MtxEvent.VideoMessage ? width*proportionalHeight : 80) + fileInfoLabel.height
width: type == MtxEvent.VideoMessage ? tooHigh ? (timelineRoot.height / divisor) / proportionalHeight : tempWidth : 250
implicitHeight: height implicitHeight: height
implicitWidth: width
MxcMedia { MxcMedia {
id: mxcmedia id: mxcmedia

View File

@ -10,10 +10,10 @@ import im.nheko 1.0
Rectangle{ Rectangle{
required property real delegateWidth //required property real delegateWidth
height: redactedLayout.implicitHeight + Nheko.paddingSmall height: redactedLayout.implicitHeight + Nheko.paddingSmall
width: redactedLayout.implicitWidth + 2 * Nheko.paddingMedium implicitWidth: redactedLayout.implicitWidth + 2 * Nheko.paddingMedium
implicitWidth: width //implicitWidth: width
radius: fontMetrics.lineSpacing / 2 + 2 * Nheko.paddingSmall radius: fontMetrics.lineSpacing / 2 + 2 * Nheko.paddingSmall
color: Nheko.colors.alternateBase color: Nheko.colors.alternateBase
@ -33,8 +33,9 @@ Rectangle{
id: redactedLabel id: redactedLabel
Layout.margins: 0 Layout.margins: 0
Layout.alignment: Qt.AlignVCenter | Qt.AlignRight Layout.alignment: Qt.AlignVCenter | Qt.AlignRight
Layout.fillWidth: true Layout.preferredWidth: implicitWidth
Layout.maximumWidth: delegateWidth - 4 * Nheko.paddingSmall - trashImg.width - 2 * Nheko.paddingMedium //Layout.fillWidth: true
//Layout.maximumWidth: delegateWidth - 4 * Nheko.paddingSmall - trashImg.width - 2 * Nheko.paddingMedium
property var redactedPair: room.formatRedactedEvent(eventId) property var redactedPair: room.formatRedactedEvent(eventId)
text: redactedPair["first"] text: redactedPair["first"]
wrapMode: Label.WordWrap wrapMode: Label.WordWrap

View File

@ -37,8 +37,8 @@ Item {
property int relatedEventCacheBuster property int relatedEventCacheBuster
property int maxWidth property int maxWidth
Layout.preferredHeight: replyContainer.height
height: replyContainer.height height: replyContainer.height
implicitHeight: replyContainer.height
implicitWidth: visible? colorLine.width+replyContainer.implicitWidth : 0 implicitWidth: visible? colorLine.width+replyContainer.implicitWidth : 0
CursorShape { CursorShape {
@ -99,6 +99,7 @@ Item {
MessageDelegate { MessageDelegate {
Layout.leftMargin: 4 Layout.leftMargin: 4
Layout.preferredHeight: height
id: reply id: reply
blurhash: r.blurhash blurhash: r.blurhash
body: r.body body: r.body
@ -125,7 +126,6 @@ Item {
enabled: false enabled: false
Layout.fillWidth: true Layout.fillWidth: true
isReply: true isReply: true
maxWidth: r.maxWidth-4
} }
} }