@paypercut/checkout-js 1.0.4 → 1.0.6

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 CHANGED
@@ -54,12 +54,12 @@ pnpm add @paypercut/checkout-js
54
54
 
55
55
  ```html
56
56
  <!-- jsDelivr (recommended with SRI) -->
57
- <script src="https://cdn.jsdelivr.net/npm/@paypercut/checkout-js@1.0.4/dist/paypercut-checkout.iife.min.js"
57
+ <script src="https://cdn.jsdelivr.net/npm/@paypercut/checkout-js@1.0.6/dist/paypercut-checkout.iife.min.js"
58
58
  integrity="sha384-..."
59
59
  crossorigin="anonymous"></script>
60
60
 
61
61
  <!-- or UNPKG -->
62
- <script src="https://unpkg.com/@paypercut/checkout-js@1.0.4/dist/paypercut-checkout.iife.min.js"></script>
62
+ <script src="https://unpkg.com/@paypercut/checkout-js@1.0.6/dist/paypercut-checkout.iife.min.js"></script>
63
63
  ```
64
64
 
65
65
  ---
@@ -203,7 +203,143 @@ Subscribe to events using the `on()` method:
203
203
  |-------|-------------|---------|
204
204
  | `loaded` | Checkout iframe has finished loading | `void` |
205
205
  | `success` | Payment completed successfully | `void` |
206
- | `error` | Payment failed or an error occurred | `void` |
206
+ | `error` | Terminal failure (tokenize, confirm, or 3DS) | `ApiErrorPayload` |
207
+ | `expired` | Checkout session expired | `void` |
208
+ | `threeds_started` | 3DS challenge flow started; SDK modal opened | `object` (flow context) |
209
+ | `threeds_complete` | 3DS challenge completed | `object` (auth result context) |
210
+ | `threeds_canceled` | 3DS challenge canceled by user | `object` (flow context) |
211
+ | `threeds_error` | 3DS challenge error | `ApiErrorPayload` (if provided) or `object` |
212
+
213
+ ### 3DS Events (enum usage recommended)
214
+
215
+ You can subscribe using string event names or the SdkEvent enum:
216
+
217
+ ```ts
218
+ import { PaypercutCheckout, SdkEvent } from '@paypercut/checkout-js';
219
+
220
+ const checkout = PaypercutCheckout({ id: 'CHK_12345', containerId: '#checkout' });
221
+
222
+ checkout.on(SdkEvent.ThreeDSStarted, (ctx) => {
223
+ // e.g., show your own UI indicator or log ctx
224
+ });
225
+
226
+ checkout.on(SdkEvent.ThreeDSComplete, (payload) => {
227
+ // 3DS completed successfully; Hosted Checkout will continue the flow
228
+ });
229
+
230
+ checkout.on(SdkEvent.ThreeDSCanceled, (payload) => {
231
+ // User canceled the 3DS challenge
232
+ });
233
+
234
+ checkout.on(SdkEvent.ThreeDSError, (err) => {
235
+ // Normalized ApiErrorPayload when available; fallback may be a raw object
236
+ console.error('3DS error:', err);
237
+ });
238
+ ```
239
+
240
+ Payload notes:
241
+ - `error`: the SDK forwards a normalized `ApiErrorPayload` when provided by Hosted Checkout.
242
+ - `threeds_error`: forwards `payload.error` when available; otherwise the raw message data.
243
+ - `threeds_*` non-error events: payload is forwarded as-is from Hosted Checkout (shape may evolve).
244
+
245
+
246
+ ### Core events (loaded, success, error, expired)
247
+
248
+ Using enums:
249
+
250
+ ```ts
251
+ import { PaypercutCheckout, SdkEvent } from '@paypercut/checkout-js';
252
+
253
+ const checkout = PaypercutCheckout({ id: 'CHK_12345', containerId: '#checkout' });
254
+
255
+ checkout.on(SdkEvent.Loaded, () => {
256
+ console.log('Checkout loaded');
257
+ });
258
+
259
+ checkout.on(SdkEvent.Success, () => {
260
+ console.log('Payment successful');
261
+ });
262
+
263
+ checkout.on(SdkEvent.Error, (err) => {
264
+ // err: ApiErrorPayload
265
+ console.error('Payment error:', err.code, err.message);
266
+ });
267
+
268
+ checkout.on(SdkEvent.Expired, () => {
269
+ console.warn('Checkout session expired');
270
+ });
271
+ ```
272
+
273
+ Or with string event names:
274
+
275
+ ```ts
276
+ checkout.on('loaded', () => {});
277
+ checkout.on('success', () => {});
278
+ checkout.on('error', (err) => {});
279
+ checkout.on('expired', () => {});
280
+ ```
281
+
282
+
283
+ ## Types
284
+
285
+ ### ApiErrorPayload (example shape)
286
+
287
+ Example only; all fields are optional and additional passthrough properties may be present.
288
+
289
+ ```ts
290
+ // Example shape for values received in `checkout.on('error', (err))`
291
+ type ApiErrorPayload = {
292
+ code?:
293
+ | 'threeds_error'
294
+ | 'threeds_authentication_failed'
295
+ | 'threeds_canceled'
296
+ | 'card_declined'
297
+ | 'authentication_failed'
298
+ | 'session_expired'
299
+ | string;
300
+ type?: 'card_error' | string;
301
+ message?: string;
302
+ status?: number; // HTTP-like status if available
303
+ decline_code?: string; // Optional decline reason
304
+ // ...other passthrough fields may be included by the server
305
+ };
306
+ ```
307
+
308
+ ### 3DS payloads (representative samples)
309
+
310
+ The SDK forwards whatever Hosted Checkout sends for 3DS events. Shapes may evolve.
311
+
312
+ ```ts
313
+ // threeds_started
314
+ {
315
+ type: 'THREEDS_START_FLOW',
316
+ checkoutId: 'CHK_12345',
317
+ // additional context as provided by Hosted Checkout (e.g., method, acsUrl)
318
+ }
319
+
320
+ // threeds_complete
321
+ {
322
+ type: 'THREEDS_COMPLETE',
323
+ checkoutId: 'CHK_12345',
324
+ // authentication result context (e.g., status: 'Y' | 'A' | 'C' | 'D')
325
+ }
326
+
327
+ // threeds_canceled
328
+ {
329
+ type: 'THREEDS_CANCELED',
330
+ checkoutId: 'CHK_12345',
331
+ }
332
+
333
+ // threeds_error (non-terminal)
334
+ // Note: terminal failures will also be emitted on `error` with ApiErrorPayload
335
+ {
336
+ type: 'THREEDS_ERROR',
337
+ checkoutId: 'CHK_12345',
338
+ error?: ApiErrorPayload,
339
+ }
340
+ ```
341
+
342
+ Tip: Prefer subscribing with the SdkEvent enum for stronger typing.
207
343
 
208
344
 
209
345
  ## Usage Examples
@@ -219,15 +355,15 @@ Subscribe to events using the `on()` method:
219
355
  <body>
220
356
  <div id="checkout"></div>
221
357
 
222
- <script src="https://cdn.jsdelivr.net/npm/@paypercut/checkout-js@1.0.4/dist/paypercut-checkout.iife.min.js"></script>
358
+ <script src="https://cdn.jsdelivr.net/npm/@paypercut/checkout-js@1.0.6/dist/paypercut-checkout.iife.min.js"></script>
223
359
  <script>
224
360
  const checkout = PaypercutCheckout({
225
361
  id: 'CHK_12345',
226
362
  containerId: '#checkout'
227
363
  });
228
364
 
