@edm-sdui/sdui 1.0.67 → 1.0.68

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.
Binary file
@@ -1,12 +1,16 @@
1
- import { Component, ElementRef, Inject, Renderer2, ViewChild, } from '@angular/core';
1
+ import { Component, ElementRef, Inject, NgZone, Renderer2, ViewChild, } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
3
  import * as i1 from "@angular/common";
4
4
  import * as i2 from "../uicomponent.component";
5
5
  import * as i3 from "../../../directives/uiview.directive";
6
6
  export class ScrollableRowComponent {
7
- constructor(uiComponent, renderer) {
7
+ constructor(uiComponent, renderer, zone) {
8
8
  this.uiComponent = uiComponent;
9
9
  this.renderer = renderer;
10
+ this.zone = zone;
11
+ this.canScroll = false;
12
+ this.canPrev = false;
13
+ this.canNext = false;
10
14
  }
11
15
  ngOnInit() {
12
16
  }
@@ -15,10 +19,74 @@ export class ScrollableRowComponent {
15
19
  this.applyElement(this.uiComponent.element);
16
20
  }
17
21
  this.scroll();
22
+ setTimeout(() => this.updateTrackState(), 0);
23
+ this.resizeListener = () => this.updateTrackState();
24
+ window.addEventListener('resize', this.resizeListener);
25
+ if (typeof ResizeObserver !== 'undefined' && this.scrollableRowElementRef?.nativeElement) {
26
+ this.zone.runOutsideAngular(() => {
27
+ this.resizeObserver = new ResizeObserver(() => {
28
+ this.zone.run(() => this.updateTrackState());
29
+ });
30
+ this.resizeObserver.observe(this.scrollableRowElementRef.nativeElement);
31
+ });
32
+ }
33
+ }
34
+ ngOnDestroy() {
35
+ if (this.resizeListener) {
36
+ window.removeEventListener('resize', this.resizeListener);
37
+ }
38
+ this.resizeObserver?.disconnect();
39
+ }
40
+ onTrackScroll() {
41
+ this.updateTrackState();
42
+ }
43
+ prev() {
44
+ this.scrollByStep(-1);
45
+ }
46
+ next() {
47
+ this.scrollByStep(1);
18
48
  }
19
49
  applyElement(element) {
20
50
  console.log(element);
21
51
  }
52
+ updateTrackState() {
53
+ const track = this.scrollableRowElementRef?.nativeElement;
54
+ if (!track) {
55
+ this.canScroll = false;
56
+ this.canPrev = false;
57
+ this.canNext = false;
58
+ return;
59
+ }
60
+ const maxLeft = Math.max(0, track.scrollWidth - track.clientWidth);
61
+ const left = Math.max(0, track.scrollLeft);
62
+ const tolerance = 2;
63
+ this.canScroll = maxLeft > tolerance;
64
+ this.canPrev = this.canScroll && left > tolerance;
65
+ this.canNext = this.canScroll && left < maxLeft - tolerance;
66
+ }
67
+ scrollByStep(direction) {
68
+ const track = this.scrollableRowElementRef?.nativeElement;
69
+ if (!track)
70
+ return;
71
+ const maxLeft = Math.max(0, track.scrollWidth - track.clientWidth);
72
+ const step = this.getTrackStep(track);
73
+ const targetLeft = Math.max(0, Math.min(maxLeft, track.scrollLeft + step * direction));
74
+ track.scrollTo({ left: targetLeft, behavior: 'smooth' });
75
+ setTimeout(() => this.updateTrackState(), 260);
76
+ }
77
+ getTrackStep(track) {
78
+ const firstItem = track.firstElementChild;
79
+ if (!firstItem)
80
+ return Math.max(1, track.clientWidth * 0.92);
81
+ const itemWidth = firstItem.getBoundingClientRect().width;
82
+ const styles = getComputedStyle(track);
83
+ const gapRaw = styles.columnGap || styles.gap || '0';
84
+ const gap = Number.parseFloat(gapRaw) || 0;
85
+ const cardStep = Math.max(1, itemWidth + gap);
86
+ const viewportStep = track.clientWidth * 0.92;
87
+ const cardsPerJump = Math.max(2, Math.floor(viewportStep / cardStep));
88
+ return Math.max(cardStep * cardsPerJump, viewportStep);
89
+ }
22
90
  scroll() {
23
91
  const scrollContainer = this.scrollableRowElementRef.nativeElement;
24
92
  let isDown = false;
@@ -47,17 +115,17 @@ export class ScrollableRowComponent {
47
115
  scrollContainer.scrollLeft = scrollLeft - walk;
48
116
  });
49
117
  }
