angular-datatables
Version:
Angular directive for DataTables
141 lines • 20.2 kB
JavaScript
/**
* @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"]}