ctt-babylon 0.12.23 → 0.12.24
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/core/babylon-fo-c3-img-txt/babylon-fo-c3-img-txt.component.mjs +11 -6
- package/esm2022/lib/components/core/babylon-he-svg-list/babylon-he-svg-list.component.mjs +27 -6
- package/esm2022/lib/components/core/babylon-list-c3-img-txt/babylon-list-c3-img-txt.component.mjs +11 -6
- package/esm2022/lib/components/external/core/lis-c4-txt-des-cta/lis-c4-txt-des-cta.component.mjs +249 -92
- package/esm2022/lib/utils/utils.mjs +42 -1
- package/fesm2022/ctt-babylon.mjs +332 -106
- package/fesm2022/ctt-babylon.mjs.map +1 -1
- package/lib/components/core/babylon-he-svg-list/babylon-he-svg-list.component.d.ts +1 -0
- package/lib/components/external/core/lis-c4-txt-des-cta/lis-c4-txt-des-cta.component.d.ts +13 -4
- package/lib/utils/utils.d.ts +3 -0
- package/package.json +1 -1
|
@@ -175,5 +175,46 @@ export class Utils {
|
|
|
175
175
|
};
|
|
176
176
|
return list.sort((a, b) => toNum(a.order) - toNum(b.order));
|
|
177
177
|
}
|
|
178
|
+
static toSlug(input) {
|
|
179
|
+
return (input ?? '')
|
|
180
|
+
.toString()
|
|
181
|
+
.trim()
|
|
182
|
+
.toLowerCase()
|
|
183
|
+
.normalize('NFD')
|
|
184
|
+
.replace(/[\u0300-\u036f]/g, '') // quita tildes
|
|
185
|
+
.replace(/&/g, 'and') // opcional
|
|
186
|
+
.replace(/[^a-z0-9\s-]/g, '') // quita caracteres raros
|
|
187
|
+
.replace(/\s+/g, '-') // espacios -> -
|
|
188
|
+
.replace(/-+/g, '-') // colapsa ---
|
|
189
|
+
.replace(/^-|-$/g, ''); // quita - al inicio/fin
|
|
190
|
+
}
|
|
191
|
+
static uniqueSlug(label, used) {
|
|
192
|
+
const base = this.toSlug(label) || 'item';
|
|
193
|
+
if (!used.has(base)) {
|
|
194
|
+
used.add(base);
|
|
195
|
+
return base;
|
|
196
|
+
}
|
|
197
|
+
let i = 2;
|
|
198
|
+
while (used.has(`${base}-${i}`))
|
|
199
|
+
i++;
|
|
200
|
+
const slug = `${base}-${i}`;
|
|
201
|
+
used.add(slug);
|
|
202
|
+
return slug;
|
|
203
|
+
}
|
|
204
|
+
static canonicalHotelTypeSlug(label) {
|
|
205
|
+
// slug base (sin tildes/raros)
|
|
206
|
+
let s = Utils.toSlug(label);
|
|
207
|
+
// normaliza plurales comunes ES (hotel(es), apartamento(s))
|
|
208
|
+
// puedes ampliar aquí si tienes "apartamentos turísticos", etc.
|
|
209
|
+
if (s === 'hoteles')
|
|
210
|
+
return 'hotel';
|
|
211
|
+
if (s === 'apartamentos')
|
|
212
|
+
return 'apartamento';
|
|
213
|
+
// regla genérica suave: si termina en 's' y no es 'dos/tres/...' etc.
|
|
214
|
+
// (evita cargarte cosas como "pools" si existieran, pero aquí aplica bien)
|
|
215
|
+
if (s.endsWith('s'))
|
|
216
|
+
s = s.slice(0, -1);
|
|
217
|
+
return s;
|
|
218
|
+
}
|
|
178
219
|
}
|
|
179
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../projects/babylon/src/lib/utils/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable } from 'rxjs';\nimport { BabylonButtonI, BodyComponent } from '../interfaces';\n\nexport class Utils {\n    // Función para convertir una cadena de snake_case a camelCase\n    static snakeToCamel(snake: string): string {\n        return snake.replace(/(_\\w)/g, (matches) => matches[1].toUpperCase());\n    }\n\n    // Función para convertir las propiedades de un objeto de snake_case a camelCase\n    static convertObjectKeysToCamelCase(obj: any): any {\n        if (Array.isArray(obj)) {\n            return obj.map((item) => this.convertObjectKeysToCamelCase(item));\n        } else if (\n            obj &&\n            typeof obj === 'object' &&\n            obj.constructor === Object\n        ) {\n            return Object.keys(obj).reduce((result, key) => {\n                const camelKey = this.snakeToCamel(key);\n                result[camelKey] = this.convertObjectKeysToCamelCase(obj[key]);\n                return result;\n            }, {} as any);\n        }\n        return obj;\n    }\n\n    static formatLinkType(linkType: string): string | undefined {\n        return linkType === 'external_link' || linkType === 'external'\n            ? 'external'\n            : linkType === 'internal_link' || linkType === 'internal'\n              ? 'internal'\n              : linkType;\n    }\n\n    static findComponent<T>(\n        body: BodyComponent<T>[],\n        name: string\n    ): T | undefined {\n        return body.find((component) => component?.name?.toLowerCase() === name)\n            ?.props;\n    }\n\n    static convertToBase64(file: File): Promise<string> {\n        return new Promise<string>((resolve, reject) => {\n            const reader = new FileReader();\n            reader.readAsDataURL(file);\n            reader.onload = () => resolve(reader.result as string);\n            reader.onerror = (error) => reject(error);\n        });\n    }\n\n    static normalizeText(text: string): string {\n        return text\n            ?.normalize('NFD') // Descompone los caracteres con acentos\n            ?.replace(/[\\u0300-\\u036f]/g, '') // Elimina los acentos\n            ?.toLowerCase();\n    }\n\n    static mapButtons(\n        button: any | any[] | undefined | null,\n        buttonType?: string\n    ): (BabylonButtonI | undefined)[] | undefined {\n        if (Array.isArray(button)) {\n            return Utils.sortButtonsByOrder(\n                button?.map((btn) =>\n                    btn.label?.length ||\n                    btn.linkValue?.length ||\n                    btn.link_value?.length\n                        ? ({\n                              type: buttonType ?? undefined,\n                              label: btn?.label ?? undefined,\n                              linkType: Utils.formatLinkType(\n                                  btn?.linkType ?? btn?.link_type\n                              ),\n                              url: (() => {\n                                  const rawLink =\n                                      btn.linkValue ?? btn.link_value;\n                                  if (rawLink && typeof rawLink === 'string') {\n                                      const trimmedValue = rawLink?.endsWith(\n                                          '/'\n                                      )\n                                          ? rawLink.slice(0, -1)\n                                          : rawLink;\n\n                                      if (/^\\+?[0-9]+$/.test(trimmedValue)) {\n                                          // Es un teléfono\n                                          return trimmedValue.startsWith('tel:')\n                                              ? trimmedValue\n                                              : `tel:${trimmedValue}`;\n                                      } else if (\n                                          /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(\n                                              trimmedValue\n                                          )\n                                      ) {\n                                          // Es un correo\n                                          return trimmedValue.startsWith(\n                                              'mailto:'\n                                          )\n                                              ? trimmedValue\n                                              : `mailto:${trimmedValue}`;\n                                      }\n\n                                      return trimmedValue;\n                                  }\n                                  return undefined;\n                              })(),\n                              icon: btn.icon ? `icon-` + btn?.icon : undefined,\n                              order: btn.order?.toString(),\n                              vars: btn.linkVars?.toString(),\n                          } as BabylonButtonI)\n                        : undefined\n                ),\n                { keepOriginal: true }\n            );\n        } else {\n            return undefined;\n        }\n    }\n\n    static loadScript$(src: string): Observable<void> {\n        return new Observable<void>((observer) => {\n            const existing = document.querySelector(`script[src=\"${src}\"]`);\n            if (existing) {\n                observer.next();\n                observer.complete();\n                return;\n            }\n\n            const script = document.createElement('script');\n            script.src = src;\n            script.type = 'text/javascript';\n            script.defer = true;\n            script.async = true;\n\n            script.onload = () => {\n                observer.next();\n                observer.complete();\n            };\n\n            script.onerror = (err) => {\n                observer.error(new Error(`Error loading script: ${src}`));\n            };\n\n            document.head.appendChild(script);\n        });\n    }\n\n    static loadStyle$(href: string): Observable<boolean> {\n        return new Observable<boolean>((observer) => {\n            const existing = document.querySelector(`link[href=\"${href}\"]`);\n            if (existing) {\n                observer.next(true);\n                observer.complete();\n                return;\n            }\n\n            const link = document.createElement('link');\n            link.rel = 'preload';\n            link.href = href;\n            link.as = 'style';\n            link.onload = () => {\n                link.onload = null;\n\n                link.rel = 'stylesheet';\n\n                observer.next(true);\n                observer.complete();\n            };\n            link.onerror = () => {\n                observer.error(\n                    new Error(`No se pudo precargar el estilo: ${href}`)\n                );\n            };\n\n            document.head.appendChild(link);\n\n            // opcional: fallback <noscript> (no lo necesita Angular pero lo puedes agregar)\n            const noscript = document.createElement('noscript');\n            noscript.innerHTML = `<link rel=\"stylesheet\" href=\"${href}\">`;\n            document.head.appendChild(noscript);\n        });\n    }\n\n    static safeOpenExternal(url: string): void {\n        if (typeof window === 'undefined' || !url) return;\n\n        const win = window.open(url, '_blank', 'noopener,noreferrer');\n        if (win) return;\n\n        // Fallback si bloquean popups\n        const a = document.createElement('a');\n        a.href = url;\n        a.target = '_blank';\n        a.rel = 'noopener noreferrer';\n        a.style.display = 'none';\n        document.body.appendChild(a);\n        a.click();\n        a.remove();\n    }\n\n    static extractNumberRange(\n        value?: string | null,\n        fallback: number[] = []\n    ): number[] {\n        if (!value) return fallback;\n\n        const match = value.match(/\\d+/);\n        if (!match) return fallback;\n\n        const n = Number(match[0]);\n        if (Number.isNaN(n) || n <= 0) return fallback;\n\n        return Array.from({ length: n }, (_, i) => i + 1);\n    }\n\n    static sortButtonsByOrder<T extends { order?: any }>(\n        buttons?: Array<T | undefined | null> | null,\n        opts?: { keepOriginal?: boolean }\n    ): T[] {\n        const arr = (buttons ?? []).filter((b): b is T => !!b);\n        if (!arr.length) return [];\n\n        const list = opts?.keepOriginal ? [...arr] : arr;\n\n        const toNum = (v: any): number => {\n            const n = Number(v);\n            return Number.isFinite(n) ? n : Number.POSITIVE_INFINITY;\n        };\n\n        return list.sort((a, b) => toNum(a.order) - toNum(b.order));\n    }\n}\n"]}
|
|
220
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../projects/babylon/src/lib/utils/utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAGlC,MAAM,OAAO,KAAK;IACd,8DAA8D;IAC9D,MAAM,CAAC,YAAY,CAAC,KAAa;QAC7B,OAAO,KAAK,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IAC1E,CAAC;IAED,gFAAgF;IAChF,MAAM,CAAC,4BAA4B,CAAC,GAAQ;QACxC,IAAI,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;YACpB,OAAO,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,4BAA4B,CAAC,IAAI,CAAC,CAAC,CAAC;SACrE;aAAM,IACH,GAAG;YACH,OAAO,GAAG,KAAK,QAAQ;YACvB,GAAG,CAAC,WAAW,KAAK,MAAM,EAC5B;YACE,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE;gBAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;gBACxC,MAAM,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,4BAA4B,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBAC/D,OAAO,MAAM,CAAC;YAClB,CAAC,EAAE,EAAS,CAAC,CAAC;SACjB;QACD,OAAO,GAAG,CAAC;IACf,CAAC;IAED,MAAM,CAAC,cAAc,CAAC,QAAgB;QAClC,OAAO,QAAQ,KAAK,eAAe,IAAI,QAAQ,KAAK,UAAU;YAC1D,CAAC,CAAC,UAAU;YACZ,CAAC,CAAC,QAAQ,KAAK,eAAe,IAAI,QAAQ,KAAK,UAAU;gBACvD,CAAC,CAAC,UAAU;gBACZ,CAAC,CAAC,QAAQ,CAAC;IACrB,CAAC;IAED,MAAM,CAAC,aAAa,CAChB,IAAwB,EACxB,IAAY;QAEZ,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,EAAE,IAAI,EAAE,WAAW,EAAE,KAAK,IAAI,CAAC;YACpE,EAAE,KAAK,CAAC;IAChB,CAAC;IAED,MAAM,CAAC,eAAe,CAAC,IAAU;QAC7B,OAAO,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YAC3C,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAChC,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;YAC3B,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,MAAgB,CAAC,CAAC;YACvD,MAAM,CAAC,OAAO,GAAG,CAAC,KAAK,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC9C,CAAC,CAAC,CAAC;IACP,CAAC;IAED,MAAM,CAAC,aAAa,CAAC,IAAY;QAC7B,OAAO,IAAI;YACP,EAAE,SAAS,CAAC,KAAK,CAAC,CAAC,wCAAwC;YAC3D,EAAE,OAAO,CAAC,kBAAkB,EAAE,EAAE,CAAC,CAAC,sBAAsB;YACxD,EAAE,WAAW,EAAE,CAAC;IACxB,CAAC;IAED,MAAM,CAAC,UAAU,CACb,MAAsC,EACtC,UAAmB;QAEnB,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACvB,OAAO,KAAK,CAAC,kBAAkB,CAC3B,MAAM,EAAE,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAChB,GAAG,CAAC,KAAK,EAAE,MAAM;gBACjB,GAAG,CAAC,SAAS,EAAE,MAAM;gBACrB,GAAG,CAAC,UAAU,EAAE,MAAM;gBAClB,CAAC,CAAE;oBACG,IAAI,EAAE,UAAU,IAAI,SAAS;oBAC7B,KAAK,EAAE,GAAG,EAAE,KAAK,IAAI,SAAS;oBAC9B,QAAQ,EAAE,KAAK,CAAC,cAAc,CAC1B,GAAG,EAAE,QAAQ,IAAI,GAAG,EAAE,SAAS,CAClC;oBACD,GAAG,EAAE,CAAC,GAAG,EAAE;wBACP,MAAM,OAAO,GACT,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,UAAU,CAAC;wBACpC,IAAI,OAAO,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE;4BACxC,MAAM,YAAY,GAAG,OAAO,EAAE,QAAQ,CAClC,GAAG,CACN;gCACG,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;gCACtB,CAAC,CAAC,OAAO,CAAC;4BAEd,IAAI,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE;gCAClC,iBAAiB;gCACjB,OAAO,YAAY,CAAC,UAAU,CAAC,MAAM,CAAC;oCAClC,CAAC,CAAC,YAAY;oCACd,CAAC,CAAC,OAAO,YAAY,EAAE,CAAC;6BAC/B;iCAAM,IACH,4BAA4B,CAAC,IAAI,CAC7B,YAAY,CACf,EACH;gCACE,eAAe;gCACf,OAAO,YAAY,CAAC,UAAU,CAC1B,SAAS,CACZ;oCACG,CAAC,CAAC,YAAY;oCACd,CAAC,CAAC,UAAU,YAAY,EAAE,CAAC;6BAClC;4BAED,OAAO,YAAY,CAAC;yBACvB;wBACD,OAAO,SAAS,CAAC;oBACrB,CAAC,CAAC,EAAE;oBACJ,IAAI,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,GAAG,GAAG,EAAE,IAAI,CAAC,CAAC,CAAC,SAAS;oBAChD,KAAK,EAAE,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE;oBAC5B,IAAI,EAAE,GAAG,CAAC,QAAQ,EAAE,QAAQ,EAAE;iBACd;gBACtB,CAAC,CAAC,SAAS,CAClB,EACD,EAAE,YAAY,EAAE,IAAI,EAAE,CACzB,CAAC;SACL;aAAM;YACH,OAAO,SAAS,CAAC;SACpB;IACL,CAAC;IAED,MAAM,CAAC,WAAW,CAAC,GAAW;QAC1B,OAAO,IAAI,UAAU,CAAO,CAAC,QAAQ,EAAE,EAAE;YACrC,MAAM,QAAQ,GAAG,QAAQ,CAAC,aAAa,CAAC,eAAe,GAAG,IAAI,CAAC,CAAC;YAChE,IAAI,QAAQ,EAAE;gBACV,QAAQ,CAAC,IAAI,EAAE,CAAC;gBAChB,QAAQ,CAAC,QAAQ,EAAE,CAAC;gBACpB,OAAO;aACV;YAED,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;YAChD,MAAM,CAAC,GAAG,GAAG,GAAG,CAAC;YACjB,MAAM,CAAC,IAAI,GAAG,iBAAiB,CAAC;YAChC,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;YACpB,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC;YAEpB,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;gBACjB,QAAQ,CAAC,IAAI,EAAE,CAAC;gBAChB,QAAQ,CAAC,QAAQ,EAAE,CAAC;YACxB,CAAC,CAAC;YAEF,MAAM,CAAC,OAAO,GAAG,CAAC,GAAG,EAAE,EAAE;gBACrB,QAAQ,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,yBAAyB,GAAG,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC;YAEF,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;IACP,CAAC;IAED,MAAM,CAAC,UAAU,CAAC,IAAY;QAC1B,OAAO,IAAI,UAAU,CAAU,CAAC,QAAQ,EAAE,EAAE;YACxC,MAAM,QAAQ,GAAG,QAAQ,CAAC,aAAa,CAAC,cAAc,IAAI,IAAI,CAAC,CAAC;YAChE,IAAI,QAAQ,EAAE;gBACV,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACpB,QAAQ,CAAC,QAAQ,EAAE,CAAC;gBACpB,OAAO;aACV;YAED,MAAM,IAAI,GAAG,QAAQ,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;YAC5C,IAAI,CAAC,GAAG,GAAG,SAAS,CAAC;YACrB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;YACjB,IAAI,CAAC,EAAE,GAAG,OAAO,CAAC;YAClB,IAAI,CAAC,MAAM,GAAG,GAAG,EAAE;gBACf,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;gBAEnB,IAAI,CAAC,GAAG,GAAG,YAAY,CAAC;gBAExB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACpB,QAAQ,CAAC,QAAQ,EAAE,CAAC;YACxB,CAAC,CAAC;YACF,IAAI,CAAC,OAAO,GAAG,GAAG,EAAE;gBAChB,QAAQ,CAAC,KAAK,CACV,IAAI,KAAK,CAAC,mCAAmC,IAAI,EAAE,CAAC,CACvD,CAAC;YACN,CAAC,CAAC;YAEF,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;YAEhC,gFAAgF;YAChF,MAAM,QAAQ,GAAG,QAAQ,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;YACpD,QAAQ,CAAC,SAAS,GAAG,gCAAgC,IAAI,IAAI,CAAC;YAC9D,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACP,CAAC;IAED,MAAM,CAAC,gBAAgB,CAAC,GAAW;QAC/B,IAAI,OAAO,MAAM,KAAK,WAAW,IAAI,CAAC,GAAG;YAAE,OAAO;QAElD,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,QAAQ,EAAE,qBAAqB,CAAC,CAAC;QAC9D,IAAI,GAAG;YAAE,OAAO;QAEhB,8BAA8B;QAC9B,MAAM,CAAC,GAAG,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;QACtC,CAAC,CAAC,IAAI,GAAG,GAAG,CAAC;QACb,CAAC,CAAC,MAAM,GAAG,QAAQ,CAAC;QACpB,CAAC,CAAC,GAAG,GAAG,qBAAqB,CAAC;QAC9B,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;QACzB,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;QAC7B,CAAC,CAAC,KAAK,EAAE,CAAC;QACV,CAAC,CAAC,MAAM,EAAE,CAAC;IACf,CAAC;IAED,MAAM,CAAC,kBAAkB,CACrB,KAAqB,EACrB,WAAqB,EAAE;QAEvB,IAAI,CAAC,KAAK;YAAE,OAAO,QAAQ,CAAC;QAE5B,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACjC,IAAI,CAAC,KAAK;YAAE,OAAO,QAAQ,CAAC;QAE5B,MAAM,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAC3B,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;YAAE,OAAO,QAAQ,CAAC;QAE/C,OAAO,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IACtD,CAAC;IAED,MAAM,CAAC,kBAAkB,CACrB,OAA4C,EAC5C,IAAiC;QAEjC,MAAM,GAAG,GAAG,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAU,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACvD,IAAI,CAAC,GAAG,CAAC,MAAM;YAAE,OAAO,EAAE,CAAC;QAE3B,MAAM,IAAI,GAAG,IAAI,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;QAEjD,MAAM,KAAK,GAAG,CAAC,CAAM,EAAU,EAAE;YAC7B,MAAM,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;YACpB,OAAO,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,iBAAiB,CAAC;QAC7D,CAAC,CAAC;QAEF,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;IAChE,CAAC;IAED,MAAM,CAAC,MAAM,CAAC,KAAa;QACvB,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC;aACf,QAAQ,EAAE;aACV,IAAI,EAAE;aACN,WAAW,EAAE;aACb,SAAS,CAAC,KAAK,CAAC;aAChB,OAAO,CAAC,kBAAkB,EAAE,EAAE,CAAC,CAAC,eAAe;aAC/C,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,WAAW;aAChC,OAAO,CAAC,eAAe,EAAE,EAAE,CAAC,CAAC,yBAAyB;aACtD,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,gBAAgB;aACrC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,cAAc;aAClC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,wBAAwB;IACxD,CAAC;IAED,MAAM,CAAC,UAAU,CAAC,KAAa,EAAE,IAAiB;QAC9C,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,MAAM,CAAC;QAC1C,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YACjB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACf,OAAO,IAAI,CAAC;SACf;QAED,IAAI,CAAC,GAAG,CAAC,CAAC;QACV,OAAO,IAAI,CAAC,GAAG,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;YAAE,CAAC,EAAE,CAAC;QACrC,MAAM,IAAI,GAAG,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC;QAC5B,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACf,OAAO,IAAI,CAAC;IAChB,CAAC;IAED,MAAM,CAAC,sBAAsB,CAAC,KAAa;QACvC,+BAA+B;QAC/B,IAAI,CAAC,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAE5B,4DAA4D;QAC5D,gEAAgE;QAChE,IAAI,CAAC,KAAK,SAAS;YAAE,OAAO,OAAO,CAAC;QACpC,IAAI,CAAC,KAAK,cAAc;YAAE,OAAO,aAAa,CAAC;QAE/C,sEAAsE;QACtE,2EAA2E;QAC3E,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC;YAAE,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAExC,OAAO,CAAC,CAAC;IACb,CAAC;CACJ","sourcesContent":["import { Observable } from 'rxjs';\nimport { BabylonButtonI, BodyComponent } from '../interfaces';\n\nexport class Utils {\n    // Función para convertir una cadena de snake_case a camelCase\n    static snakeToCamel(snake: string): string {\n        return snake.replace(/(_\\w)/g, (matches) => matches[1].toUpperCase());\n    }\n\n    // Función para convertir las propiedades de un objeto de snake_case a camelCase\n    static convertObjectKeysToCamelCase(obj: any): any {\n        if (Array.isArray(obj)) {\n            return obj.map((item) => this.convertObjectKeysToCamelCase(item));\n        } else if (\n            obj &&\n            typeof obj === 'object' &&\n            obj.constructor === Object\n        ) {\n            return Object.keys(obj).reduce((result, key) => {\n                const camelKey = this.snakeToCamel(key);\n                result[camelKey] = this.convertObjectKeysToCamelCase(obj[key]);\n                return result;\n            }, {} as any);\n        }\n        return obj;\n    }\n\n    static formatLinkType(linkType: string): string | undefined {\n        return linkType === 'external_link' || linkType === 'external'\n            ? 'external'\n            : linkType === 'internal_link' || linkType === 'internal'\n              ? 'internal'\n              : linkType;\n    }\n\n    static findComponent<T>(\n        body: BodyComponent<T>[],\n        name: string\n    ): T | undefined {\n        return body.find((component) => component?.name?.toLowerCase() === name)\n            ?.props;\n    }\n\n    static convertToBase64(file: File): Promise<string> {\n        return new Promise<string>((resolve, reject) => {\n            const reader = new FileReader();\n            reader.readAsDataURL(file);\n            reader.onload = () => resolve(reader.result as string);\n            reader.onerror = (error) => reject(error);\n        });\n    }\n\n    static normalizeText(text: string): string {\n        return text\n            ?.normalize('NFD') // Descompone los caracteres con acentos\n            ?.replace(/[\\u0300-\\u036f]/g, '') // Elimina los acentos\n            ?.toLowerCase();\n    }\n\n    static mapButtons(\n        button: any | any[] | undefined | null,\n        buttonType?: string\n    ): (BabylonButtonI | undefined)[] | undefined {\n        if (Array.isArray(button)) {\n            return Utils.sortButtonsByOrder(\n                button?.map((btn) =>\n                    btn.label?.length ||\n                    btn.linkValue?.length ||\n                    btn.link_value?.length\n                        ? ({\n                              type: buttonType ?? undefined,\n                              label: btn?.label ?? undefined,\n                              linkType: Utils.formatLinkType(\n                                  btn?.linkType ?? btn?.link_type\n                              ),\n                              url: (() => {\n                                  const rawLink =\n                                      btn.linkValue ?? btn.link_value;\n                                  if (rawLink && typeof rawLink === 'string') {\n                                      const trimmedValue = rawLink?.endsWith(\n                                          '/'\n                                      )\n                                          ? rawLink.slice(0, -1)\n                                          : rawLink;\n\n                                      if (/^\\+?[0-9]+$/.test(trimmedValue)) {\n                                          // Es un teléfono\n                                          return trimmedValue.startsWith('tel:')\n                                              ? trimmedValue\n                                              : `tel:${trimmedValue}`;\n                                      } else if (\n                                          /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(\n                                              trimmedValue\n                                          )\n                                      ) {\n                                          // Es un correo\n                                          return trimmedValue.startsWith(\n                                              'mailto:'\n                                          )\n                                              ? trimmedValue\n                                              : `mailto:${trimmedValue}`;\n                                      }\n\n                                      return trimmedValue;\n                                  }\n                                  return undefined;\n                              })(),\n                              icon: btn.icon ? `icon-` + btn?.icon : undefined,\n                              order: btn.order?.toString(),\n                              vars: btn.linkVars?.toString(),\n                          } as BabylonButtonI)\n                        : undefined\n                ),\n                { keepOriginal: true }\n            );\n        } else {\n            return undefined;\n        }\n    }\n\n    static loadScript$(src: string): Observable<void> {\n        return new Observable<void>((observer) => {\n            const existing = document.querySelector(`script[src=\"${src}\"]`);\n            if (existing) {\n                observer.next();\n                observer.complete();\n                return;\n            }\n\n            const script = document.createElement('script');\n            script.src = src;\n            script.type = 'text/javascript';\n            script.defer = true;\n            script.async = true;\n\n            script.onload = () => {\n                observer.next();\n                observer.complete();\n            };\n\n            script.onerror = (err) => {\n                observer.error(new Error(`Error loading script: ${src}`));\n            };\n\n            document.head.appendChild(script);\n        });\n    }\n\n    static loadStyle$(href: string): Observable<boolean> {\n        return new Observable<boolean>((observer) => {\n            const existing = document.querySelector(`link[href=\"${href}\"]`);\n            if (existing) {\n                observer.next(true);\n                observer.complete();\n                return;\n            }\n\n            const link = document.createElement('link');\n            link.rel = 'preload';\n            link.href = href;\n            link.as = 'style';\n            link.onload = () => {\n                link.onload = null;\n\n                link.rel = 'stylesheet';\n\n                observer.next(true);\n                observer.complete();\n            };\n            link.onerror = () => {\n                observer.error(\n                    new Error(`No se pudo precargar el estilo: ${href}`)\n                );\n            };\n\n            document.head.appendChild(link);\n\n            // opcional: fallback <noscript> (no lo necesita Angular pero lo puedes agregar)\n            const noscript = document.createElement('noscript');\n            noscript.innerHTML = `<link rel=\"stylesheet\" href=\"${href}\">`;\n            document.head.appendChild(noscript);\n        });\n    }\n\n    static safeOpenExternal(url: string): void {\n        if (typeof window === 'undefined' || !url) return;\n\n        const win = window.open(url, '_blank', 'noopener,noreferrer');\n        if (win) return;\n\n        // Fallback si bloquean popups\n        const a = document.createElement('a');\n        a.href = url;\n        a.target = '_blank';\n        a.rel = 'noopener noreferrer';\n        a.style.display = 'none';\n        document.body.appendChild(a);\n        a.click();\n        a.remove();\n    }\n\n    static extractNumberRange(\n        value?: string | null,\n        fallback: number[] = []\n    ): number[] {\n        if (!value) return fallback;\n\n        const match = value.match(/\\d+/);\n        if (!match) return fallback;\n\n        const n = Number(match[0]);\n        if (Number.isNaN(n) || n <= 0) return fallback;\n\n        return Array.from({ length: n }, (_, i) => i + 1);\n    }\n\n    static sortButtonsByOrder<T extends { order?: any }>(\n        buttons?: Array<T | undefined | null> | null,\n        opts?: { keepOriginal?: boolean }\n    ): T[] {\n        const arr = (buttons ?? []).filter((b): b is T => !!b);\n        if (!arr.length) return [];\n\n        const list = opts?.keepOriginal ? [...arr] : arr;\n\n        const toNum = (v: any): number => {\n            const n = Number(v);\n            return Number.isFinite(n) ? n : Number.POSITIVE_INFINITY;\n        };\n\n        return list.sort((a, b) => toNum(a.order) - toNum(b.order));\n    }\n\n    static toSlug(input: string): string {\n        return (input ?? '')\n            .toString()\n            .trim()\n            .toLowerCase()\n            .normalize('NFD')\n            .replace(/[\\u0300-\\u036f]/g, '') // quita tildes\n            .replace(/&/g, 'and') // opcional\n            .replace(/[^a-z0-9\\s-]/g, '') // quita caracteres raros\n            .replace(/\\s+/g, '-') // espacios -> -\n            .replace(/-+/g, '-') // colapsa ---\n            .replace(/^-|-$/g, ''); // quita - al inicio/fin\n    }\n\n    static uniqueSlug(label: string, used: Set<string>): string {\n        const base = this.toSlug(label) || 'item';\n        if (!used.has(base)) {\n            used.add(base);\n            return base;\n        }\n\n        let i = 2;\n        while (used.has(`${base}-${i}`)) i++;\n        const slug = `${base}-${i}`;\n        used.add(slug);\n        return slug;\n    }\n\n    static canonicalHotelTypeSlug(label: string): string {\n        // slug base (sin tildes/raros)\n        let s = Utils.toSlug(label);\n\n        // normaliza plurales comunes ES (hotel(es), apartamento(s))\n        // puedes ampliar aquí si tienes \"apartamentos turísticos\", etc.\n        if (s === 'hoteles') return 'hotel';\n        if (s === 'apartamentos') return 'apartamento';\n\n        // regla genérica suave: si termina en 's' y no es 'dos/tres/...' etc.\n        // (evita cargarte cosas como \"pools\" si existieran, pero aquí aplica bien)\n        if (s.endsWith('s')) s = s.slice(0, -1);\n\n        return s;\n    }\n}\n"]}
|