50
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.14", ngImport: i0, type: ScrollableRowComponent, deps: [{ token: 'uiComponent' }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); }
51
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "19.2.14", type: ScrollableRowComponent, isStandalone: false, selector: "edm-sdui-scrollable-row", viewQueries: [{ propertyName: "scrollableRowElementRef", first: true, predicate: ["scrollableRowElement"], descendants: true }], ngImport: i0, template: "<div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n *ngIf=\"uiComponent\"\n>\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n</div>\n", styles: [":host{display:contents}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.UIComponentComponent, selector: "edm-sdui-component", inputs: ["uiComponent", "translateLabel"] }, { kind: "directive", type: i3.UIViewDirective, selector: "[edmSduiView]", inputs: ["edmSduiView", "disableClick"] }] }); }
118
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.14", ngImport: i0, type: ScrollableRowComponent, deps: [{ token: 'uiComponent' }, { token: i0.Renderer2 }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
119
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "19.2.14", type: ScrollableRowComponent, isStandalone: false, selector: "edm-sdui-scrollable-row", viewQueries: [{ propertyName: "scrollableRowElementRef", first: true, predicate: ["scrollableRowElement"], descendants: true }], ngImport: i0, template: "<div\n class=\"scrollable-row-shell\"\n [class.can-scroll]=\"canScroll\"\n [class.track-scrolled]=\"canPrev\"\n [class.track-at-end]=\"canScroll && !canNext\"\n *ngIf=\"uiComponent\"\n>\n <div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n (scroll)=\"onTrackScroll()\"\n >\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n </div>\n\n <button\n *ngIf=\"canScroll && canPrev\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--prev\"\n (click)=\"prev()\"\n aria-label=\"Anterior\"\n >\n &#8592;\n </button>\n <button\n *ngIf=\"canScroll && canNext\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--next\"\n (click)=\"next()\"\n aria-label=\"Pr\u00F3ximo\"\n >\n &#8594;\n </button>\n</div>\n", styles: [":host{display:contents}.scrollable-row-shell{position:relative;width:100%}.scrollable-row-shell:before,.scrollable-row-shell:after{content:\"\";position:absolute;top:0;bottom:0;width:clamp(1.5rem,4vw,2.625rem);pointer-events:none;z-index:4;opacity:0;transition:opacity .2s ease}.scrollable-row-shell:before{left:0;background:linear-gradient(90deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell:after{right:0;background:linear-gradient(270deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell.can-scroll:after{opacity:1}.scrollable-row-shell.track-scrolled:before{opacity:1}.scrollable-row-shell.track-at-end:after{opacity:0}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-behavior:smooth;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}.scrollable-row-nav{position:absolute;top:50%;transform:translateY(-50%);width:44px;height:44px;border:1px solid rgba(255,255,255,.34);background:#070a10c7;color:#f4f6fb;font-size:1.4rem;line-height:1;display:inline-flex;align-items:center;justify-content:center;cursor:pointer;z-index:5;transition:opacity .18s ease,background-color .2s ease,border-color .2s ease}.scrollable-row-nav:hover,.scrollable-row-nav:focus{background:#121622e6;border-color:#ffffff8a;outline:none}.scrollable-row-nav--prev{left:-.75rem}.scrollable-row-nav--next{right:-.75rem}:host-context(.rounded) .scrollable-row-nav{border-radius:3.75rem}:host-context(.square) .scrollable-row-nav{border-radius:10px}::ng-deep .column:has(.scrollable-row-shell){overflow:visible}@media (max-width: 767px){.scrollable-row-nav{display:none}}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.UIComponentComponent, selector: "edm-sdui-component", inputs: ["uiComponent", "translateLabel"] }, { kind: "directive", type: i3.UIViewDirective, selector: "[edmSduiView]", inputs: ["edmSduiView", "disableClick"] }] }); }
52
120
  }
53
121
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.14", ngImport: i0, type: ScrollableRowComponent, decorators: [{
54
122
  type: Component,
55
- args: [{ selector: 'edm-sdui-scrollable-row', standalone: false, template: "<div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n *ngIf=\"uiComponent\"\n>\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n</div>\n", styles: [":host{display:contents}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}\n"] }]
123
+ args: [{ selector: 'edm-sdui-scrollable-row', standalone: false, template: "<div\n class=\"scrollable-row-shell\"\n [class.can-scroll]=\"canScroll\"\n [class.track-scrolled]=\"canPrev\"\n [class.track-at-end]=\"canScroll && !canNext\"\n *ngIf=\"uiComponent\"\n>\n <div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n (scroll)=\"onTrackScroll()\"\n >\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n </div>\n\n <button\n *ngIf=\"canScroll && canPrev\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--prev\"\n (click)=\"prev()\"\n aria-label=\"Anterior\"\n >\n &#8592;\n </button>\n <button\n *ngIf=\"canScroll && canNext\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--next\"\n (click)=\"next()\"\n aria-label=\"Pr\u00F3ximo\"\n >\n &#8594;\n </button>\n</div>\n", styles: [":host{display:contents}.scrollable-row-shell{position:relative;width:100%}.scrollable-row-shell:before,.scrollable-row-shell:after{content:\"\";position:absolute;top:0;bottom:0;width:clamp(1.5rem,4vw,2.625rem);pointer-events:none;z-index:4;opacity:0;transition:opacity .2s ease}.scrollable-row-shell:before{left:0;background:linear-gradient(90deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell:after{right:0;background:linear-gradient(270deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell.can-scroll:after{opacity:1}.scrollable-row-shell.track-scrolled:before{opacity:1}.scrollable-row-shell.track-at-end:after{opacity:0}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-behavior:smooth;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}.scrollable-row-nav{position:absolute;top:50%;transform:translateY(-50%);width:44px;height:44px;border:1px solid rgba(255,255,255,.34);background:#070a10c7;color:#f4f6fb;font-size:1.4rem;line-height:1;display:inline-flex;align-items:center;justify-content:center;cursor:pointer;z-index:5;transition:opacity .18s ease,background-color .2s ease,border-color .2s ease}.scrollable-row-nav:hover,.scrollable-row-nav:focus{background:#121622e6;border-color:#ffffff8a;outline:none}.scrollable-row-nav--prev{left:-.75rem}.scrollable-row-nav--next{right:-.75rem}:host-context(.rounded) .scrollable-row-nav{border-radius:3.75rem}:host-context(.square) .scrollable-row-nav{border-radius:10px}::ng-deep .column:has(.scrollable-row-shell){overflow:visible}@media (max-width: 767px){.scrollable-row-nav{display:none}}\n"] }]
56
124
  }], ctorParameters: () => [{ type: undefined, decorators: [{
57
125
  type: Inject,
58
126
  args: ['uiComponent']
59
- }] }, { type: i0.Renderer2 }], propDecorators: { scrollableRowElementRef: [{
127
+ }] }, { type: i0.Renderer2 }, { type: i0.NgZone }], propDecorators: { scrollableRowElementRef: [{
60
128
  type: ViewChild,
61
129
  args: ['scrollableRowElement']
62
130
  }] } });
