@gengage/assistant-fe 0.2.2 → 0.2.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (94) hide show
  1. package/dist/assistant-fe.css +1 -1
  2. package/dist/chat/catalog.d.ts +2 -0
  3. package/dist/chat/catalog.d.ts.map +1 -1
  4. package/dist/chat/components/ChatDrawer.d.ts +47 -2
  5. package/dist/chat/components/ChatDrawer.d.ts.map +1 -1
  6. package/dist/chat/components/ChoicePrompter.d.ts +3 -1
  7. package/dist/chat/components/ChoicePrompter.d.ts.map +1 -1
  8. package/dist/chat/components/ComparisonTable.d.ts.map +1 -1
  9. package/dist/chat/components/PanelTopBar.d.ts +1 -0
  10. package/dist/chat/components/PanelTopBar.d.ts.map +1 -1
  11. package/dist/chat/components/ProductSummaryCard.d.ts.map +1 -1
  12. package/dist/chat/components/ReviewHighlights.d.ts.map +1 -1
  13. package/dist/chat/components/renderUISpec.d.ts.map +1 -1
  14. package/dist/chat/index.d.ts +22 -0
  15. package/dist/chat/index.d.ts.map +1 -1
  16. package/dist/chat/locales/en.d.ts.map +1 -1
  17. package/dist/chat/locales/tr.d.ts.map +1 -1
  18. package/dist/chat/panel-manager.d.ts +6 -4
  19. package/dist/chat/panel-manager.d.ts.map +1 -1
  20. package/dist/chat/session-persistence.d.ts +6 -1
  21. package/dist/chat/session-persistence.d.ts.map +1 -1
  22. package/dist/chat/types.d.ts +23 -1
  23. package/dist/chat/types.d.ts.map +1 -1
  24. package/dist/chat.cjs +1 -1
  25. package/dist/chat.iife.js +29 -29
  26. package/dist/chat.iife.js.map +1 -1
  27. package/dist/chat.js +2 -2
  28. package/dist/common/communication-bridge.d.ts.map +1 -1
  29. package/dist/common/price-formatter.d.ts +1 -1
  30. package/dist/common/price-formatter.d.ts.map +1 -1
  31. package/dist/common/product-utils.d.ts.map +1 -1
  32. package/dist/common/quantity-stepper.d.ts +1 -1
  33. package/dist/common/quantity-stepper.d.ts.map +1 -1
  34. package/dist/common/skeleton.d.ts +9 -0
  35. package/dist/common/skeleton.d.ts.map +1 -0
  36. package/dist/common/widget-base.d.ts.map +1 -1
  37. package/dist/common.cjs +1 -1
  38. package/dist/common.js +5 -5
  39. package/dist/index-BQVIyEOM.cjs +13 -0
  40. package/dist/index-BQVIyEOM.cjs.map +1 -0
  41. package/dist/index-Cdmhc_35.js +4783 -0
  42. package/dist/index-Cdmhc_35.js.map +1 -0
  43. package/dist/{index-CUukV8wH.cjs → index-DjdiNtXX.cjs} +2 -2
  44. package/dist/{index-CUukV8wH.cjs.map → index-DjdiNtXX.cjs.map} +1 -1
  45. package/dist/{index-CpLM7vPC.js → index-hUs4nTXp.js} +3 -3
  46. package/dist/{index-CpLM7vPC.js.map → index-hUs4nTXp.js.map} +1 -1
  47. package/dist/index.cjs +1 -1
  48. package/dist/index.js +3 -3
  49. package/dist/native.cjs +1 -1
  50. package/dist/native.iife.js +29 -29
  51. package/dist/native.iife.js.map +1 -1
  52. package/dist/native.js +1 -1
  53. package/dist/qna/index.d.ts.map +1 -1
  54. package/dist/qna.cjs +1 -1
  55. package/dist/qna.cjs.map +1 -1
  56. package/dist/qna.css +1 -1
  57. package/dist/qna.iife.js +13 -13
  58. package/dist/qna.iife.js.map +1 -1
  59. package/dist/qna.js +53 -44
  60. package/dist/qna.js.map +1 -1
  61. package/dist/quantity-stepper-CQWgexMO.cjs +2 -0
  62. package/dist/quantity-stepper-CQWgexMO.cjs.map +1 -0
  63. package/dist/quantity-stepper-CjY_cpgJ.js +80 -0
  64. package/dist/quantity-stepper-CjY_cpgJ.js.map +1 -0
  65. package/dist/{schemas-D1Kd4wn8.js → schemas-C7h8aelU.js} +2 -2
  66. package/dist/schemas-C7h8aelU.js.map +1 -0
  67. package/dist/{schemas-DOxyUYVA.cjs → schemas-DXvpS-Yg.cjs} +2 -2
  68. package/dist/schemas-DXvpS-Yg.cjs.map +1 -0
  69. package/dist/simrel/components/GroupTabs.d.ts.map +1 -1
  70. package/dist/simrel/components/ProductCard.d.ts.map +1 -1
  71. package/dist/simrel/components/ProductGrid.d.ts.map +1 -1
  72. package/dist/simrel/index.d.ts.map +1 -1
  73. package/dist/simrel/locales/en.d.ts.map +1 -1
  74. package/dist/simrel/locales/tr.d.ts.map +1 -1
  75. package/dist/simrel/types.d.ts +2 -0
  76. package/dist/simrel/types.d.ts.map +1 -1
  77. package/dist/simrel.cjs +1 -1
  78. package/dist/simrel.cjs.map +1 -1
  79. package/dist/simrel.css +1 -1
  80. package/dist/simrel.iife.js +11 -11
  81. package/dist/simrel.iife.js.map +1 -1
  82. package/dist/simrel.js +149 -148
  83. package/dist/simrel.js.map +1 -1
  84. package/package.json +2 -1
  85. package/dist/index-CB8DgwIv.cjs +0 -13
  86. package/dist/index-CB8DgwIv.cjs.map +0 -1
  87. package/dist/index-EEUjXgSt.js +0 -4468
  88. package/dist/index-EEUjXgSt.js.map +0 -1
  89. package/dist/quantity-stepper-BKtPQUR1.js +0 -78
  90. package/dist/quantity-stepper-BKtPQUR1.js.map +0 -1
  91. package/dist/quantity-stepper-DU6va4sS.cjs +0 -2
  92. package/dist/quantity-stepper-DU6va4sS.cjs.map +0 -1
  93. package/dist/schemas-D1Kd4wn8.js.map +0 -1
  94. package/dist/schemas-DOxyUYVA.cjs.map +0 -1
