ctt-babylon 0.19.47 → 0.19.48

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.
@@ -56,9 +56,8 @@ export class TopC2ImgTxtComponent {
56
56
  }
57
57
  }
58
58
  get reverseLayout() {
59
- const images = this._normalizedMultimedia?.imagenes ?? [];
60
- const first = this.getImageByOrder(1, images);
61
- const second = this.getImageByOrder(2, images);
59
+ // Usar el mismo par de imágenes que se muestran (consecutivas)
60
+ const [first, second] = this.getConsecutiveImages();
62
61
  if (!this.hasText(first) && this.hasText(second)) {
63
62
  return true;
64
63
  }
@@ -94,15 +93,38 @@ export class TopC2ImgTxtComponent {
94
93
  * Devuelve la imagen principal (izquierda en layout normal, derecha si reverseLayout=true).
95
94
  * El HTML siempre usa firstImage para la columna de contenido/texto.
96
95
  */
96
+ /**
97
+ * Encuentra el par de imágenes consecutivas más bajo (por ejemplo, 1-2, 2-3, etc.)
98
+ * Si no hay pares consecutivos, usa los dos primeros ordenados.
99
+ */
100
+ getConsecutiveImages() {
101
+ const images = this.orderedImages;
102
+ if (images.length < 2)
103
+ return [images[0], undefined];
104
+ // search the lowest consecutive pair (1-2, 2-3, etc.)
105
+ for (let i = 0; i < images.length - 1; i++) {
106
+ const orderA = images[i]?.order;
107
+ const orderB = images[i + 1]?.order;
108
+ const numA = typeof orderA === 'number' ? orderA : (orderA !== undefined ? parseInt(orderA, 10) : undefined);
109
+ const numB = typeof orderB === 'number' ? orderB : (orderB !== undefined ? parseInt(orderB, 10) : undefined);
110
+ if (typeof numA === 'number' && typeof numB === 'number' && numB - numA === 1) {
111
+ return [images[i], images[i + 1]];
112
+ }
113
+ }
114
+ // Si no hay pares consecutivos, usar los dos primeros
115
+ return [images[0], images[1]];
116
+ }
97
117
  get firstImage() {
98
- return this.reverseLayout ? this.orderedImages[1] : this.orderedImages[0];
118
+ const [imgA, imgB] = this.getConsecutiveImages();
119
+ return this.reverseLayout ? imgB : imgA;
99
120
  }
100
121
  /**
101
122
  * Devuelve la imagen secundaria (derecha en layout normal, izquierda si reverseLayout=true).
102
123
  * El HTML siempre usa secondImage para la columna de imagen secundaria.
103
124
  */
104
125
  get secondImage() {
105
- return this.reverseLayout ? this.orderedImages[0] : this.orderedImages[1];
126
+ const [imgA, imgB] = this.getConsecutiveImages();
127
+ return this.reverseLayout ? imgA : imgB;
106
128
  }
107
129
  get mediaImage() {
108
130
  const first = this.normalizedMultimedia?.imagenes?.[0];
@@ -157,4 +179,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
157
179
  }], texts: [{
158
180
  type: Input
159
181
  }] } });
