btc-wallet 0.4.7-beta → 0.5.0-beta

Sign up to get free protection for your applications and to get access to all the features.
package/esm/index.js CHANGED
@@ -60,6 +60,346 @@ var BaseConnector = class {
60
60
  // src/icons/bitget.png
61
61
  var bitget_default = "";
62
62
 
63
+ // src/utils/index.ts
64
+ function shortString(str) {
65
+ if (Array.isArray(str)) {
66
+ str = "[" + str.toString() + "]";
67
+ }
68
+ if (str) {
69
+ if (typeof str.toString === "function") {
70
+ str = str.toString();
71
+ }
72
+ if (str.length <= 10) {
73
+ return str;
74
+ }
75
+ return `${str.slice(0, 5)}...${str.slice(str.length - 5, str.length)}`;
76
+ }
77
+ return "";
78
+ }
79
+ function copyToClipboard(text) {
80
+ return __async(this, null, function* () {
81
+ const clipboardCopy = () => __async(this, null, function* () {
82
+ if (navigator.clipboard) {
83
+ return navigator.clipboard.writeText(text);
84
+ } else {
85
+ const textarea = document.createElement("textarea");
86
+ textarea.value = text;
87
+ textarea.setAttribute("readonly", "");
88
+ textarea.style.position = "absolute";
89
+ textarea.style.left = "-9999px";
90
+ document.body.appendChild(textarea);
91
+ textarea.select();
92
+ const result = document.execCommand("copy");
93
+ document.body.removeChild(textarea);
94
+ if (!result) {
95
+ throw new Error("Copy to clipboard failed");
96
+ }
97
+ }
98
+ });
99
+ return new Promise((resolve, reject) => {
100
+ clipboardCopy().then(resolve).catch(reject);
101
+ });
102
+ });
103
+ }
104
+ var defaultTokenIcon = "https://static.particle.network/token-list/defaultToken/default.png";
105
+ var ipfsToSrc = (ipfs) => {
106
+ if (!ipfs || !ipfs.startsWith("ipfs://")) {
107
+ return ipfs || "";
108
+ }
109
+ return `https://ipfs.particle.network/${encodeURI(ipfs.slice(7))}`;
110
+ };
111
+ var checkBTCVersion = (accountContracts, accountContractKey, version) => {
112
+ if (!accountContracts[accountContractKey]) {
113
+ return false;
114
+ }
115
+ return accountContracts[accountContractKey].some((item) => item.version === version);
116
+ };
117
+ var delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
118
+ function retryOperation(_0, _1) {
119
+ return __async(this, arguments, function* (operation, shouldStop, {
120
+ maxRetries = 3,
121
+ delayMs = 1e3
122
+ } = {}) {
123
+ let retries = 0;
124
+ while (retries <= maxRetries) {
125
+ const result = yield operation();
126
+ if (shouldStop(result)) {
127
+ return result;
128
+ }
129
+ if (retries === maxRetries) {
130
+ console.warn("Max retries reached");
131
+ return result;
132
+ }
133
+ retries++;
134
+ yield delay(delayMs);
135
+ }
136
+ throw new Error("Unexpected execution path");
137
+ });
138
+ }
139
+ function toHex(originalString) {
140
+ const charArray = originalString.split("");
141
+ const asciiArray = charArray.map((char) => char.charCodeAt(0));
142
+ const hexArray = asciiArray.map((code) => code.toString(16));
143
+ let hexString = hexArray.join("");
144
+ hexString = hexString.replace(/(^0+)/g, "");
145
+ return hexString;
146
+ }
147
+ function isMobile() {
148
+ if (typeof window !== "undefined") {
149
+ return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(
150
+ navigator == null ? void 0 : navigator.userAgent
151
+ );
152
+ }
153
+ return false;
154
+ }
155
+
156
+ // src/utils/Dialog.ts
157
+ var Dialog = class {
158
+ static injectStyles() {
159
+ if (!document.querySelector("#dialog-styles")) {
160
+ const styleSheet = document.createElement("style");
161
+ styleSheet.id = "dialog-styles";
162
+ styleSheet.textContent = this.style;
163
+ document.head.appendChild(styleSheet);
164
+ }
165
+ }
166
+ static confirm(options) {
167
+ return new Promise((resolve) => {
168
+ this.injectStyles();
169
+ const container = document.createElement("div");
170
+ container.innerHTML = this.template;
171
+ document.body.appendChild(container);
172
+ const titleEl = container.querySelector(".dialog-title");
173
+ const messageEl = container.querySelector(".dialog-message");
174
+ const confirmBtn = container.querySelector(".dialog-confirm-btn");
175
+ const cancelBtn = container.querySelector(".dialog-cancel-btn");
176
+ if (options.title) {
177
+ titleEl.textContent = options.title;
178
+ } else {
179
+ titleEl.style.display = "none";
180
+ }
181
+ messageEl.textContent = options.message;
182
+ const cleanup = () => {
183
+ document.body.removeChild(container);
184
+ };
185
+ confirmBtn.addEventListener("click", () => {
186
+ cleanup();
187
+ resolve(true);
188
+ });
189
+ cancelBtn.addEventListener("click", () => {
190
+ cleanup();
191
+ resolve(false);
192
+ });
193
+ });
194
+ }
195
+ static alert(options) {
196
+ const messageEl = options.dangerouslyUseHTML ? { dangerouslySetInnerHTML: { __html: options.message } } : { children: options.message };
197
+ return new Promise((resolve) => {
198
+ var _a;
199
+ this.injectStyles();
200
+ const container = document.createElement("div");
201
+ container.innerHTML = this.template;
202
+ (_a = container.querySelector(".dialog-overlay")) == null ? void 0 : _a.classList.add("dialog-alert");
203
+ if (options.closable === false) {
204
+ const overlay = container.querySelector(".dialog-overlay");
205
+ overlay.style.pointerEvents = "none";
206
+ const dialogContainer = container.querySelector(".dialog-container");
207
+ dialogContainer.style.pointerEvents = "auto";
208
+ }
209
+ document.body.appendChild(container);
210
+ const titleEl = container.querySelector(".dialog-title");
211
+ const messageEl2 = container.querySelector(".dialog-message");
212
+ const confirmBtn = container.querySelector(".dialog-confirm-btn");
213
+ const cancelBtn = container.querySelector(".dialog-cancel-btn");
214
+ if (options.title) {
215
+ titleEl.textContent = options.title;
216
+ } else {
217
+ titleEl.style.display = "none";
218
+ }
219
+ messageEl2.innerHTML = options.message;
220
+ cancelBtn.style.display = "none";
221
+ if (options.closable === false) {
222
+ confirmBtn.style.display = "none";
223
+ }
224
+ const cleanup = () => {
225
+ if (options.closable === false) {
226
+ return;
227
+ }
228
+ document.body.removeChild(container);
229
+ };
230
+ confirmBtn.addEventListener("click", () => {
231
+ cleanup();
232
+ resolve();
233
+ });
234
+ });
235
+ }
236
+ };
237
+ Dialog.template = `
238
+ <div class="dialog-overlay">
239
+ <div class="dialog-container">
240
+ <div class="dialog-content">
241
+ <div class="dialog-title"></div>
242
+ <div class="dialog-message"></div>
243
+ <div class="dialog-buttons">
244
+ <button class="dialog-cancel-btn">Cancel</button>
245
+ <button class="dialog-confirm-btn">OK</button>
246
+ </div>
247
+ </div>
248
+ </div>
249
+ </div>
250
+ `;
251
+ Dialog.style = `
252
+ .dialog-overlay {
253
+ position: fixed;
254
+ top: 0;
255
+ left: 0;
256
+ right: 0;
257
+ bottom: 0;
258
+ background-color: rgba(0, 0, 0, 0.75);
259
+ display: flex;
260
+ align-items: center;
261
+ justify-content: center;
262
+ z-index: 999999;
263
+ backdrop-filter: blur(4px);
264
+ }
265
+ .dialog-container {
266
+ background: #21232f;
267
+ border-radius: 12px;
268
+ padding: 24px;
269
+ width: 350px;
270
+ box-shadow: 0 4px 24px rgba(0, 0, 0, 0.3);
271
+ }
272
+ .dialog-title {
273
+ font-size: 18px;
274
+ font-weight: 600;
275
+ margin-bottom: 16px;
276
+ color: #ffffff;
277
+ }
278
+ .dialog-message {
279
+ margin-bottom: 24px;
280
+ line-height: 1.6;
281
+ color: rgba(255, 255, 255, 0.8);
282
+ font-size: 14px;
283
+ }
284
+ .dialog-buttons {
285
+ display: flex;
286
+ justify-content: flex-end;
287
+ gap: 12px;
288
+ }
289
+ .dialog-alert .dialog-buttons {
290
+ justify-content: center;
291
+ }
292
+ .dialog-confirm-btn {
293
+ padding: 8px 24px;
294
+ background-color: #ff7a00;
295
+ color: white;
296
+ border: none;
297
+ border-radius: 6px;
298
+ cursor: pointer;
299
+ font-size: 14px;
300
+ font-weight: 500;
301
+ transition: all 0.2s ease;
302
+ }
303
+ .dialog-confirm-btn:hover {
304
+ background-color: #ff8f1f;
305
+ transform: translateY(-1px);
306
+ }
307
+ .dialog-confirm-btn:active {
308
+ transform: translateY(0);
309
+ }
310
+ .dialog-cancel-btn {
311
+ padding: 8px 24px;
312
+ background-color: rgba(255, 255, 255, 0.1);
313
+ color: rgba(255, 255, 255, 0.8);
314
+ border: none;
315
+ border-radius: 6px;
316
+ cursor: pointer;
317
+ font-size: 14px;
318
+ font-weight: 500;
319
+ transition: all 0.2s ease;
320
+ }
321
+ .dialog-cancel-btn:hover {
322
+ background-color: rgba(255, 255, 255, 0.15);
323
+ transform: translateY(-1px);
324
+ }
325
+ .dialog-cancel-btn:active {
326
+ transform: translateY(0);
327
+ }
328
+
329
+ .dialog-overlay {
330
+ animation: fadeIn 0.2s ease;
331
+ }
332
+ .dialog-container {
333
+ animation: slideIn 0.2s ease;
334
+ }
335
+ @keyframes fadeIn {
336
+ from {
337
+ opacity: 0;
338
+ }
339
+ to {
340
+ opacity: 1;
341
+ }
342
+ }
343
+ @keyframes slideIn {
344
+ from {
345
+ transform: translateY(-20px);
346
+ opacity: 0;
347
+ }
348
+ to {
349
+ transform: translateY(0);
350
+ opacity: 1;
351
+ }
352
+ }
353
+ `;
354
+
355
+ // src/connector/universalLink.ts
356
+ var MobileWalletConnect = class {
357
+ static getUniversalLink(walletId, url) {
358
+ switch (walletId) {
359
+ case "okx":
360
+ return `okx://wallet/dapp/url?dappUrl=${encodeURIComponent(url)}`;
361
+ case "bitget":
362
+ return `https://bkcode.vip?action=dapp&url=${encodeURIComponent(url)}`;
363
+ case "xverse":
364
+ return `xverse://browser?url=${encodeURIComponent(url)}`;
365
+ default:
366
+ return "";
367
+ }
368
+ }
369
+ static redirectToWallet(walletId) {
370
+ return __async(this, null, function* () {
371
+ var _a;
372
+ if (isMobile()) {
373
+ const currentUrl = window.location.href;
374
+ const universalLink = this.getUniversalLink(walletId, currentUrl);
375
+ if (!universalLink) {
376
+ try {
377
+ yield (_a = navigator.clipboard) == null ? void 0 : _a.writeText(currentUrl);
378
+ } catch (error) {
379
+ console.error(error);
380
+ }
381
+ yield Dialog.alert({
382
+ title: "Open in Wallet Browser",
383
+ message: `
384
+ <div style="display: flex; flex-direction: column; gap: 12px;">
385
+ <p>Please follow these steps:</p>
386
+ <p>1. Open ${walletId} wallet app</p>
387
+ <p>2. Find the browser feature in the wallet</p>
388
+ <p>3. Paste the URL (already copied to clipboard)</p>
389
+ </div>
390
+ `,
391
+ dangerouslyUseHTML: true
392
+ });
393
+ return false;
394
+ }
395
+ window.location.href = universalLink;
396
+ return true;
397
+ }
398
+ return false;
399
+ });
400
+ }
401
+ };
402
+
63
403
  // src/connector/injected.ts
64
404
  var InjectedConnector = class extends BaseConnector {
65
405
  constructor(propertity) {
@@ -75,21 +415,50 @@ var InjectedConnector = class extends BaseConnector {
75
415
  if (typeof window !== "undefined") {
76
416
  const props = this.propertity.split(".");
77
417
  if (props.length === 1) {
78
- return typeof window[props[0]] !== "undefined";
418
+ if (typeof window[props[0]] !== "undefined") {
419
+ return true;
420
+ }
79
421
  } else {
80
- return typeof window[props[0]] !== "undefined" && typeof window[props[0]][props[1]] !== "undefined";
422
+ if (typeof window[props[0]] !== "undefined" && typeof window[props[0]][props[1]] !== "undefined") {
423
+ return true;
424
+ }
425
+ }
426
+ if (isMobile()) {
427
+ return true;
81
428
  }
82
429
  }
83
430
  return false;
84
431
  }
85
432
  requestAccounts() {
86
433
  return __async(this, null, function* () {
434
+ if (isMobile() && !this.getProvider()) {
435
+ MobileWalletConnect.redirectToWallet(this.metadata.id);
436
+ return [];
437
+ }
87
438
  const accounts = yield this.getProviderOrThrow().requestAccounts();
88
439
  console.log("network:", yield this.getNetwork());
89
440
  console.log("\u{1F680} ~ InjectedConnector ~ requestAccounts ~ accounts:", accounts);
90
441
  return accounts;
91
442
  });
92
443
  }
444
+ getUniversalLink(url) {
445
+ url = "https://www.deltatrade.ai/";
446
+ console.log(this.metadata.id);
447
+ switch (this.metadata.id) {
448
+ case "unisat":
449
+ return `unisat://dapp?url=${encodeURIComponent(url)}`;
450
+ case "okx":
451
+ return `okx://wallet/dapp/url?dappUrl=${encodeURIComponent(url)}`;
452
+ case "bitget":
453
+ return `https://bkcode.vip?action=dapp&url=${encodeURIComponent(url)}`;
454
+ case "binance":
455
+ return `binance://dapp?url=${encodeURIComponent(url)}`;
456
+ case "xverse":
457
+ return `https://connect.xverse.app/browser?url=${encodeURIComponent(url)}`;
458
+ default:
459
+ return url;
460
+ }
461
+ }
93
462
  getAccounts() {
94
463
  return __async(this, null, function* () {
95
464
  const accounts = yield this.getProviderOrThrow().getAccounts();
@@ -306,7 +675,13 @@ var XverseConnector = class extends BaseConnector {
306
675
  __privateGet(this, _event).setMaxListeners(100);
307
676
  }
308
677
  isReady() {
309
- return typeof window !== "undefined" && typeof window.BitcoinProvider !== "undefined";
678
+ if (typeof window !== "undefined") {
679
+ if (typeof window.BitcoinProvider !== "undefined")
680
+ return true;
681
+ if (isMobile())
682
+ return true;
683
+ }
684
+ return false;
310
685
  }
311
686
  sendInscription() {
312
687
  return __async(this, null, function* () {
@@ -315,8 +690,9 @@ var XverseConnector = class extends BaseConnector {
315
690
  }
316
691
  requestAccounts() {
317
692
  return __async(this, null, function* () {
318
- if (!this.isReady()) {
319
- throw new Error(`${this.metadata.name} is not install!`);
693
+ if (isMobile() && !this.getProvider()) {
694
+ MobileWalletConnect.redirectToWallet(this.metadata.id);
695
+ return [];
320
696
  }
321
697
  const addresses = yield this.loadAccounts(__privateGet(this, _network));
322
698
  return addresses.map((item) => item.address);
@@ -637,6 +1013,22 @@ var MagicEdenConnector = class extends BaseConnector {
637
1013
  _network2 = new WeakMap();
638
1014
  _event2 = new WeakMap();
639
1015
 
1016
+ // src/icons/binance.png
1017
+ var binance_default = "";
1018
+
1019
+ // src/connector/binance.ts
1020
+ var BinanceConnector = class extends InjectedConnector {
1021
+ constructor() {
1022
+ super("binancew3w.bitcoin");
1023
+ this.metadata = {
1024
+ id: "binance",
1025
+ name: "Binance Wallet",
1026
+ icon: binance_default,
1027
+ downloadUrl: "https://www.binance.com/en/web3wallet"
1028
+ };
1029
+ }
1030
+ };
1031
+
640
1032
  // src/context/index.tsx
641
1033
  import { SmartAccount } from "@particle-network/aa";
642
1034
  import { chains as chains4 } from "@particle-network/chains";
@@ -1202,116 +1594,31 @@ var ConnectModal = ({ open, onClose }) => {
1202
1594
  button_default,
1203
1595
  {
1204
1596
  className: connect_module_default.btnDownload,
1205
- onClick: () => {
1206
- window.open(selectConnector == null ? void 0 : selectConnector.metadata.downloadUrl, "_blank");
1207
- },
1208
- children: "Get"
1209
- }
1210
- )
1211
- ] })
1212
- ] })
1213
- ] });
1214
- };
1215
- var connectModal_default = ConnectModal;
1216
-
1217
- // src/components/signModal/index.tsx
1218
- import { chains as chains2 } from "@particle-network/chains";
1219
- import { useCallback as useCallback5, useEffect as useEffect4, useMemo as useMemo4, useState as useState5 } from "react";
1220
- import { formatEther, hexToString } from "viem";
1221
-
1222
- // src/icons/check_box.svg
1223
- var check_box_default = "";
1224
-
1225
- // src/icons/check_box_blank.svg
1226
- var check_box_blank_default = "";
1227
-
1228
- // src/icons/copy.svg
1229
- var copy_default = "";
1230
-
1231
- // src/utils/index.ts
1232
- function shortString(str) {
1233
- if (Array.isArray(str)) {
1234
- str = "[" + str.toString() + "]";
1235
- }
1236
- if (str) {
1237
- if (typeof str.toString === "function") {
1238
- str = str.toString();
1239
- }
1240
- if (str.length <= 10) {
1241
- return str;
1242
- }
1243
- return `${str.slice(0, 5)}...${str.slice(str.length - 5, str.length)}`;
1244
- }
1245
- return "";
1246
- }
1247
- function copyToClipboard(text) {
1248
- return __async(this, null, function* () {
1249
- const clipboardCopy = () => __async(this, null, function* () {
1250
- if (navigator.clipboard) {
1251
- return navigator.clipboard.writeText(text);
1252
- } else {
1253
- const textarea = document.createElement("textarea");
1254
- textarea.value = text;
1255
- textarea.setAttribute("readonly", "");
1256
- textarea.style.position = "absolute";
1257
- textarea.style.left = "-9999px";
1258
- document.body.appendChild(textarea);
1259
- textarea.select();
1260
- const result = document.execCommand("copy");
1261
- document.body.removeChild(textarea);
1262
- if (!result) {
1263
- throw new Error("Copy to clipboard failed");
1264
- }
1265
- }
1266
- });
1267
- return new Promise((resolve, reject) => {
1268
- clipboardCopy().then(resolve).catch(reject);
1269
- });
1270
- });
1271
- }
1272
- var defaultTokenIcon = "https://static.particle.network/token-list/defaultToken/default.png";
1273
- var ipfsToSrc = (ipfs) => {
1274
- if (!ipfs || !ipfs.startsWith("ipfs://")) {
1275
- return ipfs || "";
1276
- }
1277
- return `https://ipfs.particle.network/${encodeURI(ipfs.slice(7))}`;
1278
- };
1279
- var checkBTCVersion = (accountContracts, accountContractKey, version) => {
1280
- if (!accountContracts[accountContractKey]) {
1281
- return false;
1282
- }
1283
- return accountContracts[accountContractKey].some((item) => item.version === version);
1597
+ onClick: () => {
1598
+ window.open(selectConnector == null ? void 0 : selectConnector.metadata.downloadUrl, "_blank");
1599
+ },
1600
+ children: "Get"
1601
+ }
1602
+ )
1603
+ ] })
1604
+ ] })
1605
+ ] });
1284
1606
  };
