@solana/connector 0.1.9 → 0.2.0

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 (62) hide show
  1. package/README.md +403 -50
  2. package/dist/chunk-4KD6HQQG.js +69 -0
  3. package/dist/chunk-4KD6HQQG.js.map +1 -0
  4. package/dist/chunk-BJAVJQLK.js +230 -0
  5. package/dist/chunk-BJAVJQLK.js.map +1 -0
  6. package/dist/{chunk-5HRJKCIL.js → chunk-BZ2VBJCZ.js} +1061 -424
  7. package/dist/chunk-BZ2VBJCZ.js.map +1 -0
  8. package/dist/{chunk-WDXEP4AJ.js → chunk-EM4KNOKG.js} +658 -190
  9. package/dist/chunk-EM4KNOKG.js.map +1 -0
  10. package/dist/chunk-HN5AJF7F.js +507 -0
  11. package/dist/chunk-HN5AJF7F.js.map +1 -0
  12. package/dist/chunk-HO6QNKFM.mjs +61 -0
  13. package/dist/chunk-HO6QNKFM.mjs.map +1 -0
  14. package/dist/chunk-HPQ5T32K.mjs +178 -0
  15. package/dist/chunk-HPQ5T32K.mjs.map +1 -0
  16. package/dist/{chunk-MAXA3HEP.mjs → chunk-IDTUFDNB.mjs} +962 -344
  17. package/dist/chunk-IDTUFDNB.mjs.map +1 -0
  18. package/dist/{chunk-P5LXUDP6.mjs → chunk-RTXUS5KG.mjs} +579 -119
  19. package/dist/chunk-RTXUS5KG.mjs.map +1 -0
  20. package/dist/{chunk-DSUCH44G.js → chunk-SITQ4JWM.js} +23 -67
  21. package/dist/chunk-SITQ4JWM.js.map +1 -0
  22. package/dist/chunk-UCISIAOG.mjs +501 -0
  23. package/dist/chunk-UCISIAOG.mjs.map +1 -0
  24. package/dist/{chunk-J7DHGLW6.mjs → chunk-ZZTY3O4N.mjs} +21 -61
  25. package/dist/chunk-ZZTY3O4N.mjs.map +1 -0
  26. package/dist/compat.d.mts +1 -1
  27. package/dist/compat.d.ts +1 -1
  28. package/dist/compat.js +10 -9
  29. package/dist/compat.js.map +1 -1
  30. package/dist/compat.mjs +2 -1
  31. package/dist/compat.mjs.map +1 -1
  32. package/dist/headless.d.mts +239 -104
  33. package/dist/headless.d.ts +239 -104
  34. package/dist/headless.js +255 -169
  35. package/dist/headless.mjs +5 -3
  36. package/dist/index.d.mts +4 -4
  37. package/dist/index.d.ts +4 -4
  38. package/dist/index.js +316 -206
  39. package/dist/index.mjs +6 -4
  40. package/dist/react.d.mts +299 -9
  41. package/dist/react.d.ts +299 -9
  42. package/dist/react.js +90 -38
  43. package/dist/react.mjs +2 -2
  44. package/dist/{standard-shim-CT49DM5l.d.mts → standard-shim-CGB88PPO.d.mts} +673 -52
  45. package/dist/{standard-shim-D9guL5fz.d.ts → standard-shim-tmnQelaJ.d.ts} +673 -52
  46. package/dist/{transaction-signer-T-KVQFi8.d.mts → transaction-signer-7NaYmP5w.d.mts} +1 -0
  47. package/dist/{transaction-signer-T-KVQFi8.d.ts → transaction-signer-7NaYmP5w.d.ts} +1 -0
  48. package/dist/walletconnect-447EY3OJ.js +28 -0
  49. package/dist/walletconnect-447EY3OJ.js.map +1 -0
  50. package/dist/walletconnect-U455PO4I.mjs +3 -0
  51. package/dist/walletconnect-U455PO4I.mjs.map +1 -0
  52. package/package.json +6 -2
  53. package/dist/chunk-5HRJKCIL.js.map +0 -1
  54. package/dist/chunk-DSUCH44G.js.map +0 -1
  55. package/dist/chunk-I6TJLYNA.js +0 -535
  56. package/dist/chunk-I6TJLYNA.js.map +0 -1
  57. package/dist/chunk-J7DHGLW6.mjs.map +0 -1
  58. package/dist/chunk-JOBLG62A.mjs +0 -476
  59. package/dist/chunk-JOBLG62A.mjs.map +0 -1
  60. package/dist/chunk-MAXA3HEP.mjs.map +0 -1
  61. package/dist/chunk-P5LXUDP6.mjs.map +0 -1
  62. package/dist/chunk-WDXEP4AJ.js.map +0 -1
@@ -1,11 +1,55 @@
1
- import { createLogger, getPublicSolanaRpcUrl, getExplorerLink, __publicField, tryCatchSync, TransactionError, isWeb3jsTransaction, Errors, prepareTransactionForWallet, convertSignedTransaction, ValidationError, isConnectorError } from './chunk-J7DHGLW6.mjs';
2
- import { Component, useTransition, useState, useCallback, useMemo } from 'react';
3
- import { jsx, jsxs } from 'react/jsx-runtime';
1
+ import { tryCatchSync, isWeb3jsTransaction, prepareTransactionForWallet, convertSignedTransaction } from './chunk-HO6QNKFM.mjs';
2
+ import { createLogger, getPublicSolanaRpcUrl, getExplorerLink, __publicField, TransactionError, Errors, ValidationError } from './chunk-ZZTY3O4N.mjs';
4
3
  import { install } from '@solana/webcrypto-ed25519-polyfill';
5
- import { address, isAddress } from '@solana/addresses';
4
+ import { Storage, createSolanaMainnet, createSolanaDevnet, createSolanaTestnet, createSolanaLocalnet } from '@wallet-ui/core';
5
+ import { isAddress, address } from '@solana/addresses';
6
+ import { z } from 'zod/v4';
6
7
  import { getBase58Decoder } from '@solana/codecs';
7
8
  import { getTransactionDecoder, assertIsTransactionWithinSizeLimit } from '@solana/transactions';
8
9
 
10
+ // src/types/session.ts
11
+ function createConnectorId(walletName) {
12
+ return `wallet-standard:${walletName.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/^-+|-+$/g, "")}`;
13
+ }
14
+ function isWalletConnectorId(value) {
15
+ return typeof value == "string" && (value.startsWith("wallet-standard:") || value === "walletconnect" || value.startsWith("mwa:"));
16
+ }
17
+ function getWalletNameFromConnectorId(connectorId) {
18
+ return connectorId === "walletconnect" ? "WalletConnect" : connectorId.startsWith("mwa:") ? connectorId.slice(4) : connectorId.startsWith("wallet-standard:") ? connectorId.slice(16).split("-").map((word) => word.charAt(0).toUpperCase() + word.slice(1)).join(" ") : connectorId;
19
+ }
20
+ function isDisconnected(status) {
21
+ return status.status === "disconnected";
22
+ }
23
+ function isConnecting(status) {
24
+ return status.status === "connecting";
25
+ }
26
+ function isConnected(status) {
27
+ return status.status === "connected";
28
+ }
29
+ function isStatusError(status) {
30
+ return status.status === "error";
31
+ }
32
+ var isWalletStatusError = isStatusError, INITIAL_WALLET_STATUS = {
33
+ status: "disconnected"
34
+ };
35
+ function toLegacyWalletState(wallet) {
36
+ switch (wallet.status) {
37
+ case "disconnected":
38
+ return { connected: false, connecting: false, selectedAccount: null, accounts: [] };
39
+ case "connecting":
40
+ return { connected: false, connecting: true, selectedAccount: null, accounts: [] };
41
+ case "connected":
42
+ return {
43
+ connected: true,
44
+ connecting: false,
45
+ selectedAccount: wallet.session.selectedAccount.address,
46
+ accounts: wallet.session.accounts.map((a) => ({ address: a.address, label: a.label }))
47
+ };
48
+ case "error":
49
+ return { connected: false, connecting: false, selectedAccount: null, accounts: [] };
50
+ }
51
+ }
52
+
9
53
  // src/lib/wallet/standard-shim.ts
