mirror of
https://github.com/status-im/status-desktop.git
synced 2025-01-12 15:24:39 +00:00
d136d9bb67
- It adds fee box into the Burn popup component. - It updates storybook accordingly. Closes #11609
270 lines
8.7 KiB
QML
270 lines
8.7 KiB
QML
import QtQuick 2.15
|
||
import QtQuick.Controls 2.14
|
||
import QtQuick.Layouts 1.14
|
||
import QtQml.Models 2.14
|
||
import QtGraphicalEffects 1.0
|
||
import QtQml 2.15
|
||
|
||
import StatusQ.Core 0.1
|
||
import StatusQ.Controls 0.1
|
||
import StatusQ.Popups.Dialog 0.1
|
||
import StatusQ.Core.Theme 0.1
|
||
import StatusQ.Core.Utils 0.1 as SQUtils
|
||
import StatusQ.Controls.Validators 0.1
|
||
import StatusQ.Components 0.1
|
||
|
||
import AppLayouts.Communities.panels 1.0
|
||
import AppLayouts.Communities.helpers 1.0
|
||
|
||
import utils 1.0
|
||
import SortFilterProxyModel 0.2
|
||
|
||
StatusDialog {
|
||
id: root
|
||
|
||
property string communityName
|
||
property bool isAsset // If asset isAsset = true; if collectible --> isAsset = false
|
||
property string tokenName
|
||
property int remainingTokens
|
||
property url tokenSource
|
||
property string chainName
|
||
|
||
// Fees related properties:
|
||
property string feeText
|
||
property string feeErrorText: ""
|
||
property bool isFeeLoading
|
||
readonly property string feeLabel: qsTr("Burn %1 token on %2").arg(root.tokenName).arg(root.chainName)
|
||
|
||
// Account expected roles: address, name, color, emoji, walletType
|
||
property var accounts
|
||
|
||
signal burnClicked(int burnAmount, string accountAddress)
|
||
signal cancelClicked
|
||
signal burnFeesRequested(int burnAmount, string accountAddress)
|
||
|
||
QtObject {
|
||
id: d
|
||
|
||
property string accountAddress
|
||
property alias amountToBurn: amountToBurnInput.text
|
||
readonly property bool isFeeError: root.feeErrorText !== ""
|
||
readonly property bool isFormValid: specificAmountButton.checked && amountToBurnInput.valid || allTokensButton.checked
|
||
|
||
function initialize() {
|
||
specificAmountButton.checked = true
|
||
amountToBurnInput.forceActiveFocus()
|
||
}
|
||
|
||
function getVerticalPadding() {
|
||
return root.topPadding + root.bottomPadding
|
||
}
|
||
|
||
function getHorizontalPadding() {
|
||
return root.leftPadding + root.rightPadding
|
||
}
|
||
}
|
||
|
||
implicitWidth: 600 // by design
|
||
implicitHeight: content.implicitHeight + footer.height + header.height + d.getVerticalPadding()
|
||
|
||
contentItem: ColumnLayout {
|
||
id: content
|
||
|
||
spacing: Style.current.padding
|
||
|
||
StatusBaseText {
|
||
Layout.fillWidth: true
|
||
|
||
text: qsTr("How many of %1’s remaining %n %2 tokens would you like to burn?", "", root.remainingTokens).arg(root.communityName).arg(root.tokenName)
|
||
wrapMode: Text.WordWrap
|
||
lineHeight: 1.2
|
||
font.pixelSize: Style.current.primaryTextFontSize
|
||
}
|
||
|
||
RowLayout {
|
||
Layout.bottomMargin: 12
|
||
Layout.leftMargin: -Style.current.halfPadding
|
||
|
||
spacing: 26
|
||
|
||
ColumnLayout {
|
||
StatusRadioButton {
|
||
id: specificAmountButton
|
||
|
||
text: qsTr("Specific amount")
|
||
font.pixelSize: Style.current.primaryTextFontSize
|
||
ButtonGroup.group: radioGroup
|
||
|
||
onToggled: if(checked) amountToBurnInput.forceActiveFocus()
|
||
}
|
||
|
||
StatusInput {
|
||
id: amountToBurnInput
|
||
|
||
Layout.preferredWidth: 192
|
||
Layout.leftMargin: 30
|
||
enabled: specificAmountButton.checked
|
||
validationMode: StatusInput.ValidationMode.OnlyWhenDirty
|
||
validators: [
|
||
StatusValidator {
|
||
validate: (value) => { return (parseInt(value) > 0 && parseInt(value) <= root.remainingTokens) }
|
||
errorMessage: qsTr("Exceeds available remaining")
|
||
},
|
||
StatusValidator {
|
||
validate: (value) => { return parseInt(value) !== 0 }
|
||
errorMessage: qsTr("Amount must be greater than 0")
|
||
},
|
||
StatusRegularExpressionValidator {
|
||
regularExpression: Constants.regularExpressions.numerical
|
||
errorMessage: qsTr("Invalid characters (0-9 only)")
|
||
}
|
||
]
|
||
}
|
||
}
|
||
|
||
StatusRadioButton {
|
||
id: allTokensButton
|
||
|
||
Layout.alignment: Qt.AlignTop
|
||
|
||
text: qsTr("All available remaining (%1)").arg(root.remainingTokens)
|
||
font.pixelSize: Style.current.primaryTextFontSize
|
||
ButtonGroup.group: radioGroup
|
||
}
|
||
|
||
ButtonGroup { id: radioGroup }
|
||
}
|
||
|
||
StatusDialogDivider {
|
||
Layout.fillWidth: true
|
||
}
|
||
|
||
FeesBox {
|
||
id: feesBox
|
||
|
||
readonly property bool triggerFeeReevaluation: {
|
||
specificAmountButton.checked
|
||
amountToBurnInput.text
|
||
allTokensButton.checked
|
||
feesBox.accountsSelector.currentIndex
|
||
|
||
requestFeeDelayTimer.restart()
|
||
return true
|
||
}
|
||
|
||
Layout.fillWidth: true
|
||
|
||
placeholderText: qsTr("Choose number of tokens to burn to see gas fees")
|
||
accountErrorText: root.feeErrorText
|
||
implicitWidth: 0
|
||
model: d.isFormValid ? singleFeeModel : undefined
|
||
accountsSelector.model: SortFilterProxyModel {
|
||
sourceModel: root.accounts
|
||
proxyRoles: [
|
||
ExpressionRole {
|
||
name: "color"
|
||
|
||
function getColor(colorId) {
|
||
return Utils.getColorForId(colorId)
|
||
}
|
||
|
||
// Direct call for singleton function is not handled properly by
|
||
// SortFilterProxyModel that's why helper function is used instead.
|
||
expression: { return getColor(model.colorId) }
|
||
}
|
||
]
|
||
filters: ValueFilter {
|
||
roleName: "walletType"
|
||
value: Constants.watchWalletType
|
||
inverted: true
|
||
}
|
||
}
|
||
|
||
accountsSelector.onCurrentIndexChanged: {
|
||
if (accountsSelector.currentIndex < 0)
|
||
return
|
||
|
||
const item = SQUtils.ModelUtils.get(accountsSelector.model, accountsSelector.currentIndex)
|
||
d.accountAddress = item.address
|
||
}
|
||
|
||
Timer {
|
||
id: requestFeeDelayTimer
|
||
|
||
interval: 500
|
||
onTriggered: {
|
||
if(specificAmountButton.checked)
|
||
root.burnFeesRequested(parseInt(amountToBurnInput.text), d.accountAddress)
|
||
else
|
||
root.burnFeesRequested(root.remainingTokens, d.accountAddress)
|
||
}
|
||
}
|
||
|
||
QtObject {
|
||
id: singleFeeModel
|
||
|
||
readonly property string title: root.feeLabel
|
||
readonly property string feeText: root.isFeeLoading ?
|
||
"" : root.feeText
|
||
readonly property bool error: d.isFeeError
|
||
}
|
||
}
|
||
}
|
||
|
||
header: StatusDialogHeader {
|
||
headline.title: qsTr("Burn %1 tokens").arg(root.tokenName)
|
||
headline.subtitle: qsTr("%n %1 remaining in smart contract", "", root.remainingTokens).arg(root.tokenName)
|
||
leftComponent: Rectangle {
|
||
height: 40
|
||
width: height
|
||
radius: root.isAsset ? height/2 : 8
|
||
color:Theme.palette.baseColor2
|
||
|
||
Image {
|
||
id: image
|
||
|
||
source: root.tokenSource
|
||
anchors.fill: parent
|
||
fillMode: Image.PreserveAspectFit
|
||
visible: false
|
||
}
|
||
|
||
OpacityMask {
|
||
anchors.fill: image
|
||
source: image
|
||
maskSource: parent
|
||
}
|
||
}
|
||
actions.closeButton.onClicked: root.close()
|
||
}
|
||
|
||
footer: StatusDialogFooter {
|
||
spacing: Style.current.padding
|
||
rightButtons: ObjectModel {
|
||
StatusButton {
|
||
text: qsTr("Cancel")
|
||
normalColor: "transparent"
|
||
|
||
onClicked: {
|
||
root.cancelClicked()
|
||
close()
|
||
}
|
||
}
|
||
|
||
StatusButton {
|
||
enabled: d.isFormValid && !d.isFeeError && !root.isFeeLoading
|
||
text: qsTr("Burn tokens")
|
||
type: StatusBaseButton.Type.Danger
|
||
onClicked: {
|
||
if(specificAmountButton.checked)
|
||
root.burnClicked(parseInt(amountToBurnInput.text), d.accountAddress)
|
||
else
|
||
root.burnClicked(root.remainingTokens, d.accountAddress)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
onOpened: d.initialize()
|
||
}
|