@sbb-esta/lyne-elements-experimental 3.1.0 → 3.3.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 +1147 -590
- package/development/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/icon-mapper.js +4 -5
- package/development/seat-reservation/common/mapper/mapper.d.ts +6 -5
- package/development/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/mapper.js +14 -11
- package/development/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts +113 -0
- 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 +25999 -649
- 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.d.ts.map +1 -1
- package/development/seat-reservation/common/translations/i18n.js +225 -194
- package/development/seat-reservation/common/types.d.ts +20 -3
- package/development/seat-reservation/common/types.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +50 -18
- 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 +352 -172
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts +29 -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 +603 -308
- 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 +54 -32
- package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts +2 -0
- 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 +107 -58
- 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 +2 -2
- package/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/icon-mapper.js +3 -4
- package/seat-reservation/common/mapper/mapper.d.ts +6 -5
- package/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/mapper.js +35 -32
- package/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts +113 -0
- package/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
- package/seat-reservation/common/mapper/seat-reservation-sample-data.js +25997 -648
- 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.d.ts.map +1 -1
- package/seat-reservation/common/translations/i18n.js +228 -197
- package/seat-reservation/common/types.d.ts +20 -3
- package/seat-reservation/common/types.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +50 -18
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.js +294 -175
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts +29 -13
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation.component.js +318 -303
- 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 +95 -84
- package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts +2 -0
- 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 +75 -63
- 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,11 +17,11 @@ var ScrollDirection;
|
|
|
17
17
|
ScrollDirection2["left"] = "left";
|
|
18
18
|
})(ScrollDirection || (ScrollDirection = {}));
|
|
19
19
|
let SeatReservationBaseElement = (() => {
|
|
20
|
-
var
|
|
20
|
+
var _seatReservations_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
|
-
let
|
|
23
|
-
let
|
|
24
|
-
let
|
|
22
|
+
let _seatReservations_decorators;
|
|
23
|
+
let _seatReservations_initializers = [];
|
|
24
|
+
let _seatReservations_extraInitializers = [];
|
|
25
25
|
let _hasNavigation_decorators;
|
|
26
26
|
let _hasNavigation_initializers = [];
|
|
27
27
|
let _hasNavigation_extraInitializers = [];
|
|
@@ -34,57 +34,82 @@ 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);
|
|
52
|
-
__privateAdd(this,
|
|
61
|
+
__privateAdd(this, _seatReservations_accessor_storage);
|
|
53
62
|
__privateAdd(this, _hasNavigation_accessor_storage);
|
|
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);
|
|
61
|
-
|
|
62
|
-
__privateSet(this,
|
|
72
|
+
__privateAdd(this, _hoveredScrollCoachIndex_accessor_storage);
|
|
73
|
+
__privateSet(this, _seatReservations_accessor_storage, __runInitializers(this, _seatReservations_initializers, null));
|
|
74
|
+
__privateSet(this, _hasNavigation_accessor_storage, (__runInitializers(this, _seatReservations_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.gapBetweenCoachDecks = 48;
|
|
89
|
+
this.coachNavButtonDim = 0;
|
|
72
90
|
this.currScrollDirection = ScrollDirection.right;
|
|
73
91
|
this.maxCalcCoachsWidth = 0;
|
|
74
92
|
this.scrollCoachsAreaWidth = 0;
|
|
93
|
+
this.scrollNavigationAreaDim = 0;
|
|
75
94
|
this.triggerCoachPositionsCollection = [];
|
|
76
95
|
this.firstTabElement = null;
|
|
77
96
|
this.lastTabElement = null;
|
|
97
|
+
this.navigationScrollArea = null;
|
|
78
98
|
this.coachScrollArea = null;
|
|
79
99
|
this.currSelectedPlace = null;
|
|
80
100
|
this.currSelectedPlaceElementId = null;
|
|
81
101
|
this.currSelectedCoachIndex = -1;
|
|
102
|
+
this.currSelectedDeckIndex = 0;
|
|
82
103
|
this.preventCoachScrollByPlaceClick = false;
|
|
83
|
-
this.selectedSeatReservationPlaces =
|
|
104
|
+
this.selectedSeatReservationPlaces = {
|
|
105
|
+
seats: [],
|
|
106
|
+
bicycles: []
|
|
107
|
+
};
|
|
84
108
|
this.seatReservationWithoutNavigationHasFocus = false;
|
|
85
109
|
this.isCoachGridFocusable = false;
|
|
86
110
|
this.isAutoScrolling = false;
|
|
87
111
|
this.isKeyboardNavigation = false;
|
|
112
|
+
this.hasMultipleDecks = false;
|
|
88
113
|
this.keyboardNavigationEvents = {
|
|
89
114
|
ArrowLeft: "ArrowLeft",
|
|
90
115
|
ArrowRight: "ArrowRight",
|
|
@@ -94,12 +119,12 @@ let SeatReservationBaseElement = (() => {
|
|
|
94
119
|
Enter: "Enter"
|
|
95
120
|
};
|
|
96
121
|
}
|
|
97
|
-
/** The seat
|
|
98
|
-
get
|
|
99
|
-
return __privateGet(this,
|
|
122
|
+
/** The seat reservations array contains all coaches and places */
|
|
123
|
+
get seatReservations() {
|
|
124
|
+
return __privateGet(this, _seatReservations_accessor_storage);
|
|
100
125
|
}
|
|
101
|
-
set
|
|
102
|
-
__privateSet(this,
|
|
126
|
+
set seatReservations(value) {
|
|
127
|
+
__privateSet(this, _seatReservations_accessor_storage, value);
|
|
103
128
|
}
|
|
104
129
|
/** The seat reservation navigation can be toggled by this property */
|
|
105
130
|
get hasNavigation() {
|
|
@@ -130,11 +155,18 @@ let SeatReservationBaseElement = (() => {
|
|
|
130
155
|
__privateSet(this, _height_accessor_storage, value);
|
|
131
156
|
}
|
|
132
157
|
/** Maximal number of possible clickable seats */
|
|
133
|
-
get
|
|
134
|
-
return __privateGet(this,
|
|
158
|
+
get maxSeatReservations() {
|
|
159
|
+
return __privateGet(this, _maxSeatReservations_accessor_storage);
|
|
160
|
+
}
|
|
161
|
+
set maxSeatReservations(value) {
|
|
162
|
+
__privateSet(this, _maxSeatReservations_accessor_storage, value);
|
|
163
|
+
}
|
|
164
|
+
/** Maximal number of possible clickable bicycle places */
|
|
165
|
+
get maxBicycleReservations() {
|
|
166
|
+
return __privateGet(this, _maxBicycleReservations_accessor_storage);
|
|
135
167
|
}
|
|
136
|
-
set
|
|
137
|
-
__privateSet(this,
|
|
168
|
+
set maxBicycleReservations(value) {
|
|
169
|
+
__privateSet(this, _maxBicycleReservations_accessor_storage, value);
|
|
138
170
|
}
|
|
139
171
|
/** Any click functionality is prevented */
|
|
140
172
|
get preventPlaceClick() {
|
|
@@ -143,6 +175,12 @@ let SeatReservationBaseElement = (() => {
|
|
|
143
175
|
set preventPlaceClick(value) {
|
|
144
176
|
__privateSet(this, _preventPlaceClick_accessor_storage, value);
|
|
145
177
|
}
|
|
178
|
+
get preselectCoachIndex() {
|
|
179
|
+
return __privateGet(this, _preselectCoachIndex_accessor_storage);
|
|
180
|
+
}
|
|
181
|
+
set preselectCoachIndex(value) {
|
|
182
|
+
__privateSet(this, _preselectCoachIndex_accessor_storage, value);
|
|
183
|
+
}
|
|
146
184
|
get selectedCoachIndex() {
|
|
147
185
|
return __privateGet(this, _selectedCoachIndex_accessor_storage);
|
|
148
186
|
}
|
|
@@ -155,44 +193,85 @@ let SeatReservationBaseElement = (() => {
|
|
|
155
193
|
set focusedCoachIndex(value) {
|
|
156
194
|
__privateSet(this, _focusedCoachIndex_accessor_storage, value);
|
|
157
195
|
}
|
|
196
|
+
//Sets the hover style when scrolling to a coach
|
|
197
|
+
get hoveredScrollCoachIndex() {
|
|
198
|
+
return __privateGet(this, _hoveredScrollCoachIndex_accessor_storage);
|
|
199
|
+
}
|
|
200
|
+
set hoveredScrollCoachIndex(value) {
|
|
201
|
+
__privateSet(this, _hoveredScrollCoachIndex_accessor_storage, value);
|
|
202
|
+
}
|
|
158
203
|
willUpdate(changedProperties) {
|
|
159
204
|
super.willUpdate(changedProperties);
|
|
160
|
-
if (changedProperties.has("
|
|
205
|
+
if (changedProperties.has("seatReservations")) {
|
|
206
|
+
this.hasMultipleDecks = this.seatReservations?.length > 1;
|
|
161
207
|
this._initSeatReservationPlaceSelection();
|
|
162
208
|
}
|
|
163
209
|
if (changedProperties.has("baseGridSize")) {
|
|
164
210
|
this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;
|
|
165
211
|
this.style?.setProperty("--sbb-seat-reservation-grid-size", `${this.baseGridSize}px`);
|
|
166
|
-
|
|
167
|
-
this._setVerticalAlignmentOffset();
|
|
168
|
-
}
|
|
212
|
+
this.initNavigationSelectionByScrollEvent();
|
|
169
213
|
}
|
|
170
214
|
if (changedProperties.has("height") && !!this.height) {
|
|
171
|
-
|
|
172
|
-
|
|
215
|
+
const seatReservationLayer = this.seatReservations[this.currSelectedDeckIndex] || null;
|
|
216
|
+
if (seatReservationLayer?.coachItems.length) {
|
|
217
|
+
this.baseGridSize = this.height / seatReservationLayer.coachItems[0].dimension.h;
|
|
173
218
|
this.coachBorderOffset = this.coachBorderPadding / this.baseGridSize;
|
|
174
219
|
this.style?.setProperty("--sbb-seat-reservation-grid-size", `${this.baseGridSize}px`);
|
|
175
|
-
|
|
176
|
-
this._setVerticalAlignmentOffset();
|
|
177
|
-
}
|
|
220
|
+
this.initNavigationSelectionByScrollEvent();
|
|
178
221
|
}
|
|
179
222
|
}
|
|
180
223
|
if (changedProperties.has("alignVertical") && this.alignVertical) {
|
|
181
|
-
this.
|
|
224
|
+
this.initNavigationSelectionByScrollEvent();
|
|
225
|
+
}
|
|
226
|
+
if (changedProperties.has("preselectCoachIndex") && this.preselectCoachIndex) {
|
|
227
|
+
setTimeout(() => this.scrollToSelectedNavCoach(this.preselectCoachIndex), 1);
|
|
182
228
|
}
|
|
183
229
|
}
|
|
230
|
+
navigateByDirectionBtn(btnDirection) {
|
|
231
|
+
this.unfocusPlaceElement();
|
|
232
|
+
let navigateToCoachIndex = this.currSelectedCoachIndex;
|
|
233
|
+
if (btnDirection == "DIRECTION_LEFT" && navigateToCoachIndex > 0) {
|
|
234
|
+
navigateToCoachIndex = this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex - 1 : 0;
|
|
235
|
+
} else if (btnDirection == "DIRECTION_RIGHT" && navigateToCoachIndex < this.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1) {
|
|
236
|
+
navigateToCoachIndex = this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex + 1 : 0;
|
|
237
|
+
}
|
|
238
|
+
this.scrollToSelectedNavCoach(navigateToCoachIndex);
|
|
239
|
+
}
|
|
240
|
+
/**
|
|
241
|
+
* Data can be prepared once for the entire component
|
|
242
|
+
* in order to avoid recurring iteration processes in rendering.
|
|
243
|
+
*/
|
|
244
|
+
initPrepairSeatReservationData() {
|
|
245
|
+
this._prepairCoachDriverArea();
|
|
246
|
+
}
|
|
184
247
|
/* Init scroll event handling for coach navigation */
|
|
185
248
|
initNavigationSelectionByScrollEvent() {
|
|
186
|
-
this.firstTabElement = this.shadowRoot?.
|
|
187
|
-
this.lastTabElement = this.shadowRoot?.
|
|
188
|
-
this.coachScrollArea = this.shadowRoot?.
|
|
189
|
-
|
|
249
|
+
this.firstTabElement = this.shadowRoot?.querySelector("#first-tab-element");
|
|
250
|
+
this.lastTabElement = this.shadowRoot?.querySelector("#last-tab-element");
|
|
251
|
+
this.coachScrollArea = this.shadowRoot?.querySelector("#sbb-sr__wrapper-scrollarea");
|
|
252
|
+
this.navigationScrollArea = this.shadowRoot?.querySelector("#sbb-sr-navigation");
|
|
253
|
+
const seatReservationDeck = this.seatReservations ? this.seatReservations[this.currSelectedDeckIndex] : null;
|
|
254
|
+
if (seatReservationDeck && seatReservationDeck.coachItems.length > 0) {
|
|
255
|
+
const coachHeightWithBorderPadding = seatReservationDeck.coachItems[0].dimension.h * this.baseGridSize + this.coachBorderPadding;
|
|
256
|
+
const gapBetweenCoachDecks = (this.seatReservations.length - 1) * this.gapBetweenCoachDecks;
|
|
257
|
+
this.style?.setProperty("--sbb-seat-reservation-height", `${coachHeightWithBorderPadding * this.seatReservations.length + gapBetweenCoachDecks}`);
|
|
258
|
+
this.style?.setProperty("--sbb-seat-reservation-decks", `${this.seatReservations.length}`);
|
|
259
|
+
}
|
|
260
|
+
if (this.navigationScrollArea) {
|
|
261
|
+
this.scrollNavigationAreaDim = this.alignVertical ? this.navigationScrollArea.getBoundingClientRect().height : this.navigationScrollArea.getBoundingClientRect().width;
|
|
262
|
+
const navCoacheList = this.navigationScrollArea.querySelector("ul > li");
|
|
263
|
+
if (navCoacheList) {
|
|
264
|
+
const firstLiEleDimension = navCoacheList?.getBoundingClientRect();
|
|
265
|
+
this.coachNavButtonDim = this.alignVertical ? firstLiEleDimension.height : firstLiEleDimension.width;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
if (this.coachScrollArea && seatReservationDeck) {
|
|
190
269
|
let currCalcTriggerPos = 0;
|
|
191
|
-
this.scrollCoachsAreaWidth = this.coachScrollArea.getBoundingClientRect().width;
|
|
192
|
-
this.triggerCoachPositionsCollection =
|
|
270
|
+
this.scrollCoachsAreaWidth = this.alignVertical ? this.coachScrollArea.getBoundingClientRect().height : this.coachScrollArea.getBoundingClientRect().width;
|
|
271
|
+
this.triggerCoachPositionsCollection = seatReservationDeck.coachItems.map((coach) => {
|
|
193
272
|
const startPosX = currCalcTriggerPos;
|
|
194
273
|
const coachWidth = this.getCalculatedDimension(coach.dimension).w;
|
|
195
|
-
currCalcTriggerPos += coachWidth;
|
|
274
|
+
currCalcTriggerPos += coachWidth + this.gapBetweenCoaches;
|
|
196
275
|
return {
|
|
197
276
|
start: startPosX,
|
|
198
277
|
end: currCalcTriggerPos,
|
|
@@ -202,11 +281,17 @@ let SeatReservationBaseElement = (() => {
|
|
|
202
281
|
this.maxCalcCoachsWidth = currCalcTriggerPos;
|
|
203
282
|
this.coachScrollArea.addEventListener("scrollend", () => {
|
|
204
283
|
const findScrollCoachIndex = this.isAutoScrolling ? this.currSelectedCoachIndex : this._getCoachIndexByScrollTriggerPosition();
|
|
284
|
+
if (this.currSelectedCoachIndex === -1) {
|
|
285
|
+
this.currSelectedCoachIndex = findScrollCoachIndex;
|
|
286
|
+
}
|
|
205
287
|
if (this._isScrollableToSelectedCoach()) {
|
|
206
288
|
this.currSelectedCoachIndex = findScrollCoachIndex;
|
|
207
289
|
} else {
|
|
208
290
|
this.currSelectedCoachIndex = findScrollCoachIndex < this.currSelectedCoachIndex ? this.currSelectedCoachIndex : findScrollCoachIndex;
|
|
209
291
|
}
|
|
292
|
+
if (!this.isAutoScrolling) {
|
|
293
|
+
this._scrollToSelectedNavigationButton(findScrollCoachIndex);
|
|
294
|
+
}
|
|
210
295
|
this.preventCoachScrollByPlaceClick = false;
|
|
211
296
|
this.updateCurrentSelectedCoach();
|
|
212
297
|
if (!this.hasNavigation) {
|
|
@@ -214,9 +299,6 @@ let SeatReservationBaseElement = (() => {
|
|
|
214
299
|
this.isAutoScrolling = false;
|
|
215
300
|
}
|
|
216
301
|
});
|
|
217
|
-
if (this.alignVertical) {
|
|
218
|
-
this._setVerticalAlignmentOffset();
|
|
219
|
-
}
|
|
220
302
|
}
|
|
221
303
|
}
|
|
222
304
|
/**
|
|
@@ -308,14 +390,102 @@ let SeatReservationBaseElement = (() => {
|
|
|
308
390
|
const isSelectedCoachIndexScrollable = this.selectedCoachIndex !== -1 || this.currSelectedCoachIndex > 0;
|
|
309
391
|
if (isSelectedCoachIndexScrollable && this._isScrollableToSelectedCoach()) {
|
|
310
392
|
this.coachScrollArea.scrollTo({
|
|
311
|
-
top: 0,
|
|
312
|
-
left: scrollToCoachPosX,
|
|
393
|
+
top: this.alignVertical ? scrollToCoachPosX : 0,
|
|
394
|
+
left: this.alignVertical ? 0 : scrollToCoachPosX,
|
|
313
395
|
behavior: "smooth"
|
|
314
396
|
});
|
|
315
397
|
} else {
|
|
316
398
|
this.updateCurrentSelectedCoach();
|
|
317
399
|
}
|
|
400
|
+
this._scrollToSelectedNavigationButton(selectedNavCoachIndex);
|
|
401
|
+
}
|
|
402
|
+
}
|
|
403
|
+
focusPlaceElement(place) {
|
|
404
|
+
this.unfocusPlaceElement();
|
|
405
|
+
if (place) {
|
|
406
|
+
this.currSelectedPlace = place;
|
|
407
|
+
this._setCurrSelectedPlaceElementId(place);
|
|
408
|
+
const selectedPlaceElement = this._getPlaceHtmlElement();
|
|
409
|
+
if (selectedPlaceElement) {
|
|
410
|
+
selectedPlaceElement.setAttribute("keyfocus", "focus");
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
unfocusPlaceElement() {
|
|
415
|
+
const selectedPlaceElement = this._getPlaceHtmlElement();
|
|
416
|
+
if (selectedPlaceElement) {
|
|
417
|
+
selectedPlaceElement.setAttribute("keyfocus", "unfocus");
|
|
418
|
+
this._setCurrSelectedPlaceElementId(null);
|
|
419
|
+
this.currSelectedPlace = null;
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
getCalculatedDimension(elementDimension, coachDimension, isOriginHeight, isStretchHeight) {
|
|
423
|
+
if (coachDimension && !isOriginHeight) {
|
|
424
|
+
elementDimension.h += this.coachBorderOffset * 2;
|
|
425
|
+
}
|
|
426
|
+
if (isStretchHeight) {
|
|
427
|
+
elementDimension.h += this.coachBorderOffset;
|
|
428
|
+
}
|
|
429
|
+
return {
|
|
430
|
+
w: this.baseGridSize * elementDimension.w,
|
|
431
|
+
h: this.baseGridSize * elementDimension.h
|
|
432
|
+
};
|
|
433
|
+
}
|
|
434
|
+
getCalculatedPosition(elementPosition, elementDimension, coachDimension, isOriginHeight) {
|
|
435
|
+
if (coachDimension && elementDimension) {
|
|
436
|
+
const endPosHeight = isOriginHeight ? coachDimension.h : coachDimension.h + this.coachBorderOffset;
|
|
437
|
+
if (elementPosition.y === 0) {
|
|
438
|
+
elementPosition.y -= this.coachBorderOffset;
|
|
439
|
+
} else if (elementPosition.y + elementDimension.h === endPosHeight) {
|
|
440
|
+
elementPosition.y += this.coachBorderOffset;
|
|
441
|
+
}
|
|
318
442
|
}
|
|
443
|
+
return {
|
|
444
|
+
x: this.baseGridSize * elementPosition.x,
|
|
445
|
+
y: this.baseGridSize * elementPosition.y,
|
|
446
|
+
z: elementPosition.z
|
|
447
|
+
};
|
|
448
|
+
}
|
|
449
|
+
/**
|
|
450
|
+
* Counts all available seats together depending on the seat type
|
|
451
|
+
*
|
|
452
|
+
* @param coachIndex
|
|
453
|
+
* @returns An Object with count of free seats and free bicycle places
|
|
454
|
+
*/
|
|
455
|
+
getAvailableFreePlacesNumFromCoach(coachIndex) {
|
|
456
|
+
const accumulator = { seats: 0, bicycles: 0 };
|
|
457
|
+
const freePlaces = this.seatReservations[this.currSelectedDeckIndex].coachItems[coachIndex].places?.reduce((accumulator2, currPlace) => {
|
|
458
|
+
if (currPlace.state !== "FREE") {
|
|
459
|
+
return accumulator2;
|
|
460
|
+
}
|
|
461
|
+
if (currPlace.type === "SEAT") {
|
|
462
|
+
accumulator2.seats++;
|
|
463
|
+
} else {
|
|
464
|
+
accumulator2.bicycles++;
|
|
465
|
+
}
|
|
466
|
+
return accumulator2;
|
|
467
|
+
}, accumulator);
|
|
468
|
+
return freePlaces ? freePlaces : accumulator;
|
|
469
|
+
}
|
|
470
|
+
/**
|
|
471
|
+
* Performs an automatic main navigation scroll to the specified selectedNavCoachIndex.
|
|
472
|
+
* Calculates the central scroll offset of the nav coach to be selected.
|
|
473
|
+
* @param selectedNavCoachIndex
|
|
474
|
+
*/
|
|
475
|
+
_scrollToSelectedNavigationButton(selectedNavCoachIndex) {
|
|
476
|
+
setTimeout(() => {
|
|
477
|
+
this.hoveredScrollCoachIndex = selectedNavCoachIndex;
|
|
478
|
+
if (this.hasNavigation && this.navigationScrollArea) {
|
|
479
|
+
const navigationAreaCenteredPosX = this.scrollNavigationAreaDim / 2;
|
|
480
|
+
const scrollButtonOffsetX = selectedNavCoachIndex * this.coachNavButtonDim;
|
|
481
|
+
const scrollOffsetX = scrollButtonOffsetX - navigationAreaCenteredPosX + this.coachNavButtonDim;
|
|
482
|
+
this.navigationScrollArea.scrollTo({
|
|
483
|
+
top: this.alignVertical ? scrollOffsetX : 0,
|
|
484
|
+
left: this.alignVertical ? 0 : scrollOffsetX,
|
|
485
|
+
behavior: "smooth"
|
|
486
|
+
});
|
|
487
|
+
}
|
|
488
|
+
}, 10);
|
|
319
489
|
}
|
|
320
490
|
/**
|
|
321
491
|
* Sets the new ScrollDirection by the new given target coach index.
|
|
@@ -325,8 +495,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
325
495
|
}
|
|
326
496
|
/**
|
|
327
497
|
* 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
|
|
498
|
+
* In case the user is currently navigating through places by keyboard and goes to previous coach,
|
|
499
|
+
* then we return the end position of the coach to get the closest scroll position of the next focus place.
|
|
330
500
|
* @returns number
|
|
331
501
|
*/
|
|
332
502
|
_getCoachScrollPositionX() {
|
|
@@ -351,8 +521,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
351
521
|
* @returns boolean
|
|
352
522
|
*/
|
|
353
523
|
_isScrollableToSelectedCoach() {
|
|
354
|
-
const currScrollPosX = this.coachScrollArea.scrollLeft;
|
|
355
|
-
const coachScrollWindowWidth = this.coachScrollArea.getBoundingClientRect().width;
|
|
524
|
+
const currScrollPosX = this.alignVertical ? this.coachScrollArea.scrollTop : this.coachScrollArea.scrollLeft;
|
|
525
|
+
const coachScrollWindowWidth = this.alignVertical ? this.coachScrollArea.getBoundingClientRect().height : this.coachScrollArea.getBoundingClientRect().width;
|
|
356
526
|
const maxScrollWidthArea = this.maxCalcCoachsWidth - coachScrollWindowWidth;
|
|
357
527
|
const currCoachTrigger = this.triggerCoachPositionsCollection[this.currSelectedCoachIndex];
|
|
358
528
|
const isScrollPosSameToCurrCoachPos = currScrollPosX === this.triggerCoachPositionsCollection[this.currSelectedCoachIndex].start;
|
|
@@ -363,7 +533,8 @@ let SeatReservationBaseElement = (() => {
|
|
|
363
533
|
* @returns number
|
|
364
534
|
*/
|
|
365
535
|
_getCoachIndexByScrollTriggerPosition() {
|
|
366
|
-
const
|
|
536
|
+
const scrollPos = this.alignVertical ? this.coachScrollArea.scrollTop : this.coachScrollArea.scrollLeft;
|
|
537
|
+
const scrollOffsetX = scrollPos + this.scrollCoachsAreaWidth / 2;
|
|
367
538
|
return this.triggerCoachPositionsCollection.findIndex((coachTrigger) => scrollOffsetX >= coachTrigger.start && scrollOffsetX <= coachTrigger.end);
|
|
368
539
|
}
|
|
369
540
|
/**
|
|
@@ -372,14 +543,24 @@ let SeatReservationBaseElement = (() => {
|
|
|
372
543
|
*/
|
|
373
544
|
_getFirstPlaceInSelecedCoach() {
|
|
374
545
|
let firstPlace = null;
|
|
375
|
-
const coach = this.
|
|
376
|
-
const firstCellId = "cell-" + this.currSelectedCoachIndex + "-0-0";
|
|
377
|
-
const placeNumber = this.shadowRoot?.querySelector("
|
|
546
|
+
const coach = this.seatReservations[this.currSelectedDeckIndex].coachItems[this.currSelectedCoachIndex];
|
|
547
|
+
const firstCellId = "cell-" + this.currSelectedDeckIndex + "-" + this.currSelectedCoachIndex + "-0-0";
|
|
548
|
+
const placeNumber = this.shadowRoot?.querySelector("#" + firstCellId)?.querySelector("sbb-seat-reservation-place-control")?.getAttribute("text") || null;
|
|
378
549
|
if (coach && placeNumber) {
|
|
379
550
|
firstPlace = coach.places?.find((place) => place.number === placeNumber) || null;
|
|
380
551
|
}
|
|
381
552
|
return firstPlace;
|
|
382
553
|
}
|
|
554
|
+
_getSwitchedCoachDeckIndexByKeyNavigation(pressedKey, coach) {
|
|
555
|
+
if (!this.currSelectedPlace || pressedKey === this.keyboardNavigationEvents.ArrowRight || pressedKey === this.keyboardNavigationEvents.ArrowLeft)
|
|
556
|
+
return null;
|
|
557
|
+
if (pressedKey === this.keyboardNavigationEvents.ArrowDown && this.currSelectedPlace.position.y + 2 === coach.dimension.h && !!this.seatReservations[this.currSelectedDeckIndex + 1]) {
|
|
558
|
+
return this.currSelectedDeckIndex + 1;
|
|
559
|
+
} else if (pressedKey === this.keyboardNavigationEvents.ArrowUp && this.currSelectedPlace.position.y === 0 && !!this.seatReservations[this.currSelectedDeckIndex - 1]) {
|
|
560
|
+
return this.currSelectedDeckIndex - 1;
|
|
561
|
+
}
|
|
562
|
+
return null;
|
|
563
|
+
}
|
|
383
564
|
/**
|
|
384
565
|
* To get the correct closest place of current pressed key and the current selected place,
|
|
385
566
|
* we have to investigate the coordinates of each place to find the closest place of the currSelectedPlaceElementId.
|
|
@@ -387,88 +568,52 @@ let SeatReservationBaseElement = (() => {
|
|
|
387
568
|
* @returns Place or null
|
|
388
569
|
*/
|
|
389
570
|
_getClosestPlaceByKeyDirection(pressedKey) {
|
|
390
|
-
const coach = this.
|
|
571
|
+
const coach = this.seatReservations[this.currSelectedDeckIndex].coachItems[this.currSelectedCoachIndex];
|
|
391
572
|
let closestPlace = null;
|
|
392
|
-
|
|
573
|
+
let coachPlaces = coach.places;
|
|
574
|
+
let switchedCoachDeckIndex = null;
|
|
575
|
+
if (coachPlaces) {
|
|
393
576
|
if (!this.currSelectedPlaceElementId) {
|
|
394
577
|
return this._getFirstPlaceInSelecedCoach();
|
|
395
578
|
} else {
|
|
396
579
|
if (this.currSelectedPlace) {
|
|
397
|
-
|
|
580
|
+
const currSelectedPlacePosition = { ...this.currSelectedPlace.position };
|
|
581
|
+
if (pressedKey) {
|
|
582
|
+
switchedCoachDeckIndex = this._getSwitchedCoachDeckIndexByKeyNavigation(pressedKey, coach);
|
|
583
|
+
if (switchedCoachDeckIndex !== null) {
|
|
584
|
+
coachPlaces = this.seatReservations[switchedCoachDeckIndex].coachItems[this.currSelectedCoachIndex].places || [];
|
|
585
|
+
currSelectedPlacePosition.y = switchedCoachDeckIndex < this.currSelectedDeckIndex ? coach.dimension.h : -1;
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
for (const place of coachPlaces) {
|
|
398
589
|
if (!pressedKey) {
|
|
399
|
-
if (this.currScrollDirection === ScrollDirection.right && place.position.y ===
|
|
590
|
+
if (this.currScrollDirection === ScrollDirection.right && place.position.y === currSelectedPlacePosition.y && (!closestPlace || place.position.x < closestPlace.position.x)) {
|
|
400
591
|
closestPlace = place;
|
|
401
|
-
} else if (this.currScrollDirection === ScrollDirection.left && place.position.y ===
|
|
592
|
+
} else if (this.currScrollDirection === ScrollDirection.left && place.position.y === currSelectedPlacePosition.y && (!closestPlace || place.position.x > closestPlace.position.x)) {
|
|
402
593
|
closestPlace = place;
|
|
403
594
|
}
|
|
404
595
|
} else {
|
|
405
596
|
if (place.number !== this.currSelectedPlace?.number) {
|
|
406
|
-
if (pressedKey === this.keyboardNavigationEvents.ArrowRight && (place.position.y ===
|
|
597
|
+
if (pressedKey === this.keyboardNavigationEvents.ArrowRight && (place.position.y === currSelectedPlacePosition.y || place.position.y === currSelectedPlacePosition.y - 1) && place.position.x > currSelectedPlacePosition.x && (!closestPlace || place.position.x < closestPlace.position.x)) {
|
|
407
598
|
closestPlace = place;
|
|
408
|
-
} else if (pressedKey === this.keyboardNavigationEvents.ArrowDown && (place.position.x ===
|
|
599
|
+
} else if (pressedKey === this.keyboardNavigationEvents.ArrowDown && (place.position.x === currSelectedPlacePosition.x || place.position.x === currSelectedPlacePosition.x + 1) && place.position.y > currSelectedPlacePosition.y && (!closestPlace || place.position.y < closestPlace.position.y)) {
|
|
409
600
|
closestPlace = place;
|
|
410
|
-
} else if (pressedKey === this.keyboardNavigationEvents.ArrowLeft && (place.position.y ===
|
|
601
|
+
} else if (pressedKey === this.keyboardNavigationEvents.ArrowLeft && (place.position.y === currSelectedPlacePosition.y || place.position.y === currSelectedPlacePosition.y + 1) && place.position.x < currSelectedPlacePosition.x && (!closestPlace || place.position.x > closestPlace.position.x)) {
|
|
411
602
|
closestPlace = place;
|
|
412
|
-
} else if (pressedKey === this.keyboardNavigationEvents.ArrowUp && (place.position.x ===
|
|
603
|
+
} else if (pressedKey === this.keyboardNavigationEvents.ArrowUp && (place.position.x === currSelectedPlacePosition.x || place.position.x === currSelectedPlacePosition.x - 1) && place.position.y < currSelectedPlacePosition.y && (!closestPlace || place.position.y > closestPlace.position.y)) {
|
|
413
604
|
closestPlace = place;
|
|
414
605
|
}
|
|
415
606
|
}
|
|
416
607
|
}
|
|
417
608
|
}
|
|
609
|
+
if (switchedCoachDeckIndex !== null && closestPlace) {
|
|
610
|
+
this.currSelectedDeckIndex = switchedCoachDeckIndex;
|
|
611
|
+
}
|
|
418
612
|
}
|
|
419
613
|
}
|
|
420
614
|
}
|
|
421
615
|
return closestPlace;
|
|
422
616
|
}
|
|
423
|
-
focusPlaceElement(place, coachIndex) {
|
|
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
617
|
// Handling for Tab navigation if an place is selected inside the coach.
|
|
473
618
|
// This controls the focused coach from the current selected coach.
|
|
474
619
|
_navigateCoachNavigationByKeyboard(tabDirection) {
|
|
@@ -476,8 +621,6 @@ let SeatReservationBaseElement = (() => {
|
|
|
476
621
|
const newFocusableIndex = tabDirection === "NEXT_TAB" ? this.getNextAvailableCoachIndex(currFocusIndex) : this.getPrevAvailableCoachIndex(currFocusIndex);
|
|
477
622
|
if (currFocusIndex === newFocusableIndex) {
|
|
478
623
|
this.unfocusPlaceElement();
|
|
479
|
-
this.selectedCoachIndex = -1;
|
|
480
|
-
this.currSelectedCoachIndex = -1;
|
|
481
624
|
this.seatReservationWithoutNavigationHasFocus = false;
|
|
482
625
|
if (tabDirection === "NEXT_TAB")
|
|
483
626
|
this.lastTabElement.focus();
|
|
@@ -492,6 +635,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
492
635
|
if (placeInCoachHasFocus) {
|
|
493
636
|
this.focusedCoachIndex = currFocusIndex;
|
|
494
637
|
this.unfocusPlaceElement();
|
|
638
|
+
return;
|
|
495
639
|
} else {
|
|
496
640
|
this.focusedCoachIndex = newFocusableIndex;
|
|
497
641
|
}
|
|
@@ -507,6 +651,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
507
651
|
this._setFocusToSelectedCoachGrid();
|
|
508
652
|
}
|
|
509
653
|
}
|
|
654
|
+
this._scrollToSelectedNavigationButton(newFocusableIndex);
|
|
510
655
|
} else {
|
|
511
656
|
this.scrollToSelectedNavCoach(newFocusableIndex);
|
|
512
657
|
}
|
|
@@ -531,33 +676,42 @@ let SeatReservationBaseElement = (() => {
|
|
|
531
676
|
}
|
|
532
677
|
getNextAvailableCoachIndex(currentIndex) {
|
|
533
678
|
const startIndex = currentIndex ?? this.currSelectedCoachIndex;
|
|
534
|
-
return startIndex < this.
|
|
679
|
+
return startIndex < this.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1 ? startIndex + 1 : startIndex;
|
|
535
680
|
}
|
|
536
681
|
getPrevAvailableCoachIndex(currentIndex) {
|
|
537
682
|
const startIndex = currentIndex ?? this.currSelectedCoachIndex;
|
|
538
683
|
return startIndex > 0 ? startIndex - 1 : startIndex;
|
|
539
684
|
}
|
|
540
685
|
updateSelectedSeatReservationPlaces(placeSelection) {
|
|
686
|
+
const placeTypeProp = placeSelection.placeType === "SEAT" ? "seats" : "bicycles";
|
|
687
|
+
const maxReservations = placeSelection.placeType === "SEAT" ? this.maxSeatReservations : this.maxBicycleReservations;
|
|
688
|
+
const currSelectedPlaces = this.selectedSeatReservationPlaces[placeTypeProp];
|
|
689
|
+
const updatedSelectedPlaces = this._updateSelectedSeatReservationPlaces(currSelectedPlaces, maxReservations, placeSelection);
|
|
690
|
+
this.selectedSeatReservationPlaces[placeTypeProp] = updatedSelectedPlaces;
|
|
691
|
+
this.dispatchEvent(new CustomEvent("selectedplaces", {
|
|
692
|
+
bubbles: true,
|
|
693
|
+
composed: true,
|
|
694
|
+
detail: this.selectedSeatReservationPlaces
|
|
695
|
+
}));
|
|
696
|
+
}
|
|
697
|
+
_updateSelectedSeatReservationPlaces(selectedSeatReservationPlaces, maxReservations, placeSelection) {
|
|
541
698
|
if (placeSelection.state === "SELECTED") {
|
|
542
699
|
const seatReservationSelection = this._getSeatReservationPlaceSelection(placeSelection);
|
|
543
700
|
if (seatReservationSelection) {
|
|
544
|
-
|
|
701
|
+
selectedSeatReservationPlaces.push(seatReservationSelection);
|
|
545
702
|
}
|
|
546
703
|
} else {
|
|
547
|
-
|
|
704
|
+
selectedSeatReservationPlaces = selectedSeatReservationPlaces.filter((_selectedPlace) => _selectedPlace.id !== placeSelection.id);
|
|
548
705
|
}
|
|
549
|
-
if (
|
|
550
|
-
|
|
706
|
+
if (maxReservations > -1 && selectedSeatReservationPlaces.length > maxReservations) {
|
|
707
|
+
const resetWithPlaceSelection = maxReservations > 0 ? placeSelection : void 0;
|
|
708
|
+
selectedSeatReservationPlaces = this._resetAllPlaceSelections(selectedSeatReservationPlaces, resetWithPlaceSelection);
|
|
551
709
|
}
|
|
552
|
-
|
|
553
|
-
bubbles: true,
|
|
554
|
-
composed: true,
|
|
555
|
-
detail: this.selectedSeatReservationPlaces
|
|
556
|
-
}));
|
|
710
|
+
return selectedSeatReservationPlaces;
|
|
557
711
|
}
|
|
558
712
|
updateCurrentSelectedPlaceInCoach(placeSelection) {
|
|
559
713
|
const coachIndex = placeSelection.coachIndex;
|
|
560
|
-
const place = this.
|
|
714
|
+
const place = this.seatReservations[this.currSelectedDeckIndex].coachItems[coachIndex].places?.find((place2) => place2.number == placeSelection.number);
|
|
561
715
|
if (!place)
|
|
562
716
|
return;
|
|
563
717
|
this.currSelectedCoachIndex = coachIndex;
|
|
@@ -570,7 +724,7 @@ let SeatReservationBaseElement = (() => {
|
|
|
570
724
|
updateCurrentSelectedCoach() {
|
|
571
725
|
this.selectedCoachIndex = this.currSelectedCoachIndex;
|
|
572
726
|
this.focusedCoachIndex = -1;
|
|
573
|
-
const coachSelection = this.
|
|
727
|
+
const coachSelection = this._getSeatReservationSelectedCoach(this.selectedCoachIndex);
|
|
574
728
|
if (coachSelection) {
|
|
575
729
|
this.dispatchEvent(new CustomEvent("selectedcoach", {
|
|
576
730
|
bubbles: true,
|
|
@@ -584,86 +738,103 @@ let SeatReservationBaseElement = (() => {
|
|
|
584
738
|
* that have the state SELECTED within the seatReservation object
|
|
585
739
|
*/
|
|
586
740
|
_initSeatReservationPlaceSelection() {
|
|
587
|
-
this.seatReservation.coachItems.map((coach, coachIndex) => {
|
|
741
|
+
this.seatReservations?.forEach((seatReservation) => seatReservation.coachItems.map((coach, coachIndex) => {
|
|
588
742
|
coach.places?.filter((place) => place.state === "SELECTED")?.forEach((place) => {
|
|
589
|
-
const preselectedPlaceSelection = mapPlaceInfosToPlaceSelection(place, coachIndex);
|
|
743
|
+
const preselectedPlaceSelection = mapPlaceInfosToPlaceSelection(place, seatReservation.deckCoachIndex, coachIndex);
|
|
590
744
|
const seatReservationPlaceSelection = this._getSeatReservationPlaceSelection(preselectedPlaceSelection);
|
|
591
|
-
if (seatReservationPlaceSelection)
|
|
592
|
-
|
|
745
|
+
if (seatReservationPlaceSelection) {
|
|
746
|
+
if (seatReservationPlaceSelection.placeType === "SEAT") {
|
|
747
|
+
this.selectedSeatReservationPlaces.seats.push(seatReservationPlaceSelection);
|
|
748
|
+
} else {
|
|
749
|
+
this.selectedSeatReservationPlaces.bicycles.push(seatReservationPlaceSelection);
|
|
750
|
+
}
|
|
751
|
+
}
|
|
593
752
|
});
|
|
594
|
-
});
|
|
753
|
+
}));
|
|
595
754
|
}
|
|
596
755
|
/**
|
|
597
756
|
* All selected places will be reset or the currentSelectedPlace was given, then we reset all except currentSelectedPlace
|
|
757
|
+
* @param reservationPlaceSelections
|
|
598
758
|
* @param currSelectedPlace
|
|
599
759
|
*/
|
|
600
|
-
_resetAllPlaceSelections(currSelectedPlace) {
|
|
601
|
-
for (const placeSelection of
|
|
760
|
+
_resetAllPlaceSelections(reservationPlaceSelections, currSelectedPlace) {
|
|
761
|
+
for (const placeSelection of reservationPlaceSelections) {
|
|
602
762
|
if (!currSelectedPlace || currSelectedPlace.id !== placeSelection.id) {
|
|
603
763
|
const placeElement = this.shadowRoot?.getElementById(placeSelection.id);
|
|
604
|
-
placeElement
|
|
764
|
+
if (placeElement) {
|
|
765
|
+
placeElement.setAttribute("state", "FREE");
|
|
766
|
+
}
|
|
605
767
|
}
|
|
606
768
|
}
|
|
607
769
|
if (currSelectedPlace) {
|
|
608
|
-
|
|
770
|
+
reservationPlaceSelections = reservationPlaceSelections.filter((_selectedPlace) => _selectedPlace.id === currSelectedPlace.id);
|
|
609
771
|
} else {
|
|
610
|
-
|
|
772
|
+
reservationPlaceSelections = [];
|
|
611
773
|
}
|
|
774
|
+
return reservationPlaceSelections;
|
|
612
775
|
}
|
|
613
776
|
_getSeatReservationPlaceSelection(currSelectedPlace) {
|
|
614
|
-
const coach = this.
|
|
777
|
+
const coach = this.seatReservations[this.currSelectedDeckIndex].coachItems[currSelectedPlace.coachIndex];
|
|
615
778
|
const place = coach.places?.find((place2) => place2.number === currSelectedPlace.number);
|
|
616
|
-
return place ? mapPlaceAndCoachToSeatReservationPlaceSelection(place, coach, currSelectedPlace.coachIndex) : null;
|
|
779
|
+
return place ? mapPlaceAndCoachToSeatReservationPlaceSelection(place, coach, currSelectedPlace.deckIndex, currSelectedPlace.coachIndex) : null;
|
|
617
780
|
}
|
|
618
|
-
|
|
619
|
-
if (!this.
|
|
781
|
+
_getSeatReservationSelectedCoach(coachIndex) {
|
|
782
|
+
if (!this.seatReservations[this.currSelectedDeckIndex].coachItems[coachIndex])
|
|
620
783
|
return null;
|
|
621
|
-
const coach = this.
|
|
622
|
-
|
|
784
|
+
const coach = this.seatReservations[this.currSelectedDeckIndex].coachItems[coachIndex];
|
|
785
|
+
const coachNumberOfFreePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);
|
|
786
|
+
return mapCoachInfosToCoachSelection(coachIndex, coach, coachNumberOfFreePlaces);
|
|
623
787
|
}
|
|
624
788
|
_setCurrSelectedPlaceElementId(place) {
|
|
625
789
|
if (place) {
|
|
626
|
-
this.currSelectedPlaceElementId = "seat-reservation__place-button-" + this.currSelectedCoachIndex + "-" + place.number;
|
|
790
|
+
this.currSelectedPlaceElementId = "seat-reservation__place-button-" + this.currSelectedDeckIndex + "-" + this.currSelectedCoachIndex + "-" + place.number;
|
|
627
791
|
} else {
|
|
628
792
|
this.currSelectedPlaceElementId = null;
|
|
629
793
|
}
|
|
630
794
|
}
|
|
631
795
|
/**
|
|
632
|
-
*
|
|
633
|
-
*
|
|
634
|
-
|
|
635
|
-
|
|
796
|
+
* Prepares all coaches with the values for whether there is a driver area left or right
|
|
797
|
+
* */
|
|
798
|
+
_prepairCoachDriverArea() {
|
|
799
|
+
this.seatReservations.forEach((seatReservation, index) => {
|
|
800
|
+
this.seatReservations[index].coachItems = seatReservation.coachItems.map((coacItem) => {
|
|
801
|
+
const coachDriverAreas = coacItem.graphicElements?.filter((graphicalElements) => graphicalElements.icon === "DRIVER_AREA");
|
|
802
|
+
if (coachDriverAreas && coachDriverAreas.length > 0) {
|
|
803
|
+
const hasLeftDriverArea = coachDriverAreas.find((driverAreaElement) => driverAreaElement.position.x === 0) || false;
|
|
804
|
+
const hasRightDriverArea = coachDriverAreas.find((driverAreaElement) => driverAreaElement.position.x > 0) || false;
|
|
805
|
+
coacItem.driverAreaSide = {
|
|
806
|
+
left: !!hasLeftDriverArea,
|
|
807
|
+
right: !!hasRightDriverArea
|
|
808
|
+
};
|
|
809
|
+
}
|
|
810
|
+
return coacItem;
|
|
811
|
+
});
|
|
812
|
+
});
|
|
813
|
+
}
|
|
814
|
+
/**
|
|
815
|
+
* Returns the current selected place HTML element by currSelectedPlaceElementId.
|
|
636
816
|
* @returns HTMLElement or null
|
|
637
817
|
*/
|
|
638
|
-
_getPlaceHtmlElement(
|
|
639
|
-
|
|
640
|
-
const coachPlaceNumberId = placeNumber ? "seat-reservation__place-button-" + currCoachIndex + "-" + placeNumber : this.currSelectedPlaceElementId;
|
|
641
|
-
return coachPlaceNumberId ? this.shadowRoot?.getElementById(coachPlaceNumberId) || null : null;
|
|
642
|
-
}
|
|
643
|
-
//Set the vertical offset
|
|
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
|
-
});
|
|
818
|
+
_getPlaceHtmlElement() {
|
|
819
|
+
return this.currSelectedPlaceElementId ? this.shadowRoot?.getElementById(this.currSelectedPlaceElementId) || null : null;
|
|
652
820
|
}
|
|
653
|
-
},
|
|
821
|
+
}, _seatReservations_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
822
|
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
|
|
655
|
-
|
|
823
|
+
_seatReservations_decorators = [property({ attribute: "seat-reservations", type: Array })];
|
|
656
824
|
_hasNavigation_decorators = [forceType(), property({ attribute: "has-navigation", type: Boolean })];
|
|
657
825
|
_alignVertical_decorators = [forceType(), property({ attribute: "align-vertical", type: Boolean })];
|
|
658
826
|
_baseGridSize_decorators = [forceType(), property({ attribute: "base-grid-size", type: Number })];
|
|
659
827
|
_height_decorators = [forceType(), property({ attribute: "height", type: Number })];
|
|
660
|
-
|
|
828
|
+
_maxSeatReservations_decorators = [forceType(), property({ attribute: "max-seat-reservations", type: Number })];
|
|
829
|
+
_maxBicycleReservations_decorators = [forceType(), property({ attribute: "max-bicycle-reservations", type: Number })];
|
|
661
830
|
_preventPlaceClick_decorators = [forceType(), property({ attribute: "prevent-place-click", type: Boolean })];
|
|
831
|
+
_preselectCoachIndex_decorators = [forceType(), property({ attribute: "preselect-coach-index", type: Number })];
|
|
662
832
|
_selectedCoachIndex_decorators = [state()];
|
|
663
833
|
_focusedCoachIndex_decorators = [state()];
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
834
|
+
_hoveredScrollCoachIndex_decorators = [state()];
|
|
835
|
+
__esDecorate(_a, null, _seatReservations_decorators, { kind: "accessor", name: "seatReservations", static: false, private: false, access: { has: (obj) => "seatReservations" in obj, get: (obj) => obj.seatReservations, set: (obj, value) => {
|
|
836
|
+
obj.seatReservations = value;
|
|
837
|
+
} }, metadata: _metadata }, _seatReservations_initializers, _seatReservations_extraInitializers);
|
|
667
838
|
__esDecorate(_a, null, _hasNavigation_decorators, { kind: "accessor", name: "hasNavigation", static: false, private: false, access: { has: (obj) => "hasNavigation" in obj, get: (obj) => obj.hasNavigation, set: (obj, value) => {
|
|
668
839
|
obj.hasNavigation = value;
|
|
669
840
|
} }, metadata: _metadata }, _hasNavigation_initializers, _hasNavigation_extraInitializers);
|
|
@@ -676,18 +847,27 @@ let SeatReservationBaseElement = (() => {
|
|
|
676
847
|
__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
848
|
obj.height = value;
|
|
678
849
|
} }, metadata: _metadata }, _height_initializers, _height_extraInitializers);
|
|
679
|
-
__esDecorate(_a, null,
|
|
680
|
-
obj.
|
|
681
|
-
} }, metadata: _metadata },
|
|
850
|
+
__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) => {
|
|
851
|
+
obj.maxSeatReservations = value;
|
|
852
|
+
} }, metadata: _metadata }, _maxSeatReservations_initializers, _maxSeatReservations_extraInitializers);
|
|
853
|
+
__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) => {
|
|
854
|
+
obj.maxBicycleReservations = value;
|
|
855
|
+
} }, metadata: _metadata }, _maxBicycleReservations_initializers, _maxBicycleReservations_extraInitializers);
|
|
682
856
|
__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
857
|
obj.preventPlaceClick = value;
|
|
684
858
|
} }, metadata: _metadata }, _preventPlaceClick_initializers, _preventPlaceClick_extraInitializers);
|
|
859
|
+
__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) => {
|
|
860
|
+
obj.preselectCoachIndex = value;
|
|
861
|
+
} }, metadata: _metadata }, _preselectCoachIndex_initializers, _preselectCoachIndex_extraInitializers);
|
|
685
862
|
__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
863
|
obj.selectedCoachIndex = value;
|
|
687
864
|
} }, metadata: _metadata }, _selectedCoachIndex_initializers, _selectedCoachIndex_extraInitializers);
|
|
688
865
|
__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
866
|
obj.focusedCoachIndex = value;
|
|
690
867
|
} }, metadata: _metadata }, _focusedCoachIndex_initializers, _focusedCoachIndex_extraInitializers);
|
|
868
|
+
__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) => {
|
|
869
|
+
obj.hoveredScrollCoachIndex = value;
|
|
870
|
+
} }, metadata: _metadata }, _hoveredScrollCoachIndex_initializers, _hoveredScrollCoachIndex_extraInitializers);
|
|
691
871
|
if (_metadata) Object.defineProperty(_a, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
|
|
692
872
|
})(), _a.events = {
|
|
693
873
|
selectedplaces: "selectedplaces",
|
|
@@ -697,4 +877,4 @@ let SeatReservationBaseElement = (() => {
|
|
|
697
877
|
export {
|
|
698
878
|
SeatReservationBaseElement
|
|
699
879
|
};
|
|
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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"}
|
|
880
|
+
//# 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  SeatReservationSelectedCoach,\n  SeatReservationSelectedPlaces,\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 class SeatReservationBaseElement extends LitElement {\n  public static readonly events = {\n    selectedplaces: 'selectedplaces',\n    selectedcoach: 'selectedcoach',\n  } as const;\n\n  /** The seat reservations array contains all coaches and places */\n  @property({ attribute: 'seat-reservations', type: Array })\n  public accessor seatReservations: 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 gap between the coaches decks by multiple deck visualization\n  protected gapBetweenCoachDecks = 48;\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 currSelectedDeckIndex: number = 0;\n  protected preventCoachScrollByPlaceClick: boolean = false;\n  protected selectedSeatReservationPlaces: SeatReservationSelectedPlaces = {\n    seats: [],\n    bicycles: [],\n  };\n  protected seatReservationWithoutNavigationHasFocus = false;\n  protected isCoachGridFocusable = false;\n  protected isAutoScrolling = false;\n  protected isKeyboardNavigation = false;\n  protected hasMultipleDecks = 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('seatReservations')) {\n      this.hasMultipleDecks = this.seatReservations?.length > 1;\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      const seatReservationLayer = this.seatReservations[this.currSelectedDeckIndex] || null;\n      if (seatReservationLayer?.coachItems.length) {\n        this.baseGridSize = this.height / seatReservationLayer.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.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1\n    ) {\n      navigateToCoachIndex =\n        this.currSelectedCoachIndex != -1 ? this.currSelectedCoachIndex + 1 : 0;\n    }\n\n    this.scrollToSelectedNavCoach(navigateToCoachIndex);\n  }\n\n  /**\n   * Data can be prepared once for the entire component\n   * in order to avoid recurring iteration processes in rendering.\n   */\n  protected initPrepairSeatReservationData(): void {\n    this._prepairCoachDriverArea();\n  }\n\n  /* Init scroll event handling for coach navigation */\n  protected initNavigationSelectionByScrollEvent(): void {\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(\n      '#sbb-sr__wrapper-scrollarea',\n    ) as HTMLElement;\n    this.navigationScrollArea = this.shadowRoot?.querySelector('#sbb-sr-navigation') as HTMLElement;\n\n    const seatReservationDeck = this.seatReservations\n      ? this.seatReservations[this.currSelectedDeckIndex]\n      : null;\n    if (seatReservationDeck && seatReservationDeck.coachItems.length > 0) {\n      // Calculates the absolute height of one coach including border padding\n      const coachHeightWithBorderPadding =\n        seatReservationDeck.coachItems[0].dimension.h * this.baseGridSize + this.coachBorderPadding;\n\n      // Calculate each gap between multiple coach the decks.\n      // No gap exist if just one coach deck exist.\n      const gapBetweenCoachDecks = (this.seatReservations.length - 1) * this.gapBetweenCoachDecks;\n\n      this.style?.setProperty(\n        '--sbb-seat-reservation-height',\n        `${coachHeightWithBorderPadding * this.seatReservations.length + gapBetweenCoachDecks}`,\n      );\n      this.style?.setProperty('--sbb-seat-reservation-decks', `${this.seatReservations.length}`);\n    }\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      if (navCoacheList) {\n        const firstLiEleDimension = navCoacheList?.getBoundingClientRect();\n        this.coachNavButtonDim = this.alignVertical\n          ? firstLiEleDimension.height\n          : firstLiEleDimension.width;\n      }\n    }\n\n    if (this.coachScrollArea && seatReservationDeck) {\n      // Init the start offset for the calculation of the coach scroll trigger positions\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 from coach width\n      this.triggerCoachPositionsCollection = seatReservationDeck.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): void {\n    this.unfocusPlaceElement();\n    if (place) {\n      this.currSelectedPlace = place;\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.seatReservations[this.currSelectedDeckIndex].coachItems[\n      coachIndex\n    ].places?.reduce((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    }, accumulator);\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    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 =\n      this.seatReservations[this.currSelectedDeckIndex].coachItems[this.currSelectedCoachIndex];\n    const firstCellId =\n      'cell-' + this.currSelectedDeckIndex + '-' + 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  private _getSwitchedCoachDeckIndexByKeyNavigation(\n    pressedKey: string,\n    coach: CoachItem,\n  ): number | null {\n    if (\n      !this.currSelectedPlace ||\n      pressedKey === this.keyboardNavigationEvents.ArrowRight ||\n      pressedKey === this.keyboardNavigationEvents.ArrowLeft\n    )\n      return null;\n\n    //CHECK DECK SWITCH DOWN\n    if (\n      pressedKey === this.keyboardNavigationEvents.ArrowDown &&\n      this.currSelectedPlace.position.y + 2 === coach.dimension.h &&\n      !!this.seatReservations[this.currSelectedDeckIndex + 1]\n    ) {\n      return this.currSelectedDeckIndex + 1;\n    }\n    //CHECK DECK SWITCH UP\n    else if (\n      pressedKey === this.keyboardNavigationEvents.ArrowUp &&\n      this.currSelectedPlace.position.y === 0 &&\n      !!this.seatReservations[this.currSelectedDeckIndex - 1]\n    ) {\n      return this.currSelectedDeckIndex - 1;\n    }\n    return null;\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 =\n      this.seatReservations[this.currSelectedDeckIndex].coachItems[this.currSelectedCoachIndex];\n    let closestPlace = null;\n    let coachPlaces = coach.places;\n    let switchedCoachDeckIndex = null;\n\n    if (coachPlaces) {\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          const currSelectedPlacePosition = { ...this.currSelectedPlace.position };\n\n          if (pressedKey) {\n            // Check whether the keyboard nagivation is used to switch to another coach deck (Up -> Down | Down -> Up).\n            // In this case, we takes the coach places we have to checked from the target deck now.\n            switchedCoachDeckIndex = this._getSwitchedCoachDeckIndexByKeyNavigation(\n              pressedKey,\n              coach,\n            );\n            if (switchedCoachDeckIndex !== null) {\n              //Assign check places from the target switch deck\n              coachPlaces =\n                this.seatReservations[switchedCoachDeckIndex].coachItems[\n                  this.currSelectedCoachIndex\n                ].places || [];\n              //We have to update y coorodinate\n              currSelectedPlacePosition.y =\n                switchedCoachDeckIndex < this.currSelectedDeckIndex ? coach.dimension.h : -1;\n            }\n          }\n\n          for (const place of coachPlaces) {\n            // If no 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 === currSelectedPlacePosition.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 === currSelectedPlacePosition.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 === currSelectedPlacePosition.y ||\n                    place.position.y === currSelectedPlacePosition.y - 1) &&\n                  place.position.x > currSelectedPlacePosition.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 === currSelectedPlacePosition.x ||\n                    place.position.x === currSelectedPlacePosition.x + 1) &&\n                  place.position.y > currSelectedPlacePosition.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 === currSelectedPlacePosition.y ||\n                    place.position.y === currSelectedPlacePosition.y + 1) &&\n                  place.position.x < currSelectedPlacePosition.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 === currSelectedPlacePosition.x ||\n                    place.position.x === currSelectedPlacePosition.x - 1) &&\n                  place.position.y < currSelectedPlacePosition.y &&\n                  (!closestPlace || place.position.y > closestPlace.position.y)\n                ) {\n                  closestPlace = place;\n                }\n              }\n            }\n          }\n\n          // If a place has been found in the changing switch coach,\n          // we can update the currSelectedDeckIndex\n          if (switchedCoachDeckIndex !== null && closestPlace) {\n            this.currSelectedDeckIndex = switchedCoachDeckIndex;\n          }\n        }\n      }\n    }\n    return closestPlace;\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.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\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.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1\n      ? startIndex + 1\n      : 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    const placeTypeProp = placeSelection.placeType === 'SEAT' ? 'seats' : 'bicycles';\n    const maxReservations =\n      placeSelection.placeType === 'SEAT' ? this.maxSeatReservations : this.maxBicycleReservations;\n    const currSelectedPlaces = this.selectedSeatReservationPlaces[placeTypeProp];\n    const updatedSelectedPlaces = this._updateSelectedSeatReservationPlaces(\n      currSelectedPlaces,\n      maxReservations,\n      placeSelection,\n    );\n\n    this.selectedSeatReservationPlaces[placeTypeProp] = updatedSelectedPlaces;\n\n    /**\n     * @type {CustomEvent<SeatReservationSelectedPlaces>}\n     * Emits when a place was selected and returns a Place array with all selected places.\n     */\n    this.dispatchEvent(\n      new CustomEvent<SeatReservationSelectedPlaces>('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    // Checks whether maxReservation is activated and the maximum number of selected places is reached\n    if (maxReservations > -1 && selectedSeatReservationPlaces.length > maxReservations) {\n      const resetWithPlaceSelection = maxReservations > 0 ? placeSelection : undefined;\n\n      selectedSeatReservationPlaces = this._resetAllPlaceSelections(\n        selectedSeatReservationPlaces,\n        resetWithPlaceSelection,\n      );\n    }\n\n    return selectedSeatReservationPlaces;\n  }\n\n  protected updateCurrentSelectedPlaceInCoach(placeSelection: PlaceSelection): void {\n    const coachIndex = placeSelection.coachIndex;\n    const place = this.seatReservations[this.currSelectedDeckIndex].coachItems[\n      coachIndex\n    ].places?.find((place) => place.number == placeSelection.number);\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._getSeatReservationSelectedCoach(this.selectedCoachIndex);\n    if (coachSelection) {\n      /**\n       * @type {CustomEvent<SeatReservationSelectedCoach>}\n       * Emits when a coach was selected and returns a CoachSelection\n       */\n      this.dispatchEvent(\n        new CustomEvent<SeatReservationSelectedCoach>('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.seatReservations?.forEach((seatReservation) =>\n      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              seatReservation.deckCoachIndex,\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  /**\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        if (placeElement) {\n          placeElement.setAttribute('state', 'FREE');\n        }\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 =\n      this.seatReservations[this.currSelectedDeckIndex].coachItems[currSelectedPlace.coachIndex];\n    const place = coach.places?.find((place) => place.number === currSelectedPlace.number);\n\n    return place\n      ? mapPlaceAndCoachToSeatReservationPlaceSelection(\n          place,\n          coach,\n          currSelectedPlace.deckIndex,\n          currSelectedPlace.coachIndex,\n        )\n      : null;\n  }\n\n  private _getSeatReservationSelectedCoach(\n    coachIndex: number,\n  ): SeatReservationSelectedCoach | null {\n    if (!this.seatReservations[this.currSelectedDeckIndex].coachItems[coachIndex]) return null;\n\n    const coach = this.seatReservations[this.currSelectedDeckIndex].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-' +\n        this.currSelectedDeckIndex +\n        '-' +\n        this.currSelectedCoachIndex +\n        '-' +\n        place.number;\n    } else {\n      this.currSelectedPlaceElementId = null;\n    }\n  }\n\n  /**\n   * Prepares all coaches with the values for whether there is a driver area left or right\n   * */\n  private _prepairCoachDriverArea(): void {\n    this.seatReservations.forEach((seatReservation, index) => {\n      this.seatReservations[index].coachItems = seatReservation.coachItems.map((coacItem) => {\n        const coachDriverAreas = coacItem.graphicElements?.filter(\n          (graphicalElements) => graphicalElements.icon === 'DRIVER_AREA',\n        );\n\n        if (coachDriverAreas && coachDriverAreas.length > 0) {\n          // Checking the driver area position whether it is present on the left or right side in a coach\n          const hasLeftDriverArea =\n            coachDriverAreas.find((driverAreaElement) => driverAreaElement.position.x === 0) ||\n            false;\n          const hasRightDriverArea =\n            coachDriverAreas.find((driverAreaElement) => driverAreaElement.position.x > 0) || false;\n\n          coacItem.driverAreaSide = {\n            left: !!hasLeftDriverArea,\n            right: !!hasRightDriverArea,\n          };\n        }\n        return coacItem;\n      });\n    });\n  }\n\n  /**\n   * Returns the current selected place HTML element by currSelectedPlaceElementId.\n   * @returns HTMLElement or null\n   */\n  private _getPlaceHtmlElement(): HTMLElement | null {\n    return this.currSelectedPlaceElementId\n      ? this.shadowRoot?.getElementById(this.currSelectedPlaceElementId) || null\n      : null;\n  }\n}\n"],"names":["ScrollDirection","accumulator","place"],"mappings":"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"}
|