UNPKG

ornamentum

Version:
250 lines 24.9 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { Component, Input } from '@angular/core'; import { get } from '../../../utility/services/object-utility.class'; import { DataFetchMode } from '../../models/data-fetch-mode.enum'; import { DragAndDropService } from '../../../utility/utility.module'; import { DataTableConfigService } from '../../services/data-table-config.service'; import { DataTableEventStateService } from '../../services/data-table-event.service'; import { DataTableDataStateService } from '../../services/data-table-data-state.service'; /** * Column title header component. Render data table column headers. */ export class DataTableColumnTitleHeaderComponent { /** * @param {?} dragAndDropService * @param {?} eventStateService * @param {?} dataStateService * @param {?} config */ constructor(dragAndDropService, eventStateService, dataStateService, config) { this.dragAndDropService = dragAndDropService; this.eventStateService = eventStateService; this.dataStateService = dataStateService; this.config = config; this.resizeInProgress = false; } /** * Header click event handler. * @param {?} column Data table column component reference. * @param {?} event Mouse event arguments object. * @return {?} */ onHeaderClick(column, event) { if (!this.resizeInProgress) { this.sortData(column); this.eventStateService.headerClickStream.emit({ column, event }); } else { this.resizeInProgress = false; // this is because I can't prevent click from mousup of the drag end } } /** * Sort data event handler. * @private * @param {?} column Data table column component reference. * @return {?} */ sortData(column) { if (column.sortable) { /** @type {?} */ const prevSortOrder = column.sortOrder; if (column.sortOrder) { column.sortOrder = column.getNewSortOrder(); } else { if (!this.config.multiColumnSortable) { /** @type {?} */ const sortColumns = this.columns.filter((/** * @param {?} item * @return {?} */ item => item.sortable)); sortColumns.forEach((/** * @param {?} sortColumn * @return {?} */ (sortColumn) => { if (sortColumn !== column) { sortColumn.sortOrder = ''; } })); } column.sortOrder = 'asc'; } if (this.config.multiColumnSortable) { if (column.sortOrder === '') { /** @type {?} */ const sortColumns = this.columns.filter((/** * @param {?} item * @return {?} */ item => item.sortable)); sortColumns.forEach((/** * @param {?} sortColumn * @return {?} */ (sortColumn) => { if (sortColumn !== column && sortColumn.sortPriority > column.sortPriority) { --sortColumn.sortPriority; } })); column.sortPriority = undefined; --this.dataStateService.currentSortPriority; } else { if (!prevSortOrder) { column.sortPriority = ++this.dataStateService.currentSortPriority; } } } this.eventStateService.dataFetchStream.next(DataFetchMode.SOFT_LOAD); } } /** * Set column width. * @param {?} width Width value in pixels. * @param {?} column Data table column component reference. * @return {?} */ setColumnWidth(width, column) { column.actualWidth = width; } /** * Column resize event handler. * @param {?} event Resize mouse event. * @param {?} column Data table column component. * @param {?} columnElement Table header cell element DOM reference. * @return {?} */ onColumnResize(event, column, columnElement) { this.resizeInProgress = true; this.dragAndDropService.drag(event, { move: (/** * @param {?} moveEvent * @param {?} dx * @return {?} */ (moveEvent, dx) => { /** @type {?} */ const newWidth = columnElement.offsetWidth + dx; if (column.resizeMinLimit !== undefined && newWidth < column.resizeMinLimit) { return; } column.actualWidth = newWidth; /** @type {?} */ let totalWidth = 0; this.columns.forEach((/** * @param {?} col * @return {?} */ col => { col.width = col.actualWidth; totalWidth += col.width; })); this.dataStateService.tableWidth = totalWidth; }) }); } /** * Set all row selected state. * @param {?} value All row selected status. * @return {?} */ set allRowSelected(value) { this.dataStateService.allRowSelected = value; this.allRowSelectedChanged(this.dataStateService.allRowSelected); this.eventStateService.allRowSelectChangeStream.emit(this.dataStateService.allRowSelected); } /** * Get all row selected state. * @return {?} */ get allRowSelected() { return this.dataStateService.allRowSelected; } /** * All row select change event handler. * @private * @param {?} selectedState Row selected status. * @return {?} */ allRowSelectedChanged(selectedState) { this.dataStateService.dataRows.forEach((/** * @param {?} row * @return {?} */ row => { /** @type {?} */ const id = get(row.item, this.config.selectTrackBy); /** @type {?} */ const index = this.dataStateService.selectedRows.indexOf(id); if (selectedState && index < 0) { this.dataStateService.selectedRows.push(id); } else if (!selectedState && index > -1) { this.dataStateService.selectedRows.splice(index, 1); } row.selected = selectedState; })); this.eventStateService.rowSelectChangeStream.emit(this.dataStateService.selectedRows); } /** * Get all row select checkbox display status. * @return {?} True if all row select checkbox should be displayed. */ get showAllRowSelectCheckbox() { return this.config.selectMode === 'multi' && this.config.showRowSelectAllCheckbox; } /** * @param {?} column * @return {?} */ showSortPriorityLabel(column) { return !!(this.config.multiColumnSortable && this.dataStateService.currentSortPriority > 1 && column.sortPriority); } } DataTableColumnTitleHeaderComponent.decorators = [ { type: Component, args: [{ exportAs: 'ngDataTableColumnTitleHeader', // tslint:disable-next-line selector: '[ngDataTableColumnTitleHeader]', template: "<th class=\"ng-data-table-expand-column-header\" *ngIf=\"config.expandableRows\"></th>\n<th class=\"ng-data-table-index-column-header\" *ngIf=\"config.showIndexColumn\">\n <span [textContent]=\"config.indexColumnTitle\"></span>\n</th>\n<th class=\"ng-data-table-select-column-header\" *ngIf=\"config.showRowSelectCheckboxColumn\">\n <div class=\"ng-data-table-checkbox-container\" *ngIf=\"showAllRowSelectCheckbox\">\n <input class=\"ng-data-table-checkbox-input\" type=\"checkbox\"\n [id]=\"dataStateService.getUniqueId('rsa', 0)\" [(ngModel)]=\"allRowSelected\"/>\n <label [for]=\"dataStateService.getUniqueId('rsa', 0)\"></label>\n </div>\n</th>\n<ng-container *ngFor=\"let column of columns\">\n <th class=\"ng-data-table-column-header\"\n #columnHeader\n *ngIf=\"column.visible\"\n (click)=\"onHeaderClick(column, $event)\"\n (ngElementWidth)=\"setColumnWidth($event, column)\"\n [class.sortable]=\"column.sortable\"\n [class.resizable]=\"column.resizable\"\n [ngClass]=\"column.cssClass\">\n <span class=\"ng-data-table-column-sort-priority\" *ngIf=\"showSortPriorityLabel(column)\"><small>{{column.sortPriority}}</small></span>\n <span *ngIf=\"!column.headerTemplate\" [textContent]=\"column.title\" class=\"ng-data-table-column-header-label\"></span>\n <ng-container *ngIf=\"column.headerTemplate\" [ngTemplateOutlet]=\"column.headerTemplate\" [ngTemplateOutletContext]=\"{ column: column }\">\n </ng-container>\n <span class=\"ng-data-table-column-sort-icon\" [hidden]=\"!column.sortable\" [ngClass]=\"column.getSortIconClass()\"></span>\n <span class=\"ng-data-table-column-resize-handle\" *ngIf=\"column.resizable\" (mousedown)=\"onColumnResize($event, column, columnHeader)\"> </span>\n </th>\n</ng-container>\n" }] } ]; /** @nocollapse */ DataTableColumnTitleHeaderComponent.ctorParameters = () => [ { type: DragAndDropService }, { type: DataTableEventStateService }, { type: DataTableDataStateService }, { type: DataTableConfigService } ]; DataTableColumnTitleHeaderComponent.propDecorators = { columns: [{ type: Input }] }; if (false) { /** * @type {?} * @private */ DataTableColumnTitleHeaderComponent.prototype.resizeInProgress; /** @type {?} */ DataTableColumnTitleHeaderComponent.prototype.columns; /** * @type {?} * @private */ DataTableColumnTitleHeaderComponent.prototype.dragAndDropService; /** * @type {?} * @private */ DataTableColumnTitleHeaderComponent.prototype.eventStateService; /** @type {?} */ DataTableColumnTitleHeaderComponent.prototype.dataStateService; /** @type {?} */ DataTableColumnTitleHeaderComponent.prototype.config; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-table-column-title-header.component.js","sourceRoot":"ng://ornamentum/","sources":["data-table/components/data-table-column-title-header/data-table-column-title-header.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from '@angular/core';\n\nimport { get } from '../../../utility/services/object-utility.class';\n\nimport { DataFetchMode } from '../../models/data-fetch-mode.enum';\n\nimport { DataTableColumnComponent } from '../data-table-column/data-table-column.component';\n\nimport { DragAndDropService } from '../../../utility/utility.module';\nimport { DataTableConfigService } from '../../services/data-table-config.service';\nimport { DataTableEventStateService } from '../../services/data-table-event.service';\nimport { DataTableDataStateService } from '../../services/data-table-data-state.service';\n\n/**\n * Column title header component. Render data table column headers.\n */\n@Component({\n  exportAs: 'ngDataTableColumnTitleHeader',\n  // tslint:disable-next-line\n  selector: '[ngDataTableColumnTitleHeader]',\n  templateUrl: './data-table-column-title-header.component.html'\n})\nexport class DataTableColumnTitleHeaderComponent {\n  private resizeInProgress = false;\n\n  @Input()\n  public columns: DataTableColumnComponent[];\n\n  constructor(\n    private dragAndDropService: DragAndDropService,\n    private eventStateService: DataTableEventStateService,\n    public dataStateService: DataTableDataStateService,\n    public config: DataTableConfigService\n  ) {}\n\n  /**\n   * Header click event handler.\n   * @param column Data table column component reference.\n   * @param event Mouse event arguments object.\n   */\n  public onHeaderClick(column: DataTableColumnComponent, event: MouseEvent): void {\n    if (!this.resizeInProgress) {\n      this.sortData(column);\n      this.eventStateService.headerClickStream.emit({ column, event });\n    } else {\n      this.resizeInProgress = false; // this is because I can't prevent click from mousup of the drag end\n    }\n  }\n\n  /**\n   * Sort data event handler.\n   * @param column Data table column component reference.\n   */\n  private sortData(column: DataTableColumnComponent): void {\n    if (column.sortable) {\n      const prevSortOrder = column.sortOrder;\n      if (column.sortOrder) {\n        column.sortOrder = column.getNewSortOrder();\n      } else {\n        if (!this.config.multiColumnSortable) {\n          const sortColumns = this.columns.filter(item => item.sortable);\n          sortColumns.forEach((sortColumn: DataTableColumnComponent) => {\n            if (sortColumn !== column) {\n              sortColumn.sortOrder = '';\n            }\n          });\n        }\n\n        column.sortOrder = 'asc';\n      }\n\n      if (this.config.multiColumnSortable) {\n        if (column.sortOrder === '') {\n          const sortColumns = this.columns.filter(item => item.sortable);\n          sortColumns.forEach((sortColumn: DataTableColumnComponent) => {\n            if (sortColumn !== column && sortColumn.sortPriority > column.sortPriority) {\n              --sortColumn.sortPriority;\n            }\n          });\n\n          column.sortPriority = undefined;\n          --this.dataStateService.currentSortPriority;\n        } else {\n          if (!prevSortOrder) {\n            column.sortPriority = ++this.dataStateService.currentSortPriority;\n          }\n        }\n      }\n\n      this.eventStateService.dataFetchStream.next(DataFetchMode.SOFT_LOAD);\n    }\n  }\n\n  /**\n   * Set column width.\n   * @param width Width value in pixels.\n   * @param column Data table column component reference.\n   */\n  public setColumnWidth(width: number, column: DataTableColumnComponent): void {\n    column.actualWidth = width;\n  }\n\n  /**\n   * Column resize event handler.\n   * @param event Resize mouse event.\n   * @param column Data table column component.\n   * @param columnElement Table header cell element DOM reference.\n   */\n  public onColumnResize(event: MouseEvent, column: DataTableColumnComponent, columnElement: HTMLTableHeaderCellElement): void {\n    this.resizeInProgress = true;\n\n    this.dragAndDropService.drag(event, {\n      move: (moveEvent: MouseEvent, dx: number) => {\n        const newWidth = columnElement.offsetWidth + dx;\n        if (column.resizeMinLimit !== undefined && newWidth < column.resizeMinLimit) {\n          return;\n        }\n\n        column.actualWidth = newWidth;\n        let totalWidth = 0;\n\n        this.columns.forEach(col => {\n          col.width = col.actualWidth;\n          totalWidth += col.width;\n        });\n\n        this.dataStateService.tableWidth = totalWidth;\n      }\n    });\n  }\n\n  /**\n   * Set all row selected state.\n   * @param value All row selected status.\n   */\n  public set allRowSelected(value: boolean) {\n    this.dataStateService.allRowSelected = value;\n    this.allRowSelectedChanged(this.dataStateService.allRowSelected);\n    this.eventStateService.allRowSelectChangeStream.emit(this.dataStateService.allRowSelected);\n  }\n\n  /**\n   * Get all row selected state.\n   */\n  public get allRowSelected(): boolean {\n    return this.dataStateService.allRowSelected;\n  }\n\n  /**\n   * All row select change event handler.\n   * @param selectedState Row selected status.\n   */\n  private allRowSelectedChanged(selectedState: boolean): void {\n    this.dataStateService.dataRows.forEach(row => {\n      const id = get(row.item, this.config.selectTrackBy);\n      const index = this.dataStateService.selectedRows.indexOf(id);\n      if (selectedState && index < 0) {\n        this.dataStateService.selectedRows.push(id);\n      } else if (!selectedState && index > -1) {\n        this.dataStateService.selectedRows.splice(index, 1);\n      }\n\n      row.selected = selectedState;\n    });\n\n    this.eventStateService.rowSelectChangeStream.emit(this.dataStateService.selectedRows);\n  }\n\n  /**\n   * Get all row select checkbox display status.\n   * @return True if all row select checkbox should be displayed.\n   */\n  public get showAllRowSelectCheckbox(): boolean {\n    return this.config.selectMode === 'multi' && this.config.showRowSelectAllCheckbox;\n  }\n\n  public showSortPriorityLabel(column: DataTableColumnComponent): boolean {\n    return !!(this.config.multiColumnSortable && this.dataStateService.currentSortPriority > 1 && column.sortPriority);\n  }\n}\n"]}