@sbb-esta/lyne-elements-experimental 3.1.0 → 3.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/custom-elements.json +1147 -590
- package/development/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/icon-mapper.js +4 -5
- package/development/seat-reservation/common/mapper/mapper.d.ts +6 -5
- package/development/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/mapper.js +14 -11
- package/development/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts +113 -0
- package/development/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
- package/development/seat-reservation/common/mapper/seat-reservation-sample-data.js +25999 -649
- package/development/seat-reservation/common/svgs.d.ts +0 -1
- package/development/seat-reservation/common/svgs.d.ts.map +1 -1
- package/development/seat-reservation/common/svgs.js +1 -3
- package/development/seat-reservation/common/translations/i18n.d.ts.map +1 -1
- package/development/seat-reservation/common/translations/i18n.js +225 -194
- package/development/seat-reservation/common/types.d.ts +20 -3
- package/development/seat-reservation/common/types.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +50 -18
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation-base-element.js +352 -172
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts +29 -13
- package/development/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation/seat-reservation.component.js +603 -308
- package/development/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +17 -10
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-assets.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-assets.js +1 -6
- package/development/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +36 -18
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts +3 -1
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +54 -32
- package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts +2 -0
- package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.js +107 -58
- package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts +1 -16
- package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts.map +1 -1
- package/development/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.js +10 -164
- package/package.json +2 -2
- package/seat-reservation/common/mapper/icon-mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/icon-mapper.js +3 -4
- package/seat-reservation/common/mapper/mapper.d.ts +6 -5
- package/seat-reservation/common/mapper/mapper.d.ts.map +1 -1
- package/seat-reservation/common/mapper/mapper.js +35 -32
- package/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts +113 -0
- package/seat-reservation/common/mapper/seat-reservation-sample-data.d.ts.map +1 -1
- package/seat-reservation/common/mapper/seat-reservation-sample-data.js +25997 -648
- package/seat-reservation/common/svgs.d.ts +0 -1
- package/seat-reservation/common/svgs.d.ts.map +1 -1
- package/seat-reservation/common/svgs.js +4 -11
- package/seat-reservation/common/translations/i18n.d.ts.map +1 -1
- package/seat-reservation/common/translations/i18n.js +228 -197
- package/seat-reservation/common/types.d.ts +20 -3
- package/seat-reservation/common/types.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts +50 -18
- package/seat-reservation/seat-reservation/seat-reservation-base-element.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation-base-element.js +294 -175
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts +29 -13
- package/seat-reservation/seat-reservation/seat-reservation.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation/seat-reservation.component.js +318 -303
- package/seat-reservation/seat-reservation-area/seat-reservation-area.component.js +22 -22
- package/seat-reservation/seat-reservation-graphic/seat-reservation-assets.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-graphic/seat-reservation-assets.js +0 -5
- package/seat-reservation/seat-reservation-graphic/seat-reservation-graphic.component.js +37 -37
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts +3 -1
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-navigation-coach/seat-reservation-navigation-coach.component.js +95 -84
- package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts +2 -0
- package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-place-control/seat-reservation-place-control.component.js +75 -63
- package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts +1 -16
- package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.d.ts.map +1 -1
- package/seat-reservation/seat-reservation-scoped/seat-reservation-scoped.component.js +14 -101
|
@@ -1,25 +1,19 @@
|
|
|
1
|
-
|
|
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 {
|
|
11
|
-
import {
|
|
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,10 @@ const style = css`*,
|
|
|
28
22
|
|
|
29
23
|
:host {
|
|
30
24
|
--sbb-seat-reservation-grid-size: 16px;
|
|
31
|
-
--sbb-seat-reservation-
|
|
25
|
+
--sbb-seat-reservation-height: 0;
|
|
26
|
+
--sbb-seat-reservation-decks: 1;
|
|
32
27
|
display: block;
|
|
28
|
+
height: inherit;
|
|
33
29
|
}
|
|
34
30
|
:host ::part(coach-floor) {
|
|
35
31
|
fill: var(--sbb-color-white);
|
|
@@ -40,29 +36,124 @@ const style = css`*,
|
|
|
40
36
|
}
|
|
41
37
|
}
|
|
42
38
|
|
|
43
|
-
.sbb-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
rotate:
|
|
48
|
-
transform-origin: top left;
|
|
39
|
+
:host(:dir(rtl)) :not(.sbb-sr-place-ctrl--type-bicycle,
|
|
40
|
+
.sbb-sr-place-ctrl--orientation-90,
|
|
41
|
+
.sbb-sr-place-ctrl--orientation-270,
|
|
42
|
+
.sbb-sr-place-ctrl--state-selected) ::part(coach-floor) {
|
|
43
|
+
rotate: calc(180 * 1deg);
|
|
49
44
|
}
|
|
50
45
|
|
|
51
|
-
|
|
52
|
-
|
|
46
|
+
[popover]:where(sbb-popover) {
|
|
47
|
+
margin: 0;
|
|
48
|
+
padding: 0;
|
|
49
|
+
border: none;
|
|
50
|
+
width: auto;
|
|
51
|
+
height: auto;
|
|
52
|
+
background-color: transparent;
|
|
53
|
+
color: inherit;
|
|
54
|
+
pointer-events: none;
|
|
53
55
|
}
|
|
54
56
|
|
|
55
|
-
.sbb-
|
|
56
|
-
|
|
57
|
+
.sbb-sr__container {
|
|
58
|
+
display: flex;
|
|
59
|
+
height: inherit;
|
|
57
60
|
}
|
|
58
61
|
|
|
59
|
-
.sbb-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
+
.sbb-sr__grid {
|
|
63
|
+
display: grid;
|
|
64
|
+
grid-template-columns: 20% 1fr 20%;
|
|
65
|
+
grid-template-rows: auto auto;
|
|
66
|
+
}
|
|
67
|
+
.sbb-sr__grid .sbb-sr-navigation-first-grid,
|
|
68
|
+
.sbb-sr__grid .sbb-sr-navigation-last-grid {
|
|
69
|
+
grid-row-start: 1;
|
|
70
|
+
position: relative;
|
|
71
|
+
}
|
|
72
|
+
.sbb-sr__grid .sbb-sr-navigation-first-grid {
|
|
73
|
+
grid-column: 1/2;
|
|
74
|
+
}
|
|
75
|
+
.sbb-sr__grid .sbb-sr-navigation-last-grid {
|
|
76
|
+
grid-column: 3/4;
|
|
77
|
+
}
|
|
78
|
+
.sbb-sr__grid .sbb-sr__component {
|
|
79
|
+
display: flex;
|
|
80
|
+
grid-column: 1/4;
|
|
81
|
+
grid-row: 1/3;
|
|
82
|
+
}
|
|
83
|
+
.sbb-sr__grid .sbb-sr-grid-inner {
|
|
84
|
+
display: grid;
|
|
85
|
+
grid-template-columns: 20% 60% 20%;
|
|
86
|
+
grid-template-rows: auto 1fr;
|
|
87
|
+
grid-gap: 0;
|
|
88
|
+
}
|
|
89
|
+
.sbb-sr__grid .sbb-sr-grid-inner .nav-grid {
|
|
90
|
+
grid-column: 2/3;
|
|
91
|
+
grid-row: 1/2;
|
|
92
|
+
display: inherit;
|
|
93
|
+
}
|
|
94
|
+
.sbb-sr__grid .sbb-sr-grid-inner .coaches-grid {
|
|
95
|
+
grid-column: 1/4;
|
|
96
|
+
grid-row: 2/3;
|
|
97
|
+
display: inherit;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
.sbb-sr__navigation-control-button {
|
|
62
101
|
position: absolute;
|
|
102
|
+
z-index: 10;
|
|
63
103
|
}
|
|
64
|
-
.sbb-
|
|
65
|
-
|
|
104
|
+
.sbb-sr__navigation-control-button#first-tab-element {
|
|
105
|
+
inset-inline-end: calc(8 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
106
|
+
}
|
|
107
|
+
.sbb-sr__navigation-control-button#last-tab-element {
|
|
108
|
+
inset-inline-start: calc(8 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
.sbb-sr-navigation-wrapper {
|
|
112
|
+
overflow: hidden;
|
|
113
|
+
padding-block: 0 calc(32 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
.sbb-sr-navigation {
|
|
117
|
+
--sbb-scrollbar-thumb-width: 0.125rem;
|
|
118
|
+
--sbb-scrollbar-thumb-width-hover: 0.25rem;
|
|
119
|
+
--sbb-scrollbar-width-firefox: thin;
|
|
120
|
+
--sbb-scrollbar-color: var(--sbb-color-black-alpha-30);
|
|
121
|
+
--sbb-scrollbar-color-hover: var(--sbb-color-black-alpha-60);
|
|
122
|
+
--sbb-scrollbar-track-color: transparent;
|
|
123
|
+
--sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
|
|
124
|
+
}
|
|
125
|
+
.sbb-sr-navigation::-webkit-scrollbar {
|
|
126
|
+
width: var(--sbb-scrollbar-width);
|
|
127
|
+
height: var(--sbb-scrollbar-width);
|
|
128
|
+
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
129
|
+
}
|
|
130
|
+
.sbb-sr-navigation::-webkit-scrollbar-corner {
|
|
131
|
+
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
132
|
+
}
|
|
133
|
+
.sbb-sr-navigation::-webkit-scrollbar-thumb {
|
|
134
|
+
background-color: var(--sbb-scrollbar-color, currentcolor);
|
|
135
|
+
border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
|
|
136
|
+
border-radius: var(--sbb-border-radius-4x);
|
|
137
|
+
background-clip: padding-box;
|
|
138
|
+
}
|
|
139
|
+
.sbb-sr-navigation::-webkit-scrollbar-thumb:hover {
|
|
140
|
+
background-color: var(--sbb-scrollbar-color-hover, currentcolor);
|
|
141
|
+
border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
|
|
142
|
+
}
|
|
143
|
+
.sbb-sr-navigation::-webkit-scrollbar-button, .sbb-sr-navigation::-webkit-scrollbar-corner {
|
|
144
|
+
display: none;
|
|
145
|
+
}
|
|
146
|
+
@supports not selector(::-webkit-scrollbar) {
|
|
147
|
+
.sbb-sr-navigation {
|
|
148
|
+
scrollbar-width: var(--sbb-scrollbar-width-firefox);
|
|
149
|
+
scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
.sbb-sr-navigation {
|
|
153
|
+
--sbb-scrollbar-color: var(--sbb-color-white-alpha-0);
|
|
154
|
+
display: flex;
|
|
155
|
+
overflow: scroll hidden;
|
|
156
|
+
position: relative;
|
|
66
157
|
}
|
|
67
158
|
|
|
68
159
|
.sbb-sr-navigation__list-coaches {
|
|
@@ -71,72 +162,244 @@ const style = css`*,
|
|
|
71
162
|
padding: 0;
|
|
72
163
|
font-size: inherit;
|
|
73
164
|
display: flex;
|
|
74
|
-
justify-content: center;
|
|
75
165
|
flex-wrap: nowrap;
|
|
76
|
-
gap: var(--sbb-
|
|
77
|
-
|
|
166
|
+
gap: calc(4 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
167
|
+
padding: calc(2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
78
168
|
}
|
|
79
169
|
|
|
80
|
-
.sbb-
|
|
81
|
-
|
|
82
|
-
|
|
170
|
+
.sbb-sr__wrapper-coach-decks {
|
|
171
|
+
display: flex;
|
|
172
|
+
overflow: hidden;
|
|
173
|
+
height: inherit;
|
|
174
|
+
}
|
|
175
|
+
.sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels {
|
|
176
|
+
display: flex;
|
|
177
|
+
justify-content: space-between;
|
|
178
|
+
height: calc(var(--sbb-seat-reservation-height) * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
179
|
+
writing-mode: vertical-lr;
|
|
180
|
+
}
|
|
181
|
+
.sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels b {
|
|
182
|
+
height: calc(50% - 16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
183
|
+
width: calc(20 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
184
|
+
margin-block-end: 16px;
|
|
185
|
+
rotate: 180deg;
|
|
186
|
+
text-align: center;
|
|
187
|
+
line-height: calc(20 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
83
188
|
}
|
|
84
189
|
|
|
85
|
-
.sbb-
|
|
190
|
+
.sbb-sr__wrapper {
|
|
191
|
+
--sbb-scrollbar-thumb-width: 0.125rem;
|
|
192
|
+
--sbb-scrollbar-thumb-width-hover: 0.25rem;
|
|
193
|
+
--sbb-scrollbar-width-firefox: thin;
|
|
194
|
+
--sbb-scrollbar-color: var(--sbb-color-black-alpha-30);
|
|
195
|
+
--sbb-scrollbar-color-hover: var(--sbb-color-black-alpha-60);
|
|
196
|
+
--sbb-scrollbar-track-color: transparent;
|
|
197
|
+
--sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
|
|
198
|
+
}
|
|
199
|
+
.sbb-sr__wrapper::-webkit-scrollbar {
|
|
200
|
+
width: var(--sbb-scrollbar-width);
|
|
201
|
+
height: var(--sbb-scrollbar-width);
|
|
202
|
+
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
203
|
+
}
|
|
204
|
+
.sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
205
|
+
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
206
|
+
}
|
|
207
|
+
.sbb-sr__wrapper::-webkit-scrollbar-thumb {
|
|
208
|
+
background-color: var(--sbb-scrollbar-color, currentcolor);
|
|
209
|
+
border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
|
|
210
|
+
border-radius: var(--sbb-border-radius-4x);
|
|
211
|
+
background-clip: padding-box;
|
|
212
|
+
}
|
|
213
|
+
.sbb-sr__wrapper::-webkit-scrollbar-thumb:hover {
|
|
214
|
+
background-color: var(--sbb-scrollbar-color-hover, currentcolor);
|
|
215
|
+
border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
|
|
216
|
+
}
|
|
217
|
+
.sbb-sr__wrapper::-webkit-scrollbar-button, .sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
218
|
+
display: none;
|
|
219
|
+
}
|
|
220
|
+
@supports not selector(::-webkit-scrollbar) {
|
|
221
|
+
.sbb-sr__wrapper {
|
|
222
|
+
scrollbar-width: var(--sbb-scrollbar-width-firefox);
|
|
223
|
+
scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
.sbb-sr__wrapper {
|
|
227
|
+
overflow-x: scroll;
|
|
228
|
+
padding-block: calc(8 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
229
|
+
}
|
|
230
|
+
.sbb-sr__wrapper .sbb-sr__parent {
|
|
86
231
|
display: flex;
|
|
87
|
-
|
|
88
|
-
|
|
232
|
+
flex-direction: column;
|
|
233
|
+
position: relative;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
.sbb-sr__list-decks {
|
|
237
|
+
list-style: none;
|
|
89
238
|
margin: 0;
|
|
239
|
+
padding: 0;
|
|
240
|
+
font-size: inherit;
|
|
241
|
+
display: flex;
|
|
242
|
+
flex-direction: column;
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
.sbb-sr__list-decks--gap {
|
|
246
|
+
gap: calc(48 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
.sbb-sr__list-item-deck {
|
|
250
|
+
position: relative;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
.sbb-sr__list-coaches {
|
|
90
254
|
list-style: none;
|
|
255
|
+
margin: 0;
|
|
256
|
+
padding: 0;
|
|
257
|
+
font-size: inherit;
|
|
258
|
+
display: flex;
|
|
259
|
+
gap: calc(4 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
91
260
|
}
|
|
92
261
|
|
|
93
262
|
.sbb-sr__item-coach {
|
|
94
|
-
padding: 0;
|
|
95
|
-
margin: 0;
|
|
96
263
|
position: relative;
|
|
97
264
|
}
|
|
98
265
|
|
|
266
|
+
.sbb-sr-popover {
|
|
267
|
+
margin: 0;
|
|
268
|
+
}
|
|
269
|
+
|
|
99
270
|
.sbb-sr-coach-wrapper__table {
|
|
100
271
|
outline: 0;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
.sbb-sr.sbb-sr__grid--vertical {
|
|
275
|
+
grid-template-columns: calc((var(--sbb-seat-reservation-height) + 24) * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) 1fr;
|
|
276
|
+
grid-template-rows: 20% 60% 20%;
|
|
277
|
+
}
|
|
278
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-first-grid {
|
|
279
|
+
grid-column: 2/3;
|
|
280
|
+
grid-row-start: 1;
|
|
281
|
+
}
|
|
282
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-first-grid #first-tab-element {
|
|
283
|
+
position: absolute;
|
|
284
|
+
inset-block-end: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
285
|
+
inset-inline-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
286
|
+
inset-inline-end: initial;
|
|
287
|
+
rotate: 90deg;
|
|
288
|
+
}
|
|
289
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-last-grid {
|
|
290
|
+
grid-column: 2/3;
|
|
291
|
+
grid-row: 3/4;
|
|
292
|
+
}
|
|
293
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-last-grid #last-tab-element {
|
|
294
|
+
inset-block-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
295
|
+
inset-inline-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
296
|
+
rotate: 90deg;
|
|
297
|
+
}
|
|
298
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-navigation-wrapper {
|
|
299
|
+
padding-block-end: initial;
|
|
300
|
+
padding-inline-start: calc(16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
301
|
+
}
|
|
302
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr__component {
|
|
303
|
+
grid-column: 1/3;
|
|
304
|
+
grid-row: 1/4;
|
|
305
|
+
}
|
|
306
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner {
|
|
307
|
+
grid-template-columns: calc((var(--sbb-seat-reservation-height) + 24) * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) 1fr;
|
|
308
|
+
grid-template-rows: 20% 60% 20%;
|
|
309
|
+
}
|
|
310
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid {
|
|
311
|
+
grid-column: 2/3;
|
|
312
|
+
grid-row: 2/3;
|
|
313
|
+
height: 100%;
|
|
314
|
+
}
|
|
315
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid .sbb-sr-navigation {
|
|
316
|
+
display: block;
|
|
317
|
+
overflow: hidden scroll;
|
|
318
|
+
height: 100%;
|
|
319
|
+
}
|
|
320
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .nav-grid .sbb-sr-navigation .sbb-sr-navigation__list-coaches {
|
|
321
|
+
flex-direction: column;
|
|
322
|
+
padding: calc(2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
323
|
+
}
|
|
324
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid {
|
|
325
|
+
grid-column: 1/2;
|
|
326
|
+
grid-row: 1/4;
|
|
327
|
+
overflow: hidden;
|
|
328
|
+
position: relative;
|
|
329
|
+
}
|
|
330
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks {
|
|
331
|
+
flex-direction: column;
|
|
332
|
+
}
|
|
333
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels {
|
|
334
|
+
flex-direction: column-reverse;
|
|
335
|
+
width: calc(100% - 24 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
336
|
+
height: initial;
|
|
337
|
+
}
|
|
338
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper-deck-labels b {
|
|
339
|
+
width: calc(50% - 16 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
340
|
+
height: calc(20 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
341
|
+
margin-bottom: 16px;
|
|
342
|
+
rotate: initial;
|
|
343
|
+
writing-mode: initial;
|
|
344
|
+
}
|
|
345
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper {
|
|
346
|
+
--sbb-scrollbar-thumb-width: 0.125rem;
|
|
347
|
+
--sbb-scrollbar-thumb-width-hover: 0.25rem;
|
|
348
|
+
--sbb-scrollbar-width-firefox: thin;
|
|
349
|
+
--sbb-scrollbar-color: var(--sbb-color-black-alpha-30);
|
|
350
|
+
--sbb-scrollbar-color-hover: var(--sbb-color-black-alpha-60);
|
|
351
|
+
--sbb-scrollbar-track-color: transparent;
|
|
352
|
+
--sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);
|
|
353
|
+
}
|
|
354
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar {
|
|
355
|
+
width: var(--sbb-scrollbar-width);
|
|
356
|
+
height: var(--sbb-scrollbar-width);
|
|
357
|
+
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
358
|
+
}
|
|
359
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
360
|
+
background-color: var(--sbb-scrollbar-track-color, transparent);
|
|
361
|
+
}
|
|
362
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-thumb {
|
|
363
|
+
background-color: var(--sbb-scrollbar-color, currentcolor);
|
|
364
|
+
border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;
|
|
365
|
+
border-radius: var(--sbb-border-radius-4x);
|
|
366
|
+
background-clip: padding-box;
|
|
367
|
+
}
|
|
368
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-thumb:hover {
|
|
369
|
+
background-color: var(--sbb-scrollbar-color-hover, currentcolor);
|
|
370
|
+
border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));
|
|
371
|
+
}
|
|
372
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-button, .sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper::-webkit-scrollbar-corner {
|
|
373
|
+
display: none;
|
|
374
|
+
}
|
|
375
|
+
@supports not selector(::-webkit-scrollbar) {
|
|
376
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper {
|
|
377
|
+
scrollbar-width: var(--sbb-scrollbar-width-firefox);
|
|
378
|
+
scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__wrapper {
|
|
382
|
+
overflow: hidden scroll;
|
|
383
|
+
height: 100%;
|
|
384
|
+
padding: 0;
|
|
385
|
+
}
|
|
386
|
+
.sbb-sr.sbb-sr__grid--vertical .sbb-sr-grid-inner .coaches-grid .sbb-sr__wrapper-coach-decks .sbb-sr__parent {
|
|
387
|
+
rotate: 90deg;
|
|
388
|
+
transform-origin: calc((var(--sbb-seat-reservation-height) + 24 - 24 - 6 * (var(--sbb-seat-reservation-decks) - 1)) / 2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem)) calc((var(--sbb-seat-reservation-height) + 24 - 24 - 6 * (var(--sbb-seat-reservation-decks) - 1)) / 2 * var(--sbb-seat-reservation-one-px-rem, 0.0625rem));
|
|
101
389
|
}`;
|
|
102
390
|
let SbbSeatReservationElement = (() => {
|
|
103
|
-
var
|
|
391
|
+
var _a;
|
|
104
392
|
let _classDecorators = [customElement("sbb-seat-reservation")];
|
|
105
393
|
let _classDescriptor;
|
|
106
394
|
let _classExtraInitializers = [];
|
|
107
395
|
let _classThis;
|
|
108
396
|
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
397
|
_a = class extends _classSuper {
|
|
125
398
|
constructor() {
|
|
126
399
|
super(...arguments);
|
|
127
|
-
|
|
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));
|
|
400
|
+
this._language = new SbbLanguageController(this);
|
|
138
401
|
this._notAreaElements = [
|
|
139
|
-
"
|
|
402
|
+
"DRIVER_AREA",
|
|
140
403
|
"COACH_PASSAGE",
|
|
141
404
|
"COACH_WALL_NO_PASSAGE",
|
|
142
405
|
"COMPARTMENT_PASSAGE",
|
|
@@ -146,41 +409,6 @@ let SbbSeatReservationElement = (() => {
|
|
|
146
409
|
];
|
|
147
410
|
this._notFixedRotatableAreaIcons = ["ENTRY_EXIT"];
|
|
148
411
|
}
|
|
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
412
|
willUpdate(changedProperties) {
|
|
185
413
|
super.willUpdate(changedProperties);
|
|
186
414
|
if (changedProperties.has("hasNavigation")) {
|
|
@@ -193,148 +421,211 @@ let SbbSeatReservationElement = (() => {
|
|
|
193
421
|
}
|
|
194
422
|
firstUpdated(changedProperties) {
|
|
195
423
|
super.firstUpdated(changedProperties);
|
|
196
|
-
this.
|
|
424
|
+
this.updateComplete.then(() => {
|
|
425
|
+
this.initPrepairSeatReservationData();
|
|
426
|
+
this.initNavigationSelectionByScrollEvent();
|
|
427
|
+
this.requestUpdate();
|
|
428
|
+
});
|
|
197
429
|
}
|
|
198
430
|
render() {
|
|
431
|
+
if (!isServer) {
|
|
432
|
+
this._determineBaseFontSize();
|
|
433
|
+
}
|
|
199
434
|
this._initVehicleSeatReservationConstruction();
|
|
200
435
|
return this._coachesHtmlTemplate || null;
|
|
201
436
|
}
|
|
202
|
-
|
|
203
|
-
|
|
437
|
+
_determineBaseFontSize() {
|
|
438
|
+
const baseFontSize = parseInt(window.getComputedStyle(document.body).fontSize, 10);
|
|
439
|
+
const onePixelInRem = 1 / baseFontSize;
|
|
440
|
+
this.style?.setProperty("--sbb-seat-reservation-one-px-rem", `${onePixelInRem + "rem"}`);
|
|
204
441
|
}
|
|
205
442
|
_initVehicleSeatReservationConstruction() {
|
|
206
|
-
const
|
|
207
|
-
const classAlignVertical = this.alignVertical ? "sbb-sr__wrapper--vertical" : "";
|
|
443
|
+
const classAlignVertical = this.alignVertical ? "sbb-sr__grid--vertical" : "";
|
|
208
444
|
this._coachesHtmlTemplate = html`
|
|
209
|
-
<div>
|
|
210
|
-
<sbb-
|
|
211
|
-
<
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
445
|
+
<div class="sbb-sr__container">
|
|
446
|
+
<div class="sbb-sr sbb-sr__grid ${classAlignVertical}">
|
|
447
|
+
<div class="sbb-sr-navigation-first-grid">
|
|
448
|
+
${this._renderNavigationControlButton("DIRECTION_LEFT")}
|
|
449
|
+
</div>
|
|
450
|
+
<div
|
|
451
|
+
class="sbb-sr__component"
|
|
452
|
+
@keydown=${(evt) => this.handleKeyboardEvent(evt)}
|
|
453
|
+
>
|
|
454
|
+
<div class="sbb-sr-grid-inner">
|
|
455
|
+
<div class="nav-grid">${this._renderNavigation()}</div>
|
|
456
|
+
<div class="coaches-grid">
|
|
457
|
+
<div class="sbb-sr__wrapper-coach-decks">
|
|
458
|
+
<div class="sbb-sr__wrapper-deck-labels">${this._renderDeckLabels()}</div>
|
|
459
|
+
<div id="sbb-sr__wrapper-scrollarea" class="sbb-sr__wrapper">
|
|
460
|
+
<div id="sbb-sr__parent-area" class="sbb-sr__parent" tabindex="-1">
|
|
461
|
+
<ul
|
|
462
|
+
class="${classMap({
|
|
463
|
+
"sbb-sr__list-decks": true,
|
|
464
|
+
"sbb-sr__list-decks--gap": this.hasMultipleDecks
|
|
465
|
+
})}"
|
|
466
|
+
>
|
|
467
|
+
${this.seatReservations?.map((seatReservation, index) => {
|
|
468
|
+
return html`<li class="sbb-sr__list-item-deck">
|
|
469
|
+
<ul class="sbb-sr__list-coaches" role="presentation">
|
|
470
|
+
${this._renderCoaches(seatReservation, index)}
|
|
471
|
+
</ul>
|
|
472
|
+
</li>`;
|
|
473
|
+
})}
|
|
474
|
+
</ul>
|
|
475
|
+
</div>
|
|
476
|
+
</div>
|
|
477
|
+
</div>
|
|
478
|
+
</div>
|
|
227
479
|
</div>
|
|
228
480
|
</div>
|
|
481
|
+
<div class="sbb-sr-navigation-last-grid">
|
|
482
|
+
${this._renderNavigationControlButton("DIRECTION_RIGHT")}
|
|
483
|
+
</div>
|
|
229
484
|
</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
485
|
</div>
|
|
241
486
|
`;
|
|
242
487
|
}
|
|
243
|
-
|
|
244
|
-
if (!this.
|
|
488
|
+
_renderDeckLabels() {
|
|
489
|
+
if (!this.hasMultipleDecks)
|
|
245
490
|
return null;
|
|
491
|
+
return this.seatReservations.map((seatReservation) => {
|
|
492
|
+
const deckDescription = getI18nSeatReservation(seatReservation.deckCoachLevel, this._language.current);
|
|
493
|
+
return html`<b aria-hidden="true">${deckDescription}</b>`;
|
|
494
|
+
});
|
|
495
|
+
}
|
|
496
|
+
_renderNavigationControlButton(btnDirection) {
|
|
497
|
+
if (!this.hasNavigation || !this.seatReservations)
|
|
498
|
+
return null;
|
|
499
|
+
const btnId = btnDirection == "DIRECTION_RIGHT" ? "last-tab-element" : "first-tab-element";
|
|
500
|
+
const btnIcon = btnDirection == "DIRECTION_RIGHT" ? "arrow-right-small" : "arrow-left-small";
|
|
501
|
+
const btnAriaDescription = btnDirection == "DIRECTION_RIGHT" ? getI18nSeatReservation("SEAT_RESERVATION_END", this._language.current) : getI18nSeatReservation("SEAT_RESERVATION_BEGIN", this._language.current);
|
|
502
|
+
let btnDisabled = true;
|
|
503
|
+
if (btnDirection == "DIRECTION_LEFT" && this.selectedCoachIndex > 0) {
|
|
504
|
+
btnDisabled = false;
|
|
505
|
+
} else if (btnDirection == "DIRECTION_RIGHT" && this.selectedCoachIndex < this.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1) {
|
|
506
|
+
btnDisabled = false;
|
|
246
507
|
}
|
|
247
|
-
return html
|
|
248
|
-
|
|
508
|
+
return html`<sbb-secondary-button
|
|
509
|
+
@click="${() => this.navigateByDirectionBtn(btnDirection)}"
|
|
510
|
+
id="${btnId}"
|
|
511
|
+
class="sbb-sr__navigation-control-button"
|
|
512
|
+
size="s"
|
|
513
|
+
icon-name="${btnIcon}"
|
|
514
|
+
type="button"
|
|
515
|
+
aria-label="${btnAriaDescription}"
|
|
516
|
+
role="contentinfo"
|
|
517
|
+
.disabledInteractive="${btnDisabled || nothing}"
|
|
518
|
+
></sbb-secondary-button>`;
|
|
519
|
+
}
|
|
520
|
+
_renderNavigation() {
|
|
521
|
+
if (!this.hasNavigation || !this.seatReservations)
|
|
522
|
+
return null;
|
|
523
|
+
return html`<div class="sbb-sr-navigation-wrapper">
|
|
524
|
+
<nav id="sbb-sr-navigation" class="sbb-sr-navigation">
|
|
249
525
|
<ul
|
|
526
|
+
id="sbb-sr__navigation-list-coaches"
|
|
250
527
|
class="sbb-sr-navigation__list-coaches"
|
|
251
528
|
aria-label="${getI18nSeatReservation("SEAT_RESERVATION_NAVIGATION", this._language.current)}"
|
|
252
529
|
>
|
|
253
|
-
${this.
|
|
530
|
+
${this.seatReservations[this.currSelectedDeckIndex]?.coachItems.map((coachItem, index) => {
|
|
531
|
+
const coachFreePlaces = this.getAvailableFreePlacesNumFromCoach(index);
|
|
254
532
|
return html`<li>
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
533
|
+
<sbb-seat-reservation-navigation-coach
|
|
534
|
+
@selectcoach=${(event) => this._onSelectNavCoach(event)}
|
|
535
|
+
@focuscoach=${() => this._onFocusNavCoach()}
|
|
536
|
+
class="${classMap({
|
|
537
|
+
"sbb-sr__navigation-coach--hover-scroll": this.hoveredScrollCoachIndex === index
|
|
538
|
+
})}"
|
|
539
|
+
index="${index}"
|
|
540
|
+
coach-id="${coachItem.id}"
|
|
541
|
+
.freePlacesByType="${coachFreePlaces}"
|
|
542
|
+
.selected=${this.selectedCoachIndex === index}
|
|
543
|
+
.focused=${this.focusedCoachIndex === index}
|
|
544
|
+
.propertyIds="${coachItem.propertyIds}"
|
|
545
|
+
.travelClass="${coachItem.travelClass}"
|
|
546
|
+
?driver-area="${!coachItem.places?.length}"
|
|
547
|
+
?first="${coachItem?.driverAreaSide?.left}"
|
|
548
|
+
?last="${coachItem?.driverAreaSide?.right}"
|
|
549
|
+
?vertical="${this.alignVertical}"
|
|
550
|
+
>
|
|
551
|
+
</sbb-seat-reservation-navigation-coach>
|
|
552
|
+
</li>`;
|
|
271
553
|
})}
|
|
272
554
|
</ul>
|
|
273
555
|
</nav>
|
|
274
|
-
|
|
556
|
+
</div>`;
|
|
275
557
|
}
|
|
276
558
|
/**
|
|
277
559
|
*
|
|
278
560
|
* @param coaches
|
|
279
561
|
* @returns
|
|
280
562
|
*/
|
|
281
|
-
_renderCoaches(
|
|
563
|
+
_renderCoaches(seatReservation, deckIndex) {
|
|
564
|
+
const coaches = JSON.parse(JSON.stringify(seatReservation?.coachItems));
|
|
282
565
|
if (!coaches) {
|
|
283
566
|
return null;
|
|
284
567
|
}
|
|
285
|
-
return coaches.map((coachItem,
|
|
568
|
+
return coaches.map((coachItem, coachIndex) => {
|
|
286
569
|
return html`
|
|
287
|
-
<li class="sbb-sr__item-coach"
|
|
570
|
+
<li class="sbb-sr__item-coach">
|
|
571
|
+
${this._renderCoachElement(coachItem, coachIndex, deckIndex)}
|
|
572
|
+
</li>
|
|
288
573
|
`;
|
|
289
574
|
});
|
|
290
575
|
}
|
|
291
|
-
_renderCoachElement(coachItem,
|
|
576
|
+
_renderCoachElement(coachItem, coachIndex, coachDeckIndex) {
|
|
292
577
|
const calculatedCoachDimension = this.getCalculatedDimension(coachItem.dimension);
|
|
293
|
-
const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem);
|
|
294
|
-
return html
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
578
|
+
const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem, coachIndex);
|
|
579
|
+
return html`<sbb-seat-reservation-scoped
|
|
580
|
+
style=${styleMap({
|
|
581
|
+
"--sbb-seat-reservation-scoped-width": calculatedCoachDimension.w,
|
|
582
|
+
"--sbb-seat-reservation-scoped-height": calculatedCoachDimension.h
|
|
583
|
+
})}
|
|
298
584
|
>
|
|
299
|
-
${this._getRenderedCoachBorders(coachItem
|
|
300
|
-
${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension)}
|
|
301
|
-
${this._getRenderedServiceElements(coachItem.serviceElements)}
|
|
585
|
+
${this._getRenderedCoachBorders(coachItem)}
|
|
586
|
+
${this._getRenderedGraphicalElements(coachItem.graphicElements || [], coachItem.dimension, coachIndex, coachDeckIndex)}
|
|
587
|
+
${this._getRenderedServiceElements(coachIndex, coachDeckIndex, coachItem.serviceElements)}
|
|
588
|
+
|
|
302
589
|
<table
|
|
303
|
-
@focus=${() => this.onFocusTableCoachAndPreselectPlace(
|
|
304
|
-
id="sbb-sr-coach-${
|
|
590
|
+
@focus=${() => this.onFocusTableCoachAndPreselectPlace(coachIndex)}
|
|
591
|
+
id="sbb-sr-coach-${coachIndex}"
|
|
305
592
|
class="sbb-sr-coach-wrapper__table"
|
|
306
|
-
aria-describedby="sbb-sr-coach-caption-${
|
|
593
|
+
aria-describedby="sbb-sr-coach-caption-${coachIndex}"
|
|
307
594
|
>
|
|
308
|
-
<caption id="sbb-sr-coach-caption-${
|
|
595
|
+
<caption id="sbb-sr-coach-caption-${coachIndex}" tabindex="-1">
|
|
309
596
|
<sbb-screen-reader-only>${descriptionTableCoachWithServices}</sbb-screen-reader-only>
|
|
310
597
|
</caption>
|
|
311
|
-
${this._getRenderedRowPlaces(coachItem,
|
|
598
|
+
${this._getRenderedRowPlaces(coachItem, coachIndex, coachDeckIndex)}
|
|
312
599
|
</table>
|
|
313
600
|
</sbb-seat-reservation-scoped>`;
|
|
314
601
|
}
|
|
315
|
-
|
|
316
|
-
|
|
602
|
+
/**
|
|
603
|
+
* @returns Returns the border graphic (COACH_BORDER_MIDDLE) of a coach with calculated border gap and coach width,
|
|
604
|
+
* depending on whether the coach is with a driver area or without.
|
|
605
|
+
*/
|
|
606
|
+
_getRenderedCoachBorders(coachItem) {
|
|
317
607
|
const COACH_PASSAGE_WIDTH = 1;
|
|
318
|
-
const
|
|
608
|
+
const allElements = coachItem.graphicElements;
|
|
609
|
+
const driverArea = allElements?.find((element) => element.icon === "DRIVER_AREA");
|
|
319
610
|
const borderWidth = driverArea ? coachItem.dimension.w - driverArea.dimension.w - COACH_PASSAGE_WIDTH : coachItem.dimension.w - COACH_PASSAGE_WIDTH * 2;
|
|
320
|
-
const
|
|
611
|
+
const borderHeight = (coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;
|
|
612
|
+
const borderOffsetX = driverArea && driverArea.position.x === 0 ? driverArea?.dimension.w * this.baseGridSize : this.baseGridSize;
|
|
321
613
|
return html`
|
|
322
|
-
<sbb-seat-reservation-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
</sbb-seat-reservation-scoped>
|
|
614
|
+
<sbb-seat-reservation-graphic
|
|
615
|
+
style=${styleMap({
|
|
616
|
+
"--sbb-seat-reservation-graphic-width": borderWidth * this.baseGridSize,
|
|
617
|
+
"--sbb-seat-reservation-graphic-height": borderHeight,
|
|
618
|
+
"--sbb-seat-reservation-graphic-top": this.coachBorderPadding * -1,
|
|
619
|
+
"--sbb-seat-reservation-graphic-left": borderOffsetX,
|
|
620
|
+
"--sbb-seat-reservation-graphic-position": "absolute"
|
|
621
|
+
})}
|
|
622
|
+
name="COACH_BORDER_MIDDLE"
|
|
623
|
+
?stretch=${true}
|
|
624
|
+
role="presentation"
|
|
625
|
+
></sbb-seat-reservation-graphic>
|
|
335
626
|
`;
|
|
336
627
|
}
|
|
337
|
-
_getRenderedRowPlaces(coach, coachIndex) {
|
|
628
|
+
_getRenderedRowPlaces(coach, coachIndex, deckIndex) {
|
|
338
629
|
if (!coach.places) {
|
|
339
630
|
return null;
|
|
340
631
|
}
|
|
@@ -349,31 +640,32 @@ let SbbSeatReservationElement = (() => {
|
|
|
349
640
|
return Object.values(tableRowPlaces).map((rowPlaces, index) => {
|
|
350
641
|
return html`
|
|
351
642
|
<tr id="row-${coachIndex}-${rowPlaces[0].position.y}" data-row-index=${index}>
|
|
352
|
-
${this._getRenderedColumnPlaces(rowPlaces, coachIndex)}
|
|
643
|
+
${this._getRenderedColumnPlaces(rowPlaces, coachIndex, deckIndex)}
|
|
353
644
|
</tr>
|
|
354
645
|
`;
|
|
355
646
|
}).flatMap((rowTemplate) => rowTemplate);
|
|
356
647
|
}
|
|
357
|
-
_getRenderedColumnPlaces(places, coachIndex) {
|
|
648
|
+
_getRenderedColumnPlaces(places, coachIndex, deckIndex) {
|
|
358
649
|
places.sort((placeA, placeB) => Number(placeA.position.x) - Number(placeB.position.x));
|
|
359
650
|
return places?.map((place, index) => {
|
|
360
|
-
const
|
|
361
|
-
const
|
|
651
|
+
const calculatedDimension = this.getCalculatedDimension(place.dimension);
|
|
652
|
+
const calculatedPosition = this.getCalculatedPosition(place.position);
|
|
362
653
|
const textRotation = this.alignVertical ? -90 : 0;
|
|
363
654
|
return html`
|
|
364
|
-
<
|
|
365
|
-
|
|
366
|
-
|
|
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}"
|
|
655
|
+
<td
|
|
656
|
+
id="cell-${deckIndex}-${coachIndex}-${place.position.y}-${index}"
|
|
657
|
+
class="graphical-element"
|
|
372
658
|
>
|
|
373
659
|
<sbb-seat-reservation-place-control
|
|
660
|
+
style=${styleMap({
|
|
661
|
+
"--sbb-seat-reservation-place-control-width": calculatedDimension.w,
|
|
662
|
+
"--sbb-seat-reservation-place-control-height": calculatedDimension.h,
|
|
663
|
+
"--sbb-seat-reservation-place-control-top": calculatedPosition.y,
|
|
664
|
+
"--sbb-seat-reservation-place-control-left": calculatedPosition.x
|
|
665
|
+
})}
|
|
374
666
|
@selectplace=${(selectPlaceEvent) => this._onSelectPlace(selectPlaceEvent)}
|
|
375
667
|
exportparts="sbb-sr-place-part"
|
|
376
|
-
id="seat-reservation__place-button-${coachIndex}-${place.number}"
|
|
668
|
+
id="seat-reservation__place-button-${deckIndex}-${coachIndex}-${place.number}"
|
|
377
669
|
class="seat-reservation-place-control"
|
|
378
670
|
data-cell-id="${coachIndex}-${place.position.y}-${index}"
|
|
379
671
|
text=${place.number}
|
|
@@ -384,17 +676,17 @@ let SbbSeatReservationElement = (() => {
|
|
|
384
676
|
rotation=${place.rotation ?? nothing}
|
|
385
677
|
text-rotation=${textRotation}
|
|
386
678
|
coach-index=${coachIndex}
|
|
679
|
+
deck-index=${deckIndex}
|
|
387
680
|
.propertyIds=${place.propertyIds}
|
|
388
681
|
.preventClick=${this.preventPlaceClick}
|
|
389
682
|
></sbb-seat-reservation-place-control>
|
|
390
|
-
</
|
|
683
|
+
</td>
|
|
391
684
|
`;
|
|
392
685
|
});
|
|
393
686
|
}
|
|
394
|
-
_getRenderedGraphicalElements(graphicalElements, coachDimension) {
|
|
395
|
-
if (!graphicalElements)
|
|
687
|
+
_getRenderedGraphicalElements(graphicalElements, coachDimension, coachIndex, coachDeckIndex) {
|
|
688
|
+
if (!graphicalElements)
|
|
396
689
|
return null;
|
|
397
|
-
}
|
|
398
690
|
return graphicalElements?.map((graphicalElement) => {
|
|
399
691
|
const icon = graphicalElement.icon ?? "";
|
|
400
692
|
const elementRotation = graphicalElement.rotation || 0;
|
|
@@ -403,16 +695,25 @@ let SbbSeatReservationElement = (() => {
|
|
|
403
695
|
if (this._notAreaElements.findIndex((notAreaElement) => notAreaElement === icon) > -1) {
|
|
404
696
|
return this._getRenderElementWithoutArea(graphicalElement, elementRotation, coachDimension);
|
|
405
697
|
}
|
|
406
|
-
return this._getRenderElementWithArea(graphicalElement, elementFixedRotation, coachDimension);
|
|
698
|
+
return this._getRenderElementWithArea(graphicalElement, elementFixedRotation, coachDimension, coachIndex, coachDeckIndex);
|
|
407
699
|
});
|
|
408
700
|
}
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
701
|
+
/**
|
|
702
|
+
* creates a rendered element with an area component
|
|
703
|
+
* @param graphicalElement
|
|
704
|
+
* @param rotation
|
|
705
|
+
* @param coachDimension
|
|
706
|
+
* @param coachIndex used to generate a unique id for the popover trigger
|
|
707
|
+
* @private
|
|
708
|
+
*/
|
|
709
|
+
_getRenderElementWithArea(graphicalElement, rotation, coachDimension, coachIndex, coachDeckIndex) {
|
|
710
|
+
const isNotTableGraphic = graphicalElement.icon?.indexOf("TABLE") === -1;
|
|
711
|
+
const areaProperty = graphicalElement.icon && isNotTableGraphic ? graphicalElement.icon : null;
|
|
412
712
|
const stretchHeight = areaProperty !== "ENTRY_EXIT";
|
|
413
713
|
const ariaLabelForArea = graphicalElement.icon ? getI18nSeatReservation(graphicalElement.icon, this._language.current) : nothing;
|
|
414
714
|
const calculatedDimension = this.getCalculatedDimension(graphicalElement.dimension, coachDimension, true, stretchHeight);
|
|
415
715
|
const calculatedPosition = this.getCalculatedPosition(graphicalElement.position, graphicalElement.dimension, coachDimension, true);
|
|
716
|
+
const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
|
|
416
717
|
let elementMounting = "free";
|
|
417
718
|
if (graphicalElement.position.y === this.coachBorderOffset * -1) {
|
|
418
719
|
elementMounting = "upper-border";
|
|
@@ -420,91 +721,78 @@ let SbbSeatReservationElement = (() => {
|
|
|
420
721
|
elementMounting = "lower-border";
|
|
421
722
|
}
|
|
422
723
|
return html`
|
|
423
|
-
<sbb-seat-reservation-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
width
|
|
428
|
-
height
|
|
429
|
-
|
|
430
|
-
|
|
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
|
|
724
|
+
<sbb-seat-reservation-area
|
|
725
|
+
id="${triggerId}"
|
|
726
|
+
class="${classMap({ "sbb-seat-reservation-area--cursor-pointer": areaProperty !== null })}"
|
|
727
|
+
style=${styleMap({
|
|
728
|
+
"--sbb-seat-reservation-area-width": calculatedDimension.w,
|
|
729
|
+
"--sbb-seat-reservation-area-height": calculatedDimension.h,
|
|
730
|
+
"--sbb-seat-reservation-area-top": calculatedPosition.y,
|
|
731
|
+
"--sbb-seat-reservation-area-left": calculatedPosition.x
|
|
435
732
|
})}
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
</sbb-seat-reservation-scoped>
|
|
733
|
+
mounting=${elementMounting}
|
|
734
|
+
background="dark"
|
|
735
|
+
aria-hidden="true"
|
|
736
|
+
>
|
|
737
|
+
${areaProperty ? html`
|
|
738
|
+
<sbb-seat-reservation-graphic
|
|
739
|
+
name=${areaProperty}
|
|
740
|
+
width=${this.baseGridSize}
|
|
741
|
+
height=${this.baseGridSize}
|
|
742
|
+
rotation=${rotation}
|
|
743
|
+
role="img"
|
|
744
|
+
aria-hidden="true"
|
|
745
|
+
></sbb-seat-reservation-graphic>
|
|
746
|
+
` : nothing}
|
|
747
|
+
</sbb-seat-reservation-area>
|
|
748
|
+
${areaProperty ? this._popover(triggerId, ariaLabelForArea) : nothing}
|
|
453
749
|
`;
|
|
454
750
|
}
|
|
455
751
|
_getRenderElementWithoutArea(graphicalElement, rotation, coachDimension) {
|
|
456
752
|
const calculatedDimension = this.getCalculatedDimension(graphicalElement.dimension, coachDimension);
|
|
457
753
|
const calculatedPosition = this.getCalculatedPosition(graphicalElement.position, graphicalElement.dimension, coachDimension);
|
|
458
|
-
const icon = graphicalElement.icon && graphicalElement.icon.indexOf("DRIVER_AREA") === -1 ? graphicalElement.icon : graphicalElement.icon?.concat("_", this.
|
|
459
|
-
return html`
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
rotation=${rotation}
|
|
473
|
-
aria-hidden="true"
|
|
474
|
-
?stretch=${true}
|
|
475
|
-
></sbb-seat-reservation-graphic>
|
|
476
|
-
</sbb-seat-reservation-scoped>
|
|
477
|
-
`;
|
|
754
|
+
const icon = graphicalElement.icon && graphicalElement.icon.indexOf("DRIVER_AREA") === -1 ? graphicalElement.icon : graphicalElement.icon?.concat("_", this.seatReservations[this.currSelectedDeckIndex].vehicleType);
|
|
755
|
+
return html` <sbb-seat-reservation-graphic
|
|
756
|
+
style=${styleMap({
|
|
757
|
+
"--sbb-seat-reservation-graphic-width": calculatedDimension.w,
|
|
758
|
+
"--sbb-seat-reservation-graphic-height": calculatedDimension.h,
|
|
759
|
+
"--sbb-seat-reservation-graphic-top": calculatedPosition.y,
|
|
760
|
+
"--sbb-seat-reservation-graphic-left": calculatedPosition.x,
|
|
761
|
+
"--sbb-seat-reservation-graphic-position": "absolute"
|
|
762
|
+
})}
|
|
763
|
+
name=${icon ?? nothing}
|
|
764
|
+
rotation=${rotation}
|
|
765
|
+
aria-hidden="true"
|
|
766
|
+
?stretch=${true}
|
|
767
|
+
></sbb-seat-reservation-graphic>`;
|
|
478
768
|
}
|
|
479
|
-
_getRenderedServiceElements(serviceElements) {
|
|
480
|
-
if (!serviceElements)
|
|
769
|
+
_getRenderedServiceElements(coachIndex, coachDeckIndex, serviceElements) {
|
|
770
|
+
if (!serviceElements)
|
|
481
771
|
return null;
|
|
482
|
-
}
|
|
483
772
|
return serviceElements?.map((serviceElement) => {
|
|
484
773
|
const titleDescription = serviceElement.icon ? getI18nSeatReservation(serviceElement.icon, this._language.current) : null;
|
|
485
|
-
const
|
|
486
|
-
const
|
|
774
|
+
const calculatedDimension = this.getCalculatedDimension(serviceElement.dimension);
|
|
775
|
+
const calculatedPosition = this.getCalculatedPosition(serviceElement.position);
|
|
487
776
|
const elementRotation = serviceElement.rotation || 0;
|
|
488
777
|
const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;
|
|
778
|
+
const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;
|
|
489
779
|
return html`
|
|
490
|
-
<sbb-seat-reservation-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
></sbb-seat-reservation-graphic>
|
|
507
|
-
</sbb-seat-reservation-scoped>
|
|
780
|
+
<sbb-seat-reservation-graphic
|
|
781
|
+
id="${triggerId}"
|
|
782
|
+
style=${styleMap({
|
|
783
|
+
"--sbb-seat-reservation-graphic-width": calculatedDimension.w,
|
|
784
|
+
"--sbb-seat-reservation-graphic-height": calculatedDimension.h,
|
|
785
|
+
"--sbb-seat-reservation-graphic-top": calculatedPosition.y,
|
|
786
|
+
"--sbb-seat-reservation-graphic-left": calculatedPosition.x,
|
|
787
|
+
"--sbb-seat-reservation-graphic-position": "absolute"
|
|
788
|
+
})}
|
|
789
|
+
class="sbb-seat-reservation-graphic--cursor-pointer"
|
|
790
|
+
name=${serviceElement.icon ?? nothing}
|
|
791
|
+
.rotation=${elementFixedRotation}
|
|
792
|
+
role="img"
|
|
793
|
+
aria-hidden="true"
|
|
794
|
+
></sbb-seat-reservation-graphic>
|
|
795
|
+
${this._popover(triggerId, titleDescription)}
|
|
508
796
|
`;
|
|
509
797
|
});
|
|
510
798
|
}
|
|
@@ -518,6 +806,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
518
806
|
this.preventCoachScrollByPlaceClick = true;
|
|
519
807
|
this.isCoachGridFocusable = false;
|
|
520
808
|
if (!this.preventPlaceClick) {
|
|
809
|
+
this.currSelectedDeckIndex = selectedPlace.deckIndex;
|
|
521
810
|
this.updateSelectedSeatReservationPlaces(selectedPlace);
|
|
522
811
|
this.updateCurrentSelectedPlaceInCoach(selectedPlace);
|
|
523
812
|
}
|
|
@@ -532,6 +821,7 @@ let SbbSeatReservationElement = (() => {
|
|
|
532
821
|
this.updateCurrentSelectedCoach();
|
|
533
822
|
this.preselectPlaceInCoach();
|
|
534
823
|
}
|
|
824
|
+
this._closePopover();
|
|
535
825
|
}
|
|
536
826
|
_onFocusNavCoach() {
|
|
537
827
|
if (!this.preventCoachScrollByPlaceClick) {
|
|
@@ -541,18 +831,43 @@ let SbbSeatReservationElement = (() => {
|
|
|
541
831
|
}
|
|
542
832
|
this.isAutoScrolling = false;
|
|
543
833
|
}
|
|
544
|
-
|
|
834
|
+
/**
|
|
835
|
+
* Creates a popover for extra service information
|
|
836
|
+
* @param triggerId
|
|
837
|
+
* @param popoverContent
|
|
838
|
+
* @private
|
|
839
|
+
*/
|
|
840
|
+
_popover(triggerId, popoverContent) {
|
|
841
|
+
return html`
|
|
842
|
+
<sbb-popover trigger="${triggerId}">
|
|
843
|
+
<p class="sbb-text-s sbb-sr-popover">${popoverContent}</p>
|
|
844
|
+
</sbb-popover>
|
|
845
|
+
`;
|
|
846
|
+
}
|
|
847
|
+
/**
|
|
848
|
+
* trigger to close all opened popovers (normally only one is opened at a time)
|
|
849
|
+
* @private
|
|
850
|
+
*/
|
|
851
|
+
_closePopover() {
|
|
852
|
+
this.shadowRoot?.querySelectorAll('sbb-popover[data-state="opened"]').forEach((popover) => popover.setAttribute("data-state", "closed"));
|
|
853
|
+
}
|
|
854
|
+
_getDescriptionTableCoach(coachItem, coachIndex) {
|
|
545
855
|
if (!coachItem.places?.length) {
|
|
546
856
|
return getI18nSeatReservation("COACH_BLOCKED_TABLE_CAPTION", this._language.current, [
|
|
547
857
|
coachItem.id
|
|
548
858
|
]);
|
|
549
859
|
}
|
|
550
|
-
let tableCoachDescription
|
|
860
|
+
let tableCoachDescription;
|
|
551
861
|
const areaDescriptions = this._getTitleDescriptionListString(coachItem.graphicElements);
|
|
552
862
|
const serviceDescriptions = this._getTitleDescriptionListString(coachItem.serviceElements);
|
|
553
863
|
tableCoachDescription = getI18nSeatReservation("COACH_TABLE_CAPTION", this._language.current, [
|
|
554
864
|
coachItem.id
|
|
555
865
|
]);
|
|
866
|
+
if (!this.hasNavigation) {
|
|
867
|
+
const freePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);
|
|
868
|
+
const freePlacesTxt = getI18nSeatReservation("COACH_AVAILABLE_NUMBER_OF_PLACES", this._language.current, [freePlaces.seats, freePlaces.bicycles]);
|
|
869
|
+
tableCoachDescription = tableCoachDescription.concat(". ").concat(freePlacesTxt).concat(". ");
|
|
870
|
+
}
|
|
556
871
|
if (!!areaDescriptions || !!serviceDescriptions) {
|
|
557
872
|
tableCoachDescription += ". " + getI18nSeatReservation("COACH_AVAILABLE_SERVICES", this._language.current) + ": ";
|
|
558
873
|
tableCoachDescription += serviceDescriptions + ", " + areaDescriptions + ".";
|
|
@@ -574,28 +889,8 @@ let SbbSeatReservationElement = (() => {
|
|
|
574
889
|
return !!translation && !descriptionAlreayExist && isValidDescription ? translation : null;
|
|
575
890
|
}).filter((description) => !!description).join(", ");
|
|
576
891
|
}
|
|
577
|
-
},
|
|
892
|
+
}, _classThis = _a, (() => {
|
|
578
893
|
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
894
|
__esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
|
|
600
895
|
_classThis = _classDescriptor.value;
|
|
601
896
|
if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
|
|
@@ -605,4 +900,4 @@ let SbbSeatReservationElement = (() => {
|
|
|
605
900
|
export {
|
|
606
901
|
SbbSeatReservationElement
|
|
607
902
|
};
|
|
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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"}
|
|
903
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"seat-reservation.component.js","sources":["../../../../../src/elements-experimental/seat-reservation/seat-reservation/seat-reservation.component.ts"],"sourcesContent":["import { SbbLanguageController } from '@sbb-esta/lyne-elements/core/controllers.js';\nimport { html, isServer, nothing } from 'lit';\nimport type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { customElement } from 'lit/decorators.js';\nimport { classMap } from 'lit/directives/class-map.js';\nimport { styleMap } from 'lit/directives/style-map.js';\n\nimport { getI18nSeatReservation } from '../common.js';\nimport type {\n  CoachItem,\n  Place,\n  ElementDimension,\n  BaseElement,\n  PlaceSelection,\n  SeatReservation,\n} from '../common.js';\n\nimport { SeatReservationBaseElement } from './seat-reservation-base-element.js';\nimport style from './seat-reservation.scss?lit&inline';\n\nimport '@sbb-esta/lyne-elements/button.js';\nimport '@sbb-esta/lyne-elements/screen-reader-only.js';\nimport '../seat-reservation-area.js';\nimport '../seat-reservation-graphic.js';\nimport '../seat-reservation-place-control.js';\nimport '../seat-reservation-navigation-coach.js';\nimport '../seat-reservation-scoped.js';\nimport '@sbb-esta/lyne-elements/popover/popover.js';\n\n/**\n * Main component for the seat reservation.\n *\n */\nexport\n@customElement('sbb-seat-reservation')\nclass SbbSeatReservationElement extends SeatReservationBaseElement {\n  public static override styles: CSSResultGroup = style;\n\n  private _language = new SbbLanguageController(this);\n  private _coachesHtmlTemplate?: TemplateResult;\n  // Graphics that should not be rendered with an area\n  private _notAreaElements = [\n    'DRIVER_AREA',\n    'COACH_PASSAGE',\n    'COACH_WALL_NO_PASSAGE',\n    'COMPARTMENT_PASSAGE',\n    'COMPARTMENT_PASSAGE_HIGH',\n    'COMPARTMENT_PASSAGE_MIDDLE',\n    'COMPARTMENT_PASSAGE_LOW',\n  ];\n\n  // Area icons that should not be fixed during rotation when vertical mode is selected\n  private _notFixedRotatableAreaIcons = ['ENTRY_EXIT'];\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('hasNavigation')) {\n      if (this.hasNavigation) {\n        this.shadowRoot\n          ?.querySelectorAll('table')\n          .forEach((table) => table.removeAttribute('tabindex'));\n      } else {\n        this.shadowRoot\n          ?.querySelectorAll('table')\n          .forEach((table) => table.setAttribute('tabindex', '0'));\n      }\n    }\n  }\n\n  protected override firstUpdated(changedProperties: PropertyValues<this>): void {\n    super.firstUpdated(changedProperties);\n\n    // We need to wait until the first update is complete to init different html element dimensions\n    this.updateComplete.then(() => {\n      this.initPrepairSeatReservationData();\n      this.initNavigationSelectionByScrollEvent();\n\n      // We have to initiate the render process manually so that the prepared data is applied directly\n      // and not only after an interaction with the component itself\n      this.requestUpdate();\n    });\n  }\n\n  protected override render(): TemplateResult | null {\n    if (!isServer) {\n      this._determineBaseFontSize();\n    }\n\n    this._initVehicleSeatReservationConstruction();\n    return this._coachesHtmlTemplate || null;\n  }\n\n  private _determineBaseFontSize(): void {\n    const baseFontSize = parseInt(window.getComputedStyle(document.body).fontSize, 10);\n    //calculate rem of 1px\n    const onePixelInRem = 1 / baseFontSize;\n    this.style?.setProperty('--sbb-seat-reservation-one-px-rem', `${onePixelInRem + 'rem'}`);\n  }\n\n  private _initVehicleSeatReservationConstruction(): void {\n    const classAlignVertical = this.alignVertical ? 'sbb-sr__grid--vertical' : '';\n    this._coachesHtmlTemplate = html`\n      <div class=\"sbb-sr__container\">\n        <div class=\"sbb-sr sbb-sr__grid ${classAlignVertical}\">\n          <div class=\"sbb-sr-navigation-first-grid\">\n            ${this._renderNavigationControlButton('DIRECTION_LEFT')}\n          </div>\n          <div\n            class=\"sbb-sr__component\"\n            @keydown=${(evt: KeyboardEvent) => this.handleKeyboardEvent(evt)}\n          >\n            <div class=\"sbb-sr-grid-inner\">\n              <div class=\"nav-grid\">${this._renderNavigation()}</div>\n              <div class=\"coaches-grid\">\n                <div class=\"sbb-sr__wrapper-coach-decks\">\n                  <div class=\"sbb-sr__wrapper-deck-labels\">${this._renderDeckLabels()}</div>\n                  <div id=\"sbb-sr__wrapper-scrollarea\" class=\"sbb-sr__wrapper\">\n                    <div id=\"sbb-sr__parent-area\" class=\"sbb-sr__parent\" tabindex=\"-1\">\n                      <ul\n                        class=\"${classMap({\n                          'sbb-sr__list-decks': true,\n                          'sbb-sr__list-decks--gap': this.hasMultipleDecks,\n                        })}\"\n                      >\n                        ${this.seatReservations?.map(\n                          (seatReservation: SeatReservation, index: number) => {\n                            return html`<li class=\"sbb-sr__list-item-deck\">\n                              <ul class=\"sbb-sr__list-coaches\" role=\"presentation\">\n                                ${this._renderCoaches(seatReservation, index)}\n                              </ul>\n                            </li>`;\n                          },\n                        )}\n                      </ul>\n                    </div>\n                  </div>\n                </div>\n              </div>\n            </div>\n          </div>\n          <div class=\"sbb-sr-navigation-last-grid\">\n            ${this._renderNavigationControlButton('DIRECTION_RIGHT')}\n          </div>\n        </div>\n      </div>\n    `;\n  }\n\n  private _renderDeckLabels(): TemplateResult[] | null {\n    if (!this.hasMultipleDecks) return null;\n\n    return this.seatReservations.map((seatReservation) => {\n      const deckDescription = getI18nSeatReservation(\n        seatReservation.deckCoachLevel,\n        this._language.current,\n      );\n      return html`<b aria-hidden=\"true\">${deckDescription}</b>`;\n    });\n  }\n\n  private _renderNavigationControlButton(btnDirection: string): TemplateResult | null {\n    if (!this.hasNavigation || !this.seatReservations) return null;\n\n    const btnId = btnDirection == 'DIRECTION_RIGHT' ? 'last-tab-element' : 'first-tab-element';\n    const btnIcon = btnDirection == 'DIRECTION_RIGHT' ? 'arrow-right-small' : 'arrow-left-small';\n    const btnAriaDescription =\n      btnDirection == 'DIRECTION_RIGHT'\n        ? getI18nSeatReservation('SEAT_RESERVATION_END', this._language.current)\n        : getI18nSeatReservation('SEAT_RESERVATION_BEGIN', this._language.current);\n    let btnDisabled = true;\n\n    if (btnDirection == 'DIRECTION_LEFT' && this.selectedCoachIndex > 0) {\n      btnDisabled = false;\n    } else if (\n      btnDirection == 'DIRECTION_RIGHT' &&\n      this.selectedCoachIndex <\n        this.seatReservations[this.currSelectedDeckIndex].coachItems.length - 1\n    ) {\n      btnDisabled = false;\n    }\n\n    return html`<sbb-secondary-button\n      @click=\"${() => this.navigateByDirectionBtn(btnDirection)}\"\n      id=\"${btnId}\"\n      class=\"sbb-sr__navigation-control-button\"\n      size=\"s\"\n      icon-name=\"${btnIcon}\"\n      type=\"button\"\n      aria-label=\"${btnAriaDescription}\"\n      role=\"contentinfo\"\n      .disabledInteractive=\"${btnDisabled || nothing}\"\n    ></sbb-secondary-button>`;\n  }\n\n  private _renderNavigation(): TemplateResult | null {\n    if (!this.hasNavigation || !this.seatReservations) return null;\n\n    return html`<div class=\"sbb-sr-navigation-wrapper\">\n      <nav id=\"sbb-sr-navigation\" class=\"sbb-sr-navigation\">\n        <ul\n          id=\"sbb-sr__navigation-list-coaches\"\n          class=\"sbb-sr-navigation__list-coaches\"\n          aria-label=\"${getI18nSeatReservation(\n            'SEAT_RESERVATION_NAVIGATION',\n            this._language.current,\n          )}\"\n        >\n          ${this.seatReservations[this.currSelectedDeckIndex]?.coachItems.map(\n            (coachItem: CoachItem, index: number) => {\n              const coachFreePlaces = this.getAvailableFreePlacesNumFromCoach(index);\n              return html`<li>\n                <sbb-seat-reservation-navigation-coach\n                  @selectcoach=${(event: CustomEvent<number>) => this._onSelectNavCoach(event)}\n                  @focuscoach=${() => this._onFocusNavCoach()}\n                  class=\"${classMap({\n                    'sbb-sr__navigation-coach--hover-scroll':\n                      this.hoveredScrollCoachIndex === index,\n                  })}\"\n                  index=\"${index}\"\n                  coach-id=\"${coachItem.id}\"\n                  .freePlacesByType=\"${coachFreePlaces}\"\n                  .selected=${this.selectedCoachIndex === index}\n                  .focused=${this.focusedCoachIndex === index}\n                  .propertyIds=\"${coachItem.propertyIds}\"\n                  .travelClass=\"${coachItem.travelClass}\"\n                  ?driver-area=\"${!coachItem.places?.length}\"\n                  ?first=\"${coachItem?.driverAreaSide?.left}\"\n                  ?last=\"${coachItem?.driverAreaSide?.right}\"\n                  ?vertical=\"${this.alignVertical}\"\n                >\n                </sbb-seat-reservation-navigation-coach>\n              </li>`;\n            },\n          )}\n        </ul>\n      </nav>\n    </div>`;\n  }\n  /**\n   *\n   * @param coaches\n   * @returns\n   */\n  private _renderCoaches(\n    seatReservation: SeatReservation,\n    deckIndex: number,\n  ): TemplateResult[] | null {\n    const coaches = JSON.parse(JSON.stringify(seatReservation?.coachItems));\n\n    if (!coaches) {\n      return null;\n    }\n    return coaches.map((coachItem: CoachItem, coachIndex: number) => {\n      return html`\n        <li class=\"sbb-sr__item-coach\">\n          ${this._renderCoachElement(coachItem, coachIndex, deckIndex)}\n        </li>\n      `;\n    });\n  }\n\n  private _renderCoachElement(\n    coachItem: CoachItem,\n    coachIndex: number,\n    coachDeckIndex: number,\n  ): TemplateResult {\n    const calculatedCoachDimension = this.getCalculatedDimension(coachItem.dimension);\n    const descriptionTableCoachWithServices = this._getDescriptionTableCoach(coachItem, coachIndex);\n\n    return html`<sbb-seat-reservation-scoped\n      style=${styleMap({\n        '--sbb-seat-reservation-scoped-width': calculatedCoachDimension.w,\n        '--sbb-seat-reservation-scoped-height': calculatedCoachDimension.h,\n      })}\n    >\n      ${this._getRenderedCoachBorders(coachItem)}\n      ${this._getRenderedGraphicalElements(\n        coachItem.graphicElements || [],\n        coachItem.dimension,\n        coachIndex,\n        coachDeckIndex,\n      )}\n      ${this._getRenderedServiceElements(coachIndex, coachDeckIndex, coachItem.serviceElements)}\n\n      <table\n        @focus=${() => this.onFocusTableCoachAndPreselectPlace(coachIndex)}\n        id=\"sbb-sr-coach-${coachIndex}\"\n        class=\"sbb-sr-coach-wrapper__table\"\n        aria-describedby=\"sbb-sr-coach-caption-${coachIndex}\"\n      >\n        <caption id=\"sbb-sr-coach-caption-${coachIndex}\" tabindex=\"-1\">\n          <sbb-screen-reader-only>${descriptionTableCoachWithServices}</sbb-screen-reader-only>\n        </caption>\n        ${this._getRenderedRowPlaces(coachItem, coachIndex, coachDeckIndex)}\n      </table>\n    </sbb-seat-reservation-scoped>`;\n  }\n\n  /**\n   * @returns Returns the border graphic (COACH_BORDER_MIDDLE) of a coach with calculated border gap and coach width,\n   * depending on whether the coach is with a driver area or without.\n   */\n  private _getRenderedCoachBorders(coachItem: CoachItem): TemplateResult {\n    const COACH_PASSAGE_WIDTH = 1;\n    const allElements = coachItem.graphicElements;\n    const driverArea = allElements?.find((element: BaseElement) => element.icon === 'DRIVER_AREA');\n    const borderWidth = driverArea\n      ? coachItem.dimension.w - driverArea.dimension.w - COACH_PASSAGE_WIDTH\n      : coachItem.dimension.w - COACH_PASSAGE_WIDTH * 2;\n    const borderHeight = (coachItem.dimension.h + this.coachBorderOffset * 2) * this.baseGridSize;\n    const borderOffsetX =\n      driverArea && driverArea.position.x === 0\n        ? driverArea?.dimension.w * this.baseGridSize\n        : this.baseGridSize;\n    return html`\n      <sbb-seat-reservation-graphic\n        style=${styleMap({\n          '--sbb-seat-reservation-graphic-width': borderWidth * this.baseGridSize,\n          '--sbb-seat-reservation-graphic-height': borderHeight,\n          '--sbb-seat-reservation-graphic-top': this.coachBorderPadding * -1,\n          '--sbb-seat-reservation-graphic-left': borderOffsetX,\n          '--sbb-seat-reservation-graphic-position': 'absolute',\n        })}\n        name=\"COACH_BORDER_MIDDLE\"\n        ?stretch=${true}\n        role=\"presentation\"\n      ></sbb-seat-reservation-graphic>\n    `;\n  }\n\n  private _getRenderedRowPlaces(\n    coach: CoachItem,\n    coachIndex: number,\n    deckIndex: number,\n  ): TemplateResult[] | null {\n    if (!coach.places) {\n      return null;\n    }\n\n    // Prepair rows with the places to render a table\n    const tableRowPlaces: Record<number, Place[]> = {};\n    for (const place of coach.places) {\n      if (!tableRowPlaces[place.position.y]) {\n        tableRowPlaces[place.position.y] = [place];\n      } else {\n        tableRowPlaces[place.position.y].push(place);\n      }\n    }\n\n    return Object.values(tableRowPlaces)\n      .map((rowPlaces: Place[], index) => {\n        return html`\n          <tr id=\"row-${coachIndex}-${rowPlaces[0].position.y}\" data-row-index=${index}>\n            ${this._getRenderedColumnPlaces(rowPlaces, coachIndex, deckIndex)}\n          </tr>\n        `;\n      })\n      .flatMap((rowTemplate) => rowTemplate);\n  }\n\n  private _getRenderedColumnPlaces(\n    places: Place[],\n    coachIndex: number,\n    deckIndex: number,\n  ): TemplateResult[] | null {\n    //Sorts each place by its ascending x coordinate\n    places.sort(\n      (placeA: Place, placeB: Place) => Number(placeA.position.x) - Number(placeB.position.x),\n    );\n\n    return places?.map((place: Place, index: number) => {\n      const calculatedDimension = this.getCalculatedDimension(place.dimension);\n      const calculatedPosition = this.getCalculatedPosition(place.position);\n      const textRotation = this.alignVertical ? -90 : 0;\n\n      return html`\n        <td\n          id=\"cell-${deckIndex}-${coachIndex}-${place.position.y}-${index}\"\n          class=\"graphical-element\"\n        >\n          <sbb-seat-reservation-place-control\n            style=${styleMap({\n              '--sbb-seat-reservation-place-control-width': calculatedDimension.w,\n              '--sbb-seat-reservation-place-control-height': calculatedDimension.h,\n              '--sbb-seat-reservation-place-control-top': calculatedPosition.y,\n              '--sbb-seat-reservation-place-control-left': calculatedPosition.x,\n            })}\n            @selectplace=${(selectPlaceEvent: CustomEvent<PlaceSelection>) =>\n              this._onSelectPlace(selectPlaceEvent)}\n            exportparts=\"sbb-sr-place-part\"\n            id=\"seat-reservation__place-button-${deckIndex}-${coachIndex}-${place.number}\"\n            class=\"seat-reservation-place-control\"\n            data-cell-id=\"${coachIndex}-${place.position.y}-${index}\"\n            text=${place.number}\n            type=${place.type}\n            state=${place.state}\n            width=${place.dimension.w * this.baseGridSize}\n            height=${place.dimension.h * this.baseGridSize}\n            rotation=${place.rotation ?? nothing}\n            text-rotation=${textRotation}\n            coach-index=${coachIndex}\n            deck-index=${deckIndex}\n            .propertyIds=${place.propertyIds}\n            .preventClick=${this.preventPlaceClick}\n          ></sbb-seat-reservation-place-control>\n        </td>\n      `;\n    });\n  }\n\n  private _getRenderedGraphicalElements(\n    graphicalElements: BaseElement[],\n    coachDimension: ElementDimension,\n    coachIndex: number,\n    coachDeckIndex: number,\n  ): TemplateResult[] | null {\n    if (!graphicalElements) return null;\n\n    return graphicalElements?.map((graphicalElement: BaseElement) => {\n      const icon = graphicalElement.icon ?? '';\n      const elementRotation = graphicalElement.rotation || 0;\n      const isNotFixedRotationGraphicalElement =\n        this._notFixedRotatableAreaIcons.indexOf(graphicalElement.icon!) === -1;\n      const elementFixedRotation =\n        this.alignVertical && isNotFixedRotationGraphicalElement\n          ? elementRotation - 90\n          : elementRotation;\n\n      //check if the current element is not an area element, since this element is drawn without an area component\n      if (this._notAreaElements.findIndex((notAreaElement) => notAreaElement === icon) > -1) {\n        return this._getRenderElementWithoutArea(graphicalElement, elementRotation, coachDimension);\n      }\n      return this._getRenderElementWithArea(\n        graphicalElement,\n        elementFixedRotation,\n        coachDimension,\n        coachIndex,\n        coachDeckIndex,\n      );\n    });\n  }\n\n  /**\n   * creates a rendered element with an area component\n   * @param graphicalElement\n   * @param rotation\n   * @param coachDimension\n   * @param coachIndex used to generate a unique id for the popover trigger\n   * @private\n   */\n  private _getRenderElementWithArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n    coachIndex: number,\n    coachDeckIndex: number,\n  ): TemplateResult {\n    // for TABLE, we use the area component itself to display the table instead of the SVG graphic.\n    // Due to different heights and widths, it wouldn't show correctly. To correct this, we would\n    // need difficult calculations for position, rotation and dimension.\n    const isNotTableGraphic = graphicalElement.icon?.indexOf('TABLE') === -1;\n    const areaProperty = graphicalElement.icon && isNotTableGraphic ? graphicalElement.icon : null;\n    const stretchHeight = areaProperty !== 'ENTRY_EXIT';\n    const ariaLabelForArea = graphicalElement.icon\n      ? getI18nSeatReservation(graphicalElement.icon, this._language.current)\n      : nothing;\n    const calculatedDimension = this.getCalculatedDimension(\n      graphicalElement.dimension,\n      coachDimension,\n      true,\n      stretchHeight,\n    );\n    const calculatedPosition = this.getCalculatedPosition(\n      graphicalElement.position,\n      graphicalElement.dimension,\n      coachDimension,\n      true,\n    );\n\n    //generate unique index number for the trigger element\n    const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;\n    let elementMounting = 'free';\n\n    if (graphicalElement.position.y === this.coachBorderOffset * -1) {\n      elementMounting = 'upper-border';\n    } else if (\n      graphicalElement.position.y + graphicalElement.dimension.h ===\n      coachDimension.h + this.coachBorderOffset\n    ) {\n      elementMounting = 'lower-border';\n    }\n\n    return html`\n      <sbb-seat-reservation-area\n        id=\"${triggerId}\"\n        class=\"${classMap({ 'sbb-seat-reservation-area--cursor-pointer': areaProperty !== null })}\"\n        style=${styleMap({\n          '--sbb-seat-reservation-area-width': calculatedDimension.w,\n          '--sbb-seat-reservation-area-height': calculatedDimension.h,\n          '--sbb-seat-reservation-area-top': calculatedPosition.y,\n          '--sbb-seat-reservation-area-left': calculatedPosition.x,\n        })}\n        mounting=${elementMounting}\n        background=\"dark\"\n        aria-hidden=\"true\"\n      >\n        ${areaProperty\n          ? html`\n              <sbb-seat-reservation-graphic\n                name=${areaProperty}\n                width=${this.baseGridSize}\n                height=${this.baseGridSize}\n                rotation=${rotation}\n                role=\"img\"\n                aria-hidden=\"true\"\n              ></sbb-seat-reservation-graphic>\n            `\n          : nothing}\n      </sbb-seat-reservation-area>\n      ${areaProperty ? this._popover(triggerId, ariaLabelForArea) : nothing}\n    `;\n  }\n\n  private _getRenderElementWithoutArea(\n    graphicalElement: BaseElement,\n    rotation: number,\n    coachDimension: ElementDimension,\n  ): TemplateResult {\n    const calculatedDimension = this.getCalculatedDimension(\n      graphicalElement.dimension,\n      coachDimension,\n    );\n    const calculatedPosition = this.getCalculatedPosition(\n      graphicalElement.position,\n      graphicalElement.dimension,\n      coachDimension,\n    );\n\n    // If the icon is the driver area, then here concat the vehicle type to get the right vehicle chassie icon\n    const icon =\n      graphicalElement.icon && graphicalElement.icon.indexOf('DRIVER_AREA') === -1\n        ? graphicalElement.icon\n        : graphicalElement.icon?.concat(\n            '_',\n            this.seatReservations[this.currSelectedDeckIndex].vehicleType,\n          );\n\n    return html` <sbb-seat-reservation-graphic\n      style=${styleMap({\n        '--sbb-seat-reservation-graphic-width': calculatedDimension.w,\n        '--sbb-seat-reservation-graphic-height': calculatedDimension.h,\n        '--sbb-seat-reservation-graphic-top': calculatedPosition.y,\n        '--sbb-seat-reservation-graphic-left': calculatedPosition.x,\n        '--sbb-seat-reservation-graphic-position': 'absolute',\n      })}\n      name=${icon ?? nothing}\n      rotation=${rotation}\n      aria-hidden=\"true\"\n      ?stretch=${true}\n    ></sbb-seat-reservation-graphic>`;\n  }\n\n  private _getRenderedServiceElements(\n    coachIndex: number,\n    coachDeckIndex: number,\n    serviceElements?: BaseElement[],\n  ): TemplateResult[] | null {\n    if (!serviceElements) return null;\n\n    return serviceElements?.map((serviceElement: BaseElement) => {\n      const titleDescription = serviceElement.icon\n        ? getI18nSeatReservation(serviceElement.icon, this._language.current)\n        : null;\n      const calculatedDimension = this.getCalculatedDimension(serviceElement.dimension);\n      const calculatedPosition = this.getCalculatedPosition(serviceElement.position);\n      const elementRotation = serviceElement.rotation || 0;\n      const elementFixedRotation = this.alignVertical ? elementRotation - 90 : elementRotation;\n\n      //generate unique index number for the trigger element\n      const triggerId = `popover-trigger-${coachDeckIndex}-${coachIndex}-${calculatedPosition.x}-${calculatedPosition.y}`;\n\n      return html`\n        <sbb-seat-reservation-graphic\n          id=\"${triggerId}\"\n          style=${styleMap({\n            '--sbb-seat-reservation-graphic-width': calculatedDimension.w,\n            '--sbb-seat-reservation-graphic-height': calculatedDimension.h,\n            '--sbb-seat-reservation-graphic-top': calculatedPosition.y,\n            '--sbb-seat-reservation-graphic-left': calculatedPosition.x,\n            '--sbb-seat-reservation-graphic-position': 'absolute',\n          })}\n          class=\"sbb-seat-reservation-graphic--cursor-pointer\"\n          name=${serviceElement.icon ?? nothing}\n          .rotation=${elementFixedRotation}\n          role=\"img\"\n          aria-hidden=\"true\"\n        ></sbb-seat-reservation-graphic>\n        ${this._popover(triggerId, titleDescription)}\n      `;\n    });\n  }\n\n  /**\n   * Manages the selected place event triggered from the place\n   * Each selection emits an array of all selected places\n   * @param selectPlaceEvent\n   */\n  private _onSelectPlace(selectPlaceEvent: CustomEvent<PlaceSelection>): void {\n    const selectedPlace = selectPlaceEvent.detail;\n    // We have to set preventCoachScrollByPlaceClick to true, to prevent automatic scrolling to the new focused place\n    this.preventCoachScrollByPlaceClick = true;\n    this.isCoachGridFocusable = false;\n    if (!this.preventPlaceClick) {\n      // Set current deck index if a places was selected\n      this.currSelectedDeckIndex = selectedPlace.deckIndex;\n      // Add place to place collection\n      this.updateSelectedSeatReservationPlaces(selectedPlace);\n      this.updateCurrentSelectedPlaceInCoach(selectedPlace);\n    }\n  }\n\n  private _onSelectNavCoach(event: CustomEvent<number>): void {\n    const selectedNavCoachIndex = event.detail;\n    this.isKeyboardNavigation = false;\n\n    if (selectedNavCoachIndex !== null && selectedNavCoachIndex !== this.currSelectedCoachIndex) {\n      this.unfocusPlaceElement();\n      this.scrollToSelectedNavCoach(selectedNavCoachIndex);\n    } else if (selectedNavCoachIndex === this.currSelectedCoachIndex) {\n      this.updateCurrentSelectedCoach();\n      this.preselectPlaceInCoach();\n    }\n\n    //close all opened popovers\n    this._closePopover();\n  }\n\n  private _onFocusNavCoach(): void {\n    if (!this.preventCoachScrollByPlaceClick) {\n      this.preselectPlaceInCoach();\n    } else {\n      this.focusPlaceElement(this.currSelectedPlace);\n    }\n    this.isAutoScrolling = false;\n  }\n\n  /**\n   * Creates a popover for extra service information\n   * @param triggerId\n   * @param popoverContent\n   * @private\n   */\n  private _popover(\n    triggerId: string,\n    popoverContent: string | null | typeof nothing,\n  ): TemplateResult {\n    return html`\n      <sbb-popover trigger=\"${triggerId}\">\n        <p class=\"sbb-text-s sbb-sr-popover\">${popoverContent}</p>\n      </sbb-popover>\n    `;\n  }\n\n  /**\n   * trigger to close all opened popovers (normally only one is opened at a time)\n   * @private\n   */\n  private _closePopover(): void {\n    this.shadowRoot\n      ?.querySelectorAll('sbb-popover[data-state=\"opened\"]')\n      .forEach((popover) => popover.setAttribute('data-state', 'closed'));\n  }\n\n  private _getDescriptionTableCoach(coachItem: CoachItem, coachIndex: number): string {\n    if (!coachItem.places?.length) {\n      return getI18nSeatReservation('COACH_BLOCKED_TABLE_CAPTION', this._language.current, [\n        coachItem.id,\n      ]);\n    }\n\n    let tableCoachDescription: string;\n    const areaDescriptions = this._getTitleDescriptionListString(coachItem.graphicElements!);\n    const serviceDescriptions = this._getTitleDescriptionListString(coachItem.serviceElements!);\n\n    tableCoachDescription = getI18nSeatReservation('COACH_TABLE_CAPTION', this._language.current, [\n      coachItem.id,\n    ]);\n\n    if (!this.hasNavigation) {\n      // Expands the number of available seats and bicycle spaces as info\n      const freePlaces = this.getAvailableFreePlacesNumFromCoach(coachIndex);\n      const freePlacesTxt = getI18nSeatReservation(\n        'COACH_AVAILABLE_NUMBER_OF_PLACES',\n        this._language.current,\n        [freePlaces.seats, freePlaces.bicycles],\n      );\n      tableCoachDescription = tableCoachDescription.concat('. ').concat(freePlacesTxt).concat('. ');\n    }\n\n    if (!!areaDescriptions || !!serviceDescriptions) {\n      tableCoachDescription +=\n        '. ' + getI18nSeatReservation('COACH_AVAILABLE_SERVICES', this._language.current) + ': ';\n      tableCoachDescription += serviceDescriptions + ', ' + areaDescriptions + '.';\n    }\n    return tableCoachDescription;\n  }\n\n  private _getTitleDescriptionListString(descriptionsElements: BaseElement[]): string {\n    const uniqueDescriptions: string[] = [];\n\n    return descriptionsElements\n      ?.map((descriptionElement) => {\n        const icon = descriptionElement.icon;\n        if (!icon) return null;\n\n        const descriptionAlreayExist = uniqueDescriptions.indexOf(icon) > -1;\n        const translation = getI18nSeatReservation(\n          descriptionElement.icon!,\n          this._language.current,\n        );\n        const isValidDescription =\n          this._notFixedRotatableAreaIcons.indexOf(icon) === -1 &&\n          this._notAreaElements.indexOf(icon) === -1;\n\n        if (!descriptionAlreayExist) {\n          uniqueDescriptions.push(descriptionElement.icon!);\n        }\n        return !!translation && !descriptionAlreayExist && isValidDescription ? translation : null;\n      })\n      .filter((description) => !!description)\n      .join(', ');\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-seat-reservation': SbbSeatReservationElement;\n  }\n}\n"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAmCM,6BAAyB,MAAA;;0BAD9B,cAAc,sBAAsB,CAAC;;;;oBACE;AAAR,EAAA,mBAAQ,YAA0B;AAAA;;AAGxD,WAAA,YAAY,IAAI,sBAAsB,IAAI;AAG1C,WAAA,mBAAmB;AAAA,QACzB;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MAAA;AAIM,WAAA,8BAA8B,CAAC,YAAY;AAAA,IAyqBrD;AAAA,IAvqBqB,WAAW,mBAAuC;AACnE,YAAM,WAAW,iBAAiB;AAElC,UAAI,kBAAkB,IAAI,eAAe,GAAG;AAC1C,YAAI,KAAK,eAAe;AACtB,eAAK,YACD,iBAAiB,OAAO,EACzB,QAAQ,CAAC,UAAU,MAAM,gBAAgB,UAAU,CAAC;AAAA,QACzD,OAAO;AACL,eAAK,YACD,iBAAiB,OAAO,EACzB,QAAQ,CAAC,UAAU,MAAM,aAAa,YAAY,GAAG,CAAC;AAAA,QAC3D;AAAA,MACF;AAAA,IACF;AAAA,IAEmB,aAAa,mBAAuC;AACrE,YAAM,aAAa,iBAAiB;AAGpC,WAAK,eAAe,KAAK,MAAK;AAC5B,aAAK,+BAAA;AACL,aAAK,qCAAA;AAIL,aAAK,cAAA;AAAA,MACP,CAAC;AAAA,IACH;AAAA,IAEmB,SAAM;AACvB,UAAI,CAAC,UAAU;AACb,aAAK,uBAAA;AAAA,MACP;AAEA,WAAK,wCAAA;AACL,aAAO,KAAK,wBAAwB;AAAA,IACtC;AAAA,IAEQ,yBAAsB;AAC5B,YAAM,eAAe,SAAS,OAAO,iBAAiB,SAAS,IAAI,EAAE,UAAU,EAAE;AAEjF,YAAM,gBAAgB,IAAI;AAC1B,WAAK,OAAO,YAAY,qCAAqC,GAAG,gBAAgB,KAAK,EAAE;AAAA,IACzF;AAAA,IAEQ,0CAAuC;AAC7C,YAAM,qBAAqB,KAAK,gBAAgB,2BAA2B;AAC3E,WAAK,uBAAuB;AAAA;AAAA,0CAEU,kBAAkB;AAAA;AAAA,cAE9C,KAAK,+BAA+B,gBAAgB,CAAC;AAAA;AAAA;AAAA;AAAA,uBAI5C,CAAC,QAAuB,KAAK,oBAAoB,GAAG,CAAC;AAAA;AAAA;AAAA,sCAGtC,KAAK,mBAAmB;AAAA;AAAA;AAAA,6DAGD,KAAK,mBAAmB;AAAA;AAAA;AAAA;AAAA,iCAIpD,SAAS;AAAA,QAChB,sBAAsB;AAAA,QACtB,2BAA2B,KAAK;AAAA,MAAA,CACjC,CAAC;AAAA;AAAA,0BAEA,KAAK,kBAAkB,IACvB,CAAC,iBAAkC,UAAiB;AAClD,eAAO;AAAA;AAAA,kCAED,KAAK,eAAe,iBAAiB,KAAK,CAAC;AAAA;AAAA;AAAA,MAGnD,CAAC,CACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,cASX,KAAK,+BAA+B,iBAAiB,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA,IAKlE;AAAA,IAEQ,oBAAiB;AACvB,UAAI,CAAC,KAAK;AAAkB,eAAO;AAEnC,aAAO,KAAK,iBAAiB,IAAI,CAAC,oBAAmB;AACnD,cAAM,kBAAkB,uBACtB,gBAAgB,gBAChB,KAAK,UAAU,OAAO;AAExB,eAAO,6BAA6B,eAAe;AAAA,MACrD,CAAC;AAAA,IACH;AAAA,IAEQ,+BAA+B,cAAoB;AACzD,UAAI,CAAC,KAAK,iBAAiB,CAAC,KAAK;AAAkB,eAAO;AAE1D,YAAM,QAAQ,gBAAgB,oBAAoB,qBAAqB;AACvE,YAAM,UAAU,gBAAgB,oBAAoB,sBAAsB;AAC1E,YAAM,qBACJ,gBAAgB,oBACZ,uBAAuB,wBAAwB,KAAK,UAAU,OAAO,IACrE,uBAAuB,0BAA0B,KAAK,UAAU,OAAO;AAC7E,UAAI,cAAc;AAElB,UAAI,gBAAgB,oBAAoB,KAAK,qBAAqB,GAAG;AACnE,sBAAc;AAAA,MAChB,WACE,gBAAgB,qBAChB,KAAK,qBACH,KAAK,iBAAiB,KAAK,qBAAqB,EAAE,WAAW,SAAS,GACxE;AACA,sBAAc;AAAA,MAChB;AAEA,aAAO;AAAA,gBACK,MAAM,KAAK,uBAAuB,YAAY,CAAC;AAAA,YACnD,KAAK;AAAA;AAAA;AAAA,mBAGE,OAAO;AAAA;AAAA,oBAEN,kBAAkB;AAAA;AAAA,8BAER,eAAe,OAAO;AAAA;AAAA,IAElD;AAAA,IAEQ,oBAAiB;AACvB,UAAI,CAAC,KAAK,iBAAiB,CAAC,KAAK;AAAkB,eAAO;AAE1D,aAAO;AAAA;AAAA;AAAA;AAAA;AAAA,wBAKa,uBACZ,+BACA,KAAK,UAAU,OAAO,CACvB;AAAA;AAAA,YAEC,KAAK,iBAAiB,KAAK,qBAAqB,GAAG,WAAW,IAC9D,CAAC,WAAsB,UAAiB;AACtC,cAAM,kBAAkB,KAAK,mCAAmC,KAAK;AACrE,eAAO;AAAA;AAAA,iCAEY,CAAC,UAA+B,KAAK,kBAAkB,KAAK,CAAC;AAAA,gCAC9D,MAAM,KAAK,iBAAA,CAAkB;AAAA,2BAClC,SAAS;AAAA,UAChB,0CACE,KAAK,4BAA4B;AAAA,QAAA,CACpC,CAAC;AAAA,2BACO,KAAK;AAAA,8BACF,UAAU,EAAE;AAAA,uCACH,eAAe;AAAA,8BACxB,KAAK,uBAAuB,KAAK;AAAA,6BAClC,KAAK,sBAAsB,KAAK;AAAA,kCAC3B,UAAU,WAAW;AAAA,kCACrB,UAAU,WAAW;AAAA,kCACrB,CAAC,UAAU,QAAQ,MAAM;AAAA,4BAC/B,WAAW,gBAAgB,IAAI;AAAA,2BAChC,WAAW,gBAAgB,KAAK;AAAA,+BAC5B,KAAK,aAAa;AAAA;AAAA;AAAA;AAAA,MAIrC,CAAC,CACF;AAAA;AAAA;AAAA;AAAA,IAIT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,eACN,iBACA,WAAiB;AAEjB,YAAM,UAAU,KAAK,MAAM,KAAK,UAAU,iBAAiB,UAAU,CAAC;AAEtE,UAAI,CAAC,SAAS;AACZ,eAAO;AAAA,MACT;AACA,aAAO,QAAQ,IAAI,CAAC,WAAsB,eAAsB;AAC9D,eAAO;AAAA;AAAA,YAED,KAAK,oBAAoB,WAAW,YAAY,SAAS,CAAC;AAAA;AAAA;AAAA,MAGlE,CAAC;AAAA,IACH;AAAA,IAEQ,oBACN,WACA,YACA,gBAAsB;AAEtB,YAAM,2BAA2B,KAAK,uBAAuB,UAAU,SAAS;AAChF,YAAM,oCAAoC,KAAK,0BAA0B,WAAW,UAAU;AAE9F,aAAO;AAAA,cACG,SAAS;AAAA,QACf,uCAAuC,yBAAyB;AAAA,QAChE,wCAAwC,yBAAyB;AAAA,MAAA,CAClE,CAAC;AAAA;AAAA,QAEA,KAAK,yBAAyB,SAAS,CAAC;AAAA,QACxC,KAAK,8BACL,UAAU,mBAAmB,CAAA,GAC7B,UAAU,WACV,YACA,cAAc,CACf;AAAA,QACC,KAAK,4BAA4B,YAAY,gBAAgB,UAAU,eAAe,CAAC;AAAA;AAAA;AAAA,iBAG9E,MAAM,KAAK,mCAAmC,UAAU,CAAC;AAAA,2BAC/C,UAAU;AAAA;AAAA,iDAEY,UAAU;AAAA;AAAA,4CAEf,UAAU;AAAA,oCAClB,iCAAiC;AAAA;AAAA,UAE3D,KAAK,sBAAsB,WAAW,YAAY,cAAc,CAAC;AAAA;AAAA;AAAA,IAGzE;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,yBAAyB,WAAoB;AACnD,YAAM,sBAAsB;AAC5B,YAAM,cAAc,UAAU;AAC9B,YAAM,aAAa,aAAa,KAAK,CAAC,YAAyB,QAAQ,SAAS,aAAa;AAC7F,YAAM,cAAc,aAChB,UAAU,UAAU,IAAI,WAAW,UAAU,IAAI,sBACjD,UAAU,UAAU,IAAI,sBAAsB;AAClD,YAAM,gBAAgB,UAAU,UAAU,IAAI,KAAK,oBAAoB,KAAK,KAAK;AACjF,YAAM,gBACJ,cAAc,WAAW,SAAS,MAAM,IACpC,YAAY,UAAU,IAAI,KAAK,eAC/B,KAAK;AACX,aAAO;AAAA;AAAA,gBAEK,SAAS;AAAA,QACf,wCAAwC,cAAc,KAAK;AAAA,QAC3D,yCAAyC;AAAA,QACzC,sCAAsC,KAAK,qBAAqB;AAAA,QAChE,uCAAuC;AAAA,QACvC,2CAA2C;AAAA,MAAA,CAC5C,CAAC;AAAA;AAAA,mBAES,IAAI;AAAA;AAAA;AAAA;AAAA,IAIrB;AAAA,IAEQ,sBACN,OACA,YACA,WAAiB;AAEjB,UAAI,CAAC,MAAM,QAAQ;AACjB,eAAO;AAAA,MACT;AAGA,YAAM,iBAA0C,CAAA;AAChD,iBAAW,SAAS,MAAM,QAAQ;AAChC,YAAI,CAAC,eAAe,MAAM,SAAS,CAAC,GAAG;AACrC,yBAAe,MAAM,SAAS,CAAC,IAAI,CAAC,KAAK;AAAA,QAC3C,OAAO;AACL,yBAAe,MAAM,SAAS,CAAC,EAAE,KAAK,KAAK;AAAA,QAC7C;AAAA,MACF;AAEA,aAAO,OAAO,OAAO,cAAc,EAChC,IAAI,CAAC,WAAoB,UAAS;AACjC,eAAO;AAAA,wBACS,UAAU,IAAI,UAAU,CAAC,EAAE,SAAS,CAAC,oBAAoB,KAAK;AAAA,cACxE,KAAK,yBAAyB,WAAW,YAAY,SAAS,CAAC;AAAA;AAAA;AAAA,MAGvE,CAAC,EACA,QAAQ,CAAC,gBAAgB,WAAW;AAAA,IACzC;AAAA,IAEQ,yBACN,QACA,YACA,WAAiB;AAGjB,aAAO,KACL,CAAC,QAAe,WAAkB,OAAO,OAAO,SAAS,CAAC,IAAI,OAAO,OAAO,SAAS,CAAC,CAAC;AAGzF,aAAO,QAAQ,IAAI,CAAC,OAAc,UAAiB;AACjD,cAAM,sBAAsB,KAAK,uBAAuB,MAAM,SAAS;AACvE,cAAM,qBAAqB,KAAK,sBAAsB,MAAM,QAAQ;AACpE,cAAM,eAAe,KAAK,gBAAgB,MAAM;AAEhD,eAAO;AAAA;AAAA,qBAEQ,SAAS,IAAI,UAAU,IAAI,MAAM,SAAS,CAAC,IAAI,KAAK;AAAA;AAAA;AAAA;AAAA,oBAIrD,SAAS;AAAA,UACf,8CAA8C,oBAAoB;AAAA,UAClE,+CAA+C,oBAAoB;AAAA,UACnE,4CAA4C,mBAAmB;AAAA,UAC/D,6CAA6C,mBAAmB;AAAA,QAAA,CACjE,CAAC;AAAA,2BACa,CAAC,qBACd,KAAK,eAAe,gBAAgB,CAAC;AAAA;AAAA,iDAEF,SAAS,IAAI,UAAU,IAAI,MAAM,MAAM;AAAA;AAAA,4BAE5D,UAAU,IAAI,MAAM,SAAS,CAAC,IAAI,KAAK;AAAA,mBAChD,MAAM,MAAM;AAAA,mBACZ,MAAM,IAAI;AAAA,oBACT,MAAM,KAAK;AAAA,oBACX,MAAM,UAAU,IAAI,KAAK,YAAY;AAAA,qBACpC,MAAM,UAAU,IAAI,KAAK,YAAY;AAAA,uBACnC,MAAM,YAAY,OAAO;AAAA,4BACpB,YAAY;AAAA,0BACd,UAAU;AAAA,yBACX,SAAS;AAAA,2BACP,MAAM,WAAW;AAAA,4BAChB,KAAK,iBAAiB;AAAA;AAAA;AAAA;AAAA,MAI9C,CAAC;AAAA,IACH;AAAA,IAEQ,8BACN,mBACA,gBACA,YACA,gBAAsB;AAEtB,UAAI,CAAC;AAAmB,eAAO;AAE/B,aAAO,mBAAmB,IAAI,CAAC,qBAAiC;AAC9D,cAAM,OAAO,iBAAiB,QAAQ;AACtC,cAAM,kBAAkB,iBAAiB,YAAY;AACrD,cAAM,qCACJ,KAAK,4BAA4B,QAAQ,iBAAiB,IAAK,MAAM;AACvE,cAAM,uBACJ,KAAK,iBAAiB,qCAClB,kBAAkB,KAClB;AAGN,YAAI,KAAK,iBAAiB,UAAU,CAAC,mBAAmB,mBAAmB,IAAI,IAAI,IAAI;AACrF,iBAAO,KAAK,6BAA6B,kBAAkB,iBAAiB,cAAc;AAAA,QAC5F;AACA,eAAO,KAAK,0BACV,kBACA,sBACA,gBACA,YACA,cAAc;AAAA,MAElB,CAAC;AAAA,IACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAUQ,0BACN,kBACA,UACA,gBACA,YACA,gBAAsB;AAKtB,YAAM,oBAAoB,iBAAiB,MAAM,QAAQ,OAAO,MAAM;AACtE,YAAM,eAAe,iBAAiB,QAAQ,oBAAoB,iBAAiB,OAAO;AAC1F,YAAM,gBAAgB,iBAAiB;AACvC,YAAM,mBAAmB,iBAAiB,OACtC,uBAAuB,iBAAiB,MAAM,KAAK,UAAU,OAAO,IACpE;AACJ,YAAM,sBAAsB,KAAK,uBAC/B,iBAAiB,WACjB,gBACA,MACA,aAAa;AAEf,YAAM,qBAAqB,KAAK,sBAC9B,iBAAiB,UACjB,iBAAiB,WACjB,gBACA,IAAI;AAIN,YAAM,YAAY,mBAAmB,cAAc,IAAI,UAAU,IAAI,mBAAmB,CAAC,IAAI,mBAAmB,CAAC;AACjH,UAAI,kBAAkB;AAEtB,UAAI,iBAAiB,SAAS,MAAM,KAAK,oBAAoB,IAAI;AAC/D,0BAAkB;AAAA,MACpB,WACE,iBAAiB,SAAS,IAAI,iBAAiB,UAAU,MACzD,eAAe,IAAI,KAAK,mBACxB;AACA,0BAAkB;AAAA,MACpB;AAEA,aAAO;AAAA;AAAA,cAEG,SAAS;AAAA,iBACN,SAAS,EAAE,6CAA6C,iBAAiB,KAAA,CAAM,CAAC;AAAA,gBACjF,SAAS;AAAA,QACf,qCAAqC,oBAAoB;AAAA,QACzD,sCAAsC,oBAAoB;AAAA,QAC1D,mCAAmC,mBAAmB;AAAA,QACtD,oCAAoC,mBAAmB;AAAA,MAAA,CACxD,CAAC;AAAA,mBACS,eAAe;AAAA;AAAA;AAAA;AAAA,UAIxB,eACE;AAAA;AAAA,uBAEW,YAAY;AAAA,wBACX,KAAK,YAAY;AAAA,yBAChB,KAAK,YAAY;AAAA,2BACf,QAAQ;AAAA;AAAA;AAAA;AAAA,gBAKvB,OAAO;AAAA;AAAA,QAEX,eAAe,KAAK,SAAS,WAAW,gBAAgB,IAAI,OAAO;AAAA;AAAA,IAEzE;AAAA,IAEQ,6BACN,kBACA,UACA,gBAAgC;AAEhC,YAAM,sBAAsB,KAAK,uBAC/B,iBAAiB,WACjB,cAAc;AAEhB,YAAM,qBAAqB,KAAK,sBAC9B,iBAAiB,UACjB,iBAAiB,WACjB,cAAc;AAIhB,YAAM,OACJ,iBAAiB,QAAQ,iBAAiB,KAAK,QAAQ,aAAa,MAAM,KACtE,iBAAiB,OACjB,iBAAiB,MAAM,OACrB,KACA,KAAK,iBAAiB,KAAK,qBAAqB,EAAE,WAAW;AAGrE,aAAO;AAAA,cACG,SAAS;AAAA,QACf,wCAAwC,oBAAoB;AAAA,QAC5D,yCAAyC,oBAAoB;AAAA,QAC7D,sCAAsC,mBAAmB;AAAA,QACzD,uCAAuC,mBAAmB;AAAA,QAC1D,2CAA2C;AAAA,MAAA,CAC5C,CAAC;AAAA,aACK,QAAQ,OAAO;AAAA,iBACX,QAAQ;AAAA;AAAA,iBAER,IAAI;AAAA;AAAA,IAEnB;AAAA,IAEQ,4BACN,YACA,gBACA,iBAA+B;AAE/B,UAAI,CAAC;AAAiB,eAAO;AAE7B,aAAO,iBAAiB,IAAI,CAAC,mBAA+B;AAC1D,cAAM,mBAAmB,eAAe,OACpC,uBAAuB,eAAe,MAAM,KAAK,UAAU,OAAO,IAClE;AACJ,cAAM,sBAAsB,KAAK,uBAAuB,eAAe,SAAS;AAChF,cAAM,qBAAqB,KAAK,sBAAsB,eAAe,QAAQ;AAC7E,cAAM,kBAAkB,eAAe,YAAY;AACnD,cAAM,uBAAuB,KAAK,gBAAgB,kBAAkB,KAAK;AAGzE,cAAM,YAAY,mBAAmB,cAAc,IAAI,UAAU,IAAI,mBAAmB,CAAC,IAAI,mBAAmB,CAAC;AAEjH,eAAO;AAAA;AAAA,gBAEG,SAAS;AAAA,kBACP,SAAS;AAAA,UACf,wCAAwC,oBAAoB;AAAA,UAC5D,yCAAyC,oBAAoB;AAAA,UAC7D,sCAAsC,mBAAmB;AAAA,UACzD,uCAAuC,mBAAmB;AAAA,UAC1D,2CAA2C;AAAA,QAAA,CAC5C,CAAC;AAAA;AAAA,iBAEK,eAAe,QAAQ,OAAO;AAAA,sBACzB,oBAAoB;AAAA;AAAA;AAAA;AAAA,UAIhC,KAAK,SAAS,WAAW,gBAAgB,CAAC;AAAA;AAAA,MAEhD,CAAC;AAAA,IACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAOQ,eAAe,kBAA6C;AAClE,YAAM,gBAAgB,iBAAiB;AAEvC,WAAK,iCAAiC;AACtC,WAAK,uBAAuB;AAC5B,UAAI,CAAC,KAAK,mBAAmB;AAE3B,aAAK,wBAAwB,cAAc;AAE3C,aAAK,oCAAoC,aAAa;AACtD,aAAK,kCAAkC,aAAa;AAAA,MACtD;AAAA,IACF;AAAA,IAEQ,kBAAkB,OAA0B;AAClD,YAAM,wBAAwB,MAAM;AACpC,WAAK,uBAAuB;AAE5B,UAAI,0BAA0B,QAAQ,0BAA0B,KAAK,wBAAwB;AAC3F,aAAK,oBAAA;AACL,aAAK,yBAAyB,qBAAqB;AAAA,MACrD,WAAW,0BAA0B,KAAK,wBAAwB;AAChE,aAAK,2BAAA;AACL,aAAK,sBAAA;AAAA,MACP;AAGA,WAAK,cAAA;AAAA,IACP;AAAA,IAEQ,mBAAgB;AACtB,UAAI,CAAC,KAAK,gCAAgC;AACxC,aAAK,sBAAA;AAAA,MACP,OAAO;AACL,aAAK,kBAAkB,KAAK,iBAAiB;AAAA,MAC/C;AACA,WAAK,kBAAkB;AAAA,IACzB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAQQ,SACN,WACA,gBAA8C;AAE9C,aAAO;AAAA,8BACmB,SAAS;AAAA,+CACQ,cAAc;AAAA;AAAA;AAAA,IAG3D;AAAA;AAAA;AAAA;AAAA;AAAA,IAMQ,gBAAa;AACnB,WAAK,YACD,iBAAiB,kCAAkC,EACpD,QAAQ,CAAC,YAAY,QAAQ,aAAa,cAAc,QAAQ,CAAC;AAAA,IACtE;AAAA,IAEQ,0BAA0B,WAAsB,YAAkB;AACxE,UAAI,CAAC,UAAU,QAAQ,QAAQ;AAC7B,eAAO,uBAAuB,+BAA+B,KAAK,UAAU,SAAS;AAAA,UACnF,UAAU;AAAA,QAAA,CACX;AAAA,MACH;AAEA,UAAI;AACJ,YAAM,mBAAmB,KAAK,+BAA+B,UAAU,eAAgB;AACvF,YAAM,sBAAsB,KAAK,+BAA+B,UAAU,eAAgB;AAE1F,8BAAwB,uBAAuB,uBAAuB,KAAK,UAAU,SAAS;AAAA,QAC5F,UAAU;AAAA,MAAA,CACX;AAED,UAAI,CAAC,KAAK,eAAe;AAEvB,cAAM,aAAa,KAAK,mCAAmC,UAAU;AACrE,cAAM,gBAAgB,uBACpB,oCACA,KAAK,UAAU,SACf,CAAC,WAAW,OAAO,WAAW,QAAQ,CAAC;AAEzC,gCAAwB,sBAAsB,OAAO,IAAI,EAAE,OAAO,aAAa,EAAE,OAAO,IAAI;AAAA,MAC9F;AAEA,UAAI,CAAC,CAAC,oBAAoB,CAAC,CAAC,qBAAqB;AAC/C,iCACE,OAAO,uBAAuB,4BAA4B,KAAK,UAAU,OAAO,IAAI;AACtF,iCAAyB,sBAAsB,OAAO,mBAAmB;AAAA,MAC3E;AACA,aAAO;AAAA,IACT;AAAA,IAEQ,+BAA+B,sBAAmC;AACxE,YAAM,qBAA+B,CAAA;AAErC,aAAO,sBACH,IAAI,CAAC,uBAAsB;AAC3B,cAAM,OAAO,mBAAmB;AAChC,YAAI,CAAC;AAAM,iBAAO;AAElB,cAAM,yBAAyB,mBAAmB,QAAQ,IAAI,IAAI;AAClE,cAAM,cAAc,uBAClB,mBAAmB,MACnB,KAAK,UAAU,OAAO;AAExB,cAAM,qBACJ,KAAK,4BAA4B,QAAQ,IAAI,MAAM,MACnD,KAAK,iBAAiB,QAAQ,IAAI,MAAM;AAE1C,YAAI,CAAC,wBAAwB;AAC3B,6BAAmB,KAAK,mBAAmB,IAAK;AAAA,QAClD;AACA,eAAO,CAAC,CAAC,eAAe,CAAC,0BAA0B,qBAAqB,cAAc;AAAA,MACxF,CAAC,EACA,OAAO,CAAC,gBAAgB,CAAC,CAAC,WAAW,EACrC,KAAK,IAAI;AAAA,IACd;AAAA;;AAzrBF,iBAAA,MAAA,mBAAA,EAAA,OAAA,WAAA,GAAA,kBAAA,EAAA,MAAA,SAAA,MAAA,WAAA,MAAA,UAAA,UAAA,GAAA,MAAA,uBAAA;;;QACyB,GAAA,SAAyB,OAD5C,kBAAA,YAAA,uBAAA,GAA0B;;;"}
|