Skip to content

Commit

Permalink
feat: add okx and bitget btc wallet (#1)
Browse files Browse the repository at this point in the history
  • Loading branch information
danielsimao authored Jun 14, 2024
1 parent b1b88c4 commit 9e30cbb
Show file tree
Hide file tree
Showing 17 changed files with 578 additions and 382 deletions.
31 changes: 21 additions & 10 deletions apps/evm/src/pages/Bridge/components/BridgeForm/BtcBridgeForm.tsx
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
import { AuthButton } from '@gobob/connect-ui';
import { Bitcoin, CurrencyAmount, Ether } from '@gobob/currency';
import { INTERVAL, useMutation, usePrices, useQuery, useQueryClient } from '@gobob/react-query';
import { useAccount as useSatsAccount, useBalance as useSatsBalance } from '@gobob/sats-wagmi';
import { BtcAddressType, useAccount as useSatsAccount, useBalance as useSatsBalance } from '@gobob/sats-wagmi';
import { BITCOIN } from '@gobob/tokens';
import { Avatar, Flex, Input, Item, P, Select, TokenInput, toast, useForm } from '@gobob/ui';
import { Alert, Avatar, Flex, Input, Item, P, Select, TokenInput, toast, useForm } from '@gobob/ui';
import { useAccount, useIsContract } from '@gobob/wagmi';
import { mergeProps } from '@react-aria/utils';
import { useDebounce } from '@uidotdev/usehooks';
Expand Down Expand Up @@ -55,7 +55,7 @@ const BtcBridgeForm = ({

const { isContract: isSmartAccount } = useIsContract({ address: evmAddress });

const { address: btcAddress, connector } = useSatsAccount();
const { address: btcAddress, connector, addressType: btcAddressType } = useSatsAccount();
const { data: satsBalance } = useSatsBalance();

const { getPrice } = usePrices({ baseUrl: import.meta.env.VITE_MARKET_DATA_API });
Expand All @@ -80,11 +80,7 @@ const BtcBridgeForm = ({
const balanceCurrencyAmount = useMemo(() => CurrencyAmount.fromRawAmount(BITCOIN, satsBalance || 0n), [satsBalance]);

const handleError = useCallback((e: any) => {
if (e.code === 4001) {
toast.error('User rejected the request');
} else {
toast.error('Something went wrong. Please try again later.');
}
toast.error(e.message);
}, []);

const quoteDataEnabled = useMemo(() => {
Expand All @@ -102,7 +98,8 @@ const BtcBridgeForm = ({
const {
data: quoteData,
isLoading: isFetchingQuote,
isError: isQuoteError
isError: isQuoteError,
error: quoteError
} = useQuery({
enabled: quoteDataEnabled,
queryKey: quoteQueryKey,
Expand Down Expand Up @@ -237,7 +234,9 @@ const BtcBridgeForm = ({

const isSubmitDisabled = isFormDisabled(form);

const isDisabled = isSubmitDisabled || !quoteData || isQuoteError;
const isTapRootAddress = btcAddressType === BtcAddressType.p2tr;

const isDisabled = isSubmitDisabled || !quoteData || isQuoteError || isTapRootAddress;

const isLoading = !isSubmitDisabled && (depositMutation.isPending || isFetchingQuote);

Expand Down Expand Up @@ -289,6 +288,18 @@ const BtcBridgeForm = ({
{isSmartAccount && (
<Input label='Recipient' placeholder='Enter destination address' {...form.getFieldProps(BRIDGE_RECIPIENT)} />
)}
{isTapRootAddress && (
<Alert status='warning'>
<P size='s'>
Unfortunately, Taproot (P2TR) addresses are not supported at this time. Please use a different address type.
</P>
</Alert>
)}
{!!quoteError && (
<Alert status='warning'>
<P size='s'>BTC bridge is currenlty unavailable. Please try again later.</P>
</Alert>
)}
<TransactionDetails
amount={receiveAmount}
amountPlaceholder={placeholderAmount}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -226,6 +226,7 @@ const ConnectModal = forwardRef<HTMLDivElement, ConnectModalProps>(
{btcWalletConnector && btcWalletAddress ? (
<ConnectedWalletSection
address={btcWalletAddress}
icon={btcWalletConnector.icon}
type='btc'
wallet={btcWalletConnector.name}
onDisconnect={handleBtcWalletDisconnect}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,12 @@ const Label = ({
) : (
<WalletIcon name={evmConnector.name} />
))}
{btcConnector && <WalletIcon name={btcConnector.name} />}
{btcConnector &&
(btcConnector.icon ? (
<Avatar rounded='none' size='2xl' src={btcConnector.icon} />
) : (
<WalletIcon name={btcConnector.name} />
))}
</StyledWallets>
<Span size='s' weight='medium'>
Wallet
Expand Down
2 changes: 2 additions & 0 deletions packages/sats-wagmi/src/assets/bitget.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export const bitkeepLogo: string =
'';
2 changes: 2 additions & 0 deletions packages/sats-wagmi/src/assets/okx.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export const okxLogo: string =
'';
29 changes: 25 additions & 4 deletions packages/sats-wagmi/src/connectors/base.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,13 @@ interface PsbtInputAccounts {

abstract class SatsConnector {
/** Unique connector id */
abstract readonly id: string;
id: string;
/** Connector name */
abstract readonly name: string;
name: string;
/** Extension or Snap homepage */
abstract homepage: string;
homepage: string;

icon?: string;

/** Whether connector is usable */
ready: boolean = false;
Expand All @@ -41,8 +43,12 @@ abstract class SatsConnector {

network: WalletNetwork;

constructor(network: WalletNetwork) {
constructor(network: WalletNetwork, id: string, name: string, homepage: string, icon?: string) {
this.network = network;
this.id = id;
this.name = name;
this.homepage = homepage;
this.icon = icon;
}

abstract connect(): Promise<void>;
Expand Down Expand Up @@ -234,12 +240,27 @@ abstract class SatsConnector {
throw new Error('Invalid network');
}

const addressType = getAddressInfo(this.paymentAddress).type;

// Ensure this is not the P2TR address for ordinals (we don't want to spend from it)
if (addressType === AddressType.p2tr) {
throw new Error('Cannot transfer using Taproot (P2TR) address. Please use another address type.');
}

// We need the public key to generate the redeem and witness script to spend the scripts
if (addressType === (AddressType.p2sh || AddressType.p2wsh)) {
if (!this.publicKey) {
throw new Error('Public key is required to spend from the selected address type');
}
}

const unsignedTransaction = await createTransferWithOpReturn(
networkType,
this.paymentAddress,
toAddress,
amount,
opReturn,
addressType,
this.publicKey
);

Expand Down
6 changes: 1 addition & 5 deletions packages/sats-wagmi/src/connectors/leather.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,14 +78,10 @@ declare global {
}

class LeatherConnector extends SatsConnector {
id = 'leather';
name = 'Leather';
homepage = 'https://leather.io/';

derivationPath: string | undefined;

constructor(network: WalletNetwork) {
super(network);
super(network, 'leather', 'Leather', 'https://leather.io/');
}

async connect(): Promise<void> {
Expand Down
7 changes: 1 addition & 6 deletions packages/sats-wagmi/src/connectors/mm-snap.ts
Original file line number Diff line number Diff line change
Expand Up @@ -69,16 +69,11 @@ const snapId = 'npm:@gobob/bob-snap';

// TODO: distinguish between payment and oridnals address
class MMSnapConnector extends SatsConnector {
id = 'metamask_snap';
name = 'MetaMask';
// TODO: add when snap is published
homepage = 'https://metamask.io/snaps/';

extendedPublicKey: ExtendedPublicKey | undefined;
snapNetwork: 'main' | 'test' = 'main';

constructor(network: WalletNetwork) {
super(network);
super(network, 'metamask_snap', 'MetaMask', 'https://snaps.metamask.io/snap/npm/gobob/bob-snap/');
}

async connect(): Promise<void> {
Expand Down
Loading

0 comments on commit 9e30cbb

Please sign in to comment.