@sbb-esta/lyne-elements-experimental 3.0.1 → 3.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (64) hide show
  1. package/custom-elements.json +908 -567
  2. package/development/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
  3. package/development/seat-reservation/common/mapper/icon-mapper.js +1 -3
  4. package/development/seat-reservation/common/mapper/mapper.d.ts +4 -3
  5. package/development/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
  6. package/development/seat-reservation/common/mapper/mapper.js +6 -4
  7. package/development/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
  8. package/development/seat-reservation/common/mapper/seat-reservation-sample-data.js +4989 -4
  9. package/development/seat-reservation/common/svgs.d.ts +0 -1
  10. package/development/seat-reservation/common/svgs.d.ts.map +1 -1
  11. package/development/seat-reservation/common/svgs.js +1 -3
  12. package/development/seat-reservation/common/translations/i18n.js +194 -194
  13. package/development/seat-reservation/common/types.d.ts +10 -0
  14. package/development/seat-reservation/common/types.d.ts.map +1 -1
  15. package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +38 -12
  16. package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
  17. package/development/seat-reservation/seat-reservation/seat-reservation-base-element.js +252 -122
  18. package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts +28 -13
  19. package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
  20. package/development/seat-reservation/seat-reservation/seat-reservation.component.js +481 -262
  21. package/development/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +17 -10
  22. package/development/seat-reservation/seat-reservation-graphic/seat-reservation-assets.d.ts.map +1 -1
  23. package/development/seat-reservation/seat-reservation-graphic/seat-reservation-assets.js +1 -6
  24. package/development/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +36 -18
  25. package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts +3 -1
  26. package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts.map +1 -1
  27. package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +26 -4
  28. package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts.map +1 -1
  29. package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.js +86 -54
  30. package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts +1 -16
  31. package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts.map +1 -1
  32. package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.js +10 -164
  33. package/package.json +3 -3
  34. package/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
  35. package/seat-reservation/common/mapper/icon-mapper.js +0 -2
  36. package/seat-reservation/common/mapper/mapper.d.ts +4 -3
  37. package/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
  38. package/seat-reservation/common/mapper/mapper.js +16 -14
  39. package/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
  40. package/seat-reservation/common/mapper/seat-reservation-sample-data.js +4988 -3
  41. package/seat-reservation/common/svgs.d.ts +0 -1
  42. package/seat-reservation/common/svgs.d.ts.map +1 -1
  43. package/seat-reservation/common/svgs.js +4 -11
  44. package/seat-reservation/common/translations/i18n.js +197 -197
  45. package/seat-reservation/common/types.d.ts +10 -0
  46. package/seat-reservation/common/types.d.ts.map +1 -1
  47. package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +38 -12
  48. package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
  49. package/seat-reservation/seat-reservation/seat-reservation-base-element.js +208 -134
  50. package/seat-reservation/seat-reservation/seat-reservation.component.d.ts +28 -13
  51. package/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
  52. package/seat-reservation/seat-reservation/seat-reservation.component.js +290 -298
  53. package/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +22 -22
  54. package/seat-reservation/seat-reservation-graphic/seat-reservation-assets.d.ts.map +1 -1
  55. package/seat-reservation/seat-reservation-graphic/seat-reservation-assets.js +0 -5
  56. package/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +37 -37
  57. package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts +3 -1
  58. package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts.map +1 -1
  59. package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +69 -58
  60. package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts.map +1 -1
  61. package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.js +28 -27
  62. package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts +1 -16
  63. package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts.map +1 -1
  64. package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.js +14 -101
@@ -1,25 +1,19 @@
1
- var __typeError = (msg) => {
2
- throw TypeError(msg);
3
- };
4
- var __accessCheck = (obj, member, msg) => member.has(obj) || __typeError("Cannot " + msg);
5
- var __privateGet = (obj, member, getter) => (__accessCheck(obj, member, "read from private field"), getter ? getter.call(obj) : member.get(obj));
6
- var __privateAdd = (obj, member, value) => member.has(obj) ? __typeError("Cannot add the same private member more than once") : member instanceof WeakSet ? member.add(obj) : member.set(obj, value);
7
- var __privateSet = (obj, member, value, setter) => (__accessCheck(obj, member, "write to private field"), setter ? setter.call(obj, value) : member.set(obj, value), value);
8
- import { __runInitializers, __esDecorate } from "tslib";
1
+ import { __esDecorate, __runInitializers } from "tslib";
9
2
  import { SbbLanguageController } from "@sbb-esta/lyne-elements/core/controllers.js";
10
- import { forceType } from "@sbb-esta/lyne-elements/core/decorators.js";
11
- import { css, html, nothing } from "lit";
12
- import { customElement, property } from "lit/decorators.js";
3
+ import { css, isServer, html, nothing } from "lit";
4
+ import { customElement } from "lit/decorators.js";
13
5
  import { classMap } from "lit/directives/class-map.js";
14
6
  import { styleMap } from "lit/directives/style-map.js";
15
7
  import { getI18nSeatReservation } from "../common.js";
16
8
  import { SeatReservationBaseElement } from "./seat-reservation-base-element.js";
9
+ import "@sbb-esta/lyne-elements/button.js";
17
10
  import "@sbb-esta/lyne-elements/screen-reader-only.js";
18
11
  import "../seat-reservation-area.js";
19
12
  import "../seat-reservation-graphic.js";
20
13
  import "../seat-reservation-place-control.js";
21
14
  import "../seat-reservation-navigation-coach.js";
22
15
  import "../seat-reservation-scoped.js";
16
+ import "@sbb-esta/lyne-elements/popover/popover.js";
23
17
  const style = css`*,
24
18
  ::before,
25
19
  ::after {
@@ -28,8 +22,9 @@ const style = css`*,
28
22
 
29
23
  :host {
30
24
  --sbb-seat-reservation-grid-size: 16px;
31
- --sbb-seat-reservation-vertical-offset: 0;
25
+ --sbb-seat-reservation-coach-height: 0;
32
26
  display: block;
27
+ height: inherit;
33
28
  }
