2024-05-15 21:22:13 +00:00
|
|
|
import QtQml 2.15
|
|
|
|
import SortFilterProxyModel 0.2
|
|
|
|
|
|
|
|
import StatusQ 0.1
|
|
|
|
import StatusQ.Core.Utils 0.1
|
|
|
|
|
|
|
|
import utils 1.0
|
|
|
|
|
|
|
|
import shared.stores 1.0
|
|
|
|
import AppLayouts.Wallet.stores 1.0 as WalletStore
|
|
|
|
|
|
|
|
QObject {
|
|
|
|
id: root
|
|
|
|
|
|
|
|
required property CurrenciesStore currencyStore
|
|
|
|
required property WalletStore.WalletAssetsStore walletAssetsStore
|
|
|
|
required property WalletStore.SwapStore swapStore
|
|
|
|
required property SwapInputParamsForm swapFormData
|
2024-06-06 14:05:31 +00:00
|
|
|
required property SwapOutputData swapOutputData
|
2024-05-15 21:22:13 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
// the below 2 properties holds the state of finding a swap proposal
|
|
|
|
property bool validSwapProposalReceived: false
|
2024-06-04 11:58:37 +00:00
|
|
|
property bool swapProposalLoading: false
|
|
|
|
|
2024-06-12 20:43:08 +00:00
|
|
|
property bool showCommunityTokens
|
|
|
|
|
2024-06-04 11:58:37 +00:00
|
|
|
// To expose the selected from and to Token from the SwapModal
|
|
|
|
readonly property var fromToken: ModelUtils.getByKey(root.walletAssetsStore.walletTokensStore.plainTokensBySymbolModel, "key", root.swapFormData.fromTokensKey)
|
|
|
|
readonly property var toToken: ModelUtils.getByKey(root.walletAssetsStore.walletTokensStore.plainTokensBySymbolModel, "key", root.swapFormData.toTokenKey)
|
|
|
|
|
2024-05-15 21:22:13 +00:00
|
|
|
readonly property var nonWatchAccounts: SortFilterProxyModel {
|
|
|
|
sourceModel: root.swapStore.accounts
|
|
|
|
filters: ValueFilter {
|
|
|
|
roleName: "walletType"
|
|
|
|
value: Constants.watchWalletType
|
|
|
|
inverted: true
|
|
|
|
}
|
|
|
|
sorters: RoleSorter { roleName: "position"; sortOrder: Qt.AscendingOrder }
|
|
|
|
proxyRoles: [
|
|
|
|
FastExpressionRole {
|
|
|
|
name: "accountBalance"
|
2024-06-06 14:05:31 +00:00
|
|
|
expression: d.processAccountBalance(model.address)
|
2024-05-15 21:22:13 +00:00
|
|
|
expectedRoles: ["address"]
|
|
|
|
},
|
|
|
|
FastExpressionRole {
|
|
|
|
name: "fromToken"
|
2024-06-04 11:58:37 +00:00
|
|
|
expression: root.fromToken
|
2024-05-15 21:22:13 +00:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
2024-05-29 15:42:26 +00:00
|
|
|
readonly property SortFilterProxyModel filteredFlatNetworksModel: SortFilterProxyModel {
|
|
|
|
sourceModel: root.swapStore.flatNetworks
|
|
|
|
filters: ValueFilter { roleName: "isTest"; value: root.swapStore.areTestNetworksEnabled }
|
|
|
|
}
|
|
|
|
|
2024-05-28 17:39:41 +00:00
|
|
|
// Model prepared to provide filtered and sorted assets as per the advanced Settings in token management
|
|
|
|
readonly property var processedAssetsModel: SortFilterProxyModel {
|
|
|
|
property real displayAssetsBelowBalanceThresholdAmount: root.walletAssetsStore.walletTokensStore.getDisplayAssetsBelowBalanceThresholdDisplayAmount()
|
2024-06-06 14:05:31 +00:00
|
|
|
sourceModel: d.assetsWithFilteredBalances
|
2024-05-28 17:39:41 +00:00
|
|
|
proxyRoles: [
|
|
|
|
FastExpressionRole {
|
|
|
|
name: "currentBalance"
|
2024-06-12 20:43:08 +00:00
|
|
|
expression: {
|
|
|
|
// FIXME recalc when selectedNetworkChainId changes
|
|
|
|
root.swapFormData.selectedNetworkChainId
|
2024-06-06 14:05:31 +00:00
|
|
|
return d.getTotalBalance(model.balances, model.decimals)
|
2024-06-12 20:43:08 +00:00
|
|
|
}
|
2024-05-28 17:39:41 +00:00
|
|
|
expectedRoles: ["balances", "decimals"]
|
|
|
|
},
|
|
|
|
FastExpressionRole {
|
|
|
|
name: "currentCurrencyBalance"
|
|
|
|
expression: {
|
|
|
|
if (!!model.marketDetails) {
|
|
|
|
return model.currentBalance * model.marketDetails.currencyPrice.amount
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
expectedRoles: ["marketDetails", "currentBalance"]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
filters: [
|
|
|
|
FastExpressionFilter {
|
|
|
|
expression: {
|
|
|
|
root.walletAssetsStore.assetsController.revision
|
|
|
|
|
|
|
|
if (!root.walletAssetsStore.assetsController.filterAcceptsSymbol(model.symbol)) // explicitely hidden
|
|
|
|
return false
|
2024-06-12 20:43:08 +00:00
|
|
|
if (!!model.communityId)
|
|
|
|
return root.showCommunityTokens
|
2024-05-28 17:39:41 +00:00
|
|
|
if (root.walletAssetsStore.walletTokensStore.displayAssetsBelowBalance)
|
|
|
|
return model.currentCurrencyBalance > processedAssetsModel.displayAssetsBelowBalanceThresholdAmount
|
|
|
|
return true
|
|
|
|
}
|
2024-06-12 20:43:08 +00:00
|
|
|
expectedRoles: ["symbol", "communityId", "currentCurrencyBalance"]
|
2024-05-28 17:39:41 +00:00
|
|
|
}
|
|
|
|
]
|
|
|
|
// FIXME sort by assetsController instead, to have the sorting/order as in the main wallet view
|
|
|
|
}
|
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
QtObject {
|
|
|
|
id: d
|
2024-05-15 21:22:13 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
property string uuid
|
2024-05-28 17:39:41 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
// Internal model filtering balances by the account selected in the AccountsModalHeader
|
|
|
|
readonly property SubmodelProxyModel assetsWithFilteredBalances: SubmodelProxyModel {
|
|
|
|
sourceModel: root.walletAssetsStore.groupedAccountAssetsModel
|
|
|
|
submodelRoleName: "balances"
|
|
|
|
delegateModel: SortFilterProxyModel {
|
|
|
|
sourceModel: submodel
|
|
|
|
|
|
|
|
filters: [
|
|
|
|
ValueFilter {
|
|
|
|
roleName: "chainId"
|
|
|
|
value: root.swapFormData.selectedNetworkChainId
|
|
|
|
enabled: root.swapFormData.selectedNetworkChainId !== -1
|
|
|
|
}/*,
|
|
|
|
// TODO enable once AccountsModalHeader is reworked!!
|
|
|
|
ValueFilter {
|
|
|
|
roleName: "account"
|
|
|
|
value: root.selectedSenderAccount.address
|
|
|
|
}*/
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
readonly property SubmodelProxyModel filteredBalancesModel: SubmodelProxyModel {
|
|
|
|
sourceModel: root.walletAssetsStore.baseGroupedAccountAssetModel
|
|
|
|
submodelRoleName: "balances"
|
|
|
|
delegateModel: SortFilterProxyModel {
|
|
|
|
sourceModel: joinModel
|
|
|
|
filters: ValueFilter {
|
2024-05-28 17:39:41 +00:00
|
|
|
roleName: "chainId"
|
|
|
|
value: root.swapFormData.selectedNetworkChainId
|
2024-06-06 14:05:31 +00:00
|
|
|
}
|
|
|
|
readonly property LeftJoinModel joinModel: LeftJoinModel {
|
|
|
|
leftModel: submodel
|
|
|
|
rightModel: root.swapStore.flatNetworks
|
|
|
|
|
|
|
|
joinRole: "chainId"
|
|
|
|
}
|
|
|
|
}
|
2024-05-28 17:39:41 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
function processAccountBalance(address) {
|
|
|
|
let network = ModelUtils.getByKey(root.filteredFlatNetworksModel, "chainId", root.swapFormData.selectedNetworkChainId)
|
|
|
|
if(!!network) {
|
|
|
|
let balancesModel = ModelUtils.getByKey(filteredBalancesModel, "tokensKey", root.swapFormData.fromTokensKey, "balances")
|
|
|
|
let accountBalance = ModelUtils.getByKey(balancesModel, "account", address)
|
|
|
|
if(!accountBalance) {
|
|
|
|
return {
|
|
|
|
balance: "0",
|
|
|
|
iconUrl: network.iconUrl,
|
|
|
|
chainColor: network.chainColor}
|
|
|
|
}
|
|
|
|
return accountBalance
|
2024-05-28 13:19:46 +00:00
|
|
|
}
|
2024-06-06 14:05:31 +00:00
|
|
|
return null
|
|
|
|
}
|
2024-05-28 13:19:46 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
/* Internal function to calculate total balance */
|
|
|
|
function getTotalBalance(balances, decimals, chainIds = [root.swapFormData.selectedNetworkChainId]) {
|
|
|
|
let totalBalance = 0
|
|
|
|
for(let i=0; i<balances.count; i++) {
|
|
|
|
let balancePerAddressPerChain = ModelUtils.get(balances, i)
|
|
|
|
if (chainIds.includes(-1) || chainIds.includes(balancePerAddressPerChain.chainId))
|
|
|
|
totalBalance += AmountsArithmetic.toNumber(balancePerAddressPerChain.balance, decimals)
|
2024-05-28 13:19:46 +00:00
|
|
|
}
|
2024-06-06 14:05:31 +00:00
|
|
|
return totalBalance
|
2024-05-28 13:19:46 +00:00
|
|
|
}
|
2024-05-15 21:22:13 +00:00
|
|
|
}
|
2024-05-28 13:19:46 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
Connections {
|
|
|
|
target: root.swapStore
|
|
|
|
function onSuggestedRoutesReady(txRoutes) {
|
|
|
|
root.swapOutputData.reset()
|
|
|
|
root.validSwapProposalReceived = false
|
|
|
|
root.swapProposalLoading = false
|
|
|
|
root.swapOutputData.rawPaths = txRoutes.rawPaths
|
|
|
|
// if valid route was found
|
|
|
|
if(txRoutes.suggestedRoutes.count === 1) {
|
|
|
|
root.validSwapProposalReceived = true
|
|
|
|
root.swapOutputData.bestRoutes = txRoutes.suggestedRoutes
|
|
|
|
root.swapOutputData.toTokenAmount = root.swapStore.getWei2Eth(txRoutes.amountToReceive, root.toToken.decimals).toString()
|
|
|
|
let gasTimeEstimate = txRoutes.gasTimeEstimate
|
|
|
|
let totalTokenFeesInFiat = 0
|
|
|
|
if (!!root.fromToken && !!root.fromToken .marketDetails && !!root.fromToken.marketDetails.currencyPrice)
|
|
|
|
totalTokenFeesInFiat = gasTimeEstimate.totalTokenFees * root.fromToken.marketDetails.currencyPrice.amount
|
|
|
|
root.swapOutputData.totalFees = root.currencyStore.getFiatValue(gasTimeEstimate.totalFeesInEth, Constants.ethToken) + totalTokenFeesInFiat
|
|
|
|
root.swapOutputData.approvalNeeded = ModelUtils.get(root.swapOutputData.bestRoutes, 0, "route").approvalRequired
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
root.swapOutputData.hasError = true
|
2024-05-15 21:22:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-05-28 17:39:41 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
function reset() {
|
|
|
|
root.swapFormData.resetFormData()
|
|
|
|
root.swapOutputData.reset()
|
|
|
|
root.validSwapProposalReceived = false
|
|
|
|
root.swapProposalLoading = false
|
|
|
|
}
|
|
|
|
|
|
|
|
function getNetworkShortNames(chainIds) {
|
|
|
|
var networkString = ""
|
|
|
|
let chainIdsArray = chainIds.split(":")
|
|
|
|
for (let i = 0; i< chainIdsArray.length; i++) {
|
|
|
|
let nwShortName = ModelUtils.getByKey(root.filteredFlatNetworksModel, "chainId", Number(chainIdsArray[i]), "shortName")
|
|
|
|
if(!!nwShortName) {
|
|
|
|
networkString = networkString + nwShortName + ':'
|
|
|
|
}
|
2024-05-28 17:39:41 +00:00
|
|
|
}
|
2024-06-06 14:05:31 +00:00
|
|
|
return networkString
|
2024-05-28 17:39:41 +00:00
|
|
|
}
|
2024-06-10 12:51:33 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
function formatCurrencyAmount(balance, symbol, options = null, locale = null) {
|
|
|
|
return root.currencyStore.formatCurrencyAmount(balance, symbol, options, locale)
|
|
|
|
}
|
2024-06-10 12:51:33 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
function formatCurrencyAmountFromBigInt(balance, symbol, decimals, options = null) {
|
|
|
|
return root.currencyStore.formatCurrencyAmountFromBigInt(balance, symbol, decimals, options)
|
|
|
|
}
|
2024-06-10 12:51:33 +00:00
|
|
|
|
2024-06-06 14:05:31 +00:00
|
|
|
function getAllChainIds() {
|
|
|
|
return ModelUtils.joinModelEntries(root.filteredFlatNetworksModel, "chainId", ":")
|
|
|
|
}
|
|
|
|
|
|
|
|
function getDisabledChainIds(enabledChainId) {
|
|
|
|
let disabledChainIds = []
|
|
|
|
let chainIds = ModelUtils.modelToFlatArray(root.filteredFlatNetworksModel, "chainId")
|
|
|
|
for (let i = 0; i < chainIds.length; i++) {
|
|
|
|
if (chainIds[i] !== enabledChainId) {
|
|
|
|
disabledChainIds.push(chainIds[i])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return disabledChainIds.join(":")
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: remove once the AccountsModalHeader is reworked!!
|
|
|
|
function getSelectedAccount(index) {
|
|
|
|
if (root.nonWatchAccounts.count > 0 && index >= 0) {
|
|
|
|
return ModelUtils.get(nonWatchAccounts, index)
|
|
|
|
}
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
function fetchSuggestedRoutes() {
|
|
|
|
let amount = !!root.swapFormData.fromTokenAmount ? AmountsArithmetic.fromString(root.swapFormData.fromTokenAmount): NaN
|
|
|
|
root.swapOutputData.reset()
|
|
|
|
|
|
|
|
if(!isNaN(amount) && !!root.fromToken && root.swapFormData.isFormFilledCorrectly()) {
|
|
|
|
let fromTokenAmountInWei = AmountsArithmetic.fromNumber(amount, !!root.fromToken ? root.fromToken.decimals: 18).toString()
|
|
|
|
|
|
|
|
root.validSwapProposalReceived = false
|
|
|
|
|
|
|
|
// Identify new swap with a different uuid
|
|
|
|
d.uuid = Utils.uuid()
|
|
|
|
|
|
|
|
let account = getSelectedAccount(root.swapFormData.selectedAccountIndex)
|
|
|
|
let accountAddress = account.address
|
|
|
|
let disabledChainIds = getDisabledChainIds(root.swapFormData.selectedNetworkChainId)
|
|
|
|
let preferedChainIds = getAllChainIds()
|
|
|
|
|
|
|
|
// TODO #14825: amount should be in BigInt string representation (fromTokenAmount * 10^decimals)
|
|
|
|
// Make sure that's replaced when the input component is integrated
|
|
|
|
root.swapStore.fetchSuggestedRoutes(accountAddress, accountAddress,
|
|
|
|
fromTokenAmountInWei, root.swapFormData.fromTokensKey, root.swapFormData.toTokenKey,
|
|
|
|
disabledChainIds, disabledChainIds, preferedChainIds,
|
|
|
|
Constants.SendType.Swap, "")
|
|
|
|
} else {
|
|
|
|
root.validSwapProposalReceived = false
|
|
|
|
root.swapProposalLoading = false
|
|
|
|
}
|
2024-06-10 12:51:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function sendApproveTx() {
|
|
|
|
let account = getSelectedAccount(root.swapFormData.selectedAccountIndex)
|
|
|
|
let accountAddress = account.address
|
|
|
|
|
|
|
|
root.swapStore.authenticateAndTransfer(d.uuid, accountAddress, accountAddress,
|
|
|
|
root.swapFormData.fromTokensKey, root.swapFormData.toTokenKey,
|
2024-06-06 14:05:31 +00:00
|
|
|
Constants.SendType.Approve, "", false, root.swapOutputData.rawPaths)
|
2024-06-10 12:51:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function sendSwapTx() {
|
|
|
|
let account = getSelectedAccount(root.swapFormData.selectedAccountIndex)
|
|
|
|
let accountAddress = account.address
|
|
|
|
|
|
|
|
root.swapStore.authenticateAndTransfer(d.uuid, accountAddress, accountAddress,
|
|
|
|
root.swapFormData.fromTokensKey, root.swapFormData.toTokenKey,
|
2024-06-06 14:05:31 +00:00
|
|
|
Constants.SendType.Swap, "", false, root.swapOutputData.rawPaths)
|
2024-06-10 12:51:33 +00:00
|
|
|
}
|
2024-05-15 21:22:13 +00:00
|
|
|
}
|