@clr/angular
Version:
Angular components for Clarity
177 lines • 17.8 kB
JavaScript
/*
* Copyright (c) 2016-2025 Broadcom. All Rights Reserved.
* The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries.
* This software is released under MIT license.
* The full license information can be found in LICENSE in the root directory of this project.
*/
import { Injectable } from '@angular/core';
import { Subject } from 'rxjs';
import * as i0 from "@angular/core";
import * as i1 from "./filters";
import * as i2 from "./sort";
import * as i3 from "./page";
export class Items {
constructor(_filters, _sort, _page) {
this._filters = _filters;
this._sort = _sort;
this._page = _page;
/**
* Indicates if the data is currently loading
*/
this.loading = false;
/**
* Whether we should use smart items for this datagrid or let the user handle
* everything.
*/
this._smart = false;
/**
* List of items currently displayed
*/
this._displayed = [];
/**
* The Observable that lets other classes subscribe to items changes
*/
this._change = new Subject();
this._allChanges = new Subject();
/**
* Tracking function to identify objects.
*/
this.trackBy = item => item;
}
get smart() {
return this._smart;
}
get all() {
return this._all;
}
set all(items) {
this._all = items;
this.emitAllChanges(items);
if (this.smart) {
this._filterItems();
}
else {
this._displayed = items;
this.emitChange();
}
}
get displayed() {
// Ideally we could return an immutable array, but we don't have it in Clarity yet.
return this._displayed;
}
// We do not want to expose the Subject itself, but the Observable which is read-only
get change() {
return this._change.asObservable();
}
get allChanges() {
return this._allChanges.asObservable();
}
/**
* Checks if we don't have data to process yet, to abort early operations
*/
get uninitialized() {
return !this._all;
}
/**
* Cleans up our subscriptions to other providers
*/
destroy() {
if (this._filtersSub) {
this._filtersSub.unsubscribe();
}
if (this._sortSub) {
this._sortSub.unsubscribe();
}
if (this._pageSub) {
this._pageSub.unsubscribe();
}
}
smartenDown() {
this._smart = false;
this.destroy();
}
smartenUp() {
this._smart = true;
/*
* These observers trigger a chain of function: filter -> sort -> paginate
* An observer up the chain re-triggers all the operations that follow it.
*/
this._filtersSub = this._filters.change.subscribe(() => this._filterItems());
this._sortSub = this._sort.change.subscribe(() => {
// Special case, if the datagrid went from sorted to unsorted, we have to re-filter
// to get the original order back
if (!this._sort.comparator) {
this._filterItems();
}
else {
this._sortItems();
}
});
this._pageSub = this._page.change.subscribe(() => this._changePage());
}
/**
* Manually recompute the list of displayed items
*/
refresh() {
if (this.smart) {
this._filterItems();
}
}
emitChange() {
this._change.next(this.displayed);
}
emitAllChanges(items) {
this._allChanges.next(items);
}
/**
* FiltersProvider items from the raw list
*/
_filterItems() {
if (this.uninitialized) {
return;
}
if (this._filters.hasActiveFilters()) {
this._filtered = this._all.filter(item => this._filters.accepts(item));
}
else {
// Work on a shallow copy of the array, to not modify the user's model
this._filtered = this._all.slice();
}
this._page.totalItems = this._filtered.length;
this._sortItems();
}
/**
* Sorts items in the filtered list
*/
_sortItems() {
if (this.uninitialized) {
return;
}
if (this._sort.comparator) {
this._filtered.sort((a, b) => this._sort.compare(a, b));
}
this._changePage();
}
/**
* Extracts the current page from the sorted list
*/
_changePage() {
// If we know we have pagination but the page size hasn't been set yet, we wait for it.
if (this.uninitialized || (this._page.activated && this._page.size === 0)) {
return;
}
if (this._page.size > 0) {
this._displayed = this._filtered.slice(this._page.firstItem, this._page.lastItem + 1);
}
else {
this._displayed = this._filtered;
}
this.emitChange();
}
}
Items.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: Items, deps: [{ token: i1.FiltersProvider }, { token: i2.Sort }, { token: i3.Page }], target: i0.ɵɵFactoryTarget.Injectable });
Items.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: Items });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: Items, decorators: [{
type: Injectable
}], ctorParameters: function () { return [{ type: i1.FiltersProvider }, { type: i2.Sort }, { type: i3.Page }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"items.js","sourceRoot":"","sources":["../../../../../../projects/angular/src/data/datagrid/providers/items.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2016-2025 Broadcom. All Rights Reserved.\n * The term \"Broadcom\" refers to Broadcom Inc. and/or its subsidiaries.\n * This software is released under MIT license.\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport { Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\nimport { Subject } from 'rxjs';\nimport { Subscription } from 'rxjs';\n\nimport { FiltersProvider } from './filters';\nimport { Page } from './page';\nimport { Sort } from './sort';\n\nexport type ClrDatagridItemsTrackByFunction<T> = (item: T) => any;\n\n@Injectable()\nexport class Items<T = any> {\n  /**\n   * Indicates if the data is currently loading\n   */\n  loading = false;\n\n  /**\n   * Subscriptions to the other providers changes.\n   */\n  private _filtersSub: Subscription;\n  private _sortSub: Subscription;\n  private _pageSub: Subscription;\n\n  /**\n   * Whether we should use smart items for this datagrid or let the user handle\n   * everything.\n   */\n  private _smart = false;\n\n  /**\n   * List of all items in the datagrid\n   */\n  private _all: T[];\n\n  /**\n   * Internal temporary step, which we preserve to avoid re-filtering or re-sorting if not necessary\n   */\n  private _filtered: T[];\n\n  /**\n   * List of items currently displayed\n   */\n  private _displayed: T[] = [];\n\n  /**\n   * The Observable that lets other classes subscribe to items changes\n   */\n  private _change = new Subject<T[]>();\n\n  private _allChanges = new Subject<T[]>();\n\n  constructor(private _filters: FiltersProvider<T>, private _sort: Sort<T>, private _page: Page) {}\n\n  get smart(): boolean {\n    return this._smart;\n  }\n\n  get all() {\n    return this._all;\n  }\n  set all(items: T[]) {\n    this._all = items;\n    this.emitAllChanges(items);\n    if (this.smart) {\n      this._filterItems();\n    } else {\n      this._displayed = items;\n      this.emitChange();\n    }\n  }\n\n  get displayed(): T[] {\n    // Ideally we could return an immutable array, but we don't have it in Clarity yet.\n    return this._displayed;\n  }\n\n  // We do not want to expose the Subject itself, but the Observable which is read-only\n  get change(): Observable<T[]> {\n    return this._change.asObservable();\n  }\n\n  get allChanges(): Observable<T[]> {\n    return this._allChanges.asObservable();\n  }\n\n  /**\n   * Checks if we don't have data to process yet, to abort early operations\n   */\n  private get uninitialized() {\n    return !this._all;\n  }\n\n  /**\n   * Tracking function to identify objects.\n   */\n  trackBy: ClrDatagridItemsTrackByFunction<T> = item => item;\n\n  /**\n   * Cleans up our subscriptions to other providers\n   */\n  destroy() {\n    if (this._filtersSub) {\n      this._filtersSub.unsubscribe();\n    }\n    if (this._sortSub) {\n      this._sortSub.unsubscribe();\n    }\n    if (this._pageSub) {\n      this._pageSub.unsubscribe();\n    }\n  }\n\n  smartenDown() {\n    this._smart = false;\n\n    this.destroy();\n  }\n\n  smartenUp() {\n    this._smart = true;\n    /*\n     * These observers trigger a chain of function: filter -> sort -> paginate\n     * An observer up the chain re-triggers all the operations that follow it.\n     */\n    this._filtersSub = this._filters.change.subscribe(() => this._filterItems());\n    this._sortSub = this._sort.change.subscribe(() => {\n      // Special case, if the datagrid went from sorted to unsorted, we have to re-filter\n      // to get the original order back\n      if (!this._sort.comparator) {\n        this._filterItems();\n      } else {\n        this._sortItems();\n      }\n    });\n    this._pageSub = this._page.change.subscribe(() => this._changePage());\n  }\n\n  /**\n   * Manually recompute the list of displayed items\n   */\n  refresh() {\n    if (this.smart) {\n      this._filterItems();\n    }\n  }\n\n  private emitChange() {\n    this._change.next(this.displayed);\n  }\n\n  private emitAllChanges(items: T[]): void {\n    this._allChanges.next(items);\n  }\n\n  /**\n   * FiltersProvider items from the raw list\n   */\n  private _filterItems() {\n    if (this.uninitialized) {\n      return;\n    }\n    if (this._filters.hasActiveFilters()) {\n      this._filtered = this._all.filter(item => this._filters.accepts(item));\n    } else {\n      // Work on a shallow copy of the array, to not modify the user's model\n      this._filtered = this._all.slice();\n    }\n    this._page.totalItems = this._filtered.length;\n    this._sortItems();\n  }\n\n  /**\n   * Sorts items in the filtered list\n   */\n  private _sortItems() {\n    if (this.uninitialized) {\n      return;\n    }\n    if (this._sort.comparator) {\n      this._filtered.sort((a, b) => this._sort.compare(a, b));\n    }\n    this._changePage();\n  }\n\n  /**\n   * Extracts the current page from the sorted list\n   */\n  private _changePage() {\n    // If we know we have pagination but the page size hasn't been set yet, we wait for it.\n    if (this.uninitialized || (this._page.activated && this._page.size === 0)) {\n      return;\n    }\n    if (this._page.size > 0) {\n      this._displayed = this._filtered.slice(this._page.firstItem, this._page.lastItem + 1);\n    } else {\n      this._displayed = this._filtered;\n    }\n    this.emitChange();\n  }\n}\n"]}