@sbb-esta/lyne-elements-experimental 3.0.1 → 3.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/custom-elements.json +908 -567
- package/development/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/icon-mapper.js +1 -3
- package/development/seat-reservation/common/mapper/mapper.d.ts +4 -3
- package/development/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/mapper.js +6 -4
- package/development/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/seat-reservation-sample-data.js +4989 -4
- package/development/seat-reservation/common/svgs.d.ts +0 -1
- package/development/seat-reservation/common/svgs.d.ts.map +1 -1
- package/development/seat-reservation/common/svgs.js +1 -3
- package/development/seat-reservation/common/translations/i18n.js +194 -194
- package/development/seat-reservation/common/types.d.ts +10 -0
- package/development/seat-reservation/common/types.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +38 -12
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.js +252 -122
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts +28 -13
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation.component.js +481 -262
- package/development/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +17 -10
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-assets.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-assets.js +1 -6
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +36 -18
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts +3 -1
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +26 -4
- package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.js +86 -54
- package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts +1 -16
- package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.js +10 -164
- package/package.json +3 -3
- package/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/icon-mapper.js +0 -2
- package/seat-reservation/common/mapper/mapper.d.ts +4 -3
- package/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/mapper.js +16 -14
- package/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
- package/seat-reservation/common/mapper/seat-reservation-sample-data.js +4988 -3
- package/seat-reservation/common/svgs.d.ts +0 -1
- package/seat-reservation/common/svgs.d.ts.map +1 -1
- package/seat-reservation/common/svgs.js +4 -11
- package/seat-reservation/common/translations/i18n.js +197 -197
- package/seat-reservation/common/types.d.ts +10 -0
- package/seat-reservation/common/types.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +38 -12
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.js +208 -134
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts +28 -13
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation.component.js +290 -298
- package/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +22 -22
- package/seat-reservation/seat-reservation-graphic/seat-reservation-assets.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-graphic/seat-reservation-assets.js +0 -5
- package/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +37 -37
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts +3 -1
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +69 -58
- package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.js +28 -27
- package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts +1 -16
- package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.js +14 -101
|
@@ -17,7 +17,7 @@ var ScrollDirection;
|
|
|
17
17
|
ScrollDirection2["left"] = "left";
|
|
18
18
|
})(ScrollDirection || (ScrollDirection = {}));
|
|
19
19
|
let SeatReservationBaseElement = (() => {
|
|
20
|
-
var _seatReservation_accessor_storage, _hasNavigation_accessor_storage, _alignVertical_accessor_storage, _baseGridSize_accessor_storage, _height_accessor_storage,
|
|
20
|
+
var _seatReservation_accessor_storage, _hasNavigation_accessor_storage, _alignVertical_accessor_storage, _baseGridSize_accessor_storage, _height_accessor_storage, _maxSeatReservations_accessor_storage, _maxBicycleReservations_accessor_storage, _preventPlaceClick_accessor_storage, _preselectCoachIndex_accessor_storage, _selectedCoachIndex_accessor_storage, _focusedCoachIndex_accessor_storage, _hoveredScrollCoachIndex_accessor_storage, _a;
|
|
21
21
|
let _classSuper = LitElement;
|
|
22
22
|
let _seatReservation_decorators;
|
|
23
23
|
let _seatReservation_initializers = [];
|
|
@@ -34,18 +34,27 @@ let SeatReservationBaseElement = (() => {
|
|
|
34
34
|
let _height_decorators;
|
|
35
35
|
let _height_initializers = [];
|
|
36
36
|
let _height_extraInitializers = [];
|
|
37
|
-
let
|
|
38
|
-
let
|
|
39
|
-
let
|
|
37
|
+
let _maxSeatReservations_decorators;
|
|
38
|
+
let _maxSeatReservations_initializers = [];
|
|
39
|
+
let _maxSeatReservations_extraInitializers = [];
|
|
40
|
+
let _maxBicycleReservations_decorators;
|
|
41
|
+
let _maxBicycleReservations_initializers = [];
|
|
42
|
+
let _maxBicycleReservations_extraInitializers = [];
|
|
40
43
|
let _preventPlaceClick_decorators;
|
|
41
44
|
let _preventPlaceClick_initializers = [];
|
|
42
45
|
let _preventPlaceClick_extraInitializers = [];
|
|
46
|
+
let _preselectCoachIndex_decorators;
|
|
47
|
+
let _preselectCoachIndex_initializers = [];
|
|
48
|
+
let _preselectCoachIndex_extraInitializers = [];
|
|
43
49
|
let _selectedCoachIndex_decorators;
|
|
44
50
|
let _selectedCoachIndex_initializers = [];
|
|
45
51
|
let _selectedCoachIndex_extraInitializers = [];
|
|
46
52
|
let _focusedCoachIndex_decorators;
|
|
47
53
|
let _focusedCoachIndex_initializers = [];
|
|
48
54
|
let _focusedCoachIndex_extraInitializers = [];
|
|
55
|
+
let _hoveredScrollCoachIndex_decorators;
|
|
56
|
+
let _hoveredScrollCoachIndex_initializers = [];
|
|
57
|
+
let _hoveredScrollCoachIndex_extraInitializers = [];
|
|
49
58
|
return _a = class extends _classSuper {
|
|
50
59
|
constructor() {
|
|
51
60
|
super(...arguments);
|
|
@@ -54,33 +63,46 @@ let SeatReservationBaseElement = (() => {
|
|
|
54
63
|
__privateAdd(this, _alignVertical_accessor_storage);
|
|
55
64
|
__privateAdd(this, _baseGridSize_accessor_storage);
|
|
56
65
|
__privateAdd(this, _height_accessor_storage);
|
|
57
|
-
__privateAdd(this,
|
|
66
|
+
__privateAdd(this, _maxSeatReservations_accessor_storage);
|
|
67
|
+
__privateAdd(this, _maxBicycleReservations_accessor_storage);
|
|
58
68
|
__privateAdd(this, _preventPlaceClick_accessor_storage);
|
|
69
|
+
__privateAdd(this, _preselectCoachIndex_accessor_storage);
|
|
59
70
|
__privateAdd(this, _selectedCoachIndex_accessor_storage);
|
|
60
71
|
__privateAdd(this, _focusedCoachIndex_accessor_storage);
|
|
72
|
+
__privateAdd(this, _hoveredScrollCoachIndex_accessor_storage);
|
|
61
73
|
__privateSet(this, _seatReservation_accessor_storage, __runInitializers(this, _seatReservation_initializers, null));
|
|
62
74
|
__privateSet(this, _hasNavigation_accessor_storage, (__runInitializers(this, _seatReservation_extraInitializers), __runInitializers(this, _hasNavigation_initializers, true)));
|
|
63
75
|
__privateSet(this, _alignVertical_accessor_storage, (__runInitializers(this, _hasNavigation_extraInitializers), __runInitializers(this, _alignVertical_initializers, false)));
|
|
64
76
|
__privateSet(this, _baseGridSize_accessor_storage, (__runInitializers(this, _alignVertical_extraInitializers), __runInitializers(this, _baseGridSize_initializers, 16)));
|
|
65
77
|
__privateSet(this, _height_accessor_storage, (__runInitializers(this, _baseGridSize_extraInitializers), __runInitializers(this, _height_initializers, null)));
|
|
66
|
-
__privateSet(this,
|
|
67
|
-
__privateSet(this,
|
|
68
|
-
__privateSet(this,
|
|
78
|
+
__privateSet(this, _maxSeatReservations_accessor_storage, (__runInitializers(this, _height_extraInitializers), __runInitializers(this, _maxSeatReservations_initializers, -1)));
|
|
79
|
+
__privateSet(this, _maxBicycleReservations_accessor_storage, (__runInitializers(this, _maxSeatReservations_extraInitializers), __runInitializers(this, _maxBicycleReservations_initializers, -1)));
|
|
80
|
+
__privateSet(this, _preventPlaceClick_accessor_storage, (__runInitializers(this, _maxBicycleReservations_extraInitializers), __runInitializers(this, _preventPlaceClick_initializers, false)));
|
|
81
|
+
__privateSet(this, _preselectCoachIndex_accessor_storage, (__runInitializers(this, _preventPlaceClick_extraInitializers), __runInitializers(this, _preselectCoachIndex_initializers, -1)));
|
|
82
|
+
__privateSet(this, _selectedCoachIndex_accessor_storage, (__runInitializers(this, _preselectCoachIndex_extraInitializers), __runInitializers(this, _selectedCoachIndex_initializers, -1)));
|
|
69
83
|
__privateSet(this, _focusedCoachIndex_accessor_storage, (__runInitializers(this, _selectedCoachIndex_extraInitializers), __runInitializers(this, _focusedCoachIndex_initializers, -1)));
|
|
70
|
-
this
|
|
84
|
+
__privateSet(this, _hoveredScrollCoachIndex_accessor_storage, (__runInitializers(this, _focusedCoachIndex_extraInitializers), __runInitializers(this, _hoveredScrollCoachIndex_initializers, -1)));
|
|
85
|
+
this.coachBorderPadding = (__runInitializers(this, _hoveredScrollCoachIndex_extraInitializers), 6);
|
|
86
|
+
this.gapBetweenCoaches = 4;
|
|
71
87
|
this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;
|
|
88
|
+
this.coachNavButtonDim = 0;
|
|
72
89
|
this.currScrollDirection = ScrollDirection.right;
|
|
73
90
|
this.maxCalcCoachsWidth = 0;
|
|
74
91
|
this.scrollCoachsAreaWidth = 0;
|
|
92
|
+
this.scrollNavigationAreaDim = 0;
|
|
75
93
|
this.triggerCoachPositionsCollection = [];
|
|
76
94
|
this.firstTabElement = null;
|
|
77
95
|
this.lastTabElement = null;
|
|
96
|
+
this.navigationScrollArea = null;
|
|
78
97
|
this.coachScrollArea = null;
|
|
79
98
|
this.currSelectedPlace = null;
|
|
80
99
|
this.currSelectedPlaceElementId = null;
|
|
81
100
|
this.currSelectedCoachIndex = -1;
|
|
82
101
|
this.preventCoachScrollByPlaceClick = false;
|
|
83
|
-
this.selectedSeatReservationPlaces =
|
|
102
|
+
this.selectedSeatReservationPlaces = {
|
|
103
|
+
seats: [],
|
|
104
|
+
bicycles: []
|
|
105
|
+
};
|
|
84
106
|
this.seatReservationWithoutNavigationHasFocus = false;
|
|
85
107
|
this.isCoachGridFocusable = false;
|
|
86
108
|
this.isAutoScrolling = false;
|
|
@@ -130,11 +152,18 @@ let SeatReservationBaseElement = (() => {
|
|
|
130
152
|
__privateSet(this, _height_accessor_storage, value);
|
|
131
153
|
}
|
|
132
154
|
/** Maximal number of possible clickable seats */
|
|
133
|
-
get
|
|
134
|
-
return __privateGet(this,
|
|
155
|
+
get maxSeatReservations() {
|
|
156
|
+
return __privateGet(this, _maxSeatReservations_accessor_storage);
|
|
157
|
+
}
|
|
158
|
+
set maxSeatReservations(value) {
|
|
159
|
+
__privateSet(this, _maxSeatReservations_accessor_storage, value);
|
|
160
|
+
}
|
|
161
|
+
/** Maximal number of possible clickable bicycle places */
|
|
162
|
+
get maxBicycleReservations() {
|
|
163
|
+
return __privateGet(this, _maxBicycleReservations_accessor_storage);
|
|
135
164
|
}
|
|
136
|
-
set
|
|
137
|
-
__privateSet(this,
|
|
165
|
+
set maxBicycleReservations(value) {
|
|
166
|
+
__privateSet(this, _maxBicycleReservations_accessor_storage, value);
|
|
138
167
|
}
|
|
139
168
|
/** Any click functionality is prevented */
|
|
140
169
|
get preventPlaceClick() {
|
|
@@ -143,6 +172,12 @@ let SeatReservationBaseElement = (() => {
|
|
|
143
172
|
set preventPlaceClick(value) {
|
|
144
173
|
__privateSet(this, _preventPlaceClick_accessor_storage, value);
|
|
145
174
|
}
|
|
175
|
+
get preselectCoachIndex() {
|
|
176
|
+
return __privateGet(this, _preselectCoachIndex_accessor_storage);
|
|
177
|
+
}
|
|
178
|
+
set preselectCoachIndex(value) {
|
|
179
|
+
__privateSet(this, _preselectCoachIndex_accessor_storage, value);
|
|
180
|
+
}
|
|
146
181
|
get selectedCoachIndex() {
|
|
147
182
|
return __privateGet(this, _selectedCoachIndex_accessor_storage);
|
|
148
183
|
}
|
|
@@ -155,6 +190,13 @@ let SeatReservationBaseElement = (() => {
|
|
|
155
190
|
set focusedCoachIndex(value) {
|
|
156
191
|
__privateSet(this, _focusedCoachIndex_accessor_storage, value);
|
|
157
192
|
}
|
|
193
|
+
//Sets the hover style when scrolling to a coach
|
|
194
|
+
get hoveredScrollCoachIndex() {
|
|
195
|
+
return __privateGet(this, _hoveredScrollCoachIndex_accessor_storage);
|
|
196
|
+
}
|
|
197
|
+
set hoveredScrollCoachIndex(value) {
|
|
198
|
+
__privateSet(this, _hoveredScrollCoachIndex_accessor_storage, value);
|
|
199
|
+
}
|
|
158
200
|
willUpdate(changedProperties) {
|
|
159
201
|
super.willUpdate(changedProperties);
|
|
160
202
|
if (changedProperties.has("seatReservation")) {
|
|
@@ -163,36 +205,56 @@ let SeatReservationBaseElement = (() => {
|
|
|
163
205
|
if (changedProperties.has("baseGridSize")) {
|
|
164
206
|
this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;
|
|
165
207
|
this.style?.setProperty("--sbb-seat-reservation-grid-size", `${this.baseGridSize}px`);
|
|
166
|
-
|
|
167
|
-
this._setVerticalAlignmentOffset();
|
|
168
|
-
}
|
|
208
|
+
this.initNavigationSelectionByScrollEvent();
|
|
169
209
|
}
|
|
170
210
|
if (changedProperties.has("height") && !!this.height) {
|
|
171
211
|
if (this.seatReservation.coachItems.length) {
|
|
172
212
|
this.baseGridSize = this.height / this.seatReservation.coachItems[0].dimension.h;
|
|
173
213
|
this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;
|
|
174
214
|
this.style?.setProperty("--sbb-seat-reservation-grid-size", `${this.baseGridSize}px`);
|
|
175
|
-
|
|
176
|
-
this._setVerticalAlignmentOffset();
|
|
177
|
-
}
|
|
215
|
+
this.initNavigationSelectionByScrollEvent();
|
|
178
216
|
}
|
|
179
217
|
}
|
|
180
218
|
if (changedProperties.has("alignVertical") && this.alignVertical) {
|
|
181
|
-
this.
|
|
219
|
+
this.initNavigationSelectionByScrollEvent();
|
|
220
|
+
}
|
|
221
|
+
if (changedProperties.has("preselectCoachIndex") && this.preselectCoachIndex) {
|
|
222
|
+
setTimeout(() => this.scrollToSelectedNavCoach(this.preselectCoachIndex), 1);
|
|
182
223
|
}
|
|
183
224
|
}
|
|
225
|
+
navigateByDirectionBtn(btnDirection) {
|
|
226
|
+
this.unfocusPlaceElement();
|
|
227
|
+
let navigateToCoachIndex = this.currSelectedCoachIndex;
|
|
228
|
+
if (btnDirection == "DIRECTION_LEFT" && navigateToCoachIndex > 0) {
|
|
229
|
+
navigateToCoachIndex = this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex - 1 : 0;
|
|
230
|
+
} else if (btnDirection == "DIRECTION_RIGHT" && navigateToCoachIndex < this.seatReservation.coachItems.length - 1) {
|
|
231
|
+
navigateToCoachIndex = this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex + 1 : 0;
|
|
232
|
+
}
|
|
233
|
+
this.scrollToSelectedNavCoach(navigateToCoachIndex);
|
|
234
|
+
}
|
|
184
235
|
/* Init scroll event handling for coach navigation */
|
|
185
236
|
initNavigationSelectionByScrollEvent() {
|
|
186
|
-
this.
|
|
187
|
-
|
|
188
|
-
|
|
237
|
+
if (this.seatReservation.coachItems.length > 0) {
|
|
238
|
+
const borderHeight = (this.seatReservation.coachItems[0].dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;
|
|
239
|
+
this.style?.setProperty("--sbb-seat-reservation-coach-height", `${borderHeight}`);
|
|
240
|
+
}
|
|
241
|
+
this.firstTabElement = this.shadowRoot?.querySelector("#first-tab-element");
|
|
242
|
+
this.lastTabElement = this.shadowRoot?.querySelector("#last-tab-element");
|
|
243
|
+
this.coachScrollArea = this.shadowRoot?.querySelector("#sbb-sr__parent-area");
|
|
244
|
+
this.navigationScrollArea = this.shadowRoot?.querySelector("#sbb-sr-navigation");
|
|
245
|
+
if (this.navigationScrollArea) {
|
|
246
|
+
this.scrollNavigationAreaDim = this.alignVertical ? this.navigationScrollArea.getBoundingClientRect().height : this.navigationScrollArea.getBoundingClientRect().width;
|
|
247
|
+
const navCoacheList = this.navigationScrollArea.querySelector("ul > li");
|
|
248
|
+
const firstLiEleDimension = navCoacheList?.getBoundingClientRect();
|
|
249
|
+
this.coachNavButtonDim = this.alignVertical ? firstLiEleDimension?.height : firstLiEleDimension?.width;
|
|
250
|
+
}
|
|
189
251
|
if (this.coachScrollArea) {
|
|
190
252
|
let currCalcTriggerPos = 0;
|
|
191
|
-
this.scrollCoachsAreaWidth = this.coachScrollArea.getBoundingClientRect().width;
|
|
253
|
+
this.scrollCoachsAreaWidth = this.alignVertical ? this.coachScrollArea.getBoundingClientRect().height : this.coachScrollArea.getBoundingClientRect().width;
|
|
192
254
|
this.triggerCoachPositionsCollection = this.seatReservation.coachItems.map((coach) => {
|
|
193
255
|
const startPosX = currCalcTriggerPos;
|
|
194
256
|
const coachWidth = this.getCalculatedDimension(coach.dimension).w;
|
|
195
|
-
currCalcTriggerPos += coachWidth;
|
|
257
|
+
currCalcTriggerPos += coachWidth + this.gapBetweenCoaches;
|
|
196
258
|
return {
|
|
197
259
|
start: startPosX,
|
|
198
260
|
end: currCalcTriggerPos,
|
|
@@ -202,11 +264,17 @@ let SeatReservationBaseElement = (() => {
|
|
|
202
264
|
this.maxCalcCoachsWidth = currCalcTriggerPos;
|
|
203
265
|
this.coachScrollArea.addEventListener("scrollend", () => {
|
|
204
266
|
const findScrollCoachIndex = this.isAutoScrolling ? this.currSelectedCoachIndex : this._getCoachIndexByScrollTriggerPosition();
|
|
267
|
+
if (this.currSelectedCoachIndex === -1) {
|
|
268
|
+
this.currSelectedCoachIndex = findScrollCoachIndex;
|
|
269
|
+
}
|
|
205
270
|
if (this._isScrollableToSelectedCoach()) {
|
|
206
271
|
this.currSelectedCoachIndex = findScrollCoachIndex;
|
|
207
272
|
} else {
|
|
208
273
|
this.currSelectedCoachIndex = findScrollCoachIndex < this.currSelectedCoachIndex ? this.currSelectedCoachIndex : findScrollCoachIndex;
|
|
209
274
|
}
|
|
275
|
+
if (!this.isAutoScrolling) {
|
|
276
|
+
this._scrollToSelectedNavigationButton(findScrollCoachIndex);
|
|
277
|
+
}
|
|
210
278
|
this.preventCoachScrollByPlaceClick = false;
|
|
211
279
|
this.updateCurrentSelectedCoach();
|
|
212
280
|
if (!this.hasNavigation) {
|
|
@@ -214,9 +282,6 @@ let SeatReservationBaseElement = (() => {
|
|
|
214
282
|
this.isAutoScrolling = false;
|
|
215
283
|
}
|
|
216
284
|
});
|
|
217
|
-
if (this.alignVertical) {
|
|
218
|
-
this._setVerticalAlignmentOffset();
|
|
219
|
-
}
|
|
220
285
|
}
|
|
221
286
|
}
|
|
222
287
|
/**
|
|
@@ -308,14 +373,105 @@ let SeatReservationBaseElement = (() => {
|
|
|
308
373
|
const isSelectedCoachIndexScrollable = this.selectedCoachIndex !== -1 || this.currSelectedCoachIndex > 0;
|
|
309
374
|
if (isSelectedCoachIndexScrollable && this._isScrollableToSelectedCoach()) {
|
|
310
375
|
this.coachScrollArea.scrollTo({
|
|
311
|
-
top: 0,
|
|
312
|
-
left: scrollToCoachPosX,
|
|
376
|
+
top: this.alignVertical ? scrollToCoachPosX : 0,
|
|
377
|
+
left: this.alignVertical ? 0 : scrollToCoachPosX,
|
|
313
378
|
behavior: "smooth"
|
|
314
379
|
});
|
|
315
380
|
} else {
|
|
316
381
|
this.updateCurrentSelectedCoach();
|
|
317
382
|
}
|
|
383
|
+
this._scrollToSelectedNavigationButton(selectedNavCoachIndex);
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
focusPlaceElement(place, coachIndex) {
|
|
387
|
+
this.unfocusPlaceElement();
|
|
388
|
+
if (place) {
|
|
389
|
+
this.currSelectedPlace = place;
|
|
390
|
+
if (coachIndex) {
|
|
391
|
+
this.currSelectedCoachIndex = coachIndex;
|
|
392
|
+
}
|
|
393
|
+
this._setCurrSelectedPlaceElementId(place);
|
|
394
|
+
const selectedPlaceElement = this._getPlaceHtmlElement();
|
|
395
|
+
if (selectedPlaceElement) {
|
|
396
|
+
selectedPlaceElement.setAttribute("keyfocus", "focus");
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
unfocusPlaceElement() {
|
|
401
|
+
const selectedPlaceElement = this._getPlaceHtmlElement();
|
|
402
|
+
if (selectedPlaceElement) {
|
|
403
|
+
selectedPlaceElement.setAttribute("keyfocus", "unfocus");
|
|
404
|
+
this._setCurrSelectedPlaceElementId(null);
|
|
405
|
+
this.currSelectedPlace = null;
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
getCalculatedDimension(elementDimension, coachDimension, isOriginHeight, isStretchHeight) {
|
|
409
|
+
if (coachDimension && !isOriginHeight) {
|
|
410
|
+
elementDimension.h += this.coachBorderOffset * 2;
|
|
411
|
+
}
|
|
412
|
+
if (isStretchHeight) {
|
|
413
|
+
elementDimension.h += this.coachBorderOffset;
|
|
414
|
+
}
|
|
415
|
+
return {
|
|
416
|
+
w: this.baseGridSize * elementDimension.w,
|
|
417
|
+
h: this.baseGridSize * elementDimension.h
|
|
418
|
+
};
|
|
419
|
+
}
|
|
420
|
+
getCalculatedPosition(elementPosition, elementDimension, coachDimension, isOriginHeight) {
|
|
421
|
+
if (coachDimension && elementDimension) {
|
|
422
|
+
const endPosHeight = isOriginHeight ? coachDimension.h : coachDimension.h + this.coachBorderOffset;
|
|
423
|
+
if (elementPosition.y === 0) {
|
|
424
|
+
elementPosition.y -= this.coachBorderOffset;
|
|
425
|
+
} else if (elementPosition.y + elementDimension.h === endPosHeight) {
|
|
426
|
+
elementPosition.y += this.coachBorderOffset;
|
|
427
|
+
}
|
|
318
428
|
}
|
|
429
|
+
return {
|
|
430
|
+
x: this.baseGridSize * elementPosition.x,
|
|
431
|
+
y: this.baseGridSize * elementPosition.y,
|
|
432
|
+
z: elementPosition.z
|
|
433
|
+
};
|
|
434
|
+
}
|
|
435
|
+
/**
|
|
436
|
+
* Counts all available seats together depending on the seat type
|
|
437
|
+
*
|
|
438
|
+
* @param coachIndex
|
|
439
|
+
* @returns An Object with count of free seats and free bicycle places
|
|
440
|
+
*/
|
|
441
|
+
getAvailableFreePlacesNumFromCoach(coachIndex) {
|
|
442
|
+
const accumulator = { seats: 0, bicycles: 0 };
|
|
443
|
+
const freePlaces = this.seatReservation.coachItems[coachIndex].places?.reduce((accumulator2, currPlace) => {
|
|
444
|
+
if (currPlace.state !== "FREE") {
|
|
445
|
+
return accumulator2;
|
|
446
|
+
}
|
|
447
|
+
if (currPlace.type === "SEAT") {
|
|
448
|
+
accumulator2.seats++;
|
|
449
|
+
} else {
|
|
450
|
+
accumulator2.bicycles++;
|
|
451
|
+
}
|
|
452
|
+
return accumulator2;
|
|
453
|
+
}, accumulator);
|
|
454
|
+
return freePlaces ? freePlaces : accumulator;
|
|
455
|
+
}
|
|
456
|
+
/**
|
|
457
|
+
* Performs an automatic main navigation scroll to the specified selectedNavCoachIndex.
|
|
458
|
+
* Calculates the central scroll offset of the nav coach to be selected.
|
|
459
|
+
* @param selectedNavCoachIndex
|
|
460
|
+
*/
|
|
461
|
+
_scrollToSelectedNavigationButton(selectedNavCoachIndex) {
|
|
462
|
+
setTimeout(() => {
|
|
463
|
+
this.hoveredScrollCoachIndex = selectedNavCoachIndex;
|
|
464
|
+
if (this.hasNavigation && this.navigationScrollArea) {
|
|
465
|
+
const navigationAreaCenteredPosX = this.scrollNavigationAreaDim / 2;
|
|
466
|
+
const scrollButtonOffsetX = selectedNavCoachIndex * this.coachNavButtonDim;
|
|
467
|
+
const scrollOffsetX = scrollButtonOffsetX - navigationAreaCenteredPosX + this.coachNavButtonDim;
|
|
468
|
+
this.navigationScrollArea.scrollTo({
|
|
469
|
+
top: this.alignVertical ? scrollOffsetX : 0,
|
|
470
|
+
left: this.alignVertical ? 0 : scrollOffsetX,
|
|
471
|
+
behavior: "smooth"
|
|
472
|
+
});
|
|
473
|
+
}
|
|
474
|
+
}, 10);
|
|
319
475
|
}
|
|
320
476
|
/**
|
|
321
477
|
* Sets the new ScrollDirection by the new given target coach index.
|
|
@@ -325,8 +481,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
325
481
|
}
|
|
326
482
|
/**
|
|
327
483
|
* Returns the scroll start or end position X from the selected coach.
|
|
328
|
-
* In case the user is
|
|
329
|
-
* then we return the end position of the coach to get
|
|
484
|
+
* In case the user is currently navigating through places by keyboard and goes to previous coach,
|
|
485
|
+
* then we return the end position of the coach to get the closest scroll position of the next focus place.
|
|
330
486
|
* @returns number
|
|
331
487
|
*/
|
|
332
488
|
_getCoachScrollPositionX() {
|
|
@@ -351,8 +507,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
351
507
|
* @returns boolean
|
|
352
508
|
*/
|
|
353
509
|
_isScrollableToSelectedCoach() {
|
|
354
|
-
const currScrollPosX = this.coachScrollArea.scrollLeft;
|
|
355
|
-
const coachScrollWindowWidth = this.coachScrollArea.getBoundingClientRect().width;
|
|
510
|
+
const currScrollPosX = this.alignVertical ? this.coachScrollArea.scrollTop : this.coachScrollArea.scrollLeft;
|
|
511
|
+
const coachScrollWindowWidth = this.alignVertical ? this.coachScrollArea.getBoundingClientRect().height : this.coachScrollArea.getBoundingClientRect().width;
|
|
356
512
|
const maxScrollWidthArea = this.maxCalcCoachsWidth - coachScrollWindowWidth;
|
|
357
513
|
const currCoachTrigger = this.triggerCoachPositionsCollection[this.currSelectedCoachIndex];
|
|
358
514
|
const isScrollPosSameToCurrCoachPos = currScrollPosX === this.triggerCoachPositionsCollection[this.currSelectedCoachIndex].start;
|
|
@@ -363,7 +519,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
363
519
|
* @returns number
|
|
364
520
|
*/
|
|
365
521
|
_getCoachIndexByScrollTriggerPosition() {
|
|
366
|
-
const
|
|
522
|
+
const scrollPos = this.alignVertical ? this.coachScrollArea.scrollTop : this.coachScrollArea.scrollLeft;
|
|
523
|
+
const scrollOffsetX = scrollPos + this.scrollCoachsAreaWidth / 2;
|
|
367
524
|
return this.triggerCoachPositionsCollection.findIndex((coachTrigger) => scrollOffsetX >= coachTrigger.start && scrollOffsetX <= coachTrigger.end);
|
|
368
525
|
}
|
|
369
526
|
/**
|
|
@@ -374,7 +531,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
374
531
|
let firstPlace = null;
|
|
375
532
|
const coach = this.seatReservation?.coachItems[this.currSelectedCoachIndex];
|
|
376
533
|
const firstCellId = "cell-" + this.currSelectedCoachIndex + "-0-0";
|
|
377
|
-
const placeNumber = this.shadowRoot?.querySelector("
|
|
534
|
+
const placeNumber = this.shadowRoot?.querySelector("#" + firstCellId)?.querySelector("sbb-seat-reservation-place-control")?.getAttribute("text") || null;
|
|
378
535
|
if (coach && placeNumber) {
|
|
379
536
|
firstPlace = coach.places?.find((place) => place.number === placeNumber) || null;
|
|
380
537
|
}
|
|
@@ -420,64 +577,13 @@ let SeatReservationBaseElement = (() => {
|
|
|
420
577
|
}
|
|
421
578
|
return closestPlace;
|
|
422
579
|
}
|
|
423
|
-
|
|
424
|
-
this.unfocusPlaceElement();
|
|
425
|
-
if (place) {
|
|
426
|
-
this.currSelectedPlace = place;
|
|
427
|
-
if (coachIndex) {
|
|
428
|
-
this.currSelectedCoachIndex = coachIndex;
|
|
429
|
-
}
|
|
430
|
-
this._setCurrSelectedPlaceElementId(place);
|
|
431
|
-
const selectedPlaceElement = this._getPlaceHtmlElement();
|
|
432
|
-
if (selectedPlaceElement) {
|
|
433
|
-
selectedPlaceElement.setAttribute("keyfocus", "focus");
|
|
434
|
-
}
|
|
435
|
-
}
|
|
436
|
-
}
|
|
437
|
-
unfocusPlaceElement() {
|
|
438
|
-
const selectedPlaceElement = this._getPlaceHtmlElement();
|
|
439
|
-
if (selectedPlaceElement) {
|
|
440
|
-
selectedPlaceElement.setAttribute("keyfocus", "unfocus");
|
|
441
|
-
this._setCurrSelectedPlaceElementId(null);
|
|
442
|
-
this.currSelectedPlace = null;
|
|
443
|
-
}
|
|
444
|
-
}
|
|
445
|
-
getCalculatedDimension(elementDimension, coachDimension, isOriginHeight, isStretchHeight) {
|
|
446
|
-
if (coachDimension && !isOriginHeight) {
|
|
447
|
-
elementDimension.h += this.coachBorderOffset * 2;
|
|
448
|
-
}
|
|
449
|
-
if (isStretchHeight) {
|
|
450
|
-
elementDimension.h += this.coachBorderOffset;
|
|
451
|
-
}
|
|
452
|
-
return {
|
|
453
|
-
w: this.baseGridSize * elementDimension.w,
|
|
454
|
-
h: this.baseGridSize * elementDimension.h
|
|
455
|
-
};
|
|
456
|
-
}
|
|
457
|
-
getCalculatedPosition(elementPosition, elementDimension, coachDimension, isOriginHeight) {
|
|
458
|
-
if (coachDimension && elementDimension) {
|
|
459
|
-
const endPosHeight = isOriginHeight ? coachDimension.h : coachDimension.h + this.coachBorderOffset;
|
|
460
|
-
if (elementPosition.y === 0) {
|
|
461
|
-
elementPosition.y -= this.coachBorderOffset;
|
|
462
|
-
} else if (elementPosition.y + elementDimension.h === endPosHeight) {
|
|
463
|
-
elementPosition.y += this.coachBorderOffset;
|
|
464
|
-
}
|
|
465
|
-
}
|
|
466
|
-
return {
|
|
467
|
-
x: this.baseGridSize * elementPosition.x,
|
|
468
|
-
y: this.baseGridSize * elementPosition.y,
|
|
469
|
-
z: elementPosition.z
|
|
470
|
-
};
|
|
471
|
-
}
|
|
472
|
-
// Handling for Tab navigation if an place is selected inside the coach.
|
|
580
|
+
// Handling for Tab navigation if a place is selected inside the coach.
|
|
473
581
|
// This controls the focused coach from the current selected coach.
|
|
474
582
|
_navigateCoachNavigationByKeyboard(tabDirection) {
|
|
475
583
|
const currFocusIndex = this.focusedCoachIndex === -1 ? this.currSelectedCoachIndex === -1 ? 0 : this.currSelectedCoachIndex : this.focusedCoachIndex;
|
|
476
584
|
const newFocusableIndex = tabDirection === "NEXT_TAB" ? this.getNextAvailableCoachIndex(currFocusIndex) : this.getPrevAvailableCoachIndex(currFocusIndex);
|
|
477
585
|
if (currFocusIndex === newFocusableIndex) {
|
|
478
586
|
this.unfocusPlaceElement();
|
|
479
|
-
this.selectedCoachIndex = -1;
|
|
480
|
-
this.currSelectedCoachIndex = -1;
|
|
481
587
|
this.seatReservationWithoutNavigationHasFocus = false;
|
|
482
588
|
if (tabDirection === "NEXT_TAB")
|
|
483
589
|
this.lastTabElement.focus();
|
|
@@ -492,6 +598,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
492
598
|
if (placeInCoachHasFocus) {
|
|
493
599
|
this.focusedCoachIndex = currFocusIndex;
|
|
494
600
|
this.unfocusPlaceElement();
|
|
601
|
+
return;
|
|
495
602
|
} else {
|
|
496
603
|
this.focusedCoachIndex = newFocusableIndex;
|
|
497
604
|
}
|
|
@@ -507,6 +614,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
507
614
|
this._setFocusToSelectedCoachGrid();
|
|
508
615
|
}
|
|
509
616
|
}
|
|
617
|
+
this._scrollToSelectedNavigationButton(newFocusableIndex);
|
|
510
618
|
} else {
|
|
511
619
|
this.scrollToSelectedNavCoach(newFocusableIndex);
|
|
512
620
|
}
|
|
@@ -538,16 +646,10 @@ let SeatReservationBaseElement = (() => {
|
|
|
538
646
|
return startIndex > 0 ? startIndex - 1 : startIndex;
|
|
539
647
|
}
|
|
540
648
|
updateSelectedSeatReservationPlaces(placeSelection) {
|
|
541
|
-
if (placeSelection.
|
|
542
|
-
|
|
543
|
-
if (seatReservationSelection) {
|
|
544
|
-
this.selectedSeatReservationPlaces.push(seatReservationSelection);
|
|
545
|
-
}
|
|
649
|
+
if (placeSelection.placeType === "SEAT") {
|
|
650
|
+
this.selectedSeatReservationPlaces.seats = this._updateSelectedSeatReservationPlaces(this.selectedSeatReservationPlaces.seats, this.maxSeatReservations, placeSelection);
|
|
546
651
|
} else {
|
|
547
|
-
this.selectedSeatReservationPlaces = this.selectedSeatReservationPlaces.
|
|
548
|
-
}
|
|
549
|
-
if (this.maxReservations && this.selectedSeatReservationPlaces.length > this.maxReservations) {
|
|
550
|
-
this._resetAllPlaceSelections(placeSelection);
|
|
652
|
+
this.selectedSeatReservationPlaces.bicycles = this._updateSelectedSeatReservationPlaces(this.selectedSeatReservationPlaces.bicycles, this.maxBicycleReservations, placeSelection);
|
|
551
653
|
}
|
|
552
654
|
this.dispatchEvent(new CustomEvent("selectedplaces", {
|
|
553
655
|
bubbles: true,
|
|
@@ -555,6 +657,24 @@ let SeatReservationBaseElement = (() => {
|
|
|
555
657
|
detail: this.selectedSeatReservationPlaces
|
|
556
658
|
}));
|
|
557
659
|
}
|
|
660
|
+
_updateSelectedSeatReservationPlaces(selectedSeatReservationPlaces, maxReservations, placeSelection) {
|
|
661
|
+
if (placeSelection.state === "SELECTED") {
|
|
662
|
+
const seatReservationSelection = this._getSeatReservationPlaceSelection(placeSelection);
|
|
663
|
+
if (seatReservationSelection) {
|
|
664
|
+
selectedSeatReservationPlaces.push(seatReservationSelection);
|
|
665
|
+
}
|
|
666
|
+
} else {
|
|
667
|
+
selectedSeatReservationPlaces = selectedSeatReservationPlaces.filter((_selectedPlace) => _selectedPlace.id !== placeSelection.id);
|
|
668
|
+
}
|
|
669
|
+
if (maxReservations > -1 && selectedSeatReservationPlaces.length > maxReservations) {
|
|
670
|
+
if (maxReservations === 0) {
|
|
671
|
+
selectedSeatReservationPlaces = this._resetAllPlaceSelections(selectedSeatReservationPlaces);
|
|
672
|
+
} else {
|
|
673
|
+
selectedSeatReservationPlaces = this._resetAllPlaceSelections(selectedSeatReservationPlaces, placeSelection);
|
|
674
|
+
}
|
|
675
|
+
}
|
|
676
|
+
return selectedSeatReservationPlaces;
|
|
677
|
+
}
|
|
558
678
|
updateCurrentSelectedPlaceInCoach(placeSelection) {
|
|
559
679
|
const coachIndex = placeSelection.coachIndex;
|
|
560
680
|
const place = this.seatReservation.coachItems[coachIndex].places?.find((place2) => place2.number == placeSelection.number);
|
|
@@ -588,27 +708,34 @@ let SeatReservationBaseElement = (() => {
|
|
|
588
708
|
coach.places?.filter((place) => place.state === "SELECTED")?.forEach((place) => {
|
|
589
709
|
const preselectedPlaceSelection = mapPlaceInfosToPlaceSelection(place, coachIndex);
|
|
590
710
|
const seatReservationPlaceSelection = this._getSeatReservationPlaceSelection(preselectedPlaceSelection);
|
|
591
|
-
if (seatReservationPlaceSelection)
|
|
592
|
-
|
|
711
|
+
if (seatReservationPlaceSelection) {
|
|
712
|
+
if (seatReservationPlaceSelection.placeType === "SEAT") {
|
|
713
|
+
this.selectedSeatReservationPlaces.seats.push(seatReservationPlaceSelection);
|
|
714
|
+
} else {
|
|
715
|
+
this.selectedSeatReservationPlaces.bicycles.push(seatReservationPlaceSelection);
|
|
716
|
+
}
|
|
717
|
+
}
|
|
593
718
|
});
|
|
594
719
|
});
|
|
595
720
|
}
|
|
596
721
|
/**
|
|
597
722
|
* All selected places will be reset or the currentSelectedPlace was given, then we reset all except currentSelectedPlace
|
|
723
|
+
* @param reservationPlaceSelections
|
|
598
724
|
* @param currSelectedPlace
|
|
599
725
|
*/
|
|
600
|
-
_resetAllPlaceSelections(currSelectedPlace) {
|
|
601
|
-
for (const placeSelection of
|
|
726
|
+
_resetAllPlaceSelections(reservationPlaceSelections, currSelectedPlace) {
|
|
727
|
+
for (const placeSelection of reservationPlaceSelections) {
|
|
602
728
|
if (!currSelectedPlace || currSelectedPlace.id !== placeSelection.id) {
|
|
603
729
|
const placeElement = this.shadowRoot?.getElementById(placeSelection.id);
|
|
604
730
|
placeElement.setAttribute("state", "FREE");
|
|
605
731
|
}
|
|
606
732
|
}
|
|
607
733
|
if (currSelectedPlace) {
|
|
608
|
-
|
|
734
|
+
reservationPlaceSelections = reservationPlaceSelections.filter((_selectedPlace) => _selectedPlace.id === currSelectedPlace.id);
|
|
609
735
|
} else {
|
|
610
|
-
|
|
736
|
+
reservationPlaceSelections = [];
|
|
611
737
|
}
|
|
738
|
+
return reservationPlaceSelections;
|
|
612
739
|
}
|
|
613
740
|
_getSeatReservationPlaceSelection(currSelectedPlace) {
|
|
614
741
|
const coach = this.seatReservation.coachItems[currSelectedPlace.coachIndex];
|
|
@@ -619,7 +746,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
619
746
|
if (!this.seatReservation.coachItems[coachIndex])
|
|
620
747
|
return null;
|
|
621
748
|
const coach = this.seatReservation.coachItems[coachIndex];
|
|
622
|
-
|
|
749
|
+
const coachNumberOfFreePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);
|
|
750
|
+
return mapCoachInfosToCoachSelection(coachIndex, coach, coachNumberOfFreePlaces);
|
|
623
751
|
}
|
|
624
752
|
_setCurrSelectedPlaceElementId(place) {
|
|
625
753
|
if (place) {
|
|
@@ -630,7 +758,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
630
758
|
}
|
|
631
759
|
/**
|
|
632
760
|
* Returns the current selected place HTML element by given placeNumber and coachIndex.
|
|
633
|
-
* If both
|
|
761
|
+
* If both doesn't exist, we try to return the place HTML element by the _currentSelectedPlaceElementId
|
|
634
762
|
* @param placeNumber optional as string
|
|
635
763
|
* @param coachIndex optional as string
|
|
636
764
|
* @returns HTMLElement or null
|
|
@@ -640,27 +768,20 @@ let SeatReservationBaseElement = (() => {
|
|
|
640
768
|
const coachPlaceNumberId = placeNumber ? "seat-reservation__place-button-" + currCoachIndex + "-" + placeNumber : this.currSelectedPlaceElementId;
|
|
641
769
|
return coachPlaceNumberId ? this.shadowRoot?.getElementById(coachPlaceNumberId) || null : null;
|
|
642
770
|
}
|
|
643
|
-
|
|
644
|
-
_setVerticalAlignmentOffset() {
|
|
645
|
-
setTimeout(() => {
|
|
646
|
-
const seatReservationWrapperElement = this.shadowRoot?.querySelector(".sbb-sr__wrapper");
|
|
647
|
-
if (seatReservationWrapperElement) {
|
|
648
|
-
const seatReservationVerticalOffset = seatReservationWrapperElement.getBoundingClientRect().width;
|
|
649
|
-
this.style?.setProperty("--sbb-seat-reservation-vertical-offset", `${seatReservationVerticalOffset}px`);
|
|
650
|
-
}
|
|
651
|
-
});
|
|
652
|
-
}
|
|
653
|
-
}, _seatReservation_accessor_storage = new WeakMap(), _hasNavigation_accessor_storage = new WeakMap(), _alignVertical_accessor_storage = new WeakMap(), _baseGridSize_accessor_storage = new WeakMap(), _height_accessor_storage = new WeakMap(), _maxReservations_accessor_storage = new WeakMap(), _preventPlaceClick_accessor_storage = new WeakMap(), _selectedCoachIndex_accessor_storage = new WeakMap(), _focusedCoachIndex_accessor_storage = new WeakMap(), (() => {
|
|
771
|
+
}, _seatReservation_accessor_storage = new WeakMap(), _hasNavigation_accessor_storage = new WeakMap(), _alignVertical_accessor_storage = new WeakMap(), _baseGridSize_accessor_storage = new WeakMap(), _height_accessor_storage = new WeakMap(), _maxSeatReservations_accessor_storage = new WeakMap(), _maxBicycleReservations_accessor_storage = new WeakMap(), _preventPlaceClick_accessor_storage = new WeakMap(), _preselectCoachIndex_accessor_storage = new WeakMap(), _selectedCoachIndex_accessor_storage = new WeakMap(), _focusedCoachIndex_accessor_storage = new WeakMap(), _hoveredScrollCoachIndex_accessor_storage = new WeakMap(), (() => {
|
|
654
772
|
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
|
|
655
773
|
_seatReservation_decorators = [property({ attribute: "seat-reservation", type: Object })];
|
|
656
774
|
_hasNavigation_decorators = [forceType(), property({ attribute: "has-navigation", type: Boolean })];
|
|
657
775
|
_alignVertical_decorators = [forceType(), property({ attribute: "align-vertical", type: Boolean })];
|
|
658
776
|
_baseGridSize_decorators = [forceType(), property({ attribute: "base-grid-size", type: Number })];
|
|
659
777
|
_height_decorators = [forceType(), property({ attribute: "height", type: Number })];
|
|
660
|
-
|
|
778
|
+
_maxSeatReservations_decorators = [forceType(), property({ attribute: "max-seat-reservations", type: Number })];
|
|
779
|
+
_maxBicycleReservations_decorators = [forceType(), property({ attribute: "max-bicycle-reservations", type: Number })];
|
|
661
780
|
_preventPlaceClick_decorators = [forceType(), property({ attribute: "prevent-place-click", type: Boolean })];
|
|
781
|
+
_preselectCoachIndex_decorators = [forceType(), property({ attribute: "preselect-coach-index", type: Number })];
|
|
662
782
|
_selectedCoachIndex_decorators = [state()];
|
|
663
783
|
_focusedCoachIndex_decorators = [state()];
|
|
784
|
+
_hoveredScrollCoachIndex_decorators = [state()];
|
|
664
785
|
__esDecorate(_a, null, _seatReservation_decorators, { kind: "accessor", name: "seatReservation", static: false, private: false, access: { has: (obj) => "seatReservation" in obj, get: (obj) => obj.seatReservation, set: (obj, value) => {
|
|
665
786
|
obj.seatReservation = value;
|
|
666
787
|
} }, metadata: _metadata }, _seatReservation_initializers, _seatReservation_extraInitializers);
|
|
@@ -676,18 +797,27 @@ let SeatReservationBaseElement = (() => {
|
|
|
676
797
|
__esDecorate(_a, null, _height_decorators, { kind: "accessor", name: "height", static: false, private: false, access: { has: (obj) => "height" in obj, get: (obj) => obj.height, set: (obj, value) => {
|
|
677
798
|
obj.height = value;
|
|
678
799
|
} }, metadata: _metadata }, _height_initializers, _height_extraInitializers);
|
|
679
|
-
__esDecorate(_a, null,
|
|
680
|
-
obj.
|
|
681
|
-
} }, metadata: _metadata },
|
|
800
|
+
__esDecorate(_a, null, _maxSeatReservations_decorators, { kind: "accessor", name: "maxSeatReservations", static: false, private: false, access: { has: (obj) => "maxSeatReservations" in obj, get: (obj) => obj.maxSeatReservations, set: (obj, value) => {
|
|
801
|
+
obj.maxSeatReservations = value;
|
|
802
|
+
} }, metadata: _metadata }, _maxSeatReservations_initializers, _maxSeatReservations_extraInitializers);
|
|
803
|
+
__esDecorate(_a, null, _maxBicycleReservations_decorators, { kind: "accessor", name: "maxBicycleReservations", static: false, private: false, access: { has: (obj) => "maxBicycleReservations" in obj, get: (obj) => obj.maxBicycleReservations, set: (obj, value) => {
|
|
804
|
+
obj.maxBicycleReservations = value;
|
|
805
|
+
} }, metadata: _metadata }, _maxBicycleReservations_initializers, _maxBicycleReservations_extraInitializers);
|
|
682
806
|
__esDecorate(_a, null, _preventPlaceClick_decorators, { kind: "accessor", name: "preventPlaceClick", static: false, private: false, access: { has: (obj) => "preventPlaceClick" in obj, get: (obj) => obj.preventPlaceClick, set: (obj, value) => {
|
|
683
807
|
obj.preventPlaceClick = value;
|
|
684
808
|
} }, metadata: _metadata }, _preventPlaceClick_initializers, _preventPlaceClick_extraInitializers);
|
|
809
|
+
__esDecorate(_a, null, _preselectCoachIndex_decorators, { kind: "accessor", name: "preselectCoachIndex", static: false, private: false, access: { has: (obj) => "preselectCoachIndex" in obj, get: (obj) => obj.preselectCoachIndex, set: (obj, value) => {
|
|
810
|
+
obj.preselectCoachIndex = value;
|
|
811
|
+
} }, metadata: _metadata }, _preselectCoachIndex_initializers, _preselectCoachIndex_extraInitializers);
|
|
685
812
|
__esDecorate(_a, null, _selectedCoachIndex_decorators, { kind: "accessor", name: "selectedCoachIndex", static: false, private: false, access: { has: (obj) => "selectedCoachIndex" in obj, get: (obj) => obj.selectedCoachIndex, set: (obj, value) => {
|
|
686
813
|
obj.selectedCoachIndex = value;
|
|
687
814
|
} }, metadata: _metadata }, _selectedCoachIndex_initializers, _selectedCoachIndex_extraInitializers);
|
|
688
815
|
__esDecorate(_a, null, _focusedCoachIndex_decorators, { kind: "accessor", name: "focusedCoachIndex", static: false, private: false, access: { has: (obj) => "focusedCoachIndex" in obj, get: (obj) => obj.focusedCoachIndex, set: (obj, value) => {
|
|
689
816
|
obj.focusedCoachIndex = value;
|
|
690
817
|
} }, metadata: _metadata }, _focusedCoachIndex_initializers, _focusedCoachIndex_extraInitializers);
|
|
818
|
+
__esDecorate(_a, null, _hoveredScrollCoachIndex_decorators, { kind: "accessor", name: "hoveredScrollCoachIndex", static: false, private: false, access: { has: (obj) => "hoveredScrollCoachIndex" in obj, get: (obj) => obj.hoveredScrollCoachIndex, set: (obj, value) => {
|
|
819
|
+
obj.hoveredScrollCoachIndex = value;
|
|
820
|
+
} }, metadata: _metadata }, _hoveredScrollCoachIndex_initializers, _hoveredScrollCoachIndex_extraInitializers);
|
|
691
821
|
if (_metadata) Object.defineProperty(_a, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
|
|
692
822
|
})(), _a.events = {
|
|
693
823
|
selectedplaces: "selectedplaces",
|
|
@@ -697,4 +827,4 @@ let SeatReservationBaseElement = (() => {
|
|
|
697
827
|
export {
|
|
698
828
|
SeatReservationBaseElement
|
|
699
829
|
};
|
|
700
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"seat-reservation-base-element.js","sources":["../../../../../src/elements-experimental/seat-reservation/seat-reservation/seat-reservation-base-element.ts"],"sourcesContent":["import { isArrowKeyOrPageKeysPressed } from '@sbb-esta/lyne-elements/core/a11y.js';\nimport { forceType } from '@sbb-esta/lyne-elements/core/decorators.js';\nimport { LitElement, type PropertyValues } from 'lit';\nimport { property, state } from 'lit/decorators.js';\n\nimport {\n  mapCoachInfosToCoachSelection,\n  mapPlaceAndCoachToSeatReservationPlaceSelection,\n  mapPlaceInfosToPlaceSelection,\n} from '../common/mapper.js';\nimport type {\n  CoachItem,\n  ElementDimension,\n  ElementPosition,\n  Place,\n  PlaceSelection,\n  SeatReservation,\n  SeatReservationCoachSelection,\n  SeatReservationPlaceSelection,\n} from '../common.js';\nimport type { SbbSeatReservationPlaceControlElement } from '../seat-reservation-place-control/seat-reservation-place-control.component.js';\nimport type { SbbSeatReservationScopedElement } from '../seat-reservation-scoped/seat-reservation-scoped.component.js';\n\nenum ScrollDirection {\n  right = 'right',\n  left = 'left',\n}\n\ninterface CoachScrollTriggerPoint {\n  start: number;\n  end: number;\n  width: number;\n}\n\nexport type SeatReservationSelectedPlacesEventDetails = SeatReservationPlaceSelection[];\n\nexport class SeatReservationBaseElement extends LitElement {\n  public static readonly events = {\n    selectedplaces: 'selectedplaces',\n    selectedcoach: 'selectedcoach',\n  } as const;\n\n  /** The seat reservation object which contains all coaches and places */\n  @property({ attribute: 'seat-reservation', type: Object })\n  public accessor seatReservation: SeatReservation = null!;\n\n  /** The seat reservation navigation can be toggled by this property */\n  @forceType()\n  @property({ attribute: 'has-navigation', type: Boolean })\n  public accessor hasNavigation: boolean = true;\n\n  /** The seat reservation area is aligned vertically */\n  @forceType()\n  @property({ attribute: 'align-vertical', type: Boolean })\n  public accessor alignVertical: boolean = false;\n\n  /** The seat reservation area's base grid size */\n  @forceType()\n  @property({ attribute: 'base-grid-size', type: Number })\n  public accessor baseGridSize: number = 16;\n\n  /** The seat reservation area's width */\n  @forceType()\n  @property({ attribute: 'height', type: Number })\n  public accessor height: number = null!;\n\n  /** Maximal number of possible clickable seats */\n  @forceType()\n  @property({ attribute: 'max-reservations', type: Number })\n  public accessor maxReservations: number = null!;\n\n  /** Any click functionality is prevented */\n  @forceType()\n  @property({ attribute: 'prevent-place-click', type: Boolean })\n  public accessor preventPlaceClick: boolean = false;\n\n  @state() protected accessor selectedCoachIndex: number = -1;\n  @state() protected accessor focusedCoachIndex: number = -1;\n\n  protected coachBorderPadding = 6;\n  protected coachBorderOffset = this.coachBorderPadding / this.baseGridSize;\n  protected currScrollDirection: ScrollDirection = ScrollDirection.right;\n  protected maxCalcCoachsWidth: number = 0;\n  protected scrollCoachsAreaWidth: number = 0;\n  protected triggerCoachPositionsCollection: CoachScrollTriggerPoint[] = [];\n  protected firstTabElement: HTMLElement = null!;\n  protected lastTabElement: HTMLElement = null!;\n  protected coachScrollArea: HTMLElement = null!;\n  protected currSelectedPlace: Place | null = null;\n  protected currSelectedPlaceElementId: string | null = null;\n  protected currSelectedCoachIndex: number = -1;\n  protected preventCoachScrollByPlaceClick: boolean = false;\n  protected selectedSeatReservationPlaces: SeatReservationPlaceSelection[] = [];\n  protected seatReservationWithoutNavigationHasFocus = false;\n  protected isCoachGridFocusable = false;\n  protected isAutoScrolling = false;\n  protected isKeyboardNavigation = false;\n  protected keyboardNavigationEvents = {\n    ArrowLeft: 'ArrowLeft',\n    ArrowRight: 'ArrowRight',\n    ArrowUp: 'ArrowUp',\n    ArrowDown: 'ArrowDown',\n    Tab: 'Tab',\n    Enter: 'Enter',\n  } as const;\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('seatReservation')) {\n      this._initSeatReservationPlaceSelection();\n    }\n\n    if (changedProperties.has('baseGridSize')) {\n      this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;\n      this.style?.setProperty('--sbb-seat-reservation-grid-size', `${this.baseGridSize}px`);\n\n      if (this.alignVertical) {\n        this._setVerticalAlignmentOffset();\n      }\n    }\n\n    // If the height is used, the baseGridSize must be recalculated\n    if (changedProperties.has('height') && !!this.height) {\n      if (this.seatReservation.coachItems.length) {\n        this.baseGridSize = this.height / this.seatReservation.coachItems[0].dimension.h;\n        this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;\n        this.style?.setProperty('--sbb-seat-reservation-grid-size', `${this.baseGridSize}px`);\n\n        if (this.alignVertical) {\n          this._setVerticalAlignmentOffset();\n        }\n      }\n    }\n\n    if (changedProperties.has('alignVertical') && this.alignVertical) {\n      this._setVerticalAlignmentOffset();\n    }\n  }\n\n  /* Init scroll event handling for coach navigation */\n  protected initNavigationSelectionByScrollEvent(): void {\n    this.firstTabElement = this.shadowRoot?.getElementById('first-tab-element') as HTMLElement;\n    this.lastTabElement = this.shadowRoot?.getElementById('last-tab-element') as HTMLElement;\n    this.coachScrollArea = this.shadowRoot?.getElementById('sbb-sr__parent-area') as HTMLElement;\n\n    if (this.coachScrollArea) {\n      let currCalcTriggerPos = 0;\n      this.scrollCoachsAreaWidth = this.coachScrollArea.getBoundingClientRect().width;\n\n      // Precalculate trigger scroll position array depends from coach width\n      this.triggerCoachPositionsCollection = this.seatReservation.coachItems.map((coach) => {\n        const startPosX = currCalcTriggerPos;\n        const coachWidth = this.getCalculatedDimension(coach.dimension).w;\n        currCalcTriggerPos += coachWidth;\n        return {\n          start: startPosX,\n          end: currCalcTriggerPos,\n          width: coachWidth,\n        } as CoachScrollTriggerPoint;\n      });\n\n      // Set maximum calculated coach width\n      this.maxCalcCoachsWidth = currCalcTriggerPos;\n\n      // At the end of a scroll Events to a coach, the reached wagon is marked as selected\n      this.coachScrollArea.addEventListener('scrollend', () => {\n        const findScrollCoachIndex = this.isAutoScrolling\n          ? this.currSelectedCoachIndex\n          : this._getCoachIndexByScrollTriggerPosition();\n        if (this._isScrollableToSelectedCoach()) {\n          this.currSelectedCoachIndex = findScrollCoachIndex;\n        } else {\n          this.currSelectedCoachIndex =\n            findScrollCoachIndex < this.currSelectedCoachIndex\n              ? this.currSelectedCoachIndex\n              : findScrollCoachIndex;\n        }\n\n        this.preventCoachScrollByPlaceClick = false;\n        this.updateCurrentSelectedCoach();\n\n        if (!this.hasNavigation) {\n          this.preselectPlaceInCoach();\n          this.isAutoScrolling = false;\n        }\n      });\n\n      // During initialization, we check vertical alignment mode. In Vertical mode we have to set the vertical offset manual for the seat reservation area,\n      // because we rotate the entire component by 90 degrees and transform the origin point to top left.\n      if (this.alignVertical) {\n        this._setVerticalAlignmentOffset();\n      }\n    }\n  }\n\n  /**\n   * If no navigation exists (property setting -> hasNavigation) and a table coach gets the focus,\n   * the first place in the coach must be automatically preselected to control the place navigation via keyboard\n   *\n   * @param focusCoachIndex\n   */\n  protected onFocusTableCoachAndPreselectPlace(focusCoachIndex: number): void {\n    if (!this.seatReservationWithoutNavigationHasFocus && !this.hasNavigation) {\n      this.seatReservationWithoutNavigationHasFocus = true;\n      this.currSelectedCoachIndex =\n        focusCoachIndex === 0\n          ? this.getNextAvailableCoachIndex(-1)\n          : this.getPrevAvailableCoachIndex(focusCoachIndex);\n      this.preselectPlaceInCoach();\n    }\n  }\n\n  /**\n   * Initialisation of Keyboard event handling to navigation between each places inside a selected coach by using [arrow] keys.\n   * With the [TAB] key the user navigation goes to the next coach navigation element and the currently selected place is automatically reset.\n   */\n  protected handleKeyboardEvent(event: KeyboardEvent): void {\n    const pressedKey = event.key;\n\n    // If any place is selected and TAB Key combination ist pressed,\n    // then we handle the next or previous coach selection\n    if (this.currSelectedPlace) {\n      if (event.shiftKey && event.keyCode === 9) {\n        this._navigateCoachNavigationByKeyboard('PREV_TAB');\n        event.preventDefault();\n        return;\n      }\n\n      if (pressedKey === this.keyboardNavigationEvents.Tab) {\n        this._navigateCoachNavigationByKeyboard('NEXT_TAB');\n        event.preventDefault();\n        return;\n      }\n    }\n\n    // Check if a coach is selected and the arrow key is pressed\n    if (this.currSelectedCoachIndex !== -1 && isArrowKeyOrPageKeysPressed(event)) {\n      event.preventDefault();\n\n      switch (pressedKey) {\n        case this.keyboardNavigationEvents.ArrowLeft:\n          {\n            const pressedLeftKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowDown\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedLeftKeyMapping);\n          }\n          break;\n        case this.keyboardNavigationEvents.ArrowRight:\n          {\n            const pressedRightKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowUp\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedRightKeyMapping);\n          }\n          break;\n        case this.keyboardNavigationEvents.ArrowUp:\n          {\n            const pressedUpKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowLeft\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedUpKeyMapping);\n          }\n          break;\n        case this.keyboardNavigationEvents.ArrowDown:\n          {\n            const pressedDownKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowRight\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedDownKeyMapping);\n          }\n          break;\n        default:\n          break;\n      }\n    }\n  }\n\n  /**\n   * Selects a place inside the coach if navigated via keyboard,\n   * otherwise the coach grid is selected (necessary for ScreenReader)\n   */\n  protected preselectPlaceInCoach(): void {\n    const closestPlace = this._getClosestPlaceByKeyDirection();\n    //If closestPlace exist, we have to unfocus previouse focused place\n    if (closestPlace) {\n      this.unfocusPlaceElement();\n    }\n\n    // Only when keyboard navigation is used and coaches are scrolled by auto scrolling,\n    // then we can set the focus on the first place in the coach.\n    if (this.isKeyboardNavigation && this.isAutoScrolling) {\n      if (closestPlace) {\n        this.focusPlaceElement(closestPlace);\n      }\n    }\n    // In cases where the preselection function is triggered by normal clicking or via screenreader via tab,\n    // we only focus the table without directly focusing the place.\n    else {\n      // We need to set the currSelectedPlace here for further correct functioning navigation via tab.\n      this.currSelectedPlace = closestPlace;\n      this._setFocusToSelectedCoachGrid();\n    }\n  }\n\n  protected scrollToSelectedNavCoach(selectedNavCoachIndex: number): void {\n    if (selectedNavCoachIndex !== this.currSelectedCoachIndex) {\n      this.isAutoScrolling = true;\n      this.isCoachGridFocusable = true;\n      this.currSelectedCoachIndex = selectedNavCoachIndex;\n      this._setScrollDirectionByCoachIndex();\n\n      const scrollToCoachPosX = this._getCoachScrollPositionX();\n      const isSelectedCoachIndexScrollable =\n        this.selectedCoachIndex !== -1 || this.currSelectedCoachIndex > 0;\n\n      // Checks whether the current scroll position allows scrolling to the next wagon or not\n      if (isSelectedCoachIndexScrollable && this._isScrollableToSelectedCoach()) {\n        this.coachScrollArea.scrollTo({\n          top: 0,\n          left: scrollToCoachPosX,\n          behavior: 'smooth',\n        });\n      } else {\n        this.updateCurrentSelectedCoach();\n      }\n    }\n  }\n\n  /**\n   * Sets the new ScrollDirection by the new given target coach index.\n   */\n  private _setScrollDirectionByCoachIndex(): void {\n    this.currScrollDirection =\n      this.currSelectedCoachIndex > this.selectedCoachIndex\n        ? ScrollDirection.right\n        : ScrollDirection.left;\n  }\n\n  /**\n   * Returns the scroll start or end position X from the selected coach.\n   * In case the user is curretnly navigate throught places by keyboard and goes to previous coach,\n   * then we return the end position of the coach to get clostest next scroll position of the next focus place.\n   * @returns number\n   */\n  private _getCoachScrollPositionX(): number {\n    const coachTriggerPoint = this.triggerCoachPositionsCollection[this.currSelectedCoachIndex];\n    const isFocusPlaceFromPreviousCoachPosition =\n      this.isKeyboardNavigation &&\n      this.currScrollDirection === ScrollDirection.left &&\n      coachTriggerPoint.width > this.scrollCoachsAreaWidth;\n    return isFocusPlaceFromPreviousCoachPosition\n      ? coachTriggerPoint.end - this.scrollCoachsAreaWidth\n      : coachTriggerPoint.start;\n  }\n\n  /**\n   * Sets the focus on the HTML table (grid) caption element so that the heading is read out when using a ScreenReader.\n   */\n  private _setFocusToSelectedCoachGrid(): void {\n    // When the user performs an action that affects the coach navigation, then the navigated table is focusable.\n    if (this.isCoachGridFocusable) {\n      this.isCoachGridFocusable = false;\n      const coachTableCaptionElement = this.shadowRoot?.querySelector(\n        '#sbb-sr-coach-caption-' + this.currSelectedCoachIndex,\n      ) as HTMLTableCaptionElement;\n      if (coachTableCaptionElement) {\n        coachTableCaptionElement.focus();\n      }\n    }\n  }\n\n  /**\n   * Returns whether the current scrolled position can be used to scroll to the selected wagon\n   * @returns boolean\n   */\n  private _isScrollableToSelectedCoach(): boolean {\n    const currScrollPosX = this.coachScrollArea.scrollLeft;\n    const coachScrollWindowWidth = this.coachScrollArea.getBoundingClientRect().width;\n    const maxScrollWidthArea = this.maxCalcCoachsWidth - coachScrollWindowWidth;\n    const currCoachTrigger = this.triggerCoachPositionsCollection[this.currSelectedCoachIndex];\n    const isScrollPosSameToCurrCoachPos =\n      currScrollPosX === this.triggerCoachPositionsCollection[this.currSelectedCoachIndex].start;\n\n    return (\n      (currScrollPosX < maxScrollWidthArea || currScrollPosX > currCoachTrigger.start) &&\n      !isScrollPosSameToCurrCoachPos\n    );\n  }\n\n  /**\n   * Returns the coach index which is currently visible in the scroll area\n   * @returns number\n   */\n  private _getCoachIndexByScrollTriggerPosition(): number {\n    const scrollOffsetX = this.coachScrollArea.scrollLeft + this.scrollCoachsAreaWidth / 2;\n    return this.triggerCoachPositionsCollection.findIndex(\n      (coachTrigger) => scrollOffsetX >= coachTrigger.start && scrollOffsetX <= coachTrigger.end,\n    );\n  }\n\n  /**\n   * Get the first place of current selected coach by table cell coordinate 0-0 id.\n   * @returns Place or null\n   */\n  private _getFirstPlaceInSelecedCoach(): Place | null {\n    let firstPlace: Place | null = null;\n    const coach = this.seatReservation?.coachItems[this.currSelectedCoachIndex];\n    const firstCellId = 'cell-' + this.currSelectedCoachIndex + '-0-0';\n    const placeNumber =\n      this.shadowRoot\n        ?.querySelector<SbbSeatReservationScopedElement>(\"[cell-id='\" + firstCellId + \"']\")\n        ?.querySelector<SbbSeatReservationPlaceControlElement>('sbb-seat-reservation-place-control')\n        ?.getAttribute('text') || null;\n\n    if (coach && placeNumber) {\n      firstPlace = coach.places?.find((place) => place.number === placeNumber) || null;\n    }\n    return firstPlace;\n  }\n\n  /**\n   * To get the correct closest place of current pressed key and the current selected place,\n   * we have to investigate the coordinates of each place to find the closest place of the currSelectedPlaceElementId.\n   * @param pressedKey\n   * @returns Place or null\n   */\n  private _getClosestPlaceByKeyDirection(pressedKey?: string): Place | null {\n    const coach = this.seatReservation?.coachItems[this.currSelectedCoachIndex];\n    let closestPlace = null;\n    if (coach.places) {\n      // If no place set, then we use initial the left-top place on the coach\n      if (!this.currSelectedPlaceElementId) {\n        return this._getFirstPlaceInSelecedCoach();\n      } else {\n        if (this.currSelectedPlace) {\n          for (const place of coach.places) {\n            // If key pressed, then we try to find the place of the current currScrollDirection\n            if (!pressedKey) {\n              //Find place from the left side of coach by y coordinate. Current currScrollDirection is RIGHT)\n              if (\n                this.currScrollDirection === ScrollDirection.right &&\n                place.position.y === this.currSelectedPlace?.position.y &&\n                (!closestPlace || place.position.x < closestPlace.position.x)\n              ) {\n                closestPlace = place;\n              }\n              //Find place from the right side of coach by y coordinate. Current currScrollDirection is LEFT\n              else if (\n                this.currScrollDirection === ScrollDirection.left &&\n                place.position.y === this.currSelectedPlace?.position.y &&\n                (!closestPlace || place.position.x > closestPlace.position.x)\n              ) {\n                closestPlace = place;\n              }\n            } else {\n              if (place.number !== this.currSelectedPlace?.number) {\n                //Key [Right] navigation, we check the place coordinates of the x-axis to get the smallest larger x place coordinate of the currently selected place\n                if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowRight &&\n                  (place.position.y === this.currSelectedPlace.position.y ||\n                    place.position.y === this.currSelectedPlace.position.y - 1) &&\n                  place.position.x > this.currSelectedPlace.position.x &&\n                  (!closestPlace || place.position.x < closestPlace.position.x)\n                ) {\n                  closestPlace = place;\n                }\n                //Key [Down] navigation, we check the place coordinates of the y-axis to get the smallest larger y place coordinate of the currently selected place\n                else if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowDown &&\n                  (place.position.x === this.currSelectedPlace.position.x ||\n                    place.position.x === this.currSelectedPlace.position.x + 1) &&\n                  place.position.y > this.currSelectedPlace.position.y &&\n                  (!closestPlace || place.position.y < closestPlace.position.y)\n                ) {\n                  closestPlace = place;\n                }\n                //Key [Left] navigation, we check the place coordinates of the x-axis to get the greatest smaller x place coordinate of the currently selected place\n                else if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowLeft &&\n                  (place.position.y === this.currSelectedPlace.position.y ||\n                    place.position.y === this.currSelectedPlace.position.y + 1) &&\n                  place.position.x < this.currSelectedPlace.position.x &&\n                  (!closestPlace || place.position.x > closestPlace.position.x)\n                ) {\n                  closestPlace = place;\n                }\n                //Key [Up] navigation, we check the place coordinates of the y-axis to get the greatest smaller y place coordinate of the currently selected place\n                else if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowUp &&\n                  (place.position.x === this.currSelectedPlace.position.x ||\n                    place.position.x === this.currSelectedPlace.position.x - 1) &&\n                  place.position.y < this.currSelectedPlace?.position.y &&\n                  (!closestPlace || place.position.y > closestPlace.position.y)\n                ) {\n                  closestPlace = place;\n                }\n              }\n            }\n          }\n        }\n      }\n    }\n    return closestPlace;\n  }\n\n  protected focusPlaceElement(place: Place | null, coachIndex?: number): void {\n    this.unfocusPlaceElement();\n    if (place) {\n      this.currSelectedPlace = place;\n      if (coachIndex) {\n        this.currSelectedCoachIndex = coachIndex;\n      }\n\n      this._setCurrSelectedPlaceElementId(place);\n\n      const selectedPlaceElement = this._getPlaceHtmlElement();\n      if (selectedPlaceElement) {\n        selectedPlaceElement.setAttribute('keyfocus', 'focus');\n      }\n    }\n  }\n\n  protected unfocusPlaceElement(): void {\n    const selectedPlaceElement = this._getPlaceHtmlElement();\n    if (selectedPlaceElement) {\n      selectedPlaceElement.setAttribute('keyfocus', 'unfocus');\n      this._setCurrSelectedPlaceElementId(null);\n      this.currSelectedPlace = null;\n    }\n  }\n\n  protected getCalculatedDimension(\n    elementDimension: ElementDimension,\n    coachDimension?: ElementDimension,\n    isOriginHeight?: boolean,\n    isStretchHeight?: boolean,\n  ): ElementDimension {\n    if (coachDimension && !isOriginHeight) {\n      elementDimension.h += this.coachBorderOffset * 2;\n    }\n\n    if (isStretchHeight) {\n      elementDimension.h += this.coachBorderOffset;\n    }\n\n    return {\n      w: this.baseGridSize * elementDimension.w,\n      h: this.baseGridSize * elementDimension.h,\n    };\n  }\n\n  protected getCalculatedPosition(\n    elementPosition: ElementPosition,\n    elementDimension?: ElementDimension,\n    coachDimension?: ElementDimension,\n    isOriginHeight?: boolean,\n  ): ElementPosition {\n    if (coachDimension && elementDimension) {\n      const endPosHeight = isOriginHeight\n        ? coachDimension.h\n        : coachDimension.h + this.coachBorderOffset;\n      //If the original element is positioned at the top or bottom of the coach, we need to recalculate the Y coordinate with the additional border padding\n      if (elementPosition.y === 0) {\n        elementPosition.y -= this.coachBorderOffset;\n      } else if (elementPosition.y + elementDimension.h === endPosHeight) {\n        elementPosition.y += this.coachBorderOffset;\n      }\n    }\n\n    return {\n      x: this.baseGridSize * elementPosition.x,\n      y: this.baseGridSize * elementPosition.y,\n      z: elementPosition.z,\n    };\n  }\n\n  // Handling for Tab navigation if an place is selected inside the coach.\n  // This controls the focused coach from the current selected coach.\n  private _navigateCoachNavigationByKeyboard(tabDirection: string): void {\n    const currFocusIndex =\n      this.focusedCoachIndex === -1\n        ? this.currSelectedCoachIndex === -1\n          ? 0\n          : this.currSelectedCoachIndex\n        : this.focusedCoachIndex;\n    // Check next or prev tab is pressed, then we need to find the next available coach index that should receive the focus\n    const newFocusableIndex: number =\n      tabDirection === 'NEXT_TAB'\n        ? this.getNextAvailableCoachIndex(currFocusIndex)\n        : this.getPrevAvailableCoachIndex(currFocusIndex);\n\n    // If the currFocusIndex equals the newFocusableIndex then we have reached the first or last tabable navigation coach Element and we have to the set the focus manual to the firstTabElement or lastTabElement.\n    if (currFocusIndex === newFocusableIndex) {\n      this.unfocusPlaceElement();\n      this.selectedCoachIndex = -1;\n      this.currSelectedCoachIndex = -1;\n      this.seatReservationWithoutNavigationHasFocus = false;\n\n      if (tabDirection === 'NEXT_TAB') this.lastTabElement.focus();\n      else this.firstTabElement.focus();\n\n      return;\n    }\n\n    if (this.hasNavigation) {\n      const selectedPlaceElement = this._getPlaceHtmlElement();\n      const placeInCoachHasFocus = selectedPlaceElement\n        ? selectedPlaceElement.getAttribute('keyfocus') === 'focus'\n        : false;\n\n      // If we tab back (PREV_TAB) and the focus is currently on place,\n      // we remove the selected state from the currently selected navigation coach and only set the focus status to it\n      if (tabDirection === 'PREV_TAB' && this.selectedCoachIndex === currFocusIndex) {\n        if (placeInCoachHasFocus) {\n          this.focusedCoachIndex = currFocusIndex;\n          this.unfocusPlaceElement();\n        } else {\n          this.focusedCoachIndex = newFocusableIndex;\n        }\n      }\n      // Only sets the focus on the new navigation coach\n      else if (newFocusableIndex !== this.currSelectedCoachIndex) {\n        this.focusedCoachIndex = newFocusableIndex;\n      } else {\n        this.focusedCoachIndex = -1;\n        this.selectedCoachIndex = newFocusableIndex;\n        // If any place was focused in coach, so we set focused again\n        if (placeInCoachHasFocus) {\n          this.focusPlaceElement(this.currSelectedPlace);\n        }\n        // If no place was selected, then we select the coach grid\n        else {\n          this.isCoachGridFocusable = true;\n          this._setFocusToSelectedCoachGrid();\n        }\n      }\n    }\n    // If no navigation exist, we scroll directly to the next tabable coach\n    else {\n      this.scrollToSelectedNavCoach(newFocusableIndex);\n    }\n  }\n\n  private _navigateToPlaceByKeyboard(pressedKey: string): void {\n    this.preventCoachScrollByPlaceClick = false;\n    this.isKeyboardNavigation = true;\n\n    if (this.focusedCoachIndex !== -1) {\n      this.focusedCoachIndex = -1;\n    }\n\n    if (!this.preventPlaceClick) {\n      const findClosestPlace = this._getClosestPlaceByKeyDirection(pressedKey);\n      if (findClosestPlace) {\n        this.focusPlaceElement(findClosestPlace);\n      }\n      // No closest place found by key navigation\n      else {\n        if (\n          pressedKey === this.keyboardNavigationEvents.ArrowRight ||\n          pressedKey === this.keyboardNavigationEvents.ArrowLeft ||\n          (this.alignVertical &&\n            (pressedKey === this.keyboardNavigationEvents.ArrowUp ||\n              pressedKey === this.keyboardNavigationEvents.ArrowDown))\n        ) {\n          // Check the current pressed key to get the next available coach index\n          const newSelectedCoachIndex =\n            pressedKey === this.keyboardNavigationEvents.ArrowRight\n              ? this.getNextAvailableCoachIndex()\n              : this.getPrevAvailableCoachIndex();\n\n          this.scrollToSelectedNavCoach(newSelectedCoachIndex);\n        }\n      }\n    }\n  }\n\n  protected getNextAvailableCoachIndex(currentIndex?: number): number {\n    const startIndex = currentIndex ?? this.currSelectedCoachIndex;\n    return startIndex < this.seatReservation.coachItems.length - 1 ? startIndex + 1 : startIndex;\n  }\n\n  protected getPrevAvailableCoachIndex(currentIndex?: number): number {\n    const startIndex = currentIndex ?? this.currSelectedCoachIndex;\n    return startIndex > 0 ? startIndex - 1 : startIndex;\n  }\n\n  protected updateSelectedSeatReservationPlaces(placeSelection: PlaceSelection): void {\n    // Add selected place to selectedSeatReservationPlaces\n    if (placeSelection.state === 'SELECTED') {\n      const seatReservationSelection = this._getSeatReservationPlaceSelection(placeSelection);\n      if (seatReservationSelection) {\n        this.selectedSeatReservationPlaces.push(seatReservationSelection);\n      }\n    }\n    // Remove selected place from selectedSeatReservationPlaces\n    else {\n      this.selectedSeatReservationPlaces = this.selectedSeatReservationPlaces.filter(\n        (_selectedPlace) => _selectedPlace.id !== placeSelection.id,\n      );\n    }\n\n    // Checks whether maxReservation is activated and the maximum number of selected places is reached\n    if (this.maxReservations && this.selectedSeatReservationPlaces.length > this.maxReservations) {\n      this._resetAllPlaceSelections(placeSelection);\n    }\n\n    /**\n     * @@type {CustomEvent<SeatReservationSelectedPlacesEventDetails>}\n     * Emits when a place was selected and returns a Place array with all selected places.\n     */\n    this.dispatchEvent(\n      new CustomEvent<SeatReservationSelectedPlacesEventDetails>('selectedplaces', {\n        bubbles: true,\n        composed: true,\n        detail: this.selectedSeatReservationPlaces,\n      }),\n    );\n  }\n\n  protected updateCurrentSelectedPlaceInCoach(placeSelection: PlaceSelection): void {\n    const coachIndex = placeSelection.coachIndex;\n    const place = this.seatReservation.coachItems[coachIndex].places?.find(\n      (place) => place.number == placeSelection.number,\n    );\n\n    if (!place) return;\n    this.currSelectedCoachIndex = coachIndex;\n    this.currSelectedPlace = place;\n    if (this.currSelectedCoachIndex !== this.selectedCoachIndex) {\n      this.updateCurrentSelectedCoach();\n    }\n\n    this._setCurrSelectedPlaceElementId(place);\n  }\n\n  protected updateCurrentSelectedCoach(): void {\n    this.selectedCoachIndex = this.currSelectedCoachIndex;\n    this.focusedCoachIndex = -1;\n    const coachSelection = this._getSeatReservationCoachSelection(this.selectedCoachIndex);\n    if (coachSelection) {\n      /**\n       * @type {CustomEvent<SeatReservationCoachSelection>}\n       * Emits when a coach was selected and returns a CoachSelection\n       */\n      this.dispatchEvent(\n        new CustomEvent<SeatReservationCoachSelection>('selectedcoach', {\n          bubbles: true,\n          composed: true,\n          detail: coachSelection,\n        }),\n      );\n    }\n  }\n\n  /**\n   * Initialization of SeatReservationPlaceSelection Array based on the transferred places\n   * that have the state SELECTED within the seatReservation object\n   */\n  private _initSeatReservationPlaceSelection(): void {\n    this.seatReservation.coachItems.map((coach: CoachItem, coachIndex: number) => {\n      coach.places\n        ?.filter((place) => place.state === 'SELECTED')\n        ?.forEach((place) => {\n          const preselectedPlaceSelection: PlaceSelection = mapPlaceInfosToPlaceSelection(\n            place,\n            coachIndex,\n          );\n          const seatReservationPlaceSelection: SeatReservationPlaceSelection | null =\n            this._getSeatReservationPlaceSelection(preselectedPlaceSelection);\n          if (seatReservationPlaceSelection)\n            this.selectedSeatReservationPlaces.push(seatReservationPlaceSelection);\n        });\n    });\n  }\n\n  /**\n   * All selected places will be reset or the currentSelectedPlace was given, then we reset all except currentSelectedPlace\n   * @param currSelectedPlace\n   */\n  private _resetAllPlaceSelections(currSelectedPlace?: PlaceSelection): void {\n    //Find all places to be needed unselect\n    for (const placeSelection of this.selectedSeatReservationPlaces) {\n      if (!currSelectedPlace || currSelectedPlace.id !== placeSelection.id) {\n        const placeElement = this.shadowRoot?.getElementById(placeSelection.id) as HTMLElement;\n        placeElement.setAttribute('state', 'FREE');\n      }\n    }\n    //Removes all selected places except the currently selected place\n    if (currSelectedPlace) {\n      this.selectedSeatReservationPlaces = this.selectedSeatReservationPlaces.filter(\n        (_selectedPlace) => _selectedPlace.id === currSelectedPlace.id,\n      );\n    } else {\n      this.selectedSeatReservationPlaces = [];\n    }\n  }\n\n  private _getSeatReservationPlaceSelection(\n    currSelectedPlace: PlaceSelection,\n  ): SeatReservationPlaceSelection | null {\n    const coach = this.seatReservation.coachItems[currSelectedPlace.coachIndex];\n    const place = coach.places?.find((place) => place.number === currSelectedPlace.number);\n\n    return place\n      ? mapPlaceAndCoachToSeatReservationPlaceSelection(place, coach, currSelectedPlace.coachIndex)\n      : null;\n  }\n\n  private _getSeatReservationCoachSelection(\n    coachIndex: number,\n  ): SeatReservationCoachSelection | null {\n    if (!this.seatReservation.coachItems[coachIndex]) return null;\n\n    const coach = this.seatReservation.coachItems[coachIndex];\n    return mapCoachInfosToCoachSelection(coachIndex, coach);\n  }\n\n  private _setCurrSelectedPlaceElementId(place: Place | null): void {\n    if (place) {\n      this.currSelectedPlaceElementId =\n        'seat-reservation__place-button-' + this.currSelectedCoachIndex + '-' + place.number;\n    } else {\n      this.currSelectedPlaceElementId = null;\n    }\n  }\n\n  /**\n   * Returns the current selected place HTML element by given placeNumber and coachIndex.\n   * If both doesnt exist, we try to return the place HTML element by the _currentSelectedPlaceElementId\n   * @param placeNumber optional as string\n   * @param coachIndex optional as string\n   * @returns HTMLElement or null\n   */\n  private _getPlaceHtmlElement(placeNumber?: string, coachIndex?: number): HTMLElement | null {\n    const currCoachIndex = coachIndex ? coachIndex : this.currSelectedCoachIndex;\n    const coachPlaceNumberId = placeNumber\n      ? 'seat-reservation__place-button-' + currCoachIndex + '-' + placeNumber\n      : this.currSelectedPlaceElementId;\n    return coachPlaceNumberId ? this.shadowRoot?.getElementById(coachPlaceNumberId) || null : null;\n  }\n\n  //Set the vertical offset\n  private _setVerticalAlignmentOffset(): void {\n    setTimeout(() => {\n      const seatReservationWrapperElement = this.shadowRoot?.querySelector(\n        '.sbb-sr__wrapper',\n      ) as HTMLElement;\n      if (seatReservationWrapperElement) {\n        const seatReservationVerticalOffset =\n          seatReservationWrapperElement.getBoundingClientRect().width;\n        this.style?.setProperty(\n          '--sbb-seat-reservation-vertical-offset',\n          `${seatReservationVerticalOffset}px`,\n        );\n      }\n    });\n  }\n}\n"],"names":["ScrollDirection","place"],"mappings":"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"}
|
|
830
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"seat-reservation-base-element.js","sources":["../../../../../src/elements-experimental/seat-reservation/seat-reservation/seat-reservation-base-element.ts"],"sourcesContent":["import { isArrowKeyOrPageKeysPressed } from '@sbb-esta/lyne-elements/core/a11y.js';\nimport { forceType } from '@sbb-esta/lyne-elements/core/decorators.js';\nimport { LitElement, type PropertyValues } from 'lit';\nimport { property, state } from 'lit/decorators.js';\n\nimport {\n  mapCoachInfosToCoachSelection,\n  mapPlaceAndCoachToSeatReservationPlaceSelection,\n  mapPlaceInfosToPlaceSelection,\n} from '../common/mapper.js';\nimport type {\n  CoachItem,\n  CoachNumberOfFreePlaces,\n  ElementDimension,\n  ElementPosition,\n  Place,\n  PlaceSelection,\n  SeatReservation,\n  SeatReservationCoachSelection,\n  SeatReservationPlaceSelection,\n} from '../common.js';\nimport type { SbbSeatReservationPlaceControlElement } from '../seat-reservation-place-control/seat-reservation-place-control.component.js';\n\nenum ScrollDirection {\n  right = 'right',\n  left = 'left',\n}\n\ninterface CoachScrollTriggerPoint {\n  start: number;\n  end: number;\n  width: number;\n}\n\nexport type SeatReservationSelectedPlacesEventDetails = {\n  seats: SeatReservationPlaceSelection[];\n  bicycles: SeatReservationPlaceSelection[];\n};\n\nexport class SeatReservationBaseElement extends LitElement {\n  public static readonly events = {\n    selectedplaces: 'selectedplaces',\n    selectedcoach: 'selectedcoach',\n  } as const;\n\n  /** The seat reservation object which contains all coaches and places */\n  @property({ attribute: 'seat-reservation', type: Object })\n  public accessor seatReservation: SeatReservation = null!;\n\n  /** The seat reservation navigation can be toggled by this property */\n  @forceType()\n  @property({ attribute: 'has-navigation', type: Boolean })\n  public accessor hasNavigation: boolean = true;\n\n  /** The seat reservation area is aligned vertically */\n  @forceType()\n  @property({ attribute: 'align-vertical', type: Boolean })\n  public accessor alignVertical: boolean = false;\n\n  /** The seat reservation area's base grid size */\n  @forceType()\n  @property({ attribute: 'base-grid-size', type: Number })\n  public accessor baseGridSize: number = 16;\n\n  /** The seat reservation area's width */\n  @forceType()\n  @property({ attribute: 'height', type: Number })\n  public accessor height: number = null!;\n\n  /** Maximal number of possible clickable seats */\n  @forceType()\n  @property({ attribute: 'max-seat-reservations', type: Number })\n  public accessor maxSeatReservations: number = -1;\n\n  /** Maximal number of possible clickable bicycle places */\n  @forceType()\n  @property({ attribute: 'max-bicycle-reservations', type: Number })\n  public accessor maxBicycleReservations: number = -1;\n\n  /** Any click functionality is prevented */\n  @forceType()\n  @property({ attribute: 'prevent-place-click', type: Boolean })\n  public accessor preventPlaceClick: boolean = false;\n\n  @forceType()\n  @property({ attribute: 'preselect-coach-index', type: Number })\n  public accessor preselectCoachIndex: number = -1;\n\n  @state() protected accessor selectedCoachIndex: number = -1;\n  @state() protected accessor focusedCoachIndex: number = -1;\n  //Sets the hover style when scrolling to a coach\n  @state() protected accessor hoveredScrollCoachIndex: number = -1;\n\n  // Describes the distance between the border of the coach and the places in pixels\n  protected coachBorderPadding = 6;\n  // Describes the gap between the coaches in pixels\n  protected gapBetweenCoaches = 4;\n  // The calculated coachBorderOffset is used to calculate the width and height of coach border graphic,\n  // but also to position other graphics that are aligned directly to the coach border.\n  protected coachBorderOffset = this.coachBorderPadding / this.baseGridSize;\n  // Describes the fix width of coach navigation button\n  protected coachNavButtonDim: number = 0;\n  protected currScrollDirection: ScrollDirection = ScrollDirection.right;\n  protected maxCalcCoachsWidth: number = 0;\n  protected scrollCoachsAreaWidth: number = 0;\n  protected scrollNavigationAreaDim: number = 0;\n  protected triggerCoachPositionsCollection: CoachScrollTriggerPoint[] = [];\n  protected firstTabElement: HTMLElement = null!;\n  protected lastTabElement: HTMLElement = null!;\n  protected navigationScrollArea: HTMLElement = null!;\n  protected coachScrollArea: HTMLElement = null!;\n  protected currSelectedPlace: Place | null = null;\n  protected currSelectedPlaceElementId: string | null = null;\n  protected currSelectedCoachIndex: number = -1;\n  protected preventCoachScrollByPlaceClick: boolean = false;\n  protected selectedSeatReservationPlaces: SeatReservationSelectedPlacesEventDetails = {\n    seats: [],\n    bicycles: [],\n  };\n  protected seatReservationWithoutNavigationHasFocus = false;\n  protected isCoachGridFocusable = false;\n  protected isAutoScrolling = false;\n  protected isKeyboardNavigation = false;\n  protected keyboardNavigationEvents = {\n    ArrowLeft: 'ArrowLeft',\n    ArrowRight: 'ArrowRight',\n    ArrowUp: 'ArrowUp',\n    ArrowDown: 'ArrowDown',\n    Tab: 'Tab',\n    Enter: 'Enter',\n  } as const;\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('seatReservation')) {\n      this._initSeatReservationPlaceSelection();\n    }\n\n    if (changedProperties.has('baseGridSize')) {\n      this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;\n      this.style?.setProperty('--sbb-seat-reservation-grid-size', `${this.baseGridSize}px`);\n\n      this.initNavigationSelectionByScrollEvent();\n    }\n\n    // If the height is used, the baseGridSize must be recalculated\n    if (changedProperties.has('height') && !!this.height) {\n      if (this.seatReservation.coachItems.length) {\n        this.baseGridSize = this.height / this.seatReservation.coachItems[0].dimension.h;\n        this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;\n        this.style?.setProperty('--sbb-seat-reservation-grid-size', `${this.baseGridSize}px`);\n\n        this.initNavigationSelectionByScrollEvent();\n      }\n    }\n\n    if (changedProperties.has('alignVertical') && this.alignVertical) {\n      this.initNavigationSelectionByScrollEvent();\n    }\n\n    if (changedProperties.has('preselectCoachIndex') && this.preselectCoachIndex) {\n      // setTimeout is neccessary because without, _getCoachScrollPositionX() would fail with NPE because\n      // the coachScrollArea is not yet initialized\n      setTimeout(() => this.scrollToSelectedNavCoach(this.preselectCoachIndex), 1);\n    }\n  }\n\n  protected navigateByDirectionBtn(btnDirection: string): void {\n    this.unfocusPlaceElement();\n    let navigateToCoachIndex = this.currSelectedCoachIndex;\n    if (btnDirection == 'DIRECTION_LEFT' && navigateToCoachIndex > 0) {\n      navigateToCoachIndex =\n        this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex - 1 : 0;\n    } else if (\n      btnDirection == 'DIRECTION_RIGHT' &&\n      navigateToCoachIndex < this.seatReservation.coachItems.length - 1\n    ) {\n      navigateToCoachIndex =\n        this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex + 1 : 0;\n    }\n\n    this.scrollToSelectedNavCoach(navigateToCoachIndex);\n  }\n\n  /* Init scroll event handling for coach navigation */\n  protected initNavigationSelectionByScrollEvent(): void {\n    if (this.seatReservation.coachItems.length > 0) {\n      const borderHeight =\n        (this.seatReservation.coachItems[0].dimension.h + this.coachBorderOffset * 2) *\n        this.baseGridSize;\n      this.style?.setProperty('--sbb-seat-reservation-coach-height', `${borderHeight}`);\n    }\n\n    this.firstTabElement = this.shadowRoot?.querySelector('#first-tab-element') as HTMLElement;\n    this.lastTabElement = this.shadowRoot?.querySelector('#last-tab-element') as HTMLElement;\n    this.coachScrollArea = this.shadowRoot?.querySelector('#sbb-sr__parent-area') as HTMLElement;\n    this.navigationScrollArea = this.shadowRoot?.querySelector('#sbb-sr-navigation') as HTMLElement;\n\n    if (this.navigationScrollArea) {\n      this.scrollNavigationAreaDim = this.alignVertical\n        ? this.navigationScrollArea.getBoundingClientRect().height\n        : this.navigationScrollArea.getBoundingClientRect().width;\n\n      // Init the coachNavButtonDim dimension, which is needed to calculate the correct scroll navigation later\n      const navCoacheList = this.navigationScrollArea.querySelector('ul > li') as HTMLUListElement;\n      const firstLiEleDimension = navCoacheList?.getBoundingClientRect();\n      this.coachNavButtonDim = this.alignVertical\n        ? firstLiEleDimension?.height\n        : firstLiEleDimension?.width;\n    }\n\n    if (this.coachScrollArea) {\n      let currCalcTriggerPos = 0;\n      this.scrollCoachsAreaWidth = this.alignVertical\n        ? this.coachScrollArea.getBoundingClientRect().height\n        : this.coachScrollArea.getBoundingClientRect().width;\n\n      // Precalculate trigger scroll position array depends on coach width\n      this.triggerCoachPositionsCollection = this.seatReservation.coachItems.map((coach) => {\n        const startPosX = currCalcTriggerPos;\n        const coachWidth = this.getCalculatedDimension(coach.dimension).w;\n\n        // Calculation of the end scroll trigger position of a coach, including the gap between the coaches\n        currCalcTriggerPos += coachWidth + this.gapBetweenCoaches;\n\n        return {\n          start: startPosX,\n          end: currCalcTriggerPos,\n          width: coachWidth,\n        } as CoachScrollTriggerPoint;\n      });\n\n      // Set maximum calculated coach width\n      this.maxCalcCoachsWidth = currCalcTriggerPos;\n\n      // At the end of a scroll Events to a coach, the reached wagon is marked as selected\n      this.coachScrollArea.addEventListener('scrollend', () => {\n        const findScrollCoachIndex = this.isAutoScrolling\n          ? this.currSelectedCoachIndex\n          : this._getCoachIndexByScrollTriggerPosition();\n\n        // In case the user uses the scrollbar without interacting with the seat reservation,\n        // the currently selected index is -1, and we have to set this value with findScrollCoachIndex.\n        if (this.currSelectedCoachIndex === -1) {\n          this.currSelectedCoachIndex = findScrollCoachIndex;\n        }\n\n        if (this._isScrollableToSelectedCoach()) {\n          this.currSelectedCoachIndex = findScrollCoachIndex;\n        } else {\n          this.currSelectedCoachIndex =\n            findScrollCoachIndex < this.currSelectedCoachIndex\n              ? this.currSelectedCoachIndex\n              : findScrollCoachIndex;\n        }\n\n        if (!this.isAutoScrolling) {\n          //When user is scrolling via scrollbar, it automatically scrolls to the focused coach in the main navigation\n          this._scrollToSelectedNavigationButton(findScrollCoachIndex);\n        }\n\n        this.preventCoachScrollByPlaceClick = false;\n        this.updateCurrentSelectedCoach();\n\n        if (!this.hasNavigation) {\n          this.preselectPlaceInCoach();\n          this.isAutoScrolling = false;\n        }\n      });\n    }\n  }\n\n  /**\n   * If no navigation exists (property setting -> hasNavigation) and a table coach gets the focus,\n   * the first place in the coach must be automatically preselected to control the place navigation via keyboard\n   *\n   * @param focusCoachIndex\n   */\n  protected onFocusTableCoachAndPreselectPlace(focusCoachIndex: number): void {\n    if (!this.seatReservationWithoutNavigationHasFocus && !this.hasNavigation) {\n      this.seatReservationWithoutNavigationHasFocus = true;\n      this.currSelectedCoachIndex =\n        focusCoachIndex === 0\n          ? this.getNextAvailableCoachIndex(-1)\n          : this.getPrevAvailableCoachIndex(focusCoachIndex);\n      this.preselectPlaceInCoach();\n    }\n  }\n\n  /**\n   * Initialisation of Keyboard event handling to navigation between each places inside a selected coach by using [arrow] keys.\n   * With the [TAB] key the user navigation goes to the next coach navigation element and the currently selected place is automatically reset.\n   */\n  protected handleKeyboardEvent(event: KeyboardEvent): void {\n    const pressedKey = event.key;\n\n    // If any place is selected and TAB Key combination ist pressed,\n    // then we handle the next or previous coach selection\n    if (this.currSelectedPlace) {\n      if (event.shiftKey && event.keyCode === 9) {\n        this._navigateCoachNavigationByKeyboard('PREV_TAB');\n        event.preventDefault();\n        return;\n      }\n\n      if (pressedKey === this.keyboardNavigationEvents.Tab) {\n        this._navigateCoachNavigationByKeyboard('NEXT_TAB');\n        event.preventDefault();\n        return;\n      }\n    }\n\n    // Check if a coach is selected and the arrow key is pressed\n    if (this.currSelectedCoachIndex !== -1 && isArrowKeyOrPageKeysPressed(event)) {\n      event.preventDefault();\n\n      switch (pressedKey) {\n        case this.keyboardNavigationEvents.ArrowLeft:\n          {\n            const pressedLeftKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowDown\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedLeftKeyMapping);\n          }\n          break;\n        case this.keyboardNavigationEvents.ArrowRight:\n          {\n            const pressedRightKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowUp\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedRightKeyMapping);\n          }\n          break;\n        case this.keyboardNavigationEvents.ArrowUp:\n          {\n            const pressedUpKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowLeft\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedUpKeyMapping);\n          }\n          break;\n        case this.keyboardNavigationEvents.ArrowDown:\n          {\n            const pressedDownKeyMapping: string = this.alignVertical\n              ? this.keyboardNavigationEvents.ArrowRight\n              : pressedKey;\n            this._navigateToPlaceByKeyboard(pressedDownKeyMapping);\n          }\n          break;\n        default:\n          break;\n      }\n    }\n  }\n\n  /**\n   * Selects a place inside the coach if navigated via keyboard,\n   * otherwise the coach grid is selected (necessary for ScreenReader)\n   */\n  protected preselectPlaceInCoach(): void {\n    const closestPlace = this._getClosestPlaceByKeyDirection();\n    //If closestPlace exist, we have to unfocus previouse focused place\n    if (closestPlace) {\n      this.unfocusPlaceElement();\n    }\n\n    // Only when keyboard navigation is used and coaches are scrolled by auto scrolling,\n    // then we can set the focus on the first place in the coach.\n    if (this.isKeyboardNavigation && this.isAutoScrolling) {\n      if (closestPlace) {\n        this.focusPlaceElement(closestPlace);\n      }\n    }\n    // In cases where the preselection function is triggered by normal clicking or via screenreader via tab,\n    // we only focus the table without directly focusing the place.\n    else {\n      // We need to set the currSelectedPlace here for further correct functioning navigation via tab.\n      this.currSelectedPlace = closestPlace;\n      this._setFocusToSelectedCoachGrid();\n    }\n  }\n\n  protected scrollToSelectedNavCoach(selectedNavCoachIndex: number): void {\n    if (selectedNavCoachIndex !== this.currSelectedCoachIndex) {\n      this.isAutoScrolling = true;\n      this.isCoachGridFocusable = true;\n      this.currSelectedCoachIndex = selectedNavCoachIndex;\n      this._setScrollDirectionByCoachIndex();\n\n      const scrollToCoachPosX = this._getCoachScrollPositionX();\n      const isSelectedCoachIndexScrollable =\n        this.selectedCoachIndex !== -1 || this.currSelectedCoachIndex > 0;\n\n      // Checks whether the current scroll position allows scrolling to the next wagon or not\n      if (isSelectedCoachIndexScrollable && this._isScrollableToSelectedCoach()) {\n        this.coachScrollArea.scrollTo({\n          top: this.alignVertical ? scrollToCoachPosX : 0,\n          left: this.alignVertical ? 0 : scrollToCoachPosX,\n          behavior: 'smooth',\n        });\n      } else {\n        this.updateCurrentSelectedCoach();\n      }\n\n      // Automatic scrolling to the selected coach in the main navigation\n      this._scrollToSelectedNavigationButton(selectedNavCoachIndex);\n    }\n  }\n\n  protected focusPlaceElement(place: Place | null, coachIndex?: number): void {\n    this.unfocusPlaceElement();\n    if (place) {\n      this.currSelectedPlace = place;\n      if (coachIndex) {\n        this.currSelectedCoachIndex = coachIndex;\n      }\n\n      this._setCurrSelectedPlaceElementId(place);\n\n      const selectedPlaceElement = this._getPlaceHtmlElement();\n      if (selectedPlaceElement) {\n        selectedPlaceElement.setAttribute('keyfocus', 'focus');\n      }\n    }\n  }\n\n  protected unfocusPlaceElement(): void {\n    const selectedPlaceElement = this._getPlaceHtmlElement();\n    if (selectedPlaceElement) {\n      selectedPlaceElement.setAttribute('keyfocus', 'unfocus');\n      this._setCurrSelectedPlaceElementId(null);\n      this.currSelectedPlace = null;\n    }\n  }\n\n  protected getCalculatedDimension(\n    elementDimension: ElementDimension,\n    coachDimension?: ElementDimension,\n    isOriginHeight?: boolean,\n    isStretchHeight?: boolean,\n  ): ElementDimension {\n    if (coachDimension && !isOriginHeight) {\n      // Since the height of the coach has been visually expanded to maintain a distance between the border of the coach and the places,\n      // some graphics must also be adapted to the height, which end with the borders of the choach\n      elementDimension.h += this.coachBorderOffset * 2;\n    }\n\n    if (isStretchHeight) {\n      // In the case of graphics are assigned directly at the border of the coach,\n      // these graphics must be expanded in height by a coachBorderOffset in order to reach their original visual position\n      elementDimension.h += this.coachBorderOffset;\n    }\n\n    return {\n      w: this.baseGridSize * elementDimension.w,\n      h: this.baseGridSize * elementDimension.h,\n    };\n  }\n\n  protected getCalculatedPosition(\n    elementPosition: ElementPosition,\n    elementDimension?: ElementDimension,\n    coachDimension?: ElementDimension,\n    isOriginHeight?: boolean,\n  ): ElementPosition {\n    if (coachDimension && elementDimension) {\n      const endPosHeight = isOriginHeight\n        ? coachDimension.h\n        : coachDimension.h + this.coachBorderOffset;\n      // If the original element is positioned at the top or bottom of the coach, we need to recalculate the Y coordinate with the additional border padding\n      if (elementPosition.y === 0) {\n        elementPosition.y -= this.coachBorderOffset;\n      } else if (elementPosition.y + elementDimension.h === endPosHeight) {\n        elementPosition.y += this.coachBorderOffset;\n      }\n    }\n\n    return {\n      x: this.baseGridSize * elementPosition.x,\n      y: this.baseGridSize * elementPosition.y,\n      z: elementPosition.z,\n    };\n  }\n\n  /**\n   * Counts all available seats together depending on the seat type\n   *\n   * @param coachIndex\n   * @returns An Object with count of free seats and free bicycle places\n   */\n  protected getAvailableFreePlacesNumFromCoach(coachIndex: number): CoachNumberOfFreePlaces {\n    const accumulator: CoachNumberOfFreePlaces = { seats: 0, bicycles: 0 };\n    const freePlaces = this.seatReservation.coachItems[coachIndex].places?.reduce(\n      (accumulator, currPlace: Place) => {\n        if (currPlace.state !== 'FREE') {\n          return accumulator;\n        }\n        // Count up depending on seat type\n        if (currPlace.type === 'SEAT') {\n          accumulator.seats++;\n        } else {\n          accumulator.bicycles++;\n        }\n        return accumulator;\n      },\n      accumulator,\n    );\n    return freePlaces ? freePlaces : accumulator;\n  }\n\n  /**\n   * Performs an automatic main navigation scroll to the specified selectedNavCoachIndex.\n   * Calculates the central scroll offset of the nav coach to be selected.\n   * @param selectedNavCoachIndex\n   */\n  private _scrollToSelectedNavigationButton(selectedNavCoachIndex: number): void {\n    //Time delay to not interfere with other executing calling scrollTo functions (coaches scrolling)\n    setTimeout(() => {\n      // Sets the hover index style for the target scroll coach\n      this.hoveredScrollCoachIndex = selectedNavCoachIndex;\n\n      if (this.hasNavigation && this.navigationScrollArea) {\n        const navigationAreaCenteredPosX = this.scrollNavigationAreaDim / 2;\n        const scrollButtonOffsetX = selectedNavCoachIndex * this.coachNavButtonDim;\n        const scrollOffsetX =\n          scrollButtonOffsetX - navigationAreaCenteredPosX + this.coachNavButtonDim;\n\n        this.navigationScrollArea.scrollTo({\n          top: this.alignVertical ? scrollOffsetX : 0,\n          left: this.alignVertical ? 0 : scrollOffsetX,\n          behavior: 'smooth',\n        });\n      }\n    }, 10);\n  }\n\n  /**\n   * Sets the new ScrollDirection by the new given target coach index.\n   */\n  private _setScrollDirectionByCoachIndex(): void {\n    this.currScrollDirection =\n      this.currSelectedCoachIndex > this.selectedCoachIndex\n        ? ScrollDirection.right\n        : ScrollDirection.left;\n  }\n\n  /**\n   * Returns the scroll start or end position X from the selected coach.\n   * In case the user is currently navigating through places by keyboard and goes to previous coach,\n   * then we return the end position of the coach to get the closest scroll position of the next focus place.\n   * @returns number\n   */\n  private _getCoachScrollPositionX(): number {\n    const coachTriggerPoint = this.triggerCoachPositionsCollection[this.currSelectedCoachIndex];\n\n    const isFocusPlaceFromPreviousCoachPosition =\n      this.isKeyboardNavigation &&\n      this.currScrollDirection === ScrollDirection.left &&\n      coachTriggerPoint.width > this.scrollCoachsAreaWidth;\n\n    return isFocusPlaceFromPreviousCoachPosition\n      ? coachTriggerPoint.end - this.scrollCoachsAreaWidth\n      : coachTriggerPoint.start;\n  }\n\n  /**\n   * Sets the focus on the HTML table (grid) caption element so that the heading is read out when using a ScreenReader.\n   */\n  private _setFocusToSelectedCoachGrid(): void {\n    // When the user performs an action that affects the coach navigation, then the navigated table is focusable.\n    if (this.isCoachGridFocusable) {\n      this.isCoachGridFocusable = false;\n      const coachTableCaptionElement = this.shadowRoot?.querySelector(\n        '#sbb-sr-coach-caption-' + this.currSelectedCoachIndex,\n      ) as HTMLTableCaptionElement;\n      if (coachTableCaptionElement) {\n        coachTableCaptionElement.focus();\n      }\n    }\n  }\n\n  /**\n   * Returns whether the current scrolled position can be used to scroll to the selected wagon\n   * @returns boolean\n   */\n  private _isScrollableToSelectedCoach(): boolean {\n    const currScrollPosX = this.alignVertical\n      ? this.coachScrollArea.scrollTop\n      : this.coachScrollArea.scrollLeft;\n    const coachScrollWindowWidth = this.alignVertical\n      ? this.coachScrollArea.getBoundingClientRect().height\n      : this.coachScrollArea.getBoundingClientRect().width;\n    const maxScrollWidthArea = this.maxCalcCoachsWidth - coachScrollWindowWidth;\n    const currCoachTrigger = this.triggerCoachPositionsCollection[this.currSelectedCoachIndex];\n    const isScrollPosSameToCurrCoachPos =\n      currScrollPosX === this.triggerCoachPositionsCollection[this.currSelectedCoachIndex].start;\n\n    return (\n      (currScrollPosX < maxScrollWidthArea || currScrollPosX > currCoachTrigger.start) &&\n      !isScrollPosSameToCurrCoachPos\n    );\n  }\n\n  /**\n   * Returns the coach index which is currently visible in the scroll area\n   * @returns number\n   */\n  private _getCoachIndexByScrollTriggerPosition(): number {\n    const scrollPos = this.alignVertical\n      ? this.coachScrollArea.scrollTop\n      : this.coachScrollArea.scrollLeft;\n    const scrollOffsetX = scrollPos + this.scrollCoachsAreaWidth / 2;\n    return this.triggerCoachPositionsCollection.findIndex(\n      (coachTrigger) => scrollOffsetX >= coachTrigger.start && scrollOffsetX <= coachTrigger.end,\n    );\n  }\n\n  /**\n   * Get the first place of current selected coach by table cell coordinate 0-0 id.\n   * @returns Place or null\n   */\n  private _getFirstPlaceInSelecedCoach(): Place | null {\n    let firstPlace: Place | null = null;\n    const coach = this.seatReservation?.coachItems[this.currSelectedCoachIndex];\n    const firstCellId = 'cell-' + this.currSelectedCoachIndex + '-0-0';\n    const placeNumber =\n      this.shadowRoot\n        ?.querySelector<HTMLTableCellElement>('#' + firstCellId)\n        ?.querySelector<SbbSeatReservationPlaceControlElement>('sbb-seat-reservation-place-control')\n        ?.getAttribute('text') || null;\n\n    if (coach && placeNumber) {\n      firstPlace = coach.places?.find((place) => place.number === placeNumber) || null;\n    }\n    return firstPlace;\n  }\n\n  /**\n   * To get the correct closest place of current pressed key and the current selected place,\n   * we have to investigate the coordinates of each place to find the closest place of the currSelectedPlaceElementId.\n   * @param pressedKey\n   * @returns Place or null\n   */\n  private _getClosestPlaceByKeyDirection(pressedKey?: string): Place | null {\n    const coach = this.seatReservation?.coachItems[this.currSelectedCoachIndex];\n    let closestPlace = null;\n    if (coach.places) {\n      // If no place set, then we use initial the left-top place on the coach\n      if (!this.currSelectedPlaceElementId) {\n        return this._getFirstPlaceInSelecedCoach();\n      } else {\n        if (this.currSelectedPlace) {\n          for (const place of coach.places) {\n            // If key pressed, then we try to find the place of the current currScrollDirection\n            if (!pressedKey) {\n              //Find place from the left side of coach by y coordinate. Current currScrollDirection is RIGHT)\n              if (\n                this.currScrollDirection === ScrollDirection.right &&\n                place.position.y === this.currSelectedPlace?.position.y &&\n                (!closestPlace || place.position.x < closestPlace.position.x)\n              ) {\n                closestPlace = place;\n              }\n              //Find place from the right side of coach by y coordinate. Current currScrollDirection is LEFT\n              else if (\n                this.currScrollDirection === ScrollDirection.left &&\n                place.position.y === this.currSelectedPlace?.position.y &&\n                (!closestPlace || place.position.x > closestPlace.position.x)\n              ) {\n                closestPlace = place;\n              }\n            } else {\n              if (place.number !== this.currSelectedPlace?.number) {\n                //Key [Right] navigation, we check the place coordinates of the x-axis to get the smallest larger x place coordinate of the currently selected place\n                if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowRight &&\n                  (place.position.y === this.currSelectedPlace.position.y ||\n                    place.position.y === this.currSelectedPlace.position.y - 1) &&\n                  place.position.x > this.currSelectedPlace.position.x &&\n                  (!closestPlace || place.position.x < closestPlace.position.x)\n                ) {\n                  closestPlace = place;\n                }\n                //Key [Down] navigation, we check the place coordinates of the y-axis to get the smallest larger y place coordinate of the currently selected place\n                else if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowDown &&\n                  (place.position.x === this.currSelectedPlace.position.x ||\n                    place.position.x === this.currSelectedPlace.position.x + 1) &&\n                  place.position.y > this.currSelectedPlace.position.y &&\n                  (!closestPlace || place.position.y < closestPlace.position.y)\n                ) {\n                  closestPlace = place;\n                }\n                //Key [Left] navigation, we check the place coordinates of the x-axis to get the greatest smaller x place coordinate of the currently selected place\n                else if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowLeft &&\n                  (place.position.y === this.currSelectedPlace.position.y ||\n                    place.position.y === this.currSelectedPlace.position.y + 1) &&\n                  place.position.x < this.currSelectedPlace.position.x &&\n                  (!closestPlace || place.position.x > closestPlace.position.x)\n                ) {\n                  closestPlace = place;\n                }\n                //Key [Up] navigation, we check the place coordinates of the y-axis to get the greatest smaller y place coordinate of the currently selected place\n                else if (\n                  pressedKey === this.keyboardNavigationEvents.ArrowUp &&\n                  (place.position.x === this.currSelectedPlace.position.x ||\n                    place.position.x === this.currSelectedPlace.position.x - 1) &&\n                  place.position.y < this.currSelectedPlace?.position.y &&\n                  (!closestPlace || place.position.y > closestPlace.position.y)\n                ) {\n                  closestPlace = place;\n                }\n              }\n            }\n          }\n        }\n      }\n    }\n    return closestPlace;\n  }\n\n  // Handling for Tab navigation if a place is selected inside the coach.\n  // This controls the focused coach from the current selected coach.\n  private _navigateCoachNavigationByKeyboard(tabDirection: string): void {\n    const currFocusIndex =\n      this.focusedCoachIndex === -1\n        ? this.currSelectedCoachIndex === -1\n          ? 0\n          : this.currSelectedCoachIndex\n        : this.focusedCoachIndex;\n    // Check next or prev tab is pressed, then we need to find the next available coach index that should receive the focus\n    const newFocusableIndex: number =\n      tabDirection === 'NEXT_TAB'\n        ? this.getNextAvailableCoachIndex(currFocusIndex)\n        : this.getPrevAvailableCoachIndex(currFocusIndex);\n\n    // If the currFocusIndex equals the newFocusableIndex then we have reached the first or last tabable navigation coach Element and we have to the set the focus manual to the firstTabElement or lastTabElement.\n    if (currFocusIndex === newFocusableIndex) {\n      this.unfocusPlaceElement();\n      this.seatReservationWithoutNavigationHasFocus = false;\n\n      if (tabDirection === 'NEXT_TAB') this.lastTabElement.focus();\n      else this.firstTabElement.focus();\n\n      return;\n    }\n\n    if (this.hasNavigation) {\n      const selectedPlaceElement = this._getPlaceHtmlElement();\n      const placeInCoachHasFocus = selectedPlaceElement\n        ? selectedPlaceElement.getAttribute('keyfocus') === 'focus'\n        : false;\n\n      // If we tab back (PREV_TAB) and the focus is currently on place,\n      // we remove the selected state from the currently selected navigation coach and only set the focus status to it\n      if (tabDirection === 'PREV_TAB' && this.selectedCoachIndex === currFocusIndex) {\n        if (placeInCoachHasFocus) {\n          this.focusedCoachIndex = currFocusIndex;\n          this.unfocusPlaceElement();\n\n          return;\n        } else {\n          this.focusedCoachIndex = newFocusableIndex;\n        }\n      }\n      // Only sets the focus on the new navigation coach\n      else if (newFocusableIndex !== this.currSelectedCoachIndex) {\n        this.focusedCoachIndex = newFocusableIndex;\n      } else {\n        this.focusedCoachIndex = -1;\n        this.selectedCoachIndex = newFocusableIndex;\n        // If any place was focused in coach, so we set focused again\n        if (placeInCoachHasFocus) {\n          this.focusPlaceElement(this.currSelectedPlace);\n        }\n        // If no place was selected, then we select the coach grid\n        else {\n          this.isCoachGridFocusable = true;\n          this._setFocusToSelectedCoachGrid();\n        }\n      }\n\n      // By Tab Navigation, perform automatic scrolling to the focused wagon\n      this._scrollToSelectedNavigationButton(newFocusableIndex);\n    }\n    // If no navigation exist, we scroll directly to the next tabable coach\n    else {\n      this.scrollToSelectedNavCoach(newFocusableIndex);\n    }\n  }\n\n  private _navigateToPlaceByKeyboard(pressedKey: string): void {\n    this.preventCoachScrollByPlaceClick = false;\n    this.isKeyboardNavigation = true;\n\n    if (this.focusedCoachIndex !== -1) {\n      this.focusedCoachIndex = -1;\n    }\n\n    if (!this.preventPlaceClick) {\n      const findClosestPlace = this._getClosestPlaceByKeyDirection(pressedKey);\n      if (findClosestPlace) {\n        this.focusPlaceElement(findClosestPlace);\n      }\n      // No closest place found by key navigation\n      else {\n        if (\n          pressedKey === this.keyboardNavigationEvents.ArrowRight ||\n          pressedKey === this.keyboardNavigationEvents.ArrowLeft ||\n          (this.alignVertical &&\n            (pressedKey === this.keyboardNavigationEvents.ArrowUp ||\n              pressedKey === this.keyboardNavigationEvents.ArrowDown))\n        ) {\n          // Check the current pressed key to get the next available coach index\n          const newSelectedCoachIndex =\n            pressedKey === this.keyboardNavigationEvents.ArrowRight\n              ? this.getNextAvailableCoachIndex()\n              : this.getPrevAvailableCoachIndex();\n\n          this.scrollToSelectedNavCoach(newSelectedCoachIndex);\n        }\n      }\n    }\n  }\n\n  protected getNextAvailableCoachIndex(currentIndex?: number): number {\n    const startIndex = currentIndex ?? this.currSelectedCoachIndex;\n    return startIndex < this.seatReservation.coachItems.length - 1 ? startIndex + 1 : startIndex;\n  }\n\n  protected getPrevAvailableCoachIndex(currentIndex?: number): number {\n    const startIndex = currentIndex ?? this.currSelectedCoachIndex;\n    return startIndex > 0 ? startIndex - 1 : startIndex;\n  }\n\n  protected updateSelectedSeatReservationPlaces(placeSelection: PlaceSelection): void {\n    if (placeSelection.placeType === 'SEAT') {\n      this.selectedSeatReservationPlaces.seats = this._updateSelectedSeatReservationPlaces(\n        this.selectedSeatReservationPlaces.seats,\n        this.maxSeatReservations,\n        placeSelection,\n      );\n    } else {\n      this.selectedSeatReservationPlaces.bicycles = this._updateSelectedSeatReservationPlaces(\n        this.selectedSeatReservationPlaces.bicycles,\n        this.maxBicycleReservations,\n        placeSelection,\n      );\n    }\n\n    /**\n     * @@type {CustomEvent<SeatReservationSelectedPlacesEventDetails>}\n     * Emits when a place was selected and returns a Place array with all selected places.\n     */\n    this.dispatchEvent(\n      new CustomEvent<SeatReservationSelectedPlacesEventDetails>('selectedplaces', {\n        bubbles: true,\n        composed: true,\n        detail: this.selectedSeatReservationPlaces,\n      }),\n    );\n  }\n\n  private _updateSelectedSeatReservationPlaces(\n    selectedSeatReservationPlaces: SeatReservationPlaceSelection[],\n    maxReservations: number,\n    placeSelection: PlaceSelection,\n  ): SeatReservationPlaceSelection[] {\n    // Add selected place to selectedSeatReservationPlaces\n    if (placeSelection.state === 'SELECTED') {\n      const seatReservationSelection = this._getSeatReservationPlaceSelection(placeSelection);\n      if (seatReservationSelection) {\n        selectedSeatReservationPlaces.push(seatReservationSelection);\n      }\n    }\n    // Remove selected place from selectedSeatReservationPlaces\n    else {\n      selectedSeatReservationPlaces = selectedSeatReservationPlaces.filter(\n        (_selectedPlace) => _selectedPlace.id !== placeSelection.id,\n      );\n    }\n\n    // Checks whether maxReservation is activated and the maximum number of selected places is reached\n    if (maxReservations > -1 && selectedSeatReservationPlaces.length > maxReservations) {\n      if (maxReservations === 0) {\n        // if maxReservation is 0(not allowed to select any place of given type), resets currently selected place\n        selectedSeatReservationPlaces = this._resetAllPlaceSelections(\n          selectedSeatReservationPlaces,\n        );\n      } else {\n        // otherwise resets all places except given one\n        selectedSeatReservationPlaces = this._resetAllPlaceSelections(\n          selectedSeatReservationPlaces,\n          placeSelection,\n        );\n      }\n    }\n\n    return selectedSeatReservationPlaces;\n  }\n\n  protected updateCurrentSelectedPlaceInCoach(placeSelection: PlaceSelection): void {\n    const coachIndex = placeSelection.coachIndex;\n    const place = this.seatReservation.coachItems[coachIndex].places?.find(\n      (place) => place.number == placeSelection.number,\n    );\n\n    if (!place) return;\n    this.currSelectedCoachIndex = coachIndex;\n    this.currSelectedPlace = place;\n    if (this.currSelectedCoachIndex !== this.selectedCoachIndex) {\n      this.updateCurrentSelectedCoach();\n    }\n\n    this._setCurrSelectedPlaceElementId(place);\n  }\n\n  protected updateCurrentSelectedCoach(): void {\n    this.selectedCoachIndex = this.currSelectedCoachIndex;\n    this.focusedCoachIndex = -1;\n    const coachSelection = this._getSeatReservationCoachSelection(this.selectedCoachIndex);\n    if (coachSelection) {\n      /**\n       * @type {CustomEvent<SeatReservationCoachSelection>}\n       * Emits when a coach was selected and returns a CoachSelection\n       */\n      this.dispatchEvent(\n        new CustomEvent<SeatReservationCoachSelection>('selectedcoach', {\n          bubbles: true,\n          composed: true,\n          detail: coachSelection,\n        }),\n      );\n    }\n  }\n\n  /**\n   * Initialization of SeatReservationPlaceSelection Array based on the transferred places\n   * that have the state SELECTED within the seatReservation object\n   */\n  private _initSeatReservationPlaceSelection(): void {\n    this.seatReservation.coachItems.map((coach: CoachItem, coachIndex: number) => {\n      coach.places\n        ?.filter((place) => place.state === 'SELECTED')\n        ?.forEach((place) => {\n          const preselectedPlaceSelection: PlaceSelection = mapPlaceInfosToPlaceSelection(\n            place,\n            coachIndex,\n          );\n          const seatReservationPlaceSelection: SeatReservationPlaceSelection | null =\n            this._getSeatReservationPlaceSelection(preselectedPlaceSelection);\n          if (seatReservationPlaceSelection) {\n            if (seatReservationPlaceSelection.placeType === 'SEAT') {\n              this.selectedSeatReservationPlaces.seats.push(seatReservationPlaceSelection);\n            } else {\n              this.selectedSeatReservationPlaces.bicycles.push(seatReservationPlaceSelection);\n            }\n          }\n        });\n    });\n  }\n  /**\n   * All selected places will be reset or the currentSelectedPlace was given, then we reset all except currentSelectedPlace\n   * @param reservationPlaceSelections\n   * @param currSelectedPlace\n   */\n  private _resetAllPlaceSelections(\n    reservationPlaceSelections: SeatReservationPlaceSelection[],\n    currSelectedPlace?: PlaceSelection,\n  ): SeatReservationPlaceSelection[] {\n    //Find all places to be needed unselect\n    for (const placeSelection of reservationPlaceSelections) {\n      if (!currSelectedPlace || currSelectedPlace.id !== placeSelection.id) {\n        const placeElement = this.shadowRoot?.getElementById(placeSelection.id) as HTMLElement;\n        placeElement.setAttribute('state', 'FREE');\n      }\n    }\n    //Removes all selected places except the currently selected place\n    if (currSelectedPlace) {\n      reservationPlaceSelections = reservationPlaceSelections.filter(\n        (_selectedPlace) => _selectedPlace.id === currSelectedPlace.id,\n      );\n    } else {\n      reservationPlaceSelections = [];\n    }\n    return reservationPlaceSelections;\n  }\n\n  private _getSeatReservationPlaceSelection(\n    currSelectedPlace: PlaceSelection,\n  ): SeatReservationPlaceSelection | null {\n    const coach = this.seatReservation.coachItems[currSelectedPlace.coachIndex];\n    const place = coach.places?.find((place) => place.number === currSelectedPlace.number);\n\n    return place\n      ? mapPlaceAndCoachToSeatReservationPlaceSelection(place, coach, currSelectedPlace.coachIndex)\n      : null;\n  }\n\n  private _getSeatReservationCoachSelection(\n    coachIndex: number,\n  ): SeatReservationCoachSelection | null {\n    if (!this.seatReservation.coachItems[coachIndex]) return null;\n\n    const coach = this.seatReservation.coachItems[coachIndex];\n    const coachNumberOfFreePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);\n    return mapCoachInfosToCoachSelection(coachIndex, coach, coachNumberOfFreePlaces);\n  }\n\n  private _setCurrSelectedPlaceElementId(place: Place | null): void {\n    if (place) {\n      this.currSelectedPlaceElementId =\n        'seat-reservation__place-button-' + this.currSelectedCoachIndex + '-' + place.number;\n    } else {\n      this.currSelectedPlaceElementId = null;\n    }\n  }\n\n  /**\n   * Returns the current selected place HTML element by given placeNumber and coachIndex.\n   * If both doesn't exist, we try to return the place HTML element by the _currentSelectedPlaceElementId\n   * @param placeNumber optional as string\n   * @param coachIndex optional as string\n   * @returns HTMLElement or null\n   */\n  private _getPlaceHtmlElement(placeNumber?: string, coachIndex?: number): HTMLElement | null {\n    const currCoachIndex = coachIndex ? coachIndex : this.currSelectedCoachIndex;\n    const coachPlaceNumberId = placeNumber\n      ? 'seat-reservation__place-button-' + currCoachIndex + '-' + placeNumber\n      : this.currSelectedPlaceElementId;\n    return coachPlaceNumberId ? this.shadowRoot?.getElementById(coachPlaceNumberId) || null : null;\n  }\n}\n"],"names":["ScrollDirection","accumulator","place"],"mappings":";;;;;;;;;;;;;AAuBA,IAAK;AAAA,CAAL,SAAKA,kBAAe;AAClBA,mBAAA,OAAA,IAAA;AACAA,mBAAA,MAAA,IAAA;AACF,GAHK,oBAAA,kBAAe,CAAA,EAAA;IAgBP,8BAA0B,MAAA;;oBAAS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAnC,SAAA,mBAAmC,YAAU;AAAA;;AAQxD;AAKA;AAKA;AAKA;AAKA;AAKA;AAKA;AAKA;AAIA;AAES;AACA;AAEA;AA5CO,yBAAA,mCAAA,kBAAA,MAAA,+BAAmC,IAAK;AAKxC,yBAAA,kCAAA,kBAAA,MAAA,kCAAA,GAAA,kBAAA,MAAA,6BAAyB,IAAI;AAK7B,yBAAA,kCAAA,kBAAA,MAAA,gCAAA,GAAA,kBAAA,MAAA,6BAAyB,KAAK;AAK9B,yBAAA,iCAAA,kBAAA,MAAA,gCAAA,GAAA,kBAAA,MAAA,4BAAuB,EAAE;AAKzB,yBAAA,2BAAA,kBAAA,MAAA,+BAAA,GAAA,kBAAA,MAAA,sBAAiB,IAAK;AAKtB,yBAAA,wCAAA,kBAAA,MAAA,yBAAA,GAAA,kBAAA,MAAA,mCAA8B,EAAE;AAKhC,yBAAA,2CAAA,kBAAA,MAAA,sCAAA,GAAA,kBAAA,MAAA,sCAAiC,EAAE;AAKnC,yBAAA,sCAAA,kBAAA,MAAA,yCAAA,GAAA,kBAAA,MAAA,iCAA6B,KAAK;AAIlC,yBAAA,wCAAA,kBAAA,MAAA,oCAAA,GAAA,kBAAA,MAAA,mCAA8B,EAAE;AAEpB,yBAAA,uCAAA,kBAAA,MAAA,sCAAA,GAAA,kBAAA,MAAA,kCAA6B,EAAE;AAC/B,yBAAA,sCAAA,kBAAA,MAAA,qCAAA,GAAA,kBAAA,MAAA,iCAA4B,EAAE;AAE9B,yBAAA,4CAAA,kBAAA,MAAA,oCAAA,GAAA,kBAAA,MAAA,uCAAkC,EAAE;AAGtD,WAAA,sBAAkB,kBAAA,MAAA,0CAAA,GAAG;AAErB,WAAA,oBAAoB;AAGpB,WAAA,oBAAoB,KAAK,qBAAqB,KAAK;AAEnD,WAAA,oBAA4B;AAC5B,WAAA,sBAAuC,gBAAgB;AACvD,WAAA,qBAA6B;AAC7B,WAAA,wBAAgC;AAChC,WAAA,0BAAkC;AAClC,WAAA,kCAA6D,CAAA;AAC7D,WAAA,kBAA+B;AAC/B,WAAA,iBAA8B;AAC9B,WAAA,uBAAoC;AACpC,WAAA,kBAA+B;AAC/B,WAAA,oBAAkC;AAClC,WAAA,6BAA4C;AAC5C,WAAA,yBAAiC;AACjC,WAAA,iCAA0C;AAC1C,WAAA,gCAA2E;AAAA,QACnF,OAAO,CAAA;AAAA,QACP,UAAU,CAAA;AAAA,MAAA;AAEF,WAAA,2CAA2C;AAC3C,WAAA,uBAAuB;AACvB,WAAA,kBAAkB;AAClB,WAAA,uBAAuB;AACvB,WAAA,2BAA2B;AAAA,QACnC,WAAW;AAAA,QACX,YAAY;AAAA,QACZ,SAAS;AAAA,QACT,WAAW;AAAA,QACX,KAAK;AAAA,QACL,OAAO;AAAA,MAAA;AAAA,IAy4BX;AAAA;AAAA,IA39BE,IAAgB,kBAAe;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAA/B,IAAgB,gBAAe,OAAA;AAAA,yBAAA,mCAAA;AAAA,IAAA;AAAA;AAAA,IAK/B,IAAgB,gBAAa;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAA7B,IAAgB,cAAa,OAAA;AAAA,yBAAA,iCAAA;AAAA,IAAA;AAAA;AAAA,IAK7B,IAAgB,gBAAa;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAA7B,IAAgB,cAAa,OAAA;AAAA,yBAAA,iCAAA;AAAA,IAAA;AAAA;AAAA,IAK7B,IAAgB,eAAY;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAA5B,IAAgB,aAAY,OAAA;AAAA,yBAAA,gCAAA;AAAA,IAAA;AAAA;AAAA,IAK5B,IAAgB,SAAM;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAAtB,IAAgB,OAAM,OAAA;AAAA,yBAAA,0BAAA;AAAA,IAAA;AAAA;AAAA,IAKtB,IAAgB,sBAAmB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAAnC,IAAgB,oBAAmB,OAAA;AAAA,yBAAA,uCAAA;AAAA,IAAA;AAAA;AAAA,IAKnC,IAAgB,yBAAsB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAAtC,IAAgB,uBAAsB,OAAA;AAAA,yBAAA,0CAAA;AAAA,IAAA;AAAA;AAAA,IAKtC,IAAgB,oBAAiB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAAjC,IAAgB,kBAAiB,OAAA;AAAA,yBAAA,qCAAA;AAAA,IAAA;AAAA,IAIjC,IAAgB,sBAAmB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAAnC,IAAgB,oBAAmB,OAAA;AAAA,yBAAA,uCAAA;AAAA,IAAA;AAAA,IAE1B,IAAmB,qBAAkB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAArC,IAAmB,mBAAkB,OAAA;AAAA,yBAAA,sCAAA;AAAA,IAAA;AAAA,IACrC,IAAmB,oBAAiB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAApC,IAAmB,kBAAiB,OAAA;AAAA,yBAAA,qCAAA;AAAA,IAAA;AAAA;AAAA,IAEpC,IAAmB,0BAAuB;AAAA,aAAA,mBAAA;AAAA,IAAA;AAAA,IAA1C,IAAmB,wBAAuB,OAAA;AAAA,yBAAA,2CAAA;AAAA,IAAA;AAAA,IAyChC,WAAW,mBAAuC;AACnE,YAAM,WAAW,iBAAiB;AAElC,UAAI,kBAAkB,IAAI,iBAAiB,GAAG;AAC5C,aAAK,mCAAA;AAAA,MACP;AAEA,UAAI,kBAAkB,IAAI,cAAc,GAAG;AACzC,aAAK,oBAAoB,KAAK,qBAAqB,KAAK;AACxD,aAAK,OAAO,YAAY,oCAAoC,GAAG,KAAK,YAAY,IAAI;AAEpF,aAAK,qCAAA;AAAA,MACP;AAGA,UAAI,kBAAkB,IAAI,QAAQ,KAAK,CAAC,CAAC,KAAK,QAAQ;AACpD,YAAI,KAAK,gBAAgB,WAAW,QAAQ;AAC1C,eAAK,eAAe,KAAK,SAAS,KAAK,gBAAgB,WAAW,CAAC,EAAE,UAAU;AAC/E,eAAK,oBAAoB,KAAK,qBAAqB,KAAK;AACxD,eAAK,OAAO,YAAY,oCAAoC,GAAG,KAAK,YAAY,IAAI;AAEpF,eAAK,qCAAA;AAAA,QACP;AAAA,MACF;AAEA,UAAI,kBAAkB,IAAI,eAAe,KAAK,KAAK,eAAe;AAChE,aAAK,qCAAA;AAAA,MACP;AAEA,UAAI,kBAAkB,IAAI,qBAAqB,KAAK,KAAK,qBAAqB;AAG5E,mBAAW,MAAM,KAAK,yBAAyB,KAAK,mBAAmB,GAAG,CAAC;AAAA,MAC7E;AAAA,IACF;AAAA,IAEU,uBAAuB,cAAoB;AACnD,WAAK,oBAAA;AACL,UAAI,uBAAuB,KAAK;AAChC,UAAI,gBAAgB,oBAAoB,uBAAuB,GAAG;AAChE,+BACE,KAAK,0BAA0B,KAAK,KAAK,yBAAyB,IAAI;AAAA,MAC1E,WACE,gBAAgB,qBAChB,uBAAuB,KAAK,gBAAgB,WAAW,SAAS,GAChE;AACA,+BACE,KAAK,0BAA0B,KAAK,KAAK,yBAAyB,IAAI;AAAA,MAC1E;AAEA,WAAK,yBAAyB,oBAAoB;AAAA,IACpD;AAAA;AAAA,IAGU,uCAAoC;AAC5C,UAAI,KAAK,gBAAgB,WAAW,SAAS,GAAG;AAC9C,cAAM,gBACH,KAAK,gBAAgB,WAAW,CAAC,EAAE,UAAU,IAAI,KAAK,oBAAoB,KAC3E,KAAK;AACP,aAAK,OAAO,YAAY,uCAAuC,GAAG,YAAY,EAAE;AAAA,MAClF;AAEA,WAAK,kBAAkB,KAAK,YAAY,cAAc,oBAAoB;AAC1E,WAAK,iBAAiB,KAAK,YAAY,cAAc,mBAAmB;AACxE,WAAK,kBAAkB,KAAK,YAAY,cAAc,sBAAsB;AAC5E,WAAK,uBAAuB,KAAK,YAAY,cAAc,oBAAoB;AAE/E,UAAI,KAAK,sBAAsB;AAC7B,aAAK,0BAA0B,KAAK,gBAChC,KAAK,qBAAqB,sBAAA,EAAwB,SAClD,KAAK,qBAAqB,sBAAA,EAAwB;AAGtD,cAAM,gBAAgB,KAAK,qBAAqB,cAAc,SAAS;AACvE,cAAM,sBAAsB,eAAe,sBAAA;AAC3C,aAAK,oBAAoB,KAAK,gBAC1B,qBAAqB,SACrB,qBAAqB;AAAA,MAC3B;AAEA,UAAI,KAAK,iBAAiB;AACxB,YAAI,qBAAqB;AACzB,aAAK,wBAAwB,KAAK,gBAC9B,KAAK,gBAAgB,sBAAA,EAAwB,SAC7C,KAAK,gBAAgB,sBAAA,EAAwB;AAGjD,aAAK,kCAAkC,KAAK,gBAAgB,WAAW,IAAI,CAAC,UAAS;AACnF,gBAAM,YAAY;AAClB,gBAAM,aAAa,KAAK,uBAAuB,MAAM,SAAS,EAAE;AAGhE,gCAAsB,aAAa,KAAK;AAExC,iBAAO;AAAA,YACL,OAAO;AAAA,YACP,KAAK;AAAA,YACL,OAAO;AAAA,UAAA;AAAA,QAEX,CAAC;AAGD,aAAK,qBAAqB;AAG1B,aAAK,gBAAgB,iBAAiB,aAAa,MAAK;AACtD,gBAAM,uBAAuB,KAAK,kBAC9B,KAAK,yBACL,KAAK,sCAAA;AAIT,cAAI,KAAK,2BAA2B,IAAI;AACtC,iBAAK,yBAAyB;AAAA,UAChC;AAEA,cAAI,KAAK,gCAAgC;AACvC,iBAAK,yBAAyB;AAAA,UAChC,OAAO;AACL,iBAAK,yBACH,uBAAuB,KAAK,yBACxB,KAAK,yBACL;AAAA,UACR;AAEA,cAAI,CAAC,KAAK,iBAAiB;AAEzB,iBAAK,kCAAkC,oBAAoB;AAAA,UAC7D;AAEA,eAAK,iCAAiC;AACtC,eAAK,2BAAA;AAEL,cAAI,CAAC,KAAK,eAAe;AACvB,iBAAK,sBAAA;AACL,iBAAK,kBAAkB;AAAA,UACzB;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAQU,mCAAmC,iBAAuB;AAClE,UAAI,CAAC,KAAK,4CAA4C,CAAC,KAAK,eAAe;AACzE,aAAK,2CAA2C;AAChD,aAAK,yBACH,oBAAoB,IAChB,KAAK,2BAA2B,EAAE,IAClC,KAAK,2BAA2B,eAAe;AACrD,aAAK,sBAAA;AAAA,MACP;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA,IAMU,oBAAoB,OAAoB;AAChD,YAAM,aAAa,MAAM;AAIzB,UAAI,KAAK,mBAAmB;AAC1B,YAAI,MAAM,YAAY,MAAM,YAAY,GAAG;AACzC,eAAK,mCAAmC,UAAU;AAClD,gBAAM,eAAA;AACN;AAAA,QACF;AAEA,YAAI,eAAe,KAAK,yBAAyB,KAAK;AACpD,eAAK,mCAAmC,UAAU;AAClD,gBAAM,eAAA;AACN;AAAA,QACF;AAAA,MACF;AAGA,UAAI,KAAK,2BAA2B,MAAM,4BAA4B,KAAK,GAAG;AAC5E,cAAM,eAAA;AAEN,gBAAQ,YAAA;AAAA,UACN,KAAK,KAAK,yBAAyB;AACjC;AACE,oBAAM,wBAAgC,KAAK,gBACvC,KAAK,yBAAyB,YAC9B;AACJ,mBAAK,2BAA2B,qBAAqB;AAAA,YACvD;AACA;AAAA,UACF,KAAK,KAAK,yBAAyB;AACjC;AACE,oBAAM,yBAAiC,KAAK,gBACxC,KAAK,yBAAyB,UAC9B;AACJ,mBAAK,2BAA2B,sBAAsB;AAAA,YACxD;AACA;AAAA,UACF,KAAK,KAAK,yBAAyB;AACjC;AACE,oBAAM,sBAA8B,KAAK,gBACrC,KAAK,yBAAyB,YAC9B;AACJ,mBAAK,2BAA2B,mBAAmB;AAAA,YACrD;AACA;AAAA,UACF,KAAK,KAAK,yBAAyB;AACjC;AACE,oBAAM,wBAAgC,KAAK,gBACvC,KAAK,yBAAyB,aAC9B;AACJ,mBAAK,2BAA2B,qBAAqB;AAAA,YACvD;AACA;AAAA,QAEA;AAAA,MAEN;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA,IAMU,wBAAqB;AAC7B,YAAM,eAAe,KAAK,+BAAA;AAE1B,UAAI,cAAc;AAChB,aAAK,oBAAA;AAAA,MACP;AAIA,UAAI,KAAK,wBAAwB,KAAK,iBAAiB;AACrD,YAAI,cAAc;AAChB,eAAK,kBAAkB,YAAY;AAAA,QACrC;AAAA,MACF,OAGK;AAEH,aAAK,oBAAoB;AACzB,aAAK,6BAAA;AAAA,MACP;AAAA,IACF;AAAA,IAEU,yBAAyB,uBAA6B;AAC9D,UAAI,0BAA0B,KAAK,wBAAwB;AACzD,aAAK,kBAAkB;AACvB,aAAK,uBAAuB;AAC5B,aAAK,yBAAyB;AAC9B,aAAK,gCAAA;AAEL,cAAM,oBAAoB,KAAK,yBAAA;AAC/B,cAAM,iCACJ,KAAK,uBAAuB,MAAM,KAAK,yBAAyB;AAGlE,YAAI,kCAAkC,KAAK,gCAAgC;AACzE,eAAK,gBAAgB,SAAS;AAAA,YAC5B,KAAK,KAAK,gBAAgB,oBAAoB;AAAA,YAC9C,MAAM,KAAK,gBAAgB,IAAI;AAAA,YAC/B,UAAU;AAAA,UAAA,CACX;AAAA,QACH,OAAO;AACL,eAAK,2BAAA;AAAA,QACP;AAGA,aAAK,kCAAkC,qBAAqB;AAAA,MAC9D;AAAA,IACF;AAAA,IAEU,kBAAkB,OAAqB,YAAmB;AAClE,WAAK,oBAAA;AACL,UAAI,OAAO;AACT,aAAK,oBAAoB;AACzB,YAAI,YAAY;AACd,eAAK,yBAAyB;AAAA,QAChC;AAEA,aAAK,+BAA+B,KAAK;AAEzC,cAAM,uBAAuB,KAAK,qBAAA;AAClC,YAAI,sBAAsB;AACxB,+BAAqB,aAAa,YAAY,OAAO;AAAA,QACvD;AAAA,MACF;AAAA,IACF;AAAA,IAEU,sBAAmB;AAC3B,YAAM,uBAAuB,KAAK,qBAAA;AAClC,UAAI,sBAAsB;AACxB,6BAAqB,aAAa,YAAY,SAAS;AACvD,aAAK,+BAA+B,IAAI;AACxC,aAAK,oBAAoB;AAAA,MAC3B;AAAA,IACF;AAAA,IAEU,uBACR,kBACA,gBACA,gBACA,iBAAyB;AAEzB,UAAI,kBAAkB,CAAC,gBAAgB;AAGrC,yBAAiB,KAAK,KAAK,oBAAoB;AAAA,MACjD;AAEA,UAAI,iBAAiB;AAGnB,yBAAiB,KAAK,KAAK;AAAA,MAC7B;AAEA,aAAO;AAAA,QACL,GAAG,KAAK,eAAe,iBAAiB;AAAA,QACxC,GAAG,KAAK,eAAe,iBAAiB;AAAA,MAAA;AAAA,IAE5C;AAAA,IAEU,sBACR,iBACA,kBACA,gBACA,gBAAwB;AAExB,UAAI,kBAAkB,kBAAkB;AACtC,cAAM,eAAe,iBACjB,eAAe,IACf,eAAe,IAAI,KAAK;AAE5B,YAAI,gBAAgB,MAAM,GAAG;AAC3B,0BAAgB,KAAK,KAAK;AAAA,QAC5B,WAAW,gBAAgB,IAAI,iBAAiB,MAAM,cAAc;AAClE,0BAAgB,KAAK,KAAK;AAAA,QAC5B;AAAA,MACF;AAEA,aAAO;AAAA,QACL,GAAG,KAAK,eAAe,gBAAgB;AAAA,QACvC,GAAG,KAAK,eAAe,gBAAgB;AAAA,QACvC,GAAG,gBAAgB;AAAA,MAAA;AAAA,IAEvB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAQU,mCAAmC,YAAkB;AAC7D,YAAM,cAAuC,EAAE,OAAO,GAAG,UAAU,EAAA;AACnE,YAAM,aAAa,KAAK,gBAAgB,WAAW,UAAU,EAAE,QAAQ,OACrE,CAACC,cAAa,cAAoB;AAChC,YAAI,UAAU,UAAU,QAAQ;AAC9B,iBAAOA;AAAAA,QACT;AAEA,YAAI,UAAU,SAAS,QAAQ;AAC7BA,uBAAY;AAAA,QACd,OAAO;AACLA,uBAAY;AAAA,QACd;AACA,eAAOA;AAAAA,MACT,GACA,WAAW;AAEb,aAAO,aAAa,aAAa;AAAA,IACnC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAOQ,kCAAkC,uBAA6B;AAErE,iBAAW,MAAK;AAEd,aAAK,0BAA0B;AAE/B,YAAI,KAAK,iBAAiB,KAAK,sBAAsB;AACnD,gBAAM,6BAA6B,KAAK,0BAA0B;AAClE,gBAAM,sBAAsB,wBAAwB,KAAK;AACzD,gBAAM,gBACJ,sBAAsB,6BAA6B,KAAK;AAE1D,eAAK,qBAAqB,SAAS;AAAA,YACjC,KAAK,KAAK,gBAAgB,gBAAgB;AAAA,YAC1C,MAAM,KAAK,gBAAgB,IAAI;AAAA,YAC/B,UAAU;AAAA,UAAA,CACX;AAAA,QACH;AAAA,MACF,GAAG,EAAE;AAAA,IACP;AAAA;AAAA;AAAA;AAAA,IAKQ,kCAA+B;AACrC,WAAK,sBACH,KAAK,yBAAyB,KAAK,qBAC/B,gBAAgB,QAChB,gBAAgB;AAAA,IACxB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAQQ,2BAAwB;AAC9B,YAAM,oBAAoB,KAAK,gCAAgC,KAAK,sBAAsB;AAE1F,YAAM,wCACJ,KAAK,wBACL,KAAK,wBAAwB,gBAAgB,QAC7C,kBAAkB,QAAQ,KAAK;AAEjC,aAAO,wCACH,kBAAkB,MAAM,KAAK,wBAC7B,kBAAkB;AAAA,IACxB;AAAA;AAAA;AAAA;AAAA,IAKQ,+BAA4B;AAElC,UAAI,KAAK,sBAAsB;AAC7B,aAAK,uBAAuB;AAC5B,cAAM,2BAA2B,KAAK,YAAY,cAChD,2BAA2B,KAAK,sBAAsB;AAExD,YAAI,0BAA0B;AAC5B,mCAAyB,MAAA;AAAA,QAC3B;AAAA,MACF;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,+BAA4B;AAClC,YAAM,iBAAiB,KAAK,gBACxB,KAAK,gBAAgB,YACrB,KAAK,gBAAgB;AACzB,YAAM,yBAAyB,KAAK,gBAChC,KAAK,gBAAgB,sBAAA,EAAwB,SAC7C,KAAK,gBAAgB,sBAAA,EAAwB;AACjD,YAAM,qBAAqB,KAAK,qBAAqB;AACrD,YAAM,mBAAmB,KAAK,gCAAgC,KAAK,sBAAsB;AACzF,YAAM,gCACJ,mBAAmB,KAAK,gCAAgC,KAAK,sBAAsB,EAAE;AAEvF,cACG,iBAAiB,sBAAsB,iBAAiB,iBAAiB,UAC1E,CAAC;AAAA,IAEL;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,wCAAqC;AAC3C,YAAM,YAAY,KAAK,gBACnB,KAAK,gBAAgB,YACrB,KAAK,gBAAgB;AACzB,YAAM,gBAAgB,YAAY,KAAK,wBAAwB;AAC/D,aAAO,KAAK,gCAAgC,UAC1C,CAAC,iBAAiB,iBAAiB,aAAa,SAAS,iBAAiB,aAAa,GAAG;AAAA,IAE9F;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,+BAA4B;AAClC,UAAI,aAA2B;AAC/B,YAAM,QAAQ,KAAK,iBAAiB,WAAW,KAAK,sBAAsB;AAC1E,YAAM,cAAc,UAAU,KAAK,yBAAyB;AAC5D,YAAM,cACJ,KAAK,YACD,cAAoC,MAAM,WAAW,GACrD,cAAqD,oCAAoC,GACzF,aAAa,MAAM,KAAK;AAE9B,UAAI,SAAS,aAAa;AACxB,qBAAa,MAAM,QAAQ,KAAK,CAAC,UAAU,MAAM,WAAW,WAAW,KAAK;AAAA,MAC9E;AACA,aAAO;AAAA,IACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAQQ,+BAA+B,YAAmB;AACxD,YAAM,QAAQ,KAAK,iBAAiB,WAAW,KAAK,sBAAsB;AAC1E,UAAI,eAAe;AACnB,UAAI,MAAM,QAAQ;AAEhB,YAAI,CAAC,KAAK,4BAA4B;AACpC,iBAAO,KAAK,6BAAA;AAAA,QACd,OAAO;AACL,cAAI,KAAK,mBAAmB;AAC1B,uBAAW,SAAS,MAAM,QAAQ;AAEhC,kBAAI,CAAC,YAAY;AAEf,oBACE,KAAK,wBAAwB,gBAAgB,SAC7C,MAAM,SAAS,MAAM,KAAK,mBAAmB,SAAS,MACrD,CAAC,gBAAgB,MAAM,SAAS,IAAI,aAAa,SAAS,IAC3D;AACA,iCAAe;AAAA,gBACjB,WAGE,KAAK,wBAAwB,gBAAgB,QAC7C,MAAM,SAAS,MAAM,KAAK,mBAAmB,SAAS,MACrD,CAAC,gBAAgB,MAAM,SAAS,IAAI,aAAa,SAAS,IAC3D;AACA,iCAAe;AAAA,gBACjB;AAAA,cACF,OAAO;AACL,oBAAI,MAAM,WAAW,KAAK,mBAAmB,QAAQ;AAEnD,sBACE,eAAe,KAAK,yBAAyB,eAC5C,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,KACpD,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,IAAI,MAC3D,MAAM,SAAS,IAAI,KAAK,kBAAkB,SAAS,MAClD,CAAC,gBAAgB,MAAM,SAAS,IAAI,aAAa,SAAS,IAC3D;AACA,mCAAe;AAAA,kBACjB,WAGE,eAAe,KAAK,yBAAyB,cAC5C,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,KACpD,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,IAAI,MAC3D,MAAM,SAAS,IAAI,KAAK,kBAAkB,SAAS,MAClD,CAAC,gBAAgB,MAAM,SAAS,IAAI,aAAa,SAAS,IAC3D;AACA,mCAAe;AAAA,kBACjB,WAGE,eAAe,KAAK,yBAAyB,cAC5C,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,KACpD,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,IAAI,MAC3D,MAAM,SAAS,IAAI,KAAK,kBAAkB,SAAS,MAClD,CAAC,gBAAgB,MAAM,SAAS,IAAI,aAAa,SAAS,IAC3D;AACA,mCAAe;AAAA,kBACjB,WAGE,eAAe,KAAK,yBAAyB,YAC5C,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,KACpD,MAAM,SAAS,MAAM,KAAK,kBAAkB,SAAS,IAAI,MAC3D,MAAM,SAAS,IAAI,KAAK,mBAAmB,SAAS,MACnD,CAAC,gBAAgB,MAAM,SAAS,IAAI,aAAa,SAAS,IAC3D;AACA,mCAAe;AAAA,kBACjB;AAAA,gBACF;AAAA,cACF;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,MACF;AACA,aAAO;AAAA,IACT;AAAA;AAAA;AAAA,IAIQ,mCAAmC,cAAoB;AAC7D,YAAM,iBACJ,KAAK,sBAAsB,KACvB,KAAK,2BAA2B,KAC9B,IACA,KAAK,yBACP,KAAK;AAEX,YAAM,oBACJ,iBAAiB,aACb,KAAK,2BAA2B,cAAc,IAC9C,KAAK,2BAA2B,cAAc;AAGpD,UAAI,mBAAmB,mBAAmB;AACxC,aAAK,oBAAA;AACL,aAAK,2CAA2C;AAEhD,YAAI,iBAAiB;AAAY,eAAK,eAAe,MAAA;AAAA;AAChD,eAAK,gBAAgB,MAAA;AAE1B;AAAA,MACF;AAEA,UAAI,KAAK,eAAe;AACtB,cAAM,uBAAuB,KAAK,qBAAA;AAClC,cAAM,uBAAuB,uBACzB,qBAAqB,aAAa,UAAU,MAAM,UAClD;AAIJ,YAAI,iBAAiB,cAAc,KAAK,uBAAuB,gBAAgB;AAC7E,cAAI,sBAAsB;AACxB,iBAAK,oBAAoB;AACzB,iBAAK,oBAAA;AAEL;AAAA,UACF,OAAO;AACL,iBAAK,oBAAoB;AAAA,UAC3B;AAAA,QACF,WAES,sBAAsB,KAAK,wBAAwB;AAC1D,eAAK,oBAAoB;AAAA,QAC3B,OAAO;AACL,eAAK,oBAAoB;AACzB,eAAK,qBAAqB;AAE1B,cAAI,sBAAsB;AACxB,iBAAK,kBAAkB,KAAK,iBAAiB;AAAA,UAC/C,OAEK;AACH,iBAAK,uBAAuB;AAC5B,iBAAK,6BAAA;AAAA,UACP;AAAA,QACF;AAGA,aAAK,kCAAkC,iBAAiB;AAAA,MAC1D,OAEK;AACH,aAAK,yBAAyB,iBAAiB;AAAA,MACjD;AAAA,IACF;AAAA,IAEQ,2BAA2B,YAAkB;AACnD,WAAK,iCAAiC;AACtC,WAAK,uBAAuB;AAE5B,UAAI,KAAK,sBAAsB,IAAI;AACjC,aAAK,oBAAoB;AAAA,MAC3B;AAEA,UAAI,CAAC,KAAK,mBAAmB;AAC3B,cAAM,mBAAmB,KAAK,+BAA+B,UAAU;AACvE,YAAI,kBAAkB;AACpB,eAAK,kBAAkB,gBAAgB;AAAA,QACzC,OAEK;AACH,cACE,eAAe,KAAK,yBAAyB,cAC7C,eAAe,KAAK,yBAAyB,aAC5C,KAAK,kBACH,eAAe,KAAK,yBAAyB,WAC5C,eAAe,KAAK,yBAAyB,YACjD;AAEA,kBAAM,wBACJ,eAAe,KAAK,yBAAyB,aACzC,KAAK,2BAAA,IACL,KAAK,2BAAA;AAEX,iBAAK,yBAAyB,qBAAqB;AAAA,UACrD;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,IAEU,2BAA2B,cAAqB;AACxD,YAAM,aAAa,gBAAgB,KAAK;AACxC,aAAO,aAAa,KAAK,gBAAgB,WAAW,SAAS,IAAI,aAAa,IAAI;AAAA,IACpF;AAAA,IAEU,2BAA2B,cAAqB;AACxD,YAAM,aAAa,gBAAgB,KAAK;AACxC,aAAO,aAAa,IAAI,aAAa,IAAI;AAAA,IAC3C;AAAA,IAEU,oCAAoC,gBAA8B;AAC1E,UAAI,eAAe,cAAc,QAAQ;AACvC,aAAK,8BAA8B,QAAQ,KAAK,qCAC9C,KAAK,8BAA8B,OACnC,KAAK,qBACL,cAAc;AAAA,MAElB,OAAO;AACL,aAAK,8BAA8B,WAAW,KAAK,qCACjD,KAAK,8BAA8B,UACnC,KAAK,wBACL,cAAc;AAAA,MAElB;AAMA,WAAK,cACH,IAAI,YAAuD,kBAAkB;AAAA,QAC3E,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ,KAAK;AAAA,MAAA,CACd,CAAC;AAAA,IAEN;AAAA,IAEQ,qCACN,+BACA,iBACA,gBAA8B;AAG9B,UAAI,eAAe,UAAU,YAAY;AACvC,cAAM,2BAA2B,KAAK,kCAAkC,cAAc;AACtF,YAAI,0BAA0B;AAC5B,wCAA8B,KAAK,wBAAwB;AAAA,QAC7D;AAAA,MACF,OAEK;AACH,wCAAgC,8BAA8B,OAC5D,CAAC,mBAAmB,eAAe,OAAO,eAAe,EAAE;AAAA,MAE/D;AAGA,UAAI,kBAAkB,MAAM,8BAA8B,SAAS,iBAAiB;AAClF,YAAI,oBAAoB,GAAG;AAEzB,0CAAgC,KAAK,yBACnC,6BAA6B;AAAA,QAEjC,OAAO;AAEL,0CAAgC,KAAK,yBACnC,+BACA,cAAc;AAAA,QAElB;AAAA,MACF;AAEA,aAAO;AAAA,IACT;AAAA,IAEU,kCAAkC,gBAA8B;AACxE,YAAM,aAAa,eAAe;AAClC,YAAM,QAAQ,KAAK,gBAAgB,WAAW,UAAU,EAAE,QAAQ,KAChE,CAACC,WAAUA,OAAM,UAAU,eAAe,MAAM;AAGlD,UAAI,CAAC;AAAO;AACZ,WAAK,yBAAyB;AAC9B,WAAK,oBAAoB;AACzB,UAAI,KAAK,2BAA2B,KAAK,oBAAoB;AAC3D,aAAK,2BAAA;AAAA,MACP;AAEA,WAAK,+BAA+B,KAAK;AAAA,IAC3C;AAAA,IAEU,6BAA0B;AAClC,WAAK,qBAAqB,KAAK;AAC/B,WAAK,oBAAoB;AACzB,YAAM,iBAAiB,KAAK,kCAAkC,KAAK,kBAAkB;AACrF,UAAI,gBAAgB;AAKlB,aAAK,cACH,IAAI,YAA2C,iBAAiB;AAAA,UAC9D,SAAS;AAAA,UACT,UAAU;AAAA,UACV,QAAQ;AAAA,QAAA,CACT,CAAC;AAAA,MAEN;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,qCAAkC;AACxC,WAAK,gBAAgB,WAAW,IAAI,CAAC,OAAkB,eAAsB;AAC3E,cAAM,QACF,OAAO,CAAC,UAAU,MAAM,UAAU,UAAU,GAC5C,QAAQ,CAAC,UAAS;AAClB,gBAAM,4BAA4C,8BAChD,OACA,UAAU;AAEZ,gBAAM,gCACJ,KAAK,kCAAkC,yBAAyB;AAClE,cAAI,+BAA+B;AACjC,gBAAI,8BAA8B,cAAc,QAAQ;AACtD,mBAAK,8BAA8B,MAAM,KAAK,6BAA6B;AAAA,YAC7E,OAAO;AACL,mBAAK,8BAA8B,SAAS,KAAK,6BAA6B;AAAA,YAChF;AAAA,UACF;AAAA,QACF,CAAC;AAAA,MACL,CAAC;AAAA,IACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,yBACN,4BACA,mBAAkC;AAGlC,iBAAW,kBAAkB,4BAA4B;AACvD,YAAI,CAAC,qBAAqB,kBAAkB,OAAO,eAAe,IAAI;AACpE,gBAAM,eAAe,KAAK,YAAY,eAAe,eAAe,EAAE;AACtE,uBAAa,aAAa,SAAS,MAAM;AAAA,QAC3C;AAAA,MACF;AAEA,UAAI,mBAAmB;AACrB,qCAA6B,2BAA2B,OACtD,CAAC,mBAAmB,eAAe,OAAO,kBAAkB,EAAE;AAAA,MAElE,OAAO;AACL,qCAA6B,CAAA;AAAA,MAC/B;AACA,aAAO;AAAA,IACT;AAAA,IAEQ,kCACN,mBAAiC;AAEjC,YAAM,QAAQ,KAAK,gBAAgB,WAAW,kBAAkB,UAAU;AAC1E,YAAM,QAAQ,MAAM,QAAQ,KAAK,CAACA,WAAUA,OAAM,WAAW,kBAAkB,MAAM;AAErF,aAAO,QACH,gDAAgD,OAAO,OAAO,kBAAkB,UAAU,IAC1F;AAAA,IACN;AAAA,IAEQ,kCACN,YAAkB;AAElB,UAAI,CAAC,KAAK,gBAAgB,WAAW,UAAU;AAAG,eAAO;AAEzD,YAAM,QAAQ,KAAK,gBAAgB,WAAW,UAAU;AACxD,YAAM,0BAA0B,KAAK,mCAAmC,UAAU;AAClF,aAAO,8BAA8B,YAAY,OAAO,uBAAuB;AAAA,IACjF;AAAA,IAEQ,+BAA+B,OAAmB;AACxD,UAAI,OAAO;AACT,aAAK,6BACH,oCAAoC,KAAK,yBAAyB,MAAM,MAAM;AAAA,MAClF,OAAO;AACL,aAAK,6BAA6B;AAAA,MACpC;AAAA,IACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IASQ,qBAAqB,aAAsB,YAAmB;AACpE,YAAM,iBAAiB,aAAa,aAAa,KAAK;AACtD,YAAM,qBAAqB,cACvB,oCAAoC,iBAAiB,MAAM,cAC3D,KAAK;AACT,aAAO,qBAAqB,KAAK,YAAY,eAAe,kBAAkB,KAAK,OAAO;AAAA,IAC5F;AAAA,EAAA,GA19BA,mDAKA,iDAKA,iDAKA,gDAKA,0CAKA,uDAKA,0DAKA,qDAIA,uDAES,sDACA,qDAEA;;mCA7CR,SAAS,EAAE,WAAW,oBAAoB,MAAM,OAAA,CAAQ,CAAC;AAIzD,gCAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,kBAAkB,MAAM,QAAA,CAAS,CAAC;AAIxD,gCAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,kBAAkB,MAAM,QAAA,CAAS,CAAC;AAIxD,+BAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,kBAAkB,MAAM,OAAA,CAAQ,CAAC;AAIvD,yBAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,UAAU,MAAM,OAAA,CAAQ,CAAC;AAI/C,sCAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,yBAAyB,MAAM,OAAA,CAAQ,CAAC;AAI9D,yCAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,4BAA4B,MAAM,OAAA,CAAQ,CAAC;AAIjE,oCAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,uBAAuB,MAAM,QAAA,CAAS,CAAC;AAG7D,sCAAA,CAAA,UAAA,GACA,SAAS,EAAE,WAAW,yBAAyB,MAAM,OAAA,CAAQ,CAAC;AAG9D,qCAAA,CAAA,OAAO;AACP,oCAAA,CAAA,OAAO;AAEP,0CAAA,CAAA,OAAO;AA5CR,iBAAA,IAAA,MAAA,6BAAA,EAAA,MAAA,YAAA,MAAA,mBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,qBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,iBAAe,KAAA,CAAA,KAAA,UAAA;AAAA,UAAf,kBAAe;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,+BAAA,kCAAA;AAK/B,iBAAA,IAAA,MAAA,2BAAA,EAAA,MAAA,YAAA,MAAA,iBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,mBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,eAAa,KAAA,CAAA,KAAA,UAAA;AAAA,UAAb,gBAAa;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,6BAAA,gCAAA;AAK7B,iBAAA,IAAA,MAAA,2BAAA,EAAA,MAAA,YAAA,MAAA,iBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,mBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,eAAa,KAAA,CAAA,KAAA,UAAA;AAAA,UAAb,gBAAa;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,6BAAA,gCAAA;AAK7B,iBAAA,IAAA,MAAA,0BAAA,EAAA,MAAA,YAAA,MAAA,gBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,kBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,cAAY,KAAA,CAAA,KAAA,UAAA;AAAA,UAAZ,eAAY;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,4BAAA,+BAAA;AAK5B,iBAAA,IAAA,MAAA,oBAAA,EAAA,MAAA,YAAA,MAAA,UAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,YAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,QAAM,KAAA,CAAA,KAAA,UAAA;AAAA,UAAN,SAAM;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,sBAAA,yBAAA;AAKtB,iBAAA,IAAA,MAAA,iCAAA,EAAA,MAAA,YAAA,MAAA,uBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,yBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,qBAAmB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAnB,sBAAmB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,mCAAA,sCAAA;AAKnC,iBAAA,IAAA,MAAA,oCAAA,EAAA,MAAA,YAAA,MAAA,0BAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,4BAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,wBAAsB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAtB,yBAAsB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,sCAAA,yCAAA;AAKtC,iBAAA,IAAA,MAAA,+BAAA,EAAA,MAAA,YAAA,MAAA,qBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,uBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,mBAAiB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAjB,oBAAiB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,iCAAA,oCAAA;AAIjC,iBAAA,IAAA,MAAA,iCAAA,EAAA,MAAA,YAAA,MAAA,uBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,yBAAA,KAAA,KAAA,CAAA,QAAA,IAAgB,qBAAmB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAnB,sBAAmB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,mCAAA,sCAAA;AAE1B,iBAAA,IAAA,MAAA,gCAAA,EAAA,MAAA,YAAA,MAAA,sBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,wBAAA,KAAA,KAAA,CAAA,QAAA,IAAmB,oBAAkB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAlB,qBAAkB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,kCAAA,qCAAA;AACrC,iBAAA,IAAA,MAAA,+BAAA,EAAA,MAAA,YAAA,MAAA,qBAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,uBAAA,KAAA,KAAA,CAAA,QAAA,IAAmB,mBAAiB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAjB,oBAAiB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,iCAAA,oCAAA;AAEpC,iBAAA,IAAA,MAAA,qCAAA,EAAA,MAAA,YAAA,MAAA,2BAAA,QAAA,OAAA,SAAA,OAAA,QAAA,EAAA,KAAA,CAAA,QAAA,6BAAA,KAAA,KAAA,CAAA,QAAA,IAAmB,yBAAuB,KAAA,CAAA,KAAA,UAAA;AAAA,UAAvB,0BAAuB;AAAA,IAAA,KAAA,UAAA,UAAA,GAAA,uCAAA,0CAAA;;QAnD5B,GAAA,SAAS;AAAA,IAC9B,gBAAgB;AAAA,IAChB,eAAe;AAAA,EAAA,GAHN;;"}
|