Add presence details to member profile
This commit is contained in:
		
							
								
								
									
										4
									
								
								TODO.md
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								TODO.md
									
									
									
									
									
								
							| @@ -1,5 +1,9 @@ | |||||||
| # TODO | # TODO | ||||||
|  |  | ||||||
|  | - profile max avatar width | ||||||
|  | - retrieve last seen time for offline members on hover/in profile | ||||||
|  | - auto-away after a configurable idle time  | ||||||
|  | - status based on process detection | ||||||
| - retry if media not found | - retry if media not found | ||||||
| - fix members not synced bug | - fix members not synced bug | ||||||
| - fix local unread counters order | - fix local unread counters order | ||||||
|   | |||||||
| @@ -67,18 +67,86 @@ HListView { | |||||||
|                 iconItem.small: true |                 iconItem.small: true | ||||||
|                 onClicked: profile.stackView.pop() |                 onClicked: profile.stackView.pop() | ||||||
|             } |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         HColumnLayout {  // no spacing between these children | ||||||
|  |             HLabel { | ||||||
|  |                 wrapMode: HLabel.Wrap | ||||||
|  |                 horizontalAlignment: Qt.AlignHCenter | ||||||
|  |                 color: utils.nameColor(member.display_name || member.id) | ||||||
|  |                 text: member.display_name.trim() || member.id | ||||||
|  |  | ||||||
|  |                 Layout.fillWidth: true | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             HLabel { | ||||||
|  |                 wrapMode: HLabel.Wrap | ||||||
|  |                 horizontalAlignment: Qt.AlignHCenter | ||||||
|  |                 color: theme.colors.dimText | ||||||
|  |                 text: member.id | ||||||
|  |                 visible: member.display_name.trim() !== "" | ||||||
|  |  | ||||||
|  |                 Layout.fillWidth: true | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         HColumnLayout { | ||||||
|  |             HLabel { | ||||||
|  |                 wrapMode: HLabel.Wrap | ||||||
|  |                 horizontalAlignment: Qt.AlignHCenter | ||||||
|  |  | ||||||
|  |                 text: | ||||||
|  |                     member.presence === "online" ? qsTr("Online") : | ||||||
|  |                     member.presence === "unavailable" ? qsTr("Unavailable") : | ||||||
|  |                     member.presence === "invisible" ? qsTr("Invisible") : | ||||||
|  |                     qsTr("Offline / Unknown") | ||||||
|  |  | ||||||
|  |                 color: | ||||||
|  |                     member.presence === "online" ? | ||||||
|  |                     theme.colors.positiveText : | ||||||
|  |  | ||||||
|  |                     member.presence === "unavailable" ? | ||||||
|  |                     theme.colors.warningText : | ||||||
|  |  | ||||||
|  |                     theme.colors.halfDimText | ||||||
|  |  | ||||||
|  |                 Layout.fillWidth: true | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             HLabel { | ||||||
|  |                 wrapMode: HLabel.Wrap | ||||||
|  |                 horizontalAlignment: Qt.AlignHCenter | ||||||
|  |                 visible: ! member.currently_active && text !== "" | ||||||
|  |                 color: theme.colors.dimText | ||||||
|  |  | ||||||
|  |                 Timer { | ||||||
|  |                     repeat: true | ||||||
|  |                     triggeredOnStart: true | ||||||
|  |  | ||||||
|  |                     running: | ||||||
|  |                         ! member.currently_active && | ||||||
|  |                         member.last_active_at > new Date(1) | ||||||
|  |  | ||||||
|  |                     interval: | ||||||
|  |                         new Date() - member.last_active_at < 60000 ? | ||||||
|  |                         1000 : | ||||||
|  |                         60000 | ||||||
|  |  | ||||||
|  |                     onTriggered: parent.text = Qt.binding(() => | ||||||
|  |                         qsTr("Last seen %1 ago").arg(utils.formatRelativeTime( | ||||||
|  |                             new Date() - member.last_active_at, false, | ||||||
|  |                         )) | ||||||
|  |                     ) | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 Layout.fillWidth: true | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         HLabel { |         HLabel { | ||||||
|             textFormat: HLabel.StyledText |  | ||||||
|             wrapMode: HLabel.Wrap |             wrapMode: HLabel.Wrap | ||||||
|             horizontalAlignment: Qt.AlignHCenter |             text: member.status_msg.trim() | ||||||
|             text: |             visible: text !== "" | ||||||
|                 utils.coloredNameHtml(member.display_name, member.user_id) + |  | ||||||
|                 (member.display_name.trim() ? |  | ||||||
|                  `<br><font color="${theme.colors.dimText}">${member.id}</font>` : |  | ||||||
|                  "") |  | ||||||
|  |  | ||||||
|             Layout.fillWidth: true |             Layout.fillWidth: true | ||||||
|             Layout.bottomMargin: theme.spacing |             Layout.bottomMargin: theme.spacing | ||||||
|   | |||||||
| @@ -328,26 +328,47 @@ QtObject { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     function formatRelativeTime(milliseconds) { |     function formatRelativeTime(milliseconds, shortForm=true) { | ||||||
|         const seconds = Math.floor(milliseconds / 1000) |         const seconds = Math.floor(milliseconds / 1000) | ||||||
|  |  | ||||||
|  |         if (shortForm) { | ||||||
|  |             return ( | ||||||
|  |                 seconds < 60 ? | ||||||
|  |                 qsTr("%1s").arg(seconds) : | ||||||
|  |  | ||||||
|  |                 seconds < 60 * 60 ? | ||||||
|  |                 qsTr("%1mn").arg(Math.floor(seconds / 60)) : | ||||||
|  |  | ||||||
|  |                 seconds < 60 * 60 * 24 ? | ||||||
|  |                 qsTr("%1h").arg(Math.floor(seconds / 60 / 60)) : | ||||||
|  |  | ||||||
|  |                 seconds < 60 * 60 * 24 * 30 ? | ||||||
|  |                 qsTr("%1d").arg(Math.floor(seconds / 60 / 60 / 24)) : | ||||||
|  |  | ||||||
|  |                 seconds < 60 * 60 * 24 * 30 * 12 ? | ||||||
|  |                 qsTr("%1mo").arg(Math.floor(seconds / 60 / 60 / 24 / 30)) : | ||||||
|  |  | ||||||
|  |                 qsTr("%1y").arg(Math.floor(seconds / 60 / 60 / 24 / 30 / 12)) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return ( |         return ( | ||||||
|             seconds < 60 ? |             seconds < 60 ? | ||||||
|             qsTr("%1s").arg(seconds) : |             qsTr("%1 seconds").arg(seconds) : | ||||||
|  |  | ||||||
|             seconds < 60 * 60 ? |             seconds < 60 * 60 ? | ||||||
|             qsTr("%1mn").arg(Math.floor(seconds / 60)) : |             qsTr("%1 minutes").arg(Math.floor(seconds / 60)) : | ||||||
|  |  | ||||||
|             seconds < 60 * 60 * 24 ? |             seconds < 60 * 60 * 24 ? | ||||||
|             qsTr("%1h").arg(Math.floor(seconds / 60 / 60)) : |             qsTr("%1 hours").arg(Math.floor(seconds / 60 / 60)) : | ||||||
|  |  | ||||||
|             seconds < 60 * 60 * 24 * 30 ? |             seconds < 60 * 60 * 24 * 30 ? | ||||||
|             qsTr("%1d").arg(Math.floor(seconds / 60 / 60 / 24)) : |             qsTr("%1 days").arg(Math.floor(seconds / 60 / 60 / 24)) : | ||||||
|  |  | ||||||
|             seconds < 60 * 60 * 24 * 30 * 12 ? |             seconds < 60 * 60 * 24 * 30 * 12 ? | ||||||
|             qsTr("%1mo").arg(Math.floor(seconds / 60 / 60 / 24 / 30)) : |             qsTr("%1 months").arg(Math.floor(seconds / 60 / 60 / 24 / 30)) : | ||||||
|  |  | ||||||
|             qsTr("%1y").arg(Math.floor(seconds / 60 / 60 / 24 / 30 / 12)) |             qsTr("%1 years").arg(Math.floor(seconds / 60 / 60 / 24 / 30 / 12)) | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	