Mp 2757 token input errors (#235)
* Refactor TokenInput * add TokenInput test + warning * change title assigned select * add unit tests for Card * remove marketAssets from broadcast store
This commit is contained in:
parent
a3b436f8dd
commit
de89ecb7ed
51
__tests__/components/Card.test.tsx
Normal file
51
__tests__/components/Card.test.tsx
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
import { render, screen } from '@testing-library/react'
|
||||||
|
import Card from 'components/Card'
|
||||||
|
import { shallow } from 'enzyme'
|
||||||
|
import Text from 'components/Text'
|
||||||
|
import Button from 'components/Button'
|
||||||
|
|
||||||
|
describe('<Card />', () => {
|
||||||
|
const defaultProps = {
|
||||||
|
children: <></>,
|
||||||
|
}
|
||||||
|
|
||||||
|
it('should render', () => {
|
||||||
|
const { container } = render(<Card {...defaultProps} />)
|
||||||
|
expect(container).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `className` prop correctly', () => {
|
||||||
|
const testClass = 'test-class'
|
||||||
|
const { container } = render(<Card {...defaultProps} className={testClass} />)
|
||||||
|
expect(container.querySelector('section')).toHaveClass(testClass)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `contentClassName` prop correctly', () => {
|
||||||
|
const testClass = 'test-class'
|
||||||
|
const { container } = render(<Card {...defaultProps} contentClassName={testClass} />)
|
||||||
|
|
||||||
|
expect(container.querySelector('div')).toHaveClass(testClass)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `title` prop as string correctly', () => {
|
||||||
|
const testTitle = 'test-title'
|
||||||
|
const wrapper = shallow(<Card {...defaultProps} title={testTitle} />)
|
||||||
|
const textComponent = wrapper.find(Text).at(0)
|
||||||
|
const text = textComponent.dive().text()
|
||||||
|
|
||||||
|
expect(text).toBe(testTitle)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `title` prop as element correctly', () => {
|
||||||
|
const testTitle = <p className='test-class'>Test title</p>
|
||||||
|
const wrapper = shallow(<Card {...defaultProps} title={testTitle} />)
|
||||||
|
expect(wrapper.find('p.test-class')).toHaveLength(1)
|
||||||
|
expect(wrapper.find(Text)).toHaveLength(0)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `id` prop as element correctly', () => {
|
||||||
|
const testId = 'test-id'
|
||||||
|
const wrapper = shallow(<Card {...defaultProps} id={testId} />)
|
||||||
|
expect(wrapper.find(`section#${testId}`).at(0)).toHaveLength(1)
|
||||||
|
})
|
||||||
|
})
|
74
__tests__/components/TokenInput.test.tsx
Normal file
74
__tests__/components/TokenInput.test.tsx
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
import { render, fireEvent } from '@testing-library/react'
|
||||||
|
import BigNumber from 'bignumber.js'
|
||||||
|
import TokenInput from 'components/TokenInput'
|
||||||
|
import { ASSETS } from 'constants/assets'
|
||||||
|
|
||||||
|
describe('<TokenInput />', () => {
|
||||||
|
const asset = ASSETS[0]
|
||||||
|
const defaultProps = {
|
||||||
|
amount: new BigNumber(1),
|
||||||
|
asset,
|
||||||
|
max: new BigNumber(100),
|
||||||
|
onChangeAsset: jest.fn(),
|
||||||
|
onChange: jest.fn(),
|
||||||
|
}
|
||||||
|
|
||||||
|
it('should render', () => {
|
||||||
|
const { container } = render(<TokenInput {...defaultProps} />)
|
||||||
|
expect(container).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `className` prop correctly', () => {
|
||||||
|
const testClass = 'test-class'
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} className={testClass} />)
|
||||||
|
expect(getByTestId('token-input-component')).toHaveClass(testClass)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `disabled` prop correctly', () => {
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} disabled />)
|
||||||
|
expect(getByTestId('token-input-component')).toHaveClass('pointer-events-none opacity-50')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `maxText` prop correctly', () => {
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} maxText='Max' />)
|
||||||
|
expect(getByTestId('token-input-max-button')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should handle `warning` prop correctly', () => {
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} warning='Warning' />)
|
||||||
|
expect(getByTestId('token-input-wrapper')).toHaveClass('border-warning')
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('should render the max button', () => {
|
||||||
|
it('when `maxText` prop is defined', () => {
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} maxText='Max' />)
|
||||||
|
expect(getByTestId('token-input-max-button')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
it('not when `maxText` prop is undefined', () => {
|
||||||
|
const { queryByTestId } = render(<TokenInput {...defaultProps} />)
|
||||||
|
expect(queryByTestId('token-input-max-button')).not.toBeInTheDocument()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('should render <Select />', () => {
|
||||||
|
it('when `hasSelect` prop is true and balances is defined', () => {
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} balances={[]} hasSelect />)
|
||||||
|
expect(getByTestId('select-component')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
it('not when `hasSelect` prop is true and balances is not defined', () => {
|
||||||
|
const { queryByTestId } = render(<TokenInput {...defaultProps} hasSelect />)
|
||||||
|
expect(queryByTestId('select-component')).not.toBeInTheDocument()
|
||||||
|
})
|
||||||
|
it('not when `hasSelect` prop is false and balances is defined', () => {
|
||||||
|
const { queryByTestId } = render(<TokenInput {...defaultProps} balances={[]} />)
|
||||||
|
expect(queryByTestId('select-component')).not.toBeInTheDocument()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should call onMaxBtnClick when the user clicks on max button', () => {
|
||||||
|
const { getByTestId } = render(<TokenInput {...defaultProps} maxText='max' />)
|
||||||
|
const maxBtn = getByTestId('token-input-max-button')
|
||||||
|
fireEvent.click(maxBtn)
|
||||||
|
expect(defaultProps.onChange).toBeCalledWith(defaultProps.max)
|
||||||
|
})
|
||||||
|
})
|
@ -85,6 +85,7 @@ export default function FundAccount(props: Props) {
|
|||||||
your Osmosis address has no assets.
|
your Osmosis address has no assets.
|
||||||
</Text>
|
</Text>
|
||||||
<TokenInputWithSlider
|
<TokenInputWithSlider
|
||||||
|
asset={asset}
|
||||||
onChange={onChangeAmount}
|
onChange={onChangeAmount}
|
||||||
onChangeAsset={onChangeAsset}
|
onChangeAsset={onChangeAsset}
|
||||||
amount={amount}
|
amount={amount}
|
||||||
|
@ -33,6 +33,7 @@ interface Props {
|
|||||||
iconClassName?: string
|
iconClassName?: string
|
||||||
hasSubmenu?: boolean
|
hasSubmenu?: boolean
|
||||||
hasFocus?: boolean
|
hasFocus?: boolean
|
||||||
|
dataTestId?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
const Button = React.forwardRef(function Button(
|
const Button = React.forwardRef(function Button(
|
||||||
@ -52,6 +53,7 @@ const Button = React.forwardRef(function Button(
|
|||||||
iconClassName,
|
iconClassName,
|
||||||
hasSubmenu,
|
hasSubmenu,
|
||||||
hasFocus,
|
hasFocus,
|
||||||
|
dataTestId,
|
||||||
}: Props,
|
}: Props,
|
||||||
ref,
|
ref,
|
||||||
) {
|
) {
|
||||||
@ -99,6 +101,7 @@ const Button = React.forwardRef(function Button(
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
|
data-testid={dataTestId}
|
||||||
className={buttonClassNames}
|
className={buttonClassNames}
|
||||||
id={id}
|
id={id}
|
||||||
ref={ref as LegacyRef<HTMLButtonElement>}
|
ref={ref as LegacyRef<HTMLButtonElement>}
|
||||||
|
@ -9,7 +9,6 @@ interface Props {
|
|||||||
contentClassName?: string
|
contentClassName?: string
|
||||||
title?: string | ReactElement
|
title?: string | ReactElement
|
||||||
id?: string
|
id?: string
|
||||||
onClick?: (e: React.MouseEvent) => void
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Card(props: Props) {
|
export default function Card(props: Props) {
|
||||||
|
3
src/components/Icons/ExclamationMarkTriangle.svg
Normal file
3
src/components/Icons/ExclamationMarkTriangle.svg
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<svg width="22" height="20" viewBox="0 0 22 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M10.9998 8.00023V12.0002M10.9998 16.0002H11.0098M9.61507 2.89195L1.39019 17.0986C0.933982 17.8866 0.70588 18.2806 0.739593 18.6039C0.768998 18.886 0.916769 19.1423 1.14613 19.309C1.40908 19.5002 1.86435 19.5002 2.77487 19.5002H19.2246C20.1352 19.5002 20.5904 19.5002 20.8534 19.309C21.0827 19.1423 21.2305 18.886 21.2599 18.6039C21.2936 18.2806 21.0655 17.8866 20.6093 17.0986L12.3844 2.89195C11.9299 2.10679 11.7026 1.71421 11.4061 1.58235C11.1474 1.46734 10.8521 1.46734 10.5935 1.58235C10.2969 1.71421 10.0696 2.10679 9.61507 2.89195Z" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 722 B |
@ -15,6 +15,7 @@ export { default as Copy } from 'components/Icons/Copy.svg'
|
|||||||
export { default as Cross } from 'components/Icons/Cross.svg'
|
export { default as Cross } from 'components/Icons/Cross.svg'
|
||||||
export { default as CrossCircled } from 'components/Icons/CrossCircled.svg'
|
export { default as CrossCircled } from 'components/Icons/CrossCircled.svg'
|
||||||
export { default as ExclamationMarkCircled } from 'components/Icons/ExclamationMarkCircled.svg'
|
export { default as ExclamationMarkCircled } from 'components/Icons/ExclamationMarkCircled.svg'
|
||||||
|
export { default as ExclamationMarkTriangle } from 'components/Icons/ExclamationMarkTriangle.svg'
|
||||||
export { default as ExternalLink } from 'components/Icons/ExternalLink.svg'
|
export { default as ExternalLink } from 'components/Icons/ExternalLink.svg'
|
||||||
export { default as Gear } from 'components/Icons/Gear.svg'
|
export { default as Gear } from 'components/Icons/Gear.svg'
|
||||||
export { default as Heart } from 'components/Icons/Heart.svg'
|
export { default as Heart } from 'components/Icons/Heart.svg'
|
||||||
|
@ -94,9 +94,10 @@ export default function FundWithdrawModalContent(props: Props) {
|
|||||||
onChangeAsset={setCurrentAsset}
|
onChangeAsset={setCurrentAsset}
|
||||||
amount={amount}
|
amount={amount}
|
||||||
max={max}
|
max={max}
|
||||||
balances={props.isFunding ? balances : props.account.deposits ?? []}
|
balances={props.isFunding ? balances : props.account.deposits}
|
||||||
accountId={!props.isFunding ? props.account.id : undefined}
|
accountId={!props.isFunding ? props.account.id : undefined}
|
||||||
hasSelect
|
hasSelect
|
||||||
|
maxText='Max'
|
||||||
/>
|
/>
|
||||||
<Divider />
|
<Divider />
|
||||||
<Button
|
<Button
|
||||||
|
@ -97,6 +97,10 @@ export default function VaultDeposit(props: Props) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getWarningText(asset: Asset) {
|
||||||
|
return `You don't have ${asset.symbol} balance in your account. Toggle custom amount to deposit.`
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='flex h-full flex-col justify-between gap-6 p-4'>
|
<div className='flex h-full flex-col justify-between gap-6 p-4'>
|
||||||
<TokenInput
|
<TokenInput
|
||||||
@ -105,6 +109,7 @@ export default function VaultDeposit(props: Props) {
|
|||||||
max={primaryMax}
|
max={primaryMax}
|
||||||
maxText='Balance'
|
maxText='Balance'
|
||||||
asset={props.primaryAsset}
|
asset={props.primaryAsset}
|
||||||
|
warning={primaryMax.isZero() ? getWarningText(props.primaryAsset) : undefined}
|
||||||
/>
|
/>
|
||||||
{!isCustomAmount && <Slider value={percentage} onChange={onChangeSlider} />}
|
{!isCustomAmount && <Slider value={percentage} onChange={onChangeSlider} />}
|
||||||
<TokenInput
|
<TokenInput
|
||||||
@ -113,6 +118,7 @@ export default function VaultDeposit(props: Props) {
|
|||||||
max={secondaryMax}
|
max={secondaryMax}
|
||||||
maxText='Balance'
|
maxText='Balance'
|
||||||
asset={props.secondaryAsset}
|
asset={props.secondaryAsset}
|
||||||
|
warning={secondaryMax.isZero() ? getWarningText(props.secondaryAsset) : undefined}
|
||||||
/>
|
/>
|
||||||
<Divider />
|
<Divider />
|
||||||
<div className='flex justify-between'>
|
<div className='flex justify-between'>
|
||||||
|
@ -46,6 +46,7 @@ export default function Option(props: Props) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
|
data-testid='option-component'
|
||||||
className={classNames(
|
className={classNames(
|
||||||
'grid grid-flow-row grid-cols-5 grid-rows-2 py-3.5 pr-4',
|
'grid grid-flow-row grid-cols-5 grid-rows-2 py-3.5 pr-4',
|
||||||
'border-b border-b-white/20 last:border-none',
|
'border-b border-b-white/20 last:border-none',
|
||||||
|
@ -48,6 +48,7 @@ export default function Select(props: Props) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
|
data-testid='select-component'
|
||||||
className={classNames(
|
className={classNames(
|
||||||
props.isParent && 'relative',
|
props.isParent && 'relative',
|
||||||
'flex min-w-fit items-center gap-2',
|
'flex min-w-fit items-center gap-2',
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import BigNumber from 'bignumber.js'
|
import BigNumber from 'bignumber.js'
|
||||||
import classNames from 'classnames'
|
import classNames from 'classnames'
|
||||||
import Image from 'next/image'
|
import Image from 'next/image'
|
||||||
import { useCallback, useEffect, useState } from 'react'
|
|
||||||
|
|
||||||
import DisplayCurrency from 'components/DisplayCurrency'
|
import DisplayCurrency from 'components/DisplayCurrency'
|
||||||
import NumberInput from 'components/NumberInput'
|
import NumberInput from 'components/NumberInput'
|
||||||
@ -12,107 +11,96 @@ import useStore from 'store'
|
|||||||
import { BN } from 'utils/helpers'
|
import { BN } from 'utils/helpers'
|
||||||
import { FormattedNumber } from 'components/FormattedNumber'
|
import { FormattedNumber } from 'components/FormattedNumber'
|
||||||
import Button from 'components/Button'
|
import Button from 'components/Button'
|
||||||
|
import { ExclamationMarkTriangle } from 'components/Icons'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
amount: BigNumber
|
amount: BigNumber
|
||||||
onChange: (amount: BigNumber) => void
|
|
||||||
className?: string
|
|
||||||
disabled?: boolean
|
|
||||||
balances?: Coin[] | null
|
|
||||||
accountId?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface SingleProps extends Props {
|
|
||||||
asset: Asset
|
asset: Asset
|
||||||
max: BigNumber
|
max: BigNumber
|
||||||
maxText: string
|
onChange: (amount: BigNumber) => void
|
||||||
|
accountId?: string
|
||||||
|
balances?: Coin[]
|
||||||
|
className?: string
|
||||||
|
disabled?: boolean
|
||||||
hasSelect?: boolean
|
hasSelect?: boolean
|
||||||
onChangeAsset?: (asset: Asset, max: BigNumber) => void
|
|
||||||
}
|
|
||||||
|
|
||||||
interface SelectProps extends Props {
|
|
||||||
asset?: Asset
|
|
||||||
max?: BigNumber
|
|
||||||
maxText?: string
|
maxText?: string
|
||||||
hasSelect: boolean
|
warning?: string
|
||||||
onChangeAsset: (asset: Asset, max: BigNumber) => void
|
onChangeAsset?: (asset: Asset) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function TokenInput(props: SingleProps | SelectProps) {
|
export default function TokenInput(props: Props) {
|
||||||
const baseCurrency = useStore((s) => s.baseCurrency)
|
const baseCurrency = useStore((s) => s.baseCurrency)
|
||||||
const [asset, setAsset] = useState<Asset>(props.asset ? props.asset : baseCurrency)
|
|
||||||
const [coin, setCoin] = useState<Coin>({
|
|
||||||
denom: props.asset ? props.asset.denom : baseCurrency.denom,
|
|
||||||
amount: '0',
|
|
||||||
})
|
|
||||||
|
|
||||||
// TODO: Refactor the useEffect
|
|
||||||
useEffect(() => {
|
|
||||||
props.onChangeAsset && props.onChangeAsset(asset, coin ? BN(coin.amount) : BN(0))
|
|
||||||
}, [coin, asset])
|
|
||||||
|
|
||||||
const updateAsset = useCallback(
|
|
||||||
(coinDenom: string) => {
|
|
||||||
const newAsset = ASSETS.find((asset) => asset.denom === coinDenom) ?? baseCurrency
|
|
||||||
const newCoin = props.balances?.find((coin) => coin.denom === coinDenom)
|
|
||||||
setAsset(newAsset)
|
|
||||||
setCoin(newCoin ?? { denom: coinDenom, amount: '0' })
|
|
||||||
},
|
|
||||||
[props.balances, baseCurrency],
|
|
||||||
)
|
|
||||||
|
|
||||||
function onMaxBtnClick() {
|
function onMaxBtnClick() {
|
||||||
if (!props.max) return
|
|
||||||
props.onChange(BN(props.max))
|
props.onChange(BN(props.max))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function onChangeAsset(denom: string) {
|
||||||
|
if (!props.onChangeAsset) return
|
||||||
|
const newAsset = ASSETS.find((asset) => asset.denom === denom) ?? baseCurrency
|
||||||
|
props.onChangeAsset(newAsset)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
|
data-testid='token-input-component'
|
||||||
className={classNames(
|
className={classNames(
|
||||||
'flex w-full flex-col gap-2 transition-opacity',
|
'flex w-full flex-col gap-2 transition-opacity',
|
||||||
props.className,
|
props.className,
|
||||||
props.disabled && 'pointer-events-none opacity-50',
|
props.disabled && 'pointer-events-none opacity-50',
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
<div className='relative isolate z-40 box-content flex h-11 w-full rounded-sm border border-white/20 bg-white/5'>
|
<div
|
||||||
|
data-testid='token-input-wrapper'
|
||||||
|
className={classNames(
|
||||||
|
'relative isolate z-40 box-content flex h-11 w-full rounded-sm border bg-white/5',
|
||||||
|
props.warning ? 'border-warning' : 'border-white/20',
|
||||||
|
)}
|
||||||
|
>
|
||||||
{props.hasSelect && props.balances ? (
|
{props.hasSelect && props.balances ? (
|
||||||
<Select
|
<Select
|
||||||
options={props.balances}
|
options={props.balances}
|
||||||
defaultValue={coin.denom}
|
defaultValue={props.asset.denom}
|
||||||
onChange={(value) => updateAsset(value)}
|
onChange={onChangeAsset}
|
||||||
title={props.accountId ? `Account ${props.accountId}` : 'Your Wallet'}
|
title={props.accountId ? `Account ${props.accountId}` : 'Your Wallet'}
|
||||||
className='border-r border-white/20 bg-white/5'
|
className='border-r border-white/20 bg-white/5'
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<div className='flex min-w-fit items-center gap-2 border-r border-white/20 bg-white/5 p-3'>
|
<div className='flex min-w-fit items-center gap-2 border-r border-white/20 bg-white/5 p-3'>
|
||||||
<Image src={asset.logo} alt='token' width={20} height={20} />
|
<Image src={props.asset.logo} alt='token' width={20} height={20} />
|
||||||
<Text>{asset.symbol}</Text>
|
<Text>{props.asset.symbol}</Text>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<NumberInput
|
<NumberInput
|
||||||
disabled={props.disabled}
|
disabled={props.disabled}
|
||||||
asset={asset}
|
asset={props.asset}
|
||||||
maxDecimals={asset.decimals}
|
maxDecimals={props.asset.decimals}
|
||||||
onChange={props.onChange}
|
onChange={props.onChange}
|
||||||
amount={props.amount}
|
amount={props.amount}
|
||||||
max={props.max}
|
max={props.max}
|
||||||
className='border-none p-3'
|
className='border-none p-3'
|
||||||
/>
|
/>
|
||||||
|
{props.warning && (
|
||||||
|
<div className='grid items-center px-2'>
|
||||||
|
<ExclamationMarkTriangle className='text-warning' />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='flex'>
|
<div className='flex'>
|
||||||
<div className='flex flex-1 items-center'>
|
<div className='flex flex-1 items-center'>
|
||||||
{props.max && props.maxText && (
|
{props.maxText && (
|
||||||
<>
|
<>
|
||||||
<Text size='xs' className='mr-1 text-white' monospace>
|
<Text size='xs' className='mr-1 text-white' monospace>
|
||||||
{`${props.maxText}:`}
|
{`${props.maxText}:`}
|
||||||
</Text>
|
</Text>
|
||||||
<FormattedNumber
|
<FormattedNumber
|
||||||
className='mr-1 text-xs text-white/50'
|
className='mr-1 text-xs text-white/50'
|
||||||
amount={props.max?.toNumber() || 0}
|
amount={props.max.toNumber()}
|
||||||
options={{ decimals: asset.decimals }}
|
options={{ decimals: props.asset.decimals }}
|
||||||
/>
|
/>
|
||||||
<Button
|
<Button
|
||||||
|
dataTestId='token-input-max-button'
|
||||||
color='tertiary'
|
color='tertiary'
|
||||||
className='h-4 bg-white/20 px-1.5 py-0.5 text-2xs'
|
className='h-4 bg-white/20 px-1.5 py-0.5 text-2xs'
|
||||||
variant='transparent'
|
variant='transparent'
|
||||||
@ -127,7 +115,7 @@ export default function TokenInput(props: SingleProps | SelectProps) {
|
|||||||
<DisplayCurrency
|
<DisplayCurrency
|
||||||
isApproximation
|
isApproximation
|
||||||
className='inline pl-0.5 text-xs text-white/50'
|
className='inline pl-0.5 text-xs text-white/50'
|
||||||
coin={{ denom: asset.denom, amount: props.amount.toString() }}
|
coin={{ denom: props.asset.denom, amount: props.amount.toString() }}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,91 +1,58 @@
|
|||||||
import BigNumber from 'bignumber.js'
|
import BigNumber from 'bignumber.js'
|
||||||
import { useCallback, useEffect, useState } from 'react'
|
import { useState } from 'react'
|
||||||
|
|
||||||
import Slider from 'components/Slider'
|
import Slider from 'components/Slider'
|
||||||
import TokenInput from 'components/TokenInput'
|
import TokenInput from 'components/TokenInput'
|
||||||
import { ASSETS } from 'constants/assets'
|
|
||||||
import { BN } from 'utils/helpers'
|
import { BN } from 'utils/helpers'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
amount: BigNumber
|
amount: BigNumber
|
||||||
|
asset: Asset
|
||||||
|
max: BigNumber
|
||||||
onChange: (amount: BigNumber) => void
|
onChange: (amount: BigNumber) => void
|
||||||
|
accountId?: string
|
||||||
|
balances?: Coin[]
|
||||||
className?: string
|
className?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
balances?: Coin[] | null
|
|
||||||
accountId?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface SingleProps extends Props {
|
|
||||||
max: BigNumber
|
|
||||||
maxText: string
|
|
||||||
asset: Asset
|
|
||||||
hasSelect?: boolean
|
hasSelect?: boolean
|
||||||
|
maxText?: string
|
||||||
onChangeAsset?: (asset: Asset) => void
|
onChangeAsset?: (asset: Asset) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
interface SelectProps extends Props {
|
export default function TokenInputWithSlider(props: Props) {
|
||||||
max?: BigNumber
|
|
||||||
maxText?: string
|
|
||||||
asset?: Asset
|
|
||||||
onChangeAsset: (asset: Asset) => void
|
|
||||||
hasSelect: boolean
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function TokenInputWithSlider(props: SingleProps | SelectProps) {
|
|
||||||
const [amount, setAmount] = useState(props.amount)
|
const [amount, setAmount] = useState(props.amount)
|
||||||
const [percentage, setPercentage] = useState(0)
|
const [percentage, setPercentage] = useState(0)
|
||||||
const [asset, setAsset] = useState<Asset>(props.asset ? props.asset : ASSETS[0])
|
|
||||||
const [max, setMax] = useState<BigNumber>(props.max ? props.max : BN(0))
|
|
||||||
|
|
||||||
const onSliderChange = useCallback(
|
function onChangeSlider(percentage: number) {
|
||||||
(percentage: number) => {
|
const newAmount = BN(percentage).div(100).times(props.max)
|
||||||
const newAmount = BN(percentage).div(100).times(max)
|
setPercentage(percentage)
|
||||||
setPercentage(percentage)
|
setAmount(newAmount)
|
||||||
setAmount(newAmount)
|
props.onChange(newAmount)
|
||||||
props.onChange(newAmount)
|
}
|
||||||
},
|
|
||||||
[props, max],
|
|
||||||
)
|
|
||||||
|
|
||||||
const onInputChange = useCallback(
|
function onChangeAmount(newAmount: BigNumber) {
|
||||||
(newAmount: BigNumber) => {
|
setAmount(newAmount)
|
||||||
setAmount(newAmount)
|
setPercentage(BN(newAmount).div(props.max).times(100).toNumber())
|
||||||
setPercentage(BN(newAmount).div(max).times(100).toNumber())
|
props.onChange(newAmount)
|
||||||
props.onChange(newAmount)
|
}
|
||||||
},
|
|
||||||
[props, max],
|
|
||||||
)
|
|
||||||
|
|
||||||
const onAssetChange = useCallback(
|
function onChangeAsset(newAsset: Asset) {
|
||||||
(newAsset: Asset, liquidtyAmount: BigNumber) => {
|
if (!props.onChangeAsset) return
|
||||||
props.onChangeAsset && props.onChangeAsset(newAsset)
|
|
||||||
setAsset(newAsset)
|
|
||||||
setMax(liquidtyAmount)
|
|
||||||
setPercentage(0)
|
|
||||||
setAmount(BN(0))
|
|
||||||
},
|
|
||||||
[props],
|
|
||||||
)
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (props.max?.isEqualTo(max)) return
|
|
||||||
|
|
||||||
setMax(props.max ?? BN(0))
|
|
||||||
setPercentage(0)
|
setPercentage(0)
|
||||||
setAmount(BN(0))
|
setAmount(BN(0))
|
||||||
setAsset(props.asset ?? ASSETS[0])
|
props.onChangeAsset(newAsset)
|
||||||
}, [props.max, props.asset, max])
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={props.className}>
|
<div className={props.className}>
|
||||||
<TokenInput
|
<TokenInput
|
||||||
asset={asset}
|
asset={props.asset}
|
||||||
onChange={(amount) => onInputChange(amount)}
|
onChange={onChangeAmount}
|
||||||
onChangeAsset={(asset: Asset, max: BigNumber) => onAssetChange(asset, max)}
|
onChangeAsset={onChangeAsset}
|
||||||
amount={amount}
|
amount={amount}
|
||||||
max={max}
|
|
||||||
maxText={props.maxText || ''}
|
|
||||||
className='mb-4'
|
className='mb-4'
|
||||||
|
max={props.max}
|
||||||
|
maxText={props.maxText}
|
||||||
disabled={props.disabled}
|
disabled={props.disabled}
|
||||||
hasSelect={props.hasSelect}
|
hasSelect={props.hasSelect}
|
||||||
balances={props.balances}
|
balances={props.balances}
|
||||||
@ -93,7 +60,7 @@ export default function TokenInputWithSlider(props: SingleProps | SelectProps) {
|
|||||||
/>
|
/>
|
||||||
<Slider
|
<Slider
|
||||||
value={percentage}
|
value={percentage}
|
||||||
onChange={(value) => onSliderChange(value)}
|
onChange={(value) => onChangeSlider(value)}
|
||||||
disabled={props.disabled}
|
disabled={props.disabled}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
@ -56,7 +56,7 @@ export default function ConnectedButton() {
|
|||||||
const disconnectWallet = () => {
|
const disconnectWallet = () => {
|
||||||
disconnect()
|
disconnect()
|
||||||
terminate()
|
terminate()
|
||||||
useStore.setState({ client: undefined, balances: null })
|
useStore.setState({ client: undefined, balances: [] })
|
||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -4,7 +4,6 @@ import { GetState, SetState } from 'zustand'
|
|||||||
|
|
||||||
import { ENV } from 'constants/env'
|
import { ENV } from 'constants/env'
|
||||||
import { Store } from 'store'
|
import { Store } from 'store'
|
||||||
import { getMarketAssets } from 'utils/assets'
|
|
||||||
import { getSingleValueFromBroadcastResult } from 'utils/broadcast'
|
import { getSingleValueFromBroadcastResult } from 'utils/broadcast'
|
||||||
import { formatAmountWithSymbol } from 'utils/formatters'
|
import { formatAmountWithSymbol } from 'utils/formatters'
|
||||||
|
|
||||||
@ -12,7 +11,6 @@ export default function createBroadcastSlice(
|
|||||||
set: SetState<Store>,
|
set: SetState<Store>,
|
||||||
get: GetState<Store>,
|
get: GetState<Store>,
|
||||||
): BroadcastSlice {
|
): BroadcastSlice {
|
||||||
const marketAssets = getMarketAssets()
|
|
||||||
return {
|
return {
|
||||||
toast: null,
|
toast: null,
|
||||||
borrow: async (options: { fee: StdFee; accountId: string; coin: Coin }) => {
|
borrow: async (options: { fee: StdFee; accountId: string; coin: Coin }) => {
|
||||||
|
@ -4,7 +4,7 @@ import { GetState, SetState } from 'zustand'
|
|||||||
export default function createCommonSlice(set: SetState<CommonSlice>, get: GetState<CommonSlice>) {
|
export default function createCommonSlice(set: SetState<CommonSlice>, get: GetState<CommonSlice>) {
|
||||||
return {
|
return {
|
||||||
accounts: null,
|
accounts: null,
|
||||||
balances: null,
|
balances: [],
|
||||||
creditAccounts: null,
|
creditAccounts: null,
|
||||||
enableAnimations: true,
|
enableAnimations: true,
|
||||||
isOpen: true,
|
isOpen: true,
|
||||||
|
2
src/types/interfaces/store/comon.d.ts
vendored
2
src/types/interfaces/store/comon.d.ts
vendored
@ -3,7 +3,7 @@ interface CommonSlice {
|
|||||||
address?: string
|
address?: string
|
||||||
enableAnimations: boolean
|
enableAnimations: boolean
|
||||||
isOpen: boolean
|
isOpen: boolean
|
||||||
balances: Coin[] | null
|
balances: Coin[]
|
||||||
selectedAccount: string | null
|
selectedAccount: string | null
|
||||||
client?: import('@marsprotocol/wallet-connector').WalletClient
|
client?: import('@marsprotocol/wallet-connector').WalletClient
|
||||||
status: import('@marsprotocol/wallet-connector').WalletConnectionStatus
|
status: import('@marsprotocol/wallet-connector').WalletConnectionStatus
|
||||||
|
@ -100,7 +100,7 @@ module.exports = {
|
|||||||
success: '#32D583',
|
success: '#32D583',
|
||||||
'success-bg': '#6CE9A6',
|
'success-bg': '#6CE9A6',
|
||||||
'vote-against': '#eb9e49',
|
'vote-against': '#eb9e49',
|
||||||
warning: '#c83333',
|
warning: '#F79009',
|
||||||
white: '#FFF',
|
white: '#FFF',
|
||||||
},
|
},
|
||||||
fontFamily: {
|
fontFamily: {
|
||||||
|
Loading…
Reference in New Issue
Block a user