@useinsider/guido 3.3.0-beta.86ea9cd → 3.3.0-beta.fb07834

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.
@@ -1,35 +1,37 @@
1
- import { useConfig as L } from "./useConfig.js";
2
- import { TemplateTypes as V } from "../enums/defaults.js";
3
- import { DISPLAY_CONDITIONS_REGEX as _, DISPLAY_CONDITIONS_EXCEPTIONS_REGEX as H, CampaignCouldNotBeSavedKey as P, CanNotMakeAnyChangesForRunningKey as G } from "../enums/html-validator.js";
1
+ import { useConfig as _ } from "./useConfig.js";
2
+ import { TemplateTypes as H } from "../enums/defaults.js";
3
+ import { DISPLAY_CONDITIONS_REGEX as P, DISPLAY_CONDITIONS_EXCEPTIONS_REGEX as G, CampaignCouldNotBeSavedKey as M, CanNotMakeAnyChangesForRunningKey as $ } from "../enums/html-validator.js";
4
4
  import { ToasterTypeOptions as c } from "../enums/toaster.js";
5
- import { itemsBlockDynamicVariables as M } from "../extensions/Blocks/Items/enums/productEnums.js";
6
- import { useRecommendationStore as $ } from "../stores/recommendation.js";
7
- import { base64EncodeWithSpecialChars as X } from "../utils/base64.js";
8
- import { useHttp as j } from "./useHttp.js";
9
- import { useToaster as q } from "./useToaster.js";
10
- import { useTranslations as z } from "./useTranslations.js";
11
- const K = /recommendation-id="(\d+)"/g;
12
- function U(a) {
13
- return [...a.matchAll(K)].map((u) => u[1]);
5
+ import { itemsBlockDynamicVariables as q } from "../extensions/Blocks/Items/enums/productEnums.js";
6
+ import { useRecommendationExtensionStore as X } from "../extensions/Blocks/Recommendation/store/recommendation.js";
7
+ import { RecommendationRequiredFieldsKey as j } from "../extensions/Blocks/Recommendation/validation/requiredFields.js";
8
+ import { useRecommendationStore as K } from "../stores/recommendation.js";
9
+ import { base64EncodeWithSpecialChars as z } from "../utils/base64.js";
10
+ import { useHttp as U } from "./useHttp.js";
11
+ import { useToaster as Y } from "./useToaster.js";
12
+ import { useTranslations as Z } from "./useTranslations.js";
13
+ const J = /recommendation-id="(\d+)"/g;
14
+ function Q(a) {
15
+ return [...a.matchAll(J)].map((u) => u[1]);
14
16
  }
15
- function Y(a, u) {
17
+ function ee(a, u) {
16
18
  return u.some((d) => a.startsWith(`${d}_`));
17
19
  }
18
- const ce = () => {
20
+ const ge = () => {
19
21
  var y, h;
20
- const { showToaster: a } = q(), { post: u } = j(), { config: d } = L(), r = z(), g = $(), p = ((h = (y = d.value) == null ? void 0 : y.partner) == null ? void 0 : h.messageType) === V.transactional, b = async (e) => {
22
+ const { showToaster: a } = Y(), { post: u } = U(), { config: d } = _(), r = Z(), g = K(), S = X(), p = ((h = (y = d.value) == null ? void 0 : y.partner) == null ? void 0 : h.messageType) === H.transactional, b = async (e) => {
21
23
  const t = await u(
22
24
  "/newsletter/template-library/check-template-html-body",
23
- { html: X(e) }
25
+ { html: z(e) }
24
26
  ), { status: n, message: l } = t.data;
25
27
  return n || a({
26
28
  type: c.Alert,
27
29
  message: n === void 0 ? l : r("newsletter.invalid-url-link-for-toaster")
28
- }), r(P), l === r(G) && a({
30
+ }), r(M), l === r($) && a({
29
31
  type: c.Alert,
30
32
  message: r("newsletter.already-in-progress")
31
33
  }), n;
32
- }, w = (e) => !["if", "endif", "else", "elif", "now"].includes(e.toLowerCase()), S = (e) => ["if", "endif"].includes(e.toLowerCase()), E = (e, s) => {
34
+ }, w = (e) => !["if", "endif", "else", "elif", "now"].includes(e.toLowerCase()), E = (e) => ["if", "endif"].includes(e.toLowerCase()), A = (e, s) => {
33
35
  const t = e.match(/({%(.*?)%})/g);
34
36
  let n = !0;
35
37
  return t !== null && !p && t.forEach((l) => {
@@ -42,13 +44,13 @@ const ce = () => {
42
44
  }), n = !1);
43
45
  }
44
46
  }), n;
45
- }, A = async (e, s, t) => {
47
+ }, k = async (e, s, t) => {
46
48
  const n = t ? await b(e) : !0;
47
- return E(e, s) && n;
48
- }, I = (e) => e.length > 0 ? !0 : (a({
49
+ return A(e, s) && n;
50
+ }, x = (e) => e.length > 0 ? !0 : (a({
49
51
  type: c.Warning,
50
52
  message: r("newsletter.html-content-is-empty")
51
- }), !1), k = (e) => {
53
+ }), !1), I = (e) => {
52
54
  const s = (e.match(/{/gm) || []).length, t = (e.match(/}/gm) || []).length;
53
55
  return s > t && a({
54
56
  type: c.Warning,
@@ -57,7 +59,7 @@ const ce = () => {
57
59
  type: c.Warning,
58
60
  message: r("custom-fields.missing-opening-braces")
59
61
  }), s === t;
60
- }, x = (e) => {
62
+ }, F = (e) => {
61
63
  const s = e.match(/{{\s*(\w+\s+((\w+\|\w+)|(\w+)))\s*}}/gm) === null;
62
64
  return s || a({
63
65
  type: c.Warning,
@@ -66,10 +68,10 @@ const ce = () => {
66
68
  }, T = (e, s) => {
67
69
  const t = e.match(/{{([a-zA-Z0-9_\s]*)}}/gm);
68
70
  if (t && !p) {
69
- const n = new Set(s.map((i) => i.toLowerCase())), l = U(e), o = [];
71
+ const n = new Set(s.map((i) => i.toLowerCase())), l = Q(e), o = [];
70
72
  if (t.forEach((i) => {
71
73
  const m = i.slice(2, -2).trim().toLowerCase();
72
- (!n.has(m) || m === "") && !Y(m, l) && o.push(m);
74
+ (!n.has(m) || m === "") && !ee(m, l) && o.push(m);
73
75
  }), o.length > 0) {
74
76
  const i = `
75
77
  <ul>
@@ -83,11 +85,11 @@ const ce = () => {
83
85
  }
84
86
  }
85
87
  return !0;
86
- }, F = (e) => {
88
+ }, R = (e) => {
87
89
  const s = e.match(/{%(.*?)%}/g), t = [];
88
90
  let n = !0;
89
91
  if (s && s.forEach((l) => {
90
- const o = l.match(_), i = l.match(H), m = (o == null ? void 0 : o.join("")) || "";
92
+ const o = l.match(P), i = l.match(G), m = (o == null ? void 0 : o.join("")) || "";
91
93
  (!o || l !== m) && !i && (a({
92
94
  type: c.Alert,
93
95
  message: r("newsletter.display-conditions-invalid-syntax")
@@ -98,7 +100,7 @@ const ce = () => {
98
100
  }), n = !1);
99
101
  const v = f.match(/^[a-zA-Z]*$/g);
100
102
  v && v.forEach((C) => {
101
- S(C) && t.push(C);
103
+ E(C) && t.push(C);
102
104
  });
103
105
  });
104
106
  }), t.length) {
@@ -109,25 +111,28 @@ const ce = () => {
109
111
  }), n = !1);
110
112
  }
111
113
  return n;
112
- }, W = (e) => {
114
+ }, B = (e) => {
113
115
  const s = (e.match(/{% /gm) || []).length, t = (e.match(/ %}/gm) || []).length, n = s === t;
114
116
  return n || a({
115
117
  type: c.Warning,
116
118
  message: r("custom-conditions.no-space-after-braces")
117
119
  }), n;
118
- }, N = (e) => (e.match(/({%(.*?)%})/g) || []).filter((t) => t.includes("if")).map((t) => (t.match(/{{.*}}/gm) || []).length).reduce((t, n) => t + n, 0) > 0 ? (a({
120
+ }, W = (e) => (e.match(/({%(.*?)%})/g) || []).filter((t) => t.includes("if")).map((t) => (t.match(/{{.*}}/gm) || []).length).reduce((t, n) => t + n, 0) > 0 ? (a({
119
121
  type: c.Warning,
120
122
  message: r("custom-conditions.no-braces-inside-if-tag")
123
+ }), !1) : !0, N = () => S.hasInvalidBlock() ? (a({
124
+ type: c.Alert,
125
+ message: r(j)
121
126
  }), !1) : !0, O = () => g.recommendationConfigs && Object.values(g.recommendationConfigs).find((s) => s.filters.find((t) => t.value === "")) !== void 0 ? (a({
122
127
  type: c.Alert,
123
128
  message: r("newsletter.fill-all-necessary-fields")
124
- }), !1) : !0, B = (e) => {
129
+ }), !1) : !0, D = (e) => {
125
130
  const s = /src="[^"]*\.(svg|pst)"/gm;
126
131
  return e.match(s) === null ? !0 : (a({
127
132
  type: c.Alert,
128
133
  message: r("newsletter.invalid-image-type")
129
134
  }), !1);
130
- }, R = (e) => {
135
+ }, L = (e) => {
131
136
  const n = new DOMParser().parseFromString(e, "text/html").querySelectorAll(".checkbox-block-v2");
132
137
  return Array.from(n).find((o) => {
133
138
  var i;
@@ -136,7 +141,7 @@ const ce = () => {
136
141
  type: c.Alert,
137
142
  message: r("unsubscribe-templates.select-checkbox-groups")
138
143
  }), !1) : !0;
139
- }, D = (e) => {
144
+ }, V = (e) => {
140
145
  const n = new DOMParser().parseFromString(e, "text/html").querySelectorAll(".radio-button-v2");
141
146
  return Array.from(n).find((o) => {
142
147
  var i;
@@ -150,12 +155,12 @@ const ce = () => {
150
155
  var o, i;
151
156
  const n = [
152
157
  ...s.map((m) => m.value),
153
- ...M,
158
+ ...q,
154
159
  ...((i = (o = d.value) == null ? void 0 : o.template) == null ? void 0 : i.customFieldAttributes) ?? []
155
160
  ];
156
- return await A(e, n, t) && I(e) && k(e) && x(e) && T(e, n) && F(e) && W(e) && N(e) && O() && B(e) && R(e) && D(e);
161
+ return await k(e, n, t) && x(e) && I(e) && F(e) && T(e, n) && R(e) && B(e) && W(e) && N() && O() && D(e) && L(e) && V(e);
157
162
  } };
158
163
  };
159
164
  export {
160
- ce as useHtmlValidator
165
+ ge as useHtmlValidator
161
166
  };
@@ -1,33 +1,33 @@
1
- var k = Object.defineProperty;
2
- var _ = (r, n, t) => n in r ? k(r, n, { enumerable: !0, configurable: !0, writable: !0, value: t }) : r[n] = t;
3
- var g = (r, n, t) => _(r, typeof n != "symbol" ? n + "" : n, t);
4
- import { BlockId as B } from "../../../enums/block.js";
5
- import { getMigrationBannerHtml as b } from "../../../utils/migrationBannerHtml.js";
6
- import { Block as R, BlockCompositionType as y, ModificationDescription as C } from "../../../node_modules/@stripoinc/ui-editor-extensions/dist/esm/index.js";
7
- import { regenerateMobileProductRows as A } from "./controls/main/utils.js";
8
- import { ensureMobileCssRulesExist as d, setMobileLayoutOptOut as p, hasMobileLayoutOptOut as D } from "./controls/mobileLayout/cssRules.js";
9
- import { RecommendationConfigService as s } from "./services/configService.js";
10
- import { useRecommendationExtensionStore as f } from "./store/recommendation.js";
1
+ var I = Object.defineProperty;
2
+ var k = (r, n, t) => n in r ? I(r, n, { enumerable: !0, configurable: !0, writable: !0, value: t }) : r[n] = t;
3
+ var u = (r, n, t) => k(r, typeof n != "symbol" ? n + "" : n, t);
4
+ import { BlockId as _ } from "../../../enums/block.js";
5
+ import { getMigrationBannerHtml as B } from "../../../utils/migrationBannerHtml.js";
6
+ import { Block as b, BlockCompositionType as R, ModificationDescription as y } from "../../../node_modules/@stripoinc/ui-editor-extensions/dist/esm/index.js";
7
+ import { regenerateMobileProductRows as C } from "./controls/main/utils.js";
8
+ import { ensureMobileCssRulesExist as g, setMobileLayoutOptOut as d, hasMobileLayoutOptOut as A } from "./controls/mobileLayout/cssRules.js";
9
+ import { RecommendationConfigService as c } from "./services/configService.js";
10
+ import { useRecommendationExtensionStore as p } from "./store/recommendation.js";
11
11
  import { getDefaultTemplate as E } from "./templates/grid/template.js";
12
- const h = B.Recommendation, l = "recommendation-block-v2", m = "recommendation-id";
13
- let I = !1;
14
- class v extends R {
12
+ const f = _.Recommendation, l = "recommendation-block-v2", m = "recommendation-id";
13
+ let h = !1;
14
+ class q extends b {
15
15
  constructor() {
16
16
  super();
17
17
  /**
18
18
  * Stores the ID generated in getTemplate() so onCreated() can reuse it.
19
19
  * This avoids generating a new (different) ID in onCreated().
20
20
  */
21
- g(this, "_pendingBlockId", null);
21
+ u(this, "_pendingBlockId", null);
22
22
  }
23
23
  getId() {
24
- return h;
24
+ return f;
25
25
  }
26
26
  getIcon() {
27
27
  return "recommendation-icon";
28
28
  }
29
29
  getBlockCompositionType() {
30
- return y.CONTAINER;
30
+ return R.CONTAINER;
31
31
  }
32
32
  getName() {
33
33
  return this.api.translate("Recommendation Block");
@@ -38,8 +38,8 @@ class v extends R {
38
38
  );
39
39
  }
40
40
  getSettingsPanelTitleHtml() {
41
- return b(
42
- h,
41
+ return B(
42
+ f,
43
43
  this.api.translate("Recommendation Block"),
44
44
  this.api.translate("This block is switched from the Old Version to the New Version. We recommend you check the Recommendation block and test your message to ensure it works properly.")
45
45
  );
@@ -71,20 +71,13 @@ class v extends R {
71
71
  return;
72
72
  const i = this._pendingBlockId ?? this._generateNextId();
73
73
  this._pendingBlockId = null, this._assignRecommendationId(t, i);
74
- const { config: o, wasFreshDrop: c } = s.initializeConfig(
75
- this.api,
76
- t,
77
- { recommendationId: i }
78
- ), a = f();
79
- if (a.setCurrentBlock(i), c) {
80
- d(this.api);
81
- const u = this._getBlockElement(t);
82
- u && (p(this.api, u, !0), A({
83
- currentNode: t,
84
- documentModifier: this.api.getDocumentModifier()
85
- }));
86
- }
87
- a.patchCurrentBlockConfig({ language: o.language }, { triggerRefetch: !1 });
74
+ const o = c.initializeConfig(this.api, t, { recommendationId: i }), s = p();
75
+ s.setCurrentBlock(i), g(this.api);
76
+ const a = this._getBlockElement(t);
77
+ a && (d(this.api, a, !0), C({
78
+ currentNode: t,
79
+ documentModifier: this.api.getDocumentModifier()
80
+ })), s.patchCurrentBlockConfig({ language: o.language }, { triggerRefetch: !1 });
88
81
  }
89
82
  /**
90
83
  * Called when the document changes or template is loaded
@@ -97,20 +90,20 @@ class v extends R {
97
90
  if (!(!t || !("getNodeConfig" in t))) {
98
91
  if (!this._getRecommendationId(t)) {
99
92
  const e = this._generateNextId();
100
- this._assignRecommendationId(t, e), s.hasConfig(t) && s.updateConfig(
93
+ this._assignRecommendationId(t, e), c.hasConfig(t) && c.updateConfig(
101
94
  this.api,
102
95
  t,
103
96
  { recommendationId: e },
104
97
  "Assign recommendation ID to legacy block"
105
98
  );
106
99
  }
107
- s.needsMigration(t) && this._migrateFromLegacy(t);
100
+ c.needsMigration(t) && this._migrateFromLegacy(t);
108
101
  try {
109
- I || (d(this.api), I = !0);
110
- const e = s.getConfig(t), i = this._getBlockElement(t);
102
+ h || (g(this.api), h = !0);
103
+ const e = c.getConfig(t), i = this._getBlockElement(t);
111
104
  if (i) {
112
105
  const o = !e.mobileLayoutEnabled;
113
- D(i) !== o && p(this.api, i, o);
106
+ A(i) !== o && d(this.api, i, o);
114
107
  }
115
108
  } catch {
116
109
  }
@@ -124,7 +117,7 @@ class v extends R {
124
117
  */
125
118
  onDelete(t) {
126
119
  const e = this._getRecommendationId(t);
127
- e && f().removeBlockState(e);
120
+ e && p().removeBlockState(e);
128
121
  }
129
122
  /**
130
123
  * Generates the next unique recommendation ID by scanning all existing blocks
@@ -136,7 +129,7 @@ class v extends R {
136
129
  const e = this.api.getDocumentRoot();
137
130
  e && "querySelectorAll" in e && e.querySelectorAll(`.${l}`).forEach((o) => {
138
131
  if ("getAttribute" in o) {
139
- const c = o.getAttribute(m), a = c ? parseInt(c) : 0;
132
+ const s = o.getAttribute(m), a = s ? parseInt(s) : 0;
140
133
  a > t && (t = a);
141
134
  }
142
135
  });
@@ -155,7 +148,7 @@ class v extends R {
155
148
  if (!i)
156
149
  return;
157
150
  const o = this.api.getDocumentModifier();
158
- o.modifyHtml(i).setAttribute(m, e.toString()), o.apply(new C(`Assign recommendation ID ${e}`));
151
+ o.modifyHtml(i).setAttribute(m, e.toString()), o.apply(new y(`Assign recommendation ID ${e}`));
159
152
  }
160
153
  /**
161
154
  * Gets the recommendation-id from a block node
@@ -185,10 +178,10 @@ class v extends R {
185
178
  * Migrate configuration from legacy format
186
179
  */
187
180
  _migrateFromLegacy(t) {
188
- s.migrateFromDataAttributes(this.api, t);
181
+ c.migrateFromDataAttributes(this.api, t);
189
182
  }
190
183
  }
191
184
  export {
192
- h as BLOCK_ID,
193
- v as RecommendationBlock
185
+ f as BLOCK_ID,
186
+ q as RecommendationBlock
194
187
  };
@@ -1,5 +1,5 @@
1
- import { ModificationDescription as C } from "../../../../node_modules/@stripoinc/ui-editor-extensions/dist/esm/index.js";
2
- import { CURRENT_CONFIG_VERSION as l, DEFAULT_NODE_CONFIG as c } from "../constants/defaultConfig.js";
1
+ import { ModificationDescription as h } from "../../../../node_modules/@stripoinc/ui-editor-extensions/dist/esm/index.js";
2
+ import { CURRENT_CONFIG_VERSION as l, DEFAULT_NODE_CONFIG as r } from "../constants/defaultConfig.js";
3
3
  import { setCurrencyAttributes as D } from "../controls/main/utils.js";
4
4
  import { hasMinimalConfig as A } from "../types/nodeConfig.js";
5
5
  function S(e) {
@@ -8,7 +8,7 @@ function S(e) {
8
8
  function N(e) {
9
9
  return e === "." || e === "," || e === " " || e === "";
10
10
  }
11
- class M {
11
+ class V {
12
12
  // ========================================================================
13
13
  // Read Operations
14
14
  // ========================================================================
@@ -83,7 +83,7 @@ class M {
83
83
  * @returns The new complete configuration
84
84
  */
85
85
  static updateConfig(i, t, o, n) {
86
- const r = this.getConfig(t), u = this.deepMerge(r, o);
86
+ const c = this.getConfig(t), u = this.deepMerge(c, o);
87
87
  return this.saveConfig(i, t, u, n), u;
88
88
  }
89
89
  /**
@@ -91,28 +91,21 @@ class M {
91
91
  *
92
92
  * Called when a block is first created (dropped into template).
93
93
  * Can optionally merge in partial config from migration.
94
- *
95
- * The `wasFreshDrop` flag distinguishes a brand-new drop (no prior config)
96
- * from a clone (Stripo replays the source's setNodeConfig payload before
97
- * onCreated fires). Callers use this to skip side-effects already inherited
98
- * from the source.
99
94
  * @example
100
95
  * // In Block.onCreated lifecycle
101
- * const { config, wasFreshDrop } = RecommendationConfigService.initializeConfig(this.api, node);
96
+ * RecommendationConfigService.initializeConfig(this.api, node);
102
97
  * @param api - Stripo extension API with document modifier
103
98
  * @param node - The immutable HTML node to initialize
104
99
  * @param partialConfig - Optional partial config to merge with defaults
105
- * @returns The initialized configuration and whether the node was a fresh drop
100
+ * @returns The initialized configuration
106
101
  */
107
102
  static initializeConfig(i, t, o) {
108
- if (this.hasConfig(t))
109
- return { config: o ? this.updateConfig(i, t, o, "Initialize recommendation block") : this.getConfig(t), wasFreshDrop: !1 };
110
103
  const n = o ? this.mergeWithDefaults(o) : this.cloneDefaults();
111
104
  return this.saveConfig(i, t, n, "Initialize recommendation block"), D({
112
105
  currentNode: t,
113
106
  documentModifier: i.getDocumentModifier(),
114
107
  currency: n.currency
115
- }), { config: n, wasFreshDrop: !0 };
108
+ }), n;
116
109
  }
117
110
  /**
118
111
  * Save complete configuration to a node
@@ -125,9 +118,9 @@ class M {
125
118
  */
126
119
  static saveConfig(i, t, o, n) {
127
120
  try {
128
- i.getDocumentModifier().modifyHtml(t).setNodeConfig(o).apply(new C(n));
129
- } catch (r) {
130
- console.warn("[RecommendationConfigService] Failed to save config:", r);
121
+ i.getDocumentModifier().modifyHtml(t).setNodeConfig(o).apply(new h(n));
122
+ } catch (c) {
123
+ console.warn("[RecommendationConfigService] Failed to save config:", c);
131
124
  }
132
125
  }
133
126
  // ========================================================================
@@ -164,29 +157,29 @@ class M {
164
157
  s && typeof s == "object" && (Object.assign(o, s), o.configVersion = l);
165
158
  } catch {
166
159
  }
167
- const r = t.getAttribute("data-layout");
168
- r === "list" || r === "horizontal" ? o.layout = "list" : (r === "grid" || r === "vertical") && (o.layout = "grid");
160
+ const c = t.getAttribute("data-layout");
161
+ c === "list" || c === "horizontal" ? o.layout = "list" : (c === "grid" || c === "vertical") && (o.layout = "grid");
169
162
  const u = t.getAttribute("data-card-composition");
170
163
  u && (o.composition = u.split(",").filter(Boolean));
171
- const b = t.getAttribute("data-column-spacing");
172
- b && (o.columnSpacing = parseInt(b) || c.columnSpacing);
173
- const p = t.getAttribute("data-row-spacing");
174
- if (p && (o.rowSpacing = parseInt(p) || c.rowSpacing), !o.currency) {
164
+ const p = t.getAttribute("data-column-spacing");
165
+ p && (o.columnSpacing = parseInt(p) || r.columnSpacing);
166
+ const b = t.getAttribute("data-row-spacing");
167
+ if (b && (o.rowSpacing = parseInt(b) || r.rowSpacing), !o.currency) {
175
168
  const s = t.getAttribute("currency"), y = t.getAttribute("currency-symbol"), d = t.getAttribute("currency-alignment"), f = t.getAttribute("currency-thousand-separator"), g = t.getAttribute("currency-decimal-separator"), m = t.getAttribute("currency-decimal-count");
176
169
  if (s || y || d || f || g || m) {
177
- const a = c.currency, h = m ? parseInt(m) : NaN;
170
+ const a = r.currency, C = m ? parseInt(m) : NaN;
178
171
  o.currency = {
179
172
  code: s ?? a.code,
180
173
  symbol: y ?? a.symbol,
181
174
  alignment: d === "0" ? "before" : "after",
182
- decimalCount: Number.isFinite(h) ? h : a.decimalCount,
175
+ decimalCount: Number.isFinite(C) ? C : a.decimalCount,
183
176
  decimalSeparator: S(g) ? g : a.decimalSeparator,
184
177
  thousandSeparator: N(f) ? f : a.thousandSeparator
185
178
  };
186
179
  }
187
180
  }
188
181
  }
189
- return this.initializeConfig(i, t, o).config;
182
+ return this.initializeConfig(i, t, o);
190
183
  }
191
184
  /**
192
185
  * Check if configuration needs migration
@@ -204,12 +197,12 @@ class M {
204
197
  */
205
198
  static cloneDefaults() {
206
199
  return {
207
- ...c,
208
- currency: { ...c.currency },
209
- omnibusPrice: { ...c.omnibusPrice },
210
- omnibusDiscount: { ...c.omnibusDiscount },
211
- composition: [...c.composition],
212
- visibility: { ...c.visibility },
200
+ ...r,
201
+ currency: { ...r.currency },
202
+ omnibusPrice: { ...r.omnibusPrice },
203
+ omnibusDiscount: { ...r.omnibusDiscount },
204
+ composition: [...r.composition],
205
+ visibility: { ...r.visibility },
213
206
  filters: [],
214
207
  productIds: [],
215
208
  recommendationId: 0
@@ -279,5 +272,5 @@ class M {
279
272
  }
280
273
  }
281
274
  export {
282
- M as RecommendationConfigService
275
+ V as RecommendationConfigService
283
276
  };
@@ -7,8 +7,9 @@ import { EXCLUDED_ALGORITHM_IDS as D } from "../constants/defaultConfig.js";
7
7
  import { getDefaultProducts as g } from "../templates/utils.js";
8
8
  import { generateCompleteFilterQuery as b } from "../utils/filterUtil.js";
9
9
  import { isFilterValid as w } from "../validation/filterSchema.js";
10
+ import { isConfigValid as v } from "../validation/requiredFields.js";
10
11
  const h = C();
11
- let u = null, m = null, d = null;
12
+ let m = null, u = null, d = null;
12
13
  function k() {
13
14
  return {
14
15
  cardsInRow: F,
@@ -48,7 +49,7 @@ function I() {
48
49
  filterSnapshot: null
49
50
  };
50
51
  }
51
- const v = () => ({
52
+ const N = () => ({
52
53
  recommendationCampaignUrls: {},
53
54
  activePredictiveAlgorithms: [],
54
55
  languages: {},
@@ -57,8 +58,8 @@ const v = () => ({
57
58
  blockStates: {},
58
59
  currentRecommendationId: null,
59
60
  configVersion: 0
60
- }), _ = P("guidoRecommendationExtension", {
61
- state: () => v(),
61
+ }), M = P("guidoRecommendationExtension", {
62
+ state: () => N(),
62
63
  getters: {
63
64
  // ====================================================================
64
65
  // Proxy Getters — Backward Compatible Access to Current Block State
@@ -293,11 +294,11 @@ const v = () => ({
293
294
  // ====================================================================
294
295
  async fetchRecommendationCreateData() {
295
296
  if (!this.activePredictiveAlgorithms.length) {
296
- if (u) {
297
- await u;
297
+ if (m) {
298
+ await m;
298
299
  return;
299
300
  }
300
- u = (async () => {
301
+ m = (async () => {
301
302
  const {
302
303
  activePredictiveAlgorithms: t,
303
304
  languages: e,
@@ -310,26 +311,26 @@ const v = () => ({
310
311
  this.currencyList = r;
311
312
  })();
312
313
  try {
313
- await u;
314
+ await m;
314
315
  } finally {
315
- u = null;
316
+ m = null;
316
317
  }
317
318
  }
318
319
  },
319
320
  async fetchRecommendationFilters() {
320
321
  if (!Object.keys(this.filterList).length) {
321
- if (m) {
322
- await m;
322
+ if (u) {
323
+ await u;
323
324
  return;
324
325
  }
325
- m = (async () => {
326
+ u = (async () => {
326
327
  const t = await h.fetchRecommendationFilters();
327
328
  this.filterList = t;
328
329
  })();
329
330
  try {
330
- await m;
331
+ await u;
331
332
  } finally {
332
- m = null;
333
+ u = null;
333
334
  }
334
335
  }
335
336
  },
@@ -398,6 +399,14 @@ const v = () => ({
398
399
  generateFilterQuery() {
399
400
  return b(this.recommendationConfigs.filters);
400
401
  },
402
+ /**
403
+ * Validation-only check invoked at save-CTA time. Defined as an action
404
+ * (not a getter) so reading it does not register reactive tracking on
405
+ * every block's recommendationConfigs across user edits.
406
+ */
407
+ hasInvalidBlock() {
408
+ return Object.values(this.blockStates).some((t) => !v(t.recommendationConfigs, this));
409
+ },
401
410
  // ====================================================================
402
411
  // Per-Block Product Fetching
403
412
  // ====================================================================
@@ -443,5 +452,5 @@ const v = () => ({
443
452
  }
444
453
  });
445
454
  export {
446
- _ as useRecommendationExtensionStore
455
+ M as useRecommendationExtensionStore
447
456
  };
@@ -0,0 +1,33 @@
1
+ const o = [
2
+ {
3
+ key: "locale",
4
+ getValue: (e) => e.language,
5
+ getAvailableOptions: (e) => Object.keys(e.languages)
6
+ },
7
+ {
8
+ key: "currency",
9
+ getValue: (e) => e.currencySettings.value,
10
+ getAvailableOptions: (e) => e.currencyList.map((n) => n.text)
11
+ }
12
+ ], l = "newsletter.recommendation-fill-required-fields";
13
+ function u(e, n) {
14
+ return o.filter((t) => {
15
+ var a;
16
+ if (t.condition && !t.condition(e))
17
+ return !1;
18
+ const i = t.getValue(e);
19
+ if (!i)
20
+ return !0;
21
+ const r = (a = t.getAvailableOptions) == null ? void 0 : a.call(t, n);
22
+ return r !== void 0 && !r.includes(i);
23
+ }).map((t) => t.key);
24
+ }
25
+ function s(e, n) {
26
+ return u(e, n).length === 0;
27
+ }
28
+ export {
29
+ o as REQUIRED_RECOMMENDATION_FIELDS,
30
+ l as RecommendationRequiredFieldsKey,
31
+ u as getInvalidFields,
32
+ s as isConfigValid
33
+ };
@@ -95,23 +95,15 @@ export declare class RecommendationConfigService {
95
95
  *
96
96
  * Called when a block is first created (dropped into template).
97
97
  * Can optionally merge in partial config from migration.
98
- *
99
- * The `wasFreshDrop` flag distinguishes a brand-new drop (no prior config)
100
- * from a clone (Stripo replays the source's setNodeConfig payload before
101
- * onCreated fires). Callers use this to skip side-effects already inherited
102
- * from the source.
103
98
  * @example
104
99
  * // In Block.onCreated lifecycle
105
- * const { config, wasFreshDrop } = RecommendationConfigService.initializeConfig(this.api, node);
100
+ * RecommendationConfigService.initializeConfig(this.api, node);
106
101
  * @param api - Stripo extension API with document modifier
107
102
  * @param node - The immutable HTML node to initialize
108
103
  * @param partialConfig - Optional partial config to merge with defaults
109
- * @returns The initialized configuration and whether the node was a fresh drop
104
+ * @returns The initialized configuration
110
105
  */
111
- static initializeConfig(api: DocumentModifierApi, node: ImmutableHtmlNode, partialConfig?: PartialNodeConfig): {
112
- config: RecommendationNodeConfig;
113
- wasFreshDrop: boolean;
114
- };
106
+ static initializeConfig(api: DocumentModifierApi, node: ImmutableHtmlNode, partialConfig?: PartialNodeConfig): RecommendationNodeConfig;
115
107
  /**
116
108
  * Save complete configuration to a node
117
109
  *
@@ -1,5 +1,5 @@
1
1
  import type { Orientation, Languages, Currency, NumericSeparator, FiltersResponse, Filter, RecommendationProduct } from '@@/Types/recommendation';
2
- interface PerBlockConfigs {
2
+ export interface PerBlockConfigs {
3
3
  cardsInRow: number;
4
4
  currencySettings: {
5
5
  name: string;
@@ -272,6 +272,12 @@ export declare const useRecommendationExtensionStore: import("pinia").StoreDefin
272
272
  deleteFilter(filter: Filter): void;
273
273
  addFilter(filter: Filter): void;
274
274
  generateFilterQuery(): string;
275
+ /**
276
+ * Validation-only check invoked at save-CTA time. Defined as an action
277
+ * (not a getter) so reading it does not register reactive tracking on
278
+ * every block's recommendationConfigs across user edits.
279
+ */
280
+ hasInvalidBlock(): boolean;
275
281
  fetchRecommendationProducts(): Promise<void>;
276
282
  _doFetchProducts(): Promise<void>;
277
283
  }>;
@@ -0,0 +1,21 @@
1
+ import type { PerBlockConfigs } from '../store/recommendation';
2
+ import type { Currency, Languages } from '@@/Types/recommendation';
3
+ /**
4
+ * Structural slice of the recommendation extension store that descriptors may read.
5
+ * Add new fields here when a future descriptor needs them.
6
+ */
7
+ export interface ExtensionStoreSlice {
8
+ languages: Languages;
9
+ currencyList: Currency[];
10
+ }
11
+ interface RequiredField {
12
+ key: string;
13
+ getValue: (config: PerBlockConfigs) => string;
14
+ getAvailableOptions?: (store: ExtensionStoreSlice) => string[];
15
+ condition?: (config: PerBlockConfigs) => boolean;
16
+ }
17
+ export declare const REQUIRED_RECOMMENDATION_FIELDS: RequiredField[];
18
+ export declare const RecommendationRequiredFieldsKey = "newsletter.recommendation-fill-required-fields";
19
+ export declare function getInvalidFields(config: PerBlockConfigs, store: ExtensionStoreSlice): string[];
20
+ export declare function isConfigValid(config: PerBlockConfigs, store: ExtensionStoreSlice): boolean;
21
+ export {};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@useinsider/guido",
3
- "version": "3.3.0-beta.86ea9cd",
3
+ "version": "3.3.0-beta.fb07834",
4
4
  "description": "Guido is a Vue + TypeScript wrapper for Email Plugin. Easily embed the email editor in your Vue applications.",
5
5
  "main": "./dist/guido.umd.cjs",
6
6
  "module": "./dist/library.js",