@scania-nl/tegel-angular-extensions 0.0.7 → 0.0.9
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 +508 -54
- package/esm2022/index.mjs +9 -1
- package/esm2022/lib/components/tae-date-time-picker/tae-date-time-picker.component.mjs +366 -0
- package/esm2022/lib/components/tae-date-time-picker/tae-date-time-picker.constants.mjs +24 -0
- package/esm2022/lib/components/tae-date-time-picker/tae-date-time-picker.utils.mjs +148 -0
- package/esm2022/lib/directives/typed-template.directive.mjs +38 -0
- package/esm2022/lib/env/angular/provide-runtime-config.mjs +7 -5
- package/esm2022/lib/modal/components/modal/modal.component.mjs +128 -0
- package/esm2022/lib/modal/components/modal-host/modal-host.component.mjs +32 -0
- package/esm2022/lib/modal/directives/dynamic-component.directive.mjs +135 -0
- package/esm2022/lib/modal/modal.service.mjs +156 -0
- package/esm2022/lib/modal/provide-modal.mjs +25 -0
- package/esm2022/lib/modal/schema/modal-ref.mjs +2 -0
- package/esm2022/lib/modal/schema/modal.config.mjs +15 -0
- package/esm2022/lib/modal/schema/modal.model.mjs +2 -0
- package/esm2022/lib/modal/schema/modal.types.mjs +3 -0
- package/esm2022/lib/toast/provide-toast.mjs +7 -1
- package/esm2022/lib/utils/angular-component-io.mjs +2 -0
- package/index.d.ts +415 -4
- package/package.json +1 -1
package/esm2022/index.mjs
CHANGED
|
@@ -5,6 +5,12 @@ export { provideToast } from './lib/toast/provide-toast';
|
|
|
5
5
|
export { DEFAULT_TOAST_CONFIG, TOAST_CONFIG, } from './lib/toast/toast.config';
|
|
6
6
|
export { ToastService } from './lib/toast/toast.service';
|
|
7
7
|
// --------------------------------------------------
|
|
8
|
+
// Modal Module Exports
|
|
9
|
+
// --------------------------------------------------
|
|
10
|
+
export { ModalService } from './lib/modal/modal.service';
|
|
11
|
+
export { provideModal } from './lib/modal/provide-modal';
|
|
12
|
+
export { DEFAULT_MODAL_CONFIG, MODAL_CONFIG, } from './lib/modal/schema/modal.config';
|
|
13
|
+
// --------------------------------------------------
|
|
8
14
|
// Env Module Exports
|
|
9
15
|
// --------------------------------------------------
|
|
10
16
|
// Core
|
|
@@ -18,8 +24,10 @@ export { provideStaticConfig } from './lib/env/angular/provide-static-config';
|
|
|
18
24
|
// --------------------------------------------------
|
|
19
25
|
export { BarcodeScannerDirective } from './lib/directives/barcode-scanner.directive';
|
|
20
26
|
export { HardRefreshDirective } from './lib/directives/hard-refresh.directive';
|
|
27
|
+
export { TypedTemplateDirective } from './lib/directives/typed-template.directive';
|
|
21
28
|
// --------------------------------------------------
|
|
22
29
|
// Component Module Exports
|
|
23
30
|
// --------------------------------------------------
|
|
31
|
+
export { TaeDateTimePickerComponent } from './lib/components/tae-date-time-picker/tae-date-time-picker.component';
|
|
24
32
|
export { TaeFooterComponent } from './lib/components/tae-footer/tae-footer.component';
|
|
25
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
33
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzL3RlZ2VsLWFuZ3VsYXItZXh0ZW5zaW9ucy9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEscURBQXFEO0FBQ3JELHVCQUF1QjtBQUN2QixxREFBcUQ7QUFDckQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ3pELE9BQU8sRUFDTCxvQkFBb0IsRUFDcEIsWUFBWSxHQUViLE1BQU0sMEJBQTBCLENBQUM7QUFDbEMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBS3pELHFEQUFxRDtBQUNyRCx1QkFBdUI7QUFDdkIscURBQXFEO0FBQ3JELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFFekQsT0FBTyxFQUNMLG9CQUFvQixFQUNwQixZQUFZLEdBRWIsTUFBTSxpQ0FBaUMsQ0FBQztBQVN6QyxxREFBcUQ7QUFDckQscUJBQXFCO0FBQ3JCLHFEQUFxRDtBQUNyRCxPQUFPO0FBQ1AsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLCtCQUErQixDQUFDO0FBQzdELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUU3RCxVQUFVO0FBQ1YsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sMENBQTBDLENBQUM7QUFDaEYsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0seUNBQXlDLENBQUM7QUFFOUUscURBQXFEO0FBQ3JELDJCQUEyQjtBQUMzQixxREFBcUQ7QUFDckQsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNENBQTRDLENBQUM7QUFDckYsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0seUNBQXlDLENBQUM7QUFDL0UsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sMkNBQTJDLENBQUM7QUFFbkYscURBQXFEO0FBQ3JELDJCQUEyQjtBQUMzQixxREFBcUQ7QUFDckQsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sc0VBQXNFLENBQUM7QUFDbEgsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sa0RBQWtELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuLy8gVG9hc3QgTW9kdWxlIEV4cG9ydHNcbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5leHBvcnQgeyBwcm92aWRlVG9hc3QgfSBmcm9tICcuL2xpYi90b2FzdC9wcm92aWRlLXRvYXN0JztcbmV4cG9ydCB7XG4gIERFRkFVTFRfVE9BU1RfQ09ORklHLFxuICBUT0FTVF9DT05GSUcsXG4gIFRvYXN0Q29uZmlnLFxufSBmcm9tICcuL2xpYi90b2FzdC90b2FzdC5jb25maWcnO1xuZXhwb3J0IHsgVG9hc3RTZXJ2aWNlIH0gZnJvbSAnLi9saWIvdG9hc3QvdG9hc3Quc2VydmljZSc7XG5cbi8vIE1vZGVsc1xuZXhwb3J0IHsgVG9hc3QsIFRvYXN0T3B0aW9ucyB9IGZyb20gJy4vbGliL3RvYXN0L21vZGVscy90b2FzdC5tb2RlbCc7XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4vLyBNb2RhbCBNb2R1bGUgRXhwb3J0c1xuLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbmV4cG9ydCB7IE1vZGFsU2VydmljZSB9IGZyb20gJy4vbGliL21vZGFsL21vZGFsLnNlcnZpY2UnO1xuZXhwb3J0IHsgcHJvdmlkZU1vZGFsIH0gZnJvbSAnLi9saWIvbW9kYWwvcHJvdmlkZS1tb2RhbCc7XG5leHBvcnQgeyBNb2RhbFJlZiB9IGZyb20gJy4vbGliL21vZGFsL3NjaGVtYS9tb2RhbC1yZWYnO1xuZXhwb3J0IHtcbiAgREVGQVVMVF9NT0RBTF9DT05GSUcsXG4gIE1PREFMX0NPTkZJRyxcbiAgTW9kYWxDb25maWcsXG59IGZyb20gJy4vbGliL21vZGFsL3NjaGVtYS9tb2RhbC5jb25maWcnO1xuZXhwb3J0IHsgTW9kYWwsIE1vZGFsT3B0aW9ucyB9IGZyb20gJy4vbGliL21vZGFsL3NjaGVtYS9tb2RhbC5tb2RlbCc7XG5leHBvcnQge1xuICBNb2RhbEFjdGlvbnNQb3NpdGlvbixcbiAgTW9kYWxBbGVydERpYWxvZ1JvbGUsXG4gIE1vZGFsQnV0dG9uLFxuICBNb2RhbFNpemUsXG59IGZyb20gJy4vbGliL21vZGFsL3NjaGVtYS9tb2RhbC50eXBlcyc7XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4vLyBFbnYgTW9kdWxlIEV4cG9ydHNcbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4vLyBDb3JlXG5leHBvcnQgeyBjcmVhdGVFbnZLaXQgfSBmcm9tICcuL2xpYi9lbnYvY29yZS9jcmVhdGUtZW52LWtpdCc7XG5leHBvcnQgeyBwYXJzZUVudkZpbGUgfSBmcm9tICcuL2xpYi9lbnYvY29yZS9wYXJzZS1lbnYtZmlsZSc7XG5cbi8vIEFuZ3VsYXJcbmV4cG9ydCB7IHByb3ZpZGVSdW50aW1lQ29uZmlnIH0gZnJvbSAnLi9saWIvZW52L2FuZ3VsYXIvcHJvdmlkZS1ydW50aW1lLWNvbmZpZyc7XG5leHBvcnQgeyBwcm92aWRlU3RhdGljQ29uZmlnIH0gZnJvbSAnLi9saWIvZW52L2FuZ3VsYXIvcHJvdmlkZS1zdGF0aWMtY29uZmlnJztcblxuLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbi8vIERpcmVjdGl2ZSBNb2R1bGUgRXhwb3J0c1xuLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbmV4cG9ydCB7IEJhcmNvZGVTY2FubmVyRGlyZWN0aXZlIH0gZnJvbSAnLi9saWIvZGlyZWN0aXZlcy9iYXJjb2RlLXNjYW5uZXIuZGlyZWN0aXZlJztcbmV4cG9ydCB7IEhhcmRSZWZyZXNoRGlyZWN0aXZlIH0gZnJvbSAnLi9saWIvZGlyZWN0aXZlcy9oYXJkLXJlZnJlc2guZGlyZWN0aXZlJztcbmV4cG9ydCB7IFR5cGVkVGVtcGxhdGVEaXJlY3RpdmUgfSBmcm9tICcuL2xpYi9kaXJlY3RpdmVzL3R5cGVkLXRlbXBsYXRlLmRpcmVjdGl2ZSc7XG5cbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4vLyBDb21wb25lbnQgTW9kdWxlIEV4cG9ydHNcbi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5leHBvcnQgeyBUYWVEYXRlVGltZVBpY2tlckNvbXBvbmVudCB9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvdGFlLWRhdGUtdGltZS1waWNrZXIvdGFlLWRhdGUtdGltZS1waWNrZXIuY29tcG9uZW50JztcbmV4cG9ydCB7IFRhZUZvb3RlckNvbXBvbmVudCB9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvdGFlLWZvb3Rlci90YWUtZm9vdGVyLmNvbXBvbmVudCc7XG4iXX0=
|
|
@@ -0,0 +1,366 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, computed, effect, ElementRef, forwardRef, input, signal, } from '@angular/core';
|
|
3
|
+
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
4
|
+
import { TegelModule } from '@scania/tegel-angular-17';
|
|
5
|
+
import { noop } from 'rxjs';
|
|
6
|
+
import { HOURS, MINUTES, MONTHS, SECONDS, WEEK_DAYS, } from './tae-date-time-picker.constants';
|
|
7
|
+
import { buildCalendarDays, formatDisplayDateTime, parseDatePartsFromIso, } from './tae-date-time-picker.utils';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
/**
|
|
10
|
+
* Standalone date/time picker that integrates with Angular forms via CVA.
|
|
11
|
+
*
|
|
12
|
+
* Supports `date`, `datetime`, and `time` modes, exposes a typed API via inputs,
|
|
13
|
+
* and emits form updates through the registered CVA callbacks.
|
|
14
|
+
*
|
|
15
|
+
* Adheres to Tegel design guidelines and can be used as a drop-in replacement.
|
|
16
|
+
*/
|
|
17
|
+
export class TaeDateTimePickerComponent {
|
|
18
|
+
elementRef;
|
|
19
|
+
// Inputs (public API)
|
|
20
|
+
label = input('');
|
|
21
|
+
labelPosition = input('outside');
|
|
22
|
+
size = input('lg');
|
|
23
|
+
state = input(undefined);
|
|
24
|
+
helper = input('');
|
|
25
|
+
mode = input('date');
|
|
26
|
+
// Form control state
|
|
27
|
+
internalValue = signal('');
|
|
28
|
+
displayValue = computed(() => formatDisplayDateTime(this.internalValue(), this.mode()));
|
|
29
|
+
suppressChange = signal(false);
|
|
30
|
+
disabled = signal(false);
|
|
31
|
+
// UI metadata
|
|
32
|
+
labelId = `tae-dtp-label-${crypto.randomUUID()}`;
|
|
33
|
+
// Picker visibility
|
|
34
|
+
showPicker = signal(false);
|
|
35
|
+
// Calendar view state
|
|
36
|
+
currentMonth = signal(0);
|
|
37
|
+
currentYear = signal(0);
|
|
38
|
+
calendarDays = computed(() => this.buildCalendar());
|
|
39
|
+
// Calendar selection
|
|
40
|
+
selectedDay = signal(null);
|
|
41
|
+
selectedMonth = signal(null);
|
|
42
|
+
selectedYear = signal(null);
|
|
43
|
+
// Time selection
|
|
44
|
+
selectedHour = signal(null);
|
|
45
|
+
selectedMinute = signal(null);
|
|
46
|
+
selectedSecond = signal(null);
|
|
47
|
+
// UI constants (static lists)
|
|
48
|
+
weekDays = WEEK_DAYS;
|
|
49
|
+
months = MONTHS;
|
|
50
|
+
hours = HOURS;
|
|
51
|
+
minutes = MINUTES;
|
|
52
|
+
seconds = SECONDS;
|
|
53
|
+
// Derived display values
|
|
54
|
+
placeholder = computed(() => {
|
|
55
|
+
if (this.mode() === 'datetime')
|
|
56
|
+
return 'dd-MM-YYYY HH:mm:ss';
|
|
57
|
+
if (this.mode() === 'time')
|
|
58
|
+
return 'HH:mm:ss';
|
|
59
|
+
return 'dd-MM-YYYY';
|
|
60
|
+
});
|
|
61
|
+
currentMonthName = computed(() => this.months[this.currentMonth()]);
|
|
62
|
+
// CVA callbacks (Angular Forms)
|
|
63
|
+
onChange = noop;
|
|
64
|
+
onTouched = noop;
|
|
65
|
+
updateScheduled = false;
|
|
66
|
+
constructor(elementRef) {
|
|
67
|
+
this.elementRef = elementRef;
|
|
68
|
+
const today = this.getToday();
|
|
69
|
+
this.currentMonth.set(today.getMonth());
|
|
70
|
+
this.currentYear.set(today.getFullYear());
|
|
71
|
+
// Sync internal ISO value when date/time parts change.
|
|
72
|
+
effect(() => {
|
|
73
|
+
const deps = [
|
|
74
|
+
this.selectedDay(),
|
|
75
|
+
this.selectedMonth(),
|
|
76
|
+
this.selectedYear(),
|
|
77
|
+
this.selectedHour(),
|
|
78
|
+
this.selectedMinute(),
|
|
79
|
+
this.selectedSecond(),
|
|
80
|
+
this.mode(),
|
|
81
|
+
];
|
|
82
|
+
void deps;
|
|
83
|
+
if (this.suppressChange() || this.updateScheduled)
|
|
84
|
+
return;
|
|
85
|
+
// Schedule the update for the next microtask so Angular can finish the
|
|
86
|
+
// current change detection pass before we emit a form value change.
|
|
87
|
+
// This avoids ExpressionChangedAfterItHasBeenCheckedError.
|
|
88
|
+
this.updateScheduled = true;
|
|
89
|
+
queueMicrotask(() => {
|
|
90
|
+
this.updateScheduled = false;
|
|
91
|
+
this.updateValue();
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
// Normalize time parts when in date-only mode.
|
|
95
|
+
effect(() => {
|
|
96
|
+
if (this.suppressChange())
|
|
97
|
+
return;
|
|
98
|
+
if (this.mode() !== 'date')
|
|
99
|
+
return;
|
|
100
|
+
if (this.selectedHour() === null &&
|
|
101
|
+
this.selectedMinute() === null &&
|
|
102
|
+
this.selectedSecond() === null) {
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
this.selectedHour.set(null);
|
|
106
|
+
this.selectedMinute.set(null);
|
|
107
|
+
this.selectedSecond.set(null);
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
// ControlValueAccessor
|
|
111
|
+
/**
|
|
112
|
+
* Writes an ISO value from the outside into the picker state.
|
|
113
|
+
* @param value ISO date/datetime string.
|
|
114
|
+
*/
|
|
115
|
+
writeValue(value) {
|
|
116
|
+
this.suppressChange.set(true);
|
|
117
|
+
this.internalValue.set(value || '');
|
|
118
|
+
if (!value) {
|
|
119
|
+
const today = this.getToday();
|
|
120
|
+
this.resetSelection();
|
|
121
|
+
this.currentMonth.set(today.getMonth());
|
|
122
|
+
this.currentYear.set(today.getFullYear());
|
|
123
|
+
this.suppressChange.set(false);
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
if (value) {
|
|
127
|
+
if (this.mode() === 'time') {
|
|
128
|
+
const [hour, minute, second] = value.split(':');
|
|
129
|
+
this.selectedHour.set(hour || null);
|
|
130
|
+
this.selectedMinute.set(minute || null);
|
|
131
|
+
this.selectedSecond.set(second || null);
|
|
132
|
+
}
|
|
133
|
+
else if (this.mode() === 'datetime') {
|
|
134
|
+
const [datePart, timePart] = value.split('T');
|
|
135
|
+
this.parseDatePart(datePart);
|
|
136
|
+
if (timePart) {
|
|
137
|
+
const [hour, minute, second] = timePart.split(':');
|
|
138
|
+
this.selectedHour.set(hour || null);
|
|
139
|
+
this.selectedMinute.set(minute || null);
|
|
140
|
+
this.selectedSecond.set(second || null);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
this.parseDatePart(value);
|
|
145
|
+
}
|
|
146
|
+
const selectedYear = this.selectedYear();
|
|
147
|
+
const selectedMonth = this.selectedMonth();
|
|
148
|
+
if (selectedYear && selectedMonth !== null) {
|
|
149
|
+
this.currentYear.set(selectedYear);
|
|
150
|
+
this.currentMonth.set(selectedMonth);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
this.suppressChange.set(false);
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Registers the CVA change handler.
|
|
157
|
+
* @param fn Callback invoked on value changes.
|
|
158
|
+
*/
|
|
159
|
+
registerOnChange(fn) {
|
|
160
|
+
this.onChange = fn;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Registers the CVA touched handler.
|
|
164
|
+
* @param fn Callback invoked on touch.
|
|
165
|
+
*/
|
|
166
|
+
registerOnTouched(fn) {
|
|
167
|
+
this.onTouched = fn;
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Updates the disabled state for the control.
|
|
171
|
+
* @param isDisabled Whether the control is disabled.
|
|
172
|
+
*/
|
|
173
|
+
setDisabledState(isDisabled) {
|
|
174
|
+
this.disabled.set(isDisabled);
|
|
175
|
+
if (isDisabled) {
|
|
176
|
+
this.showPicker.set(false);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
// Host/DOM handlers
|
|
180
|
+
/**
|
|
181
|
+
* Closes the picker when clicking outside the component.
|
|
182
|
+
* @param event Document click event.
|
|
183
|
+
*/
|
|
184
|
+
onDocumentClick(event) {
|
|
185
|
+
if (!this.elementRef.nativeElement.contains(event.target)) {
|
|
186
|
+
this.showPicker.set(false);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
/** Marks the control as touched. */
|
|
190
|
+
onBlur() {
|
|
191
|
+
this.onTouched();
|
|
192
|
+
}
|
|
193
|
+
// UI actions
|
|
194
|
+
/**
|
|
195
|
+
* Toggles the picker popover.
|
|
196
|
+
* @param event Click event from the control.
|
|
197
|
+
*/
|
|
198
|
+
togglePicker(event) {
|
|
199
|
+
event.preventDefault();
|
|
200
|
+
event.stopPropagation();
|
|
201
|
+
this.showPicker.update((value) => !value);
|
|
202
|
+
}
|
|
203
|
+
/** Moves the calendar to the previous month. */
|
|
204
|
+
previousMonth() {
|
|
205
|
+
const month = this.currentMonth() - 1;
|
|
206
|
+
if (month < 0) {
|
|
207
|
+
this.currentMonth.set(11);
|
|
208
|
+
this.currentYear.update((value) => value - 1);
|
|
209
|
+
return;
|
|
210
|
+
}
|
|
211
|
+
this.currentMonth.set(month);
|
|
212
|
+
}
|
|
213
|
+
/** Moves the calendar to the next month. */
|
|
214
|
+
nextMonth() {
|
|
215
|
+
const month = this.currentMonth() + 1;
|
|
216
|
+
if (month > 11) {
|
|
217
|
+
this.currentMonth.set(0);
|
|
218
|
+
this.currentYear.update((value) => value + 1);
|
|
219
|
+
return;
|
|
220
|
+
}
|
|
221
|
+
this.currentMonth.set(month);
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Builds the calendar day grid for the current month.
|
|
225
|
+
* @returns Calendar day entries for rendering.
|
|
226
|
+
*/
|
|
227
|
+
buildCalendar() {
|
|
228
|
+
if (this.mode() === 'time') {
|
|
229
|
+
return [];
|
|
230
|
+
}
|
|
231
|
+
const currentYear = this.currentYear();
|
|
232
|
+
const currentMonth = this.currentMonth();
|
|
233
|
+
const today = this.getToday();
|
|
234
|
+
const selected = this.getSelectedDateParts();
|
|
235
|
+
return buildCalendarDays(currentYear, currentMonth, selected, today);
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* Selects a date from the calendar grid.
|
|
239
|
+
* @param day Calendar day entry.
|
|
240
|
+
*/
|
|
241
|
+
selectDate(day) {
|
|
242
|
+
this.selectedDay.set(day.date);
|
|
243
|
+
this.selectedMonth.set(day.month);
|
|
244
|
+
this.selectedYear.set(day.year);
|
|
245
|
+
// In date-only mode, close picker after selection
|
|
246
|
+
if (this.mode() === 'date') {
|
|
247
|
+
this.showPicker.set(false);
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
/**
|
|
251
|
+
* Selects an hour in datetime mode.
|
|
252
|
+
* @param hour Two-digit hour string.
|
|
253
|
+
*/
|
|
254
|
+
selectHour(hour) {
|
|
255
|
+
this.selectedHour.set(hour);
|
|
256
|
+
}
|
|
257
|
+
/**
|
|
258
|
+
* Selects a minute in datetime mode.
|
|
259
|
+
* @param minute Two-digit minute string.
|
|
260
|
+
*/
|
|
261
|
+
selectMinute(minute) {
|
|
262
|
+
this.selectedMinute.set(minute);
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* Selects a second in datetime mode.
|
|
266
|
+
* @param second Two-digit second string.
|
|
267
|
+
*/
|
|
268
|
+
selectSecond(second) {
|
|
269
|
+
this.selectedSecond.set(second);
|
|
270
|
+
}
|
|
271
|
+
/** Confirms the selection and closes the picker. */
|
|
272
|
+
confirm() {
|
|
273
|
+
this.showPicker.set(false);
|
|
274
|
+
this.onTouched();
|
|
275
|
+
}
|
|
276
|
+
// Internal helpers
|
|
277
|
+
/**
|
|
278
|
+
* Parses an ISO date segment into selected date parts.
|
|
279
|
+
* @param datePart ISO date segment.
|
|
280
|
+
*/
|
|
281
|
+
parseDatePart(datePart) {
|
|
282
|
+
const parsed = parseDatePartsFromIso(datePart);
|
|
283
|
+
if (!parsed)
|
|
284
|
+
return;
|
|
285
|
+
this.selectedYear.set(parsed.year);
|
|
286
|
+
this.selectedMonth.set(parsed.month);
|
|
287
|
+
this.selectedDay.set(parsed.day);
|
|
288
|
+
}
|
|
289
|
+
/** Computes and emits the ISO value from current parts. */
|
|
290
|
+
updateValue() {
|
|
291
|
+
if (this.mode() === 'time') {
|
|
292
|
+
const hour = this.selectedHour();
|
|
293
|
+
const minute = this.selectedMinute();
|
|
294
|
+
const second = this.selectedSecond();
|
|
295
|
+
if (!hour || !minute || !second)
|
|
296
|
+
return;
|
|
297
|
+
const nextValue = `${hour}:${minute}:${second}`;
|
|
298
|
+
if (nextValue === this.internalValue())
|
|
299
|
+
return;
|
|
300
|
+
this.internalValue.set(nextValue);
|
|
301
|
+
this.onChange(nextValue);
|
|
302
|
+
return;
|
|
303
|
+
}
|
|
304
|
+
const selectedDay = this.selectedDay();
|
|
305
|
+
const selectedMonth = this.selectedMonth();
|
|
306
|
+
const selectedYear = this.selectedYear();
|
|
307
|
+
if (selectedDay && selectedMonth !== null && selectedYear) {
|
|
308
|
+
const month = (selectedMonth + 1).toString().padStart(2, '0');
|
|
309
|
+
const day = selectedDay.toString().padStart(2, '0');
|
|
310
|
+
const datePart = `${selectedYear}-${month}-${day}`;
|
|
311
|
+
const nextValue = this.mode() === 'datetime'
|
|
312
|
+
? `${datePart}T${this.selectedHour() ?? '00'}:${this.selectedMinute() ?? '00'}:${this.selectedSecond() ?? '00'}`
|
|
313
|
+
: datePart;
|
|
314
|
+
if (nextValue === this.internalValue())
|
|
315
|
+
return;
|
|
316
|
+
this.internalValue.set(nextValue);
|
|
317
|
+
this.onChange(nextValue);
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
/** Resets all selected date/time parts to defaults. */
|
|
321
|
+
resetSelection() {
|
|
322
|
+
this.selectedDay.set(null);
|
|
323
|
+
this.selectedMonth.set(null);
|
|
324
|
+
this.selectedYear.set(null);
|
|
325
|
+
this.selectedHour.set(null);
|
|
326
|
+
this.selectedMinute.set(null);
|
|
327
|
+
this.selectedSecond.set(null);
|
|
328
|
+
}
|
|
329
|
+
/**
|
|
330
|
+
* Returns current selected date parts when fully defined.
|
|
331
|
+
* @returns Date parts or `null` if incomplete.
|
|
332
|
+
*/
|
|
333
|
+
getSelectedDateParts() {
|
|
334
|
+
const day = this.selectedDay();
|
|
335
|
+
const month = this.selectedMonth();
|
|
336
|
+
const year = this.selectedYear();
|
|
337
|
+
if (day === null || month === null || year === null)
|
|
338
|
+
return null;
|
|
339
|
+
return { day, month, year };
|
|
340
|
+
}
|
|
341
|
+
/** Returns today's date in local time. */
|
|
342
|
+
getToday() {
|
|
343
|
+
return new Date();
|
|
344
|
+
}
|
|
345
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: TaeDateTimePickerComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
346
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.17", type: TaeDateTimePickerComponent, isStandalone: true, selector: "tae-date-time-picker", inputs: { label: { classPropertyName: "label", publicName: "label", isSignal: true, isRequired: false, transformFunction: null }, labelPosition: { classPropertyName: "labelPosition", publicName: "labelPosition", isSignal: true, isRequired: false, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: true, isRequired: false, transformFunction: null }, state: { classPropertyName: "state", publicName: "state", isSignal: true, isRequired: false, transformFunction: null }, helper: { classPropertyName: "helper", publicName: "helper", isSignal: true, isRequired: false, transformFunction: null }, mode: { classPropertyName: "mode", publicName: "mode", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "document:click": "onDocumentClick($event)" } }, providers: [
|
|
347
|
+
{
|
|
348
|
+
provide: NG_VALUE_ACCESSOR,
|
|
349
|
+
useExisting: forwardRef(() => TaeDateTimePickerComponent),
|
|
350
|
+
multi: true,
|
|
351
|
+
},
|
|
352
|
+
], ngImport: i0, template: "<div class=\"date-picker-wrapper\">\r\n <label\n class=\"date-picker-label\"\n [class.outside]=\"labelPosition() === 'outside'\"\n [attr.for]=\"labelId\"\n >{{ label() }}</label\n >\n <div\n [id]=\"labelId\"\n class=\"date-picker-container\"\n [class.error]=\"state() === 'error'\"\n [class.size-sm]=\"size() === 'sm'\"\r\n [class.size-md]=\"size() === 'md'\"\r\n [class.size-lg]=\"size() === 'lg'\"\r\n [class.disabled]=\"disabled()\"\r\n (click)=\"!disabled() && togglePicker($event)\"\r\n (keydown.enter)=\"!disabled() && togglePicker($event)\"\r\n (keydown.space)=\"!disabled() && togglePicker($event)\"\r\n role=\"button\"\r\n tabindex=\"0\"\r\n >\r\n <span class=\"display-value\" [class.placeholder]=\"!displayValue()\">{{\r\n displayValue() || placeholder()\r\n }}</span>\r\n <div class=\"picker-icon\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"16\"\r\n height=\"16\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <rect x=\"3\" y=\"4\" width=\"18\" height=\"18\" rx=\"2\" ry=\"2\"></rect>\r\n <line x1=\"16\" y1=\"2\" x2=\"16\" y2=\"6\"></line>\r\n <line x1=\"8\" y1=\"2\" x2=\"8\" y2=\"6\"></line>\r\n <line x1=\"3\" y1=\"10\" x2=\"21\" y2=\"10\"></line>\r\n </svg>\r\n </div>\r\n </div>\r\n\r\n @if (showPicker()) {\n <div class=\"picker-popover\" [class.datetime-mode]=\"mode() !== 'date'\">\n <div class=\"picker-content\">\n <!-- Calendar Section -->\n @if (mode() !== 'time') {\n <div class=\"calendar-section\">\n <div class=\"calendar-header\">\n <button type=\"button\" class=\"nav-btn\" (click)=\"previousMonth()\">\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"16\"\r\n height=\"16\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <polyline points=\"15 18 9 12 15 6\"></polyline>\r\n </svg>\r\n </button>\r\n <span class=\"month-year\"\r\n >{{ currentMonthName() }} {{ currentYear() }}</span\r\n >\r\n <button type=\"button\" class=\"nav-btn\" (click)=\"nextMonth()\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"16\"\r\n height=\"16\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <polyline points=\"9 18 15 12 9 6\"></polyline>\r\n </svg>\r\n </button>\n </div>\n <div class=\"calendar-weekdays\">\n @for (day of weekDays; track day) {\n <span class=\"weekday\">{{ day }}</span>\n }\n </div>\n <div class=\"calendar-days\">\n @for (\n day of calendarDays();\n track day.date + '-' + day.month + '-' + day.year\n ) {\n <button\n type=\"button\"\n class=\"calendar-day\"\n [disabled]=\"disabled()\"\n [class.other-month]=\"!day.isCurrentMonth\"\n [class.today]=\"day.isToday\"\n [class.selected]=\"day.isSelected\"\n (click)=\"selectDate(day); $event.stopPropagation()\"\n >\n {{ day.date }}\n </button>\n }\n </div>\n </div>\n }\n\n <!-- Time Section (only for datetime mode) -->\n @if (mode() !== 'date') {\n <div class=\"time-section\">\n <div class=\"time-section-header\">\r\n <span>Time (24h)</span>\r\n </div>\r\n <div class=\"time-picker-body\">\r\n <div class=\"time-column\">\r\n <div class=\"time-column-label\">Hour</div>\r\n <div class=\"time-column-scroll\">\r\n @for (hour of hours; track hour) {\r\n <button\r\n type=\"button\"\r\n class=\"time-option\"\r\n [disabled]=\"disabled()\"\r\n [class.selected]=\"hour === selectedHour()\"\n (click)=\"selectHour(hour); $event.stopPropagation()\"\n >\n {{ hour }}\n </button>\n }\r\n </div>\r\n </div>\r\n <div class=\"time-separator\">:</div>\r\n <div class=\"time-column\">\r\n <div class=\"time-column-label\">Min</div>\r\n <div class=\"time-column-scroll\">\r\n @for (minute of minutes; track minute) {\r\n <button\r\n type=\"button\"\r\n class=\"time-option\"\r\n [disabled]=\"disabled()\"\r\n [class.selected]=\"minute === selectedMinute()\"\n (click)=\"selectMinute(minute); $event.stopPropagation()\"\n >\n {{ minute }}\n </button>\n }\r\n </div>\r\n </div>\r\n <div class=\"time-separator\">:</div>\r\n <div class=\"time-column\">\r\n <div class=\"time-column-label\">Sec</div>\r\n <div class=\"time-column-scroll\">\r\n @for (second of seconds; track second) {\r\n <button\r\n type=\"button\"\r\n class=\"time-option\"\r\n [disabled]=\"disabled()\"\r\n [class.selected]=\"second === selectedSecond()\"\n (click)=\"selectSecond(second); $event.stopPropagation()\"\n >\n {{ second }}\n </button>\n }\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n }\r\n </div>\r\n\r\n <!-- Footer -->\r\n <div class=\"picker-footer\">\r\n <button\r\n type=\"button\"\r\n class=\"confirm-btn\"\r\n [disabled]=\"disabled()\"\r\n (click)=\"confirm(); $event.stopPropagation()\"\r\n >\r\n OK\r\n </button>\r\n </div>\r\n </div>\r\n }\r\n\r\n @if (helper()) {\r\n <span class=\"helper-text\" [class.error]=\"state() === 'error'\">{{\r\n helper()\r\n }}</span>\r\n }\r\n</div>\r\n", styles: [".date-picker-wrapper{display:flex;flex-direction:column;position:relative}.date-picker-label{font:var(--tds-detail-05);letter-spacing:var(--tds-detail-05-ls);color:var(--tds-text-field-label-color, #1d2229);margin-bottom:8px}.date-picker-label.no-label{display:none}.date-picker-container{position:relative;display:flex;align-items:center;background-color:var(--tds-text-field-background, #f9fafb);border:none;border-bottom:1px solid var(--tds-text-field-border-bottom, #b0b7c4);border-radius:4px 4px 0 0;cursor:pointer;transition:border-color .17s ease,background-color .17s ease;box-sizing:border-box}.date-picker-container:hover{background-color:var(--tds-text-field-background-hover, #edeff3)}.date-picker-container:focus-within,.date-picker-container.focus{border-bottom-color:var(--tds-text-field-border-bottom-focus, #2b70d3);border-bottom-width:2px}.date-picker-container.error{border-bottom-color:var(--tds-negative, #ff2340);border-bottom-width:2px}.date-picker-container.size-sm{height:40px;padding:0 16px}.date-picker-container.size-md{height:48px;padding:0 16px}.date-picker-container.size-lg{height:56px;padding:0 16px}.date-picker-container.disabled{cursor:not-allowed;background-color:var(--tds-text-field-background, #f9fafb);opacity:.6}.display-value{flex:1;font:var(--tds-detail-02);letter-spacing:var(--tds-detail-02-ls);color:var(--tds-text-field-color, #1d2229)}.display-value.placeholder{color:var(--tds-text-field-placeholder, #868d99)}.picker-icon{display:flex;align-items:center;justify-content:center;color:var(--tds-grey-600, #6b6f77);margin-left:8px}.picker-popover{position:absolute;top:100%;left:0;margin-top:4px;background:#fff;border:1px solid var(--tds-grey-300, #cdd1db);border-radius:8px;box-shadow:0 4px 16px #0000001f;z-index:1000;min-width:280px}.picker-content{display:flex;flex-direction:row}.calendar-section{padding:12px;flex:1;min-width:280px}.calendar-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:12px}.nav-btn{width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:transparent;border:none;border-radius:4px;cursor:pointer;color:var(--tds-grey-600, #6b6f77);transition:background-color .15s ease}.nav-btn:hover{background-color:var(--tds-grey-100, #f4f5f7);color:var(--tds-grey-958, #1d2229)}.month-year{font-size:14px;font-weight:600;color:var(--tds-grey-958, #1d2229)}.calendar-weekdays{display:grid;grid-template-columns:repeat(7,1fr);gap:2px;margin-bottom:8px}.weekday{font-size:12px;font-weight:500;color:var(--tds-grey-600, #6b6f77);text-align:center;padding:4px}.calendar-days{display:grid;grid-template-columns:repeat(7,1fr);gap:2px}.calendar-day{width:36px;height:36px;display:flex;align-items:center;justify-content:center;background:transparent;border:none;border-radius:4px;cursor:pointer;font-size:14px;color:var(--tds-grey-958, #1d2229);transition:background-color .15s ease}.calendar-day:hover{background-color:var(--tds-grey-100, #f4f5f7)}.calendar-day.other-month{color:var(--tds-grey-400, #b0b3bb)}.calendar-day.today{border:1px solid var(--tds-blue-500, #2b70d3)}.calendar-day.selected{background-color:var(--tds-blue-500, #2b70d3);color:#fff}.calendar-day:disabled{cursor:not-allowed;opacity:.6}.time-section{padding:12px;border-left:1px solid var(--tds-grey-200, #e7e9ed);min-width:160px}.time-section-header{font-size:14px;font-weight:500;color:var(--tds-grey-958, #1d2229);margin-bottom:12px;text-align:center}.time-picker-body{display:flex;align-items:flex-start;gap:4px}.time-column{flex:1;display:flex;flex-direction:column}.time-column-label{font-size:12px;color:var(--tds-grey-600, #6b6f77);text-align:center;padding:4px 0 8px;font-weight:500}.time-column-scroll{max-height:200px;overflow-y:auto;display:flex;flex-direction:column;gap:2px}.time-column-scroll::-webkit-scrollbar{width:6px}.time-column-scroll::-webkit-scrollbar-track{background:var(--tds-grey-100, #f4f5f7);border-radius:3px}.time-column-scroll::-webkit-scrollbar-thumb{background:var(--tds-grey-400, #b0b3bb);border-radius:3px}.time-option{padding:6px 8px;border:none;background:transparent;cursor:pointer;font-size:14px;color:var(--tds-grey-958, #1d2229);border-radius:4px;text-align:center;transition:background-color .15s ease}.time-option:hover{background-color:var(--tds-grey-100, #f4f5f7)}.time-option.selected{background-color:var(--tds-blue-500, #2b70d3);color:#fff}.time-option:disabled{cursor:not-allowed;opacity:.6}.time-separator{font-size:20px;font-weight:500;color:var(--tds-grey-600, #6b6f77);padding-top:28px}.picker-footer{padding:8px 12px;border-top:1px solid var(--tds-grey-200, #e7e9ed);display:flex;justify-content:flex-end}.confirm-btn{padding:8px 24px;background-color:var(--tds-blue-500, #2b70d3);color:#fff;border:none;border-radius:4px;font-size:14px;font-weight:500;cursor:pointer;transition:background-color .2s ease}.confirm-btn:hover{background-color:var(--tds-blue-600, #2563c2)}.confirm-btn:disabled{cursor:not-allowed;opacity:.6}.helper-text{font:var(--tds-detail-05);letter-spacing:var(--tds-detail-05-ls);color:var(--tds-text-field-helper, #6b6f77);margin-top:8px}.helper-text.error{color:var(--tds-negative, #ff2340)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: TegelModule }] });
|
|
353
|
+
}
|
|
354
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: TaeDateTimePickerComponent, decorators: [{
|
|
355
|
+
type: Component,
|
|
356
|
+
args: [{ selector: 'tae-date-time-picker', imports: [CommonModule, TegelModule], host: {
|
|
357
|
+
'(document:click)': 'onDocumentClick($event)',
|
|
358
|
+
}, providers: [
|
|
359
|
+
{
|
|
360
|
+
provide: NG_VALUE_ACCESSOR,
|
|
361
|
+
useExisting: forwardRef(() => TaeDateTimePickerComponent),
|
|
362
|
+
multi: true,
|
|
363
|
+
},
|
|
364
|
+
], template: "<div class=\"date-picker-wrapper\">\r\n <label\n class=\"date-picker-label\"\n [class.outside]=\"labelPosition() === 'outside'\"\n [attr.for]=\"labelId\"\n >{{ label() }}</label\n >\n <div\n [id]=\"labelId\"\n class=\"date-picker-container\"\n [class.error]=\"state() === 'error'\"\n [class.size-sm]=\"size() === 'sm'\"\r\n [class.size-md]=\"size() === 'md'\"\r\n [class.size-lg]=\"size() === 'lg'\"\r\n [class.disabled]=\"disabled()\"\r\n (click)=\"!disabled() && togglePicker($event)\"\r\n (keydown.enter)=\"!disabled() && togglePicker($event)\"\r\n (keydown.space)=\"!disabled() && togglePicker($event)\"\r\n role=\"button\"\r\n tabindex=\"0\"\r\n >\r\n <span class=\"display-value\" [class.placeholder]=\"!displayValue()\">{{\r\n displayValue() || placeholder()\r\n }}</span>\r\n <div class=\"picker-icon\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"16\"\r\n height=\"16\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <rect x=\"3\" y=\"4\" width=\"18\" height=\"18\" rx=\"2\" ry=\"2\"></rect>\r\n <line x1=\"16\" y1=\"2\" x2=\"16\" y2=\"6\"></line>\r\n <line x1=\"8\" y1=\"2\" x2=\"8\" y2=\"6\"></line>\r\n <line x1=\"3\" y1=\"10\" x2=\"21\" y2=\"10\"></line>\r\n </svg>\r\n </div>\r\n </div>\r\n\r\n @if (showPicker()) {\n <div class=\"picker-popover\" [class.datetime-mode]=\"mode() !== 'date'\">\n <div class=\"picker-content\">\n <!-- Calendar Section -->\n @if (mode() !== 'time') {\n <div class=\"calendar-section\">\n <div class=\"calendar-header\">\n <button type=\"button\" class=\"nav-btn\" (click)=\"previousMonth()\">\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"16\"\r\n height=\"16\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <polyline points=\"15 18 9 12 15 6\"></polyline>\r\n </svg>\r\n </button>\r\n <span class=\"month-year\"\r\n >{{ currentMonthName() }} {{ currentYear() }}</span\r\n >\r\n <button type=\"button\" class=\"nav-btn\" (click)=\"nextMonth()\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"16\"\r\n height=\"16\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <polyline points=\"9 18 15 12 9 6\"></polyline>\r\n </svg>\r\n </button>\n </div>\n <div class=\"calendar-weekdays\">\n @for (day of weekDays; track day) {\n <span class=\"weekday\">{{ day }}</span>\n }\n </div>\n <div class=\"calendar-days\">\n @for (\n day of calendarDays();\n track day.date + '-' + day.month + '-' + day.year\n ) {\n <button\n type=\"button\"\n class=\"calendar-day\"\n [disabled]=\"disabled()\"\n [class.other-month]=\"!day.isCurrentMonth\"\n [class.today]=\"day.isToday\"\n [class.selected]=\"day.isSelected\"\n (click)=\"selectDate(day); $event.stopPropagation()\"\n >\n {{ day.date }}\n </button>\n }\n </div>\n </div>\n }\n\n <!-- Time Section (only for datetime mode) -->\n @if (mode() !== 'date') {\n <div class=\"time-section\">\n <div class=\"time-section-header\">\r\n <span>Time (24h)</span>\r\n </div>\r\n <div class=\"time-picker-body\">\r\n <div class=\"time-column\">\r\n <div class=\"time-column-label\">Hour</div>\r\n <div class=\"time-column-scroll\">\r\n @for (hour of hours; track hour) {\r\n <button\r\n type=\"button\"\r\n class=\"time-option\"\r\n [disabled]=\"disabled()\"\r\n [class.selected]=\"hour === selectedHour()\"\n (click)=\"selectHour(hour); $event.stopPropagation()\"\n >\n {{ hour }}\n </button>\n }\r\n </div>\r\n </div>\r\n <div class=\"time-separator\">:</div>\r\n <div class=\"time-column\">\r\n <div class=\"time-column-label\">Min</div>\r\n <div class=\"time-column-scroll\">\r\n @for (minute of minutes; track minute) {\r\n <button\r\n type=\"button\"\r\n class=\"time-option\"\r\n [disabled]=\"disabled()\"\r\n [class.selected]=\"minute === selectedMinute()\"\n (click)=\"selectMinute(minute); $event.stopPropagation()\"\n >\n {{ minute }}\n </button>\n }\r\n </div>\r\n </div>\r\n <div class=\"time-separator\">:</div>\r\n <div class=\"time-column\">\r\n <div class=\"time-column-label\">Sec</div>\r\n <div class=\"time-column-scroll\">\r\n @for (second of seconds; track second) {\r\n <button\r\n type=\"button\"\r\n class=\"time-option\"\r\n [disabled]=\"disabled()\"\r\n [class.selected]=\"second === selectedSecond()\"\n (click)=\"selectSecond(second); $event.stopPropagation()\"\n >\n {{ second }}\n </button>\n }\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n }\r\n </div>\r\n\r\n <!-- Footer -->\r\n <div class=\"picker-footer\">\r\n <button\r\n type=\"button\"\r\n class=\"confirm-btn\"\r\n [disabled]=\"disabled()\"\r\n (click)=\"confirm(); $event.stopPropagation()\"\r\n >\r\n OK\r\n </button>\r\n </div>\r\n </div>\r\n }\r\n\r\n @if (helper()) {\r\n <span class=\"helper-text\" [class.error]=\"state() === 'error'\">{{\r\n helper()\r\n }}</span>\r\n }\r\n</div>\r\n", styles: [".date-picker-wrapper{display:flex;flex-direction:column;position:relative}.date-picker-label{font:var(--tds-detail-05);letter-spacing:var(--tds-detail-05-ls);color:var(--tds-text-field-label-color, #1d2229);margin-bottom:8px}.date-picker-label.no-label{display:none}.date-picker-container{position:relative;display:flex;align-items:center;background-color:var(--tds-text-field-background, #f9fafb);border:none;border-bottom:1px solid var(--tds-text-field-border-bottom, #b0b7c4);border-radius:4px 4px 0 0;cursor:pointer;transition:border-color .17s ease,background-color .17s ease;box-sizing:border-box}.date-picker-container:hover{background-color:var(--tds-text-field-background-hover, #edeff3)}.date-picker-container:focus-within,.date-picker-container.focus{border-bottom-color:var(--tds-text-field-border-bottom-focus, #2b70d3);border-bottom-width:2px}.date-picker-container.error{border-bottom-color:var(--tds-negative, #ff2340);border-bottom-width:2px}.date-picker-container.size-sm{height:40px;padding:0 16px}.date-picker-container.size-md{height:48px;padding:0 16px}.date-picker-container.size-lg{height:56px;padding:0 16px}.date-picker-container.disabled{cursor:not-allowed;background-color:var(--tds-text-field-background, #f9fafb);opacity:.6}.display-value{flex:1;font:var(--tds-detail-02);letter-spacing:var(--tds-detail-02-ls);color:var(--tds-text-field-color, #1d2229)}.display-value.placeholder{color:var(--tds-text-field-placeholder, #868d99)}.picker-icon{display:flex;align-items:center;justify-content:center;color:var(--tds-grey-600, #6b6f77);margin-left:8px}.picker-popover{position:absolute;top:100%;left:0;margin-top:4px;background:#fff;border:1px solid var(--tds-grey-300, #cdd1db);border-radius:8px;box-shadow:0 4px 16px #0000001f;z-index:1000;min-width:280px}.picker-content{display:flex;flex-direction:row}.calendar-section{padding:12px;flex:1;min-width:280px}.calendar-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:12px}.nav-btn{width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:transparent;border:none;border-radius:4px;cursor:pointer;color:var(--tds-grey-600, #6b6f77);transition:background-color .15s ease}.nav-btn:hover{background-color:var(--tds-grey-100, #f4f5f7);color:var(--tds-grey-958, #1d2229)}.month-year{font-size:14px;font-weight:600;color:var(--tds-grey-958, #1d2229)}.calendar-weekdays{display:grid;grid-template-columns:repeat(7,1fr);gap:2px;margin-bottom:8px}.weekday{font-size:12px;font-weight:500;color:var(--tds-grey-600, #6b6f77);text-align:center;padding:4px}.calendar-days{display:grid;grid-template-columns:repeat(7,1fr);gap:2px}.calendar-day{width:36px;height:36px;display:flex;align-items:center;justify-content:center;background:transparent;border:none;border-radius:4px;cursor:pointer;font-size:14px;color:var(--tds-grey-958, #1d2229);transition:background-color .15s ease}.calendar-day:hover{background-color:var(--tds-grey-100, #f4f5f7)}.calendar-day.other-month{color:var(--tds-grey-400, #b0b3bb)}.calendar-day.today{border:1px solid var(--tds-blue-500, #2b70d3)}.calendar-day.selected{background-color:var(--tds-blue-500, #2b70d3);color:#fff}.calendar-day:disabled{cursor:not-allowed;opacity:.6}.time-section{padding:12px;border-left:1px solid var(--tds-grey-200, #e7e9ed);min-width:160px}.time-section-header{font-size:14px;font-weight:500;color:var(--tds-grey-958, #1d2229);margin-bottom:12px;text-align:center}.time-picker-body{display:flex;align-items:flex-start;gap:4px}.time-column{flex:1;display:flex;flex-direction:column}.time-column-label{font-size:12px;color:var(--tds-grey-600, #6b6f77);text-align:center;padding:4px 0 8px;font-weight:500}.time-column-scroll{max-height:200px;overflow-y:auto;display:flex;flex-direction:column;gap:2px}.time-column-scroll::-webkit-scrollbar{width:6px}.time-column-scroll::-webkit-scrollbar-track{background:var(--tds-grey-100, #f4f5f7);border-radius:3px}.time-column-scroll::-webkit-scrollbar-thumb{background:var(--tds-grey-400, #b0b3bb);border-radius:3px}.time-option{padding:6px 8px;border:none;background:transparent;cursor:pointer;font-size:14px;color:var(--tds-grey-958, #1d2229);border-radius:4px;text-align:center;transition:background-color .15s ease}.time-option:hover{background-color:var(--tds-grey-100, #f4f5f7)}.time-option.selected{background-color:var(--tds-blue-500, #2b70d3);color:#fff}.time-option:disabled{cursor:not-allowed;opacity:.6}.time-separator{font-size:20px;font-weight:500;color:var(--tds-grey-600, #6b6f77);padding-top:28px}.picker-footer{padding:8px 12px;border-top:1px solid var(--tds-grey-200, #e7e9ed);display:flex;justify-content:flex-end}.confirm-btn{padding:8px 24px;background-color:var(--tds-blue-500, #2b70d3);color:#fff;border:none;border-radius:4px;font-size:14px;font-weight:500;cursor:pointer;transition:background-color .2s ease}.confirm-btn:hover{background-color:var(--tds-blue-600, #2563c2)}.confirm-btn:disabled{cursor:not-allowed;opacity:.6}.helper-text{font:var(--tds-detail-05);letter-spacing:var(--tds-detail-05-ls);color:var(--tds-text-field-helper, #6b6f77);margin-top:8px}.helper-text.error{color:var(--tds-negative, #ff2340)}\n"] }]
|
|
365
|
+
}], ctorParameters: () => [{ type: i0.ElementRef }] });
|
|
366
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tae-date-time-picker.component.js","sourceRoot":"","sources":["../../../../../../../libs/tegel-angular-extensions/src/lib/components/tae-date-time-picker/tae-date-time-picker.component.ts","../../../../../../../libs/tegel-angular-extensions/src/lib/components/tae-date-time-picker/tae-date-time-picker.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,SAAS,EACT,QAAQ,EACR,MAAM,EACN,UAAU,EACV,UAAU,EACV,KAAK,EACL,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AACvD,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EACL,KAAK,EACL,OAAO,EACP,MAAM,EACN,OAAO,EACP,SAAS,GACV,MAAM,kCAAkC,CAAC;AAC1C,OAAO,EACL,iBAAiB,EACjB,qBAAqB,EACrB,qBAAqB,GACtB,MAAM,8BAA8B,CAAC;;AAItC;;;;;;;GAOG;AAiBH,MAAM,OAAO,0BAA0B;IAqDR;IApD7B,sBAAsB;IACb,KAAK,GAAG,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,aAAa,GAAG,KAAK,CAAoC,SAAS,CAAC,CAAC;IACpE,IAAI,GAAG,KAAK,CAAqB,IAAI,CAAC,CAAC;IACvC,KAAK,GAAG,KAAK,CAAsB,SAAS,CAAC,CAAC;IAC9C,MAAM,GAAG,KAAK,CAAC,EAAE,CAAC,CAAC;IACnB,IAAI,GAAG,KAAK,CAA+B,MAAM,CAAC,CAAC;IAE5D,qBAAqB;IACJ,aAAa,GAAG,MAAM,CAAC,EAAE,CAAC,CAAC;IACnC,YAAY,GAAG,QAAQ,CAAC,GAAG,EAAE,CACpC,qBAAqB,CAAC,IAAI,CAAC,aAAa,EAAE,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CACzD,CAAC;IACe,cAAc,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IACvC,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IAClC,cAAc;IACL,OAAO,GAAG,iBAAiB,MAAM,CAAC,UAAU,EAAE,EAAE,CAAC;IAC1D,oBAAoB;IACX,UAAU,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IACpC,sBAAsB;IACb,YAAY,GAAG,MAAM,CAAS,CAAC,CAAC,CAAC;IACjC,WAAW,GAAG,MAAM,CAAS,CAAC,CAAC,CAAC;IAChC,YAAY,GAAG,QAAQ,CAAgB,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC;IAC5E,qBAAqB;IACZ,WAAW,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAC1C,aAAa,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAC5C,YAAY,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IACpD,iBAAiB;IACR,YAAY,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAC3C,cAAc,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAC7C,cAAc,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAEtD,8BAA8B;IACrB,QAAQ,GAAG,SAAS,CAAC;IACrB,MAAM,GAAG,MAAM,CAAC;IAChB,KAAK,GAAG,KAAK,CAAC;IACd,OAAO,GAAG,OAAO,CAAC;IAClB,OAAO,GAAG,OAAO,CAAC;IAE3B,yBAAyB;IAChB,WAAW,GAAG,QAAQ,CAAC,GAAG,EAAE;QACnC,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,UAAU;YAAE,OAAO,qBAAqB,CAAC;QAC7D,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,MAAM;YAAE,OAAO,UAAU,CAAC;QAC9C,OAAO,YAAY,CAAC;IACtB,CAAC,CAAC,CAAC;IACM,gBAAgB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC;IAE7E,gCAAgC;IACxB,QAAQ,GAA4B,IAAI,CAAC;IACzC,SAAS,GAAe,IAAI,CAAC;IAC7B,eAAe,GAAG,KAAK,CAAC;IAEhC,YAA6B,UAAsB;QAAtB,eAAU,GAAV,UAAU,CAAY;QACjD,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC9B,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;QACxC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;QAE1C,uDAAuD;QACvD,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,IAAI,GAAG;gBACX,IAAI,CAAC,WAAW,EAAE;gBAClB,IAAI,CAAC,aAAa,EAAE;gBACpB,IAAI,CAAC,YAAY,EAAE;gBACnB,IAAI,CAAC,YAAY,EAAE;gBACnB,IAAI,CAAC,cAAc,EAAE;gBACrB,IAAI,CAAC,cAAc,EAAE;gBACrB,IAAI,CAAC,IAAI,EAAE;aACZ,CAAC;YACF,KAAK,IAAI,CAAC;YACV,IAAI,IAAI,CAAC,cAAc,EAAE,IAAI,IAAI,CAAC,eAAe;gBAAE,OAAO;YAC1D,uEAAuE;YACvE,oEAAoE;YACpE,2DAA2D;YAC3D,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;YAC5B,cAAc,CAAC,GAAG,EAAE;gBAClB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;gBAC7B,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,+CAA+C;QAC/C,MAAM,CAAC,GAAG,EAAE;YACV,IAAI,IAAI,CAAC,cAAc,EAAE;gBAAE,OAAO;YAClC,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,MAAM;gBAAE,OAAO;YACnC,IACE,IAAI,CAAC,YAAY,EAAE,KAAK,IAAI;gBAC5B,IAAI,CAAC,cAAc,EAAE,KAAK,IAAI;gBAC9B,IAAI,CAAC,cAAc,EAAE,KAAK,IAAI,EAC9B,CAAC;gBACD,OAAO;YACT,CAAC;YACD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YAC5B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YAC9B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,uBAAuB;IACvB;;;OAGG;IACH,UAAU,CAAC,KAAa;QACtB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC9B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC;QAEpC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;YAC9B,IAAI,CAAC,cAAc,EAAE,CAAC;YACtB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;YACxC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;YAC1C,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAC/B,OAAO;QACT,CAAC;QAED,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,MAAM,EAAE,CAAC;gBAC3B,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAChD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC;gBACpC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,IAAI,IAAI,CAAC,CAAC;gBACxC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,IAAI,IAAI,CAAC,CAAC;YAC1C,CAAC;iBAAM,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,UAAU,EAAE,CAAC;gBACtC,MAAM,CAAC,QAAQ,EAAE,QAAQ,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAC9C,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;gBAC7B,IAAI,QAAQ,EAAE,CAAC;oBACb,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;oBACnD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC;oBACpC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,IAAI,IAAI,CAAC,CAAC;oBACxC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,IAAI,IAAI,CAAC,CAAC;gBAC1C,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;YAC5B,CAAC;YAED,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACzC,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;YAC3C,IAAI,YAAY,IAAI,aAAa,KAAK,IAAI,EAAE,CAAC;gBAC3C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;gBACnC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;YACvC,CAAC;QACH,CAAC;QACD,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED;;;OAGG;IACH,gBAAgB,CAAC,EAA2B;QAC1C,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED;;;OAGG;IACH,iBAAiB,CAAC,EAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,gBAAgB,CAAC,UAAmB;QAClC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAC9B,IAAI,UAAU,EAAE,CAAC;YACf,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,oBAAoB;IACpB;;;OAGG;IACH,eAAe,CAAC,KAAiB;QAC/B,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC;YAC1D,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,oCAAoC;IACpC,MAAM;QACJ,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,aAAa;IACb;;;OAGG;IACH,YAAY,CAAC,KAAY;QACvB,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,KAAK,CAAC,eAAe,EAAE,CAAC;QACxB,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC;IAC5C,CAAC;IAED,gDAAgD;IAChD,aAAa;QACX,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC;QACtC,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;YACd,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC1B,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YAC9C,OAAO;QACT,CAAC;QACD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,4CAA4C;IAC5C,SAAS;QACP,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC;QACtC,IAAI,KAAK,GAAG,EAAE,EAAE,CAAC;YACf,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YACzB,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YAC9C,OAAO;QACT,CAAC;QACD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACK,aAAa;QACnB,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,MAAM,EAAE,CAAC;YAC3B,OAAO,EAAE,CAAC;QACZ,CAAC;QACD,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QACvC,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QACzC,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC9B,MAAM,QAAQ,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAE7C,OAAO,iBAAiB,CAAC,WAAW,EAAE,YAAY,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;IACvE,CAAC;IAED;;;OAGG;IACH,UAAU,CAAC,GAAgB;QACzB,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC/B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAClC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEhC,kDAAkD;QAClD,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,MAAM,EAAE,CAAC;YAC3B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,UAAU,CAAC,IAAY;QACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACH,YAAY,CAAC,MAAc;QACzB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAClC,CAAC;IAED;;;OAGG;IACH,YAAY,CAAC,MAAc;QACzB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAClC,CAAC;IAED,oDAAoD;IACpD,OAAO;QACL,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC3B,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,mBAAmB;IACnB;;;OAGG;IACK,aAAa,CAAC,QAAgB;QACpC,MAAM,MAAM,GAAG,qBAAqB,CAAC,QAAQ,CAAC,CAAC;QAC/C,IAAI,CAAC,MAAM;YAAE,OAAO;QACpB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACnC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACrC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IACnC,CAAC;IAED,2DAA2D;IACnD,WAAW;QACjB,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,MAAM,EAAE,CAAC;YAC3B,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACjC,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACrC,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACrC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM;gBAAE,OAAO;YACxC,MAAM,SAAS,GAAG,GAAG,IAAI,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;YAChD,IAAI,SAAS,KAAK,IAAI,CAAC,aAAa,EAAE;gBAAE,OAAO;YAC/C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YAClC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YACzB,OAAO;QACT,CAAC;QAED,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QACvC,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QAC3C,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QACzC,IAAI,WAAW,IAAI,aAAa,KAAK,IAAI,IAAI,YAAY,EAAE,CAAC;YAC1D,MAAM,KAAK,GAAG,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;YAC9D,MAAM,GAAG,GAAG,WAAW,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;YACpD,MAAM,QAAQ,GAAG,GAAG,YAAY,IAAI,KAAK,IAAI,GAAG,EAAE,CAAC;YAEnD,MAAM,SAAS,GACb,IAAI,CAAC,IAAI,EAAE,KAAK,UAAU;gBACxB,CAAC,CAAC,GAAG,QAAQ,IAAI,IAAI,CAAC,YAAY,EAAE,IAAI,IAAI,IAAI,IAAI,CAAC,cAAc,EAAE,IAAI,IAAI,IAAI,IAAI,CAAC,cAAc,EAAE,IAAI,IAAI,EAAE;gBAChH,CAAC,CAAC,QAAQ,CAAC;YACf,IAAI,SAAS,KAAK,IAAI,CAAC,aAAa,EAAE;gBAAE,OAAO;YAC/C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YAClC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAED,uDAAuD;IAC/C,cAAc;QACpB,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC3B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC7B,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC5B,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC5B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC9B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;IAED;;;OAGG;IACK,oBAAoB;QAC1B,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC/B,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACnC,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QACjC,IAAI,GAAG,KAAK,IAAI,IAAI,KAAK,KAAK,IAAI,IAAI,IAAI,KAAK,IAAI;YAAE,OAAO,IAAI,CAAC;QACjE,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;IAC9B,CAAC;IAED,0CAA0C;IAClC,QAAQ;QACd,OAAO,IAAI,IAAI,EAAE,CAAC;IACpB,CAAC;wGAhWU,0BAA0B;4FAA1B,0BAA0B,03BAR1B;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,0BAA0B,CAAC;gBACzD,KAAK,EAAE,IAAI;aACZ;SACF,0BClDH,u9NAiMA,wmKDzJY,YAAY,8BAAE,WAAW;;4FAYxB,0BAA0B;kBAhBtC,SAAS;+BACE,sBAAsB,WAGvB,CAAC,YAAY,EAAE,WAAW,CAAC,QAC9B;wBACJ,kBAAkB,EAAE,yBAAyB;qBAC9C,aACU;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,2BAA2B,CAAC;4BACzD,KAAK,EAAE,IAAI;yBACZ;qBACF","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  computed,\n  effect,\n  ElementRef,\n  forwardRef,\n  input,\n  signal,\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { TegelModule } from '@scania/tegel-angular-17';\nimport { noop } from 'rxjs';\nimport {\n  HOURS,\n  MINUTES,\n  MONTHS,\n  SECONDS,\n  WEEK_DAYS,\n} from './tae-date-time-picker.constants';\nimport {\n  buildCalendarDays,\n  formatDisplayDateTime,\n  parseDatePartsFromIso,\n} from './tae-date-time-picker.utils';\n\nimport type { CalendarDay, DateParts } from './tae-date-time-picker.utils';\n\n/**\n * Standalone date/time picker that integrates with Angular forms via CVA.\n *\n * Supports `date`, `datetime`, and `time` modes, exposes a typed API via inputs,\n * and emits form updates through the registered CVA callbacks.\n *\n * Adheres to Tegel design guidelines and can be used as a drop-in replacement.\n */\n@Component({\n  selector: 'tae-date-time-picker',\n  templateUrl: './tae-date-time-picker.component.html',\n  styleUrls: ['./tae-date-time-picker.component.scss'],\n  imports: [CommonModule, TegelModule],\n  host: {\n    '(document:click)': 'onDocumentClick($event)',\n  },\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => TaeDateTimePickerComponent),\n      multi: true,\n    },\n  ],\n})\nexport class TaeDateTimePickerComponent implements ControlValueAccessor {\n  // Inputs (public API)\n  readonly label = input('');\n  readonly labelPosition = input<'inside' | 'outside' | 'no-label'>('outside');\n  readonly size = input<'sm' | 'md' | 'lg'>('lg');\n  readonly state = input<'error' | undefined>(undefined);\n  readonly helper = input('');\n  readonly mode = input<'date' | 'datetime' | 'time'>('date');\n\n  // Form control state\n  private readonly internalValue = signal('');\n  readonly displayValue = computed(() =>\n    formatDisplayDateTime(this.internalValue(), this.mode()),\n  );\n  private readonly suppressChange = signal(false);\n  readonly disabled = signal(false);\n  // UI metadata\n  readonly labelId = `tae-dtp-label-${crypto.randomUUID()}`;\n  // Picker visibility\n  readonly showPicker = signal(false);\n  // Calendar view state\n  readonly currentMonth = signal<number>(0);\n  readonly currentYear = signal<number>(0);\n  readonly calendarDays = computed<CalendarDay[]>(() => this.buildCalendar());\n  // Calendar selection\n  readonly selectedDay = signal<number | null>(null);\n  readonly selectedMonth = signal<number | null>(null);\n  readonly selectedYear = signal<number | null>(null);\n  // Time selection\n  readonly selectedHour = signal<string | null>(null);\n  readonly selectedMinute = signal<string | null>(null);\n  readonly selectedSecond = signal<string | null>(null);\n\n  // UI constants (static lists)\n  readonly weekDays = WEEK_DAYS;\n  readonly months = MONTHS;\n  readonly hours = HOURS;\n  readonly minutes = MINUTES;\n  readonly seconds = SECONDS;\n\n  // Derived display values\n  readonly placeholder = computed(() => {\n    if (this.mode() === 'datetime') return 'dd-MM-YYYY HH:mm:ss';\n    if (this.mode() === 'time') return 'HH:mm:ss';\n    return 'dd-MM-YYYY';\n  });\n  readonly currentMonthName = computed(() => this.months[this.currentMonth()]);\n\n  // CVA callbacks (Angular Forms)\n  private onChange: (value: string) => void = noop;\n  private onTouched: () => void = noop;\n  private updateScheduled = false;\n\n  constructor(private readonly elementRef: ElementRef) {\n    const today = this.getToday();\n    this.currentMonth.set(today.getMonth());\n    this.currentYear.set(today.getFullYear());\n\n    // Sync internal ISO value when date/time parts change.\n    effect(() => {\n      const deps = [\n        this.selectedDay(),\n        this.selectedMonth(),\n        this.selectedYear(),\n        this.selectedHour(),\n        this.selectedMinute(),\n        this.selectedSecond(),\n        this.mode(),\n      ];\n      void deps;\n      if (this.suppressChange() || this.updateScheduled) return;\n      // Schedule the update for the next microtask so Angular can finish the\n      // current change detection pass before we emit a form value change.\n      // This avoids ExpressionChangedAfterItHasBeenCheckedError.\n      this.updateScheduled = true;\n      queueMicrotask(() => {\n        this.updateScheduled = false;\n        this.updateValue();\n      });\n    });\n\n    // Normalize time parts when in date-only mode.\n    effect(() => {\n      if (this.suppressChange()) return;\n      if (this.mode() !== 'date') return;\n      if (\n        this.selectedHour() === null &&\n        this.selectedMinute() === null &&\n        this.selectedSecond() === null\n      ) {\n        return;\n      }\n      this.selectedHour.set(null);\n      this.selectedMinute.set(null);\n      this.selectedSecond.set(null);\n    });\n  }\n\n  // ControlValueAccessor\n  /**\n   * Writes an ISO value from the outside into the picker state.\n   * @param value ISO date/datetime string.\n   */\n  writeValue(value: string): void {\n    this.suppressChange.set(true);\n    this.internalValue.set(value || '');\n\n    if (!value) {\n      const today = this.getToday();\n      this.resetSelection();\n      this.currentMonth.set(today.getMonth());\n      this.currentYear.set(today.getFullYear());\n      this.suppressChange.set(false);\n      return;\n    }\n\n    if (value) {\n      if (this.mode() === 'time') {\n        const [hour, minute, second] = value.split(':');\n        this.selectedHour.set(hour || null);\n        this.selectedMinute.set(minute || null);\n        this.selectedSecond.set(second || null);\n      } else if (this.mode() === 'datetime') {\n        const [datePart, timePart] = value.split('T');\n        this.parseDatePart(datePart);\n        if (timePart) {\n          const [hour, minute, second] = timePart.split(':');\n          this.selectedHour.set(hour || null);\n          this.selectedMinute.set(minute || null);\n          this.selectedSecond.set(second || null);\n        }\n      } else {\n        this.parseDatePart(value);\n      }\n\n      const selectedYear = this.selectedYear();\n      const selectedMonth = this.selectedMonth();\n      if (selectedYear && selectedMonth !== null) {\n        this.currentYear.set(selectedYear);\n        this.currentMonth.set(selectedMonth);\n      }\n    }\n    this.suppressChange.set(false);\n  }\n\n  /**\n   * Registers the CVA change handler.\n   * @param fn Callback invoked on value changes.\n   */\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n\n  /**\n   * Registers the CVA touched handler.\n   * @param fn Callback invoked on touch.\n   */\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  /**\n   * Updates the disabled state for the control.\n   * @param isDisabled Whether the control is disabled.\n   */\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled.set(isDisabled);\n    if (isDisabled) {\n      this.showPicker.set(false);\n    }\n  }\n\n  // Host/DOM handlers\n  /**\n   * Closes the picker when clicking outside the component.\n   * @param event Document click event.\n   */\n  onDocumentClick(event: MouseEvent): void {\n    if (!this.elementRef.nativeElement.contains(event.target)) {\n      this.showPicker.set(false);\n    }\n  }\n\n  /** Marks the control as touched. */\n  onBlur(): void {\n    this.onTouched();\n  }\n\n  // UI actions\n  /**\n   * Toggles the picker popover.\n   * @param event Click event from the control.\n   */\n  togglePicker(event: Event): void {\n    event.preventDefault();\n    event.stopPropagation();\n    this.showPicker.update((value) => !value);\n  }\n\n  /** Moves the calendar to the previous month. */\n  previousMonth(): void {\n    const month = this.currentMonth() - 1;\n    if (month < 0) {\n      this.currentMonth.set(11);\n      this.currentYear.update((value) => value - 1);\n      return;\n    }\n    this.currentMonth.set(month);\n  }\n\n  /** Moves the calendar to the next month. */\n  nextMonth(): void {\n    const month = this.currentMonth() + 1;\n    if (month > 11) {\n      this.currentMonth.set(0);\n      this.currentYear.update((value) => value + 1);\n      return;\n    }\n    this.currentMonth.set(month);\n  }\n\n  /**\n   * Builds the calendar day grid for the current month.\n   * @returns Calendar day entries for rendering.\n   */\n  private buildCalendar(): CalendarDay[] {\n    if (this.mode() === 'time') {\n      return [];\n    }\n    const currentYear = this.currentYear();\n    const currentMonth = this.currentMonth();\n    const today = this.getToday();\n    const selected = this.getSelectedDateParts();\n\n    return buildCalendarDays(currentYear, currentMonth, selected, today);\n  }\n\n  /**\n   * Selects a date from the calendar grid.\n   * @param day Calendar day entry.\n   */\n  selectDate(day: CalendarDay): void {\n    this.selectedDay.set(day.date);\n    this.selectedMonth.set(day.month);\n    this.selectedYear.set(day.year);\n\n    // In date-only mode, close picker after selection\n    if (this.mode() === 'date') {\n      this.showPicker.set(false);\n    }\n  }\n\n  /**\n   * Selects an hour in datetime mode.\n   * @param hour Two-digit hour string.\n   */\n  selectHour(hour: string): void {\n    this.selectedHour.set(hour);\n  }\n\n  /**\n   * Selects a minute in datetime mode.\n   * @param minute Two-digit minute string.\n   */\n  selectMinute(minute: string): void {\n    this.selectedMinute.set(minute);\n  }\n\n  /**\n   * Selects a second in datetime mode.\n   * @param second Two-digit second string.\n   */\n  selectSecond(second: string): void {\n    this.selectedSecond.set(second);\n  }\n\n  /** Confirms the selection and closes the picker. */\n  confirm(): void {\n    this.showPicker.set(false);\n    this.onTouched();\n  }\n\n  // Internal helpers\n  /**\n   * Parses an ISO date segment into selected date parts.\n   * @param datePart ISO date segment.\n   */\n  private parseDatePart(datePart: string): void {\n    const parsed = parseDatePartsFromIso(datePart);\n    if (!parsed) return;\n    this.selectedYear.set(parsed.year);\n    this.selectedMonth.set(parsed.month);\n    this.selectedDay.set(parsed.day);\n  }\n\n  /** Computes and emits the ISO value from current parts. */\n  private updateValue(): void {\n    if (this.mode() === 'time') {\n      const hour = this.selectedHour();\n      const minute = this.selectedMinute();\n      const second = this.selectedSecond();\n      if (!hour || !minute || !second) return;\n      const nextValue = `${hour}:${minute}:${second}`;\n      if (nextValue === this.internalValue()) return;\n      this.internalValue.set(nextValue);\n      this.onChange(nextValue);\n      return;\n    }\n\n    const selectedDay = this.selectedDay();\n    const selectedMonth = this.selectedMonth();\n    const selectedYear = this.selectedYear();\n    if (selectedDay && selectedMonth !== null && selectedYear) {\n      const month = (selectedMonth + 1).toString().padStart(2, '0');\n      const day = selectedDay.toString().padStart(2, '0');\n      const datePart = `${selectedYear}-${month}-${day}`;\n\n      const nextValue =\n        this.mode() === 'datetime'\n          ? `${datePart}T${this.selectedHour() ?? '00'}:${this.selectedMinute() ?? '00'}:${this.selectedSecond() ?? '00'}`\n          : datePart;\n      if (nextValue === this.internalValue()) return;\n      this.internalValue.set(nextValue);\n      this.onChange(nextValue);\n    }\n  }\n\n  /** Resets all selected date/time parts to defaults. */\n  private resetSelection(): void {\n    this.selectedDay.set(null);\n    this.selectedMonth.set(null);\n    this.selectedYear.set(null);\n    this.selectedHour.set(null);\n    this.selectedMinute.set(null);\n    this.selectedSecond.set(null);\n  }\n\n  /**\n   * Returns current selected date parts when fully defined.\n   * @returns Date parts or `null` if incomplete.\n   */\n  private getSelectedDateParts(): DateParts | null {\n    const day = this.selectedDay();\n    const month = this.selectedMonth();\n    const year = this.selectedYear();\n    if (day === null || month === null || year === null) return null;\n    return { day, month, year };\n  }\n\n  /** Returns today's date in local time. */\n  private getToday(): Date {\n    return new Date();\n  }\n}\n","<div class=\"date-picker-wrapper\">\r\n  <label\n    class=\"date-picker-label\"\n    [class.outside]=\"labelPosition() === 'outside'\"\n    [attr.for]=\"labelId\"\n    >{{ label() }}</label\n  >\n  <div\n    [id]=\"labelId\"\n    class=\"date-picker-container\"\n    [class.error]=\"state() === 'error'\"\n    [class.size-sm]=\"size() === 'sm'\"\r\n    [class.size-md]=\"size() === 'md'\"\r\n    [class.size-lg]=\"size() === 'lg'\"\r\n    [class.disabled]=\"disabled()\"\r\n    (click)=\"!disabled() && togglePicker($event)\"\r\n    (keydown.enter)=\"!disabled() && togglePicker($event)\"\r\n    (keydown.space)=\"!disabled() && togglePicker($event)\"\r\n    role=\"button\"\r\n    tabindex=\"0\"\r\n  >\r\n    <span class=\"display-value\" [class.placeholder]=\"!displayValue()\">{{\r\n      displayValue() || placeholder()\r\n    }}</span>\r\n    <div class=\"picker-icon\">\r\n      <svg\r\n        xmlns=\"http://www.w3.org/2000/svg\"\r\n        width=\"16\"\r\n        height=\"16\"\r\n        viewBox=\"0 0 24 24\"\r\n        fill=\"none\"\r\n        stroke=\"currentColor\"\r\n        stroke-width=\"2\"\r\n        stroke-linecap=\"round\"\r\n        stroke-linejoin=\"round\"\r\n      >\r\n        <rect x=\"3\" y=\"4\" width=\"18\" height=\"18\" rx=\"2\" ry=\"2\"></rect>\r\n        <line x1=\"16\" y1=\"2\" x2=\"16\" y2=\"6\"></line>\r\n        <line x1=\"8\" y1=\"2\" x2=\"8\" y2=\"6\"></line>\r\n        <line x1=\"3\" y1=\"10\" x2=\"21\" y2=\"10\"></line>\r\n      </svg>\r\n    </div>\r\n  </div>\r\n\r\n  @if (showPicker()) {\n    <div class=\"picker-popover\" [class.datetime-mode]=\"mode() !== 'date'\">\n      <div class=\"picker-content\">\n        <!-- Calendar Section -->\n        @if (mode() !== 'time') {\n          <div class=\"calendar-section\">\n          <div class=\"calendar-header\">\n            <button type=\"button\" class=\"nav-btn\" (click)=\"previousMonth()\">\n              <svg\r\n                xmlns=\"http://www.w3.org/2000/svg\"\r\n                width=\"16\"\r\n                height=\"16\"\r\n                viewBox=\"0 0 24 24\"\r\n                fill=\"none\"\r\n                stroke=\"currentColor\"\r\n                stroke-width=\"2\"\r\n                stroke-linecap=\"round\"\r\n                stroke-linejoin=\"round\"\r\n              >\r\n                <polyline points=\"15 18 9 12 15 6\"></polyline>\r\n              </svg>\r\n            </button>\r\n            <span class=\"month-year\"\r\n              >{{ currentMonthName() }} {{ currentYear() }}</span\r\n            >\r\n            <button type=\"button\" class=\"nav-btn\" (click)=\"nextMonth()\">\r\n              <svg\r\n                xmlns=\"http://www.w3.org/2000/svg\"\r\n                width=\"16\"\r\n                height=\"16\"\r\n                viewBox=\"0 0 24 24\"\r\n                fill=\"none\"\r\n                stroke=\"currentColor\"\r\n                stroke-width=\"2\"\r\n                stroke-linecap=\"round\"\r\n                stroke-linejoin=\"round\"\r\n              >\r\n                <polyline points=\"9 18 15 12 9 6\"></polyline>\r\n              </svg>\r\n            </button>\n          </div>\n          <div class=\"calendar-weekdays\">\n            @for (day of weekDays; track day) {\n              <span class=\"weekday\">{{ day }}</span>\n            }\n          </div>\n          <div class=\"calendar-days\">\n            @for (\n              day of calendarDays();\n              track day.date + '-' + day.month + '-' + day.year\n            ) {\n              <button\n                type=\"button\"\n                class=\"calendar-day\"\n                [disabled]=\"disabled()\"\n                [class.other-month]=\"!day.isCurrentMonth\"\n                [class.today]=\"day.isToday\"\n                [class.selected]=\"day.isSelected\"\n                (click)=\"selectDate(day); $event.stopPropagation()\"\n              >\n                {{ day.date }}\n              </button>\n            }\n          </div>\n        </div>\n        }\n\n        <!-- Time Section (only for datetime mode) -->\n        @if (mode() !== 'date') {\n          <div class=\"time-section\">\n            <div class=\"time-section-header\">\r\n              <span>Time (24h)</span>\r\n            </div>\r\n            <div class=\"time-picker-body\">\r\n              <div class=\"time-column\">\r\n                <div class=\"time-column-label\">Hour</div>\r\n                <div class=\"time-column-scroll\">\r\n                  @for (hour of hours; track hour) {\r\n                    <button\r\n                      type=\"button\"\r\n                      class=\"time-option\"\r\n                      [disabled]=\"disabled()\"\r\n                      [class.selected]=\"hour === selectedHour()\"\n                      (click)=\"selectHour(hour); $event.stopPropagation()\"\n                    >\n                      {{ hour }}\n                    </button>\n                  }\r\n                </div>\r\n              </div>\r\n              <div class=\"time-separator\">:</div>\r\n              <div class=\"time-column\">\r\n                <div class=\"time-column-label\">Min</div>\r\n                <div class=\"time-column-scroll\">\r\n                  @for (minute of minutes; track minute) {\r\n                    <button\r\n                      type=\"button\"\r\n                      class=\"time-option\"\r\n                      [disabled]=\"disabled()\"\r\n                      [class.selected]=\"minute === selectedMinute()\"\n                      (click)=\"selectMinute(minute); $event.stopPropagation()\"\n                    >\n                      {{ minute }}\n                    </button>\n                  }\r\n                </div>\r\n              </div>\r\n              <div class=\"time-separator\">:</div>\r\n              <div class=\"time-column\">\r\n                <div class=\"time-column-label\">Sec</div>\r\n                <div class=\"time-column-scroll\">\r\n                  @for (second of seconds; track second) {\r\n                    <button\r\n                      type=\"button\"\r\n                      class=\"time-option\"\r\n                      [disabled]=\"disabled()\"\r\n                      [class.selected]=\"second === selectedSecond()\"\n                      (click)=\"selectSecond(second); $event.stopPropagation()\"\n                    >\n                      {{ second }}\n                    </button>\n                  }\r\n                </div>\r\n              </div>\r\n            </div>\r\n          </div>\r\n        }\r\n      </div>\r\n\r\n      <!-- Footer -->\r\n      <div class=\"picker-footer\">\r\n        <button\r\n          type=\"button\"\r\n          class=\"confirm-btn\"\r\n          [disabled]=\"disabled()\"\r\n          (click)=\"confirm(); $event.stopPropagation()\"\r\n        >\r\n          OK\r\n        </button>\r\n      </div>\r\n    </div>\r\n  }\r\n\r\n  @if (helper()) {\r\n    <span class=\"helper-text\" [class.error]=\"state() === 'error'\">{{\r\n      helper()\r\n    }}</span>\r\n  }\r\n</div>\r\n"]}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
/** Short weekday labels */
|
|
2
|
+
export const WEEK_DAYS = ['Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa', 'Su'];
|
|
3
|
+
/** Full month names for display in the calendar header. */
|
|
4
|
+
export const MONTHS = [
|
|
5
|
+
'January',
|
|
6
|
+
'February',
|
|
7
|
+
'March',
|
|
8
|
+
'April',
|
|
9
|
+
'May',
|
|
10
|
+
'June',
|
|
11
|
+
'July',
|
|
12
|
+
'August',
|
|
13
|
+
'September',
|
|
14
|
+
'October',
|
|
15
|
+
'November',
|
|
16
|
+
'December',
|
|
17
|
+
];
|
|
18
|
+
/** Zero-padded hours for 24-hour time selection. */
|
|
19
|
+
export const HOURS = Array.from({ length: 24 }, (_, i) => i.toString().padStart(2, '0'));
|
|
20
|
+
/** Zero-padded minutes for time selection. */
|
|
21
|
+
export const MINUTES = Array.from({ length: 60 }, (_, i) => i.toString().padStart(2, '0'));
|
|
22
|
+
/** Zero-padded seconds for time selection. */
|
|
23
|
+
export const SECONDS = Array.from({ length: 60 }, (_, i) => i.toString().padStart(2, '0'));
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFlLWRhdGUtdGltZS1waWNrZXIuY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy90ZWdlbC1hbmd1bGFyLWV4dGVuc2lvbnMvc3JjL2xpYi9jb21wb25lbnRzL3RhZS1kYXRlLXRpbWUtcGlja2VyL3RhZS1kYXRlLXRpbWUtcGlja2VyLmNvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSwyQkFBMkI7QUFDM0IsTUFBTSxDQUFDLE1BQU0sU0FBUyxHQUFHLENBQUMsSUFBSSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsSUFBSSxDQUFVLENBQUM7QUFFN0UsMkRBQTJEO0FBQzNELE1BQU0sQ0FBQyxNQUFNLE1BQU0sR0FBRztJQUNwQixTQUFTO0lBQ1QsVUFBVTtJQUNWLE9BQU87SUFDUCxPQUFPO0lBQ1AsS0FBSztJQUNMLE1BQU07SUFDTixNQUFNO0lBQ04sUUFBUTtJQUNSLFdBQVc7SUFDWCxTQUFTO0lBQ1QsVUFBVTtJQUNWLFVBQVU7Q0FDRixDQUFDO0FBRVgsb0RBQW9EO0FBQ3BELE1BQU0sQ0FBQyxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsTUFBTSxFQUFFLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQ3ZELENBQUMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxFQUFFLEdBQUcsQ0FBQyxDQUM5QixDQUFDO0FBRUYsOENBQThDO0FBQzlDLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsTUFBTSxFQUFFLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQ3pELENBQUMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxFQUFFLEdBQUcsQ0FBQyxDQUM5QixDQUFDO0FBRUYsOENBQThDO0FBQzlDLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsTUFBTSxFQUFFLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQ3pELENBQUMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxFQUFFLEdBQUcsQ0FBQyxDQUM5QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqIFNob3J0IHdlZWtkYXkgbGFiZWxzICovXG5leHBvcnQgY29uc3QgV0VFS19EQVlTID0gWydNbycsICdUdScsICdXZScsICdUaCcsICdGcicsICdTYScsICdTdSddIGFzIGNvbnN0O1xuXG4vKiogRnVsbCBtb250aCBuYW1lcyBmb3IgZGlzcGxheSBpbiB0aGUgY2FsZW5kYXIgaGVhZGVyLiAqL1xuZXhwb3J0IGNvbnN0IE1PTlRIUyA9IFtcbiAgJ0phbnVhcnknLFxuICAnRmVicnVhcnknLFxuICAnTWFyY2gnLFxuICAnQXByaWwnLFxuICAnTWF5JyxcbiAgJ0p1bmUnLFxuICAnSnVseScsXG4gICdBdWd1c3QnLFxuICAnU2VwdGVtYmVyJyxcbiAgJ09jdG9iZXInLFxuICAnTm92ZW1iZXInLFxuICAnRGVjZW1iZXInLFxuXSBhcyBjb25zdDtcblxuLyoqIFplcm8tcGFkZGVkIGhvdXJzIGZvciAyNC1ob3VyIHRpbWUgc2VsZWN0aW9uLiAqL1xuZXhwb3J0IGNvbnN0IEhPVVJTID0gQXJyYXkuZnJvbSh7IGxlbmd0aDogMjQgfSwgKF8sIGkpID0+XG4gIGkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpLFxuKTtcblxuLyoqIFplcm8tcGFkZGVkIG1pbnV0ZXMgZm9yIHRpbWUgc2VsZWN0aW9uLiAqL1xuZXhwb3J0IGNvbnN0IE1JTlVURVMgPSBBcnJheS5mcm9tKHsgbGVuZ3RoOiA2MCB9LCAoXywgaSkgPT5cbiAgaS50b1N0cmluZygpLnBhZFN0YXJ0KDIsICcwJyksXG4pO1xuXG4vKiogWmVyby1wYWRkZWQgc2Vjb25kcyBmb3IgdGltZSBzZWxlY3Rpb24uICovXG5leHBvcnQgY29uc3QgU0VDT05EUyA9IEFycmF5LmZyb20oeyBsZW5ndGg6IDYwIH0sIChfLCBpKSA9PlxuICBpLnRvU3RyaW5nKCkucGFkU3RhcnQoMiwgJzAnKSxcbik7XG4iXX0=
|