@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.
- package/app/components/image-carousel/image-carousel.component.d.ts +6 -2
- package/app/components/product-related/product-related.component.d.ts +1 -0
- package/bundles/colijnit-product.umd.js +70 -48
- package/bundles/colijnit-product.umd.js.map +1 -1
- package/colijnit-product.metadata.json +1 -1
- package/esm2015/app/components/image-carousel/image-carousel.component.js +100 -86
- package/esm2015/app/components/image-carousel/image-carousel.module.js +6 -2
- package/esm2015/app/components/product-related/product-related.component.js +7 -2
- package/esm2015/app/product-version.js +3 -3
- package/fesm2015/colijnit-product.js +111 -91
- package/fesm2015/colijnit-product.js.map +1 -1
- package/package.json +1 -1
|
@@ -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
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
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
|
-
|
|
163
|
-
|
|
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
|
-
|
|
181
|
-
//
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
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
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
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
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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,
|
|
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
|
|
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.
|
|
7
|
-
this.publishDate = "
|
|
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,
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvZHVjdC12ZXJzaW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2FwcC9wcm9kdWN0LXZlcnNpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsdURBQXVEO0FBQ3ZELE1BQU0sT0FBTyxPQUFPO0lBQXBCO1FBQ1MsU0FBSSxHQUFHLG1CQUFtQixDQUFDO1FBQzNCLGdCQUFXLEdBQUcsc0NBQXNDLENBQUM7UUFDckQsV0FBTSxHQUFHLFNBQVMsQ0FBQztRQUNuQixnQkFBVyxHQUFHLG9CQUFvQixDQUFDO0lBQzVDLENBQUM7Q0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIHRoaXMgZmlsZSBpcyBkeW5hbWljYWxseSBjcmVhdGVkLCBkbyBub3QgY2hhbmdlIHRoaXNcbmV4cG9ydCBjbGFzcyBWZXJzaW9uIHtcbiAgcHVibGljIG5hbWUgPSBcIkBjb2xpam5pdC9wcm9kdWN0XCI7XG4gIHB1YmxpYyBkZXNjcmlwdGlvbiA9IFwiUHJvZHVjdCBkZXRhaWwgcGFnZSBwcm9qZWN0IGZvciBpT25lXCI7XG4gIHB1YmxpYyBzeW1WZXIgPSBcIjI2MC4xLjNcIjtcbiAgcHVibGljIHB1Ymxpc2hEYXRlID0gXCIyNC0zLTIwMjYgMTY6MjE6NDBcIjtcbn0iXX0=
|