@colijnit/product 260.1.2 → 260.1.3

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.
@@ -2,15 +2,19 @@ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, Host
2
2
  import { CoDocument } from '@colijnit/mainapi/build/model/co-document.bo';
3
3
  import { ProductConnectorService } from '../../service/product-connector.service';
4
4
  import { ProductEventService } from '../../service/product-event.service';
5
+ import { IconEnum } from '../../enum/icon.enum';
5
6
  import { DomSanitizer } from '@angular/platform-browser';
7
+ import { IconCacheService } from '../../service/icon-cache.service';
6
8
  export class ImageViewModel {
7
9
  }
8
10
  export class ImageCarouselComponent {
9
- constructor(_ione, _appEventService, _changeDetector, _domSanitizer) {
11
+ constructor(iconCacheService, _ione, _appEventService, _changeDetector, _domSanitizer) {
12
+ this.iconCacheService = iconCacheService;
10
13
  this._ione = _ione;
11
14
  this._appEventService = _appEventService;
12
15
  this._changeDetector = _changeDetector;
13
16
  this._domSanitizer = _domSanitizer;
17
+ this.icons = IconEnum;
14
18
  this.isPopupOpen = false;
15
19
  this.showRefresh = false;
16
20
  this.resizing = false;
@@ -148,51 +152,63 @@ export class ImageCarouselComponent {
148
152
  const ctx = resizeCanvas.getContext('2d');
149
153
  const img = document.createElement('img');
150
154
  img.crossOrigin = 'anonymous';
155
+ const handleFallback = () => {
156
+ // When a CDN does not accept our tainted canvas, we fallback to the original source.
157
+ imageViewModel.source = this._domSanitizer.bypassSecurityTrustUrl(source);
158
+ imageViewModel.originalSource = source;
159
+ this._changeDetector.detectChanges();
160
+ };
161
+ img.onerror = () => {
162
+ handleFallback();
163
+ };
151
164
  img.onload = () => {
152
- ctx.imageSmoothingEnabled = true;
153
- ctx.imageSmoothingQuality = 'high';
154
- const ow = img.width;
155
- const oh = img.height;
156
- const aspect = ow / oh;
157
- let newW = this._resizeCanvasHeight;
158
- let newH = this._resizeCanvasHeight;
159
- if (ow > oh) {
160
- newH = this._resizeCanvasHeight / aspect;
165
+ try {
166
+ ctx.imageSmoothingEnabled = true;
167
+ ctx.imageSmoothingQuality = 'high';
168
+ const aspect = img.width / img.height;
169
+ let newW = this._resizeCanvasHeight;
170
+ let newH = this._resizeCanvasHeight;
171
+ if (img.width > img.height) {
172
+ newH = this._resizeCanvasHeight / aspect;
173
+ }
174
+ else {
175
+ newW = this._resizeCanvasHeight * aspect;
176
+ }
177
+ resizeCanvas.width = Math.round(newW);
178
+ resizeCanvas.height = Math.round(newH);
179
+ ctx.clearRect(0, 0, resizeCanvas.width, resizeCanvas.height);
180
+ ctx.drawImage(img, 0, 0, resizeCanvas.width, resizeCanvas.height);
181
+ const mime = this._handleMimeTypes(source);
182
+ const resizedSource = resizeCanvas.toDataURL(mime);
183
+ imageViewModel.source = this._domSanitizer.bypassSecurityTrustUrl(resizedSource);
184
+ imageViewModel.originalSource = source;
185
+ this._changeDetector.detectChanges();
161
186
  }
162
- else {
163
- newW = this._resizeCanvasHeight * aspect;
187
+ catch (error) {
188
+ handleFallback();
164
189
  }
165
- resizeCanvas.width = Math.round(newW);
166
- resizeCanvas.height = Math.round(newH);
167
- // Ensure transparent background before drawing
168
- ctx.clearRect(0, 0, resizeCanvas.width, resizeCanvas.height);
169
- ctx.drawImage(img, 0, 0, resizeCanvas.width, resizeCanvas.height);
170
- const mime = this._detectPreferredMime(source);
171
- const resizedSource = mime === 'image/jpeg'
172
- ? resizeCanvas.toDataURL('image/jpeg', 0.92) // only if original was JPEG
173
- : resizeCanvas.toDataURL(mime); // PNG/WebP keep alpha
174
- imageViewModel.source = this._domSanitizer.bypassSecurityTrustUrl(resizedSource);
175
- imageViewModel.originalSource = source;
176
- this._changeDetector.detectChanges();
177
190
  };
178
191
  img.src = source;
179
192
  }
180
- _detectPreferredMime(source) {
181
- // Data URI check
182
- const m = source.match(/^data:(image\/[a-zA-Z+.-]+);base64,/);
183
- if (m) {
184
- const t = m[1].toLowerCase();
185
- if (t === 'image/png' || t === 'image/webp' || t === 'image/jpeg')
186
- return t;
193
+ _handleMimeTypes(imageSource) {
194
+ // When we got base64 data
195
+ if (imageSource.startsWith('data:')) {
196
+ const match = imageSource.match(/^data:(image\/(png|webp|jpeg|jpg))/i);
197
+ return match ? match[1].toLowerCase() : 'image/png';
198
+ }
199
+ // when we got an URL.
200
+ const extension = imageSource.substring(imageSource.lastIndexOf('.') + 1).toLowerCase();
201
+ switch (extension) {
202
+ case 'png':
203
+ return 'image/png';
204
+ case 'webp':
205
+ return 'image/webp';
206
+ case 'jpg':
207
+ case 'jpeg':
208
+ return 'image/jpeg';
209
+ default:
210
+ return 'image/png';
187
211
  }
188
- const lower = source.toLowerCase();
189
- if (lower.endsWith('.png'))
190
- return 'image/png';
191
- if (lower.endsWith('.webp'))
192
- return 'image/webp';
193
- if (lower.endsWith('.jpg') || lower.endsWith('.jpeg'))
194
- return 'image/jpeg';
195
- return 'image/png';
196
212
  }
197
213
  _scrollCarouselToIndex() {
198
214
  if (this.currentIndex > -1 && this.currentIndex <= this.images.length) {
@@ -211,65 +227,63 @@ ImageCarouselComponent.decorators = [
211
227
  { type: Component, args: [{
212
228
  selector: 'app-image-carousel',
213
229
  template: `
214
- <div id="product_page_carousel">
215
- <div id="product_page_carousel_items">
216
- <co-loader [isShown]="true" *ngIf="showLoader"></co-loader>
217
- <div #carousel class="inner-carousel">
218
- <!-- This has been taken out of the for loop to prevent flashing images when updating. -->
219
- <div *ngIf="imageViewModels[0]" class="carousel-item" [id]="'slide-0'" [class.active]="isCurrentIndex(0)" (click)="handleShowImage(imageViewModels[0])">
220
- <img [src]="imageViewModels[0].source">
221
- </div>
222
- <div *ngFor="let imageViewModel of imageViewModels.slice(1); let index = index" class="carousel-item"
223
- [id]="'slide-' + (index + 1)" [class.active]="isCurrentIndex((index + 1))" (click)="handleShowImage(imageViewModel)">
224
- <img [src]="imageViewModel.source">
225
- </div>
226
- <div class="carousel-scroller-layer" *ngIf="imageViewModels && imageViewModels.length > 1">
227
- <div class="carousel-item-scroller prev" (click)="gotoPrevSlide()" *ngIf="currentIndex > 0"></div>
228
- <div class="carousel-item-scroller next" (click)="gotoNextSlide()" *ngIf="currentIndex < images.length - 1"></div>
229
- </div>
230
- </div>
231
- <!--
232
- <co-icon class="selector-type-icon refresh-button" [iconData]="iconCache.getIcon(icons.Refresh)" (click)="onForceRenderImage()" [class.loading]="showLoader" *ngIf="showRefresh"></co-icon>
233
- -->
230
+ <div id="product_page_carousel">
231
+ <div id="product_page_carousel_items">
232
+ <co-loader [isShown]="true" *ngIf="showLoader"></co-loader>
233
+ <div #carousel class="inner-carousel">
234
+ <!-- This has been taken out of the for loop to prevent flashing images when updating. -->
235
+ <div *ngIf="imageViewModels[0]" class="carousel-item" [id]="'slide-0'" [class.active]="isCurrentIndex(0)"
236
+ (click)="handleShowImage(imageViewModels[0])">
237
+ <img [src]="imageViewModels[0].source">
238
+ </div>
239
+ <div *ngFor="let imageViewModel of imageViewModels.slice(1); let index = index" class="carousel-item"
240
+ [id]="'slide-' + (index + 1)" [class.active]="isCurrentIndex((index + 1))" (click)="handleShowImage(imageViewModel)">
241
+ <img [src]="imageViewModel.source">
234
242
  </div>
243
+ <div class="carousel-scroller-layer" *ngIf="imageViewModels && imageViewModels.length > 1">
244
+ <div class="carousel-item-scroller prev" (click)="gotoPrevSlide()" *ngIf="currentIndex > 0"></div>
245
+ <div class="carousel-item-scroller next" (click)="gotoNextSlide()" *ngIf="currentIndex < images.length - 1"></div>
246
+ </div>
247
+ </div>
248
+ <!--
249
+ <co-icon class="selector-type-icon refresh-button" [iconData]="iconCache.getIcon(icons.Refresh)" (click)="onForceRenderImage()" [class.loading]="showLoader" *ngIf="showRefresh"></co-icon>
250
+ -->
251
+ </div>
235
252
 
236
- <div id="product_page_carousel_thumbs">
237
- <co-scroll-container class="scroll-container" *ngIf="imageViewModels && imageViewModels.length > 1">
238
- <div *ngFor="let imageViewModel of imageViewModels; let index = index" class="carousel-thumb"
239
- [class.active]="index === currentIndex">
240
- <img [src]="imageViewModel.source" (click)="handleThumbClick(index)"/>
241
- </div>
242
- </co-scroll-container>
253
+ <div id="product_page_carousel_thumbs">
254
+ <co-scroll-container class="scroll-container" *ngIf="imageViewModels && imageViewModels.length > 1">
255
+ <div *ngFor="let imageViewModel of imageViewModels; let index = index" class="carousel-thumb"
256
+ [class.active]="index === currentIndex">
257
+ <img [src]="imageViewModel.source" (click)="handleThumbClick(index)"/>
243
258
  </div>
259
+ </co-scroll-container>
244
260
  </div>
245
- <!-- Modal (real popup) -->
246
- <div
247
- class="image-modal"
248
- *ngIf="isPopupOpen"
249
- (click)="closePopup()"
250
- role="dialog"
251
- aria-modal="true"
252
- aria-label="Image preview"
253
- >
254
- <div class="image-modal__content" (click)="$event.stopPropagation()">
255
- <button
256
- class="image-modal__close"
257
- type="button"
258
- aria-label="Close"
259
- (click)="closePopup()"
260
- >
261
- ×
262
- </button>
261
+ </div>
263
262
 
264
- <img [src]="selectedImage?.originalSource" alt="Image preview" />
263
+ <div cdkOverlayOrigin #trigger="cdkOverlayOrigin"></div>
264
+ <ng-template
265
+ cdkConnectedOverlay
266
+ [cdkConnectedOverlayOrigin]="trigger"
267
+ [cdkConnectedOverlayOpen]="isPopupOpen"
268
+ [cdkConnectedOverlayHasBackdrop]="true"
269
+ (backdropClick)="closePopup()">
270
+ <div class="image-modal" role="dialog" aria-modal="true" aria-label="Image preview">
271
+ <div class="image-modal__content">
272
+ <button class="image-modal__close" type="button" aria-label="Close" (click)="closePopup()">
273
+ <co-icon [iconData]="iconCacheService.getIcon(icons.CrossSkinny)"></co-icon>
274
+ </button>
275
+ <img [src]="selectedImage?.originalSource" alt="Image preview"/>
265
276
  </div>
266
277
  </div>
278
+ </ng-template>
279
+
267
280
  `,
268
281
  changeDetection: ChangeDetectionStrategy.OnPush,
269
282
  styles: [":host{max-height:540px;height:100%;position:relative}:host:not(.resizing) .inner-carousel{scroll-behavior:smooth;-webkit-overflow-scrolling:touch;scroll-snap-type:x mandatory}#product_page_carousel{position:relative}#product_page_carousel .refresh-button{position:absolute;bottom:10px;right:10px;background:#fff}#product_page_carousel .refresh-button.loading{animation:spin 1s linear infinite}#product_page_carousel .refresh-button:hover{box-shadow:none;background:#74B77F;transition:all .2s ease-in-out}#product_page_carousel .refresh-button:hover ::ng-deep svg path{fill:#fff!important}#product_page_carousel #product_page_carousel_items{position:relative;margin-bottom:10px}#product_page_carousel #product_page_carousel_items ::ng-deep co-loader{position:absolute}#product_page_carousel .inner-carousel{display:flex;flex-direction:row;align-items:center;overflow:hidden;max-height:500px;border:1px solid #efefef}#product_page_carousel .carousel-item{max-height:500px;width:100%;display:flex;cursor:zoom-in;flex-shrink:0;flex-grow:0}#product_page_carousel .carousel-item img{width:100%;height:auto;-o-object-fit:contain;object-fit:contain}#product_page_carousel .carousel-scroller-layer{height:100%;width:100%;position:absolute;pointer-events:none;top:0;left:0}#product_page_carousel #product_page_carousel_thumbs{display:flex;justify-content:flex-start;height:80px;margin-left:auto;margin-right:auto}#product_page_carousel #product_page_carousel_thumbs ::ng-deep co-scroll-container{padding:0 22px}#product_page_carousel #product_page_carousel_thumbs ::ng-deep co-scroll-container .content-wrapper{padding:0}#product_page_carousel #product_page_carousel_thumbs .carousel-thumb{opacity:1;cursor:pointer;transition:all .2s ease;padding:4px;border:1px solid #f6f5f4}#product_page_carousel #product_page_carousel_thumbs .carousel-thumb.active,#product_page_carousel #product_page_carousel_thumbs .carousel-thumb:hover{border-color:#22313c}#product_page_carousel #product_page_carousel_thumbs .carousel-thumb:not(:last-child){margin-right:10px}#product_page_carousel #product_page_carousel_thumbs .carousel-thumb img{height:68px}.image-modal{position:fixed;inset:0;background:rgba(0,0,0,.8);display:flex;align-items:center;justify-content:center;z-index:1000}.image-modal__content{position:relative;max-width:90vw;max-height:90vh}.image-modal__content img{max-width:90vw;max-height:90vh;-o-object-fit:contain;object-fit:contain;display:block}.image-modal__close{position:fixed;top:15px;right:30px;background:transparent;border:none;color:#fff;font-size:60px;line-height:1;cursor:pointer}@media screen and (max-width: 650px){#product_page_carousel_thumbs{height:57px!important}#product_page_carousel_thumbs .carousel-thumb img{height:50px!important}}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"]
270
283
  },] }
271
284
  ];
272
285
  ImageCarouselComponent.ctorParameters = () => [
286
+ { type: IconCacheService },
273
287
  { type: ProductConnectorService },
274
288
  { type: ProductEventService },
275
289
  { type: ChangeDetectorRef },
@@ -285,4 +299,4 @@ ImageCarouselComponent.propDecorators = {
285
299
  gotoPrevSlide: [{ type: HostListener, args: ['swiperight',] }],
286
300
  resizing: [{ type: HostBinding, args: ['class.resizing',] }]
287
301
  };
288
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-carousel.component.js","sourceRoot":"","sources":["../../../../../src/app/components/image-carousel/image-carousel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  ElementRef,\r\n  HostBinding,\r\n  HostListener,\r\n  Input,\r\n  OnDestroy,\r\n  ViewChild\r\n} from '@angular/core';\r\nimport {CoDocument} from '@colijnit/mainapi/build/model/co-document.bo';\r\nimport {ProductConnectorService} from '../../service/product-connector.service';\r\nimport {ProductEventService} from '../../service/product-event.service';\r\nimport {Subscription} from 'rxjs';\r\nimport {IconEnum} from '../../enum/icon.enum';\r\nimport {DomSanitizer, SafeUrl} from '@angular/platform-browser';\r\nimport {DocumentContent} from '@colijnit/articleapi/build/model/document-content';\r\n\r\nexport class ImageViewModel {\r\n  public image: CoDocument | string;\r\n  public source: SafeUrl;\r\n  public originalSource: string;\r\n}\r\n\r\n@Component({\r\n  selector: 'app-image-carousel',\r\n  template: `\r\n      <div id=\"product_page_carousel\">\r\n          <div id=\"product_page_carousel_items\">\r\n              <co-loader [isShown]=\"true\" *ngIf=\"showLoader\"></co-loader>\r\n              <div #carousel class=\"inner-carousel\">\r\n                  <!-- This has been taken out of the for loop to prevent flashing images when updating. -->\r\n                  <div *ngIf=\"imageViewModels[0]\" class=\"carousel-item\" [id]=\"'slide-0'\" [class.active]=\"isCurrentIndex(0)\" (click)=\"handleShowImage(imageViewModels[0])\">\r\n                      <img [src]=\"imageViewModels[0].source\">\r\n                  </div>\r\n                  <div *ngFor=\"let imageViewModel of imageViewModels.slice(1); let index = index\" class=\"carousel-item\"\r\n                       [id]=\"'slide-' + (index + 1)\" [class.active]=\"isCurrentIndex((index + 1))\" (click)=\"handleShowImage(imageViewModel)\">\r\n                      <img [src]=\"imageViewModel.source\">\r\n                  </div>\r\n                  <div class=\"carousel-scroller-layer\" *ngIf=\"imageViewModels && imageViewModels.length > 1\">\r\n                      <div class=\"carousel-item-scroller prev\" (click)=\"gotoPrevSlide()\" *ngIf=\"currentIndex > 0\"></div>\r\n                      <div class=\"carousel-item-scroller next\" (click)=\"gotoNextSlide()\" *ngIf=\"currentIndex < images.length - 1\"></div>\r\n                  </div>\r\n              </div>\r\n              <!--\r\n                      <co-icon class=\"selector-type-icon refresh-button\" [iconData]=\"iconCache.getIcon(icons.Refresh)\" (click)=\"onForceRenderImage()\" [class.loading]=\"showLoader\" *ngIf=\"showRefresh\"></co-icon>\r\n              -->\r\n          </div>\r\n\r\n          <div id=\"product_page_carousel_thumbs\">\r\n              <co-scroll-container class=\"scroll-container\" *ngIf=\"imageViewModels && imageViewModels.length > 1\">\r\n                  <div *ngFor=\"let imageViewModel of imageViewModels; let index = index\" class=\"carousel-thumb\"\r\n                       [class.active]=\"index === currentIndex\">\r\n                      <img [src]=\"imageViewModel.source\" (click)=\"handleThumbClick(index)\"/>\r\n                  </div>\r\n              </co-scroll-container>\r\n          </div>\r\n      </div>\r\n      <!-- Modal (real popup) -->\r\n      <div\r\n        class=\"image-modal\"\r\n        *ngIf=\"isPopupOpen\"\r\n        (click)=\"closePopup()\"\r\n        role=\"dialog\"\r\n        aria-modal=\"true\"\r\n        aria-label=\"Image preview\"\r\n      >\r\n        <div class=\"image-modal__content\" (click)=\"$event.stopPropagation()\">\r\n          <button\r\n            class=\"image-modal__close\"\r\n            type=\"button\"\r\n            aria-label=\"Close\"\r\n            (click)=\"closePopup()\"\r\n          >\r\n            ×\r\n          </button>\r\n\r\n          <img [src]=\"selectedImage?.originalSource\" alt=\"Image preview\" />\r\n        </div>\r\n      </div>\r\n  `,\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  styleUrls: ['./image-carousel.component.scss']\r\n})\r\nexport class ImageCarouselComponent implements OnDestroy {\r\n  @ViewChild('carousel', {read: ElementRef})\r\n  public carousel: ElementRef;\r\n\r\n  public isPopupOpen = false;\r\n  public selectedImage?: ImageViewModel;\r\n\r\n  // Close on ESC\r\n  @HostListener('document:keydown.escape')\r\n  onEsc(): void {\r\n    if (this.isPopupOpen) {\r\n      this.closePopup();\r\n    }\r\n  }\r\n\r\n  @Input()\r\n  public showRefresh: boolean = false;\r\n\r\n  @Input()\r\n  public set images(value: (CoDocument | string)[]) {\r\n    if (value && value.length > 0) {\r\n      this._images = this._filterValidImages(value);\r\n      this._loadAndRescaleImages();\r\n      this._changeDetector.detectChanges();\r\n    }\r\n  }\r\n\r\n  public get images(): (CoDocument | string)[] {\r\n    return this._images;\r\n  }\r\n\r\n  @HostListener('window:resize')\r\n  public handleWindowResize(): void {\r\n    this.resizing = true;\r\n    this._scrollCarouselToIndex();\r\n    clearTimeout(this._resizeTimer);\r\n    this._resizeTimer = setTimeout(() => {\r\n      this.resizing = false;\r\n    }, 200);\r\n  }\r\n\r\n  @HostListener('swipeleft')\r\n  public gotoNextSlide(): void {\r\n    this.currentIndex++;\r\n  }\r\n\r\n  @HostListener('swiperight')\r\n  public gotoPrevSlide(): void {\r\n    this.currentIndex--;\r\n  }\r\n\r\n  public isCurrentIndex(index: number): boolean {\r\n    return this.currentIndex === index;\r\n  }\r\n\r\n  public get currentIndex(): number {\r\n    return this._currentIndex;\r\n  }\r\n\r\n  public set currentIndex(value: number) {\r\n    this._currentIndex = value;\r\n    this._scrollCarouselToIndex();\r\n  }\r\n\r\n  @HostBinding('class.resizing')\r\n  public resizing = false;\r\n\r\n  public imageViewModels: ImageViewModel[] = [];\r\n  public showLoader: boolean = false;\r\n\r\n  private _resizeTimer: any;\r\n  private _currentIndex = 0;\r\n  private _images: (CoDocument | string)[] = [];\r\n  private _subs: Subscription[] = [];\r\n\r\n  private _resizeCanvasHeight: number = 500;\r\n\r\n  constructor(\r\n    private _ione: ProductConnectorService,\r\n    private _appEventService: ProductEventService,\r\n    private _changeDetector: ChangeDetectorRef,\r\n    private _domSanitizer: DomSanitizer\r\n  ) {\r\n    this._subs.push(\r\n      this._appEventService.onRenderStarted.subscribe(() => {\r\n        this.showLoader = true;\r\n        this._changeDetector.detectChanges();\r\n        setTimeout(() => {\r\n          this.showLoader = false;\r\n          this._changeDetector.detectChanges();\r\n        }, 10000);\r\n      }),\r\n      this._appEventService.onDraftRenderImageReceived.subscribe((event: CustomEvent) => {\r\n        if (event && event.detail) {\r\n          if (this._images[0] instanceof CoDocument) {\r\n            this._images.unshift(event.detail);\r\n          } else {\r\n            this._images[0] = event.detail;\r\n          }\r\n          this._loadAndRescaleImages();\r\n        }\r\n        this.showLoader = false;\r\n        this._changeDetector.detectChanges();\r\n      })\r\n    );\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.carousel = undefined;\r\n    this._subs.forEach(s => s.unsubscribe());\r\n  }\r\n\r\n  public handleThumbClick(index: number): void {\r\n    this.currentIndex = index;\r\n  }\r\n\r\n  public onForceRenderImage(): void {\r\n    this._appEventService.onForceRenderImage.next();\r\n  }\r\n\r\n  public handleShowImage(imageViewModel: ImageViewModel): void {\r\n    if (imageViewModel && imageViewModel.originalSource) {\r\n      this.selectedImage = imageViewModel;\r\n      this.isPopupOpen = true;\r\n      this._changeDetector.markForCheck();\r\n    }\r\n  }\r\n\r\n  public closePopup(): void {\r\n    this.isPopupOpen = false;\r\n    this.selectedImage = undefined;\r\n    this._changeDetector.markForCheck();\r\n  }\r\n\r\n  private _filterValidImages(value: (CoDocument | string)[]): (CoDocument | string)[] {\r\n    if (!value) {\r\n      return [];\r\n    }\r\n    return value.filter((doc: CoDocument | string) => {\r\n      const pattern = /\\.(jpg|jpeg|png|gif|bmp|tiff|webp)$/i;\r\n      return typeof doc === 'string' || (pattern.test(doc.fileName) || !!doc.filePath);\r\n    });\r\n  }\r\n\r\n  private _loadAndRescaleImages(): void {\r\n    this.imageViewModels.length = 0;\r\n    if (this._images) {\r\n      this._images.forEach((i: CoDocument | string) => {\r\n        let imageViewModel = new ImageViewModel();\r\n        imageViewModel.image = i;\r\n        if (typeof i === 'string') { // is a rendered image coming from configurator\r\n          this._resizeAndSanitizeSource(i, imageViewModel);\r\n        } else {\r\n          if (i.filePath) {\r\n            this._resizeAndSanitizeSource(i.filePath, imageViewModel);\r\n          } else if (i.documentBody) {\r\n            this._resizeAndSanitizeSource(i.documentBodyAsDataUri, imageViewModel);\r\n          } else {\r\n            this._ione.getDocumentContent(i.documentId, false).then((content: DocumentContent) => {\r\n              if (content) {\r\n                i.documentBody = content.documentContent;\r\n                this._resizeAndSanitizeSource(i.documentBodyAsDataUri, imageViewModel);\r\n                this._changeDetector.detectChanges();\r\n              }\r\n            });\r\n          }\r\n        }\r\n        this.imageViewModels.push(imageViewModel);\r\n        this._changeDetector.detectChanges();\r\n      });\r\n    }\r\n  }\r\n\r\n  private _resizeAndSanitizeSource(source: string, imageViewModel: ImageViewModel): void {\r\n    const resizeCanvas: HTMLCanvasElement = document.createElement('canvas');\r\n    const ctx = resizeCanvas.getContext('2d')!;\r\n    const img: HTMLImageElement = document.createElement('img');\r\n    img.crossOrigin = 'anonymous';\r\n\r\n    img.onload = () => {\r\n      ctx.imageSmoothingEnabled = true;\r\n      ctx.imageSmoothingQuality = 'high';\r\n\r\n      const ow = img.width;\r\n      const oh = img.height;\r\n      const aspect = ow / oh;\r\n\r\n      let newW = this._resizeCanvasHeight;\r\n      let newH = this._resizeCanvasHeight;\r\n      if (ow > oh) {\r\n        newH = this._resizeCanvasHeight / aspect;\r\n      } else {\r\n        newW = this._resizeCanvasHeight * aspect;\r\n      }\r\n\r\n      resizeCanvas.width = Math.round(newW);\r\n      resizeCanvas.height = Math.round(newH);\r\n\r\n      // Ensure transparent background before drawing\r\n      ctx.clearRect(0, 0, resizeCanvas.width, resizeCanvas.height);\r\n\r\n      ctx.drawImage(img, 0, 0, resizeCanvas.width, resizeCanvas.height);\r\n\r\n      const mime = this._detectPreferredMime(source);\r\n      const resizedSource =\r\n        mime === 'image/jpeg'\r\n          ? resizeCanvas.toDataURL('image/jpeg', 0.92) // only if original was JPEG\r\n          : resizeCanvas.toDataURL(mime);              // PNG/WebP keep alpha\r\n\r\n      imageViewModel.source = this._domSanitizer.bypassSecurityTrustUrl(resizedSource);\r\n      imageViewModel.originalSource = source;\r\n      this._changeDetector.detectChanges();\r\n    };\r\n\r\n    img.src = source;\r\n  }\r\n\r\n  private _detectPreferredMime(source: string): 'image/png' | 'image/webp' | 'image/jpeg' {\r\n    // Data URI check\r\n    const m = source.match(/^data:(image\\/[a-zA-Z+.-]+);base64,/);\r\n    if (m) {\r\n      const t = m[1].toLowerCase();\r\n      if (t === 'image/png' || t === 'image/webp' || t === 'image/jpeg') return t as any;\r\n    }\r\n\r\n    const lower = source.toLowerCase();\r\n    if (lower.endsWith('.png')) return 'image/png';\r\n    if (lower.endsWith('.webp')) return 'image/webp';\r\n    if (lower.endsWith('.jpg') || lower.endsWith('.jpeg')) return 'image/jpeg';\r\n\r\n    return 'image/png';\r\n  }\r\n\r\n  private _scrollCarouselToIndex(): void {\r\n    if (this.currentIndex > -1 && this.currentIndex <= this.images.length) {\r\n      const movePx: number = this.currentIndex * this.carousel.nativeElement.clientWidth;\r\n      if (this.carousel && this.carousel.nativeElement) {\r\n        this.carousel.nativeElement.scrollLeft = movePx;\r\n        // this.carousel.nativeElement.scrollTo({\r\n        //   left: movePx,\r\n        //   behavior: 'smooth'\r\n        // });\r\n      }\r\n    }\r\n  }\r\n\r\n}\r\n"]}
302
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-carousel.component.js","sourceRoot":"","sources":["../../../../../src/app/components/image-carousel/image-carousel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  ElementRef,\r\n  HostBinding,\r\n  HostListener,\r\n  Input,\r\n  OnDestroy,\r\n  ViewChild\r\n} from '@angular/core';\r\nimport {CoDocument} from '@colijnit/mainapi/build/model/co-document.bo';\r\nimport {ProductConnectorService} from '../../service/product-connector.service';\r\nimport {ProductEventService} from '../../service/product-event.service';\r\nimport {Subscription} from 'rxjs';\r\nimport {IconEnum} from '../../enum/icon.enum';\r\nimport {DomSanitizer, SafeUrl} from '@angular/platform-browser';\r\nimport {DocumentContent} from '@colijnit/articleapi/build/model/document-content';\r\nimport {IconModule} from '@colijnit/corecomponents_v12';\r\nimport {ScrollContainerModule} from '@colijnit/corecomponents';\r\nimport {OverlayModule} from '@angular/cdk/overlay';\r\nimport {IconCacheService} from '../../service/icon-cache.service';\r\n\r\nexport class ImageViewModel {\r\n  public image: CoDocument | string;\r\n  public source: SafeUrl;\r\n  public originalSource: string;\r\n}\r\n\r\n@Component({\r\n  selector: 'app-image-carousel',\r\n  template: `\r\n    <div id=\"product_page_carousel\">\r\n      <div id=\"product_page_carousel_items\">\r\n        <co-loader [isShown]=\"true\" *ngIf=\"showLoader\"></co-loader>\r\n        <div #carousel class=\"inner-carousel\">\r\n          <!-- This has been taken out of the for loop to prevent flashing images when updating. -->\r\n          <div *ngIf=\"imageViewModels[0]\" class=\"carousel-item\" [id]=\"'slide-0'\" [class.active]=\"isCurrentIndex(0)\"\r\n               (click)=\"handleShowImage(imageViewModels[0])\">\r\n            <img [src]=\"imageViewModels[0].source\">\r\n          </div>\r\n          <div *ngFor=\"let imageViewModel of imageViewModels.slice(1); let index = index\" class=\"carousel-item\"\r\n               [id]=\"'slide-' + (index + 1)\" [class.active]=\"isCurrentIndex((index + 1))\" (click)=\"handleShowImage(imageViewModel)\">\r\n            <img [src]=\"imageViewModel.source\">\r\n          </div>\r\n          <div class=\"carousel-scroller-layer\" *ngIf=\"imageViewModels && imageViewModels.length > 1\">\r\n            <div class=\"carousel-item-scroller prev\" (click)=\"gotoPrevSlide()\" *ngIf=\"currentIndex > 0\"></div>\r\n            <div class=\"carousel-item-scroller next\" (click)=\"gotoNextSlide()\" *ngIf=\"currentIndex < images.length - 1\"></div>\r\n          </div>\r\n        </div>\r\n        <!--\r\n                <co-icon class=\"selector-type-icon refresh-button\" [iconData]=\"iconCache.getIcon(icons.Refresh)\" (click)=\"onForceRenderImage()\" [class.loading]=\"showLoader\" *ngIf=\"showRefresh\"></co-icon>\r\n        -->\r\n      </div>\r\n\r\n      <div id=\"product_page_carousel_thumbs\">\r\n        <co-scroll-container class=\"scroll-container\" *ngIf=\"imageViewModels && imageViewModels.length > 1\">\r\n          <div *ngFor=\"let imageViewModel of imageViewModels; let index = index\" class=\"carousel-thumb\"\r\n               [class.active]=\"index === currentIndex\">\r\n            <img [src]=\"imageViewModel.source\" (click)=\"handleThumbClick(index)\"/>\r\n          </div>\r\n        </co-scroll-container>\r\n      </div>\r\n    </div>\r\n\r\n    <div cdkOverlayOrigin #trigger=\"cdkOverlayOrigin\"></div>\r\n    <ng-template\r\n      cdkConnectedOverlay\r\n      [cdkConnectedOverlayOrigin]=\"trigger\"\r\n      [cdkConnectedOverlayOpen]=\"isPopupOpen\"\r\n      [cdkConnectedOverlayHasBackdrop]=\"true\"\r\n      (backdropClick)=\"closePopup()\">\r\n      <div class=\"image-modal\" role=\"dialog\" aria-modal=\"true\" aria-label=\"Image preview\">\r\n        <div class=\"image-modal__content\">\r\n          <button class=\"image-modal__close\" type=\"button\" aria-label=\"Close\" (click)=\"closePopup()\">\r\n            <co-icon [iconData]=\"iconCacheService.getIcon(icons.CrossSkinny)\"></co-icon>\r\n          </button>\r\n          <img [src]=\"selectedImage?.originalSource\" alt=\"Image preview\"/>\r\n        </div>\r\n      </div>\r\n    </ng-template>\r\n\r\n  `,\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  styleUrls: ['./image-carousel.component.scss']\r\n})\r\nexport class ImageCarouselComponent implements OnDestroy {\r\n\r\n  public readonly icons: typeof IconEnum = IconEnum;\r\n\r\n  @ViewChild('carousel', {read: ElementRef})\r\n  public carousel: ElementRef;\r\n\r\n  public isPopupOpen = false;\r\n  public selectedImage?: ImageViewModel;\r\n\r\n  // Close on ESC\r\n  @HostListener('document:keydown.escape')\r\n  onEsc(): void {\r\n    if (this.isPopupOpen) {\r\n      this.closePopup();\r\n    }\r\n  }\r\n\r\n  @Input()\r\n  public showRefresh: boolean = false;\r\n\r\n  @Input()\r\n  public set images(value: (CoDocument | string)[]) {\r\n    if (value && value.length > 0) {\r\n      this._images = this._filterValidImages(value);\r\n      this._loadAndRescaleImages();\r\n      this._changeDetector.detectChanges();\r\n    }\r\n  }\r\n\r\n  public get images(): (CoDocument | string)[] {\r\n    return this._images;\r\n  }\r\n\r\n  @HostListener('window:resize')\r\n  public handleWindowResize(): void {\r\n    this.resizing = true;\r\n    this._scrollCarouselToIndex();\r\n    clearTimeout(this._resizeTimer);\r\n    this._resizeTimer = setTimeout(() => {\r\n      this.resizing = false;\r\n    }, 200);\r\n  }\r\n\r\n  @HostListener('swipeleft')\r\n  public gotoNextSlide(): void {\r\n    this.currentIndex++;\r\n  }\r\n\r\n  @HostListener('swiperight')\r\n  public gotoPrevSlide(): void {\r\n    this.currentIndex--;\r\n  }\r\n\r\n  public isCurrentIndex(index: number): boolean {\r\n    return this.currentIndex === index;\r\n  }\r\n\r\n  public get currentIndex(): number {\r\n    return this._currentIndex;\r\n  }\r\n\r\n  public set currentIndex(value: number) {\r\n    this._currentIndex = value;\r\n    this._scrollCarouselToIndex();\r\n  }\r\n\r\n  @HostBinding('class.resizing')\r\n  public resizing = false;\r\n\r\n  public imageViewModels: ImageViewModel[] = [];\r\n  public showLoader: boolean = false;\r\n\r\n  private _resizeTimer: any;\r\n  private _currentIndex = 0;\r\n  private _images: (CoDocument | string)[] = [];\r\n  private _subs: Subscription[] = [];\r\n\r\n  private _resizeCanvasHeight: number = 500;\r\n\r\n  constructor(\r\n    public iconCacheService: IconCacheService,\r\n    private _ione: ProductConnectorService,\r\n    private _appEventService: ProductEventService,\r\n    private _changeDetector: ChangeDetectorRef,\r\n    private _domSanitizer: DomSanitizer\r\n  ) {\r\n    this._subs.push(\r\n      this._appEventService.onRenderStarted.subscribe(() => {\r\n        this.showLoader = true;\r\n        this._changeDetector.detectChanges();\r\n        setTimeout(() => {\r\n          this.showLoader = false;\r\n          this._changeDetector.detectChanges();\r\n        }, 10000);\r\n      }),\r\n      this._appEventService.onDraftRenderImageReceived.subscribe((event: CustomEvent) => {\r\n        if (event && event.detail) {\r\n          if (this._images[0] instanceof CoDocument) {\r\n            this._images.unshift(event.detail);\r\n          } else {\r\n            this._images[0] = event.detail;\r\n          }\r\n          this._loadAndRescaleImages();\r\n        }\r\n        this.showLoader = false;\r\n        this._changeDetector.detectChanges();\r\n      })\r\n    );\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.carousel = undefined;\r\n    this._subs.forEach(s => s.unsubscribe());\r\n  }\r\n\r\n  public handleThumbClick(index: number): void {\r\n    this.currentIndex = index;\r\n  }\r\n\r\n  public onForceRenderImage(): void {\r\n    this._appEventService.onForceRenderImage.next();\r\n  }\r\n\r\n  public handleShowImage(imageViewModel: ImageViewModel): void {\r\n    if (imageViewModel && imageViewModel.originalSource) {\r\n      this.selectedImage = imageViewModel;\r\n      this.isPopupOpen = true;\r\n      this._changeDetector.markForCheck();\r\n    }\r\n  }\r\n\r\n  public closePopup(): void {\r\n    this.isPopupOpen = false;\r\n    this.selectedImage = undefined;\r\n    this._changeDetector.markForCheck();\r\n  }\r\n\r\n  private _filterValidImages(value: (CoDocument | string)[]): (CoDocument | string)[] {\r\n    if (!value) {\r\n      return [];\r\n    }\r\n    return value.filter((doc: CoDocument | string) => {\r\n      const pattern = /\\.(jpg|jpeg|png|gif|bmp|tiff|webp)$/i;\r\n      return typeof doc === 'string' || (pattern.test(doc.fileName) || !!doc.filePath);\r\n    });\r\n  }\r\n\r\n  private _loadAndRescaleImages(): void {\r\n    this.imageViewModels.length = 0;\r\n    if (this._images) {\r\n      this._images.forEach((i: CoDocument | string) => {\r\n        let imageViewModel = new ImageViewModel();\r\n        imageViewModel.image = i;\r\n        if (typeof i === 'string') { // is a rendered image coming from configurator\r\n          this._resizeAndSanitizeSource(i, imageViewModel);\r\n        } else {\r\n          if (i.filePath) {\r\n            this._resizeAndSanitizeSource(i.filePath, imageViewModel);\r\n          } else if (i.documentBody) {\r\n            this._resizeAndSanitizeSource(i.documentBodyAsDataUri, imageViewModel);\r\n          } else {\r\n            this._ione.getDocumentContent(i.documentId, false).then((content: DocumentContent) => {\r\n              if (content) {\r\n                i.documentBody = content.documentContent;\r\n                this._resizeAndSanitizeSource(i.documentBodyAsDataUri, imageViewModel);\r\n                this._changeDetector.detectChanges();\r\n              }\r\n            });\r\n          }\r\n        }\r\n        this.imageViewModels.push(imageViewModel);\r\n        this._changeDetector.detectChanges();\r\n      });\r\n    }\r\n  }\r\n\r\n  private _resizeAndSanitizeSource(source: string, imageViewModel: ImageViewModel): void {\r\n    const resizeCanvas: HTMLCanvasElement = document.createElement('canvas');\r\n    const ctx = resizeCanvas.getContext('2d')!;\r\n    const img: HTMLImageElement = document.createElement('img');\r\n    img.crossOrigin = 'anonymous';\r\n\r\n    const handleFallback = () => {\r\n      // When a CDN does not accept our tainted canvas, we fallback to the original source.\r\n      imageViewModel.source = this._domSanitizer.bypassSecurityTrustUrl(source);\r\n      imageViewModel.originalSource = source;\r\n      this._changeDetector.detectChanges();\r\n    };\r\n\r\n    img.onerror = () => {\r\n      handleFallback();\r\n    };\r\n\r\n    img.onload = () => {\r\n      try {\r\n        ctx.imageSmoothingEnabled = true;\r\n        ctx.imageSmoothingQuality = 'high';\r\n\r\n        const aspect: number = img.width / img.height;\r\n        let newW: number = this._resizeCanvasHeight;\r\n        let newH: number = this._resizeCanvasHeight;\r\n\r\n        if (img.width > img.height) {\r\n          newH = this._resizeCanvasHeight / aspect;\r\n        } else {\r\n          newW = this._resizeCanvasHeight * aspect;\r\n        }\r\n        resizeCanvas.width = Math.round(newW);\r\n        resizeCanvas.height = Math.round(newH);\r\n\r\n        ctx.clearRect(0, 0, resizeCanvas.width, resizeCanvas.height);\r\n        ctx.drawImage(img, 0, 0, resizeCanvas.width, resizeCanvas.height);\r\n\r\n        const mime: string = this._handleMimeTypes(source);\r\n        const resizedSource: string = resizeCanvas.toDataURL(mime);\r\n\r\n        imageViewModel.source = this._domSanitizer.bypassSecurityTrustUrl(resizedSource);\r\n        imageViewModel.originalSource = source;\r\n        this._changeDetector.detectChanges();\r\n      } catch (error) {\r\n        handleFallback();\r\n      }\r\n    };\r\n    img.src = source;\r\n  }\r\n\r\n  private _handleMimeTypes(imageSource: string): string {\r\n    // When we got base64 data\r\n    if (imageSource.startsWith('data:')) {\r\n      const match: RegExpMatchArray = imageSource.match(/^data:(image\\/(png|webp|jpeg|jpg))/i);\r\n      return match ? match[1].toLowerCase() : 'image/png';\r\n    }\r\n    // when we got an URL.\r\n    const extension: string = imageSource.substring(imageSource.lastIndexOf('.') + 1).toLowerCase();\r\n    switch (extension) {\r\n      case 'png':\r\n        return 'image/png';\r\n      case 'webp':\r\n        return 'image/webp';\r\n      case 'jpg':\r\n      case 'jpeg':\r\n        return 'image/jpeg';\r\n      default:\r\n        return 'image/png';\r\n    }\r\n  }\r\n\r\n  private _scrollCarouselToIndex(): void {\r\n    if (this.currentIndex > -1 && this.currentIndex <= this.images.length) {\r\n      const movePx: number = this.currentIndex * this.carousel.nativeElement.clientWidth;\r\n      if (this.carousel && this.carousel.nativeElement) {\r\n        this.carousel.nativeElement.scrollLeft = movePx;\r\n        // this.carousel.nativeElement.scrollTo({\r\n        //   left: movePx,\r\n        //   behavior: 'smooth'\r\n        // });\r\n      }\r\n    }\r\n  }\r\n\r\n}\r\n"]}
@@ -2,6 +2,8 @@ import { NgModule } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import { ImageCarouselComponent } from './image-carousel.component';
4
4
  import { /*IconModule, */ LoaderModule, ScrollContainerModule } from '@colijnit/corecomponents';
5
+ import { OverlayModule } from '@angular/cdk/overlay';
6
+ import { IconModule } from '@colijnit/corecomponents_v12';
5
7
  export class ImageCarouselModule {
6
8
  }
7
9
  ImageCarouselModule.decorators = [
@@ -10,7 +12,9 @@ ImageCarouselModule.decorators = [
10
12
  CommonModule,
11
13
  LoaderModule,
12
14
  // IconModule,
13
- ScrollContainerModule
15
+ ScrollContainerModule,
16
+ OverlayModule,
17
+ IconModule
14
18
  ],
15
19
  declarations: [
16
20
  ImageCarouselComponent
@@ -20,4 +24,4 @@ ImageCarouselModule.decorators = [
20
24
  ]
21
25
  },] }
22
26
  ];
23
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW1hZ2UtY2Fyb3VzZWwubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9jb21wb25lbnRzL2ltYWdlLWNhcm91c2VsL2ltYWdlLWNhcm91c2VsLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsUUFBUSxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQ3ZDLE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsc0JBQXNCLEVBQUMsTUFBTSw0QkFBNEIsQ0FBQztBQUNsRSxPQUFPLEVBQUMsZ0JBQWdCLENBQUEsWUFBWSxFQUFFLHFCQUFxQixFQUFDLE1BQU0sMEJBQTBCLENBQUM7QUFnQjdGLE1BQU0sT0FBTyxtQkFBbUI7OztZQWQvQixRQUFRLFNBQUM7Z0JBQ04sT0FBTyxFQUFFO29CQUNMLFlBQVk7b0JBQ1osWUFBWTtvQkFDWixjQUFjO29CQUNkLHFCQUFxQjtpQkFDeEI7Z0JBQ0QsWUFBWSxFQUFFO29CQUNWLHNCQUFzQjtpQkFDekI7Z0JBQ0QsT0FBTyxFQUFFO29CQUNMLHNCQUFzQjtpQkFDekI7YUFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7TmdNb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQge0NvbW1vbk1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHtJbWFnZUNhcm91c2VsQ29tcG9uZW50fSBmcm9tICcuL2ltYWdlLWNhcm91c2VsLmNvbXBvbmVudCc7XHJcbmltcG9ydCB7LypJY29uTW9kdWxlLCAqL0xvYWRlck1vZHVsZSwgU2Nyb2xsQ29udGFpbmVyTW9kdWxlfSBmcm9tICdAY29saWpuaXQvY29yZWNvbXBvbmVudHMnO1xyXG5cclxuQE5nTW9kdWxlKHtcclxuICAgIGltcG9ydHM6IFtcclxuICAgICAgICBDb21tb25Nb2R1bGUsXHJcbiAgICAgICAgTG9hZGVyTW9kdWxlLFxyXG4gICAgICAgIC8vIEljb25Nb2R1bGUsXHJcbiAgICAgICAgU2Nyb2xsQ29udGFpbmVyTW9kdWxlXHJcbiAgICBdLFxyXG4gICAgZGVjbGFyYXRpb25zOiBbXHJcbiAgICAgICAgSW1hZ2VDYXJvdXNlbENvbXBvbmVudFxyXG4gICAgXSxcclxuICAgIGV4cG9ydHM6IFtcclxuICAgICAgICBJbWFnZUNhcm91c2VsQ29tcG9uZW50XHJcbiAgICBdXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBJbWFnZUNhcm91c2VsTW9kdWxlIHtcclxufVxyXG4iXX0=
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW1hZ2UtY2Fyb3VzZWwubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9jb21wb25lbnRzL2ltYWdlLWNhcm91c2VsL2ltYWdlLWNhcm91c2VsLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsUUFBUSxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQ3ZDLE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsc0JBQXNCLEVBQUMsTUFBTSw0QkFBNEIsQ0FBQztBQUNsRSxPQUFPLEVBQUMsZ0JBQWdCLENBQUEsWUFBWSxFQUFFLHFCQUFxQixFQUFDLE1BQU0sMEJBQTBCLENBQUM7QUFDN0YsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLHNCQUFzQixDQUFDO0FBQ25ELE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSw4QkFBOEIsQ0FBQztBQWtCeEQsTUFBTSxPQUFPLG1CQUFtQjs7O1lBaEIvQixRQUFRLFNBQUM7Z0JBQ04sT0FBTyxFQUFFO29CQUNMLFlBQVk7b0JBQ1osWUFBWTtvQkFDWixjQUFjO29CQUNkLHFCQUFxQjtvQkFDckIsYUFBYTtvQkFDYixVQUFVO2lCQUNiO2dCQUNELFlBQVksRUFBRTtvQkFDVixzQkFBc0I7aUJBQ3pCO2dCQUNELE9BQU8sRUFBRTtvQkFDTCxzQkFBc0I7aUJBQ3pCO2FBQ0oiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge05nTW9kdWxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHtDb21tb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XHJcbmltcG9ydCB7SW1hZ2VDYXJvdXNlbENvbXBvbmVudH0gZnJvbSAnLi9pbWFnZS1jYXJvdXNlbC5jb21wb25lbnQnO1xyXG5pbXBvcnQgey8qSWNvbk1vZHVsZSwgKi9Mb2FkZXJNb2R1bGUsIFNjcm9sbENvbnRhaW5lck1vZHVsZX0gZnJvbSAnQGNvbGlqbml0L2NvcmVjb21wb25lbnRzJztcclxuaW1wb3J0IHtPdmVybGF5TW9kdWxlfSBmcm9tICdAYW5ndWxhci9jZGsvb3ZlcmxheSc7XHJcbmltcG9ydCB7SWNvbk1vZHVsZX0gZnJvbSAnQGNvbGlqbml0L2NvcmVjb21wb25lbnRzX3YxMic7XHJcblxyXG5ATmdNb2R1bGUoe1xyXG4gICAgaW1wb3J0czogW1xyXG4gICAgICAgIENvbW1vbk1vZHVsZSxcclxuICAgICAgICBMb2FkZXJNb2R1bGUsXHJcbiAgICAgICAgLy8gSWNvbk1vZHVsZSxcclxuICAgICAgICBTY3JvbGxDb250YWluZXJNb2R1bGUsXHJcbiAgICAgICAgT3ZlcmxheU1vZHVsZSxcclxuICAgICAgICBJY29uTW9kdWxlXHJcbiAgICBdLFxyXG4gICAgZGVjbGFyYXRpb25zOiBbXHJcbiAgICAgICAgSW1hZ2VDYXJvdXNlbENvbXBvbmVudFxyXG4gICAgXSxcclxuICAgIGV4cG9ydHM6IFtcclxuICAgICAgICBJbWFnZUNhcm91c2VsQ29tcG9uZW50XHJcbiAgICBdXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBJbWFnZUNhcm91c2VsTW9kdWxlIHtcclxufVxyXG4iXX0=
@@ -86,11 +86,16 @@ export class ProductRelatedComponent {
86
86
  return __awaiter(this, void 0, void 0, function* () {
87
87
  this.articleViewModels.forEach(avm => {
88
88
  if (avm.article.image) {
89
- this._iOne.getImageForCoDocument(avm.article.image, false).then(imageContent => avm.imageData = imageContent.image);
89
+ this._iOne.getImageForCoDocument(avm.article.image, false).then(imageContent => this._handleImageData(avm, imageContent));
90
90
  }
91
91
  });
92
92
  });
93
93
  }
94
+ _handleImageData(avm, imageContent) {
95
+ if (imageContent && imageContent.image) {
96
+ avm.imageData = imageContent.image;
97
+ }
98
+ }
94
99
  }
95
100
  ProductRelatedComponent.decorators = [
96
101
  { type: Component, args: [{
@@ -134,4 +139,4 @@ ProductRelatedComponent.propDecorators = {
134
139
  createFrozenArticle: [{ type: Input }],
135
140
  articles: [{ type: Input }]
136
141
  };
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-related.component.js","sourceRoot":"","sources":["../../../../../src/app/components/product-related/product-related.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {Component, Input} from '@angular/core';\r\nimport {ArticleListObject} from '@colijnit/articleapi/build/model/article-list-object';\r\nimport {ProductConnectorService} from '../../service/product-connector.service';\r\nimport {ProductEventService} from '../../service/product-event.service';\r\nimport {ProductConnectorAdapterService} from '../../service/product-connector-adapter.service';\r\nimport {ArticleQuantity} from '../../model/article-quantity.model';\r\nimport {ConfiguratorStatisticsEnvironment} from '@colijnit/articleapi/build/model/configurator-statistics-environment';\r\nimport {ProductSettingsService} from '../../service/product-settings.service';\r\nimport {ArticleFullObject} from '@colijnit/articleapi/build/model/article-full-object';\r\nimport {ArticleListObjectExtended} from '@colijnit/articleapi/build/model/article-list-object-extended.bo';\r\nimport {ExternalSource} from '@colijnit/articleapi/build/model/external-source.bo';\r\n\r\nexport class ArticleViewModel {\r\n    public article: ArticleListObject;\r\n    public imageData: string;\r\n    constructor(article: ArticleListObject, imageData?: string) {\r\n        this.article = article;\r\n        if (imageData) {\r\n            this.imageData = imageData;\r\n        }\r\n    }\r\n}\r\n\r\n@Component({\r\n    selector: 'app-product-related',\r\n    template: `\r\n        <div *ngIf=\"articles && articles.length > 0\">\r\n            <app-header [label]=\"label\" [amount]=\"articles?.length\" *ngIf=\"label\"></app-header>\r\n            <co-scroll-container>\r\n                <div class=\"article-wrapper\" *ngFor=\"let articleViewModel of articleViewModels\">\r\n                    <co-article-tile \r\n                        [imageData]=\"articleViewModel.imageData\"\r\n                        [description]=\"articleViewModel.article.description\"\r\n                        [price]=\"articleViewModel.article.price\"\r\n                        [level]=\"articleViewModel.article.stockStatus\"\r\n                        [hasCartButton]=\"true\"\r\n                        [isSmallModus]=\"isSmallModus\"\r\n                        [hasConfigureButton]=\"articleViewModel.article.goodType === 'B'\"\r\n                        (contentClick)=\"handleContentClick(articleViewModel.article)\"\r\n                        (cartButtonClick)=\"handleAddToCartClick(articleViewModel.article)\"\r\n                        (configureButtonClick)=\"handleContentClick(articleViewModel.article)\"\r\n                        (threeDButtonClick)=\"handleThreeDButtonClick(articleViewModel.article)\"\r\n                    ></co-article-tile>\r\n                </div>\r\n            </co-scroll-container>\r\n        </div>\r\n    `,\r\n    styleUrls: ['./product-related.component.scss']\r\n})\r\nexport class ProductRelatedComponent {\r\n\r\n    @Input()\r\n    public refType: number;\r\n\r\n    @Input()\r\n    public label: string;\r\n\r\n    @Input()\r\n    public externalSource: ExternalSource;\r\n\r\n    @Input()\r\n    public isSmallModus: boolean = true;\r\n\r\n    @Input()\r\n    public createFrozenArticle: boolean = true;\r\n\r\n    @Input()\r\n    public set articles(value: ArticleListObject[]) {\r\n        if (value) {\r\n            if (this.refType && this.refType !== undefined) {\r\n                if (value.length > 0) {\r\n                    value.forEach((x: ArticleListObject) => {\r\n                        if (x.refType === this.refType) {\r\n                            this._articles.push(x);\r\n                        }\r\n                    });\r\n                }\r\n            } else {\r\n                this._articles = value;\r\n            }\r\n            this.articleViewModels.length = 0;\r\n            this._articles.forEach(a => this.articleViewModels.push(new ArticleViewModel(a)));\r\n            this._loadImages();\r\n        }\r\n    }\r\n\r\n    public get articles(): ArticleListObject[] {\r\n        return this._articles;\r\n    }\r\n\r\n    public articleViewModels: ArticleViewModel[] = [];\r\n    private _articles: ArticleListObject[] = [];\r\n\r\n    constructor(\r\n        private _iOne: ProductConnectorService,\r\n        private _appEventService: ProductEventService,\r\n        private _productConnectorAdapterService: ProductConnectorAdapterService,\r\n        private _settingsService: ProductSettingsService\r\n    ) {\r\n    }\r\n\r\n    public handleContentClick(article: ArticleListObject): void {\r\n        this._appEventService.onAlternativeClick.next(article);\r\n    }\r\n\r\n    public async handleAddToCartClick(article: ArticleListObject): Promise<void> {\r\n        const quantity: number = 1;\r\n        const articleFull: ArticleFullObject = await this._productConnectorAdapterService.getArticleFullObject(article.goodId, true);\r\n        if (articleFull) {\r\n            if (this.createFrozenArticle) {\r\n                const article: ArticleListObjectExtended = this._iOne.convertArticleFullObjectToArticleExtended(articleFull);\r\n                this._appEventService.onAddToCart.next({article: article, quantity: quantity});\r\n                // this._appEventService.onAddToCart.next({ article: await this._getJSONFromArticleObject({article: articleFull, quantity: quantity}), quantity: quantity });\r\n            } else {\r\n                const article: ArticleListObjectExtended = this._iOne.convertArticleFullObjectToArticleExtended(articleFull);\r\n                this._appEventService.onAddToCart.next({article: article, quantity: quantity});\r\n            }\r\n        }\r\n    }\r\n\r\n    public handleConfigureButtonClick(article: ArticleListObject): void {\r\n\r\n    }\r\n\r\n    public handleThreeDButtonClick(article: ArticleListObject): void {\r\n\r\n    }\r\n\r\n    private async _getJSONFromArticleObject(article: ArticleQuantity): Promise<string> {\r\n        const configuratorStatistics = new ConfiguratorStatisticsEnvironment();\r\n        configuratorStatistics.userType = 'iOneProductPage';\r\n        configuratorStatistics.transactionType = 'Sales';\r\n        configuratorStatistics.actionDomain = 'bundle';\r\n        configuratorStatistics.sessionId = this._settingsService.settings.session ? this._settingsService.settings.session.sessionId : 'unknown';\r\n        configuratorStatistics.webHost = window.location.host;\r\n        configuratorStatistics.bundleHost = window.location.host;\r\n        return await this._iOne.getJsonArticleFlatTree(\r\n            article.article.goodId, article.article.goodType, article.quantity, !!this.externalSource, true, configuratorStatistics);\r\n    }\r\n\r\n    private async _loadImages(): Promise<void> {\r\n        this.articleViewModels.forEach(avm => {\r\n            if (avm.article.image) {\r\n                this._iOne.getImageForCoDocument(avm.article.image, false).then(imageContent => avm.imageData = imageContent.image);\r\n            }\r\n        })\r\n    }\r\n}\r\n"]}
142
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-related.component.js","sourceRoot":"","sources":["../../../../../src/app/components/product-related/product-related.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {Component, Input} from '@angular/core';\r\nimport {ArticleListObject} from '@colijnit/articleapi/build/model/article-list-object';\r\nimport {ProductConnectorService} from '../../service/product-connector.service';\r\nimport {ProductEventService} from '../../service/product-event.service';\r\nimport {ProductConnectorAdapterService} from '../../service/product-connector-adapter.service';\r\nimport {ArticleQuantity} from '../../model/article-quantity.model';\r\nimport {ConfiguratorStatisticsEnvironment} from '@colijnit/articleapi/build/model/configurator-statistics-environment';\r\nimport {ProductSettingsService} from '../../service/product-settings.service';\r\nimport {ArticleFullObject} from '@colijnit/articleapi/build/model/article-full-object';\r\nimport {ArticleListObjectExtended} from '@colijnit/articleapi/build/model/article-list-object-extended.bo';\r\nimport {ExternalSource} from '@colijnit/articleapi/build/model/external-source.bo';\r\nimport {ImageContent} from '@colijnit/mainapi/build/model/image-content.bo';\r\n\r\nexport class ArticleViewModel {\r\n    public article: ArticleListObject;\r\n    public imageData: string;\r\n    constructor(article: ArticleListObject, imageData?: string) {\r\n        this.article = article;\r\n        if (imageData) {\r\n            this.imageData = imageData;\r\n        }\r\n    }\r\n}\r\n\r\n@Component({\r\n    selector: 'app-product-related',\r\n    template: `\r\n        <div *ngIf=\"articles && articles.length > 0\">\r\n            <app-header [label]=\"label\" [amount]=\"articles?.length\" *ngIf=\"label\"></app-header>\r\n            <co-scroll-container>\r\n                <div class=\"article-wrapper\" *ngFor=\"let articleViewModel of articleViewModels\">\r\n                    <co-article-tile \r\n                        [imageData]=\"articleViewModel.imageData\"\r\n                        [description]=\"articleViewModel.article.description\"\r\n                        [price]=\"articleViewModel.article.price\"\r\n                        [level]=\"articleViewModel.article.stockStatus\"\r\n                        [hasCartButton]=\"true\"\r\n                        [isSmallModus]=\"isSmallModus\"\r\n                        [hasConfigureButton]=\"articleViewModel.article.goodType === 'B'\"\r\n                        (contentClick)=\"handleContentClick(articleViewModel.article)\"\r\n                        (cartButtonClick)=\"handleAddToCartClick(articleViewModel.article)\"\r\n                        (configureButtonClick)=\"handleContentClick(articleViewModel.article)\"\r\n                        (threeDButtonClick)=\"handleThreeDButtonClick(articleViewModel.article)\"\r\n                    ></co-article-tile>\r\n                </div>\r\n            </co-scroll-container>\r\n        </div>\r\n    `,\r\n    styleUrls: ['./product-related.component.scss']\r\n})\r\nexport class ProductRelatedComponent {\r\n\r\n    @Input()\r\n    public refType: number;\r\n\r\n    @Input()\r\n    public label: string;\r\n\r\n    @Input()\r\n    public externalSource: ExternalSource;\r\n\r\n    @Input()\r\n    public isSmallModus: boolean = true;\r\n\r\n    @Input()\r\n    public createFrozenArticle: boolean = true;\r\n\r\n    @Input()\r\n    public set articles(value: ArticleListObject[]) {\r\n        if (value) {\r\n            if (this.refType && this.refType !== undefined) {\r\n                if (value.length > 0) {\r\n                    value.forEach((x: ArticleListObject) => {\r\n                        if (x.refType === this.refType) {\r\n                            this._articles.push(x);\r\n                        }\r\n                    });\r\n                }\r\n            } else {\r\n                this._articles = value;\r\n            }\r\n            this.articleViewModels.length = 0;\r\n            this._articles.forEach(a => this.articleViewModels.push(new ArticleViewModel(a)));\r\n            this._loadImages();\r\n        }\r\n    }\r\n\r\n    public get articles(): ArticleListObject[] {\r\n        return this._articles;\r\n    }\r\n\r\n    public articleViewModels: ArticleViewModel[] = [];\r\n    private _articles: ArticleListObject[] = [];\r\n\r\n    constructor(\r\n        private _iOne: ProductConnectorService,\r\n        private _appEventService: ProductEventService,\r\n        private _productConnectorAdapterService: ProductConnectorAdapterService,\r\n        private _settingsService: ProductSettingsService\r\n    ) {\r\n    }\r\n\r\n    public handleContentClick(article: ArticleListObject): void {\r\n        this._appEventService.onAlternativeClick.next(article);\r\n    }\r\n\r\n    public async handleAddToCartClick(article: ArticleListObject): Promise<void> {\r\n        const quantity: number = 1;\r\n        const articleFull: ArticleFullObject = await this._productConnectorAdapterService.getArticleFullObject(article.goodId, true);\r\n        if (articleFull) {\r\n            if (this.createFrozenArticle) {\r\n                const article: ArticleListObjectExtended = this._iOne.convertArticleFullObjectToArticleExtended(articleFull);\r\n                this._appEventService.onAddToCart.next({article: article, quantity: quantity});\r\n                // this._appEventService.onAddToCart.next({ article: await this._getJSONFromArticleObject({article: articleFull, quantity: quantity}), quantity: quantity });\r\n            } else {\r\n                const article: ArticleListObjectExtended = this._iOne.convertArticleFullObjectToArticleExtended(articleFull);\r\n                this._appEventService.onAddToCart.next({article: article, quantity: quantity});\r\n            }\r\n        }\r\n    }\r\n\r\n    public handleConfigureButtonClick(article: ArticleListObject): void {\r\n\r\n    }\r\n\r\n    public handleThreeDButtonClick(article: ArticleListObject): void {\r\n\r\n    }\r\n\r\n    private async _getJSONFromArticleObject(article: ArticleQuantity): Promise<string> {\r\n        const configuratorStatistics = new ConfiguratorStatisticsEnvironment();\r\n        configuratorStatistics.userType = 'iOneProductPage';\r\n        configuratorStatistics.transactionType = 'Sales';\r\n        configuratorStatistics.actionDomain = 'bundle';\r\n        configuratorStatistics.sessionId = this._settingsService.settings.session ? this._settingsService.settings.session.sessionId : 'unknown';\r\n        configuratorStatistics.webHost = window.location.host;\r\n        configuratorStatistics.bundleHost = window.location.host;\r\n        return await this._iOne.getJsonArticleFlatTree(\r\n            article.article.goodId, article.article.goodType, article.quantity, !!this.externalSource, true, configuratorStatistics);\r\n    }\r\n\r\n    private async _loadImages(): Promise<void> {\r\n        this.articleViewModels.forEach(avm => {\r\n            if (avm.article.image) {\r\n                this._iOne.getImageForCoDocument(avm.article.image, false).then(\r\n                  imageContent => this._handleImageData(avm, imageContent));\r\n            }\r\n        })\r\n    }\r\n\r\n    private _handleImageData(avm: ArticleViewModel, imageContent: ImageContent): void {\r\n        if (imageContent && imageContent.image) {\r\n            avm.imageData = imageContent.image;\r\n        }\r\n    }\r\n}\r\n"]}
@@ -3,8 +3,8 @@ export class Version {
3
3
  constructor() {
4
4
  this.name = "@colijnit/product";
5
5
  this.description = "Product detail page project for iOne";
6
- this.symVer = "260.1.2";
7
- this.publishDate = "5-1-2026 18:13:02";
6
+ this.symVer = "260.1.3";
7
+ this.publishDate = "24-3-2026 16:21:40";
8
8
  }
9
9
  }
10
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvZHVjdC12ZXJzaW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2FwcC9wcm9kdWN0LXZlcnNpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsdURBQXVEO0FBQ3ZELE1BQU0sT0FBTyxPQUFPO0lBQXBCO1FBQ1MsU0FBSSxHQUFHLG1CQUFtQixDQUFDO1FBQzNCLGdCQUFXLEdBQUcsc0NBQXNDLENBQUM7UUFDckQsV0FBTSxHQUFHLFNBQVMsQ0FBQztRQUNuQixnQkFBVyxHQUFHLG1CQUFtQixDQUFDO0lBQzNDLENBQUM7Q0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIHRoaXMgZmlsZSBpcyBkeW5hbWljYWxseSBjcmVhdGVkLCBkbyBub3QgY2hhbmdlIHRoaXNcbmV4cG9ydCBjbGFzcyBWZXJzaW9uIHtcbiAgcHVibGljIG5hbWUgPSBcIkBjb2xpam5pdC9wcm9kdWN0XCI7XG4gIHB1YmxpYyBkZXNjcmlwdGlvbiA9IFwiUHJvZHVjdCBkZXRhaWwgcGFnZSBwcm9qZWN0IGZvciBpT25lXCI7XG4gIHB1YmxpYyBzeW1WZXIgPSBcIjI2MC4xLjJcIjtcbiAgcHVibGljIHB1Ymxpc2hEYXRlID0gXCI1LTEtMjAyNiAxODoxMzowMlwiO1xufSJdfQ==
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvZHVjdC12ZXJzaW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2FwcC9wcm9kdWN0LXZlcnNpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsdURBQXVEO0FBQ3ZELE1BQU0sT0FBTyxPQUFPO0lBQXBCO1FBQ1MsU0FBSSxHQUFHLG1CQUFtQixDQUFDO1FBQzNCLGdCQUFXLEdBQUcsc0NBQXNDLENBQUM7UUFDckQsV0FBTSxHQUFHLFNBQVMsQ0FBQztRQUNuQixnQkFBVyxHQUFHLG9CQUFvQixDQUFDO0lBQzVDLENBQUM7Q0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIHRoaXMgZmlsZSBpcyBkeW5hbWljYWxseSBjcmVhdGVkLCBkbyBub3QgY2hhbmdlIHRoaXNcbmV4cG9ydCBjbGFzcyBWZXJzaW9uIHtcbiAgcHVibGljIG5hbWUgPSBcIkBjb2xpam5pdC9wcm9kdWN0XCI7XG4gIHB1YmxpYyBkZXNjcmlwdGlvbiA9IFwiUHJvZHVjdCBkZXRhaWwgcGFnZSBwcm9qZWN0IGZvciBpT25lXCI7XG4gIHB1YmxpYyBzeW1WZXIgPSBcIjI2MC4xLjNcIjtcbiAgcHVibGljIHB1Ymxpc2hEYXRlID0gXCIyNC0zLTIwMjYgMTY6MjE6NDBcIjtcbn0iXX0=