160
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"top-c2-img-txt.component.js","sourceRoot":"","sources":["../../../../../../../../projects/babylon/src/lib/components/external/core/top-c2-img-txt/top-c2-img-txt.component.ts","../../../../../../../../projects/babylon/src/lib/components/external/core/top-c2-img-txt/top-c2-img-txt.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, ViewEncapsulation } from '@angular/core';\nimport { BabylonLinkTypeDirective } from '../../../../directives/link-type/link-type.directive';\nimport {\n    BabylonButtonI,\n    BabylonImageI,\n    BabylonMultimediaI,\n    BabylonRoomDetailI,\n    BabylonTextInfoI,\n} from '../../../../interfaces';\nimport { getOrderedImages, getImagesByTagFromMultimedia } from '../../../../utils/mutimedia.utils';\nimport { TopC2ImgTxtContent } from './top-c2-img-txt.interfaces';\n\n@Component({\n    selector: 'top-c2-img-txt',\n    standalone: true,\n    imports: [CommonModule, BabylonLinkTypeDirective],\n    templateUrl: './top-c2-img-txt.component.html',\n    styleUrl: './top-c2-img-txt.component.scss',\n    encapsulation: ViewEncapsulation.None,\n})\nexport class TopC2ImgTxtComponent {\n    @Input() identifier?: string;\n    private _multimedia?: BabylonMultimediaI | BabylonRoomDetailI;\n\n    @Input()\n    set multimedia(value: BabylonMultimediaI | BabylonRoomDetailI | undefined) {\n        this._multimedia = value;\n        this.updateNormalizedMultimedia();\n    }\n\n    get multimedia(): BabylonMultimediaI | BabylonRoomDetailI | undefined {\n        return this._multimedia;\n    }\n    @Input() buttons?: BabylonButtonI[];\n    @Input() content?: TopC2ImgTxtContent;\n    @Input() reverse = false;\n    @Input() imageOrder?: number | string;\n    @Input() texts?: BabylonTextInfoI;\n\n    private _normalizedMultimedia?: BabylonMultimediaI | undefined;\n    private readonly imageErrors = new Set<string>();\n\n    private updateNormalizedMultimedia(): void {\n        const multimedia = this._multimedia;\n        if (!multimedia) {\n            this._normalizedMultimedia = undefined;\n            return;\n        }\n        const TAG = 'TopC2ImgTxt';\n        if (this.isRoomDetail(multimedia)) {\n            let imagenes = (multimedia.multimedia?.imagenes ?? multimedia.images ?? []) as BabylonImageI[];\n            imagenes = getImagesByTagFromMultimedia({ imagenes }, TAG);\n            const [first, ...rest] = imagenes;\n            const contentImage: BabylonImageI | undefined = first\n                ? {\n                      ...first,\n                      title: first.title || multimedia.texts?.name,\n                      description:\n                          first.description || multimedia.texts?.description,\n                  }\n                : undefined;\n            this.texts = multimedia.texts;\n            this._normalizedMultimedia = {\n                imagenes: contentImage ? [contentImage, ...rest] : imagenes\n            };\n            return;\n        }\n        // Si no es RoomDetail, NO filtramos por tag, solo devolvemos las imágenes tal cual\n        if ('imagenes' in multimedia && Array.isArray((multimedia as any).imagenes)) {\n            this._normalizedMultimedia = {\n                ...multimedia,\n                imagenes: (multimedia as any).imagenes\n            };\n            return;\n        }\n        this._normalizedMultimedia = multimedia;\n    }\n\n    ngOnChanges(changes: any): void {\n        if (changes['multimedia']) {\n            this.updateNormalizedMultimedia();\n        }\n    }\n\n    get reverseLayout(): boolean {\n        const images = this._normalizedMultimedia?.imagenes ?? [];\n        const first = this.getImageByOrder(1, images);\n        const second = this.getImageByOrder(2, images);\n        if (!this.hasText(first) && this.hasText(second)) {\n            return true;\n        }\n        return this.content?.reverse ?? this.reverse;\n    }\n\n    get contentImage(): BabylonImageI | undefined {\n        const images = this.normalizedMultimedia?.imagenes ?? [];\n        const first = this.getImageByOrder(1, images);\n        const second = this.getImageByOrder(2, images);\n        if (this.hasText(first)) return first;\n        if (this.hasText(second)) return second;\n        return first ?? second;\n    }\n\n    private getImageByOrder(\n        order: number | string | undefined,\n        images: BabylonImageI[]\n    ): BabylonImageI | undefined {\n        if (order == null) {\n            return images[0];\n        }\n        return images.find((img) => {\n            const imgOrder = (img as any).order;\n            return imgOrder != null && imgOrder.toString() === order.toString();\n        });\n    }\n\n    /**\n     * Devuelve las imágenes filtradas por tag y ordenadas por order ascendente.\n     * Se usa como base para mostrar imágenes en el HTML, asegurando que siempre estén en el orden correcto.\n     */\n    get orderedImages(): BabylonImageI[] {\n        return getOrderedImages(this.normalizedMultimedia?.imagenes ?? []);\n    }\n\n    /**\n     * Devuelve la imagen principal (izquierda en layout normal, derecha si reverseLayout=true).\n     * El HTML siempre usa firstImage para la columna de contenido/texto.\n     */\n    get firstImage(): BabylonImageI | undefined {\n        return this.reverseLayout ? this.orderedImages[1] : this.orderedImages[0];\n    }\n\n    /**\n     * Devuelve la imagen secundaria (derecha en layout normal, izquierda si reverseLayout=true).\n     * El HTML siempre usa secondImage para la columna de imagen secundaria.\n     */\n    get secondImage(): BabylonImageI | undefined {\n        return this.reverseLayout ? this.orderedImages[0] : this.orderedImages[1];\n    }\n\n    get mediaImage(): BabylonImageI | undefined {\n        const first = this.normalizedMultimedia?.imagenes?.[0];\n        const second = this.normalizedMultimedia?.imagenes?.[1];\n\n        if (this.contentImage === first) return second ?? first;\n        if (this.contentImage === second) return first ?? second;\n\n        return second ?? first;\n    }\n\n    get ctaButton(): BabylonButtonI | undefined {\n        return this.contentImage?.buttons?.[0] ?? this.buttons?.[0];\n    }\n\n    private hasText(image?: BabylonImageI): boolean {\n        return !!(\n            image?.title?.trim() ||\n            image?.subtitle?.trim() ||\n            image?.description?.trim()\n        );\n    }\n\n    private isRoomDetail(\n        value: BabylonMultimediaI | BabylonRoomDetailI\n    ): value is BabylonRoomDetailI {\n        return 'images' in value || 'texts' in value;\n    }\n\n    get normalizedMultimedia(): BabylonMultimediaI | undefined {\n        return this._normalizedMultimedia;\n    }\n\n    hasImageSrc(image?: BabylonImageI): boolean {\n        return !!image?.img?.src?.trim();\n    }\n\n    hasImageError(key: 'first' | 'second'): boolean {\n        return this.imageErrors.has(key);\n    }\n\n    onImageError(key: 'first' | 'second'): void {\n        this.imageErrors.add(key);\n    }\n}\n","@if (firstImage || secondImage) {\n    <section\n        class=\"mdl-topc2imgtext\"\n        [class.reverse]=\"reverseLayout\"\n        [attr.id]=\"identifier\"\n    >\n        <div class=\"mdl-container\">\n            <div class=\"m-columleft\">\n                <div class=\"m-content\">\n                    @if (firstImage?.title || texts?.title || texts?.name) {\n                        <div\n                            class=\"m-titulo\"\n                            [innerHTML]=\"firstImage?.title || texts?.title || texts?.name\"\n                        ></div>\n                    }\n                    @if (firstImage?.subtitle || texts?.subtitle) {\n                        <div class=\"m-subtitulo\">\n                            {{ firstImage?.subtitle || texts?.subtitle }}\n                        </div>\n                    }\n                    <div class=\"m-image\" *ngIf=\"firstImage\">\n                        @if (hasImageSrc(firstImage) && !hasImageError('first')) {\n                            <img\n                                [src]=\"firstImage?.img?.src\"\n                                [alt]=\"\n                                    firstImage?.img?.alt ?? firstImage?.alt ?? ''\n                                \"\n                                loading=\"lazy\"\n                                (error)=\"onImageError('first')\"\n                            />\n                        } @else {\n                            <div class=\"m-image-placeholder\" aria-hidden=\"true\"></div>\n                        }\n                    </div>\n                    @if (firstImage?.description || texts?.description) {\n                        <div\n                            class=\"m-texto\"\n                            [innerHTML]=\"\n                                firstImage?.description || texts?.description\n                            \"\n                        ></div>\n                    }\n                    @if (ctaButton; as btn) {\n                        <a\n                            class=\"m-buttonD\"\n                            [href]=\"btn?.url\"\n                            [linkType]=\"btn?.linkType\"\n                            [attr.target]=\"btn?.target || '_self'\"\n                            [attr.rel]=\"\n                                (btn?.target || '_self') === '_blank'\n                                    ? 'noopener noreferrer'\n                                    : null\n                            \"\n                            [attr.aria-label]=\"btn?.label\"\n                        >\n                            {{ btn?.label }}\n                        </a>\n                    }\n                </div>\n            </div>\n            <div class=\"m-columright\" *ngIf=\"secondImage\">\n                <div class=\"m-image\">\n                    @if (hasImageSrc(secondImage) && !hasImageError('second')) {\n                        <img\n                            [src]=\"secondImage?.img?.src\"\n                            [alt]=\"secondImage?.img?.alt ?? secondImage?.alt ?? ''\"\n                            loading=\"lazy\"\n                            (error)=\"onImageError('second')\"\n                        />\n                    } @else {\n                        <div class=\"m-image-placeholder\" aria-hidden=\"true\"></div>\n                    }\n                </div>\n            </div>\n        </div>\n    </section>\n}\n"]}
182
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"top-c2-img-txt.component.js","sourceRoot":"","sources":["../../../../../../../../projects/babylon/src/lib/components/external/core/top-c2-img-txt/top-c2-img-txt.component.ts","../../../../../../../../projects/babylon/src/lib/components/external/core/top-c2-img-txt/top-c2-img-txt.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, ViewEncapsulation } from '@angular/core';\nimport { BabylonLinkTypeDirective } from '../../../../directives/link-type/link-type.directive';\nimport {\n    BabylonButtonI,\n    BabylonImageI,\n    BabylonMultimediaI,\n    BabylonRoomDetailI,\n    BabylonTextInfoI,\n} from '../../../../interfaces';\nimport { getOrderedImages, getImagesByTagFromMultimedia } from '../../../../utils/mutimedia.utils';\nimport { TopC2ImgTxtContent } from './top-c2-img-txt.interfaces';\n\n@Component({\n    selector: 'top-c2-img-txt',\n    standalone: true,\n    imports: [CommonModule, BabylonLinkTypeDirective],\n    templateUrl: './top-c2-img-txt.component.html',\n    styleUrl: './top-c2-img-txt.component.scss',\n    encapsulation: ViewEncapsulation.None,\n})\nexport class TopC2ImgTxtComponent {\n    @Input() identifier?: string;\n    private _multimedia?: BabylonMultimediaI | BabylonRoomDetailI;\n\n    @Input()\n    set multimedia(value: BabylonMultimediaI | BabylonRoomDetailI | undefined) {\n        this._multimedia = value;\n        this.updateNormalizedMultimedia();\n    }\n\n    get multimedia(): BabylonMultimediaI | BabylonRoomDetailI | undefined {\n        return this._multimedia;\n    }\n    @Input() buttons?: BabylonButtonI[];\n    @Input() content?: TopC2ImgTxtContent;\n    @Input() reverse = false;\n    @Input() imageOrder?: number | string;\n    @Input() texts?: BabylonTextInfoI;\n\n    private _normalizedMultimedia?: BabylonMultimediaI | undefined;\n    private readonly imageErrors = new Set<string>();\n\n    private updateNormalizedMultimedia(): void {\n        const multimedia = this._multimedia;\n        if (!multimedia) {\n            this._normalizedMultimedia = undefined;\n            return;\n        }\n        const TAG = 'TopC2ImgTxt';\n        if (this.isRoomDetail(multimedia)) {\n            let imagenes = (multimedia.multimedia?.imagenes ?? multimedia.images ?? []) as BabylonImageI[];\n            imagenes = getImagesByTagFromMultimedia({ imagenes }, TAG);\n            const [first, ...rest] = imagenes;\n            const contentImage: BabylonImageI | undefined = first\n                ? {\n                      ...first,\n                      title: first.title || multimedia.texts?.name,\n                      description:\n                          first.description || multimedia.texts?.description,\n                  }\n                : undefined;\n            this.texts = multimedia.texts;\n            this._normalizedMultimedia = {\n                imagenes: contentImage ? [contentImage, ...rest] : imagenes\n            };\n            return;\n        }\n        // Si no es RoomDetail, NO filtramos por tag, solo devolvemos las imágenes tal cual\n        if ('imagenes' in multimedia && Array.isArray((multimedia as any).imagenes)) {\n            this._normalizedMultimedia = {\n                ...multimedia,\n                imagenes: (multimedia as any).imagenes\n            };\n            return;\n        }\n        this._normalizedMultimedia = multimedia;\n    }\n\n    ngOnChanges(changes: any): void {\n        if (changes['multimedia']) {\n            this.updateNormalizedMultimedia();\n        }\n    }\n\n    get reverseLayout(): boolean {\n        // Usar el mismo par de imágenes que se muestran (consecutivas)\n        const [first, second] = this.getConsecutiveImages();\n        if (!this.hasText(first) && this.hasText(second)) {\n            return true;\n        }\n        return this.content?.reverse ?? this.reverse;\n    }\n\n    get contentImage(): BabylonImageI | undefined {\n        const images = this.normalizedMultimedia?.imagenes ?? [];\n        const first = this.getImageByOrder(1, images);\n        const second = this.getImageByOrder(2, images);\n        if (this.hasText(first)) return first;\n        if (this.hasText(second)) return second;\n        return first ?? second;\n    }\n\n    private getImageByOrder(\n        order: number | string | undefined,\n        images: BabylonImageI[]\n    ): BabylonImageI | undefined {\n        if (order == null) {\n            return images[0];\n        }\n        return images.find((img) => {\n            const imgOrder = (img as any).order;\n            return imgOrder != null && imgOrder.toString() === order.toString();\n        });\n    }\n\n    /**\n     * Devuelve las imágenes filtradas por tag y ordenadas por order ascendente.\n     * Se usa como base para mostrar imágenes en el HTML, asegurando que siempre estén en el orden correcto.\n     */\n    get orderedImages(): BabylonImageI[] {\n        return getOrderedImages(this.normalizedMultimedia?.imagenes ?? []);\n    }\n\n    /**\n     * Devuelve la imagen principal (izquierda en layout normal, derecha si reverseLayout=true).\n     * El HTML siempre usa firstImage para la columna de contenido/texto.\n     */\n    /**\n     * Encuentra el par de imágenes consecutivas más bajo (por ejemplo, 1-2, 2-3, etc.)\n     * Si no hay pares consecutivos, usa los dos primeros ordenados.\n     */\n    private getConsecutiveImages(): [BabylonImageI | undefined, BabylonImageI | undefined] {\n        const images = this.orderedImages;\n        if (images.length < 2) return [images[0], undefined];\n        // search the lowest consecutive pair (1-2, 2-3, etc.)\n        for (let i = 0; i < images.length - 1; i++) {\n            const orderA = images[i]?.order;\n            const orderB = images[i + 1]?.order;\n            const numA = typeof orderA === 'number' ? orderA : (orderA !== undefined ? parseInt(orderA as any, 10) : undefined);\n            const numB = typeof orderB === 'number' ? orderB : (orderB !== undefined ? parseInt(orderB as any, 10) : undefined);\n            if (typeof numA === 'number' && typeof numB === 'number' && numB - numA === 1) {\n                return [images[i], images[i + 1]];\n            }\n        }\n        // Si no hay pares consecutivos, usar los dos primeros\n        return [images[0], images[1]];\n    }\n\n    get firstImage(): BabylonImageI | undefined {\n        const [imgA, imgB] = this.getConsecutiveImages();\n        return this.reverseLayout ? imgB : imgA;\n    }\n\n    /**\n     * Devuelve la imagen secundaria (derecha en layout normal, izquierda si reverseLayout=true).\n     * El HTML siempre usa secondImage para la columna de imagen secundaria.\n     */\n    get secondImage(): BabylonImageI | undefined {\n        const [imgA, imgB] = this.getConsecutiveImages();\n        return this.reverseLayout ? imgA : imgB;\n    }\n\n    get mediaImage(): BabylonImageI | undefined {\n        const first = this.normalizedMultimedia?.imagenes?.[0];\n        const second = this.normalizedMultimedia?.imagenes?.[1];\n\n        if (this.contentImage === first) return second ?? first;\n        if (this.contentImage === second) return first ?? second;\n\n        return second ?? first;\n    }\n\n    get ctaButton(): BabylonButtonI | undefined {\n        return this.contentImage?.buttons?.[0] ?? this.buttons?.[0];\n    }\n\n    private hasText(image?: BabylonImageI): boolean {\n        return !!(\n            image?.title?.trim() ||\n            image?.subtitle?.trim() ||\n            image?.description?.trim()\n        );\n    }\n\n    private isRoomDetail(\n        value: BabylonMultimediaI | BabylonRoomDetailI\n    ): value is BabylonRoomDetailI {\n        return 'images' in value || 'texts' in value;\n    }\n\n    get normalizedMultimedia(): BabylonMultimediaI | undefined {\n        return this._normalizedMultimedia;\n    }\n\n    hasImageSrc(image?: BabylonImageI): boolean {\n        return !!image?.img?.src?.trim();\n    }\n\n    hasImageError(key: 'first' | 'second'): boolean {\n        return this.imageErrors.has(key);\n    }\n\n    onImageError(key: 'first' | 'second'): void {\n        this.imageErrors.add(key);\n    }\n}\n","@if (firstImage || secondImage) {\n    <section\n        class=\"mdl-topc2imgtext\"\n        [class.reverse]=\"reverseLayout\"\n        [attr.id]=\"identifier\"\n    >\n        <div class=\"mdl-container\">\n            <div class=\"m-columleft\">\n                <div class=\"m-content\">\n                    @if (firstImage?.title || texts?.title || texts?.name) {\n                        <div\n                            class=\"m-titulo\"\n                            [innerHTML]=\"firstImage?.title || texts?.title || texts?.name\"\n                        ></div>\n                    }\n                    @if (firstImage?.subtitle || texts?.subtitle) {\n                        <div class=\"m-subtitulo\">\n                            {{ firstImage?.subtitle || texts?.subtitle }}\n                        </div>\n                    }\n                    <div class=\"m-image\" *ngIf=\"firstImage\">\n                        @if (hasImageSrc(firstImage) && !hasImageError('first')) {\n                            <img\n                                [src]=\"firstImage?.img?.src\"\n                                [alt]=\"\n                                    firstImage?.img?.alt ?? firstImage?.alt ?? ''\n                                \"\n                                loading=\"lazy\"\n                                (error)=\"onImageError('first')\"\n                            />\n                        } @else {\n                            <div class=\"m-image-placeholder\" aria-hidden=\"true\"></div>\n                        }\n                    </div>\n                    @if (firstImage?.description || texts?.description) {\n                        <div\n                            class=\"m-texto\"\n                            [innerHTML]=\"\n                                firstImage?.description || texts?.description\n                            \"\n                        ></div>\n                    }\n                    @if (ctaButton; as btn) {\n                        <a\n                            class=\"m-buttonD\"\n                            [href]=\"btn?.url\"\n                            [linkType]=\"btn?.linkType\"\n                            [attr.target]=\"btn?.target || '_self'\"\n                            [attr.rel]=\"\n                                (btn?.target || '_self') === '_blank'\n                                    ? 'noopener noreferrer'\n                                    : null\n                            \"\n                            [attr.aria-label]=\"btn?.label\"\n                        >\n                            {{ btn?.label }}\n                        </a>\n                    }\n                </div>\n            </div>\n            <div class=\"m-columright\" *ngIf=\"secondImage\">\n                <div class=\"m-image\">\n                    @if (hasImageSrc(secondImage) && !hasImageError('second')) {\n                        <img\n                            [src]=\"secondImage?.img?.src\"\n                            [alt]=\"secondImage?.img?.alt ?? secondImage?.alt ?? ''\"\n                            loading=\"lazy\"\n                            (error)=\"onImageError('second')\"\n                        />\n                    } @else {\n                        <div class=\"m-image-placeholder\" aria-hidden=\"true\"></div>\n                    }\n                </div>\n            </div>\n        </div>\n    </section>\n}\n"]}
@@ -29869,9 +29869,8 @@ class TopC2ImgTxtComponent {
29869
29869
  }
