@cartridge/controller 0.7.7 → 0.7.8

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 (39) hide show
  1. package/.turbo/turbo-build$colon$deps.log +23 -23
  2. package/.turbo/turbo-build.log +31 -31
  3. package/dist/controller.cjs +472 -3
  4. package/dist/controller.cjs.map +1 -1
  5. package/dist/controller.d.cts +1 -1
  6. package/dist/controller.d.ts +1 -1
  7. package/dist/controller.js +472 -3
  8. package/dist/controller.js.map +1 -1
  9. package/dist/index.cjs +1908 -1214
  10. package/dist/index.cjs.map +1 -1
  11. package/dist/index.d.cts +46 -3
  12. package/dist/index.d.ts +46 -3
  13. package/dist/index.js +1905 -1215
  14. package/dist/index.js.map +1 -1
  15. package/dist/node/index.cjs +1 -1
  16. package/dist/node/index.cjs.map +1 -1
  17. package/dist/node/index.d.cts +2 -2
  18. package/dist/node/index.d.ts +2 -2
  19. package/dist/node/index.js +1 -1
  20. package/dist/node/index.js.map +1 -1
  21. package/dist/{provider-CTDncg8U.d.cts → provider-BeCgS86X.d.cts} +64 -1
  22. package/dist/{provider-CTDncg8U.d.ts → provider-BeCgS86X.d.ts} +64 -1
  23. package/dist/session/index.cjs +1 -1
  24. package/dist/session/index.cjs.map +1 -1
  25. package/dist/session/index.d.cts +2 -2
  26. package/dist/session/index.d.ts +2 -2
  27. package/dist/session/index.js +1 -1
  28. package/dist/session/index.js.map +1 -1
  29. package/package.json +8 -5
  30. package/src/iframe/base.ts +1 -1
  31. package/src/iframe/keychain.ts +11 -0
  32. package/src/index.ts +1 -0
  33. package/src/types.ts +24 -0
  34. package/src/wallets/argent/index.ts +122 -0
  35. package/src/wallets/bridge.ts +165 -0
  36. package/src/wallets/index.ts +5 -0
  37. package/src/wallets/metamask/index.ts +210 -0
  38. package/src/wallets/phantom/index.ts +124 -0
  39. package/src/wallets/types.ts +32 -0
@@ -1,5 +1,8 @@
1
1
  import { shortString, WalletAccount, constants, Provider } from 'starknet';
2
2
  import { connectToChild } from '@cartridge/penpal';
3
+ import { MetaMaskSDK } from '@metamask/sdk';
4
+ import { connect } from 'starknetkit';
5
+ import { InjectedConnector } from 'starknetkit/injected';
3
6
  import { Permission } from '@starknet-io/types-js';
4
7
 
5
8
  // src/account.ts
