@ksangkuk10/wallet-controller 1.7.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (124) hide show
  1. package/README.md +7 -0
  2. package/_commonjs/controller.js +704 -0
  3. package/_commonjs/env.js +9 -0
  4. package/_commonjs/exception/isError.js +15 -0
  5. package/_commonjs/exception/mapExtensionTxError.js +46 -0
  6. package/_commonjs/exception/mapWalletConnectError.js +69 -0
  7. package/_commonjs/getChainOptions.js +57 -0
  8. package/_commonjs/index.js +22 -0
  9. package/_commonjs/modules/connect-modal/index.js +80 -0
  10. package/_commonjs/modules/connect-modal/style.js +161 -0
  11. package/_commonjs/modules/extension-router/ExtensionRouter.js +241 -0
  12. package/_commonjs/modules/extension-router/index.js +19 -0
  13. package/_commonjs/modules/extension-router/modal.js +80 -0
  14. package/_commonjs/modules/extension-router/modal.style.js +161 -0
  15. package/_commonjs/modules/extension-router/multiChannel.js +18 -0
  16. package/_commonjs/modules/extension-router/session.js +37 -0
  17. package/_commonjs/modules/extension-router/types.js +16 -0
  18. package/_commonjs/modules/legacy-extension/LegacyExtensionConnector.js +143 -0
  19. package/_commonjs/modules/legacy-extension/createFixedExtension.js +236 -0
  20. package/_commonjs/modules/legacy-extension/index.js +19 -0
  21. package/_commonjs/modules/readonly-wallet/connect.js +24 -0
  22. package/_commonjs/modules/readonly-wallet/index.js +20 -0
  23. package/_commonjs/modules/readonly-wallet/modal.js +123 -0
  24. package/_commonjs/modules/readonly-wallet/modal.style.js +204 -0
  25. package/_commonjs/modules/readonly-wallet/storage.js +40 -0
  26. package/_commonjs/modules/readonly-wallet/types.js +3 -0
  27. package/_commonjs/modules/walletconnect/connect.js +421 -0
  28. package/_commonjs/modules/walletconnect/errors.js +51 -0
  29. package/_commonjs/modules/walletconnect/impl/socket-transport/index.js +204 -0
  30. package/_commonjs/modules/walletconnect/impl/socket-transport/network.js +30 -0
  31. package/_commonjs/modules/walletconnect/index.js +22 -0
  32. package/_commonjs/modules/walletconnect/modal.js +147 -0
  33. package/_commonjs/modules/walletconnect/modal.style.js +138 -0
  34. package/_commonjs/modules/walletconnect/types.js +13 -0
  35. package/_commonjs/operators/getExtensions.js +55 -0
  36. package/_commonjs/operators/toConnectedWallet.js +24 -0
  37. package/_commonjs/operators/toLcdClient.js +15 -0
  38. package/_commonjs/utils/browser-check.js +47 -0
  39. package/_commonjs/utils/checkExtensionReady.js +28 -0
  40. package/_commonjs/utils/sortConnections.js +13 -0
  41. package/_commonjs/verifyBytes.js +19 -0
  42. package/controller.d.ts +220 -0
  43. package/controller.js +697 -0
  44. package/env.d.ts +2 -0
  45. package/env.js +5 -0
  46. package/exception/isError.d.ts +3 -0
  47. package/exception/isError.js +11 -0
  48. package/exception/mapExtensionTxError.d.ts +5 -0
  49. package/exception/mapExtensionTxError.js +41 -0
  50. package/exception/mapWalletConnectError.d.ts +6 -0
  51. package/exception/mapWalletConnectError.js +63 -0
  52. package/getChainOptions.d.ts +3 -0
  53. package/getChainOptions.js +53 -0
  54. package/index.d.ts +6 -0
  55. package/index.js +6 -0
  56. package/modules/connect-modal/index.d.ts +2 -0
  57. package/modules/connect-modal/index.js +76 -0
  58. package/modules/connect-modal/style.d.ts +1 -0
  59. package/modules/connect-modal/style.js +158 -0
  60. package/modules/extension-router/ExtensionRouter.d.ts +43 -0
  61. package/modules/extension-router/ExtensionRouter.js +237 -0
  62. package/modules/extension-router/index.d.ts +2 -0
  63. package/modules/extension-router/index.js +3 -0
  64. package/modules/extension-router/modal.d.ts +2 -0
  65. package/modules/extension-router/modal.js +76 -0
  66. package/modules/extension-router/modal.style.d.ts +1 -0
  67. package/modules/extension-router/modal.style.js +158 -0
  68. package/modules/extension-router/multiChannel.d.ts +13 -0
  69. package/modules/extension-router/multiChannel.js +14 -0
  70. package/modules/extension-router/session.d.ts +8 -0
  71. package/modules/extension-router/session.js +31 -0
  72. package/modules/extension-router/types.d.ts +35 -0
  73. package/modules/extension-router/types.js +13 -0
  74. package/modules/legacy-extension/LegacyExtensionConnector.d.ts +30 -0
  75. package/modules/legacy-extension/LegacyExtensionConnector.js +139 -0
  76. package/modules/legacy-extension/createFixedExtension.d.ts +46 -0
  77. package/modules/legacy-extension/createFixedExtension.js +232 -0
  78. package/modules/legacy-extension/index.d.ts +2 -0
  79. package/modules/legacy-extension/index.js +3 -0
  80. package/modules/readonly-wallet/connect.d.ts +8 -0
  81. package/modules/readonly-wallet/connect.js +19 -0
  82. package/modules/readonly-wallet/index.d.ts +3 -0
  83. package/modules/readonly-wallet/index.js +4 -0
  84. package/modules/readonly-wallet/modal.d.ts +7 -0
  85. package/modules/readonly-wallet/modal.js +119 -0
  86. package/modules/readonly-wallet/modal.style.d.ts +1 -0
  87. package/modules/readonly-wallet/modal.style.js +201 -0
  88. package/modules/readonly-wallet/storage.d.ts +4 -0
  89. package/modules/readonly-wallet/storage.js +34 -0
  90. package/modules/readonly-wallet/types.d.ts +5 -0
  91. package/modules/readonly-wallet/types.js +2 -0
  92. package/modules/walletconnect/connect.d.ts +42 -0
  93. package/modules/walletconnect/connect.js +390 -0
  94. package/modules/walletconnect/errors.d.ts +22 -0
  95. package/modules/walletconnect/errors.js +41 -0
  96. package/modules/walletconnect/impl/socket-transport/index.d.ts +39 -0
  97. package/modules/walletconnect/impl/socket-transport/index.js +199 -0
  98. package/modules/walletconnect/impl/socket-transport/network.d.ts +8 -0
  99. package/modules/walletconnect/impl/socket-transport/network.js +28 -0
  100. package/modules/walletconnect/index.d.ts +5 -0
  101. package/modules/walletconnect/index.js +6 -0
  102. package/modules/walletconnect/modal.d.ts +12 -0
  103. package/modules/walletconnect/modal.js +143 -0
  104. package/modules/walletconnect/modal.style.d.ts +1 -0
  105. package/modules/walletconnect/modal.style.js +135 -0
  106. package/modules/walletconnect/types.d.ts +24 -0
  107. package/modules/walletconnect/types.js +10 -0
  108. package/operators/getExtensions.d.ts +9 -0
  109. package/operators/getExtensions.js +51 -0
  110. package/operators/toConnectedWallet.d.ts +4 -0
  111. package/operators/toConnectedWallet.js +20 -0
  112. package/operators/toLcdClient.d.ts +4 -0
  113. package/operators/toLcdClient.js +11 -0
  114. package/package.json +550 -0
  115. package/utils/browser-check.d.ts +3 -0
  116. package/utils/browser-check.js +38 -0
  117. package/utils/checkExtensionReady.d.ts +6 -0
  118. package/utils/checkExtensionReady.js +24 -0
  119. package/utils/sortConnections.d.ts +2 -0
  120. package/utils/sortConnections.js +9 -0
  121. package/verifyBytes.d.ts +4 -0
  122. package/verifyBytes.js +15 -0
  123. package/~/.npm/_cacache/content-v2/sha512/ad/b5/8c87dfae7c208906a88f1997b323933e7efb4e481bcfdc559cb13199d077e1b40abcb161561a293ca59cd98aae224cd0877555e6e7a2e11021bc55ebedf4 +0 -0
  124. package/~/.npm/_cacache/index-v5/6a/39/c03f62d7353a7cc1411e180ec37c393f9fedcfd081fa567f67bf32e6fb68 +2 -0
