Skip to content

Commit e157a2a

Browse files
Wowliancallil
andcommitted
feat(lists): allow selecting and adding token lists (#1023)
* more list stuff Use the selected list instead of the default list, but also use the default list start list selection code * move token warning to a modal, fix the install issue * add/remove/enter key * handle enter on currency select for ETHER * change slippage tolerance to be a slider * make ui closer to the mocks * commit slider changes * back to tabs * copy changes * bump list version * some styling for the list select * bump uniswap default list version * use contract calls to get ens names and addresses * show list logo * fix failing integration test * .eth.link * list introduction screen * remove showSendWithSwap * fix integration and unit tests * resolve ENS names * logos from ens * fix the lint errors * some refactoring to better support using a the library provider from the user for resolving ENS names * load list info from the list url for the introduction page * make it slightly harder to remove a list * minor clean up, some help text and links * remove icon from list update popup * show added/removed tokens * add GA everywhere, don't debounce contenthash lookups * show tags * fix tag key * tag display, list rendering, needs optimization * fix list fetching in firefox, style issue in safari * sort the lists, clean up styling * use client provider when possible * show token warning for url loaded tokens * improve the warning modal * some refactoring to fix the list fetching on networks other than mainnet * fix tests * some minor improvements * increase timeout to maybe fix integration tests which pass locally * build for tests using the dev network url * reset the lists if we deleted the other two copies * improve how we handle updating the default list of lists * fix integration test * Update token list selection styles * fix external links, reuse the on click outside code, show add errors * show the list origin instead of the full url * fix update list link * show host instead of hostname do not automatically dismiss major version upgrades for lists * fix link to tokenlists.org * add uma * clean up styling in list rows * bump token list version * bump token list version again * hover symbol to see currency name * bump version * add cmc lists, dharma list Co-authored-by: Callil Capuozzo <[email protected]>
1 parent 306940b commit e157a2a

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

84 files changed

+3797
-1075
lines changed

.github/workflows/tests.yaml

+2
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,8 @@ jobs:
2929
- run: yarn install --frozen-lockfile
3030
- run: yarn cypress install
3131
- run: yarn build
32+
env:
33+
REACT_APP_NETWORK_URL: "https://mainnet.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847"
3234
- run: yarn integration-test
3335

3436
unit-tests:

cypress.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,5 +3,6 @@
33
"pluginsFile": false,
44
"fixturesFolder": false,
55
"supportFile": "cypress/support/index.js",
6-
"video": false
6+
"video": false,
7+
"defaultCommandTimeout": 10000
78
}

cypress/integration/lists.test.ts

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
describe('Swap', () => {
2+
beforeEach(() => cy.visit('/swap'))
3+
4+
it('list selection persists', () => {
5+
cy.get('#swap-currency-output .open-currency-select-button').click()
6+
cy.get('#select-default-uniswap-list .select-button').click()
7+
cy.reload()
8+
cy.get('#swap-currency-output .open-currency-select-button').click()
9+
cy.get('#select-default-uniswap-list').should('not.exist')
10+
})
11+
})

