UNPKG

@particle-network/wallet

Version:

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

4 lines 93.5 kB
{ "version": 3, "sources": ["../../src/index.ts", "../../src/walletPlugin.ts", "../../src/utils.ts", "../../src/config.ts", "../../src/html.ts", "../../src/iconsBase64.ts", "../../src/messageHandler.ts", "../../src/types.ts", "../../src/style.ts", "../../src/walletUrl.ts"], "sourcesContent": ["import { WalletEntryPlugin } from './walletPlugin';\n\nexport type {\n CustomEventHandler,\n CustomMessageType,\n EthereumProvider,\n MessageType,\n ProjectConfig,\n SolanaWallet,\n WalletConfig,\n WalletCore,\n WalletCustomStyle,\n WalletOption,\n} from './types';\n\nexport { EntryPosition } from './types';\n\nconst walletEntryPlugin = new WalletEntryPlugin();\n\nexport { walletEntryPlugin, type WalletEntryPlugin };\n", "import debounce from 'lodash/debounce.js';\nimport globalConfig from './config';\nimport htmlRender from './html';\nimport { downArrowDarkBase64, walletIconDarkBase64 } from './iconsBase64';\nimport { handleCustomEvent, handleRpc, sendEthereumEvent } from './messageHandler';\nimport styleRender from './style';\nimport {\n EntryPosition,\n IframeWalletMessageType,\n MessageType,\n walletIframeId,\n walletIframeMask,\n type ProjectConfig,\n type WalletConfig,\n type WalletCore,\n type WalletOption,\n} from './types';\nimport { isNullish, isServer } from './utils';\nimport { walletUrl } from './walletUrl';\n\nconst fullScreenClass = 'particle-pwe-full-screen-iframe-content';\n\nlet isListen = false;\n\nlet timer: any = null;\n\nlet draggie: any;\n\nexport class WalletEntryPlugin {\n static WALLET_BTN_POSITION = 'walletBtnPosition';\n\n private _walletOptions!: WalletOption;\n\n private projectConfig: ProjectConfig | undefined;\n\n private _walletCore: WalletCore | undefined;\n\n /**\n * wallet is center\n */\n private centerWallet = false;\n\n walletCreated = false;\n\n get walletCore() {\n return this._walletCore;\n }\n\n get walletOptions() {\n return this._walletOptions;\n }\n\n constructor() {\n if (typeof window !== 'undefined') {\n window.addEventListener('message', (event) => {\n if (event?.data?.name === 'particle-network-wallet') {\n console.log('handleWalletEvent', event);\n const type = event?.data?.data?.type;\n if (type === 'logout') {\n this.walletEntryDestroy();\n }\n }\n });\n\n // @ts-ignore\n window.walletEntryPlugin = this;\n }\n }\n\n init(projectConfig: ProjectConfig, options: WalletOption = {}) {\n this.projectConfig = projectConfig;\n this._walletOptions = {\n entryPosition: EntryPosition.BR,\n themeType: 'light',\n ...options,\n widgetIntegration: options.widgetIntegration ?? 'modal',\n visible: options.visible ?? true,\n };\n\n if (options.walletUrl) {\n globalConfig.updateWalletUrl(options.walletUrl);\n }\n\n const storedPosition = this.getButtonStorageData().position;\n if (storedPosition && storedPosition !== this._walletOptions.entryPosition) {\n localStorage.removeItem(WalletEntryPlugin.WALLET_BTN_POSITION);\n localStorage.removeItem(WalletEntryPlugin.WALLET_BTN_POSITION + '_window');\n this.setInitialButtonPosition();\n }\n\n this.centerWallet = Boolean(this._walletOptions.entryPosition === EntryPosition.MC || !this._walletOptions.visible);\n\n if (this.centerWallet) {\n this._walletOptions.topMenuType = 'close';\n }\n\n this.destroy();\n if (typeof window !== 'undefined') {\n window.removeEventListener('message', this.#onMessage);\n window.addEventListener('message', this.#onMessage);\n }\n }\n\n private setInitialButtonPosition() {\n const position = this._walletOptions.entryPosition;\n const { innerWidth, innerHeight } = window;\n let x, y, direction;\n\n switch (position) {\n case EntryPosition.BR:\n x = innerWidth;\n y = innerHeight;\n direction = 'right';\n break;\n case EntryPosition.BL:\n x = 0;\n y = innerHeight;\n direction = 'left';\n break;\n case EntryPosition.TR:\n x = innerWidth;\n y = 0;\n direction = 'right';\n break;\n case EntryPosition.TL:\n x = 0;\n y = 0;\n direction = 'left';\n break;\n default:\n x = innerWidth;\n y = innerHeight;\n direction = 'right';\n }\n\n this.setButtonStorageData({ x, y, direction });\n }\n\n overrideWalletOption(options: Partial<WalletOption>) {\n if (!this.projectConfig) {\n throw new Error('Please init wallet entry plugin first!');\n }\n this._walletOptions = {\n ...this._walletOptions,\n ...options,\n };\n\n if (this.walletCore && this.walletCreated) {\n this.walletEntryCreate();\n } else {\n this.walletEntryDestroy();\n }\n }\n\n setWalletCore(walletCore: WalletCore) {\n if (!this.projectConfig) {\n throw new Error('Please init wallet entry plugin first!');\n }\n if (!walletCore.ethereum && !walletCore.solana) {\n throw new Error('Please provide ethereum or solana wallet!');\n }\n if (this._walletCore) {\n this._walletCore.ethereum?.removeListener?.('accountsChanged', this.#onEthereumAccountsChanged);\n this._walletCore.ethereum?.removeListener?.('chainChanged', this.#onEthereumChainChanged);\n }\n this._walletCore = walletCore;\n if (this._walletCore.ethereum) {\n this._walletCore.ethereum?.removeListener?.('accountsChanged', this.#onEthereumAccountsChanged);\n this._walletCore.ethereum?.on?.('accountsChanged', this.#onEthereumAccountsChanged);\n\n this._walletCore.ethereum?.removeListener?.('chainChanged', this.#onEthereumChainChanged);\n this._walletCore.ethereum?.on?.('chainChanged', this.#onEthereumChainChanged);\n }\n }\n\n walletEntryCreate() {\n this.destroy();\n if (this._walletOptions?.widgetIntegration !== 'embedded') {\n this.walletEntryRander();\n\n if (this._walletOptions.preload) {\n const iframe: HTMLIFrameElement = this.getWalletIFrameImpl();\n const iframeContent = document.querySelector('.particle-pwe-iframe-content') as HTMLDivElement;\n if (!iframeContent?.childElementCount) {\n iframe.style.width = '100%';\n iframe.style.height = '100%';\n iframeContent?.appendChild(iframe);\n }\n }\n }\n this.walletCreated = true;\n }\n\n walletEntryDestroy() {\n this.destroy();\n }\n\n destroy() {\n if (isServer()) return;\n draggie?.destroy?.();\n window.removeEventListener('resize', this?.resize?.(), false);\n document.querySelector('.particle-wallet-entry-container')?.remove?.();\n\n const walletBtn = document.querySelector('.particle-pwe-btn') as HTMLButtonElement;\n walletBtn && walletBtn.removeAttribute('data-position');\n\n const iframe = document.getElementById(walletIframeId);\n if (iframe) {\n iframe.remove();\n }\n this.walletCreated = false;\n }\n\n private getWalletUrl(options?: WalletConfig): string {\n if (!this.projectConfig) {\n throw new Error('Please init wallet entry plugin first!');\n }\n const url = walletUrl({\n options,\n projectConfig: this.projectConfig,\n walletOptions: this._walletOptions,\n customEventHandler: this._walletCore?.customEventHandler,\n });\n return url;\n }\n\n public getWalletIFrame(options?: WalletConfig): HTMLIFrameElement {\n if (this._walletOptions.widgetIntegration !== 'embedded') {\n throw new Error('Only `embedded` mode can control the iframe.');\n }\n\n return this.getWalletIFrameImpl(options);\n }\n\n private getWalletIFrameImpl(options?: WalletConfig): HTMLIFrameElement {\n if (!this.projectConfig) {\n throw new Error('Please init wallet entry plugin first!');\n }\n if (!this._walletCore) {\n throw new Error('Please call `setWalletCore` first!');\n }\n\n let iframe: HTMLIFrameElement | null = document.getElementById(walletIframeId) as HTMLIFrameElement;\n if (!iframe) {\n iframe = this.createIFrame();\n }\n\n const url = this.getWalletUrl(options);\n if (url != iframe.getAttribute('src')) {\n iframe.src = url;\n }\n return iframe;\n }\n\n private getWalletIFrameMask() {\n let mask: HTMLDivElement | null = document.getElementById(walletIframeMask) as HTMLDivElement;\n if (!mask) {\n mask = document.createElement('div');\n mask.setAttribute('id', walletIframeMask);\n mask.style.position = 'fixed';\n mask.style.top = '0';\n mask.style.left = '0';\n mask.style.width = '100%';\n mask.style.height = '100%';\n mask.style.zIndex = '1000';\n mask.style.backgroundColor = 'rgba(71, 88, 107, 0.24)';\n mask.style.backdropFilter = 'blur(6px)';\n mask.style.display = 'none';\n document.body.appendChild(mask);\n }\n return mask;\n }\n\n private createIFrame() {\n let iframe: HTMLIFrameElement | null = document.getElementById(walletIframeId) as HTMLIFrameElement;\n if (!iframe) {\n iframe = document.createElement('iframe');\n iframe.style.width = '100%';\n iframe.style.height = '100%';\n iframe.style.border = 'none';\n iframe.allow = 'camera';\n iframe.id = walletIframeId;\n iframe.src = this.getWalletUrl();\n }\n\n return iframe;\n }\n\n /**\n * open modal wallet\n * @param params\n * @param params.windowSize large | small\n * @param params.pathName The path to the web wallet page, which is '/' by default.\n * @param params.query Go to the query parameter of the wallet page\n * @returns\n */\n public openWallet(\n params?: {\n windowSize?: 'large' | 'small';\n } & WalletConfig\n ) {\n console.log('open wallet', this._walletOptions, params);\n if (this._walletOptions?.widgetIntegration === 'embedded') {\n throw new Error('embedded mode not support `openWallet`');\n }\n\n const walletIcon = document.querySelector('.particle-pwe-wallet-icon') as HTMLDivElement;\n const downArrow = document.querySelector('.particle-pwe-down-arrow');\n const iframeContent = document.querySelector('.particle-pwe-iframe-content') as HTMLDivElement;\n\n if (params?.windowSize === 'large' || window.screen.width < 600) {\n iframeContent?.classList.add(fullScreenClass);\n }\n const modalBorderRadius =\n this._walletOptions?.customStyle?.[this._walletOptions?.themeType || 'light']?.cardBorderRadius ?? 18;\n if (!isNullish(modalBorderRadius)) {\n iframeContent.style.borderRadius = `${modalBorderRadius}px`;\n }\n iframeContent?.classList.add('particle-pwe-iframe-content-show');\n\n const uiMode = this._walletOptions?.themeType || 'light';\n\n if (uiMode == 'dark') {\n iframeContent?.classList.add('particle-pwe-iframe-content-dark');\n iframeContent?.classList.remove('particle-pwe-iframe-content-light');\n } else {\n iframeContent?.classList.add('particle-pwe-iframe-content-light');\n iframeContent?.classList.remove('particle-pwe-iframe-content-dark');\n }\n\n if (this.centerWallet) {\n iframeContent.classList.add('particle-pwe-iframe-center-content');\n }\n\n const iframe: HTMLIFrameElement = this.getWalletIFrameImpl(params);\n if (!iframeContent.childElementCount) {\n iframe.style.width = '100%';\n iframe.style.height = '100%';\n iframeContent?.appendChild(iframe);\n }\n\n walletIcon?.classList.add('particle-pwe-wallet-icon-hide');\n downArrow?.classList.remove('particle-pwe-down-arrow-hide');\n if (this.centerWallet) {\n const iframeMask = this.getWalletIFrameMask();\n iframeMask.style.display = 'block';\n }\n this.updateIframeContentPosition();\n }\n\n // ----- wallet event\n\n #onMessage = (event: MessageEvent) => {\n if (event.data?.type === IframeWalletMessageType && this._walletCore) {\n const messageType = event.data?.message?.messageType;\n if (messageType === MessageType.EthereumRpc || messageType === MessageType.SolanaRpc) {\n handleRpc(event, this._walletCore);\n } else {\n handleCustomEvent(event, this._walletCore.customEventHandler);\n }\n }\n };\n\n #onEthereumAccountsChanged = (args: unknown) => {\n sendEthereumEvent('accountsChanged', args);\n };\n\n #onEthereumChainChanged = (args: unknown) => {\n sendEthereumEvent('chainChanged', args);\n };\n\n // ---- render wallet icon\n\n private setWalletIcon() {\n const walletIconEl = document.querySelector('.particle-pwe-wallet-icon') as HTMLImageElement;\n const downArrowEl = document.querySelector('.particle-pwe-down-arrow') as HTMLImageElement;\n\n walletIconEl && walletIconEl.setAttribute('src', walletIconDarkBase64);\n downArrowEl && downArrowEl.setAttribute('src', downArrowDarkBase64);\n\n const walletBtn = document.querySelector('.particle-pwe-btn') as HTMLButtonElement;\n if (walletBtn && this._walletOptions.visible && this._walletOptions.entryPosition !== EntryPosition.MC) {\n walletBtn.style.opacity = '1';\n }\n }\n\n private updateIframeContentPosition() {\n const iframeContent = document.querySelector('.particle-pwe-iframe-content') as HTMLDivElement;\n if (!iframeContent || iframeContent?.style?.display === 'none') {\n return;\n }\n const walletBtn = document.querySelector('.particle-pwe-btn') as HTMLButtonElement;\n\n if (!this._walletOptions.visible) {\n walletBtn.style.display = 'none';\n }\n\n const walletBtnRect = walletBtn.getBoundingClientRect();\n const iframeContentRect = iframeContent.getBoundingClientRect();\n const windowHeight = window.innerHeight;\n const windowWidth = window.innerWidth;\n const iframeContentHeight = iframeContentRect.height;\n const iframeContentWidth = iframeContentRect.width;\n const walletBtnHeight = walletBtnRect.height;\n // const walletBtnWidth = walletBtnRect.width;\n const walletBtnTop = walletBtnRect.top;\n const walletBtnLeft = walletBtnRect.left;\n const walletBtnBottom = walletBtnRect.bottom;\n const walletBtnRight = walletBtnRect.right;\n\n if (this.centerWallet) {\n // Screen center\n iframeContent.style.top = `${(windowHeight - iframeContentHeight) / 2}px`;\n iframeContent.style.left = `${(windowWidth - iframeContentWidth) / 2}px`;\n } else if (\n walletBtnBottom + iframeContentHeight + 10 < windowHeight &&\n walletBtnRight + iframeContentWidth + 10 < windowWidth\n ) {\n iframeContent.style.top = walletBtnBottom + 10 + 'px';\n iframeContent.style.left = walletBtnLeft + 'px';\n } else if (\n walletBtnBottom + iframeContentHeight + 10 < windowHeight &&\n walletBtnLeft - iframeContentWidth - 10 > 0\n ) {\n iframeContent.style.top = walletBtnBottom + 10 + 'px';\n iframeContent.style.left = walletBtnRight - iframeContentWidth + 'px';\n } else if (walletBtnTop - iframeContentHeight - 10 > 0 && walletBtnRight + iframeContentWidth + 10 < windowWidth) {\n iframeContent.style.top = walletBtnTop - iframeContentHeight - 10 + 'px';\n iframeContent.style.left = walletBtnLeft + 'px';\n } else if (walletBtnTop - iframeContentHeight - 10 > 0 && walletBtnLeft - iframeContentWidth - 10 > 0) {\n iframeContent.style.top = walletBtnTop - iframeContentHeight - 10 + 'px';\n iframeContent.style.left = walletBtnRight - iframeContentWidth + 'px';\n } else if (walletBtnRight + iframeContentWidth + 10 < windowWidth) {\n const top = walletBtnTop + walletBtnHeight / 2 - iframeContentHeight / 2;\n iframeContent.style.top =\n top < 30\n ? 30 + 'px'\n : top > windowHeight - iframeContentHeight - 30\n ? windowHeight - iframeContentHeight - 30 + 'px'\n : top + 'px';\n iframeContent.style.left = walletBtnRight + 10 + 'px';\n } else if (walletBtnLeft - iframeContentWidth - 10 > 0) {\n const top = walletBtnTop + walletBtnHeight / 2 - iframeContentHeight / 2;\n iframeContent.style.top =\n top < 30\n ? 30 + 'px'\n : top > windowHeight - iframeContentHeight - 30\n ? windowHeight - iframeContentHeight - 30 + 'px'\n : top + 'px';\n iframeContent.style.left = walletBtnLeft - iframeContentWidth - 10 + 'px';\n } else {\n console.log('no space');\n }\n }\n\n private setButtonStorageData(data: any) {\n const { innerWidth, innerHeight } = window;\n const position = this._walletOptions.entryPosition || EntryPosition.BR;\n localStorage.setItem(WalletEntryPlugin.WALLET_BTN_POSITION, `${data.x},${data.y},${data.direction},${position}`);\n localStorage.setItem(WalletEntryPlugin.WALLET_BTN_POSITION + '_window', `${innerWidth},${innerHeight}`);\n }\n\n private getButtonStorageData() {\n const { innerWidth, innerHeight } = window;\n const [x, y, direction, position] =\n localStorage?.getItem(WalletEntryPlugin.WALLET_BTN_POSITION)?.split?.(',') || [];\n let [width, height]: any =\n localStorage.getItem(WalletEntryPlugin.WALLET_BTN_POSITION + '_window')?.split?.(',') || [];\n width = Number(width || 0);\n height = Number(height || 0);\n if (width && height && (width != innerWidth || height != innerHeight)) {\n localStorage.removeItem(WalletEntryPlugin.WALLET_BTN_POSITION);\n localStorage.removeItem(WalletEntryPlugin.WALLET_BTN_POSITION + '_window');\n return this.getInitialPositionData();\n }\n\n return {\n x: Number(x) || 0,\n y: Number(y) || 0,\n position: position || this._walletOptions?.entryPosition,\n direction: direction || this.getDirectionFromPosition(this._walletOptions?.entryPosition),\n };\n }\n\n private getInitialPositionData() {\n const position = this._walletOptions?.entryPosition || EntryPosition.BR;\n const { innerWidth, innerHeight } = window;\n let x, y;\n\n switch (position) {\n case EntryPosition.BR:\n x = innerWidth;\n y = innerHeight;\n break;\n case EntryPosition.BL:\n x = 0;\n y = innerHeight;\n break;\n case EntryPosition.TR:\n x = innerWidth;\n y = 0;\n break;\n case EntryPosition.TL:\n x = 0;\n y = 0;\n break;\n default:\n x = innerWidth;\n y = innerHeight;\n }\n\n return {\n x,\n y,\n position,\n direction: this.getDirectionFromPosition(position),\n };\n }\n\n private getDirectionFromPosition(position?: EntryPosition): 'left' | 'right' {\n return position === EntryPosition.BL || position === EntryPosition.TL ? 'left' : 'right';\n }\n\n private updateWalletBtnPosition(x: number, y: number, type = '') {\n const { width } = document.body.getBoundingClientRect();\n const { direction } = this.getButtonStorageData();\n const isRight = x > width / 2 || (type == 'windowResize' && direction === 'right');\n let left = isRight ? width - 60 : 0;\n let top = y;\n let newDirection = 'left';\n\n const safeDistance = window.screen.width > 600 ? 30 : 10;\n\n if (left < safeDistance) {\n left = safeDistance;\n newDirection = 'left';\n } else if (left > width - (60 + safeDistance)) {\n left = width - (60 + safeDistance);\n newDirection = 'right';\n }\n if (top < safeDistance) {\n top = safeDistance;\n } else if (top > window.innerHeight - (60 + safeDistance)) {\n top = window.innerHeight - (60 + safeDistance);\n }\n\n return {\n left,\n top,\n direction: newDirection,\n };\n }\n\n public closeWallet() {\n const iframeContent: HTMLDivElement | null = document.querySelector('.particle-pwe-iframe-content');\n iframeContent?.classList.remove('particle-pwe-iframe-content-show');\n iframeContent?.classList.remove(fullScreenClass);\n const walletIcon = document.querySelector('.particle-pwe-wallet-icon') as HTMLDivElement;\n const downArrow = document.querySelector('.particle-pwe-down-arrow');\n walletIcon?.classList.remove('particle-pwe-wallet-icon-hide');\n downArrow?.classList.add('particle-pwe-down-arrow-hide');\n const iframeMask = this.getWalletIFrameMask();\n iframeMask.style.display = 'none';\n }\n\n private resize() {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const that = this;\n return () => {\n if (isServer()) return;\n clearTimeout(timer);\n\n timer = setTimeout(() => {\n const walletBtn: HTMLDivElement | null = document.querySelector('.particle-pwe-btn');\n\n const { x: btnX, y: btnY, position } = this.getButtonStorageData();\n\n let x, y;\n if (btnX && btnY) {\n x = btnX;\n y = btnY;\n } else {\n switch (position) {\n case EntryPosition.BL:\n x = 0;\n y = window.innerHeight;\n break;\n case EntryPosition.TL:\n x = 0;\n y = 0;\n break;\n case EntryPosition.TR:\n x = window.innerWidth;\n y = 0;\n break;\n case EntryPosition.BR:\n default:\n x = window.innerWidth;\n y = window.innerHeight;\n break;\n }\n }\n\n const { left, top, direction } = that.updateWalletBtnPosition(x, y, 'windowResize');\n if (walletBtn) {\n walletBtn.style.left = left + 'px';\n walletBtn.style.top = top + 'px';\n }\n\n this.setButtonStorageData({\n x: left,\n y: top,\n direction,\n });\n\n that.updateIframeContentPosition();\n\n (async () => {\n let count = 0;\n do {\n count++;\n await this.sleep(100);\n this.updateIframeContentPosition();\n } while (count < 15);\n })();\n }, 10);\n };\n }\n\n private async sleep(ms: number) {\n return new Promise((resolve) => setTimeout(resolve, ms));\n }\n\n private async walletEntryRander() {\n if (isServer()) return;\n styleRender();\n htmlRender();\n this.setWalletIcon();\n\n const walletBtn = document.querySelector('.particle-pwe-btn') as HTMLButtonElement;\n // const maxZIndex = Math.max.apply(\n // null,\n // Array.from(document.querySelectorAll('*')).map((item) => {\n // return Number(window.getComputedStyle(item).zIndex) || 0;\n // })\n // );\n\n // walletBtn.style.zIndex = maxZIndex + 1;\n\n const { x: btnX, y: btnY } = this.getButtonStorageData();\n\n if (btnX && btnY) {\n if ((Number(btnX) || 0) < window.innerWidth && (Number(btnY) || 0) < window.innerHeight) {\n const { left, top } = this.updateWalletBtnPosition(Number(btnX) || 0, Number(btnY) || 0);\n walletBtn.style.left = left + 'px';\n walletBtn.style.top = top + 'px';\n this.updateIframeContentPosition();\n } else {\n const { left, top } = this.updateWalletBtnPosition(window.innerWidth, window.innerHeight);\n walletBtn.style.left = left + 'px';\n walletBtn.style.top = top + 'px';\n this.updateIframeContentPosition();\n }\n } else {\n // default wallet btn position\n const position = this._walletOptions.entryPosition || EntryPosition.BL;\n let top, left;\n if (position === EntryPosition.BR) {\n top = window.innerHeight;\n left = window.innerWidth;\n } else if (position === EntryPosition.TR) {\n top = 0;\n left = window.innerWidth;\n } else if (position === EntryPosition.TL) {\n top = 0;\n left = 0;\n } else {\n top = window.innerHeight;\n left = 0;\n }\n\n const positionData = this.updateWalletBtnPosition(left, top);\n walletBtn.style.left = positionData.left + 'px';\n walletBtn.style.top = positionData.top + 'px';\n walletBtn.setAttribute('data-position', position?.toLowerCase());\n this.updateIframeContentPosition();\n }\n\n let isDraggie = false;\n\n const Module = await import('draggabilly');\n const Draggabilly = Module.default ? Module.default : Module;\n //@ts-ignore\n draggie = new Draggabilly(walletBtn);\n\n draggie.on('dragStart', () => {\n isDraggie = true;\n walletBtn.style.cursor = 'move';\n });\n\n draggie.on('dragMove', () => {\n this.updateIframeContentPosition();\n });\n\n draggie.on('dragEnd', (event: any, pointer: any) => {\n event.stopPropagation();\n walletBtn.style.cursor = 'pointer';\n const { clientX: x, clientY: y } = pointer;\n const { left, top, direction } = this.updateWalletBtnPosition(x, y - 35);\n walletBtn.style.left = left + 'px';\n walletBtn.style.top = top + 'px';\n this.updateIframeContentPosition();\n\n (async () => {\n let count = 0;\n do {\n count++;\n await this.sleep(20);\n this.updateIframeContentPosition();\n } while (count < 70);\n })();\n\n this.setButtonStorageData({\n x: left,\n y: top,\n direction,\n });\n\n setTimeout(() => {\n isDraggie = false;\n }, 50);\n });\n\n const controlIframe = debounce(() => {\n if (!isDraggie) {\n const iframeContent = document.querySelector('.particle-pwe-iframe-content') as HTMLDivElement;\n if (iframeContent?.classList?.contains?.('particle-pwe-iframe-content-show')) {\n this.closeWallet();\n } else {\n this.openWallet();\n }\n }\n }, 30);\n\n walletBtn.addEventListener('touchend', (event: any) => {\n setTimeout(() => {\n controlIframe();\n });\n });\n\n walletBtn.addEventListener('click', (event: any) => {\n event?.stopPropagation?.();\n setTimeout(() => {\n controlIframe();\n });\n });\n\n if (!isListen) {\n isListen = true;\n\n // @ts-ignore\n window.addEventListener('resize', window.walletEntryPlugin.resize(), false);\n window.addEventListener(\n 'message',\n (events: any) => {\n if ((window as any).walletEntryPlugin?.walletOptions?.widgetIntegration === 'embedded') {\n return;\n }\n if (events?.data?.type === 'PARTICLE_WALLET_RESIZE_IFRAME') {\n const walletEntryPlugin = (window as any).walletEntryPlugin;\n const iframeContent = document.querySelector('.particle-pwe-iframe-content') as HTMLDivElement;\n const classList = iframeContent?.classList;\n if (classList?.contains?.(fullScreenClass)) {\n if (!this._walletOptions?.visible || window.screen.width < 600) {\n walletEntryPlugin.closeWallet();\n } else {\n classList?.remove?.(fullScreenClass);\n walletEntryPlugin.resize();\n walletEntryPlugin.updateIframeContentPosition();\n }\n } else {\n classList?.add?.(fullScreenClass);\n }\n } else if (events?.data?.type === 'PARTICLE_WALLET_CLOSE_IFRAME') {\n // @ts-ignore\n window.walletEntryPlugin.closeWallet();\n }\n },\n false\n );\n }\n }\n}\n", "export const isServer = () => {\n return typeof window === 'undefined';\n};\n\nexport function isNullish(x: unknown): boolean {\n return x === undefined || x === null;\n}\n", "import { isServer } from './utils';\n\ntype Env = {\n walletUrl: string;\n};\n\nclass GlobalConfig {\n /**\n * custom wallet url\n */\n #customWalletUrl: string | undefined;\n\n #devEnv: Env = {\n walletUrl: 'https://wallet-debug.particle.network',\n };\n\n #stagingEnv: Env = {\n walletUrl: 'https://wallet-staging.particle.network',\n };\n\n #productionEnv: Env = {\n walletUrl: 'https://wallet.particle.network',\n };\n\n get version() {\n // replace when build\n return 'web_' + '2.1.1';\n }\n\n get env() {\n let currentEnv = this.#productionEnv;\n\n // @ts-ignore\n if (!isServer() && window.__PARTICLE_ENVIRONMENT__ === 'development') {\n currentEnv = this.#devEnv;\n // @ts-ignore\n } else if (!isServer() && window.__PARTICLE_ENVIRONMENT__ === 'staging') {\n currentEnv = this.#stagingEnv;\n } else {\n currentEnv = this.#productionEnv;\n }\n\n if (this.#customWalletUrl) {\n currentEnv.walletUrl = this.#customWalletUrl;\n }\n\n return currentEnv;\n }\n\n updateWalletUrl(url: string) {\n if (url) {\n this.#customWalletUrl = url;\n }\n }\n}\n\nconst globalConfig = new GlobalConfig();\nexport default globalConfig;\n", "const html = `\n <button class=\"particle-pwe-btn\">\n <img class=\"particle-pwe-img particle-pwe-wallet-icon\" src=\"\" alt=\"\" />\n <img class=\"particle-pwe-img particle-pwe-down-arrow particle-pwe-down-arrow-hide\" src=\"\" alt=\"\" />\n </button>\n <div class=\"particle-pwe-iframe-content\">\n </div>\n`;\n\nconst walletEntryRender = () => {\n const className = 'particle-wallet-entry-container';\n const el = document.querySelector('.' + className);\n el && el.remove();\n const EL = document.createElement('div');\n EL.classList.add(className);\n EL.innerHTML = html;\n document.body.appendChild(EL);\n};\n\nexport default walletEntryRender;\n", "export const walletIconDarkBase64 =\n '';\n\nexport const downArrowDarkBase64 =\n '