63
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Nyb2xsYWJsZS1yb3cuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9zZHVpL3NyYy9saWIvY29tcG9uZW50cy91aWNvbXBvbmVudC9zY3JvbGxhYmxlLXJvdy9zY3JvbGxhYmxlLXJvdy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3NkdWkvc3JjL2xpYi9jb21wb25lbnRzL3VpY29tcG9uZW50L3Njcm9sbGFibGUtcm93L3Njcm9sbGFibGUtcm93LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFFTCxTQUFTLEVBQ1QsVUFBVSxFQUNWLE1BQU0sRUFFTixTQUFTLEVBQ1QsU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDOzs7OztBQWN2QixNQUFNLE9BQU8sc0JBQXNCO0lBR2pDLFlBQ2dDLFdBQXdCLEVBQzlDLFFBQW1CO1FBREcsZ0JBQVcsR0FBWCxXQUFXLENBQWE7UUFDOUMsYUFBUSxHQUFSLFFBQVEsQ0FBVztJQUN6QixDQUFDO0lBRUwsUUFBUTtJQUNSLENBQUM7SUFFRCxlQUFlO1FBQ2IsSUFBSSxJQUFJLENBQUMsV0FBVyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQzdCLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUM5QyxDQUFDO1FBQ0QsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDO0lBQ2hCLENBQUM7SUFFTyxZQUFZLENBQUMsT0FBa0I7UUFDckMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQTtJQUN0QixDQUFDO0lBR08sTUFBTTtRQUNaLE1BQU0sZUFBZSxHQUFHLElBQUksQ0FBQyx1QkFBdUIsQ0FBQyxhQUFhLENBQUM7UUFDbkUsSUFBSSxNQUFNLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksTUFBTSxHQUFHLENBQUMsQ0FBQztRQUNmLElBQUksVUFBVSxHQUFHLENBQUMsQ0FBQztRQUVuQixJQUFJLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxlQUFlLEVBQUUsV0FBVyxFQUFFLENBQUMsQ0FBYSxFQUFFLEVBQUU7WUFDbkUsTUFBTSxHQUFHLElBQUksQ0FBQztZQUNkLGVBQWUsQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1lBQzNDLE1BQU0sR0FBRyxDQUFDLENBQUMsS0FBSyxHQUFHLGVBQWUsQ0FBQyxVQUFVLENBQUM7WUFDOUMsVUFBVSxHQUFHLGVBQWUsQ0FBQyxVQUFVLENBQUM7UUFDMUMsQ0FBQyxDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxlQUFlLEVBQUUsWUFBWSxFQUFFLEdBQUcsRUFBRTtZQUN2RCxNQUFNLEdBQUcsS0FBSyxDQUFDO1lBQ2YsZUFBZSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7UUFDaEQsQ0FBQyxDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxlQUFlLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBRTtZQUNwRCxNQUFNLEdBQUcsS0FBSyxDQUFDO1lBQ2YsZUFBZSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7UUFDaEQsQ0FBQyxDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxlQUFlLEVBQUUsV0FBVyxFQUFFLENBQUMsQ0FBYSxFQUFFLEVBQUU7WUFDbkUsSUFBSSxDQUFDLE1BQU07Z0JBQUUsT0FBTztZQUNwQixDQUFDLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDbkIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFDLEtBQUssR0FBRyxlQUFlLENBQUMsVUFBVSxDQUFDO1lBQy9DLE1BQU0sSUFBSSxHQUFHLENBQUMsQ0FBQyxHQUFHLE1BQU0sQ0FBQyxHQUFHLEdBQUcsQ0FBQztZQUNoQyxlQUFlLENBQUMsVUFBVSxHQUFHLFVBQVUsR0FBRyxJQUFJLENBQUM7UUFDakQsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDOytHQXJEVSxzQkFBc0Isa0JBSXZCLGFBQWE7bUdBSlosc0JBQXNCLHFOQ3RCbkMsd1JBWUE7OzRGRFVhLHNCQUFzQjtrQkFObEMsU0FBUzsrQkFDRSx5QkFBeUIsY0FHdkIsS0FBSzs7MEJBTWQsTUFBTTsyQkFBQyxhQUFhO2lFQUhZLHVCQUF1QjtzQkFBekQsU0FBUzt1QkFBQyxzQkFBc0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBZnRlclZpZXdJbml0LFxuICBDb21wb25lbnQsXG4gIEVsZW1lbnRSZWYsXG4gIEluamVjdCxcbiAgT25Jbml0LFxuICBSZW5kZXJlcjIsXG4gIFZpZXdDaGlsZCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBVSUNvbXBvbmVudCB9IGZyb20gJy4uLy4uLy4uL2NvcmUvdWljb21wb3NpdGlvbi9tb2RlbHMvdWljb21wb25lbnQnO1xuaW1wb3J0IHsgVUlFbGVtZW50IH0gZnJvbSAnLi4vLi4vLi4vY29yZS91aWNvbXBvc2l0aW9uL21vZGVscy91aWVsZW1lbnQnO1xuaW1wb3J0IHsgY29sb3JNYXBwaW5nIH0gZnJvbSAnLi4vLi4vLi4vY29yZS91aXRoZW1lL21hcHBpbmcvY29sb3ItbWFwcGluZyc7XG5pbXBvcnQgeyBGb250U2l6ZU1hcHBpbmdTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vLi4vY29yZS9zZXJ2aWNlcy9mb250LXNpemUtbWFwcGluZy5zZXJ2aWNlJztcbmltcG9ydCB7IGFzc2V0TWFwcGluZyB9IGZyb20gJy4uLy4uLy4uL2NvcmUvdWl0aGVtZS9tYXBwaW5nL2Fzc2V0LW1hcHBpbmcnO1xuaW1wb3J0IHsgVUlBc3NldCB9IGZyb20gJy4uLy4uLy4uL2NvcmUvdWl0aGVtZS9lbnVtcy91aWFzc2V0JztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnZWRtLXNkdWktc2Nyb2xsYWJsZS1yb3cnLFxuICB0ZW1wbGF0ZVVybDogJy4vc2Nyb2xsYWJsZS1yb3cuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybDogJy4vc2Nyb2xsYWJsZS1yb3cuY29tcG9uZW50LnNjc3MnLFxuICBzdGFuZGFsb25lOiBmYWxzZSxcbn0pXG5leHBvcnQgY2xhc3MgU2Nyb2xsYWJsZVJvd0NvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCwgQWZ0ZXJWaWV3SW5pdCB7XG4gIEBWaWV3Q2hpbGQoJ3Njcm9sbGFibGVSb3dFbGVtZW50Jykgc2Nyb2xsYWJsZVJvd0VsZW1lbnRSZWYhOiBFbGVtZW50UmVmPEhUTUxEaXZFbGVtZW50PjtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBASW5qZWN0KCd1aUNvbXBvbmVudCcpIHB1YmxpYyB1aUNvbXBvbmVudDogVUlDb21wb25lbnQsXG4gICAgcHJpdmF0ZSByZW5kZXJlcjogUmVuZGVyZXIyLFxuICApIHsgfVxuXG4gIG5nT25Jbml0KCk6IHZvaWQge1xuICB9XG5cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnVpQ29tcG9uZW50LmVsZW1lbnQpIHtcbiAgICAgIHRoaXMuYXBwbHlFbGVtZW50KHRoaXMudWlDb21wb25lbnQuZWxlbWVudCk7XG4gICAgfVxuICAgIHRoaXMuc2Nyb2xsKCk7XG4gIH1cblxuICBwcml2YXRlIGFwcGx5RWxlbWVudChlbGVtZW50OiBVSUVsZW1lbnQpIHtcbiAgICBjb25zb2xlLmxvZyhlbGVtZW50KVxuICB9XG5cblxuICBwcml2YXRlIHNjcm9sbCgpIHtcbiAgICBjb25zdCBzY3JvbGxDb250YWluZXIgPSB0aGlzLnNjcm9sbGFibGVSb3dFbGVtZW50UmVmLm5hdGl2ZUVsZW1lbnQ7XG4gICAgbGV0IGlzRG93biA9IGZhbHNlO1xuICAgIGxldCBzdGFydFggPSAwO1xuICAgIGxldCBzY3JvbGxMZWZ0ID0gMDtcblxuICAgIHRoaXMucmVuZGVyZXIubGlzdGVuKHNjcm9sbENvbnRhaW5lciwgJ21vdXNlZG93bicsIChlOiBNb3VzZUV2ZW50KSA9PiB7XG4gICAgICBpc0Rvd24gPSB0cnVlO1xuICAgICAgc2Nyb2xsQ29udGFpbmVyLmNsYXNzTGlzdC5hZGQoJ3Njcm9sbGluZycpO1xuICAgICAgc3RhcnRYID0gZS5wYWdlWCAtIHNjcm9sbENvbnRhaW5lci5vZmZzZXRMZWZ0O1xuICAgICAgc2Nyb2xsTGVmdCA9IHNjcm9sbENvbnRhaW5lci5zY3JvbGxMZWZ0O1xuICAgIH0pO1xuXG4gICAgdGhpcy5yZW5kZXJlci5saXN0ZW4oc2Nyb2xsQ29udGFpbmVyLCAnbW91c2VsZWF2ZScsICgpID0+IHtcbiAgICAgIGlzRG93biA9IGZhbHNlO1xuICAgICAgc2Nyb2xsQ29udGFpbmVyLmNsYXNzTGlzdC5yZW1vdmUoJ3Njcm9sbGluZycpO1xuICAgIH0pO1xuXG4gICAgdGhpcy5yZW5kZXJlci5saXN0ZW4oc2Nyb2xsQ29udGFpbmVyLCAnbW91c2V1cCcsICgpID0+IHtcbiAgICAgIGlzRG93biA9IGZhbHNlO1xuICAgICAgc2Nyb2xsQ29udGFpbmVyLmNsYXNzTGlzdC5yZW1vdmUoJ3Njcm9sbGluZycpO1xuICAgIH0pO1xuXG4gICAgdGhpcy5yZW5kZXJlci5saXN0ZW4oc2Nyb2xsQ29udGFpbmVyLCAnbW91c2Vtb3ZlJywgKGU6IE1vdXNlRXZlbnQpID0+IHtcbiAgICAgIGlmICghaXNEb3duKSByZXR1cm47XG4gICAgICBlLnByZXZlbnREZWZhdWx0KCk7XG4gICAgICBjb25zdCB4ID0gZS5wYWdlWCAtIHNjcm9sbENvbnRhaW5lci5vZmZzZXRMZWZ0O1xuICAgICAgY29uc3Qgd2FsayA9ICh4IC0gc3RhcnRYKSAqIDEuNTtcbiAgICAgIHNjcm9sbENvbnRhaW5lci5zY3JvbGxMZWZ0ID0gc2Nyb2xsTGVmdCAtIHdhbGs7XG4gICAgfSk7XG4gIH1cbn1cbiIsIjxkaXZcbiAgI3Njcm9sbGFibGVSb3dFbGVtZW50XG4gIFtlZG1TZHVpVmlld109XCJ1aUNvbXBvbmVudC5lbGVtZW50XCJcbiAgY2xhc3M9XCJzY3JvbGxhYmxlLXJvd1wiXG4gICpuZ0lmPVwidWlDb21wb25lbnRcIlxuPlxuICA8ZWRtLXNkdWktY29tcG9uZW50XG4gICAgKm5nRm9yPVwibGV0IGNvbXBvbmVudCBvZiB1aUNvbXBvbmVudC5jb21wb25lbnRzXCJcbiAgICBbdWlDb21wb25lbnRdPVwiY29tcG9uZW50XCJcbiAgPlxuICA8L2VkbS1zZHVpLWNvbXBvbmVudD5cbjwvZGl2PlxuIl19
131
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable-row.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdui/src/lib/components/uicomponent/scrollable-row/scrollable-row.component.ts","../../../../../../../../libs/sdui/src/lib/components/uicomponent/scrollable-row/scrollable-row.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEL,SAAS,EACT,UAAU,EACV,MAAM,EACN,MAAM,EAGN,SAAS,EACT,SAAS,GACV,MAAM,eAAe,CAAC;;;;;AAUvB,MAAM,OAAO,sBAAsB;IAUjC,YACgC,WAAwB,EAC9C,QAAmB,EACnB,IAAY;QAFU,gBAAW,GAAX,WAAW,CAAa;QAC9C,aAAQ,GAAR,QAAQ,CAAW;QACnB,SAAI,GAAJ,IAAI,CAAQ;QAVf,cAAS,GAAY,KAAK,CAAC;QAC3B,YAAO,GAAY,KAAK,CAAC;QACzB,YAAO,GAAY,KAAK,CAAC;IAS5B,CAAC;IAEL,QAAQ;IACR,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;YAC7B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QAC9C,CAAC;QACD,IAAI,CAAC,MAAM,EAAE,CAAC;QACd,UAAU,CAAC,GAAS,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,EAAE,CAAC,CAAC,CAAC;QAEnD,IAAI,CAAC,cAAc,GAAG,GAAS,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC1D,MAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;QAEvD,IAAI,OAAO,cAAc,KAAK,WAAW,IAAI,IAAI,CAAC,uBAAuB,EAAE,aAAa,EAAE,CAAC;YACzF,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAS,EAAE;gBACrC,IAAI,CAAC,cAAc,GAAG,IAAI,cAAc,CAAC,GAAS,EAAE;oBAClD,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAS,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC;gBACrD,CAAC,CAAC,CAAC;gBACH,IAAI,CAAC,cAAe,CAAC,OAAO,CAAC,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,CAAC;YAC3E,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,MAAM,CAAC,mBAAmB,CAAC,QAAQ,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;QAC5D,CAAC;QACD,IAAI,CAAC,cAAc,EAAE,UAAU,EAAE,CAAC;IACpC,CAAC;IAED,aAAa;QACX,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED,IAAI;QACF,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;IACxB,CAAC;IAED,IAAI;QACF,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IACvB,CAAC;IAEO,YAAY,CAAC,OAAkB;QACrC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;IACtB,CAAC;IAEO,gBAAgB;QACtB,MAAM,KAAK,GAAG,IAAI,CAAC,uBAAuB,EAAE,aAAa,CAAC;QAC1D,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;YACvB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YACrB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YACrB,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAW,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,CAAC;QAC3E,MAAM,IAAI,GAAW,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC,UAAU,CAAC,CAAC;QACnD,MAAM,SAAS,GAAW,CAAC,CAAC;QAE5B,IAAI,CAAC,SAAS,GAAG,OAAO,GAAG,SAAS,CAAC;QACrC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,SAAS,IAAI,IAAI,GAAG,SAAS,CAAC;QAClD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,SAAS,IAAI,IAAI,GAAG,OAAO,GAAG,SAAS,CAAC;IAC9D,CAAC;IAEO,YAAY,CAAC,SAAiB;QACpC,MAAM,KAAK,GAAG,IAAI,CAAC,uBAAuB,EAAE,aAAa,CAAC;QAC1D,IAAI,CAAC,KAAK;YAAE,OAAO;QAEnB,MAAM,OAAO,GAAW,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,CAAC;QAC3E,MAAM,IAAI,GAAW,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAC9C,MAAM,UAAU,GAAW,IAAI,CAAC,GAAG,CACjC,CAAC,EACD,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,UAAU,GAAG,IAAI,GAAG,SAAS,CAAC,CACvD,CAAC;QAEF,KAAK,CAAC,QAAQ,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,CAAC;QACzD,UAAU,CAAC,GAAS,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,EAAE,GAAG,CAAC,CAAC;IACvD,CAAC;IAEO,YAAY,CAAC,KAAkB;QACrC,MAAM,SAAS,GAAG,KAAK,CAAC,iBAAuC,CAAC;QAChE,IAAI,CAAC,SAAS;YAAE,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC,CAAC;QAE7D,MAAM,SAAS,GAAW,SAAS,CAAC,qBAAqB,EAAE,CAAC,KAAK,CAAC;QAClE,MAAM,MAAM,GAAwB,gBAAgB,CAAC,KAAK,CAAC,CAAC;QAC5D,MAAM,MAAM,GAAW,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,GAAG,IAAI,GAAG,CAAC;QAC7D,MAAM,GAAG,GAAW,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACnD,MAAM,QAAQ,GAAW,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,SAAS,GAAG,GAAG,CAAC,CAAC;QACtD,MAAM,YAAY,GAAW,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC;QACtD,MAAM,YAAY,GAAW,IAAI,CAAC,GAAG,CACnC,CAAC,EACD,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,QAAQ,CAAC,CACpC,CAAC;QAEF,OAAO,IAAI,CAAC,GAAG,CAAC,QAAQ,GAAG,YAAY,EAAE,YAAY,CAAC,CAAC;IACzD,CAAC;IAEO,MAAM;QACZ,MAAM,eAAe,GAAG,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC;QACnE,IAAI,MAAM,GAAG,KAAK,CAAC;QACnB,IAAI,MAAM,GAAG,CAAC,CAAC;QACf,IAAI,UAAU,GAAG,CAAC,CAAC;QAEnB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,eAAe,EAAE,WAAW,EAAE,CAAC,CAAa,EAAE,EAAE;YACnE,MAAM,GAAG,IAAI,CAAC;YACd,eAAe,CAAC,SAAS,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;YAC3C,MAAM,GAAG,CAAC,CAAC,KAAK,GAAG,eAAe,CAAC,UAAU,CAAC;YAC9C,UAAU,GAAG,eAAe,CAAC,UAAU,CAAC;QAC1C,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,eAAe,EAAE,YAAY,EAAE,GAAG,EAAE;YACvD,MAAM,GAAG,KAAK,CAAC;YACf,eAAe,CAAC,SAAS,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,eAAe,EAAE,SAAS,EAAE,GAAG,EAAE;YACpD,MAAM,GAAG,KAAK,CAAC;YACf,eAAe,CAAC,SAAS,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,eAAe,EAAE,WAAW,EAAE,CAAC,CAAa,EAAE,EAAE;YACnE,IAAI,CAAC,MAAM;gBAAE,OAAO;YACpB,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,GAAG,eAAe,CAAC,UAAU,CAAC;YAC/C,MAAM,IAAI,GAAG,CAAC,CAAC,GAAG,MAAM,CAAC,GAAG,GAAG,CAAC;YAChC,eAAe,CAAC,UAAU,GAAG,UAAU,GAAG,IAAI,CAAC;QACjD,CAAC,CAAC,CAAC;IACL,CAAC;+GA/IU,sBAAsB,kBAWvB,aAAa;mGAXZ,sBAAsB,qNCpBnC,26BAuCA;;4FDnBa,sBAAsB;kBANlC,SAAS;+BACE,yBAAyB,cAGvB,KAAK;;0BAad,MAAM;2BAAC,aAAa;sFAVY,uBAAuB;sBAAzD,SAAS;uBAAC,sBAAsB","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  Inject,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Renderer2,\n  ViewChild,\n} from '@angular/core';\nimport { UIComponent } from '../../../core/uicomposition/models/uicomponent';\nimport { UIElement } from '../../../core/uicomposition/models/uielement';\n\n@Component({\n  selector: 'edm-sdui-scrollable-row',\n  templateUrl: './scrollable-row.component.html',\n  styleUrl: './scrollable-row.component.scss',\n  standalone: false,\n})\nexport class ScrollableRowComponent implements OnInit, AfterViewInit, OnDestroy {\n  @ViewChild('scrollableRowElement') scrollableRowElementRef!: ElementRef<HTMLDivElement>;\n\n  public canScroll: boolean = false;\n  public canPrev: boolean = false;\n  public canNext: boolean = false;\n\n  private resizeListener?: () => void;\n  private resizeObserver?: ResizeObserver;\n\n  constructor(\n    @Inject('uiComponent') public uiComponent: UIComponent,\n    private renderer: Renderer2,\n    private zone: NgZone,\n  ) { }\n\n  ngOnInit(): void {\n  }\n\n  ngAfterViewInit(): void {\n    if (this.uiComponent.element) {\n      this.applyElement(this.uiComponent.element);\n    }\n    this.scroll();\n    setTimeout((): void => this.updateTrackState(), 0);\n\n    this.resizeListener = (): void => this.updateTrackState();\n    window.addEventListener('resize', this.resizeListener);\n\n    if (typeof ResizeObserver !== 'undefined' && this.scrollableRowElementRef?.nativeElement) {\n      this.zone.runOutsideAngular((): void => {\n        this.resizeObserver = new ResizeObserver((): void => {\n          this.zone.run((): void => this.updateTrackState());\n        });\n        this.resizeObserver!.observe(this.scrollableRowElementRef.nativeElement);\n      });\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.resizeListener) {\n      window.removeEventListener('resize', this.resizeListener);\n    }\n    this.resizeObserver?.disconnect();\n  }\n\n  onTrackScroll(): void {\n    this.updateTrackState();\n  }\n\n  prev(): void {\n    this.scrollByStep(-1);\n  }\n\n  next(): void {\n    this.scrollByStep(1);\n  }\n\n  private applyElement(element: UIElement) {\n    console.log(element)\n  }\n\n  private updateTrackState(): void {\n    const track = this.scrollableRowElementRef?.nativeElement;\n    if (!track) {\n      this.canScroll = false;\n      this.canPrev = false;\n      this.canNext = false;\n      return;\n    }\n\n    const maxLeft: number = Math.max(0, track.scrollWidth - track.clientWidth);\n    const left: number = Math.max(0, track.scrollLeft);\n    const tolerance: number = 2;\n\n    this.canScroll = maxLeft > tolerance;\n    this.canPrev = this.canScroll && left > tolerance;\n    this.canNext = this.canScroll && left < maxLeft - tolerance;\n  }\n\n  private scrollByStep(direction: 1 | -1): void {\n    const track = this.scrollableRowElementRef?.nativeElement;\n    if (!track) return;\n\n    const maxLeft: number = Math.max(0, track.scrollWidth - track.clientWidth);\n    const step: number = this.getTrackStep(track);\n    const targetLeft: number = Math.max(\n      0,\n      Math.min(maxLeft, track.scrollLeft + step * direction),\n    );\n\n    track.scrollTo({ left: targetLeft, behavior: 'smooth' });\n    setTimeout((): void => this.updateTrackState(), 260);\n  }\n\n  private getTrackStep(track: HTMLElement): number {\n    const firstItem = track.firstElementChild as HTMLElement | null;\n    if (!firstItem) return Math.max(1, track.clientWidth * 0.92);\n\n    const itemWidth: number = firstItem.getBoundingClientRect().width;\n    const styles: CSSStyleDeclaration = getComputedStyle(track);\n    const gapRaw: string = styles.columnGap || styles.gap || '0';\n    const gap: number = Number.parseFloat(gapRaw) || 0;\n    const cardStep: number = Math.max(1, itemWidth + gap);\n    const viewportStep: number = track.clientWidth * 0.92;\n    const cardsPerJump: number = Math.max(\n      2,\n      Math.floor(viewportStep / cardStep),\n    );\n\n    return Math.max(cardStep * cardsPerJump, viewportStep);\n  }\n\n  private scroll() {\n    const scrollContainer = this.scrollableRowElementRef.nativeElement;\n    let isDown = false;\n    let startX = 0;\n    let scrollLeft = 0;\n\n    this.renderer.listen(scrollContainer, 'mousedown', (e: MouseEvent) => {\n      isDown = true;\n      scrollContainer.classList.add('scrolling');\n      startX = e.pageX - scrollContainer.offsetLeft;\n      scrollLeft = scrollContainer.scrollLeft;\n    });\n\n    this.renderer.listen(scrollContainer, 'mouseleave', () => {\n      isDown = false;\n      scrollContainer.classList.remove('scrolling');\n    });\n\n    this.renderer.listen(scrollContainer, 'mouseup', () => {\n      isDown = false;\n      scrollContainer.classList.remove('scrolling');\n    });\n\n    this.renderer.listen(scrollContainer, 'mousemove', (e: MouseEvent) => {\n      if (!isDown) return;\n      e.preventDefault();\n      const x = e.pageX - scrollContainer.offsetLeft;\n      const walk = (x - startX) * 1.5;\n      scrollContainer.scrollLeft = scrollLeft - walk;\n    });\n  }\n}\n","<div\n  class=\"scrollable-row-shell\"\n  [class.can-scroll]=\"canScroll\"\n  [class.track-scrolled]=\"canPrev\"\n  [class.track-at-end]=\"canScroll && !canNext\"\n  *ngIf=\"uiComponent\"\n>\n  <div\n    #scrollableRowElement\n    [edmSduiView]=\"uiComponent.element\"\n    class=\"scrollable-row\"\n    (scroll)=\"onTrackScroll()\"\n  >\n    <edm-sdui-component\n      *ngFor=\"let component of uiComponent.components\"\n      [uiComponent]=\"component\"\n    >\n    </edm-sdui-component>\n  </div>\n\n  <button\n    *ngIf=\"canScroll && canPrev\"\n    type=\"button\"\n    class=\"scrollable-row-nav scrollable-row-nav--prev\"\n    (click)=\"prev()\"\n    aria-label=\"Anterior\"\n  >\n    &#8592;\n  </button>\n  <button\n    *ngIf=\"canScroll && canNext\"\n    type=\"button\"\n    class=\"scrollable-row-nav scrollable-row-nav--next\"\n    (click)=\"next()\"\n    aria-label=\"Próximo\"\n  >\n    &#8594;\n  </button>\n</div>\n"]}
@@ -4131,9 +4131,13 @@ var picker_component = /*#__PURE__*/Object.freeze({
4131
4131
  });
