@qite/tide-booking-component 1.2.3 → 1.2.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (137) hide show
  1. package/.vs/ProjectSettings.json +3 -3
  2. package/.vs/VSWorkspaceState.json +5 -5
  3. package/README.md +8 -8
  4. package/build/build-cjs/booking-product/components/footer.d.ts +1 -0
  5. package/build/build-cjs/booking-product/components/rooms.d.ts +2 -0
  6. package/build/build-cjs/booking-wizard/features/booking/selectors.d.ts +1 -1
  7. package/build/build-cjs/booking-wizard/features/flight-options/flight-option-flight.d.ts +1 -1
  8. package/build/build-cjs/booking-wizard/features/flight-options/flight-option.d.ts +1 -1
  9. package/build/build-cjs/booking-wizard/types.d.ts +5 -0
  10. package/build/build-cjs/index.js +1453 -735
  11. package/build/build-cjs/shared/utils/localization-util.d.ts +4 -0
  12. package/build/build-esm/booking-product/components/footer.d.ts +1 -0
  13. package/build/build-esm/booking-product/components/rooms.d.ts +2 -0
  14. package/build/build-esm/booking-wizard/features/booking/selectors.d.ts +1 -1
  15. package/build/build-esm/booking-wizard/features/flight-options/flight-option-flight.d.ts +1 -1
  16. package/build/build-esm/booking-wizard/features/flight-options/flight-option.d.ts +1 -1
  17. package/build/build-esm/booking-wizard/types.d.ts +5 -0
  18. package/build/build-esm/index.js +1455 -737
  19. package/build/build-esm/shared/utils/localization-util.d.ts +4 -0
  20. package/package.json +75 -77
  21. package/src/booking-product/components/age-select.tsx +35 -41
  22. package/src/booking-product/components/amount-input.tsx +78 -64
  23. package/src/booking-product/components/date-range-picker/calendar-day.tsx +58 -54
  24. package/src/booking-product/components/date-range-picker/calendar.tsx +178 -176
  25. package/src/booking-product/components/date-range-picker/index.tsx +196 -181
  26. package/src/booking-product/components/dates.tsx +136 -132
  27. package/src/booking-product/components/footer.tsx +69 -64
  28. package/src/booking-product/components/header.tsx +79 -68
  29. package/src/booking-product/components/icon.tsx +251 -208
  30. package/src/booking-product/components/product.tsx +314 -277
  31. package/src/booking-product/components/rooms.tsx +195 -177
  32. package/src/booking-product/index.tsx +30 -30
  33. package/src/booking-product/settings-context.ts +14 -14
  34. package/src/booking-product/types.ts +28 -28
  35. package/src/booking-product/utils/api.ts +25 -25
  36. package/src/booking-product/utils/price.ts +29 -27
  37. package/src/booking-wizard/api-settings-slice.ts +24 -24
  38. package/src/booking-wizard/components/icon.tsx +508 -60
  39. package/src/booking-wizard/components/message.tsx +34 -34
  40. package/src/booking-wizard/components/multi-range-filter.tsx +113 -113
  41. package/src/booking-wizard/components/product-card.tsx +37 -37
  42. package/src/booking-wizard/components/step-indicator.tsx +51 -51
  43. package/src/booking-wizard/components/step-route.tsx +27 -27
  44. package/src/booking-wizard/declarations.d.ts +4 -4
  45. package/src/booking-wizard/features/booking/api.ts +49 -45
  46. package/src/booking-wizard/features/booking/booking-self-contained.tsx +384 -357
  47. package/src/booking-wizard/features/booking/booking-slice.ts +662 -603
  48. package/src/booking-wizard/features/booking/booking.tsx +356 -349
  49. package/src/booking-wizard/features/booking/constants.ts +16 -16
  50. package/src/booking-wizard/features/booking/selectors.ts +441 -408
  51. package/src/booking-wizard/features/confirmation/confirmation.tsx +97 -94
  52. package/src/booking-wizard/features/error/error.tsx +78 -75
  53. package/src/booking-wizard/features/flight-options/flight-filter.tsx +432 -344
  54. package/src/booking-wizard/features/flight-options/flight-option-flight.tsx +385 -351
  55. package/src/booking-wizard/features/flight-options/flight-option-modal.tsx +229 -214
  56. package/src/booking-wizard/features/flight-options/flight-option.tsx +81 -66
  57. package/src/booking-wizard/features/flight-options/flight-utils.ts +516 -401
  58. package/src/booking-wizard/features/flight-options/index.tsx +196 -177
  59. package/src/booking-wizard/features/price-details/price-details-api.ts +24 -24
  60. package/src/booking-wizard/features/price-details/price-details-slice.ts +178 -171
  61. package/src/booking-wizard/features/price-details/util.ts +155 -155
  62. package/src/booking-wizard/features/product-options/no-options.tsx +21 -21
  63. package/src/booking-wizard/features/product-options/none-option.tsx +120 -120
  64. package/src/booking-wizard/features/product-options/option-booking-airline-group.tsx +64 -66
  65. package/src/booking-wizard/features/product-options/option-booking-group.tsx +216 -210
  66. package/src/booking-wizard/features/product-options/option-item.tsx +317 -318
  67. package/src/booking-wizard/features/product-options/option-pax-card.tsx +201 -117
  68. package/src/booking-wizard/features/product-options/option-pax-group.tsx +175 -169
  69. package/src/booking-wizard/features/product-options/option-room.tsx +321 -314
  70. package/src/booking-wizard/features/product-options/option-unit-group.tsx +198 -192
  71. package/src/booking-wizard/features/product-options/option-units-card.tsx +185 -102
  72. package/src/booking-wizard/features/product-options/options-form.tsx +459 -437
  73. package/src/booking-wizard/features/room-options/index.tsx +187 -172
  74. package/src/booking-wizard/features/room-options/room-utils.ts +190 -143
  75. package/src/booking-wizard/features/room-options/room.tsx +160 -124
  76. package/src/booking-wizard/features/room-options/traveler-rooms.tsx +75 -63
  77. package/src/booking-wizard/features/sidebar/index.tsx +76 -76
  78. package/src/booking-wizard/features/sidebar/sidebar-flight.tsx +68 -68
  79. package/src/booking-wizard/features/sidebar/sidebar-util.ts +177 -177
  80. package/src/booking-wizard/features/sidebar/sidebar.tsx +364 -346
  81. package/src/booking-wizard/features/summary/summary-booking-option-pax.tsx +25 -25
  82. package/src/booking-wizard/features/summary/summary-booking-option-unit.tsx +25 -25
  83. package/src/booking-wizard/features/summary/summary-flight.tsx +39 -39
  84. package/src/booking-wizard/features/summary/summary-per-booking-option-group.tsx +69 -57
  85. package/src/booking-wizard/features/summary/summary-per-pax-option-group.tsx +63 -51
  86. package/src/booking-wizard/features/summary/summary-per-unit-option-group.tsx +66 -54
  87. package/src/booking-wizard/features/summary/summary-slice.ts +28 -28
  88. package/src/booking-wizard/features/summary/summary.tsx +674 -641
  89. package/src/booking-wizard/features/travelers-form/travelers-form-slice.ts +164 -164
  90. package/src/booking-wizard/features/travelers-form/travelers-form.tsx +754 -755
  91. package/src/booking-wizard/features/travelers-form/type-ahead-input.tsx +101 -101
  92. package/src/booking-wizard/features/travelers-form/validate-form.ts +245 -245
  93. package/src/booking-wizard/index.tsx +36 -36
  94. package/src/booking-wizard/settings-context.ts +60 -60
  95. package/src/booking-wizard/store.ts +31 -31
  96. package/src/booking-wizard/types.ts +276 -271
  97. package/src/index.ts +4 -5
  98. package/src/shared/components/loader.tsx +16 -16
  99. package/src/shared/translations/en-GB.json +232 -0
  100. package/src/shared/translations/fr-BE.json +233 -233
  101. package/src/shared/translations/nl-BE.json +232 -232
  102. package/src/shared/types.ts +4 -4
  103. package/src/shared/utils/localization-util.ts +62 -56
  104. package/src/shared/utils/query-string-util.ts +119 -116
  105. package/src/shared/utils/tide-api-utils.ts +36 -36
  106. package/styles/booking-product-variables.scss +394 -288
  107. package/styles/booking-product.scss +446 -440
  108. package/styles/booking-wizard-variables.scss +871 -0
  109. package/styles/booking-wizard.scss +59 -4
  110. package/styles/components/_animations.scss +39 -39
  111. package/styles/components/_base.scss +107 -106
  112. package/styles/components/_booking.scss +879 -0
  113. package/styles/components/_button.scss +238 -185
  114. package/styles/components/_checkbox.scss +219 -0
  115. package/styles/components/_cta.scss +208 -67
  116. package/styles/components/_date-list.scss +41 -0
  117. package/styles/components/_date-range-picker.scss +225 -224
  118. package/styles/components/_decrement-increment.scss +35 -37
  119. package/styles/components/_dropdown.scss +72 -74
  120. package/styles/components/_flight-option.scss +1429 -0
  121. package/styles/components/_form.scss +1583 -136
  122. package/styles/components/_info-message.scss +71 -0
  123. package/styles/components/_input.scss +25 -0
  124. package/styles/components/_list.scss +187 -0
  125. package/styles/components/_loader.scss +72 -71
  126. package/styles/components/_mixins.scss +550 -518
  127. package/styles/components/_placeholders.scss +166 -166
  128. package/styles/components/_pricing-summary.scss +155 -0
  129. package/styles/components/_qsm.scss +17 -20
  130. package/styles/components/_radiobutton.scss +170 -0
  131. package/styles/components/_select-wrapper.scss +80 -0
  132. package/styles/components/_spinner.scss +29 -0
  133. package/styles/components/_step-indicators.scss +168 -0
  134. package/styles/components/_table.scss +81 -0
  135. package/styles/components/_tree.scss +530 -0
  136. package/styles/components/_typeahead.scss +281 -0
  137. package/styles/components/_variables.scss +89 -89
