@tolle_/tolle-ui 0.0.1-beta

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.
Files changed (77) hide show
  1. package/README.md +35 -0
  2. package/esm2022/lib/accordion-item.component.mjs +78 -0
  3. package/esm2022/lib/accordion.component.mjs +60 -0
  4. package/esm2022/lib/badge.component.mjs +76 -0
  5. package/esm2022/lib/button-group.component.mjs +25 -0
  6. package/esm2022/lib/button.component.mjs +70 -0
  7. package/esm2022/lib/calendar.component.mjs +315 -0
  8. package/esm2022/lib/card.component.mjs +94 -0
  9. package/esm2022/lib/checkbox.component.mjs +100 -0
  10. package/esm2022/lib/data-table.component.mjs +332 -0
  11. package/esm2022/lib/date-picker.component.mjs +232 -0
  12. package/esm2022/lib/date-range-picker.component.mjs +208 -0
  13. package/esm2022/lib/input.component.mjs +134 -0
  14. package/esm2022/lib/masked-input.component.mjs +179 -0
  15. package/esm2022/lib/modal-ref.mjs +31 -0
  16. package/esm2022/lib/modal-stack.service.mjs +26 -0
  17. package/esm2022/lib/modal.component.mjs +98 -0
  18. package/esm2022/lib/modal.mjs +27 -0
  19. package/esm2022/lib/modal.service.mjs +65 -0
  20. package/esm2022/lib/multi-select.component.mjs +231 -0
  21. package/esm2022/lib/pagination.component.mjs +279 -0
  22. package/esm2022/lib/range-calendar.component.mjs +285 -0
  23. package/esm2022/lib/select-group.component.mjs +28 -0
  24. package/esm2022/lib/select-item.component.mjs +84 -0
  25. package/esm2022/lib/select-separator.component.mjs +24 -0
  26. package/esm2022/lib/select.component.mjs +261 -0
  27. package/esm2022/lib/select.service.mjs +21 -0
  28. package/esm2022/lib/skeleton.component.mjs +34 -0
  29. package/esm2022/lib/switch.component.mjs +133 -0
  30. package/esm2022/lib/toast.service.mjs +59 -0
  31. package/esm2022/lib/tolle-cell.directive.mjs +22 -0
  32. package/esm2022/lib/tolle-config.mjs +11 -0
  33. package/esm2022/lib/tooltip.directive.mjs +71 -0
  34. package/esm2022/lib/types/date-range.mjs +2 -0
  35. package/esm2022/lib/utils/cn.mjs +6 -0
  36. package/esm2022/public-api.mjs +36 -0
  37. package/esm2022/tolle_-tolle-ui.mjs +5 -0
  38. package/fesm2022/tolle_-tolle-ui.mjs +3553 -0
  39. package/fesm2022/tolle_-tolle-ui.mjs.map +1 -0
  40. package/index.d.ts +5 -0
  41. package/lib/accordion-item.component.d.ts +13 -0
  42. package/lib/accordion.component.d.ts +14 -0
  43. package/lib/badge.component.d.ts +14 -0
  44. package/lib/button-group.component.d.ts +8 -0
  45. package/lib/button.component.d.ts +16 -0
  46. package/lib/calendar.component.d.ts +35 -0
  47. package/lib/card.component.d.ts +32 -0
  48. package/lib/checkbox.component.d.ts +23 -0
  49. package/lib/data-table.component.d.ts +45 -0
  50. package/lib/date-picker.component.d.ts +35 -0
  51. package/lib/date-range-picker.component.d.ts +36 -0
  52. package/lib/input.component.d.ts +27 -0
  53. package/lib/masked-input.component.d.ts +36 -0
  54. package/lib/modal-ref.d.ts +16 -0
  55. package/lib/modal-stack.service.d.ts +12 -0
  56. package/lib/modal.component.d.ts +19 -0
  57. package/lib/modal.d.ts +29 -0
  58. package/lib/modal.service.d.ts +18 -0
  59. package/lib/multi-select.component.d.ts +47 -0
  60. package/lib/pagination.component.d.ts +36 -0
  61. package/lib/range-calendar.component.d.ts +37 -0
  62. package/lib/select-group.component.d.ts +8 -0
  63. package/lib/select-item.component.d.ts +18 -0
  64. package/lib/select-separator.component.d.ts +8 -0
  65. package/lib/select.component.d.ts +45 -0
  66. package/lib/select.service.d.ts +10 -0
  67. package/lib/skeleton.component.d.ts +10 -0
  68. package/lib/switch.component.d.ts +39 -0
  69. package/lib/toast.service.d.ts +24 -0
  70. package/lib/tolle-cell.directive.d.ts +9 -0
  71. package/lib/tolle-config.d.ts +9 -0
  72. package/lib/tooltip.directive.d.ts +15 -0
  73. package/lib/types/date-range.d.ts +4 -0
  74. package/lib/utils/cn.d.ts +2 -0
  75. package/package.json +32 -0
  76. package/public-api.d.ts +32 -0
  77. package/theme.css +211 -0
