Compare commits
25 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6d5625a1f8 | ||
|
|
d425ff64b4 | ||
|
|
2f47fdf71d | ||
|
|
9f8719f2a5 | ||
|
|
77b640c41b | ||
|
|
d2f98bc9b4 | ||
|
|
e4dd4f9283 | ||
|
|
7798443919 | ||
|
|
392d78b9da | ||
|
|
231289732c | ||
|
|
140ff7a674 | ||
|
|
3fbc4e34f4 | ||
|
|
b964953daf | ||
|
|
649fd9c845 | ||
|
|
6347e63a15 | ||
|
|
bdcb9a8a0a | ||
|
|
8d90bb7a39 | ||
|
|
d70b456855 | ||
|
|
fbb797fa54 | ||
|
|
8ace518311 | ||
|
|
67c776c995 | ||
|
|
719754c46c | ||
|
|
9170af888e | ||
|
|
b258f557d1 | ||
|
|
9d8c7f8e12 |
3
.env
3
.env
@@ -1,2 +1,3 @@
|
|||||||
REACT_APP_CHAIN_ID="1"
|
REACT_APP_CHAIN_ID="1"
|
||||||
REACT_APP_NETWORK_URL="https://mainnet.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847"
|
REACT_APP_NETWORK_URL="https://mainnet.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847"
|
||||||
|
REACT_APP_WALLETCONNECT_BRIDGE_URL="https://uniswap.bridge.walletconnect.org"
|
||||||
38
.github/workflows/integration-tests.yaml
vendored
Normal file
38
.github/workflows/integration-tests.yaml
vendored
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
name: Integration Tests
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
integration-tests:
|
||||||
|
name: Cypress
|
||||||
|
runs-on: ubuntu-16.04
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Set up node
|
||||||
|
uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: 14
|
||||||
|
registry-url: https://registry.npmjs.org
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: yarn install --frozen-lockfile
|
||||||
|
|
||||||
|
- run: yarn cypress install
|
||||||
|
- run: yarn build
|
||||||
|
env:
|
||||||
|
CI: false
|
||||||
|
REACT_APP_NETWORK_URL: "https://mainnet.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847"
|
||||||
|
|
||||||
|
- run: yarn integration-test
|
||||||
|
env:
|
||||||
|
CYPRESS_INTEGRATION_TEST_PRIVATE_KEY: ${{ secrets.CYPRESS_INTEGRATION_TEST_PRIVATE_KEY }}
|
||||||
|
|
||||||
|
|
||||||
21
.github/workflows/lint.yml
vendored
21
.github/workflows/lint.yml
vendored
@@ -14,33 +14,20 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Check out Git repository
|
- name: Checkout
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
- name: Set up node
|
- name: Set up node
|
||||||
uses: actions/setup-node@v1
|
uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
node-version: 12
|
node-version: 14
|
||||||
always-auth: true
|
|
||||||
registry-url: https://registry.npmjs.org
|
registry-url: https://registry.npmjs.org
|
||||||
|
|
||||||
- name: Set output of cache
|
|
||||||
id: yarn-cache
|
|
||||||
run: echo "::set-output name=dir::$(yarn cache dir)"
|
|
||||||
|
|
||||||
- name: Node dependency cache
|
|
||||||
uses: actions/cache@v1
|
|
||||||
with:
|
|
||||||
path: ${{ steps.yarn-cache.outputs.dir }}
|
|
||||||
key: yarn-${{ hashFiles('**/yarn.lock') }}
|
|
||||||
restore-keys: |
|
|
||||||
yarn-
|
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: yarn install --frozen-lockfile
|
run: yarn install --frozen-lockfile
|
||||||
|
|
||||||
- name: Run linters
|
- name: Run linters
|
||||||
uses: wearerequired/lint-action@77d70b9a07ecb93bc98dc46dc27d96c4f004d035
|
uses: wearerequired/lint-action@b98b0918aa71490373d2eca9e8e39a9bc1cc2517
|
||||||
with:
|
with:
|
||||||
github_token: ${{ secrets.github_token }}
|
github_token: ${{ secrets.github_token }}
|
||||||
eslint: true
|
eslint: true
|
||||||
|
|||||||
18
.github/workflows/release.yaml
vendored
18
.github/workflows/release.yaml
vendored
@@ -15,7 +15,7 @@ jobs:
|
|||||||
changelog: ${{ steps.github_tag_action.outputs.changelog }}
|
changelog: ${{ steps.github_tag_action.outputs.changelog }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v1
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
- name: Bump version and push tag
|
- name: Bump version and push tag
|
||||||
id: github_tag_action
|
id: github_tag_action
|
||||||
@@ -31,12 +31,12 @@ jobs:
|
|||||||
if: ${{ needs.bump_version.outputs.new_tag != null }}
|
if: ${{ needs.bump_version.outputs.new_tag != null }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v1
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
- uses: actions/setup-node@v1
|
- name: Set up node
|
||||||
|
uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
node-version: '12'
|
node-version: 14
|
||||||
always-auth: true
|
|
||||||
registry-url: https://registry.npmjs.org
|
registry-url: https://registry.npmjs.org
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
@@ -54,6 +54,14 @@ jobs:
|
|||||||
pinata-api-key: ${{ secrets.PINATA_API_KEY }}
|
pinata-api-key: ${{ secrets.PINATA_API_KEY }}
|
||||||
pinata-secret-api-key: ${{ secrets.PINATA_API_SECRET_KEY }}
|
pinata-secret-api-key: ${{ secrets.PINATA_API_SECRET_KEY }}
|
||||||
|
|
||||||
|
- name: Pin to Crust
|
||||||
|
uses: crustio/ipfs-crust-action@v1.0.8
|
||||||
|
continue-on-error: true
|
||||||
|
timeout-minutes: 2
|
||||||
|
with:
|
||||||
|
cid: ${{ steps.upload.outputs.hash }}
|
||||||
|
seeds: ${{ secrets.CRUST_SEEDS }}
|
||||||
|
|
||||||
- name: Convert CIDv0 to CIDv1
|
- name: Convert CIDv0 to CIDv1
|
||||||
id: convert_cidv0
|
id: convert_cidv0
|
||||||
uses: uniswap/convert-cidv0-cidv1@v1.0.0
|
uses: uniswap/convert-cidv0-cidv1@v1.0.0
|
||||||
|
|||||||
65
.github/workflows/tests.yaml
vendored
65
.github/workflows/tests.yaml
vendored
@@ -1,65 +0,0 @@
|
|||||||
name: Tests
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- main
|
|
||||||
pull_request:
|
|
||||||
branches:
|
|
||||||
- main
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
integration-tests:
|
|
||||||
name: Integration tests
|
|
||||||
runs-on: ubuntu-16.04
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v1
|
|
||||||
- uses: actions/setup-node@v1
|
|
||||||
with:
|
|
||||||
node-version: '12'
|
|
||||||
always-auth: true
|
|
||||||
registry-url: https://registry.npmjs.org
|
|
||||||
|
|
||||||
- name: Get yarn cache directory path
|
|
||||||
id: yarn-cache-dir-path
|
|
||||||
run: echo "::set-output name=dir::$(yarn cache dir)"
|
|
||||||
- uses: actions/cache@v1
|
|
||||||
id: yarn-cache # use this to check for `cache-hit` (`steps.yarn-cache.outputs.cache-hit != 'true'`)
|
|
||||||
with:
|
|
||||||
path: ${{ steps.yarn-cache-dir-path.outputs.dir }}
|
|
||||||
key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-yarn-
|
|
||||||
- run: yarn install --frozen-lockfile
|
|
||||||
- run: yarn cypress install
|
|
||||||
- run: yarn build
|
|
||||||
env:
|
|
||||||
CI: false
|
|
||||||
REACT_APP_NETWORK_URL: "https://mainnet.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847"
|
|
||||||
- run: yarn integration-test
|
|
||||||
|
|
||||||
unit-tests:
|
|
||||||
name: Unit tests
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v1
|
|
||||||
- uses: actions/setup-node@v1
|
|
||||||
with:
|
|
||||||
node-version: '12'
|
|
||||||
always-auth: true
|
|
||||||
registry-url: https://registry.npmjs.org
|
|
||||||
|
|
||||||
- name: Get yarn cache directory path
|
|
||||||
id: yarn-cache-dir-path
|
|
||||||
run: echo "::set-output name=dir::$(yarn cache dir)"
|
|
||||||
- uses: actions/cache@v1
|
|
||||||
id: yarn-cache # use this to check for `cache-hit` (`steps.yarn-cache.outputs.cache-hit != 'true'`)
|
|
||||||
with:
|
|
||||||
path: ${{ steps.yarn-cache-dir-path.outputs.dir }}
|
|
||||||
key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-yarn-
|
|
||||||
- run: yarn install --frozen-lockfile
|
|
||||||
- run: yarn test
|
|
||||||
|
|
||||||
28
.github/workflows/unit-tests.yaml
vendored
Normal file
28
.github/workflows/unit-tests.yaml
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
name: Unit Tests
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
unit-tests:
|
||||||
|
name: Unit tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Set up node
|
||||||
|
uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: 14
|
||||||
|
registry-url: https://registry.npmjs.org
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: yarn install --frozen-lockfile
|
||||||
|
|
||||||
|
- name: Run unit tests
|
||||||
|
run: yarn test
|
||||||
@@ -1,8 +1,9 @@
|
|||||||
# Uniswap Interface
|
# Uniswap Interface
|
||||||
|
|
||||||
[](https://github.com/Uniswap/uniswap-interface/actions?query=workflow%3ALint)
|
[](https://github.com/Uniswap/uniswap-interface/actions/workflows/unit-tests.yaml)
|
||||||
[](https://github.com/Uniswap/uniswap-interface/actions?query=workflow%3ATests)
|
[](https://github.com/Uniswap/uniswap-interface/actions/workflows/integration-tests.yaml)
|
||||||
[](https://prettier.io/)
|
[](https://github.com/Uniswap/uniswap-interface/actions/workflows/lint.yml)
|
||||||
|
[](https://github.com/Uniswap/uniswap-interface/actions/workflows/release.yaml)
|
||||||
|
|
||||||
An open source interface for Uniswap -- a protocol for decentralized exchange of Ethereum tokens.
|
An open source interface for Uniswap -- a protocol for decentralized exchange of Ethereum tokens.
|
||||||
|
|
||||||
@@ -56,7 +57,7 @@ The interface will not work on other networks.
|
|||||||
|
|
||||||
## Contributions
|
## Contributions
|
||||||
|
|
||||||
**Please open all pull requests against the `master` branch.**
|
**Please open all pull requests against the `main` branch.**
|
||||||
CI checks will run against all PRs.
|
CI checks will run against all PRs.
|
||||||
|
|
||||||
## Accessing Uniswap Interface V1
|
## Accessing Uniswap Interface V1
|
||||||
|
|||||||
@@ -8,13 +8,15 @@ import { JsonRpcProvider } from '@ethersproject/providers'
|
|||||||
import { Wallet } from '@ethersproject/wallet'
|
import { Wallet } from '@ethersproject/wallet'
|
||||||
import { _Eip1193Bridge } from '@ethersproject/experimental/lib/eip1193-bridge'
|
import { _Eip1193Bridge } from '@ethersproject/experimental/lib/eip1193-bridge'
|
||||||
|
|
||||||
// never send real ether to this, obviously
|
const TEST_PRIVATE_KEY = Cypress.env('INTEGRATION_TEST_PRIVATE_KEY')
|
||||||
const PRIVATE_KEY_TEST_NEVER_USE = '0xad20c82497421e9784f18460ad2fe84f73569068e98e270b3e63743268af5763'
|
|
||||||
|
|
||||||
// address of the above key
|
// address of the above key
|
||||||
export const TEST_ADDRESS_NEVER_USE = '0x0fF2D1eFd7A57B7562b2bf27F3f37899dB27F4a5'
|
export const TEST_ADDRESS_NEVER_USE = new Wallet(TEST_PRIVATE_KEY).address
|
||||||
|
|
||||||
export const TEST_ADDRESS_NEVER_USE_SHORTENED = '0x0fF2...F4a5'
|
export const TEST_ADDRESS_NEVER_USE_SHORTENED = `${TEST_ADDRESS_NEVER_USE.substr(
|
||||||
|
0,
|
||||||
|
6
|
||||||
|
)}...${TEST_ADDRESS_NEVER_USE.substr(-4, 4)}`
|
||||||
|
|
||||||
class CustomizedBridge extends _Eip1193Bridge {
|
class CustomizedBridge extends _Eip1193Bridge {
|
||||||
async sendAsync(...args) {
|
async sendAsync(...args) {
|
||||||
@@ -75,7 +77,7 @@ Cypress.Commands.overwrite('visit', (original, url, options) => {
|
|||||||
options && options.onBeforeLoad && options.onBeforeLoad(win)
|
options && options.onBeforeLoad && options.onBeforeLoad(win)
|
||||||
win.localStorage.clear()
|
win.localStorage.clear()
|
||||||
const provider = new JsonRpcProvider('https://rinkeby.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847', 4)
|
const provider = new JsonRpcProvider('https://rinkeby.infura.io/v3/4bf032f2d38a4ed6bb975b80d6340847', 4)
|
||||||
const signer = new Wallet(PRIVATE_KEY_TEST_NEVER_USE, provider)
|
const signer = new Wallet(TEST_PRIVATE_KEY, provider)
|
||||||
win.ethereum = new CustomizedBridge(signer, provider)
|
win.ethereum = new CustomizedBridge(signer, provider)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -94,9 +94,6 @@
|
|||||||
"workbox-routing": "^6.1.0",
|
"workbox-routing": "^6.1.0",
|
||||||
"workbox-strategies": "^6.1.0"
|
"workbox-strategies": "^6.1.0"
|
||||||
},
|
},
|
||||||
"resolutions": {
|
|
||||||
"@walletconnect/web3-provider": "1.1.1-alpha.0"
|
|
||||||
},
|
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "react-scripts start",
|
"start": "react-scripts start",
|
||||||
"start:service-worker": "yarn build && yarn serve -s build",
|
"start:service-worker": "yarn build && yarn serve -s build",
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 2.8 KiB After Width: | Height: | Size: 2.6 KiB |
File diff suppressed because one or more lines are too long
|
Before Width: | Height: | Size: 7.9 KiB After Width: | Height: | Size: 14 KiB |
@@ -115,7 +115,13 @@ export default function Modal({
|
|||||||
{fadeTransition.map(
|
{fadeTransition.map(
|
||||||
({ item, key, props }) =>
|
({ item, key, props }) =>
|
||||||
item && (
|
item && (
|
||||||
<StyledDialogOverlay key={key} style={props} onDismiss={onDismiss} initialFocusRef={initialFocusRef}>
|
<StyledDialogOverlay
|
||||||
|
key={key}
|
||||||
|
style={props}
|
||||||
|
onDismiss={onDismiss}
|
||||||
|
initialFocusRef={initialFocusRef}
|
||||||
|
unstable_lockFocusAcrossFrames={false}
|
||||||
|
>
|
||||||
<StyledDialogContent
|
<StyledDialogContent
|
||||||
{...(isMobile
|
{...(isMobile
|
||||||
? {
|
? {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import React, { useContext, useRef, useState } from 'react'
|
import React, { useContext, useRef, useState } from 'react'
|
||||||
import { Settings, X } from 'react-feather'
|
import { Settings, X } from 'react-feather'
|
||||||
|
import ReactGA from 'react-ga'
|
||||||
import { Text } from 'rebass'
|
import { Text } from 'rebass'
|
||||||
import styled, { ThemeContext } from 'styled-components'
|
import styled, { ThemeContext } from 'styled-components'
|
||||||
import { useOnClickOutside } from '../../hooks/useOnClickOutside'
|
import { useOnClickOutside } from '../../hooks/useOnClickOutside'
|
||||||
@@ -236,7 +237,13 @@ export default function SettingsTab() {
|
|||||||
<Toggle
|
<Toggle
|
||||||
id="toggle-disable-multihop-button"
|
id="toggle-disable-multihop-button"
|
||||||
isActive={singleHopOnly}
|
isActive={singleHopOnly}
|
||||||
toggle={() => (singleHopOnly ? setSingleHopOnly(false) : setSingleHopOnly(true))}
|
toggle={() => {
|
||||||
|
ReactGA.event({
|
||||||
|
category: 'Routing',
|
||||||
|
action: singleHopOnly ? 'disable single hop' : 'enable single hop'
|
||||||
|
})
|
||||||
|
setSingleHopOnly(!singleHopOnly)
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</RowBetween>
|
</RowBetween>
|
||||||
</AutoColumn>
|
</AutoColumn>
|
||||||
|
|||||||
@@ -1,21 +1,21 @@
|
|||||||
import { Token } from '@uniswap/sdk'
|
import { Token } from '@uniswap/sdk'
|
||||||
import React, { useCallback } from 'react'
|
import React from 'react'
|
||||||
import Modal from '../Modal'
|
import Modal from '../Modal'
|
||||||
import { ImportToken } from 'components/SearchModal/ImportToken'
|
import { ImportToken } from 'components/SearchModal/ImportToken'
|
||||||
|
|
||||||
export default function TokenWarningModal({
|
export default function TokenWarningModal({
|
||||||
isOpen,
|
isOpen,
|
||||||
tokens,
|
tokens,
|
||||||
onConfirm
|
onConfirm,
|
||||||
|
onDismiss
|
||||||
}: {
|
}: {
|
||||||
isOpen: boolean
|
isOpen: boolean
|
||||||
tokens: Token[]
|
tokens: Token[]
|
||||||
onConfirm: () => void
|
onConfirm: () => void
|
||||||
|
onDismiss: () => void
|
||||||
}) {
|
}) {
|
||||||
const handleDismiss = useCallback(() => null, [])
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal isOpen={isOpen} onDismiss={handleDismiss} maxHeight={90}>
|
<Modal isOpen={isOpen} onDismiss={onDismiss} maxHeight={100}>
|
||||||
<ImportToken tokens={tokens} handleCurrencySelect={onConfirm} />
|
<ImportToken tokens={tokens} handleCurrencySelect={onConfirm} />
|
||||||
</Modal>
|
</Modal>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -105,7 +105,7 @@ export function AdvancedSwapDetails({ trade }: AdvancedSwapDetailsProps) {
|
|||||||
{!showRoute && (
|
{!showRoute && (
|
||||||
<AutoColumn style={{ padding: '12px 16px 0 16px' }}>
|
<AutoColumn style={{ padding: '12px 16px 0 16px' }}>
|
||||||
<InfoLink
|
<InfoLink
|
||||||
href={'https://uniswap.info/pair/' + trade.route.pairs[0].liquidityToken.address}
|
href={'https://info.uniswap.org/pair/' + trade.route.pairs[0].liquidityToken.address}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
>
|
>
|
||||||
View pair analytics ↗
|
View pair analytics ↗
|
||||||
|
|||||||
@@ -64,7 +64,6 @@ export default function UnsupportedCurrencyFooter({
|
|||||||
<AutoColumn gap="lg">
|
<AutoColumn gap="lg">
|
||||||
<RowBetween>
|
<RowBetween>
|
||||||
<TYPE.mediumHeader>Unsupported Assets</TYPE.mediumHeader>
|
<TYPE.mediumHeader>Unsupported Assets</TYPE.mediumHeader>
|
||||||
|
|
||||||
<CloseIcon onClick={() => setShowDetails(false)} />
|
<CloseIcon onClick={() => setShowDetails(false)} />
|
||||||
</RowBetween>
|
</RowBetween>
|
||||||
{tokens.map(token => {
|
{tokens.map(token => {
|
||||||
|
|||||||
@@ -6,10 +6,12 @@ import { PortisConnector } from '@web3-react/portis-connector'
|
|||||||
|
|
||||||
import { FortmaticConnector } from './Fortmatic'
|
import { FortmaticConnector } from './Fortmatic'
|
||||||
import { NetworkConnector } from './NetworkConnector'
|
import { NetworkConnector } from './NetworkConnector'
|
||||||
|
import UNISWAP_LOGO_URL from '../assets/svg/logo.svg'
|
||||||
|
|
||||||
const NETWORK_URL = process.env.REACT_APP_NETWORK_URL
|
const NETWORK_URL = process.env.REACT_APP_NETWORK_URL
|
||||||
const FORMATIC_KEY = process.env.REACT_APP_FORTMATIC_KEY
|
const FORMATIC_KEY = process.env.REACT_APP_FORTMATIC_KEY
|
||||||
const PORTIS_ID = process.env.REACT_APP_PORTIS_ID
|
const PORTIS_ID = process.env.REACT_APP_PORTIS_ID
|
||||||
|
const WALLETCONNECT_BRIDGE_URL = process.env.REACT_APP_WALLETCONNECT_BRIDGE_URL
|
||||||
|
|
||||||
export const NETWORK_CHAIN_ID: number = parseInt(process.env.REACT_APP_CHAIN_ID ?? '1')
|
export const NETWORK_CHAIN_ID: number = parseInt(process.env.REACT_APP_CHAIN_ID ?? '1')
|
||||||
|
|
||||||
@@ -33,7 +35,7 @@ export const injected = new InjectedConnector({
|
|||||||
// mainnet only
|
// mainnet only
|
||||||
export const walletconnect = new WalletConnectConnector({
|
export const walletconnect = new WalletConnectConnector({
|
||||||
rpc: { 1: NETWORK_URL },
|
rpc: { 1: NETWORK_URL },
|
||||||
bridge: 'https://bridge.walletconnect.org',
|
bridge: WALLETCONNECT_BRIDGE_URL,
|
||||||
qrcode: true,
|
qrcode: true,
|
||||||
pollingInterval: 15000
|
pollingInterval: 15000
|
||||||
})
|
})
|
||||||
@@ -54,6 +56,5 @@ export const portis = new PortisConnector({
|
|||||||
export const walletlink = new WalletLinkConnector({
|
export const walletlink = new WalletLinkConnector({
|
||||||
url: NETWORK_URL,
|
url: NETWORK_URL,
|
||||||
appName: 'Uniswap',
|
appName: 'Uniswap',
|
||||||
appLogoUrl:
|
appLogoUrl: UNISWAP_LOGO_URL
|
||||||
'https://mpng.pngfly.com/20181202/bex/kisspng-emoji-domain-unicorn-pin-badges-sticker-unicorn-tumblr-emoji-unicorn-iphoneemoji-5c046729264a77.5671679315437924251569.jpg'
|
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -14,13 +14,16 @@ type ChainTokenList = {
|
|||||||
readonly [chainId in ChainId]: Token[]
|
readonly [chainId in ChainId]: Token[]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const AMPL = new Token(ChainId.MAINNET, '0xD46bA6D942050d489DBd938a2C909A5d5039A161', 9, 'AMPL', 'Ampleforth')
|
||||||
export const DAI = new Token(ChainId.MAINNET, '0x6B175474E89094C44Da98b954EedeAC495271d0F', 18, 'DAI', 'Dai Stablecoin')
|
export const DAI = new Token(ChainId.MAINNET, '0x6B175474E89094C44Da98b954EedeAC495271d0F', 18, 'DAI', 'Dai Stablecoin')
|
||||||
export const USDC = new Token(ChainId.MAINNET, '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48', 6, 'USDC', 'USD//C')
|
export const USDC = new Token(ChainId.MAINNET, '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48', 6, 'USDC', 'USD//C')
|
||||||
export const USDT = new Token(ChainId.MAINNET, '0xdAC17F958D2ee523a2206206994597C13D831ec7', 6, 'USDT', 'Tether USD')
|
export const USDT = new Token(ChainId.MAINNET, '0xdAC17F958D2ee523a2206206994597C13D831ec7', 6, 'USDT', 'Tether USD')
|
||||||
export const COMP = new Token(ChainId.MAINNET, '0xc00e94Cb662C3520282E6f5717214004A7f26888', 18, 'COMP', 'Compound')
|
|
||||||
export const MKR = new Token(ChainId.MAINNET, '0x9f8F72aA9304c8B593d555F12eF6589cC3A579A2', 18, 'MKR', 'Maker')
|
|
||||||
export const AMPL = new Token(ChainId.MAINNET, '0xD46bA6D942050d489DBd938a2C909A5d5039A161', 9, 'AMPL', 'Ampleforth')
|
|
||||||
export const WBTC = new Token(ChainId.MAINNET, '0x2260FAC5E5542a773Aa44fBCfeDf7C193bc2C599', 8, 'WBTC', 'Wrapped BTC')
|
export const WBTC = new Token(ChainId.MAINNET, '0x2260FAC5E5542a773Aa44fBCfeDf7C193bc2C599', 8, 'WBTC', 'Wrapped BTC')
|
||||||
|
export const FEI = new Token(ChainId.MAINNET, '0x956F47F50A910163D8BF957Cf5846D573E7f87CA', 18, 'FEI', 'Fei USD')
|
||||||
|
export const TRIBE = new Token(ChainId.MAINNET, '0xc7283b66Eb1EB5FB86327f08e1B5816b0720212B', 18, 'TRIBE', 'Tribe')
|
||||||
|
export const FRAX = new Token(ChainId.MAINNET, '0x853d955aCEf822Db058eb8505911ED77F175b99e', 18, 'FRAX', 'Frax')
|
||||||
|
export const FXS = new Token(ChainId.MAINNET, '0x3432B6A60D23Ca0dFCa7761B7ab56459D9C964D0', 18, 'FXS', 'Frax Share')
|
||||||
|
export const renBTC = new Token(ChainId.MAINNET, '0xEB4C2781e4ebA804CE9a9803C67d0893436bB27D', 8, 'renBTC', 'renBTC')
|
||||||
|
|
||||||
// Block time here is slightly higher (~1s) than average in order to avoid ongoing proposals past the displayed time
|
// Block time here is slightly higher (~1s) than average in order to avoid ongoing proposals past the displayed time
|
||||||
export const AVERAGE_BLOCK_TIME_IN_SECS = 13
|
export const AVERAGE_BLOCK_TIME_IN_SECS = 13
|
||||||
@@ -62,7 +65,20 @@ const WETH_ONLY: ChainTokenList = {
|
|||||||
// used to construct intermediary pairs for trading
|
// used to construct intermediary pairs for trading
|
||||||
export const BASES_TO_CHECK_TRADES_AGAINST: ChainTokenList = {
|
export const BASES_TO_CHECK_TRADES_AGAINST: ChainTokenList = {
|
||||||
...WETH_ONLY,
|
...WETH_ONLY,
|
||||||
[ChainId.MAINNET]: [...WETH_ONLY[ChainId.MAINNET], DAI, USDC, USDT, COMP, MKR, WBTC]
|
[ChainId.MAINNET]: [...WETH_ONLY[ChainId.MAINNET], DAI, USDC, USDT, WBTC]
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ADDITIONAL_BASES: { [chainId in ChainId]?: { [tokenAddress: string]: Token[] } } = {
|
||||||
|
[ChainId.MAINNET]: {
|
||||||
|
'0xA948E86885e12Fb09AfEF8C52142EBDbDf73cD18': [new Token(ChainId.MAINNET, UNI_ADDRESS, 18, 'UNI', 'Uniswap')],
|
||||||
|
'0x561a4717537ff4AF5c687328c0f7E90a319705C0': [new Token(ChainId.MAINNET, UNI_ADDRESS, 18, 'UNI', 'Uniswap')],
|
||||||
|
[FEI.address]: [TRIBE],
|
||||||
|
[TRIBE.address]: [FEI],
|
||||||
|
[FRAX.address]: [FXS],
|
||||||
|
[FXS.address]: [FRAX],
|
||||||
|
[WBTC.address]: [renBTC],
|
||||||
|
[renBTC.address]: [WBTC]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,8 +1,4 @@
|
|||||||
// used to mark unsupported tokens, these are hosted lists of unsupported tokens
|
// used to mark unsupported tokens, these are hosted lists of unsupported tokens
|
||||||
/**
|
|
||||||
* @TODO add list from blockchain association
|
|
||||||
*/
|
|
||||||
export const UNSUPPORTED_LIST_URLS: string[] = []
|
|
||||||
|
|
||||||
const COMPOUND_LIST = 'https://raw.githubusercontent.com/compound-finance/token-list/master/compound.tokenlist.json'
|
const COMPOUND_LIST = 'https://raw.githubusercontent.com/compound-finance/token-list/master/compound.tokenlist.json'
|
||||||
const UMA_LIST = 'https://umaproject.org/uma.tokenlist.json'
|
const UMA_LIST = 'https://umaproject.org/uma.tokenlist.json'
|
||||||
@@ -17,6 +13,9 @@ const CMC_ALL_LIST = 'defi.cmc.eth'
|
|||||||
const CMC_STABLECOIN = 'stablecoin.cmc.eth'
|
const CMC_STABLECOIN = 'stablecoin.cmc.eth'
|
||||||
const KLEROS_LIST = 't2crtokens.eth'
|
const KLEROS_LIST = 't2crtokens.eth'
|
||||||
const GEMINI_LIST = 'https://www.gemini.com/uniswap/manifest.json'
|
const GEMINI_LIST = 'https://www.gemini.com/uniswap/manifest.json'
|
||||||
|
const BA_LIST = 'https://raw.githubusercontent.com/The-Blockchain-Association/sec-notice-list/master/ba-sec-list.json'
|
||||||
|
|
||||||
|
export const UNSUPPORTED_LIST_URLS: string[] = [BA_LIST]
|
||||||
|
|
||||||
// lower index == higher priority for token import
|
// lower index == higher priority for token import
|
||||||
export const DEFAULT_LIST_OF_LISTS: string[] = [
|
export const DEFAULT_LIST_OF_LISTS: string[] = [
|
||||||
|
|||||||
@@ -17,6 +17,14 @@
|
|||||||
"decimals": 6,
|
"decimals": 6,
|
||||||
"chainId": 1,
|
"chainId": 1,
|
||||||
"logoURI": "https://raw.githubusercontent.com/trustwallet/assets/master/blockchains/ethereum/assets/0x4922a015c4407F87432B179bb209e125432E4a2A/logo.png"
|
"logoURI": "https://raw.githubusercontent.com/trustwallet/assets/master/blockchains/ethereum/assets/0x4922a015c4407F87432B179bb209e125432E4a2A/logo.png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Grump Cat",
|
||||||
|
"address": "0x93B2FfF814FCaEFFB01406e80B4Ecd89Ca6A021b",
|
||||||
|
"symbol": "GRUMPY",
|
||||||
|
"decimals": 9,
|
||||||
|
"chainId": 1,
|
||||||
|
"logoURI": "https://raw.githubusercontent.com/trustwallet/assets/master/blockchains/ethereum/assets/0x4922a015c4407F87432B179bb209e125432E4a2A/logo.png"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,12 @@ import { Currency, CurrencyAmount, Pair, Token, Trade } from '@uniswap/sdk'
|
|||||||
import flatMap from 'lodash.flatmap'
|
import flatMap from 'lodash.flatmap'
|
||||||
import { useMemo } from 'react'
|
import { useMemo } from 'react'
|
||||||
|
|
||||||
import { BASES_TO_CHECK_TRADES_AGAINST, CUSTOM_BASES, BETTER_TRADE_LESS_HOPS_THRESHOLD } from '../constants'
|
import {
|
||||||
|
BASES_TO_CHECK_TRADES_AGAINST,
|
||||||
|
CUSTOM_BASES,
|
||||||
|
BETTER_TRADE_LESS_HOPS_THRESHOLD,
|
||||||
|
ADDITIONAL_BASES
|
||||||
|
} from '../constants'
|
||||||
import { PairState, usePairs } from '../data/Reserves'
|
import { PairState, usePairs } from '../data/Reserves'
|
||||||
import { wrappedCurrency } from '../utils/wrappedCurrency'
|
import { wrappedCurrency } from '../utils/wrappedCurrency'
|
||||||
|
|
||||||
@@ -14,17 +19,22 @@ import { useUserSingleHopOnly } from 'state/user/hooks'
|
|||||||
function useAllCommonPairs(currencyA?: Currency, currencyB?: Currency): Pair[] {
|
function useAllCommonPairs(currencyA?: Currency, currencyB?: Currency): Pair[] {
|
||||||
const { chainId } = useActiveWeb3React()
|
const { chainId } = useActiveWeb3React()
|
||||||
|
|
||||||
const bases: Token[] = chainId ? BASES_TO_CHECK_TRADES_AGAINST[chainId] : []
|
|
||||||
|
|
||||||
const [tokenA, tokenB] = chainId
|
const [tokenA, tokenB] = chainId
|
||||||
? [wrappedCurrency(currencyA, chainId), wrappedCurrency(currencyB, chainId)]
|
? [wrappedCurrency(currencyA, chainId), wrappedCurrency(currencyB, chainId)]
|
||||||
: [undefined, undefined]
|
: [undefined, undefined]
|
||||||
|
|
||||||
|
const bases: Token[] = useMemo(() => {
|
||||||
|
if (!chainId) return []
|
||||||
|
|
||||||
|
const common = BASES_TO_CHECK_TRADES_AGAINST[chainId] ?? []
|
||||||
|
const additionalA = tokenA ? ADDITIONAL_BASES[chainId]?.[tokenA.address] ?? [] : []
|
||||||
|
const additionalB = tokenB ? ADDITIONAL_BASES[chainId]?.[tokenB.address] ?? [] : []
|
||||||
|
|
||||||
|
return [...common, ...additionalA, ...additionalB]
|
||||||
|
}, [chainId, tokenA, tokenB])
|
||||||
|
|
||||||
const basePairs: [Token, Token][] = useMemo(
|
const basePairs: [Token, Token][] = useMemo(
|
||||||
() =>
|
() => flatMap(bases, (base): [Token, Token][] => bases.map(otherBase => [base, otherBase])),
|
||||||
flatMap(bases, (base): [Token, Token][] => bases.map(otherBase => [base, otherBase])).filter(
|
|
||||||
([t0, t1]) => t0.address !== t1.address
|
|
||||||
),
|
|
||||||
[bases]
|
[bases]
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -46,10 +56,9 @@ function useAllCommonPairs(currencyA?: Currency, currencyB?: Currency): Pair[] {
|
|||||||
.filter(([tokenA, tokenB]) => {
|
.filter(([tokenA, tokenB]) => {
|
||||||
if (!chainId) return true
|
if (!chainId) return true
|
||||||
const customBases = CUSTOM_BASES[chainId]
|
const customBases = CUSTOM_BASES[chainId]
|
||||||
if (!customBases) return true
|
|
||||||
|
|
||||||
const customBasesA: Token[] | undefined = customBases[tokenA.address]
|
const customBasesA: Token[] | undefined = customBases?.[tokenA.address]
|
||||||
const customBasesB: Token[] | undefined = customBases[tokenB.address]
|
const customBasesB: Token[] | undefined = customBases?.[tokenB.address]
|
||||||
|
|
||||||
if (!customBasesA && !customBasesB) return true
|
if (!customBasesA && !customBasesB) return true
|
||||||
|
|
||||||
@@ -150,18 +159,18 @@ export function useTradeExactOut(currencyIn?: Currency, currencyAmountOut?: Curr
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function useIsTransactionUnsupported(currencyIn?: Currency, currencyOut?: Currency): boolean {
|
export function useIsTransactionUnsupported(currencyIn?: Currency, currencyOut?: Currency): boolean {
|
||||||
const unsupportedToken: { [address: string]: Token } = useUnsupportedTokens()
|
const unsupportedTokens: { [address: string]: Token } = useUnsupportedTokens()
|
||||||
const { chainId } = useActiveWeb3React()
|
const { chainId } = useActiveWeb3React()
|
||||||
|
|
||||||
const tokenIn = wrappedCurrency(currencyIn, chainId)
|
const tokenIn = wrappedCurrency(currencyIn, chainId)
|
||||||
const tokenOut = wrappedCurrency(currencyOut, chainId)
|
const tokenOut = wrappedCurrency(currencyOut, chainId)
|
||||||
|
|
||||||
// if unsupported list loaded & either token on list, mark as unsupported
|
// if unsupported list loaded & either token on list, mark as unsupported
|
||||||
if (unsupportedToken) {
|
if (unsupportedTokens) {
|
||||||
if (tokenIn && Object.keys(unsupportedToken).includes(tokenIn.address)) {
|
if (tokenIn && Object.keys(unsupportedTokens).includes(tokenIn.address)) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
if (tokenOut && Object.keys(unsupportedToken).includes(tokenOut.address)) {
|
if (tokenOut && Object.keys(unsupportedTokens).includes(tokenOut.address)) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,8 +28,14 @@ if (!!window.ethereum) {
|
|||||||
|
|
||||||
const GOOGLE_ANALYTICS_ID: string | undefined = process.env.REACT_APP_GOOGLE_ANALYTICS_ID
|
const GOOGLE_ANALYTICS_ID: string | undefined = process.env.REACT_APP_GOOGLE_ANALYTICS_ID
|
||||||
if (typeof GOOGLE_ANALYTICS_ID === 'string') {
|
if (typeof GOOGLE_ANALYTICS_ID === 'string') {
|
||||||
ReactGA.initialize(GOOGLE_ANALYTICS_ID)
|
ReactGA.initialize(GOOGLE_ANALYTICS_ID, {
|
||||||
|
gaOptions: {
|
||||||
|
storage: 'none',
|
||||||
|
storeGac: false
|
||||||
|
}
|
||||||
|
})
|
||||||
ReactGA.set({
|
ReactGA.set({
|
||||||
|
anonymizeIp: true,
|
||||||
customBrowserType: !isMobile ? 'desktop' : 'web3' in window || 'ethereum' in window ? 'mobileWeb3' : 'mobileRegular'
|
customBrowserType: !isMobile ? 'desktop' : 'web3' in window || 'ethereum' in window ? 'mobileWeb3' : 'mobileRegular'
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -48,8 +48,9 @@ import Loader from '../../components/Loader'
|
|||||||
import { useIsTransactionUnsupported } from 'hooks/Trades'
|
import { useIsTransactionUnsupported } from 'hooks/Trades'
|
||||||
import UnsupportedCurrencyFooter from 'components/swap/UnsupportedCurrencyFooter'
|
import UnsupportedCurrencyFooter from 'components/swap/UnsupportedCurrencyFooter'
|
||||||
import { isTradeBetter } from 'utils/trades'
|
import { isTradeBetter } from 'utils/trades'
|
||||||
|
import { RouteComponentProps } from 'react-router-dom'
|
||||||
|
|
||||||
export default function Swap() {
|
export default function Swap({ history }: RouteComponentProps) {
|
||||||
const loadedUrlParams = useDefaultsFromURLSearch()
|
const loadedUrlParams = useDefaultsFromURLSearch()
|
||||||
|
|
||||||
// token warning stuff
|
// token warning stuff
|
||||||
@@ -97,6 +98,7 @@ export default function Swap() {
|
|||||||
currencies,
|
currencies,
|
||||||
inputError: swapInputError
|
inputError: swapInputError
|
||||||
} = useDerivedSwapInfo()
|
} = useDerivedSwapInfo()
|
||||||
|
|
||||||
const { wrapType, execute: onWrap, inputError: wrapInputError } = useWrapCallback(
|
const { wrapType, execute: onWrap, inputError: wrapInputError } = useWrapCallback(
|
||||||
currencies[Field.INPUT],
|
currencies[Field.INPUT],
|
||||||
currencies[Field.OUTPUT],
|
currencies[Field.OUTPUT],
|
||||||
@@ -142,6 +144,12 @@ export default function Swap() {
|
|||||||
[onUserInput]
|
[onUserInput]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// reset if they close warning without tokens in params
|
||||||
|
const handleDismissTokenWarning = useCallback(() => {
|
||||||
|
setDismissTokenWarning(true)
|
||||||
|
history.push('/swap/')
|
||||||
|
}, [history])
|
||||||
|
|
||||||
// modal and loading
|
// modal and loading
|
||||||
const [{ showConfirm, tradeToConfirm, swapErrorMessage, attemptingTxn, txHash }, setSwapState] = useState<{
|
const [{ showConfirm, tradeToConfirm, swapErrorMessage, attemptingTxn, txHash }, setSwapState] = useState<{
|
||||||
showConfirm: boolean
|
showConfirm: boolean
|
||||||
@@ -297,6 +305,7 @@ export default function Swap() {
|
|||||||
isOpen={importTokensNotInDefault.length > 0 && !dismissTokenWarning}
|
isOpen={importTokensNotInDefault.length > 0 && !dismissTokenWarning}
|
||||||
tokens={importTokensNotInDefault}
|
tokens={importTokensNotInDefault}
|
||||||
onConfirm={handleConfirmTokenWarning}
|
onConfirm={handleConfirmTokenWarning}
|
||||||
|
onDismiss={handleDismissTokenWarning}
|
||||||
/>
|
/>
|
||||||
<SwapPoolTabs active={'swap'} />
|
<SwapPoolTabs active={'swap'} />
|
||||||
<AppBody>
|
<AppBody>
|
||||||
|
|||||||
@@ -29,18 +29,16 @@ function fetchClaim(account: string, chainId: ChainId): Promise<UserClaimData |
|
|||||||
|
|
||||||
return (CLAIM_PROMISES[key] =
|
return (CLAIM_PROMISES[key] =
|
||||||
CLAIM_PROMISES[key] ??
|
CLAIM_PROMISES[key] ??
|
||||||
fetch(`https://gentle-frost-9e74.uniswap.workers.dev/${chainId}/${formatted}`)
|
fetch('https://merkle-drop-1.uniswap.workers.dev/', {
|
||||||
.then(res => {
|
body: JSON.stringify({ chainId, address: formatted }),
|
||||||
if (res.status === 200) {
|
headers: {
|
||||||
return res.json()
|
'Content-Type': 'application/json',
|
||||||
} else {
|
'Referrer-Policy': 'no-referrer'
|
||||||
console.debug(`No claim for account ${formatted} on chain ID ${chainId}`)
|
},
|
||||||
return null
|
method: 'POST'
|
||||||
}
|
})
|
||||||
})
|
.then(res => (res.ok ? res.json() : console.log(`No claim for account ${formatted} on chain ID ${chainId}`)))
|
||||||
.catch(error => {
|
.catch(error => console.error('Failed to get claim data', error)))
|
||||||
console.error('Failed to get claim data', error)
|
|
||||||
}))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// parse distributorContract blob and detect if user has claim data
|
// parse distributorContract blob and detect if user has claim data
|
||||||
|
|||||||
@@ -103,10 +103,8 @@ function combineMaps(map1: TokenAddressMap, map2: TokenAddressMap): TokenAddress
|
|||||||
// merge tokens contained within lists from urls
|
// merge tokens contained within lists from urls
|
||||||
function useCombinedTokenMapFromUrls(urls: string[] | undefined): TokenAddressMap {
|
function useCombinedTokenMapFromUrls(urls: string[] | undefined): TokenAddressMap {
|
||||||
const lists = useAllLists()
|
const lists = useAllLists()
|
||||||
|
|
||||||
return useMemo(() => {
|
return useMemo(() => {
|
||||||
if (!urls) return EMPTY_LIST
|
if (!urls) return EMPTY_LIST
|
||||||
|
|
||||||
return (
|
return (
|
||||||
urls
|
urls
|
||||||
.slice()
|
.slice()
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { DEFAULT_ACTIVE_LIST_URLS } from './../../constants/lists'
|
import { DEFAULT_ACTIVE_LIST_URLS, UNSUPPORTED_LIST_URLS } from './../../constants/lists'
|
||||||
import { createReducer } from '@reduxjs/toolkit'
|
import { createReducer } from '@reduxjs/toolkit'
|
||||||
import { getVersionUpgrade, VersionUpgrade } from '@uniswap/token-lists'
|
import { getVersionUpgrade, VersionUpgrade } from '@uniswap/token-lists'
|
||||||
import { TokenList } from '@uniswap/token-lists/dist/types'
|
import { TokenList } from '@uniswap/token-lists/dist/types'
|
||||||
@@ -36,7 +36,7 @@ type Mutable<T> = { -readonly [P in keyof T]: T[P] extends ReadonlyArray<infer U
|
|||||||
const initialState: ListsState = {
|
const initialState: ListsState = {
|
||||||
lastInitializedDefaultListOfLists: DEFAULT_LIST_OF_LISTS,
|
lastInitializedDefaultListOfLists: DEFAULT_LIST_OF_LISTS,
|
||||||
byUrl: {
|
byUrl: {
|
||||||
...DEFAULT_LIST_OF_LISTS.reduce<Mutable<ListsState['byUrl']>>((memo, listUrl) => {
|
...DEFAULT_LIST_OF_LISTS.concat(...UNSUPPORTED_LIST_URLS).reduce<Mutable<ListsState['byUrl']>>((memo, listUrl) => {
|
||||||
memo[listUrl] = NEW_LIST_STATE
|
memo[listUrl] = NEW_LIST_STATE
|
||||||
return memo
|
return memo
|
||||||
}, {})
|
}, {})
|
||||||
|
|||||||
@@ -7,9 +7,10 @@ import { useFetchListCallback } from '../../hooks/useFetchListCallback'
|
|||||||
import useInterval from '../../hooks/useInterval'
|
import useInterval from '../../hooks/useInterval'
|
||||||
import useIsWindowVisible from '../../hooks/useIsWindowVisible'
|
import useIsWindowVisible from '../../hooks/useIsWindowVisible'
|
||||||
import { AppDispatch } from '../index'
|
import { AppDispatch } from '../index'
|
||||||
import { acceptListUpdate } from './actions'
|
import { acceptListUpdate, removeList } from './actions'
|
||||||
import { useActiveListUrls } from './hooks'
|
import { useActiveListUrls } from './hooks'
|
||||||
import { useAllInactiveTokens } from 'hooks/Tokens'
|
import { useAllInactiveTokens } from 'hooks/Tokens'
|
||||||
|
import { UNSUPPORTED_LIST_URLS } from 'constants/lists'
|
||||||
|
|
||||||
export default function Updater(): null {
|
export default function Updater(): null {
|
||||||
const { library } = useActiveWeb3React()
|
const { library } = useActiveWeb3React()
|
||||||
@@ -34,6 +35,13 @@ export default function Updater(): null {
|
|||||||
// fetch all lists every 10 minutes, but only after we initialize library
|
// fetch all lists every 10 minutes, but only after we initialize library
|
||||||
useInterval(fetchAllListsCallback, library ? 1000 * 60 * 10 : null)
|
useInterval(fetchAllListsCallback, library ? 1000 * 60 * 10 : null)
|
||||||
|
|
||||||
|
// hot fix for fetching error
|
||||||
|
useEffect(() => {
|
||||||
|
if (lists['https://tokens.coingecko.com/uniswap/all.json']) {
|
||||||
|
dispatch(removeList('https://tokens.coingecko.com/uniswap/all.json'))
|
||||||
|
}
|
||||||
|
}, [dispatch, lists])
|
||||||
|
|
||||||
// whenever a list is not loaded and not loading, try again to load it
|
// whenever a list is not loaded and not loading, try again to load it
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
Object.keys(lists).forEach(listUrl => {
|
Object.keys(lists).forEach(listUrl => {
|
||||||
@@ -44,6 +52,16 @@ export default function Updater(): null {
|
|||||||
})
|
})
|
||||||
}, [dispatch, fetchList, library, lists])
|
}, [dispatch, fetchList, library, lists])
|
||||||
|
|
||||||
|
// if any lists from unsupported lists are loaded, check them too (in case new updates since last visit)
|
||||||
|
useEffect(() => {
|
||||||
|
Object.keys(UNSUPPORTED_LIST_URLS).forEach(listUrl => {
|
||||||
|
const list = lists[listUrl]
|
||||||
|
if (!list || (!list.current && !list.loadingRequestId && !list.error)) {
|
||||||
|
fetchList(listUrl).catch(error => console.debug('list added fetching error', error))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, [dispatch, fetchList, library, lists])
|
||||||
|
|
||||||
// automatically update lists if versions are minor/patch
|
// automatically update lists if versions are minor/patch
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
Object.keys(lists).forEach(listUrl => {
|
Object.keys(lists).forEach(listUrl => {
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import { ChainId, Pair, Token } from '@uniswap/sdk'
|
import { ChainId, Pair, Token } from '@uniswap/sdk'
|
||||||
import flatMap from 'lodash.flatmap'
|
import flatMap from 'lodash.flatmap'
|
||||||
import ReactGA from 'react-ga'
|
|
||||||
import { useCallback, useMemo } from 'react'
|
import { useCallback, useMemo } from 'react'
|
||||||
import { shallowEqual, useDispatch, useSelector } from 'react-redux'
|
import { shallowEqual, useDispatch, useSelector } from 'react-redux'
|
||||||
import { BASES_TO_TRACK_LIQUIDITY_FOR, PINNED_PAIRS } from '../../constants'
|
import { BASES_TO_TRACK_LIQUIDITY_FOR, PINNED_PAIRS } from '../../constants'
|
||||||
@@ -92,10 +91,6 @@ export function useUserSingleHopOnly(): [boolean, (newSingleHopOnly: boolean) =>
|
|||||||
|
|
||||||
const setSingleHopOnly = useCallback(
|
const setSingleHopOnly = useCallback(
|
||||||
(newSingleHopOnly: boolean) => {
|
(newSingleHopOnly: boolean) => {
|
||||||
ReactGA.event({
|
|
||||||
category: 'Routing',
|
|
||||||
action: newSingleHopOnly ? 'enable single hop' : 'disable single hop'
|
|
||||||
})
|
|
||||||
dispatch(updateUserSingleHopOnly({ userSingleHopOnly: newSingleHopOnly }))
|
dispatch(updateUserSingleHopOnly({ userSingleHopOnly: newSingleHopOnly }))
|
||||||
},
|
},
|
||||||
[dispatch]
|
[dispatch]
|
||||||
@@ -162,7 +157,7 @@ export function useUserAddedTokens(): Token[] {
|
|||||||
|
|
||||||
return useMemo(() => {
|
return useMemo(() => {
|
||||||
if (!chainId) return []
|
if (!chainId) return []
|
||||||
return Object.values(serializedTokensMap[chainId as ChainId] ?? {}).map(deserializeToken)
|
return Object.values(serializedTokensMap?.[chainId as ChainId] ?? {}).map(deserializeToken)
|
||||||
}, [serializedTokensMap, chainId])
|
}, [serializedTokensMap, chainId])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -111,11 +111,17 @@ export default createReducer(initialState, builder =>
|
|||||||
state.userSingleHopOnly = action.payload.userSingleHopOnly
|
state.userSingleHopOnly = action.payload.userSingleHopOnly
|
||||||
})
|
})
|
||||||
.addCase(addSerializedToken, (state, { payload: { serializedToken } }) => {
|
.addCase(addSerializedToken, (state, { payload: { serializedToken } }) => {
|
||||||
|
if (!state.tokens) {
|
||||||
|
state.tokens = {}
|
||||||
|
}
|
||||||
state.tokens[serializedToken.chainId] = state.tokens[serializedToken.chainId] || {}
|
state.tokens[serializedToken.chainId] = state.tokens[serializedToken.chainId] || {}
|
||||||
state.tokens[serializedToken.chainId][serializedToken.address] = serializedToken
|
state.tokens[serializedToken.chainId][serializedToken.address] = serializedToken
|
||||||
state.timestamp = currentTimestamp()
|
state.timestamp = currentTimestamp()
|
||||||
})
|
})
|
||||||
.addCase(removeSerializedToken, (state, { payload: { address, chainId } }) => {
|
.addCase(removeSerializedToken, (state, { payload: { address, chainId } }) => {
|
||||||
|
if (!state.tokens) {
|
||||||
|
state.tokens = {}
|
||||||
|
}
|
||||||
state.tokens[chainId] = state.tokens[chainId] || {}
|
state.tokens[chainId] = state.tokens[chainId] || {}
|
||||||
delete state.tokens[chainId][address]
|
delete state.tokens[chainId][address]
|
||||||
state.timestamp = currentTimestamp()
|
state.timestamp = currentTimestamp()
|
||||||
|
|||||||
Reference in New Issue
Block a user