34
29
  :host ::part(coach-floor) {
35
30
  fill: var(--sbb-color-white);
@@ -40,29 +35,127 @@ const style = css`*,
40
35
  }
41
36
  }
42
37
 
43
- .sbb-sr__wrapper--vertical {
44
- width: fit-content;
45
- position: relative;
46
- inset-inline-start: var(--sbb-seat-reservation-vertical-offset);
47
- rotate: 90deg;
48
- transform-origin: top left;
38
+ :host(:dir(rtl)) :not(.sbb-sr-place-ctrl--type-bicycle,
39
+ .sbb-sr-place-ctrl--orientation-90,
40
+ .sbb-sr-place-ctrl--orientation-270,
41
+ .sbb-sr-place-ctrl--state-selected) ::part(coach-floor) {
42
+ rotate: calc(180 * 1deg);
49
43
  }
50
44
 
51
- .sbb-sr__wrapper--vertical .sbb-sr__parent {
52
- overflow-y: hidden;
45
+ [popover]:where(sbb-popover) {
46
+ margin: 0;
47
+ padding: 0;
48
+ border: none;
49
+ width: auto;
50
+ height: auto;
51
+ background-color: transparent;
52
+ color: inherit;
53
+ pointer-events: none;
53
54
  }
54
55
 
55
- .sbb-sr__wrapper--vertical .sbb-sr__list-coaches {
56
- width: 100vw;
56
+ .sbb-sr__container {
57
+ display: flex;
58
+ height: inherit;
57
59
  }
58
60
 
59
- .sbb-sr-navigation--vertical {
60
- inset-inline-start: var(--sbb-seat-reservation-vertical-offset);
61
- margin-inline-start: 1.25rem;
61
+ .sbb-sr__grid {
62
+ display: grid;
63
+ grid-template-columns: 20% 1fr 20%;
64
+ grid-template-rows: auto auto;
65
+ }
66
+ .sbb-sr__grid .sbb-sr-navigation-first-grid,
67
+ .sbb-sr__grid .sbb-sr-navigation-last-grid {
68
+ grid-row-start: 1;
69
+ position: relative;
70
+ }
71
+ .sbb-sr__grid .sbb-sr-navigation-first-grid {
72
+ grid-column: 1/2;
73
+ }
74
+ .sbb-sr__grid .sbb-sr-navigation-last-grid {
75
+ grid-column: 3/4;
76
+ }
77
+ .sbb-sr__grid .sbb-sr__component {
78
+ display: flex;
79
+ grid-column: 1/4;
80
+ grid-row: 1/3;
81
+ }
82
+ .sbb-sr__grid .sbb-sr-grid-inner {
83
+ display: grid;
84
+ grid-template-columns: 20% 60% 20%;
85
+ grid-template-rows: auto 1fr;
86
+ grid-gap: 0;
87
+ }
88
+ .sbb-sr__grid .sbb-sr-grid-inner .nav-grid {
89
+ grid-column: 2/3;
90
+ grid-row: 1/2;
91
+ display: inherit;
92
+ }
93
+ .sbb-sr__grid .sbb-sr-grid-inner .coaches-grid {
94
+ grid-column: 1/4;
95
+ grid-row: 2/3;
96
+ display: inherit;
97
+ }
98
+ .sbb-sr__grid .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper {
99
+ overflow: hidden;
100
+ }
101
+
102
+ .sbb-sr__navigation-control-button {
62
103
  position: absolute;
104
+ z-index: 10;
63
105
  }
64
- .sbb-sr-navigation--vertical .sbb-sr-navigation__list-coaches {
65
- flex-direction: column;
106
+ .sbb-sr__navigation-control-button#first-tab-element {
107
+ inset-inline-end: var(--sbb-spacing-fixed-2x);
108
+ }
109
+ .sbb-sr__navigation-control-button#last-tab-element {
110
+ inset-inline-start: var(--sbb-spacing-fixed-2x);
111
+ }
112
+
113
+ .sbb-sr-navigation-wrapper {
114
+ overflow: hidden;
115
+ padding-block: 0 var(--sbb-spacing-fixed-8x);
116
+ }
117
+
118
+ .sbb-sr-navigation {
119
+ --sbb-scrollbar-thumb-width: 0.125rem;
120
+ --sbb-scrollbar-thumb-width-hover: 0.25rem;
121
+ --sbb-scrollbar-width-firefox: thin;
122
+ --sbb-scrollbar-color: var(--sbb-color-black-alpha-30);
123
+ --sbb-scrollbar-color-hover: var(--sbb-color-black-alpha-60);
124
+ --sbb-scrollbar-track-color: transparent;
125
+ --sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
126
+ }
127
+ .sbb-sr-navigation::-webkit-scrollbar {
128
+ width: var(--sbb-scrollbar-width);
129
+ height: var(--sbb-scrollbar-width);
130
+ background-color: var(--sbb-scrollbar-track-color, transparent);
131
+ }
132
+ .sbb-sr-navigation::-webkit-scrollbar-corner {
133
+ background-color: var(--sbb-scrollbar-track-color, transparent);
134
+ }
135
+ .sbb-sr-navigation::-webkit-scrollbar-thumb {
136
+ background-color: var(--sbb-scrollbar-color, currentcolor);
137
+ border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
138
+ border-radius: var(--sbb-border-radius-4x);
139
+ background-clip: padding-box;
140
+ }
141
+ .sbb-sr-navigation::-webkit-scrollbar-thumb:hover {
142
+ background-color: var(--sbb-scrollbar-color-hover, currentcolor);
143
+ border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
144
+ }
145
+ .sbb-sr-navigation::-webkit-scrollbar-button, .sbb-sr-navigation::-webkit-scrollbar-corner {
146
+ display: none;
147
+ }
148
+ @supports not selector(::-webkit-scrollbar) {
149
+ .sbb-sr-navigation {
150
+ scrollbar-width: var(--sbb-scrollbar-width-firefox);
151
+ scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
152
+ }
153
+ }
154
+ .sbb-sr-navigation {
155
+ --sbb-scrollbar-color: var(--sbb-color-white-alpha-0);
156
+ display: flex;
157
+ overflow: scroll hidden;
158
+ position: relative;
66
159
  }
67
160
 
68
161
  .sbb-sr-navigation__list-coaches {
@@ -71,70 +164,189 @@ const style = css`*,
71
164
  padding: 0;
72
165
  font-size: inherit;
73
166
  display: flex;
74
- justify-content: center;
75
167
  flex-wrap: nowrap;
76
168
  gap: var(--sbb-spacing-fixed-1x);
77
- min-height: 3.75rem;
169
+ padding: 0.125rem;
78
170
  }
79
171
 
172
+ .sbb-sr__parent {
173
+ --sbb-scrollbar-thumb-width: 0.125rem;
174
+ --sbb-scrollbar-thumb-width-hover: 0.25rem;
175
+ --sbb-scrollbar-width-firefox: thin;
176
+ --sbb-scrollbar-color: var(--sbb-color-black-alpha-30);
177
+ --sbb-scrollbar-color-hover: var(--sbb-color-black-alpha-60);
178
+ --sbb-scrollbar-track-color: transparent;
179
+ --sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
180
+ }
181
+ .sbb-sr__parent::-webkit-scrollbar {
182
+ width: var(--sbb-scrollbar-width);
183
+ height: var(--sbb-scrollbar-width);
184
+ background-color: var(--sbb-scrollbar-track-color, transparent);
185
+ }
186
+ .sbb-sr__parent::-webkit-scrollbar-corner {
187
+ background-color: var(--sbb-scrollbar-track-color, transparent);
188
+ }
189
+ .sbb-sr__parent::-webkit-scrollbar-thumb {
190
+ background-color: var(--sbb-scrollbar-color, currentcolor);
191
+ border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
192
+ border-radius: var(--sbb-border-radius-4x);
193
+ background-clip: padding-box;
194
+ }
195
+ .sbb-sr__parent::-webkit-scrollbar-thumb:hover {
196
+ background-color: var(--sbb-scrollbar-color-hover, currentcolor);
197
+ border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
198
+ }
199
+ .sbb-sr__parent::-webkit-scrollbar-button, .sbb-sr__parent::-webkit-scrollbar-corner {
200
+ display: none;
201
+ }
202
+ @supports not selector(::-webkit-scrollbar) {
203
+ .sbb-sr__parent {
204
+ scrollbar-width: var(--sbb-scrollbar-width-firefox);
205
+ scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
206
+ }
207
+ }
80
208
  .sbb-sr__parent {
81
209
  overflow-x: scroll;
82
- padding: 3rem 0;
210
+ padding-block: var(--sbb-spacing-fixed-2x) var(--sbb-spacing-fixed-4x);
83
211
  }
84
212
 
85
213
  .sbb-sr__list-coaches {
214
+ list-style: none;
215
+ margin: 0;
216
+ padding: 0;
217
+ font-size: inherit;
86
218
  display: flex;
87
219
  gap: 0.25rem;
88
- padding: 0;
89
- margin: 0;
90
- list-style: none;
91
220
  }
92
221
 
93
222
  .sbb-sr__item-coach {
94
- padding: 0;
95
- margin: 0;
96
223
  position: relative;
97
224
  }
98
225
 
