@elderbyte/ngx-starter 18.9.0 → 18.11.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/esm2022/lib/common/time/public_api.mjs +3 -1
- package/esm2022/lib/common/time/temporal-plain-date-interval.mjs +34 -0
- package/esm2022/lib/common/time/temporal-util.mjs +187 -0
- package/esm2022/lib/components/errors/error-util.mjs +2 -2
- package/esm2022/lib/components/time/elder-interval-input/elder-interval-input.component.mjs +13 -4
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker-binding.directive.mjs +45 -0
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker-toggle/elder-interval-picker-toggle.component.mjs +203 -0
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker.component.mjs +269 -0
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker.service.mjs +703 -0
- package/esm2022/lib/components/time/elder-time.module.mjs +22 -2
- package/fesm2022/elderbyte-ngx-starter.mjs +1791 -376
- package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/common/time/public_api.d.ts +2 -0
- package/lib/common/time/temporal-plain-date-interval.d.ts +15 -0
- package/lib/common/time/temporal-util.d.ts +34 -0
- package/lib/components/time/elder-interval-input/elder-interval-input.component.d.ts +4 -1
- package/lib/components/time/elder-interval-picker/elder-interval-picker-binding.directive.d.ts +30 -0
- package/lib/components/time/elder-interval-picker/elder-interval-picker-toggle/elder-interval-picker-toggle.component.d.ts +48 -0
- package/lib/components/time/elder-interval-picker/elder-interval-picker.component.d.ts +73 -0
- package/lib/components/time/elder-interval-picker/elder-interval-picker.service.d.ts +130 -0
- package/lib/components/time/elder-time.module.d.ts +8 -2
- package/package.json +4 -2
- package/src/assets/i18n/de.json +137 -0
- package/src/assets/i18n/en.json +123 -0
- package/src/lib/components/time/elder-interval-picker/elder-interval-picker.component.scss +36 -0
|
@@ -0,0 +1,269 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { ChangeDetectionStrategy, Component, effect, ElementRef, EventEmitter, input, Input, Output, ViewChild, } from '@angular/core';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import { MatButtonModule } from '@angular/material/button';
|
|
5
|
+
import { DateAdapter } from '@angular/material/core';
|
|
6
|
+
import { MatCalendar, MatDatepickerModule, MatDatepickerToggle, MatDateRangeInput, MatDateRangePicker } from '@angular/material/datepicker';
|
|
7
|
+
import { MatFormFieldModule, MatHint, MatLabel, } from '@angular/material/form-field';
|
|
8
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
9
|
+
import { MatInputModule } from '@angular/material/input';
|
|
10
|
+
import { MatMenuModule } from '@angular/material/menu';
|
|
11
|
+
import { MatTooltip } from '@angular/material/tooltip';
|
|
12
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
13
|
+
import { TranslateModule } from '@ngx-translate/core';
|
|
14
|
+
import { TemporalUtil } from '../../../common/time/temporal-util';
|
|
15
|
+
import { CustomDateAdapter } from '../../../components/time/date-adapters/custom-date-adapter';
|
|
16
|
+
import { ElderLocalDateInputComponent } from '../../../components/time/elder-local-date-input/elder-local-date-input.component';
|
|
17
|
+
import { ElderIntervalPickerService } from './elder-interval-picker.service';
|
|
18
|
+
import * as i0 from "@angular/core";
|
|
19
|
+
import * as i1 from "./elder-interval-picker.service";
|
|
20
|
+
import * as i2 from "@angular/material/datepicker";
|
|
21
|
+
import * as i3 from "@angular/material/form-field";
|
|
22
|
+
import * as i4 from "@angular/material/input";
|
|
23
|
+
import * as i5 from "@angular/common";
|
|
24
|
+
import * as i6 from "@angular/material/icon";
|
|
25
|
+
import * as i7 from "@angular/forms";
|
|
26
|
+
import * as i8 from "@angular/material/button";
|
|
27
|
+
import * as i9 from "@ngx-translate/core";
|
|
28
|
+
import * as i10 from "@angular/material/menu";
|
|
29
|
+
export class ElderIntervalPickerComponent {
|
|
30
|
+
/***************************************************************************
|
|
31
|
+
* *
|
|
32
|
+
* Constructor *
|
|
33
|
+
* *
|
|
34
|
+
**************************************************************************/
|
|
35
|
+
constructor(svc) {
|
|
36
|
+
this.svc = svc;
|
|
37
|
+
/***************************************************************************
|
|
38
|
+
* *
|
|
39
|
+
* Fields *
|
|
40
|
+
* *
|
|
41
|
+
**************************************************************************/
|
|
42
|
+
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
43
|
+
this.calendarAnchorDateCssClassFn = (cellDate, view) => {
|
|
44
|
+
return this.calendarAnchorDateCssClassFnBody(cellDate, view);
|
|
45
|
+
};
|
|
46
|
+
this.isEmittingOnChange = true;
|
|
47
|
+
this.intervalInputMode = 'date-range';
|
|
48
|
+
this.showAnchor = false;
|
|
49
|
+
this.isAnchorReadOnly = false;
|
|
50
|
+
this.isResultTextVisible = true;
|
|
51
|
+
this.isHelpTextsVisible = false;
|
|
52
|
+
this.dateChange = new EventEmitter();
|
|
53
|
+
this.inputInterval = input(null);
|
|
54
|
+
this.inputAnchorDateTime = input(null);
|
|
55
|
+
/***************************************************************************
|
|
56
|
+
* *
|
|
57
|
+
* Effects *
|
|
58
|
+
* *
|
|
59
|
+
**************************************************************************/
|
|
60
|
+
this._updateMatCalendarElAnchorCssClassEffect = effect(() => {
|
|
61
|
+
const anchorDate = this.svc.anchorDate();
|
|
62
|
+
if (anchorDate) {
|
|
63
|
+
this.calendarStart.updateTodaysDate();
|
|
64
|
+
this.calendarEnd.updateTodaysDate();
|
|
65
|
+
}
|
|
66
|
+
});
|
|
67
|
+
this._updateMatCalendarElSelectionEffect = effect(() => {
|
|
68
|
+
const startDate = this.svc.startDate();
|
|
69
|
+
const endDate = this.svc.endDate();
|
|
70
|
+
if (!this.calendarStart || !this.calendarEnd) {
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
if (!startDate && !endDate) {
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
const calendar1 = this.calendarStart;
|
|
77
|
+
const calendar2 = this.calendarEnd;
|
|
78
|
+
setTimeout(() => {
|
|
79
|
+
// timeout is needed to run run calendar update last, otherwise there are issues
|
|
80
|
+
if (startDate) {
|
|
81
|
+
const startJsDate = TemporalUtil.getJSDateFromPlainDateTime(startDate);
|
|
82
|
+
calendar1.activeDate = startJsDate;
|
|
83
|
+
}
|
|
84
|
+
if (endDate) {
|
|
85
|
+
const endJsDate = TemporalUtil.getJSDateFromPlainDateTime(endDate);
|
|
86
|
+
calendar2.activeDate = endJsDate;
|
|
87
|
+
}
|
|
88
|
+
}, 0);
|
|
89
|
+
});
|
|
90
|
+
this.svc.setupInputInterval(this.inputInterval);
|
|
91
|
+
this.svc.setupInputAnchorDateTime(this.inputAnchorDateTime);
|
|
92
|
+
this.svc.setupEventEmitter(this.dateChange);
|
|
93
|
+
}
|
|
94
|
+
/***************************************************************************
|
|
95
|
+
* *
|
|
96
|
+
* Life Cycle *
|
|
97
|
+
* *
|
|
98
|
+
**************************************************************************/
|
|
99
|
+
ngOnInit() {
|
|
100
|
+
if (this.inputInterval) {
|
|
101
|
+
this.svc.pullInputInterval();
|
|
102
|
+
}
|
|
103
|
+
if (this.inputAnchorDateTime()) {
|
|
104
|
+
this.svc.pullInputAnchorDateTime();
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
ngAfterViewInit() {
|
|
108
|
+
this.svc.isEmittingOnChange.set(this.isEmittingOnChange);
|
|
109
|
+
this.setupDateInputCalendarFocusPrevention();
|
|
110
|
+
}
|
|
111
|
+
ngOnDestroy() {
|
|
112
|
+
this.removeEventListeners();
|
|
113
|
+
}
|
|
114
|
+
/***************************************************************************
|
|
115
|
+
* *
|
|
116
|
+
* Public API *
|
|
117
|
+
* *
|
|
118
|
+
**************************************************************************/
|
|
119
|
+
setupDateInputCalendarFocusPrevention() {
|
|
120
|
+
// Prevents the focus on the calendar when the input field is changed via the keyboard
|
|
121
|
+
this.calendarElFocusEventHandler = (event) => {
|
|
122
|
+
if (!event || !event.relatedTarget) {
|
|
123
|
+
return;
|
|
124
|
+
}
|
|
125
|
+
event.preventDefault();
|
|
126
|
+
const relTarget = event.relatedTarget;
|
|
127
|
+
if (relTarget.name === 'startDateInput') {
|
|
128
|
+
this.startDateInput.nativeElement.focus();
|
|
129
|
+
return;
|
|
130
|
+
}
|
|
131
|
+
if (relTarget.name === 'endDateInput') {
|
|
132
|
+
this.endDateInput.nativeElement.focus();
|
|
133
|
+
return;
|
|
134
|
+
}
|
|
135
|
+
};
|
|
136
|
+
const nativeElementStart = this.calendarStartElRef.nativeElement;
|
|
137
|
+
const nativeElementEnd = this.calendarEndElRef.nativeElement;
|
|
138
|
+
nativeElementStart.addEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
139
|
+
nativeElementEnd.addEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
140
|
+
}
|
|
141
|
+
clearStartDateTime() {
|
|
142
|
+
this.svc.clearStartDateTime();
|
|
143
|
+
this.svc.resetCalendarMode();
|
|
144
|
+
if (this.isEmittingOnChange) {
|
|
145
|
+
this.svc.emitMainDates();
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
clearEndDateTime() {
|
|
149
|
+
this.svc.clearEndDateTime();
|
|
150
|
+
this.svc.resetCalendarMode();
|
|
151
|
+
if (this.isEmittingOnChange) {
|
|
152
|
+
this.svc.emitMainDates();
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
clearStartAndEndDateTimes() {
|
|
156
|
+
this.svc.clearStartDateTime();
|
|
157
|
+
this.svc.clearEndDateTime();
|
|
158
|
+
this.svc.resetCalendarMode();
|
|
159
|
+
if (this.isEmittingOnChange) {
|
|
160
|
+
this.svc.emitMainDates();
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
handleStartCalendarChange(changedDate) {
|
|
164
|
+
this.svc.calendarMode.set('days');
|
|
165
|
+
const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate);
|
|
166
|
+
this.svc.startDate.set(newPlainDate);
|
|
167
|
+
}
|
|
168
|
+
handleEndCalendarChange(changedDate) {
|
|
169
|
+
this.svc.calendarMode.set('days');
|
|
170
|
+
const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate);
|
|
171
|
+
this.svc.endDate.set(newPlainDate);
|
|
172
|
+
}
|
|
173
|
+
/***************************************************************************
|
|
174
|
+
* *
|
|
175
|
+
* Private methods *
|
|
176
|
+
* *
|
|
177
|
+
**************************************************************************/
|
|
178
|
+
removeEventListeners() {
|
|
179
|
+
if (!this.calendarStartElRef || !this.calendarEndElRef) {
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
if (!this.calendarStartElRef.nativeElement || !this.calendarEndElRef.nativeElement) {
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
const nativeElementStart = this.calendarStartElRef.nativeElement;
|
|
186
|
+
const nativeElementEnd = this.calendarEndElRef.nativeElement;
|
|
187
|
+
nativeElementStart.removeEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
188
|
+
nativeElementEnd.removeEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
189
|
+
}
|
|
190
|
+
calendarAnchorDateCssClassFnBody(cellDate, view) {
|
|
191
|
+
const anchorDate = this.svc.fixedAnchorDate();
|
|
192
|
+
if (!anchorDate) {
|
|
193
|
+
return '';
|
|
194
|
+
}
|
|
195
|
+
const date = TemporalUtil.getJSDateFromPlainDateTime(anchorDate);
|
|
196
|
+
if (!date) {
|
|
197
|
+
return '';
|
|
198
|
+
}
|
|
199
|
+
if (cellDate.getDate() === date.getDate() &&
|
|
200
|
+
cellDate.getMonth() === date.getMonth() &&
|
|
201
|
+
cellDate.getFullYear() === date.getFullYear()) {
|
|
202
|
+
return 'elder-custom-anchor-date';
|
|
203
|
+
}
|
|
204
|
+
return '';
|
|
205
|
+
}
|
|
206
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.2", ngImport: i0, type: ElderIntervalPickerComponent, deps: [{ token: i1.ElderIntervalPickerService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
207
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.2", type: ElderIntervalPickerComponent, isStandalone: true, selector: "elder-interval-picker", inputs: { isEmittingOnChange: { classPropertyName: "isEmittingOnChange", publicName: "isEmittingOnChange", isSignal: false, isRequired: false, transformFunction: null }, intervalInputMode: { classPropertyName: "intervalInputMode", publicName: "intervalInputMode", isSignal: false, isRequired: false, transformFunction: null }, showAnchor: { classPropertyName: "showAnchor", publicName: "showAnchor", isSignal: false, isRequired: false, transformFunction: null }, isAnchorReadOnly: { classPropertyName: "isAnchorReadOnly", publicName: "isAnchorReadOnly", isSignal: false, isRequired: false, transformFunction: null }, isResultTextVisible: { classPropertyName: "isResultTextVisible", publicName: "isResultTextVisible", isSignal: false, isRequired: false, transformFunction: null }, isHelpTextsVisible: { classPropertyName: "isHelpTextsVisible", publicName: "isHelpTextsVisible", isSignal: false, isRequired: false, transformFunction: null }, inputInterval: { classPropertyName: "inputInterval", publicName: "inputInterval", isSignal: true, isRequired: false, transformFunction: null }, inputAnchorDateTime: { classPropertyName: "inputAnchorDateTime", publicName: "inputAnchorDateTime", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { dateChange: "dateChange" }, providers: [
|
|
208
|
+
ElderIntervalPickerService,
|
|
209
|
+
{ provide: DateAdapter, useClass: CustomDateAdapter },
|
|
210
|
+
], viewQueries: [{ propertyName: "calendarStart", first: true, predicate: ["rangeCalendarStart"], descendants: true }, { propertyName: "calendarEnd", first: true, predicate: ["rangeCalendarEnd"], descendants: true }, { propertyName: "calendarStartElRef", first: true, predicate: ["rangeCalendarStart"], descendants: true, read: ElementRef }, { propertyName: "calendarEndElRef", first: true, predicate: ["rangeCalendarEnd"], descendants: true, read: ElementRef }, { propertyName: "startDateInput", first: true, predicate: ["startDateInput"], descendants: true }, { propertyName: "endDateInput", first: true, predicate: ["endDateInput"], descendants: true }], ngImport: i0, template: "<div class=\"interval-picker-component p-md layout-col gap-xxl\">\n <div class=\"layout-row gap-xxl place-between-start\" style=\"gap: 5%\">\n <div class=\"layout-col gap-md pt-xs\">\n <ng-container *ngTemplateOutlet=\"intervalSelector\"></ng-container>\n </div>\n <div class=\"layout-col\">\n <!-- smart shift -->\n <div class=\"layout-row place-around-center\">\n <button mat-icon-button (click)=\"svc.smartShift(-1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ svc.viewIntervalMessage() }}</span>\n <button mat-icon-button (click)=\"svc.smartShift(1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <!-- calendars and input controls -->\n <ng-container *ngTemplateOutlet=\"calendarsTemplate\"></ng-container>\n </div>\n <div class=\"pt-xs\" style=\"width: 25%\">\n <div class=\"layout-col gap-lg\">\n <div class=\"layout-col gap-xs place-start-start\">\n @if (isHelpTextsVisible) {\n <h5 class=\"mat-h5\">{{ \"intervalPicker.selectSmartInterval\" | translate }}</h5>\n }\n <!-- select current -->\n <button mat-button (click)=\"svc.selectCurrentWeek()\">\n {{ \"intervalPicker.currentPeriod.week\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectCurrentMonth()\">\n {{ \"intervalPicker.currentPeriod.month\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectCurrentQuarter()\">\n {{ \"intervalPicker.currentPeriod.quarter\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectCurrentYear()\">\n {{ \"intervalPicker.currentPeriod.year\" | translate }}\n </button>\n </div>\n <div class=\"fixed-shifts-container\">\n <ng-container *ngTemplateOutlet=\"fixedShifts\"></ng-container>\n </div>\n </div>\n </div>\n </div>\n <div class=\"layout-row place-between-center gap-xxl pt-sm\">\n <div>\n @if (showAnchor) {\n <ng-container *ngTemplateOutlet=\"anchorInputField\"></ng-container>\n }\n </div>\n @if (isResultTextVisible) {\n <!-- result interval -->\n <div class=\"layout-col place-center-center\">\n <div class=\"date-interval mat-caption pt-xs\">\n @if (svc.startDateTimeAsJSDate()) {\n {{ svc.startDateTimeAsJSDate() | date : \"dd.MM. y, HH:mm:ss\" }}\n } @else {\n {{ \"intervalPicker.startDateNotSet\" | translate }}\n }\n <span> - </span>\n @if (svc.endDateTimeAsJSDate()) {\n {{ svc.endDateTimeAsJSDate() | date : \"dd.MM. y, HH:mm:ss\" }}\n } @else {\n {{ \"intervalPicker.endDateNotSet\" | translate }}\n }\n </div>\n <div>\n <span class=\"mat-caption\">{{ svc.deltaHumanReadable() || \" \" }}</span>\n </div>\n </div>\n }\n <div class=\"layout-row gap-lg\">\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"clearStartAndEndDateTimes()\"\n [disabled]=\"svc.isStartAndEndDatesEmpty()\"\n >\n {{ \"intervalPicker.clear\" | translate }}\n </button>\n\n @if(!this.isEmittingOnChange) {\n <button color=\"primary\" mat-raised-button (click)=\"svc.emitMainDates()\">\n {{ \"actions.ok\" | translate }}\n </button>\n }\n </div>\n </div>\n</div>\n\n<ng-template #intervalSelector>\n <div class=\"layout-col\">\n @if (isHelpTextsVisible) {\n <h5 class=\"mat-h5\">{{ \"intervalPicker.selectAbsoluteInterval\" | translate }}</h5>\n }\n <div class=\"layout-row select-buttons-container gap-sm\">\n <div class=\"layout-col gap-xs place-start-stretch\">\n <button mat-button (click)=\"svc.selectCurrentDay()\">\n @if(svc.isFixedAnchorDateSet()) {\n {{ \"intervalPicker.anchorDay\" | translate }}\n } @else {\n {{ \"intervalPicker.today\" | translate }}\n }\n </button>\n <button mat-button (click)=\"svc.selectYesterday()\">\n @if (svc.isFixedAnchorDateSet()) {\n {{ \"intervalPicker.last\" | translate }} 1 {{ \"intervalPicker.day\" | translate }}\n } @else {\n {{ \"intervalPicker.yesterday\" | translate }}\n }\n </button>\n <button mat-button (click)=\"svc.selectLastSevenDays()\">\n {{ \"intervalPicker.last\" | translate }} 7 {{ \"intervalPicker.days\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectLastThirtyDays()\">\n {{ \"intervalPicker.last\" | translate }} 30 {{ \"intervalPicker.days\" | translate }}\n </button>\n @if (intervalInputMode === 'date-time-range') {\n <div class=\"pt-sm\"></div>\n <button mat-button (click)=\"svc.selectLastFiveMinutes()\">\n {{ \"intervalPicker.last\" | translate }} 5 {{ \"intervalPicker.minutes\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectLastHour()\">\n {{ \"intervalPicker.last\" | translate }} {{ \"intervalPicker.hour\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectLast24Hours()\">\n {{ \"intervalPicker.last\" | translate }} 24 {{ \"intervalPicker.hours\" | translate }}\n </button>\n }\n </div>\n </div>\n </div>\n</ng-template>\n\n<ng-template #calendarsTemplate>\n <div class=\"layout-row place-around-center\" style=\"align-items: flex-start; min-height: 280px\">\n <mat-calendar\n #rangeCalendarStart\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"svc.dateRangeForCalendar()\"\n (selectedChange)=\"handleStartCalendarChange($event)\"\n [maxDate]=\"svc.endDateAsJSDate()\"\n [dateClass]=\"calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n <mat-calendar\n #rangeCalendarEnd\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"svc.dateRangeForCalendar()\"\n (selectedChange)=\"handleEndCalendarChange($event)\"\n [minDate]=\"svc.startDateAsJSDate()\"\n [dateClass]=\"calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n </div>\n <div class=\"layout-row place-around-center gap-xl pt-sm\">\n <div class=\"layout-row place-around-center\">\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.startDate\" | translate }}</mat-label>\n <input\n #startDateInput\n name=\"startDateInput\"\n matInput\n type=\"date\"\n [ngModel]=\"svc.startDateHtmlString()\"\n (ngModelChange)=\"svc.setStartDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearStartDateTime()\"\n [disabled]=\"!this.svc.startDate()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n <div class=\"layout-row place-around-center\">\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.endDate\" | translate }}</mat-label>\n <input\n #endDateInput\n name=\"endDateInput\"\n matInput\n type=\"date\"\n [ngModel]=\"svc.endDateHtmlString()\"\n (ngModelChange)=\"svc.setEndDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearEndDateTime()\"\n [disabled]=\"!this.svc.endDate()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n </div>\n <br />\n @if (intervalInputMode === 'date-time-range') {\n <div class=\"layout-row place-around-center gap-xl pt-sm\">\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.startTime\" | translate }}</mat-label>\n <input\n name=\"startTimeInput\"\n matInput\n type=\"time\"\n #startTimeControl=\"ngModel\"\n step=\"1\"\n [ngModel]=\"svc.getStartTimeString()\"\n (ngModelChange)=\"svc.setStartTimeFromString($event, startTimeControl)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.svc.clearStartTime()\"\n [disabled]=\"!this.svc.isStartTimeSetAndNotMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.endTime\" | translate }}</mat-label>\n <input\n name=\"endTimeInput\"\n matInput\n type=\"time\"\n #endTimeControl=\"ngModel\"\n step=\"1\"\n [ngModel]=\"svc.getEndTimeString()\"\n (ngModelChange)=\"svc.setEndTimeFromString($event, endTimeControl)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.svc.clearEndTime()\"\n [disabled]=\"!this.svc.isEndTimeSetAndNotMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n }\n</ng-template>\n\n<ng-template #fixedShifts>\n @if (isHelpTextsVisible) {\n <h5 class=\"mat-h5\">{{ \"intervalPicker.shiftInterval\" | translate }}</h5>\n }\n <div class=\"layout-col place-start-stretch\">\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftDay(-1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.day\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftDay(1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftMonth(-1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.month\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftMonth(1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftYear(-1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.year\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftYear(1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n\n @if (intervalInputMode === 'date-time-range') {\n <!-- shift minute -->\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"svc.shiftTime(-1, 'minutes')\"\n [disabled]=\"svc.isRangeNotSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.minute\" | translate }}</span>\n <button\n mat-icon-button\n (click)=\"svc.shiftTime(1, 'minutes')\"\n [disabled]=\"svc.isRangeNotSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <!-- shift hour -->\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftTime(-1, 'hours')\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.hour\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftTime(1, 'hours')\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n }\n </div>\n</ng-template>\n\n<ng-template #anchorInputField>\n <div class=\"layout-row flex-none gap-md\">\n <mat-menu #anchorMenu=\"matMenu\">\n @if (svc.startDate()) {\n <button\n mat-menu-item\n (click)=\"svc.setFixedAnchorPointToEndDateTime()\"\n [disabled]=\"!svc.endDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n <span>{{ \"intervalPicker.endDate\" | translate }}</span>\n </button>\n } @if (svc.endDate()) {\n <button\n mat-menu-item\n (click)=\"svc.setFixedAnchorPointToStartDateTime()\"\n [disabled]=\"!svc.startDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n <span>{{ \"intervalPicker.startDate\" | translate }}</span>\n </button>\n } @if (svc.fixedAnchorDate()) {\n <button\n mat-menu-item\n (click)=\"this.svc.resetFixedAnchorPoint()\"\n [disabled]=\"!svc.fixedAnchorDate() || isAnchorReadOnly\"\n >\n <mat-icon>close</mat-icon>\n <span>{{ \"intervalPicker.clear\" | translate }}</span>\n </button>\n }\n </mat-menu>\n\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.anchorDate\" | translate }}</mat-label>\n <elder-local-date-input\n name=\"anchorDateInput\"\n (valueUpdated)=\"svc.setAnchorDateFromForm($event)\"\n [value]=\"svc.getAnchorPointAsLocaleDate()\"\n [readonly]=\"isAnchorReadOnly\"\n >\n </elder-local-date-input>\n @if (!isAnchorReadOnly) {\n <button\n mat-icon-button\n matSuffix\n [matMenuTriggerFor]=\"anchorMenu\"\n [disabled]=\"svc.isStartAndEndDatesEmpty() && !svc.fixedAnchorDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">more_horiz</mat-icon>\n </button>\n }\n </mat-form-field>\n @if (intervalInputMode === 'date-time-range') {\n <mat-form-field class=\"input-control-container-short\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.anchorTime\" | translate }}</mat-label>\n <input\n matInput\n name=\"anchorTimeInput\"\n type=\"time\"\n #anchorTimeControl=\"ngModel\"\n step=\"1\"\n [readonly]=\"isAnchorReadOnly\"\n [ngModel]=\"svc.getAnchorTimeString()\"\n (ngModelChange)=\"svc.setAnchorTimeFromForm($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.svc.resetFixedAnchorTime()\"\n [disabled]=\"!svc.isFixedAnchorTimeSetAndNotMidnight() || isAnchorReadOnly\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n }\n </div>\n</ng-template>\n", styles: [".interval-picker-component{min-width:840px;max-width:100%}.fixed-shifts-container{max-width:160px}.input-control-container{width:192px;max-width:100%}.input-control-container-short{width:140px;max-width:100%}.select-buttons-container button{white-space:nowrap;text-align:left;justify-content:flex-start}::ng-deep .interval-picker-component .mat-calendar-body-cell.elder-custom-anchor-date .mat-calendar-body-cell-content{border-radius:50%;background-color:#def;background-color:#b4d2ebbf}\n"], dependencies: [{ kind: "ngmodule", type: MatDatepickerModule }, { kind: "component", type: i2.MatCalendar, selector: "mat-calendar", inputs: ["headerComponent", "startAt", "startView", "selected", "minDate", "maxDate", "dateFilter", "dateClass", "comparisonStart", "comparisonEnd", "startDateAccessibleName", "endDateAccessibleName"], outputs: ["selectedChange", "yearSelected", "monthSelected", "viewChanged", "_userSelection", "_userDragDrop"], exportAs: ["matCalendar"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "directive", type: i3.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i5.DatePipe, name: "date" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i6.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i8.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i8.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "component", type: ElderLocalDateInputComponent, selector: "elder-local-date-input", inputs: ["zone", "autoDatePicker", "arrows", "today", "center", "datePickerTouchUi", "allowNull", "datePickerEnabled", "isoValue", "dateValue", "isoDateValue"], outputs: ["blurred", "valueUpdatedBlur", "isoValueChange", "dateValueChange", "isoDateValueChange"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i9.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: MatMenuModule }, { kind: "component", type: i10.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i10.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i10.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
211
|
+
}
|
|
212
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.2", ngImport: i0, type: ElderIntervalPickerComponent, decorators: [{
|
|
213
|
+
type: Component,
|
|
214
|
+
args: [{ selector: 'elder-interval-picker', standalone: true, imports: [
|
|
215
|
+
MatDateRangePicker,
|
|
216
|
+
MatDatepickerModule,
|
|
217
|
+
MatDateRangeInput,
|
|
218
|
+
MatFormFieldModule,
|
|
219
|
+
MatLabel,
|
|
220
|
+
MatHint,
|
|
221
|
+
MatDatepickerToggle,
|
|
222
|
+
MatInputModule,
|
|
223
|
+
CommonModule,
|
|
224
|
+
MatIconModule,
|
|
225
|
+
FormsModule,
|
|
226
|
+
MatButtonModule,
|
|
227
|
+
MatCalendar,
|
|
228
|
+
ElderLocalDateInputComponent,
|
|
229
|
+
MatTooltip,
|
|
230
|
+
TranslateModule,
|
|
231
|
+
MatMenuModule,
|
|
232
|
+
], providers: [
|
|
233
|
+
ElderIntervalPickerService,
|
|
234
|
+
{ provide: DateAdapter, useClass: CustomDateAdapter },
|
|
235
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"interval-picker-component p-md layout-col gap-xxl\">\n <div class=\"layout-row gap-xxl place-between-start\" style=\"gap: 5%\">\n <div class=\"layout-col gap-md pt-xs\">\n <ng-container *ngTemplateOutlet=\"intervalSelector\"></ng-container>\n </div>\n <div class=\"layout-col\">\n <!-- smart shift -->\n <div class=\"layout-row place-around-center\">\n <button mat-icon-button (click)=\"svc.smartShift(-1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ svc.viewIntervalMessage() }}</span>\n <button mat-icon-button (click)=\"svc.smartShift(1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <!-- calendars and input controls -->\n <ng-container *ngTemplateOutlet=\"calendarsTemplate\"></ng-container>\n </div>\n <div class=\"pt-xs\" style=\"width: 25%\">\n <div class=\"layout-col gap-lg\">\n <div class=\"layout-col gap-xs place-start-start\">\n @if (isHelpTextsVisible) {\n <h5 class=\"mat-h5\">{{ \"intervalPicker.selectSmartInterval\" | translate }}</h5>\n }\n <!-- select current -->\n <button mat-button (click)=\"svc.selectCurrentWeek()\">\n {{ \"intervalPicker.currentPeriod.week\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectCurrentMonth()\">\n {{ \"intervalPicker.currentPeriod.month\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectCurrentQuarter()\">\n {{ \"intervalPicker.currentPeriod.quarter\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectCurrentYear()\">\n {{ \"intervalPicker.currentPeriod.year\" | translate }}\n </button>\n </div>\n <div class=\"fixed-shifts-container\">\n <ng-container *ngTemplateOutlet=\"fixedShifts\"></ng-container>\n </div>\n </div>\n </div>\n </div>\n <div class=\"layout-row place-between-center gap-xxl pt-sm\">\n <div>\n @if (showAnchor) {\n <ng-container *ngTemplateOutlet=\"anchorInputField\"></ng-container>\n }\n </div>\n @if (isResultTextVisible) {\n <!-- result interval -->\n <div class=\"layout-col place-center-center\">\n <div class=\"date-interval mat-caption pt-xs\">\n @if (svc.startDateTimeAsJSDate()) {\n {{ svc.startDateTimeAsJSDate() | date : \"dd.MM. y, HH:mm:ss\" }}\n } @else {\n {{ \"intervalPicker.startDateNotSet\" | translate }}\n }\n <span> - </span>\n @if (svc.endDateTimeAsJSDate()) {\n {{ svc.endDateTimeAsJSDate() | date : \"dd.MM. y, HH:mm:ss\" }}\n } @else {\n {{ \"intervalPicker.endDateNotSet\" | translate }}\n }\n </div>\n <div>\n <span class=\"mat-caption\">{{ svc.deltaHumanReadable() || \" \" }}</span>\n </div>\n </div>\n }\n <div class=\"layout-row gap-lg\">\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"clearStartAndEndDateTimes()\"\n [disabled]=\"svc.isStartAndEndDatesEmpty()\"\n >\n {{ \"intervalPicker.clear\" | translate }}\n </button>\n\n @if(!this.isEmittingOnChange) {\n <button color=\"primary\" mat-raised-button (click)=\"svc.emitMainDates()\">\n {{ \"actions.ok\" | translate }}\n </button>\n }\n </div>\n </div>\n</div>\n\n<ng-template #intervalSelector>\n <div class=\"layout-col\">\n @if (isHelpTextsVisible) {\n <h5 class=\"mat-h5\">{{ \"intervalPicker.selectAbsoluteInterval\" | translate }}</h5>\n }\n <div class=\"layout-row select-buttons-container gap-sm\">\n <div class=\"layout-col gap-xs place-start-stretch\">\n <button mat-button (click)=\"svc.selectCurrentDay()\">\n @if(svc.isFixedAnchorDateSet()) {\n {{ \"intervalPicker.anchorDay\" | translate }}\n } @else {\n {{ \"intervalPicker.today\" | translate }}\n }\n </button>\n <button mat-button (click)=\"svc.selectYesterday()\">\n @if (svc.isFixedAnchorDateSet()) {\n {{ \"intervalPicker.last\" | translate }} 1 {{ \"intervalPicker.day\" | translate }}\n } @else {\n {{ \"intervalPicker.yesterday\" | translate }}\n }\n </button>\n <button mat-button (click)=\"svc.selectLastSevenDays()\">\n {{ \"intervalPicker.last\" | translate }} 7 {{ \"intervalPicker.days\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectLastThirtyDays()\">\n {{ \"intervalPicker.last\" | translate }} 30 {{ \"intervalPicker.days\" | translate }}\n </button>\n @if (intervalInputMode === 'date-time-range') {\n <div class=\"pt-sm\"></div>\n <button mat-button (click)=\"svc.selectLastFiveMinutes()\">\n {{ \"intervalPicker.last\" | translate }} 5 {{ \"intervalPicker.minutes\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectLastHour()\">\n {{ \"intervalPicker.last\" | translate }} {{ \"intervalPicker.hour\" | translate }}\n </button>\n <button mat-button (click)=\"svc.selectLast24Hours()\">\n {{ \"intervalPicker.last\" | translate }} 24 {{ \"intervalPicker.hours\" | translate }}\n </button>\n }\n </div>\n </div>\n </div>\n</ng-template>\n\n<ng-template #calendarsTemplate>\n <div class=\"layout-row place-around-center\" style=\"align-items: flex-start; min-height: 280px\">\n <mat-calendar\n #rangeCalendarStart\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"svc.dateRangeForCalendar()\"\n (selectedChange)=\"handleStartCalendarChange($event)\"\n [maxDate]=\"svc.endDateAsJSDate()\"\n [dateClass]=\"calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n <mat-calendar\n #rangeCalendarEnd\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"svc.dateRangeForCalendar()\"\n (selectedChange)=\"handleEndCalendarChange($event)\"\n [minDate]=\"svc.startDateAsJSDate()\"\n [dateClass]=\"calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n </div>\n <div class=\"layout-row place-around-center gap-xl pt-sm\">\n <div class=\"layout-row place-around-center\">\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.startDate\" | translate }}</mat-label>\n <input\n #startDateInput\n name=\"startDateInput\"\n matInput\n type=\"date\"\n [ngModel]=\"svc.startDateHtmlString()\"\n (ngModelChange)=\"svc.setStartDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearStartDateTime()\"\n [disabled]=\"!this.svc.startDate()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n <div class=\"layout-row place-around-center\">\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.endDate\" | translate }}</mat-label>\n <input\n #endDateInput\n name=\"endDateInput\"\n matInput\n type=\"date\"\n [ngModel]=\"svc.endDateHtmlString()\"\n (ngModelChange)=\"svc.setEndDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearEndDateTime()\"\n [disabled]=\"!this.svc.endDate()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n </div>\n <br />\n @if (intervalInputMode === 'date-time-range') {\n <div class=\"layout-row place-around-center gap-xl pt-sm\">\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.startTime\" | translate }}</mat-label>\n <input\n name=\"startTimeInput\"\n matInput\n type=\"time\"\n #startTimeControl=\"ngModel\"\n step=\"1\"\n [ngModel]=\"svc.getStartTimeString()\"\n (ngModelChange)=\"svc.setStartTimeFromString($event, startTimeControl)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.svc.clearStartTime()\"\n [disabled]=\"!this.svc.isStartTimeSetAndNotMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.endTime\" | translate }}</mat-label>\n <input\n name=\"endTimeInput\"\n matInput\n type=\"time\"\n #endTimeControl=\"ngModel\"\n step=\"1\"\n [ngModel]=\"svc.getEndTimeString()\"\n (ngModelChange)=\"svc.setEndTimeFromString($event, endTimeControl)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.svc.clearEndTime()\"\n [disabled]=\"!this.svc.isEndTimeSetAndNotMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n }\n</ng-template>\n\n<ng-template #fixedShifts>\n @if (isHelpTextsVisible) {\n <h5 class=\"mat-h5\">{{ \"intervalPicker.shiftInterval\" | translate }}</h5>\n }\n <div class=\"layout-col place-start-stretch\">\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftDay(-1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.day\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftDay(1)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftMonth(-1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.month\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftMonth(1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftYear(-1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.year\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftYear(1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n\n @if (intervalInputMode === 'date-time-range') {\n <!-- shift minute -->\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"svc.shiftTime(-1, 'minutes')\"\n [disabled]=\"svc.isRangeNotSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.minute\" | translate }}</span>\n <button\n mat-icon-button\n (click)=\"svc.shiftTime(1, 'minutes')\"\n [disabled]=\"svc.isRangeNotSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <!-- shift hour -->\n <div class=\"layout-row place-between-center\">\n <button mat-icon-button (click)=\"svc.shiftTime(-1, 'hours')\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ \"intervalPicker.hour\" | translate }}</span>\n <button mat-icon-button (click)=\"svc.shiftTime(1, 'hours')\" [disabled]=\"svc.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n }\n </div>\n</ng-template>\n\n<ng-template #anchorInputField>\n <div class=\"layout-row flex-none gap-md\">\n <mat-menu #anchorMenu=\"matMenu\">\n @if (svc.startDate()) {\n <button\n mat-menu-item\n (click)=\"svc.setFixedAnchorPointToEndDateTime()\"\n [disabled]=\"!svc.endDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n <span>{{ \"intervalPicker.endDate\" | translate }}</span>\n </button>\n } @if (svc.endDate()) {\n <button\n mat-menu-item\n (click)=\"svc.setFixedAnchorPointToStartDateTime()\"\n [disabled]=\"!svc.startDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n <span>{{ \"intervalPicker.startDate\" | translate }}</span>\n </button>\n } @if (svc.fixedAnchorDate()) {\n <button\n mat-menu-item\n (click)=\"this.svc.resetFixedAnchorPoint()\"\n [disabled]=\"!svc.fixedAnchorDate() || isAnchorReadOnly\"\n >\n <mat-icon>close</mat-icon>\n <span>{{ \"intervalPicker.clear\" | translate }}</span>\n </button>\n }\n </mat-menu>\n\n <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.anchorDate\" | translate }}</mat-label>\n <elder-local-date-input\n name=\"anchorDateInput\"\n (valueUpdated)=\"svc.setAnchorDateFromForm($event)\"\n [value]=\"svc.getAnchorPointAsLocaleDate()\"\n [readonly]=\"isAnchorReadOnly\"\n >\n </elder-local-date-input>\n @if (!isAnchorReadOnly) {\n <button\n mat-icon-button\n matSuffix\n [matMenuTriggerFor]=\"anchorMenu\"\n [disabled]=\"svc.isStartAndEndDatesEmpty() && !svc.fixedAnchorDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">more_horiz</mat-icon>\n </button>\n }\n </mat-form-field>\n @if (intervalInputMode === 'date-time-range') {\n <mat-form-field class=\"input-control-container-short\" [subscriptSizing]=\"'dynamic'\">\n <mat-label>{{ \"intervalPicker.anchorTime\" | translate }}</mat-label>\n <input\n matInput\n name=\"anchorTimeInput\"\n type=\"time\"\n #anchorTimeControl=\"ngModel\"\n step=\"1\"\n [readonly]=\"isAnchorReadOnly\"\n [ngModel]=\"svc.getAnchorTimeString()\"\n (ngModelChange)=\"svc.setAnchorTimeFromForm($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.svc.resetFixedAnchorTime()\"\n [disabled]=\"!svc.isFixedAnchorTimeSetAndNotMidnight() || isAnchorReadOnly\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n }\n </div>\n</ng-template>\n", styles: [".interval-picker-component{min-width:840px;max-width:100%}.fixed-shifts-container{max-width:160px}.input-control-container{width:192px;max-width:100%}.input-control-container-short{width:140px;max-width:100%}.select-buttons-container button{white-space:nowrap;text-align:left;justify-content:flex-start}::ng-deep .interval-picker-component .mat-calendar-body-cell.elder-custom-anchor-date .mat-calendar-body-cell-content{border-radius:50%;background-color:#def;background-color:#b4d2ebbf}\n"] }]
|
|
236
|
+
}], ctorParameters: () => [{ type: i1.ElderIntervalPickerService }], propDecorators: { calendarStart: [{
|
|
237
|
+
type: ViewChild,
|
|
238
|
+
args: ['rangeCalendarStart']
|
|
239
|
+
}], calendarEnd: [{
|
|
240
|
+
type: ViewChild,
|
|
241
|
+
args: ['rangeCalendarEnd']
|
|
242
|
+
}], calendarStartElRef: [{
|
|
243
|
+
type: ViewChild,
|
|
244
|
+
args: ['rangeCalendarStart', { read: ElementRef }]
|
|
245
|
+
}], calendarEndElRef: [{
|
|
246
|
+
type: ViewChild,
|
|
247
|
+
args: ['rangeCalendarEnd', { read: ElementRef }]
|
|
248
|
+
}], startDateInput: [{
|
|
249
|
+
type: ViewChild,
|
|
250
|
+
args: ['startDateInput']
|
|
251
|
+
}], endDateInput: [{
|
|
252
|
+
type: ViewChild,
|
|
253
|
+
args: ['endDateInput']
|
|
254
|
+
}], isEmittingOnChange: [{
|
|
255
|
+
type: Input
|
|
256
|
+
}], intervalInputMode: [{
|
|
257
|
+
type: Input
|
|
258
|
+
}], showAnchor: [{
|
|
259
|
+
type: Input
|
|
260
|
+
}], isAnchorReadOnly: [{
|
|
261
|
+
type: Input
|
|
262
|
+
}], isResultTextVisible: [{
|
|
263
|
+
type: Input
|
|
264
|
+
}], isHelpTextsVisible: [{
|
|
265
|
+
type: Input
|
|
266
|
+
}], dateChange: [{
|
|
267
|
+
type: Output
|
|
268
|
+
}] } });
|
|
269
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-interval-picker.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/time/elder-interval-picker/elder-interval-picker.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/time/elder-interval-picker/elder-interval-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  effect,\n  EffectRef,\n  ElementRef,\n  EventEmitter,\n  input,\n  Input,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport { MatButtonModule } from '@angular/material/button';\nimport { DateAdapter } from '@angular/material/core';\nimport {\n  MatCalendar,\n  MatCalendarCellClassFunction,\n  MatDatepickerModule,\n  MatDatepickerToggle,\n  MatDateRangeInput,\n  MatDateRangePicker\n} from '@angular/material/datepicker';\nimport {\n  MatFormFieldModule,\n  MatHint,\n  MatLabel,\n} from '@angular/material/form-field';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatTooltip } from '@angular/material/tooltip';\nimport { LoggerFactory } from '@elderbyte/ts-logger';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { Interval } from '../../../common/time/interval';\nimport { TemporalUtil } from '../../../common/time/temporal-util';\nimport { CustomDateAdapter } from '../../../components/time/date-adapters/custom-date-adapter';\nimport { IntervalInputMode } from '../../../components/time/elder-interval-input/elder-interval-input.component';\nimport { ElderLocalDateInputComponent } from '../../../components/time/elder-local-date-input/elder-local-date-input.component';\nimport { ElderIntervalPickerService } from './elder-interval-picker.service';\n\n@Component({\n  selector: 'elder-interval-picker',\n  standalone: true,\n  imports: [\n    MatDateRangePicker,\n    MatDatepickerModule,\n    MatDateRangeInput,\n    MatFormFieldModule,\n    MatLabel,\n    MatHint,\n    MatDatepickerToggle,\n    MatInputModule,\n    CommonModule,\n    MatIconModule,\n    FormsModule,\n    MatButtonModule,\n    MatCalendar,\n    ElderLocalDateInputComponent,\n    MatTooltip,\n    TranslateModule,\n    MatMenuModule,\n  ],\n  providers: [\n    ElderIntervalPickerService,\n    { provide: DateAdapter, useClass: CustomDateAdapter },\n  ],\n  templateUrl: './elder-interval-picker.component.html',\n  styleUrl: './elder-interval-picker.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ElderIntervalPickerComponent {\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n  public readonly calendarAnchorDateCssClassFn: MatCalendarCellClassFunction<Date> =\n    (cellDate: Date, view: string): string => {\n      return this.calendarAnchorDateCssClassFnBody(cellDate, view);\n    };\n\n  @ViewChild('rangeCalendarStart') calendarStart!: MatCalendar<Date>;\n  @ViewChild('rangeCalendarEnd') calendarEnd!: MatCalendar<Date>;\n\n  @ViewChild('rangeCalendarStart', { read: ElementRef }) calendarStartElRef!: ElementRef;\n  @ViewChild('rangeCalendarEnd', { read: ElementRef }) calendarEndElRef!: ElementRef;\n\n  @ViewChild('startDateInput') startDateInput!: ElementRef;\n  @ViewChild('endDateInput') endDateInput!: ElementRef;\n\n  @Input() isEmittingOnChange = true;\n  @Input() intervalInputMode: IntervalInputMode = 'date-range';\n  @Input() showAnchor = false;\n  @Input() isAnchorReadOnly = false;\n  @Input() isResultTextVisible = true;\n  @Input() isHelpTextsVisible = false;\n  @Output() dateChange = new EventEmitter<Interval>();\n  inputInterval = input<Interval>(null);\n  inputAnchorDateTime = input<Date>(null);\n\n  private calendarElFocusEventHandler: (event: FocusEvent) => void;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(public svc: ElderIntervalPickerService) {\n    this.svc.setupInputInterval(this.inputInterval);\n    this.svc.setupInputAnchorDateTime(this.inputAnchorDateTime);\n    this.svc.setupEventEmitter(this.dateChange);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  ngOnInit(): void {\n    if (this.inputInterval) {\n      this.svc.pullInputInterval();\n    }\n    if (this.inputAnchorDateTime()) {\n      this.svc.pullInputAnchorDateTime();\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.svc.isEmittingOnChange.set(this.isEmittingOnChange);\n    this.setupDateInputCalendarFocusPrevention();\n  }\n\n  ngOnDestroy(): void {\n    this.removeEventListeners();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  setupDateInputCalendarFocusPrevention(): void {\n    // Prevents the focus on the calendar when the input field is changed via the keyboard\n    this.calendarElFocusEventHandler = (event: FocusEvent): void => {\n      if (!event || !event.relatedTarget) {\n        return;\n\n      }\n      event.preventDefault();\n      const relTarget = event.relatedTarget as HTMLInputElement;\n\n      if (relTarget.name === 'startDateInput') {\n        this.startDateInput.nativeElement.focus();\n        return;\n      } \n      if (relTarget.name === 'endDateInput') {\n        this.endDateInput.nativeElement.focus();\n        return;\n      }\n    }\n\n    const nativeElementStart = this.calendarStartElRef.nativeElement as HTMLElement;\n    const nativeElementEnd = this.calendarEndElRef.nativeElement as HTMLElement;\n\n    nativeElementStart.addEventListener('focus', this.calendarElFocusEventHandler, true);\n    nativeElementEnd.addEventListener('focus', this.calendarElFocusEventHandler, true);\n  }\n\n  clearStartDateTime(): void {\n    this.svc.clearStartDateTime();\n\n    this.svc.resetCalendarMode();\n\n    if (this.isEmittingOnChange) {\n      this.svc.emitMainDates();\n    }\n  }\n\n  clearEndDateTime(): void {\n    this.svc.clearEndDateTime();\n\n    this.svc.resetCalendarMode();\n\n    if (this.isEmittingOnChange) {\n      this.svc.emitMainDates();\n    }\n  }\n\n  clearStartAndEndDateTimes(): void {\n    this.svc.clearStartDateTime();\n    this.svc.clearEndDateTime();\n\n    this.svc.resetCalendarMode();\n\n    if (this.isEmittingOnChange) {\n      this.svc.emitMainDates();\n    }\n  }\n\n  handleStartCalendarChange(changedDate: Date): void {\n    this.svc.calendarMode.set('days');\n    const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate);\n    this.svc.startDate.set(newPlainDate);\n  }\n\n  handleEndCalendarChange(changedDate: Date): void {\n    this.svc.calendarMode.set('days');\n    const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate);\n    this.svc.endDate.set(newPlainDate);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private removeEventListeners(): void {\n    if (!this.calendarStartElRef || !this.calendarEndElRef) {\n      return;\n    }\n\n    if (!this.calendarStartElRef.nativeElement || !this.calendarEndElRef.nativeElement) {\n      return;\n    }\n\n    const nativeElementStart = this.calendarStartElRef.nativeElement as HTMLElement;\n    const nativeElementEnd = this.calendarEndElRef.nativeElement as HTMLElement;\n\n    nativeElementStart.removeEventListener('focus', this.calendarElFocusEventHandler, true);\n    nativeElementEnd.removeEventListener('focus', this.calendarElFocusEventHandler, true);\n  }\n\n  private calendarAnchorDateCssClassFnBody(\n    cellDate: Date,\n    view: string\n  ): string {\n    const anchorDate = this.svc.fixedAnchorDate();\n\n    if (!anchorDate) {\n      return '';\n    }\n\n    const date = TemporalUtil.getJSDateFromPlainDateTime(anchorDate);\n\n    if (!date) {\n      return '';\n    }\n\n    if (\n      cellDate.getDate() === date.getDate() &&\n      cellDate.getMonth() === date.getMonth() &&\n      cellDate.getFullYear() === date.getFullYear()\n    ) {\n      return 'elder-custom-anchor-date';\n    }\n\n    return '';\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Effects                                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly _updateMatCalendarElAnchorCssClassEffect: EffectRef = effect(\n    (): void => {\n      const anchorDate = this.svc.anchorDate();\n\n      if (anchorDate) {\n        this.calendarStart.updateTodaysDate();\n        this.calendarEnd.updateTodaysDate();\n      }\n    }\n  );\n\n  private readonly _updateMatCalendarElSelectionEffect: EffectRef = effect(\n    (): void => {\n      const startDate = this.svc.startDate();\n      const endDate = this.svc.endDate();\n\n      if (!this.calendarStart || !this.calendarEnd) {\n        return;\n      }\n\n      if (!startDate && !endDate) {\n        return;\n      }\n\n      const calendar1 = this.calendarStart;\n      const calendar2 = this.calendarEnd;\n\n      setTimeout(() => {\n        // timeout is needed to run run calendar update last, otherwise there are issues\n        if (startDate) {\n          const startJsDate =\n            TemporalUtil.getJSDateFromPlainDateTime(startDate);\n          calendar1.activeDate = startJsDate;\n        }\n\n        if (endDate) {\n          const endJsDate = TemporalUtil.getJSDateFromPlainDateTime(endDate);\n          calendar2.activeDate = endJsDate;\n        }\n      }, 0);\n    }\n  );\n}\n","<div class=\"interval-picker-component p-md layout-col gap-xxl\">\n  <div class=\"layout-row gap-xxl place-between-start\" style=\"gap: 5%\">\n    <div class=\"layout-col gap-md pt-xs\">\n      <ng-container *ngTemplateOutlet=\"intervalSelector\"></ng-container>\n    </div>\n    <div class=\"layout-col\">\n      <!-- smart shift -->\n      <div class=\"layout-row place-around-center\">\n        <button mat-icon-button (click)=\"svc.smartShift(-1)\" [disabled]=\"svc.isRangeNotSet()\">\n          <mat-icon>keyboard_double_arrow_left</mat-icon>\n        </button>\n        <span>{{ svc.viewIntervalMessage() }}</span>\n        <button mat-icon-button (click)=\"svc.smartShift(1)\" [disabled]=\"svc.isRangeNotSet()\">\n          <mat-icon>keyboard_double_arrow_right</mat-icon>\n        </button>\n      </div>\n      <!-- calendars and input controls -->\n      <ng-container *ngTemplateOutlet=\"calendarsTemplate\"></ng-container>\n    </div>\n    <div class=\"pt-xs\" style=\"width: 25%\">\n      <div class=\"layout-col gap-lg\">\n        <div class=\"layout-col gap-xs place-start-start\">\n          @if (isHelpTextsVisible) {\n          <h5 class=\"mat-h5\">{{ \"intervalPicker.selectSmartInterval\" | translate }}</h5>\n          }\n          <!-- select current -->\n          <button mat-button (click)=\"svc.selectCurrentWeek()\">\n            {{ \"intervalPicker.currentPeriod.week\" | translate }}\n          </button>\n          <button mat-button (click)=\"svc.selectCurrentMonth()\">\n            {{ \"intervalPicker.currentPeriod.month\" | translate }}\n          </button>\n          <button mat-button (click)=\"svc.selectCurrentQuarter()\">\n            {{ \"intervalPicker.currentPeriod.quarter\" | translate }}\n          </button>\n          <button mat-button (click)=\"svc.selectCurrentYear()\">\n            {{ \"intervalPicker.currentPeriod.year\" | translate }}\n          </button>\n        </div>\n        <div class=\"fixed-shifts-container\">\n          <ng-container *ngTemplateOutlet=\"fixedShifts\"></ng-container>\n        </div>\n      </div>\n    </div>\n  </div>\n  <div class=\"layout-row place-between-center gap-xxl pt-sm\">\n    <div>\n      @if (showAnchor) {\n      <ng-container *ngTemplateOutlet=\"anchorInputField\"></ng-container>\n      }\n    </div>\n    @if (isResultTextVisible) {\n    <!-- result interval -->\n    <div class=\"layout-col place-center-center\">\n      <div class=\"date-interval mat-caption pt-xs\">\n        @if (svc.startDateTimeAsJSDate()) {\n        {{ svc.startDateTimeAsJSDate() | date : \"dd.MM. y, HH:mm:ss\" }}\n        } @else {\n        {{ \"intervalPicker.startDateNotSet\" | translate }}\n        }\n        <span> - </span>\n        @if (svc.endDateTimeAsJSDate()) {\n        {{ svc.endDateTimeAsJSDate() | date : \"dd.MM. y, HH:mm:ss\" }}\n        } @else {\n        {{ \"intervalPicker.endDateNotSet\" | translate }}\n        }\n      </div>\n      <div>\n        <span class=\"mat-caption\">{{ svc.deltaHumanReadable() || \"&nbsp;\" }}</span>\n      </div>\n    </div>\n    }\n    <div class=\"layout-row gap-lg\">\n      <button\n        mat-raised-button\n        color=\"primary\"\n        (click)=\"clearStartAndEndDateTimes()\"\n        [disabled]=\"svc.isStartAndEndDatesEmpty()\"\n      >\n        {{ \"intervalPicker.clear\" | translate }}\n      </button>\n\n      @if(!this.isEmittingOnChange) {\n      <button color=\"primary\" mat-raised-button (click)=\"svc.emitMainDates()\">\n        {{ \"actions.ok\" | translate }}\n      </button>\n      }\n    </div>\n  </div>\n</div>\n\n<ng-template #intervalSelector>\n  <div class=\"layout-col\">\n    @if (isHelpTextsVisible) {\n    <h5 class=\"mat-h5\">{{ \"intervalPicker.selectAbsoluteInterval\" | translate }}</h5>\n    }\n    <div class=\"layout-row select-buttons-container gap-sm\">\n      <div class=\"layout-col gap-xs place-start-stretch\">\n        <button mat-button (click)=\"svc.selectCurrentDay()\">\n          @if(svc.isFixedAnchorDateSet()) {\n          {{ \"intervalPicker.anchorDay\" | translate }}\n          } @else {\n          {{ \"intervalPicker.today\" | translate }}\n          }\n        </button>\n        <button mat-button (click)=\"svc.selectYesterday()\">\n          @if (svc.isFixedAnchorDateSet()) {\n          {{ \"intervalPicker.last\" | translate }} 1 {{ \"intervalPicker.day\" | translate }}\n          } @else {\n          {{ \"intervalPicker.yesterday\" | translate }}\n          }\n        </button>\n        <button mat-button (click)=\"svc.selectLastSevenDays()\">\n          {{ \"intervalPicker.last\" | translate }} 7 {{ \"intervalPicker.days\" | translate }}\n        </button>\n        <button mat-button (click)=\"svc.selectLastThirtyDays()\">\n          {{ \"intervalPicker.last\" | translate }} 30 {{ \"intervalPicker.days\" | translate }}\n        </button>\n        @if (intervalInputMode === 'date-time-range') {\n        <div class=\"pt-sm\"></div>\n        <button mat-button (click)=\"svc.selectLastFiveMinutes()\">\n          {{ \"intervalPicker.last\" | translate }} 5 {{ \"intervalPicker.minutes\" | translate }}\n        </button>\n        <button mat-button (click)=\"svc.selectLastHour()\">\n          {{ \"intervalPicker.last\" | translate }} {{ \"intervalPicker.hour\" | translate }}\n        </button>\n        <button mat-button (click)=\"svc.selectLast24Hours()\">\n          {{ \"intervalPicker.last\" | translate }} 24 {{ \"intervalPicker.hours\" | translate }}\n        </button>\n        }\n      </div>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #calendarsTemplate>\n  <div class=\"layout-row place-around-center\" style=\"align-items: flex-start; min-height: 280px\">\n    <mat-calendar\n      #rangeCalendarStart\n      style=\"width: 210px; max-width: 100%\"\n      [selected]=\"svc.dateRangeForCalendar()\"\n      (selectedChange)=\"handleStartCalendarChange($event)\"\n      [maxDate]=\"svc.endDateAsJSDate()\"\n      [dateClass]=\"calendarAnchorDateCssClassFn\"\n    >\n    </mat-calendar>\n    <mat-calendar\n      #rangeCalendarEnd\n      style=\"width: 210px; max-width: 100%\"\n      [selected]=\"svc.dateRangeForCalendar()\"\n      (selectedChange)=\"handleEndCalendarChange($event)\"\n      [minDate]=\"svc.startDateAsJSDate()\"\n      [dateClass]=\"calendarAnchorDateCssClassFn\"\n    >\n    </mat-calendar>\n  </div>\n  <div class=\"layout-row place-around-center gap-xl pt-sm\">\n    <div class=\"layout-row place-around-center\">\n      <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n        <mat-label>{{ \"intervalPicker.startDate\" | translate }}</mat-label>\n        <input\n          #startDateInput\n          name=\"startDateInput\"\n          matInput\n          type=\"date\"\n          [ngModel]=\"svc.startDateHtmlString()\"\n          (ngModelChange)=\"svc.setStartDateFromHtmlString($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.clearStartDateTime()\"\n          [disabled]=\"!this.svc.startDate()\"\n        >\n          <mat-icon>close</mat-icon>\n        </button>\n      </mat-form-field>\n    </div>\n    <div class=\"layout-row place-around-center\">\n      <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n        <mat-label>{{ \"intervalPicker.endDate\" | translate }}</mat-label>\n        <input\n          #endDateInput\n          name=\"endDateInput\"\n          matInput\n          type=\"date\"\n          [ngModel]=\"svc.endDateHtmlString()\"\n          (ngModelChange)=\"svc.setEndDateFromHtmlString($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.clearEndDateTime()\"\n          [disabled]=\"!this.svc.endDate()\"\n        >\n          <mat-icon>close</mat-icon>\n        </button>\n      </mat-form-field>\n    </div>\n  </div>\n  <br />\n  @if (intervalInputMode === 'date-time-range') {\n  <div class=\"layout-row place-around-center gap-xl pt-sm\">\n    <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n      <mat-label>{{ \"intervalPicker.startTime\" | translate }}</mat-label>\n      <input\n        name=\"startTimeInput\"\n        matInput\n        type=\"time\"\n        #startTimeControl=\"ngModel\"\n        step=\"1\"\n        [ngModel]=\"svc.getStartTimeString()\"\n        (ngModelChange)=\"svc.setStartTimeFromString($event, startTimeControl)\"\n      />\n      <button\n        mat-icon-button\n        matSuffix\n        (click)=\"this.svc.clearStartTime()\"\n        [disabled]=\"!this.svc.isStartTimeSetAndNotMidnight()\"\n      >\n        <mat-icon>close</mat-icon>\n      </button>\n    </mat-form-field>\n    <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n      <mat-label>{{ \"intervalPicker.endTime\" | translate }}</mat-label>\n      <input\n        name=\"endTimeInput\"\n        matInput\n        type=\"time\"\n        #endTimeControl=\"ngModel\"\n        step=\"1\"\n        [ngModel]=\"svc.getEndTimeString()\"\n        (ngModelChange)=\"svc.setEndTimeFromString($event, endTimeControl)\"\n      />\n      <button\n        mat-icon-button\n        matSuffix\n        (click)=\"this.svc.clearEndTime()\"\n        [disabled]=\"!this.svc.isEndTimeSetAndNotMidnight()\"\n      >\n        <mat-icon>close</mat-icon>\n      </button>\n    </mat-form-field>\n  </div>\n  }\n</ng-template>\n\n<ng-template #fixedShifts>\n  @if (isHelpTextsVisible) {\n  <h5 class=\"mat-h5\">{{ \"intervalPicker.shiftInterval\" | translate }}</h5>\n  }\n  <div class=\"layout-col place-start-stretch\">\n    <div class=\"layout-row place-between-center\">\n      <button mat-icon-button (click)=\"svc.shiftDay(-1)\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ \"intervalPicker.day\" | translate }}</span>\n      <button mat-icon-button (click)=\"svc.shiftDay(1)\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_right</mat-icon>\n      </button>\n    </div>\n    <div class=\"layout-row place-between-center\">\n      <button mat-icon-button (click)=\"svc.shiftMonth(-1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ \"intervalPicker.month\" | translate }}</span>\n      <button mat-icon-button (click)=\"svc.shiftMonth(1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_right</mat-icon>\n      </button>\n    </div>\n    <div class=\"layout-row place-between-center\">\n      <button mat-icon-button (click)=\"svc.shiftYear(-1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ \"intervalPicker.year\" | translate }}</span>\n      <button mat-icon-button (click)=\"svc.shiftYear(1, true)\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_right</mat-icon>\n      </button>\n    </div>\n\n    @if (intervalInputMode === 'date-time-range') {\n    <!-- shift minute -->\n    <div class=\"layout-row place-between-center\">\n      <button\n        mat-icon-button\n        (click)=\"svc.shiftTime(-1, 'minutes')\"\n        [disabled]=\"svc.isRangeNotSet()\"\n      >\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ \"intervalPicker.minute\" | translate }}</span>\n      <button\n        mat-icon-button\n        (click)=\"svc.shiftTime(1, 'minutes')\"\n        [disabled]=\"svc.isRangeNotSet()\"\n      >\n        <mat-icon>keyboard_double_arrow_right</mat-icon>\n      </button>\n    </div>\n    <!-- shift hour -->\n    <div class=\"layout-row place-between-center\">\n      <button mat-icon-button (click)=\"svc.shiftTime(-1, 'hours')\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ \"intervalPicker.hour\" | translate }}</span>\n      <button mat-icon-button (click)=\"svc.shiftTime(1, 'hours')\" [disabled]=\"svc.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_right</mat-icon>\n      </button>\n    </div>\n    }\n  </div>\n</ng-template>\n\n<ng-template #anchorInputField>\n  <div class=\"layout-row flex-none gap-md\">\n    <mat-menu #anchorMenu=\"matMenu\">\n      @if (svc.startDate()) {\n      <button\n        mat-menu-item\n        (click)=\"svc.setFixedAnchorPointToEndDateTime()\"\n        [disabled]=\"!svc.endDate()\"\n      >\n        <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n        <span>{{ \"intervalPicker.endDate\" | translate }}</span>\n      </button>\n      } @if (svc.endDate()) {\n      <button\n        mat-menu-item\n        (click)=\"svc.setFixedAnchorPointToStartDateTime()\"\n        [disabled]=\"!svc.startDate()\"\n      >\n        <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n        <span>{{ \"intervalPicker.startDate\" | translate }}</span>\n      </button>\n      } @if (svc.fixedAnchorDate()) {\n      <button\n        mat-menu-item\n        (click)=\"this.svc.resetFixedAnchorPoint()\"\n        [disabled]=\"!svc.fixedAnchorDate() || isAnchorReadOnly\"\n      >\n        <mat-icon>close</mat-icon>\n        <span>{{ \"intervalPicker.clear\" | translate }}</span>\n      </button>\n      }\n    </mat-menu>\n\n    <mat-form-field class=\"input-control-container\" [subscriptSizing]=\"'dynamic'\">\n      <mat-label>{{ \"intervalPicker.anchorDate\" | translate }}</mat-label>\n      <elder-local-date-input\n        name=\"anchorDateInput\"\n        (valueUpdated)=\"svc.setAnchorDateFromForm($event)\"\n        [value]=\"svc.getAnchorPointAsLocaleDate()\"\n        [readonly]=\"isAnchorReadOnly\"\n      >\n      </elder-local-date-input>\n      @if (!isAnchorReadOnly) {\n      <button\n        mat-icon-button\n        matSuffix\n        [matMenuTriggerFor]=\"anchorMenu\"\n        [disabled]=\"svc.isStartAndEndDatesEmpty() && !svc.fixedAnchorDate()\"\n      >\n        <mat-icon class=\"material-symbols-outlined\">more_horiz</mat-icon>\n      </button>\n      }\n    </mat-form-field>\n    @if (intervalInputMode === 'date-time-range') {\n    <mat-form-field class=\"input-control-container-short\" [subscriptSizing]=\"'dynamic'\">\n      <mat-label>{{ \"intervalPicker.anchorTime\" | translate }}</mat-label>\n      <input\n        matInput\n        name=\"anchorTimeInput\"\n        type=\"time\"\n        #anchorTimeControl=\"ngModel\"\n        step=\"1\"\n        [readonly]=\"isAnchorReadOnly\"\n        [ngModel]=\"svc.getAnchorTimeString()\"\n        (ngModelChange)=\"svc.setAnchorTimeFromForm($event)\"\n      />\n      <button\n        mat-icon-button\n        matSuffix\n        (click)=\"this.svc.resetFixedAnchorTime()\"\n        [disabled]=\"!svc.isFixedAnchorTimeSetAndNotMidnight() || isAnchorReadOnly\"\n      >\n        <mat-icon>close</mat-icon>\n      </button>\n    </mat-form-field>\n    }\n  </div>\n</ng-template>\n"]}
|