feat(Wallet): add activity filter basic API
Add the possibility of retrieving the data of wallet activity based on the given filter criteria. Major changes: - Bump status-go with the new equivalent API - Add temporary developer tools - Debugging ActivityView QML component to test filter and display the activity - Add activity Nim package and synchronous controller and model Considerations - Have the model synchronous for the first iteration and then move to async while implementing the fetching mechanism - Use QtObject instances in the model instead of roles over items as agreed with the team - Simplify the implementation by having a simple presentation layer using backend (service also in the future) - Small required fixes and improvements Closes: #10633 Updates #10366
This commit is contained in:
parent
cba67b3b23
commit
22df203653
1
nim.cfg
1
nim.cfg
|
@ -1,3 +1,4 @@
|
||||||
# we need to link C++ libraries
|
# we need to link C++ libraries
|
||||||
gcc.linkerexe="g++"
|
gcc.linkerexe="g++"
|
||||||
|
|
||||||
|
path = "src"
|
|
@ -0,0 +1,131 @@
|
||||||
|
import NimQml, logging, std/json, sequtils, sugar, options
|
||||||
|
|
||||||
|
import model
|
||||||
|
import entry
|
||||||
|
|
||||||
|
import ../transactions/item
|
||||||
|
import ../transactions/module as transactions_module
|
||||||
|
|
||||||
|
import backend/activity as backend_activity
|
||||||
|
import backend/backend as backend
|
||||||
|
import backend/transactions
|
||||||
|
|
||||||
|
import app_service/service/transaction/service as transaction_service
|
||||||
|
|
||||||
|
proc toRef*[T](obj: T): ref T =
|
||||||
|
new(result)
|
||||||
|
result[] = obj
|
||||||
|
|
||||||
|
QtObject:
|
||||||
|
type
|
||||||
|
Controller* = ref object of QObject
|
||||||
|
model: Model
|
||||||
|
transactionsModule: transactions_module.AccessInterface
|
||||||
|
currentActivityFilter: backend_activity.ActivityFilter
|
||||||
|
|
||||||
|
proc setup(self: Controller) =
|
||||||
|
self.QObject.setup
|
||||||
|
|
||||||
|
proc delete*(self: Controller) =
|
||||||
|
self.QObject.delete
|
||||||
|
|
||||||
|
proc newController*(transactionsModule: transactions_module.AccessInterface): Controller =
|
||||||
|
new(result, delete)
|
||||||
|
result.model = newModel()
|
||||||
|
result.transactionsModule = transactionsModule
|
||||||
|
result.currentActivityFilter = backend_activity.getIncludeAllActivityFilter()
|
||||||
|
result.setup()
|
||||||
|
|
||||||
|
proc getModel*(self: Controller): QVariant {.slot.} =
|
||||||
|
return newQVariant(self.model)
|
||||||
|
|
||||||
|
QtProperty[QVariant] model:
|
||||||
|
read = getModel
|
||||||
|
|
||||||
|
# TODO: move it to service, make it async and lazy load details for transactions
|
||||||
|
proc backendToPresentation(self: Controller, backendEnties: seq[backend_activity.ActivityEntry]): seq[entry.ActivityEntry] =
|
||||||
|
var multiTransactionsIds: seq[int] = @[]
|
||||||
|
var transactionIdentities: seq[backend.TransactionIdentity] = @[]
|
||||||
|
var pendingTransactionIdentities: seq[backend.TransactionIdentity] = @[]
|
||||||
|
|
||||||
|
# Extract metadata required to fetch details
|
||||||
|
# TODO: temporary here to show the working API. Will be done as required on a detail request from UI
|
||||||
|
for backendEntry in backendEnties:
|
||||||
|
case backendEntry.transactionType:
|
||||||
|
of MultiTransaction:
|
||||||
|
multiTransactionsIds.add(backendEntry.id)
|
||||||
|
of SimpleTransaction:
|
||||||
|
transactionIdentities.add(backendEntry.transaction.get())
|
||||||
|
of PendingTransaction:
|
||||||
|
pendingTransactionIdentities.add(backendEntry.transaction.get())
|
||||||
|
|
||||||
|
var multiTransactions: seq[MultiTransactionDto] = @[]
|
||||||
|
if len(multiTransactionsIds) > 0:
|
||||||
|
multiTransactions = transaction_service.getMultiTransactions(multiTransactionsIds)
|
||||||
|
|
||||||
|
var transactions: seq[Item] = @[]
|
||||||
|
if len(transactionIdentities) > 0:
|
||||||
|
let response = backend.getTransfersForIdentities(transactionIdentities)
|
||||||
|
let res = response.result
|
||||||
|
if response.error != nil or res.kind != JArray or res.len == 0:
|
||||||
|
raise newException(Defect, "failed fetching transaction details")
|
||||||
|
|
||||||
|
let transactionsDtos = res.getElems().map(x => x.toTransactionDto())
|
||||||
|
transactions = self.transactionsModule.transactionsToItems(transactionsDtos, @[])
|
||||||
|
|
||||||
|
var pendingTransactions: seq[Item] = @[]
|
||||||
|
if len(pendingTransactionIdentities) > 0:
|
||||||
|
let response = backend.getPendingTransactionsForIdentities(pendingTransactionIdentities)
|
||||||
|
let res = response.result
|
||||||
|
if response.error != nil or res.kind != JArray or res.len == 0:
|
||||||
|
raise newException(Defect, "failed fetching pending transactions details")
|
||||||
|
|
||||||
|
let pendingTransactionsDtos = res.getElems().map(x => x.toPendingTransactionDto())
|
||||||
|
pendingTransactions = self.transactionsModule.transactionsToItems(pendingTransactionsDtos, @[])
|
||||||
|
|
||||||
|
# Merge detailed transaction info in order
|
||||||
|
result = newSeq[entry.ActivityEntry](multiTransactions.len + transactions.len + pendingTransactions.len)
|
||||||
|
var mtIndex = 0
|
||||||
|
var tIndex = 0
|
||||||
|
var ptIndex = 0
|
||||||
|
for i in low(backendEnties) .. high(backendEnties):
|
||||||
|
let backendEntry = backendEnties[i]
|
||||||
|
case backendEntry.transactionType:
|
||||||
|
of MultiTransaction:
|
||||||
|
result[i] = entry.newMultiTransactionActivityEntry(multiTransactions[mtIndex])
|
||||||
|
mtIndex += 1
|
||||||
|
of SimpleTransaction:
|
||||||
|
let refInstance = new(Item)
|
||||||
|
refInstance[] = transactions[tIndex]
|
||||||
|
result[i] = entry.newTransactionActivityEntry(refInstance, false)
|
||||||
|
tIndex += 1
|
||||||
|
of PendingTransaction:
|
||||||
|
let refInstance = new(Item)
|
||||||
|
refInstance[] = pendingTransactions[ptIndex]
|
||||||
|
result[i] = entry.newTransactionActivityEntry(refInstance, true)
|
||||||
|
ptIndex += 1
|
||||||
|
|
||||||
|
proc refreshData*(self: Controller) {.slot.} =
|
||||||
|
# result type is RpcResponse
|
||||||
|
let response = backend_activity.getActivityEntries(@["0x0000000000000000000000000000000000000001"], @[1], self.currentActivityFilter, 0, 10)
|
||||||
|
# RPC returns null for result in case of empty array
|
||||||
|
if response.error != nil or (response.result.kind != JArray and response.result.kind != JNull):
|
||||||
|
error "error fetching activity entries: ", response.error
|
||||||
|
return
|
||||||
|
|
||||||
|
if response.result.kind == JNull:
|
||||||
|
self.model.setEntries(@[])
|
||||||
|
return
|
||||||
|
|
||||||
|
var backendEnties = newSeq[backend_activity.ActivityEntry](response.result.len)
|
||||||
|
for i in 0 ..< response.result.len:
|
||||||
|
backendEnties[i] = fromJson(response.result[i], backend_activity.ActivityEntry)
|
||||||
|
let entries = self.backendToPresentation(backendEnties)
|
||||||
|
self.model.setEntries(entries)
|
||||||
|
|
||||||
|
# TODO: add all parameters and separate in different methods
|
||||||
|
proc updateFilter*(self: Controller, startTimestamp: int, endTimestamp: int) {.slot.} =
|
||||||
|
# Update filter
|
||||||
|
self.currentActivityFilter.period = backend_activity.newPeriod(startTimestamp, endTimestamp)
|
||||||
|
|
||||||
|
self.refreshData()
|
|
@ -0,0 +1,127 @@
|
||||||
|
import NimQml, tables, json, strformat, sequtils, strutils, logging
|
||||||
|
|
||||||
|
import ../transactions/view
|
||||||
|
import ../transactions/item
|
||||||
|
import ./backend/transactions
|
||||||
|
|
||||||
|
# The ActivityEntry contains one of the following instances transaction, pensing transaction or multi-transaction
|
||||||
|
# It is used to display an activity history entry in the QML UI
|
||||||
|
QtObject:
|
||||||
|
type
|
||||||
|
ActivityEntry* = ref object of QObject
|
||||||
|
multi_transaction: MultiTransactionDto
|
||||||
|
transaction: ref Item
|
||||||
|
isPending: bool
|
||||||
|
|
||||||
|
proc setup(self: ActivityEntry) =
|
||||||
|
self.QObject.setup
|
||||||
|
|
||||||
|
proc delete*(self: ActivityEntry) =
|
||||||
|
self.QObject.delete
|
||||||
|
|
||||||
|
proc newMultiTransactionActivityEntry*(mt: MultiTransactionDto): ActivityEntry =
|
||||||
|
new(result, delete)
|
||||||
|
result.multi_transaction = mt
|
||||||
|
result.transaction = nil
|
||||||
|
result.isPending = false
|
||||||
|
result.setup()
|
||||||
|
|
||||||
|
proc newTransactionActivityEntry*(tr: ref Item, isPending: bool): ActivityEntry =
|
||||||
|
new(result, delete)
|
||||||
|
result.multi_transaction = nil
|
||||||
|
result.transaction = tr
|
||||||
|
result.isPending = isPending
|
||||||
|
result.setup()
|
||||||
|
|
||||||
|
proc isMultiTransaction*(self: ActivityEntry): bool {.slot.} =
|
||||||
|
return self.multi_transaction != nil
|
||||||
|
|
||||||
|
QtProperty[bool] isMultiTransaction:
|
||||||
|
read = isMultiTransaction
|
||||||
|
|
||||||
|
proc isPendingTransaction*(self: ActivityEntry): bool {.slot.} =
|
||||||
|
return (not self.isMultiTransaction()) and self.isPending
|
||||||
|
|
||||||
|
QtProperty[bool] isPendingTransaction:
|
||||||
|
read = isPendingTransaction
|
||||||
|
|
||||||
|
proc `$`*(self: ActivityEntry): string =
|
||||||
|
let mtStr = if self.multi_transaction != nil: $(self.multi_transaction.id) else: "0"
|
||||||
|
let trStr = if self.transaction != nil: $(self.transaction[]) else: "nil"
|
||||||
|
|
||||||
|
return fmt"""ActivityEntry(
|
||||||
|
multi_transaction.id:{mtStr},
|
||||||
|
transaction:{trStr},
|
||||||
|
isPending:{self.isPending}
|
||||||
|
)"""
|
||||||
|
|
||||||
|
proc getMultiTransaction*(self: ActivityEntry): MultiTransactionDto =
|
||||||
|
if not self.isMultiTransaction():
|
||||||
|
raise newException(Defect, "ActivityEntry is not a MultiTransaction")
|
||||||
|
return self.multi_transaction
|
||||||
|
|
||||||
|
proc getTransaction*(self: ActivityEntry, pending: bool): ref Item =
|
||||||
|
if self.isMultiTransaction() or self.isPending != pending:
|
||||||
|
raise newException(Defect, "ActivityEntry is not a " & (if pending: "pending" else: "") & " Transaction")
|
||||||
|
return self.transaction
|
||||||
|
|
||||||
|
proc getSender*(self: ActivityEntry): string {.slot.} =
|
||||||
|
# TODO: lookup sender's name from addressbook and cache it or in advance
|
||||||
|
if self.isMultiTransaction():
|
||||||
|
return self.multi_transaction.fromAddress
|
||||||
|
|
||||||
|
return self.transaction[].getfrom()
|
||||||
|
|
||||||
|
QtProperty[string] sender:
|
||||||
|
read = getSender
|
||||||
|
|
||||||
|
proc getRecipient*(self: ActivityEntry): string {.slot.} =
|
||||||
|
# TODO: lookup recipient name from addressbook and cache it or in advance
|
||||||
|
if self.isMultiTransaction():
|
||||||
|
return self.multi_transaction.toAddress
|
||||||
|
|
||||||
|
return self.transaction[].getTo()
|
||||||
|
|
||||||
|
QtProperty[string] recipient:
|
||||||
|
read = getRecipient
|
||||||
|
|
||||||
|
# TODO: use CurrencyAmount?
|
||||||
|
proc getFromAmount*(self: ActivityEntry): string {.slot.} =
|
||||||
|
if self.isMultiTransaction():
|
||||||
|
return self.multi_transaction.fromAmount
|
||||||
|
error "getFromAmount: ActivityEntry is not a MultiTransaction"
|
||||||
|
return "0"
|
||||||
|
|
||||||
|
QtProperty[string] fromAmount:
|
||||||
|
read = getFromAmount
|
||||||
|
|
||||||
|
proc getToAmount*(self: ActivityEntry): string {.slot.} =
|
||||||
|
if not self.isMultiTransaction():
|
||||||
|
error "getToAmount: ActivityEntry is not a MultiTransaction"
|
||||||
|
return "0"
|
||||||
|
|
||||||
|
return self.multi_transaction.fromAmount
|
||||||
|
|
||||||
|
QtProperty[string] toAmount:
|
||||||
|
read = getToAmount
|
||||||
|
|
||||||
|
proc getAmount*(self: ActivityEntry): QVariant {.slot.} =
|
||||||
|
if not self.isMultiTransaction():
|
||||||
|
error "getAmount: ActivityEntry is not an transaction.Item"
|
||||||
|
return newQVariant(0)
|
||||||
|
|
||||||
|
return newQVariant(self.transaction[].getValue())
|
||||||
|
|
||||||
|
QtProperty[QVariant] amount:
|
||||||
|
read = getAmount
|
||||||
|
|
||||||
|
proc getTimestamp*(self: ActivityEntry): int {.slot.} =
|
||||||
|
if self.isMultiTransaction():
|
||||||
|
return self.multi_transaction.timestamp
|
||||||
|
# TODO: should we account for self.transaction[].isTimeStamp?
|
||||||
|
return self.transaction[].getTimestamp()
|
||||||
|
|
||||||
|
QtProperty[int] timestamp:
|
||||||
|
read = getTimestamp
|
||||||
|
|
||||||
|
# TODO: properties - type, fromChains, toChains, fromAsset, toAsset, assetName
|
|
@ -0,0 +1,90 @@
|
||||||
|
import NimQml, Tables, strutils, strformat, sequtils
|
||||||
|
|
||||||
|
import ./entry
|
||||||
|
|
||||||
|
# TODO - DEV: remove this
|
||||||
|
import app_service/service/transaction/dto
|
||||||
|
import app/modules/shared_models/currency_amount
|
||||||
|
import ../transactions/item as transaction
|
||||||
|
|
||||||
|
type
|
||||||
|
ModelRole {.pure.} = enum
|
||||||
|
ActivityEntryRole = UserRole + 1
|
||||||
|
|
||||||
|
QtObject:
|
||||||
|
type
|
||||||
|
Model* = ref object of QAbstractListModel
|
||||||
|
entries: seq[ActivityEntry]
|
||||||
|
hasMore: bool
|
||||||
|
|
||||||
|
proc delete(self: Model) =
|
||||||
|
self.entries = @[]
|
||||||
|
self.QAbstractListModel.delete
|
||||||
|
|
||||||
|
proc setup(self: Model) =
|
||||||
|
self.QAbstractListModel.setup
|
||||||
|
|
||||||
|
proc newModel*(): Model =
|
||||||
|
new(result, delete)
|
||||||
|
result.entries = @[]
|
||||||
|
result.setup
|
||||||
|
result.hasMore = true
|
||||||
|
|
||||||
|
proc `$`*(self: Model): string =
|
||||||
|
for i in 0 ..< self.entries.len:
|
||||||
|
result &= fmt"""[{i}]:({$self.entries[i]})"""
|
||||||
|
|
||||||
|
proc countChanged(self: Model) {.signal.}
|
||||||
|
|
||||||
|
proc getCount*(self: Model): int {.slot.} =
|
||||||
|
self.entries.len
|
||||||
|
|
||||||
|
QtProperty[int] count:
|
||||||
|
read = getCount
|
||||||
|
notify = countChanged
|
||||||
|
|
||||||
|
method rowCount(self: Model, index: QModelIndex = nil): int =
|
||||||
|
return self.entries.len
|
||||||
|
|
||||||
|
method roleNames(self: Model): Table[int, string] =
|
||||||
|
{
|
||||||
|
ModelRole.ActivityEntryRole.int:"activityEntry",
|
||||||
|
}.toTable
|
||||||
|
|
||||||
|
method data(self: Model, index: QModelIndex, role: int): QVariant =
|
||||||
|
if (not index.isValid):
|
||||||
|
return
|
||||||
|
|
||||||
|
if (index.row < 0 or index.row >= self.entries.len):
|
||||||
|
return
|
||||||
|
|
||||||
|
let entry = self.entries[index.row]
|
||||||
|
let enumRole = role.ModelRole
|
||||||
|
|
||||||
|
case enumRole:
|
||||||
|
of ModelRole.ActivityEntryRole:
|
||||||
|
result = newQVariant(entry)
|
||||||
|
|
||||||
|
proc setEntries*(self: Model, entries: seq[ActivityEntry]) =
|
||||||
|
self.beginResetModel()
|
||||||
|
self.entries = entries
|
||||||
|
self.endResetModel()
|
||||||
|
self.countChanged()
|
||||||
|
|
||||||
|
# TODO: update data
|
||||||
|
|
||||||
|
# TODO: fetch more
|
||||||
|
|
||||||
|
proc hasMoreChanged*(self: Model) {.signal.}
|
||||||
|
|
||||||
|
proc getHasMore*(self: Model): bool {.slot.} =
|
||||||
|
return self.hasMore
|
||||||
|
|
||||||
|
proc setHasMore*(self: Model, hasMore: bool) {.slot.} =
|
||||||
|
self.hasMore = hasMore
|
||||||
|
self.hasMoreChanged()
|
||||||
|
|
||||||
|
QtProperty[bool] hasMore:
|
||||||
|
read = getHasMore
|
||||||
|
write = setHasMore
|
||||||
|
notify = hasMoreChanged
|
|
@ -16,6 +16,8 @@ import ./networks/module as networks_module
|
||||||
import ./overview/module as overview_module
|
import ./overview/module as overview_module
|
||||||
import ./send/module as send_module
|
import ./send/module as send_module
|
||||||
|
|
||||||
|
import ./activity/controller as activityc
|
||||||
|
|
||||||
import ../../../global/global_singleton
|
import ../../../global/global_singleton
|
||||||
import ../../../core/eventemitter
|
import ../../../core/eventemitter
|
||||||
import ../../../../app_service/service/keycard/service as keycard_service
|
import ../../../../app_service/service/keycard/service as keycard_service
|
||||||
|
@ -34,7 +36,6 @@ import ../../../../app_service/service/network_connection/service as network_con
|
||||||
logScope:
|
logScope:
|
||||||
topics = "wallet-section-module"
|
topics = "wallet-section-module"
|
||||||
|
|
||||||
import io_interface
|
|
||||||
export io_interface
|
export io_interface
|
||||||
|
|
||||||
type
|
type
|
||||||
|
@ -42,7 +43,7 @@ type
|
||||||
delegate: delegate_interface.AccessInterface
|
delegate: delegate_interface.AccessInterface
|
||||||
events: EventEmitter
|
events: EventEmitter
|
||||||
moduleLoaded: bool
|
moduleLoaded: bool
|
||||||
controller: Controller
|
controller: controller.Controller
|
||||||
view: View
|
view: View
|
||||||
filter: Filter
|
filter: Filter
|
||||||
|
|
||||||
|
@ -61,6 +62,8 @@ type
|
||||||
accountsService: accounts_service.Service
|
accountsService: accounts_service.Service
|
||||||
walletAccountService: wallet_account_service.Service
|
walletAccountService: wallet_account_service.Service
|
||||||
|
|
||||||
|
activityController: activityc.Controller
|
||||||
|
|
||||||
proc newModule*(
|
proc newModule*(
|
||||||
delegate: delegate_interface.AccessInterface,
|
delegate: delegate_interface.AccessInterface,
|
||||||
events: EventEmitter,
|
events: EventEmitter,
|
||||||
|
@ -85,7 +88,6 @@ proc newModule*(
|
||||||
result.walletAccountService = walletAccountService
|
result.walletAccountService = walletAccountService
|
||||||
result.moduleLoaded = false
|
result.moduleLoaded = false
|
||||||
result.controller = newController(result, settingsService, walletAccountService, currencyService, networkService)
|
result.controller = newController(result, settingsService, walletAccountService, currencyService, networkService)
|
||||||
result.view = newView(result)
|
|
||||||
|
|
||||||
result.accountsModule = accounts_module.newModule(result, events, walletAccountService, networkService, currencyService)
|
result.accountsModule = accounts_module.newModule(result, events, walletAccountService, networkService, currencyService)
|
||||||
result.allTokensModule = all_tokens_module.newModule(result, events, tokenService, walletAccountService)
|
result.allTokensModule = all_tokens_module.newModule(result, events, tokenService, walletAccountService)
|
||||||
|
@ -99,6 +101,10 @@ proc newModule*(
|
||||||
result.networksModule = networks_module.newModule(result, events, networkService, walletAccountService, settingsService)
|
result.networksModule = networks_module.newModule(result, events, networkService, walletAccountService, settingsService)
|
||||||
result.filter = initFilter(result.controller)
|
result.filter = initFilter(result.controller)
|
||||||
|
|
||||||
|
result.activityController = activityc.newController(result.transactionsModule)
|
||||||
|
result.view = newView(result, result.activityController)
|
||||||
|
|
||||||
|
|
||||||
method delete*(self: Module) =
|
method delete*(self: Module) =
|
||||||
self.accountsModule.delete
|
self.accountsModule.delete
|
||||||
self.allTokensModule.delete
|
self.allTokensModule.delete
|
||||||
|
@ -110,6 +116,7 @@ method delete*(self: Module) =
|
||||||
self.sendModule.delete
|
self.sendModule.delete
|
||||||
self.controller.delete
|
self.controller.delete
|
||||||
self.view.delete
|
self.view.delete
|
||||||
|
self.activityController.delete
|
||||||
|
|
||||||
if not self.addAccountModule.isNil:
|
if not self.addAccountModule.isNil:
|
||||||
self.addAccountModule.delete
|
self.addAccountModule.delete
|
||||||
|
|
|
@ -9,6 +9,7 @@ import ../../../shared_modules/keycard_popup/io_interface as keycard_shared_modu
|
||||||
|
|
||||||
import ../../../../core/[main]
|
import ../../../../core/[main]
|
||||||
import ../../../../core/tasks/[qt, threadpool]
|
import ../../../../core/tasks/[qt, threadpool]
|
||||||
|
import ./backend/transactions
|
||||||
|
|
||||||
type
|
type
|
||||||
Controller* = ref object of RootObj
|
Controller* = ref object of RootObj
|
||||||
|
@ -122,4 +123,4 @@ proc findTokenSymbolByAddress*(self: Controller, address: string): string =
|
||||||
return self.walletAccountService.findTokenSymbolByAddress(address)
|
return self.walletAccountService.findTokenSymbolByAddress(address)
|
||||||
|
|
||||||
proc getMultiTransactions*(self: Controller, transactionIDs: seq[int]): seq[MultiTransactionDto] =
|
proc getMultiTransactions*(self: Controller, transactionIDs: seq[int]): seq[MultiTransactionDto] =
|
||||||
return self.transactionService.getMultiTransactions(transactionIDs)
|
return transaction_service.getMultiTransactions(transactionIDs)
|
||||||
|
|
|
@ -3,6 +3,8 @@ import ../../../../../app_service/service/collectible/dto as CollectibleDto
|
||||||
import ../../../../../app_service/service/transaction/dto
|
import ../../../../../app_service/service/transaction/dto
|
||||||
export TransactionDto, CollectibleDto
|
export TransactionDto, CollectibleDto
|
||||||
|
|
||||||
|
import ./item
|
||||||
|
|
||||||
type
|
type
|
||||||
AccessInterface* {.pure inheritable.} = ref object of RootObj
|
AccessInterface* {.pure inheritable.} = ref object of RootObj
|
||||||
## Abstract class for any input/interaction with this module.
|
## Abstract class for any input/interaction with this module.
|
||||||
|
@ -43,10 +45,10 @@ method setIsNonArchivalNode*(self: AccessInterface, isNonArchivalNode: bool) {.b
|
||||||
method transactionWasSent*(self: AccessInterface, result: string) {.base.} =
|
method transactionWasSent*(self: AccessInterface, result: string) {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method getChainIdForChat*(self: AccessInterface): int =
|
method getChainIdForChat*(self: AccessInterface): int {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method getChainIdForBrowser*(self: AccessInterface): int =
|
method getChainIdForBrowser*(self: AccessInterface): int {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
method refreshTransactions*(self: AccessInterface) {.base.} =
|
method refreshTransactions*(self: AccessInterface) {.base.} =
|
||||||
|
@ -60,3 +62,6 @@ method viewDidLoad*(self: AccessInterface) {.base.} =
|
||||||
|
|
||||||
method getLastTxBlockNumber*(self: AccessInterface): string {.base.} =
|
method getLastTxBlockNumber*(self: AccessInterface): string {.base.} =
|
||||||
raise newException(ValueError, "No implementation available")
|
raise newException(ValueError, "No implementation available")
|
||||||
|
|
||||||
|
method transactionsToItems*(self: AccessInterface, transactions: seq[TransactionDto], collectibles: seq[CollectibleDto]): seq[Item] {.base.} =
|
||||||
|
raise newException(ValueError, "No implementation available")
|
|
@ -240,6 +240,7 @@ proc getTxStatus*(self: Item): string =
|
||||||
proc getValue*(self: Item): CurrencyAmount =
|
proc getValue*(self: Item): CurrencyAmount =
|
||||||
return self.value
|
return self.value
|
||||||
|
|
||||||
|
# TODO: fix naming
|
||||||
proc getfrom*(self: Item): string =
|
proc getfrom*(self: Item): string =
|
||||||
return self.fro
|
return self.fro
|
||||||
|
|
||||||
|
|
|
@ -59,26 +59,10 @@ proc getResolvedSymbol*(self: Module, transaction: TransactionDto): string =
|
||||||
else:
|
else:
|
||||||
result = "ETH"
|
result = "ETH"
|
||||||
|
|
||||||
proc transactionsToItems(self: Module, transactions: seq[TransactionDto], collectibles: seq[CollectibleDto]) : seq[Item] =
|
method transactionsToItems*(self: Module, transactions: seq[TransactionDto], collectibles: seq[CollectibleDto]): seq[Item] =
|
||||||
let gweiFormat = self.controller.getCurrencyFormat("Gwei")
|
let gweiFormat = self.controller.getCurrencyFormat("Gwei")
|
||||||
let ethFormat = self.controller.getCurrencyFormat("ETH")
|
let ethFormat = self.controller.getCurrencyFormat("ETH")
|
||||||
|
|
||||||
# TODO: Continue merging multi-transactions with transactions
|
|
||||||
#
|
|
||||||
# let transactionIDs = transactions.filter(t => t.multiTransactionID != MultiTransactionMissingID).map(t => t.multiTransactionID)
|
|
||||||
# let multiTransactions = self.controller.getMultiTransactions(transactionIDs)
|
|
||||||
# for mt in multiTransactions:
|
|
||||||
# let mtItem = multiTransactionToItem(mt)
|
|
||||||
#
|
|
||||||
# Tip: depending of the new design best will be to replace the transaction.View
|
|
||||||
# with a new ActivityEntry that contains eighter a transaction or a multi-transaction
|
|
||||||
# Refactor transaction Model to serve ActivityEntry istead of Views
|
|
||||||
#
|
|
||||||
# Here we should filter all transactions that are part of a multi-transaciton
|
|
||||||
# and add them to the multi-transaction View associated with an ActivityEntry
|
|
||||||
# and the remaining "free" transactions to the corresponding ActivityEntry
|
|
||||||
# TODO: check TransactionsItem changes
|
|
||||||
|
|
||||||
transactions.map(t => (block:
|
transactions.map(t => (block:
|
||||||
if t.typeValue == ERC721_TRANSACTION_TYPE:
|
if t.typeValue == ERC721_TRANSACTION_TYPE:
|
||||||
for c in collectibles:
|
for c in collectibles:
|
||||||
|
|
|
@ -1,9 +1,10 @@
|
||||||
import strformat
|
import strformat
|
||||||
|
|
||||||
import ../../../../../app_service/service/transaction/dto
|
import ./backend/transactions
|
||||||
|
|
||||||
const MultiTransactionMissingID* = 0
|
const MultiTransactionMissingID* = 0
|
||||||
|
|
||||||
|
# TODO: make it a Qt object to be referenced in QML via ActivityView
|
||||||
type
|
type
|
||||||
MultiTransactionItem* = object
|
MultiTransactionItem* = object
|
||||||
id: int
|
id: int
|
||||||
|
|
|
@ -10,6 +10,8 @@ import ../../../shared_models/currency_amount
|
||||||
import ./item
|
import ./item
|
||||||
import ./multi_transaction_item
|
import ./multi_transaction_item
|
||||||
|
|
||||||
|
import ./backend/transactions
|
||||||
|
|
||||||
proc hex2GweiCurrencyAmount(hexValueStr: string, gweiFormat: CurrencyFormatDto): CurrencyAmount =
|
proc hex2GweiCurrencyAmount(hexValueStr: string, gweiFormat: CurrencyFormatDto): CurrencyAmount =
|
||||||
let value = parseFloat(singletonInstance.utils.hex2Gwei(hexValueStr))
|
let value = parseFloat(singletonInstance.utils.hex2Gwei(hexValueStr))
|
||||||
return currencyAmountToItem(value, gweiFormat)
|
return currencyAmountToItem(value, gweiFormat)
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import NimQml, json
|
import NimQml, json
|
||||||
|
|
||||||
|
import ./activity/controller as activityc
|
||||||
import ./io_interface
|
import ./io_interface
|
||||||
import ../../shared_models/currency_amount
|
import ../../shared_models/currency_amount
|
||||||
|
|
||||||
|
@ -13,6 +14,7 @@ QtObject:
|
||||||
isMnemonicBackedUp: bool
|
isMnemonicBackedUp: bool
|
||||||
tmpAmount: float # shouldn't be used anywhere except in prepare*/getPrepared* procs
|
tmpAmount: float # shouldn't be used anywhere except in prepare*/getPrepared* procs
|
||||||
tmpSymbol: string # shouldn't be used anywhere except in prepare*/getPrepared* procs
|
tmpSymbol: string # shouldn't be used anywhere except in prepare*/getPrepared* procs
|
||||||
|
activityController: activityc.Controller
|
||||||
|
|
||||||
proc setup(self: View) =
|
proc setup(self: View) =
|
||||||
self.QObject.setup
|
self.QObject.setup
|
||||||
|
@ -20,9 +22,10 @@ QtObject:
|
||||||
proc delete*(self: View) =
|
proc delete*(self: View) =
|
||||||
self.QObject.delete
|
self.QObject.delete
|
||||||
|
|
||||||
proc newView*(delegate: io_interface.AccessInterface): View =
|
proc newView*(delegate: io_interface.AccessInterface, activityController: activityc.Controller): View =
|
||||||
new(result, delete)
|
new(result, delete)
|
||||||
result.delegate = delegate
|
result.delegate = delegate
|
||||||
|
result.activityController = activityController
|
||||||
result.setup()
|
result.setup()
|
||||||
|
|
||||||
proc load*(self: View) =
|
proc load*(self: View) =
|
||||||
|
@ -121,3 +124,8 @@ QtObject:
|
||||||
proc destroyAddAccountPopup*(self: View) {.signal.}
|
proc destroyAddAccountPopup*(self: View) {.signal.}
|
||||||
proc emitDestroyAddAccountPopup*(self: View) =
|
proc emitDestroyAddAccountPopup*(self: View) =
|
||||||
self.destroyAddAccountPopup()
|
self.destroyAddAccountPopup()
|
||||||
|
|
||||||
|
proc getActivityController(self: View): QVariant {.slot.} =
|
||||||
|
return newQVariant(self.activityController)
|
||||||
|
QtProperty[QVariant] activityController:
|
||||||
|
read = getActivityController
|
||||||
|
|
|
@ -1,12 +1,14 @@
|
||||||
import json, strutils, stint, json_serialization, strformat
|
import json, strutils, stint, json_serialization, strformat
|
||||||
|
|
||||||
import
|
import
|
||||||
web3/ethtypes, json_serialization
|
web3/ethtypes
|
||||||
|
|
||||||
include ../../common/json_utils
|
include ../../common/json_utils
|
||||||
import ../network/dto
|
import ../network/dto
|
||||||
import ../../common/conversion as service_conversion
|
import ../../common/conversion as service_conversion
|
||||||
|
|
||||||
|
import ./backend/transactions
|
||||||
|
|
||||||
type
|
type
|
||||||
PendingTransactionTypeDto* {.pure.} = enum
|
PendingTransactionTypeDto* {.pure.} = enum
|
||||||
RegisterENS = "RegisterENS",
|
RegisterENS = "RegisterENS",
|
||||||
|
@ -20,20 +22,6 @@ type
|
||||||
proc event*(self:PendingTransactionTypeDto):string =
|
proc event*(self:PendingTransactionTypeDto):string =
|
||||||
result = "transaction:" & $self
|
result = "transaction:" & $self
|
||||||
|
|
||||||
type
|
|
||||||
MultiTransactionType* = enum
|
|
||||||
MultiTransactionSend = 0, MultiTransactionSwap = 1, MultiTransactionBridge = 2
|
|
||||||
|
|
||||||
type MultiTransactionDto* = ref object of RootObj
|
|
||||||
id* {.serializedFieldName("id").}: int
|
|
||||||
timestamp* {.serializedFieldName("timestamp").}: int
|
|
||||||
fromAddress* {.serializedFieldName("fromAddress").}: string
|
|
||||||
toAddress* {.serializedFieldName("toAddress").}: string
|
|
||||||
fromAsset* {.serializedFieldName("fromAsset").}: string
|
|
||||||
toAsset* {.serializedFieldName("toAsset").}: string
|
|
||||||
fromAmount* {.serializedFieldName("fromAmount").}: string
|
|
||||||
multiTxtype* {.serializedFieldName("type").}: MultiTransactionType
|
|
||||||
|
|
||||||
type
|
type
|
||||||
TransactionDto* = ref object of RootObj
|
TransactionDto* = ref object of RootObj
|
||||||
id*: string
|
id*: string
|
||||||
|
|
|
@ -18,7 +18,7 @@ import ../token/service as token_service
|
||||||
import ../settings/service as settings_service
|
import ../settings/service as settings_service
|
||||||
import ../collectible/dto
|
import ../collectible/dto
|
||||||
import ../eth/dto/transaction as transaction_data_dto
|
import ../eth/dto/transaction as transaction_data_dto
|
||||||
import ../eth/dto/[method_dto, coder, method_dto]
|
import ../eth/dto/[coder, method_dto]
|
||||||
import ./dto as transaction_dto
|
import ./dto as transaction_dto
|
||||||
import ./cryptoRampDto
|
import ./cryptoRampDto
|
||||||
import ../eth/utils as eth_utils
|
import ../eth/utils as eth_utils
|
||||||
|
@ -158,16 +158,6 @@ QtObject:
|
||||||
error "error: ", errDescription
|
error "error: ", errDescription
|
||||||
return
|
return
|
||||||
|
|
||||||
proc getMultiTransactions*(self: Service, transactionIDs: seq[int]): seq[MultiTransactionDto] =
|
|
||||||
try:
|
|
||||||
let response = transactions.getMultiTransactions(transactionIDs).result
|
|
||||||
|
|
||||||
return response.getElems().map(x => x.toMultiTransactionDto())
|
|
||||||
except Exception as e:
|
|
||||||
let errDescription = e.msg
|
|
||||||
error "error: ", errDescription
|
|
||||||
return
|
|
||||||
|
|
||||||
proc getAllTransactions*(self: Service, address: string): seq[TransactionDto] =
|
proc getAllTransactions*(self: Service, address: string): seq[TransactionDto] =
|
||||||
if not self.allTransactions.hasKey(address):
|
if not self.allTransactions.hasKey(address):
|
||||||
return @[]
|
return @[]
|
||||||
|
@ -375,7 +365,7 @@ QtObject:
|
||||||
fromAsset: tokenSymbol,
|
fromAsset: tokenSymbol,
|
||||||
toAsset: tokenSymbol,
|
toAsset: tokenSymbol,
|
||||||
fromAmount: "0x" & amountToSend.toHex,
|
fromAmount: "0x" & amountToSend.toHex,
|
||||||
multiTxtype: MultiTransactionType.MultiTransactionSend,
|
multiTxtype: transactions.MultiTransactionType.MultiTransactionSend,
|
||||||
),
|
),
|
||||||
paths,
|
paths,
|
||||||
password,
|
password,
|
||||||
|
@ -442,7 +432,7 @@ QtObject:
|
||||||
fromAsset: tokenSymbol,
|
fromAsset: tokenSymbol,
|
||||||
toAsset: tokenSymbol,
|
toAsset: tokenSymbol,
|
||||||
fromAmount: "0x" & amountToSend.toHex,
|
fromAmount: "0x" & amountToSend.toHex,
|
||||||
multiTxtype: MultiTransactionType.MultiTransactionSend,
|
multiTxtype: transactions.MultiTransactionType.MultiTransactionSend,
|
||||||
),
|
),
|
||||||
paths,
|
paths,
|
||||||
password,
|
password,
|
||||||
|
@ -547,3 +537,13 @@ QtObject:
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
error "Error getting latest block number", message = e.msg
|
error "Error getting latest block number", message = e.msg
|
||||||
return ""
|
return ""
|
||||||
|
|
||||||
|
proc getMultiTransactions*(transactionIDs: seq[int]): seq[MultiTransactionDto] =
|
||||||
|
try:
|
||||||
|
let response = transactions.getMultiTransactions(transactionIDs).result
|
||||||
|
|
||||||
|
return response.getElems().map(x => x.toMultiTransactionDto())
|
||||||
|
except Exception as e:
|
||||||
|
let errDescription = e.msg
|
||||||
|
error "error: ", errDescription
|
||||||
|
return
|
||||||
|
|
|
@ -0,0 +1,126 @@
|
||||||
|
import times, strformat
|
||||||
|
import json, json_serialization
|
||||||
|
import options
|
||||||
|
import ./core, ./response_type
|
||||||
|
from ./gen import rpc
|
||||||
|
import ./backend
|
||||||
|
import transactions
|
||||||
|
|
||||||
|
export response_type
|
||||||
|
|
||||||
|
# TODO: consider using common status-go types via protobuf
|
||||||
|
# TODO: consider using flags instead of list of enums
|
||||||
|
type
|
||||||
|
Period* = object
|
||||||
|
startTimestamp*: int
|
||||||
|
endTimestamp*: int
|
||||||
|
|
||||||
|
# see status-go/services/wallet/activity/filter.go Type
|
||||||
|
ActivityType* {.pure.} = enum
|
||||||
|
Send, Receive, Buy, Swap, Bridge
|
||||||
|
|
||||||
|
# see status-go/services/wallet/activity/filter.go Status
|
||||||
|
ActivityStatus* {.pure.} = enum
|
||||||
|
Failed, Pending, Complete, Finalized
|
||||||
|
|
||||||
|
# see status-go/services/wallet/activity/filter.go TokenType
|
||||||
|
TokenType* {.pure.} = enum
|
||||||
|
Asset, Collectibles
|
||||||
|
|
||||||
|
# see status-go/services/wallet/activity/filter.go Filter
|
||||||
|
ActivityFilter* = object
|
||||||
|
period* {.serializedFieldName("period").}: Period
|
||||||
|
types* {.serializedFieldName("types").}: seq[ActivityType]
|
||||||
|
statuses* {.serializedFieldName("statuses").}: seq[ActivityStatus]
|
||||||
|
tokenTypes* {.serializedFieldName("tokenTypes").}: seq[TokenType]
|
||||||
|
counterpartyAddresses* {.serializedFieldName("counterpartyAddresses").}: seq[string]
|
||||||
|
|
||||||
|
proc newPeriod*(startTime: Option[DateTime], endTime: Option[DateTime]): Period =
|
||||||
|
if startTime.isSome:
|
||||||
|
result.startTimestamp = startTime.get().toTime().toUnix().int
|
||||||
|
else:
|
||||||
|
result.startTimestamp = 0
|
||||||
|
if endTime.isSome:
|
||||||
|
result.endTimestamp = endTime.get().toTime().toUnix().int
|
||||||
|
else:
|
||||||
|
result.endTimestamp = 0
|
||||||
|
|
||||||
|
proc newPeriod*(startTimestamp: int, endTimestamp: int): Period =
|
||||||
|
result.startTimestamp = startTimestamp
|
||||||
|
result.endTimestamp = endTimestamp
|
||||||
|
|
||||||
|
proc getIncludeAllActivityFilter*(): ActivityFilter =
|
||||||
|
result = ActivityFilter(period: newPeriod(none(DateTime), none(DateTime)), types: @[], statuses: @[], tokenTypes: @[], counterpartyAddresses: @[])
|
||||||
|
|
||||||
|
# Empty sequence for paramters means include all
|
||||||
|
proc newActivityFilter*(period: Period, activityType: seq[ActivityType], activityStatus: seq[ActivityStatus], tokenType: seq[TokenType], counterpartyAddress: seq[string]): ActivityFilter =
|
||||||
|
result.period = period
|
||||||
|
result.types = activityType
|
||||||
|
result.statuses = activityStatus
|
||||||
|
result.tokenTypes = tokenType
|
||||||
|
result.counterpartyAddresses = counterpartyAddress
|
||||||
|
|
||||||
|
# Mirrors status-go/services/wallet/activity/activity.go PayloadType
|
||||||
|
type
|
||||||
|
PayloadType* {.pure.} = enum
|
||||||
|
MultiTransaction = 1
|
||||||
|
SimpleTransaction
|
||||||
|
PendingTransaction
|
||||||
|
|
||||||
|
# Define toJson proc for PayloadType
|
||||||
|
proc toJson*(x: PayloadType): JsonNode {.inline.} =
|
||||||
|
return %*(ord(x))
|
||||||
|
|
||||||
|
# Define fromJson proc for PayloadType
|
||||||
|
proc fromJson*(x: JsonNode, T: typedesc[PayloadType]): PayloadType {.inline.} =
|
||||||
|
return cast[PayloadType](x.getInt())
|
||||||
|
|
||||||
|
# TODO: hide internals behind safe interface
|
||||||
|
type
|
||||||
|
ActivityEntry* = object
|
||||||
|
transactionType* {.serializedFieldName("transactionType").}: PayloadType
|
||||||
|
transaction* {.serializedFieldName("transaction").}: Option[TransactionIdentity]
|
||||||
|
id* {.serializedFieldName("id").}: int
|
||||||
|
timestamp* {.serializedFieldName("timestamp").}: int
|
||||||
|
activityType* {.serializedFieldName("activityType").}: MultiTransactionType
|
||||||
|
|
||||||
|
proc fromJson[T](jsonObj: JsonNode, TT: typedesc[Option[T]]): Option[T] =
|
||||||
|
if jsonObj.kind != JNull:
|
||||||
|
return some(to(jsonObj, T))
|
||||||
|
else:
|
||||||
|
return none(T)
|
||||||
|
|
||||||
|
proc toJson[T](obj: Option[T]): JsonNode =
|
||||||
|
if obj.isSome:
|
||||||
|
toJson(obj.get())
|
||||||
|
else:
|
||||||
|
newJNull()
|
||||||
|
|
||||||
|
# Define toJson proc for PayloadType
|
||||||
|
proc toJson*(ae: ActivityEntry): JsonNode {.inline.} =
|
||||||
|
return %*(ae)
|
||||||
|
|
||||||
|
# Define fromJson proc for PayloadType
|
||||||
|
proc fromJson*(e: JsonNode, T: typedesc[ActivityEntry]): ActivityEntry {.inline.} =
|
||||||
|
result = T(
|
||||||
|
transactionType: fromJson(e["transactionType"], PayloadType),
|
||||||
|
transaction: if e.hasKey("transaction"): fromJson(e["transaction"], Option[TransactionIdentity]) else: none(TransactionIdentity),
|
||||||
|
id: e["id"].getInt(),
|
||||||
|
timestamp: e["timestamp"].getInt()
|
||||||
|
)
|
||||||
|
|
||||||
|
proc `$`*(self: ActivityEntry): string =
|
||||||
|
let transactionStr = if self.transaction.isSome: $self.transaction.get() else: "none(TransactionIdentity)"
|
||||||
|
return fmt"""ActivityEntry(
|
||||||
|
transactionType:{self.transactionType.int},
|
||||||
|
transaction:{transactionStr},
|
||||||
|
id:{self.id},
|
||||||
|
timestamp:{self.timestamp},
|
||||||
|
)"""
|
||||||
|
|
||||||
|
rpc(getActivityEntries, "wallet"):
|
||||||
|
addresses: seq[string]
|
||||||
|
chainIds: seq[int]
|
||||||
|
filter: ActivityFilter
|
||||||
|
offset: int
|
||||||
|
limit: int
|
|
@ -1,4 +1,4 @@
|
||||||
import json, json_serialization
|
import json, json_serialization, strformat
|
||||||
import ./core, ./response_type
|
import ./core, ./response_type
|
||||||
from ./gen import rpc
|
from ./gen import rpc
|
||||||
|
|
||||||
|
@ -96,6 +96,25 @@ rpc(getTokensBalancesForChainIDs, "wallet"):
|
||||||
rpc(getPendingTransactionsByChainIDs, "wallet"):
|
rpc(getPendingTransactionsByChainIDs, "wallet"):
|
||||||
chainIds: seq[int]
|
chainIds: seq[int]
|
||||||
|
|
||||||
|
type
|
||||||
|
TransactionIdentity* = ref object
|
||||||
|
chainId* {.serializedFieldName("chainId").}: int
|
||||||
|
hash* {.serializedFieldName("hash").}: string
|
||||||
|
address* {.serializedFieldName("address").}: string
|
||||||
|
|
||||||
|
proc `$`*(self: TransactionIdentity): string =
|
||||||
|
return fmt"""TransactionIdentity(
|
||||||
|
chainId:{self.chainId},
|
||||||
|
hash:{self.hash},
|
||||||
|
address:{self.address},
|
||||||
|
)"""
|
||||||
|
|
||||||
|
rpc(getPendingTransactionsForIdentities, "wallet"):
|
||||||
|
identities = seq[TransactionIdentity]
|
||||||
|
|
||||||
|
rpc(getTransfersForIdentities, "wallet"):
|
||||||
|
identities = seq[TransactionIdentity]
|
||||||
|
|
||||||
rpc(getWalletToken, "wallet"):
|
rpc(getWalletToken, "wallet"):
|
||||||
accounts: seq[string]
|
accounts: seq[string]
|
||||||
|
|
||||||
|
|
|
@ -1,10 +1,27 @@
|
||||||
import json, stint
|
import json, stint, json_serialization
|
||||||
|
|
||||||
import ../app_service/service/transaction/dto
|
|
||||||
import ../app_service/service/eth/dto/transaction
|
import ../app_service/service/eth/dto/transaction
|
||||||
import ./core as core
|
import ./core as core
|
||||||
import ../app_service/common/utils
|
import ../app_service/common/utils
|
||||||
|
|
||||||
|
# mirrors the MultiTransactionType from status-go, services/wallet/transfer/transaction.go
|
||||||
|
type
|
||||||
|
MultiTransactionType* = enum
|
||||||
|
MultiTransactionSend = 0, MultiTransactionSwap = 1, MultiTransactionBridge = 2
|
||||||
|
|
||||||
|
MultiTransactionDto* = ref object of RootObj
|
||||||
|
id* {.serializedFieldName("id").}: int
|
||||||
|
timestamp* {.serializedFieldName("timestamp").}: int
|
||||||
|
fromAddress* {.serializedFieldName("fromAddress").}: string
|
||||||
|
toAddress* {.serializedFieldName("toAddress").}: string
|
||||||
|
fromAsset* {.serializedFieldName("fromAsset").}: string
|
||||||
|
toAsset* {.serializedFieldName("toAsset").}: string
|
||||||
|
fromAmount* {.serializedFieldName("fromAmount").}: string
|
||||||
|
multiTxtype* {.serializedFieldName("type").}: MultiTransactionType
|
||||||
|
|
||||||
|
proc getTransactionByHash*(chainId: int, hash: string): RpcResponse[JsonNode] {.raises: [Exception].} =
|
||||||
|
core.callPrivateRPCWithChainId("eth_getTransactionByHash", chainId, %* [hash])
|
||||||
|
|
||||||
proc checkRecentHistory*(chainIds: seq[int], addresses: seq[string]) {.raises: [Exception].} =
|
proc checkRecentHistory*(chainIds: seq[int], addresses: seq[string]) {.raises: [Exception].} =
|
||||||
let payload = %* [chainIds, addresses]
|
let payload = %* [chainIds, addresses]
|
||||||
discard core.callPrivateRPC("wallet_checkRecentHistoryForChainIDs", payload)
|
discard core.callPrivateRPC("wallet_checkRecentHistoryForChainIDs", payload)
|
||||||
|
|
|
@ -29,6 +29,7 @@ QtObject {
|
||||||
|
|
||||||
property var walletSectionInst: walletSection
|
property var walletSectionInst: walletSection
|
||||||
property var totalCurrencyBalance: walletSection.totalCurrencyBalance
|
property var totalCurrencyBalance: walletSection.totalCurrencyBalance
|
||||||
|
property var activityController: walletSection.activityController
|
||||||
property string signingPhrase: walletSection.signingPhrase
|
property string signingPhrase: walletSection.signingPhrase
|
||||||
property string mnemonicBackedUp: walletSection.isMnemonicBackedUp
|
property string mnemonicBackedUp: walletSection.isMnemonicBackedUp
|
||||||
|
|
||||||
|
|
|
@ -83,6 +83,13 @@ Item {
|
||||||
width: implicitWidth
|
width: implicitWidth
|
||||||
text: qsTr("Activity")
|
text: qsTr("Activity")
|
||||||
}
|
}
|
||||||
|
// TODO - DEV: remove me
|
||||||
|
// currentIndex: 3
|
||||||
|
// StatusTabButton {
|
||||||
|
// rightPadding: 0
|
||||||
|
// width: implicitWidth
|
||||||
|
// text: qsTr("DEV activity")
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
StackLayout {
|
StackLayout {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
@ -114,6 +121,14 @@ Item {
|
||||||
stack.currentIndex = 3
|
stack.currentIndex = 3
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// TODO: replace with the real activity view
|
||||||
|
// Enable for debugging activity filter
|
||||||
|
// ActivityView {
|
||||||
|
// Layout.fillWidth: true
|
||||||
|
// Layout.fillHeight: true
|
||||||
|
|
||||||
|
// controller: RootStore.activityController
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
CollectibleDetailView {
|
CollectibleDetailView {
|
||||||
|
|
|
@ -0,0 +1,113 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQml 2.15
|
||||||
|
import QtQuick.Controls 2.15
|
||||||
|
import QtQuick.Layouts 1.15
|
||||||
|
|
||||||
|
import StatusQ.Core 0.1
|
||||||
|
import StatusQ.Components 0.1
|
||||||
|
import StatusQ.Controls 0.1
|
||||||
|
import StatusQ.Core.Theme 0.1
|
||||||
|
|
||||||
|
import SortFilterProxyModel 0.2
|
||||||
|
|
||||||
|
import utils 1.0
|
||||||
|
|
||||||
|
import "../panels"
|
||||||
|
import "../popups"
|
||||||
|
import "../stores"
|
||||||
|
import "../controls"
|
||||||
|
|
||||||
|
// Temporary developer view to test the filter APIs
|
||||||
|
Item {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
property var controller
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
id: filterLayout
|
||||||
|
|
||||||
|
readonly property int millisInADay: 24 * 60 * 60 * 1000
|
||||||
|
property int start: fromSlider.value > 0 ? Math.floor(new Date(new Date() - (fromSlider.value * millisInADay)).getTime() / 1000) : 0
|
||||||
|
property int end: toSlider.value > 0 ? Math.floor(new Date(new Date() - (toSlider.value * millisInADay)).getTime() / 1000) : 0
|
||||||
|
|
||||||
|
function updateFilter() { controller.updateFilter(start, end) }
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
Label { text: "Past Days Span: 100" }
|
||||||
|
Slider {
|
||||||
|
id: fromSlider
|
||||||
|
|
||||||
|
Layout.preferredWidth: 200
|
||||||
|
Layout.preferredHeight: 50
|
||||||
|
|
||||||
|
from: 100
|
||||||
|
to: 0
|
||||||
|
|
||||||
|
stepSize: 1
|
||||||
|
value: 0
|
||||||
|
|
||||||
|
onPressedChanged: { if (!pressed) filterLayout.updateFilter() }
|
||||||
|
}
|
||||||
|
Label { text: `${fromSlider.value}d - ${toSlider.value}d` }
|
||||||
|
Slider {
|
||||||
|
id: toSlider
|
||||||
|
|
||||||
|
Layout.preferredWidth: 200
|
||||||
|
Layout.preferredHeight: 50
|
||||||
|
|
||||||
|
enabled: fromSlider.value > 1
|
||||||
|
|
||||||
|
from: fromSlider.value - 1
|
||||||
|
to: 0
|
||||||
|
|
||||||
|
stepSize: 1
|
||||||
|
value: 0
|
||||||
|
|
||||||
|
onPressedChanged: { if (!pressed) filterLayout.updateFilter() }
|
||||||
|
}
|
||||||
|
Label { text: "0" }
|
||||||
|
}
|
||||||
|
Label { text: `Interval: ${filterLayout.start > 0 ? root.epochToDateStr(filterLayout.start) : "all time"} - ${filterLayout.end > 0 ? root.epochToDateStr(filterLayout.end) : "now"}` }
|
||||||
|
}
|
||||||
|
|
||||||
|
ListView {
|
||||||
|
id: listView
|
||||||
|
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.fillHeight: true
|
||||||
|
|
||||||
|
model: controller.model
|
||||||
|
|
||||||
|
delegate: Item {
|
||||||
|
width: parent ? parent.width : 0
|
||||||
|
height: itemLayout.implicitHeight
|
||||||
|
|
||||||
|
readonly property var entry: model.activityEntry
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
id: itemLayout
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
Label { text: entry.isMultiTransaction ? "MT" : entry.isPendingTransaction ? "PT" : " T" }
|
||||||
|
Label { text: `[${root.epochToDateStr(entry.timestamp)}] ` }
|
||||||
|
Label { text: entry.isMultiTransaction ? entry.fromAmount : entry.amount }
|
||||||
|
Label { text: "from"; Layout.leftMargin: 5; Layout.rightMargin: 5 }
|
||||||
|
Label { text: entry.sender; Layout.maximumWidth: 200; elide: Text.ElideMiddle }
|
||||||
|
Label { text: "to"; Layout.leftMargin: 5; Layout.rightMargin: 5 }
|
||||||
|
Label { text: entry.recipient; Layout.maximumWidth: 200; elide: Text.ElideMiddle }
|
||||||
|
Label { text: "got"; Layout.leftMargin: 5; Layout.rightMargin: 5; visible: entry.isMultiTransaction }
|
||||||
|
Label { text: entry.toAmount; Layout.leftMargin: 5; Layout.rightMargin: 5; visible: entry.isMultiTransaction }
|
||||||
|
RowLayout {} // Spacer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function epochToDateStr(epochTimestamp) {
|
||||||
|
var date = new Date(epochTimestamp * 1000);
|
||||||
|
return date.toLocaleString(Qt.locale(), "dd-MM-yyyy hh:mm");
|
||||||
|
}
|
||||||
|
}
|
|
@ -13,3 +13,4 @@ TokenListView 1.0 TokenListView.qml
|
||||||
TransactionPreview 1.0 TransactionPreview.qml
|
TransactionPreview 1.0 TransactionPreview.qml
|
||||||
TransactionSigner 1.0 TransactionSigner.qml
|
TransactionSigner 1.0 TransactionSigner.qml
|
||||||
TransactionStackView 1.0 TransactionStackView.qml
|
TransactionStackView 1.0 TransactionStackView.qml
|
||||||
|
ActivityView 1.0 ActivityView.qml
|
||||||
|
|
Loading…
Reference in New Issue