otx-btc-wallet-connectors 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +554 -0
- package/dist/base-IAFq7sd8.d.mts +53 -0
- package/dist/base-IAFq7sd8.d.ts +53 -0
- package/dist/binance/index.d.mts +81 -0
- package/dist/binance/index.d.ts +81 -0
- package/dist/binance/index.js +13 -0
- package/dist/binance/index.js.map +1 -0
- package/dist/binance/index.mjs +4 -0
- package/dist/binance/index.mjs.map +1 -0
- package/dist/bitget/index.d.mts +84 -0
- package/dist/bitget/index.d.ts +84 -0
- package/dist/bitget/index.js +13 -0
- package/dist/bitget/index.js.map +1 -0
- package/dist/bitget/index.mjs +4 -0
- package/dist/bitget/index.mjs.map +1 -0
- package/dist/chunk-5Z5Q2Y75.mjs +91 -0
- package/dist/chunk-5Z5Q2Y75.mjs.map +1 -0
- package/dist/chunk-7KK2LZLZ.mjs +208 -0
- package/dist/chunk-7KK2LZLZ.mjs.map +1 -0
- package/dist/chunk-AW2JZIHR.mjs +753 -0
- package/dist/chunk-AW2JZIHR.mjs.map +1 -0
- package/dist/chunk-EIJOSZXZ.js +331 -0
- package/dist/chunk-EIJOSZXZ.js.map +1 -0
- package/dist/chunk-EQHR7P7G.js +541 -0
- package/dist/chunk-EQHR7P7G.js.map +1 -0
- package/dist/chunk-EWRXLZO4.mjs +539 -0
- package/dist/chunk-EWRXLZO4.mjs.map +1 -0
- package/dist/chunk-FISNQZZ7.js +802 -0
- package/dist/chunk-FISNQZZ7.js.map +1 -0
- package/dist/chunk-HL4WDMGS.js +200 -0
- package/dist/chunk-HL4WDMGS.js.map +1 -0
- package/dist/chunk-IPYWR76I.js +314 -0
- package/dist/chunk-IPYWR76I.js.map +1 -0
- package/dist/chunk-JYYNWR5G.js +142 -0
- package/dist/chunk-JYYNWR5G.js.map +1 -0
- package/dist/chunk-LNKTYZJM.js +701 -0
- package/dist/chunk-LNKTYZJM.js.map +1 -0
- package/dist/chunk-LVZMONQL.mjs +699 -0
- package/dist/chunk-LVZMONQL.mjs.map +1 -0
- package/dist/chunk-MFXLQWOE.js +93 -0
- package/dist/chunk-MFXLQWOE.js.map +1 -0
- package/dist/chunk-NBIA4TTE.mjs +204 -0
- package/dist/chunk-NBIA4TTE.mjs.map +1 -0
- package/dist/chunk-O4DD2XJ2.js +206 -0
- package/dist/chunk-O4DD2XJ2.js.map +1 -0
- package/dist/chunk-P7HVBU2B.mjs +140 -0
- package/dist/chunk-P7HVBU2B.mjs.map +1 -0
- package/dist/chunk-Q7QVQYEB.js +210 -0
- package/dist/chunk-Q7QVQYEB.js.map +1 -0
- package/dist/chunk-RLZEG6KL.mjs +329 -0
- package/dist/chunk-RLZEG6KL.mjs.map +1 -0
- package/dist/chunk-SYLDBJ75.mjs +246 -0
- package/dist/chunk-SYLDBJ75.mjs.map +1 -0
- package/dist/chunk-TTEUU3CI.mjs +198 -0
- package/dist/chunk-TTEUU3CI.mjs.map +1 -0
- package/dist/chunk-V66BXDTR.mjs +292 -0
- package/dist/chunk-V66BXDTR.mjs.map +1 -0
- package/dist/chunk-X77ZT4OI.js +268 -0
- package/dist/chunk-X77ZT4OI.js.map +1 -0
- package/dist/imtoken/index.d.mts +116 -0
- package/dist/imtoken/index.d.ts +116 -0
- package/dist/imtoken/index.js +14 -0
- package/dist/imtoken/index.js.map +1 -0
- package/dist/imtoken/index.mjs +5 -0
- package/dist/imtoken/index.mjs.map +1 -0
- package/dist/index.d.mts +14 -0
- package/dist/index.d.ts +14 -0
- package/dist/index.js +170 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +13 -0
- package/dist/index.mjs.map +1 -0
- package/dist/ledger/index.d.mts +290 -0
- package/dist/ledger/index.d.ts +290 -0
- package/dist/ledger/index.js +14 -0
- package/dist/ledger/index.js.map +1 -0
- package/dist/ledger/index.mjs +5 -0
- package/dist/ledger/index.mjs.map +1 -0
- package/dist/okx/index.d.mts +88 -0
- package/dist/okx/index.d.ts +88 -0
- package/dist/okx/index.js +13 -0
- package/dist/okx/index.js.map +1 -0
- package/dist/okx/index.mjs +4 -0
- package/dist/okx/index.mjs.map +1 -0
- package/dist/phantom/index.d.mts +96 -0
- package/dist/phantom/index.d.ts +96 -0
- package/dist/phantom/index.js +14 -0
- package/dist/phantom/index.js.map +1 -0
- package/dist/phantom/index.mjs +5 -0
- package/dist/phantom/index.mjs.map +1 -0
- package/dist/psbt-builder-CFOs69Z5.d.mts +131 -0
- package/dist/psbt-builder-CFOs69Z5.d.ts +131 -0
- package/dist/trezor/index.d.mts +155 -0
- package/dist/trezor/index.d.ts +155 -0
- package/dist/trezor/index.js +14 -0
- package/dist/trezor/index.js.map +1 -0
- package/dist/trezor/index.mjs +5 -0
- package/dist/trezor/index.mjs.map +1 -0
- package/dist/unisat/index.d.mts +75 -0
- package/dist/unisat/index.d.ts +75 -0
- package/dist/unisat/index.js +13 -0
- package/dist/unisat/index.js.map +1 -0
- package/dist/unisat/index.mjs +4 -0
- package/dist/unisat/index.mjs.map +1 -0
- package/dist/utils/index.d.mts +398 -0
- package/dist/utils/index.d.ts +398 -0
- package/dist/utils/index.js +120 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/index.mjs +3 -0
- package/dist/utils/index.mjs.map +1 -0
- package/dist/xverse/index.d.mts +79 -0
- package/dist/xverse/index.d.ts +79 -0
- package/dist/xverse/index.js +13 -0
- package/dist/xverse/index.js.map +1 -0
- package/dist/xverse/index.mjs +4 -0
- package/dist/xverse/index.mjs.map +1 -0
- package/package.json +108 -0
- package/src/base.ts +132 -0
- package/src/binance/BinanceConnector.ts +307 -0
- package/src/binance/index.ts +1 -0
- package/src/bitget/BitgetConnector.ts +301 -0
- package/src/bitget/index.ts +1 -0
- package/src/imtoken/ImTokenConnector.ts +420 -0
- package/src/imtoken/index.ts +2 -0
- package/src/index.ts +78 -0
- package/src/ledger/LedgerConnector.ts +1019 -0
- package/src/ledger/index.ts +8 -0
- package/src/okx/OKXConnector.ts +230 -0
- package/src/okx/index.ts +1 -0
- package/src/phantom/PhantomConnector.ts +381 -0
- package/src/phantom/index.ts +2 -0
- package/src/trezor/TrezorConnector.ts +824 -0
- package/src/trezor/index.ts +6 -0
- package/src/unisat/UnisatConnector.ts +312 -0
- package/src/unisat/index.ts +1 -0
- package/src/utils/blockstream.ts +230 -0
- package/src/utils/btc-service.ts +364 -0
- package/src/utils/index.ts +56 -0
- package/src/utils/mempool.ts +232 -0
- package/src/utils/psbt-builder.ts +492 -0
- package/src/utils/types.ts +183 -0
- package/src/xverse/XverseConnector.ts +479 -0
- package/src/xverse/index.ts +1 -0
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var chunkEIJOSZXZ_js = require('../chunk-EIJOSZXZ.js');
|
|
4
|
+
require('../chunk-MFXLQWOE.js');
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
Object.defineProperty(exports, "XverseConnector", {
|
|
9
|
+
enumerable: true,
|
|
10
|
+
get: function () { return chunkEIJOSZXZ_js.XverseConnector; }
|
|
11
|
+
});
|
|
12
|
+
//# sourceMappingURL=out.js.map
|
|
13
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"names":[],"mappings":""}
|
package/package.json
ADDED
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "otx-btc-wallet-connectors",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Bitcoin wallet connectors for otx-btc-wallet",
|
|
5
|
+
"license": "MIT",
|
|
6
|
+
"main": "./dist/index.js",
|
|
7
|
+
"module": "./dist/index.mjs",
|
|
8
|
+
"types": "./dist/index.d.ts",
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"import": "./dist/index.mjs",
|
|
12
|
+
"require": "./dist/index.js",
|
|
13
|
+
"types": "./dist/index.d.ts"
|
|
14
|
+
},
|
|
15
|
+
"./unisat": {
|
|
16
|
+
"import": "./dist/unisat/index.mjs",
|
|
17
|
+
"require": "./dist/unisat/index.js",
|
|
18
|
+
"types": "./dist/unisat/index.d.ts"
|
|
19
|
+
},
|
|
20
|
+
"./xverse": {
|
|
21
|
+
"import": "./dist/xverse/index.mjs",
|
|
22
|
+
"require": "./dist/xverse/index.js",
|
|
23
|
+
"types": "./dist/xverse/index.d.ts"
|
|
24
|
+
},
|
|
25
|
+
"./okx": {
|
|
26
|
+
"import": "./dist/okx/index.mjs",
|
|
27
|
+
"require": "./dist/okx/index.js",
|
|
28
|
+
"types": "./dist/okx/index.d.ts"
|
|
29
|
+
},
|
|
30
|
+
"./leather": {
|
|
31
|
+
"import": "./dist/leather/index.mjs",
|
|
32
|
+
"require": "./dist/leather/index.js",
|
|
33
|
+
"types": "./dist/leather/index.d.ts"
|
|
34
|
+
},
|
|
35
|
+
"./phantom": {
|
|
36
|
+
"import": "./dist/phantom/index.mjs",
|
|
37
|
+
"require": "./dist/phantom/index.js",
|
|
38
|
+
"types": "./dist/phantom/index.d.ts"
|
|
39
|
+
},
|
|
40
|
+
"./bitget": {
|
|
41
|
+
"import": "./dist/bitget/index.mjs",
|
|
42
|
+
"require": "./dist/bitget/index.js",
|
|
43
|
+
"types": "./dist/bitget/index.d.ts"
|
|
44
|
+
},
|
|
45
|
+
"./binance": {
|
|
46
|
+
"import": "./dist/binance/index.mjs",
|
|
47
|
+
"require": "./dist/binance/index.js",
|
|
48
|
+
"types": "./dist/binance/index.d.ts"
|
|
49
|
+
},
|
|
50
|
+
"./ledger": {
|
|
51
|
+
"import": "./dist/ledger/index.mjs",
|
|
52
|
+
"require": "./dist/ledger/index.js",
|
|
53
|
+
"types": "./dist/ledger/index.d.ts"
|
|
54
|
+
},
|
|
55
|
+
"./trezor": {
|
|
56
|
+
"import": "./dist/trezor/index.mjs",
|
|
57
|
+
"require": "./dist/trezor/index.js",
|
|
58
|
+
"types": "./dist/trezor/index.d.ts"
|
|
59
|
+
},
|
|
60
|
+
"./imtoken": {
|
|
61
|
+
"import": "./dist/imtoken/index.mjs",
|
|
62
|
+
"require": "./dist/imtoken/index.js",
|
|
63
|
+
"types": "./dist/imtoken/index.d.ts"
|
|
64
|
+
},
|
|
65
|
+
"./utils": {
|
|
66
|
+
"import": "./dist/utils/index.mjs",
|
|
67
|
+
"require": "./dist/utils/index.js",
|
|
68
|
+
"types": "./dist/utils/index.d.ts"
|
|
69
|
+
}
|
|
70
|
+
},
|
|
71
|
+
"files": [
|
|
72
|
+
"dist",
|
|
73
|
+
"src"
|
|
74
|
+
],
|
|
75
|
+
"sideEffects": false,
|
|
76
|
+
"dependencies": {
|
|
77
|
+
"@ledgerhq/hw-app-btc": "^10.4.1",
|
|
78
|
+
"@ledgerhq/hw-transport-webusb": "^6.30.0",
|
|
79
|
+
"@trezor/connect-web": "^9.4.3",
|
|
80
|
+
"bitcoinjs-lib": "^7.0.1",
|
|
81
|
+
"sats-connect": "^2.2.0",
|
|
82
|
+
"otx-btc-wallet-core": "0.1.0"
|
|
83
|
+
},
|
|
84
|
+
"devDependencies": {
|
|
85
|
+
"typescript": "^5.3.2",
|
|
86
|
+
"tsup": "^8.0.1"
|
|
87
|
+
},
|
|
88
|
+
"keywords": [
|
|
89
|
+
"bitcoin",
|
|
90
|
+
"wallet",
|
|
91
|
+
"unisat",
|
|
92
|
+
"xverse",
|
|
93
|
+
"okx",
|
|
94
|
+
"leather",
|
|
95
|
+
"phantom",
|
|
96
|
+
"bitget",
|
|
97
|
+
"binance",
|
|
98
|
+
"ledger",
|
|
99
|
+
"trezor",
|
|
100
|
+
"imtoken"
|
|
101
|
+
],
|
|
102
|
+
"scripts": {
|
|
103
|
+
"build": "tsup",
|
|
104
|
+
"dev": "tsup --watch",
|
|
105
|
+
"clean": "rm -rf dist",
|
|
106
|
+
"typecheck": "tsc --noEmit"
|
|
107
|
+
}
|
|
108
|
+
}
|
package/src/base.ts
ADDED
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
BitcoinConnector,
|
|
3
|
+
WalletAccount,
|
|
4
|
+
BitcoinNetwork,
|
|
5
|
+
SignPsbtOptions,
|
|
6
|
+
AddressType,
|
|
7
|
+
} from 'otx-btc-wallet-core';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Abstract base class for all wallet connectors
|
|
11
|
+
* Provides common functionality and enforces the BitcoinConnector interface
|
|
12
|
+
*/
|
|
13
|
+
export abstract class BaseConnector implements BitcoinConnector {
|
|
14
|
+
abstract readonly id: string;
|
|
15
|
+
abstract readonly name: string;
|
|
16
|
+
abstract readonly icon: string;
|
|
17
|
+
|
|
18
|
+
private _ready = false;
|
|
19
|
+
private _accountsChangedCallbacks: Set<(accounts: WalletAccount[]) => void> =
|
|
20
|
+
new Set();
|
|
21
|
+
private _networkChangedCallbacks: Set<(network: BitcoinNetwork) => void> =
|
|
22
|
+
new Set();
|
|
23
|
+
|
|
24
|
+
constructor() {
|
|
25
|
+
this.checkReady();
|
|
26
|
+
// Some wallets inject after DOMContentLoaded
|
|
27
|
+
if (typeof window !== 'undefined') {
|
|
28
|
+
if (document.readyState === 'complete') {
|
|
29
|
+
this.checkReady();
|
|
30
|
+
} else {
|
|
31
|
+
window.addEventListener('load', () => this.checkReady());
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
get ready(): boolean {
|
|
37
|
+
return this._ready;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
protected set ready(value: boolean) {
|
|
41
|
+
this._ready = value;
|
|
42
|
+
}
|
|
43
|
+
protected abstract getProvider(): unknown;
|
|
44
|
+
protected checkReady(): void {
|
|
45
|
+
this._ready = !!this.getProvider();
|
|
46
|
+
}
|
|
47
|
+
protected ensureInstalled(): void {
|
|
48
|
+
if (!this.ready) {
|
|
49
|
+
throw new Error(`${this.name} wallet is not installed`);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
protected inferAddressType(address: string): AddressType {
|
|
53
|
+
if (address.startsWith('bc1q') || address.startsWith('tb1q')) {
|
|
54
|
+
return 'segwit';
|
|
55
|
+
}
|
|
56
|
+
if (address.startsWith('bc1p') || address.startsWith('tb1p')) {
|
|
57
|
+
return 'taproot';
|
|
58
|
+
}
|
|
59
|
+
if (address.startsWith('3') || address.startsWith('2')) {
|
|
60
|
+
return 'nested-segwit';
|
|
61
|
+
}
|
|
62
|
+
return 'legacy';
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
protected handleError(error: unknown): never {
|
|
66
|
+
console.error('Exception in connector', this.id, error);
|
|
67
|
+
// Throw raw error for other cases
|
|
68
|
+
throw error;
|
|
69
|
+
}
|
|
70
|
+
// Abstract methods that must be implemented
|
|
71
|
+
abstract connect(network?: BitcoinNetwork): Promise<WalletAccount>;
|
|
72
|
+
abstract disconnect(): Promise<void>;
|
|
73
|
+
abstract getAccounts(): Promise<WalletAccount[]>;
|
|
74
|
+
abstract signMessage(message: string): Promise<string>;
|
|
75
|
+
abstract signPsbt(psbtHex: string, options?: SignPsbtOptions): Promise<string>;
|
|
76
|
+
abstract sendTransaction(to: string, satoshis: number): Promise<string>;
|
|
77
|
+
abstract getNetwork(): Promise<BitcoinNetwork>;
|
|
78
|
+
|
|
79
|
+
// Optional methods - can be overridden
|
|
80
|
+
signPsbts?(
|
|
81
|
+
psbtHexs: string[],
|
|
82
|
+
options?: SignPsbtOptions
|
|
83
|
+
): Promise<string[]>;
|
|
84
|
+
|
|
85
|
+
switchNetwork?(network: BitcoinNetwork): Promise<void>;
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* Subscribe to account changes
|
|
89
|
+
*/
|
|
90
|
+
onAccountsChanged(callback: (accounts: WalletAccount[]) => void): () => void {
|
|
91
|
+
this._accountsChangedCallbacks.add(callback);
|
|
92
|
+
return () => {
|
|
93
|
+
this._accountsChangedCallbacks.delete(callback);
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* Subscribe to network changes
|
|
99
|
+
*/
|
|
100
|
+
onNetworkChanged(callback: (network: BitcoinNetwork) => void): () => void {
|
|
101
|
+
this._networkChangedCallbacks.add(callback);
|
|
102
|
+
return () => {
|
|
103
|
+
this._networkChangedCallbacks.delete(callback);
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* Emit account change event
|
|
109
|
+
*/
|
|
110
|
+
protected emitAccountsChanged(accounts: WalletAccount[]): void {
|
|
111
|
+
for (const callback of this._accountsChangedCallbacks) {
|
|
112
|
+
callback(accounts);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
/**
|
|
117
|
+
* Emit network change event
|
|
118
|
+
*/
|
|
119
|
+
protected emitNetworkChanged(network: BitcoinNetwork): void {
|
|
120
|
+
for (const callback of this._networkChangedCallbacks) {
|
|
121
|
+
callback(network);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
/**
|
|
126
|
+
* Clean up all listeners
|
|
127
|
+
*/
|
|
128
|
+
protected cleanup(): void {
|
|
129
|
+
this._accountsChangedCallbacks.clear();
|
|
130
|
+
this._networkChangedCallbacks.clear();
|
|
131
|
+
}
|
|
132
|
+
}
|
|
@@ -0,0 +1,307 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
WalletAccount,
|
|
3
|
+
BitcoinNetwork,
|
|
4
|
+
SignPsbtOptions,
|
|
5
|
+
} from 'otx-btc-wallet-core';
|
|
6
|
+
import { BaseConnector } from '../base';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Binance sign PSBT options
|
|
10
|
+
* @see https://developers.binance.com/docs/binance-w3w/bitcoin-provider
|
|
11
|
+
*/
|
|
12
|
+
interface BinanceSignPsbtOptions {
|
|
13
|
+
autoFinalized?: boolean;
|
|
14
|
+
toSignInputs?: Array<{
|
|
15
|
+
index: number;
|
|
16
|
+
address?: string;
|
|
17
|
+
publicKey?: string;
|
|
18
|
+
sighashTypes?: number[];
|
|
19
|
+
disableTweakSigner?: boolean;
|
|
20
|
+
}>;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Binance Bitcoin provider interface
|
|
25
|
+
* Provider is accessible via window.binancew3w.bitcoin
|
|
26
|
+
* @see https://developers.binance.com/docs/binance-w3w/bitcoin-provider
|
|
27
|
+
*/
|
|
28
|
+
interface BinanceBitcoinProvider {
|
|
29
|
+
requestAccounts(): Promise<string[]>;
|
|
30
|
+
getAccounts(): Promise<string[]>;
|
|
31
|
+
getPublicKey(): Promise<string>;
|
|
32
|
+
getBalance(): Promise<{ confirmed: number; unconfirmed: number; total: number }>;
|
|
33
|
+
getNetwork(): Promise<string>;
|
|
34
|
+
switchNetwork(network: 'livenet' | 'testnet' | 'signet'): Promise<boolean>;
|
|
35
|
+
signMessage(msg: string, type?: 'ecdsa' | 'bip322-simple'): Promise<string>;
|
|
36
|
+
signPsbt(psbtHex: string, options?: BinanceSignPsbtOptions): Promise<string>;
|
|
37
|
+
signPsbts(psbtHexs: string[], options?: BinanceSignPsbtOptions[]): Promise<string[]>;
|
|
38
|
+
pushPsbt(psbtHex: string): Promise<string>;
|
|
39
|
+
sendBitcoin(toAddress: string, satoshis: number, options?: { feeRate?: number }): Promise<string>;
|
|
40
|
+
on(event: 'accountsChanged', callback: (accounts: string[]) => void): void;
|
|
41
|
+
on(event: 'networkChanged', callback: (network: string) => void): void;
|
|
42
|
+
removeListener(event: 'accountsChanged', callback: (accounts: string[]) => void): void;
|
|
43
|
+
removeListener(event: 'networkChanged', callback: (network: string) => void): void;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Extend window type
|
|
47
|
+
declare global {
|
|
48
|
+
interface Window {
|
|
49
|
+
binancew3w?: {
|
|
50
|
+
bitcoin?: BinanceBitcoinProvider;
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Binance wallet icon
|
|
56
|
+
const BINANCE_ICON =
|
|
57
|
+
'';
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Binance network type
|
|
61
|
+
*/
|
|
62
|
+
type BinanceNetwork = 'livenet' | 'testnet' | 'signet';
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Binance Wallet Connector
|
|
66
|
+
*
|
|
67
|
+
* @see https://developers.binance.com/docs/binance-w3w/bitcoin-provider
|
|
68
|
+
*/
|
|
69
|
+
export class BinanceConnector extends BaseConnector {
|
|
70
|
+
readonly id = 'binance';
|
|
71
|
+
readonly name = 'Binance Wallet';
|
|
72
|
+
readonly icon = BINANCE_ICON;
|
|
73
|
+
|
|
74
|
+
private _removeAccountChangeListener: (() => void) | null = null;
|
|
75
|
+
private _removeNetworkChangeListener: (() => void) | null = null;
|
|
76
|
+
|
|
77
|
+
protected getProvider(): BinanceBitcoinProvider | undefined {
|
|
78
|
+
if (typeof window === 'undefined') return undefined;
|
|
79
|
+
return window.binancew3w?.bitcoin;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
async connect(network: BitcoinNetwork = 'mainnet'): Promise<WalletAccount> {
|
|
83
|
+
this.ensureInstalled();
|
|
84
|
+
const provider = this.getProvider()!;
|
|
85
|
+
|
|
86
|
+
try {
|
|
87
|
+
const accounts = await provider.requestAccounts();
|
|
88
|
+
if (!accounts || accounts.length === 0) {
|
|
89
|
+
throw new Error('No accounts found');
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Switch network if needed
|
|
93
|
+
await this.ensureNetwork(provider, network);
|
|
94
|
+
|
|
95
|
+
const publicKey = await provider.getPublicKey();
|
|
96
|
+
|
|
97
|
+
// Setup event listeners
|
|
98
|
+
this.setupEventListeners();
|
|
99
|
+
|
|
100
|
+
return {
|
|
101
|
+
address: accounts[0] ?? '',
|
|
102
|
+
publicKey,
|
|
103
|
+
type: this.inferAddressType(accounts[0] ?? ''),
|
|
104
|
+
};
|
|
105
|
+
} catch (error) {
|
|
106
|
+
this.handleError(error);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
private async ensureNetwork(provider: BinanceBitcoinProvider, network: BitcoinNetwork): Promise<void> {
|
|
111
|
+
const currentNetwork = await provider.getNetwork();
|
|
112
|
+
const targetNetwork = this.mapToBinanceNetwork(network);
|
|
113
|
+
|
|
114
|
+
if (currentNetwork !== targetNetwork) {
|
|
115
|
+
await provider.switchNetwork(targetNetwork);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
private setupEventListeners(): void {
|
|
120
|
+
const provider = this.getProvider();
|
|
121
|
+
if (!provider) return;
|
|
122
|
+
|
|
123
|
+
// Remove existing listeners
|
|
124
|
+
this.removeEventListeners();
|
|
125
|
+
|
|
126
|
+
// Account change listener
|
|
127
|
+
const handleAccountsChanged = (accounts: string[]) => {
|
|
128
|
+
if (!accounts || accounts.length === 0) {
|
|
129
|
+
this.emitAccountsChanged([]);
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
provider.getPublicKey().then((publicKey) => {
|
|
134
|
+
const walletAccounts: WalletAccount[] = accounts.map((address) => ({
|
|
135
|
+
address,
|
|
136
|
+
publicKey,
|
|
137
|
+
type: this.inferAddressType(address),
|
|
138
|
+
}));
|
|
139
|
+
this.emitAccountsChanged(walletAccounts);
|
|
140
|
+
}).catch(() => {
|
|
141
|
+
this.emitAccountsChanged([]);
|
|
142
|
+
});
|
|
143
|
+
};
|
|
144
|
+
|
|
145
|
+
// Network change listener
|
|
146
|
+
const handleNetworkChanged = (network: string) => {
|
|
147
|
+
const btcNetwork = this.mapNetwork(network);
|
|
148
|
+
this.emitNetworkChanged(btcNetwork);
|
|
149
|
+
};
|
|
150
|
+
|
|
151
|
+
provider.on('accountsChanged', handleAccountsChanged);
|
|
152
|
+
provider.on('networkChanged', handleNetworkChanged);
|
|
153
|
+
|
|
154
|
+
this._removeAccountChangeListener = () => {
|
|
155
|
+
provider.removeListener('accountsChanged', handleAccountsChanged);
|
|
156
|
+
};
|
|
157
|
+
this._removeNetworkChangeListener = () => {
|
|
158
|
+
provider.removeListener('networkChanged', handleNetworkChanged);
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
private removeEventListeners(): void {
|
|
163
|
+
this._removeAccountChangeListener?.();
|
|
164
|
+
this._removeAccountChangeListener = null;
|
|
165
|
+
|
|
166
|
+
this._removeNetworkChangeListener?.();
|
|
167
|
+
this._removeNetworkChangeListener = null;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
async disconnect(): Promise<void> {
|
|
171
|
+
this.removeEventListeners();
|
|
172
|
+
this.cleanup();
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
async getAccounts(): Promise<WalletAccount[]> {
|
|
176
|
+
this.ensureInstalled();
|
|
177
|
+
const provider = this.getProvider()!;
|
|
178
|
+
|
|
179
|
+
try {
|
|
180
|
+
const addresses = await provider.getAccounts();
|
|
181
|
+
const publicKey = await provider.getPublicKey();
|
|
182
|
+
|
|
183
|
+
return addresses.map((address) => ({
|
|
184
|
+
address,
|
|
185
|
+
publicKey,
|
|
186
|
+
type: this.inferAddressType(address),
|
|
187
|
+
}));
|
|
188
|
+
} catch (error) {
|
|
189
|
+
this.handleError(error);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
async signMessage(message: string): Promise<string> {
|
|
194
|
+
this.ensureInstalled();
|
|
195
|
+
const provider = this.getProvider()!;
|
|
196
|
+
|
|
197
|
+
try {
|
|
198
|
+
return await provider.signMessage(message);
|
|
199
|
+
} catch (error) {
|
|
200
|
+
this.handleError(error);
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
async signPsbt(psbtHex: string, options?: SignPsbtOptions): Promise<string> {
|
|
205
|
+
this.ensureInstalled();
|
|
206
|
+
const provider = this.getProvider()!;
|
|
207
|
+
|
|
208
|
+
try {
|
|
209
|
+
const binanceOptions: BinanceSignPsbtOptions = {
|
|
210
|
+
autoFinalized: options?.autoFinalize ?? true,
|
|
211
|
+
};
|
|
212
|
+
|
|
213
|
+
if (options?.toSignInputs) {
|
|
214
|
+
binanceOptions.toSignInputs = options.toSignInputs;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
return await provider.signPsbt(psbtHex, binanceOptions);
|
|
218
|
+
} catch (error) {
|
|
219
|
+
this.handleError(error);
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
async signPsbts(psbtHexs: string[], options?: SignPsbtOptions): Promise<string[]> {
|
|
224
|
+
this.ensureInstalled();
|
|
225
|
+
const provider = this.getProvider()!;
|
|
226
|
+
|
|
227
|
+
try {
|
|
228
|
+
const binanceOptions: BinanceSignPsbtOptions = {
|
|
229
|
+
autoFinalized: options?.autoFinalize ?? true,
|
|
230
|
+
};
|
|
231
|
+
|
|
232
|
+
if (options?.toSignInputs) {
|
|
233
|
+
binanceOptions.toSignInputs = options.toSignInputs;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
// Create options array for each PSBT
|
|
237
|
+
const optionsArray = psbtHexs.map(() => binanceOptions);
|
|
238
|
+
|
|
239
|
+
return await provider.signPsbts(psbtHexs, optionsArray);
|
|
240
|
+
} catch (error) {
|
|
241
|
+
this.handleError(error);
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
async sendTransaction(to: string, satoshis: number): Promise<string> {
|
|
246
|
+
this.ensureInstalled();
|
|
247
|
+
const provider = this.getProvider()!;
|
|
248
|
+
|
|
249
|
+
try {
|
|
250
|
+
return await provider.sendBitcoin(to, satoshis);
|
|
251
|
+
} catch (error) {
|
|
252
|
+
this.handleError(error);
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
async getNetwork(): Promise<BitcoinNetwork> {
|
|
257
|
+
this.ensureInstalled();
|
|
258
|
+
const provider = this.getProvider()!;
|
|
259
|
+
|
|
260
|
+
try {
|
|
261
|
+
const network = await provider.getNetwork();
|
|
262
|
+
return this.mapNetwork(network);
|
|
263
|
+
} catch (error) {
|
|
264
|
+
this.handleError(error);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
async switchNetwork(network: BitcoinNetwork): Promise<void> {
|
|
269
|
+
this.ensureInstalled();
|
|
270
|
+
const provider = this.getProvider()!;
|
|
271
|
+
|
|
272
|
+
try {
|
|
273
|
+
const binanceNetwork = this.mapToBinanceNetwork(network);
|
|
274
|
+
await provider.switchNetwork(binanceNetwork);
|
|
275
|
+
} catch (error) {
|
|
276
|
+
this.handleError(error);
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
private mapNetwork(network: string): BitcoinNetwork {
|
|
281
|
+
switch (network.toLowerCase()) {
|
|
282
|
+
case 'livenet':
|
|
283
|
+
case 'mainnet':
|
|
284
|
+
return 'mainnet';
|
|
285
|
+
case 'testnet':
|
|
286
|
+
return 'testnet';
|
|
287
|
+
case 'signet':
|
|
288
|
+
return 'signet';
|
|
289
|
+
default:
|
|
290
|
+
return 'mainnet';
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
private mapToBinanceNetwork(network: BitcoinNetwork): BinanceNetwork {
|
|
295
|
+
switch (network) {
|
|
296
|
+
case 'mainnet':
|
|
297
|
+
return 'livenet';
|
|
298
|
+
case 'testnet':
|
|
299
|
+
case 'testnet4':
|
|
300
|
+
return 'testnet';
|
|
301
|
+
case 'signet':
|
|
302
|
+
return 'signet';
|
|
303
|
+
default:
|
|
304
|
+
return 'livenet';
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { BinanceConnector } from './BinanceConnector';
|