access-layers-wallet 1.0.2
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.
- package/README.md +213 -0
- package/dist/index.d.mts +168 -0
- package/dist/index.d.ts +168 -0
- package/dist/index.js +3392 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +3346 -0
- package/dist/index.mjs.map +1 -0
- package/package.json +80 -0
package/dist/index.mjs
ADDED
|
@@ -0,0 +1,3346 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __defProps = Object.defineProperties;
|
|
4
|
+
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
|
5
|
+
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
6
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
7
|
+
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
|
8
|
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
9
|
+
var __spreadValues = (a, b) => {
|
|
10
|
+
for (var prop in b || (b = {}))
|
|
11
|
+
if (__hasOwnProp.call(b, prop))
|
|
12
|
+
__defNormalProp(a, prop, b[prop]);
|
|
13
|
+
if (__getOwnPropSymbols)
|
|
14
|
+
for (var prop of __getOwnPropSymbols(b)) {
|
|
15
|
+
if (__propIsEnum.call(b, prop))
|
|
16
|
+
__defNormalProp(a, prop, b[prop]);
|
|
17
|
+
}
|
|
18
|
+
return a;
|
|
19
|
+
};
|
|
20
|
+
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
|
|
21
|
+
var __objRest = (source, exclude) => {
|
|
22
|
+
var target = {};
|
|
23
|
+
for (var prop in source)
|
|
24
|
+
if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
|
|
25
|
+
target[prop] = source[prop];
|
|
26
|
+
if (source != null && __getOwnPropSymbols)
|
|
27
|
+
for (var prop of __getOwnPropSymbols(source)) {
|
|
28
|
+
if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
|
|
29
|
+
target[prop] = source[prop];
|
|
30
|
+
}
|
|
31
|
+
return target;
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
// src/components/ConnectWalletHandler.tsx
|
|
35
|
+
import { useEffect as useEffect2, useState as useState2, useCallback as useCallback2 } from "react";
|
|
36
|
+
|
|
37
|
+
// components/ui/dialog.tsx
|
|
38
|
+
import * as React from "react";
|
|
39
|
+
import * as DialogPrimitive from "@radix-ui/react-dialog";
|
|
40
|
+
import { X } from "lucide-react";
|
|
41
|
+
|
|
42
|
+
// lib/utils.ts
|
|
43
|
+
import { clsx } from "clsx";
|
|
44
|
+
import { twMerge } from "tailwind-merge";
|
|
45
|
+
function cn(...inputs) {
|
|
46
|
+
return twMerge(clsx(inputs));
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
// components/ui/dialog.tsx
|
|
50
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
51
|
+
var Dialog = DialogPrimitive.Root;
|
|
52
|
+
var DialogPortal = DialogPrimitive.Portal;
|
|
53
|
+
var DialogOverlay = React.forwardRef((_a, ref) => {
|
|
54
|
+
var _b = _a, { className } = _b, props = __objRest(_b, ["className"]);
|
|
55
|
+
return /* @__PURE__ */ jsx(DialogPortal, { children: /* @__PURE__ */ jsx(
|
|
56
|
+
DialogPrimitive.Overlay,
|
|
57
|
+
__spreadValues({
|
|
58
|
+
ref,
|
|
59
|
+
className: cn(
|
|
60
|
+
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0",
|
|
61
|
+
className
|
|
62
|
+
)
|
|
63
|
+
}, props)
|
|
64
|
+
) });
|
|
65
|
+
});
|
|
66
|
+
DialogOverlay.displayName = DialogPrimitive.Overlay.displayName;
|
|
67
|
+
var DialogContent = React.forwardRef((_a, ref) => {
|
|
68
|
+
var _b = _a, { className, children } = _b, props = __objRest(_b, ["className", "children"]);
|
|
69
|
+
return /* @__PURE__ */ jsxs(DialogPortal, { children: [
|
|
70
|
+
/* @__PURE__ */ jsx(DialogOverlay, {}),
|
|
71
|
+
/* @__PURE__ */ jsxs(
|
|
72
|
+
DialogPrimitive.Content,
|
|
73
|
+
__spreadProps(__spreadValues({
|
|
74
|
+
ref,
|
|
75
|
+
className: cn(
|
|
76
|
+
"fixed left-[50%] top-[50%] z-50 grid w-full max-w-lg translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg",
|
|
77
|
+
className
|
|
78
|
+
)
|
|
79
|
+
}, props), {
|
|
80
|
+
children: [
|
|
81
|
+
children,
|
|
82
|
+
/* @__PURE__ */ jsxs(DialogPrimitive.Close, { className: "absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground", children: [
|
|
83
|
+
/* @__PURE__ */ jsx(X, { className: "h-4 w-4" }),
|
|
84
|
+
/* @__PURE__ */ jsx("span", { className: "sr-only", children: "Close" })
|
|
85
|
+
] })
|
|
86
|
+
]
|
|
87
|
+
})
|
|
88
|
+
)
|
|
89
|
+
] });
|
|
90
|
+
});
|
|
91
|
+
DialogContent.displayName = DialogPrimitive.Content.displayName;
|
|
92
|
+
var DialogHeader = (_a) => {
|
|
93
|
+
var _b = _a, {
|
|
94
|
+
className
|
|
95
|
+
} = _b, props = __objRest(_b, [
|
|
96
|
+
"className"
|
|
97
|
+
]);
|
|
98
|
+
return /* @__PURE__ */ jsx(
|
|
99
|
+
"div",
|
|
100
|
+
__spreadValues({
|
|
101
|
+
className: cn(
|
|
102
|
+
"flex flex-col space-y-1.5 text-center sm:text-left",
|
|
103
|
+
className
|
|
104
|
+
)
|
|
105
|
+
}, props)
|
|
106
|
+
);
|
|
107
|
+
};
|
|
108
|
+
DialogHeader.displayName = "DialogHeader";
|
|
109
|
+
var DialogFooter = (_a) => {
|
|
110
|
+
var _b = _a, {
|
|
111
|
+
className
|
|
112
|
+
} = _b, props = __objRest(_b, [
|
|
113
|
+
"className"
|
|
114
|
+
]);
|
|
115
|
+
return /* @__PURE__ */ jsx(
|
|
116
|
+
"div",
|
|
117
|
+
__spreadValues({
|
|
118
|
+
className: cn(
|
|
119
|
+
"flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2",
|
|
120
|
+
className
|
|
121
|
+
)
|
|
122
|
+
}, props)
|
|
123
|
+
);
|
|
124
|
+
};
|
|
125
|
+
DialogFooter.displayName = "DialogFooter";
|
|
126
|
+
var DialogTitle = React.forwardRef((_a, ref) => {
|
|
127
|
+
var _b = _a, { className } = _b, props = __objRest(_b, ["className"]);
|
|
128
|
+
return /* @__PURE__ */ jsx(
|
|
129
|
+
DialogPrimitive.Title,
|
|
130
|
+
__spreadValues({
|
|
131
|
+
ref,
|
|
132
|
+
className: cn(
|
|
133
|
+
"text-lg font-semibold leading-none tracking-tight",
|
|
134
|
+
className
|
|
135
|
+
)
|
|
136
|
+
}, props)
|
|
137
|
+
);
|
|
138
|
+
});
|
|
139
|
+
DialogTitle.displayName = DialogPrimitive.Title.displayName;
|
|
140
|
+
var DialogDescription = React.forwardRef((_a, ref) => {
|
|
141
|
+
var _b = _a, { className } = _b, props = __objRest(_b, ["className"]);
|
|
142
|
+
return /* @__PURE__ */ jsx(
|
|
143
|
+
DialogPrimitive.Description,
|
|
144
|
+
__spreadValues({
|
|
145
|
+
ref,
|
|
146
|
+
className: cn("text-sm text-muted-foreground", className)
|
|
147
|
+
}, props)
|
|
148
|
+
);
|
|
149
|
+
});
|
|
150
|
+
DialogDescription.displayName = DialogPrimitive.Description.displayName;
|
|
151
|
+
|
|
152
|
+
// src/components/ConnectWalletHandler.tsx
|
|
153
|
+
import { VisuallyHidden } from "@radix-ui/react-visually-hidden";
|
|
154
|
+
import { toast as toast2 } from "sonner";
|
|
155
|
+
|
|
156
|
+
// components/ui/button.tsx
|
|
157
|
+
import * as React2 from "react";
|
|
158
|
+
import { Slot } from "@radix-ui/react-slot";
|
|
159
|
+
import { cva } from "class-variance-authority";
|
|
160
|
+
import { jsx as jsx2 } from "react/jsx-runtime";
|
|
161
|
+
var buttonVariants = cva(
|
|
162
|
+
"inline-flex items-center justify-center whitespace-nowrap rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50",
|
|
163
|
+
{
|
|
164
|
+
variants: {
|
|
165
|
+
variant: {
|
|
166
|
+
default: "bg-primary text-primary-foreground hover:bg-primary/90",
|
|
167
|
+
destructive: "bg-destructive text-destructive-foreground hover:bg-destructive/90",
|
|
168
|
+
outline: "border border-input bg-background hover:bg-accent hover:text-accent-foreground",
|
|
169
|
+
secondary: "bg-secondary text-secondary-foreground hover:bg-secondary/80",
|
|
170
|
+
ghost: "hover:bg-accent hover:text-accent-foreground",
|
|
171
|
+
link: "text-primary underline-offset-4 hover:underline"
|
|
172
|
+
},
|
|
173
|
+
size: {
|
|
174
|
+
default: "h-10 px-4 py-2",
|
|
175
|
+
sm: "h-9 rounded-md px-3",
|
|
176
|
+
lg: "h-11 rounded-md px-8",
|
|
177
|
+
icon: "h-10 w-10"
|
|
178
|
+
}
|
|
179
|
+
},
|
|
180
|
+
defaultVariants: {
|
|
181
|
+
variant: "default",
|
|
182
|
+
size: "default"
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
);
|
|
186
|
+
var Button = React2.forwardRef(
|
|
187
|
+
(_a, ref) => {
|
|
188
|
+
var _b = _a, { className, variant, size, asChild = false } = _b, props = __objRest(_b, ["className", "variant", "size", "asChild"]);
|
|
189
|
+
const Comp = asChild ? Slot : "button";
|
|
190
|
+
return /* @__PURE__ */ jsx2(
|
|
191
|
+
Comp,
|
|
192
|
+
__spreadValues({
|
|
193
|
+
className: cn(buttonVariants({ variant, size, className })),
|
|
194
|
+
ref
|
|
195
|
+
}, props)
|
|
196
|
+
);
|
|
197
|
+
}
|
|
198
|
+
);
|
|
199
|
+
Button.displayName = "Button";
|
|
200
|
+
|
|
201
|
+
// src/components/ConnectWalletHandler.tsx
|
|
202
|
+
import { motion } from "framer-motion";
|
|
203
|
+
import {
|
|
204
|
+
Loader2,
|
|
205
|
+
X as X2,
|
|
206
|
+
ExternalLink
|
|
207
|
+
} from "lucide-react";
|
|
208
|
+
|
|
209
|
+
// hooks/useSupraMultiWallet.ts
|
|
210
|
+
import { useState, useEffect, useCallback } from "react";
|
|
211
|
+
import nacl from "tweetnacl";
|
|
212
|
+
import { useRouter } from "next/navigation";
|
|
213
|
+
import { toast } from "sonner";
|
|
214
|
+
import {
|
|
215
|
+
SupraChainId,
|
|
216
|
+
initSdk
|
|
217
|
+
} from "ribbit-wallet-connect";
|
|
218
|
+
|
|
219
|
+
// lib/logger.ts
|
|
220
|
+
var Logger = class {
|
|
221
|
+
constructor() {
|
|
222
|
+
this.isDevelopment = process.env.NODE_ENV === "development";
|
|
223
|
+
}
|
|
224
|
+
log(level, message, context, error) {
|
|
225
|
+
if (!this.isDevelopment && level === "debug") {
|
|
226
|
+
return;
|
|
227
|
+
}
|
|
228
|
+
const entry = {
|
|
229
|
+
level,
|
|
230
|
+
message,
|
|
231
|
+
timestamp: Date.now(),
|
|
232
|
+
context,
|
|
233
|
+
error
|
|
234
|
+
};
|
|
235
|
+
const logMethod = level === "error" ? console.error : level === "warn" ? console.warn : console.log;
|
|
236
|
+
if (error) {
|
|
237
|
+
logMethod(`[${level.toUpperCase()}] ${message}`, __spreadProps(__spreadValues({}, context), {
|
|
238
|
+
error: {
|
|
239
|
+
name: error.name,
|
|
240
|
+
message: error.message,
|
|
241
|
+
stack: error.stack
|
|
242
|
+
}
|
|
243
|
+
}));
|
|
244
|
+
} else {
|
|
245
|
+
logMethod(`[${level.toUpperCase()}] ${message}`, context || {});
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
debug(message, context) {
|
|
249
|
+
this.log("debug", message, context);
|
|
250
|
+
}
|
|
251
|
+
info(message, context) {
|
|
252
|
+
this.log("info", message, context);
|
|
253
|
+
}
|
|
254
|
+
warn(message, context) {
|
|
255
|
+
this.log("warn", message, context);
|
|
256
|
+
}
|
|
257
|
+
error(message, error, context) {
|
|
258
|
+
this.log("error", message, context, error);
|
|
259
|
+
}
|
|
260
|
+
};
|
|
261
|
+
var logger = new Logger();
|
|
262
|
+
|
|
263
|
+
// lib/errors.ts
|
|
264
|
+
var WalletError = class _WalletError extends Error {
|
|
265
|
+
constructor(message, code, cause) {
|
|
266
|
+
super(message);
|
|
267
|
+
this.code = code;
|
|
268
|
+
this.cause = cause;
|
|
269
|
+
this.name = "WalletError";
|
|
270
|
+
Object.setPrototypeOf(this, _WalletError.prototype);
|
|
271
|
+
}
|
|
272
|
+
};
|
|
273
|
+
var WalletConnectionError = class _WalletConnectionError extends WalletError {
|
|
274
|
+
constructor(message, cause) {
|
|
275
|
+
super(message, "WALLET_CONNECTION_ERROR", cause);
|
|
276
|
+
this.name = "WalletConnectionError";
|
|
277
|
+
Object.setPrototypeOf(this, _WalletConnectionError.prototype);
|
|
278
|
+
}
|
|
279
|
+
};
|
|
280
|
+
var TransactionError = class _TransactionError extends WalletError {
|
|
281
|
+
constructor(message, txHash, cause) {
|
|
282
|
+
super(message, "TRANSACTION_ERROR", cause);
|
|
283
|
+
this.txHash = txHash;
|
|
284
|
+
this.name = "TransactionError";
|
|
285
|
+
Object.setPrototypeOf(this, _TransactionError.prototype);
|
|
286
|
+
}
|
|
287
|
+
};
|
|
288
|
+
var SignMessageError = class _SignMessageError extends WalletError {
|
|
289
|
+
constructor(message, cause) {
|
|
290
|
+
super(message, "SIGN_MESSAGE_ERROR", cause);
|
|
291
|
+
this.name = "SignMessageError";
|
|
292
|
+
Object.setPrototypeOf(this, _SignMessageError.prototype);
|
|
293
|
+
}
|
|
294
|
+
};
|
|
295
|
+
var WalletNotInstalledError = class _WalletNotInstalledError extends WalletError {
|
|
296
|
+
constructor(walletType) {
|
|
297
|
+
super(`Wallet ${walletType} is not installed`, "WALLET_NOT_INSTALLED");
|
|
298
|
+
this.name = "WalletNotInstalledError";
|
|
299
|
+
Object.setPrototypeOf(this, _WalletNotInstalledError.prototype);
|
|
300
|
+
}
|
|
301
|
+
};
|
|
302
|
+
var NetworkError = class _NetworkError extends WalletError {
|
|
303
|
+
constructor(message, cause) {
|
|
304
|
+
super(message, "NETWORK_ERROR", cause);
|
|
305
|
+
this.name = "NetworkError";
|
|
306
|
+
Object.setPrototypeOf(this, _NetworkError.prototype);
|
|
307
|
+
}
|
|
308
|
+
};
|
|
309
|
+
|
|
310
|
+
// lib/storage.ts
|
|
311
|
+
function setStorageItem(key, value, preferredType = "localStorage") {
|
|
312
|
+
const types = preferredType === "localStorage" ? ["localStorage", "sessionStorage", "cookie"] : preferredType === "sessionStorage" ? ["sessionStorage", "localStorage", "cookie"] : ["cookie", "localStorage", "sessionStorage"];
|
|
313
|
+
for (const type of types) {
|
|
314
|
+
try {
|
|
315
|
+
switch (type) {
|
|
316
|
+
case "localStorage":
|
|
317
|
+
if (typeof window !== "undefined" && window.localStorage) {
|
|
318
|
+
window.localStorage.setItem(key, value);
|
|
319
|
+
return true;
|
|
320
|
+
}
|
|
321
|
+
break;
|
|
322
|
+
case "sessionStorage":
|
|
323
|
+
if (typeof window !== "undefined" && window.sessionStorage) {
|
|
324
|
+
window.sessionStorage.setItem(key, value);
|
|
325
|
+
return true;
|
|
326
|
+
}
|
|
327
|
+
break;
|
|
328
|
+
case "cookie":
|
|
329
|
+
if (typeof document !== "undefined") {
|
|
330
|
+
document.cookie = `${key}=${encodeURIComponent(value)}; path=/; max-age=${60 * 60 * 24 * 30}; SameSite=Lax`;
|
|
331
|
+
return true;
|
|
332
|
+
}
|
|
333
|
+
break;
|
|
334
|
+
}
|
|
335
|
+
} catch (error) {
|
|
336
|
+
logger.warn(`Failed to set ${key} in ${type}`, { error });
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
return false;
|
|
340
|
+
}
|
|
341
|
+
function getStorageItem(key) {
|
|
342
|
+
try {
|
|
343
|
+
if (typeof window !== "undefined" && window.localStorage) {
|
|
344
|
+
const value = window.localStorage.getItem(key);
|
|
345
|
+
if (value !== null) return value;
|
|
346
|
+
}
|
|
347
|
+
} catch (error) {
|
|
348
|
+
logger.warn(`Failed to read ${key} from localStorage`, { error });
|
|
349
|
+
}
|
|
350
|
+
try {
|
|
351
|
+
if (typeof window !== "undefined" && window.sessionStorage) {
|
|
352
|
+
const value = window.sessionStorage.getItem(key);
|
|
353
|
+
if (value !== null) return value;
|
|
354
|
+
}
|
|
355
|
+
} catch (error) {
|
|
356
|
+
logger.warn(`Failed to read ${key} from sessionStorage`, { error });
|
|
357
|
+
}
|
|
358
|
+
try {
|
|
359
|
+
if (typeof document !== "undefined") {
|
|
360
|
+
const match = document.cookie.split("; ").find((r) => r.startsWith(key + "="));
|
|
361
|
+
if (match) {
|
|
362
|
+
return decodeURIComponent(match.split("=").slice(1).join("="));
|
|
363
|
+
}
|
|
364
|
+
}
|
|
365
|
+
} catch (error) {
|
|
366
|
+
logger.warn(`Failed to read ${key} from cookie`, { error });
|
|
367
|
+
}
|
|
368
|
+
return null;
|
|
369
|
+
}
|
|
370
|
+
function removeStorageItem(key) {
|
|
371
|
+
try {
|
|
372
|
+
if (typeof window !== "undefined" && window.localStorage) {
|
|
373
|
+
window.localStorage.removeItem(key);
|
|
374
|
+
}
|
|
375
|
+
} catch (error) {
|
|
376
|
+
}
|
|
377
|
+
try {
|
|
378
|
+
if (typeof window !== "undefined" && window.sessionStorage) {
|
|
379
|
+
window.sessionStorage.removeItem(key);
|
|
380
|
+
}
|
|
381
|
+
} catch (error) {
|
|
382
|
+
}
|
|
383
|
+
try {
|
|
384
|
+
if (typeof document !== "undefined") {
|
|
385
|
+
document.cookie = `${key}=; path=/; expires=Thu, 01 Jan 1970 00:00:00 GMT`;
|
|
386
|
+
}
|
|
387
|
+
} catch (error) {
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
// lib/constants.ts
|
|
392
|
+
var STORAGE_KEYS = {
|
|
393
|
+
SELECTED_WALLET: "multiwallet.selectedWallet",
|
|
394
|
+
RECENT_WALLET: "recent_wallet_type",
|
|
395
|
+
USER_PROFILE_CACHE: "user_profile_cache",
|
|
396
|
+
USER_PROFILE_CACHE_TIMESTAMP: "user_profile_cache_timestamp",
|
|
397
|
+
IS_SIGNING_WALLET: "isSigningWallet",
|
|
398
|
+
STARKEY_ACCOUNTS: "starkey.accounts.0"
|
|
399
|
+
};
|
|
400
|
+
var CACHE_TTL = {
|
|
401
|
+
PROFILE: 600 * 1e3,
|
|
402
|
+
// 10 minutes
|
|
403
|
+
BALANCE: 30 * 1e3
|
|
404
|
+
// 30 seconds
|
|
405
|
+
};
|
|
406
|
+
var TIMEOUTS = {
|
|
407
|
+
WALLET_DETECTION_POLL: 3e3,
|
|
408
|
+
// 3 seconds (reduced from 5s)
|
|
409
|
+
WALLET_CHECK_INTERVAL: 2e3,
|
|
410
|
+
// 2 seconds
|
|
411
|
+
WALLET_CHECK_MAX_DURATION: 3e4,
|
|
412
|
+
// 30 seconds
|
|
413
|
+
CONNECTION_TIMEOUT: 3e4,
|
|
414
|
+
// 30 seconds
|
|
415
|
+
BALANCE_POLL_INTERVAL: 3e4,
|
|
416
|
+
// 30 seconds
|
|
417
|
+
MODAL_CLOSE_DELAY: 2500,
|
|
418
|
+
// 2.5 seconds
|
|
419
|
+
CLICK_OUTSIDE_DELAY: 2e3
|
|
420
|
+
// 2 seconds
|
|
421
|
+
};
|
|
422
|
+
var WALLET_DOWNLOAD_URLS = {
|
|
423
|
+
starkey: "https://chromewebstore.google.com/detail/starkey-wallet-the-offici/hcjhpkgbmechpabifbggldplacolbkoh",
|
|
424
|
+
ribbit: "https://ribbitwallet.com"
|
|
425
|
+
};
|
|
426
|
+
var DEFAULT_CHAIN_IDS = {
|
|
427
|
+
TESTNET: "6",
|
|
428
|
+
MAINNET: "8"
|
|
429
|
+
};
|
|
430
|
+
var ERROR_MESSAGES = {
|
|
431
|
+
WALLET_NOT_INSTALLED: "Wallet extension not installed",
|
|
432
|
+
CONNECTION_FAILED: "Failed to connect wallet",
|
|
433
|
+
CONNECTION_REJECTED: "Connection rejected by user",
|
|
434
|
+
NO_ACCOUNT_FOUND: "No account found",
|
|
435
|
+
SIGNING_FAILED: "Message signing failed",
|
|
436
|
+
SIGNING_REJECTED: "Message signing rejected",
|
|
437
|
+
TRANSACTION_FAILED: "Transaction failed",
|
|
438
|
+
TRANSACTION_REJECTED: "Transaction rejected",
|
|
439
|
+
NETWORK_ERROR: "Network error occurred",
|
|
440
|
+
UNSUPPORTED_WALLET: "Unsupported wallet type",
|
|
441
|
+
INVALID_ADDRESS: "Invalid wallet address"
|
|
442
|
+
};
|
|
443
|
+
var CONNECTION_MESSAGES = {
|
|
444
|
+
CONNECTING: "Connecting to wallet...",
|
|
445
|
+
SIGNING: "Please sign the message to verify your account",
|
|
446
|
+
SUCCESS: "Successfully connected",
|
|
447
|
+
ERROR: "Connection failed",
|
|
448
|
+
CONNECTED_NOT_SIGNED: "Connected but signature rejected"
|
|
449
|
+
};
|
|
450
|
+
|
|
451
|
+
// lib/crypto.ts
|
|
452
|
+
function generateNonce() {
|
|
453
|
+
if (typeof window !== "undefined" && window.crypto && window.crypto.getRandomValues) {
|
|
454
|
+
const array = new Uint32Array(1);
|
|
455
|
+
window.crypto.getRandomValues(array);
|
|
456
|
+
return array[0].toString(36) + Date.now().toString(36);
|
|
457
|
+
}
|
|
458
|
+
return Math.random().toString(36).substring(2) + Date.now().toString(36);
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
// lib/config.ts
|
|
462
|
+
var defaultConfig = {
|
|
463
|
+
rpcUrl: {
|
|
464
|
+
testnet: "https://rpc-testnet.supra.com",
|
|
465
|
+
mainnet: "https://rpc-mainnet.supra.com"
|
|
466
|
+
},
|
|
467
|
+
chainId: process.env.NEXT_PUBLIC_SUPRA_CHAIN_ID || DEFAULT_CHAIN_IDS.TESTNET,
|
|
468
|
+
enableLogging: process.env.NODE_ENV === "development",
|
|
469
|
+
balancePollInterval: 3e4,
|
|
470
|
+
walletDetectionTimeout: 3e3
|
|
471
|
+
};
|
|
472
|
+
var config = __spreadValues({}, defaultConfig);
|
|
473
|
+
function initConfig(userConfig) {
|
|
474
|
+
var _a, _b, _c;
|
|
475
|
+
config = {
|
|
476
|
+
rpcUrl: {
|
|
477
|
+
testnet: ((_a = userConfig == null ? void 0 : userConfig.rpcUrl) == null ? void 0 : _a.testnet) || defaultConfig.rpcUrl.testnet,
|
|
478
|
+
mainnet: ((_b = userConfig == null ? void 0 : userConfig.rpcUrl) == null ? void 0 : _b.mainnet) || defaultConfig.rpcUrl.mainnet
|
|
479
|
+
},
|
|
480
|
+
chainId: (userConfig == null ? void 0 : userConfig.chainId) || defaultConfig.chainId,
|
|
481
|
+
enableLogging: (_c = userConfig == null ? void 0 : userConfig.enableLogging) != null ? _c : defaultConfig.enableLogging,
|
|
482
|
+
balancePollInterval: (userConfig == null ? void 0 : userConfig.balancePollInterval) || defaultConfig.balancePollInterval,
|
|
483
|
+
walletDetectionTimeout: (userConfig == null ? void 0 : userConfig.walletDetectionTimeout) || defaultConfig.walletDetectionTimeout
|
|
484
|
+
};
|
|
485
|
+
}
|
|
486
|
+
function getRpcUrl(chainId) {
|
|
487
|
+
const targetChainId = chainId || config.chainId;
|
|
488
|
+
const isMainnet = targetChainId === DEFAULT_CHAIN_IDS.MAINNET;
|
|
489
|
+
return isMainnet ? config.rpcUrl.mainnet : config.rpcUrl.testnet;
|
|
490
|
+
}
|
|
491
|
+
function getChainId() {
|
|
492
|
+
return config.chainId;
|
|
493
|
+
}
|
|
494
|
+
initConfig();
|
|
495
|
+
|
|
496
|
+
// hooks/useSupraMultiWallet.ts
|
|
497
|
+
var WALLET_CONFIGS = {
|
|
498
|
+
starkey: {
|
|
499
|
+
capabilities: {
|
|
500
|
+
signMessage: true,
|
|
501
|
+
accountSwitching: true,
|
|
502
|
+
networkSwitching: true,
|
|
503
|
+
rawTransactions: true,
|
|
504
|
+
eventListeners: true,
|
|
505
|
+
tokenRevalidation: true
|
|
506
|
+
},
|
|
507
|
+
provider: () => {
|
|
508
|
+
var _a;
|
|
509
|
+
return typeof window !== "undefined" && ((_a = window == null ? void 0 : window.starkey) == null ? void 0 : _a.supra);
|
|
510
|
+
}
|
|
511
|
+
},
|
|
512
|
+
ribbit: {
|
|
513
|
+
capabilities: {
|
|
514
|
+
signMessage: true,
|
|
515
|
+
accountSwitching: false,
|
|
516
|
+
// Ribbit doesn't support account switching
|
|
517
|
+
networkSwitching: false,
|
|
518
|
+
// Ribbit network switching happens in-app
|
|
519
|
+
rawTransactions: true,
|
|
520
|
+
eventListeners: false,
|
|
521
|
+
tokenRevalidation: false
|
|
522
|
+
// Ribbit doesn't support token revalidation
|
|
523
|
+
},
|
|
524
|
+
provider: () => initSdk()
|
|
525
|
+
}
|
|
526
|
+
};
|
|
527
|
+
var WALLET_EVENTS = {
|
|
528
|
+
CONNECTED: "wallet-connected",
|
|
529
|
+
PRESIGNED_STATE: "presigned-state",
|
|
530
|
+
POSTSIGNED_STATE: "postsigned-state",
|
|
531
|
+
ERROR: "wallet-error"
|
|
532
|
+
};
|
|
533
|
+
var setStoredWalletType = (walletType) => {
|
|
534
|
+
setStorageItem(STORAGE_KEYS.SELECTED_WALLET, walletType);
|
|
535
|
+
};
|
|
536
|
+
var getStoredWalletType = () => {
|
|
537
|
+
const stored = getStorageItem(STORAGE_KEYS.SELECTED_WALLET);
|
|
538
|
+
if (stored && ["starkey", "ribbit"].includes(stored)) {
|
|
539
|
+
return stored;
|
|
540
|
+
}
|
|
541
|
+
return "starkey";
|
|
542
|
+
};
|
|
543
|
+
var clearStoredWalletType = () => {
|
|
544
|
+
removeStorageItem(STORAGE_KEYS.SELECTED_WALLET);
|
|
545
|
+
};
|
|
546
|
+
var useSupraMultiWallet = () => {
|
|
547
|
+
const router = useRouter();
|
|
548
|
+
const [selectedWallet, setSelectedWallet] = useState("starkey");
|
|
549
|
+
const [walletCapabilities, setWalletCapabilities] = useState(
|
|
550
|
+
WALLET_CONFIGS["starkey"].capabilities
|
|
551
|
+
);
|
|
552
|
+
useEffect(() => {
|
|
553
|
+
const stored = getStoredWalletType();
|
|
554
|
+
if (stored !== "starkey") {
|
|
555
|
+
setSelectedWallet(stored);
|
|
556
|
+
setWalletCapabilities(WALLET_CONFIGS[stored].capabilities);
|
|
557
|
+
}
|
|
558
|
+
}, []);
|
|
559
|
+
const [supraProvider, setSupraProvider] = useState(
|
|
560
|
+
WALLET_CONFIGS.starkey.provider()
|
|
561
|
+
);
|
|
562
|
+
const [ribbitProvider, setRibbitProvider] = useState(
|
|
563
|
+
WALLET_CONFIGS.ribbit.provider()
|
|
564
|
+
);
|
|
565
|
+
const [isExtensionInstalled, setIsExtensionInstalled] = useState(false);
|
|
566
|
+
const [accounts, setAccounts] = useState([]);
|
|
567
|
+
const [networkData, setNetworkData] = useState();
|
|
568
|
+
const [balance, setBalance] = useState("");
|
|
569
|
+
const [loading, setLoading] = useState(false);
|
|
570
|
+
const [justRequestedRelative, setJustRequestedRelative] = useState(false);
|
|
571
|
+
const [transactions, setTransactions] = useState([]);
|
|
572
|
+
const [selectedChainId, setSelectedChainId] = useState("");
|
|
573
|
+
const addTransactions = (hash) => {
|
|
574
|
+
setTransactions((prev) => [{ hash }, ...prev]);
|
|
575
|
+
};
|
|
576
|
+
const getCurrentProvider = () => {
|
|
577
|
+
switch (selectedWallet) {
|
|
578
|
+
case "starkey": {
|
|
579
|
+
return supraProvider;
|
|
580
|
+
}
|
|
581
|
+
case "ribbit": {
|
|
582
|
+
return ribbitProvider;
|
|
583
|
+
}
|
|
584
|
+
default: {
|
|
585
|
+
return supraProvider;
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
};
|
|
589
|
+
const checkExtensionInstalled = useCallback(async () => {
|
|
590
|
+
switch (selectedWallet) {
|
|
591
|
+
case "starkey": {
|
|
592
|
+
const provider = WALLET_CONFIGS.starkey.provider();
|
|
593
|
+
setSupraProvider(provider);
|
|
594
|
+
setIsExtensionInstalled(!!provider);
|
|
595
|
+
return !!provider;
|
|
596
|
+
}
|
|
597
|
+
case "ribbit": {
|
|
598
|
+
const provider = WALLET_CONFIGS.ribbit.provider();
|
|
599
|
+
if (!provider) {
|
|
600
|
+
setRibbitProvider(null);
|
|
601
|
+
setIsExtensionInstalled(false);
|
|
602
|
+
return false;
|
|
603
|
+
}
|
|
604
|
+
try {
|
|
605
|
+
setRibbitProvider(provider);
|
|
606
|
+
setIsExtensionInstalled(true);
|
|
607
|
+
return true;
|
|
608
|
+
} catch (error) {
|
|
609
|
+
logger.error("Error checking Ribbit wallet readiness", error);
|
|
610
|
+
setRibbitProvider(null);
|
|
611
|
+
setIsExtensionInstalled(false);
|
|
612
|
+
return false;
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
default: {
|
|
616
|
+
return false;
|
|
617
|
+
}
|
|
618
|
+
}
|
|
619
|
+
}, [selectedWallet]);
|
|
620
|
+
const updateAccounts = useCallback(async () => {
|
|
621
|
+
const provider = getCurrentProvider();
|
|
622
|
+
if (!provider) return;
|
|
623
|
+
try {
|
|
624
|
+
switch (selectedWallet) {
|
|
625
|
+
case "starkey": {
|
|
626
|
+
const responseAcc = await provider.account();
|
|
627
|
+
const newAccounts = responseAcc.length > 0 ? responseAcc : [];
|
|
628
|
+
setAccounts(newAccounts);
|
|
629
|
+
if (responseAcc.length > 0) {
|
|
630
|
+
setStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS, responseAcc[0]);
|
|
631
|
+
}
|
|
632
|
+
if (newAccounts.length > 0) {
|
|
633
|
+
const balance2 = await provider.balance();
|
|
634
|
+
if (balance2) {
|
|
635
|
+
setBalance(`${balance2.formattedBalance} ${balance2.displayUnit}`);
|
|
636
|
+
}
|
|
637
|
+
const networkData2 = await provider.getChainId();
|
|
638
|
+
setNetworkData(networkData2 || {});
|
|
639
|
+
}
|
|
640
|
+
break;
|
|
641
|
+
}
|
|
642
|
+
case "ribbit": {
|
|
643
|
+
const wallet = provider.getWalletInfo();
|
|
644
|
+
if (wallet == null ? void 0 : wallet.connected) {
|
|
645
|
+
setAccounts([wallet.walletAddress]);
|
|
646
|
+
const walletBalanceRequest = {
|
|
647
|
+
chainId: parseInt(getChainId()),
|
|
648
|
+
resourceType: "0x1::supra_coin::SupraCoin",
|
|
649
|
+
decimals: 8
|
|
650
|
+
};
|
|
651
|
+
const balanceStr = await provider.getWalletBalance(walletBalanceRequest);
|
|
652
|
+
logger.debug("Ribbit balance response", { balance: balanceStr });
|
|
653
|
+
setBalance(`${(balanceStr == null ? void 0 : balanceStr.balance) || 0} SUPRA`);
|
|
654
|
+
} else {
|
|
655
|
+
setAccounts([]);
|
|
656
|
+
}
|
|
657
|
+
break;
|
|
658
|
+
}
|
|
659
|
+
default: {
|
|
660
|
+
setAccounts([]);
|
|
661
|
+
break;
|
|
662
|
+
}
|
|
663
|
+
}
|
|
664
|
+
} catch (error) {
|
|
665
|
+
logger.error("Failed to update accounts", error, { walletType: selectedWallet });
|
|
666
|
+
setAccounts([]);
|
|
667
|
+
switch (selectedWallet) {
|
|
668
|
+
case "starkey": {
|
|
669
|
+
removeStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS);
|
|
670
|
+
break;
|
|
671
|
+
}
|
|
672
|
+
case "ribbit": {
|
|
673
|
+
break;
|
|
674
|
+
}
|
|
675
|
+
}
|
|
676
|
+
}
|
|
677
|
+
}, [selectedWallet]);
|
|
678
|
+
useEffect(() => {
|
|
679
|
+
let mounted = true;
|
|
680
|
+
const initProvider = async () => {
|
|
681
|
+
const isInstalled = await checkExtensionInstalled();
|
|
682
|
+
if (mounted && isInstalled) {
|
|
683
|
+
updateAccounts();
|
|
684
|
+
}
|
|
685
|
+
};
|
|
686
|
+
initProvider();
|
|
687
|
+
return () => {
|
|
688
|
+
mounted = false;
|
|
689
|
+
};
|
|
690
|
+
}, [selectedWallet, checkExtensionInstalled, updateAccounts]);
|
|
691
|
+
useEffect(() => {
|
|
692
|
+
const checkExtension = async () => {
|
|
693
|
+
return await checkExtensionInstalled();
|
|
694
|
+
};
|
|
695
|
+
checkExtension().then((isInstalled) => {
|
|
696
|
+
if (isInstalled && selectedWallet === "ribbit") {
|
|
697
|
+
updateAccounts();
|
|
698
|
+
}
|
|
699
|
+
});
|
|
700
|
+
const intv = setInterval(async () => {
|
|
701
|
+
const isInstalled = await checkExtension();
|
|
702
|
+
if (isInstalled) {
|
|
703
|
+
clearInterval(intv);
|
|
704
|
+
if (selectedWallet === "ribbit") {
|
|
705
|
+
updateAccounts();
|
|
706
|
+
}
|
|
707
|
+
}
|
|
708
|
+
}, 1e3);
|
|
709
|
+
setTimeout(() => {
|
|
710
|
+
clearInterval(intv);
|
|
711
|
+
}, TIMEOUTS.WALLET_DETECTION_POLL);
|
|
712
|
+
return () => clearInterval(intv);
|
|
713
|
+
}, [selectedWallet, updateAccounts, checkExtensionInstalled]);
|
|
714
|
+
const checkIsExtensionInstalled = useCallback(() => {
|
|
715
|
+
const intervalId = setInterval(async () => {
|
|
716
|
+
const isInstalled = await checkExtensionInstalled();
|
|
717
|
+
if (isInstalled) {
|
|
718
|
+
clearInterval(intervalId);
|
|
719
|
+
updateAccounts();
|
|
720
|
+
}
|
|
721
|
+
}, 500);
|
|
722
|
+
setTimeout(() => clearInterval(intervalId), TIMEOUTS.WALLET_DETECTION_POLL);
|
|
723
|
+
}, [updateAccounts, checkExtensionInstalled]);
|
|
724
|
+
const updateBalance = useCallback(async () => {
|
|
725
|
+
const provider = getCurrentProvider();
|
|
726
|
+
if (!provider || !accounts.length) {
|
|
727
|
+
setBalance("");
|
|
728
|
+
return;
|
|
729
|
+
}
|
|
730
|
+
try {
|
|
731
|
+
switch (selectedWallet) {
|
|
732
|
+
case "starkey": {
|
|
733
|
+
const balance2 = await provider.balance();
|
|
734
|
+
if (balance2) {
|
|
735
|
+
setBalance(`${balance2.formattedBalance} ${balance2.displayUnit}`);
|
|
736
|
+
}
|
|
737
|
+
break;
|
|
738
|
+
}
|
|
739
|
+
case "ribbit": {
|
|
740
|
+
const walletBalanceRequest = {
|
|
741
|
+
chainId: parseInt(getChainId()),
|
|
742
|
+
resourceType: "0x1::supra_coin::SupraCoin",
|
|
743
|
+
decimals: 8
|
|
744
|
+
};
|
|
745
|
+
const balanceStr = await provider.getWalletBalance(
|
|
746
|
+
walletBalanceRequest
|
|
747
|
+
);
|
|
748
|
+
logger.debug("Ribbit balance response", { balance: balanceStr });
|
|
749
|
+
setBalance(`${(balanceStr == null ? void 0 : balanceStr.balance) || 0} SUPRA`);
|
|
750
|
+
break;
|
|
751
|
+
}
|
|
752
|
+
default: {
|
|
753
|
+
setBalance("");
|
|
754
|
+
break;
|
|
755
|
+
}
|
|
756
|
+
}
|
|
757
|
+
} catch (error) {
|
|
758
|
+
logger.error("Failed to update balance", error, { walletType: selectedWallet });
|
|
759
|
+
setBalance("");
|
|
760
|
+
}
|
|
761
|
+
}, [selectedWallet, accounts]);
|
|
762
|
+
const getNetworkData = useCallback(async () => {
|
|
763
|
+
const provider = getCurrentProvider();
|
|
764
|
+
if (!provider) return {};
|
|
765
|
+
try {
|
|
766
|
+
switch (selectedWallet) {
|
|
767
|
+
case "starkey": {
|
|
768
|
+
const data = await provider.getChainId();
|
|
769
|
+
setNetworkData(data || {});
|
|
770
|
+
return data || {};
|
|
771
|
+
}
|
|
772
|
+
case "ribbit": {
|
|
773
|
+
const chainId = parseInt(getChainId());
|
|
774
|
+
const mockNetworkData = { chainId: chainId.toString() };
|
|
775
|
+
setNetworkData(mockNetworkData);
|
|
776
|
+
return mockNetworkData;
|
|
777
|
+
}
|
|
778
|
+
default: {
|
|
779
|
+
setNetworkData({});
|
|
780
|
+
return {};
|
|
781
|
+
}
|
|
782
|
+
}
|
|
783
|
+
} catch (error) {
|
|
784
|
+
logger.error("Failed to get network data", error, { walletType: selectedWallet });
|
|
785
|
+
setNetworkData({});
|
|
786
|
+
return {};
|
|
787
|
+
}
|
|
788
|
+
}, [selectedWallet]);
|
|
789
|
+
const connectWallet = async (walletType) => {
|
|
790
|
+
if (walletType) {
|
|
791
|
+
updateSelectedWallet(walletType);
|
|
792
|
+
}
|
|
793
|
+
const provider = walletType ? WALLET_CONFIGS[walletType].provider() : getCurrentProvider();
|
|
794
|
+
if (!provider) {
|
|
795
|
+
const error = new WalletNotInstalledError(walletType || selectedWallet);
|
|
796
|
+
logger.warn("Wallet not installed", { walletType: walletType || selectedWallet });
|
|
797
|
+
toast.error(ERROR_MESSAGES.WALLET_NOT_INSTALLED, {
|
|
798
|
+
description: `Please install the ${walletType || selectedWallet} extension`
|
|
799
|
+
});
|
|
800
|
+
return false;
|
|
801
|
+
}
|
|
802
|
+
setLoading(true);
|
|
803
|
+
try {
|
|
804
|
+
switch (walletType || selectedWallet) {
|
|
805
|
+
case "starkey": {
|
|
806
|
+
await provider.connect();
|
|
807
|
+
logger.info("Starkey wallet connection approved");
|
|
808
|
+
await updateAccounts();
|
|
809
|
+
const responseAcc = await provider.account();
|
|
810
|
+
if (responseAcc.length === 0) {
|
|
811
|
+
throw new WalletConnectionError(ERROR_MESSAGES.NO_ACCOUNT_FOUND);
|
|
812
|
+
}
|
|
813
|
+
if (responseAcc.length) {
|
|
814
|
+
setAccounts(responseAcc);
|
|
815
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
816
|
+
setStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS, responseAcc[0]);
|
|
817
|
+
window.dispatchEvent(
|
|
818
|
+
new CustomEvent(WALLET_EVENTS.PRESIGNED_STATE, {
|
|
819
|
+
detail: {
|
|
820
|
+
timestamp: Date.now(),
|
|
821
|
+
account: responseAcc[0]
|
|
822
|
+
}
|
|
823
|
+
})
|
|
824
|
+
);
|
|
825
|
+
window.dispatchEvent(
|
|
826
|
+
new CustomEvent(WALLET_EVENTS.CONNECTED, {
|
|
827
|
+
detail: {
|
|
828
|
+
timestamp: Date.now(),
|
|
829
|
+
account: responseAcc[0],
|
|
830
|
+
wallet: "starkey"
|
|
831
|
+
}
|
|
832
|
+
})
|
|
833
|
+
);
|
|
834
|
+
try {
|
|
835
|
+
const message = "To verify your account, please sign this message.";
|
|
836
|
+
const nonce = generateNonce();
|
|
837
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
838
|
+
logger.debug("Signature request initiated", { account: responseAcc[0] });
|
|
839
|
+
const signatureResponse = await provider.signMessage({
|
|
840
|
+
message: hexMessage,
|
|
841
|
+
nonce
|
|
842
|
+
});
|
|
843
|
+
if (signatureResponse) {
|
|
844
|
+
logger.info("Signature approved", { account: responseAcc[0] });
|
|
845
|
+
window.dispatchEvent(
|
|
846
|
+
new CustomEvent(WALLET_EVENTS.POSTSIGNED_STATE, {
|
|
847
|
+
detail: {
|
|
848
|
+
timestamp: Date.now(),
|
|
849
|
+
account: responseAcc[0]
|
|
850
|
+
}
|
|
851
|
+
})
|
|
852
|
+
);
|
|
853
|
+
}
|
|
854
|
+
} catch (signError) {
|
|
855
|
+
const error = new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, signError);
|
|
856
|
+
logger.error("Signing failed", error, { account: responseAcc[0] });
|
|
857
|
+
window.dispatchEvent(
|
|
858
|
+
new CustomEvent(WALLET_EVENTS.ERROR, {
|
|
859
|
+
detail: {
|
|
860
|
+
timestamp: Date.now(),
|
|
861
|
+
error
|
|
862
|
+
}
|
|
863
|
+
})
|
|
864
|
+
);
|
|
865
|
+
throw error;
|
|
866
|
+
}
|
|
867
|
+
}
|
|
868
|
+
break;
|
|
869
|
+
}
|
|
870
|
+
case "ribbit": {
|
|
871
|
+
const dappMetadata = {
|
|
872
|
+
name: "multiwallet",
|
|
873
|
+
description: "NFT Marketplace and Lootbox Platform",
|
|
874
|
+
logo: window.location.origin + "/favicon.ico",
|
|
875
|
+
url: window.location.origin
|
|
876
|
+
};
|
|
877
|
+
const response = await provider.connectToWallet(
|
|
878
|
+
dappMetadata
|
|
879
|
+
);
|
|
880
|
+
if (response == null ? void 0 : response.connected) {
|
|
881
|
+
logger.info("Ribbit wallet connection approved");
|
|
882
|
+
}
|
|
883
|
+
if (response.walletAddress == null) {
|
|
884
|
+
throw new WalletConnectionError(ERROR_MESSAGES.NO_ACCOUNT_FOUND);
|
|
885
|
+
}
|
|
886
|
+
if (response == null ? void 0 : response.connected) {
|
|
887
|
+
await updateAccounts();
|
|
888
|
+
if (response.walletAddress) {
|
|
889
|
+
setAccounts([response.walletAddress]);
|
|
890
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
891
|
+
window.dispatchEvent(
|
|
892
|
+
new CustomEvent(WALLET_EVENTS.PRESIGNED_STATE, {
|
|
893
|
+
detail: {
|
|
894
|
+
timestamp: Date.now(),
|
|
895
|
+
account: response.walletAddress
|
|
896
|
+
}
|
|
897
|
+
})
|
|
898
|
+
);
|
|
899
|
+
window.dispatchEvent(
|
|
900
|
+
new CustomEvent(WALLET_EVENTS.CONNECTED, {
|
|
901
|
+
detail: {
|
|
902
|
+
timestamp: Date.now(),
|
|
903
|
+
account: response.walletAddress,
|
|
904
|
+
wallet: "ribbit"
|
|
905
|
+
}
|
|
906
|
+
})
|
|
907
|
+
);
|
|
908
|
+
try {
|
|
909
|
+
const message = "To verify your account, please sign this message.";
|
|
910
|
+
const nonce = generateNonce();
|
|
911
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
912
|
+
logger.debug("Signature request initiated", { account: response.walletAddress });
|
|
913
|
+
const signatureResponse = await provider.signMessage({
|
|
914
|
+
message: hexMessage,
|
|
915
|
+
nonce: parseInt(nonce),
|
|
916
|
+
chainId: parseInt(getChainId())
|
|
917
|
+
});
|
|
918
|
+
if (signatureResponse && signatureResponse.approved) {
|
|
919
|
+
logger.info("Signature approved", { account: response.walletAddress });
|
|
920
|
+
window.dispatchEvent(
|
|
921
|
+
new CustomEvent(WALLET_EVENTS.POSTSIGNED_STATE, {
|
|
922
|
+
detail: {
|
|
923
|
+
timestamp: Date.now(),
|
|
924
|
+
account: response.walletAddress
|
|
925
|
+
}
|
|
926
|
+
})
|
|
927
|
+
);
|
|
928
|
+
} else {
|
|
929
|
+
throw new SignMessageError(signatureResponse.error || ERROR_MESSAGES.SIGNING_REJECTED);
|
|
930
|
+
}
|
|
931
|
+
} catch (signError) {
|
|
932
|
+
const error = new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, signError);
|
|
933
|
+
logger.error("Signing failed", error, { account: response.walletAddress });
|
|
934
|
+
window.dispatchEvent(
|
|
935
|
+
new CustomEvent(WALLET_EVENTS.ERROR, {
|
|
936
|
+
detail: {
|
|
937
|
+
timestamp: Date.now(),
|
|
938
|
+
error
|
|
939
|
+
}
|
|
940
|
+
})
|
|
941
|
+
);
|
|
942
|
+
throw error;
|
|
943
|
+
}
|
|
944
|
+
}
|
|
945
|
+
} else {
|
|
946
|
+
throw new WalletConnectionError(ERROR_MESSAGES.CONNECTION_REJECTED);
|
|
947
|
+
}
|
|
948
|
+
break;
|
|
949
|
+
}
|
|
950
|
+
default: {
|
|
951
|
+
throw new WalletConnectionError(
|
|
952
|
+
`${ERROR_MESSAGES.UNSUPPORTED_WALLET}: ${walletType || selectedWallet}`
|
|
953
|
+
);
|
|
954
|
+
}
|
|
955
|
+
}
|
|
956
|
+
return true;
|
|
957
|
+
} catch (error) {
|
|
958
|
+
const walletError = error instanceof WalletConnectionError ? error : new WalletConnectionError(ERROR_MESSAGES.CONNECTION_FAILED, error);
|
|
959
|
+
logger.error("Wallet connection failed", walletError, {
|
|
960
|
+
walletType: walletType || selectedWallet
|
|
961
|
+
});
|
|
962
|
+
window.dispatchEvent(
|
|
963
|
+
new CustomEvent(WALLET_EVENTS.ERROR, {
|
|
964
|
+
detail: {
|
|
965
|
+
timestamp: Date.now(),
|
|
966
|
+
error: walletError
|
|
967
|
+
}
|
|
968
|
+
})
|
|
969
|
+
);
|
|
970
|
+
return false;
|
|
971
|
+
} finally {
|
|
972
|
+
setLoading(false);
|
|
973
|
+
}
|
|
974
|
+
};
|
|
975
|
+
const disconnectWallet = async () => {
|
|
976
|
+
const provider = getCurrentProvider();
|
|
977
|
+
if (!provider) return;
|
|
978
|
+
try {
|
|
979
|
+
switch (selectedWallet) {
|
|
980
|
+
case "starkey": {
|
|
981
|
+
await provider.disconnect();
|
|
982
|
+
break;
|
|
983
|
+
}
|
|
984
|
+
case "ribbit": {
|
|
985
|
+
await provider.disconnect();
|
|
986
|
+
break;
|
|
987
|
+
}
|
|
988
|
+
}
|
|
989
|
+
resetWalletData();
|
|
990
|
+
clearStoredWalletType();
|
|
991
|
+
router.push("/");
|
|
992
|
+
} catch (error) {
|
|
993
|
+
logger.error("Wallet disconnect failed", error, { walletType: selectedWallet });
|
|
994
|
+
resetWalletData();
|
|
995
|
+
clearStoredWalletType();
|
|
996
|
+
}
|
|
997
|
+
};
|
|
998
|
+
const resetWalletData = () => {
|
|
999
|
+
setAccounts([]);
|
|
1000
|
+
setBalance("");
|
|
1001
|
+
setNetworkData({});
|
|
1002
|
+
switch (selectedWallet) {
|
|
1003
|
+
case "starkey": {
|
|
1004
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
1005
|
+
removeStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS);
|
|
1006
|
+
break;
|
|
1007
|
+
}
|
|
1008
|
+
case "ribbit": {
|
|
1009
|
+
break;
|
|
1010
|
+
}
|
|
1011
|
+
}
|
|
1012
|
+
};
|
|
1013
|
+
const getSequenceNumber = async (address) => {
|
|
1014
|
+
const rpcUrl = getRpcUrl();
|
|
1015
|
+
const data = await fetch(`${rpcUrl}/rpc/v1/accounts/${address}`);
|
|
1016
|
+
if (!data.ok) {
|
|
1017
|
+
throw new NetworkError(`Failed to fetch sequence number for ${address}`);
|
|
1018
|
+
}
|
|
1019
|
+
const accountData = await data.json();
|
|
1020
|
+
return accountData.sequence_number;
|
|
1021
|
+
};
|
|
1022
|
+
const sendRawTransaction = async (moduleAddress, moduleName, functionName, params, runTimeParams = [], txExpiryTime = Math.ceil(Date.now() / 1e3) + 3e3) => {
|
|
1023
|
+
const provider = getCurrentProvider();
|
|
1024
|
+
if (!provider || !accounts.length || !moduleAddress || !moduleName || !functionName)
|
|
1025
|
+
return;
|
|
1026
|
+
try {
|
|
1027
|
+
switch (selectedWallet) {
|
|
1028
|
+
case "starkey": {
|
|
1029
|
+
if (!walletCapabilities.rawTransactions) {
|
|
1030
|
+
throw new TransactionError("Raw transactions not supported by current wallet");
|
|
1031
|
+
}
|
|
1032
|
+
let networkData2 = await getNetworkData();
|
|
1033
|
+
const currentChainId = getChainId();
|
|
1034
|
+
if (networkData2.chainId !== currentChainId) {
|
|
1035
|
+
setSelectedChainId(currentChainId);
|
|
1036
|
+
await provider.changeNetwork({
|
|
1037
|
+
chainId: currentChainId
|
|
1038
|
+
});
|
|
1039
|
+
}
|
|
1040
|
+
const rawTxPayload = [
|
|
1041
|
+
accounts[0],
|
|
1042
|
+
0,
|
|
1043
|
+
// sequence number
|
|
1044
|
+
moduleAddress,
|
|
1045
|
+
moduleName,
|
|
1046
|
+
functionName,
|
|
1047
|
+
runTimeParams,
|
|
1048
|
+
params,
|
|
1049
|
+
{}
|
|
1050
|
+
];
|
|
1051
|
+
const data = await provider.createRawTransactionData(rawTxPayload);
|
|
1052
|
+
const txHash = await provider.sendTransaction({
|
|
1053
|
+
data,
|
|
1054
|
+
from: accounts[0],
|
|
1055
|
+
to: moduleAddress,
|
|
1056
|
+
chainId: currentChainId,
|
|
1057
|
+
value: ""
|
|
1058
|
+
});
|
|
1059
|
+
addTransactions(txHash || "failed");
|
|
1060
|
+
logger.info("Transaction sent successfully", { txHash, walletType: "starkey" });
|
|
1061
|
+
return txHash;
|
|
1062
|
+
}
|
|
1063
|
+
case "ribbit": {
|
|
1064
|
+
if (!walletCapabilities.rawTransactions) {
|
|
1065
|
+
throw new TransactionError("Raw transactions not supported by current wallet");
|
|
1066
|
+
}
|
|
1067
|
+
const currentChainId = getChainId();
|
|
1068
|
+
let chainId = SupraChainId.TESTNET;
|
|
1069
|
+
if (currentChainId === DEFAULT_CHAIN_IDS.TESTNET) {
|
|
1070
|
+
chainId = SupraChainId.TESTNET;
|
|
1071
|
+
} else if (currentChainId === DEFAULT_CHAIN_IDS.MAINNET) {
|
|
1072
|
+
chainId = SupraChainId.MAINNET;
|
|
1073
|
+
}
|
|
1074
|
+
const rawTxnRequest = {
|
|
1075
|
+
sender: accounts[0],
|
|
1076
|
+
// Use actual sender address
|
|
1077
|
+
moduleAddress,
|
|
1078
|
+
// Use provided module address
|
|
1079
|
+
moduleName,
|
|
1080
|
+
// Use provided module name
|
|
1081
|
+
functionName,
|
|
1082
|
+
// Use provided function name
|
|
1083
|
+
typeArgs: runTimeParams,
|
|
1084
|
+
// Use converted runtime parameters
|
|
1085
|
+
args: params || [],
|
|
1086
|
+
// Use provided parameters
|
|
1087
|
+
chainId
|
|
1088
|
+
};
|
|
1089
|
+
const rawTxnBase64 = await provider.createRawTransactionBuffer(rawTxnRequest);
|
|
1090
|
+
const response = await provider.signAndSendRawTransaction({
|
|
1091
|
+
rawTxn: rawTxnBase64,
|
|
1092
|
+
chainId,
|
|
1093
|
+
meta: {
|
|
1094
|
+
description: `Call ${moduleName}::${functionName}`
|
|
1095
|
+
// Dynamic description
|
|
1096
|
+
}
|
|
1097
|
+
});
|
|
1098
|
+
if (response.approved) {
|
|
1099
|
+
const txHash = response.txHash || response.result || "success";
|
|
1100
|
+
addTransactions(txHash);
|
|
1101
|
+
logger.info("Transaction sent successfully", { txHash, walletType: "ribbit" });
|
|
1102
|
+
return txHash;
|
|
1103
|
+
} else {
|
|
1104
|
+
throw new TransactionError(response.error || ERROR_MESSAGES.TRANSACTION_REJECTED);
|
|
1105
|
+
}
|
|
1106
|
+
}
|
|
1107
|
+
default: {
|
|
1108
|
+
throw new TransactionError(
|
|
1109
|
+
`Raw transactions not supported for wallet: ${selectedWallet}`
|
|
1110
|
+
);
|
|
1111
|
+
}
|
|
1112
|
+
}
|
|
1113
|
+
} catch (error) {
|
|
1114
|
+
const txError = error instanceof TransactionError ? error : new TransactionError(ERROR_MESSAGES.TRANSACTION_FAILED, void 0, error);
|
|
1115
|
+
logger.error("Transaction failed", txError, {
|
|
1116
|
+
walletType: selectedWallet,
|
|
1117
|
+
moduleAddress,
|
|
1118
|
+
moduleName,
|
|
1119
|
+
functionName
|
|
1120
|
+
});
|
|
1121
|
+
throw txError;
|
|
1122
|
+
}
|
|
1123
|
+
};
|
|
1124
|
+
const signMessage = async (message, nonce, account, forceSign = false) => {
|
|
1125
|
+
const provider = getCurrentProvider();
|
|
1126
|
+
if (!provider) return;
|
|
1127
|
+
const secureNonce = nonce || generateNonce();
|
|
1128
|
+
switch (selectedWallet) {
|
|
1129
|
+
case "starkey": {
|
|
1130
|
+
if (!walletCapabilities.signMessage) {
|
|
1131
|
+
throw new SignMessageError("Message signing not supported by current wallet");
|
|
1132
|
+
}
|
|
1133
|
+
if (!accounts.length && !account) return;
|
|
1134
|
+
if (!accounts.length && account) {
|
|
1135
|
+
accounts[0] = account;
|
|
1136
|
+
}
|
|
1137
|
+
if (getStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET) === "true" && !forceSign) {
|
|
1138
|
+
return;
|
|
1139
|
+
}
|
|
1140
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "true");
|
|
1141
|
+
try {
|
|
1142
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
1143
|
+
const response = await provider.signMessage({
|
|
1144
|
+
message: hexMessage,
|
|
1145
|
+
nonce: secureNonce
|
|
1146
|
+
});
|
|
1147
|
+
const { publicKey, signature } = response;
|
|
1148
|
+
const verified = nacl.sign.detached.verify(
|
|
1149
|
+
new TextEncoder().encode(message),
|
|
1150
|
+
Uint8Array.from(Buffer.from(signature.slice(2), "hex")),
|
|
1151
|
+
Uint8Array.from(Buffer.from(publicKey.slice(2), "hex"))
|
|
1152
|
+
);
|
|
1153
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
1154
|
+
logger.info("Message signed successfully", { walletType: "starkey" });
|
|
1155
|
+
return __spreadProps(__spreadValues({}, response), { verified });
|
|
1156
|
+
} catch (error) {
|
|
1157
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
1158
|
+
throw new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, error);
|
|
1159
|
+
}
|
|
1160
|
+
}
|
|
1161
|
+
case "ribbit": {
|
|
1162
|
+
if (!walletCapabilities.signMessage) {
|
|
1163
|
+
throw new SignMessageError("Message signing not supported by current wallet");
|
|
1164
|
+
}
|
|
1165
|
+
if (!accounts.length && !account) return;
|
|
1166
|
+
if (!accounts.length && account) {
|
|
1167
|
+
accounts[0] = account;
|
|
1168
|
+
}
|
|
1169
|
+
if (getStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET) === "true" && !forceSign) {
|
|
1170
|
+
return;
|
|
1171
|
+
}
|
|
1172
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "true");
|
|
1173
|
+
try {
|
|
1174
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
1175
|
+
const response = await provider.signMessage({
|
|
1176
|
+
message: hexMessage,
|
|
1177
|
+
nonce: parseInt(secureNonce),
|
|
1178
|
+
chainId: parseInt(getChainId())
|
|
1179
|
+
});
|
|
1180
|
+
if (response.approved && response.publicKey && response.signature) {
|
|
1181
|
+
const { publicKey, signature } = response;
|
|
1182
|
+
const verified = nacl.sign.detached.verify(
|
|
1183
|
+
new TextEncoder().encode(message),
|
|
1184
|
+
Uint8Array.from(Buffer.from(signature.slice(2), "hex")),
|
|
1185
|
+
Uint8Array.from(Buffer.from(publicKey.slice(2), "hex"))
|
|
1186
|
+
);
|
|
1187
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
1188
|
+
logger.info("Message signed successfully", { walletType: "ribbit" });
|
|
1189
|
+
return __spreadProps(__spreadValues({}, response), { verified });
|
|
1190
|
+
} else {
|
|
1191
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
1192
|
+
throw new SignMessageError(response.error || ERROR_MESSAGES.SIGNING_REJECTED);
|
|
1193
|
+
}
|
|
1194
|
+
} catch (error) {
|
|
1195
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
1196
|
+
throw new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, error);
|
|
1197
|
+
}
|
|
1198
|
+
}
|
|
1199
|
+
default: {
|
|
1200
|
+
throw new SignMessageError(
|
|
1201
|
+
`Message signing not supported for wallet: ${selectedWallet}`
|
|
1202
|
+
);
|
|
1203
|
+
}
|
|
1204
|
+
}
|
|
1205
|
+
};
|
|
1206
|
+
useEffect(() => {
|
|
1207
|
+
if (selectedWallet === "starkey" && walletCapabilities.eventListeners) {
|
|
1208
|
+
const handleExtensionEvents = (event) => {
|
|
1209
|
+
var _a, _b, _c;
|
|
1210
|
+
if ((_b = (_a = event == null ? void 0 : event.data) == null ? void 0 : _a.name) == null ? void 0 : _b.startsWith("starkey-")) {
|
|
1211
|
+
switch ((_c = event == null ? void 0 : event.data) == null ? void 0 : _c.name) {
|
|
1212
|
+
case "starkey-extension-installed": {
|
|
1213
|
+
checkIsExtensionInstalled();
|
|
1214
|
+
break;
|
|
1215
|
+
}
|
|
1216
|
+
case "starkey-wallet-updated": {
|
|
1217
|
+
(async () => {
|
|
1218
|
+
const responseAcc = await supraProvider.account();
|
|
1219
|
+
if (responseAcc.length) {
|
|
1220
|
+
setAccounts(responseAcc);
|
|
1221
|
+
window.dispatchEvent(
|
|
1222
|
+
new CustomEvent(WALLET_EVENTS.CONNECTED, {
|
|
1223
|
+
detail: {
|
|
1224
|
+
timestamp: Date.now(),
|
|
1225
|
+
account: responseAcc[0]
|
|
1226
|
+
}
|
|
1227
|
+
})
|
|
1228
|
+
);
|
|
1229
|
+
await updateBalance();
|
|
1230
|
+
await getNetworkData();
|
|
1231
|
+
} else {
|
|
1232
|
+
logger.debug("Starkey wallet updated: No accounts found - Resetting");
|
|
1233
|
+
resetWalletData();
|
|
1234
|
+
}
|
|
1235
|
+
setLoading(false);
|
|
1236
|
+
})();
|
|
1237
|
+
break;
|
|
1238
|
+
}
|
|
1239
|
+
case "starkey-wallet-disconnected": {
|
|
1240
|
+
resetWalletData();
|
|
1241
|
+
router.push("/");
|
|
1242
|
+
setLoading(false);
|
|
1243
|
+
break;
|
|
1244
|
+
}
|
|
1245
|
+
case "starkey-window-removed": {
|
|
1246
|
+
setLoading(false);
|
|
1247
|
+
break;
|
|
1248
|
+
}
|
|
1249
|
+
}
|
|
1250
|
+
}
|
|
1251
|
+
};
|
|
1252
|
+
checkIsExtensionInstalled();
|
|
1253
|
+
window.addEventListener("message", handleExtensionEvents);
|
|
1254
|
+
return () => window.removeEventListener("message", handleExtensionEvents);
|
|
1255
|
+
}
|
|
1256
|
+
}, [selectedWallet, walletCapabilities, supraProvider]);
|
|
1257
|
+
const getAvailableWallets = useCallback(() => {
|
|
1258
|
+
const availableWallets = [];
|
|
1259
|
+
Object.entries(WALLET_CONFIGS).forEach(([walletType, config2]) => {
|
|
1260
|
+
const provider = config2.provider();
|
|
1261
|
+
const isInstalled = !!provider;
|
|
1262
|
+
switch (walletType) {
|
|
1263
|
+
case "starkey": {
|
|
1264
|
+
availableWallets.push({
|
|
1265
|
+
type: "starkey",
|
|
1266
|
+
name: "Starkey Wallet",
|
|
1267
|
+
isInstalled,
|
|
1268
|
+
capabilities: config2.capabilities
|
|
1269
|
+
});
|
|
1270
|
+
break;
|
|
1271
|
+
}
|
|
1272
|
+
case "ribbit": {
|
|
1273
|
+
availableWallets.push({
|
|
1274
|
+
type: "ribbit",
|
|
1275
|
+
name: "Ribbit Wallet",
|
|
1276
|
+
isInstalled,
|
|
1277
|
+
capabilities: config2.capabilities
|
|
1278
|
+
});
|
|
1279
|
+
break;
|
|
1280
|
+
}
|
|
1281
|
+
}
|
|
1282
|
+
});
|
|
1283
|
+
return availableWallets;
|
|
1284
|
+
}, []);
|
|
1285
|
+
const updateSelectedWallet = (walletType) => {
|
|
1286
|
+
setSelectedWallet(walletType);
|
|
1287
|
+
setWalletCapabilities(WALLET_CONFIGS[walletType].capabilities);
|
|
1288
|
+
setStoredWalletType(walletType);
|
|
1289
|
+
};
|
|
1290
|
+
return {
|
|
1291
|
+
// New wallet selection functionality
|
|
1292
|
+
selectedWallet,
|
|
1293
|
+
walletCapabilities,
|
|
1294
|
+
getAvailableWallets,
|
|
1295
|
+
// Add this new function
|
|
1296
|
+
// Existing interface (unchanged)
|
|
1297
|
+
getCurrentProvider,
|
|
1298
|
+
isExtensionInstalled,
|
|
1299
|
+
accounts,
|
|
1300
|
+
networkData,
|
|
1301
|
+
balance,
|
|
1302
|
+
transactions,
|
|
1303
|
+
selectedChainId,
|
|
1304
|
+
connectWallet,
|
|
1305
|
+
// Now accepts optional walletType parameter
|
|
1306
|
+
disconnectWallet,
|
|
1307
|
+
sendRawTransaction,
|
|
1308
|
+
signMessage,
|
|
1309
|
+
setSelectedChainId,
|
|
1310
|
+
// switchToChain,
|
|
1311
|
+
loading
|
|
1312
|
+
// authFetch,
|
|
1313
|
+
// checkAndRevalidateToken,
|
|
1314
|
+
// signIn,
|
|
1315
|
+
};
|
|
1316
|
+
};
|
|
1317
|
+
var useSupraMultiWallet_default = useSupraMultiWallet;
|
|
1318
|
+
|
|
1319
|
+
// src/assets/walletIcons/Starkey.png
|
|
1320
|
+
var Starkey_default = "";
|
|
1321
|
+
|
|
1322
|
+
// src/assets/walletIcons/Ribbit.jpg
|
|
1323
|
+
var Ribbit_default = "";
|
|
1324
|
+
|
|
1325
|
+
// src/components/ConnectWalletHandler.tsx
|
|
1326
|
+
import { Fragment, jsx as jsx3, jsxs as jsxs2 } from "react/jsx-runtime";
|
|
1327
|
+
var PROFILE_CACHE_KEY = STORAGE_KEYS.USER_PROFILE_CACHE;
|
|
1328
|
+
var PROFILE_CACHE_TIMESTAMP_KEY = STORAGE_KEYS.USER_PROFILE_CACHE_TIMESTAMP;
|
|
1329
|
+
var WALLET_INFO = {
|
|
1330
|
+
starkey: {
|
|
1331
|
+
name: "Starkey Wallet",
|
|
1332
|
+
icon: /* @__PURE__ */ jsx3(
|
|
1333
|
+
"img",
|
|
1334
|
+
{
|
|
1335
|
+
src: Starkey_default,
|
|
1336
|
+
alt: "Starkey Wallet",
|
|
1337
|
+
className: "w-10 h-10 rounded-full"
|
|
1338
|
+
}
|
|
1339
|
+
),
|
|
1340
|
+
downloadUrl: WALLET_DOWNLOAD_URLS.starkey
|
|
1341
|
+
},
|
|
1342
|
+
ribbit: {
|
|
1343
|
+
name: "Ribbit Wallet",
|
|
1344
|
+
icon: /* @__PURE__ */ jsx3(
|
|
1345
|
+
"img",
|
|
1346
|
+
{
|
|
1347
|
+
src: Ribbit_default,
|
|
1348
|
+
alt: "Ribbit Wallet",
|
|
1349
|
+
className: "w-10 h-10 rounded-full"
|
|
1350
|
+
}
|
|
1351
|
+
),
|
|
1352
|
+
downloadUrl: WALLET_DOWNLOAD_URLS.ribbit
|
|
1353
|
+
}
|
|
1354
|
+
};
|
|
1355
|
+
var ConnectWalletHandler = ({
|
|
1356
|
+
onConnect,
|
|
1357
|
+
onDisconnect,
|
|
1358
|
+
children
|
|
1359
|
+
}) => {
|
|
1360
|
+
const starKeyWalletHook = useSupraMultiWallet_default();
|
|
1361
|
+
const getAvailableWallets = starKeyWalletHook.getAvailableWallets;
|
|
1362
|
+
const [loading, setLoading] = useState2(false);
|
|
1363
|
+
const [showWalletModal, setShowWalletModal] = useState2(false);
|
|
1364
|
+
const [userProfile, setUserProfile] = useState2(null);
|
|
1365
|
+
const [walletBalance, setWalletBalance] = useState2("0.00");
|
|
1366
|
+
const [availableWallets, setAvailableWallets] = useState2([]);
|
|
1367
|
+
const [recentWallet, setRecentWallet] = useState2(null);
|
|
1368
|
+
const [selectedWallet, setSelectedWallet] = useState2(null);
|
|
1369
|
+
const [hoveredWallet, setHoveredWallet] = useState2(null);
|
|
1370
|
+
const [connectionStage, setConnectionStage] = useState2("idle");
|
|
1371
|
+
const [connectionStageStartTime, setConnectionStageStartTime] = useState2(null);
|
|
1372
|
+
const [canClickOutside, setCanClickOutside] = useState2(false);
|
|
1373
|
+
const getProfileFromCache = useCallback2(() => {
|
|
1374
|
+
try {
|
|
1375
|
+
if (typeof window === "undefined") return null;
|
|
1376
|
+
const cachedData = getStorageItem(PROFILE_CACHE_KEY);
|
|
1377
|
+
const timestamp = getStorageItem(PROFILE_CACHE_TIMESTAMP_KEY);
|
|
1378
|
+
if (!cachedData || !timestamp) return null;
|
|
1379
|
+
const now = Date.now();
|
|
1380
|
+
const cacheTime = parseInt(timestamp);
|
|
1381
|
+
if (now - cacheTime > CACHE_TTL.PROFILE) {
|
|
1382
|
+
return null;
|
|
1383
|
+
}
|
|
1384
|
+
return JSON.parse(cachedData);
|
|
1385
|
+
} catch (error) {
|
|
1386
|
+
logger.error("Failed to read profile from cache", error);
|
|
1387
|
+
return null;
|
|
1388
|
+
}
|
|
1389
|
+
}, []);
|
|
1390
|
+
const updateWalletBalance = useCallback2(async () => {
|
|
1391
|
+
const provider = starKeyWalletHook.getCurrentProvider();
|
|
1392
|
+
if (!provider) return;
|
|
1393
|
+
try {
|
|
1394
|
+
await new Promise((resolve) => setTimeout(resolve, 1));
|
|
1395
|
+
switch (starKeyWalletHook.selectedWallet) {
|
|
1396
|
+
case "starkey": {
|
|
1397
|
+
const balance = await provider.balance();
|
|
1398
|
+
if (balance && balance.formattedBalance) {
|
|
1399
|
+
setWalletBalance(balance.formattedBalance);
|
|
1400
|
+
}
|
|
1401
|
+
break;
|
|
1402
|
+
}
|
|
1403
|
+
case "ribbit": {
|
|
1404
|
+
const walletBalanceRequest = {
|
|
1405
|
+
chainId: parseInt(process.env.NEXT_PUBLIC_SUPRA_CHAIN_ID || "6"),
|
|
1406
|
+
resourceType: "0x1::supra_coin::SupraCoin",
|
|
1407
|
+
decimals: 8
|
|
1408
|
+
};
|
|
1409
|
+
try {
|
|
1410
|
+
const balance = await provider.getWalletBalance(walletBalanceRequest);
|
|
1411
|
+
const balanceStr = balance.balance;
|
|
1412
|
+
logger.debug("Ribbit balance fetched", { balance: balanceStr });
|
|
1413
|
+
if (balanceStr) {
|
|
1414
|
+
setWalletBalance(balanceStr);
|
|
1415
|
+
}
|
|
1416
|
+
} catch (error) {
|
|
1417
|
+
logger.error("Failed to fetch Ribbit balance", error);
|
|
1418
|
+
}
|
|
1419
|
+
break;
|
|
1420
|
+
}
|
|
1421
|
+
}
|
|
1422
|
+
} catch (error) {
|
|
1423
|
+
logger.error("Failed to update wallet balance", error);
|
|
1424
|
+
}
|
|
1425
|
+
}, [starKeyWalletHook.selectedWallet, starKeyWalletHook.getCurrentProvider]);
|
|
1426
|
+
const connectWallet = useCallback2(async (walletType) => {
|
|
1427
|
+
if (walletType) {
|
|
1428
|
+
setLoading(true);
|
|
1429
|
+
setSelectedWallet(walletType);
|
|
1430
|
+
setConnectionStage("connecting");
|
|
1431
|
+
try {
|
|
1432
|
+
const success = await starKeyWalletHook.connectWallet(walletType);
|
|
1433
|
+
if (success) {
|
|
1434
|
+
setStorageItem(STORAGE_KEYS.RECENT_WALLET, walletType);
|
|
1435
|
+
setRecentWallet(walletType);
|
|
1436
|
+
if (starKeyWalletHook.accounts.length > 0) {
|
|
1437
|
+
onConnect == null ? void 0 : onConnect(starKeyWalletHook.accounts[0]);
|
|
1438
|
+
}
|
|
1439
|
+
}
|
|
1440
|
+
} catch (error) {
|
|
1441
|
+
logger.error("Failed to connect wallet", error, { walletType });
|
|
1442
|
+
toast2.error(`Connection failed: ${error.message || "Unknown error"}`);
|
|
1443
|
+
setConnectionStage("idle");
|
|
1444
|
+
} finally {
|
|
1445
|
+
setLoading(false);
|
|
1446
|
+
}
|
|
1447
|
+
} else {
|
|
1448
|
+
handleConnectClick();
|
|
1449
|
+
}
|
|
1450
|
+
}, [starKeyWalletHook, onConnect]);
|
|
1451
|
+
const handleConnectClick = () => {
|
|
1452
|
+
const installedWallets = availableWallets.filter((w) => w.isInstalled);
|
|
1453
|
+
if (installedWallets.length === 0) {
|
|
1454
|
+
setShowWalletModal(true);
|
|
1455
|
+
} else if (installedWallets.length === 1) {
|
|
1456
|
+
setShowWalletModal(true);
|
|
1457
|
+
} else {
|
|
1458
|
+
setShowWalletModal(true);
|
|
1459
|
+
}
|
|
1460
|
+
};
|
|
1461
|
+
const handleDisconnectWallet = useCallback2(async () => {
|
|
1462
|
+
setLoading(true);
|
|
1463
|
+
try {
|
|
1464
|
+
await starKeyWalletHook.disconnectWallet();
|
|
1465
|
+
setUserProfile(null);
|
|
1466
|
+
setWalletBalance("0.00");
|
|
1467
|
+
onDisconnect == null ? void 0 : onDisconnect();
|
|
1468
|
+
} catch (error) {
|
|
1469
|
+
logger.error("Failed to disconnect wallet", error);
|
|
1470
|
+
} finally {
|
|
1471
|
+
setLoading(false);
|
|
1472
|
+
}
|
|
1473
|
+
}, [starKeyWalletHook, onDisconnect]);
|
|
1474
|
+
const getConnectionStageInfo = useCallback2(() => {
|
|
1475
|
+
const wallet = selectedWallet ? WALLET_INFO[selectedWallet] : null;
|
|
1476
|
+
switch (connectionStage) {
|
|
1477
|
+
case "connecting":
|
|
1478
|
+
return {
|
|
1479
|
+
title: `Waiting for ${(wallet == null ? void 0 : wallet.name) || "Wallet"}`,
|
|
1480
|
+
subtitle: CONNECTION_MESSAGES.CONNECTING,
|
|
1481
|
+
buttonText: "Connecting"
|
|
1482
|
+
};
|
|
1483
|
+
case "signing":
|
|
1484
|
+
return {
|
|
1485
|
+
title: "Sign to verify",
|
|
1486
|
+
subtitle: CONNECTION_MESSAGES.SIGNING,
|
|
1487
|
+
buttonText: "Signing"
|
|
1488
|
+
};
|
|
1489
|
+
case "success":
|
|
1490
|
+
return {
|
|
1491
|
+
title: `Connected to ${(wallet == null ? void 0 : wallet.name.replace(" Wallet", "")) || "Wallet"}`,
|
|
1492
|
+
subtitle: CONNECTION_MESSAGES.SUCCESS,
|
|
1493
|
+
buttonText: "Connected"
|
|
1494
|
+
};
|
|
1495
|
+
case "error":
|
|
1496
|
+
return {
|
|
1497
|
+
title: "Error",
|
|
1498
|
+
subtitle: CONNECTION_MESSAGES.ERROR,
|
|
1499
|
+
buttonText: "Close"
|
|
1500
|
+
};
|
|
1501
|
+
case "connected-not-signed":
|
|
1502
|
+
return {
|
|
1503
|
+
title: `Connected to ${(wallet == null ? void 0 : wallet.name.replace(" Wallet", "")) || "Wallet"}`,
|
|
1504
|
+
subtitle: CONNECTION_MESSAGES.CONNECTED_NOT_SIGNED,
|
|
1505
|
+
buttonText: "Connected"
|
|
1506
|
+
};
|
|
1507
|
+
default:
|
|
1508
|
+
return null;
|
|
1509
|
+
}
|
|
1510
|
+
}, [connectionStage, selectedWallet]);
|
|
1511
|
+
useEffect2(() => {
|
|
1512
|
+
const recent = getStorageItem(STORAGE_KEYS.RECENT_WALLET);
|
|
1513
|
+
if (recent && ["starkey", "ribbit"].includes(recent)) {
|
|
1514
|
+
setRecentWallet(recent);
|
|
1515
|
+
}
|
|
1516
|
+
}, []);
|
|
1517
|
+
useEffect2(() => {
|
|
1518
|
+
const checkWallets = () => {
|
|
1519
|
+
const wallets = getAvailableWallets();
|
|
1520
|
+
setAvailableWallets(wallets);
|
|
1521
|
+
};
|
|
1522
|
+
checkWallets();
|
|
1523
|
+
const interval = setInterval(checkWallets, TIMEOUTS.WALLET_CHECK_INTERVAL);
|
|
1524
|
+
const timeout = setTimeout(() => clearInterval(interval), TIMEOUTS.WALLET_CHECK_MAX_DURATION);
|
|
1525
|
+
return () => {
|
|
1526
|
+
clearInterval(interval);
|
|
1527
|
+
clearTimeout(timeout);
|
|
1528
|
+
};
|
|
1529
|
+
}, [getAvailableWallets]);
|
|
1530
|
+
useEffect2(() => {
|
|
1531
|
+
if (starKeyWalletHook.accounts.length > 0) {
|
|
1532
|
+
updateWalletBalance();
|
|
1533
|
+
}
|
|
1534
|
+
}, [starKeyWalletHook.accounts, starKeyWalletHook.selectedWallet]);
|
|
1535
|
+
useEffect2(() => {
|
|
1536
|
+
const handlePresignedState = () => {
|
|
1537
|
+
setConnectionStage(() => "signing");
|
|
1538
|
+
};
|
|
1539
|
+
const handlePostsignedState = () => {
|
|
1540
|
+
setConnectionStage("success");
|
|
1541
|
+
setTimeout(() => {
|
|
1542
|
+
setConnectionStage("idle");
|
|
1543
|
+
setShowWalletModal(false);
|
|
1544
|
+
updateWalletBalance();
|
|
1545
|
+
}, TIMEOUTS.MODAL_CLOSE_DELAY);
|
|
1546
|
+
};
|
|
1547
|
+
const handleWalletError = () => {
|
|
1548
|
+
if (connectionStage == "signing") {
|
|
1549
|
+
setConnectionStage(() => "connected-not-signed");
|
|
1550
|
+
} else {
|
|
1551
|
+
setConnectionStage(() => "error");
|
|
1552
|
+
}
|
|
1553
|
+
setTimeout(() => {
|
|
1554
|
+
setShowWalletModal(false);
|
|
1555
|
+
setSelectedWallet(null);
|
|
1556
|
+
setConnectionStage("idle");
|
|
1557
|
+
setLoading(false);
|
|
1558
|
+
}, TIMEOUTS.MODAL_CLOSE_DELAY);
|
|
1559
|
+
};
|
|
1560
|
+
const handleStarkeyEvents = (event) => {
|
|
1561
|
+
var _a, _b, _c;
|
|
1562
|
+
if ((_b = (_a = event == null ? void 0 : event.data) == null ? void 0 : _a.name) == null ? void 0 : _b.startsWith("starkey-")) {
|
|
1563
|
+
switch ((_c = event == null ? void 0 : event.data) == null ? void 0 : _c.name) {
|
|
1564
|
+
case "starkey-wallet-updated":
|
|
1565
|
+
case "starkey-wallet-connected":
|
|
1566
|
+
setTimeout(updateWalletBalance, 1);
|
|
1567
|
+
break;
|
|
1568
|
+
case "starkey-wallet-disconnected":
|
|
1569
|
+
setWalletBalance("0.00");
|
|
1570
|
+
break;
|
|
1571
|
+
}
|
|
1572
|
+
}
|
|
1573
|
+
};
|
|
1574
|
+
window.addEventListener("presigned-state", handlePresignedState);
|
|
1575
|
+
window.addEventListener("postsigned-state", handlePostsignedState);
|
|
1576
|
+
window.addEventListener("wallet-error", handleWalletError);
|
|
1577
|
+
window.addEventListener("message", handleStarkeyEvents);
|
|
1578
|
+
return () => {
|
|
1579
|
+
window.removeEventListener("presigned-state", handlePresignedState);
|
|
1580
|
+
window.removeEventListener("postsigned-state", handlePostsignedState);
|
|
1581
|
+
window.removeEventListener("wallet-error", handleWalletError);
|
|
1582
|
+
window.removeEventListener("message", handleStarkeyEvents);
|
|
1583
|
+
};
|
|
1584
|
+
}, [connectionStage]);
|
|
1585
|
+
useEffect2(() => {
|
|
1586
|
+
const handleProfileUpdated = (event) => {
|
|
1587
|
+
const { address, username, profileImage } = event.detail;
|
|
1588
|
+
if (starKeyWalletHook.accounts[0] === address) {
|
|
1589
|
+
setUserProfile({ address, username, profileImage });
|
|
1590
|
+
}
|
|
1591
|
+
};
|
|
1592
|
+
window.addEventListener("profile-updated", handleProfileUpdated);
|
|
1593
|
+
return () => {
|
|
1594
|
+
window.removeEventListener("profile-updated", handleProfileUpdated);
|
|
1595
|
+
};
|
|
1596
|
+
}, [starKeyWalletHook.accounts]);
|
|
1597
|
+
useEffect2(() => {
|
|
1598
|
+
if (starKeyWalletHook.accounts.length > 0) {
|
|
1599
|
+
const handleVisibilityChange = () => {
|
|
1600
|
+
if (!document.hidden) {
|
|
1601
|
+
updateWalletBalance();
|
|
1602
|
+
}
|
|
1603
|
+
};
|
|
1604
|
+
document.addEventListener("visibilitychange", handleVisibilityChange);
|
|
1605
|
+
const balanceInterval = setInterval(() => {
|
|
1606
|
+
if (!document.hidden) {
|
|
1607
|
+
updateWalletBalance();
|
|
1608
|
+
}
|
|
1609
|
+
}, TIMEOUTS.BALANCE_POLL_INTERVAL);
|
|
1610
|
+
return () => {
|
|
1611
|
+
clearInterval(balanceInterval);
|
|
1612
|
+
document.removeEventListener("visibilitychange", handleVisibilityChange);
|
|
1613
|
+
};
|
|
1614
|
+
}
|
|
1615
|
+
}, [starKeyWalletHook.accounts, updateWalletBalance]);
|
|
1616
|
+
useEffect2(() => {
|
|
1617
|
+
if (connectionStage === "connecting" || connectionStage === "signing") {
|
|
1618
|
+
setConnectionStageStartTime(Date.now());
|
|
1619
|
+
setCanClickOutside(false);
|
|
1620
|
+
const timer = setTimeout(() => {
|
|
1621
|
+
setCanClickOutside(true);
|
|
1622
|
+
}, TIMEOUTS.CLICK_OUTSIDE_DELAY);
|
|
1623
|
+
return () => clearTimeout(timer);
|
|
1624
|
+
} else {
|
|
1625
|
+
setConnectionStageStartTime(null);
|
|
1626
|
+
setCanClickOutside(false);
|
|
1627
|
+
}
|
|
1628
|
+
}, [connectionStage]);
|
|
1629
|
+
const handleModalClose = useCallback2((open) => {
|
|
1630
|
+
if (!open) {
|
|
1631
|
+
if (connectionStage === "idle" || connectionStage === "success" || connectionStage === "connected-not-signed" || connectionStage === "error" || canClickOutside && (connectionStage === "connecting" || connectionStage === "signing")) {
|
|
1632
|
+
setShowWalletModal(false);
|
|
1633
|
+
setSelectedWallet(null);
|
|
1634
|
+
setLoading(false);
|
|
1635
|
+
setCanClickOutside(false);
|
|
1636
|
+
setConnectionStageStartTime(null);
|
|
1637
|
+
setTimeout(() => {
|
|
1638
|
+
setConnectionStage("idle");
|
|
1639
|
+
}, 100);
|
|
1640
|
+
}
|
|
1641
|
+
}
|
|
1642
|
+
}, [connectionStage, canClickOutside]);
|
|
1643
|
+
return /* @__PURE__ */ jsxs2(Fragment, { children: [
|
|
1644
|
+
children({
|
|
1645
|
+
isConnected: starKeyWalletHook.accounts.length > 0,
|
|
1646
|
+
accounts: starKeyWalletHook.accounts,
|
|
1647
|
+
loading: loading || starKeyWalletHook.loading,
|
|
1648
|
+
balance: walletBalance,
|
|
1649
|
+
userProfile,
|
|
1650
|
+
handleConnect: handleConnectClick,
|
|
1651
|
+
handleDisconnect: handleDisconnectWallet
|
|
1652
|
+
}),
|
|
1653
|
+
/* @__PURE__ */ jsxs2(
|
|
1654
|
+
Dialog,
|
|
1655
|
+
{
|
|
1656
|
+
open: showWalletModal,
|
|
1657
|
+
onOpenChange: handleModalClose,
|
|
1658
|
+
children: [
|
|
1659
|
+
/* @__PURE__ */ jsx3(VisuallyHidden, { asChild: true, children: /* @__PURE__ */ jsx3(DialogTitle, { children: "Select a wallet" }) }),
|
|
1660
|
+
/* @__PURE__ */ jsx3(VisuallyHidden, { asChild: true, children: /* @__PURE__ */ jsx3(DialogDescription, { children: "Select a wallet to connect to MyDApp" }) }),
|
|
1661
|
+
/* @__PURE__ */ jsxs2(
|
|
1662
|
+
DialogContent,
|
|
1663
|
+
{
|
|
1664
|
+
className: "z-[100] px-4 py-6 w-[90%] mx-auto max-w-sm bg-gradient-to-br from-brand-dark via-gray-900 to-brand-dark border border-brand-dark sm:rounded-3xl rounded-3xl",
|
|
1665
|
+
children: [
|
|
1666
|
+
canClickOutside && (connectionStage === "connecting" || connectionStage === "signing") && /* @__PURE__ */ jsx3("div", { className: "absolute top-4 right-4", children: /* @__PURE__ */ jsx3(X2, { className: "h-5 w-5" }) }),
|
|
1667
|
+
/* @__PURE__ */ jsx3("div", { className: "", children: connectionStage === "idle" ? /* @__PURE__ */ jsxs2(Fragment, { children: [
|
|
1668
|
+
/* @__PURE__ */ jsx3("p", { className: "text-center text-sm text-brand-light/75 mb-6 text-white", children: "Select Wallet" }),
|
|
1669
|
+
/* @__PURE__ */ jsx3("div", { className: "space-y-3", children: availableWallets.map((wallet) => {
|
|
1670
|
+
return /* @__PURE__ */ jsx3("div", { className: "w-full relative", children: /* @__PURE__ */ jsxs2(
|
|
1671
|
+
motion.button,
|
|
1672
|
+
{
|
|
1673
|
+
initial: { opacity: 0, y: 20 },
|
|
1674
|
+
animate: { opacity: 1, y: 0 },
|
|
1675
|
+
transition: { duration: 0.05, ease: "easeInOut" },
|
|
1676
|
+
onClick: () => {
|
|
1677
|
+
wallet.isInstalled ? connectWallet(wallet.type) : window.open(WALLET_INFO[wallet.type].downloadUrl, "_blank");
|
|
1678
|
+
},
|
|
1679
|
+
type: "button",
|
|
1680
|
+
disabled: loading && wallet.isInstalled,
|
|
1681
|
+
onMouseEnter: () => setHoveredWallet(wallet.type),
|
|
1682
|
+
onMouseLeave: () => setHoveredWallet(null),
|
|
1683
|
+
className: "w-full px-4 py-2 rounded-2xl border border-gray-950/60 hover:border-gray-800/80 bg-gray-950/20 hover:bg-gray-900/40 transition-all duration-300 flex items-center gap-4 group",
|
|
1684
|
+
children: [
|
|
1685
|
+
/* @__PURE__ */ jsx3("div", { className: "flex-shrink-0", children: WALLET_INFO[wallet.type].icon }),
|
|
1686
|
+
/* @__PURE__ */ jsxs2("div", { className: "flex-1 text-left", children: [
|
|
1687
|
+
/* @__PURE__ */ jsx3("h3", { className: "font-medium text-white", children: WALLET_INFO[wallet.type].name }),
|
|
1688
|
+
!wallet.isInstalled && /* @__PURE__ */ jsx3("p", { className: "text-sm text-gray-400", children: "Not installed - Click to download" })
|
|
1689
|
+
] }),
|
|
1690
|
+
/* @__PURE__ */ jsx3("div", { className: "flex-shrink-0", children: recentWallet === wallet.type && wallet.isInstalled ? /* @__PURE__ */ jsx3("span", { className: "bg-gray-800/60 text-gray-300 text-xs px-3 py-1 rounded-full border border-gray-700/40", children: "Recent" }) : wallet.isInstalled ? loading && wallet.type === selectedWallet ? /* @__PURE__ */ jsx3(Loader2, { className: "h-4 w-4 animate-spin text-gray-400" }) : hoveredWallet === wallet.type && /* @__PURE__ */ jsx3("span", { className: "text-gray-400 text-sm font-medium", children: "Connect" }) : /* @__PURE__ */ jsx3(ExternalLink, { className: "h-4 w-4 text-gray-500" }) })
|
|
1691
|
+
]
|
|
1692
|
+
}
|
|
1693
|
+
) }, wallet.type);
|
|
1694
|
+
}) })
|
|
1695
|
+
] }) : /* @__PURE__ */ jsx3("div", { className: "w-full text-center py-8", children: (() => {
|
|
1696
|
+
const stageInfo = getConnectionStageInfo();
|
|
1697
|
+
const wallet = selectedWallet ? WALLET_INFO[selectedWallet] : null;
|
|
1698
|
+
return /* @__PURE__ */ jsxs2(Fragment, { children: [
|
|
1699
|
+
/* @__PURE__ */ jsx3("div", { className: "relative flex justify-center mb-6", children: connectionStage === "success" || connectionStage === "connected-not-signed" ? /* @__PURE__ */ jsx3("div", { className: "w-20 h-20 rounded-full border-4 border-green-500 flex items-center justify-center", children: wallet && /* @__PURE__ */ jsx3("div", { className: "w-12 h-12 flex items-center justify-center", children: wallet.icon }) }) : connectionStage === "error" ? /* @__PURE__ */ jsx3("div", { className: "w-20 h-20 rounded-full border-4 border-red-500 flex items-center justify-center", children: wallet && /* @__PURE__ */ jsx3("div", { className: "w-12 h-12 flex items-center justify-center", children: wallet.icon }) }) : /* @__PURE__ */ jsxs2("div", { className: "relative", children: [
|
|
1700
|
+
/* @__PURE__ */ jsx3("div", { className: "w-20 h-20 rounded-full border-4 border-gray-600 border-t-gray-400 animate-spin" }),
|
|
1701
|
+
/* @__PURE__ */ jsx3("div", { className: "absolute inset-0 flex items-center justify-center", children: wallet && /* @__PURE__ */ jsx3("div", { className: "w-12 h-12 flex items-center justify-center", children: wallet.icon }) })
|
|
1702
|
+
] }) }),
|
|
1703
|
+
/* @__PURE__ */ jsx3("h3", { className: "text-lg font-semibold text-white mb-2", children: stageInfo == null ? void 0 : stageInfo.title }),
|
|
1704
|
+
/* @__PURE__ */ jsx3("p", { className: "text-sm text-gray-400 mb-6", children: stageInfo == null ? void 0 : stageInfo.subtitle }),
|
|
1705
|
+
/* @__PURE__ */ jsx3(
|
|
1706
|
+
Button,
|
|
1707
|
+
{
|
|
1708
|
+
disabled: true,
|
|
1709
|
+
className: "w-full bg-gray-700 text-gray-400 cursor-not-allowed rounded-2xl py-3",
|
|
1710
|
+
children: stageInfo == null ? void 0 : stageInfo.buttonText
|
|
1711
|
+
}
|
|
1712
|
+
)
|
|
1713
|
+
] });
|
|
1714
|
+
})() }) })
|
|
1715
|
+
]
|
|
1716
|
+
}
|
|
1717
|
+
)
|
|
1718
|
+
]
|
|
1719
|
+
}
|
|
1720
|
+
)
|
|
1721
|
+
] });
|
|
1722
|
+
};
|
|
1723
|
+
|
|
1724
|
+
// src/hooks/useSupraMultiWallet.ts
|
|
1725
|
+
import { useState as useState3, useEffect as useEffect3, useCallback as useCallback3 } from "react";
|
|
1726
|
+
import nacl2 from "tweetnacl";
|
|
1727
|
+
import { useRouter as useRouter2 } from "next/navigation";
|
|
1728
|
+
import { toast as toast3 } from "sonner";
|
|
1729
|
+
import {
|
|
1730
|
+
SupraChainId as SupraChainId2,
|
|
1731
|
+
initSdk as initSdk2
|
|
1732
|
+
} from "ribbit-wallet-connect";
|
|
1733
|
+
var WALLET_CONFIGS2 = {
|
|
1734
|
+
starkey: {
|
|
1735
|
+
capabilities: {
|
|
1736
|
+
signMessage: true,
|
|
1737
|
+
accountSwitching: true,
|
|
1738
|
+
networkSwitching: true,
|
|
1739
|
+
rawTransactions: true,
|
|
1740
|
+
eventListeners: true,
|
|
1741
|
+
tokenRevalidation: true
|
|
1742
|
+
},
|
|
1743
|
+
provider: () => {
|
|
1744
|
+
var _a;
|
|
1745
|
+
return typeof window !== "undefined" && ((_a = window == null ? void 0 : window.starkey) == null ? void 0 : _a.supra);
|
|
1746
|
+
}
|
|
1747
|
+
},
|
|
1748
|
+
ribbit: {
|
|
1749
|
+
capabilities: {
|
|
1750
|
+
signMessage: true,
|
|
1751
|
+
accountSwitching: false,
|
|
1752
|
+
// Ribbit doesn't support account switching
|
|
1753
|
+
networkSwitching: false,
|
|
1754
|
+
// Ribbit network switching happens in-app
|
|
1755
|
+
rawTransactions: true,
|
|
1756
|
+
eventListeners: false,
|
|
1757
|
+
tokenRevalidation: false
|
|
1758
|
+
// Ribbit doesn't support token revalidation
|
|
1759
|
+
},
|
|
1760
|
+
provider: () => initSdk2()
|
|
1761
|
+
}
|
|
1762
|
+
};
|
|
1763
|
+
var WALLET_EVENTS2 = {
|
|
1764
|
+
CONNECTED: "wallet-connected",
|
|
1765
|
+
PRESIGNED_STATE: "presigned-state",
|
|
1766
|
+
POSTSIGNED_STATE: "postsigned-state",
|
|
1767
|
+
ERROR: "wallet-error"
|
|
1768
|
+
};
|
|
1769
|
+
var setStoredWalletType2 = (walletType) => {
|
|
1770
|
+
setStorageItem(STORAGE_KEYS.SELECTED_WALLET, walletType);
|
|
1771
|
+
};
|
|
1772
|
+
var getStoredWalletType2 = () => {
|
|
1773
|
+
const stored = getStorageItem(STORAGE_KEYS.SELECTED_WALLET);
|
|
1774
|
+
if (stored && ["starkey", "ribbit"].includes(stored)) {
|
|
1775
|
+
return stored;
|
|
1776
|
+
}
|
|
1777
|
+
return "starkey";
|
|
1778
|
+
};
|
|
1779
|
+
var clearStoredWalletType2 = () => {
|
|
1780
|
+
removeStorageItem(STORAGE_KEYS.SELECTED_WALLET);
|
|
1781
|
+
};
|
|
1782
|
+
var useSupraMultiWallet2 = () => {
|
|
1783
|
+
const router = useRouter2();
|
|
1784
|
+
const [selectedWallet, setSelectedWallet] = useState3("starkey");
|
|
1785
|
+
const [walletCapabilities, setWalletCapabilities] = useState3(
|
|
1786
|
+
WALLET_CONFIGS2["starkey"].capabilities
|
|
1787
|
+
);
|
|
1788
|
+
useEffect3(() => {
|
|
1789
|
+
const stored = getStoredWalletType2();
|
|
1790
|
+
if (stored !== "starkey") {
|
|
1791
|
+
setSelectedWallet(stored);
|
|
1792
|
+
setWalletCapabilities(WALLET_CONFIGS2[stored].capabilities);
|
|
1793
|
+
}
|
|
1794
|
+
}, []);
|
|
1795
|
+
const [supraProvider, setSupraProvider] = useState3(
|
|
1796
|
+
WALLET_CONFIGS2.starkey.provider()
|
|
1797
|
+
);
|
|
1798
|
+
const [ribbitProvider, setRibbitProvider] = useState3(
|
|
1799
|
+
WALLET_CONFIGS2.ribbit.provider()
|
|
1800
|
+
);
|
|
1801
|
+
const [isExtensionInstalled, setIsExtensionInstalled] = useState3(false);
|
|
1802
|
+
const [accounts, setAccounts] = useState3([]);
|
|
1803
|
+
const [networkData, setNetworkData] = useState3();
|
|
1804
|
+
const [balance, setBalance] = useState3("");
|
|
1805
|
+
const [loading, setLoading] = useState3(false);
|
|
1806
|
+
const [justRequestedRelative, setJustRequestedRelative] = useState3(false);
|
|
1807
|
+
const [transactions, setTransactions] = useState3([]);
|
|
1808
|
+
const [selectedChainId, setSelectedChainId] = useState3("");
|
|
1809
|
+
const addTransactions = (hash) => {
|
|
1810
|
+
setTransactions((prev) => [{ hash }, ...prev]);
|
|
1811
|
+
};
|
|
1812
|
+
const getCurrentProvider = () => {
|
|
1813
|
+
switch (selectedWallet) {
|
|
1814
|
+
case "starkey": {
|
|
1815
|
+
return supraProvider;
|
|
1816
|
+
}
|
|
1817
|
+
case "ribbit": {
|
|
1818
|
+
return ribbitProvider;
|
|
1819
|
+
}
|
|
1820
|
+
default: {
|
|
1821
|
+
return supraProvider;
|
|
1822
|
+
}
|
|
1823
|
+
}
|
|
1824
|
+
};
|
|
1825
|
+
const checkExtensionInstalled = useCallback3(async () => {
|
|
1826
|
+
switch (selectedWallet) {
|
|
1827
|
+
case "starkey": {
|
|
1828
|
+
const provider = WALLET_CONFIGS2.starkey.provider();
|
|
1829
|
+
setSupraProvider(provider);
|
|
1830
|
+
setIsExtensionInstalled(!!provider);
|
|
1831
|
+
return !!provider;
|
|
1832
|
+
}
|
|
1833
|
+
case "ribbit": {
|
|
1834
|
+
const provider = WALLET_CONFIGS2.ribbit.provider();
|
|
1835
|
+
if (!provider) {
|
|
1836
|
+
setRibbitProvider(null);
|
|
1837
|
+
setIsExtensionInstalled(false);
|
|
1838
|
+
return false;
|
|
1839
|
+
}
|
|
1840
|
+
try {
|
|
1841
|
+
setRibbitProvider(provider);
|
|
1842
|
+
setIsExtensionInstalled(true);
|
|
1843
|
+
return true;
|
|
1844
|
+
} catch (error) {
|
|
1845
|
+
logger.error("Error checking Ribbit wallet readiness", error);
|
|
1846
|
+
setRibbitProvider(null);
|
|
1847
|
+
setIsExtensionInstalled(false);
|
|
1848
|
+
return false;
|
|
1849
|
+
}
|
|
1850
|
+
}
|
|
1851
|
+
default: {
|
|
1852
|
+
return false;
|
|
1853
|
+
}
|
|
1854
|
+
}
|
|
1855
|
+
}, [selectedWallet]);
|
|
1856
|
+
const updateAccounts = useCallback3(async () => {
|
|
1857
|
+
const provider = getCurrentProvider();
|
|
1858
|
+
if (!provider) return;
|
|
1859
|
+
try {
|
|
1860
|
+
switch (selectedWallet) {
|
|
1861
|
+
case "starkey": {
|
|
1862
|
+
const responseAcc = await provider.account();
|
|
1863
|
+
const newAccounts = responseAcc.length > 0 ? responseAcc : [];
|
|
1864
|
+
setAccounts(newAccounts);
|
|
1865
|
+
if (responseAcc.length > 0) {
|
|
1866
|
+
setStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS, responseAcc[0]);
|
|
1867
|
+
}
|
|
1868
|
+
if (newAccounts.length > 0) {
|
|
1869
|
+
const balance2 = await provider.balance();
|
|
1870
|
+
if (balance2) {
|
|
1871
|
+
setBalance(`${balance2.formattedBalance} ${balance2.displayUnit}`);
|
|
1872
|
+
}
|
|
1873
|
+
const networkData2 = await provider.getChainId();
|
|
1874
|
+
setNetworkData(networkData2 || {});
|
|
1875
|
+
}
|
|
1876
|
+
break;
|
|
1877
|
+
}
|
|
1878
|
+
case "ribbit": {
|
|
1879
|
+
const wallet = provider.getWalletInfo();
|
|
1880
|
+
if (wallet == null ? void 0 : wallet.connected) {
|
|
1881
|
+
setAccounts([wallet.walletAddress]);
|
|
1882
|
+
const walletBalanceRequest = {
|
|
1883
|
+
chainId: parseInt(getChainId()),
|
|
1884
|
+
resourceType: "0x1::supra_coin::SupraCoin",
|
|
1885
|
+
decimals: 8
|
|
1886
|
+
};
|
|
1887
|
+
const balanceStr = await provider.getWalletBalance(walletBalanceRequest);
|
|
1888
|
+
logger.debug("Ribbit balance response", { balance: balanceStr });
|
|
1889
|
+
setBalance(`${(balanceStr == null ? void 0 : balanceStr.balance) || 0} SUPRA`);
|
|
1890
|
+
} else {
|
|
1891
|
+
setAccounts([]);
|
|
1892
|
+
}
|
|
1893
|
+
break;
|
|
1894
|
+
}
|
|
1895
|
+
default: {
|
|
1896
|
+
setAccounts([]);
|
|
1897
|
+
break;
|
|
1898
|
+
}
|
|
1899
|
+
}
|
|
1900
|
+
} catch (error) {
|
|
1901
|
+
logger.error("Failed to update accounts", error, { walletType: selectedWallet });
|
|
1902
|
+
setAccounts([]);
|
|
1903
|
+
switch (selectedWallet) {
|
|
1904
|
+
case "starkey": {
|
|
1905
|
+
removeStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS);
|
|
1906
|
+
break;
|
|
1907
|
+
}
|
|
1908
|
+
case "ribbit": {
|
|
1909
|
+
break;
|
|
1910
|
+
}
|
|
1911
|
+
}
|
|
1912
|
+
}
|
|
1913
|
+
}, [selectedWallet]);
|
|
1914
|
+
useEffect3(() => {
|
|
1915
|
+
let mounted = true;
|
|
1916
|
+
const initProvider = async () => {
|
|
1917
|
+
const isInstalled = await checkExtensionInstalled();
|
|
1918
|
+
if (mounted && isInstalled) {
|
|
1919
|
+
updateAccounts();
|
|
1920
|
+
}
|
|
1921
|
+
};
|
|
1922
|
+
initProvider();
|
|
1923
|
+
return () => {
|
|
1924
|
+
mounted = false;
|
|
1925
|
+
};
|
|
1926
|
+
}, [selectedWallet, checkExtensionInstalled, updateAccounts]);
|
|
1927
|
+
useEffect3(() => {
|
|
1928
|
+
const checkExtension = async () => {
|
|
1929
|
+
return await checkExtensionInstalled();
|
|
1930
|
+
};
|
|
1931
|
+
checkExtension().then((isInstalled) => {
|
|
1932
|
+
if (isInstalled && selectedWallet === "ribbit") {
|
|
1933
|
+
updateAccounts();
|
|
1934
|
+
}
|
|
1935
|
+
});
|
|
1936
|
+
const intv = setInterval(async () => {
|
|
1937
|
+
const isInstalled = await checkExtension();
|
|
1938
|
+
if (isInstalled) {
|
|
1939
|
+
clearInterval(intv);
|
|
1940
|
+
if (selectedWallet === "ribbit") {
|
|
1941
|
+
updateAccounts();
|
|
1942
|
+
}
|
|
1943
|
+
}
|
|
1944
|
+
}, 1e3);
|
|
1945
|
+
setTimeout(() => {
|
|
1946
|
+
clearInterval(intv);
|
|
1947
|
+
}, TIMEOUTS.WALLET_DETECTION_POLL);
|
|
1948
|
+
return () => clearInterval(intv);
|
|
1949
|
+
}, [selectedWallet, updateAccounts, checkExtensionInstalled]);
|
|
1950
|
+
const checkIsExtensionInstalled = useCallback3(() => {
|
|
1951
|
+
const intervalId = setInterval(async () => {
|
|
1952
|
+
const isInstalled = await checkExtensionInstalled();
|
|
1953
|
+
if (isInstalled) {
|
|
1954
|
+
clearInterval(intervalId);
|
|
1955
|
+
updateAccounts();
|
|
1956
|
+
}
|
|
1957
|
+
}, 500);
|
|
1958
|
+
setTimeout(() => clearInterval(intervalId), TIMEOUTS.WALLET_DETECTION_POLL);
|
|
1959
|
+
}, [updateAccounts, checkExtensionInstalled]);
|
|
1960
|
+
const updateBalance = useCallback3(async () => {
|
|
1961
|
+
const provider = getCurrentProvider();
|
|
1962
|
+
if (!provider || !accounts.length) {
|
|
1963
|
+
setBalance("");
|
|
1964
|
+
return;
|
|
1965
|
+
}
|
|
1966
|
+
try {
|
|
1967
|
+
switch (selectedWallet) {
|
|
1968
|
+
case "starkey": {
|
|
1969
|
+
const balance2 = await provider.balance();
|
|
1970
|
+
if (balance2) {
|
|
1971
|
+
setBalance(`${balance2.formattedBalance} ${balance2.displayUnit}`);
|
|
1972
|
+
}
|
|
1973
|
+
break;
|
|
1974
|
+
}
|
|
1975
|
+
case "ribbit": {
|
|
1976
|
+
const walletBalanceRequest = {
|
|
1977
|
+
chainId: parseInt(getChainId()),
|
|
1978
|
+
resourceType: "0x1::supra_coin::SupraCoin",
|
|
1979
|
+
decimals: 8
|
|
1980
|
+
};
|
|
1981
|
+
const balanceStr = await provider.getWalletBalance(
|
|
1982
|
+
walletBalanceRequest
|
|
1983
|
+
);
|
|
1984
|
+
logger.debug("Ribbit balance response", { balance: balanceStr });
|
|
1985
|
+
setBalance(`${(balanceStr == null ? void 0 : balanceStr.balance) || 0} SUPRA`);
|
|
1986
|
+
break;
|
|
1987
|
+
}
|
|
1988
|
+
default: {
|
|
1989
|
+
setBalance("");
|
|
1990
|
+
break;
|
|
1991
|
+
}
|
|
1992
|
+
}
|
|
1993
|
+
} catch (error) {
|
|
1994
|
+
logger.error("Failed to update balance", error, { walletType: selectedWallet });
|
|
1995
|
+
setBalance("");
|
|
1996
|
+
}
|
|
1997
|
+
}, [selectedWallet, accounts]);
|
|
1998
|
+
const getNetworkData = useCallback3(async () => {
|
|
1999
|
+
const provider = getCurrentProvider();
|
|
2000
|
+
if (!provider) return {};
|
|
2001
|
+
try {
|
|
2002
|
+
switch (selectedWallet) {
|
|
2003
|
+
case "starkey": {
|
|
2004
|
+
const data = await provider.getChainId();
|
|
2005
|
+
setNetworkData(data || {});
|
|
2006
|
+
return data || {};
|
|
2007
|
+
}
|
|
2008
|
+
case "ribbit": {
|
|
2009
|
+
const chainId = parseInt(getChainId());
|
|
2010
|
+
const mockNetworkData = { chainId: chainId.toString() };
|
|
2011
|
+
setNetworkData(mockNetworkData);
|
|
2012
|
+
return mockNetworkData;
|
|
2013
|
+
}
|
|
2014
|
+
default: {
|
|
2015
|
+
setNetworkData({});
|
|
2016
|
+
return {};
|
|
2017
|
+
}
|
|
2018
|
+
}
|
|
2019
|
+
} catch (error) {
|
|
2020
|
+
logger.error("Failed to get network data", error, { walletType: selectedWallet });
|
|
2021
|
+
setNetworkData({});
|
|
2022
|
+
return {};
|
|
2023
|
+
}
|
|
2024
|
+
}, [selectedWallet]);
|
|
2025
|
+
const connectWallet = async (walletType) => {
|
|
2026
|
+
if (walletType) {
|
|
2027
|
+
updateSelectedWallet(walletType);
|
|
2028
|
+
}
|
|
2029
|
+
const provider = walletType ? WALLET_CONFIGS2[walletType].provider() : getCurrentProvider();
|
|
2030
|
+
if (!provider) {
|
|
2031
|
+
const error = new WalletNotInstalledError(walletType || selectedWallet);
|
|
2032
|
+
logger.warn("Wallet not installed", { walletType: walletType || selectedWallet });
|
|
2033
|
+
toast3.error(ERROR_MESSAGES.WALLET_NOT_INSTALLED, {
|
|
2034
|
+
description: `Please install the ${walletType || selectedWallet} extension`
|
|
2035
|
+
});
|
|
2036
|
+
return false;
|
|
2037
|
+
}
|
|
2038
|
+
setLoading(true);
|
|
2039
|
+
try {
|
|
2040
|
+
switch (walletType || selectedWallet) {
|
|
2041
|
+
case "starkey": {
|
|
2042
|
+
await provider.connect();
|
|
2043
|
+
logger.info("Starkey wallet connection approved");
|
|
2044
|
+
await updateAccounts();
|
|
2045
|
+
const responseAcc = await provider.account();
|
|
2046
|
+
if (responseAcc.length === 0) {
|
|
2047
|
+
throw new WalletConnectionError(ERROR_MESSAGES.NO_ACCOUNT_FOUND);
|
|
2048
|
+
}
|
|
2049
|
+
if (responseAcc.length) {
|
|
2050
|
+
setAccounts(responseAcc);
|
|
2051
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2052
|
+
setStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS, responseAcc[0]);
|
|
2053
|
+
window.dispatchEvent(
|
|
2054
|
+
new CustomEvent(WALLET_EVENTS2.PRESIGNED_STATE, {
|
|
2055
|
+
detail: {
|
|
2056
|
+
timestamp: Date.now(),
|
|
2057
|
+
account: responseAcc[0]
|
|
2058
|
+
}
|
|
2059
|
+
})
|
|
2060
|
+
);
|
|
2061
|
+
window.dispatchEvent(
|
|
2062
|
+
new CustomEvent(WALLET_EVENTS2.CONNECTED, {
|
|
2063
|
+
detail: {
|
|
2064
|
+
timestamp: Date.now(),
|
|
2065
|
+
account: responseAcc[0],
|
|
2066
|
+
wallet: "starkey"
|
|
2067
|
+
}
|
|
2068
|
+
})
|
|
2069
|
+
);
|
|
2070
|
+
try {
|
|
2071
|
+
const message = "To verify your account, please sign this message.";
|
|
2072
|
+
const nonce = generateNonce();
|
|
2073
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
2074
|
+
logger.debug("Signature request initiated", { account: responseAcc[0] });
|
|
2075
|
+
const signatureResponse = await provider.signMessage({
|
|
2076
|
+
message: hexMessage,
|
|
2077
|
+
nonce
|
|
2078
|
+
});
|
|
2079
|
+
if (signatureResponse) {
|
|
2080
|
+
logger.info("Signature approved", { account: responseAcc[0] });
|
|
2081
|
+
window.dispatchEvent(
|
|
2082
|
+
new CustomEvent(WALLET_EVENTS2.POSTSIGNED_STATE, {
|
|
2083
|
+
detail: {
|
|
2084
|
+
timestamp: Date.now(),
|
|
2085
|
+
account: responseAcc[0]
|
|
2086
|
+
}
|
|
2087
|
+
})
|
|
2088
|
+
);
|
|
2089
|
+
}
|
|
2090
|
+
} catch (signError) {
|
|
2091
|
+
const error = new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, signError);
|
|
2092
|
+
logger.error("Signing failed", error, { account: responseAcc[0] });
|
|
2093
|
+
window.dispatchEvent(
|
|
2094
|
+
new CustomEvent(WALLET_EVENTS2.ERROR, {
|
|
2095
|
+
detail: {
|
|
2096
|
+
timestamp: Date.now(),
|
|
2097
|
+
error
|
|
2098
|
+
}
|
|
2099
|
+
})
|
|
2100
|
+
);
|
|
2101
|
+
throw error;
|
|
2102
|
+
}
|
|
2103
|
+
}
|
|
2104
|
+
break;
|
|
2105
|
+
}
|
|
2106
|
+
case "ribbit": {
|
|
2107
|
+
const dappMetadata = {
|
|
2108
|
+
name: "multiwallet",
|
|
2109
|
+
description: "NFT Marketplace and Lootbox Platform",
|
|
2110
|
+
logo: window.location.origin + "/favicon.ico",
|
|
2111
|
+
url: window.location.origin
|
|
2112
|
+
};
|
|
2113
|
+
const response = await provider.connectToWallet(
|
|
2114
|
+
dappMetadata
|
|
2115
|
+
);
|
|
2116
|
+
if (response == null ? void 0 : response.connected) {
|
|
2117
|
+
logger.info("Ribbit wallet connection approved");
|
|
2118
|
+
}
|
|
2119
|
+
if (response.walletAddress == null) {
|
|
2120
|
+
throw new WalletConnectionError(ERROR_MESSAGES.NO_ACCOUNT_FOUND);
|
|
2121
|
+
}
|
|
2122
|
+
if (response == null ? void 0 : response.connected) {
|
|
2123
|
+
await updateAccounts();
|
|
2124
|
+
if (response.walletAddress) {
|
|
2125
|
+
setAccounts([response.walletAddress]);
|
|
2126
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2127
|
+
window.dispatchEvent(
|
|
2128
|
+
new CustomEvent(WALLET_EVENTS2.PRESIGNED_STATE, {
|
|
2129
|
+
detail: {
|
|
2130
|
+
timestamp: Date.now(),
|
|
2131
|
+
account: response.walletAddress
|
|
2132
|
+
}
|
|
2133
|
+
})
|
|
2134
|
+
);
|
|
2135
|
+
window.dispatchEvent(
|
|
2136
|
+
new CustomEvent(WALLET_EVENTS2.CONNECTED, {
|
|
2137
|
+
detail: {
|
|
2138
|
+
timestamp: Date.now(),
|
|
2139
|
+
account: response.walletAddress,
|
|
2140
|
+
wallet: "ribbit"
|
|
2141
|
+
}
|
|
2142
|
+
})
|
|
2143
|
+
);
|
|
2144
|
+
try {
|
|
2145
|
+
const message = "To verify your account, please sign this message.";
|
|
2146
|
+
const nonce = generateNonce();
|
|
2147
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
2148
|
+
logger.debug("Signature request initiated", { account: response.walletAddress });
|
|
2149
|
+
const signatureResponse = await provider.signMessage({
|
|
2150
|
+
message: hexMessage,
|
|
2151
|
+
nonce: parseInt(nonce),
|
|
2152
|
+
chainId: parseInt(getChainId())
|
|
2153
|
+
});
|
|
2154
|
+
if (signatureResponse && signatureResponse.approved) {
|
|
2155
|
+
logger.info("Signature approved", { account: response.walletAddress });
|
|
2156
|
+
window.dispatchEvent(
|
|
2157
|
+
new CustomEvent(WALLET_EVENTS2.POSTSIGNED_STATE, {
|
|
2158
|
+
detail: {
|
|
2159
|
+
timestamp: Date.now(),
|
|
2160
|
+
account: response.walletAddress
|
|
2161
|
+
}
|
|
2162
|
+
})
|
|
2163
|
+
);
|
|
2164
|
+
} else {
|
|
2165
|
+
throw new SignMessageError(signatureResponse.error || ERROR_MESSAGES.SIGNING_REJECTED);
|
|
2166
|
+
}
|
|
2167
|
+
} catch (signError) {
|
|
2168
|
+
const error = new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, signError);
|
|
2169
|
+
logger.error("Signing failed", error, { account: response.walletAddress });
|
|
2170
|
+
window.dispatchEvent(
|
|
2171
|
+
new CustomEvent(WALLET_EVENTS2.ERROR, {
|
|
2172
|
+
detail: {
|
|
2173
|
+
timestamp: Date.now(),
|
|
2174
|
+
error
|
|
2175
|
+
}
|
|
2176
|
+
})
|
|
2177
|
+
);
|
|
2178
|
+
throw error;
|
|
2179
|
+
}
|
|
2180
|
+
}
|
|
2181
|
+
} else {
|
|
2182
|
+
throw new WalletConnectionError(ERROR_MESSAGES.CONNECTION_REJECTED);
|
|
2183
|
+
}
|
|
2184
|
+
break;
|
|
2185
|
+
}
|
|
2186
|
+
default: {
|
|
2187
|
+
throw new WalletConnectionError(
|
|
2188
|
+
`${ERROR_MESSAGES.UNSUPPORTED_WALLET}: ${walletType || selectedWallet}`
|
|
2189
|
+
);
|
|
2190
|
+
}
|
|
2191
|
+
}
|
|
2192
|
+
return true;
|
|
2193
|
+
} catch (error) {
|
|
2194
|
+
const walletError = error instanceof WalletConnectionError ? error : new WalletConnectionError(ERROR_MESSAGES.CONNECTION_FAILED, error);
|
|
2195
|
+
logger.error("Wallet connection failed", walletError, {
|
|
2196
|
+
walletType: walletType || selectedWallet
|
|
2197
|
+
});
|
|
2198
|
+
window.dispatchEvent(
|
|
2199
|
+
new CustomEvent(WALLET_EVENTS2.ERROR, {
|
|
2200
|
+
detail: {
|
|
2201
|
+
timestamp: Date.now(),
|
|
2202
|
+
error: walletError
|
|
2203
|
+
}
|
|
2204
|
+
})
|
|
2205
|
+
);
|
|
2206
|
+
return false;
|
|
2207
|
+
} finally {
|
|
2208
|
+
setLoading(false);
|
|
2209
|
+
}
|
|
2210
|
+
};
|
|
2211
|
+
const disconnectWallet = async () => {
|
|
2212
|
+
const provider = getCurrentProvider();
|
|
2213
|
+
if (!provider) return;
|
|
2214
|
+
try {
|
|
2215
|
+
switch (selectedWallet) {
|
|
2216
|
+
case "starkey": {
|
|
2217
|
+
await provider.disconnect();
|
|
2218
|
+
break;
|
|
2219
|
+
}
|
|
2220
|
+
case "ribbit": {
|
|
2221
|
+
await provider.disconnect();
|
|
2222
|
+
break;
|
|
2223
|
+
}
|
|
2224
|
+
}
|
|
2225
|
+
resetWalletData();
|
|
2226
|
+
clearStoredWalletType2();
|
|
2227
|
+
router.push("/");
|
|
2228
|
+
} catch (error) {
|
|
2229
|
+
logger.error("Wallet disconnect failed", error, { walletType: selectedWallet });
|
|
2230
|
+
resetWalletData();
|
|
2231
|
+
clearStoredWalletType2();
|
|
2232
|
+
}
|
|
2233
|
+
};
|
|
2234
|
+
const resetWalletData = () => {
|
|
2235
|
+
setAccounts([]);
|
|
2236
|
+
setBalance("");
|
|
2237
|
+
setNetworkData({});
|
|
2238
|
+
switch (selectedWallet) {
|
|
2239
|
+
case "starkey": {
|
|
2240
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2241
|
+
removeStorageItem(STORAGE_KEYS.STARKEY_ACCOUNTS);
|
|
2242
|
+
break;
|
|
2243
|
+
}
|
|
2244
|
+
case "ribbit": {
|
|
2245
|
+
break;
|
|
2246
|
+
}
|
|
2247
|
+
}
|
|
2248
|
+
};
|
|
2249
|
+
const getSequenceNumber = async (address) => {
|
|
2250
|
+
const rpcUrl = getRpcUrl();
|
|
2251
|
+
const data = await fetch(`${rpcUrl}/rpc/v1/accounts/${address}`);
|
|
2252
|
+
if (!data.ok) {
|
|
2253
|
+
throw new NetworkError(`Failed to fetch sequence number for ${address}`);
|
|
2254
|
+
}
|
|
2255
|
+
const accountData = await data.json();
|
|
2256
|
+
return accountData.sequence_number;
|
|
2257
|
+
};
|
|
2258
|
+
const sendRawTransaction = async (moduleAddress, moduleName, functionName, params, runTimeParams = [], txExpiryTime = Math.ceil(Date.now() / 1e3) + 3e3) => {
|
|
2259
|
+
const provider = getCurrentProvider();
|
|
2260
|
+
if (!provider || !accounts.length || !moduleAddress || !moduleName || !functionName)
|
|
2261
|
+
return;
|
|
2262
|
+
try {
|
|
2263
|
+
switch (selectedWallet) {
|
|
2264
|
+
case "starkey": {
|
|
2265
|
+
if (!walletCapabilities.rawTransactions) {
|
|
2266
|
+
throw new TransactionError("Raw transactions not supported by current wallet");
|
|
2267
|
+
}
|
|
2268
|
+
let networkData2 = await getNetworkData();
|
|
2269
|
+
const currentChainId = getChainId();
|
|
2270
|
+
if (networkData2.chainId !== currentChainId) {
|
|
2271
|
+
setSelectedChainId(currentChainId);
|
|
2272
|
+
await provider.changeNetwork({
|
|
2273
|
+
chainId: currentChainId
|
|
2274
|
+
});
|
|
2275
|
+
}
|
|
2276
|
+
const rawTxPayload = [
|
|
2277
|
+
accounts[0],
|
|
2278
|
+
0,
|
|
2279
|
+
// sequence number
|
|
2280
|
+
moduleAddress,
|
|
2281
|
+
moduleName,
|
|
2282
|
+
functionName,
|
|
2283
|
+
runTimeParams,
|
|
2284
|
+
params,
|
|
2285
|
+
{}
|
|
2286
|
+
];
|
|
2287
|
+
const data = await provider.createRawTransactionData(rawTxPayload);
|
|
2288
|
+
const txHash = await provider.sendTransaction({
|
|
2289
|
+
data,
|
|
2290
|
+
from: accounts[0],
|
|
2291
|
+
to: moduleAddress,
|
|
2292
|
+
chainId: currentChainId,
|
|
2293
|
+
value: ""
|
|
2294
|
+
});
|
|
2295
|
+
addTransactions(txHash || "failed");
|
|
2296
|
+
logger.info("Transaction sent successfully", { txHash, walletType: "starkey" });
|
|
2297
|
+
return txHash;
|
|
2298
|
+
}
|
|
2299
|
+
case "ribbit": {
|
|
2300
|
+
if (!walletCapabilities.rawTransactions) {
|
|
2301
|
+
throw new TransactionError("Raw transactions not supported by current wallet");
|
|
2302
|
+
}
|
|
2303
|
+
const currentChainId = getChainId();
|
|
2304
|
+
let chainId = SupraChainId2.TESTNET;
|
|
2305
|
+
if (currentChainId === DEFAULT_CHAIN_IDS.TESTNET) {
|
|
2306
|
+
chainId = SupraChainId2.TESTNET;
|
|
2307
|
+
} else if (currentChainId === DEFAULT_CHAIN_IDS.MAINNET) {
|
|
2308
|
+
chainId = SupraChainId2.MAINNET;
|
|
2309
|
+
}
|
|
2310
|
+
const rawTxnRequest = {
|
|
2311
|
+
sender: accounts[0],
|
|
2312
|
+
// Use actual sender address
|
|
2313
|
+
moduleAddress,
|
|
2314
|
+
// Use provided module address
|
|
2315
|
+
moduleName,
|
|
2316
|
+
// Use provided module name
|
|
2317
|
+
functionName,
|
|
2318
|
+
// Use provided function name
|
|
2319
|
+
typeArgs: runTimeParams,
|
|
2320
|
+
// Use converted runtime parameters
|
|
2321
|
+
args: params || [],
|
|
2322
|
+
// Use provided parameters
|
|
2323
|
+
chainId
|
|
2324
|
+
};
|
|
2325
|
+
const rawTxnBase64 = await provider.createRawTransactionBuffer(rawTxnRequest);
|
|
2326
|
+
const response = await provider.signAndSendRawTransaction({
|
|
2327
|
+
rawTxn: rawTxnBase64,
|
|
2328
|
+
chainId,
|
|
2329
|
+
meta: {
|
|
2330
|
+
description: `Call ${moduleName}::${functionName}`
|
|
2331
|
+
// Dynamic description
|
|
2332
|
+
}
|
|
2333
|
+
});
|
|
2334
|
+
if (response.approved) {
|
|
2335
|
+
const txHash = response.txHash || response.result || "success";
|
|
2336
|
+
addTransactions(txHash);
|
|
2337
|
+
logger.info("Transaction sent successfully", { txHash, walletType: "ribbit" });
|
|
2338
|
+
return txHash;
|
|
2339
|
+
} else {
|
|
2340
|
+
throw new TransactionError(response.error || ERROR_MESSAGES.TRANSACTION_REJECTED);
|
|
2341
|
+
}
|
|
2342
|
+
}
|
|
2343
|
+
default: {
|
|
2344
|
+
throw new TransactionError(
|
|
2345
|
+
`Raw transactions not supported for wallet: ${selectedWallet}`
|
|
2346
|
+
);
|
|
2347
|
+
}
|
|
2348
|
+
}
|
|
2349
|
+
} catch (error) {
|
|
2350
|
+
const txError = error instanceof TransactionError ? error : new TransactionError(ERROR_MESSAGES.TRANSACTION_FAILED, void 0, error);
|
|
2351
|
+
logger.error("Transaction failed", txError, {
|
|
2352
|
+
walletType: selectedWallet,
|
|
2353
|
+
moduleAddress,
|
|
2354
|
+
moduleName,
|
|
2355
|
+
functionName
|
|
2356
|
+
});
|
|
2357
|
+
throw txError;
|
|
2358
|
+
}
|
|
2359
|
+
};
|
|
2360
|
+
const signMessage = async (message, nonce, account, forceSign = false) => {
|
|
2361
|
+
const provider = getCurrentProvider();
|
|
2362
|
+
if (!provider) return;
|
|
2363
|
+
const secureNonce = nonce || generateNonce();
|
|
2364
|
+
switch (selectedWallet) {
|
|
2365
|
+
case "starkey": {
|
|
2366
|
+
if (!walletCapabilities.signMessage) {
|
|
2367
|
+
throw new SignMessageError("Message signing not supported by current wallet");
|
|
2368
|
+
}
|
|
2369
|
+
if (!accounts.length && !account) return;
|
|
2370
|
+
if (!accounts.length && account) {
|
|
2371
|
+
accounts[0] = account;
|
|
2372
|
+
}
|
|
2373
|
+
if (getStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET) === "true" && !forceSign) {
|
|
2374
|
+
return;
|
|
2375
|
+
}
|
|
2376
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "true");
|
|
2377
|
+
try {
|
|
2378
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
2379
|
+
const response = await provider.signMessage({
|
|
2380
|
+
message: hexMessage,
|
|
2381
|
+
nonce: secureNonce
|
|
2382
|
+
});
|
|
2383
|
+
const { publicKey, signature } = response;
|
|
2384
|
+
const verified = nacl2.sign.detached.verify(
|
|
2385
|
+
new TextEncoder().encode(message),
|
|
2386
|
+
Uint8Array.from(Buffer.from(signature.slice(2), "hex")),
|
|
2387
|
+
Uint8Array.from(Buffer.from(publicKey.slice(2), "hex"))
|
|
2388
|
+
);
|
|
2389
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2390
|
+
logger.info("Message signed successfully", { walletType: "starkey" });
|
|
2391
|
+
return __spreadProps(__spreadValues({}, response), { verified });
|
|
2392
|
+
} catch (error) {
|
|
2393
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2394
|
+
throw new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, error);
|
|
2395
|
+
}
|
|
2396
|
+
}
|
|
2397
|
+
case "ribbit": {
|
|
2398
|
+
if (!walletCapabilities.signMessage) {
|
|
2399
|
+
throw new SignMessageError("Message signing not supported by current wallet");
|
|
2400
|
+
}
|
|
2401
|
+
if (!accounts.length && !account) return;
|
|
2402
|
+
if (!accounts.length && account) {
|
|
2403
|
+
accounts[0] = account;
|
|
2404
|
+
}
|
|
2405
|
+
if (getStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET) === "true" && !forceSign) {
|
|
2406
|
+
return;
|
|
2407
|
+
}
|
|
2408
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "true");
|
|
2409
|
+
try {
|
|
2410
|
+
const hexMessage = "0x" + Buffer.from(message, "utf8").toString("hex");
|
|
2411
|
+
const response = await provider.signMessage({
|
|
2412
|
+
message: hexMessage,
|
|
2413
|
+
nonce: parseInt(secureNonce),
|
|
2414
|
+
chainId: parseInt(getChainId())
|
|
2415
|
+
});
|
|
2416
|
+
if (response.approved && response.publicKey && response.signature) {
|
|
2417
|
+
const { publicKey, signature } = response;
|
|
2418
|
+
const verified = nacl2.sign.detached.verify(
|
|
2419
|
+
new TextEncoder().encode(message),
|
|
2420
|
+
Uint8Array.from(Buffer.from(signature.slice(2), "hex")),
|
|
2421
|
+
Uint8Array.from(Buffer.from(publicKey.slice(2), "hex"))
|
|
2422
|
+
);
|
|
2423
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2424
|
+
logger.info("Message signed successfully", { walletType: "ribbit" });
|
|
2425
|
+
return __spreadProps(__spreadValues({}, response), { verified });
|
|
2426
|
+
} else {
|
|
2427
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2428
|
+
throw new SignMessageError(response.error || ERROR_MESSAGES.SIGNING_REJECTED);
|
|
2429
|
+
}
|
|
2430
|
+
} catch (error) {
|
|
2431
|
+
setStorageItem(STORAGE_KEYS.IS_SIGNING_WALLET, "false");
|
|
2432
|
+
throw new SignMessageError(ERROR_MESSAGES.SIGNING_FAILED, error);
|
|
2433
|
+
}
|
|
2434
|
+
}
|
|
2435
|
+
default: {
|
|
2436
|
+
throw new SignMessageError(
|
|
2437
|
+
`Message signing not supported for wallet: ${selectedWallet}`
|
|
2438
|
+
);
|
|
2439
|
+
}
|
|
2440
|
+
}
|
|
2441
|
+
};
|
|
2442
|
+
useEffect3(() => {
|
|
2443
|
+
if (selectedWallet === "starkey" && walletCapabilities.eventListeners) {
|
|
2444
|
+
const handleExtensionEvents = (event) => {
|
|
2445
|
+
var _a, _b, _c;
|
|
2446
|
+
if ((_b = (_a = event == null ? void 0 : event.data) == null ? void 0 : _a.name) == null ? void 0 : _b.startsWith("starkey-")) {
|
|
2447
|
+
switch ((_c = event == null ? void 0 : event.data) == null ? void 0 : _c.name) {
|
|
2448
|
+
case "starkey-extension-installed": {
|
|
2449
|
+
checkIsExtensionInstalled();
|
|
2450
|
+
break;
|
|
2451
|
+
}
|
|
2452
|
+
case "starkey-wallet-updated": {
|
|
2453
|
+
(async () => {
|
|
2454
|
+
const responseAcc = await supraProvider.account();
|
|
2455
|
+
if (responseAcc.length) {
|
|
2456
|
+
setAccounts(responseAcc);
|
|
2457
|
+
window.dispatchEvent(
|
|
2458
|
+
new CustomEvent(WALLET_EVENTS2.CONNECTED, {
|
|
2459
|
+
detail: {
|
|
2460
|
+
timestamp: Date.now(),
|
|
2461
|
+
account: responseAcc[0]
|
|
2462
|
+
}
|
|
2463
|
+
})
|
|
2464
|
+
);
|
|
2465
|
+
await updateBalance();
|
|
2466
|
+
await getNetworkData();
|
|
2467
|
+
} else {
|
|
2468
|
+
logger.debug("Starkey wallet updated: No accounts found - Resetting");
|
|
2469
|
+
resetWalletData();
|
|
2470
|
+
}
|
|
2471
|
+
setLoading(false);
|
|
2472
|
+
})();
|
|
2473
|
+
break;
|
|
2474
|
+
}
|
|
2475
|
+
case "starkey-wallet-disconnected": {
|
|
2476
|
+
resetWalletData();
|
|
2477
|
+
router.push("/");
|
|
2478
|
+
setLoading(false);
|
|
2479
|
+
break;
|
|
2480
|
+
}
|
|
2481
|
+
case "starkey-window-removed": {
|
|
2482
|
+
setLoading(false);
|
|
2483
|
+
break;
|
|
2484
|
+
}
|
|
2485
|
+
}
|
|
2486
|
+
}
|
|
2487
|
+
};
|
|
2488
|
+
checkIsExtensionInstalled();
|
|
2489
|
+
window.addEventListener("message", handleExtensionEvents);
|
|
2490
|
+
return () => window.removeEventListener("message", handleExtensionEvents);
|
|
2491
|
+
}
|
|
2492
|
+
}, [selectedWallet, walletCapabilities, supraProvider]);
|
|
2493
|
+
const getAvailableWallets = useCallback3(() => {
|
|
2494
|
+
const availableWallets = [];
|
|
2495
|
+
Object.entries(WALLET_CONFIGS2).forEach(([walletType, config2]) => {
|
|
2496
|
+
const provider = config2.provider();
|
|
2497
|
+
const isInstalled = !!provider;
|
|
2498
|
+
switch (walletType) {
|
|
2499
|
+
case "starkey": {
|
|
2500
|
+
availableWallets.push({
|
|
2501
|
+
type: "starkey",
|
|
2502
|
+
name: "Starkey Wallet",
|
|
2503
|
+
isInstalled,
|
|
2504
|
+
capabilities: config2.capabilities
|
|
2505
|
+
});
|
|
2506
|
+
break;
|
|
2507
|
+
}
|
|
2508
|
+
case "ribbit": {
|
|
2509
|
+
availableWallets.push({
|
|
2510
|
+
type: "ribbit",
|
|
2511
|
+
name: "Ribbit Wallet",
|
|
2512
|
+
isInstalled,
|
|
2513
|
+
capabilities: config2.capabilities
|
|
2514
|
+
});
|
|
2515
|
+
break;
|
|
2516
|
+
}
|
|
2517
|
+
}
|
|
2518
|
+
});
|
|
2519
|
+
return availableWallets;
|
|
2520
|
+
}, []);
|
|
2521
|
+
const updateSelectedWallet = (walletType) => {
|
|
2522
|
+
setSelectedWallet(walletType);
|
|
2523
|
+
setWalletCapabilities(WALLET_CONFIGS2[walletType].capabilities);
|
|
2524
|
+
setStoredWalletType2(walletType);
|
|
2525
|
+
};
|
|
2526
|
+
return {
|
|
2527
|
+
// New wallet selection functionality
|
|
2528
|
+
selectedWallet,
|
|
2529
|
+
walletCapabilities,
|
|
2530
|
+
getAvailableWallets,
|
|
2531
|
+
// Add this new function
|
|
2532
|
+
// Existing interface (unchanged)
|
|
2533
|
+
getCurrentProvider,
|
|
2534
|
+
isExtensionInstalled,
|
|
2535
|
+
accounts,
|
|
2536
|
+
networkData,
|
|
2537
|
+
balance,
|
|
2538
|
+
transactions,
|
|
2539
|
+
selectedChainId,
|
|
2540
|
+
connectWallet,
|
|
2541
|
+
// Now accepts optional walletType parameter
|
|
2542
|
+
disconnectWallet,
|
|
2543
|
+
sendRawTransaction,
|
|
2544
|
+
signMessage,
|
|
2545
|
+
setSelectedChainId,
|
|
2546
|
+
// switchToChain,
|
|
2547
|
+
loading
|
|
2548
|
+
// authFetch,
|
|
2549
|
+
// checkAndRevalidateToken,
|
|
2550
|
+
// signIn,
|
|
2551
|
+
};
|
|
2552
|
+
};
|
|
2553
|
+
var useSupraMultiWallet_default2 = useSupraMultiWallet2;
|
|
2554
|
+
|
|
2555
|
+
// src/hooks/useConversionUtils.ts
|
|
2556
|
+
import { useCallback as useCallback4 } from "react";
|
|
2557
|
+
import { BCS as BCS3, TxnBuilderTypes } from "supra-l1-sdk-core";
|
|
2558
|
+
|
|
2559
|
+
// src/lib/abis/supra_account.ts
|
|
2560
|
+
var supraAccountABI = {
|
|
2561
|
+
address: "0x1",
|
|
2562
|
+
name: "supra_account",
|
|
2563
|
+
friends: [
|
|
2564
|
+
"0x1::genesis",
|
|
2565
|
+
"0x1::resource_account",
|
|
2566
|
+
"0x1::transaction_fee",
|
|
2567
|
+
"0x1::transaction_validation"
|
|
2568
|
+
],
|
|
2569
|
+
exposed_functions: [
|
|
2570
|
+
{
|
|
2571
|
+
name: "assert_account_exists",
|
|
2572
|
+
visibility: "public",
|
|
2573
|
+
is_entry: false,
|
|
2574
|
+
is_view: false,
|
|
2575
|
+
generic_type_params: [],
|
|
2576
|
+
params: ["address"],
|
|
2577
|
+
return: []
|
|
2578
|
+
},
|
|
2579
|
+
{
|
|
2580
|
+
name: "assert_account_is_registered_for_apt",
|
|
2581
|
+
visibility: "public",
|
|
2582
|
+
is_entry: false,
|
|
2583
|
+
is_view: false,
|
|
2584
|
+
generic_type_params: [],
|
|
2585
|
+
params: ["address"],
|
|
2586
|
+
return: []
|
|
2587
|
+
},
|
|
2588
|
+
{
|
|
2589
|
+
name: "assert_account_is_registered_for_supra",
|
|
2590
|
+
visibility: "public",
|
|
2591
|
+
is_entry: false,
|
|
2592
|
+
is_view: false,
|
|
2593
|
+
generic_type_params: [],
|
|
2594
|
+
params: ["address"],
|
|
2595
|
+
return: []
|
|
2596
|
+
},
|
|
2597
|
+
{
|
|
2598
|
+
name: "batch_transfer",
|
|
2599
|
+
visibility: "public",
|
|
2600
|
+
is_entry: true,
|
|
2601
|
+
is_view: false,
|
|
2602
|
+
generic_type_params: [],
|
|
2603
|
+
params: ["&signer", "vector<address>", "vector<u64>"],
|
|
2604
|
+
return: []
|
|
2605
|
+
},
|
|
2606
|
+
{
|
|
2607
|
+
name: "batch_transfer_coins",
|
|
2608
|
+
visibility: "public",
|
|
2609
|
+
is_entry: true,
|
|
2610
|
+
is_view: false,
|
|
2611
|
+
generic_type_params: [{ constraints: [] }],
|
|
2612
|
+
params: ["&signer", "vector<address>", "vector<u64>"],
|
|
2613
|
+
return: []
|
|
2614
|
+
},
|
|
2615
|
+
{
|
|
2616
|
+
name: "burn_from_fungible_store",
|
|
2617
|
+
visibility: "friend",
|
|
2618
|
+
is_entry: false,
|
|
2619
|
+
is_view: false,
|
|
2620
|
+
generic_type_params: [],
|
|
2621
|
+
params: ["&0x1::fungible_asset::BurnRef", "address", "u64"],
|
|
2622
|
+
return: []
|
|
2623
|
+
},
|
|
2624
|
+
{
|
|
2625
|
+
name: "can_receive_direct_coin_transfers",
|
|
2626
|
+
visibility: "public",
|
|
2627
|
+
is_entry: false,
|
|
2628
|
+
is_view: true,
|
|
2629
|
+
generic_type_params: [],
|
|
2630
|
+
params: ["address"],
|
|
2631
|
+
return: ["bool"]
|
|
2632
|
+
},
|
|
2633
|
+
{
|
|
2634
|
+
name: "create_account",
|
|
2635
|
+
visibility: "public",
|
|
2636
|
+
is_entry: true,
|
|
2637
|
+
is_view: false,
|
|
2638
|
+
generic_type_params: [],
|
|
2639
|
+
params: ["address"],
|
|
2640
|
+
return: []
|
|
2641
|
+
},
|
|
2642
|
+
{
|
|
2643
|
+
name: "deposit_coins",
|
|
2644
|
+
visibility: "public",
|
|
2645
|
+
is_entry: false,
|
|
2646
|
+
is_view: false,
|
|
2647
|
+
generic_type_params: [{ constraints: [] }],
|
|
2648
|
+
params: ["address", "0x1::coin::Coin<T0>"],
|
|
2649
|
+
return: []
|
|
2650
|
+
},
|
|
2651
|
+
{
|
|
2652
|
+
name: "is_fungible_balance_at_least",
|
|
2653
|
+
visibility: "friend",
|
|
2654
|
+
is_entry: false,
|
|
2655
|
+
is_view: false,
|
|
2656
|
+
generic_type_params: [],
|
|
2657
|
+
params: ["address", "u64"],
|
|
2658
|
+
return: ["bool"]
|
|
2659
|
+
},
|
|
2660
|
+
{
|
|
2661
|
+
name: "register_supra",
|
|
2662
|
+
visibility: "friend",
|
|
2663
|
+
is_entry: false,
|
|
2664
|
+
is_view: false,
|
|
2665
|
+
generic_type_params: [],
|
|
2666
|
+
params: ["&signer"],
|
|
2667
|
+
return: []
|
|
2668
|
+
},
|
|
2669
|
+
{
|
|
2670
|
+
name: "set_allow_direct_coin_transfers",
|
|
2671
|
+
visibility: "public",
|
|
2672
|
+
is_entry: true,
|
|
2673
|
+
is_view: false,
|
|
2674
|
+
generic_type_params: [],
|
|
2675
|
+
params: ["&signer", "bool"],
|
|
2676
|
+
return: []
|
|
2677
|
+
},
|
|
2678
|
+
{
|
|
2679
|
+
name: "transfer",
|
|
2680
|
+
visibility: "public",
|
|
2681
|
+
is_entry: true,
|
|
2682
|
+
is_view: false,
|
|
2683
|
+
generic_type_params: [],
|
|
2684
|
+
params: ["&signer", "address", "u64"],
|
|
2685
|
+
return: []
|
|
2686
|
+
},
|
|
2687
|
+
{
|
|
2688
|
+
name: "transfer_coins",
|
|
2689
|
+
visibility: "public",
|
|
2690
|
+
is_entry: true,
|
|
2691
|
+
is_view: false,
|
|
2692
|
+
generic_type_params: [{ constraints: [] }],
|
|
2693
|
+
params: ["&signer", "address", "u64"],
|
|
2694
|
+
return: []
|
|
2695
|
+
}
|
|
2696
|
+
],
|
|
2697
|
+
structs: [
|
|
2698
|
+
{
|
|
2699
|
+
name: "DirectCoinTransferConfigUpdated",
|
|
2700
|
+
is_native: false,
|
|
2701
|
+
abilities: ["drop", "store"],
|
|
2702
|
+
generic_type_params: [],
|
|
2703
|
+
fields: [
|
|
2704
|
+
{ name: "account", type: "address" },
|
|
2705
|
+
{ name: "new_allow_direct_transfers", type: "bool" }
|
|
2706
|
+
]
|
|
2707
|
+
},
|
|
2708
|
+
{
|
|
2709
|
+
name: "DirectCoinTransferConfigUpdatedEvent",
|
|
2710
|
+
is_native: false,
|
|
2711
|
+
abilities: ["drop", "store"],
|
|
2712
|
+
generic_type_params: [],
|
|
2713
|
+
fields: [{ name: "new_allow_direct_transfers", type: "bool" }]
|
|
2714
|
+
},
|
|
2715
|
+
{
|
|
2716
|
+
name: "DirectTransferConfig",
|
|
2717
|
+
is_native: false,
|
|
2718
|
+
abilities: ["key"],
|
|
2719
|
+
generic_type_params: [],
|
|
2720
|
+
fields: [
|
|
2721
|
+
{ name: "allow_arbitrary_coin_transfers", type: "bool" },
|
|
2722
|
+
{
|
|
2723
|
+
name: "update_coin_transfer_events",
|
|
2724
|
+
type: "0x1::event::EventHandle<0x1::supra_account::DirectCoinTransferConfigUpdatedEvent>"
|
|
2725
|
+
}
|
|
2726
|
+
]
|
|
2727
|
+
}
|
|
2728
|
+
]
|
|
2729
|
+
};
|
|
2730
|
+
|
|
2731
|
+
// src/lib/utils.ts
|
|
2732
|
+
import { clsx as clsx2 } from "clsx";
|
|
2733
|
+
import { twMerge as twMerge2 } from "tailwind-merge";
|
|
2734
|
+
function cn2(...inputs) {
|
|
2735
|
+
return twMerge2(clsx2(inputs));
|
|
2736
|
+
}
|
|
2737
|
+
function standardizeAddress(address) {
|
|
2738
|
+
let cleanAddress = address.replace(/^0x/, "");
|
|
2739
|
+
if (cleanAddress.length < 64) {
|
|
2740
|
+
cleanAddress = cleanAddress.padStart(64, "0");
|
|
2741
|
+
}
|
|
2742
|
+
if (cleanAddress.length > 64) {
|
|
2743
|
+
throw new Error(`Address ${address} is not a valid address`);
|
|
2744
|
+
}
|
|
2745
|
+
return `0x${cleanAddress}`;
|
|
2746
|
+
}
|
|
2747
|
+
|
|
2748
|
+
// src/lib/abiStorage.ts
|
|
2749
|
+
var abiStorage = {
|
|
2750
|
+
[standardizeAddress("0x1")]: {
|
|
2751
|
+
supra_account: supraAccountABI
|
|
2752
|
+
}
|
|
2753
|
+
};
|
|
2754
|
+
function getStoredABI(moduleAddress, moduleName) {
|
|
2755
|
+
var _a;
|
|
2756
|
+
const standardizedAddress = standardizeAddress(moduleAddress);
|
|
2757
|
+
const normalizedName = moduleName;
|
|
2758
|
+
if ((_a = abiStorage[standardizedAddress]) == null ? void 0 : _a[normalizedName]) {
|
|
2759
|
+
return abiStorage[standardizedAddress][normalizedName];
|
|
2760
|
+
}
|
|
2761
|
+
return null;
|
|
2762
|
+
}
|
|
2763
|
+
|
|
2764
|
+
// src/hooks/useConversionUtils.ts
|
|
2765
|
+
var useConversionUtils = () => {
|
|
2766
|
+
const stringToUint8Array = useCallback4((humanReadableStr) => {
|
|
2767
|
+
return BCS3.bcsToBytes(new TxnBuilderTypes.Identifier(humanReadableStr));
|
|
2768
|
+
}, []);
|
|
2769
|
+
const serializeString = useCallback4((humanReadableStr) => {
|
|
2770
|
+
return BCS3.bcsSerializeStr(humanReadableStr);
|
|
2771
|
+
}, []);
|
|
2772
|
+
const addressToUint8Array = useCallback4((cryptoAddress) => {
|
|
2773
|
+
return BCS3.bcsToBytes(TxnBuilderTypes.AccountAddress.fromHex(cryptoAddress));
|
|
2774
|
+
}, []);
|
|
2775
|
+
const deserializeString = (uint8Array) => {
|
|
2776
|
+
return BCS3.bcsSerializeStr(uint8Array);
|
|
2777
|
+
};
|
|
2778
|
+
const deserializeVector = (uint8Array) => {
|
|
2779
|
+
const deserializer = new BCS3.Deserializer(uint8Array);
|
|
2780
|
+
return BCS3.deserializeVector(deserializer, BCS3.bcsSerializeU8);
|
|
2781
|
+
};
|
|
2782
|
+
const serializeUint8 = useCallback4((value) => {
|
|
2783
|
+
const num = typeof value === "string" ? parseInt(value, 10) : value;
|
|
2784
|
+
if (num < 0 || num > 255) {
|
|
2785
|
+
throw new Error(`u8 value out of range: ${num}`);
|
|
2786
|
+
}
|
|
2787
|
+
return BCS3.bcsSerializeU8(num);
|
|
2788
|
+
}, []);
|
|
2789
|
+
const serializeUint16 = useCallback4((value) => {
|
|
2790
|
+
let num;
|
|
2791
|
+
if (typeof value === "string") {
|
|
2792
|
+
num = parseInt(value, 10);
|
|
2793
|
+
} else if (typeof value === "bigint") {
|
|
2794
|
+
num = Number(value);
|
|
2795
|
+
} else {
|
|
2796
|
+
num = value;
|
|
2797
|
+
}
|
|
2798
|
+
if (num < 0 || num > 65535) {
|
|
2799
|
+
throw new Error(`u16 value out of range: ${num}`);
|
|
2800
|
+
}
|
|
2801
|
+
return BCS3.bcsSerializeU16(num);
|
|
2802
|
+
}, []);
|
|
2803
|
+
const serializeUint32 = useCallback4((value) => {
|
|
2804
|
+
let num;
|
|
2805
|
+
if (typeof value === "string") {
|
|
2806
|
+
num = parseInt(value, 10);
|
|
2807
|
+
} else if (typeof value === "bigint") {
|
|
2808
|
+
num = Number(value);
|
|
2809
|
+
} else {
|
|
2810
|
+
num = value;
|
|
2811
|
+
}
|
|
2812
|
+
if (num < 0 || num > 4294967295) {
|
|
2813
|
+
throw new Error(`u32 value out of range: ${num}`);
|
|
2814
|
+
}
|
|
2815
|
+
return BCS3.bcsSerializeU32(num);
|
|
2816
|
+
}, []);
|
|
2817
|
+
const serializeUint64 = useCallback4((value) => {
|
|
2818
|
+
let num;
|
|
2819
|
+
if (typeof value === "string") {
|
|
2820
|
+
num = BigInt(value);
|
|
2821
|
+
} else if (typeof value === "number") {
|
|
2822
|
+
num = BigInt(value);
|
|
2823
|
+
} else {
|
|
2824
|
+
num = value;
|
|
2825
|
+
}
|
|
2826
|
+
if (num < 0) {
|
|
2827
|
+
throw new Error(`u64 value cannot be negative: ${num}`);
|
|
2828
|
+
}
|
|
2829
|
+
return BCS3.bcsSerializeUint64(num);
|
|
2830
|
+
}, []);
|
|
2831
|
+
const serializeUint128 = useCallback4((value) => {
|
|
2832
|
+
let num;
|
|
2833
|
+
if (typeof value === "string") {
|
|
2834
|
+
num = BigInt(value);
|
|
2835
|
+
} else if (typeof value === "number") {
|
|
2836
|
+
num = BigInt(value);
|
|
2837
|
+
} else {
|
|
2838
|
+
num = value;
|
|
2839
|
+
}
|
|
2840
|
+
if (num < 0) {
|
|
2841
|
+
throw new Error(`u128 value cannot be negative: ${num}`);
|
|
2842
|
+
}
|
|
2843
|
+
return BCS3.bcsSerializeU128(num);
|
|
2844
|
+
}, []);
|
|
2845
|
+
const serializeU256 = useCallback4((value) => {
|
|
2846
|
+
return BCS3.bcsSerializeU256(value);
|
|
2847
|
+
}, []);
|
|
2848
|
+
const serializeBool = useCallback4((value) => {
|
|
2849
|
+
return BCS3.bcsSerializeBool(value);
|
|
2850
|
+
}, []);
|
|
2851
|
+
const serializeVector = useCallback4((values, type) => {
|
|
2852
|
+
const serializer = new BCS3.Serializer();
|
|
2853
|
+
serializer.serializeU32AsUleb128(values.length);
|
|
2854
|
+
values.forEach((value) => {
|
|
2855
|
+
if (type === "u64") {
|
|
2856
|
+
serializer.serializeU64(value);
|
|
2857
|
+
} else if (type === "bool") {
|
|
2858
|
+
serializer.serializeBool(value);
|
|
2859
|
+
} else if (type === "string") {
|
|
2860
|
+
serializer.serializeStr(value);
|
|
2861
|
+
} else if (type === "address") {
|
|
2862
|
+
const accountAddress = TxnBuilderTypes.AccountAddress.fromHex(value);
|
|
2863
|
+
serializer.serializeFixedBytes(accountAddress.address);
|
|
2864
|
+
} else {
|
|
2865
|
+
serializer.serializeStr(value);
|
|
2866
|
+
}
|
|
2867
|
+
});
|
|
2868
|
+
return serializer.getBytes();
|
|
2869
|
+
}, []);
|
|
2870
|
+
const hexToString = (hex, type) => {
|
|
2871
|
+
if (!hex) {
|
|
2872
|
+
return "";
|
|
2873
|
+
}
|
|
2874
|
+
if (type !== "String") {
|
|
2875
|
+
try {
|
|
2876
|
+
return BigInt(hex).toString();
|
|
2877
|
+
} catch (error) {
|
|
2878
|
+
console.error("Error converting hex to string:", error);
|
|
2879
|
+
return hex;
|
|
2880
|
+
}
|
|
2881
|
+
}
|
|
2882
|
+
try {
|
|
2883
|
+
const cleanHex = hex.slice(2);
|
|
2884
|
+
return Buffer.from(cleanHex, "hex").toString().slice(1);
|
|
2885
|
+
} catch (error) {
|
|
2886
|
+
console.error("Error converting hex to string:", error);
|
|
2887
|
+
return hex;
|
|
2888
|
+
}
|
|
2889
|
+
};
|
|
2890
|
+
const stringToHex = (str) => {
|
|
2891
|
+
const encoder = new TextEncoder();
|
|
2892
|
+
const bytes = encoder.encode(str);
|
|
2893
|
+
let hexString = "";
|
|
2894
|
+
for (let i = 0; i < bytes.length; i++) {
|
|
2895
|
+
const hex = bytes[i].toString(16).padStart(2, "0");
|
|
2896
|
+
hexString += hex;
|
|
2897
|
+
}
|
|
2898
|
+
return hexString;
|
|
2899
|
+
};
|
|
2900
|
+
const serializeValueByType = useCallback4(
|
|
2901
|
+
(value, type, serializer) => {
|
|
2902
|
+
var _a;
|
|
2903
|
+
const ser = serializer || new BCS3.Serializer();
|
|
2904
|
+
const shouldReturnBytes = !serializer;
|
|
2905
|
+
if (type.startsWith("0x1::option::Option<")) {
|
|
2906
|
+
if (value === null || value === void 0) {
|
|
2907
|
+
ser.serializeU8(0);
|
|
2908
|
+
} else {
|
|
2909
|
+
ser.serializeU8(1);
|
|
2910
|
+
const innerType = type.slice(21, -1);
|
|
2911
|
+
serializeValueByType(value, innerType, ser);
|
|
2912
|
+
}
|
|
2913
|
+
return shouldReturnBytes ? ser.getBytes() : new Uint8Array(0);
|
|
2914
|
+
}
|
|
2915
|
+
if (type.startsWith("vector<")) {
|
|
2916
|
+
const vectorMatch = type.match(/vector<(.+)>$/);
|
|
2917
|
+
if (!vectorMatch) {
|
|
2918
|
+
throw new Error(`Invalid vector type format: ${type}`);
|
|
2919
|
+
}
|
|
2920
|
+
const innerType = vectorMatch[1];
|
|
2921
|
+
if (innerType === "u8") {
|
|
2922
|
+
let bytes;
|
|
2923
|
+
if (typeof value === "string") {
|
|
2924
|
+
const cleanHex = value.startsWith("0x") ? value.slice(2) : value;
|
|
2925
|
+
bytes = new Uint8Array(
|
|
2926
|
+
((_a = cleanHex.match(/.{1,2}/g)) == null ? void 0 : _a.map((byte) => parseInt(byte, 16))) || []
|
|
2927
|
+
);
|
|
2928
|
+
} else if (value instanceof Uint8Array) {
|
|
2929
|
+
bytes = value;
|
|
2930
|
+
} else if (Array.isArray(value)) {
|
|
2931
|
+
bytes = new Uint8Array(
|
|
2932
|
+
value.map((item) => {
|
|
2933
|
+
const u8 = typeof item === "string" ? parseInt(item, 10) : item;
|
|
2934
|
+
if (u8 < 0 || u8 > 255) {
|
|
2935
|
+
throw new Error(`u8 value out of range in vector: ${u8}`);
|
|
2936
|
+
}
|
|
2937
|
+
return u8;
|
|
2938
|
+
})
|
|
2939
|
+
);
|
|
2940
|
+
} else {
|
|
2941
|
+
throw new Error(
|
|
2942
|
+
`Expected string, Uint8Array, or number[] for vector<u8>, got ${typeof value}`
|
|
2943
|
+
);
|
|
2944
|
+
}
|
|
2945
|
+
ser.serializeU32AsUleb128(bytes.length);
|
|
2946
|
+
for (let i = 0; i < bytes.length; i++) {
|
|
2947
|
+
ser.serializeU8(bytes[i]);
|
|
2948
|
+
}
|
|
2949
|
+
} else {
|
|
2950
|
+
if (!Array.isArray(value)) {
|
|
2951
|
+
throw new Error(
|
|
2952
|
+
`Expected array for vector<${innerType}>, got ${typeof value}`
|
|
2953
|
+
);
|
|
2954
|
+
}
|
|
2955
|
+
ser.serializeU32AsUleb128(value.length);
|
|
2956
|
+
for (const item of value) {
|
|
2957
|
+
serializeValueByType(item, innerType, ser);
|
|
2958
|
+
}
|
|
2959
|
+
}
|
|
2960
|
+
return shouldReturnBytes ? ser.getBytes() : new Uint8Array(0);
|
|
2961
|
+
}
|
|
2962
|
+
if (type.startsWith("0x1::object::Object")) {
|
|
2963
|
+
if (typeof value !== "string") {
|
|
2964
|
+
throw new Error(`Expected string for Object, got ${typeof value}`);
|
|
2965
|
+
}
|
|
2966
|
+
const objectAddress = TxnBuilderTypes.AccountAddress.fromHex(value);
|
|
2967
|
+
ser.serializeFixedBytes(objectAddress.address);
|
|
2968
|
+
return shouldReturnBytes ? ser.getBytes() : new Uint8Array(0);
|
|
2969
|
+
}
|
|
2970
|
+
switch (type) {
|
|
2971
|
+
case "address":
|
|
2972
|
+
if (typeof value !== "string") {
|
|
2973
|
+
throw new Error(`Expected string for address, got ${typeof value}`);
|
|
2974
|
+
}
|
|
2975
|
+
const accountAddress = TxnBuilderTypes.AccountAddress.fromHex(value);
|
|
2976
|
+
ser.serializeFixedBytes(accountAddress.address);
|
|
2977
|
+
break;
|
|
2978
|
+
case "u8":
|
|
2979
|
+
const u8 = typeof value === "string" ? parseInt(value, 10) : value;
|
|
2980
|
+
if (u8 < 0 || u8 > 255) {
|
|
2981
|
+
throw new Error(`u8 value out of range: ${u8}`);
|
|
2982
|
+
}
|
|
2983
|
+
ser.serializeU8(u8);
|
|
2984
|
+
break;
|
|
2985
|
+
case "u16":
|
|
2986
|
+
const u16 = typeof value === "string" ? parseInt(value, 10) : typeof value === "bigint" ? Number(value) : value;
|
|
2987
|
+
if (u16 < 0 || u16 > 65535) {
|
|
2988
|
+
throw new Error(`u16 value out of range: ${u16}`);
|
|
2989
|
+
}
|
|
2990
|
+
ser.serializeU16(u16);
|
|
2991
|
+
break;
|
|
2992
|
+
case "u32":
|
|
2993
|
+
const u32 = typeof value === "string" ? parseInt(value, 10) : typeof value === "bigint" ? Number(value) : value;
|
|
2994
|
+
if (u32 < 0 || u32 > 4294967295) {
|
|
2995
|
+
throw new Error(`u32 value out of range: ${u32}`);
|
|
2996
|
+
}
|
|
2997
|
+
ser.serializeU32(u32);
|
|
2998
|
+
break;
|
|
2999
|
+
case "u64":
|
|
3000
|
+
const u64 = typeof value === "string" ? BigInt(value) : typeof value === "number" ? BigInt(value) : value;
|
|
3001
|
+
if (u64 < 0) {
|
|
3002
|
+
throw new Error(`u64 value cannot be negative: ${u64}`);
|
|
3003
|
+
}
|
|
3004
|
+
ser.serializeU64(u64);
|
|
3005
|
+
break;
|
|
3006
|
+
case "u128":
|
|
3007
|
+
const u128 = typeof value === "string" ? BigInt(value) : typeof value === "number" ? BigInt(value) : value;
|
|
3008
|
+
if (u128 < 0) {
|
|
3009
|
+
throw new Error(`u128 value cannot be negative: ${u128}`);
|
|
3010
|
+
}
|
|
3011
|
+
ser.serializeU128(u128);
|
|
3012
|
+
break;
|
|
3013
|
+
case "u256":
|
|
3014
|
+
if (typeof value !== "bigint") {
|
|
3015
|
+
throw new Error(`Expected bigint for u256, got ${typeof value}`);
|
|
3016
|
+
}
|
|
3017
|
+
ser.serializeU256(value);
|
|
3018
|
+
break;
|
|
3019
|
+
case "bool":
|
|
3020
|
+
if (typeof value !== "boolean") {
|
|
3021
|
+
throw new Error(`Expected boolean, got ${typeof value}`);
|
|
3022
|
+
}
|
|
3023
|
+
ser.serializeBool(value);
|
|
3024
|
+
break;
|
|
3025
|
+
case "0x1::string::String":
|
|
3026
|
+
if (typeof value !== "string") {
|
|
3027
|
+
throw new Error(`Expected string, got ${typeof value}`);
|
|
3028
|
+
}
|
|
3029
|
+
ser.serializeStr(value);
|
|
3030
|
+
break;
|
|
3031
|
+
default:
|
|
3032
|
+
throw new Error(`Unsupported type: ${type}`);
|
|
3033
|
+
}
|
|
3034
|
+
return shouldReturnBytes ? ser.getBytes() : new Uint8Array(0);
|
|
3035
|
+
},
|
|
3036
|
+
[]
|
|
3037
|
+
);
|
|
3038
|
+
const serializeArgsFromTypes = useCallback4(
|
|
3039
|
+
(args, paramTypes) => {
|
|
3040
|
+
if (args.length !== paramTypes.length) {
|
|
3041
|
+
throw new Error(
|
|
3042
|
+
`Argument count mismatch: expected ${paramTypes.length}, got ${args.length}`
|
|
3043
|
+
);
|
|
3044
|
+
}
|
|
3045
|
+
return args.map((arg, index) => {
|
|
3046
|
+
try {
|
|
3047
|
+
const paramType = paramTypes[index].replace("&signer", "").trim();
|
|
3048
|
+
return serializeValueByType(arg, paramType);
|
|
3049
|
+
} catch (error) {
|
|
3050
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
3051
|
+
throw new Error(
|
|
3052
|
+
`Failed to serialize argument ${index} (${paramTypes[index]}): ${errorMessage}`
|
|
3053
|
+
);
|
|
3054
|
+
}
|
|
3055
|
+
});
|
|
3056
|
+
},
|
|
3057
|
+
[serializeValueByType]
|
|
3058
|
+
);
|
|
3059
|
+
const fetchModuleABI = useCallback4(
|
|
3060
|
+
async (moduleAddress, moduleName, rpcUrl) => {
|
|
3061
|
+
const storedABI = getStoredABI(moduleAddress, moduleName);
|
|
3062
|
+
if (storedABI) {
|
|
3063
|
+
return storedABI;
|
|
3064
|
+
}
|
|
3065
|
+
const baseUrl = rpcUrl || (process.env.NEXT_PUBLIC_SUPRA_CHAIN_ID === "8" ? "https://rpc-mainnet.supra.com" : "https://rpc-testnet.supra.com");
|
|
3066
|
+
const url = `${baseUrl}/rpc/v3/accounts/${moduleAddress}/modules/${moduleName}`;
|
|
3067
|
+
try {
|
|
3068
|
+
const response = await fetch(url);
|
|
3069
|
+
if (!response.ok) {
|
|
3070
|
+
throw new Error(`Failed to fetch ABI: ${response.statusText}`);
|
|
3071
|
+
}
|
|
3072
|
+
const data = await response.json();
|
|
3073
|
+
return data.abi;
|
|
3074
|
+
} catch (error) {
|
|
3075
|
+
console.error("Error fetching module ABI:", error);
|
|
3076
|
+
throw new Error(
|
|
3077
|
+
`Failed to fetch module ABI: ${error instanceof Error ? error.message : String(error)}`
|
|
3078
|
+
);
|
|
3079
|
+
}
|
|
3080
|
+
},
|
|
3081
|
+
[]
|
|
3082
|
+
);
|
|
3083
|
+
const getFunctionParamTypes = useCallback4(
|
|
3084
|
+
async (moduleAddress, moduleName, functionName, rpcUrl) => {
|
|
3085
|
+
const moduleABI = await fetchModuleABI(moduleAddress, moduleName, rpcUrl);
|
|
3086
|
+
if (!moduleABI.exposed_functions) {
|
|
3087
|
+
throw new Error("Invalid module ABI response");
|
|
3088
|
+
}
|
|
3089
|
+
const functionDef = moduleABI.exposed_functions.find(
|
|
3090
|
+
(func) => func.name === functionName
|
|
3091
|
+
);
|
|
3092
|
+
if (!functionDef) {
|
|
3093
|
+
throw new Error(`Function ${functionName} not found in module ${moduleName}`);
|
|
3094
|
+
}
|
|
3095
|
+
return functionDef.params.filter((param) => {
|
|
3096
|
+
const trimmed = param.trim();
|
|
3097
|
+
return trimmed !== "signer" && trimmed !== "&signer";
|
|
3098
|
+
});
|
|
3099
|
+
},
|
|
3100
|
+
[fetchModuleABI]
|
|
3101
|
+
);
|
|
3102
|
+
const serializeTransactionArgs = useCallback4(
|
|
3103
|
+
async (args, moduleAddress, moduleName, functionName, rpcUrl) => {
|
|
3104
|
+
const paramTypes = await getFunctionParamTypes(
|
|
3105
|
+
moduleAddress,
|
|
3106
|
+
moduleName,
|
|
3107
|
+
functionName,
|
|
3108
|
+
rpcUrl
|
|
3109
|
+
);
|
|
3110
|
+
return serializeArgsFromTypes(args, paramTypes);
|
|
3111
|
+
},
|
|
3112
|
+
[getFunctionParamTypes, serializeArgsFromTypes]
|
|
3113
|
+
);
|
|
3114
|
+
return {
|
|
3115
|
+
stringToUint8Array,
|
|
3116
|
+
addressToUint8Array,
|
|
3117
|
+
serializeString,
|
|
3118
|
+
serializeUint8,
|
|
3119
|
+
serializeUint16,
|
|
3120
|
+
serializeUint32,
|
|
3121
|
+
serializeUint64,
|
|
3122
|
+
serializeUint128,
|
|
3123
|
+
serializeU256,
|
|
3124
|
+
serializeBool,
|
|
3125
|
+
serializeVector,
|
|
3126
|
+
deserializeString,
|
|
3127
|
+
deserializeVector,
|
|
3128
|
+
hexToString,
|
|
3129
|
+
stringToHex,
|
|
3130
|
+
// ABI-based serialization
|
|
3131
|
+
serializeValueByType,
|
|
3132
|
+
serializeArgsFromTypes,
|
|
3133
|
+
fetchModuleABI,
|
|
3134
|
+
getFunctionParamTypes,
|
|
3135
|
+
serializeTransactionArgs
|
|
3136
|
+
// Main function to use
|
|
3137
|
+
};
|
|
3138
|
+
};
|
|
3139
|
+
var useConversionUtils_default = useConversionUtils;
|
|
3140
|
+
|
|
3141
|
+
// src/components/ui/button.tsx
|
|
3142
|
+
import * as React3 from "react";
|
|
3143
|
+
import { Slot as Slot2 } from "@radix-ui/react-slot";
|
|
3144
|
+
import { cva as cva2 } from "class-variance-authority";
|
|
3145
|
+
import { jsx as jsx4 } from "react/jsx-runtime";
|
|
3146
|
+
var buttonVariants2 = cva2(
|
|
3147
|
+
"inline-flex items-center justify-center whitespace-nowrap rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50",
|
|
3148
|
+
{
|
|
3149
|
+
variants: {
|
|
3150
|
+
variant: {
|
|
3151
|
+
default: "bg-primary text-primary-foreground hover:bg-primary/90",
|
|
3152
|
+
destructive: "bg-destructive text-destructive-foreground hover:bg-destructive/90",
|
|
3153
|
+
outline: "border border-input bg-background hover:bg-accent hover:text-accent-foreground",
|
|
3154
|
+
secondary: "bg-secondary text-secondary-foreground hover:bg-secondary/80",
|
|
3155
|
+
ghost: "hover:bg-accent hover:text-accent-foreground",
|
|
3156
|
+
link: "text-primary underline-offset-4 hover:underline"
|
|
3157
|
+
},
|
|
3158
|
+
size: {
|
|
3159
|
+
default: "h-10 px-4 py-2",
|
|
3160
|
+
sm: "h-9 rounded-md px-3",
|
|
3161
|
+
lg: "h-11 rounded-md px-8",
|
|
3162
|
+
icon: "h-10 w-10"
|
|
3163
|
+
}
|
|
3164
|
+
},
|
|
3165
|
+
defaultVariants: {
|
|
3166
|
+
variant: "default",
|
|
3167
|
+
size: "default"
|
|
3168
|
+
}
|
|
3169
|
+
}
|
|
3170
|
+
);
|
|
3171
|
+
var Button2 = React3.forwardRef(
|
|
3172
|
+
(_a, ref) => {
|
|
3173
|
+
var _b = _a, { className, variant, size, asChild = false } = _b, props = __objRest(_b, ["className", "variant", "size", "asChild"]);
|
|
3174
|
+
const Comp = asChild ? Slot2 : "button";
|
|
3175
|
+
return /* @__PURE__ */ jsx4(
|
|
3176
|
+
Comp,
|
|
3177
|
+
__spreadValues({
|
|
3178
|
+
className: cn2(buttonVariants2({ variant, size, className })),
|
|
3179
|
+
ref
|
|
3180
|
+
}, props)
|
|
3181
|
+
);
|
|
3182
|
+
}
|
|
3183
|
+
);
|
|
3184
|
+
Button2.displayName = "Button";
|
|
3185
|
+
|
|
3186
|
+
// src/components/ui/dialog.tsx
|
|
3187
|
+
import * as React4 from "react";
|
|
3188
|
+
import * as DialogPrimitive2 from "@radix-ui/react-dialog";
|
|
3189
|
+
import { X as X3 } from "lucide-react";
|
|
3190
|
+
import { jsx as jsx5, jsxs as jsxs3 } from "react/jsx-runtime";
|
|
3191
|
+
var Dialog2 = DialogPrimitive2.Root;
|
|
3192
|
+
var DialogTrigger = DialogPrimitive2.Trigger;
|
|
3193
|
+
var DialogPortal2 = DialogPrimitive2.Portal;
|
|
3194
|
+
var DialogClose = DialogPrimitive2.Close;
|
|
3195
|
+
var DialogOverlay2 = React4.forwardRef((_a, ref) => {
|
|
3196
|
+
var _b = _a, { className } = _b, props = __objRest(_b, ["className"]);
|
|
3197
|
+
return /* @__PURE__ */ jsx5(DialogPortal2, { children: /* @__PURE__ */ jsx5(
|
|
3198
|
+
DialogPrimitive2.Overlay,
|
|
3199
|
+
__spreadValues({
|
|
3200
|
+
ref,
|
|
3201
|
+
className: cn2(
|
|
3202
|
+
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0",
|
|
3203
|
+
className
|
|
3204
|
+
)
|
|
3205
|
+
}, props)
|
|
3206
|
+
) });
|
|
3207
|
+
});
|
|
3208
|
+
DialogOverlay2.displayName = DialogPrimitive2.Overlay.displayName;
|
|
3209
|
+
var DialogContent2 = React4.forwardRef((_a, ref) => {
|
|
3210
|
+
var _b = _a, { className, children } = _b, props = __objRest(_b, ["className", "children"]);
|
|
3211
|
+
return /* @__PURE__ */ jsxs3(DialogPortal2, { children: [
|
|
3212
|
+
/* @__PURE__ */ jsx5(DialogOverlay2, {}),
|
|
3213
|
+
/* @__PURE__ */ jsxs3(
|
|
3214
|
+
DialogPrimitive2.Content,
|
|
3215
|
+
__spreadProps(__spreadValues({
|
|
3216
|
+
ref,
|
|
3217
|
+
className: cn2(
|
|
3218
|
+
"fixed left-[50%] top-[50%] z-50 grid w-full max-w-lg translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg",
|
|
3219
|
+
className
|
|
3220
|
+
)
|
|
3221
|
+
}, props), {
|
|
3222
|
+
children: [
|
|
3223
|
+
children,
|
|
3224
|
+
/* @__PURE__ */ jsxs3(DialogPrimitive2.Close, { className: "absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground", children: [
|
|
3225
|
+
/* @__PURE__ */ jsx5(X3, { className: "h-4 w-4" }),
|
|
3226
|
+
/* @__PURE__ */ jsx5("span", { className: "sr-only", children: "Close" })
|
|
3227
|
+
] })
|
|
3228
|
+
]
|
|
3229
|
+
})
|
|
3230
|
+
)
|
|
3231
|
+
] });
|
|
3232
|
+
});
|
|
3233
|
+
DialogContent2.displayName = DialogPrimitive2.Content.displayName;
|
|
3234
|
+
var DialogHeader2 = (_a) => {
|
|
3235
|
+
var _b = _a, {
|
|
3236
|
+
className
|
|
3237
|
+
} = _b, props = __objRest(_b, [
|
|
3238
|
+
"className"
|
|
3239
|
+
]);
|
|
3240
|
+
return /* @__PURE__ */ jsx5(
|
|
3241
|
+
"div",
|
|
3242
|
+
__spreadValues({
|
|
3243
|
+
className: cn2(
|
|
3244
|
+
"flex flex-col space-y-1.5 text-center sm:text-left",
|
|
3245
|
+
className
|
|
3246
|
+
)
|
|
3247
|
+
}, props)
|
|
3248
|
+
);
|
|
3249
|
+
};
|
|
3250
|
+
DialogHeader2.displayName = "DialogHeader";
|
|
3251
|
+
var DialogFooter2 = (_a) => {
|
|
3252
|
+
var _b = _a, {
|
|
3253
|
+
className
|
|
3254
|
+
} = _b, props = __objRest(_b, [
|
|
3255
|
+
"className"
|
|
3256
|
+
]);
|
|
3257
|
+
return /* @__PURE__ */ jsx5(
|
|
3258
|
+
"div",
|
|
3259
|
+
__spreadValues({
|
|
3260
|
+
className: cn2(
|
|
3261
|
+
"flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2",
|
|
3262
|
+
className
|
|
3263
|
+
)
|
|
3264
|
+
}, props)
|
|
3265
|
+
);
|
|
3266
|
+
};
|
|
3267
|
+
DialogFooter2.displayName = "DialogFooter";
|
|
3268
|
+
var DialogTitle2 = React4.forwardRef((_a, ref) => {
|
|
3269
|
+
var _b = _a, { className } = _b, props = __objRest(_b, ["className"]);
|
|
3270
|
+
return /* @__PURE__ */ jsx5(
|
|
3271
|
+
DialogPrimitive2.Title,
|
|
3272
|
+
__spreadValues({
|
|
3273
|
+
ref,
|
|
3274
|
+
className: cn2(
|
|
3275
|
+
"text-lg font-semibold leading-none tracking-tight",
|
|
3276
|
+
className
|
|
3277
|
+
)
|
|
3278
|
+
}, props)
|
|
3279
|
+
);
|
|
3280
|
+
});
|
|
3281
|
+
DialogTitle2.displayName = DialogPrimitive2.Title.displayName;
|
|
3282
|
+
var DialogDescription2 = React4.forwardRef((_a, ref) => {
|
|
3283
|
+
var _b = _a, { className } = _b, props = __objRest(_b, ["className"]);
|
|
3284
|
+
return /* @__PURE__ */ jsx5(
|
|
3285
|
+
DialogPrimitive2.Description,
|
|
3286
|
+
__spreadValues({
|
|
3287
|
+
ref,
|
|
3288
|
+
className: cn2("text-sm text-muted-foreground", className)
|
|
3289
|
+
}, props)
|
|
3290
|
+
);
|
|
3291
|
+
});
|
|
3292
|
+
DialogDescription2.displayName = DialogPrimitive2.Description.displayName;
|
|
3293
|
+
|
|
3294
|
+
// src/components/ui/sonner.tsx
|
|
3295
|
+
import { Toaster as Sonner } from "sonner";
|
|
3296
|
+
import { jsx as jsx6 } from "react/jsx-runtime";
|
|
3297
|
+
var Toaster = (_a) => {
|
|
3298
|
+
var _b = _a, { theme = "system" } = _b, props = __objRest(_b, ["theme"]);
|
|
3299
|
+
return /* @__PURE__ */ jsx6(
|
|
3300
|
+
Sonner,
|
|
3301
|
+
__spreadValues({
|
|
3302
|
+
theme,
|
|
3303
|
+
className: "toaster group",
|
|
3304
|
+
toastOptions: {
|
|
3305
|
+
classNames: {
|
|
3306
|
+
toast: "group toast group-[.toaster]:bg-background group-[.toaster]:text-foreground group-[.toaster]:border-border group-[.toaster]:shadow-lg",
|
|
3307
|
+
description: "group-[.toast]:text-muted-foreground",
|
|
3308
|
+
actionButton: "group-[.toast]:bg-primary group-[.toast]:text-primary-foreground",
|
|
3309
|
+
cancelButton: "group-[.toast]:bg-muted group-[.toast]:text-muted-foreground"
|
|
3310
|
+
}
|
|
3311
|
+
}
|
|
3312
|
+
}, props)
|
|
3313
|
+
);
|
|
3314
|
+
};
|
|
3315
|
+
|
|
3316
|
+
// src/assets/walletIcons/index.ts
|
|
3317
|
+
var WALLET_ICONS = {
|
|
3318
|
+
starkey: Starkey_default,
|
|
3319
|
+
ribbit: Ribbit_default
|
|
3320
|
+
};
|
|
3321
|
+
export {
|
|
3322
|
+
Button2 as Button,
|
|
3323
|
+
ConnectWalletHandler,
|
|
3324
|
+
Dialog2 as Dialog,
|
|
3325
|
+
DialogClose,
|
|
3326
|
+
DialogContent2 as DialogContent,
|
|
3327
|
+
DialogDescription2 as DialogDescription,
|
|
3328
|
+
DialogFooter2 as DialogFooter,
|
|
3329
|
+
DialogHeader2 as DialogHeader,
|
|
3330
|
+
DialogOverlay2 as DialogOverlay,
|
|
3331
|
+
DialogPortal2 as DialogPortal,
|
|
3332
|
+
DialogTitle2 as DialogTitle,
|
|
3333
|
+
DialogTrigger,
|
|
3334
|
+
Toaster,
|
|
3335
|
+
WALLET_EVENTS2 as WALLET_EVENTS,
|
|
3336
|
+
WALLET_ICONS,
|
|
3337
|
+
buttonVariants2 as buttonVariants,
|
|
3338
|
+
cn2 as cn,
|
|
3339
|
+
getStoredABI,
|
|
3340
|
+
Ribbit_default as ribbitIcon,
|
|
3341
|
+
standardizeAddress,
|
|
3342
|
+
Starkey_default as starkeyIcon,
|
|
3343
|
+
useConversionUtils_default as useConversionUtils,
|
|
3344
|
+
useSupraMultiWallet_default2 as useSupraMultiWallet
|
|
3345
|
+
};
|
|
3346
|
+
//# sourceMappingURL=index.mjs.map
|