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