4132
4132
 
4133
4133
  class ScrollableRowComponent {
4134
- constructor(uiComponent, renderer) {
4134
+ constructor(uiComponent, renderer, zone) {
4135
4135
  this.uiComponent = uiComponent;
4136
4136
  this.renderer = renderer;
4137
+ this.zone = zone;
4138
+ this.canScroll = false;
4139
+ this.canPrev = false;
4140
+ this.canNext = false;
4137
4141
  }
4138
4142
  ngOnInit() {
4139
4143
  }
@@ -4142,10 +4146,74 @@ class ScrollableRowComponent {
4142
4146
  this.applyElement(this.uiComponent.element);
4143
4147
  }
4144
4148
  this.scroll();
4149
+ setTimeout(() => this.updateTrackState(), 0);
4150
+ this.resizeListener = () => this.updateTrackState();
4151
+ window.addEventListener('resize', this.resizeListener);
4152
+ if (typeof ResizeObserver !== 'undefined' && this.scrollableRowElementRef?.nativeElement) {
4153
+ this.zone.runOutsideAngular(() => {
4154
+ this.resizeObserver = new ResizeObserver(() => {
4155
+ this.zone.run(() => this.updateTrackState());
4156
+ });
4157
+ this.resizeObserver.observe(this.scrollableRowElementRef.nativeElement);
4158
+ });
4159
+ }
4160
+ }
4161
+ ngOnDestroy() {
4162
+ if (this.resizeListener) {
4163
+ window.removeEventListener('resize', this.resizeListener);
4164
+ }
4165
+ this.resizeObserver?.disconnect();
4166
+ }
4167
+ onTrackScroll() {
4168
+ this.updateTrackState();
4169
+ }
4170
+ prev() {
4171
+ this.scrollByStep(-1);
4172
+ }
4173
+ next() {
4174
+ this.scrollByStep(1);
4145
4175
  }
4146
4176
  applyElement(element) {
4147
4177
  console.log(element);
4148
4178
  }
4179
+ updateTrackState() {
4180
+ const track = this.scrollableRowElementRef?.nativeElement;
4181
+ if (!track) {
4182
+ this.canScroll = false;
4183
+ this.canPrev = false;
4184
+ this.canNext = false;
4185
+ return;
4186
+ }
4187
+ const maxLeft = Math.max(0, track.scrollWidth - track.clientWidth);
4188
+ const left = Math.max(0, track.scrollLeft);
4189
+ const tolerance = 2;
4190
+ this.canScroll = maxLeft > tolerance;
4191
+ this.canPrev = this.canScroll && left > tolerance;
4192
+ this.canNext = this.canScroll && left < maxLeft - tolerance;
4193
+ }
4194
+ scrollByStep(direction) {
4195
+ const track = this.scrollableRowElementRef?.nativeElement;
4196
+ if (!track)
4197
+ return;
4198
+ const maxLeft = Math.max(0, track.scrollWidth - track.clientWidth);
4199
+ const step = this.getTrackStep(track);
4200
+ const targetLeft = Math.max(0, Math.min(maxLeft, track.scrollLeft + step * direction));
4201
+ track.scrollTo({ left: targetLeft, behavior: 'smooth' });
4202
+ setTimeout(() => this.updateTrackState(), 260);
4203
+ }
4204
+ getTrackStep(track) {
4205
+ const firstItem = track.firstElementChild;
4206
+ if (!firstItem)
4207
+ return Math.max(1, track.clientWidth * 0.92);
4208
+ const itemWidth = firstItem.getBoundingClientRect().width;
4209
+ const styles = getComputedStyle(track);
4210
+ const gapRaw = styles.columnGap || styles.gap || '0';
4211
+ const gap = Number.parseFloat(gapRaw) || 0;
4212
+ const cardStep = Math.max(1, itemWidth + gap);
4213
+ const viewportStep = track.clientWidth * 0.92;
4214
+ const cardsPerJump = Math.max(2, Math.floor(viewportStep / cardStep));
4215
+ return Math.max(cardStep * cardsPerJump, viewportStep);
4216
+ }
4149
4217
  scroll() {
4150
4218
  const scrollContainer = this.scrollableRowElementRef.nativeElement;
4151
4219
  let isDown = false;
@@ -4174,16 +4242,16 @@ class ScrollableRowComponent {
4174
4242
  scrollContainer.scrollLeft = scrollLeft - walk;
4175
4243
  });
4176
4244
  }
