mango-v4-ui/components/modals/UserSetupModal.tsx

627 lines
23 KiB
TypeScript
Raw Normal View History

2022-12-20 17:14:35 -08:00
import { Transition } from '@headlessui/react'
import {
ArrowDownTrayIcon,
ArrowTopRightOnSquareIcon,
2022-12-20 17:14:35 -08:00
CheckCircleIcon,
ChevronDownIcon,
2022-12-20 17:14:35 -08:00
ExclamationCircleIcon,
PencilIcon,
} from '@heroicons/react/20/solid'
import { useWallet } from '@solana/wallet-adapter-react'
import mangoStore from '@store/mangoStore'
import Decimal from 'decimal.js'
import useMangoAccount from 'hooks/useMangoAccount'
import useSolBalance from 'hooks/useSolBalance'
import { useTranslation } from 'next-i18next'
import Image from 'next/image'
import {
ChangeEvent,
ReactNode,
useCallback,
useEffect,
useMemo,
useState,
} from 'react'
import { notify } from 'utils/notifications'
import ActionTokenList from '../account/ActionTokenList'
import ButtonGroup from '../forms/ButtonGroup'
import Input from '../forms/Input'
import Label from '../forms/Label'
2023-05-21 22:24:04 -07:00
// import ParticlesBackground from '../ParticlesBackground'
// import EditNftProfilePic from '../profile/EditNftProfilePic'
// import EditProfileForm from '../profile/EditProfileForm'
2022-12-20 17:14:35 -08:00
import Button, { LinkButton } from '../shared/Button'
import InlineNotification from '../shared/InlineNotification'
import Loading from '../shared/Loading'
import MaxAmountButton from '../shared/MaxAmountButton'
import SolBalanceWarnings from '../shared/SolBalanceWarnings'
import Modal from '../shared/Modal'
2023-01-03 15:58:31 -08:00
import NumberFormat, { NumberFormatValues } from 'react-number-format'
import { withValueLimit } from '@components/swap/MarketSwapForm'
2023-01-29 20:13:38 -08:00
import useBanksWithBalances from 'hooks/useBanksWithBalances'
import BankAmountWithValue from '@components/shared/BankAmountWithValue'
2023-02-27 23:20:11 -08:00
import { isMangoError } from 'types'
2023-03-07 18:54:36 -08:00
import ColorBlur from '@components/ColorBlur'
import useLocalStorageState from 'hooks/useLocalStorageState'
2023-08-24 05:05:22 -07:00
import { ACCEPT_TERMS_KEY, MAX_ACCOUNTS } from 'utils/constants'
2023-04-05 20:24:29 -07:00
import { ACCOUNT_ACTIONS_NUMBER_FORMAT_CLASSES } from '@components/BorrowForm'
import { WalletReadyState } from '@solana/wallet-adapter-base'
2022-12-20 17:14:35 -08:00
const UserSetupModal = ({
isOpen,
onClose,
}: {
isOpen: boolean
onClose: () => void
}) => {
const { t } = useTranslation(['common', 'onboarding', 'swap'])
2023-07-22 18:28:50 -07:00
const { connected, select, wallet, wallets, publicKey, connect } = useWallet()
2022-12-20 17:14:35 -08:00
const { mangoAccount } = useMangoAccount()
const mangoAccountLoading = mangoStore((s) => s.mangoAccount.initialLoad)
const [accountName, setAccountName] = useState('')
const [loadingAccount, setLoadingAccount] = useState(false)
const [showSetupStep, setShowSetupStep] = useState(0)
const [depositToken, setDepositToken] = useState('USDC')
const [depositAmount, setDepositAmount] = useState('')
const [submitDeposit, setSubmitDeposit] = useState(false)
const [sizePercentage, setSizePercentage] = useState('')
// const [showEditProfilePic, setShowEditProfilePic] = useState(false)
2022-12-20 17:14:35 -08:00
const { maxSolDeposit } = useSolBalance()
2023-01-29 20:13:38 -08:00
const banks = useBanksWithBalances('walletBalance')
const [, setAcceptTerms] = useLocalStorageState(ACCEPT_TERMS_KEY, '')
const [walletsToDisplay, setWalletstoDisplay] = useState<'default' | 'all'>(
2023-07-21 11:47:53 -07:00
'default',
)
const walletsDisplayed = useMemo(() => {
const firstFive = wallets.slice(0, 5)
const detectedWallets = wallets.filter(
2023-07-20 16:48:43 -07:00
(w) =>
w.readyState === WalletReadyState.Installed ||
2023-07-21 11:47:53 -07:00
w.readyState === WalletReadyState.Loadable,
)
if (walletsToDisplay === 'default') {
return detectedWallets.length > firstFive.length
? detectedWallets
: firstFive
} else {
return wallets
}
}, [walletsToDisplay, wallets])
2022-12-20 17:14:35 -08:00
useEffect(() => {
if (connected) {
setShowSetupStep(2)
}
}, [connected])
const handleCreateAccount = useCallback(async () => {
const client = mangoStore.getState().client
const group = mangoStore.getState().group
const actions = mangoStore.getState().actions
if (!group || !publicKey) return
2022-12-20 17:14:35 -08:00
setLoadingAccount(true)
try {
const { signature: tx } = await client.createMangoAccount(
2022-12-20 17:14:35 -08:00
group,
0,
accountName || 'Account 1',
2023-08-23 18:40:40 -07:00
parseInt(MAX_ACCOUNTS.tokenAccounts), // tokens
parseInt(MAX_ACCOUNTS.spotOpenOrders), // serum3
parseInt(MAX_ACCOUNTS.perpAccounts), // perps
parseInt(MAX_ACCOUNTS.perpOpenOrders), // perp Oo
2022-12-20 17:14:35 -08:00
)
actions.fetchMangoAccounts(publicKey)
2022-12-20 17:14:35 -08:00
if (tx) {
actions.fetchWalletTokens(publicKey) // need to update sol balance after account rent
2022-12-20 17:14:35 -08:00
setShowSetupStep(3)
notify({
title: t('new-account-success'),
type: 'success',
txid: tx,
})
}
2023-02-27 23:20:11 -08:00
} catch (e) {
if (isMangoError(e)) {
notify({
title: t('new-account-failed'),
txid: e?.txid,
type: 'error',
})
}
2022-12-20 17:14:35 -08:00
console.error(e)
} finally {
setLoadingAccount(false)
}
}, [accountName, publicKey, t])
2022-12-20 17:14:35 -08:00
const handleDeposit = useCallback(async () => {
const client = mangoStore.getState().client
const group = mangoStore.getState().group
const actions = mangoStore.getState().actions
const mangoAccount = mangoStore.getState().mangoAccount.current
2023-02-27 23:20:11 -08:00
const bank = group?.banksMapByName.get(depositToken)?.[0]
2022-12-20 17:14:35 -08:00
2023-02-27 23:20:11 -08:00
if (!mangoAccount || !group || !bank) return
2022-12-20 17:14:35 -08:00
try {
setSubmitDeposit(true)
const { signature: tx, slot } = await client.tokenDeposit(
2022-12-20 17:14:35 -08:00
group,
mangoAccount,
bank.mint,
2023-07-21 11:47:53 -07:00
parseFloat(depositAmount),
2022-12-20 17:14:35 -08:00
)
notify({
title: 'Transaction confirmed',
type: 'success',
txid: tx,
})
await actions.reloadMangoAccount(slot)
2022-12-20 17:14:35 -08:00
setSubmitDeposit(false)
onClose()
// setShowSetupStep(4)
2023-02-27 23:20:11 -08:00
} catch (e) {
setSubmitDeposit(false)
console.error(e)
if (!isMangoError(e)) return
2022-12-20 17:14:35 -08:00
notify({
title: 'Transaction failed',
description: e.message,
txid: e?.txid,
type: 'error',
})
}
2023-02-27 23:20:11 -08:00
}, [depositAmount, depositToken, onClose])
2022-12-20 17:14:35 -08:00
useEffect(() => {
if (mangoAccount && showSetupStep === 2) {
onClose()
}
}, [mangoAccount, showSetupStep, onClose])
const depositBank = useMemo(() => {
2023-01-29 20:13:38 -08:00
return banks.find((b) => b.bank.name === depositToken)?.bank
2022-12-20 17:14:35 -08:00
}, [depositToken, banks])
const tokenMax = useMemo(() => {
2023-01-29 20:13:38 -08:00
const bank = banks.find((b) => b.bank.name === depositToken)
2022-12-20 17:14:35 -08:00
if (bank) {
2023-01-29 20:13:38 -08:00
return { amount: bank.walletBalance, decimals: bank.bank.mintDecimals }
2022-12-20 17:14:35 -08:00
}
return { amount: 0, decimals: 0 }
}, [banks, depositToken])
2023-01-03 15:58:31 -08:00
const showInsufficientBalance =
tokenMax.amount < Number(depositAmount) ||
(depositToken === 'SOL' && maxSolDeposit <= 0)
2022-12-20 17:14:35 -08:00
const setMax = useCallback(() => {
const max = new Decimal(tokenMax.amount).toDecimalPlaces(
tokenMax.decimals,
2023-07-21 11:47:53 -07:00
Decimal.ROUND_FLOOR,
)
setDepositAmount(max.toString())
setSizePercentage('100')
}, [tokenMax])
2022-12-20 17:14:35 -08:00
const handleSizePercentage = useCallback(
(percentage: string) => {
setSizePercentage(percentage)
const amount = new Decimal(tokenMax.amount)
.mul(percentage)
.div(100)
.toDecimalPlaces(tokenMax.decimals, Decimal.ROUND_FLOOR)
2022-12-20 17:14:35 -08:00
setDepositAmount(amount.toString())
},
2023-07-21 11:47:53 -07:00
[tokenMax],
2022-12-20 17:14:35 -08:00
)
const handleNextStep = () => {
if (showSetupStep === 0) {
setAcceptTerms(Date.now())
}
2022-12-20 17:14:35 -08:00
setShowSetupStep(showSetupStep + 1)
}
return (
<Modal isOpen={isOpen} onClose={onClose} fullScreen disableOutsideClose>
2023-03-07 18:54:36 -08:00
<div className="grid h-screen overflow-auto bg-th-bkg-1 text-left lg:grid-cols-2">
2023-05-21 22:24:04 -07:00
<ColorBlur
width="66%"
height="300px"
className="-left-20 -top-20 bg-th-button opacity-10 brightness-125"
2023-05-21 22:24:04 -07:00
/>
<ColorBlur
width="50%"
height="100%"
className="-bottom-20 -right-20 bg-th-bkg-1 opacity-30 mix-blend-multiply"
2022-12-20 17:14:35 -08:00
/>
<img
className={`absolute left-6 top-6 h-10 w-10 flex-shrink-0`}
2022-12-20 17:14:35 -08:00
src="/logos/logo-mark.svg"
alt="next"
/>
<div className="absolute left-0 top-0 z-10 flex h-1.5 w-full flex-grow bg-th-bkg-3">
2022-12-20 17:14:35 -08:00
<div
style={{
width: `${(showSetupStep / 3) * 100}%`,
2022-12-20 17:14:35 -08:00
}}
className="flex bg-th-active transition-all duration-700 ease-out"
/>
</div>
<div className="col-span-1 flex flex-col items-center justify-center p-6 pt-24">
<UserSetupTransition show={showSetupStep === 0}>
2023-03-07 15:54:13 -08:00
<h2 className="mb-4 font-display text-3xl tracking-normal md:text-5xl lg:max-w-[400px] lg:text-6xl">
2022-12-20 17:14:35 -08:00
{t('onboarding:intro-heading')}
</h2>
2023-03-07 18:54:36 -08:00
<p className="text-base sm:mb-2 lg:text-lg">
2023-03-07 15:54:13 -08:00
{t('onboarding:intro-desc')}
</p>
<div className="mb-3 space-y-2 py-3">
2023-03-07 15:54:13 -08:00
<CheckBullet text={t('onboarding:bullet-1')} />
<CheckBullet text={t('onboarding:bullet-2')} />
<CheckBullet text={t('onboarding:bullet-3')} />
<CheckBullet text={t('onboarding:bullet-4')} />
2022-12-20 17:14:35 -08:00
</div>
<p className="mb-6 flex flex-wrap">
<span className="mr-1">{t('accept-terms-desc')}</span>
<a
className="flex items-center"
href="https://docs.mango.markets/legal/terms-of-use"
rel="noopener noreferrer"
target="_blank"
>
{t('terms-of-use')}
<ArrowTopRightOnSquareIcon className="ml-1 h-4 w-4 flex-shrink-0" />
</a>
2023-07-18 20:04:22 -07:00
<span className="mx-1">and</span>
<a
className="flex items-center"
href="https://docs.mango.markets/mango-markets/risks"
rel="noopener noreferrer"
target="_blank"
>
{t('risks')}
<ArrowTopRightOnSquareIcon className="ml-1 h-4 w-4 flex-shrink-0" />
</a>
</p>
<Button className="mb-12" onClick={handleNextStep} size="large">
{t('agree-and-continue')}
2022-12-20 17:14:35 -08:00
</Button>
</UserSetupTransition>
<UserSetupTransition delay show={showSetupStep === 1}>
{showSetupStep === 1 ? (
<div>
<h2 className="mb-6 font-display text-3xl tracking-normal md:text-5xl lg:text-6xl">
{t('onboarding:connect-wallet')}
</h2>
<p className="mb-2 text-base">
{t('onboarding:choose-wallet')}
</p>
<div className="space-y-2">
{walletsDisplayed?.map((w) => (
2022-12-20 17:14:35 -08:00
<button
className={`col-span-1 w-full rounded-md border px-4 py-3 text-base font-normal focus:outline-none md:hover:cursor-pointer md:hover:border-th-fgd-4 ${
2022-12-20 17:14:35 -08:00
w.adapter.name === wallet?.adapter.name
? 'border-th-active text-th-fgd-1 md:hover:border-th-active'
: 'border-th-bkg-4 text-th-fgd-2'
2022-12-20 17:14:35 -08:00
}`}
onClick={() => {
2023-07-22 18:28:50 -07:00
if (wallet) {
connect()
}
2022-12-20 17:14:35 -08:00
select(w.adapter.name)
}}
key={w.adapter.name}
>
<div className="flex items-center justify-between">
<div className="flex items-center">
<img
src={w.adapter.icon}
className="mr-2 h-5 w-5"
alt={`${w.adapter.name} icon`}
/>
<div className="ml-2">{w.adapter.name}</div>
</div>
2023-07-20 16:48:43 -07:00
{w.readyState === WalletReadyState.Installed ||
w.readyState === WalletReadyState.Loadable ? (
<div className="text-xs">Detected</div>
) : null}
2022-12-20 17:14:35 -08:00
</div>
</button>
))}
</div>
{walletsToDisplay !== 'all' ? (
<button
className="mt-4 flex w-full items-center justify-center text-base text-th-fgd-3 hover:text-th-fgd-1"
onClick={() => setWalletstoDisplay('all')}
>
<div>More</div>
<div>
<ChevronDownIcon className={`h-5 w-5 flex-shrink-0`} />
2022-12-20 17:14:35 -08:00
</div>
</button>
) : null}
2022-12-20 17:14:35 -08:00
</div>
) : null}
</UserSetupTransition>
<UserSetupTransition
delay
show={showSetupStep === 2 && !mangoAccountLoading}
>
{showSetupStep === 2 ? (
<div>
<div className="pb-6">
<h2 className="mb-4 font-display text-3xl tracking-normal md:text-5xl lg:text-6xl">
{t('onboarding:create-account')}
</h2>
<p className="text-base">
{t('onboarding:create-account-desc')}
</p>
</div>
2023-01-03 15:58:31 -08:00
<div className="mb-4">
2022-12-20 17:14:35 -08:00
<Label text={t('account-name')} optional />
<Input
type="text"
name="name"
id="name"
placeholder="e.g. Main Account"
value={accountName}
onChange={(e: ChangeEvent<HTMLInputElement>) =>
setAccountName(e.target.value)
}
2023-01-25 01:19:12 -08:00
maxLength={30}
2022-12-20 17:14:35 -08:00
/>
</div>
<SolBalanceWarnings className="mt-4" />
2023-01-03 15:58:31 -08:00
<div className="mt-2">
2022-12-20 17:14:35 -08:00
<InlineNotification
type="info"
desc={t('insufficient-sol')}
/>
<div className="mt-10">
<Button
2023-01-03 15:58:31 -08:00
className="mb-6 flex items-center justify-center"
2022-12-20 17:14:35 -08:00
disabled={maxSolDeposit <= 0}
onClick={handleCreateAccount}
size="large"
>
{loadingAccount ? (
<Loading />
) : (
<div className="flex items-center justify-center">
{t('create-account')}
</div>
)}
</Button>
<LinkButton onClick={onClose}>
2023-04-20 19:32:20 -07:00
{t('onboarding:skip')}
2022-12-20 17:14:35 -08:00
</LinkButton>
</div>
</div>
</div>
) : null}
</UserSetupTransition>
<UserSetupTransition delay show={showSetupStep === 3}>
{showSetupStep === 3 ? (
<div className="relative">
<h2 className="mb-6 font-display text-3xl tracking-normal md:text-5xl lg:text-6xl">
{t('onboarding:fund-account')}
</h2>
<UserSetupTransition show={depositToken.length > 0}>
<div className="mb-4">
<SolBalanceWarnings
amount={depositAmount}
className="mt-4"
2022-12-20 17:14:35 -08:00
setAmount={setDepositAmount}
selectedToken={depositToken}
/>
</div>
<div className="flex justify-between">
<Label text={t('amount')} />
<MaxAmountButton
className="mb-2"
2023-01-23 19:04:05 -08:00
decimals={tokenMax.decimals}
2023-01-03 15:58:31 -08:00
label="Max"
onClick={setMax}
2023-01-23 19:04:05 -08:00
value={tokenMax.amount}
2022-12-20 17:14:35 -08:00
/>
</div>
<div className="mb-6 grid grid-cols-2">
<button
2023-01-03 15:58:31 -08:00
className="col-span-1 flex items-center rounded-lg rounded-r-none border border-r-0 border-th-input-border bg-th-input-bkg px-4"
2022-12-20 17:14:35 -08:00
onClick={() => setDepositToken('')}
>
<div className="flex w-full items-center justify-between">
<div className="flex items-center">
<Image
alt=""
width="20"
height="20"
src={`/icons/${depositToken.toLowerCase()}.svg`}
/>
<p className="ml-2 text-xl font-bold text-th-fgd-1">
{depositToken}
</p>
</div>
<PencilIcon className="ml-2 h-5 w-5 text-th-fgd-3" />
</div>
</button>
2023-01-03 15:58:31 -08:00
<NumberFormat
name="amountIn"
id="amountIn"
inputMode="decimal"
thousandSeparator=","
allowNegative={false}
isNumericString={true}
decimalScale={tokenMax.decimals || 6}
2023-04-05 20:24:29 -07:00
className={ACCOUNT_ACTIONS_NUMBER_FORMAT_CLASSES}
2022-12-20 17:14:35 -08:00
placeholder="0.00"
value={depositAmount}
2023-01-03 15:58:31 -08:00
onValueChange={(e: NumberFormatValues) => {
setDepositAmount(
2023-07-21 11:47:53 -07:00
!Number.isNaN(Number(e.value)) ? e.value : '',
2023-01-03 15:58:31 -08:00
)
}}
isAllowed={withValueLimit}
2022-12-20 17:14:35 -08:00
/>
<div className="col-span-2 mt-2">
<ButtonGroup
activeValue={sizePercentage}
onChange={(p) => handleSizePercentage(p)}
values={['10', '25', '50', '75', '100']}
unit="%"
/>
</div>
</div>
2023-01-06 02:32:54 -08:00
{depositBank ? (
<div className="border-y border-th-bkg-3">
<div className="flex justify-between px-2 py-4">
<p>{t('deposit-amount')}</p>
<p className="font-mono text-th-fgd-2">
2023-01-29 20:13:38 -08:00
<BankAmountWithValue
amount={depositAmount}
bank={depositBank}
/>
{/* {depositAmount ? (
2023-01-06 02:32:54 -08:00
<>
2023-01-24 17:12:13 -08:00
<FormatNumericValue
value={depositAmount}
decimals={depositBank.mintDecimals}
/>{' '}
2023-01-06 02:32:54 -08:00
<span className="text-xs text-th-fgd-3">
(
2023-01-24 16:54:24 -08:00
<FormatNumericValue
value={
depositBank.uiPrice * Number(depositAmount)
}
decimals={2}
isUsd
/>
2023-01-06 02:32:54 -08:00
)
</span>
</>
) : (
<>
0{' '}
<span className="text-xs text-th-fgd-3">
($0.00)
</span>
</>
2023-01-29 20:13:38 -08:00
)} */}
2023-01-06 02:32:54 -08:00
</p>
</div>
2022-12-20 17:14:35 -08:00
</div>
2023-01-06 02:32:54 -08:00
) : null}
2022-12-20 17:14:35 -08:00
<Button
2023-01-06 02:32:54 -08:00
className="mb-6 mt-10 flex items-center justify-center"
2022-12-20 17:14:35 -08:00
disabled={
2023-02-25 12:02:21 -08:00
!depositAmount || !depositToken || showInsufficientBalance
2022-12-20 17:14:35 -08:00
}
onClick={handleDeposit}
size="large"
>
{submitDeposit ? (
<Loading />
) : showInsufficientBalance ? (
<div className="flex items-center">
<ExclamationCircleIcon className="mr-2 h-5 w-5 flex-shrink-0" />
{t('swap:insufficient-balance', {
symbol: depositToken,
})}
</div>
) : (
<div className="flex items-center justify-center">
<ArrowDownTrayIcon className="mr-2 h-5 w-5" />
{t('deposit')}
</div>
)}
</Button>
<LinkButton onClick={onClose}>
2023-04-20 19:32:20 -07:00
{t('onboarding:skip')}
2022-12-20 17:14:35 -08:00
</LinkButton>
</UserSetupTransition>
<UserSetupTransition show={depositToken.length === 0}>
<div
2023-03-07 18:54:36 -08:00
className="thin-scroll absolute top-[62px] w-full overflow-auto md:top-[74px] lg:top-36"
2022-12-20 17:14:35 -08:00
style={{ height: 'calc(100vh - 380px)' }}
>
<div className="flex items-center px-4 pb-2">
<div className="w-1/4">
2022-12-20 17:14:35 -08:00
<p className="text-xs">{t('token')}</p>
</div>
<div className="w-1/4 text-right">
2022-12-20 17:14:35 -08:00
<p className="text-xs">{t('deposit-rate')}</p>
</div>
<div className="w-1/2 text-right">
2022-12-20 17:14:35 -08:00
<p className="whitespace-nowrap text-xs">
{t('wallet-balance')}
</p>
</div>
</div>
<ActionTokenList
banks={banks}
onSelect={setDepositToken}
showDepositRates
valueKey="walletBalance"
/>
</div>
</UserSetupTransition>
</div>
) : null}
</UserSetupTransition>
</div>
2023-05-21 22:24:04 -07:00
<div className="relative col-span-1 hidden h-screen lg:block">
{/* <ParticlesBackground /> */}
<img
className={`absolute left-1/2 top-1/2 h-auto w-[95%] max-w-[700px] -translate-x-1/2 -translate-y-1/2 ${
showSetupStep !== 0 ? 'hidden lg:block' : 'hidden sm:block'
}`}
src="/images/onboarding-image@1x.png"
alt="next"
/>
2022-12-20 17:14:35 -08:00
</div>
</div>
</Modal>
)
}
export default UserSetupModal
2023-03-07 15:54:13 -08:00
const CheckBullet = ({ text }: { text: string }) => {
return (
<div className="flex items-center space-x-2">
<CheckCircleIcon className="h-5 w-5 text-th-active" />
<p className="text-base text-th-fgd-2">{text}</p>
</div>
)
}
2022-12-20 17:14:35 -08:00
const UserSetupTransition = ({
show,
children,
delay = false,
}: {
show: boolean
children: ReactNode
delay?: boolean
}) => {
return (
<Transition
appear
2023-03-07 18:54:36 -08:00
className="h-full w-full max-w-md"
2022-12-20 17:14:35 -08:00
show={show}
enter={`transition ease-in duration-300 ${delay ? 'delay-300' : ''}`}
enterFrom="opacity-0"
enterTo="opacity-100"
leave="transition ease-out duration-300"
leaveFrom="opacity-100"
leaveTo="opacity-0"
>
{children}
</Transition>
)
}