10
54
  var registry = null, registryInitPromise = null, registryInitResolve = null, ready = new Promise((resolve, reject) => {
11
55
  if (typeof window > "u") {
@@ -620,8 +664,43 @@ var logger2 = createLogger("WalletAuthenticity"), WalletAuthenticityVerifier = c
620
664
  }
621
665
  };
622
666
 
667
+ // src/lib/wallet/wallet-icon-overrides.ts
668
+ var PHANTOM_ICON = "", SOLFLARE_ICON = "", BACKPACK_ICON = "", WALLET_ICON_OVERRIDES = {
669
+ Phantom: PHANTOM_ICON,
670
+ Solflare: SOLFLARE_ICON,
671
+ Backpack: BACKPACK_ICON
672
+ }, ICON_PROXY_CACHE = /* @__PURE__ */ new WeakMap();
673
+ function getWalletIconOverride(walletName) {
674
+ return WALLET_ICON_OVERRIDES[walletName];
675
+ }
676
+ function createIconProxy(wallet, icon) {
677
+ let cached = ICON_PROXY_CACHE.get(wallet);
678
+ if (cached) return cached;
679
+ let proxy = new Proxy(wallet, {
680
+ get(target, prop) {
681
+ if (prop === "icon") return icon;
682
+ let value = Reflect.get(target, prop, target);
683
+ return typeof value == "function" ? value.bind(target) : value;
684
+ }
685
+ });
686
+ return ICON_PROXY_CACHE.set(wallet, proxy), proxy;
687
+ }
688
+ function applyWalletIconOverride(wallet) {
689
+ let override = getWalletIconOverride(wallet.name);
690
+ if (!override || wallet.icon === override) return wallet;
691
+ if (Object.isExtensible(wallet))
692
+ try {
693
+ return wallet.icon = override, wallet;
694
+ } catch {
695
+ }
696
+ return createIconProxy(wallet, override);
697
+ }
698
+
623
699
  // src/lib/wallet/detector.ts
624
700
  var logger3 = createLogger("WalletDetector");
701
+ function isSolanaWallet(wallet) {
702
+ return Array.isArray(wallet.chains) && wallet.chains.some((chain) => typeof chain == "string" && chain.startsWith("solana:"));
703
+ }
625
704
  function hasFeature(wallet, featureName) {
626
705
  return wallet.features != null && wallet.features[featureName] !== void 0;
627
706
  }
@@ -644,6 +723,8 @@ var WalletDetector = class extends BaseCollaborator {
644
723
  constructor(stateManager, eventEmitter, debug = false) {
645
724
  super({ stateManager, eventEmitter, debug }, "WalletDetector");
646
725
  __publicField(this, "unsubscribers", []);
726
+ /** Map from stable connector ID to Wallet reference (not stored in state) */
727
+ __publicField(this, "connectorRegistry", /* @__PURE__ */ new Map());
647
728
  }
648
729
  /**
649
730
  * Initialize wallet detection (synchronous)
@@ -659,11 +740,10 @@ var WalletDetector = class extends BaseCollaborator {
659
740
  if (!(typeof window > "u"))
660
741
  try {
661
742
  let walletsApi = getWalletsRegistry(), update = () => {
662
- let ws = walletsApi.get(), previousCount = this.getState().wallets.length, newCount = ws.length;
663
- newCount !== previousCount && this.log("\u{1F50D} WalletDetector: found wallets:", newCount);
664
- let unique = this.deduplicateWallets(ws);
665
- this.stateManager.updateState({
666
- wallets: unique.map((w) => this.mapToWalletInfo(w))
743
+ let solanaWallets = walletsApi.get().filter(isSolanaWallet), unique = this.deduplicateWallets(solanaWallets), previousCount = this.getState().wallets.length, newCount = unique.length;
744
+ newCount !== previousCount && this.log("\u{1F50D} WalletDetector: found wallets:", newCount), this.updateConnectorRegistry(unique), this.stateManager.updateState({
745
+ wallets: unique.map((w) => this.mapToWalletInfo(w)),
746
+ connectors: unique.map((w) => this.mapToConnectorMetadata(w))
667
747
  }), newCount !== previousCount && newCount > 0 && this.eventEmitter.emit({
668
748
  type: "wallets:detected",
669
749
  count: newCount,
@@ -741,20 +821,64 @@ var WalletDetector = class extends BaseCollaborator {
741
821
  return null;
742
822
  }
743
823
  /**
744
- * Get currently detected wallets
824
+ * Get currently detected wallets (legacy)
825
+ * @deprecated Use getConnectors() for vNext API
745
826
  */
746
827
  getDetectedWallets() {
747
828
  return this.getState().wallets;
748
829
  }
749
830
  /**
750
- * Convert a Wallet Standard wallet to WalletInfo with capability checks
831
+ * Get all available connectors (vNext API)
832
+ */
833
+ getConnectors() {
834
+ return this.getState().connectors;
835
+ }
836
+ /**
837
+ * Get a wallet by its stable connector ID (vNext API)
838
+ * Returns the Wallet reference for connection operations
839
+ */
840
+ getConnectorById(connectorId) {
841
+ return this.connectorRegistry.get(connectorId);
842
+ }
843
+ /**
844
+ * Get connector metadata by ID
845
+ */
846
+ getConnectorMetadata(connectorId) {
847
+ return this.getState().connectors.find((c) => c.id === connectorId);
848
+ }
849
+ /**
850
+ * Update the connector registry map
851
+ */
852
+ updateConnectorRegistry(wallets) {
853
+ this.connectorRegistry.clear();
854
+ for (let wallet of wallets) {
855
+ let connectorId = createConnectorId(wallet.name);
856
+ this.connectorRegistry.set(connectorId, wallet);
857
+ }
858
+ }
859
+ /**
860
+ * Convert a Wallet Standard wallet to WalletConnectorMetadata (serializable)
861
+ */
862
+ mapToConnectorMetadata(wallet) {
863
+ let walletWithIcon = applyWalletIconOverride(wallet), hasConnect = hasFeature(walletWithIcon, "standard:connect"), isSolana = isSolanaWallet(walletWithIcon), ready2 = hasConnect && isSolana;
864
+ return {
865
+ id: createConnectorId(wallet.name),
866
+ name: wallet.name,
867
+ icon: typeof walletWithIcon.icon == "string" ? walletWithIcon.icon : "",
868
+ ready: ready2,
869
+ chains: walletWithIcon.chains ?? [],
870
+ features: Object.keys(walletWithIcon.features ?? {})
871
+ };
872
+ }
873
+ /**
874
+ * Convert a Wallet Standard wallet to WalletInfo with capability checks (legacy)
751
875
  */
752
876
  mapToWalletInfo(wallet) {
753
- let hasConnect = hasFeature(wallet, "standard:connect"), hasDisconnect = hasFeature(wallet, "standard:disconnect"), isSolana = Array.isArray(wallet.chains) && wallet.chains.some((c) => typeof c == "string" && c.includes("solana"));
877
+ let walletWithIcon = applyWalletIconOverride(wallet), hasConnect = hasFeature(walletWithIcon, "standard:connect"), isSolana = isSolanaWallet(walletWithIcon);
754
878
  return {
755
- wallet,
879
+ wallet: walletWithIcon,
756
880
  installed: true,
757
- connectable: hasConnect && hasDisconnect && isSolana
881
+ connectable: hasConnect && isSolana
758
882
  };
759
883
  }
760
884
  /**
@@ -797,26 +921,258 @@ var ConnectionManager = class extends BaseCollaborator {
797
921
  __publicField(this, "walletChangeUnsub", null);
798
922
  __publicField(this, "pollTimer", null);
799
923
  __publicField(this, "pollAttempts", 0);
924
+ __publicField(this, "connectAttemptId", 0);
925
+ __publicField(this, "pendingWallet", null);
926
+ __publicField(this, "pendingWalletName", null);
800
927
  this.walletStorage = walletStorage;
801
928
  }
929
+ // ========================================================================
930
+ // vNext Connection Methods (connector-id based, silent-first)
931
+ // ========================================================================
932
+ /**
933
+ * Connect to a wallet using the vNext API with silent-first support.
934
+ *
935
+ * @param wallet - Wallet Standard wallet instance
936
+ * @param connectorId - Stable connector identifier
937
+ * @param options - Connection options (silent mode, preferred account, etc.)
938
+ */
939
+ async connectWallet(wallet, connectorId, options) {
940
+ if (typeof window > "u") return;
941
+ let attemptId = ++this.connectAttemptId;
942
+ this.pendingWallet = wallet, this.pendingWalletName = wallet.name;
943
+ let { silent = false, allowInteractiveFallback = true, preferredAccount } = options ?? {};
944
+ this.updateWalletStatus({
945
+ status: "connecting",
946
+ connectorId
947
+ }), this.stateManager.updateState({ connecting: true }, true), this.eventEmitter.emit({
948
+ type: "connecting",
949
+ wallet: wallet.name,
950
+ timestamp: (/* @__PURE__ */ new Date()).toISOString()
951
+ });
952
+ try {
953
+ let connect = getConnectFeature(wallet);
954
+ if (!connect) throw new Error(`Wallet ${wallet.name} does not support standard connect`);
955
+ let result;
956
+ if (silent)
957
+ try {
958
+ if (result = await connect({ silent: true }), attemptId !== this.connectAttemptId) throw new Error("Connection cancelled");
959
+ let hasAccounts = result.accounts.length > 0 || wallet.accounts.length > 0;
960
+ if (!hasAccounts && allowInteractiveFallback)
961
+ this.log("Silent connect returned no accounts, trying interactive..."), result = await connect({ silent: false });
962
+ else if (!hasAccounts)
963
+ throw new Error("Silent connection failed: no accounts returned");
964
+ } catch (silentError) {
965
+ if (attemptId !== this.connectAttemptId) throw new Error("Connection cancelled");
966
+ if (allowInteractiveFallback)
967
+ this.log("Silent connect failed, trying interactive...", silentError), result = await connect({ silent: false });
968
+ else
969
+ throw silentError;
970
+ }
971
+ else
972
+ result = await connect({ silent: false });
973
+ if (attemptId !== this.connectAttemptId) throw new Error("Connection cancelled");
974
+ let walletAccounts = wallet.accounts, accountMap = /* @__PURE__ */ new Map();
975
+ for (let a of [...walletAccounts, ...result.accounts]) accountMap.set(a.address, a);
976
+ let sessionAccounts = Array.from(accountMap.values()).map((a) => ({
977
+ address: a.address,
978
+ label: a.label,
979
+ account: a
980
+ })), legacyAccounts = sessionAccounts.map((a) => this.toAccountInfo(a.account)), selectedAccount = sessionAccounts[0];
981
+ if (preferredAccount) {
982
+ let preferred = sessionAccounts.find((a) => a.address === preferredAccount);
983
+ preferred && (selectedAccount = preferred);
984
+ }
985
+ let session = this.createSession(wallet, connectorId, sessionAccounts, selectedAccount);
986
+ this.updateWalletStatus({
987
+ status: "connected",
988
+ session
989
+ }), this.stateManager.updateState(
990
+ {
991
+ selectedWallet: wallet,
992
+ connected: true,
993
+ connecting: false,
994
+ accounts: legacyAccounts,
995
+ selectedAccount: selectedAccount?.address ?? null
996
+ },
997
+ true
998
+ ), this.log("\u2705 Connection successful (vNext)", {
999
+ connectorId,
1000
+ selectedAccount: selectedAccount?.address,
1001
+ accountsCount: sessionAccounts.length
1002
+ }), selectedAccount && this.eventEmitter.emit({
1003
+ type: "wallet:connected",
1004
+ wallet: wallet.name,
1005
+ account: selectedAccount.address,
1006
+ timestamp: (/* @__PURE__ */ new Date()).toISOString()
1007
+ }), this.walletStorage && (!("isAvailable" in this.walletStorage) || typeof this.walletStorage.isAvailable != "function" || this.walletStorage.isAvailable()) && this.walletStorage.set(wallet.name), this.subscribeToWalletEventsVNext(wallet, connectorId);
1008
+ } catch (e) {
1009
+ if (attemptId !== this.connectAttemptId) throw e;
1010
+ let errorMessage = e instanceof Error ? e.message : String(e), error = e instanceof Error ? e : new Error(errorMessage);
1011
+ throw this.updateWalletStatus({
1012
+ status: "error",
1013
+ error,
1014
+ connectorId,
1015
+ recoverable: this.isRecoverableError(error)
1016
+ }), this.stateManager.updateState(
1017
+ {
1018
+ selectedWallet: null,
1019
+ connected: false,
1020
+ connecting: false,
1021
+ accounts: [],
1022
+ selectedAccount: null
1023
+ },
1024
+ true
1025
+ ), this.eventEmitter.emit({
1026
+ type: "connection:failed",
1027
+ wallet: wallet.name,
1028
+ error: errorMessage,
1029
+ timestamp: (/* @__PURE__ */ new Date()).toISOString()
1030
+ }), e;
1031
+ } finally {
1032
+ this.pendingWallet === wallet && (this.pendingWallet = null, this.pendingWalletName = null);
1033
+ }
1034
+ }
1035
+ /**
1036
+ * Create a WalletSession object
1037
+ */
1038
+ createSession(wallet, connectorId, accounts, selectedAccount) {
1039
+ let listeners = /* @__PURE__ */ new Set();
1040
+ return {
1041
+ connectorId,
1042
+ accounts,
1043
+ selectedAccount,
1044
+ onAccountsChanged: (listener) => (listeners.add(listener), () => listeners.delete(listener)),
1045
+ selectAccount: async (address) => {
1046
+ if (accounts.find((a) => a.address === address))
1047
+ return this.selectAccount(address);
1048
+ }
1049
+ };
1050
+ }
1051
+ /**
1052
+ * Update wallet status in state
1053
+ */
1054
+ updateWalletStatus(status) {
1055
+ this.stateManager.updateState({ wallet: status });
1056
+ }
1057
+ /**
1058
+ * Subscribe to wallet events (vNext version with improved account change handling)
1059
+ */
1060
+ subscribeToWalletEventsVNext(wallet, connectorId) {
1061
+ this.walletChangeUnsub && (this.walletChangeUnsub(), this.walletChangeUnsub = null), this.stopPollingWalletAccounts();
1062
+ let eventsOn = getEventsFeature(wallet);
1063
+ if (!eventsOn) {
1064
+ this.startPollingWalletAccountsVNext(wallet, connectorId);
1065
+ return;
1066
+ }
1067
+ try {
1068
+ this.walletChangeUnsub = eventsOn("change", (properties) => {
1069
+ let changeAccounts = properties?.accounts ?? [];
1070
+ this.handleAccountsChanged(changeAccounts, connectorId, wallet);
1071
+ });
1072
+ } catch {
1073
+ this.startPollingWalletAccountsVNext(wallet, connectorId);
1074
+ }
1075
+ }
1076
+ /**
1077
+ * Handle accounts changed event with proper validation
1078
+ */
1079
+ handleAccountsChanged(newAccounts, connectorId, wallet) {
1080
+ let state = this.getState();
1081
+ if (newAccounts.length === 0) {
1082
+ this.log("No accounts available, disconnecting..."), this.disconnect();
1083
+ return;
1084
+ }
1085
+ let sessionAccounts = newAccounts.map((a) => ({
1086
+ address: a.address,
1087
+ label: a.label,
1088
+ account: a
1089
+ })), legacyAccounts = sessionAccounts.map((a) => this.toAccountInfo(a.account)), currentSelected = state.selectedAccount, selectedAccount = sessionAccounts.find((a) => a.address === currentSelected);
1090
+ if (selectedAccount || (selectedAccount = sessionAccounts[0], this.eventEmitter.emit({
1091
+ type: "account:changed",
1092
+ account: selectedAccount.address,
1093
+ timestamp: (/* @__PURE__ */ new Date()).toISOString()
1094
+ })), state.wallet.status === "connected") {
1095
+ let session = this.createSession(wallet, connectorId, sessionAccounts, selectedAccount);
1096
+ this.updateWalletStatus({
1097
+ status: "connected",
1098
+ session
1099
+ });
1100
+ }
1101
+ this.stateManager.updateState({
1102
+ accounts: legacyAccounts,
1103
+ selectedAccount: selectedAccount.address
1104
+ });
1105
+ }
1106
+ /**
1107
+ * Start polling wallet accounts (vNext version)
1108
+ */
1109
+ startPollingWalletAccountsVNext(wallet, connectorId) {
1110
+ if (this.pollTimer) return;
1111
+ this.pollAttempts = 0;
1112
+ let poll = () => {
1113
+ if (this.pollAttempts >= 20) {
1114
+ this.stopPollingWalletAccounts(), this.log("Stopped wallet polling after max attempts");
1115
+ return;
1116
+ }
1117
+ try {
1118
+ let walletAccounts = wallet.accounts;
1119
+ walletAccounts.length > 0 && (this.handleAccountsChanged(walletAccounts, connectorId, wallet), this.pollAttempts = 0);
1120
+ } catch (error) {
1121
+ this.log("Wallet polling error:", error);
1122
+ }
1123
+ this.pollAttempts++;
1124
+ let intervalIndex = Math.min(this.pollAttempts, POLL_INTERVALS_MS.length - 1), interval = POLL_INTERVALS_MS[intervalIndex];
1125
+ this.pollTimer = setTimeout(poll, interval);
1126
+ };
1127
+ poll();
1128
+ }
802
1129
  /**
803
- * Connect to a wallet
1130
+ * Check if an error is recoverable
1131
+ */
1132
+ isRecoverableError(error) {
1133
+ let message = error.message.toLowerCase();
1134
+ return message.includes("user rejected") || message.includes("user denied") || message.includes("cancelled") || message.includes("canceled");
1135
+ }
1136
+ // ========================================================================
1137
+ // Legacy Connection Methods (kept for backwards compatibility)
1138
+ // ========================================================================
1139
+ /**
1140
+ * Connect to a wallet (legacy API)
1141
+ * @deprecated Use connectWallet() instead
804
1142
  */
805
1143
  async connect(wallet, walletName) {
806
1144
  if (typeof window > "u") return;
807
- let name = walletName || wallet.name;
1145
+ let name = walletName || wallet.name, attemptId = ++this.connectAttemptId;
1146
+ this.pendingWallet = wallet, this.pendingWalletName = name;
1147
+ let connectorId = createConnectorId(name);
808
1148
  this.eventEmitter.emit({
809
1149
  type: "connecting",
810
1150
  wallet: name,
811
1151
  timestamp: (/* @__PURE__ */ new Date()).toISOString()
1152
+ }), this.updateWalletStatus({
1153
+ status: "connecting",
1154
+ connectorId
812
1155
  }), this.stateManager.updateState({ connecting: true }, true);
813
1156
  try {
814
1157
  let connect = getConnectFeature(wallet);
815
1158
  if (!connect) throw new Error(`Wallet ${name} does not support standard connect`);
816
- let result = await connect({ silent: false }), walletAccounts = wallet.accounts, accountMap = /* @__PURE__ */ new Map();
1159
+ let result = await connect({ silent: false });
1160
+ if (attemptId !== this.connectAttemptId)
1161
+ throw new Error("Connection cancelled");
1162
+ let walletAccounts = wallet.accounts, accountMap = /* @__PURE__ */ new Map();
817
1163
  for (let a of [...walletAccounts, ...result.accounts]) accountMap.set(a.address, a);
818
- let accounts = Array.from(accountMap.values()).map((a) => this.toAccountInfo(a)), state = this.getState(), previouslySelected = state.selectedAccount, previousAddresses = new Set(state.accounts.map((a) => a.address)), selected = accounts.find((a) => !previousAddresses.has(a.address))?.address ?? previouslySelected ?? accounts[0]?.address ?? null;
819
- this.stateManager.updateState(
1164
+ let sessionAccounts = Array.from(accountMap.values()).map((a) => ({
1165
+ address: a.address,
1166
+ label: a.label,
1167
+ account: a
1168
+ }));
1169
+ if (sessionAccounts.length === 0)
1170
+ throw new Error(`Wallet ${name} connected but returned no accounts`);
1171
+ let accounts = sessionAccounts.map((a) => this.toAccountInfo(a.account)), state = this.getState(), previouslySelected = state.selectedAccount, previousAddresses = new Set(state.accounts.map((a) => a.address)), selected = accounts.find((a) => !previousAddresses.has(a.address))?.address ?? previouslySelected ?? accounts[0]?.address ?? null, selectedSessionAccount = (selected ? sessionAccounts.find((a) => a.address === selected) : void 0) ?? sessionAccounts[0], session = this.createSession(wallet, connectorId, sessionAccounts, selectedSessionAccount);
1172
+ this.updateWalletStatus({
1173
+ status: "connected",
1174
+ session
1175
+ }), this.stateManager.updateState(
820
1176
  {
821
1177
  selectedWallet: wallet,
822
1178
  connected: true,
@@ -840,7 +1196,9 @@ var ConnectionManager = class extends BaseCollaborator {
840
1196
  accountsCount: accounts.length
841
1197
  }), this.walletStorage && (!("isAvailable" in this.walletStorage) || typeof this.walletStorage.isAvailable != "function" || this.walletStorage.isAvailable() ? this.walletStorage.set(name) : this.log("Storage not available (private browsing?), skipping wallet persistence")), this.subscribeToWalletEvents();
842
1198
  } catch (e) {
843
- let errorMessage = e instanceof Error ? e.message : String(e);
1199
+ if (attemptId !== this.connectAttemptId)
1200
+ throw e;
1201
+ let errorMessage = e instanceof Error ? e.message : String(e), error = e instanceof Error ? e : new Error(errorMessage);
844
1202
  throw this.eventEmitter.emit({
845
1203
  type: "connection:failed",
846
1204
  wallet: name,
@@ -848,9 +1206,14 @@ var ConnectionManager = class extends BaseCollaborator {
848
1206
  timestamp: (/* @__PURE__ */ new Date()).toISOString()
849
1207
  }), this.eventEmitter.emit({
850
1208
  type: "error",
851
- error: e instanceof Error ? e : new Error(errorMessage),
1209
+ error,
852
1210
  context: "wallet-connection",
853
1211
  timestamp: (/* @__PURE__ */ new Date()).toISOString()
1212
+ }), this.updateWalletStatus({
1213
+ status: "error",
1214
+ error,
1215
+ connectorId,
1216
+ recoverable: this.isRecoverableError(error)
854
1217
  }), this.stateManager.updateState(
855
1218
  {
856
1219
  selectedWallet: null,
@@ -861,22 +1224,21 @@ var ConnectionManager = class extends BaseCollaborator {
861
1224
  },
862
1225
  true
863
1226
  ), e;
1227
+ } finally {
1228
+ this.pendingWallet === wallet && this.pendingWalletName === name && (this.pendingWallet = null, this.pendingWalletName = null);
864
1229
  }
865
1230
  }
866
1231
  /**
867
1232
  * Disconnect from wallet
868
1233
  */
869
1234
  async disconnect() {
870
- this.walletChangeUnsub && (this.walletChangeUnsub(), this.walletChangeUnsub = null), this.stopPollingWalletAccounts();
871
- let wallet = this.getState().selectedWallet;
872
- if (wallet) {
873
- let disconnect = getDisconnectFeature(wallet);
874
- disconnect && await disconnect();
875
- }
876
- this.stateManager.updateState(
1235
+ this.connectAttemptId++, this.walletChangeUnsub && (this.walletChangeUnsub(), this.walletChangeUnsub = null), this.stopPollingWalletAccounts();
1236
+ let wallet = this.getState().selectedWallet ?? this.pendingWallet;
1237
+ if (this.pendingWallet = null, this.pendingWalletName = null, this.updateWalletStatus(INITIAL_WALLET_STATUS), this.stateManager.updateState(
877
1238
  {
878
1239
  selectedWallet: null,
879
1240
  connected: false,
1241
+ connecting: false,
880
1242
  accounts: [],
881
1243
  selectedAccount: null
882
1244
  },
@@ -884,7 +1246,14 @@ var ConnectionManager = class extends BaseCollaborator {
884
1246
  ), this.eventEmitter.emit({
885
1247
  type: "wallet:disconnected",
886
1248
  timestamp: (/* @__PURE__ */ new Date()).toISOString()
887
- }), this.walletStorage && "clear" in this.walletStorage && typeof this.walletStorage.clear == "function" ? this.walletStorage.clear() : this.walletStorage?.set(void 0);
1249
+ }), this.walletStorage && "clear" in this.walletStorage && typeof this.walletStorage.clear == "function" ? this.walletStorage.clear() : this.walletStorage?.set(void 0), wallet) {
1250
+ let disconnect = getDisconnectFeature(wallet);
1251
+ if (disconnect)
1252
+ try {
1253
+ await disconnect();
1254
+ } catch {
1255
+ }
1256
+ }
888
1257
  }
889
1258
  /**
890
1259
  * Select a different account
@@ -992,16 +1361,49 @@ var ConnectionManager = class extends BaseCollaborator {
992
1361
 
993
1362
  // src/lib/wallet/auto-connector.ts
994
1363
  var logger4 = createLogger("AutoConnector"), MIN_ADDRESS_LENGTH = 30, AutoConnector = class {
995
- constructor(walletDetector, connectionManager, stateManager, walletStorage, debug = false) {
1364
+ constructor(walletDetector, connectionManager, stateManager, walletStorage, debug = false, walletStateStorage) {
996
1365
  __publicField(this, "walletDetector");
997
1366
  __publicField(this, "connectionManager");
998
1367
  __publicField(this, "stateManager");
1368
+ /** Legacy wallet name storage */
999
1369
  __publicField(this, "walletStorage");
1370
+ /** vNext wallet state storage (connector ID + account) */
1371
+ __publicField(this, "walletStateStorage");
1000
1372
  __publicField(this, "debug");
1001
- this.walletDetector = walletDetector, this.connectionManager = connectionManager, this.stateManager = stateManager, this.walletStorage = walletStorage, this.debug = debug;
1373
+ this.walletDetector = walletDetector, this.connectionManager = connectionManager, this.stateManager = stateManager, this.walletStorage = walletStorage, this.walletStateStorage = walletStateStorage, this.debug = debug;
1002
1374
  }
1003
1375
  async attemptAutoConnect() {
1004
- return await this.attemptInstantConnect() ? true : (await this.attemptStandardConnect(), this.stateManager.getSnapshot().connected);
1376
+ return this.walletStateStorage && await this.attemptVNextAutoConnect() || await this.attemptInstantConnect() ? true : (await this.attemptStandardConnect(), this.stateManager.getSnapshot().connected);
1377
+ }
1378
+ /**
1379
+ * vNext auto-connect using stored connector ID with silent-first approach.
1380
+ * This won't prompt the user unless they explicitly initiated the connect.
1381
+ */
1382
+ async attemptVNextAutoConnect() {
1383
+ let walletState = this.walletStateStorage?.get();
1384
+ if (!walletState || !walletState.autoConnect)
1385
+ return this.debug && logger4.debug("vNext auto-connect: No stored wallet state or autoConnect disabled"), false;
1386
+ let { connectorId, lastAccount } = walletState;
1387
+ await ready;
1388
+ let wallet = this.walletDetector.getConnectorById(connectorId);
1389
+ if (!wallet)
1390
+ return this.debug && logger4.debug("vNext auto-connect: Connector not found", { connectorId }), false;
1391
+ try {
1392
+ return this.debug && logger4.info("vNext auto-connect: Attempting silent connect", {
1393
+ connectorId,
1394
+ lastAccount
1395
+ }), await this.connectionManager.connectWallet(wallet, connectorId, {
1396
+ silent: true,
1397
+ allowInteractiveFallback: false,
1398
+ // Don't prompt on auto-connect
1399
+ preferredAccount: lastAccount
1400
+ }), this.debug && logger4.info("vNext auto-connect: Silent connect successful", { connectorId }), true;
1401
+ } catch (error) {
1402
+ return this.debug && logger4.debug("vNext auto-connect: Silent connect failed (expected for first-time or revoked)", {
1403
+ connectorId,
1404
+ error: error instanceof Error ? error.message : error
1405
+ }), false;
1406
+ }
1005
1407
  }
1006
1408
  async attemptInstantConnect() {
1007
1409
  let storedWalletName = this.walletStorage?.get();
@@ -1092,12 +1494,12 @@ var logger4 = createLogger("AutoConnector"), MIN_ADDRESS_LENGTH = 30, AutoConnec
1092
1494
  ],
1093
1495
  features,
1094
1496
  accounts: []
1095
- };
1497
+ }, walletWithIcon = applyWalletIconOverride(wallet);
1096
1498
  this.stateManager.updateState(
1097
1499
  {
1098
1500
  wallets: [
1099
1501
  {
1100
- wallet,
1502
+ wallet: walletWithIcon,
1101
1503
  installed: true,
1102
1504
  connectable: true
1103
1505
  }
@@ -1105,7 +1507,7 @@ var logger4 = createLogger("AutoConnector"), MIN_ADDRESS_LENGTH = 30, AutoConnec
1105
1507
  },
1106
1508
  true
1107
1509
  ), await ready;
1108
- let walletsApi = getWalletsRegistry(), registryWallet = walletsApi.get().find((w) => w.name === storedWalletName), walletToUse = registryWallet || wallet;
1510
+ let walletsApi = getWalletsRegistry(), registryWallet = walletsApi.get().find((w) => w.name === storedWalletName), walletToUse = applyWalletIconOverride(registryWallet || walletWithIcon);
1109
1511
  return this.debug && logger4.info("Attempting to connect via instant auto-connect", {
1110
1512
  walletName: storedWalletName,
1111
1513
  usingRegistry: !!registryWallet
@@ -1319,8 +1721,13 @@ var logger5 = createLogger("ConnectorClient"), ConnectorClient = class {
1319
1721
  __publicField(this, "healthMonitor");
1320
1722
  __publicField(this, "initialized", false);
1321
1723
  __publicField(this, "config");
1724
+ __publicField(this, "walletConnectRegistration", null);
1322
1725
  this.config = config;
1323
1726
  let initialState = {
1727
+ // vNext wallet status
1728
+ wallet: INITIAL_WALLET_STATUS,
1729
+ connectors: [],
1730
+ // Legacy fields (for backwards compatibility)
1324
1731
  wallets: [],
1325
1732
  selectedWallet: null,
1326
1733
  connected: false,
@@ -1362,7 +1769,9 @@ var logger5 = createLogger("ConnectorClient"), ConnectorClient = class {
1362
1769
  initialize() {
1363
1770
  if (typeof window > "u" || this.initialized) return;
1364
1771
  let { error } = tryCatchSync(() => {
1365
- this.walletDetector.initialize(), this.config.autoConnect && setTimeout(() => {
1772
+ this.walletDetector.initialize(), this.config.walletConnect?.enabled && this.initializeWalletConnect().catch((err) => {
1773
+ this.config.debug && logger5.error("WalletConnect initialization failed", { error: err });
1774
+ }), this.config.autoConnect && setTimeout(() => {
1366
1775
  this.autoConnector.attemptAutoConnect().catch((err) => {
1367
1776
  this.config.debug && logger5.error("Auto-connect error", { error: err });
1368
1777
  });
@@ -1370,11 +1779,62 @@ var logger5 = createLogger("ConnectorClient"), ConnectorClient = class {
1370
1779
  });
1371
1780
  error && this.config.debug && logger5.error("Connector initialization failed", { error });
1372
1781
  }
1782
+ /**
1783
+ * Initialize WalletConnect integration
1784
+ * Dynamically imports and registers the WalletConnect wallet
1785
+ */
1786
+ async initializeWalletConnect() {
1787
+ if (this.config.walletConnect?.enabled)
1788
+ try {
1789
+ let { registerWalletConnectWallet } = await import('./walletconnect-U455PO4I.mjs');
1790
+ this.walletConnectRegistration = await registerWalletConnectWallet(this.config.walletConnect), this.config.debug && logger5.info("WalletConnect wallet registered successfully");
1791
+ } catch (error) {
1792
+ this.config.debug && logger5.error("Failed to register WalletConnect wallet", { error });
1793
+ }
1794
+ }
1795
+ // ========================================================================
1796
+ // vNext Wallet Actions (connector-id based)
1797
+ // ========================================================================
1798
+ /**
1799
+ * Connect to a wallet using its stable connector ID.
1800
+ * This is the recommended way to connect in vNext.
1801
+ *
1802
+ * @param connectorId - Stable connector identifier
1803
+ * @param options - Connection options (silent mode, preferred account, etc.)
1804
+ */
1805
+ async connectWallet(connectorId, options) {
1806
+ let connector = this.walletDetector.getConnectorById(connectorId);
1807
+ if (!connector)
1808
+ throw new Error(`Connector ${connectorId} not found`);
1809
+ await this.connectionManager.connectWallet(connector, connectorId, options);
1810
+ }
1811
+ /**
1812
+ * Disconnect the current wallet session.
1813
+ * This is the vNext equivalent of disconnect().
1814
+ */
1815
+ async disconnectWallet() {
1816
+ await this.connectionManager.disconnect();
1817
+ }
1818
+ /**
1819
+ * Get a connector by its ID (for advanced use cases).
1820
+ */
1821
+ getConnector(connectorId) {
1822
+ return this.walletDetector.getConnectorById(connectorId);
1823
+ }
1824
+ // ========================================================================
1825
+ // Legacy Actions (kept for backwards compatibility)
1826
+ // ========================================================================
1827
+ /**
1828
+ * @deprecated Use `connectWallet(connectorId)` instead.
1829
+ */
1373
1830
  async select(walletName) {
1374
1831
  let wallet = this.stateManager.getSnapshot().wallets.find((w) => w.wallet.name === walletName)?.wallet;
1375
1832
  if (!wallet) throw new Error(`Wallet ${walletName} not found`);
1376
1833
  await this.connectionManager.connect(wallet, walletName);
1377
1834
  }
1835
+ /**
1836
+ * @deprecated Use `disconnectWallet()` instead.
1837
+ */
1378
1838
  async disconnect() {
1379
1839
  await this.connectionManager.disconnect();
1380
1840
  }
@@ -1462,288 +1922,23 @@ var logger5 = createLogger("ConnectorClient"), ConnectorClient = class {
1462
1922
  this.debugMetrics.resetMetrics();
1463
1923
  }
1464
1924
  destroy() {
1465
- this.connectionManager.disconnect().catch(() => {
1466
- }), this.walletDetector.destroy(), this.eventEmitter.offAll(), this.stateManager.clear();
1467
- }
1468
- };
1469
- var logger6 = createLogger("ErrorBoundary"), WalletErrorType = /* @__PURE__ */ ((WalletErrorType2) => (WalletErrorType2.CONNECTION_FAILED = "CONNECTION_FAILED", WalletErrorType2.TRANSACTION_FAILED = "TRANSACTION_FAILED", WalletErrorType2.NETWORK_ERROR = "NETWORK_ERROR", WalletErrorType2.WALLET_NOT_FOUND = "WALLET_NOT_FOUND", WalletErrorType2.USER_REJECTED = "USER_REJECTED", WalletErrorType2.INSUFFICIENT_FUNDS = "INSUFFICIENT_FUNDS", WalletErrorType2.UNKNOWN_ERROR = "UNKNOWN_ERROR", WalletErrorType2))(WalletErrorType || {}), ErrorLogger = class {
1470
- static log(error, errorInfo, context) {
1471
- if (process.env.NODE_ENV === "development" && logger6.error(error.message, {
1472
- error,
1473
- errorInfo,
1474
- context
1475
- }), process.env.NODE_ENV === "production" && typeof window < "u")
1925
+ if (this.walletConnectRegistration)
1476
1926
  try {
1477
- let gtag = window.gtag;
1478
- typeof gtag == "function" && gtag("event", "exception", {
1479
- description: error.message,
1480
- fatal: false,
1481
- custom_map: { error_type: "wallet_error", ...context }
1482
- });
1483
- } catch {
1927
+ this.walletConnectRegistration.unregister(), this.walletConnectRegistration = null;
1928
+ } catch (error) {
1929
+ this.config.debug && logger5.warn("Error unregistering WalletConnect wallet", { error });
1484
1930
  }
1931
+ this.connectionManager.disconnect().catch(() => {
1932
+ }), this.walletDetector.destroy(), this.eventEmitter.offAll(), this.stateManager.clear();
1485
1933
  }
1486
1934
  };
1487
- function classifyError(error) {
1488
- if (isConnectorError(error))
1489
- return {
1490
- ...error,
1491
- type: {
1492
- WALLET_NOT_CONNECTED: "CONNECTION_FAILED" /* CONNECTION_FAILED */,
1493
- WALLET_NOT_FOUND: "WALLET_NOT_FOUND" /* WALLET_NOT_FOUND */,
1494
- CONNECTION_FAILED: "CONNECTION_FAILED" /* CONNECTION_FAILED */,
1495
- USER_REJECTED: "USER_REJECTED" /* USER_REJECTED */,
1496
- RPC_ERROR: "NETWORK_ERROR" /* NETWORK_ERROR */,
1497
- NETWORK_TIMEOUT: "NETWORK_ERROR" /* NETWORK_ERROR */,
1498
- SIGNING_FAILED: "TRANSACTION_FAILED" /* TRANSACTION_FAILED */,
1499
- SEND_FAILED: "TRANSACTION_FAILED" /* TRANSACTION_FAILED */
1500
- }[error.code] || "UNKNOWN_ERROR" /* UNKNOWN_ERROR */,
1501
- recoverable: error.recoverable,
1502
- context: error.context
1503
- };
1504
- let walletError = error;
1505
- if (walletError.type) return walletError;
1506
- let type = "UNKNOWN_ERROR" /* UNKNOWN_ERROR */, recoverable = false;
1507
- return error.message.includes("User rejected") || error.message.includes("User denied") ? (type = "USER_REJECTED" /* USER_REJECTED */, recoverable = true) : error.message.includes("Insufficient funds") ? (type = "INSUFFICIENT_FUNDS" /* INSUFFICIENT_FUNDS */, recoverable = false) : error.message.includes("Network") || error.message.includes("fetch") ? (type = "NETWORK_ERROR" /* NETWORK_ERROR */, recoverable = true) : error.message.includes("Wallet not found") || error.message.includes("not installed") ? (type = "WALLET_NOT_FOUND" /* WALLET_NOT_FOUND */, recoverable = true) : (error.message.includes("Failed to connect") || error.message.includes("Connection")) && (type = "CONNECTION_FAILED" /* CONNECTION_FAILED */, recoverable = true), {
1508
- ...error,
1509
- type,
1510
- recoverable,
1511
- context: { originalMessage: error.message }
1512
- };
1513
- }
1514
- var ConnectorErrorBoundary = class extends Component {
1515
- constructor(props) {
1516
- super(props);
1517
- __publicField(this, "retryTimeouts", /* @__PURE__ */ new Set());
1518
- __publicField(this, "retry", () => {
1519
- let { maxRetries = 3 } = this.props;
1520
- this.state.retryCount >= maxRetries || this.setState((prevState) => ({
1521
- hasError: false,
1522
- error: null,
1523
- errorInfo: null,
1524
- retryCount: prevState.retryCount + 1
1525
- }));
1526
- });
1527
- this.state = {
1528
- hasError: false,
1529
- error: null,
1530
- errorInfo: null,
1531
- errorId: "",
1532
- retryCount: 0
1533
- };
1534
- }
1535
- static getDerivedStateFromError(error) {
1536
- return {
1537
- hasError: true,
1538
- error,
1539
- errorId: `error_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`
1540
- };
1541
- }
1542
- componentDidCatch(error, errorInfo) {
1543
- this.setState({ errorInfo }), ErrorLogger.log(error, errorInfo, {
1544
- retryCount: this.state.retryCount,
1545
- errorId: this.state.errorId,
1546
- timestamp: (/* @__PURE__ */ new Date()).toISOString()
1547
- }), this.props.onError?.(error, errorInfo);
1548
- }
1549
- componentWillUnmount() {
1550
- this.retryTimeouts.forEach((timeout) => clearTimeout(timeout));
1551
- }
1552
- render() {
1553
- if (this.state.hasError && this.state.error) {
1554
- let walletError = classifyError(this.state.error);
1555
- return this.props.fallback ? this.props.fallback(walletError, this.retry) : /* @__PURE__ */ jsx(DefaultErrorFallback, { error: walletError, onRetry: this.retry });
1556
- }
1557
- return this.props.children;
1558
- }
1559
- };
1560
- function DefaultErrorFallback({ error, onRetry }) {
1561
- let [isPending, startTransition] = useTransition(), [isRetrying, setIsRetrying] = useState(false), handleRetry = useCallback(() => {
1562
- setIsRetrying(true), startTransition(() => {
1563
- setTimeout(() => {
1564
- onRetry(), setIsRetrying(false);
1565
- }, 500);
1566
- });
1567
- }, [onRetry]), { title, message, actionText, showRetry } = useMemo(() => {
1568
- switch (error.type) {
1569
- case "USER_REJECTED" /* USER_REJECTED */:
1570
- return {
1571
- title: "Transaction Cancelled",
1572
- message: "You cancelled the transaction. No problem!",
1573
- actionText: "Try Again",
1574
- showRetry: true
1575
- };
1576
- case "WALLET_NOT_FOUND" /* WALLET_NOT_FOUND */:
1577
- return {
1578
- title: "Wallet Not Found",
1579
- message: "Please install a supported Solana wallet to continue.",
1580
- actionText: "Check Wallets",
1581
- showRetry: true
1582
- };
1583
- case "NETWORK_ERROR" /* NETWORK_ERROR */:
1584
- return {
1585
- title: "Network Error",
1586
- message: "Having trouble connecting. Please check your internet connection.",
1587
- actionText: "Retry",
1588
- showRetry: true
1589
- };
1590
- case "INSUFFICIENT_FUNDS" /* INSUFFICIENT_FUNDS */:
1591
- return {
1592
- title: "Insufficient Funds",
1593
- message: "You don't have enough SOL for this transaction.",
1594
- actionText: "Add Funds",
1595
- showRetry: false
1596
- };
1597
- default:
1598
- return {
1599
- title: "Something went wrong",
1600
- message: "An unexpected error occurred. Please try again.",
1601
- actionText: "Retry",
1602
- showRetry: error.recoverable
1603
- };
1604
- }
1605
- }, [error.type, error.recoverable]);
1606
- return /* @__PURE__ */ jsxs(
1607
- "div",
1608
- {
1609
- style: {
1610
- display: "flex",
1611
- flexDirection: "column",
1612
- alignItems: "center",
1613
- justifyContent: "center",
1614
- padding: "2rem",
1615
- textAlign: "center",
1616
- borderRadius: "12px",
1617
- border: "1px solid #e5e7eb",
1618
- backgroundColor: "#fafafa",
1619
- maxWidth: "400px",
1620
- margin: "0 auto"
1621
- },
1622
- children: [
1623
- /* @__PURE__ */ jsx(
1624
- "div",
1625
- {
1626
- style: {
1627
- width: "48px",
1628
- height: "48px",
1629
- borderRadius: "50%",
1630
- backgroundColor: "#fee2e2",
1631
- display: "flex",
1632
- alignItems: "center",
1633
- justifyContent: "center",
1634
- marginBottom: "1rem"
1635
- },
1636
- children: /* @__PURE__ */ jsx("svg", { width: "24", height: "24", viewBox: "0 0 24 24", fill: "#dc2626", children: /* @__PURE__ */ jsx("path", { d: "M12 9v3.75m-9.303 3.376c-.866 1.5.217 3.374 1.948 3.374h14.71c1.73 0 2.813-1.874 1.948-3.374L13.949 3.378c-.866-1.5-3.032-1.5-3.898 0L2.697 16.126zM12 15.75h.007v.008H12v-.008z" }) })
1637
- }
1638
- ),
1639
- /* @__PURE__ */ jsx(
1640
- "h3",
1641
- {
1642
- style: {
1643
- margin: "0 0 0.5rem 0",
1644
- fontSize: "1.125rem",
1645
- fontWeight: "600",
1646
- color: "#111827"
1647
- },
1648
- children: title
1649
- }
1650
- ),
1651
- /* @__PURE__ */ jsx(
1652
- "p",
1653
- {
1654
- style: {
1655
- margin: "0 0 1.5rem 0",
1656
- fontSize: "0.875rem",
1657
- color: "#6b7280",
1658
- lineHeight: "1.5"
1659
- },
1660
- children: message
1661
- }
1662
- ),
1663
- /* @__PURE__ */ jsxs("div", { style: { display: "flex", gap: "0.75rem", flexWrap: "wrap" }, children: [
1664
- showRetry && /* @__PURE__ */ jsx(
1665
- "button",
1666
- {
1667
- onClick: handleRetry,
1668
- disabled: isPending || isRetrying,
1669
- style: {
1670
- padding: "0.5rem 1rem",
1671
- backgroundColor: "#3b82f6",
1672
- color: "white",
1673
- border: "none",
1674
- borderRadius: "6px",
1675
- fontSize: "0.875rem",
1676
- fontWeight: "500",
1677
- cursor: isPending || isRetrying ? "wait" : "pointer",
1678
- opacity: isPending || isRetrying ? 0.7 : 1,
1679
- transition: "all 0.2s"
1680
- },
1681
- children: isRetrying ? "Retrying..." : actionText
1682
- }
1683
- ),
1684
- /* @__PURE__ */ jsx(
1685
- "button",
1686
- {
1687
- onClick: () => window.location.reload(),
1688
- style: {
1689
- padding: "0.5rem 1rem",
1690
- backgroundColor: "transparent",
1691
- color: "#6b7280",
1692
- border: "1px solid #d1d5db",
1693
- borderRadius: "6px",
1694
- fontSize: "0.875rem",
1695
- fontWeight: "500",
1696
- cursor: "pointer",
1697
- transition: "all 0.2s"
1698
- },
1699
- children: "Refresh Page"
1700
- }
1701
- )
1702
- ] }),
1703
- process.env.NODE_ENV === "development" && /* @__PURE__ */ jsxs(
1704
- "details",
1705
- {
1706
- style: {
1707
- marginTop: "1rem",
1708
- fontSize: "0.75rem",
1709
- color: "#6b7280",
1710
- width: "100%"
1711
- },
1712
- children: [
1713
- /* @__PURE__ */ jsx("summary", { style: { cursor: "pointer", marginBottom: "0.5rem" }, children: "Error Details" }),
1714
- /* @__PURE__ */ jsx(
1715
- "pre",
1716
- {
1717
- style: {
1718
- whiteSpace: "pre-wrap",
1719
- wordBreak: "break-all",
1720
- backgroundColor: "#f3f4f6",
1721
- padding: "0.5rem",
1722
- borderRadius: "4px",
1723
- overflow: "auto",
1724
- maxHeight: "200px"
1725
- },
1726
- children: error.message
1727
- }
1728
- )
1729
- ]
1730
- }
1731
- )
1732
- ]
1733
- }
1734
- );
1735
- }
1736
- function withErrorBoundary(Component2, errorBoundaryProps) {
1737
- let WrappedComponent = (props) => /* @__PURE__ */ jsx(ConnectorErrorBoundary, { ...errorBoundaryProps, children: /* @__PURE__ */ jsx(Component2, { ...props }) });
1738
- return WrappedComponent.displayName = `withErrorBoundary(${Component2.displayName || Component2.name})`, WrappedComponent;
1739
- }
1740
- var logger7 = createLogger("Polyfills"), installed = false;
1935
+ var logger6 = createLogger("Polyfills"), installed = false;
1741
1936
  function installPolyfills() {
1742
1937
  if (!(installed || typeof window > "u"))
1743
1938
  try {
1744
- install(), installed = true, logger7.info("Browser compatibility polyfills installed");
1939
+ install(), installed = true, logger6.info("Browser compatibility polyfills installed");
1745
1940
  } catch (error) {
1746
- logger7 && logger7.warn("Failed to install polyfills", { error }), installed = true;
1941
+ logger6 && logger6.warn("Failed to install polyfills", { error }), installed = true;
1747
1942
  }
1748
1943
  }
1749
1944
  function isPolyfillInstalled() {
@@ -1764,6 +1959,404 @@ function getPolyfillStatus() {
1764
1959
  environment: typeof window < "u" ? "browser" : "server"
1765
1960
  };
1766
1961
  }
1962
+ var logger7 = createLogger("EnhancedStorage"), STORAGE_VERSION = "v1", EnhancedStorage = class extends Storage {
1963
+ constructor(key, initial, options) {
1964
+ super(key, initial);
1965
+ this.options = options;
1966
+ __publicField(this, "errorHandlers", /* @__PURE__ */ new Set());
1967
+ __publicField(this, "validators", []);
1968
+ __publicField(this, "memoryFallback");
1969
+ this.memoryFallback = initial, options?.onError && this.errorHandlers.add(options.onError), options?.validator && this.validators.push(options.validator);
1970
+ }
1971
+ set(value) {
1972
+ try {
1973
+ return this.validate(value) ? (super.set(value), this.memoryFallback = value, true) : (logger7.warn("Validation failed", { key: this.key }), false);
1974
+ } catch (error) {
1975
+ return this.handleError(error), this.options?.useMemoryFallback ? (this.memoryFallback = value, true) : false;
1976
+ }
1977
+ }
1978
+ get() {
1979
+ try {
1980
+ return super.get();
1981
+ } catch (error) {
1982
+ return this.handleError(error), this.options?.useMemoryFallback ? this.memoryFallback : this.initial;
1983
+ }
1984
+ }
1985
+ validate(value) {
1986
+ return this.validators.every((validator) => validator(value));
1987
+ }
1988
+ addValidator(validator) {
1989
+ return this.validators.push(validator), this;
1990
+ }
1991
+ onError(handler) {
1992
+ return this.errorHandlers.add(handler), this;
1993
+ }
1994
+ transform(transformer) {
1995
+ return transformer(this.get());
1996
+ }
1997
+ reset() {
1998
+ this.set(this.initial);
1999
+ }
2000
+ clear() {
2001
+ try {
2002
+ typeof window < "u" && window.localStorage && window.localStorage.removeItem(this.key), this.reset();
2003
+ } catch (error) {
2004
+ this.handleError(error);
2005
+ }
2006
+ }
2007
+ isAvailable() {
2008
+ try {
2009
+ if (typeof window > "u") return false;
2010
+ let testKey = `__storage_test_${this.key}__`;
2011
+ return window.localStorage.setItem(testKey, "test"), window.localStorage.removeItem(testKey), true;
2012
+ } catch {
2013
+ return false;
2014
+ }
2015
+ }
2016
+ static migrate(oldKey, newStorage) {
2017
+ try {
2018
+ if (typeof window > "u") return false;
2019
+ let oldValue = window.localStorage.getItem(oldKey);
2020
+ if (oldValue) {
2021
+ let parsed = JSON.parse(oldValue);
2022
+ return newStorage.set(parsed), window.localStorage.removeItem(oldKey), true;
2023
+ }
2024
+ return false;
2025
+ } catch {
2026
+ return false;
2027
+ }
2028
+ }
2029
+ handleError(error) {
2030
+ logger7.error("Storage error", { key: this.key, error }), this.errorHandlers.forEach((handler) => {
2031
+ try {
2032
+ handler(error);
2033
+ } catch (err) {
2034
+ logger7.error("Error in error handler", { error: err });
2035
+ }
2036
+ });
2037
+ }
2038
+ };
2039
+ function createEnhancedStorageAccount(options) {
2040
+ let key = options?.key ?? `connector-kit:${STORAGE_VERSION}:account`;
2041
+ return new EnhancedStorage(key, options?.initial, {
2042
+ validator: options?.validator,
2043
+ onError: options?.onError,
2044
+ useMemoryFallback: true
2045
+ // Always fallback for SSR
2046
+ });
2047
+ }
2048
+ function createEnhancedStorageCluster(options) {
2049
+ let key = options?.key ?? `connector-kit:${STORAGE_VERSION}:cluster`, storage = new EnhancedStorage(key, options?.initial ?? "solana:mainnet", {
2050
+ onError: options?.onError,
2051
+ useMemoryFallback: true
2052
+ });
2053
+ return options?.validClusters && storage.addValidator((clusterId) => options.validClusters.includes(clusterId)), storage;
2054
+ }
2055
+ function createEnhancedStorageWallet(options) {
2056
+ let key = options?.key ?? `connector-kit:${STORAGE_VERSION}:wallet`;
2057
+ return new EnhancedStorage(key, options?.initial, {
2058
+ onError: options?.onError,
2059
+ useMemoryFallback: true
2060
+ });
2061
+ }
2062
+ var WALLET_STATE_VERSION = 1;
2063
+ function createEnhancedStorageWalletState(options) {
2064
+ let key = options?.key ?? `connector-kit:${STORAGE_VERSION}:wallet-state`, legacyKey = `connector-kit:${STORAGE_VERSION}:wallet`, storage = new EnhancedStorage(key, options?.initial ?? null, {
2065
+ onError: options?.onError,
2066
+ useMemoryFallback: true
2067
+ });
2068
+ if (typeof window < "u" && storage.isAvailable())
2069
+ try {
2070
+ if (!storage.get()) {
2071
+ let legacyValue = window.localStorage.getItem(legacyKey);
2072
+ if (legacyValue) {
2073
+ let legacyWalletName = JSON.parse(legacyValue);
2074
+ if (legacyWalletName && typeof legacyWalletName == "string") {
2075
+ let connectorId = options?.migrateLegacy ? options.migrateLegacy(legacyWalletName) : createConnectorId(legacyWalletName);
2076
+ if (connectorId) {
2077
+ let migratedState = {
2078
+ version: WALLET_STATE_VERSION,
2079
+ connectorId,
2080
+ autoConnect: true,
2081
+ lastConnected: (/* @__PURE__ */ new Date()).toISOString()
2082
+ };
2083
+ storage.set(migratedState), logger7.info("Migrated legacy wallet storage", {
2084
+ from: legacyWalletName,
2085
+ to: connectorId
2086
+ }), window.localStorage.removeItem(legacyKey);
2087
+ }
2088
+ }
2089
+ }
2090
+ }
2091
+ } catch (error) {
2092
+ logger7.warn("Failed to migrate legacy wallet storage", { error });
2093
+ }
2094
+ return storage;
2095
+ }
2096
+ function saveWalletState(storage, connectorId, account, autoConnect = true) {
2097
+ let state = {
2098
+ version: WALLET_STATE_VERSION,
2099
+ connectorId,
2100
+ lastAccount: account,
2101
+ autoConnect,
2102
+ lastConnected: (/* @__PURE__ */ new Date()).toISOString()
2103
+ };
2104
+ storage.set(state);
2105
+ }
2106
+ function clearWalletState(storage) {
2107
+ storage.set(null);
2108
+ }
2109
+ var EnhancedStorageAdapter = class {
2110
+ constructor(storage) {
2111
+ this.storage = storage;
2112
+ }
2113
+ get() {
2114
+ return this.storage.get();
2115
+ }
2116
+ set(value) {
2117
+ this.storage.set(value);
2118
+ }
2119
+ subscribe(callback) {
2120
+ return this.storage.value.subscribe(callback);
2121
+ }
2122
+ validate(value) {
2123
+ return this.storage.validate(value);
2124
+ }
2125
+ reset() {
2126
+ this.storage.reset();
2127
+ }
2128
+ clear() {
2129
+ this.storage.clear();
2130
+ }
2131
+ isAvailable() {
2132
+ return this.storage.isAvailable();
2133
+ }
2134
+ transform(transformer) {
2135
+ return this.storage.transform(transformer);
2136
+ }
2137
+ addValidator(validator) {
2138
+ return this.storage.addValidator(validator), this;
2139
+ }
2140
+ onError(handler) {
2141
+ return this.storage.onError(handler), this;
2142
+ }
2143
+ };
2144
+ var logger8 = createLogger("DefaultConfig");
2145
+ function getDefaultConfig(options) {
2146
+ let {
2147
+ appName,
2148
+ appUrl,
2149
+ autoConnect = true,
2150
+ debug,
2151
+ network = "mainnet-beta",
2152
+ enableMobile = true,
2153
+ storage,
2154
+ clusters,
2155
+ customClusters = [],
2156
+ persistClusterSelection = true,
2157
+ clusterStorageKey,
2158
+ enableErrorBoundary = true,
2159
+ maxRetries = DEFAULT_MAX_RETRIES,
2160
+ onError,
2161
+ imageProxy,
2162
+ programLabels,
2163
+ coingecko,
2164
+ walletConnect
2165
+ } = options, defaultClusters = clusters ?? [
2166
+ createSolanaMainnet(),
2167
+ createSolanaDevnet(),
2168
+ createSolanaTestnet(),
2169
+ ...network === "localnet" ? [createSolanaLocalnet()] : [],
2170
+ ...customClusters || []
2171
+ ], validClusterIds = defaultClusters.map((c) => c.id), accountStorage = createEnhancedStorageAccount({
2172
+ validator: (address) => address ? isAddress(address) : true,
2173
+ onError: (error) => {
2174
+ debug && logger8.error("Account Storage error", { error }), onError && onError(error, {
2175
+ componentStack: "account-storage"
2176
+ });
2177
+ }
2178
+ }), clusterStorage = createEnhancedStorageCluster({
2179
+ key: clusterStorageKey,
2180
+ initial: getInitialCluster(network),
2181
+ validClusters: persistClusterSelection ? validClusterIds : void 0,
2182
+ onError: (error) => {
2183
+ debug && logger8.error("Cluster Storage error", { error }), onError && onError(error, {
2184
+ componentStack: "cluster-storage"
2185
+ });
2186
+ }
2187
+ }), walletStorage = createEnhancedStorageWallet({
2188
+ onError: (error) => {
2189
+ debug && logger8.error("Wallet Storage error", { error }), onError && onError(error, {
2190
+ componentStack: "wallet-storage"
2191
+ });
2192
+ }
2193
+ });
2194
+ if (typeof window < "u") {
2195
+ let oldAccountKey = "connector-kit:account", oldWalletKey = "connector-kit:wallet", oldClusterKey = clusterStorageKey || "connector-kit:cluster";
2196
+ EnhancedStorage.migrate(oldAccountKey, accountStorage), EnhancedStorage.migrate(oldWalletKey, walletStorage), EnhancedStorage.migrate(oldClusterKey, clusterStorage);
2197
+ }
2198
+ let defaultStorage = storage ?? {
2199
+ account: new EnhancedStorageAdapter(accountStorage),
2200
+ cluster: new EnhancedStorageAdapter(clusterStorage),
2201
+ wallet: new EnhancedStorageAdapter(walletStorage)
2202
+ }, walletConnectConfig = buildWalletConnectConfig(walletConnect, appName, appUrl, clusterStorageKey ?? "connector-kit:v1:cluster");
2203
+ return {
2204
+ autoConnect,
2205
+ debug: debug ?? process.env.NODE_ENV === "development",
2206
+ storage: defaultStorage,
2207
+ appName,
2208
+ appUrl,
2209
+ enableMobile,
2210
+ network,
2211
+ cluster: {
2212
+ clusters: defaultClusters,
2213
+ persistSelection: persistClusterSelection,
2214
+ initialCluster: getInitialCluster(network)
2215
+ },
2216
+ errorBoundary: {
2217
+ enabled: enableErrorBoundary,
2218
+ maxRetries,
2219
+ onError
2220
+ },
2221
+ imageProxy,
2222
+ programLabels,
2223
+ coingecko,
2224
+ walletConnect: walletConnectConfig
2225
+ };
2226
+ }
2227
+ function buildWalletConnectConfig(walletConnect, appName, appUrl, clusterStorageKey) {
2228
+ if (!walletConnect) return;
2229
+ let configProjectId = typeof walletConnect == "object" ? walletConnect.projectId : void 0, envProjectId = typeof process < "u" ? process.env.NEXT_PUBLIC_WALLETCONNECT_PROJECT_ID : void 0, projectId = configProjectId || envProjectId;
2230
+ if (!projectId) {
2231
+ (typeof walletConnect == "object" || walletConnect === true) && logger8.warn(
2232
+ "WalletConnect enabled but no project ID found. Set NEXT_PUBLIC_WALLETCONNECT_PROJECT_ID or provide projectId in config."
2233
+ );
2234
+ return;
2235
+ }
2236
+ let origin = appUrl || (typeof window < "u" ? window.location.origin : "http://localhost:3000"), customMetadata = typeof walletConnect == "object" ? walletConnect.metadata : void 0, customDefaultChain = typeof walletConnect == "object" ? walletConnect.defaultChain : void 0, customRelayUrl = typeof walletConnect == "object" ? walletConnect.relayUrl : void 0;
2237
+ return {
2238
+ enabled: true,
2239
+ projectId,
2240
+ metadata: {
2241
+ name: customMetadata?.name ?? appName,
2242
+ description: customMetadata?.description ?? `${appName} - Powered by ConnectorKit`,
2243
+ url: customMetadata?.url ?? origin,
2244
+ icons: customMetadata?.icons ?? [`${origin}/icon.svg`]
2245
+ },
2246
+ defaultChain: customDefaultChain ?? "solana:mainnet",
2247
+ relayUrl: customRelayUrl,
2248
+ // Auto-sync with cluster storage
2249
+ getCurrentChain: () => {
2250
+ if (typeof window > "u") return "solana:mainnet";
2251
+ let storageKey = clusterStorageKey || "connector-kit:v1:cluster";
2252
+ try {
2253
+ let stored = localStorage.getItem(storageKey);
2254
+ if (stored) {
2255
+ let id = JSON.parse(stored);
2256
+ if (id === "solana:mainnet" || id === "solana:devnet" || id === "solana:testnet")
2257
+ return id;
2258
+ }
2259
+ } catch {
2260
+ }
2261
+ return customDefaultChain ?? "solana:mainnet";
2262
+ }
2263
+ // Note: onDisplayUri, onSessionEstablished, onSessionDisconnected are auto-wired by AppProvider
2264
+ };
2265
+ }
2266
+ function getInitialCluster(network = "mainnet-beta") {
2267
+ return toClusterId(network);
2268
+ }
2269
+ function getDefaultMobileConfig(options) {
2270
+ let baseUrl = options.appUrl || (typeof window < "u" ? window.location.origin : "https://localhost:3000");
2271
+ return {
2272
+ appIdentity: {
2273
+ name: options.appName,
2274
+ uri: baseUrl,
2275
+ icon: `${baseUrl}/favicon.ico`
2276
+ },
2277
+ cluster: options.network || "mainnet-beta"
2278
+ };
2279
+ }
2280
+ var solanaNetworkSchema = z.enum(["mainnet", "mainnet-beta", "devnet", "testnet", "localnet"]), solanaClusterIdSchema = z.string().regex(/^solana:(mainnet|devnet|testnet|localnet|[a-zA-Z0-9-]+)$/, {
2281
+ message: 'Cluster ID must be in format "solana:<network>" (e.g., "solana:mainnet")'
2282
+ }), urlSchema = z.string().url("Invalid URL format"), optionalUrlSchema = urlSchema.optional(), coinGeckoConfigSchema = z.strictObject({
2283
+ apiKey: z.string().optional(),
2284
+ isPro: z.boolean().optional(),
2285
+ maxRetries: z.number().int().positive().max(10).optional(),
2286
+ baseDelay: z.number().int().positive().max(3e4).optional(),
2287
+ maxTimeout: z.number().int().positive().max(12e4).optional()
2288
+ }).optional(), walletConnectMetadataSchema = z.object({
2289
+ name: z.string().min(1, "WalletConnect app name is required"),
2290
+ description: z.string(),
2291
+ url: urlSchema,
2292
+ icons: z.array(z.string())
2293
+ }), walletConnectObjectConfigSchema = z.object({
2294
+ enabled: z.boolean().optional(),
2295
+ projectId: z.string().min(1, "WalletConnect projectId is required"),
2296
+ metadata: walletConnectMetadataSchema,
2297
+ defaultChain: z.enum(["solana:mainnet", "solana:devnet", "solana:testnet"]).optional(),
2298
+ onDisplayUri: z.custom((val) => typeof val == "function").optional(),
2299
+ onSessionEstablished: z.custom((val) => typeof val == "function").optional(),
2300
+ onSessionDisconnected: z.custom((val) => typeof val == "function").optional(),
2301
+ relayUrl: urlSchema.optional()
2302
+ }), walletConnectConfigSchema = z.union([z.literal(true), walletConnectObjectConfigSchema]).optional(), storageAdapterSchema = z.looseObject({
2303
+ get: z.custom((val) => typeof val == "function"),
2304
+ set: z.custom((val) => typeof val == "function")
2305
+ }), storageConfigSchema = z.object({
2306
+ account: storageAdapterSchema,
2307
+ cluster: storageAdapterSchema,
2308
+ wallet: storageAdapterSchema
2309
+ }).optional(), solanaClusterSchema = z.object({
2310
+ id: solanaClusterIdSchema,
2311
+ label: z.string().min(1, "Cluster label cannot be empty"),
2312
+ url: urlSchema,
2313
+ urlWs: urlSchema.optional()
2314
+ }), clusterConfigSchema = z.object({
2315
+ clusters: z.array(solanaClusterSchema).optional(),
2316
+ persistSelection: z.boolean().optional(),
2317
+ initialCluster: solanaClusterIdSchema.optional()
2318
+ }).optional(), defaultConfigOptionsSchema = z.object({
2319
+ // Required
2320
+ appName: z.string().min(1, "Application name is required"),
2321
+ // Optional strings
2322
+ appUrl: optionalUrlSchema,
2323
+ imageProxy: z.string().optional(),
2324
+ clusterStorageKey: z.string().optional(),
2325
+ // Optional booleans
2326
+ autoConnect: z.boolean().optional(),
2327
+ debug: z.boolean().optional(),
2328
+ enableMobile: z.boolean().optional(),
2329
+ persistClusterSelection: z.boolean().optional(),
2330
+ enableErrorBoundary: z.boolean().optional(),
2331
+ // Network
2332
+ network: solanaNetworkSchema.optional(),
2333
+ // Numbers
2334
+ maxRetries: z.number().int().positive().max(10).optional(),
2335
+ // Complex types
2336
+ storage: storageConfigSchema,
2337
+ clusters: z.array(solanaClusterSchema).optional(),
2338
+ customClusters: z.array(solanaClusterSchema).optional(),
2339
+ programLabels: z.record(z.string(), z.string()).optional(),
2340
+ coingecko: coinGeckoConfigSchema,
2341
+ walletConnect: walletConnectConfigSchema,
2342
+ // Functions (can't validate implementation, just existence)
2343
+ onError: z.custom((val) => typeof val == "function").optional()
2344
+ }); z.strictObject({
2345
+ autoConnect: z.boolean().optional(),
2346
+ debug: z.boolean().optional(),
2347
+ storage: storageConfigSchema,
2348
+ cluster: clusterConfigSchema,
2349
+ imageProxy: z.string().optional(),
2350
+ programLabels: z.record(z.string(), z.string()).optional(),
2351
+ coingecko: coinGeckoConfigSchema,
2352
+ walletConnect: walletConnectConfigSchema
2353
+ }).optional();
2354
+ function validateConfigOptions(options) {
2355
+ return defaultConfigOptionsSchema.safeParse(options);
2356
+ }
2357
+ function parseConfigOptions(options) {
2358
+ return defaultConfigOptionsSchema.parse(options);
2359
+ }
1767
2360
 
1768
2361
  // src/utils/formatting.ts
1769
2362
  function formatAddress(address, options = {}) {
@@ -1975,7 +2568,7 @@ async function copySignatureToClipboard(signature, options) {
1975
2568
  }
1976
2569
 
1977
2570
  // src/lib/transaction/transaction-validator.ts
1978
- var logger8 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232, MIN_TRANSACTION_SIZE = 64, TransactionValidator = class {
2571
+ var logger9 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232, MIN_TRANSACTION_SIZE = 64, TransactionValidator = class {
1979
2572
  /**
1980
2573
  * Validate a transaction before signing
1981
2574
  *
@@ -1998,7 +2591,7 @@ var logger8 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232,
1998
2591
  try {
1999
2592
  serialized = transaction.serialize();
2000
2593
  } catch (serializeError) {
2001
- logger8.debug("Transaction not yet serializable (may need signing)", {
2594
+ logger9.debug("Transaction not yet serializable (may need signing)", {
2002
2595
  error: serializeError instanceof Error ? serializeError.message : String(serializeError)
2003
2596
  });
2004
2597
  }
@@ -2008,10 +2601,10 @@ var logger8 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232,
2008
2601
  return errors.push(
2009
2602
  "Transaction type not recognized - must be a Transaction object with serialize() or Uint8Array"
2010
2603
  ), { valid: false, errors, warnings };
2011
- serialized && (size = serialized.length, size > maxSize && (errors.push(`Transaction too large: ${size} bytes (max ${maxSize} bytes)`), logger8.warn("Transaction exceeds maximum size", { size, maxSize })), size < minSize && warnings.push(`Transaction is very small: ${size} bytes (min recommended ${minSize} bytes)`), size === 0 && errors.push("Transaction is empty (0 bytes)"), this.hasSuspiciousPattern(serialized) && warnings.push("Transaction contains unusual patterns - please review carefully"));
2604
+ serialized && (size = serialized.length, size > maxSize && (errors.push(`Transaction too large: ${size} bytes (max ${maxSize} bytes)`), logger9.warn("Transaction exceeds maximum size", { size, maxSize })), size < minSize && warnings.push(`Transaction is very small: ${size} bytes (min recommended ${minSize} bytes)`), size === 0 && errors.push("Transaction is empty (0 bytes)"), this.hasSuspiciousPattern(serialized) && warnings.push("Transaction contains unusual patterns - please review carefully"));
2012
2605
  } catch (error) {
2013
2606
  let errorMessage = error instanceof Error ? error.message : String(error);
2014
- errors.push(`Transaction validation failed: ${errorMessage}`), logger8.error("Validation error", { error: errorMessage });
2607
+ errors.push(`Transaction validation failed: ${errorMessage}`), logger9.error("Validation error", { error: errorMessage });
2015
2608
  }
2016
2609
  if (checkDuplicateSignatures && typeof transaction == "object" && "signatures" in transaction) {
2017
2610
  let signatures = transaction.signatures;
@@ -2024,7 +2617,7 @@ var logger8 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232,
2024
2617
  }
2025
2618
  strict && warnings.length > 0 && (errors.push(...warnings.map((w) => `Strict mode: ${w}`)), warnings.length = 0);
2026
2619
  let valid = errors.length === 0;
2027
- return valid ? warnings.length > 0 ? logger8.debug("Transaction validation passed with warnings", { warnings, size }) : logger8.debug("Transaction validation passed", { size }) : logger8.warn("Transaction validation failed", { errors, size }), {
2620
+ return valid ? warnings.length > 0 ? logger9.debug("Transaction validation passed with warnings", { warnings, size }) : logger9.debug("Transaction validation passed", { size }) : logger9.warn("Transaction validation failed", { errors, size }), {
2028
2621
  valid,
2029
2622
  errors,
2030
2623
  warnings,
@@ -2064,7 +2657,7 @@ var logger8 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232,
2064
2657
  let result = this.validate(transaction, options);
2065
2658
  if (!result.valid)
2066
2659
  throw new Error(`Transaction validation failed: ${result.errors.join(", ")}`);
2067
- result.warnings.length > 0 && logger8.warn("Transaction validation warnings", { warnings: result.warnings });
2660
+ result.warnings.length > 0 && logger9.warn("Transaction validation warnings", { warnings: result.warnings });
2068
2661
  }
2069
2662
  /**
2070
2663
  * Batch validate multiple transactions
@@ -2075,10 +2668,10 @@ var logger8 = createLogger("TransactionValidator"), MAX_TRANSACTION_SIZE = 1232,
2075
2668
  * @returns Array of validation results
2076
2669
  */
2077
2670
  static validateBatch(transactions, options) {
2078
- return transactions.map((tx, index) => (logger8.debug(`Validating transaction ${index + 1}/${transactions.length}`), this.validate(tx, options)));
2671
+ return transactions.map((tx, index) => (logger9.debug(`Validating transaction ${index + 1}/${transactions.length}`), this.validate(tx, options)));
2079
2672
  }
2080
2673
  };
2081
- var logger9 = createLogger("TransactionSigner");
2674
+ var logger10 = createLogger("TransactionSigner");
2082
2675
  function signatureBytesToBase58(bytes) {
2083
2676
  if (bytes.length !== 64)
2084
2677
  throw new Error(`Invalid signature length: expected 64 bytes, got ${bytes.length}`);
@@ -2100,6 +2693,24 @@ function extractSignatureString(result) {
2100
2693
  }
2101
2694
  throw new Error("Unexpected wallet response format for signAndSendTransaction");
2102
2695
  }
2696
+ function extractSignatureBytes(result) {
2697
+ if (result instanceof Uint8Array)
2698
+ return result;
2699
+ if (Array.isArray(result)) {
2700
+ let [first] = result;
2701
+ if (!first)
2702
+ throw new Error("Wallet returned empty results array");
2703
+ return extractSignatureBytes(first);
2704
+ }
2705
+ if (result && typeof result == "object") {
2706
+ let record = result;
2707
+ if ("signature" in record)
2708
+ return extractSignatureBytes(record.signature);
2709
+ if (Array.isArray(record.signatures) && record.signatures.length > 0)
2710
+ return extractSignatureBytes(record.signatures[0]);
2711
+ }
2712
+ throw new Error("Unexpected wallet response format for signMessage");
2713
+ }
2103
2714
  function createTransactionSigner(config) {
2104
2715
  let { wallet, account, cluster, eventEmitter } = config;
2105
2716
  if (!wallet || !account)
@@ -2116,11 +2727,11 @@ function createTransactionSigner(config) {
2116
2727
  throw Errors.featureNotSupported("transaction signing");
2117
2728
  let validation = TransactionValidator.validate(transaction);
2118
2729
  if (!validation.valid)
2119
- throw logger9.error("Transaction validation failed", { errors: validation.errors }), Errors.invalidTransaction(validation.errors.join(", "));
2120
- validation.warnings.length > 0 && logger9.warn("Transaction validation warnings", { warnings: validation.warnings });
2730
+ throw logger10.error("Transaction validation failed", { errors: validation.errors }), Errors.invalidTransaction(validation.errors.join(", "));
2731
+ validation.warnings.length > 0 && logger10.warn("Transaction validation warnings", { warnings: validation.warnings });
2121
2732
  try {
2122
2733
  let signFeature = features["solana:signTransaction"], { serialized, wasWeb3js } = prepareTransactionForWallet(transaction);
2123
- logger9.debug("Signing transaction", {
2734
+ logger10.debug("Signing transaction", {
2124
2735
  wasWeb3js,
2125
2736
  serializedLength: serialized.length,
2126
2737
  serializedType: serialized.constructor.name,
@@ -2129,26 +2740,26 @@ function createTransactionSigner(config) {
2129
2740
  });
2130
2741
  let result, usedFormat = "";
2131
2742
  try {
2132
- logger9.debug("Trying array format: transactions: [Uint8Array]"), result = await signFeature.signTransaction({
2743
+ logger10.debug("Trying array format: transactions: [Uint8Array]"), result = await signFeature.signTransaction({
2133
2744
  account,
2134
2745
  transactions: [serialized],
2135
2746
  ...cluster ? { chain: cluster.id } : {}
2136
2747
  }), usedFormat = "array";
2137
2748
  } catch (err1) {
2138
2749
  let error1 = err1 instanceof Error ? err1 : new Error(String(err1));
2139
- logger9.debug("Array format failed, trying singular format", { error: error1.message });
2750
+ logger10.debug("Array format failed, trying singular format", { error: error1.message });
2140
2751
  try {
2141
- logger9.debug("Trying singular format: transaction: Uint8Array"), result = await signFeature.signTransaction({
2752
+ logger10.debug("Trying singular format: transaction: Uint8Array"), result = await signFeature.signTransaction({
2142
2753
  account,
2143
2754
  transaction: serialized,
2144
2755
  ...cluster ? { chain: cluster.id } : {}
2145
2756
  }), usedFormat = "singular";
2146
2757
  } catch (err2) {
2147
2758
  let error2 = err2 instanceof Error ? err2 : new Error(String(err2));
2148
- throw logger9.error("Both array and singular formats failed", { error: error2.message }), error2;
2759
+ throw logger10.error("Both array and singular formats failed", { error: error2.message }), error2;
2149
2760
  }
2150
2761
  }
2151
- logger9.debug("Wallet signed successfully", { format: usedFormat });
2762
+ logger10.debug("Wallet signed successfully", { format: usedFormat });
2152
2763
  let signedTx;
2153
2764
  if (Array.isArray(result.signedTransactions) && result.signedTransactions[0])
2154
2765
  signedTx = result.signedTransactions[0];
@@ -2160,23 +2771,23 @@ function createTransactionSigner(config) {
2160
2771
  signedTx = result;
2161
2772
  else
2162
2773
  throw new Error(`Unexpected wallet response format: ${JSON.stringify(Object.keys(result))}`);
2163
- if (logger9.debug("Extracted signed transaction", {
2774
+ if (logger10.debug("Extracted signed transaction", {
2164
2775
  hasSignedTx: !!signedTx,
2165
2776
  signedTxType: signedTx?.constructor?.name,
2166
2777
  signedTxLength: signedTx?.length,
2167
2778
  isUint8Array: signedTx instanceof Uint8Array,
2168
2779
  hasSerialize: typeof signedTx?.serialize == "function"
2169
2780
  }), signedTx && typeof signedTx.serialize == "function")
2170
- return logger9.debug("Wallet returned web3.js object directly, no conversion needed"), signedTx;
2781
+ return logger10.debug("Wallet returned web3.js object directly, no conversion needed"), signedTx;
2171
2782
  if (signedTx && signedTx.signedTransaction) {
2172
- logger9.debug("Found signedTransaction property");
2783
+ logger10.debug("Found signedTransaction property");
2173
2784
  let bytes = signedTx.signedTransaction;
2174
2785
  if (bytes instanceof Uint8Array)
2175
2786
  return await convertSignedTransaction(bytes, wasWeb3js);
2176
2787
  }
2177
2788
  if (signedTx instanceof Uint8Array)
2178
2789
  return await convertSignedTransaction(signedTx, wasWeb3js);
2179
- throw logger9.error("Unexpected wallet response format", {
2790
+ throw logger10.error("Unexpected wallet response format", {
2180
2791
  type: typeof signedTx,
2181
2792
  constructor: signedTx?.constructor?.name
2182
2793
  }), new ValidationError(
@@ -2196,9 +2807,15 @@ function createTransactionSigner(config) {
2196
2807
  account,
2197
2808
  transactions: serializedTxs,
2198
2809
  ...cluster ? { chain: cluster.id } : {}
2199
- });
2810
+ }), signedBytesArray;
2811
+ if (Array.isArray(result))
2812
+ signedBytesArray = result.map((item) => item.signedTransaction);
2813
+ else if ("signedTransactions" in result)
2814
+ signedBytesArray = result.signedTransactions;
2815
+ else
2816
+ throw new Error("Unexpected signAllTransactions response format");
2200
2817
  return await Promise.all(
2201
- result.signedTransactions.map(
2818
+ signedBytesArray.map(
2202
2819
  (signedBytes) => convertSignedTransaction(signedBytes, wasWeb3js)
2203
2820
  )
2204
2821
  );
@@ -2292,11 +2909,12 @@ function createTransactionSigner(config) {
2292
2909
  ...capabilities.canSignMessage && {
2293
2910
  async signMessage(message) {
2294
2911
  try {
2295
- return (await features["solana:signMessage"].signMessage({
2912
+ let result = await features["solana:signMessage"].signMessage({
2296
2913
  account,
2297
2914
  message,
2298
2915
  ...cluster ? { chain: cluster.id } : {}
2299
- })).signature;
2916
+ });
2917
+ return extractSignatureBytes(result);
2300
2918
  } catch (error) {
2301
2919
  throw new TransactionError("SIGNING_FAILED", "Failed to sign message", void 0, error);
2302
2920
  }
@@ -2317,7 +2935,7 @@ var TransactionSignerError = class extends TransactionError {
2317
2935
  function isTransactionSignerError(error) {
2318
2936
  return error instanceof TransactionSignerError || error instanceof TransactionError;
2319
2937
  }
2320
- var logger10 = createLogger("KitTransactionSigner");
2938
+ var logger11 = createLogger("KitTransactionSigner");
2321
2939
  function encodeShortVecLength(value) {
2322
2940
  let bytes = [], remaining = value;
2323
2941
  for (; remaining >= 128; )
@@ -2392,7 +3010,7 @@ function createKitTransactionSigner(connectorSigner) {
2392
3010
  async modifyAndSignTransactions(transactions) {
2393
3011
  let transactionData = transactions.map((tx) => {
2394
3012
  let messageBytes = new Uint8Array(tx.messageBytes), { numSigners } = parseMessageSigners(messageBytes), wireFormat = createTransactionBytesForSigning(messageBytes, numSigners);
2395
- return logger10.debug("Preparing wire format for wallet", {
3013
+ return logger11.debug("Preparing wire format for wallet", {
2396
3014
  signerAddress,
2397
3015
  messageBytesLength: messageBytes.length,
2398
3016
  wireFormatLength: wireFormat.length,
@@ -2421,14 +3039,14 @@ function createKitTransactionSigner(connectorSigner) {
2421
3039
  throw new Error("Web3.js transaction without serialize method");
2422
3040
  } else
2423
3041
  throw new Error("Unknown signed transaction format");
2424
- if (logger10.debug("Wallet returned signed transaction", {
3042
+ if (logger11.debug("Wallet returned signed transaction", {
2425
3043
  returnedLength: signedTxBytes.length,
2426
3044
  sentLength: wireFormat.length,
2427
3045
  lengthsMatch: signedTxBytes.length === wireFormat.length,
2428
3046
  signedFirstBytes: Array.from(signedTxBytes.slice(0, 20)),
2429
3047
  sentFirstBytes: Array.from(wireFormat.slice(0, 20))
2430
3048
  }), signedTxBytes.length !== wireFormat.length) {
2431
- logger10.warn("Wallet modified transaction! Using wallet version", {
3049
+ logger11.warn("Wallet modified transaction! Using wallet version", {
2432
3050
  originalLength: wireFormat.length,
2433
3051
  modifiedLength: signedTxBytes.length,
2434
3052
  difference: signedTxBytes.length - wireFormat.length
@@ -2439,13 +3057,13 @@ function createKitTransactionSigner(connectorSigner) {
2439
3057
  lifetimeConstraint: originalWithLifetime.lifetimeConstraint
2440
3058
  } : {}
2441
3059
  };
2442
- return logger10.debug("Using modified transaction from wallet", {
3060
+ return logger11.debug("Using modified transaction from wallet", {
2443
3061
  modifiedMessageBytesLength: walletTransaction.messageBytes.length,
2444
3062
  signatures: Object.keys(walletTransaction.signatures)
2445
3063
  }), assertIsTransactionWithinSizeLimit(result), result;
2446
3064
  }
2447
3065
  let extractedSignatureBytes = extractSignature(signedTxBytes), signatureBase58 = getBase58Decoder().decode(extractedSignatureBytes);
2448
- logger10.debug("Extracted signature from wallet (unmodified)", {
3066
+ logger11.debug("Extracted signature from wallet (unmodified)", {
2449
3067
  signerAddress,
2450
3068
  signatureLength: extractedSignatureBytes.length,
2451
3069
  signatureBase58
@@ -2460,7 +3078,7 @@ function createKitTransactionSigner(connectorSigner) {
2460
3078
  };
2461
3079
  return assertIsTransactionWithinSizeLimit(signedTransaction), signedTransaction;
2462
3080
  } catch (error) {
2463
- return logger10.error("Failed to decode signed transaction", { error }), assertIsTransactionWithinSizeLimit(originalTransaction), originalTransaction;
3081
+ return logger11.error("Failed to decode signed transaction", { error }), assertIsTransactionWithinSizeLimit(originalTransaction), originalTransaction;
2464
3082
  }
2465
3083
  });
2466
3084
  }
@@ -2468,6 +3086,6 @@ function createKitTransactionSigner(connectorSigner) {
2468
3086
  }
2469
3087
  var createGillTransactionSigner = createKitTransactionSigner;
2470
3088
 
2471
- export { ClipboardErrorType, ConnectorClient, ConnectorErrorBoundary, DEFAULT_MAX_RETRIES, PUBLIC_RPC_ENDPOINTS, TransactionSignerError, WalletErrorType, copyAddressToClipboard, copySignatureToClipboard, copyToClipboard, createGillTransactionSigner, createKitTransactionSigner, createTransactionSigner, formatAddress, formatBigIntBalance, formatBigIntUsd, formatLamportsToSolSafe, formatNumber, formatSOL, formatTokenAmount, formatTokenBalanceSafe, getAddressUrl, getBlockUrl, getChainIdForWalletStandard, getClusterChainId, getClusterExplorerUrl, getClusterName, getClusterRpcUrl, getClusterType, getDefaultRpcUrl, getNetworkDisplayName, getPolyfillStatus, getTokenUrl, getTransactionUrl, getWalletsRegistry, installPolyfills, isClipboardAvailable, isCryptoAvailable, isDevnet, isDevnetCluster, isLocalCluster, isLocalnet, isMainnet, isMainnetCluster, isPolyfillInstalled, isTestnet, isTestnetCluster, isTransactionSignerError, normalizeNetwork, ready, toClusterId, truncate, withErrorBoundary };
2472
- //# sourceMappingURL=chunk-MAXA3HEP.mjs.map
2473
- //# sourceMappingURL=chunk-MAXA3HEP.mjs.map
3089
+ export { ClipboardErrorType, ConnectorClient, EnhancedStorage, EnhancedStorageAdapter, INITIAL_WALLET_STATUS, PUBLIC_RPC_ENDPOINTS, TransactionSignerError, WALLET_STATE_VERSION, clearWalletState, copyAddressToClipboard, copySignatureToClipboard, copyToClipboard, createConnectorId, createEnhancedStorageAccount, createEnhancedStorageCluster, createEnhancedStorageWallet, createEnhancedStorageWalletState, createGillTransactionSigner, createKitTransactionSigner, createTransactionSigner, formatAddress, formatBigIntBalance, formatBigIntUsd, formatLamportsToSolSafe, formatNumber, formatSOL, formatTokenAmount, formatTokenBalanceSafe, getAddressUrl, getBlockUrl, getChainIdForWalletStandard, getClusterChainId, getClusterExplorerUrl, getClusterName, getClusterRpcUrl, getClusterType, getDefaultConfig, getDefaultMobileConfig, getDefaultRpcUrl, getNetworkDisplayName, getPolyfillStatus, getTokenUrl, getTransactionUrl, getWalletNameFromConnectorId, getWalletsRegistry, installPolyfills, isClipboardAvailable, isConnected, isConnecting, isCryptoAvailable, isDevnet, isDevnetCluster, isDisconnected, isLocalCluster, isLocalnet, isMainnet, isMainnetCluster, isPolyfillInstalled, isStatusError, isTestnet, isTestnetCluster, isTransactionSignerError, isWalletConnectorId, isWalletStatusError, normalizeNetwork, parseConfigOptions, ready, saveWalletState, toClusterId, toLegacyWalletState, truncate, validateConfigOptions };
3090
+ //# sourceMappingURL=chunk-IDTUFDNB.mjs.map
3091
+ //# sourceMappingURL=chunk-IDTUFDNB.mjs.map