Skip to content

Commit

Permalink
fix: display msg if tokenId already exists
Browse files Browse the repository at this point in the history
  • Loading branch information
sahar-fehri committed Sep 29, 2023
1 parent c686a18 commit 6a6074d
Show file tree
Hide file tree
Showing 4 changed files with 154 additions and 3 deletions.
3 changes: 3 additions & 0 deletions app/_locales/en/messages.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

25 changes: 23 additions & 2 deletions ui/components/multichain/import-nfts-modal/import-nfts-modal.js
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,8 @@ import {
ModalOverlay,
} from '../../component-library';
import Tooltip from '../../ui/tooltip';
import { useNftsCollections } from '../../../hooks/useNftsCollections';
import { checkTokenIdExists } from '../../../helpers/utils/util';

export const ImportNftsModal = ({ onClose }) => {
const t = useI18nContext();
Expand All @@ -67,14 +69,15 @@ export const ImportNftsModal = ({ onClose }) => {
tokenId: initialTokenId,
ignoreErc20Token,
} = useSelector((state) => state.appState.importNftsModal);

const existingNfts = useNftsCollections();
const [nftAddress, setNftAddress] = useState(initialTokenAddress ?? '');
const [tokenId, setTokenId] = useState(initialTokenId ?? '');
const [disabled, setDisabled] = useState(true);
const [nftAddFailed, setNftAddFailed] = useState(false);
const trackEvent = useContext(MetaMetricsContext);
const [nftAddressValidationError, setNftAddressValidationError] =
useState(null);
const [duplicateTokenIdError, setDuplicateTokenIdError] = useState(null);

const handleAddNft = async () => {
try {
Expand Down Expand Up @@ -140,7 +143,23 @@ export const ImportNftsModal = ({ onClose }) => {
};

const validateAndSetTokenId = (val) => {
setDisabled(!isValidHexAddress(nftAddress) || !val || isNaN(Number(val)));
setDuplicateTokenIdError(null);
// Check if tokenId is already imported
const tokenIdExists = checkTokenIdExists(
nftAddress,
val,
existingNfts.collections,
);
if (tokenIdExists) {
setDuplicateTokenIdError(t('nftAlreadyAdded'));
}
setDisabled(
!isValidHexAddress(nftAddress) ||
!val ||
isNaN(Number(val)) ||
tokenIdExists,
);

setTokenId(val);
};

Expand Down Expand Up @@ -250,6 +269,8 @@ export const ImportNftsModal = ({ onClose }) => {
validateAndSetTokenId(e.target.value);
setNftAddFailed(false);
}}
helpText={duplicateTokenIdError}
error={duplicateTokenIdError}
/>
</Box>
</Box>
Expand Down
38 changes: 37 additions & 1 deletion ui/helpers/utils/util.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,11 @@ import bowser from 'bowser';
///: BEGIN:ONLY_INCLUDE_IN(snaps)
import { getSnapPrefix } from '@metamask/snaps-utils';
import { WALLET_SNAP_PERMISSION_KEY } from '@metamask/rpc-methods';
// eslint-disable-next-line import/no-duplicates
import { isObject } from '@metamask/utils';
///: END:ONLY_INCLUDE_IN
// eslint-disable-next-line import/no-duplicates
import { isStrictHexString } from '@metamask/utils';
import { CHAIN_IDS, NETWORK_TYPES } from '../../../shared/constants/network';
import {
toChecksumHexAddress,
Expand All @@ -30,8 +33,10 @@ import {
SNAPS_METADATA,
} from '../../../shared/constants/snaps';
///: END:ONLY_INCLUDE_IN

// formatData :: ( date: <Unix Timestamp> ) -> String
import { isEqualCaseInsensitive } from '../../../shared/modules/string-utils';
import { hexToDecimal } from '../../../shared/modules/conversion.utils';

export function formatDate(date, format = "M/d/y 'at' T") {
if (!date) {
return '';
Expand Down Expand Up @@ -631,3 +636,34 @@ export const getNetworkNameFromProviderType = (providerName) => {
export const isAbleToExportAccount = (keyringType = '') => {
return !keyringType.includes('Hardware') && !keyringType.includes('Snap');
};

/**
* Checks if a tokenId in Hex or decimal format already exists in an object.
*
* @param {string} address - collection address.
* @param {string} tokenId - tokenId to search for
* @param {*} obj - object to look into
* @returns {boolean} `false` if tokenId does not already exist.
*/
export const checkTokenIdExists = (address, tokenId, obj) => {
// check if input tokenId is hexadecimal
// If it is convert to decimal and compare with existing tokens
const isHex = isStrictHexString(tokenId);
let convertedTokenId = tokenId;
if (isHex) {
// Convert to decimal
convertedTokenId = hexToDecimal(tokenId);
}

if (obj[address]) {
const value = obj[address];
return lodash.some(value.nfts, (nft) => {
return (
nft.address === address &&
(isEqualCaseInsensitive(nft.tokenId, tokenId) ||
isEqualCaseInsensitive(nft.tokenId, convertedTokenId.toString()))
);
});
}
return false;
};
91 changes: 91 additions & 0 deletions ui/helpers/utils/util.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -929,4 +929,95 @@ describe('util', () => {
expect(util.getNetworkNameFromProviderType('rpc')).toStrictEqual('');
});
});

describe('checkTokenIdExists()', () => {
const data = {
'0x2df920B180c58766951395c26ecF1EC2063490Fa': {
collectionName: 'Numbers',
nfts: [
{
address: '0x2df920B180c58766951395c26ecF1EC2063490Fa',
description: 'Numbers',
favorite: false,
name: 'Numbers #132',
tokenId: '132',
},
],
},
'0x2df920B180c58766951395c26ecF1EC206343334': {
collectionName: 'toto',
nfts: [
{
address: '0x2df920B180c58766951395c26ecF1EC206343334',
description: 'toto',
favorite: false,
name: 'toto#3453',
tokenId: '3453',
},
],
},
'0xf4910C763eD4e47A585E2D34baA9A4b611aE448C': {
collectionName: 'foo',
nfts: [
{
address: '0xf4910C763eD4e47A585E2D34baA9A4b611aE448C',
description: 'foo',
favorite: false,
name: 'toto#111486581076844052489180254627234340268504869259922513413248833349282110111749',
tokenId:
'111486581076844052489180254627234340268504869259922513413248833349282110111749',
},
],
},
};
it('should return true if it exists', () => {
expect(
util.checkTokenIdExists(
'0x2df920B180c58766951395c26ecF1EC206343334',
'3453',
data,
),
).toBeTruthy();
});

it('should return true if it exists in decimal format', () => {
expect(
util.checkTokenIdExists(
'0x2df920B180c58766951395c26ecF1EC206343334',
'0xD7D',
data,
),
).toBeTruthy();
});

it('should return true if is exists but input is not decimal nor hex', () => {
expect(
util.checkTokenIdExists(
'0xf4910C763eD4e47A585E2D34baA9A4b611aE448C',
'111486581076844052489180254627234340268504869259922513413248833349282110111749',
data,
),
).toBeTruthy();
});

it('should return false if it does not exists', () => {
expect(
util.checkTokenIdExists(
'0x2df920B180c58766951395c26ecF1EC206343334',
'1122',
data,
),
).toBeFalsy();
});

it('should return false if it address does not exists', () => {
expect(
util.checkTokenIdExists(
'0xB0BdaBea57B0BDABeA57b0bdABEA57b0BDabEa57',
'1122',
data,
),
).toBeFalsy();
});
});
});

0 comments on commit 6a6074d

Please sign in to comment.