@@ -174,7 +177,7 @@ var IFrame = class {
174
177
  this.container = container;
175
178
  connectToChild({
176
179
  iframe: this.iframe,
177
- methods: { close: () => this.close(), ...methods }
180
+ methods: { close: (_origin) => () => this.close(), ...methods }
178
181
  }).promise.then(onConnect);
179
182
  this.resize();
180
183
  window.addEventListener("resize", () => this.resize());
@@ -238,11 +241,472 @@ var IFrame = class {
238
241
  // src/constants.ts
239
242
  var KEYCHAIN_URL = "https://x.cartridge.gg";
240
243
  var PROFILE_URL = "https://profile.cartridge.gg";
244
+ var MetaMaskWallet = class {
245
+ type = "metamask";
246
+ platform = "ethereum";
247
+ MMSDK;
248
+ account = void 0;
249
+ constructor() {
250
+ this.MMSDK = new MetaMaskSDK({
251
+ dappMetadata: {
252
+ name: "Cartridge Controller",
253
+ url: window.location.href
254
+ }
255
+ });
256
+ }
257
+ isAvailable() {
258
+ return typeof window !== "undefined" && !!window.ethereum?.isMetaMask;
259
+ }
260
+ getInfo() {
261
+ const available = this.isAvailable();
262
+ return {
263
+ type: this.type,
264
+ available,
265
+ version: available ? window.ethereum?.version || "Unknown" : void 0,
266
+ chainId: available ? window.ethereum?.chainId : void 0,
267
+ name: "MetaMask",
268
+ platform: this.platform
269
+ };
270
+ }
271
+ async connect() {
272
+ if (this.account) {
273
+ return { success: true, wallet: this.type, account: this.account };
274
+ }
275
+ try {
276
+ if (!this.isAvailable()) {
277
+ throw new Error("MetaMask is not available");
278
+ }
279
+ const accounts = await this.MMSDK.connect();
280
+ if (accounts && accounts.length > 0) {
281
+ this.account = accounts[0];
282
+ return { success: true, wallet: this.type, account: this.account };
283
+ }
284
+ throw new Error("No accounts found");
285
+ } catch (error) {
286
+ console.error(`Error connecting to MetaMask:`, error);
287
+ return {
288
+ success: false,
289
+ wallet: this.type,
290
+ error: error.message || "Unknown error"
291
+ };
292
+ }
293
+ }
294
+ async signTransaction(transaction) {
295
+ try {
296
+ if (!this.isAvailable() || !this.account) {
297
+ throw new Error("MetaMask is not connected");
298
+ }
299
+ const ethereum = this.MMSDK.getProvider();
300
+ if (!ethereum) {
301
+ throw new Error("MetaMask is not connected");
302
+ }
303
+ const result = await ethereum.request({
304
+ method: "eth_sendTransaction",
305
+ params: [transaction]
306
+ });
307
+ return { success: true, wallet: this.type, result };
308
+ } catch (error) {
309
+ console.error(`Error signing transaction with MetaMask:`, error);
310
+ return {
311
+ success: false,
312
+ wallet: this.type,
313
+ error: error.message || "Unknown error"
314
+ };
315
+ }
316
+ }
317
+ async signMessage(message) {
318
+ try {
319
+ if (!this.isAvailable() || !this.account) {
320
+ throw new Error("MetaMask is not connected");
321
+ }
322
+ const result = await this.MMSDK.connectAndSign({
323
+ msg: message
324
+ });
325
+ return { success: true, wallet: this.type, result };
326
+ } catch (error) {
327
+ console.error(`Error signing message with MetaMask:`, error);
328
+ return {
329
+ success: false,
330
+ wallet: this.type,
331
+ error: error.message || "Unknown error"
332
+ };
333
+ }
334
+ }
335
+ async signTypedData(data) {
336
+ try {
337
+ if (!this.isAvailable() || !this.account) {
338
+ throw new Error("MetaMask is not connected");
339
+ }
340
+ const ethereum = this.MMSDK.getProvider();
341
+ if (!ethereum) {
342
+ throw new Error("MetaMask is not connected");
343
+ }
344
+ const result = await ethereum.request({
345
+ method: "eth_signTypedData_v4",
346
+ params: [this.account, JSON.stringify(data)]
347
+ });
348
+ return { success: true, wallet: this.type, result };
349
+ } catch (error) {
350
+ console.error(`Error signing typed data with MetaMask:`, error);
351
+ return {
352
+ success: false,
353
+ wallet: this.type,
354
+ error: error.message || "Unknown error"
355
+ };
356
+ }
357
+ }
358
+ async switchChain(chainId) {
359
+ try {
360
+ if (!this.isAvailable()) {
361
+ throw new Error("MetaMask is not available");
362
+ }
363
+ const ethereum = this.MMSDK.getProvider();
364
+ if (!ethereum) {
365
+ throw new Error("MetaMask is not connected");
366
+ }
367
+ try {
368
+ await ethereum.request({
369
+ method: "wallet_switchEthereumChain",
370
+ params: [{ chainId }]
371
+ });
372
+ return true;
373
+ } catch (error) {
374
+ if (error.code === 4902) {
375
+ console.warn("Chain not added to MetaMask");
376
+ }
377
+ throw error;
378
+ }
379
+ } catch (error) {
380
+ console.error(`Error switching chain for MetaMask:`, error);
381
+ return false;
382
+ }
383
+ }
384
+ async getBalance(tokenAddress) {
385
+ try {
386
+ if (!this.isAvailable() || !this.account) {
387
+ throw new Error("MetaMask is not connected");
388
+ }
389
+ if (tokenAddress) {
390
+ return {
391
+ success: false,
392
+ wallet: this.type,
393
+ error: "Not implemented for ERC20"
394
+ };
395
+ } else {
396
+ const ethereum = this.MMSDK.getProvider();
397
+ if (!ethereum) {
398
+ throw new Error("MetaMask is not connected");
399
+ }
400
+ const balance = await ethereum.request({
401
+ method: "eth_getBalance",
402
+ params: [this.account, "latest"]
403
+ });
404
+ return { success: true, wallet: this.type, result: balance };
405
+ }
406
+ } catch (error) {
407
+ console.error(`Error getting balance from MetaMask:`, error);
408
+ return {
409
+ success: false,
410
+ wallet: this.type,
411
+ error: error.message || "Unknown error"
412
+ };
413
+ }
414
+ }
415
+ };
416
+
417
+ // src/wallets/phantom/index.ts
418
+ var PhantomWallet = class {
419
+ type = "phantom";
420
+ platform = "solana";
421
+ account = void 0;
422
+ isAvailable() {
423
+ return typeof window !== "undefined" && !!window.solana?.isPhantom;
424
+ }
425
+ getInfo() {
426
+ const available = this.isAvailable();
427
+ return {
428
+ type: this.type,
429
+ available,
430
+ version: "Unknown",
431
+ name: "Phantom",
432
+ platform: this.platform
433
+ };
434
+ }
435
+ async connect() {
436
+ if (this.account) {
437
+ return { success: true, wallet: this.type, account: this.account };
438
+ }
439
+ try {
440
+ if (!this.isAvailable()) {
441
+ throw new Error("Phantom is not available");
442
+ }
443
+ const response = await window.solana.connect();
444
+ if (response.publicKey) {
445
+ this.account = response.publicKey.toString();
446
+ return { success: true, wallet: this.type, account: this.account };
447
+ }
448
+ throw new Error("No accounts found");
449
+ } catch (error) {
450
+ console.error(`Error connecting to Phantom:`, error);
451
+ return {
452
+ success: false,
453
+ wallet: this.type,
454
+ error: error.message || "Unknown error"
455
+ };
456
+ }
457
+ }
458
+ async signTransaction(transaction) {
459
+ try {
460
+ if (!this.isAvailable() || !this.account) {
461
+ throw new Error("Phantom is not connected");
462
+ }
463
+ const result = await window.solana.signTransaction(transaction);
464
+ return { success: true, wallet: this.type, result };
465
+ } catch (error) {
466
+ console.error(`Error signing transaction with Phantom:`, error);
467
+ return {
468
+ success: false,
469
+ wallet: this.type,
470
+ error: error.message || "Unknown error"
471
+ };
472
+ }
473
+ }
474
+ async signMessage(message) {
475
+ try {
476
+ if (!this.isAvailable() || !this.account) {
477
+ throw new Error("Phantom is not connected");
478
+ }
479
+ const encodedMessage = new TextEncoder().encode(message);
480
+ const result = await window.solana.signMessage(encodedMessage, "utf8");
481
+ return { success: true, wallet: this.type, result };
482
+ } catch (error) {
483
+ console.error(`Error signing message with Phantom:`, error);
484
+ return {
485
+ success: false,
486
+ wallet: this.type,
487
+ error: error.message || "Unknown error"
488
+ };
489
+ }
490
+ }
491
+ async switchChain(_chainId) {
492
+ console.warn("Chain switching not supported for Phantom");
493
+ return false;
494
+ }
495
+ async getBalance(_tokenAddress) {
496
+ try {
497
+ if (!this.isAvailable() || !this.account) {
498
+ throw new Error("Phantom is not connected");
499
+ }
500
+ return {
501
+ success: true,
502
+ wallet: this.type,
503
+ result: "Implement based on Phantom API"
504
+ };
505
+ } catch (error) {
506
+ console.error(`Error getting balance from Phantom:`, error);
507
+ return {
508
+ success: false,
509
+ wallet: this.type,
510
+ error: error.message || "Unknown error"
511
+ };
512
+ }
513
+ }
514
+ };
515
+ var ArgentWallet = class {
516
+ type = "argent";
517
+ platform = "starknet";
518
+ wallet = void 0;
519
+ account = void 0;
520
+ isAvailable() {
521
+ return typeof window !== "undefined" && !!window.starknet_argentX;
522
+ }
523
+ getInfo() {
524
+ const available = this.isAvailable();
525
+ return {
526
+ type: this.type,
527
+ available,
528
+ version: available ? window.starknet_argentX?.version || "Unknown" : void 0,
529
+ chainId: available ? window.starknet_argentX?.chainId : void 0,
530
+ name: "Argent",
531
+ platform: this.platform
532
+ };
533
+ }
534
+ async connect() {
535
+ if (this.account) {
536
+ return { success: true, wallet: this.type, account: this.account };
537
+ }
538
+ try {
539
+ if (!this.isAvailable()) {
540
+ throw new Error("Argent is not available");
541
+ }
542
+ const { wallet, connectorData } = await connect({
543
+ connectors: [new InjectedConnector({ options: { id: "argentX" } })]
544
+ });
545
+ if (!wallet) {
546
+ throw new Error("No wallet found");
547
+ }
548
+ this.wallet = wallet;
549
+ this.account = connectorData?.account;
550
+ return { success: true, wallet: this.type, account: this.account };
551
+ } catch (error) {
552
+ console.error(`Error connecting to Argent:`, error);
553
+ return {
554
+ success: false,
555
+ wallet: this.type,
556
+ error: error.message || "Unknown error"
557
+ };
558
+ }
559
+ }
560
+ async signTypedData(data) {
561
+ try {
562
+ if (!this.isAvailable() || !this.wallet) {
563
+ throw new Error("Argent is not connected");
564
+ }
565
+ console.log("signTypedData", data);
566
+ const sig = await this.wallet.request({
567
+ type: "wallet_signTypedData",
568
+ params: data
569
+ });
570
+ return { success: true, wallet: this.type, result: sig };
571
+ } catch (error) {
572
+ console.error(`Error signing typed data with Argent:`, error);
573
+ return {
574
+ success: false,
575
+ wallet: this.type,
576
+ error: error.message || "Unknown error"
577
+ };
578
+ }
579
+ }
580
+ async switchChain(_chainId) {
581
+ console.warn(
582
+ "Chain switching for Argent may require custom implementation"
583
+ );
584
+ return false;
585
+ }
586
+ async getBalance(_tokenAddress) {
587
+ try {
588
+ if (!this.isAvailable() || !this.wallet) {
589
+ throw new Error("Argent is not connected");
590
+ }
591
+ return {
592
+ success: true,
593
+ wallet: this.type,
594
+ result: "Implement based on Argent API"
595
+ };
596
+ } catch (error) {
597
+ console.error(`Error getting balance from Argent:`, error);
598
+ return {
599
+ success: false,
600
+ wallet: this.type,
601
+ error: error.message || "Unknown error"
602
+ };
603
+ }
604
+ }
605
+ };
606
+
607
+ // src/wallets/bridge.ts
608
+ var WalletBridge = class {
609
+ walletAdapters;
610
+ connectedWallets = /* @__PURE__ */ new Map();
611
+ constructor() {
612
+ this.walletAdapters = /* @__PURE__ */ new Map();
613
+ this.walletAdapters.set("metamask", new MetaMaskWallet());
614
+ this.walletAdapters.set("phantom", new PhantomWallet());
615
+ this.walletAdapters.set("argent", new ArgentWallet());
616
+ if (typeof window !== "undefined") {
617
+ window.wallet_bridge = this;
618
+ }
619
+ }
620
+ getIFrameMethods() {
621
+ return {
622
+ externalDetectWallets: (_origin) => () => this.detectWallets(),
623
+ externalConnectWallet: (_origin) => (type) => this.connectWallet(type),
624
+ externalSignMessage: (_origin) => (type, message) => this.signMessage(type, message),
625
+ externalSignTypedData: (_origin) => (type, data) => this.signTypedData(type, data),
626
+ externalGetBalance: (_origin) => (type, tokenAddress) => this.getBalance(type, tokenAddress)
627
+ };
628
+ }
629
+ async detectWallets() {
630
+ const wallets = Array.from(this.walletAdapters.values()).map(
631
+ (adapter) => adapter.getInfo()
632
+ );
633
+ return wallets;
634
+ }
635
+ getWalletAdapter(type) {
636
+ const adapter = this.walletAdapters.get(type);
637
+ if (!adapter) {
638
+ throw new Error(`Unsupported wallet type: ${type}`);
639
+ }
640
+ return adapter;
641
+ }
642
+ handleError(type, error, operation) {
643
+ const errorMessage = error instanceof Error ? error.message : "Unknown error";
644
+ console.error(`Error ${operation} with ${type} wallet:`, error);
645
+ return { success: false, wallet: type, error: errorMessage };
646
+ }
647
+ async connectWallet(type) {
648
+ try {
649
+ if (this.connectedWallets.has(type)) {
650
+ const wallet2 = this.connectedWallets.get(type);
651
+ return { success: true, wallet: type, account: wallet2.type };
652
+ }
653
+ const wallet = this.getWalletAdapter(type);
654
+ const response = await wallet.connect();
655
+ if (response.success) {
656
+ this.connectedWallets.set(type, wallet);
657
+ }
658
+ return response;
659
+ } catch (error) {
660
+ return this.handleError(type, error, "connecting to");
661
+ }
662
+ }
663
+ async signMessage(type, message) {
664
+ try {
665
+ if (!this.connectedWallets.has(type)) {
666
+ throw new Error(`Wallet ${type} is not connected`);
667
+ }
668
+ const wallet = this.connectedWallets.get(type);
669
+ if (!wallet.signMessage) {
670
+ throw new Error(`Wallet ${type} does not support signing messages`);
671
+ }
672
+ return await wallet.signMessage(message);
673
+ } catch (error) {
674
+ return this.handleError(type, error, "signing message with");
675
+ }
676
+ }
677
+ async signTypedData(type, data) {
678
+ try {
679
+ if (!this.connectedWallets.has(type)) {
680
+ throw new Error(`Wallet ${type} is not connected`);
681
+ }
682
+ const wallet = this.connectedWallets.get(type);
683
+ if (!wallet.signTypedData) {
684
+ throw new Error(`Wallet ${type} does not support signing typed data`);
685
+ }
686
+ return await wallet.signTypedData(data);
687
+ } catch (error) {
688
+ return this.handleError(type, error, "signing typed data with");
689
+ }
690
+ }
691
+ async getBalance(type, tokenAddress) {
692
+ try {
693
+ if (!this.connectedWallets.has(type)) {
694
+ throw new Error(`Wallet ${type} is not connected`);
695
+ }
696
+ const wallet = this.connectedWallets.get(type);
697
+ return await wallet.getBalance(tokenAddress);
698
+ } catch (error) {
699
+ return this.handleError(type, error, "getting balance from");
700
+ }
701
+ }
702
+ };
241
703
 
242
704
  // src/iframe/keychain.ts
243
705
  var KeychainIFrame = class extends IFrame {
706
+ walletBridge;
244
707
  constructor({ url, policies, ...iframeOptions }) {
245
708
  const _url = new URL(url ?? KEYCHAIN_URL);
709
+ const walletBridge = new WalletBridge();
246
710
  if (policies) {
247
711
  _url.searchParams.set(
248
712
  "policies",
@@ -252,8 +716,13 @@ var KeychainIFrame = class extends IFrame {
252
716
  super({
253
717
  ...iframeOptions,
254
718
  id: "controller-keychain",
255
- url: _url
719
+ url: _url,
720
+ methods: walletBridge.getIFrameMethods()
256
721
  });
722
+ this.walletBridge = walletBridge;
723
+ }
724
+ getWalletBridge() {
725
+ return this.walletBridge;
257
726
  }
258
727
  };
259
728
 
@@ -315,7 +784,7 @@ var NotReadyToConnect = class _NotReadyToConnect extends Error {
315
784
 
316
785
  // package.json
317
786
  var package_default = {
318
- version: "0.7.7"};
787
+ version: "0.7.8"};
319
788
 
320
789
  // src/icon.ts
321
790
  var icon = "";