1285
- var delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
1286
- function retryOperation(_0, _1) {
1287
- return __async(this, arguments, function* (operation, shouldStop, {
1288
- maxRetries = 3,
1289
- delayMs = 1e3
1290
- } = {}) {
1291
- let retries = 0;
1292
- while (retries <= maxRetries) {
1293
- const result = yield operation();
1294
- if (shouldStop(result)) {
1295
- return result;
1296
- }
1297
- if (retries === maxRetries) {
1298
- console.warn("Max retries reached");
1299
- return result;
1300
- }
1301
- retries++;
1302
- yield delay(delayMs);
1303
- }
1304
- throw new Error("Unexpected execution path");
1305
- });
1306
- }
1307
- function toHex(originalString) {
1308
- const charArray = originalString.split("");
1309
- const asciiArray = charArray.map((char) => char.charCodeAt(0));
1310
- const hexArray = asciiArray.map((code) => code.toString(16));
1311
- let hexString = hexArray.join("");
1312
- hexString = hexString.replace(/(^0+)/g, "");
1313
- return hexString;
1314
- }
1607
+ var connectModal_default = ConnectModal;
1608
+
1609
+ // src/components/signModal/index.tsx
1610
+ import { chains as chains2 } from "@particle-network/chains";
1611
+ import { useCallback as useCallback5, useEffect as useEffect4, useMemo as useMemo4, useState as useState5 } from "react";
1612
+ import { formatEther, hexToString } from "viem";
1613
+
1614
+ // src/icons/check_box.svg
1615
+ var check_box_default = "";
1616
+
1617
+ // src/icons/check_box_blank.svg
1618
+ var check_box_blank_default = "";
1619
+
1620
+ // src/icons/copy.svg
1621
+ var copy_default = "";
1315
1622
 
