status-desktop/ui/imports/shared/views/profile/ProfileShowcaseCollectiblesView.qml
2024-04-05 12:31:11 +03:00

209 lines
7.8 KiB
QML

import QtQuick 2.15
import QtQuick.Controls 2.15
import QtQuick.Layouts 1.15
import QtQml.Models 2.15
import StatusQ.Core 0.1
import StatusQ.Components 0.1
import StatusQ.Controls 0.1
import StatusQ.Core.Theme 0.1
import StatusQ.Popups.Dialog 0.1
import StatusQ.Core.Utils 0.1 as StatusQUtils
import StatusQ.Popups 0.1
import shared.controls 1.0
import utils 1.0
Item {
id: root
required property string mainDisplayName
required property var collectiblesModel
required property var walletStore
property alias cellWidth: collectiblesView.cellWidth
property alias cellHeight: collectiblesView.cellHeight
signal closeRequested()
signal visitCommunity(var model)
StatusBaseText {
anchors.centerIn: parent
visible: (collectiblesView.count === 0)
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
color: Theme.palette.directColor1
text: qsTr("%1 has not shared any collectibles").arg(root.mainDisplayName)
}
StatusGridView {
id: collectiblesView
anchors.fill: parent
topMargin: Style.current.bigPadding
bottomMargin: Style.current.bigPadding
leftMargin: Style.current.bigPadding
visible: count
model: root.collectiblesModel
ScrollBar.vertical: StatusScrollBar { anchors.right: parent.right; anchors.rightMargin: width / 2 }
delegate: Item {
id: delegateItem
function getCollectibleURL() {
const networkShortName = root.walletStore.getNetworkShortNames(model.chainId);
return root.walletStore.getOpenSeaCollectibleUrl(networkShortName, model.contractAddress, model.tokenId)
}
function openCollectibleURL() {
const link = getCollectibleURL();
Global.openLinkWithConfirmation(link, StatusQUtils.StringUtils.extractDomainFromLink(link));
}
function openCollectionURL() {
let networkShortName = root.walletStore.getNetworkShortNames(model.chainId);
let link = root.walletStore.getOpenSeaCollectionUrl(networkShortName, model.contractAddress)
Global.openLinkWithConfirmation(link, StatusQUtils.StringUtils.extractDomainFromLink(link));
}
width: GridView.view.cellWidth - Style.current.padding
height: GridView.view.cellHeight - Style.current.padding
HoverHandler {
id: hoverHandler
cursorShape: hovered ? Qt.PointingHandCursor : undefined
}
StatusRoundedImage {
id: collectibleImage
anchors.fill: parent
color: !!model.backgroundColor ? model.backgroundColor : "transparent"
radius: Style.current.radius
showLoadingIndicator: true
isLoading: image.isLoading || !model.imageUrl
image.fillMode: Image.PreserveAspectCrop
image.source: model.imageUrl ?? ""
TapHandler {
acceptedButtons: Qt.LeftButton | Qt.RightButton
onSingleTapped: {
if ((eventPoint.event.button === Qt.LeftButton) && (model.communityId !== "")) {
root.visitCommunity(model)
} else {
if (eventPoint.event.button === Qt.LeftButton) {
delegateItem.openCollectibleURL()
} else {
Global.openMenu(delegatesActionsMenu, collectibleImage, { communityId: model.communityId, url: getCollectibleURL()});
}
}
}
}
}
Image {
id: gradient
anchors.fill: collectibleImage
visible: hoverHandler.hovered
source: Style.png("profile/gradient")
}
//TODO Add drop shadow
Control {
id: amountControl
width: (amountText.contentWidth + Style.current.padding)
height: 24
anchors.left: parent.left
anchors.leftMargin: 12
anchors.top: parent.top
anchors.topMargin: 12
//TODO TBD, https://github.com/status-im/status-desktop/issues/13782
visible: (model.userHas > 1)
background: Rectangle {
radius: 30
color: amountControl.hovered ? Theme.palette.indirectColor1 : Theme.palette.indirectColor2
}
contentItem: StatusBaseText {
id: amountText
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
font.pixelSize: Style.current.asideTextFontSize
text: "x"+model.userHas
}
}
StatusRoundButton {
implicitWidth: 24
implicitHeight: 24
anchors.right: parent.right
anchors.rightMargin: 12
anchors.top: parent.top
anchors.topMargin: 12
visible: (hoverHandler.hovered && model.communityId === "")
type: StatusFlatRoundButton.Type.Secondary
icon.name: "external"
icon.width: 16
icon.height: 16
radius: width/2
icon.color: Theme.palette.directColor1
icon.hoverColor: icon.color
color: hovered ? Theme.palette.indirectColor1 : Theme.palette.indirectColor2
onClicked: {
delegateItem.openCollectibleURL()
}
}
ExpandableTag {
id: expandableTag
readonly property bool isCommunity: model.communityId != ""
readonly property bool isCollection: model.collectionUid != ""
visible: isCommunity || (isCollection && hoverHandler.hovered)
tagHeaderText: model.name ?? ""
tagName: isCommunity ? (model.communityName ?? "")
: (model.collectionName ?? "")
tagImage: isCommunity ? (model.communityImage ?? "")
: (hovered ? "external" : "gallery")
isIcon: !isCommunity
backgroundColor: hovered ? Style.current.background : Theme.palette.indirectColor2
expanded: hoverHandler.hovered || hovered
onTagClicked: {
if (isCommunity) {
Global.switchToCommunity(model.communityId);
root.closeRequested();
} else {
delegateItem.openCollectionURL()
}
}
}
}
}
Component {
id: delegatesActionsMenu
StatusMenu {
id: contextMenu
property string url
property string communityId
StatusAction {
text: qsTr("Visit community")
enabled: !!contextMenu.communityId
icon.name: "communities"
onTriggered: {
Global.switchToCommunity(contextMenu.communityId);
root.closeRequested();
}
}
StatusAction {
text: qsTr("View on Opensea")
enabled: contextMenu.communityId === ""
icon.name: "link"
onTriggered: {
Global.openLinkWithConfirmation(contextMenu.url, StatusQUtils.StringUtils.extractDomainFromLink(contextMenu.url));
}
}
}
}
}