hopesf-angular-dev-utils 1.0.0
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/LICENSE +21 -0
- package/README.md +60 -0
- package/esm2022/hopesf-angular-dev-utils.mjs +5 -0
- package/esm2022/lib/components/button/button.component.mjs +68 -0
- package/esm2022/lib/components/card/card.component.mjs +78 -0
- package/esm2022/lib/components/input/input.component.mjs +141 -0
- package/esm2022/lib/components/modal/modal.component.mjs +102 -0
- package/esm2022/lib/components/spinner/spinner.component.mjs +44 -0
- package/esm2022/lib/components/table/table.component.mjs +240 -0
- package/esm2022/lib/models/types.mjs +2 -0
- package/esm2022/lib/services/modal.service.mjs +102 -0
- package/esm2022/public-api.mjs +15 -0
- package/fesm2022/hopesf-angular-dev-utils.mjs +765 -0
- package/fesm2022/hopesf-angular-dev-utils.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/button/button.component.d.ts +16 -0
- package/lib/components/card/card.component.d.ts +12 -0
- package/lib/components/input/input.component.d.ts +27 -0
- package/lib/components/modal/modal.component.d.ts +17 -0
- package/lib/components/spinner/spinner.component.d.ts +11 -0
- package/lib/components/table/table.component.d.ts +29 -0
- package/lib/models/types.d.ts +27 -0
- package/lib/services/modal.service.d.ts +21 -0
- package/package.json +49 -0
- package/public-api.d.ts +8 -0
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ContentChild } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
export class TableComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.data = [];
|
|
8
|
+
this.columns = [];
|
|
9
|
+
this.config = {
|
|
10
|
+
sortable: true,
|
|
11
|
+
pageable: true,
|
|
12
|
+
pageSize: 10,
|
|
13
|
+
striped: true,
|
|
14
|
+
hoverable: true
|
|
15
|
+
};
|
|
16
|
+
this.rowClicked = new EventEmitter();
|
|
17
|
+
this.sortChanged = new EventEmitter();
|
|
18
|
+
this.sortColumn = '';
|
|
19
|
+
this.sortDirection = 'asc';
|
|
20
|
+
this.currentPage = 1;
|
|
21
|
+
this.hasEmptyState = false;
|
|
22
|
+
}
|
|
23
|
+
ngOnInit() {
|
|
24
|
+
this.config = { ...this.getDefaultConfig(), ...this.config };
|
|
25
|
+
}
|
|
26
|
+
get sortedData() {
|
|
27
|
+
if (!this.config.sortable || !this.sortColumn) {
|
|
28
|
+
return this.data;
|
|
29
|
+
}
|
|
30
|
+
return [...this.data].sort((a, b) => {
|
|
31
|
+
const aVal = this.getNestedValue(a, this.sortColumn);
|
|
32
|
+
const bVal = this.getNestedValue(b, this.sortColumn);
|
|
33
|
+
if (aVal < bVal)
|
|
34
|
+
return this.sortDirection === 'asc' ? -1 : 1;
|
|
35
|
+
if (aVal > bVal)
|
|
36
|
+
return this.sortDirection === 'asc' ? 1 : -1;
|
|
37
|
+
return 0;
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
get paginatedData() {
|
|
41
|
+
if (!this.config.pageable) {
|
|
42
|
+
return this.sortedData;
|
|
43
|
+
}
|
|
44
|
+
const start = (this.currentPage - 1) * (this.config.pageSize || 10);
|
|
45
|
+
const end = start + (this.config.pageSize || 10);
|
|
46
|
+
return this.sortedData.slice(start, end);
|
|
47
|
+
}
|
|
48
|
+
get totalPages() {
|
|
49
|
+
return Math.ceil(this.data.length / (this.config.pageSize || 10));
|
|
50
|
+
}
|
|
51
|
+
onSort(column) {
|
|
52
|
+
if (!column.sortable || !this.config.sortable)
|
|
53
|
+
return;
|
|
54
|
+
if (this.sortColumn === column.key) {
|
|
55
|
+
this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';
|
|
56
|
+
}
|
|
57
|
+
else {
|
|
58
|
+
this.sortColumn = column.key;
|
|
59
|
+
this.sortDirection = 'asc';
|
|
60
|
+
}
|
|
61
|
+
this.sortChanged.emit({ column: column.key, direction: this.sortDirection });
|
|
62
|
+
}
|
|
63
|
+
goToPage(page) {
|
|
64
|
+
if (page >= 1 && page <= this.totalPages) {
|
|
65
|
+
this.currentPage = page;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
getCellValue(row, column) {
|
|
69
|
+
if (column.cellTemplate) {
|
|
70
|
+
return column.cellTemplate(row);
|
|
71
|
+
}
|
|
72
|
+
return this.getNestedValue(row, column.key);
|
|
73
|
+
}
|
|
74
|
+
getNestedValue(obj, path) {
|
|
75
|
+
return path.split('.').reduce((current, prop) => current?.[prop], obj);
|
|
76
|
+
}
|
|
77
|
+
getDefaultConfig() {
|
|
78
|
+
return {
|
|
79
|
+
sortable: true,
|
|
80
|
+
pageable: true,
|
|
81
|
+
pageSize: 10,
|
|
82
|
+
striped: true,
|
|
83
|
+
hoverable: true
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
87
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TableComponent, isStandalone: true, selector: "adu-table", inputs: { data: "data", columns: "columns", config: "config" }, outputs: { rowClicked: "rowClicked", sortChanged: "sortChanged" }, queries: [{ propertyName: "cellTemplate", first: true, predicate: ["cellTemplate"], descendants: true }], ngImport: i0, template: `
|
|
88
|
+
<div class="adu-table-container">
|
|
89
|
+
<table class="adu-table" [class.adu-table-striped]="config.striped" [class.adu-table-hoverable]="config.hoverable">
|
|
90
|
+
<thead class="adu-table-header">
|
|
91
|
+
<tr>
|
|
92
|
+
<th
|
|
93
|
+
*ngFor="let column of columns"
|
|
94
|
+
[style.width]="column.width"
|
|
95
|
+
[class.adu-table-sortable]="column.sortable && config.sortable"
|
|
96
|
+
(click)="onSort(column)"
|
|
97
|
+
class="adu-table-th"
|
|
98
|
+
>
|
|
99
|
+
<div class="flex items-center justify-between">
|
|
100
|
+
<span>{{ column.label }}</span>
|
|
101
|
+
<span *ngIf="column.sortable && config.sortable" class="adu-table-sort-icon">
|
|
102
|
+
<span *ngIf="sortColumn === column.key">
|
|
103
|
+
{{ sortDirection === 'asc' ? '↑' : '↓' }}
|
|
104
|
+
</span>
|
|
105
|
+
<span *ngIf="sortColumn !== column.key" class="text-gray-300">↕</span>
|
|
106
|
+
</span>
|
|
107
|
+
</div>
|
|
108
|
+
</th>
|
|
109
|
+
</tr>
|
|
110
|
+
</thead>
|
|
111
|
+
<tbody class="adu-table-body">
|
|
112
|
+
<tr *ngFor="let row of paginatedData; let i = index" class="adu-table-row">
|
|
113
|
+
<td *ngFor="let column of columns" class="adu-table-td">
|
|
114
|
+
<ng-container *ngIf="cellTemplate; else defaultCell">
|
|
115
|
+
<ng-container *ngTemplateOutlet="cellTemplate; context: { $implicit: row, column: column }"></ng-container>
|
|
116
|
+
</ng-container>
|
|
117
|
+
<ng-template #defaultCell>
|
|
118
|
+
{{ getCellValue(row, column) }}
|
|
119
|
+
</ng-template>
|
|
120
|
+
</td>
|
|
121
|
+
</tr>
|
|
122
|
+
<tr *ngIf="paginatedData.length === 0" class="adu-table-empty">
|
|
123
|
+
<td [attr.colspan]="columns.length" class="text-center py-8 text-gray-500">
|
|
124
|
+
<ng-content select="[empty-state]"></ng-content>
|
|
125
|
+
<span *ngIf="!hasEmptyState">No data available</span>
|
|
126
|
+
</td>
|
|
127
|
+
</tr>
|
|
128
|
+
</tbody>
|
|
129
|
+
</table>
|
|
130
|
+
|
|
131
|
+
<!-- Pagination -->
|
|
132
|
+
<div *ngIf="config.pageable && totalPages > 1" class="adu-table-pagination">
|
|
133
|
+
<button
|
|
134
|
+
class="adu-pagination-btn"
|
|
135
|
+
[disabled]="currentPage === 1"
|
|
136
|
+
(click)="goToPage(currentPage - 1)"
|
|
137
|
+
>
|
|
138
|
+
Previous
|
|
139
|
+
</button>
|
|
140
|
+
|
|
141
|
+
<div class="adu-pagination-info">
|
|
142
|
+
Page {{ currentPage }} of {{ totalPages }}
|
|
143
|
+
</div>
|
|
144
|
+
|
|
145
|
+
<button
|
|
146
|
+
class="adu-pagination-btn"
|
|
147
|
+
[disabled]="currentPage === totalPages"
|
|
148
|
+
(click)="goToPage(currentPage + 1)"
|
|
149
|
+
>
|
|
150
|
+
Next
|
|
151
|
+
</button>
|
|
152
|
+
</div>
|
|
153
|
+
</div>
|
|
154
|
+
`, isInline: true, styles: [".adu-table-container{width:100%;overflow-x:auto;border-radius:.5rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.adu-table{width:100%;border-collapse:collapse}.adu-table-header{border-bottom-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-table-th{padding:.75rem 1.5rem;text-align:left;font-size:.75rem;line-height:1rem;font-weight:500;text-transform:uppercase;letter-spacing:.05em;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.adu-table-sortable{cursor:pointer;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.adu-table-sortable:hover{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.adu-table-sort-icon{margin-left:.5rem;display:inline-block}.adu-table-body>:not([hidden])~:not([hidden]){--tw-divide-y-reverse: 0;border-top-width:calc(1px * calc(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px * var(--tw-divide-y-reverse));--tw-divide-opacity: 1;border-color:rgb(229 231 235 / var(--tw-divide-opacity, 1))}.adu-table-row{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.adu-table-striped .adu-table-row:nth-child(2n){--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-table-hoverable .adu-table-row:hover{--tw-bg-opacity: 1;background-color:rgb(239 246 255 / var(--tw-bg-opacity, 1))}.adu-table-td{padding:1rem 1.5rem;font-size:.875rem;line-height:1.25rem;--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.adu-table-empty{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-table-pagination{display:flex;align-items:center;justify-content:space-between;border-top-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1));padding:.75rem 1.5rem}.adu-pagination-btn{border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem 1rem;font-size:.875rem;line-height:1.25rem;font-weight:500;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1));transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.adu-pagination-btn:hover{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-pagination-btn:disabled{cursor:not-allowed;opacity:.5}.adu-pagination-info{font-size:.875rem;line-height:1.25rem;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}\n"], 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"] }] }); }
|
|
155
|
+
}
|
|
156
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TableComponent, decorators: [{
|
|
157
|
+
type: Component,
|
|
158
|
+
args: [{ selector: 'adu-table', standalone: true, imports: [CommonModule], template: `
|
|
159
|
+
<div class="adu-table-container">
|
|
160
|
+
<table class="adu-table" [class.adu-table-striped]="config.striped" [class.adu-table-hoverable]="config.hoverable">
|
|
161
|
+
<thead class="adu-table-header">
|
|
162
|
+
<tr>
|
|
163
|
+
<th
|
|
164
|
+
*ngFor="let column of columns"
|
|
165
|
+
[style.width]="column.width"
|
|
166
|
+
[class.adu-table-sortable]="column.sortable && config.sortable"
|
|
167
|
+
(click)="onSort(column)"
|
|
168
|
+
class="adu-table-th"
|
|
169
|
+
>
|
|
170
|
+
<div class="flex items-center justify-between">
|
|
171
|
+
<span>{{ column.label }}</span>
|
|
172
|
+
<span *ngIf="column.sortable && config.sortable" class="adu-table-sort-icon">
|
|
173
|
+
<span *ngIf="sortColumn === column.key">
|
|
174
|
+
{{ sortDirection === 'asc' ? '↑' : '↓' }}
|
|
175
|
+
</span>
|
|
176
|
+
<span *ngIf="sortColumn !== column.key" class="text-gray-300">↕</span>
|
|
177
|
+
</span>
|
|
178
|
+
</div>
|
|
179
|
+
</th>
|
|
180
|
+
</tr>
|
|
181
|
+
</thead>
|
|
182
|
+
<tbody class="adu-table-body">
|
|
183
|
+
<tr *ngFor="let row of paginatedData; let i = index" class="adu-table-row">
|
|
184
|
+
<td *ngFor="let column of columns" class="adu-table-td">
|
|
185
|
+
<ng-container *ngIf="cellTemplate; else defaultCell">
|
|
186
|
+
<ng-container *ngTemplateOutlet="cellTemplate; context: { $implicit: row, column: column }"></ng-container>
|
|
187
|
+
</ng-container>
|
|
188
|
+
<ng-template #defaultCell>
|
|
189
|
+
{{ getCellValue(row, column) }}
|
|
190
|
+
</ng-template>
|
|
191
|
+
</td>
|
|
192
|
+
</tr>
|
|
193
|
+
<tr *ngIf="paginatedData.length === 0" class="adu-table-empty">
|
|
194
|
+
<td [attr.colspan]="columns.length" class="text-center py-8 text-gray-500">
|
|
195
|
+
<ng-content select="[empty-state]"></ng-content>
|
|
196
|
+
<span *ngIf="!hasEmptyState">No data available</span>
|
|
197
|
+
</td>
|
|
198
|
+
</tr>
|
|
199
|
+
</tbody>
|
|
200
|
+
</table>
|
|
201
|
+
|
|
202
|
+
<!-- Pagination -->
|
|
203
|
+
<div *ngIf="config.pageable && totalPages > 1" class="adu-table-pagination">
|
|
204
|
+
<button
|
|
205
|
+
class="adu-pagination-btn"
|
|
206
|
+
[disabled]="currentPage === 1"
|
|
207
|
+
(click)="goToPage(currentPage - 1)"
|
|
208
|
+
>
|
|
209
|
+
Previous
|
|
210
|
+
</button>
|
|
211
|
+
|
|
212
|
+
<div class="adu-pagination-info">
|
|
213
|
+
Page {{ currentPage }} of {{ totalPages }}
|
|
214
|
+
</div>
|
|
215
|
+
|
|
216
|
+
<button
|
|
217
|
+
class="adu-pagination-btn"
|
|
218
|
+
[disabled]="currentPage === totalPages"
|
|
219
|
+
(click)="goToPage(currentPage + 1)"
|
|
220
|
+
>
|
|
221
|
+
Next
|
|
222
|
+
</button>
|
|
223
|
+
</div>
|
|
224
|
+
</div>
|
|
225
|
+
`, styles: [".adu-table-container{width:100%;overflow-x:auto;border-radius:.5rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.adu-table{width:100%;border-collapse:collapse}.adu-table-header{border-bottom-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-table-th{padding:.75rem 1.5rem;text-align:left;font-size:.75rem;line-height:1rem;font-weight:500;text-transform:uppercase;letter-spacing:.05em;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}.adu-table-sortable{cursor:pointer;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.adu-table-sortable:hover{--tw-bg-opacity: 1;background-color:rgb(243 244 246 / var(--tw-bg-opacity, 1))}.adu-table-sort-icon{margin-left:.5rem;display:inline-block}.adu-table-body>:not([hidden])~:not([hidden]){--tw-divide-y-reverse: 0;border-top-width:calc(1px * calc(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px * var(--tw-divide-y-reverse));--tw-divide-opacity: 1;border-color:rgb(229 231 235 / var(--tw-divide-opacity, 1))}.adu-table-row{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.adu-table-striped .adu-table-row:nth-child(2n){--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-table-hoverable .adu-table-row:hover{--tw-bg-opacity: 1;background-color:rgb(239 246 255 / var(--tw-bg-opacity, 1))}.adu-table-td{padding:1rem 1.5rem;font-size:.875rem;line-height:1.25rem;--tw-text-opacity: 1;color:rgb(17 24 39 / var(--tw-text-opacity, 1))}.adu-table-empty{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-table-pagination{display:flex;align-items:center;justify-content:space-between;border-top-width:1px;--tw-border-opacity: 1;border-color:rgb(229 231 235 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1));padding:.75rem 1.5rem}.adu-pagination-btn{border-radius:.375rem;border-width:1px;--tw-border-opacity: 1;border-color:rgb(209 213 219 / var(--tw-border-opacity, 1));--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1));padding:.5rem 1rem;font-size:.875rem;line-height:1.25rem;font-weight:500;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1));transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.adu-pagination-btn:hover{--tw-bg-opacity: 1;background-color:rgb(249 250 251 / var(--tw-bg-opacity, 1))}.adu-pagination-btn:disabled{cursor:not-allowed;opacity:.5}.adu-pagination-info{font-size:.875rem;line-height:1.25rem;--tw-text-opacity: 1;color:rgb(55 65 81 / var(--tw-text-opacity, 1))}\n"] }]
|
|
226
|
+
}], propDecorators: { data: [{
|
|
227
|
+
type: Input
|
|
228
|
+
}], columns: [{
|
|
229
|
+
type: Input
|
|
230
|
+
}], config: [{
|
|
231
|
+
type: Input
|
|
232
|
+
}], cellTemplate: [{
|
|
233
|
+
type: ContentChild,
|
|
234
|
+
args: ['cellTemplate']
|
|
235
|
+
}], rowClicked: [{
|
|
236
|
+
type: Output
|
|
237
|
+
}], sortChanged: [{
|
|
238
|
+
type: Output
|
|
239
|
+
}] } });
|
|
240
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/table/table.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ContentChild, TemplateRef } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { TableColumn, TableConfig } from '../../models/types';\n\n@Component({\n  selector: 'adu-table',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <div class=\"adu-table-container\">\n      <table class=\"adu-table\" [class.adu-table-striped]=\"config.striped\" [class.adu-table-hoverable]=\"config.hoverable\">\n        <thead class=\"adu-table-header\">\n          <tr>\n            <th \n              *ngFor=\"let column of columns\"\n              [style.width]=\"column.width\"\n              [class.adu-table-sortable]=\"column.sortable && config.sortable\"\n              (click)=\"onSort(column)\"\n              class=\"adu-table-th\"\n            >\n              <div class=\"flex items-center justify-between\">\n                <span>{{ column.label }}</span>\n                <span *ngIf=\"column.sortable && config.sortable\" class=\"adu-table-sort-icon\">\n                  <span *ngIf=\"sortColumn === column.key\">\n                    {{ sortDirection === 'asc' ? '↑' : '↓' }}\n                  </span>\n                  <span *ngIf=\"sortColumn !== column.key\" class=\"text-gray-300\">↕</span>\n                </span>\n              </div>\n            </th>\n          </tr>\n        </thead>\n        <tbody class=\"adu-table-body\">\n          <tr *ngFor=\"let row of paginatedData; let i = index\" class=\"adu-table-row\">\n            <td *ngFor=\"let column of columns\" class=\"adu-table-td\">\n              <ng-container *ngIf=\"cellTemplate; else defaultCell\">\n                <ng-container *ngTemplateOutlet=\"cellTemplate; context: { $implicit: row, column: column }\"></ng-container>\n              </ng-container>\n              <ng-template #defaultCell>\n                {{ getCellValue(row, column) }}\n              </ng-template>\n            </td>\n          </tr>\n          <tr *ngIf=\"paginatedData.length === 0\" class=\"adu-table-empty\">\n            <td [attr.colspan]=\"columns.length\" class=\"text-center py-8 text-gray-500\">\n              <ng-content select=\"[empty-state]\"></ng-content>\n              <span *ngIf=\"!hasEmptyState\">No data available</span>\n            </td>\n          </tr>\n        </tbody>\n      </table>\n\n      <!-- Pagination -->\n      <div *ngIf=\"config.pageable && totalPages > 1\" class=\"adu-table-pagination\">\n        <button \n          class=\"adu-pagination-btn\" \n          [disabled]=\"currentPage === 1\"\n          (click)=\"goToPage(currentPage - 1)\"\n        >\n          Previous\n        </button>\n\n        <div class=\"adu-pagination-info\">\n          Page {{ currentPage }} of {{ totalPages }}\n        </div>\n\n        <button \n          class=\"adu-pagination-btn\"\n          [disabled]=\"currentPage === totalPages\"\n          (click)=\"goToPage(currentPage + 1)\"\n        >\n          Next\n        </button>\n      </div>\n    </div>\n  `,\n  styles: [`\n    .adu-table-container {\n      @apply w-full overflow-x-auto bg-white rounded-lg border border-gray-200;\n    }\n\n    .adu-table {\n      @apply w-full border-collapse;\n    }\n\n    .adu-table-header {\n      @apply bg-gray-50 border-b border-gray-200;\n    }\n\n    .adu-table-th {\n      @apply px-6 py-3 text-left text-xs font-medium text-gray-700 uppercase tracking-wider;\n    }\n\n    .adu-table-sortable {\n      @apply cursor-pointer hover:bg-gray-100 transition-colors;\n    }\n\n    .adu-table-sort-icon {\n      @apply ml-2 inline-block;\n    }\n\n    .adu-table-body {\n      @apply divide-y divide-gray-200;\n    }\n\n    .adu-table-row {\n      @apply transition-colors;\n    }\n\n    .adu-table-striped .adu-table-row:nth-child(even) {\n      @apply bg-gray-50;\n    }\n\n    .adu-table-hoverable .adu-table-row:hover {\n      @apply bg-blue-50;\n    }\n\n    .adu-table-td {\n      @apply px-6 py-4 text-sm text-gray-900;\n    }\n\n    .adu-table-empty {\n      @apply bg-gray-50;\n    }\n\n    .adu-table-pagination {\n      @apply flex items-center justify-between px-6 py-3 border-t border-gray-200 bg-gray-50;\n    }\n\n    .adu-pagination-btn {\n      @apply px-4 py-2 text-sm font-medium text-gray-700 bg-white border border-gray-300 rounded-md;\n      @apply hover:bg-gray-50 disabled:opacity-50 disabled:cursor-not-allowed transition-colors;\n    }\n\n    .adu-pagination-info {\n      @apply text-sm text-gray-700;\n    }\n  `]\n})\nexport class TableComponent<T = any> {\n  @Input() data: T[] = [];\n  @Input() columns: TableColumn<T>[] = [];\n  @Input() config: TableConfig = {\n    sortable: true,\n    pageable: true,\n    pageSize: 10,\n    striped: true,\n    hoverable: true\n  };\n\n  @ContentChild('cellTemplate') cellTemplate?: TemplateRef<any>;\n  @Output() rowClicked = new EventEmitter<T>();\n  @Output() sortChanged = new EventEmitter<{ column: string; direction: 'asc' | 'desc' }>();\n\n  sortColumn = '';\n  sortDirection: 'asc' | 'desc' = 'asc';\n  currentPage = 1;\n  hasEmptyState = false;\n\n  ngOnInit() {\n    this.config = { ...this.getDefaultConfig(), ...this.config };\n  }\n\n  get sortedData(): T[] {\n    if (!this.config.sortable || !this.sortColumn) {\n      return this.data;\n    }\n\n    return [...this.data].sort((a, b) => {\n      const aVal = this.getNestedValue(a, this.sortColumn);\n      const bVal = this.getNestedValue(b, this.sortColumn);\n\n      if (aVal < bVal) return this.sortDirection === 'asc' ? -1 : 1;\n      if (aVal > bVal) return this.sortDirection === 'asc' ? 1 : -1;\n      return 0;\n    });\n  }\n\n  get paginatedData(): T[] {\n    if (!this.config.pageable) {\n      return this.sortedData;\n    }\n\n    const start = (this.currentPage - 1) * (this.config.pageSize || 10);\n    const end = start + (this.config.pageSize || 10);\n    return this.sortedData.slice(start, end);\n  }\n\n  get totalPages(): number {\n    return Math.ceil(this.data.length / (this.config.pageSize || 10));\n  }\n\n  onSort(column: TableColumn<T>): void {\n    if (!column.sortable || !this.config.sortable) return;\n\n    if (this.sortColumn === column.key) {\n      this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';\n    } else {\n      this.sortColumn = column.key;\n      this.sortDirection = 'asc';\n    }\n\n    this.sortChanged.emit({ column: column.key, direction: this.sortDirection });\n  }\n\n  goToPage(page: number): void {\n    if (page >= 1 && page <= this.totalPages) {\n      this.currentPage = page;\n    }\n  }\n\n  getCellValue(row: T, column: TableColumn<T>): any {\n    if (column.cellTemplate) {\n      return column.cellTemplate(row);\n    }\n    return this.getNestedValue(row, column.key);\n  }\n\n  private getNestedValue(obj: any, path: string): any {\n    return path.split('.').reduce((current, prop) => current?.[prop], obj);\n  }\n\n  private getDefaultConfig(): TableConfig {\n    return {\n      sortable: true,\n      pageable: true,\n      pageSize: 10,\n      striped: true,\n      hoverable: true\n    };\n  }\n}\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL21vZGVscy90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgQnV0dG9uVmFyaWFudCA9ICdwcmltYXJ5JyB8ICdzZWNvbmRhcnknIHwgJ3N1Y2Nlc3MnIHwgJ2RhbmdlcicgfCAnd2FybmluZycgfCAnZ2hvc3QnO1xuZXhwb3J0IHR5cGUgQnV0dG9uU2l6ZSA9ICdzbScgfCAnbWQnIHwgJ2xnJztcblxuZXhwb3J0IHR5cGUgSW5wdXRUeXBlID0gJ3RleHQnIHwgJ2VtYWlsJyB8ICdwYXNzd29yZCcgfCAnbnVtYmVyJyB8ICd0ZWwnIHwgJ3VybCc7XG5leHBvcnQgdHlwZSBJbnB1dFNpemUgPSAnc20nIHwgJ21kJyB8ICdsZyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGFibGVDb2x1bW48VCA9IGFueT4ge1xuICBrZXk6IHN0cmluZztcbiAgbGFiZWw6IHN0cmluZztcbiAgc29ydGFibGU/OiBib29sZWFuO1xuICB3aWR0aD86IHN0cmluZztcbiAgY2VsbFRlbXBsYXRlPzogKHJvdzogVCkgPT4gc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFRhYmxlQ29uZmlnIHtcbiAgc29ydGFibGU/OiBib29sZWFuO1xuICBwYWdlYWJsZT86IGJvb2xlYW47XG4gIHBhZ2VTaXplPzogbnVtYmVyO1xuICBzdHJpcGVkPzogYm9vbGVhbjtcbiAgaG92ZXJhYmxlPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb2RhbENvbmZpZyB7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBzaXplPzogJ3NtJyB8ICdtZCcgfCAnbGcnIHwgJ3hsJyB8ICdmdWxsJztcbiAgY2xvc2VPbkJhY2tkcm9wPzogYm9vbGVhbjtcbiAgY2xvc2VPbkVzY2FwZT86IGJvb2xlYW47XG4gIHNob3dDbG9zZUJ1dHRvbj86IGJvb2xlYW47XG59XG5cbmV4cG9ydCB0eXBlIFNwaW5uZXJTaXplID0gJ3NtJyB8ICdtZCcgfCAnbGcnO1xuZXhwb3J0IHR5cGUgU3Bpbm5lckNvbG9yID0gJ3ByaW1hcnknIHwgJ3NlY29uZGFyeScgfCAnd2hpdGUnO1xuIl19
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { Injectable, createComponent } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class ModalService {
|
|
5
|
+
constructor(appRef, injector) {
|
|
6
|
+
this.appRef = appRef;
|
|
7
|
+
this.injector = injector;
|
|
8
|
+
this.modalComponentRef = null;
|
|
9
|
+
this.closeSubject = new Subject();
|
|
10
|
+
}
|
|
11
|
+
open(component, config = {}) {
|
|
12
|
+
// Close existing modal if any
|
|
13
|
+
if (this.modalComponentRef) {
|
|
14
|
+
this.close();
|
|
15
|
+
}
|
|
16
|
+
// Create modal wrapper
|
|
17
|
+
const modalWrapper = document.createElement('div');
|
|
18
|
+
modalWrapper.className = this.getModalWrapperClasses(config);
|
|
19
|
+
// Create backdrop
|
|
20
|
+
const backdrop = document.createElement('div');
|
|
21
|
+
backdrop.className = 'adu-modal-backdrop';
|
|
22
|
+
if (config.closeOnBackdrop !== false) {
|
|
23
|
+
backdrop.onclick = () => this.close();
|
|
24
|
+
}
|
|
25
|
+
// Create modal content
|
|
26
|
+
const modalContent = document.createElement('div');
|
|
27
|
+
modalContent.className = this.getModalContentClasses(config);
|
|
28
|
+
modalContent.onclick = (e) => e.stopPropagation();
|
|
29
|
+
// Create close button if needed
|
|
30
|
+
if (config.showCloseButton !== false) {
|
|
31
|
+
const closeBtn = document.createElement('button');
|
|
32
|
+
closeBtn.className = 'adu-modal-close';
|
|
33
|
+
closeBtn.innerHTML = '×';
|
|
34
|
+
closeBtn.onclick = () => this.close();
|
|
35
|
+
modalContent.appendChild(closeBtn);
|
|
36
|
+
}
|
|
37
|
+
// Add title if provided
|
|
38
|
+
if (config.title) {
|
|
39
|
+
const titleEl = document.createElement('div');
|
|
40
|
+
titleEl.className = 'adu-modal-title';
|
|
41
|
+
titleEl.textContent = config.title;
|
|
42
|
+
modalContent.appendChild(titleEl);
|
|
43
|
+
}
|
|
44
|
+
// Create component container
|
|
45
|
+
const componentContainer = document.createElement('div');
|
|
46
|
+
componentContainer.className = 'adu-modal-body';
|
|
47
|
+
modalContent.appendChild(componentContainer);
|
|
48
|
+
// Assemble modal
|
|
49
|
+
modalWrapper.appendChild(backdrop);
|
|
50
|
+
modalWrapper.appendChild(modalContent);
|
|
51
|
+
document.body.appendChild(modalWrapper);
|
|
52
|
+
// Create and attach component
|
|
53
|
+
this.modalComponentRef = createComponent(component, {
|
|
54
|
+
environmentInjector: this.injector,
|
|
55
|
+
hostElement: componentContainer
|
|
56
|
+
});
|
|
57
|
+
this.appRef.attachView(this.modalComponentRef.hostView);
|
|
58
|
+
// Handle ESC key
|
|
59
|
+
if (config.closeOnEscape !== false) {
|
|
60
|
+
const escHandler = (e) => {
|
|
61
|
+
if (e.key === 'Escape') {
|
|
62
|
+
this.close();
|
|
63
|
+
document.removeEventListener('keydown', escHandler);
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
document.addEventListener('keydown', escHandler);
|
|
67
|
+
}
|
|
68
|
+
return {
|
|
69
|
+
close: (result) => this.close(result),
|
|
70
|
+
afterClosed: () => this.closeSubject
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
close(result) {
|
|
74
|
+
if (this.modalComponentRef) {
|
|
75
|
+
this.appRef.detachView(this.modalComponentRef.hostView);
|
|
76
|
+
this.modalComponentRef.destroy();
|
|
77
|
+
this.modalComponentRef = null;
|
|
78
|
+
// Remove modal from DOM
|
|
79
|
+
const modals = document.querySelectorAll('.adu-modal-wrapper');
|
|
80
|
+
modals.forEach(modal => modal.remove());
|
|
81
|
+
this.closeSubject.next(result);
|
|
82
|
+
this.closeSubject.complete();
|
|
83
|
+
this.closeSubject = new Subject();
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
getModalWrapperClasses(config) {
|
|
87
|
+
return 'adu-modal-wrapper';
|
|
88
|
+
}
|
|
89
|
+
getModalContentClasses(config) {
|
|
90
|
+
const size = config.size || 'md';
|
|
91
|
+
return `adu-modal-content adu-modal-${size}`;
|
|
92
|
+
}
|
|
93
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ModalService, deps: [{ token: i0.ApplicationRef }, { token: i0.EnvironmentInjector }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
94
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ModalService, providedIn: 'root' }); }
|
|
95
|
+
}
|
|
96
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ModalService, decorators: [{
|
|
97
|
+
type: Injectable,
|
|
98
|
+
args: [{
|
|
99
|
+
providedIn: 'root'
|
|
100
|
+
}]
|
|
101
|
+
}], ctorParameters: () => [{ type: i0.ApplicationRef }, { type: i0.EnvironmentInjector }] });
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.service.js","sourceRoot":"","sources":["../../../../../src/lib/services/modal.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, ComponentRef, ApplicationRef, createComponent, EnvironmentInjector, Type } from '@angular/core';\nimport { Subject } from 'rxjs';\nimport { ModalConfig } from '../models/types';\n\nexport interface ModalRef<T = any> {\n  close: (result?: T) => void;\n  afterClosed: () => Subject<T | undefined>;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ModalService {\n  private modalComponentRef: ComponentRef<any> | null = null;\n  private closeSubject = new Subject<any>();\n\n  constructor(\n    private appRef: ApplicationRef,\n    private injector: EnvironmentInjector\n  ) { }\n\n  open<T, R = any>(component: Type<T>, config: ModalConfig = {}): ModalRef<R> {\n    // Close existing modal if any\n    if (this.modalComponentRef) {\n      this.close();\n    }\n\n    // Create modal wrapper\n    const modalWrapper = document.createElement('div');\n    modalWrapper.className = this.getModalWrapperClasses(config);\n\n    // Create backdrop\n    const backdrop = document.createElement('div');\n    backdrop.className = 'adu-modal-backdrop';\n    if (config.closeOnBackdrop !== false) {\n      backdrop.onclick = () => this.close();\n    }\n\n    // Create modal content\n    const modalContent = document.createElement('div');\n    modalContent.className = this.getModalContentClasses(config);\n    modalContent.onclick = (e) => e.stopPropagation();\n\n    // Create close button if needed\n    if (config.showCloseButton !== false) {\n      const closeBtn = document.createElement('button');\n      closeBtn.className = 'adu-modal-close';\n      closeBtn.innerHTML = '×';\n      closeBtn.onclick = () => this.close();\n      modalContent.appendChild(closeBtn);\n    }\n\n    // Add title if provided\n    if (config.title) {\n      const titleEl = document.createElement('div');\n      titleEl.className = 'adu-modal-title';\n      titleEl.textContent = config.title;\n      modalContent.appendChild(titleEl);\n    }\n\n    // Create component container\n    const componentContainer = document.createElement('div');\n    componentContainer.className = 'adu-modal-body';\n    modalContent.appendChild(componentContainer);\n\n    // Assemble modal\n    modalWrapper.appendChild(backdrop);\n    modalWrapper.appendChild(modalContent);\n    document.body.appendChild(modalWrapper);\n\n    // Create and attach component\n    this.modalComponentRef = createComponent(component, {\n      environmentInjector: this.injector,\n      hostElement: componentContainer\n    });\n\n    this.appRef.attachView(this.modalComponentRef.hostView);\n\n    // Handle ESC key\n    if (config.closeOnEscape !== false) {\n      const escHandler = (e: KeyboardEvent) => {\n        if (e.key === 'Escape') {\n          this.close();\n          document.removeEventListener('keydown', escHandler);\n        }\n      };\n      document.addEventListener('keydown', escHandler);\n    }\n\n    return {\n      close: (result?: R) => this.close(result),\n      afterClosed: () => this.closeSubject as Subject<R | undefined>\n    };\n  }\n\n  close(result?: any): void {\n    if (this.modalComponentRef) {\n      this.appRef.detachView(this.modalComponentRef.hostView);\n      this.modalComponentRef.destroy();\n      this.modalComponentRef = null;\n\n      // Remove modal from DOM\n      const modals = document.querySelectorAll('.adu-modal-wrapper');\n      modals.forEach(modal => modal.remove());\n\n      this.closeSubject.next(result);\n      this.closeSubject.complete();\n      this.closeSubject = new Subject<any>();\n    }\n  }\n\n  private getModalWrapperClasses(config: ModalConfig): string {\n    return 'adu-modal-wrapper';\n  }\n\n  private getModalContentClasses(config: ModalConfig): string {\n    const size = config.size || 'md';\n    return `adu-modal-content adu-modal-${size}`;\n  }\n}\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Public API Surface of angular-dev-utils
|
|
3
|
+
*/
|
|
4
|
+
// Components
|
|
5
|
+
export * from './lib/components/button/button.component';
|
|
6
|
+
export * from './lib/components/input/input.component';
|
|
7
|
+
export * from './lib/components/card/card.component';
|
|
8
|
+
export * from './lib/components/modal/modal.component';
|
|
9
|
+
export * from './lib/components/table/table.component';
|
|
10
|
+
export * from './lib/components/spinner/spinner.component';
|
|
11
|
+
// Services
|
|
12
|
+
export * from './lib/services/modal.service';
|
|
13
|
+
// Models & Types
|
|
14
|
+
export * from './lib/models/types';
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsYUFBYTtBQUNiLGNBQWMsMENBQTBDLENBQUM7QUFDekQsY0FBYyx3Q0FBd0MsQ0FBQztBQUN2RCxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsd0NBQXdDLENBQUM7QUFDdkQsY0FBYyx3Q0FBd0MsQ0FBQztBQUN2RCxjQUFjLDRDQUE0QyxDQUFDO0FBRTNELFdBQVc7QUFDWCxjQUFjLDhCQUE4QixDQUFDO0FBRTdDLGlCQUFpQjtBQUNqQixjQUFjLG9CQUFvQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBhbmd1bGFyLWRldi11dGlsc1xuICovXG5cbi8vIENvbXBvbmVudHNcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMvYnV0dG9uL2J1dHRvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY29tcG9uZW50cy9pbnB1dC9pbnB1dC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY29tcG9uZW50cy9jYXJkL2NhcmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMvbW9kYWwvbW9kYWwuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMvdGFibGUvdGFibGUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMvc3Bpbm5lci9zcGlubmVyLmNvbXBvbmVudCc7XG5cbi8vIFNlcnZpY2VzXG5leHBvcnQgKiBmcm9tICcuL2xpYi9zZXJ2aWNlcy9tb2RhbC5zZXJ2aWNlJztcblxuLy8gTW9kZWxzICYgVHlwZXNcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGVscy90eXBlcyc7XG4iXX0=
|