@elderbyte/ngx-starter 18.12.4 → 18.12.6
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/data/data-context/indexed-entities.mjs +13 -2
- package/esm2022/lib/common/selection/selection-model.mjs +4 -3
- package/esm2022/lib/components/data-view/common/selection/data-context-selection.directive.mjs +4 -2
- package/esm2022/lib/components/time/elder-interval-picker/controller/interval-anchor-controller.mjs +21 -0
- package/esm2022/lib/components/time/elder-interval-picker/controller/interval-calendar-controller.mjs +24 -0
- package/esm2022/lib/components/time/elder-interval-picker/controller/interval-controller.mjs +43 -0
- package/esm2022/lib/components/time/elder-interval-picker/controller/interval-form-controller.mjs +69 -0
- package/esm2022/lib/components/time/elder-interval-picker/controller/interval-select-controller.mjs +117 -0
- package/esm2022/lib/components/time/elder-interval-picker/controller/interval-shift-controller.mjs +44 -0
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker.component.mjs +87 -130
- package/esm2022/lib/components/time/elder-interval-picker/interval-picker-binding/elder-interval-picker-binding.directive.mjs +41 -0
- package/esm2022/lib/components/time/elder-interval-picker/interval-picker-toggle/elder-interval-picker-toggle.component.mjs +144 -0
- package/esm2022/lib/components/time/elder-interval-picker/manager/interval-picker-state-manager.mjs +93 -0
- package/esm2022/lib/components/time/elder-interval-picker/model/anchor-state.mjs +54 -0
- package/esm2022/lib/components/time/elder-interval-picker/model/interval-state.mjs +37 -0
- package/esm2022/lib/components/time/elder-interval-picker/model/smart-shift-state.mjs +10 -0
- package/esm2022/lib/components/time/elder-interval-picker/presenter/interval-picker-presenter.mjs +96 -0
- package/esm2022/lib/components/time/elder-interval-picker/util/interval-picker-util.mjs +194 -0
- package/esm2022/lib/components/time/elder-time.module.mjs +5 -5
- package/fesm2022/elderbyte-ngx-starter.mjs +789 -892
- package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/components/time/elder-interval-picker/controller/interval-anchor-controller.d.ts +9 -0
- package/lib/components/time/elder-interval-picker/controller/interval-calendar-controller.d.ts +7 -0
- package/lib/components/time/elder-interval-picker/controller/interval-controller.d.ts +35 -0
- package/lib/components/time/elder-interval-picker/controller/interval-form-controller.d.ts +12 -0
- package/lib/components/time/elder-interval-picker/controller/interval-select-controller.d.ts +22 -0
- package/lib/components/time/elder-interval-picker/controller/interval-shift-controller.d.ts +13 -0
- package/lib/components/time/elder-interval-picker/elder-interval-picker.component.d.ts +26 -34
- package/lib/components/time/elder-interval-picker/{elder-interval-picker-binding.directive.d.ts → interval-picker-binding/elder-interval-picker-binding.directive.d.ts} +3 -4
- package/lib/components/time/elder-interval-picker/interval-picker-toggle/elder-interval-picker-toggle.component.d.ts +37 -0
- package/lib/components/time/elder-interval-picker/manager/interval-picker-state-manager.d.ts +46 -0
- package/lib/components/time/elder-interval-picker/model/anchor-state.d.ts +20 -0
- package/lib/components/time/elder-interval-picker/model/interval-state.d.ts +16 -0
- package/lib/components/time/elder-interval-picker/model/smart-shift-state.d.ts +8 -0
- package/lib/components/time/elder-interval-picker/presenter/interval-picker-presenter.d.ts +30 -0
- package/lib/components/time/elder-interval-picker/util/interval-picker-util.d.ts +20 -0
- package/lib/components/time/elder-time.module.d.ts +4 -4
- package/package.json +1 -1
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker-binding.directive.mjs +0 -40
- package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker-toggle/elder-interval-picker-toggle.component.mjs +0 -202
- package/esm2022/lib/components/time/elder-interval-picker/model/interval-picker-model-editor.mjs +0 -284
- package/esm2022/lib/components/time/elder-interval-picker/model/interval-picker-model.mjs +0 -445
- package/esm2022/lib/components/time/elder-interval-picker/model/plain-date-interval-message-renderer.mjs +0 -37
- package/esm2022/lib/components/time/elder-interval-picker/model/plain-temporal-duration-message-renderer.mjs +0 -25
- package/lib/components/time/elder-interval-picker/elder-interval-picker-toggle/elder-interval-picker-toggle.component.d.ts +0 -55
- package/lib/components/time/elder-interval-picker/model/interval-picker-model-editor.d.ts +0 -77
- package/lib/components/time/elder-interval-picker/model/interval-picker-model.d.ts +0 -113
- package/lib/components/time/elder-interval-picker/model/plain-date-interval-message-renderer.d.ts +0 -7
- package/lib/components/time/elder-interval-picker/model/plain-temporal-duration-message-renderer.d.ts +0 -4
package/esm2022/lib/components/time/elder-interval-picker/elder-interval-picker.component.mjs
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
|
-
import {
|
|
2
|
+
import { ChangeDetectionStrategy, Component, ElementRef, Input, input, Output, ViewChild, } from '@angular/core';
|
|
3
|
+
import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
|
|
3
4
|
import { FormsModule } from '@angular/forms';
|
|
4
5
|
import { MatButtonModule } from '@angular/material/button';
|
|
5
6
|
import { DateAdapter } from '@angular/material/core';
|
|
@@ -10,61 +11,80 @@ import { MatInputModule } from '@angular/material/input';
|
|
|
10
11
|
import { MatMenuModule } from '@angular/material/menu';
|
|
11
12
|
import { MatTooltip } from '@angular/material/tooltip';
|
|
12
13
|
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
14
|
+
import { Temporal } from '@js-temporal/polyfill';
|
|
13
15
|
import { TranslateModule } from '@ngx-translate/core';
|
|
14
|
-
import {
|
|
16
|
+
import { map } from 'rxjs';
|
|
17
|
+
import { coerceInterval, coerceIntervalIsoStr } from '../../../common/public_api';
|
|
15
18
|
import { CustomDateAdapter } from '../date-adapters/custom-date-adapter';
|
|
16
|
-
import {
|
|
17
|
-
import
|
|
19
|
+
import { ElderIntervalInputComponent, } from '../elder-interval-input/elder-interval-input.component';
|
|
20
|
+
import IntervalController from './controller/interval-controller';
|
|
21
|
+
import IntervalPickerStateManager from './manager/interval-picker-state-manager';
|
|
22
|
+
import IntervalPickerPresenter from './presenter/interval-picker-presenter';
|
|
23
|
+
import IntervalPickerUtil from './util/interval-picker-util';
|
|
18
24
|
import * as i0 from "@angular/core";
|
|
19
|
-
import * as i1 from "
|
|
20
|
-
import * as i2 from "@angular/material/
|
|
21
|
-
import * as i3 from "@angular/material/
|
|
22
|
-
import * as i4 from "@angular/
|
|
23
|
-
import * as i5 from "@angular/
|
|
24
|
-
import * as i6 from "@angular/
|
|
25
|
-
import * as i7 from "@angular/
|
|
26
|
-
import * as i8 from "@
|
|
27
|
-
import * as i9 from "@
|
|
28
|
-
import * as i10 from "@angular/material/menu";
|
|
25
|
+
import * as i1 from "@angular/material/datepicker";
|
|
26
|
+
import * as i2 from "@angular/material/form-field";
|
|
27
|
+
import * as i3 from "@angular/material/input";
|
|
28
|
+
import * as i4 from "@angular/common";
|
|
29
|
+
import * as i5 from "@angular/material/icon";
|
|
30
|
+
import * as i6 from "@angular/forms";
|
|
31
|
+
import * as i7 from "@angular/material/button";
|
|
32
|
+
import * as i8 from "@ngx-translate/core";
|
|
33
|
+
import * as i9 from "@angular/material/menu";
|
|
29
34
|
export class ElderIntervalPickerComponent {
|
|
30
|
-
set interval(value) {
|
|
31
|
-
this.model.setInterval(value);
|
|
32
|
-
}
|
|
33
|
-
set anchorDateTime(value) {
|
|
34
|
-
this.model.setAnchorDateTime(value);
|
|
35
|
-
}
|
|
36
35
|
/***************************************************************************
|
|
37
36
|
* *
|
|
38
37
|
* Constructor *
|
|
39
38
|
* *
|
|
40
39
|
**************************************************************************/
|
|
41
|
-
constructor(
|
|
42
|
-
this.model = model;
|
|
40
|
+
constructor() {
|
|
43
41
|
/***************************************************************************
|
|
44
42
|
* *
|
|
45
43
|
* Fields *
|
|
46
44
|
* *
|
|
47
45
|
**************************************************************************/
|
|
48
46
|
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
49
|
-
this.
|
|
50
|
-
this.
|
|
51
|
-
this.
|
|
52
|
-
this.
|
|
53
|
-
this.
|
|
54
|
-
this.
|
|
55
|
-
this.
|
|
56
|
-
this.
|
|
57
|
-
this.
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
this.
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
47
|
+
this.autoEmitMode = true;
|
|
48
|
+
this.emitOnChange = true;
|
|
49
|
+
this.emitType = 'interval';
|
|
50
|
+
this.intervalInputMode = input('date-time-range');
|
|
51
|
+
this.showAnchor = input(false);
|
|
52
|
+
this.anchorReadOnly = input(false);
|
|
53
|
+
this.externalAnchorDateTime = input(Temporal.Now.plainDateTimeISO());
|
|
54
|
+
this.externalInterval = input(null);
|
|
55
|
+
this.manager = new IntervalPickerStateManager();
|
|
56
|
+
this.controller = new IntervalController(this.manager);
|
|
57
|
+
this.presenter = new IntervalPickerPresenter(this.manager.interval$, this.manager.anchor$, this.manager.smartShift$);
|
|
58
|
+
this.manager.intervalChange.pipe(takeUntilDestroyed()).subscribe({
|
|
59
|
+
next: (state) => this.tryUpdateMatCalendarActiveDate(state),
|
|
60
|
+
});
|
|
61
|
+
this.intervalChange = this.manager.intervalChange.pipe(map((srcInterval) => {
|
|
62
|
+
if (!srcInterval) {
|
|
63
|
+
return null;
|
|
64
|
+
}
|
|
65
|
+
const elderInterval = IntervalPickerUtil.createElderInterval(srcInterval);
|
|
66
|
+
if (this.emitType === 'iso') {
|
|
67
|
+
console.log('emitting type iso', coerceIntervalIsoStr(elderInterval));
|
|
68
|
+
return coerceIntervalIsoStr(elderInterval);
|
|
64
69
|
}
|
|
70
|
+
console.log('emitting type elder interval', elderInterval);
|
|
71
|
+
return elderInterval;
|
|
72
|
+
}));
|
|
73
|
+
this.manager.anchor$.pipe(takeUntilDestroyed()).subscribe((anchor) => {
|
|
74
|
+
IntervalPickerUtil.updateMatCalendarTodayDate(this.calendarStart, this.calendarEnd);
|
|
75
|
+
});
|
|
76
|
+
const externalAnchorObservable = toObservable(this.externalAnchorDateTime);
|
|
77
|
+
externalAnchorObservable.pipe(takeUntilDestroyed()).subscribe({
|
|
78
|
+
next: (date) => {
|
|
79
|
+
this.manager.setExternalAnchor(date);
|
|
80
|
+
},
|
|
65
81
|
});
|
|
66
|
-
|
|
67
|
-
|
|
82
|
+
const externalIntervalObservable = toObservable(this.externalInterval);
|
|
83
|
+
externalIntervalObservable.pipe(takeUntilDestroyed()).subscribe({
|
|
84
|
+
next: (interval) => {
|
|
85
|
+
const result = coerceInterval(interval);
|
|
86
|
+
this.manager.setInterval(IntervalPickerUtil.transformToIntervalState(result || null));
|
|
87
|
+
},
|
|
68
88
|
});
|
|
69
89
|
}
|
|
70
90
|
/***************************************************************************
|
|
@@ -72,19 +92,21 @@ export class ElderIntervalPickerComponent {
|
|
|
72
92
|
* Life Cycle *
|
|
73
93
|
* *
|
|
74
94
|
**************************************************************************/
|
|
95
|
+
ngOnInit() { }
|
|
96
|
+
ngOnDestroy() {
|
|
97
|
+
this.removeEventListeners();
|
|
98
|
+
}
|
|
75
99
|
ngAfterViewInit() {
|
|
76
|
-
this.model.isEmittingOnChange.set(this.emitOnChange());
|
|
77
100
|
this.setupDateInputCalendarFocusPrevention();
|
|
78
101
|
setTimeout(() => {
|
|
79
|
-
this
|
|
102
|
+
// this prevents ngmodel issue
|
|
103
|
+
this.manager.isInitialized = true;
|
|
104
|
+
console.log('externalInterval', this.externalInterval());
|
|
80
105
|
}, 0);
|
|
81
106
|
}
|
|
82
|
-
ngOnDestroy() {
|
|
83
|
-
this.removeEventListeners();
|
|
84
|
-
}
|
|
85
107
|
/***************************************************************************
|
|
86
108
|
* *
|
|
87
|
-
*
|
|
109
|
+
* Private methods *
|
|
88
110
|
* *
|
|
89
111
|
**************************************************************************/
|
|
90
112
|
setupDateInputCalendarFocusPrevention() {
|
|
@@ -109,43 +131,6 @@ export class ElderIntervalPickerComponent {
|
|
|
109
131
|
nativeElementStart.addEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
110
132
|
nativeElementEnd.addEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
111
133
|
}
|
|
112
|
-
clearStartDateTime() {
|
|
113
|
-
this.model.clearStartDateTime();
|
|
114
|
-
this.model.resetCalendarMode();
|
|
115
|
-
if (this.emitOnChange()) {
|
|
116
|
-
this.model.emitMainDates();
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
clearEndDateTime() {
|
|
120
|
-
this.model.clearEndDateTime();
|
|
121
|
-
this.model.resetCalendarMode();
|
|
122
|
-
if (this.emitOnChange()) {
|
|
123
|
-
this.model.emitMainDates();
|
|
124
|
-
}
|
|
125
|
-
}
|
|
126
|
-
clearStartAndEndDateTimes() {
|
|
127
|
-
this.model.clearStartDateTime();
|
|
128
|
-
this.model.clearEndDateTime();
|
|
129
|
-
this.model.resetCalendarMode();
|
|
130
|
-
if (this.emitOnChange()) {
|
|
131
|
-
this.model.emitMainDates();
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
handleStartCalendarChange(changedDate) {
|
|
135
|
-
this.model.calendarMode.set('days');
|
|
136
|
-
const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate);
|
|
137
|
-
this.model.setStartDate(newPlainDate);
|
|
138
|
-
}
|
|
139
|
-
handleEndCalendarChange(changedDate) {
|
|
140
|
-
this.model.calendarMode.set('days');
|
|
141
|
-
const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate);
|
|
142
|
-
this.model.setEndDate(newPlainDate);
|
|
143
|
-
}
|
|
144
|
-
/***************************************************************************
|
|
145
|
-
* *
|
|
146
|
-
* Private methods *
|
|
147
|
-
* *
|
|
148
|
-
**************************************************************************/
|
|
149
134
|
removeEventListeners() {
|
|
150
135
|
if (!this.calendarStartElRef || !this.calendarEndElRef) {
|
|
151
136
|
return;
|
|
@@ -158,47 +143,16 @@ export class ElderIntervalPickerComponent {
|
|
|
158
143
|
nativeElementStart.removeEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
159
144
|
nativeElementEnd.removeEventListener('focus', this.calendarElFocusEventHandler, true);
|
|
160
145
|
}
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
return '';
|
|
146
|
+
tryUpdateMatCalendarActiveDate(interval) {
|
|
147
|
+
try {
|
|
148
|
+
IntervalPickerUtil.updateMatCalendarActiveDate(this.calendarStart, this.calendarEnd, interval.startDate, interval.endDate);
|
|
165
149
|
}
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
return '';
|
|
150
|
+
catch (e) {
|
|
151
|
+
console.debug('Failed to update calendar active date', e);
|
|
169
152
|
}
|
|
170
|
-
if (cellDate.getDate() === date.getDate() &&
|
|
171
|
-
cellDate.getMonth() === date.getMonth() &&
|
|
172
|
-
cellDate.getFullYear() === date.getFullYear()) {
|
|
173
|
-
return 'elder-custom-anchor-date';
|
|
174
|
-
}
|
|
175
|
-
return '';
|
|
176
|
-
}
|
|
177
|
-
updateMatCalendarActiveDate(startDate, endDate) {
|
|
178
|
-
if (!this.calendarStart || !this.calendarEnd) {
|
|
179
|
-
return;
|
|
180
|
-
}
|
|
181
|
-
if (!startDate && !endDate) {
|
|
182
|
-
return;
|
|
183
|
-
}
|
|
184
|
-
const calendarStart = this.calendarStart;
|
|
185
|
-
const calendarEnd = this.calendarEnd;
|
|
186
|
-
setTimeout(() => {
|
|
187
|
-
// timeout is needed to run calendar update last, otherwise there are issues
|
|
188
|
-
if (startDate) {
|
|
189
|
-
calendarStart.activeDate = TemporalUtil.getJSDateFromPlainDateTime(startDate);
|
|
190
|
-
}
|
|
191
|
-
if (endDate) {
|
|
192
|
-
calendarEnd.activeDate = TemporalUtil.getJSDateFromPlainDateTime(endDate);
|
|
193
|
-
}
|
|
194
|
-
}, 0);
|
|
195
|
-
}
|
|
196
|
-
updateMatCalendarTodayDate() {
|
|
197
|
-
this.calendarStart.updateTodaysDate();
|
|
198
|
-
this.calendarEnd.updateTodaysDate();
|
|
199
153
|
}
|
|
200
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: ElderIntervalPickerComponent, deps: [
|
|
201
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.5", type: ElderIntervalPickerComponent, isStandalone: true, selector: "elder-interval-picker", inputs: { emitOnChange: { classPropertyName: "emitOnChange", publicName: "emitOnChange", isSignal: true, isRequired: false, transformFunction: null }, intervalInputMode: { classPropertyName: "intervalInputMode", publicName: "intervalInputMode", isSignal: true, isRequired: false, transformFunction: null }, showAnchor: { classPropertyName: "showAnchor", publicName: "showAnchor", isSignal: true, isRequired: false, transformFunction: null }, anchorReadOnly: { classPropertyName: "anchorReadOnly", publicName: "anchorReadOnly", isSignal: true, isRequired: false, transformFunction: null }, showResultText: { classPropertyName: "showResultText", publicName: "showResultText", isSignal: true, isRequired: false, transformFunction: null }, showHelpText: { classPropertyName: "showHelpText", publicName: "showHelpText", isSignal: true, isRequired: false, transformFunction: null }, lastExcludesToday: { classPropertyName: "lastExcludesToday", publicName: "lastExcludesToday", isSignal: true, isRequired: false, transformFunction: null }, interval: { classPropertyName: "interval", publicName: "interval", isSignal: false, isRequired: false, transformFunction: null }, anchorDateTime: { classPropertyName: "anchorDateTime", publicName: "anchorDateTime", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { dateChange: "dateChange" }, providers: [IntervalPickerModel, { provide: DateAdapter, useClass: CustomDateAdapter }], 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)=\"model.editor.smartShift(-1)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ model.viewIntervalMessage() }}</span>\n <button mat-icon-button (click)=\"model.editor.smartShift(1)\" [disabled]=\"model.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 (showHelpText()) {\n <h5 class=\"mat-h5\">{{ 'intervalPicker.selectSmartInterval' | translate }}</h5>\n }\n <!-- select current -->\n <button mat-button (click)=\"model.editor.selectCurrentWeek()\">\n {{ 'intervalPicker.currentPeriod.week' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectCurrentMonth()\">\n {{ 'intervalPicker.currentPeriod.month' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectCurrentQuarter()\">\n {{ 'intervalPicker.currentPeriod.quarter' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.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 (showResultText()) {\n <!-- result interval -->\n <div class=\"layout-col place-center-center\">\n <div class=\"date-interval mat-caption pt-xs\">\n @if (model.startDateTimeAsJSDate()) {\n {{ model.startDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.startDateNotSet' | translate }}\n }\n <span> - </span>\n @if (model.endDateTimeAsJSDate()) {\n {{ model.endDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.endDateNotSet' | translate }}\n }\n </div>\n <div>\n <span class=\"mat-caption\">{{ model.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]=\"model.isStartAndEndDatesEmpty()\"\n >\n {{ 'intervalPicker.clear' | translate }}\n </button>\n\n @if (!this.emitOnChange()) {\n <button color=\"primary\" mat-raised-button (click)=\"model.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 (showHelpText()) {\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)=\"model.editor.selectCurrentDay()\">\n @if (model.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.anchorDay' | translate }}\n } @else {\n {{ 'intervalPicker.today' | translate }}\n }\n </button>\n <button mat-button (click)=\"model.editor.selectYesterday()\">\n @if (model.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.last' | translate }} 1 {{ 'intervalPicker.day' | translate }}\n } @else {\n {{ 'intervalPicker.yesterday' | translate }}\n }\n </button>\n @if (lastExcludesToday()) {\n <button mat-button (click)=\"model.editor.selectLastSevenDaysExcludingToday()\">\n {{ 'intervalPicker.last' | translate }} 7 {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectLastThirtyDaysExcludingToday()\">\n {{ 'intervalPicker.last' | translate }} 30 {{ 'intervalPicker.days' | translate }}\n </button>\n } @else {\n <button mat-button (click)=\"model.editor.selectLastSevenDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 7 {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectLastThirtyDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 30 {{ 'intervalPicker.days' | translate }}\n </button>\n }\n @if (intervalInputMode() === 'date-time-range') {\n <div class=\"pt-sm\"></div>\n <button mat-button (click)=\"model.editor.selectLastFiveMinutes()\">\n {{ 'intervalPicker.last' | translate }} 5 {{ 'intervalPicker.minutes' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectLastHour()\">\n {{ 'intervalPicker.last' | translate }} {{ 'intervalPicker.hour' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.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]=\"model.dateRangeForCalendar()\"\n (selectedChange)=\"handleStartCalendarChange($event)\"\n [maxDate]=\"model.endDateAsJSDate()\"\n [dateClass]=\"calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n <mat-calendar\n #rangeCalendarEnd\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"model.dateRangeForCalendar()\"\n (selectedChange)=\"handleEndCalendarChange($event)\"\n [minDate]=\"model.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]=\"model.startDateHtmlString()\"\n (ngModelChange)=\"model.setStartDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearStartDateTime()\"\n [disabled]=\"!this.model.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]=\"model.endDateHtmlString()\"\n (ngModelChange)=\"model.setEndDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearEndDateTime()\"\n [disabled]=\"!this.model.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]=\"model.getStartTimeString()\"\n (ngModelChange)=\"model.setStartTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.model.clearStartTime()\"\n [disabled]=\"!this.model.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]=\"model.getEndTimeString()\"\n (ngModelChange)=\"model.setEndTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.model.clearEndTime()\"\n [disabled]=\"!this.model.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 (showHelpText()) {\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)=\"model.editor.shiftDay(-1)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.day' | translate }}</span>\n <button mat-icon-button (click)=\"model.editor.shiftDay(1)\" [disabled]=\"model.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)=\"model.editor.shiftMonth(-1, true)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.month' | translate }}</span>\n <button mat-icon-button (click)=\"model.editor.shiftMonth(1, true)\" [disabled]=\"model.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)=\"model.editor.shiftYear(-1, true)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.year' | translate }}</span>\n <button mat-icon-button (click)=\"model.editor.shiftYear(1, true)\" [disabled]=\"model.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)=\"model.editor.shiftTime(-1, 'minutes')\"\n [disabled]=\"model.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)=\"model.editor.shiftTime(1, 'minutes')\"\n [disabled]=\"model.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\n mat-icon-button\n (click)=\"model.editor.shiftTime(-1, 'hours')\"\n [disabled]=\"model.isRangeNotSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.hour' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"model.editor.shiftTime(1, 'hours')\"\n [disabled]=\"model.isRangeNotSet()\"\n >\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 (model.startDate()) {\n <button\n mat-menu-item\n (click)=\"model.setFixedAnchorPointToEndDateTime()\"\n [disabled]=\"!model.endDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n <span>{{ 'intervalPicker.endDate' | translate }}</span>\n </button>\n }\n @if (model.endDate()) {\n <button\n mat-menu-item\n (click)=\"model.setFixedAnchorPointToStartDateTime()\"\n [disabled]=\"!model.startDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n <span>{{ 'intervalPicker.startDate' | translate }}</span>\n </button>\n }\n @if (model.fixedAnchorDate()) {\n <button\n mat-menu-item\n (click)=\"this.model.resetFixedAnchorPoint()\"\n [disabled]=\"!model.fixedAnchorDate() || anchorReadOnly()\"\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)=\"model.setAnchorDateFromForm($event)\"\n [value]=\"model.getAnchorPointAsLocaleDate()\"\n [readonly]=\"anchorReadOnly()\"\n >\n </elder-local-date-input>\n @if (!anchorReadOnly()) {\n <button\n mat-icon-button\n matSuffix\n [matMenuTriggerFor]=\"anchorMenu\"\n [disabled]=\"model.isStartAndEndDatesEmpty() && !model.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]=\"anchorReadOnly()\"\n [ngModel]=\"model.getAnchorTimeString()\"\n (ngModelChange)=\"model.setAnchorTimeFromForm($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.model.resetFixedAnchorTime()\"\n [disabled]=\"!model.isFixedAnchorTimeSetAndNotMidnight() || anchorReadOnly()\"\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 }); }
|
|
154
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: ElderIntervalPickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
155
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.5", type: ElderIntervalPickerComponent, isStandalone: true, selector: "elder-interval-picker", inputs: { autoEmitMode: { classPropertyName: "autoEmitMode", publicName: "autoEmitMode", isSignal: false, isRequired: false, transformFunction: null }, emitOnChange: { classPropertyName: "emitOnChange", publicName: "emitOnChange", isSignal: false, isRequired: false, transformFunction: null }, emitType: { classPropertyName: "emitType", publicName: "emitType", isSignal: false, isRequired: false, transformFunction: null }, intervalInputMode: { classPropertyName: "intervalInputMode", publicName: "intervalInputMode", isSignal: true, isRequired: false, transformFunction: null }, showAnchor: { classPropertyName: "showAnchor", publicName: "showAnchor", isSignal: true, isRequired: false, transformFunction: null }, anchorReadOnly: { classPropertyName: "anchorReadOnly", publicName: "anchorReadOnly", isSignal: true, isRequired: false, transformFunction: null }, externalAnchorDateTime: { classPropertyName: "externalAnchorDateTime", publicName: "externalAnchorDateTime", isSignal: true, isRequired: false, transformFunction: null }, externalInterval: { classPropertyName: "externalInterval", publicName: "externalInterval", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { intervalChange: "intervalChange" }, providers: [{ provide: DateAdapter, useClass: CustomDateAdapter }], 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 <div class=\"layout-col\">\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)=\"controller.select.selectCurrentDay()\">\n @if (presenter.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.anchorDay' | translate }}\n } @else {\n {{ 'intervalPicker.today' | translate }}\n }\n </button>\n <button mat-button (click)=\"controller.select.selectYesterday()\">\n @if (presenter.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.last' | translate }} 1 {{ 'intervalPicker.day' | translate }}\n } @else {\n {{ 'intervalPicker.yesterday' | translate }}\n }\n </button>\n <button mat-button (click)=\"controller.select.selectLastSevenDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 7\n {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLastThirtyDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 30\n {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLast365daysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 365\n {{ 'intervalPicker.days' | translate }}\n </button>\n @if (intervalInputMode() === 'date-time-range') {\n <div class=\"pt-sm\"></div>\n <button mat-button (click)=\"controller.select.selectLastFiveMinutes()\">\n {{ 'intervalPicker.last' | translate }} 5\n {{ 'intervalPicker.minutes' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLastHour()\">\n {{ 'intervalPicker.last' | translate }} {{ 'intervalPicker.hour' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLast24Hours()\">\n {{ 'intervalPicker.last' | translate }} 24\n {{ 'intervalPicker.hours' | translate }}\n </button>\n }\n </div>\n </div>\n </div>\n </div>\n <div class=\"layout-col\">\n <!-- smart shift -->\n <div class=\"layout-row place-around-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.smartShift(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ presenter.smartShiftMessage() }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.smartShift(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <!-- calendars and input controls -->\n <div\n class=\"layout-row place-around-center\"\n style=\"align-items: flex-start; min-height: 280px\"\n >\n <mat-calendar\n #rangeCalendarStart\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"presenter.dateRangeForCalendar()\"\n (selectedChange)=\"controller.calendar.setStartDateFromJSDate($event)\"\n [maxDate]=\"presenter.endDateTimeAsJSDate()\"\n [dateClass]=\"presenter.calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n <mat-calendar\n #rangeCalendarEnd\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"presenter.dateRangeForCalendar()\"\n (selectedChange)=\"controller.calendar.setEndDateFromJSDate($event)\"\n [minDate]=\"presenter.startDateTimeAsJSDate()\"\n [dateClass]=\"presenter.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]=\"presenter.startDateHtmlString()\"\n (ngModelChange)=\"controller.form.setStartDateFromString($event)\"\n />\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]=\"presenter.endDateHtmlString()\"\n (ngModelChange)=\"controller.form.setEndDateFromString($event)\"\n />\n </mat-form-field>\n </div>\n </div>\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]=\"presenter.startTimeHtmlString()\"\n (ngModelChange)=\"controller.form.setStartTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"controller.clearStartTime()\"\n [disabled]=\"!presenter.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]=\"presenter.endTimeHtmlString()\"\n (ngModelChange)=\"controller.form.setEndTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"controller.clearEndTime()\"\n [disabled]=\"!presenter.isEndTimeSetAndNotMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n }\n <br />\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 <!-- select current -->\n <button mat-button (click)=\"controller.select.selectCurrentWeek()\">\n {{ 'intervalPicker.currentPeriod.week' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectCurrentMonth()\">\n {{ 'intervalPicker.currentPeriod.month' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectCurrentQuarter()\">\n {{ 'intervalPicker.currentPeriod.quarter' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectCurrentYear()\">\n {{ 'intervalPicker.currentPeriod.year' | translate }}\n </button>\n </div>\n <div class=\"fixed-shifts-container\">\n <div class=\"layout-col place-start-stretch\">\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftDay(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.day' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftDay(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftMonth(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.month' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftMonth(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftYear(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.year' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftYear(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n\n @if (true) {\n <!-- shift minute -->\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftMinute(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\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)=\"controller.shift.shiftMinute(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\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\n mat-icon-button\n (click)=\"controller.shift.shiftHour(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.hour' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftHour(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n }\n </div>\n </div>\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 <div class=\"layout-row flex-none gap-md\">\n <mat-menu #anchorMenu=\"matMenu\">\n @if (presenter.startDateHtmlString()) {\n <button\n mat-menu-item\n (click)=\"controller.anchor.setFixedAnchorPointToEndDateTime()\"\n [disabled]=\"!presenter.startDateHtmlString()\"\n >\n <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n <span>{{ 'intervalPicker.endDate' | translate }}</span>\n </button>\n }\n @if (presenter.endDateHtmlString()) {\n <button\n mat-menu-item\n (click)=\"controller.anchor.setFixedAnchorPointToStartDateTime()\"\n [disabled]=\"!presenter.endDateHtmlString()\"\n >\n <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n <span>{{ 'intervalPicker.startDate' | translate }}</span>\n </button>\n }\n @if (presenter.isFixedAnchorDateSet()) {\n <button\n mat-menu-item\n (click)=\"controller.anchor.resetAnchor()\"\n [disabled]=\"!presenter.isFixedAnchorDateSet() || anchorReadOnly()\"\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 <input\n #startDateInput\n name=\"anchorDateInput\"\n matInput\n type=\"date\"\n [ngModel]=\"presenter.anchorDateHtmlString()\"\n (ngModelChange)=\"controller.form.setAnchorDateFromString($event)\"\n [readonly]=\"anchorReadOnly()\"\n />\n @if (!anchorReadOnly()) {\n <button mat-icon-button matSuffix [matMenuTriggerFor]=\"anchorMenu\" [disabled]=\"presenter.isAnchorMenuDisabled()\">\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]=\"anchorReadOnly()\"\n [ngModel]=\"presenter.anchorTimeHtmlString()\"\n (ngModelChange)=\"controller.form.setAnchorTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"controller.anchor.resetAnchorTime()\"\n [disabled]=\"anchorReadOnly() || presenter.isAnchorTimeMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n }\n </div>\n }\n </div>\n <!-- result interval -->\n <div class=\"layout-col place-center-center\">\n <div class=\"date-interval mat-caption pt-xs\">\n @if (presenter.startDateTimeAsJSDate()) {\n {{ presenter.startDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.startDateNotSet' | translate }}\n }\n <span> - </span>\n @if (presenter.endDateTimeAsJSDate()) {\n {{ presenter.endDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.endDateNotSet' | translate }}\n }\n </div>\n <div>\n <span class=\"mat-caption\">{{ presenter.deltaHumanReadable() || ' ' }}</span>\n </div>\n </div>\n <div class=\"layout-row gap-lg\">\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"controller.clearInterval()\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n {{ 'intervalPicker.clear' | translate }}\n </button>\n\n @if (!this.autoEmitMode) {\n <button color=\"primary\" mat-raised-button (click)=\"controller.manualEmit()\">\n {{ 'actions.ok' | translate }}\n </button>\n }\n </div>\n</div>\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: i1.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: i2.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i2.MatLabel, selector: "mat-label" }, { kind: "directive", type: i2.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i3.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: "pipe", type: i4.DatePipe, name: "date" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.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: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i6.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i7.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i7.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i8.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: MatMenuModule }, { kind: "component", type: i9.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: i9.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i9.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 }); }
|
|
202
156
|
}
|
|
203
157
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: ElderIntervalPickerComponent, decorators: [{
|
|
204
158
|
type: Component,
|
|
@@ -216,12 +170,21 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImpor
|
|
|
216
170
|
FormsModule,
|
|
217
171
|
MatButtonModule,
|
|
218
172
|
MatCalendar,
|
|
219
|
-
ElderLocalDateInputComponent,
|
|
220
173
|
MatTooltip,
|
|
221
174
|
TranslateModule,
|
|
222
175
|
MatMenuModule,
|
|
223
|
-
], providers: [IntervalPickerModel, { provide: DateAdapter, useClass: CustomDateAdapter }], 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)=\"model.editor.smartShift(-1)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ model.viewIntervalMessage() }}</span>\n <button mat-icon-button (click)=\"model.editor.smartShift(1)\" [disabled]=\"model.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 (showHelpText()) {\n <h5 class=\"mat-h5\">{{ 'intervalPicker.selectSmartInterval' | translate }}</h5>\n }\n <!-- select current -->\n <button mat-button (click)=\"model.editor.selectCurrentWeek()\">\n {{ 'intervalPicker.currentPeriod.week' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectCurrentMonth()\">\n {{ 'intervalPicker.currentPeriod.month' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectCurrentQuarter()\">\n {{ 'intervalPicker.currentPeriod.quarter' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.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 (showResultText()) {\n <!-- result interval -->\n <div class=\"layout-col place-center-center\">\n <div class=\"date-interval mat-caption pt-xs\">\n @if (model.startDateTimeAsJSDate()) {\n {{ model.startDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.startDateNotSet' | translate }}\n }\n <span> - </span>\n @if (model.endDateTimeAsJSDate()) {\n {{ model.endDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.endDateNotSet' | translate }}\n }\n </div>\n <div>\n <span class=\"mat-caption\">{{ model.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]=\"model.isStartAndEndDatesEmpty()\"\n >\n {{ 'intervalPicker.clear' | translate }}\n </button>\n\n @if (!this.emitOnChange()) {\n <button color=\"primary\" mat-raised-button (click)=\"model.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 (showHelpText()) {\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)=\"model.editor.selectCurrentDay()\">\n @if (model.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.anchorDay' | translate }}\n } @else {\n {{ 'intervalPicker.today' | translate }}\n }\n </button>\n <button mat-button (click)=\"model.editor.selectYesterday()\">\n @if (model.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.last' | translate }} 1 {{ 'intervalPicker.day' | translate }}\n } @else {\n {{ 'intervalPicker.yesterday' | translate }}\n }\n </button>\n @if (lastExcludesToday()) {\n <button mat-button (click)=\"model.editor.selectLastSevenDaysExcludingToday()\">\n {{ 'intervalPicker.last' | translate }} 7 {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectLastThirtyDaysExcludingToday()\">\n {{ 'intervalPicker.last' | translate }} 30 {{ 'intervalPicker.days' | translate }}\n </button>\n } @else {\n <button mat-button (click)=\"model.editor.selectLastSevenDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 7 {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectLastThirtyDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 30 {{ 'intervalPicker.days' | translate }}\n </button>\n }\n @if (intervalInputMode() === 'date-time-range') {\n <div class=\"pt-sm\"></div>\n <button mat-button (click)=\"model.editor.selectLastFiveMinutes()\">\n {{ 'intervalPicker.last' | translate }} 5 {{ 'intervalPicker.minutes' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.selectLastHour()\">\n {{ 'intervalPicker.last' | translate }} {{ 'intervalPicker.hour' | translate }}\n </button>\n <button mat-button (click)=\"model.editor.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]=\"model.dateRangeForCalendar()\"\n (selectedChange)=\"handleStartCalendarChange($event)\"\n [maxDate]=\"model.endDateAsJSDate()\"\n [dateClass]=\"calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n <mat-calendar\n #rangeCalendarEnd\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"model.dateRangeForCalendar()\"\n (selectedChange)=\"handleEndCalendarChange($event)\"\n [minDate]=\"model.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]=\"model.startDateHtmlString()\"\n (ngModelChange)=\"model.setStartDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearStartDateTime()\"\n [disabled]=\"!this.model.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]=\"model.endDateHtmlString()\"\n (ngModelChange)=\"model.setEndDateFromHtmlString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.clearEndDateTime()\"\n [disabled]=\"!this.model.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]=\"model.getStartTimeString()\"\n (ngModelChange)=\"model.setStartTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.model.clearStartTime()\"\n [disabled]=\"!this.model.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]=\"model.getEndTimeString()\"\n (ngModelChange)=\"model.setEndTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.model.clearEndTime()\"\n [disabled]=\"!this.model.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 (showHelpText()) {\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)=\"model.editor.shiftDay(-1)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.day' | translate }}</span>\n <button mat-icon-button (click)=\"model.editor.shiftDay(1)\" [disabled]=\"model.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)=\"model.editor.shiftMonth(-1, true)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.month' | translate }}</span>\n <button mat-icon-button (click)=\"model.editor.shiftMonth(1, true)\" [disabled]=\"model.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)=\"model.editor.shiftYear(-1, true)\" [disabled]=\"model.isRangeNotSet()\">\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.year' | translate }}</span>\n <button mat-icon-button (click)=\"model.editor.shiftYear(1, true)\" [disabled]=\"model.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)=\"model.editor.shiftTime(-1, 'minutes')\"\n [disabled]=\"model.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)=\"model.editor.shiftTime(1, 'minutes')\"\n [disabled]=\"model.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\n mat-icon-button\n (click)=\"model.editor.shiftTime(-1, 'hours')\"\n [disabled]=\"model.isRangeNotSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.hour' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"model.editor.shiftTime(1, 'hours')\"\n [disabled]=\"model.isRangeNotSet()\"\n >\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 (model.startDate()) {\n <button\n mat-menu-item\n (click)=\"model.setFixedAnchorPointToEndDateTime()\"\n [disabled]=\"!model.endDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n <span>{{ 'intervalPicker.endDate' | translate }}</span>\n </button>\n }\n @if (model.endDate()) {\n <button\n mat-menu-item\n (click)=\"model.setFixedAnchorPointToStartDateTime()\"\n [disabled]=\"!model.startDate()\"\n >\n <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n <span>{{ 'intervalPicker.startDate' | translate }}</span>\n </button>\n }\n @if (model.fixedAnchorDate()) {\n <button\n mat-menu-item\n (click)=\"this.model.resetFixedAnchorPoint()\"\n [disabled]=\"!model.fixedAnchorDate() || anchorReadOnly()\"\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)=\"model.setAnchorDateFromForm($event)\"\n [value]=\"model.getAnchorPointAsLocaleDate()\"\n [readonly]=\"anchorReadOnly()\"\n >\n </elder-local-date-input>\n @if (!anchorReadOnly()) {\n <button\n mat-icon-button\n matSuffix\n [matMenuTriggerFor]=\"anchorMenu\"\n [disabled]=\"model.isStartAndEndDatesEmpty() && !model.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]=\"anchorReadOnly()\"\n [ngModel]=\"model.getAnchorTimeString()\"\n (ngModelChange)=\"model.setAnchorTimeFromForm($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"this.model.resetFixedAnchorTime()\"\n [disabled]=\"!model.isFixedAnchorTimeSetAndNotMidnight() || anchorReadOnly()\"\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"] }]
|
|
224
|
-
|
|
176
|
+
ElderIntervalInputComponent,
|
|
177
|
+
MatMenuModule,
|
|
178
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, providers: [{ provide: DateAdapter, useClass: CustomDateAdapter }], 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 <div class=\"layout-col\">\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)=\"controller.select.selectCurrentDay()\">\n @if (presenter.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.anchorDay' | translate }}\n } @else {\n {{ 'intervalPicker.today' | translate }}\n }\n </button>\n <button mat-button (click)=\"controller.select.selectYesterday()\">\n @if (presenter.isFixedAnchorDateSet()) {\n {{ 'intervalPicker.last' | translate }} 1 {{ 'intervalPicker.day' | translate }}\n } @else {\n {{ 'intervalPicker.yesterday' | translate }}\n }\n </button>\n <button mat-button (click)=\"controller.select.selectLastSevenDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 7\n {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLastThirtyDaysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 30\n {{ 'intervalPicker.days' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLast365daysIncludingToday()\">\n {{ 'intervalPicker.last' | translate }} 365\n {{ 'intervalPicker.days' | translate }}\n </button>\n @if (intervalInputMode() === 'date-time-range') {\n <div class=\"pt-sm\"></div>\n <button mat-button (click)=\"controller.select.selectLastFiveMinutes()\">\n {{ 'intervalPicker.last' | translate }} 5\n {{ 'intervalPicker.minutes' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLastHour()\">\n {{ 'intervalPicker.last' | translate }} {{ 'intervalPicker.hour' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectLast24Hours()\">\n {{ 'intervalPicker.last' | translate }} 24\n {{ 'intervalPicker.hours' | translate }}\n </button>\n }\n </div>\n </div>\n </div>\n </div>\n <div class=\"layout-col\">\n <!-- smart shift -->\n <div class=\"layout-row place-around-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.smartShift(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ presenter.smartShiftMessage() }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.smartShift(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <!-- calendars and input controls -->\n <div\n class=\"layout-row place-around-center\"\n style=\"align-items: flex-start; min-height: 280px\"\n >\n <mat-calendar\n #rangeCalendarStart\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"presenter.dateRangeForCalendar()\"\n (selectedChange)=\"controller.calendar.setStartDateFromJSDate($event)\"\n [maxDate]=\"presenter.endDateTimeAsJSDate()\"\n [dateClass]=\"presenter.calendarAnchorDateCssClassFn\"\n >\n </mat-calendar>\n <mat-calendar\n #rangeCalendarEnd\n style=\"width: 210px; max-width: 100%\"\n [selected]=\"presenter.dateRangeForCalendar()\"\n (selectedChange)=\"controller.calendar.setEndDateFromJSDate($event)\"\n [minDate]=\"presenter.startDateTimeAsJSDate()\"\n [dateClass]=\"presenter.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]=\"presenter.startDateHtmlString()\"\n (ngModelChange)=\"controller.form.setStartDateFromString($event)\"\n />\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]=\"presenter.endDateHtmlString()\"\n (ngModelChange)=\"controller.form.setEndDateFromString($event)\"\n />\n </mat-form-field>\n </div>\n </div>\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]=\"presenter.startTimeHtmlString()\"\n (ngModelChange)=\"controller.form.setStartTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"controller.clearStartTime()\"\n [disabled]=\"!presenter.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]=\"presenter.endTimeHtmlString()\"\n (ngModelChange)=\"controller.form.setEndTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"controller.clearEndTime()\"\n [disabled]=\"!presenter.isEndTimeSetAndNotMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n </div>\n }\n <br />\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 <!-- select current -->\n <button mat-button (click)=\"controller.select.selectCurrentWeek()\">\n {{ 'intervalPicker.currentPeriod.week' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectCurrentMonth()\">\n {{ 'intervalPicker.currentPeriod.month' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectCurrentQuarter()\">\n {{ 'intervalPicker.currentPeriod.quarter' | translate }}\n </button>\n <button mat-button (click)=\"controller.select.selectCurrentYear()\">\n {{ 'intervalPicker.currentPeriod.year' | translate }}\n </button>\n </div>\n <div class=\"fixed-shifts-container\">\n <div class=\"layout-col place-start-stretch\">\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftDay(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.day' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftDay(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftMonth(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.month' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftMonth(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftYear(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.year' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftYear(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n\n @if (true) {\n <!-- shift minute -->\n <div class=\"layout-row place-between-center\">\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftMinute(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\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)=\"controller.shift.shiftMinute(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\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\n mat-icon-button\n (click)=\"controller.shift.shiftHour(-1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_left</mat-icon>\n </button>\n <span>{{ 'intervalPicker.hour' | translate }}</span>\n <button\n mat-icon-button\n (click)=\"controller.shift.shiftHour(1)\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n <mat-icon>keyboard_double_arrow_right</mat-icon>\n </button>\n </div>\n }\n </div>\n </div>\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 <div class=\"layout-row flex-none gap-md\">\n <mat-menu #anchorMenu=\"matMenu\">\n @if (presenter.startDateHtmlString()) {\n <button\n mat-menu-item\n (click)=\"controller.anchor.setFixedAnchorPointToEndDateTime()\"\n [disabled]=\"!presenter.startDateHtmlString()\"\n >\n <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n <span>{{ 'intervalPicker.endDate' | translate }}</span>\n </button>\n }\n @if (presenter.endDateHtmlString()) {\n <button\n mat-menu-item\n (click)=\"controller.anchor.setFixedAnchorPointToStartDateTime()\"\n [disabled]=\"!presenter.endDateHtmlString()\"\n >\n <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n <span>{{ 'intervalPicker.startDate' | translate }}</span>\n </button>\n }\n @if (presenter.isFixedAnchorDateSet()) {\n <button\n mat-menu-item\n (click)=\"controller.anchor.resetAnchor()\"\n [disabled]=\"!presenter.isFixedAnchorDateSet() || anchorReadOnly()\"\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 <input\n #startDateInput\n name=\"anchorDateInput\"\n matInput\n type=\"date\"\n [ngModel]=\"presenter.anchorDateHtmlString()\"\n (ngModelChange)=\"controller.form.setAnchorDateFromString($event)\"\n [readonly]=\"anchorReadOnly()\"\n />\n @if (!anchorReadOnly()) {\n <button mat-icon-button matSuffix [matMenuTriggerFor]=\"anchorMenu\" [disabled]=\"presenter.isAnchorMenuDisabled()\">\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]=\"anchorReadOnly()\"\n [ngModel]=\"presenter.anchorTimeHtmlString()\"\n (ngModelChange)=\"controller.form.setAnchorTimeFromString($event)\"\n />\n <button\n mat-icon-button\n matSuffix\n (click)=\"controller.anchor.resetAnchorTime()\"\n [disabled]=\"anchorReadOnly() || presenter.isAnchorTimeMidnight()\"\n >\n <mat-icon>close</mat-icon>\n </button>\n </mat-form-field>\n }\n </div>\n }\n </div>\n <!-- result interval -->\n <div class=\"layout-col place-center-center\">\n <div class=\"date-interval mat-caption pt-xs\">\n @if (presenter.startDateTimeAsJSDate()) {\n {{ presenter.startDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.startDateNotSet' | translate }}\n }\n <span> - </span>\n @if (presenter.endDateTimeAsJSDate()) {\n {{ presenter.endDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n } @else {\n {{ 'intervalPicker.endDateNotSet' | translate }}\n }\n </div>\n <div>\n <span class=\"mat-caption\">{{ presenter.deltaHumanReadable() || ' ' }}</span>\n </div>\n </div>\n <div class=\"layout-row gap-lg\">\n <button\n mat-raised-button\n color=\"primary\"\n (click)=\"controller.clearInterval()\"\n [disabled]=\"!presenter.isValidIntervalSet()\"\n >\n {{ 'intervalPicker.clear' | translate }}\n </button>\n\n @if (!this.autoEmitMode) {\n <button color=\"primary\" mat-raised-button (click)=\"controller.manualEmit()\">\n {{ 'actions.ok' | translate }}\n </button>\n }\n </div>\n</div>\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"] }]
|
|
179
|
+
}], ctorParameters: () => [], propDecorators: { intervalChange: [{
|
|
180
|
+
type: Output
|
|
181
|
+
}], autoEmitMode: [{
|
|
182
|
+
type: Input
|
|
183
|
+
}], emitOnChange: [{
|
|
184
|
+
type: Input
|
|
185
|
+
}], emitType: [{
|
|
186
|
+
type: Input
|
|
187
|
+
}], calendarStart: [{
|
|
225
188
|
type: ViewChild,
|
|
226
189
|
args: ['rangeCalendarStart']
|
|
227
190
|
}], calendarEnd: [{
|
|
@@ -239,11 +202,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImpor
|
|
|
239
202
|
}], endDateInput: [{
|
|
240
203
|
type: ViewChild,
|
|
241
204
|
args: ['endDateInput']
|
|
242
|
-
}], dateChange: [{
|
|
243
|
-
type: Output
|
|
244
|
-
}], interval: [{
|
|
245
|
-
type: Input
|
|
246
|
-
}], anchorDateTime: [{
|
|
247
|
-
type: Input
|
|
248
205
|
}] } });
|
|
249
|
-
//# 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  AfterViewInit,\n  booleanAttribute,\n  ChangeDetectionStrategy,\n  Component,\n  effect,\n  ElementRef,\n  EventEmitter,\n  Input,\n  input,\n  OnDestroy,\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 { MatFormFieldModule, MatHint, MatLabel } 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 '../date-adapters/custom-date-adapter'\nimport { IntervalInputMode } from '../elder-interval-input/elder-interval-input.component'\nimport { ElderLocalDateInputComponent } from '../elder-local-date-input/elder-local-date-input.component'\nimport { IntervalPickerModel } from './model/interval-picker-model'\nimport { Temporal } from '@js-temporal/polyfill'\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: [IntervalPickerModel, { provide: DateAdapter, useClass: CustomDateAdapter }],\n  templateUrl: './elder-interval-picker.component.html',\n  styleUrl: './elder-interval-picker.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ElderIntervalPickerComponent implements AfterViewInit, OnDestroy {\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name)\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  public readonly emitOnChange = input(true, { transform: (v) => booleanAttribute(v) })\n  public readonly intervalInputMode = input<IntervalInputMode>('date-range')\n  public readonly showAnchor = input(false, { transform: (v) => booleanAttribute(v) })\n  public readonly anchorReadOnly = input(false, { transform: (v) => booleanAttribute(v) })\n  public readonly showResultText = input(true, { transform: (v) => booleanAttribute(v) })\n  public readonly showHelpText = input(false, { transform: (v) => booleanAttribute(v) })\n  public readonly lastExcludesToday = input(false, { transform: (v) => booleanAttribute(v) })\n\n  @Output() dateChange = new EventEmitter<Interval>()\n\n  private calendarElFocusEventHandler: (event: FocusEvent) => void\n\n  @Input()\n  public set interval(value: Interval) {\n    this.model.setInterval(value)\n  }\n\n  @Input()\n  public set anchorDateTime(value: Date) {\n    this.model.setAnchorDateTime(value)\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(public model: IntervalPickerModel) {\n    this.model.setupEventEmitter(this.dateChange)\n\n    effect((): void => {\n      if (this.model.anchorDate()) {\n        this.updateMatCalendarTodayDate()\n      }\n    })\n\n    effect((): void => {\n      this.updateMatCalendarActiveDate(this.model.startDate(), this.model.endDate())\n    })\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngAfterViewInit(): void {\n    this.model.isEmittingOnChange.set(this.emitOnChange())\n    this.setupDateInputCalendarFocusPrevention()\n    setTimeout(() => {\n      this.model.setInitialized()\n    }, 0)\n  }\n\n  public ngOnDestroy(): void {\n    this.removeEventListeners()\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public 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      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  public clearStartDateTime(): void {\n    this.model.clearStartDateTime()\n\n    this.model.resetCalendarMode()\n\n    if (this.emitOnChange()) {\n      this.model.emitMainDates()\n    }\n  }\n\n  public clearEndDateTime(): void {\n    this.model.clearEndDateTime()\n\n    this.model.resetCalendarMode()\n\n    if (this.emitOnChange()) {\n      this.model.emitMainDates()\n    }\n  }\n\n  public clearStartAndEndDateTimes(): void {\n    this.model.clearStartDateTime()\n    this.model.clearEndDateTime()\n\n    this.model.resetCalendarMode()\n\n    if (this.emitOnChange()) {\n      this.model.emitMainDates()\n    }\n  }\n\n  public handleStartCalendarChange(changedDate: Date): void {\n    this.model.calendarMode.set('days')\n    const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate)\n    this.model.setStartDate(newPlainDate)\n  }\n\n  public handleEndCalendarChange(changedDate: Date): void {\n    this.model.calendarMode.set('days')\n    const newPlainDate = TemporalUtil.getPlainDateFromJSDate(changedDate)\n    this.model.setEndDate(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  public readonly calendarAnchorDateCssClassFn: MatCalendarCellClassFunction<Date> = (\n    cellDate: Date,\n    view: string\n  ): string => {\n    return this.calendarAnchorDateCssClassFnBody(cellDate, view)\n  }\n\n  private calendarAnchorDateCssClassFnBody(cellDate: Date, view: string): string {\n    const anchorDate = this.model.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  private updateMatCalendarActiveDate(startDate: Temporal.PlainDate, endDate: Temporal.PlainDate) {\n    if (!this.calendarStart || !this.calendarEnd) {\n      return\n    }\n\n    if (!startDate && !endDate) {\n      return\n    }\n\n    const calendarStart = this.calendarStart\n    const calendarEnd = this.calendarEnd\n\n    setTimeout(() => {\n      // timeout is needed to run calendar update last, otherwise there are issues\n      if (startDate) {\n        calendarStart.activeDate = TemporalUtil.getJSDateFromPlainDateTime(startDate)\n      }\n\n      if (endDate) {\n        calendarEnd.activeDate = TemporalUtil.getJSDateFromPlainDateTime(endDate)\n      }\n    }, 0)\n  }\n\n  private updateMatCalendarTodayDate(): void {\n    this.calendarStart.updateTodaysDate()\n    this.calendarEnd.updateTodaysDate()\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)=\"model.editor.smartShift(-1)\" [disabled]=\"model.isRangeNotSet()\">\n          <mat-icon>keyboard_double_arrow_left</mat-icon>\n        </button>\n        <span>{{ model.viewIntervalMessage() }}</span>\n        <button mat-icon-button (click)=\"model.editor.smartShift(1)\" [disabled]=\"model.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 (showHelpText()) {\n            <h5 class=\"mat-h5\">{{ 'intervalPicker.selectSmartInterval' | translate }}</h5>\n          }\n          <!-- select current -->\n          <button mat-button (click)=\"model.editor.selectCurrentWeek()\">\n            {{ 'intervalPicker.currentPeriod.week' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.selectCurrentMonth()\">\n            {{ 'intervalPicker.currentPeriod.month' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.selectCurrentQuarter()\">\n            {{ 'intervalPicker.currentPeriod.quarter' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.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 (showResultText()) {\n      <!-- result interval -->\n      <div class=\"layout-col place-center-center\">\n        <div class=\"date-interval mat-caption pt-xs\">\n          @if (model.startDateTimeAsJSDate()) {\n            {{ model.startDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n          } @else {\n            {{ 'intervalPicker.startDateNotSet' | translate }}\n          }\n          <span> - </span>\n          @if (model.endDateTimeAsJSDate()) {\n            {{ model.endDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n          } @else {\n            {{ 'intervalPicker.endDateNotSet' | translate }}\n          }\n        </div>\n        <div>\n          <span class=\"mat-caption\">{{ model.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]=\"model.isStartAndEndDatesEmpty()\"\n      >\n        {{ 'intervalPicker.clear' | translate }}\n      </button>\n\n      @if (!this.emitOnChange()) {\n        <button color=\"primary\" mat-raised-button (click)=\"model.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 (showHelpText()) {\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)=\"model.editor.selectCurrentDay()\">\n          @if (model.isFixedAnchorDateSet()) {\n            {{ 'intervalPicker.anchorDay' | translate }}\n          } @else {\n            {{ 'intervalPicker.today' | translate }}\n          }\n        </button>\n        <button mat-button (click)=\"model.editor.selectYesterday()\">\n          @if (model.isFixedAnchorDateSet()) {\n            {{ 'intervalPicker.last' | translate }} 1 {{ 'intervalPicker.day' | translate }}\n          } @else {\n            {{ 'intervalPicker.yesterday' | translate }}\n          }\n        </button>\n        @if (lastExcludesToday()) {\n          <button mat-button (click)=\"model.editor.selectLastSevenDaysExcludingToday()\">\n            {{ 'intervalPicker.last' | translate }} 7 {{ 'intervalPicker.days' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.selectLastThirtyDaysExcludingToday()\">\n            {{ 'intervalPicker.last' | translate }} 30 {{ 'intervalPicker.days' | translate }}\n          </button>\n        } @else {\n          <button mat-button (click)=\"model.editor.selectLastSevenDaysIncludingToday()\">\n            {{ 'intervalPicker.last' | translate }} 7 {{ 'intervalPicker.days' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.selectLastThirtyDaysIncludingToday()\">\n            {{ 'intervalPicker.last' | translate }} 30 {{ 'intervalPicker.days' | translate }}\n          </button>\n        }\n        @if (intervalInputMode() === 'date-time-range') {\n          <div class=\"pt-sm\"></div>\n          <button mat-button (click)=\"model.editor.selectLastFiveMinutes()\">\n            {{ 'intervalPicker.last' | translate }} 5 {{ 'intervalPicker.minutes' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.selectLastHour()\">\n            {{ 'intervalPicker.last' | translate }} {{ 'intervalPicker.hour' | translate }}\n          </button>\n          <button mat-button (click)=\"model.editor.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]=\"model.dateRangeForCalendar()\"\n      (selectedChange)=\"handleStartCalendarChange($event)\"\n      [maxDate]=\"model.endDateAsJSDate()\"\n      [dateClass]=\"calendarAnchorDateCssClassFn\"\n    >\n    </mat-calendar>\n    <mat-calendar\n      #rangeCalendarEnd\n      style=\"width: 210px; max-width: 100%\"\n      [selected]=\"model.dateRangeForCalendar()\"\n      (selectedChange)=\"handleEndCalendarChange($event)\"\n      [minDate]=\"model.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]=\"model.startDateHtmlString()\"\n          (ngModelChange)=\"model.setStartDateFromHtmlString($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.clearStartDateTime()\"\n          [disabled]=\"!this.model.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]=\"model.endDateHtmlString()\"\n          (ngModelChange)=\"model.setEndDateFromHtmlString($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.clearEndDateTime()\"\n          [disabled]=\"!this.model.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]=\"model.getStartTimeString()\"\n          (ngModelChange)=\"model.setStartTimeFromString($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.model.clearStartTime()\"\n          [disabled]=\"!this.model.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]=\"model.getEndTimeString()\"\n          (ngModelChange)=\"model.setEndTimeFromString($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.model.clearEndTime()\"\n          [disabled]=\"!this.model.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 (showHelpText()) {\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)=\"model.editor.shiftDay(-1)\" [disabled]=\"model.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ 'intervalPicker.day' | translate }}</span>\n      <button mat-icon-button (click)=\"model.editor.shiftDay(1)\" [disabled]=\"model.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)=\"model.editor.shiftMonth(-1, true)\" [disabled]=\"model.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ 'intervalPicker.month' | translate }}</span>\n      <button mat-icon-button (click)=\"model.editor.shiftMonth(1, true)\" [disabled]=\"model.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)=\"model.editor.shiftYear(-1, true)\" [disabled]=\"model.isRangeNotSet()\">\n        <mat-icon>keyboard_double_arrow_left</mat-icon>\n      </button>\n      <span>{{ 'intervalPicker.year' | translate }}</span>\n      <button mat-icon-button (click)=\"model.editor.shiftYear(1, true)\" [disabled]=\"model.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)=\"model.editor.shiftTime(-1, 'minutes')\"\n          [disabled]=\"model.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)=\"model.editor.shiftTime(1, 'minutes')\"\n          [disabled]=\"model.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\n          mat-icon-button\n          (click)=\"model.editor.shiftTime(-1, 'hours')\"\n          [disabled]=\"model.isRangeNotSet()\"\n        >\n          <mat-icon>keyboard_double_arrow_left</mat-icon>\n        </button>\n        <span>{{ 'intervalPicker.hour' | translate }}</span>\n        <button\n          mat-icon-button\n          (click)=\"model.editor.shiftTime(1, 'hours')\"\n          [disabled]=\"model.isRangeNotSet()\"\n        >\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 (model.startDate()) {\n        <button\n          mat-menu-item\n          (click)=\"model.setFixedAnchorPointToEndDateTime()\"\n          [disabled]=\"!model.endDate()\"\n        >\n          <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n          <span>{{ 'intervalPicker.endDate' | translate }}</span>\n        </button>\n      }\n      @if (model.endDate()) {\n        <button\n          mat-menu-item\n          (click)=\"model.setFixedAnchorPointToStartDateTime()\"\n          [disabled]=\"!model.startDate()\"\n        >\n          <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n          <span>{{ 'intervalPicker.startDate' | translate }}</span>\n        </button>\n      }\n      @if (model.fixedAnchorDate()) {\n        <button\n          mat-menu-item\n          (click)=\"this.model.resetFixedAnchorPoint()\"\n          [disabled]=\"!model.fixedAnchorDate() || anchorReadOnly()\"\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)=\"model.setAnchorDateFromForm($event)\"\n        [value]=\"model.getAnchorPointAsLocaleDate()\"\n        [readonly]=\"anchorReadOnly()\"\n      >\n      </elder-local-date-input>\n      @if (!anchorReadOnly()) {\n        <button\n          mat-icon-button\n          matSuffix\n          [matMenuTriggerFor]=\"anchorMenu\"\n          [disabled]=\"model.isStartAndEndDatesEmpty() && !model.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]=\"anchorReadOnly()\"\n          [ngModel]=\"model.getAnchorTimeString()\"\n          (ngModelChange)=\"model.setAnchorTimeFromForm($event)\"\n        />\n        <button\n          mat-icon-button\n          matSuffix\n          (click)=\"this.model.resetFixedAnchorTime()\"\n          [disabled]=\"!model.isFixedAnchorTimeSetAndNotMidnight() || anchorReadOnly()\"\n        >\n          <mat-icon>close</mat-icon>\n        </button>\n      </mat-form-field>\n    }\n  </div>\n</ng-template>\n"]}
|
|
206
|
+
//# 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  ElementRef,\n  Input,\n  input,\n  Output,\n  ViewChild,\n} from '@angular/core'\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop'\nimport { FormsModule } from '@angular/forms'\nimport { MatButtonModule } from '@angular/material/button'\nimport { DateAdapter } from '@angular/material/core'\nimport {\n  MatCalendar,\n  MatDatepickerModule,\n  MatDatepickerToggle,\n  MatDateRangeInput,\n  MatDateRangePicker,\n} from '@angular/material/datepicker'\nimport { MatFormFieldModule, MatHint, MatLabel } 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 { Temporal } from '@js-temporal/polyfill'\nimport { TranslateModule } from '@ngx-translate/core'\nimport { map, Observable } from 'rxjs'\nimport { coerceInterval, coerceIntervalIsoStr } from '../../../common/public_api'\nimport { Interval, IsoIntervalStr } from '../../../common/time/interval'\nimport { CustomDateAdapter } from '../date-adapters/custom-date-adapter'\nimport {\n  ElderIntervalInputComponent,\n  IntervalInputMode,\n} from '../elder-interval-input/elder-interval-input.component'\nimport IntervalController from './controller/interval-controller'\nimport IntervalPickerStateManager from './manager/interval-picker-state-manager'\nimport IntervalState from './model/interval-state'\nimport IntervalPickerPresenter from './presenter/interval-picker-presenter'\nimport IntervalPickerUtil from './util/interval-picker-util'\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    MatTooltip,\n    TranslateModule,\n    MatMenuModule,\n    ElderIntervalInputComponent,\n    MatMenuModule,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  templateUrl: './elder-interval-picker.component.html',\n  styleUrl: './elder-interval-picker.component.scss',\n  providers: [{ provide: DateAdapter, useClass: CustomDateAdapter }],\n})\nexport class ElderIntervalPickerComponent {\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name)\n\n  // Private\n  private readonly manager: IntervalPickerStateManager\n\n  // Public\n  public readonly controller: IntervalController\n  public readonly presenter: IntervalPickerPresenter\n\n  @Output()\n  public readonly intervalChange: Observable<Interval | IsoIntervalStr | null>\n\n  @Input()\n  public autoEmitMode = true\n\n  @Input()\n  public emitOnChange = true\n\n  @Input()\n  public emitType: 'interval' | 'iso' = 'interval'\n\n  intervalInputMode = input<IntervalInputMode>('date-time-range')\n  showAnchor = input<boolean>(false)\n  anchorReadOnly = input<boolean>(false)\n  externalAnchorDateTime = input<Temporal.PlainDateTime | null>(Temporal.Now.plainDateTimeISO())\n  externalInterval = input<Interval | IsoIntervalStr | null>(null)\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  private calendarElFocusEventHandler: (event: FocusEvent) => void\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor() {\n    this.manager = new IntervalPickerStateManager()\n    this.controller = new IntervalController(this.manager)\n    this.presenter = new IntervalPickerPresenter(\n      this.manager.interval$,\n      this.manager.anchor$,\n      this.manager.smartShift$\n    )\n\n    this.manager.intervalChange.pipe(takeUntilDestroyed()).subscribe({\n      next: (state) => this.tryUpdateMatCalendarActiveDate(state),\n    })\n\n    this.intervalChange = this.manager.intervalChange.pipe(\n      map((srcInterval: any) => {\n        if (!srcInterval) {\n          return null\n        }\n\n        const elderInterval = IntervalPickerUtil.createElderInterval(srcInterval)\n\n        if (this.emitType === 'iso') {\n          console.log('emitting type iso', coerceIntervalIsoStr(elderInterval))\n          return coerceIntervalIsoStr(elderInterval)\n        }\n\n        console.log('emitting type elder interval', elderInterval)\n        return elderInterval\n      })\n    )\n\n    this.manager.anchor$.pipe(takeUntilDestroyed()).subscribe((anchor) => {\n      IntervalPickerUtil.updateMatCalendarTodayDate(this.calendarStart, this.calendarEnd)\n    })\n\n    const externalAnchorObservable = toObservable(this.externalAnchorDateTime)\n    externalAnchorObservable.pipe(takeUntilDestroyed()).subscribe({\n      next: (date) => {\n        this.manager.setExternalAnchor(date)\n      },\n    })\n\n    const externalIntervalObservable = toObservable(this.externalInterval)\n    externalIntervalObservable.pipe(takeUntilDestroyed()).subscribe({\n      next: (interval: any) => {\n        const result = coerceInterval(interval)\n        this.manager.setInterval(IntervalPickerUtil.transformToIntervalState(result || null))\n      },\n    })\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  ngOnInit(): void {}\n\n  ngOnDestroy(): void {\n    this.removeEventListeners()\n  }\n\n  ngAfterViewInit(): void {\n    this.setupDateInputCalendarFocusPrevention()\n\n    setTimeout(() => {\n      // this prevents ngmodel issue\n      this.manager.isInitialized = true\n      console.log('externalInterval', this.externalInterval())\n    }, 0)\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private 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      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  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 tryUpdateMatCalendarActiveDate(interval: IntervalState): void {\n    try {\n      IntervalPickerUtil.updateMatCalendarActiveDate(\n        this.calendarStart,\n        this.calendarEnd,\n        interval.startDate,\n        interval.endDate\n      )\n    } catch (e) {\n      console.debug('Failed to update calendar active date', e)\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      <div class=\"layout-col\">\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)=\"controller.select.selectCurrentDay()\">\n              @if (presenter.isFixedAnchorDateSet()) {\n                {{ 'intervalPicker.anchorDay' | translate }}\n              } @else {\n                {{ 'intervalPicker.today' | translate }}\n              }\n            </button>\n            <button mat-button (click)=\"controller.select.selectYesterday()\">\n              @if (presenter.isFixedAnchorDateSet()) {\n                {{ 'intervalPicker.last' | translate }} 1 {{ 'intervalPicker.day' | translate }}\n              } @else {\n                {{ 'intervalPicker.yesterday' | translate }}\n              }\n            </button>\n            <button mat-button (click)=\"controller.select.selectLastSevenDaysIncludingToday()\">\n              {{ 'intervalPicker.last' | translate }} 7\n              {{ 'intervalPicker.days' | translate }}\n            </button>\n            <button mat-button (click)=\"controller.select.selectLastThirtyDaysIncludingToday()\">\n              {{ 'intervalPicker.last' | translate }} 30\n              {{ 'intervalPicker.days' | translate }}\n            </button>\n            <button mat-button (click)=\"controller.select.selectLast365daysIncludingToday()\">\n              {{ 'intervalPicker.last' | translate }} 365\n              {{ 'intervalPicker.days' | translate }}\n            </button>\n            @if (intervalInputMode() === 'date-time-range') {\n              <div class=\"pt-sm\"></div>\n              <button mat-button (click)=\"controller.select.selectLastFiveMinutes()\">\n                {{ 'intervalPicker.last' | translate }} 5\n                {{ 'intervalPicker.minutes' | translate }}\n              </button>\n              <button mat-button (click)=\"controller.select.selectLastHour()\">\n                {{ 'intervalPicker.last' | translate }} {{ 'intervalPicker.hour' | translate }}\n              </button>\n              <button mat-button (click)=\"controller.select.selectLast24Hours()\">\n                {{ 'intervalPicker.last' | translate }} 24\n                {{ 'intervalPicker.hours' | translate }}\n              </button>\n            }\n          </div>\n        </div>\n      </div>\n    </div>\n    <div class=\"layout-col\">\n      <!-- smart shift -->\n      <div class=\"layout-row place-around-center\">\n        <button\n          mat-icon-button\n          (click)=\"controller.shift.smartShift(-1)\"\n          [disabled]=\"!presenter.isValidIntervalSet()\"\n        >\n          <mat-icon>keyboard_double_arrow_left</mat-icon>\n        </button>\n        <span>{{ presenter.smartShiftMessage() }}</span>\n        <button\n          mat-icon-button\n          (click)=\"controller.shift.smartShift(1)\"\n          [disabled]=\"!presenter.isValidIntervalSet()\"\n        >\n          <mat-icon>keyboard_double_arrow_right</mat-icon>\n        </button>\n      </div>\n      <!-- calendars and input controls -->\n      <div\n        class=\"layout-row place-around-center\"\n        style=\"align-items: flex-start; min-height: 280px\"\n      >\n        <mat-calendar\n          #rangeCalendarStart\n          style=\"width: 210px; max-width: 100%\"\n          [selected]=\"presenter.dateRangeForCalendar()\"\n          (selectedChange)=\"controller.calendar.setStartDateFromJSDate($event)\"\n          [maxDate]=\"presenter.endDateTimeAsJSDate()\"\n          [dateClass]=\"presenter.calendarAnchorDateCssClassFn\"\n        >\n        </mat-calendar>\n        <mat-calendar\n          #rangeCalendarEnd\n          style=\"width: 210px; max-width: 100%\"\n          [selected]=\"presenter.dateRangeForCalendar()\"\n          (selectedChange)=\"controller.calendar.setEndDateFromJSDate($event)\"\n          [minDate]=\"presenter.startDateTimeAsJSDate()\"\n          [dateClass]=\"presenter.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]=\"presenter.startDateHtmlString()\"\n              (ngModelChange)=\"controller.form.setStartDateFromString($event)\"\n            />\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]=\"presenter.endDateHtmlString()\"\n              (ngModelChange)=\"controller.form.setEndDateFromString($event)\"\n            />\n          </mat-form-field>\n        </div>\n      </div>\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]=\"presenter.startTimeHtmlString()\"\n              (ngModelChange)=\"controller.form.setStartTimeFromString($event)\"\n            />\n            <button\n              mat-icon-button\n              matSuffix\n              (click)=\"controller.clearStartTime()\"\n              [disabled]=\"!presenter.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]=\"presenter.endTimeHtmlString()\"\n              (ngModelChange)=\"controller.form.setEndTimeFromString($event)\"\n            />\n            <button\n              mat-icon-button\n              matSuffix\n              (click)=\"controller.clearEndTime()\"\n              [disabled]=\"!presenter.isEndTimeSetAndNotMidnight()\"\n            >\n              <mat-icon>close</mat-icon>\n            </button>\n          </mat-form-field>\n        </div>\n      }\n      <br />\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          <!-- select current -->\n          <button mat-button (click)=\"controller.select.selectCurrentWeek()\">\n            {{ 'intervalPicker.currentPeriod.week' | translate }}\n          </button>\n          <button mat-button (click)=\"controller.select.selectCurrentMonth()\">\n            {{ 'intervalPicker.currentPeriod.month' | translate }}\n          </button>\n          <button mat-button (click)=\"controller.select.selectCurrentQuarter()\">\n            {{ 'intervalPicker.currentPeriod.quarter' | translate }}\n          </button>\n          <button mat-button (click)=\"controller.select.selectCurrentYear()\">\n            {{ 'intervalPicker.currentPeriod.year' | translate }}\n          </button>\n        </div>\n        <div class=\"fixed-shifts-container\">\n          <div class=\"layout-col place-start-stretch\">\n            <div class=\"layout-row place-between-center\">\n              <button\n                mat-icon-button\n                (click)=\"controller.shift.shiftDay(-1)\"\n                [disabled]=\"!presenter.isValidIntervalSet()\"\n              >\n                <mat-icon>keyboard_double_arrow_left</mat-icon>\n              </button>\n              <span>{{ 'intervalPicker.day' | translate }}</span>\n              <button\n                mat-icon-button\n                (click)=\"controller.shift.shiftDay(1)\"\n                [disabled]=\"!presenter.isValidIntervalSet()\"\n              >\n                <mat-icon>keyboard_double_arrow_right</mat-icon>\n              </button>\n            </div>\n            <div class=\"layout-row place-between-center\">\n              <button\n                mat-icon-button\n                (click)=\"controller.shift.shiftMonth(-1)\"\n                [disabled]=\"!presenter.isValidIntervalSet()\"\n              >\n                <mat-icon>keyboard_double_arrow_left</mat-icon>\n              </button>\n              <span>{{ 'intervalPicker.month' | translate }}</span>\n              <button\n                mat-icon-button\n                (click)=\"controller.shift.shiftMonth(1)\"\n                [disabled]=\"!presenter.isValidIntervalSet()\"\n              >\n                <mat-icon>keyboard_double_arrow_right</mat-icon>\n              </button>\n            </div>\n            <div class=\"layout-row place-between-center\">\n              <button\n                mat-icon-button\n                (click)=\"controller.shift.shiftYear(-1)\"\n                [disabled]=\"!presenter.isValidIntervalSet()\"\n              >\n                <mat-icon>keyboard_double_arrow_left</mat-icon>\n              </button>\n              <span>{{ 'intervalPicker.year' | translate }}</span>\n              <button\n                mat-icon-button\n                (click)=\"controller.shift.shiftYear(1)\"\n                [disabled]=\"!presenter.isValidIntervalSet()\"\n              >\n                <mat-icon>keyboard_double_arrow_right</mat-icon>\n              </button>\n            </div>\n\n            @if (true) {\n              <!-- shift minute -->\n              <div class=\"layout-row place-between-center\">\n                <button\n                  mat-icon-button\n                  (click)=\"controller.shift.shiftMinute(-1)\"\n                  [disabled]=\"!presenter.isValidIntervalSet()\"\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)=\"controller.shift.shiftMinute(1)\"\n                  [disabled]=\"!presenter.isValidIntervalSet()\"\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\n                  mat-icon-button\n                  (click)=\"controller.shift.shiftHour(-1)\"\n                  [disabled]=\"!presenter.isValidIntervalSet()\"\n                >\n                  <mat-icon>keyboard_double_arrow_left</mat-icon>\n                </button>\n                <span>{{ 'intervalPicker.hour' | translate }}</span>\n                <button\n                  mat-icon-button\n                  (click)=\"controller.shift.shiftHour(1)\"\n                  [disabled]=\"!presenter.isValidIntervalSet()\"\n                >\n                  <mat-icon>keyboard_double_arrow_right</mat-icon>\n                </button>\n              </div>\n            }\n          </div>\n        </div>\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      <div class=\"layout-row flex-none gap-md\">\n        <mat-menu #anchorMenu=\"matMenu\">\n          @if (presenter.startDateHtmlString()) {\n            <button\n              mat-menu-item\n              (click)=\"controller.anchor.setFixedAnchorPointToEndDateTime()\"\n              [disabled]=\"!presenter.startDateHtmlString()\"\n            >\n              <mat-icon class=\"material-symbols-outlined\">login</mat-icon>\n              <span>{{ 'intervalPicker.endDate' | translate }}</span>\n            </button>\n          }\n          @if (presenter.endDateHtmlString()) {\n            <button\n              mat-menu-item\n              (click)=\"controller.anchor.setFixedAnchorPointToStartDateTime()\"\n              [disabled]=\"!presenter.endDateHtmlString()\"\n            >\n              <mat-icon class=\"material-symbols-outlined\">logout</mat-icon>\n              <span>{{ 'intervalPicker.startDate' | translate }}</span>\n            </button>\n          }\n          @if (presenter.isFixedAnchorDateSet()) {\n            <button\n              mat-menu-item\n              (click)=\"controller.anchor.resetAnchor()\"\n              [disabled]=\"!presenter.isFixedAnchorDateSet() || anchorReadOnly()\"\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          <input\n            #startDateInput\n            name=\"anchorDateInput\"\n            matInput\n            type=\"date\"\n            [ngModel]=\"presenter.anchorDateHtmlString()\"\n            (ngModelChange)=\"controller.form.setAnchorDateFromString($event)\"\n            [readonly]=\"anchorReadOnly()\"\n          />\n          @if (!anchorReadOnly()) {\n            <button mat-icon-button matSuffix [matMenuTriggerFor]=\"anchorMenu\" [disabled]=\"presenter.isAnchorMenuDisabled()\">\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]=\"anchorReadOnly()\"\n              [ngModel]=\"presenter.anchorTimeHtmlString()\"\n              (ngModelChange)=\"controller.form.setAnchorTimeFromString($event)\"\n            />\n            <button\n              mat-icon-button\n              matSuffix\n              (click)=\"controller.anchor.resetAnchorTime()\"\n              [disabled]=\"anchorReadOnly() || presenter.isAnchorTimeMidnight()\"\n            >\n              <mat-icon>close</mat-icon>\n            </button>\n          </mat-form-field>\n        }\n      </div>\n    }\n  </div>\n  <!-- result interval -->\n  <div class=\"layout-col place-center-center\">\n    <div class=\"date-interval mat-caption pt-xs\">\n      @if (presenter.startDateTimeAsJSDate()) {\n        {{ presenter.startDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n      } @else {\n        {{ 'intervalPicker.startDateNotSet' | translate }}\n      }\n      <span> - </span>\n      @if (presenter.endDateTimeAsJSDate()) {\n        {{ presenter.endDateTimeAsJSDate() | date: 'dd.MM. y, HH:mm:ss' }}\n      } @else {\n        {{ 'intervalPicker.endDateNotSet' | translate }}\n      }\n    </div>\n    <div>\n      <span class=\"mat-caption\">{{ presenter.deltaHumanReadable() || '&nbsp;' }}</span>\n    </div>\n  </div>\n  <div class=\"layout-row gap-lg\">\n    <button\n      mat-raised-button\n      color=\"primary\"\n      (click)=\"controller.clearInterval()\"\n      [disabled]=\"!presenter.isValidIntervalSet()\"\n    >\n      {{ 'intervalPicker.clear' | translate }}\n    </button>\n\n    @if (!this.autoEmitMode) {\n      <button color=\"primary\" mat-raised-button (click)=\"controller.manualEmit()\">\n        {{ 'actions.ok' | translate }}\n      </button>\n    }\n  </div>\n</div>\n"]}
|