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
JavaScript
"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 = "