226
+ .sbb-sr-popover {
227
+ margin: 0;
228
+ }
229
+
99
230
  .sbb-sr-coach-wrapper__table {
100
231
  outline: 0;
232
+ }
233
+
234
+ .sbb-sr.sbb-sr__grid--vertical {
235
+ grid-template-columns: calc((var(--sbb-seat-reservation-coach-height) + 28) * var(--sbb-seat-reservation-one-px-rem)) 1fr;
236
+ grid-template-rows: 20% 60% 20%;
237
+ }
238
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-first-grid {
239
+ grid-column: 2/3;
240
+ grid-row-start: 1;
241
+ }
242
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-first-grid #first-tab-element {
243
+ position: absolute;
244
+ inset-block-end: var(--sbb-spacing-fixed-4x);
245
+ inset-inline-start: var(--sbb-spacing-fixed-8x);
246
+ inset-inline-end: initial;
247
+ rotate: 90deg;
248
+ }
249
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-last-grid {
250
+ grid-column: 2/3;
251
+ grid-row: 3/4;
252
+ }
253
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-last-grid #last-tab-element {
254
+ inset-block-start: var(--sbb-spacing-fixed-4x);
255
+ inset-inline-start: var(--sbb-spacing-fixed-8x);
256
+ rotate: 90deg;
257
+ }
258
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-wrapper {
259
+ padding-block-end: initial;
260
+ padding-inline-start: var(--sbb-spacing-fixed-8x);
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);
265
+ }
266
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr__component {
267
+ grid-column: 1/3;
268
+ grid-row: 1/4;
269
+ }
270
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner {
271
+ grid-template-columns: calc((var(--sbb-seat-reservation-coach-height) + 28) * var(--sbb-seat-reservation-one-px-rem)) 1fr;
272
+ grid-template-rows: 20% 60% 20%;
273
+ }
274
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid {
275
+ grid-column: 2/3;
276
+ grid-row: 2/3;
277
+ height: 100%;
278
+ }
279
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid .sbb-sr-navigation {
280
+ display: block;
281
+ overflow: hidden scroll;
282
+ height: 100%;
283
+ }
284
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid .sbb-sr-navigation .sbb-sr-navigation__list-coaches {
285
+ flex-direction: column;
286
+ padding: 0.125rem 0.125rem;
287
+ }
288
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid {
289
+ grid-column: 1/2;
290
+ grid-row: 1/4;
291
+ overflow: hidden;
292
+ position: relative;
293
+ }
294
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent {
295
+ --sbb-scrollbar-thumb-width: 0.125rem;
296
+ --sbb-scrollbar-thumb-width-hover: 0.25rem;
297
+ --sbb-scrollbar-width-firefox: thin;
298
+ --sbb-scrollbar-color: var(--sbb-color-black-alpha-30);
299
+ --sbb-scrollbar-color-hover: var(--sbb-color-black-alpha-60);
300
+ --sbb-scrollbar-track-color: transparent;
301
+ --sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
302
+ }
303
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent::-webkit-scrollbar {
304
+ width: var(--sbb-scrollbar-width);
305
+ height: var(--sbb-scrollbar-width);
306
+ background-color: var(--sbb-scrollbar-track-color, transparent);
307
+ }
308
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent::-webkit-scrollbar-corner {
309
+ background-color: var(--sbb-scrollbar-track-color, transparent);
310
+ }
311
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent::-webkit-scrollbar-thumb {
312
+ background-color: var(--sbb-scrollbar-color, currentcolor);
313
+ border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
314
+ border-radius: var(--sbb-border-radius-4x);
315
+ background-clip: padding-box;
316
+ }
317
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent::-webkit-scrollbar-thumb:hover {
318
+ background-color: var(--sbb-scrollbar-color-hover, currentcolor);
319
+ border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
320
+ }
321
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent::-webkit-scrollbar-button, .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent::-webkit-scrollbar-corner {
322
+ display: none;
323
+ }
324
+ @supports not selector(::-webkit-scrollbar) {
325
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent {
326
+ scrollbar-width: var(--sbb-scrollbar-width-firefox);
327
+ scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
328
+ }
329
+ }
330
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent {
331
+ overflow: hidden scroll;
332
+ padding: 0 var(--sbb-spacing-fixed-4x) 0 0;
333
+ height: 100%;
334
+ }
335
+ .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__parent .sbb-sr__list-coaches {
336
+ rotate: 90deg;
337
+ transform-origin: calc((var(--sbb-seat-reservation-coach-height) / 2 - 3) * var(--sbb-seat-reservation-one-px-rem)) calc((var(--sbb-seat-reservation-coach-height) / 2 - 3) * var(--sbb-seat-reservation-one-px-rem));
101
338
  }`;
102
339
  let SbbSeatReservationElement = (() => {
103
- var _seatReservation_accessor_storage, _hasNavigation_accessor_storage, _alignVertical_accessor_storage, _maxReservations_accessor_storage, _preventPlaceClick_accessor_storage, _a;
340
+ var _a;
104
341
  let _classDecorators = [customElement("sbb-seat-reservation")];
105
342
  let _classDescriptor;
106
343
  let _classExtraInitializers = [];
107
344
  let _classThis;
108
345
  let _classSuper = SeatReservationBaseElement;
109
- let _seatReservation_decorators;
110
- let _seatReservation_initializers = [];
111
- let _seatReservation_extraInitializers = [];
112
- let _hasNavigation_decorators;
113
- let _hasNavigation_initializers = [];
114
- let _hasNavigation_extraInitializers = [];
115
- let _alignVertical_decorators;
116
- let _alignVertical_initializers = [];
117
- let _alignVertical_extraInitializers = [];
118
- let _maxReservations_decorators;
119
- let _maxReservations_initializers = [];
120
- let _maxReservations_extraInitializers = [];
121
- let _preventPlaceClick_decorators;
122
- let _preventPlaceClick_initializers = [];
123
- let _preventPlaceClick_extraInitializers = [];
124
346
  _a = class extends _classSuper {
125
347
  constructor() {
126
348
  super(...arguments);
127
- __privateAdd(this, _seatReservation_accessor_storage);
128
- __privateAdd(this, _hasNavigation_accessor_storage);
129
- __privateAdd(this, _alignVertical_accessor_storage);
130
- __privateAdd(this, _maxReservations_accessor_storage);
131
- __privateAdd(this, _preventPlaceClick_accessor_storage);
132
- __privateSet(this, _seatReservation_accessor_storage, __runInitializers(this, _seatReservation_initializers, null));
133
- __privateSet(this, _hasNavigation_accessor_storage, (__runInitializers(this, _seatReservation_extraInitializers), __runInitializers(this, _hasNavigation_initializers, true)));
134
- __privateSet(this, _alignVertical_accessor_storage, (__runInitializers(this, _hasNavigation_extraInitializers), __runInitializers(this, _alignVertical_initializers, false)));
135
- __privateSet(this, _maxReservations_accessor_storage, (__runInitializers(this, _alignVertical_extraInitializers), __runInitializers(this, _maxReservations_initializers, null)));
136
- __privateSet(this, _preventPlaceClick_accessor_storage, (__runInitializers(this, _maxReservations_extraInitializers), __runInitializers(this, _preventPlaceClick_initializers, false)));
137
- this._language = (__runInitializers(this, _preventPlaceClick_extraInitializers), new SbbLanguageController(this));
349
+ this._language = new SbbLanguageController(this);
138
350
  this._notAreaElements = [
139
351
  "DRIVER_AREA_FULL",
140
352
  "COACH_PASSAGE",
@@ -146,41 +358,6 @@ let SbbSeatReservationElement = (() => {
146
358
  ];
147
359
  this._notFixedRotatableAreaIcons = ["ENTRY_EXIT"];
148
360
  }
149
- /** The seat reservation object which contains all coaches and places */
150
- get seatReservation() {
151
- return __privateGet(this, _seatReservation_accessor_storage);
152
- }
153
- set seatReservation(value) {
154
- __privateSet(this, _seatReservation_accessor_storage, value);
155
- }
156
- /** The seat reservation navigation can be toggled by this property */
157
- get hasNavigation() {
158
- return __privateGet(this, _hasNavigation_accessor_storage);
159
- }
160
- set hasNavigation(value) {
161
- __privateSet(this, _hasNavigation_accessor_storage, value);
162
- }
163
- /** Controls the visual representation of seat reservation in a horizonal or vertical alignment */
164
- get alignVertical() {
165
- return __privateGet(this, _alignVertical_accessor_storage);
166
- }
167
- set alignVertical(value) {
168
- __privateSet(this, _alignVertical_accessor_storage, value);
169
- }
170
- /** Maximal number of possible clickable seats */
171
- get maxReservations() {
172
- return __privateGet(this, _maxReservations_accessor_storage);
173
- }
174
- set maxReservations(value) {
175
- __privateSet(this, _maxReservations_accessor_storage, value);
176
- }
177
- /** Any click functionality is prevented */
178
- get preventPlaceClick() {
179
- return __privateGet(this, _preventPlaceClick_accessor_storage);
180
- }
181
- set preventPlaceClick(value) {
182
- __privateSet(this, _preventPlaceClick_accessor_storage, value);
183
- }
184
361
  willUpdate(changedProperties) {
185
362
  super.willUpdate(changedProperties);
186
363
  if (changedProperties.has("hasNavigation")) {
@@ -193,70 +370,103 @@ let SbbSeatReservationElement = (() => {
193
370
  }
194
371
  firstUpdated(changedProperties) {
195
372
  super.firstUpdated(changedProperties);
196
- this._componentSetup();
373
+ this.updateComplete.then(() => {
374
+ this.initNavigationSelectionByScrollEvent();
375
+ });
197
376
  }
198
377
  render() {
378
+ if (!isServer) {
379
+ this._determineBaseFontSize();
380
+ }
199
381
  this._initVehicleSeatReservationConstruction();
200
382
  return this._coachesHtmlTemplate || null;
201
383
  }
202
- _componentSetup() {
203
- this.initNavigationSelectionByScrollEvent();
384
+ _determineBaseFontSize() {
385
+ const baseFontSize = parseInt(window.getComputedStyle(document.body).fontSize, 10);
386
+ const onePixelInRem = 1 / baseFontSize;
387
+ this.style?.setProperty("--sbb-seat-reservation-one-px-rem", `${onePixelInRem + "rem"}`);
204
388
  }
205
389
  _initVehicleSeatReservationConstruction() {
206
390
  const coachItems = JSON.parse(JSON.stringify(this.seatReservation?.coachItems));
207
- const classAlignVertical = this.alignVertical ? "sbb-sr__wrapper--vertical" : "";
391
+ const classAlignVertical = this.alignVertical ? "sbb-sr__grid--vertical" : "";
208
392
  this._coachesHtmlTemplate = html`