1316
1623
  // src/utils/ethereumUtils.ts
1317
1624
  import {
@@ -2569,15 +2876,15 @@ var walletConfig = {
2569
2876
  accountContractId: "acc.toalice.near",
2570
2877
  bridgeContractId: "brg.toalice.near",
2571
2878
  walletUrl: "https://wallet-stg.satoshibridge.top",
2572
- bridgeUrl: "https://ramp.satos.network/"
2879
+ bridgeUrl: "https://old.ramp.satos.network"
2573
2880
  },
2574
2881
  mainnet: {
2575
- base_url: "https://api.mainnet.satoshibridge.top",
2576
- token: "nbtc.toalice.near",
2577
- accountContractId: "acc.toalice.near",
2578
- bridgeContractId: "brg.toalice.near",
2882
+ base_url: "https://api.satos.network",
2883
+ token: "nbtc.bridge.near",
2884
+ accountContractId: "acc.ref-labs.near",
2885
+ bridgeContractId: "btc-connector.bridge.near",
2579
2886
  walletUrl: "https://wallet.satoshibridge.top",
2580
- bridgeUrl: "https://www.satoshibridge.top/"
2887
+ bridgeUrl: "https://ramp.satos.network"
2581
2888
  }
2582
2889
  };
2583
2890
  var nearRpcUrls = {
@@ -2650,6 +2957,7 @@ function createFloatingButtonWithIframe({
2650
2957
  userSelect: "none"
2651
2958
  });
2652
2959
  document.body.appendChild(button);
2960
+ updateIframePosition(iframe, right, bottom, windowWidth, windowHeight);
2653
2961
  let isDragging = false;
2654
2962
  let startX = 0;
2655
2963
  let startY = 0;
@@ -3081,205 +3389,6 @@ function getWhitelist(url) {
3081
3389
  });
