ms-data-grid 0.0.46 → 0.0.48

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,149 @@
1
+ import { Injectable } from '@angular/core';
2
+ import * as XLSX from 'xlsx-js-style';
3
+ import { saveAs } from 'file-saver';
4
+ import * as i0 from "@angular/core";
5
+ export class ExportService {
6
+ constructor() { }
7
+ /**
8
+ * Exports data to CSV or Excel.
9
+ * @param columns List of column definitions (each having name/key/is_visible)
10
+ * @param data The array of data objects
11
+ * @param fileName Name of the exported file (without extension)
12
+ * @param format 'csv' | 'xlsx'
13
+ */
14
+ exportData(columns, data, fileName, format = 'xlsx', rowSelectedIndexes = new Set(), styleConfig = {}) {
15
+ const { headerBgColor = '#228B22', headerTextColor = '#FFFFFF', bodyTextColor = '#000000', fontFamily = 'Calibri', fontWeight = 'normal', zoomScale = 120, headerRowHeight = 20, selectedRowBgColor = '#8AC5FF', } = styleConfig;
16
+ const visibleColumns = columns.filter(col => col.is_visible && col.type !== 'image' && col.type !== 'logo');
17
+ const headers = visibleColumns.map(col => col.header || col.headerName || col.name);
18
+ const flattenValue = (value) => {
19
+ if (value == null)
20
+ return '';
21
+ if (Array.isArray(value))
22
+ return value.map(v => (typeof v === 'object' ? flattenValue(v) : v)).join(', ');
23
+ if (typeof value === 'object')
24
+ return Object.entries(value)
25
+ .map(([k, v]) => `${k}: ${flattenValue(v)}`)
26
+ .join(', ');
27
+ return value;
28
+ };
29
+ const exportData = data.map(row => visibleColumns.reduce((acc, col) => {
30
+ const header = col.headerName || col.name || col.header;
31
+ let value = this.getNestedValue(row, col.field);
32
+ switch (col.type) {
33
+ case 'boolean':
34
+ value = value ? 'Yes' : 'No';
35
+ break;
36
+ case 'date':
37
+ if (value) {
38
+ const date = new Date(value);
39
+ if (!isNaN(date.getTime())) {
40
+ value = date.toLocaleDateString('en-US', {
41
+ day: '2-digit',
42
+ month: 'long',
43
+ year: 'numeric',
44
+ });
45
+ }
46
+ }
47
+ break;
48
+ default:
49
+ value = flattenValue(value);
50
+ }
51
+ acc[header] = value;
52
+ return acc;
53
+ }, {}));
54
+ const worksheet = XLSX.utils.json_to_sheet(exportData, { header: headers });
55
+ // ✅ Define a reusable border style
56
+ const borderStyle = {
57
+ top: { style: 'thin', color: { rgb: 'CCCCCC' } },
58
+ bottom: { style: 'thin', color: { rgb: 'CCCCCC' } },
59
+ left: { style: 'thin', color: { rgb: 'CCCCCC' } },
60
+ right: { style: 'thin', color: { rgb: 'CCCCCC' } },
61
+ };
62
+ // ✅ Apply header styles
63
+ headers.forEach((header, index) => {
64
+ const cellAddress = XLSX.utils.encode_cell({ r: 0, c: index });
65
+ if (!worksheet[cellAddress])
66
+ return;
67
+ worksheet[cellAddress].s = {
68
+ fill: { fgColor: { rgb: headerBgColor.replace('#', '') } },
69
+ font: {
70
+ name: fontFamily,
71
+ sz: 14,
72
+ color: { rgb: headerTextColor.replace('#', '') },
73
+ bold: fontWeight === 'bold',
74
+ },
75
+ alignment: { horizontal: 'center', vertical: 'center' },
76
+ border: borderStyle, // ✅ Added border here
77
+ };
78
+ });
79
+ // ✅ Apply body styles + highlight selected rows
80
+ for (let r = 1; r <= exportData.length; r++) {
81
+ const isSelected = rowSelectedIndexes?.has(r - 1);
82
+ const rowBgColor = isSelected ? selectedRowBgColor : '#FFFFFF';
83
+ for (let c = 0; c < headers.length; c++) {
84
+ const col = visibleColumns[c];
85
+ const cellAddress = XLSX.utils.encode_cell({ r, c });
86
+ const cell = worksheet[cellAddress];
87
+ if (!cell)
88
+ continue;
89
+ let horizontalAlign = 'left';
90
+ if (['number', 'boolean', 'date'].includes(col.type))
91
+ horizontalAlign = 'center';
92
+ cell.s = {
93
+ fill: { fgColor: { rgb: rowBgColor.replace('#', '') } },
94
+ font: {
95
+ name: fontFamily,
96
+ sz: 14,
97
+ color: { rgb: bodyTextColor.replace('#', '') },
98
+ bold: false,
99
+ },
100
+ alignment: { horizontal: horizontalAlign, vertical: 'center', wrapText: true },
101
+ border: borderStyle, // ✅ Added border here too
102
+ };
103
+ }
104
+ }
105
+ // ✅ Adjust column widths
106
+ const colWidths = headers.map(header => {
107
+ let maxLength = header.length;
108
+ exportData.forEach(row => {
109
+ const val = row[header];
110
+ if (val != null) {
111
+ const len = val.toString().length;
112
+ if (len > maxLength)
113
+ maxLength = len;
114
+ }
115
+ });
116
+ return { wch: Math.min(maxLength + 2, 64) };
117
+ });
118
+ worksheet['!cols'] = colWidths;
119
+ worksheet['!rows'] = [{ hpt: headerRowHeight }];
120
+ worksheet['!zoom'] = zoomScale;
121
+ const workbook = { Sheets: { Data: worksheet }, SheetNames: ['Data'] };
122
+ if (format === 'csv') {
123
+ const csv = XLSX.utils.sheet_to_csv(worksheet);
124
+ const blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });
125
+ saveAs(blob, `${fileName}.csv`);
126
+ }
127
+ else {
128
+ const excelBuffer = XLSX.write(workbook, { bookType: 'xlsx', type: 'array' });
129
+ const blob = new Blob([excelBuffer], {
130
+ type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8',
131
+ });
132
+ saveAs(blob, `${fileName}.xlsx`);
133
+ }
134
+ }
135
+ getNestedValue(obj, path) {
136
+ if (!path || typeof obj !== 'object')
137
+ return obj;
138
+ return path.split('.').reduce((acc, key) => (acc && acc[key] !== undefined ? acc[key] : ''), obj);
139
+ }
140
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ExportService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
141
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ExportService, providedIn: 'root' }); }
142
+ }
143
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ExportService, decorators: [{
144
+ type: Injectable,
145
+ args: [{
146
+ providedIn: 'root'
147
+ }]
148
+ }], ctorParameters: function () { return []; } });
149
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"export.service.js","sourceRoot":"","sources":["../../../../../projects/data-grid/src/lib/services/export.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport * as XLSX from 'xlsx-js-style';\r\nimport { saveAs } from 'file-saver';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class ExportService {\r\n\r\n  constructor() {}\r\n\r\n  /**\r\n   * Exports data to CSV or Excel.\r\n   * @param columns List of column definitions (each having name/key/is_visible)\r\n   * @param data The array of data objects\r\n   * @param fileName Name of the exported file (without extension)\r\n   * @param format 'csv' | 'xlsx'\r\n   */\r\nexportData(\r\n  columns: any[],\r\n  data: any[],\r\n  fileName: string,\r\n  format: 'csv' | 'xlsx' = 'xlsx',\r\n  rowSelectedIndexes: Set<number> | null = new Set(),\r\n  styleConfig: {\r\n    headerBgColor?: string;\r\n    headerTextColor?: string;\r\n    bodyTextColor?: string;\r\n    fontFamily?: string;\r\n    fontWeight?: 'normal' | 'bold';\r\n    zoomScale?: number;\r\n    headerRowHeight?: number;\r\n    selectedRowBgColor?: string;\r\n  } = {}\r\n) {\r\n  const {\r\n    headerBgColor = '#228B22',\r\n    headerTextColor = '#FFFFFF',\r\n    bodyTextColor = '#000000',\r\n    fontFamily = 'Calibri',\r\n    fontWeight = 'normal',\r\n    zoomScale = 120,\r\n    headerRowHeight = 20,\r\n    selectedRowBgColor = '#8AC5FF',\r\n  } = styleConfig;\r\n\r\n  const visibleColumns = columns.filter(\r\n    col => col.is_visible && col.type !== 'image' && col.type !== 'logo'\r\n  );\r\n\r\n  const headers = visibleColumns.map(col => col.header || col.headerName || col.name);\r\n\r\n  const flattenValue = (value: any): any => {\r\n    if (value == null) return '';\r\n    if (Array.isArray(value))\r\n      return value.map(v => (typeof v === 'object' ? flattenValue(v) : v)).join(', ');\r\n    if (typeof value === 'object')\r\n      return Object.entries(value)\r\n        .map(([k, v]) => `${k}: ${flattenValue(v)}`)\r\n        .join(', ');\r\n    return value;\r\n  };\r\n\r\n  const exportData = data.map(row =>\r\n    visibleColumns.reduce((acc: any, col) => {\r\n      const header = col.headerName || col.name || col.header;\r\n      let value = this.getNestedValue(row, col.field);\r\n\r\n      switch (col.type) {\r\n        case 'boolean':\r\n          value = value ? 'Yes' : 'No';\r\n          break;\r\n        case 'date':\r\n          if (value) {\r\n            const date = new Date(value);\r\n            if (!isNaN(date.getTime())) {\r\n              value = date.toLocaleDateString('en-US', {\r\n                day: '2-digit',\r\n                month: 'long',\r\n                year: 'numeric',\r\n              });\r\n            }\r\n          }\r\n          break;\r\n        default:\r\n          value = flattenValue(value);\r\n      }\r\n\r\n      acc[header] = value;\r\n      return acc;\r\n    }, {})\r\n  );\r\n\r\n  const worksheet: XLSX.WorkSheet = XLSX.utils.json_to_sheet(exportData, { header: headers });\r\n\r\n  // ✅ Define a reusable border style\r\n  const borderStyle = {\r\n    top: { style: 'thin', color: { rgb: 'CCCCCC' } },\r\n    bottom: { style: 'thin', color: { rgb: 'CCCCCC' } },\r\n    left: { style: 'thin', color: { rgb: 'CCCCCC' } },\r\n    right: { style: 'thin', color: { rgb: 'CCCCCC' } },\r\n  };\r\n\r\n  // ✅ Apply header styles\r\n  headers.forEach((header, index) => {\r\n    const cellAddress = XLSX.utils.encode_cell({ r: 0, c: index });\r\n    if (!worksheet[cellAddress]) return;\r\n\r\n    worksheet[cellAddress].s = {\r\n      fill: { fgColor: { rgb: headerBgColor.replace('#', '') } },\r\n      font: {\r\n        name: fontFamily,\r\n        sz: 14,\r\n        color: { rgb: headerTextColor.replace('#', '') },\r\n        bold: fontWeight === 'bold',\r\n      },\r\n      alignment: { horizontal: 'center', vertical: 'center' },\r\n      border: borderStyle, // ✅ Added border here\r\n    };\r\n  });\r\n\r\n  // ✅ Apply body styles + highlight selected rows\r\n  for (let r = 1; r <= exportData.length; r++) {\r\n    const isSelected = rowSelectedIndexes?.has(r - 1);\r\n    const rowBgColor = isSelected ? selectedRowBgColor : '#FFFFFF';\r\n\r\n    for (let c = 0; c < headers.length; c++) {\r\n      const col = visibleColumns[c];\r\n      const cellAddress = XLSX.utils.encode_cell({ r, c });\r\n      const cell = worksheet[cellAddress];\r\n      if (!cell) continue;\r\n\r\n      let horizontalAlign = 'left';\r\n      if (['number', 'boolean', 'date'].includes(col.type)) horizontalAlign = 'center';\r\n\r\n      cell.s = {\r\n        fill: { fgColor: { rgb: rowBgColor.replace('#', '') } },\r\n        font: {\r\n          name: fontFamily,\r\n          sz: 14,\r\n          color: { rgb: bodyTextColor.replace('#', '') },\r\n          bold: false,\r\n        },\r\n        alignment: { horizontal: horizontalAlign, vertical: 'center', wrapText: true },\r\n        border: borderStyle, // ✅ Added border here too\r\n      };\r\n    }\r\n  }\r\n\r\n  // ✅ Adjust column widths\r\n  const colWidths = headers.map(header => {\r\n    let maxLength = header.length;\r\n    exportData.forEach(row => {\r\n      const val = row[header];\r\n      if (val != null) {\r\n        const len = val.toString().length;\r\n        if (len > maxLength) maxLength = len;\r\n      }\r\n    });\r\n    return { wch: Math.min(maxLength + 2, 64) };\r\n  });\r\n  worksheet['!cols'] = colWidths;\r\n\r\n  worksheet['!rows'] = [{ hpt: headerRowHeight }];\r\n  worksheet['!zoom'] = zoomScale;\r\n\r\n  const workbook: XLSX.WorkBook = { Sheets: { Data: worksheet }, SheetNames: ['Data'] };\r\n\r\n  if (format === 'csv') {\r\n    const csv = XLSX.utils.sheet_to_csv(worksheet);\r\n    const blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });\r\n    saveAs(blob, `${fileName}.csv`);\r\n  } else {\r\n    const excelBuffer: any = XLSX.write(workbook, { bookType: 'xlsx', type: 'array' });\r\n    const blob = new Blob([excelBuffer], {\r\n      type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8',\r\n    });\r\n    saveAs(blob, `${fileName}.xlsx`);\r\n  }\r\n}\r\n\r\n\r\n\r\n\r\nprivate getNestedValue(obj: any, path: string): any {\r\n  if (!path || typeof obj !== 'object') return obj;\r\n  return path.split('.').reduce((acc, key) => (acc && acc[key] !== undefined ? acc[key] : ''), obj);\r\n}\r\n}\r\n"]}
@@ -1,8 +1,6 @@
1
1
  /*
2
2
  * Public API Surface of data-grid
3
3
  */
