ng-virtual-list 18.7.13 → 18.7.15

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.
@@ -1,7 +1,8 @@
1
1
  import { Injectable } from '@angular/core';
2
- import { Subject, tap } from 'rxjs';
3
- import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
4
2
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
4
+ import { Subject, tap } from 'rxjs';
5
+ import { FocusAlignments } from './enums';
5
6
  import { MethodsForSelectingTypes } from './enums/method-for-selecting-types';
6
7
  import { DEFAULT_COLLAPSE_BY_CLICK, DEFAULT_SELECT_BY_CLICK } from './const';
7
8
  import * as i0 from "@angular/core";
@@ -25,7 +26,17 @@ export class NgVirtualListService {
25
26
  collapseByClick = DEFAULT_COLLAPSE_BY_CLICK;
26
27
  _trackBox;
27
28
  listElement = null;
28
- collection = [];
29
+ _$displayItems = new BehaviorSubject([]);
30
+ $displayItems = this._$displayItems.asObservable();
31
+ _collection = [];
32
+ set collection(v) {
33
+ if (this._collection === v) {
34
+ return;
35
+ }
36
+ this._collection = v;
37
+ this._$displayItems.next(v);
38
+ }
39
+ get collection() { return this._collection; }
29
40
  constructor() {
30
41
  this._$methodOfSelecting.pipe(takeUntilDestroyed(), tap(v => {
31
42
  switch (v) {
@@ -164,11 +175,11 @@ export class NgVirtualListService {
164
175
  }
165
176
  }
166
177
  itemToFocus;
167
- focus(element) {
178
+ focus(element, align = FocusAlignments.CENTER) {
168
179
  element.focus({ preventScroll: true });
169
180
  if (element.parentElement) {
170
181
  const pos = parseFloat(element.parentElement?.getAttribute('position') ?? '0');
171
- this.itemToFocus?.(element, pos);
182
+ this.itemToFocus?.(element, pos, align);
172
183
  }
173
184
  }
174
185
  areaFocus(id) {
@@ -190,4 +201,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
190
201
  providedIn: 'root'
191
202
  }]
192
203
  }], ctorParameters: () => [] });
193
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-virtual-list.service.js","sourceRoot":"","sources":["../../../../projects/ng-virtual-list/src/lib/ng-virtual-list.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { Subject, tap } from 'rxjs';\r\nimport { TrackBox } from './utils/trackBox';\r\nimport { IRenderVirtualListItem } from './models';\r\nimport { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';\r\nimport { Id } from './types';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { MethodsForSelectingTypes } from './enums/method-for-selecting-types';\r\nimport { DEFAULT_COLLAPSE_BY_CLICK, DEFAULT_SELECT_BY_CLICK } from './const';\r\nimport { IRenderVirtualListCollection } from './models/render-collection.model';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class NgVirtualListService {\r\n  private _nextComponentId: number = 0;\r\n\r\n  private _$itemClick = new Subject<IRenderVirtualListItem<any> | undefined>();\r\n  $itemClick = this._$itemClick.asObservable();\r\n\r\n  private _$selectedIds = new BehaviorSubject<Array<Id> | Id | undefined>(undefined);\r\n  $selectedIds = this._$selectedIds.asObservable();\r\n\r\n  private _$collapsedIds = new BehaviorSubject<Array<Id>>([]);\r\n  $collapsedIds = this._$collapsedIds.asObservable();\r\n\r\n  private _$methodOfSelecting = new BehaviorSubject<MethodsForSelectingTypes>(0);\r\n  $methodOfSelecting = this._$methodOfSelecting.asObservable();\r\n\r\n  set methodOfSelecting(v: MethodsForSelectingTypes) {\r\n    this._$methodOfSelecting.next(v);\r\n  }\r\n\r\n  private _$focusedId = new BehaviorSubject<Id | null>(null);\r\n  $focusedId = this._$focusedId.asObservable();\r\n  get focusedId() { return this._$focusedId.getValue(); }\r\n\r\n  selectByClick: boolean = DEFAULT_SELECT_BY_CLICK;\r\n\r\n  collapseByClick: boolean = DEFAULT_COLLAPSE_BY_CLICK;\r\n\r\n  private _trackBox: TrackBox | undefined;\r\n\r\n  listElement: HTMLDivElement | null = null;\r\n\r\n  collection: IRenderVirtualListCollection = [];\r\n\r\n  constructor() {\r\n    this._$methodOfSelecting.pipe(\r\n      takeUntilDestroyed(),\r\n      tap(v => {\r\n        switch (v) {\r\n          case MethodsForSelectingTypes.SELECT: {\r\n            const curr = this._$selectedIds.getValue();\r\n            if (typeof curr !== 'number' && typeof curr !== 'string') {\r\n              this._$selectedIds.next(undefined);\r\n            }\r\n            break;\r\n          }\r\n          case MethodsForSelectingTypes.MULTI_SELECT: {\r\n            if (!Array.isArray(this._$selectedIds.getValue())) {\r\n              this._$selectedIds.next([]);\r\n            }\r\n            break;\r\n          }\r\n          case MethodsForSelectingTypes.NONE:\r\n          default: {\r\n            this._$selectedIds.next(undefined);\r\n            break;\r\n          }\r\n        }\r\n      }),\r\n    ).subscribe();\r\n  }\r\n\r\n  setSelectedIds(ids: Array<Id> | Id | undefined) {\r\n    if (JSON.stringify(this._$selectedIds.getValue()) !== JSON.stringify(ids)) {\r\n      this._$selectedIds.next(ids);\r\n    }\r\n  }\r\n\r\n  setCollapsedIds(ids: Array<Id>) {\r\n    if (JSON.stringify(this._$collapsedIds.getValue()) !== JSON.stringify(ids)) {\r\n      this._$collapsedIds.next(ids);\r\n    }\r\n  }\r\n\r\n  itemClick(data: IRenderVirtualListItem | undefined) {\r\n    this._$itemClick.next(data);\r\n    if (this.collapseByClick) {\r\n      this.collapse(data);\r\n    }\r\n    if (this.selectByClick) {\r\n      this.select(data);\r\n    }\r\n  }\r\n\r\n  update() {\r\n    this._trackBox?.changes();\r\n  }\r\n\r\n  /**\r\n   * Selects a list item\r\n   * @param data \r\n   * @param selected - If the value is undefined, then the toggle method is executed, if false or true, then the selection/deselection is performed.\r\n   */\r\n  select(data: IRenderVirtualListItem | undefined, selected: boolean | undefined = undefined) {\r\n    if (data && data.config.selectable) {\r\n      switch (this._$methodOfSelecting.getValue()) {\r\n        case MethodsForSelectingTypes.SELECT: {\r\n          const curr = this._$selectedIds.getValue() as (Id | undefined);\r\n          if (selected === undefined) {\r\n            this._$selectedIds.next(curr !== data?.id ? data?.id : undefined);\r\n          } else {\r\n            this._$selectedIds.next(selected ? data?.id : undefined);\r\n          }\r\n          break;\r\n        }\r\n        case MethodsForSelectingTypes.MULTI_SELECT: {\r\n          const curr = [...(this._$selectedIds.getValue() || []) as Array<Id>], index = curr.indexOf(data.id);\r\n          if (selected === undefined) {\r\n            if (index > -1) {\r\n              curr.splice(index, 1);\r\n              this._$selectedIds.next(curr);\r\n            } else {\r\n              this._$selectedIds.next([...curr, data.id]);\r\n            }\r\n          } else if (selected) {\r\n            if (index > -1) {\r\n              this._$selectedIds.next(curr);\r\n            } else {\r\n              this._$selectedIds.next([...curr, data.id]);\r\n            }\r\n          } else {\r\n            if (index > -1) {\r\n              curr.splice(index, 1);\r\n              this._$selectedIds.next(curr);\r\n            } else {\r\n              this._$selectedIds.next(curr);\r\n            }\r\n          }\r\n          break;\r\n        }\r\n        case MethodsForSelectingTypes.NONE:\r\n        default: {\r\n          this._$selectedIds.next(undefined);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n    * Collapse list items\r\n    * @param data \r\n    * @param collapsed - If the value is undefined, then the toggle method is executed, if false or true, then the collapse/expand is performed.\r\n    */\r\n  collapse(data: IRenderVirtualListItem | undefined, collapsed: boolean | undefined = undefined) {\r\n    if (data && data.config.sticky > 0 && data.config.collapsable) {\r\n      const curr = [...(this._$collapsedIds.getValue() || []) as Array<Id>], index = curr.indexOf(data.id);\r\n      if (collapsed === undefined) {\r\n        if (index > -1) {\r\n          curr.splice(index, 1);\r\n          this._$collapsedIds.next(curr);\r\n        } else {\r\n          this._$collapsedIds.next([...curr, data.id]);\r\n        }\r\n      } else if (collapsed) {\r\n        if (index > -1) {\r\n          this._$collapsedIds.next(curr);\r\n        } else {\r\n          this._$collapsedIds.next([...curr, data.id]);\r\n        }\r\n      } else {\r\n        if (index > -1) {\r\n          curr.splice(index, 1);\r\n          this._$collapsedIds.next(curr);\r\n        } else {\r\n          this._$collapsedIds.next(curr);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  itemToFocus: ((element: HTMLElement, position: number) => void) | undefined;\r\n\r\n  focus(element: HTMLElement) {\r\n    element.focus({ preventScroll: true });\r\n    if (element.parentElement) {\r\n      const pos = parseFloat(element.parentElement?.getAttribute('position') ?? '0');\r\n      this.itemToFocus?.(element, pos);\r\n    }\r\n  }\r\n\r\n  areaFocus(id: Id | null) {\r\n    this._$focusedId.next(id);\r\n  }\r\n\r\n  initialize(trackBox: TrackBox) {\r\n    this._trackBox = trackBox;\r\n  }\r\n\r\n  generateComponentId() {\r\n    return this._nextComponentId = this._nextComponentId === Number.MAX_SAFE_INTEGER\r\n      ? 0 : this._nextComponentId + 1;\r\n  }\r\n}\r\n"]}
204
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-virtual-list.service.js","sourceRoot":"","sources":["../../../../projects/ng-virtual-list/src/lib/ng-virtual-list.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';\r\nimport { Subject, tap } from 'rxjs';\r\nimport { TrackBox } from './utils/trackBox';\r\nimport { IRenderVirtualListItem } from './models';\r\nimport { IRenderVirtualListCollection } from './models/render-collection.model';\r\nimport { FocusAlignments } from './enums';\r\nimport { MethodsForSelectingTypes } from './enums/method-for-selecting-types';\r\nimport { DEFAULT_COLLAPSE_BY_CLICK, DEFAULT_SELECT_BY_CLICK } from './const';\r\nimport { FocusAlignment, Id } from './types';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class NgVirtualListService {\r\n  private _nextComponentId: number = 0;\r\n\r\n  private _$itemClick = new Subject<IRenderVirtualListItem<any> | undefined>();\r\n  $itemClick = this._$itemClick.asObservable();\r\n\r\n  private _$selectedIds = new BehaviorSubject<Array<Id> | Id | undefined>(undefined);\r\n  $selectedIds = this._$selectedIds.asObservable();\r\n\r\n  private _$collapsedIds = new BehaviorSubject<Array<Id>>([]);\r\n  $collapsedIds = this._$collapsedIds.asObservable();\r\n\r\n  private _$methodOfSelecting = new BehaviorSubject<MethodsForSelectingTypes>(0);\r\n  $methodOfSelecting = this._$methodOfSelecting.asObservable();\r\n\r\n  set methodOfSelecting(v: MethodsForSelectingTypes) {\r\n    this._$methodOfSelecting.next(v);\r\n  }\r\n\r\n  private _$focusedId = new BehaviorSubject<Id | null>(null);\r\n  $focusedId = this._$focusedId.asObservable();\r\n  get focusedId() { return this._$focusedId.getValue(); }\r\n\r\n  selectByClick: boolean = DEFAULT_SELECT_BY_CLICK;\r\n\r\n  collapseByClick: boolean = DEFAULT_COLLAPSE_BY_CLICK;\r\n\r\n  private _trackBox: TrackBox | undefined;\r\n\r\n  listElement: HTMLDivElement | null = null;\r\n\r\n  private _$displayItems = new BehaviorSubject<IRenderVirtualListCollection>([]);\r\n  readonly $displayItems = this._$displayItems.asObservable();\r\n\r\n  private _collection: IRenderVirtualListCollection = [];\r\n  set collection(v: IRenderVirtualListCollection) {\r\n    if (this._collection === v) {\r\n      return;\r\n    }\r\n\r\n    this._collection = v;\r\n\r\n    this._$displayItems.next(v);\r\n  }\r\n  get collection() { return this._collection; }\r\n\r\n  constructor() {\r\n    this._$methodOfSelecting.pipe(\r\n      takeUntilDestroyed(),\r\n      tap(v => {\r\n        switch (v) {\r\n          case MethodsForSelectingTypes.SELECT: {\r\n            const curr = this._$selectedIds.getValue();\r\n            if (typeof curr !== 'number' && typeof curr !== 'string') {\r\n              this._$selectedIds.next(undefined);\r\n            }\r\n            break;\r\n          }\r\n          case MethodsForSelectingTypes.MULTI_SELECT: {\r\n            if (!Array.isArray(this._$selectedIds.getValue())) {\r\n              this._$selectedIds.next([]);\r\n            }\r\n            break;\r\n          }\r\n          case MethodsForSelectingTypes.NONE:\r\n          default: {\r\n            this._$selectedIds.next(undefined);\r\n            break;\r\n          }\r\n        }\r\n      }),\r\n    ).subscribe();\r\n  }\r\n\r\n  setSelectedIds(ids: Array<Id> | Id | undefined) {\r\n    if (JSON.stringify(this._$selectedIds.getValue()) !== JSON.stringify(ids)) {\r\n      this._$selectedIds.next(ids);\r\n    }\r\n  }\r\n\r\n  setCollapsedIds(ids: Array<Id>) {\r\n    if (JSON.stringify(this._$collapsedIds.getValue()) !== JSON.stringify(ids)) {\r\n      this._$collapsedIds.next(ids);\r\n    }\r\n  }\r\n\r\n  itemClick(data: IRenderVirtualListItem | undefined) {\r\n    this._$itemClick.next(data);\r\n    if (this.collapseByClick) {\r\n      this.collapse(data);\r\n    }\r\n    if (this.selectByClick) {\r\n      this.select(data);\r\n    }\r\n  }\r\n\r\n  update() {\r\n    this._trackBox?.changes();\r\n  }\r\n\r\n  /**\r\n   * Selects a list item\r\n   * @param data \r\n   * @param selected - If the value is undefined, then the toggle method is executed, if false or true, then the selection/deselection is performed.\r\n   */\r\n  select(data: IRenderVirtualListItem | undefined, selected: boolean | undefined = undefined) {\r\n    if (data && data.config.selectable) {\r\n      switch (this._$methodOfSelecting.getValue()) {\r\n        case MethodsForSelectingTypes.SELECT: {\r\n          const curr = this._$selectedIds.getValue() as (Id | undefined);\r\n          if (selected === undefined) {\r\n            this._$selectedIds.next(curr !== data?.id ? data?.id : undefined);\r\n          } else {\r\n            this._$selectedIds.next(selected ? data?.id : undefined);\r\n          }\r\n          break;\r\n        }\r\n        case MethodsForSelectingTypes.MULTI_SELECT: {\r\n          const curr = [...(this._$selectedIds.getValue() || []) as Array<Id>], index = curr.indexOf(data.id);\r\n          if (selected === undefined) {\r\n            if (index > -1) {\r\n              curr.splice(index, 1);\r\n              this._$selectedIds.next(curr);\r\n            } else {\r\n              this._$selectedIds.next([...curr, data.id]);\r\n            }\r\n          } else if (selected) {\r\n            if (index > -1) {\r\n              this._$selectedIds.next(curr);\r\n            } else {\r\n              this._$selectedIds.next([...curr, data.id]);\r\n            }\r\n          } else {\r\n            if (index > -1) {\r\n              curr.splice(index, 1);\r\n              this._$selectedIds.next(curr);\r\n            } else {\r\n              this._$selectedIds.next(curr);\r\n            }\r\n          }\r\n          break;\r\n        }\r\n        case MethodsForSelectingTypes.NONE:\r\n        default: {\r\n          this._$selectedIds.next(undefined);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n    * Collapse list items\r\n    * @param data \r\n    * @param collapsed - If the value is undefined, then the toggle method is executed, if false or true, then the collapse/expand is performed.\r\n    */\r\n  collapse(data: IRenderVirtualListItem | undefined, collapsed: boolean | undefined = undefined) {\r\n    if (data && data.config.sticky > 0 && data.config.collapsable) {\r\n      const curr = [...(this._$collapsedIds.getValue() || []) as Array<Id>], index = curr.indexOf(data.id);\r\n      if (collapsed === undefined) {\r\n        if (index > -1) {\r\n          curr.splice(index, 1);\r\n          this._$collapsedIds.next(curr);\r\n        } else {\r\n          this._$collapsedIds.next([...curr, data.id]);\r\n        }\r\n      } else if (collapsed) {\r\n        if (index > -1) {\r\n          this._$collapsedIds.next(curr);\r\n        } else {\r\n          this._$collapsedIds.next([...curr, data.id]);\r\n        }\r\n      } else {\r\n        if (index > -1) {\r\n          curr.splice(index, 1);\r\n          this._$collapsedIds.next(curr);\r\n        } else {\r\n          this._$collapsedIds.next(curr);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  itemToFocus: ((element: HTMLElement, position: number, align: FocusAlignment) => void) | undefined;\r\n\r\n  focus(element: HTMLElement, align: FocusAlignment = FocusAlignments.CENTER) {\r\n    element.focus({ preventScroll: true });\r\n    if (element.parentElement) {\r\n      const pos = parseFloat(element.parentElement?.getAttribute('position') ?? '0');\r\n      this.itemToFocus?.(element, pos, align);\r\n    }\r\n  }\r\n\r\n  areaFocus(id: Id | null) {\r\n    this._$focusedId.next(id);\r\n  }\r\n\r\n  initialize(trackBox: TrackBox) {\r\n    this._trackBox = trackBox;\r\n  }\r\n\r\n  generateComponentId() {\r\n    return this._nextComponentId = this._nextComponentId === Number.MAX_SAFE_INTEGER\r\n      ? 0 : this._nextComponentId + 1;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9jdXMtYWxpZ25tZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctdmlydHVhbC1saXN0L3NyYy9saWIvdHlwZXMvZm9jdXMtYWxpZ25tZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGb2N1c0FsaWdubWVudHMgfSBmcm9tIFwiLi4vZW51bXNcIjtcclxuXHJcbi8qKlxyXG4gKiBGb2N1cyBBbGlnbm1lbnRcclxuICogQGxpbmsgaHR0cHM6Ly9naXRodWIuY29tL0Rqb25ueVgvbmctdmlydHVhbC1saXN0L2Jsb2IvMTgueC9wcm9qZWN0cy9uZy12aXJ0dWFsLWxpc3Qvc3JjL2xpYi90eXBlcy9mb2N1cy1hbGlnbm1lbnQudHNcclxuICogQGF1dGhvciBFdmdlbmlpIEdyZWJlbm5pa292XHJcbiAqIEBlbWFpbCBkam9ubnl4QGdtYWlsLmNvbVxyXG4gKi9cclxuZXhwb3J0IHR5cGUgRm9jdXNBbGlnbm1lbnQgPSBGb2N1c0FsaWdubWVudHMgfCAnbm9uZScgfCAnc3RhcnQnIHwgJ2NlbnRlcicgfCAnZW5kJztcclxuIl19
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy12aXJ0dWFsLWxpc3Qvc3JjL2xpYi90eXBlcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSWQgfSBmcm9tICcuL2lkJztcclxuaW1wb3J0IHsgSVNpemUgfSBmcm9tICcuL3NpemUnO1xyXG5pbXBvcnQgeyBJUmVjdCB9IGZyb20gJy4vcmVjdCc7XHJcblxyXG5leHBvcnQgdHlwZSB7XHJcbiAgICBJZCxcclxuICAgIElTaXplLFxyXG4gICAgSVJlY3QsXHJcbn0iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy12aXJ0dWFsLWxpc3Qvc3JjL2xpYi90eXBlcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSWQgfSBmcm9tICcuL2lkJztcclxuaW1wb3J0IHsgSVNpemUgfSBmcm9tICcuL3NpemUnO1xyXG5pbXBvcnQgeyBJUmVjdCB9IGZyb20gJy4vcmVjdCc7XHJcbmltcG9ydCB7IEZvY3VzQWxpZ25tZW50IH0gZnJvbSAnLi9mb2N1cy1hbGlnbm1lbnQnO1xyXG5cclxuZXhwb3J0IHR5cGUge1xyXG4gICAgSWQsXHJcbiAgICBJU2l6ZSxcclxuICAgIElSZWN0LFxyXG4gICAgRm9jdXNBbGlnbm1lbnQsXHJcbn0iXX0=
@@ -1,8 +1,8 @@
1
1
  import * as i0 from '@angular/core';
2
2
  import { Injectable, inject, ChangeDetectorRef, signal, ElementRef, Component, ChangeDetectionStrategy, viewChild, output, input, ViewContainerRef, ViewEncapsulation, ViewChild, NgModule, NO_ERRORS_SCHEMA } from '@angular/core';
3
- import { Subject, tap, fromEvent, combineLatest, map, filter, distinctUntilChanged, switchMap, of } from 'rxjs';
4
- import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
5
3
  import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
4
+ import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
5
+ import { Subject, tap, fromEvent, combineLatest, map, filter, distinctUntilChanged, debounceTime, switchMap, of } from 'rxjs';
6
6
  import * as i1 from '@angular/common';
7
7
  import { CommonModule } from '@angular/common';
8
8
 
@@ -82,6 +82,20 @@ var SnappingMethods;
82
82
  SnappingMethods["ADVANCED"] = "advanced";
83
83
  })(SnappingMethods || (SnappingMethods = {}));
