@sbb-esta/lyne-elements-experimental 3.2.0 → 3.3.2
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 +341 -125
- package/development/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/icon-mapper.js +4 -3
- package/development/seat-reservation/common/mapper/mapper.d.ts +4 -4
- package/development/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/mapper.js +9 -8
- 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 +20373 -8
- package/development/seat-reservation/common/translations/i18n.d.ts.map +1 -1
- package/development/seat-reservation/common/translations/i18n.js +35 -4
- package/development/seat-reservation/common/types.d.ts +10 -3
- package/development/seat-reservation/common/types.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +20 -14
- 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 +129 -79
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts +11 -10
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation.component.js +208 -132
- package/development/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +1 -1
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +4 -4
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +28 -28
- 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 +22 -5
- 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 -2
- package/seat-reservation/common/mapper/mapper.d.ts +4 -4
- package/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/mapper.js +29 -28
- 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 +20375 -11
- package/seat-reservation/common/translations/i18n.d.ts.map +1 -1
- package/seat-reservation/common/translations/i18n.js +34 -3
- package/seat-reservation/common/types.d.ts +10 -3
- package/seat-reservation/common/types.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +20 -14
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.js +191 -146
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts +11 -10
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation.component.js +211 -188
- package/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +1 -1
- package/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +1 -1
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +74 -74
- 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 +64 -53
|
@@ -22,7 +22,8 @@ const style = css`*,
|
|
|
22
22
|
|
|
23
23
|
:host {
|
|
24
24
|
--sbb-seat-reservation-grid-size: 16px;
|
|
25
|
-
--sbb-seat-reservation-
|
|
25
|
+
--sbb-seat-reservation-height: 0;
|
|
26
|
+
--sbb-seat-reservation-decks: 1;
|
|
26
27
|
display: block;
|
|
27
28
|
height: inherit;
|
|
28
29
|
}
|
|
@@ -95,24 +96,21 @@ const style = css`*,
|
|
|
95
96
|
grid-row: 2/3;
|
|
96
97
|
display: inherit;
|
|
97
98
|
}
|
|
98
|
-
.sbb-sr__grid .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper {
|
|
99
|
-
overflow: hidden;
|
|
100
|
-
}
|
|
101
99
|
|
|
102
100
|
.sbb-sr__navigation-control-button {
|
|
103
101
|
position: absolute;
|
|
104
102
|
z-index: 10;
|
|
105
103
|
}
|
|
106
104
|
.sbb-sr__navigation-control-button#first-tab-element {
|
|
107
|
-
inset-inline-end: var(--sbb-
|
|
105
|
+
inset-inline-end: calc(8 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
108
106
|
}
|
|
109
107
|
.sbb-sr__navigation-control-button#last-tab-element {
|
|
110
|
-
inset-inline-start: var(--sbb-
|
|
108
|
+
inset-inline-start: calc(8 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
111
109
|
}
|
|
112
110
|
|
|
113
111
|
.sbb-sr-navigation-wrapper {
|
|
114
112
|
overflow: hidden;
|
|
115
|
-
padding-block: 0 var(--sbb-
|
|
113
|
+
padding-block: 0 calc(32 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
116
114
|
}
|
|
117
115
|
|
|
118
116
|
.sbb-sr-navigation {
|
|
@@ -165,11 +163,31 @@ const style = css`*,
|
|
|
165
163
|
font-size: inherit;
|
|
166
164
|
display: flex;
|
|
167
165
|
flex-wrap: nowrap;
|
|
168
|
-
gap: var(--sbb-
|
|
169
|
-
padding: 0.
|
|
166
|
+
gap: calc(4 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
167
|
+
padding: calc(2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
170
168
|
}
|
|
171
169
|
|
|
172
|
-
.sbb-
|
|
170
|
+
.sbb-sr__wrapper-coach-decks {
|
|
171
|
+
display: flex;
|
|
172
|
+
overflow: hidden;
|
|
173
|
+
height: inherit;
|
|
174
|
+
}
|
|
175
|
+
.sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels {
|
|
176
|
+
display: flex;
|
|
177
|
+
justify-content: space-between;
|
|
178
|
+
height: calc(var(--sbb-seat-reservation-height) * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
179
|
+
writing-mode: vertical-lr;
|
|
180
|
+
}
|
|
181
|
+
.sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels b {
|
|
182
|
+
height: calc(50% - 16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
183
|
+
width: calc(20 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
184
|
+
margin-block-end: 16px;
|
|
185
|
+
rotate: 180deg;
|
|
186
|
+
text-align: center;
|
|
187
|
+
line-height: calc(20 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
.sbb-sr__wrapper {
|
|
173
191
|
--sbb-scrollbar-thumb-width: 0.125rem;
|
|
174
192
|
--sbb-scrollbar-thumb-width-hover: 0.25rem;
|
|
175
193
|
--sbb-scrollbar-width-firefox: thin;
|
|
@@ -178,36 +196,58 @@ const style = css`*,
|
|
|
178
196
|
--sbb-scrollbar-track-color: transparent;
|
|
179
197
|
--sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
|
|
180
198
|
}
|
|
181
|
-
.sbb-
|
|
199
|
+
.sbb-sr__wrapper::-webkit-scrollbar {
|
|
182
200
|
width: var(--sbb-scrollbar-width);
|
|
183
201
|
height: var(--sbb-scrollbar-width);
|
|
184
202
|
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
185
203
|
}
|
|
186
|
-
.sbb-
|
|
204
|
+
.sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
187
205
|
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
188
206
|
}
|
|
189
|
-
.sbb-
|
|
207
|
+
.sbb-sr__wrapper::-webkit-scrollbar-thumb {
|
|
190
208
|
background-color: var(--sbb-scrollbar-color, currentcolor);
|
|
191
209
|
border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
|
|
192
210
|
border-radius: var(--sbb-border-radius-4x);
|
|
193
211
|
background-clip: padding-box;
|
|
194
212
|
}
|
|
195
|
-
.sbb-
|
|
213
|
+
.sbb-sr__wrapper::-webkit-scrollbar-thumb:hover {
|
|
196
214
|
background-color: var(--sbb-scrollbar-color-hover, currentcolor);
|
|
197
215
|
border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
|
|
198
216
|
}
|
|
199
|
-
.sbb-
|
|
217
|
+
.sbb-sr__wrapper::-webkit-scrollbar-button, .sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
200
218
|
display: none;
|
|
201
219
|
}
|
|
202
220
|
@supports not selector(::-webkit-scrollbar) {
|
|
203
|
-
.sbb-
|
|
221
|
+
.sbb-sr__wrapper {
|
|
204
222
|
scrollbar-width: var(--sbb-scrollbar-width-firefox);
|
|
205
223
|
scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
|
|
206
224
|
}
|
|
207
225
|
}
|
|
208
|
-
.sbb-
|
|
226
|
+
.sbb-sr__wrapper {
|
|
209
227
|
overflow-x: scroll;
|
|
210
|
-
padding-block: var(--sbb-
|
|
228
|
+
padding-block: calc(8 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
229
|
+
}
|
|
230
|
+
.sbb-sr__wrapper .sbb-sr__parent {
|
|
231
|
+
display: flex;
|
|
232
|
+
flex-direction: column;
|
|
233
|
+
position: relative;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
.sbb-sr__list-decks {
|
|
237
|
+
list-style: none;
|
|
238
|
+
margin: 0;
|
|
239
|
+
padding: 0;
|
|
240
|
+
font-size: inherit;
|
|
241
|
+
display: flex;
|
|
242
|
+
flex-direction: column;
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
.sbb-sr__list-decks--gap {
|
|
246
|
+
gap: calc(48 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
.sbb-sr__list-item-deck {
|
|
250
|
+
position: relative;
|
|
211
251
|
}
|
|
212
252
|
|
|
213
253
|
.sbb-sr__list-coaches {
|
|
@@ -216,7 +256,7 @@ const style = css`*,
|
|
|
216
256
|
padding: 0;
|
|
217
257
|
font-size: inherit;
|
|
218
258
|
display: flex;
|
|
219
|
-
gap: 0.
|
|
259
|
+
gap: calc(4 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
220
260
|
}
|
|
221
261
|
|
|
222
262
|
.sbb-sr__item-coach {
|
|
@@ -232,7 +272,7 @@ const style = css`*,
|
|
|
232
272
|
}
|
|
233
273
|
|
|
234
274
|
.sbb-sr.sbb-sr__grid--vertical {
|
|
235
|
-
grid-template-columns: calc((var(--sbb-seat-reservation-
|
|
275
|
+
grid-template-columns: calc((var(--sbb-seat-reservation-height) + 24) * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) 1fr;
|
|
236
276
|
grid-template-rows: 20% 60% 20%;
|
|
237
277
|
}
|
|
238
278
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-first-grid {
|
|
@@ -241,8 +281,8 @@ const style = css`*,
|
|
|
241
281
|
}
|
|
242
282
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-first-grid #first-tab-element {
|
|
243
283
|
position: absolute;
|
|
244
|
-
inset-block-end: var(--sbb-
|
|
245
|
-
inset-inline-start: var(--sbb-
|
|
284
|
+
inset-block-end: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
285
|
+
inset-inline-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
246
286
|
inset-inline-end: initial;
|
|
247
287
|
rotate: 90deg;
|
|
248
288
|
}
|
|
@@ -251,24 +291,20 @@ const style = css`*,
|
|
|
251
291
|
grid-row: 3/4;
|
|
252
292
|
}
|
|
253
293
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-last-grid #last-tab-element {
|
|
254
|
-
inset-block-start: var(--sbb-
|
|
255
|
-
inset-inline-start: var(--sbb-
|
|
294
|
+
inset-block-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
295
|
+
inset-inline-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
256
296
|
rotate: 90deg;
|
|
257
297
|
}
|
|
258
298
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-wrapper {
|
|
259
299
|
padding-block-end: initial;
|
|
260
|
-
padding-inline-start: var(--sbb-
|
|
261
|
-
}
|
|
262
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-wrapper {
|
|
263
|
-
padding-block-end: initial;
|
|
264
|
-
padding-inline-start: var(--sbb-spacing-fixed-8x);
|
|
300
|
+
padding-inline-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
265
301
|
}
|
|
266
302
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr__component {
|
|
267
303
|
grid-column: 1/3;
|
|
268
304
|
grid-row: 1/4;
|
|
269
305
|
}
|
|
270
306
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner {
|
|
271
|
-
grid-template-columns: calc((var(--sbb-seat-reservation-
|
|
307
|
+
grid-template-columns: calc((var(--sbb-seat-reservation-height) + 24) * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) 1fr;
|
|
272
308
|
grid-template-rows: 20% 60% 20%;
|
|
273
309
|
}
|
|
274
310
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid {
|
|
@@ -283,7 +319,7 @@ const style = css`*,
|
|
|
283
319
|
}
|
|
284
320
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid .sbb-sr-navigation .sbb-sr-navigation__list-coaches {
|
|
285
321
|
flex-direction: column;
|
|
286
|
-
padding:
|
|
322
|
+
padding: calc(2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
287
323
|
}
|
|
288
324
|
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid {
|
|
289
325
|
grid-column: 1/2;
|
|
@@ -291,7 +327,22 @@ const style = css`*,
|
|
|
291
327
|
overflow: hidden;
|
|
292
328
|
position: relative;
|
|
293
329
|
}
|
|
294
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
330
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks {
|
|
331
|
+
flex-direction: column;
|
|
332
|
+
}
|
|
333
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels {
|
|
334
|
+
flex-direction: column-reverse;
|
|
335
|
+
width: calc(100% - 24 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
336
|
+
height: initial;
|
|
337
|
+
}
|
|
338
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels b {
|
|
339
|
+
width: calc(50% - 16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
340
|
+
height: calc(20 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
341
|
+
margin-bottom: 16px;
|
|
342
|
+
rotate: initial;
|
|
343
|
+
writing-mode: initial;
|
|
344
|
+
}
|
|
345
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper {
|
|
295
346
|
--sbb-scrollbar-thumb-width: 0.125rem;
|
|
296
347
|
--sbb-scrollbar-thumb-width-hover: 0.25rem;
|
|
297
348
|
--sbb-scrollbar-width-firefox: thin;
|
|
@@ -300,41 +351,41 @@ const style = css`*,
|
|
|
300
351
|
--sbb-scrollbar-track-color: transparent;
|
|
301
352
|
--sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
|
|
302
353
|
}
|
|
303
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
354
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar {
|
|
304
355
|
width: var(--sbb-scrollbar-width);
|
|
305
356
|
height: var(--sbb-scrollbar-width);
|
|
306
357
|
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
307
358
|
}
|
|
308
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
359
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
309
360
|
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
310
361
|
}
|
|
311
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
362
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-thumb {
|
|
312
363
|
background-color: var(--sbb-scrollbar-color, currentcolor);
|
|
313
364
|
border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
|
|
314
365
|
border-radius: var(--sbb-border-radius-4x);
|
|
315
366
|
background-clip: padding-box;
|
|
316
367
|
}
|
|
317
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
368
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-thumb:hover {
|
|
318
369
|
background-color: var(--sbb-scrollbar-color-hover, currentcolor);
|
|
319
370
|
border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
|
|
320
371
|
}
|
|
321
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
372
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-button, .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
322
373
|
display: none;
|
|
323
374
|
}
|
|
324
375
|
@supports not selector(::-webkit-scrollbar) {
|
|
325
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
376
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper {
|
|
326
377
|
scrollbar-width: var(--sbb-scrollbar-width-firefox);
|
|
327
378
|
scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
|
|
328
379
|
}
|
|
329
380
|
}
|
|
330
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
381
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper {
|
|
331
382
|
overflow: hidden scroll;
|
|
332
|
-
padding: 0 var(--sbb-spacing-fixed-4x) 0 0;
|
|
333
383
|
height: 100%;
|
|
384
|
+
padding: 0;
|
|
334
385
|
}
|
|
335
|
-
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-
|
|
386
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__parent {
|
|
336
387
|
rotate: 90deg;
|
|
337
|
-
transform-origin: calc((var(--sbb-seat-reservation-
|
|
388
|
+
transform-origin: calc((var(--sbb-seat-reservation-height) + 24 - 24 - 6 * (var(--sbb-seat-reservation-decks) - 1)) / 2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) calc((var(--sbb-seat-reservation-height) + 24 - 24 - 6 * (var(--sbb-seat-reservation-decks) - 1)) / 2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
338
389
|
}`;
|
|
339
390
|
let SbbSeatReservationElement = (() => {
|
|
340
391
|
var _a;
|
|
@@ -348,7 +399,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
348
399
|
super(...arguments);
|
|
349
400
|
this._language = new SbbLanguageController(this);
|
|
350
401
|
this._notAreaElements = [
|
|
351
|
-
"
|
|
402
|
+
"DRIVER_AREA",
|
|
352
403
|
"COACH_PASSAGE",
|
|
353
404
|
"COACH_WALL_NO_PASSAGE",
|
|
354
405
|
"COMPARTMENT_PASSAGE",
|
|
@@ -371,7 +422,9 @@ let SbbSeatReservationElement = (() => {
|
|
|
371
422
|
firstUpdated(changedProperties) {
|
|
372
423
|
super.firstUpdated(changedProperties);
|
|
373
424
|
this.updateComplete.then(() => {
|
|
425
|
+
this.initPrepairSeatReservationData();
|
|
374
426
|
this.initNavigationSelectionByScrollEvent();
|
|
427
|
+
this.requestUpdate();
|
|
375
428
|
});
|
|
376
429
|
}
|
|
377
430
|
render() {
|
|
@@ -387,7 +440,6 @@ let SbbSeatReservationElement = (() => {
|
|
|
387
440
|
this.style?.setProperty("--sbb-seat-reservation-one-px-rem", `${onePixelInRem + "rem"}`);
|
|
388
441
|
}
|
|
389
442
|
_initVehicleSeatReservationConstruction() {
|
|
390
|
-
const coachItems = JSON.parse(JSON.stringify(this.seatReservation?.coachItems));
|
|
391
443
|
const classAlignVertical = this.alignVertical ? "sbb-sr__grid--vertical" : "";
|
|
392
444
|
this._coachesHtmlTemplate = html`
|
|
393
445
|
<div class="sbb-sr__container">
|
|
@@ -402,11 +454,25 @@ let SbbSeatReservationElement = (() => {
|
|
|
402
454
|
<div class="sbb-sr-grid-inner">
|
|
403
455
|
<div class="nav-grid">${this._renderNavigation()}</div>
|
|
404
456
|
<div class="coaches-grid">
|
|
405
|
-
<div class="sbb-sr__wrapper">
|
|
406
|
-
<div
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
457
|
+
<div class="sbb-sr__wrapper-coach-decks">
|
|
458
|
+
<div class="sbb-sr__wrapper-deck-labels">${this._renderDeckLabels()}</div>
|
|
459
|
+
<div id="sbb-sr__wrapper-scrollarea" class="sbb-sr__wrapper">
|
|
460
|
+
<div id="sbb-sr__parent-area" class="sbb-sr__parent" tabindex="-1">
|
|
461
|
+
<ul
|
|
462
|
+
class="${classMap({
|
|
463
|
+
"sbb-sr__list-decks": true,
|
|
464
|
+
"sbb-sr__list-decks--gap": this.hasMultipleDecks
|
|
465
|
+
})}"
|
|
466
|
+
>
|
|
467
|
+
${this.seatReservations?.map((seatReservation, index) => {
|
|
468
|
+
return html`<li class="sbb-sr__list-item-deck">
|
|
469
|
+
<ul class="sbb-sr__list-coaches" role="presentation">
|
|
470
|
+
${this._renderCoaches(seatReservation, index)}
|
|
471
|
+
</ul>
|
|
472
|
+
</li>`;
|
|
473
|
+
})}
|
|
474
|
+
</ul>
|
|
475
|
+
</div>
|
|
410
476
|
</div>
|
|
411
477
|
</div>
|
|
412
478
|
</div>
|
|
@@ -419,17 +485,24 @@ let SbbSeatReservationElement = (() => {
|
|
|
419
485
|
</div>
|
|
420
486
|
`;
|
|
421
487
|
}
|
|
488
|
+
_renderDeckLabels() {
|
|
489
|
+
if (!this.hasMultipleDecks)
|
|
490
|
+
return null;
|
|
491
|
+
return this.seatReservations.map((seatReservation) => {
|
|
492
|
+
const deckDescription = getI18nSeatReservation(seatReservation.deckCoachLevel, this._language.current);
|
|
493
|
+
return html`<b aria-hidden="true">${deckDescription}</b>`;
|
|
494
|
+
});
|
|
495
|
+
}
|
|
422
496
|
_renderNavigationControlButton(btnDirection) {
|
|
423
|
-
if (!this.hasNavigation)
|
|
497
|
+
if (!this.hasNavigation || !this.seatReservations)
|
|
424
498
|
return null;
|
|
425
|
-
}
|
|
426
499
|
const btnId = btnDirection == "DIRECTION_RIGHT" ? "last-tab-element" : "first-tab-element";
|
|
427
500
|
const btnIcon = btnDirection == "DIRECTION_RIGHT" ? "arrow-right-small" : "arrow-left-small";
|
|
428
501
|
const btnAriaDescription = btnDirection == "DIRECTION_RIGHT" ? getI18nSeatReservation("SEAT_RESERVATION_END", this._language.current) : getI18nSeatReservation("SEAT_RESERVATION_BEGIN", this._language.current);
|
|
429
502
|
let btnDisabled = true;
|
|
430
503
|
if (btnDirection == "DIRECTION_LEFT" && this.selectedCoachIndex > 0) {
|
|
431
504
|
btnDisabled = false;
|
|
432
|
-
} else if (btnDirection == "DIRECTION_RIGHT" && this.selectedCoachIndex < this.
|
|
505
|
+
} else if (btnDirection == "DIRECTION_RIGHT" && this.selectedCoachIndex < this.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1) {
|
|
433
506
|
btnDisabled = false;
|
|
434
507
|
}
|
|
435
508
|
return html`<sbb-secondary-button
|
|
@@ -445,9 +518,8 @@ let SbbSeatReservationElement = (() => {
|
|
|
445
518
|
></sbb-secondary-button>`;
|
|
446
519
|
}
|
|
447
520
|
_renderNavigation() {
|
|
448
|
-
if (!this.hasNavigation)
|
|
521
|
+
if (!this.hasNavigation || !this.seatReservations)
|
|
449
522
|
return null;
|
|
450
|
-
}
|
|
451
523
|
return html`<div class="sbb-sr-navigation-wrapper">
|
|
452
524
|
<nav id="sbb-sr-navigation" class="sbb-sr-navigation">
|
|
453
525
|
<ul
|
|
@@ -455,29 +527,29 @@ let SbbSeatReservationElement = (() => {
|
|
|
455
527
|
class="sbb-sr-navigation__list-coaches"
|
|
456
528
|
aria-label="${getI18nSeatReservation("SEAT_RESERVATION_NAVIGATION", this._language.current)}"
|
|
457
529
|
>
|
|
458
|
-
${this.
|
|
530
|
+
${this.seatReservations[this.currSelectedDeckIndex]?.coachItems.map((coachItem, index) => {
|
|
459
531
|
const coachFreePlaces = this.getAvailableFreePlacesNumFromCoach(index);
|
|
460
532
|
return html`<li>
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
533
|
+
<sbb-seat-reservation-navigation-coach
|
|
534
|
+
@selectcoach=${(event) => this._onSelectNavCoach(event)}
|
|
535
|
+
@focuscoach=${() => this._onFocusNavCoach()}
|
|
536
|
+
class="${classMap({
|
|
465
537
|
"sbb-sr__navigation-coach--hover-scroll": this.hoveredScrollCoachIndex === index
|
|
466
538
|
})}"
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
539
|
+
index="${index}"
|
|
540
|
+
coach-id="${coachItem.id}"
|
|
541
|
+
.freePlacesByType="${coachFreePlaces}"
|
|
542
|
+
.selected=${this.selectedCoachIndex === index}
|
|
543
|
+
.focused=${this.focusedCoachIndex === index}
|
|
544
|
+
.propertyIds="${coachItem.propertyIds}"
|
|
545
|
+
.travelClass="${coachItem.travelClass}"
|
|
546
|
+
?driver-area="${!coachItem.places?.length}"
|
|
547
|
+
?first="${coachItem?.driverAreaSide?.left}"
|
|
548
|
+
?last="${coachItem?.driverAreaSide?.right}"
|
|
549
|
+
?vertical="${this.alignVertical}"
|
|
550
|
+
>
|
|
551
|
+
</sbb-seat-reservation-navigation-coach>
|
|
552
|
+
</li>`;
|
|
481
553
|
})}
|
|
482
554
|
</ul>
|
|
483
555
|
</nav>
|
|
@@ -488,39 +560,42 @@ let SbbSeatReservationElement = (() => {
|
|
|
488
560
|
* @param coaches
|
|
489
561
|
* @returns
|
|
490
562
|
*/
|
|
491
|
-
_renderCoaches(
|
|
563
|
+
_renderCoaches(seatReservation, deckIndex) {
|
|
564
|
+
const coaches = JSON.parse(JSON.stringify(seatReservation?.coachItems));
|
|
492
565
|
if (!coaches) {
|
|
493
566
|
return null;
|
|
494
567
|
}
|
|
495
|
-
return coaches.map((coachItem,
|
|
568
|
+
return coaches.map((coachItem, coachIndex) => {
|
|
496
569
|
return html`
|
|
497
|
-
<li class="sbb-sr__item-coach"
|
|
570
|
+
<li class="sbb-sr__item-coach">
|
|
571
|
+
${this._renderCoachElement(coachItem, coachIndex, deckIndex)}
|
|
572
|
+
</li>
|
|
498
573
|
`;
|
|
499
574
|
});
|
|
500
575
|
}
|
|
501
|
-
_renderCoachElement(coachItem,
|
|
576
|
+
_renderCoachElement(coachItem, coachIndex, coachDeckIndex) {
|
|
502
577
|
const calculatedCoachDimension = this.getCalculatedDimension(coachItem.dimension);
|
|
503
|
-
const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem,
|
|
578
|
+
const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem, coachIndex);
|
|
504
579
|
return html`<sbb-seat-reservation-scoped
|
|
505
580
|
style=${styleMap({
|
|
506
581
|
"--sbb-seat-reservation-scoped-width": calculatedCoachDimension.w,
|
|
507
582
|
"--sbb-seat-reservation-scoped-height": calculatedCoachDimension.h
|
|
508
583
|
})}
|
|
509
584
|
>
|
|
510
|
-
${this._getRenderedCoachBorders(coachItem
|
|
511
|
-
${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension,
|
|
512
|
-
${this._getRenderedServiceElements(
|
|
585
|
+
${this._getRenderedCoachBorders(coachItem)}
|
|
586
|
+
${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension, coachIndex, coachDeckIndex)}
|
|
587
|
+
${this._getRenderedServiceElements(coachIndex, coachDeckIndex, coachItem.serviceElements)}
|
|
513
588
|
|
|
514
589
|
<table
|
|
515
|
-
@focus=${() => this.onFocusTableCoachAndPreselectPlace(
|
|
516
|
-
id="sbb-sr-coach-${
|
|
590
|
+
@focus=${() => this.onFocusTableCoachAndPreselectPlace(coachIndex)}
|
|
591
|
+
id="sbb-sr-coach-${coachIndex}"
|
|
517
592
|
class="sbb-sr-coach-wrapper__table"
|
|
518
|
-
aria-describedby="sbb-sr-coach-caption-${
|
|
593
|
+
aria-describedby="sbb-sr-coach-caption-${coachIndex}"
|
|
519
594
|
>
|
|
520
|
-
<caption id="sbb-sr-coach-caption-${
|
|
595
|
+
<caption id="sbb-sr-coach-caption-${coachIndex}" tabindex="-1">
|
|
521
596
|
<sbb-screen-reader-only>${descriptionTableCoachWithServices}</sbb-screen-reader-only>
|
|
522
597
|
</caption>
|
|
523
|
-
${this._getRenderedRowPlaces(coachItem,
|
|
598
|
+
${this._getRenderedRowPlaces(coachItem, coachIndex, coachDeckIndex)}
|
|
524
599
|
</table>
|
|
525
600
|
</sbb-seat-reservation-scoped>`;
|
|
526
601
|
}
|
|
@@ -528,13 +603,13 @@ let SbbSeatReservationElement = (() => {
|
|
|
528
603
|
* @returns Returns the border graphic (COACH_BORDER_MIDDLE) of a coach with calculated border gap and coach width,
|
|
529
604
|
* depending on whether the coach is with a driver area or without.
|
|
530
605
|
*/
|
|
531
|
-
_getRenderedCoachBorders(coachItem
|
|
606
|
+
_getRenderedCoachBorders(coachItem) {
|
|
532
607
|
const COACH_PASSAGE_WIDTH = 1;
|
|
533
608
|
const allElements = coachItem.graphicElements;
|
|
534
|
-
const driverArea = allElements?.find((element) => element.icon === "
|
|
609
|
+
const driverArea = allElements?.find((element) => element.icon === "DRIVER_AREA");
|
|
535
610
|
const borderWidth = driverArea ? coachItem.dimension.w - driverArea.dimension.w - COACH_PASSAGE_WIDTH : coachItem.dimension.w - COACH_PASSAGE_WIDTH * 2;
|
|
536
611
|
const borderHeight = (coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;
|
|
537
|
-
const borderOffsetX =
|
|
612
|
+
const borderOffsetX = driverArea && driverArea.position.x === 0 ? driverArea?.dimension.w * this.baseGridSize : this.baseGridSize;
|
|
538
613
|
return html`
|
|
539
614
|
<sbb-seat-reservation-graphic
|
|
540
615
|
style=${styleMap({
|
|
@@ -550,7 +625,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
550
625
|
></sbb-seat-reservation-graphic>
|
|
551
626
|
`;
|
|
552
627
|
}
|
|
553
|
-
_getRenderedRowPlaces(coach, coachIndex) {
|
|
628
|
+
_getRenderedRowPlaces(coach, coachIndex, deckIndex) {
|
|
554
629
|
if (!coach.places) {
|
|
555
630
|
return null;
|
|
556
631
|
}
|
|
@@ -565,19 +640,22 @@ let SbbSeatReservationElement = (() => {
|
|
|
565
640
|
return Object.values(tableRowPlaces).map((rowPlaces, index) => {
|
|
566
641
|
return html`
|
|
567
642
|
<tr id="row-${coachIndex}-${rowPlaces[0].position.y}" data-row-index=${index}>
|
|
568
|
-
${this._getRenderedColumnPlaces(rowPlaces, coachIndex)}
|
|
643
|
+
${this._getRenderedColumnPlaces(rowPlaces, coachIndex, deckIndex)}
|
|
569
644
|
</tr>
|
|
570
645
|
`;
|
|
571
646
|
}).flatMap((rowTemplate) => rowTemplate);
|
|
572
647
|
}
|
|
573
|
-
_getRenderedColumnPlaces(places, coachIndex) {
|
|
648
|
+
_getRenderedColumnPlaces(places, coachIndex, deckIndex) {
|
|
574
649
|
places.sort((placeA, placeB) => Number(placeA.position.x) - Number(placeB.position.x));
|
|
575
650
|
return places?.map((place, index) => {
|
|
576
651
|
const calculatedDimension = this.getCalculatedDimension(place.dimension);
|
|
577
652
|
const calculatedPosition = this.getCalculatedPosition(place.position);
|
|
578
653
|
const textRotation = this.alignVertical ? -90 : 0;
|
|
579
654
|
return html`
|
|
580
|
-
<td
|
|
655
|
+
<td
|
|
656
|
+
id="cell-${deckIndex}-${coachIndex}-${place.position.y}-${index}"
|
|
657
|
+
class="graphical-element"
|
|
658
|
+
>
|
|
581
659
|
<sbb-seat-reservation-place-control
|
|
582
660
|
style=${styleMap({
|
|
583
661
|
"--sbb-seat-reservation-place-control-width": calculatedDimension.w,
|
|
@@ -587,7 +665,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
587
665
|
})}
|
|
588
666
|
@selectplace=${(selectPlaceEvent) => this._onSelectPlace(selectPlaceEvent)}
|
|
589
667
|
exportparts="sbb-sr-place-part"
|
|
590
|
-
id="seat-reservation__place-button-${coachIndex}-${place.number}"
|
|
668
|
+
id="seat-reservation__place-button-${deckIndex}-${coachIndex}-${place.number}"
|
|
591
669
|
class="seat-reservation-place-control"
|
|
592
670
|
data-cell-id="${coachIndex}-${place.position.y}-${index}"
|
|
593
671
|
text=${place.number}
|
|
@@ -598,6 +676,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
598
676
|
rotation=${place.rotation ?? nothing}
|
|
599
677
|
text-rotation=${textRotation}
|
|
600
678
|
coach-index=${coachIndex}
|
|
679
|
+
deck-index=${deckIndex}
|
|
601
680
|
.propertyIds=${place.propertyIds}
|
|
602
681
|
.preventClick=${this.preventPlaceClick}
|
|
603
682
|
></sbb-seat-reservation-place-control>
|
|
@@ -605,10 +684,9 @@ let SbbSeatReservationElement = (() => {
|
|
|
605
684
|
`;
|
|
606
685
|
});
|
|
607
686
|
}
|
|
608
|
-
_getRenderedGraphicalElements(graphicalElements, coachDimension, coachIndex) {
|
|
609
|
-
if (!graphicalElements)
|
|
687
|
+
_getRenderedGraphicalElements(graphicalElements, coachDimension, coachIndex, coachDeckIndex) {
|
|
688
|
+
if (!graphicalElements)
|
|
610
689
|
return null;
|
|
611
|
-
}
|
|
612
690
|
return graphicalElements?.map((graphicalElement) => {
|
|
613
691
|
const icon = graphicalElement.icon ?? "";
|
|
614
692
|
const elementRotation = graphicalElement.rotation || 0;
|
|
@@ -617,7 +695,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
617
695
|
if (this._notAreaElements.findIndex((notAreaElement) => notAreaElement === icon) > -1) {
|
|
618
696
|
return this._getRenderElementWithoutArea(graphicalElement, elementRotation, coachDimension);
|
|
619
697
|
}
|
|
620
|
-
return this._getRenderElementWithArea(graphicalElement, elementFixedRotation, coachDimension, coachIndex);
|
|
698
|
+
return this._getRenderElementWithArea(graphicalElement, elementFixedRotation, coachDimension, coachIndex, coachDeckIndex);
|
|
621
699
|
});
|
|
622
700
|
}
|
|
623
701
|
/**
|
|
@@ -628,14 +706,14 @@ let SbbSeatReservationElement = (() => {
|
|
|
628
706
|
* @param coachIndex used to generate a unique id for the popover trigger
|
|
629
707
|
* @private
|
|
630
708
|
*/
|
|
631
|
-
_getRenderElementWithArea(graphicalElement, rotation, coachDimension, coachIndex) {
|
|
709
|
+
_getRenderElementWithArea(graphicalElement, rotation, coachDimension, coachIndex, coachDeckIndex) {
|
|
632
710
|
const isNotTableGraphic = graphicalElement.icon?.indexOf("TABLE") === -1;
|
|
633
711
|
const areaProperty = graphicalElement.icon && isNotTableGraphic ? graphicalElement.icon : null;
|
|
634
712
|
const stretchHeight = areaProperty !== "ENTRY_EXIT";
|
|
635
713
|
const ariaLabelForArea = graphicalElement.icon ? getI18nSeatReservation(graphicalElement.icon, this._language.current) : nothing;
|
|
636
714
|
const calculatedDimension = this.getCalculatedDimension(graphicalElement.dimension, coachDimension, true, stretchHeight);
|
|
637
715
|
const calculatedPosition = this.getCalculatedPosition(graphicalElement.position, graphicalElement.dimension, coachDimension, true);
|
|
638
|
-
const triggerId = `popover-trigger-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
|
|
716
|
+
const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
|
|
639
717
|
let elementMounting = "free";
|
|
640
718
|
if (graphicalElement.position.y === this.coachBorderOffset * -1) {
|
|
641
719
|
elementMounting = "upper-border";
|
|
@@ -645,6 +723,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
645
723
|
return html`
|
|
646
724
|
<sbb-seat-reservation-area
|
|
647
725
|
id="${triggerId}"
|
|
726
|
+
class="${classMap({ "sbb-seat-reservation-area--cursor-pointer": areaProperty !== null })}"
|
|
648
727
|
style=${styleMap({
|
|
649
728
|
"--sbb-seat-reservation-area-width": calculatedDimension.w,
|
|
650
729
|
"--sbb-seat-reservation-area-height": calculatedDimension.h,
|
|
@@ -657,9 +736,6 @@ let SbbSeatReservationElement = (() => {
|
|
|
657
736
|
>
|
|
658
737
|
${areaProperty ? html`
|
|
659
738
|
<sbb-seat-reservation-graphic
|
|
660
|
-
style=${styleMap({
|
|
661
|
-
"--sbb-seat-reservation--cursor-pointer": "pointer"
|
|
662
|
-
})}
|
|
663
739
|
name=${areaProperty}
|
|
664
740
|
width=${this.baseGridSize}
|
|
665
741
|
height=${this.baseGridSize}
|
|
@@ -669,33 +745,13 @@ let SbbSeatReservationElement = (() => {
|
|
|
669
745
|
></sbb-seat-reservation-graphic>
|
|
670
746
|
` : nothing}
|
|
671
747
|
</sbb-seat-reservation-area>
|
|
672
|
-
${this._popover(triggerId, ariaLabelForArea)}
|
|
673
|
-
`;
|
|
674
|
-
}
|
|
675
|
-
/**
|
|
676
|
-
* Creates a popover for extra service information
|
|
677
|
-
* @param triggerId
|
|
678
|
-
* @param popoverContent
|
|
679
|
-
* @private
|
|
680
|
-
*/
|
|
681
|
-
_popover(triggerId, popoverContent) {
|
|
682
|
-
return html`
|
|
683
|
-
<sbb-popover trigger="${triggerId}">
|
|
684
|
-
<p class="sbb-text-s sbb-sr-popover">${popoverContent}</p>
|
|
685
|
-
</sbb-popover>
|
|
748
|
+
${areaProperty ? this._popover(triggerId, ariaLabelForArea) : nothing}
|
|
686
749
|
`;
|
|
687
750
|
}
|
|
688
|
-
/**
|
|
689
|
-
* trigger to close all opened popovers (normally only one is opened at a time)
|
|
690
|
-
* @private
|
|
691
|
-
*/
|
|
692
|
-
_closePopover() {
|
|
693
|
-
this.shadowRoot?.querySelectorAll('sbb-popover[data-state="opened"]').forEach((popover) => popover.setAttribute("data-state", "closed"));
|
|
694
|
-
}
|
|
695
751
|
_getRenderElementWithoutArea(graphicalElement, rotation, coachDimension) {
|
|
696
752
|
const calculatedDimension = this.getCalculatedDimension(graphicalElement.dimension, coachDimension);
|
|
697
753
|
const calculatedPosition = this.getCalculatedPosition(graphicalElement.position, graphicalElement.dimension, coachDimension);
|
|
698
|
-
const icon = graphicalElement.icon && graphicalElement.icon.indexOf("DRIVER_AREA") === -1 ? graphicalElement.icon : graphicalElement.icon?.concat("_", this.
|
|
754
|
+
const icon = graphicalElement.icon && graphicalElement.icon.indexOf("DRIVER_AREA") === -1 ? graphicalElement.icon : graphicalElement.icon?.concat("_", this.seatReservations[this.currSelectedDeckIndex].vehicleType);
|
|
699
755
|
return html` <sbb-seat-reservation-graphic
|
|
700
756
|
style=${styleMap({
|
|
701
757
|
"--sbb-seat-reservation-graphic-width": calculatedDimension.w,
|
|
@@ -710,17 +766,16 @@ let SbbSeatReservationElement = (() => {
|
|
|
710
766
|
?stretch=${true}
|
|
711
767
|
></sbb-seat-reservation-graphic>`;
|
|
712
768
|
}
|
|
713
|
-
_getRenderedServiceElements(coachIndex, serviceElements) {
|
|
714
|
-
if (!serviceElements)
|
|
769
|
+
_getRenderedServiceElements(coachIndex, coachDeckIndex, serviceElements) {
|
|
770
|
+
if (!serviceElements)
|
|
715
771
|
return null;
|
|
716
|
-
}
|
|
717
772
|
return serviceElements?.map((serviceElement) => {
|
|
718
773
|
const titleDescription = serviceElement.icon ? getI18nSeatReservation(serviceElement.icon, this._language.current) : null;
|
|
719
774
|
const calculatedDimension = this.getCalculatedDimension(serviceElement.dimension);
|
|
720
775
|
const calculatedPosition = this.getCalculatedPosition(serviceElement.position);
|
|
721
776
|
const elementRotation = serviceElement.rotation || 0;
|
|
722
777
|
const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;
|
|
723
|
-
const triggerId = `popover-trigger-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
|
|
778
|
+
const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
|
|
724
779
|
return html`
|
|
725
780
|
<sbb-seat-reservation-graphic
|
|
726
781
|
id="${triggerId}"
|
|
@@ -729,9 +784,9 @@ let SbbSeatReservationElement = (() => {
|
|
|
729
784
|
"--sbb-seat-reservation-graphic-height": calculatedDimension.h,
|
|
730
785
|
"--sbb-seat-reservation-graphic-top": calculatedPosition.y,
|
|
731
786
|
"--sbb-seat-reservation-graphic-left": calculatedPosition.x,
|
|
732
|
-
"--sbb-seat-reservation-graphic-position": "absolute"
|
|
733
|
-
"--sbb-seat-reservation--cursor-pointer": "pointer"
|
|
787
|
+
"--sbb-seat-reservation-graphic-position": "absolute"
|
|
734
788
|
})}
|
|
789
|
+
class="sbb-seat-reservation-graphic--cursor-pointer"
|
|
735
790
|
name=${serviceElement.icon ?? nothing}
|
|
736
791
|
.rotation=${elementFixedRotation}
|
|
737
792
|
role="img"
|
|
@@ -751,6 +806,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
751
806
|
this.preventCoachScrollByPlaceClick = true;
|
|
752
807
|
this.isCoachGridFocusable = false;
|
|
753
808
|
if (!this.preventPlaceClick) {
|
|
809
|
+
this.currSelectedDeckIndex = selectedPlace.deckIndex;
|
|
754
810
|
this.updateSelectedSeatReservationPlaces(selectedPlace);
|
|
755
811
|
this.updateCurrentSelectedPlaceInCoach(selectedPlace);
|
|
756
812
|
}
|
|
@@ -775,6 +831,26 @@ let SbbSeatReservationElement = (() => {
|
|
|
775
831
|
}
|
|
776
832
|
this.isAutoScrolling = false;
|
|
777
833
|
}
|
|
834
|
+
/**
|
|
835
|
+
* Creates a popover for extra service information
|
|
836
|
+
* @param triggerId
|
|
837
|
+
* @param popoverContent
|
|
838
|
+
* @private
|
|
839
|
+
*/
|
|
840
|
+
_popover(triggerId, popoverContent) {
|
|
841
|
+
return html`
|
|
842
|
+
<sbb-popover trigger="${triggerId}">
|
|
843
|
+
<p class="sbb-text-s sbb-sr-popover">${popoverContent}</p>
|
|
844
|
+
</sbb-popover>
|
|
845
|
+
`;
|
|
846
|
+
}
|
|
847
|
+
/**
|
|
848
|
+
* trigger to close all opened popovers (normally only one is opened at a time)
|
|
849
|
+
* @private
|
|
850
|
+
*/
|
|
851
|
+
_closePopover() {
|
|
852
|
+
this.shadowRoot?.querySelectorAll('sbb-popover[data-state="opened"]').forEach((popover) => popover.setAttribute("data-state", "closed"));
|
|
853
|
+
}
|
|
778
854
|
_getDescriptionTableCoach(coachItem, coachIndex) {
|
|
779
855
|
if (!coachItem.places?.length) {
|
|
780
856
|
return getI18nSeatReservation("COACH_BLOCKED_TABLE_CAPTION", this._language.current, [
|
|
@@ -824,4 +900,4 @@ let SbbSeatReservationElement = (() => {
|
|
|
824
900
|
export {
|
|
825
901
|
SbbSeatReservationElement
|
|
826
902
|
};
|
|
827
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"seat-reservation.component.js","sources":["../../../../../src/elements-experimental/seat-reservation/seat-reservation/seat-reservation.component.ts"],"sourcesContent":["import { SbbLanguageController } from '@sbb-esta/lyne-elements/core/controllers.js';\nimport { html, isServer, nothing } from 'lit';\nimport type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { customElement } from 'lit/decorators.js';\nimport { classMap } from 'lit/directives/class-map.js';\nimport { styleMap } from 'lit/directives/style-map.js';\n\nimport { getI18nSeatReservation } from '../common.js';\nimport type { CoachItem, Place, ElementDimension, BaseElement, PlaceSelection } from '../common.js';\n\nimport { SeatReservationBaseElement } from './seat-reservation-base-element.js';\nimport style from './seat-reservation.scss?lit&inline';\n\nimport '@sbb-esta/lyne-elements/button.js';\nimport '@sbb-esta/lyne-elements/screen-reader-only.js';\nimport '../seat-reservation-area.js';\nimport '../seat-reservation-graphic.js';\nimport '../seat-reservation-place-control.js';\nimport '../seat-reservation-navigation-coach.js';\nimport '../seat-reservation-scoped.js';\nimport '@sbb-esta/lyne-elements/popover/popover.js';\n\n/**\n * Main component for the seat reservation.\n *\n */\nexport\n@customElement('sbb-seat-reservation')\nclass SbbSeatReservationElement extends SeatReservationBaseElement {\n  public static override styles: CSSResultGroup = style;\n\n  private _language = new SbbLanguageController(this);\n  private _coachesHtmlTemplate?: TemplateResult;\n\n  // Graphics that should not be rendered with an area\n  private _notAreaElements = [\n    'DRIVER_AREA_FULL',\n    'COACH_PASSAGE',\n    'COACH_WALL_NO_PASSAGE',\n    'COMPARTMENT_PASSAGE',\n    'COMPARTMENT_PASSAGE_HIGH',\n    'COMPARTMENT_PASSAGE_MIDDLE',\n    'COMPARTMENT_PASSAGE_LOW',\n  ];\n\n  // Area icons that should not be fixed during rotation when vertical mode is selected\n  private _notFixedRotatableAreaIcons = ['ENTRY_EXIT'];\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('hasNavigation')) {\n      if (this.hasNavigation) {\n        this.shadowRoot\n          ?.querySelectorAll('table')\n          .forEach((table) => table.removeAttribute('tabindex'));\n      } else {\n        this.shadowRoot\n          ?.querySelectorAll('table')\n          .forEach((table) => table.setAttribute('tabindex', '0'));\n      }\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    // We need to wait until the first update is complete to init different html element dimensions\n    this.updateComplete.then(() => {\n      this.initNavigationSelectionByScrollEvent();\n    });\n  }\n\n  protected override render(): TemplateResult | null {\n    if (!isServer) {\n      this._determineBaseFontSize();\n    }\n\n    this._initVehicleSeatReservationConstruction();\n    return this._coachesHtmlTemplate || null;\n  }\n\n  private _determineBaseFontSize(): void {\n    const baseFontSize = parseInt(window.getComputedStyle(document.body).fontSize, 10);\n    //calculate rem of 1px\n    const onePixelInRem = 1 / baseFontSize;\n    this.style?.setProperty('--sbb-seat-reservation-one-px-rem', `${onePixelInRem + 'rem'}`);\n  }\n\n  private _initVehicleSeatReservationConstruction(): void {\n    const coachItems = JSON.parse(JSON.stringify(this.seatReservation?.coachItems));\n    const classAlignVertical = this.alignVertical ? 'sbb-sr__grid--vertical' : '';\n    this._coachesHtmlTemplate = html`\n      <div class=\"sbb-sr__container\">\n        <div class=\"sbb-sr sbb-sr__grid ${classAlignVertical}\">\n          <div class=\"sbb-sr-navigation-first-grid\">\n            ${this._renderNavigationControlButton('DIRECTION_LEFT')}\n          </div>\n          <div\n            class=\"sbb-sr__component\"\n            @keydown=${(evt: KeyboardEvent) => this.handleKeyboardEvent(evt)}\n          >\n            <div class=\"sbb-sr-grid-inner\">\n              <div class=\"nav-grid\">${this._renderNavigation()}</div>\n              <div class=\"coaches-grid\">\n                <div class=\"sbb-sr__wrapper\">\n                  <div id=\"sbb-sr__parent-area\" class=\"sbb-sr__parent\" tabindex=\"-1\">\n                    <ul class=\"sbb-sr__list-coaches\" role=\"presentation\">\n                      ${this._renderCoaches(coachItems)}\n                    </ul>\n                  </div>\n                </div>\n              </div>\n            </div>\n          </div>\n          <div class=\"sbb-sr-navigation-last-grid\">\n            ${this._renderNavigationControlButton('DIRECTION_RIGHT')}\n          </div>\n        </div>\n      </div>\n    `;\n  }\n\n  private _renderNavigationControlButton(btnDirection: string): TemplateResult | null {\n    if (!this.hasNavigation) {\n      return null;\n    }\n\n    const btnId = btnDirection == 'DIRECTION_RIGHT' ? 'last-tab-element' : 'first-tab-element';\n    const btnIcon = btnDirection == 'DIRECTION_RIGHT' ? 'arrow-right-small' : 'arrow-left-small';\n    const btnAriaDescription =\n      btnDirection == 'DIRECTION_RIGHT'\n        ? getI18nSeatReservation('SEAT_RESERVATION_END', this._language.current)\n        : getI18nSeatReservation('SEAT_RESERVATION_BEGIN', this._language.current);\n    let btnDisabled = true;\n\n    if (btnDirection == 'DIRECTION_LEFT' && this.selectedCoachIndex > 0) {\n      btnDisabled = false;\n    } else if (\n      btnDirection == 'DIRECTION_RIGHT' &&\n      this.selectedCoachIndex < this.seatReservation.coachItems.length - 1\n    ) {\n      btnDisabled = false;\n    }\n\n    return html`<sbb-secondary-button\n      @click=\"${() => this.navigateByDirectionBtn(btnDirection)}\"\n      id=\"${btnId}\"\n      class=\"sbb-sr__navigation-control-button\"\n      size=\"s\"\n      icon-name=\"${btnIcon}\"\n      type=\"button\"\n      aria-label=\"${btnAriaDescription}\"\n      role=\"contentinfo\"\n      .disabledInteractive=\"${btnDisabled || nothing}\"\n    ></sbb-secondary-button>`;\n  }\n\n  private _renderNavigation(): TemplateResult | null {\n    if (!this.hasNavigation) {\n      return null;\n    }\n\n    return html`<div class=\"sbb-sr-navigation-wrapper\">\n      <nav id=\"sbb-sr-navigation\" class=\"sbb-sr-navigation\">\n        <ul\n          id=\"sbb-sr__navigation-list-coaches\"\n          class=\"sbb-sr-navigation__list-coaches\"\n          aria-label=\"${getI18nSeatReservation(\n            'SEAT_RESERVATION_NAVIGATION',\n            this._language.current,\n          )}\"\n        >\n          ${this.seatReservation?.coachItems.map((coachItem: CoachItem, index: number) => {\n            const coachFreePlaces = this.getAvailableFreePlacesNumFromCoach(index);\n            return html`<li>\n              <sbb-seat-reservation-navigation-coach\n                @selectcoach=${(event: CustomEvent<number>) => this._onSelectNavCoach(event)}\n                @focuscoach=${() => this._onFocusNavCoach()}\n                class=\"${classMap({\n                  'sbb-sr__navigation-coach--hover-scroll': this.hoveredScrollCoachIndex === index,\n                })}\"\n                index=\"${index}\"\n                coach-id=\"${coachItem.id}\"\n                .freePlacesByType=\"${coachFreePlaces}\"\n                .selected=${this.selectedCoachIndex === index}\n                .focused=${this.focusedCoachIndex === index}\n                .propertyIds=\"${coachItem.propertyIds}\"\n                .travelClass=\"${coachItem.travelClass}\"\n                ?driver-area=\"${!coachItem.places?.length}\"\n                ?first=\"${index === 0}\"\n                ?last=\"${index === this.seatReservation?.coachItems.length - 1}\"\n                ?vertical=\"${this.alignVertical}\"\n              >\n              </sbb-seat-reservation-navigation-coach>\n            </li>`;\n          })}\n        </ul>\n      </nav>\n    </div>`;\n  }\n  /**\n   *\n   * @param coaches\n   * @returns\n   */\n  private _renderCoaches(coaches?: CoachItem[]): TemplateResult[] | null {\n    if (!coaches) {\n      return null;\n    }\n    return coaches.map((coachItem: CoachItem, index: number) => {\n      return html`\n        <li class=\"sbb-sr__item-coach\">${this._renderCoachElement(coachItem, index)}</li>\n      `;\n    });\n  }\n\n  private _renderCoachElement(coachItem: CoachItem, index: number): TemplateResult {\n    const calculatedCoachDimension = this.getCalculatedDimension(coachItem.dimension);\n    const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem, index);\n\n    return html`<sbb-seat-reservation-scoped\n      style=${styleMap({\n        '--sbb-seat-reservation-scoped-width': calculatedCoachDimension.w,\n        '--sbb-seat-reservation-scoped-height': calculatedCoachDimension.h,\n      })}\n    >\n      ${this._getRenderedCoachBorders(coachItem, index)}\n      ${this._getRenderedGraphicalElements(\n        coachItem.graphicElements || [],\n        coachItem.dimension,\n        index,\n      )}\n      ${this._getRenderedServiceElements(index, coachItem.serviceElements)}\n\n      <table\n        @focus=${() => this.onFocusTableCoachAndPreselectPlace(index)}\n        id=\"sbb-sr-coach-${index}\"\n        class=\"sbb-sr-coach-wrapper__table\"\n        aria-describedby=\"sbb-sr-coach-caption-${index}\"\n      >\n        <caption id=\"sbb-sr-coach-caption-${index}\" tabindex=\"-1\">\n          <sbb-screen-reader-only>${descriptionTableCoachWithServices}</sbb-screen-reader-only>\n        </caption>\n        ${this._getRenderedRowPlaces(coachItem, index)}\n      </table>\n    </sbb-seat-reservation-scoped>`;\n  }\n\n  /**\n   * @returns Returns the border graphic (COACH_BORDER_MIDDLE) of a coach with calculated border gap and coach width,\n   * depending on whether the coach is with a driver area or without.\n   */\n  private _getRenderedCoachBorders(coachItem: CoachItem, coachIndex: number): TemplateResult {\n    const COACH_PASSAGE_WIDTH = 1;\n    const allElements = coachItem.graphicElements;\n    const driverArea = allElements?.find(\n      (element: BaseElement) => element.icon === 'DRIVER_AREA_FULL',\n    );\n    const borderWidth = driverArea\n      ? coachItem.dimension.w - driverArea.dimension.w - COACH_PASSAGE_WIDTH\n      : coachItem.dimension.w - COACH_PASSAGE_WIDTH * 2;\n    const borderHeight = (coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;\n    const borderOffsetX =\n      coachIndex === 0 && driverArea\n        ? driverArea?.dimension.w * this.baseGridSize\n        : this.baseGridSize;\n    return html`\n      <sbb-seat-reservation-graphic\n        style=${styleMap({\n          '--sbb-seat-reservation-graphic-width': borderWidth * this.baseGridSize,\n          '--sbb-seat-reservation-graphic-height': borderHeight,\n          '--sbb-seat-reservation-graphic-top': this.coachBorderPadding * -1,\n          '--sbb-seat-reservation-graphic-left': borderOffsetX,\n          '--sbb-seat-reservation-graphic-position': 'absolute',\n        })}\n        name=\"COACH_BORDER_MIDDLE\"\n        ?stretch=${true}\n        role=\"presentation\"\n      ></sbb-seat-reservation-graphic>\n    `;\n  }\n\n  private _getRenderedRowPlaces(coach: CoachItem, coachIndex: number): TemplateResult[] | null {\n    if (!coach.places) {\n      return null;\n    }\n\n    // Prepair rows with the places to render a table\n    const tableRowPlaces: Record<number, Place[]> = {};\n    for (const place of coach.places) {\n      if (!tableRowPlaces[place.position.y]) {\n        tableRowPlaces[place.position.y] = [place];\n      } else {\n        tableRowPlaces[place.position.y].push(place);\n      }\n    }\n\n    return Object.values(tableRowPlaces)\n      .map((rowPlaces: Place[], index) => {\n        return html`\n          <tr id=\"row-${coachIndex}-${rowPlaces[0].position.y}\" data-row-index=${index}>\n            ${this._getRenderedColumnPlaces(rowPlaces, coachIndex)}\n          </tr>\n        `;\n      })\n      .flatMap((rowTemplate) => rowTemplate);\n  }\n\n  private _getRenderedColumnPlaces(places: Place[], coachIndex: number): TemplateResult[] | null {\n    //Sorts each place by its ascending x coordinate\n    places.sort(\n      (placeA: Place, placeB: Place) => Number(placeA.position.x) - Number(placeB.position.x),\n    );\n\n    return places?.map((place: Place, index: number) => {\n      const calculatedDimension = this.getCalculatedDimension(place.dimension);\n      const calculatedPosition = this.getCalculatedPosition(place.position);\n      const textRotation = this.alignVertical ? -90 : 0;\n\n      return html`\n        <td id=\"cell-${coachIndex}-${place.position.y}-${index}\" class=\"graphical-element\">\n          <sbb-seat-reservation-place-control\n            style=${styleMap({\n              '--sbb-seat-reservation-place-control-width': calculatedDimension.w,\n              '--sbb-seat-reservation-place-control-height': calculatedDimension.h,\n              '--sbb-seat-reservation-place-control-top': calculatedPosition.y,\n              '--sbb-seat-reservation-place-control-left': calculatedPosition.x,\n            })}\n            @selectplace=${(selectPlaceEvent: CustomEvent<PlaceSelection>) =>\n              this._onSelectPlace(selectPlaceEvent)}\n            exportparts=\"sbb-sr-place-part\"\n            id=\"seat-reservation__place-button-${coachIndex}-${place.number}\"\n            class=\"seat-reservation-place-control\"\n            data-cell-id=\"${coachIndex}-${place.position.y}-${index}\"\n            text=${place.number}\n            type=${place.type}\n            state=${place.state}\n            width=${place.dimension.w * this.baseGridSize}\n            height=${place.dimension.h * this.baseGridSize}\n            rotation=${place.rotation ?? nothing}\n            text-rotation=${textRotation}\n            coach-index=${coachIndex}\n            .propertyIds=${place.propertyIds}\n            .preventClick=${this.preventPlaceClick}\n          ></sbb-seat-reservation-place-control>\n        </td>\n      `;\n    });\n  }\n\n  private _getRenderedGraphicalElements(\n    graphicalElements: BaseElement[],\n    coachDimension: ElementDimension,\n    coachIndex: number,\n  ): TemplateResult[] | null {\n    if (!graphicalElements) {\n      return null;\n    }\n\n    return graphicalElements?.map((graphicalElement: BaseElement) => {\n      const icon = graphicalElement.icon ?? '';\n      const elementRotation = graphicalElement.rotation || 0;\n      const isNotFixedRotationGraphicalElement =\n        this._notFixedRotatableAreaIcons.indexOf(graphicalElement.icon!) === -1;\n      const elementFixedRotation =\n        this.alignVertical && isNotFixedRotationGraphicalElement\n          ? elementRotation - 90\n          : elementRotation;\n\n      //check if the current element is not an area element, since this element is drawn without an area component\n      if (this._notAreaElements.findIndex((notAreaElement) => notAreaElement === icon) > -1) {\n        return this._getRenderElementWithoutArea(graphicalElement, elementRotation, coachDimension);\n      }\n      return this._getRenderElementWithArea(\n        graphicalElement,\n        elementFixedRotation,\n        coachDimension,\n        coachIndex,\n      );\n    });\n  }\n\n  /**\n   * creates a rendered element with an area component\n   * @param graphicalElement\n   * @param rotation\n   * @param coachDimension\n   * @param coachIndex used to generate a unique id for the popover trigger\n   * @private\n   */\n  private _getRenderElementWithArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n    coachIndex: number,\n  ): TemplateResult {\n    // for TABLE, we use the area component itself to display the table instead of the SVG graphic.\n    // Due to different heights and widths, it wouldn't show correctly. To correct this, we would\n    // need difficult calculations for position, rotation and dimension.\n    const isNotTableGraphic = graphicalElement.icon?.indexOf('TABLE') === -1;\n    const areaProperty = graphicalElement.icon && isNotTableGraphic ? graphicalElement.icon : null;\n    const stretchHeight = areaProperty !== 'ENTRY_EXIT';\n    const ariaLabelForArea = graphicalElement.icon\n      ? getI18nSeatReservation(graphicalElement.icon, this._language.current)\n      : nothing;\n    const calculatedDimension = this.getCalculatedDimension(\n      graphicalElement.dimension,\n      coachDimension,\n      true,\n      stretchHeight,\n    );\n    const calculatedPosition = this.getCalculatedPosition(\n      graphicalElement.position,\n      graphicalElement.dimension,\n      coachDimension,\n      true,\n    );\n\n    //generate unique index number for the trigger element\n    const triggerId = `popover-trigger-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;\n\n    let elementMounting = 'free';\n    if (graphicalElement.position.y === this.coachBorderOffset * -1) {\n      elementMounting = 'upper-border';\n    } else if (\n      graphicalElement.position.y + graphicalElement.dimension.h ===\n      coachDimension.h + this.coachBorderOffset\n    ) {\n      elementMounting = 'lower-border';\n    }\n\n    return html`\n      <sbb-seat-reservation-area\n        id=\"${triggerId}\"\n        style=${styleMap({\n          '--sbb-seat-reservation-area-width': calculatedDimension.w,\n          '--sbb-seat-reservation-area-height': calculatedDimension.h,\n          '--sbb-seat-reservation-area-top': calculatedPosition.y,\n          '--sbb-seat-reservation-area-left': calculatedPosition.x,\n        })}\n        mounting=${elementMounting}\n        background=\"dark\"\n        aria-hidden=\"true\"\n      >\n        ${areaProperty\n          ? html`\n              <sbb-seat-reservation-graphic\n                style=${styleMap({\n                  '--sbb-seat-reservation--cursor-pointer': 'pointer',\n                })}\n                name=${areaProperty}\n                width=${this.baseGridSize}\n                height=${this.baseGridSize}\n                rotation=${rotation}\n                role=\"img\"\n                aria-hidden=\"true\"\n              ></sbb-seat-reservation-graphic>\n            `\n          : nothing}\n      </sbb-seat-reservation-area>\n      ${this._popover(triggerId, ariaLabelForArea)}\n    `;\n  }\n\n  /**\n   * Creates a popover for extra service information\n   * @param triggerId\n   * @param popoverContent\n   * @private\n   */\n  private _popover(\n    triggerId: string,\n    popoverContent: string | null | typeof nothing,\n  ): TemplateResult {\n    return html`\n      <sbb-popover trigger=\"${triggerId}\">\n        <p class=\"sbb-text-s sbb-sr-popover\">${popoverContent}</p>\n      </sbb-popover>\n    `;\n  }\n\n  /**\n   * trigger to close all opened popovers (normally only one is opened at a time)\n   * @private\n   */\n  private _closePopover(): void {\n    this.shadowRoot\n      ?.querySelectorAll('sbb-popover[data-state=\"opened\"]')\n      .forEach((popover) => popover.setAttribute('data-state', 'closed'));\n  }\n\n  private _getRenderElementWithoutArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n  ): TemplateResult {\n    const calculatedDimension = this.getCalculatedDimension(\n      graphicalElement.dimension,\n      coachDimension,\n    );\n    const calculatedPosition = this.getCalculatedPosition(\n      graphicalElement.position,\n      graphicalElement.dimension,\n      coachDimension,\n    );\n\n    // If the icon is the driver area, then here concat the vehicle type to get the right vehicle chassie icon\n    const icon =\n      graphicalElement.icon && graphicalElement.icon.indexOf('DRIVER_AREA') === -1\n        ? graphicalElement.icon\n        : graphicalElement.icon?.concat('_', this.seatReservation.vehicleType);\n\n    return html` <sbb-seat-reservation-graphic\n      style=${styleMap({\n        '--sbb-seat-reservation-graphic-width': calculatedDimension.w,\n        '--sbb-seat-reservation-graphic-height': calculatedDimension.h,\n        '--sbb-seat-reservation-graphic-top': calculatedPosition.y,\n        '--sbb-seat-reservation-graphic-left': calculatedPosition.x,\n        '--sbb-seat-reservation-graphic-position': 'absolute',\n      })}\n      name=${icon ?? nothing}\n      rotation=${rotation}\n      aria-hidden=\"true\"\n      ?stretch=${true}\n    ></sbb-seat-reservation-graphic>`;\n  }\n\n  private _getRenderedServiceElements(\n    coachIndex: number,\n    serviceElements?: BaseElement[],\n  ): TemplateResult[] | null {\n    if (!serviceElements) {\n      return null;\n    }\n\n    return serviceElements?.map((serviceElement: BaseElement) => {\n      const titleDescription = serviceElement.icon\n        ? getI18nSeatReservation(serviceElement.icon, this._language.current)\n        : null;\n      const calculatedDimension = this.getCalculatedDimension(serviceElement.dimension);\n      const calculatedPosition = this.getCalculatedPosition(serviceElement.position);\n      const elementRotation = serviceElement.rotation || 0;\n      const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;\n\n      //generate unique index number for the trigger element\n      const triggerId = `popover-trigger-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;\n\n      return html`\n        <sbb-seat-reservation-graphic\n          id=\"${triggerId}\"\n          style=${styleMap({\n            '--sbb-seat-reservation-graphic-width': calculatedDimension.w,\n            '--sbb-seat-reservation-graphic-height': calculatedDimension.h,\n            '--sbb-seat-reservation-graphic-top': calculatedPosition.y,\n            '--sbb-seat-reservation-graphic-left': calculatedPosition.x,\n            '--sbb-seat-reservation-graphic-position': 'absolute',\n            '--sbb-seat-reservation--cursor-pointer': 'pointer',\n          })}\n          name=${serviceElement.icon ?? nothing}\n          .rotation=${elementFixedRotation}\n          role=\"img\"\n          aria-hidden=\"true\"\n        ></sbb-seat-reservation-graphic>\n        ${this._popover(triggerId, titleDescription)}\n      `;\n    });\n  }\n\n  /**\n   * Manages the selected place event triggered from the place\n   * Each selection emits an array of all selected places\n   * @param selectPlaceEvent\n   */\n  private _onSelectPlace(selectPlaceEvent: CustomEvent<PlaceSelection>): void {\n    const selectedPlace = selectPlaceEvent.detail;\n    // We have to set preventCoachScrollByPlaceClick to true, to prevent automatic scrolling to the new focused place\n    this.preventCoachScrollByPlaceClick = true;\n    this.isCoachGridFocusable = false;\n    if (!this.preventPlaceClick) {\n      // Add place to place collection\n      this.updateSelectedSeatReservationPlaces(selectedPlace);\n      this.updateCurrentSelectedPlaceInCoach(selectedPlace);\n    }\n  }\n\n  private _onSelectNavCoach(event: CustomEvent<number>): void {\n    const selectedNavCoachIndex = event.detail;\n    this.isKeyboardNavigation = false;\n\n    if (selectedNavCoachIndex !== null && selectedNavCoachIndex !== this.currSelectedCoachIndex) {\n      this.unfocusPlaceElement();\n      this.scrollToSelectedNavCoach(selectedNavCoachIndex);\n    } else if (selectedNavCoachIndex === this.currSelectedCoachIndex) {\n      this.updateCurrentSelectedCoach();\n      this.preselectPlaceInCoach();\n    }\n\n    //close all opened popovers\n    this._closePopover();\n  }\n\n  private _onFocusNavCoach(): void {\n    if (!this.preventCoachScrollByPlaceClick) {\n      this.preselectPlaceInCoach();\n    } else {\n      this.focusPlaceElement(this.currSelectedPlace);\n    }\n    this.isAutoScrolling = false;\n  }\n\n  private _getDescriptionTableCoach(coachItem: CoachItem, coachIndex: number): string {\n    if (!coachItem.places?.length) {\n      return getI18nSeatReservation('COACH_BLOCKED_TABLE_CAPTION', this._language.current, [\n        coachItem.id,\n      ]);\n    }\n\n    let tableCoachDescription: string;\n    const areaDescriptions = this._getTitleDescriptionListString(coachItem.graphicElements!);\n    const serviceDescriptions = this._getTitleDescriptionListString(coachItem.serviceElements!);\n\n    tableCoachDescription = getI18nSeatReservation('COACH_TABLE_CAPTION', this._language.current, [\n      coachItem.id,\n    ]);\n\n    if (!this.hasNavigation) {\n      // Expands the number of available seats and bicycle spaces as info\n      const freePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);\n      const freePlacesTxt = getI18nSeatReservation(\n        'COACH_AVAILABLE_NUMBER_OF_PLACES',\n        this._language.current,\n        [freePlaces.seats, freePlaces.bicycles],\n      );\n      tableCoachDescription = tableCoachDescription.concat('. ').concat(freePlacesTxt).concat('. ');\n    }\n\n    if (!!areaDescriptions || !!serviceDescriptions) {\n      tableCoachDescription +=\n        '. ' + getI18nSeatReservation('COACH_AVAILABLE_SERVICES', this._language.current) + ': ';\n      tableCoachDescription += serviceDescriptions + ', ' + areaDescriptions + '.';\n    }\n    return tableCoachDescription;\n  }\n\n  private _getTitleDescriptionListString(descriptionsElements: BaseElement[]): string {\n    const uniqueDescriptions: string[] = [];\n\n    return descriptionsElements\n      ?.map((descriptionElement) => {\n        const icon = descriptionElement.icon;\n        if (!icon) return null;\n\n        const descriptionAlreayExist = uniqueDescriptions.indexOf(icon) > -1;\n        const translation = getI18nSeatReservation(\n          descriptionElement.icon!,\n          this._language.current,\n        );\n        const isValidDescription =\n          this._notFixedRotatableAreaIcons.indexOf(icon) === -1 &&\n          this._notAreaElements.indexOf(icon) === -1;\n\n        if (!descriptionAlreayExist) {\n          uniqueDescriptions.push(descriptionElement.icon!);\n        }\n        return !!translation && !descriptionAlreayExist && isValidDescription ? translation : null;\n      })\n      .filter((description) => !!description)\n      .join(', ');\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-seat-reservation': SbbSeatReservationElement;\n  }\n}\n"],"names":[],"mappings":"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"}
|
|
903
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"seat-reservation.component.js","sources":["../../../../../src/elements-experimental/seat-reservation/seat-reservation/seat-reservation.component.ts"],"sourcesContent":["import { SbbLanguageController } from '@sbb-esta/lyne-elements/core/controllers.js';\nimport { html, isServer, nothing } from 'lit';\nimport type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { customElement } from 'lit/decorators.js';\nimport { classMap } from 'lit/directives/class-map.js';\nimport { styleMap } from 'lit/directives/style-map.js';\n\nimport { getI18nSeatReservation } from '../common.js';\nimport type {\n  CoachItem,\n  Place,\n  ElementDimension,\n  BaseElement,\n  PlaceSelection,\n  SeatReservation,\n} from '../common.js';\n\nimport { SeatReservationBaseElement } from './seat-reservation-base-element.js';\nimport style from './seat-reservation.scss?lit&inline';\n\nimport '@sbb-esta/lyne-elements/button.js';\nimport '@sbb-esta/lyne-elements/screen-reader-only.js';\nimport '../seat-reservation-area.js';\nimport '../seat-reservation-graphic.js';\nimport '../seat-reservation-place-control.js';\nimport '../seat-reservation-navigation-coach.js';\nimport '../seat-reservation-scoped.js';\nimport '@sbb-esta/lyne-elements/popover/popover.js';\n\n/**\n * Main component for the seat reservation.\n *\n */\nexport\n@customElement('sbb-seat-reservation')\nclass SbbSeatReservationElement extends SeatReservationBaseElement {\n  public static override styles: CSSResultGroup = style;\n\n  private _language = new SbbLanguageController(this);\n  private _coachesHtmlTemplate?: TemplateResult;\n  // Graphics that should not be rendered with an area\n  private _notAreaElements = [\n    'DRIVER_AREA',\n    'COACH_PASSAGE',\n    'COACH_WALL_NO_PASSAGE',\n    'COMPARTMENT_PASSAGE',\n    'COMPARTMENT_PASSAGE_HIGH',\n    'COMPARTMENT_PASSAGE_MIDDLE',\n    'COMPARTMENT_PASSAGE_LOW',\n  ];\n\n  // Area icons that should not be fixed during rotation when vertical mode is selected\n  private _notFixedRotatableAreaIcons = ['ENTRY_EXIT'];\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('hasNavigation')) {\n      if (this.hasNavigation) {\n        this.shadowRoot\n          ?.querySelectorAll('table')\n          .forEach((table) => table.removeAttribute('tabindex'));\n      } else {\n        this.shadowRoot\n          ?.querySelectorAll('table')\n          .forEach((table) => table.setAttribute('tabindex', '0'));\n      }\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    // We need to wait until the first update is complete to init different html element dimensions\n    this.updateComplete.then(() => {\n      this.initPrepairSeatReservationData();\n      this.initNavigationSelectionByScrollEvent();\n\n      // We have to initiate the render process manually so that the prepared data is applied directly\n      // and not only after an interaction with the component itself\n      this.requestUpdate();\n    });\n  }\n\n  protected override render(): TemplateResult | null {\n    if (!isServer) {\n      this._determineBaseFontSize();\n    }\n\n    this._initVehicleSeatReservationConstruction();\n    return this._coachesHtmlTemplate || null;\n  }\n\n  private _determineBaseFontSize(): void {\n    const baseFontSize = parseInt(window.getComputedStyle(document.body).fontSize, 10);\n    //calculate rem of 1px\n    const onePixelInRem = 1 / baseFontSize;\n    this.style?.setProperty('--sbb-seat-reservation-one-px-rem', `${onePixelInRem + 'rem'}`);\n  }\n\n  private _initVehicleSeatReservationConstruction(): void {\n    const classAlignVertical = this.alignVertical ? 'sbb-sr__grid--vertical' : '';\n    this._coachesHtmlTemplate = html`\n      <div class=\"sbb-sr__container\">\n        <div class=\"sbb-sr sbb-sr__grid ${classAlignVertical}\">\n          <div class=\"sbb-sr-navigation-first-grid\">\n            ${this._renderNavigationControlButton('DIRECTION_LEFT')}\n          </div>\n          <div\n            class=\"sbb-sr__component\"\n            @keydown=${(evt: KeyboardEvent) => this.handleKeyboardEvent(evt)}\n          >\n            <div class=\"sbb-sr-grid-inner\">\n              <div class=\"nav-grid\">${this._renderNavigation()}</div>\n              <div class=\"coaches-grid\">\n                <div class=\"sbb-sr__wrapper-coach-decks\">\n                  <div class=\"sbb-sr__wrapper-deck-labels\">${this._renderDeckLabels()}</div>\n                  <div id=\"sbb-sr__wrapper-scrollarea\" class=\"sbb-sr__wrapper\">\n                    <div id=\"sbb-sr__parent-area\" class=\"sbb-sr__parent\" tabindex=\"-1\">\n                      <ul\n                        class=\"${classMap({\n                          'sbb-sr__list-decks': true,\n                          'sbb-sr__list-decks--gap': this.hasMultipleDecks,\n                        })}\"\n                      >\n                        ${this.seatReservations?.map(\n                          (seatReservation: SeatReservation, index: number) => {\n                            return html`<li class=\"sbb-sr__list-item-deck\">\n                              <ul class=\"sbb-sr__list-coaches\" role=\"presentation\">\n                                ${this._renderCoaches(seatReservation, index)}\n                              </ul>\n                            </li>`;\n                          },\n                        )}\n                      </ul>\n                    </div>\n                  </div>\n                </div>\n              </div>\n            </div>\n          </div>\n          <div class=\"sbb-sr-navigation-last-grid\">\n            ${this._renderNavigationControlButton('DIRECTION_RIGHT')}\n          </div>\n        </div>\n      </div>\n    `;\n  }\n\n  private _renderDeckLabels(): TemplateResult[] | null {\n    if (!this.hasMultipleDecks) return null;\n\n    return this.seatReservations.map((seatReservation) => {\n      const deckDescription = getI18nSeatReservation(\n        seatReservation.deckCoachLevel,\n        this._language.current,\n      );\n      return html`<b aria-hidden=\"true\">${deckDescription}</b>`;\n    });\n  }\n\n  private _renderNavigationControlButton(btnDirection: string): TemplateResult | null {\n    if (!this.hasNavigation || !this.seatReservations) return null;\n\n    const btnId = btnDirection == 'DIRECTION_RIGHT' ? 'last-tab-element' : 'first-tab-element';\n    const btnIcon = btnDirection == 'DIRECTION_RIGHT' ? 'arrow-right-small' : 'arrow-left-small';\n    const btnAriaDescription =\n      btnDirection == 'DIRECTION_RIGHT'\n        ? getI18nSeatReservation('SEAT_RESERVATION_END', this._language.current)\n        : getI18nSeatReservation('SEAT_RESERVATION_BEGIN', this._language.current);\n    let btnDisabled = true;\n\n    if (btnDirection == 'DIRECTION_LEFT' && this.selectedCoachIndex > 0) {\n      btnDisabled = false;\n    } else if (\n      btnDirection == 'DIRECTION_RIGHT' &&\n      this.selectedCoachIndex <\n        this.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1\n    ) {\n      btnDisabled = false;\n    }\n\n    return html`<sbb-secondary-button\n      @click=\"${() => this.navigateByDirectionBtn(btnDirection)}\"\n      id=\"${btnId}\"\n      class=\"sbb-sr__navigation-control-button\"\n      size=\"s\"\n      icon-name=\"${btnIcon}\"\n      type=\"button\"\n      aria-label=\"${btnAriaDescription}\"\n      role=\"contentinfo\"\n      .disabledInteractive=\"${btnDisabled || nothing}\"\n    ></sbb-secondary-button>`;\n  }\n\n  private _renderNavigation(): TemplateResult | null {\n    if (!this.hasNavigation || !this.seatReservations) return null;\n\n    return html`<div class=\"sbb-sr-navigation-wrapper\">\n      <nav id=\"sbb-sr-navigation\" class=\"sbb-sr-navigation\">\n        <ul\n          id=\"sbb-sr__navigation-list-coaches\"\n          class=\"sbb-sr-navigation__list-coaches\"\n          aria-label=\"${getI18nSeatReservation(\n            'SEAT_RESERVATION_NAVIGATION',\n            this._language.current,\n          )}\"\n        >\n          ${this.seatReservations[this.currSelectedDeckIndex]?.coachItems.map(\n            (coachItem: CoachItem, index: number) => {\n              const coachFreePlaces = this.getAvailableFreePlacesNumFromCoach(index);\n              return html`<li>\n                <sbb-seat-reservation-navigation-coach\n                  @selectcoach=${(event: CustomEvent<number>) => this._onSelectNavCoach(event)}\n                  @focuscoach=${() => this._onFocusNavCoach()}\n                  class=\"${classMap({\n                    'sbb-sr__navigation-coach--hover-scroll':\n                      this.hoveredScrollCoachIndex === index,\n                  })}\"\n                  index=\"${index}\"\n                  coach-id=\"${coachItem.id}\"\n                  .freePlacesByType=\"${coachFreePlaces}\"\n                  .selected=${this.selectedCoachIndex === index}\n                  .focused=${this.focusedCoachIndex === index}\n                  .propertyIds=\"${coachItem.propertyIds}\"\n                  .travelClass=\"${coachItem.travelClass}\"\n                  ?driver-area=\"${!coachItem.places?.length}\"\n                  ?first=\"${coachItem?.driverAreaSide?.left}\"\n                  ?last=\"${coachItem?.driverAreaSide?.right}\"\n                  ?vertical=\"${this.alignVertical}\"\n                >\n                </sbb-seat-reservation-navigation-coach>\n              </li>`;\n            },\n          )}\n        </ul>\n      </nav>\n    </div>`;\n  }\n  /**\n   *\n   * @param coaches\n   * @returns\n   */\n  private _renderCoaches(\n    seatReservation: SeatReservation,\n    deckIndex: number,\n  ): TemplateResult[] | null {\n    const coaches = JSON.parse(JSON.stringify(seatReservation?.coachItems));\n\n    if (!coaches) {\n      return null;\n    }\n    return coaches.map((coachItem: CoachItem, coachIndex: number) => {\n      return html`\n        <li class=\"sbb-sr__item-coach\">\n          ${this._renderCoachElement(coachItem, coachIndex, deckIndex)}\n        </li>\n      `;\n    });\n  }\n\n  private _renderCoachElement(\n    coachItem: CoachItem,\n    coachIndex: number,\n    coachDeckIndex: number,\n  ): TemplateResult {\n    const calculatedCoachDimension = this.getCalculatedDimension(coachItem.dimension);\n    const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem, coachIndex);\n\n    return html`<sbb-seat-reservation-scoped\n      style=${styleMap({\n        '--sbb-seat-reservation-scoped-width': calculatedCoachDimension.w,\n        '--sbb-seat-reservation-scoped-height': calculatedCoachDimension.h,\n      })}\n    >\n      ${this._getRenderedCoachBorders(coachItem)}\n      ${this._getRenderedGraphicalElements(\n        coachItem.graphicElements || [],\n        coachItem.dimension,\n        coachIndex,\n        coachDeckIndex,\n      )}\n      ${this._getRenderedServiceElements(coachIndex, coachDeckIndex, coachItem.serviceElements)}\n\n      <table\n        @focus=${() => this.onFocusTableCoachAndPreselectPlace(coachIndex)}\n        id=\"sbb-sr-coach-${coachIndex}\"\n        class=\"sbb-sr-coach-wrapper__table\"\n        aria-describedby=\"sbb-sr-coach-caption-${coachIndex}\"\n      >\n        <caption id=\"sbb-sr-coach-caption-${coachIndex}\" tabindex=\"-1\">\n          <sbb-screen-reader-only>${descriptionTableCoachWithServices}</sbb-screen-reader-only>\n        </caption>\n        ${this._getRenderedRowPlaces(coachItem, coachIndex, coachDeckIndex)}\n      </table>\n    </sbb-seat-reservation-scoped>`;\n  }\n\n  /**\n   * @returns Returns the border graphic (COACH_BORDER_MIDDLE) of a coach with calculated border gap and coach width,\n   * depending on whether the coach is with a driver area or without.\n   */\n  private _getRenderedCoachBorders(coachItem: CoachItem): TemplateResult {\n    const COACH_PASSAGE_WIDTH = 1;\n    const allElements = coachItem.graphicElements;\n    const driverArea = allElements?.find((element: BaseElement) => element.icon === 'DRIVER_AREA');\n    const borderWidth = driverArea\n      ? coachItem.dimension.w - driverArea.dimension.w - COACH_PASSAGE_WIDTH\n      : coachItem.dimension.w - COACH_PASSAGE_WIDTH * 2;\n    const borderHeight = (coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;\n    const borderOffsetX =\n      driverArea && driverArea.position.x === 0\n        ? driverArea?.dimension.w * this.baseGridSize\n        : this.baseGridSize;\n    return html`\n      <sbb-seat-reservation-graphic\n        style=${styleMap({\n          '--sbb-seat-reservation-graphic-width': borderWidth * this.baseGridSize,\n          '--sbb-seat-reservation-graphic-height': borderHeight,\n          '--sbb-seat-reservation-graphic-top': this.coachBorderPadding * -1,\n          '--sbb-seat-reservation-graphic-left': borderOffsetX,\n          '--sbb-seat-reservation-graphic-position': 'absolute',\n        })}\n        name=\"COACH_BORDER_MIDDLE\"\n        ?stretch=${true}\n        role=\"presentation\"\n      ></sbb-seat-reservation-graphic>\n    `;\n  }\n\n  private _getRenderedRowPlaces(\n    coach: CoachItem,\n    coachIndex: number,\n    deckIndex: number,\n  ): TemplateResult[] | null {\n    if (!coach.places) {\n      return null;\n    }\n\n    // Prepair rows with the places to render a table\n    const tableRowPlaces: Record<number, Place[]> = {};\n    for (const place of coach.places) {\n      if (!tableRowPlaces[place.position.y]) {\n        tableRowPlaces[place.position.y] = [place];\n      } else {\n        tableRowPlaces[place.position.y].push(place);\n      }\n    }\n\n    return Object.values(tableRowPlaces)\n      .map((rowPlaces: Place[], index) => {\n        return html`\n          <tr id=\"row-${coachIndex}-${rowPlaces[0].position.y}\" data-row-index=${index}>\n            ${this._getRenderedColumnPlaces(rowPlaces, coachIndex, deckIndex)}\n          </tr>\n        `;\n      })\n      .flatMap((rowTemplate) => rowTemplate);\n  }\n\n  private _getRenderedColumnPlaces(\n    places: Place[],\n    coachIndex: number,\n    deckIndex: number,\n  ): TemplateResult[] | null {\n    //Sorts each place by its ascending x coordinate\n    places.sort(\n      (placeA: Place, placeB: Place) => Number(placeA.position.x) - Number(placeB.position.x),\n    );\n\n    return places?.map((place: Place, index: number) => {\n      const calculatedDimension = this.getCalculatedDimension(place.dimension);\n      const calculatedPosition = this.getCalculatedPosition(place.position);\n      const textRotation = this.alignVertical ? -90 : 0;\n\n      return html`\n        <td\n          id=\"cell-${deckIndex}-${coachIndex}-${place.position.y}-${index}\"\n          class=\"graphical-element\"\n        >\n          <sbb-seat-reservation-place-control\n            style=${styleMap({\n              '--sbb-seat-reservation-place-control-width': calculatedDimension.w,\n              '--sbb-seat-reservation-place-control-height': calculatedDimension.h,\n              '--sbb-seat-reservation-place-control-top': calculatedPosition.y,\n              '--sbb-seat-reservation-place-control-left': calculatedPosition.x,\n            })}\n            @selectplace=${(selectPlaceEvent: CustomEvent<PlaceSelection>) =>\n              this._onSelectPlace(selectPlaceEvent)}\n            exportparts=\"sbb-sr-place-part\"\n            id=\"seat-reservation__place-button-${deckIndex}-${coachIndex}-${place.number}\"\n            class=\"seat-reservation-place-control\"\n            data-cell-id=\"${coachIndex}-${place.position.y}-${index}\"\n            text=${place.number}\n            type=${place.type}\n            state=${place.state}\n            width=${place.dimension.w * this.baseGridSize}\n            height=${place.dimension.h * this.baseGridSize}\n            rotation=${place.rotation ?? nothing}\n            text-rotation=${textRotation}\n            coach-index=${coachIndex}\n            deck-index=${deckIndex}\n            .propertyIds=${place.propertyIds}\n            .preventClick=${this.preventPlaceClick}\n          ></sbb-seat-reservation-place-control>\n        </td>\n      `;\n    });\n  }\n\n  private _getRenderedGraphicalElements(\n    graphicalElements: BaseElement[],\n    coachDimension: ElementDimension,\n    coachIndex: number,\n    coachDeckIndex: number,\n  ): TemplateResult[] | null {\n    if (!graphicalElements) return null;\n\n    return graphicalElements?.map((graphicalElement: BaseElement) => {\n      const icon = graphicalElement.icon ?? '';\n      const elementRotation = graphicalElement.rotation || 0;\n      const isNotFixedRotationGraphicalElement =\n        this._notFixedRotatableAreaIcons.indexOf(graphicalElement.icon!) === -1;\n      const elementFixedRotation =\n        this.alignVertical && isNotFixedRotationGraphicalElement\n          ? elementRotation - 90\n          : elementRotation;\n\n      //check if the current element is not an area element, since this element is drawn without an area component\n      if (this._notAreaElements.findIndex((notAreaElement) => notAreaElement === icon) > -1) {\n        return this._getRenderElementWithoutArea(graphicalElement, elementRotation, coachDimension);\n      }\n      return this._getRenderElementWithArea(\n        graphicalElement,\n        elementFixedRotation,\n        coachDimension,\n        coachIndex,\n        coachDeckIndex,\n      );\n    });\n  }\n\n  /**\n   * creates a rendered element with an area component\n   * @param graphicalElement\n   * @param rotation\n   * @param coachDimension\n   * @param coachIndex used to generate a unique id for the popover trigger\n   * @private\n   */\n  private _getRenderElementWithArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n    coachIndex: number,\n    coachDeckIndex: number,\n  ): TemplateResult {\n    // for TABLE, we use the area component itself to display the table instead of the SVG graphic.\n    // Due to different heights and widths, it wouldn't show correctly. To correct this, we would\n    // need difficult calculations for position, rotation and dimension.\n    const isNotTableGraphic = graphicalElement.icon?.indexOf('TABLE') === -1;\n    const areaProperty = graphicalElement.icon && isNotTableGraphic ? graphicalElement.icon : null;\n    const stretchHeight = areaProperty !== 'ENTRY_EXIT';\n    const ariaLabelForArea = graphicalElement.icon\n      ? getI18nSeatReservation(graphicalElement.icon, this._language.current)\n      : nothing;\n    const calculatedDimension = this.getCalculatedDimension(\n      graphicalElement.dimension,\n      coachDimension,\n      true,\n      stretchHeight,\n    );\n    const calculatedPosition = this.getCalculatedPosition(\n      graphicalElement.position,\n      graphicalElement.dimension,\n      coachDimension,\n      true,\n    );\n\n    //generate unique index number for the trigger element\n    const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;\n    let elementMounting = 'free';\n\n    if (graphicalElement.position.y === this.coachBorderOffset * -1) {\n      elementMounting = 'upper-border';\n    } else if (\n      graphicalElement.position.y + graphicalElement.dimension.h ===\n      coachDimension.h + this.coachBorderOffset\n    ) {\n      elementMounting = 'lower-border';\n    }\n\n    return html`\n      <sbb-seat-reservation-area\n        id=\"${triggerId}\"\n        class=\"${classMap({ 'sbb-seat-reservation-area--cursor-pointer': areaProperty !== null })}\"\n        style=${styleMap({\n          '--sbb-seat-reservation-area-width': calculatedDimension.w,\n          '--sbb-seat-reservation-area-height': calculatedDimension.h,\n          '--sbb-seat-reservation-area-top': calculatedPosition.y,\n          '--sbb-seat-reservation-area-left': calculatedPosition.x,\n        })}\n        mounting=${elementMounting}\n        background=\"dark\"\n        aria-hidden=\"true\"\n      >\n        ${areaProperty\n          ? html`\n              <sbb-seat-reservation-graphic\n                name=${areaProperty}\n                width=${this.baseGridSize}\n                height=${this.baseGridSize}\n                rotation=${rotation}\n                role=\"img\"\n                aria-hidden=\"true\"\n              ></sbb-seat-reservation-graphic>\n            `\n          : nothing}\n      </sbb-seat-reservation-area>\n      ${areaProperty ? this._popover(triggerId, ariaLabelForArea) : nothing}\n    `;\n  }\n\n  private _getRenderElementWithoutArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n  ): TemplateResult {\n    const calculatedDimension = this.getCalculatedDimension(\n      graphicalElement.dimension,\n      coachDimension,\n    );\n    const calculatedPosition = this.getCalculatedPosition(\n      graphicalElement.position,\n      graphicalElement.dimension,\n      coachDimension,\n    );\n\n    // If the icon is the driver area, then here concat the vehicle type to get the right vehicle chassie icon\n    const icon =\n      graphicalElement.icon && graphicalElement.icon.indexOf('DRIVER_AREA') === -1\n        ? graphicalElement.icon\n        : graphicalElement.icon?.concat(\n            '_',\n            this.seatReservations[this.currSelectedDeckIndex].vehicleType,\n          );\n\n    return html` <sbb-seat-reservation-graphic\n      style=${styleMap({\n        '--sbb-seat-reservation-graphic-width': calculatedDimension.w,\n        '--sbb-seat-reservation-graphic-height': calculatedDimension.h,\n        '--sbb-seat-reservation-graphic-top': calculatedPosition.y,\n        '--sbb-seat-reservation-graphic-left': calculatedPosition.x,\n        '--sbb-seat-reservation-graphic-position': 'absolute',\n      })}\n      name=${icon ?? nothing}\n      rotation=${rotation}\n      aria-hidden=\"true\"\n      ?stretch=${true}\n    ></sbb-seat-reservation-graphic>`;\n  }\n\n  private _getRenderedServiceElements(\n    coachIndex: number,\n    coachDeckIndex: number,\n    serviceElements?: BaseElement[],\n  ): TemplateResult[] | null {\n    if (!serviceElements) return null;\n\n    return serviceElements?.map((serviceElement: BaseElement) => {\n      const titleDescription = serviceElement.icon\n        ? getI18nSeatReservation(serviceElement.icon, this._language.current)\n        : null;\n      const calculatedDimension = this.getCalculatedDimension(serviceElement.dimension);\n      const calculatedPosition = this.getCalculatedPosition(serviceElement.position);\n      const elementRotation = serviceElement.rotation || 0;\n      const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;\n\n      //generate unique index number for the trigger element\n      const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;\n\n      return html`\n        <sbb-seat-reservation-graphic\n          id=\"${triggerId}\"\n          style=${styleMap({\n            '--sbb-seat-reservation-graphic-width': calculatedDimension.w,\n            '--sbb-seat-reservation-graphic-height': calculatedDimension.h,\n            '--sbb-seat-reservation-graphic-top': calculatedPosition.y,\n            '--sbb-seat-reservation-graphic-left': calculatedPosition.x,\n            '--sbb-seat-reservation-graphic-position': 'absolute',\n          })}\n          class=\"sbb-seat-reservation-graphic--cursor-pointer\"\n          name=${serviceElement.icon ?? nothing}\n          .rotation=${elementFixedRotation}\n          role=\"img\"\n          aria-hidden=\"true\"\n        ></sbb-seat-reservation-graphic>\n        ${this._popover(triggerId, titleDescription)}\n      `;\n    });\n  }\n\n  /**\n   * Manages the selected place event triggered from the place\n   * Each selection emits an array of all selected places\n   * @param selectPlaceEvent\n   */\n  private _onSelectPlace(selectPlaceEvent: CustomEvent<PlaceSelection>): void {\n    const selectedPlace = selectPlaceEvent.detail;\n    // We have to set preventCoachScrollByPlaceClick to true, to prevent automatic scrolling to the new focused place\n    this.preventCoachScrollByPlaceClick = true;\n    this.isCoachGridFocusable = false;\n    if (!this.preventPlaceClick) {\n      // Set current deck index if a places was selected\n      this.currSelectedDeckIndex = selectedPlace.deckIndex;\n      // Add place to place collection\n      this.updateSelectedSeatReservationPlaces(selectedPlace);\n      this.updateCurrentSelectedPlaceInCoach(selectedPlace);\n    }\n  }\n\n  private _onSelectNavCoach(event: CustomEvent<number>): void {\n    const selectedNavCoachIndex = event.detail;\n    this.isKeyboardNavigation = false;\n\n    if (selectedNavCoachIndex !== null && selectedNavCoachIndex !== this.currSelectedCoachIndex) {\n      this.unfocusPlaceElement();\n      this.scrollToSelectedNavCoach(selectedNavCoachIndex);\n    } else if (selectedNavCoachIndex === this.currSelectedCoachIndex) {\n      this.updateCurrentSelectedCoach();\n      this.preselectPlaceInCoach();\n    }\n\n    //close all opened popovers\n    this._closePopover();\n  }\n\n  private _onFocusNavCoach(): void {\n    if (!this.preventCoachScrollByPlaceClick) {\n      this.preselectPlaceInCoach();\n    } else {\n      this.focusPlaceElement(this.currSelectedPlace);\n    }\n    this.isAutoScrolling = false;\n  }\n\n  /**\n   * Creates a popover for extra service information\n   * @param triggerId\n   * @param popoverContent\n   * @private\n   */\n  private _popover(\n    triggerId: string,\n    popoverContent: string | null | typeof nothing,\n  ): TemplateResult {\n    return html`\n      <sbb-popover trigger=\"${triggerId}\">\n        <p class=\"sbb-text-s sbb-sr-popover\">${popoverContent}</p>\n      </sbb-popover>\n    `;\n  }\n\n  /**\n   * trigger to close all opened popovers (normally only one is opened at a time)\n   * @private\n   */\n  private _closePopover(): void {\n    this.shadowRoot\n      ?.querySelectorAll('sbb-popover[data-state=\"opened\"]')\n      .forEach((popover) => popover.setAttribute('data-state', 'closed'));\n  }\n\n  private _getDescriptionTableCoach(coachItem: CoachItem, coachIndex: number): string {\n    if (!coachItem.places?.length) {\n      return getI18nSeatReservation('COACH_BLOCKED_TABLE_CAPTION', this._language.current, [\n        coachItem.id,\n      ]);\n    }\n\n    let tableCoachDescription: string;\n    const areaDescriptions = this._getTitleDescriptionListString(coachItem.graphicElements!);\n    const serviceDescriptions = this._getTitleDescriptionListString(coachItem.serviceElements!);\n\n    tableCoachDescription = getI18nSeatReservation('COACH_TABLE_CAPTION', this._language.current, [\n      coachItem.id,\n    ]);\n\n    if (!this.hasNavigation) {\n      // Expands the number of available seats and bicycle spaces as info\n      const freePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);\n      const freePlacesTxt = getI18nSeatReservation(\n        'COACH_AVAILABLE_NUMBER_OF_PLACES',\n        this._language.current,\n        [freePlaces.seats, freePlaces.bicycles],\n      );\n      tableCoachDescription = tableCoachDescription.concat('. ').concat(freePlacesTxt).concat('. ');\n    }\n\n    if (!!areaDescriptions || !!serviceDescriptions) {\n      tableCoachDescription +=\n        '. ' + getI18nSeatReservation('COACH_AVAILABLE_SERVICES', this._language.current) + ': ';\n      tableCoachDescription += serviceDescriptions + ', ' + areaDescriptions + '.';\n    }\n    return tableCoachDescription;\n  }\n\n  private _getTitleDescriptionListString(descriptionsElements: BaseElement[]): string {\n    const uniqueDescriptions: string[] = [];\n\n    return descriptionsElements\n      ?.map((descriptionElement) => {\n        const icon = descriptionElement.icon;\n        if (!icon) return null;\n\n        const descriptionAlreayExist = uniqueDescriptions.indexOf(icon) > -1;\n        const translation = getI18nSeatReservation(\n          descriptionElement.icon!,\n          this._language.current,\n        );\n        const isValidDescription =\n          this._notFixedRotatableAreaIcons.indexOf(icon) === -1 &&\n          this._notAreaElements.indexOf(icon) === -1;\n\n        if (!descriptionAlreayExist) {\n          uniqueDescriptions.push(descriptionElement.icon!);\n        }\n        return !!translation && !descriptionAlreayExist && isValidDescription ? translation : null;\n      })\n      .filter((description) => !!description)\n      .join(', ');\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-seat-reservation': SbbSeatReservationElement;\n  }\n}\n"],"names":[],"mappings":"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"}
|