134 lines
4.4 KiB
JavaScript
134 lines
4.4 KiB
JavaScript
import { getGetServerSideProps } from '@/api/ssrApollo'
|
|
import Layout from '@/components/layout'
|
|
import styles from '@/styles/wallet.module.css'
|
|
import Link from 'next/link'
|
|
import { useWallets } from '@/components/wallet'
|
|
import { useEffect, useState } from 'react'
|
|
import dynamic from 'next/dynamic'
|
|
|
|
const WalletCard = dynamic(() => import('@/components/wallet-card'), { ssr: false })
|
|
|
|
export const getServerSideProps = getGetServerSideProps({ authRequired: true })
|
|
|
|
async function reorder (wallets, sourceIndex, targetIndex) {
|
|
const newOrder = [...wallets]
|
|
|
|
const [source] = newOrder.splice(sourceIndex, 1)
|
|
const newTargetIndex = sourceIndex < targetIndex ? targetIndex - 1 : targetIndex
|
|
const append = sourceIndex < targetIndex
|
|
|
|
newOrder.splice(newTargetIndex + (append ? 1 : 0), 0, source)
|
|
|
|
await Promise.all(
|
|
newOrder.map((w, i) =>
|
|
w.setPriority(i).catch(console.error)
|
|
)
|
|
)
|
|
}
|
|
|
|
export default function Wallet ({ ssrData }) {
|
|
const { wallets } = useWallets()
|
|
|
|
const [mounted, setMounted] = useState(false)
|
|
const [sourceIndex, setSourceIndex] = useState(null)
|
|
const [targetIndex, setTargetIndex] = useState(null)
|
|
|
|
useEffect(() => {
|
|
// mounted is required since draggable is false
|
|
// for wallets only available on the client during SSR
|
|
// and thus we need to render the component again on the client
|
|
setMounted(true)
|
|
}, [])
|
|
|
|
const onDragStart = (i) => (e) => {
|
|
// e.dataTransfer.dropEffect = 'move'
|
|
// We can only use the DataTransfer API inside the drop event
|
|
// see https://html.spec.whatwg.org/multipage/dnd.html#security-risks-in-the-drag-and-drop-model
|
|
// e.dataTransfer.setData('text/plain', name)
|
|
// That's why we use React state instead
|
|
setSourceIndex(i)
|
|
}
|
|
|
|
const onDragEnter = (i) => (e) => {
|
|
setTargetIndex(i)
|
|
}
|
|
|
|
const onDragEnd = async (e) => {
|
|
setSourceIndex(null)
|
|
setTargetIndex(null)
|
|
|
|
if (sourceIndex === targetIndex) return
|
|
|
|
await reorder(wallets, sourceIndex, targetIndex)
|
|
}
|
|
|
|
const onTouchStart = (i) => async (e) => {
|
|
if (sourceIndex !== null) {
|
|
await reorder(wallets, sourceIndex, i)
|
|
setSourceIndex(null)
|
|
} else {
|
|
setSourceIndex(i)
|
|
}
|
|
}
|
|
|
|
return (
|
|
<Layout>
|
|
<div className='py-5 w-100'>
|
|
<h2 className='mb-2 text-center'>attach wallets</h2>
|
|
<h6 className='text-muted text-center'>attach wallets to supplement your SN wallet</h6>
|
|
<div className='text-center'>
|
|
<Link href='/wallet/logs' className='text-muted fw-bold text-underline'>
|
|
wallet logs
|
|
</Link>
|
|
</div>
|
|
<div className={styles.walletGrid} onDragEnd={onDragEnd}>
|
|
{wallets
|
|
.sort((w1, w2) => {
|
|
// enabled/configured wallets always come before disabled/unconfigured wallets
|
|
if ((w1.enabled && !w2.enabled) || (w1.isConfigured && !w2.isConfigured)) {
|
|
return -1
|
|
} else if ((w2.enabled && !w1.enabled) || (w2.isConfigured && !w1.isConfigured)) {
|
|
return 1
|
|
}
|
|
|
|
const delta = w1.priority - w2.priority
|
|
// delta is NaN if either priority is undefined
|
|
if (!Number.isNaN(delta) && delta !== 0) return delta
|
|
|
|
// if both wallets have an id, use that as tie breaker
|
|
// since that's the order in which autowithdrawals are attempted
|
|
if (w1.id && w2.id) return Number(w1.id) - Number(w2.id)
|
|
|
|
// else we will use the card title as tie breaker
|
|
return w1.card.title < w2.card.title ? -1 : 1
|
|
})
|
|
.map((w, i) => {
|
|
const draggable = mounted && w.enabled
|
|
|
|
return (
|
|
<div
|
|
key={w.name}
|
|
draggable={draggable}
|
|
style={{ cursor: draggable ? 'move' : 'default' }}
|
|
onDragStart={draggable ? onDragStart(i) : undefined}
|
|
onTouchStart={draggable ? onTouchStart(i) : undefined}
|
|
onDragEnter={draggable ? onDragEnter(i) : undefined}
|
|
className={
|
|
!draggable
|
|
? ''
|
|
: (`${sourceIndex === i ? styles.drag : ''} ${draggable && targetIndex === i ? styles.drop : ''}`)
|
|
}
|
|
suppressHydrationWarning
|
|
>
|
|
<WalletCard wallet={w} />
|
|
</div>
|
|
)
|
|
}
|
|
)}
|
|
|
|
</div>
|
|
</div>
|
|
</Layout>
|
|
)
|
|
}
|