4177
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.14", ngImport: i0, type: ScrollableRowComponent, deps: [{ token: 'uiComponent' }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); }
4178
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "19.2.14", type: ScrollableRowComponent, isStandalone: false, selector: "edm-sdui-scrollable-row", viewQueries: [{ propertyName: "scrollableRowElementRef", first: true, predicate: ["scrollableRowElement"], descendants: true }], ngImport: i0, template: "<div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n *ngIf=\"uiComponent\"\n>\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n</div>\n", styles: [":host{display:contents}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}\n"], dependencies: [{ kind: "directive", type: i1$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1$1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: UIComponentComponent, selector: "edm-sdui-component", inputs: ["uiComponent", "translateLabel"] }, { kind: "directive", type: UIViewDirective, selector: "[edmSduiView]", inputs: ["edmSduiView", "disableClick"] }] }); }
4245
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.14", ngImport: i0, type: ScrollableRowComponent, deps: [{ token: 'uiComponent' }, { token: i0.Renderer2 }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
4246
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "19.2.14", type: ScrollableRowComponent, isStandalone: false, selector: "edm-sdui-scrollable-row", viewQueries: [{ propertyName: "scrollableRowElementRef", first: true, predicate: ["scrollableRowElement"], descendants: true }], ngImport: i0, template: "<div\n class=\"scrollable-row-shell\"\n [class.can-scroll]=\"canScroll\"\n [class.track-scrolled]=\"canPrev\"\n [class.track-at-end]=\"canScroll && !canNext\"\n *ngIf=\"uiComponent\"\n>\n <div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n (scroll)=\"onTrackScroll()\"\n >\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n </div>\n\n <button\n *ngIf=\"canScroll && canPrev\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--prev\"\n (click)=\"prev()\"\n aria-label=\"Anterior\"\n >\n &#8592;\n </button>\n <button\n *ngIf=\"canScroll && canNext\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--next\"\n (click)=\"next()\"\n aria-label=\"Pr\u00F3ximo\"\n >\n &#8594;\n </button>\n</div>\n", styles: [":host{display:contents}.scrollable-row-shell{position:relative;width:100%}.scrollable-row-shell:before,.scrollable-row-shell:after{content:\"\";position:absolute;top:0;bottom:0;width:clamp(1.5rem,4vw,2.625rem);pointer-events:none;z-index:4;opacity:0;transition:opacity .2s ease}.scrollable-row-shell:before{left:0;background:linear-gradient(90deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell:after{right:0;background:linear-gradient(270deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell.can-scroll:after{opacity:1}.scrollable-row-shell.track-scrolled:before{opacity:1}.scrollable-row-shell.track-at-end:after{opacity:0}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-behavior:smooth;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}.scrollable-row-nav{position:absolute;top:50%;transform:translateY(-50%);width:44px;height:44px;border:1px solid rgba(255,255,255,.34);background:#070a10c7;color:#f4f6fb;font-size:1.4rem;line-height:1;display:inline-flex;align-items:center;justify-content:center;cursor:pointer;z-index:5;transition:opacity .18s ease,background-color .2s ease,border-color .2s ease}.scrollable-row-nav:hover,.scrollable-row-nav:focus{background:#121622e6;border-color:#ffffff8a;outline:none}.scrollable-row-nav--prev{left:-.75rem}.scrollable-row-nav--next{right:-.75rem}:host-context(.rounded) .scrollable-row-nav{border-radius:3.75rem}:host-context(.square) .scrollable-row-nav{border-radius:10px}::ng-deep .column:has(.scrollable-row-shell){overflow:visible}@media (max-width: 767px){.scrollable-row-nav{display:none}}\n"], dependencies: [{ kind: "directive", type: i1$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1$1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: UIComponentComponent, selector: "edm-sdui-component", inputs: ["uiComponent", "translateLabel"] }, { kind: "directive", type: UIViewDirective, selector: "[edmSduiView]", inputs: ["edmSduiView", "disableClick"] }] }); }
4179
4247
  }
