UNPKG

@cisstech/nge

Version:

NG Essentials is a collection of libraries for Angular developers.

117 lines 27.6 kB
import { Component, ContentChildren, EventEmitter, Input, Output, } from '@angular/core'; import { ListTemplateComponent } from './list-template.component'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; import * as i2 from "@angular/forms"; import * as i3 from "ngx-pipes"; export class ListComponent { constructor() { this.items = []; this.selectable = false; this.filterBy = []; this.selections = []; this.selectionsChange = new EventEmitter(); this._selectionStates = {}; this._noResultTemplate = null; this._emptyStateTemplate = null; this._empty = false; } get hasSelection() { return !!this.selections.length; } get classes() { if (!this.containerClass) { return {}; } return { [this.containerClass]: true, }; } ngOnChanges() { this._empty = !this.items?.length; setTimeout(() => { this.checkSelections(); }, 300); } ngAfterContentInit() { this._noResultTemplate = this.templates.find((e) => e.slot === 'noresult')?.template || null; this._emptyStateTemplate = this.templates.find((e) => e.slot === 'empty')?.template || null; } unselect(item) { const id = item[this.idField]; this.selections = this.selections.filter((e) => e !== item); this._selectionStates[id] = false; this.selectionsChange.emit(this.selections); } _trackBy(index, item) { if (this.trackBy) { return item[this.trackBy] ?? index; } return index; } _template(context, slot) { return (this.templates.find((e) => { if (e.slot === slot && e.when) { return e.when(context); } return false; })?.template || this.templates.find((e) => e.slot === slot && !e.when)?.template || null); } _isSelected(item) { return !!this.selections.find((e) => e === item); } _toggleSelection(item) { for (let i = 0; i < this.selections.length; i++) { if (this.equals(this.selections[i], item)) { this.selections.splice(i, 1); this.selectionsChange.emit(this.selections); return; } } this.selections.push(item); this.selectionsChange.emit(this.selections); } equals(a, b) { return a[this.idField] === b[this.idField]; } checkSelections() { this.selections = this.selections.filter((selection) => { if (this.items.find((item) => this.equals(item, selection))) { return true; } delete this._selectionStates[selection[this.idField]]; return false; }); this.selectionsChange.emit(this.selections); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: ListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.1", type: ListComponent, selector: "ui-list", inputs: { idField: "idField", items: "items", trackBy: "trackBy", selectable: "selectable", filter: "filter", filterBy: "filterBy", selections: "selections", containerClass: "containerClass" }, outputs: { selectionsChange: "selectionsChange" }, queries: [{ propertyName: "templates", predicate: ListTemplateComponent }], usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"selectable\">\n <ng-container *ngFor=\"let item of selections; trackBy: _trackBy.bind(this)\">\n <ng-container *ngTemplateOutlet=\"_template(item, 'selection'); context: {item: item}\"></ng-container>\n </ng-container>\n</ng-container>\n\n<!-- TEMPLATE FOR EMPTY STATE -->\n<ng-template #emptyStateTemplate>\n <!-- SHOW EMPTY STATE TEMPLATE IF FILTER IS DEFINED -->\n <ng-container *ngIf=\"!filter?.trim(); else: noResult\">\n <ng-container *ngTemplateOutlet=\"_emptyStateTemplate\"></ng-container>\n </ng-container>\n <!-- SHOW NO RESULT TEMPLATE IF FILTER IS DEFINED -->\n <ng-template #noResult>\n <ng-container *ngTemplateOutlet=\"_noResultTemplate; context: { $implicit: filter }\"></ng-container>\n </ng-template>\n</ng-template>\n\n<!-- APPLY CURRENT FILTER TO ITEMS IF NEEDED AND STORE THE RESULT TO A VARIABLE `filtered` -->\n<ng-container *ngIf=\"(filter ? (items | filterBy: filterBy: filter) : items) as filtered\">\n <!-- SHOW EMPTY STATE IF NO ITEM ELSE SHOW ROWS -->\n <ng-container *ngIf=\"filtered.length; else: emptyStateTemplate\">\n <ul class=\"ui-list-container\" [ngClass]=\"classes\">\n <ng-container *ngFor=\"let item of filtered; trackBy: _trackBy.bind(this) let index=index; let first=first; let last=last; let even=even; let odd=odd\">\n <ng-container *ngIf=\"{item: item, items: filtered, odd: odd, last: last, even: even, index: index, first: first} as context\">\n <div class=\"ui-list-item\">\n <!-- HEADER -->\n <ng-container *ngIf=\"_template(context, 'header') as header\">\n <li class=\"ui-list-item-header\">\n <ng-container *ngTemplateOutlet=\"header; context: context\"></ng-container>\n </li>\n </ng-container>\n <!-- ROW -->\n <ng-container *ngIf=\"_template(context, 'row') as row\">\n <li class=\"ui-list-item-content\" [class.selectable]=\"selectable\">\n <ng-container *ngTemplateOutlet=\"row; context: context\"></ng-container>\n <!-- CHECKBOX -->\n <label class=\"ui-list-label\" *ngIf=\"selectable\">\n <input\n class=\"ui-list-label__checkbox\"\n type=\"checkbox\"\n [(ngModel)]=\"_selectionStates[$any(item)[idField]]\"\n (ngModelChange)=\"_toggleSelection(item)\"/>\n <span class=\"ui-list-label__text\">\n <span class=\"ui-list-label__check\">\n <span class=\"ui-list-label__icon\">\u2713</span>\n </span>\n </span>\n </label>\n </li>\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n </ul>\n </ng-container>\n</ng-container>\n", styles: [":host{display:block;min-height:100%;height:auto;width:100%;position:relative;--inner-margin-bottom: 12px}ul,li{margin:0;padding:0;list-style:none}li{display:block}li.selectable{display:grid;grid-template-columns:90% 10%;column-gap:12px;align-items:center}.ui-list-item{margin-bottom:var(--inner-margin-bottom)}.ui-list-label__checkbox{display:none}.ui-list-label__check{display:inline-block;border-radius:50%;border:5px solid rgba(0,0,0,.1);background:#fff;vertical-align:middle;margin-right:20px;width:2em;height:2em;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:border .3s ease}.ui-list-label__check .label__icon{opacity:.2;color:transparent;transition:opacity .3s .1s ease;-webkit-text-stroke:3px rgba(0,0,0,.5)}.ui-list-label__check:hover{border:5px solid rgba(0,0,0,.2)}.ui-list-label__checkbox:checked+.ui-list-label__text .ui-list-label__check{animation:check .5s cubic-bezier(.895,.03,.685,.22) forwards}.ui-list-label__checkbox:checked+.ui-list-label__text .ui-list-label__check .ui-list-label__icon{opacity:1;transform:scale(0);color:#fff;-webkit-text-stroke:0;animation:icon .3s cubic-bezier(1,.008,.565,1.65) .1s 1 forwards}.spacer{flex:1}@keyframes icon{0%{opacity:0;transform:scale(.3)}to{opacity:1;transform:scale(1)}}@keyframes check{0%{width:1.5em;height:1.5em;border-width:5px}10%{width:1.5em;height:1.5em;opacity:.1;background:#0003;border-width:15px}12%{width:1.5em;height:1.5em;opacity:.4;background:#0000001a;border-width:0}50%{width:2em;height:2em;background:#00d478;border:0;opacity:.6}to{width:2em;height:2em;background:#00d478;border:0;opacity:1}}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "pipe", type: i3.FilterByPipe, name: "filterBy" }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.1", ngImport: i0, type: ListComponent, decorators: [{ type: Component, args: [{ selector: 'ui-list', template: "<ng-container *ngIf=\"selectable\">\n <ng-container *ngFor=\"let item of selections; trackBy: _trackBy.bind(this)\">\n <ng-container *ngTemplateOutlet=\"_template(item, 'selection'); context: {item: item}\"></ng-container>\n </ng-container>\n</ng-container>\n\n<!-- TEMPLATE FOR EMPTY STATE -->\n<ng-template #emptyStateTemplate>\n <!-- SHOW EMPTY STATE TEMPLATE IF FILTER IS DEFINED -->\n <ng-container *ngIf=\"!filter?.trim(); else: noResult\">\n <ng-container *ngTemplateOutlet=\"_emptyStateTemplate\"></ng-container>\n </ng-container>\n <!-- SHOW NO RESULT TEMPLATE IF FILTER IS DEFINED -->\n <ng-template #noResult>\n <ng-container *ngTemplateOutlet=\"_noResultTemplate; context: { $implicit: filter }\"></ng-container>\n </ng-template>\n</ng-template>\n\n<!-- APPLY CURRENT FILTER TO ITEMS IF NEEDED AND STORE THE RESULT TO A VARIABLE `filtered` -->\n<ng-container *ngIf=\"(filter ? (items | filterBy: filterBy: filter) : items) as filtered\">\n <!-- SHOW EMPTY STATE IF NO ITEM ELSE SHOW ROWS -->\n <ng-container *ngIf=\"filtered.length; else: emptyStateTemplate\">\n <ul class=\"ui-list-container\" [ngClass]=\"classes\">\n <ng-container *ngFor=\"let item of filtered; trackBy: _trackBy.bind(this) let index=index; let first=first; let last=last; let even=even; let odd=odd\">\n <ng-container *ngIf=\"{item: item, items: filtered, odd: odd, last: last, even: even, index: index, first: first} as context\">\n <div class=\"ui-list-item\">\n <!-- HEADER -->\n <ng-container *ngIf=\"_template(context, 'header') as header\">\n <li class=\"ui-list-item-header\">\n <ng-container *ngTemplateOutlet=\"header; context: context\"></ng-container>\n </li>\n </ng-container>\n <!-- ROW -->\n <ng-container *ngIf=\"_template(context, 'row') as row\">\n <li class=\"ui-list-item-content\" [class.selectable]=\"selectable\">\n <ng-container *ngTemplateOutlet=\"row; context: context\"></ng-container>\n <!-- CHECKBOX -->\n <label class=\"ui-list-label\" *ngIf=\"selectable\">\n <input\n class=\"ui-list-label__checkbox\"\n type=\"checkbox\"\n [(ngModel)]=\"_selectionStates[$any(item)[idField]]\"\n (ngModelChange)=\"_toggleSelection(item)\"/>\n <span class=\"ui-list-label__text\">\n <span class=\"ui-list-label__check\">\n <span class=\"ui-list-label__icon\">\u2713</span>\n </span>\n </span>\n </label>\n </li>\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n </ul>\n </ng-container>\n</ng-container>\n", styles: [":host{display:block;min-height:100%;height:auto;width:100%;position:relative;--inner-margin-bottom: 12px}ul,li{margin:0;padding:0;list-style:none}li{display:block}li.selectable{display:grid;grid-template-columns:90% 10%;column-gap:12px;align-items:center}.ui-list-item{margin-bottom:var(--inner-margin-bottom)}.ui-list-label__checkbox{display:none}.ui-list-label__check{display:inline-block;border-radius:50%;border:5px solid rgba(0,0,0,.1);background:#fff;vertical-align:middle;margin-right:20px;width:2em;height:2em;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:border .3s ease}.ui-list-label__check .label__icon{opacity:.2;color:transparent;transition:opacity .3s .1s ease;-webkit-text-stroke:3px rgba(0,0,0,.5)}.ui-list-label__check:hover{border:5px solid rgba(0,0,0,.2)}.ui-list-label__checkbox:checked+.ui-list-label__text .ui-list-label__check{animation:check .5s cubic-bezier(.895,.03,.685,.22) forwards}.ui-list-label__checkbox:checked+.ui-list-label__text .ui-list-label__check .ui-list-label__icon{opacity:1;transform:scale(0);color:#fff;-webkit-text-stroke:0;animation:icon .3s cubic-bezier(1,.008,.565,1.65) .1s 1 forwards}.spacer{flex:1}@keyframes icon{0%{opacity:0;transform:scale(.3)}to{opacity:1;transform:scale(1)}}@keyframes check{0%{width:1.5em;height:1.5em;border-width:5px}10%{width:1.5em;height:1.5em;opacity:.1;background:#0003;border-width:15px}12%{width:1.5em;height:1.5em;opacity:.4;background:#0000001a;border-width:0}50%{width:2em;height:2em;background:#00d478;border:0;opacity:.6}to{width:2em;height:2em;background:#00d478;border:0;opacity:1}}\n"] }] }], propDecorators: { templates: [{ type: ContentChildren, args: [ListTemplateComponent] }], idField: [{ type: Input }], items: [{ type: Input }], trackBy: [{ type: Input }], selectable: [{ type: Input }], filter: [{ type: Input }], filterBy: [{ type: Input }], selections: [{ type: Input }], containerClass: [{ type: Input }], selectionsChange: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"list.component.js","sourceRoot":"","sources":["../../../../../../projects/nge/ui/list/src/list.component.ts","../../../../../../projects/nge/ui/list/src/list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  Component,\n  ContentChildren,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  QueryList,\n  TemplateRef,\n} from '@angular/core'\nimport { ListContext, ListTemplateSlots } from './list'\nimport { ListTemplateComponent } from './list-template.component'\n\n@Component({\n  selector: 'ui-list',\n  templateUrl: './list.component.html',\n  styleUrls: ['./list.component.scss'],\n})\nexport class ListComponent<T> implements OnChanges, AfterContentInit {\n  @ContentChildren(ListTemplateComponent)\n  templates!: QueryList<ListTemplateComponent<T>>\n\n  @Input()\n  idField!: string\n\n  @Input()\n  items: T[] = []\n\n  @Input()\n  trackBy?: string\n\n  @Input()\n  selectable = false\n\n  @Input()\n  filter?: string\n\n  @Input()\n  filterBy: string[] = []\n\n  @Input()\n  selections: T[] = []\n\n  @Input()\n  containerClass?: string\n\n  @Output()\n  selectionsChange = new EventEmitter<T[]>()\n\n  _selectionStates: Record<string, boolean> = {}\n  _noResultTemplate: TemplateRef<any> | null = null\n  _emptyStateTemplate: TemplateRef<any> | null = null\n\n  _empty = false\n\n  get hasSelection() {\n    return !!this.selections.length\n  }\n\n  protected get classes() {\n    if (!this.containerClass) {\n      return {}\n    }\n    return {\n      [this.containerClass]: true,\n    }\n  }\n\n  ngOnChanges() {\n    this._empty = !this.items?.length\n    setTimeout(() => {\n      this.checkSelections()\n    }, 300)\n  }\n\n  ngAfterContentInit() {\n    this._noResultTemplate = this.templates.find((e) => e.slot === 'noresult')?.template || null\n    this._emptyStateTemplate = this.templates.find((e) => e.slot === 'empty')?.template || null\n  }\n\n  unselect(item: T) {\n    const id = (item as any)[this.idField]\n    this.selections = this.selections.filter((e) => e !== item)\n    this._selectionStates[id] = false\n    this.selectionsChange.emit(this.selections)\n  }\n\n  _trackBy(index: number, item: any): any {\n    if (this.trackBy) {\n      return item[this.trackBy] ?? index\n    }\n    return index\n  }\n\n  _template(context: T | ListContext<T>, slot: ListTemplateSlots): TemplateRef<any> | null {\n    return (\n      this.templates.find((e) => {\n        if (e.slot === slot && e.when) {\n          return e.when(context)\n        }\n        return false\n      })?.template ||\n      this.templates.find((e) => e.slot === slot && !e.when)?.template ||\n      null\n    )\n  }\n\n  _isSelected(item: T): boolean {\n    return !!this.selections.find((e) => e === item)\n  }\n\n  _toggleSelection(item: T): void {\n    for (let i = 0; i < this.selections.length; i++) {\n      if (this.equals(this.selections[i], item)) {\n        this.selections.splice(i, 1)\n        this.selectionsChange.emit(this.selections)\n        return\n      }\n    }\n    this.selections.push(item)\n    this.selectionsChange.emit(this.selections)\n  }\n\n  private equals(a: any, b: any) {\n    return a[this.idField] === b[this.idField]\n  }\n\n  private checkSelections() {\n    this.selections = this.selections.filter((selection: any) => {\n      if (this.items.find((item) => this.equals(item, selection))) {\n        return true\n      }\n      delete this._selectionStates[selection[this.idField]]\n      return false\n    })\n    this.selectionsChange.emit(this.selections)\n  }\n}\n","<ng-container *ngIf=\"selectable\">\n  <ng-container *ngFor=\"let item of selections; trackBy: _trackBy.bind(this)\">\n      <ng-container *ngTemplateOutlet=\"_template(item, 'selection'); context: {item: item}\"></ng-container>\n  </ng-container>\n</ng-container>\n\n<!-- TEMPLATE FOR EMPTY STATE -->\n<ng-template #emptyStateTemplate>\n  <!-- SHOW EMPTY STATE TEMPLATE IF FILTER IS DEFINED -->\n  <ng-container *ngIf=\"!filter?.trim(); else: noResult\">\n    <ng-container *ngTemplateOutlet=\"_emptyStateTemplate\"></ng-container>\n  </ng-container>\n  <!-- SHOW NO RESULT TEMPLATE IF FILTER IS DEFINED -->\n  <ng-template #noResult>\n    <ng-container *ngTemplateOutlet=\"_noResultTemplate; context: { $implicit: filter }\"></ng-container>\n  </ng-template>\n</ng-template>\n\n<!-- APPLY CURRENT FILTER TO ITEMS IF NEEDED AND STORE THE RESULT TO A VARIABLE `filtered` -->\n<ng-container *ngIf=\"(filter ? (items | filterBy: filterBy: filter) : items) as filtered\">\n  <!-- SHOW EMPTY STATE IF NO ITEM ELSE SHOW ROWS -->\n  <ng-container *ngIf=\"filtered.length; else: emptyStateTemplate\">\n  <ul class=\"ui-list-container\" [ngClass]=\"classes\">\n    <ng-container *ngFor=\"let item of filtered; trackBy: _trackBy.bind(this) let index=index; let first=first; let last=last; let even=even; let odd=odd\">\n      <ng-container *ngIf=\"{item: item, items: filtered, odd: odd, last: last, even: even, index: index, first: first} as context\">\n        <div class=\"ui-list-item\">\n          <!-- HEADER -->\n          <ng-container *ngIf=\"_template(context, 'header') as header\">\n            <li class=\"ui-list-item-header\">\n              <ng-container *ngTemplateOutlet=\"header; context: context\"></ng-container>\n            </li>\n          </ng-container>\n          <!-- ROW -->\n          <ng-container *ngIf=\"_template(context, 'row') as row\">\n            <li class=\"ui-list-item-content\" [class.selectable]=\"selectable\">\n              <ng-container *ngTemplateOutlet=\"row; context: context\"></ng-container>\n              <!-- CHECKBOX -->\n              <label class=\"ui-list-label\" *ngIf=\"selectable\">\n                <input\n                  class=\"ui-list-label__checkbox\"\n                  type=\"checkbox\"\n                  [(ngModel)]=\"_selectionStates[$any(item)[idField]]\"\n                  (ngModelChange)=\"_toggleSelection(item)\"/>\n                <span class=\"ui-list-label__text\">\n                  <span class=\"ui-list-label__check\">\n                    <span class=\"ui-list-label__icon\">✓</span>\n                  </span>\n                </span>\n              </label>\n            </li>\n          </ng-container>\n        </div>\n      </ng-container>\n    </ng-container>\n  </ul>\n  </ng-container>\n</ng-container>\n"]}