4
- export * from './lib/data-grid.service';
5
4
  export * from './lib/data-grid.module';
6
5
  export * from './lib/data-grid/data-grid.component';
7
- export * from './lib/directives/draggable-header.directive';
8
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2RhdGEtZ3JpZC9zcmMvcHVibGljLWFwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMseUJBQXlCLENBQUM7QUFDeEMsY0FBYyx3QkFBd0IsQ0FBQztBQUN2QyxjQUFjLHFDQUFxQyxDQUFDO0FBQ3BELGNBQWMsNkNBQTZDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxyXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgZGF0YS1ncmlkXHJcbiAqL1xyXG5cclxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGF0YS1ncmlkLnNlcnZpY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9kYXRhLWdyaWQubW9kdWxlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGF0YS1ncmlkL2RhdGEtZ3JpZC5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9kaXJlY3RpdmVzL2RyYWdnYWJsZS1oZWFkZXIuZGlyZWN0aXZlJztcclxuIl19
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2RhdGEtZ3JpZC9zcmMvcHVibGljLWFwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsd0JBQXdCLENBQUM7QUFDdkMsY0FBYyxxQ0FBcUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXHJcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBkYXRhLWdyaWRcclxuICovXHJcblxyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9kYXRhLWdyaWQubW9kdWxlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGF0YS1ncmlkL2RhdGEtZ3JpZC5jb21wb25lbnQnO1xyXG4iXX0=