4180
4248
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.14", ngImport: i0, type: ScrollableRowComponent, decorators: [{
4181
4249
  type: Component,
4182
- args: [{ selector: 'edm-sdui-scrollable-row', standalone: false, template: "<div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n *ngIf=\"uiComponent\"\n>\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n</div>\n", styles: [":host{display:contents}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}\n"] }]
4250
+ args: [{ selector: 'edm-sdui-scrollable-row', standalone: false, template: "<div\n class=\"scrollable-row-shell\"\n [class.can-scroll]=\"canScroll\"\n [class.track-scrolled]=\"canPrev\"\n [class.track-at-end]=\"canScroll && !canNext\"\n *ngIf=\"uiComponent\"\n>\n <div\n #scrollableRowElement\n [edmSduiView]=\"uiComponent.element\"\n class=\"scrollable-row\"\n (scroll)=\"onTrackScroll()\"\n >\n <edm-sdui-component\n *ngFor=\"let component of uiComponent.components\"\n [uiComponent]=\"component\"\n >\n </edm-sdui-component>\n </div>\n\n <button\n *ngIf=\"canScroll && canPrev\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--prev\"\n (click)=\"prev()\"\n aria-label=\"Anterior\"\n >\n &#8592;\n </button>\n <button\n *ngIf=\"canScroll && canNext\"\n type=\"button\"\n class=\"scrollable-row-nav scrollable-row-nav--next\"\n (click)=\"next()\"\n aria-label=\"Pr\u00F3ximo\"\n >\n &#8594;\n </button>\n</div>\n", styles: [":host{display:contents}.scrollable-row-shell{position:relative;width:100%}.scrollable-row-shell:before,.scrollable-row-shell:after{content:\"\";position:absolute;top:0;bottom:0;width:clamp(1.5rem,4vw,2.625rem);pointer-events:none;z-index:4;opacity:0;transition:opacity .2s ease}.scrollable-row-shell:before{left:0;background:linear-gradient(90deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell:after{right:0;background:linear-gradient(270deg,var(--bg) 0%,transparent 100%)}.scrollable-row-shell.can-scroll:after{opacity:1}.scrollable-row-shell.track-scrolled:before{opacity:1}.scrollable-row-shell.track-at-end:after{opacity:0}.scrollable-row{display:flex;flex-direction:row;overflow-x:auto;overflow-y:hidden;white-space:nowrap;width:100%;-webkit-overflow-scrolling:touch;scrollbar-width:none;cursor:pointer;scroll-behavior:smooth;scroll-padding-left:var(--sdui-content-inset-left, 0px);scroll-padding-right:var(--sdui-content-inset-right, 0px)}.scrollable-row::-webkit-scrollbar{display:none}.scrollable-row>*{flex-shrink:0}.scrollable-row-nav{position:absolute;top:50%;transform:translateY(-50%);width:44px;height:44px;border:1px solid rgba(255,255,255,.34);background:#070a10c7;color:#f4f6fb;font-size:1.4rem;line-height:1;display:inline-flex;align-items:center;justify-content:center;cursor:pointer;z-index:5;transition:opacity .18s ease,background-color .2s ease,border-color .2s ease}.scrollable-row-nav:hover,.scrollable-row-nav:focus{background:#121622e6;border-color:#ffffff8a;outline:none}.scrollable-row-nav--prev{left:-.75rem}.scrollable-row-nav--next{right:-.75rem}:host-context(.rounded) .scrollable-row-nav{border-radius:3.75rem}:host-context(.square) .scrollable-row-nav{border-radius:10px}::ng-deep .column:has(.scrollable-row-shell){overflow:visible}@media (max-width: 767px){.scrollable-row-nav{display:none}}\n"] }]
4183
4251
  }], ctorParameters: () => [{ type: undefined, decorators: [{
4184
4252
  type: Inject,
4185
4253
  args: ['uiComponent']
4186
- }] }, { type: i0.Renderer2 }], propDecorators: { scrollableRowElementRef: [{
4254
+ }] }, { type: i0.Renderer2 }, { type: i0.NgZone }], propDecorators: { scrollableRowElementRef: [{
4187
4255
  type: ViewChild,
4188
4256
  args: ['scrollableRowElement']
4189
4257
  }] } });