84
84
 
85
+ /**
86
+ * Focus Alignments.
87
+ * @link https://github.com/DjonnyX/ng-virtual-list/blob/18.x/projects/ng-virtual-list/src/lib/enums/focus-alignments.ts
88
+ * @author Evgenii Grebennikov
89
+ * @email djonnyx@gmail.com
90
+ */
91
+ var FocusAlignments;
92
+ (function (FocusAlignments) {
93
+ FocusAlignments["NONE"] = "none";
94
+ FocusAlignments["START"] = "start";
95
+ FocusAlignments["CENTER"] = "center";
96
+ FocusAlignments["END"] = "end";
97
+ })(FocusAlignments || (FocusAlignments = {}));
98
+
85
99
  const DEFAULT_ITEM_SIZE = 24;
86
100
  const DEFAULT_BUFFER_SIZE = 2;
87
101
  const DEFAULT_MAX_BUFFER_SIZE = 10;
@@ -98,6 +112,7 @@ const DISPLAY_OBJECTS_LENGTH_MESUREMENT_ERROR = 1;
98
112
  const MAX_SCROLL_TO_ITERATIONS = 5;
99
113
  const DEFAULT_SNAPPING_METHOD = SnappingMethods.NORMAL;
100
114
  const DEFAULT_SELECT_METHOD = MethodsForSelecting.NONE;
