2024-06-19 00:51:49 +02:00
|
|
|
import QtQuick 2.15
|
|
|
|
import QtQuick.Controls 2.15
|
|
|
|
import QtQuick.Layouts 1.15
|
|
|
|
import QtGraphicalEffects 1.0
|
|
|
|
|
|
|
|
import StatusQ 0.1
|
|
|
|
import StatusQ.Components 0.1
|
|
|
|
import StatusQ.Components.private 0.1
|
|
|
|
import StatusQ.Core 0.1
|
|
|
|
import StatusQ.Core.Utils 0.1
|
|
|
|
import StatusQ.Core.Theme 0.1
|
|
|
|
import StatusQ.Popups.Dialog 0.1
|
|
|
|
|
|
|
|
import AppLayouts.Wallet.views 1.0
|
|
|
|
|
|
|
|
import utils 1.0
|
|
|
|
import shared.controls 1.0
|
|
|
|
|
|
|
|
ComboBox {
|
|
|
|
id: root
|
|
|
|
|
|
|
|
// expected model structure:
|
|
|
|
// tokensKey, name, symbol, decimals, currencyBalanceAsString (computed), marketDetails, balances -> [ chainId, address, balance, iconUrl ]
|
|
|
|
|
|
|
|
// input API
|
|
|
|
property string nonInteractiveDelegateKey
|
|
|
|
|
|
|
|
// output API
|
|
|
|
readonly property string currentTokensKey: d.currentTokensKey
|
|
|
|
readonly property alias searchString: searchBox.text
|
|
|
|
|
|
|
|
/**
|
|
|
|
Emitted when a token gets selected
|
|
|
|
*/
|
|
|
|
signal tokenSelected(string tokensKey)
|
|
|
|
|
|
|
|
// manipulation
|
|
|
|
function selectToken(tokensKey) {
|
|
|
|
const idx = ModelUtils.indexOf(model, "tokensKey", tokensKey)
|
|
|
|
if (idx === -1) {
|
|
|
|
console.warn("TokenSelector::selectToken: unknown tokensKey:", tokensKey)
|
|
|
|
tokensKey = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
currentIndex = idx
|
|
|
|
d.currentTokensKey = tokensKey
|
|
|
|
root.tokenSelected(tokensKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
function reset() {
|
|
|
|
selectToken("")
|
|
|
|
}
|
|
|
|
|
|
|
|
QtObject {
|
|
|
|
id: d
|
|
|
|
|
|
|
|
// NB: internal tracking; the ComboBox currentValue is not persistent,
|
|
|
|
// i.e. relying on currentValue is not safe
|
|
|
|
property string currentTokensKey
|
|
|
|
|
|
|
|
readonly property bool isTokenSelected: !!currentTokensKey
|
|
|
|
|
|
|
|
// NB: handle cases when our currently selected token disappears from the model -> reset
|
|
|
|
readonly property Connections _conn: Connections {
|
|
|
|
target: model ?? null
|
|
|
|
function onModelReset() {
|
|
|
|
if (d.isTokenSelected && !root.popup.opened)
|
|
|
|
root.selectToken(d.currentTokensKey)
|
|
|
|
}
|
|
|
|
function onRowsRemoved() {
|
|
|
|
if (d.isTokenSelected && !root.popup.opened)
|
|
|
|
root.selectToken(d.currentTokensKey)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
font.family: Theme.palette.baseFont.name
|
|
|
|
font.pixelSize: Style.current.additionalTextSize
|
|
|
|
spacing: Style.current.halfPadding
|
|
|
|
verticalPadding: 10
|
|
|
|
leftPadding: 12
|
|
|
|
rightPadding: leftPadding + indicator.width + spacing
|
|
|
|
opacity: enabled ? 1 : 0.3
|
|
|
|
|
|
|
|
popup.width: 380
|
2024-07-02 13:59:21 +02:00
|
|
|
popup.height: 380
|
2024-06-19 00:51:49 +02:00
|
|
|
popup.x: root.width - popup.width
|
2024-07-02 13:59:21 +02:00
|
|
|
popup.margins: Style.current.xlPadding*2
|
2024-06-19 00:51:49 +02:00
|
|
|
popup.background: Rectangle {
|
|
|
|
color: Theme.palette.statusSelect.menuItemBackgroundColor
|
|
|
|
radius: Style.current.radius
|
|
|
|
layer.enabled: true
|
|
|
|
layer.effect: DropShadow {
|
|
|
|
horizontalOffset: 0
|
|
|
|
verticalOffset: 4
|
|
|
|
radius: 12
|
|
|
|
samples: 25
|
|
|
|
spread: 0.2
|
|
|
|
color: Theme.palette.dropShadow
|
|
|
|
}
|
|
|
|
}
|
|
|
|
popup.contentItem: ColumnLayout {
|
2024-07-02 13:59:21 +02:00
|
|
|
Connections {
|
|
|
|
target: root.popup
|
|
|
|
function onOpened() {
|
|
|
|
listview.positionViewAtBeginning()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-19 00:51:49 +02:00
|
|
|
spacing: 0
|
|
|
|
SearchBox {
|
|
|
|
Layout.fillWidth: true
|
|
|
|
id: searchBox
|
|
|
|
objectName: "searchBox"
|
|
|
|
|
|
|
|
input.leftPadding: root.leftPadding
|
|
|
|
input.rightPadding: root.leftPadding
|
|
|
|
minimumHeight: 56
|
|
|
|
maximumHeight: 56
|
|
|
|
placeholderText: qsTr("Search asset name or symbol")
|
|
|
|
input.showBackground: false
|
|
|
|
focus: visible
|
|
|
|
onVisibleChanged: if (!visible) input.edit.clear()
|
|
|
|
}
|
|
|
|
StatusDialogDivider {
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
StatusListView {
|
|
|
|
id: listview
|
|
|
|
objectName: "tokenSelectorListview"
|
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.preferredHeight: contentHeight
|
|
|
|
Layout.fillHeight: true
|
|
|
|
|
|
|
|
model: root.popup.visible ? root.delegateModel : null
|
|
|
|
currentIndex: root.highlightedIndex
|
|
|
|
|
2024-07-04 00:52:05 +02:00
|
|
|
section.property: "sectionName"
|
2024-06-19 00:51:49 +02:00
|
|
|
section.delegate: StatusBaseText {
|
|
|
|
required property string section
|
2024-07-04 00:52:05 +02:00
|
|
|
visible: searchBox.text === ""
|
2024-06-19 00:51:49 +02:00
|
|
|
width: parent.width
|
|
|
|
elide: Text.ElideRight
|
2024-07-04 00:52:05 +02:00
|
|
|
text: visible ? section : ""
|
2024-06-19 00:51:49 +02:00
|
|
|
color: Theme.palette.baseColor1
|
|
|
|
padding: Style.current.padding
|
|
|
|
}
|
|
|
|
}
|
2024-07-02 20:40:27 +02:00
|
|
|
StatusBaseText {
|
|
|
|
Layout.fillWidth: true
|
2024-07-04 00:52:05 +02:00
|
|
|
Layout.fillHeight: true
|
2024-07-02 20:40:27 +02:00
|
|
|
Layout.alignment: Qt.AlignCenter
|
|
|
|
horizontalAlignment: Text.AlignHCenter
|
|
|
|
verticalAlignment: Text.AlignVCenter
|
|
|
|
visible: listview.count === 0
|
|
|
|
color: Theme.palette.baseColor1
|
|
|
|
text: qsTr("No assets found")
|
|
|
|
}
|
2024-06-19 00:51:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
background: StatusComboboxBackground {
|
|
|
|
border.width: 0
|
|
|
|
color: {
|
|
|
|
if (d.isTokenSelected)
|
|
|
|
return "transparent"
|
|
|
|
return root.hovered ? Theme.palette.primaryColor2 : Theme.palette.primaryColor3
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
contentItem: Loader {
|
|
|
|
height: 40 // by design
|
|
|
|
sourceComponent: d.isTokenSelected ? iconTextContentItem : textContentItem
|
|
|
|
}
|
|
|
|
|
|
|
|
indicator: StatusComboboxIndicator {
|
|
|
|
anchors.right: parent.right
|
|
|
|
anchors.rightMargin: root.leftPadding
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
color: Theme.palette.primaryColor1
|
|
|
|
}
|
|
|
|
|
|
|
|
delegate: TokenSelectorAssetDelegate {
|
|
|
|
required property var model
|
|
|
|
required property int index
|
|
|
|
|
|
|
|
highlighted: tokensKey === d.currentTokensKey
|
|
|
|
interactive: tokensKey !== root.nonInteractiveDelegateKey
|
|
|
|
|
|
|
|
tokensKey: model.tokensKey
|
|
|
|
name: model.name
|
|
|
|
symbol: model.symbol
|
2024-06-25 15:37:42 +02:00
|
|
|
currencyBalanceAsString: model.currencyBalanceAsString ?? ""
|
|
|
|
iconSource: model.iconSource
|
2024-06-19 00:51:49 +02:00
|
|
|
balancesModel: model.balances
|
|
|
|
|
|
|
|
onAssetSelected: (tokensKey) => root.selectToken(tokensKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: textContentItem
|
|
|
|
StatusBaseText {
|
2024-06-18 19:24:07 +02:00
|
|
|
objectName: "tokenSelectorContentItemText"
|
2024-06-19 00:51:49 +02:00
|
|
|
font.pixelSize: root.font.pixelSize
|
|
|
|
font.weight: Font.Medium
|
|
|
|
color: Theme.palette.primaryColor1
|
|
|
|
text: qsTr("Select asset")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: iconTextContentItem
|
|
|
|
RowLayout {
|
|
|
|
spacing: root.spacing
|
|
|
|
StatusRoundedImage {
|
2024-06-18 19:24:07 +02:00
|
|
|
objectName: "tokenSelectorIcon"
|
2024-06-19 00:51:49 +02:00
|
|
|
Layout.preferredWidth: 20
|
|
|
|
Layout.preferredHeight: 20
|
2024-06-25 15:37:42 +02:00
|
|
|
image.source: ModelUtils.getByKey(model, "tokensKey", d.currentTokensKey, "iconSource")
|
2024-06-19 00:51:49 +02:00
|
|
|
}
|
|
|
|
StatusBaseText {
|
2024-06-18 19:24:07 +02:00
|
|
|
objectName: "tokenSelectorContentItemText"
|
2024-06-19 00:51:49 +02:00
|
|
|
font.pixelSize: 28
|
|
|
|
color: root.hovered ? Theme.palette.blue : Theme.palette.darkBlue
|
2024-06-25 15:37:42 +02:00
|
|
|
text: ModelUtils.getByKey(model, "tokensKey", d.currentTokensKey, "symbol")
|
2024-06-19 00:51:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|