@sbb-esta/lyne-elements-dev 4.9.0-dev.1774875578 → 4.9.0-dev.1774887348
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/calendar/calendar/calendar.component.js +1 -1
- package/calendar/calendar.js +1 -1
- package/{calendar.component-Dk8Xun-b.js → calendar.component-BRIW9HVu.js} +88 -186
- package/calendar.js +1 -1
- package/calendar.pure.js +1 -1
- package/core/datetime/date-adapter.js +13 -13
- package/core/datetime.js +4 -4
- package/core.js +47 -47
- package/custom-elements.json +1050 -939
- package/development/calendar/calendar/calendar.component.d.ts +1 -12
- package/development/calendar/calendar/calendar.component.d.ts.map +1 -1
- package/development/calendar/calendar/calendar.component.js +1 -1
- package/development/calendar/calendar.js +1 -1
- package/development/calendar.component-CsAl9EAg.js +1678 -0
- package/development/calendar.js +1 -1
- package/development/calendar.pure.js +1 -1
- package/development/core/datetime/date-adapter.d.ts +9 -2
- package/development/core/datetime/date-adapter.d.ts.map +1 -1
- package/development/core/datetime/date-adapter.js +11 -4
- package/development/core/datetime.js +2 -2
- package/development/core.js +2 -2
- package/package.json +2 -2
- package/development/calendar.component-CRvTYw1K.js +0 -2223
|
@@ -0,0 +1,1678 @@
|
|
|
1
|
+
import { __esDecorate, __runInitializers } from "tslib";
|
|
2
|
+
import { css, html, isServer, nothing } from "lit";
|
|
3
|
+
import { property, state } from "lit/decorators.js";
|
|
4
|
+
import { SbbElement } from "./core/base-elements.js";
|
|
5
|
+
import { forceType, handleDistinctChange, plainDate } from "./core/decorators.js";
|
|
6
|
+
import { boxSizingStyles } from "./core/styles.js";
|
|
7
|
+
import { SbbSecondaryButtonElement } from "./button.pure.js";
|
|
8
|
+
import { SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove } from "./core/controllers.js";
|
|
9
|
+
import { i18nCalendarDateSelection, i18nCalendarWeekNumber, i18nNextMonth, i18nNextYear, i18nNextYearRange, i18nPreviousMonth, i18nPreviousYear, i18nPreviousYearRange, i18nYearMonthSelection } from "./core/i18n.js";
|
|
10
|
+
import "./icon.js";
|
|
11
|
+
import { isArrowKeyOrPageKeysPressed } from "./core/a11y.js";
|
|
12
|
+
import { readConfig } from "./core/config.js";
|
|
13
|
+
import { DAYS_PER_ROW, MONDAY, MONTHS_PER_PAGE, MONTHS_PER_ROW, THURSDAY, TUESDAY, WEDNESDAY, YEARS_PER_PAGE, YEARS_PER_ROW, defaultDateAdapter } from "./core/datetime.js";
|
|
14
|
+
import { SbbCalendarDayElement } from "./calendar/calendar-day/calendar-day.component.js";
|
|
15
|
+
import { SbbCalendarMonthElement } from "./calendar/calendar-month/calendar-month.component.js";
|
|
16
|
+
import { SbbCalendarWeekdayElement } from "./calendar/calendar-weekday/calendar-weekday.component.js";
|
|
17
|
+
import { SbbCalendarWeeknumberElement } from "./calendar/calendar-weeknumber/calendar-weeknumber.component.js";
|
|
18
|
+
import { SbbCalendarYearElement } from "./calendar/calendar-year/calendar-year.component.js";
|
|
19
|
+
import "./screen-reader-only.js";
|
|
20
|
+
//#region src/elements/calendar/calendar/calendar.scss?lit&inline
|
|
21
|
+
var calendar_default = css`:host {
|
|
22
|
+
display: block;
|
|
23
|
+
width: max-content;
|
|
24
|
+
--sbb-calendar-cell-size: 2.75rem;
|
|
25
|
+
--sbb-calendar-cell-border-radius: calc(
|
|
26
|
+
var(--sbb-border-radius-4x) + var(--sbb-calendar-cell-border-width)
|
|
27
|
+
);
|
|
28
|
+
--sbb-calendar-cell-transition-duration: var(
|
|
29
|
+
--sbb-disable-animation-duration,
|
|
30
|
+
var(--sbb-animation-duration-2x)
|
|
31
|
+
);
|
|
32
|
+
}
|
|
33
|
+
@media (min-width: calc(37.5rem)) {
|
|
34
|
+
:host {
|
|
35
|
+
--sbb-calendar-control-view-change-height: 3rem;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
:host(:is(:state(enhanced),[state--enhanced])) {
|
|
40
|
+
--sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
.sbb-calendar__wrapper {
|
|
44
|
+
width: 100%;
|
|
45
|
+
display: block;
|
|
46
|
+
transition-duration: var(--sbb-calendar-cell-transition-duration);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
.sbb-calendar__controls {
|
|
50
|
+
width: 100%;
|
|
51
|
+
display: inline-flex;
|
|
52
|
+
align-items: center;
|
|
53
|
+
gap: var(--sbb-calendar-control-gap);
|
|
54
|
+
margin-block-end: var(--sbb-calendar-control-margin-block-end);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
.sbb-calendar__controls-month {
|
|
58
|
+
width: 100%;
|
|
59
|
+
display: flex;
|
|
60
|
+
gap: var(--sbb-calendar-tables-gap);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
#sbb-calendar__controls-previous,
|
|
64
|
+
#sbb-calendar__controls-next {
|
|
65
|
+
-webkit-tap-highlight-color: transparent;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
.sbb-calendar__controls-change-date {
|
|
69
|
+
appearance: none;
|
|
70
|
+
box-sizing: border-box;
|
|
71
|
+
margin: 0;
|
|
72
|
+
outline: none;
|
|
73
|
+
border: none;
|
|
74
|
+
border-radius: 0;
|
|
75
|
+
padding: 0;
|
|
76
|
+
background: none;
|
|
77
|
+
-webkit-tap-highlight-color: transparent;
|
|
78
|
+
color: inherit;
|
|
79
|
+
display: flex;
|
|
80
|
+
align-items: center;
|
|
81
|
+
margin: auto;
|
|
82
|
+
height: var(--sbb-calendar-control-view-change-height);
|
|
83
|
+
font-size: var(--sbb-text-font-size-s);
|
|
84
|
+
letter-spacing: var(--sbb-typo-letter-spacing-text);
|
|
85
|
+
text-transform: capitalize;
|
|
86
|
+
cursor: var(--sbb-cursor-pointer);
|
|
87
|
+
padding-inline: var(--sbb-calendar-control-view-change-padding-inline);
|
|
88
|
+
border-radius: var(--sbb-border-radius-infinity);
|
|
89
|
+
background-color: var(--sbb-calendar-control-view-change-background);
|
|
90
|
+
color: var(--sbb-calendar-control-view-change-color);
|
|
91
|
+
transition-duration: var(--sbb-calendar-cell-transition-duration);
|
|
92
|
+
transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);
|
|
93
|
+
transition-property: background-color, padding-block-end;
|
|
94
|
+
}
|
|
95
|
+
.sbb-calendar__controls-change-date:disabled {
|
|
96
|
+
--sbb-calendar-control-view-change-background: var(--sbb-color-milk);
|
|
97
|
+
--sbb-calendar-control-view-change-background: light-dark(
|
|
98
|
+
var(--sbb-color-milk),
|
|
99
|
+
var(--sbb-color-anthracite)
|
|
100
|
+
);
|
|
101
|
+
--sbb-calendar-control-view-change-color: var(--sbb-color-granite);
|
|
102
|
+
--sbb-calendar-control-view-change-color: light-dark(
|
|
103
|
+
var(--sbb-color-granite),
|
|
104
|
+
var(--sbb-color-aluminium)
|
|
105
|
+
);
|
|
106
|
+
cursor: unset;
|
|
107
|
+
}
|
|
108
|
+
.sbb-calendar__controls-change-date:focus-visible {
|
|
109
|
+
outline-offset: var(--sbb-focus-outline-offset);
|
|
110
|
+
outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);
|
|
111
|
+
outline-offset: var(--sbb-spacing-fixed-1x);
|
|
112
|
+
}
|
|
113
|
+
@media (any-hover: hover) {
|
|
114
|
+
.sbb-calendar__controls-change-date:not(:active, :disabled):hover {
|
|
115
|
+
padding-block-end: var(--sbb-calendar-cell-hover-shift);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
.sbb-calendar__controls-change-date:not(:disabled):active {
|
|
119
|
+
--sbb-calendar-control-view-change-background: var(--sbb-background-color-3);
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
.sbb-calendar__table-month-view,
|
|
123
|
+
.sbb-calendar__table-year-view {
|
|
124
|
+
--sbb-calendar-table-column-spaces: 6;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
.sbb-calendar__table-overflow-break {
|
|
128
|
+
contain: layout;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
.sbb-calendar__table-container {
|
|
132
|
+
display: flex;
|
|
133
|
+
gap: var(--sbb-calendar-tables-gap);
|
|
134
|
+
margin-inline: var(--sbb-calendar-margin);
|
|
135
|
+
--sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));
|
|
136
|
+
--sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));
|
|
137
|
+
--sbb-calendar-start-offset: min(
|
|
138
|
+
0px,
|
|
139
|
+
-1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))
|
|
140
|
+
);
|
|
141
|
+
--sbb-calendar-margin: var(--sbb-calendar-start-offset);
|
|
142
|
+
}
|
|
143
|
+
:host(:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
144
|
+
--sbb-calendar-min-width: calc(
|
|
145
|
+
2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
146
|
+
);
|
|
147
|
+
--sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));
|
|
148
|
+
}
|
|
149
|
+
:host([orientation=horizontal][week-numbers]) .sbb-calendar__table-container {
|
|
150
|
+
--sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));
|
|
151
|
+
}
|
|
152
|
+
:host([orientation=horizontal][week-numbers]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
153
|
+
--sbb-calendar-min-width: calc(
|
|
154
|
+
2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
155
|
+
);
|
|
156
|
+
}
|
|
157
|
+
:host([orientation=vertical]) .sbb-calendar__table-container {
|
|
158
|
+
min-width: var(--sbb-calendar-min-width);
|
|
159
|
+
--sbb-calendar-start-offset: 0;
|
|
160
|
+
}
|
|
161
|
+
:host([orientation=vertical]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {
|
|
162
|
+
--sbb-calendar-min-width: calc(
|
|
163
|
+
13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)
|
|
164
|
+
);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
.sbb-calendar__table {
|
|
168
|
+
width: 100%;
|
|
169
|
+
border-collapse: collapse;
|
|
170
|
+
height: max-content;
|
|
171
|
+
animation-name: show;
|
|
172
|
+
animation-duration: var(--sbb-calendar-table-animation-duration);
|
|
173
|
+
}
|
|
174
|
+
.sbb-calendar__table.sbb-calendar__table-hide {
|
|
175
|
+
--sbb-calendar-cell-transition-duration: 0ms;
|
|
176
|
+
animation-name: hide;
|
|
177
|
+
animation-duration: var(--sbb-calendar-table-animation-duration);
|
|
178
|
+
}
|
|
179
|
+
:host(:not(:is(:state(wide),[state--wide]))) .sbb-calendar__table {
|
|
180
|
+
min-width: 100%;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
.sbb-calendar__table-header {
|
|
184
|
+
text-align: center;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
.sbb-calendar__table-body {
|
|
188
|
+
text-align: center;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
.sbb-calendar__table-header-cell,
|
|
192
|
+
.sbb-calendar__table-header-cell-vertical {
|
|
193
|
+
width: var(--sbb-calendar-cell-size);
|
|
194
|
+
color: var(--sbb-calendar-header-color);
|
|
195
|
+
padding: 0;
|
|
196
|
+
font-size: var(--sbb-text-font-size-xs);
|
|
197
|
+
font-weight: normal;
|
|
198
|
+
letter-spacing: var(--sbb-typo-letter-spacing-text);
|
|
199
|
+
}
|
|
200
|
+
:host(:not([multiple])) .sbb-calendar__table-header-cell,
|
|
201
|
+
:host(:not([multiple])) .sbb-calendar__table-header-cell-vertical {
|
|
202
|
+
height: var(--sbb-calendar-cell-size);
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
.sbb-calendar__table-header-cell-vertical {
|
|
206
|
+
padding-block-end: var(--sbb-calendar-cell-padding-block-end);
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
.sbb-calendar__table-data {
|
|
210
|
+
position: relative;
|
|
211
|
+
padding: 0;
|
|
212
|
+
text-align: center;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
.sbb-calendar__day-cell {
|
|
216
|
+
padding-block-end: var(--sbb-calendar-cell-padding-block-end);
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
:host(:not(:is(:state(enhanced),[state--enhanced]))) sbb-calendar-day {
|
|
220
|
+
--sbb-calendar-cell-justify-content: center;
|
|
221
|
+
--sbb-calendar-day-height: 2.75rem;
|
|
222
|
+
--sbb-calendar-day-extra-display: none;
|
|
223
|
+
--sbb-calendar-day-value-height: unset;
|
|
224
|
+
--sbb-calendar-day-crossed-out-top: 50%;
|
|
225
|
+
--sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
@keyframes show {
|
|
229
|
+
from {
|
|
230
|
+
opacity: 0;
|
|
231
|
+
transform: translateY(var(--sbb-calendar-table-animation-shift));
|
|
232
|
+
}
|
|
233
|
+
to {
|
|
234
|
+
opacity: 1;
|
|
235
|
+
transform: translateY(0%);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
@keyframes hide {
|
|
239
|
+
from {
|
|
240
|
+
opacity: 1;
|
|
241
|
+
transform: translateY(0%);
|
|
242
|
+
}
|
|
243
|
+
to {
|
|
244
|
+
opacity: 0;
|
|
245
|
+
transform: translateY(var(--sbb-calendar-table-animation-shift));
|
|
246
|
+
}
|
|
247
|
+
}`;
|
|
248
|
+
//#endregion
|
|
249
|
+
//#region src/elements/calendar/calendar/calendar.component.ts
|
|
250
|
+
SbbSecondaryButtonElement.define();
|
|
251
|
+
SbbCalendarDayElement.define();
|
|
252
|
+
SbbCalendarMonthElement.define();
|
|
253
|
+
SbbCalendarYearElement.define();
|
|
254
|
+
SbbCalendarWeekdayElement.define();
|
|
255
|
+
SbbCalendarWeeknumberElement.define();
|
|
256
|
+
var SbbMonthChangeEvent = class extends Event {
|
|
257
|
+
get range() {
|
|
258
|
+
return this._range;
|
|
259
|
+
}
|
|
260
|
+
constructor(range) {
|
|
261
|
+
super("monthchange", {
|
|
262
|
+
bubbles: true,
|
|
263
|
+
composed: true
|
|
264
|
+
});
|
|
265
|
+
this._range = Object.freeze(range || []);
|
|
266
|
+
}
|
|
267
|
+
};
|
|
268
|
+
/**
|
|
269
|
+
* It displays a calendar which allows choosing a date.
|
|
270
|
+
*
|
|
271
|
+
* @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.
|
|
272
|
+
*/
|
|
273
|
+
var SbbCalendarElement = (() => {
|
|
274
|
+
let _classSuper = SbbElement;
|
|
275
|
+
let _instanceExtraInitializers = [];
|
|
276
|
+
let _wide_decorators;
|
|
277
|
+
let _wide_initializers = [];
|
|
278
|
+
let _wide_extraInitializers = [];
|
|
279
|
+
let _view_decorators;
|
|
280
|
+
let _view_initializers = [];
|
|
281
|
+
let _view_extraInitializers = [];
|
|
282
|
+
let _min_decorators;
|
|
283
|
+
let _min_initializers = [];
|
|
284
|
+
let _min_extraInitializers = [];
|
|
285
|
+
let _max_decorators;
|
|
286
|
+
let _max_initializers = [];
|
|
287
|
+
let _max_extraInitializers = [];
|
|
288
|
+
let _multiple_decorators;
|
|
289
|
+
let _multiple_initializers = [];
|
|
290
|
+
let _multiple_extraInitializers = [];
|
|
291
|
+
let _set_selected_decorators;
|
|
292
|
+
let __selected_decorators;
|
|
293
|
+
let __selected_initializers = [];
|
|
294
|
+
let __selected_extraInitializers = [];
|
|
295
|
+
let _dateFilter_decorators;
|
|
296
|
+
let _dateFilter_initializers = [];
|
|
297
|
+
let _dateFilter_extraInitializers = [];
|
|
298
|
+
let _orientation_decorators;
|
|
299
|
+
let _orientation_initializers = [];
|
|
300
|
+
let _orientation_extraInitializers = [];
|
|
301
|
+
let _weekNumbers_decorators;
|
|
302
|
+
let _weekNumbers_initializers = [];
|
|
303
|
+
let _weekNumbers_extraInitializers = [];
|
|
304
|
+
let __activeDate_decorators;
|
|
305
|
+
let __activeDate_initializers = [];
|
|
306
|
+
let __activeDate_extraInitializers = [];
|
|
307
|
+
let _set__wide_decorators;
|
|
308
|
+
let __calendarView_decorators;
|
|
309
|
+
let __calendarView_initializers = [];
|
|
310
|
+
let __calendarView_extraInitializers = [];
|
|
311
|
+
let __initialized_decorators;
|
|
312
|
+
let __initialized_initializers = [];
|
|
313
|
+
let __initialized_extraInitializers = [];
|
|
314
|
+
return class SbbCalendarElement extends _classSuper {
|
|
315
|
+
static {
|
|
316
|
+
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
|
|
317
|
+
_wide_decorators = [forceType(), property({ type: Boolean })];
|
|
318
|
+
_view_decorators = [property()];
|
|
319
|
+
_min_decorators = [plainDate(), property()];
|
|
320
|
+
_max_decorators = [plainDate(), property()];
|
|
321
|
+
_multiple_decorators = [
|
|
322
|
+
forceType(),
|
|
323
|
+
handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)),
|
|
324
|
+
property({ type: Boolean })
|
|
325
|
+
];
|
|
326
|
+
_set_selected_decorators = [property()];
|
|
327
|
+
__selected_decorators = [state()];
|
|
328
|
+
_dateFilter_decorators = [property({ attribute: "date-filter" })];
|
|
329
|
+
_orientation_decorators = [property({ reflect: true })];
|
|
330
|
+
_weekNumbers_decorators = [forceType(), property({
|
|
331
|
+
attribute: "week-numbers",
|
|
332
|
+
type: Boolean
|
|
333
|
+
})];
|
|
334
|
+
__activeDate_decorators = [state()];
|
|
335
|
+
_set__wide_decorators = [state()];
|
|
336
|
+
__calendarView_decorators = [state()];
|
|
337
|
+
__initialized_decorators = [state()];
|
|
338
|
+
__esDecorate(this, null, _wide_decorators, {
|
|
339
|
+
kind: "accessor",
|
|
340
|
+
name: "wide",
|
|
341
|
+
static: false,
|
|
342
|
+
private: false,
|
|
343
|
+
access: {
|
|
344
|
+
has: (obj) => "wide" in obj,
|
|
345
|
+
get: (obj) => obj.wide,
|
|
346
|
+
set: (obj, value) => {
|
|
347
|
+
obj.wide = value;
|
|
348
|
+
}
|
|
349
|
+
},
|
|
350
|
+
metadata: _metadata
|
|
351
|
+
}, _wide_initializers, _wide_extraInitializers);
|
|
352
|
+
__esDecorate(this, null, _view_decorators, {
|
|
353
|
+
kind: "accessor",
|
|
354
|
+
name: "view",
|
|
355
|
+
static: false,
|
|
356
|
+
private: false,
|
|
357
|
+
access: {
|
|
358
|
+
has: (obj) => "view" in obj,
|
|
359
|
+
get: (obj) => obj.view,
|
|
360
|
+
set: (obj, value) => {
|
|
361
|
+
obj.view = value;
|
|
362
|
+
}
|
|
363
|
+
},
|
|
364
|
+
metadata: _metadata
|
|
365
|
+
}, _view_initializers, _view_extraInitializers);
|
|
366
|
+
__esDecorate(this, null, _min_decorators, {
|
|
367
|
+
kind: "accessor",
|
|
368
|
+
name: "min",
|
|
369
|
+
static: false,
|
|
370
|
+
private: false,
|
|
371
|
+
access: {
|
|
372
|
+
has: (obj) => "min" in obj,
|
|
373
|
+
get: (obj) => obj.min,
|
|
374
|
+
set: (obj, value) => {
|
|
375
|
+
obj.min = value;
|
|
376
|
+
}
|
|
377
|
+
},
|
|
378
|
+
metadata: _metadata
|
|
379
|
+
}, _min_initializers, _min_extraInitializers);
|
|
380
|
+
__esDecorate(this, null, _max_decorators, {
|
|
381
|
+
kind: "accessor",
|
|
382
|
+
name: "max",
|
|
383
|
+
static: false,
|
|
384
|
+
private: false,
|
|
385
|
+
access: {
|
|
386
|
+
has: (obj) => "max" in obj,
|
|
387
|
+
get: (obj) => obj.max,
|
|
388
|
+
set: (obj, value) => {
|
|
389
|
+
obj.max = value;
|
|
390
|
+
}
|
|
391
|
+
},
|
|
392
|
+
metadata: _metadata
|
|
393
|
+
}, _max_initializers, _max_extraInitializers);
|
|
394
|
+
__esDecorate(this, null, _multiple_decorators, {
|
|
395
|
+
kind: "accessor",
|
|
396
|
+
name: "multiple",
|
|
397
|
+
static: false,
|
|
398
|
+
private: false,
|
|
399
|
+
access: {
|
|
400
|
+
has: (obj) => "multiple" in obj,
|
|
401
|
+
get: (obj) => obj.multiple,
|
|
402
|
+
set: (obj, value) => {
|
|
403
|
+
obj.multiple = value;
|
|
404
|
+
}
|
|
405
|
+
},
|
|
406
|
+
metadata: _metadata
|
|
407
|
+
}, _multiple_initializers, _multiple_extraInitializers);
|
|
408
|
+
__esDecorate(this, null, _set_selected_decorators, {
|
|
409
|
+
kind: "setter",
|
|
410
|
+
name: "selected",
|
|
411
|
+
static: false,
|
|
412
|
+
private: false,
|
|
413
|
+
access: {
|
|
414
|
+
has: (obj) => "selected" in obj,
|
|
415
|
+
set: (obj, value) => {
|
|
416
|
+
obj.selected = value;
|
|
417
|
+
}
|
|
418
|
+
},
|
|
419
|
+
metadata: _metadata
|
|
420
|
+
}, null, _instanceExtraInitializers);
|
|
421
|
+
__esDecorate(this, null, __selected_decorators, {
|
|
422
|
+
kind: "accessor",
|
|
423
|
+
name: "_selected",
|
|
424
|
+
static: false,
|
|
425
|
+
private: false,
|
|
426
|
+
access: {
|
|
427
|
+
has: (obj) => "_selected" in obj,
|
|
428
|
+
get: (obj) => obj._selected,
|
|
429
|
+
set: (obj, value) => {
|
|
430
|
+
obj._selected = value;
|
|
431
|
+
}
|
|
432
|
+
},
|
|
433
|
+
metadata: _metadata
|
|
434
|
+
}, __selected_initializers, __selected_extraInitializers);
|
|
435
|
+
__esDecorate(this, null, _dateFilter_decorators, {
|
|
436
|
+
kind: "accessor",
|
|
437
|
+
name: "dateFilter",
|
|
438
|
+
static: false,
|
|
439
|
+
private: false,
|
|
440
|
+
access: {
|
|
441
|
+
has: (obj) => "dateFilter" in obj,
|
|
442
|
+
get: (obj) => obj.dateFilter,
|
|
443
|
+
set: (obj, value) => {
|
|
444
|
+
obj.dateFilter = value;
|
|
445
|
+
}
|
|
446
|
+
},
|
|
447
|
+
metadata: _metadata
|
|
448
|
+
}, _dateFilter_initializers, _dateFilter_extraInitializers);
|
|
449
|
+
__esDecorate(this, null, _orientation_decorators, {
|
|
450
|
+
kind: "accessor",
|
|
451
|
+
name: "orientation",
|
|
452
|
+
static: false,
|
|
453
|
+
private: false,
|
|
454
|
+
access: {
|
|
455
|
+
has: (obj) => "orientation" in obj,
|
|
456
|
+
get: (obj) => obj.orientation,
|
|
457
|
+
set: (obj, value) => {
|
|
458
|
+
obj.orientation = value;
|
|
459
|
+
}
|
|
460
|
+
},
|
|
461
|
+
metadata: _metadata
|
|
462
|
+
}, _orientation_initializers, _orientation_extraInitializers);
|
|
463
|
+
__esDecorate(this, null, _weekNumbers_decorators, {
|
|
464
|
+
kind: "accessor",
|
|
465
|
+
name: "weekNumbers",
|
|
466
|
+
static: false,
|
|
467
|
+
private: false,
|
|
468
|
+
access: {
|
|
469
|
+
has: (obj) => "weekNumbers" in obj,
|
|
470
|
+
get: (obj) => obj.weekNumbers,
|
|
471
|
+
set: (obj, value) => {
|
|
472
|
+
obj.weekNumbers = value;
|
|
473
|
+
}
|
|
474
|
+
},
|
|
475
|
+
metadata: _metadata
|
|
476
|
+
}, _weekNumbers_initializers, _weekNumbers_extraInitializers);
|
|
477
|
+
__esDecorate(this, null, __activeDate_decorators, {
|
|
478
|
+
kind: "accessor",
|
|
479
|
+
name: "_activeDate",
|
|
480
|
+
static: false,
|
|
481
|
+
private: false,
|
|
482
|
+
access: {
|
|
483
|
+
has: (obj) => "_activeDate" in obj,
|
|
484
|
+
get: (obj) => obj._activeDate,
|
|
485
|
+
set: (obj, value) => {
|
|
486
|
+
obj._activeDate = value;
|
|
487
|
+
}
|
|
488
|
+
},
|
|
489
|
+
metadata: _metadata
|
|
490
|
+
}, __activeDate_initializers, __activeDate_extraInitializers);
|
|
491
|
+
__esDecorate(this, null, _set__wide_decorators, {
|
|
492
|
+
kind: "setter",
|
|
493
|
+
name: "_wide",
|
|
494
|
+
static: false,
|
|
495
|
+
private: false,
|
|
496
|
+
access: {
|
|
497
|
+
has: (obj) => "_wide" in obj,
|
|
498
|
+
set: (obj, value) => {
|
|
499
|
+
obj._wide = value;
|
|
500
|
+
}
|
|
501
|
+
},
|
|
502
|
+
metadata: _metadata
|
|
503
|
+
}, null, _instanceExtraInitializers);
|
|
504
|
+
__esDecorate(this, null, __calendarView_decorators, {
|
|
505
|
+
kind: "accessor",
|
|
506
|
+
name: "_calendarView",
|
|
507
|
+
static: false,
|
|
508
|
+
private: false,
|
|
509
|
+
access: {
|
|
510
|
+
has: (obj) => "_calendarView" in obj,
|
|
511
|
+
get: (obj) => obj._calendarView,
|
|
512
|
+
set: (obj, value) => {
|
|
513
|
+
obj._calendarView = value;
|
|
514
|
+
}
|
|
515
|
+
},
|
|
516
|
+
metadata: _metadata
|
|
517
|
+
}, __calendarView_initializers, __calendarView_extraInitializers);
|
|
518
|
+
__esDecorate(this, null, __initialized_decorators, {
|
|
519
|
+
kind: "accessor",
|
|
520
|
+
name: "_initialized",
|
|
521
|
+
static: false,
|
|
522
|
+
private: false,
|
|
523
|
+
access: {
|
|
524
|
+
has: (obj) => "_initialized" in obj,
|
|
525
|
+
get: (obj) => obj._initialized,
|
|
526
|
+
set: (obj, value) => {
|
|
527
|
+
obj._initialized = value;
|
|
528
|
+
}
|
|
529
|
+
},
|
|
530
|
+
metadata: _metadata
|
|
531
|
+
}, __initialized_initializers, __initialized_extraInitializers);
|
|
532
|
+
if (_metadata) Object.defineProperty(this, Symbol.metadata, {
|
|
533
|
+
enumerable: true,
|
|
534
|
+
configurable: true,
|
|
535
|
+
writable: true,
|
|
536
|
+
value: _metadata
|
|
537
|
+
});
|
|
538
|
+
}
|
|
539
|
+
static {
|
|
540
|
+
this.elementName = "sbb-calendar";
|
|
541
|
+
}
|
|
542
|
+
static {
|
|
543
|
+
this.styles = [boxSizingStyles, calendar_default];
|
|
544
|
+
}
|
|
545
|
+
static {
|
|
546
|
+
this.events = {
|
|
547
|
+
dateselected: "dateselected",
|
|
548
|
+
monthchange: "monthchange"
|
|
549
|
+
};
|
|
550
|
+
}
|
|
551
|
+
#wide_accessor_storage;
|
|
552
|
+
/** If set to true, two months are displayed */
|
|
553
|
+
get wide() {
|
|
554
|
+
return this.#wide_accessor_storage;
|
|
555
|
+
}
|
|
556
|
+
set wide(value) {
|
|
557
|
+
this.#wide_accessor_storage = value;
|
|
558
|
+
}
|
|
559
|
+
#view_accessor_storage;
|
|
560
|
+
/** The initial view of the calendar which should be displayed on opening. */
|
|
561
|
+
get view() {
|
|
562
|
+
return this.#view_accessor_storage;
|
|
563
|
+
}
|
|
564
|
+
set view(value) {
|
|
565
|
+
this.#view_accessor_storage = value;
|
|
566
|
+
}
|
|
567
|
+
#min_accessor_storage;
|
|
568
|
+
/**
|
|
569
|
+
* The minimum valid date. Accepts a date object or null.
|
|
570
|
+
* Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
|
|
571
|
+
*/
|
|
572
|
+
get min() {
|
|
573
|
+
return this.#min_accessor_storage;
|
|
574
|
+
}
|
|
575
|
+
set min(value) {
|
|
576
|
+
this.#min_accessor_storage = value;
|
|
577
|
+
}
|
|
578
|
+
#max_accessor_storage;
|
|
579
|
+
/**
|
|
580
|
+
* The maximum valid date. Accepts a date object or null.
|
|
581
|
+
* Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
|
|
582
|
+
*/
|
|
583
|
+
get max() {
|
|
584
|
+
return this.#max_accessor_storage;
|
|
585
|
+
}
|
|
586
|
+
set max(value) {
|
|
587
|
+
this.#max_accessor_storage = value;
|
|
588
|
+
}
|
|
589
|
+
#multiple_accessor_storage;
|
|
590
|
+
/** Whether the calendar allows for multiple date selection. */
|
|
591
|
+
get multiple() {
|
|
592
|
+
return this.#multiple_accessor_storage;
|
|
593
|
+
}
|
|
594
|
+
set multiple(value) {
|
|
595
|
+
this.#multiple_accessor_storage = value;
|
|
596
|
+
}
|
|
597
|
+
/**
|
|
598
|
+
* The selected date: accepts a date object, or, if `multiple`, an array of dates.
|
|
599
|
+
*/
|
|
600
|
+
set selected(value) {
|
|
601
|
+
if (Array.isArray(value)) this._selected = value.map((dateLike) => this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(dateLike))).filter((date) => date !== null).filter((date) => !this._isDayInRange(this._dateAdapter.toIso8601(date)) || this._dateFilter(date));
|
|
602
|
+
else {
|
|
603
|
+
const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
|
|
604
|
+
if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) this._selected = selectedDate;
|
|
605
|
+
else this._selected = null;
|
|
606
|
+
}
|
|
607
|
+
}
|
|
608
|
+
get selected() {
|
|
609
|
+
return this._selected;
|
|
610
|
+
}
|
|
611
|
+
#_selected_accessor_storage;
|
|
612
|
+
get _selected() {
|
|
613
|
+
return this.#_selected_accessor_storage;
|
|
614
|
+
}
|
|
615
|
+
set _selected(value) {
|
|
616
|
+
this.#_selected_accessor_storage = value;
|
|
617
|
+
}
|
|
618
|
+
#dateFilter_accessor_storage;
|
|
619
|
+
/** A function used to filter out dates. */
|
|
620
|
+
get dateFilter() {
|
|
621
|
+
return this.#dateFilter_accessor_storage;
|
|
622
|
+
}
|
|
623
|
+
set dateFilter(value) {
|
|
624
|
+
this.#dateFilter_accessor_storage = value;
|
|
625
|
+
}
|
|
626
|
+
#orientation_accessor_storage;
|
|
627
|
+
/** The orientation of days in the calendar. */
|
|
628
|
+
get orientation() {
|
|
629
|
+
return this.#orientation_accessor_storage;
|
|
630
|
+
}
|
|
631
|
+
set orientation(value) {
|
|
632
|
+
this.#orientation_accessor_storage = value;
|
|
633
|
+
}
|
|
634
|
+
#weekNumbers_accessor_storage;
|
|
635
|
+
/** Whether it has to display the week numbers in addition to week days. */
|
|
636
|
+
get weekNumbers() {
|
|
637
|
+
return this.#weekNumbers_accessor_storage;
|
|
638
|
+
}
|
|
639
|
+
set weekNumbers(value) {
|
|
640
|
+
this.#weekNumbers_accessor_storage = value;
|
|
641
|
+
}
|
|
642
|
+
#_activeDate_accessor_storage;
|
|
643
|
+
/** The currently active date. */
|
|
644
|
+
get _activeDate() {
|
|
645
|
+
return this.#_activeDate_accessor_storage;
|
|
646
|
+
}
|
|
647
|
+
set _activeDate(value) {
|
|
648
|
+
this.#_activeDate_accessor_storage = value;
|
|
649
|
+
}
|
|
650
|
+
/** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
|
|
651
|
+
set _wide(wide) {
|
|
652
|
+
this.toggleState("wide", wide);
|
|
653
|
+
this._wideInternal = wide;
|
|
654
|
+
}
|
|
655
|
+
get _wide() {
|
|
656
|
+
return this._wideInternal;
|
|
657
|
+
}
|
|
658
|
+
#_calendarView_accessor_storage;
|
|
659
|
+
get _calendarView() {
|
|
660
|
+
return this.#_calendarView_accessor_storage;
|
|
661
|
+
}
|
|
662
|
+
set _calendarView(value) {
|
|
663
|
+
this.#_calendarView_accessor_storage = value;
|
|
664
|
+
}
|
|
665
|
+
/** A list of calendar's cells corresponding to days, months or years depending on the view. */
|
|
666
|
+
get _cells() {
|
|
667
|
+
return Array.from((this._calendarView === "day" ? Array.from(this.shadowRoot.querySelectorAll("slot")).flatMap((e) => e.assignedElements({ flatten: true })) : this.shadowRoot?.querySelectorAll(`sbb-calendar-${this._calendarView}`)) ?? []);
|
|
668
|
+
}
|
|
669
|
+
#_initialized_accessor_storage;
|
|
670
|
+
get _initialized() {
|
|
671
|
+
return this.#_initialized_accessor_storage;
|
|
672
|
+
}
|
|
673
|
+
set _initialized(value) {
|
|
674
|
+
this.#_initialized_accessor_storage = value;
|
|
675
|
+
}
|
|
676
|
+
constructor() {
|
|
677
|
+
super();
|
|
678
|
+
this.#wide_accessor_storage = (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false));
|
|
679
|
+
this.#view_accessor_storage = (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day"));
|
|
680
|
+
this.#min_accessor_storage = (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null));
|
|
681
|
+
this.#max_accessor_storage = (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null));
|
|
682
|
+
this.#multiple_accessor_storage = (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false));
|
|
683
|
+
this.#_selected_accessor_storage = (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null));
|
|
684
|
+
this.#dateFilter_accessor_storage = (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null));
|
|
685
|
+
this.#orientation_accessor_storage = (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal"));
|
|
686
|
+
this.#weekNumbers_accessor_storage = (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false));
|
|
687
|
+
this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
|
|
688
|
+
this.#_activeDate_accessor_storage = __runInitializers(this, __activeDate_initializers, this._dateAdapter.today());
|
|
689
|
+
this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
|
|
690
|
+
this.#_calendarView_accessor_storage = __runInitializers(this, __calendarView_initializers, "day");
|
|
691
|
+
this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
|
|
692
|
+
/** Information about the rendered day view; used in keyboard navigation. */
|
|
693
|
+
this._keyboardNavigationDayViewParameters = {
|
|
694
|
+
firstDayInView: null,
|
|
695
|
+
lastDayInView: null,
|
|
696
|
+
firstMonthOffset: 0,
|
|
697
|
+
firstMonthLength: 0,
|
|
698
|
+
secondMonthOffset: 0
|
|
699
|
+
};
|
|
700
|
+
/** Grid of calendar cells representing the dates of the month. */
|
|
701
|
+
this._weeks = [];
|
|
702
|
+
/** An array containing all the month names in the current language. */
|
|
703
|
+
this._monthNames = this._dateAdapter.getMonthNames("long");
|
|
704
|
+
this._enhancedVariant = false;
|
|
705
|
+
/** Whether the focus should be reset on focusCell. */
|
|
706
|
+
this._resetFocus = false;
|
|
707
|
+
/** Whether an element inside the calendar is currently focused. */
|
|
708
|
+
this._containingFocus = false;
|
|
709
|
+
this.#_initialized_accessor_storage = __runInitializers(this, __initialized_initializers, false);
|
|
710
|
+
this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
|
|
711
|
+
this._monthNames = this._dateAdapter.getMonthNames("long");
|
|
712
|
+
this._createMonthRows();
|
|
713
|
+
}));
|
|
714
|
+
this._mediaMatcher = new SbbMediaMatcherController(this, { [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init() });
|
|
715
|
+
this._onSlotChange = () => {
|
|
716
|
+
this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
|
|
717
|
+
this.toggleState("enhanced", this._enhancedVariant);
|
|
718
|
+
this._setTabIndex();
|
|
719
|
+
};
|
|
720
|
+
this._createMonthRows();
|
|
721
|
+
this._setWeekdays();
|
|
722
|
+
this.addEventListener("focusin", () => this._containingFocus = true);
|
|
723
|
+
this.addEventListener("focusout", () => this._containingFocus = false);
|
|
724
|
+
this.addEventListener("click", (e) => {
|
|
725
|
+
const day = e.target.closest("sbb-calendar-day");
|
|
726
|
+
if (day) this._selectDate(day.value);
|
|
727
|
+
});
|
|
728
|
+
this.addEventListener("keydown", (e) => {
|
|
729
|
+
if (e.target.localName === "sbb-calendar-day") this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
|
|
730
|
+
});
|
|
731
|
+
}
|
|
732
|
+
_dateFilter(date) {
|
|
733
|
+
return this.dateFilter?.(date) ?? true;
|
|
734
|
+
}
|
|
735
|
+
/** Resets the active month according to the new state of the calendar. */
|
|
736
|
+
resetPosition() {
|
|
737
|
+
this._resetCalendarView();
|
|
738
|
+
this._init();
|
|
739
|
+
}
|
|
740
|
+
connectedCallback() {
|
|
741
|
+
super.connectedCallback();
|
|
742
|
+
this.resetPosition();
|
|
743
|
+
this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
|
|
744
|
+
}
|
|
745
|
+
disconnectedCallback() {
|
|
746
|
+
super.disconnectedCallback();
|
|
747
|
+
this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
|
|
748
|
+
}
|
|
749
|
+
/** @internal */
|
|
750
|
+
focus() {
|
|
751
|
+
this._resetFocus = true;
|
|
752
|
+
this._focusCell();
|
|
753
|
+
}
|
|
754
|
+
willUpdate(changedProperties) {
|
|
755
|
+
super.willUpdate(changedProperties);
|
|
756
|
+
if (!this._initialized) return;
|
|
757
|
+
if (changedProperties.has("wide") || changedProperties.has("orientation")) this.resetPosition();
|
|
758
|
+
if (changedProperties.has("view")) {
|
|
759
|
+
this._setChosenYear();
|
|
760
|
+
this._chosenMonth = void 0;
|
|
761
|
+
this._nextCalendarView = this._calendarView = this.view;
|
|
762
|
+
}
|
|
763
|
+
}
|
|
764
|
+
updated(changedProperties) {
|
|
765
|
+
super.updated(changedProperties);
|
|
766
|
+
this._setTabIndex();
|
|
767
|
+
this._focusCell();
|
|
768
|
+
}
|
|
769
|
+
/**
|
|
770
|
+
* The `_selected` state should be adapted when the `multiple` property changes:
|
|
771
|
+
* - if it changes to true, the '_selected' is set to an array;
|
|
772
|
+
* - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
|
|
773
|
+
*/
|
|
774
|
+
_onMultipleChanged(isMultiple) {
|
|
775
|
+
if (isMultiple && !Array.isArray(this._selected)) this._selected = this._selected ? [this._selected] : [];
|
|
776
|
+
if (!isMultiple && Array.isArray(this._selected)) this._selected = this._selected.length ? this._selected[0] : null;
|
|
777
|
+
}
|
|
778
|
+
/** Initializes the component. */
|
|
779
|
+
_init(activeDate) {
|
|
780
|
+
if (isServer) return;
|
|
781
|
+
else if (this.hydrationRequired) {
|
|
782
|
+
this.hydrationComplete.then(() => this._init());
|
|
783
|
+
return;
|
|
784
|
+
}
|
|
785
|
+
if (activeDate) this._assignActiveDate(activeDate);
|
|
786
|
+
this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
|
|
787
|
+
this._weeks = this._createWeekRows(this._activeDate);
|
|
788
|
+
this._years = this._createYearRows();
|
|
789
|
+
this._weekNumbers = this._weeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
|
|
790
|
+
this._nextMonthWeeks = [[]];
|
|
791
|
+
this._nextMonthYears = [[]];
|
|
792
|
+
if (this._wide) {
|
|
793
|
+
const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
|
|
794
|
+
this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
|
|
795
|
+
this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
|
|
796
|
+
this._nextMonthWeekNumbers = this._nextMonthWeeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
|
|
797
|
+
}
|
|
798
|
+
this._initialized = true;
|
|
799
|
+
}
|
|
800
|
+
/** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
|
|
801
|
+
_focusCell() {
|
|
802
|
+
if (this._resetFocus) {
|
|
803
|
+
this._getFirstFocusable()?.focus();
|
|
804
|
+
this._resetFocus = false;
|
|
805
|
+
}
|
|
806
|
+
}
|
|
807
|
+
/** Creates the array of weekdays. */
|
|
808
|
+
_setWeekdays() {
|
|
809
|
+
const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
|
|
810
|
+
const weekdays = this._dateAdapter.getDayOfWeekNames("long").map((long, i) => ({
|
|
811
|
+
long,
|
|
812
|
+
narrow: narrowWeekdays[i]
|
|
813
|
+
}));
|
|
814
|
+
const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
|
|
815
|
+
this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
|
|
816
|
+
}
|
|
817
|
+
/** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
|
|
818
|
+
_createWeekRows(value, isSecondMonthInView = false) {
|
|
819
|
+
const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
|
|
820
|
+
const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
|
|
821
|
+
if (!isSecondMonthInView) {
|
|
822
|
+
this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
|
|
823
|
+
this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
|
|
824
|
+
this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
|
|
825
|
+
this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
|
|
826
|
+
} else {
|
|
827
|
+
this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
|
|
828
|
+
this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
|
|
829
|
+
}
|
|
830
|
+
return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
|
|
831
|
+
}
|
|
832
|
+
/**
|
|
833
|
+
* Creates the rows for each week in orientation='horizontal'.
|
|
834
|
+
*
|
|
835
|
+
* Iterates through the days of the months, creates a Day object for each and pushes it into and array.
|
|
836
|
+
* Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
|
|
837
|
+
*
|
|
838
|
+
* The result is a matrix in which every row is a week (or part of it, considering offset).
|
|
839
|
+
*/
|
|
840
|
+
_createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
|
|
841
|
+
const weeks = [[]];
|
|
842
|
+
for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
|
|
843
|
+
if (cell === DAYS_PER_ROW) {
|
|
844
|
+
weeks.push([]);
|
|
845
|
+
cell = 0;
|
|
846
|
+
}
|
|
847
|
+
const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
|
|
848
|
+
weeks[weeks.length - 1].push(this._mapDateToDay(date));
|
|
849
|
+
}
|
|
850
|
+
return weeks;
|
|
851
|
+
}
|
|
852
|
+
/**
|
|
853
|
+
* Creates the rows for each week in orientation='vertical'.
|
|
854
|
+
*
|
|
855
|
+
* Creates a matrix with seven empty rows.
|
|
856
|
+
* Iterates through the days of the months, creates a Day object for each
|
|
857
|
+
* and pushes it into the correct array considering the offset at the beginning of the month.
|
|
858
|
+
* Each seven days (including offset) restarts from the first.
|
|
859
|
+
*
|
|
860
|
+
* The result is a matrix in which every row is a set of weekdays, so:
|
|
861
|
+
* - row 0: all the Mondays;
|
|
862
|
+
* - row 1: all the Tuesdays;
|
|
863
|
+
* - ...
|
|
864
|
+
* - row 7: all the Sundays.
|
|
865
|
+
*/
|
|
866
|
+
_createWeekRowsVertical(value, daysInMonth, weekOffset) {
|
|
867
|
+
const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
|
|
868
|
+
for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
|
|
869
|
+
if (cell === DAYS_PER_ROW) cell = 0;
|
|
870
|
+
const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
|
|
871
|
+
weeks[cell].push(this._mapDateToDay(date));
|
|
872
|
+
}
|
|
873
|
+
return weeks;
|
|
874
|
+
}
|
|
875
|
+
_mapDateToDay(date) {
|
|
876
|
+
return {
|
|
877
|
+
value: this._dateAdapter.toIso8601(date),
|
|
878
|
+
dateValue: date,
|
|
879
|
+
dayValue: String(this._dateAdapter.getDate(date)),
|
|
880
|
+
monthValue: String(this._dateAdapter.getMonth(date)),
|
|
881
|
+
yearValue: String(this._dateAdapter.getYear(date)),
|
|
882
|
+
weekValue: this._getWeek(date),
|
|
883
|
+
weekDayValue: this._dateAdapter.getDayOfWeek(date)
|
|
884
|
+
};
|
|
885
|
+
}
|
|
886
|
+
_getWeek(date) {
|
|
887
|
+
const firstDayOfYear = this._dateAdapter.createDate(this._dateAdapter.getYear(date), 1, 1);
|
|
888
|
+
const weekday = this._dateAdapter.getDayOfWeek(firstDayOfYear);
|
|
889
|
+
let weekIndex = [
|
|
890
|
+
MONDAY,
|
|
891
|
+
TUESDAY,
|
|
892
|
+
WEDNESDAY,
|
|
893
|
+
THURSDAY
|
|
894
|
+
].includes(weekday) ? 1 : 0;
|
|
895
|
+
let weekStart = this._dateAdapter.addCalendarDays(firstDayOfYear, this._dateAdapter.getFirstWeekOffset(firstDayOfYear) * -1);
|
|
896
|
+
while (this._dateAdapter.compareDate(weekStart, date) <= 0) {
|
|
897
|
+
const weekEnd = this._dateAdapter.addCalendarDays(weekStart, 6);
|
|
898
|
+
if (this._dateAdapter.compareDate(date, weekEnd) <= 0) return weekIndex > 0 ? weekIndex : this._getWeek(this._dateAdapter.addCalendarDays(firstDayOfYear, -1));
|
|
899
|
+
weekStart = this._dateAdapter.addCalendarDays(weekStart, DAYS_PER_ROW);
|
|
900
|
+
weekIndex++;
|
|
901
|
+
}
|
|
902
|
+
throw new Error("The provided date is invalid");
|
|
903
|
+
}
|
|
904
|
+
/** Force the conversion to ISO8601 formatted string. */
|
|
905
|
+
_mapValueToISODate(value) {
|
|
906
|
+
return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
|
|
907
|
+
}
|
|
908
|
+
/** Creates the rows for the month selection view. */
|
|
909
|
+
_createMonthRows() {
|
|
910
|
+
const months = new Array(12).fill(null).map((_, i) => ({
|
|
911
|
+
value: String(i + 1).padStart(2, "0"),
|
|
912
|
+
monthValue: i + 1
|
|
913
|
+
}));
|
|
914
|
+
const rows = 12 / MONTHS_PER_ROW;
|
|
915
|
+
const monthArray = [];
|
|
916
|
+
for (let i = 0; i < rows; i++) monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
|
|
917
|
+
this._months = monthArray;
|
|
918
|
+
}
|
|
919
|
+
/** Creates the rows for the year selection view. */
|
|
920
|
+
_createYearRows(offset = 0) {
|
|
921
|
+
const startValueYearView = this._getStartValueYearView();
|
|
922
|
+
const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
|
|
923
|
+
const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
|
|
924
|
+
const yearArray = [];
|
|
925
|
+
for (let i = 0; i < rows; i++) yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
|
|
926
|
+
return yearArray;
|
|
927
|
+
}
|
|
928
|
+
/**
|
|
929
|
+
* Calculates the first year that will be shown in the year selection panel.
|
|
930
|
+
* If `minDate` and `maxDate` are both null, the starting year is calculated as
|
|
931
|
+
* the multiple of YEARS_PER_PAGE closest to and less than activeDate,
|
|
932
|
+
* e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
|
|
933
|
+
* while with `activeDate` = 2000, the function will return 1992 (24 * 82).
|
|
934
|
+
* If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
|
|
935
|
+
* it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
|
|
936
|
+
* If both are not null, `maxDate` has priority over `minDate`.
|
|
937
|
+
*/
|
|
938
|
+
_getStartValueYearView() {
|
|
939
|
+
let startingYear = 0;
|
|
940
|
+
if (this.max) startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
|
|
941
|
+
else if (this.min) startingYear = this._dateAdapter.getYear(this.min);
|
|
942
|
+
const activeYear = this._dateAdapter.getYear(this._activeDate);
|
|
943
|
+
return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
|
|
944
|
+
}
|
|
945
|
+
/** Checks if date is within the min-max range. */
|
|
946
|
+
_isDayInRange(dateString) {
|
|
947
|
+
if (!this.min && !this.max) return true;
|
|
948
|
+
const date = this._dateAdapter.deserialize(dateString);
|
|
949
|
+
return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
|
|
950
|
+
}
|
|
951
|
+
/** Emits the selected date and sets it internally. */
|
|
952
|
+
_selectDate(day) {
|
|
953
|
+
this._chosenMonth = void 0;
|
|
954
|
+
this._setChosenYear();
|
|
955
|
+
if (this.multiple) {
|
|
956
|
+
if (this._selected && this._selected.length > 0) {
|
|
957
|
+
const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
|
|
958
|
+
if (indexOfSelectedDay !== -1) this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
|
|
959
|
+
else this._selected = [...this._selected, day];
|
|
960
|
+
} else this._selected = [day];
|
|
961
|
+
this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
|
|
962
|
+
} else if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
|
|
963
|
+
this._selected = day;
|
|
964
|
+
this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
|
|
965
|
+
}
|
|
966
|
+
}
|
|
967
|
+
/**
|
|
968
|
+
* Handle multiple dates selection via weekNumber / weekDay buttons:
|
|
969
|
+
* - if Cmd or Ctrl are pressed, add the new date to the current ones;
|
|
970
|
+
* - if not,
|
|
971
|
+
* - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
|
|
972
|
+
* - if not, the selected dates are the new ones.
|
|
973
|
+
*/
|
|
974
|
+
_selectMultipleDates(days) {
|
|
975
|
+
const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
|
|
976
|
+
const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
|
|
977
|
+
const daysToAddSet = new Set(daysToAdd);
|
|
978
|
+
const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
|
|
979
|
+
this._selected = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet).map((s) => this._dateAdapter.deserialize(s));
|
|
980
|
+
this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
|
|
981
|
+
}
|
|
982
|
+
/**
|
|
983
|
+
* Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
|
|
984
|
+
*/
|
|
985
|
+
_emitDateSelectedEvent(detail) {
|
|
986
|
+
/** @type {CustomEvent<T | T[]>} Event emitted on date selection. */
|
|
987
|
+
this.dispatchEvent(new CustomEvent("dateselected", {
|
|
988
|
+
detail,
|
|
989
|
+
composed: true,
|
|
990
|
+
bubbles: true
|
|
991
|
+
}));
|
|
992
|
+
}
|
|
993
|
+
_emitMonthChange() {
|
|
994
|
+
const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
|
|
995
|
+
/**
|
|
996
|
+
* @type {SbbMonthChangeEvent}
|
|
997
|
+
* Emits when the month changes.
|
|
998
|
+
* The `range` property contains the days array of the chosen month.
|
|
999
|
+
*/
|
|
1000
|
+
this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
|
|
1001
|
+
}
|
|
1002
|
+
/**
|
|
1003
|
+
* In case of multiple selection, newly added days must be added to the existing ones, without duplication.
|
|
1004
|
+
* If the days to add are exactly the same as the selected ones, the set must be emptied.
|
|
1005
|
+
*/
|
|
1006
|
+
_updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
|
|
1007
|
+
if (daysToAdd.every((day) => selectedSet.has(day))) daysToAddSet.forEach((day) => selectedSet.delete(day));
|
|
1008
|
+
else daysToAddSet.forEach((day) => selectedSet.add(day));
|
|
1009
|
+
return Array.from(selectedSet);
|
|
1010
|
+
}
|
|
1011
|
+
_setChosenYear() {
|
|
1012
|
+
if (this.view === "month") {
|
|
1013
|
+
let selectedDate;
|
|
1014
|
+
if (this.multiple) selectedDate = this.selected.at(-1);
|
|
1015
|
+
else selectedDate = this.selected;
|
|
1016
|
+
this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
|
|
1017
|
+
} else this._chosenYear = void 0;
|
|
1018
|
+
}
|
|
1019
|
+
_assignActiveDate(date) {
|
|
1020
|
+
if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
|
|
1021
|
+
this._activeDate = this.min;
|
|
1022
|
+
return;
|
|
1023
|
+
}
|
|
1024
|
+
if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
|
|
1025
|
+
this._activeDate = this.max;
|
|
1026
|
+
return;
|
|
1027
|
+
}
|
|
1028
|
+
this._activeDate = date;
|
|
1029
|
+
}
|
|
1030
|
+
/** Goes to the month identified by the shift. */
|
|
1031
|
+
_goToDifferentMonth(months) {
|
|
1032
|
+
this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
|
|
1033
|
+
this._emitMonthChange();
|
|
1034
|
+
}
|
|
1035
|
+
_goToDifferentYear(years) {
|
|
1036
|
+
this._chosenYear += years;
|
|
1037
|
+
this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
|
|
1038
|
+
this._init();
|
|
1039
|
+
}
|
|
1040
|
+
_goToDifferentYearRange(years) {
|
|
1041
|
+
this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
|
|
1042
|
+
}
|
|
1043
|
+
_prevDisabled(prevDate) {
|
|
1044
|
+
if (!this.min) return false;
|
|
1045
|
+
return this._dateAdapter.compareDate(prevDate, this.min) < 0;
|
|
1046
|
+
}
|
|
1047
|
+
_nextDisabled(nextDate) {
|
|
1048
|
+
if (!this.max) return false;
|
|
1049
|
+
return this._dateAdapter.compareDate(nextDate, this.max) > 0;
|
|
1050
|
+
}
|
|
1051
|
+
/** Checks if the "previous month" button should be disabled. */
|
|
1052
|
+
_previousMonthDisabled() {
|
|
1053
|
+
const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
|
|
1054
|
+
return this._prevDisabled(prevMonth);
|
|
1055
|
+
}
|
|
1056
|
+
/** Checks if the "next month" button should be disabled. */
|
|
1057
|
+
_nextMonthDisabled() {
|
|
1058
|
+
let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
|
|
1059
|
+
nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
|
|
1060
|
+
return this._nextDisabled(nextMonth);
|
|
1061
|
+
}
|
|
1062
|
+
/** Checks if the "previous year" button should be disabled. */
|
|
1063
|
+
_previousYearDisabled() {
|
|
1064
|
+
const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
|
|
1065
|
+
return this._prevDisabled(prevYear);
|
|
1066
|
+
}
|
|
1067
|
+
/** Checks if the "next year" button should be disabled. */
|
|
1068
|
+
_nextYearDisabled() {
|
|
1069
|
+
const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
|
|
1070
|
+
return this._nextDisabled(nextYear);
|
|
1071
|
+
}
|
|
1072
|
+
/** Checks if the "previous year" button should be disabled in year view. */
|
|
1073
|
+
_previousYearRangeDisabled() {
|
|
1074
|
+
const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
|
|
1075
|
+
return this._prevDisabled(prevYear);
|
|
1076
|
+
}
|
|
1077
|
+
/** Checks if the "next year" button should be disabled in year view. */
|
|
1078
|
+
_nextYearRangeDisabled() {
|
|
1079
|
+
const years = this._wide ? this._nextMonthYears : this._years;
|
|
1080
|
+
const lastYearRange = years[years.length - 1];
|
|
1081
|
+
const lastYear = lastYearRange[lastYearRange.length - 1];
|
|
1082
|
+
const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
|
|
1083
|
+
return this._nextDisabled(nextYear);
|
|
1084
|
+
}
|
|
1085
|
+
_handleTableBlur(eventTarget) {
|
|
1086
|
+
if (eventTarget?.localName !== "sbb-calendar-day") this._setTabIndex();
|
|
1087
|
+
}
|
|
1088
|
+
_setTabIndex() {
|
|
1089
|
+
Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach((day) => day.tabIndex = -1);
|
|
1090
|
+
const firstFocusable = this._getFirstFocusable();
|
|
1091
|
+
if (firstFocusable) firstFocusable.tabIndex = 0;
|
|
1092
|
+
}
|
|
1093
|
+
/** Get the element in the calendar to assign focus. */
|
|
1094
|
+
_getFirstFocusable() {
|
|
1095
|
+
if (this._calendarView === "day") {
|
|
1096
|
+
const selectedOrCurrent = this._cells.find((e) => e.matches(":is(:state(selected),[state--selected])")) ?? this._cells.find((e) => e.matches(":is(:state(current),[state--current])"));
|
|
1097
|
+
return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
|
|
1098
|
+
} else {
|
|
1099
|
+
const selectedOrCurrent = this.shadowRoot?.querySelector(":is(:state(selected),[state--selected])") ?? this.shadowRoot?.querySelector(":is(:state(current),[state--current])");
|
|
1100
|
+
return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(`sbb-calendar-${this._calendarView}:not([disabled])`);
|
|
1101
|
+
}
|
|
1102
|
+
}
|
|
1103
|
+
/**
|
|
1104
|
+
* In `day` view in `vertical` orientation,
|
|
1105
|
+
* if the first of the month is not a Monday, it is not the first rendered element in the table,
|
|
1106
|
+
* so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.
|
|
1107
|
+
*
|
|
1108
|
+
* To solve this, the element with the lowest `value` is taken (ISO String are ordered).
|
|
1109
|
+
*/
|
|
1110
|
+
_getFirstFocusableDay() {
|
|
1111
|
+
const cells = this._cells;
|
|
1112
|
+
const daysInView = cells.filter((e) => !e.disabled);
|
|
1113
|
+
if (!daysInView || daysInView.length === 0) return null;
|
|
1114
|
+
else {
|
|
1115
|
+
const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
|
|
1116
|
+
return cells.find((e) => e.matches(`[slot="${firstElement}"]`)) ?? null;
|
|
1117
|
+
}
|
|
1118
|
+
}
|
|
1119
|
+
_handleKeyboardEvent(event, day) {
|
|
1120
|
+
if (isArrowKeyOrPageKeysPressed(event)) event.preventDefault();
|
|
1121
|
+
const cells = this._cells;
|
|
1122
|
+
const index = cells.findIndex((e) => e === event.target);
|
|
1123
|
+
let nextEl;
|
|
1124
|
+
if (day) nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
|
|
1125
|
+
else nextEl = this._navigateByKeyboard(event, index, cells);
|
|
1126
|
+
const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
|
|
1127
|
+
if (nextEl !== activeEl) {
|
|
1128
|
+
nextEl.tabIndex = 0;
|
|
1129
|
+
nextEl?.focus();
|
|
1130
|
+
activeEl.tabIndex = -1;
|
|
1131
|
+
}
|
|
1132
|
+
}
|
|
1133
|
+
_navigateByKeyboardDayView(evt, index, cells, day) {
|
|
1134
|
+
const arrowsOffset = this.orientation === "horizontal" ? {
|
|
1135
|
+
leftRight: 1,
|
|
1136
|
+
upDown: DAYS_PER_ROW
|
|
1137
|
+
} : {
|
|
1138
|
+
leftRight: DAYS_PER_ROW,
|
|
1139
|
+
upDown: 1
|
|
1140
|
+
};
|
|
1141
|
+
const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
|
|
1142
|
+
switch (evt.key) {
|
|
1143
|
+
case "ArrowUp": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
|
|
1144
|
+
case "ArrowDown": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
|
|
1145
|
+
case "ArrowLeft": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
|
|
1146
|
+
case "ArrowRight": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
|
|
1147
|
+
case "PageUp": if (this.orientation === "horizontal") {
|
|
1148
|
+
const delta = (+day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW) - +day.dayValue;
|
|
1149
|
+
return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
|
|
1150
|
+
} else {
|
|
1151
|
+
const delta = (Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) - 1) * DAYS_PER_ROW - offsetForVertical + 1 - +day.dayValue;
|
|
1152
|
+
return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
|
|
1153
|
+
}
|
|
1154
|
+
case "PageDown": if (this.orientation === "horizontal") {
|
|
1155
|
+
const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
|
|
1156
|
+
const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
|
|
1157
|
+
const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
|
|
1158
|
+
const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
|
|
1159
|
+
const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
|
|
1160
|
+
return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
|
|
1161
|
+
} else {
|
|
1162
|
+
const delta = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) * DAYS_PER_ROW - offsetForVertical - +day.dayValue;
|
|
1163
|
+
return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
|
|
1164
|
+
}
|
|
1165
|
+
case "Home": return this._findDayFirst(cells, index, day, 1);
|
|
1166
|
+
case "End": {
|
|
1167
|
+
const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
|
|
1168
|
+
const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
|
|
1169
|
+
const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
|
|
1170
|
+
return this._findDayLast(cells, index, firstNextMonth);
|
|
1171
|
+
}
|
|
1172
|
+
default: return cells[index];
|
|
1173
|
+
}
|
|
1174
|
+
}
|
|
1175
|
+
_isDayOutOfView(date) {
|
|
1176
|
+
return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
|
|
1177
|
+
}
|
|
1178
|
+
_findDayArrows(cells, index, date, delta) {
|
|
1179
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
|
|
1180
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1181
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1182
|
+
if (!nextCell || nextCell.disabled) return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
|
|
1183
|
+
return nextCell;
|
|
1184
|
+
}
|
|
1185
|
+
_findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
|
|
1186
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
|
|
1187
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1188
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1189
|
+
if (!nextCell || nextCell.disabled) return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
|
|
1190
|
+
return nextCell;
|
|
1191
|
+
}
|
|
1192
|
+
_findDayFirst(cells, index, day, date) {
|
|
1193
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
|
|
1194
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1195
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1196
|
+
if (!nextCell || nextCell.disabled) return this._findDayFirst(cells, index, day, date + 1);
|
|
1197
|
+
return nextCell;
|
|
1198
|
+
}
|
|
1199
|
+
_findDayLast(cells, index, firstNextMonth) {
|
|
1200
|
+
const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
|
|
1201
|
+
if (this._isDayOutOfView(newDateValue)) return cells[index];
|
|
1202
|
+
const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
|
|
1203
|
+
if (!nextCell || nextCell.disabled) return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
|
|
1204
|
+
return nextCell;
|
|
1205
|
+
}
|
|
1206
|
+
/**
|
|
1207
|
+
* Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
|
|
1208
|
+
* the keyboard input and the position of the current element in the list.
|
|
1209
|
+
* In the day view, the `day?: Day` parameter is mandatory for calculation,
|
|
1210
|
+
* while in month and year view it's not due to the fixed amount of rendered cells.
|
|
1211
|
+
*/
|
|
1212
|
+
_navigateByKeyboard(evt, index, cells) {
|
|
1213
|
+
const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
|
|
1214
|
+
switch (evt.key) {
|
|
1215
|
+
case "ArrowUp": return this._findNext(cells, index, -verticalOffset);
|
|
1216
|
+
case "ArrowDown": return this._findNext(cells, index, verticalOffset);
|
|
1217
|
+
case "ArrowLeft": return this._findNext(cells, index, -1);
|
|
1218
|
+
case "ArrowRight": return this._findNext(cells, index, 1);
|
|
1219
|
+
case "Home": return this._findFirst(cells, offsetForWideMode);
|
|
1220
|
+
case "PageUp": return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
|
|
1221
|
+
case "PageDown": return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
|
|
1222
|
+
case "End": return this._findLast(cells, lastElementIndexForWideMode - 1);
|
|
1223
|
+
default: return cells[index];
|
|
1224
|
+
}
|
|
1225
|
+
}
|
|
1226
|
+
/**
|
|
1227
|
+
* Calculates the parameters needed in keyboard navigation in year and month view.
|
|
1228
|
+
* @param index The starting element's index in the cell array.
|
|
1229
|
+
* @param isYearView Whether the displayed `view` is the year one.
|
|
1230
|
+
*/
|
|
1231
|
+
_calculateParametersForKeyboardNavigation(index, isYearView) {
|
|
1232
|
+
const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
|
|
1233
|
+
const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
|
|
1234
|
+
const indexInView = offset === 0 ? index : index - elementsPerPage;
|
|
1235
|
+
return {
|
|
1236
|
+
verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
|
|
1237
|
+
elementIndexForWideMode: indexInView,
|
|
1238
|
+
offsetForWideMode: index - indexInView,
|
|
1239
|
+
lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
|
|
1240
|
+
};
|
|
1241
|
+
}
|
|
1242
|
+
/**
|
|
1243
|
+
* Gets the next element of the provided array starting from `index` by adding `delta`.
|
|
1244
|
+
* If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
|
|
1245
|
+
*/
|
|
1246
|
+
_findNext(days, index, delta) {
|
|
1247
|
+
let nextIndex = index + delta;
|
|
1248
|
+
while (nextIndex < days.length && days[nextIndex]?.disabled) nextIndex += delta;
|
|
1249
|
+
return days[nextIndex] ?? days[index];
|
|
1250
|
+
}
|
|
1251
|
+
/** Find the first enabled element in the provided array. */
|
|
1252
|
+
_findFirst(days, firstOfCurrentMonth) {
|
|
1253
|
+
return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
|
|
1254
|
+
}
|
|
1255
|
+
/** Find the last enabled element in the provided array. */
|
|
1256
|
+
_findLast(days, lastOfCurrentMonth) {
|
|
1257
|
+
return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
|
|
1258
|
+
}
|
|
1259
|
+
/** Find the first enabled element in the same column of the provided array. */
|
|
1260
|
+
_findFirstOnColumn(days, index, offset, verticalOffset) {
|
|
1261
|
+
const nextIndex = index % verticalOffset + offset;
|
|
1262
|
+
return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
|
|
1263
|
+
}
|
|
1264
|
+
/** Find the last enabled element in the same column of the provided array. */
|
|
1265
|
+
_findLastOnColumn(days, index, offset, verticalOffset) {
|
|
1266
|
+
const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
|
|
1267
|
+
return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
|
|
1268
|
+
}
|
|
1269
|
+
_resetCalendarViewAndEmitMonthChange(initTransition = false) {
|
|
1270
|
+
this._resetCalendarView(initTransition);
|
|
1271
|
+
this._emitMonthChange();
|
|
1272
|
+
}
|
|
1273
|
+
_resetCalendarView(initTransition = false) {
|
|
1274
|
+
if (this._containingFocus) this._resetFocus = true;
|
|
1275
|
+
this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
|
|
1276
|
+
this._setChosenYear();
|
|
1277
|
+
this._chosenMonth = void 0;
|
|
1278
|
+
this._init();
|
|
1279
|
+
this._nextCalendarView = this._calendarView = this.view;
|
|
1280
|
+
if (initTransition) this._startTableTransition();
|
|
1281
|
+
}
|
|
1282
|
+
/** Render the view for the day selection. */
|
|
1283
|
+
_renderDayView() {
|
|
1284
|
+
const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
|
|
1285
|
+
return html`
|
|
1286
|
+
<div class="sbb-calendar__controls">
|
|
1287
|
+
${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
|
|
1288
|
+
<div class="sbb-calendar__controls-month">
|
|
1289
|
+
${this._createLabelForDayView(this._activeDate)}
|
|
1290
|
+
${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
|
|
1291
|
+
<sbb-screen-reader-only role="status">
|
|
1292
|
+
${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
|
|
1293
|
+
</sbb-screen-reader-only>
|
|
1294
|
+
</div>
|
|
1295
|
+
${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
|
|
1296
|
+
</div>
|
|
1297
|
+
<div class="sbb-calendar__table-overflow-break">
|
|
1298
|
+
<div class="sbb-calendar__table-container sbb-calendar__table-day-view">
|
|
1299
|
+
${this.orientation === "horizontal" ? html`
|
|
1300
|
+
${this._createDayTable(this._weeks, this._weekNumbers)}
|
|
1301
|
+
${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
|
|
1302
|
+
` : html`
|
|
1303
|
+
${this._createDayTableVertical(this._weeks, this._weekNumbers)}
|
|
1304
|
+
${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
|
|
1305
|
+
`}
|
|
1306
|
+
</div>
|
|
1307
|
+
</div>
|
|
1308
|
+
`;
|
|
1309
|
+
}
|
|
1310
|
+
/** Creates the label with the month for the daily view. */
|
|
1311
|
+
_createLabelForDayView(d) {
|
|
1312
|
+
const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
|
|
1313
|
+
return html`
|
|
1314
|
+
<button
|
|
1315
|
+
type="button"
|
|
1316
|
+
class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
|
|
1317
|
+
aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
|
|
1318
|
+
@click=${() => {
|
|
1319
|
+
this._resetFocus = true;
|
|
1320
|
+
this._nextCalendarView = "year";
|
|
1321
|
+
this._startTableTransition();
|
|
1322
|
+
}}
|
|
1323
|
+
>
|
|
1324
|
+
${monthLabel}
|
|
1325
|
+
<sbb-icon name="chevron-small-down-small"></sbb-icon>
|
|
1326
|
+
</button>
|
|
1327
|
+
`;
|
|
1328
|
+
}
|
|
1329
|
+
/** Creates the aria-label for the daily view. */
|
|
1330
|
+
_createAriaLabelForDayView(...dates) {
|
|
1331
|
+
let monthLabel = "";
|
|
1332
|
+
for (const d of dates) if (d) monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
|
|
1333
|
+
return monthLabel;
|
|
1334
|
+
}
|
|
1335
|
+
/** Creates the calendar table for the daily view. */
|
|
1336
|
+
_createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
|
|
1337
|
+
const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
|
|
1338
|
+
const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
|
|
1339
|
+
return html`
|
|
1340
|
+
<table
|
|
1341
|
+
class="sbb-calendar__table"
|
|
1342
|
+
@focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
|
|
1343
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1344
|
+
>
|
|
1345
|
+
<thead class="sbb-calendar__table-header">
|
|
1346
|
+
<tr>
|
|
1347
|
+
${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
|
|
1348
|
+
${this._weekdays.map((weekDay, index) => html`
|
|
1349
|
+
<th class="sbb-calendar__table-header-cell">
|
|
1350
|
+
${this.multiple ? html`
|
|
1351
|
+
<sbb-calendar-weekday
|
|
1352
|
+
.value=${weekDay}
|
|
1353
|
+
@click=${() => {
|
|
1354
|
+
const days = weeksForSelectMultipleWeekDays.filter((day) => day.weekDayValue === (index + 1) % 7);
|
|
1355
|
+
this._selectMultipleDates(days);
|
|
1356
|
+
}}
|
|
1357
|
+
></sbb-calendar-weekday>
|
|
1358
|
+
` : html`
|
|
1359
|
+
<sbb-screen-reader-only>${weekDay.long}</sbb-screen-reader-only>
|
|
1360
|
+
<span aria-hidden="true">${weekDay.narrow}</span>
|
|
1361
|
+
`}
|
|
1362
|
+
</th>
|
|
1363
|
+
`)}
|
|
1364
|
+
</tr>
|
|
1365
|
+
</thead>
|
|
1366
|
+
<tbody class="sbb-calendar__table-body">
|
|
1367
|
+
${weeks.map((week, rowIndex) => {
|
|
1368
|
+
const firstRowOffset = DAYS_PER_ROW - week.length;
|
|
1369
|
+
if (rowIndex === 0 && firstRowOffset) return html`
|
|
1370
|
+
<tr>
|
|
1371
|
+
${this.weekNumbers ? html`
|
|
1372
|
+
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1373
|
+
${this.multiple ? html`
|
|
1374
|
+
<sbb-calendar-weeknumber
|
|
1375
|
+
.value=${weekNumbers[0]}
|
|
1376
|
+
@click=${() => {
|
|
1377
|
+
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
|
|
1378
|
+
this._selectMultipleDates(days);
|
|
1379
|
+
}}
|
|
1380
|
+
></sbb-calendar-weeknumber>
|
|
1381
|
+
` : html`
|
|
1382
|
+
<sbb-screen-reader-only
|
|
1383
|
+
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only
|
|
1384
|
+
>
|
|
1385
|
+
<span aria-hidden="true">${weekNumbers[0]}</span>
|
|
1386
|
+
`}
|
|
1387
|
+
</td>
|
|
1388
|
+
` : nothing}
|
|
1389
|
+
${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
|
|
1390
|
+
${this._createDayCells(week)}
|
|
1391
|
+
</tr>
|
|
1392
|
+
`;
|
|
1393
|
+
return html`
|
|
1394
|
+
<tr>
|
|
1395
|
+
${this.weekNumbers ? html`
|
|
1396
|
+
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1397
|
+
${this.multiple ? html`
|
|
1398
|
+
<sbb-calendar-weeknumber
|
|
1399
|
+
.value=${weekNumbers[rowIndex]}
|
|
1400
|
+
@click=${() => {
|
|
1401
|
+
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
|
|
1402
|
+
this._selectMultipleDates(days);
|
|
1403
|
+
}}
|
|
1404
|
+
></sbb-calendar-weeknumber>
|
|
1405
|
+
` : html`
|
|
1406
|
+
<sbb-screen-reader-only
|
|
1407
|
+
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only
|
|
1408
|
+
>
|
|
1409
|
+
<span aria-hidden="true">${weekNumbers[rowIndex]}</span>
|
|
1410
|
+
`}
|
|
1411
|
+
</td>
|
|
1412
|
+
` : nothing}
|
|
1413
|
+
${this._createDayCells(week)}
|
|
1414
|
+
</tr>
|
|
1415
|
+
`;
|
|
1416
|
+
})}
|
|
1417
|
+
</tbody>
|
|
1418
|
+
</table>
|
|
1419
|
+
`;
|
|
1420
|
+
}
|
|
1421
|
+
/** Creates the table in orientation='vertical'. */
|
|
1422
|
+
_createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
|
|
1423
|
+
const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
|
|
1424
|
+
const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
|
|
1425
|
+
return html`
|
|
1426
|
+
<table
|
|
1427
|
+
class="sbb-calendar__table"
|
|
1428
|
+
@focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
|
|
1429
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1430
|
+
>
|
|
1431
|
+
${this.weekNumbers ? html`
|
|
1432
|
+
<thead class="sbb-calendar__table-header">
|
|
1433
|
+
<tr>
|
|
1434
|
+
${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
|
|
1435
|
+
${weekNumbers.map((weekNumber) => html`
|
|
1436
|
+
<th class="sbb-calendar__table-header-cell">
|
|
1437
|
+
${this.multiple ? html`
|
|
1438
|
+
<sbb-calendar-weeknumber
|
|
1439
|
+
.value=${weekNumber}
|
|
1440
|
+
@click=${() => {
|
|
1441
|
+
const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
|
|
1442
|
+
this._selectMultipleDates(days);
|
|
1443
|
+
}}
|
|
1444
|
+
></sbb-calendar-weeknumber>
|
|
1445
|
+
` : html`
|
|
1446
|
+
<sbb-screen-reader-only
|
|
1447
|
+
>${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only
|
|
1448
|
+
>
|
|
1449
|
+
<span aria-hidden="true">${weekNumber}</span>
|
|
1450
|
+
`}
|
|
1451
|
+
</th>
|
|
1452
|
+
`)}
|
|
1453
|
+
</tr>
|
|
1454
|
+
</thead>
|
|
1455
|
+
` : nothing}
|
|
1456
|
+
<tbody class="sbb-calendar__table-body">
|
|
1457
|
+
${weeks.map((week, rowIndex) => {
|
|
1458
|
+
const weekday = this._weekdays[rowIndex];
|
|
1459
|
+
const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
|
|
1460
|
+
return html`
|
|
1461
|
+
<tr>
|
|
1462
|
+
${nextMonthActiveDate ? nothing : html`
|
|
1463
|
+
<td class="sbb-calendar__table-header-cell-vertical">
|
|
1464
|
+
${this.multiple ? html`
|
|
1465
|
+
<sbb-calendar-weekday
|
|
1466
|
+
.value=${weekday}
|
|
1467
|
+
@click=${() => this._selectMultipleDates(selectableDays)}
|
|
1468
|
+
>
|
|
1469
|
+
${weekday.narrow}
|
|
1470
|
+
</sbb-calendar-weekday>
|
|
1471
|
+
` : html`
|
|
1472
|
+
<sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>
|
|
1473
|
+
<span aria-hidden="true">${weekday.narrow}</span>
|
|
1474
|
+
`}
|
|
1475
|
+
</td>
|
|
1476
|
+
`}
|
|
1477
|
+
${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
|
|
1478
|
+
${this._createDayCells(week)}
|
|
1479
|
+
</tr>
|
|
1480
|
+
`;
|
|
1481
|
+
})}
|
|
1482
|
+
</tbody>
|
|
1483
|
+
</table>
|
|
1484
|
+
`;
|
|
1485
|
+
}
|
|
1486
|
+
/** Creates the cells for the daily view. */
|
|
1487
|
+
_createDayCells(week) {
|
|
1488
|
+
return week.map((day) => {
|
|
1489
|
+
return html`
|
|
1490
|
+
<td class="sbb-calendar__table-data sbb-calendar__day-cell">
|
|
1491
|
+
<slot name=${day.value}>
|
|
1492
|
+
<sbb-calendar-day
|
|
1493
|
+
slot=${day.value}
|
|
1494
|
+
@click=${() => this._selectDate(day.dateValue)}
|
|
1495
|
+
@keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
|
|
1496
|
+
></sbb-calendar-day>
|
|
1497
|
+
</slot>
|
|
1498
|
+
</td>
|
|
1499
|
+
`;
|
|
1500
|
+
});
|
|
1501
|
+
}
|
|
1502
|
+
/** Render the view for the month selection. */
|
|
1503
|
+
_renderMonthView() {
|
|
1504
|
+
return html`
|
|
1505
|
+
<div class="sbb-calendar__controls">
|
|
1506
|
+
${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
|
|
1507
|
+
<div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
|
|
1508
|
+
${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
|
|
1509
|
+
</div>
|
|
1510
|
+
<div class="sbb-calendar__table-overflow-break">
|
|
1511
|
+
<div class="sbb-calendar__table-container sbb-calendar__table-month-view">
|
|
1512
|
+
${this._createMonthTable(this._months, this._chosenYear)}
|
|
1513
|
+
${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
|
|
1514
|
+
</div>
|
|
1515
|
+
</div>
|
|
1516
|
+
`;
|
|
1517
|
+
}
|
|
1518
|
+
/** Creates the label with the year for the monthly view. */
|
|
1519
|
+
_createLabelForMonthView() {
|
|
1520
|
+
return html` <button
|
|
1521
|
+
type="button"
|
|
1522
|
+
id="sbb-calendar__month-selection"
|
|
1523
|
+
class="sbb-calendar__controls-change-date"
|
|
1524
|
+
aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
|
|
1525
|
+
@click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
|
|
1526
|
+
>
|
|
1527
|
+
${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
|
|
1528
|
+
<sbb-icon name="chevron-small-up-small"></sbb-icon>
|
|
1529
|
+
</button>
|
|
1530
|
+
<sbb-screen-reader-only role="status"> ${this._chosenYear} </sbb-screen-reader-only>`;
|
|
1531
|
+
}
|
|
1532
|
+
/** Creates the table for the month selection view. */
|
|
1533
|
+
_createMonthTable(months, year) {
|
|
1534
|
+
return html`
|
|
1535
|
+
<table
|
|
1536
|
+
class="sbb-calendar__table"
|
|
1537
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1538
|
+
>
|
|
1539
|
+
${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
|
|
1540
|
+
<tr>
|
|
1541
|
+
<th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
|
|
1542
|
+
</tr>
|
|
1543
|
+
</thead>` : nothing}
|
|
1544
|
+
<tbody class="sbb-calendar__table-body">
|
|
1545
|
+
${months.map((row) => html`
|
|
1546
|
+
<tr>
|
|
1547
|
+
${row.map((month) => {
|
|
1548
|
+
return html`
|
|
1549
|
+
<td class="sbb-calendar__table-data">
|
|
1550
|
+
<sbb-calendar-month
|
|
1551
|
+
.value="${year}-${month.value}"
|
|
1552
|
+
@click=${() => this._onMonthSelection(month.monthValue, year)}
|
|
1553
|
+
@keydown=${(evt) => this._handleKeyboardEvent(evt)}
|
|
1554
|
+
>
|
|
1555
|
+
</sbb-calendar-month>
|
|
1556
|
+
</td>
|
|
1557
|
+
`;
|
|
1558
|
+
})}
|
|
1559
|
+
</tr>
|
|
1560
|
+
`)}
|
|
1561
|
+
</tbody>
|
|
1562
|
+
</table>
|
|
1563
|
+
`;
|
|
1564
|
+
}
|
|
1565
|
+
/** Select the month and change the view to day selection. */
|
|
1566
|
+
_onMonthSelection(month, year) {
|
|
1567
|
+
this._chosenMonth = month;
|
|
1568
|
+
this._nextCalendarView = "day";
|
|
1569
|
+
this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
|
|
1570
|
+
this._startTableTransition();
|
|
1571
|
+
this._emitMonthChange();
|
|
1572
|
+
}
|
|
1573
|
+
/** Render the view for the year selection. */
|
|
1574
|
+
_renderYearView() {
|
|
1575
|
+
return html`
|
|
1576
|
+
<div class="sbb-calendar__controls">
|
|
1577
|
+
${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
|
|
1578
|
+
<div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
|
|
1579
|
+
${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
|
|
1580
|
+
</div>
|
|
1581
|
+
<div class="sbb-calendar__table-overflow-break">
|
|
1582
|
+
<div class="sbb-calendar__table-container sbb-calendar__table-year-view">
|
|
1583
|
+
${this._createYearTable(this._years)}
|
|
1584
|
+
${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
|
|
1585
|
+
</div>
|
|
1586
|
+
</div>
|
|
1587
|
+
`;
|
|
1588
|
+
}
|
|
1589
|
+
/** Creates the button arrow for all the views. */
|
|
1590
|
+
_getArrow(direction, click, ariaLabel, disabled) {
|
|
1591
|
+
return html`<sbb-secondary-button
|
|
1592
|
+
size="m"
|
|
1593
|
+
icon-name="chevron-small-${direction}-small"
|
|
1594
|
+
aria-label=${ariaLabel}
|
|
1595
|
+
@click=${click}
|
|
1596
|
+
?disabled=${disabled}
|
|
1597
|
+
id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
|
|
1598
|
+
></sbb-secondary-button>`;
|
|
1599
|
+
}
|
|
1600
|
+
/** Creates the label with the year range for the yearly view. */
|
|
1601
|
+
_createLabelForYearView() {
|
|
1602
|
+
const firstYear = this._years.flat()[0];
|
|
1603
|
+
const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
|
|
1604
|
+
const yearLabel = `${firstYear} - ${lastYearArray[lastYearArray.length - 1]}`;
|
|
1605
|
+
return html`
|
|
1606
|
+
<button
|
|
1607
|
+
type="button"
|
|
1608
|
+
id="sbb-calendar__year-selection"
|
|
1609
|
+
class="sbb-calendar__controls-change-date"
|
|
1610
|
+
aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
|
|
1611
|
+
@click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
|
|
1612
|
+
>
|
|
1613
|
+
${yearLabel}
|
|
1614
|
+
<sbb-icon name="chevron-small-up-small"></sbb-icon>
|
|
1615
|
+
</button>
|
|
1616
|
+
<sbb-screen-reader-only role="status"> ${yearLabel} </sbb-screen-reader-only>
|
|
1617
|
+
`;
|
|
1618
|
+
}
|
|
1619
|
+
/** Creates the table for the year selection view. */
|
|
1620
|
+
_createYearTable(years, shiftRight = false) {
|
|
1621
|
+
return html` <table
|
|
1622
|
+
class="sbb-calendar__table"
|
|
1623
|
+
@animationend=${(e) => this._tableAnimationEnd(e)}
|
|
1624
|
+
>
|
|
1625
|
+
<tbody class="sbb-calendar__table-body">
|
|
1626
|
+
${years.map((row) => html` <tr>
|
|
1627
|
+
${row.map((year) => {
|
|
1628
|
+
return html`
|
|
1629
|
+
<td class="sbb-calendar__table-data">
|
|
1630
|
+
<sbb-calendar-year
|
|
1631
|
+
.value=${String(year)}
|
|
1632
|
+
@keydown=${(evt) => this._handleKeyboardEvent(evt)}
|
|
1633
|
+
@click=${() => this._onYearSelection(year, shiftRight)}
|
|
1634
|
+
>
|
|
1635
|
+
</sbb-calendar-year>
|
|
1636
|
+
</td>
|
|
1637
|
+
`;
|
|
1638
|
+
})}
|
|
1639
|
+
</tr>`)}
|
|
1640
|
+
</tbody>
|
|
1641
|
+
</table>`;
|
|
1642
|
+
}
|
|
1643
|
+
/** Select the year and change the view to month selection. */
|
|
1644
|
+
_onYearSelection(year, rightSide) {
|
|
1645
|
+
this._chosenYear = rightSide ? year - 1 : year;
|
|
1646
|
+
this._nextCalendarView = "month";
|
|
1647
|
+
this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
|
|
1648
|
+
this._startTableTransition();
|
|
1649
|
+
}
|
|
1650
|
+
_getView() {
|
|
1651
|
+
if (isServer || this.hydrationRequired) return html`${nothing}`;
|
|
1652
|
+
switch (this._calendarView) {
|
|
1653
|
+
case "year": return this._renderYearView();
|
|
1654
|
+
case "month": return this._renderMonthView();
|
|
1655
|
+
default: return this._renderDayView();
|
|
1656
|
+
}
|
|
1657
|
+
}
|
|
1658
|
+
_tableAnimationEnd(event) {
|
|
1659
|
+
const table = event.target;
|
|
1660
|
+
if (event.animationName === "hide") {
|
|
1661
|
+
table.classList.remove("sbb-calendar__table-hide");
|
|
1662
|
+
if (this._containingFocus) this._resetFocus = true;
|
|
1663
|
+
this._calendarView = this._nextCalendarView;
|
|
1664
|
+
} else if (event.animationName === "show") this.internals.states.delete("transition");
|
|
1665
|
+
}
|
|
1666
|
+
_startTableTransition() {
|
|
1667
|
+
this.internals.states.add("transition");
|
|
1668
|
+
this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
|
|
1669
|
+
}
|
|
1670
|
+
render() {
|
|
1671
|
+
return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
|
|
1672
|
+
}
|
|
1673
|
+
};
|
|
1674
|
+
})();
|
|
1675
|
+
//#endregion
|
|
1676
|
+
export { SbbMonthChangeEvent as n, SbbCalendarElement as t };
|
|
1677
|
+
|
|
1678
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"calendar.component-CsAl9EAg.js","names":[],"sources":["../../../src/elements/calendar/calendar/calendar.scss?lit&inline","../../../src/elements/calendar/calendar/calendar.component.ts"],"sourcesContent":["@use '../../core/styles' as sbb;\n\n:host {\n  display: block;\n\n  // We add width definition to host, to make overwriting easy for consumers.\n  width: max-content;\n\n  --sbb-calendar-cell-size: #{sbb.px-to-rem-build(44)};\n\n  // Match Figma design\n  --sbb-calendar-cell-border-radius: calc(\n    var(--sbb-border-radius-4x) + var(--sbb-calendar-cell-border-width)\n  );\n  --sbb-calendar-cell-transition-duration: var(\n    --sbb-disable-animation-duration,\n    var(--sbb-animation-duration-2x)\n  );\n\n  @include sbb.mq($from: small) {\n    --sbb-calendar-control-view-change-height: #{sbb.px-to-rem-build(48)};\n  }\n}\n\n:host(:state(enhanced)) {\n  --sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);\n}\n\n.sbb-calendar__wrapper {\n  width: 100%;\n  display: block;\n  transition-duration: var(--sbb-calendar-cell-transition-duration);\n}\n\n.sbb-calendar__controls {\n  width: 100%;\n  display: inline-flex;\n  align-items: center;\n  gap: var(--sbb-calendar-control-gap);\n  margin-block-end: var(--sbb-calendar-control-margin-block-end);\n}\n\n.sbb-calendar__controls-month {\n  width: 100%;\n  display: flex;\n  gap: var(--sbb-calendar-tables-gap);\n}\n\n#sbb-calendar__controls-previous,\n#sbb-calendar__controls-next {\n  -webkit-tap-highlight-color: transparent;\n}\n\n.sbb-calendar__controls-change-date {\n  @include sbb.button-reset;\n\n  display: flex;\n  align-items: center;\n  margin: auto;\n  height: var(--sbb-calendar-control-view-change-height);\n  font-size: var(--sbb-text-font-size-s);\n  letter-spacing: var(--sbb-typo-letter-spacing-text);\n  text-transform: capitalize;\n  cursor: var(--sbb-cursor-pointer);\n  padding-inline: var(--sbb-calendar-control-view-change-padding-inline);\n  border-radius: var(--sbb-border-radius-infinity);\n  background-color: var(--sbb-calendar-control-view-change-background);\n  color: var(--sbb-calendar-control-view-change-color);\n  transition-duration: var(--sbb-calendar-cell-transition-duration);\n  transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);\n  transition-property: background-color, padding-block-end;\n\n  &:disabled {\n    --sbb-calendar-control-view-change-background: light-dark(\n      var(--sbb-color-milk),\n      var(--sbb-color-anthracite)\n    );\n    --sbb-calendar-control-view-change-color: light-dark(\n      var(--sbb-color-granite),\n      var(--sbb-color-aluminium)\n    );\n\n    cursor: unset;\n  }\n\n  &:focus-visible {\n    @include sbb.focus-outline;\n\n    outline-offset: var(--sbb-spacing-fixed-1x);\n  }\n\n  @include sbb.hover-mq {\n    &:not(:active, :disabled):hover {\n      padding-block-end: var(--sbb-calendar-cell-hover-shift);\n    }\n  }\n\n  &:not(:disabled):active {\n    --sbb-calendar-control-view-change-background: var(--sbb-background-color-3);\n  }\n}\n\n.sbb-calendar__table-month-view,\n.sbb-calendar__table-year-view {\n  --sbb-calendar-table-column-spaces: 6;\n}\n\n.sbb-calendar__table-overflow-break {\n  // We have to cut the negative margin which is built by the containing sbb-calendar__table-container.\n  // Overflow: hidden would cut the focus outline. Therefore we take `contain` here.\n  contain: layout;\n}\n\n.sbb-calendar__table-container {\n  display: flex;\n  gap: var(--sbb-calendar-tables-gap);\n  margin-inline: var(--sbb-calendar-margin);\n\n  // The padding of the first and last column should not be visible if calendar is stretched.\n  // Therefore we need a negative inline margin.\n  // As we don't want to squeeze, the margin should never be greater than zero.\n\n  // Min width is equals to the normal width of the calendar\n  --sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));\n\n  // The overflow variable is equals the difference between the actual width and the min width.\n  --sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));\n\n  // The start offset is negative margin which should overlap the parent container. Should never be a positive value.\n  --sbb-calendar-start-offset: min(\n    0px,\n    -1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))\n  );\n  --sbb-calendar-margin: var(--sbb-calendar-start-offset);\n\n  :host(:state(wide)) & {\n    --sbb-calendar-min-width: calc(\n      2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)\n    );\n    --sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));\n  }\n\n  :host([orientation='horizontal'][week-numbers]) & {\n    --sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));\n  }\n\n  :host([orientation='horizontal'][week-numbers]:state(wide)) & {\n    --sbb-calendar-min-width: calc(\n      2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)\n    );\n  }\n\n  // The container's min-width is set in vertical mode;\n  // the value `calc(7 * var(--sbb-calendar-cell-size))` is fine\n  // considering that the maximum number of weeks in a month is 6, plus 1 for the header at the table's left side.\n  :host([orientation='vertical']) & {\n    min-width: var(--sbb-calendar-min-width);\n\n    // The variable that defines the margin is set to 0 to correctly display the table if consumers set a custom width.\n    --sbb-calendar-start-offset: 0;\n  }\n\n  // The min-width in wide is calculated as (max number of weeks in a month) * (displayed months) + (header column) = 6 * 2 + 1 = 13.\n  :host([orientation='vertical']:state(wide)) & {\n    --sbb-calendar-min-width: calc(\n      13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)\n    );\n  }\n}\n\n.sbb-calendar__table {\n  width: 100%;\n  border-collapse: collapse;\n  height: max-content;\n\n  animation: {\n    name: show;\n    duration: var(--sbb-calendar-table-animation-duration);\n  }\n\n  &.sbb-calendar__table-hide {\n    --sbb-calendar-cell-transition-duration: 0ms;\n\n    animation: {\n      name: hide;\n      duration: var(--sbb-calendar-table-animation-duration);\n    }\n  }\n\n  :host(:not(:state(wide))) & {\n    // Due to a Safari iOS rendering bug we need to define min-width as well.\n    // Otherwise, after orientation change, there is a wrong width if placed in an sbb-dialog.\n    min-width: 100%;\n  }\n}\n\n.sbb-calendar__table-header {\n  // Depending where in the DOM it's used, it's possible that browser defaults set text-align to left.\n  text-align: center;\n}\n\n.sbb-calendar__table-body {\n  // Depending where in the DOM it's used, it's possible that browser defaults set text-align to left.\n  text-align: center;\n}\n\n.sbb-calendar__table-header-cell,\n.sbb-calendar__table-header-cell-vertical {\n  width: var(--sbb-calendar-cell-size);\n  color: var(--sbb-calendar-header-color);\n  padding: 0;\n  font-size: var(--sbb-text-font-size-xs);\n  font-weight: normal;\n  letter-spacing: var(--sbb-typo-letter-spacing-text);\n\n  :host(:not([multiple])) & {\n    height: var(--sbb-calendar-cell-size);\n  }\n}\n\n.sbb-calendar__table-header-cell-vertical {\n  padding-block-end: var(--sbb-calendar-cell-padding-block-end);\n}\n\n.sbb-calendar__table-data {\n  position: relative;\n  padding: 0;\n  text-align: center;\n}\n\n.sbb-calendar__day-cell {\n  padding-block-end: var(--sbb-calendar-cell-padding-block-end);\n}\n\nsbb-calendar-day {\n  :host(:not(:state(enhanced))) & {\n    --sbb-calendar-cell-justify-content: center;\n    --sbb-calendar-day-height: #{sbb.px-to-rem-build(44)};\n    --sbb-calendar-day-extra-display: none;\n    --sbb-calendar-day-value-height: unset;\n    --sbb-calendar-day-crossed-out-top: 50%;\n    --sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);\n  }\n}\n\n@keyframes show {\n  from {\n    opacity: 0;\n    transform: translateY(var(--sbb-calendar-table-animation-shift));\n  }\n\n  to {\n    opacity: 1;\n    transform: translateY(0%);\n  }\n}\n\n@keyframes hide {\n  from {\n    opacity: 1;\n    transform: translateY(0%);\n  }\n\n  to {\n    opacity: 0;\n    transform: translateY(var(--sbb-calendar-table-animation-shift));\n  }\n}\n","import {\n  type CSSResultGroup,\n  html,\n  isServer,\n  nothing,\n  type PropertyValues,\n  type TemplateResult,\n} from 'lit';\nimport { property, state } from 'lit/decorators.js';\n\nimport { SbbSecondaryButtonElement } from '../../button.pure.ts';\nimport { isArrowKeyOrPageKeysPressed } from '../../core/a11y.ts';\nimport { SbbElement } from '../../core/base-elements.ts';\nimport { readConfig } from '../../core/config.ts';\nimport {\n  SbbLanguageController,\n  SbbMediaMatcherController,\n  SbbMediaQueryBreakpointLargeAndAbove,\n} from '../../core/controllers.ts';\nimport type { DateAdapter } from '../../core/datetime.ts';\nimport {\n  DAYS_PER_ROW,\n  defaultDateAdapter,\n  MONDAY,\n  MONTHS_PER_PAGE,\n  MONTHS_PER_ROW,\n  THURSDAY,\n  TUESDAY,\n  WEDNESDAY,\n  YEARS_PER_PAGE,\n  YEARS_PER_ROW,\n} from '../../core/datetime.ts';\nimport { forceType, handleDistinctChange, plainDate } from '../../core/decorators.ts';\nimport {\n  i18nCalendarDateSelection,\n  i18nCalendarWeekNumber,\n  i18nNextMonth,\n  i18nNextYear,\n  i18nNextYearRange,\n  i18nPreviousMonth,\n  i18nPreviousYear,\n  i18nPreviousYearRange,\n  i18nYearMonthSelection,\n} from '../../core/i18n.ts';\nimport type { SbbOrientation } from '../../core/interfaces.ts';\nimport { boxSizingStyles } from '../../core/styles.ts';\nimport { SbbCalendarDayElement } from '../calendar-day/calendar-day.component.ts';\nimport { SbbCalendarMonthElement } from '../calendar-month/calendar-month.component.ts';\nimport { SbbCalendarWeekdayElement } from '../calendar-weekday/calendar-weekday.component.ts';\nimport { SbbCalendarWeeknumberElement } from '../calendar-weeknumber/calendar-weeknumber.component.ts';\nimport { SbbCalendarYearElement } from '../calendar-year/calendar-year.component.ts';\nimport type { SbbCalendarCellBaseElement } from '../common.ts';\n\nimport style from './calendar.scss?lit&inline';\n\nimport '../../icon.ts';\nimport '../../screen-reader-only.ts';\n\n// TODO(breaking-change): Remove call to define.\nSbbSecondaryButtonElement.define();\n\nSbbCalendarDayElement.define();\nSbbCalendarMonthElement.define();\nSbbCalendarYearElement.define();\nSbbCalendarWeekdayElement.define();\nSbbCalendarWeeknumberElement.define();\n\nexport class SbbMonthChangeEvent extends Event {\n  private readonly _range: readonly Day[];\n\n  public get range(): readonly Day[] {\n    return this._range;\n  }\n\n  public constructor(range: readonly Day[]) {\n    super('monthchange', { bubbles: true, composed: true });\n    this._range = Object.freeze(range || []);\n  }\n}\n\n/**\n * Parameters needed in year and month views to correctly calculate the next element in keyboard navigation.\n *\n * The cell's index and the element's index in its month / year batch must be distinguished:\n * the first is the index of the element in the array of all the rendered cells, while the second is the index of the element relative to its table.\n * In non-wide mode, the wto are the same, while in wide mode the cell's index can go from 0 to 47 for years and from 0 to 23 for months,\n * while the element index goes from 0 to, respectively, 23 and 11.\n */\ninterface CalendarKeyboardNavigationMonthYearViewsParameters {\n  /** The element index within its year or month range. */\n  elementIndexForWideMode: number;\n  /** In wide mode, the index of the first element in the second panel, or, alternatively, the number of elements in the first panel. */\n  offsetForWideMode: number;\n  /** The index of the last element within the element's month (or year range). */\n  lastElementIndexForWideMode: number;\n  /** The number of cells displayed in a single row, depending on the rendered view. */\n  verticalOffset: number;\n}\n\n/**\n * Parameters needed in day view to correctly calculate the next element in keyboard navigation.\n *\n * In orientation='vertical', it's not possible to rely on any array/index to calculate the element to navigate to,\n * so calculations on dates must be done, which should consider view boundaries, offsets and month's length.\n */\ninterface CalendarKeyboardNavigationDayViewParameters {\n  /** The first day rendered. */\n  firstDayInView: string | null;\n  /** The last day rendered. It depends on the 'wide' value. */\n  lastDayInView: string | null;\n  /** The offset from the first day of the week (Monday) of the first rendered month. */\n  firstMonthOffset: number;\n  /** The number of days in the first rendered month. */\n  firstMonthLength: number;\n  /** The offset from the first day of the week (Monday) of the second rendered month. If wide is false, it's equal to zero. */\n  secondMonthOffset: number;\n}\n\nexport interface Day<T = Date> {\n  /** Date as ISO string. */\n  value: string;\n  dayValue: string;\n  monthValue: string;\n  yearValue: string;\n  dateValue: T;\n  weekValue: number;\n  weekDayValue: number;\n}\n\n/** @deprecated use MonthCell */\nexport interface Month {\n  value: string;\n  longValue: string;\n  monthValue: number;\n}\n\nexport interface MonthCell {\n  value: string;\n  monthValue: number;\n}\n\nexport interface Weekday {\n  long: string;\n  narrow: string;\n}\n\nexport type CalendarView = 'day' | 'month' | 'year';\n\n/**\n * It displays a calendar which allows choosing a date.\n *\n * @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.\n */\nexport class SbbCalendarElement<T = Date> extends SbbElement {\n  public static override readonly elementName: string = 'sbb-calendar';\n  public static override styles: CSSResultGroup = [boxSizingStyles, style];\n  public static readonly events = {\n    dateselected: 'dateselected',\n    monthchange: 'monthchange',\n  } as const;\n\n  /** If set to true, two months are displayed */\n  @forceType()\n  @property({ type: Boolean })\n  public accessor wide: boolean = false;\n\n  /** The initial view of the calendar which should be displayed on opening. */\n  @property() public accessor view: CalendarView = 'day';\n\n  /**\n   * The minimum valid date. Accepts a date object or null.\n   * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.\n   */\n  @plainDate()\n  @property()\n  public accessor min: T | null = null;\n\n  /**\n   * The maximum valid date. Accepts a date object or null.\n   * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.\n   */\n  @plainDate()\n  @property()\n  public accessor max: T | null = null;\n\n  /** Whether the calendar allows for multiple date selection. */\n  @forceType()\n  @handleDistinctChange((e: SbbCalendarElement<T>, newValue: boolean) =>\n    e._onMultipleChanged(newValue),\n  )\n  @property({ type: Boolean })\n  public accessor multiple: boolean = false;\n\n  /**\n   * The selected date: accepts a date object, or, if `multiple`, an array of dates.\n   */\n  @property()\n  public set selected(value: T | T[] | null) {\n    if (Array.isArray(value)) {\n      this._selected = value\n        .map((dateLike: T) =>\n          this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(dateLike)),\n        )\n        .filter((date: T | null): date is T => date !== null)\n        .filter(\n          (date: T) =>\n            !this._isDayInRange(this._dateAdapter.toIso8601(date)) || this._dateFilter(date),\n        );\n    } else {\n      const selectedDate = this._dateAdapter.getValidDateOrNull(\n        this._dateAdapter.deserialize(value),\n      );\n      if (\n        !!selectedDate &&\n        (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) ||\n          this._dateFilter(selectedDate))\n      ) {\n        this._selected = selectedDate;\n      } else {\n        this._selected = null;\n      }\n    }\n  }\n  public get selected(): T | T[] | null {\n    return this._selected;\n  }\n  @state() private accessor _selected: T | T[] | null = null;\n\n  /** A function used to filter out dates. */\n  @property({ attribute: 'date-filter' })\n  public accessor dateFilter: ((date: T | null) => boolean) | null = null;\n\n  /** The orientation of days in the calendar. */\n  @property({ reflect: true }) public accessor orientation: SbbOrientation = 'horizontal';\n\n  /** Whether it has to display the week numbers in addition to week days. */\n  @forceType()\n  @property({ attribute: 'week-numbers', type: Boolean })\n  public accessor weekNumbers: boolean = false;\n\n  private _dateAdapter: DateAdapter<T> = readConfig().datetime?.dateAdapter ?? defaultDateAdapter;\n\n  /** The currently active date. */\n  @state() private accessor _activeDate: T = this._dateAdapter.today();\n\n  /** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */\n  @state()\n  private set _wide(wide: boolean) {\n    this.toggleState('wide', wide);\n    this._wideInternal = wide;\n  }\n  private get _wide(): boolean {\n    return this._wideInternal;\n  }\n  // We keep the state in a field because of the WebKit bug https://bugs.webkit.org/show_bug.cgi?id=303467.\n  // TODO: re-check whether field is needed\n  private _wideInternal: boolean = false;\n\n  @state() private accessor _calendarView: CalendarView = 'day';\n\n  private _nextCalendarView: CalendarView = 'day';\n\n  /** Information about the rendered day view; used in keyboard navigation. */\n  private _keyboardNavigationDayViewParameters: CalendarKeyboardNavigationDayViewParameters = {\n    firstDayInView: null,\n    lastDayInView: null,\n    firstMonthOffset: 0,\n    firstMonthLength: 0,\n    secondMonthOffset: 0,\n  };\n\n  /** A list of days, in two formats (long and single char). */\n  private _weekdays!: Weekday[];\n\n  /** Grid of calendar cells representing the dates of the month. */\n  private _weeks: Day<T>[][] = [];\n\n  /** Grid of calendar cells representing months. */\n  private _months!: MonthCell[][];\n\n  /** Grid of calendar cells representing years. */\n  private _years!: number[][];\n\n  /** Grid of calendar cells representing years for the wide view. */\n  private _nextMonthYears!: number[][];\n\n  /** Grid of calendar cells representing the dates of the next month. */\n  private _nextMonthWeeks!: Day<T>[][];\n\n  /** An array containing all the month names in the current language. */\n  private _monthNames: string[] = this._dateAdapter.getMonthNames('long');\n\n  /** An array containing the weeks' numbers for the current month. */\n  private _weekNumbers!: number[];\n\n  /** An array containing the weeks' numbers for the next month in wide mode. */\n  private _nextMonthWeekNumbers!: number[];\n\n  private _enhancedVariant: boolean = false;\n\n  /** A list of calendar's cells corresponding to days, months or years depending on the view. */\n  private get _cells(): SbbCalendarCellBaseElement[] {\n    return Array.from<SbbCalendarCellBaseElement>(\n      (this._calendarView === 'day'\n        ? (Array.from(this.shadowRoot!.querySelectorAll('slot')).flatMap((e: HTMLSlotElement) =>\n            e.assignedElements({ flatten: true }),\n          ) as SbbCalendarDayElement[])\n        : this.shadowRoot?.querySelectorAll(`sbb-calendar-${this._calendarView}`)) ?? [],\n    );\n  }\n\n  /** The chosen year in the year selection view. */\n  private _chosenYear?: number;\n\n  /** The chosen month in the year selection view. */\n  private _chosenMonth?: number;\n\n  /** Whether the focus should be reset on focusCell. */\n  private _resetFocus = false;\n\n  /** Whether an element inside the calendar is currently focused. */\n  private _containingFocus = false;\n\n  @state()\n  private accessor _initialized = false;\n\n  private _language = new SbbLanguageController(this).withHandler(() => {\n    this._monthNames = this._dateAdapter.getMonthNames('long');\n    this._createMonthRows();\n  });\n  private _mediaMatcher = new SbbMediaMatcherController(this, {\n    [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init(),\n  });\n\n  public constructor() {\n    super();\n    this._createMonthRows();\n    this._setWeekdays();\n\n    // We need to track the focus as we should only take focus into the calendar, when the\n    // focus was once set into the calendar.\n    // For shadow DOM compatibility we need to track this programmatically.\n    this.addEventListener('focusin', () => (this._containingFocus = true));\n    this.addEventListener('focusout', () => (this._containingFocus = false));\n    this.addEventListener('click', (e) => {\n      const day = (e.target as HTMLElement).closest<SbbCalendarDayElement<T>>('sbb-calendar-day');\n      if (day) {\n        this._selectDate(day.value!);\n      }\n    });\n    this.addEventListener('keydown', (e) => {\n      if ((e.target as HTMLElement).localName === 'sbb-calendar-day') {\n        this._handleKeyboardEvent(\n          e,\n          this._mapDateToDay((e.target as SbbCalendarDayElement).value! as T),\n        );\n      }\n    });\n  }\n\n  private _dateFilter(date: T): boolean {\n    return this.dateFilter?.(date) ?? true;\n  }\n\n  /** Resets the active month according to the new state of the calendar. */\n  public resetPosition(): void {\n    this._resetCalendarView();\n    this._init();\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    this.resetPosition();\n    this.shadowRoot?.addEventListener('slotchange', this._onSlotChange, { capture: true });\n  }\n\n  public override disconnectedCallback(): void {\n    super.disconnectedCallback();\n    this.shadowRoot?.removeEventListener('slotchange', this._onSlotChange, { capture: true });\n  }\n\n  /** @internal */\n  public override focus(): void {\n    this._resetFocus = true;\n    this._focusCell();\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (!this._initialized) {\n      return;\n    }\n\n    if (changedProperties.has('wide') || changedProperties.has('orientation')) {\n      this.resetPosition();\n    }\n\n    if (changedProperties.has('view')) {\n      this._setChosenYear();\n      this._chosenMonth = undefined;\n      this._nextCalendarView = this._calendarView = this.view;\n    }\n  }\n\n  protected override updated(changedProperties: PropertyValues<this>): void {\n    super.updated(changedProperties);\n    // The calendar needs to calculate tab-indexes on first render,\n    // and every time a date is selected or the month view changes.\n    this._setTabIndex();\n\n    // When changing view to year/month, the tabindex is changed, but the focused element is getting lost.\n    // We need to call `_focusCell()` method explicitly to correctly set the focus.\n    this._focusCell();\n  }\n\n  private _onSlotChange = (): void => {\n    this._enhancedVariant = Array.from(this.children).some(\n      (c) => c.localName === 'sbb-calendar-day',\n    );\n    this.toggleState('enhanced', this._enhancedVariant);\n    this._setTabIndex();\n  };\n\n  /**\n   * The `_selected` state should be adapted when the `multiple` property changes:\n   *   - if it changes to true, the '_selected' is set to an array;\n   *   - if it changes to false, the first available option is set as 'value' otherwise it's set to null.\n   */\n  private _onMultipleChanged(isMultiple: boolean): void {\n    if (isMultiple && !Array.isArray(this._selected)) {\n      this._selected = this._selected ? [this._selected as T] : [];\n    }\n    if (!isMultiple && Array.isArray(this._selected)) {\n      this._selected = (this._selected as T[]).length ? (this._selected as T[])[0] : null;\n    }\n  }\n\n  /** Initializes the component. */\n  private _init(activeDate?: T): void {\n    // Due to its complexity, the calendar is only initialized on client side\n    if (isServer) {\n      return;\n    } else if (this.hydrationRequired) {\n      this.hydrationComplete.then(() => this._init());\n      return;\n    }\n\n    if (activeDate) {\n      this._assignActiveDate(activeDate);\n    }\n    this._wide =\n      (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;\n    this._weeks = this._createWeekRows(this._activeDate);\n    this._years = this._createYearRows();\n    this._weekNumbers = this._weeks\n      .flat()\n      .sort((a, b) => a.value.localeCompare(b.value))\n      .map((day: Day<T>) => day.weekValue)\n      .filter((v, i, a) => a.indexOf(v) === i);\n    this._nextMonthWeeks = [[]];\n    this._nextMonthYears = [[]];\n    if (this._wide) {\n      const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);\n      this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);\n      this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);\n      this._nextMonthWeekNumbers = this._nextMonthWeeks\n        .flat()\n        .sort((a, b) => a.value.localeCompare(b.value))\n        .map((day: Day<T>) => day.weekValue)\n        .filter((v, i, a) => a.indexOf(v) === i);\n    }\n    this._initialized = true;\n  }\n\n  /** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */\n  private _focusCell(): void {\n    if (this._resetFocus) {\n      this._getFirstFocusable()?.focus();\n      this._resetFocus = false;\n    }\n  }\n\n  /** Creates the array of weekdays. */\n  private _setWeekdays(): void {\n    const narrowWeekdays: string[] = this._dateAdapter.getDayOfWeekNames('narrow');\n    const longWeekdays: string[] = this._dateAdapter.getDayOfWeekNames('long');\n    const weekdays: Weekday[] = longWeekdays.map((long: string, i: number) => ({\n      long,\n      narrow: narrowWeekdays[i],\n    }));\n\n    // Rotates the labels for days of the week based on the configured first day of the week.\n    const firstDayOfWeek: number = this._dateAdapter.getFirstDayOfWeek();\n    this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));\n  }\n\n  /** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */\n  private _createWeekRows(value: T, isSecondMonthInView = false): Day<T>[][] {\n    const daysInMonth: number = this._dateAdapter.getNumDaysInMonth(value);\n    const weekOffset: number = this._dateAdapter.getFirstWeekOffset(value);\n    if (!isSecondMonthInView) {\n      this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;\n      this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;\n      this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(\n        this._dateAdapter.createDate(\n          this._dateAdapter.getYear(value),\n          this._dateAdapter.getMonth(value),\n          1,\n        ),\n      );\n      this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(\n        this._dateAdapter.createDate(\n          this._dateAdapter.getYear(value),\n          this._dateAdapter.getMonth(value),\n          daysInMonth,\n        ),\n      );\n    } else {\n      this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;\n      this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(\n        this._dateAdapter.createDate(\n          this._dateAdapter.getYear(value),\n          this._dateAdapter.getMonth(value),\n          daysInMonth,\n        ),\n      );\n    }\n    return this.orientation === 'horizontal'\n      ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset)\n      : this._createWeekRowsVertical(value, daysInMonth, weekOffset);\n  }\n\n  /**\n   * Creates the rows for each week in orientation='horizontal'.\n   *\n   * Iterates through the days of the months, creates a Day object for each and pushes it into and array.\n   * Each seven days (considering the offset at the beginning of the month) restarts from an empty array.\n   *\n   * The result is a matrix in which every row is a week (or part of it, considering offset).\n   */\n  private _createWeekRowsHorizontal(value: T, daysInMonth: number, weekOffset: number): Day<T>[][] {\n    const weeks: Day<T>[][] = [[]];\n    for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {\n      if (cell === DAYS_PER_ROW) {\n        weeks.push([]);\n        cell = 0;\n      }\n      const date = this._dateAdapter.createDate(\n        this._dateAdapter.getYear(value),\n        this._dateAdapter.getMonth(value),\n        i + 1,\n      )!;\n      weeks[weeks.length - 1].push(this._mapDateToDay(date));\n    }\n    return weeks;\n  }\n\n  /**\n   * Creates the rows for each week in orientation='vertical'.\n   *\n   * Creates a matrix with seven empty rows.\n   * Iterates through the days of the months, creates a Day object for each\n   * and pushes it into the correct array considering the offset at the beginning of the month.\n   * Each seven days (including offset) restarts from the first.\n   *\n   * The result is a matrix in which every row is a set of weekdays, so:\n   * - row 0: all the Mondays;\n   * - row 1: all the Tuesdays;\n   * - ...\n   * - row 7: all the Sundays.\n   */\n  private _createWeekRowsVertical(value: T, daysInMonth: number, weekOffset: number): Day<T>[][] {\n    const weeks: Day<T>[][] = Array.from({ length: DAYS_PER_ROW }, () => []);\n    for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {\n      if (cell === DAYS_PER_ROW) {\n        cell = 0;\n      }\n      const date = this._dateAdapter.createDate(\n        this._dateAdapter.getYear(value),\n        this._dateAdapter.getMonth(value),\n        i + 1,\n      )!;\n      weeks[cell].push(this._mapDateToDay(date));\n    }\n    return weeks;\n  }\n\n  private _mapDateToDay(date: T): Day<T> {\n    const isoDate = this._dateAdapter.toIso8601(date);\n    return {\n      value: isoDate,\n      dateValue: date,\n      dayValue: String(this._dateAdapter.getDate(date)),\n      monthValue: String(this._dateAdapter.getMonth(date)),\n      yearValue: String(this._dateAdapter.getYear(date)),\n      // TODO: Improve performance of this, by keeping track of the\n      // week number while iterating through the days.\n      weekValue: this._getWeek(date),\n      weekDayValue: this._dateAdapter.getDayOfWeek(date),\n    };\n  }\n\n  private _getWeek(date: T): number {\n    const firstDayOfYear = this._dateAdapter.createDate(this._dateAdapter.getYear(date), 1, 1);\n    const weekday = this._dateAdapter.getDayOfWeek(firstDayOfYear);\n\n    let weekIndex = [MONDAY, TUESDAY, WEDNESDAY, THURSDAY].includes(weekday) ? 1 : 0;\n    let weekStart = this._dateAdapter.addCalendarDays(\n      firstDayOfYear,\n      this._dateAdapter.getFirstWeekOffset(firstDayOfYear) * -1,\n    );\n    while (this._dateAdapter.compareDate(weekStart, date) <= 0) {\n      const weekEnd = this._dateAdapter.addCalendarDays(weekStart, 6);\n      if (this._dateAdapter.compareDate(date, weekEnd) <= 0) {\n        return weekIndex > 0\n          ? weekIndex\n          : this._getWeek(this._dateAdapter.addCalendarDays(firstDayOfYear, -1));\n      }\n      weekStart = this._dateAdapter.addCalendarDays(weekStart, DAYS_PER_ROW);\n      weekIndex++;\n    }\n\n    throw new Error('The provided date is invalid');\n  }\n\n  /** Force the conversion to ISO8601 formatted string. */\n  private _mapValueToISODate(value: string | Date): string {\n    return typeof value === 'string' ? value : this._dateAdapter.toIso8601(value as T);\n  }\n\n  /** Creates the rows for the month selection view. */\n  private _createMonthRows(): void {\n    const months: MonthCell[] = new Array(12).fill(null).map(\n      (_, i: number): MonthCell => ({\n        value: String(i + 1).padStart(2, '0'),\n        monthValue: i + 1,\n      }),\n    );\n    const rows: number = 12 / MONTHS_PER_ROW;\n    const monthArray: MonthCell[][] = [];\n    for (let i: number = 0; i < rows; i++) {\n      monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));\n    }\n    this._months = monthArray;\n  }\n\n  /** Creates the rows for the year selection view. */\n  private _createYearRows(offset: number = 0): number[][] {\n    const startValueYearView: number = this._getStartValueYearView();\n    const allYears: number[] = new Array(YEARS_PER_PAGE)\n      .fill(0)\n      .map((_, i: number) => startValueYearView + offset + i);\n    const rows: number = YEARS_PER_PAGE / YEARS_PER_ROW;\n    const yearArray: number[][] = [];\n    for (let i: number = 0; i < rows; i++) {\n      yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));\n    }\n    return yearArray;\n  }\n\n  /**\n   * Calculates the first year that will be shown in the year selection panel.\n   * If `minDate` and `maxDate` are both null, the starting year is calculated as\n   * the multiple of YEARS_PER_PAGE closest to and less than activeDate,\n   * e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),\n   * while with `activeDate` = 2000, the function will return 1992 (24 * 82).\n   * If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,\n   * it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.\n   * If both are not null, `maxDate` has priority over `minDate`.\n   */\n  private _getStartValueYearView(): number {\n    let startingYear = 0;\n    if (this.max) {\n      startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;\n    } else if (this.min) {\n      startingYear = this._dateAdapter.getYear(this.min);\n    }\n    const activeYear = this._dateAdapter.getYear(this._activeDate);\n    return (\n      activeYear -\n      ((((activeYear - startingYear) % YEARS_PER_PAGE) + YEARS_PER_PAGE) % YEARS_PER_PAGE)\n    );\n  }\n\n  /** Checks if date is within the min-max range. */\n  private _isDayInRange(dateString: string): boolean {\n    if (!this.min && !this.max) {\n      return true;\n    }\n    const date = this._dateAdapter.deserialize(dateString)!;\n    return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));\n  }\n\n  /** Emits the selected date and sets it internally. */\n  private _selectDate(day: T): void {\n    this._chosenMonth = undefined;\n    this._setChosenYear();\n    if (this.multiple) {\n      // Check if _selected has elements\n      if (this._selected && (this._selected as T[]).length > 0) {\n        const indexOfSelectedDay: number = (this._selected as T[]).findIndex(\n          (sel) => this._dateAdapter.compareDate(sel, day) === 0,\n        );\n        // If the selected date is already in the _selected array, remove it, otherwise add it\n        if (indexOfSelectedDay !== -1) {\n          this._selected = (this._selected as T[]).filter((_, i) => i !== indexOfSelectedDay);\n        } else {\n          this._selected = [...(this._selected as T[]), day];\n        }\n      } else {\n        // If _selected is empty, set it\n        this._selected = [day];\n      }\n      this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)!));\n    } else {\n      // In single selection, check if the day is already selected\n      if (!this._selected || this._dateAdapter.compareDate(this._selected as T, day) !== 0) {\n        this._selected = day;\n        this._emitDateSelectedEvent(this._dateAdapter.deserialize(day)!);\n      }\n    }\n  }\n\n  /**\n   * Handle multiple dates selection via weekNumber / weekDay buttons:\n   * - if Cmd or Ctrl are pressed, add the new date to the current ones;\n   * - if not,\n   *     - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;\n   *     - if not, the selected dates are the new ones.\n   */\n  private _selectMultipleDates(days: Day<T>[]): void {\n    // Filter disabled days by matching the provided `days` parameter against the enabled cells.\n    // Since the buttons' value is set to the Day's interface value (ISO string), there's no need to deserialize it.\n    const enabledDays: string[] = (this._cells as SbbCalendarDayElement[])\n      .filter((e) => !e.disabled)\n      .map((e) => this._mapValueToISODate(e.value!));\n    const daysToAdd: string[] = days\n      .map((e: Day<T>) => e.value)\n      .filter((isoDate: string) => enabledDays.includes(isoDate));\n    const daysToAddSet = new Set(daysToAdd);\n    const selectedSet = new Set((this._selected as T[]).map((s) => this._dateAdapter.toIso8601(s)));\n    const selStrings = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet);\n    this._selected = selStrings.map((s) => this._dateAdapter.deserialize(s)!);\n\n    this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)!));\n  }\n\n  /**\n   * Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).\n   */\n  private _emitDateSelectedEvent(detail: T | T[]): void {\n    /** @type {CustomEvent<T | T[]>} Event emitted on date selection. */\n    this.dispatchEvent(\n      new CustomEvent<T | T[]>('dateselected', {\n        detail,\n        composed: true,\n        bubbles: true,\n      }),\n    );\n  }\n\n  private _emitMonthChange(): void {\n    // FIXME: the name of this variable appears as event name in the readme\n    //  due to a bug in the custom-elements-manifest library.\n    //  https://github.com/open-wc/custom-elements-manifest/issues/149\n    const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks)\n      .flat()\n      .sort((a, b) => a.value.localeCompare(b.value)) as Day[];\n    /**\n     * @type {SbbMonthChangeEvent}\n     * Emits when the month changes.\n     * The `range` property contains the days array of the chosen month.\n     */\n    this.dispatchEvent(new SbbMonthChangeEvent(monthchange));\n  }\n\n  /**\n   * In case of multiple selection, newly added days must be added to the existing ones, without duplication.\n   * If the days to add are exactly the same as the selected ones, the set must be emptied.\n   */\n  private _updateSelectedWithMultipleDates(\n    daysToAdd: string[],\n    daysToAddSet: Set<string>,\n    selectedSet: Set<string>,\n  ): string[] {\n    if (daysToAdd.every((day: string) => selectedSet.has(day))) {\n      daysToAddSet.forEach((day: string) => selectedSet.delete(day));\n    } else {\n      daysToAddSet.forEach((day: string) => selectedSet.add(day));\n    }\n    return Array.from(selectedSet);\n  }\n\n  private _setChosenYear(): void {\n    if (this.view === 'month') {\n      let selectedDate: T | undefined;\n      if (this.multiple) {\n        selectedDate = (this.selected as T[]).at(-1);\n      } else {\n        selectedDate = this.selected as T;\n      }\n      this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());\n    } else {\n      this._chosenYear = undefined;\n    }\n  }\n\n  private _assignActiveDate(date: T): void {\n    if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {\n      this._activeDate = this.min;\n      return;\n    }\n    if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {\n      this._activeDate = this.max;\n      return;\n    }\n    this._activeDate = date;\n  }\n\n  /** Goes to the month identified by the shift. */\n  private _goToDifferentMonth(months: number): void {\n    this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));\n    this._emitMonthChange();\n  }\n\n  private _goToDifferentYear(years: number): void {\n    this._chosenYear! += years;\n    // Can't use `_assignActiveDate(...)` here, because it will set it to min/max value if argument is out of range\n    this._activeDate = this._dateAdapter.createDate(\n      this._chosenYear!,\n      this._dateAdapter.getMonth(this._activeDate),\n      this._dateAdapter.getDate(this._activeDate),\n    );\n    this._init();\n  }\n\n  private _goToDifferentYearRange(years: number): void {\n    this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));\n  }\n\n  private _prevDisabled(prevDate: T): boolean {\n    if (!this.min) {\n      return false;\n    }\n    return this._dateAdapter.compareDate(prevDate, this.min) < 0;\n  }\n\n  private _nextDisabled(nextDate: T): boolean {\n    if (!this.max) {\n      return false;\n    }\n    return this._dateAdapter.compareDate(nextDate, this.max) > 0;\n  }\n\n  /** Checks if the \"previous month\" button should be disabled. */\n  private _previousMonthDisabled(): boolean {\n    const prevMonth = this._dateAdapter.addCalendarDays(\n      this._activeDate,\n      this._dateAdapter.getDate(this._activeDate) * -1,\n    );\n    return this._prevDisabled(prevMonth);\n  }\n\n  /** Checks if the \"next month\" button should be disabled. */\n  private _nextMonthDisabled(): boolean {\n    let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);\n    nextMonth = this._dateAdapter.createDate(\n      this._dateAdapter.getYear(nextMonth),\n      this._dateAdapter.getMonth(nextMonth),\n      1,\n    );\n    return this._nextDisabled(nextMonth);\n  }\n\n  /** Checks if the \"previous year\" button should be disabled. */\n  private _previousYearDisabled(): boolean {\n    const prevYear = this._dateAdapter.createDate(\n      this._dateAdapter.getYear(this._activeDate) - 1,\n      12,\n      31,\n    );\n    return this._prevDisabled(prevYear);\n  }\n\n  /** Checks if the \"next year\" button should be disabled. */\n  private _nextYearDisabled(): boolean {\n    const nextYear = this._dateAdapter.createDate(\n      this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1),\n      1,\n      1,\n    );\n    return this._nextDisabled(nextYear);\n  }\n\n  /** Checks if the \"previous year\" button should be disabled in year view. */\n  private _previousYearRangeDisabled(): boolean {\n    const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);\n    return this._prevDisabled(prevYear);\n  }\n\n  /** Checks if the \"next year\" button should be disabled in year view. */\n  private _nextYearRangeDisabled(): boolean {\n    const years = this._wide ? this._nextMonthYears : this._years;\n    const lastYearRange = years[years.length - 1];\n    const lastYear = lastYearRange[lastYearRange.length - 1];\n    const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);\n    return this._nextDisabled(nextYear);\n  }\n\n  private _handleTableBlur(eventTarget: HTMLElement): void {\n    if (eventTarget?.localName !== 'sbb-calendar-day') {\n      this._setTabIndex();\n    }\n  }\n\n  private _setTabIndex(): void {\n    Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach(\n      (day) => (day.tabIndex = -1),\n    );\n    const firstFocusable = this._getFirstFocusable();\n    if (firstFocusable) {\n      firstFocusable.tabIndex = 0;\n    }\n  }\n\n  /** Get the element in the calendar to assign focus. */\n  private _getFirstFocusable(): SbbCalendarCellBaseElement | null {\n    if (this._calendarView === 'day') {\n      const selectedOrCurrent =\n        this._cells.find((e) => e.matches(':state(selected)')) ??\n        this._cells.find((e) => e.matches(':state(current)'));\n      return selectedOrCurrent && !selectedOrCurrent.disabled\n        ? selectedOrCurrent\n        : this._getFirstFocusableDay();\n    } else {\n      const selectedOrCurrent =\n        this.shadowRoot?.querySelector<SbbCalendarCellBaseElement>(':state(selected)') ??\n        this.shadowRoot?.querySelector<SbbCalendarCellBaseElement>(':state(current)');\n      return selectedOrCurrent && !selectedOrCurrent.disabled\n        ? selectedOrCurrent\n        : this.shadowRoot!.querySelector<SbbCalendarCellBaseElement>(\n            `sbb-calendar-${this._calendarView}:not([disabled])`,\n          );\n    }\n  }\n\n  /**\n   * In `day` view in `vertical` orientation,\n   * if the first of the month is not a Monday, it is not the first rendered element in the table,\n   * so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.\n   *\n   * To solve this, the element with the lowest `value` is taken (ISO String are ordered).\n   */\n  private _getFirstFocusableDay(): SbbCalendarDayElement | null {\n    const cells = this._cells as SbbCalendarDayElement[];\n    const daysInView = cells.filter((e) => !e.disabled);\n    if (!daysInView || daysInView.length === 0) {\n      return null;\n    } else {\n      const firstElement = daysInView\n        .map((e): string => this._dateAdapter.toIso8601(e.value! as T))\n        .sort()[0];\n      return cells.find((e) => e.matches(`[slot=\"${firstElement}\"]`))! ?? null;\n    }\n  }\n\n  private _handleKeyboardEvent(event: KeyboardEvent, day?: Day<T>): void {\n    if (isArrowKeyOrPageKeysPressed(event)) {\n      event.preventDefault();\n    }\n    // Gets the currently rendered table's cell;\n    // they could be days, months or years based on the current selection view.\n    // If `wide` is true, years are doubled in number and days are (roughly) doubled too, affecting the `index` calculation.\n    const cells = this._cells;\n    const index: number = cells.findIndex((e) => e === event.target);\n    let nextEl: SbbCalendarCellBaseElement;\n    if (day) {\n      nextEl = this._navigateByKeyboardDayView(event, index, cells as SbbCalendarDayElement[], day);\n    } else {\n      nextEl = this._navigateByKeyboard(event, index, cells);\n    }\n    const activeEl = (this._enhancedVariant ? document : this.shadowRoot!)\n      .activeElement as SbbCalendarCellBaseElement;\n    if (nextEl !== activeEl) {\n      nextEl.tabIndex = 0;\n      nextEl?.focus();\n      activeEl.tabIndex = -1;\n    }\n  }\n\n  private _navigateByKeyboardDayView(\n    evt: KeyboardEvent,\n    index: number,\n    cells: SbbCalendarDayElement[],\n    day: Day<T>,\n  ): SbbCalendarDayElement {\n    const arrowsOffset =\n      this.orientation === 'horizontal'\n        ? { leftRight: 1, upDown: DAYS_PER_ROW }\n        : { leftRight: DAYS_PER_ROW, upDown: 1 };\n    const offsetForVertical: number =\n      index < this._keyboardNavigationDayViewParameters.firstMonthLength\n        ? this._keyboardNavigationDayViewParameters.firstMonthOffset\n        : this._keyboardNavigationDayViewParameters.secondMonthOffset;\n\n    switch (evt.key) {\n      case 'ArrowUp':\n        return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);\n      case 'ArrowDown':\n        return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);\n      case 'ArrowLeft':\n        return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);\n      case 'ArrowRight':\n        return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);\n      case 'PageUp': {\n        if (this.orientation === 'horizontal') {\n          const firstOfWeek: number = +day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW;\n          const delta: number = firstOfWeek - +day.dayValue;\n          return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);\n        } else {\n          const weekNumber: number = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW);\n          const firstOfWeek: number = (weekNumber - 1) * DAYS_PER_ROW - offsetForVertical + 1;\n          const delta: number = firstOfWeek - +day.dayValue;\n          return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);\n        }\n      }\n      case 'PageDown': {\n        if (this.orientation === 'horizontal') {\n          const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;\n          const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;\n          const firstNextMonth: T = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);\n          const lastOfMonth: number = this._dateAdapter.getDate(\n            this._dateAdapter.addCalendarDays(firstNextMonth, -1),\n          );\n          const delta: number =\n            Math.trunc((lastOfMonth - +day.dayValue!) / DAYS_PER_ROW) * DAYS_PER_ROW;\n          return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);\n        } else {\n          const weekNumber: number = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW);\n          const lastOfWeek: number = weekNumber * DAYS_PER_ROW - offsetForVertical;\n          const delta: number = lastOfWeek - +day.dayValue;\n          return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);\n        }\n      }\n      case 'Home': {\n        return this._findDayFirst(cells, index, day, 1);\n      }\n      case 'End': {\n        const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;\n        const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;\n        const firstNextMonth: T = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);\n        return this._findDayLast(cells, index, firstNextMonth);\n      }\n      default:\n        return cells[index];\n    }\n  }\n\n  private _isDayOutOfView(date: string): boolean {\n    return (\n      date < this._keyboardNavigationDayViewParameters.firstDayInView! ||\n      date > this._keyboardNavigationDayViewParameters.lastDayInView!\n    );\n  }\n\n  private _findDayArrows(\n    cells: SbbCalendarDayElement[],\n    index: number,\n    date: T,\n    delta: number,\n  ): SbbCalendarDayElement {\n    const newDateValue = this._dateAdapter.toIso8601(\n      this._dateAdapter.addCalendarDays(date, delta),\n    );\n    if (this._isDayOutOfView(newDateValue)) {\n      return cells[index];\n    }\n    const nextCell = cells.find((e) => this._mapValueToISODate(e.value!) === newDateValue);\n    if (!nextCell || nextCell.disabled) {\n      return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue)!, delta);\n    }\n    return nextCell;\n  }\n\n  private _findDayPageUpDown(\n    cells: SbbCalendarDayElement[],\n    index: number,\n    day: Day<T>,\n    delta: number,\n    deltaIfDisabled: number,\n  ): SbbCalendarDayElement {\n    const newDateValue = this._dateAdapter.toIso8601(\n      this._dateAdapter.addCalendarDays(day.dateValue, delta),\n    );\n    if (this._isDayOutOfView(newDateValue)) {\n      return cells[index];\n    }\n    const nextCell = cells.find((e) => this._mapValueToISODate(e.value!) === newDateValue);\n    if (!nextCell || nextCell.disabled) {\n      return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);\n    }\n    return nextCell;\n  }\n\n  private _findDayFirst(\n    cells: SbbCalendarDayElement[],\n    index: number,\n    day: Day<T>,\n    date: number,\n  ): SbbCalendarDayElement {\n    const newDateValue = this._dateAdapter.toIso8601(\n      this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date),\n    );\n    if (this._isDayOutOfView(newDateValue)) {\n      return cells[index];\n    }\n    const nextCell = cells.find((e) => this._mapValueToISODate(e.value!) === newDateValue);\n    if (!nextCell || nextCell.disabled) {\n      return this._findDayFirst(cells, index, day, date + 1);\n    }\n    return nextCell;\n  }\n\n  private _findDayLast(\n    cells: SbbCalendarDayElement[],\n    index: number,\n    firstNextMonth: T,\n  ): SbbCalendarDayElement {\n    const newDateValue = this._dateAdapter.toIso8601(\n      this._dateAdapter.addCalendarDays(firstNextMonth, -1),\n    );\n    if (this._isDayOutOfView(newDateValue)) {\n      return cells[index];\n    }\n    const nextCell = cells.find((e) => this._mapValueToISODate(e.value!) === newDateValue);\n    if (!nextCell || nextCell.disabled) {\n      return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue)!);\n    }\n    return nextCell;\n  }\n\n  /**\n   * Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),\n   * the keyboard input and the position of the current element in the list.\n   * In the day view, the `day?: Day` parameter is mandatory for calculation,\n   * while in month and year view it's not due to the fixed amount of rendered cells.\n   */\n  private _navigateByKeyboard(\n    evt: KeyboardEvent,\n    index: number,\n    cells: SbbCalendarCellBaseElement[],\n  ): SbbCalendarCellBaseElement {\n    const {\n      elementIndexForWideMode,\n      offsetForWideMode,\n      lastElementIndexForWideMode,\n      verticalOffset,\n    }: CalendarKeyboardNavigationMonthYearViewsParameters =\n      this._calculateParametersForKeyboardNavigation(index, this._calendarView === 'year');\n\n    switch (evt.key) {\n      case 'ArrowUp':\n        return this._findNext(cells, index, -verticalOffset);\n      case 'ArrowDown':\n        return this._findNext(cells, index, verticalOffset);\n      case 'ArrowLeft':\n        return this._findNext(cells, index, -1);\n      case 'ArrowRight':\n        return this._findNext(cells, index, 1);\n      case 'Home':\n        return this._findFirst(cells, offsetForWideMode);\n      case 'PageUp':\n        return this._findFirstOnColumn(\n          cells,\n          elementIndexForWideMode,\n          offsetForWideMode,\n          verticalOffset,\n        );\n      case 'PageDown':\n        return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);\n      case 'End':\n        return this._findLast(cells, lastElementIndexForWideMode - 1);\n      default:\n        return cells[index];\n    }\n  }\n\n  /**\n   * Calculates the parameters needed in keyboard navigation in year and month view.\n   * @param index The starting element's index in the cell array.\n   * @param isYearView Whether the displayed `view` is the year one.\n   */\n  private _calculateParametersForKeyboardNavigation(\n    index: number,\n    isYearView: boolean,\n  ): CalendarKeyboardNavigationMonthYearViewsParameters {\n    const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;\n    const offset: number = Math.trunc(index / elementsPerPage) * elementsPerPage;\n    const indexInView: number = offset === 0 ? index : index - elementsPerPage;\n    return {\n      verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,\n      elementIndexForWideMode: indexInView,\n      offsetForWideMode: index - indexInView,\n      lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2,\n    };\n  }\n\n  /**\n   * Gets the next element of the provided array starting from `index` by adding `delta`.\n   * If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.\n   */\n  private _findNext(\n    days: SbbCalendarCellBaseElement[],\n    index: number,\n    delta: number,\n  ): SbbCalendarCellBaseElement {\n    let nextIndex = index + delta;\n    while (nextIndex < days.length && days[nextIndex]?.disabled) {\n      nextIndex += delta;\n    }\n    return days[nextIndex] ?? days[index];\n  }\n\n  /** Find the first enabled element in the provided array. */\n  private _findFirst(\n    days: SbbCalendarCellBaseElement[],\n    firstOfCurrentMonth: number,\n  ): SbbCalendarCellBaseElement {\n    return !days[firstOfCurrentMonth].disabled\n      ? days[firstOfCurrentMonth]\n      : this._findNext(days, firstOfCurrentMonth, 1);\n  }\n\n  /** Find the last enabled element in the provided array. */\n  private _findLast(\n    days: SbbCalendarCellBaseElement[],\n    lastOfCurrentMonth: number,\n  ): SbbCalendarCellBaseElement {\n    return !days[lastOfCurrentMonth].disabled\n      ? days[lastOfCurrentMonth]\n      : this._findNext(days, lastOfCurrentMonth, -1);\n  }\n\n  /** Find the first enabled element in the same column of the provided array. */\n  private _findFirstOnColumn(\n    days: SbbCalendarCellBaseElement[],\n    index: number,\n    offset: number,\n    verticalOffset: number,\n  ): SbbCalendarCellBaseElement {\n    const nextIndex = (index % verticalOffset) + offset;\n    return !days[nextIndex].disabled\n      ? days[nextIndex]\n      : this._findNext(days, nextIndex, verticalOffset);\n  }\n\n  /** Find the last enabled element in the same column of the provided array. */\n  private _findLastOnColumn(\n    days: SbbCalendarCellBaseElement[],\n    index: number,\n    offset: number,\n    verticalOffset: number,\n  ): SbbCalendarCellBaseElement {\n    const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;\n    return !days[nextIndex].disabled\n      ? days[nextIndex]\n      : this._findNext(days, nextIndex, -verticalOffset);\n  }\n\n  private _resetCalendarViewAndEmitMonthChange(initTransition = false): void {\n    this._resetCalendarView(initTransition);\n    this._emitMonthChange();\n  }\n\n  private _resetCalendarView(initTransition = false): void {\n    if (this._containingFocus) {\n      this._resetFocus = true;\n    }\n    this._activeDate =\n      (this.multiple ? (this._selected as T[]).at(-1) : (this._selected as T)) ??\n      this._dateAdapter.today();\n    this._setChosenYear();\n    this._chosenMonth = undefined;\n    this._init();\n    this._nextCalendarView = this._calendarView = this.view;\n\n    if (initTransition) {\n      this._startTableTransition();\n    }\n  }\n\n  /** Render the view for the day selection. */\n  private _renderDayView(): TemplateResult {\n    const nextMonthActiveDate = this._wide\n      ? this._dateAdapter.addCalendarMonths(this._activeDate, 1)\n      : undefined;\n    return html`\n      <div class=\"sbb-calendar__controls\">\n        ${this._getArrow(\n          'left',\n          () => this._goToDifferentMonth(-1),\n          i18nPreviousMonth[this._language.current],\n          this._previousMonthDisabled(),\n        )}\n        <div class=\"sbb-calendar__controls-month\">\n          ${this._createLabelForDayView(this._activeDate)}\n          ${this._wide ? this._createLabelForDayView(nextMonthActiveDate!) : nothing}\n          <sbb-screen-reader-only role=\"status\">\n            ${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate!)}\n          </sbb-screen-reader-only>\n        </div>\n        ${this._getArrow(\n          'right',\n          () => this._goToDifferentMonth(1),\n          i18nNextMonth[this._language.current],\n          this._nextMonthDisabled(),\n        )}\n      </div>\n      <div class=\"sbb-calendar__table-overflow-break\">\n        <div class=\"sbb-calendar__table-container sbb-calendar__table-day-view\">\n          ${this.orientation === 'horizontal'\n            ? html`\n                ${this._createDayTable(this._weeks, this._weekNumbers)}\n                ${this._wide\n                  ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true)\n                  : nothing}\n              `\n            : html`\n                ${this._createDayTableVertical(this._weeks, this._weekNumbers)}\n                ${this._wide\n                  ? this._createDayTableVertical(\n                      this._nextMonthWeeks,\n                      this._nextMonthWeekNumbers,\n                      nextMonthActiveDate,\n                    )\n                  : nothing}\n              `}\n        </div>\n      </div>\n    `;\n  }\n\n  /** Creates the label with the month for the daily view. */\n  private _createLabelForDayView(d: T): TemplateResult {\n    const monthLabel = `${\n      this._monthNames[this._dateAdapter.getMonth(d) - 1]\n    } ${this._dateAdapter.getYear(d)}`;\n    return html`\n      <button\n        type=\"button\"\n        class=\"sbb-calendar__date-selection sbb-calendar__controls-change-date\"\n        aria-label=\"${i18nYearMonthSelection[this._language.current]} ${monthLabel}\"\n        @click=${() => {\n          this._resetFocus = true;\n          this._nextCalendarView = 'year';\n          this._startTableTransition();\n        }}\n      >\n        ${monthLabel}\n        <sbb-icon name=\"chevron-small-down-small\"></sbb-icon>\n      </button>\n    `;\n  }\n\n  /** Creates the aria-label for the daily view. */\n  private _createAriaLabelForDayView(...dates: T[]): string {\n    let monthLabel = '';\n    for (const d of dates) {\n      if (d) {\n        monthLabel += `${\n          this._monthNames[this._dateAdapter.getMonth(d) - 1]\n        } ${this._dateAdapter.getYear(d)} `;\n      }\n    }\n    return monthLabel;\n  }\n\n  /** Creates the calendar table for the daily view. */\n  private _createDayTable(\n    weeks: Day<T>[][],\n    weekNumbers: number[],\n    isWideNextMonth: boolean = false,\n  ): TemplateResult {\n    const weeksForSelectMultipleWeekNumbers: Day<T>[] = (\n      this._wide\n        ? [...this._weeks, ...this._nextMonthWeeks]\n        : isWideNextMonth\n          ? this._nextMonthWeeks\n          : this._weeks\n    ).flat();\n    const weeksForSelectMultipleWeekDays: Day<T>[] = (\n      isWideNextMonth ? this._nextMonthWeeks : this._weeks\n    ).flat();\n    return html`\n      <table\n        class=\"sbb-calendar__table\"\n        @focusout=${(event: FocusEvent) =>\n          this._handleTableBlur(event.relatedTarget as HTMLElement)}\n        @animationend=${(e: AnimationEvent) => this._tableAnimationEnd(e)}\n      >\n        <thead class=\"sbb-calendar__table-header\">\n          <tr>\n            ${this.weekNumbers ? html`<th class=\"sbb-calendar__table-header-cell\"></th>` : nothing}\n            ${this._weekdays.map(\n              (weekDay: Weekday, index: number) => html`\n                <th class=\"sbb-calendar__table-header-cell\">\n                  ${this.multiple\n                    ? html`\n                        <sbb-calendar-weekday\n                          .value=${weekDay}\n                          @click=${() => {\n                            // NOTE: Sundays have index 7, while their weekDayValue is 0\n                            const days: Day<T>[] = weeksForSelectMultipleWeekDays.filter(\n                              (day: Day<T>) => day.weekDayValue === (index + 1) % 7,\n                            )!;\n                            this._selectMultipleDates(days);\n                          }}\n                        ></sbb-calendar-weekday>\n                      `\n                    : html`\n                        <sbb-screen-reader-only>${weekDay.long}</sbb-screen-reader-only>\n                        <span aria-hidden=\"true\">${weekDay.narrow}</span>\n                      `}\n                </th>\n              `,\n            )}\n          </tr>\n        </thead>\n        <tbody class=\"sbb-calendar__table-body\">\n          ${weeks.map((week: Day<T>[], rowIndex: number) => {\n            const firstRowOffset: number = DAYS_PER_ROW - week.length;\n            if (rowIndex === 0 && firstRowOffset) {\n              return html`\n                <tr>\n                  ${this.weekNumbers\n                    ? html`\n                        <td class=\"sbb-calendar__table-header-cell-vertical\">\n                          ${this.multiple\n                            ? html`\n                                <sbb-calendar-weeknumber\n                                  .value=${weekNumbers[0]}\n                                  @click=${() => {\n                                    const days: Day<T>[] = weeksForSelectMultipleWeekNumbers.filter(\n                                      (day: Day<T>) => day.weekValue === weekNumbers[0],\n                                    )!;\n                                    this._selectMultipleDates(days);\n                                  }}\n                                ></sbb-calendar-weeknumber>\n                              `\n                            : html`\n                                <sbb-screen-reader-only\n                                  >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</sbb-screen-reader-only\n                                >\n                                <span aria-hidden=\"true\">${weekNumbers[0]}</span>\n                              `}\n                        </td>\n                      `\n                    : nothing}\n                  ${[...Array(firstRowOffset).keys()].map(\n                    () => html`<td class=\"sbb-calendar__table-data\"></td>`,\n                  )}\n                  ${this._createDayCells(week)}\n                </tr>\n              `;\n            }\n            return html`\n              <tr>\n                ${this.weekNumbers\n                  ? html`\n                      <td class=\"sbb-calendar__table-header-cell-vertical\">\n                        ${this.multiple\n                          ? html`\n                              <sbb-calendar-weeknumber\n                                .value=${weekNumbers[rowIndex]}\n                                @click=${() => {\n                                  const days: Day<T>[] = weeksForSelectMultipleWeekNumbers.filter(\n                                    (day: Day<T>) => day.weekValue === weekNumbers[rowIndex],\n                                  )!;\n                                  this._selectMultipleDates(days);\n                                }}\n                              ></sbb-calendar-weeknumber>\n                            `\n                          : html`\n                              <sbb-screen-reader-only\n                                >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</sbb-screen-reader-only\n                              >\n                              <span aria-hidden=\"true\">${weekNumbers[rowIndex]}</span>\n                            `}\n                      </td>\n                    `\n                  : nothing}\n                ${this._createDayCells(week)}\n              </tr>\n            `;\n          })}\n        </tbody>\n      </table>\n    `;\n  }\n\n  /** Creates the table in orientation='vertical'. */\n  private _createDayTableVertical(\n    weeks: Day<T>[][],\n    weekNumbers: number[],\n    nextMonthActiveDate?: T,\n  ): TemplateResult {\n    const weekOffset = this._dateAdapter.getFirstWeekOffset(\n      nextMonthActiveDate ?? this._activeDate,\n    );\n    const weeksForSelectMultipleWeekNumbers: Day<T>[] = (\n      this._wide\n        ? [...this._weeks, ...this._nextMonthWeeks]\n        : nextMonthActiveDate\n          ? this._nextMonthWeeks\n          : this._weeks\n    ).flat();\n    return html`\n      <table\n        class=\"sbb-calendar__table\"\n        @focusout=${(event: FocusEvent) =>\n          this._handleTableBlur(event.relatedTarget as HTMLElement)}\n        @animationend=${(e: AnimationEvent) => this._tableAnimationEnd(e)}\n      >\n        ${this.weekNumbers\n          ? html`\n              <thead class=\"sbb-calendar__table-header\">\n                <tr>\n                  ${nextMonthActiveDate\n                    ? nothing\n                    : html`<th class=\"sbb-calendar__table-data\"></th>`}\n                  ${weekNumbers.map(\n                    (weekNumber: number) => html`\n                      <th class=\"sbb-calendar__table-header-cell\">\n                        ${this.multiple\n                          ? html`\n                              <sbb-calendar-weeknumber\n                                .value=${weekNumber}\n                                @click=${() => {\n                                  const days: Day<T>[] = weeksForSelectMultipleWeekNumbers.filter(\n                                    (day: Day<T>) => day.weekValue === weekNumber,\n                                  )!;\n                                  this._selectMultipleDates(days);\n                                }}\n                              ></sbb-calendar-weeknumber>\n                            `\n                          : html`\n                              <sbb-screen-reader-only\n                                >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</sbb-screen-reader-only\n                              >\n                              <span aria-hidden=\"true\">${weekNumber}</span>\n                            `}\n                      </th>\n                    `,\n                  )}\n                </tr>\n              </thead>\n            `\n          : nothing}\n        <tbody class=\"sbb-calendar__table-body\">\n          ${weeks.map((week: Day<T>[], rowIndex: number) => {\n            const weekday = this._weekdays[rowIndex];\n            const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;\n            return html`\n              <tr>\n                ${nextMonthActiveDate\n                  ? nothing\n                  : html`\n                      <td class=\"sbb-calendar__table-header-cell-vertical\">\n                        ${this.multiple\n                          ? html`\n                              <sbb-calendar-weekday\n                                .value=${weekday}\n                                @click=${() => this._selectMultipleDates(selectableDays)}\n                              >\n                                ${weekday.narrow}\n                              </sbb-calendar-weekday>\n                            `\n                          : html`\n                              <sbb-screen-reader-only>${weekday.long}</sbb-screen-reader-only>\n                              <span aria-hidden=\"true\">${weekday.narrow}</span>\n                            `}\n                      </td>\n                    `}\n                ${rowIndex < weekOffset\n                  ? html`<td class=\"sbb-calendar__table-data\"></td>`\n                  : nothing}\n                ${this._createDayCells(week)}\n              </tr>\n            `;\n          })}\n        </tbody>\n      </table>\n    `;\n  }\n\n  /** Creates the cells for the daily view. */\n  private _createDayCells(week: Day<T>[]): TemplateResult[] {\n    return week.map((day: Day<T>) => {\n      return html`\n        <td class=\"sbb-calendar__table-data sbb-calendar__day-cell\">\n          <slot name=${day.value}>\n            <sbb-calendar-day\n              slot=${day.value}\n              @click=${() => this._selectDate(day.dateValue)}\n              @keydown=${(evt: KeyboardEvent) => this._handleKeyboardEvent(evt, day)}\n            ></sbb-calendar-day>\n          </slot>\n        </td>\n      `;\n    });\n  }\n\n  /** Render the view for the month selection. */\n  private _renderMonthView(): TemplateResult {\n    return html`\n      <div class=\"sbb-calendar__controls\">\n        ${this._getArrow(\n          'left',\n          () => this._goToDifferentYear(-1),\n          i18nPreviousYear[this._language.current],\n          this._previousYearDisabled(),\n        )}\n        <div class=\"sbb-calendar__controls-month\">${this._createLabelForMonthView()}</div>\n        ${this._getArrow(\n          'right',\n          () => this._goToDifferentYear(1),\n          i18nNextYear[this._language.current],\n          this._nextYearDisabled(),\n        )}\n      </div>\n      <div class=\"sbb-calendar__table-overflow-break\">\n        <div class=\"sbb-calendar__table-container sbb-calendar__table-month-view\">\n          ${this._createMonthTable(this._months, this._chosenYear!)}\n          ${this._wide ? this._createMonthTable(this._months, this._chosenYear! + 1) : nothing}\n        </div>\n      </div>\n    `;\n  }\n\n  /** Creates the label with the year for the monthly view. */\n  private _createLabelForMonthView(): TemplateResult {\n    return html` <button\n        type=\"button\"\n        id=\"sbb-calendar__month-selection\"\n        class=\"sbb-calendar__controls-change-date\"\n        aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}\n        @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}\n      >\n        ${this._chosenYear} ${this._wide ? ` - ${this._chosenYear! + 1}` : nothing}\n        <sbb-icon name=\"chevron-small-up-small\"></sbb-icon>\n      </button>\n      <sbb-screen-reader-only role=\"status\"> ${this._chosenYear} </sbb-screen-reader-only>`;\n  }\n\n  /** Creates the table for the month selection view. */\n  private _createMonthTable(months: MonthCell[][], year: number): TemplateResult {\n    return html`\n      <table\n        class=\"sbb-calendar__table\"\n        @animationend=${(e: AnimationEvent) => this._tableAnimationEnd(e)}\n      >\n        ${this._wide\n          ? html`<thead class=\"sbb-calendar__table-header\" aria-hidden=\"true\">\n              <tr>\n                <th class=\"sbb-calendar__table-header-cell\" colspan=${MONTHS_PER_ROW}>${year}</th>\n              </tr>\n            </thead>`\n          : nothing}\n        <tbody class=\"sbb-calendar__table-body\">\n          ${months.map(\n            (row: MonthCell[]) => html`\n              <tr>\n                ${row.map((month: MonthCell) => {\n                  return html`\n                    <td class=\"sbb-calendar__table-data\">\n                      <sbb-calendar-month\n                        .value=\"${year}-${month.value}\"\n                        @click=${() => this._onMonthSelection(month.monthValue, year)}\n                        @keydown=${(evt: KeyboardEvent) => this._handleKeyboardEvent(evt)}\n                      >\n                      </sbb-calendar-month>\n                    </td>\n                  `;\n                })}\n              </tr>\n            `,\n          )}\n        </tbody>\n      </table>\n    `;\n  }\n\n  /** Select the month and change the view to day selection. */\n  private _onMonthSelection(month: number, year: number): void {\n    this._chosenMonth = month;\n    this._nextCalendarView = 'day';\n    this._init(\n      this._dateAdapter.createDate(\n        year,\n        this._chosenMonth,\n        this._dateAdapter.getDate(this._activeDate),\n      ),\n    );\n    this._startTableTransition();\n    this._emitMonthChange();\n  }\n\n  /** Render the view for the year selection. */\n  private _renderYearView(): TemplateResult {\n    return html`\n      <div class=\"sbb-calendar__controls\">\n        ${this._getArrow(\n          'left',\n          () => this._goToDifferentYearRange(-YEARS_PER_PAGE),\n          i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current],\n          this._previousYearRangeDisabled(),\n        )}\n        <div class=\"sbb-calendar__controls-month\">${this._createLabelForYearView()}</div>\n        ${this._getArrow(\n          'right',\n          () => this._goToDifferentYearRange(YEARS_PER_PAGE),\n          i18nNextYearRange(YEARS_PER_PAGE)[this._language.current],\n          this._nextYearRangeDisabled(),\n        )}\n      </div>\n      <div class=\"sbb-calendar__table-overflow-break\">\n        <div class=\"sbb-calendar__table-container sbb-calendar__table-year-view\">\n          ${this._createYearTable(this._years)}\n          ${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}\n        </div>\n      </div>\n    `;\n  }\n\n  /** Creates the button arrow for all the views. */\n  private _getArrow(\n    direction: 'left' | 'right',\n    click: () => void,\n    ariaLabel: string,\n    disabled: boolean,\n  ): TemplateResult {\n    return html`<sbb-secondary-button\n      size=\"m\"\n      icon-name=\"chevron-small-${direction}-small\"\n      aria-label=${ariaLabel}\n      @click=${click}\n      ?disabled=${disabled}\n      id=\"sbb-calendar__controls-${direction === 'left' ? 'previous' : 'next'}\"\n    ></sbb-secondary-button>`;\n  }\n\n  /** Creates the label with the year range for the yearly view. */\n  private _createLabelForYearView(): TemplateResult {\n    const firstYear: number = this._years.flat()[0];\n    const lastYearArray: number[] = (this._wide ? this._nextMonthYears : this._years).flat();\n    const lastYear: number = lastYearArray[lastYearArray.length - 1];\n    const yearLabel = `${firstYear} - ${lastYear}`;\n    return html`\n      <button\n        type=\"button\"\n        id=\"sbb-calendar__year-selection\"\n        class=\"sbb-calendar__controls-change-date\"\n        aria-label=\"${i18nCalendarDateSelection[this._language.current]} ${yearLabel}\"\n        @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}\n      >\n        ${yearLabel}\n        <sbb-icon name=\"chevron-small-up-small\"></sbb-icon>\n      </button>\n      <sbb-screen-reader-only role=\"status\"> ${yearLabel} </sbb-screen-reader-only>\n    `;\n  }\n\n  /** Creates the table for the year selection view. */\n  private _createYearTable(years: number[][], shiftRight = false): TemplateResult {\n    return html` <table\n      class=\"sbb-calendar__table\"\n      @animationend=${(e: AnimationEvent) => this._tableAnimationEnd(e)}\n    >\n      <tbody class=\"sbb-calendar__table-body\">\n        ${years.map(\n          (row: number[]) =>\n            html` <tr>\n              ${row.map((year: number) => {\n                return html`\n                  <td class=\"sbb-calendar__table-data\">\n                    <sbb-calendar-year\n                      .value=${String(year)}\n                      @keydown=${(evt: KeyboardEvent) => this._handleKeyboardEvent(evt)}\n                      @click=${() => this._onYearSelection(year, shiftRight)}\n                    >\n                    </sbb-calendar-year>\n                  </td>\n                `;\n              })}\n            </tr>`,\n        )}\n      </tbody>\n    </table>`;\n  }\n\n  /** Select the year and change the view to month selection. */\n  private _onYearSelection(year: number, rightSide: boolean): void {\n    this._chosenYear = rightSide ? year - 1 : year;\n    this._nextCalendarView = 'month';\n    this._assignActiveDate(\n      this._dateAdapter.createDate(\n        this._chosenYear,\n        this._dateAdapter.getMonth(this._activeDate),\n        this._dateAdapter.getDate(this._activeDate),\n      ),\n    );\n    this._startTableTransition();\n  }\n\n  private _getView(): TemplateResult {\n    if (isServer || this.hydrationRequired) {\n      // TODO: We disable SSR for calendar for now. Figure out, if there is a way\n      // to enable it, while considering i18n and date information.\n      return html`${nothing}`;\n    }\n    switch (this._calendarView) {\n      case 'year':\n        return this._renderYearView();\n      case 'month':\n        return this._renderMonthView();\n      case 'day':\n      default:\n        return this._renderDayView();\n    }\n  }\n\n  private _tableAnimationEnd(event: AnimationEvent): void {\n    const table = event.target as HTMLElement;\n    if (event.animationName === 'hide') {\n      table.classList.remove('sbb-calendar__table-hide');\n      if (this._containingFocus) {\n        this._resetFocus = true;\n      }\n      this._calendarView = this._nextCalendarView;\n    } else if (event.animationName === 'show') {\n      this.internals.states.delete('transition');\n    }\n  }\n\n  private _startTableTransition(): void {\n    this.internals.states.add('transition');\n    this.shadowRoot\n      ?.querySelectorAll('table')\n      ?.forEach((e) => e.classList.toggle('sbb-calendar__table-hide'));\n  }\n\n  protected override render(): TemplateResult {\n    return html`<div class=\"sbb-calendar__wrapper\">${this._getView()}</div>`;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-calendar': SbbCalendarElement;\n  }\n  interface HTMLElementEventMap {\n    monthchange: SbbMonthChangeEvent;\n  }\n}\n"],"mappings":"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"}
|