UNPKG

angular-datatables

Version:
141 lines 20.2 kB
/** * @license * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://raw.githubusercontent.com/l-lin/angular-datatables/master/LICENSE */ import { Directive, ElementRef, Input, Renderer2, ViewContainerRef } from '@angular/core'; import { Subject } from 'rxjs'; import * as i0 from "@angular/core"; export class DataTableDirective { constructor(el, vcr, renderer) { this.el = el; this.vcr = vcr; this.renderer = renderer; /** * The DataTable option you pass to configure your table. */ this.dtOptions = {}; } ngOnInit() { if (this.dtTrigger) { this.dtTrigger.subscribe((options) => { this.displayTable(options); }); } else { this.displayTable(null); } } ngOnDestroy() { if (this.dtTrigger) { this.dtTrigger.unsubscribe(); } if (this.dt) { this.dt.destroy(true); } } displayTable(dtOptions) { // assign new options if provided if (dtOptions) { this.dtOptions = dtOptions; } this.dtInstance = new Promise((resolve, reject) => { Promise.resolve(this.dtOptions).then(resolvedDTOptions => { // validate object const isTableEmpty = Object.keys(resolvedDTOptions).length === 0 && $('tbody tr', this.el.nativeElement).length === 0; if (isTableEmpty) { reject('Both the table and dtOptions cannot be empty'); return; } // Set a column unique if (resolvedDTOptions.columns) { resolvedDTOptions.columns.forEach(col => { if ((col.id ?? '').trim() === '') { col.id = this.getColumnUniqueId(); } }); } // Using setTimeout as a "hack" to be "part" of NgZone setTimeout(() => { // Assign DT properties here let options = { rowCallback: (row, data, index) => { if (resolvedDTOptions.columns) { const columns = resolvedDTOptions.columns; this.applyNgPipeTransform(row, columns); this.applyNgRefTemplate(row, columns, data); } // run user specified row callback if provided. if (resolvedDTOptions.rowCallback) { resolvedDTOptions.rowCallback(row, data, index); } } }; // merge user's config with ours options = Object.assign({}, resolvedDTOptions, options); this.dt = $(this.el.nativeElement).DataTable(options); resolve(this.dt); }); }); }); } applyNgPipeTransform(row, columns) { // Filter columns with pipe declared const colsWithPipe = columns.filter(x => x.ngPipeInstance && !x.ngTemplateRef); colsWithPipe.forEach(el => { const pipe = el.ngPipeInstance; const pipeArgs = el.ngPipeArgs || []; // find index of column using `data` attr const i = columns.filter(c => c.visible !== false).findIndex(e => e.id === el.id); // get <td> element which holds data using index const rowFromCol = row.childNodes.item(i); // Transform data with Pipe and PipeArgs const rowVal = $(rowFromCol).text(); const rowValAfter = pipe.transform(rowVal, ...pipeArgs); // Apply transformed string to <td> $(rowFromCol).text(rowValAfter); }); } applyNgRefTemplate(row, columns, data) { // Filter columns using `ngTemplateRef` const colsWithTemplate = columns.filter(x => x.ngTemplateRef && !x.ngPipeInstance); colsWithTemplate.forEach(el => { const { ref, context } = el.ngTemplateRef; // get <td> element which holds data using index const i = columns.filter(c => c.visible !== false).findIndex(e => e.id === el.id); const cellFromIndex = row.childNodes.item(i); // reset cell before applying transform $(cellFromIndex).html(''); // render onto DOM // finalize context to be sent to user const _context = Object.assign({}, context, context?.userData, { adtData: data }); const instance = this.vcr.createEmbeddedView(ref, _context); this.renderer.appendChild(cellFromIndex, instance.rootNodes[0]); }); } getColumnUniqueId() { let result = ''; const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; for (let i = 0; i < 6; i++) { const randomIndex = Math.floor(Math.random() * characters.length); result += characters.charAt(randomIndex); } return result.trim(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: DataTableDirective, deps: [{ token: i0.ElementRef }, { token: i0.ViewContainerRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.0.5", type: DataTableDirective, selector: "[datatable]", inputs: { dtOptions: "dtOptions", dtTrigger: "dtTrigger" }, ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.5", ngImport: i0, type: DataTableDirective, decorators: [{ type: Directive, args: [{ selector: '[datatable]' }] }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.ViewContainerRef }, { type: i0.Renderer2 }], propDecorators: { dtOptions: [{ type: Input }], dtTrigger: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angular-datatables.directive.js","sourceRoot":"","sources":["../../../../lib/src/angular-datatables.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://raw.githubusercontent.com/l-lin/angular-datatables/master/LICENSE\n */\n\nimport { Directive, ElementRef, Input, OnDestroy, OnInit, Renderer2, ViewContainerRef } from '@angular/core';\nimport { Subject } from 'rxjs';\nimport { ADTSettings, ADTColumns } from './models/settings';\nimport { Api } from 'datatables.net';\n\n@Directive({\n  selector: '[datatable]'\n})\nexport class DataTableDirective implements OnDestroy, OnInit {\n  /**\n   * The DataTable option you pass to configure your table.\n   */\n  @Input()\n  dtOptions: ADTSettings = {};\n\n  /**\n   * This trigger is used if one wants to trigger manually the DT rendering\n   * Useful when rendering angular rendered DOM\n   */\n  @Input()\n  dtTrigger!: Subject<ADTSettings>;\n\n  /**\n   * The DataTable instance built by the jQuery library [DataTables](datatables.net).\n   *\n   * It's possible to execute the [DataTables APIs](https://datatables.net/reference/api/) with\n   * this variable.\n   */\n  dtInstance!: Promise<Api>;\n\n  // Only used for destroying the table when destroying this directive\n  private dt!: Api;\n\n  constructor(\n    private el: ElementRef,\n    private vcr: ViewContainerRef,\n    private renderer: Renderer2\n  ) { }\n\n  ngOnInit(): void {\n    if (this.dtTrigger) {\n      this.dtTrigger.subscribe((options) => {\n        this.displayTable(options);\n      });\n    } else {\n      this.displayTable(null);\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.dtTrigger) {\n      this.dtTrigger.unsubscribe();\n    }\n    if (this.dt) {\n      this.dt.destroy(true);\n    }\n  }\n\n  private displayTable(dtOptions: ADTSettings | null): void {\n    // assign new options if provided\n    if (dtOptions) {\n      this.dtOptions = dtOptions;\n    }\n    this.dtInstance = new Promise((resolve, reject) => {\n      Promise.resolve(this.dtOptions).then(resolvedDTOptions => {\n        // validate object\n        const isTableEmpty = Object.keys(resolvedDTOptions).length === 0 && $('tbody tr', this.el.nativeElement).length === 0;\n        if (isTableEmpty) {\n          reject('Both the table and dtOptions cannot be empty');\n          return;\n        }\n\n        // Set a column unique\n        if (resolvedDTOptions.columns) {\n          resolvedDTOptions.columns.forEach(col => {\n            if ((col.id ?? '').trim() === '') {\n              col.id = this.getColumnUniqueId();\n            }\n          });\n        }\n\n        // Using setTimeout as a \"hack\" to be \"part\" of NgZone\n        setTimeout(() => {\n          // Assign DT properties here\n          let options: ADTSettings = {\n            rowCallback: (row, data, index) => {\n              if (resolvedDTOptions.columns) {\n                const columns = resolvedDTOptions.columns;\n                this.applyNgPipeTransform(row, columns);\n                this.applyNgRefTemplate(row, columns, data);\n              }\n\n              // run user specified row callback if provided.\n              if (resolvedDTOptions.rowCallback) {\n                resolvedDTOptions.rowCallback(row, data, index);\n              }\n            }\n          };\n          // merge user's config with ours\n          options = Object.assign({}, resolvedDTOptions, options);\n          this.dt = $(this.el.nativeElement).DataTable(options);\n          resolve(this.dt);\n        });\n      });\n    });\n  }\n\n  private applyNgPipeTransform(row: Node, columns: ADTColumns[]): void {\n    // Filter columns with pipe declared\n    const colsWithPipe = columns.filter(x => x.ngPipeInstance && !x.ngTemplateRef);\n    colsWithPipe.forEach(el => {\n      const pipe = el.ngPipeInstance!;\n      const pipeArgs = el.ngPipeArgs || [];\n      // find index of column using `data` attr\n      const i = columns.filter(c => c.visible !== false).findIndex(e => e.id === el.id);\n      // get <td> element which holds data using index\n      const rowFromCol = row.childNodes.item(i);\n      // Transform data with Pipe and PipeArgs\n      const rowVal = $(rowFromCol).text();\n      const rowValAfter = pipe.transform(rowVal, ...pipeArgs);\n      // Apply transformed string to <td>\n      $(rowFromCol).text(rowValAfter);\n    });\n  }\n\n  private applyNgRefTemplate(row: Node, columns: ADTColumns[], data: Object): void {\n    // Filter columns using `ngTemplateRef`\n    const colsWithTemplate = columns.filter(x => x.ngTemplateRef && !x.ngPipeInstance);\n    colsWithTemplate.forEach(el => {\n      const { ref, context } = el.ngTemplateRef!;\n      // get <td> element which holds data using index\n      const i = columns.filter(c => c.visible !== false).findIndex(e => e.id === el.id);\n      const cellFromIndex = row.childNodes.item(i);\n      // reset cell before applying transform\n      $(cellFromIndex).html('');\n      // render onto DOM\n      // finalize context to be sent to user\n      const _context = Object.assign({}, context, context?.userData, {\n        adtData: data\n      });\n      const instance = this.vcr.createEmbeddedView(ref, _context);\n      this.renderer.appendChild(cellFromIndex, instance.rootNodes[0]);\n    });\n  }\n\n  private getColumnUniqueId(): string {\n    let result = '';\n    const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\n\n    for (let i = 0; i < 6; i++) {\n      const randomIndex = Math.floor(Math.random() * characters.length);\n      result += characters.charAt(randomIndex);\n    }\n\n    return result.trim();\n  }\n\n}\n"]}