@libs-ui/components-scroll-measure-items-direction-horizontal 0.2.356-25 → 0.2.356-26
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/direction-horizontal.directive.d.ts +9 -0
- package/esm2022/direction-horizontal.directive.mjs +69 -36
- package/fesm2022/libs-ui-components-scroll-measure-items-direction-horizontal.mjs +68 -35
- package/fesm2022/libs-ui-components-scroll-measure-items-direction-horizontal.mjs.map +1 -1
- package/package.json +1 -2
|
@@ -2,11 +2,20 @@ import { AfterViewInit } from '@angular/core';
|
|
|
2
2
|
import { IScrollMeasureItemDirectionHorizontalFunctionsControl } from './interfaces/direction-horizontal.interface';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
export declare class LibsUiScrollMeasureItemDirectionHorizontalDirective implements AfterViewInit {
|
|
5
|
+
/** Số item buffer render thêm sau viewport để tránh trắng vùng khi scroll. */
|
|
6
|
+
private readonly bufferAhead;
|
|
5
7
|
private viewPortItems;
|
|
6
8
|
private start;
|
|
7
9
|
private end;
|
|
8
10
|
private storeItems;
|
|
9
11
|
private divVirtual;
|
|
12
|
+
private resizeObserver?;
|
|
13
|
+
/** Chặn ResizeObserver / subscribe được tạo trong setTimeout sau khi directive đã destroy. */
|
|
14
|
+
private timerAfterViewInit?;
|
|
15
|
+
/** Token nhận dạng lần effect chạy hiện tại. Async đo width xong sẽ bỏ kết quả nếu token đổi. */
|
|
16
|
+
private executionToken;
|
|
17
|
+
/** Style + DOM hook của divVirtual chỉ cần khởi tạo một lần cho cả vòng đời directive. */
|
|
18
|
+
private divVirtualInitialized;
|
|
10
19
|
readonly elementScroll: import("@angular/core").InputSignal<HTMLElement>;
|
|
11
20
|
readonly functionGetWidthItem: import("@angular/core").InputSignal<(item: any) => Promise<number>>;
|
|
12
21
|
readonly items: import("@angular/core").InputSignal<any[]>;
|
|
@@ -2,16 +2,24 @@
|
|
|
2
2
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
3
3
|
import { DestroyRef, Directive, ElementRef, effect, inject, input, output, signal, untracked } from '@angular/core';
|
|
4
4
|
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
5
|
-
import { get } from '@libs-ui/utils';
|
|
6
5
|
import { fromEvent } from 'rxjs';
|
|
7
6
|
import * as i0 from "@angular/core";
|
|
8
7
|
export class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
9
8
|
// #region PROPERTY
|
|
9
|
+
/** Số item buffer render thêm sau viewport để tránh trắng vùng khi scroll. */
|
|
10
|
+
bufferAhead = 3;
|
|
10
11
|
viewPortItems = signal([]);
|
|
11
12
|
start = signal(0);
|
|
12
13
|
end = signal(0);
|
|
13
14
|
storeItems = signal([]);
|
|
14
15
|
divVirtual = document.createElement('div');
|
|
16
|
+
resizeObserver;
|
|
17
|
+
/** Chặn ResizeObserver / subscribe được tạo trong setTimeout sau khi directive đã destroy. */
|
|
18
|
+
timerAfterViewInit;
|
|
19
|
+
/** Token nhận dạng lần effect chạy hiện tại. Async đo width xong sẽ bỏ kết quả nếu token đổi. */
|
|
20
|
+
executionToken = 0;
|
|
21
|
+
/** Style + DOM hook của divVirtual chỉ cần khởi tạo một lần cho cả vòng đời directive. */
|
|
22
|
+
divVirtualInitialized = false;
|
|
15
23
|
// #region INPUT
|
|
16
24
|
elementScroll = input.required();
|
|
17
25
|
functionGetWidthItem = input.required();
|
|
@@ -26,59 +34,79 @@ export class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
|
26
34
|
destroyRef = inject(DestroyRef);
|
|
27
35
|
constructor() {
|
|
28
36
|
effect(() => {
|
|
29
|
-
const scrollEl = untracked(() => this.ScrollContainer);
|
|
30
|
-
if (!scrollEl || !untracked(() => this.functionGetWidthItem())) {
|
|
31
|
-
return;
|
|
32
|
-
}
|
|
33
37
|
const items = this.items();
|
|
38
|
+
const myToken = ++this.executionToken;
|
|
34
39
|
untracked(async () => {
|
|
40
|
+
const scrollEl = this.ScrollContainer;
|
|
41
|
+
const functionGetWidthItem = this.functionGetWidthItem();
|
|
42
|
+
if (!scrollEl || !functionGetWidthItem) {
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
// Đo width tất cả item song song — rút ngắn thời gian cho list lớn (vd 3000 cột).
|
|
46
|
+
const widths = await Promise.all(items.map((it) => functionGetWidthItem(it)));
|
|
47
|
+
// Bị huỷ vì có lượt items() mới chạy sau — bỏ kết quả cũ tránh đè state.
|
|
48
|
+
if (myToken !== this.executionToken) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
35
51
|
this.start.set(0);
|
|
36
52
|
const scrollLeft = scrollEl.scrollLeft;
|
|
37
53
|
const containerWidth = scrollEl.getBoundingClientRect().width;
|
|
38
54
|
let width = 0;
|
|
55
|
+
let endLocal = 0;
|
|
39
56
|
const newItems = [];
|
|
40
57
|
for (let i = 0; i < items.length; i++) {
|
|
41
|
-
const
|
|
42
|
-
const newItem = { ref:
|
|
43
|
-
const itemWidth = await this.functionGetWidthItem()(item);
|
|
44
|
-
newItem.itemWidth = itemWidth;
|
|
45
|
-
newItem.start = width;
|
|
58
|
+
const itemWidth = widths[i];
|
|
59
|
+
const newItem = { ref: items[i], itemWidth, start: width, end: 0 };
|
|
46
60
|
width += itemWidth;
|
|
47
61
|
newItem.end = width;
|
|
48
62
|
if (newItem.end <= containerWidth + scrollLeft) {
|
|
49
|
-
|
|
63
|
+
endLocal = i;
|
|
50
64
|
}
|
|
51
65
|
newItems.push(newItem);
|
|
52
66
|
}
|
|
67
|
+
// Set end một lần thay vì ghi mỗi vòng lặp — tránh notify consumer N lần.
|
|
68
|
+
this.end.set(endLocal);
|
|
53
69
|
this.storeItems.set(newItems);
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
.
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
70
|
+
// Khởi tạo style + DOM hook của divVirtual đúng một lần.
|
|
71
|
+
if (!this.divVirtualInitialized) {
|
|
72
|
+
this.divVirtual.style.position = 'absolute';
|
|
73
|
+
this.divVirtual.style.top = '0px';
|
|
74
|
+
this.divVirtual.style.left = '0px';
|
|
75
|
+
this.divVirtual.style.height = '1px';
|
|
76
|
+
this.divVirtual.style.visibility = 'hidden';
|
|
77
|
+
this.elementRef.nativeElement.classList.add('relative');
|
|
78
|
+
this.elementRef.nativeElement.classList.remove('w-full');
|
|
79
|
+
this.elementRef.nativeElement.style.paddingLeft = '0px';
|
|
80
|
+
scrollEl.append(this.divVirtual);
|
|
81
|
+
this.divVirtualInitialized = true;
|
|
82
|
+
this.outDivVirtual.emit(this.divVirtual);
|
|
83
|
+
}
|
|
84
|
+
// Width tổng thể đổi theo dataset → cần update mỗi lượt.
|
|
62
85
|
this.divVirtual.style.width = `${width}px`;
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
this.
|
|
67
|
-
this.elementRef.nativeElement.style.paddingLeft = '0px';
|
|
68
|
-
scrollEl.append(this.divVirtual);
|
|
69
|
-
this.outDivVirtual.emit(this.divVirtual);
|
|
86
|
+
// containerWidth lúc đầu có thể = 0 (DOM chưa layout), khiến end giữ
|
|
87
|
+
// nguyên 0 và chỉ render 3 cột đầu. Re-đo bằng handlerScroll sau khi vòng
|
|
88
|
+
// lặp await xong (layout thường đã hoàn tất) để end ra đúng giá trị.
|
|
89
|
+
this.handlerScroll();
|
|
70
90
|
});
|
|
71
91
|
});
|
|
72
92
|
this.destroyRef.onDestroy(() => {
|
|
93
|
+
clearTimeout(this.timerAfterViewInit);
|
|
73
94
|
this.divVirtual.remove();
|
|
95
|
+
this.resizeObserver?.disconnect();
|
|
74
96
|
});
|
|
75
97
|
}
|
|
76
98
|
ngAfterViewInit() {
|
|
77
|
-
setTimeout(() => {
|
|
99
|
+
this.timerAfterViewInit = setTimeout(() => {
|
|
78
100
|
fromEvent(this.ScrollContainer, 'scroll')
|
|
79
101
|
.pipe(takeUntilDestroyed(this.destroyRef))
|
|
80
102
|
.subscribe(() => this.handlerScroll());
|
|
81
103
|
this.outFunctionControl.emit(this.FunctionsControl);
|
|
104
|
+
// Re-tính viewport sau khi container có kích thước thật.
|
|
105
|
+
// Lần đầu render, getBoundingClientRect().width có thể = 0 nên end giữ giá trị 0,
|
|
106
|
+
// chỉ hiển thị 3 cột đầu (do slice(start, end + bufferAhead)). ResizeObserver fire ngay khi
|
|
107
|
+
// observe và mỗi lần container đổi size sẽ kéo end về đúng giá trị.
|
|
108
|
+
this.resizeObserver = new ResizeObserver(() => this.handlerScroll());
|
|
109
|
+
this.resizeObserver.observe(this.ScrollContainer);
|
|
82
110
|
});
|
|
83
111
|
}
|
|
84
112
|
// #region PUBLIC API
|
|
@@ -104,28 +132,29 @@ export class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
|
104
132
|
const scrollLeft = scrollEl.scrollLeft;
|
|
105
133
|
const containerWidth = scrollEl.getBoundingClientRect().width;
|
|
106
134
|
let start = this.start();
|
|
135
|
+
let endLocal = this.end();
|
|
107
136
|
let isSetStartPositioned = false;
|
|
108
137
|
for (let i = 0; i < store.length; i++) {
|
|
109
138
|
const item = store[i];
|
|
110
139
|
if (item.end > containerWidth + scrollLeft) {
|
|
111
140
|
break;
|
|
112
141
|
}
|
|
113
|
-
|
|
142
|
+
endLocal = i;
|
|
114
143
|
if (item.end >= scrollLeft && !isSetStartPositioned) {
|
|
115
144
|
start = i;
|
|
116
145
|
isSetStartPositioned = true;
|
|
117
146
|
}
|
|
118
147
|
}
|
|
148
|
+
if (endLocal !== this.end()) {
|
|
149
|
+
this.end.set(endLocal);
|
|
150
|
+
}
|
|
119
151
|
if (start !== this.start()) {
|
|
120
152
|
this.start.set(start);
|
|
121
153
|
const paddingLeft = store[start].start;
|
|
122
154
|
this.elementRef.nativeElement.style.paddingLeft = `${paddingLeft}px`;
|
|
123
155
|
this.outPaddingLeft.emit(paddingLeft);
|
|
124
156
|
}
|
|
125
|
-
|
|
126
|
-
this.start.set(0);
|
|
127
|
-
}
|
|
128
|
-
this.viewPortItems.set(store.slice(this.start(), this.end() + 3).map((item) => item.ref));
|
|
157
|
+
this.viewPortItems.set(store.slice(this.start(), this.end() + this.bufferAhead).map((item) => item.ref));
|
|
129
158
|
this.outViewPortItem.emit(this.viewPortItems());
|
|
130
159
|
}
|
|
131
160
|
async scrollInto(itemScroll) {
|
|
@@ -146,10 +175,14 @@ export class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
|
146
175
|
this.ScrollContainer.scrollLeft = itemOfIndex.start;
|
|
147
176
|
}
|
|
148
177
|
async reCalculatorViewPort() {
|
|
149
|
-
let width = 0;
|
|
150
178
|
const items = this.storeItems();
|
|
151
|
-
|
|
152
|
-
|
|
179
|
+
const fn = this.functionGetWidthItem();
|
|
180
|
+
// Đo song song để rút ngắn thời gian thay vì await tuần tự.
|
|
181
|
+
const widths = await Promise.all(items.map((it) => fn(it.ref)));
|
|
182
|
+
let width = 0;
|
|
183
|
+
for (let i = 0; i < items.length; i++) {
|
|
184
|
+
const item = items[i];
|
|
185
|
+
const itemWidth = widths[i];
|
|
153
186
|
item.itemWidth = itemWidth;
|
|
154
187
|
item.start = width;
|
|
155
188
|
width += itemWidth;
|
|
@@ -170,4 +203,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
170
203
|
standalone: true,
|
|
171
204
|
}]
|
|
172
205
|
}], ctorParameters: () => [] });
|
|
173
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"direction-horizontal.directive.js","sourceRoot":"","sources":["../../../../../../libs-ui/components/scroll-measure-items/direction-horizontal/src/direction-horizontal.directive.ts"],"names":[],"mappings":"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","sourcesContent":["// CONVENTION-EXCEPT: any — Generic directive works with unknown item types from consumer\n/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { AfterViewInit, DestroyRef, Directive, ElementRef, effect, inject, input, output, signal, untracked } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { get } from '@libs-ui/utils';\nimport { fromEvent } from 'rxjs';\nimport { IScrollMeasureItemDirectionHorizontalFunctionsControl, IScrollMeasureStoreItemConvert } from './interfaces/direction-horizontal.interface';\n@Directive({\n  // eslint-disable-next-line @angular-eslint/directive-selector\n  selector: '[LibsUiScrollMeasureItemDirectionHorizontalDirective]',\n  standalone: true,\n})\nexport class LibsUiScrollMeasureItemDirectionHorizontalDirective implements AfterViewInit {\n  // #region PROPERTY\n  private viewPortItems = signal<Array<any>>([]);\n  private start = signal<number>(0);\n  private end = signal<number>(0);\n  private storeItems = signal<Array<IScrollMeasureStoreItemConvert>>([]);\n  private divVirtual = document.createElement('div');\n\n  // #region INPUT\n  readonly elementScroll = input.required<HTMLElement>();\n  readonly functionGetWidthItem = input.required<(item: any) => Promise<number>>();\n  readonly items = input.required<Array<any>>();\n\n  // #region OUTPUT\n  readonly outViewPortItem = output<Array<any>>();\n  readonly outFunctionControl = output<IScrollMeasureItemDirectionHorizontalFunctionsControl>();\n  readonly outDivVirtual = output<HTMLDivElement>();\n  readonly outPaddingLeft = output<number>();\n\n  // #region INJECT\n  private readonly elementRef = inject(ElementRef);\n  private readonly destroyRef = inject(DestroyRef);\n\n  constructor() {\n    effect(() => {\n      const scrollEl = untracked(() => this.ScrollContainer);\n      if (!scrollEl || !untracked(() => this.functionGetWidthItem())) {\n        return;\n      }\n      const items = this.items();\n      untracked(async () => {\n        this.start.set(0);\n        const scrollLeft = scrollEl.scrollLeft;\n        const containerWidth = scrollEl.getBoundingClientRect().width;\n        let width = 0;\n        const newItems: Array<IScrollMeasureStoreItemConvert> = [];\n\n        for (let i = 0; i < items.length; i++) {\n          const item = items[i];\n          const newItem: IScrollMeasureStoreItemConvert = { ref: item, itemWidth: 0, start: 0, end: 0 };\n          const itemWidth = await this.functionGetWidthItem()(item);\n          newItem.itemWidth = itemWidth;\n          newItem.start = width;\n          width += itemWidth;\n          newItem.end = width;\n          if (newItem.end <= containerWidth + scrollLeft) {\n            this.end.set(i);\n          }\n          newItems.push(newItem);\n        }\n\n        this.storeItems.set(newItems);\n        this.viewPortItems.set(\n          this.storeItems()\n            .slice(this.start(), this.end() + 3)\n            .map((item) => item.ref)\n        );\n        this.outViewPortItem.emit(this.viewPortItems());\n        this.outPaddingLeft.emit(get(this.storeItems(), `[${this.start()}].start`, 0));\n\n        this.divVirtual.style.position = 'absolute';\n        this.divVirtual.style.top = '0px';\n        this.divVirtual.style.left = '0px';\n        this.divVirtual.style.width = `${width}px`;\n        this.divVirtual.style.height = '1px';\n        this.divVirtual.style.visibility = 'hidden';\n        this.elementRef.nativeElement.classList.add('relative');\n        this.elementRef.nativeElement.classList.remove('w-full');\n        this.elementRef.nativeElement.style.paddingLeft = '0px';\n        scrollEl.append(this.divVirtual);\n        this.outDivVirtual.emit(this.divVirtual);\n      });\n    });\n\n    this.destroyRef.onDestroy(() => {\n      this.divVirtual.remove();\n    });\n  }\n\n  ngAfterViewInit() {\n    setTimeout(() => {\n      fromEvent<Event>(this.ScrollContainer, 'scroll')\n        .pipe(takeUntilDestroyed(this.destroyRef))\n        .subscribe(() => this.handlerScroll());\n      this.outFunctionControl.emit(this.FunctionsControl);\n    });\n  }\n\n  // #region PUBLIC API\n  public get FunctionsControl(): IScrollMeasureItemDirectionHorizontalFunctionsControl {\n    return {\n      scrollInto: this.scrollInto.bind(this),\n      scrollToPosition: this.scrollToPosition.bind(this),\n      scrollToIndex: this.scrollToIndex.bind(this),\n      reCalculatorViewPort: this.reCalculatorViewPort.bind(this),\n      getViewPortItems: () => this.viewPortItems(),\n    };\n  }\n\n  // #region PRIVATE METHODS\n  private get ScrollContainer(): HTMLElement {\n    return this.elementScroll() || this.elementRef.nativeElement;\n  }\n\n  private handlerScroll() {\n    const store = this.storeItems();\n    if (!store?.length) {\n      return;\n    }\n    const scrollEl = this.ScrollContainer;\n    const scrollLeft = scrollEl.scrollLeft;\n    const containerWidth = scrollEl.getBoundingClientRect().width;\n    let start = this.start();\n    let isSetStartPositioned = false;\n\n    for (let i = 0; i < store.length; i++) {\n      const item = store[i];\n      if (item.end > containerWidth + scrollLeft) {\n        break;\n      }\n      this.end.set(i);\n      if (item.end >= scrollLeft && !isSetStartPositioned) {\n        start = i;\n        isSetStartPositioned = true;\n      }\n    }\n\n    if (start !== this.start()) {\n      this.start.set(start);\n      const paddingLeft = store[start].start;\n      this.elementRef.nativeElement.style.paddingLeft = `${paddingLeft}px`;\n      this.outPaddingLeft.emit(paddingLeft);\n    }\n\n    if (this.start() < 0) {\n      this.start.set(0);\n    }\n    this.viewPortItems.set(store.slice(this.start(), this.end() + 3).map((item) => item.ref));\n    this.outViewPortItem.emit(this.viewPortItems());\n  }\n\n  private async scrollInto(itemScroll: any) {\n    const itemFound = this.storeItems().find((item) => item.ref === itemScroll);\n    if (!itemFound) {\n      return;\n    }\n    this.ScrollContainer.scrollLeft = itemFound.start;\n  }\n\n  private async scrollToPosition(position: number) {\n    this.ScrollContainer.scrollLeft = position;\n  }\n\n  private async scrollToIndex(index: number) {\n    const itemOfIndex = this.storeItems()[index];\n    if (!itemOfIndex) {\n      return;\n    }\n    this.ScrollContainer.scrollLeft = itemOfIndex.start;\n  }\n\n  private async reCalculatorViewPort() {\n    let width = 0;\n    const items = this.storeItems();\n\n    for (const item of items) {\n      const itemWidth = await this.functionGetWidthItem()(item.ref);\n      item.itemWidth = itemWidth;\n      item.start = width;\n      width += itemWidth;\n      item.end = width;\n    }\n\n    this.storeItems.set([...items]);\n    this.divVirtual.style.width = `${width}px`;\n    this.handlerScroll();\n  }\n}\n"]}
|
|
206
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"direction-horizontal.directive.js","sourceRoot":"","sources":["../../../../../../libs-ui/components/scroll-measure-items/direction-horizontal/src/direction-horizontal.directive.ts"],"names":[],"mappings":"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","sourcesContent":["// CONVENTION-EXCEPT: any — Generic directive works with unknown item types from consumer\n/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { AfterViewInit, DestroyRef, Directive, ElementRef, effect, inject, input, output, signal, untracked } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { fromEvent } from 'rxjs';\nimport { IScrollMeasureItemDirectionHorizontalFunctionsControl, IScrollMeasureStoreItemConvert } from './interfaces/direction-horizontal.interface';\n@Directive({\n  // eslint-disable-next-line @angular-eslint/directive-selector\n  selector: '[LibsUiScrollMeasureItemDirectionHorizontalDirective]',\n  standalone: true,\n})\nexport class LibsUiScrollMeasureItemDirectionHorizontalDirective implements AfterViewInit {\n  // #region PROPERTY\n  /** Số item buffer render thêm sau viewport để tránh trắng vùng khi scroll. */\n  private readonly bufferAhead = 3;\n\n  private viewPortItems = signal<Array<any>>([]);\n  private start = signal<number>(0);\n  private end = signal<number>(0);\n  private storeItems = signal<Array<IScrollMeasureStoreItemConvert>>([]);\n  private divVirtual = document.createElement('div');\n  private resizeObserver?: ResizeObserver;\n  /** Chặn ResizeObserver / subscribe được tạo trong setTimeout sau khi directive đã destroy. */\n  private timerAfterViewInit?: number;\n  /** Token nhận dạng lần effect chạy hiện tại. Async đo width xong sẽ bỏ kết quả nếu token đổi. */\n  private executionToken = 0;\n  /** Style + DOM hook của divVirtual chỉ cần khởi tạo một lần cho cả vòng đời directive. */\n  private divVirtualInitialized = false;\n\n  // #region INPUT\n  readonly elementScroll = input.required<HTMLElement>();\n  readonly functionGetWidthItem = input.required<(item: any) => Promise<number>>();\n  readonly items = input.required<Array<any>>();\n\n  // #region OUTPUT\n  readonly outViewPortItem = output<Array<any>>();\n  readonly outFunctionControl = output<IScrollMeasureItemDirectionHorizontalFunctionsControl>();\n  readonly outDivVirtual = output<HTMLDivElement>();\n  readonly outPaddingLeft = output<number>();\n\n  // #region INJECT\n  private readonly elementRef = inject(ElementRef);\n  private readonly destroyRef = inject(DestroyRef);\n\n  constructor() {\n    effect(() => {\n      const items = this.items();\n      const myToken = ++this.executionToken;\n      untracked(async () => {\n        const scrollEl = this.ScrollContainer;\n        const functionGetWidthItem = this.functionGetWidthItem();\n        if (!scrollEl || !functionGetWidthItem) {\n          return;\n        }\n\n        // Đo width tất cả item song song — rút ngắn thời gian cho list lớn (vd 3000 cột).\n        const widths = await Promise.all(items.map((it) => functionGetWidthItem(it)));\n        // Bị huỷ vì có lượt items() mới chạy sau — bỏ kết quả cũ tránh đè state.\n        if (myToken !== this.executionToken) {\n          return;\n        }\n\n        this.start.set(0);\n        const scrollLeft = scrollEl.scrollLeft;\n        const containerWidth = scrollEl.getBoundingClientRect().width;\n        let width = 0;\n        let endLocal = 0;\n        const newItems: Array<IScrollMeasureStoreItemConvert> = [];\n\n        for (let i = 0; i < items.length; i++) {\n          const itemWidth = widths[i];\n          const newItem: IScrollMeasureStoreItemConvert = { ref: items[i], itemWidth, start: width, end: 0 };\n          width += itemWidth;\n          newItem.end = width;\n          if (newItem.end <= containerWidth + scrollLeft) {\n            endLocal = i;\n          }\n          newItems.push(newItem);\n        }\n        // Set end một lần thay vì ghi mỗi vòng lặp — tránh notify consumer N lần.\n        this.end.set(endLocal);\n        this.storeItems.set(newItems);\n\n        // Khởi tạo style + DOM hook của divVirtual đúng một lần.\n        if (!this.divVirtualInitialized) {\n          this.divVirtual.style.position = 'absolute';\n          this.divVirtual.style.top = '0px';\n          this.divVirtual.style.left = '0px';\n          this.divVirtual.style.height = '1px';\n          this.divVirtual.style.visibility = 'hidden';\n          this.elementRef.nativeElement.classList.add('relative');\n          this.elementRef.nativeElement.classList.remove('w-full');\n          this.elementRef.nativeElement.style.paddingLeft = '0px';\n          scrollEl.append(this.divVirtual);\n          this.divVirtualInitialized = true;\n          this.outDivVirtual.emit(this.divVirtual);\n        }\n        // Width tổng thể đổi theo dataset → cần update mỗi lượt.\n        this.divVirtual.style.width = `${width}px`;\n\n        // containerWidth lúc đầu có thể = 0 (DOM chưa layout), khiến end giữ\n        // nguyên 0 và chỉ render 3 cột đầu. Re-đo bằng handlerScroll sau khi vòng\n        // lặp await xong (layout thường đã hoàn tất) để end ra đúng giá trị.\n        this.handlerScroll();\n      });\n    });\n\n    this.destroyRef.onDestroy(() => {\n      clearTimeout(this.timerAfterViewInit);\n      this.divVirtual.remove();\n      this.resizeObserver?.disconnect();\n    });\n  }\n\n  ngAfterViewInit() {\n    this.timerAfterViewInit = setTimeout(() => {\n      fromEvent<Event>(this.ScrollContainer, 'scroll')\n        .pipe(takeUntilDestroyed(this.destroyRef))\n        .subscribe(() => this.handlerScroll());\n      this.outFunctionControl.emit(this.FunctionsControl);\n\n      // Re-tính viewport sau khi container có kích thước thật.\n      // Lần đầu render, getBoundingClientRect().width có thể = 0 nên end giữ giá trị 0,\n      // chỉ hiển thị 3 cột đầu (do slice(start, end + bufferAhead)). ResizeObserver fire ngay khi\n      // observe và mỗi lần container đổi size sẽ kéo end về đúng giá trị.\n      this.resizeObserver = new ResizeObserver(() => this.handlerScroll());\n      this.resizeObserver.observe(this.ScrollContainer);\n    });\n  }\n\n  // #region PUBLIC API\n  public get FunctionsControl(): IScrollMeasureItemDirectionHorizontalFunctionsControl {\n    return {\n      scrollInto: this.scrollInto.bind(this),\n      scrollToPosition: this.scrollToPosition.bind(this),\n      scrollToIndex: this.scrollToIndex.bind(this),\n      reCalculatorViewPort: this.reCalculatorViewPort.bind(this),\n      getViewPortItems: () => this.viewPortItems(),\n    };\n  }\n\n  // #region PRIVATE METHODS\n  private get ScrollContainer(): HTMLElement {\n    return this.elementScroll() || this.elementRef.nativeElement;\n  }\n\n  private handlerScroll() {\n    const store = this.storeItems();\n    if (!store?.length) {\n      return;\n    }\n    const scrollEl = this.ScrollContainer;\n    const scrollLeft = scrollEl.scrollLeft;\n    const containerWidth = scrollEl.getBoundingClientRect().width;\n    let start = this.start();\n    let endLocal = this.end();\n    let isSetStartPositioned = false;\n\n    for (let i = 0; i < store.length; i++) {\n      const item = store[i];\n      if (item.end > containerWidth + scrollLeft) {\n        break;\n      }\n      endLocal = i;\n      if (item.end >= scrollLeft && !isSetStartPositioned) {\n        start = i;\n        isSetStartPositioned = true;\n      }\n    }\n    if (endLocal !== this.end()) {\n      this.end.set(endLocal);\n    }\n\n    if (start !== this.start()) {\n      this.start.set(start);\n      const paddingLeft = store[start].start;\n      this.elementRef.nativeElement.style.paddingLeft = `${paddingLeft}px`;\n      this.outPaddingLeft.emit(paddingLeft);\n    }\n    this.viewPortItems.set(store.slice(this.start(), this.end() + this.bufferAhead).map((item) => item.ref));\n    this.outViewPortItem.emit(this.viewPortItems());\n  }\n\n  private async scrollInto(itemScroll: any) {\n    const itemFound = this.storeItems().find((item) => item.ref === itemScroll);\n    if (!itemFound) {\n      return;\n    }\n    this.ScrollContainer.scrollLeft = itemFound.start;\n  }\n\n  private async scrollToPosition(position: number) {\n    this.ScrollContainer.scrollLeft = position;\n  }\n\n  private async scrollToIndex(index: number) {\n    const itemOfIndex = this.storeItems()[index];\n    if (!itemOfIndex) {\n      return;\n    }\n    this.ScrollContainer.scrollLeft = itemOfIndex.start;\n  }\n\n  private async reCalculatorViewPort() {\n    const items = this.storeItems();\n    const fn = this.functionGetWidthItem();\n    // Đo song song để rút ngắn thời gian thay vì await tuần tự.\n    const widths = await Promise.all(items.map((it) => fn(it.ref)));\n\n    let width = 0;\n    for (let i = 0; i < items.length; i++) {\n      const item = items[i];\n      const itemWidth = widths[i];\n      item.itemWidth = itemWidth;\n      item.start = width;\n      width += itemWidth;\n      item.end = width;\n    }\n\n    this.storeItems.set([...items]);\n    this.divVirtual.style.width = `${width}px`;\n    this.handlerScroll();\n  }\n}\n"]}
|
|
@@ -1,18 +1,26 @@
|
|
|
1
1
|
import * as i0 from '@angular/core';
|
|
2
2
|
import { signal, input, output, inject, ElementRef, DestroyRef, effect, untracked, Directive } from '@angular/core';
|
|
3
3
|
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
4
|
-
import { get } from '@libs-ui/utils';
|
|
5
4
|
import { fromEvent } from 'rxjs';
|
|
6
5
|
|
|
7
6
|
// CONVENTION-EXCEPT: any — Generic directive works with unknown item types from consumer
|
|
8
7
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
9
8
|
class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
10
9
|
// #region PROPERTY
|
|
10
|
+
/** Số item buffer render thêm sau viewport để tránh trắng vùng khi scroll. */
|
|
11
|
+
bufferAhead = 3;
|
|
11
12
|
viewPortItems = signal([]);
|
|
12
13
|
start = signal(0);
|
|
13
14
|
end = signal(0);
|
|
14
15
|
storeItems = signal([]);
|
|
15
16
|
divVirtual = document.createElement('div');
|
|
17
|
+
resizeObserver;
|
|
18
|
+
/** Chặn ResizeObserver / subscribe được tạo trong setTimeout sau khi directive đã destroy. */
|
|
19
|
+
timerAfterViewInit;
|
|
20
|
+
/** Token nhận dạng lần effect chạy hiện tại. Async đo width xong sẽ bỏ kết quả nếu token đổi. */
|
|
21
|
+
executionToken = 0;
|
|
22
|
+
/** Style + DOM hook của divVirtual chỉ cần khởi tạo một lần cho cả vòng đời directive. */
|
|
23
|
+
divVirtualInitialized = false;
|
|
16
24
|
// #region INPUT
|
|
17
25
|
elementScroll = input.required();
|
|
18
26
|
functionGetWidthItem = input.required();
|
|
@@ -27,59 +35,79 @@ class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
|
27
35
|
destroyRef = inject(DestroyRef);
|
|
28
36
|
constructor() {
|
|
29
37
|
effect(() => {
|
|
30
|
-
const scrollEl = untracked(() => this.ScrollContainer);
|
|
31
|
-
if (!scrollEl || !untracked(() => this.functionGetWidthItem())) {
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
38
|
const items = this.items();
|
|
39
|
+
const myToken = ++this.executionToken;
|
|
35
40
|
untracked(async () => {
|
|
41
|
+
const scrollEl = this.ScrollContainer;
|
|
42
|
+
const functionGetWidthItem = this.functionGetWidthItem();
|
|
43
|
+
if (!scrollEl || !functionGetWidthItem) {
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
// Đo width tất cả item song song — rút ngắn thời gian cho list lớn (vd 3000 cột).
|
|
47
|
+
const widths = await Promise.all(items.map((it) => functionGetWidthItem(it)));
|
|
48
|
+
// Bị huỷ vì có lượt items() mới chạy sau — bỏ kết quả cũ tránh đè state.
|
|
49
|
+
if (myToken !== this.executionToken) {
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
36
52
|
this.start.set(0);
|
|
37
53
|
const scrollLeft = scrollEl.scrollLeft;
|
|
38
54
|
const containerWidth = scrollEl.getBoundingClientRect().width;
|
|
39
55
|
let width = 0;
|
|
56
|
+
let endLocal = 0;
|
|
40
57
|
const newItems = [];
|
|
41
58
|
for (let i = 0; i < items.length; i++) {
|
|
42
|
-
const
|
|
43
|
-
const newItem = { ref:
|
|
44
|
-
const itemWidth = await this.functionGetWidthItem()(item);
|
|
45
|
-
newItem.itemWidth = itemWidth;
|
|
46
|
-
newItem.start = width;
|
|
59
|
+
const itemWidth = widths[i];
|
|
60
|
+
const newItem = { ref: items[i], itemWidth, start: width, end: 0 };
|
|
47
61
|
width += itemWidth;
|
|
48
62
|
newItem.end = width;
|
|
49
63
|
if (newItem.end <= containerWidth + scrollLeft) {
|
|
50
|
-
|
|
64
|
+
endLocal = i;
|
|
51
65
|
}
|
|
52
66
|
newItems.push(newItem);
|
|
53
67
|
}
|
|
68
|
+
// Set end một lần thay vì ghi mỗi vòng lặp — tránh notify consumer N lần.
|
|
69
|
+
this.end.set(endLocal);
|
|
54
70
|
this.storeItems.set(newItems);
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
.
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
71
|
+
// Khởi tạo style + DOM hook của divVirtual đúng một lần.
|
|
72
|
+
if (!this.divVirtualInitialized) {
|
|
73
|
+
this.divVirtual.style.position = 'absolute';
|
|
74
|
+
this.divVirtual.style.top = '0px';
|
|
75
|
+
this.divVirtual.style.left = '0px';
|
|
76
|
+
this.divVirtual.style.height = '1px';
|
|
77
|
+
this.divVirtual.style.visibility = 'hidden';
|
|
78
|
+
this.elementRef.nativeElement.classList.add('relative');
|
|
79
|
+
this.elementRef.nativeElement.classList.remove('w-full');
|
|
80
|
+
this.elementRef.nativeElement.style.paddingLeft = '0px';
|
|
81
|
+
scrollEl.append(this.divVirtual);
|
|
82
|
+
this.divVirtualInitialized = true;
|
|
83
|
+
this.outDivVirtual.emit(this.divVirtual);
|
|
84
|
+
}
|
|
85
|
+
// Width tổng thể đổi theo dataset → cần update mỗi lượt.
|
|
63
86
|
this.divVirtual.style.width = `${width}px`;
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
this.
|
|
68
|
-
this.elementRef.nativeElement.style.paddingLeft = '0px';
|
|
69
|
-
scrollEl.append(this.divVirtual);
|
|
70
|
-
this.outDivVirtual.emit(this.divVirtual);
|
|
87
|
+
// containerWidth lúc đầu có thể = 0 (DOM chưa layout), khiến end giữ
|
|
88
|
+
// nguyên 0 và chỉ render 3 cột đầu. Re-đo bằng handlerScroll sau khi vòng
|
|
89
|
+
// lặp await xong (layout thường đã hoàn tất) để end ra đúng giá trị.
|
|
90
|
+
this.handlerScroll();
|
|
71
91
|
});
|
|
72
92
|
});
|
|
73
93
|
this.destroyRef.onDestroy(() => {
|
|
94
|
+
clearTimeout(this.timerAfterViewInit);
|
|
74
95
|
this.divVirtual.remove();
|
|
96
|
+
this.resizeObserver?.disconnect();
|
|
75
97
|
});
|
|
76
98
|
}
|
|
77
99
|
ngAfterViewInit() {
|
|
78
|
-
setTimeout(() => {
|
|
100
|
+
this.timerAfterViewInit = setTimeout(() => {
|
|
79
101
|
fromEvent(this.ScrollContainer, 'scroll')
|
|
80
102
|
.pipe(takeUntilDestroyed(this.destroyRef))
|
|
81
103
|
.subscribe(() => this.handlerScroll());
|
|
82
104
|
this.outFunctionControl.emit(this.FunctionsControl);
|
|
105
|
+
// Re-tính viewport sau khi container có kích thước thật.
|
|
106
|
+
// Lần đầu render, getBoundingClientRect().width có thể = 0 nên end giữ giá trị 0,
|
|
107
|
+
// chỉ hiển thị 3 cột đầu (do slice(start, end + bufferAhead)). ResizeObserver fire ngay khi
|
|
108
|
+
// observe và mỗi lần container đổi size sẽ kéo end về đúng giá trị.
|
|
109
|
+
this.resizeObserver = new ResizeObserver(() => this.handlerScroll());
|
|
110
|
+
this.resizeObserver.observe(this.ScrollContainer);
|
|
83
111
|
});
|
|
84
112
|
}
|
|
85
113
|
// #region PUBLIC API
|
|
@@ -105,28 +133,29 @@ class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
|
105
133
|
const scrollLeft = scrollEl.scrollLeft;
|
|
106
134
|
const containerWidth = scrollEl.getBoundingClientRect().width;
|
|
107
135
|
let start = this.start();
|
|
136
|
+
let endLocal = this.end();
|
|
108
137
|
let isSetStartPositioned = false;
|
|
109
138
|
for (let i = 0; i < store.length; i++) {
|
|
110
139
|
const item = store[i];
|
|
111
140
|
if (item.end > containerWidth + scrollLeft) {
|
|
112
141
|
break;
|
|
113
142
|
}
|
|
114
|
-
|
|
143
|
+
endLocal = i;
|
|
115
144
|
if (item.end >= scrollLeft && !isSetStartPositioned) {
|
|
116
145
|
start = i;
|
|
117
146
|
isSetStartPositioned = true;
|
|
118
147
|
}
|
|
119
148
|
}
|
|
149
|
+
if (endLocal !== this.end()) {
|
|
150
|
+
this.end.set(endLocal);
|
|
151
|
+
}
|
|
120
152
|
if (start !== this.start()) {
|
|
121
153
|
this.start.set(start);
|
|
122
154
|
const paddingLeft = store[start].start;
|
|
123
155
|
this.elementRef.nativeElement.style.paddingLeft = `${paddingLeft}px`;
|
|
124
156
|
this.outPaddingLeft.emit(paddingLeft);
|
|
125
157
|
}
|
|
126
|
-
|
|
127
|
-
this.start.set(0);
|
|
128
|
-
}
|
|
129
|
-
this.viewPortItems.set(store.slice(this.start(), this.end() + 3).map((item) => item.ref));
|
|
158
|
+
this.viewPortItems.set(store.slice(this.start(), this.end() + this.bufferAhead).map((item) => item.ref));
|
|
130
159
|
this.outViewPortItem.emit(this.viewPortItems());
|
|
131
160
|
}
|
|
132
161
|
async scrollInto(itemScroll) {
|
|
@@ -147,10 +176,14 @@ class LibsUiScrollMeasureItemDirectionHorizontalDirective {
|
|
|
147
176
|
this.ScrollContainer.scrollLeft = itemOfIndex.start;
|
|
148
177
|
}
|
|
149
178
|
async reCalculatorViewPort() {
|
|
150
|
-
let width = 0;
|
|
151
179
|
const items = this.storeItems();
|
|
152
|
-
|
|
153
|
-
|
|
180
|
+
const fn = this.functionGetWidthItem();
|
|
181
|
+
// Đo song song để rút ngắn thời gian thay vì await tuần tự.
|
|
182
|
+
const widths = await Promise.all(items.map((it) => fn(it.ref)));
|
|
183
|
+
let width = 0;
|
|
184
|
+
for (let i = 0; i < items.length; i++) {
|
|
185
|
+
const item = items[i];
|
|
186
|
+
const itemWidth = widths[i];
|
|
154
187
|
item.itemWidth = itemWidth;
|
|
155
188
|
item.start = width;
|
|
156
189
|
width += itemWidth;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"libs-ui-components-scroll-measure-items-direction-horizontal.mjs","sources":["../../../../../../libs-ui/components/scroll-measure-items/direction-horizontal/src/direction-horizontal.directive.ts","../../../../../../libs-ui/components/scroll-measure-items/direction-horizontal/src/libs-ui-components-scroll-measure-items-direction-horizontal.ts"],"sourcesContent":["// CONVENTION-EXCEPT: any — Generic directive works with unknown item types from consumer\n/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { AfterViewInit, DestroyRef, Directive, ElementRef, effect, inject, input, output, signal, untracked } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { get } from '@libs-ui/utils';\nimport { fromEvent } from 'rxjs';\nimport { IScrollMeasureItemDirectionHorizontalFunctionsControl, IScrollMeasureStoreItemConvert } from './interfaces/direction-horizontal.interface';\n@Directive({\n // eslint-disable-next-line @angular-eslint/directive-selector\n selector: '[LibsUiScrollMeasureItemDirectionHorizontalDirective]',\n standalone: true,\n})\nexport class LibsUiScrollMeasureItemDirectionHorizontalDirective implements AfterViewInit {\n // #region PROPERTY\n private viewPortItems = signal<Array<any>>([]);\n private start = signal<number>(0);\n private end = signal<number>(0);\n private storeItems = signal<Array<IScrollMeasureStoreItemConvert>>([]);\n private divVirtual = document.createElement('div');\n\n // #region INPUT\n readonly elementScroll = input.required<HTMLElement>();\n readonly functionGetWidthItem = input.required<(item: any) => Promise<number>>();\n readonly items = input.required<Array<any>>();\n\n // #region OUTPUT\n readonly outViewPortItem = output<Array<any>>();\n readonly outFunctionControl = output<IScrollMeasureItemDirectionHorizontalFunctionsControl>();\n readonly outDivVirtual = output<HTMLDivElement>();\n readonly outPaddingLeft = output<number>();\n\n // #region INJECT\n private readonly elementRef = inject(ElementRef);\n private readonly destroyRef = inject(DestroyRef);\n\n constructor() {\n effect(() => {\n const scrollEl = untracked(() => this.ScrollContainer);\n if (!scrollEl || !untracked(() => this.functionGetWidthItem())) {\n return;\n }\n const items = this.items();\n untracked(async () => {\n this.start.set(0);\n const scrollLeft = scrollEl.scrollLeft;\n const containerWidth = scrollEl.getBoundingClientRect().width;\n let width = 0;\n const newItems: Array<IScrollMeasureStoreItemConvert> = [];\n\n for (let i = 0; i < items.length; i++) {\n const item = items[i];\n const newItem: IScrollMeasureStoreItemConvert = { ref: item, itemWidth: 0, start: 0, end: 0 };\n const itemWidth = await this.functionGetWidthItem()(item);\n newItem.itemWidth = itemWidth;\n newItem.start = width;\n width += itemWidth;\n newItem.end = width;\n if (newItem.end <= containerWidth + scrollLeft) {\n this.end.set(i);\n }\n newItems.push(newItem);\n }\n\n this.storeItems.set(newItems);\n this.viewPortItems.set(\n this.storeItems()\n .slice(this.start(), this.end() + 3)\n .map((item) => item.ref)\n );\n this.outViewPortItem.emit(this.viewPortItems());\n this.outPaddingLeft.emit(get(this.storeItems(), `[${this.start()}].start`, 0));\n\n this.divVirtual.style.position = 'absolute';\n this.divVirtual.style.top = '0px';\n this.divVirtual.style.left = '0px';\n this.divVirtual.style.width = `${width}px`;\n this.divVirtual.style.height = '1px';\n this.divVirtual.style.visibility = 'hidden';\n this.elementRef.nativeElement.classList.add('relative');\n this.elementRef.nativeElement.classList.remove('w-full');\n this.elementRef.nativeElement.style.paddingLeft = '0px';\n scrollEl.append(this.divVirtual);\n this.outDivVirtual.emit(this.divVirtual);\n });\n });\n\n this.destroyRef.onDestroy(() => {\n this.divVirtual.remove();\n });\n }\n\n ngAfterViewInit() {\n setTimeout(() => {\n fromEvent<Event>(this.ScrollContainer, 'scroll')\n .pipe(takeUntilDestroyed(this.destroyRef))\n .subscribe(() => this.handlerScroll());\n this.outFunctionControl.emit(this.FunctionsControl);\n });\n }\n\n // #region PUBLIC API\n public get FunctionsControl(): IScrollMeasureItemDirectionHorizontalFunctionsControl {\n return {\n scrollInto: this.scrollInto.bind(this),\n scrollToPosition: this.scrollToPosition.bind(this),\n scrollToIndex: this.scrollToIndex.bind(this),\n reCalculatorViewPort: this.reCalculatorViewPort.bind(this),\n getViewPortItems: () => this.viewPortItems(),\n };\n }\n\n // #region PRIVATE METHODS\n private get ScrollContainer(): HTMLElement {\n return this.elementScroll() || this.elementRef.nativeElement;\n }\n\n private handlerScroll() {\n const store = this.storeItems();\n if (!store?.length) {\n return;\n }\n const scrollEl = this.ScrollContainer;\n const scrollLeft = scrollEl.scrollLeft;\n const containerWidth = scrollEl.getBoundingClientRect().width;\n let start = this.start();\n let isSetStartPositioned = false;\n\n for (let i = 0; i < store.length; i++) {\n const item = store[i];\n if (item.end > containerWidth + scrollLeft) {\n break;\n }\n this.end.set(i);\n if (item.end >= scrollLeft && !isSetStartPositioned) {\n start = i;\n isSetStartPositioned = true;\n }\n }\n\n if (start !== this.start()) {\n this.start.set(start);\n const paddingLeft = store[start].start;\n this.elementRef.nativeElement.style.paddingLeft = `${paddingLeft}px`;\n this.outPaddingLeft.emit(paddingLeft);\n }\n\n if (this.start() < 0) {\n this.start.set(0);\n }\n this.viewPortItems.set(store.slice(this.start(), this.end() + 3).map((item) => item.ref));\n this.outViewPortItem.emit(this.viewPortItems());\n }\n\n private async scrollInto(itemScroll: any) {\n const itemFound = this.storeItems().find((item) => item.ref === itemScroll);\n if (!itemFound) {\n return;\n }\n this.ScrollContainer.scrollLeft = itemFound.start;\n }\n\n private async scrollToPosition(position: number) {\n this.ScrollContainer.scrollLeft = position;\n }\n\n private async scrollToIndex(index: number) {\n const itemOfIndex = this.storeItems()[index];\n if (!itemOfIndex) {\n return;\n }\n this.ScrollContainer.scrollLeft = itemOfIndex.start;\n }\n\n private async reCalculatorViewPort() {\n let width = 0;\n const items = this.storeItems();\n\n for (const item of items) {\n const itemWidth = await this.functionGetWidthItem()(item.ref);\n item.itemWidth = itemWidth;\n item.start = width;\n width += itemWidth;\n item.end = width;\n }\n\n this.storeItems.set([...items]);\n this.divVirtual.style.width = `${width}px`;\n this.handlerScroll();\n }\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"libs-ui-components-scroll-measure-items-direction-horizontal.mjs","sources":["../../../../../../libs-ui/components/scroll-measure-items/direction-horizontal/src/direction-horizontal.directive.ts","../../../../../../libs-ui/components/scroll-measure-items/direction-horizontal/src/libs-ui-components-scroll-measure-items-direction-horizontal.ts"],"sourcesContent":["// CONVENTION-EXCEPT: any — Generic directive works with unknown item types from consumer\n/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { AfterViewInit, DestroyRef, Directive, ElementRef, effect, inject, input, output, signal, untracked } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { fromEvent } from 'rxjs';\nimport { IScrollMeasureItemDirectionHorizontalFunctionsControl, IScrollMeasureStoreItemConvert } from './interfaces/direction-horizontal.interface';\n@Directive({\n // eslint-disable-next-line @angular-eslint/directive-selector\n selector: '[LibsUiScrollMeasureItemDirectionHorizontalDirective]',\n standalone: true,\n})\nexport class LibsUiScrollMeasureItemDirectionHorizontalDirective implements AfterViewInit {\n // #region PROPERTY\n /** Số item buffer render thêm sau viewport để tránh trắng vùng khi scroll. */\n private readonly bufferAhead = 3;\n\n private viewPortItems = signal<Array<any>>([]);\n private start = signal<number>(0);\n private end = signal<number>(0);\n private storeItems = signal<Array<IScrollMeasureStoreItemConvert>>([]);\n private divVirtual = document.createElement('div');\n private resizeObserver?: ResizeObserver;\n /** Chặn ResizeObserver / subscribe được tạo trong setTimeout sau khi directive đã destroy. */\n private timerAfterViewInit?: number;\n /** Token nhận dạng lần effect chạy hiện tại. Async đo width xong sẽ bỏ kết quả nếu token đổi. */\n private executionToken = 0;\n /** Style + DOM hook của divVirtual chỉ cần khởi tạo một lần cho cả vòng đời directive. */\n private divVirtualInitialized = false;\n\n // #region INPUT\n readonly elementScroll = input.required<HTMLElement>();\n readonly functionGetWidthItem = input.required<(item: any) => Promise<number>>();\n readonly items = input.required<Array<any>>();\n\n // #region OUTPUT\n readonly outViewPortItem = output<Array<any>>();\n readonly outFunctionControl = output<IScrollMeasureItemDirectionHorizontalFunctionsControl>();\n readonly outDivVirtual = output<HTMLDivElement>();\n readonly outPaddingLeft = output<number>();\n\n // #region INJECT\n private readonly elementRef = inject(ElementRef);\n private readonly destroyRef = inject(DestroyRef);\n\n constructor() {\n effect(() => {\n const items = this.items();\n const myToken = ++this.executionToken;\n untracked(async () => {\n const scrollEl = this.ScrollContainer;\n const functionGetWidthItem = this.functionGetWidthItem();\n if (!scrollEl || !functionGetWidthItem) {\n return;\n }\n\n // Đo width tất cả item song song — rút ngắn thời gian cho list lớn (vd 3000 cột).\n const widths = await Promise.all(items.map((it) => functionGetWidthItem(it)));\n // Bị huỷ vì có lượt items() mới chạy sau — bỏ kết quả cũ tránh đè state.\n if (myToken !== this.executionToken) {\n return;\n }\n\n this.start.set(0);\n const scrollLeft = scrollEl.scrollLeft;\n const containerWidth = scrollEl.getBoundingClientRect().width;\n let width = 0;\n let endLocal = 0;\n const newItems: Array<IScrollMeasureStoreItemConvert> = [];\n\n for (let i = 0; i < items.length; i++) {\n const itemWidth = widths[i];\n const newItem: IScrollMeasureStoreItemConvert = { ref: items[i], itemWidth, start: width, end: 0 };\n width += itemWidth;\n newItem.end = width;\n if (newItem.end <= containerWidth + scrollLeft) {\n endLocal = i;\n }\n newItems.push(newItem);\n }\n // Set end một lần thay vì ghi mỗi vòng lặp — tránh notify consumer N lần.\n this.end.set(endLocal);\n this.storeItems.set(newItems);\n\n // Khởi tạo style + DOM hook của divVirtual đúng một lần.\n if (!this.divVirtualInitialized) {\n this.divVirtual.style.position = 'absolute';\n this.divVirtual.style.top = '0px';\n this.divVirtual.style.left = '0px';\n this.divVirtual.style.height = '1px';\n this.divVirtual.style.visibility = 'hidden';\n this.elementRef.nativeElement.classList.add('relative');\n this.elementRef.nativeElement.classList.remove('w-full');\n this.elementRef.nativeElement.style.paddingLeft = '0px';\n scrollEl.append(this.divVirtual);\n this.divVirtualInitialized = true;\n this.outDivVirtual.emit(this.divVirtual);\n }\n // Width tổng thể đổi theo dataset → cần update mỗi lượt.\n this.divVirtual.style.width = `${width}px`;\n\n // containerWidth lúc đầu có thể = 0 (DOM chưa layout), khiến end giữ\n // nguyên 0 và chỉ render 3 cột đầu. Re-đo bằng handlerScroll sau khi vòng\n // lặp await xong (layout thường đã hoàn tất) để end ra đúng giá trị.\n this.handlerScroll();\n });\n });\n\n this.destroyRef.onDestroy(() => {\n clearTimeout(this.timerAfterViewInit);\n this.divVirtual.remove();\n this.resizeObserver?.disconnect();\n });\n }\n\n ngAfterViewInit() {\n this.timerAfterViewInit = setTimeout(() => {\n fromEvent<Event>(this.ScrollContainer, 'scroll')\n .pipe(takeUntilDestroyed(this.destroyRef))\n .subscribe(() => this.handlerScroll());\n this.outFunctionControl.emit(this.FunctionsControl);\n\n // Re-tính viewport sau khi container có kích thước thật.\n // Lần đầu render, getBoundingClientRect().width có thể = 0 nên end giữ giá trị 0,\n // chỉ hiển thị 3 cột đầu (do slice(start, end + bufferAhead)). ResizeObserver fire ngay khi\n // observe và mỗi lần container đổi size sẽ kéo end về đúng giá trị.\n this.resizeObserver = new ResizeObserver(() => this.handlerScroll());\n this.resizeObserver.observe(this.ScrollContainer);\n });\n }\n\n // #region PUBLIC API\n public get FunctionsControl(): IScrollMeasureItemDirectionHorizontalFunctionsControl {\n return {\n scrollInto: this.scrollInto.bind(this),\n scrollToPosition: this.scrollToPosition.bind(this),\n scrollToIndex: this.scrollToIndex.bind(this),\n reCalculatorViewPort: this.reCalculatorViewPort.bind(this),\n getViewPortItems: () => this.viewPortItems(),\n };\n }\n\n // #region PRIVATE METHODS\n private get ScrollContainer(): HTMLElement {\n return this.elementScroll() || this.elementRef.nativeElement;\n }\n\n private handlerScroll() {\n const store = this.storeItems();\n if (!store?.length) {\n return;\n }\n const scrollEl = this.ScrollContainer;\n const scrollLeft = scrollEl.scrollLeft;\n const containerWidth = scrollEl.getBoundingClientRect().width;\n let start = this.start();\n let endLocal = this.end();\n let isSetStartPositioned = false;\n\n for (let i = 0; i < store.length; i++) {\n const item = store[i];\n if (item.end > containerWidth + scrollLeft) {\n break;\n }\n endLocal = i;\n if (item.end >= scrollLeft && !isSetStartPositioned) {\n start = i;\n isSetStartPositioned = true;\n }\n }\n if (endLocal !== this.end()) {\n this.end.set(endLocal);\n }\n\n if (start !== this.start()) {\n this.start.set(start);\n const paddingLeft = store[start].start;\n this.elementRef.nativeElement.style.paddingLeft = `${paddingLeft}px`;\n this.outPaddingLeft.emit(paddingLeft);\n }\n this.viewPortItems.set(store.slice(this.start(), this.end() + this.bufferAhead).map((item) => item.ref));\n this.outViewPortItem.emit(this.viewPortItems());\n }\n\n private async scrollInto(itemScroll: any) {\n const itemFound = this.storeItems().find((item) => item.ref === itemScroll);\n if (!itemFound) {\n return;\n }\n this.ScrollContainer.scrollLeft = itemFound.start;\n }\n\n private async scrollToPosition(position: number) {\n this.ScrollContainer.scrollLeft = position;\n }\n\n private async scrollToIndex(index: number) {\n const itemOfIndex = this.storeItems()[index];\n if (!itemOfIndex) {\n return;\n }\n this.ScrollContainer.scrollLeft = itemOfIndex.start;\n }\n\n private async reCalculatorViewPort() {\n const items = this.storeItems();\n const fn = this.functionGetWidthItem();\n // Đo song song để rút ngắn thời gian thay vì await tuần tự.\n const widths = await Promise.all(items.map((it) => fn(it.ref)));\n\n let width = 0;\n for (let i = 0; i < items.length; i++) {\n const item = items[i];\n const itemWidth = widths[i];\n item.itemWidth = itemWidth;\n item.start = width;\n width += itemWidth;\n item.end = width;\n }\n\n this.storeItems.set([...items]);\n this.divVirtual.style.width = `${width}px`;\n this.handlerScroll();\n }\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":";;;;;AAAA;AACA;MAUa,mDAAmD,CAAA;;;IAG7C,WAAW,GAAG,CAAC;AAExB,IAAA,aAAa,GAAG,MAAM,CAAa,EAAE,CAAC;AACtC,IAAA,KAAK,GAAG,MAAM,CAAS,CAAC,CAAC;AACzB,IAAA,GAAG,GAAG,MAAM,CAAS,CAAC,CAAC;AACvB,IAAA,UAAU,GAAG,MAAM,CAAwC,EAAE,CAAC;AAC9D,IAAA,UAAU,GAAG,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC;AAC1C,IAAA,cAAc;;AAEd,IAAA,kBAAkB;;IAElB,cAAc,GAAG,CAAC;;IAElB,qBAAqB,GAAG,KAAK;;AAG5B,IAAA,aAAa,GAAG,KAAK,CAAC,QAAQ,EAAe;AAC7C,IAAA,oBAAoB,GAAG,KAAK,CAAC,QAAQ,EAAkC;AACvE,IAAA,KAAK,GAAG,KAAK,CAAC,QAAQ,EAAc;;IAGpC,eAAe,GAAG,MAAM,EAAc;IACtC,kBAAkB,GAAG,MAAM,EAAyD;IACpF,aAAa,GAAG,MAAM,EAAkB;IACxC,cAAc,GAAG,MAAM,EAAU;;AAGzB,IAAA,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC;AAC/B,IAAA,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC;AAEhD,IAAA,WAAA,GAAA;QACE,MAAM,CAAC,MAAK;AACV,YAAA,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE;AAC1B,YAAA,MAAM,OAAO,GAAG,EAAE,IAAI,CAAC,cAAc;YACrC,SAAS,CAAC,YAAW;AACnB,gBAAA,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe;AACrC,gBAAA,MAAM,oBAAoB,GAAG,IAAI,CAAC,oBAAoB,EAAE;AACxD,gBAAA,IAAI,CAAC,QAAQ,IAAI,CAAC,oBAAoB,EAAE;oBACtC;gBACF;;gBAGA,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,oBAAoB,CAAC,EAAE,CAAC,CAAC,CAAC;;AAE7E,gBAAA,IAAI,OAAO,KAAK,IAAI,CAAC,cAAc,EAAE;oBACnC;gBACF;AAEA,gBAAA,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;AACjB,gBAAA,MAAM,UAAU,GAAG,QAAQ,CAAC,UAAU;gBACtC,MAAM,cAAc,GAAG,QAAQ,CAAC,qBAAqB,EAAE,CAAC,KAAK;gBAC7D,IAAI,KAAK,GAAG,CAAC;gBACb,IAAI,QAAQ,GAAG,CAAC;gBAChB,MAAM,QAAQ,GAA0C,EAAE;AAE1D,gBAAA,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;AACrC,oBAAA,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,CAAC;oBAC3B,MAAM,OAAO,GAAmC,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,SAAS,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,EAAE;oBAClG,KAAK,IAAI,SAAS;AAClB,oBAAA,OAAO,CAAC,GAAG,GAAG,KAAK;oBACnB,IAAI,OAAO,CAAC,GAAG,IAAI,cAAc,GAAG,UAAU,EAAE;wBAC9C,QAAQ,GAAG,CAAC;oBACd;AACA,oBAAA,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC;gBACxB;;AAEA,gBAAA,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC;AACtB,gBAAA,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,QAAQ,CAAC;;AAG7B,gBAAA,IAAI,CAAC,IAAI,CAAC,qBAAqB,EAAE;oBAC/B,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,QAAQ,GAAG,UAAU;oBAC3C,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,GAAG,KAAK;oBACjC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,GAAG,KAAK;oBAClC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,MAAM,GAAG,KAAK;oBACpC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,UAAU,GAAG,QAAQ;oBAC3C,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,CAAC;oBACvD,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC;oBACxD,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,KAAK,CAAC,WAAW,GAAG,KAAK;AACvD,oBAAA,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC;AAChC,oBAAA,IAAI,CAAC,qBAAqB,GAAG,IAAI;oBACjC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;gBAC1C;;gBAEA,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,KAAK,GAAG,CAAA,EAAG,KAAK,CAAA,EAAA,CAAI;;;;gBAK1C,IAAI,CAAC,aAAa,EAAE;AACtB,YAAA,CAAC,CAAC;AACJ,QAAA,CAAC,CAAC;AAEF,QAAA,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,MAAK;AAC7B,YAAA,YAAY,CAAC,IAAI,CAAC,kBAAkB,CAAC;AACrC,YAAA,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE;AACxB,YAAA,IAAI,CAAC,cAAc,EAAE,UAAU,EAAE;AACnC,QAAA,CAAC,CAAC;IACJ;IAEA,eAAe,GAAA;AACb,QAAA,IAAI,CAAC,kBAAkB,GAAG,UAAU,CAAC,MAAK;AACxC,YAAA,SAAS,CAAQ,IAAI,CAAC,eAAe,EAAE,QAAQ;AAC5C,iBAAA,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC;iBACxC,SAAS,CAAC,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;YACxC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC;;;;;AAMnD,YAAA,IAAI,CAAC,cAAc,GAAG,IAAI,cAAc,CAAC,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;YACpE,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC;AACnD,QAAA,CAAC,CAAC;IACJ;;AAGA,IAAA,IAAW,gBAAgB,GAAA;QACzB,OAAO;YACL,UAAU,EAAE,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC;YACtC,gBAAgB,EAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC;YAClD,aAAa,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC;YAC5C,oBAAoB,EAAE,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC;AAC1D,YAAA,gBAAgB,EAAE,MAAM,IAAI,CAAC,aAAa,EAAE;SAC7C;IACH;;AAGA,IAAA,IAAY,eAAe,GAAA;QACzB,OAAO,IAAI,CAAC,aAAa,EAAE,IAAI,IAAI,CAAC,UAAU,CAAC,aAAa;IAC9D;IAEQ,aAAa,GAAA;AACnB,QAAA,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,EAAE;AAC/B,QAAA,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE;YAClB;QACF;AACA,QAAA,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe;AACrC,QAAA,MAAM,UAAU,GAAG,QAAQ,CAAC,UAAU;QACtC,MAAM,cAAc,GAAG,QAAQ,CAAC,qBAAqB,EAAE,CAAC,KAAK;AAC7D,QAAA,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE;AACxB,QAAA,IAAI,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE;QACzB,IAAI,oBAAoB,GAAG,KAAK;AAEhC,QAAA,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;AACrC,YAAA,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC;YACrB,IAAI,IAAI,CAAC,GAAG,GAAG,cAAc,GAAG,UAAU,EAAE;gBAC1C;YACF;YACA,QAAQ,GAAG,CAAC;YACZ,IAAI,IAAI,CAAC,GAAG,IAAI,UAAU,IAAI,CAAC,oBAAoB,EAAE;gBACnD,KAAK,GAAG,CAAC;gBACT,oBAAoB,GAAG,IAAI;YAC7B;QACF;AACA,QAAA,IAAI,QAAQ,KAAK,IAAI,CAAC,GAAG,EAAE,EAAE;AAC3B,YAAA,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC;QACxB;AAEA,QAAA,IAAI,KAAK,KAAK,IAAI,CAAC,KAAK,EAAE,EAAE;AAC1B,YAAA,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC;YACrB,MAAM,WAAW,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,KAAK;AACtC,YAAA,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,KAAK,CAAC,WAAW,GAAG,CAAA,EAAG,WAAW,CAAA,EAAA,CAAI;AACpE,YAAA,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,WAAW,CAAC;QACvC;AACA,QAAA,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,KAAK,IAAI,CAAC,GAAG,CAAC,CAAC;QACxG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;IACjD;IAEQ,MAAM,UAAU,CAAC,UAAe,EAAA;QACtC,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,KAAK,IAAI,CAAC,GAAG,KAAK,UAAU,CAAC;QAC3E,IAAI,CAAC,SAAS,EAAE;YACd;QACF;QACA,IAAI,CAAC,eAAe,CAAC,UAAU,GAAG,SAAS,CAAC,KAAK;IACnD;IAEQ,MAAM,gBAAgB,CAAC,QAAgB,EAAA;AAC7C,QAAA,IAAI,CAAC,eAAe,CAAC,UAAU,GAAG,QAAQ;IAC5C;IAEQ,MAAM,aAAa,CAAC,KAAa,EAAA;QACvC,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,KAAK,CAAC;QAC5C,IAAI,CAAC,WAAW,EAAE;YAChB;QACF;QACA,IAAI,CAAC,eAAe,CAAC,UAAU,GAAG,WAAW,CAAC,KAAK;IACrD;AAEQ,IAAA,MAAM,oBAAoB,GAAA;AAChC,QAAA,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,EAAE;AAC/B,QAAA,MAAM,EAAE,GAAG,IAAI,CAAC,oBAAoB,EAAE;;QAEtC,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;QAE/D,IAAI,KAAK,GAAG,CAAC;AACb,QAAA,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;AACrC,YAAA,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC;AACrB,YAAA,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,CAAC;AAC3B,YAAA,IAAI,CAAC,SAAS,GAAG,SAAS;AAC1B,YAAA,IAAI,CAAC,KAAK,GAAG,KAAK;YAClB,KAAK,IAAI,SAAS;AAClB,YAAA,IAAI,CAAC,GAAG,GAAG,KAAK;QAClB;QAEA,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC;QAC/B,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,KAAK,GAAG,CAAA,EAAG,KAAK,CAAA,EAAA,CAAI;QAC1C,IAAI,CAAC,aAAa,EAAE;IACtB;wGAnNW,mDAAmD,EAAA,IAAA,EAAA,EAAA,EAAA,MAAA,EAAA,EAAA,CAAA,eAAA,CAAA,SAAA,EAAA,CAAA;4FAAnD,mDAAmD,EAAA,YAAA,EAAA,IAAA,EAAA,QAAA,EAAA,uDAAA,EAAA,MAAA,EAAA,EAAA,aAAA,EAAA,EAAA,iBAAA,EAAA,eAAA,EAAA,UAAA,EAAA,eAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,oBAAA,EAAA,EAAA,iBAAA,EAAA,sBAAA,EAAA,UAAA,EAAA,sBAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,KAAA,EAAA,EAAA,iBAAA,EAAA,OAAA,EAAA,UAAA,EAAA,OAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,EAAA,OAAA,EAAA,EAAA,eAAA,EAAA,iBAAA,EAAA,kBAAA,EAAA,oBAAA,EAAA,aAAA,EAAA,eAAA,EAAA,cAAA,EAAA,gBAAA,EAAA,EAAA,QAAA,EAAA,EAAA,EAAA,CAAA;;4FAAnD,mDAAmD,EAAA,UAAA,EAAA,CAAA;kBAL/D,SAAS;AAAC,YAAA,IAAA,EAAA,CAAA;;AAET,oBAAA,QAAQ,EAAE,uDAAuD;AACjE,oBAAA,UAAU,EAAE,IAAI;AACjB,iBAAA;;;ACVD;;AAEG;;;;"}
|
package/package.json
CHANGED
|
@@ -1,9 +1,8 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@libs-ui/components-scroll-measure-items-direction-horizontal",
|
|
3
|
-
"version": "0.2.356-
|
|
3
|
+
"version": "0.2.356-26",
|
|
4
4
|
"peerDependencies": {
|
|
5
5
|
"@angular/core": ">=18.0.0",
|
|
6
|
-
"@libs-ui/utils": "0.2.356-25",
|
|
7
6
|
"rxjs": "~7.8.0"
|
|
8
7
|
},
|
|
9
8
|
"sideEffects": false,
|