Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add okx btc wallet #1

Merged
merged 4 commits into from
Jun 14, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
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