package/dist/qna.js CHANGED
@@ -1,4 +1,4 @@
1
- import { b as q, c as w, a as x, r as Q, o as g, s as p, u as L, d as R, _ as C, e as B, f as N, B as P, t as O, g as H, h as U, w as j, i as D, j as m, k as F, l as z, m as M, n as G } from "./schemas-D1Kd4wn8.js";
1
+ import { b as q, c as w, a as x, r as Q, o as _, s as p, u as L, d as R, _ as E, e as B, f as P, B as N, t as O, g as H, h as U, w as j, i as D, j as y, k as F, l as z, m as M, n as G } from "./schemas-C7h8aelU.js";
2
2
  async function W(n, e, t) {
3
3
  const i = q("launcher_action", e), o = { uiSpecs: [], actions: [] }, s = {
4
4
  method: "POST",
@@ -13,17 +13,17 @@ async function W(n, e, t) {
13
13
  onEvent: (c) => {
14
14
  const l = x(c);
15
15
  if (l && (l.type === "ui_spec" && o.uiSpecs.push(l.spec), l.type === "ui_spec" && l.spec.elements)) {
16
- for (const d of Object.values(l.spec.elements))
17
- if (d.type === "ActionButton" && d.props?.action) {
18
- const h = d.props.action;
19
- o.actions.push(h);
16
+ for (const h of Object.values(l.spec.elements))
17
+ if (h.type === "ActionButton" && h.props?.action) {
18
+ const d = h.props.action;
19
+ o.actions.push(d);
20
20
  }
21
21
  }
22
22
  }
23
23
  };
24
24
  return t !== void 0 && (r.signal = t), await w(a, r), o;
25
25
  }
26
- function y(n) {
26
+ function A(n) {
27
27
  const e = document.createElement("div");
28
28
  e.className = "gengage-qna-buttons", e.setAttribute("role", "group"), e.setAttribute("aria-label", n.quickQuestionsAriaLabel ?? "Quick questions"), n.orientation === "vertical" && (e.style.flexDirection = "column");
29
29
  for (const t of n.actions) {
@@ -75,7 +75,7 @@ function $(n) {
75
75
  s && clearInterval(s), a && clearTimeout(a);
76
76
  }, e;
77
77
  }
78
- function _(n, e) {
78
+ function m(n, e) {
79
79
  if (!n || typeof n != "object") return null;
80
80
  const t = n, i = t.type;
81
81
  if (typeof i != "string" || i.length === 0) return null;
@@ -84,62 +84,62 @@ function _(n, e) {
84
84
  const r = { title: a, type: i };
85
85
  return s !== void 0 && (r.payload = s), r;
86
86
  }
87
- function A(n) {
87
+ function T(n) {
88
88
  const e = {
89
89
  title: n.title,
90
90
  type: n.type
91
91
  };
92
92
  return n.payload !== void 0 && (e.payload = n.payload), e;
93
93
  }
94
- function T(n, e) {
94
+ function C(n, e) {
95
95
  const t = [], i = n.props?.actions;
96
96
  if (Array.isArray(i))
97
97
  for (const a of i) {
98
- const r = _(a);
98
+ const r = m(a);
99
99
  r && t.push(r);
100
100
  }
101
101
  const o = n.props?.buttons;
102
102
  if (Array.isArray(o))
103
103
  for (const a of o) {
104
104
  if (!a || typeof a != "object") continue;
105
- const r = a, c = typeof r.label == "string" ? r.label : void 0, l = _(r.action, c);
105
+ const r = a, c = typeof r.label == "string" ? r.label : void 0, l = m(r.action, c);
106
106
  l && t.push(l);
107
107
  }
108
108
  if (n.children)
109
109
  for (const a of n.children) {
110
110
  const r = e.elements[a];
111
111
  if (!r || r.type !== "ActionButton") continue;
112
- const c = typeof r.props?.label == "string" ? r.props.label : void 0, l = _(r.props?.action, c);
112
+ const c = typeof r.props?.label == "string" ? r.props.label : void 0, l = m(r.props?.action, c);
113
113
  l && t.push(l);
114
114
  }
115
115
  const s = /* @__PURE__ */ new Set();
116
116
  return t.filter((a) => s.has(a.title) ? !1 : (s.add(a.title), !0));
117
117
  }
118
- const E = {
118
+ const I = {
119
119
  ButtonRow: ({ element: n, spec: e, context: t }) => {
120
- const i = T(n, e).map(A), o = n.props?.orientation, s = {
120
+ const i = C(n, e).map(T), o = n.props?.orientation, s = {
121
121
  actions: i,
122
122
  onAction: t.onAction,
123
123
  defaultCtaText: t.i18n.defaultCtaText,
124
124
  quickQuestionsAriaLabel: t.i18n.quickQuestionsAriaLabel
125
125
  };
126
- return t.onOpenChat !== void 0 && (s.onOpenChat = t.onOpenChat), t.ctaText !== void 0 && (s.ctaText = t.ctaText), (o === "horizontal" || o === "vertical") && (s.orientation = o), y(s);
126
+ return t.onOpenChat !== void 0 && (s.onOpenChat = t.onOpenChat), t.ctaText !== void 0 && (s.ctaText = t.ctaText), (o === "horizontal" || o === "vertical") && (s.orientation = o), A(s);
127
127
  },
128
128
  ActionButtons: ({ element: n, spec: e, context: t }) => {
129
129
  const o = {
130
- actions: T(n, e).map(A),
130
+ actions: C(n, e).map(T),
131
131
  onAction: t.onAction,
132
132
  defaultCtaText: t.i18n.defaultCtaText,
133
133
  quickQuestionsAriaLabel: t.i18n.quickQuestionsAriaLabel
134
134
  };
135
- return t.onOpenChat !== void 0 && (o.onOpenChat = t.onOpenChat), t.ctaText !== void 0 && (o.ctaText = t.ctaText), y(o);
135
+ return t.onOpenChat !== void 0 && (o.onOpenChat = t.onOpenChat), t.ctaText !== void 0 && (o.ctaText = t.ctaText), A(o);
136
136
  },
137
137
  ActionButton: ({ element: n, context: e }) => {
138
138
  const t = document.createElement("button");
139
139
  t.className = "gengage-qna-button", t.type = "button";
140
140
  const i = n.props?.label;
141
141
  typeof i == "string" ? t.textContent = i : t.textContent = e.i18n.defaultCtaText;
142
- const o = _(n.props?.action, typeof i == "string" ? i : void 0);
142
+ const o = m(n.props?.action, typeof i == "string" ? i : void 0);
143
143
  return o && t.addEventListener("click", () => e.onAction(o)), t;
144
144
  },
145
145
  TextInput: ({ element: n, context: e }) => {
@@ -162,7 +162,7 @@ const E = {
162
162
  // Backend may send ProductCard but QNA intentionally skips it;
163
163
  // the product is shown once chat opens via openWithAction().
164
164
  ProductCard: () => null
165
- }, I = ({
165
+ }, v = ({
166
166
  element: n,
167
167
  renderElement: e
168
168
  }) => {
@@ -176,9 +176,9 @@ const E = {
176
176
  return t;
177
177
  };
178
178
  function J() {
179
- return { ...E };
179
+ return { ...I };
180
180
  }
181
- function V(n, e, t = E, i = I) {
181
+ function V(n, e, t = I, i = v) {
182
182
  return Q({
183
183
  spec: n,
184
184
  context: e,
@@ -187,7 +187,7 @@ function V(n, e, t = E, i = I) {
187
187
  unknownRenderer: i
188
188
  });
189
189
  }
190
- const v = {
190
+ const S = {
191
191
  quickQuestionsAriaLabel: "Hızlı sorular",
192
192
  askQuestionAriaLabel: "Soru sorun",
193
193
  defaultInputPlaceholder: "Bir soru sorun...",
@@ -206,23 +206,23 @@ function K(n) {
206
206
  return n ? n.toLowerCase().split("-")[0] ?? "tr" : "tr";
207
207
  }
208
208
  function X(n) {
209
- return K(n) === "en" ? Y : v;
209
+ return K(n) === "en" ? Y : S;
210
210
  }
211
- const Z = g({
211
+ const Z = _({
212
212
  title: p(),
213
213
  type: p(),
214
214
  payload: B().optional()
215
- }), tt = g({
215
+ }), tt = _({
216
216
  label: p(),
217
217
  action: Z,
218
- variant: C(["primary", "outline", "ghost"]).optional()
219
- }), et = g({
220
- orientation: C(["horizontal", "vertical"]).optional(),
218
+ variant: E(["primary", "outline", "ghost"]).optional()
219
+ }), et = _({
220
+ orientation: E(["horizontal", "vertical"]).optional(),
221
221
  wrap: R().optional()
222
- }), nt = g({
223
- placeholder: L([p(), N(p())]).optional(),
222
+ }), nt = _({
223
+ placeholder: L([p(), P(p())]).optional(),
224
224
  ctaLabel: p().optional()
225
- }), ot = g({
225
+ }), ot = _({
226
226
  text: p()
227
227
  }), ct = {
228
228
  components: {
@@ -244,9 +244,9 @@ const Z = g({
244
244
  }
245
245
  }
246
246
  };
247
- class it extends P {
247
+ class it extends N {
248
248
  constructor() {
249
- super(...arguments), this._abortController = null, this._debounceTimer = null, this._contentEl = null, this._i18n = v, this._actionHandler = this._handleAction.bind(this), this._openChatHandler = this._handleOpenChat.bind(this);
249
+ super(...arguments), this._abortController = null, this._debounceTimer = null, this._contentEl = null, this._i18n = S, this._actionHandler = this._handleAction.bind(this), this._openChatHandler = this._handleOpenChat.bind(this);
250
250
  }
251
251
  async onInit(e) {
252
252
  this._i18n = this._resolveI18n(e), this._contentEl = document.createElement("div"), this._contentEl.className = "gengage-qna-container", this.root.appendChild(this._contentEl);
@@ -326,20 +326,29 @@ class it extends P {
326
326
  const u = document.createElement("h3");
327
327
  u.className = "gengage-qna-heading", u.textContent = this.config.staticQuestionText, this._contentEl.appendChild(u);
328
328
  }
329
- const d = this.config.inputPlaceholder, h = c.actions.length > 0 && d === !0 ? c.actions.map((u) => u.title) : d === !0 ? this._i18n.defaultInputPlaceholder : d ?? this._i18n.defaultInputPlaceholder, f = {
329
+ const h = this.config.inputPlaceholder;
330
+ let d;
331
+ if (h !== !0)
332
+ d = h ?? this._i18n.defaultInputPlaceholder;
333
+ else if (c.actions.length > 0) {
334
+ const u = c.actions.filter((f) => f.type === "user_message" || f.title.includes("?")).map((f) => f.title);
335
+ d = u.length > 0 ? u : this._i18n.defaultInputPlaceholder;
336
+ } else
337
+ d = this._i18n.defaultInputPlaceholder;
338
+ const g = {
330
339
  onAction: this._actionHandler,
331
340
  i18n: this._i18n
332
341
  };
333
- this.config.hideButtonRowCta || (f.onOpenChat = this._openChatHandler, this.config.ctaText !== void 0 && (f.ctaText = this.config.ctaText)), h !== void 0 && (f.inputPlaceholder = h);
334
- const S = this._buildFallbackActionsSpec(c.actions), b = (c.uiSpecs.length > 0 ? c.uiSpecs : [S]).filter((u) => Object.keys(u.elements).length > 0);
342
+ this.config.hideButtonRowCta || (g.onOpenChat = this._openChatHandler, this.config.ctaText !== void 0 && (g.ctaText = this.config.ctaText)), d !== void 0 && (g.inputPlaceholder = d);
343
+ const k = this._buildFallbackActionsSpec(c.actions), b = (c.uiSpecs.length > 0 ? c.uiSpecs : [k]).filter((u) => Object.keys(u.elements).length > 0);
335
344
  for (const u of b) {
336
- const k = this._renderUISpec(u, f);
337
- this._contentEl.appendChild(k);
345
+ const f = this._renderUISpec(u, g);
346
+ this._contentEl.appendChild(f);
338
347
  }
339
- b.length > 0 && D("qna"), !this._specIncludesType(b, "TextInput") && this._appendStandaloneInput(f, h);
348
+ b.length > 0 && D("qna"), !this._specIncludesType(b, "TextInput") && this._appendStandaloneInput(g, d);
340
349
  } catch (a) {
341
350
  if (a instanceof DOMException && a.name === "AbortError") return;
342
- if (m("gengage:global:error", {
351
+ if (y("gengage:global:error", {
343
352
  source: "qna",
344
353
  code: "FETCH_ERROR",
345
354
  message: F(this.config.locale)
@@ -378,7 +387,7 @@ class it extends P {
378
387
  return M(e, this.config.renderer?.registry);
379
388
  }
380
389
  _renderUISpec(e, t) {
381
- const i = this._resolveUISpecRegistry(), o = this.config.renderer?.unknownRenderer ?? I, s = (c, l) => V(c, l, i, o), a = this.config.renderer?.renderUISpec;
390
+ const i = this._resolveUISpecRegistry(), o = this.config.renderer?.unknownRenderer ?? v, s = (c, l) => V(c, l, i, o), a = this.config.renderer?.renderUISpec;
382
391
  return a ? a(e, t, {
383
392
  registry: i,
384
393
  unknownRenderer: o,
@@ -433,11 +442,11 @@ class it extends P {
433
442
  this._contentEl.appendChild(o);
434
443
  }
435
444
  _handleAction(e) {
436
- G(e.title, e.type), this.config.onActionSelected?.(e), m("gengage:qna:action", e);
445
+ G(e.title, e.type), this.config.onActionSelected?.(e), y("gengage:qna:action", e);
437
446
  }
438
447
  _handleOpenChat() {
439
448
  const e = this._contentEl?.querySelector(".gengage-qna-input");
440
- e && e.focus(), this.config.onOpenChat?.(), m("gengage:qna:open-chat", {});
449
+ e && e.focus(), this.config.onOpenChat?.(), y("gengage:qna:open-chat", {});
441
450
  }
442
451
  }
443
452
  function lt() {
@@ -447,7 +456,7 @@ export {
447
456
  it as GengageQNA,
448
457
  J as createDefaultQnaUISpecRegistry,
449
458
  lt as createQNAWidget,
450
- I as defaultQnaUnknownUISpecRenderer,
459
+ v as defaultQnaUnknownUISpecRenderer,
451
460
  ct as qnaCatalog,
452
461
  V as renderQnaUISpec
453
462
  };
package/dist/qna.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"qna.js","sources":["../src/qna/api.ts","../src/qna/components/ButtonRow.ts","../src/qna/components/TextInput.ts","../src/qna/components/renderUISpec.ts","../src/qna/locales/tr.ts","../src/qna/locales/en.ts","../src/qna/locales/index.ts","../src/qna/catalog.ts","../src/qna/index.ts"],"sourcesContent":["import { consumeStream } from '../common/streaming.js';\nimport { buildChatEndpointUrl } from '../common/api-paths.js';\nimport { adaptBackendEvent } from '../common/protocol-adapter.js';\nimport type { StreamEvent, UISpec } from '../common/types.js';\nimport type { ChatTransportConfig } from '../common/api-paths.js';\n\nexport interface LauncherActionRequest {\n account_id: string;\n session_id: string;\n correlation_id: string;\n sku: string;\n page_type?: string;\n locale?: string;\n output_language?: string;\n mode?: string;\n}\n\nexport interface LauncherActionResult {\n uiSpecs: UISpec[];\n actions: Array<{ title: string; type: string; payload?: unknown }>;\n}\n\nexport async function fetchLauncherActions(\n request: LauncherActionRequest,\n transport: ChatTransportConfig,\n signal?: AbortSignal,\n): Promise<LauncherActionResult> {\n const url = buildChatEndpointUrl('launcher_action', transport);\n const result: LauncherActionResult = { uiSpecs: [], actions: [] };\n\n const fetchInit: RequestInit = {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(request),\n };\n if (signal !== undefined) fetchInit.signal = signal;\n const response = await fetch(url, fetchInit);\n\n if (!response.ok) {\n throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n }\n\n const streamOpts: import('../common/streaming.js').StreamOptions = {\n onEvent: (event: StreamEvent) => {\n const normalized = adaptBackendEvent(event as unknown as Record<string, unknown>);\n\n if (!normalized) return;\n\n if (normalized.type === 'ui_spec') {\n result.uiSpecs.push(normalized.spec);\n }\n\n if (normalized.type === 'ui_spec' && normalized.spec.elements) {\n for (const el of Object.values(normalized.spec.elements)) {\n if (el.type === 'ActionButton' && el.props?.['action']) {\n const action = el.props['action'] as { title: string; type: string; payload?: unknown };\n result.actions.push(action);\n }\n }\n }\n },\n };\n if (signal !== undefined) streamOpts.signal = signal;\n\n await consumeStream(response, streamOpts);\n\n return result;\n}\n","import type { ActionPayload } from '../../common/types.js';\n\nexport interface ButtonRowOptions {\n actions: Array<{ title: string; type: string; payload?: unknown }>;\n onAction: (action: ActionPayload) => void;\n ctaText?: string;\n defaultCtaText?: string;\n onOpenChat?: () => void;\n orientation?: 'horizontal' | 'vertical';\n quickQuestionsAriaLabel?: string;\n}\n\nexport function renderButtonRow(options: ButtonRowOptions): HTMLElement {\n const container = document.createElement('div');\n container.className = 'gengage-qna-buttons';\n container.setAttribute('role', 'group');\n container.setAttribute('aria-label', options.quickQuestionsAriaLabel ?? 'Quick questions');\n\n if (options.orientation === 'vertical') {\n container.style.flexDirection = 'column';\n }\n\n for (const action of options.actions) {\n const button = document.createElement('button');\n button.className = 'gengage-qna-button';\n button.textContent = action.title;\n button.type = 'button';\n button.addEventListener('click', () => {\n const actionPayload: ActionPayload = {\n title: action.title,\n type: action.type,\n };\n if (action.payload !== undefined) actionPayload.payload = action.payload;\n options.onAction(actionPayload);\n });\n container.appendChild(button);\n }\n\n if (options.ctaText || options.onOpenChat) {\n const cta = document.createElement('button');\n cta.className = 'gengage-qna-cta';\n cta.textContent = options.ctaText ?? options.defaultCtaText ?? 'Ask something else';\n cta.type = 'button';\n cta.addEventListener('click', () => {\n options.onOpenChat?.();\n });\n container.appendChild(cta);\n }\n\n return container;\n}\n","import type { ActionPayload } from '../../common/types.js';\n\nexport interface TextInputOptions {\n placeholders?: string | string[];\n ctaLabel?: string;\n defaultInputPlaceholder?: string;\n askQuestionAriaLabel?: string;\n sendButtonText?: string;\n sendQuestionAriaLabel?: string;\n onSubmit: (action: ActionPayload) => void;\n}\n\nexport function renderTextInput(options: TextInputOptions): HTMLElement {\n const container = document.createElement('div');\n container.className = 'gengage-qna-input-wrapper';\n\n const input = document.createElement('input');\n input.type = 'text';\n input.className = 'gengage-qna-input';\n input.setAttribute('aria-label', options.askQuestionAriaLabel ?? 'Ask a question');\n\n // Rotating placeholders\n const placeholders = Array.isArray(options.placeholders)\n ? options.placeholders\n : options.placeholders\n ? [options.placeholders]\n : [options.defaultInputPlaceholder ?? 'Ask a question...'];\n\n let placeholderIndex = 0;\n input.placeholder = placeholders[0] ?? '';\n\n let rotateTimer: ReturnType<typeof setInterval> | null = null;\n let fadeTimer: ReturnType<typeof setTimeout> | null = null;\n if (placeholders.length > 1) {\n rotateTimer = setInterval(() => {\n input.classList.add('gengage-qna-input--fade');\n fadeTimer = setTimeout(() => {\n placeholderIndex = (placeholderIndex + 1) % placeholders.length;\n input.placeholder = placeholders[placeholderIndex] ?? '';\n input.classList.remove('gengage-qna-input--fade');\n }, 180);\n }, 3000);\n }\n\n const sendBtn = document.createElement('button');\n sendBtn.className = 'gengage-qna-send';\n sendBtn.type = 'button';\n sendBtn.textContent = options.ctaLabel ?? options.sendButtonText ?? 'Ask';\n sendBtn.setAttribute('aria-label', options.sendQuestionAriaLabel ?? 'Send question');\n\n const submit = () => {\n const text = input.value.trim();\n if (!text) return;\n options.onSubmit({\n title: text,\n type: 'user_message',\n // Align with chat composer: user_message payload is plain string.\n payload: text,\n });\n input.value = '';\n };\n\n sendBtn.addEventListener('click', submit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Enter') {\n e.preventDefault();\n submit();\n }\n });\n\n container.appendChild(input);\n container.appendChild(sendBtn);\n\n // Cleanup method\n (container as HTMLElement & { _cleanup?: () => void })._cleanup = () => {\n if (rotateTimer) clearInterval(rotateTimer);\n if (fadeTimer) clearTimeout(fadeTimer);\n };\n\n return container;\n}\n","import { renderUISpecWithRegistry } from '../../common/renderer/index.js';\nimport type { UISpecDomRegistry, UISpecDomUnknownRenderer } from '../../common/renderer/index.js';\nimport type { UISpec, UIElement, ActionPayload } from '../../common/types.js';\nimport type { QNAUISpecRenderContext } from '../types.js';\nimport { renderButtonRow } from './ButtonRow.js';\nimport { renderTextInput } from './TextInput.js';\n\nexport type QNAUISpecRegistry = UISpecDomRegistry<QNAUISpecRenderContext>;\n\nfunction toActionPayload(raw: unknown, labelFallback?: string): ActionPayload | null {\n if (!raw || typeof raw !== 'object') return null;\n const obj = raw as Record<string, unknown>;\n const type = obj['type'];\n if (typeof type !== 'string' || type.length === 0) return null;\n\n const titleRaw = obj['title'];\n const payload = obj['payload'];\n const title = typeof titleRaw === 'string' && titleRaw.length > 0 ? titleRaw : labelFallback;\n if (!title) return null;\n\n const action: ActionPayload = { title, type };\n if (payload !== undefined) action.payload = payload;\n return action;\n}\n\nfunction actionPayloadToRowAction(action: ActionPayload): { title: string; type: string; payload?: unknown } {\n const rowAction: { title: string; type: string; payload?: unknown } = {\n title: action.title,\n type: action.type,\n };\n if (action.payload !== undefined) rowAction.payload = action.payload;\n return rowAction;\n}\n\nfunction collectButtonRowActions(element: UIElement, spec: UISpec): ActionPayload[] {\n const actions: ActionPayload[] = [];\n\n const directActions = element.props?.['actions'];\n if (Array.isArray(directActions)) {\n for (const rawAction of directActions) {\n const normalized = toActionPayload(rawAction);\n if (normalized) actions.push(normalized);\n }\n }\n\n const directButtons = element.props?.['buttons'];\n if (Array.isArray(directButtons)) {\n for (const rawButton of directButtons) {\n if (!rawButton || typeof rawButton !== 'object') continue;\n const obj = rawButton as Record<string, unknown>;\n const label = typeof obj['label'] === 'string' ? obj['label'] : undefined;\n const action = toActionPayload(obj['action'], label);\n if (action) actions.push(action);\n }\n }\n\n if (element.children) {\n for (const childId of element.children) {\n const child = spec.elements[childId];\n if (!child || child.type !== 'ActionButton') continue;\n const label = typeof child.props?.['label'] === 'string' ? child.props['label'] : undefined;\n const action = toActionPayload(child.props?.['action'], label);\n if (action) actions.push(action);\n }\n }\n\n // Deduplicate by title — backend may provide same actions via props and children\n const seen = new Set<string>();\n return actions.filter((a) => {\n if (seen.has(a.title)) return false;\n seen.add(a.title);\n return true;\n });\n}\n\nconst DEFAULT_QNA_UI_SPEC_REGISTRY: QNAUISpecRegistry = {\n ButtonRow: ({ element, spec, context }) => {\n const rowActions = collectButtonRowActions(element, spec).map(actionPayloadToRowAction);\n const orientation = element.props?.['orientation'];\n\n const options: import('./ButtonRow.js').ButtonRowOptions = {\n actions: rowActions,\n onAction: context.onAction,\n defaultCtaText: context.i18n.defaultCtaText,\n quickQuestionsAriaLabel: context.i18n.quickQuestionsAriaLabel,\n };\n if (context.onOpenChat !== undefined) options.onOpenChat = context.onOpenChat;\n if (context.ctaText !== undefined) options.ctaText = context.ctaText;\n if (orientation === 'horizontal' || orientation === 'vertical') options.orientation = orientation;\n return renderButtonRow(options);\n },\n\n ActionButtons: ({ element, spec, context }) => {\n const rowActions = collectButtonRowActions(element, spec).map(actionPayloadToRowAction);\n const options: import('./ButtonRow.js').ButtonRowOptions = {\n actions: rowActions,\n onAction: context.onAction,\n defaultCtaText: context.i18n.defaultCtaText,\n quickQuestionsAriaLabel: context.i18n.quickQuestionsAriaLabel,\n };\n if (context.onOpenChat !== undefined) options.onOpenChat = context.onOpenChat;\n if (context.ctaText !== undefined) options.ctaText = context.ctaText;\n return renderButtonRow(options);\n },\n\n ActionButton: ({ element, context }) => {\n const button = document.createElement('button');\n button.className = 'gengage-qna-button';\n button.type = 'button';\n\n const label = element.props?.['label'];\n if (typeof label === 'string') {\n button.textContent = label;\n } else {\n button.textContent = context.i18n.defaultCtaText;\n }\n\n const action = toActionPayload(element.props?.['action'], typeof label === 'string' ? label : undefined);\n if (action) {\n button.addEventListener('click', () => context.onAction(action));\n }\n return button;\n },\n\n TextInput: ({ element, context }) => {\n const placeholder = element.props?.['placeholder'];\n const placeholders =\n typeof placeholder === 'string' || Array.isArray(placeholder) ? placeholder : context.inputPlaceholder;\n // Keep input submit label independent from the quick-question CTA text.\n const ctaLabel = typeof element.props?.['ctaLabel'] === 'string' ? element.props['ctaLabel'] : undefined;\n\n const options: import('./TextInput.js').TextInputOptions = {\n onSubmit: context.onAction,\n askQuestionAriaLabel: context.i18n.askQuestionAriaLabel,\n defaultInputPlaceholder: context.i18n.defaultInputPlaceholder,\n sendButtonText: context.i18n.sendButton,\n sendQuestionAriaLabel: context.i18n.sendQuestionAriaLabel,\n };\n if (placeholders !== undefined) options.placeholders = placeholders;\n if (ctaLabel !== undefined) options.ctaLabel = ctaLabel;\n return renderTextInput(options);\n },\n\n QuestionHeading: ({ element }) => {\n const heading = document.createElement('h3');\n heading.className = 'gengage-qna-heading';\n const text = element.props?.['text'];\n heading.textContent = typeof text === 'string' ? text : '';\n return heading;\n },\n\n // ProductCard: no-op — QNA renders only question pills + text input.\n // Backend may send ProductCard but QNA intentionally skips it;\n // the product is shown once chat opens via openWithAction().\n ProductCard: () => null,\n};\n\nexport const defaultQnaUnknownUISpecRenderer: UISpecDomUnknownRenderer<QNAUISpecRenderContext> = ({\n element,\n renderElement,\n}) => {\n if (import.meta.env?.DEV) {\n console.warn(`[gengage:qna] Unknown ui_spec component type: ${element.type}`);\n }\n if (!element.children || element.children.length === 0) {\n return null;\n }\n const wrapper = document.createElement('div');\n for (const childId of element.children) {\n const rendered = renderElement(childId);\n if (rendered) wrapper.appendChild(rendered);\n }\n return wrapper;\n};\n\nexport function createDefaultQnaUISpecRegistry(): QNAUISpecRegistry {\n return { ...DEFAULT_QNA_UI_SPEC_REGISTRY };\n}\n\nexport function renderQnaUISpec(\n spec: UISpec,\n context: QNAUISpecRenderContext,\n registry = DEFAULT_QNA_UI_SPEC_REGISTRY,\n unknownRenderer: UISpecDomUnknownRenderer<QNAUISpecRenderContext> = defaultQnaUnknownUISpecRenderer,\n): HTMLElement {\n return renderUISpecWithRegistry({\n spec,\n context,\n registry,\n containerClassName: 'gengage-qna-uispec',\n unknownRenderer,\n });\n}\n","import type { QNAI18n } from '../types.js';\n\nexport const QNA_I18N_TR: QNAI18n = {\n quickQuestionsAriaLabel: 'Hızlı sorular',\n askQuestionAriaLabel: 'Soru sorun',\n defaultInputPlaceholder: 'Bir soru sorun...',\n sendButton: 'Sor',\n sendQuestionAriaLabel: 'Soruyu gönder',\n defaultCtaText: 'Başka bir şey sor',\n};\n","import type { QNAI18n } from '../types.js';\n\nexport const QNA_I18N_EN: QNAI18n = {\n quickQuestionsAriaLabel: 'Quick questions',\n askQuestionAriaLabel: 'Ask a question',\n defaultInputPlaceholder: 'Ask a question...',\n sendButton: 'Ask',\n sendQuestionAriaLabel: 'Send question',\n defaultCtaText: 'Ask something else',\n};\n","import type { QNAI18n } from '../types.js';\nimport { QNA_I18N_TR } from './tr.js';\nimport { QNA_I18N_EN } from './en.js';\n\nfunction normalizeLocale(locale?: string): string {\n if (!locale) return 'tr';\n return locale.toLowerCase().split('-')[0] ?? 'tr';\n}\n\nexport function resolveQnaLocale(locale?: string): QNAI18n {\n switch (normalizeLocale(locale)) {\n case 'en':\n return QNA_I18N_EN;\n default:\n return QNA_I18N_TR;\n }\n}\n\nexport { QNA_I18N_TR, QNA_I18N_EN };\n","/**\n * QNA widget — json-render catalog definition.\n *\n * Backend endpoint: POST /chat/launcher_action\n * Response format: NDJSON stream, each line a StreamEvent.\n *\n * The backend returns a `ui_spec` event whose elements are typed to the\n * component names defined below. The frontend renders them via ./registry.\n *\n * Customising:\n * Fork this repo and replace ./registry implementations with your own.\n * The catalog schema (component names + prop types) is the API contract\n * with the backend — do not rename components without a backend release.\n */\n\nimport { z } from 'zod';\n\nconst ActionPayloadSchema = z.object({\n title: z.string(),\n type: z.string(),\n payload: z.unknown().optional(),\n});\n\nexport const ActionButtonSchema = z.object({\n label: z.string(),\n action: ActionPayloadSchema,\n variant: z.enum(['primary', 'outline', 'ghost']).optional(),\n});\n\nexport const ButtonRowSchema = z.object({\n orientation: z.enum(['horizontal', 'vertical']).optional(),\n wrap: z.boolean().optional(),\n});\n\nexport const TextInputSchema = z.object({\n placeholder: z.union([z.string(), z.array(z.string())]).optional(),\n ctaLabel: z.string().optional(),\n});\n\nexport const QuestionHeadingSchema = z.object({\n text: z.string(),\n});\n\nexport const qnaCatalog = {\n components: {\n ActionButton: {\n schema: ActionButtonSchema,\n description: 'A single clickable QNA action button.',\n },\n ButtonRow: {\n schema: ButtonRowSchema,\n description: 'Container for a group of QNA action buttons.',\n },\n TextInput: {\n schema: TextInputSchema,\n description: 'Free-text input with rotating placeholder and CTA button.',\n },\n QuestionHeading: {\n schema: QuestionHeadingSchema,\n description: 'A heading displayed above the QNA button group.',\n },\n },\n} as const;\n\nexport type QNACatalog = typeof qnaCatalog;\nexport type QNAComponentName = keyof QNACatalog['components'];\n","/**\n * QNA widget -- public entry point.\n *\n * Renders contextual action buttons (and an optional free-text input) on\n * product or content pages. When a user taps a button, the widget opens\n * the Chat widget with that action pre-loaded.\n *\n * Backend: POST /chat/launcher_action\n * Protocol: NDJSON stream -> json-render UISpec -> ActionButton / ButtonRow / TextInput\n */\n\nimport type { ActionPayload, PageContext, UISpec, UIElement } from '../common/types.js';\nimport type { ChatTransportConfig } from '../common/api-paths.js';\nimport type { UISpecRenderHelpers } from '../common/renderer/index.js';\nimport { mergeUISpecRegistry } from '../common/renderer/index.js';\nimport { BaseWidget } from '../common/widget-base.js';\nimport { dispatch } from '../common/events.js';\nimport { getGlobalErrorMessage } from '../common/global-error-toast.js';\nimport {\n streamStartEvent,\n streamDoneEvent,\n streamErrorEvent,\n widgetHistorySnapshotEvent,\n} from '../common/analytics-events.js';\nimport { fetchLauncherActions } from './api.js';\nimport * as ga from '../common/ga-datalayer.js';\nimport {\n createDefaultQnaUISpecRegistry,\n defaultQnaUnknownUISpecRenderer,\n renderQnaUISpec,\n} from './components/renderUISpec.js';\nimport type { QNAWidgetConfig, QNAI18n, QNAUISpecRenderContext } from './types.js';\nimport { QNA_I18N_TR, resolveQnaLocale } from './locales/index.js';\n\n// Inline CSS import marker - Vite will bundle this\nimport './components/qna.css';\n\n/**\n * Contextual Q&A action buttons for product pages.\n * Renders quick-action buttons that open the chat widget with a pre-built query.\n *\n * @example\n * ```ts\n * import { GengageQNA, wireQNAToChat, bootstrapSession } from '@gengage/assistant-fe';\n *\n * const qna = new GengageQNA();\n * await qna.init({\n * accountId: 'mystore',\n * middlewareUrl: 'https://chat.gengage.ai',\n * mountTarget: '#qna-section',\n * pageContext: { pageType: 'pdp', sku: '12345' },\n * session: { sessionId: bootstrapSession() },\n * });\n * wireQNAToChat(); // Wire button clicks to chat.openWithAction()\n * ```\n */\nexport class GengageQNA extends BaseWidget<QNAWidgetConfig> {\n private _abortController: AbortController | null = null;\n private _debounceTimer: ReturnType<typeof setTimeout> | null = null;\n private _contentEl: HTMLElement | null = null;\n private _lastSku: string | undefined;\n private _i18n: QNAI18n = QNA_I18N_TR;\n\n protected async onInit(config: QNAWidgetConfig): Promise<void> {\n this._i18n = this._resolveI18n(config);\n\n this._contentEl = document.createElement('div');\n this._contentEl.className = 'gengage-qna-container';\n this.root.appendChild(this._contentEl);\n\n const sku = config.pageContext?.sku;\n if (sku) {\n this._lastSku = sku;\n await this._fetchAndRender(sku);\n }\n\n this.isVisible = true;\n ga.trackInit('qna');\n }\n\n protected onUpdate(context: Partial<PageContext>): void {\n const newSku = context.sku;\n if (!newSku || newSku === this._lastSku) return;\n\n // Debounce rapid SPA navigations\n if (this._debounceTimer) clearTimeout(this._debounceTimer);\n this._debounceTimer = setTimeout(() => {\n this._debounceTimer = null;\n this._lastSku = newSku;\n void this._fetchAndRender(newSku);\n }, 50);\n }\n\n protected onShow(): void {\n if (this._contentEl) {\n this._contentEl.style.opacity = '0';\n this._contentEl.style.transition = 'opacity 0.2s ease-in';\n requestAnimationFrame(() => {\n if (this._contentEl) this._contentEl.style.opacity = '1';\n });\n }\n }\n\n protected onHide(): void {\n // Preserve fetched data for re-show\n }\n\n protected onDestroy(): void {\n this._abort();\n if (this._debounceTimer) {\n clearTimeout(this._debounceTimer);\n this._debounceTimer = null;\n }\n if (this._contentEl) {\n this._cleanupTextInputTimers();\n this._contentEl.remove();\n this._contentEl = null;\n }\n }\n\n // ---------------------------------------------------------------------------\n // Internal\n // ---------------------------------------------------------------------------\n\n private _abort(): void {\n this._abortController?.abort();\n this._abortController = null;\n }\n\n /** Clean up TextInput placeholder rotation timers to prevent interval leaks. */\n private _cleanupTextInputTimers(): void {\n if (!this._contentEl) return;\n const wrappers = this._contentEl.querySelectorAll('.gengage-qna-input-wrapper');\n for (const wrapper of wrappers) {\n (wrapper as HTMLElement & { _cleanup?: () => void })._cleanup?.();\n }\n }\n\n private async _fetchAndRender(sku: string): Promise<void> {\n this._abort();\n this._abortController = new AbortController();\n\n if (!this._contentEl) return;\n // Clean up TextInput timers before clearing DOM to prevent interval leaks\n this._cleanupTextInputTimers();\n this._contentEl.innerHTML = '';\n\n // Show loading dots\n const loading = this._createLoadingIndicator();\n this._contentEl.appendChild(loading);\n\n const transport: ChatTransportConfig = {\n middlewareUrl: this.config.middlewareUrl,\n };\n\n const requestId = crypto.randomUUID();\n const fetchStart = Date.now();\n\n this.track(\n streamStartEvent(this.analyticsContext(), {\n endpoint: 'launcher_action',\n request_id: requestId,\n widget: 'qna',\n }),\n );\n\n try {\n const launcherReq: import('./api.js').LauncherActionRequest = {\n account_id: this.config.accountId,\n session_id: this.config.session?.sessionId ?? '',\n correlation_id: this.config.session?.sessionId ?? '',\n sku,\n locale: this.config.locale ?? 'tr',\n };\n const pageType = this.config.pageContext?.pageType;\n if (pageType !== undefined) launcherReq.page_type = pageType;\n\n const result = await fetchLauncherActions(launcherReq, transport, this._abortController.signal);\n\n this.track(\n streamDoneEvent(this.analyticsContext(), {\n request_id: requestId,\n latency_ms: Date.now() - fetchStart,\n chunk_count: result.actions.length,\n widget: 'qna',\n }),\n );\n\n this.track(\n widgetHistorySnapshotEvent(this.analyticsContext(), {\n message_count: result.actions.length,\n history_ref: requestId,\n redaction_level: 'none',\n widget: 'qna',\n }),\n );\n\n if (!this._contentEl) return;\n this._contentEl.innerHTML = '';\n\n const hasQuestionHeading = this._specIncludesType(result.uiSpecs, 'QuestionHeading');\n\n // Render heading if configured and backend didn't provide one\n if (!hasQuestionHeading && this.config.showStaticQuestion && this.config.staticQuestionText) {\n const heading = document.createElement('h3');\n heading.className = 'gengage-qna-heading';\n heading.textContent = this.config.staticQuestionText;\n this._contentEl.appendChild(heading);\n }\n\n const cfgPlaceholders = this.config.inputPlaceholder;\n const effectivePlaceholders =\n result.actions.length > 0 && cfgPlaceholders === true\n ? result.actions.map((a) => a.title)\n : cfgPlaceholders === true\n ? this._i18n.defaultInputPlaceholder\n : (cfgPlaceholders ?? this._i18n.defaultInputPlaceholder);\n\n const renderContext: QNAUISpecRenderContext = {\n onAction: this._actionHandler,\n i18n: this._i18n,\n };\n if (!this.config.hideButtonRowCta) {\n renderContext.onOpenChat = this._openChatHandler;\n if (this.config.ctaText !== undefined) renderContext.ctaText = this.config.ctaText;\n }\n if (effectivePlaceholders !== undefined) renderContext.inputPlaceholder = effectivePlaceholders;\n\n const fallbackSpec = this._buildFallbackActionsSpec(result.actions);\n const specsToRender = result.uiSpecs.length > 0 ? result.uiSpecs : [fallbackSpec];\n const nonEmptySpecs = specsToRender.filter((spec) => Object.keys(spec.elements).length > 0);\n\n for (const spec of nonEmptySpecs) {\n const rendered = this._renderUISpec(spec, renderContext);\n this._contentEl.appendChild(rendered);\n }\n\n if (nonEmptySpecs.length > 0) {\n ga.trackShow('qna');\n }\n\n const shouldRenderStandaloneInput = !this._specIncludesType(nonEmptySpecs, 'TextInput');\n if (shouldRenderStandaloneInput) {\n this._appendStandaloneInput(renderContext, effectivePlaceholders);\n }\n } catch (err) {\n if (err instanceof DOMException && err.name === 'AbortError') return;\n\n dispatch('gengage:global:error', {\n source: 'qna',\n code: 'FETCH_ERROR',\n message: getGlobalErrorMessage(this.config.locale),\n });\n\n this.track(\n streamErrorEvent(this.analyticsContext(), {\n request_id: requestId,\n error_code: 'FETCH_ERROR',\n error_message: err instanceof Error ? err.message : String(err),\n widget: 'qna',\n }),\n );\n\n // Keep QNA usable during backend hiccups: render at least free-text input.\n if (this._contentEl) {\n this._cleanupTextInputTimers();\n this._contentEl.innerHTML = '';\n const fallbackPlaceholders =\n this.config.inputPlaceholder === true\n ? this._i18n.defaultInputPlaceholder\n : (this.config.inputPlaceholder ?? this._i18n.defaultInputPlaceholder);\n const fallbackContext: QNAUISpecRenderContext = {\n onAction: this._actionHandler,\n i18n: this._i18n,\n onOpenChat: this._openChatHandler,\n };\n if (this.config.ctaText !== undefined) fallbackContext.ctaText = this.config.ctaText;\n this._appendStandaloneInput(fallbackContext, fallbackPlaceholders);\n }\n\n if (import.meta.env?.DEV) {\n console.error('[gengage:qna] Failed to fetch launcher actions:', err);\n }\n }\n }\n\n private _createLoadingIndicator(): HTMLElement {\n const el = document.createElement('div');\n el.className = 'gengage-qna-loading';\n for (let i = 0; i < 3; i++) {\n const dot = document.createElement('div');\n dot.className = 'gengage-qna-loading-dot';\n el.appendChild(dot);\n }\n return el;\n }\n\n private _resolveI18n(config: QNAWidgetConfig): QNAI18n {\n const base = resolveQnaLocale(config.locale);\n return { ...base, ...config.i18n };\n }\n\n private _resolveUISpecRegistry() {\n const baseRegistry = createDefaultQnaUISpecRegistry();\n return mergeUISpecRegistry(baseRegistry, this.config.renderer?.registry);\n }\n\n private _renderUISpec(spec: UISpec, context: QNAUISpecRenderContext): HTMLElement {\n const registry = this._resolveUISpecRegistry();\n const unknownRenderer = this.config.renderer?.unknownRenderer ?? defaultQnaUnknownUISpecRenderer;\n const defaultRender = (inputSpec: UISpec, inputContext: QNAUISpecRenderContext) =>\n renderQnaUISpec(inputSpec, inputContext, registry, unknownRenderer);\n\n const override = this.config.renderer?.renderUISpec;\n if (!override) return defaultRender(spec, context);\n\n const helpers: UISpecRenderHelpers<QNAUISpecRenderContext> = {\n registry,\n unknownRenderer,\n defaultRender,\n };\n return override(spec, context, helpers);\n }\n\n private _specIncludesType(specs: UISpec[], type: string): boolean {\n for (const spec of specs) {\n for (const element of Object.values(spec.elements)) {\n if (element.type === type) return true;\n }\n }\n return false;\n }\n\n private _buildFallbackActionsSpec(actions: Array<{ title: string; type: string; payload?: unknown }>): UISpec {\n if (actions.length === 0) {\n return { root: 'root', elements: {} };\n }\n\n const elements: Record<string, UIElement> = {};\n const childIds: string[] = [];\n for (let i = 0; i < actions.length; i++) {\n const action = actions[i]!;\n const id = `action-${i}`;\n childIds.push(id);\n elements[id] = {\n type: 'ActionButton',\n props: {\n label: action.title,\n action: {\n title: action.title,\n type: action.type,\n payload: action.payload,\n },\n },\n };\n }\n elements['root'] = {\n type: 'ButtonRow',\n children: childIds,\n };\n return {\n root: 'root',\n elements,\n };\n }\n\n private _appendStandaloneInput(context: QNAUISpecRenderContext, placeholder?: string | string[]): void {\n if (!this._contentEl) return;\n const inputSpec: UISpec = {\n root: 'root',\n elements: {\n root: {\n type: 'TextInput',\n props: {\n placeholder,\n },\n },\n },\n };\n const renderedInput = this._renderUISpec(inputSpec, context);\n this._contentEl.appendChild(renderedInput);\n }\n\n private _handleAction(action: ActionPayload): void {\n ga.trackSuggestedQuestion(action.title, action.type);\n this.config.onActionSelected?.(action);\n dispatch('gengage:qna:action', action);\n }\n\n private _handleOpenChat(): void {\n // Couple CTA with the inline QNA text input when available.\n const input = this._contentEl?.querySelector<HTMLInputElement>('.gengage-qna-input');\n if (input) {\n input.focus();\n }\n this.config.onOpenChat?.();\n dispatch('gengage:qna:open-chat', {});\n }\n\n _actionHandler = this._handleAction.bind(this);\n _openChatHandler = this._handleOpenChat.bind(this);\n}\n\nexport function createQNAWidget(): GengageQNA {\n return new GengageQNA();\n}\n\nexport type { QNAWidgetConfig, QNAUIComponents, QNAI18n, QNAUISpecRenderContext, QNARendererConfig } from './types.js';\nexport {\n renderQnaUISpec,\n createDefaultQnaUISpecRegistry,\n defaultQnaUnknownUISpecRenderer,\n} from './components/renderUISpec.js';\nexport type { QNAUISpecRegistry } from './components/renderUISpec.js';\nexport { qnaCatalog } from './catalog.js';\nexport type { QNACatalog, QNAComponentName } from './catalog.js';\n"],"names":["fetchLauncherActions","request","transport","signal","url","buildChatEndpointUrl","result","fetchInit","response","streamOpts","event","normalized","adaptBackendEvent","el","action","consumeStream","renderButtonRow","options","container","button","actionPayload","cta","renderTextInput","input","placeholders","placeholderIndex","rotateTimer","fadeTimer","sendBtn","submit","text","e","toActionPayload","raw","labelFallback","obj","type","titleRaw","payload","title","actionPayloadToRowAction","rowAction","collectButtonRowActions","element","spec","actions","directActions","rawAction","directButtons","rawButton","label","childId","child","seen","DEFAULT_QNA_UI_SPEC_REGISTRY","context","rowActions","orientation","placeholder","ctaLabel","heading","defaultQnaUnknownUISpecRenderer","renderElement","wrapper","rendered","createDefaultQnaUISpecRegistry","renderQnaUISpec","registry","unknownRenderer","renderUISpecWithRegistry","QNA_I18N_TR","QNA_I18N_EN","normalizeLocale","locale","resolveQnaLocale","ActionPayloadSchema","z.object","z.string","z.unknown","ActionButtonSchema","z.enum","ButtonRowSchema","z.boolean","TextInputSchema","z.union","z.array","QuestionHeadingSchema","qnaCatalog","GengageQNA","BaseWidget","config","sku","ga.trackInit","newSku","wrappers","loading","requestId","fetchStart","streamStartEvent","launcherReq","pageType","streamDoneEvent","widgetHistorySnapshotEvent","cfgPlaceholders","effectivePlaceholders","a","renderContext","fallbackSpec","nonEmptySpecs","ga.trackShow","err","dispatch","getGlobalErrorMessage","streamErrorEvent","fallbackPlaceholders","fallbackContext","i","dot","baseRegistry","mergeUISpecRegistry","defaultRender","inputSpec","inputContext","override","specs","elements","childIds","id","renderedInput","ga.trackSuggestedQuestion","createQNAWidget"],"mappings":"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"}
1
+ {"version":3,"file":"qna.js","sources":["../src/qna/api.ts","../src/qna/components/ButtonRow.ts","../src/qna/components/TextInput.ts","../src/qna/components/renderUISpec.ts","../src/qna/locales/tr.ts","../src/qna/locales/en.ts","../src/qna/locales/index.ts","../src/qna/catalog.ts","../src/qna/index.ts"],"sourcesContent":["import { consumeStream } from '../common/streaming.js';\nimport { buildChatEndpointUrl } from '../common/api-paths.js';\nimport { adaptBackendEvent } from '../common/protocol-adapter.js';\nimport type { StreamEvent, UISpec } from '../common/types.js';\nimport type { ChatTransportConfig } from '../common/api-paths.js';\n\nexport interface LauncherActionRequest {\n account_id: string;\n session_id: string;\n correlation_id: string;\n sku: string;\n page_type?: string;\n locale?: string;\n output_language?: string;\n mode?: string;\n}\n\nexport interface LauncherActionResult {\n uiSpecs: UISpec[];\n actions: Array<{ title: string; type: string; payload?: unknown }>;\n}\n\nexport async function fetchLauncherActions(\n request: LauncherActionRequest,\n transport: ChatTransportConfig,\n signal?: AbortSignal,\n): Promise<LauncherActionResult> {\n const url = buildChatEndpointUrl('launcher_action', transport);\n const result: LauncherActionResult = { uiSpecs: [], actions: [] };\n\n const fetchInit: RequestInit = {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(request),\n };\n if (signal !== undefined) fetchInit.signal = signal;\n const response = await fetch(url, fetchInit);\n\n if (!response.ok) {\n throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n }\n\n const streamOpts: import('../common/streaming.js').StreamOptions = {\n onEvent: (event: StreamEvent) => {\n const normalized = adaptBackendEvent(event as unknown as Record<string, unknown>);\n\n if (!normalized) return;\n\n if (normalized.type === 'ui_spec') {\n result.uiSpecs.push(normalized.spec);\n }\n\n if (normalized.type === 'ui_spec' && normalized.spec.elements) {\n for (const el of Object.values(normalized.spec.elements)) {\n if (el.type === 'ActionButton' && el.props?.['action']) {\n const action = el.props['action'] as { title: string; type: string; payload?: unknown };\n result.actions.push(action);\n }\n }\n }\n },\n };\n if (signal !== undefined) streamOpts.signal = signal;\n\n await consumeStream(response, streamOpts);\n\n return result;\n}\n","import type { ActionPayload } from '../../common/types.js';\n\nexport interface ButtonRowOptions {\n actions: Array<{ title: string; type: string; payload?: unknown }>;\n onAction: (action: ActionPayload) => void;\n ctaText?: string;\n defaultCtaText?: string;\n onOpenChat?: () => void;\n orientation?: 'horizontal' | 'vertical';\n quickQuestionsAriaLabel?: string;\n}\n\nexport function renderButtonRow(options: ButtonRowOptions): HTMLElement {\n const container = document.createElement('div');\n container.className = 'gengage-qna-buttons';\n container.setAttribute('role', 'group');\n container.setAttribute('aria-label', options.quickQuestionsAriaLabel ?? 'Quick questions');\n\n if (options.orientation === 'vertical') {\n container.style.flexDirection = 'column';\n }\n\n for (const action of options.actions) {\n const button = document.createElement('button');\n button.className = 'gengage-qna-button';\n button.textContent = action.title;\n button.type = 'button';\n button.addEventListener('click', () => {\n const actionPayload: ActionPayload = {\n title: action.title,\n type: action.type,\n };\n if (action.payload !== undefined) actionPayload.payload = action.payload;\n options.onAction(actionPayload);\n });\n container.appendChild(button);\n }\n\n if (options.ctaText || options.onOpenChat) {\n const cta = document.createElement('button');\n cta.className = 'gengage-qna-cta';\n cta.textContent = options.ctaText ?? options.defaultCtaText ?? 'Ask something else';\n cta.type = 'button';\n cta.addEventListener('click', () => {\n options.onOpenChat?.();\n });\n container.appendChild(cta);\n }\n\n return container;\n}\n","import type { ActionPayload } from '../../common/types.js';\n\nexport interface TextInputOptions {\n placeholders?: string | string[];\n ctaLabel?: string;\n defaultInputPlaceholder?: string;\n askQuestionAriaLabel?: string;\n sendButtonText?: string;\n sendQuestionAriaLabel?: string;\n onSubmit: (action: ActionPayload) => void;\n}\n\nexport function renderTextInput(options: TextInputOptions): HTMLElement {\n const container = document.createElement('div');\n container.className = 'gengage-qna-input-wrapper';\n\n const input = document.createElement('input');\n input.type = 'text';\n input.className = 'gengage-qna-input';\n input.setAttribute('aria-label', options.askQuestionAriaLabel ?? 'Ask a question');\n\n // Rotating placeholders\n const placeholders = Array.isArray(options.placeholders)\n ? options.placeholders\n : options.placeholders\n ? [options.placeholders]\n : [options.defaultInputPlaceholder ?? 'Ask a question...'];\n\n let placeholderIndex = 0;\n input.placeholder = placeholders[0] ?? '';\n\n let rotateTimer: ReturnType<typeof setInterval> | null = null;\n let fadeTimer: ReturnType<typeof setTimeout> | null = null;\n if (placeholders.length > 1) {\n rotateTimer = setInterval(() => {\n input.classList.add('gengage-qna-input--fade');\n fadeTimer = setTimeout(() => {\n placeholderIndex = (placeholderIndex + 1) % placeholders.length;\n input.placeholder = placeholders[placeholderIndex] ?? '';\n input.classList.remove('gengage-qna-input--fade');\n }, 180);\n }, 3000);\n }\n\n const sendBtn = document.createElement('button');\n sendBtn.className = 'gengage-qna-send';\n sendBtn.type = 'button';\n sendBtn.textContent = options.ctaLabel ?? options.sendButtonText ?? 'Ask';\n sendBtn.setAttribute('aria-label', options.sendQuestionAriaLabel ?? 'Send question');\n\n const submit = () => {\n const text = input.value.trim();\n if (!text) return;\n options.onSubmit({\n title: text,\n type: 'user_message',\n // Align with chat composer: user_message payload is plain string.\n payload: text,\n });\n input.value = '';\n };\n\n sendBtn.addEventListener('click', submit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Enter') {\n e.preventDefault();\n submit();\n }\n });\n\n container.appendChild(input);\n container.appendChild(sendBtn);\n\n // Cleanup method\n (container as HTMLElement & { _cleanup?: () => void })._cleanup = () => {\n if (rotateTimer) clearInterval(rotateTimer);\n if (fadeTimer) clearTimeout(fadeTimer);\n };\n\n return container;\n}\n","import { renderUISpecWithRegistry } from '../../common/renderer/index.js';\nimport type { UISpecDomRegistry, UISpecDomUnknownRenderer } from '../../common/renderer/index.js';\nimport type { UISpec, UIElement, ActionPayload } from '../../common/types.js';\nimport type { QNAUISpecRenderContext } from '../types.js';\nimport { renderButtonRow } from './ButtonRow.js';\nimport { renderTextInput } from './TextInput.js';\n\nexport type QNAUISpecRegistry = UISpecDomRegistry<QNAUISpecRenderContext>;\n\nfunction toActionPayload(raw: unknown, labelFallback?: string): ActionPayload | null {\n if (!raw || typeof raw !== 'object') return null;\n const obj = raw as Record<string, unknown>;\n const type = obj['type'];\n if (typeof type !== 'string' || type.length === 0) return null;\n\n const titleRaw = obj['title'];\n const payload = obj['payload'];\n const title = typeof titleRaw === 'string' && titleRaw.length > 0 ? titleRaw : labelFallback;\n if (!title) return null;\n\n const action: ActionPayload = { title, type };\n if (payload !== undefined) action.payload = payload;\n return action;\n}\n\nfunction actionPayloadToRowAction(action: ActionPayload): { title: string; type: string; payload?: unknown } {\n const rowAction: { title: string; type: string; payload?: unknown } = {\n title: action.title,\n type: action.type,\n };\n if (action.payload !== undefined) rowAction.payload = action.payload;\n return rowAction;\n}\n\nfunction collectButtonRowActions(element: UIElement, spec: UISpec): ActionPayload[] {\n const actions: ActionPayload[] = [];\n\n const directActions = element.props?.['actions'];\n if (Array.isArray(directActions)) {\n for (const rawAction of directActions) {\n const normalized = toActionPayload(rawAction);\n if (normalized) actions.push(normalized);\n }\n }\n\n const directButtons = element.props?.['buttons'];\n if (Array.isArray(directButtons)) {\n for (const rawButton of directButtons) {\n if (!rawButton || typeof rawButton !== 'object') continue;\n const obj = rawButton as Record<string, unknown>;\n const label = typeof obj['label'] === 'string' ? obj['label'] : undefined;\n const action = toActionPayload(obj['action'], label);\n if (action) actions.push(action);\n }\n }\n\n if (element.children) {\n for (const childId of element.children) {\n const child = spec.elements[childId];\n if (!child || child.type !== 'ActionButton') continue;\n const label = typeof child.props?.['label'] === 'string' ? child.props['label'] : undefined;\n const action = toActionPayload(child.props?.['action'], label);\n if (action) actions.push(action);\n }\n }\n\n // Deduplicate by title — backend may provide same actions via props and children\n const seen = new Set<string>();\n return actions.filter((a) => {\n if (seen.has(a.title)) return false;\n seen.add(a.title);\n return true;\n });\n}\n\nconst DEFAULT_QNA_UI_SPEC_REGISTRY: QNAUISpecRegistry = {\n ButtonRow: ({ element, spec, context }) => {\n const rowActions = collectButtonRowActions(element, spec).map(actionPayloadToRowAction);\n const orientation = element.props?.['orientation'];\n\n const options: import('./ButtonRow.js').ButtonRowOptions = {\n actions: rowActions,\n onAction: context.onAction,\n defaultCtaText: context.i18n.defaultCtaText,\n quickQuestionsAriaLabel: context.i18n.quickQuestionsAriaLabel,\n };\n if (context.onOpenChat !== undefined) options.onOpenChat = context.onOpenChat;\n if (context.ctaText !== undefined) options.ctaText = context.ctaText;\n if (orientation === 'horizontal' || orientation === 'vertical') options.orientation = orientation;\n return renderButtonRow(options);\n },\n\n ActionButtons: ({ element, spec, context }) => {\n const rowActions = collectButtonRowActions(element, spec).map(actionPayloadToRowAction);\n const options: import('./ButtonRow.js').ButtonRowOptions = {\n actions: rowActions,\n onAction: context.onAction,\n defaultCtaText: context.i18n.defaultCtaText,\n quickQuestionsAriaLabel: context.i18n.quickQuestionsAriaLabel,\n };\n if (context.onOpenChat !== undefined) options.onOpenChat = context.onOpenChat;\n if (context.ctaText !== undefined) options.ctaText = context.ctaText;\n return renderButtonRow(options);\n },\n\n ActionButton: ({ element, context }) => {\n const button = document.createElement('button');\n button.className = 'gengage-qna-button';\n button.type = 'button';\n\n const label = element.props?.['label'];\n if (typeof label === 'string') {\n button.textContent = label;\n } else {\n button.textContent = context.i18n.defaultCtaText;\n }\n\n const action = toActionPayload(element.props?.['action'], typeof label === 'string' ? label : undefined);\n if (action) {\n button.addEventListener('click', () => context.onAction(action));\n }\n return button;\n },\n\n TextInput: ({ element, context }) => {\n const placeholder = element.props?.['placeholder'];\n const placeholders =\n typeof placeholder === 'string' || Array.isArray(placeholder) ? placeholder : context.inputPlaceholder;\n // Keep input submit label independent from the quick-question CTA text.\n const ctaLabel = typeof element.props?.['ctaLabel'] === 'string' ? element.props['ctaLabel'] : undefined;\n\n const options: import('./TextInput.js').TextInputOptions = {\n onSubmit: context.onAction,\n askQuestionAriaLabel: context.i18n.askQuestionAriaLabel,\n defaultInputPlaceholder: context.i18n.defaultInputPlaceholder,\n sendButtonText: context.i18n.sendButton,\n sendQuestionAriaLabel: context.i18n.sendQuestionAriaLabel,\n };\n if (placeholders !== undefined) options.placeholders = placeholders;\n if (ctaLabel !== undefined) options.ctaLabel = ctaLabel;\n return renderTextInput(options);\n },\n\n QuestionHeading: ({ element }) => {\n const heading = document.createElement('h3');\n heading.className = 'gengage-qna-heading';\n const text = element.props?.['text'];\n heading.textContent = typeof text === 'string' ? text : '';\n return heading;\n },\n\n // ProductCard: no-op — QNA renders only question pills + text input.\n // Backend may send ProductCard but QNA intentionally skips it;\n // the product is shown once chat opens via openWithAction().\n ProductCard: () => null,\n};\n\nexport const defaultQnaUnknownUISpecRenderer: UISpecDomUnknownRenderer<QNAUISpecRenderContext> = ({\n element,\n renderElement,\n}) => {\n if (import.meta.env?.DEV) {\n console.warn(`[gengage:qna] Unknown ui_spec component type: ${element.type}`);\n }\n if (!element.children || element.children.length === 0) {\n return null;\n }\n const wrapper = document.createElement('div');\n for (const childId of element.children) {\n const rendered = renderElement(childId);\n if (rendered) wrapper.appendChild(rendered);\n }\n return wrapper;\n};\n\nexport function createDefaultQnaUISpecRegistry(): QNAUISpecRegistry {\n return { ...DEFAULT_QNA_UI_SPEC_REGISTRY };\n}\n\nexport function renderQnaUISpec(\n spec: UISpec,\n context: QNAUISpecRenderContext,\n registry = DEFAULT_QNA_UI_SPEC_REGISTRY,\n unknownRenderer: UISpecDomUnknownRenderer<QNAUISpecRenderContext> = defaultQnaUnknownUISpecRenderer,\n): HTMLElement {\n return renderUISpecWithRegistry({\n spec,\n context,\n registry,\n containerClassName: 'gengage-qna-uispec',\n unknownRenderer,\n });\n}\n","import type { QNAI18n } from '../types.js';\n\nexport const QNA_I18N_TR: QNAI18n = {\n quickQuestionsAriaLabel: 'Hızlı sorular',\n askQuestionAriaLabel: 'Soru sorun',\n defaultInputPlaceholder: 'Bir soru sorun...',\n sendButton: 'Sor',\n sendQuestionAriaLabel: 'Soruyu gönder',\n defaultCtaText: 'Başka bir şey sor',\n};\n","import type { QNAI18n } from '../types.js';\n\nexport const QNA_I18N_EN: QNAI18n = {\n quickQuestionsAriaLabel: 'Quick questions',\n askQuestionAriaLabel: 'Ask a question',\n defaultInputPlaceholder: 'Ask a question...',\n sendButton: 'Ask',\n sendQuestionAriaLabel: 'Send question',\n defaultCtaText: 'Ask something else',\n};\n","import type { QNAI18n } from '../types.js';\nimport { QNA_I18N_TR } from './tr.js';\nimport { QNA_I18N_EN } from './en.js';\n\nfunction normalizeLocale(locale?: string): string {\n if (!locale) return 'tr';\n return locale.toLowerCase().split('-')[0] ?? 'tr';\n}\n\nexport function resolveQnaLocale(locale?: string): QNAI18n {\n switch (normalizeLocale(locale)) {\n case 'en':\n return QNA_I18N_EN;\n default:\n return QNA_I18N_TR;\n }\n}\n\nexport { QNA_I18N_TR, QNA_I18N_EN };\n","/**\n * QNA widget — json-render catalog definition.\n *\n * Backend endpoint: POST /chat/launcher_action\n * Response format: NDJSON stream, each line a StreamEvent.\n *\n * The backend returns a `ui_spec` event whose elements are typed to the\n * component names defined below. The frontend renders them via ./registry.\n *\n * Customising:\n * Fork this repo and replace ./registry implementations with your own.\n * The catalog schema (component names + prop types) is the API contract\n * with the backend — do not rename components without a backend release.\n */\n\nimport { z } from 'zod';\n\nconst ActionPayloadSchema = z.object({\n title: z.string(),\n type: z.string(),\n payload: z.unknown().optional(),\n});\n\nexport const ActionButtonSchema = z.object({\n label: z.string(),\n action: ActionPayloadSchema,\n variant: z.enum(['primary', 'outline', 'ghost']).optional(),\n});\n\nexport const ButtonRowSchema = z.object({\n orientation: z.enum(['horizontal', 'vertical']).optional(),\n wrap: z.boolean().optional(),\n});\n\nexport const TextInputSchema = z.object({\n placeholder: z.union([z.string(), z.array(z.string())]).optional(),\n ctaLabel: z.string().optional(),\n});\n\nexport const QuestionHeadingSchema = z.object({\n text: z.string(),\n});\n\nexport const qnaCatalog = {\n components: {\n ActionButton: {\n schema: ActionButtonSchema,\n description: 'A single clickable QNA action button.',\n },\n ButtonRow: {\n schema: ButtonRowSchema,\n description: 'Container for a group of QNA action buttons.',\n },\n TextInput: {\n schema: TextInputSchema,\n description: 'Free-text input with rotating placeholder and CTA button.',\n },\n QuestionHeading: {\n schema: QuestionHeadingSchema,\n description: 'A heading displayed above the QNA button group.',\n },\n },\n} as const;\n\nexport type QNACatalog = typeof qnaCatalog;\nexport type QNAComponentName = keyof QNACatalog['components'];\n","/**\n * QNA widget -- public entry point.\n *\n * Renders contextual action buttons (and an optional free-text input) on\n * product or content pages. When a user taps a button, the widget opens\n * the Chat widget with that action pre-loaded.\n *\n * Backend: POST /chat/launcher_action\n * Protocol: NDJSON stream -> json-render UISpec -> ActionButton / ButtonRow / TextInput\n */\n\nimport type { ActionPayload, PageContext, UISpec, UIElement } from '../common/types.js';\nimport type { ChatTransportConfig } from '../common/api-paths.js';\nimport type { UISpecRenderHelpers } from '../common/renderer/index.js';\nimport { mergeUISpecRegistry } from '../common/renderer/index.js';\nimport { BaseWidget } from '../common/widget-base.js';\nimport { dispatch } from '../common/events.js';\nimport { getGlobalErrorMessage } from '../common/global-error-toast.js';\nimport {\n streamStartEvent,\n streamDoneEvent,\n streamErrorEvent,\n widgetHistorySnapshotEvent,\n} from '../common/analytics-events.js';\nimport { fetchLauncherActions } from './api.js';\nimport * as ga from '../common/ga-datalayer.js';\nimport {\n createDefaultQnaUISpecRegistry,\n defaultQnaUnknownUISpecRenderer,\n renderQnaUISpec,\n} from './components/renderUISpec.js';\nimport type { QNAWidgetConfig, QNAI18n, QNAUISpecRenderContext } from './types.js';\nimport { QNA_I18N_TR, resolveQnaLocale } from './locales/index.js';\n\n// Inline CSS import marker - Vite will bundle this\nimport './components/qna.css';\n\n/**\n * Contextual Q&A action buttons for product pages.\n * Renders quick-action buttons that open the chat widget with a pre-built query.\n *\n * @example\n * ```ts\n * import { GengageQNA, wireQNAToChat, bootstrapSession } from '@gengage/assistant-fe';\n *\n * const qna = new GengageQNA();\n * await qna.init({\n * accountId: 'mystore',\n * middlewareUrl: 'https://chat.gengage.ai',\n * mountTarget: '#qna-section',\n * pageContext: { pageType: 'pdp', sku: '12345' },\n * session: { sessionId: bootstrapSession() },\n * });\n * wireQNAToChat(); // Wire button clicks to chat.openWithAction()\n * ```\n */\nexport class GengageQNA extends BaseWidget<QNAWidgetConfig> {\n private _abortController: AbortController | null = null;\n private _debounceTimer: ReturnType<typeof setTimeout> | null = null;\n private _contentEl: HTMLElement | null = null;\n private _lastSku: string | undefined;\n private _i18n: QNAI18n = QNA_I18N_TR;\n\n protected async onInit(config: QNAWidgetConfig): Promise<void> {\n this._i18n = this._resolveI18n(config);\n\n this._contentEl = document.createElement('div');\n this._contentEl.className = 'gengage-qna-container';\n this.root.appendChild(this._contentEl);\n\n const sku = config.pageContext?.sku;\n if (sku) {\n this._lastSku = sku;\n await this._fetchAndRender(sku);\n }\n\n this.isVisible = true;\n ga.trackInit('qna');\n }\n\n protected onUpdate(context: Partial<PageContext>): void {\n const newSku = context.sku;\n if (!newSku || newSku === this._lastSku) return;\n\n // Debounce rapid SPA navigations\n if (this._debounceTimer) clearTimeout(this._debounceTimer);\n this._debounceTimer = setTimeout(() => {\n this._debounceTimer = null;\n this._lastSku = newSku;\n void this._fetchAndRender(newSku);\n }, 50);\n }\n\n protected onShow(): void {\n if (this._contentEl) {\n this._contentEl.style.opacity = '0';\n this._contentEl.style.transition = 'opacity 0.2s ease-in';\n requestAnimationFrame(() => {\n if (this._contentEl) this._contentEl.style.opacity = '1';\n });\n }\n }\n\n protected onHide(): void {\n // Preserve fetched data for re-show\n }\n\n protected onDestroy(): void {\n this._abort();\n if (this._debounceTimer) {\n clearTimeout(this._debounceTimer);\n this._debounceTimer = null;\n }\n if (this._contentEl) {\n this._cleanupTextInputTimers();\n this._contentEl.remove();\n this._contentEl = null;\n }\n }\n\n // ---------------------------------------------------------------------------\n // Internal\n // ---------------------------------------------------------------------------\n\n private _abort(): void {\n this._abortController?.abort();\n this._abortController = null;\n }\n\n /** Clean up TextInput placeholder rotation timers to prevent interval leaks. */\n private _cleanupTextInputTimers(): void {\n if (!this._contentEl) return;\n const wrappers = this._contentEl.querySelectorAll('.gengage-qna-input-wrapper');\n for (const wrapper of wrappers) {\n (wrapper as HTMLElement & { _cleanup?: () => void })._cleanup?.();\n }\n }\n\n private async _fetchAndRender(sku: string): Promise<void> {\n this._abort();\n this._abortController = new AbortController();\n\n if (!this._contentEl) return;\n // Clean up TextInput timers before clearing DOM to prevent interval leaks\n this._cleanupTextInputTimers();\n this._contentEl.innerHTML = '';\n\n // Show loading dots\n const loading = this._createLoadingIndicator();\n this._contentEl.appendChild(loading);\n\n const transport: ChatTransportConfig = {\n middlewareUrl: this.config.middlewareUrl,\n };\n\n const requestId = crypto.randomUUID();\n const fetchStart = Date.now();\n\n this.track(\n streamStartEvent(this.analyticsContext(), {\n endpoint: 'launcher_action',\n request_id: requestId,\n widget: 'qna',\n }),\n );\n\n try {\n const launcherReq: import('./api.js').LauncherActionRequest = {\n account_id: this.config.accountId,\n session_id: this.config.session?.sessionId ?? '',\n correlation_id: this.config.session?.sessionId ?? '',\n sku,\n locale: this.config.locale ?? 'tr',\n };\n const pageType = this.config.pageContext?.pageType;\n if (pageType !== undefined) launcherReq.page_type = pageType;\n\n const result = await fetchLauncherActions(launcherReq, transport, this._abortController.signal);\n\n this.track(\n streamDoneEvent(this.analyticsContext(), {\n request_id: requestId,\n latency_ms: Date.now() - fetchStart,\n chunk_count: result.actions.length,\n widget: 'qna',\n }),\n );\n\n this.track(\n widgetHistorySnapshotEvent(this.analyticsContext(), {\n message_count: result.actions.length,\n history_ref: requestId,\n redaction_level: 'none',\n widget: 'qna',\n }),\n );\n\n if (!this._contentEl) return;\n this._contentEl.innerHTML = '';\n\n const hasQuestionHeading = this._specIncludesType(result.uiSpecs, 'QuestionHeading');\n\n // Render heading if configured and backend didn't provide one\n if (!hasQuestionHeading && this.config.showStaticQuestion && this.config.staticQuestionText) {\n const heading = document.createElement('h3');\n heading.className = 'gengage-qna-heading';\n heading.textContent = this.config.staticQuestionText;\n this._contentEl.appendChild(heading);\n }\n\n const cfgPlaceholders = this.config.inputPlaceholder;\n let effectivePlaceholders: string | string[];\n if (cfgPlaceholders !== true) {\n effectivePlaceholders = cfgPlaceholders ?? this._i18n.defaultInputPlaceholder;\n } else if (result.actions.length > 0) {\n // Only use question-like actions as rotating placeholders\n const filtered = result.actions\n .filter((a) => a.type === 'user_message' || a.title.includes('?'))\n .map((a) => a.title);\n effectivePlaceholders = filtered.length > 0 ? filtered : this._i18n.defaultInputPlaceholder;\n } else {\n effectivePlaceholders = this._i18n.defaultInputPlaceholder;\n }\n\n const renderContext: QNAUISpecRenderContext = {\n onAction: this._actionHandler,\n i18n: this._i18n,\n };\n if (!this.config.hideButtonRowCta) {\n renderContext.onOpenChat = this._openChatHandler;\n if (this.config.ctaText !== undefined) renderContext.ctaText = this.config.ctaText;\n }\n if (effectivePlaceholders !== undefined) renderContext.inputPlaceholder = effectivePlaceholders;\n\n const fallbackSpec = this._buildFallbackActionsSpec(result.actions);\n const specsToRender = result.uiSpecs.length > 0 ? result.uiSpecs : [fallbackSpec];\n const nonEmptySpecs = specsToRender.filter((spec) => Object.keys(spec.elements).length > 0);\n\n for (const spec of nonEmptySpecs) {\n const rendered = this._renderUISpec(spec, renderContext);\n this._contentEl.appendChild(rendered);\n }\n\n if (nonEmptySpecs.length > 0) {\n ga.trackShow('qna');\n }\n\n const shouldRenderStandaloneInput = !this._specIncludesType(nonEmptySpecs, 'TextInput');\n if (shouldRenderStandaloneInput) {\n this._appendStandaloneInput(renderContext, effectivePlaceholders);\n }\n } catch (err) {\n if (err instanceof DOMException && err.name === 'AbortError') return;\n\n dispatch('gengage:global:error', {\n source: 'qna',\n code: 'FETCH_ERROR',\n message: getGlobalErrorMessage(this.config.locale),\n });\n\n this.track(\n streamErrorEvent(this.analyticsContext(), {\n request_id: requestId,\n error_code: 'FETCH_ERROR',\n error_message: err instanceof Error ? err.message : String(err),\n widget: 'qna',\n }),\n );\n\n // Keep QNA usable during backend hiccups: render at least free-text input.\n if (this._contentEl) {\n this._cleanupTextInputTimers();\n this._contentEl.innerHTML = '';\n const fallbackPlaceholders =\n this.config.inputPlaceholder === true\n ? this._i18n.defaultInputPlaceholder\n : (this.config.inputPlaceholder ?? this._i18n.defaultInputPlaceholder);\n const fallbackContext: QNAUISpecRenderContext = {\n onAction: this._actionHandler,\n i18n: this._i18n,\n onOpenChat: this._openChatHandler,\n };\n if (this.config.ctaText !== undefined) fallbackContext.ctaText = this.config.ctaText;\n this._appendStandaloneInput(fallbackContext, fallbackPlaceholders);\n }\n\n if (import.meta.env?.DEV) {\n console.error('[gengage:qna] Failed to fetch launcher actions:', err);\n }\n }\n }\n\n private _createLoadingIndicator(): HTMLElement {\n const el = document.createElement('div');\n el.className = 'gengage-qna-loading';\n for (let i = 0; i < 3; i++) {\n const dot = document.createElement('div');\n dot.className = 'gengage-qna-loading-dot';\n el.appendChild(dot);\n }\n return el;\n }\n\n private _resolveI18n(config: QNAWidgetConfig): QNAI18n {\n const base = resolveQnaLocale(config.locale);\n return { ...base, ...config.i18n };\n }\n\n private _resolveUISpecRegistry() {\n const baseRegistry = createDefaultQnaUISpecRegistry();\n return mergeUISpecRegistry(baseRegistry, this.config.renderer?.registry);\n }\n\n private _renderUISpec(spec: UISpec, context: QNAUISpecRenderContext): HTMLElement {\n const registry = this._resolveUISpecRegistry();\n const unknownRenderer = this.config.renderer?.unknownRenderer ?? defaultQnaUnknownUISpecRenderer;\n const defaultRender = (inputSpec: UISpec, inputContext: QNAUISpecRenderContext) =>\n renderQnaUISpec(inputSpec, inputContext, registry, unknownRenderer);\n\n const override = this.config.renderer?.renderUISpec;\n if (!override) return defaultRender(spec, context);\n\n const helpers: UISpecRenderHelpers<QNAUISpecRenderContext> = {\n registry,\n unknownRenderer,\n defaultRender,\n };\n return override(spec, context, helpers);\n }\n\n private _specIncludesType(specs: UISpec[], type: string): boolean {\n for (const spec of specs) {\n for (const element of Object.values(spec.elements)) {\n if (element.type === type) return true;\n }\n }\n return false;\n }\n\n private _buildFallbackActionsSpec(actions: Array<{ title: string; type: string; payload?: unknown }>): UISpec {\n if (actions.length === 0) {\n return { root: 'root', elements: {} };\n }\n\n const elements: Record<string, UIElement> = {};\n const childIds: string[] = [];\n for (let i = 0; i < actions.length; i++) {\n const action = actions[i]!;\n const id = `action-${i}`;\n childIds.push(id);\n elements[id] = {\n type: 'ActionButton',\n props: {\n label: action.title,\n action: {\n title: action.title,\n type: action.type,\n payload: action.payload,\n },\n },\n };\n }\n elements['root'] = {\n type: 'ButtonRow',\n children: childIds,\n };\n return {\n root: 'root',\n elements,\n };\n }\n\n private _appendStandaloneInput(context: QNAUISpecRenderContext, placeholder?: string | string[]): void {\n if (!this._contentEl) return;\n const inputSpec: UISpec = {\n root: 'root',\n elements: {\n root: {\n type: 'TextInput',\n props: {\n placeholder,\n },\n },\n },\n };\n const renderedInput = this._renderUISpec(inputSpec, context);\n this._contentEl.appendChild(renderedInput);\n }\n\n private _handleAction(action: ActionPayload): void {\n ga.trackSuggestedQuestion(action.title, action.type);\n this.config.onActionSelected?.(action);\n dispatch('gengage:qna:action', action);\n }\n\n private _handleOpenChat(): void {\n // Couple CTA with the inline QNA text input when available.\n const input = this._contentEl?.querySelector<HTMLInputElement>('.gengage-qna-input');\n if (input) {\n input.focus();\n }\n this.config.onOpenChat?.();\n dispatch('gengage:qna:open-chat', {});\n }\n\n _actionHandler = this._handleAction.bind(this);\n _openChatHandler = this._handleOpenChat.bind(this);\n}\n\nexport function createQNAWidget(): GengageQNA {\n return new GengageQNA();\n}\n\nexport type { QNAWidgetConfig, QNAUIComponents, QNAI18n, QNAUISpecRenderContext, QNARendererConfig } from './types.js';\nexport {\n renderQnaUISpec,\n createDefaultQnaUISpecRegistry,\n defaultQnaUnknownUISpecRenderer,\n} from './components/renderUISpec.js';\nexport type { QNAUISpecRegistry } from './components/renderUISpec.js';\nexport { qnaCatalog } from './catalog.js';\nexport type { QNACatalog, QNAComponentName } from './catalog.js';\n"],"names":["fetchLauncherActions","request","transport","signal","url","buildChatEndpointUrl","result","fetchInit","response","streamOpts","event","normalized","adaptBackendEvent","el","action","consumeStream","renderButtonRow","options","container","button","actionPayload","cta","renderTextInput","input","placeholders","placeholderIndex","rotateTimer","fadeTimer","sendBtn","submit","text","e","toActionPayload","raw","labelFallback","obj","type","titleRaw","payload","title","actionPayloadToRowAction","rowAction","collectButtonRowActions","element","spec","actions","directActions","rawAction","directButtons","rawButton","label","childId","child","seen","DEFAULT_QNA_UI_SPEC_REGISTRY","context","rowActions","orientation","placeholder","ctaLabel","heading","defaultQnaUnknownUISpecRenderer","renderElement","wrapper","rendered","createDefaultQnaUISpecRegistry","renderQnaUISpec","registry","unknownRenderer","renderUISpecWithRegistry","QNA_I18N_TR","QNA_I18N_EN","normalizeLocale","locale","resolveQnaLocale","ActionPayloadSchema","z.object","z.string","z.unknown","ActionButtonSchema","z.enum","ButtonRowSchema","z.boolean","TextInputSchema","z.union","z.array","QuestionHeadingSchema","qnaCatalog","GengageQNA","BaseWidget","config","sku","ga.trackInit","newSku","wrappers","loading","requestId","fetchStart","streamStartEvent","launcherReq","pageType","streamDoneEvent","widgetHistorySnapshotEvent","cfgPlaceholders","effectivePlaceholders","filtered","a","renderContext","fallbackSpec","nonEmptySpecs","ga.trackShow","err","dispatch","getGlobalErrorMessage","streamErrorEvent","fallbackPlaceholders","fallbackContext","i","dot","baseRegistry","mergeUISpecRegistry","defaultRender","inputSpec","inputContext","override","specs","elements","childIds","id","renderedInput","ga.trackSuggestedQuestion","createQNAWidget"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ "use strict";const y={currencySymbol:"TL",currencyPosition:"suffix",thousandsSeparator:".",decimalSeparator:",",alwaysShowDecimals:!0};function x(t,d){const c=Number(t);if(!Number.isFinite(c)||c<0)return t;const e={...y,...d},a=c%1!==0||e.alwaysShowDecimals?c.toFixed(2):c.toFixed(0),s=a.indexOf("."),n=s===-1?a:a.slice(0,s),o=s===-1?void 0:a.slice(s+1),l=n.replace(/\B(?=(\d{3})+(?!\d))/g,e.thousandsSeparator);let i;return o!==void 0?i=`${l}${e.decimalSeparator}${o}`:i=l,e.currencySymbol?e.currencyPosition==="prefix"?`${e.currencySymbol}${i}`:`${i} ${e.currencySymbol}`:i}function b(t){return Number.isFinite(t)?Math.max(0,Math.min(5,t)):0}function S(t){return Number.isFinite(t)?Math.max(0,Math.min(100,Math.round(t))):0}function C(t){const d=b(t),c=Math.floor(d),e=d-c>=.5,u=5-c-(e?1:0),a=document.createElement("span");if(a.className="gengage-star-rating",a.setAttribute("role","img"),a.setAttribute("aria-label",`${d.toFixed(1)} out of 5 stars`),c>0&&a.appendChild(document.createTextNode("★".repeat(c))),e){const s=document.createElement("span");s.className="gengage-star-half",s.textContent="☆";const n=document.createElement("span");n.textContent="★",s.appendChild(n),a.appendChild(s)}return u>0&&a.appendChild(document.createTextNode("☆".repeat(u))),a}function E(t){t.addEventListener("error",()=>{t.style.display="none"},{once:!0})}function M(t){const d=t.min??1,c=t.max??99,e=Math.min(d,c),u=Math.max(d,c),a=Math.max(e,Math.min(u,t.initial??e)),s=t.compact??!1;let n=a;const o=document.createElement("div");o.className=`gengage-qty-stepper${s?" gengage-qty-stepper--compact":""}`;const l=document.createElement("button");l.className="gengage-qty-btn",l.type="button",l.textContent=t.decreaseSymbol??"−",l.setAttribute("aria-label",t.decreaseLabel??"Decrease");const i=document.createElement("span");i.className="gengage-qty-value",i.textContent=String(n),i.setAttribute("aria-live","polite"),i.setAttribute("aria-atomic","true");const m=document.createElement("button");m.className="gengage-qty-btn",m.type="button",m.textContent=t.increaseSymbol??"+",m.setAttribute("aria-label",t.increaseLabel??"Increase");const r=document.createElement("button");r.className="gengage-qty-submit",r.type="button";const f='<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="9" cy="21" r="1"/><circle cx="20" cy="21" r="1"/><path d="M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6"/></svg>';s?(r.innerHTML=t.submitIcon??f,r.title=t.label??"Add to Cart"):r.textContent=t.label??"Add to Cart";function g(){l.disabled=n<=e,m.disabled=n>=u}return l.addEventListener("click",p=>{p.stopPropagation(),n>e&&(n--,i.textContent=String(n),g())}),m.addEventListener("click",p=>{p.stopPropagation(),n<u&&(n++,i.textContent=String(n),g())}),r.addEventListener("click",p=>{p.stopPropagation(),t.onSubmit(n);const h=r.innerHTML;r.textContent="✓",r.classList.add("gengage-qty-submit--success"),r.disabled=!0,setTimeout(()=>{r.innerHTML=h,r.classList.remove("gengage-qty-submit--success"),r.disabled=!1},1200)}),o.addEventListener("click",p=>{p.stopPropagation()}),g(),o.appendChild(l),o.appendChild(i),o.appendChild(m),o.appendChild(r),o}exports.addImageErrorHandler=E;exports.clampDiscount=S;exports.clampRating=b;exports.createQuantityStepper=M;exports.createStarRatingElement=C;exports.formatPrice=x;
2
+ //# sourceMappingURL=quantity-stepper-CQWgexMO.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"quantity-stepper-CQWgexMO.cjs","sources":["../src/common/price-formatter.ts","../src/common/product-utils.ts","../src/common/quantity-stepper.ts"],"sourcesContent":["/**\n * Configurable price formatting.\n *\n * Defaults to Turkish locale (dot thousands, comma decimal, TL suffix).\n * Configure via widget config `pricing` field for any locale/currency.\n */\n\nexport interface PriceFormatConfig {\n /** Currency symbol. Default: 'TL' */\n currencySymbol?: string;\n /** Where to place the symbol. Default: 'suffix' */\n currencyPosition?: 'prefix' | 'suffix';\n /** Separator between thousands. Default: '.' (Turkish) */\n thousandsSeparator?: string;\n /** Decimal point character. Default: ',' (Turkish) */\n decimalSeparator?: string;\n /** Whether to show decimal part for whole numbers. Default: true */\n alwaysShowDecimals?: boolean;\n}\n\nconst TURKISH_DEFAULTS: Required<PriceFormatConfig> = {\n currencySymbol: 'TL',\n currencyPosition: 'suffix',\n thousandsSeparator: '.',\n decimalSeparator: ',',\n alwaysShowDecimals: true,\n};\n\n/**\n * Formats a raw numeric price string into the configured locale format.\n *\n * Examples (default Turkish):\n * \"17990\" → \"17.990 TL\"\n * \"17990.5\" → \"17.990,50 TL\"\n *\n * Examples (GBP prefix):\n * \"17990\" with { currencySymbol: '£', currencyPosition: 'prefix', thousandsSeparator: ',', decimalSeparator: '.' }\n * → \"£17,990\"\n *\n * Returns the input as-is if it's not a valid number.\n */\nexport function formatPrice(raw: string, config?: PriceFormatConfig): string {\n const num = Number(raw);\n if (!Number.isFinite(num) || num < 0) return raw;\n\n const resolved = { ...TURKISH_DEFAULTS, ...config };\n\n const hasDecimals = num % 1 !== 0;\n const fixed = hasDecimals || resolved.alwaysShowDecimals ? num.toFixed(2) : num.toFixed(0);\n const dotIdx = fixed.indexOf('.');\n const intPart = dotIdx === -1 ? fixed : fixed.slice(0, dotIdx);\n const decPart = dotIdx === -1 ? undefined : fixed.slice(dotIdx + 1);\n\n // Add thousands separators to integer part\n const withSeparators = intPart.replace(/\\B(?=(\\d{3})+(?!\\d))/g, resolved.thousandsSeparator);\n\n let formatted: string;\n if (decPart !== undefined) {\n formatted = `${withSeparators}${resolved.decimalSeparator}${decPart}`;\n } else {\n formatted = withSeparators;\n }\n\n if (resolved.currencySymbol) {\n if (resolved.currencyPosition === 'prefix') {\n return `${resolved.currencySymbol}${formatted}`;\n }\n return `${formatted} ${resolved.currencySymbol}`;\n }\n\n return formatted;\n}\n","/**\n * Shared product rendering utilities.\n *\n * Extracted from chat/renderUISpec and simrel/ProductCard to eliminate\n * duplication and provide consistent behavior across all widgets.\n */\n\n/** Clamp a rating value to the 0–5 range. Returns 0 for NaN/non-finite. */\nexport function clampRating(value: number): number {\n if (!Number.isFinite(value)) return 0;\n return Math.max(0, Math.min(5, value));\n}\n\n/** Clamp a discount percentage to the 0–100 range, rounded to integer. Returns 0 for NaN/non-finite. */\nexport function clampDiscount(value: number): number {\n if (!Number.isFinite(value)) return 0;\n return Math.max(0, Math.min(100, Math.round(value)));\n}\n\n/**\n * Render a star rating string with full, half, and empty stars.\n *\n * @param rating - A numeric rating (will be clamped to 0–5).\n * @param halfStars - Whether to render half-star characters. Defaults to true.\n * @returns A string like \"★★★½☆\" or \"★★★☆☆\" (without half-stars).\n */\nexport function renderStarRating(rating: number, halfStars: boolean = true): string {\n const clamped = clampRating(rating);\n if (halfStars) {\n const full = Math.floor(clamped);\n const half = clamped - full >= 0.5 ? 1 : 0;\n const empty = 5 - full - half;\n return '\\u2605'.repeat(full) + (half ? '\\u00BD' : '') + '\\u2606'.repeat(empty);\n }\n const rounded = Math.round(clamped);\n return '\\u2605'.repeat(rounded) + '\\u2606'.repeat(5 - rounded);\n}\n\n/**\n * Create a star rating DOM element with proper half-filled star rendering.\n *\n * Uses a CSS-clipped full star overlaid on an empty star for the half-star,\n * giving a visually accurate half-filled appearance instead of the \"½\" character.\n *\n * @param rating - A numeric rating (will be clamped to 0–5).\n * @returns An HTMLSpanElement containing the star icons.\n */\nexport function createStarRatingElement(rating: number): HTMLSpanElement {\n const clamped = clampRating(rating);\n const full = Math.floor(clamped);\n const hasHalf = clamped - full >= 0.5;\n const empty = 5 - full - (hasHalf ? 1 : 0);\n\n const container = document.createElement('span');\n container.className = 'gengage-star-rating';\n container.setAttribute('role', 'img');\n container.setAttribute('aria-label', `${clamped.toFixed(1)} out of 5 stars`);\n\n if (full > 0) {\n container.appendChild(document.createTextNode('\\u2605'.repeat(full)));\n }\n\n if (hasHalf) {\n const halfStar = document.createElement('span');\n halfStar.className = 'gengage-star-half';\n halfStar.textContent = '\\u2606';\n const filled = document.createElement('span');\n filled.textContent = '\\u2605';\n halfStar.appendChild(filled);\n container.appendChild(halfStar);\n }\n\n if (empty > 0) {\n container.appendChild(document.createTextNode('\\u2606'.repeat(empty)));\n }\n\n return container;\n}\n\n/**\n * Attach a one-time error handler that hides the image on load failure.\n *\n * Works with any HTMLImageElement. Hides the element by setting\n * `display: none` so layout doesn't break from broken images.\n */\nexport function addImageErrorHandler(img: HTMLImageElement): void {\n img.addEventListener(\n 'error',\n () => {\n img.style.display = 'none';\n },\n { once: true },\n );\n}\n","export interface QuantityStepperOptions {\n min?: number | undefined;\n max?: number | undefined;\n initial?: number | undefined;\n label?: string | undefined;\n compact?: boolean | undefined;\n decreaseLabel?: string | undefined;\n increaseLabel?: string | undefined;\n /** Symbol for decrease button (default: '\\u2212' minus sign). */\n decreaseSymbol?: string | undefined;\n /** Symbol for increase button (default: '+'). */\n increaseSymbol?: string | undefined;\n /** Icon HTML for compact mode submit button (default: cart SVG icon). */\n submitIcon?: string | undefined;\n onSubmit: (quantity: number) => void;\n}\n\n/**\n * Creates a quantity stepper with [−] [value] [+] and a submit button.\n * Compact mode renders a cart icon button; full mode renders a labeled button.\n */\nexport function createQuantityStepper(options: QuantityStepperOptions): HTMLElement {\n const rawMin = options.min ?? 1;\n const rawMax = options.max ?? 99;\n // Ensure min <= max; swap if caller provided inverted range\n const min = Math.min(rawMin, rawMax);\n const max = Math.max(rawMin, rawMax);\n const initial = Math.max(min, Math.min(max, options.initial ?? min));\n const compact = options.compact ?? false;\n\n let quantity = initial;\n\n const container = document.createElement('div');\n container.className = `gengage-qty-stepper${compact ? ' gengage-qty-stepper--compact' : ''}`;\n\n const decBtn = document.createElement('button');\n decBtn.className = 'gengage-qty-btn';\n decBtn.type = 'button';\n decBtn.textContent = options.decreaseSymbol ?? '\\u2212'; // minus sign\n decBtn.setAttribute('aria-label', options.decreaseLabel ?? 'Decrease');\n\n const valueEl = document.createElement('span');\n valueEl.className = 'gengage-qty-value';\n valueEl.textContent = String(quantity);\n valueEl.setAttribute('aria-live', 'polite');\n valueEl.setAttribute('aria-atomic', 'true');\n\n const incBtn = document.createElement('button');\n incBtn.className = 'gengage-qty-btn';\n incBtn.type = 'button';\n incBtn.textContent = options.increaseSymbol ?? '+';\n incBtn.setAttribute('aria-label', options.increaseLabel ?? 'Increase');\n\n const submitBtn = document.createElement('button');\n submitBtn.className = 'gengage-qty-submit';\n submitBtn.type = 'button';\n\n const defaultCartSvg =\n '<svg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><circle cx=\"9\" cy=\"21\" r=\"1\"/><circle cx=\"20\" cy=\"21\" r=\"1\"/><path d=\"M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6\"/></svg>';\n\n if (compact) {\n submitBtn.innerHTML = options.submitIcon ?? defaultCartSvg;\n submitBtn.title = options.label ?? 'Add to Cart';\n } else {\n submitBtn.textContent = options.label ?? 'Add to Cart';\n }\n\n function updateButtons(): void {\n decBtn.disabled = quantity <= min;\n incBtn.disabled = quantity >= max;\n }\n\n decBtn.addEventListener('click', (e) => {\n e.stopPropagation();\n if (quantity > min) {\n quantity--;\n valueEl.textContent = String(quantity);\n updateButtons();\n }\n });\n\n incBtn.addEventListener('click', (e) => {\n e.stopPropagation();\n if (quantity < max) {\n quantity++;\n valueEl.textContent = String(quantity);\n updateButtons();\n }\n });\n\n submitBtn.addEventListener('click', (e) => {\n e.stopPropagation();\n options.onSubmit(quantity);\n // Brief visual feedback: show checkmark then revert\n const original = submitBtn.innerHTML;\n submitBtn.textContent = '\\u2713'; // checkmark\n submitBtn.classList.add('gengage-qty-submit--success');\n submitBtn.disabled = true;\n setTimeout(() => {\n submitBtn.innerHTML = original;\n submitBtn.classList.remove('gengage-qty-submit--success');\n submitBtn.disabled = false;\n }, 1200);\n });\n\n // Prevent card click when interacting with stepper\n container.addEventListener('click', (e) => {\n e.stopPropagation();\n });\n\n updateButtons();\n\n container.appendChild(decBtn);\n container.appendChild(valueEl);\n container.appendChild(incBtn);\n container.appendChild(submitBtn);\n\n return container;\n}\n"],"names":["TURKISH_DEFAULTS","formatPrice","raw","config","num","resolved","fixed","dotIdx","intPart","decPart","withSeparators","formatted","clampRating","value","clampDiscount","createStarRatingElement","rating","clamped","full","hasHalf","empty","container","halfStar","filled","addImageErrorHandler","img","createQuantityStepper","options","rawMin","rawMax","min","max","initial","compact","quantity","decBtn","valueEl","incBtn","submitBtn","defaultCartSvg","updateButtons","e","original"],"mappings":"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"}
@@ -0,0 +1,80 @@
1
+ const h = {
2
+ currencySymbol: "TL",
3
+ currencyPosition: "suffix",
4
+ thousandsSeparator: ".",
5
+ decimalSeparator: ",",
6
+ alwaysShowDecimals: !0
7
+ };
8
+ function x(e, d) {
9
+ const c = Number(e);
10
+ if (!Number.isFinite(c) || c < 0) return e;
11
+ const t = { ...h, ...d }, a = c % 1 !== 0 || t.alwaysShowDecimals ? c.toFixed(2) : c.toFixed(0), s = a.indexOf("."), n = s === -1 ? a : a.slice(0, s), o = s === -1 ? void 0 : a.slice(s + 1), l = n.replace(/\B(?=(\d{3})+(?!\d))/g, t.thousandsSeparator);
12
+ let i;
13
+ return o !== void 0 ? i = `${l}${t.decimalSeparator}${o}` : i = l, t.currencySymbol ? t.currencyPosition === "prefix" ? `${t.currencySymbol}${i}` : `${i} ${t.currencySymbol}` : i;
14
+ }
15
+ function y(e) {
16
+ return Number.isFinite(e) ? Math.max(0, Math.min(5, e)) : 0;
17
+ }
18
+ function S(e) {
19
+ return Number.isFinite(e) ? Math.max(0, Math.min(100, Math.round(e))) : 0;
20
+ }
21
+ function C(e) {
22
+ const d = y(e), c = Math.floor(d), t = d - c >= 0.5, u = 5 - c - (t ? 1 : 0), a = document.createElement("span");
23
+ if (a.className = "gengage-star-rating", a.setAttribute("role", "img"), a.setAttribute("aria-label", `${d.toFixed(1)} out of 5 stars`), c > 0 && a.appendChild(document.createTextNode("★".repeat(c))), t) {
24
+ const s = document.createElement("span");
25
+ s.className = "gengage-star-half", s.textContent = "☆";
26
+ const n = document.createElement("span");
27
+ n.textContent = "★", s.appendChild(n), a.appendChild(s);
28
+ }
29
+ return u > 0 && a.appendChild(document.createTextNode("☆".repeat(u))), a;
30
+ }
31
+ function E(e) {
32
+ e.addEventListener(
33
+ "error",
34
+ () => {
35
+ e.style.display = "none";
36
+ },
37
+ { once: !0 }
38
+ );
39
+ }
40
+ function M(e) {
41
+ const d = e.min ?? 1, c = e.max ?? 99, t = Math.min(d, c), u = Math.max(d, c), a = Math.max(t, Math.min(u, e.initial ?? t)), s = e.compact ?? !1;
42
+ let n = a;
43
+ const o = document.createElement("div");
44
+ o.className = `gengage-qty-stepper${s ? " gengage-qty-stepper--compact" : ""}`;
45
+ const l = document.createElement("button");
46
+ l.className = "gengage-qty-btn", l.type = "button", l.textContent = e.decreaseSymbol ?? "−", l.setAttribute("aria-label", e.decreaseLabel ?? "Decrease");
47
+ const i = document.createElement("span");
48
+ i.className = "gengage-qty-value", i.textContent = String(n), i.setAttribute("aria-live", "polite"), i.setAttribute("aria-atomic", "true");
49
+ const m = document.createElement("button");
50
+ m.className = "gengage-qty-btn", m.type = "button", m.textContent = e.increaseSymbol ?? "+", m.setAttribute("aria-label", e.increaseLabel ?? "Increase");
51
+ const r = document.createElement("button");
52
+ r.className = "gengage-qty-submit", r.type = "button";
53
+ const b = '<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="9" cy="21" r="1"/><circle cx="20" cy="21" r="1"/><path d="M1 1h4l2.68 13.39a2 2 0 0 0 2 1.61h9.72a2 2 0 0 0 2-1.61L23 6H6"/></svg>';
54
+ s ? (r.innerHTML = e.submitIcon ?? b, r.title = e.label ?? "Add to Cart") : r.textContent = e.label ?? "Add to Cart";
55
+ function g() {
56
+ l.disabled = n <= t, m.disabled = n >= u;
57
+ }
58
+ return l.addEventListener("click", (p) => {
59
+ p.stopPropagation(), n > t && (n--, i.textContent = String(n), g());
60
+ }), m.addEventListener("click", (p) => {
61
+ p.stopPropagation(), n < u && (n++, i.textContent = String(n), g());
62
+ }), r.addEventListener("click", (p) => {
63
+ p.stopPropagation(), e.onSubmit(n);
64
+ const f = r.innerHTML;
65
+ r.textContent = "✓", r.classList.add("gengage-qty-submit--success"), r.disabled = !0, setTimeout(() => {
66
+ r.innerHTML = f, r.classList.remove("gengage-qty-submit--success"), r.disabled = !1;
67
+ }, 1200);
68
+ }), o.addEventListener("click", (p) => {
69
+ p.stopPropagation();
70
+ }), g(), o.appendChild(l), o.appendChild(i), o.appendChild(m), o.appendChild(r), o;
71
+ }
72
+ export {
73
+ E as a,
74
+ C as b,
75
+ S as c,
76
+ M as d,
77
+ y as e,
78
+ x as f
79
+ };
80
+ //# sourceMappingURL=quantity-stepper-CjY_cpgJ.js.map