@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/dist/index.d.ts CHANGED
@@ -111,7 +111,20 @@ interface PaypercutCheckoutOptions {
111
111
  /**
112
112
  * Event names that can be emitted by the checkout
113
113
  */
114
- type EventName = 'loaded' | 'success' | 'error' | 'expired';
114
+ type EventName = 'loaded' | 'success' | 'error' | 'expired' | 'threeds_complete' | 'threeds_error' | 'threeds_canceled' | 'threeds_started';
115
+ /**
116
+ * Preferred enum for SDK event names (use instead of hardcoded strings)
117
+ */
118
+ declare enum SdkEvent {
119
+ Loaded = "loaded",
120
+ Success = "success",
121
+ Error = "error",
122
+ Expired = "expired",
123
+ ThreeDSComplete = "threeds_complete",
124
+ ThreeDSError = "threeds_error",
125
+ ThreeDSCanceled = "threeds_canceled",
126
+ ThreeDSStarted = "threeds_started"
127
+ }
115
128
  /**
116
129
  * Checkout instance interface
117
130
  */
@@ -123,11 +136,11 @@ interface CheckoutInstance {
123
136
  /** Destroy instance and cleanup all listeners */
124
137
  destroy(): void;
125
138
  /** Subscribe to events. Returns unsubscribe function */
126
- on(event: EventName, handler: (...args: any[]) => void): () => void;
139
+ on(event: EventName | SdkEvent, handler: (...args: any[]) => void): () => void;
127
140
  /** Subscribe to event that auto-unsubscribes after first emission */
128
- once(event: EventName, handler: (...args: any[]) => void): () => void;
141
+ once(event: EventName | SdkEvent, handler: (...args: any[]) => void): () => void;
129
142
  /** Unsubscribe from events */
130
- off(event: EventName, handler: (...args: any[]) => void): void;
143
+ off(event: EventName | SdkEvent, handler: (...args: any[]) => void): void;
131
144
  /** Check if checkout is currently mounted */
132
145
  isMounted(): boolean;
133
146
  }
@@ -149,5 +162,5 @@ interface PaypercutCheckoutStatic {
149
162
  */
150
163
  declare const PaypercutCheckout: PaypercutCheckoutStatic;
151
164
 
152
- export { LOCALES, PaymentMethod, PaypercutCheckout, UIMode, WalletOption, PaypercutCheckout as default };
165
+ export { LOCALES, PaymentMethod, PaypercutCheckout, SdkEvent, UIMode, WalletOption, PaypercutCheckout as default };
153
166
  export type { CheckoutInstance, EventName, Locale, PaypercutCheckoutOptions, PaypercutCheckoutStatic };
package/dist/index.mjs CHANGED
@@ -80,7 +80,7 @@ class Emitter {
80
80
  * Production configuration (for merchants)
81
81
  */
82
82
  const productionConfig = {
83
- version: "1.0.4",
83
+ version: "1.0.6",
84
84
  defaultCheckoutOrigin: 'https://buy.paypercut.io',
85
85
  allowedOrigins: [
86
86
  'https://buy.paypercut.io',
@@ -265,6 +265,20 @@ function validateUIMode(mode) {
265
265
  console.warn(`[PaypercutCheckout] Invalid ui_mode: "${mode}". Valid options: ${Object.values(UIMode).join(', ')}`);
266
266
  return undefined;
267
267
  }
268
+ /**
269
+ * Preferred enum for SDK event names (use instead of hardcoded strings)
270
+ */
271
+ var SdkEvent;
272
+ (function (SdkEvent) {
273
+ SdkEvent["Loaded"] = "loaded";
274
+ SdkEvent["Success"] = "success";
275
+ SdkEvent["Error"] = "error";
276
+ SdkEvent["Expired"] = "expired";
277
+ SdkEvent["ThreeDSComplete"] = "threeds_complete";
278
+ SdkEvent["ThreeDSError"] = "threeds_error";
279
+ SdkEvent["ThreeDSCanceled"] = "threeds_canceled";
280
+ SdkEvent["ThreeDSStarted"] = "threeds_started";
281
+ })(SdkEvent || (SdkEvent = {}));
268
282
 
269
283
  /**
270
284
  * Internal implementation of CheckoutInstance
@@ -276,6 +290,9 @@ class CheckoutImpl {
276
290
  this.mounted = false;
277
291
  this.destroyed = false;
278
292
  this.iframe = null;
293
+ // 3DS Modal state
294
+ this.threeDSModal = null;
295
+ this.threeDSIframe = null;
279
296
  // Bind message handler
280
297
  this.messageHandler = this.onMessage.bind(this);
281
298
  window.addEventListener('message', this.messageHandler);
@@ -315,7 +332,6 @@ class CheckoutImpl {
315
332
  if (this.options.appearance?.preset) {
316
333
  url.searchParams.set('preset', this.options.appearance.preset);
317
334
  }
318
- console.log('this.options', this.options);
319
335
  if (this.options.form_only !== undefined) {
320
336
  url.searchParams.set('form_only', String(this.options.form_only));
321
337
  }
@@ -354,16 +370,39 @@ class CheckoutImpl {
354
370
  // Handle specific events
355
371
  switch (data.type) {
356
372
  case 'CHECKOUT_LOADED':
357
- this.emitter.emit('loaded');
373
+ this.emitter.emit(SdkEvent.Loaded);
358
374
  break;
359
375
  case 'CHECKOUT_SUCCESS':
360
- this.emitter.emit('success');
376
+ this.emitter.emit(SdkEvent.Success);
361
377
  break;
362
378
  case 'CHECKOUT_ERROR':
363
- this.emitter.emit('error');
379
+ // Forward error payload (if provided) to SDK consumers
380
+ this.emitter.emit(SdkEvent.Error, (data && data.error) ?? data);
364
381
  break;
365
382
  case 'CHECKOUT_EXPIRED':
366
- this.emitter.emit('expired');
383
+ this.emitter.emit(SdkEvent.Expired);
384
+ break;
385
+ case 'THREEDS_START_FLOW':
386
+ this.show3DSModal(data);
387
+ this.emitter.emit(SdkEvent.ThreeDSStarted);
388
+ break;
389
+ case 'THREEDS_READY':
390
+ this.handle3DSReady();
391
+ break;
392
+ case 'THREEDS_COMPLETE':
393
+ this.postToIframe(data);
394
+ this.close3DSModal();
395
+ this.emitter.emit(SdkEvent.ThreeDSComplete);
396
+ break;
397
+ case 'THREEDS_CANCELED':
398
+ this.postToIframe(data);
399
+ this.close3DSModal();
400
+ this.emitter.emit(SdkEvent.ThreeDSCanceled);
401
+ break;
402
+ case 'THREEDS_ERROR':
403
+ this.postToIframe(data);
404
+ this.close3DSModal();
405
+ this.emitter.emit(SdkEvent.ThreeDSError, data.error);
367
406
  break;
368
407
  }
369
408
  }
@@ -372,7 +411,6 @@ class CheckoutImpl {
372
411
  */
373
412
  render() {
374
413
  if (this.mounted) {
375
- console.warn('[PaypercutCheckout] Already mounted');
376
414
  return;
377
415
  }
378
416
  try {
@@ -383,6 +421,9 @@ class CheckoutImpl {
383
421
  this.iframe.src = this.buildSrc();
384
422
  this.iframe.allow = 'payment *; clipboard-write';
385
423
  this.iframe.setAttribute('frameborder', '0');
424
+ // Allow Payment Request API inside iframe and necessary sandbox permissions for wallets
425
+ this.iframe.setAttribute('allowpaymentrequest', 'true');
426
+ this.iframe.setAttribute('sandbox', 'allow-scripts allow-forms allow-same-origin allow-popups allow-popups-to-escape-sandbox');
386
427
  // Apply default styles - just construct URL and assign to iframe
387
428
  Object.assign(this.iframe.style, {
388
429
  width: '100%',
@@ -409,7 +450,6 @@ class CheckoutImpl {
409
450
  */
410
451
  submit() {
411
452
  if (!this.mounted) {
412
- console.warn('[PaypercutCheckout] Cannot submit: checkout not mounted');
413
453
  return;
414
454
  }
415
455
  try {
@@ -417,7 +457,6 @@ class CheckoutImpl {
417
457
  type: 'START_PROCESSING',
418
458
  checkoutId: this.options.id,
419
459
  });
420
- console.log('[PaypercutCheckout] Submit payment message sent');
421
460
  }
422
461
  catch (err) {
423
462
  console.error('[PaypercutCheckout] Failed to submit payment:', err);
@@ -425,7 +464,7 @@ class CheckoutImpl {
425
464
  }
426
465
  }
427
466
  /**
428
- * Send message to iframe - used for submit and other commands
467
+ * Send message to hosted/embedded checkout iframe - used for submit payment and communicating 3DS events
429
468
  */
430
469
  postToIframe(message) {
431
470
  if (!this.iframe?.contentWindow) {
@@ -435,6 +474,100 @@ class CheckoutImpl {
435
474
  const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);
436
475
  this.iframe.contentWindow.postMessage(message, checkoutOrigin);
437
476
  }
477
+ /**
478
+ * Show 3DS modal with challenge/decoupled flow
479
+ */
480
+ show3DSModal(data) {
481
+ // Create modal backdrop
482
+ this.threeDSModal = document.createElement('div');
483
+ this.threeDSModal.id = 'paypercut-3ds-modal';
484
+ Object.assign(this.threeDSModal.style, {
485
+ position: 'fixed',
486
+ top: '0',
487
+ left: '0',
488
+ width: '100%',
489
+ height: '100%',
490
+ // backgroundColor: 'rgba(0, 0, 0, 0.5)',
491
+ display: 'flex',
492
+ alignItems: 'center',
493
+ justifyContent: 'center',
494
+ zIndex: '999999',
495
+ });
496
+ // Create container for iframe + close button
497
+ const modalContainer = document.createElement('div');
498
+ Object.assign(modalContainer.style, {
499
+ position: 'relative',
500
+ display: 'flex',
501
+ flexDirection: 'column',
502
+ backgroundColor: '#ffffff',
503
+ borderRadius: '24px',
504
+ });
505
+ // Create iframe for 3DS page
506
+ this.threeDSIframe = document.createElement('iframe');
507
+ const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);
508
+ // Build URL: baseUrl/c/{checkoutId}/threeds with query params
509
+ const threeDSUrl = new URL(`/c/${this.options.id}/threeds`, checkoutOrigin);
510
+ threeDSUrl.searchParams.set('sessionId', data.sessionId);
511
+ threeDSUrl.searchParams.set('step', data.step);
512
+ if (data.challengeUrl) {
513
+ threeDSUrl.searchParams.set('challengeUrl', data.challengeUrl);
514
+ }
515
+ if (data.acsTransactionId) {
516
+ threeDSUrl.searchParams.set('acsTransactionId', data.acsTransactionId);
517
+ }
518
+ if (data.threeDSVersion) {
519
+ threeDSUrl.searchParams.set('threeDSVersion', data.threeDSVersion);
520
+ }
521
+ this.threeDSIframe.src = threeDSUrl.toString();
522
+ this.threeDSIframe.allow = 'payment *';
523
+ this.threeDSIframe.setAttribute('frameborder', '0');
524
+ // Responsive modal sizing (60% of viewport, min 400px, max 800px width / 600px height)
525
+ const viewportWidth = window.innerWidth;
526
+ const viewportHeight = window.innerHeight;
527
+ const modalWidth = Math.min(Math.max(viewportWidth * 0.6, 400), 800);
528
+ const modalHeight = Math.min(Math.max(viewportHeight * 0.6, 400), 600);
529
+ Object.assign(this.threeDSIframe.style, {
530
+ width: `${modalWidth}px`,
531
+ height: `${modalHeight}px`,
532
+ border: 'none',
533
+ borderRadius: '24px',
534
+ backgroundColor: 'white',
535
+ overflow: 'auto',
536
+ });
537
+ // Append iframe
538
+ modalContainer.appendChild(this.threeDSIframe);
539
+ // Append container to modal backdrop
540
+ this.threeDSModal.appendChild(modalContainer);
541
+ document.body.appendChild(this.threeDSModal);
542
+ // Store 3DS data for later use
543
+ this.pending3DSData = data;
544
+ }
545
+ /**
546
+ * Handle THREEDS_READY message - send THREEDS_INIT with challenge data
547
+ */
548
+ handle3DSReady() {
549
+ const data = this.pending3DSData;
550
+ if (!data || !this.threeDSIframe?.contentWindow) {
551
+ console.error('[PaypercutCheckout] No pending 3DS data or iframe not ready');
552
+ return;
553
+ }
554
+ const checkoutOrigin = getCheckoutOrigin(this.options.hostedCheckoutUrl);
555
+ this.threeDSIframe.contentWindow.postMessage({
556
+ type: 'THREEDS_INIT',
557
+ checkoutId: this.options.id,
558
+ }, checkoutOrigin);
559
+ }
560
+ /**
561
+ * Close 3DS modal and cleanup
562
+ */
563
+ close3DSModal() {
564
+ if (this.threeDSModal) {
565
+ this.threeDSModal.remove();
566
+ this.threeDSModal = null;
567
+ }
568
+ this.threeDSIframe = null;
569
+ delete this.pending3DSData;
570
+ }
438
571
  /**
439
572
  * Destroy the checkout instance and cleanup (idempotent)
440
573
  */
@@ -453,6 +586,8 @@ class CheckoutImpl {
453
586
  console.error('[PaypercutCheckout] Error removing iframe:', err);
454
587
  }
455
588
  }
589
+ // Cleanup 3DS modal if open
590
+ this.close3DSModal();
456
591
  // Only set mounted = false after successful DOM removal
457
592
  this.mounted = false;
458
593
  this.destroyed = true;
@@ -496,5 +631,5 @@ const PaypercutCheckout = function (options) {
496
631
  // Add static version property
497
632
  PaypercutCheckout.version = config.version;
498
633
 
499
- export { LOCALES, PaymentMethod, PaypercutCheckout, UIMode, WalletOption, PaypercutCheckout as default };
634
+ export { LOCALES, PaymentMethod, PaypercutCheckout, SdkEvent, UIMode, WalletOption, PaypercutCheckout as default };
500
635
  //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","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":[],"mappings":"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"}
1
+ {"version":3,"file":"index.mjs","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":[],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- var PaypercutCheckout=function(){"use strict";class t{constructor(){this.handlers=new Map}on(t,e){return this.handlers.has(t)||this.handlers.set(t,new Set),this.handlers.get(t).add(e),()=>this.off(t,e)}once(t,e){const o=(...s)=>{e(...s),this.off(t,o)};return this.on(t,o)}off(t,e){this.handlers.get(t)?.delete(e)}emit(t,...e){this.handlers.get(t)?.forEach(o=>{try{o(...e)}catch(e){console.error(`[Emitter] Error in ${t} handler:`,e)}})}clear(){this.handlers.clear()}}const e={version:"1.0.4",defaultCheckoutOrigin:"https://buy.paypercut.io",allowedOrigins:["https://buy.paypercut.io"]};function o(t){return e.defaultCheckoutOrigin}const s=new Set(["bg","bg-BG","en","en-GB","el","el-GR","ro","ro-RO","hr","hr-HR","pl","pl-PL","cs","cs-CZ","sl","sl-SI","sk","sk-SK"]);var i,n,r;function a(t){return Object.values(n).includes(t)}function c(t){return Object.values(r).includes(t)}function h(t){var e;if(t)return e=t,Object.values(i).includes(e)?t:void console.warn(`[PaypercutCheckout] Invalid ui_mode: "${t}". Valid options: ${Object.values(i).join(", ")}`)}!function(t){t.CUSTOM="custom",t.EMBEDDED="embedded",t.HOSTED="hosted"}(i||(i={})),function(t){t.CARD="card"}(n||(n={})),function(t){t.APPLE_PAY="apple_pay",t.GOOGLE_PAY="google_pay"}(r||(r={}));class u{constructor(e){this.options=e,this.emitter=new t,this.mounted=!1,this.destroyed=!1,this.iframe=null,this.messageHandler=this.onMessage.bind(this),window.addEventListener("message",this.messageHandler)}buildSrc(){const t=o(this.options.hostedCheckoutUrl),e=new URL(`/c/${this.options.id}`,t);if(this.options.locale){const t=(i=this.options.locale)&&"auto"!==i?s.has(i)?i:(console.warn(`[PaypercutCheckout] Locale "${i}" is not supported. Falling back to "en".`),"en"):"en";e.searchParams.set("locale",t)}var i;if(this.options.ui_mode){const t=h(this.options.ui_mode);t&&e.searchParams.set("ui_mode",t)}if(this.options.payment_methods&&this.options.payment_methods.length>0){(function(t){const e=[];for(const o of t)a(o)?e.push(o):console.warn(`[PaypercutCheckout] Invalid payment method: "${o}". Skipping.`);return 0===e.length&&(console.warn('[PaypercutCheckout] No valid payment methods provided. Defaulting to "card".'),e.push(n.CARD)),e})(this.options.payment_methods).forEach(t=>{e.searchParams.append("payment_methods",t)})}if(this.options.wallet_options&&this.options.wallet_options.length>0){(function(t){const e=[];for(const o of t)c(o)?e.push(o):console.warn(`[PaypercutCheckout] Invalid wallet option: "${o}". Skipping.`);return e})(this.options.wallet_options).forEach(t=>{e.searchParams.append("wallet_options",t)})}return this.options.appearance?.preset&&e.searchParams.set("preset",this.options.appearance.preset),console.log("this.options",this.options),void 0!==this.options.form_only&&e.searchParams.set("form_only",String(this.options.form_only)),e.toString()}getContainer(){const t="string"==typeof this.options.containerId?document.querySelector(this.options.containerId):this.options.containerId;if(!t)throw new Error(`Container not found: ${this.options.containerId}`);return t}onMessage(t){if(o=t.origin,!e.allowedOrigins.includes(o))return;var o;const s=t.data;if(s&&"object"==typeof s&&"type"in s&&(!s.checkoutId||s.checkoutId===this.options.id))switch(s.type){case"CHECKOUT_LOADED":this.emitter.emit("loaded");break;case"CHECKOUT_SUCCESS":this.emitter.emit("success");break;case"CHECKOUT_ERROR":this.emitter.emit("error");break;case"CHECKOUT_EXPIRED":this.emitter.emit("expired")}}render(){if(this.mounted)console.warn("[PaypercutCheckout] Already mounted");else try{const t=this.getContainer();this.iframe=document.createElement("iframe"),this.iframe.id="paypercut-checkout-iframe",this.iframe.src=this.buildSrc(),this.iframe.allow="payment *; clipboard-write",this.iframe.setAttribute("frameborder","0"),Object.assign(this.iframe.style,{width:"100%",height:"100%",border:"none",display:"block"}),t.appendChild(this.iframe),this.mounted=!0}catch(t){throw console.error("[PaypercutCheckout] Failed to render:",t),t}}submit(){if(this.mounted)try{this.postToIframe({type:"START_PROCESSING",checkoutId:this.options.id}),console.log("[PaypercutCheckout] Submit payment message sent")}catch(t){throw console.error("[PaypercutCheckout] Failed to submit payment:",t),t}else console.warn("[PaypercutCheckout] Cannot submit: checkout not mounted")}postToIframe(t){if(!this.iframe?.contentWindow)return void console.error("[PaypercutCheckout] Cannot post message: iframe not mounted");const e=o(this.options.hostedCheckoutUrl);this.iframe.contentWindow.postMessage(t,e)}destroy(){if(!this.destroyed){if(this.iframe)try{this.iframe.remove(),this.iframe=null}catch(t){console.error("[PaypercutCheckout] Error removing iframe:",t)}this.mounted=!1,this.destroyed=!0,window.removeEventListener("message",this.messageHandler),this.emitter.clear()}}on(t,e){return this.emitter.on(t,e)}once(t,e){return this.emitter.once(t,e)}off(t,e){this.emitter.off(t,e)}isMounted(){return this.mounted}}const d=function(t){return new u(t)};return d.version=e.version,d}();
1
+ var PaypercutCheckout=function(){"use strict";class e{constructor(){this.handlers=new Map}on(e,t){return this.handlers.has(e)||this.handlers.set(e,new Set),this.handlers.get(e).add(t),()=>this.off(e,t)}once(e,t){const s=(...o)=>{t(...o),this.off(e,s)};return this.on(e,s)}off(e,t){this.handlers.get(e)?.delete(t)}emit(e,...t){this.handlers.get(e)?.forEach(s=>{try{s(...t)}catch(t){console.error(`[Emitter] Error in ${e} handler:`,t)}})}clear(){this.handlers.clear()}}const t={version:"1.0.6",defaultCheckoutOrigin:"https://buy.paypercut.io",allowedOrigins:["https://buy.paypercut.io"]};function s(e){return t.defaultCheckoutOrigin}const o=new Set(["bg","bg-BG","en","en-GB","el","el-GR","ro","ro-RO","hr","hr-HR","pl","pl-PL","cs","cs-CZ","sl","sl-SI","sk","sk-SK"]);var r,i,n,a;function h(e){return Object.values(i).includes(e)}function c(e){return Object.values(n).includes(e)}function d(e){var t;if(e)return t=e,Object.values(r).includes(t)?e:void console.warn(`[PaypercutCheckout] Invalid ui_mode: "${e}". Valid options: ${Object.values(r).join(", ")}`)}!function(e){e.CUSTOM="custom",e.EMBEDDED="embedded",e.HOSTED="hosted"}(r||(r={})),function(e){e.CARD="card"}(i||(i={})),function(e){e.APPLE_PAY="apple_pay",e.GOOGLE_PAY="google_pay"}(n||(n={})),function(e){e.Loaded="loaded",e.Success="success",e.Error="error",e.Expired="expired",e.ThreeDSComplete="threeds_complete",e.ThreeDSError="threeds_error",e.ThreeDSCanceled="threeds_canceled",e.ThreeDSStarted="threeds_started"}(a||(a={}));class l{constructor(t){this.options=t,this.emitter=new e,this.mounted=!1,this.destroyed=!1,this.iframe=null,this.threeDSModal=null,this.threeDSIframe=null,this.messageHandler=this.onMessage.bind(this),window.addEventListener("message",this.messageHandler)}buildSrc(){const e=s(this.options.hostedCheckoutUrl),t=new URL(`/c/${this.options.id}`,e);if(this.options.locale){const e=(r=this.options.locale)&&"auto"!==r?o.has(r)?r:(console.warn(`[PaypercutCheckout] Locale "${r}" is not supported. Falling back to "en".`),"en"):"en";t.searchParams.set("locale",e)}var r;if(this.options.ui_mode){const e=d(this.options.ui_mode);e&&t.searchParams.set("ui_mode",e)}if(this.options.payment_methods&&this.options.payment_methods.length>0){(function(e){const t=[];for(const s of e)h(s)?t.push(s):console.warn(`[PaypercutCheckout] Invalid payment method: "${s}". Skipping.`);return 0===t.length&&(console.warn('[PaypercutCheckout] No valid payment methods provided. Defaulting to "card".'),t.push(i.CARD)),t})(this.options.payment_methods).forEach(e=>{t.searchParams.append("payment_methods",e)})}if(this.options.wallet_options&&this.options.wallet_options.length>0){(function(e){const t=[];for(const s of e)c(s)?t.push(s):console.warn(`[PaypercutCheckout] Invalid wallet option: "${s}". Skipping.`);return t})(this.options.wallet_options).forEach(e=>{t.searchParams.append("wallet_options",e)})}return this.options.appearance?.preset&&t.searchParams.set("preset",this.options.appearance.preset),void 0!==this.options.form_only&&t.searchParams.set("form_only",String(this.options.form_only)),t.toString()}getContainer(){const e="string"==typeof this.options.containerId?document.querySelector(this.options.containerId):this.options.containerId;if(!e)throw new Error(`Container not found: ${this.options.containerId}`);return e}onMessage(e){if(s=e.origin,!t.allowedOrigins.includes(s))return;var s;const o=e.data;if(o&&"object"==typeof o&&"type"in o&&(!o.checkoutId||o.checkoutId===this.options.id))switch(o.type){case"CHECKOUT_LOADED":this.emitter.emit(a.Loaded);break;case"CHECKOUT_SUCCESS":this.emitter.emit(a.Success);break;case"CHECKOUT_ERROR":this.emitter.emit(a.Error,(o&&o.error)??o);break;case"CHECKOUT_EXPIRED":this.emitter.emit(a.Expired);break;case"THREEDS_START_FLOW":this.show3DSModal(o),this.emitter.emit(a.ThreeDSStarted);break;case"THREEDS_READY":this.handle3DSReady();break;case"THREEDS_COMPLETE":this.postToIframe(o),this.close3DSModal(),this.emitter.emit(a.ThreeDSComplete);break;case"THREEDS_CANCELED":this.postToIframe(o),this.close3DSModal(),this.emitter.emit(a.ThreeDSCanceled);break;case"THREEDS_ERROR":this.postToIframe(o),this.close3DSModal(),this.emitter.emit(a.ThreeDSError,o.error)}}render(){if(!this.mounted)try{const e=this.getContainer();this.iframe=document.createElement("iframe"),this.iframe.id="paypercut-checkout-iframe",this.iframe.src=this.buildSrc(),this.iframe.allow="payment *; clipboard-write",this.iframe.setAttribute("frameborder","0"),this.iframe.setAttribute("allowpaymentrequest","true"),this.iframe.setAttribute("sandbox","allow-scripts allow-forms allow-same-origin allow-popups allow-popups-to-escape-sandbox"),Object.assign(this.iframe.style,{width:"100%",height:"100%",border:"none",display:"block"}),e.appendChild(this.iframe),this.mounted=!0}catch(e){throw console.error("[PaypercutCheckout] Failed to render:",e),e}}submit(){if(this.mounted)try{this.postToIframe({type:"START_PROCESSING",checkoutId:this.options.id})}catch(e){throw console.error("[PaypercutCheckout] Failed to submit payment:",e),e}}postToIframe(e){if(!this.iframe?.contentWindow)return void console.error("[PaypercutCheckout] Cannot post message: iframe not mounted");const t=s(this.options.hostedCheckoutUrl);this.iframe.contentWindow.postMessage(e,t)}show3DSModal(e){this.threeDSModal=document.createElement("div"),this.threeDSModal.id="paypercut-3ds-modal",Object.assign(this.threeDSModal.style,{position:"fixed",top:"0",left:"0",width:"100%",height:"100%",display:"flex",alignItems:"center",justifyContent:"center",zIndex:"999999"});const t=document.createElement("div");Object.assign(t.style,{position:"relative",display:"flex",flexDirection:"column",backgroundColor:"#ffffff",borderRadius:"24px"}),this.threeDSIframe=document.createElement("iframe");const o=s(this.options.hostedCheckoutUrl),r=new URL(`/c/${this.options.id}/threeds`,o);r.searchParams.set("sessionId",e.sessionId),r.searchParams.set("step",e.step),e.challengeUrl&&r.searchParams.set("challengeUrl",e.challengeUrl),e.acsTransactionId&&r.searchParams.set("acsTransactionId",e.acsTransactionId),e.threeDSVersion&&r.searchParams.set("threeDSVersion",e.threeDSVersion),this.threeDSIframe.src=r.toString(),this.threeDSIframe.allow="payment *",this.threeDSIframe.setAttribute("frameborder","0");const i=window.innerWidth,n=window.innerHeight,a=Math.min(Math.max(.6*i,400),800),h=Math.min(Math.max(.6*n,400),600);Object.assign(this.threeDSIframe.style,{width:`${a}px`,height:`${h}px`,border:"none",borderRadius:"24px",backgroundColor:"white",overflow:"auto"}),t.appendChild(this.threeDSIframe),this.threeDSModal.appendChild(t),document.body.appendChild(this.threeDSModal),this.pending3DSData=e}handle3DSReady(){if(!this.pending3DSData||!this.threeDSIframe?.contentWindow)return void console.error("[PaypercutCheckout] No pending 3DS data or iframe not ready");const e=s(this.options.hostedCheckoutUrl);this.threeDSIframe.contentWindow.postMessage({type:"THREEDS_INIT",checkoutId:this.options.id},e)}close3DSModal(){this.threeDSModal&&(this.threeDSModal.remove(),this.threeDSModal=null),this.threeDSIframe=null,delete this.pending3DSData}destroy(){if(!this.destroyed){if(this.iframe)try{this.iframe.remove(),this.iframe=null}catch(e){console.error("[PaypercutCheckout] Error removing iframe:",e)}this.close3DSModal(),this.mounted=!1,this.destroyed=!0,window.removeEventListener("message",this.messageHandler),this.emitter.clear()}}on(e,t){return this.emitter.on(e,t)}once(e,t){return this.emitter.once(e,t)}off(e,t){this.emitter.off(e,t)}isMounted(){return this.mounted}}const p=function(e){return new l(e)};return p.version=t.version,p}();
2
2
  //# sourceMappingURL=paypercut-checkout.iife.min.js.map