import { getGetServerSideProps } from '@/api/ssrApollo' import { Form, ClientInput, PasswordInput, CheckboxGroup, Checkbox } from '@/components/form' import { CenterLayout } from '@/components/layout' import { WalletSecurityBanner } from '@/components/banners' import { WalletLogs } from '@/components/wallet-logger' import { useToast } from '@/components/toast' import { useRouter } from 'next/router' import { useWallet } from '@/wallets/index' import Info from '@/components/info' import Text from '@/components/text' import { autowithdrawInitial, AutowithdrawSettings } from '@/components/autowithdraw-shared' import { canReceive, canSend, isConfigured } from '@/wallets/common' import { SSR } from '@/lib/constants' import WalletButtonBar from '@/components/wallet-buttonbar' import { useWalletConfigurator } from '@/wallets/config' import { useCallback, useEffect, useMemo, useState } from 'react' import { useMe } from '@/components/me' import validateWallet from '@/wallets/validate' import { ValidationError } from 'yup' import { useFormikContext } from 'formik' import useDarkMode from '@/components/dark-mode' export const getServerSideProps = getGetServerSideProps({ authRequired: true }) export default function WalletSettings () { const toaster = useToast() const router = useRouter() const { wallet: name } = router.query const wallet = useWallet(name) const { me } = useMe() const { save, detach } = useWalletConfigurator(wallet) const [dark] = useDarkMode() const [imgSrc, setImgSrc] = useState(wallet?.def.card?.image?.src) const initial = useMemo(() => { const initial = wallet?.def.fields.reduce((acc, field) => { // We still need to run over all wallet fields via reduce // even though we use wallet.config as the initial value // since wallet.config is empty when wallet is not configured. // Also, wallet.config includes general fields like // 'enabled' and 'priority' which are not defined in wallet.fields. return { ...acc, [field.name]: wallet?.config?.[field.name] || field.defaultValue || '' } }, wallet?.config) if (wallet?.def.fields.every(f => f.clientOnly)) { return initial } return { ...initial, ...autowithdrawInitial({ me }) } }, [wallet, me]) const validate = useCallback(async (data) => { try { await validateWallet(wallet.def, data, { yupOptions: { abortEarly: false }, topLevel: false, skipGenerated: true }) } catch (error) { if (error instanceof ValidationError) { return error.inner.reduce((acc, error) => { acc[error.path] = error.message return acc }, {}) } throw error } }, [wallet.def]) const { card: { image, title, subtitle } } = wallet?.def || { card: {} } useEffect(() => { if (!imgSrc) return // wallet.png <-> wallet-dark.png setImgSrc(dark ? image?.src.replace(/\.([a-z]{3})$/, '-dark.$1') : image?.src) }, [dark]) return ( {image ? {title} :

{title}

}
{subtitle}
{ try { const newConfig = !isConfigured(wallet) // enable wallet if wallet was just configured if (newConfig) { values.enabled = true } await save(values, values.enabled) toaster.success('saved settings') router.push('/settings/wallets') } catch (err) { console.error(err) toaster.danger(err.message || err.toString?.()) } }} > {wallet && } { try { await detach() toaster.success('saved settings') router.push('/settings/wallets') } catch (err) { console.error(err) const message = 'failed to detach: ' + err.message || err.toString?.() toaster.danger(message) } }} />
{wallet && }
) } function SendWarningBanner ({ walletDef }) { const { values } = useFormikContext() if (!canSend({ def: walletDef, config: values }) || !walletDef.requiresConfig) return null return } function ReceiveSettings ({ walletDef }) { const { values } = useFormikContext() return canReceive({ def: walletDef, config: values }) && } function WalletFields ({ wallet }) { return wallet.def.fields .map(({ name, label = '', type, help, optional, editable, requiredWithout, validate, clientOnly, serverOnly, generated, ...props }, i) => { const rawProps = { ...props, name, initialValue: wallet.config?.[name], readOnly: !SSR && isConfigured(wallet) && editable === false && !!wallet.config?.[name], groupClassName: props.hidden ? 'd-none' : undefined, label: label ? (
{label} {/* help can be a string or object to customize the label */} {help && ( {help.text || help} )} {optional && ( {typeof optional === 'boolean' ? 'optional' : {optional}} )}
) : undefined, required: !optional, autoFocus: i === 0 } if (type === 'text') { return } if (type === 'password') { return } return null }) }