mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
Refactor transaction list token filtering (#8669)
The transaction list now filters by token in the `TransactionList` component instead of in the transaction selector. This was done in preparation for the asset page work. Technically this approach is slightly less efficient than before, as we're now filtering the transactions after they've been grouped together rather than beforehand. The difference is minimal though, and this method is more correct. The old filtering was broken because it inappropriately filtered out cancel transactions. Cancel transactions always have the `to` address set to the same as the `from` address, and the token filter only returned transactions where the `to` address was set to the token address. Now that we're only filtering by the `to` address of the initial transaction, token transaction groups will be included in their entirety, including any cancel transactions.
This commit is contained in:
parent
e89540fd94
commit
95a95ee4bc
@ -1,4 +1,5 @@
|
|||||||
import React, { useMemo, useEffect, useRef, useState, useCallback } from 'react'
|
import React, { useMemo, useEffect, useRef, useState, useCallback } from 'react'
|
||||||
|
import PropTypes from 'prop-types'
|
||||||
import { useSelector, useDispatch } from 'react-redux'
|
import { useSelector, useDispatch } from 'react-redux'
|
||||||
import {
|
import {
|
||||||
nonceSortedCompletedTransactionsSelector,
|
nonceSortedCompletedTransactionsSelector,
|
||||||
@ -14,15 +15,38 @@ import Button from '../../ui/button'
|
|||||||
|
|
||||||
const PAGE_INCREMENT = 10
|
const PAGE_INCREMENT = 10
|
||||||
|
|
||||||
export default function TransactionList () {
|
const getTransactionGroupRecipientAddressFilter = (recipientAddress) => {
|
||||||
|
return ({ initialTransaction: { txParams } }) => txParams && txParams.to === recipientAddress
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function TransactionList ({ tokenAddress }) {
|
||||||
const [limit, setLimit] = useState(PAGE_INCREMENT)
|
const [limit, setLimit] = useState(PAGE_INCREMENT)
|
||||||
const t = useI18nContext()
|
const t = useI18nContext()
|
||||||
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const pendingTransactions = useSelector(nonceSortedPendingTransactionsSelector)
|
const unfilteredPendingTransactions = useSelector(nonceSortedPendingTransactionsSelector)
|
||||||
const completedTransactions = useSelector(nonceSortedCompletedTransactionsSelector)
|
const unfilteredCompletedTransactions = useSelector(nonceSortedCompletedTransactionsSelector)
|
||||||
const { transactionTime: transactionTimeFeatureActive } = useSelector(getFeatureFlags)
|
const { transactionTime: transactionTimeFeatureActive } = useSelector(getFeatureFlags)
|
||||||
|
|
||||||
|
const pendingTransactions = useMemo(
|
||||||
|
() => (
|
||||||
|
tokenAddress && tokenAddress.startsWith('0x')
|
||||||
|
? unfilteredPendingTransactions
|
||||||
|
.filter(getTransactionGroupRecipientAddressFilter(tokenAddress))
|
||||||
|
: unfilteredPendingTransactions
|
||||||
|
),
|
||||||
|
[unfilteredPendingTransactions, tokenAddress]
|
||||||
|
)
|
||||||
|
const completedTransactions = useMemo(
|
||||||
|
() => (
|
||||||
|
tokenAddress && tokenAddress.startsWith('0x')
|
||||||
|
? unfilteredCompletedTransactions
|
||||||
|
.filter(getTransactionGroupRecipientAddressFilter(tokenAddress))
|
||||||
|
: unfilteredCompletedTransactions
|
||||||
|
),
|
||||||
|
[unfilteredCompletedTransactions, tokenAddress]
|
||||||
|
)
|
||||||
|
|
||||||
const { fetchGasEstimates, fetchBasicGasAndTimeEstimates } = useMemo(() => ({
|
const { fetchGasEstimates, fetchBasicGasAndTimeEstimates } = useMemo(() => ({
|
||||||
fetchGasEstimates: (blockTime) => dispatch(actions.fetchGasEstimates(blockTime)),
|
fetchGasEstimates: (blockTime) => dispatch(actions.fetchGasEstimates(blockTime)),
|
||||||
fetchBasicGasAndTimeEstimates: () => dispatch(actions.fetchBasicGasAndTimeEstimates()),
|
fetchBasicGasAndTimeEstimates: () => dispatch(actions.fetchBasicGasAndTimeEstimates()),
|
||||||
@ -96,3 +120,11 @@ export default function TransactionList () {
|
|||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TransactionList.propTypes = {
|
||||||
|
tokenAddress: PropTypes.string,
|
||||||
|
}
|
||||||
|
|
||||||
|
TransactionList.defaultProps = {
|
||||||
|
tokenAddress: undefined,
|
||||||
|
}
|
||||||
|
@ -256,7 +256,7 @@ export default class Home extends PureComponent {
|
|||||||
data-testid="home__history-tab"
|
data-testid="home__history-tab"
|
||||||
name="History"
|
name="History"
|
||||||
>
|
>
|
||||||
<TransactionList />
|
<TransactionList tokenAddress={selectedToken?.address} />
|
||||||
</Tab>
|
</Tab>
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</div>
|
</div>
|
||||||
|
@ -139,57 +139,6 @@ describe('Transaction Selectors', function () {
|
|||||||
assert(Array.isArray(selectedTx))
|
assert(Array.isArray(selectedTx))
|
||||||
assert.deepEqual(selectedTx, orderedTxList)
|
assert.deepEqual(selectedTx, orderedTxList)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('returns token transactions from currentNetworkTxList when selectedTokenAddress is valid', function () {
|
|
||||||
|
|
||||||
const state = {
|
|
||||||
metamask: {
|
|
||||||
provider: {
|
|
||||||
nickname: 'mainnet',
|
|
||||||
},
|
|
||||||
featureFlags: {
|
|
||||||
showIncomingTransactions: false,
|
|
||||||
},
|
|
||||||
selectedAddress: '0xAddress',
|
|
||||||
selectedTokenAddress: '0xToken',
|
|
||||||
currentNetworkTxList: [
|
|
||||||
{
|
|
||||||
id: 0,
|
|
||||||
time: 0,
|
|
||||||
txParams: {
|
|
||||||
from: '0xAddress',
|
|
||||||
to: '0xToken',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 1,
|
|
||||||
time: 1,
|
|
||||||
txParams: {
|
|
||||||
from: '0xAddress',
|
|
||||||
to: '0xRecipient',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 2,
|
|
||||||
time: 2,
|
|
||||||
txParams: {
|
|
||||||
from: '0xAddress',
|
|
||||||
to: '0xToken',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
const orderedTxList = state.metamask.currentNetworkTxList
|
|
||||||
.filter((tx) => tx.txParams.to === '0xToken')
|
|
||||||
.sort((a, b) => b.time - a.time)
|
|
||||||
|
|
||||||
const selectedTx = transactionsSelector(state)
|
|
||||||
|
|
||||||
assert(Array.isArray(selectedTx))
|
|
||||||
assert.deepEqual(selectedTx, orderedTxList)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('nonceSortedTransactionsSelector', function () {
|
describe('nonceSortedTransactionsSelector', function () {
|
||||||
|
@ -10,7 +10,6 @@ import {
|
|||||||
TRANSACTION_TYPE_RETRY,
|
TRANSACTION_TYPE_RETRY,
|
||||||
} from '../../../app/scripts/controllers/transactions/enums'
|
} from '../../../app/scripts/controllers/transactions/enums'
|
||||||
import { hexToDecimal } from '../helpers/utils/conversions.util'
|
import { hexToDecimal } from '../helpers/utils/conversions.util'
|
||||||
import { selectedTokenAddressSelector } from './tokens'
|
|
||||||
import { getFastPriceEstimateInHexWEI } from './custom-gas'
|
import { getFastPriceEstimateInHexWEI } from './custom-gas'
|
||||||
import {
|
import {
|
||||||
getSelectedToken,
|
getSelectedToken,
|
||||||
@ -80,23 +79,14 @@ export const transactionSubSelector = createSelector(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
const transactionSelectorReturnHelper = (selectedTokenAddress, transactions) => {
|
|
||||||
return selectedTokenAddress
|
|
||||||
? transactions
|
|
||||||
.filter(({ txParams }) => txParams && txParams.to === selectedTokenAddress)
|
|
||||||
.sort((a, b) => b.time - a.time)
|
|
||||||
: transactions
|
|
||||||
.sort((a, b) => b.time - a.time)
|
|
||||||
}
|
|
||||||
|
|
||||||
export const transactionsSelector = createSelector(
|
export const transactionsSelector = createSelector(
|
||||||
selectedTokenAddressSelector,
|
|
||||||
transactionSubSelector,
|
transactionSubSelector,
|
||||||
selectedAddressTxListSelector,
|
selectedAddressTxListSelector,
|
||||||
(selectedTokenAddress, subSelectorTxList = [], selectedAddressTxList = []) => {
|
(subSelectorTxList = [], selectedAddressTxList = []) => {
|
||||||
const txsToRender = selectedAddressTxList.concat(subSelectorTxList)
|
const txsToRender = selectedAddressTxList.concat(subSelectorTxList)
|
||||||
|
|
||||||
return transactionSelectorReturnHelper(selectedTokenAddress, txsToRender)
|
return txsToRender
|
||||||
|
.sort((a, b) => b.time - a.time)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user