@@ -0,0 +1,332 @@
1
+ import { Component, Input, ContentChildren } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { FormsModule } from '@angular/forms';
4
+ import { cn } from './utils/cn';
5
+ import { PaginationComponent } from '@tolle/ui/pagination.component';
6
+ import { InputComponent } from '@tolle/ui/input.component';
7
+ import { TolleCellDirective } from '@tolle/ui/tolle-cell.directive';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/common";
10
+ import * as i2 from "@angular/forms";
11
+ export class DataTableComponent {
12
+ data = [];
13
+ columns = [];
14
+ searchable = true;
15
+ paginate = true;
16
+ pageSize = 10;
17
+ expandable = false;
18
+ size = 'default';
19
+ // Track which rows are open
20
+ expandedRows = new Set();
21
+ // Use ContentChildren to grab the tolleCell templates from the user's HTML
22
+ cellTemplates;
23
+ // Keep this as an Input for the main expansion slot
24
+ expandedTemplate;
25
+ filteredData = [];
26
+ pagedData = [];
27
+ searchTerm = '';
28
+ currentPage = 1;
29
+ sortKey = '';
30
+ sortDir = null;
31
+ // 2. Map Size to Padding for Cells
32
+ get cellPaddingClass() {
33
+ switch (this.size) {
34
+ case 'xs': return 'p-1 px-4'; // Ultra-compact
35
+ case 'sm': return 'p-2 px-4'; // Dense
36
+ case 'lg': return 'p-6 px-4'; // Spacious
37
+ default: return 'p-4'; // Standard (16px)
38
+ }
39
+ }
40
+ // 3. Map Size to Padding for Header (usually slightly shorter than cells)
41
+ get headerPaddingClass() {
42
+ switch (this.size) {
43
+ case 'xs': return 'h-7 px-4';
44
+ case 'sm': return 'h-9 px-4';
45
+ case 'lg': return 'h-14 px-4';
46
+ default: return 'h-12 px-4';
47
+ }
48
+ }
49
+ // 4. Map Size to Font Sizes
50
+ get fontSizeClass() {
51
+ switch (this.size) {
52
+ case 'xs': return 'text-[11px]';
53
+ case 'sm': return 'text-xs';
54
+ case 'lg': return 'text-base';
55
+ default: return 'text-sm';
56
+ }
57
+ }
58
+ cn = cn;
59
+ ngOnInit() { this.refreshTable(); }
60
+ ngOnChanges(changes) {
61
+ if (changes['data']) {
62
+ this.refreshTable();
63
+ }
64
+ }
65
+ // --- Search & Sort & Page Logic ---
66
+ // (Your existing implementation of applySearch, applySort, and updatePage goes here)
67
+ refreshTable() { this.applySearch(); this.applySort(); this.updatePage(); }
68
+ onSearch() { this.currentPage = 1; this.refreshTable(); }
69
+ applySearch() {
70
+ if (!this.searchTerm) {
71
+ this.filteredData = [...this.data];
72
+ return;
73
+ }
74
+ const q = this.searchTerm.toLowerCase();
75
+ this.filteredData = this.data.filter(row => Object.values(row).some(val => String(val).toLowerCase().includes(q)));
76
+ }
77
+ applySort() {
78
+ if (!this.sortKey || !this.sortDir)
79
+ return;
80
+ this.filteredData.sort((a, b) => {
81
+ const valA = a[this.sortKey];
82
+ const valB = b[this.sortKey];
83
+ if (valA < valB)
84
+ return this.sortDir === 'asc' ? -1 : 1;
85
+ if (valA > valB)
86
+ return this.sortDir === 'asc' ? 1 : -1;
87
+ return 0;
88
+ });
89
+ }
90
+ updatePage() {
91
+ if (!this.paginate) {
92
+ this.pagedData = this.filteredData;
93
+ return;
94
+ }
95
+ const start = (this.currentPage - 1) * this.pageSize;
96
+ const end = start + this.pageSize;
97
+ this.pagedData = this.filteredData.slice(start, end);
98
+ }
99
+ // --- Helpers ---
100
+ toggleSort(key) {
101
+ if (this.sortKey === key) {
102
+ this.sortDir = this.sortDir === 'asc' ? 'desc' : this.sortDir === 'desc' ? null : 'asc';
103
+ }
104
+ else {
105
+ this.sortKey = key;
106
+ this.sortDir = 'asc';
107
+ }
108
+ this.refreshTable();
109
+ }
110
+ getSortIcon(key) {
111
+ if (this.sortKey !== key || !this.sortDir)
112
+ return 'ri-arrow-up-down-line opacity-30';
113
+ return this.sortDir === 'asc' ? 'ri-arrow-up-line' : 'ri-arrow-down-line';
114
+ }
115
+ toggleRow(index) {
116
+ if (this.expandedRows.has(index))
117
+ this.expandedRows.delete(index);
118
+ else
119
+ this.expandedRows.add(index);
120
+ }
121
+ // Helper to find the right cell template
122
+ getTemplate(key) {
123
+ return this.cellTemplates?.find(t => t.name === key);
124
+ }
125
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DataTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
126
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: DataTableComponent, isStandalone: true, selector: "tolle-data-table", inputs: { data: "data", columns: "columns", searchable: "searchable", paginate: "paginate", pageSize: "pageSize", expandable: "expandable", size: "size", expandedTemplate: "expandedTemplate" }, queries: [{ propertyName: "cellTemplates", predicate: TolleCellDirective }], usesOnChanges: true, ngImport: i0, template: `
127
+ <div class="space-y-4">
128
+ <div *ngIf="searchable" class="flex items-center py-2">
129
+ <tolle-input
130
+ [size]="size === 'lg' ? 'default' : 'sm'"
131
+ class="max-w-sm"
132
+ placeholder="Filter records..."
133
+ [(ngModel)]="searchTerm"
134
+ (ngModelChange)="onSearch()">
135
+ <i prefix class="ri-search-line"></i>
136
+ </tolle-input>
137
+ </div>
138
+
139
+ <div class="rounded-md border border-border bg-background overflow-hidden shadow-sm">
140
+ <table class="w-full text-sm">
141
+ <thead class="border-b bg-muted/30">
142
+ <tr>
143
+ <th *ngIf="expandable" [class]="cn('px-4', size === 'xs' ? 'w-[32px]' : 'w-[48px]')"></th>
144
+ <th *ngFor="let col of columns"
145
+ [class]="cn(
146
+ 'font-medium text-muted-foreground transition-all',
147
+ headerPaddingClass,
148
+ fontSizeClass,
149
+ col.class
150
+ )">
151
+ <div *ngIf="col.sortable; else simpleHeader" (click)="toggleSort(col.key)" class="flex items-center gap-1 cursor-pointer hover:text-foreground">
152
+ {{ col.label }}
153
+ <i [class]="getSortIcon(col.key)"></i>
154
+ </div>
155
+ <ng-template #simpleHeader>{{ col.label }}</ng-template>
156
+ </th>
157
+ </tr>
158
+ </thead>
159
+ <tbody class="divide-y divide-border">
160
+ <ng-container *ngFor="let row of pagedData; let i = index">
161
+ <tr class="hover:bg-muted/50 transition-colors">
162
+ <td *ngIf="expandable" class="px-4">
163
+ <button (click)="toggleRow(i)"
164
+ [class]="cn(
165
+ 'flex items-center justify-center rounded-md hover:bg-accent text-muted-foreground hover:text-foreground',
166
+ size === 'xs' ? 'h-6 w-6' : 'h-8 w-8'
167
+ )">
168
+ <i [class]="expandedRows.has(i) ? 'ri-arrow-down-s-line' : 'ri-arrow-right-s-line'"></i>
169
+ </button>
170
+ </td>
171
+
172
+ <td *ngFor="let col of columns"
173
+ [class]="cn(
174
+ 'align-middle transition-all',
175
+ cellPaddingClass,
176
+ fontSizeClass,
177
+ col.class
178
+ )">
179
+ <ng-container *ngIf="getTemplate(col.key) as cell; else defaultValue">
180
+ <ng-container *ngTemplateOutlet="cell.template; context: { $implicit: row[col.key], row: row }"></ng-container>
181
+ </ng-container>
182
+ <ng-template #defaultValue>
183
+ <span class="text-foreground">{{ row[col.key] }}</span>
184
+ </ng-template>
185
+ </td>
186
+ </tr>
187
+
188
+ <tr *ngIf="expandedRows.has(i)" class="bg-muted/10">
189
+ <td [attr.colspan]="columns.length + (expandable ? 1 : 0)" class="p-0">
190
+ <div class="p-6 border-b border-dashed border-border">
191
+ <ng-container *ngIf="expandedTemplate; else defaultExpanded">
192
+ <ng-container *ngTemplateOutlet="expandedTemplate; context: { row: row }"></ng-container>
193
+ </ng-container>
194
+ <ng-template #defaultExpanded>
195
+ <div class="text-xs text-muted-foreground italic">No details available.</div>
196
+ </ng-template>
197
+ </div>
198
+ </td>
199
+ </tr>
200
+ </ng-container>
201
+ </tbody>
202
+ </table>
203
+ </div>
204
+
205
+ <tolle-pagination
206
+ *ngIf="paginate"
207
+ [totalRecords]="filteredData.length"
208
+ [currentPage]="currentPage"
209
+ [currentPageSize]="pageSize"
210
+ (onPageNumberChange)="updatePage()"
211
+ (onPageSizeChange)="updatePage()"
212
+ ></tolle-pagination>
213
+ </div>
214
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { 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: "ngmodule", type: FormsModule }, { 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: "component", type: PaginationComponent, selector: "tolle-pagination", inputs: ["class", "showPageLinks", "showPageOptions", "showCurrentPageInfo", "currentPageInfoTemplate", "totalRecords", "currentPageSize", "currentPage", "pageSizeOptions"], outputs: ["onPageNumberChange", "onPageSizeChange"] }, { kind: "component", type: InputComponent, selector: "tolle-input", inputs: ["type", "placeholder", "disabled", "error", "size", "containerClass", "class"] }] });
215
+ }
216
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DataTableComponent, decorators: [{
217
+ type: Component,
218
+ args: [{
219
+ selector: 'tolle-data-table',
220
+ standalone: true,
221
+ imports: [CommonModule, FormsModule, PaginationComponent, InputComponent],
222
+ template: `
223
+ <div class="space-y-4">
224
+ <div *ngIf="searchable" class="flex items-center py-2">
225
+ <tolle-input
226
+ [size]="size === 'lg' ? 'default' : 'sm'"
227
+ class="max-w-sm"
228
+ placeholder="Filter records..."
229
+ [(ngModel)]="searchTerm"
230
+ (ngModelChange)="onSearch()">
231
+ <i prefix class="ri-search-line"></i>
232
+ </tolle-input>
233
+ </div>
234
+
235
+ <div class="rounded-md border border-border bg-background overflow-hidden shadow-sm">
236
+ <table class="w-full text-sm">
237
+ <thead class="border-b bg-muted/30">
238
+ <tr>
239
+ <th *ngIf="expandable" [class]="cn('px-4', size === 'xs' ? 'w-[32px]' : 'w-[48px]')"></th>
240
+ <th *ngFor="let col of columns"
241
+ [class]="cn(
242
+ 'font-medium text-muted-foreground transition-all',
243
+ headerPaddingClass,
244
+ fontSizeClass,
245
+ col.class
246
+ )">
247
+ <div *ngIf="col.sortable; else simpleHeader" (click)="toggleSort(col.key)" class="flex items-center gap-1 cursor-pointer hover:text-foreground">
248
+ {{ col.label }}
249
+ <i [class]="getSortIcon(col.key)"></i>
250
+ </div>
251
+ <ng-template #simpleHeader>{{ col.label }}</ng-template>
252
+ </th>
253
+ </tr>
254
+ </thead>
255
+ <tbody class="divide-y divide-border">
256
+ <ng-container *ngFor="let row of pagedData; let i = index">
257
+ <tr class="hover:bg-muted/50 transition-colors">
258
+ <td *ngIf="expandable" class="px-4">
259
+ <button (click)="toggleRow(i)"
260
+ [class]="cn(
261
+ 'flex items-center justify-center rounded-md hover:bg-accent text-muted-foreground hover:text-foreground',
262
+ size === 'xs' ? 'h-6 w-6' : 'h-8 w-8'
263
+ )">
264
+ <i [class]="expandedRows.has(i) ? 'ri-arrow-down-s-line' : 'ri-arrow-right-s-line'"></i>
265
+ </button>
266
+ </td>
267
+
268
+ <td *ngFor="let col of columns"
269
+ [class]="cn(
270
+ 'align-middle transition-all',
271
+ cellPaddingClass,
272
+ fontSizeClass,
273
+ col.class
274
+ )">
275
+ <ng-container *ngIf="getTemplate(col.key) as cell; else defaultValue">
276
+ <ng-container *ngTemplateOutlet="cell.template; context: { $implicit: row[col.key], row: row }"></ng-container>
277
+ </ng-container>
278
+ <ng-template #defaultValue>
279
+ <span class="text-foreground">{{ row[col.key] }}</span>
280
+ </ng-template>
281
+ </td>
282
+ </tr>
283
+
284
+ <tr *ngIf="expandedRows.has(i)" class="bg-muted/10">
285
+ <td [attr.colspan]="columns.length + (expandable ? 1 : 0)" class="p-0">
286
+ <div class="p-6 border-b border-dashed border-border">
287
+ <ng-container *ngIf="expandedTemplate; else defaultExpanded">
288
+ <ng-container *ngTemplateOutlet="expandedTemplate; context: { row: row }"></ng-container>
289
+ </ng-container>
290
+ <ng-template #defaultExpanded>
291
+ <div class="text-xs text-muted-foreground italic">No details available.</div>
292
+ </ng-template>
293
+ </div>
294
+ </td>
295
+ </tr>
296
+ </ng-container>
297
+ </tbody>
298
+ </table>
299
+ </div>
300
+
301
+ <tolle-pagination
302
+ *ngIf="paginate"
303
+ [totalRecords]="filteredData.length"
304
+ [currentPage]="currentPage"
305
+ [currentPageSize]="pageSize"
306
+ (onPageNumberChange)="updatePage()"
307
+ (onPageSizeChange)="updatePage()"
308
+ ></tolle-pagination>
309
+ </div>
310
+ `
311
+ }]
312
+ }], propDecorators: { data: [{
313
+ type: Input
314
+ }], columns: [{
315
+ type: Input
316
+ }], searchable: [{
317
+ type: Input
318
+ }], paginate: [{
319
+ type: Input
320
+ }], pageSize: [{
321
+ type: Input
322
+ }], expandable: [{
323
+ type: Input
324
+ }], size: [{
325
+ type: Input
326
+ }], cellTemplates: [{
327
+ type: ContentChildren,
328
+ args: [TolleCellDirective]
329
+ }], expandedTemplate: [{
330
+ type: Input
331
+ }] } });
332
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-table.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/data-table.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, OnInit, OnChanges, SimpleChanges,\n  TemplateRef, ContentChildren, QueryList\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { cn } from './utils/cn';\nimport {PaginationComponent} from '@tolle/ui/pagination.component';\nimport {InputComponent} from '@tolle/ui/input.component';\nimport {TolleCellDirective} from '@tolle/ui/tolle-cell.directive';\n\nexport interface TableColumn {\n  key: string;\n  label: string;\n  sortable?: boolean;\n  class?: string;\n}\n\n@Component({\n  selector: 'tolle-data-table',\n  standalone: true,\n  imports: [CommonModule, FormsModule, PaginationComponent, InputComponent],\n  template: `\n    <div class=\"space-y-4\">\n      <div *ngIf=\"searchable\" class=\"flex items-center py-2\">\n        <tolle-input\n          [size]=\"size === 'lg' ? 'default' : 'sm'\"\n          class=\"max-w-sm\"\n          placeholder=\"Filter records...\"\n          [(ngModel)]=\"searchTerm\"\n          (ngModelChange)=\"onSearch()\">\n          <i prefix class=\"ri-search-line\"></i>\n        </tolle-input>\n      </div>\n\n      <div class=\"rounded-md border border-border bg-background overflow-hidden shadow-sm\">\n        <table class=\"w-full text-sm\">\n          <thead class=\"border-b bg-muted/30\">\n            <tr>\n              <th *ngIf=\"expandable\" [class]=\"cn('px-4', size === 'xs' ? 'w-[32px]' : 'w-[48px]')\"></th>\n              <th *ngFor=\"let col of columns\"\n                  [class]=\"cn(\n                  'font-medium text-muted-foreground transition-all',\n                  headerPaddingClass,\n                  fontSizeClass,\n                  col.class\n                )\">\n                <div *ngIf=\"col.sortable; else simpleHeader\" (click)=\"toggleSort(col.key)\" class=\"flex items-center gap-1 cursor-pointer hover:text-foreground\">\n                  {{ col.label }}\n                  <i [class]=\"getSortIcon(col.key)\"></i>\n                </div>\n                <ng-template #simpleHeader>{{ col.label }}</ng-template>\n              </th>\n            </tr>\n          </thead>\n          <tbody class=\"divide-y divide-border\">\n            <ng-container *ngFor=\"let row of pagedData; let i = index\">\n              <tr class=\"hover:bg-muted/50 transition-colors\">\n                <td *ngIf=\"expandable\" class=\"px-4\">\n                  <button (click)=\"toggleRow(i)\"\n                          [class]=\"cn(\n                      'flex items-center justify-center rounded-md hover:bg-accent text-muted-foreground hover:text-foreground',\n                      size === 'xs' ? 'h-6 w-6' : 'h-8 w-8'\n                    )\">\n                    <i [class]=\"expandedRows.has(i) ? 'ri-arrow-down-s-line' : 'ri-arrow-right-s-line'\"></i>\n                  </button>\n                </td>\n\n                <td *ngFor=\"let col of columns\"\n                    [class]=\"cn(\n                    'align-middle transition-all',\n                    cellPaddingClass,\n                    fontSizeClass,\n                    col.class\n                  )\">\n                  <ng-container *ngIf=\"getTemplate(col.key) as cell; else defaultValue\">\n                    <ng-container *ngTemplateOutlet=\"cell.template; context: { $implicit: row[col.key], row: row }\"></ng-container>\n                  </ng-container>\n                  <ng-template #defaultValue>\n                    <span class=\"text-foreground\">{{ row[col.key] }}</span>\n                  </ng-template>\n                </td>\n              </tr>\n\n              <tr *ngIf=\"expandedRows.has(i)\" class=\"bg-muted/10\">\n                <td [attr.colspan]=\"columns.length + (expandable ? 1 : 0)\" class=\"p-0\">\n                   <div class=\"p-6 border-b border-dashed border-border\">\n                      <ng-container *ngIf=\"expandedTemplate; else defaultExpanded\">\n                        <ng-container *ngTemplateOutlet=\"expandedTemplate; context: { row: row }\"></ng-container>\n                      </ng-container>\n                      <ng-template #defaultExpanded>\n                        <div class=\"text-xs text-muted-foreground italic\">No details available.</div>\n                      </ng-template>\n                   </div>\n                </td>\n              </tr>\n            </ng-container>\n          </tbody>\n        </table>\n      </div>\n\n      <tolle-pagination\n        *ngIf=\"paginate\"\n        [totalRecords]=\"filteredData.length\"\n        [currentPage]=\"currentPage\"\n        [currentPageSize]=\"pageSize\"\n        (onPageNumberChange)=\"updatePage()\"\n        (onPageSizeChange)=\"updatePage()\"\n      ></tolle-pagination>\n    </div>\n  `\n})\nexport class DataTableComponent implements OnInit, OnChanges {\n  @Input() data: any[] = [];\n  @Input() columns: TableColumn[] = [];\n  @Input() searchable = true;\n  @Input() paginate = true;\n  @Input() pageSize = 10;\n  @Input() expandable = false;\n  @Input() size: 'xs' | 'sm' | 'default' | 'lg' = 'default';\n  // Track which rows are open\n  expandedRows = new Set<number>();\n\n  // Use ContentChildren to grab the tolleCell templates from the user's HTML\n  @ContentChildren(TolleCellDirective) cellTemplates!: QueryList<TolleCellDirective>;\n\n  // Keep this as an Input for the main expansion slot\n  @Input() expandedTemplate?: TemplateRef<any>;\n\n  filteredData: any[] = [];\n  pagedData: any[] = [];\n  searchTerm = '';\n  currentPage = 1;\n  sortKey = '';\n  sortDir: 'asc' | 'desc' | null = null;\n\n  // 2. Map Size to Padding for Cells\n  get cellPaddingClass(): string {\n    switch (this.size) {\n      case 'xs': return 'p-1 px-4';    // Ultra-compact\n      case 'sm': return 'p-2 px-4';    // Dense\n      case 'lg': return 'p-6 px-4';    // Spacious\n      default:   return 'p-4';         // Standard (16px)\n    }\n  }\n\n  // 3. Map Size to Padding for Header (usually slightly shorter than cells)\n  get headerPaddingClass(): string {\n    switch (this.size) {\n      case 'xs': return 'h-7 px-4';\n      case 'sm': return 'h-9 px-4';\n      case 'lg': return 'h-14 px-4';\n      default:   return 'h-12 px-4';\n    }\n  }\n\n  // 4. Map Size to Font Sizes\n  get fontSizeClass(): string {\n    switch (this.size) {\n      case 'xs': return 'text-[11px]';\n      case 'sm': return 'text-xs';\n      case 'lg': return 'text-base';\n      default:   return 'text-sm';\n    }\n  }\n\n  protected cn = cn;\n\n  ngOnInit() { this.refreshTable(); }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['data']) { this.refreshTable(); }\n  }\n\n  // --- Search & Sort & Page Logic ---\n  // (Your existing implementation of applySearch, applySort, and updatePage goes here)\n  refreshTable() { this.applySearch(); this.applySort(); this.updatePage(); }\n  onSearch() { this.currentPage = 1; this.refreshTable(); }\n\n  private applySearch() {\n    if (!this.searchTerm) { this.filteredData = [...this.data]; return; }\n    const q = this.searchTerm.toLowerCase();\n    this.filteredData = this.data.filter(row =>\n      Object.values(row).some(val => String(val).toLowerCase().includes(q))\n    );\n  }\n\n  private applySort() {\n    if (!this.sortKey || !this.sortDir) return;\n    this.filteredData.sort((a, b) => {\n      const valA = a[this.sortKey]; const valB = b[this.sortKey];\n      if (valA < valB) return this.sortDir === 'asc' ? -1 : 1;\n      if (valA > valB) return this.sortDir === 'asc' ? 1 : -1;\n      return 0;\n    });\n  }\n\n  updatePage() {\n    if (!this.paginate) { this.pagedData = this.filteredData; return; }\n    const start = (this.currentPage - 1) * this.pageSize;\n    const end = start + this.pageSize;\n    this.pagedData = this.filteredData.slice(start, end);\n  }\n\n  // --- Helpers ---\n  toggleSort(key: string) {\n    if (this.sortKey === key) {\n      this.sortDir = this.sortDir === 'asc' ? 'desc' : this.sortDir === 'desc' ? null : 'asc';\n    } else { this.sortKey = key; this.sortDir = 'asc'; }\n    this.refreshTable();\n  }\n\n  getSortIcon(key: string) {\n    if (this.sortKey !== key || !this.sortDir) return 'ri-arrow-up-down-line opacity-30';\n    return this.sortDir === 'asc' ? 'ri-arrow-up-line' : 'ri-arrow-down-line';\n  }\n\n  toggleRow(index: number) {\n    if (this.expandedRows.has(index)) this.expandedRows.delete(index);\n    else this.expandedRows.add(index);\n  }\n\n  // Helper to find the right cell template\n  getTemplate(key: string): TolleCellDirective | undefined {\n    return this.cellTemplates?.find(t => t.name === key);\n  }\n}\n"]}
@@ -0,0 +1,232 @@
1
+ import { Component, Input, forwardRef, ViewChild, HostListener } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';
4
+ import { computePosition, flip, shift, offset, autoUpdate } from '@floating-ui/dom';
5
+ import { format, parse, isValid, startOfDay } from 'date-fns';
6
+ import { cn } from './utils/cn';
7
+ import { MaskedInputComponent } from './masked-input.component';
8
+ import { CalendarComponent } from './calendar.component';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "@angular/common";
11
+ import * as i2 from "@angular/forms";
12
+ export class DatePickerComponent {
13
+ cdr;
14
+ placeholder = 'MM/DD/YYYY';
15
+ disabled = false;
16
+ class = '';
17
+ disablePastDates = false;
18
+ triggerContainer;
19
+ popover;
20
+ value = null;
21
+ inputValue = '';
22
+ isOpen = false;
23
+ cleanupAutoUpdate;
24
+ constructor(cdr) {
25
+ this.cdr = cdr;
26
+ }
27
+ // --- Logic ---
28
+ onInputChange(str) {
29
+ if (str?.length === 10) {
30
+ const parsed = parse(str, 'MM/dd/yyyy', new Date());
31
+ if (isValid(parsed)) {
32
+ this.value = startOfDay(parsed);
33
+ this.onChange(this.value);
34
+ }
35
+ }
36
+ else if (!str) {
37
+ this.value = null;
38
+ this.onChange(null);
39
+ }
40
+ }
41
+ onCalendarChange(date) {
42
+ this.value = date;
43
+ this.inputValue = format(date, 'MM/dd/yyyy');
44
+ this.onChange(this.value);
45
+ this.close();
46
+ }
47
+ togglePopover(event) {
48
+ event.stopPropagation(); // Prevent bubbling to document
49
+ if (this.disabled)
50
+ return;
51
+ this.isOpen ? this.close() : this.open();
52
+ }
53
+ open() {
54
+ this.isOpen = true;
55
+ setTimeout(() => this.updatePosition());
56
+ }
57
+ close() {
58
+ this.isOpen = false;
59
+ if (this.cleanupAutoUpdate)
60
+ this.cleanupAutoUpdate();
61
+ }
62
+ clear(event) {
63
+ event.stopPropagation(); // CRITICAL: Stop the calendar from opening
64
+ this.value = null;
65
+ this.inputValue = '';
66
+ this.onChange(null);
67
+ this.cdr.markForCheck();
68
+ }
69
+ // --- Positioning ---
70
+ updatePosition() {
71
+ if (!this.triggerContainer || !this.popover)
72
+ return;
73
+ this.cleanupAutoUpdate = autoUpdate(this.triggerContainer.nativeElement, this.popover.nativeElement, () => {
74
+ computePosition(this.triggerContainer.nativeElement, this.popover.nativeElement, {
75
+ placement: 'bottom-end', // Aligned to the right where the icon is
76
+ middleware: [offset(4), flip(), shift({ padding: 8 })],
77
+ }).then(({ x, y }) => {
78
+ Object.assign(this.popover.nativeElement.style, {
79
+ left: `${x}px`,
80
+ top: `${y}px`,
81
+ visibility: 'visible',
82
+ });
83
+ });
84
+ });
85
+ }
86
+ onClickOutside(event) {
87
+ if (this.isOpen &&
88
+ !this.triggerContainer.nativeElement.contains(event.target) &&
89
+ !this.popover.nativeElement.contains(event.target)) {
90
+ this.close();
91
+ }
92
+ }
93
+ // --- CVA ---
94
+ onChange = () => { };
95
+ onTouched = () => { };
96
+ writeValue(val) {
97
+ if (val) {
98
+ const date = new Date(val);
99
+ if (isValid(date)) {
100
+ this.value = startOfDay(date);
101
+ this.inputValue = format(this.value, 'MM/dd/yyyy');
102
+ }
103
+ }
104
+ else {
105
+ this.value = null;
106
+ this.inputValue = '';
107
+ }
108
+ this.cdr.markForCheck();
109
+ }
110
+ registerOnChange(fn) { this.onChange = fn; }
111
+ registerOnTouched(fn) { this.onTouched = fn; }
112
+ setDisabledState(isDisabled) { this.disabled = isDisabled; }
113
+ cn = cn;
114
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DatePickerComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
115
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: DatePickerComponent, isStandalone: true, selector: "tolle-date-picker", inputs: { placeholder: "placeholder", disabled: "disabled", class: "class", disablePastDates: "disablePastDates" }, host: { listeners: { "document:mousedown": "onClickOutside($event)" } }, providers: [
116
+ {
117
+ provide: NG_VALUE_ACCESSOR,
118
+ useExisting: forwardRef(() => DatePickerComponent),
119
+ multi: true
120
+ }
121
+ ], viewQueries: [{ propertyName: "triggerContainer", first: true, predicate: ["triggerContainer"], descendants: true }, { propertyName: "popover", first: true, predicate: ["popover"], descendants: true }], ngImport: i0, template: `
122
+ <div class="relative w-full" #triggerContainer>
123
+ <tolle-masked-input
124
+ #maskInput
125
+ [mask]="'00/00/0000'"
126
+ [placeholder]="placeholder"
127
+ [disabled]="disabled"
128
+ [(ngModel)]="inputValue"
129
+ (ngModelChange)="onInputChange($event)"
130
+ [class]="cn(class)"
131
+ >
132
+ <div suffix class="flex items-center gap-1.5 cursor-pointer">
133
+ <i
134
+ *ngIf="value && !disabled"
135
+ (click)="clear($event)"
136
+ class="ri-close-line cursor-pointer text-muted-foreground hover:text-foreground transition-colors"
137
+ ></i>
138
+
139
+ <i
140
+ (click)="togglePopover($event)"
141
+ class="ri-calendar-line cursor-pointer text-muted-foreground hover:text-primary transition-colors"
142
+ ></i>
143
+ </div>
144
+ </tolle-masked-input>
145
+
146
+ <div
147
+ #popover
148
+ *ngIf="isOpen"
149
+ class="absolute bg-popover z-50 max-w-max left-0 right-0 overflow-hidden rounded-md border border-border text-popover-foreground bg-background shadow-md"
150
+ style="visibility: hidden; top: 0; left: 0;"
151
+ >
152
+ <tolle-calendar
153
+ [(ngModel)]="value"
154
+ (ngModelChange)="onCalendarChange($event)"
155
+ [disablePastDates]="disablePastDates"
156
+ ></tolle-calendar>
157
+ </div>
158
+ </div>
159
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { 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: "component", type: MaskedInputComponent, selector: "tolle-masked-input", inputs: ["mask", "placeholder", "type", "disabled", "class", "error", "size", "returnRaw"] }, { kind: "component", type: CalendarComponent, selector: "tolle-calendar", inputs: ["class", "disablePastDates"] }] });
160
+ }
161
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DatePickerComponent, decorators: [{
162
+ type: Component,
163
+ args: [{
164
+ selector: 'tolle-date-picker',
165
+ standalone: true,
166
+ imports: [CommonModule, FormsModule, MaskedInputComponent, CalendarComponent],
167
+ providers: [
168
+ {
169
+ provide: NG_VALUE_ACCESSOR,
170
+ useExisting: forwardRef(() => DatePickerComponent),
171
+ multi: true
172
+ }
173
+ ],
174
+ template: `
175
+ <div class="relative w-full" #triggerContainer>
176
+ <tolle-masked-input
177
+ #maskInput
178
+ [mask]="'00/00/0000'"
179
+ [placeholder]="placeholder"
180
+ [disabled]="disabled"
181
+ [(ngModel)]="inputValue"
182
+ (ngModelChange)="onInputChange($event)"
183
+ [class]="cn(class)"
184
+ >
185
+ <div suffix class="flex items-center gap-1.5 cursor-pointer">
186
+ <i
187
+ *ngIf="value && !disabled"
188
+ (click)="clear($event)"
189
+ class="ri-close-line cursor-pointer text-muted-foreground hover:text-foreground transition-colors"
190
+ ></i>
191
+
192
+ <i
193
+ (click)="togglePopover($event)"
194
+ class="ri-calendar-line cursor-pointer text-muted-foreground hover:text-primary transition-colors"
195
+ ></i>
196
+ </div>
197
+ </tolle-masked-input>
198
+
199
+ <div
200
+ #popover
201
+ *ngIf="isOpen"
202
+ class="absolute bg-popover z-50 max-w-max left-0 right-0 overflow-hidden rounded-md border border-border text-popover-foreground bg-background shadow-md"
203
+ style="visibility: hidden; top: 0; left: 0;"
204
+ >
205
+ <tolle-calendar
206
+ [(ngModel)]="value"
207
+ (ngModelChange)="onCalendarChange($event)"
208
+ [disablePastDates]="disablePastDates"
209
+ ></tolle-calendar>
210
+ </div>
211
+ </div>
212
+ `
213
+ }]
214
+ }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { placeholder: [{
215
+ type: Input
216
+ }], disabled: [{
217
+ type: Input
218
+ }], class: [{
219
+ type: Input
220
+ }], disablePastDates: [{
221
+ type: Input
222
+ }], triggerContainer: [{
223
+ type: ViewChild,
224
+ args: ['triggerContainer']
225
+ }], popover: [{
226
+ type: ViewChild,
227
+ args: ['popover']
228
+ }], onClickOutside: [{
229
+ type: HostListener,
230
+ args: ['document:mousedown', ['$event']]
231
+ }] } });
232
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-picker.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/date-picker.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, forwardRef, ElementRef, ViewChild, HostListener, ChangeDetectorRef\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport { computePosition, flip, shift, offset, autoUpdate } from '@floating-ui/dom';\nimport { format, parse, isValid, startOfDay } from 'date-fns';\nimport { cn } from './utils/cn';\nimport { MaskedInputComponent } from './masked-input.component';\nimport { CalendarComponent } from './calendar.component';\n\n@Component({\n  selector: 'tolle-date-picker',\n  standalone: true,\n  imports: [CommonModule, FormsModule, MaskedInputComponent, CalendarComponent],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => DatePickerComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"relative w-full\" #triggerContainer>\n      <tolle-masked-input\n        #maskInput\n        [mask]=\"'00/00/0000'\"\n        [placeholder]=\"placeholder\"\n        [disabled]=\"disabled\"\n        [(ngModel)]=\"inputValue\"\n        (ngModelChange)=\"onInputChange($event)\"\n        [class]=\"cn(class)\"\n      >\n        <div suffix class=\"flex items-center gap-1.5 cursor-pointer\">\n          <i\n            *ngIf=\"value && !disabled\"\n            (click)=\"clear($event)\"\n            class=\"ri-close-line cursor-pointer text-muted-foreground hover:text-foreground transition-colors\"\n          ></i>\n\n          <i\n            (click)=\"togglePopover($event)\"\n            class=\"ri-calendar-line cursor-pointer text-muted-foreground hover:text-primary transition-colors\"\n          ></i>\n        </div>\n      </tolle-masked-input>\n\n      <div\n        #popover\n        *ngIf=\"isOpen\"\n        class=\"absolute bg-popover z-50 max-w-max left-0 right-0 overflow-hidden rounded-md border border-border text-popover-foreground bg-background shadow-md\"\n        style=\"visibility: hidden; top: 0; left: 0;\"\n      >\n        <tolle-calendar\n          [(ngModel)]=\"value\"\n          (ngModelChange)=\"onCalendarChange($event)\"\n          [disablePastDates]=\"disablePastDates\"\n        ></tolle-calendar>\n      </div>\n    </div>\n  `\n})\nexport class DatePickerComponent implements ControlValueAccessor {\n  @Input() placeholder = 'MM/DD/YYYY';\n  @Input() disabled = false;\n  @Input() class = '';\n  @Input() disablePastDates = false;\n\n  @ViewChild('triggerContainer') triggerContainer!: ElementRef;\n  @ViewChild('popover') popover!: ElementRef;\n\n  value: Date | null = null;\n  inputValue: string = '';\n  isOpen = false;\n  cleanupAutoUpdate?: () => void;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  // --- Logic ---\n\n  onInputChange(str: string) {\n    if (str?.length === 10) {\n      const parsed = parse(str, 'MM/dd/yyyy', new Date());\n      if (isValid(parsed)) {\n        this.value = startOfDay(parsed);\n        this.onChange(this.value);\n      }\n    } else if (!str) {\n      this.value = null;\n      this.onChange(null);\n    }\n  }\n\n  onCalendarChange(date: Date) {\n    this.value = date;\n    this.inputValue = format(date, 'MM/dd/yyyy');\n    this.onChange(this.value);\n    this.close();\n  }\n\n  togglePopover(event: MouseEvent) {\n    event.stopPropagation(); // Prevent bubbling to document\n    if (this.disabled) return;\n    this.isOpen ? this.close() : this.open();\n  }\n\n  open() {\n    this.isOpen = true;\n    setTimeout(() => this.updatePosition());\n  }\n\n  close() {\n    this.isOpen = false;\n    if (this.cleanupAutoUpdate) this.cleanupAutoUpdate();\n  }\n\n  clear(event: MouseEvent) {\n    event.stopPropagation(); // CRITICAL: Stop the calendar from opening\n    this.value = null;\n    this.inputValue = '';\n    this.onChange(null);\n    this.cdr.markForCheck();\n  }\n\n  // --- Positioning ---\n\n  private updatePosition() {\n    if (!this.triggerContainer || !this.popover) return;\n\n    this.cleanupAutoUpdate = autoUpdate(\n      this.triggerContainer.nativeElement,\n      this.popover.nativeElement,\n      () => {\n        computePosition(this.triggerContainer.nativeElement, this.popover.nativeElement, {\n          placement: 'bottom-end', // Aligned to the right where the icon is\n          middleware: [offset(4), flip(), shift({ padding: 8 })],\n        }).then(({ x, y }) => {\n          Object.assign(this.popover.nativeElement.style, {\n            left: `${x}px`,\n            top: `${y}px`,\n            visibility: 'visible',\n          });\n        });\n      }\n    );\n  }\n\n  @HostListener('document:mousedown', ['$event'])\n  onClickOutside(event: MouseEvent) {\n    if (this.isOpen &&\n      !this.triggerContainer.nativeElement.contains(event.target) &&\n      !this.popover.nativeElement.contains(event.target)) {\n      this.close();\n    }\n  }\n\n  // --- CVA ---\n  onChange: any = () => {};\n  onTouched: any = () => {};\n\n  writeValue(val: any): void {\n    if (val) {\n      const date = new Date(val);\n      if (isValid(date)) {\n        this.value = startOfDay(date);\n        this.inputValue = format(this.value, 'MM/dd/yyyy');\n      }\n    } else {\n      this.value = null;\n      this.inputValue = '';\n    }\n    this.cdr.markForCheck();\n  }\n\n  registerOnChange(fn: any): void { this.onChange = fn; }\n  registerOnTouched(fn: any): void { this.onTouched = fn; }\n  setDisabledState(isDisabled: boolean): void { this.disabled = isDisabled; }\n  protected cn = cn;\n}\n"]}