209
- <div>
210
- <sbb-screen-reader-only>
211
- <input
212
- id="first-tab-element"
213
- role="contentinfo"
214
- type="text"
215
- aria-label="${getI18nSeatReservation("SEAT_RESERVATION_BEGIN", this._language.current)}"
216
- readonly
217
- />
218
- </sbb-screen-reader-only>
219
-
220
- <div @keydown=${(evt) => this.handleKeyboardEvent(evt)}>
221
- ${this._renderNavigation()}
222
- <div class="sbb-sr__wrapper ${classAlignVertical}">
223
- <div id="sbb-sr__parent-area" class="sbb-sr__parent" tabindex="-1">
224
- <ul class="sbb-sr__list-coaches" role="presentation">
225
- ${this._renderCoaches(coachItems)}
226
- </ul>
393
+ <div class="sbb-sr__container">
394
+ <div class="sbb-sr sbb-sr__grid ${classAlignVertical}">
395
+ <div class="sbb-sr-navigation-first-grid">
396
+ ${this._renderNavigationControlButton("DIRECTION_LEFT")}
397
+ </div>
398
+ <div
399
+ class="sbb-sr__component"
400
+ @keydown=${(evt) => this.handleKeyboardEvent(evt)}
401
+ >
402
+ <div class="sbb-sr-grid-inner">
403
+ <div class="nav-grid">${this._renderNavigation()}</div>
404
+ <div class="coaches-grid">
405
+ <div class="sbb-sr__wrapper">
406
+ <div id="sbb-sr__parent-area" class="sbb-sr__parent" tabindex="-1">
407
+ <ul class="sbb-sr__list-coaches" role="presentation">
408
+ ${this._renderCoaches(coachItems)}
409
+ </ul>
410
+ </div>
411
+ </div>
412
+ </div>
227
413
  </div>
228
414
  </div>
415
+ <div class="sbb-sr-navigation-last-grid">
416
+ ${this._renderNavigationControlButton("DIRECTION_RIGHT")}
417
+ </div>
229
418
  </div>
230
-
231
- <sbb-screen-reader-only>
232
- <input
233
- id="last-tab-element"
234
- role="contentinfo"
235
- type="text"
236
- aria-label="${getI18nSeatReservation("SEAT_RESERVATION_END", this._language.current)}"
237
- readonly
238
- />
239
- </sbb-screen-reader-only>
240
419
  </div>
241
420
  `;
242
421
  }
422
+ _renderNavigationControlButton(btnDirection) {
423
+ if (!this.hasNavigation) {
424
+ return null;
425
+ }
426
+ const btnId = btnDirection == "DIRECTION_RIGHT" ? "last-tab-element" : "first-tab-element";
427
+ const btnIcon = btnDirection == "DIRECTION_RIGHT" ? "arrow-right-small" : "arrow-left-small";
428
+ const btnAriaDescription = btnDirection == "DIRECTION_RIGHT" ? getI18nSeatReservation("SEAT_RESERVATION_END", this._language.current) : getI18nSeatReservation("SEAT_RESERVATION_BEGIN", this._language.current);
429
+ let btnDisabled = true;
430
+ if (btnDirection == "DIRECTION_LEFT" && this.selectedCoachIndex > 0) {
431
+ btnDisabled = false;
432
+ } else if (btnDirection == "DIRECTION_RIGHT" && this.selectedCoachIndex < this.seatReservation.coachItems.length - 1) {
433
+ btnDisabled = false;
434
+ }
435
+ return html`<sbb-secondary-button
436
+ @click="${() => this.navigateByDirectionBtn(btnDirection)}"
437
+ id="${btnId}"
438
+ class="sbb-sr__navigation-control-button"
439
+ size="s"
440
+ icon-name="${btnIcon}"
441
+ type="button"
442
+ aria-label="${btnAriaDescription}"
443
+ role="contentinfo"
444
+ .disabledInteractive="${btnDisabled || nothing}"
445
+ ></sbb-secondary-button>`;
446
+ }
243
447
  _renderNavigation() {
244
448
  if (!this.hasNavigation) {
245
449
  return null;
246
450
  }
247
- return html`
248
- <nav class="${classMap({ "sbb-sr-navigation--vertical": this.alignVertical })}">
451
+ return html`<div class="sbb-sr-navigation-wrapper">
452
+ <nav id="sbb-sr-navigation" class="sbb-sr-navigation">
249
453
  <ul
454
+ id="sbb-sr__navigation-list-coaches"
250
455
  class="sbb-sr-navigation__list-coaches"
251
456
  aria-label="${getI18nSeatReservation("SEAT_RESERVATION_NAVIGATION", this._language.current)}"
252
457
  >
253
458
  ${this.seatReservation?.coachItems.map((coachItem, index) => {
459
+ const coachFreePlaces = this.getAvailableFreePlacesNumFromCoach(index);
254
460
  return html`<li>
255
461
  <sbb-seat-reservation-navigation-coach
256
462
  @selectcoach=${(event) => this._onSelectNavCoach(event)}
257
463
  @focuscoach=${() => this._onFocusNavCoach()}
464
+ class="${classMap({
465
+ "sbb-sr__navigation-coach--hover-scroll": this.hoveredScrollCoachIndex === index
466
+ })}"
258
467
  index="${index}"
259
468
  coach-id="${coachItem.id}"
469
+ .freePlacesByType="${coachFreePlaces}"
260
470
  .selected=${this.selectedCoachIndex === index}
261
471
  .focused=${this.focusedCoachIndex === index}
262
472
  .propertyIds="${coachItem.propertyIds}"
@@ -271,7 +481,7 @@ let SbbSeatReservationElement = (() => {
271
481
  })}
272
482
  </ul>
273
483
  </nav>
274
- `;
484
+ </div>`;
275
485
  }
