2023-07-04 17:11:41 +02:00
|
|
|
import QtQuick 2.15
|
|
|
|
import QtQuick.Controls 2.15
|
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
import StatusQ 0.1
|
2023-07-04 17:11:41 +02:00
|
|
|
import StatusQ.Core 0.1
|
|
|
|
import StatusQ.Components 0.1
|
|
|
|
import StatusQ.Controls 0.1
|
|
|
|
import StatusQ.Core.Theme 0.1
|
|
|
|
import StatusQ.Core.Utils 0.1
|
|
|
|
|
|
|
|
import SortFilterProxyModel 0.2
|
|
|
|
|
|
|
|
import utils 1.0
|
|
|
|
|
|
|
|
StatusListView {
|
|
|
|
id: root
|
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
property var walletAssetsModel
|
2023-07-04 17:11:41 +02:00
|
|
|
property bool hasPermissions
|
|
|
|
property var uniquePermissionTokenKeys
|
|
|
|
|
|
|
|
// read/write properties
|
2023-08-02 19:39:42 +02:00
|
|
|
property string selectedAirdropAddress
|
|
|
|
property var selectedSharedAddresses: []
|
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
property var getCurrencyAmount: function (balance, symbol){}
|
|
|
|
|
2023-08-02 19:39:42 +02:00
|
|
|
signal addressesChanged()
|
2023-07-04 17:11:41 +02:00
|
|
|
|
|
|
|
leftMargin: d.absLeftMargin
|
|
|
|
topMargin: Style.current.padding
|
|
|
|
rightMargin: Style.current.padding
|
|
|
|
bottomMargin: Style.current.padding
|
|
|
|
|
|
|
|
QtObject {
|
|
|
|
id: d
|
|
|
|
|
|
|
|
// UI
|
|
|
|
readonly property int absLeftMargin: 12
|
|
|
|
|
|
|
|
readonly property ButtonGroup airdropGroup: ButtonGroup {
|
|
|
|
exclusive: true
|
|
|
|
}
|
|
|
|
|
|
|
|
readonly property ButtonGroup addressesGroup: ButtonGroup {
|
|
|
|
exclusive: false
|
|
|
|
}
|
|
|
|
|
|
|
|
function selectFirstAvailableAirdropAddress() {
|
|
|
|
root.selectedAirdropAddress = ModelUtils.modelToFlatArray(root.model, "address").find(address => selectedSharedAddresses.includes(address))
|
|
|
|
}
|
2024-01-24 16:35:53 +00:00
|
|
|
|
|
|
|
function getTotalBalance(balances, decimals, symbol) {
|
|
|
|
let totalBalance = 0
|
|
|
|
for(let i=0; i<balances.count; i++) {
|
|
|
|
let balancePerAddressPerChain = ModelUtils.get(balances, i)
|
|
|
|
totalBalance += AmountsArithmetic.toNumber(balancePerAddressPerChain.balance, decimals)
|
|
|
|
}
|
|
|
|
return totalBalance
|
|
|
|
}
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
spacing: Style.current.halfPadding
|
|
|
|
delegate: StatusListItem {
|
2023-08-24 16:36:38 -04:00
|
|
|
readonly property string address: model.address.toLowerCase()
|
|
|
|
|
|
|
|
id: listItem
|
2023-07-04 17:11:41 +02:00
|
|
|
width: ListView.view.width - ListView.view.leftMargin - ListView.view.rightMargin
|
|
|
|
statusListItemTitle.font.weight: Font.Medium
|
|
|
|
title: model.name
|
|
|
|
tertiaryTitle: !walletAccountAssetsModel.count && root.hasPermissions ? qsTr("No relevant tokens") : ""
|
2024-01-24 16:35:53 +00:00
|
|
|
property string accountAddress: model.address
|
|
|
|
|
|
|
|
SubmodelProxyModel {
|
|
|
|
id: filteredBalances
|
|
|
|
sourceModel: root.walletAssetsModel
|
|
|
|
submodelRoleName: "balances"
|
|
|
|
delegateModel: SortFilterProxyModel {
|
|
|
|
sourceModel: submodel
|
|
|
|
filters: FastExpressionFilter {
|
|
|
|
expression: listItem.accountAddress === model.account
|
|
|
|
expectedRoles: ["account"]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-07-04 17:11:41 +02:00
|
|
|
tagsModel: SortFilterProxyModel {
|
|
|
|
id: walletAccountAssetsModel
|
2024-01-24 16:35:53 +00:00
|
|
|
sourceModel: filteredBalances
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
function filterPredicate(symbol) {
|
|
|
|
return root.uniquePermissionTokenKeys.includes(symbol.toUpperCase())
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
proxyRoles: FastExpressionRole {
|
|
|
|
name: "enabledNetworkBalance"
|
|
|
|
expression: d.getTotalBalance(model.balances, model.decimals, model.symbol)
|
|
|
|
expectedRoles: ["balances", "decimals", "symbol"]
|
|
|
|
}
|
|
|
|
filters: FastExpressionFilter {
|
|
|
|
expression: walletAccountAssetsModel.filterPredicate(model.symbol)
|
|
|
|
expectedRoles: ["symbol"]
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
2024-01-24 16:35:53 +00:00
|
|
|
sorters: FastExpressionSorter {
|
2023-07-04 17:11:41 +02:00
|
|
|
expression: {
|
2024-01-29 21:49:59 +02:00
|
|
|
if (modelLeft.enabledNetworkBalance > modelRight.enabledNetworkBalance)
|
|
|
|
return -1 // descending, biggest first
|
|
|
|
else if (modelLeft.enabledNetworkBalance < modelRight.enabledNetworkBalance)
|
|
|
|
return 1
|
|
|
|
else
|
|
|
|
return 0
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
2024-01-24 16:35:53 +00:00
|
|
|
expectedRoles: ["enabledNetworkBalance"]
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
statusListItemInlineTagsSlot.spacing: Style.current.padding
|
|
|
|
tagsDelegate: Row {
|
|
|
|
spacing: 4
|
|
|
|
StatusRoundedImage {
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
width: 16
|
|
|
|
height: 16
|
|
|
|
image.source: Constants.tokenIcon(model.symbol.toUpperCase())
|
|
|
|
}
|
|
|
|
StatusBaseText {
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
font.pixelSize: Theme.tertiaryTextFontSize
|
2024-01-24 16:35:53 +00:00
|
|
|
text: LocaleUtils.currencyAmountToLocaleString(root.getCurrencyAmount(model.enabledNetworkBalance, model.symbol))
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-09 14:31:58 +02:00
|
|
|
asset.color: !!model.color ? model.color : ""
|
2023-07-04 17:11:41 +02:00
|
|
|
asset.emoji: model.emoji
|
|
|
|
asset.name: !model.emoji ? "filled-account": ""
|
|
|
|
asset.letterSize: 14
|
|
|
|
asset.isLetterIdenticon: !!model.emoji
|
|
|
|
asset.isImage: asset.isLetterIdenticon
|
|
|
|
|
|
|
|
components: [
|
|
|
|
StatusFlatButton {
|
|
|
|
ButtonGroup.group: d.airdropGroup
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
icon.name: "airdrop"
|
|
|
|
icon.color: hovered ? Theme.palette.primaryColor3 :
|
|
|
|
checked ? Theme.palette.primaryColor1 : disabledTextColor
|
|
|
|
checkable: true
|
2023-09-18 12:23:30 +02:00
|
|
|
checked: listItem.address === root.selectedAirdropAddress.toLowerCase()
|
2023-07-04 17:11:41 +02:00
|
|
|
enabled: shareAddressCheckbox.checked && root.selectedSharedAddresses.length > 1 // last cannot be unchecked
|
|
|
|
visible: shareAddressCheckbox.checked
|
|
|
|
opacity: enabled ? 1.0 : 0.3
|
2023-08-24 16:36:38 -04:00
|
|
|
onCheckedChanged: if (checked) root.selectedAirdropAddress = listItem.address
|
2023-08-02 19:39:42 +02:00
|
|
|
onToggled: root.addressesChanged()
|
2023-07-04 17:11:41 +02:00
|
|
|
|
|
|
|
StatusToolTip {
|
|
|
|
text: qsTr("Use this address for any Community airdrops")
|
|
|
|
visible: parent.hovered
|
|
|
|
delay: 500
|
|
|
|
}
|
|
|
|
},
|
|
|
|
StatusCheckBox {
|
|
|
|
id: shareAddressCheckbox
|
|
|
|
ButtonGroup.group: d.addressesGroup
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
checkable: true
|
2023-09-18 12:23:30 +02:00
|
|
|
checked: root.selectedSharedAddresses.some((address) => address.toLowerCase() === listItem.address )
|
2023-07-04 17:11:41 +02:00
|
|
|
enabled: !(root.selectedSharedAddresses.length === 1 && checked) // last cannot be unchecked
|
|
|
|
onToggled: {
|
|
|
|
// handle selected addresses
|
2023-09-18 12:23:30 +02:00
|
|
|
const index = root.selectedSharedAddresses.findIndex((address) => address.toLowerCase() === listItem.address)
|
2023-07-04 17:11:41 +02:00
|
|
|
const selectedSharedAddressesCopy = Object.assign([], root.selectedSharedAddresses) // deep copy
|
|
|
|
if (index === -1) {
|
2023-08-24 16:36:38 -04:00
|
|
|
selectedSharedAddressesCopy.push(listItem.address)
|
2023-07-04 17:11:41 +02:00
|
|
|
} else {
|
|
|
|
selectedSharedAddressesCopy.splice(index, 1)
|
|
|
|
}
|
|
|
|
root.selectedSharedAddresses = selectedSharedAddressesCopy
|
|
|
|
|
|
|
|
// switch to next available airdrop address when unchecking
|
2023-09-18 12:23:30 +02:00
|
|
|
if (!checked && listItem.address === root.selectedAirdropAddress.toLowerCase()) {
|
2023-07-04 17:11:41 +02:00
|
|
|
d.selectFirstAvailableAirdropAddress()
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
|
|
|
|
root.addressesChanged()
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|