|
@ -1,7 +1,34 @@ |
|
|
import { CheckIcon, CopyIcon, ExternalLinkIcon } from "@chakra-ui/icons"; |
|
|
import { CheckIcon, CopyIcon, ExternalLinkIcon } from "@chakra-ui/icons"; |
|
|
import { Box, Center, Divider, HStack, IconButton, Skeleton, Text, useClipboard } from "@chakra-ui/react"; |
|
|
import { |
|
|
|
|
|
Box, |
|
|
|
|
|
Button, |
|
|
|
|
|
Center, |
|
|
|
|
|
Divider, |
|
|
|
|
|
FormControl, |
|
|
|
|
|
FormHelperText, |
|
|
|
|
|
FormLabel, |
|
|
|
|
|
Heading, |
|
|
|
|
|
HStack, |
|
|
|
|
|
IconButton, |
|
|
|
|
|
Input, |
|
|
|
|
|
Link, |
|
|
|
|
|
NumberDecrementStepper, |
|
|
|
|
|
NumberIncrementStepper, |
|
|
|
|
|
NumberInput, |
|
|
|
|
|
NumberInputField, |
|
|
|
|
|
NumberInputStepper, |
|
|
|
|
|
Skeleton, |
|
|
|
|
|
Text, |
|
|
|
|
|
useClipboard, |
|
|
|
|
|
useToast, |
|
|
|
|
|
VStack, |
|
|
|
|
|
} from "@chakra-ui/react"; |
|
|
import * as React from "react"; |
|
|
import * as React from "react"; |
|
|
|
|
|
import { useState } from "react"; |
|
|
|
|
|
import { useAsync } from "react-async"; |
|
|
import { useNavigate } from "react-router-dom"; |
|
|
import { useNavigate } from "react-router-dom"; |
|
|
|
|
|
import { postWithdraw } from "../App"; |
|
|
|
|
|
import createErrorToast from "./ErrorToast"; |
|
|
import Timestamp from "./Timestamp"; |
|
|
import Timestamp from "./Timestamp"; |
|
|
import { WalletInfo } from "./Types"; |
|
|
import { WalletInfo } from "./Types"; |
|
|
|
|
|
|
|
@ -14,14 +41,46 @@ export default function Wallet( |
|
|
walletInfo, |
|
|
walletInfo, |
|
|
}: WalletProps, |
|
|
}: WalletProps, |
|
|
) { |
|
|
) { |
|
|
|
|
|
const toast = useToast(); |
|
|
const { hasCopied, onCopy } = useClipboard(walletInfo ? walletInfo.address : ""); |
|
|
const { hasCopied, onCopy } = useClipboard(walletInfo ? walletInfo.address : ""); |
|
|
const { balance, address, last_updated_at } = walletInfo || {}; |
|
|
const { balance, address, last_updated_at } = walletInfo || {}; |
|
|
|
|
|
|
|
|
|
|
|
const [withdrawAmount, setWithdrawAmount] = useState(0); |
|
|
|
|
|
const [fee, setFee] = useState(1); |
|
|
|
|
|
const [withdrawAddress, setWithdrawAddress] = useState(""); |
|
|
|
|
|
|
|
|
|
|
|
let { run: runWithdraw, isLoading: isWithdrawing } = useAsync({ |
|
|
|
|
|
deferFn: async () => { |
|
|
|
|
|
try { |
|
|
|
|
|
const url = await postWithdraw({ |
|
|
|
|
|
amount: withdrawAmount, |
|
|
|
|
|
fee, |
|
|
|
|
|
address: withdrawAddress, |
|
|
|
|
|
}); |
|
|
|
|
|
window.open(url, "_blank"); |
|
|
|
|
|
toast({ |
|
|
|
|
|
title: "Withdraw successful", |
|
|
|
|
|
description: <Link href={url} isExternal> |
|
|
|
|
|
{url} |
|
|
|
|
|
</Link>, |
|
|
|
|
|
status: "info", |
|
|
|
|
|
duration: 10000, |
|
|
|
|
|
isClosable: true, |
|
|
|
|
|
}); |
|
|
|
|
|
} catch (e) { |
|
|
|
|
|
console.log(`Caught an error: ${e}`); |
|
|
|
|
|
createErrorToast(toast, e); |
|
|
|
|
|
} |
|
|
|
|
|
}, |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
return ( |
|
|
return ( |
|
|
<Center> |
|
|
<Center> |
|
|
<Box shadow={"md"} marginBottom={5} padding={5} boxSize={"sm"}> |
|
|
<Box shadow={"md"} marginBottom={5} padding={5}> |
|
|
<Center><Text fontWeight={"bold"}>Your wallet</Text></Center> |
|
|
<Center> |
|
|
<HStack> |
|
|
<Heading size="sm">Wallet Details</Heading> |
|
|
|
|
|
</Center> |
|
|
|
|
|
<HStack padding={2}> |
|
|
<Text align={"left"}>Balance:</Text> |
|
|
<Text align={"left"}>Balance:</Text> |
|
|
<Skeleton isLoaded={balance != null}> |
|
|
<Skeleton isLoaded={balance != null}> |
|
|
<Text>{balance} BTC</Text> |
|
|
<Text>{balance} BTC</Text> |
|
@ -45,6 +104,65 @@ export default function Wallet( |
|
|
<Timestamp timestamp={last_updated_at!} /> |
|
|
<Timestamp timestamp={last_updated_at!} /> |
|
|
</Skeleton> |
|
|
</Skeleton> |
|
|
</HStack> |
|
|
</HStack> |
|
|
|
|
|
|
|
|
|
|
|
<Divider marginTop={2} marginBottom={2} /> |
|
|
|
|
|
|
|
|
|
|
|
<VStack padding={2}> |
|
|
|
|
|
<form onSubmit={runWithdraw}> |
|
|
|
|
|
<Heading as="h3" size="sm">Withdraw</Heading> |
|
|
|
|
|
<FormControl id="address"> |
|
|
|
|
|
<FormLabel>Address</FormLabel> |
|
|
|
|
|
<Input |
|
|
|
|
|
onChange={(event) => setWithdrawAddress(event.target.value)} |
|
|
|
|
|
value={withdrawAddress} |
|
|
|
|
|
placeholder="Target address" |
|
|
|
|
|
> |
|
|
|
|
|
</Input> |
|
|
|
|
|
</FormControl> |
|
|
|
|
|
<HStack> |
|
|
|
|
|
<FormControl id="amount"> |
|
|
|
|
|
<FormLabel>Amount</FormLabel> |
|
|
|
|
|
<NumberInput |
|
|
|
|
|
min={0} |
|
|
|
|
|
max={balance} |
|
|
|
|
|
default={0} |
|
|
|
|
|
onChange={(_, amount) => setWithdrawAmount(amount)} |
|
|
|
|
|
value={withdrawAmount} |
|
|
|
|
|
precision={8} |
|
|
|
|
|
step={0.001} |
|
|
|
|
|
placeholder="How much do you want to withdraw? (0 to withdraw all)" |
|
|
|
|
|
> |
|
|
|
|
|
<NumberInputField /> |
|
|
|
|
|
<NumberInputStepper> |
|
|
|
|
|
<NumberIncrementStepper /> |
|
|
|
|
|
<NumberDecrementStepper /> |
|
|
|
|
|
</NumberInputStepper> |
|
|
|
|
|
</NumberInput> |
|
|
|
|
|
<FormHelperText>How much do you want to withdraw? (0 to withdraw all)</FormHelperText> |
|
|
|
|
|
</FormControl> |
|
|
|
|
|
<FormControl id="fee" w={"30%"}> |
|
|
|
|
|
<FormLabel>Fee</FormLabel> |
|
|
|
|
|
<NumberInput |
|
|
|
|
|
min={1} |
|
|
|
|
|
max={100} |
|
|
|
|
|
default={0} |
|
|
|
|
|
onChange={(_, amount) => setFee(amount)} |
|
|
|
|
|
value={fee} |
|
|
|
|
|
step={1} |
|
|
|
|
|
placeholder="In sats/vbyte" |
|
|
|
|
|
> |
|
|
|
|
|
<NumberInputField /> |
|
|
|
|
|
<NumberInputStepper> |
|
|
|
|
|
<NumberIncrementStepper /> |
|
|
|
|
|
<NumberDecrementStepper /> |
|
|
|
|
|
</NumberInputStepper> |
|
|
|
|
|
</NumberInput> |
|
|
|
|
|
<FormHelperText>In sats/vbyte</FormHelperText> |
|
|
|
|
|
</FormControl> |
|
|
|
|
|
</HStack> |
|
|
|
|
|
<Button type="submit" isLoading={isWithdrawing}>Withdraw</Button> |
|
|
|
|
|
</form> |
|
|
|
|
|
</VStack> |
|
|
</Box> |
|
|
</Box> |
|
|
</Center> |
|
|
</Center> |
|
|
); |
|
|
); |
|
|