UNPKG

@particle-network/wallet

Version:

A wallet component that can be easily injected into the developer's website to provide rich custom styles.

783 lines (771 loc) 66.4 kB
"use client"; 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 __privateSet = (obj, member, value, setter) => { __accessCheck(obj, member, "write to private field"); setter ? setter.call(obj, value) : member.set(obj, value); return value; }; // src/walletPlugin.ts import debounce from "lodash/debounce.js"; // src/utils.ts var isServer = () => { return typeof window === "undefined"; }; function isNullish(x) { return x === void 0 || x === null; } // src/config.ts var _customWalletUrl, _devEnv, _stagingEnv, _productionEnv; var GlobalConfig = class { constructor() { __privateAdd(this, _customWalletUrl, void 0); __privateAdd(this, _devEnv, { walletUrl: "https://wallet-debug.particle.network" }); __privateAdd(this, _stagingEnv, { walletUrl: "https://wallet-staging.particle.network" }); __privateAdd(this, _productionEnv, { walletUrl: "https://wallet.particle.network" }); } get version() { return "web_2.1.1"; } get env() { let currentEnv = __privateGet(this, _productionEnv); if (!isServer() && window.__PARTICLE_ENVIRONMENT__ === "development") { currentEnv = __privateGet(this, _devEnv); } else if (!isServer() && window.__PARTICLE_ENVIRONMENT__ === "staging") { currentEnv = __privateGet(this, _stagingEnv); } else { currentEnv = __privateGet(this, _productionEnv); } if (__privateGet(this, _customWalletUrl)) { currentEnv.walletUrl = __privateGet(this, _customWalletUrl); } return currentEnv; } updateWalletUrl(url) { if (url) { __privateSet(this, _customWalletUrl, url); } } }; _customWalletUrl = new WeakMap(); _devEnv = new WeakMap(); _stagingEnv = new WeakMap(); _productionEnv = new WeakMap(); var globalConfig = new GlobalConfig(); var config_default = globalConfig; // src/html.ts var html = ` <button class="particle-pwe-btn"> <img class="particle-pwe-img particle-pwe-wallet-icon" src="" alt="" /> <img class="particle-pwe-img particle-pwe-down-arrow particle-pwe-down-arrow-hide" src="" alt="" /> </button> <div class="particle-pwe-iframe-content"> </div> `; var walletEntryRender = () => { const className = "particle-wallet-entry-container"; const el = document.querySelector("." + className); el && el.remove(); const EL = document.createElement("div"); EL.classList.add(className); EL.innerHTML = html; document.body.appendChild(EL); }; var html_default = walletEntryRender; // src/iconsBase64.ts var walletIconDarkBase64 = ""; var downArrowDarkBase64 = ""; // src/messageHandler.ts import sha256 from "crypto-js/sha256.js"; import stringify from "fast-json-stable-stringify"; import { v4 as uuid } from "uuid"; // src/types.ts var IframeWalletMessageType = "particle-auth-core-iframe-wallet-message"; var walletIframeId = "particle-auth-core-iframe-wallet"; var walletIframeMask = "particle-auth-core-iframe-wallet-mask"; var EntryPosition = /* @__PURE__ */ ((EntryPosition2) => { EntryPosition2["BR"] = "bottom-right"; EntryPosition2["BL"] = "bottom-left"; EntryPosition2["TR"] = "top-right"; EntryPosition2["TL"] = "top-left"; EntryPosition2["MC"] = "middle-center"; return EntryPosition2; })(EntryPosition || {}); // src/messageHandler.ts var walletPluginIds = [walletIframeId]; function checkMessage(message) { const { id, messageType, data } = message; const { nonce, date, hash, iframeId } = JSON.parse(Buffer.from(id, "base64").toString()); const hashBody = stringify({ state: { nonce, date, iframeId }, messageType, data }); const hashValue = sha256(hashBody).toString(); if (hashValue === hash) { return { state: { nonce, date, hash, iframeId }, messageType, data }; } } function buildMessage(data, messageType, nonce) { const state = { nonce: nonce || uuid(), date: Date.now() }; const hashValue = sha256( stringify({ state, messageType, data }) ).toString().toLowerCase(); const id = Buffer.from( JSON.stringify({ ...state, hash: hashValue }) ).toString("base64"); return { id, messageType, data }; } var handleEthereumRpc = async (data, provider) => { if (!provider) { throw { code: 4200, message: "Wallet plugin not support EVM chains." }; } const result = await provider.request(data); return result; }; var handleSolanaRpc = async (data, solana) => { var _a, _b; if (!solana) { throw { code: 4200, message: "Wallet plugin not support Solana chains." }; } if (data.method === "solana_requestAccounts") { const publicAddress = solana.selectedAddress || ((_a = solana.publicKey) == null ? void 0 : _a.toBase58()); if (!publicAddress) { await solana.connect(); } return solana.selectedAddress || ((_b = solana.publicKey) == null ? void 0 : _b.toBase58()); } else if (data.method === "solana_chainId") { return solana.chainId; } else if (data.method === "solana_signTransaction") { const { VersionedTransaction } = await import("@solana/web3.js"); const result = await solana.signTransaction( VersionedTransaction.deserialize(Buffer.from(data.params[0], "base64")) ); return Buffer.from(result.serialize()).toString("base64"); } else if (data.method === "solana_signAllTransactions") { const { VersionedTransaction } = await import("@solana/web3.js"); const txs = data.params[0].map( (tx) => VersionedTransaction.deserialize(Buffer.from(tx, "base64")) ); const result = await solana.signAllTransactions(txs); return result.map((tx) => Buffer.from(tx.serialize()).toString("base64")); } else if (data.method === "solana_signAndSendTransaction") { const { VersionedTransaction } = await import("@solana/web3.js"); const { signature } = await solana.signAndSendTransaction( VersionedTransaction.deserialize(Buffer.from(data.params[0], "base64")) ); return signature; } else if (data.method === "solana_signMessage") { const result = await solana.signMessage(Buffer.from(data.params[0], "base64")); return Buffer.from(result.signature).toString("base64"); } else if (data.method === "solana_switchChain") { if (!solana.switchChain) { throw { code: 4200, message: "Solana Wallet not support switch chain." }; } await solana.switchChain(data.params[0]); } }; var handleRpc = async (event, walletCore) => { var _a, _b, _c; const messageBody = checkMessage((_a = event.data) == null ? void 0 : _a.message); if (messageBody) { if (!walletPluginIds.includes(messageBody.state.iframeId)) { walletPluginIds.push(messageBody.state.iframeId); } let message; try { let result; if (messageBody.messageType === "ethereum-rpc" /* EthereumRpc */) { result = await handleEthereumRpc(messageBody.data, walletCore.ethereum); } else { result = await handleSolanaRpc(messageBody.data, walletCore.solana); } message = buildMessage({ result }, messageBody.messageType, messageBody.state.nonce); } catch (error) { message = buildMessage( { error: { message: error.message || error.stack || error.details || error, code: error.code } }, messageBody.messageType, messageBody.state.nonce ); } (_c = (_b = window.document.getElementById(messageBody.state.iframeId)) == null ? void 0 : _b.contentWindow) == null ? void 0 : _c.postMessage( { type: IframeWalletMessageType, message }, "*" ); } else { } }; var handleCustomEvent = async (event, customEventHandler) => { var _a, _b, _c; const messageBody = checkMessage((_a = event.data) == null ? void 0 : _a.message); if (messageBody) { if (!walletPluginIds.includes(messageBody.state.iframeId)) { walletPluginIds.push(messageBody.state.iframeId); } let message; try { if (!customEventHandler) { message = buildMessage( { error: { code: 4200, message: "Wallet plugin not support custom event." } }, messageBody.messageType, messageBody.state.nonce ); } else { const result = await customEventHandler(messageBody.messageType, messageBody.data); message = buildMessage({ result }, messageBody.messageType, messageBody.state.nonce); } } catch (error) { message = buildMessage( { error: { message: error.message || error.stack || error.toString(), code: error.code } }, messageBody.messageType, messageBody.state.nonce ); } (_c = (_b = window.document.getElementById(messageBody.state.iframeId)) == null ? void 0 : _b.contentWindow) == null ? void 0 : _c.postMessage( { type: IframeWalletMessageType, message }, "*" ); } else { } }; var sendEthereumEvent = (event, args) => { var _a, _b; const message = buildMessage({ name: event, args }, "ethereum-event" /* EthereumEvent */); for (const iframeId of walletPluginIds) { (_b = (_a = window.document.getElementById(iframeId)) == null ? void 0 : _a.contentWindow) == null ? void 0 : _b.postMessage( { type: IframeWalletMessageType, message }, "*" ); } }; // src/style.ts var style = ` .particle-wallet-entry-container .particle-pwe-btn { background: none; border: none; cursor: pointer; height: 60px; margin: 0; outline: none; padding: 0; position: fixed; width: 60px; border-radius: 60px; z-index: 1000; opacity: 0; } .particle-wallet-entry-container .particle-pwe-btn[data-position]{ opacity: 1; } .particle-wallet-entry-container .particle-pwe-btn[data-position=middle-center] { opacity: 0 !important; } .particle-wallet-entry-container .particle-pwe-btn:not(.is-dragging) { -webkit-transition: all 0.2s; transition: all 0.2s; } .particle-wallet-entry-container .particle-pwe-btn > img { height: 100%; width: 100%; -webkit-box-shadow: 2px 2px 10px 3px rgba(0, 0, 0, 0.1); box-shadow: 2px 2px 10px 3px rgba(0, 0, 0, 0.1); border-radius: 60px; } .particle-wallet-entry-container .particle-pwe-btn .particle-pwe-wallet-icon { display: block; } .particle-wallet-entry-container .particle-pwe-btn .particle-pwe-wallet-icon:not(.particle-pwe-wallet-icon-hide) { -webkit-animation: particle-pwe-wallet-icon-show 0.3s ease-in-out; animation: particle-pwe-wallet-icon-show 0.3s ease-in-out; } @-webkit-keyframes particle-pwe-wallet-icon-show { 0% { -webkit-transform: scale(0.8); transform: scale(0.8); } 100% { -webkit-transform: scale(1); transform: scale(1); } } @keyframes particle-pwe-wallet-icon-show { 0% { -webkit-transform: scale(0.6); transform: scale(0.6); } 100% { -webkit-transform: scale(1); transform: scale(1); } } .particle-wallet-entry-container .particle-pwe-btn .particle-pwe-wallet-icon.particle-pwe-wallet-icon-hide { display: none; } .particle-wallet-entry-container .particle-pwe-btn .particle-pwe-down-arrow { display: block; } .particle-wallet-entry-container .particle-pwe-btn .particle-pwe-down-arrow:not(.particle-pwe-down-arrow-hide) { -webkit-animation: particle-pwe-down-arrow-show 0.3s ease-in-out; animation: particle-pwe-down-arrow-show 0.3s ease-in-out; } @-webkit-keyframes particle-pwe-down-arrow-show { 0% { -webkit-transform: scale(0.6); transform: scale(0.6); } 100% { -webkit-transform: scale(1); transform: scale(1); } } @keyframes particle-pwe-down-arrow-show { 0% { -webkit-transform: scale(0.6); transform: scale(0.6); } 100% { -webkit-transform: scale(1); transform: scale(1); } } .particle-wallet-entry-container .particle-pwe-btn .particle-pwe-down-arrow.particle-pwe-down-arrow-hide { display: none; } .particle-wallet-entry-container .particle-pwe-iframe-content { background-color: #fff; border: none; border-radius: 10px; -webkit-box-shadow: -1px 3px 11px 2px #00000073; box-shadow: -1px 3px 11px 2px #00000073; display: none; height: 650px; overflow: hidden; position: fixed; width: 400px; z-index: 10000; } .particle-wallet-entry-container .particle-pwe-iframe-content.particle-pwe-full-screen-iframe-content { top: 0 !important; left: 0 !important; width: 100% !important; height: 100% !important; border-radius: 0 !important; } @media screen and (max-height: 660px) { .particle-wallet-entry-container .particle-pwe-iframe-content { height: 600px; width: 360px; } } .particle-wallet-entry-container .particle-pwe-iframe-content.particle-pwe-iframe-content-show { display: block; } .particle-pwe-iframe-center-content{ box-shadow: -1px 3px 11px 2px #0000002b !important; } .particle-pwe-iframe-content-dark.particle-pwe-iframe-center-content{ box-shadow: -1px 3px 11px 2px #dddddd0d !important; } .particle-pwe-iframe-content-dark{ background-color: #000 !important; } .particle-pwe-iframe-content-light{ background-color: #fff !important; } .particle-wallet-entry-container .particle-pwe-iframe-content iframe { border: none; height: 100%; width: 100%; } `; var renderStyle = () => { const className = "particle-wallet-entry-style"; const el = document.querySelector("." + className); el && el.remove(); const styleEl = document.createElement("style"); styleEl.classList.add(className); styleEl.innerHTML = style; document.head.appendChild(styleEl); }; var style_default = renderStyle; // src/walletUrl.ts var openTimestamp = new Date().getTime(); function walletUrl({ projectConfig, walletOptions, options, customEventHandler }) { const { pathName = "/", query = {} } = options || {}; let url = `${config_default.env.walletUrl}${pathName}?iframeid=${walletIframeId}&${JSON.stringify(query).replace(/[{}"]/g, "").split(",").filter((value) => !!(value == null ? void 0 : value.trim())).map((keyValue) => { const [key, value] = keyValue.split(":"); return `${encodeURIComponent(key)}=${encodeURIComponent(value.trim())}`; }).join("&")}`; const { projectId, clientKey, appId } = projectConfig; const { erc4337, language, themeType, customStyle } = walletOptions; const topMenuType = (options == null ? void 0 : options.topMenuType) || (walletOptions == null ? void 0 : walletOptions.topMenuType); url += `&projectConfig=${encodeURIComponent( Buffer.from(JSON.stringify({ projectId, clientKey, appId })).toString("base64") )}`; if (themeType) { url += "&theme=" + themeType; } if (language) { url += "&language=" + language; } if (erc4337) { url += `&erc4337=${encodeURIComponent(JSON.stringify(erc4337))}`; } if (topMenuType) { url += `&topMenuType=${topMenuType}`; } if (customEventHandler) { if (customEventHandler.type) { url += `&supportCustomEvent=${customEventHandler.type}`; } else { url += "&supportCustomEvent=true"; } } url += `&openTimestamp=${openTimestamp}`; if (customStyle) { let supportChains = customStyle.supportChains; if (supportChains) { supportChains = supportChains.map((chain) => { return { name: chain.name, id: chain.id, chainType: chain.chainType }; }); } url += `&customStyle=${encodeURIComponent( Buffer.from(JSON.stringify({ ...customStyle, supportChains })).toString("base64") )}`; } return url; } // src/walletPlugin.ts var fullScreenClass = "particle-pwe-full-screen-iframe-content"; var isListen = false; var timer = null; var draggie; var _onMessage, _onEthereumAccountsChanged, _onEthereumChainChanged; var _WalletEntryPlugin = class { constructor() { this.centerWallet = false; this.walletCreated = false; __privateAdd(this, _onMessage, (event) => { var _a, _b, _c; if (((_a = event.data) == null ? void 0 : _a.type) === IframeWalletMessageType && this._walletCore) { const messageType = (_c = (_b = event.data) == null ? void 0 : _b.message) == null ? void 0 : _c.messageType; if (messageType === "ethereum-rpc" /* EthereumRpc */ || messageType === "solana-rpc" /* SolanaRpc */) { handleRpc(event, this._walletCore); } else { handleCustomEvent(event, this._walletCore.customEventHandler); } } }); __privateAdd(this, _onEthereumAccountsChanged, (args) => { sendEthereumEvent("accountsChanged", args); }); __privateAdd(this, _onEthereumChainChanged, (args) => { sendEthereumEvent("chainChanged", args); }); if (typeof window !== "undefined") { window.addEventListener("message", (event) => { var _a, _b, _c; if (((_a = event == null ? void 0 : event.data) == null ? void 0 : _a.name) === "particle-network-wallet") { const type = (_c = (_b = event == null ? void 0 : event.data) == null ? void 0 : _b.data) == null ? void 0 : _c.type; if (type === "logout") { this.walletEntryDestroy(); } } }); window.walletEntryPlugin = this; } } get walletCore() { return this._walletCore; } get walletOptions() { return this._walletOptions; } init(projectConfig, options = {}) { var _a, _b; this.projectConfig = projectConfig; this._walletOptions = { entryPosition: "bottom-right" /* BR */, themeType: "light", ...options, widgetIntegration: (_a = options.widgetIntegration) != null ? _a : "modal", visible: (_b = options.visible) != null ? _b : true }; if (options.walletUrl) { config_default.updateWalletUrl(options.walletUrl); } const storedPosition = this.getButtonStorageData().position; if (storedPosition && storedPosition !== this._walletOptions.entryPosition) { localStorage.removeItem(_WalletEntryPlugin.WALLET_BTN_POSITION); localStorage.removeItem(_WalletEntryPlugin.WALLET_BTN_POSITION + "_window"); this.setInitialButtonPosition(); } this.centerWallet = Boolean(this._walletOptions.entryPosition === "middle-center" /* MC */ || !this._walletOptions.visible); if (this.centerWallet) { this._walletOptions.topMenuType = "close"; } this.destroy(); if (typeof window !== "undefined") { window.removeEventListener("message", __privateGet(this, _onMessage)); window.addEventListener("message", __privateGet(this, _onMessage)); } } setInitialButtonPosition() { const position = this._walletOptions.entryPosition; const { innerWidth, innerHeight } = window; let x, y, direction; switch (position) { case "bottom-right" /* BR */: x = innerWidth; y = innerHeight; direction = "right"; break; case "bottom-left" /* BL */: x = 0; y = innerHeight; direction = "left"; break; case "top-right" /* TR */: x = innerWidth; y = 0; direction = "right"; break; case "top-left" /* TL */: x = 0; y = 0; direction = "left"; break; default: x = innerWidth; y = innerHeight; direction = "right"; } this.setButtonStorageData({ x, y, direction }); } overrideWalletOption(options) { if (!this.projectConfig) { throw new Error("Please init wallet entry plugin first!"); } this._walletOptions = { ...this._walletOptions, ...options }; if (this.walletCore && this.walletCreated) { this.walletEntryCreate(); } else { this.walletEntryDestroy(); } } setWalletCore(walletCore) { var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l; if (!this.projectConfig) { throw new Error("Please init wallet entry plugin first!"); } if (!walletCore.ethereum && !walletCore.solana) { throw new Error("Please provide ethereum or solana wallet!"); } if (this._walletCore) { (_b = (_a = this._walletCore.ethereum) == null ? void 0 : _a.removeListener) == null ? void 0 : _b.call(_a, "accountsChanged", __privateGet(this, _onEthereumAccountsChanged)); (_d = (_c = this._walletCore.ethereum) == null ? void 0 : _c.removeListener) == null ? void 0 : _d.call(_c, "chainChanged", __privateGet(this, _onEthereumChainChanged)); } this._walletCore = walletCore; if (this._walletCore.ethereum) { (_f = (_e = this._walletCore.ethereum) == null ? void 0 : _e.removeListener) == null ? void 0 : _f.call(_e, "accountsChanged", __privateGet(this, _onEthereumAccountsChanged)); (_h = (_g = this._walletCore.ethereum) == null ? void 0 : _g.on) == null ? void 0 : _h.call(_g, "accountsChanged", __privateGet(this, _onEthereumAccountsChanged)); (_j = (_i = this._walletCore.ethereum) == null ? void 0 : _i.removeListener) == null ? void 0 : _j.call(_i, "chainChanged", __privateGet(this, _onEthereumChainChanged)); (_l = (_k = this._walletCore.ethereum) == null ? void 0 : _k.on) == null ? void 0 : _l.call(_k, "chainChanged", __privateGet(this, _onEthereumChainChanged)); } } walletEntryCreate() { var _a; this.destroy(); if (((_a = this._walletOptions) == null ? void 0 : _a.widgetIntegration) !== "embedded") { this.walletEntryRander(); if (this._walletOptions.preload) { const iframe = this.getWalletIFrameImpl(); const iframeContent = document.querySelector(".particle-pwe-iframe-content"); if (!(iframeContent == null ? void 0 : iframeContent.childElementCount)) { iframe.style.width = "100%"; iframe.style.height = "100%"; iframeContent == null ? void 0 : iframeContent.appendChild(iframe); } } } this.walletCreated = true; } walletEntryDestroy() { this.destroy(); } destroy() { var _a, _b, _c, _d; if (isServer()) return; (_a = draggie == null ? void 0 : draggie.destroy) == null ? void 0 : _a.call(draggie); window.removeEventListener("resize", (_b = this == null ? void 0 : this.resize) == null ? void 0 : _b.call(this), false); (_d = (_c = document.querySelector(".particle-wallet-entry-container")) == null ? void 0 : _c.remove) == null ? void 0 : _d.call(_c); const walletBtn = document.querySelector(".particle-pwe-btn"); walletBtn && walletBtn.removeAttribute("data-position"); const iframe = document.getElementById(walletIframeId); if (iframe) { iframe.remove(); } this.walletCreated = false; } getWalletUrl(options) { var _a; if (!this.projectConfig) { throw new Error("Please init wallet entry plugin first!"); } const url = walletUrl({ options, projectConfig: this.projectConfig, walletOptions: this._walletOptions, customEventHandler: (_a = this._walletCore) == null ? void 0 : _a.customEventHandler }); return url; } getWalletIFrame(options) { if (this._walletOptions.widgetIntegration !== "embedded") { throw new Error("Only `embedded` mode can control the iframe."); } return this.getWalletIFrameImpl(options); } getWalletIFrameImpl(options) { if (!this.projectConfig) { throw new Error("Please init wallet entry plugin first!"); } if (!this._walletCore) { throw new Error("Please call `setWalletCore` first!"); } let iframe = document.getElementById(walletIframeId); if (!iframe) { iframe = this.createIFrame(); } const url = this.getWalletUrl(options); if (url != iframe.getAttribute("src")) { iframe.src = url; } return iframe; } getWalletIFrameMask() { let mask = document.getElementById(walletIframeMask); if (!mask) { mask = document.createElement("div"); mask.setAttribute("id", walletIframeMask); mask.style.position = "fixed"; mask.style.top = "0"; mask.style.left = "0"; mask.style.width = "100%"; mask.style.height = "100%"; mask.style.zIndex = "1000"; mask.style.backgroundColor = "rgba(71, 88, 107, 0.24)"; mask.style.backdropFilter = "blur(6px)"; mask.style.display = "none"; document.body.appendChild(mask); } return mask; } createIFrame() { let iframe = document.getElementById(walletIframeId); if (!iframe) { iframe = document.createElement("iframe"); iframe.style.width = "100%"; iframe.style.height = "100%"; iframe.style.border = "none"; iframe.allow = "camera"; iframe.id = walletIframeId; iframe.src = this.getWalletUrl(); } return iframe; } openWallet(params) { var _a, _b, _c, _d, _e, _f, _g; if (((_a = this._walletOptions) == null ? void 0 : _a.widgetIntegration) === "embedded") { throw new Error("embedded mode not support `openWallet`"); } const walletIcon = document.querySelector(".particle-pwe-wallet-icon"); const downArrow = document.querySelector(".particle-pwe-down-arrow"); const iframeContent = document.querySelector(".particle-pwe-iframe-content"); if ((params == null ? void 0 : params.windowSize) === "large" || window.screen.width < 600) { iframeContent == null ? void 0 : iframeContent.classList.add(fullScreenClass); } const modalBorderRadius = (_f = (_e = (_d = (_b = this._walletOptions) == null ? void 0 : _b.customStyle) == null ? void 0 : _d[((_c = this._walletOptions) == null ? void 0 : _c.themeType) || "light"]) == null ? void 0 : _e.cardBorderRadius) != null ? _f : 18; if (!isNullish(modalBorderRadius)) { iframeContent.style.borderRadius = `${modalBorderRadius}