3082
3390
  }
3083
3391
 
3084
- // src/utils/Dialog.ts
3085
- var Dialog = class {
3086
- static injectStyles() {
3087
- if (!document.querySelector("#dialog-styles")) {
3088
- const styleSheet = document.createElement("style");
3089
- styleSheet.id = "dialog-styles";
3090
- styleSheet.textContent = this.style;
3091
- document.head.appendChild(styleSheet);
3092
- }
3093
- }
3094
- static confirm(options) {
3095
- return new Promise((resolve) => {
3096
- this.injectStyles();
3097
- const container = document.createElement("div");
3098
- container.innerHTML = this.template;
3099
- document.body.appendChild(container);
3100
- const titleEl = container.querySelector(".dialog-title");
3101
- const messageEl = container.querySelector(".dialog-message");
3102
- const confirmBtn = container.querySelector(".dialog-confirm-btn");
3103
- const cancelBtn = container.querySelector(".dialog-cancel-btn");
3104
- if (options.title) {
3105
- titleEl.textContent = options.title;
3106
- } else {
3107
- titleEl.style.display = "none";
3108
- }
3109
- messageEl.textContent = options.message;
3110
- const cleanup = () => {
3111
- document.body.removeChild(container);
3112
- };
3113
- confirmBtn.addEventListener("click", () => {
3114
- cleanup();
3115
- resolve(true);
3116
- });
3117
- cancelBtn.addEventListener("click", () => {
3118
- cleanup();
3119
- resolve(false);
3120
- });
3121
- });
3122
- }
3123
- static alert(options) {
3124
- const messageEl = options.dangerouslyUseHTML ? { dangerouslySetInnerHTML: { __html: options.message } } : { children: options.message };
3125
- return new Promise((resolve) => {
3126
- var _a;
3127
- this.injectStyles();
3128
- const container = document.createElement("div");
3129
- container.innerHTML = this.template;
3130
- (_a = container.querySelector(".dialog-overlay")) == null ? void 0 : _a.classList.add("dialog-alert");
3131
- if (options.closable === false) {
3132
- const overlay = container.querySelector(".dialog-overlay");
3133
- overlay.style.pointerEvents = "none";
3134
- const dialogContainer = container.querySelector(".dialog-container");
3135
- dialogContainer.style.pointerEvents = "auto";
3136
- }
3137
- document.body.appendChild(container);
3138
- const titleEl = container.querySelector(".dialog-title");
3139
- const messageEl2 = container.querySelector(".dialog-message");
3140
- const confirmBtn = container.querySelector(".dialog-confirm-btn");
3141
- const cancelBtn = container.querySelector(".dialog-cancel-btn");
3142
- if (options.title) {
3143
- titleEl.textContent = options.title;
3144
- } else {
3145
- titleEl.style.display = "none";
3146
- }
3147
- messageEl2.innerHTML = options.message;
3148
- cancelBtn.style.display = "none";
3149
- if (options.closable === false) {
3150
- confirmBtn.style.display = "none";
3151
- }
3152
- const cleanup = () => {
3153
- if (options.closable === false) {
3154
- return;
3155
- }
3156
- document.body.removeChild(container);
3157
- };
3158
- confirmBtn.addEventListener("click", () => {
3159
- cleanup();
3160
- resolve();
3161
- });
3162
- });
3163
- }
3164
- };
3165
- Dialog.template = `
3166
- <div class="dialog-overlay">
3167
- <div class="dialog-container">
3168
- <div class="dialog-content">
3169
- <div class="dialog-title"></div>
3170
- <div class="dialog-message"></div>
3171
- <div class="dialog-buttons">
3172
- <button class="dialog-cancel-btn">Cancel</button>
3173
- <button class="dialog-confirm-btn">OK</button>
3174
- </div>
3175
- </div>
3176
- </div>
3177
- </div>
3178
- `;
3179
- Dialog.style = `
3180
- .dialog-overlay {
3181
- position: fixed;
3182
- top: 0;
3183
- left: 0;
3184
- right: 0;
3185
- bottom: 0;
3186
- background-color: rgba(0, 0, 0, 0.75);
3187
- display: flex;
3188
- align-items: center;
3189
- justify-content: center;
3190
- z-index: 999999;
3191
- backdrop-filter: blur(4px);
3192
- }
3193
- .dialog-container {
3194
- background: #21232f;
3195
- border-radius: 12px;
3196
- padding: 24px;
3197
- width: 350px;
3198
- box-shadow: 0 4px 24px rgba(0, 0, 0, 0.3);
3199
- }
3200
- .dialog-title {
3201
- font-size: 18px;
3202
- font-weight: 600;
3203
- margin-bottom: 16px;
3204
- color: #ffffff;
3205
- }
3206
- .dialog-message {
3207
- margin-bottom: 24px;
3208
- line-height: 1.6;
3209
- color: rgba(255, 255, 255, 0.8);
3210
- font-size: 14px;
3211
- }
3212
- .dialog-buttons {
3213
- display: flex;
3214
- justify-content: flex-end;
3215
- gap: 12px;
3216
- }
3217
- .dialog-alert .dialog-buttons {
3218
- justify-content: center;
3219
- }
3220
- .dialog-confirm-btn {
3221
- padding: 8px 24px;
3222
- background-color: #ff7a00;
3223
- color: white;
3224
- border: none;
3225
- border-radius: 6px;
3226
- cursor: pointer;
3227
- font-size: 14px;
3228
- font-weight: 500;
3229
- transition: all 0.2s ease;
3230
- }
3231
- .dialog-confirm-btn:hover {
3232
- background-color: #ff8f1f;
3233
- transform: translateY(-1px);
3234
- }
3235
- .dialog-confirm-btn:active {
3236
- transform: translateY(0);
3237
- }
3238
- .dialog-cancel-btn {
3239
- padding: 8px 24px;
3240
- background-color: rgba(255, 255, 255, 0.1);
3241
- color: rgba(255, 255, 255, 0.8);
3242
- border: none;
3243
- border-radius: 6px;
3244
- cursor: pointer;
3245
- font-size: 14px;
3246
- font-weight: 500;
3247
- transition: all 0.2s ease;
3248
- }
3249
- .dialog-cancel-btn:hover {
3250
- background-color: rgba(255, 255, 255, 0.15);
3251
- transform: translateY(-1px);
3252
- }
3253
- .dialog-cancel-btn:active {
3254
- transform: translateY(0);
3255
- }
3256
-
3257
- .dialog-overlay {
3258
- animation: fadeIn 0.2s ease;
3259
- }
3260
- .dialog-container {
3261
- animation: slideIn 0.2s ease;
3262
- }
3263
- @keyframes fadeIn {
3264
- from {
3265
- opacity: 0;
3266
- }
3267
- to {
3268
- opacity: 1;
3269
- }
3270
- }
3271
- @keyframes slideIn {
3272
- from {
3273
- transform: translateY(-20px);
3274
- opacity: 0;
3275
- }
3276
- to {
3277
- transform: translateY(0);
3278
- opacity: 1;
3279
- }
3280
- }
3281
- `;
3282
-
3283
3392
  // src/core/btcUtils.ts