229
- checkout.on('success', function(data) {
230
- alert('Payment successful! Transaction ID: ' + data.transactionId);
365
+ checkout.on('success', function() {
366
+ alert('Payment successful!');
231
367
  });
232
368
 
233
369
  checkout.on('error', function(error) {
@@ -250,8 +386,8 @@ const checkout = PaypercutCheckout({
250
386
  containerId: '#checkout'
251
387
  });
252
388
 
253
- checkout.on('success', (data) => {
254
- console.log('Payment successful:', data);
389
+ checkout.on('success', () => {
390
+ console.log('Payment successful');
255
391
  // Redirect to success page
256
392
  window.location.href = '/success';
257
393
  });
@@ -289,9 +425,9 @@ export function CheckoutComponent({ checkoutId }: { checkoutId: string }) {
289
425
  console.log('Checkout loaded');
290
426
  });
291
427
 
292
- checkout.on('success', (data) => {
428
+ checkout.on('success', () => {
293
429
  setStatus('success');
294
- console.log('Payment successful:', data);
430
+ console.log('Payment successful');
295
431
  });
296
432
 
297
433
  checkout.on('error', (error) => {
@@ -342,8 +478,8 @@ export function ModalCheckout({ checkoutId, onClose }: { checkoutId: string; onC
342
478
  setIsLoading(false);
343
479
  });
344
480
 
345
- checkout.on('success', (data) => {
346
- console.log('Payment successful:', data);
481
+ checkout.on('success', () => {
482
+ console.log('Payment successful');
347
483
  onClose();
348
484
  });
349
485
 
package/dist/index.cjs CHANGED
@@ -84,7 +84,7 @@ class Emitter {
84
84
  * Production configuration (for merchants)
85
85
  */
86
86
  const productionConfig = {
87
- version: "1.0.4",
87
+ version: "1.0.6",
88
88
  defaultCheckoutOrigin: 'https://buy.paypercut.io',
89
89
  allowedOrigins: [
90
90
  'https://buy.paypercut.io',
@@ -269,6 +269,20 @@ function validateUIMode(mode) {
269
269
  console.warn(`[PaypercutCheckout] Invalid ui_mode: "${mode}". Valid options: ${Object.values(exports.UIMode).join(', ')}`);
270
270
  return undefined;
271
271
  }
272
+ /**
273
+ * Preferred enum for SDK event names (use instead of hardcoded strings)
274
+ */
275
+ exports.SdkEvent = void 0;
276
+ (function (SdkEvent) {
277
+ SdkEvent["Loaded"] = "loaded";
278
+ SdkEvent["Success"] = "success";
279
+ SdkEvent["Error"] = "error";
280
+ SdkEvent["Expired"] = "expired";
281
+ SdkEvent["ThreeDSComplete"] = "threeds_complete";
282
+ SdkEvent["ThreeDSError"] = "threeds_error";
283
+ SdkEvent["ThreeDSCanceled"] = "threeds_canceled";
284
+ SdkEvent["ThreeDSStarted"] = "threeds_started";
285
+ })(exports.SdkEvent || (exports.SdkEvent = {}));
272
286
 
273
287
  /**
274
288
  * Internal implementation of CheckoutInstance
@@ -280,6 +294,9 @@ class CheckoutImpl {
280
294
  this.mounted = false;
281
295
  this.destroyed = false;
282
296
  this.iframe = null;
297
+ // 3DS Modal state
298
+ this.threeDSModal = null;
299
+ this.threeDSIframe = null;
283
300
  // Bind message handler
284
301
  this.messageHandler = this.onMessage.bind(this);
285
302
  window.addEventListener('message', this.messageHandler);
@@ -319,7 +336,6 @@ class CheckoutImpl {
319
336
  if (this.options.appearance?.preset) {
320
337
  url.searchParams.set('preset', this.options.appearance.preset);
321
338
  }
322
- console.log('this.options', this.options);
323
339
  if (this.options.form_only !== undefined) {
324
340
  url.searchParams.set('form_only', String(this.options.form_only));
325
341
  }
@@ -358,16 +374,39 @@ class CheckoutImpl {
358
374
  // Handle specific events
359
375
  switch (data.type) {
360
376
  case 'CHECKOUT_LOADED':
361
- this.emitter.emit('loaded');
377
+ this.emitter.emit(exports.SdkEvent.Loaded);
362
378
  break;
363
379
  case 'CHECKOUT_SUCCESS':
364
- this.emitter.emit('success');
380
+ this.emitter.emit(exports.SdkEvent.Success);
365
381
  break;
366
382
  case 'CHECKOUT_ERROR':
367
- this.emitter.emit('error');
383
+ // Forward error payload (if provided) to SDK consumers
384
+ this.emitter.emit(exports.SdkEvent.Error, (data && data.error) ?? data);
368
385
  break;
369
386
  case 'CHECKOUT_EXPIRED':
370
- this.emitter.emit('expired');
387
+ this.emitter.emit(exports.SdkEvent.Expired);
388
+ break;
389
+ case 'THREEDS_START_FLOW':
390
+ this.show3DSModal(data);
391
+ this.emitter.emit(exports.SdkEvent.ThreeDSStarted);
392
+ break;
393
+ case 'THREEDS_READY':
394
+ this.handle3DSReady();
395
+ break;
396
+ case 'THREEDS_COMPLETE':
397
+ this.postToIframe(data);
398
+ this.close3DSModal();
399
+ this.emitter.emit(exports.SdkEvent.ThreeDSComplete);
400
+ break;
401
+ case 'THREEDS_CANCELED':
402
+ this.postToIframe(data);
403
+ this.close3DSModal();
404
+ this.emitter.emit(exports.SdkEvent.ThreeDSCanceled);
405
+ break;
406
+ case 'THREEDS_ERROR':
407
+ this.postToIframe(data);
408
+ this.close3DSModal();
409
+ this.emitter.emit(exports.SdkEvent.ThreeDSError, data.error);
371
410
  break;
372
411
  }
373
412
  }
@@ -376,7 +415,6 @@ class CheckoutImpl {
376
415
  */
377
416
  render() {
378
417
  if (this.mounted) {
379
- console.warn('[PaypercutCheckout] Already mounted');
380
418
  return;
381
419
  }
382
420
  try {
@@ -387,6 +425,9 @@ class CheckoutImpl {
387
425
  this.iframe.src = this.buildSrc();
388
426
  this.iframe.allow = 'payment *; clipboard-write';
389
427
  this.iframe.setAttribute('frameborder', '0');
428
+ // Allow Payment Request API inside iframe and necessary sandbox permissions for wallets
429
+ this.iframe.setAttribute('allowpaymentrequest', 'true');
430
+ this.iframe.setAttribute('sandbox', 'allow-scripts allow-forms allow-same-origin allow-popups allow-popups-to-escape-sandbox');
390
431
  // Apply default styles - just construct URL and assign to iframe
391
432
  Object.assign(this.iframe.style, {
392
433
  width: '100%',
@@ -413,7 +454,6 @@ class CheckoutImpl {
413
454
  */
414
455
  submit() {
415
456
  if (!this.mounted) {
416
- console.warn('[PaypercutCheckout] Cannot submit: checkout not mounted');
417
457
  return;
418
458
  }
419
459
  try {
@@ -421,7 +461,6 @@ class CheckoutImpl {
421
461
  type: 'START_PROCESSING',
422
462
  checkoutId: this.options.id,
423
463
  });
424
- console.log('[PaypercutCheckout] Submit payment message sent');
425
464
  }
426
465
  catch (err) {
427
466
  console.error('[PaypercutCheckout] Failed to submit payment:', err);
@@ -429,7 +468,7 @@ class CheckoutImpl {
429
468
  }
430
469
  }
431
470
  /**
432
- * Send message to iframe - used for submit and other commands
471
+ * Send message to hosted/embedded checkout iframe - used for submit payment and communicating 3DS events
433
472
  */
434
473
  postToIframe(message) {
435
474
  if (!this.iframe?.contentWindow) {
@@ -439,6 +478,100 @@ class CheckoutImpl {
439
478
  const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);
440
479
  this.iframe.contentWindow.postMessage(message, checkoutOrigin);
441
480
  }
481
+ /**
482
+ * Show 3DS modal with challenge/decoupled flow
483
+ */
484
+ show3DSModal(data) {
485
+ // Create modal backdrop
486
+ this.threeDSModal = document.createElement('div');
487
+ this.threeDSModal.id = 'paypercut-3ds-modal';
488
+ Object.assign(this.threeDSModal.style, {
489
+ position: 'fixed',
490
+ top: '0',
491
+ left: '0',
492
+ width: '100%',
493
+ height: '100%',
494
+ // backgroundColor: 'rgba(0, 0, 0, 0.5)',
495
+ display: 'flex',
496
+ alignItems: 'center',
497
+ justifyContent: 'center',
498
+ zIndex: '999999',
499
+ });
500
+ // Create container for iframe + close button
501
+ const modalContainer = document.createElement('div');
502
+ Object.assign(modalContainer.style, {
503
+ position: 'relative',
504
+ display: 'flex',
505
+ flexDirection: 'column',
506
+ backgroundColor: '#ffffff',
507
+ borderRadius: '24px',
508
+ });
509
+ // Create iframe for 3DS page
510
+ this.threeDSIframe = document.createElement('iframe');
511
+ const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);
512
+ // Build URL: baseUrl/c/{checkoutId}/threeds with query params
513
+ const threeDSUrl = new URL(`/c/${this.options.id}/threeds`, checkoutOrigin);
514
+ threeDSUrl.searchParams.set('sessionId', data.sessionId);
515
+ threeDSUrl.searchParams.set('step', data.step);
516
+ if (data.challengeUrl) {
517
+ threeDSUrl.searchParams.set('challengeUrl', data.challengeUrl);
518
+ }
519
+ if (data.acsTransactionId) {
520
+ threeDSUrl.searchParams.set('acsTransactionId', data.acsTransactionId);
521
+ }
522
+ if (data.threeDSVersion) {
523
+ threeDSUrl.searchParams.set('threeDSVersion', data.threeDSVersion);
524
+ }
525
+ this.threeDSIframe.src = threeDSUrl.toString();
526
+ this.threeDSIframe.allow = 'payment *';
527
+ this.threeDSIframe.setAttribute('frameborder', '0');
528
+ // Responsive modal sizing (60% of viewport, min 400px, max 800px width / 600px height)
529
+ const viewportWidth = window.innerWidth;
530
+ const viewportHeight = window.innerHeight;
531
+ const modalWidth = Math.min(Math.max(viewportWidth * 0.6, 400), 800);
532
+ const modalHeight = Math.min(Math.max(viewportHeight * 0.6, 400), 600);
533
+ Object.assign(this.threeDSIframe.style, {
534
+ width: `${modalWidth}px`,
535
+ height: `${modalHeight}px`,
536
+ border: 'none',
537
+ borderRadius: '24px',
538
+ backgroundColor: 'white',
539
+ overflow: 'auto',
540
+ });
541
+ // Append iframe
542
+ modalContainer.appendChild(this.threeDSIframe);
543
+ // Append container to modal backdrop
544
+ this.threeDSModal.appendChild(modalContainer);
545
+ document.body.appendChild(this.threeDSModal);
546
+ // Store 3DS data for later use
547
+ this.pending3DSData = data;
548
+ }
549
+ /**
550
+ * Handle THREEDS_READY message - send THREEDS_INIT with challenge data
551
+ */
552
+ handle3DSReady() {
553
+ const data = this.pending3DSData;
554
+ if (!data || !this.threeDSIframe?.contentWindow) {
555
+ console.error('[PaypercutCheckout] No pending 3DS data or iframe not ready');
556
+ return;
557
+ }
558
+ const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);
559
+ this.threeDSIframe.contentWindow.postMessage({
560
+ type: 'THREEDS_INIT',
561
+ checkoutId: this.options.id,
562
+ }, checkoutOrigin);
563
+ }
564
+ /**
565
+ * Close 3DS modal and cleanup
566
+ */
567
+ close3DSModal() {
568
+ if (this.threeDSModal) {
569
+ this.threeDSModal.remove();
570
+ this.threeDSModal = null;
571
+ }
572
+ this.threeDSIframe = null;
573
+ delete this.pending3DSData;
574
+ }
442
575
  /**
443
576
  * Destroy the checkout instance and cleanup (idempotent)
444
577
  */
@@ -457,6 +590,8 @@ class CheckoutImpl {
457
590
  console.error('[PaypercutCheckout] Error removing iframe:', err);
458
591
  }
459
592
  }
593
+ // Cleanup 3DS modal if open
594
+ this.close3DSModal();
460
595
  // Only set mounted = false after successful DOM removal
461
596
  this.mounted = false;
462
597
  this.destroyed = true;
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","sources":["../src/utils/emitter.ts","../src/config.ts","../src/types/locales.ts","../src/types/checkout.ts","../src/index.ts"],"sourcesContent":["/**\n * Simple event emitter for handling checkout events\n * Supports subscribing, unsubscribing, and emitting events\n */\nexport class Emitter<T extends string = string> {\n private handlers = new Map<T, Set<Function>>();\n\n /**\n * Subscribe to an event\n * @param event - Event name to listen to\n * @param handler - Callback function to execute when event is emitted\n * @returns Unsubscribe function\n */\n on(event: T, handler: Function): () => void {\n if (!this.handlers.has(event)) {\n this.handlers.set(event, new Set());\n }\n this.handlers.get(event)!.add(handler);\n\n // Return unsubscribe function\n return () => this.off(event, handler);\n }\n\n /**\n * Subscribe to an event that auto-unsubscribes after first emission\n *\n * Common use case: waiting for 'loaded' event or handling first 'success'.\n * Without this helper, developers would need to manually unsubscribe inside\n * their handler, which is error-prone and leads to memory leaks if forgotten.\n *\n * @param event - Event name to listen to\n * @param handler - Callback function to execute when event is emitted\n * @returns Unsubscribe function\n */\n once(event: T, handler: Function): () => void {\n const wrappedHandler = (...args: any[]) => {\n handler(...args);\n this.off(event, wrappedHandler);\n };\n return this.on(event, wrappedHandler);\n }\n\n /**\n * Unsubscribe from an event\n * @param event - Event name to stop listening to\n * @param handler - Callback function to remove\n */\n off(event: T, handler: Function): void {\n this.handlers.get(event)?.delete(handler);\n }\n\n /**\n * Emit an event with optional arguments\n * @param event - Event name to emit\n * @param args - Arguments to pass to event handlers\n */\n emit(event: T, ...args: any[]): void {\n this.handlers.get(event)?.forEach(h => {\n try {\n h(...args);\n } catch (err) {\n console.error(`[Emitter] Error in ${event} handler:`, err);\n }\n });\n }\n\n /**\n * Clear all event handlers\n */\n clear(): void {\n this.handlers.clear();\n }\n}\n\n","/**\n * Build-time configuration\n * \n * This file is processed at build time with different values for:\n * - Production build (for merchants)\n * - Internal build (for team development)\n */\n\n/**\n * Build mode - replaced at build time\n */\ndeclare const __BUILD_MODE__: 'production' | 'internal';\n\n/**\n * SDK version - replaced at build time\n */\ndeclare const __VERSION__: string;\n\n/**\n * Configuration interface\n */\nexport interface BuildConfig {\n /** SDK version */\n version: string;\n \n /** Build mode */\n mode: 'production' | 'internal';\n \n /** Default checkout origin (production URL) */\n defaultCheckoutOrigin: string;\n \n /** Allowed origins for postMessage validation */\n allowedOrigins: string[];\n \n /** Whether hostedCheckoutUrl override is allowed */\n allowOriginOverride: boolean;\n}\n\n/**\n * Production configuration (for merchants)\n */\nconst productionConfig: BuildConfig = {\n version: __VERSION__,\n mode: 'production',\n defaultCheckoutOrigin: 'https://buy.paypercut.io',\n allowedOrigins: [\n 'https://buy.paypercut.io',\n ],\n allowOriginOverride: false\n};\n\n/**\n * Internal configuration (for team development)\n */\nconst internalConfig: BuildConfig = {\n version: __VERSION__,\n mode: 'internal',\n defaultCheckoutOrigin: 'https://buy.paypercut.io',\n allowedOrigins: [\n 'https://buy.paypercut.io',\n 'http://localhost:3000',\n 'http://localhost:3001',\n 'http://127.0.0.1:3000',\n 'http://127.0.0.1:3001'\n ],\n allowOriginOverride: true\n};\n\n/**\n * Active configuration (selected at build time)\n */\nexport const config: BuildConfig = __BUILD_MODE__ === 'internal' \n ? internalConfig \n : productionConfig;\n\n/**\n * Helper to check if origin is allowed\n */\nexport function isOriginAllowed(origin: string): boolean {\n return config.allowedOrigins.includes(origin);\n}\n\n/**\n * Get the checkout origin (with optional override for internal builds)\n */\nexport function getCheckoutOrigin(override?: string): string {\n // Only allow override in internal builds\n if (override && config.allowOriginOverride) {\n return override;\n }\n \n return config.defaultCheckoutOrigin;\n}\n\n","/**\n * Supported locales for Paypercut Checkout\n * \n * @remarks\n * Single source of truth for all supported locale codes.\n * \n * Supported languages:\n * - Bulgarian: 'bg', 'bg-BG'\n * - English: 'en', 'en-GB'\n * - Greek: 'el', 'el-GR'\n * - Romanian: 'ro', 'ro-RO'\n * - Croatian: 'hr', 'hr-HR'\n * - Polish: 'pl', 'pl-PL'\n * - Czech: 'cs', 'cs-CZ'\n * - Slovenian: 'sl', 'sl-SI'\n * - Slovak: 'sk', 'sk-SK'\n * \n * @example\n * ```typescript\n * const checkout = PaypercutCheckout({\n * locale: 'bg' // or 'bg-BG', 'en', 'en-GB', etc.\n * });\n * ```\n */\nexport const LOCALES = [\n 'bg', 'bg-BG',\n 'en', 'en-GB',\n 'el', 'el-GR',\n 'ro', 'ro-RO',\n 'hr', 'hr-HR',\n 'pl', 'pl-PL',\n 'cs', 'cs-CZ',\n 'sl', 'sl-SI',\n 'sk', 'sk-SK',\n] as const;\n\n/**\n * Locale type - union of all supported locale codes plus 'auto'\n * @example\n * ```typescript\n * const locale: Locale = 'bg';\n * const autoLocale: Locale = 'auto';\n * ```\n */\nexport type Locale = typeof LOCALES[number] | 'auto';\n\n/**\n * Fast runtime check using Set for O(1) lookup\n * @internal\n */\nconst LOCALE_SET: ReadonlySet<string> = new Set(LOCALES);\n\n/**\n * Normalize and validate locale\n * \n * @param locale - Locale code to normalize\n * @returns Normalized locale code or 'en' as fallback\n * \n * @remarks\n * - 'auto' or empty → 'en'\n * - Unsupported locale → 'en' with console warning\n * - Supported locale → original value\n * \n * @example\n * ```typescript\n * normalizeLocale('auto') // → 'en'\n * normalizeLocale('bg') // → 'bg'\n * normalizeLocale('de') // → 'en' (with warning)\n * ```\n */\nexport function normalizeLocale(locale: string | Locale): typeof LOCALES[number] | 'en' {\n if (!locale || locale === 'auto') return 'en';\n if (LOCALE_SET.has(locale)) return locale as typeof LOCALES[number];\n console.warn(`[PaypercutCheckout] Locale \"${locale}\" is not supported. Falling back to \"en\".`);\n return 'en';\n}\n\n","import { Locale } from './locales';\n\n/**\n * UI mode for checkout\n */\nexport enum UIMode {\n /** Custom UI mode - merchant provides their own submit button */\n CUSTOM = 'custom',\n /** Embedded mode - checkout embedded in merchant page */\n EMBEDDED = 'embedded',\n /** Hosted mode - full-page checkout experience */\n HOSTED = 'hosted',\n}\n\n/**\n * Type guard for UIMode\n */\nexport function isValidUIMode(value: string): value is UIMode {\n return Object.values(UIMode).includes(value as UIMode);\n}\n\n/**\n * Payment method types\n */\nexport enum PaymentMethod {\n /** Card payment (credit/debit) */\n CARD = 'card',\n}\n\n/**\n * Type guard for PaymentMethod\n */\nexport function isValidPaymentMethod(value: string): value is PaymentMethod {\n return Object.values(PaymentMethod).includes(value as PaymentMethod);\n}\n\n/**\n * Wallet options for digital wallets\n */\nexport enum WalletOption {\n /** Apple Pay */\n APPLE_PAY = 'apple_pay',\n /** Google Pay */\n GOOGLE_PAY = 'google_pay',\n}\n\n/**\n * Type guard for WalletOption\n */\nexport function isValidWalletOption(value: string): value is WalletOption {\n return Object.values(WalletOption).includes(value as WalletOption);\n}\n\n/**\n * Validate payment methods array\n */\nexport function validatePaymentMethods(methods: string[]): PaymentMethod[] {\n const validated: PaymentMethod[] = [];\n\n for (const method of methods) {\n if (isValidPaymentMethod(method)) {\n validated.push(method as PaymentMethod);\n } else {\n console.warn(`[PaypercutCheckout] Invalid payment method: \"${method}\". Skipping.`);\n }\n }\n\n // Default to card if no valid methods\n if (validated.length === 0) {\n console.warn('[PaypercutCheckout] No valid payment methods provided. Defaulting to \"card\".');\n validated.push(PaymentMethod.CARD);\n }\n\n return validated;\n}\n\n/**\n * Validate wallet options array\n */\nexport function validateWalletOptions(options: string[]): WalletOption[] {\n const validated: WalletOption[] = [];\n\n for (const option of options) {\n if (isValidWalletOption(option)) {\n validated.push(option as WalletOption);\n } else {\n console.warn(`[PaypercutCheckout] Invalid wallet option: \"${option}\". Skipping.`);\n }\n }\n\n return validated;\n}\n\n/**\n * Validate UI mode\n */\nexport function validateUIMode(mode: string | undefined): UIMode | undefined {\n if (!mode) {\n return undefined;\n }\n\n if (isValidUIMode(mode)) {\n return mode as UIMode;\n }\n\n console.warn(\n `[PaypercutCheckout] Invalid ui_mode: \"${mode}\". Valid options: ${Object.values(UIMode).join(', ')}`,\n );\n return undefined;\n}\n\n/**\n * Configuration options for PaypercutCheckout\n */\nexport interface PaypercutCheckoutOptions {\n /** Checkout session identifier (e.g., 'CHK_12345') */\n id: string;\n\n /** CSS selector or HTMLElement where iframe mounts (required) */\n containerId: string | HTMLElement;\n\n /**\n * Optional: Custom hosted checkout URL (only available in internal builds)\n * Production builds will ignore this option for security\n */\n hostedCheckoutUrl?: string;\n\n /**\n * Optional: Locale for checkout UI\n * @default 'en'\n */\n locale?: Locale | string;\n\n /**\n * Optional: UI mode for checkout\n */\n ui_mode?: UIMode | `${UIMode}`;\n\n /**\n * Optional: Payment methods to enable\n * @default ['card']\n */\n payment_methods?: (PaymentMethod | `${PaymentMethod}`)[];\n\n /**\n * Optional: Digital wallet options\n * Can include both or just one\n */\n wallet_options?: (WalletOption | `${WalletOption}`)[];\n\n /**\n * Optional: Show only the payment form without header/footer\n * @default false\n */\n form_only?: boolean;\n\n /**\n * Optional: Appearance options\n */\n appearance?: {\n preset?: 'inline' | 'modal';\n theme?: 'light' | 'dark' | 'none' | 'flat';\n variables: {};\n rules: {};\n labels: 'floating' | 'above' | 'none';\n };\n}\n\n/**\n * Event names that can be emitted by the checkout\n */\nexport type EventName =\n | 'loaded' // iframe finished loading\n | 'success' // payment successful\n | 'error' // payment or system error\n | 'expired'; // checkout expired or already paid\n\n/**\n * Checkout instance interface\n */\nexport interface CheckoutInstance {\n /** Mount and render the iframe into the container */\n render(): void;\n\n /** Submit payment - sends message to hosted checkout to confirm payment */\n submit(): void;\n\n /** Destroy instance and cleanup all listeners */\n destroy(): void;\n\n /** Subscribe to events. Returns unsubscribe function */\n on(event: EventName, handler: (...args: any[]) => void): () => void;\n\n /** Subscribe to event that auto-unsubscribes after first emission */\n once(event: EventName, handler: (...args: any[]) => void): () => void;\n\n /** Unsubscribe from events */\n off(event: EventName, handler: (...args: any[]) => void): void;\n\n /** Check if checkout is currently mounted */\n isMounted(): boolean;\n}\n\n/**\n * PaypercutCheckout static interface (callable and constructable)\n */\nexport interface PaypercutCheckoutStatic {\n /** Callable factory function */\n (options: PaypercutCheckoutOptions): CheckoutInstance;\n\n /** Constructor support */\n new (options: PaypercutCheckoutOptions): CheckoutInstance;\n\n /** SDK version */\n version: string;\n}\n","import { Emitter } from './utils/emitter';\nimport { config, isOriginAllowed, getCheckoutOrigin } from './config';\n\nimport {\n PaypercutCheckoutOptions,\n CheckoutInstance,\n PaypercutCheckoutStatic,\n EventName,\n validatePaymentMethods,\n validateWalletOptions,\n validateUIMode,\n} from './types';\nimport { normalizeLocale } from './types';\n\n// Re-export types and enums for consumers\nexport type { PaypercutCheckoutOptions, CheckoutInstance, PaypercutCheckoutStatic, EventName };\n\nexport { UIMode, PaymentMethod, WalletOption } from './types/checkout';\n\nexport type { Locale } from './types/locales';\nexport { LOCALES } from './types/locales';\n\n/**\n * Internal implementation of CheckoutInstance\n */\nclass CheckoutImpl implements CheckoutInstance {\n private emitter = new Emitter<EventName>();\n private mounted = false;\n private destroyed = false;\n private iframe: HTMLIFrameElement | null = null;\n private messageHandler: (evt: MessageEvent) => void;\n\n constructor(private options: PaypercutCheckoutOptions) {\n // Bind message handler\n this.messageHandler = this.onMessage.bind(this);\n window.addEventListener('message', this.messageHandler);\n }\n\n /**\n * Build the iframe source URL with query parameters\n */\n private buildSrc(): string {\n const baseUrl = getCheckoutOrigin(this.options.hostedCheckoutUrl);\n const url = new URL(`/c/${this.options.id}`, baseUrl);\n\n // Add locale parameter\n if (this.options.locale) {\n const normalizedLocale = normalizeLocale(this.options.locale);\n url.searchParams.set('locale', normalizedLocale);\n }\n\n // Add ui_mode parameter\n if (this.options.ui_mode) {\n const validatedUIMode = validateUIMode(this.options.ui_mode);\n if (validatedUIMode) {\n url.searchParams.set('ui_mode', validatedUIMode);\n }\n }\n\n // Add payment_methods parameters (repeated for each method)\n if (this.options.payment_methods && this.options.payment_methods.length > 0) {\n const validatedMethods = validatePaymentMethods(this.options.payment_methods as string[]);\n validatedMethods.forEach((method) => {\n url.searchParams.append('payment_methods', method);\n });\n }\n\n // Add wallet_options parameters (repeated for each wallet)\n if (this.options.wallet_options && this.options.wallet_options.length > 0) {\n const validatedWallets = validateWalletOptions(this.options.wallet_options as string[]);\n validatedWallets.forEach((wallet) => {\n url.searchParams.append('wallet_options', wallet);\n });\n }\n\n if (this.options.appearance?.preset) {\n url.searchParams.set('preset', this.options.appearance.preset);\n }\n\n console.log('this.options', this.options);\n\n if (this.options.form_only !== undefined) {\n url.searchParams.set('form_only', String(this.options.form_only));\n }\n\n return url.toString();\n }\n\n /**\n * Get the container element from selector or HTMLElement\n */\n private getContainer(): HTMLElement {\n const container =\n typeof this.options.containerId === 'string'\n ? document.querySelector(this.options.containerId)\n : this.options.containerId;\n\n if (!container) {\n throw new Error(`Container not found: ${this.options.containerId}`);\n }\n\n return container as HTMLElement;\n }\n\n /**\n * Handle incoming postMessage events from iframe\n */\n private onMessage(evt: MessageEvent): void {\n // Validate origin against allowed origins (build-time whitelist)\n if (!isOriginAllowed(evt.origin)) {\n return;\n }\n\n // Validate structure\n const data = evt.data;\n if (!data || typeof data !== 'object' || !('type' in data)) {\n return;\n }\n\n // Filter messages by checkout session ID to prevent cross-instance message handling\n // This ensures each checkout instance only processes its own messages\n if (data.checkoutId && data.checkoutId !== this.options.id) {\n return; // Message is for a different checkout instance\n }\n\n // Handle specific events\n switch (data.type) {\n case 'CHECKOUT_LOADED':\n this.emitter.emit('loaded');\n break;\n case 'CHECKOUT_SUCCESS':\n this.emitter.emit('success');\n break;\n case 'CHECKOUT_ERROR':\n this.emitter.emit('error');\n break;\n case 'CHECKOUT_EXPIRED':\n this.emitter.emit('expired');\n break;\n }\n }\n\n /**\n * Render the checkout iframe into the container\n */\n render(): void {\n if (this.mounted) {\n console.warn('[PaypercutCheckout] Already mounted');\n return;\n }\n\n try {\n const container = this.getContainer();\n\n // Create iframe\n this.iframe = document.createElement('iframe');\n this.iframe.id = 'paypercut-checkout-iframe';\n this.iframe.src = this.buildSrc();\n this.iframe.allow = 'payment *; clipboard-write';\n this.iframe.setAttribute('frameborder', '0');\n\n // Apply default styles - just construct URL and assign to iframe\n Object.assign(this.iframe.style, {\n width: '100%',\n height: '100%',\n border: 'none',\n display: 'block',\n });\n\n // Listen for iframe load event (fallback)\n // This ensures 'loaded' event is always emitted even if hosted checkout\n // doesn't send CHECKOUT_LOADED message\n /*this.iframe.addEventListener('load', () => {\n this.emitter.emit('loaded');\n });*/\n\n container.appendChild(this.iframe);\n this.mounted = true;\n } catch (err) {\n console.error('[PaypercutCheckout] Failed to render:', err);\n throw err;\n }\n }\n\n /**\n * Submit payment - sends message to hosted checkout to confirm payment\n */\n submit(): void {\n if (!this.mounted) {\n console.warn('[PaypercutCheckout] Cannot submit: checkout not mounted');\n return;\n }\n\n try {\n this.postToIframe({\n type: 'START_PROCESSING',\n checkoutId: this.options.id,\n });\n console.log('[PaypercutCheckout] Submit payment message sent');\n } catch (err) {\n console.error('[PaypercutCheckout] Failed to submit payment:', err);\n throw err;\n }\n }\n\n /**\n * Send message to iframe - used for submit and other commands\n */\n private postToIframe(message: any): void {\n if (!this.iframe?.contentWindow) {\n console.error('[PaypercutCheckout] Cannot post message: iframe not mounted');\n return;\n }\n\n const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);\n this.iframe.contentWindow.postMessage(message, checkoutOrigin);\n }\n\n /**\n * Destroy the checkout instance and cleanup (idempotent)\n */\n destroy(): void {\n // Early return if already destroyed\n if (this.destroyed) {\n return;\n }\n\n // Remove iframe from DOM\n if (this.iframe) {\n try {\n this.iframe.remove();\n this.iframe = null;\n } catch (err) {\n console.error('[PaypercutCheckout] Error removing iframe:', err);\n }\n }\n\n // Only set mounted = false after successful DOM removal\n this.mounted = false;\n this.destroyed = true;\n\n // Cleanup event listeners\n window.removeEventListener('message', this.messageHandler);\n this.emitter.clear();\n }\n\n /**\n * Subscribe to an event\n */\n on(event: EventName, handler: (...args: any[]) => void): () => void {\n return this.emitter.on(event, handler);\n }\n\n /**\n * Subscribe to event that auto-unsubscribes after first emission\n */\n once(event: EventName, handler: (...args: any[]) => void): () => void {\n return this.emitter.once(event, handler);\n }\n\n /**\n * Unsubscribe from an event\n */\n off(event: EventName, handler: (...args: any[]) => void): void {\n this.emitter.off(event, handler);\n }\n\n /**\n * Check if checkout is currently mounted\n */\n isMounted(): boolean {\n return this.mounted;\n }\n}\n\n/**\n * Factory function that works both as callable and constructable.\n * Always returns a new CheckoutImpl instance - no prototype manipulation needed.\n */\nconst PaypercutCheckout: PaypercutCheckoutStatic = function (\n this: unknown,\n options: PaypercutCheckoutOptions,\n): CheckoutInstance {\n // Always return a new instance, regardless of how it's called\n return new CheckoutImpl(options);\n} as unknown as PaypercutCheckoutStatic;\n\n// Add static version property\n(PaypercutCheckout as any).version = config.version;\n\n// Export as default and named\nexport default PaypercutCheckout;\nexport { PaypercutCheckout };\n"],"names":["UIMode","PaymentMethod","WalletOption"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","sources":["../src/utils/emitter.ts","../src/config.ts","../src/types/locales.ts","../src/types/checkout.ts","../src/index.ts"],"sourcesContent":["/**\n * Simple event emitter for handling checkout events\n * Supports subscribing, unsubscribing, and emitting events\n */\nexport class Emitter<T extends string = string> {\n private handlers = new Map<T, Set<Function>>();\n\n /**\n * Subscribe to an event\n * @param event - Event name to listen to\n * @param handler - Callback function to execute when event is emitted\n * @returns Unsubscribe function\n */\n on(event: T, handler: Function): () => void {\n if (!this.handlers.has(event)) {\n this.handlers.set(event, new Set());\n }\n this.handlers.get(event)!.add(handler);\n\n // Return unsubscribe function\n return () => this.off(event, handler);\n }\n\n /**\n * Subscribe to an event that auto-unsubscribes after first emission\n *\n * Common use case: waiting for 'loaded' event or handling first 'success'.\n * Without this helper, developers would need to manually unsubscribe inside\n * their handler, which is error-prone and leads to memory leaks if forgotten.\n *\n * @param event - Event name to listen to\n * @param handler - Callback function to execute when event is emitted\n * @returns Unsubscribe function\n */\n once(event: T, handler: Function): () => void {\n const wrappedHandler = (...args: any[]) => {\n handler(...args);\n this.off(event, wrappedHandler);\n };\n return this.on(event, wrappedHandler);\n }\n\n /**\n * Unsubscribe from an event\n * @param event - Event name to stop listening to\n * @param handler - Callback function to remove\n */\n off(event: T, handler: Function): void {\n this.handlers.get(event)?.delete(handler);\n }\n\n /**\n * Emit an event with optional arguments\n * @param event - Event name to emit\n * @param args - Arguments to pass to event handlers\n */\n emit(event: T, ...args: any[]): void {\n this.handlers.get(event)?.forEach(h => {\n try {\n h(...args);\n } catch (err) {\n console.error(`[Emitter] Error in ${event} handler:`, err);\n }\n });\n }\n\n /**\n * Clear all event handlers\n */\n clear(): void {\n this.handlers.clear();\n }\n}\n\n","/**\n * Build-time configuration\n * \n * This file is processed at build time with different values for:\n * - Production build (for merchants)\n * - Internal build (for team development)\n */\n\n/**\n * Build mode - replaced at build time\n */\ndeclare const __BUILD_MODE__: 'production' | 'internal';\n\n/**\n * SDK version - replaced at build time\n */\ndeclare const __VERSION__: string;\n\n/**\n * Configuration interface\n */\nexport interface BuildConfig {\n /** SDK version */\n version: string;\n \n /** Build mode */\n mode: 'production' | 'internal';\n \n /** Default checkout origin (production URL) */\n defaultCheckoutOrigin: string;\n \n /** Allowed origins for postMessage validation */\n allowedOrigins: string[];\n \n /** Whether hostedCheckoutUrl override is allowed */\n allowOriginOverride: boolean;\n}\n\n/**\n * Production configuration (for merchants)\n */\nconst productionConfig: BuildConfig = {\n version: __VERSION__,\n mode: 'production',\n defaultCheckoutOrigin: 'https://buy.paypercut.io',\n allowedOrigins: [\n 'https://buy.paypercut.io',\n ],\n allowOriginOverride: false\n};\n\n/**\n * Internal configuration (for team development)\n */\nconst internalConfig: BuildConfig = {\n version: __VERSION__,\n mode: 'internal',\n defaultCheckoutOrigin: 'https://buy.paypercut.io',\n allowedOrigins: [\n 'https://buy.paypercut.io',\n 'http://localhost:3000',\n 'http://localhost:3001',\n 'http://127.0.0.1:3000',\n 'http://127.0.0.1:3001'\n ],\n allowOriginOverride: true\n};\n\n/**\n * Active configuration (selected at build time)\n */\nexport const config: BuildConfig = __BUILD_MODE__ === 'internal' \n ? internalConfig \n : productionConfig;\n\n/**\n * Helper to check if origin is allowed\n */\nexport function isOriginAllowed(origin: string): boolean {\n return config.allowedOrigins.includes(origin);\n}\n\n/**\n * Get the checkout origin (with optional override for internal builds)\n */\nexport function getCheckoutOrigin(override?: string): string {\n // Only allow override in internal builds\n if (override && config.allowOriginOverride) {\n return override;\n }\n \n return config.defaultCheckoutOrigin;\n}\n\n","/**\n * Supported locales for Paypercut Checkout\n * \n * @remarks\n * Single source of truth for all supported locale codes.\n * \n * Supported languages:\n * - Bulgarian: 'bg', 'bg-BG'\n * - English: 'en', 'en-GB'\n * - Greek: 'el', 'el-GR'\n * - Romanian: 'ro', 'ro-RO'\n * - Croatian: 'hr', 'hr-HR'\n * - Polish: 'pl', 'pl-PL'\n * - Czech: 'cs', 'cs-CZ'\n * - Slovenian: 'sl', 'sl-SI'\n * - Slovak: 'sk', 'sk-SK'\n * \n * @example\n * ```typescript\n * const checkout = PaypercutCheckout({\n * locale: 'bg' // or 'bg-BG', 'en', 'en-GB', etc.\n * });\n * ```\n */\nexport const LOCALES = [\n 'bg', 'bg-BG',\n 'en', 'en-GB',\n 'el', 'el-GR',\n 'ro', 'ro-RO',\n 'hr', 'hr-HR',\n 'pl', 'pl-PL',\n 'cs', 'cs-CZ',\n 'sl', 'sl-SI',\n 'sk', 'sk-SK',\n] as const;\n\n/**\n * Locale type - union of all supported locale codes plus 'auto'\n * @example\n * ```typescript\n * const locale: Locale = 'bg';\n * const autoLocale: Locale = 'auto';\n * ```\n */\nexport type Locale = typeof LOCALES[number] | 'auto';\n\n/**\n * Fast runtime check using Set for O(1) lookup\n * @internal\n */\nconst LOCALE_SET: ReadonlySet<string> = new Set(LOCALES);\n\n/**\n * Normalize and validate locale\n * \n * @param locale - Locale code to normalize\n * @returns Normalized locale code or 'en' as fallback\n * \n * @remarks\n * - 'auto' or empty → 'en'\n * - Unsupported locale → 'en' with console warning\n * - Supported locale → original value\n * \n * @example\n * ```typescript\n * normalizeLocale('auto') // → 'en'\n * normalizeLocale('bg') // → 'bg'\n * normalizeLocale('de') // → 'en' (with warning)\n * ```\n */\nexport function normalizeLocale(locale: string | Locale): typeof LOCALES[number] | 'en' {\n if (!locale || locale === 'auto') return 'en';\n if (LOCALE_SET.has(locale)) return locale as typeof LOCALES[number];\n console.warn(`[PaypercutCheckout] Locale \"${locale}\" is not supported. Falling back to \"en\".`);\n return 'en';\n}\n\n","import { Locale } from './locales';\n\n/**\n * UI mode for checkout\n */\nexport enum UIMode {\n /** Custom UI mode - merchant provides their own submit button */\n CUSTOM = 'custom',\n /** Embedded mode - checkout embedded in merchant page */\n EMBEDDED = 'embedded',\n /** Hosted mode - full-page checkout experience */\n HOSTED = 'hosted',\n}\n\n/**\n * Type guard for UIMode\n */\nexport function isValidUIMode(value: string): value is UIMode {\n return Object.values(UIMode).includes(value as UIMode);\n}\n\n/**\n * Payment method types\n */\nexport enum PaymentMethod {\n /** Card payment (credit/debit) */\n CARD = 'card',\n}\n\n/**\n * Type guard for PaymentMethod\n */\nexport function isValidPaymentMethod(value: string): value is PaymentMethod {\n return Object.values(PaymentMethod).includes(value as PaymentMethod);\n}\n\n/**\n * Wallet options for digital wallets\n */\nexport enum WalletOption {\n /** Apple Pay */\n APPLE_PAY = 'apple_pay',\n /** Google Pay */\n GOOGLE_PAY = 'google_pay',\n}\n\n/**\n * Type guard for WalletOption\n */\nexport function isValidWalletOption(value: string): value is WalletOption {\n return Object.values(WalletOption).includes(value as WalletOption);\n}\n\n/**\n * Validate payment methods array\n */\nexport function validatePaymentMethods(methods: string[]): PaymentMethod[] {\n const validated: PaymentMethod[] = [];\n\n for (const method of methods) {\n if (isValidPaymentMethod(method)) {\n validated.push(method as PaymentMethod);\n } else {\n console.warn(`[PaypercutCheckout] Invalid payment method: \"${method}\". Skipping.`);\n }\n }\n\n // Default to card if no valid methods\n if (validated.length === 0) {\n console.warn('[PaypercutCheckout] No valid payment methods provided. Defaulting to \"card\".');\n validated.push(PaymentMethod.CARD);\n }\n\n return validated;\n}\n\n/**\n * Validate wallet options array\n */\nexport function validateWalletOptions(options: string[]): WalletOption[] {\n const validated: WalletOption[] = [];\n\n for (const option of options) {\n if (isValidWalletOption(option)) {\n validated.push(option as WalletOption);\n } else {\n console.warn(`[PaypercutCheckout] Invalid wallet option: \"${option}\". Skipping.`);\n }\n }\n\n return validated;\n}\n\n/**\n * Validate UI mode\n */\nexport function validateUIMode(mode: string | undefined): UIMode | undefined {\n if (!mode) {\n return undefined;\n }\n\n if (isValidUIMode(mode)) {\n return mode as UIMode;\n }\n\n console.warn(\n `[PaypercutCheckout] Invalid ui_mode: \"${mode}\". Valid options: ${Object.values(UIMode).join(', ')}`,\n );\n return undefined;\n}\n\n/**\n * Configuration options for PaypercutCheckout\n */\nexport interface PaypercutCheckoutOptions {\n /** Checkout session identifier (e.g., 'CHK_12345') */\n id: string;\n\n /** CSS selector or HTMLElement where iframe mounts (required) */\n containerId: string | HTMLElement;\n\n /**\n * Optional: Custom hosted checkout URL (only available in internal builds)\n * Production builds will ignore this option for security\n */\n hostedCheckoutUrl?: string;\n\n /**\n * Optional: Locale for checkout UI\n * @default 'en'\n */\n locale?: Locale | string;\n\n /**\n * Optional: UI mode for checkout\n */\n ui_mode?: UIMode | `${UIMode}`;\n\n /**\n * Optional: Payment methods to enable\n * @default ['card']\n */\n payment_methods?: (PaymentMethod | `${PaymentMethod}`)[];\n\n /**\n * Optional: Digital wallet options\n * Can include both or just one\n */\n wallet_options?: (WalletOption | `${WalletOption}`)[];\n\n /**\n * Optional: Show only the payment form without header/footer\n * @default false\n */\n form_only?: boolean;\n\n /**\n * Optional: Appearance options\n */\n appearance?: {\n preset?: 'inline' | 'modal';\n theme?: 'light' | 'dark' | 'none' | 'flat';\n variables: {};\n rules: {};\n labels: 'floating' | 'above' | 'none';\n };\n}\n\n/**\n * Event names that can be emitted by the checkout\n */\nexport type EventName =\n | 'loaded' // iframe finished loading\n | 'success' // payment successful\n | 'error' // payment or system error\n | 'expired' // checkout expired or already paid\n | 'threeds_complete' // checkout paid successfully\n | 'threeds_error' // checkout threeds error\n | 'threeds_canceled' // checkout threeds canceled\n | 'threeds_started'; // checkout threeds started\n\n/**\n * Preferred enum for SDK event names (use instead of hardcoded strings)\n */\nexport enum SdkEvent {\n Loaded = 'loaded',\n Success = 'success',\n Error = 'error',\n Expired = 'expired',\n ThreeDSComplete = 'threeds_complete',\n ThreeDSError = 'threeds_error',\n ThreeDSCanceled = 'threeds_canceled',\n ThreeDSStarted = 'threeds_started',\n}\n\n/**\n * Checkout instance interface\n */\nexport interface CheckoutInstance {\n /** Mount and render the iframe into the container */\n render(): void;\n\n /** Submit payment - sends message to hosted checkout to confirm payment */\n submit(): void;\n\n /** Destroy instance and cleanup all listeners */\n destroy(): void;\n\n /** Subscribe to events. Returns unsubscribe function */\n on(event: EventName | SdkEvent, handler: (...args: any[]) => void): () => void;\n\n /** Subscribe to event that auto-unsubscribes after first emission */\n once(event: EventName | SdkEvent, handler: (...args: any[]) => void): () => void;\n\n /** Unsubscribe from events */\n off(event: EventName | SdkEvent, handler: (...args: any[]) => void): void;\n\n /** Check if checkout is currently mounted */\n isMounted(): boolean;\n}\n\n/**\n * PaypercutCheckout static interface (callable and constructable)\n */\nexport interface PaypercutCheckoutStatic {\n /** Callable factory function */\n (options: PaypercutCheckoutOptions): CheckoutInstance;\n\n /** Constructor support */\n new (options: PaypercutCheckoutOptions): CheckoutInstance;\n\n /** SDK version */\n version: string;\n}\n","import { Emitter } from './utils/emitter';\nimport { config, isOriginAllowed, getCheckoutOrigin } from './config';\n\nimport {\n PaypercutCheckoutOptions,\n CheckoutInstance,\n PaypercutCheckoutStatic,\n EventName,\n SdkEvent,\n validatePaymentMethods,\n validateWalletOptions,\n validateUIMode,\n} from './types';\nimport { normalizeLocale } from './types';\n\n// Re-export types and enums for consumers\nexport type { PaypercutCheckoutOptions, CheckoutInstance, PaypercutCheckoutStatic, EventName };\n\nexport { UIMode, PaymentMethod, WalletOption } from './types/checkout';\n\n\n// Re-export SDK event enums for consumers\nexport { SdkEvent } from './types/checkout';\n\nexport type { Locale } from './types/locales';\nexport { LOCALES } from './types/locales';\n\n/**\n * Internal implementation of CheckoutInstance\n */\nclass CheckoutImpl implements CheckoutInstance {\n private emitter = new Emitter<EventName>();\n private mounted = false;\n private destroyed = false;\n private iframe: HTMLIFrameElement | null = null;\n private messageHandler: (evt: MessageEvent) => void;\n\n // 3DS Modal state\n private threeDSModal: HTMLDivElement | null = null;\n private threeDSIframe: HTMLIFrameElement | null = null;\n\n constructor(private options: PaypercutCheckoutOptions) {\n // Bind message handler\n this.messageHandler = this.onMessage.bind(this);\n window.addEventListener('message', this.messageHandler);\n }\n\n /**\n * Build the iframe source URL with query parameters\n */\n private buildSrc(): string {\n const baseUrl = getCheckoutOrigin(this.options.hostedCheckoutUrl);\n const url = new URL(`/c/${this.options.id}`, baseUrl);\n\n // Add locale parameter\n if (this.options.locale) {\n const normalizedLocale = normalizeLocale(this.options.locale);\n url.searchParams.set('locale', normalizedLocale);\n }\n\n // Add ui_mode parameter\n if (this.options.ui_mode) {\n const validatedUIMode = validateUIMode(this.options.ui_mode);\n if (validatedUIMode) {\n url.searchParams.set('ui_mode', validatedUIMode);\n }\n }\n\n // Add payment_methods parameters (repeated for each method)\n if (this.options.payment_methods && this.options.payment_methods.length > 0) {\n const validatedMethods = validatePaymentMethods(this.options.payment_methods as string[]);\n validatedMethods.forEach((method) => {\n url.searchParams.append('payment_methods', method);\n });\n }\n\n // Add wallet_options parameters (repeated for each wallet)\n if (this.options.wallet_options && this.options.wallet_options.length > 0) {\n const validatedWallets = validateWalletOptions(this.options.wallet_options as string[]);\n validatedWallets.forEach((wallet) => {\n url.searchParams.append('wallet_options', wallet);\n });\n }\n\n if (this.options.appearance?.preset) {\n url.searchParams.set('preset', this.options.appearance.preset);\n }\n\n if (this.options.form_only !== undefined) {\n url.searchParams.set('form_only', String(this.options.form_only));\n }\n\n return url.toString();\n }\n\n /**\n * Get the container element from selector or HTMLElement\n */\n private getContainer(): HTMLElement {\n const container =\n typeof this.options.containerId === 'string'\n ? document.querySelector(this.options.containerId)\n : this.options.containerId;\n\n if (!container) {\n throw new Error(`Container not found: ${this.options.containerId}`);\n }\n\n return container as HTMLElement;\n }\n\n /**\n * Handle incoming postMessage events from iframe\n */\n private onMessage(evt: MessageEvent): void {\n // Validate origin against allowed origins (build-time whitelist)\n if (!isOriginAllowed(evt.origin)) {\n return;\n }\n\n // Validate structure\n const data = evt.data;\n if (!data || typeof data !== 'object' || !('type' in data)) {\n return;\n }\n\n // Filter messages by checkout session ID to prevent cross-instance message handling\n // This ensures each checkout instance only processes its own messages\n if (data.checkoutId && data.checkoutId !== this.options.id) {\n return; // Message is for a different checkout instance\n }\n\n // Handle specific events\n switch (data.type) {\n case 'CHECKOUT_LOADED':\n this.emitter.emit(SdkEvent.Loaded);\n break;\n case 'CHECKOUT_SUCCESS':\n this.emitter.emit(SdkEvent.Success);\n break;\n case 'CHECKOUT_ERROR':\n // Forward error payload (if provided) to SDK consumers\n this.emitter.emit(SdkEvent.Error, (data && (data as any).error) ?? data);\n break;\n case 'CHECKOUT_EXPIRED':\n this.emitter.emit(SdkEvent.Expired);\n break;\n case 'THREEDS_START_FLOW':\n this.show3DSModal(data);\n this.emitter.emit(SdkEvent.ThreeDSStarted);\n break;\n case 'THREEDS_READY':\n this.handle3DSReady();\n break;\n case 'THREEDS_COMPLETE':\n this.postToIframe(data);\n this.close3DSModal();\n this.emitter.emit(SdkEvent.ThreeDSComplete);\n break;\n case 'THREEDS_CANCELED':\n this.postToIframe(data);\n this.close3DSModal();\n this.emitter.emit(SdkEvent.ThreeDSCanceled);\n break;\n case 'THREEDS_ERROR':\n this.postToIframe(data);\n this.close3DSModal();\n this.emitter.emit(SdkEvent.ThreeDSError, data.error);\n break;\n }\n }\n\n /**\n * Render the checkout iframe into the container\n */\n render(): void {\n if (this.mounted) {\n return;\n }\n\n try {\n const container = this.getContainer();\n\n // Create iframe\n this.iframe = document.createElement('iframe');\n this.iframe.id = 'paypercut-checkout-iframe';\n this.iframe.src = this.buildSrc();\n this.iframe.allow = 'payment *; clipboard-write';\n this.iframe.setAttribute('frameborder', '0');\n // Allow Payment Request API inside iframe and necessary sandbox permissions for wallets\n this.iframe.setAttribute('allowpaymentrequest', 'true');\n this.iframe.setAttribute(\n 'sandbox',\n 'allow-scripts allow-forms allow-same-origin allow-popups allow-popups-to-escape-sandbox',\n );\n\n\n // Apply default styles - just construct URL and assign to iframe\n Object.assign(this.iframe.style, {\n width: '100%',\n height: '100%',\n border: 'none',\n display: 'block',\n });\n\n // Listen for iframe load event (fallback)\n // This ensures 'loaded' event is always emitted even if hosted checkout\n // doesn't send CHECKOUT_LOADED message\n /*this.iframe.addEventListener('load', () => {\n this.emitter.emit('loaded');\n });*/\n\n container.appendChild(this.iframe);\n this.mounted = true;\n } catch (err) {\n console.error('[PaypercutCheckout] Failed to render:', err);\n throw err;\n }\n }\n\n /**\n * Submit payment - sends message to hosted checkout to confirm payment\n */\n submit(): void {\n if (!this.mounted) {\n return;\n }\n\n try {\n this.postToIframe({\n type: 'START_PROCESSING',\n checkoutId: this.options.id,\n });\n } catch (err) {\n console.error('[PaypercutCheckout] Failed to submit payment:', err);\n throw err;\n }\n }\n\n /**\n * Send message to hosted/embedded checkout iframe - used for submit payment and communicating 3DS events\n */\n private postToIframe(message: any): void {\n if (!this.iframe?.contentWindow) {\n console.error('[PaypercutCheckout] Cannot post message: iframe not mounted');\n return;\n }\n\n const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);\n this.iframe.contentWindow.postMessage(message, checkoutOrigin);\n }\n\n /**\n * Show 3DS modal with challenge/decoupled flow\n */\n private show3DSModal(data: {\n sessionId: string;\n step: 'challenge' | 'decoupled_waiting';\n challengeUrl?: string;\n acsTransactionId?: string;\n threeDSVersion?: string;\n cardBrand?: string;\n }): void {\n\n // Create modal backdrop\n this.threeDSModal = document.createElement('div');\n this.threeDSModal.id = 'paypercut-3ds-modal';\n Object.assign(this.threeDSModal.style, {\n position: 'fixed',\n top: '0',\n left: '0',\n width: '100%',\n height: '100%',\n // backgroundColor: 'rgba(0, 0, 0, 0.5)',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n zIndex: '999999',\n });\n\n // Create container for iframe + close button\n const modalContainer = document.createElement('div');\n Object.assign(modalContainer.style, {\n position: 'relative',\n display: 'flex',\n flexDirection: 'column',\n backgroundColor: '#ffffff',\n borderRadius: '24px',\n });\n\n // Create iframe for 3DS page\n this.threeDSIframe = document.createElement('iframe');\n const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);\n\n // Build URL: baseUrl/c/{checkoutId}/threeds with query params\n const threeDSUrl = new URL(`/c/${this.options.id}/threeds`, checkoutOrigin);\n threeDSUrl.searchParams.set('sessionId', data.sessionId);\n threeDSUrl.searchParams.set('step', data.step);\n\n if (data.challengeUrl) {\n threeDSUrl.searchParams.set('challengeUrl', data.challengeUrl);\n }\n if (data.acsTransactionId) {\n threeDSUrl.searchParams.set('acsTransactionId', data.acsTransactionId);\n }\n if (data.threeDSVersion) {\n threeDSUrl.searchParams.set('threeDSVersion', data.threeDSVersion);\n }\n\n this.threeDSIframe.src = threeDSUrl.toString();\n this.threeDSIframe.allow = 'payment *';\n this.threeDSIframe.setAttribute('frameborder', '0');\n\n // Responsive modal sizing (60% of viewport, min 400px, max 800px width / 600px height)\n const viewportWidth = window.innerWidth;\n const viewportHeight = window.innerHeight;\n const modalWidth = Math.min(Math.max(viewportWidth * 0.6, 400), 800);\n const modalHeight = Math.min(Math.max(viewportHeight * 0.6, 400), 600);\n\n Object.assign(this.threeDSIframe.style, {\n width: `${modalWidth}px`,\n height: `${modalHeight}px`,\n border: 'none',\n borderRadius: '24px',\n backgroundColor: 'white',\n overflow: 'auto',\n });\n\n // Append iframe\n modalContainer.appendChild(this.threeDSIframe);\n\n // Append container to modal backdrop\n this.threeDSModal.appendChild(modalContainer);\n document.body.appendChild(this.threeDSModal);\n\n // Store 3DS data for later use\n (this as any).pending3DSData = data;\n }\n\n /**\n * Handle THREEDS_READY message - send THREEDS_INIT with challenge data\n */\n private handle3DSReady(): void {\n const data = (this as any).pending3DSData;\n if (!data || !this.threeDSIframe?.contentWindow) {\n console.error('[PaypercutCheckout] No pending 3DS data or iframe not ready');\n return;\n }\n\n const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);\n this.threeDSIframe.contentWindow.postMessage(\n {\n type: 'THREEDS_INIT',\n checkoutId: this.options.id,\n },\n checkoutOrigin,\n );\n }\n\n /**\n * Close 3DS modal and cleanup\n */\n private close3DSModal(): void {\n if (this.threeDSModal) {\n this.threeDSModal.remove();\n this.threeDSModal = null;\n }\n\n this.threeDSIframe = null;\n delete (this as any).pending3DSData;\n }\n\n /**\n * Destroy the checkout instance and cleanup (idempotent)\n */\n destroy(): void {\n // Early return if already destroyed\n if (this.destroyed) {\n return;\n }\n\n // Remove iframe from DOM\n if (this.iframe) {\n try {\n this.iframe.remove();\n this.iframe = null;\n } catch (err) {\n console.error('[PaypercutCheckout] Error removing iframe:', err);\n }\n }\n\n // Cleanup 3DS modal if open\n this.close3DSModal();\n\n // Only set mounted = false after successful DOM removal\n this.mounted = false;\n this.destroyed = true;\n\n // Cleanup event listeners\n window.removeEventListener('message', this.messageHandler);\n this.emitter.clear();\n }\n\n /**\n * Subscribe to an event\n */\n on(event: EventName, handler: (...args: any[]) => void): () => void {\n return this.emitter.on(event, handler);\n }\n\n /**\n * Subscribe to event that auto-unsubscribes after first emission\n */\n once(event: EventName, handler: (...args: any[]) => void): () => void {\n return this.emitter.once(event, handler);\n }\n\n /**\n * Unsubscribe from an event\n */\n off(event: EventName, handler: (...args: any[]) => void): void {\n this.emitter.off(event, handler);\n }\n\n /**\n * Check if checkout is currently mounted\n */\n isMounted(): boolean {\n return this.mounted;\n }\n}\n\n/**\n * Factory function that works both as callable and constructable.\n * Always returns a new CheckoutImpl instance - no prototype manipulation needed.\n */\nconst PaypercutCheckout: PaypercutCheckoutStatic = function (\n this: unknown,\n options: PaypercutCheckoutOptions,\n): CheckoutInstance {\n // Always return a new instance, regardless of how it's called\n return new CheckoutImpl(options);\n} as unknown as PaypercutCheckoutStatic;\n\n// Add static version property\n(PaypercutCheckout as any).version = config.version;\n\n// Export as default and named\nexport default PaypercutCheckout;\nexport { PaypercutCheckout };\n"],"names":["UIMode","PaymentMethod","WalletOption","SdkEvent"],"mappings":"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"}