UNPKG

btc-wallet

Version:

BTC Wallet is a toolkit that enables Bitcoin usage on the NEAR blockchain through the Satoshi protocol.

796 lines (780 loc) 278 kB
"use client"; "use strict"; var __create = Object.create; var __defProp = Object.defineProperty; var __defProps = Object.defineProperties; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropDescs = Object.getOwnPropertyDescriptors; var __getOwnPropNames = Object.getOwnPropertyNames; var __getOwnPropSymbols = Object.getOwnPropertySymbols; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __propIsEnum = Object.prototype.propertyIsEnumerable; var __pow = Math.pow; var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __spreadValues = (a, b) => { for (var prop in b || (b = {})) if (__hasOwnProp.call(b, prop)) __defNormalProp(a, prop, b[prop]); if (__getOwnPropSymbols) for (var prop of __getOwnPropSymbols(b)) { if (__propIsEnum.call(b, prop)) __defNormalProp(a, prop, b[prop]); } return a; }; var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b)); var __objRest = (source, exclude) => { var target = {}; for (var prop in source) if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0) target[prop] = source[prop]; if (source != null && __getOwnPropSymbols) for (var prop of __getOwnPropSymbols(source)) { if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop)) target[prop] = source[prop]; } return target; }; var __export = (target, all) => { for (var name in all) __defProp(target, name, { get: all[name], enumerable: true }); }; var __copyProps = (to, from, except, desc) => { if (from && typeof from === "object" || typeof from === "function") { for (let key of __getOwnPropNames(from)) if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); } return to; }; var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps( isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod )); var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); var __accessCheck = (obj, member, msg) => { if (!member.has(obj)) throw TypeError("Cannot " + msg); }; var __privateGet = (obj, member, getter) => { __accessCheck(obj, member, "read from private field"); return getter ? getter.call(obj) : member.get(obj); }; var __privateAdd = (obj, member, value) => { if (member.has(obj)) throw TypeError("Cannot add the same private member more than once"); member instanceof WeakSet ? member.add(obj) : member.set(obj, value); }; var __async = (__this, __arguments, generator) => { return new Promise((resolve, reject) => { var fulfilled = (value) => { try { step(generator.next(value)); } catch (e) { reject(e); } }; var rejected = (value) => { try { step(generator.throw(value)); } catch (e) { reject(e); } }; var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected); step((generator = generator.apply(__this, __arguments)).next()); }); }; // src/index.ts var src_exports = {}; __export(src_exports, { BaseConnector: () => BaseConnector, BinanceConnector: () => BinanceConnector, BitgetConnector: () => BitgetConnector, BtcWalletSelectorContextProvider: () => BtcWalletSelectorContextProvider, BybitConnector: () => BybitConnector, ConnectProvider: () => ConnectProvider, GateConnector: () => GateConnector, InjectedConnector: () => InjectedConnector, MagicEdenConnector: () => MagicEdenConnector, OKXConnector: () => OKXConnector, TokenPocketConnector: () => TokenPocketConnector, UnisatConnector: () => UnisatConnector, WizzConnector: () => WizzConnector, XverseConnector: () => XverseConnector, btcRpcUrls: () => btcRpcUrls, calculateGasFee: () => calculateGasFee, calculateGasLimit: () => calculateGasLimit, calculateGasStrategy: () => calculateGasStrategy, calculateWithdraw: () => calculateWithdraw, checkBridgeTransactionStatus: () => checkBridgeTransactionStatus, checkGasTokenDebt: () => checkGasTokenDebt, checkNewAccount: () => checkNewAccount, checkSatoshiWhitelist: () => checkSatoshiWhitelist, estimateDepositAmount: () => estimateDepositAmount, executeBTCDepositAndAction: () => executeBTCDepositAndAction, getAddressByPublicKeyBase58: () => getAddressByPublicKeyBase58, getBtcBalance: () => getBtcBalance, getBtcGasPrice: () => getBtcGasPrice, getBtcUtxos: () => getBtcUtxos, getCsnaAccountId: () => getCsnaAccountId, getDepositAmount: () => getDepositAmount, getPublicKeyBase58: () => getPublicKeyBase58, getVersion: () => getVersion, getWalletConfig: () => getWalletConfig, getWithdrawTransaction: () => getWithdrawTransaction, nearRpcUrls: () => nearRpcUrls, sendBitcoin: () => sendBitcoin, setupBTCWallet: () => setupBTCWallet, setupWalletSelectorModal: () => setupWalletSelectorModal, signMessage: () => signMessage, useAccountContract: () => useAccountContract, useAccounts: () => useAccounts, useBTCProvider: () => useBTCProvider, useBtcWalletSelector: () => useBtcWalletSelector, useConnectModal: () => useConnectModal, useConnector: () => useConnector, useETHProvider: () => useETHProvider, walletConfig: () => walletConfig }); module.exports = __toCommonJS(src_exports); // src/connector/base.ts var BaseConnector = class { }; // src/icons/bitget.png var bitget_default = ""; // src/utils/index.ts function shortString(str) { if (Array.isArray(str)) { str = "[" + str.toString() + "]"; } if (str) { if (typeof str.toString === "function") { str = str.toString(); } if (str.length <= 10) { return str; } return `${str.slice(0, 5)}...${str.slice(str.length - 5, str.length)}`; } return ""; } function copyToClipboard(text) { return __async(this, null, function* () { const clipboardCopy = () => __async(this, null, function* () { if (navigator.clipboard) { return navigator.clipboard.writeText(text); } else { const textarea = document.createElement("textarea"); textarea.value = text; textarea.setAttribute("readonly", ""); textarea.style.position = "absolute"; textarea.style.left = "-9999px"; document.body.appendChild(textarea); textarea.select(); const result = document.execCommand("copy"); document.body.removeChild(textarea); if (!result) { throw new Error("Copy to clipboard failed"); } } }); return new Promise((resolve, reject) => { clipboardCopy().then(resolve).catch(reject); }); }); } var defaultTokenIcon = "https://static.particle.network/token-list/defaultToken/default.png"; var ipfsToSrc = (ipfs) => { if (!ipfs || !ipfs.startsWith("ipfs://")) { return ipfs || ""; } return `https://ipfs.particle.network/${encodeURI(ipfs.slice(7))}`; }; var checkBTCVersion = (accountContracts, accountContractKey, version) => { if (!accountContracts[accountContractKey]) { return false; } return accountContracts[accountContractKey].some((item) => item.version === version); }; var delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms)); function retryOperation(_0, _1) { return __async(this, arguments, function* (operation, shouldStop, { maxRetries = 3, delayMs = 1e3 } = {}) { let retries = 0; while (retries <= maxRetries) { const result = yield operation(); if (shouldStop(result)) { return result; } if (retries === maxRetries) { console.warn("Max retries reached"); return result; } retries++; yield delay(delayMs); } throw new Error("Unexpected execution path"); }); } function toHex(originalString) { const charArray = originalString.split(""); const asciiArray = charArray.map((char) => char.charCodeAt(0)); const hexArray = asciiArray.map((code) => code.toString(16)); let hexString = hexArray.join(""); hexString = hexString.replace(/(^0+)/g, ""); return hexString; } function isMobile() { if (typeof window !== "undefined") { return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test( navigator == null ? void 0 : navigator.userAgent ); } return false; } function safeJSONParse(str) { try { return JSON.parse(str); } catch (e) { console.error("safeJSONParse", e); return void 0; } } function safeJSONStringify(obj) { try { return JSON.stringify(obj); } catch (e) { console.error("safeJSONStringify", e); return void 0; } } function storageStore(namespace, options) { if (typeof window === "undefined") return; const _namespace = namespace || "SATOSHI_WALLET_DEFAULT"; const storage5 = (options == null ? void 0 : options.storage) || (window == null ? void 0 : window.localStorage); const namespaceKey = (key) => { return _namespace + ":" + key; }; return { set(key, value) { const _value = safeJSONStringify(value); _value ? storage5.setItem(namespaceKey(key), _value) : storage5.removeItem(namespaceKey(key)); }, get(key) { const _value = storage5.getItem(namespaceKey(key)); return _value ? safeJSONParse(_value) : void 0; }, remove(key) { storage5.removeItem(namespaceKey(key)); }, clearAll: function clearAll() { for (const key in storage5) { if (key.startsWith(namespace + ":")) { storage5.removeItem(key); } } } }; } // src/utils/Dialog.ts var Dialog = class { static injectStyles() { if (!document.querySelector("#dialog-styles")) { const styleSheet = document.createElement("style"); styleSheet.id = "dialog-styles"; styleSheet.textContent = this.style; document.head.appendChild(styleSheet); } } static confirm(options) { return new Promise((resolve) => { this.injectStyles(); const container = document.createElement("div"); container.innerHTML = this.template; document.body.appendChild(container); const titleEl = container.querySelector(".dialog-title"); const messageEl = container.querySelector(".dialog-message"); const confirmBtn = container.querySelector(".dialog-confirm-btn"); const cancelBtn = container.querySelector(".dialog-cancel-btn"); if (options.title) { titleEl.textContent = options.title; } else { titleEl.style.display = "none"; } messageEl.textContent = options.message; const cleanup = () => { document.body.removeChild(container); }; confirmBtn.addEventListener("click", () => { cleanup(); resolve(true); }); cancelBtn.addEventListener("click", () => { cleanup(); resolve(false); }); }); } static alert(options) { const messageEl = options.dangerouslyUseHTML ? { dangerouslySetInnerHTML: { __html: options.message } } : { children: options.message }; return new Promise((resolve) => { var _a; this.injectStyles(); const container = document.createElement("div"); container.innerHTML = this.template; (_a = container.querySelector(".dialog-overlay")) == null ? void 0 : _a.classList.add("dialog-alert"); if (options.closable === false) { const overlay = container.querySelector(".dialog-overlay"); overlay.style.pointerEvents = "none"; const dialogContainer = container.querySelector(".dialog-container"); dialogContainer.style.pointerEvents = "auto"; } document.body.appendChild(container); const titleEl = container.querySelector(".dialog-title"); const messageEl2 = container.querySelector(".dialog-message"); const confirmBtn = container.querySelector(".dialog-confirm-btn"); const cancelBtn = container.querySelector(".dialog-cancel-btn"); if (options.title) { titleEl.textContent = options.title; } else { titleEl.style.display = "none"; } messageEl2.innerHTML = options.message; cancelBtn.style.display = "none"; if (options.closable === false) { confirmBtn.style.display = "none"; } const cleanup = () => { if (options.closable === false) { return; } document.body.removeChild(container); }; confirmBtn.addEventListener("click", () => { cleanup(); resolve(); }); }); } static openModal({ title, titleStyle, content }) { return new Promise((resolve) => { const modalContainer = document.createElement("div"); modalContainer.innerHTML = this.template; document.body.appendChild(modalContainer); const btns = modalContainer.querySelector(".dialog-buttons"); btns.style.display = "none"; this.injectStyles(); const titleEl = modalContainer.querySelector(".dialog-title"); if (title) { titleEl.textContent = title; if (titleStyle) { titleEl.style.cssText = titleStyle; } } else { titleEl.style.display = "none"; } const cleanup = () => { document.body.removeChild(modalContainer); }; const close = () => { cleanup(); resolve(null); }; const messageEl = modalContainer.querySelector(".dialog-message"); messageEl.appendChild(content(resolve, close)); const overlay = modalContainer.querySelector(".dialog-overlay"); overlay.addEventListener("click", (e) => { if (e.target === overlay) { close(); } }); }); } }; Dialog.template = ` <div class="dialog-overlay"> <div class="dialog-container"> <div class="dialog-content"> <div class="dialog-title"></div> <div class="dialog-message"></div> <div class="dialog-buttons"> <button class="dialog-cancel-btn">Cancel</button> <button class="dialog-confirm-btn">OK</button> </div> </div> </div> </div> `; Dialog.style = ` .dialog-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background-color: rgba(0, 0, 0, 0.75); display: flex; align-items: center; justify-content: center; z-index: 999999; backdrop-filter: blur(4px); } .dialog-container { background: #131313; border-radius: 12px; padding: 24px; width: 350px; box-shadow: 0 4px 24px rgba(0, 0, 0, 0.3); } .dialog-title { font-size: 18px; font-weight: 600; margin-bottom: 16px; color: #ffffff; } .dialog-message { margin-bottom: 24px; line-height: 1.6; color: rgba(255, 255, 255, 0.8); font-size: 14px; } .dialog-buttons { display: flex; justify-content: flex-end; gap: 12px; } .dialog-alert .dialog-buttons { justify-content: center; } .dialog-confirm-btn { padding: 8px 24px; background-color: #ff7a00; color: white; border: none; border-radius: 6px; cursor: pointer; font-size: 14px; font-weight: 500; transition: all 0.2s ease; } .dialog-confirm-btn:hover { background-color: #ff8f1f; transform: translateY(-1px); } .dialog-confirm-btn:active { transform: translateY(0); } .dialog-cancel-btn { padding: 8px 24px; background-color: rgba(255, 255, 255, 0.1); color: rgba(255, 255, 255, 0.8); border: none; border-radius: 6px; cursor: pointer; font-size: 14px; font-weight: 500; transition: all 0.2s ease; } .dialog-cancel-btn:hover { background-color: rgba(255, 255, 255, 0.15); transform: translateY(-1px); } .dialog-cancel-btn:active { transform: translateY(0); } .dialog-overlay { animation: fadeIn 0.2s ease; } .dialog-container { animation: slideIn 0.2s ease; } @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } @keyframes slideIn { from { transform: translateY(-20px); opacity: 0; } to { transform: translateY(0); opacity: 1; } } .option-list { display: flex; flex-direction: column; gap: 12px; } .option-item { display: flex; align-items: center; gap: 12px; color: #fff; border: 1px solid transparent; background-color: hsla(0, 0%, 100%, .05); border-radius: 8px; padding: 8px 16px; transition: all 0.15s ease; cursor: pointer; font-size: 14px; font-weight: bold; &:hover { border-color: #ff7a00; color: #ff7a00; background-color: hsla(0, 0%, 100%, .08); } } `; // src/connector/universalLink.ts var MobileWalletConnect = class { static getUniversalLink(walletId, url) { switch (walletId) { case "okx": return `okx://wallet/dapp/url?dappUrl=${encodeURIComponent(url)}`; case "bitget": return `bitkeep://bkconnect?action=dapp&url=${encodeURIComponent(url)}`; case "xverse": return `xverse://browser?url=${encodeURIComponent(url)}`; default: return ""; } } static redirectToWallet(walletId) { return __async(this, null, function* () { if (isMobile()) { const currentUrl = window.location.href; const universalLink = this.getUniversalLink(walletId, currentUrl); const showGuideDialog = () => __async(this, null, function* () { var _a; try { yield (_a = navigator.clipboard) == null ? void 0 : _a.writeText(currentUrl); } catch (error) { console.error(error); } yield Dialog.alert({ title: "Open in Wallet Browser", message: ` <div style="display: flex; flex-direction: column; gap: 12px;"> <p>Please follow these steps:</p> <p>1. Open ${walletId} wallet app</p> <p>2. Find the browser feature in the wallet</p> <p>3. Paste the URL (already copied to clipboard)</p> </div> `, dangerouslyUseHTML: true }); }); if (!universalLink) { yield showGuideDialog(); return false; } const openWallet = () => { const iframe = document.createElement("iframe"); iframe.style.display = "none"; iframe.src = universalLink; document.body.appendChild(iframe); setTimeout(() => __async(this, null, function* () { document.body.removeChild(iframe); yield showGuideDialog(); }), 2e3); }; window.location.href = universalLink; setTimeout(openWallet, 100); return true; } return false; }); } }; // src/connector/injected.ts var InjectedConnector = class extends BaseConnector { constructor(propertity) { var _a; super(); this.propertity = propertity; const props = (_a = this.propertity) == null ? void 0 : _a.split("."); if (!this.propertity || props.length > 2) { throw new Error("please input valid propertity"); } } isReady() { if (typeof window !== "undefined") { const props = this.propertity.split("."); if (props.length === 1) { if (typeof window[props[0]] !== "undefined") { return true; } } else { if (typeof window[props[0]] !== "undefined" && typeof window[props[0]][props[1]] !== "undefined") { return true; } } if (isMobile()) { return true; } } return false; } requestAccounts() { return __async(this, null, function* () { if (isMobile()) { try { this.getProvider(); } catch (error) { yield MobileWalletConnect.redirectToWallet(this.metadata.id); return []; } } const accounts = yield this.getProviderOrThrow().requestAccounts(); console.log("network:", yield this.getNetwork()); console.log("\u{1F680} ~ InjectedConnector ~ requestAccounts ~ accounts:", accounts); return accounts; }); } getAccounts() { return __async(this, null, function* () { const accounts = yield this.getProviderOrThrow().getAccounts(); return accounts; }); } getPublicKey() { return __async(this, null, function* () { return this.getProviderOrThrow().getPublicKey(); }); } signMessage(signStr, type) { return __async(this, null, function* () { const addresses = yield this.getAccounts(); if (addresses.length === 0) { throw new Error(`${this.metadata.name} not connected!`); } return this.getProviderOrThrow().signMessage(signStr, type); }); } on(event, handler) { var _a; const provider = this.getProvider(); return (_a = provider == null ? void 0 : provider.on) == null ? void 0 : _a.call(provider, event, handler); } removeListener(event, handler) { var _a; const provider = this.getProvider(); return (_a = provider == null ? void 0 : provider.removeListener) == null ? void 0 : _a.call(provider, event, handler); } getProvider() { if (this.isReady()) { const props = this.propertity.split("."); if (props.length === 1) { return window[props[0]]; } else { return window[props[0]][props[1]]; } } } getProviderOrThrow() { const provider = this.getProvider(); if (!provider) { throw new Error(`${this.metadata.name} is not install or not create Bitcoin wallet!`); } return provider; } getNetwork() { return __async(this, null, function* () { return this.getProviderOrThrow().getNetwork(); }); } switchNetwork(network) { return __async(this, null, function* () { return this.getProviderOrThrow().switchNetwork(network); }); } sendBitcoin(toAddress, satoshis, options) { return __async(this, null, function* () { return this.getProviderOrThrow().sendBitcoin(toAddress, satoshis, options); }); } sendInscription(address, inscriptionId, options) { return __async(this, null, function* () { const result = yield this.getProviderOrThrow().sendInscription(address, inscriptionId, options); if (typeof result === "string") { return { txid: result }; } return result; }); } disconnect() { } }; // src/connector/bitget.ts var BitgetConnector = class extends InjectedConnector { constructor() { super("bitkeep.unisat"); this.metadata = { id: "bitget", name: "Bitget Wallet", icon: bitget_default, downloadUrl: "https://web3.bitget.com/en/wallet-download" }; } }; // src/icons/bybit.png var bybit_default = ""; // src/connector/bybit.ts var BybitConnector = class extends InjectedConnector { constructor() { super("bybitWallet.bitcoin"); this.metadata = { id: "bybit", name: "Bybit Wallet", icon: bybit_default, downloadUrl: "https://www.bybit.com/download/" }; } }; // src/icons/okx.svg var okx_default = ""; // src/connector/okx.ts var OKXConnector = class extends InjectedConnector { constructor() { super("okxwallet.bitcoin"); this.metadata = { id: "okx", name: "OKX Wallet", icon: okx_default, downloadUrl: "https://www.okx.com/download" }; } }; // src/icons/tokenpocket.png var tokenpocket_default = ""; // src/connector/tokenPocket.ts var TokenPocketConnector = class extends InjectedConnector { constructor() { super("tokenpocket.bitcoin"); this.metadata = { id: "tokenpocket", name: "TokenPocket", icon: tokenpocket_default, downloadUrl: "https://www.tokenpocket.pro/en/download/app" }; } }; // src/icons/unisat.svg var unisat_default = "