import QtQuick 2.15 import QtTest 1.15 import SortFilterProxyModel 0.2 import StatusQ 0.1 import StatusQ.Core.Utils 0.1 import StatusQ.Core.Theme 0.1 import StatusQ.Core.Utils 0.1 as SQUtils import StatusQ.Core.Backpressure 0.1 import Models 1.0 import utils 1.0 import AppLayouts.Wallet.popups.buy 1.0 import AppLayouts.Wallet.stores 1.0 import shared.stores 1.0 Item { id: root width: 600 height: 800 QtObject { id: d } Component { id: componentUnderTest BuyCryptoModal { id: buySellModal buyProvidersModel: buyCryptoStore.providersModel isBuyProvidersModelLoading: buyCryptoStore.areProvidersLoading currentCurrency: currencyStore.currentCurrency walletAccountsModel: WalletAccountsModel{} networksModel: NetworksModel.flatNetworks areTestNetworksEnabled: true plainTokensBySymbolModel: assetsStore.walletTokensStore.plainTokensBySymbolModel groupedAccountAssetsModel: assetsStore.groupedAccountAssetsModel buyCryptoInputParamsForm: BuyCryptoParamsForm { selectedWalletAddress: "0x7F47C2e18a4BBf5487E6fb082eC2D9Ab0E6d7240" selectedNetworkChainId: 11155111 selectedTokenKey: "ETH" } Component.onCompleted: { fetchProviders.connect(buyCryptoStore.fetchProviders) fetchProviderUrl.connect(buyCryptoStore.fetchProviderUrl) buyCryptoStore.providerUrlReady.connect(buySellModal.providerUrlReady) } // Temporary assignments to make tests run independently readonly property var currencyStore: CurrenciesStore {} readonly property var buyCryptoStore: BuyCryptoStore { readonly property var providersModel: OnRampProvidersModel{} property bool areProvidersLoading signal providerUrlReady(string uuid ,string url) function fetchProviders() { console.warn("fetchProviders called >>") areProvidersLoading = true debounceFetchProvidersList() } function fetchProviderUrl(uuid, providerID, isRecurrent, accountAddress = "", chainID = 0, symbol = "") { console.warn("fetchProviderUrl called >> uuid: ", uuid, "providerID: ",providerID , "isRecurrent: ", isRecurrent, "accountAddress: ", accountAddress, "chainID: ", chainID, "symbol: ", symbol) buySellModal.uuid = uuid debounceFetchProviderUrl() } } readonly property ModelEntry selectedAccountEntry: ModelEntry { sourceModel: walletAccountsModel key: "address" value: buyCryptoInputParamsForm.selectedWalletAddress } readonly property ModelEntry selectedProviderEntry: ModelEntry { sourceModel: buyCryptoStore.providersModel key: "id" value: buyCryptoInputParamsForm.selectedProviderId } readonly property var recurrentOnRampProvidersModel: SortFilterProxyModel { sourceModel: buyProvidersModel filters: ValueFilter { roleName: "supportsRecurrentPurchase" value: true } } readonly property var assetsStore: WalletAssetsStore { id: thisWalletAssetStore walletTokensStore: TokensStore { plainTokensBySymbolModel: TokensBySymbolModel {} } readonly property var baseGroupedAccountAssetModel: GroupedAccountsAssetsModel {} assetsWithFilteredBalances: thisWalletAssetStore.groupedAccountsAssetsModel } property string uuid property var debounceFetchProviderUrl: Backpressure.debounce(root, 500, function() { buySellModal.buyCryptoStore.providerUrlReady(uuid, "xxxx") }) property var debounceFetchProvidersList: Backpressure.debounce(root, 500, function() { buySellModal.buyCryptoStore.areProvidersLoading = false }) } } SignalSpy { id: notificationSpy target: Global signalName: "openLinkWithConfirmation" } TestCase { name: "BuyCryptoModal" when: windowShown property BuyCryptoModal controlUnderTest: null function init() { notificationSpy.clear() controlUnderTest = createTemporaryObject(componentUnderTest, root) } function launchPopup() { verify(!!controlUnderTest) controlUnderTest.buyCryptoInputParamsForm.selectedWalletAddress = "0x7F47C2e18a4BBf5487E6fb082eC2D9Ab0E6d7240" controlUnderTest.buyCryptoInputParamsForm.selectedNetworkChainId = 11155111 controlUnderTest.buyCryptoInputParamsForm.selectedTokenKey = "ETH" controlUnderTest.open() verify(!!controlUnderTest.opened) } function testDelegateItems(providersList, modelToCompareAgainst) { for(let i =0; i< providersList.count; i++) { let delegateUnderTest = providersList.itemAtIndex(i) verify(!!delegateUnderTest) tryCompare(delegateUnderTest, "title", modelToCompareAgainst.get(i).name) compare(delegateUnderTest.subTitle, modelToCompareAgainst.get(i).description) compare(delegateUnderTest.asset.name, modelToCompareAgainst.get(i).logoUrl) compare(delegateUnderTest.isUrlLoading, false) const feesText = findChild(delegateUnderTest, "feesText") verify(!!feesText) compare(feesText.text, modelToCompareAgainst.get(i).fees) const loadingIndicator = findChild(delegateUnderTest, "loadingIndicator") verify(!!loadingIndicator) verify(!loadingIndicator.visible) var extraIcon = null if (modelToCompareAgainst.get(i).urlsNeedParameters) { extraIcon = findChild(delegateUnderTest, "chevron-down-icon") verify(!!extraIcon) compare(extraIcon.icon, "chevron-down") compare(extraIcon.rotation, 270) compare(extraIcon.color, Theme.palette.baseColor1) } else { extraIcon = findChild(delegateUnderTest, "tiny/external-icon") verify(!!extraIcon) compare(extraIcon.icon, "tiny/external") compare(extraIcon.rotation, 0) compare(extraIcon.color, Theme.palette.baseColor1) } // Hover over the item and check hovered state mouseMove(delegateUnderTest, delegateUnderTest.width/2, delegateUnderTest.height/2) verify(delegateUnderTest.sensor.containsMouse) compare(extraIcon.color, Theme.palette.directColor1) verify(delegateUnderTest.color, Theme.palette.baseColor2) } } function testDelegateMouseClicksForProvidersThatNeedParams(delegateUnderTest, modelData) { const loadingIndicator = findChild(delegateUnderTest, "loadingIndicator") verify(!!loadingIndicator) verify(!loadingIndicator.visible) verify(!controlUnderTest.replaceItem) // test mouse click tryCompare(notificationSpy, "count", 0) mouseClick(delegateUnderTest) waitForRendering(controlUnderTest.replaceLoader) verify(controlUnderTest.replaceItem) const selectParamsPanel = findChild(controlUnderTest, "selectParamsPanel") verify(!!selectParamsPanel) // title should not change verify(controlUnderTest.stackTitle, qsTr("Buy assets for %1").arg(!!controlUnderTest.selectedAccountEntry.item ? controlUnderTest.selectedAccountEntry.item.name: "")) compare(controlUnderTest.rightButtons.length, 2) verify(controlUnderTest.rightButtons[0].visible) verify(controlUnderTest.rightButtons[1].enabled) verify(controlUnderTest.rightButtons[0].text, qsTr("Buy via %1").arg(!!controlUnderTest.selectedProviderEntry.item ? controlUnderTest.selectedProviderEntry.item.name: "")) verify(!controlUnderTest.rightButtons[1].visible) verify(controlUnderTest.backButton.visible) const selectParamsForBuyCryptoPanelHeader = findChild(selectParamsPanel, "selectParamsForBuyCryptoPanelHeader") verify(!!selectParamsForBuyCryptoPanelHeader) compare(selectParamsForBuyCryptoPanelHeader.title, qsTr("Buy via %1").arg(!!controlUnderTest.selectedProviderEntry.item ? controlUnderTest.selectedProviderEntry.item.name: "")) compare(selectParamsForBuyCryptoPanelHeader.subTitle, qsTr("Select which network and asset")) compare(selectParamsForBuyCryptoPanelHeader.statusListItemTitle.color, Theme.palette.directColor1) compare(selectParamsForBuyCryptoPanelHeader.asset.name, !!controlUnderTest.selectedProviderEntry.item ? controlUnderTest.selectedProviderEntry.item .logoUrl: "") compare(selectParamsForBuyCryptoPanelHeader.color, Theme.palette.transparent) compare(selectParamsForBuyCryptoPanelHeader.enabled, false) const networkFilter = findChild(selectParamsPanel, "networkFilter") verify(!!networkFilter) compare(networkFilter.selection, [controlUnderTest.buyCryptoInputParamsForm.selectedNetworkChainId]) const tokenSelector = findChild(selectParamsPanel, "tokenSelector") verify(!!tokenSelector) compare(tokenSelector.currentTokensKey, controlUnderTest.buyCryptoInputParamsForm.selectedTokenKey) const selectedTokenItem = findChild(selectParamsPanel, "selectedTokenItem") verify(!!selectedTokenItem) const modelDataToTest = ModelUtils.getByKey(tokenSelector.model, "tokensKey", tokenSelector.currentTokensKey) const tokenSelectorIcon = findChild(selectedTokenItem, "tokenSelectorIcon") verify(!!tokenSelectorIcon) compare(tokenSelectorIcon.image.source, modelDataToTest.iconSource) const tokenSelectorContentItemName = findChild(selectedTokenItem, "tokenSelectorContentItemName") verify(!!tokenSelectorContentItemName) compare(tokenSelectorContentItemName.text, modelDataToTest.name) const tokenSelectorContentItemSymbol = findChild(selectedTokenItem, "tokenSelectorContentItemSymbol") verify(!!tokenSelectorContentItemSymbol) compare(tokenSelectorContentItemSymbol.text, modelDataToTest.symbol) //switch to a network that has no tokens and ensure its reset controlUnderTest.buyCryptoInputParamsForm.selectedNetworkChainId = 421613 waitForRendering(selectParamsPanel) const nothingSelectedContentItem = findChild(selectParamsPanel, "tokenSelectorContentItemText") verify(!!nothingSelectedContentItem) verify(!selectedTokenItem.visible) verify(!controlUnderTest.rightButtons[0].enabled) // switch back a network and token thats valid and check if clicking buy button works properly controlUnderTest.buyCryptoInputParamsForm.selectedNetworkChainId = 11155111 controlUnderTest.buyCryptoInputParamsForm.selectedTokenKey = "ETH" waitForRendering(selectParamsPanel) verify(controlUnderTest.rightButtons[0].enabled) mouseClick(controlUnderTest.rightButtons[0]) verify(controlUnderTest.rightButtons[0].loading) tryCompare(notificationSpy, "count", 1) compare(notificationSpy.signalArguments[0][0], "xxxx") compare(notificationSpy.signalArguments[0][1], modelData.hostname) notificationSpy.clear() // popup should be closed verify(!controlUnderTest.opened) } function testDelegateMouseClicksForProvidersThatNeedNoParams(delegateUnderTest, modelData) { // test provider that need no parameters and we directly redirect to the site const loadingIndicator = findChild(delegateUnderTest, "loadingIndicator") verify(!!loadingIndicator) verify(!loadingIndicator.visible) const extraIcon = findChild(delegateUnderTest, "tiny/external-icon") verify(!!extraIcon) verify(!extraIcon.visble) // test mouse click tryCompare(notificationSpy, "count", 0) mouseClick(delegateUnderTest) verify(loadingIndicator.visible) tryCompare(notificationSpy, "count", 1) compare(notificationSpy.signalArguments[0][0], "xxxx") compare(notificationSpy.signalArguments[0][1], modelData.hostname) notificationSpy.clear() // popup should be closed verify(!controlUnderTest.opened) } function test_launchAndCloseModal() { launchPopup() // close popup controlUnderTest.close() verify(!controlUnderTest.opened) } function test_ModalFooter() { // Launch modal launchPopup() // check if footer has Done button and action on button clicked compare(controlUnderTest.rightButtons.length, 2) verify(!controlUnderTest.rightButtons[0].visible) verify(controlUnderTest.rightButtons[1].visible) compare(controlUnderTest.rightButtons[1].text, qsTr("Done")) mouseClick(controlUnderTest.rightButtons[1]) verify(!controlUnderTest.backButton.visible) // popup should be closed verify(!controlUnderTest.opened) } function test_modalContent() { // Launch modal launchPopup() verify(controlUnderTest.stackTitle, qsTr("Buy assets for %1").arg(!!controlUnderTest.selectedAccountEntry.item ? controlUnderTest.selectedAccountEntry.item.name: "")) // find tab bar const tabBar = findChild(controlUnderTest, "tabBar") verify(!!tabBar) // find providers list const providersList = findChild(controlUnderTest, "providersList") waitForRendering(providersList) verify(!!providersList) // should have 2 items compare(tabBar.count, 2) // current index set should be to 0 compare(tabBar.currentIndex, 0) // item 0 should have text "One time" compare(tabBar.itemAt(0).text, qsTr("One time")) // item 1 should have text "Recurrent" compare(tabBar.itemAt(1).text, qsTr("Recurrent")) // close popup controlUnderTest.close() verify(!controlUnderTest.opened) } function test_modalContent_OneTime_tab() { notificationSpy.clear() // Launch modal launchPopup() // find tab bar const tabBar = findChild(controlUnderTest, "tabBar") verify(!!tabBar) // find providers list const providersList = findChild(controlUnderTest, "providersList") verify(!!providersList) tryCompare(controlUnderTest, "isBuyProvidersModelLoading", false) mouseClick(tabBar.itemAt(0)) compare(tabBar.currentIndex, 0) // verify that 4 items are listed compare(providersList.count, 4) // check if delegate contents are as expected testDelegateItems(providersList, controlUnderTest.buyProvidersModel) controlUnderTest.close() } function test_modalContent_OneTime_tab_mouseClicks() { notificationSpy.clear() // Launch modal launchPopup() // find providers list const providersList = findChild(controlUnderTest, "providersList") verify(!!providersList) for(let i =0; i< controlUnderTest.buyProvidersModel.count; i++) { notificationSpy.clear() launchPopup() verify(controlUnderTest.opened) tryCompare(controlUnderTest, "isBuyProvidersModelLoading", false) let delegateUnderTest = providersList.itemAtIndex(i) verify(!!delegateUnderTest) waitForRendering(delegateUnderTest) // test provider that need parameters like network and token to be selected const modelData = controlUnderTest.buyProvidersModel.get(i) verify(!!modelData) if (modelData.urlsNeedParameters) { testDelegateMouseClicksForProvidersThatNeedParams(delegateUnderTest, modelData) } else { testDelegateMouseClicksForProvidersThatNeedNoParams(delegateUnderTest, modelData) } } controlUnderTest.close() } function test_modalContent_recurrent_tab() { notificationSpy.clear() // Launch modal launchPopup() // find tab bar const tabBar = findChild(controlUnderTest, "tabBar") verify(!!tabBar) // find providers list const providersList = findChild(controlUnderTest, "providersList") verify(!!providersList) tryCompare(controlUnderTest, "isBuyProvidersModelLoading", false) // check data in "Recurrent" tab -------------------------------------------------------- mouseClick(tabBar.itemAt(1)) compare(tabBar.currentIndex, 1) waitForRendering(providersList) verify(!!providersList) // verify that 1 item is listed compare(providersList.count, 1) // check if delegate contents are as expected testDelegateItems(providersList, controlUnderTest.recurrentOnRampProvidersModel) controlUnderTest.close() } function test_modalContent_Recurrent_tab_mouseClicks() { notificationSpy.clear() // Launch modal launchPopup() // find tab bar const tabBar = findChild(controlUnderTest, "tabBar") verify(!!tabBar) // find providers list const providersList = findChild(controlUnderTest, "providersList") verify(!!providersList) mouseClick(tabBar.itemAt(1)) compare(tabBar.currentIndex, 1) waitForRendering(providersList) verify(!!providersList) for(let i =0; i< controlUnderTest.recurrentOnRampProvidersModel.count; i++) { notificationSpy.clear() launchPopup() verify(controlUnderTest.opened) tryCompare(controlUnderTest, "isBuyProvidersModelLoading", false) let delegateUnderTest = providersList.itemAtIndex(i) verify(!!delegateUnderTest) waitForRendering(delegateUnderTest) // test provider that need parameters like network and token to be selected const modelData = controlUnderTest.recurrentOnRampProvidersModel.get(i) verify(!!modelData) if (modelData.urlsNeedParameters) { testDelegateMouseClicksForProvidersThatNeedParams(delegateUnderTest, modelData) } else { testDelegateMouseClicksForProvidersThatNeedNoParams(delegateUnderTest, modelData) } } } } }