feat(LinkPreviews): Integrate Link previews with the backend (#12523)
* feat(StatusQ): Adding numberToLocaleStringInCompactForm function to LocaleUtils This function will format the number in a compact form E.g: 1000 -> 1K; 1000000 -> 1M; 1100000 -> 1.1M + adding tests fix(statusQ): Update numberToLocaleStringInCompactForm to return the locale number when greter than 999T fix(StatusQ): extend the test_numberToLocaleStringInCompactForm with new data * feat(LinkPreviews): Update the link preview area in StatusChatInput to use the new model Changes: 1. Create a new component `LinkPreviewMiniCardDelegate.qml` that filters the model data to properly fill the link preview card with the needed data based on the preview type 2. Update storybook pages 3. Small updates to LinkPreviewMiniCard * feat(LinkPreviews): Update the link previews in message history to use the new backend Changes: 1. Create delegate items for LinkPreviewCard and gif link preview to clean the LinksMessageView component and filter the model data based on the preview type 2. Remove UserProfileCard and reuse the LinkPreviewCard to display contacts link previews 3. Update LinkPreviewCard so that it can accommodate status link previews (communities, channels, contacts). The generic properties (title, description, footer) have been dropped and replaced with specialised properties for each preview type. 4. Fix LinkPreviewCard layout to better accommodate different content variants (missing properties, long/short title, missing description, missing icon) 5. Fixing the link preview context menu and click actions fix: Move inline components to separate files Fixing the linux builds using Qt 5.15.2 affected by this bug: https://bugreports.qt.io/browse/QTBUG-89180 * fix: Align LinkPreviewMiniCard implementation with LinkPreviewCard and remove state based model filtering
This commit is contained in:
parent
5c4dd60f1e
commit
4a30d13bdc
|
@ -4,6 +4,7 @@ import QtQuick.Layouts 1.15
|
||||||
import QtGraphicalEffects 1.15
|
import QtGraphicalEffects 1.15
|
||||||
|
|
||||||
import Storybook 1.0
|
import Storybook 1.0
|
||||||
|
import Models 1.0
|
||||||
|
|
||||||
import StatusQ.Core.Theme 0.1
|
import StatusQ.Core.Theme 0.1
|
||||||
import shared.controls.chat 1.0
|
import shared.controls.chat 1.0
|
||||||
|
@ -30,7 +31,7 @@ SplitView {
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
width: parent.width
|
width: parent.width
|
||||||
imagePreviewArray: ["https://picsum.photos/200/300?random=1", "https://picsum.photos/200/300?random=1"]
|
imagePreviewArray: ["https://picsum.photos/200/300?random=1", "https://picsum.photos/200/300?random=1"]
|
||||||
linkPreviewModel: showLinkPreviewSettings ? emptyModel : linkPreviewListModel
|
linkPreviewModel: showLinkPreviewSettings ? emptyModel : mockedLinkPreviewModel
|
||||||
showLinkPreviewSettings: !linkPreviewEnabledSwitch.checked
|
showLinkPreviewSettings: !linkPreviewEnabledSwitch.checked
|
||||||
visible: hasContent
|
visible: hasContent
|
||||||
|
|
||||||
|
@ -78,123 +79,8 @@ SplitView {
|
||||||
id: emptyModel
|
id: emptyModel
|
||||||
}
|
}
|
||||||
|
|
||||||
ListModel {
|
LinkPreviewModel {
|
||||||
id: linkPreviewListModel
|
id: mockedLinkPreviewModel
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
immutable: false
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=1"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: false
|
|
||||||
immutable: false
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=2"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
immutable: false
|
|
||||||
hostname: ""
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=3"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=4"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
immutable: false
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=5"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=6"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
immutable: false
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=7"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
immutable: false
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=8"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
ListElement {
|
|
||||||
url: "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
unfurled: true
|
|
||||||
immutable: false
|
|
||||||
hostname: "youtube.com"
|
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
description: ""
|
|
||||||
linkType: 0
|
|
||||||
thumbnailWidth: 480
|
|
||||||
thumbnailHeight: 360
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=9"
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,16 +2,48 @@ import QtQuick 2.15
|
||||||
import QtQuick.Controls 2.15
|
import QtQuick.Controls 2.15
|
||||||
import QtQuick.Layouts 1.15
|
import QtQuick.Layouts 1.15
|
||||||
|
|
||||||
|
import Qt.labs.settings 1.0
|
||||||
|
|
||||||
import StatusQ.Core.Theme 0.1
|
import StatusQ.Core.Theme 0.1
|
||||||
|
|
||||||
import shared.controls.chat 1.0
|
import shared.controls.chat 1.0
|
||||||
import utils 1.0
|
import utils 1.0
|
||||||
|
|
||||||
|
|
||||||
SplitView {
|
SplitView {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
property alias logoSettings: previewCard.logoSettings
|
|
||||||
property string ytBannerQuality: "hqdefault"
|
property string ytBannerQuality: "hqdefault"
|
||||||
|
property string image: Style.png("tokens/SOCKS")
|
||||||
|
property string banner: rawImageCheck.checked ? rawImageCheck.rawImageData : "https://img.youtube.com/vi/yHN1M7vcPKU/%1.jpg".arg(root.ytBannerQuality)
|
||||||
|
property bool globalUtilsReady: false
|
||||||
|
|
||||||
|
// globalUtilsInst mock
|
||||||
|
QtObject {
|
||||||
|
function getEmojiHashAsJson(publicKey) {
|
||||||
|
return JSON.stringify(["👨🏻🍼", "🏃🏿♂️", "🌇", "🤶🏿", "🏮","🤷🏻♂️", "🤦🏻", "📣", "🤎", "👷🏽", "😺", "🥞", "🔃", "🧝🏽♂️"])
|
||||||
|
}
|
||||||
|
function getColorId(publicKey) { return 4 }
|
||||||
|
|
||||||
|
function getCompressedPk(publicKey) { return "zx3sh" + publicKey }
|
||||||
|
|
||||||
|
function getColorHashAsJson(publicKey) {
|
||||||
|
return JSON.stringify([{4: 0, segmentLength: 1},
|
||||||
|
{5: 19, segmentLength: 2}])
|
||||||
|
}
|
||||||
|
|
||||||
|
function isCompressedPubKey(publicKey) { return true }
|
||||||
|
|
||||||
|
Component.onCompleted: {
|
||||||
|
Utils.globalUtilsInst = this
|
||||||
|
root.globalUtilsReady = true
|
||||||
|
|
||||||
|
}
|
||||||
|
Component.onDestruction: {
|
||||||
|
root.globalUtilsReady = false
|
||||||
|
Utils.globalUtilsInst = {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Pane {
|
Pane {
|
||||||
SplitView.fillWidth: true
|
SplitView.fillWidth: true
|
||||||
|
@ -19,22 +51,96 @@ SplitView {
|
||||||
|
|
||||||
LinkPreviewCard {
|
LinkPreviewCard {
|
||||||
id: previewCard
|
id: previewCard
|
||||||
bannerImageSource: "https://img.youtube.com/vi/yHN1M7vcPKU/%1.jpg".arg(root.ytBannerQuality)
|
type: 1
|
||||||
title: titleInput.text
|
linkData {
|
||||||
description: descriptionInput.text
|
title: titleInput.text
|
||||||
footer: footerInput.text
|
description: descriptionInput.text
|
||||||
logoSettings.name: Style.png("tokens/SOCKS")
|
domain: footerInput.text
|
||||||
logoSettings.isImage: true
|
thumbnail: root.banner
|
||||||
logoSettings.isLetterIdenticon: false
|
image: root.image
|
||||||
|
}
|
||||||
|
userData {
|
||||||
|
name: userNameInput.text
|
||||||
|
publicKey: "zQ3shgmVJjmwwhkfAemjDizYJtv9nzot7QD4iRJ52ZkgdU6Ci"
|
||||||
|
bio: bioInput.text
|
||||||
|
image: root.image
|
||||||
|
ensVerified: false
|
||||||
|
}
|
||||||
|
communityData {
|
||||||
|
name: titleInput.text
|
||||||
|
description: descriptionInput.text
|
||||||
|
banner: root.banner
|
||||||
|
image: root.image
|
||||||
|
membersCount: parseInt(membersCountInput.text)
|
||||||
|
activeMembersCount: parseInt(activeMembersCountInput.text)
|
||||||
|
color: "orchid"
|
||||||
|
}
|
||||||
|
channelData {
|
||||||
|
name: titleInput.text
|
||||||
|
description: descriptionInput.text
|
||||||
|
emoji: ""
|
||||||
|
color: "blue"
|
||||||
|
communityData {
|
||||||
|
name: "Community" + titleInput.text
|
||||||
|
description: "Community" + descriptionInput.text
|
||||||
|
banner: root.banner
|
||||||
|
image: root.image
|
||||||
|
membersCount: parseInt(membersCountInput.text)
|
||||||
|
activeMembersCount: parseInt(activeMembersCountInput.text)
|
||||||
|
color: "orchid"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Pane {
|
|
||||||
|
ScrollView {
|
||||||
SplitView.preferredWidth: 500
|
SplitView.preferredWidth: 500
|
||||||
SplitView.fillHeight: true
|
SplitView.fillHeight: true
|
||||||
|
leftPadding: 10
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
id: layout
|
||||||
spacing: 24
|
spacing: 24
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
Label {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: "Card type"
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("Link")
|
||||||
|
checked: previewCard.type === Constants.LinkPreviewType.Standard
|
||||||
|
onToggled: previewCard.type = Constants.LinkPreviewType.Standard
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("Contact")
|
||||||
|
checked: previewCard.type === Constants.LinkPreviewType.StatusContact
|
||||||
|
onToggled: previewCard.type = Constants.LinkPreviewType.StatusContact
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("Community")
|
||||||
|
checked: previewCard.type === Constants.LinkPreviewType.StatusCommunity
|
||||||
|
onToggled: {
|
||||||
|
previewCard.type = Constants.LinkPreviewType.StatusCommunity
|
||||||
|
titleInput.text = "Socks"
|
||||||
|
descriptionInput.text = "Community description goes here. If blank it will enable multi line title."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("Channel")
|
||||||
|
checked: previewCard.type === Constants.LinkPreviewType.StatusCommunityChannel
|
||||||
|
onToggled: {
|
||||||
|
previewCard.type = Constants.LinkPreviewType.StatusCommunityChannel
|
||||||
|
titleInput.text = "general"
|
||||||
|
descriptionInput.text = "Channel description goes here. If blank it will enable multi line title."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ColumnLayout {
|
||||||
|
visible: previewCard.type !== Constants.LinkPreviewType.StatusContact
|
||||||
Label {
|
Label {
|
||||||
text: "Title"
|
text: "Title"
|
||||||
}
|
}
|
||||||
|
@ -65,77 +171,125 @@ SplitView {
|
||||||
onClicked: descriptionInput.text = "Link description goes here. If blank it will enable multi line title."
|
onClicked: descriptionInput.text = "Link description goes here. If blank it will enable multi line title."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
ColumnLayout {
|
||||||
|
visible: previewCard.type === Constants.LinkPreviewType.Standard
|
||||||
|
Label {
|
||||||
|
text: "Footer"
|
||||||
|
}
|
||||||
|
|
||||||
|
TextField {
|
||||||
|
id: footerInput
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: "X"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
visible: previewCard.type === Constants.LinkPreviewType.StatusCommunity
|
||||||
|
Label {
|
||||||
|
text: "MembersCount"
|
||||||
|
}
|
||||||
|
TextField {
|
||||||
|
id: membersCountInput
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
inputMethodHints: Qt.ImhDigitsOnly
|
||||||
|
text: "629200"
|
||||||
|
}
|
||||||
|
TextField {
|
||||||
|
id: activeMembersCountInput
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
inputMethodHints: Qt.ImhDigitsOnly
|
||||||
|
text: "112100"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
visible: previewCard.type === Constants.LinkPreviewType.StatusContact
|
||||||
Label {
|
Label {
|
||||||
text: "Footer"
|
text: "User name"
|
||||||
}
|
}
|
||||||
|
|
||||||
TextField {
|
TextField {
|
||||||
id: footerInput
|
id: userNameInput
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: footerTypeCommunity.footerRichText
|
text: "Test user name"
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: "Bio"
|
||||||
|
}
|
||||||
|
RowLayout {
|
||||||
|
TextField {
|
||||||
|
id: bioInput
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: "User bio description goes here. If blank it will enable multi line title."
|
||||||
|
}
|
||||||
|
Button {
|
||||||
|
text: "clear"
|
||||||
|
onClicked: bioInput.text = ""
|
||||||
|
}
|
||||||
|
Button {
|
||||||
|
text: "Set"
|
||||||
|
onClicked: bioInput.text = "User bio description goes here. If blank it will enable multi line title."
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
visible: previewCard.type === Constants.LinkPreviewType.StatusCommunityChannel
|
||||||
Label {
|
Label {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: "Logo"
|
text: "channel settings"
|
||||||
}
|
}
|
||||||
|
CheckBox {
|
||||||
RadioButton {
|
|
||||||
text: qsTr("No logo")
|
|
||||||
checked: root.logoSettings.name === "" && root.logoSettings.emoji === ""
|
|
||||||
onToggled: {
|
|
||||||
root.logoSettings.name = ""
|
|
||||||
root.logoSettings.emoji = ""
|
|
||||||
root.logoSettings.isImage = false
|
|
||||||
root.logoSettings.isLetterIdenticon = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
RadioButton {
|
|
||||||
readonly property string rawImageData: ""
|
|
||||||
text: qsTr("Raw image")
|
|
||||||
checked: root.logoSettings.name === rawImageData
|
|
||||||
onToggled: {
|
|
||||||
root.logoSettings.name = rawImageData
|
|
||||||
root.logoSettings.isImage = true
|
|
||||||
root.logoSettings.isLetterIdenticon = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
RadioButton {
|
|
||||||
text: qsTr("QRC asset: SOCKS")
|
|
||||||
checked: root.logoSettings.name = Style.png("tokens/SOCKS")
|
|
||||||
onToggled:{
|
|
||||||
root.logoSettings.name = Style.png("tokens/SOCKS")
|
|
||||||
root.logoSettings.isImage = true
|
|
||||||
root.logoSettings.isLetterIdenticon = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
RadioButton {
|
|
||||||
text: qsTr("Letter identicon")
|
|
||||||
checked: root.logoSettings.name = "github.com"
|
|
||||||
onToggled: {
|
|
||||||
root.logoSettings.name = "github.com"
|
|
||||||
root.logoSettings.emoji = ""
|
|
||||||
root.logoSettings.isLetterIdenticon = true
|
|
||||||
root.logoSettings.color = "blue"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
RadioButton {
|
|
||||||
text: qsTr("Emoji")
|
text: qsTr("Emoji")
|
||||||
checked: root.logoSettings.emoji === "👋"
|
checked: previewCard.channelData.emoji === "👋"
|
||||||
onToggled: {
|
onToggled: previewCard.channelData.emoji = checked ? "👋" : ""
|
||||||
root.logoSettings.emoji = "👋"
|
}
|
||||||
root.logoSettings.isLetterIdenticon = true
|
RadioButton {
|
||||||
root.logoSettings.color = "orchid"
|
text: qsTr("Blue channel color")
|
||||||
}
|
checked: previewCard.channelData.color === "blue"
|
||||||
|
onToggled: previewCard.channelData.color = "blue"
|
||||||
|
}
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("Red channel color")
|
||||||
|
checked: previewCard.channelData.color === "red"
|
||||||
|
onToggled: previewCard.channelData.color = "red"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: "Logo"
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("no image")
|
||||||
|
checked: root.image === ""
|
||||||
|
onToggled: root.image = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
readonly property string rawImageData: ""
|
||||||
|
text: qsTr("Raw image")
|
||||||
|
checked: root.image === rawImageData
|
||||||
|
onToggled: root.image = rawImageData
|
||||||
|
}
|
||||||
|
|
||||||
|
RadioButton {
|
||||||
|
text: qsTr("QRC asset: SOCKS")
|
||||||
|
checked: root.image === Style.png("tokens/SOCKS")
|
||||||
|
onToggled: root.image = Style.png("tokens/SOCKS")
|
||||||
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
visible: previewCard.type !== Constants.LinkPreviewType.StatusContact
|
||||||
Label {
|
Label {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: "Banner size"
|
text: "Banner size"
|
||||||
|
@ -157,29 +311,17 @@ SplitView {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CheckBox {
|
||||||
|
id: rawImageCheck
|
||||||
|
readonly property string rawImageData: ""
|
||||||
|
text: qsTr("Raw image banner")
|
||||||
|
checked: root.banner === rawImageData
|
||||||
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
Label {
|
Label {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: "Footer type"
|
text: "UserName"
|
||||||
}
|
|
||||||
RadioButton {
|
|
||||||
id: footerTypeCommunity
|
|
||||||
property string footerRichText: `<img src="%1" width="16" height="16" style="vertical-align: top" ></img><font color="%2"> 629.2K </font> <img src="%3" width="16" height="16" style="vertical-align: top" ><font color="%2">112.1K</font>`.arg(Style.svg("group")).arg(Theme.palette.directColor1).arg(Style.svg("active-members"))
|
|
||||||
text: qsTr("Community")
|
|
||||||
checked: footerInput.text === footerRichText
|
|
||||||
onToggled: footerInput.text = footerRichText
|
|
||||||
}
|
|
||||||
RadioButton {
|
|
||||||
property string footerRichText: `%1 <img src="%2" width="16" height="16" style="vertical-align: top" ><font color="%2"> %3</font>`.arg(qsTr("Channel in")).arg(Style.png("tokens/SOCKS")).arg(qsTr("Doodles"))
|
|
||||||
text: qsTr("Channel")
|
|
||||||
checked: footerInput.text === footerRichText
|
|
||||||
onToggled: footerInput.text = footerRichText
|
|
||||||
}
|
|
||||||
RadioButton {
|
|
||||||
text: qsTr("Link domain")
|
|
||||||
property string footerText: "X"
|
|
||||||
checked: footerInput.text === footerText
|
|
||||||
onToggled: footerInput.text = footerText
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -201,4 +343,12 @@ SplitView {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Settings {
|
||||||
|
property alias linkType: previewCard.type
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// category: Controls
|
||||||
|
|
||||||
|
// https://www.figma.com/file/Mr3rqxxgKJ2zMQ06UAKiWL/💬-Chat⎜Desktop?type=design&node-id=22347-219545&mode=design&t=bODv5MUGQgU9ThJF-0
|
||||||
|
|
|
@ -25,15 +25,39 @@ SplitView {
|
||||||
LinkPreviewMiniCard {
|
LinkPreviewMiniCard {
|
||||||
id: previewMiniCard
|
id: previewMiniCard
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
titleStr: type === LinkPreviewMiniCard.Type.User ? userNameInput.text : titleInput.text
|
|
||||||
domain: domainInput.text
|
|
||||||
favIconUrl: faviconInput.text
|
|
||||||
previewState: stateInput.currentIndex
|
|
||||||
thumbnailImageUrl: externalImageInput.text
|
|
||||||
type: previewTypeInput.currentIndex
|
type: previewTypeInput.currentIndex
|
||||||
communityName: communityNameInput.text
|
previewState: stateInput.currentIndex
|
||||||
channelName: channelNameInput.text
|
linkData {
|
||||||
}
|
title: titleInput.text
|
||||||
|
description: ""
|
||||||
|
domain: domainInput.text
|
||||||
|
thumbnail: externalImageInput.text
|
||||||
|
image: faviconInput.text
|
||||||
|
}
|
||||||
|
userData {
|
||||||
|
name: userNameInput.text
|
||||||
|
publicKey: "zQ3shgmVJjmwwhkfAemjDizYJtv9nzot7QD4iRJ52ZkgdU6Ci"
|
||||||
|
image: faviconInput.text
|
||||||
|
ensVerified: false
|
||||||
|
}
|
||||||
|
communityData {
|
||||||
|
name: communityNameInput.text
|
||||||
|
banner: externalImageInput.text
|
||||||
|
image: faviconInput.text
|
||||||
|
color: "orchid"
|
||||||
|
}
|
||||||
|
channelData {
|
||||||
|
name: channelNameInput.text
|
||||||
|
emoji: ""
|
||||||
|
color: "blue"
|
||||||
|
communityData {
|
||||||
|
name: communityNameInput.text
|
||||||
|
banner: externalImageInput.text
|
||||||
|
image: faviconInput.text
|
||||||
|
color: "orchid"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Pane {
|
Pane {
|
||||||
|
@ -50,7 +74,7 @@ SplitView {
|
||||||
id: previewTypeInput
|
id: previewTypeInput
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
model: ["link", "image", "community", "channel", "user profile"]
|
model: ["unknown", "standard", "user profile", "community", "channel"]
|
||||||
}
|
}
|
||||||
Label {
|
Label {
|
||||||
text: "Community name"
|
text: "Community name"
|
||||||
|
|
|
@ -2,87 +2,15 @@ import QtQuick 2.15
|
||||||
import QtQuick.Controls 2.15
|
import QtQuick.Controls 2.15
|
||||||
import QtQuick.Layouts 1.15
|
import QtQuick.Layouts 1.15
|
||||||
|
|
||||||
|
import Models 1.0
|
||||||
|
|
||||||
import shared.views.chat 1.0
|
import shared.views.chat 1.0
|
||||||
import shared.stores 1.0
|
import shared.stores 1.0
|
||||||
|
|
||||||
SplitView {
|
SplitView {
|
||||||
|
|
||||||
ListModel {
|
LinkPreviewModel {
|
||||||
id: mockedLinkPreviewModel
|
id: mockedLinkPreviewModel
|
||||||
|
|
||||||
// Create the model dynamically, because `ListElement` doesnt suppport nested elements
|
|
||||||
Component.onCompleted: {
|
|
||||||
const emptyObject = {
|
|
||||||
"unfurled": true,
|
|
||||||
"immutable": false,
|
|
||||||
"empty": false,
|
|
||||||
"url": "",
|
|
||||||
"previewType": 1,
|
|
||||||
"standardPreview": {},
|
|
||||||
"standardPreviewThumbnail": {},
|
|
||||||
"statusContactPreview": {},
|
|
||||||
"statusContactPreviewThumbnail": {},
|
|
||||||
"statusCommunityPreview": {},
|
|
||||||
"statusCommunityPreviewIcon": {},
|
|
||||||
"statusCommunityPreviewBanner": {},
|
|
||||||
"statusCommunityChannelPreview": {},
|
|
||||||
"statusCommunityChannelCommunityPreview": {},
|
|
||||||
"statusCommunityChannelCommunityPreviewIcon": {},
|
|
||||||
"statusCommunityChannelCommunityPreviewBanner": {},
|
|
||||||
}
|
|
||||||
|
|
||||||
const preview1 = Object.assign({}, emptyObject)
|
|
||||||
preview1.url = "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
|
||||||
preview1.previewType = 1
|
|
||||||
preview1.standardPreview = {}
|
|
||||||
preview1.standardPreviewThumbnail = {}
|
|
||||||
preview1.standardPreview.hostname = "www.youtube.com"
|
|
||||||
preview1.standardPreview.title = "PSY - GANGNAM STYLE(강남스타일) M/V"
|
|
||||||
preview1.standardPreview.description = "PSY - ‘I LUV IT’ M/V @ https://youtu.be/Xvjnoagk6GU PSY - ‘New Face’ M/V @https://youtu.be/OwJPPaEyqhI PSY - 8TH ALBUM '4X2=8' on iTunes @ https://smarturl.it/PSY_8thAlbum PSY - GANGNAM STYLE(강남스타일) on iTunes @ http://smarturl.it/PsyGangnam #PSY #싸이 #GANGNAMSTYLE #강남스타일 More about PSY@ http://www.psyp..."
|
|
||||||
preview1.standardPreview.linkType = 0
|
|
||||||
preview1.standardPreviewThumbnail.width = 480
|
|
||||||
preview1.standardPreviewThumbnail.height = 360
|
|
||||||
preview1.standardPreviewThumbnail.url = "https://i.ytimg.com/vi/9bZkp7q19f0/hqdefault.jpg"
|
|
||||||
preview1.standardPreviewThumbnail.dataUri = ""
|
|
||||||
|
|
||||||
|
|
||||||
const preview2 = Object.assign({}, emptyObject)
|
|
||||||
preview2.url = "https://status.app/u/Ow==#zQ3shgmVJjmwwhkfAemjDizYJtv9nzot7QD4iRJ52ZkgdU6Ci"
|
|
||||||
preview2.previewType = 2
|
|
||||||
preview2.statusContactPreview = {}
|
|
||||||
preview2.statusContactPreview.publicKey = "zQ3shgmVJjmwwhkfAemjDizYJtv9nzot7QD4iRJ52ZkgdU6Ci"
|
|
||||||
preview2.statusContactPreview.displayName = "Test contact display name"
|
|
||||||
preview2.statusContactPreview.description = "Test description"
|
|
||||||
preview2.statusContactPreviewThumbnail = {}
|
|
||||||
preview2.statusContactPreviewThumbnail.width = 64
|
|
||||||
preview2.statusContactPreviewThumbnail.height = 64
|
|
||||||
preview2.statusContactPreviewThumbnail.url = "https://placehold.co/64x64"
|
|
||||||
preview2.statusContactPreviewThumbnail.dataUri = ""
|
|
||||||
|
|
||||||
const preview3 = Object.assign({}, emptyObject)
|
|
||||||
preview3.url = "https://status.app/c/ixiACjAKDlRlc3QgQ29tbXVuaXR5Eg9PcGVuIGZvciBhbnlvbmUYdiIHI0ZGMDAwMCoCHwkD#zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
|
||||||
preview3.previewType = 3
|
|
||||||
preview3.statusCommunityPreview = {}
|
|
||||||
preview3.statusCommunityPreview.communityId = "zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
|
||||||
preview3.statusCommunityPreview.displayName = "Test community display name"
|
|
||||||
preview3.statusCommunityPreview.description = "Test community description"
|
|
||||||
preview3.statusCommunityPreview.membersCount = 10
|
|
||||||
preview3.statusCommunityPreview.color = "#123456"
|
|
||||||
preview3.statusCommunityPreviewIcon = {}
|
|
||||||
preview3.statusCommunityPreviewIcon.width = 64
|
|
||||||
preview3.statusCommunityPreviewIcon.height = 64
|
|
||||||
preview3.statusCommunityPreviewIcon.url = "https://placehold.co/64x64"
|
|
||||||
preview3.statusCommunityPreviewIcon.dataUri = ""
|
|
||||||
preview3.statusCommunityPreviewBanner = {}
|
|
||||||
preview3.statusCommunityPreviewBanner.width = 320
|
|
||||||
preview3.statusCommunityPreviewBanner.height = 180
|
|
||||||
preview3.statusCommunityPreviewBanner.url = "https://placehold.co/320x180"
|
|
||||||
preview3.statusCommunityPreviewBanner.dataUri = ""
|
|
||||||
|
|
||||||
mockedLinkPreviewModel.append(preview1)
|
|
||||||
mockedLinkPreviewModel.append(preview2)
|
|
||||||
mockedLinkPreviewModel.append(preview3)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Pane {
|
Pane {
|
||||||
|
@ -90,38 +18,19 @@ SplitView {
|
||||||
SplitView.fillWidth: true
|
SplitView.fillWidth: true
|
||||||
SplitView.fillHeight: true
|
SplitView.fillHeight: true
|
||||||
|
|
||||||
component LinkPreviewObject: QtObject {
|
|
||||||
required property string url
|
|
||||||
required property bool unfurled
|
|
||||||
required property bool empty
|
|
||||||
required property int previewType
|
|
||||||
}
|
|
||||||
|
|
||||||
component StandardPreviewObject: QtObject {
|
|
||||||
required property string hostname
|
|
||||||
required property string title
|
|
||||||
required property string description
|
|
||||||
required property int linkType // 0 = link, 1 = image
|
|
||||||
}
|
|
||||||
|
|
||||||
component ThumbnailObject: QtObject {
|
|
||||||
required property int width
|
|
||||||
required property int height
|
|
||||||
required property string url
|
|
||||||
required property string dataUri
|
|
||||||
}
|
|
||||||
|
|
||||||
LinksMessageView {
|
LinksMessageView {
|
||||||
id: linksMessageView
|
id: linksMessageView
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
|
|
||||||
store: {}
|
isOnline: true
|
||||||
messageStore: {}
|
playAnimations: true
|
||||||
linkPreviewModel: mockedLinkPreviewModel
|
linkPreviewModel: mockedLinkPreviewModel
|
||||||
gifLinks: [ "https://media.tenor.com/qN_ytiwLh24AAAAC/cold.gif" ]
|
gifLinks: [ "https://media.tenor.com/qN_ytiwLh24AAAAC/cold.gif" ]
|
||||||
isCurrentUser: true
|
isCurrentUser: true
|
||||||
|
|
||||||
|
gifUnfurlingEnabled: false
|
||||||
|
canAskToUnfurlGifs: true
|
||||||
onImageClicked: {
|
onImageClicked: {
|
||||||
console.log("image clicked")
|
console.log("image clicked")
|
||||||
}
|
}
|
||||||
|
@ -148,18 +57,28 @@ SplitView {
|
||||||
}
|
}
|
||||||
CheckBox {
|
CheckBox {
|
||||||
text: qsTr("Enabled")
|
text: qsTr("Enabled")
|
||||||
checked: RootStore.gifUnfurlingEnabled
|
checked: linksMessageView.gifUnfurlingEnabled
|
||||||
onToggled: RootStore.gifUnfurlingEnabled = !RootStore.gifUnfurlingEnabled
|
onToggled: linksMessageView.gifUnfurlingEnabled = !linksMessageView.gifUnfurlingEnabled
|
||||||
}
|
}
|
||||||
CheckBox {
|
CheckBox {
|
||||||
text: qsTr("Never ask about GIF unfurling again")
|
text: qsTr("Can ask about GIF unfurling")
|
||||||
checked: RootStore.neverAskAboutUnfurlingAgain
|
checked: linksMessageView.canAskToUnfurlGifs
|
||||||
onClicked: RootStore.neverAskAboutUnfurlingAgain = !RootStore.neverAskAboutUnfurlingAgain
|
onClicked: linksMessageView.canAskToUnfurlGifs = !linksMessageView.canAskToUnfurlGifs
|
||||||
}
|
}
|
||||||
Button {
|
Button {
|
||||||
text: qsTr("Reset local `askAboutUnfurling` setting")
|
text: qsTr("Reset local `askAboutUnfurling` setting")
|
||||||
onClicked: linksMessageView.resetLocalAskAboutUnfurling()
|
onClicked: linksMessageView.resetLocalAskAboutUnfurling()
|
||||||
}
|
}
|
||||||
|
CheckBox {
|
||||||
|
text: qsTr("Play animations")
|
||||||
|
checked: linksMessageView.playAnimations
|
||||||
|
onToggled: linksMessageView.playAnimations = !linksMessageView.playAnimations
|
||||||
|
}
|
||||||
|
CheckBox {
|
||||||
|
text: qsTr("Is online")
|
||||||
|
checked: linksMessageView.isOnline
|
||||||
|
onToggled: linksMessageView.isOnline = !linksMessageView.isOnline
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -74,7 +74,7 @@ SplitView {
|
||||||
id: fakeUsersModel
|
id: fakeUsersModel
|
||||||
}
|
}
|
||||||
|
|
||||||
ListModel {
|
LinkPreviewModel {
|
||||||
id: fakeLinksModel
|
id: fakeLinksModel
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -172,19 +172,12 @@ SplitView {
|
||||||
fakeLinksModel.clear()
|
fakeLinksModel.clear()
|
||||||
words.forEach(function(word){
|
words.forEach(function(word){
|
||||||
if(Utils.isURL(word)) {
|
if(Utils.isURL(word)) {
|
||||||
fakeLinksModel.append({
|
const linkPreview = fakeLinksModel.getStandardLinkPreview()
|
||||||
url: encodeURI(word),
|
linkPreview.url = encodeURI(word)
|
||||||
unfurled: d.linkPreviewsEnabled,
|
linkPreview.unfurled = Math.random() > 0.2
|
||||||
immutable: !d.linkPreviewsEnabled,
|
linkPreview.immutable = !d.linkPreviewsEnabled
|
||||||
hostname: Math.floor(Math.random() * 2) ? "youtube.com" : "",
|
linkPreview.empty = Math.random() > 0.7
|
||||||
title: "PSY - GANGNAM STYLE(강남스타일) M/V",
|
fakeLinksModel.append(linkPreview)
|
||||||
description: "This is the description of the link",
|
|
||||||
linkType: Math.floor(Math.random() * 3),
|
|
||||||
thumbnailWidth: 480,
|
|
||||||
thumbnailHeight: 360,
|
|
||||||
thumbnailUrl: "https://picsum.photos/480/360?random=1",
|
|
||||||
thumbnailDataUri: ""
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,86 +0,0 @@
|
||||||
import QtQuick 2.15
|
|
||||||
import QtQuick.Controls 2.15
|
|
||||||
import QtQuick.Layouts 1.15
|
|
||||||
|
|
||||||
import shared.controls.chat 1.0
|
|
||||||
import utils 1.0
|
|
||||||
|
|
||||||
SplitView {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
property bool globalUtilsReady: false
|
|
||||||
|
|
||||||
// globalUtilsInst mock
|
|
||||||
QtObject {
|
|
||||||
function getEmojiHashAsJson(publicKey) {
|
|
||||||
return JSON.stringify(["👨🏻🍼", "🏃🏿♂️", "🌇", "🤶🏿", "🏮","🤷🏻♂️", "🤦🏻", "📣", "🤎", "👷🏽", "😺", "🥞", "🔃", "🧝🏽♂️"])
|
|
||||||
}
|
|
||||||
function getColorId(publicKey) { return 4 }
|
|
||||||
|
|
||||||
function getCompressedPk(publicKey) { return "zx3sh" + publicKey }
|
|
||||||
|
|
||||||
function getColorHashAsJson(publicKey) {
|
|
||||||
return JSON.stringify([{4: 0, segmentLength: 1},
|
|
||||||
{5: 19, segmentLength: 2}])
|
|
||||||
}
|
|
||||||
|
|
||||||
function isCompressedPubKey(publicKey) { return true }
|
|
||||||
|
|
||||||
Component.onCompleted: {
|
|
||||||
Utils.globalUtilsInst = this
|
|
||||||
root.globalUtilsReady = true
|
|
||||||
|
|
||||||
}
|
|
||||||
Component.onDestruction: {
|
|
||||||
root.globalUtilsReady = false
|
|
||||||
Utils.globalUtilsInst = {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Pane {
|
|
||||||
SplitView.fillWidth: true
|
|
||||||
SplitView.fillHeight: true
|
|
||||||
|
|
||||||
Loader {
|
|
||||||
anchors.centerIn: parent
|
|
||||||
active: root.globalUtilsReady
|
|
||||||
sourceComponent: UserProfileCard {
|
|
||||||
id: userProfileCard
|
|
||||||
userName: nameInput.text
|
|
||||||
userPublicKey: "0x1234567890"
|
|
||||||
userBio: bioInput.text
|
|
||||||
userImage: "
|
|
||||||
nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC"
|
|
||||||
ensVerified: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Pane {
|
|
||||||
SplitView.fillWidth: true
|
|
||||||
SplitView.fillHeight: true
|
|
||||||
SplitView.minimumWidth: 300
|
|
||||||
|
|
||||||
ColumnLayout {
|
|
||||||
Label {
|
|
||||||
text: "userName"
|
|
||||||
}
|
|
||||||
TextField {
|
|
||||||
id: nameInput
|
|
||||||
text: "John Doe"
|
|
||||||
}
|
|
||||||
Label {
|
|
||||||
text: "userBio"
|
|
||||||
}
|
|
||||||
TextField {
|
|
||||||
id: bioInput
|
|
||||||
text: "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed non risus. Suspendisse lectus tortor, dignissim sit amet, adipiscing nec, ultricies sed, dolor."
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://www.figma.com/file/Mr3rqxxgKJ2zMQ06UAKiWL/💬-Chat⎜Desktop?type=design&node-id=21961-655678&mode=design&t=JiMnPfMaLPWlrFK3-0
|
|
|
@ -0,0 +1,136 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
ListModel {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
|
||||||
|
function getStandardLinkPreview() {
|
||||||
|
const preview = Object.assign({}, emptyObject)
|
||||||
|
preview.url = "https://www.youtube.com/watch?v=9bZkp7q19f0"
|
||||||
|
preview.previewType = 1
|
||||||
|
preview.standardPreview = {}
|
||||||
|
preview.standardPreviewThumbnail = {}
|
||||||
|
preview.standardPreview.hostname = "www.youtube.com"
|
||||||
|
preview.standardPreview.title = "PSY - GANGNAM STYLE(강남스타일) M/V"
|
||||||
|
preview.standardPreview.description = "PSY - ‘I LUV IT’ M/V @ https://youtu.be/Xvjnoagk6GU PSY - ‘New Face’ M/V @https://youtu.be/OwJPPaEyqhI PSY - 8TH ALBUM '4X2=8' on iTunes @ https://smarturl.it/PSY_8thAlbum PSY - GANGNAM STYLE(강남스타일) on iTunes @ http://smarturl.it/PsyGangnam #PSY #싸이 #GANGNAMSTYLE #강남스타일 More about PSY@ http://www.psyp..."
|
||||||
|
preview.standardPreview.linkType = 0
|
||||||
|
preview.standardPreviewThumbnail.width = 480
|
||||||
|
preview.standardPreviewThumbnail.height = 360
|
||||||
|
preview.standardPreviewThumbnail.url = "https://i.ytimg.com/vi/9bZkp7q19f0/hqdefault.jpg"
|
||||||
|
preview.standardPreviewThumbnail.dataUri = ""
|
||||||
|
return preview
|
||||||
|
}
|
||||||
|
|
||||||
|
function getImageLinkPreview() {
|
||||||
|
const preview = Object.assign({}, emptyObject)
|
||||||
|
preview.url = "https://i.ytimg.com/vi/9bZkp7q19f0/hqdefault.jpg"
|
||||||
|
preview.previewType = 1
|
||||||
|
preview.standardPreview = {}
|
||||||
|
preview.standardPreviewThumbnail = {}
|
||||||
|
preview.standardPreview.hostname = "i.ytimg.com"
|
||||||
|
preview.standardPreview.title = "Image_link_preview.png"
|
||||||
|
preview.standardPreview.description = "Image link preview"
|
||||||
|
preview.standardPreview.linkType = 1
|
||||||
|
preview.standardPreviewThumbnail.width = 480
|
||||||
|
preview.standardPreviewThumbnail.height = 360
|
||||||
|
preview.standardPreviewThumbnail.url = "https://i.ytimg.com/vi/9bZkp7q19f0/hqdefault.jpg"
|
||||||
|
preview.standardPreviewThumbnail.dataUri = ""
|
||||||
|
return preview
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCommunityLinkPreview() {
|
||||||
|
const preview = Object.assign({}, emptyObject)
|
||||||
|
preview.url = "https://status.app/c/ixiACjAKDlRlc3QgQ29tbXVuaXR5Eg9PcGVuIGZvciBhbnlvbmUYdiIHI0ZGMDAwMCoCHwkD#zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
||||||
|
preview.previewType = 3
|
||||||
|
preview.statusCommunityPreview = {}
|
||||||
|
preview.statusCommunityPreview.communityId = "zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
||||||
|
preview.statusCommunityPreview.displayName = "Test community display name"
|
||||||
|
preview.statusCommunityPreview.description = "Test community description"
|
||||||
|
preview.statusCommunityPreview.membersCount = 10
|
||||||
|
preview.statusCommunityPreview.color = "#123456"
|
||||||
|
preview.statusCommunityPreviewIcon = {}
|
||||||
|
preview.statusCommunityPreviewIcon.width = 64
|
||||||
|
preview.statusCommunityPreviewIcon.height = 64
|
||||||
|
preview.statusCommunityPreviewIcon.url = "https://picsum.photos/64/64?random=1"
|
||||||
|
preview.statusCommunityPreviewIcon.dataUri = ""
|
||||||
|
preview.statusCommunityPreviewBanner = {}
|
||||||
|
preview.statusCommunityPreviewBanner.width = 320
|
||||||
|
preview.statusCommunityPreviewBanner.height = 180
|
||||||
|
preview.statusCommunityPreviewBanner.url = "https://picsum.photos/320/180?random=1"
|
||||||
|
preview.statusCommunityPreviewBanner.dataUri = ""
|
||||||
|
return preview
|
||||||
|
}
|
||||||
|
|
||||||
|
function getChannelLinkPreview() {
|
||||||
|
const preview = Object.assign({}, emptyObject)
|
||||||
|
preview.url = "https://status.app/c/ixiACjAKDlRlc3QgQ29tbXVuaXR5Eg9PcGVuIGZvciBhbnlvbmUYdiIHI0ZGMDAwMCoCHwkD#zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
||||||
|
preview.previewType = 4
|
||||||
|
preview.statusCommunityChannelPreview = {}
|
||||||
|
preview.statusCommunityChannelPreview.channelUuid = "zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
||||||
|
preview.statusCommunityChannelPreview.emoji = "👋"
|
||||||
|
preview.statusCommunityChannelPreview.displayName = "general"
|
||||||
|
preview.statusCommunityChannelPreview.description = "Test channel description"
|
||||||
|
preview.statusCommunityChannelPreview.color = "#122456"
|
||||||
|
preview.statusCommunityChannelCommunityPreview = {}
|
||||||
|
preview.statusCommunityChannelCommunityPreview.communityId = "zQ3shnd55dNx9yTihuL6XMbmyM6UNjzU6jk77h5Js31jxcT5V"
|
||||||
|
preview.statusCommunityChannelCommunityPreview.displayName = "Doodles"
|
||||||
|
preview.statusCommunityChannelCommunityPreview.description = "Test community description"
|
||||||
|
preview.statusCommunityChannelCommunityPreview.membersCount = 10
|
||||||
|
preview.statusCommunityChannelCommunityPreview.color = "#123456"
|
||||||
|
preview.statusCommunityChannelCommunityPreviewIcon = {}
|
||||||
|
preview.statusCommunityChannelCommunityPreviewIcon.width = 64
|
||||||
|
preview.statusCommunityChannelCommunityPreviewIcon.height = 64
|
||||||
|
preview.statusCommunityChannelCommunityPreviewIcon.url = "https://picsum.photos/64/64?random=1"
|
||||||
|
preview.statusCommunityChannelCommunityPreviewIcon.dataUri = ""
|
||||||
|
preview.statusCommunityChannelCommunityPreviewBanner = {}
|
||||||
|
preview.statusCommunityChannelCommunityPreviewBanner.width = 320
|
||||||
|
preview.statusCommunityChannelCommunityPreviewBanner.height = 180
|
||||||
|
preview.statusCommunityChannelCommunityPreviewBanner.url = "https://picsum.photos/320/180?random=1"
|
||||||
|
preview.statusCommunityChannelCommunityPreviewBanner.dataUri = ""
|
||||||
|
return preview
|
||||||
|
}
|
||||||
|
|
||||||
|
function getContactLinkPreview() {
|
||||||
|
const preview = Object.assign({}, emptyObject)
|
||||||
|
preview.url = "https://status.app/u/Ow==#zQ3shgmVJjmwwhkfAemjDizYJtv9nzot7QD4iRJ52ZkgdU6Ci"
|
||||||
|
preview.previewType = 2
|
||||||
|
preview.statusContactPreview = {}
|
||||||
|
preview.statusContactPreview.publicKey = "zQ3shgmVJjmwwhkfAemjDizYJtv9nzot7QD4iRJ52ZkgdU6Ci"
|
||||||
|
preview.statusContactPreview.displayName = "Test contact display name"
|
||||||
|
preview.statusContactPreview.description = "Test description"
|
||||||
|
preview.statusContactPreviewThumbnail = {}
|
||||||
|
preview.statusContactPreviewThumbnail.width = 64
|
||||||
|
preview.statusContactPreviewThumbnail.height = 64
|
||||||
|
preview.statusContactPreviewThumbnail.url = "https://picsum.photos/64/64?random=1"
|
||||||
|
preview.statusContactPreviewThumbnail.dataUri = ""
|
||||||
|
return preview
|
||||||
|
}
|
||||||
|
|
||||||
|
readonly property var emptyObject: {
|
||||||
|
"unfurled": true,
|
||||||
|
"immutable": false,
|
||||||
|
"empty": false,
|
||||||
|
"url": "",
|
||||||
|
"previewType": 1,
|
||||||
|
"standardPreview": {},
|
||||||
|
"standardPreviewThumbnail": {},
|
||||||
|
"statusContactPreview": {},
|
||||||
|
"statusContactPreviewThumbnail": {},
|
||||||
|
"statusCommunityPreview": {},
|
||||||
|
"statusCommunityPreviewIcon": {},
|
||||||
|
"statusCommunityPreviewBanner": {},
|
||||||
|
"statusCommunityChannelPreview": {},
|
||||||
|
"statusCommunityChannelCommunityPreview": {},
|
||||||
|
"statusCommunityChannelCommunityPreviewIcon": {},
|
||||||
|
"statusCommunityChannelCommunityPreviewBanner": {},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the model dynamically, because `ListElement` doesnt suppport nested elements
|
||||||
|
Component.onCompleted: {
|
||||||
|
append(getStandardLinkPreview())
|
||||||
|
append(getImageLinkPreview())
|
||||||
|
append(getCommunityLinkPreview())
|
||||||
|
append(getChannelLinkPreview())
|
||||||
|
append(getContactLinkPreview())
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,6 +6,7 @@ ChannelsModel 1.0 ChannelsModel.qml
|
||||||
CollectiblesModel 1.0 CollectiblesModel.qml
|
CollectiblesModel 1.0 CollectiblesModel.qml
|
||||||
FeesModel 1.0 FeesModel.qml
|
FeesModel 1.0 FeesModel.qml
|
||||||
IconModel 1.0 IconModel.qml
|
IconModel 1.0 IconModel.qml
|
||||||
|
LinkPreviewModel 1.0 LinkPreviewModel.qml
|
||||||
MintedTokensModel 1.0 MintedTokensModel.qml
|
MintedTokensModel 1.0 MintedTokensModel.qml
|
||||||
RecipientModel 1.0 RecipientModel.qml
|
RecipientModel 1.0 RecipientModel.qml
|
||||||
TokenHoldersModel 1.0 TokenHoldersModel.qml
|
TokenHoldersModel 1.0 TokenHoldersModel.qml
|
||||||
|
|
|
@ -80,6 +80,30 @@ QtObject {
|
||||||
return num.toLocaleString(locale, 'f', precision)
|
return num.toLocaleString(locale, 'f', precision)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function numberToLocaleStringInCompactForm(num, locale = null) {
|
||||||
|
locale = locale || Qt.locale()
|
||||||
|
const numberOfDigits = integralPartLength(num)
|
||||||
|
let oneArgStrFormat = "%1"
|
||||||
|
let formattedNumber = num
|
||||||
|
let multiplier = 1
|
||||||
|
if(numberOfDigits >=4 && numberOfDigits < 7) { // 1K - 999K
|
||||||
|
multiplier = 1 / 1000
|
||||||
|
oneArgStrFormat = qsTr("%1K", "Thousand")
|
||||||
|
} else if(numberOfDigits >= 7 && numberOfDigits < 10) { // 1M - 999M
|
||||||
|
multiplier = 1 / 1000000
|
||||||
|
oneArgStrFormat = qsTr("%1M", "Million")
|
||||||
|
} else if(numberOfDigits >= 10 && numberOfDigits < 13) { // 1B - 999B
|
||||||
|
multiplier = 1 / 1000000000
|
||||||
|
oneArgStrFormat = qsTr("%1B", "Billion")
|
||||||
|
} else if(numberOfDigits >= 13 && numberOfDigits < 16) { // 1T - 999T
|
||||||
|
multiplier = 1 / 1000000000000
|
||||||
|
oneArgStrFormat = qsTr("%1T", "Trillion")
|
||||||
|
}
|
||||||
|
|
||||||
|
const stringNumber = numberToLocaleString(num * multiplier, 2, locale)
|
||||||
|
return oneArgStrFormat.arg(stripTrailingZeroes(stringNumber, locale))
|
||||||
|
}
|
||||||
|
|
||||||
function numberFromLocaleString(num, locale = null) {
|
function numberFromLocaleString(num, locale = null) {
|
||||||
locale = locale || Qt.locale()
|
locale = locale || Qt.locale()
|
||||||
try {
|
try {
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M6.46069 13.6656C6.42831 13.957 6.78989 14.1174 6.98417 13.8977L12.6016 7.54762C12.8588 7.25683 12.6524 6.79857 12.2641 6.79857H9.09442C8.91541 6.79857 8.77613 6.64297 8.7959 6.46505L9.25486 2.33445C9.28724 2.04302 8.92565 1.88265 8.73137 2.10227L3.11397 8.45238C2.85673 8.74317 3.06318 9.20143 3.45142 9.20143H6.28554C6.64358 9.20143 6.92212 9.51264 6.88258 9.86848L6.46069 13.6656Z" fill="black"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 552 B |
|
@ -123,4 +123,168 @@ TestCase {
|
||||||
compare(LocaleUtils.currencyAmountToLocaleString(
|
compare(LocaleUtils.currencyAmountToLocaleString(
|
||||||
data.amount, null, locale), data.amountString)
|
data.amount, null, locale), data.amountString)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function test_numberToLocaleStringInCompactForm_data() {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
amount: NaN,
|
||||||
|
amountString: "nan"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: null,
|
||||||
|
amountString: "0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: "",
|
||||||
|
amountString: "0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: "string",
|
||||||
|
amountString: "nan"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: {},
|
||||||
|
amountString: "nan"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1,
|
||||||
|
amountString: "-1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1.1,
|
||||||
|
amountString: "-1.1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1.1234,
|
||||||
|
amountString: "-1.12"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000,
|
||||||
|
amountString: "-1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000.1,
|
||||||
|
amountString: "-1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -100000,
|
||||||
|
amountString: "-100K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1001,
|
||||||
|
amountString: "-1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1100,
|
||||||
|
amountString: "-1.1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000000,
|
||||||
|
amountString: "-1M"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1100000.123,
|
||||||
|
amountString: "-1.1M"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000000000,
|
||||||
|
amountString: "-1B"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1100000000,
|
||||||
|
amountString: "-1.1B"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000000000.123,
|
||||||
|
amountString: "-1B"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000000000000,
|
||||||
|
amountString: "-1T"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -999000000000000,
|
||||||
|
amountString: "-999T"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: -1000000000000000,
|
||||||
|
amountString: "-1,000,000,000,000,000"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 0,
|
||||||
|
amountString: "0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1,
|
||||||
|
amountString: "1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1.1,
|
||||||
|
amountString: "1.1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1.1234,
|
||||||
|
amountString: "1.12"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000,
|
||||||
|
amountString: "1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000.1,
|
||||||
|
amountString: "1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 100000,
|
||||||
|
amountString: "100K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1001,
|
||||||
|
amountString: "1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1100,
|
||||||
|
amountString: "1.1K"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000000,
|
||||||
|
amountString: "1M"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1100000.123,
|
||||||
|
amountString: "1.1M"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000000000,
|
||||||
|
amountString: "1B"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1100000000,
|
||||||
|
amountString: "1.1B"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000000000.123,
|
||||||
|
amountString: "1B"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000000000000,
|
||||||
|
amountString: "1T"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 999000000000000,
|
||||||
|
amountString: "999T"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
amount: 1000000000000000,
|
||||||
|
amountString: "1,000,000,000,000,000"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
function test_numberToLocaleStringInCompactForm(data) {
|
||||||
|
const locale = Qt.locale("en_US")
|
||||||
|
|
||||||
|
compare(LocaleUtils.numberToLocaleStringInCompactForm(
|
||||||
|
data.amount, locale), data.amountString)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -277,6 +277,10 @@ Item {
|
||||||
popups.openConfirmExternalLinkPopup(link, domain)
|
popups.openConfirmExternalLinkPopup(link, domain)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function onActivateDeepLink(link: string) {
|
||||||
|
appMain.rootStore.mainModuleInst.activateStatusDeepLink(link)
|
||||||
|
}
|
||||||
|
|
||||||
function onPlaySendMessageSound() {
|
function onPlaySendMessageSound() {
|
||||||
sendMessageSound.stop()
|
sendMessageSound.stop()
|
||||||
sendMessageSound.play()
|
sendMessageSound.play()
|
||||||
|
|
|
@ -6,7 +6,7 @@ import QtQuick.Layouts 1.15
|
||||||
import StatusQ.Core 0.1
|
import StatusQ.Core 0.1
|
||||||
|
|
||||||
import shared.status 1.0
|
import shared.status 1.0
|
||||||
import shared.controls.chat 1.0
|
import shared.controls.delegates 1.0
|
||||||
|
|
||||||
import utils 1.0
|
import utils 1.0
|
||||||
|
|
||||||
|
@ -99,57 +99,8 @@ Control {
|
||||||
Repeater {
|
Repeater {
|
||||||
id: linkPreviewRepeater
|
id: linkPreviewRepeater
|
||||||
model: d.filteredModel
|
model: d.filteredModel
|
||||||
delegate: LinkPreviewMiniCard {
|
delegate: LinkPreviewMiniCardDelegate {
|
||||||
// Model properties
|
|
||||||
|
|
||||||
required property int index
|
required property int index
|
||||||
required property bool unfurled
|
|
||||||
required property bool empty
|
|
||||||
required property string url
|
|
||||||
required property bool immutable
|
|
||||||
required property int previewType
|
|
||||||
required property var standardPreview
|
|
||||||
required property var standardPreviewThumbnail
|
|
||||||
required property var statusContactPreview
|
|
||||||
required property var statusContactPreviewThumbnail
|
|
||||||
required property var statusCommunityPreview
|
|
||||||
required property var statusCommunityPreviewIcon
|
|
||||||
required property var statusCommunityPreviewBanner
|
|
||||||
required property var statusCommunityChannelPreview
|
|
||||||
required property var statusCommunityChannelCommunityPreview
|
|
||||||
required property var statusCommunityChannelCommunityPreviewIcon
|
|
||||||
required property var statusCommunityChannelCommunityPreviewBanner
|
|
||||||
|
|
||||||
readonly property var thumbnail: {
|
|
||||||
switch (previewType) {
|
|
||||||
case Constants.Standard:
|
|
||||||
return standardPreviewThumbnail
|
|
||||||
case Constants.StatusContact:
|
|
||||||
return statusContactPreviewThumbnail
|
|
||||||
case Constants.StatusCommunity:
|
|
||||||
return statusCommunityPreviewIcon
|
|
||||||
case Constants.StatusCommunityChannel:
|
|
||||||
return statusCommunityChannelCommunityPreviewIcon
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly property string thumbnailUrl: thumbnail ? thumbnail.url : ""
|
|
||||||
readonly property string thumbnailDataUri: thumbnail ? thumbnail.dataUri : ""
|
|
||||||
|
|
||||||
|
|
||||||
Layout.preferredHeight: 64
|
|
||||||
|
|
||||||
titleStr: standardPreview ? standardPreview.title : statusContactPreview ? statusContactPreview.displayName : ""
|
|
||||||
domain: standardPreview ? standardPreview.hostname : "" //TODO: use domain when available
|
|
||||||
favIconUrl: "" //TODO: use favicon when available
|
|
||||||
communityName: statusCommunityPreview ? statusCommunityPreview.displayName : ""
|
|
||||||
channelName: statusCommunityChannelPreview ? statusCommunityChannelPreview.displayName : ""
|
|
||||||
|
|
||||||
thumbnailImageUrl: thumbnailUrl.length > 0 ? thumbnailUrl : thumbnailDataUri
|
|
||||||
type: getCardType(previewType, standardPreview)
|
|
||||||
previewState: unfurled && !empty ? LinkPreviewMiniCard.State.Loaded :
|
|
||||||
unfurled && empty ? LinkPreviewMiniCard.State.LoadingFailed :
|
|
||||||
!unfurled ? LinkPreviewMiniCard.State.Loading : LinkPreviewMiniCard.State.Invalid
|
|
||||||
|
|
||||||
onClose: root.dismissLinkPreview(d.filteredModel.mapToSource(index))
|
onClose: root.dismissLinkPreview(d.filteredModel.mapToSource(index))
|
||||||
onRetry: root.linkReload(url)
|
onRetry: root.linkReload(url)
|
||||||
|
|
|
@ -6,27 +6,24 @@ import StatusQ.Core 0.1
|
||||||
import StatusQ.Core.Theme 0.1
|
import StatusQ.Core.Theme 0.1
|
||||||
import StatusQ.Components 0.1
|
import StatusQ.Components 0.1
|
||||||
|
|
||||||
|
import shared.controls 1.0
|
||||||
import shared.status 1.0
|
import shared.status 1.0
|
||||||
import utils 1.0
|
import utils 1.0
|
||||||
|
|
||||||
|
import "./private"
|
||||||
|
|
||||||
CalloutCard {
|
CalloutCard {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
/// The title of the callout card
|
readonly property LinkData linkData: LinkData { }
|
||||||
required property string title
|
readonly property UserData userData: UserData { }
|
||||||
required property string description
|
readonly property CommunityData communityData: CommunityData { }
|
||||||
required property string footer
|
readonly property ChannelData channelData: ChannelData { }
|
||||||
|
|
||||||
|
property int type: Constants.LinkPreviewType.NoPreview
|
||||||
|
|
||||||
property bool highlight: false
|
property bool highlight: false
|
||||||
|
|
||||||
property StatusAssetSettings logoSettings: StatusAssetSettings {
|
|
||||||
width: 28
|
|
||||||
height: 28
|
|
||||||
bgRadius: bgWidth / 2
|
|
||||||
}
|
|
||||||
|
|
||||||
property string bannerImageSource: ""
|
|
||||||
|
|
||||||
signal clicked(var mouse)
|
signal clicked(var mouse)
|
||||||
|
|
||||||
borderWidth: 1
|
borderWidth: 1
|
||||||
|
@ -41,71 +38,103 @@ CalloutCard {
|
||||||
}
|
}
|
||||||
|
|
||||||
contentItem: ColumnLayout {
|
contentItem: ColumnLayout {
|
||||||
StatusImage {
|
Loader {
|
||||||
id: bannerImage
|
id: bannerImageLoader
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: d.bannerImageMargins
|
Layout.leftMargin: d.bannerImageMargins
|
||||||
Layout.rightMargin: d.bannerImageMargins
|
Layout.rightMargin: d.bannerImageMargins
|
||||||
Layout.topMargin: d.bannerImageMargins
|
Layout.topMargin: d.bannerImageMargins
|
||||||
Layout.preferredHeight: 170
|
Layout.preferredHeight: 170
|
||||||
asynchronous: true
|
active: !!d.bannerImageSource
|
||||||
source: root.bannerImageSource
|
sourceComponent: StatusImage {
|
||||||
fillMode: Image.PreserveAspectCrop
|
id: bannerImage
|
||||||
layer.enabled: true
|
asynchronous: true
|
||||||
layer.effect: root.clippingEffect
|
source: d.bannerImageSource
|
||||||
|
fillMode: Image.PreserveAspectCrop
|
||||||
|
layer.enabled: true
|
||||||
|
layer.effect: root.clippingEffect
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.margins: 12
|
Layout.margins: 12
|
||||||
|
Loader {
|
||||||
|
id: userImageLoader
|
||||||
|
Layout.alignment: Qt.AlignLeft | Qt.AlignVCenter
|
||||||
|
Layout.topMargin: 8
|
||||||
|
Layout.bottomMargin: 8
|
||||||
|
visible: active
|
||||||
|
active: root.type === Constants.LinkPreviewType.StatusContact
|
||||||
|
sourceComponent: UserImage {
|
||||||
|
interactive: false
|
||||||
|
imageWidth: 58
|
||||||
|
imageHeight: imageWidth
|
||||||
|
ensVerified: root.userData.ensVerified
|
||||||
|
name: root.userData.name
|
||||||
|
pubkey: root.userData.publicKey
|
||||||
|
image: root.userData.image
|
||||||
|
}
|
||||||
|
}
|
||||||
RowLayout {
|
RowLayout {
|
||||||
|
id: titleLayout
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.maximumHeight: root.description.length ? 28 : 72
|
Layout.maximumHeight: description.text.length ? 28 : 72
|
||||||
|
Layout.minimumHeight: 18
|
||||||
StatusSmartIdenticon {
|
StatusSmartIdenticon {
|
||||||
Layout.alignment: Qt.AlignLeft | Qt.AlignTop
|
id: logo
|
||||||
|
Layout.alignment: Qt.AlignTop
|
||||||
Layout.preferredWidth: 28
|
Layout.preferredWidth: 28
|
||||||
Layout.preferredHeight: 28
|
Layout.preferredHeight: 28
|
||||||
asset: root.logoSettings
|
asset.width: width
|
||||||
name: root.logoSettings.name
|
asset.height: height
|
||||||
visible: !!root.logoSettings.name.length || !!root.logoSettings.emoji.length
|
visible: false
|
||||||
}
|
}
|
||||||
StatusBaseText {
|
StatusBaseText {
|
||||||
text: root.title
|
id: title
|
||||||
|
// One line centered next to the logo
|
||||||
|
// Two or more lines, or no logo, top aligned
|
||||||
|
readonly property bool centerText: lineCount == 1 && height === logo.height && logo.visible
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
|
Layout.alignment: Qt.AlignTop
|
||||||
|
Layout.topMargin: verticalAlignment === Text.AlignTop && contentHeight < logo.height ? (logo.height - contentHeight) / 2 : 0
|
||||||
font.pixelSize: 13
|
font.pixelSize: 13
|
||||||
font.weight: Font.Medium
|
font.weight: Font.Medium
|
||||||
wrapMode: Text.Wrap
|
wrapMode: Text.Wrap
|
||||||
elide: Text.ElideRight
|
elide: Text.ElideRight
|
||||||
verticalAlignment: Text.AlignVCenter
|
verticalAlignment: centerText ? Text.AlignVCenter : Text.AlignTop
|
||||||
|
visible: text.length
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
EmojiHash {
|
||||||
|
Layout.topMargin: 4
|
||||||
|
Layout.bottomMargin: 6
|
||||||
|
visible: root.type === Constants.LinkPreviewType.StatusContact
|
||||||
|
publicKey: root.userData.publicKey
|
||||||
|
oneRow: true
|
||||||
|
}
|
||||||
StatusBaseText {
|
StatusBaseText {
|
||||||
|
id: description
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
text: root.description
|
|
||||||
font.pixelSize: 12
|
font.pixelSize: 12
|
||||||
wrapMode: Text.Wrap
|
wrapMode: Text.Wrap
|
||||||
elide: Text.ElideRight
|
elide: Text.ElideRight
|
||||||
color: Theme.palette.baseColor1
|
color: Theme.palette.baseColor1
|
||||||
visible: root.description.length
|
visible: description.text.length
|
||||||
}
|
}
|
||||||
StatusBaseText {
|
Loader {
|
||||||
id: linkSite
|
id: footerLoader
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: root.footer
|
visible: active
|
||||||
font.pixelSize: 12
|
sourceComponent: FooterText {
|
||||||
lineHeight: 16
|
}
|
||||||
lineHeightMode: Text.FixedHeight
|
|
||||||
color: Theme.palette.baseColor1
|
|
||||||
elide: Text.ElideRight
|
|
||||||
verticalAlignment: Text.AlignBottom
|
|
||||||
textFormat: Text.RichText
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
anchors.fill: root
|
anchors.fill: root
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
|
@ -114,9 +143,151 @@ CalloutCard {
|
||||||
onClicked: root.clicked(mouse)
|
onClicked: root.clicked(mouse)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
component FooterText: StatusBaseText {
|
||||||
|
font.pixelSize: 12
|
||||||
|
lineHeight: 16
|
||||||
|
lineHeightMode: Text.FixedHeight
|
||||||
|
color: Theme.palette.baseColor1
|
||||||
|
elide: Text.ElideRight
|
||||||
|
wrapMode: Text.Wrap
|
||||||
|
verticalAlignment: Text.AlignBottom
|
||||||
|
text: root.linkData.domain
|
||||||
|
maximumLineCount: 1
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: channelFooterComponent
|
||||||
|
RowLayout {
|
||||||
|
spacing: 4
|
||||||
|
FooterText {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
text: qsTr("Channel in")
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
}
|
||||||
|
StatusRoundedImage {
|
||||||
|
Layout.preferredHeight: 16
|
||||||
|
Layout.preferredWidth: height
|
||||||
|
image.source: channelData.communityData.image
|
||||||
|
}
|
||||||
|
FooterText {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: channelData.communityData.name
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
color: Theme.palette.directColor1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: communityFooterComponent
|
||||||
|
RowLayout {
|
||||||
|
spacing: 2
|
||||||
|
StatusIcon {
|
||||||
|
icon: "group"
|
||||||
|
color: Theme.palette.directColor1
|
||||||
|
width: 16
|
||||||
|
height: width
|
||||||
|
}
|
||||||
|
FooterText {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.fillWidth: communityData.activeMembersCount === -1 //TODO: remove magic number once we have activeMembersCount
|
||||||
|
color: Theme.palette.directColor1
|
||||||
|
text: LocaleUtils.numberToLocaleStringInCompactForm(communityData.membersCount)
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
}
|
||||||
|
StatusIcon {
|
||||||
|
Layout.leftMargin: 6
|
||||||
|
icon: "active-members"
|
||||||
|
color: Theme.palette.directColor1
|
||||||
|
width: 16
|
||||||
|
height: width
|
||||||
|
visible: communityData.activeMembersCount > -1
|
||||||
|
}
|
||||||
|
FooterText {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.fillHeight: true
|
||||||
|
color: Theme.palette.directColor1
|
||||||
|
text: LocaleUtils.numberToLocaleStringInCompactForm(communityData.activeMembersCount)
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
visible: communityData.activeMembersCount > -1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//behavior
|
||||||
|
states: [
|
||||||
|
State {
|
||||||
|
name: "noPreview"
|
||||||
|
when: root.type === Constants.LinkPreviewType.NoPreview
|
||||||
|
PropertyChanges { target: root; visible: false }
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "linkPreview"
|
||||||
|
when: root.type === Constants.LinkPreviewType.Standard
|
||||||
|
PropertyChanges {
|
||||||
|
target: logo
|
||||||
|
visible: !!root.linkData.image
|
||||||
|
name: root.linkData.domain
|
||||||
|
asset.name: root.linkData.image
|
||||||
|
asset.isImage: !!root.linkData.image
|
||||||
|
asset.color: Theme.palette.baseColor2
|
||||||
|
}
|
||||||
|
PropertyChanges { target: bannerImageLoader; visible: true }
|
||||||
|
PropertyChanges { target: title; text: root.linkData.title }
|
||||||
|
PropertyChanges { target: description; text: root.linkData.description }
|
||||||
|
PropertyChanges { target: d; bannerImageSource: root.linkData.thumbnail }
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "community"
|
||||||
|
when: root.type === Constants.LinkPreviewType.StatusCommunity
|
||||||
|
PropertyChanges {
|
||||||
|
target: logo
|
||||||
|
visible: true
|
||||||
|
name: root.communityData.name
|
||||||
|
asset.name: root.communityData.image
|
||||||
|
asset.isImage: !!root.communityData.image
|
||||||
|
asset.color: root.communityData.color
|
||||||
|
}
|
||||||
|
PropertyChanges { target: bannerImageLoader; visible: true }
|
||||||
|
PropertyChanges { target: title; text: root.communityData.name }
|
||||||
|
PropertyChanges { target: description; text: root.communityData.description }
|
||||||
|
PropertyChanges { target: d; bannerImageSource: root.communityData.banner }
|
||||||
|
PropertyChanges { target: footerLoader; active: true; visible: true; sourceComponent: communityFooterComponent }
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "channel"
|
||||||
|
when: root.type === Constants.LinkPreviewType.StatusCommunityChannel
|
||||||
|
PropertyChanges {
|
||||||
|
target: logo
|
||||||
|
visible: true
|
||||||
|
name: root.channelData.name
|
||||||
|
asset.name: ""
|
||||||
|
asset.isImage: false
|
||||||
|
asset.color: root.channelData.color
|
||||||
|
asset.emoji: root.channelData.emoji
|
||||||
|
}
|
||||||
|
PropertyChanges { target: bannerImageLoader; visible: true }
|
||||||
|
PropertyChanges { target: title; text: "#" + root.channelData.name }
|
||||||
|
PropertyChanges { target: description; text: root.channelData.description || root.channelData.communityData.description }
|
||||||
|
PropertyChanges { target: d; bannerImageSource: root.channelData.communityData.banner }
|
||||||
|
PropertyChanges { target: footerLoader; active: true; visible: true; sourceComponent: channelFooterComponent }
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "contact"
|
||||||
|
when: root.type === Constants.LinkPreviewType.StatusContact
|
||||||
|
PropertyChanges { target: root; implicitHeight: 187 }
|
||||||
|
PropertyChanges { target: bannerImageLoader; visible: false }
|
||||||
|
PropertyChanges { target: footerLoader; active: false; visible: !root.userData.bio; Layout.fillHeight: true }
|
||||||
|
PropertyChanges { target: title; text: root.userData.name }
|
||||||
|
PropertyChanges { target: description; text: root.userData.bio; Layout.minimumHeight: 32; visible: true }
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
id: d
|
id: d
|
||||||
property real bannerImageMargins: 1 / Screen.devicePixelRatio // image size isn't pixel perfect..
|
property real bannerImageMargins: 1 / Screen.devicePixelRatio // image size isn't pixel perfect..
|
||||||
property bool highlight: root.highlight || root.hovered
|
property bool highlight: root.highlight || root.hovered
|
||||||
|
property string bannerImageSource: ""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,8 @@ import shared 1.0
|
||||||
|
|
||||||
import utils 1.0
|
import utils 1.0
|
||||||
|
|
||||||
|
import "./private" 1.0
|
||||||
|
|
||||||
CalloutCard {
|
CalloutCard {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
|
@ -20,45 +22,11 @@ CalloutCard {
|
||||||
Loaded
|
Loaded
|
||||||
}
|
}
|
||||||
|
|
||||||
enum Type {
|
readonly property LinkData linkData: LinkData { }
|
||||||
Unknown = 0,
|
readonly property UserData userData: UserData { }
|
||||||
Link,
|
readonly property CommunityData communityData: CommunityData { }
|
||||||
Image,
|
readonly property ChannelData channelData: ChannelData { }
|
||||||
Community,
|
|
||||||
Channel,
|
|
||||||
User
|
|
||||||
}
|
|
||||||
|
|
||||||
function getCardType(previewType, standardLinkPreview) {
|
|
||||||
switch (previewType) {
|
|
||||||
case Constants.StatusContact:
|
|
||||||
return LinkPreviewMiniCard.Type.User
|
|
||||||
case Constants.StatusCommunity:
|
|
||||||
return LinkPreviewMiniCard.Type.Community
|
|
||||||
case Constants.StatusCommunityChannel:
|
|
||||||
return LinkPreviewMiniCard.Type.Channel
|
|
||||||
case Constants.Standard:
|
|
||||||
if (!standardLinkPreview)
|
|
||||||
return LinkPreviewMiniCard.Type.Unknown
|
|
||||||
switch (standardLinkPreview.linkType) {
|
|
||||||
case Constants.StandardLinkPreviewType.Link:
|
|
||||||
return LinkPreviewMiniCard.Type.Link
|
|
||||||
case Constants.StandardLinkPreviewType.Image:
|
|
||||||
return LinkPreviewMiniCard.Type.Image
|
|
||||||
default:
|
|
||||||
return LinkPreviewMiniCard.Type.Unknown
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return LinkPreviewMiniCard.Type.Unknown
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
required property string titleStr
|
|
||||||
required property string domain
|
|
||||||
required property string communityName
|
|
||||||
required property string channelName
|
|
||||||
required property url favIconUrl
|
|
||||||
required property url thumbnailImageUrl
|
|
||||||
required property int previewState
|
required property int previewState
|
||||||
required property int type
|
required property int type
|
||||||
|
|
||||||
|
@ -106,47 +74,78 @@ CalloutCard {
|
||||||
},
|
},
|
||||||
State {
|
State {
|
||||||
name: "loaded"
|
name: "loaded"
|
||||||
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === LinkPreviewMiniCard.Type.Link
|
when: root.previewState === LinkPreviewMiniCard.State.Loaded &&
|
||||||
|
root.type === Constants.LinkPreviewType.Standard &&
|
||||||
|
root.linkData.type === Constants.StandardLinkPreviewType.Link
|
||||||
PropertyChanges {
|
PropertyChanges {
|
||||||
target: root; visible: true; dashedBorder: false; borderWidth: 0;
|
target: root; visible: true; dashedBorder: false; borderWidth: 0;
|
||||||
backgroundColor: root.containsMouse ? Theme.palette.directColor8 : Theme.palette.indirectColor1;
|
backgroundColor: root.containsMouse ? Theme.palette.directColor8 : Theme.palette.indirectColor1;
|
||||||
borderColor: backgroundColor;
|
borderColor: backgroundColor;
|
||||||
}
|
}
|
||||||
PropertyChanges { target: loadingAnimation; visible: false; }
|
PropertyChanges { target: loadingAnimation; visible: false; }
|
||||||
PropertyChanges { target: titleText; text: root.titleStr; color: Theme.palette.directColor1 }
|
PropertyChanges { target: titleText; text: root.linkData.title; color: Theme.palette.directColor1 }
|
||||||
PropertyChanges { target: subtitleText; visible: true; }
|
PropertyChanges { target: subtitleText; visible: true; text: root.linkData.domain; }
|
||||||
PropertyChanges { target: reloadButton; visible: false; }
|
PropertyChanges { target: reloadButton; visible: false; }
|
||||||
PropertyChanges { target: favIcon; visible: true }
|
|
||||||
},
|
|
||||||
State {
|
|
||||||
name: "loadedImage"
|
|
||||||
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === LinkPreviewMiniCard.Type.Image
|
|
||||||
extend: "loaded"
|
|
||||||
PropertyChanges { target: thumbnailImage; visible: root.thumbnailImageUrl != "" }
|
|
||||||
PropertyChanges { target: favIcon; visible: true; name: root.domain; asset.isLetterIdenticon: true; asset.color: Theme.palette.baseColor2; }
|
|
||||||
},
|
|
||||||
State {
|
|
||||||
name: "loadedCommunity"
|
|
||||||
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === LinkPreviewMiniCard.Type.Community
|
|
||||||
extend: "loaded"
|
|
||||||
PropertyChanges { target: titleText; text: root.communityName; }
|
|
||||||
},
|
|
||||||
State {
|
|
||||||
name: "loadedChannel"
|
|
||||||
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === LinkPreviewMiniCard.Type.Channel
|
|
||||||
extend: "loaded"
|
|
||||||
PropertyChanges { target: titleText; text: root.communityName; Layout.fillWidth: false; Layout.maximumWidth: Math.min(92, implicitWidth); }
|
|
||||||
PropertyChanges { target: secondTitleText; text: root.channelName; visible: true; }
|
|
||||||
},
|
|
||||||
State {
|
|
||||||
name: "loadedUser"
|
|
||||||
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === LinkPreviewMiniCard.Type.User
|
|
||||||
extend: "loaded"
|
|
||||||
PropertyChanges {
|
PropertyChanges {
|
||||||
target: favIcon
|
target: favIcon
|
||||||
visible: true
|
visible: true
|
||||||
name: root.titleStr
|
name: root.linkData.title
|
||||||
asset.isLetterIdenticon: true
|
asset.isLetterIdenticon: !root.linkData.image
|
||||||
|
asset.color: Theme.palette.baseColor2
|
||||||
|
}
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "loadedImage"
|
||||||
|
when: root.previewState === LinkPreviewMiniCard.State.Loaded &&
|
||||||
|
root.type === Constants.LinkPreviewType.Standard &&
|
||||||
|
root.linkData.type === Constants.StandardLinkPreviewType.Image
|
||||||
|
extend: "loaded"
|
||||||
|
PropertyChanges { target: thumbnailImage; visible: root.linkData.thumbnail != ""; image.source: root.linkData.thumbnail; }
|
||||||
|
PropertyChanges { target: favIcon; visible: true; name: root.linkData.domain; asset.isLetterIdenticon: true; asset.color: Theme.palette.baseColor2; }
|
||||||
|
PropertyChanges { target: subtitleText; visible: true; text: root.linkData.domain; }
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "loadedCommunity"
|
||||||
|
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === Constants.LinkPreviewType.StatusCommunity
|
||||||
|
extend: "loaded"
|
||||||
|
PropertyChanges { target: titleText; text: root.communityData.name; }
|
||||||
|
PropertyChanges { target: subtitleText; visible: true; text: Constants.externalStatusLink; }
|
||||||
|
PropertyChanges {
|
||||||
|
target: favIcon
|
||||||
|
visible: true
|
||||||
|
name: root.communityData.name
|
||||||
|
asset.isLetterIdenticon: root.communityData.image.length === 0
|
||||||
|
asset.color: root.communityData.color
|
||||||
|
asset.name: root.communityData.image
|
||||||
|
}
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "loadedChannel"
|
||||||
|
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === Constants.LinkPreviewType.StatusCommunityChannel
|
||||||
|
extend: "loadedCommunity"
|
||||||
|
PropertyChanges { target: titleText; text: root.channelData.communityData.name; Layout.fillWidth: false; Layout.maximumWidth: Math.min(92, implicitWidth); }
|
||||||
|
PropertyChanges { target: secondTitleText; text: "#" + root.channelData.name; visible: true; }
|
||||||
|
PropertyChanges {
|
||||||
|
target: favIcon
|
||||||
|
visible: true
|
||||||
|
name: root.channelData.communityData.name
|
||||||
|
asset.isLetterIdenticon: root.channelData.communityData.image.length === 0
|
||||||
|
asset.color: root.channelData.communityData.color
|
||||||
|
asset.name: root.channelData.communityData.image
|
||||||
|
}
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
name: "loadedUser"
|
||||||
|
when: root.previewState === LinkPreviewMiniCard.State.Loaded && root.type === Constants.LinkPreviewType.StatusContact
|
||||||
|
extend: "loaded"
|
||||||
|
PropertyChanges { target: titleText; text: root.userData.name; Layout.fillWidth: false; Layout.maximumWidth: Math.min(92, implicitWidth); }
|
||||||
|
PropertyChanges { target: subtitleText; visible: true; text: Constants.externalStatusLink; }
|
||||||
|
PropertyChanges {
|
||||||
|
target: favIcon
|
||||||
|
visible: true
|
||||||
|
name: root.userData.name
|
||||||
|
asset.name: root.userData.image
|
||||||
|
asset.isLetterIdenticon: root.userData.image.length === 0
|
||||||
asset.charactersLen: 2
|
asset.charactersLen: 2
|
||||||
asset.color: Theme.palette.miscColor9
|
asset.color: Theme.palette.miscColor9
|
||||||
}
|
}
|
||||||
|
@ -174,8 +173,6 @@ CalloutCard {
|
||||||
Layout.preferredWidth: 16
|
Layout.preferredWidth: 16
|
||||||
Layout.preferredHeight: 16
|
Layout.preferredHeight: 16
|
||||||
visible: false
|
visible: false
|
||||||
name: root.titleStr
|
|
||||||
asset.name: root.favIconUrl
|
|
||||||
asset.letterSize: asset.charactersLen == 1 ? 10 : 7
|
asset.letterSize: asset.charactersLen == 1 ? 10 : 7
|
||||||
}
|
}
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
@ -190,7 +187,6 @@ CalloutCard {
|
||||||
spacing: 0
|
spacing: 0
|
||||||
StatusBaseText {
|
StatusBaseText {
|
||||||
id: titleText
|
id: titleText
|
||||||
text: root.titleStr
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
font.pixelSize: Style.current.additionalTextSize
|
font.pixelSize: Style.current.additionalTextSize
|
||||||
|
@ -227,7 +223,6 @@ CalloutCard {
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
font.pixelSize: Style.current.tertiaryTextFontSize
|
font.pixelSize: Style.current.tertiaryTextFontSize
|
||||||
color: Theme.palette.baseColor1
|
color: Theme.palette.baseColor1
|
||||||
text: root.domain
|
|
||||||
wrapMode: Text.WordWrap
|
wrapMode: Text.WordWrap
|
||||||
elide: Text.ElideRight
|
elide: Text.ElideRight
|
||||||
}
|
}
|
||||||
|
@ -239,7 +234,6 @@ CalloutCard {
|
||||||
implicitWidth: 34
|
implicitWidth: 34
|
||||||
implicitHeight: 34
|
implicitHeight: 34
|
||||||
radius: 4
|
radius: 4
|
||||||
image.source: root.thumbnailImageUrl
|
|
||||||
visible: false
|
visible: false
|
||||||
}
|
}
|
||||||
StatusFlatButton {
|
StatusFlatButton {
|
||||||
|
|
|
@ -1,78 +0,0 @@
|
||||||
import QtQuick 2.15
|
|
||||||
import QtQuick.Layouts 1.15
|
|
||||||
|
|
||||||
import StatusQ.Core 0.1
|
|
||||||
import StatusQ.Core.Theme 0.1
|
|
||||||
|
|
||||||
import shared.status 1.0
|
|
||||||
import shared.controls 1.0
|
|
||||||
import shared.controls.chat 1.0
|
|
||||||
|
|
||||||
import utils 1.0
|
|
||||||
|
|
||||||
|
|
||||||
CalloutCard {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
required property string userName
|
|
||||||
required property string userPublicKey
|
|
||||||
required property string userBio
|
|
||||||
required property var userImage
|
|
||||||
required property bool ensVerified
|
|
||||||
|
|
||||||
signal clicked()
|
|
||||||
|
|
||||||
implicitWidth: 305
|
|
||||||
implicitHeight: 187
|
|
||||||
|
|
||||||
padding: 12
|
|
||||||
|
|
||||||
contentItem: ColumnLayout {
|
|
||||||
spacing: 0
|
|
||||||
UserImage {
|
|
||||||
Layout.alignment: Qt.AlignLeft | Qt.AlignVCenter
|
|
||||||
name: root.userName
|
|
||||||
pubkey: root.userPublicKey
|
|
||||||
image: root.userImage
|
|
||||||
interactive: false
|
|
||||||
imageWidth: 58
|
|
||||||
imageHeight: imageWidth
|
|
||||||
ensVerified: root.ensVerified
|
|
||||||
}
|
|
||||||
|
|
||||||
StatusBaseText {
|
|
||||||
id: contactName
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.topMargin: 12
|
|
||||||
font.pixelSize: Style.current.additionalTextSize
|
|
||||||
font.weight: Font.Medium
|
|
||||||
elide: Text.ElideRight
|
|
||||||
text: root.userName
|
|
||||||
}
|
|
||||||
|
|
||||||
EmojiHash {
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.topMargin: 4
|
|
||||||
publicKey: root.userPublicKey
|
|
||||||
oneRow: true
|
|
||||||
}
|
|
||||||
|
|
||||||
StatusBaseText {
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.fillHeight: true
|
|
||||||
Layout.topMargin: 15
|
|
||||||
font.pixelSize: Style.current.tertiaryTextFontSize
|
|
||||||
color: Theme.palette.baseColor1
|
|
||||||
text: root.userBio
|
|
||||||
wrapMode: Text.WordWrap
|
|
||||||
elide: Text.ElideRight
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
MouseArea {
|
|
||||||
anchors.fill: root
|
|
||||||
hoverEnabled: true
|
|
||||||
cursorShape: Qt.PointingHandCursor
|
|
||||||
onClicked: root.clicked()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
property string name
|
||||||
|
property string description
|
||||||
|
property string emoji
|
||||||
|
property string color
|
||||||
|
readonly property CommunityData communityData: CommunityData {}
|
||||||
|
}
|
|
@ -0,0 +1,11 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
property string name
|
||||||
|
property string description
|
||||||
|
property string banner
|
||||||
|
property string image
|
||||||
|
property string color
|
||||||
|
property int membersCount
|
||||||
|
property int activeMembersCount: -1 // TODO: implement this and remove the magic number
|
||||||
|
}
|
|
@ -0,0 +1,10 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
property string title
|
||||||
|
property string description
|
||||||
|
property string domain
|
||||||
|
property string thumbnail
|
||||||
|
property string image
|
||||||
|
property int type
|
||||||
|
}
|
|
@ -0,0 +1,9 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
property string name
|
||||||
|
property string publicKey
|
||||||
|
property string bio
|
||||||
|
property string image
|
||||||
|
property bool ensVerified
|
||||||
|
}
|
|
@ -19,6 +19,5 @@ SendTransactionButton 1.0 SendTransactionButton.qml
|
||||||
SentMessage 1.0 SentMessage.qml
|
SentMessage 1.0 SentMessage.qml
|
||||||
StateBubble 1.0 StateBubble.qml
|
StateBubble 1.0 StateBubble.qml
|
||||||
UserImage 1.0 UserImage.qml
|
UserImage 1.0 UserImage.qml
|
||||||
UserProfileCard 1.0 UserProfileCard.qml
|
|
||||||
UsernameLabel 1.0 UsernameLabel.qml
|
UsernameLabel 1.0 UsernameLabel.qml
|
||||||
VerificationLabel 1.0 VerificationLabel.qml
|
VerificationLabel 1.0 VerificationLabel.qml
|
||||||
|
|
|
@ -0,0 +1,91 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
import StatusQ.Core 0.1
|
||||||
|
import StatusQ.Core.Theme 0.1
|
||||||
|
|
||||||
|
import utils 1.0
|
||||||
|
|
||||||
|
import shared.controls.chat 1.0
|
||||||
|
|
||||||
|
LinkPreviewCard {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Model properties
|
||||||
|
* The following properties are required to be set by the user of this component
|
||||||
|
* unfurled: Whether the link has been unfurled or not
|
||||||
|
* empty: Whether the link preview is empty or not
|
||||||
|
* url: The url of the link
|
||||||
|
* immutable: Whether the link preview can be updated
|
||||||
|
* previewType: The type of the preview. See Constants.LinkPreviewType
|
||||||
|
* standardPreview: The standard preview data (title, description, linkType, hostname)
|
||||||
|
* standardPreviewThumbnail: The standard preview thumbnail data (url, dataUri)
|
||||||
|
* statusContactPreview: The status contact preview data (displayName, publicKey, description, icon)
|
||||||
|
* statusContactPreviewThumbnail: The status contact preview thumbnail data (url, dataUri)
|
||||||
|
* statusCommunityPreview: The status community preview data (communityId, displayName, description, membersCount, color)
|
||||||
|
* statusCommunityPreviewIcon: The status community preview icon data (url, dataUri)
|
||||||
|
* statusCommunityPreviewBanner: The status community preview banner data (url, dataUri)
|
||||||
|
* statusCommunityChannelPreview: The status community channel preview data (channelId, displayName, description, emoji, color)
|
||||||
|
* statusCommunityChannelCommunityPreview: The status community channel community preview data (communityId, displayName, description, membersCount, color)
|
||||||
|
* statusCommunityChannelCommunityPreviewIcon: The status community channel community preview icon data (url, dataUri)
|
||||||
|
* statusCommunityChannelCommunityPreviewBanner: The status community channel community preview banner data (url, dataUri)
|
||||||
|
*/
|
||||||
|
required property bool unfurled
|
||||||
|
required property bool empty
|
||||||
|
required property string url
|
||||||
|
required property bool immutable
|
||||||
|
required property int previewType
|
||||||
|
required property var standardPreview
|
||||||
|
required property var standardPreviewThumbnail
|
||||||
|
required property var statusContactPreview
|
||||||
|
required property var statusContactPreviewThumbnail
|
||||||
|
required property var statusCommunityPreview
|
||||||
|
required property var statusCommunityPreviewIcon
|
||||||
|
required property var statusCommunityPreviewBanner
|
||||||
|
required property var statusCommunityChannelPreview
|
||||||
|
required property var statusCommunityChannelCommunityPreview
|
||||||
|
required property var statusCommunityChannelCommunityPreviewIcon
|
||||||
|
required property var statusCommunityChannelCommunityPreviewBanner
|
||||||
|
|
||||||
|
//View properties
|
||||||
|
property bool isCurrentUser: false
|
||||||
|
|
||||||
|
leftTail: !isCurrentUser
|
||||||
|
type: root.previewType
|
||||||
|
linkData {
|
||||||
|
title: standardPreview ? standardPreview.title : ""
|
||||||
|
description: standardPreview ? standardPreview.description : ""
|
||||||
|
domain: standardPreview ? standardPreview.hostname : "" //TODO: Use domainName when available
|
||||||
|
thumbnail: standardPreviewThumbnail ? (standardPreviewThumbnail.url || standardPreviewThumbnail.dataUri) || "" : ""
|
||||||
|
image: "" //TODO: usefavicon when available
|
||||||
|
}
|
||||||
|
userData {
|
||||||
|
name: statusContactPreview ? statusContactPreview.displayName : ""
|
||||||
|
publicKey: statusContactPreview ? statusContactPreview.publicKey : ""
|
||||||
|
bio: statusContactPreview ? statusContactPreview.description : ""
|
||||||
|
image: statusContactPreviewThumbnail ? (statusContactPreviewThumbnail.url || statusContactPreviewThumbnail.dataUri) || "" : ""
|
||||||
|
ensVerified: false // not supported yet
|
||||||
|
}
|
||||||
|
communityData {
|
||||||
|
name: statusCommunityPreview ? statusCommunityPreview.displayName : ""
|
||||||
|
description: statusCommunityPreview ? statusCommunityPreview.description : ""
|
||||||
|
banner: statusCommunityPreviewBanner ? (statusCommunityPreviewBanner.url || statusCommunityPreviewBanner.dataUri) || "" : ""
|
||||||
|
image: statusCommunityPreviewIcon ? (statusCommunityPreviewIcon.url || statusCommunityPreviewIcon.dataUri) || "" : ""
|
||||||
|
membersCount: statusCommunityPreview ? statusCommunityPreview.membersCount : 0
|
||||||
|
color: statusCommunityPreview ? statusCommunityPreview.color : ""
|
||||||
|
}
|
||||||
|
channelData {
|
||||||
|
name: statusCommunityChannelPreview ? statusCommunityChannelPreview.displayName : ""
|
||||||
|
description: statusCommunityChannelPreview ? statusCommunityChannelPreview.description : ""
|
||||||
|
emoji: statusCommunityChannelPreview ? statusCommunityChannelPreview.emoji : ""
|
||||||
|
color: statusCommunityChannelPreview ? statusCommunityChannelPreview.color : ""
|
||||||
|
communityData {
|
||||||
|
name: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.displayName : ""
|
||||||
|
description: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.description : ""
|
||||||
|
banner: statusCommunityChannelCommunityPreviewBanner ? (statusCommunityChannelCommunityPreviewBanner.url || statusCommunityChannelCommunityPreviewBanner.dataUri) || "" : ""
|
||||||
|
image: statusCommunityChannelCommunityPreviewIcon ? (statusCommunityChannelCommunityPreviewIcon.url || statusCommunityChannelCommunityPreviewIcon.dataUri) || "" : ""
|
||||||
|
membersCount: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.membersCount : 0
|
||||||
|
color: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.color : ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,80 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
import utils 1.0
|
||||||
|
|
||||||
|
import shared.controls.chat 1.0
|
||||||
|
import shared.status 1.0
|
||||||
|
|
||||||
|
import StatusQ.Core 0.1
|
||||||
|
|
||||||
|
CalloutCard {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
required property string link
|
||||||
|
required property bool playAnimation
|
||||||
|
required property bool isOnline
|
||||||
|
required property bool isCurrentUser
|
||||||
|
|
||||||
|
readonly property bool isPlaying: linkImage.playing
|
||||||
|
readonly property alias imageAlias: linkImage.imageAlias
|
||||||
|
|
||||||
|
|
||||||
|
signal clicked(var mouse)
|
||||||
|
|
||||||
|
implicitWidth: linkImage.width
|
||||||
|
implicitHeight: linkImage.height
|
||||||
|
leftTail: !isCurrentUser
|
||||||
|
|
||||||
|
StatusChatImageLoader {
|
||||||
|
id: linkImage
|
||||||
|
|
||||||
|
property bool localAnimationEnabled: true
|
||||||
|
|
||||||
|
objectName: "LinksMessageView_unfurledImageComponent_linkImage"
|
||||||
|
anchors.centerIn: parent
|
||||||
|
source: root.link
|
||||||
|
imageWidth: 300
|
||||||
|
isCurrentUser: root.isCurrentUser
|
||||||
|
playing: root.playAnimation && localAnimationEnabled
|
||||||
|
isOnline: root.isOnline
|
||||||
|
asynchronous: true
|
||||||
|
isAnimated: true
|
||||||
|
onClicked: {
|
||||||
|
if (isAnimated && !playing)
|
||||||
|
localAnimationEnabled = true
|
||||||
|
else
|
||||||
|
root.clicked(mouse)
|
||||||
|
}
|
||||||
|
imageAlias.cache: localAnimationEnabled // GIFs can only loop/play properly with cache enabled
|
||||||
|
Loader {
|
||||||
|
width: 45
|
||||||
|
height: 38
|
||||||
|
anchors.left: parent.left
|
||||||
|
anchors.leftMargin: 12
|
||||||
|
anchors.bottom: parent.bottom
|
||||||
|
anchors.bottomMargin: 12
|
||||||
|
active: linkImage.isAnimated && !linkImage.playing
|
||||||
|
sourceComponent: Item {
|
||||||
|
anchors.fill: parent
|
||||||
|
Rectangle {
|
||||||
|
anchors.fill: parent
|
||||||
|
color: "black"
|
||||||
|
radius: Style.current.radius
|
||||||
|
opacity: .4
|
||||||
|
}
|
||||||
|
StatusBaseText {
|
||||||
|
anchors.centerIn: parent
|
||||||
|
text: "GIF"
|
||||||
|
font.pixelSize: 13
|
||||||
|
color: "white"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Timer {
|
||||||
|
id: animationPlayingTimer
|
||||||
|
interval: 10000
|
||||||
|
running: linkImage.isAnimated && linkImage.playing
|
||||||
|
onTriggered: { linkImage.localAnimationEnabled = false }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,86 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
|
||||||
|
import utils 1.0
|
||||||
|
import shared.controls.chat 1.0
|
||||||
|
|
||||||
|
LinkPreviewMiniCard {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Model properties
|
||||||
|
* The following properties are required to be set by the user of this component
|
||||||
|
* unfurled: Whether the link has been unfurled or not
|
||||||
|
* empty: Whether the link preview is empty or not
|
||||||
|
* url: The url of the link
|
||||||
|
* immutable: Whether the link preview can be updated
|
||||||
|
* previewType: The type of the preview. See Constants.LinkPreviewType
|
||||||
|
* standardPreview: The standard preview data (title, description, linkType, hostname)
|
||||||
|
* standardPreviewThumbnail: The standard preview thumbnail data (url, dataUri)
|
||||||
|
* statusContactPreview: The status contact preview data (displayName, publicKey, description, icon)
|
||||||
|
* statusContactPreviewThumbnail: The status contact preview thumbnail data (url, dataUri)
|
||||||
|
* statusCommunityPreview: The status community preview data (communityId, displayName, description, membersCount, color)
|
||||||
|
* statusCommunityPreviewIcon: The status community preview icon data (url, dataUri)
|
||||||
|
* statusCommunityPreviewBanner: The status community preview banner data (url, dataUri)
|
||||||
|
* statusCommunityChannelPreview: The status community channel preview data (channelId, displayName, description, emoji, color)
|
||||||
|
* statusCommunityChannelCommunityPreview: The status community channel community preview data (communityId, displayName, description, membersCount, color)
|
||||||
|
* statusCommunityChannelCommunityPreviewIcon: The status community channel community preview icon data (url, dataUri)
|
||||||
|
* statusCommunityChannelCommunityPreviewBanner: The status community channel community preview banner data (url, dataUri)
|
||||||
|
*/
|
||||||
|
required property bool unfurled
|
||||||
|
required property bool empty
|
||||||
|
required property string url
|
||||||
|
required property bool immutable
|
||||||
|
required property int previewType
|
||||||
|
required property var standardPreview
|
||||||
|
required property var standardPreviewThumbnail
|
||||||
|
required property var statusContactPreview
|
||||||
|
required property var statusContactPreviewThumbnail
|
||||||
|
required property var statusCommunityPreview
|
||||||
|
required property var statusCommunityPreviewIcon
|
||||||
|
required property var statusCommunityPreviewBanner
|
||||||
|
required property var statusCommunityChannelPreview
|
||||||
|
required property var statusCommunityChannelCommunityPreview
|
||||||
|
required property var statusCommunityChannelCommunityPreviewIcon
|
||||||
|
required property var statusCommunityChannelCommunityPreviewBanner
|
||||||
|
|
||||||
|
previewState: !root.unfurled ? LinkPreviewMiniCard.State.Loading : root.unfurled && !root.empty ? LinkPreviewMiniCard.State.Loaded : LinkPreviewMiniCard.State.LoadingFailed
|
||||||
|
type: root.previewType
|
||||||
|
|
||||||
|
linkData {
|
||||||
|
title: standardPreview ? standardPreview.title : ""
|
||||||
|
description: standardPreview ? standardPreview.description : ""
|
||||||
|
domain: standardPreview ? standardPreview.hostname : "" //TODO: Use domainName when available
|
||||||
|
thumbnail: standardPreviewThumbnail ? (standardPreviewThumbnail.url || standardPreviewThumbnail.dataUri) || "" : ""
|
||||||
|
image: "" //TODO: usefavicon when available
|
||||||
|
type: standardPreview ? standardPreview.linkType : Constants.StandardLinkPreviewType.Link
|
||||||
|
}
|
||||||
|
userData {
|
||||||
|
name: statusContactPreview ? statusContactPreview.displayName : ""
|
||||||
|
publicKey: statusContactPreview ? statusContactPreview.publicKey : ""
|
||||||
|
bio: statusContactPreview ? statusContactPreview.description : ""
|
||||||
|
image: statusContactPreviewThumbnail ? (statusContactPreviewThumbnail.url || statusContactPreviewThumbnail.dataUri) || "" : ""
|
||||||
|
ensVerified: false // not supported yet
|
||||||
|
}
|
||||||
|
communityData {
|
||||||
|
name: statusCommunityPreview ? statusCommunityPreview.displayName : ""
|
||||||
|
description: statusCommunityPreview ? statusCommunityPreview.description : ""
|
||||||
|
banner: statusCommunityPreviewBanner ? (statusCommunityPreviewBanner.url || statusCommunityPreviewBanner.dataUri) || "" : ""
|
||||||
|
image: statusCommunityPreviewIcon ? (statusCommunityPreviewIcon.url || statusCommunityPreviewIcon.dataUri) || "" : ""
|
||||||
|
membersCount: statusCommunityPreview ? statusCommunityPreview.membersCount : 0
|
||||||
|
color: statusCommunityPreview ? statusCommunityPreview.color : ""
|
||||||
|
}
|
||||||
|
channelData {
|
||||||
|
name: statusCommunityChannelPreview ? statusCommunityChannelPreview.displayName : ""
|
||||||
|
description: statusCommunityChannelPreview ? statusCommunityChannelPreview.description : ""
|
||||||
|
emoji: statusCommunityChannelPreview ? statusCommunityChannelPreview.emoji : ""
|
||||||
|
color: statusCommunityChannelPreview ? statusCommunityChannelPreview.color : ""
|
||||||
|
communityData {
|
||||||
|
name: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.displayName : ""
|
||||||
|
description: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.description : ""
|
||||||
|
banner: statusCommunityChannelCommunityPreviewBanner ? (statusCommunityChannelCommunityPreviewBanner.url || statusCommunityChannelCommunityPreviewBanner.dataUri) || "" : ""
|
||||||
|
image: statusCommunityChannelCommunityPreviewIcon ? (statusCommunityChannelCommunityPreviewIcon.url || statusCommunityChannelCommunityPreviewIcon.dataUri) || "" : ""
|
||||||
|
membersCount: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.membersCount : 0
|
||||||
|
color: statusCommunityChannelCommunityPreview ? statusCommunityChannelCommunityPreview.color : ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1 +1,4 @@
|
||||||
ContactListItemDelegate 1.0 ContactListItemDelegate.qml
|
ContactListItemDelegate 1.0 ContactListItemDelegate.qml
|
||||||
|
LinkPreviewCardDelegate 1.0 LinkPreviewCardDelegate.qml
|
||||||
|
LinkPreviewGifDelegate 1.0 LinkPreviewGifDelegate.qml
|
||||||
|
LinkPreviewMiniCardDelegate 1.0 LinkPreviewMiniCardDelegate.qml
|
||||||
|
|
|
@ -1201,7 +1201,7 @@ Rectangle {
|
||||||
}
|
}
|
||||||
control.fileUrlsAndSources = urls
|
control.fileUrlsAndSources = urls
|
||||||
}
|
}
|
||||||
onImageClicked: (chatImage) => Global.openImagePopup(chatImage)
|
onImageClicked: (chatImage) => Global.openImagePopup(chatImage, "")
|
||||||
onLinkReload: (link) => control.linkPreviewReloaded(link)
|
onLinkReload: (link) => control.linkPreviewReloaded(link)
|
||||||
onLinkClicked: (link) => Global.openLink(link)
|
onLinkClicked: (link) => Global.openLink(link)
|
||||||
onEnableLinkPreview: () => control.enableLinkPreview()
|
onEnableLinkPreview: () => control.enableLinkPreview()
|
||||||
|
|
|
@ -7,6 +7,8 @@ StatusMenu {
|
||||||
|
|
||||||
property string url
|
property string url
|
||||||
property string imageSource
|
property string imageSource
|
||||||
|
property string domain
|
||||||
|
property bool requireConfirmationOnOpen: false
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
id: d
|
id: d
|
||||||
|
@ -43,6 +45,6 @@ StatusMenu {
|
||||||
text: qsTr("Open link")
|
text: qsTr("Open link")
|
||||||
icon.name: "browser"
|
icon.name: "browser"
|
||||||
enabled: d.isUnfurled
|
enabled: d.isUnfurled
|
||||||
onTriggered: Global.openLink(root.url)
|
onTriggered: requireConfirmationOnOpen ? Global.openLinkWithConfirmation(root.url, root.domain) : Global.openLink(root.url)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,24 +10,29 @@ import StatusQ.Components 0.1
|
||||||
import shared.controls 1.0
|
import shared.controls 1.0
|
||||||
import shared.status 1.0
|
import shared.status 1.0
|
||||||
import shared.panels 1.0
|
import shared.panels 1.0
|
||||||
import shared.stores 1.0
|
|
||||||
import shared.controls.chat 1.0
|
import shared.controls.delegates 1.0
|
||||||
|
|
||||||
Flow {
|
Flow {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
required property var store
|
required property bool isOnline
|
||||||
required property var messageStore
|
required property bool playAnimations
|
||||||
|
|
||||||
required property var linkPreviewModel
|
required property var linkPreviewModel
|
||||||
required property var gifLinks
|
required property var gifLinks
|
||||||
|
|
||||||
required property bool isCurrentUser
|
required property bool isCurrentUser
|
||||||
|
|
||||||
|
required property bool gifUnfurlingEnabled
|
||||||
|
required property bool canAskToUnfurlGifs
|
||||||
|
|
||||||
readonly property alias hoveredLink: linksRepeater.hoveredUrl
|
readonly property alias hoveredLink: linksRepeater.hoveredUrl
|
||||||
property string highlightLink: ""
|
property string highlightLink: ""
|
||||||
|
|
||||||
signal imageClicked(var image, var mouse, var imageSource, string url)
|
signal imageClicked(var image, var mouse, string imageSource, string url)
|
||||||
|
signal openContextMenu(var item, string url, string domain)
|
||||||
|
signal setNeverAskAboutUnfurlingAgain(bool neverAskAgain)
|
||||||
|
|
||||||
function resetLocalAskAboutUnfurling() {
|
function resetLocalAskAboutUnfurling() {
|
||||||
d.localAskAboutUnfurling = true
|
d.localAskAboutUnfurling = true
|
||||||
|
@ -45,15 +50,25 @@ Flow {
|
||||||
Loader {
|
Loader {
|
||||||
visible: active
|
visible: active
|
||||||
active: root.gifLinks && root.gifLinks.length > 0
|
active: root.gifLinks && root.gifLinks.length > 0
|
||||||
&& !RootStore.gifUnfurlingEnabled
|
&& !root.gifUnfurlingEnabled
|
||||||
&& d.localAskAboutUnfurling && !RootStore.neverAskAboutUnfurlingAgain
|
&& d.localAskAboutUnfurling && root.canAskToUnfurlGifs
|
||||||
sourceComponent: enableLinkComponent
|
sourceComponent: enableLinkComponent
|
||||||
}
|
}
|
||||||
|
|
||||||
Repeater {
|
Repeater {
|
||||||
id: tempRepeater
|
id: tempRepeater
|
||||||
model: RootStore.gifUnfurlingEnabled ? gifLinks : []
|
visible: root.canAskToUnfurlGifs
|
||||||
delegate: gifComponent
|
model: root.gifUnfurlingEnabled ? gifLinks : []
|
||||||
|
|
||||||
|
delegate: LinkPreviewGifDelegate {
|
||||||
|
required property string modelData
|
||||||
|
|
||||||
|
link: modelData
|
||||||
|
isOnline: root.isOnline
|
||||||
|
isCurrentUser: root.isCurrentUser
|
||||||
|
playAnimation: root.playAnimations
|
||||||
|
onClicked: root.imageClicked(imageAlias, mouse, link, link)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Repeater {
|
Repeater {
|
||||||
|
@ -62,154 +77,26 @@ Flow {
|
||||||
property string hoveredUrl: ""
|
property string hoveredUrl: ""
|
||||||
|
|
||||||
model: root.linkPreviewModel
|
model: root.linkPreviewModel
|
||||||
delegate: Loader {
|
delegate: LinkPreviewCardDelegate {
|
||||||
id: linkMessageLoader
|
id: delegate
|
||||||
// properties from the model
|
isCurrentUser: root.isCurrentUser
|
||||||
|
highlight: url === root.highlightLink
|
||||||
required property bool unfurled
|
onHoveredChanged: {
|
||||||
required property bool empty
|
linksRepeater.hoveredUrl = hovered ? url : ""
|
||||||
required property string url
|
|
||||||
required property bool immutable
|
|
||||||
required property int previewType
|
|
||||||
required property var standardPreview
|
|
||||||
required property var standardPreviewThumbnail
|
|
||||||
required property var statusContactPreview
|
|
||||||
required property var statusContactPreviewThumbnail
|
|
||||||
required property var statusCommunityPreview
|
|
||||||
required property var statusCommunityPreviewIcon
|
|
||||||
required property var statusCommunityPreviewBanner
|
|
||||||
required property var statusCommunityChannelPreview
|
|
||||||
required property var statusCommunityChannelCommunityPreview
|
|
||||||
required property var statusCommunityChannelCommunityPreviewIcon
|
|
||||||
required property var statusCommunityChannelCommunityPreviewBanner
|
|
||||||
|
|
||||||
readonly property string hostname: standardPreview ? standardPreview.hostname : ""
|
|
||||||
readonly property string title: standardPreview ? standardPreview.title : ""
|
|
||||||
readonly property string description: standardPreview ? standardPreview.description : ""
|
|
||||||
readonly property int standardLinkType: standardPreview ? standardPreview.linkType : ""
|
|
||||||
readonly property int thumbnailWidth: standardPreviewThumbnail ? standardPreviewThumbnail.width : ""
|
|
||||||
readonly property int thumbnailHeight: standardPreviewThumbnail ? standardPreviewThumbnail.height : ""
|
|
||||||
readonly property string thumbnailUrl: standardPreviewThumbnail ? standardPreviewThumbnail.url : ""
|
|
||||||
readonly property string thumbnailDataUri: standardPreviewThumbnail ? standardPreviewThumbnail.dataUri : ""
|
|
||||||
|
|
||||||
asynchronous: true
|
|
||||||
active: unfurled && !empty
|
|
||||||
|
|
||||||
StateGroup {
|
|
||||||
//Using StateGroup as a warkardound for https://bugreports.qt.io/browse/QTBUG-47796
|
|
||||||
states: [
|
|
||||||
State {
|
|
||||||
name: "standardLinkPreview"
|
|
||||||
when: linkMessageLoader.previewType === Constants.LinkPreviewType.Standard
|
|
||||||
PropertyChanges { target: linkMessageLoader; sourceComponent: standardLinkPreviewCard }
|
|
||||||
},
|
|
||||||
State {
|
|
||||||
name: "statusContactLinkPreview"
|
|
||||||
when: linkMessageLoader.previewType === Constants.LinkPreviewType.StatusContact
|
|
||||||
PropertyChanges { target: linkMessageLoader; sourceComponent: unfurledProfileLinkComponent }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Component {
|
|
||||||
id: standardLinkPreviewCard
|
|
||||||
LinkPreviewCard {
|
|
||||||
leftTail: !root.isCurrentUser // WARNING: Is this by design?
|
|
||||||
bannerImageSource: standardPreviewThumbnail ? standardPreviewThumbnail.url : ""
|
|
||||||
title: standardPreview ? standardPreview.title : ""
|
|
||||||
description: standardPreview ? standardPreview.description : ""
|
|
||||||
footer: standardPreview ? standardPreview.hostname : ""
|
|
||||||
highlight: root.highlightLink === url
|
|
||||||
onClicked: (mouse) => {
|
onClicked: (mouse) => {
|
||||||
switch (mouse.button) {
|
if(mouse.button === Qt.RightButton) {
|
||||||
case Qt.RightButton:
|
const domain = previewType === Constants.LinkPreviewType.Standard ? linkData.domain : Constants.externalStatusLink
|
||||||
root.imageClicked(unfurledLink, mouse, "", url) // request a dumb context menu with just "copy/open link" items
|
root.openContextMenu(delegate, url, domain)
|
||||||
break
|
return
|
||||||
default:
|
|
||||||
Global.openLinkWithConfirmation(url, hostname)
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Component {
|
if(previewType === Constants.LinkPreviewType.Standard) {
|
||||||
id: unfurledProfileLinkComponent
|
Global.openLinkWithConfirmation(url, linkData.domain)
|
||||||
UserProfileCard {
|
return
|
||||||
id: unfurledProfileLink
|
}
|
||||||
|
|
||||||
leftTail: !root.isCurrentUser
|
Global.activateDeepLink(url)
|
||||||
userName: statusContactPreview && statusContactPreview.displayName ? statusContactPreview.displayName : ""
|
|
||||||
userPublicKey: statusContactPreview && statusContactPreview.publicKey ? statusContactPreview.publicKey : ""
|
|
||||||
userBio: statusContactPreview && statusContactPreview.description ? statusContactPreview.description : ""
|
|
||||||
userImage: statusContactPreviewThumbnail ? statusContactPreviewThumbnail.url : ""
|
|
||||||
ensVerified: false // not supported yet
|
|
||||||
onClicked: {
|
|
||||||
Global.openProfilePopup(userPublicKey)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//TODO: Remove this once we have gif support in new unfurling flow
|
|
||||||
Component {
|
|
||||||
id: gifComponent
|
|
||||||
CalloutCard {
|
|
||||||
implicitWidth: linkImage.width
|
|
||||||
implicitHeight: linkImage.height
|
|
||||||
leftTail: !root.isCurrentUser
|
|
||||||
StatusChatImageLoader {
|
|
||||||
id: linkImage
|
|
||||||
readonly property bool globalAnimationEnabled: root.messageStore.playAnimation
|
|
||||||
readonly property string urlLink: modelData
|
|
||||||
property bool localAnimationEnabled: true
|
|
||||||
objectName: "LinksMessageView_unfurledImageComponent_linkImage"
|
|
||||||
anchors.centerIn: parent
|
|
||||||
source: urlLink
|
|
||||||
imageWidth: 300
|
|
||||||
isCurrentUser: root.isCurrentUser
|
|
||||||
playing: globalAnimationEnabled && localAnimationEnabled
|
|
||||||
isOnline: root.store.mainModuleInst.isOnline
|
|
||||||
asynchronous: true
|
|
||||||
isAnimated: true
|
|
||||||
onClicked: {
|
|
||||||
if (!playing)
|
|
||||||
localAnimationEnabled = true
|
|
||||||
else
|
|
||||||
root.imageClicked(linkImage.imageAlias, mouse, source, urlLink)
|
|
||||||
}
|
|
||||||
imageAlias.cache: localAnimationEnabled // GIFs can only loop/play properly with cache enabled
|
|
||||||
Loader {
|
|
||||||
width: 45
|
|
||||||
height: 38
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.leftMargin: 12
|
|
||||||
anchors.bottom: parent.bottom
|
|
||||||
anchors.bottomMargin: 12
|
|
||||||
active: linkImage.isAnimated && !linkImage.playing
|
|
||||||
sourceComponent: Item {
|
|
||||||
anchors.fill: parent
|
|
||||||
Rectangle {
|
|
||||||
anchors.fill: parent
|
|
||||||
color: "black"
|
|
||||||
radius: Style.current.radius
|
|
||||||
opacity: .4
|
|
||||||
}
|
|
||||||
StatusBaseText {
|
|
||||||
anchors.centerIn: parent
|
|
||||||
text: "GIF"
|
|
||||||
font.pixelSize: 13
|
|
||||||
color: "white"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Timer {
|
|
||||||
id: animationPlayingTimer
|
|
||||||
interval: 10000
|
|
||||||
running: linkImage.isAnimated && linkImage.playing
|
|
||||||
onTriggered: { linkImage.localAnimationEnabled = false }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -308,7 +195,7 @@ Flow {
|
||||||
text: qsTr("Don't ask me again")
|
text: qsTr("Don't ask me again")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
onClicked: RootStore.setNeverAskAboutUnfurlingAgain(true)
|
onClicked: root.setNeverAskAboutUnfurlingAgain(true)
|
||||||
Component.onCompleted: {
|
Component.onCompleted: {
|
||||||
background.radius = Style.current.padding;
|
background.radius = Style.current.padding;
|
||||||
}
|
}
|
||||||
|
@ -316,5 +203,5 @@ Flow {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,7 @@ import shared.controls 1.0
|
||||||
import shared.popups 1.0
|
import shared.popups 1.0
|
||||||
import shared.views.chat 1.0
|
import shared.views.chat 1.0
|
||||||
import shared.controls.chat 1.0
|
import shared.controls.chat 1.0
|
||||||
|
import shared.stores 1.0
|
||||||
|
|
||||||
import StatusQ.Core 0.1
|
import StatusQ.Core 0.1
|
||||||
import StatusQ.Core.Theme 0.1
|
import StatusQ.Core.Theme 0.1
|
||||||
|
@ -570,7 +571,7 @@ Loader {
|
||||||
rootStore.chatCommunitySectionModule.switchToChannel(link.replace("#", ""))
|
rootStore.chatCommunitySectionModule.switchToChannel(link.replace("#", ""))
|
||||||
return
|
return
|
||||||
} else if (Utils.isStatusDeepLink(link)) {
|
} else if (Utils.isStatusDeepLink(link)) {
|
||||||
rootStore.activateStatusDeepLink(link)
|
Global.activateDeepLink(link)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -755,14 +756,20 @@ Loader {
|
||||||
id: linksMessageView
|
id: linksMessageView
|
||||||
linkPreviewModel: root.linkPreviewModel
|
linkPreviewModel: root.linkPreviewModel
|
||||||
gifLinks: root.gifLinks
|
gifLinks: root.gifLinks
|
||||||
messageStore: root.messageStore
|
playAnimations: root.messageStore.playAnimation
|
||||||
store: root.rootStore
|
isOnline: root.rootStore.mainModuleInst.isOnline
|
||||||
isCurrentUser: root.amISender
|
isCurrentUser: root.amISender
|
||||||
highlightLink: delegate.hoveredLink
|
highlightLink: delegate.hoveredLink
|
||||||
onImageClicked: (image, mouse, imageSource, url) => {
|
onImageClicked: (image, mouse, imageSource, url) => {
|
||||||
d.onImageClicked(image, mouse, imageSource, url)
|
d.onImageClicked(image, mouse, imageSource, url)
|
||||||
}
|
}
|
||||||
|
onOpenContextMenu: (item, url, domain) => {
|
||||||
|
Global.openMenu(imageContextMenuComponent, item, { url: url, domain: domain, requireConfirmationOnOpen: true })
|
||||||
|
}
|
||||||
onHoveredLinkChanged: delegate.highlightedLink = linksMessageView.hoveredLink
|
onHoveredLinkChanged: delegate.highlightedLink = linksMessageView.hoveredLink
|
||||||
|
gifUnfurlingEnabled: RootStore.gifUnfurlingEnabled
|
||||||
|
canAskToUnfurlGifs: !RootStore.neverAskAboutUnfurlingAgain
|
||||||
|
onSetNeverAskAboutUnfurlingAgain: RootStore.setNeverAskAboutUnfurlingAgain(neverAskAgain)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -56,6 +56,7 @@ QtObject {
|
||||||
|
|
||||||
signal openLink(string link)
|
signal openLink(string link)
|
||||||
signal openLinkWithConfirmation(string link, string domain)
|
signal openLinkWithConfirmation(string link, string domain)
|
||||||
|
signal activateDeepLink(string link)
|
||||||
|
|
||||||
signal setNthEnabledSectionActive(int nthSection)
|
signal setNthEnabledSectionActive(int nthSection)
|
||||||
signal appSectionBySectionTypeChanged(int sectionType, int subsection)
|
signal appSectionBySectionTypeChanged(int sectionType, int subsection)
|
||||||
|
|
Loading…
Reference in New Issue