Compare commits
14 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
2c96de053e | ||
|
d336ed9d3d | ||
|
05f95531e7 | ||
|
78cb58329e | ||
|
0e561f80ef | ||
|
51e3bee414 | ||
|
57fc9e1eb7 | ||
|
f6660bef03 | ||
|
ae0bedf24b | ||
|
206c999835 | ||
|
3bd0b1c9be | ||
|
3cc7cecf6a | ||
|
9620365349 | ||
|
8e4e8a90ab |
CODEOWNERS
cypress/e2e/swap
functions
api/image/nfts/collection
nfts/collection
src
assets/images
components
Banner/AndroidAnnouncementBanner
Logo
NavBar
SearchModal
swap
connection
constants
hooks
locales
af-ZA.poar-SA.poca-ES.pocs-CZ.poda-DK.pode-DE.poel-GR.poes-ES.pofi-FI.pofr-FR.pohe-IL.pohu-HU.poid-ID.poit-IT.poja-JP.poko-KR.ponl-NL.pono-NO.popl-PL.popt-BR.popt-PT.poro-RO.poru-RU.posl-SI.posr-SP.posv-SE.posw-TZ.poth-TH.potr-TR.pouk-UA.povi-VN.pozh-CN.pozh-TW.po
nft
state
utils
1
CODEOWNERS
Normal file
1
CODEOWNERS
Normal file
@ -0,0 +1 @@
|
||||
* @uniswap/web-admins
|
@ -1,6 +1,7 @@
|
||||
import { BigNumber } from '@ethersproject/bignumber'
|
||||
import { InterfaceSectionName } from '@uniswap/analytics-events'
|
||||
import { CurrencyAmount } from '@uniswap/sdk-core'
|
||||
import { FeatureFlag } from 'featureFlags'
|
||||
|
||||
import { DEFAULT_DEADLINE_FROM_NOW } from '../../../src/constants/misc'
|
||||
import { DAI, USDC_MAINNET } from '../../../src/constants/tokens'
|
||||
@ -64,7 +65,9 @@ describe('Swap errors', () => {
|
||||
})
|
||||
|
||||
it('slippage failure', () => {
|
||||
cy.visit(`/swap?inputCurrency=${USDC_MAINNET.address}&outputCurrency=${DAI.address}`)
|
||||
cy.visit(`/swap?inputCurrency=${USDC_MAINNET.address}&outputCurrency=${DAI.address}`, {
|
||||
featureFlags: [{ name: FeatureFlag.uniswapXDefaultEnabled, value: false }],
|
||||
})
|
||||
cy.hardhat({ automine: false }).then(async (hardhat) => {
|
||||
await hardhat.fund(hardhat.wallet, CurrencyAmount.fromRawAmount(USDC_MAINNET, 500e6))
|
||||
await hardhat.mine()
|
||||
@ -87,6 +90,7 @@ describe('Swap errors', () => {
|
||||
cy.get(getTestSelector('open-settings-dialog-button')).click()
|
||||
cy.get(getTestSelector('max-slippage-settings')).click()
|
||||
cy.get(getTestSelector('slippage-input')).clear().type('0.01')
|
||||
cy.get(getTestSelector('toggle-uniswap-x-button')).click() // turn off uniswapx
|
||||
cy.get('body').click('topRight') // close modal
|
||||
cy.get(getTestSelector('slippage-input')).should('not.exist')
|
||||
|
||||
|
@ -4,7 +4,7 @@ import { FeatureFlag } from 'featureFlags'
|
||||
import { USDC_MAINNET } from '../../../src/constants/tokens'
|
||||
import { getBalance, getTestSelector } from '../../utils'
|
||||
|
||||
describe('Swap with fees', () => {
|
||||
describe.skip('Swap with fees', () => {
|
||||
describe('Classic swaps', () => {
|
||||
beforeEach(() => {
|
||||
cy.visit('/swap', { featureFlags: [{ name: FeatureFlag.feesEnabled, value: true }] })
|
||||
|
@ -42,7 +42,8 @@ function stubSwapTxReceipt() {
|
||||
})
|
||||
}
|
||||
|
||||
describe('UniswapX Toggle', () => {
|
||||
// TODO: FIX THESE TESTS where we should NOT stub for pricing requests
|
||||
describe.skip('UniswapX Toggle', () => {
|
||||
beforeEach(() => {
|
||||
stubNonPriceQuoteWith(QuoteWhereUniswapXIsBetter)
|
||||
cy.visit(`/swap/?inputCurrency=${USDC_MAINNET.address}&outputCurrency=${DAI.address}`, {
|
||||
@ -90,7 +91,7 @@ describe('UniswapX Toggle', () => {
|
||||
})
|
||||
})
|
||||
|
||||
describe('UniswapX Orders', () => {
|
||||
describe.skip('UniswapX Orders', () => {
|
||||
beforeEach(() => {
|
||||
stubNonPriceQuoteWith(QuoteWhereUniswapXIsBetter)
|
||||
cy.intercept(OrderSubmissionEndpoint, { fixture: 'uniswapx/orderResponse.json' })
|
||||
@ -183,7 +184,7 @@ describe('UniswapX Orders', () => {
|
||||
})
|
||||
})
|
||||
|
||||
describe('UniswapX Eth Input', () => {
|
||||
describe.skip('UniswapX Eth Input', () => {
|
||||
beforeEach(() => {
|
||||
stubNonPriceQuoteWith(QuoteWithEthInput)
|
||||
cy.intercept(OrderSubmissionEndpoint, { fixture: 'uniswapx/orderResponse.json' })
|
||||
@ -236,7 +237,7 @@ describe('UniswapX Eth Input', () => {
|
||||
cy.contains('Swapped')
|
||||
})
|
||||
|
||||
it('switches swap input to WETH after wrap', () => {
|
||||
it('keeps ETH as the input currency before wrap completes', () => {
|
||||
// Setup a swap
|
||||
cy.get('#swap-currency-input .token-amount-input').type('1')
|
||||
cy.wait('@quote')
|
||||
@ -250,16 +251,25 @@ describe('UniswapX Eth Input', () => {
|
||||
|
||||
// Close review modal before wrap is confirmed on chain
|
||||
cy.get(getTestSelector('confirmation-close-icon')).click()
|
||||
// Confirm ETH is still the input token before wrap succeeds
|
||||
cy.contains('ETH')
|
||||
})
|
||||
|
||||
it('switches swap input to WETH after wrap', () => {
|
||||
// Setup a swap
|
||||
cy.get('#swap-currency-input .token-amount-input').type('1')
|
||||
cy.wait('@quote')
|
||||
|
||||
// Prompt ETH wrap and confirm
|
||||
cy.get('#swap-button').click()
|
||||
cy.contains('Confirm swap').click()
|
||||
cy.wait('@eth_sendRawTransaction')
|
||||
cy.hardhat().then((hardhat) => hardhat.mine())
|
||||
|
||||
// Confirm wrap is successful and WETH is now input token
|
||||
cy.contains('Wrapped')
|
||||
cy.contains('WETH')
|
||||
|
||||
// Reopen review modal and continue swap
|
||||
cy.get('#swap-button').click()
|
||||
cy.contains('Confirm swap').click()
|
||||
|
||||
// Approve WETH spend
|
||||
cy.wait('@eth_sendRawTransaction')
|
||||
cy.hardhat().then((hardhat) => hardhat.mine())
|
||||
@ -274,10 +284,15 @@ describe('UniswapX Eth Input', () => {
|
||||
|
||||
// Verify swap success
|
||||
cy.contains('Swapped')
|
||||
|
||||
// Close modal
|
||||
cy.get(getTestSelector('confirmation-close-icon')).click()
|
||||
// The input currency should now be WETH
|
||||
cy.contains('WETH')
|
||||
})
|
||||
})
|
||||
|
||||
describe('UniswapX activity history', () => {
|
||||
describe.skip('UniswapX activity history', () => {
|
||||
beforeEach(() => {
|
||||
cy.intercept(QuoteEndpoint, { fixture: QuoteWhereUniswapXIsBetter })
|
||||
cy.intercept(OrderSubmissionEndpoint, { fixture: 'uniswapx/orderResponse.json' })
|
||||
|
@ -7,19 +7,19 @@ const collectionImageUrls = [
|
||||
'http://127.0.0.1:3000/api/image/nfts/collection/0x49cf6f5d44e70224e2e23fdcdd2c053f30ada28b',
|
||||
]
|
||||
|
||||
const nonexistentImageUrls = [
|
||||
'http://127.0.0.1:3000/api/image/nfts/collection/0xed5af388653567af2f388e6224dc7c4b3241c545',
|
||||
]
|
||||
|
||||
test.each([...collectionImageUrls, ...nonexistentImageUrls])('collectionImageUrl', async (url) => {
|
||||
test.each([...collectionImageUrls])('collectionImageUrl', async (url) => {
|
||||
const response = await fetch(new Request(url))
|
||||
expect(response.status).toBe(200)
|
||||
expect(response.headers.get('content-type')).toBe('image/png')
|
||||
})
|
||||
|
||||
const nonexistentImageUrls = [
|
||||
'http://127.0.0.1:3000/api/image/nfts/collection/0xed5af388653567af2f388e6224dc7c4b3241c545',
|
||||
]
|
||||
|
||||
const invalidCollectionImageUrls = ['http://127.0.0.1:3000/api/image/nfts/collection/0xd3adb33f']
|
||||
|
||||
test.each(invalidCollectionImageUrls)('invalidAssetImageUrl', async (url) => {
|
||||
test.each([...invalidCollectionImageUrls, ...nonexistentImageUrls])('invalidAssetImageUrl', async (url) => {
|
||||
const response = await fetch(new Request(url))
|
||||
expect(response.status).toBeOneOf([404, 500])
|
||||
})
|
||||
|
@ -443,151 +443,3 @@ exports[`should inject metadata for collections 3`] = `
|
||||
</html>
|
||||
"
|
||||
`;
|
||||
|
||||
exports[`should inject metadata for collections 4`] = `
|
||||
"<!DOCTYPE html>
|
||||
<html translate="no">
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
|
||||
<title>Uniswap Interface</title>
|
||||
|
||||
<!--
|
||||
will be replaced with the URL of the \`public\` folder during build.
|
||||
Only files inside the \`public\` folder can be referenced from the HTML.
|
||||
-->
|
||||
<link rel="shortcut icon" type="image/png" href="/favicon.png" />
|
||||
<link rel="apple-touch-icon" sizes="192x192" href="/images/192x192_App_Icon.png" />
|
||||
<link rel="apple-touch-icon" sizes="512x512" href="/images/512x512_App_Icon.png" />
|
||||
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
||||
<meta name="theme-color" content="#fff" />
|
||||
<meta
|
||||
http-equiv="Content-Security-Policy"
|
||||
|
||||
content="script-src 'self' 'unsafe-inline'"
|
||||
|
||||
/>
|
||||
|
||||
<!--
|
||||
Apple Smart App Banner for Safari on iOS
|
||||
https://developer.apple.com/documentation/webkit/promoting_apps_with_smart_app_banners
|
||||
-->
|
||||
<meta name="apple-itunes-app" content="app-id=6443944476">
|
||||
|
||||
<!--
|
||||
manifest.json provides metadata used when the app is installed as a PWA.
|
||||
See https://developers.google.com/web/fundamentals/web-app-manifest/
|
||||
-->
|
||||
<link rel="manifest" href="/manifest.json" />
|
||||
|
||||
<link rel="preconnect" href="https://api.uniswap.org/" crossorigin/>
|
||||
<link rel="preconnect" href="https://mainnet.infura.io/" crossorigin/>
|
||||
|
||||
<link rel="preload" href="/fonts/Basel-Book.woff" as="font" type="font/woff" crossorigin />
|
||||
<link rel="preload" href="/fonts/Basel-Book.woff2" as="font" type="font/woff2" crossorigin />
|
||||
<link rel="preload" href="/fonts/Basel-Medium.woff" as="font" type="font/woff" crossorigin />
|
||||
<link rel="preload" href="/fonts/Basel-Medium.woff2" as="font" type="font/woff2" crossorigin />
|
||||
|
||||
<style>
|
||||
* {
|
||||
font-family: 'Basel', sans-serif;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
/**
|
||||
Explicitly load Basel var from public/ so it does not block LCP's critical path.
|
||||
*/
|
||||
@font-face {
|
||||
font-family: 'Basel';
|
||||
font-weight: 535;
|
||||
font-style: normal;
|
||||
font-display: block;
|
||||
src:
|
||||
url('/fonts/Basel-Medium.woff2') format('woff2'),
|
||||
url('/fonts/Basel-Medium.woff') format('woff');
|
||||
}
|
||||
|
||||
@font-face {
|
||||
font-family: 'Basel';
|
||||
font-weight: 485;
|
||||
font-style: normal;
|
||||
font-display: block;
|
||||
src:
|
||||
url('/fonts/Basel-Book.woff') format('woff2'),
|
||||
url('/fonts/Basel-Book.woff') format('woff');
|
||||
}
|
||||
|
||||
@supports (font-variation-settings: normal) {
|
||||
* {
|
||||
font-family: 'Basel', sans-serif;
|
||||
}
|
||||
}
|
||||
|
||||
html,
|
||||
body {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
button {
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
html {
|
||||
font-size: 16px;
|
||||
font-weight: 485;
|
||||
font-variant: none;
|
||||
font-smooth: always;
|
||||
text-rendering: optimizeLegibility !important;
|
||||
-webkit-font-smoothing: antialiased !important;
|
||||
-moz-osx-font-smoothing: grayscale;
|
||||
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
|
||||
}
|
||||
|
||||
/* Use this to apply network-specific gradient backgrounds, in RadialGradientByChainUpdater.ts */
|
||||
#background-radial-gradient {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
pointer-events: none;
|
||||
width: 200vw;
|
||||
height: 200vh;
|
||||
transform: translate(-50vw, -100vh);
|
||||
z-index: -1;
|
||||
}
|
||||
|
||||
html,
|
||||
body,
|
||||
#root {
|
||||
min-height: 100%;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
html {
|
||||
background: linear-gradient(rgb(19, 19, 19) 0%, rgb(19, 19, 19) 100%);
|
||||
}
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: light) {
|
||||
html {
|
||||
background: radial-gradient(100% 100% at 50% 0%, rgba(255, 184, 226, 0) 0%, rgba(255, 255, 255, 0) 100%), rgb(255, 255, 255);
|
||||
}
|
||||
}
|
||||
</style>
|
||||
<script defer src="/static/js/bundle.js"></script><meta property="og:title" content="0xed5af388653567af2f388e6224dc7c4b3241c545 on Uniswap"/><meta property="og:image" content="http://127.0.0.1:3000/api/image/nfts/collection/0xed5af388653567af2f388e6224dc7c4b3241c545"/><meta property="og:image:width" content="1200"/><meta property="og:image:height" content="630"/><meta property="og:image:alt" content="0xed5af388653567af2f388e6224dc7c4b3241c545 on Uniswap"/><meta property="og:type" content="website"/><meta property="og:url" content="http://127.0.0.1:3000/nfts/collection/0xed5af388653567af2f388e6224dc7c4b3241c545"/><meta property="twitter:card" content="summary_large_image"/><meta property="twitter:title" content="0xed5af388653567af2f388e6224dc7c4b3241c545 on Uniswap"/><meta property="twitter:image" content="http://127.0.0.1:3000/api/image/nfts/collection/0xed5af388653567af2f388e6224dc7c4b3241c545"/><meta property="twitter:image:alt" content="0xed5af388653567af2f388e6224dc7c4b3241c545 on Uniswap"/></head>
|
||||
|
||||
<body>
|
||||
<noscript>You need to enable JavaScript to run this app.</noscript>
|
||||
|
||||
<div id="root">
|
||||
<!-- Triggers the font to load immediately and then is replaced by the app -->
|
||||
<div> </div>
|
||||
</div>
|
||||
|
||||
<div id="background-radial-gradient"></div>
|
||||
</body>
|
||||
</html>
|
||||
"
|
||||
`;
|
||||
|
@ -16,15 +16,7 @@ const collections = [
|
||||
},
|
||||
]
|
||||
|
||||
const nonexistentCollections = [
|
||||
{
|
||||
address: '0xed5af388653567af2f388e6224dc7c4b3241c545',
|
||||
collectionName: '0xed5af388653567af2f388e6224dc7c4b3241c545',
|
||||
image: 'http://127.0.0.1:3000/api/image/nfts/collection/0xed5af388653567af2f388e6224dc7c4b3241c545',
|
||||
},
|
||||
]
|
||||
|
||||
test.each([...collections, ...nonexistentCollections])('should inject metadata for collections', async (collection) => {
|
||||
test.each([...collections])('should inject metadata for collections', async (collection) => {
|
||||
const url = 'http://127.0.0.1:3000/nfts/collection/' + collection.address
|
||||
const body = await fetch(new Request(url)).then((res) => res.text())
|
||||
expect(body).toMatchSnapshot()
|
||||
@ -42,24 +34,33 @@ test.each([...collections, ...nonexistentCollections])('should inject metadata f
|
||||
expect(body).toContain(`<meta property="twitter:image:alt" content="${collection.collectionName} on Uniswap"/>`)
|
||||
})
|
||||
|
||||
const nonexistentCollections = [
|
||||
{
|
||||
address: '0xed5af388653567af2f388e6224dc7c4b3241c545',
|
||||
},
|
||||
]
|
||||
|
||||
const invalidCollections = [
|
||||
{
|
||||
address: '0xd3adb33f',
|
||||
},
|
||||
]
|
||||
|
||||
test.each(invalidCollections)('should not inject metadata for nonexistent collections', async (collection) => {
|
||||
const url = 'http://127.0.0.1:3000/nfts/collection/' + collection.address
|
||||
const body = await fetch(new Request(url)).then((res) => res.text())
|
||||
expect(body).not.toContain('og:title')
|
||||
expect(body).not.toContain('og:image')
|
||||
expect(body).not.toContain('og:image:width')
|
||||
expect(body).not.toContain('og:image:height')
|
||||
expect(body).not.toContain('og:type')
|
||||
expect(body).not.toContain('og:url')
|
||||
expect(body).not.toContain('og:image:alt')
|
||||
expect(body).not.toContain('twitter:card')
|
||||
expect(body).not.toContain('twitter:title')
|
||||
expect(body).not.toContain('twitter:image')
|
||||
expect(body).not.toContain('twitter:image:alt')
|
||||
})
|
||||
test.each([...invalidCollections, ...nonexistentCollections])(
|
||||
'should not inject metadata for nonexistent collections',
|
||||
async (collection) => {
|
||||
const url = 'http://127.0.0.1:3000/nfts/collection/' + collection.address
|
||||
const body = await fetch(new Request(url)).then((res) => res.text())
|
||||
expect(body).not.toContain('og:title')
|
||||
expect(body).not.toContain('og:image')
|
||||
expect(body).not.toContain('og:image:width')
|
||||
expect(body).not.toContain('og:image:height')
|
||||
expect(body).not.toContain('og:type')
|
||||
expect(body).not.toContain('og:url')
|
||||
expect(body).not.toContain('og:image:alt')
|
||||
expect(body).not.toContain('twitter:card')
|
||||
expect(body).not.toContain('twitter:title')
|
||||
expect(body).not.toContain('twitter:image')
|
||||
expect(body).not.toContain('twitter:image:alt')
|
||||
}
|
||||
)
|
||||
|
BIN
src/assets/images/santa-hat.png
Normal file
BIN
src/assets/images/santa-hat.png
Normal file
Binary file not shown.
After ![]() (image error) Size: 4.3 KiB |
@ -51,7 +51,12 @@ export default function AndroidAnnouncementBanner() {
|
||||
<ThemedText.LabelMicro>
|
||||
<Trans>Available now - download from the Google Play Store today</Trans>
|
||||
</ThemedText.LabelMicro>
|
||||
<DownloadButton onClick={onClick}>
|
||||
<DownloadButton
|
||||
onClick={(e) => {
|
||||
e.stopPropagation()
|
||||
onClick()
|
||||
}}
|
||||
>
|
||||
<Trans>Download now</Trans>
|
||||
</DownloadButton>
|
||||
</TextContainer>
|
||||
|
23
src/components/Logo/HolidayOrnament.tsx
Normal file
23
src/components/Logo/HolidayOrnament.tsx
Normal file
@ -0,0 +1,23 @@
|
||||
import { ReactElement } from 'react'
|
||||
import styled from 'styled-components'
|
||||
|
||||
import SantaHat from '../../assets/images/santa-hat.png'
|
||||
|
||||
const SantaHatImage = styled.img`
|
||||
position: absolute;
|
||||
top: 5px;
|
||||
right: 6px;
|
||||
height: 18px;
|
||||
`
|
||||
|
||||
const Christmas = <SantaHatImage src={SantaHat} alt="Santa hat" />
|
||||
|
||||
const MONTH_TO_ORNAMENT: { [date: string]: ReactElement } = {
|
||||
'12': Christmas,
|
||||
}
|
||||
|
||||
export default function HolidayOrnament() {
|
||||
// months in javascript are 0 indexed...
|
||||
const currentMonth = `${new Date().getMonth() + 1}`
|
||||
return MONTH_TO_ORNAMENT[currentMonth] || null
|
||||
}
|
29
src/components/Logo/UniIcon.tsx
Normal file
29
src/components/Logo/UniIcon.tsx
Normal file
File diff suppressed because one or more lines are too long
@ -1,6 +1,7 @@
|
||||
import { Trans } from '@lingui/macro'
|
||||
import { useWeb3React } from '@web3-react/core'
|
||||
import { useAccountDrawer } from 'components/AccountDrawer'
|
||||
import { UniIcon } from 'components/Logo/UniIcon'
|
||||
import Web3Status from 'components/Web3Status'
|
||||
import { useInfoExplorePageEnabled } from 'featureFlags/flags/infoExplore'
|
||||
import { chainIdToBackendName } from 'graphql/data/util'
|
||||
@ -9,7 +10,6 @@ import { useIsNftPage } from 'hooks/useIsNftPage'
|
||||
import { useIsPoolsPage } from 'hooks/useIsPoolsPage'
|
||||
import { Box } from 'nft/components/Box'
|
||||
import { Row } from 'nft/components/Flex'
|
||||
import { UniIcon } from 'nft/components/icons'
|
||||
import { useProfilePageState } from 'nft/hooks'
|
||||
import { ProfilePageStateType } from 'nft/types'
|
||||
import { ReactNode, useCallback } from 'react'
|
||||
|
@ -23,7 +23,7 @@ import { CloseIcon, ThemedText } from 'theme/components'
|
||||
import { UserAddedToken } from 'types/tokens'
|
||||
import { splitHiddenTokens } from 'utils/splitHiddenTokens'
|
||||
|
||||
import { useDefaultActiveTokens, useIsUserAddedToken, useSearchInactiveTokenLists, useToken } from '../../hooks/Tokens'
|
||||
import { useDefaultActiveTokens, useSearchInactiveTokenLists, useToken } from '../../hooks/Tokens'
|
||||
import { isAddress } from '../../utils'
|
||||
import Column from '../Column'
|
||||
import Row, { RowBetween } from '../Row'
|
||||
@ -76,7 +76,6 @@ export function CurrencySearch({
|
||||
const debouncedQuery = useDebounce(searchQuery, 200)
|
||||
const isAddressSearch = isAddress(debouncedQuery)
|
||||
const searchToken = useToken(debouncedQuery)
|
||||
const searchTokenIsAdded = useIsUserAddedToken(searchToken)
|
||||
|
||||
const defaultTokens = useDefaultActiveTokens(chainId)
|
||||
|
||||
@ -289,7 +288,7 @@ export function CurrencySearch({
|
||||
)}
|
||||
</PaddedColumn>
|
||||
<Separator />
|
||||
{searchToken && !searchTokenIsAdded ? (
|
||||
{searchToken ? (
|
||||
<Column style={{ padding: '20px 0', height: '100%' }}>
|
||||
<CurrencyRow
|
||||
currency={searchToken}
|
||||
|
@ -31,7 +31,6 @@ import { ThemedText } from 'theme/components'
|
||||
import invariant from 'tiny-invariant'
|
||||
import { isL2ChainId } from 'utils/chains'
|
||||
import { SignatureExpiredError } from 'utils/errors'
|
||||
import { NumberType, useFormatter } from 'utils/formatNumbers'
|
||||
import { formatSwapPriceUpdatedEventProperties } from 'utils/loggingFormatters'
|
||||
import { didUserReject } from 'utils/swapErrorToUserReadableMessage'
|
||||
import { tradeMeaningfullyDiffers } from 'utils/tradeMeaningFullyDiffer'
|
||||
@ -82,7 +81,6 @@ function useConfirmModalState({
|
||||
const [confirmModalState, setConfirmModalState] = useState<ConfirmModalState>(ConfirmModalState.REVIEWING)
|
||||
const [approvalError, setApprovalError] = useState<PendingModalError>()
|
||||
const [pendingModalSteps, setPendingModalSteps] = useState<PendingConfirmModalState[]>([])
|
||||
const { formatCurrencyAmount } = useFormatter()
|
||||
|
||||
// This is a function instead of a memoized value because we do _not_ want it to update as the allowance changes.
|
||||
// For example, if the user needs to complete 3 steps initially, we should always show 3 step indicators
|
||||
@ -117,14 +115,7 @@ function useConfirmModalState({
|
||||
const nativeCurrency = useNativeCurrency(chainId)
|
||||
|
||||
const [wrapTxHash, setWrapTxHash] = useState<string>()
|
||||
const { execute: onWrap } = useWrapCallback(
|
||||
nativeCurrency,
|
||||
trade.inputAmount.currency,
|
||||
formatCurrencyAmount({
|
||||
amount: trade.inputAmount,
|
||||
type: NumberType.SwapTradeAmount,
|
||||
})
|
||||
)
|
||||
const { execute: onWrap } = useWrapCallback(nativeCurrency, trade.inputAmount.currency, trade.inputAmount.toExact())
|
||||
const wrapConfirmed = useIsTransactionConfirmed(wrapTxHash)
|
||||
const prevWrapConfirmed = usePrevious(wrapConfirmed)
|
||||
const catchUserReject = async (e: any, errorType: PendingModalError) => {
|
||||
@ -142,9 +133,6 @@ function useConfirmModalState({
|
||||
onWrap?.()
|
||||
.then((wrapTxHash) => {
|
||||
setWrapTxHash(wrapTxHash)
|
||||
// After the wrap has succeeded, reset the input currency to be WETH
|
||||
// because the trade will be on WETH -> token
|
||||
onCurrencySelection(Field.INPUT, trade.inputAmount.currency)
|
||||
sendAnalyticsEvent(InterfaceEventName.WRAP_TOKEN_TXN_SUBMITTED, {
|
||||
chain_id: chainId,
|
||||
token_symbol: maximumAmountIn?.currency.symbol,
|
||||
@ -192,7 +180,6 @@ function useConfirmModalState({
|
||||
onWrap,
|
||||
trace,
|
||||
trade,
|
||||
onCurrencySelection,
|
||||
]
|
||||
)
|
||||
|
||||
@ -209,10 +196,20 @@ function useConfirmModalState({
|
||||
useEffect(() => {
|
||||
// If the wrapping step finished, trigger the next step (allowance or swap).
|
||||
if (wrapConfirmed && !prevWrapConfirmed) {
|
||||
// After the wrap has succeeded, reset the input currency to be WETH
|
||||
// because the trade will be on WETH -> token
|
||||
onCurrencySelection(Field.INPUT, trade.inputAmount.currency)
|
||||
// moves on to either approve WETH or to swap submission
|
||||
performStep(pendingModalSteps[1])
|
||||
}
|
||||
}, [pendingModalSteps, performStep, prevWrapConfirmed, wrapConfirmed])
|
||||
}, [
|
||||
pendingModalSteps,
|
||||
performStep,
|
||||
prevWrapConfirmed,
|
||||
wrapConfirmed,
|
||||
onCurrencySelection,
|
||||
trade.inputAmount.currency,
|
||||
])
|
||||
|
||||
useEffect(() => {
|
||||
if (
|
||||
|
@ -3,7 +3,7 @@ import { URI_AVAILABLE, WalletConnect, WalletConnectConstructorArgs } from '@web
|
||||
import { sendAnalyticsEvent } from 'analytics'
|
||||
import { L1_CHAIN_IDS, L2_CHAIN_IDS } from 'constants/chains'
|
||||
import { Z_INDEX } from 'theme/zIndex'
|
||||
import { isIOS } from 'utils/userAgent'
|
||||
import { isAndroid, isIOS } from 'utils/userAgent'
|
||||
|
||||
import { RPC_URLS } from '../constants/networks'
|
||||
|
||||
@ -83,7 +83,7 @@ export class UniwalletConnect extends WalletConnectV2 {
|
||||
this.events.emit(UniwalletConnect.UNI_URI_AVAILABLE, `https://uniswap.org/app/wc?uri=${uri}`)
|
||||
|
||||
// Opens deeplink to Uniswap Wallet if on iOS
|
||||
if (isIOS) {
|
||||
if (isIOS || isAndroid) {
|
||||
// Using window.location.href to open the deep link ensures smooth navigation and leverages OS handling for installed apps,
|
||||
// avoiding potential popup blockers or inconsistent behavior associated with window.open
|
||||
window.location.href = `uniswap://wc?uri=${encodeURIComponent(uri)}`
|
||||
|
@ -90,7 +90,13 @@ export const MATIC_MAINNET = new Token(
|
||||
'MATIC',
|
||||
'Polygon Matic'
|
||||
)
|
||||
const MATIC_POLYGON = new Token(ChainId.POLYGON, '0x0000000000000000000000000000000000001010', 18, 'MATIC', 'Matic')
|
||||
export const MATIC_POLYGON = new Token(
|
||||
ChainId.POLYGON,
|
||||
'0x0000000000000000000000000000000000001010',
|
||||
18,
|
||||
'MATIC',
|
||||
'Matic'
|
||||
)
|
||||
export const DAI_POLYGON = new Token(
|
||||
ChainId.POLYGON,
|
||||
'0x8f3Cf7ad23Cd3CaDbD9735AFf958023239c6A063',
|
||||
@ -143,13 +149,6 @@ export const WBTC_OPTIMISM = new Token(
|
||||
'WBTC',
|
||||
'Wrapped BTC'
|
||||
)
|
||||
const MATIC_POLYGON_MUMBAI = new Token(
|
||||
ChainId.POLYGON_MUMBAI,
|
||||
'0x0000000000000000000000000000000000001010',
|
||||
18,
|
||||
'MATIC',
|
||||
'Matic'
|
||||
)
|
||||
export const WETH_POLYGON_MUMBAI = new Token(
|
||||
ChainId.POLYGON_MUMBAI,
|
||||
'0xa6fa4fb5f76172d178d61b04b0ecd319c5d1c0aa',
|
||||
@ -359,14 +358,21 @@ export function isPolygon(chainId: number): chainId is ChainId.POLYGON | ChainId
|
||||
return chainId === ChainId.POLYGON_MUMBAI || chainId === ChainId.POLYGON
|
||||
}
|
||||
|
||||
function getPolygonNativeCurrency(chainId: number) {
|
||||
switch (chainId) {
|
||||
case ChainId.POLYGON:
|
||||
return MATIC_POLYGON
|
||||
case ChainId.POLYGON_MUMBAI:
|
||||
return MATIC_POLYGON_MUMBAI
|
||||
default:
|
||||
throw new Error('Not polygon')
|
||||
class PolygonNativeCurrency extends NativeCurrency {
|
||||
equals(other: Currency): boolean {
|
||||
return other.isNative && other.chainId === this.chainId
|
||||
}
|
||||
|
||||
get wrapped(): Token {
|
||||
if (!isPolygon(this.chainId)) throw new Error('Not Polygon')
|
||||
const wrapped = WRAPPED_NATIVE_CURRENCY[this.chainId]
|
||||
invariant(wrapped instanceof Token)
|
||||
return wrapped
|
||||
}
|
||||
|
||||
public constructor(chainId: number) {
|
||||
if (!isPolygon(chainId)) throw new Error('Not Polygon')
|
||||
super(chainId, 18, 'MATIC', 'Matic')
|
||||
}
|
||||
}
|
||||
|
||||
@ -433,7 +439,7 @@ export function nativeOnChain(chainId: number): NativeCurrency | Token {
|
||||
if (cachedNativeCurrency[chainId]) return cachedNativeCurrency[chainId]
|
||||
let nativeCurrency: NativeCurrency | Token
|
||||
if (isPolygon(chainId)) {
|
||||
nativeCurrency = getPolygonNativeCurrency(chainId)
|
||||
nativeCurrency = new PolygonNativeCurrency(chainId)
|
||||
} else if (isCelo(chainId)) {
|
||||
nativeCurrency = getCeloNativeCurrency(chainId)
|
||||
} else if (isBsc(chainId)) {
|
||||
|
@ -1,4 +1,3 @@
|
||||
import ms from 'ms'
|
||||
import { useEffect } from 'react'
|
||||
import { ApplicationModal, setOpenModal } from 'state/application/reducer'
|
||||
import { useAppDispatch } from 'state/hooks'
|
||||
@ -7,34 +6,23 @@ export default function useAccountRiskCheck(account: string | null | undefined)
|
||||
const dispatch = useAppDispatch()
|
||||
|
||||
useEffect(() => {
|
||||
if (account) {
|
||||
const riskCheckLocalStorageKey = `risk-check-${account}`
|
||||
const now = Date.now()
|
||||
try {
|
||||
// Check local browser cache
|
||||
const storedTime = localStorage.getItem(riskCheckLocalStorageKey)
|
||||
const checkExpirationTime = storedTime ? parseInt(storedTime) : now - 1
|
||||
if (checkExpirationTime < Date.now()) {
|
||||
const headers = new Headers({ 'Content-Type': 'application/json' })
|
||||
fetch('https://api.uniswap.org/v1/screen', {
|
||||
method: 'POST',
|
||||
headers,
|
||||
body: JSON.stringify({ address: account }),
|
||||
})
|
||||
.then((res) => res.json())
|
||||
.then((data) => {
|
||||
if (data.block) {
|
||||
dispatch(setOpenModal(ApplicationModal.BLOCKED_ACCOUNT))
|
||||
}
|
||||
})
|
||||
.catch(() => {
|
||||
dispatch(setOpenModal(null))
|
||||
})
|
||||
if (!account) return
|
||||
|
||||
// TODO: add back local browser cacheing (revisit 11/13/2023)
|
||||
const headers = new Headers({ 'Content-Type': 'application/json' })
|
||||
fetch('https://api.uniswap.org/v1/screen', {
|
||||
method: 'POST',
|
||||
headers,
|
||||
body: JSON.stringify({ address: account }),
|
||||
})
|
||||
.then((res) => res.json())
|
||||
.then((data) => {
|
||||
if (data.block) {
|
||||
dispatch(setOpenModal(ApplicationModal.BLOCKED_ACCOUNT))
|
||||
}
|
||||
} finally {
|
||||
// Set item to have 1 day local cache storage
|
||||
localStorage.setItem(riskCheckLocalStorageKey, (now + ms(`1d`)).toString())
|
||||
}
|
||||
}
|
||||
})
|
||||
.catch(() => {
|
||||
dispatch(setOpenModal(null))
|
||||
})
|
||||
}, [account, dispatch])
|
||||
}
|
||||
|
3816
src/locales/af-ZA.po
Normal file
3816
src/locales/af-ZA.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/ar-SA.po
Normal file
3816
src/locales/ar-SA.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/ca-ES.po
Normal file
3816
src/locales/ca-ES.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/cs-CZ.po
Normal file
3816
src/locales/cs-CZ.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/da-DK.po
Normal file
3816
src/locales/da-DK.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/de-DE.po
Normal file
3816
src/locales/de-DE.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/el-GR.po
Normal file
3816
src/locales/el-GR.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/es-ES.po
Normal file
3816
src/locales/es-ES.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/fi-FI.po
Normal file
3816
src/locales/fi-FI.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/fr-FR.po
Normal file
3816
src/locales/fr-FR.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/he-IL.po
Normal file
3816
src/locales/he-IL.po
Normal file
File diff suppressed because it is too large
Load Diff
3817
src/locales/hu-HU.po
Normal file
3817
src/locales/hu-HU.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/id-ID.po
Normal file
3816
src/locales/id-ID.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/it-IT.po
Normal file
3816
src/locales/it-IT.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/ja-JP.po
Normal file
3816
src/locales/ja-JP.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/ko-KR.po
Normal file
3816
src/locales/ko-KR.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/nl-NL.po
Normal file
3816
src/locales/nl-NL.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/no-NO.po
Normal file
3816
src/locales/no-NO.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/pl-PL.po
Normal file
3816
src/locales/pl-PL.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/pt-BR.po
Normal file
3816
src/locales/pt-BR.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/pt-PT.po
Normal file
3816
src/locales/pt-PT.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/ro-RO.po
Normal file
3816
src/locales/ro-RO.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/ru-RU.po
Normal file
3816
src/locales/ru-RU.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/sl-SI.po
Normal file
3816
src/locales/sl-SI.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/sr-SP.po
Normal file
3816
src/locales/sr-SP.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/sv-SE.po
Normal file
3816
src/locales/sv-SE.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/sw-TZ.po
Normal file
3816
src/locales/sw-TZ.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/th-TH.po
Normal file
3816
src/locales/th-TH.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/tr-TR.po
Normal file
3816
src/locales/tr-TR.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/uk-UA.po
Normal file
3816
src/locales/uk-UA.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/vi-VN.po
Normal file
3816
src/locales/vi-VN.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/zh-CN.po
Normal file
3816
src/locales/zh-CN.po
Normal file
File diff suppressed because it is too large
Load Diff
3816
src/locales/zh-TW.po
Normal file
3816
src/locales/zh-TW.po
Normal file
File diff suppressed because it is too large
Load Diff
@ -4,10 +4,11 @@ import { InterfaceModalName, NFTEventName } from '@uniswap/analytics-events'
|
||||
import { Trace, useTrace } from 'analytics'
|
||||
import clsx from 'clsx'
|
||||
import { OpacityHoverState } from 'components/Common'
|
||||
import { UniIcon } from 'components/Logo/UniIcon'
|
||||
import { Box } from 'nft/components/Box'
|
||||
import { Portal } from 'nft/components/common/Portal'
|
||||
import { Row } from 'nft/components/Flex'
|
||||
import { BackArrowIcon, ChevronUpIcon, LightningBoltIcon, TwitterIcon, UniIcon } from 'nft/components/icons'
|
||||
import { BackArrowIcon, ChevronUpIcon, LightningBoltIcon, TwitterIcon } from 'nft/components/icons'
|
||||
import { Overlay, stopPropagation } from 'nft/components/modals/Overlay'
|
||||
import { themeVars, vars } from 'nft/css/sprinkles.css'
|
||||
import { useIsMobile, useNativeUsdPrice, useSendTransaction, useTransactionResponse } from 'nft/hooks'
|
||||
|
File diff suppressed because one or more lines are too long
@ -10,4 +10,5 @@ export const blocklistedCollections = [
|
||||
'0x8e52fb89b6311bd9ec36bd7cea9a0c311fd27a92',
|
||||
'0x2079c2765462af6d78a9ccbddb6ff3c6d4ba2e24',
|
||||
'0xd4d871419714b778ebec2e22c7c53572b573706e',
|
||||
'0x383e2070a38f3205be0f3ea4fb05a5c13ade6cf4',
|
||||
]
|
||||
|
@ -13,7 +13,7 @@ const defaultState = {
|
||||
user: {},
|
||||
_persist: {
|
||||
rehydrated: true,
|
||||
version: 2,
|
||||
version: 4,
|
||||
},
|
||||
application: {
|
||||
chainId: null,
|
||||
|
@ -4,6 +4,8 @@ import { MigrationConfig } from 'redux-persist/es/createMigrate'
|
||||
import { migration0 } from './migrations/0'
|
||||
import { migration1 } from './migrations/1'
|
||||
import { migration2 } from './migrations/2'
|
||||
import { migration3 } from './migrations/3'
|
||||
import { migration4 } from './migrations/4'
|
||||
import { legacyLocalStorageMigration } from './migrations/legacy'
|
||||
|
||||
/**
|
||||
@ -19,6 +21,8 @@ export const migrations: MigrationManifest = {
|
||||
0: migration0,
|
||||
1: migration1,
|
||||
2: migration2,
|
||||
3: migration3,
|
||||
4: migration4,
|
||||
}
|
||||
|
||||
// We use a custom migration function for the initial state, because redux-persist
|
||||
|
@ -26,7 +26,7 @@ export const migration3 = (state: PersistAppStateV3 | undefined) => {
|
||||
}
|
||||
for (const [chainId, address] of Object.entries(USDCe_ADDRESSES)) {
|
||||
const chainIdKey = Number(chainId) as ChainId
|
||||
if (state.user.tokens[chainIdKey]?.[address]) {
|
||||
if (state.user.tokens?.[chainIdKey]?.[address]) {
|
||||
state.user.tokens[chainIdKey][address] = serializeToken(
|
||||
new Token(chainIdKey, address, 6, 'USDC.e', 'Bridged USDC')
|
||||
)
|
||||
@ -40,7 +40,7 @@ export const migration3 = (state: PersistAppStateV3 | undefined) => {
|
||||
'USDbC',
|
||||
'USD Base Coin'
|
||||
)
|
||||
if (state.user.tokens[ChainId.BASE]?.[USDbC_BASE.address]) {
|
||||
if (state.user.tokens?.[ChainId.BASE]?.[USDbC_BASE.address]) {
|
||||
state.user.tokens[ChainId.BASE][USDbC_BASE.address] = serializeToken(USDbC_BASE)
|
||||
}
|
||||
return {
|
||||
|
46
src/state/migrations/4.test.ts
Normal file
46
src/state/migrations/4.test.ts
Normal file
@ -0,0 +1,46 @@
|
||||
import { DEFAULT_LOCALE } from 'constants/locales'
|
||||
import { createMigrate } from 'redux-persist'
|
||||
import { RouterPreference } from 'state/routing/types'
|
||||
import { SlippageTolerance } from 'state/user/types'
|
||||
|
||||
import { migration1 } from './1'
|
||||
import { migration2 } from './2'
|
||||
import { migration3 } from './3'
|
||||
import { migration4, PersistAppStateV4 } from './4'
|
||||
|
||||
const previousState: PersistAppStateV4 = {
|
||||
user: {
|
||||
userLocale: 'de-DE',
|
||||
userRouterPreference: RouterPreference.API,
|
||||
userHideClosedPositions: false,
|
||||
userSlippageTolerance: SlippageTolerance.Auto,
|
||||
userSlippageToleranceHasBeenMigratedToAuto: true,
|
||||
userDeadline: 1800,
|
||||
tokens: {},
|
||||
pairs: {},
|
||||
timestamp: Date.now(),
|
||||
hideAndroidAnnouncementBanner: false,
|
||||
},
|
||||
_persist: {
|
||||
version: 3,
|
||||
rehydrated: true,
|
||||
},
|
||||
}
|
||||
|
||||
describe('migration to v4', () => {
|
||||
it('should migrate users who currently have German as their set locale', async () => {
|
||||
const migrator = createMigrate(
|
||||
{
|
||||
1: migration1,
|
||||
2: migration2,
|
||||
3: migration3,
|
||||
4: migration4,
|
||||
},
|
||||
{ debug: false }
|
||||
)
|
||||
const result: any = await migrator(previousState, 4)
|
||||
expect(result.user.userLocale).toEqual(DEFAULT_LOCALE)
|
||||
|
||||
expect(result?._persist.version).toEqual(4)
|
||||
})
|
||||
})
|
28
src/state/migrations/4.ts
Normal file
28
src/state/migrations/4.ts
Normal file
@ -0,0 +1,28 @@
|
||||
import { DEFAULT_LOCALE } from 'constants/locales'
|
||||
import { PersistState } from 'redux-persist'
|
||||
import { UserState } from 'state/user/reducer'
|
||||
|
||||
export type PersistAppStateV4 = {
|
||||
_persist: PersistState
|
||||
} & { user?: UserState }
|
||||
|
||||
/**
|
||||
* Migration to set german locale to default locale, after
|
||||
* the german locale was removed from supported locales.
|
||||
*/
|
||||
export const migration4 = (state: PersistAppStateV4 | undefined) => {
|
||||
if (state?.user) {
|
||||
if (state.user.userLocale === 'de-DE') {
|
||||
state.user.userLocale = DEFAULT_LOCALE
|
||||
}
|
||||
|
||||
return {
|
||||
...state,
|
||||
_persist: {
|
||||
...state._persist,
|
||||
version: 4,
|
||||
},
|
||||
}
|
||||
}
|
||||
return state
|
||||
}
|
@ -44,7 +44,7 @@ export type AppState = ReturnType<typeof appReducer>
|
||||
|
||||
const persistConfig: PersistConfig<AppState> = {
|
||||
key: 'interface',
|
||||
version: 2, // see migrations.ts for more details about this version
|
||||
version: 4, // see migrations.ts for more details about this version
|
||||
storage: localForage.createInstance({
|
||||
name: 'redux',
|
||||
}),
|
||||
|
@ -1,5 +1,5 @@
|
||||
import { MaxUint256, PERMIT2_ADDRESS } from '@uniswap/permit2-sdk'
|
||||
import { Currency } from '@uniswap/sdk-core'
|
||||
import { ChainId, Currency } from '@uniswap/sdk-core'
|
||||
import ERC20_ABI from 'abis/erc20.json'
|
||||
import { Erc20, Weth } from 'abis/types'
|
||||
import WETH_ABI from 'abis/weth.json'
|
||||
@ -26,6 +26,12 @@ export async function getApproveInfo(
|
||||
// If any of these arguments aren't provided, then we cannot generate approval cost info
|
||||
if (!account || !usdCostPerGas) return { needsApprove: false }
|
||||
|
||||
// routing-api under estimates gas for Arbitrum swaps so it inflates cost per gas by a lot
|
||||
// so disable showing approves for Arbitrum until routing-api gives more accurate gas estimates
|
||||
if (currency.chainId === ChainId.ARBITRUM_ONE || currency.chainId === ChainId.ARBITRUM_GOERLI) {
|
||||
return { needsApprove: false }
|
||||
}
|
||||
|
||||
const provider = DEPRECATED_RPC_PROVIDERS[currency.chainId as SupportedInterfaceChain]
|
||||
const tokenContract = getContract(currency.address, ERC20_ABI, provider) as Erc20
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { nativeOnChain } from 'constants/tokens'
|
||||
import { MATIC_POLYGON, nativeOnChain } from 'constants/tokens'
|
||||
import { Chain } from 'graphql/data/__generated__/types-and-hooks'
|
||||
import { supportedChainIdFromGQLChain } from 'graphql/data/util'
|
||||
|
||||
@ -10,8 +10,11 @@ export function getNativeTokenDBAddress(chain: Chain): string | undefined {
|
||||
switch (chain) {
|
||||
// Celo & Polygon have precompiles for their native tokens
|
||||
case Chain.Celo:
|
||||
case Chain.Polygon:
|
||||
return nativeOnChain(pageChainId).wrapped.address
|
||||
case Chain.Polygon:
|
||||
// Like Celo, native MATIC does have a ERC20 precompile, but we use WMATIC in routing/pools
|
||||
// So instead of returning nativeOnChain().wrapped.address, should directly use the precompile address here
|
||||
return MATIC_POLYGON.address
|
||||
case Chain.Ethereum:
|
||||
case Chain.Arbitrum:
|
||||
case Chain.EthereumGoerli:
|
||||
|
Loading…
x
Reference in New Issue
Block a user