package/controller.js ADDED
@@ -0,0 +1,697 @@
1
+ import { AccAddress, PublicKey, SimplePublicKey, Tx, } from '@xpla/xpla.js';
2
+ import { ConnectType, WalletApp, WalletStatus, } from '@ksangkuk10/wallet-types';
3
+ import { WebExtensionTxStatus, } from '@ksangkuk10/web-extension-interface';
4
+ import deepEqual from 'fast-deep-equal';
5
+ import { BehaviorSubject, combineLatest, firstValueFrom, } from 'rxjs';
6
+ import { filter, map } from 'rxjs/operators';
7
+ import { CHROME_EXTENSION_INSTALL_URL, DEFAULT_CHROME_EXTENSION_COMPATIBLE_BROWSER_CHECK, } from './env';
8
+ import { mapExtensionSignBytesError, mapExtensionTxError, } from './exception/mapExtensionTxError';
9
+ import { mapWalletConnectError, mapWalletConnectSignError, mapWalletConnectSignBytesError, } from './exception/mapWalletConnectError';
10
+ import { selectConnection } from './modules/connect-modal';
11
+ import { ExtensionRouter, ExtensionRouterStatus, } from './modules/extension-router';
12
+ import { getXplaExtensions, } from './modules/extension-router/multiChannel';
13
+ import { connect as reConnect, connectIfSessionExists as reConnectIfSessionExists, readonlyWalletModal, } from './modules/readonly-wallet';
14
+ import { connect as wcConnect, connectIfSessionExists as wcConnectIfSessionExists, WalletConnectSessionStatus, } from './modules/walletconnect';
15
+ import { getExtensions } from './operators/getExtensions';
16
+ import { toConnectedWallet } from './operators/toConnectedWallet';
17
+ import { toLcdClient } from './operators/toLcdClient';
18
+ import { isDesktopChrome } from './utils/browser-check';
19
+ import { checkExtensionReady } from './utils/checkExtensionReady';
20
+ import { sortConnections } from './utils/sortConnections';
21
+ const CONNECTIONS = {
22
+ [ConnectType.READONLY]: {
23
+ type: ConnectType.READONLY,
24
+ name: 'View an address',
25
+ icon: 'https://assets.xpla.io/icon/wallet-provider/readonly.svg',
26
+ },
27
+ [ConnectType.WALLETCONNECT]: {
28
+ type: ConnectType.WALLETCONNECT,
29
+ name: 'Wallet Connect',
30
+ icon: 'https://assets.xpla.io/icon/wallet-provider/walletconnect.svg',
31
+ },
32
+ };
33
+ const DEFAULT_WAITING_CHROME_EXTENSION_INSTALL_CHECK = 1000 * 3;
34
+ const WALLETCONNECT_SUPPORT_FEATURES = new Set([
35
+ 'post', 'sign', 'sign-bytes'
36
+ ]);
37
+ const EMPTY_SUPPORT_FEATURES = new Set();
38
+ //noinspection ES6MissingAwait
39
+ export class WalletController {
40
+ constructor(options) {
41
+ var _a;
42
+ this.options = options;
43
+ this.extension = null;
44
+ this.walletConnect = null;
45
+ this.readonlyWallet = null;
46
+ this.disableReadonlyWallet = null;
47
+ this.disableExtension = null;
48
+ this.disableWalletConnect = null;
49
+ /**
50
+ * Some mobile wallet emulates the behavior of chrome extension.
51
+ * It confirms that the current connection environment is such a wallet.
52
+ * (If you are running connect() by checking availableConnectType, you do not need to use this API.)
53
+ *
54
+ * @see Wallet#isChromeExtensionCompatibleBrowser
55
+ */
56
+ this.isChromeExtensionCompatibleBrowser = () => {
57
+ var _a;
58
+ return ((_a = this.options.dangerously__chromeExtensionCompatibleBrowserCheck) !== null && _a !== void 0 ? _a : DEFAULT_CHROME_EXTENSION_COMPATIBLE_BROWSER_CHECK)(navigator.userAgent);
59
+ };
60
+ /**
61
+ * available connect types on the browser
62
+ *
63
+ * @see Wallet#availableConnectTypes
64
+ */
65
+ this.availableConnectTypes = () => {
66
+ return this._availableConnectTypes.asObservable();
67
+ };
68
+ /**
69
+ * available connections includes identifier, name, icon
70
+ *
71
+ * @see Wallet#availableConnections
72
+ */
73
+ this.availableConnections = () => {
74
+ return this._availableConnectTypes.pipe(map((connectTypes) => {
75
+ const connections = [];
76
+ for (const connectType of connectTypes) {
77
+ if (connectType === ConnectType.EXTENSION) {
78
+ const xplaExtensions = getXplaExtensions();
79
+ for (const xplaExtension of xplaExtensions) {
80
+ connections.push(memoConnection(ConnectType.EXTENSION, xplaExtension.name, xplaExtension.icon, xplaExtension.identifier));
81
+ }
82
+ }
83
+ else {
84
+ connections.push(CONNECTIONS[connectType]);
85
+ }
86
+ }
87
+ return sortConnections(connections);
88
+ }));
89
+ };
90
+ /**
91
+ * available install types on the browser
92
+ *
93
+ * in this time, this only contains [ConnectType.EXTENSION]
94
+ *
95
+ * @see Wallet#availableInstallTypes
96
+ */
97
+ this.availableInstallTypes = () => {
98
+ return this._availableInstallTypes.asObservable();
99
+ };
100
+ /**
101
+ * available installations includes identifier, name, icon, url
102
+ *
103
+ * @see Wallet#availableInstallations
104
+ */
105
+ this.availableInstallations = () => {
106
+ return combineLatest([this.availableConnections(), getExtensions()]).pipe(map(([connections, extensions]) => {
107
+ const installedIdentifiers = new Set(connections
108
+ .filter(({ type, identifier }) => {
109
+ return type === ConnectType.EXTENSION && !!identifier;
110
+ })
111
+ .map(({ identifier }) => {
112
+ return identifier;
113
+ }));
114
+ return extensions
115
+ .filter(({ identifier }) => {
116
+ return !installedIdentifiers.has(identifier);
117
+ })
118
+ .map(({ name, identifier, icon, url }) => {
119
+ return {
120
+ type: ConnectType.EXTENSION,
121
+ identifier,
122
+ name,
123
+ icon,
124
+ url,
125
+ };
126
+ });
127
+ }));
128
+ };
129
+ /**
130
+ * @see Wallet#status
131
+ * @see Wallet#network
132
+ * @see Wallet#wallets
133
+ */
134
+ this.states = () => {
135
+ return this._states.asObservable();
136
+ };
137
+ /** get connectedWallet */
138
+ this.connectedWallet = () => {
139
+ return this._states.pipe(toConnectedWallet(this));
140
+ };
141
+ /** get lcdClient */
142
+ this.lcdClient = (lcdClientConfig) => {
143
+ return this._states.pipe(toLcdClient(lcdClientConfig));
144
+ };
145
+ /**
146
+ * reload the connected wallet states
147
+ *
148
+ * in this time, this only work on the ConnectType.EXTENSION
149
+ *
150
+ * @see Wallet#recheckStatus
151
+ */
152
+ this.refetchStates = () => {
153
+ var _a;
154
+ if (this.disableExtension) {
155
+ (_a = this.extension) === null || _a === void 0 ? void 0 : _a.refetchStates();
156
+ }
157
+ };
158
+ /**
159
+ * @deprecated Please use availableInstallations
160
+ *
161
+ * install for the connect type
162
+ *
163
+ * @see Wallet#install
164
+ */
165
+ this.install = (type) => {
166
+ if (type === ConnectType.EXTENSION) {
167
+ // TODO separate install links by browser types
168
+ window.open(CHROME_EXTENSION_INSTALL_URL, '_blank');
169
+ }
170
+ else {
171
+ console.warn(`[WalletController] ConnectType "${type}" does not support install() function`);
172
+ }
173
+ };
174
+ /**
175
+ * connect to wallet
176
+ *
177
+ * @see Wallet#connect
178
+ */
179
+ this.connect = async (_type, _identifier, _walletApp) => {
180
+ var _a, _b, _c, _d;
181
+ let type;
182
+ let identifier;
183
+ if (!!_type) {
184
+ type = _type;
185
+ identifier = _identifier;
186
+ }
187
+ else {
188
+ const connections = await firstValueFrom(this.availableConnections());
189
+ const selector = (_a = this.options.selectConnection) !== null && _a !== void 0 ? _a : selectConnection;
190
+ const selected = await selector(connections);
191
+ if (!selected) {
192
+ return;
193
+ }
194
+ type = selected[0];
195
+ identifier = selected[1];
196
+ }
197
+ switch (type) {
198
+ case ConnectType.READONLY:
199
+ const networks = Object.keys(this.options.walletConnectChainIds).map((chainId) => this.options.walletConnectChainIds[+chainId]);
200
+ const createReadonlyWalletSession = (_d = (_c = (_b = this.options).createReadonlyWalletSession) === null || _c === void 0 ? void 0 : _c.call(_b, networks)) !== null && _d !== void 0 ? _d : readonlyWalletModal({ networks });
201
+ const readonlyWalletSession = await createReadonlyWalletSession;
202
+ if (readonlyWalletSession) {
203
+ this.enableReadonlyWallet(reConnect(readonlyWalletSession));
204
+ }
205
+ break;
206
+ case ConnectType.WALLETCONNECT:
207
+ this.enableWalletConnect(wcConnect(this.options, false, _walletApp));
208
+ break;
209
+ case ConnectType.EXTENSION:
210
+ if (!this.extension) {
211
+ throw new Error(`extension instance is not created!`);
212
+ }
213
+ this.extension.connect(identifier);
214
+ this.enableExtension();
215
+ break;
216
+ default:
217
+ throw new Error(`Unknown ConnectType!`);
218
+ }
219
+ };
220
+ /**
221
+ * manual connect to read only session
222
+ *
223
+ * @see Wallet#connectReadonly
224
+ */
225
+ this.connectReadonly = (xplaAddress, network) => {
226
+ this.enableReadonlyWallet(reConnect({
227
+ xplaAddress,
228
+ network,
229
+ }));
230
+ };
231
+ /** @see Wallet#disconnect */
232
+ this.disconnect = () => {
233
+ var _a, _b, _c;
234
+ (_a = this.disableReadonlyWallet) === null || _a === void 0 ? void 0 : _a.call(this);
235
+ this.disableReadonlyWallet = null;
236
+ (_b = this.disableExtension) === null || _b === void 0 ? void 0 : _b.call(this);
237
+ this.disableExtension = null;
238
+ (_c = this.disableWalletConnect) === null || _c === void 0 ? void 0 : _c.call(this);
239
+ this.disableWalletConnect = null;
240
+ this.updateStates(this._notConnected);
241
+ };
242
+ /**
243
+ * @see Wallet#post
244
+ * @param tx
245
+ * @param xplaAddress only available new extension
246
+ */
247
+ this.post = async (tx, xplaAddress, walletApp) => {
248
+ // ---------------------------------------------
249
+ // extension
250
+ // ---------------------------------------------
251
+ if (this.disableExtension) {
252
+ return new Promise((resolve, reject) => {
253
+ if (!this.extension) {
254
+ reject(new Error(`extension instance not created!`));
255
+ return;
256
+ }
257
+ const subscription = this.extension.post(tx, xplaAddress).subscribe({
258
+ next: (txResult) => {
259
+ if (txResult.status === WebExtensionTxStatus.SUCCEED) {
260
+ resolve({
261
+ ...tx,
262
+ result: txResult.payload,
263
+ success: true,
264
+ });
265
+ subscription.unsubscribe();
266
+ }
267
+ },
268
+ error: (error) => {
269
+ reject(mapExtensionTxError(tx, error));
270
+ subscription.unsubscribe();
271
+ },
272
+ });
273
+ });
274
+ }
275
+ // ---------------------------------------------
276
+ // wallet connect
277
+ // ---------------------------------------------
278
+ else if (this.walletConnect) {
279
+ return this.walletConnect
280
+ .post(tx, walletApp)
281
+ .then((result) => ({
282
+ ...tx,
283
+ result,
284
+ success: true,
285
+ }))
286
+ .catch((error) => {
287
+ throw mapWalletConnectError(tx, error);
288
+ });
289
+ }
290
+ else {
291
+ throw new Error(`There are no connections that can be posting tx!`);
292
+ }
293
+ };
294
+ /**
295
+ * @see Wallet#sign
296
+ * @param tx
297
+ * @param xplaAddress only available new extension
298
+ */
299
+ this.sign = async (tx, xplaAddress, walletApp) => {
300
+ if (this.disableExtension) {
301
+ return new Promise((resolve, reject) => {
302
+ if (!this.extension) {
303
+ reject(new Error(`extension instance is not created!`));
304
+ return;
305
+ }
306
+ const subscription = this.extension.sign(tx, xplaAddress).subscribe({
307
+ next: (txResult) => {
308
+ if (txResult.status === WebExtensionTxStatus.SUCCEED) {
309
+ resolve({
310
+ ...tx,
311
+ result: Tx.fromData(txResult.payload),
312
+ success: true,
313
+ });
314
+ subscription.unsubscribe();
315
+ }
316
+ },
317
+ error: (error) => {
318
+ reject(mapExtensionTxError(tx, error));
319
+ subscription.unsubscribe();
320
+ },
321
+ });
322
+ });
323
+ }
324
+ // ---------------------------------------------
325
+ // wallet connect
326
+ // ---------------------------------------------
327
+ else if (this.walletConnect) {
328
+ if (walletApp && walletApp === WalletApp.XPLA_GAMES) {
329
+ throw new Error(`There are no connections that can be signing!`);
330
+ }
331
+ return this.walletConnect
332
+ .sign(tx, walletApp)
333
+ .then((result) => {
334
+ return {
335
+ ...tx,
336
+ result: Tx.fromData(result, false),
337
+ success: true,
338
+ };
339
+ })
340
+ .catch((error) => {
341
+ throw mapWalletConnectSignError(tx, error);
342
+ });
343
+ }
344
+ else {
345
+ throw new Error(`There are no connections that can be signing!`);
346
+ }
347
+ };
348
+ /**
349
+ * @see Wallet#signBytes
350
+ * @param bytes
351
+ * @param xplaAddress only available new extension
352
+ */
353
+ this.signBytes = async (bytes, xplaAddress, walletApp) => {
354
+ if (this.disableExtension) {
355
+ return new Promise((resolve, reject) => {
356
+ if (!this.extension) {
357
+ reject(new Error(`extension instance is not created!`));
358
+ return;
359
+ }
360
+ const subscription = this.extension
361
+ .signBytes(bytes, xplaAddress)
362
+ .subscribe({
363
+ next: (txResult) => {
364
+ if (txResult.status === WebExtensionTxStatus.SUCCEED) {
365
+ resolve({
366
+ result: {
367
+ recid: txResult.payload.recid,
368
+ signature: Uint8Array.from(Buffer.from(txResult.payload.signature, 'base64')),
369
+ public_key: txResult.payload.public_key
370
+ ? PublicKey.fromData(txResult.payload.public_key)
371
+ : undefined,
372
+ },
373
+ success: true,
374
+ });
375
+ subscription.unsubscribe();
376
+ }
377
+ },
378
+ error: (error) => {
379
+ reject(mapExtensionSignBytesError(bytes, error));
380
+ subscription.unsubscribe();
381
+ },
382
+ });
383
+ });
384
+ }
385
+ // ---------------------------------------------
386
+ // wallet connect
387
+ // ---------------------------------------------
388
+ else if (this.walletConnect) {
389
+ if (walletApp && walletApp === WalletApp.XPLA_GAMES) {
390
+ throw new Error(`There are no connections that can be signing bytes!`);
391
+ }
392
+ return this.walletConnect
393
+ .signBytes(bytes, walletApp)
394
+ .then((result) => {
395
+ const key = new SimplePublicKey(String(result.public_key)).toData();
396
+ return {
397
+ result: {
398
+ recid: result.recid,
399
+ signature: Uint8Array.from(Buffer.from(result.signature, 'base64')),
400
+ public_key: key
401
+ ? PublicKey.fromData(key)
402
+ : undefined,
403
+ },
404
+ success: true,
405
+ };
406
+ })
407
+ .catch((error) => {
408
+ throw mapWalletConnectSignBytesError(bytes, error);
409
+ });
410
+ }
411
+ else {
412
+ throw new Error(`There are no connections that can be signing bytes!`);
413
+ }
414
+ };
415
+ /**
416
+ * @see Wallet#hasCW20Tokens
417
+ * @param chainID
418
+ * @param tokenAddrs Token addresses
419
+ */
420
+ this.hasCW20Tokens = async (chainID, ...tokenAddrs) => {
421
+ if (this.availableExtensionFeature('cw20-token')) {
422
+ return this.extension.hasCW20Tokens(chainID, ...tokenAddrs);
423
+ }
424
+ throw new Error(`Does not support hasCW20Tokens() on this connection`);
425
+ };
426
+ /**
427
+ * @see Wallet#addCW20Tokens
428
+ * @param chainID
429
+ * @param tokenAddrs Token addresses
430
+ */
431
+ this.addCW20Tokens = async (chainID, ...tokenAddrs) => {
432
+ if (this.availableExtensionFeature('cw20-token')) {
433
+ return this.extension.addCW20Tokens(chainID, ...tokenAddrs);
434
+ }
435
+ throw new Error(`Does not support addCW20Tokens() on this connection`);
436
+ };
437
+ /**
438
+ * @see Wallet#hasNetwork
439
+ * @param network
440
+ */
441
+ this.hasNetwork = (network) => {
442
+ if (this.availableExtensionFeature('network')) {
443
+ return this.extension.hasNetwork(network);
444
+ }
445
+ throw new Error(`Does not support hasNetwork() on this connection`);
446
+ };
447
+ /**
448
+ * @see Wallet#hasNetwork
449
+ * @param network
450
+ */
451
+ this.addNetwork = (network) => {
452
+ if (this.availableExtensionFeature('network')) {
453
+ return this.extension.addNetwork(network);
454
+ }
455
+ throw new Error(`Does not support addNetwork() on this connection`);
456
+ };
457
+ // ================================================================
458
+ // internal
459
+ // connect type changing
460
+ // ================================================================
461
+ this.availableExtensionFeature = (feature) => {
462
+ if (this.disableExtension && this.extension) {
463
+ const states = this.extension.getLastStates();
464
+ return (states.type === ExtensionRouterStatus.WALLET_CONNECTED &&
465
+ states.supportFeatures.has(feature));
466
+ }
467
+ };
468
+ this.updateStates = (next) => {
469
+ const prev = this._states.getValue();
470
+ if (next.status === WalletStatus.WALLET_CONNECTED &&
471
+ next.wallets.length === 0) {
472
+ next = {
473
+ status: WalletStatus.WALLET_NOT_CONNECTED,
474
+ network: next.network,
475
+ };
476
+ }
477
+ if (prev.status !== next.status || !deepEqual(prev, next)) {
478
+ this._states.next(next);
479
+ }
480
+ };
481
+ this.enableReadonlyWallet = (readonlyWallet) => {
482
+ var _a, _b, _c;
483
+ (_a = this.disableWalletConnect) === null || _a === void 0 ? void 0 : _a.call(this);
484
+ (_b = this.disableExtension) === null || _b === void 0 ? void 0 : _b.call(this);
485
+ if (this.readonlyWallet === readonlyWallet ||
486
+ (((_c = this.readonlyWallet) === null || _c === void 0 ? void 0 : _c.xplaAddress) === readonlyWallet.xplaAddress &&
487
+ this.readonlyWallet.network === readonlyWallet.network)) {
488
+ return;
489
+ }
490
+ if (this.readonlyWallet) {
491
+ this.readonlyWallet.disconnect();
492
+ }
493
+ this.readonlyWallet = readonlyWallet;
494
+ this.updateStates({
495
+ status: WalletStatus.WALLET_CONNECTED,
496
+ network: readonlyWallet.network,
497
+ wallets: [
498
+ {
499
+ connectType: ConnectType.READONLY,
500
+ xplaAddress: readonlyWallet.xplaAddress,
501
+ design: 'readonly',
502
+ },
503
+ ],
504
+ supportFeatures: EMPTY_SUPPORT_FEATURES,
505
+ connection: CONNECTIONS.READONLY,
506
+ });
507
+ this.disableReadonlyWallet = () => {
508
+ readonlyWallet.disconnect();
509
+ this.readonlyWallet = null;
510
+ this.disableReadonlyWallet = null;
511
+ };
512
+ };
513
+ this.enableExtension = () => {
514
+ var _a, _b;
515
+ (_a = this.disableReadonlyWallet) === null || _a === void 0 ? void 0 : _a.call(this);
516
+ (_b = this.disableWalletConnect) === null || _b === void 0 ? void 0 : _b.call(this);
517
+ if (this.disableExtension || !this.extension) {
518
+ return;
519
+ }
520
+ const extensionSubscription = this.extension.states().subscribe({
521
+ next: (extensionStates) => {
522
+ if (extensionStates.type === ExtensionRouterStatus.WALLET_CONNECTED &&
523
+ AccAddress.validate(extensionStates.wallet.xplaAddress)) {
524
+ this.updateStates({
525
+ status: WalletStatus.WALLET_CONNECTED,
526
+ network: extensionStates.network,
527
+ wallets: [
528
+ {
529
+ connectType: ConnectType.EXTENSION,
530
+ xplaAddress: extensionStates.wallet.xplaAddress,
531
+ design: extensionStates.wallet.design,
532
+ },
533
+ ],
534
+ supportFeatures: extensionStates.supportFeatures,
535
+ connection: memoConnection(ConnectType.EXTENSION, extensionStates.extensionInfo.name, extensionStates.extensionInfo.icon, extensionStates.extensionInfo.identifier),
536
+ });
537
+ }
538
+ else {
539
+ this.updateStates(this._notConnected);
540
+ }
541
+ },
542
+ });
543
+ this.disableExtension = () => {
544
+ var _a;
545
+ (_a = this.extension) === null || _a === void 0 ? void 0 : _a.disconnect();
546
+ extensionSubscription.unsubscribe();
547
+ this.disableExtension = null;
548
+ };
549
+ };
550
+ this.enableWalletConnect = (walletConnect) => {
551
+ var _a, _b;
552
+ (_a = this.disableReadonlyWallet) === null || _a === void 0 ? void 0 : _a.call(this);
553
+ (_b = this.disableExtension) === null || _b === void 0 ? void 0 : _b.call(this);
554
+ if (this.walletConnect === walletConnect) {
555
+ return;
556
+ }
557
+ if (this.walletConnect) {
558
+ this.walletConnect.disconnect();
559
+ }
560
+ this.walletConnect = walletConnect;
561
+ const subscribeWalletConnect = (wc) => {
562
+ return wc.session().subscribe({
563
+ next: (status) => {
564
+ var _a;
565
+ switch (status.status) {
566
+ case WalletConnectSessionStatus.CONNECTED:
567
+ this.updateStates({
568
+ status: WalletStatus.WALLET_CONNECTED,
569
+ network: (_a = this.options.walletConnectChainIds[status.chainId]) !== null && _a !== void 0 ? _a : this.options.defaultNetwork,
570
+ wallets: [
571
+ {
572
+ connectType: ConnectType.WALLETCONNECT,
573
+ xplaAddress: status.xplaAddress,
574
+ design: 'walletconnect',
575
+ },
576
+ ],
577
+ supportFeatures: WALLETCONNECT_SUPPORT_FEATURES,
578
+ connection: CONNECTIONS.WALLETCONNECT,
579
+ });
580
+ break;
581
+ default:
582
+ this.updateStates(this._notConnected);
583
+ break;
584
+ }
585
+ },
586
+ });
587
+ };
588
+ const walletConnectSessionSubscription = subscribeWalletConnect(walletConnect);
589
+ this.disableWalletConnect = () => {
590
+ var _a;
591
+ (_a = this.walletConnect) === null || _a === void 0 ? void 0 : _a.disconnect();
592
+ this.walletConnect = null;
593
+ walletConnectSessionSubscription.unsubscribe();
594
+ this.disableWalletConnect = null;
595
+ };
596
+ };
597
+ this._notConnected = {
598
+ status: WalletStatus.WALLET_NOT_CONNECTED,
599
+ network: options.defaultNetwork,
600
+ };
601
+ this._initializing = {
602
+ status: WalletStatus.INITIALIZING,
603
+ network: options.defaultNetwork,
604
+ };
605
+ this._availableConnectTypes = new BehaviorSubject([
606
+ ConnectType.READONLY,
607
+ ConnectType.WALLETCONNECT,
608
+ ]);
609
+ this._availableInstallTypes = new BehaviorSubject([]);
610
+ this._states = new BehaviorSubject(this._initializing);
611
+ let numSessionCheck = 0;
612
+ // wait checking the availability of the chrome extension
613
+ // 0. check if extension wallet session is exists
614
+ checkExtensionReady((_a = options.waitingChromeExtensionInstallCheck) !== null && _a !== void 0 ? _a : DEFAULT_WAITING_CHROME_EXTENSION_INSTALL_CHECK, this.isChromeExtensionCompatibleBrowser()).then((ready) => {
615
+ var _a;
616
+ if (ready) {
617
+ this._availableConnectTypes.next([
618
+ ConnectType.EXTENSION,
619
+ ConnectType.WALLETCONNECT,
620
+ ConnectType.READONLY,
621
+ ]);
622
+ this.extension = new ExtensionRouter({
623
+ hostWindow: window,
624
+ selectExtension: options.selectExtension,
625
+ dangerously__chromeExtensionCompatibleBrowserCheck: (_a = options.dangerously__chromeExtensionCompatibleBrowserCheck) !== null && _a !== void 0 ? _a : DEFAULT_CHROME_EXTENSION_COMPATIBLE_BROWSER_CHECK,
626
+ defaultNetwork: options.defaultNetwork,
627
+ });
628
+ const subscription = this.extension
629
+ .states()
630
+ .pipe(filter(({ type }) => type !== ExtensionRouterStatus.INITIALIZING))
631
+ .subscribe((extensionStates) => {
632
+ try {
633
+ subscription.unsubscribe();
634
+ }
635
+ catch (_a) { }
636
+ if (extensionStates.type === ExtensionRouterStatus.WALLET_CONNECTED &&
637
+ !this.disableWalletConnect &&
638
+ !this.disableReadonlyWallet) {
639
+ this.enableExtension();
640
+ }
641
+ else if (numSessionCheck === 0) {
642
+ numSessionCheck += 1;
643
+ }
644
+ else {
645
+ this.updateStates(this._notConnected);
646
+ }
647
+ });
648
+ }
649
+ else {
650
+ if (isDesktopChrome(this.isChromeExtensionCompatibleBrowser())) {
651
+ this._availableInstallTypes.next([ConnectType.EXTENSION]);
652
+ }
653
+ if (numSessionCheck === 0) {
654
+ numSessionCheck += 1;
655
+ }
656
+ else {
657
+ this.updateStates(this._notConnected);
658
+ }
659
+ }
660
+ });
661
+ // 1. check if readonly wallet session is exists
662
+ const draftReadonlyWallet = reConnectIfSessionExists();
663
+ if (draftReadonlyWallet) {
664
+ this.enableReadonlyWallet(draftReadonlyWallet);
665
+ return;
666
+ }
667
+ // 2. check if walletconnect sesison is exists
668
+ const draftWalletConnect = wcConnectIfSessionExists(options);
669
+ if (draftWalletConnect &&
670
+ draftWalletConnect.getLatestSession().status ===
671
+ WalletConnectSessionStatus.CONNECTED) {
672
+ this.enableWalletConnect(draftWalletConnect);
673
+ }
674
+ else if (numSessionCheck === 0) {
675
+ numSessionCheck += 1;
676
+ }
677
+ else {
678
+ this.updateStates(this._notConnected);
679
+ }
680
+ }
681
+ }
682
+ const memoizedConnections = new Map();
683
+ function memoConnection(connectType, name, icon, identifier = '') {
684
+ const key = [connectType, name, icon, identifier].join(';');
685
+ if (memoizedConnections.has(key)) {
686
+ return memoizedConnections.get(key);
687
+ }
688
+ const connection = {
689
+ type: connectType,
690
+ name,
691
+ icon,
692
+ identifier,
693
+ };
694
+ memoizedConnections.set(key, connection);
695
+ return connection;
696
+ }
697
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controller.js","sourceRoot":"","sources":["../../../src/@ksangkuk10/wallet-controller/controller.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AccAddress,\n  CreateTxOptions,\n  LCDClient,\n  PublicKey,\n  SimplePublicKey,\n  Tx,\n  SignMode,\n} from '@xpla/xpla.js';\nimport {\n  ConnectedWallet,\n  Connection,\n  ConnectType,\n  Installation,\n  NetworkInfo,\n  SignBytesResult,\n  SignResult,\n  TxResult,\n  WalletApp,\n  WalletLCDClientConfig,\n  WalletStates,\n  WalletStatus,\n} from '@ksangkuk10/wallet-types';\nimport {\n  XplaWebExtensionFeatures,\n  WebExtensionTxStatus,\n} from '@ksangkuk10/web-extension-interface';\nimport deepEqual from 'fast-deep-equal';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  firstValueFrom,\n  Observable,\n  Subscription,\n} from 'rxjs';\nimport { filter, map } from 'rxjs/operators';\nimport {\n  CHROME_EXTENSION_INSTALL_URL,\n  DEFAULT_CHROME_EXTENSION_COMPATIBLE_BROWSER_CHECK,\n} from './env';\nimport {\n  mapExtensionSignBytesError,\n  mapExtensionTxError,\n} from './exception/mapExtensionTxError';\nimport { \n  mapWalletConnectError,\n  mapWalletConnectSignError,\n  mapWalletConnectSignBytesError,\n} from './exception/mapWalletConnectError';\nimport { selectConnection } from './modules/connect-modal';\nimport {\n  ExtensionRouter,\n  ExtensionRouterStatus,\n} from './modules/extension-router';\nimport {\n  ExtensionInfo,\n  getXplaExtensions,\n} from './modules/extension-router/multiChannel';\nimport {\n  connect as reConnect,\n  connectIfSessionExists as reConnectIfSessionExists,\n  ReadonlyWalletController,\n  readonlyWalletModal,\n  ReadonlyWalletSession,\n} from './modules/readonly-wallet';\nimport {\n  connect as wcConnect,\n  connectIfSessionExists as wcConnectIfSessionExists,\n  WalletConnectController,\n  WalletConnectControllerOptions,\n  WalletConnectSessionStatus,\n} from './modules/walletconnect';\nimport { getExtensions } from './operators/getExtensions';\nimport { toConnectedWallet } from './operators/toConnectedWallet';\nimport { toLcdClient } from './operators/toLcdClient';\nimport { isDesktopChrome } from './utils/browser-check';\nimport { checkExtensionReady } from './utils/checkExtensionReady';\nimport { sortConnections } from './utils/sortConnections';\n\nexport interface WalletControllerOptions\n  extends WalletConnectControllerOptions {\n  /**\n   * ⚠️ Don't hardcoding this, use getChain Options()\n   *\n   * fallback network if controller is not connected\n   */\n  defaultNetwork: NetworkInfo;\n\n  /**\n   * ⚠️ Don't hardcoding this, use getChain Options()\n   *\n   * for walletconnect\n   *\n   * The network rules passed by the Xpla Mobile are 0 is testnet, 1 is mainnet.\n   *\n   * Always set testnet for 0 and mainnet for 1.\n   *\n   * @example\n   * ```\n   * const mainnet: NetworkInfo = {\n   *  name: 'mainnet',\n   *  chainID: 'dimension-1',\n   *  lcd: 'https://lcd.xpla.net',\n   * }\n   *\n   * const testnet: NetworkInfo = {\n   *  name: 'testnet',\n   *  chainID: 'cube-1',\n   *  lcd: 'https://lcd.xpla.net',\n   * }\n   *\n   * const walletConnectChainIds: Record<number, NetworkInfo> = {\n   *   0: testnet,\n   *   1: mainnet,\n   * }\n   *\n   * <WalletProvider walletConnectChainIds={walletConnectChainIds}>\n   * ```\n   */\n  walletConnectChainIds: Record<number, NetworkInfo>;\n\n  /**\n   * run at executing the `connect(ConnectType.READONLY)`\n   */\n  createReadonlyWalletSession?: (\n    networks: NetworkInfo[],\n  ) => Promise<ReadonlyWalletSession | null>;\n\n  /**\n   * run at executing the `connect()` - only used when does not input ConnectType\n   */\n  selectConnection?: (\n    connections: Connection[],\n  ) => Promise<[type: ConnectType, identifier: string | undefined] | null>;\n\n  /**\n   * run at executing the `connect(ConnectType.EXTENSION)`\n   * if user installed multiple wallets\n   */\n  selectExtension?: (\n    extensionInfos: ExtensionInfo[],\n  ) => Promise<ExtensionInfo | null>;\n\n  /**\n   * milliseconds to wait checking chrome extension is installed\n   *\n   * @default 1000 * 3 miliseconds\n   */\n  waitingChromeExtensionInstallCheck?: number;\n\n  /**\n   * ⚠️ This API is an option for wallet developers. Please don't use dApp developers.\n   *\n   * @example\n   * ```\n   * <WalletProvider dangerously__chromeExtensionCompatibleBrowserCheck={(userAgent: string) => {\n   *   return /MyWallet\\//.test(userAgent);\n   * }}>\n   * ```\n   */\n  dangerously__chromeExtensionCompatibleBrowserCheck?: (\n    userAgent: string,\n  ) => boolean;\n}\n\nconst CONNECTIONS = {\n  [ConnectType.READONLY]: {\n    type: ConnectType.READONLY,\n    name: 'View an address',\n    icon: 'https://assets.xpla.io/icon/wallet-provider/readonly.svg',\n  } as Connection,\n  [ConnectType.WALLETCONNECT]: {\n    type: ConnectType.WALLETCONNECT,\n    name: 'Wallet Connect',\n    icon: 'https://assets.xpla.io/icon/wallet-provider/walletconnect.svg',\n  } as Connection,\n} as const;\n\nconst DEFAULT_WAITING_CHROME_EXTENSION_INSTALL_CHECK = 1000 * 3;\n\nconst WALLETCONNECT_SUPPORT_FEATURES = new Set<XplaWebExtensionFeatures>([\n  'post', 'sign', 'sign-bytes'\n]);\n\nconst EMPTY_SUPPORT_FEATURES = new Set<XplaWebExtensionFeatures>();\n\n//noinspection ES6MissingAwait\nexport class WalletController {\n  private extension: ExtensionRouter | null = null;\n  private walletConnect: WalletConnectController | null = null;\n  private readonlyWallet: ReadonlyWalletController | null = null;\n\n  private _availableConnectTypes: BehaviorSubject<ConnectType[]>;\n  private _availableInstallTypes: BehaviorSubject<ConnectType[]>;\n  private _states: BehaviorSubject<WalletStates>;\n\n  private disableReadonlyWallet: (() => void) | null = null;\n  private disableExtension: (() => void) | null = null;\n  private disableWalletConnect: (() => void) | null = null;\n\n  private readonly _notConnected: WalletStates;\n  private readonly _initializing: WalletStates;\n\n  constructor(readonly options: WalletControllerOptions) {\n    this._notConnected = {\n      status: WalletStatus.WALLET_NOT_CONNECTED,\n      network: options.defaultNetwork,\n    };\n\n    this._initializing = {\n      status: WalletStatus.INITIALIZING,\n      network: options.defaultNetwork,\n    };\n\n    this._availableConnectTypes = new BehaviorSubject<ConnectType[]>([\n      ConnectType.READONLY,\n      ConnectType.WALLETCONNECT,\n    ]);\n\n    this._availableInstallTypes = new BehaviorSubject<ConnectType[]>([]);\n\n    this._states = new BehaviorSubject<WalletStates>(this._initializing);\n\n    let numSessionCheck: number = 0;\n\n    // wait checking the availability of the chrome extension\n    // 0. check if extension wallet session is exists\n    checkExtensionReady(\n      options.waitingChromeExtensionInstallCheck ??\n        DEFAULT_WAITING_CHROME_EXTENSION_INSTALL_CHECK,\n      this.isChromeExtensionCompatibleBrowser(),\n    ).then((ready: boolean) => {\n      if (ready) {\n        this._availableConnectTypes.next([\n          ConnectType.EXTENSION,\n          ConnectType.WALLETCONNECT,\n          ConnectType.READONLY,\n        ]);\n\n        this.extension = new ExtensionRouter({\n          hostWindow: window,\n          selectExtension: options.selectExtension,\n          dangerously__chromeExtensionCompatibleBrowserCheck:\n            options.dangerously__chromeExtensionCompatibleBrowserCheck ??\n            DEFAULT_CHROME_EXTENSION_COMPATIBLE_BROWSER_CHECK,\n          defaultNetwork: options.defaultNetwork,\n        });\n\n        const subscription = this.extension\n          .states()\n          .pipe(\n            filter(({ type }) => type !== ExtensionRouterStatus.INITIALIZING),\n          )\n          .subscribe((extensionStates) => {\n            try {\n              subscription.unsubscribe();\n            } catch {}\n\n            if (\n              extensionStates.type === ExtensionRouterStatus.WALLET_CONNECTED &&\n              !this.disableWalletConnect &&\n              !this.disableReadonlyWallet\n            ) {\n              this.enableExtension();\n            } else if (numSessionCheck === 0) {\n              numSessionCheck += 1;\n            } else {\n              this.updateStates(this._notConnected);\n            }\n          });\n      } else {\n        if (isDesktopChrome(this.isChromeExtensionCompatibleBrowser())) {\n          this._availableInstallTypes.next([ConnectType.EXTENSION]);\n        }\n\n        if (numSessionCheck === 0) {\n          numSessionCheck += 1;\n        } else {\n          this.updateStates(this._notConnected);\n        }\n      }\n    });\n\n    // 1. check if readonly wallet session is exists\n    const draftReadonlyWallet = reConnectIfSessionExists();\n\n    if (draftReadonlyWallet) {\n      this.enableReadonlyWallet(draftReadonlyWallet);\n      return;\n    }\n\n    // 2. check if walletconnect sesison is exists\n    const draftWalletConnect = wcConnectIfSessionExists(options);\n\n    if (\n      draftWalletConnect &&\n      draftWalletConnect.getLatestSession().status ===\n        WalletConnectSessionStatus.CONNECTED\n    ) {\n      this.enableWalletConnect(draftWalletConnect);\n    } else if (numSessionCheck === 0) {\n      numSessionCheck += 1;\n    } else {\n      this.updateStates(this._notConnected);\n    }\n  }\n\n  /**\n   * Some mobile wallet emulates the behavior of chrome extension.\n   * It confirms that the current connection environment is such a wallet.\n   * (If you are running connect() by checking availableConnectType, you do not need to use this API.)\n   *\n   * @see Wallet#isChromeExtensionCompatibleBrowser\n   */\n  isChromeExtensionCompatibleBrowser = (): boolean => {\n    return (\n      this.options.dangerously__chromeExtensionCompatibleBrowserCheck ??\n      DEFAULT_CHROME_EXTENSION_COMPATIBLE_BROWSER_CHECK\n    )(navigator.userAgent);\n  };\n\n  /**\n   * available connect types on the browser\n   *\n   * @see Wallet#availableConnectTypes\n   */\n  availableConnectTypes = (): Observable<ConnectType[]> => {\n    return this._availableConnectTypes.asObservable();\n  };\n\n  /**\n   * available connections includes identifier, name, icon\n   *\n   * @see Wallet#availableConnections\n   */\n  availableConnections = (): Observable<Connection[]> => {\n    return this._availableConnectTypes.pipe(\n      map((connectTypes) => {\n        const connections: Connection[] = [];\n\n        for (const connectType of connectTypes) {\n          if (connectType === ConnectType.EXTENSION) {\n            const xplaExtensions = getXplaExtensions();\n\n            for (const xplaExtension of xplaExtensions) {\n              connections.push(\n                memoConnection(\n                  ConnectType.EXTENSION,\n                  xplaExtension.name,\n                  xplaExtension.icon,\n                  xplaExtension.identifier,\n                ),\n              );\n            }\n          } else {\n            connections.push(CONNECTIONS[connectType]);\n          }\n        }\n\n        return sortConnections(connections);\n      }),\n    );\n  };\n\n  /**\n   * available install types on the browser\n   *\n   * in this time, this only contains [ConnectType.EXTENSION]\n   *\n   * @see Wallet#availableInstallTypes\n   */\n  availableInstallTypes = (): Observable<ConnectType[]> => {\n    return this._availableInstallTypes.asObservable();\n  };\n\n  /**\n   * available installations includes identifier, name, icon, url\n   *\n   * @see Wallet#availableInstallations\n   */\n  availableInstallations = (): Observable<Installation[]> => {\n    return combineLatest([this.availableConnections(), getExtensions()]).pipe(\n      map(([connections, extensions]) => {\n        const installedIdentifiers = new Set<string>(\n          connections\n            .filter(({ type, identifier }) => {\n              return type === ConnectType.EXTENSION && !!identifier;\n            })\n            .map(({ identifier }) => {\n              return identifier!;\n            }),\n        );\n\n        return extensions\n          .filter(({ identifier }) => {\n            return !installedIdentifiers.has(identifier);\n          })\n          .map(({ name, identifier, icon, url }) => {\n            return {\n              type: ConnectType.EXTENSION,\n              identifier,\n              name,\n              icon,\n              url,\n            };\n          });\n      }),\n    );\n  };\n\n  /**\n   * @see Wallet#status\n   * @see Wallet#network\n   * @see Wallet#wallets\n   */\n  states = (): Observable<WalletStates> => {\n    return this._states.asObservable();\n  };\n\n  /** get connectedWallet */\n  connectedWallet = (): Observable<ConnectedWallet | undefined> => {\n    return this._states.pipe(toConnectedWallet(this));\n  };\n\n  /** get lcdClient */\n  lcdClient = (\n    lcdClientConfig?: WalletLCDClientConfig,\n  ): Observable<LCDClient> => {\n    return this._states.pipe(toLcdClient(lcdClientConfig));\n  };\n\n  /**\n   * reload the connected wallet states\n   *\n   * in this time, this only work on the ConnectType.EXTENSION\n   *\n   * @see Wallet#recheckStatus\n   */\n  refetchStates = () => {\n    if (this.disableExtension) {\n      this.extension?.refetchStates();\n    }\n  };\n\n  /**\n   * @deprecated Please use availableInstallations\n   *\n   * install for the connect type\n   *\n   * @see Wallet#install\n   */\n  install = (type: ConnectType) => {\n    if (type === ConnectType.EXTENSION) {\n      // TODO separate install links by browser types\n      window.open(CHROME_EXTENSION_INSTALL_URL, '_blank');\n    } else {\n      console.warn(\n        `[WalletController] ConnectType \"${type}\" does not support install() function`,\n      );\n    }\n  };\n\n  /**\n   * connect to wallet\n   *\n   * @see Wallet#connect\n   */\n  connect = async (_type?: ConnectType, _identifier?: string, _walletApp?: WalletApp | boolean) => {\n    let type: ConnectType;\n    let identifier: string | undefined;\n\n    if (!!_type) {\n      type = _type;\n      identifier = _identifier;\n    } else {\n      const connections = await firstValueFrom(this.availableConnections());\n      const selector = this.options.selectConnection ?? selectConnection;\n      const selected = await selector(connections);\n\n      if (!selected) {\n        return;\n      }\n\n      type = selected[0];\n      identifier = selected[1];\n    }\n\n    switch (type) {\n      case ConnectType.READONLY:\n        const networks: NetworkInfo[] = Object.keys(\n          this.options.walletConnectChainIds,\n        ).map((chainId) => this.options.walletConnectChainIds[+chainId]);\n\n        const createReadonlyWalletSession =\n          this.options.createReadonlyWalletSession?.(networks) ??\n          readonlyWalletModal({ networks });\n\n        const readonlyWalletSession = await createReadonlyWalletSession;\n\n        if (readonlyWalletSession) {\n          this.enableReadonlyWallet(reConnect(readonlyWalletSession));\n        }\n        break;\n      case ConnectType.WALLETCONNECT:\n        this.enableWalletConnect(wcConnect(this.options, false, _walletApp));\n        break;\n      case ConnectType.EXTENSION:\n        if (!this.extension) {\n          throw new Error(`extension instance is not created!`);\n        }\n\n        this.extension.connect(identifier);\n        this.enableExtension();\n        break;\n      default:\n        throw new Error(`Unknown ConnectType!`);\n    }\n  };\n\n  /**\n   * manual connect to read only session\n   *\n   * @see Wallet#connectReadonly\n   */\n  connectReadonly = (xplaAddress: string, network: NetworkInfo) => {\n    this.enableReadonlyWallet(\n      reConnect({\n        xplaAddress,\n        network,\n      }),\n    );\n  };\n\n  /** @see Wallet#disconnect */\n  disconnect = () => {\n    this.disableReadonlyWallet?.();\n    this.disableReadonlyWallet = null;\n\n    this.disableExtension?.();\n    this.disableExtension = null;\n\n    this.disableWalletConnect?.();\n    this.disableWalletConnect = null;\n\n    this.updateStates(this._notConnected);\n  };\n\n  /**\n   * @see Wallet#post\n   * @param tx\n   * @param xplaAddress only available new extension\n   */\n  post = async (\n    tx: CreateTxOptions,\n    xplaAddress?: string,\n    walletApp?: WalletApp | boolean,\n  ): Promise<TxResult> => {\n    // ---------------------------------------------\n    // extension\n    // ---------------------------------------------\n    if (this.disableExtension) {\n      return new Promise<TxResult>((resolve, reject) => {\n        if (!this.extension) {\n          reject(new Error(`extension instance not created!`));\n          return;\n        }\n\n        const subscription = this.extension.post(tx, xplaAddress).subscribe({\n          next: (txResult) => {\n            if (txResult.status === WebExtensionTxStatus.SUCCEED) {\n              resolve({\n                ...tx,\n                result: txResult.payload,\n                success: true,\n              });\n              subscription.unsubscribe();\n            }\n          },\n          error: (error) => {\n            reject(mapExtensionTxError(tx, error));\n            subscription.unsubscribe();\n          },\n        });\n      });\n    }\n    // ---------------------------------------------\n    // wallet connect\n    // ---------------------------------------------\n    else if (this.walletConnect) {\n      return this.walletConnect\n        .post(tx, walletApp)\n        .then(\n          (result) =>\n            ({\n              ...tx,\n              result,\n              success: true,\n            } as TxResult),\n        )\n        .catch((error) => {\n          throw mapWalletConnectError(tx, error);\n        });\n    } else {\n      throw new Error(`There are no connections that can be posting tx!`);\n    }\n  };\n\n  /**\n   * @see Wallet#sign\n   * @param tx\n   * @param xplaAddress only available new extension\n   */\n  sign = async (\n    tx: CreateTxOptions & {\n      sequence?: number;\n      accountNumber?: number;\n      signMode?: SignMode;\n    },\n    xplaAddress?: string,\n    walletApp?: WalletApp | boolean,\n  ): Promise<SignResult> => {\n    if (this.disableExtension) {\n      return new Promise<SignResult>((resolve, reject) => {\n        if (!this.extension) {\n          reject(new Error(`extension instance is not created!`));\n          return;\n        }\n\n        const subscription = this.extension.sign(tx, xplaAddress).subscribe({\n          next: (txResult) => {\n            if (txResult.status === WebExtensionTxStatus.SUCCEED) {\n              resolve({\n                ...tx,\n                result: Tx.fromData(txResult.payload),\n                success: true,\n              });\n              subscription.unsubscribe();\n            }\n          },\n          error: (error) => {\n            reject(mapExtensionTxError(tx, error));\n            subscription.unsubscribe();\n          },\n        });\n      });\n    }\n    // ---------------------------------------------\n    // wallet connect\n    // ---------------------------------------------\n    else if (this.walletConnect) {\n      if (walletApp && walletApp === WalletApp.XPLA_GAMES) {\n        throw new Error(`There are no connections that can be signing!`);\n      }\n\n      return this.walletConnect\n        .sign(tx, walletApp)\n        .then(\n          (result) => {\n            return {\n                ...tx,\n                result: Tx.fromData(result, false),\n                success: true,\n            }\n          }\n        )\n        .catch((error) => {\n          throw mapWalletConnectSignError(tx, error);\n        });\n    } else {\n      throw new Error(`There are no connections that can be signing!`);\n    }\n  };\n\n  /**\n   * @see Wallet#signBytes\n   * @param bytes\n   * @param xplaAddress only available new extension\n   */\n  signBytes = async (\n    bytes: Buffer,\n    xplaAddress?: string,\n    walletApp?: WalletApp | boolean,\n  ): Promise<SignBytesResult> => {\n    if (this.disableExtension) {\n      return new Promise<SignBytesResult>((resolve, reject) => {\n        if (!this.extension) {\n          reject(new Error(`extension instance is not created!`));\n          return;\n        }\n\n        const subscription = this.extension\n          .signBytes(bytes, xplaAddress)\n          .subscribe({\n            next: (txResult) => {\n              if (txResult.status === WebExtensionTxStatus.SUCCEED) {\n                resolve({\n                  result: {\n                    recid: txResult.payload.recid,\n                    signature: Uint8Array.from(\n                      Buffer.from(txResult.payload.signature, 'base64'),\n                    ),\n                    public_key: txResult.payload.public_key\n                      ? PublicKey.fromData(txResult.payload.public_key)\n                      : undefined,\n                  },\n                  success: true,\n                });\n                subscription.unsubscribe();\n              }\n            },\n            error: (error) => {\n              reject(mapExtensionSignBytesError(bytes, error));\n              subscription.unsubscribe();\n            },\n          });\n      });\n    }\n    // ---------------------------------------------\n    // wallet connect\n    // ---------------------------------------------\n    else if (this.walletConnect) {\n      if (walletApp && walletApp === WalletApp.XPLA_GAMES) {\n        throw new Error(`There are no connections that can be signing bytes!`);\n      }\n\n      return this.walletConnect\n        .signBytes(bytes, walletApp)\n        .then(\n          (result) => {\n            const key = new SimplePublicKey(String(result.public_key)).toData()\n            return {\n              result: {\n                recid: result.recid,\n                signature: Uint8Array.from(\n                  Buffer.from(result.signature, 'base64'),\n                ),\n                public_key: key\n                  ? PublicKey.fromData(key)\n                  : undefined,\n              },\n              success: true,\n            }\n          }\n        )\n        .catch((error) => {\n          throw mapWalletConnectSignBytesError(bytes, error);\n        });\n    } else {\n      throw new Error(`There are no connections that can be signing bytes!`);\n    }\n  };\n\n  /**\n   * @see Wallet#hasCW20Tokens\n   * @param chainID\n   * @param tokenAddrs Token addresses\n   */\n  hasCW20Tokens = async (\n    chainID: string,\n    ...tokenAddrs: string[]\n  ): Promise<{ [tokenAddr: string]: boolean }> => {\n    if (this.availableExtensionFeature('cw20-token')) {\n      return this.extension!.hasCW20Tokens(chainID, ...tokenAddrs);\n    }\n\n    throw new Error(`Does not support hasCW20Tokens() on this connection`);\n  };\n\n  /**\n   * @see Wallet#addCW20Tokens\n   * @param chainID\n   * @param tokenAddrs Token addresses\n   */\n  addCW20Tokens = async (\n    chainID: string,\n    ...tokenAddrs: string[]\n  ): Promise<{ [tokenAddr: string]: boolean }> => {\n    if (this.availableExtensionFeature('cw20-token')) {\n      return this.extension!.addCW20Tokens(chainID, ...tokenAddrs);\n    }\n\n    throw new Error(`Does not support addCW20Tokens() on this connection`);\n  };\n\n  /**\n   * @see Wallet#hasNetwork\n   * @param network\n   */\n  hasNetwork = (network: Omit<NetworkInfo, 'name'>): Promise<boolean> => {\n    if (this.availableExtensionFeature('network')) {\n      return this.extension!.hasNetwork(network);\n    }\n\n    throw new Error(`Does not support hasNetwork() on this connection`);\n  };\n\n  /**\n   * @see Wallet#hasNetwork\n   * @param network\n   */\n  addNetwork = (network: NetworkInfo): Promise<boolean> => {\n    if (this.availableExtensionFeature('network')) {\n      return this.extension!.addNetwork(network);\n    }\n\n    throw new Error(`Does not support addNetwork() on this connection`);\n  };\n\n  // ================================================================\n  // internal\n  // connect type changing\n  // ================================================================\n  private availableExtensionFeature = (feature: XplaWebExtensionFeatures) => {\n    if (this.disableExtension && this.extension) {\n      const states = this.extension.getLastStates();\n\n      return (\n        states.type === ExtensionRouterStatus.WALLET_CONNECTED &&\n        states.supportFeatures.has(feature)\n      );\n    }\n  };\n\n  private updateStates = (next: WalletStates) => {\n    const prev = this._states.getValue();\n\n    if (\n      next.status === WalletStatus.WALLET_CONNECTED &&\n      next.wallets.length === 0\n    ) {\n      next = {\n        status: WalletStatus.WALLET_NOT_CONNECTED,\n        network: next.network,\n      };\n    }\n\n    if (prev.status !== next.status || !deepEqual(prev, next)) {\n      this._states.next(next);\n    }\n  };\n\n  private enableReadonlyWallet = (readonlyWallet: ReadonlyWalletController) => {\n    this.disableWalletConnect?.();\n    this.disableExtension?.();\n\n    if (\n      this.readonlyWallet === readonlyWallet ||\n      (this.readonlyWallet?.xplaAddress === readonlyWallet.xplaAddress &&\n        this.readonlyWallet.network === readonlyWallet.network)\n    ) {\n      return;\n    }\n\n    if (this.readonlyWallet) {\n      this.readonlyWallet.disconnect();\n    }\n\n    this.readonlyWallet = readonlyWallet;\n\n    this.updateStates({\n      status: WalletStatus.WALLET_CONNECTED,\n      network: readonlyWallet.network,\n      wallets: [\n        {\n          connectType: ConnectType.READONLY,\n          xplaAddress: readonlyWallet.xplaAddress,\n          design: 'readonly',\n        },\n      ],\n      supportFeatures: EMPTY_SUPPORT_FEATURES,\n      connection: CONNECTIONS.READONLY,\n    });\n\n    this.disableReadonlyWallet = () => {\n      readonlyWallet.disconnect();\n      this.readonlyWallet = null;\n      this.disableReadonlyWallet = null;\n    };\n  };\n\n  private enableExtension = () => {\n    this.disableReadonlyWallet?.();\n    this.disableWalletConnect?.();\n\n    if (this.disableExtension || !this.extension) {\n      return;\n    }\n\n    const extensionSubscription = this.extension.states().subscribe({\n      next: (extensionStates) => {\n        if (\n          extensionStates.type === ExtensionRouterStatus.WALLET_CONNECTED &&\n          AccAddress.validate(extensionStates.wallet.xplaAddress)\n        ) {\n          this.updateStates({\n            status: WalletStatus.WALLET_CONNECTED,\n            network: extensionStates.network,\n            wallets: [\n              {\n                connectType: ConnectType.EXTENSION,\n                xplaAddress: extensionStates.wallet.xplaAddress,\n                design: extensionStates.wallet.design,\n              },\n            ],\n            supportFeatures: extensionStates.supportFeatures,\n            connection: memoConnection(\n              ConnectType.EXTENSION,\n              extensionStates.extensionInfo.name,\n              extensionStates.extensionInfo.icon,\n              extensionStates.extensionInfo.identifier,\n            ),\n          });\n        } else {\n          this.updateStates(this._notConnected);\n        }\n      },\n    });\n\n    this.disableExtension = () => {\n      this.extension?.disconnect();\n      extensionSubscription.unsubscribe();\n      this.disableExtension = null;\n    };\n  };\n\n  private enableWalletConnect = (walletConnect: WalletConnectController) => {\n    this.disableReadonlyWallet?.();\n    this.disableExtension?.();\n\n    if (this.walletConnect === walletConnect) {\n      return;\n    }\n\n    if (this.walletConnect) {\n      this.walletConnect.disconnect();\n    }\n\n    this.walletConnect = walletConnect;\n\n    const subscribeWalletConnect = (\n      wc: WalletConnectController,\n    ): Subscription => {\n      return wc.session().subscribe({\n        next: (status) => {\n          switch (status.status) {\n            case WalletConnectSessionStatus.CONNECTED:\n              this.updateStates({\n                status: WalletStatus.WALLET_CONNECTED,\n                network:\n                  this.options.walletConnectChainIds[status.chainId] ??\n                  this.options.defaultNetwork,\n                wallets: [\n                  {\n                    connectType: ConnectType.WALLETCONNECT,\n                    xplaAddress: status.xplaAddress,\n                    design: 'walletconnect',\n                  },\n                ],\n                supportFeatures: WALLETCONNECT_SUPPORT_FEATURES,\n                connection: CONNECTIONS.WALLETCONNECT,\n              });\n              break;\n            default:\n              this.updateStates(this._notConnected);\n              break;\n          }\n        },\n      });\n    };\n\n    const walletConnectSessionSubscription =\n      subscribeWalletConnect(walletConnect);\n\n    this.disableWalletConnect = () => {\n      this.walletConnect?.disconnect();\n      this.walletConnect = null;\n      walletConnectSessionSubscription.unsubscribe();\n      this.disableWalletConnect = null;\n    };\n  };\n}\n\nconst memoizedConnections = new Map<string, Connection>();\n\nfunction memoConnection(\n  connectType: ConnectType,\n  name: string,\n  icon: string,\n  identifier: string | undefined = '',\n): Connection {\n  const key = [connectType, name, icon, identifier].join(';');\n\n  if (memoizedConnections.has(key)) {\n    return memoizedConnections.get(key)!;\n  }\n\n  const connection: Connection = {\n    type: connectType,\n    name,\n    icon,\n    identifier,\n  };\n\n  memoizedConnections.set(key, connection);\n\n  return connection;\n}\n"]}