@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.
- package/README.md +35 -0
- package/esm2022/lib/accordion-item.component.mjs +78 -0
- package/esm2022/lib/accordion.component.mjs +60 -0
- package/esm2022/lib/badge.component.mjs +76 -0
- package/esm2022/lib/button-group.component.mjs +25 -0
- package/esm2022/lib/button.component.mjs +70 -0
- package/esm2022/lib/calendar.component.mjs +315 -0
- package/esm2022/lib/card.component.mjs +94 -0
- package/esm2022/lib/checkbox.component.mjs +100 -0
- package/esm2022/lib/data-table.component.mjs +332 -0
- package/esm2022/lib/date-picker.component.mjs +232 -0
- package/esm2022/lib/date-range-picker.component.mjs +208 -0
- package/esm2022/lib/input.component.mjs +134 -0
- package/esm2022/lib/masked-input.component.mjs +179 -0
- package/esm2022/lib/modal-ref.mjs +31 -0
- package/esm2022/lib/modal-stack.service.mjs +26 -0
- package/esm2022/lib/modal.component.mjs +98 -0
- package/esm2022/lib/modal.mjs +27 -0
- package/esm2022/lib/modal.service.mjs +65 -0
- package/esm2022/lib/multi-select.component.mjs +231 -0
- package/esm2022/lib/pagination.component.mjs +279 -0
- package/esm2022/lib/range-calendar.component.mjs +285 -0
- package/esm2022/lib/select-group.component.mjs +28 -0
- package/esm2022/lib/select-item.component.mjs +84 -0
- package/esm2022/lib/select-separator.component.mjs +24 -0
- package/esm2022/lib/select.component.mjs +261 -0
- package/esm2022/lib/select.service.mjs +21 -0
- package/esm2022/lib/skeleton.component.mjs +34 -0
- package/esm2022/lib/switch.component.mjs +133 -0
- package/esm2022/lib/toast.service.mjs +59 -0
- package/esm2022/lib/tolle-cell.directive.mjs +22 -0
- package/esm2022/lib/tolle-config.mjs +11 -0
- package/esm2022/lib/tooltip.directive.mjs +71 -0
- package/esm2022/lib/types/date-range.mjs +2 -0
- package/esm2022/lib/utils/cn.mjs +6 -0
- package/esm2022/public-api.mjs +36 -0
- package/esm2022/tolle_-tolle-ui.mjs +5 -0
- package/fesm2022/tolle_-tolle-ui.mjs +3553 -0
- package/fesm2022/tolle_-tolle-ui.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/accordion-item.component.d.ts +13 -0
- package/lib/accordion.component.d.ts +14 -0
- package/lib/badge.component.d.ts +14 -0
- package/lib/button-group.component.d.ts +8 -0
- package/lib/button.component.d.ts +16 -0
- package/lib/calendar.component.d.ts +35 -0
- package/lib/card.component.d.ts +32 -0
- package/lib/checkbox.component.d.ts +23 -0
- package/lib/data-table.component.d.ts +45 -0
- package/lib/date-picker.component.d.ts +35 -0
- package/lib/date-range-picker.component.d.ts +36 -0
- package/lib/input.component.d.ts +27 -0
- package/lib/masked-input.component.d.ts +36 -0
- package/lib/modal-ref.d.ts +16 -0
- package/lib/modal-stack.service.d.ts +12 -0
- package/lib/modal.component.d.ts +19 -0
- package/lib/modal.d.ts +29 -0
- package/lib/modal.service.d.ts +18 -0
- package/lib/multi-select.component.d.ts +47 -0
- package/lib/pagination.component.d.ts +36 -0
- package/lib/range-calendar.component.d.ts +37 -0
- package/lib/select-group.component.d.ts +8 -0
- package/lib/select-item.component.d.ts +18 -0
- package/lib/select-separator.component.d.ts +8 -0
- package/lib/select.component.d.ts +45 -0
- package/lib/select.service.d.ts +10 -0
- package/lib/skeleton.component.d.ts +10 -0
- package/lib/switch.component.d.ts +39 -0
- package/lib/toast.service.d.ts +24 -0
- package/lib/tolle-cell.directive.d.ts +9 -0
- package/lib/tolle-config.d.ts +9 -0
- package/lib/tooltip.directive.d.ts +15 -0
- package/lib/types/date-range.d.ts +4 -0
- package/lib/utils/cn.d.ts +2 -0
- package/package.json +32 -0
- package/public-api.d.ts +32 -0
- 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"]}
|