cypress/integration/swap.test.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,8 @@
11
describe('Swap', () => {
2-
beforeEach(() => cy.visit('/swap'))
2+
beforeEach(() => {
3+
cy.clearLocalStorage()
4+
cy.visit('/swap')
5+
})
36
it('can enter an amount into input', () => {
47
cy.get('#swap-currency-input .token-amount-input')
58
.type('0.001', { delay: 200 })
@@ -32,6 +35,7 @@ describe('Swap', () => {
3235

3336
it('can swap ETH for DAI', () => {
3437
cy.get('#swap-currency-output .open-currency-select-button').click()
38+
cy.get('#select-default-uniswap-list .select-button').click()
3539
cy.get('.token-item-0xc7AD46e0b8a400Bb3C915120d284AafbA8fc4735').should('be.visible')
3640
cy.get('.token-item-0xc7AD46e0b8a400Bb3C915120d284AafbA8fc4735').click({ force: true })
3741
cy.get('#swap-currency-input .token-amount-input').should('be.visible')

cypress/integration/token-warning.ts

+4-7
Original file line numberDiff line numberDiff line change
@@ -6,14 +6,11 @@ describe('Warning', () => {
66
it('Check that warning is displayed', () => {
77
cy.get('.token-warning-container').should('be.visible')
88
})
9-
it('Check that warning hides after button dismissal.', () => {
9+
it('Check that warning hides after button dismissal', () => {
10+
cy.get('.token-dismiss-button').should('be.disabled')
11+
cy.get('.understand-checkbox').click()
12+
cy.get('.token-dismiss-button').should('not.be.disabled')
1013
cy.get('.token-dismiss-button').click()
1114
cy.get('.token-warning-container').should('not.be.visible')
1215
})
13-
it('Check supression persists across sessions.', () => {
14-
cy.get('.token-warning-container').should('be.visible')
15-
cy.get('.token-dismiss-button').click()
16-
cy.reload()
17-
cy.get('.token-warning-container').should('not.be.visible')
18-
})
1916
})

package.json

+8-1
Original file line numberDiff line numberDiff line change
@@ -11,20 +11,23 @@
1111
"@reduxjs/toolkit": "^1.3.5",
1212
"@types/jest": "^25.2.1",
1313
"@types/lodash.flatmap": "^4.5.6",
14+
"@types/multicodec": "^1.0.0",
1415
"@types/node": "^13.13.5",
1516
"@types/qs": "^6.9.2",
1617
"@types/react": "^16.9.34",
1718
"@types/react-dom": "^16.9.7",
1819
"@types/react-redux": "^7.1.8",
1920
"@types/react-router-dom": "^5.0.0",
21+
"@types/react-virtualized-auto-sizer": "^1.0.0",
2022
"@types/react-window": "^1.8.2",
2123
"@types/rebass": "^4.0.5",
2224
"@types/styled-components": "^5.1.0",
2325
"@types/testing-library__cypress": "^5.0.5",
2426
"@typescript-eslint/eslint-plugin": "^2.31.0",
2527
"@typescript-eslint/parser": "^2.31.0",
28+
"@uniswap/default-token-list": "^1.3.0",
2629
"@uniswap/sdk": "3.0.3-beta.1",
27-
"@uniswap/token-lists": "^1.0.0-beta.11",
30+
"@uniswap/token-lists": "^1.0.0-beta.14",
2831
"@uniswap/v2-core": "1.0.0",
2932
"@uniswap/v2-periphery": "^1.1.0-beta.0",
3033
"@web3-react/core": "^6.0.9",
@@ -34,6 +37,7 @@
3437
"@web3-react/walletconnect-connector": "^6.1.1",
3538
"@web3-react/walletlink-connector": "^6.0.9",
3639
"ajv": "^6.12.3",
40+
"cids": "^1.0.0",
3741
"copy-to-clipboard": "^3.2.0",
3842
"cross-env": "^7.0.2",
3943
"cypress": "^4.11.0",
@@ -49,6 +53,8 @@
4953
"inter-ui": "^3.13.1",
5054
"jazzicon": "^1.5.0",
5155
"lodash.flatmap": "^4.5.0",
56+
"multicodec": "^2.0.0",
57+
"multihashes": "^3.0.1",
5258
"polished": "^3.3.2",
5359
"prettier": "^1.17.0",
5460
"qs": "^6.9.4",
@@ -64,6 +70,7 @@
6470
"react-scripts": "^3.4.1",
6571
"react-spring": "^8.0.27",
6672
"react-use-gesture": "^6.0.14",
73+
"react-virtualized-auto-sizer": "^1.0.2",
6774
"react-window": "^1.8.5",
6875
"rebass": "^4.0.7",
6976
"redux-localstorage-simple": "^2.2.0",

src/components/Card/index.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,11 +19,11 @@ export const LightCard = styled(Card)`
1919
`
2020

2121
export const GreyCard = styled(Card)`
22-
background-color: ${({ theme }) => theme.advancedBG};
22+
background-color: ${({ theme }) => theme.bg3};
2323
`
2424

2525
export const OutlineCard = styled(Card)`
26-
border: 1px solid ${({ theme }) => theme.advancedBG};
26+
border: 1px solid ${({ theme }) => theme.bg3};
2727
`
2828

2929
export const YellowCard = styled(Card)`

src/components/CurrencyInputPanel/index.tsx

+1-4
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,6 @@ interface CurrencyInputPanelProps {
126126
hideBalance?: boolean
127127
pair?: Pair | null
128128
hideInput?: boolean
129-
showSendWithSwap?: boolean
130129
otherCurrency?: Currency | null
131130
id: string
132131
showCommonBases?: boolean
@@ -144,7 +143,6 @@ export default function CurrencyInputPanel({
144143
hideBalance = false,
145144
pair = null, // used for double token logo
146145
hideInput = false,
147-
showSendWithSwap = false,
148146
otherCurrency = null,
149147
id,
150148
showCommonBases
@@ -238,8 +236,7 @@ export default function CurrencyInputPanel({
238236
isOpen={modalOpen}
239237
onDismiss={handleDismissSearch}
240238
onCurrencySelect={onCurrencySelect}
241-
showSendWithSwap={showSendWithSwap}
242-
hiddenCurrency={currency}
239+
selectedCurrency={currency}
243240
otherSelectedCurrency={otherCurrency}
244241
showCommonBases={showCommonBases}
245242
/>

src/components/CurrencyLogo/index.tsx

+20-60
Original file line numberDiff line numberDiff line change
@@ -1,94 +1,54 @@
11
import { Currency, ETHER, Token } from '@uniswap/sdk'
2-
import React, { useState } from 'react'
2+
import React, { useMemo } from 'react'
33
import styled from 'styled-components'
44

55
import EthereumLogo from '../../assets/images/ethereum-logo.png'
6+
import useHttpLocations from '../../hooks/useHttpLocations'
67
import { WrappedTokenInfo } from '../../state/lists/hooks'
7-
import uriToHttp from '../../utils/uriToHttp'
8+
import Logo from '../Logo'
89

9-
const getTokenLogoURL = address =>
10+
const getTokenLogoURL = (address: string) =>
1011
`https://raw.githubusercontent.com/trustwallet/assets/master/blockchains/ethereum/assets/${address}/logo.png`
11-
const BAD_URIS: { [tokenAddress: string]: true } = {}
1212

13-
const Image = styled.img<{ size: string }>`
13+
const StyledEthereumLogo = styled.img<{ size: string }>`
1414
width: ${({ size }) => size};
1515
height: ${({ size }) => size};
16-
background-color: white;
17-
border-radius: 1rem;
1816
box-shadow: 0px 6px 10px rgba(0, 0, 0, 0.075);
17+
border-radius: 24px;
1918
`
2019

21-
const Emoji = styled.span<{ size?: string }>`
22-
display: flex;
23-
align-items: center;
24-
justify-content: center;
25-
font-size: ${({ size }) => size};
26-
width: ${({ size }) => size};
27-
height: ${({ size }) => size};
28-
margin-bottom: -4px;
29-
`
30-
31-
const StyledEthereumLogo = styled.img<{ size: string }>`
20+
const StyledLogo = styled(Logo)<{ size: string }>`
3221
width: ${({ size }) => size};
3322
height: ${({ size }) => size};
34-
box-shadow: 0px 6px 10px rgba(0, 0, 0, 0.075);
35-
border-radius: 24px;
3623
`
3724

3825
export default function CurrencyLogo({
3926
currency,
4027
size = '24px',
41-
...rest
28+
style
4229
}: {
4330
currency?: Currency
4431
size?: string
4532
style?: React.CSSProperties
4633
}) {
47-
const [, refresh] = useState<number>(0)
48-
49-
if (currency === ETHER) {
50-
return <StyledEthereumLogo src={EthereumLogo} size={size} {...rest} />
51-
}
34+
const uriLocations = useHttpLocations(currency instanceof WrappedTokenInfo ? currency.logoURI : undefined)
5235

53-
if (currency instanceof Token) {
54-
let uri: string | undefined
36+
const srcs: string[] = useMemo(() => {
37+
if (currency === ETHER) return []
5538

56-
if (currency instanceof WrappedTokenInfo) {
57-
if (currency.logoURI && !BAD_URIS[currency.logoURI]) {
58-
uri = uriToHttp(currency.logoURI).filter(s => !BAD_URIS[s])[0]
39+
if (currency instanceof Token) {
40+
if (currency instanceof WrappedTokenInfo) {
41+
return [...uriLocations, getTokenLogoURL(currency.address)]
5942
}
60-
}
6143

62-
if (!uri) {
63-
const defaultUri = getTokenLogoURL(currency.address)
64-
if (!BAD_URIS[defaultUri]) {
65-
uri = defaultUri
66-
}
44+
return [getTokenLogoURL(currency.address)]
6745
}
46+
return []
47+
}, [currency, uriLocations])
6848

69-
if (uri) {
70-
return (
71-
<Image
72-
{...rest}
73-
alt={`${currency.name} Logo`}
74-
src={uri}
75-
size={size}
76-
onError={() => {
77-
if (currency instanceof Token) {
78-
BAD_URIS[uri] = true
79-
}
80-
refresh(i => i + 1)
81-
}}
82-
/>
83-
)
84-
}
49+
if (currency === ETHER) {
50+
return <StyledEthereumLogo src={EthereumLogo} size={size} style={style} />
8551
}
8652

87-
return (
88-
<Emoji {...rest} size={size}>
89-
<span role="img" aria-label="Thinking">
90-
🤔
91-
</span>
92-
</Emoji>
93-
)
53+
return <StyledLogo size={size} srcs={srcs} alt={`${currency?.symbol ?? 'token'} logo`} style={style} />
9454
}

src/components/ListLogo/index.tsx

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import React from 'react'
2+
import styled from 'styled-components'
3+
import useHttpLocations from '../../hooks/useHttpLocations'
4+
5+
import Logo from '../Logo'
6+
7+
const StyledListLogo = styled(Logo)<{ size: string }>`
8+
width: ${({ size }) => size};
9+
height: ${({ size }) => size};
10+
`
11+
12+
export default function ListLogo({
13+
logoURI,
14+
style,
15+
size = '24px',
16+
alt
17+
}: {
18+
logoURI: string
19+
size?: string
20+
style?: React.CSSProperties
21+
alt?: string
22+
}) {
23+
const srcs: string[] = useHttpLocations(logoURI)
24+
25+
return <StyledListLogo alt={alt} size={size} srcs={srcs} style={style} />
26+
}

src/components/Logo/index.tsx

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import React, { useState } from 'react'
2+
import { AlertTriangle } from 'react-feather'
3+
import { ImageProps } from 'rebass'
4+
5+
const BAD_SRCS: { [tokenAddress: string]: true } = {}
6+
7+
export interface LogoProps extends Pick<ImageProps, 'style' | 'alt' | 'className'> {
8+
srcs: string[]
9+
}
10+
11+
/**
12+
* Renders an image by sequentially trying a list of URIs, and then eventually a fallback triangle alert
13+
*/
14+
export default function Logo({ srcs, alt, ...rest }: LogoProps) {
15+
const [, refresh] = useState<number>(0)
16+
17+
const src: string | undefined = srcs.find(src => !BAD_SRCS[src])
18+
19+
if (src) {
20+
return (
21+
<img
22+
{...rest}
23+
alt={alt}
24+
src={src}
25+
onError={() => {
26+
if (src) BAD_SRCS[src] = true
27+
refresh(i => i + 1)
28+
}}
29+
/>
30+
)
31+
}
32+
33+
return <AlertTriangle {...rest} />
34+
}

src/components/Menu/index.tsx

+3-19
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
1-
import React, { useRef, useEffect } from 'react'
1+
import React, { useRef } from 'react'
22
import { Info, BookOpen, Code, PieChart, MessageCircle } from 'react-feather'
33
import styled from 'styled-components'
44
import { ReactComponent as MenuIcon } from '../../assets/images/menu.svg'
5+
import { useOnClickOutside } from '../../hooks/useOnClickOutside'
56
import useToggle from '../../hooks/useToggle'
67

78
import { ExternalLink } from '../../theme'
@@ -83,24 +84,7 @@ export default function Menu() {
8384
const node = useRef<HTMLDivElement>()
8485
const [open, toggle] = useToggle(false)
8586

86-
useEffect(() => {
87-
const handleClickOutside = e => {
88-
if (node.current?.contains(e.target) ?? false) {
89-
return
90-
}
91-
toggle()
92-
}
93-
94-
if (open) {
95-
document.addEventListener('mousedown', handleClickOutside)
96-
} else {
97-
document.removeEventListener('mousedown', handleClickOutside)
98-
}
99-
100-
return () => {
101-
document.removeEventListener('mousedown', handleClickOutside)
102-
}
103-
}, [open, toggle])
87+
useOnClickOutside(node, open ? toggle : undefined)
10488

10589
return (
10690
<StyledMenu ref={node}>

src/components/Modal/index.tsx

+1
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ const StyledDialogContent = styled(({ minHeight, maxHeight, mobile, isOpen, ...r
3838
box-shadow: 0 4px 8px 0 ${({ theme }) => transparentize(0.95, theme.shadow1)};
3939
padding: 0px;
4040
width: 50vw;
41+
overflow: hidden;
4142
4243
align-self: ${({ mobile }) => (mobile ? 'flex-end' : 'center')};
4344

0 commit comments

Comments
 (0)