115
+ const DEFAULT_SCREEN_READER_MESSAGE = 'Showing items $1 to $2';
101
116
  // presets
102
117
  const BEHAVIOR_AUTO = 'auto';
103
118
  const BEHAVIOR_INSTANT = 'instant';
@@ -188,7 +203,17 @@ class NgVirtualListService {
188
203
  collapseByClick = DEFAULT_COLLAPSE_BY_CLICK;
189
204
  _trackBox;
190
205
  listElement = null;
191
- collection = [];
206
+ _$displayItems = new BehaviorSubject([]);
207
+ $displayItems = this._$displayItems.asObservable();
208
+ _collection = [];
209
+ set collection(v) {
210
+ if (this._collection === v) {
211
+ return;
212
+ }
213
+ this._collection = v;
214
+ this._$displayItems.next(v);
215
+ }
216
+ get collection() { return this._collection; }
192
217
  constructor() {
193
218
  this._$methodOfSelecting.pipe(takeUntilDestroyed(), tap(v => {
194
219
  switch (v) {
@@ -327,11 +352,11 @@ class NgVirtualListService {
327
352
  }
328
353
  }
329
354
  itemToFocus;
330
- focus(element) {
355
+ focus(element, align = FocusAlignments.CENTER) {
331
356
  element.focus({ preventScroll: true });
332
357
  if (element.parentElement) {
333
358
  const pos = parseFloat(element.parentElement?.getAttribute('position') ?? '0');
334
- this.itemToFocus?.(element, pos);
359
+ this.itemToFocus?.(element, pos, align);
335
360
  }
336
361
  }
337
362
  areaFocus(id) {
@@ -410,7 +435,7 @@ const validateFunction = (value, undefinable = false, nullable = false) => {
410
435
  return (undefinable && isUndefinable(value)) || (nullable && isNullable(value)) || typeof value === 'function';
411
436
  };
412
437
 
413
- const ATTR_AREA_SELECTED = 'area-selected', TABINDEX = 'ng-vl-index', KEY_SPACE = " ", KEY_ARR_LEFT = "ArrowLeft", KEY_ARR_UP = "ArrowUp", KEY_ARR_RIGHT = "ArrowRight", KEY_ARR_DOWN = "ArrowDown", EVENT_FOCUS_IN = 'focusin', EVENT_FOCUS_OUT = 'focusout', EVENT_KEY_DOWN = 'keydown';
438
+ const ATTR_AREA_SELECTED = 'area-selected', TABINDEX = 'ng-vl-index', POSITION = 'position', POSITION_ZERO = '0', ID = 'item-id', KEY_SPACE = " ", KEY_ARR_LEFT = "ArrowLeft", KEY_ARR_UP = "ArrowUp", KEY_ARR_RIGHT = "ArrowRight", KEY_ARR_DOWN = "ArrowDown", EVENT_FOCUS_IN = 'focusin', EVENT_FOCUS_OUT = 'focusout', EVENT_KEY_DOWN = 'keydown';
414
439
  const getElementByIndex = (index) => {
415
440
  return `[${TABINDEX}="${index}"]`;
416
441
  };
@@ -431,7 +456,7 @@ class NgVirtualListItemComponent extends BaseVirtualListItemComponent {
431
456
  _isCollapsed = false;
432
457
  config = signal({});
433
458
  measures = signal(undefined);
434
- focus = signal(false);
459
+ focused = signal(false);
435
460
  part = signal(PART_DEFAULT_ITEM);
436
461
  regular = false;
437
462
  data = signal(undefined);
@@ -497,21 +522,28 @@ class NgVirtualListItemComponent extends BaseVirtualListItemComponent {
497
522
  }
498
523
  this._service.collapse(data, collapsed);
499
524
  };
525
+ _focusHandler = () =>
526
+ /**
527
+ * Focus a list item
528
+ */
529
+ (align = FocusAlignments.CENTER) => {
530
+ this.focus(align);
531
+ };
500
532
  constructor() {
501
533
  super();
502
534
  this._id = this._service.generateComponentId();
503
535
  this._elementRef.nativeElement.setAttribute('id', String(this._id));
504
- const $data = toObservable(this.data), $focus = toObservable(this.focus);
505
- $focus.pipe(takeUntilDestroyed(), tap(v => {
536
+ const $data = toObservable(this.data), $focused = toObservable(this.focused);
537
+ $focused.pipe(takeUntilDestroyed(), tap(v => {
506
538
  this._service.areaFocus(v ? this._id : this._service.focusedId === this._id ? null : this._service.focusedId);
507
539
  })).subscribe();
508
540
  fromEvent(this.element, EVENT_FOCUS_IN).pipe(takeUntilDestroyed(), tap(e => {
509
- this.focus.set(true);
541
+ this.focused.set(true);
510
542
  this.updateConfig(this._data);
511
543
  this.updatePartStr(this._data, this._isSelected, this._isCollapsed);
512
544
  })).subscribe(),
513
545
  fromEvent(this.element, EVENT_FOCUS_OUT).pipe(takeUntilDestroyed(), tap(e => {
514
- this.focus.set(false);
546
+ this.focused.set(false);
515
547
  this.updateConfig(this._data);
516
548
  this.updatePartStr(this._data, this._isSelected, this._isCollapsed);
517
549
  })).subscribe(),
@@ -614,23 +646,34 @@ class NgVirtualListItemComponent extends BaseVirtualListItemComponent {
614
646
  }
615
647
  }
616
648
  }
649
+ focus(align = FocusAlignments.CENTER) {
650
+ if (this._service.listElement) {
651
+ const tabIndex = this._data?.config?.tabIndex ?? 0;
652
+ let index = tabIndex;
653
+ const el = this._service.listElement.querySelector(getElementByIndex(index));
654
+ if (el) {
655
+ this._service.focus(el, align);
656
+ }
657
+ }
658
+ }
617
659
  updateMeasures(v) {
618
660
  this.measures.set(v?.measures ? { ...v.measures } : undefined);
619
661
  }
620
662
  updateConfig(v) {
621
663
  this.config.set({
622
- ...v?.config || {}, selected: this._isSelected, collapsed: this._isCollapsed, focus: this.focus(),
623
- collapse: this._collapseHandler(v), select: this._selectHandler(v)
664
+ ...v?.config || {}, selected: this._isSelected, collapsed: this._isCollapsed, focused: this.focused(),
665
+ collapse: this._collapseHandler(v), select: this._selectHandler(v), focus: this._focusHandler(),
624
666
  });
625
667
  }
626
668
  update() {
627
669
  const data = this._data, regular = this.regular, length = this._regularLength;
628
670
  if (data) {
671
+ this._elementRef.nativeElement.setAttribute(ID, `${data.id}`);
629
672
  const styles = this._elementRef.nativeElement.style;
630
673
  styles.zIndex = data.config.zIndex;
631
674
  if (data.config.snapped) {
632
- this._elementRef.nativeElement.setAttribute('position', data.config.sticky === 1 ? '0' : `${data.config.isVertical ? data.measures.y : data.measures.x}`);
633
- styles.transform = data.config.sticky === 1 ? ZEROS_TRANSLATE_3D : `${TRANSLATE_3D}(${data.config.isVertical ? 0 : data.measures.x}${PX}, ${data.config.isVertical ? data.measures.y : 0}${PX} , 0)`;
675
+ this._elementRef.nativeElement.setAttribute(POSITION, data.config.sticky === 1 ? POSITION_ZERO : `${data.config.isVertical ? data.measures.y : data.measures.x}`);
676
+ styles.transform = data.config.sticky === 1 ? ZEROS_TRANSLATE_3D : `${TRANSLATE_3D}(${data.config.isVertical ? 0 : data.measures.x}${PX}, ${data.config.isVertical ? data.measures.y : 0}${PX}, ${POSITION_ZERO})`;
634
677
  ;
635
678
  if (!data.config.isSnappingMethodAdvanced) {
636
679
  styles.position = POSITION_STICKY;
@@ -639,17 +682,20 @@ class NgVirtualListItemComponent extends BaseVirtualListItemComponent {
639
682
  else {
640
683
  styles.position = POSITION_ABSOLUTE;
641
684
  if (regular) {
642
- this._elementRef.nativeElement.setAttribute('position', '0');
643
- styles.transform = `${TRANSLATE_3D}(${data.config.isVertical ? 0 : data.measures.delta}${PX}, ${data.config.isVertical ? data.measures.delta : 0}${PX} , 0)`;
685
+ this._elementRef.nativeElement.setAttribute(POSITION, POSITION_ZERO);
686
+ styles.transform = `${TRANSLATE_3D}(${data.config.isVertical ? 0 : data.measures.delta}${PX}, ${data.config.isVertical ? data.measures.delta : 0}${PX}, ${POSITION_ZERO})`;
644
687
  }
645
688
  else {
646
- this._elementRef.nativeElement.setAttribute('position', `${data.config.isVertical ? data.measures.y : data.measures.x}`);
647
- styles.transform = `${TRANSLATE_3D}(${data.config.isVertical ? 0 : data.measures.x}${PX}, ${data.config.isVertical ? data.measures.y : 0}${PX} , 0)`;
689
+ this._elementRef.nativeElement.setAttribute(POSITION, `${data.config.isVertical ? data.measures.y : data.measures.x}`);
690
+ styles.transform = `${TRANSLATE_3D}(${data.config.isVertical ? 0 : data.measures.x}${PX}, ${data.config.isVertical ? data.measures.y : 0}${PX}, ${POSITION_ZERO})`;
648
691
  }
649
692
  }
650
693
  styles.height = data.config.isVertical ? data.config.dynamic ? SIZE_AUTO : `${data.measures.height}${PX}` : regular ? length : SIZE_100_PERSENT;
651
694
  styles.width = data.config.isVertical ? regular ? length : SIZE_100_PERSENT : data.config.dynamic ? SIZE_AUTO : `${data.measures.width}${PX}`;
652
695
  }
696
+ else {
697
+ this._elementRef.nativeElement.removeAttribute(ID);
698
+ }
653
699
  this._cdr.markForCheck();
654
700
  }
655
701
  updatePartStr(v, isSelected, isCollapsed) {
@@ -671,7 +717,7 @@ class NgVirtualListItemComponent extends BaseVirtualListItemComponent {
671
717
  if (v ? v.config.new : false) {
672
718
  part += PART_ITEM_NEW;
673
719
  }
674
- if (this.focus()) {
720
+ if (this.focused()) {
675
721
  part += PART_ITEM_FOCUSED;
676
722
  }
677
723
  this.part.set(part);
@@ -718,14 +764,14 @@ class NgVirtualListItemComponent extends BaseVirtualListItemComponent {
718
764
  this._service.itemClick(this._data);
719
765
  }
720
766
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgVirtualListItemComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
721
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: NgVirtualListItemComponent, selector: "ng-virtual-list-item", host: { attributes: { "role": "listitem" }, classAttribute: "ngvl__item" }, usesInheritance: true, ngImport: i0, template: "@if (data(); as item) {\r\n <div #listItem [part]=\"part()\" class=\"ngvl-item__container\" [attr.ng-vl-index]=\"item.config.tabIndex || -1\" tabindex=\"0\" [ngClass]=\"{'snapped': item.config.snapped,\r\n 'snapped-out': item.config.snappedOut, 'focus': focus()}\" (click)=\"onClickHandler()\">\r\n @if (itemRenderer(); as renderer) {\r\n <ng-container [ngTemplateOutlet]=\"renderer\"\r\n [ngTemplateOutletContext]=\"{data: item.data || {}, measures: measures(), config: config()}\" />\r\n }\r\n </div>\r\n}\r\n", styles: [":host{display:block;position:absolute;left:0;top:0;box-sizing:border-box;overflow:hidden}.ngvl-item__container{margin:0;padding:0;overflow:hidden;background-color:#fff;width:inherit;height:inherit;box-sizing:border-box}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
767
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: NgVirtualListItemComponent, selector: "ng-virtual-list-item", host: { attributes: { "role": "listitem" }, classAttribute: "ngvl__item" }, usesInheritance: true, ngImport: i0, template: "@if (data(); as item) {\r\n <div #listItem [part]=\"part()\" class=\"ngvl-item__container\" [attr.ng-vl-index]=\"item.config.tabIndex || -1\" tabindex=\"0\" [ngClass]=\"{'snapped': item.config.snapped,\r\n 'snapped-out': item.config.snappedOut, 'focus': focused()}\" (click)=\"onClickHandler()\">\r\n @if (itemRenderer(); as renderer) {\r\n <ng-container [ngTemplateOutlet]=\"renderer\"\r\n [ngTemplateOutletContext]=\"{data: item.data || {}, measures: measures(), config: config()}\" />\r\n }\r\n </div>\r\n}\r\n", styles: [":host{display:block;position:absolute;left:0;top:0;box-sizing:border-box;overflow:hidden}.ngvl-item__container{margin:0;padding:0;overflow:hidden;background-color:#fff;width:inherit;height:inherit;box-sizing:border-box}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
722
768
  }
723
769
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgVirtualListItemComponent, decorators: [{
724
770
  type: Component,
725
771
  args: [{ selector: 'ng-virtual-list-item', standalone: false, host: {
726
772
  'class': 'ngvl__item',
727
773
  'role': 'listitem',
728
- }, changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (data(); as item) {\r\n <div #listItem [part]=\"part()\" class=\"ngvl-item__container\" [attr.ng-vl-index]=\"item.config.tabIndex || -1\" tabindex=\"0\" [ngClass]=\"{'snapped': item.config.snapped,\r\n 'snapped-out': item.config.snappedOut, 'focus': focus()}\" (click)=\"onClickHandler()\">\r\n @if (itemRenderer(); as renderer) {\r\n <ng-container [ngTemplateOutlet]=\"renderer\"\r\n [ngTemplateOutletContext]=\"{data: item.data || {}, measures: measures(), config: config()}\" />\r\n }\r\n </div>\r\n}\r\n", styles: [":host{display:block;position:absolute;left:0;top:0;box-sizing:border-box;overflow:hidden}.ngvl-item__container{margin:0;padding:0;overflow:hidden;background-color:#fff;width:inherit;height:inherit;box-sizing:border-box}\n"] }]
774
+ }, changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (data(); as item) {\r\n <div #listItem [part]=\"part()\" class=\"ngvl-item__container\" [attr.ng-vl-index]=\"item.config.tabIndex || -1\" tabindex=\"0\" [ngClass]=\"{'snapped': item.config.snapped,\r\n 'snapped-out': item.config.snappedOut, 'focus': focused()}\" (click)=\"onClickHandler()\">\r\n @if (itemRenderer(); as renderer) {\r\n <ng-container [ngTemplateOutlet]=\"renderer\"\r\n [ngTemplateOutletContext]=\"{data: item.data || {}, measures: measures(), config: config()}\" />\r\n }\r\n </div>\r\n}\r\n", styles: [":host{display:block;position:absolute;left:0;top:0;box-sizing:border-box;overflow:hidden}.ngvl-item__container{margin:0;padding:0;overflow:hidden;background-color:#fff;width:inherit;height:inherit;box-sizing:border-box}\n"] }]
729
775
  }], ctorParameters: () => [] });
730
776
 
731
777
  /**
@@ -2165,7 +2211,7 @@ const isCollectionMode = (src, expected) => {
2165
2211
  return NORMAL_ALIASES.includes(src);
2166
2212
  };
2167
2213
 
2168
- const ROLE_LIST = 'list', ROLE_LIST_BOX = 'listbox';
2214
+ const ROLE_LIST = 'list', ROLE_LIST_BOX = 'listbox', ITEM_ID = 'item-id', ITEM_CONTAINER = 'ngvl-item__container';
2169
2215
  const validateScrollIteration = (value) => {
2170
2216
  return Number.isNaN(value) || (value < 0) ? 0 : value > MAX_SCROLL_TO_ITERATIONS ? MAX_SCROLL_TO_ITERATIONS : value;
2171
2217
  }, validateId = (id) => {
@@ -2183,6 +2229,35 @@ const validateScrollIteration = (value) => {
2183
2229
  if (!valid) {
2184
2230
  throw Error('The "iteration" parameter must be of type `number`.');
2185
2231
  }
2232
+ }, validateFocusAlignment = (align) => {
2233
+ const valid = validateString(align) && (align === 'none' || align === 'start' || align === 'center' || align === 'end');
2234
+ if (!valid) {
2235
+ throw Error('The "align" parameter must have the value `none`, `start`, `center` or `end`.');
2236
+ }
2237
+ };
2238
+ const formatScreenReaderMessage = (items, messagePattern, scrollSize, isVertical, bounds) => {
2239
+ if (!messagePattern) {
2240
+ return '';
2241
+ }
2242
+ const list = items ?? [], size = isVertical ? bounds.height : bounds.width;
2243
+ let start = Number.NaN, end = Number.NaN, prevItem;
2244
+ for (let i = 0, l = list.length; i < l; i++) {
2245
+ const item = list[i], position = isVertical ? item.measures.y : item.measures.x, itemSize = isVertical ? item.measures.height : item.measures.width;
2246
+ if (((position + itemSize) >= scrollSize) && Number.isNaN(start)) {
2247
+ start = item.index + 1;
2248
+ }
2249
+ if ((position >= (scrollSize + size)) && Number.isNaN(end) && prevItem) {
2250
+ end = prevItem.index + 1;
2251
+ }
2252
+ prevItem = item;
2253
+ }
2254
+ if (Number.isNaN(start) || Number.isNaN(end)) {
2255
+ return '';
2256
+ }
2257
+ let formatted = messagePattern ?? '';
2258
+ formatted = formatted.replace('$1', `${start}`);
2259
+ formatted = formatted.replace('$2', `${end}`);
2260
+ return formatted;
2186
2261
  };
2187
2262
  /**
2188
2263
  * Virtual list component.
@@ -2569,6 +2644,24 @@ class NgVirtualListComponent {
2569
2644
  * The name of the property by which tracking is performed
2570
2645
  */
2571
2646
  trackBy = input(TRACK_BY_PROPERTY_NAME, { ...this._trackByOptions });
2647
+ _screenReaderMessageOptions = {
2648
+ transform: (v) => {
2649
+ const valid = validateString(v);
2650
+ if (!valid) {
2651
+ console.error('The "screenReaderMessage" parameter must be of type `string`.');
2652
+ return DEFAULT_SCREEN_READER_MESSAGE;
2653
+ }
2654
+ return v;
2655
+ },
2656
+ };
2657
+ /**
2658
+ * Message for screen reader.
2659
+ * The message format is: "some text $1 some text $2",
2660
+ * where $1 is the number of the first element of the screen collection,
2661
+ * $2 is the number of the last element of the screen collection.
2662
+ */
2663
+ screenReaderMessage = input(DEFAULT_SCREEN_READER_MESSAGE, { ...this._screenReaderMessageOptions });
2664
+ screenReaderFormattedMessage = signal(this.screenReaderMessage());
2572
2665
  _isNotSelecting = this.getIsNotSelecting();
2573
2666
  get isNotSelecting() { return this._isNotSelecting; }
2574
2667
  _isSingleSelecting = this.getIsSingleSelecting();
@@ -2647,12 +2740,33 @@ class NgVirtualListComponent {
2647
2740
  this._scrollSize.set(actualScrollSize);
2648
2741
  }
2649
2742
  };
2650
- itemToFocus = (element, position) => {
2743
+ itemToFocus = (element, position, align = FocusAlignments.CENTER) => {
2651
2744
  const container = this._container()?.nativeElement;
2652
2745
  if (container) {
2653
- const { width, height } = this._bounds(), { width: elementWidth, height: elementHeight } = element.getBoundingClientRect(), isVertical = this._isVertical, pos = isVertical ? position - (height - elementHeight) * .5 : position - (width - elementWidth) * .5;
2654
- const params = { [this._isVertical ? TOP_PROP_NAME : LEFT_PROP_NAME]: pos, behavior: 'instant' };
2655
- container.scrollTo(params);
2746
+ const { width, height } = this._bounds(), { width: elementWidth, height: elementHeight } = element.getBoundingClientRect(), isVertical = this._isVertical;
2747
+ let pos = Number.NaN;
2748
+ switch (align) {
2749
+ case FocusAlignments.START: {
2750
+ pos = isVertical ? position : position;
2751
+ break;
2752
+ }
2753
+ case FocusAlignments.CENTER: {
2754
+ pos = isVertical ? position - (height - elementHeight) * .5 : position - (width - elementWidth) * .5;
2755
+ break;
2756
+ }
2757
+ case FocusAlignments.END: {
2758
+ pos = isVertical ? position - (height - elementHeight) : position - (width - elementWidth);
2759
+ break;
2760
+ }
2761
+ case FocusAlignments.NONE:
2762
+ default: {
2763
+ break;
2764
+ }
2765
+ }
2766
+ if (!Number.isNaN(pos)) {
2767
+ const params = { [this._isVertical ? TOP_PROP_NAME : LEFT_PROP_NAME]: pos, behavior: 'instant' };
2768
+ container.scrollTo(params);
2769
+ }
2656
2770
  }
2657
2771
  };
2658
2772
  _elementRef = inject((ElementRef));
@@ -2725,7 +2839,10 @@ class NgVirtualListComponent {
2725
2839
  $trackBy.pipe(takeUntilDestroyed(), tap(v => {
2726
2840
  this._trackBox.trackingPropertyName = v;
2727
2841
  })).subscribe();
2728
- const $bounds = toObservable(this._bounds).pipe(filter(b => !!b)), $items = toObservable(this.items).pipe(map(i => !i ? [] : i)), $scrollSize = toObservable(this._scrollSize), $itemSize = toObservable(this.itemSize).pipe(map(v => v <= 0 ? DEFAULT_ITEM_SIZE : v)), $bufferSize = toObservable(this.bufferSize).pipe(map(v => v < 0 ? DEFAULT_BUFFER_SIZE : v)), $maxBufferSize = toObservable(this.maxBufferSize).pipe(map(v => v < 0 ? DEFAULT_BUFFER_SIZE : v)), $itemConfigMap = toObservable(this.itemConfigMap).pipe(map(v => !v ? {} : v)), $snap = toObservable(this.snap), $isVertical = toObservable(this.direction).pipe(map(v => this.getIsVertical(v || DEFAULT_DIRECTION))), $isLazy = toObservable(this.collectionMode).pipe(map(v => this.getIsLazy(v || DEFAULT_COLLECTION_MODE))), $dynamicSize = toObservable(this.dynamicSize), $enabledBufferOptimization = toObservable(this.enabledBufferOptimization), $snappingMethod = toObservable(this.snappingMethod).pipe(map(v => this.getIsSnappingMethodAdvanced(v || DEFAULT_SNAPPING_METHOD))), $methodForSelecting = toObservable(this.methodForSelecting), $selectedIds = toObservable(this.selectedIds), $collapsedIds = toObservable(this.collapsedIds).pipe(map(v => Array.isArray(v) ? v : [])), $collapsedItemIds = toObservable(this._collapsedItemIds).pipe(map(v => Array.isArray(v) ? v : [])), $actualItems = toObservable(this._actualItems), $cacheVersion = toObservable(this._cacheVersion);
2842
+ const $bounds = toObservable(this._bounds).pipe(filter(b => !!b)), $items = toObservable(this.items).pipe(map(i => !i ? [] : i)), $scrollSize = toObservable(this._scrollSize), $itemSize = toObservable(this.itemSize).pipe(map(v => v <= 0 ? DEFAULT_ITEM_SIZE : v)), $bufferSize = toObservable(this.bufferSize).pipe(map(v => v < 0 ? DEFAULT_BUFFER_SIZE : v)), $maxBufferSize = toObservable(this.maxBufferSize).pipe(map(v => v < 0 ? DEFAULT_BUFFER_SIZE : v)), $itemConfigMap = toObservable(this.itemConfigMap).pipe(map(v => !v ? {} : v)), $snap = toObservable(this.snap), $isVertical = toObservable(this.direction).pipe(map(v => this.getIsVertical(v || DEFAULT_DIRECTION))), $isLazy = toObservable(this.collectionMode).pipe(map(v => this.getIsLazy(v || DEFAULT_COLLECTION_MODE))), $dynamicSize = toObservable(this.dynamicSize), $enabledBufferOptimization = toObservable(this.enabledBufferOptimization), $snappingMethod = toObservable(this.snappingMethod).pipe(map(v => this.getIsSnappingMethodAdvanced(v || DEFAULT_SNAPPING_METHOD))), $methodForSelecting = toObservable(this.methodForSelecting), $selectedIds = toObservable(this.selectedIds), $collapsedIds = toObservable(this.collapsedIds).pipe(map(v => Array.isArray(v) ? v : [])), $collapsedItemIds = toObservable(this._collapsedItemIds).pipe(map(v => Array.isArray(v) ? v : [])), $actualItems = toObservable(this._actualItems), $screenReaderMessage = toObservable(this.screenReaderMessage), $displayItems = this._service.$displayItems, $cacheVersion = toObservable(this._cacheVersion);
2843
+ combineLatest([$displayItems, $screenReaderMessage, $isVertical, $scrollSize, $bounds]).pipe(takeUntilDestroyed(), distinctUntilChanged(), debounceTime(100), tap(([items, screenReaderMessage, isVertical, scrollSize, bounds]) => {
2844
+ this.screenReaderFormattedMessage.set(formatScreenReaderMessage(items, screenReaderMessage, scrollSize, isVertical, bounds));
2845
+ })).subscribe();
2729
2846
  $isLazy.pipe(takeUntilDestroyed(), tap(v => {
2730
2847
  this._trackBox.isLazy = v;
2731
2848
  })).subscribe();
@@ -2978,6 +3095,20 @@ class NgVirtualListComponent {
2978
3095
  validateId(id);
2979
3096
  return this._trackBox.getItemBounds(id);
2980
3097
  }
3098
+ /**
3099
+ * Focus an list item by a given id.
3100
+ */
3101
+ focus(id, align = FocusAlignments.NONE) {
3102
+ validateId(id);
3103
+ validateFocusAlignment(align);
3104
+ const el = this._list()?.nativeElement.querySelector(`[${ITEM_ID}="${id}"]`);
3105
+ if (el) {
3106
+ const focusedEl = el.querySelector(`.${ITEM_CONTAINER}`);
3107
+ if (focusedEl) {
3108
+ this._service.focus(focusedEl, align);
3109
+ }
3110
+ }
3111
+ }
2981
3112
  /**
2982
3113
  * The method scrolls the list to the element with the given id and returns the value of the scrolled area.
2983
3114
  * Behavior accepts the values ​​"auto", "instant" and "smooth".
@@ -3139,13 +3270,13 @@ class NgVirtualListComponent {
3139
3270
  }
3140
3271
  }
3141
3272
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgVirtualListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
3142
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: NgVirtualListComponent, selector: "ng-virtual-list", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: true, transformFunction: null }, selectedIds: { classPropertyName: "selectedIds", publicName: "selectedIds", isSignal: true, isRequired: false, transformFunction: null }, collapsedIds: { classPropertyName: "collapsedIds", publicName: "collapsedIds", isSignal: true, isRequired: false, transformFunction: null }, selectByClick: { classPropertyName: "selectByClick", publicName: "selectByClick", isSignal: true, isRequired: false, transformFunction: null }, collapseByClick: { classPropertyName: "collapseByClick", publicName: "collapseByClick", isSignal: true, isRequired: false, transformFunction: null }, snap: { classPropertyName: "snap", publicName: "snap", isSignal: true, isRequired: false, transformFunction: null }, enabledBufferOptimization: { classPropertyName: "enabledBufferOptimization", publicName: "enabledBufferOptimization", isSignal: true, isRequired: false, transformFunction: null }, itemRenderer: { classPropertyName: "itemRenderer", publicName: "itemRenderer", isSignal: true, isRequired: true, transformFunction: null }, itemConfigMap: { classPropertyName: "itemConfigMap", publicName: "itemConfigMap", isSignal: true, isRequired: false, transformFunction: null }, itemSize: { classPropertyName: "itemSize", publicName: "itemSize", isSignal: true, isRequired: false, transformFunction: null }, dynamicSize: { classPropertyName: "dynamicSize", publicName: "dynamicSize", isSignal: true, isRequired: false, transformFunction: null }, direction: { classPropertyName: "direction", publicName: "direction", isSignal: true, isRequired: false, transformFunction: null }, collectionMode: { classPropertyName: "collectionMode", publicName: "collectionMode", isSignal: true, isRequired: false, transformFunction: null }, bufferSize: { classPropertyName: "bufferSize", publicName: "bufferSize", isSignal: true, isRequired: false, transformFunction: null }, maxBufferSize: { classPropertyName: "maxBufferSize", publicName: "maxBufferSize", isSignal: true, isRequired: false, transformFunction: null }, snappingMethod: { classPropertyName: "snappingMethod", publicName: "snappingMethod", isSignal: true, isRequired: false, transformFunction: null }, methodForSelecting: { classPropertyName: "methodForSelecting", publicName: "methodForSelecting", isSignal: true, isRequired: false, transformFunction: null }, trackBy: { classPropertyName: "trackBy", publicName: "trackBy", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { onScroll: "onScroll", onScrollEnd: "onScrollEnd", onViewportChange: "onViewportChange", onItemClick: "onItemClick", onSelect: "onSelect", onCollapse: "onCollapse", onScrollReachStart: "onScrollReachStart", onScrollReachEnd: "onScrollReachEnd" }, host: { styleAttribute: "position: relative;" }, providers: [NgVirtualListService], viewQueries: [{ propertyName: "_snappedContainer", first: true, predicate: ["snapped"], descendants: true, isSignal: true }, { propertyName: "_container", first: true, predicate: ["container"], descendants: true, isSignal: true }, { propertyName: "_list", first: true, predicate: ["list"], descendants: true, isSignal: true }, { propertyName: "_listContainerRef", first: true, predicate: ["renderersContainer"], descendants: true, read: ViewContainerRef }, { propertyName: "_snapContainerRef", first: true, predicate: ["snapRendererContainer"], descendants: true, read: ViewContainerRef }], ngImport: i0, template: "@if (snap()) {\r\n <div #snapped part=\"snapped-item\" class=\"ngvl__list-snapper\">\r\n <ng-container #snapRendererContainer></ng-container>\r\n </div>\r\n}\r\n<div #container part=\"scroller\" class=\"ngvl__scroller\">\r\n <div [attr.aria-orientation]=\"orientation\" [attr.aria-activedescendant]=\"focusedElement()\" #list part=\"list\" class=\"ngvl__list\">\r\n <ng-container #renderersContainer></ng-container>\r\n </div>\r\n</div>", styles: [":host{position:relative;display:block;width:400px;overflow:hidden}:host(.horizontal){height:48px}:host(.horizontal) .ngvl__list{display:inline-flex}:host(.horizontal) .ngvl__scroller{overflow:auto hidden}:host(.vertical) .ngvl__scroller{overflow:hidden auto}:host(.vertical){height:320px}.ngvl__scroller{overflow:auto;width:100%;height:100%}.ngvl__list-snapper{pointer-events:none;position:absolute;list-style:none;left:0;top:0;z-index:1}.ngvl__list{position:relative;list-style:none;padding:0;margin:0;width:100%;height:100%}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.ShadowDom });
3273
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: NgVirtualListComponent, selector: "ng-virtual-list", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: true, transformFunction: null }, selectedIds: { classPropertyName: "selectedIds", publicName: "selectedIds", isSignal: true, isRequired: false, transformFunction: null }, collapsedIds: { classPropertyName: "collapsedIds", publicName: "collapsedIds", isSignal: true, isRequired: false, transformFunction: null }, selectByClick: { classPropertyName: "selectByClick", publicName: "selectByClick", isSignal: true, isRequired: false, transformFunction: null }, collapseByClick: { classPropertyName: "collapseByClick", publicName: "collapseByClick", isSignal: true, isRequired: false, transformFunction: null }, snap: { classPropertyName: "snap", publicName: "snap", isSignal: true, isRequired: false, transformFunction: null }, enabledBufferOptimization: { classPropertyName: "enabledBufferOptimization", publicName: "enabledBufferOptimization", isSignal: true, isRequired: false, transformFunction: null }, itemRenderer: { classPropertyName: "itemRenderer", publicName: "itemRenderer", isSignal: true, isRequired: true, transformFunction: null }, itemConfigMap: { classPropertyName: "itemConfigMap", publicName: "itemConfigMap", isSignal: true, isRequired: false, transformFunction: null }, itemSize: { classPropertyName: "itemSize", publicName: "itemSize", isSignal: true, isRequired: false, transformFunction: null }, dynamicSize: { classPropertyName: "dynamicSize", publicName: "dynamicSize", isSignal: true, isRequired: false, transformFunction: null }, direction: { classPropertyName: "direction", publicName: "direction", isSignal: true, isRequired: false, transformFunction: null }, collectionMode: { classPropertyName: "collectionMode", publicName: "collectionMode", isSignal: true, isRequired: false, transformFunction: null }, bufferSize: { classPropertyName: "bufferSize", publicName: "bufferSize", isSignal: true, isRequired: false, transformFunction: null }, maxBufferSize: { classPropertyName: "maxBufferSize", publicName: "maxBufferSize", isSignal: true, isRequired: false, transformFunction: null }, snappingMethod: { classPropertyName: "snappingMethod", publicName: "snappingMethod", isSignal: true, isRequired: false, transformFunction: null }, methodForSelecting: { classPropertyName: "methodForSelecting", publicName: "methodForSelecting", isSignal: true, isRequired: false, transformFunction: null }, trackBy: { classPropertyName: "trackBy", publicName: "trackBy", isSignal: true, isRequired: false, transformFunction: null }, screenReaderMessage: { classPropertyName: "screenReaderMessage", publicName: "screenReaderMessage", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { onScroll: "onScroll", onScrollEnd: "onScrollEnd", onViewportChange: "onViewportChange", onItemClick: "onItemClick", onSelect: "onSelect", onCollapse: "onCollapse", onScrollReachStart: "onScrollReachStart", onScrollReachEnd: "onScrollReachEnd" }, host: { styleAttribute: "position: relative;" }, providers: [NgVirtualListService], viewQueries: [{ propertyName: "_snappedContainer", first: true, predicate: ["snapped"], descendants: true, isSignal: true }, { propertyName: "_container", first: true, predicate: ["container"], descendants: true, isSignal: true }, { propertyName: "_list", first: true, predicate: ["list"], descendants: true, isSignal: true }, { propertyName: "_listContainerRef", first: true, predicate: ["renderersContainer"], descendants: true, read: ViewContainerRef }, { propertyName: "_snapContainerRef", first: true, predicate: ["snapRendererContainer"], descendants: true, read: ViewContainerRef }], ngImport: i0, template: "<div aria-live=\"polite\" aria-atomic=\"true\" class=\"ngvl__screen-reader\">\r\n {{ screenReaderFormattedMessage() }}\r\n</div>\r\n\r\n@if (snap()) {\r\n <div #snapped part=\"snapped-item\" class=\"ngvl__list-snapper\">\r\n <ng-container #snapRendererContainer></ng-container>\r\n </div>\r\n}\r\n<div #container part=\"scroller\" class=\"ngvl__scroller\">\r\n <div [attr.aria-orientation]=\"orientation\" [attr.aria-activedescendant]=\"focusedElement()\" tabindex=\"0\" #list\r\n part=\"list\" class=\"ngvl__list\">\r\n <ng-container #renderersContainer></ng-container>\r\n </div>\r\n</div>", styles: [":host{position:relative;display:block;width:400px;overflow:hidden}:host(.horizontal){height:48px}:host(.horizontal) .ngvl__list{display:inline-flex}:host(.horizontal) .ngvl__scroller{overflow:auto hidden}:host(.vertical) .ngvl__scroller{overflow:hidden auto}:host(.vertical){height:320px}.ngvl__scroller{overflow:auto;width:100%;height:100%}.ngvl__list-snapper{pointer-events:none;position:absolute;list-style:none;left:0;top:0;z-index:1}.ngvl__list{position:relative;list-style:none;padding:0;margin:0;width:100%;height:100%}.ngvl__screen-reader{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.ShadowDom });
3143
3274
  }
3144
3275
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgVirtualListComponent, decorators: [{
3145
3276
  type: Component,
3146
3277
  args: [{ selector: 'ng-virtual-list', standalone: false, host: {
3147
3278
  'style': 'position: relative;'
3148
- }, changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.ShadowDom, providers: [NgVirtualListService], template: "@if (snap()) {\r\n <div #snapped part=\"snapped-item\" class=\"ngvl__list-snapper\">\r\n <ng-container #snapRendererContainer></ng-container>\r\n </div>\r\n}\r\n<div #container part=\"scroller\" class=\"ngvl__scroller\">\r\n <div [attr.aria-orientation]=\"orientation\" [attr.aria-activedescendant]=\"focusedElement()\" #list part=\"list\" class=\"ngvl__list\">\r\n <ng-container #renderersContainer></ng-container>\r\n </div>\r\n</div>", styles: [":host{position:relative;display:block;width:400px;overflow:hidden}:host(.horizontal){height:48px}:host(.horizontal) .ngvl__list{display:inline-flex}:host(.horizontal) .ngvl__scroller{overflow:auto hidden}:host(.vertical) .ngvl__scroller{overflow:hidden auto}:host(.vertical){height:320px}.ngvl__scroller{overflow:auto;width:100%;height:100%}.ngvl__list-snapper{pointer-events:none;position:absolute;list-style:none;left:0;top:0;z-index:1}.ngvl__list{position:relative;list-style:none;padding:0;margin:0;width:100%;height:100%}\n"] }]
3279
+ }, changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.ShadowDom, providers: [NgVirtualListService], template: "<div aria-live=\"polite\" aria-atomic=\"true\" class=\"ngvl__screen-reader\">\r\n {{ screenReaderFormattedMessage() }}\r\n</div>\r\n\r\n@if (snap()) {\r\n <div #snapped part=\"snapped-item\" class=\"ngvl__list-snapper\">\r\n <ng-container #snapRendererContainer></ng-container>\r\n </div>\r\n}\r\n<div #container part=\"scroller\" class=\"ngvl__scroller\">\r\n <div [attr.aria-orientation]=\"orientation\" [attr.aria-activedescendant]=\"focusedElement()\" tabindex=\"0\" #list\r\n part=\"list\" class=\"ngvl__list\">\r\n <ng-container #renderersContainer></ng-container>\r\n </div>\r\n</div>", styles: [":host{position:relative;display:block;width:400px;overflow:hidden}:host(.horizontal){height:48px}:host(.horizontal) .ngvl__list{display:inline-flex}:host(.horizontal) .ngvl__scroller{overflow:auto hidden}:host(.vertical) .ngvl__scroller{overflow:hidden auto}:host(.vertical){height:320px}.ngvl__scroller{overflow:auto;width:100%;height:100%}.ngvl__list-snapper{pointer-events:none;position:absolute;list-style:none;left:0;top:0;z-index:1}.ngvl__list{position:relative;list-style:none;padding:0;margin:0;width:100%;height:100%}.ngvl__screen-reader{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}\n"] }]
3149
3280
  }], ctorParameters: () => [], propDecorators: { _listContainerRef: [{
3150
3281
  type: ViewChild,
3151
3282
  args: ['renderersContainer', { read: ViewContainerRef }]
@@ -3177,5 +3308,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
3177
3308
  * Generated bundle index. Do not edit.
3178
3309
  */
3179
3310
 
3180
- export { CollectionModes, Directions, MethodsForSelecting, NgVirtualListComponent, NgVirtualListItemComponent, NgVirtualListModule, ScrollEvent, SnappingMethods, debounce, toggleClassName };
3311
+ export { CollectionModes, Directions, FocusAlignments, MethodsForSelecting, NgVirtualListComponent, NgVirtualListItemComponent, NgVirtualListModule, ScrollEvent, SnappingMethods, debounce, toggleClassName };
3181
3312
  //# sourceMappingURL=ng-virtual-list.mjs.map