276
486
  /**
277
487
  *
@@ -290,15 +500,17 @@ let SbbSeatReservationElement = (() => {
290
500
  }
291
501
  _renderCoachElement(coachItem, index) {
292
502
  const calculatedCoachDimension = this.getCalculatedDimension(coachItem.dimension);
293
- const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem);
294
- return html` <sbb-seat-reservation-scoped
295
- scoped-classes="coach-wrapper"
296
- height="${calculatedCoachDimension.h}px"
297
- width="${calculatedCoachDimension.w}px"
503
+ const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem, index);
504
+ return html`<sbb-seat-reservation-scoped
505
+ style=${styleMap({
506
+ "--sbb-seat-reservation-scoped-width": calculatedCoachDimension.w,
507
+ "--sbb-seat-reservation-scoped-height": calculatedCoachDimension.h
508
+ })}
298
509
  >
299
510
  ${this._getRenderedCoachBorders(coachItem, index)}
300
- ${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension)}
301
- ${this._getRenderedServiceElements(coachItem.serviceElements)}
511
+ ${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension, index)}
512
+ ${this._getRenderedServiceElements(index, coachItem.serviceElements)}
513
+
302
514
  <table
303
515
  @focus=${() => this.onFocusTableCoachAndPreselectPlace(index)}
304
516
  id="sbb-sr-coach-${index}"
@@ -312,26 +524,30 @@ let SbbSeatReservationElement = (() => {
312
524
  </table>
313
525
  </sbb-seat-reservation-scoped>`;
314
526
  }
527
+ /**
528
+ * @returns Returns the border graphic (COACH_BORDER_MIDDLE) of a coach with calculated border gap and coach width,
529
+ * depending on whether the coach is with a driver area or without.
530
+ */
315
531
  _getRenderedCoachBorders(coachItem, coachIndex) {
316
- const allElements = coachItem.graphicElements;
317
532
  const COACH_PASSAGE_WIDTH = 1;
533
+ const allElements = coachItem.graphicElements;
318
534
  const driverArea = allElements?.find((element) => element.icon === "DRIVER_AREA_FULL");
319
535
  const borderWidth = driverArea ? coachItem.dimension.w - driverArea.dimension.w - COACH_PASSAGE_WIDTH : coachItem.dimension.w - COACH_PASSAGE_WIDTH * 2;
536
+ const borderHeight = (coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;
320
537
  const borderOffsetX = coachIndex === 0 && driverArea ? driverArea?.dimension.w * this.baseGridSize : this.baseGridSize;
321
538
  return html`
322
- <sbb-seat-reservation-scoped
323
- scoped-classes="coach-border"
324
- inset-block-start="${this.coachBorderPadding * -1}px"
325
- inset-inline-start="${borderOffsetX}px"
326
- >
327
- <sbb-seat-reservation-graphic
328
- name="COACH_BORDER_MIDDLE"
329
- width=${borderWidth * this.baseGridSize}
330
- height=${(coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize}
331
- ?stretch=${true}
332
- role="presentation"
333
- ></sbb-seat-reservation-graphic>
334
- </sbb-seat-reservation-scoped>
539
+ <sbb-seat-reservation-graphic
540
+ style=${styleMap({
541
+ "--sbb-seat-reservation-graphic-width": borderWidth * this.baseGridSize,
542
+ "--sbb-seat-reservation-graphic-height": borderHeight,
543
+ "--sbb-seat-reservation-graphic-top": this.coachBorderPadding * -1,
544
+ "--sbb-seat-reservation-graphic-left": borderOffsetX,
545
+ "--sbb-seat-reservation-graphic-position": "absolute"
546
+ })}
547
+ name="COACH_BORDER_MIDDLE"
548
+ ?stretch=${true}
549
+ role="presentation"
550
+ ></sbb-seat-reservation-graphic>
335
551
  `;
336
552
  }
337
553
  _getRenderedRowPlaces(coach, coachIndex) {
@@ -357,20 +573,18 @@ let SbbSeatReservationElement = (() => {
357
573
  _getRenderedColumnPlaces(places, coachIndex) {
358
574
  places.sort((placeA, placeB) => Number(placeA.position.x) - Number(placeB.position.x));
359
575
  return places?.map((place, index) => {
360
- const calculatedInternalDimension = this.getCalculatedDimension(place.dimension);
361
- const calculatedInternalPosition = this.getCalculatedPosition(place.position);
576
+ const calculatedDimension = this.getCalculatedDimension(place.dimension);
577
+ const calculatedPosition = this.getCalculatedPosition(place.position);
362
578
  const textRotation = this.alignVertical ? -90 : 0;
363
579
  return html`
364
- <sbb-seat-reservation-scoped
365
- scoped-classes="graphical-element"
366
- inset-block-start="${calculatedInternalPosition.y}px"
367
- inset-inline-start="${calculatedInternalPosition.x}px"
368
- width="${calculatedInternalDimension.w}px"
369
- height="${calculatedInternalDimension.h}px"
370
- z-index="${place.position.z}"
371
- cell-id="cell-${coachIndex}-${place.position.y}-${index}"
372
- >
580
+ <td id="cell-${coachIndex}-${place.position.y}-${index}" class="graphical-element">
373
581
  <sbb-seat-reservation-place-control
582
+ style=${styleMap({
583
+ "--sbb-seat-reservation-place-control-width": calculatedDimension.w,
584
+ "--sbb-seat-reservation-place-control-height": calculatedDimension.h,
585
+ "--sbb-seat-reservation-place-control-top": calculatedPosition.y,
586
+ "--sbb-seat-reservation-place-control-left": calculatedPosition.x
587
+ })}
374
588
  @selectplace=${(selectPlaceEvent) => this._onSelectPlace(selectPlaceEvent)}
375
589
  exportparts="sbb-sr-place-part"
376
590
  id="seat-reservation__place-button-${coachIndex}-${place.number}"
@@ -387,11 +601,11 @@ let SbbSeatReservationElement = (() => {
387
601
  .propertyIds=${place.propertyIds}
388
602
  .preventClick=${this.preventPlaceClick}
389
603
  ></sbb-seat-reservation-place-control>
390
- </sbb-seat-reservation-scoped>
604
+ </td>
391
605
  `;
392
606
  });
393
607
  }
394
- _getRenderedGraphicalElements(graphicalElements, coachDimension) {
608
+ _getRenderedGraphicalElements(graphicalElements, coachDimension, coachIndex) {
395
609
  if (!graphicalElements) {
396
610
  return null;
397
611
  }
@@ -403,16 +617,25 @@ let SbbSeatReservationElement = (() => {
403
617
  if (this._notAreaElements.findIndex((notAreaElement) => notAreaElement === icon) > -1) {
404
618
  return this._getRenderElementWithoutArea(graphicalElement, elementRotation, coachDimension);
405
619
  }
406
- return this._getRenderElementWithArea(graphicalElement, elementFixedRotation, coachDimension);
620
+ return this._getRenderElementWithArea(graphicalElement, elementFixedRotation, coachDimension, coachIndex);
407
621
  });
408
622
  }
409
- _getRenderElementWithArea(graphicalElement, rotation, coachDimension) {
410
- const isNotTableGraphicTempFix = graphicalElement.icon?.indexOf("TABLE") === -1;
411
- const areaProperty = graphicalElement.icon && isNotTableGraphicTempFix ? graphicalElement.icon : null;
623
+ /**
624
+ * creates a rendered element with an area component
625
+ * @param graphicalElement
626
+ * @param rotation
627
+ * @param coachDimension
628
+ * @param coachIndex used to generate a unique id for the popover trigger
629
+ * @private
630
+ */
631
+ _getRenderElementWithArea(graphicalElement, rotation, coachDimension, coachIndex) {
632
+ const isNotTableGraphic = graphicalElement.icon?.indexOf("TABLE") === -1;
633
+ const areaProperty = graphicalElement.icon && isNotTableGraphic ? graphicalElement.icon : null;
412
634
  const stretchHeight = areaProperty !== "ENTRY_EXIT";
413
635
  const ariaLabelForArea = graphicalElement.icon ? getI18nSeatReservation(graphicalElement.icon, this._language.current) : nothing;
414
636
  const calculatedDimension = this.getCalculatedDimension(graphicalElement.dimension, coachDimension, true, stretchHeight);
415
637
  const calculatedPosition = this.getCalculatedPosition(graphicalElement.position, graphicalElement.dimension, coachDimension, true);
638
+ const triggerId = `popover-trigger-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
416
639
  let elementMounting = "free";
417
640
  if (graphicalElement.position.y === this.coachBorderOffset * -1) {
418
641
  elementMounting = "upper-border";
@@ -420,91 +643,101 @@ let SbbSeatReservationElement = (() => {
420
643
  elementMounting = "lower-border";
421
644
  }
422
645
  return html`
423
- <sbb-seat-reservation-scoped
424
- scoped-classes="graphical-element"
425
- inset-block-start="${calculatedPosition.y}px"
426
- inset-inline-start="${calculatedPosition.x}px"
427
- width="${calculatedDimension.w}px"
428
- height="${calculatedDimension.h}px"
429
- z-index="${graphicalElement.position.z}"
646
+ <sbb-seat-reservation-area
647
+ id="${triggerId}"
648
+ style=${styleMap({
649
+ "--sbb-seat-reservation-area-width": calculatedDimension.w,
650
+ "--sbb-seat-reservation-area-height": calculatedDimension.h,
651
+ "--sbb-seat-reservation-area-top": calculatedPosition.y,
652
+ "--sbb-seat-reservation-area-left": calculatedPosition.x
653
+ })}
654
+ mounting=${elementMounting}
655
+ background="dark"
656
+ aria-hidden="true"
430
657
  >
431
- <sbb-seat-reservation-area
432
- style=${styleMap({
433
- "--sbb-reservation-area-width": graphicalElement.dimension.w * this.baseGridSize,
434
- "--sbb-reservation-area-height": graphicalElement.dimension.h * this.baseGridSize
658
+ ${areaProperty ? html`
659
+ <sbb-seat-reservation-graphic
660
+ style=${styleMap({
661
+ "--sbb-seat-reservation--cursor-pointer": "pointer"
435
662
  })}
436
- mounting=${elementMounting}
437
- background="dark"
438
- aria-hidden="true"
439
- title=${ariaLabelForArea}
440
- >
441
- ${areaProperty ? html`
442
- <sbb-seat-reservation-graphic
443
- name=${areaProperty}
444
- rotation=${rotation}
445
- width=${this.baseGridSize}
446
- height=${this.baseGridSize}
447
- role="img"
448
- aria-hidden="true"
449
- ></sbb-seat-reservation-graphic>
450
- ` : nothing}
451
- </sbb-seat-reservation-area>
452
- </sbb-seat-reservation-scoped>
663
+ name=${areaProperty}
664
+ width=${this.baseGridSize}
665
+ height=${this.baseGridSize}
666
+ rotation=${rotation}
667
+ role="img"
668
+ aria-hidden="true"
669
+ ></sbb-seat-reservation-graphic>
670
+ ` : nothing}
671
+ </sbb-seat-reservation-area>
672
+ ${this._popover(triggerId, ariaLabelForArea)}
453
673
  `;
454
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>
686
+ `;
687
+ }
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
+ }
455
695
  _getRenderElementWithoutArea(graphicalElement, rotation, coachDimension) {
456
696
  const calculatedDimension = this.getCalculatedDimension(graphicalElement.dimension, coachDimension);
457
697
  const calculatedPosition = this.getCalculatedPosition(graphicalElement.position, graphicalElement.dimension, coachDimension);
458
698
  const icon = graphicalElement.icon && graphicalElement.icon.indexOf("DRIVER_AREA") === -1 ? graphicalElement.icon : graphicalElement.icon?.concat("_", this.seatReservation.vehicleType);
459
- return html`
460
- <sbb-seat-reservation-scoped
461
- scoped-classes="graphical-element"
462
- inset-block-start="${calculatedPosition.y}px"
463
- inset-inline-start="${calculatedPosition.x}px"
464
- width="${calculatedDimension.w}px"
465
- height="${calculatedDimension.h}px"
466
- z-index="${graphicalElement.position.z}"
467
- >
468
- <sbb-seat-reservation-graphic
469
- name=${icon ?? nothing}
470
- width=${graphicalElement.dimension.w * this.baseGridSize}
471
- height=${graphicalElement.dimension.h * this.baseGridSize}
472
- rotation=${rotation}
473
- aria-hidden="true"
474
- ?stretch=${true}
475
- ></sbb-seat-reservation-graphic>
476
- </sbb-seat-reservation-scoped>
477
- `;
699
+ return html` <sbb-seat-reservation-graphic
700
+ style=${styleMap({
701
+ "--sbb-seat-reservation-graphic-width": calculatedDimension.w,
702
+ "--sbb-seat-reservation-graphic-height": calculatedDimension.h,
703
+ "--sbb-seat-reservation-graphic-top": calculatedPosition.y,
704
+ "--sbb-seat-reservation-graphic-left": calculatedPosition.x,
705
+ "--sbb-seat-reservation-graphic-position": "absolute"
706
+ })}
707
+ name=${icon ?? nothing}
708
+ rotation=${rotation}
709
+ aria-hidden="true"
710
+ ?stretch=${true}
711
+ ></sbb-seat-reservation-graphic>`;
478
712
  }
479
- _getRenderedServiceElements(serviceElements) {
713
+ _getRenderedServiceElements(coachIndex, serviceElements) {
480
714
  if (!serviceElements) {
481
715
  return null;
482
716
  }
483
717
  return serviceElements?.map((serviceElement) => {
484
718
  const titleDescription = serviceElement.icon ? getI18nSeatReservation(serviceElement.icon, this._language.current) : null;
485
- const calculatedcCmpartmentNumberDimension = this.getCalculatedDimension(serviceElement.dimension);
486
- const calculatedcCmpartmentNumberPosition = this.getCalculatedPosition(serviceElement.position);
719
+ const calculatedDimension = this.getCalculatedDimension(serviceElement.dimension);
720
+ const calculatedPosition = this.getCalculatedPosition(serviceElement.position);
487
721
  const elementRotation = serviceElement.rotation || 0;
488
722
  const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;
723
+ const triggerId = `popover-trigger-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
489
724
  return html`
490
- <sbb-seat-reservation-scoped
491
- scoped-classes="graphical-element"
492
- inset-block-start="${calculatedcCmpartmentNumberPosition.y}px"
493
- inset-inline-start="${calculatedcCmpartmentNumberPosition.x}px"
494
- width="${calculatedcCmpartmentNumberDimension.w}px"
495
- height="${calculatedcCmpartmentNumberDimension.h}px"
496
- z-index="${serviceElement.position.z}"
497
- >
498
- <sbb-seat-reservation-graphic
499
- name=${serviceElement.icon ?? nothing}
500
- width=${serviceElement.dimension.w * this.baseGridSize}
501
- height=${serviceElement.dimension.h * this.baseGridSize}
502
- .rotation=${elementFixedRotation}
503
- role="img"
504
- aria-hidden="true"
505
- title=${titleDescription ?? nothing}
506
- ></sbb-seat-reservation-graphic>
507
- </sbb-seat-reservation-scoped>
725
+ <sbb-seat-reservation-graphic
726
+ id="${triggerId}"
727
+ style=${styleMap({
728
+ "--sbb-seat-reservation-graphic-width": calculatedDimension.w,
729
+ "--sbb-seat-reservation-graphic-height": calculatedDimension.h,
730
+ "--sbb-seat-reservation-graphic-top": calculatedPosition.y,
731
+ "--sbb-seat-reservation-graphic-left": calculatedPosition.x,
732
+ "--sbb-seat-reservation-graphic-position": "absolute",
733
+ "--sbb-seat-reservation--cursor-pointer": "pointer"
734
+ })}
735
+ name=${serviceElement.icon ?? nothing}
736
+ .rotation=${elementFixedRotation}
737
+ role="img"
738
+ aria-hidden="true"
739
+ ></sbb-seat-reservation-graphic>
740
+ ${this._popover(triggerId, titleDescription)}
508
741
  `;
509
742
  });
510
743
  }
@@ -532,6 +765,7 @@ let SbbSeatReservationElement = (() => {
532
765
  this.updateCurrentSelectedCoach();
533
766
  this.preselectPlaceInCoach();
534
767
  }
768
+ this._closePopover();
535
769
  }
536
770
  _onFocusNavCoach() {
537
771
  if (!this.preventCoachScrollByPlaceClick) {
@@ -541,18 +775,23 @@ let SbbSeatReservationElement = (() => {
541
775
  }
542
776
  this.isAutoScrolling = false;
543
777
  }
544
- _getDescriptionTableCoach(coachItem) {
778
+ _getDescriptionTableCoach(coachItem, coachIndex) {
545
779
  if (!coachItem.places?.length) {
546
780
  return getI18nSeatReservation("COACH_BLOCKED_TABLE_CAPTION", this._language.current, [
547
781
  coachItem.id
548
782
  ]);
549
783
  }
550
- let tableCoachDescription = "";
784
+ let tableCoachDescription;
551
785
  const areaDescriptions = this._getTitleDescriptionListString(coachItem.graphicElements);
552
786
  const serviceDescriptions = this._getTitleDescriptionListString(coachItem.serviceElements);
553
787
  tableCoachDescription = getI18nSeatReservation("COACH_TABLE_CAPTION", this._language.current, [
554
788
  coachItem.id
555
789
  ]);
790
+ if (!this.hasNavigation) {
791
+ const freePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);
792
+ const freePlacesTxt = getI18nSeatReservation("COACH_AVAILABLE_NUMBER_OF_PLACES", this._language.current, [freePlaces.seats, freePlaces.bicycles]);
793
+ tableCoachDescription = tableCoachDescription.concat(". ").concat(freePlacesTxt).concat(". ");
794
+ }
556
795
  if (!!areaDescriptions || !!serviceDescriptions) {
557
796
  tableCoachDescription += ". " + getI18nSeatReservation("COACH_AVAILABLE_SERVICES", this._language.current) + ": ";
558
797
  tableCoachDescription += serviceDescriptions + ", " + areaDescriptions + ".";
@@ -574,28 +813,8 @@ let SbbSeatReservationElement = (() => {
574
813
  return !!translation && !descriptionAlreayExist && isValidDescription ? translation : null;
575
814
  }).filter((description) => !!description).join(", ");
576
815
  }
577
- }, _seatReservation_accessor_storage = new WeakMap(), _hasNavigation_accessor_storage = new WeakMap(), _alignVertical_accessor_storage = new WeakMap(), _maxReservations_accessor_storage = new WeakMap(), _preventPlaceClick_accessor_storage = new WeakMap(), _classThis = _a, (() => {
816
+ }, _classThis = _a, (() => {
578
817
  const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
579
- _seatReservation_decorators = [property({ attribute: "seat-reservation", type: Object })];
580
- _hasNavigation_decorators = [forceType(), property({ attribute: "has-navigation", type: Boolean })];
581
- _alignVertical_decorators = [forceType(), property({ attribute: "align-vertical", type: Boolean })];
582
- _maxReservations_decorators = [forceType(), property({ attribute: "max-reservations", type: Number })];
583
- _preventPlaceClick_decorators = [forceType(), property({ attribute: "prevent-place-click", type: Boolean })];
584
- __esDecorate(_a, null, _seatReservation_decorators, { kind: "accessor", name: "seatReservation", static: false, private: false, access: { has: (obj) => "seatReservation" in obj, get: (obj) => obj.seatReservation, set: (obj, value) => {
585
- obj.seatReservation = value;
586
- } }, metadata: _metadata }, _seatReservation_initializers, _seatReservation_extraInitializers);
587
- __esDecorate(_a, null, _hasNavigation_decorators, { kind: "accessor", name: "hasNavigation", static: false, private: false, access: { has: (obj) => "hasNavigation" in obj, get: (obj) => obj.hasNavigation, set: (obj, value) => {
588
- obj.hasNavigation = value;
589
- } }, metadata: _metadata }, _hasNavigation_initializers, _hasNavigation_extraInitializers);
590
- __esDecorate(_a, null, _alignVertical_decorators, { kind: "accessor", name: "alignVertical", static: false, private: false, access: { has: (obj) => "alignVertical" in obj, get: (obj) => obj.alignVertical, set: (obj, value) => {
591
- obj.alignVertical = value;
592
- } }, metadata: _metadata }, _alignVertical_initializers, _alignVertical_extraInitializers);
593
- __esDecorate(_a, null, _maxReservations_decorators, { kind: "accessor", name: "maxReservations", static: false, private: false, access: { has: (obj) => "maxReservations" in obj, get: (obj) => obj.maxReservations, set: (obj, value) => {
594
- obj.maxReservations = value;
595
- } }, metadata: _metadata }, _maxReservations_initializers, _maxReservations_extraInitializers);
596
- __esDecorate(_a, null, _preventPlaceClick_decorators, { kind: "accessor", name: "preventPlaceClick", static: false, private: false, access: { has: (obj) => "preventPlaceClick" in obj, get: (obj) => obj.preventPlaceClick, set: (obj, value) => {
597
- obj.preventPlaceClick = value;
598
- } }, metadata: _metadata }, _preventPlaceClick_initializers, _preventPlaceClick_extraInitializers);
599
818
  __esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
600
819
  _classThis = _classDescriptor.value;
601
820
  if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
@@ -605,4 +824,4 @@ let SbbSeatReservationElement = (() => {
605
824
  export {
606
825
  SbbSeatReservationElement
607
826
  };
608
- //# 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 { forceType } from '@sbb-esta/lyne-elements/core/decorators.js';\nimport { html, nothing } from 'lit';\nimport type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { customElement, property } 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/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';\n\n/**\n * Describe the purpose of the component with a single short sentence.\n */\nexport\n@customElement('sbb-seat-reservation')\nclass SbbSeatReservationElement extends SeatReservationBaseElement {\n  public static override styles: CSSResultGroup = style;\n\n  /** The seat reservation object which contains all coaches and places */\n  @property({ attribute: 'seat-reservation', type: Object })\n  public override accessor seatReservation: SeatReservation = null!;\n\n  /** The seat reservation navigation can be toggled by this property */\n  @forceType()\n  @property({ attribute: 'has-navigation', type: Boolean })\n  public override accessor hasNavigation: boolean = true;\n\n  /** Controls the visual representation of seat reservation in a horizonal or vertical alignment */\n  @forceType()\n  @property({ attribute: 'align-vertical', type: Boolean })\n  public override accessor alignVertical: boolean = false;\n\n  /** Maximal number of possible clickable seats */\n  @forceType()\n  @property({ attribute: 'max-reservations', type: Number })\n  public override accessor maxReservations: number = null!;\n\n  /** Any click functionality is prevented */\n  @forceType()\n  @property({ attribute: 'prevent-place-click', type: Boolean })\n  public override accessor preventPlaceClick: boolean = false;\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_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  // 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    this._componentSetup();\n  }\n\n  protected override render(): TemplateResult | null {\n    this._initVehicleSeatReservationConstruction();\n    return this._coachesHtmlTemplate || null;\n  }\n\n  private _componentSetup(): void {\n    this.initNavigationSelectionByScrollEvent();\n  }\n\n  private _initVehicleSeatReservationConstruction(): void {\n    const coachItems = JSON.parse(JSON.stringify(this.seatReservation?.coachItems));\n    const classAlignVertical = this.alignVertical ? 'sbb-sr__wrapper--vertical' : '';\n\n    this._coachesHtmlTemplate = html`\n      <div>\n        <sbb-screen-reader-only>\n          <input\n            id=\"first-tab-element\"\n            role=\"contentinfo\"\n            type=\"text\"\n            aria-label=\"${getI18nSeatReservation('SEAT_RESERVATION_BEGIN', this._language.current)}\"\n            readonly\n          />\n        </sbb-screen-reader-only>\n\n        <div @keydown=${(evt: KeyboardEvent) => this.handleKeyboardEvent(evt)}>\n          ${this._renderNavigation()}\n          <div class=\"sbb-sr__wrapper ${classAlignVertical}\">\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\n        <sbb-screen-reader-only>\n          <input\n            id=\"last-tab-element\"\n            role=\"contentinfo\"\n            type=\"text\"\n            aria-label=\"${getI18nSeatReservation('SEAT_RESERVATION_END', this._language.current)}\"\n            readonly\n          />\n        </sbb-screen-reader-only>\n      </div>\n    `;\n  }\n\n  private _renderNavigation(): TemplateResult | null {\n    if (!this.hasNavigation) {\n      return null;\n    }\n\n    return html`\n      <nav class=\"${classMap({ 'sbb-sr-navigation--vertical': this.alignVertical })}\">\n        <ul\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            return html`<li>\n              <sbb-seat-reservation-navigation-coach\n                @selectcoach=${(event: CustomEvent<number>) => this._onSelectNavCoach(event)}\n                @focuscoach=${() => this._onFocusNavCoach()}\n                index=\"${index}\"\n                coach-id=\"${coachItem.id}\"\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    `;\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);\n\n    return html` <sbb-seat-reservation-scoped\n      scoped-classes=\"coach-wrapper\"\n      height=\"${calculatedCoachDimension.h}px\"\n      width=\"${calculatedCoachDimension.w}px\"\n    >\n      ${this._getRenderedCoachBorders(coachItem, index)}\n      ${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension)}\n      ${this._getRenderedServiceElements(coachItem.serviceElements)}\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  private _getRenderedCoachBorders(coachItem: CoachItem, coachIndex: number): TemplateResult {\n    const allElements = coachItem.graphicElements;\n    const COACH_PASSAGE_WIDTH = 1;\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 borderOffsetX =\n      coachIndex === 0 && driverArea\n        ? driverArea?.dimension.w * this.baseGridSize\n        : this.baseGridSize;\n\n    return html`\n      <sbb-seat-reservation-scoped\n        scoped-classes=\"coach-border\"\n        inset-block-start=\"${this.coachBorderPadding * -1}px\"\n        inset-inline-start=\"${borderOffsetX}px\"\n      >\n        <sbb-seat-reservation-graphic\n          name=\"COACH_BORDER_MIDDLE\"\n          width=${borderWidth * this.baseGridSize}\n          height=${(coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize}\n          ?stretch=${true}\n          role=\"presentation\"\n        ></sbb-seat-reservation-graphic>\n      </sbb-seat-reservation-scoped>\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 calculatedInternalDimension = this.getCalculatedDimension(place.dimension);\n      const calculatedInternalPosition = this.getCalculatedPosition(place.position);\n      const textRotation = this.alignVertical ? -90 : 0;\n\n      return html`\n        <sbb-seat-reservation-scoped\n          scoped-classes=\"graphical-element\"\n          inset-block-start=\"${calculatedInternalPosition.y}px\"\n          inset-inline-start=\"${calculatedInternalPosition.x}px\"\n          width=\"${calculatedInternalDimension.w}px\"\n          height=\"${calculatedInternalDimension.h}px\"\n          z-index=\"${place.position.z}\"\n          cell-id=\"cell-${coachIndex}-${place.position.y}-${index}\"\n        >\n          <sbb-seat-reservation-place-control\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        </sbb-seat-reservation-scoped>\n      `;\n    });\n  }\n\n  private _getRenderedGraphicalElements(\n    graphicalElements: BaseElement[],\n    coachDimension: ElementDimension,\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(graphicalElement, elementFixedRotation, coachDimension);\n    });\n  }\n\n  private _getRenderElementWithArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n  ): TemplateResult {\n    // TODO -> isNotTableGraphicTempFix is temp fix to show coach tables as area and not as svg graphic.\n    // The problem here is that when using TABLE svg graphics,\n    // they are displayed distorted due to different heights and widths and this is not visually good.\n    const isNotTableGraphicTempFix = graphicalElement.icon?.indexOf('TABLE') === -1;\n\n    const areaProperty =\n      graphicalElement.icon && isNotTableGraphicTempFix ? 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    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-scoped\n        scoped-classes=\"graphical-element\"\n        inset-block-start=\"${calculatedPosition.y}px\"\n        inset-inline-start=\"${calculatedPosition.x}px\"\n        width=\"${calculatedDimension.w}px\"\n        height=\"${calculatedDimension.h}px\"\n        z-index=\"${graphicalElement.position.z}\"\n      >\n        <sbb-seat-reservation-area\n          style=${styleMap({\n            '--sbb-reservation-area-width': graphicalElement.dimension.w * this.baseGridSize,\n            '--sbb-reservation-area-height': graphicalElement.dimension.h * this.baseGridSize,\n          })}\n          mounting=${elementMounting}\n          background=\"dark\"\n          aria-hidden=\"true\"\n          title=${ariaLabelForArea}\n        >\n          ${areaProperty\n            ? html`\n                <sbb-seat-reservation-graphic\n                  name=${areaProperty}\n                  rotation=${rotation}\n                  width=${this.baseGridSize}\n                  height=${this.baseGridSize}\n                  role=\"img\"\n                  aria-hidden=\"true\"\n                ></sbb-seat-reservation-graphic>\n              `\n            : nothing}\n        </sbb-seat-reservation-area>\n      </sbb-seat-reservation-scoped>\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('_', this.seatReservation.vehicleType);\n\n    return html`\n      <sbb-seat-reservation-scoped\n        scoped-classes=\"graphical-element\"\n        inset-block-start=\"${calculatedPosition.y}px\"\n        inset-inline-start=\"${calculatedPosition.x}px\"\n        width=\"${calculatedDimension.w}px\"\n        height=\"${calculatedDimension.h}px\"\n        z-index=\"${graphicalElement.position.z}\"\n      >\n        <sbb-seat-reservation-graphic\n          name=${icon ?? nothing}\n          width=${graphicalElement.dimension.w * this.baseGridSize}\n          height=${graphicalElement.dimension.h * this.baseGridSize}\n          rotation=${rotation}\n          aria-hidden=\"true\"\n          ?stretch=${true}\n        ></sbb-seat-reservation-graphic>\n      </sbb-seat-reservation-scoped>\n    `;\n  }\n\n  private _getRenderedServiceElements(serviceElements?: BaseElement[]): 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 calculatedcCmpartmentNumberDimension = this.getCalculatedDimension(\n        serviceElement.dimension,\n      );\n      const calculatedcCmpartmentNumberPosition = this.getCalculatedPosition(\n        serviceElement.position,\n      );\n      const elementRotation = serviceElement.rotation || 0;\n      const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;\n\n      return html`\n        <sbb-seat-reservation-scoped\n          scoped-classes=\"graphical-element\"\n          inset-block-start=\"${calculatedcCmpartmentNumberPosition.y}px\"\n          inset-inline-start=\"${calculatedcCmpartmentNumberPosition.x}px\"\n          width=\"${calculatedcCmpartmentNumberDimension.w}px\"\n          height=\"${calculatedcCmpartmentNumberDimension.h}px\"\n          z-index=\"${serviceElement.position.z}\"\n        >\n          <sbb-seat-reservation-graphic\n            name=${serviceElement.icon ?? nothing}\n            width=${serviceElement.dimension.w * this.baseGridSize}\n            height=${serviceElement.dimension.h * this.baseGridSize}\n            .rotation=${elementFixedRotation}\n            role=\"img\"\n            aria-hidden=\"true\"\n            title=${titleDescription ?? nothing}\n          ></sbb-seat-reservation-graphic>\n        </sbb-seat-reservation-scoped>\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\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): 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 = '';\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 (!!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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"}
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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"}