@@ -1,603 +1,662 @@
1
- import { PayloadAction, createAsyncThunk, createSlice } from "@reduxjs/toolkit";
2
- import {
3
- BookingAttributes,
4
- BookingOptions,
5
- GroupedFlights,
6
- ProductAttributes,
7
- } from "../../types";
8
-
9
- import {
10
- AirlineBookingPackageOption,
11
- AirportBookingPackageOption,
12
- BookingAirlineGroup,
13
- BookingAirportGroup,
14
- BookingOptionGroup,
15
- BookingOptionPax,
16
- BookingOptionUnit,
17
- BookingPackage,
18
- BookingPackageDetailsRequest,
19
- BookingPackageFlight,
20
- BookingPackagePax,
21
- BookingPackageRequest,
22
- BookingPackageRequestRoom,
23
- BookingPackageRoom,
24
- BookingTravelAgent,
25
- GenerateBookingAccommodationRequest,
26
- PerBookingPackageOption,
27
- } from "@qite/tide-client/build/types";
28
- import { first, isEmpty, isNil, range } from "lodash";
29
- import { RootState } from "../../store";
30
- import { selectAgentId } from "../travelers-form/travelers-form-slice";
31
- import packageApi from "./api";
32
- import { BookingStep, OPTIONS_FORM_STEP } from "./constants";
33
- import {
34
- selectAccommodationCodes,
35
- selectBookingAttributes,
36
- selectBookingRooms,
37
- selectLanguageCode,
38
- selectOfficeId,
39
- selectProductAttributes,
40
- selectProductCode,
41
- } from "./selectors";
42
-
43
- export interface BookingState {
44
- officeId: number;
45
- languageCode: string;
46
- productAttributes?: ProductAttributes;
47
- bookingAttributes?: BookingAttributes;
48
- calculateDeposit: boolean;
49
- bookingNumber?: string;
50
- isRetry: boolean;
51
- package?: BookingPackage;
52
- agents?: BookingTravelAgent[];
53
- isBusy: boolean;
54
- skipPaymentWithAgent: boolean;
55
- generatePaymentUrl: boolean;
56
- isUnavailable?: boolean;
57
- tagIds?: number[];
58
- agentAdressId?: number;
59
- remarks?: string;
60
- voucherCodes?: string[];
61
- bookingOptions: BookingOptions;
62
- bookingType: string;
63
- currentStep: BookingStep;
64
- translations?: {
65
- language: string;
66
- value: any;
67
- }[];
68
- accommodationViewId?: number;
69
- accommodationViews?: { [key: string]: string };
70
- }
71
-
72
- const initialState: BookingState = {
73
- officeId: 1,
74
- languageCode: "nl-BE",
75
- bookingOptions: {
76
- b2b: {
77
- tagIds: [],
78
- entryStatus: 2,
79
- customEntryStatusId: undefined,
80
- },
81
- b2b2c: {
82
- tagIds: [],
83
- entryStatus: 2,
84
- customEntryStatusId: undefined,
85
- },
86
- b2c: {
87
- tagIds: [],
88
- entryStatus: 0,
89
- customEntryStatusId: undefined,
90
- },
91
- },
92
- bookingType: "b2c",
93
- productAttributes: undefined,
94
- bookingAttributes: undefined,
95
- calculateDeposit: false,
96
- bookingNumber: undefined,
97
- isRetry: false,
98
- package: undefined,
99
- isBusy: false,
100
- skipPaymentWithAgent: false,
101
- generatePaymentUrl: false,
102
- tagIds: [],
103
- agentAdressId: undefined,
104
- currentStep: OPTIONS_FORM_STEP,
105
- translations: undefined,
106
- };
107
-
108
- export const fetchPackage = createAsyncThunk(
109
- "booking/fetchPackage",
110
- async (_, { dispatch }) => {
111
- dispatch(setFetchingPackage(true));
112
- await dispatch(fetchAgents());
113
- await dispatch(fetchPackageDetails());
114
- await dispatch(fetchAccommodationViews());
115
- dispatch(setFetchingPackage(false));
116
- }
117
- );
118
-
119
- const fetchAgents = createAsyncThunk(
120
- "booking/agents",
121
- async (_, { dispatch, getState, signal }) => {
122
- const settings = getState() as RootState;
123
- return await packageApi.fetchAgents(signal, settings.apiSettings);
124
- }
125
- );
126
-
127
- const fetchPackageDetails = createAsyncThunk(
128
- "booking/details",
129
- async (_, { dispatch, getState, signal }) => {
130
- const state = getState() as RootState;
131
-
132
- const officeId = selectOfficeId(state);
133
- const productAttributes = selectProductAttributes(state);
134
- const bookingAttributes = selectBookingAttributes(state);
135
- const agentId = selectAgentId(state);
136
- const rooms = selectBookingRooms(state);
137
- const languageCode = selectLanguageCode(state);
138
-
139
- if (isNil(productAttributes)) {
140
- throw Error("productAttributes could not be found");
141
- }
142
-
143
- if (isNil(bookingAttributes)) {
144
- throw Error("bookingAttributes could not be found");
145
- }
146
-
147
- if (!rooms?.length) {
148
- throw Error("rooms could not be found");
149
- }
150
-
151
- var requestRooms = rooms?.map((x, i) => {
152
- var room = { index: i, pax: [] } as BookingPackageRequestRoom;
153
- range(0, x.adults).forEach(() => {
154
- room.pax.push({
155
- age: 30,
156
- } as BookingPackagePax);
157
- });
158
- x.childAges.forEach((x) => {
159
- room.pax.push({
160
- age: x,
161
- } as BookingPackagePax);
162
- });
163
- return room;
164
- });
165
-
166
- const isAllotment =
167
- bookingAttributes.tourCode ||
168
- bookingAttributes.allotmentName ||
169
- (bookingAttributes.allotmentIds && bookingAttributes.allotmentIds.length);
170
-
171
- const request = {
172
- officeId: officeId,
173
- agentId: agentId,
174
- payload: {
175
- searchType: isAllotment ? 1 : 0,
176
- catalogueId: bookingAttributes.catalog,
177
- productCode: productAttributes.productCode,
178
- fromDate: bookingAttributes.startDate,
179
- toDate: bookingAttributes.endDate,
180
- includeFlights: bookingAttributes.includeFlights,
181
- allotmentName: bookingAttributes.allotmentName,
182
- allotmentIds: bookingAttributes.allotmentIds ?? [],
183
- tourCode: bookingAttributes.tourCode,
184
- rooms: requestRooms,
185
- } as BookingPackageDetailsRequest,
186
- } as BookingPackageRequest<BookingPackageDetailsRequest>;
187
-
188
- return await packageApi.fetchDetails(
189
- request,
190
- signal,
191
- languageCode,
192
- state.apiSettings
193
- );
194
- }
195
- );
196
-
197
- const fetchAccommodationViews = createAsyncThunk(
198
- "booking/accommodationViews",
199
- async (_, { dispatch, getState, signal }) => {
200
- const state = getState() as RootState;
201
- if (!state.booking.accommodationViewId) return Promise.resolve();
202
-
203
- const languageCode = selectLanguageCode(state);
204
- const accommodationCodes = selectAccommodationCodes(state);
205
- const productCode = selectProductCode(state);
206
-
207
- if (!productCode) {
208
- throw Error('No product selected');
209
- }
210
-
211
- const request = {
212
- languageCode: languageCode,
213
- productCode: productCode,
214
- accommodationCodes: accommodationCodes,
215
- contentViewId: state.booking.accommodationViewId
216
- } as GenerateBookingAccommodationRequest;
217
-
218
- return await packageApi.fetchAccommodationViews(request, signal, state.apiSettings);
219
- }
220
- );
221
-
222
- const getActiveOption = (state: BookingState) => {
223
- if (state.package) return state.package.options.find((x) => x.isSelected);
224
- return null;
225
- };
226
-
227
- const changeOutwardFlight = (state: BookingPackage, flight: BookingPackageFlight) => {
228
- const currentOutwardFlight = state.outwardFlights.find(x => x.isSelected)!;
229
- const currentReturnFlight = state.returnFlights.find(x => x.isSelected)!;
230
-
231
- if (currentOutwardFlight?.entryLineGuid == flight.entryLineGuid) return;
232
-
233
- const newFlight = state.outwardFlights.find(x => x.entryLineGuid == flight.entryLineGuid);
234
- if (newFlight) {
235
- newFlight.isSelected = true;
236
- currentOutwardFlight.isSelected = false;
237
-
238
- if (newFlight.externalGuid) {
239
- if (currentOutwardFlight.externalGuid !== newFlight.externalGuid) {
240
- const newReturnFlight = state.returnFlights.find(x => x.externalGuid === newFlight.externalGuid)!;
241
-
242
- currentReturnFlight.isSelected = false;
243
- newReturnFlight.isSelected = true;
244
- }
245
- } else if (currentReturnFlight.externalGuid) {
246
- const firstInternal = state.returnFlights.find(x => !x.externalGuid);
247
-
248
- if (firstInternal) {
249
- currentReturnFlight.isSelected = false;
250
- firstInternal.isSelected = true;
251
- }
252
- }
253
- }
254
- }
255
-
256
- const changeReturnFlight = (state: BookingPackage, flight: BookingPackageFlight) => {
257
- const currentReturnFlight = state.returnFlights.find(x => x.isSelected)!;
258
-
259
- if (currentReturnFlight?.entryLineGuid == flight.entryLineGuid) return;
260
-
261
- const newFlight = state.outwardFlights.find(x => x.entryLineGuid == flight.entryLineGuid);
262
- if (newFlight) {
263
- newFlight.isSelected = true;
264
- currentReturnFlight.isSelected = false;
265
- }
266
- }
267
-
268
- const changePackageOption = (state: BookingPackage) => {
269
- const selectedOutward = state.outwardFlights.find(x => x.isSelected)!;
270
- const selectedReturn = state.returnFlights.find(x => x.isSelected)!;
271
- const validOptions = selectedOutward.validOptions.filter(x => selectedReturn.validOptions.some(y => x === y));
272
-
273
- const currentOption = state.options.find(x => x.isSelected)!;
274
- if (validOptions.some(x => x === currentOption.id)) return;
275
-
276
- const firstOption = state.options.find(x => validOptions.some(y => y === x.id))!;
277
- currentOption.isSelected = false;
278
- firstOption.isSelected = true;
279
-
280
- const currentRooms = currentOption.rooms.map(r => {
281
- const selectedOption = r.options.find(o => o.isSelected)!;
282
-
283
- return {
284
- accommodation: selectedOption?.accommodationCode,
285
- regime: selectedOption?.regimeCode
286
- };
287
- });
288
-
289
- firstOption.rooms.forEach((r, i) => {
290
- const currentRoom = currentRooms[i];
291
-
292
- const selectedOption = r.options.find(o => o.isSelected);
293
- const selection = r.options.find(x => x.accommodationCode === currentRoom.accommodation && x.regimeCode === currentRoom.regime);
294
-
295
- if (selection) {
296
- if (selection.entryLineGuid !== selectedOption?.entryLineGuid) {
297
- if (selectedOption) selectedOption.isSelected = false;
298
- selection.isSelected = true;
299
- }
300
- } else {
301
- const accommodationSelection = r.options.find(x => x.accommodationCode === currentRoom.accommodation);
302
- if (accommodationSelection) {
303
- if (accommodationSelection.entryLineGuid !== selectedOption?.entryLineGuid) {
304
- if (selectedOption) selectedOption.isSelected = false;
305
- accommodationSelection.isSelected = true;
306
- }
307
- } else {
308
- const firstOption = r.options[0];
309
- if (firstOption.entryLineGuid !== selectedOption?.entryLineGuid) {
310
- if (selectedOption) selectedOption.isSelected = false;
311
- firstOption.isSelected = true;
312
- }
313
- }
314
- }
315
- });
316
- }
317
-
318
- const bookingSlice = createSlice({
319
- name: "booking",
320
- initialState,
321
- reducers: {
322
- setOfficeId(state, action: PayloadAction<number>) {
323
- state.officeId = action.payload;
324
- },
325
- setLanguageCode(state, action: PayloadAction<string>) {
326
- state.languageCode = action.payload;
327
- },
328
- setTranslations(state, action: PayloadAction<any>) {
329
- state.translations = action.payload;
330
- },
331
- setBookingOptions(state, action: PayloadAction<BookingOptions>) {
332
- state.bookingOptions = action.payload;
333
- },
334
- setBookingType(state, action: PayloadAction<string>) {
335
- state.bookingType = action.payload;
336
- },
337
- setProductAttributes(state, action: PayloadAction<ProductAttributes>) {
338
- state.productAttributes = action.payload;
339
- },
340
- setBookingAttributes(state, action: PayloadAction<BookingAttributes>) {
341
- state.bookingAttributes = action.payload;
342
- },
343
- setCalculateDeposit(state, action: PayloadAction<boolean>) {
344
- state.calculateDeposit = action.payload;
345
- },
346
- setBookingNumber(state, action: PayloadAction<string>) {
347
- state.bookingNumber = action.payload;
348
- },
349
- setIsRetry(state, action: PayloadAction<boolean>) {
350
- state.isRetry = action.payload;
351
- },
352
- setFetchingPackage(state, action: PayloadAction<boolean>) {
353
- state.isBusy = action.payload;
354
- },
355
- setPackage(state, action: PayloadAction<BookingPackage>) {
356
- state.package = action.payload;
357
- },
358
- setPackageRooms(state, action: PayloadAction<BookingPackageRoom[]>) {
359
- const option = getActiveOption(state);
360
- if (option) option.rooms = action.payload;
361
- },
362
- setPackageOptionPax(state, action: PayloadAction<BookingOptionPax[]>) {
363
- const option = getActiveOption(state);
364
- if (option) option.optionPax = action.payload;
365
- },
366
- setPackageOptionUnits(state, action: PayloadAction<BookingOptionUnit[]>) {
367
- const option = getActiveOption(state);
368
- if (option) option.optionUnits = action.payload;
369
- },
370
- setSkipPayment(state, action: PayloadAction<boolean>) {
371
- state.skipPaymentWithAgent = action.payload;
372
- },
373
- setGeneratePaymentUrl(state, action: PayloadAction<boolean>) {
374
- state.generatePaymentUrl = action.payload;
375
- },
376
- setPackageGroups(
377
- state,
378
- action: PayloadAction<BookingOptionGroup<PerBookingPackageOption>[]>
379
- ) {
380
- const option = getActiveOption(state);
381
- if (option) option.groups = action.payload;
382
- },
383
- setPackageAirlineGroups(
384
- state,
385
- action: PayloadAction<BookingAirlineGroup<AirlineBookingPackageOption>[]>
386
- ) {
387
- const option = getActiveOption(state);
388
- if (option) option.airlineGroups = action.payload;
389
- },
390
- setPackageAirportGroups(
391
- state,
392
- action: PayloadAction<BookingAirportGroup<AirportBookingPackageOption>[]>
393
- ) {
394
- const option = getActiveOption(state);
395
- if (option) option.airportGroups = action.payload;
396
- },
397
- setTagIds(state, action: PayloadAction<number[] | undefined>) {
398
- state.tagIds = action.payload;
399
- },
400
- setAgentAdressId(state, action: PayloadAction<number | undefined>) {
401
- state.agentAdressId = action.payload;
402
- },
403
- setBookingRemarks(state, action: PayloadAction<string>) {
404
- state.remarks = action.payload;
405
- },
406
- setVoucherCodes(state, action: PayloadAction<string[]>) {
407
- state.voucherCodes = action.payload;
408
- },
409
- setCurrentStep(state, action: PayloadAction<BookingStep>) {
410
- document.body.scrollTop = 0; // For Safari
411
- document.documentElement.scrollTop = 0; // For Chrome, Firefox, IE and Opera
412
-
413
- state.currentStep = action.payload;
414
- },
415
- setFlights(state, action: PayloadAction<GroupedFlights>) {
416
- if (!state.package) return;
417
-
418
- changeOutwardFlight(state.package, action.payload.selectedOutward);
419
- changeReturnFlight(state.package, action.payload.selectedReturn);
420
- changePackageOption(state.package);
421
- },
422
- setAccommodationViewId(state, action: PayloadAction<number>) {
423
- state.accommodationViewId = action.payload;
424
- }
425
- },
426
- extraReducers: (builder) => {
427
- builder.addCase(fetchPackageDetails.fulfilled, (state, action) => {
428
- if (action.payload) {
429
- if (action.payload.errorCode) {
430
- console.error(
431
- action.payload.errorCode,
432
- action.payload.errorMessage,
433
- action.payload.errorDetails
434
- );
435
-
436
- state.isUnavailable = true;
437
- return;
438
- }
439
-
440
- if (!action.payload.payload) {
441
- state.isUnavailable = true;
442
- return;
443
- }
444
-
445
- const bookingRooms = state.bookingAttributes?.rooms;
446
- const flight = state.bookingAttributes?.flight;
447
- const packageDetails = action.payload.payload;
448
-
449
- let activeOption = packageDetails.options.find((x) => x.isSelected)!;
450
-
451
- if (flight) {
452
- const selectedOutward = packageDetails.outwardFlights.find(
453
- (x) => x.isSelected
454
- );
455
- const outwardFlights = packageDetails.outwardFlights.filter(
456
- (x) =>
457
- x.code === flight.outwardCode &&
458
- x.flightMetaData.flightLines[0]!.flightClass ===
459
- flight.outwardClass &&
460
- x.flightMetaData.flightLines.map((y) => y.number).join(",") ===
461
- flight.outwardNumbers.join(",")
462
- );
463
-
464
- // ook bij identieke vertrekvluchten eerst kijken als de returnflight kan gekoppeld worden.
465
- // op die manier moet het juiste koppel selected worden.
466
- // Enkel werkend met externe vluchten
467
- let outwardFlight: BookingPackageFlight | undefined = undefined;
468
- if (!isEmpty(outwardFlights)) {
469
- const returnExternalGuids = packageDetails.returnFlights
470
- .filter(
471
- (x) =>
472
- x.code === flight.returnCode &&
473
- x.flightMetaData.flightLines[0]!.flightClass ===
474
- flight.returnClass &&
475
- x.flightMetaData.flightLines
476
- .map((y) => y.number)
477
- .join(",") === flight.returnNumbers.join(",")
478
- )
479
- .map((f) => f.externalGuid)
480
- .filter((e) => e);
481
- outwardFlight = outwardFlights.find((o) =>
482
- returnExternalGuids.includes(o.externalGuid)
483
- );
484
- if (!outwardFlight) {
485
- outwardFlight = first(outwardFlights);
486
- }
487
- }
488
-
489
- if (selectedOutward && outwardFlight) {
490
- selectedOutward.isSelected = false;
491
- outwardFlight.isSelected = true;
492
- }
493
-
494
- const selectedReturn = packageDetails.returnFlights.find(
495
- (x) => x.isSelected
496
- );
497
- const returnFlight = outwardFlight?.externalGuid
498
- ? packageDetails.returnFlights.find(
499
- (x) => x.externalGuid === outwardFlight?.externalGuid
500
- )
501
- : packageDetails.returnFlights.find(
502
- (x) =>
503
- x.code === flight.returnCode &&
504
- x.flightMetaData.flightLines[0]!.flightClass ===
505
- flight.returnClass &&
506
- x.flightMetaData.flightLines
507
- .map((y) => y.number)
508
- .join(",") === flight.returnNumbers.join(",")
509
- );
510
-
511
- if (selectedReturn && returnFlight) {
512
- selectedReturn.isSelected = false;
513
- returnFlight.isSelected = true;
514
- }
515
-
516
- if (outwardFlight && returnFlight) {
517
- if (!outwardFlight.validOptions.some((x) => x == activeOption.id)) {
518
- activeOption.isSelected = false;
519
-
520
- activeOption = packageDetails.options.find((x) =>
521
- outwardFlight?.validOptions.some((y) => y === x.id)
522
- )!;
523
- activeOption.isSelected = true;
524
- }
525
- }
526
- }
527
-
528
- if (
529
- activeOption &&
530
- bookingRooms?.some((x) => x.accommodationCode || x.regimeCode)
531
- ) {
532
- bookingRooms.forEach((room, i) => {
533
- if (room.accommodationCode || room.regimeCode) {
534
- activeOption.rooms[i].options = activeOption.rooms[i].options.map(
535
- (ro) => ({
536
- ...ro,
537
- isSelected:
538
- ro.accommodationCode == room.accommodationCode &&
539
- ro.regimeCode == room.regimeCode,
540
- })
541
- );
542
- }
543
-
544
- // Fallback to an option that has the requested accommodation OR regime if the requested option is not available. If no fallback is available, select the first option.
545
- if (!activeOption.rooms[i].options.some(x => x.isSelected)) {
546
- const fallbackOption = activeOption.rooms[i].options.find(x => x.accommodationCode == room.accommodationCode || x.regimeCode == room.regimeCode);
547
- if (fallbackOption) {
548
- fallbackOption.isSelected = true;
549
- } else {
550
- activeOption.rooms[i].options[0].isSelected = true;
551
- }
552
- }
553
- });
554
- }
555
-
556
- state.package = packageDetails;
557
- }
558
- });
559
- builder.addCase(fetchAgents.fulfilled, (state, action) => {
560
- if (action.payload) {
561
- state.agents = action.payload;
562
- }
563
- });
564
- builder.addCase(fetchAccommodationViews.fulfilled, (state, action) => {
565
- if (action.payload) {
566
- state.accommodationViews = action.payload;
567
- }
568
- });
569
- },
570
- });
571
-
572
- export const {
573
- setOfficeId,
574
- setLanguageCode,
575
- setTranslations,
576
- setBookingOptions,
577
- setBookingType,
578
- setProductAttributes,
579
- setBookingAttributes,
580
- setCalculateDeposit,
581
- setBookingNumber,
582
- setIsRetry,
583
- setFetchingPackage,
584
- setPackage,
585
- setPackageRooms,
586
- setPackageOptionPax,
587
- setPackageOptionUnits,
588
- setPackageGroups,
589
- setSkipPayment,
590
- setGeneratePaymentUrl,
591
- setTagIds,
592
- setAgentAdressId,
593
- setBookingRemarks,
594
- setVoucherCodes,
595
- setCurrentStep,
596
- setPackageAirlineGroups,
597
- setPackageAirportGroups,
598
- setFlights,
599
- setAccommodationViewId
600
- } = bookingSlice.actions;
601
-
602
- export default bookingSlice.reducer;
603
-
1
+ import { PayloadAction, createAsyncThunk, createSlice } from "@reduxjs/toolkit";
2
+ import {
3
+ BookingAttributes,
4
+ BookingOptions,
5
+ GroupedFlights,
6
+ ProductAttributes,
7
+ } from "../../types";
8
+
9
+ import {
10
+ AirlineBookingPackageOption,
11
+ AirportBookingPackageOption,
12
+ BookingAirlineGroup,
13
+ BookingAirportGroup,
14
+ BookingOptionGroup,
15
+ BookingOptionPax,
16
+ BookingOptionUnit,
17
+ BookingPackage,
18
+ BookingPackageDetailsRequest,
19
+ BookingPackageFlight,
20
+ BookingPackagePax,
21
+ BookingPackageRequest,
22
+ BookingPackageRequestRoom,
23
+ BookingPackageRoom,
24
+ BookingTravelAgent,
25
+ GenerateBookingAccommodationRequest,
26
+ PerBookingPackageOption,
27
+ SelectedFlight,
28
+ } from "@qite/tide-client/build/types";
29
+ import { first, isEmpty, isNil, range } from "lodash";
30
+ import { RootState } from "../../store";
31
+ import { selectAgentId } from "../travelers-form/travelers-form-slice";
32
+ import packageApi from "./api";
33
+ import { BookingStep, OPTIONS_FORM_STEP } from "./constants";
34
+ import {
35
+ selectAccommodationCodes,
36
+ selectBookingAttributes,
37
+ selectBookingRooms,
38
+ selectLanguageCode,
39
+ selectOfficeId,
40
+ selectProductAttributes,
41
+ selectProductCode,
42
+ } from "./selectors";
43
+
44
+ export interface BookingState {
45
+ officeId: number;
46
+ languageCode: string;
47
+ productAttributes?: ProductAttributes;
48
+ bookingAttributes?: BookingAttributes;
49
+ calculateDeposit: boolean;
50
+ bookingNumber?: string;
51
+ isRetry: boolean;
52
+ package?: BookingPackage;
53
+ agents?: BookingTravelAgent[];
54
+ isBusy: boolean;
55
+ skipPaymentWithAgent: boolean;
56
+ generatePaymentUrl: boolean;
57
+ isUnavailable?: boolean;
58
+ tagIds?: number[];
59
+ agentAdressId?: number;
60
+ remarks?: string;
61
+ voucherCodes?: string[];
62
+ bookingOptions: BookingOptions;
63
+ bookingType: string;
64
+ currentStep: BookingStep;
65
+ translations?: {
66
+ language: string;
67
+ value: any;
68
+ }[];
69
+ accommodationViewId?: number;
70
+ accommodationViews?: { [key: string]: string };
71
+ }
72
+
73
+ const initialState: BookingState = {
74
+ officeId: 1,
75
+ languageCode: "nl-BE",
76
+ bookingOptions: {
77
+ b2b: {
78
+ tagIds: [],
79
+ entryStatus: 2,
80
+ customEntryStatusId: undefined,
81
+ },
82
+ b2b2c: {
83
+ tagIds: [],
84
+ entryStatus: 2,
85
+ customEntryStatusId: undefined,
86
+ },
87
+ b2c: {
88
+ tagIds: [],
89
+ entryStatus: 0,
90
+ customEntryStatusId: undefined,
91
+ },
92
+ },
93
+ bookingType: "b2c",
94
+ productAttributes: undefined,
95
+ bookingAttributes: undefined,
96
+ calculateDeposit: false,
97
+ bookingNumber: undefined,
98
+ isRetry: false,
99
+ package: undefined,
100
+ isBusy: false,
101
+ skipPaymentWithAgent: false,
102
+ generatePaymentUrl: false,
103
+ tagIds: [],
104
+ agentAdressId: undefined,
105
+ currentStep: OPTIONS_FORM_STEP,
106
+ translations: undefined,
107
+ };
108
+
109
+ export const fetchPackage = createAsyncThunk(
110
+ "booking/fetchPackage",
111
+ async (_, { dispatch }) => {
112
+ dispatch(setFetchingPackage(true));
113
+ await dispatch(fetchAgents());
114
+ await dispatch(fetchPackageDetails());
115
+ await dispatch(fetchAccommodationViews());
116
+ dispatch(setFetchingPackage(false));
117
+ }
118
+ );
119
+
120
+ const fetchAgents = createAsyncThunk(
121
+ "booking/agents",
122
+ async (_, { dispatch, getState, signal }) => {
123
+ const settings = getState() as RootState;
124
+ return await packageApi.fetchAgents(signal, settings.apiSettings);
125
+ }
126
+ );
127
+
128
+ const fetchPackageDetails = createAsyncThunk(
129
+ "booking/details",
130
+ async (_, { dispatch, getState, signal }) => {
131
+ const state = getState() as RootState;
132
+
133
+ const officeId = selectOfficeId(state);
134
+ const productAttributes = selectProductAttributes(state);
135
+ const bookingAttributes = selectBookingAttributes(state);
136
+ const agentId = selectAgentId(state);
137
+ const rooms = selectBookingRooms(state);
138
+ const languageCode = selectLanguageCode(state);
139
+
140
+ if (isNil(productAttributes)) {
141
+ throw Error("productAttributes could not be found");
142
+ }
143
+
144
+ if (isNil(bookingAttributes)) {
145
+ throw Error("bookingAttributes could not be found");
146
+ }
147
+
148
+ if (!rooms?.length) {
149
+ throw Error("rooms could not be found");
150
+ }
151
+
152
+ var requestRooms = rooms?.map((x, i) => {
153
+ var room = { index: i, pax: [] } as BookingPackageRequestRoom;
154
+ range(0, x.adults).forEach(() => {
155
+ room.pax.push({
156
+ age: 30,
157
+ } as BookingPackagePax);
158
+ });
159
+ x.childAges.forEach((x) => {
160
+ room.pax.push({
161
+ age: x,
162
+ } as BookingPackagePax);
163
+ });
164
+ return room;
165
+ });
166
+
167
+ const isAllotment =
168
+ bookingAttributes.tourCode ||
169
+ bookingAttributes.allotmentName ||
170
+ (bookingAttributes.allotmentIds && bookingAttributes.allotmentIds.length);
171
+
172
+ let searchType = isAllotment
173
+ ? 1 // ALLOTMENT
174
+ : 0; // DEFAULT;
175
+
176
+ let outwardFlight: SelectedFlight | undefined;
177
+ let returnFlight: SelectedFlight | undefined;
178
+ if (bookingAttributes.flightRouteId && bookingAttributes.flight) {
179
+ searchType = 3; // FLIGHT;
180
+
181
+ outwardFlight = {
182
+ flightCode: bookingAttributes.flight.outwardCode,
183
+ startDateTime: bookingAttributes.flight.outwardDepartureDate,
184
+ endDateTime: bookingAttributes.flight.outwardArrivalDate,
185
+ } as SelectedFlight;
186
+
187
+ if (bookingAttributes.flight.returnCode) {
188
+ returnFlight = {
189
+ flightCode: bookingAttributes.flight.returnCode,
190
+ startDateTime: bookingAttributes.flight.returnDepartureDate,
191
+ endDateTime: bookingAttributes.flight.returnArrivalDate,
192
+ } as SelectedFlight;
193
+ }
194
+ }
195
+
196
+ const request = {
197
+ officeId: officeId,
198
+ agentId: agentId,
199
+ payload: {
200
+ searchType: searchType,
201
+ catalogueId: bookingAttributes.catalog,
202
+ productCode: productAttributes.productCode,
203
+ fromDate: bookingAttributes.startDate,
204
+ toDate: bookingAttributes.endDate,
205
+ includeFlights: bookingAttributes.includeFlights,
206
+ allotmentName: bookingAttributes.allotmentName,
207
+ allotmentIds: bookingAttributes.allotmentIds ?? [],
208
+ tourCode: bookingAttributes.tourCode,
209
+ rooms: requestRooms,
210
+ routeId: bookingAttributes.flightRouteId,
211
+ outwardFlight: outwardFlight,
212
+ returnFlight: returnFlight,
213
+ } as BookingPackageDetailsRequest,
214
+ } as BookingPackageRequest<BookingPackageDetailsRequest>;
215
+
216
+ return await packageApi.fetchDetails(
217
+ request,
218
+ signal,
219
+ languageCode,
220
+ state.apiSettings
221
+ );
222
+ }
223
+ );
224
+
225
+ const fetchAccommodationViews = createAsyncThunk(
226
+ "booking/accommodationViews",
227
+ async (_, { dispatch, getState, signal }) => {
228
+ const state = getState() as RootState;
229
+ if (!state.booking.accommodationViewId) return Promise.resolve();
230
+
231
+ const languageCode = selectLanguageCode(state);
232
+ const accommodationCodes = selectAccommodationCodes(state);
233
+ const productCode = selectProductCode(state);
234
+
235
+ if (!productCode) {
236
+ throw Error("No product selected");
237
+ }
238
+
239
+ const request = {
240
+ languageCode: languageCode,
241
+ productCode: productCode,
242
+ accommodationCodes: accommodationCodes,
243
+ contentViewId: state.booking.accommodationViewId,
244
+ } as GenerateBookingAccommodationRequest;
245
+
246
+ return await packageApi.fetchAccommodationViews(
247
+ request,
248
+ signal,
249
+ state.apiSettings
250
+ );
251
+ }
252
+ );
253
+
254
+ const getActiveOption = (state: BookingState) => {
255
+ if (state.package) return state.package.options.find((x) => x.isSelected);
256
+ return null;
257
+ };
258
+
259
+ const changeOutwardFlight = (
260
+ state: BookingPackage,
261
+ flight: BookingPackageFlight
262
+ ) => {
263
+ const currentOutwardFlight = state.outwardFlights.find((x) => x.isSelected)!;
264
+ const currentReturnFlight = state.returnFlights.find((x) => x.isSelected)!;
265
+
266
+ if (currentOutwardFlight?.entryLineGuid == flight.entryLineGuid) return;
267
+
268
+ const newFlight = state.outwardFlights.find(
269
+ (x) => x.entryLineGuid == flight.entryLineGuid
270
+ );
271
+ if (newFlight) {
272
+ newFlight.isSelected = true;
273
+ currentOutwardFlight.isSelected = false;
274
+
275
+ if (newFlight.externalGuid) {
276
+ if (currentOutwardFlight.externalGuid !== newFlight.externalGuid) {
277
+ const newReturnFlight = state.returnFlights.find(
278
+ (x) => x.externalGuid === newFlight.externalGuid
279
+ )!;
280
+
281
+ currentReturnFlight.isSelected = false;
282
+ newReturnFlight.isSelected = true;
283
+ }
284
+ } else if (currentReturnFlight.externalGuid) {
285
+ const firstInternal = state.returnFlights.find((x) => !x.externalGuid);
286
+
287
+ if (firstInternal) {
288
+ currentReturnFlight.isSelected = false;
289
+ firstInternal.isSelected = true;
290
+ }
291
+ }
292
+ }
293
+ };
294
+
295
+ const changeReturnFlight = (
296
+ state: BookingPackage,
297
+ flight: BookingPackageFlight
298
+ ) => {
299
+ const currentReturnFlight = state.returnFlights.find((x) => x.isSelected)!;
300
+
301
+ if (currentReturnFlight?.entryLineGuid == flight.entryLineGuid) return;
302
+
303
+ const newFlight = state.outwardFlights.find(
304
+ (x) => x.entryLineGuid == flight.entryLineGuid
305
+ );
306
+ if (newFlight) {
307
+ newFlight.isSelected = true;
308
+ currentReturnFlight.isSelected = false;
309
+ }
310
+ };
311
+
312
+ const changePackageOption = (state: BookingPackage) => {
313
+ const selectedOutward = state.outwardFlights.find((x) => x.isSelected)!;
314
+ const selectedReturn = state.returnFlights.find((x) => x.isSelected)!;
315
+ const validOptions = selectedOutward.validOptions.filter((x) =>
316
+ selectedReturn.validOptions.some((y) => x === y)
317
+ );
318
+
319
+ const currentOption = state.options.find((x) => x.isSelected)!;
320
+ if (validOptions.some((x) => x === currentOption.id)) return;
321
+
322
+ const firstOption = state.options.find((x) =>
323
+ validOptions.some((y) => y === x.id)
324
+ )!;
325
+ currentOption.isSelected = false;
326
+ firstOption.isSelected = true;
327
+
328
+ const currentRooms = currentOption.rooms.map((r) => {
329
+ const selectedOption = r.options.find((o) => o.isSelected)!;
330
+
331
+ return {
332
+ accommodation: selectedOption?.accommodationCode,
333
+ regime: selectedOption?.regimeCode,
334
+ };
335
+ });
336
+
337
+ firstOption.rooms.forEach((r, i) => {
338
+ const currentRoom = currentRooms[i];
339
+
340
+ const selectedOption = r.options.find((o) => o.isSelected);
341
+ const selection = r.options.find(
342
+ (x) =>
343
+ x.accommodationCode === currentRoom.accommodation &&
344
+ x.regimeCode === currentRoom.regime
345
+ );
346
+
347
+ if (selection) {
348
+ if (selection.entryLineGuid !== selectedOption?.entryLineGuid) {
349
+ if (selectedOption) selectedOption.isSelected = false;
350
+ selection.isSelected = true;
351
+ }
352
+ } else {
353
+ const accommodationSelection = r.options.find(
354
+ (x) => x.accommodationCode === currentRoom.accommodation
355
+ );
356
+ if (accommodationSelection) {
357
+ if (
358
+ accommodationSelection.entryLineGuid !== selectedOption?.entryLineGuid
359
+ ) {
360
+ if (selectedOption) selectedOption.isSelected = false;
361
+ accommodationSelection.isSelected = true;
362
+ }
363
+ } else {
364
+ const firstOption = r.options[0];
365
+ if (firstOption.entryLineGuid !== selectedOption?.entryLineGuid) {
366
+ if (selectedOption) selectedOption.isSelected = false;
367
+ firstOption.isSelected = true;
368
+ }
369
+ }
370
+ }
371
+ });
372
+ };
373
+
374
+ const bookingSlice = createSlice({
375
+ name: "booking",
376
+ initialState,
377
+ reducers: {
378
+ setOfficeId(state, action: PayloadAction<number>) {
379
+ state.officeId = action.payload;
380
+ },
381
+ setLanguageCode(state, action: PayloadAction<string>) {
382
+ state.languageCode = action.payload;
383
+ },
384
+ setTranslations(state, action: PayloadAction<any>) {
385
+ state.translations = action.payload;
386
+ },
387
+ setBookingOptions(state, action: PayloadAction<BookingOptions>) {
388
+ state.bookingOptions = action.payload;
389
+ },
390
+ setBookingType(state, action: PayloadAction<string>) {
391
+ state.bookingType = action.payload;
392
+ },
393
+ setProductAttributes(state, action: PayloadAction<ProductAttributes>) {
394
+ state.productAttributes = action.payload;
395
+ },
396
+ setBookingAttributes(state, action: PayloadAction<BookingAttributes>) {
397
+ state.bookingAttributes = action.payload;
398
+ },
399
+ setCalculateDeposit(state, action: PayloadAction<boolean>) {
400
+ state.calculateDeposit = action.payload;
401
+ },
402
+ setBookingNumber(state, action: PayloadAction<string>) {
403
+ state.bookingNumber = action.payload;
404
+ },
405
+ setIsRetry(state, action: PayloadAction<boolean>) {
406
+ state.isRetry = action.payload;
407
+ },
408
+ setFetchingPackage(state, action: PayloadAction<boolean>) {
409
+ state.isBusy = action.payload;
410
+ },
411
+ setPackage(state, action: PayloadAction<BookingPackage>) {
412
+ state.package = action.payload;
413
+ },
414
+ setPackageRooms(state, action: PayloadAction<BookingPackageRoom[]>) {
415
+ const option = getActiveOption(state);
416
+ if (option) option.rooms = action.payload;
417
+ },
418
+ setPackageOptionPax(state, action: PayloadAction<BookingOptionPax[]>) {
419
+ const option = getActiveOption(state);
420
+ if (option) option.optionPax = action.payload;
421
+ },
422
+ setPackageOptionUnits(state, action: PayloadAction<BookingOptionUnit[]>) {
423
+ const option = getActiveOption(state);
424
+ if (option) option.optionUnits = action.payload;
425
+ },
426
+ setSkipPayment(state, action: PayloadAction<boolean>) {
427
+ state.skipPaymentWithAgent = action.payload;
428
+ },
429
+ setGeneratePaymentUrl(state, action: PayloadAction<boolean>) {
430
+ state.generatePaymentUrl = action.payload;
431
+ },
432
+ setPackageGroups(
433
+ state,
434
+ action: PayloadAction<BookingOptionGroup<PerBookingPackageOption>[]>
435
+ ) {
436
+ const option = getActiveOption(state);
437
+ if (option) option.groups = action.payload;
438
+ },
439
+ setPackageAirlineGroups(
440
+ state,
441
+ action: PayloadAction<BookingAirlineGroup<AirlineBookingPackageOption>[]>
442
+ ) {
443
+ const option = getActiveOption(state);
444
+ if (option) option.airlineGroups = action.payload;
445
+ },
446
+ setPackageAirportGroups(
447
+ state,
448
+ action: PayloadAction<BookingAirportGroup<AirportBookingPackageOption>[]>
449
+ ) {
450
+ const option = getActiveOption(state);
451
+ if (option) option.airportGroups = action.payload;
452
+ },
453
+ setTagIds(state, action: PayloadAction<number[] | undefined>) {
454
+ state.tagIds = action.payload;
455
+ },
456
+ setAgentAdressId(state, action: PayloadAction<number | undefined>) {
457
+ state.agentAdressId = action.payload;
458
+ },
459
+ setBookingRemarks(state, action: PayloadAction<string>) {
460
+ state.remarks = action.payload;
461
+ },
462
+ setVoucherCodes(state, action: PayloadAction<string[]>) {
463
+ state.voucherCodes = action.payload;
464
+ },
465
+ setCurrentStep(state, action: PayloadAction<BookingStep>) {
466
+ document.body.scrollTop = 0; // For Safari
467
+ document.documentElement.scrollTop = 0; // For Chrome, Firefox, IE and Opera
468
+
469
+ state.currentStep = action.payload;
470
+ },
471
+ setFlights(state, action: PayloadAction<GroupedFlights>) {
472
+ if (!state.package) return;
473
+
474
+ changeOutwardFlight(state.package, action.payload.selectedOutward);
475
+ changeReturnFlight(state.package, action.payload.selectedReturn);
476
+ changePackageOption(state.package);
477
+ },
478
+ setAccommodationViewId(state, action: PayloadAction<number>) {
479
+ state.accommodationViewId = action.payload;
480
+ },
481
+ },
482
+ extraReducers: (builder) => {
483
+ builder.addCase(fetchPackageDetails.fulfilled, (state, action) => {
484
+ if (action.payload) {
485
+ if (action.payload.errorCode) {
486
+ console.error(
487
+ action.payload.errorCode,
488
+ action.payload.errorMessage,
489
+ action.payload.errorDetails
490
+ );
491
+
492
+ state.isUnavailable = true;
493
+ return;
494
+ }
495
+
496
+ if (!action.payload.payload) {
497
+ state.isUnavailable = true;
498
+ return;
499
+ }
500
+
501
+ const bookingRooms = state.bookingAttributes?.rooms;
502
+ const flight = state.bookingAttributes?.flight;
503
+ const packageDetails = action.payload.payload;
504
+
505
+ let activeOption = packageDetails.options.find((x) => x.isSelected)!;
506
+
507
+ if (flight) {
508
+ const selectedOutward = packageDetails.outwardFlights.find(
509
+ (x) => x.isSelected
510
+ );
511
+ const outwardFlights = packageDetails.outwardFlights.filter(
512
+ (x) =>
513
+ x.code === flight.outwardCode &&
514
+ x.flightMetaData.flightLines[0]!.flightClass ===
515
+ flight.outwardClass &&
516
+ x.flightMetaData.flightLines.map((y) => y.number).join(",") ===
517
+ flight.outwardNumbers.join(",")
518
+ );
519
+
520
+ // ook bij identieke vertrekvluchten eerst kijken als de returnflight kan gekoppeld worden.
521
+ // op die manier moet het juiste koppel selected worden.
522
+ // Enkel werkend met externe vluchten
523
+ let outwardFlight: BookingPackageFlight | undefined = undefined;
524
+ if (!isEmpty(outwardFlights)) {
525
+ const returnExternalGuids = packageDetails.returnFlights
526
+ .filter(
527
+ (x) =>
528
+ x.code === flight.returnCode &&
529
+ x.flightMetaData.flightLines[0]!.flightClass ===
530
+ flight.returnClass &&
531
+ x.flightMetaData.flightLines
532
+ .map((y) => y.number)
533
+ .join(",") === flight.returnNumbers.join(",")
534
+ )
535
+ .map((f) => f.externalGuid)
536
+ .filter((e) => e);
537
+ outwardFlight = outwardFlights.find((o) =>
538
+ returnExternalGuids.includes(o.externalGuid)
539
+ );
540
+ if (!outwardFlight) {
541
+ outwardFlight = first(outwardFlights);
542
+ }
543
+ }
544
+
545
+ if (selectedOutward && outwardFlight) {
546
+ selectedOutward.isSelected = false;
547
+ outwardFlight.isSelected = true;
548
+ }
549
+
550
+ const selectedReturn = packageDetails.returnFlights.find(
551
+ (x) => x.isSelected
552
+ );
553
+ const returnFlight = outwardFlight?.externalGuid
554
+ ? packageDetails.returnFlights.find(
555
+ (x) => x.externalGuid === outwardFlight?.externalGuid
556
+ )
557
+ : packageDetails.returnFlights.find(
558
+ (x) =>
559
+ x.code === flight.returnCode &&
560
+ x.flightMetaData.flightLines[0]!.flightClass ===
561
+ flight.returnClass &&
562
+ x.flightMetaData.flightLines
563
+ .map((y) => y.number)
564
+ .join(",") === flight.returnNumbers.join(",")
565
+ );
566
+
567
+ if (selectedReturn && returnFlight) {
568
+ selectedReturn.isSelected = false;
569
+ returnFlight.isSelected = true;
570
+ }
571
+
572
+ if (outwardFlight && returnFlight) {
573
+ if (!outwardFlight.validOptions.some((x) => x == activeOption.id)) {
574
+ activeOption.isSelected = false;
575
+
576
+ activeOption = packageDetails.options.find((x) =>
577
+ outwardFlight?.validOptions.some((y) => y === x.id)
578
+ )!;
579
+ activeOption.isSelected = true;
580
+ }
581
+ }
582
+ }
583
+
584
+ if (
585
+ activeOption &&
586
+ bookingRooms?.some((x) => x.accommodationCode || x.regimeCode)
587
+ ) {
588
+ bookingRooms.forEach((room, i) => {
589
+ if (room.accommodationCode || room.regimeCode) {
590
+ activeOption.rooms[i].options = activeOption.rooms[i].options.map(
591
+ (ro) => ({
592
+ ...ro,
593
+ isSelected:
594
+ ro.accommodationCode == room.accommodationCode &&
595
+ ro.regimeCode == room.regimeCode,
596
+ })
597
+ );
598
+ }
599
+
600
+ // Fallback to an option that has the requested accommodation OR regime if the requested option is not available. If no fallback is available, select the first option.
601
+ if (!activeOption.rooms[i].options.some((x) => x.isSelected)) {
602
+ const fallbackOption = activeOption.rooms[i].options.find(
603
+ (x) =>
604
+ x.accommodationCode == room.accommodationCode ||
605
+ x.regimeCode == room.regimeCode
606
+ );
607
+ if (fallbackOption) {
608
+ fallbackOption.isSelected = true;
609
+ } else {
610
+ activeOption.rooms[i].options[0].isSelected = true;
611
+ }
612
+ }
613
+ });
614
+ }
615
+
616
+ state.package = packageDetails;
617
+ }
618
+ });
619
+ builder.addCase(fetchAgents.fulfilled, (state, action) => {
620
+ if (action.payload) {
621
+ state.agents = action.payload;
622
+ }
623
+ });
624
+ builder.addCase(fetchAccommodationViews.fulfilled, (state, action) => {
625
+ if (action.payload) {
626
+ state.accommodationViews = action.payload;
627
+ }
628
+ });
629
+ },
630
+ });
631
+
632
+ export const {
633
+ setOfficeId,
634
+ setLanguageCode,
635
+ setTranslations,
636
+ setBookingOptions,
637
+ setBookingType,
638
+ setProductAttributes,
639
+ setBookingAttributes,
640
+ setCalculateDeposit,
641
+ setBookingNumber,
642
+ setIsRetry,
643
+ setFetchingPackage,
644
+ setPackage,
645
+ setPackageRooms,
646
+ setPackageOptionPax,
647
+ setPackageOptionUnits,
648
+ setPackageGroups,
649
+ setSkipPayment,
650
+ setGeneratePaymentUrl,
651
+ setTagIds,
652
+ setAgentAdressId,
653
+ setBookingRemarks,
654
+ setVoucherCodes,
655
+ setCurrentStep,
656
+ setPackageAirlineGroups,
657
+ setPackageAirportGroups,
658
+ setFlights,
659
+ setAccommodationViewId,
660
+ } = bookingSlice.actions;
661
+
662
+ export default bookingSlice.reducer;