3284
3393
  import bitcoin from "bitcoinjs-lib";
3285
3394
  import coinselect from "coinselect";
@@ -3396,7 +3505,7 @@ function getBtcBalance() {
3396
3505
  const { account } = yield retryOperation(getBtcProvider, (res) => !!res.account);
3397
3506
  if (!account) {
3398
3507
  console.error("BTC Account is not available.");
3399
- return { rawBalance: 0, balance: 0, maxSpendableBalance: 0 };
3508
+ return { rawBalance: 0, balance: 0, availableBalance: 0 };
3400
3509
  }
3401
3510
  const btcRpcUrl = yield getBtcRpcUrl();
3402
3511
  const utxos = yield fetch(`${btcRpcUrl}/address/${account}/utxo`).then((res) => res.json());
@@ -3404,11 +3513,12 @@ function getBtcBalance() {
3404
3513
  const rawBalance = (utxos == null ? void 0 : utxos.reduce((acc, cur) => acc + cur.value, 0)) || 0;
3405
3514
  const balance = rawBalance / __pow(10, btcDecimals);
3406
3515
  const feeRate = yield getBtcGasPrice();
3407
- const inputSize = ((utxos == null ? void 0 : utxos.length) || 0) * 66;
3408
- const outputSize = 34;
3409
- const overheadSize = 10;
3516
+ const inputSize = ((utxos == null ? void 0 : utxos.length) || 0) * 69;
3517
+ const outputSize = 33 * 2;
3518
+ const overheadSize = 11;
3410
3519
  const estimatedTxSize = inputSize + outputSize + overheadSize;
3411
- const estimatedFee = estimatedTxSize * feeRate;
3520
+ const estimatedFee = Math.ceil(estimatedTxSize * feeRate);
3521
+ console.log("estimatedFee:", estimatedFee);
3412
3522
  const availableRawBalance = (rawBalance - estimatedFee).toFixed(0);
3413
3523
  const availableBalance = new Big(availableRawBalance).div(__pow(10, btcDecimals)).round(btcDecimals, Big.roundDown).toNumber();
3414
3524
  return {
@@ -4277,13 +4387,14 @@ function setupBTCWallet({
4277
4387
 
4278
4388
  // src/index.ts
4279
4389
  var getVersion = () => {
4280
- return "0.4.7-beta";
4390
+ return "0.5.0-beta";
4281
4391
  };
4282
4392
  if (typeof window !== "undefined") {
4283
4393
  window.__BTC_WALLET_VERSION = getVersion();
4284
4394
  }
4285
4395
  export {
4286
4396
  BaseConnector,
4397
+ BinanceConnector,
4287
4398
  BitgetConnector,
4288
4399
  BtcWalletSelectorContextProvider,
4289
4400
  BybitConnector,