29870
29870
  }
29871
29871
  get reverseLayout() {
29872
- const images = this._normalizedMultimedia?.imagenes ?? [];
29873
- const first = this.getImageByOrder(1, images);
29874
- const second = this.getImageByOrder(2, images);
29872
+ // Usar el mismo par de imágenes que se muestran (consecutivas)
29873
+ const [first, second] = this.getConsecutiveImages();
29875
29874
  if (!this.hasText(first) && this.hasText(second)) {
29876
29875
  return true;
29877
29876
  }
@@ -29907,15 +29906,38 @@ class TopC2ImgTxtComponent {
29907
29906
  * Devuelve la imagen principal (izquierda en layout normal, derecha si reverseLayout=true).
29908
29907
  * El HTML siempre usa firstImage para la columna de contenido/texto.
29909
29908
  */
29909
+ /**
29910
+ * Encuentra el par de imágenes consecutivas más bajo (por ejemplo, 1-2, 2-3, etc.)
29911
+ * Si no hay pares consecutivos, usa los dos primeros ordenados.
29912
+ */
29913
+ getConsecutiveImages() {
29914
+ const images = this.orderedImages;
29915
+ if (images.length < 2)
29916
+ return [images[0], undefined];
29917
+ // search the lowest consecutive pair (1-2, 2-3, etc.)
29918
+ for (let i = 0; i < images.length - 1; i++) {
29919
+ const orderA = images[i]?.order;
29920
+ const orderB = images[i + 1]?.order;
29921
+ const numA = typeof orderA === 'number' ? orderA : (orderA !== undefined ? parseInt(orderA, 10) : undefined);
29922
+ const numB = typeof orderB === 'number' ? orderB : (orderB !== undefined ? parseInt(orderB, 10) : undefined);
29923
+ if (typeof numA === 'number' && typeof numB === 'number' && numB - numA === 1) {
29924
+ return [images[i], images[i + 1]];
29925
+ }
29926
+ }
29927
+ // Si no hay pares consecutivos, usar los dos primeros
29928
+ return [images[0], images[1]];
29929
+ }
29910
29930
  get firstImage() {
29911
- return this.reverseLayout ? this.orderedImages[1] : this.orderedImages[0];
29931
+ const [imgA, imgB] = this.getConsecutiveImages();
29932
+ return this.reverseLayout ? imgB : imgA;
29912
29933
  }
29913
29934
  /**
29914
29935
  * Devuelve la imagen secundaria (derecha en layout normal, izquierda si reverseLayout=true).
29915
29936
  * El HTML siempre usa secondImage para la columna de imagen secundaria.
29916
29937
  */
29917
29938
  get secondImage() {
29918
- return this.reverseLayout ? this.orderedImages[0] : this.orderedImages[1];
29939
+ const [imgA, imgB] = this.getConsecutiveImages();
29940
+ return this.reverseLayout ? imgA : imgB;
29919
29941
  }
29920
29942
  get mediaImage() {
29921
29943
  const first = this.normalizedMultimedia?.imagenes?.[0];