2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
import QtQml 2.15
|
|
|
|
import QtQuick 2.13
|
|
|
|
import QtQuick.Layouts 1.13
|
|
|
|
|
|
|
|
import shared.controls 1.0
|
2024-01-30 14:15:58 +01:00
|
|
|
import shared.popups 1.0
|
2024-02-20 08:43:31 +01:00
|
|
|
import shared.popups.send 1.0
|
2023-09-11 07:20:36 -03:00
|
|
|
import utils 1.0
|
|
|
|
|
|
|
|
import SortFilterProxyModel 0.2
|
|
|
|
|
2023-10-30 23:18:58 +01:00
|
|
|
import StatusQ 0.1
|
2023-09-11 07:20:36 -03:00
|
|
|
import StatusQ.Controls 0.1
|
|
|
|
import StatusQ.Core 0.1
|
|
|
|
import StatusQ.Core.Theme 0.1
|
|
|
|
|
|
|
|
import "../controls"
|
|
|
|
|
|
|
|
Item {
|
|
|
|
id: root
|
2023-11-28 20:16:18 +01:00
|
|
|
|
2023-09-11 07:20:36 -03:00
|
|
|
property var assetsModel
|
2023-11-28 20:16:18 +01:00
|
|
|
property string selectedSenderAccount
|
2023-09-11 07:20:36 -03:00
|
|
|
property var collectiblesModel
|
2023-10-30 23:18:58 +01:00
|
|
|
property var networksModel
|
2023-09-11 07:20:36 -03:00
|
|
|
property string currentCurrencySymbol
|
|
|
|
property bool onlyAssets: true
|
2024-01-30 14:15:58 +01:00
|
|
|
property string searchText
|
2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
implicitWidth: holdingItemSelector.implicitWidth
|
|
|
|
implicitHeight: holdingItemSelector.implicitHeight
|
|
|
|
|
2024-01-30 14:15:58 +01:00
|
|
|
property var formatCurrentCurrencyAmount: function(balance){}
|
|
|
|
property var formatCurrencyAmountFromBigInt: function(balance, symbol, decimals){}
|
2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
signal itemHovered(string holdingId, var holdingType)
|
|
|
|
signal itemSelected(string holdingId, var holdingType)
|
|
|
|
|
|
|
|
property alias selectedItem: holdingItemSelector.selectedItem
|
|
|
|
property alias hoveredItem: holdingItemSelector.hoveredItem
|
|
|
|
|
|
|
|
function setSelectedItem(item, holdingType) {
|
|
|
|
d.browsingHoldingType = holdingType
|
|
|
|
holdingItemSelector.selectedItem = null
|
|
|
|
d.currentHoldingType = holdingType
|
|
|
|
holdingItemSelector.selectedItem = item
|
|
|
|
}
|
|
|
|
|
|
|
|
function setHoveredItem(item, holdingType) {
|
|
|
|
d.browsingHoldingType = holdingType
|
|
|
|
holdingItemSelector.hoveredItem = null
|
|
|
|
d.currentHoldingType = holdingType
|
|
|
|
holdingItemSelector.hoveredItem = item
|
|
|
|
}
|
|
|
|
|
|
|
|
QtObject {
|
|
|
|
id: d
|
|
|
|
// Internal management properties and signals:
|
|
|
|
readonly property var holdingTypes: onlyAssets ?
|
2023-11-07 23:45:47 +01:00
|
|
|
[Constants.TokenType.ERC20] :
|
|
|
|
[Constants.TokenType.ERC20, Constants.TokenType.ERC721]
|
2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
readonly property var tabsModel: onlyAssets ?
|
|
|
|
[qsTr("Assets")] :
|
|
|
|
[qsTr("Assets"), qsTr("Collectibles")]
|
|
|
|
|
|
|
|
readonly property var updateSearchText: Backpressure.debounce(root, 1000, function(inputText) {
|
|
|
|
searchText = inputText
|
|
|
|
})
|
2024-02-20 08:43:31 +01:00
|
|
|
|
2023-09-11 07:20:36 -03:00
|
|
|
function isAsset(type) {
|
2023-11-07 23:45:47 +01:00
|
|
|
return type === Constants.TokenType.ERC20
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
2023-11-07 23:45:47 +01:00
|
|
|
property int browsingHoldingType: Constants.TokenType.ERC20
|
2023-09-11 07:20:36 -03:00
|
|
|
readonly property bool isCurrentBrowsingTypeAsset: isAsset(browsingHoldingType)
|
|
|
|
readonly property bool isBrowsingCollection: !isCurrentBrowsingTypeAsset && !!collectiblesModel && collectiblesModel.currentCollectionUid !== ""
|
|
|
|
property string currentBrowsingCollectionName
|
|
|
|
|
2023-11-07 23:45:47 +01:00
|
|
|
property var currentHoldingType: Constants.TokenType.Unknown
|
2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
readonly property string uppercaseSearchText: searchText.toUpperCase()
|
|
|
|
|
|
|
|
property var assetTextFn: function (asset) {
|
2023-09-12 16:26:38 +02:00
|
|
|
return !!asset && asset.symbol ? asset.symbol : ""
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
property var assetIconSourceFn: function (asset) {
|
2023-09-12 16:26:38 +02:00
|
|
|
return !!asset && asset.symbol ? Style.png("tokens/%1".arg(asset.symbol)) : ""
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
property var collectibleTextFn: function (item) {
|
|
|
|
if (!!item) {
|
|
|
|
return !!item.collectionName ? item.collectionName + ": " + item.name : item.name
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
property var collectibleIconSourceFn: function (item) {
|
2023-09-12 16:26:38 +02:00
|
|
|
return !!item && item.iconUrl ? item.iconUrl : ""
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
2023-10-30 23:18:58 +01:00
|
|
|
readonly property RolesRenamingModel renamedAllNetworksModel: RolesRenamingModel {
|
|
|
|
sourceModel: root.networksModel
|
|
|
|
mapping: RoleRename {
|
|
|
|
from: "iconUrl"
|
|
|
|
to: "networkIconUrl"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
readonly property LeftJoinModel collectibleNetworksJointModel: LeftJoinModel {
|
|
|
|
leftModel: root.collectiblesModel
|
|
|
|
rightModel: d.renamedAllNetworksModel
|
|
|
|
joinRole: "chainId"
|
|
|
|
}
|
|
|
|
|
2023-09-11 07:20:36 -03:00
|
|
|
property var collectibleComboBoxModel: SortFilterProxyModel {
|
2023-10-30 23:18:58 +01:00
|
|
|
sourceModel: d.collectibleNetworksJointModel
|
2024-02-20 08:43:31 +01:00
|
|
|
proxyRoles: [
|
|
|
|
FastExpressionRole {
|
|
|
|
name: "isCommunityAsset"
|
|
|
|
expression: !!model.communityId
|
|
|
|
expectedRoles: ["communityId"]
|
|
|
|
}
|
|
|
|
]
|
2023-09-11 07:20:36 -03:00
|
|
|
filters: [
|
|
|
|
ExpressionFilter {
|
|
|
|
expression: {
|
|
|
|
return d.uppercaseSearchText === "" || name.toUpperCase().startsWith(d.uppercaseSearchText)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
2024-02-20 08:43:31 +01:00
|
|
|
sorters: [
|
|
|
|
RoleSorter {
|
|
|
|
roleName: "isCommunityAsset"
|
|
|
|
sortOrder: Qt.DescendingOrder
|
|
|
|
},
|
|
|
|
RoleSorter {
|
|
|
|
roleName: "isCollection"
|
|
|
|
sortOrder: Qt.DescendingOrder
|
|
|
|
}
|
|
|
|
]
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
readonly property string searchPlaceholderText: {
|
|
|
|
if (isCurrentBrowsingTypeAsset) {
|
|
|
|
return qsTr("Search for token or enter token address")
|
|
|
|
} else if (isBrowsingCollection) {
|
|
|
|
return qsTr("Search %1").arg(d.currentBrowsingCollectionName ?? qsTr("collectibles in collection"))
|
|
|
|
} else {
|
|
|
|
return qsTr("Search collectibles")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// By design values:
|
|
|
|
readonly property int padding: 16
|
|
|
|
readonly property int headerTopMargin: 5
|
|
|
|
readonly property int tabBarTopMargin: 20
|
|
|
|
readonly property int tabBarHeight: 35
|
|
|
|
readonly property int bottomInset: 20
|
|
|
|
readonly property int assetContentIconSize: 21
|
|
|
|
readonly property int collectibleContentIconSize: 28
|
|
|
|
readonly property int assetContentTextSize: 28
|
|
|
|
readonly property int collectibleContentTextSize: 15
|
2024-02-20 08:43:31 +01:00
|
|
|
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
HoldingItemSelector {
|
|
|
|
id: holdingItemSelector
|
2023-10-03 23:10:02 +02:00
|
|
|
width: parent.width
|
|
|
|
height: parent.height
|
2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
defaultIconSource: Style.png("tokens/DEFAULT-TOKEN@3x")
|
|
|
|
placeholderText: d.isCurrentBrowsingTypeAsset ? qsTr("Select token") : qsTr("Select collectible")
|
2024-02-20 08:43:31 +01:00
|
|
|
property bool hasCommunityTokens: false
|
|
|
|
|
2023-09-11 07:20:36 -03:00
|
|
|
comboBoxDelegate: Item {
|
|
|
|
property var itemModel: model // read 'model' from the delegate's context
|
|
|
|
width: loader.width
|
|
|
|
height: loader.height
|
|
|
|
Loader {
|
|
|
|
id: loader
|
|
|
|
|
|
|
|
// inject model properties to the loaded item's context
|
|
|
|
// common
|
|
|
|
property var model: itemModel
|
|
|
|
property var chainId: model.chainId
|
|
|
|
property var name: model.name
|
|
|
|
// asset
|
|
|
|
property var symbol: model.symbol
|
|
|
|
property var totalBalance: model.totalBalance
|
2023-11-28 20:16:18 +01:00
|
|
|
property var marketDetails: model.marketDetails
|
2023-09-11 07:20:36 -03:00
|
|
|
property var decimals: model.decimals
|
|
|
|
property var balances: model.balances
|
|
|
|
// collectible
|
|
|
|
property var uid: model.uid
|
|
|
|
property var iconUrl: model.iconUrl
|
2023-10-30 23:18:58 +01:00
|
|
|
property var networkIconUrl: model.networkIconUrl
|
2023-09-11 07:20:36 -03:00
|
|
|
property var collectionUid: model.collectionUid
|
2024-02-20 08:43:31 +01:00
|
|
|
property var communityId: model.communityId
|
2023-09-11 07:20:36 -03:00
|
|
|
property var collectionName: model.collectionName
|
|
|
|
property var isCollection: model.isCollection
|
|
|
|
|
|
|
|
sourceComponent: d.isCurrentBrowsingTypeAsset ? assetComboBoxDelegate : collectibleComboBoxDelegate
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
comboBoxPopupHeader: headerComponent
|
|
|
|
itemTextFn: d.isCurrentBrowsingTypeAsset ? d.assetTextFn : d.collectibleTextFn
|
|
|
|
itemIconSourceFn: d.isCurrentBrowsingTypeAsset ? d.assetIconSourceFn : d.collectibleIconSourceFn
|
2024-01-30 14:15:58 +01:00
|
|
|
comboBoxModel: d.isCurrentBrowsingTypeAsset ? root.assetsModel : d.collectibleComboBoxModel
|
2024-02-20 08:43:31 +01:00
|
|
|
onComboBoxModelChanged: updateHasCommunityTokens()
|
|
|
|
|
|
|
|
function updateHasCommunityTokens() {
|
|
|
|
hasCommunityTokens = Helpers.modelHasCommunityTokens(comboBoxModel, d.isCurrentBrowsingTypeAsset)
|
|
|
|
}
|
2023-09-11 07:20:36 -03:00
|
|
|
|
|
|
|
contentIconSize: d.isAsset(d.currentHoldingType) ? d.assetContentIconSize : d.collectibleContentIconSize
|
|
|
|
contentTextSize: d.isAsset(d.currentHoldingType) ? d.assetContentTextSize : d.collectibleContentTextSize
|
2024-01-30 14:15:58 +01:00
|
|
|
comboBoxListViewSection.property: "isCommunityAsset"
|
2024-02-20 08:43:31 +01:00
|
|
|
comboBoxListViewSection.delegate: AssetsSectionDelegate {
|
|
|
|
height: !!text ? 52 : 0 // if we bind to some property instead of hardcoded value it wont work nice when switching tabs or going inside collection and back
|
|
|
|
width: ListView.view.width
|
|
|
|
required property bool section
|
|
|
|
text: Helpers.assetsSectionTitle(section, holdingItemSelector.hasCommunityTokens, d.isBrowsingCollection, d.isCurrentBrowsingTypeAsset)
|
|
|
|
onOpenInfoPopup: Global.openPopup(communityInfoPopupCmp)
|
|
|
|
}
|
2024-01-30 14:15:58 +01:00
|
|
|
comboBoxControl.popup.onClosed: comboBoxControl.popup.contentItem.headerItem.clear()
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: communityInfoPopupCmp
|
|
|
|
CommunityAssetsInfoPopup {}
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: headerComponent
|
|
|
|
ColumnLayout {
|
2024-01-30 14:15:58 +01:00
|
|
|
function clear() {
|
|
|
|
searchInput.input.edit.clear()
|
|
|
|
}
|
|
|
|
|
2023-09-11 07:20:36 -03:00
|
|
|
width: holdingItemSelector.comboBoxControl.popup.width
|
|
|
|
Layout.topMargin: d.headerTopMargin
|
|
|
|
spacing: -1 // Used to overlap rectangles from row components
|
|
|
|
|
|
|
|
StatusTabBar {
|
|
|
|
id: tabBar
|
|
|
|
|
|
|
|
visible: !root.onlyAssets
|
|
|
|
Layout.preferredHeight: d.tabBarHeight
|
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.leftMargin: d.padding
|
|
|
|
Layout.rightMargin: d.padding
|
|
|
|
Layout.topMargin: d.tabBarTopMargin
|
|
|
|
Layout.bottomMargin: 6
|
|
|
|
currentIndex: d.holdingTypes.indexOf(d.browsingHoldingType)
|
|
|
|
|
|
|
|
onCurrentIndexChanged: {
|
|
|
|
if (currentIndex >= 0) {
|
|
|
|
d.browsingHoldingType = d.holdingTypes[currentIndex]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Repeater {
|
|
|
|
id: tabLabelsRepeater
|
|
|
|
model: d.tabsModel
|
|
|
|
|
|
|
|
StatusTabButton {
|
|
|
|
text: modelData
|
|
|
|
width: implicitWidth
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-12 16:26:38 +02:00
|
|
|
CollectibleBackButtonWithInfo {
|
2023-09-11 07:20:36 -03:00
|
|
|
Layout.fillWidth: true
|
|
|
|
visible: d.isBrowsingCollection
|
2023-09-12 16:26:38 +02:00
|
|
|
count: collectiblesModel.count
|
|
|
|
name: d.currentBrowsingCollectionName
|
|
|
|
onBackClicked: {
|
|
|
|
if (!d.isCurrentBrowsingTypeAsset) {
|
2024-02-20 08:43:31 +01:00
|
|
|
searchInput.reset()
|
2023-09-12 16:26:38 +02:00
|
|
|
root.collectiblesModel.currentCollectionUid = ""
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Rectangle {
|
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.preferredHeight: searchInput.input.implicitHeight
|
|
|
|
|
|
|
|
color: "transparent"
|
|
|
|
border.color: Theme.palette.baseColor2
|
|
|
|
border.width: 1
|
|
|
|
|
|
|
|
StatusInput {
|
|
|
|
id: searchInput
|
|
|
|
anchors.fill: parent
|
|
|
|
|
|
|
|
input.showBackground: false
|
|
|
|
placeholderText: d.searchPlaceholderText
|
|
|
|
onTextChanged: Qt.callLater(d.updateSearchText, text)
|
|
|
|
input.clearable: true
|
|
|
|
input.implicitHeight: 56
|
|
|
|
input.rightComponent: StatusFlatRoundButton {
|
|
|
|
icon.name: "search"
|
|
|
|
type: StatusFlatRoundButton.Type.Secondary
|
|
|
|
enabled: false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: assetComboBoxDelegate
|
|
|
|
TokenBalancePerChainDelegate {
|
|
|
|
objectName: "AssetSelector_ItemDelegate_" + symbol
|
|
|
|
width: holdingItemSelector.comboBoxControl.popup.width
|
2023-11-28 20:16:18 +01:00
|
|
|
selectedSenderAccount: root.selectedSenderAccount
|
2023-10-30 23:18:58 +01:00
|
|
|
balancesModel: LeftJoinModel {
|
|
|
|
leftModel: balances
|
|
|
|
rightModel: root.networksModel
|
|
|
|
joinRole: "chainId"
|
|
|
|
}
|
2023-09-11 07:20:36 -03:00
|
|
|
onTokenSelected: {
|
|
|
|
holdingItemSelector.selectedItem = selectedToken
|
2023-11-07 23:45:47 +01:00
|
|
|
d.currentHoldingType = Constants.TokenType.ERC20
|
|
|
|
root.itemSelected(selectedToken.symbol, Constants.TokenType.ERC20)
|
2023-09-11 07:20:36 -03:00
|
|
|
holdingItemSelector.comboBoxControl.popup.close()
|
|
|
|
}
|
2024-01-30 14:15:58 +01:00
|
|
|
formatCurrentCurrencyAmount: function(balance){
|
|
|
|
return root.formatCurrentCurrencyAmount(balance)
|
2023-11-28 20:16:18 +01:00
|
|
|
}
|
2024-01-30 14:15:58 +01:00
|
|
|
formatCurrencyAmountFromBigInt: function(balance, symbol, decimals){
|
|
|
|
return root.formatCurrencyAmountFromBigInt(balance, symbol, decimals)
|
2023-11-28 20:16:18 +01:00
|
|
|
}
|
2023-09-11 07:20:36 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: collectibleComboBoxDelegate
|
|
|
|
CollectibleNestedDelegate {
|
|
|
|
objectName: "CollectibleSelector_ItemDelegate_" + collectionUid
|
|
|
|
width: holdingItemSelector.comboBoxControl.popup.width
|
2024-02-20 08:43:31 +01:00
|
|
|
numItems: isCollection ? (!!communityId ?
|
|
|
|
root.collectiblesModel.getNumberOfCollectiblesInCommunity(communityId) :
|
|
|
|
root.collectiblesModel.getNumberOfCollectiblesInCollection(collectionUid)) : 0
|
2023-09-11 07:20:36 -03:00
|
|
|
onItemSelected: {
|
|
|
|
if (isCollection) {
|
|
|
|
d.currentBrowsingCollectionName = collectionName
|
2024-02-20 08:43:31 +01:00
|
|
|
if (!!communityId)
|
|
|
|
root.collectiblesModel.currentCollectionUid = communityId
|
|
|
|
else
|
|
|
|
root.collectiblesModel.currentCollectionUid = collectionUid
|
2023-09-11 07:20:36 -03:00
|
|
|
} else {
|
|
|
|
holdingItemSelector.selectedItem = selectedItem
|
2023-11-07 23:45:47 +01:00
|
|
|
d.currentHoldingType = Constants.TokenType.ERC721
|
|
|
|
root.itemSelected(selectedItem.uid, Constants.TokenType.ERC721)
|
2023-09-11 07:20:36 -03:00
|
|
|
holdingItemSelector.comboBoxControl.popup.close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|