@zucchinifi/dapp-sdk 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.mjs ADDED
@@ -0,0 +1,727 @@
1
+ // src/utils.ts
2
+ function toZats(amount) {
3
+ if (typeof amount === "number") {
4
+ amount = amount.toString();
5
+ }
6
+ amount = amount.replace(/,/g, "");
7
+ const parts = amount.split(".");
8
+ let whole = parts[0];
9
+ let fraction = parts[1] || "";
10
+ if (fraction.length > 8) {
11
+ throw new Error("Amount has more than 8 decimal places");
12
+ }
13
+ while (fraction.length < 8) {
14
+ fraction += "0";
15
+ }
16
+ const total = whole + fraction;
17
+ return BigInt(total);
18
+ }
19
+ function fromZats(amount) {
20
+ const zats = BigInt(amount).toString();
21
+ const padded = zats.padStart(9, "0");
22
+ const whole = padded.slice(0, -8);
23
+ const fraction = padded.slice(-8);
24
+ const cleanFraction = fraction.replace(/0+$/, "");
25
+ if (cleanFraction === "") {
26
+ return whole;
27
+ }
28
+ return `${whole}.${cleanFraction}`;
29
+ }
30
+ function shortenAddress(address, chars = 4) {
31
+ if (!address) return "";
32
+ if (address.length <= chars * 2 + 3) return address;
33
+ return `${address.slice(0, chars + 3)}...${address.slice(-chars)}`;
34
+ }
35
+
36
+ // src/zip321.ts
37
+ import { bech32 as bech322, bech32m as bech32m2 } from "bech32";
38
+ import bs58check2 from "bs58check";
39
+
40
+ // src/unified_address.ts
41
+ import { bech32m, bech32 } from "bech32";
42
+ import bs58check from "bs58check";
43
+
44
+ // src/f4jumble.ts
45
+ import { blake2bInit, blake2bUpdate, blake2bFinal } from "blakejs";
46
+ function get_h_pers(i) {
47
+ const p = new Uint8Array(16);
48
+ p.set([85, 65, 95, 70, 52, 74, 117, 109, 98, 108, 101, 95, 72, i, 0, 0]);
49
+ return p;
50
+ }
51
+ function get_g_pers(i, j) {
52
+ const p = new Uint8Array(16);
53
+ p.set([85, 65, 95, 70, 52, 74, 117, 109, 98, 108, 101, 95, 71, i, j & 255, j >> 8 & 255]);
54
+ return p;
55
+ }
56
+ function xor(target, source) {
57
+ const len = Math.min(target.length, source.length);
58
+ for (let i = 0; i < len; i++) {
59
+ target[i] ^= source[i];
60
+ }
61
+ }
62
+ var State = class {
63
+ constructor(message) {
64
+ const leftLen = Math.min(64, Math.floor(message.length / 2));
65
+ this.left = message.subarray(0, leftLen);
66
+ this.right = message.subarray(leftLen);
67
+ }
68
+ h_round(i) {
69
+ const ctx = blake2bInit(this.left.length, null, null, get_h_pers(i));
70
+ blake2bUpdate(ctx, this.right);
71
+ const hash = blake2bFinal(ctx);
72
+ xor(this.left, hash);
73
+ }
74
+ g_round(i) {
75
+ const outBytes = 64;
76
+ const numChunks = Math.ceil(this.right.length / outBytes);
77
+ for (let j = 0; j < numChunks; j++) {
78
+ const ctx = blake2bInit(outBytes, null, null, get_g_pers(i, j));
79
+ blake2bUpdate(ctx, this.left);
80
+ const hash = blake2bFinal(ctx);
81
+ const offset = j * outBytes;
82
+ const targetChunk = this.right.subarray(offset, offset + outBytes);
83
+ xor(targetChunk, hash);
84
+ }
85
+ }
86
+ apply_f4jumble() {
87
+ this.g_round(0);
88
+ this.h_round(0);
89
+ this.g_round(1);
90
+ this.h_round(1);
91
+ }
92
+ apply_f4jumble_inv() {
93
+ this.h_round(1);
94
+ this.g_round(1);
95
+ this.h_round(0);
96
+ this.g_round(0);
97
+ }
98
+ };
99
+ function f4jumble(message) {
100
+ const msgCopy = new Uint8Array(message);
101
+ const state = new State(msgCopy);
102
+ state.apply_f4jumble();
103
+ return msgCopy;
104
+ }
105
+ function f4jumble_inv(message) {
106
+ const msgCopy = new Uint8Array(message);
107
+ const state = new State(msgCopy);
108
+ state.apply_f4jumble_inv();
109
+ return msgCopy;
110
+ }
111
+
112
+ // src/unified_address.ts
113
+ var TYPE_P2PKH = 0;
114
+ var TYPE_P2SH = 1;
115
+ var TYPE_SAPLING = 2;
116
+ var TYPE_ORCHARD = 3;
117
+ function parseUnifiedAddress(unifiedAddress) {
118
+ if (!unifiedAddress.startsWith("u")) {
119
+ throw new Error("Not a Unified Address (must start with 'u')");
120
+ }
121
+ const { prefix: hrp, words: decodedWords } = bech32m.decode(unifiedAddress, 1e3);
122
+ if (hrp !== "u" && hrp !== "utest") {
123
+ throw new Error(`Invalid prefix: ${hrp} `);
124
+ }
125
+ const unjumbled = f4jumble_inv(new Uint8Array(bech32m.fromWords(decodedWords)));
126
+ if (!unjumbled) throw new Error("F4Jumble failed");
127
+ const padding = new Uint8Array(16);
128
+ const hrpBytes = new TextEncoder().encode(hrp);
129
+ padding.set(hrpBytes);
130
+ const suffix = unjumbled.slice(-16);
131
+ let isValidPadding = true;
132
+ for (let i = 0; i < 16; i++) {
133
+ if (suffix[i] !== padding[i]) {
134
+ isValidPadding = false;
135
+ break;
136
+ }
137
+ }
138
+ if (!isValidPadding) {
139
+ throw new Error("Invalid padding");
140
+ }
141
+ const data = unjumbled.slice(0, -16);
142
+ const receivers = [];
143
+ let offset = 0;
144
+ let lastTypeCode = -1;
145
+ while (offset < data.length) {
146
+ const { value: typeCode, bytes: typeCodeLength } = readCompactSize(data, offset);
147
+ offset += typeCodeLength;
148
+ const { value: length, bytes: lengthLength } = readCompactSize(data, offset);
149
+ offset += lengthLength;
150
+ if (offset + length > data.length) {
151
+ throw new Error("Invalid receiver length");
152
+ }
153
+ const receiverData = data.slice(offset, offset + length);
154
+ offset += length;
155
+ if (typeCode < lastTypeCode) {
156
+ throw new Error("Receivers not sorted by type code");
157
+ }
158
+ lastTypeCode = typeCode;
159
+ switch (typeCode) {
160
+ case TYPE_P2PKH:
161
+ if (length !== 20) throw new Error("Invalid P2PKH length");
162
+ const p2pkhPrefix = hrp === "u" ? [28, 184] : [29, 37];
163
+ receivers.push({ type: "p2pkh", typeCode, data: receiverData, address: bs58check.encode(Buffer.concat([Buffer.from(p2pkhPrefix), Buffer.from(receiverData)])) });
164
+ break;
165
+ case TYPE_P2SH:
166
+ if (length !== 20) throw new Error("Invalid P2SH length");
167
+ const p2shPrefix = hrp === "u" ? [28, 189] : [28, 186];
168
+ receivers.push({ type: "p2sh", typeCode, data: receiverData, address: bs58check.encode(Buffer.concat([Buffer.from(p2shPrefix), Buffer.from(receiverData)])) });
169
+ break;
170
+ case TYPE_SAPLING:
171
+ if (length !== 43) throw new Error("Invalid Sapling length");
172
+ const saplingHrp = hrp === "u" ? "zs" : "ztestsapling";
173
+ receivers.push({ type: "sapling", typeCode, data: receiverData, address: bech32.encode(saplingHrp, bech32.toWords(receiverData), 1e3) });
174
+ break;
175
+ case TYPE_ORCHARD:
176
+ if (length !== 43) throw new Error("Invalid Orchard length");
177
+ const tlv = new Uint8Array(1 + 1 + receiverData.length);
178
+ tlv[0] = 3;
179
+ tlv[1] = receiverData.length;
180
+ tlv.set(receiverData, 2);
181
+ const payloadWithPadding = new Uint8Array(tlv.length + 16);
182
+ payloadWithPadding.set(tlv);
183
+ payloadWithPadding.set(padding, tlv.length);
184
+ const jumbledOrchard = f4jumble(payloadWithPadding);
185
+ const orchardWords = bech32m.toWords(jumbledOrchard);
186
+ const orchardAddr = bech32m.encode(hrp, orchardWords, 1e3);
187
+ receivers.push({ type: "orchard", typeCode, data: receiverData, address: orchardAddr });
188
+ break;
189
+ default:
190
+ receivers.push({ type: "unknown", typeCode, data: receiverData, address: "" });
191
+ }
192
+ }
193
+ const result = {
194
+ unknown: []
195
+ };
196
+ for (const receiver of receivers) {
197
+ switch (receiver.type) {
198
+ case "p2pkh":
199
+ case "p2sh":
200
+ result.transparent = receiver.address;
201
+ break;
202
+ case "sapling":
203
+ result.sapling = receiver.address;
204
+ break;
205
+ case "orchard":
206
+ result.orchard = receiver.address;
207
+ break;
208
+ case "unknown":
209
+ result.unknown?.push({ typeCode: receiver.typeCode, data: receiver.data });
210
+ break;
211
+ }
212
+ }
213
+ return result;
214
+ }
215
+ function readCompactSize(data, offset) {
216
+ const first = data[offset];
217
+ if (first < 253) {
218
+ return { value: first, bytes: 1 };
219
+ } else if (first === 253) {
220
+ const val = data[offset + 1] | data[offset + 2] << 8;
221
+ return { value: val, bytes: 3 };
222
+ } else if (first === 254) {
223
+ const val = data[offset + 1] | data[offset + 2] << 8 | data[offset + 3] << 16 | data[offset + 4] << 24;
224
+ return { value: val, bytes: 5 };
225
+ } else {
226
+ throw new Error("CompactSize 8 bytes not supported");
227
+ }
228
+ }
229
+
230
+ // src/zip321.ts
231
+ var Zip321ParsingError = class extends Error {
232
+ constructor(message) {
233
+ super(message);
234
+ this.name = "Zip321ParsingError";
235
+ }
236
+ };
237
+ var Zip321Parser = class {
238
+ /**
239
+ * Parses a zcash: URI into a list of Payment objects.
240
+ * @param uri The zcash URI string.
241
+ * @param options Parsing options.
242
+ * @returns An array of Payment objects.
243
+ * @throws Zip321ParsingError if the URI is invalid.
244
+ */
245
+ static parse(uri, options = {}) {
246
+ if (!uri.startsWith("zcash:")) {
247
+ throw new Zip321ParsingError("Invalid scheme: must start with zcash:");
248
+ }
249
+ const parts = uri.substring(6).split("?");
250
+ const path = parts[0];
251
+ const query = parts[1];
252
+ if (parts.length > 2) {
253
+ throw new Zip321ParsingError("Invalid URI: multiple question marks");
254
+ }
255
+ const params = new URLSearchParams(query || "");
256
+ const payments = /* @__PURE__ */ new Map();
257
+ const getPayment = (index) => {
258
+ if (!payments.has(index)) {
259
+ payments.set(index, { address: "" });
260
+ }
261
+ return payments.get(index);
262
+ };
263
+ if (path && path.length > 0) {
264
+ if (!this.isValidAddress(path)) {
265
+ throw new Zip321ParsingError(`Invalid address in path: ${path}`);
266
+ }
267
+ const p = getPayment(0);
268
+ p.address = path;
269
+ }
270
+ params.forEach((value, key) => {
271
+ const match = key.match(/^([a-zA-Z][a-zA-Z0-9+-]*)(\.(\d+))?$/);
272
+ if (!match) {
273
+ if (key.startsWith("req-")) {
274
+ throw new Zip321ParsingError(`Unknown required parameter: ${key}`);
275
+ }
276
+ return;
277
+ }
278
+ const paramName = match[1];
279
+ const indexStr = match[3];
280
+ let index = 0;
281
+ if (indexStr) {
282
+ if (indexStr.startsWith("0")) {
283
+ throw new Zip321ParsingError(`Invalid index (leading zero): ${indexStr}`);
284
+ }
285
+ index = parseInt(indexStr, 10);
286
+ if (index > 9999) {
287
+ throw new Zip321ParsingError(`Index out of range: ${indexStr}`);
288
+ }
289
+ } else {
290
+ index = 0;
291
+ }
292
+ if (index === 0 && paramName === "address" && path.length > 0) {
293
+ throw new Zip321ParsingError("Duplicate address: defined in path and query");
294
+ }
295
+ const payment = getPayment(index);
296
+ switch (paramName) {
297
+ case "address":
298
+ if (payment.address) throw new Zip321ParsingError(`Duplicate address for index ${index}`);
299
+ if (!this.isValidAddress(value)) throw new Zip321ParsingError(`Invalid address: ${value}`);
300
+ payment.address = value;
301
+ break;
302
+ case "amount":
303
+ if (payment.amount !== void 0) throw new Zip321ParsingError(`Duplicate amount for index ${index}`);
304
+ if (payment.assetId) throw new Zip321ParsingError(`Amount and req-asset cannot both be present for index ${index}`);
305
+ payment.amount = this.parseAmount(value);
306
+ break;
307
+ case "req-asset":
308
+ if (payment.assetId) throw new Zip321ParsingError(`Duplicate req-asset for index ${index}`);
309
+ if (payment.amount !== void 0) throw new Zip321ParsingError(`Amount and req-asset cannot both be present for index ${index}`);
310
+ const assetData = this.parseAsset(value);
311
+ payment.assetId = assetData.assetId;
312
+ payment.amount = assetData.amount;
313
+ break;
314
+ case "memo":
315
+ if (payment.memo) throw new Zip321ParsingError(`Duplicate memo for index ${index}`);
316
+ if (!this.isValidBase64Url(value)) throw new Zip321ParsingError(`Invalid Base64URL memo: ${value}`);
317
+ payment.memo = value;
318
+ break;
319
+ case "label":
320
+ if (payment.label) throw new Zip321ParsingError(`Duplicate label for index ${index}`);
321
+ payment.label = value;
322
+ break;
323
+ case "message":
324
+ if (payment.message) throw new Zip321ParsingError(`Duplicate message for index ${index}`);
325
+ payment.message = value;
326
+ break;
327
+ default:
328
+ if (paramName.startsWith("req-")) {
329
+ throw new Zip321ParsingError(`Unknown required parameter: ${paramName}`);
330
+ }
331
+ if (!payment.otherParams) payment.otherParams = {};
332
+ payment.otherParams[paramName] = value;
333
+ break;
334
+ }
335
+ });
336
+ const result = [];
337
+ const sortedIndices = Array.from(payments.keys()).sort((a, b) => a - b);
338
+ for (const idx of sortedIndices) {
339
+ const p = payments.get(idx);
340
+ if (!p.address) {
341
+ throw new Zip321ParsingError(`Missing address for index ${idx}`);
342
+ }
343
+ if (this.isTransparent(p.address) && p.memo) {
344
+ throw new Zip321ParsingError(`Memo not allowed for transparent address at index ${idx}`);
345
+ }
346
+ if ((this.isTransparent(p.address) || this.isSapling(p.address)) && p.assetId) {
347
+ throw new Zip321ParsingError(`Custom assets not allowed for transparent/Sapling address at index ${idx}`);
348
+ }
349
+ result.push(p);
350
+ }
351
+ if (result.length === 0) {
352
+ throw new Zip321ParsingError("No payments found in URI");
353
+ }
354
+ if (options.useOrchard) {
355
+ for (const p of result) {
356
+ if (this.isUnified(p.address)) {
357
+ try {
358
+ const ua = parseUnifiedAddress(p.address);
359
+ if (ua.orchard) {
360
+ p.address = ua.orchard;
361
+ }
362
+ } catch (e) {
363
+ console.warn("Failed to extract Orchard receiver from UA:", e);
364
+ }
365
+ }
366
+ }
367
+ }
368
+ return result;
369
+ }
370
+ static isValidAddress(address) {
371
+ return this.isTransparent(address) || this.isSapling(address) || this.isUnified(address);
372
+ }
373
+ static isTransparent(address) {
374
+ try {
375
+ bs58check2.decode(address);
376
+ return address.startsWith("t");
377
+ } catch {
378
+ return false;
379
+ }
380
+ }
381
+ static isSapling(address) {
382
+ try {
383
+ const decoded = bech322.decode(address);
384
+ return decoded.prefix === "zs" || decoded.prefix === "ztestsapling";
385
+ } catch {
386
+ return false;
387
+ }
388
+ }
389
+ static isUnified(address) {
390
+ try {
391
+ const decoded = bech32m2.decode(address, 1e3);
392
+ return decoded.prefix === "u" || decoded.prefix === "utest";
393
+ } catch (e) {
394
+ return false;
395
+ }
396
+ }
397
+ static parseAmount(value) {
398
+ if (!/^\d+(\.\d{1,8})?$/.test(value)) {
399
+ throw new Zip321ParsingError(`Invalid amount format: ${value}`);
400
+ }
401
+ const parts = value.split(".");
402
+ let whole = BigInt(parts[0]);
403
+ let fraction = parts[1] || "";
404
+ fraction = fraction.padEnd(8, "0");
405
+ const zatoshis = whole * 100000000n + BigInt(fraction);
406
+ if (zatoshis > 2100000000000000n) {
407
+ throw new Zip321ParsingError(`Amount exceeds max supply: ${value}`);
408
+ }
409
+ return zatoshis;
410
+ }
411
+ static parseAsset(base64) {
412
+ if (!this.isValidBase64Url(base64)) {
413
+ throw new Zip321ParsingError(`Invalid Base64URL asset: ${base64}`);
414
+ }
415
+ const b64 = base64.replace(/-/g, "+").replace(/_/g, "/");
416
+ const pad = b64.length % 4;
417
+ const padded = pad ? b64 + "=".repeat(4 - pad) : b64;
418
+ let buffer;
419
+ try {
420
+ const binString = atob(padded);
421
+ buffer = new Uint8Array(binString.length);
422
+ for (let i = 0; i < binString.length; i++) {
423
+ buffer[i] = binString.charCodeAt(i);
424
+ }
425
+ } catch (e) {
426
+ throw new Zip321ParsingError(`Failed to decode asset base64: ${e}`);
427
+ }
428
+ if (buffer[0] !== 0) {
429
+ throw new Zip321ParsingError(`Unsupported asset version: ${buffer[0]}`);
430
+ }
431
+ const amountBytes = buffer.slice(1, 9);
432
+ let amount = 0n;
433
+ for (let i = 0; i < 8; i++) {
434
+ amount += BigInt(amountBytes[i]) << BigInt(i * 8);
435
+ }
436
+ const assetIdBytes = buffer.slice(9);
437
+ const assetId = Array.from(assetIdBytes).map((b) => b.toString(16).padStart(2, "0")).join("");
438
+ return { assetId, amount };
439
+ }
440
+ static isValidBase64Url(str) {
441
+ return /^[A-Za-z0-9\-_]*$/.test(str);
442
+ }
443
+ };
444
+ var Zip321Generator = class {
445
+ /**
446
+ * Creates a ZIP 321 compliant URI from a list of payments.
447
+ * @param payments List of Payment objects.
448
+ * @param options Generation options.
449
+ * @returns The formatted zcash: URI.
450
+ * @throws Error if payments list is empty or invalid.
451
+ */
452
+ static create(payments, options = {}) {
453
+ if (!payments || payments.length === 0) {
454
+ throw new Error("Payments list cannot be empty");
455
+ }
456
+ const processedPayments = payments.map((p) => {
457
+ if (options.useOrchard && Zip321Parser["isUnified"](p.address)) {
458
+ try {
459
+ const ua = parseUnifiedAddress(p.address);
460
+ if (ua.orchard) {
461
+ return { ...p, address: ua.orchard };
462
+ }
463
+ } catch (e) {
464
+ console.warn("Failed to extract Orchard receiver from UA:", e);
465
+ }
466
+ }
467
+ return p;
468
+ });
469
+ processedPayments.forEach((p, i) => {
470
+ if (!p.address) throw new Error(`Payment at index ${i} missing address`);
471
+ });
472
+ const parts = [];
473
+ const primary = processedPayments[0];
474
+ let uri = `zcash:${primary.address}`;
475
+ const addParam = (key, value, index) => {
476
+ if (value === void 0) return;
477
+ const paramKey = index !== void 0 && index > 0 ? `${key}.${index}` : key;
478
+ parts.push(`${paramKey}=${value}`);
479
+ };
480
+ if (primary.assetId) {
481
+ addParam("req-asset", this.formatAsset(primary.assetId, primary.amount || 0n));
482
+ } else if (primary.amount !== void 0) {
483
+ addParam("amount", this.formatAmount(primary.amount));
484
+ }
485
+ addParam("memo", primary.memo);
486
+ addParam("label", primary.label);
487
+ addParam("message", primary.message);
488
+ if (primary.otherParams) {
489
+ Object.entries(primary.otherParams).forEach(([k, v]) => addParam(k, v));
490
+ }
491
+ for (let i = 1; i < processedPayments.length; i++) {
492
+ const p = processedPayments[i];
493
+ addParam("address", p.address, i);
494
+ if (p.assetId) {
495
+ addParam("req-asset", this.formatAsset(p.assetId, p.amount || 0n), i);
496
+ } else if (p.amount !== void 0) {
497
+ addParam("amount", this.formatAmount(p.amount), i);
498
+ }
499
+ addParam("memo", p.memo, i);
500
+ addParam("label", p.label, i);
501
+ addParam("message", p.message, i);
502
+ if (p.otherParams) {
503
+ Object.entries(p.otherParams).forEach(([k, v]) => addParam(k, v, i));
504
+ }
505
+ }
506
+ if (parts.length > 0) {
507
+ uri += `?${parts.join("&")}`;
508
+ }
509
+ return uri;
510
+ }
511
+ static formatAmount(zatoshis) {
512
+ const whole = zatoshis / 100000000n;
513
+ const frac = zatoshis % 100000000n;
514
+ if (frac === 0n) {
515
+ return whole.toString();
516
+ }
517
+ let fracStr = frac.toString().padStart(8, "0");
518
+ fracStr = fracStr.replace(/0+$/, "");
519
+ return `${whole}.${fracStr}`;
520
+ }
521
+ static formatAsset(assetId, amount) {
522
+ const buffer = new Uint8Array(1 + 8 + 32);
523
+ buffer[0] = 0;
524
+ for (let i = 0; i < 8; i++) {
525
+ buffer[1 + i] = Number(amount >> BigInt(i * 8) & 0xffn);
526
+ }
527
+ for (let i = 0; i < 32; i++) {
528
+ buffer[9 + i] = parseInt(assetId.substr(i * 2, 2), 16);
529
+ }
530
+ let binary = "";
531
+ for (let i = 0; i < buffer.length; i++) {
532
+ binary += String.fromCharCode(buffer[i]);
533
+ }
534
+ const b64 = btoa(binary);
535
+ return b64.replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/, "");
536
+ }
537
+ };
538
+ function parsePaymentUri(uri, options = {}) {
539
+ return Zip321Parser.parse(uri, options);
540
+ }
541
+ function createPaymentUri(payments, options = {}) {
542
+ return Zip321Generator.create(payments, options);
543
+ }
544
+
545
+ // src/qrcode.ts
546
+ import QRCode from "qrcode";
547
+ var DEFAULT_LOGO = "";
548
+ async function generateQRCode(text, options = {}) {
549
+ const {
550
+ width = 300,
551
+ margin = 2,
552
+ color = { dark: "#000000", light: "#ffffff" },
553
+ logoUrl = DEFAULT_LOGO,
554
+ logoWidth = 60,
555
+ logoHeight = 60
556
+ } = options;
557
+ try {
558
+ const qrDataUrl = await QRCode.toDataURL(text, {
559
+ width,
560
+ margin,
561
+ color,
562
+ errorCorrectionLevel: "H"
563
+ // High error correction for logo overlay
564
+ });
565
+ if (!logoUrl) {
566
+ return qrDataUrl;
567
+ }
568
+ if (typeof document === "undefined") {
569
+ console.warn("Logo overlay requires browser environment (Canvas API). Returning plain QR code.");
570
+ return qrDataUrl;
571
+ }
572
+ return new Promise((resolve, reject) => {
573
+ const canvas = document.createElement("canvas");
574
+ canvas.width = width;
575
+ canvas.height = width;
576
+ const ctx = canvas.getContext("2d");
577
+ if (!ctx) {
578
+ reject(new Error("Failed to get canvas context"));
579
+ return;
580
+ }
581
+ const qrImage = new Image();
582
+ qrImage.onload = () => {
583
+ ctx.drawImage(qrImage, 0, 0);
584
+ const logo = new Image();
585
+ logo.crossOrigin = "Anonymous";
586
+ logo.onload = () => {
587
+ const x = (width - logoWidth) / 2;
588
+ const y = (width - logoHeight) / 2;
589
+ ctx.drawImage(logo, x, y, logoWidth, logoHeight);
590
+ resolve(canvas.toDataURL("image/png"));
591
+ };
592
+ logo.onerror = (err) => {
593
+ console.warn("Failed to load logo image", err);
594
+ resolve(qrDataUrl);
595
+ };
596
+ logo.src = logoUrl;
597
+ };
598
+ qrImage.onerror = (err) => reject(err);
599
+ qrImage.src = qrDataUrl;
600
+ });
601
+ } catch (error) {
602
+ throw new Error(`Failed to generate QR code: ${error}`);
603
+ }
604
+ }
605
+
606
+ // src/index.ts
607
+ var ZucchiniSDK = class {
608
+ /**
609
+ * Check if the Zucchini extension is installed and the provider is injected.
610
+ */
611
+ get isInstalled() {
612
+ return typeof window !== "undefined" && !!window.zucchini;
613
+ }
614
+ /**
615
+ * Get the provider instance.
616
+ * @throws Error if Zucchini is not installed.
617
+ */
618
+ get provider() {
619
+ if (!this.isInstalled) {
620
+ throw new Error("Zucchini Wallet is not installed");
621
+ }
622
+ return window.zucchini;
623
+ }
624
+ /**
625
+ * Connect to the wallet.
626
+ * Requests permission from the user to access their accounts.
627
+ * @param permissions List of permissions to request.
628
+ */
629
+ async connect(permissions = []) {
630
+ return this.request({ method: "connect", params: { permissions } });
631
+ }
632
+ /**
633
+ * Send a transaction.
634
+ * Requires 'send_transaction' permission.
635
+ */
636
+ async sendTransaction(args) {
637
+ return this.request({
638
+ method: "sendTransaction",
639
+ params: args
640
+ });
641
+ }
642
+ async getBalance() {
643
+ return this.request({
644
+ method: "getBalance"
645
+ });
646
+ }
647
+ async getAddresses() {
648
+ return this.request({
649
+ method: "getAddresses"
650
+ });
651
+ }
652
+ async getNetwork() {
653
+ return this.request({
654
+ method: "getNetwork"
655
+ });
656
+ }
657
+ /**
658
+ * Send a generic request to the wallet.
659
+ */
660
+ async request(args) {
661
+ return this.provider.request(args);
662
+ }
663
+ on(event, handler) {
664
+ this.provider.on(event, handler);
665
+ }
666
+ off(event, handler) {
667
+ this.provider.off(event, handler);
668
+ }
669
+ async disconnect() {
670
+ return this.request({ method: "disconnect" });
671
+ }
672
+ async isConnected() {
673
+ try {
674
+ return await this.request({ method: "isConnected" });
675
+ } catch (e) {
676
+ return false;
677
+ }
678
+ }
679
+ async getUniversalViewingKey() {
680
+ return this.request({ method: "getUniversalViewingKey" });
681
+ }
682
+ async getAllViewingKeys() {
683
+ return this.request({ method: "getAllViewingKeys" });
684
+ }
685
+ async getWalletStatus() {
686
+ return this.request({ method: "getWalletStatus" });
687
+ }
688
+ async getTokens() {
689
+ const API_BASE_URL = "https://api.zucchinifi.xyz";
690
+ const res = await fetch(`${API_BASE_URL}/tokens`);
691
+ if (!res.ok) throw new Error("Failed to fetch tokens");
692
+ return res.json();
693
+ }
694
+ async getSwapQuote(args) {
695
+ const API_BASE_URL = "https://api.zucchinifi.xyz";
696
+ const { from, to, amount, recipient, refundAddress } = args;
697
+ const params = new URLSearchParams({
698
+ from,
699
+ to,
700
+ amount,
701
+ recipient: recipient || "0x0000000000000000000000000000000000000000",
702
+ refundAddress: refundAddress || "0x0000000000000000000000000000000000000000",
703
+ dry: "true"
704
+ });
705
+ const res = await fetch(`${API_BASE_URL}/swap/quote?${params.toString()}`);
706
+ const data = await res.json();
707
+ if (data.error) {
708
+ throw new Error(data.error);
709
+ }
710
+ return data.quote;
711
+ }
712
+ };
713
+ var zucchini = new ZucchiniSDK();
714
+ export {
715
+ Zip321Generator,
716
+ Zip321Parser,
717
+ Zip321ParsingError,
718
+ ZucchiniSDK,
719
+ createPaymentUri,
720
+ fromZats,
721
+ generateQRCode,
722
+ parsePaymentUri,
723
+ parseUnifiedAddress,
724
+ shortenAddress,
725
+ toZats,
726
+ zucchini
727
+ };