import QtQuick 2.12 import QtQuick.Controls 2.14 import QtQuick.Layouts 1.14 import StatusQ.Core 0.1 import StatusQ.Core.Theme 0.1 import StatusQ.Popups 0.1 import StatusQ.Components 0.1 import StatusQ.Controls 0.1 import utils 1.0 import "../stores" Item { id: derivedAddresses property int selectedAccountType: RootStore.defaultSelectedType property string selectedKeyUid: RootStore.defaultSelectedKeyUid property bool selectedKeyUidMigratedToKeycard: RootStore.defaultSelectedKeyUidMigratedToKeycard property string selectedPath: "" property string pathSubFix: "" property bool isLoading: RootStore.derivedAddressesLoading property bool pathError: Utils.isInvalidPath(RootStore.derivedAddressesError) property alias selectedAddress: selectedDerivedAddress.title property alias selectedAddressAvailable: selectedDerivedAddress.enabled function reset() { RootStore.resetDerivedAddressModel() _internal.nextSelectableAddressIndex = 0 selectedDerivedAddress.pathSubFix = 0 selectedDerivedAddress.title = "---" selectedDerivedAddress.subTitle = qsTr("No activity") } onIsLoadingChanged: { if(isLoading) { selectedDerivedAddress.title = qsTr("Pending") selectedDerivedAddress.subTitle = "" } } onPathErrorChanged: { if(pathError) { selectedDerivedAddress.title = qsTr("Invalid path") selectedDerivedAddress.subTitle = "" } } QtObject { id: _internal readonly property bool showEnterPinPassButton: !RootStore.loggedInUserAuthenticated && derivedAddresses.selectedAccountType !== SelectGeneratedAccount.AddAccountType.ImportSeedPhrase && derivedAddresses.selectedAccountType !== SelectGeneratedAccount.AddAccountType.ImportPrivateKey property int pageSize: 6 property int noOfPages: Math.ceil(RootStore.derivedAddressesList.count/pageSize) property int lastPageSize: RootStore.derivedAddressesList.count - ((noOfPages -1) * pageSize) property bool isLastPage: stackLayout.currentIndex == (noOfPages - 1) property int nextSelectableAddressIndex: RootStore.getNextSelectableDerivedAddressIndex() onNextSelectableAddressIndexChanged: { stackLayout.currentIndex = nextSelectableAddressIndex/_internal.pageSize if(nextSelectableAddressIndex >= 0 && nextSelectableAddressIndex < RootStore.derivedAddressesList.count) { selectedDerivedAddress.title = RootStore.getDerivedAddressData(nextSelectableAddressIndex) selectedDerivedAddress.hasActivity = RootStore.getDerivedAddressHasActivityData(nextSelectableAddressIndex) selectedDerivedAddress.subTitle = RootStore.getDerivedAddressHasActivityData(nextSelectableAddressIndex) ? qsTr("Has Activity"): qsTr("No Activity") selectedDerivedAddress.enabled = !RootStore.getDerivedAddressAlreadyCreatedData(nextSelectableAddressIndex) selectedDerivedAddress.pathSubFix = nextSelectableAddressIndex } } // dimensions property int popupWidth: 359 property int maxAddressWidth: 102 function runAction() { if (derivedAddresses.selectedKeyUidMigratedToKeycard) RootStore.authenticateUserAndDeriveAddressOnKeycardForPath(derivedAddresses.selectedKeyUid, derivedAddresses.selectedPath) else RootStore.authenticateUser() } } Connections { target: RootStore.derivedAddressesList onModelReset: { _internal.pageSize = 0 _internal.pageSize = 6 _internal.nextSelectableAddressIndex = -1 _internal.nextSelectableAddressIndex = RootStore.getNextSelectableDerivedAddressIndex() } } ColumnLayout { id: layout width: parent.width spacing: 7 StatusBaseText { id: inputLabel width: parent.width text: qsTr("Account") font.pixelSize: 15 color: selectedDerivedAddress.enabled ? Theme.palette.directColor1 : Theme.palette.baseColor1 } StatusListItem { id: selectedDerivedAddress property int pathSubFix: 0 property bool hasActivity: false implicitWidth: parent.width visible: !_internal.showEnterPinPassButton color: "transparent" border.width: 1 border.color: Theme.palette.baseColor2 title: "---" subTitle: selectedDerivedAddress.hasActivity ? qsTr("Has Activity"): qsTr("No Activity") statusListItemSubTitle.color: selectedDerivedAddress.hasActivity ? Theme.palette.primaryColor1 : Theme.palette.baseColor1 statusListItemTitle.wrapMode: Text.NoWrap statusListItemTitle.width: _internal.maxAddressWidth statusListItemTitle.elide: Qt.ElideMiddle statusListItemTitle.anchors.left: undefined statusListItemTitle.anchors.right: undefined components: [ StatusIcon { width: 24 height: 24 icon: "chevron-down" color: Theme.palette.baseColor1 visible: RootStore.derivedAddressesList.count > 1 } ] onClicked: { if(RootStore.derivedAddressesList.count > 0 && RootStore.derivedAddressesList.count !== 1) derivedAddressPopup.popup(derivedAddresses.x - layout.width - Style.current.bigPadding , derivedAddresses.y + layout.height + 8) } enabled: RootStore.derivedAddressesList.count > 0 Component.onCompleted: derivedAddresses.pathSubFix = Qt.binding(function() { return pathSubFix}) } StatusButton { visible: _internal.showEnterPinPassButton text: qsTr("Preview address") highlighted: focus onClicked: _internal.runAction() } } StatusMenu { id: derivedAddressPopup width: _internal.popupWidth contentItem: Column { StackLayout { id: stackLayout Layout.fillWidth:true Layout.fillHeight: true Repeater { id: pageModel model: _internal.noOfPages delegate: Page { id: page contentItem: ColumnLayout { Repeater { id: repeater model: _internal.isLastPage ? _internal.lastPageSize : _internal.pageSize delegate: StatusListItem { id: element property int actualIndex: index + (stackLayout.currentIndex* _internal.pageSize) property bool hasActivity: { if(actualIndex >= 0 && actualIndex < RootStore.derivedAddressesList.count) { return RootStore.getDerivedAddressHasActivityData(actualIndex) } return false } implicitWidth: derivedAddressPopup.width statusListItemTitle.wrapMode: Text.NoWrap statusListItemTitle.width: _internal.maxAddressWidth statusListItemTitle.elide: Qt.ElideMiddle statusListItemTitle.anchors.left: undefined statusListItemTitle.anchors.right: undefined title: { if(actualIndex >= 0 && actualIndex < RootStore.derivedAddressesList.count) { return RootStore.getDerivedAddressData(actualIndex) } return "" } subTitle: element.hasActivity ? qsTr("Has Activity"): qsTr("No Activity") statusListItemSubTitle.color: element.hasActivity ? Theme.palette.primaryColor1 : Theme.palette.baseColor1 enabled: { if(actualIndex >= 0 && actualIndex < RootStore.derivedAddressesList.count) { return !RootStore.getDerivedAddressAlreadyCreatedData(actualIndex) } return true } components: [ StatusBaseText { text: element.actualIndex font.pixelSize: 15 color: Theme.palette.baseColor1 }, Rectangle { radius: width/2 height: 5 width: 5 color: Theme.palette.primaryColor1 visible: element.hasActivity anchors.verticalCenter: parent.verticalCenter } ] onClicked: { selectedDerivedAddress.title = title selectedDerivedAddress.subTitle = subTitle selectedDerivedAddress.pathSubFix = actualIndex selectedDerivedAddress.hasActivity = element.hasActivity derivedAddressPopup.close() } Component.onCompleted: { if(RootStore.derivedAddressesList.count === 1 && index === 0) { selectedDerivedAddress.title = title selectedDerivedAddress.subTitle = subTitle selectedDerivedAddress.pathSubFix = actualIndex selectedDerivedAddress.hasActivity = element.hasActivity selectedDerivedAddress.enabled = !RootStore.getDerivedAddressAlreadyCreatedData(index) } } } } } background: Rectangle { implicitWidth: stackLayout.width implicitHeight: stackLayout.height color: Theme.palette.statusMenu.backgroundColor radius: 8 } } } } PageIndicator { id: pageIndicator anchors.horizontalCenter: parent.horizontalCenter interactive: true currentIndex: stackLayout.currentIndex count: stackLayout.count onCurrentIndexChanged: stackLayout.currentIndex = currentIndex } } } }