79 lines
1.8 KiB
TypeScript
79 lines
1.8 KiB
TypeScript
|
import { Button, Loading } from '@nextui-org/react'
|
||
|
import dynamic from 'next/dynamic'
|
||
|
import Image from 'next/image'
|
||
|
import { Fragment, useState } from 'react'
|
||
|
import s from './styles.module.css'
|
||
|
|
||
|
/**
|
||
|
* You can use normal import if you are not within next / ssr environment
|
||
|
* @info https://nextjs.org/docs/advanced-features/dynamic-import
|
||
|
*/
|
||
|
const ReactQrReader = dynamic(() => import('react-qr-reader-es6'), { ssr: false })
|
||
|
|
||
|
/**
|
||
|
* Types
|
||
|
*/
|
||
|
interface IProps {
|
||
|
onConnect: (uri: string) => Promise<void>
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Component
|
||
|
*/
|
||
|
export default function QrReader({ onConnect }: IProps) {
|
||
|
const [show, setShow] = useState(false)
|
||
|
const [loading, setLoading] = useState(false)
|
||
|
|
||
|
function onError() {
|
||
|
setShow(false)
|
||
|
}
|
||
|
|
||
|
async function onScan(data: string | null) {
|
||
|
if (data) {
|
||
|
await onConnect(data)
|
||
|
setShow(false)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function onShowScanner() {
|
||
|
setLoading(true)
|
||
|
setShow(true)
|
||
|
}
|
||
|
|
||
|
return (
|
||
|
<div className={s.container}>
|
||
|
{show ? (
|
||
|
<Fragment>
|
||
|
{loading && <Loading css={{ position: 'absolute' }} />}
|
||
|
<div className={s.mask}>
|
||
|
<ReactQrReader
|
||
|
onLoad={() => setLoading(false)}
|
||
|
showViewFinder={false}
|
||
|
onError={onError}
|
||
|
onScan={onScan}
|
||
|
style={{ width: '100%' }}
|
||
|
/>
|
||
|
</div>
|
||
|
</Fragment>
|
||
|
) : (
|
||
|
<div className={`${s.container} ${s.placeholder}`}>
|
||
|
<Image
|
||
|
src="/qr-icon.svg"
|
||
|
width={100}
|
||
|
height={100}
|
||
|
alt="qr code icon"
|
||
|
className={s.icon}
|
||
|
/>
|
||
|
<Button
|
||
|
color="gradient"
|
||
|
css={{ marginTop: '$10', width: '100%' }}
|
||
|
onClick={onShowScanner}
|
||
|
>
|
||
|
Scan QR code
|
||
|
</Button>
|
||
|
</div>
|
||
|
)}
|
||
|
</div>
|
||
|
)
|
||
|
}
|