@sbb-esta/lyne-elements-dev 5.0.0-next.3-dev.1779188320 → 5.0.0-next.3-dev.1779200097

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 (162) hide show
  1. package/autocomplete/autocomplete-base-element.js +2 -2
  2. package/autocomplete/autocomplete.component.js +1 -1
  3. package/{autocomplete-base-element-DRGjFF9_.js → autocomplete-base-element-6leI6UF4.js} +24 -14
  4. package/autocomplete.js +4 -4
  5. package/autocomplete.pure.js +3 -3
  6. package/calendar/calendar/calendar.component.js +2 -2
  7. package/{calendar.component-oS_T4s4k.js → calendar.component-BK4w5GWV.js} +35 -29
  8. package/calendar.js +4 -4
  9. package/calendar.pure.js +3 -3
  10. package/carousel/carousel/carousel.component.js +1 -1
  11. package/carousel/carousel-item/carousel-item.component.js +2 -2
  12. package/carousel/carousel-list/carousel-list.component.js +1 -1
  13. package/{carousel-item.component-DyZAdXL_.js → carousel-item.component-B6WmZPkW.js} +12 -2
  14. package/{carousel-list.component-Ai0uJWB7.js → carousel-list.component-CnZehdy6.js} +6 -13
  15. package/{carousel.component-9oMMOyWo.js → carousel.component-ZrkGeJgZ.js} +1 -1
  16. package/carousel.js +5 -5
  17. package/carousel.pure.js +4 -4
  18. package/chip/chip-group/chip-group.component.js +1 -1
  19. package/{chip-group.component-DJORKUef.js → chip-group.component-BIP7c2G0.js} +2 -7
  20. package/chip.js +1 -1
  21. package/chip.pure.js +1 -1
  22. package/core/overlay/overlay-outside-pointer-event-listener.js +1 -1
  23. package/custom-elements.json +3886 -3393
  24. package/datepicker/datepicker/datepicker.component.js +1 -1
  25. package/development/autocomplete/autocomplete-base-element.d.ts +6 -3
  26. package/development/autocomplete/autocomplete-base-element.d.ts.map +1 -1
  27. package/development/autocomplete/autocomplete-base-element.js +2 -2
  28. package/development/autocomplete/autocomplete.component.js +1 -1
  29. package/development/autocomplete-base-element-qnPoRLvM.js +695 -0
  30. package/development/autocomplete.js +2 -2
  31. package/development/autocomplete.pure.js +2 -2
  32. package/development/calendar/calendar/calendar.component.d.ts +8 -0
  33. package/development/calendar/calendar/calendar.component.d.ts.map +1 -1
  34. package/development/calendar/calendar/calendar.component.js +2 -2
  35. package/development/calendar.component-BKWvdgzO.js +1461 -0
  36. package/development/calendar.js +2 -2
  37. package/development/calendar.pure.js +2 -2
  38. package/development/carousel/carousel/carousel.component.js +1 -1
  39. package/development/carousel/carousel-item/carousel-item.component.d.ts +8 -6
  40. package/development/carousel/carousel-item/carousel-item.component.d.ts.map +1 -1
  41. package/development/carousel/carousel-item/carousel-item.component.js +2 -2
  42. package/development/carousel/carousel-list/carousel-list.component.d.ts.map +1 -1
  43. package/development/carousel/carousel-list/carousel-list.component.js +1 -1
  44. package/development/{carousel-item.component-B_0wdIAu.js → carousel-item.component-nuo7jAt3.js} +16 -4
  45. package/development/carousel-list.component-B8PUREwS.js +126 -0
  46. package/development/carousel.component-D_KIJ9Bb.js +139 -0
  47. package/development/carousel.js +4 -4
  48. package/development/carousel.pure.js +4 -4
  49. package/development/chip/chip-group/chip-group.component.d.ts +1 -23
  50. package/development/chip/chip-group/chip-group.component.d.ts.map +1 -1
  51. package/development/chip/chip-group/chip-group.component.js +1 -1
  52. package/development/chip-group.component-CVizDsAz.js +395 -0
  53. package/development/chip.js +1 -1
  54. package/development/chip.pure.js +1 -1
  55. package/development/core/overlay/overlay-outside-pointer-event-listener.d.ts +1 -1
  56. package/development/core/overlay/overlay-outside-pointer-event-listener.d.ts.map +1 -1
  57. package/development/core/overlay/overlay-outside-pointer-event-listener.js +2 -2
  58. package/development/core.d.ts +0 -1
  59. package/development/core.d.ts.map +1 -1
  60. package/development/datepicker/datepicker/datepicker.component.d.ts +1 -1
  61. package/development/datepicker/datepicker/datepicker.component.d.ts.map +1 -1
  62. package/development/datepicker/datepicker/datepicker.component.js +3 -3
  63. package/development/dialog/dialog/dialog.component.d.ts +2 -3
  64. package/development/dialog/dialog/dialog.component.d.ts.map +1 -1
  65. package/development/dialog/dialog/dialog.component.js +1 -1
  66. package/development/{dialog.component-AItB8ixG.js → dialog.component-D8B2dVck.js} +4 -7
  67. package/development/dialog.js +1 -1
  68. package/development/dialog.pure.js +1 -1
  69. package/development/file-selector/common/file-selector-common.d.ts +5 -0
  70. package/development/file-selector/common/file-selector-common.d.ts.map +1 -1
  71. package/development/file-selector/common/file-selector-common.js +2 -2
  72. package/development/{file-selector-common-CGL95Rnw.js → file-selector-common-mZ5UYxOF.js} +17 -8
  73. package/development/file-selector.js +2 -2
  74. package/development/file-selector.pure.js +2 -2
  75. package/development/form-field/form-field/form-field.component.d.ts +1 -1
  76. package/development/form-field/form-field/form-field.component.d.ts.map +1 -1
  77. package/development/form-field.component-CPWBCHer.js +1 -1
  78. package/development/overlay/overlay-base-element.d.ts +4 -4
  79. package/development/overlay/overlay-base-element.d.ts.map +1 -1
  80. package/development/overlay/overlay-base-element.js +3 -9
  81. package/development/overlay/overlay.component.d.ts +3 -3
  82. package/development/overlay/overlay.component.d.ts.map +1 -1
  83. package/development/overlay/overlay.component.js +1 -1
  84. package/development/{overlay.component-rFmitc_r.js → overlay.component-YSCuGeme.js} +4 -7
  85. package/development/overlay.js +1 -1
  86. package/development/overlay.pure.js +1 -1
  87. package/development/paginator/common/paginator-common.d.ts +11 -6
  88. package/development/paginator/common/paginator-common.d.ts.map +1 -1
  89. package/development/paginator/common/paginator-common.js +2 -2
  90. package/development/paginator-common-B7iba0wQ.js +432 -0
  91. package/development/paginator.js +2 -2
  92. package/development/paginator.pure.js +2 -2
  93. package/development/popover/popover.component.d.ts +7 -0
  94. package/development/popover/popover.component.d.ts.map +1 -1
  95. package/development/popover/popover.component.js +2 -2
  96. package/development/popover.component-CKUZbWbV.js +578 -0
  97. package/development/popover.js +2 -2
  98. package/development/popover.pure.js +2 -2
  99. package/development/{step.component-CcjtUFyR.js → step.component-B_YHm1tz.js} +32 -10
  100. package/development/stepper/step/step.component.d.ts +14 -1
  101. package/development/stepper/step/step.component.d.ts.map +1 -1
  102. package/development/stepper/step/step.component.js +2 -2
  103. package/development/stepper/stepper/stepper.component.d.ts +9 -5
  104. package/development/stepper/stepper/stepper.component.d.ts.map +1 -1
  105. package/development/stepper/stepper/stepper.component.js +1 -1
  106. package/development/stepper.component-BeqeUIRN.js +443 -0
  107. package/development/stepper.js +3 -3
  108. package/development/stepper.pure.js +3 -3
  109. package/development/tab-group.component-CGMBihsQ.js +297 -0
  110. package/development/{tab-label.component-CNX8bsJL.js → tab-label.component-CQOLO2t7.js} +11 -13
  111. package/development/tabs/tab-group/tab-group.component.d.ts +15 -8
  112. package/development/tabs/tab-group/tab-group.component.d.ts.map +1 -1
  113. package/development/tabs/tab-group/tab-group.component.js +2 -2
  114. package/development/tabs/tab-label/tab-label.component.d.ts.map +1 -1
  115. package/development/tabs/tab-label/tab-label.component.js +1 -1
  116. package/development/tabs.js +3 -3
  117. package/development/tabs.pure.js +3 -3
  118. package/dialog/dialog/dialog.component.js +1 -1
  119. package/{dialog.component-4O3__zyd.js → dialog.component-CsBryhM-.js} +3 -6
  120. package/dialog.js +1 -1
  121. package/dialog.pure.js +1 -1
  122. package/file-selector/common/file-selector-common.js +2 -2
  123. package/{file-selector-common-BE780nIa.js → file-selector-common-VNwWFEmM.js} +26 -19
  124. package/file-selector.js +5 -5
  125. package/file-selector.pure.js +4 -4
  126. package/overlay/overlay-base-element.js +2 -8
  127. package/overlay/overlay.component.js +1 -1
  128. package/{overlay.component-C80XN61v.js → overlay.component-myJflfLY.js} +3 -6
  129. package/overlay.js +1 -1
  130. package/overlay.pure.js +1 -1
  131. package/package.json +2 -2
  132. package/paginator/common/paginator-common.js +2 -2
  133. package/{paginator-common-uoCIcl4X.js → paginator-common-BWewgXga.js} +42 -24
  134. package/paginator.js +5 -5
  135. package/paginator.pure.js +4 -4
  136. package/popover/popover.component.js +2 -2
  137. package/{popover.component-DJruKzY-.js → popover.component-Dm_86bPY.js} +19 -7
  138. package/popover.js +3 -3
  139. package/popover.pure.js +2 -2
  140. package/{step.component-BY6_2Wh2.js → step.component-tJh-Pb_E.js} +23 -8
  141. package/stepper/step/step.component.js +2 -2
  142. package/stepper/stepper/stepper.component.js +1 -1
  143. package/{stepper.component-lh0YSbT1.js → stepper.component-BCl77LVa.js} +20 -3
  144. package/stepper.js +4 -4
  145. package/stepper.pure.js +3 -3
  146. package/{tab-group.component-t4eeCNf-.js → tab-group.component-CDhggNpe.js} +27 -2
  147. package/{tab-label.component-DvvSohaG.js → tab-label.component-BuF2qt1T.js} +23 -22
  148. package/tabs/tab-group/tab-group.component.js +2 -2
  149. package/tabs/tab-label/tab-label.component.js +1 -1
  150. package/tabs.js +7 -7
  151. package/tabs.pure.js +6 -6
  152. package/development/autocomplete-base-element-DQiZ6hhk.js +0 -683
  153. package/development/calendar.component-CjlcDyS5.js +0 -1450
  154. package/development/carousel-list.component-DpDh5tqB.js +0 -133
  155. package/development/carousel.component-ClWvnZ8s.js +0 -139
  156. package/development/chip-group.component-Sgmi-kSy.js +0 -403
  157. package/development/core/interfaces/overlay-close-details.d.ts +0 -6
  158. package/development/core/interfaces/overlay-close-details.d.ts.map +0 -1
  159. package/development/paginator-common-BCeNUko_.js +0 -410
  160. package/development/popover.component-BmTfHMFw.js +0 -564
  161. package/development/stepper.component-80S-oXrH.js +0 -422
  162. package/development/tab-group.component-D4fvd6Uu.js +0 -265
@@ -1,1450 +0,0 @@
1
- import { __esDecorate, __runInitializers } from "tslib";
2
- import { html, isServer, nothing, unsafeCSS } from "lit";
3
- import { property, state } from "lit/decorators.js";
4
- import { DAYS_PER_ROW, MONDAY, MONTHS_PER_PAGE, MONTHS_PER_ROW, SbbElement, SbbLanguageController, SbbMediaMatcherController, SbbMediaQueryBreakpointLargeAndAbove, THURSDAY, TUESDAY, WEDNESDAY, YEARS_PER_PAGE, YEARS_PER_ROW, defaultDateAdapter, forceType, handleDistinctChange, i18nCalendarDateSelection, i18nCalendarWeekNumber, i18nNextMonth, i18nNextYear, i18nNextYearRange, i18nPreviousMonth, i18nPreviousYear, i18nPreviousYearRange, i18nYearMonthSelection, isArrowKeyOrPageKeysPressed, plainDate, readConfig, screenReaderOnlyStyles } from "./core.js";
5
- import { SbbSecondaryButtonElement } from "./button.pure.js";
6
- import { SbbIconElement } from "./icon.pure.js";
7
- import { SbbCalendarDayElement } from "./calendar/calendar-day/calendar-day.component.js";
8
- import { SbbCalendarMonthElement } from "./calendar/calendar-month/calendar-month.component.js";
9
- import { SbbCalendarWeekdayElement } from "./calendar/calendar-weekday/calendar-weekday.component.js";
10
- import { SbbCalendarWeeknumberElement } from "./calendar/calendar-weeknumber/calendar-weeknumber.component.js";
11
- import { SbbCalendarYearElement } from "./calendar/calendar-year/calendar-year.component.js";
12
- //#region src/elements/calendar/calendar/calendar.scss?inline
13
- var calendar_default = ":host {\n display: block;\n width: max-content;\n line-height: var(--sbb-typo-line-height-text);\n --sbb-calendar-cell-size: 2.75rem;\n --sbb-calendar-cell-border-radius: calc(\n var(--sbb-border-radius-4x) + var(--sbb-calendar-cell-border-width)\n );\n --sbb-calendar-cell-transition-duration: var(\n --sbb-disable-animation-duration,\n var(--sbb-animation-duration-2x)\n );\n}\n@media (min-width: calc(37.5rem)) {\n :host {\n --sbb-calendar-control-view-change-height: 3rem;\n }\n}\n\n:host(:is(:state(enhanced),[state--enhanced])) {\n --sbb-calendar-cell-padding-block-end: var(--sbb-spacing-fixed-3x);\n}\n\n.sbb-calendar__wrapper {\n width: 100%;\n display: block;\n transition-duration: var(--sbb-calendar-cell-transition-duration);\n}\n\n.sbb-calendar__controls {\n width: 100%;\n display: inline-flex;\n align-items: center;\n gap: var(--sbb-calendar-control-gap);\n margin-block-end: var(--sbb-calendar-control-margin-block-end);\n}\n\n.sbb-calendar__controls-month {\n width: 100%;\n display: flex;\n gap: var(--sbb-calendar-tables-gap);\n}\n\n#sbb-calendar__controls-previous,\n#sbb-calendar__controls-next {\n -webkit-tap-highlight-color: transparent;\n}\n\n.sbb-calendar__controls-change-date {\n appearance: none;\n box-sizing: border-box;\n margin: 0;\n outline: none;\n border: none;\n border-radius: 0;\n padding: 0;\n background: none;\n -webkit-tap-highlight-color: transparent;\n color: inherit;\n display: flex;\n align-items: center;\n margin: auto;\n height: var(--sbb-calendar-control-view-change-height);\n font-size: var(--sbb-text-font-size-s);\n letter-spacing: var(--sbb-typo-letter-spacing-text);\n line-height: var(--sbb-typo-letter-spacing-text);\n text-transform: capitalize;\n cursor: var(--sbb-cursor-pointer);\n padding-inline: var(--sbb-calendar-control-view-change-padding-inline);\n border-radius: var(--sbb-border-radius-infinity);\n background-color: var(--sbb-calendar-control-view-change-background);\n color: var(--sbb-calendar-control-view-change-color);\n transition-duration: var(--sbb-calendar-cell-transition-duration);\n transition-timing-function: var(--sbb-calendar-cell-transition-easing-function);\n transition-property: background-color, padding-block-end;\n}\n.sbb-calendar__controls-change-date:disabled {\n --sbb-calendar-control-view-change-background: var(--sbb-color-milk);\n --sbb-calendar-control-view-change-background: light-dark(\n var(--sbb-color-milk),\n var(--sbb-color-anthracite)\n );\n --sbb-calendar-control-view-change-color: var(--sbb-color-granite);\n --sbb-calendar-control-view-change-color: light-dark(\n var(--sbb-color-granite),\n var(--sbb-color-aluminium)\n );\n cursor: unset;\n}\n.sbb-calendar__controls-change-date:focus-visible {\n outline-offset: var(--sbb-focus-outline-offset);\n outline: var(--sbb-focus-outline-color) var(--sbb-focus-outline-style, solid) var(--sbb-focus-outline-width);\n outline-offset: var(--sbb-spacing-fixed-1x);\n}\n@media (any-hover: hover) {\n .sbb-calendar__controls-change-date:not(:active, :disabled):hover {\n padding-block-end: var(--sbb-calendar-cell-hover-shift);\n }\n}\n.sbb-calendar__controls-change-date:not(:disabled):active {\n --sbb-calendar-control-view-change-background: var(--sbb-background-color-3);\n}\n\n.sbb-calendar__table-month-view,\n.sbb-calendar__table-year-view {\n --sbb-calendar-table-column-spaces: 6;\n}\n\n.sbb-calendar__table-overflow-break {\n contain: layout;\n}\n\n.sbb-calendar__table-container {\n display: flex;\n gap: var(--sbb-calendar-tables-gap);\n margin-inline: var(--sbb-calendar-margin);\n --sbb-calendar-min-width: calc(7 * var(--sbb-calendar-cell-size));\n --sbb-calendar-overflow: calc(100% - var(--sbb-calendar-min-width));\n --sbb-calendar-start-offset: min(\n 0px,\n -1 * (var(--sbb-calendar-overflow) / var(--sbb-calendar-table-column-spaces))\n );\n --sbb-calendar-margin: var(--sbb-calendar-start-offset);\n}\n:host(:is(:state(wide),[state--wide])) .sbb-calendar__table-container {\n --sbb-calendar-min-width: calc(\n 2 * 7 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)\n );\n --sbb-calendar-margin: calc(0.5 * var(--sbb-calendar-start-offset));\n}\n:host([orientation=horizontal][week-numbers]) .sbb-calendar__table-container {\n --sbb-calendar-min-width: calc(8 * var(--sbb-calendar-cell-size));\n}\n:host([orientation=horizontal][week-numbers]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {\n --sbb-calendar-min-width: calc(\n 2 * 8 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)\n );\n}\n:host([orientation=vertical]) .sbb-calendar__table-container {\n min-width: var(--sbb-calendar-min-width);\n --sbb-calendar-start-offset: 0;\n}\n:host([orientation=vertical]:is(:state(wide),[state--wide])) .sbb-calendar__table-container {\n --sbb-calendar-min-width: calc(\n 13 * var(--sbb-calendar-cell-size) + var(--sbb-calendar-tables-gap)\n );\n}\n\n.sbb-calendar__table {\n width: 100%;\n border-collapse: collapse;\n height: max-content;\n animation-name: show;\n animation-duration: var(--sbb-calendar-table-animation-duration);\n}\n.sbb-calendar__table.sbb-calendar__table-hide {\n --sbb-calendar-cell-transition-duration: 0ms;\n animation-name: hide;\n animation-duration: var(--sbb-calendar-table-animation-duration);\n}\n:host(:not(:is(:state(wide),[state--wide]))) .sbb-calendar__table {\n min-width: 100%;\n}\n\n.sbb-calendar__table-header {\n text-align: center;\n}\n\n.sbb-calendar__table-body {\n text-align: center;\n}\n\n.sbb-calendar__table-header-cell,\n.sbb-calendar__table-header-cell-vertical {\n width: var(--sbb-calendar-cell-size);\n color: var(--sbb-calendar-header-color);\n padding: 0;\n font-size: var(--sbb-text-font-size-xs);\n font-weight: normal;\n letter-spacing: var(--sbb-typo-letter-spacing-text);\n}\n:host(:not([multiple])) .sbb-calendar__table-header-cell,\n:host(:not([multiple])) .sbb-calendar__table-header-cell-vertical {\n height: var(--sbb-calendar-cell-size);\n}\n\n.sbb-calendar__table-header-cell-vertical {\n padding-block-end: var(--sbb-calendar-cell-padding-block-end);\n}\n\n.sbb-calendar__table-data {\n position: relative;\n padding: 0;\n text-align: center;\n}\n\n.sbb-calendar__day-cell {\n padding-block-end: var(--sbb-calendar-cell-padding-block-end);\n}\n\n:host(:not(:is(:state(enhanced),[state--enhanced]))) sbb-calendar-day {\n --sbb-calendar-cell-justify-content: center;\n --sbb-calendar-day-height: 2.75rem;\n --sbb-calendar-day-extra-display: none;\n --sbb-calendar-day-value-height: unset;\n --sbb-calendar-day-crossed-out-top: 50%;\n --sbb-calendar-day-crossed-out-translate: translate(-50%, -50%) rotate(-45deg);\n}\n\n@keyframes show {\n from {\n opacity: 0;\n transform: translateY(var(--sbb-calendar-table-animation-shift));\n }\n to {\n opacity: 1;\n transform: translateY(0%);\n }\n}\n@keyframes hide {\n from {\n opacity: 1;\n transform: translateY(0%);\n }\n to {\n opacity: 0;\n transform: translateY(var(--sbb-calendar-table-animation-shift));\n }\n}";
14
- //#endregion
15
- //#region src/elements/calendar/calendar/calendar.component.ts
16
- var SbbMonthChangeEvent = class extends Event {
17
- get range() {
18
- return this._range;
19
- }
20
- constructor(range) {
21
- super("monthchange", {
22
- bubbles: true,
23
- composed: true
24
- });
25
- this._range = Object.freeze(range || []);
26
- }
27
- };
28
- /**
29
- * It displays a calendar which allows choosing a date.
30
- *
31
- * @slot - Use the unnamed slot to add customized `sbb-calendar-day` elements.
32
- */
33
- var SbbCalendarElement = (() => {
34
- let _classSuper = SbbElement;
35
- let _instanceExtraInitializers = [];
36
- let _wide_decorators;
37
- let _wide_initializers = [];
38
- let _wide_extraInitializers = [];
39
- let _view_decorators;
40
- let _view_initializers = [];
41
- let _view_extraInitializers = [];
42
- let _min_decorators;
43
- let _min_initializers = [];
44
- let _min_extraInitializers = [];
45
- let _max_decorators;
46
- let _max_initializers = [];
47
- let _max_extraInitializers = [];
48
- let _multiple_decorators;
49
- let _multiple_initializers = [];
50
- let _multiple_extraInitializers = [];
51
- let _set_selected_decorators;
52
- let __selected_decorators;
53
- let __selected_initializers = [];
54
- let __selected_extraInitializers = [];
55
- let _dateFilter_decorators;
56
- let _dateFilter_initializers = [];
57
- let _dateFilter_extraInitializers = [];
58
- let _orientation_decorators;
59
- let _orientation_initializers = [];
60
- let _orientation_extraInitializers = [];
61
- let _weekNumbers_decorators;
62
- let _weekNumbers_initializers = [];
63
- let _weekNumbers_extraInitializers = [];
64
- let __activeDate_decorators;
65
- let __activeDate_initializers = [];
66
- let __activeDate_extraInitializers = [];
67
- let _set__wide_decorators;
68
- let __calendarView_decorators;
69
- let __calendarView_initializers = [];
70
- let __calendarView_extraInitializers = [];
71
- let __initialized_decorators;
72
- let __initialized_initializers = [];
73
- let __initialized_extraInitializers = [];
74
- return class SbbCalendarElement extends _classSuper {
75
- static {
76
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
77
- _wide_decorators = [forceType(), property({ type: Boolean })];
78
- _view_decorators = [property()];
79
- _min_decorators = [plainDate(), property()];
80
- _max_decorators = [plainDate(), property()];
81
- _multiple_decorators = [
82
- forceType(),
83
- handleDistinctChange((e, newValue) => e._onMultipleChanged(newValue)),
84
- property({ type: Boolean })
85
- ];
86
- _set_selected_decorators = [property()];
87
- __selected_decorators = [state()];
88
- _dateFilter_decorators = [property({ attribute: "date-filter" })];
89
- _orientation_decorators = [property({ reflect: true })];
90
- _weekNumbers_decorators = [forceType(), property({
91
- attribute: "week-numbers",
92
- type: Boolean
93
- })];
94
- __activeDate_decorators = [state()];
95
- _set__wide_decorators = [state()];
96
- __calendarView_decorators = [state()];
97
- __initialized_decorators = [state()];
98
- __esDecorate(this, null, _wide_decorators, {
99
- kind: "accessor",
100
- name: "wide",
101
- static: false,
102
- private: false,
103
- access: {
104
- has: (obj) => "wide" in obj,
105
- get: (obj) => obj.wide,
106
- set: (obj, value) => {
107
- obj.wide = value;
108
- }
109
- },
110
- metadata: _metadata
111
- }, _wide_initializers, _wide_extraInitializers);
112
- __esDecorate(this, null, _view_decorators, {
113
- kind: "accessor",
114
- name: "view",
115
- static: false,
116
- private: false,
117
- access: {
118
- has: (obj) => "view" in obj,
119
- get: (obj) => obj.view,
120
- set: (obj, value) => {
121
- obj.view = value;
122
- }
123
- },
124
- metadata: _metadata
125
- }, _view_initializers, _view_extraInitializers);
126
- __esDecorate(this, null, _min_decorators, {
127
- kind: "accessor",
128
- name: "min",
129
- static: false,
130
- private: false,
131
- access: {
132
- has: (obj) => "min" in obj,
133
- get: (obj) => obj.min,
134
- set: (obj, value) => {
135
- obj.min = value;
136
- }
137
- },
138
- metadata: _metadata
139
- }, _min_initializers, _min_extraInitializers);
140
- __esDecorate(this, null, _max_decorators, {
141
- kind: "accessor",
142
- name: "max",
143
- static: false,
144
- private: false,
145
- access: {
146
- has: (obj) => "max" in obj,
147
- get: (obj) => obj.max,
148
- set: (obj, value) => {
149
- obj.max = value;
150
- }
151
- },
152
- metadata: _metadata
153
- }, _max_initializers, _max_extraInitializers);
154
- __esDecorate(this, null, _multiple_decorators, {
155
- kind: "accessor",
156
- name: "multiple",
157
- static: false,
158
- private: false,
159
- access: {
160
- has: (obj) => "multiple" in obj,
161
- get: (obj) => obj.multiple,
162
- set: (obj, value) => {
163
- obj.multiple = value;
164
- }
165
- },
166
- metadata: _metadata
167
- }, _multiple_initializers, _multiple_extraInitializers);
168
- __esDecorate(this, null, _set_selected_decorators, {
169
- kind: "setter",
170
- name: "selected",
171
- static: false,
172
- private: false,
173
- access: {
174
- has: (obj) => "selected" in obj,
175
- set: (obj, value) => {
176
- obj.selected = value;
177
- }
178
- },
179
- metadata: _metadata
180
- }, null, _instanceExtraInitializers);
181
- __esDecorate(this, null, __selected_decorators, {
182
- kind: "accessor",
183
- name: "_selected",
184
- static: false,
185
- private: false,
186
- access: {
187
- has: (obj) => "_selected" in obj,
188
- get: (obj) => obj._selected,
189
- set: (obj, value) => {
190
- obj._selected = value;
191
- }
192
- },
193
- metadata: _metadata
194
- }, __selected_initializers, __selected_extraInitializers);
195
- __esDecorate(this, null, _dateFilter_decorators, {
196
- kind: "accessor",
197
- name: "dateFilter",
198
- static: false,
199
- private: false,
200
- access: {
201
- has: (obj) => "dateFilter" in obj,
202
- get: (obj) => obj.dateFilter,
203
- set: (obj, value) => {
204
- obj.dateFilter = value;
205
- }
206
- },
207
- metadata: _metadata
208
- }, _dateFilter_initializers, _dateFilter_extraInitializers);
209
- __esDecorate(this, null, _orientation_decorators, {
210
- kind: "accessor",
211
- name: "orientation",
212
- static: false,
213
- private: false,
214
- access: {
215
- has: (obj) => "orientation" in obj,
216
- get: (obj) => obj.orientation,
217
- set: (obj, value) => {
218
- obj.orientation = value;
219
- }
220
- },
221
- metadata: _metadata
222
- }, _orientation_initializers, _orientation_extraInitializers);
223
- __esDecorate(this, null, _weekNumbers_decorators, {
224
- kind: "accessor",
225
- name: "weekNumbers",
226
- static: false,
227
- private: false,
228
- access: {
229
- has: (obj) => "weekNumbers" in obj,
230
- get: (obj) => obj.weekNumbers,
231
- set: (obj, value) => {
232
- obj.weekNumbers = value;
233
- }
234
- },
235
- metadata: _metadata
236
- }, _weekNumbers_initializers, _weekNumbers_extraInitializers);
237
- __esDecorate(this, null, __activeDate_decorators, {
238
- kind: "accessor",
239
- name: "_activeDate",
240
- static: false,
241
- private: false,
242
- access: {
243
- has: (obj) => "_activeDate" in obj,
244
- get: (obj) => obj._activeDate,
245
- set: (obj, value) => {
246
- obj._activeDate = value;
247
- }
248
- },
249
- metadata: _metadata
250
- }, __activeDate_initializers, __activeDate_extraInitializers);
251
- __esDecorate(this, null, _set__wide_decorators, {
252
- kind: "setter",
253
- name: "_wide",
254
- static: false,
255
- private: false,
256
- access: {
257
- has: (obj) => "_wide" in obj,
258
- set: (obj, value) => {
259
- obj._wide = value;
260
- }
261
- },
262
- metadata: _metadata
263
- }, null, _instanceExtraInitializers);
264
- __esDecorate(this, null, __calendarView_decorators, {
265
- kind: "accessor",
266
- name: "_calendarView",
267
- static: false,
268
- private: false,
269
- access: {
270
- has: (obj) => "_calendarView" in obj,
271
- get: (obj) => obj._calendarView,
272
- set: (obj, value) => {
273
- obj._calendarView = value;
274
- }
275
- },
276
- metadata: _metadata
277
- }, __calendarView_initializers, __calendarView_extraInitializers);
278
- __esDecorate(this, null, __initialized_decorators, {
279
- kind: "accessor",
280
- name: "_initialized",
281
- static: false,
282
- private: false,
283
- access: {
284
- has: (obj) => "_initialized" in obj,
285
- get: (obj) => obj._initialized,
286
- set: (obj, value) => {
287
- obj._initialized = value;
288
- }
289
- },
290
- metadata: _metadata
291
- }, __initialized_initializers, __initialized_extraInitializers);
292
- if (_metadata) Object.defineProperty(this, Symbol.metadata, {
293
- enumerable: true,
294
- configurable: true,
295
- writable: true,
296
- value: _metadata
297
- });
298
- }
299
- static {
300
- this.elementName = "sbb-calendar";
301
- }
302
- static {
303
- this.elementDependencies = [
304
- SbbCalendarDayElement,
305
- SbbCalendarMonthElement,
306
- SbbCalendarWeekdayElement,
307
- SbbCalendarWeeknumberElement,
308
- SbbCalendarYearElement,
309
- SbbIconElement,
310
- SbbSecondaryButtonElement
311
- ];
312
- }
313
- static {
314
- this.styles = [screenReaderOnlyStyles, unsafeCSS(calendar_default)];
315
- }
316
- static {
317
- this.events = {
318
- dateselected: "dateselected",
319
- monthchange: "monthchange"
320
- };
321
- }
322
- #wide_accessor_storage;
323
- /** If set to true, two months are displayed */
324
- get wide() {
325
- return this.#wide_accessor_storage;
326
- }
327
- set wide(value) {
328
- this.#wide_accessor_storage = value;
329
- }
330
- #view_accessor_storage;
331
- /** The initial view of the calendar which should be displayed on opening. */
332
- get view() {
333
- return this.#view_accessor_storage;
334
- }
335
- set view(value) {
336
- this.#view_accessor_storage = value;
337
- }
338
- #min_accessor_storage;
339
- /**
340
- * The minimum valid date. Accepts a date object or null.
341
- * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
342
- */
343
- get min() {
344
- return this.#min_accessor_storage;
345
- }
346
- set min(value) {
347
- this.#min_accessor_storage = value;
348
- }
349
- #max_accessor_storage;
350
- /**
351
- * The maximum valid date. Accepts a date object or null.
352
- * Accepts an ISO8601 formatted string (e.g. 2024-12-24) as attribute.
353
- */
354
- get max() {
355
- return this.#max_accessor_storage;
356
- }
357
- set max(value) {
358
- this.#max_accessor_storage = value;
359
- }
360
- #multiple_accessor_storage;
361
- /** Whether the calendar allows for multiple date selection. */
362
- get multiple() {
363
- return this.#multiple_accessor_storage;
364
- }
365
- set multiple(value) {
366
- this.#multiple_accessor_storage = value;
367
- }
368
- /**
369
- * The selected date: accepts a date object, or, if `multiple`, an array of dates.
370
- */
371
- set selected(value) {
372
- if (Array.isArray(value)) this._selected = value.map((dateLike) => this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(dateLike))).filter((date) => date !== null).filter((date) => !this._isDayInRange(this._dateAdapter.toIso8601(date)) || this._dateFilter(date));
373
- else {
374
- const selectedDate = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(value));
375
- if (!!selectedDate && (!this._isDayInRange(this._dateAdapter.toIso8601(selectedDate)) || this._dateFilter(selectedDate))) this._selected = selectedDate;
376
- else this._selected = null;
377
- }
378
- }
379
- get selected() {
380
- return this._selected;
381
- }
382
- #_selected_accessor_storage;
383
- get _selected() {
384
- return this.#_selected_accessor_storage;
385
- }
386
- set _selected(value) {
387
- this.#_selected_accessor_storage = value;
388
- }
389
- #dateFilter_accessor_storage;
390
- /** A function used to filter out dates. */
391
- get dateFilter() {
392
- return this.#dateFilter_accessor_storage;
393
- }
394
- set dateFilter(value) {
395
- this.#dateFilter_accessor_storage = value;
396
- }
397
- #orientation_accessor_storage;
398
- /** The orientation of days in the calendar. */
399
- get orientation() {
400
- return this.#orientation_accessor_storage;
401
- }
402
- set orientation(value) {
403
- this.#orientation_accessor_storage = value;
404
- }
405
- #weekNumbers_accessor_storage;
406
- /** Whether it has to display the week numbers in addition to week days. */
407
- get weekNumbers() {
408
- return this.#weekNumbers_accessor_storage;
409
- }
410
- set weekNumbers(value) {
411
- this.#weekNumbers_accessor_storage = value;
412
- }
413
- #_activeDate_accessor_storage;
414
- /** The currently active date. */
415
- get _activeDate() {
416
- return this.#_activeDate_accessor_storage;
417
- }
418
- set _activeDate(value) {
419
- this.#_activeDate_accessor_storage = value;
420
- }
421
- /** The current wide property considering property value and breakpoints. From zero to small `wide` has always to be false. */
422
- set _wide(wide) {
423
- this.toggleState("wide", wide);
424
- this._wideInternal = wide;
425
- }
426
- get _wide() {
427
- return this._wideInternal;
428
- }
429
- #_calendarView_accessor_storage;
430
- get _calendarView() {
431
- return this.#_calendarView_accessor_storage;
432
- }
433
- set _calendarView(value) {
434
- this.#_calendarView_accessor_storage = value;
435
- }
436
- /** A list of calendar's cells corresponding to days, months or years depending on the view. */
437
- get _cells() {
438
- return Array.from((this._calendarView === "day" ? Array.from(this.shadowRoot.querySelectorAll("slot")).flatMap((e) => e.assignedElements({ flatten: true })) : this.shadowRoot?.querySelectorAll(`sbb-calendar-${this._calendarView}`)) ?? []);
439
- }
440
- #_initialized_accessor_storage;
441
- get _initialized() {
442
- return this.#_initialized_accessor_storage;
443
- }
444
- set _initialized(value) {
445
- this.#_initialized_accessor_storage = value;
446
- }
447
- constructor() {
448
- super();
449
- this.#wide_accessor_storage = (__runInitializers(this, _instanceExtraInitializers), __runInitializers(this, _wide_initializers, false));
450
- this.#view_accessor_storage = (__runInitializers(this, _wide_extraInitializers), __runInitializers(this, _view_initializers, "day"));
451
- this.#min_accessor_storage = (__runInitializers(this, _view_extraInitializers), __runInitializers(this, _min_initializers, null));
452
- this.#max_accessor_storage = (__runInitializers(this, _min_extraInitializers), __runInitializers(this, _max_initializers, null));
453
- this.#multiple_accessor_storage = (__runInitializers(this, _max_extraInitializers), __runInitializers(this, _multiple_initializers, false));
454
- this.#_selected_accessor_storage = (__runInitializers(this, _multiple_extraInitializers), __runInitializers(this, __selected_initializers, null));
455
- this.#dateFilter_accessor_storage = (__runInitializers(this, __selected_extraInitializers), __runInitializers(this, _dateFilter_initializers, null));
456
- this.#orientation_accessor_storage = (__runInitializers(this, _dateFilter_extraInitializers), __runInitializers(this, _orientation_initializers, "horizontal"));
457
- this.#weekNumbers_accessor_storage = (__runInitializers(this, _orientation_extraInitializers), __runInitializers(this, _weekNumbers_initializers, false));
458
- this._dateAdapter = (__runInitializers(this, _weekNumbers_extraInitializers), readConfig().datetime?.dateAdapter ?? defaultDateAdapter);
459
- this.#_activeDate_accessor_storage = __runInitializers(this, __activeDate_initializers, this._dateAdapter.today());
460
- this._wideInternal = (__runInitializers(this, __activeDate_extraInitializers), false);
461
- this.#_calendarView_accessor_storage = __runInitializers(this, __calendarView_initializers, "day");
462
- this._nextCalendarView = (__runInitializers(this, __calendarView_extraInitializers), "day");
463
- /** Information about the rendered day view; used in keyboard navigation. */
464
- this._keyboardNavigationDayViewParameters = {
465
- firstDayInView: null,
466
- lastDayInView: null,
467
- firstMonthOffset: 0,
468
- firstMonthLength: 0,
469
- secondMonthOffset: 0
470
- };
471
- /** Grid of calendar cells representing the dates of the month. */
472
- this._weeks = [];
473
- /** An array containing all the month names in the current language. */
474
- this._monthNames = this._dateAdapter.getMonthNames("long");
475
- this._enhancedVariant = false;
476
- /** Whether the focus should be reset on focusCell. */
477
- this._resetFocus = false;
478
- /** Whether an element inside the calendar is currently focused. */
479
- this._containingFocus = false;
480
- this.#_initialized_accessor_storage = __runInitializers(this, __initialized_initializers, false);
481
- this._language = (__runInitializers(this, __initialized_extraInitializers), new SbbLanguageController(this).withHandler(() => {
482
- this._monthNames = this._dateAdapter.getMonthNames("long");
483
- this._createMonthRows();
484
- }));
485
- this._mediaMatcher = new SbbMediaMatcherController(this, { [SbbMediaQueryBreakpointLargeAndAbove]: () => this._init() });
486
- this._onSlotChange = () => {
487
- this._enhancedVariant = Array.from(this.children).some((c) => c.localName === "sbb-calendar-day");
488
- this.toggleState("enhanced", this._enhancedVariant);
489
- this._setTabIndex();
490
- };
491
- this._createMonthRows();
492
- this._setWeekdays();
493
- this.addEventListener("focusin", () => this._containingFocus = true);
494
- this.addEventListener("focusout", () => this._containingFocus = false);
495
- this.addEventListener("click", (e) => {
496
- const day = e.target.closest("sbb-calendar-day");
497
- if (day) this._selectDate(day.value);
498
- });
499
- this.addEventListener("keydown", (e) => {
500
- if (e.target.localName === "sbb-calendar-day") this._handleKeyboardEvent(e, this._mapDateToDay(e.target.value));
501
- });
502
- }
503
- _dateFilter(date) {
504
- return this.dateFilter?.(date) ?? true;
505
- }
506
- /** Resets the active month according to the new state of the calendar. */
507
- resetPosition() {
508
- this._resetCalendarView();
509
- this._init();
510
- }
511
- connectedCallback() {
512
- super.connectedCallback();
513
- this.resetPosition();
514
- this.shadowRoot?.addEventListener("slotchange", this._onSlotChange, { capture: true });
515
- }
516
- disconnectedCallback() {
517
- super.disconnectedCallback();
518
- this.shadowRoot?.removeEventListener("slotchange", this._onSlotChange, { capture: true });
519
- }
520
- /** @internal */
521
- focus() {
522
- this._resetFocus = true;
523
- this._focusCell();
524
- }
525
- willUpdate(changedProperties) {
526
- super.willUpdate(changedProperties);
527
- if (!this._initialized) return;
528
- if (changedProperties.has("wide") || changedProperties.has("orientation")) this.resetPosition();
529
- if (changedProperties.has("view")) {
530
- this._setChosenYear();
531
- this._chosenMonth = void 0;
532
- this._nextCalendarView = this._calendarView = this.view;
533
- }
534
- }
535
- updated(changedProperties) {
536
- super.updated(changedProperties);
537
- this._setTabIndex();
538
- this._focusCell();
539
- }
540
- /**
541
- * The `_selected` state should be adapted when the `multiple` property changes:
542
- * - if it changes to true, the '_selected' is set to an array;
543
- * - if it changes to false, the first available option is set as 'value' otherwise it's set to null.
544
- */
545
- _onMultipleChanged(isMultiple) {
546
- if (isMultiple && !Array.isArray(this._selected)) this._selected = this._selected ? [this._selected] : [];
547
- if (!isMultiple && Array.isArray(this._selected)) this._selected = this._selected.length ? this._selected[0] : null;
548
- }
549
- /** Initializes the component. */
550
- _init(activeDate) {
551
- if (isServer) return;
552
- else if (this.hydrationRequired) {
553
- this.hydrationComplete.then(() => this._init());
554
- return;
555
- }
556
- if (activeDate) this._assignActiveDate(activeDate);
557
- this._wide = (this._mediaMatcher.matches(SbbMediaQueryBreakpointLargeAndAbove) ?? false) && this.wide;
558
- this._weeks = this._createWeekRows(this._activeDate);
559
- this._years = this._createYearRows();
560
- this._weekNumbers = this._weeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
561
- this._nextMonthWeeks = [[]];
562
- this._nextMonthYears = [[]];
563
- if (this._wide) {
564
- const nextMonthDate = this._dateAdapter.addCalendarMonths(this._activeDate, 1);
565
- this._nextMonthWeeks = this._createWeekRows(nextMonthDate, true);
566
- this._nextMonthYears = this._createYearRows(YEARS_PER_PAGE);
567
- this._nextMonthWeekNumbers = this._nextMonthWeeks.flat().sort((a, b) => a.value.localeCompare(b.value)).map((day) => day.weekValue).filter((v, i, a) => a.indexOf(v) === i);
568
- }
569
- this._initialized = true;
570
- }
571
- /** Focuses on a day cell prioritizing the selected day, the current day, and lastly, the first selectable day. */
572
- _focusCell() {
573
- if (this._resetFocus) {
574
- this._getFirstFocusable()?.focus();
575
- this._resetFocus = false;
576
- }
577
- }
578
- /** Creates the array of weekdays. */
579
- _setWeekdays() {
580
- const narrowWeekdays = this._dateAdapter.getDayOfWeekNames("narrow");
581
- const weekdays = this._dateAdapter.getDayOfWeekNames("long").map((long, i) => ({
582
- long,
583
- narrow: narrowWeekdays[i]
584
- }));
585
- const firstDayOfWeek = this._dateAdapter.getFirstDayOfWeek();
586
- this._weekdays = weekdays.slice(firstDayOfWeek).concat(weekdays.slice(0, firstDayOfWeek));
587
- }
588
- /** Creates the rows along the horizontal direction and sets the parameters used in keyboard navigation. */
589
- _createWeekRows(value, isSecondMonthInView = false) {
590
- const daysInMonth = this._dateAdapter.getNumDaysInMonth(value);
591
- const weekOffset = this._dateAdapter.getFirstWeekOffset(value);
592
- if (!isSecondMonthInView) {
593
- this._keyboardNavigationDayViewParameters.firstMonthLength = daysInMonth;
594
- this._keyboardNavigationDayViewParameters.firstMonthOffset = weekOffset;
595
- this._keyboardNavigationDayViewParameters.firstDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), 1));
596
- this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
597
- } else {
598
- this._keyboardNavigationDayViewParameters.secondMonthOffset = weekOffset;
599
- this._keyboardNavigationDayViewParameters.lastDayInView = this._dateAdapter.toIso8601(this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), daysInMonth));
600
- }
601
- return this.orientation === "horizontal" ? this._createWeekRowsHorizontal(value, daysInMonth, weekOffset) : this._createWeekRowsVertical(value, daysInMonth, weekOffset);
602
- }
603
- /**
604
- * Creates the rows for each week in orientation='horizontal'.
605
- *
606
- * Iterates through the days of the months, creates a Day object for each and pushes it into and array.
607
- * Each seven days (considering the offset at the beginning of the month) restarts from an empty array.
608
- *
609
- * The result is a matrix in which every row is a week (or part of it, considering offset).
610
- */
611
- _createWeekRowsHorizontal(value, daysInMonth, weekOffset) {
612
- const weeks = [[]];
613
- for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
614
- if (cell === DAYS_PER_ROW) {
615
- weeks.push([]);
616
- cell = 0;
617
- }
618
- const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
619
- weeks[weeks.length - 1].push(this._mapDateToDay(date));
620
- }
621
- return weeks;
622
- }
623
- /**
624
- * Creates the rows for each week in orientation='vertical'.
625
- *
626
- * Creates a matrix with seven empty rows.
627
- * Iterates through the days of the months, creates a Day object for each
628
- * and pushes it into the correct array considering the offset at the beginning of the month.
629
- * Each seven days (including offset) restarts from the first.
630
- *
631
- * The result is a matrix in which every row is a set of weekdays, so:
632
- * - row 0: all the Mondays;
633
- * - row 1: all the Tuesdays;
634
- * - ...
635
- * - row 7: all the Sundays.
636
- */
637
- _createWeekRowsVertical(value, daysInMonth, weekOffset) {
638
- const weeks = Array.from({ length: DAYS_PER_ROW }, () => []);
639
- for (let i = 0, cell = weekOffset; i < daysInMonth; i++, cell++) {
640
- if (cell === DAYS_PER_ROW) cell = 0;
641
- const date = this._dateAdapter.createDate(this._dateAdapter.getYear(value), this._dateAdapter.getMonth(value), i + 1);
642
- weeks[cell].push(this._mapDateToDay(date));
643
- }
644
- return weeks;
645
- }
646
- _mapDateToDay(date) {
647
- return {
648
- value: this._dateAdapter.toIso8601(date),
649
- dateValue: date,
650
- dayValue: String(this._dateAdapter.getDate(date)),
651
- monthValue: String(this._dateAdapter.getMonth(date)),
652
- yearValue: String(this._dateAdapter.getYear(date)),
653
- weekValue: this._getWeek(date),
654
- weekDayValue: this._dateAdapter.getDayOfWeek(date)
655
- };
656
- }
657
- _getWeek(date) {
658
- const firstDayOfYear = this._dateAdapter.createDate(this._dateAdapter.getYear(date), 1, 1);
659
- const weekday = this._dateAdapter.getDayOfWeek(firstDayOfYear);
660
- let weekIndex = [
661
- MONDAY,
662
- TUESDAY,
663
- WEDNESDAY,
664
- THURSDAY
665
- ].includes(weekday) ? 1 : 0;
666
- let weekStart = this._dateAdapter.addCalendarDays(firstDayOfYear, this._dateAdapter.getFirstWeekOffset(firstDayOfYear) * -1);
667
- while (this._dateAdapter.compareDate(weekStart, date) <= 0) {
668
- const weekEnd = this._dateAdapter.addCalendarDays(weekStart, 6);
669
- if (this._dateAdapter.compareDate(date, weekEnd) <= 0) return weekIndex > 0 ? weekIndex : this._getWeek(this._dateAdapter.addCalendarDays(firstDayOfYear, -1));
670
- weekStart = this._dateAdapter.addCalendarDays(weekStart, DAYS_PER_ROW);
671
- weekIndex++;
672
- }
673
- throw new Error("The provided date is invalid");
674
- }
675
- /** Force the conversion to ISO8601 formatted string. */
676
- _mapValueToISODate(value) {
677
- return typeof value === "string" ? value : this._dateAdapter.toIso8601(value);
678
- }
679
- /** Creates the rows for the month selection view. */
680
- _createMonthRows() {
681
- const months = new Array(12).fill(null).map((_, i) => ({
682
- value: String(i + 1).padStart(2, "0"),
683
- monthValue: i + 1
684
- }));
685
- const rows = 12 / MONTHS_PER_ROW;
686
- const monthArray = [];
687
- for (let i = 0; i < rows; i++) monthArray.push(months.slice(MONTHS_PER_ROW * i, MONTHS_PER_ROW * (i + 1)));
688
- this._months = monthArray;
689
- }
690
- /** Creates the rows for the year selection view. */
691
- _createYearRows(offset = 0) {
692
- const startValueYearView = this._getStartValueYearView();
693
- const allYears = new Array(YEARS_PER_PAGE).fill(0).map((_, i) => startValueYearView + offset + i);
694
- const rows = YEARS_PER_PAGE / YEARS_PER_ROW;
695
- const yearArray = [];
696
- for (let i = 0; i < rows; i++) yearArray.push(allYears.slice(YEARS_PER_ROW * i, YEARS_PER_ROW * (i + 1)));
697
- return yearArray;
698
- }
699
- /**
700
- * Calculates the first year that will be shown in the year selection panel.
701
- * If `minDate` and `maxDate` are both null, the starting year is calculated as
702
- * the multiple of YEARS_PER_PAGE closest to and less than activeDate,
703
- * e.g., with `YEARS_PER_PAGE` = 24 and `activeDate` = 2020, the function will return 2016 (24 * 83),
704
- * while with `activeDate` = 2000, the function will return 1992 (24 * 82).
705
- * If `minDate` is not null, it returns the corresponding year; if `maxDate` is not null,
706
- * it returns the corresponding year minus `YEARS_PER_PAGE`, so that the `maxDate` is the last rendered year.
707
- * If both are not null, `maxDate` has priority over `minDate`.
708
- */
709
- _getStartValueYearView() {
710
- let startingYear = 0;
711
- if (this.max) startingYear = this._dateAdapter.getYear(this.max) - YEARS_PER_PAGE + 1;
712
- else if (this.min) startingYear = this._dateAdapter.getYear(this.min);
713
- const activeYear = this._dateAdapter.getYear(this._activeDate);
714
- return activeYear - ((activeYear - startingYear) % YEARS_PER_PAGE + YEARS_PER_PAGE) % YEARS_PER_PAGE;
715
- }
716
- /** Checks if date is within the min-max range. */
717
- _isDayInRange(dateString) {
718
- if (!this.min && !this.max) return true;
719
- const date = this._dateAdapter.deserialize(dateString);
720
- return this._dateAdapter.sameDate(date, this._dateAdapter.clampDate(date, this.min, this.max));
721
- }
722
- /** Emits the selected date and sets it internally. */
723
- _selectDate(day) {
724
- this._chosenMonth = void 0;
725
- this._setChosenYear();
726
- if (this.multiple) {
727
- if (this._selected && this._selected.length > 0) {
728
- const indexOfSelectedDay = this._selected.findIndex((sel) => this._dateAdapter.compareDate(sel, day) === 0);
729
- if (indexOfSelectedDay !== -1) this._selected = this._selected.filter((_, i) => i !== indexOfSelectedDay);
730
- else this._selected = [...this._selected, day];
731
- } else this._selected = [day];
732
- this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
733
- } else if (!this._selected || this._dateAdapter.compareDate(this._selected, day) !== 0) {
734
- this._selected = day;
735
- this._emitDateSelectedEvent(this._dateAdapter.deserialize(day));
736
- }
737
- }
738
- /**
739
- * Handle multiple dates selection via weekNumber / weekDay buttons:
740
- * - if Cmd or Ctrl are pressed, add the new date to the current ones;
741
- * - if not,
742
- * - if the new dates are the same of the current ones, it means that the same button has been clicked twice, so do nothing;
743
- * - if not, the selected dates are the new ones.
744
- */
745
- _selectMultipleDates(days) {
746
- const enabledDays = this._cells.filter((e) => !e.disabled).map((e) => this._mapValueToISODate(e.value));
747
- const daysToAdd = days.map((e) => e.value).filter((isoDate) => enabledDays.includes(isoDate));
748
- const daysToAddSet = new Set(daysToAdd);
749
- const selectedSet = new Set(this._selected.map((s) => this._dateAdapter.toIso8601(s)));
750
- const selStrings = this._updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet);
751
- this._selected = selStrings.map((s) => this._dateAdapter.deserialize(s));
752
- this._emitDateSelectedEvent(this._selected.map((e) => this._dateAdapter.deserialize(e)));
753
- }
754
- /**
755
- * Emits the dateselected event given the detail (as T or T[] based on the value of the multiple flag).
756
- */
757
- _emitDateSelectedEvent(detail) {
758
- /** @type {CustomEvent<T | T[]>} Event emitted on date selection. */
759
- this.dispatchEvent(new CustomEvent("dateselected", {
760
- detail,
761
- composed: true,
762
- bubbles: true
763
- }));
764
- }
765
- _emitMonthChange() {
766
- const monthchange = (this.wide ? [...this._weeks, ...this._nextMonthWeeks] : this._weeks).flat().sort((a, b) => a.value.localeCompare(b.value));
767
- /**
768
- * @type {SbbMonthChangeEvent}
769
- * Emits when the month changes.
770
- * The `range` property contains the days array of the chosen month.
771
- */
772
- this.dispatchEvent(new SbbMonthChangeEvent(monthchange));
773
- }
774
- /**
775
- * In case of multiple selection, newly added days must be added to the existing ones, without duplication.
776
- * If the days to add are exactly the same as the selected ones, the set must be emptied.
777
- */
778
- _updateSelectedWithMultipleDates(daysToAdd, daysToAddSet, selectedSet) {
779
- if (daysToAdd.every((day) => selectedSet.has(day))) daysToAddSet.forEach((day) => selectedSet.delete(day));
780
- else daysToAddSet.forEach((day) => selectedSet.add(day));
781
- return Array.from(selectedSet);
782
- }
783
- _setChosenYear() {
784
- if (this.view === "month") {
785
- let selectedDate;
786
- if (this.multiple) selectedDate = this.selected.at(-1);
787
- else selectedDate = this.selected;
788
- this._chosenYear = this._dateAdapter.getYear(selectedDate ?? this._dateAdapter.today());
789
- } else this._chosenYear = void 0;
790
- }
791
- _assignActiveDate(date) {
792
- if (this.min && this._dateAdapter.compareDate(this.min, date) > 0) {
793
- this._activeDate = this.min;
794
- return;
795
- }
796
- if (this.max && this._dateAdapter.compareDate(this.max, date) < 0) {
797
- this._activeDate = this.max;
798
- return;
799
- }
800
- this._activeDate = date;
801
- }
802
- /** Goes to the month identified by the shift. */
803
- _goToDifferentMonth(months) {
804
- this._init(this._dateAdapter.addCalendarMonths(this._activeDate, months));
805
- this._emitMonthChange();
806
- }
807
- _goToDifferentYear(years) {
808
- this._chosenYear += years;
809
- this._activeDate = this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate));
810
- this._init();
811
- }
812
- _goToDifferentYearRange(years) {
813
- this._init(this._dateAdapter.addCalendarYears(this._activeDate, years));
814
- }
815
- _prevDisabled(prevDate) {
816
- if (!this.min) return false;
817
- return this._dateAdapter.compareDate(prevDate, this.min) < 0;
818
- }
819
- _nextDisabled(nextDate) {
820
- if (!this.max) return false;
821
- return this._dateAdapter.compareDate(nextDate, this.max) > 0;
822
- }
823
- /** Checks if the "previous month" button should be disabled. */
824
- _previousMonthDisabled() {
825
- const prevMonth = this._dateAdapter.addCalendarDays(this._activeDate, this._dateAdapter.getDate(this._activeDate) * -1);
826
- return this._prevDisabled(prevMonth);
827
- }
828
- /** Checks if the "next month" button should be disabled. */
829
- _nextMonthDisabled() {
830
- let nextMonth = this._dateAdapter.addCalendarMonths(this._activeDate, this._wide ? 2 : 1);
831
- nextMonth = this._dateAdapter.createDate(this._dateAdapter.getYear(nextMonth), this._dateAdapter.getMonth(nextMonth), 1);
832
- return this._nextDisabled(nextMonth);
833
- }
834
- /** Checks if the "previous year" button should be disabled. */
835
- _previousYearDisabled() {
836
- const prevYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) - 1, 12, 31);
837
- return this._prevDisabled(prevYear);
838
- }
839
- /** Checks if the "next year" button should be disabled. */
840
- _nextYearDisabled() {
841
- const nextYear = this._dateAdapter.createDate(this._dateAdapter.getYear(this._activeDate) + (this._wide ? 2 : 1), 1, 1);
842
- return this._nextDisabled(nextYear);
843
- }
844
- /** Checks if the "previous year" button should be disabled in year view. */
845
- _previousYearRangeDisabled() {
846
- const prevYear = this._dateAdapter.createDate(this._years[0][0] - 1, 12, 31);
847
- return this._prevDisabled(prevYear);
848
- }
849
- /** Checks if the "next year" button should be disabled in year view. */
850
- _nextYearRangeDisabled() {
851
- const years = this._wide ? this._nextMonthYears : this._years;
852
- const lastYearRange = years[years.length - 1];
853
- const lastYear = lastYearRange[lastYearRange.length - 1];
854
- const nextYear = this._dateAdapter.createDate(lastYear + 1, 1, 1);
855
- return this._nextDisabled(nextYear);
856
- }
857
- _handleTableBlur(eventTarget) {
858
- if (eventTarget?.localName !== "sbb-calendar-day") this._setTabIndex();
859
- }
860
- _setTabIndex() {
861
- Array.from(this._cells.filter((e) => e.tabIndex === 0) ?? []).forEach((day) => day.tabIndex = -1);
862
- const firstFocusable = this._getFirstFocusable();
863
- if (firstFocusable) firstFocusable.tabIndex = 0;
864
- }
865
- /** Get the element in the calendar to assign focus. */
866
- _getFirstFocusable() {
867
- if (this._calendarView === "day") {
868
- const selectedOrCurrent = this._cells.find((e) => e.matches(":is(:state(selected),[state--selected])")) ?? this._cells.find((e) => e.matches(":is(:state(current),[state--current])"));
869
- return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this._getFirstFocusableDay();
870
- } else {
871
- const selectedOrCurrent = this.shadowRoot?.querySelector(":is(:state(selected),[state--selected])") ?? this.shadowRoot?.querySelector(":is(:state(current),[state--current])");
872
- return selectedOrCurrent && !selectedOrCurrent.disabled ? selectedOrCurrent : this.shadowRoot.querySelector(`sbb-calendar-${this._calendarView}:not([disabled])`);
873
- }
874
- }
875
- /**
876
- * In `day` view in `vertical` orientation,
877
- * if the first of the month is not a Monday, it is not the first rendered element in the table,
878
- * so `this.shadowRoot!.querySelector('sbb-calendar-day:not([disabled])')` will return a wrong value.
879
- *
880
- * To solve this, the element with the lowest `value` is taken (ISO String are ordered).
881
- */
882
- _getFirstFocusableDay() {
883
- const cells = this._cells;
884
- const daysInView = cells.filter((e) => !e.disabled);
885
- if (!daysInView || daysInView.length === 0) return null;
886
- else {
887
- const firstElement = daysInView.map((e) => this._dateAdapter.toIso8601(e.value)).sort()[0];
888
- return cells.find((e) => e.matches(`[slot="${firstElement}"]`)) ?? null;
889
- }
890
- }
891
- _handleKeyboardEvent(event, day) {
892
- if (isArrowKeyOrPageKeysPressed(event)) event.preventDefault();
893
- const cells = this._cells;
894
- const index = cells.findIndex((e) => e === event.target);
895
- let nextEl;
896
- if (day) nextEl = this._navigateByKeyboardDayView(event, index, cells, day);
897
- else nextEl = this._navigateByKeyboard(event, index, cells);
898
- const activeEl = (this._enhancedVariant ? document : this.shadowRoot).activeElement;
899
- if (nextEl !== activeEl) {
900
- nextEl.tabIndex = 0;
901
- nextEl?.focus();
902
- activeEl.tabIndex = -1;
903
- }
904
- }
905
- _navigateByKeyboardDayView(evt, index, cells, day) {
906
- const arrowsOffset = this.orientation === "horizontal" ? {
907
- leftRight: 1,
908
- upDown: DAYS_PER_ROW
909
- } : {
910
- leftRight: DAYS_PER_ROW,
911
- upDown: 1
912
- };
913
- const offsetForVertical = index < this._keyboardNavigationDayViewParameters.firstMonthLength ? this._keyboardNavigationDayViewParameters.firstMonthOffset : this._keyboardNavigationDayViewParameters.secondMonthOffset;
914
- switch (evt.key) {
915
- case "ArrowUp": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.upDown);
916
- case "ArrowDown": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.upDown);
917
- case "ArrowLeft": return this._findDayArrows(cells, index, day.dateValue, -arrowsOffset.leftRight);
918
- case "ArrowRight": return this._findDayArrows(cells, index, day.dateValue, arrowsOffset.leftRight);
919
- case "PageUp": if (this.orientation === "horizontal") {
920
- const delta = (+day.dayValue % DAYS_PER_ROW || DAYS_PER_ROW) - +day.dayValue;
921
- return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
922
- } else {
923
- const delta = (Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) - 1) * DAYS_PER_ROW - offsetForVertical + 1 - +day.dayValue;
924
- return this._findDayPageUpDown(cells, index, day, delta, arrowsOffset.upDown);
925
- }
926
- case "PageDown": if (this.orientation === "horizontal") {
927
- const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
928
- const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
929
- const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
930
- const lastOfMonth = this._dateAdapter.getDate(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
931
- const delta = Math.trunc((lastOfMonth - +day.dayValue) / DAYS_PER_ROW) * DAYS_PER_ROW;
932
- return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
933
- } else {
934
- const delta = Math.ceil((+day.dayValue + offsetForVertical) / DAYS_PER_ROW) * DAYS_PER_ROW - offsetForVertical - +day.dayValue;
935
- return this._findDayPageUpDown(cells, index, day, delta, -arrowsOffset.upDown);
936
- }
937
- case "Home": return this._findDayFirst(cells, index, day, 1);
938
- case "End": {
939
- const monthInBounds = +day.monthValue + 1 > 12 ? 1 : +day.monthValue + 1;
940
- const yearInBounds = +day.monthValue + 1 > 12 ? +day.yearValue + 1 : +day.yearValue;
941
- const firstNextMonth = this._dateAdapter.createDate(yearInBounds, monthInBounds, 1);
942
- return this._findDayLast(cells, index, firstNextMonth);
943
- }
944
- default: return cells[index];
945
- }
946
- }
947
- _isDayOutOfView(date) {
948
- return date < this._keyboardNavigationDayViewParameters.firstDayInView || date > this._keyboardNavigationDayViewParameters.lastDayInView;
949
- }
950
- _findDayArrows(cells, index, date, delta) {
951
- const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(date, delta));
952
- if (this._isDayOutOfView(newDateValue)) return cells[index];
953
- const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
954
- if (!nextCell || nextCell.disabled) return this._findDayArrows(cells, index, this._dateAdapter.deserialize(newDateValue), delta);
955
- return nextCell;
956
- }
957
- _findDayPageUpDown(cells, index, day, delta, deltaIfDisabled) {
958
- const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(day.dateValue, delta));
959
- if (this._isDayOutOfView(newDateValue)) return cells[index];
960
- const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
961
- if (!nextCell || nextCell.disabled) return this._findDayPageUpDown(cells, index, day, delta + deltaIfDisabled, deltaIfDisabled);
962
- return nextCell;
963
- }
964
- _findDayFirst(cells, index, day, date) {
965
- const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.createDate(+day.yearValue, +day.monthValue, date));
966
- if (this._isDayOutOfView(newDateValue)) return cells[index];
967
- const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
968
- if (!nextCell || nextCell.disabled) return this._findDayFirst(cells, index, day, date + 1);
969
- return nextCell;
970
- }
971
- _findDayLast(cells, index, firstNextMonth) {
972
- const newDateValue = this._dateAdapter.toIso8601(this._dateAdapter.addCalendarDays(firstNextMonth, -1));
973
- if (this._isDayOutOfView(newDateValue)) return cells[index];
974
- const nextCell = cells.find((e) => this._mapValueToISODate(e.value) === newDateValue);
975
- if (!nextCell || nextCell.disabled) return this._findDayLast(cells, index, this._dateAdapter.deserialize(newDateValue));
976
- return nextCell;
977
- }
978
- /**
979
- * Gets the index of the element to move to, based on a list of elements (which can be potentially disabled),
980
- * the keyboard input and the position of the current element in the list.
981
- * In the day view, the `day?: Day` parameter is mandatory for calculation,
982
- * while in month and year view it's not due to the fixed amount of rendered cells.
983
- */
984
- _navigateByKeyboard(evt, index, cells) {
985
- const { elementIndexForWideMode, offsetForWideMode, lastElementIndexForWideMode, verticalOffset } = this._calculateParametersForKeyboardNavigation(index, this._calendarView === "year");
986
- switch (evt.key) {
987
- case "ArrowUp": return this._findNext(cells, index, -verticalOffset);
988
- case "ArrowDown": return this._findNext(cells, index, verticalOffset);
989
- case "ArrowLeft": return this._findNext(cells, index, -1);
990
- case "ArrowRight": return this._findNext(cells, index, 1);
991
- case "Home": return this._findFirst(cells, offsetForWideMode);
992
- case "PageUp": return this._findFirstOnColumn(cells, elementIndexForWideMode, offsetForWideMode, verticalOffset);
993
- case "PageDown": return this._findLastOnColumn(cells, index, lastElementIndexForWideMode, verticalOffset);
994
- case "End": return this._findLast(cells, lastElementIndexForWideMode - 1);
995
- default: return cells[index];
996
- }
997
- }
998
- /**
999
- * Calculates the parameters needed in keyboard navigation in year and month view.
1000
- * @param index The starting element's index in the cell array.
1001
- * @param isYearView Whether the displayed `view` is the year one.
1002
- */
1003
- _calculateParametersForKeyboardNavigation(index, isYearView) {
1004
- const elementsPerPage = isYearView ? YEARS_PER_PAGE : MONTHS_PER_PAGE;
1005
- const offset = Math.trunc(index / elementsPerPage) * elementsPerPage;
1006
- const indexInView = offset === 0 ? index : index - elementsPerPage;
1007
- return {
1008
- verticalOffset: isYearView ? YEARS_PER_ROW : MONTHS_PER_ROW,
1009
- elementIndexForWideMode: indexInView,
1010
- offsetForWideMode: index - indexInView,
1011
- lastElementIndexForWideMode: offset === 0 ? elementsPerPage : elementsPerPage * 2
1012
- };
1013
- }
1014
- /**
1015
- * Gets the next element of the provided array starting from `index` by adding `delta`.
1016
- * If the found element is disabled, it continues adding `delta` until it finds an enabled one in the array bounds.
1017
- */
1018
- _findNext(days, index, delta) {
1019
- let nextIndex = index + delta;
1020
- while (nextIndex < days.length && days[nextIndex]?.disabled) nextIndex += delta;
1021
- return days[nextIndex] ?? days[index];
1022
- }
1023
- /** Find the first enabled element in the provided array. */
1024
- _findFirst(days, firstOfCurrentMonth) {
1025
- return !days[firstOfCurrentMonth].disabled ? days[firstOfCurrentMonth] : this._findNext(days, firstOfCurrentMonth, 1);
1026
- }
1027
- /** Find the last enabled element in the provided array. */
1028
- _findLast(days, lastOfCurrentMonth) {
1029
- return !days[lastOfCurrentMonth].disabled ? days[lastOfCurrentMonth] : this._findNext(days, lastOfCurrentMonth, -1);
1030
- }
1031
- /** Find the first enabled element in the same column of the provided array. */
1032
- _findFirstOnColumn(days, index, offset, verticalOffset) {
1033
- const nextIndex = index % verticalOffset + offset;
1034
- return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, verticalOffset);
1035
- }
1036
- /** Find the last enabled element in the same column of the provided array. */
1037
- _findLastOnColumn(days, index, offset, verticalOffset) {
1038
- const nextIndex = index + Math.trunc((offset - index - 1) / verticalOffset) * verticalOffset;
1039
- return !days[nextIndex].disabled ? days[nextIndex] : this._findNext(days, nextIndex, -verticalOffset);
1040
- }
1041
- _resetCalendarViewAndEmitMonthChange(initTransition = false) {
1042
- this._resetCalendarView(initTransition);
1043
- this._emitMonthChange();
1044
- }
1045
- _resetCalendarView(initTransition = false) {
1046
- if (this._containingFocus) this._resetFocus = true;
1047
- this._activeDate = (this.multiple ? this._selected.at(-1) : this._selected) ?? this._dateAdapter.today();
1048
- this._setChosenYear();
1049
- this._chosenMonth = void 0;
1050
- this._init();
1051
- this._nextCalendarView = this._calendarView = this.view;
1052
- if (initTransition) this._startTableTransition();
1053
- }
1054
- /** Render the view for the day selection. */
1055
- _renderDayView() {
1056
- const nextMonthActiveDate = this._wide ? this._dateAdapter.addCalendarMonths(this._activeDate, 1) : void 0;
1057
- return html`
1058
- <div class="sbb-calendar__controls">
1059
- ${this._getArrow("left", () => this._goToDifferentMonth(-1), i18nPreviousMonth[this._language.current], this._previousMonthDisabled())}
1060
- <div class="sbb-calendar__controls-month">
1061
- ${this._createLabelForDayView(this._activeDate)}
1062
- ${this._wide ? this._createLabelForDayView(nextMonthActiveDate) : nothing}
1063
- <span class="sbb-screen-reader-only" role="status">
1064
- ${this._createAriaLabelForDayView(this._activeDate, nextMonthActiveDate)}
1065
- </span>
1066
- </div>
1067
- ${this._getArrow("right", () => this._goToDifferentMonth(1), i18nNextMonth[this._language.current], this._nextMonthDisabled())}
1068
- </div>
1069
- <div class="sbb-calendar__table-overflow-break">
1070
- <div class="sbb-calendar__table-container sbb-calendar__table-day-view">
1071
- ${this.orientation === "horizontal" ? html`
1072
- ${this._createDayTable(this._weeks, this._weekNumbers)}
1073
- ${this._wide ? this._createDayTable(this._nextMonthWeeks, this._nextMonthWeekNumbers, true) : nothing}
1074
- ` : html`
1075
- ${this._createDayTableVertical(this._weeks, this._weekNumbers)}
1076
- ${this._wide ? this._createDayTableVertical(this._nextMonthWeeks, this._nextMonthWeekNumbers, nextMonthActiveDate) : nothing}
1077
- `}
1078
- </div>
1079
- </div>
1080
- `;
1081
- }
1082
- /** Creates the label with the month for the daily view. */
1083
- _createLabelForDayView(d) {
1084
- const monthLabel = `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)}`;
1085
- return html`
1086
- <button
1087
- type="button"
1088
- class="sbb-calendar__date-selection sbb-calendar__controls-change-date"
1089
- aria-label="${i18nYearMonthSelection[this._language.current]} ${monthLabel}"
1090
- @click=${() => {
1091
- this._resetFocus = true;
1092
- this._nextCalendarView = "year";
1093
- this._startTableTransition();
1094
- }}
1095
- >
1096
- ${monthLabel}
1097
- <sbb-icon name="chevron-small-down-small"></sbb-icon>
1098
- </button>
1099
- `;
1100
- }
1101
- /** Creates the aria-label for the daily view. */
1102
- _createAriaLabelForDayView(...dates) {
1103
- let monthLabel = "";
1104
- for (const d of dates) if (d) monthLabel += `${this._monthNames[this._dateAdapter.getMonth(d) - 1]} ${this._dateAdapter.getYear(d)} `;
1105
- return monthLabel;
1106
- }
1107
- /** Creates the calendar table for the daily view. */
1108
- _createDayTable(weeks, weekNumbers, isWideNextMonth = false) {
1109
- const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1110
- const weeksForSelectMultipleWeekDays = (isWideNextMonth ? this._nextMonthWeeks : this._weeks).flat();
1111
- return html`
1112
- <table
1113
- class="sbb-calendar__table"
1114
- @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1115
- @animationend=${(e) => this._tableAnimationEnd(e)}
1116
- >
1117
- <thead class="sbb-calendar__table-header">
1118
- <tr>
1119
- ${this.weekNumbers ? html`<th class="sbb-calendar__table-header-cell"></th>` : nothing}
1120
- ${this._weekdays.map((weekDay, index) => html`
1121
- <th class="sbb-calendar__table-header-cell">
1122
- ${this.multiple ? html`
1123
- <sbb-calendar-weekday
1124
- .value=${weekDay}
1125
- @click=${() => {
1126
- const days = weeksForSelectMultipleWeekDays.filter((day) => day.weekDayValue === (index + 1) % 7);
1127
- this._selectMultipleDates(days);
1128
- }}
1129
- ></sbb-calendar-weekday>
1130
- ` : html`
1131
- <span class="sbb-screen-reader-only">${weekDay.long}</span>
1132
- <span aria-hidden="true">${weekDay.narrow}</span>
1133
- `}
1134
- </th>
1135
- `)}
1136
- </tr>
1137
- </thead>
1138
- <tbody class="sbb-calendar__table-body">
1139
- ${weeks.map((week, rowIndex) => {
1140
- const firstRowOffset = DAYS_PER_ROW - week.length;
1141
- if (rowIndex === 0 && firstRowOffset) return html`
1142
- <tr>
1143
- ${this.weekNumbers ? html`
1144
- <td class="sbb-calendar__table-header-cell-vertical">
1145
- ${this.multiple ? html`
1146
- <sbb-calendar-weeknumber
1147
- .value=${weekNumbers[0]}
1148
- @click=${() => {
1149
- const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[0]);
1150
- this._selectMultipleDates(days);
1151
- }}
1152
- ></sbb-calendar-weeknumber>
1153
- ` : html`
1154
- <span class="sbb-screen-reader-only"
1155
- >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[0]}`}</span
1156
- >
1157
- <span aria-hidden="true">${weekNumbers[0]}</span>
1158
- `}
1159
- </td>
1160
- ` : nothing}
1161
- ${[...Array(firstRowOffset).keys()].map(() => html`<td class="sbb-calendar__table-data"></td>`)}
1162
- ${this._createDayCells(week)}
1163
- </tr>
1164
- `;
1165
- return html`
1166
- <tr>
1167
- ${this.weekNumbers ? html`
1168
- <td class="sbb-calendar__table-header-cell-vertical">
1169
- ${this.multiple ? html`
1170
- <sbb-calendar-weeknumber
1171
- .value=${weekNumbers[rowIndex]}
1172
- @click=${() => {
1173
- const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumbers[rowIndex]);
1174
- this._selectMultipleDates(days);
1175
- }}
1176
- ></sbb-calendar-weeknumber>
1177
- ` : html`
1178
- <span class="sbb-screen-reader-only"
1179
- >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumbers[rowIndex]}`}</span
1180
- >
1181
- <span aria-hidden="true">${weekNumbers[rowIndex]}</span>
1182
- `}
1183
- </td>
1184
- ` : nothing}
1185
- ${this._createDayCells(week)}
1186
- </tr>
1187
- `;
1188
- })}
1189
- </tbody>
1190
- </table>
1191
- `;
1192
- }
1193
- /** Creates the table in orientation='vertical'. */
1194
- _createDayTableVertical(weeks, weekNumbers, nextMonthActiveDate) {
1195
- const weekOffset = this._dateAdapter.getFirstWeekOffset(nextMonthActiveDate ?? this._activeDate);
1196
- const weeksForSelectMultipleWeekNumbers = (this._wide ? [...this._weeks, ...this._nextMonthWeeks] : nextMonthActiveDate ? this._nextMonthWeeks : this._weeks).flat();
1197
- return html`
1198
- <table
1199
- class="sbb-calendar__table"
1200
- @focusout=${(event) => this._handleTableBlur(event.relatedTarget)}
1201
- @animationend=${(e) => this._tableAnimationEnd(e)}
1202
- >
1203
- ${this.weekNumbers ? html`
1204
- <thead class="sbb-calendar__table-header">
1205
- <tr>
1206
- ${nextMonthActiveDate ? nothing : html`<th class="sbb-calendar__table-data"></th>`}
1207
- ${weekNumbers.map((weekNumber) => html`
1208
- <th class="sbb-calendar__table-header-cell">
1209
- ${this.multiple ? html`
1210
- <sbb-calendar-weeknumber
1211
- .value=${weekNumber}
1212
- @click=${() => {
1213
- const days = weeksForSelectMultipleWeekNumbers.filter((day) => day.weekValue === weekNumber);
1214
- this._selectMultipleDates(days);
1215
- }}
1216
- ></sbb-calendar-weeknumber>
1217
- ` : html`
1218
- <span class="sbb-screen-reader-only"
1219
- >${`${i18nCalendarWeekNumber[this._language.current]} ${weekNumber}`}</span
1220
- >
1221
- <span aria-hidden="true">${weekNumber}</span>
1222
- `}
1223
- </th>
1224
- `)}
1225
- </tr>
1226
- </thead>
1227
- ` : nothing}
1228
- <tbody class="sbb-calendar__table-body">
1229
- ${weeks.map((week, rowIndex) => {
1230
- const weekday = this._weekdays[rowIndex];
1231
- const selectableDays = this._wide ? [...week, ...this._nextMonthWeeks[rowIndex]] : week;
1232
- return html`
1233
- <tr>
1234
- ${nextMonthActiveDate ? nothing : html`
1235
- <td class="sbb-calendar__table-header-cell-vertical">
1236
- ${this.multiple ? html`
1237
- <sbb-calendar-weekday
1238
- .value=${weekday}
1239
- @click=${() => this._selectMultipleDates(selectableDays)}
1240
- >
1241
- ${weekday.narrow}
1242
- </sbb-calendar-weekday>
1243
- ` : html`
1244
- <span class="sbb-screen-reader-only">${weekday.long}</span>
1245
- <span aria-hidden="true">${weekday.narrow}</span>
1246
- `}
1247
- </td>
1248
- `}
1249
- ${rowIndex < weekOffset ? html`<td class="sbb-calendar__table-data"></td>` : nothing}
1250
- ${this._createDayCells(week)}
1251
- </tr>
1252
- `;
1253
- })}
1254
- </tbody>
1255
- </table>
1256
- `;
1257
- }
1258
- /** Creates the cells for the daily view. */
1259
- _createDayCells(week) {
1260
- return week.map((day) => {
1261
- return html`
1262
- <td class="sbb-calendar__table-data sbb-calendar__day-cell">
1263
- <slot name=${day.value}>
1264
- <sbb-calendar-day
1265
- slot=${day.value}
1266
- @click=${() => this._selectDate(day.dateValue)}
1267
- @keydown=${(evt) => this._handleKeyboardEvent(evt, day)}
1268
- ></sbb-calendar-day>
1269
- </slot>
1270
- </td>
1271
- `;
1272
- });
1273
- }
1274
- /** Render the view for the month selection. */
1275
- _renderMonthView() {
1276
- return html`
1277
- <div class="sbb-calendar__controls">
1278
- ${this._getArrow("left", () => this._goToDifferentYear(-1), i18nPreviousYear[this._language.current], this._previousYearDisabled())}
1279
- <div class="sbb-calendar__controls-month">${this._createLabelForMonthView()}</div>
1280
- ${this._getArrow("right", () => this._goToDifferentYear(1), i18nNextYear[this._language.current], this._nextYearDisabled())}
1281
- </div>
1282
- <div class="sbb-calendar__table-overflow-break">
1283
- <div class="sbb-calendar__table-container sbb-calendar__table-month-view">
1284
- ${this._createMonthTable(this._months, this._chosenYear)}
1285
- ${this._wide ? this._createMonthTable(this._months, this._chosenYear + 1) : nothing}
1286
- </div>
1287
- </div>
1288
- `;
1289
- }
1290
- /** Creates the label with the year for the monthly view. */
1291
- _createLabelForMonthView() {
1292
- return html` <button
1293
- type="button"
1294
- id="sbb-calendar__month-selection"
1295
- class="sbb-calendar__controls-change-date"
1296
- aria-label=${`${i18nCalendarDateSelection[this._language.current]} ${this._chosenYear}`}
1297
- @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
1298
- >
1299
- ${this._chosenYear} ${this._wide ? ` - ${this._chosenYear + 1}` : nothing}
1300
- <sbb-icon name="chevron-small-up-small"></sbb-icon>
1301
- </button>
1302
- <span class="sbb-screen-reader-only" role="status"> ${this._chosenYear} </span>`;
1303
- }
1304
- /** Creates the table for the month selection view. */
1305
- _createMonthTable(months, year) {
1306
- return html`
1307
- <table
1308
- class="sbb-calendar__table"
1309
- @animationend=${(e) => this._tableAnimationEnd(e)}
1310
- >
1311
- ${this._wide ? html`<thead class="sbb-calendar__table-header" aria-hidden="true">
1312
- <tr>
1313
- <th class="sbb-calendar__table-header-cell" colspan=${MONTHS_PER_ROW}>${year}</th>
1314
- </tr>
1315
- </thead>` : nothing}
1316
- <tbody class="sbb-calendar__table-body">
1317
- ${months.map((row) => html`
1318
- <tr>
1319
- ${row.map((month) => {
1320
- return html`
1321
- <td class="sbb-calendar__table-data">
1322
- <sbb-calendar-month
1323
- .value="${year}-${month.value}"
1324
- @click=${() => this._onMonthSelection(month.monthValue, year)}
1325
- @keydown=${(evt) => this._handleKeyboardEvent(evt)}
1326
- >
1327
- </sbb-calendar-month>
1328
- </td>
1329
- `;
1330
- })}
1331
- </tr>
1332
- `)}
1333
- </tbody>
1334
- </table>
1335
- `;
1336
- }
1337
- /** Select the month and change the view to day selection. */
1338
- _onMonthSelection(month, year) {
1339
- this._chosenMonth = month;
1340
- this._nextCalendarView = "day";
1341
- this._init(this._dateAdapter.createDate(year, this._chosenMonth, this._dateAdapter.getDate(this._activeDate)));
1342
- this._startTableTransition();
1343
- this._emitMonthChange();
1344
- }
1345
- /** Render the view for the year selection. */
1346
- _renderYearView() {
1347
- return html`
1348
- <div class="sbb-calendar__controls">
1349
- ${this._getArrow("left", () => this._goToDifferentYearRange(-YEARS_PER_PAGE), i18nPreviousYearRange(YEARS_PER_PAGE)[this._language.current], this._previousYearRangeDisabled())}
1350
- <div class="sbb-calendar__controls-month">${this._createLabelForYearView()}</div>
1351
- ${this._getArrow("right", () => this._goToDifferentYearRange(YEARS_PER_PAGE), i18nNextYearRange(YEARS_PER_PAGE)[this._language.current], this._nextYearRangeDisabled())}
1352
- </div>
1353
- <div class="sbb-calendar__table-overflow-break">
1354
- <div class="sbb-calendar__table-container sbb-calendar__table-year-view">
1355
- ${this._createYearTable(this._years)}
1356
- ${this._wide ? this._createYearTable(this._nextMonthYears, true) : nothing}
1357
- </div>
1358
- </div>
1359
- `;
1360
- }
1361
- /** Creates the button arrow for all the views. */
1362
- _getArrow(direction, click, ariaLabel, disabled) {
1363
- return html`<sbb-secondary-button
1364
- size="m"
1365
- icon-name="chevron-small-${direction}-small"
1366
- aria-label=${ariaLabel}
1367
- @click=${click}
1368
- ?disabled=${disabled}
1369
- id="sbb-calendar__controls-${direction === "left" ? "previous" : "next"}"
1370
- ></sbb-secondary-button>`;
1371
- }
1372
- /** Creates the label with the year range for the yearly view. */
1373
- _createLabelForYearView() {
1374
- const firstYear = this._years.flat()[0];
1375
- const lastYearArray = (this._wide ? this._nextMonthYears : this._years).flat();
1376
- const yearLabel = `${firstYear} - ${lastYearArray[lastYearArray.length - 1]}`;
1377
- return html`
1378
- <button
1379
- type="button"
1380
- id="sbb-calendar__year-selection"
1381
- class="sbb-calendar__controls-change-date"
1382
- aria-label="${i18nCalendarDateSelection[this._language.current]} ${yearLabel}"
1383
- @click=${() => this._resetCalendarViewAndEmitMonthChange(true)}
1384
- >
1385
- ${yearLabel}
1386
- <sbb-icon name="chevron-small-up-small"></sbb-icon>
1387
- </button>
1388
- <span class="sbb-screen-reader-only" role="status"> ${yearLabel} </span>
1389
- `;
1390
- }
1391
- /** Creates the table for the year selection view. */
1392
- _createYearTable(years, shiftRight = false) {
1393
- return html` <table
1394
- class="sbb-calendar__table"
1395
- @animationend=${(e) => this._tableAnimationEnd(e)}
1396
- >
1397
- <tbody class="sbb-calendar__table-body">
1398
- ${years.map((row) => html` <tr>
1399
- ${row.map((year) => {
1400
- return html`
1401
- <td class="sbb-calendar__table-data">
1402
- <sbb-calendar-year
1403
- .value=${String(year)}
1404
- @keydown=${(evt) => this._handleKeyboardEvent(evt)}
1405
- @click=${() => this._onYearSelection(year, shiftRight)}
1406
- >
1407
- </sbb-calendar-year>
1408
- </td>
1409
- `;
1410
- })}
1411
- </tr>`)}
1412
- </tbody>
1413
- </table>`;
1414
- }
1415
- /** Select the year and change the view to month selection. */
1416
- _onYearSelection(year, rightSide) {
1417
- this._chosenYear = rightSide ? year - 1 : year;
1418
- this._nextCalendarView = "month";
1419
- this._assignActiveDate(this._dateAdapter.createDate(this._chosenYear, this._dateAdapter.getMonth(this._activeDate), this._dateAdapter.getDate(this._activeDate)));
1420
- this._startTableTransition();
1421
- }
1422
- _getView() {
1423
- if (isServer || this.hydrationRequired) return html`${nothing}`;
1424
- switch (this._calendarView) {
1425
- case "year": return this._renderYearView();
1426
- case "month": return this._renderMonthView();
1427
- default: return this._renderDayView();
1428
- }
1429
- }
1430
- _tableAnimationEnd(event) {
1431
- const table = event.target;
1432
- if (event.animationName === "hide") {
1433
- table.classList.remove("sbb-calendar__table-hide");
1434
- if (this._containingFocus) this._resetFocus = true;
1435
- this._calendarView = this._nextCalendarView;
1436
- } else if (event.animationName === "show") this.internals.states.delete("transition");
1437
- }
1438
- _startTableTransition() {
1439
- this.internals.states.add("transition");
1440
- this.shadowRoot?.querySelectorAll("table")?.forEach((e) => e.classList.toggle("sbb-calendar__table-hide"));
1441
- }
1442
- render() {
1443
- return html`<div class="sbb-calendar__wrapper">${this._getView()}</div>`;
1444
- }
1445
- };
1446
- })();
1447
- //#endregion
1448
- export { SbbMonthChangeEvent as n, SbbCalendarElement as t };
1449
-
1450
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIuY29tcG9uZW50LUNqbGNEeVM1LmpzIiwibmFtZXMiOltdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9lbGVtZW50cy9jYWxlbmRhci9jYWxlbmRhci9jYWxlbmRhci5zY3NzP2lubGluZSIsIi4uLy4uLy4uL3NyYy9lbGVtZW50cy9jYWxlbmRhci9jYWxlbmRhci9jYWxlbmRhci5jb21wb25lbnQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiQHVzZSAnLi4vLi4vY29yZS9zdHlsZXMnIGFzIHNiYjtcblxuOmhvc3Qge1xuICBkaXNwbGF5OiBibG9jaztcblxuICAvLyBXZSBhZGQgd2lkdGggZGVmaW5pdGlvbiB0byBob3N0LCB0byBtYWtlIG92ZXJ3cml0aW5nIGVhc3kgZm9yIGNvbnN1bWVycy5cbiAgd2lkdGg6IG1heC1jb250ZW50O1xuXG4gIC8vIFNldCB0aGUgbGluZS1oZWlnaHQgZm9yIGFkamFjZW50IGVsZW1lbnRzIGluIG9yZGVyIHRvIHByZXZlbnQgZnJvbSBpbmZsdWVuY2Ugb2YgYSBjb25zdW1lci1zZXQgbGluZS1oZWlnaHQuXG4gIGxpbmUtaGVpZ2h0OiB2YXIoLS1zYmItdHlwby1saW5lLWhlaWdodC10ZXh0KTtcblxuICAtLXNiYi1jYWxlbmRhci1jZWxsLXNpemU6ICN7c2JiLnB4LXRvLXJlbS1idWlsZCg0NCl9O1xuXG4gIC8vIE1hdGNoIEZpZ21hIGRlc2lnblxuICAtLXNiYi1jYWxlbmRhci1jZWxsLWJvcmRlci1yYWRpdXM6IGNhbGMoXG4gICAgdmFyKC0tc2JiLWJvcmRlci1yYWRpdXMtNHgpICsgdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtYm9yZGVyLXdpZHRoKVxuICApO1xuICAtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb246IHZhcihcbiAgICAtLXNiYi1kaXNhYmxlLWFuaW1hdGlvbi1kdXJhdGlvbixcbiAgICB2YXIoLS1zYmItYW5pbWF0aW9uLWR1cmF0aW9uLTJ4KVxuICApO1xuXG4gIEBpbmNsdWRlIHNiYi5tcSgkZnJvbTogc21hbGwpIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWhlaWdodDogI3tzYmIucHgtdG8tcmVtLWJ1aWxkKDQ4KX07XG4gIH1cbn1cblxuOmhvc3QoOnN0YXRlKGVuaGFuY2VkKSkge1xuICAtLXNiYi1jYWxlbmRhci1jZWxsLXBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItc3BhY2luZy1maXhlZC0zeCk7XG59XG5cbi5zYmItY2FsZW5kYXJfX3dyYXBwZXIge1xuICB3aWR0aDogMTAwJTtcbiAgZGlzcGxheTogYmxvY2s7XG4gIHRyYW5zaXRpb24tZHVyYXRpb246IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb24pO1xufVxuXG4uc2JiLWNhbGVuZGFyX19jb250cm9scyB7XG4gIHdpZHRoOiAxMDAlO1xuICBkaXNwbGF5OiBpbmxpbmUtZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgZ2FwOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC1nYXApO1xuICBtYXJnaW4tYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY29udHJvbC1tYXJnaW4tYmxvY2stZW5kKTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGgge1xuICB3aWR0aDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgZ2FwOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGVzLWdhcCk7XG59XG5cbiNzYmItY2FsZW5kYXJfX2NvbnRyb2xzLXByZXZpb3VzLFxuI3NiYi1jYWxlbmRhcl9fY29udHJvbHMtbmV4dCB7XG4gIC13ZWJraXQtdGFwLWhpZ2hsaWdodC1jb2xvcjogdHJhbnNwYXJlbnQ7XG59XG5cbi5zYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlIHtcbiAgQGluY2x1ZGUgc2JiLmJ1dHRvbi1yZXNldDtcblxuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBtYXJnaW46IGF1dG87XG4gIGhlaWdodDogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtaGVpZ2h0KTtcbiAgZm9udC1zaXplOiB2YXIoLS1zYmItdGV4dC1mb250LXNpemUtcyk7XG4gIGxldHRlci1zcGFjaW5nOiB2YXIoLS1zYmItdHlwby1sZXR0ZXItc3BhY2luZy10ZXh0KTtcbiAgbGluZS1oZWlnaHQ6IHZhcigtLXNiYi10eXBvLWxldHRlci1zcGFjaW5nLXRleHQpO1xuICB0ZXh0LXRyYW5zZm9ybTogY2FwaXRhbGl6ZTtcbiAgY3Vyc29yOiB2YXIoLS1zYmItY3Vyc29yLXBvaW50ZXIpO1xuICBwYWRkaW5nLWlubGluZTogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtcGFkZGluZy1pbmxpbmUpO1xuICBib3JkZXItcmFkaXVzOiB2YXIoLS1zYmItYm9yZGVyLXJhZGl1cy1pbmZpbml0eSk7XG4gIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWJhY2tncm91bmQpO1xuICBjb2xvcjogdmFyKC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtY29sb3IpO1xuICB0cmFuc2l0aW9uLWR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC10cmFuc2l0aW9uLWR1cmF0aW9uKTtcbiAgdHJhbnNpdGlvbi10aW1pbmctZnVuY3Rpb246IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZWFzaW5nLWZ1bmN0aW9uKTtcbiAgdHJhbnNpdGlvbi1wcm9wZXJ0eTogYmFja2dyb3VuZC1jb2xvciwgcGFkZGluZy1ibG9jay1lbmQ7XG5cbiAgJjpkaXNhYmxlZCB7XG4gICAgLS1zYmItY2FsZW5kYXItY29udHJvbC12aWV3LWNoYW5nZS1iYWNrZ3JvdW5kOiBsaWdodC1kYXJrKFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLW1pbGspLFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWFudGhyYWNpdGUpXG4gICAgKTtcbiAgICAtLXNiYi1jYWxlbmRhci1jb250cm9sLXZpZXctY2hhbmdlLWNvbG9yOiBsaWdodC1kYXJrKFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWdyYW5pdGUpLFxuICAgICAgdmFyKC0tc2JiLWNvbG9yLWFsdW1pbml1bSlcbiAgICApO1xuXG4gICAgY3Vyc29yOiB1bnNldDtcbiAgfVxuXG4gICY6Zm9jdXMtdmlzaWJsZSB7XG4gICAgQGluY2x1ZGUgc2JiLmZvY3VzLW91dGxpbmU7XG5cbiAgICBvdXRsaW5lLW9mZnNldDogdmFyKC0tc2JiLXNwYWNpbmctZml4ZWQtMXgpO1xuICB9XG5cbiAgQGluY2x1ZGUgc2JiLmhvdmVyLW1xIHtcbiAgICAmOm5vdCg6YWN0aXZlLCA6ZGlzYWJsZWQpOmhvdmVyIHtcbiAgICAgIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1ob3Zlci1zaGlmdCk7XG4gICAgfVxuICB9XG5cbiAgJjpub3QoOmRpc2FibGVkKTphY3RpdmUge1xuICAgIC0tc2JiLWNhbGVuZGFyLWNvbnRyb2wtdmlldy1jaGFuZ2UtYmFja2dyb3VuZDogdmFyKC0tc2JiLWJhY2tncm91bmQtY29sb3ItMyk7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtbW9udGgtdmlldyxcbi5zYmItY2FsZW5kYXJfX3RhYmxlLXllYXItdmlldyB7XG4gIC0tc2JiLWNhbGVuZGFyLXRhYmxlLWNvbHVtbi1zcGFjZXM6IDY7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrIHtcbiAgLy8gV2UgaGF2ZSB0byBjdXQgdGhlIG5lZ2F0aXZlIG1hcmdpbiB3aGljaCBpcyBidWlsdCBieSB0aGUgY29udGFpbmluZyBzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lci5cbiAgLy8gT3ZlcmZsb3c6IGhpZGRlbiB3b3VsZCBjdXQgdGhlIGZvY3VzIG91dGxpbmUuIFRoZXJlZm9yZSB3ZSB0YWtlIGBjb250YWluYCBoZXJlLlxuICBjb250YWluOiBsYXlvdXQ7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciB7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGdhcDogdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApO1xuICBtYXJnaW4taW5saW5lOiB2YXIoLS1zYmItY2FsZW5kYXItbWFyZ2luKTtcblxuICAvLyBUaGUgcGFkZGluZyBvZiB0aGUgZmlyc3QgYW5kIGxhc3QgY29sdW1uIHNob3VsZCBub3QgYmUgdmlzaWJsZSBpZiBjYWxlbmRhciBpcyBzdHJldGNoZWQuXG4gIC8vIFRoZXJlZm9yZSB3ZSBuZWVkIGEgbmVnYXRpdmUgaW5saW5lIG1hcmdpbi5cbiAgLy8gQXMgd2UgZG9uJ3Qgd2FudCB0byBzcXVlZXplLCB0aGUgbWFyZ2luIHNob3VsZCBuZXZlciBiZSBncmVhdGVyIHRoYW4gemVyby5cblxuICAvLyBNaW4gd2lkdGggaXMgZXF1YWxzIHRvIHRoZSBub3JtYWwgd2lkdGggb2YgdGhlIGNhbGVuZGFyXG4gIC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aDogY2FsYyg3ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpO1xuXG4gIC8vIFRoZSBvdmVyZmxvdyB2YXJpYWJsZSBpcyBlcXVhbHMgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiB0aGUgYWN0dWFsIHdpZHRoIGFuZCB0aGUgbWluIHdpZHRoLlxuICAtLXNiYi1jYWxlbmRhci1vdmVyZmxvdzogY2FsYygxMDAlIC0gdmFyKC0tc2JiLWNhbGVuZGFyLW1pbi13aWR0aCkpO1xuXG4gIC8vIFRoZSBzdGFydCBvZmZzZXQgaXMgbmVnYXRpdmUgbWFyZ2luIHdoaWNoIHNob3VsZCBvdmVybGFwIHRoZSBwYXJlbnQgY29udGFpbmVyLiBTaG91bGQgbmV2ZXIgYmUgYSBwb3NpdGl2ZSB2YWx1ZS5cbiAgLS1zYmItY2FsZW5kYXItc3RhcnQtb2Zmc2V0OiBtaW4oXG4gICAgMHB4LFxuICAgIC0xICogKHZhcigtLXNiYi1jYWxlbmRhci1vdmVyZmxvdykgLyB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtY29sdW1uLXNwYWNlcykpXG4gICk7XG4gIC0tc2JiLWNhbGVuZGFyLW1hcmdpbjogdmFyKC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldCk7XG5cbiAgOmhvc3QoOnN0YXRlKHdpZGUpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoXG4gICAgICAyICogNyAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpICsgdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApXG4gICAgKTtcbiAgICAtLXNiYi1jYWxlbmRhci1tYXJnaW46IGNhbGMoMC41ICogdmFyKC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldCkpO1xuICB9XG5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSdob3Jpem9udGFsJ11bd2Vlay1udW1iZXJzXSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKDggKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSk7XG4gIH1cblxuICA6aG9zdChbb3JpZW50YXRpb249J2hvcml6b250YWwnXVt3ZWVrLW51bWJlcnNdOnN0YXRlKHdpZGUpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1taW4td2lkdGg6IGNhbGMoXG4gICAgICAyICogOCAqIHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpICsgdmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlcy1nYXApXG4gICAgKTtcbiAgfVxuXG4gIC8vIFRoZSBjb250YWluZXIncyBtaW4td2lkdGggaXMgc2V0IGluIHZlcnRpY2FsIG1vZGU7XG4gIC8vIHRoZSB2YWx1ZSBgY2FsYyg3ICogdmFyKC0tc2JiLWNhbGVuZGFyLWNlbGwtc2l6ZSkpYCBpcyBmaW5lXG4gIC8vIGNvbnNpZGVyaW5nIHRoYXQgdGhlIG1heGltdW0gbnVtYmVyIG9mIHdlZWtzIGluIGEgbW9udGggaXMgNiwgcGx1cyAxIGZvciB0aGUgaGVhZGVyIGF0IHRoZSB0YWJsZSdzIGxlZnQgc2lkZS5cbiAgOmhvc3QoW29yaWVudGF0aW9uPSd2ZXJ0aWNhbCddKSAmIHtcbiAgICBtaW4td2lkdGg6IHZhcigtLXNiYi1jYWxlbmRhci1taW4td2lkdGgpO1xuXG4gICAgLy8gVGhlIHZhcmlhYmxlIHRoYXQgZGVmaW5lcyB0aGUgbWFyZ2luIGlzIHNldCB0byAwIHRvIGNvcnJlY3RseSBkaXNwbGF5IHRoZSB0YWJsZSBpZiBjb25zdW1lcnMgc2V0IGEgY3VzdG9tIHdpZHRoLlxuICAgIC0tc2JiLWNhbGVuZGFyLXN0YXJ0LW9mZnNldDogMDtcbiAgfVxuXG4gIC8vIFRoZSBtaW4td2lkdGggaW4gd2lkZSBpcyBjYWxjdWxhdGVkIGFzIChtYXggbnVtYmVyIG9mIHdlZWtzIGluIGEgbW9udGgpICogKGRpc3BsYXllZCBtb250aHMpICsgKGhlYWRlciBjb2x1bW4pID0gNiAqIDIgKyAxID0gMTMuXG4gIDpob3N0KFtvcmllbnRhdGlvbj0ndmVydGljYWwnXTpzdGF0ZSh3aWRlKSkgJiB7XG4gICAgLS1zYmItY2FsZW5kYXItbWluLXdpZHRoOiBjYWxjKFxuICAgICAgMTMgKiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKSArIHZhcigtLXNiYi1jYWxlbmRhci10YWJsZXMtZ2FwKVxuICAgICk7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUge1xuICB3aWR0aDogMTAwJTtcbiAgYm9yZGVyLWNvbGxhcHNlOiBjb2xsYXBzZTtcbiAgaGVpZ2h0OiBtYXgtY29udGVudDtcblxuICBhbmltYXRpb246IHtcbiAgICBuYW1lOiBzaG93O1xuICAgIGR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtYW5pbWF0aW9uLWR1cmF0aW9uKTtcbiAgfVxuXG4gICYuc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jZWxsLXRyYW5zaXRpb24tZHVyYXRpb246IDBtcztcblxuICAgIGFuaW1hdGlvbjoge1xuICAgICAgbmFtZTogaGlkZTtcbiAgICAgIGR1cmF0aW9uOiB2YXIoLS1zYmItY2FsZW5kYXItdGFibGUtYW5pbWF0aW9uLWR1cmF0aW9uKTtcbiAgICB9XG4gIH1cblxuICA6aG9zdCg6bm90KDpzdGF0ZSh3aWRlKSkpICYge1xuICAgIC8vIER1ZSB0byBhIFNhZmFyaSBpT1MgcmVuZGVyaW5nIGJ1ZyB3ZSBuZWVkIHRvIGRlZmluZSBtaW4td2lkdGggYXMgd2VsbC5cbiAgICAvLyBPdGhlcndpc2UsIGFmdGVyIG9yaWVudGF0aW9uIGNoYW5nZSwgdGhlcmUgaXMgYSB3cm9uZyB3aWR0aCBpZiBwbGFjZWQgaW4gYW4gc2JiLWRpYWxvZy5cbiAgICBtaW4td2lkdGg6IDEwMCU7XG4gIH1cbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyIHtcbiAgLy8gRGVwZW5kaW5nIHdoZXJlIGluIHRoZSBET00gaXQncyB1c2VkLCBpdCdzIHBvc3NpYmxlIHRoYXQgYnJvd3NlciBkZWZhdWx0cyBzZXQgdGV4dC1hbGlnbiB0byBsZWZ0LlxuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWJvZHkge1xuICAvLyBEZXBlbmRpbmcgd2hlcmUgaW4gdGhlIERPTSBpdCdzIHVzZWQsIGl0J3MgcG9zc2libGUgdGhhdCBicm93c2VyIGRlZmF1bHRzIHNldCB0ZXh0LWFsaWduIHRvIGxlZnQuXG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwsXG4uc2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbCB7XG4gIHdpZHRoOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1zaXplKTtcbiAgY29sb3I6IHZhcigtLXNiYi1jYWxlbmRhci1oZWFkZXItY29sb3IpO1xuICBwYWRkaW5nOiAwO1xuICBmb250LXNpemU6IHZhcigtLXNiYi10ZXh0LWZvbnQtc2l6ZS14cyk7XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG4gIGxldHRlci1zcGFjaW5nOiB2YXIoLS1zYmItdHlwby1sZXR0ZXItc3BhY2luZy10ZXh0KTtcblxuICA6aG9zdCg6bm90KFttdWx0aXBsZV0pKSAmIHtcbiAgICBoZWlnaHQ6IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXNpemUpO1xuICB9XG59XG5cbi5zYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlci1jZWxsLXZlcnRpY2FsIHtcbiAgcGFkZGluZy1ibG9jay1lbmQ6IHZhcigtLXNiYi1jYWxlbmRhci1jZWxsLXBhZGRpbmctYmxvY2stZW5kKTtcbn1cblxuLnNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YSB7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgcGFkZGluZzogMDtcbiAgdGV4dC1hbGlnbjogY2VudGVyO1xufVxuXG4uc2JiLWNhbGVuZGFyX19kYXktY2VsbCB7XG4gIHBhZGRpbmctYmxvY2stZW5kOiB2YXIoLS1zYmItY2FsZW5kYXItY2VsbC1wYWRkaW5nLWJsb2NrLWVuZCk7XG59XG5cbnNiYi1jYWxlbmRhci1kYXkge1xuICA6aG9zdCg6bm90KDpzdGF0ZShlbmhhbmNlZCkpKSAmIHtcbiAgICAtLXNiYi1jYWxlbmRhci1jZWxsLWp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1oZWlnaHQ6ICN7c2JiLnB4LXRvLXJlbS1idWlsZCg0NCl9O1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1leHRyYS1kaXNwbGF5OiBub25lO1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS12YWx1ZS1oZWlnaHQ6IHVuc2V0O1xuICAgIC0tc2JiLWNhbGVuZGFyLWRheS1jcm9zc2VkLW91dC10b3A6IDUwJTtcbiAgICAtLXNiYi1jYWxlbmRhci1kYXktY3Jvc3NlZC1vdXQtdHJhbnNsYXRlOiB0cmFuc2xhdGUoLTUwJSwgLTUwJSkgcm90YXRlKC00NWRlZyk7XG4gIH1cbn1cblxuQGtleWZyYW1lcyBzaG93IHtcbiAgZnJvbSB7XG4gICAgb3BhY2l0eTogMDtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkodmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1zaGlmdCkpO1xuICB9XG5cbiAgdG8ge1xuICAgIG9wYWNpdHk6IDE7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDAlKTtcbiAgfVxufVxuXG5Aa2V5ZnJhbWVzIGhpZGUge1xuICBmcm9tIHtcbiAgICBvcGFjaXR5OiAxO1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwJSk7XG4gIH1cblxuICB0byB7XG4gICAgb3BhY2l0eTogMDtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkodmFyKC0tc2JiLWNhbGVuZGFyLXRhYmxlLWFuaW1hdGlvbi1zaGlmdCkpO1xuICB9XG59XG4iLCJpbXBvcnQge1xuICB0eXBlIENTU1Jlc3VsdEdyb3VwLFxuICBodG1sLFxuICBpc1NlcnZlcixcbiAgbm90aGluZyxcbiAgdHlwZSBQcm9wZXJ0eVZhbHVlcyxcbiAgdHlwZSBUZW1wbGF0ZVJlc3VsdCxcbiAgdW5zYWZlQ1NTLFxufSBmcm9tICdsaXQnO1xuaW1wb3J0IHsgcHJvcGVydHksIHN0YXRlIH0gZnJvbSAnbGl0L2RlY29yYXRvcnMuanMnO1xuXG5pbXBvcnQgeyBTYmJTZWNvbmRhcnlCdXR0b25FbGVtZW50IH0gZnJvbSAnLi4vLi4vYnV0dG9uLnB1cmUudHMnO1xuaW1wb3J0IHtcbiAgdHlwZSBEYXRlQWRhcHRlcixcbiAgREFZU19QRVJfUk9XLFxuICBkZWZhdWx0RGF0ZUFkYXB0ZXIsXG4gIGZvcmNlVHlwZSxcbiAgaGFuZGxlRGlzdGluY3RDaGFuZ2UsXG4gIGkxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb24sXG4gIGkxOG5DYWxlbmRhcldlZWtOdW1iZXIsXG4gIGkxOG5OZXh0TW9udGgsXG4gIGkxOG5OZXh0WWVhcixcbiAgaTE4bk5leHRZZWFyUmFuZ2UsXG4gIGkxOG5QcmV2aW91c01vbnRoLFxuICBpMThuUHJldmlvdXNZZWFyLFxuICBpMThuUHJldmlvdXNZZWFyUmFuZ2UsXG4gIGkxOG5ZZWFyTW9udGhTZWxlY3Rpb24sXG4gIGlzQXJyb3dLZXlPclBhZ2VLZXlzUHJlc3NlZCxcbiAgTU9OREFZLFxuICBNT05USFNfUEVSX1BBR0UsXG4gIE1PTlRIU19QRVJfUk9XLFxuICBwbGFpbkRhdGUsXG4gIHJlYWRDb25maWcsXG4gIFNiYkVsZW1lbnQsXG4gIHR5cGUgU2JiRWxlbWVudFR5cGUsXG4gIFNiYkxhbmd1YWdlQ29udHJvbGxlcixcbiAgU2JiTWVkaWFNYXRjaGVyQ29udHJvbGxlcixcbiAgU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlLFxuICBzY3JlZW5SZWFkZXJPbmx5U3R5bGVzLFxuICBUSFVSU0RBWSxcbiAgVFVFU0RBWSxcbiAgV0VETkVTREFZLFxuICBZRUFSU19QRVJfUEFHRSxcbiAgWUVBUlNfUEVSX1JPVyxcbn0gZnJvbSAnLi4vLi4vY29yZS50cyc7XG5pbXBvcnQgeyBTYmJJY29uRWxlbWVudCB9IGZyb20gJy4uLy4uL2ljb24ucHVyZS50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhckRheUVsZW1lbnQgfSBmcm9tICcuLi9jYWxlbmRhci1kYXkvY2FsZW5kYXItZGF5LmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhck1vbnRoRWxlbWVudCB9IGZyb20gJy4uL2NhbGVuZGFyLW1vbnRoL2NhbGVuZGFyLW1vbnRoLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhcldlZWtkYXlFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItd2Vla2RheS9jYWxlbmRhci13ZWVrZGF5LmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhcldlZWtudW1iZXJFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXItd2Vla251bWJlci9jYWxlbmRhci13ZWVrbnVtYmVyLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgeyBTYmJDYWxlbmRhclllYXJFbGVtZW50IH0gZnJvbSAnLi4vY2FsZW5kYXIteWVhci9jYWxlbmRhci15ZWFyLmNvbXBvbmVudC50cyc7XG5pbXBvcnQgdHlwZSB7IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50IH0gZnJvbSAnLi4vY29tbW9uL2NhbGVuZGFyLWNlbGwtYmFzZS1lbGVtZW50LnRzJztcblxuaW1wb3J0IHN0eWxlIGZyb20gJy4vY2FsZW5kYXIuc2Nzcz9pbmxpbmUnO1xuXG5leHBvcnQgY2xhc3MgU2JiTW9udGhDaGFuZ2VFdmVudCBleHRlbmRzIEV2ZW50IHtcbiAgcHJpdmF0ZSByZWFkb25seSBfcmFuZ2U6IHJlYWRvbmx5IERheVtdO1xuXG4gIHB1YmxpYyBnZXQgcmFuZ2UoKTogcmVhZG9ubHkgRGF5W10ge1xuICAgIHJldHVybiB0aGlzLl9yYW5nZTtcbiAgfVxuXG4gIHB1YmxpYyBjb25zdHJ1Y3RvcihyYW5nZTogcmVhZG9ubHkgRGF5W10pIHtcbiAgICBzdXBlcignbW9udGhjaGFuZ2UnLCB7IGJ1YmJsZXM6IHRydWUsIGNvbXBvc2VkOiB0cnVlIH0pO1xuICAgIHRoaXMuX3JhbmdlID0gT2JqZWN0LmZyZWV6ZShyYW5nZSB8fCBbXSk7XG4gIH1cbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIG5lZWRlZCBpbiB5ZWFyIGFuZCBtb250aCB2aWV3cyB0byBjb3JyZWN0bHkgY2FsY3VsYXRlIHRoZSBuZXh0IGVsZW1lbnQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi5cbiAqXG4gKiBUaGUgY2VsbCdzIGluZGV4IGFuZCB0aGUgZWxlbWVudCdzIGluZGV4IGluIGl0cyBtb250aCAvIHllYXIgYmF0Y2ggbXVzdCBiZSBkaXN0aW5ndWlzaGVkOlxuICogdGhlIGZpcnN0IGlzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCBpbiB0aGUgYXJyYXkgb2YgYWxsIHRoZSByZW5kZXJlZCBjZWxscywgd2hpbGUgdGhlIHNlY29uZCBpcyB0aGUgaW5kZXggb2YgdGhlIGVsZW1lbnQgcmVsYXRpdmUgdG8gaXRzIHRhYmxlLlxuICogSW4gbm9uLXdpZGUgbW9kZSwgdGhlIHd0byBhcmUgdGhlIHNhbWUsIHdoaWxlIGluIHdpZGUgbW9kZSB0aGUgY2VsbCdzIGluZGV4IGNhbiBnbyBmcm9tIDAgdG8gNDcgZm9yIHllYXJzIGFuZCBmcm9tIDAgdG8gMjMgZm9yIG1vbnRocyxcbiAqIHdoaWxlIHRoZSBlbGVtZW50IGluZGV4IGdvZXMgZnJvbSAwIHRvLCByZXNwZWN0aXZlbHksIDIzIGFuZCAxMS5cbiAqL1xuaW50ZXJmYWNlIENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzIHtcbiAgLyoqIFRoZSBlbGVtZW50IGluZGV4IHdpdGhpbiBpdHMgeWVhciBvciBtb250aCByYW5nZS4gKi9cbiAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIEluIHdpZGUgbW9kZSwgdGhlIGluZGV4IG9mIHRoZSBmaXJzdCBlbGVtZW50IGluIHRoZSBzZWNvbmQgcGFuZWwsIG9yLCBhbHRlcm5hdGl2ZWx5LCB0aGUgbnVtYmVyIG9mIGVsZW1lbnRzIGluIHRoZSBmaXJzdCBwYW5lbC4gKi9cbiAgb2Zmc2V0Rm9yV2lkZU1vZGU6IG51bWJlcjtcbiAgLyoqIFRoZSBpbmRleCBvZiB0aGUgbGFzdCBlbGVtZW50IHdpdGhpbiB0aGUgZWxlbWVudCdzIG1vbnRoIChvciB5ZWFyIHJhbmdlKS4gKi9cbiAgbGFzdEVsZW1lbnRJbmRleEZvcldpZGVNb2RlOiBudW1iZXI7XG4gIC8qKiBUaGUgbnVtYmVyIG9mIGNlbGxzIGRpc3BsYXllZCBpbiBhIHNpbmdsZSByb3csIGRlcGVuZGluZyBvbiB0aGUgcmVuZGVyZWQgdmlldy4gKi9cbiAgdmVydGljYWxPZmZzZXQ6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIG5lZWRlZCBpbiBkYXkgdmlldyB0byBjb3JyZWN0bHkgY2FsY3VsYXRlIHRoZSBuZXh0IGVsZW1lbnQgaW4ga2V5Ym9hcmQgbmF2aWdhdGlvbi5cbiAqXG4gKiBJbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLCBpdCdzIG5vdCBwb3NzaWJsZSB0byByZWx5IG9uIGFueSBhcnJheS9pbmRleCB0byBjYWxjdWxhdGUgdGhlIGVsZW1lbnQgdG8gbmF2aWdhdGUgdG8sXG4gKiBzbyBjYWxjdWxhdGlvbnMgb24gZGF0ZXMgbXVzdCBiZSBkb25lLCB3aGljaCBzaG91bGQgY29uc2lkZXIgdmlldyBib3VuZGFyaWVzLCBvZmZzZXRzIGFuZCBtb250aCdzIGxlbmd0aC5cbiAqL1xuaW50ZXJmYWNlIENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMge1xuICAvKiogVGhlIGZpcnN0IGRheSByZW5kZXJlZC4gKi9cbiAgZmlyc3REYXlJblZpZXc6IHN0cmluZyB8IG51bGw7XG4gIC8qKiBUaGUgbGFzdCBkYXkgcmVuZGVyZWQuIEl0IGRlcGVuZHMgb24gdGhlICd3aWRlJyB2YWx1ZS4gKi9cbiAgbGFzdERheUluVmlldzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIFRoZSBvZmZzZXQgZnJvbSB0aGUgZmlyc3QgZGF5IG9mIHRoZSB3ZWVrIChNb25kYXkpIG9mIHRoZSBmaXJzdCByZW5kZXJlZCBtb250aC4gKi9cbiAgZmlyc3RNb250aE9mZnNldDogbnVtYmVyO1xuICAvKiogVGhlIG51bWJlciBvZiBkYXlzIGluIHRoZSBmaXJzdCByZW5kZXJlZCBtb250aC4gKi9cbiAgZmlyc3RNb250aExlbmd0aDogbnVtYmVyO1xuICAvKiogVGhlIG9mZnNldCBmcm9tIHRoZSBmaXJzdCBkYXkgb2YgdGhlIHdlZWsgKE1vbmRheSkgb2YgdGhlIHNlY29uZCByZW5kZXJlZCBtb250aC4gSWYgd2lkZSBpcyBmYWxzZSwgaXQncyBlcXVhbCB0byB6ZXJvLiAqL1xuICBzZWNvbmRNb250aE9mZnNldDogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERheTxUID0gRGF0ZT4ge1xuICAvKiogRGF0ZSBhcyBJU08gc3RyaW5nLiAqL1xuICB2YWx1ZTogc3RyaW5nO1xuICBkYXlWYWx1ZTogc3RyaW5nO1xuICBtb250aFZhbHVlOiBzdHJpbmc7XG4gIHllYXJWYWx1ZTogc3RyaW5nO1xuICBkYXRlVmFsdWU6IFQ7XG4gIHdlZWtWYWx1ZTogbnVtYmVyO1xuICB3ZWVrRGF5VmFsdWU6IG51bWJlcjtcbn1cblxuLyoqIEBkZXByZWNhdGVkIHVzZSBNb250aENlbGwgKi9cbmV4cG9ydCBpbnRlcmZhY2UgTW9udGgge1xuICB2YWx1ZTogc3RyaW5nO1xuICBsb25nVmFsdWU6IHN0cmluZztcbiAgbW9udGhWYWx1ZTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1vbnRoQ2VsbCB7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIG1vbnRoVmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBXZWVrZGF5IHtcbiAgbG9uZzogc3RyaW5nO1xuICBuYXJyb3c6IHN0cmluZztcbn1cblxuLyoqXG4gKiBJdCBkaXNwbGF5cyBhIGNhbGVuZGFyIHdoaWNoIGFsbG93cyBjaG9vc2luZyBhIGRhdGUuXG4gKlxuICogQHNsb3QgLSBVc2UgdGhlIHVubmFtZWQgc2xvdCB0byBhZGQgY3VzdG9taXplZCBgc2JiLWNhbGVuZGFyLWRheWAgZWxlbWVudHMuXG4gKi9cbmV4cG9ydCBjbGFzcyBTYmJDYWxlbmRhckVsZW1lbnQ8VCA9IERhdGU+IGV4dGVuZHMgU2JiRWxlbWVudCB7XG4gIHB1YmxpYyBzdGF0aWMgb3ZlcnJpZGUgcmVhZG9ubHkgZWxlbWVudE5hbWU6IHN0cmluZyA9ICdzYmItY2FsZW5kYXInO1xuICBwdWJsaWMgc3RhdGljIG92ZXJyaWRlIGVsZW1lbnREZXBlbmRlbmNpZXM6IFNiYkVsZW1lbnRUeXBlW10gPSBbXG4gICAgU2JiQ2FsZW5kYXJEYXlFbGVtZW50LFxuICAgIFNiYkNhbGVuZGFyTW9udGhFbGVtZW50LFxuICAgIFNiYkNhbGVuZGFyV2Vla2RheUVsZW1lbnQsXG4gICAgU2JiQ2FsZW5kYXJXZWVrbnVtYmVyRWxlbWVudCxcbiAgICBTYmJDYWxlbmRhclllYXJFbGVtZW50LFxuICAgIFNiYkljb25FbGVtZW50LFxuICAgIFNiYlNlY29uZGFyeUJ1dHRvbkVsZW1lbnQsXG4gIF07XG4gIHB1YmxpYyBzdGF0aWMgb3ZlcnJpZGUgc3R5bGVzOiBDU1NSZXN1bHRHcm91cCA9IFtzY3JlZW5SZWFkZXJPbmx5U3R5bGVzLCB1bnNhZmVDU1Moc3R5bGUpXTtcbiAgcHVibGljIHN0YXRpYyByZWFkb25seSBldmVudHMgPSB7XG4gICAgZGF0ZXNlbGVjdGVkOiAnZGF0ZXNlbGVjdGVkJyxcbiAgICBtb250aGNoYW5nZTogJ21vbnRoY2hhbmdlJyxcbiAgfSBhcyBjb25zdDtcblxuICAvKiogSWYgc2V0IHRvIHRydWUsIHR3byBtb250aHMgYXJlIGRpc3BsYXllZCAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3Igd2lkZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKiBUaGUgaW5pdGlhbCB2aWV3IG9mIHRoZSBjYWxlbmRhciB3aGljaCBzaG91bGQgYmUgZGlzcGxheWVkIG9uIG9wZW5pbmcuICovXG4gIEBwcm9wZXJ0eSgpIHB1YmxpYyBhY2Nlc3NvciB2aWV3OiAnZGF5JyB8ICdtb250aCcgfCAneWVhcicgPSAnZGF5JztcblxuICAvKipcbiAgICogVGhlIG1pbmltdW0gdmFsaWQgZGF0ZS4gQWNjZXB0cyBhIGRhdGUgb2JqZWN0IG9yIG51bGwuXG4gICAqIEFjY2VwdHMgYW4gSVNPODYwMSBmb3JtYXR0ZWQgc3RyaW5nIChlLmcuIDIwMjQtMTItMjQpIGFzIGF0dHJpYnV0ZS5cbiAgICovXG4gIEBwbGFpbkRhdGUoKVxuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgYWNjZXNzb3IgbWluOiBUIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqXG4gICAqIFRoZSBtYXhpbXVtIHZhbGlkIGRhdGUuIEFjY2VwdHMgYSBkYXRlIG9iamVjdCBvciBudWxsLlxuICAgKiBBY2NlcHRzIGFuIElTTzg2MDEgZm9ybWF0dGVkIHN0cmluZyAoZS5nLiAyMDI0LTEyLTI0KSBhcyBhdHRyaWJ1dGUuXG4gICAqL1xuICBAcGxhaW5EYXRlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIG1heDogVCB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBjYWxlbmRhciBhbGxvd3MgZm9yIG11bHRpcGxlIGRhdGUgc2VsZWN0aW9uLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQGhhbmRsZURpc3RpbmN0Q2hhbmdlKChlOiBTYmJDYWxlbmRhckVsZW1lbnQ8VD4sIG5ld1ZhbHVlOiBib29sZWFuKSA9PlxuICAgIGUuX29uTXVsdGlwbGVDaGFuZ2VkKG5ld1ZhbHVlKSxcbiAgKVxuICBAcHJvcGVydHkoeyB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBtdWx0aXBsZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBUaGUgc2VsZWN0ZWQgZGF0ZTogYWNjZXB0cyBhIGRhdGUgb2JqZWN0LCBvciwgaWYgYG11bHRpcGxlYCwgYW4gYXJyYXkgb2YgZGF0ZXMuXG4gICAqL1xuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgc2V0IHNlbGVjdGVkKHZhbHVlOiBUIHwgVFtdIHwgbnVsbCkge1xuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSB2YWx1ZVxuICAgICAgICAubWFwKChkYXRlTGlrZTogVCkgPT5cbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRWYWxpZERhdGVPck51bGwodGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZGF0ZUxpa2UpKSxcbiAgICAgICAgKVxuICAgICAgICAuZmlsdGVyKChkYXRlOiBUIHwgbnVsbCk6IGRhdGUgaXMgVCA9PiBkYXRlICE9PSBudWxsKVxuICAgICAgICAuZmlsdGVyKFxuICAgICAgICAgIChkYXRlOiBUKSA9PlxuICAgICAgICAgICAgIXRoaXMuX2lzRGF5SW5SYW5nZSh0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZGF0ZSkpIHx8IHRoaXMuX2RhdGVGaWx0ZXIoZGF0ZSksXG4gICAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHNlbGVjdGVkRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFZhbGlkRGF0ZU9yTnVsbChcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUodmFsdWUpLFxuICAgICAgKTtcbiAgICAgIGlmIChcbiAgICAgICAgISFzZWxlY3RlZERhdGUgJiZcbiAgICAgICAgKCF0aGlzLl9pc0RheUluUmFuZ2UodGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHNlbGVjdGVkRGF0ZSkpIHx8XG4gICAgICAgICAgdGhpcy5fZGF0ZUZpbHRlcihzZWxlY3RlZERhdGUpKVxuICAgICAgKSB7XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gc2VsZWN0ZWREYXRlO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBudWxsO1xuICAgICAgfVxuICAgIH1cbiAgfVxuICBwdWJsaWMgZ2V0IHNlbGVjdGVkKCk6IFQgfCBUW10gfCBudWxsIHtcbiAgICByZXR1cm4gdGhpcy5fc2VsZWN0ZWQ7XG4gIH1cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfc2VsZWN0ZWQ6IFQgfCBUW10gfCBudWxsID0gbnVsbDtcblxuICAvKiogQSBmdW5jdGlvbiB1c2VkIHRvIGZpbHRlciBvdXQgZGF0ZXMuICovXG4gIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2RhdGUtZmlsdGVyJyB9KVxuICBwdWJsaWMgYWNjZXNzb3IgZGF0ZUZpbHRlcjogKChkYXRlOiBUIHwgbnVsbCkgPT4gYm9vbGVhbikgfCBudWxsID0gbnVsbDtcblxuICAvKiogVGhlIG9yaWVudGF0aW9uIG9mIGRheXMgaW4gdGhlIGNhbGVuZGFyLiAqL1xuICBAcHJvcGVydHkoeyByZWZsZWN0OiB0cnVlIH0pIHB1YmxpYyBhY2Nlc3NvciBvcmllbnRhdGlvbjogJ2hvcml6b250YWwnIHwgJ3ZlcnRpY2FsJyA9XG4gICAgJ2hvcml6b250YWwnO1xuXG4gIC8qKiBXaGV0aGVyIGl0IGhhcyB0byBkaXNwbGF5IHRoZSB3ZWVrIG51bWJlcnMgaW4gYWRkaXRpb24gdG8gd2VlayBkYXlzLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnd2Vlay1udW1iZXJzJywgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3Igd2Vla051bWJlcnM6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBwcml2YXRlIF9kYXRlQWRhcHRlcjogRGF0ZUFkYXB0ZXI8VD4gPSByZWFkQ29uZmlnKCkuZGF0ZXRpbWU/LmRhdGVBZGFwdGVyID8/IGRlZmF1bHREYXRlQWRhcHRlcjtcblxuICAvKiogVGhlIGN1cnJlbnRseSBhY3RpdmUgZGF0ZS4gKi9cbiAgQHN0YXRlKCkgcHJpdmF0ZSBhY2Nlc3NvciBfYWN0aXZlRGF0ZTogVCA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvZGF5KCk7XG5cbiAgLyoqIFRoZSBjdXJyZW50IHdpZGUgcHJvcGVydHkgY29uc2lkZXJpbmcgcHJvcGVydHkgdmFsdWUgYW5kIGJyZWFrcG9pbnRzLiBGcm9tIHplcm8gdG8gc21hbGwgYHdpZGVgIGhhcyBhbHdheXMgdG8gYmUgZmFsc2UuICovXG4gIEBzdGF0ZSgpXG4gIHByaXZhdGUgc2V0IF93aWRlKHdpZGU6IGJvb2xlYW4pIHtcbiAgICB0aGlzLnRvZ2dsZVN0YXRlKCd3aWRlJywgd2lkZSk7XG4gICAgdGhpcy5fd2lkZUludGVybmFsID0gd2lkZTtcbiAgfVxuICBwcml2YXRlIGdldCBfd2lkZSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5fd2lkZUludGVybmFsO1xuICB9XG4gIC8vIFdlIGtlZXAgdGhlIHN0YXRlIGluIGEgZmllbGQgYmVjYXVzZSBvZiB0aGUgV2ViS2l0IGJ1ZyBodHRwczovL2J1Z3Mud2Via2l0Lm9yZy9zaG93X2J1Zy5jZ2k/aWQ9MzAzNDY3LlxuICAvLyBUT0RPOiByZS1jaGVjayB3aGV0aGVyIGZpZWxkIGlzIG5lZWRlZFxuICBwcml2YXRlIF93aWRlSW50ZXJuYWw6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBAc3RhdGUoKSBwcml2YXRlIGFjY2Vzc29yIF9jYWxlbmRhclZpZXc6IFNiYkNhbGVuZGFyRWxlbWVudFsndmlldyddID0gJ2RheSc7XG5cbiAgcHJpdmF0ZSBfbmV4dENhbGVuZGFyVmlldzogU2JiQ2FsZW5kYXJFbGVtZW50Wyd2aWV3J10gPSAnZGF5JztcblxuICAvKiogSW5mb3JtYXRpb24gYWJvdXQgdGhlIHJlbmRlcmVkIGRheSB2aWV3OyB1c2VkIGluIGtleWJvYXJkIG5hdmlnYXRpb24uICovXG4gIHByaXZhdGUgX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzOiBDYWxlbmRhcktleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzID0ge1xuICAgIGZpcnN0RGF5SW5WaWV3OiBudWxsLFxuICAgIGxhc3REYXlJblZpZXc6IG51bGwsXG4gICAgZmlyc3RNb250aE9mZnNldDogMCxcbiAgICBmaXJzdE1vbnRoTGVuZ3RoOiAwLFxuICAgIHNlY29uZE1vbnRoT2Zmc2V0OiAwLFxuICB9O1xuXG4gIC8qKiBBIGxpc3Qgb2YgZGF5cywgaW4gdHdvIGZvcm1hdHMgKGxvbmcgYW5kIHNpbmdsZSBjaGFyKS4gKi9cbiAgcHJpdmF0ZSBfd2Vla2RheXMhOiBXZWVrZGF5W107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIHRoZSBkYXRlcyBvZiB0aGUgbW9udGguICovXG4gIHByaXZhdGUgX3dlZWtzOiBEYXk8VD5bXVtdID0gW107XG5cbiAgLyoqIEdyaWQgb2YgY2FsZW5kYXIgY2VsbHMgcmVwcmVzZW50aW5nIG1vbnRocy4gKi9cbiAgcHJpdmF0ZSBfbW9udGhzITogTW9udGhDZWxsW11bXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgeWVhcnMuICovXG4gIHByaXZhdGUgX3llYXJzITogbnVtYmVyW11bXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgeWVhcnMgZm9yIHRoZSB3aWRlIHZpZXcuICovXG4gIHByaXZhdGUgX25leHRNb250aFllYXJzITogbnVtYmVyW11bXTtcblxuICAvKiogR3JpZCBvZiBjYWxlbmRhciBjZWxscyByZXByZXNlbnRpbmcgdGhlIGRhdGVzIG9mIHRoZSBuZXh0IG1vbnRoLiAqL1xuICBwcml2YXRlIF9uZXh0TW9udGhXZWVrcyE6IERheTxUPltdW107XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgYWxsIHRoZSBtb250aCBuYW1lcyBpbiB0aGUgY3VycmVudCBsYW5ndWFnZS4gKi9cbiAgcHJpdmF0ZSBfbW9udGhOYW1lczogc3RyaW5nW10gPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aE5hbWVzKCdsb25nJyk7XG5cbiAgLyoqIEFuIGFycmF5IGNvbnRhaW5pbmcgdGhlIHdlZWtzJyBudW1iZXJzIGZvciB0aGUgY3VycmVudCBtb250aC4gKi9cbiAgcHJpdmF0ZSBfd2Vla051bWJlcnMhOiBudW1iZXJbXTtcblxuICAvKiogQW4gYXJyYXkgY29udGFpbmluZyB0aGUgd2Vla3MnIG51bWJlcnMgZm9yIHRoZSBuZXh0IG1vbnRoIGluIHdpZGUgbW9kZS4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoV2Vla051bWJlcnMhOiBudW1iZXJbXTtcblxuICBwcml2YXRlIF9lbmhhbmNlZFZhcmlhbnQ6IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKiogQSBsaXN0IG9mIGNhbGVuZGFyJ3MgY2VsbHMgY29ycmVzcG9uZGluZyB0byBkYXlzLCBtb250aHMgb3IgeWVhcnMgZGVwZW5kaW5nIG9uIHRoZSB2aWV3LiAqL1xuICBwcml2YXRlIGdldCBfY2VsbHMoKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnRbXSB7XG4gICAgcmV0dXJuIEFycmF5LmZyb208U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KFxuICAgICAgKHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ2RheSdcbiAgICAgICAgPyAoQXJyYXkuZnJvbSh0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3JBbGwoJ3Nsb3QnKSkuZmxhdE1hcCgoZTogSFRNTFNsb3RFbGVtZW50KSA9PlxuICAgICAgICAgICAgZS5hc3NpZ25lZEVsZW1lbnRzKHsgZmxhdHRlbjogdHJ1ZSB9KSxcbiAgICAgICAgICApIGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdKVxuICAgICAgICA6IHRoaXMuc2hhZG93Um9vdD8ucXVlcnlTZWxlY3RvckFsbChgc2JiLWNhbGVuZGFyLSR7dGhpcy5fY2FsZW5kYXJWaWV3fWApKSA/PyBbXSxcbiAgICApO1xuICB9XG5cbiAgLyoqIFRoZSBjaG9zZW4geWVhciBpbiB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY2hvc2VuWWVhcj86IG51bWJlcjtcblxuICAvKiogVGhlIGNob3NlbiBtb250aCBpbiB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY2hvc2VuTW9udGg/OiBudW1iZXI7XG5cbiAgLyoqIFdoZXRoZXIgdGhlIGZvY3VzIHNob3VsZCBiZSByZXNldCBvbiBmb2N1c0NlbGwuICovXG4gIHByaXZhdGUgX3Jlc2V0Rm9jdXMgPSBmYWxzZTtcblxuICAvKiogV2hldGhlciBhbiBlbGVtZW50IGluc2lkZSB0aGUgY2FsZW5kYXIgaXMgY3VycmVudGx5IGZvY3VzZWQuICovXG4gIHByaXZhdGUgX2NvbnRhaW5pbmdGb2N1cyA9IGZhbHNlO1xuXG4gIEBzdGF0ZSgpXG4gIHByaXZhdGUgYWNjZXNzb3IgX2luaXRpYWxpemVkID0gZmFsc2U7XG5cbiAgcHJpdmF0ZSBfbGFuZ3VhZ2UgPSBuZXcgU2JiTGFuZ3VhZ2VDb250cm9sbGVyKHRoaXMpLndpdGhIYW5kbGVyKCgpID0+IHtcbiAgICB0aGlzLl9tb250aE5hbWVzID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGhOYW1lcygnbG9uZycpO1xuICAgIHRoaXMuX2NyZWF0ZU1vbnRoUm93cygpO1xuICB9KTtcbiAgcHJpdmF0ZSBfbWVkaWFNYXRjaGVyID0gbmV3IFNiYk1lZGlhTWF0Y2hlckNvbnRyb2xsZXIodGhpcywge1xuICAgIFtTYmJNZWRpYVF1ZXJ5QnJlYWtwb2ludExhcmdlQW5kQWJvdmVdOiAoKSA9PiB0aGlzLl9pbml0KCksXG4gIH0pO1xuXG4gIHB1YmxpYyBjb25zdHJ1Y3RvcigpIHtcbiAgICBzdXBlcigpO1xuICAgIHRoaXMuX2NyZWF0ZU1vbnRoUm93cygpO1xuICAgIHRoaXMuX3NldFdlZWtkYXlzKCk7XG5cbiAgICAvLyBXZSBuZWVkIHRvIHRyYWNrIHRoZSBmb2N1cyBhcyB3ZSBzaG91bGQgb25seSB0YWtlIGZvY3VzIGludG8gdGhlIGNhbGVuZGFyLCB3aGVuIHRoZVxuICAgIC8vIGZvY3VzIHdhcyBvbmNlIHNldCBpbnRvIHRoZSBjYWxlbmRhci5cbiAgICAvLyBGb3Igc2hhZG93IERPTSBjb21wYXRpYmlsaXR5IHdlIG5lZWQgdG8gdHJhY2sgdGhpcyBwcm9ncmFtbWF0aWNhbGx5LlxuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignZm9jdXNpbicsICgpID0+ICh0aGlzLl9jb250YWluaW5nRm9jdXMgPSB0cnVlKSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKCdmb2N1c291dCcsICgpID0+ICh0aGlzLl9jb250YWluaW5nRm9jdXMgPSBmYWxzZSkpO1xuICAgIHRoaXMuYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoZSkgPT4ge1xuICAgICAgY29uc3QgZGF5ID0gKGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50KS5jbG9zZXN0PFNiYkNhbGVuZGFyRGF5RWxlbWVudDxUPj4oJ3NiYi1jYWxlbmRhci1kYXknKTtcbiAgICAgIGlmIChkYXkpIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0RGF0ZShkYXkudmFsdWUhKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXIoJ2tleWRvd24nLCAoZSkgPT4ge1xuICAgICAgaWYgKChlLnRhcmdldCBhcyBIVE1MRWxlbWVudCkubG9jYWxOYW1lID09PSAnc2JiLWNhbGVuZGFyLWRheScpIHtcbiAgICAgICAgdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChcbiAgICAgICAgICBlLFxuICAgICAgICAgIHRoaXMuX21hcERhdGVUb0RheSgoZS50YXJnZXQgYXMgU2JiQ2FsZW5kYXJEYXlFbGVtZW50KS52YWx1ZSEgYXMgVCksXG4gICAgICAgICk7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIF9kYXRlRmlsdGVyKGRhdGU6IFQpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5kYXRlRmlsdGVyPy4oZGF0ZSkgPz8gdHJ1ZTtcbiAgfVxuXG4gIC8qKiBSZXNldHMgdGhlIGFjdGl2ZSBtb250aCBhY2NvcmRpbmcgdG8gdGhlIG5ldyBzdGF0ZSBvZiB0aGUgY2FsZW5kYXIuICovXG4gIHB1YmxpYyByZXNldFBvc2l0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Q2FsZW5kYXJWaWV3KCk7XG4gICAgdGhpcy5faW5pdCgpO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGNvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmNvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgdGhpcy5yZXNldFBvc2l0aW9uKCk7XG4gICAgdGhpcy5zaGFkb3dSb290Py5hZGRFdmVudExpc3RlbmVyKCdzbG90Y2hhbmdlJywgdGhpcy5fb25TbG90Q2hhbmdlLCB7IGNhcHR1cmU6IHRydWUgfSk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgZGlzY29ubmVjdGVkQ2FsbGJhY2soKTogdm9pZCB7XG4gICAgc3VwZXIuZGlzY29ubmVjdGVkQ2FsbGJhY2soKTtcbiAgICB0aGlzLnNoYWRvd1Jvb3Q/LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ3Nsb3RjaGFuZ2UnLCB0aGlzLl9vblNsb3RDaGFuZ2UsIHsgY2FwdHVyZTogdHJ1ZSB9KTtcbiAgfVxuXG4gIC8qKiBAaW50ZXJuYWwgKi9cbiAgcHVibGljIG92ZXJyaWRlIGZvY3VzKCk6IHZvaWQge1xuICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgIHRoaXMuX2ZvY3VzQ2VsbCgpO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHdpbGxVcGRhdGUoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIud2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllcyk7XG5cbiAgICBpZiAoIXRoaXMuX2luaXRpYWxpemVkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnd2lkZScpIHx8IGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnb3JpZW50YXRpb24nKSkge1xuICAgICAgdGhpcy5yZXNldFBvc2l0aW9uKCk7XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygndmlldycpKSB7XG4gICAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgICB0aGlzLl9jaG9zZW5Nb250aCA9IHVuZGVmaW5lZDtcbiAgICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLnZpZXc7XG4gICAgfVxuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIudXBkYXRlZChjaGFuZ2VkUHJvcGVydGllcyk7XG4gICAgLy8gVGhlIGNhbGVuZGFyIG5lZWRzIHRvIGNhbGN1bGF0ZSB0YWItaW5kZXhlcyBvbiBmaXJzdCByZW5kZXIsXG4gICAgLy8gYW5kIGV2ZXJ5IHRpbWUgYSBkYXRlIGlzIHNlbGVjdGVkIG9yIHRoZSBtb250aCB2aWV3IGNoYW5nZXMuXG4gICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcblxuICAgIC8vIFdoZW4gY2hhbmdpbmcgdmlldyB0byB5ZWFyL21vbnRoLCB0aGUgdGFiaW5kZXggaXMgY2hhbmdlZCwgYnV0IHRoZSBmb2N1c2VkIGVsZW1lbnQgaXMgZ2V0dGluZyBsb3N0LlxuICAgIC8vIFdlIG5lZWQgdG8gY2FsbCBgX2ZvY3VzQ2VsbCgpYCBtZXRob2QgZXhwbGljaXRseSB0byBjb3JyZWN0bHkgc2V0IHRoZSBmb2N1cy5cbiAgICB0aGlzLl9mb2N1c0NlbGwoKTtcbiAgfVxuXG4gIHByaXZhdGUgX29uU2xvdENoYW5nZSA9ICgpOiB2b2lkID0+IHtcbiAgICB0aGlzLl9lbmhhbmNlZFZhcmlhbnQgPSBBcnJheS5mcm9tKHRoaXMuY2hpbGRyZW4pLnNvbWUoXG4gICAgICAoYykgPT4gYy5sb2NhbE5hbWUgPT09ICdzYmItY2FsZW5kYXItZGF5JyxcbiAgICApO1xuICAgIHRoaXMudG9nZ2xlU3RhdGUoJ2VuaGFuY2VkJywgdGhpcy5fZW5oYW5jZWRWYXJpYW50KTtcbiAgICB0aGlzLl9zZXRUYWJJbmRleCgpO1xuICB9O1xuXG4gIC8qKlxuICAgKiBUaGUgYF9zZWxlY3RlZGAgc3RhdGUgc2hvdWxkIGJlIGFkYXB0ZWQgd2hlbiB0aGUgYG11bHRpcGxlYCBwcm9wZXJ0eSBjaGFuZ2VzOlxuICAgKiAgIC0gaWYgaXQgY2hhbmdlcyB0byB0cnVlLCB0aGUgJ19zZWxlY3RlZCcgaXMgc2V0IHRvIGFuIGFycmF5O1xuICAgKiAgIC0gaWYgaXQgY2hhbmdlcyB0byBmYWxzZSwgdGhlIGZpcnN0IGF2YWlsYWJsZSBvcHRpb24gaXMgc2V0IGFzICd2YWx1ZScgb3RoZXJ3aXNlIGl0J3Mgc2V0IHRvIG51bGwuXG4gICAqL1xuICBwcml2YXRlIF9vbk11bHRpcGxlQ2hhbmdlZChpc011bHRpcGxlOiBib29sZWFuKTogdm9pZCB7XG4gICAgaWYgKGlzTXVsdGlwbGUgJiYgIUFycmF5LmlzQXJyYXkodGhpcy5fc2VsZWN0ZWQpKSB7XG4gICAgICB0aGlzLl9zZWxlY3RlZCA9IHRoaXMuX3NlbGVjdGVkID8gW3RoaXMuX3NlbGVjdGVkIGFzIFRdIDogW107XG4gICAgfVxuICAgIGlmICghaXNNdWx0aXBsZSAmJiBBcnJheS5pc0FycmF5KHRoaXMuX3NlbGVjdGVkKSkge1xuICAgICAgdGhpcy5fc2VsZWN0ZWQgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5sZW5ndGggPyAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKVswXSA6IG51bGw7XG4gICAgfVxuICB9XG5cbiAgLyoqIEluaXRpYWxpemVzIHRoZSBjb21wb25lbnQuICovXG4gIHByaXZhdGUgX2luaXQoYWN0aXZlRGF0ZT86IFQpOiB2b2lkIHtcbiAgICAvLyBEdWUgdG8gaXRzIGNvbXBsZXhpdHksIHRoZSBjYWxlbmRhciBpcyBvbmx5IGluaXRpYWxpemVkIG9uIGNsaWVudCBzaWRlXG4gICAgaWYgKGlzU2VydmVyKSB7XG4gICAgICByZXR1cm47XG4gICAgfSBlbHNlIGlmICh0aGlzLmh5ZHJhdGlvblJlcXVpcmVkKSB7XG4gICAgICB0aGlzLmh5ZHJhdGlvbkNvbXBsZXRlLnRoZW4oKCkgPT4gdGhpcy5faW5pdCgpKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAoYWN0aXZlRGF0ZSkge1xuICAgICAgdGhpcy5fYXNzaWduQWN0aXZlRGF0ZShhY3RpdmVEYXRlKTtcbiAgICB9XG4gICAgdGhpcy5fd2lkZSA9XG4gICAgICAodGhpcy5fbWVkaWFNYXRjaGVyLm1hdGNoZXMoU2JiTWVkaWFRdWVyeUJyZWFrcG9pbnRMYXJnZUFuZEFib3ZlKSA/PyBmYWxzZSkgJiYgdGhpcy53aWRlO1xuICAgIHRoaXMuX3dlZWtzID0gdGhpcy5fY3JlYXRlV2Vla1Jvd3ModGhpcy5fYWN0aXZlRGF0ZSk7XG4gICAgdGhpcy5feWVhcnMgPSB0aGlzLl9jcmVhdGVZZWFyUm93cygpO1xuICAgIHRoaXMuX3dlZWtOdW1iZXJzID0gdGhpcy5fd2Vla3NcbiAgICAgIC5mbGF0KClcbiAgICAgIC5zb3J0KChhLCBiKSA9PiBhLnZhbHVlLmxvY2FsZUNvbXBhcmUoYi52YWx1ZSkpXG4gICAgICAubWFwKChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSlcbiAgICAgIC5maWx0ZXIoKHYsIGksIGEpID0+IGEuaW5kZXhPZih2KSA9PT0gaSk7XG4gICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MgPSBbW11dO1xuICAgIHRoaXMuX25leHRNb250aFllYXJzID0gW1tdXTtcbiAgICBpZiAodGhpcy5fd2lkZSkge1xuICAgICAgY29uc3QgbmV4dE1vbnRoRGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIDEpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MgPSB0aGlzLl9jcmVhdGVXZWVrUm93cyhuZXh0TW9udGhEYXRlLCB0cnVlKTtcbiAgICAgIHRoaXMuX25leHRNb250aFllYXJzID0gdGhpcy5fY3JlYXRlWWVhclJvd3MoWUVBUlNfUEVSX1BBR0UpO1xuICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla051bWJlcnMgPSB0aGlzLl9uZXh0TW9udGhXZWVrc1xuICAgICAgICAuZmxhdCgpXG4gICAgICAgIC5zb3J0KChhLCBiKSA9PiBhLnZhbHVlLmxvY2FsZUNvbXBhcmUoYi52YWx1ZSkpXG4gICAgICAgIC5tYXAoKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlKVxuICAgICAgICAuZmlsdGVyKCh2LCBpLCBhKSA9PiBhLmluZGV4T2YodikgPT09IGkpO1xuICAgIH1cbiAgICB0aGlzLl9pbml0aWFsaXplZCA9IHRydWU7XG4gIH1cblxuICAvKiogRm9jdXNlcyBvbiBhIGRheSBjZWxsIHByaW9yaXRpemluZyB0aGUgc2VsZWN0ZWQgZGF5LCB0aGUgY3VycmVudCBkYXksIGFuZCBsYXN0bHksIHRoZSBmaXJzdCBzZWxlY3RhYmxlIGRheS4gKi9cbiAgcHJpdmF0ZSBfZm9jdXNDZWxsKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLl9yZXNldEZvY3VzKSB7XG4gICAgICB0aGlzLl9nZXRGaXJzdEZvY3VzYWJsZSgpPy5mb2N1cygpO1xuICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBhcnJheSBvZiB3ZWVrZGF5cy4gKi9cbiAgcHJpdmF0ZSBfc2V0V2Vla2RheXMoKTogdm9pZCB7XG4gICAgY29uc3QgbmFycm93V2Vla2RheXM6IHN0cmluZ1tdID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF5T2ZXZWVrTmFtZXMoJ25hcnJvdycpO1xuICAgIGNvbnN0IGxvbmdXZWVrZGF5czogc3RyaW5nW10gPSB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWtOYW1lcygnbG9uZycpO1xuICAgIGNvbnN0IHdlZWtkYXlzOiBXZWVrZGF5W10gPSBsb25nV2Vla2RheXMubWFwKChsb25nOiBzdHJpbmcsIGk6IG51bWJlcikgPT4gKHtcbiAgICAgIGxvbmcsXG4gICAgICBuYXJyb3c6IG5hcnJvd1dlZWtkYXlzW2ldLFxuICAgIH0pKTtcblxuICAgIC8vIFJvdGF0ZXMgdGhlIGxhYmVscyBmb3IgZGF5cyBvZiB0aGUgd2VlayBiYXNlZCBvbiB0aGUgY29uZmlndXJlZCBmaXJzdCBkYXkgb2YgdGhlIHdlZWsuXG4gICAgY29uc3QgZmlyc3REYXlPZldlZWs6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0RGF5T2ZXZWVrKCk7XG4gICAgdGhpcy5fd2Vla2RheXMgPSB3ZWVrZGF5cy5zbGljZShmaXJzdERheU9mV2VlaykuY29uY2F0KHdlZWtkYXlzLnNsaWNlKDAsIGZpcnN0RGF5T2ZXZWVrKSk7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgcm93cyBhbG9uZyB0aGUgaG9yaXpvbnRhbCBkaXJlY3Rpb24gYW5kIHNldHMgdGhlIHBhcmFtZXRlcnMgdXNlZCBpbiBrZXlib2FyZCBuYXZpZ2F0aW9uLiAqL1xuICBwcml2YXRlIF9jcmVhdGVXZWVrUm93cyh2YWx1ZTogVCwgaXNTZWNvbmRNb250aEluVmlldyA9IGZhbHNlKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3QgZGF5c0luTW9udGg6IG51bWJlciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldE51bURheXNJbk1vbnRoKHZhbHVlKTtcbiAgICBjb25zdCB3ZWVrT2Zmc2V0OiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRGaXJzdFdlZWtPZmZzZXQodmFsdWUpO1xuICAgIGlmICghaXNTZWNvbmRNb250aEluVmlldykge1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aExlbmd0aCA9IGRheXNJbk1vbnRoO1xuICAgICAgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aE9mZnNldCA9IHdlZWtPZmZzZXQ7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5maXJzdERheUluVmlldyA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHZhbHVlKSxcbiAgICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aCh2YWx1ZSksXG4gICAgICAgICAgMSxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgICB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5sYXN0RGF5SW5WaWV3ID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodmFsdWUpLFxuICAgICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgICBkYXlzSW5Nb250aCxcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLnNlY29uZE1vbnRoT2Zmc2V0ID0gd2Vla09mZnNldDtcbiAgICAgIHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmxhc3REYXlJblZpZXcgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodmFsdWUpLFxuICAgICAgICAgIGRheXNJbk1vbnRoLFxuICAgICAgICApLFxuICAgICAgKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgPyB0aGlzLl9jcmVhdGVXZWVrUm93c0hvcml6b250YWwodmFsdWUsIGRheXNJbk1vbnRoLCB3ZWVrT2Zmc2V0KVxuICAgICAgOiB0aGlzLl9jcmVhdGVXZWVrUm93c1ZlcnRpY2FsKHZhbHVlLCBkYXlzSW5Nb250aCwgd2Vla09mZnNldCk7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyB0aGUgcm93cyBmb3IgZWFjaCB3ZWVrIGluIG9yaWVudGF0aW9uPSdob3Jpem9udGFsJy5cbiAgICpcbiAgICogSXRlcmF0ZXMgdGhyb3VnaCB0aGUgZGF5cyBvZiB0aGUgbW9udGhzLCBjcmVhdGVzIGEgRGF5IG9iamVjdCBmb3IgZWFjaCBhbmQgcHVzaGVzIGl0IGludG8gYW5kIGFycmF5LlxuICAgKiBFYWNoIHNldmVuIGRheXMgKGNvbnNpZGVyaW5nIHRoZSBvZmZzZXQgYXQgdGhlIGJlZ2lubmluZyBvZiB0aGUgbW9udGgpIHJlc3RhcnRzIGZyb20gYW4gZW1wdHkgYXJyYXkuXG4gICAqXG4gICAqIFRoZSByZXN1bHQgaXMgYSBtYXRyaXggaW4gd2hpY2ggZXZlcnkgcm93IGlzIGEgd2VlayAob3IgcGFydCBvZiBpdCwgY29uc2lkZXJpbmcgb2Zmc2V0KS5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzSG9yaXpvbnRhbCh2YWx1ZTogVCwgZGF5c0luTW9udGg6IG51bWJlciwgd2Vla09mZnNldDogbnVtYmVyKTogRGF5PFQ+W11bXSB7XG4gICAgY29uc3Qgd2Vla3M6IERheTxUPltdW10gPSBbW11dO1xuICAgIGZvciAobGV0IGkgPSAwLCBjZWxsID0gd2Vla09mZnNldDsgaSA8IGRheXNJbk1vbnRoOyBpKyssIGNlbGwrKykge1xuICAgICAgaWYgKGNlbGwgPT09IERBWVNfUEVSX1JPVykge1xuICAgICAgICB3ZWVrcy5wdXNoKFtdKTtcbiAgICAgICAgY2VsbCA9IDA7XG4gICAgICB9XG4gICAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgaSArIDEsXG4gICAgICApITtcbiAgICAgIHdlZWtzW3dlZWtzLmxlbmd0aCAtIDFdLnB1c2godGhpcy5fbWFwRGF0ZVRvRGF5KGRhdGUpKTtcbiAgICB9XG4gICAgcmV0dXJuIHdlZWtzO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgdGhlIHJvd3MgZm9yIGVhY2ggd2VlayBpbiBvcmllbnRhdGlvbj0ndmVydGljYWwnLlxuICAgKlxuICAgKiBDcmVhdGVzIGEgbWF0cml4IHdpdGggc2V2ZW4gZW1wdHkgcm93cy5cbiAgICogSXRlcmF0ZXMgdGhyb3VnaCB0aGUgZGF5cyBvZiB0aGUgbW9udGhzLCBjcmVhdGVzIGEgRGF5IG9iamVjdCBmb3IgZWFjaFxuICAgKiBhbmQgcHVzaGVzIGl0IGludG8gdGhlIGNvcnJlY3QgYXJyYXkgY29uc2lkZXJpbmcgdGhlIG9mZnNldCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoZSBtb250aC5cbiAgICogRWFjaCBzZXZlbiBkYXlzIChpbmNsdWRpbmcgb2Zmc2V0KSByZXN0YXJ0cyBmcm9tIHRoZSBmaXJzdC5cbiAgICpcbiAgICogVGhlIHJlc3VsdCBpcyBhIG1hdHJpeCBpbiB3aGljaCBldmVyeSByb3cgaXMgYSBzZXQgb2Ygd2Vla2RheXMsIHNvOlxuICAgKiAtIHJvdyAwOiBhbGwgdGhlIE1vbmRheXM7XG4gICAqIC0gcm93IDE6IGFsbCB0aGUgVHVlc2RheXM7XG4gICAqIC0gLi4uXG4gICAqIC0gcm93IDc6IGFsbCB0aGUgU3VuZGF5cy5cbiAgICovXG4gIHByaXZhdGUgX2NyZWF0ZVdlZWtSb3dzVmVydGljYWwodmFsdWU6IFQsIGRheXNJbk1vbnRoOiBudW1iZXIsIHdlZWtPZmZzZXQ6IG51bWJlcik6IERheTxUPltdW10ge1xuICAgIGNvbnN0IHdlZWtzOiBEYXk8VD5bXVtdID0gQXJyYXkuZnJvbSh7IGxlbmd0aDogREFZU19QRVJfUk9XIH0sICgpID0+IFtdKTtcbiAgICBmb3IgKGxldCBpID0gMCwgY2VsbCA9IHdlZWtPZmZzZXQ7IGkgPCBkYXlzSW5Nb250aDsgaSsrLCBjZWxsKyspIHtcbiAgICAgIGlmIChjZWxsID09PSBEQVlTX1BFUl9ST1cpIHtcbiAgICAgICAgY2VsbCA9IDA7XG4gICAgICB9XG4gICAgICBjb25zdCBkYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih2YWx1ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKHZhbHVlKSxcbiAgICAgICAgaSArIDEsXG4gICAgICApITtcbiAgICAgIHdlZWtzW2NlbGxdLnB1c2godGhpcy5fbWFwRGF0ZVRvRGF5KGRhdGUpKTtcbiAgICB9XG4gICAgcmV0dXJuIHdlZWtzO1xuICB9XG5cbiAgcHJpdmF0ZSBfbWFwRGF0ZVRvRGF5KGRhdGU6IFQpOiBEYXk8VD4ge1xuICAgIGNvbnN0IGlzb0RhdGUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZGF0ZSk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHZhbHVlOiBpc29EYXRlLFxuICAgICAgZGF0ZVZhbHVlOiBkYXRlLFxuICAgICAgZGF5VmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKGRhdGUpKSxcbiAgICAgIG1vbnRoVmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXRNb250aChkYXRlKSksXG4gICAgICB5ZWFyVmFsdWU6IFN0cmluZyh0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKGRhdGUpKSxcbiAgICAgIC8vIFRPRE86IEltcHJvdmUgcGVyZm9ybWFuY2Ugb2YgdGhpcywgYnkga2VlcGluZyB0cmFjayBvZiB0aGVcbiAgICAgIC8vIHdlZWsgbnVtYmVyIHdoaWxlIGl0ZXJhdGluZyB0aHJvdWdoIHRoZSBkYXlzLlxuICAgICAgd2Vla1ZhbHVlOiB0aGlzLl9nZXRXZWVrKGRhdGUpLFxuICAgICAgd2Vla0RheVZhbHVlOiB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWsoZGF0ZSksXG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgX2dldFdlZWsoZGF0ZTogVCk6IG51bWJlciB7XG4gICAgY29uc3QgZmlyc3REYXlPZlllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIoZGF0ZSksIDEsIDEpO1xuICAgIGNvbnN0IHdlZWtkYXkgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXlPZldlZWsoZmlyc3REYXlPZlllYXIpO1xuXG4gICAgbGV0IHdlZWtJbmRleCA9IFtNT05EQVksIFRVRVNEQVksIFdFRE5FU0RBWSwgVEhVUlNEQVldLmluY2x1ZGVzKHdlZWtkYXkpID8gMSA6IDA7XG4gICAgbGV0IHdlZWtTdGFydCA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyhcbiAgICAgIGZpcnN0RGF5T2ZZZWFyLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0Rmlyc3RXZWVrT2Zmc2V0KGZpcnN0RGF5T2ZZZWFyKSAqIC0xLFxuICAgICk7XG4gICAgd2hpbGUgKHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHdlZWtTdGFydCwgZGF0ZSkgPD0gMCkge1xuICAgICAgY29uc3Qgd2Vla0VuZCA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyRGF5cyh3ZWVrU3RhcnQsIDYpO1xuICAgICAgaWYgKHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKGRhdGUsIHdlZWtFbmQpIDw9IDApIHtcbiAgICAgICAgcmV0dXJuIHdlZWtJbmRleCA+IDBcbiAgICAgICAgICA/IHdlZWtJbmRleFxuICAgICAgICAgIDogdGhpcy5fZ2V0V2Vlayh0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZmlyc3REYXlPZlllYXIsIC0xKSk7XG4gICAgICB9XG4gICAgICB3ZWVrU3RhcnQgPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMod2Vla1N0YXJ0LCBEQVlTX1BFUl9ST1cpO1xuICAgICAgd2Vla0luZGV4Kys7XG4gICAgfVxuXG4gICAgdGhyb3cgbmV3IEVycm9yKCdUaGUgcHJvdmlkZWQgZGF0ZSBpcyBpbnZhbGlkJyk7XG4gIH1cblxuICAvKiogRm9yY2UgdGhlIGNvbnZlcnNpb24gdG8gSVNPODYwMSBmb3JtYXR0ZWQgc3RyaW5nLiAqL1xuICBwcml2YXRlIF9tYXBWYWx1ZVRvSVNPRGF0ZSh2YWx1ZTogc3RyaW5nIHwgRGF0ZSk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycgPyB2YWx1ZSA6IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMSh2YWx1ZSBhcyBUKTtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSByb3dzIGZvciB0aGUgbW9udGggc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZU1vbnRoUm93cygpOiB2b2lkIHtcbiAgICBjb25zdCBtb250aHM6IE1vbnRoQ2VsbFtdID0gbmV3IEFycmF5KDEyKS5maWxsKG51bGwpLm1hcChcbiAgICAgIChfLCBpOiBudW1iZXIpOiBNb250aENlbGwgPT4gKHtcbiAgICAgICAgdmFsdWU6IFN0cmluZyhpICsgMSkucGFkU3RhcnQoMiwgJzAnKSxcbiAgICAgICAgbW9udGhWYWx1ZTogaSArIDEsXG4gICAgICB9KSxcbiAgICApO1xuICAgIGNvbnN0IHJvd3M6IG51bWJlciA9IDEyIC8gTU9OVEhTX1BFUl9ST1c7XG4gICAgY29uc3QgbW9udGhBcnJheTogTW9udGhDZWxsW11bXSA9IFtdO1xuICAgIGZvciAobGV0IGk6IG51bWJlciA9IDA7IGkgPCByb3dzOyBpKyspIHtcbiAgICAgIG1vbnRoQXJyYXkucHVzaChtb250aHMuc2xpY2UoTU9OVEhTX1BFUl9ST1cgKiBpLCBNT05USFNfUEVSX1JPVyAqIChpICsgMSkpKTtcbiAgICB9XG4gICAgdGhpcy5fbW9udGhzID0gbW9udGhBcnJheTtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSByb3dzIGZvciB0aGUgeWVhciBzZWxlY3Rpb24gdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlWWVhclJvd3Mob2Zmc2V0OiBudW1iZXIgPSAwKTogbnVtYmVyW11bXSB7XG4gICAgY29uc3Qgc3RhcnRWYWx1ZVllYXJWaWV3OiBudW1iZXIgPSB0aGlzLl9nZXRTdGFydFZhbHVlWWVhclZpZXcoKTtcbiAgICBjb25zdCBhbGxZZWFyczogbnVtYmVyW10gPSBuZXcgQXJyYXkoWUVBUlNfUEVSX1BBR0UpXG4gICAgICAuZmlsbCgwKVxuICAgICAgLm1hcCgoXywgaTogbnVtYmVyKSA9PiBzdGFydFZhbHVlWWVhclZpZXcgKyBvZmZzZXQgKyBpKTtcbiAgICBjb25zdCByb3dzOiBudW1iZXIgPSBZRUFSU19QRVJfUEFHRSAvIFlFQVJTX1BFUl9ST1c7XG4gICAgY29uc3QgeWVhckFycmF5OiBudW1iZXJbXVtdID0gW107XG4gICAgZm9yIChsZXQgaTogbnVtYmVyID0gMDsgaSA8IHJvd3M7IGkrKykge1xuICAgICAgeWVhckFycmF5LnB1c2goYWxsWWVhcnMuc2xpY2UoWUVBUlNfUEVSX1JPVyAqIGksIFlFQVJTX1BFUl9ST1cgKiAoaSArIDEpKSk7XG4gICAgfVxuICAgIHJldHVybiB5ZWFyQXJyYXk7XG4gIH1cblxuICAvKipcbiAgICogQ2FsY3VsYXRlcyB0aGUgZmlyc3QgeWVhciB0aGF0IHdpbGwgYmUgc2hvd24gaW4gdGhlIHllYXIgc2VsZWN0aW9uIHBhbmVsLlxuICAgKiBJZiBgbWluRGF0ZWAgYW5kIGBtYXhEYXRlYCBhcmUgYm90aCBudWxsLCB0aGUgc3RhcnRpbmcgeWVhciBpcyBjYWxjdWxhdGVkIGFzXG4gICAqIHRoZSBtdWx0aXBsZSBvZiBZRUFSU19QRVJfUEFHRSBjbG9zZXN0IHRvIGFuZCBsZXNzIHRoYW4gYWN0aXZlRGF0ZSxcbiAgICogZS5nLiwgd2l0aCBgWUVBUlNfUEVSX1BBR0VgID0gMjQgYW5kIGBhY3RpdmVEYXRlYCA9IDIwMjAsIHRoZSBmdW5jdGlvbiB3aWxsIHJldHVybiAyMDE2ICgyNCAqIDgzKSxcbiAgICogd2hpbGUgd2l0aCBgYWN0aXZlRGF0ZWAgPSAyMDAwLCB0aGUgZnVuY3Rpb24gd2lsbCByZXR1cm4gMTk5MiAoMjQgKiA4MikuXG4gICAqIElmIGBtaW5EYXRlYCBpcyBub3QgbnVsbCwgaXQgcmV0dXJucyB0aGUgY29ycmVzcG9uZGluZyB5ZWFyOyBpZiBgbWF4RGF0ZWAgaXMgbm90IG51bGwsXG4gICAqIGl0IHJldHVybnMgdGhlIGNvcnJlc3BvbmRpbmcgeWVhciBtaW51cyBgWUVBUlNfUEVSX1BBR0VgLCBzbyB0aGF0IHRoZSBgbWF4RGF0ZWAgaXMgdGhlIGxhc3QgcmVuZGVyZWQgeWVhci5cbiAgICogSWYgYm90aCBhcmUgbm90IG51bGwsIGBtYXhEYXRlYCBoYXMgcHJpb3JpdHkgb3ZlciBgbWluRGF0ZWAuXG4gICAqL1xuICBwcml2YXRlIF9nZXRTdGFydFZhbHVlWWVhclZpZXcoKTogbnVtYmVyIHtcbiAgICBsZXQgc3RhcnRpbmdZZWFyID0gMDtcbiAgICBpZiAodGhpcy5tYXgpIHtcbiAgICAgIHN0YXJ0aW5nWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5tYXgpIC0gWUVBUlNfUEVSX1BBR0UgKyAxO1xuICAgIH0gZWxzZSBpZiAodGhpcy5taW4pIHtcbiAgICAgIHN0YXJ0aW5nWWVhciA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIodGhpcy5taW4pO1xuICAgIH1cbiAgICBjb25zdCBhY3RpdmVZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9hY3RpdmVEYXRlKTtcbiAgICByZXR1cm4gKFxuICAgICAgYWN0aXZlWWVhciAtXG4gICAgICAoKCgoYWN0aXZlWWVhciAtIHN0YXJ0aW5nWWVhcikgJSBZRUFSU19QRVJfUEFHRSkgKyBZRUFSU19QRVJfUEFHRSkgJSBZRUFSU19QRVJfUEFHRSlcbiAgICApO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiBkYXRlIGlzIHdpdGhpbiB0aGUgbWluLW1heCByYW5nZS4gKi9cbiAgcHJpdmF0ZSBfaXNEYXlJblJhbmdlKGRhdGVTdHJpbmc6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5taW4gJiYgIXRoaXMubWF4KSB7XG4gICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG4gICAgY29uc3QgZGF0ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKGRhdGVTdHJpbmcpITtcbiAgICByZXR1cm4gdGhpcy5fZGF0ZUFkYXB0ZXIuc2FtZURhdGUoZGF0ZSwgdGhpcy5fZGF0ZUFkYXB0ZXIuY2xhbXBEYXRlKGRhdGUsIHRoaXMubWluLCB0aGlzLm1heCkpO1xuICB9XG5cbiAgLyoqIEVtaXRzIHRoZSBzZWxlY3RlZCBkYXRlIGFuZCBzZXRzIGl0IGludGVybmFsbHkuICovXG4gIHByaXZhdGUgX3NlbGVjdERhdGUoZGF5OiBUKTogdm9pZCB7XG4gICAgdGhpcy5fY2hvc2VuTW9udGggPSB1bmRlZmluZWQ7XG4gICAgdGhpcy5fc2V0Q2hvc2VuWWVhcigpO1xuICAgIGlmICh0aGlzLm11bHRpcGxlKSB7XG4gICAgICAvLyBDaGVjayBpZiBfc2VsZWN0ZWQgaGFzIGVsZW1lbnRzXG4gICAgICBpZiAodGhpcy5fc2VsZWN0ZWQgJiYgKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkubGVuZ3RoID4gMCkge1xuICAgICAgICBjb25zdCBpbmRleE9mU2VsZWN0ZWREYXk6IG51bWJlciA9ICh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLmZpbmRJbmRleChcbiAgICAgICAgICAoc2VsKSA9PiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShzZWwsIGRheSkgPT09IDAsXG4gICAgICAgICk7XG4gICAgICAgIC8vIElmIHRoZSBzZWxlY3RlZCBkYXRlIGlzIGFscmVhZHkgaW4gdGhlIF9zZWxlY3RlZCBhcnJheSwgcmVtb3ZlIGl0LCBvdGhlcndpc2UgYWRkIGl0XG4gICAgICAgIGlmIChpbmRleE9mU2VsZWN0ZWREYXkgIT09IC0xKSB7XG4gICAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSAodGhpcy5fc2VsZWN0ZWQgYXMgVFtdKS5maWx0ZXIoKF8sIGkpID0+IGkgIT09IGluZGV4T2ZTZWxlY3RlZERheSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBbLi4uKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSksIGRheV07XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIElmIF9zZWxlY3RlZCBpcyBlbXB0eSwgc2V0IGl0XG4gICAgICAgIHRoaXMuX3NlbGVjdGVkID0gW2RheV07XG4gICAgICB9XG4gICAgICB0aGlzLl9lbWl0RGF0ZVNlbGVjdGVkRXZlbnQodGhpcy5fc2VsZWN0ZWQubWFwKChlKSA9PiB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShlKSEpKTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gSW4gc2luZ2xlIHNlbGVjdGlvbiwgY2hlY2sgaWYgdGhlIGRheSBpcyBhbHJlYWR5IHNlbGVjdGVkXG4gICAgICBpZiAoIXRoaXMuX3NlbGVjdGVkIHx8IHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKHRoaXMuX3NlbGVjdGVkIGFzIFQsIGRheSkgIT09IDApIHtcbiAgICAgICAgdGhpcy5fc2VsZWN0ZWQgPSBkYXk7XG4gICAgICAgIHRoaXMuX2VtaXREYXRlU2VsZWN0ZWRFdmVudCh0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShkYXkpISk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEhhbmRsZSBtdWx0aXBsZSBkYXRlcyBzZWxlY3Rpb24gdmlhIHdlZWtOdW1iZXIgLyB3ZWVrRGF5IGJ1dHRvbnM6XG4gICAqIC0gaWYgQ21kIG9yIEN0cmwgYXJlIHByZXNzZWQsIGFkZCB0aGUgbmV3IGRhdGUgdG8gdGhlIGN1cnJlbnQgb25lcztcbiAgICogLSBpZiBub3QsXG4gICAqICAgICAtIGlmIHRoZSBuZXcgZGF0ZXMgYXJlIHRoZSBzYW1lIG9mIHRoZSBjdXJyZW50IG9uZXMsIGl0IG1lYW5zIHRoYXQgdGhlIHNhbWUgYnV0dG9uIGhhcyBiZWVuIGNsaWNrZWQgdHdpY2UsIHNvIGRvIG5vdGhpbmc7XG4gICAqICAgICAtIGlmIG5vdCwgdGhlIHNlbGVjdGVkIGRhdGVzIGFyZSB0aGUgbmV3IG9uZXMuXG4gICAqL1xuICBwcml2YXRlIF9zZWxlY3RNdWx0aXBsZURhdGVzKGRheXM6IERheTxUPltdKTogdm9pZCB7XG4gICAgLy8gRmlsdGVyIGRpc2FibGVkIGRheXMgYnkgbWF0Y2hpbmcgdGhlIHByb3ZpZGVkIGBkYXlzYCBwYXJhbWV0ZXIgYWdhaW5zdCB0aGUgZW5hYmxlZCBjZWxscy5cbiAgICAvLyBTaW5jZSB0aGUgYnV0dG9ucycgdmFsdWUgaXMgc2V0IHRvIHRoZSBEYXkncyBpbnRlcmZhY2UgdmFsdWUgKElTTyBzdHJpbmcpLCB0aGVyZSdzIG5vIG5lZWQgdG8gZGVzZXJpYWxpemUgaXQuXG4gICAgY29uc3QgZW5hYmxlZERheXM6IHN0cmluZ1tdID0gKHRoaXMuX2NlbGxzIGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdKVxuICAgICAgLmZpbHRlcigoZSkgPT4gIWUuZGlzYWJsZWQpXG4gICAgICAubWFwKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkpO1xuICAgIGNvbnN0IGRheXNUb0FkZDogc3RyaW5nW10gPSBkYXlzXG4gICAgICAubWFwKChlOiBEYXk8VD4pID0+IGUudmFsdWUpXG4gICAgICAuZmlsdGVyKChpc29EYXRlOiBzdHJpbmcpID0+IGVuYWJsZWREYXlzLmluY2x1ZGVzKGlzb0RhdGUpKTtcbiAgICBjb25zdCBkYXlzVG9BZGRTZXQgPSBuZXcgU2V0KGRheXNUb0FkZCk7XG4gICAgY29uc3Qgc2VsZWN0ZWRTZXQgPSBuZXcgU2V0KCh0aGlzLl9zZWxlY3RlZCBhcyBUW10pLm1hcCgocykgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKHMpKSk7XG4gICAgY29uc3Qgc2VsU3RyaW5ncyA9IHRoaXMuX3VwZGF0ZVNlbGVjdGVkV2l0aE11bHRpcGxlRGF0ZXMoZGF5c1RvQWRkLCBkYXlzVG9BZGRTZXQsIHNlbGVjdGVkU2V0KTtcbiAgICB0aGlzLl9zZWxlY3RlZCA9IHNlbFN0cmluZ3MubWFwKChzKSA9PiB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShzKSEpO1xuXG4gICAgdGhpcy5fZW1pdERhdGVTZWxlY3RlZEV2ZW50KHRoaXMuX3NlbGVjdGVkLm1hcCgoZSkgPT4gdGhpcy5fZGF0ZUFkYXB0ZXIuZGVzZXJpYWxpemUoZSkhKSk7XG4gIH1cblxuICAvKipcbiAgICogRW1pdHMgdGhlIGRhdGVzZWxlY3RlZCBldmVudCBnaXZlbiB0aGUgZGV0YWlsIChhcyBUIG9yIFRbXSBiYXNlZCBvbiB0aGUgdmFsdWUgb2YgdGhlIG11bHRpcGxlIGZsYWcpLlxuICAgKi9cbiAgcHJpdmF0ZSBfZW1pdERhdGVTZWxlY3RlZEV2ZW50KGRldGFpbDogVCB8IFRbXSk6IHZvaWQge1xuICAgIC8qKiBAdHlwZSB7Q3VzdG9tRXZlbnQ8VCB8IFRbXT59IEV2ZW50IGVtaXR0ZWQgb24gZGF0ZSBzZWxlY3Rpb24uICovXG4gICAgdGhpcy5kaXNwYXRjaEV2ZW50KFxuICAgICAgbmV3IEN1c3RvbUV2ZW50PFQgfCBUW10+KCdkYXRlc2VsZWN0ZWQnLCB7XG4gICAgICAgIGRldGFpbCxcbiAgICAgICAgY29tcG9zZWQ6IHRydWUsXG4gICAgICAgIGJ1YmJsZXM6IHRydWUsXG4gICAgICB9KSxcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBfZW1pdE1vbnRoQ2hhbmdlKCk6IHZvaWQge1xuICAgIC8vIEZJWE1FOiB0aGUgbmFtZSBvZiB0aGlzIHZhcmlhYmxlIGFwcGVhcnMgYXMgZXZlbnQgbmFtZSBpbiB0aGUgcmVhZG1lXG4gICAgLy8gIGR1ZSB0byBhIGJ1ZyBpbiB0aGUgY3VzdG9tLWVsZW1lbnRzLW1hbmlmZXN0IGxpYnJhcnkuXG4gICAgLy8gIGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuLXdjL2N1c3RvbS1lbGVtZW50cy1tYW5pZmVzdC9pc3N1ZXMvMTQ5XG4gICAgY29uc3QgbW9udGhjaGFuZ2UgPSAodGhpcy53aWRlID8gWy4uLnRoaXMuX3dlZWtzLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc10gOiB0aGlzLl93ZWVrcylcbiAgICAgIC5mbGF0KClcbiAgICAgIC5zb3J0KChhLCBiKSA9PiBhLnZhbHVlLmxvY2FsZUNvbXBhcmUoYi52YWx1ZSkpIGFzIERheVtdO1xuICAgIC8qKlxuICAgICAqIEB0eXBlIHtTYmJNb250aENoYW5nZUV2ZW50fVxuICAgICAqIEVtaXRzIHdoZW4gdGhlIG1vbnRoIGNoYW5nZXMuXG4gICAgICogVGhlIGByYW5nZWAgcHJvcGVydHkgY29udGFpbnMgdGhlIGRheXMgYXJyYXkgb2YgdGhlIGNob3NlbiBtb250aC5cbiAgICAgKi9cbiAgICB0aGlzLmRpc3BhdGNoRXZlbnQobmV3IFNiYk1vbnRoQ2hhbmdlRXZlbnQobW9udGhjaGFuZ2UpKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBJbiBjYXNlIG9mIG11bHRpcGxlIHNlbGVjdGlvbiwgbmV3bHkgYWRkZWQgZGF5cyBtdXN0IGJlIGFkZGVkIHRvIHRoZSBleGlzdGluZyBvbmVzLCB3aXRob3V0IGR1cGxpY2F0aW9uLlxuICAgKiBJZiB0aGUgZGF5cyB0byBhZGQgYXJlIGV4YWN0bHkgdGhlIHNhbWUgYXMgdGhlIHNlbGVjdGVkIG9uZXMsIHRoZSBzZXQgbXVzdCBiZSBlbXB0aWVkLlxuICAgKi9cbiAgcHJpdmF0ZSBfdXBkYXRlU2VsZWN0ZWRXaXRoTXVsdGlwbGVEYXRlcyhcbiAgICBkYXlzVG9BZGQ6IHN0cmluZ1tdLFxuICAgIGRheXNUb0FkZFNldDogU2V0PHN0cmluZz4sXG4gICAgc2VsZWN0ZWRTZXQ6IFNldDxzdHJpbmc+LFxuICApOiBzdHJpbmdbXSB7XG4gICAgaWYgKGRheXNUb0FkZC5ldmVyeSgoZGF5OiBzdHJpbmcpID0+IHNlbGVjdGVkU2V0LmhhcyhkYXkpKSkge1xuICAgICAgZGF5c1RvQWRkU2V0LmZvckVhY2goKGRheTogc3RyaW5nKSA9PiBzZWxlY3RlZFNldC5kZWxldGUoZGF5KSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGRheXNUb0FkZFNldC5mb3JFYWNoKChkYXk6IHN0cmluZykgPT4gc2VsZWN0ZWRTZXQuYWRkKGRheSkpO1xuICAgIH1cbiAgICByZXR1cm4gQXJyYXkuZnJvbShzZWxlY3RlZFNldCk7XG4gIH1cblxuICBwcml2YXRlIF9zZXRDaG9zZW5ZZWFyKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnZpZXcgPT09ICdtb250aCcpIHtcbiAgICAgIGxldCBzZWxlY3RlZERhdGU6IFQgfCB1bmRlZmluZWQ7XG4gICAgICBpZiAodGhpcy5tdWx0aXBsZSkge1xuICAgICAgICBzZWxlY3RlZERhdGUgPSAodGhpcy5zZWxlY3RlZCBhcyBUW10pLmF0KC0xKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNlbGVjdGVkRGF0ZSA9IHRoaXMuc2VsZWN0ZWQgYXMgVDtcbiAgICAgIH1cbiAgICAgIHRoaXMuX2Nob3NlblllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXRZZWFyKHNlbGVjdGVkRGF0ZSA/PyB0aGlzLl9kYXRlQWRhcHRlci50b2RheSgpKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5fY2hvc2VuWWVhciA9IHVuZGVmaW5lZDtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9hc3NpZ25BY3RpdmVEYXRlKGRhdGU6IFQpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5taW4gJiYgdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUodGhpcy5taW4sIGRhdGUpID4gMCkge1xuICAgICAgdGhpcy5fYWN0aXZlRGF0ZSA9IHRoaXMubWluO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBpZiAodGhpcy5tYXggJiYgdGhpcy5fZGF0ZUFkYXB0ZXIuY29tcGFyZURhdGUodGhpcy5tYXgsIGRhdGUpIDwgMCkge1xuICAgICAgdGhpcy5fYWN0aXZlRGF0ZSA9IHRoaXMubWF4O1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLl9hY3RpdmVEYXRlID0gZGF0ZTtcbiAgfVxuXG4gIC8qKiBHb2VzIHRvIHRoZSBtb250aCBpZGVudGlmaWVkIGJ5IHRoZSBzaGlmdC4gKi9cbiAgcHJpdmF0ZSBfZ29Ub0RpZmZlcmVudE1vbnRoKG1vbnRoczogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5faW5pdCh0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhck1vbnRocyh0aGlzLl9hY3RpdmVEYXRlLCBtb250aHMpKTtcbiAgICB0aGlzLl9lbWl0TW9udGhDaGFuZ2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX2dvVG9EaWZmZXJlbnRZZWFyKHllYXJzOiBudW1iZXIpOiB2b2lkIHtcbiAgICB0aGlzLl9jaG9zZW5ZZWFyISArPSB5ZWFycztcbiAgICAvLyBDYW4ndCB1c2UgYF9hc3NpZ25BY3RpdmVEYXRlKC4uLilgIGhlcmUsIGJlY2F1c2UgaXQgd2lsbCBzZXQgaXQgdG8gbWluL21heCB2YWx1ZSBpZiBhcmd1bWVudCBpcyBvdXQgb2YgcmFuZ2VcbiAgICB0aGlzLl9hY3RpdmVEYXRlID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2Nob3NlblllYXIhLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICk7XG4gICAgdGhpcy5faW5pdCgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfZ29Ub0RpZmZlcmVudFllYXJSYW5nZSh5ZWFyczogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5faW5pdCh0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhclllYXJzKHRoaXMuX2FjdGl2ZURhdGUsIHllYXJzKSk7XG4gIH1cblxuICBwcml2YXRlIF9wcmV2RGlzYWJsZWQocHJldkRhdGU6IFQpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMubWluKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9kYXRlQWRhcHRlci5jb21wYXJlRGF0ZShwcmV2RGF0ZSwgdGhpcy5taW4pIDwgMDtcbiAgfVxuXG4gIHByaXZhdGUgX25leHREaXNhYmxlZChuZXh0RGF0ZTogVCk6IGJvb2xlYW4ge1xuICAgIGlmICghdGhpcy5tYXgpIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2RhdGVBZGFwdGVyLmNvbXBhcmVEYXRlKG5leHREYXRlLCB0aGlzLm1heCkgPiAwO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJwcmV2aW91cyBtb250aFwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX3ByZXZpb3VzTW9udGhEaXNhYmxlZCgpOiBib29sZWFuIHtcbiAgICBjb25zdCBwcmV2TW9udGggPSB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoXG4gICAgICB0aGlzLl9hY3RpdmVEYXRlLFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0RGF0ZSh0aGlzLl9hY3RpdmVEYXRlKSAqIC0xLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX3ByZXZEaXNhYmxlZChwcmV2TW9udGgpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJuZXh0IG1vbnRoXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfbmV4dE1vbnRoRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgbGV0IG5leHRNb250aCA9IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIHRoaXMuX3dpZGUgPyAyIDogMSk7XG4gICAgbmV4dE1vbnRoID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldFllYXIobmV4dE1vbnRoKSxcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKG5leHRNb250aCksXG4gICAgICAxLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX25leHREaXNhYmxlZChuZXh0TW9udGgpO1xuICB9XG5cbiAgLyoqIENoZWNrcyBpZiB0aGUgXCJwcmV2aW91cyB5ZWFyXCIgYnV0dG9uIHNob3VsZCBiZSBkaXNhYmxlZC4gKi9cbiAgcHJpdmF0ZSBfcHJldmlvdXNZZWFyRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgcHJldlllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9hY3RpdmVEYXRlKSAtIDEsXG4gICAgICAxMixcbiAgICAgIDMxLFxuICAgICk7XG4gICAgcmV0dXJuIHRoaXMuX3ByZXZEaXNhYmxlZChwcmV2WWVhcik7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcIm5leHQgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQuICovXG4gIHByaXZhdGUgX25leHRZZWFyRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgbmV4dFllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcih0aGlzLl9hY3RpdmVEYXRlKSArICh0aGlzLl93aWRlID8gMiA6IDEpLFxuICAgICAgMSxcbiAgICAgIDEsXG4gICAgKTtcbiAgICByZXR1cm4gdGhpcy5fbmV4dERpc2FibGVkKG5leHRZZWFyKTtcbiAgfVxuXG4gIC8qKiBDaGVja3MgaWYgdGhlIFwicHJldmlvdXMgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQgaW4geWVhciB2aWV3LiAqL1xuICBwcml2YXRlIF9wcmV2aW91c1llYXJSYW5nZURpc2FibGVkKCk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IHByZXZZZWFyID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh0aGlzLl95ZWFyc1swXVswXSAtIDEsIDEyLCAzMSk7XG4gICAgcmV0dXJuIHRoaXMuX3ByZXZEaXNhYmxlZChwcmV2WWVhcik7XG4gIH1cblxuICAvKiogQ2hlY2tzIGlmIHRoZSBcIm5leHQgeWVhclwiIGJ1dHRvbiBzaG91bGQgYmUgZGlzYWJsZWQgaW4geWVhciB2aWV3LiAqL1xuICBwcml2YXRlIF9uZXh0WWVhclJhbmdlRGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgeWVhcnMgPSB0aGlzLl93aWRlID8gdGhpcy5fbmV4dE1vbnRoWWVhcnMgOiB0aGlzLl95ZWFycztcbiAgICBjb25zdCBsYXN0WWVhclJhbmdlID0geWVhcnNbeWVhcnMubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgbGFzdFllYXIgPSBsYXN0WWVhclJhbmdlW2xhc3RZZWFyUmFuZ2UubGVuZ3RoIC0gMV07XG4gICAgY29uc3QgbmV4dFllYXIgPSB0aGlzLl9kYXRlQWRhcHRlci5jcmVhdGVEYXRlKGxhc3RZZWFyICsgMSwgMSwgMSk7XG4gICAgcmV0dXJuIHRoaXMuX25leHREaXNhYmxlZChuZXh0WWVhcik7XG4gIH1cblxuICBwcml2YXRlIF9oYW5kbGVUYWJsZUJsdXIoZXZlbnRUYXJnZXQ6IEhUTUxFbGVtZW50KTogdm9pZCB7XG4gICAgaWYgKGV2ZW50VGFyZ2V0Py5sb2NhbE5hbWUgIT09ICdzYmItY2FsZW5kYXItZGF5Jykge1xuICAgICAgdGhpcy5fc2V0VGFiSW5kZXgoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9zZXRUYWJJbmRleCgpOiB2b2lkIHtcbiAgICBBcnJheS5mcm9tKHRoaXMuX2NlbGxzLmZpbHRlcigoZSkgPT4gZS50YWJJbmRleCA9PT0gMCkgPz8gW10pLmZvckVhY2goXG4gICAgICAoZGF5KSA9PiAoZGF5LnRhYkluZGV4ID0gLTEpLFxuICAgICk7XG4gICAgY29uc3QgZmlyc3RGb2N1c2FibGUgPSB0aGlzLl9nZXRGaXJzdEZvY3VzYWJsZSgpO1xuICAgIGlmIChmaXJzdEZvY3VzYWJsZSkge1xuICAgICAgZmlyc3RGb2N1c2FibGUudGFiSW5kZXggPSAwO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBHZXQgdGhlIGVsZW1lbnQgaW4gdGhlIGNhbGVuZGFyIHRvIGFzc2lnbiBmb2N1cy4gKi9cbiAgcHJpdmF0ZSBfZ2V0Rmlyc3RGb2N1c2FibGUoKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQgfCBudWxsIHtcbiAgICBpZiAodGhpcy5fY2FsZW5kYXJWaWV3ID09PSAnZGF5Jykge1xuICAgICAgY29uc3Qgc2VsZWN0ZWRPckN1cnJlbnQgPVxuICAgICAgICB0aGlzLl9jZWxscy5maW5kKChlKSA9PiBlLm1hdGNoZXMoJzpzdGF0ZShzZWxlY3RlZCknKSkgPz9cbiAgICAgICAgdGhpcy5fY2VsbHMuZmluZCgoZSkgPT4gZS5tYXRjaGVzKCc6c3RhdGUoY3VycmVudCknKSk7XG4gICAgICByZXR1cm4gc2VsZWN0ZWRPckN1cnJlbnQgJiYgIXNlbGVjdGVkT3JDdXJyZW50LmRpc2FibGVkXG4gICAgICAgID8gc2VsZWN0ZWRPckN1cnJlbnRcbiAgICAgICAgOiB0aGlzLl9nZXRGaXJzdEZvY3VzYWJsZURheSgpO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBzZWxlY3RlZE9yQ3VycmVudCA9XG4gICAgICAgIHRoaXMuc2hhZG93Um9vdD8ucXVlcnlTZWxlY3RvcjxTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudD4oJzpzdGF0ZShzZWxlY3RlZCknKSA/P1xuICAgICAgICB0aGlzLnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3I8U2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ+KCc6c3RhdGUoY3VycmVudCknKTtcbiAgICAgIHJldHVybiBzZWxlY3RlZE9yQ3VycmVudCAmJiAhc2VsZWN0ZWRPckN1cnJlbnQuZGlzYWJsZWRcbiAgICAgICAgPyBzZWxlY3RlZE9yQ3VycmVudFxuICAgICAgICA6IHRoaXMuc2hhZG93Um9vdCEucXVlcnlTZWxlY3RvcjxTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudD4oXG4gICAgICAgICAgICBgc2JiLWNhbGVuZGFyLSR7dGhpcy5fY2FsZW5kYXJWaWV3fTpub3QoW2Rpc2FibGVkXSlgLFxuICAgICAgICAgICk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEluIGBkYXlgIHZpZXcgaW4gYHZlcnRpY2FsYCBvcmllbnRhdGlvbixcbiAgICogaWYgdGhlIGZpcnN0IG9mIHRoZSBtb250aCBpcyBub3QgYSBNb25kYXksIGl0IGlzIG5vdCB0aGUgZmlyc3QgcmVuZGVyZWQgZWxlbWVudCBpbiB0aGUgdGFibGUsXG4gICAqIHNvIGB0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3IoJ3NiYi1jYWxlbmRhci1kYXk6bm90KFtkaXNhYmxlZF0pJylgIHdpbGwgcmV0dXJuIGEgd3JvbmcgdmFsdWUuXG4gICAqXG4gICAqIFRvIHNvbHZlIHRoaXMsIHRoZSBlbGVtZW50IHdpdGggdGhlIGxvd2VzdCBgdmFsdWVgIGlzIHRha2VuIChJU08gU3RyaW5nIGFyZSBvcmRlcmVkKS5cbiAgICovXG4gIHByaXZhdGUgX2dldEZpcnN0Rm9jdXNhYmxlRGF5KCk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB8IG51bGwge1xuICAgIGNvbnN0IGNlbGxzID0gdGhpcy5fY2VsbHMgYXMgU2JiQ2FsZW5kYXJEYXlFbGVtZW50W107XG4gICAgY29uc3QgZGF5c0luVmlldyA9IGNlbGxzLmZpbHRlcigoZSkgPT4gIWUuZGlzYWJsZWQpO1xuICAgIGlmICghZGF5c0luVmlldyB8fCBkYXlzSW5WaWV3Lmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IGZpcnN0RWxlbWVudCA9IGRheXNJblZpZXdcbiAgICAgICAgLm1hcCgoZSk6IHN0cmluZyA9PiB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoZS52YWx1ZSEgYXMgVCkpXG4gICAgICAgIC5zb3J0KClbMF07XG4gICAgICByZXR1cm4gY2VsbHMuZmluZCgoZSkgPT4gZS5tYXRjaGVzKGBbc2xvdD1cIiR7Zmlyc3RFbGVtZW50fVwiXWApKSEgPz8gbnVsbDtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9oYW5kbGVLZXlib2FyZEV2ZW50KGV2ZW50OiBLZXlib2FyZEV2ZW50LCBkYXk/OiBEYXk8VD4pOiB2b2lkIHtcbiAgICBpZiAoaXNBcnJvd0tleU9yUGFnZUtleXNQcmVzc2VkKGV2ZW50KSkge1xuICAgICAgZXZlbnQucHJldmVudERlZmF1bHQoKTtcbiAgICB9XG4gICAgLy8gR2V0cyB0aGUgY3VycmVudGx5IHJlbmRlcmVkIHRhYmxlJ3MgY2VsbDtcbiAgICAvLyB0aGV5IGNvdWxkIGJlIGRheXMsIG1vbnRocyBvciB5ZWFycyBiYXNlZCBvbiB0aGUgY3VycmVudCBzZWxlY3Rpb24gdmlldy5cbiAgICAvLyBJZiBgd2lkZWAgaXMgdHJ1ZSwgeWVhcnMgYXJlIGRvdWJsZWQgaW4gbnVtYmVyIGFuZCBkYXlzIGFyZSAocm91Z2hseSkgZG91YmxlZCB0b28sIGFmZmVjdGluZyB0aGUgYGluZGV4YCBjYWxjdWxhdGlvbi5cbiAgICBjb25zdCBjZWxscyA9IHRoaXMuX2NlbGxzO1xuICAgIGNvbnN0IGluZGV4OiBudW1iZXIgPSBjZWxscy5maW5kSW5kZXgoKGUpID0+IGUgPT09IGV2ZW50LnRhcmdldCk7XG4gICAgbGV0IG5leHRFbDogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQ7XG4gICAgaWYgKGRheSkge1xuICAgICAgbmV4dEVsID0gdGhpcy5fbmF2aWdhdGVCeUtleWJvYXJkRGF5VmlldyhldmVudCwgaW5kZXgsIGNlbGxzIGFzIFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLCBkYXkpO1xuICAgIH0gZWxzZSB7XG4gICAgICBuZXh0RWwgPSB0aGlzLl9uYXZpZ2F0ZUJ5S2V5Ym9hcmQoZXZlbnQsIGluZGV4LCBjZWxscyk7XG4gICAgfVxuICAgIGNvbnN0IGFjdGl2ZUVsID0gKHRoaXMuX2VuaGFuY2VkVmFyaWFudCA/IGRvY3VtZW50IDogdGhpcy5zaGFkb3dSb290ISlcbiAgICAgIC5hY3RpdmVFbGVtZW50IGFzIFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50O1xuICAgIGlmIChuZXh0RWwgIT09IGFjdGl2ZUVsKSB7XG4gICAgICBuZXh0RWwudGFiSW5kZXggPSAwO1xuICAgICAgbmV4dEVsPy5mb2N1cygpO1xuICAgICAgYWN0aXZlRWwudGFiSW5kZXggPSAtMTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9uYXZpZ2F0ZUJ5S2V5Ym9hcmREYXlWaWV3KFxuICAgIGV2dDogS2V5Ym9hcmRFdmVudCxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSxcbiAgICBkYXk6IERheTxUPixcbiAgKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBhcnJvd3NPZmZzZXQgPVxuICAgICAgdGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnXG4gICAgICAgID8geyBsZWZ0UmlnaHQ6IDEsIHVwRG93bjogREFZU19QRVJfUk9XIH1cbiAgICAgICAgOiB7IGxlZnRSaWdodDogREFZU19QRVJfUk9XLCB1cERvd246IDEgfTtcbiAgICBjb25zdCBvZmZzZXRGb3JWZXJ0aWNhbDogbnVtYmVyID1cbiAgICAgIGluZGV4IDwgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3RNb250aExlbmd0aFxuICAgICAgICA/IHRoaXMuX2tleWJvYXJkTmF2aWdhdGlvbkRheVZpZXdQYXJhbWV0ZXJzLmZpcnN0TW9udGhPZmZzZXRcbiAgICAgICAgOiB0aGlzLl9rZXlib2FyZE5hdmlnYXRpb25EYXlWaWV3UGFyYW1ldGVycy5zZWNvbmRNb250aE9mZnNldDtcblxuICAgIHN3aXRjaCAoZXZ0LmtleSkge1xuICAgICAgY2FzZSAnQXJyb3dVcCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgLWFycm93c09mZnNldC51cERvd24pO1xuICAgICAgY2FzZSAnQXJyb3dEb3duJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlBcnJvd3MoY2VsbHMsIGluZGV4LCBkYXkuZGF0ZVZhbHVlLCBhcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgIGNhc2UgJ0Fycm93TGVmdCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgLWFycm93c09mZnNldC5sZWZ0UmlnaHQpO1xuICAgICAgY2FzZSAnQXJyb3dSaWdodCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5QXJyb3dzKGNlbGxzLCBpbmRleCwgZGF5LmRhdGVWYWx1ZSwgYXJyb3dzT2Zmc2V0LmxlZnRSaWdodCk7XG4gICAgICBjYXNlICdQYWdlVXAnOiB7XG4gICAgICAgIGlmICh0aGlzLm9yaWVudGF0aW9uID09PSAnaG9yaXpvbnRhbCcpIHtcbiAgICAgICAgICBjb25zdCBmaXJzdE9mV2VlazogbnVtYmVyID0gK2RheS5kYXlWYWx1ZSAlIERBWVNfUEVSX1JPVyB8fCBEQVlTX1BFUl9ST1c7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9IGZpcnN0T2ZXZWVrIC0gK2RheS5kYXlWYWx1ZTtcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCBhcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCB3ZWVrTnVtYmVyOiBudW1iZXIgPSBNYXRoLmNlaWwoKCtkYXkuZGF5VmFsdWUgKyBvZmZzZXRGb3JWZXJ0aWNhbCkgLyBEQVlTX1BFUl9ST1cpO1xuICAgICAgICAgIGNvbnN0IGZpcnN0T2ZXZWVrOiBudW1iZXIgPSAod2Vla051bWJlciAtIDEpICogREFZU19QRVJfUk9XIC0gb2Zmc2V0Rm9yVmVydGljYWwgKyAxO1xuICAgICAgICAgIGNvbnN0IGRlbHRhOiBudW1iZXIgPSBmaXJzdE9mV2VlayAtICtkYXkuZGF5VmFsdWU7XG4gICAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlQYWdlVXBEb3duKGNlbGxzLCBpbmRleCwgZGF5LCBkZWx0YSwgYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIGNhc2UgJ1BhZ2VEb3duJzoge1xuICAgICAgICBpZiAodGhpcy5vcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnKSB7XG4gICAgICAgICAgY29uc3QgbW9udGhJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/IDEgOiArZGF5Lm1vbnRoVmFsdWUgKyAxO1xuICAgICAgICAgIGNvbnN0IHllYXJJbkJvdW5kcyA9ICtkYXkubW9udGhWYWx1ZSArIDEgPiAxMiA/ICtkYXkueWVhclZhbHVlICsgMSA6ICtkYXkueWVhclZhbHVlO1xuICAgICAgICAgIGNvbnN0IGZpcnN0TmV4dE1vbnRoOiBUID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh5ZWFySW5Cb3VuZHMsIG1vbnRoSW5Cb3VuZHMsIDEpO1xuICAgICAgICAgIGNvbnN0IGxhc3RPZk1vbnRoOiBudW1iZXIgPSB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKFxuICAgICAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGZpcnN0TmV4dE1vbnRoLCAtMSksXG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCBkZWx0YTogbnVtYmVyID1cbiAgICAgICAgICAgIE1hdGgudHJ1bmMoKGxhc3RPZk1vbnRoIC0gK2RheS5kYXlWYWx1ZSEpIC8gREFZU19QRVJfUk9XKSAqIERBWVNfUEVSX1JPVztcbiAgICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhLCAtYXJyb3dzT2Zmc2V0LnVwRG93bik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc3Qgd2Vla051bWJlcjogbnVtYmVyID0gTWF0aC5jZWlsKCgrZGF5LmRheVZhbHVlICsgb2Zmc2V0Rm9yVmVydGljYWwpIC8gREFZU19QRVJfUk9XKTtcbiAgICAgICAgICBjb25zdCBsYXN0T2ZXZWVrOiBudW1iZXIgPSB3ZWVrTnVtYmVyICogREFZU19QRVJfUk9XIC0gb2Zmc2V0Rm9yVmVydGljYWw7XG4gICAgICAgICAgY29uc3QgZGVsdGE6IG51bWJlciA9IGxhc3RPZldlZWsgLSArZGF5LmRheVZhbHVlO1xuICAgICAgICAgIHJldHVybiB0aGlzLl9maW5kRGF5UGFnZVVwRG93bihjZWxscywgaW5kZXgsIGRheSwgZGVsdGEsIC1hcnJvd3NPZmZzZXQudXBEb3duKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgY2FzZSAnSG9tZSc6IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlGaXJzdChjZWxscywgaW5kZXgsIGRheSwgMSk7XG4gICAgICB9XG4gICAgICBjYXNlICdFbmQnOiB7XG4gICAgICAgIGNvbnN0IG1vbnRoSW5Cb3VuZHMgPSArZGF5Lm1vbnRoVmFsdWUgKyAxID4gMTIgPyAxIDogK2RheS5tb250aFZhbHVlICsgMTtcbiAgICAgICAgY29uc3QgeWVhckluQm91bmRzID0gK2RheS5tb250aFZhbHVlICsgMSA+IDEyID8gK2RheS55ZWFyVmFsdWUgKyAxIDogK2RheS55ZWFyVmFsdWU7XG4gICAgICAgIGNvbnN0IGZpcnN0TmV4dE1vbnRoOiBUID0gdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZSh5ZWFySW5Cb3VuZHMsIG1vbnRoSW5Cb3VuZHMsIDEpO1xuICAgICAgICByZXR1cm4gdGhpcy5fZmluZERheUxhc3QoY2VsbHMsIGluZGV4LCBmaXJzdE5leHRNb250aCk7XG4gICAgICB9XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2lzRGF5T3V0T2ZWaWV3KGRhdGU6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHJldHVybiAoXG4gICAgICBkYXRlIDwgdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMuZmlyc3REYXlJblZpZXchIHx8XG4gICAgICBkYXRlID4gdGhpcy5fa2V5Ym9hcmROYXZpZ2F0aW9uRGF5Vmlld1BhcmFtZXRlcnMubGFzdERheUluVmlldyFcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBfZmluZERheUFycm93cyhcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBkYXRlOiBULFxuICAgIGRlbHRhOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGRhdGUsIGRlbHRhKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheUFycm93cyhjZWxscywgaW5kZXgsIHRoaXMuX2RhdGVBZGFwdGVyLmRlc2VyaWFsaXplKG5ld0RhdGVWYWx1ZSkhLCBkZWx0YSk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIHByaXZhdGUgX2ZpbmREYXlQYWdlVXBEb3duKFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckRheUVsZW1lbnRbXSxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGRheTogRGF5PFQ+LFxuICAgIGRlbHRhOiBudW1iZXIsXG4gICAgZGVsdGFJZkRpc2FibGVkOiBudW1iZXIsXG4gICk6IFNiYkNhbGVuZGFyRGF5RWxlbWVudCB7XG4gICAgY29uc3QgbmV3RGF0ZVZhbHVlID0gdGhpcy5fZGF0ZUFkYXB0ZXIudG9Jc284NjAxKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuYWRkQ2FsZW5kYXJEYXlzKGRheS5kYXRlVmFsdWUsIGRlbHRhKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheVBhZ2VVcERvd24oY2VsbHMsIGluZGV4LCBkYXksIGRlbHRhICsgZGVsdGFJZkRpc2FibGVkLCBkZWx0YUlmRGlzYWJsZWQpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5Rmlyc3QoXG4gICAgY2VsbHM6IFNiYkNhbGVuZGFyRGF5RWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGF5OiBEYXk8VD4sXG4gICAgZGF0ZTogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckRheUVsZW1lbnQge1xuICAgIGNvbnN0IG5ld0RhdGVWYWx1ZSA9IHRoaXMuX2RhdGVBZGFwdGVyLnRvSXNvODYwMShcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoK2RheS55ZWFyVmFsdWUsICtkYXkubW9udGhWYWx1ZSwgZGF0ZSksXG4gICAgKTtcbiAgICBpZiAodGhpcy5faXNEYXlPdXRPZlZpZXcobmV3RGF0ZVZhbHVlKSkge1xuICAgICAgcmV0dXJuIGNlbGxzW2luZGV4XTtcbiAgICB9XG4gICAgY29uc3QgbmV4dENlbGwgPSBjZWxscy5maW5kKChlKSA9PiB0aGlzLl9tYXBWYWx1ZVRvSVNPRGF0ZShlLnZhbHVlISkgPT09IG5ld0RhdGVWYWx1ZSk7XG4gICAgaWYgKCFuZXh0Q2VsbCB8fCBuZXh0Q2VsbC5kaXNhYmxlZCkge1xuICAgICAgcmV0dXJuIHRoaXMuX2ZpbmREYXlGaXJzdChjZWxscywgaW5kZXgsIGRheSwgZGF0ZSArIDEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV4dENlbGw7XG4gIH1cblxuICBwcml2YXRlIF9maW5kRGF5TGFzdChcbiAgICBjZWxsczogU2JiQ2FsZW5kYXJEYXlFbGVtZW50W10sXG4gICAgaW5kZXg6IG51bWJlcixcbiAgICBmaXJzdE5leHRNb250aDogVCxcbiAgKTogU2JiQ2FsZW5kYXJEYXlFbGVtZW50IHtcbiAgICBjb25zdCBuZXdEYXRlVmFsdWUgPSB0aGlzLl9kYXRlQWRhcHRlci50b0lzbzg2MDEoXG4gICAgICB0aGlzLl9kYXRlQWRhcHRlci5hZGRDYWxlbmRhckRheXMoZmlyc3ROZXh0TW9udGgsIC0xKSxcbiAgICApO1xuICAgIGlmICh0aGlzLl9pc0RheU91dE9mVmlldyhuZXdEYXRlVmFsdWUpKSB7XG4gICAgICByZXR1cm4gY2VsbHNbaW5kZXhdO1xuICAgIH1cbiAgICBjb25zdCBuZXh0Q2VsbCA9IGNlbGxzLmZpbmQoKGUpID0+IHRoaXMuX21hcFZhbHVlVG9JU09EYXRlKGUudmFsdWUhKSA9PT0gbmV3RGF0ZVZhbHVlKTtcbiAgICBpZiAoIW5leHRDZWxsIHx8IG5leHRDZWxsLmRpc2FibGVkKSB7XG4gICAgICByZXR1cm4gdGhpcy5fZmluZERheUxhc3QoY2VsbHMsIGluZGV4LCB0aGlzLl9kYXRlQWRhcHRlci5kZXNlcmlhbGl6ZShuZXdEYXRlVmFsdWUpISk7XG4gICAgfVxuICAgIHJldHVybiBuZXh0Q2VsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXRzIHRoZSBpbmRleCBvZiB0aGUgZWxlbWVudCB0byBtb3ZlIHRvLCBiYXNlZCBvbiBhIGxpc3Qgb2YgZWxlbWVudHMgKHdoaWNoIGNhbiBiZSBwb3RlbnRpYWxseSBkaXNhYmxlZCksXG4gICAqIHRoZSBrZXlib2FyZCBpbnB1dCBhbmQgdGhlIHBvc2l0aW9uIG9mIHRoZSBjdXJyZW50IGVsZW1lbnQgaW4gdGhlIGxpc3QuXG4gICAqIEluIHRoZSBkYXkgdmlldywgdGhlIGBkYXk/OiBEYXlgIHBhcmFtZXRlciBpcyBtYW5kYXRvcnkgZm9yIGNhbGN1bGF0aW9uLFxuICAgKiB3aGlsZSBpbiBtb250aCBhbmQgeWVhciB2aWV3IGl0J3Mgbm90IGR1ZSB0byB0aGUgZml4ZWQgYW1vdW50IG9mIHJlbmRlcmVkIGNlbGxzLlxuICAgKi9cbiAgcHJpdmF0ZSBfbmF2aWdhdGVCeUtleWJvYXJkKFxuICAgIGV2dDogS2V5Ym9hcmRFdmVudCxcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGNlbGxzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICApOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB7XG4gICAgY29uc3Qge1xuICAgICAgZWxlbWVudEluZGV4Rm9yV2lkZU1vZGUsXG4gICAgICBvZmZzZXRGb3JXaWRlTW9kZSxcbiAgICAgIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZSxcbiAgICAgIHZlcnRpY2FsT2Zmc2V0LFxuICAgIH06IENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzID1cbiAgICAgIHRoaXMuX2NhbGN1bGF0ZVBhcmFtZXRlcnNGb3JLZXlib2FyZE5hdmlnYXRpb24oaW5kZXgsIHRoaXMuX2NhbGVuZGFyVmlldyA9PT0gJ3llYXInKTtcblxuICAgIHN3aXRjaCAoZXZ0LmtleSkge1xuICAgICAgY2FzZSAnQXJyb3dVcCc6XG4gICAgICAgIHJldHVybiB0aGlzLl9maW5kTmV4dChjZWxscywgaW5kZXgsIC12ZXJ0aWNhbE9mZnNldCk7XG4gICAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCB2ZXJ0aWNhbE9mZnNldCk7XG4gICAgICBjYXNlICdBcnJvd0xlZnQnOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZE5leHQoY2VsbHMsIGluZGV4LCAtMSk7XG4gICAgICBjYXNlICdBcnJvd1JpZ2h0JzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmROZXh0KGNlbGxzLCBpbmRleCwgMSk7XG4gICAgICBjYXNlICdIb21lJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRGaXJzdChjZWxscywgb2Zmc2V0Rm9yV2lkZU1vZGUpO1xuICAgICAgY2FzZSAnUGFnZVVwJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRGaXJzdE9uQ29sdW1uKFxuICAgICAgICAgIGNlbGxzLFxuICAgICAgICAgIGVsZW1lbnRJbmRleEZvcldpZGVNb2RlLFxuICAgICAgICAgIG9mZnNldEZvcldpZGVNb2RlLFxuICAgICAgICAgIHZlcnRpY2FsT2Zmc2V0LFxuICAgICAgICApO1xuICAgICAgY2FzZSAnUGFnZURvd24nOlxuICAgICAgICByZXR1cm4gdGhpcy5fZmluZExhc3RPbkNvbHVtbihjZWxscywgaW5kZXgsIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZSwgdmVydGljYWxPZmZzZXQpO1xuICAgICAgY2FzZSAnRW5kJzpcbiAgICAgICAgcmV0dXJuIHRoaXMuX2ZpbmRMYXN0KGNlbGxzLCBsYXN0RWxlbWVudEluZGV4Rm9yV2lkZU1vZGUgLSAxKTtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHJldHVybiBjZWxsc1tpbmRleF07XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENhbGN1bGF0ZXMgdGhlIHBhcmFtZXRlcnMgbmVlZGVkIGluIGtleWJvYXJkIG5hdmlnYXRpb24gaW4geWVhciBhbmQgbW9udGggdmlldy5cbiAgICogQHBhcmFtIGluZGV4IFRoZSBzdGFydGluZyBlbGVtZW50J3MgaW5kZXggaW4gdGhlIGNlbGwgYXJyYXkuXG4gICAqIEBwYXJhbSBpc1llYXJWaWV3IFdoZXRoZXIgdGhlIGRpc3BsYXllZCBgdmlld2AgaXMgdGhlIHllYXIgb25lLlxuICAgKi9cbiAgcHJpdmF0ZSBfY2FsY3VsYXRlUGFyYW1ldGVyc0ZvcktleWJvYXJkTmF2aWdhdGlvbihcbiAgICBpbmRleDogbnVtYmVyLFxuICAgIGlzWWVhclZpZXc6IGJvb2xlYW4sXG4gICk6IENhbGVuZGFyS2V5Ym9hcmROYXZpZ2F0aW9uTW9udGhZZWFyVmlld3NQYXJhbWV0ZXJzIHtcbiAgICBjb25zdCBlbGVtZW50c1BlclBhZ2UgPSBpc1llYXJWaWV3ID8gWUVBUlNfUEVSX1BBR0UgOiBNT05USFNfUEVSX1BBR0U7XG4gICAgY29uc3Qgb2Zmc2V0OiBudW1iZXIgPSBNYXRoLnRydW5jKGluZGV4IC8gZWxlbWVudHNQZXJQYWdlKSAqIGVsZW1lbnRzUGVyUGFnZTtcbiAgICBjb25zdCBpbmRleEluVmlldzogbnVtYmVyID0gb2Zmc2V0ID09PSAwID8gaW5kZXggOiBpbmRleCAtIGVsZW1lbnRzUGVyUGFnZTtcbiAgICByZXR1cm4ge1xuICAgICAgdmVydGljYWxPZmZzZXQ6IGlzWWVhclZpZXcgPyBZRUFSU19QRVJfUk9XIDogTU9OVEhTX1BFUl9ST1csXG4gICAgICBlbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogaW5kZXhJblZpZXcsXG4gICAgICBvZmZzZXRGb3JXaWRlTW9kZTogaW5kZXggLSBpbmRleEluVmlldyxcbiAgICAgIGxhc3RFbGVtZW50SW5kZXhGb3JXaWRlTW9kZTogb2Zmc2V0ID09PSAwID8gZWxlbWVudHNQZXJQYWdlIDogZWxlbWVudHNQZXJQYWdlICogMixcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIEdldHMgdGhlIG5leHQgZWxlbWVudCBvZiB0aGUgcHJvdmlkZWQgYXJyYXkgc3RhcnRpbmcgZnJvbSBgaW5kZXhgIGJ5IGFkZGluZyBgZGVsdGFgLlxuICAgKiBJZiB0aGUgZm91bmQgZWxlbWVudCBpcyBkaXNhYmxlZCwgaXQgY29udGludWVzIGFkZGluZyBgZGVsdGFgIHVudGlsIGl0IGZpbmRzIGFuIGVuYWJsZWQgb25lIGluIHRoZSBhcnJheSBib3VuZHMuXG4gICAqL1xuICBwcml2YXRlIF9maW5kTmV4dChcbiAgICBkYXlzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgZGVsdGE6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIGxldCBuZXh0SW5kZXggPSBpbmRleCArIGRlbHRhO1xuICAgIHdoaWxlIChuZXh0SW5kZXggPCBkYXlzLmxlbmd0aCAmJiBkYXlzW25leHRJbmRleF0/LmRpc2FibGVkKSB7XG4gICAgICBuZXh0SW5kZXggKz0gZGVsdGE7XG4gICAgfVxuICAgIHJldHVybiBkYXlzW25leHRJbmRleF0gPz8gZGF5c1tpbmRleF07XG4gIH1cblxuICAvKiogRmluZCB0aGUgZmlyc3QgZW5hYmxlZCBlbGVtZW50IGluIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZEZpcnN0KFxuICAgIGRheXM6IFNiYkNhbGVuZGFyQ2VsbEJhc2VFbGVtZW50W10sXG4gICAgZmlyc3RPZkN1cnJlbnRNb250aDogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB7XG4gICAgcmV0dXJuICFkYXlzW2ZpcnN0T2ZDdXJyZW50TW9udGhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbZmlyc3RPZkN1cnJlbnRNb250aF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgZmlyc3RPZkN1cnJlbnRNb250aCwgMSk7XG4gIH1cblxuICAvKiogRmluZCB0aGUgbGFzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHByb3ZpZGVkIGFycmF5LiAqL1xuICBwcml2YXRlIF9maW5kTGFzdChcbiAgICBkYXlzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICAgIGxhc3RPZkN1cnJlbnRNb250aDogbnVtYmVyLFxuICApOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudCB7XG4gICAgcmV0dXJuICFkYXlzW2xhc3RPZkN1cnJlbnRNb250aF0uZGlzYWJsZWRcbiAgICAgID8gZGF5c1tsYXN0T2ZDdXJyZW50TW9udGhdXG4gICAgICA6IHRoaXMuX2ZpbmROZXh0KGRheXMsIGxhc3RPZkN1cnJlbnRNb250aCwgLTEpO1xuICB9XG5cbiAgLyoqIEZpbmQgdGhlIGZpcnN0IGVuYWJsZWQgZWxlbWVudCBpbiB0aGUgc2FtZSBjb2x1bW4gb2YgdGhlIHByb3ZpZGVkIGFycmF5LiAqL1xuICBwcml2YXRlIF9maW5kRmlyc3RPbkNvbHVtbihcbiAgICBkYXlzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgdmVydGljYWxPZmZzZXQ6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIGNvbnN0IG5leHRJbmRleCA9IChpbmRleCAlIHZlcnRpY2FsT2Zmc2V0KSArIG9mZnNldDtcbiAgICByZXR1cm4gIWRheXNbbmV4dEluZGV4XS5kaXNhYmxlZFxuICAgICAgPyBkYXlzW25leHRJbmRleF1cbiAgICAgIDogdGhpcy5fZmluZE5leHQoZGF5cywgbmV4dEluZGV4LCB2ZXJ0aWNhbE9mZnNldCk7XG4gIH1cblxuICAvKiogRmluZCB0aGUgbGFzdCBlbmFibGVkIGVsZW1lbnQgaW4gdGhlIHNhbWUgY29sdW1uIG9mIHRoZSBwcm92aWRlZCBhcnJheS4gKi9cbiAgcHJpdmF0ZSBfZmluZExhc3RPbkNvbHVtbihcbiAgICBkYXlzOiBTYmJDYWxlbmRhckNlbGxCYXNlRWxlbWVudFtdLFxuICAgIGluZGV4OiBudW1iZXIsXG4gICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgdmVydGljYWxPZmZzZXQ6IG51bWJlcixcbiAgKTogU2JiQ2FsZW5kYXJDZWxsQmFzZUVsZW1lbnQge1xuICAgIGNvbnN0IG5leHRJbmRleCA9IGluZGV4ICsgTWF0aC50cnVuYygob2Zmc2V0IC0gaW5kZXggLSAxKSAvIHZlcnRpY2FsT2Zmc2V0KSAqIHZlcnRpY2FsT2Zmc2V0O1xuICAgIHJldHVybiAhZGF5c1tuZXh0SW5kZXhdLmRpc2FibGVkXG4gICAgICA/IGRheXNbbmV4dEluZGV4XVxuICAgICAgOiB0aGlzLl9maW5kTmV4dChkYXlzLCBuZXh0SW5kZXgsIC12ZXJ0aWNhbE9mZnNldCk7XG4gIH1cblxuICBwcml2YXRlIF9yZXNldENhbGVuZGFyVmlld0FuZEVtaXRNb250aENoYW5nZShpbml0VHJhbnNpdGlvbiA9IGZhbHNlKTogdm9pZCB7XG4gICAgdGhpcy5fcmVzZXRDYWxlbmRhclZpZXcoaW5pdFRyYW5zaXRpb24pO1xuICAgIHRoaXMuX2VtaXRNb250aENoYW5nZSgpO1xuICB9XG5cbiAgcHJpdmF0ZSBfcmVzZXRDYWxlbmRhclZpZXcoaW5pdFRyYW5zaXRpb24gPSBmYWxzZSk6IHZvaWQge1xuICAgIGlmICh0aGlzLl9jb250YWluaW5nRm9jdXMpIHtcbiAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgIH1cbiAgICB0aGlzLl9hY3RpdmVEYXRlID1cbiAgICAgICh0aGlzLm11bHRpcGxlID8gKHRoaXMuX3NlbGVjdGVkIGFzIFRbXSkuYXQoLTEpIDogKHRoaXMuX3NlbGVjdGVkIGFzIFQpKSA/P1xuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIudG9kYXkoKTtcbiAgICB0aGlzLl9zZXRDaG9zZW5ZZWFyKCk7XG4gICAgdGhpcy5fY2hvc2VuTW9udGggPSB1bmRlZmluZWQ7XG4gICAgdGhpcy5faW5pdCgpO1xuICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLnZpZXc7XG5cbiAgICBpZiAoaW5pdFRyYW5zaXRpb24pIHtcbiAgICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqIFJlbmRlciB0aGUgdmlldyBmb3IgdGhlIGRheSBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX3JlbmRlckRheVZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IG5leHRNb250aEFjdGl2ZURhdGUgPSB0aGlzLl93aWRlXG4gICAgICA/IHRoaXMuX2RhdGVBZGFwdGVyLmFkZENhbGVuZGFyTW9udGhzKHRoaXMuX2FjdGl2ZURhdGUsIDEpXG4gICAgICA6IHVuZGVmaW5lZDtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzXCI+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ2xlZnQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRNb250aCgtMSksXG4gICAgICAgICAgaTE4blByZXZpb3VzTW9udGhbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fcHJldmlvdXNNb250aERpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLW1vbnRoXCI+XG4gICAgICAgICAgJHt0aGlzLl9jcmVhdGVMYWJlbEZvckRheVZpZXcodGhpcy5fYWN0aXZlRGF0ZSl9XG4gICAgICAgICAgJHt0aGlzLl93aWRlID8gdGhpcy5fY3JlYXRlTGFiZWxGb3JEYXlWaWV3KG5leHRNb250aEFjdGl2ZURhdGUhKSA6IG5vdGhpbmd9XG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJzYmItc2NyZWVuLXJlYWRlci1vbmx5XCIgcm9sZT1cInN0YXR1c1wiPlxuICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVBcmlhTGFiZWxGb3JEYXlWaWV3KHRoaXMuX2FjdGl2ZURhdGUsIG5leHRNb250aEFjdGl2ZURhdGUhKX1cbiAgICAgICAgICA8L3NwYW4+XG4gICAgICAgIDwvZGl2PlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdyaWdodCcsXG4gICAgICAgICAgKCkgPT4gdGhpcy5fZ29Ub0RpZmZlcmVudE1vbnRoKDEpLFxuICAgICAgICAgIGkxOG5OZXh0TW9udGhbdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF0sXG4gICAgICAgICAgdGhpcy5fbmV4dE1vbnRoRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgIDwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtb3ZlcmZsb3ctYnJlYWtcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtY29udGFpbmVyIHNiYi1jYWxlbmRhcl9fdGFibGUtZGF5LXZpZXdcIj5cbiAgICAgICAgICAke3RoaXMub3JpZW50YXRpb24gPT09ICdob3Jpem9udGFsJ1xuICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5VGFibGUodGhpcy5fd2Vla3MsIHRoaXMuX3dlZWtOdW1iZXJzKX1cbiAgICAgICAgICAgICAgICAke3RoaXMuX3dpZGVcbiAgICAgICAgICAgICAgICAgID8gdGhpcy5fY3JlYXRlRGF5VGFibGUodGhpcy5fbmV4dE1vbnRoV2Vla3MsIHRoaXMuX25leHRNb250aFdlZWtOdW1iZXJzLCB0cnVlKVxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICBgXG4gICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlUYWJsZVZlcnRpY2FsKHRoaXMuX3dlZWtzLCB0aGlzLl93ZWVrTnVtYmVycyl9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl93aWRlXG4gICAgICAgICAgICAgICAgICA/IHRoaXMuX2NyZWF0ZURheVRhYmxlVmVydGljYWwoXG4gICAgICAgICAgICAgICAgICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla3MsXG4gICAgICAgICAgICAgICAgICAgICAgdGhpcy5fbmV4dE1vbnRoV2Vla051bWJlcnMsXG4gICAgICAgICAgICAgICAgICAgICAgbmV4dE1vbnRoQWN0aXZlRGF0ZSxcbiAgICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICBgfVxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgbGFiZWwgd2l0aCB0aGUgbW9udGggZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVMYWJlbEZvckRheVZpZXcoZDogVCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCBtb250aExhYmVsID0gYCR7XG4gICAgICB0aGlzLl9tb250aE5hbWVzW3RoaXMuX2RhdGVBZGFwdGVyLmdldE1vbnRoKGQpIC0gMV1cbiAgICB9ICR7dGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkKX1gO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2RhdGUtc2VsZWN0aW9uIHNiYi1jYWxlbmRhcl9fY29udHJvbHMtY2hhbmdlLWRhdGVcIlxuICAgICAgICBhcmlhLWxhYmVsPVwiJHtpMThuWWVhck1vbnRoU2VsZWN0aW9uW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke21vbnRoTGFiZWx9XCJcbiAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgIHRoaXMuX3Jlc2V0Rm9jdXMgPSB0cnVlO1xuICAgICAgICAgIHRoaXMuX25leHRDYWxlbmRhclZpZXcgPSAneWVhcic7XG4gICAgICAgICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgICAgICAgfX1cbiAgICAgID5cbiAgICAgICAgJHttb250aExhYmVsfVxuICAgICAgICA8c2JiLWljb24gbmFtZT1cImNoZXZyb24tc21hbGwtZG93bi1zbWFsbFwiPjwvc2JiLWljb24+XG4gICAgICA8L2J1dHRvbj5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIGFyaWEtbGFiZWwgZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVBcmlhTGFiZWxGb3JEYXlWaWV3KC4uLmRhdGVzOiBUW10pOiBzdHJpbmcge1xuICAgIGxldCBtb250aExhYmVsID0gJyc7XG4gICAgZm9yIChjb25zdCBkIG9mIGRhdGVzKSB7XG4gICAgICBpZiAoZCkge1xuICAgICAgICBtb250aExhYmVsICs9IGAke1xuICAgICAgICAgIHRoaXMuX21vbnRoTmFtZXNbdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgoZCkgLSAxXVxuICAgICAgICB9ICR7dGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0WWVhcihkKX0gYDtcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIG1vbnRoTGFiZWw7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgY2FsZW5kYXIgdGFibGUgZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlUYWJsZShcbiAgICB3ZWVrczogRGF5PFQ+W11bXSxcbiAgICB3ZWVrTnVtYmVyczogbnVtYmVyW10sXG4gICAgaXNXaWRlTmV4dE1vbnRoOiBib29sZWFuID0gZmFsc2UsXG4gICk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla051bWJlcnM6IERheTxUPltdID0gKFxuICAgICAgdGhpcy5fd2lkZVxuICAgICAgICA/IFsuLi50aGlzLl93ZWVrcywgLi4udGhpcy5fbmV4dE1vbnRoV2Vla3NdXG4gICAgICAgIDogaXNXaWRlTmV4dE1vbnRoXG4gICAgICAgICAgPyB0aGlzLl9uZXh0TW9udGhXZWVrc1xuICAgICAgICAgIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICBjb25zdCB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla0RheXM6IERheTxUPltdID0gKFxuICAgICAgaXNXaWRlTmV4dE1vbnRoID8gdGhpcy5fbmV4dE1vbnRoV2Vla3MgOiB0aGlzLl93ZWVrc1xuICAgICkuZmxhdCgpO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPHRhYmxlXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICAgIEBmb2N1c291dD0keyhldmVudDogRm9jdXNFdmVudCkgPT5cbiAgICAgICAgICB0aGlzLl9oYW5kbGVUYWJsZUJsdXIoZXZlbnQucmVsYXRlZFRhcmdldCBhcyBIVE1MRWxlbWVudCl9XG4gICAgICAgIEBhbmltYXRpb25lbmQ9JHsoZTogQW5pbWF0aW9uRXZlbnQpID0+IHRoaXMuX3RhYmxlQW5pbWF0aW9uRW5kKGUpfVxuICAgICAgPlxuICAgICAgICA8dGhlYWQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlclwiPlxuICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICR7dGhpcy53ZWVrTnVtYmVycyA/IGh0bWxgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbFwiPjwvdGg+YCA6IG5vdGhpbmd9XG4gICAgICAgICAgICAke3RoaXMuX3dlZWtkYXlzLm1hcChcbiAgICAgICAgICAgICAgKHdlZWtEYXk6IFdlZWtkYXksIGluZGV4OiBudW1iZXIpID0+IGh0bWxgXG4gICAgICAgICAgICAgICAgPHRoIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbFwiPlxuICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXItd2Vla2RheVxuICAgICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHt3ZWVrRGF5fVxuICAgICAgICAgICAgICAgICAgICAgICAgICBAY2xpY2s9JHsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gTk9URTogU3VuZGF5cyBoYXZlIGluZGV4IDcsIHdoaWxlIHRoZWlyIHdlZWtEYXlWYWx1ZSBpcyAwXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZGF5czogRGF5PFQ+W10gPSB3ZWVrc0ZvclNlbGVjdE11bHRpcGxlV2Vla0RheXMuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla0RheVZhbHVlID09PSAoaW5kZXggKyAxKSAlIDcsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgKSE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci13ZWVrZGF5PlxuICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJzYmItc2NyZWVuLXJlYWRlci1vbmx5XCI+JHt3ZWVrRGF5Lmxvbmd9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrRGF5Lm5hcnJvd308L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICA8L3RoPlxuICAgICAgICAgICAgICBgLFxuICAgICAgICAgICAgKX1cbiAgICAgICAgICA8L3RyPlxuICAgICAgICA8L3RoZWFkPlxuICAgICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgICAke3dlZWtzLm1hcCgod2VlazogRGF5PFQ+W10sIHJvd0luZGV4OiBudW1iZXIpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGZpcnN0Um93T2Zmc2V0OiBudW1iZXIgPSBEQVlTX1BFUl9ST1cgLSB3ZWVrLmxlbmd0aDtcbiAgICAgICAgICAgIGlmIChyb3dJbmRleCA9PT0gMCAmJiBmaXJzdFJvd09mZnNldCkge1xuICAgICAgICAgICAgICByZXR1cm4gaHRtbGBcbiAgICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgICAke3RoaXMud2Vla051bWJlcnNcbiAgICAgICAgICAgICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbFwiPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAke3RoaXMubXVsdGlwbGVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXItd2Vla251bWJlclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT0ke3dlZWtOdW1iZXJzWzBdfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRheTogRGF5PFQ+KSA9PiBkYXkud2Vla1ZhbHVlID09PSB3ZWVrTnVtYmVyc1swXSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICkhO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhkYXlzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+PC9zYmItY2FsZW5kYXItd2Vla251bWJlcj5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwic2JiLXNjcmVlbi1yZWFkZXItb25seVwiXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPiR7YCR7aTE4bkNhbGVuZGFyV2Vla051bWJlclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt3ZWVrTnVtYmVyc1swXX1gfTwvc3BhblxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla051bWJlcnNbMF19PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+XG4gICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgICAgICAke1suLi5BcnJheShmaXJzdFJvd09mZnNldCkua2V5cygpXS5tYXAoXG4gICAgICAgICAgICAgICAgICAgICgpID0+IGh0bWxgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+PC90ZD5gLFxuICAgICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5Q2VsbHMod2Vlayl9XG4gICAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgICAgYDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVybiBodG1sYFxuICAgICAgICAgICAgICA8dHI+XG4gICAgICAgICAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzXG4gICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXItY2VsbC12ZXJ0aWNhbFwiPlxuICAgICAgICAgICAgICAgICAgICAgICAgJHt0aGlzLm11bHRpcGxlXG4gICAgICAgICAgICAgICAgICAgICAgICAgID8gaHRtbGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXItd2Vla251bWJlclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAudmFsdWU9JHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBkYXlzOiBEYXk8VD5bXSA9IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVycy5maWx0ZXIoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF5OiBEYXk8VD4pID0+IGRheS53ZWVrVmFsdWUgPT09IHdlZWtOdW1iZXJzW3Jvd0luZGV4XSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKGRheXMpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLXdlZWtudW1iZXI+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcIlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+JHtgJHtpMThuQ2FsZW5kYXJXZWVrTnVtYmVyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3dlZWtOdW1iZXJzW3Jvd0luZGV4XX1gfTwvc3BhblxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrTnVtYmVyc1tyb3dJbmRleF19PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGB9XG4gICAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICAgICAgICAgICR7dGhpcy5fY3JlYXRlRGF5Q2VsbHMod2Vlayl9XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICBgO1xuICAgICAgICAgIH0pfVxuICAgICAgICA8L3Rib2R5PlxuICAgICAgPC90YWJsZT5cbiAgICBgO1xuICB9XG5cbiAgLyoqIENyZWF0ZXMgdGhlIHRhYmxlIGluIG9yaWVudGF0aW9uPSd2ZXJ0aWNhbCcuICovXG4gIHByaXZhdGUgX2NyZWF0ZURheVRhYmxlVmVydGljYWwoXG4gICAgd2Vla3M6IERheTxUPltdW10sXG4gICAgd2Vla051bWJlcnM6IG51bWJlcltdLFxuICAgIG5leHRNb250aEFjdGl2ZURhdGU/OiBULFxuICApOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgY29uc3Qgd2Vla09mZnNldCA9IHRoaXMuX2RhdGVBZGFwdGVyLmdldEZpcnN0V2Vla09mZnNldChcbiAgICAgIG5leHRNb250aEFjdGl2ZURhdGUgPz8gdGhpcy5fYWN0aXZlRGF0ZSxcbiAgICApO1xuICAgIGNvbnN0IHdlZWtzRm9yU2VsZWN0TXVsdGlwbGVXZWVrTnVtYmVyczogRGF5PFQ+W10gPSAoXG4gICAgICB0aGlzLl93aWRlXG4gICAgICAgID8gWy4uLnRoaXMuX3dlZWtzLCAuLi50aGlzLl9uZXh0TW9udGhXZWVrc11cbiAgICAgICAgOiBuZXh0TW9udGhBY3RpdmVEYXRlXG4gICAgICAgICAgPyB0aGlzLl9uZXh0TW9udGhXZWVrc1xuICAgICAgICAgIDogdGhpcy5fd2Vla3NcbiAgICApLmZsYXQoKTtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDx0YWJsZVxuICAgICAgICBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGVcIlxuICAgICAgICBAZm9jdXNvdXQ9JHsoZXZlbnQ6IEZvY3VzRXZlbnQpID0+XG4gICAgICAgICAgdGhpcy5faGFuZGxlVGFibGVCbHVyKGV2ZW50LnJlbGF0ZWRUYXJnZXQgYXMgSFRNTEVsZW1lbnQpfVxuICAgICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICAgID5cbiAgICAgICAgJHt0aGlzLndlZWtOdW1iZXJzXG4gICAgICAgICAgPyBodG1sYFxuICAgICAgICAgICAgICA8dGhlYWQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWhlYWRlclwiPlxuICAgICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAgICR7bmV4dE1vbnRoQWN0aXZlRGF0ZVxuICAgICAgICAgICAgICAgICAgICA/IG5vdGhpbmdcbiAgICAgICAgICAgICAgICAgICAgOiBodG1sYDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPjwvdGg+YH1cbiAgICAgICAgICAgICAgICAgICR7d2Vla051bWJlcnMubWFwKFxuICAgICAgICAgICAgICAgICAgICAod2Vla051bWJlcjogbnVtYmVyKSA9PiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtudW1iZXJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla051bWJlcn1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRheXM6IERheTxUPltdID0gd2Vla3NGb3JTZWxlY3RNdWx0aXBsZVdlZWtOdW1iZXJzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkYXk6IERheTxUPikgPT4gZGF5LndlZWtWYWx1ZSA9PT0gd2Vla051bWJlcixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApITtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLl9zZWxlY3RNdWx0aXBsZURhdGVzKGRheXMpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPjwvc2JiLWNhbGVuZGFyLXdlZWtudW1iZXI+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYFxuICAgICAgICAgICAgICAgICAgICAgICAgICA6IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcIlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+JHtgJHtpMThuQ2FsZW5kYXJXZWVrTnVtYmVyW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdfSAke3dlZWtOdW1iZXJ9YH08L3NwYW5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxzcGFuIGFyaWEtaGlkZGVuPVwidHJ1ZVwiPiR7d2Vla051bWJlcn08L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAgICAgICA8L3RoPlxuICAgICAgICAgICAgICAgICAgICBgLFxuICAgICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgICA8L3RoZWFkPlxuICAgICAgICAgICAgYFxuICAgICAgICAgIDogbm90aGluZ31cbiAgICAgICAgPHRib2R5IGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1ib2R5XCI+XG4gICAgICAgICAgJHt3ZWVrcy5tYXAoKHdlZWs6IERheTxUPltdLCByb3dJbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICBjb25zdCB3ZWVrZGF5ID0gdGhpcy5fd2Vla2RheXNbcm93SW5kZXhdO1xuICAgICAgICAgICAgY29uc3Qgc2VsZWN0YWJsZURheXMgPSB0aGlzLl93aWRlID8gWy4uLndlZWssIC4uLnRoaXMuX25leHRNb250aFdlZWtzW3Jvd0luZGV4XV0gOiB3ZWVrO1xuICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgIDx0cj5cbiAgICAgICAgICAgICAgICAke25leHRNb250aEFjdGl2ZURhdGVcbiAgICAgICAgICAgICAgICAgID8gbm90aGluZ1xuICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGwtdmVydGljYWxcIj5cbiAgICAgICAgICAgICAgICAgICAgICAgICR7dGhpcy5tdWx0aXBsZVxuICAgICAgICAgICAgICAgICAgICAgICAgICA/IGh0bWxgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8c2JiLWNhbGVuZGFyLXdlZWtkYXlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7d2Vla2RheX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fc2VsZWN0TXVsdGlwbGVEYXRlcyhzZWxlY3RhYmxlRGF5cyl9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICR7d2Vla2RheS5uYXJyb3d9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3NiYi1jYWxlbmRhci13ZWVrZGF5PlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBodG1sYFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJzYmItc2NyZWVuLXJlYWRlci1vbmx5XCI+JHt3ZWVrZGF5Lmxvbmd9PC9zcGFuPlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4gYXJpYS1oaWRkZW49XCJ0cnVlXCI+JHt3ZWVrZGF5Lm5hcnJvd308L3NwYW4+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgYH1cbiAgICAgICAgICAgICAgICAgICAgICA8L3RkPlxuICAgICAgICAgICAgICAgICAgICBgfVxuICAgICAgICAgICAgICAgICR7cm93SW5kZXggPCB3ZWVrT2Zmc2V0XG4gICAgICAgICAgICAgICAgICA/IGh0bWxgPHRkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1kYXRhXCI+PC90ZD5gXG4gICAgICAgICAgICAgICAgICA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgICAgJHt0aGlzLl9jcmVhdGVEYXlDZWxscyh3ZWVrKX1cbiAgICAgICAgICAgICAgPC90cj5cbiAgICAgICAgICAgIGA7XG4gICAgICAgICAgfSl9XG4gICAgICAgIDwvdGJvZHk+XG4gICAgICA8L3RhYmxlPlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgY2VsbHMgZm9yIHRoZSBkYWlseSB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVEYXlDZWxscyh3ZWVrOiBEYXk8VD5bXSk6IFRlbXBsYXRlUmVzdWx0W10ge1xuICAgIHJldHVybiB3ZWVrLm1hcCgoZGF5OiBEYXk8VD4pID0+IHtcbiAgICAgIHJldHVybiBodG1sYFxuICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGEgc2JiLWNhbGVuZGFyX19kYXktY2VsbFwiPlxuICAgICAgICAgIDxzbG90IG5hbWU9JHtkYXkudmFsdWV9PlxuICAgICAgICAgICAgPHNiYi1jYWxlbmRhci1kYXlcbiAgICAgICAgICAgICAgc2xvdD0ke2RheS52YWx1ZX1cbiAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fc2VsZWN0RGF0ZShkYXkuZGF0ZVZhbHVlKX1cbiAgICAgICAgICAgICAgQGtleWRvd249JHsoZXZ0OiBLZXlib2FyZEV2ZW50KSA9PiB0aGlzLl9oYW5kbGVLZXlib2FyZEV2ZW50KGV2dCwgZGF5KX1cbiAgICAgICAgICAgID48L3NiYi1jYWxlbmRhci1kYXk+XG4gICAgICAgICAgPC9zbG90PlxuICAgICAgICA8L3RkPlxuICAgICAgYDtcbiAgICB9KTtcbiAgfVxuXG4gIC8qKiBSZW5kZXIgdGhlIHZpZXcgZm9yIHRoZSBtb250aCBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX3JlbmRlck1vbnRoVmlldygpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9sc1wiPlxuICAgICAgICAke3RoaXMuX2dldEFycm93KFxuICAgICAgICAgICdsZWZ0JyxcbiAgICAgICAgICAoKSA9PiB0aGlzLl9nb1RvRGlmZmVyZW50WWVhcigtMSksXG4gICAgICAgICAgaTE4blByZXZpb3VzWWVhclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9wcmV2aW91c1llYXJEaXNhYmxlZCgpLFxuICAgICAgICApfVxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWNhbGVuZGFyX19jb250cm9scy1tb250aFwiPiR7dGhpcy5fY3JlYXRlTGFiZWxGb3JNb250aFZpZXcoKX08L2Rpdj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAncmlnaHQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRZZWFyKDEpLFxuICAgICAgICAgIGkxOG5OZXh0WWVhclt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9uZXh0WWVhckRpc2FibGVkKCksXG4gICAgICAgICl9XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLW92ZXJmbG93LWJyZWFrXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWNvbnRhaW5lciBzYmItY2FsZW5kYXJfX3RhYmxlLW1vbnRoLXZpZXdcIj5cbiAgICAgICAgICAke3RoaXMuX2NyZWF0ZU1vbnRoVGFibGUodGhpcy5fbW9udGhzLCB0aGlzLl9jaG9zZW5ZZWFyISl9XG4gICAgICAgICAgJHt0aGlzLl93aWRlID8gdGhpcy5fY3JlYXRlTW9udGhUYWJsZSh0aGlzLl9tb250aHMsIHRoaXMuX2Nob3NlblllYXIhICsgMSkgOiBub3RoaW5nfVxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgbGFiZWwgd2l0aCB0aGUgeWVhciBmb3IgdGhlIG1vbnRobHkgdmlldy4gKi9cbiAgcHJpdmF0ZSBfY3JlYXRlTGFiZWxGb3JNb250aFZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYCA8YnV0dG9uXG4gICAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgICBpZD1cInNiYi1jYWxlbmRhcl9fbW9udGgtc2VsZWN0aW9uXCJcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlXCJcbiAgICAgICAgYXJpYS1sYWJlbD0ke2Ake2kxOG5DYWxlbmRhckRhdGVTZWxlY3Rpb25bdGhpcy5fbGFuZ3VhZ2UuY3VycmVudF19ICR7dGhpcy5fY2hvc2VuWWVhcn1gfVxuICAgICAgICBAY2xpY2s9JHsoKSA9PiB0aGlzLl9yZXNldENhbGVuZGFyVmlld0FuZEVtaXRNb250aENoYW5nZSh0cnVlKX1cbiAgICAgID5cbiAgICAgICAgJHt0aGlzLl9jaG9zZW5ZZWFyfSAke3RoaXMuX3dpZGUgPyBgIC0gJHt0aGlzLl9jaG9zZW5ZZWFyISArIDF9YCA6IG5vdGhpbmd9XG4gICAgICAgIDxzYmItaWNvbiBuYW1lPVwiY2hldnJvbi1zbWFsbC11cC1zbWFsbFwiPjwvc2JiLWljb24+XG4gICAgICA8L2J1dHRvbj5cbiAgICAgIDxzcGFuIGNsYXNzPVwic2JiLXNjcmVlbi1yZWFkZXItb25seVwiIHJvbGU9XCJzdGF0dXNcIj4gJHt0aGlzLl9jaG9zZW5ZZWFyfSA8L3NwYW4+YDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSB0YWJsZSBmb3IgdGhlIG1vbnRoIHNlbGVjdGlvbiB2aWV3LiAqL1xuICBwcml2YXRlIF9jcmVhdGVNb250aFRhYmxlKG1vbnRoczogTW9udGhDZWxsW11bXSwgeWVhcjogbnVtYmVyKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPHRhYmxlXG4gICAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICAgIEBhbmltYXRpb25lbmQ9JHsoZTogQW5pbWF0aW9uRXZlbnQpID0+IHRoaXMuX3RhYmxlQW5pbWF0aW9uRW5kKGUpfVxuICAgICAgPlxuICAgICAgICAke3RoaXMuX3dpZGVcbiAgICAgICAgICA/IGh0bWxgPHRoZWFkIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZS1oZWFkZXJcIiBhcmlhLWhpZGRlbj1cInRydWVcIj5cbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgIDx0aCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtaGVhZGVyLWNlbGxcIiBjb2xzcGFuPSR7TU9OVEhTX1BFUl9ST1d9PiR7eWVhcn08L3RoPlxuICAgICAgICAgICAgICA8L3RyPlxuICAgICAgICAgICAgPC90aGVhZD5gXG4gICAgICAgICAgOiBub3RoaW5nfVxuICAgICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgICAke21vbnRocy5tYXAoXG4gICAgICAgICAgICAocm93OiBNb250aENlbGxbXSkgPT4gaHRtbGBcbiAgICAgICAgICAgICAgPHRyPlxuICAgICAgICAgICAgICAgICR7cm93Lm1hcCgobW9udGg6IE1vbnRoQ2VsbCkgPT4ge1xuICAgICAgICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtZGF0YVwiPlxuICAgICAgICAgICAgICAgICAgICAgIDxzYmItY2FsZW5kYXItbW9udGhcbiAgICAgICAgICAgICAgICAgICAgICAgIC52YWx1ZT1cIiR7eWVhcn0tJHttb250aC52YWx1ZX1cIlxuICAgICAgICAgICAgICAgICAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fb25Nb250aFNlbGVjdGlvbihtb250aC5tb250aFZhbHVlLCB5ZWFyKX1cbiAgICAgICAgICAgICAgICAgICAgICAgIEBrZXlkb3duPSR7KGV2dDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChldnQpfVxuICAgICAgICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgICAgICA8L3NiYi1jYWxlbmRhci1tb250aD5cbiAgICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICAgIGA7XG4gICAgICAgICAgICAgICAgfSl9XG4gICAgICAgICAgICAgIDwvdHI+XG4gICAgICAgICAgICBgLFxuICAgICAgICAgICl9XG4gICAgICAgIDwvdGJvZHk+XG4gICAgICA8L3RhYmxlPlxuICAgIGA7XG4gIH1cblxuICAvKiogU2VsZWN0IHRoZSBtb250aCBhbmQgY2hhbmdlIHRoZSB2aWV3IHRvIGRheSBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX29uTW9udGhTZWxlY3Rpb24obW9udGg6IG51bWJlciwgeWVhcjogbnVtYmVyKTogdm9pZCB7XG4gICAgdGhpcy5fY2hvc2VuTW9udGggPSBtb250aDtcbiAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gJ2RheSc7XG4gICAgdGhpcy5faW5pdChcbiAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmNyZWF0ZURhdGUoXG4gICAgICAgIHllYXIsXG4gICAgICAgIHRoaXMuX2Nob3Nlbk1vbnRoLFxuICAgICAgICB0aGlzLl9kYXRlQWRhcHRlci5nZXREYXRlKHRoaXMuX2FjdGl2ZURhdGUpLFxuICAgICAgKSxcbiAgICApO1xuICAgIHRoaXMuX3N0YXJ0VGFibGVUcmFuc2l0aW9uKCk7XG4gICAgdGhpcy5fZW1pdE1vbnRoQ2hhbmdlKCk7XG4gIH1cblxuICAvKiogUmVuZGVyIHRoZSB2aWV3IGZvciB0aGUgeWVhciBzZWxlY3Rpb24uICovXG4gIHByaXZhdGUgX3JlbmRlclllYXJWaWV3KCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGBcbiAgICAgIDxkaXYgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzXCI+XG4gICAgICAgICR7dGhpcy5fZ2V0QXJyb3coXG4gICAgICAgICAgJ2xlZnQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRZZWFyUmFuZ2UoLVlFQVJTX1BFUl9QQUdFKSxcbiAgICAgICAgICBpMThuUHJldmlvdXNZZWFyUmFuZ2UoWUVBUlNfUEVSX1BBR0UpW3RoaXMuX2xhbmd1YWdlLmN1cnJlbnRdLFxuICAgICAgICAgIHRoaXMuX3ByZXZpb3VzWWVhclJhbmdlRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fY29udHJvbHMtbW9udGhcIj4ke3RoaXMuX2NyZWF0ZUxhYmVsRm9yWWVhclZpZXcoKX08L2Rpdj5cbiAgICAgICAgJHt0aGlzLl9nZXRBcnJvdyhcbiAgICAgICAgICAncmlnaHQnLFxuICAgICAgICAgICgpID0+IHRoaXMuX2dvVG9EaWZmZXJlbnRZZWFyUmFuZ2UoWUVBUlNfUEVSX1BBR0UpLFxuICAgICAgICAgIGkxOG5OZXh0WWVhclJhbmdlKFlFQVJTX1BFUl9QQUdFKVt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XSxcbiAgICAgICAgICB0aGlzLl9uZXh0WWVhclJhbmdlRGlzYWJsZWQoKSxcbiAgICAgICAgKX1cbiAgICAgIDwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtb3ZlcmZsb3ctYnJlYWtcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fdGFibGUtY29udGFpbmVyIHNiYi1jYWxlbmRhcl9fdGFibGUteWVhci12aWV3XCI+XG4gICAgICAgICAgJHt0aGlzLl9jcmVhdGVZZWFyVGFibGUodGhpcy5feWVhcnMpfVxuICAgICAgICAgICR7dGhpcy5fd2lkZSA/IHRoaXMuX2NyZWF0ZVllYXJUYWJsZSh0aGlzLl9uZXh0TW9udGhZZWFycywgdHJ1ZSkgOiBub3RoaW5nfVxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgIGA7XG4gIH1cblxuICAvKiogQ3JlYXRlcyB0aGUgYnV0dG9uIGFycm93IGZvciBhbGwgdGhlIHZpZXdzLiAqL1xuICBwcml2YXRlIF9nZXRBcnJvdyhcbiAgICBkaXJlY3Rpb246ICdsZWZ0JyB8ICdyaWdodCcsXG4gICAgY2xpY2s6ICgpID0+IHZvaWQsXG4gICAgYXJpYUxhYmVsOiBzdHJpbmcsXG4gICAgZGlzYWJsZWQ6IGJvb2xlYW4sXG4gICk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICByZXR1cm4gaHRtbGA8c2JiLXNlY29uZGFyeS1idXR0b25cbiAgICAgIHNpemU9XCJtXCJcbiAgICAgIGljb24tbmFtZT1cImNoZXZyb24tc21hbGwtJHtkaXJlY3Rpb259LXNtYWxsXCJcbiAgICAgIGFyaWEtbGFiZWw9JHthcmlhTGFiZWx9XG4gICAgICBAY2xpY2s9JHtjbGlja31cbiAgICAgID9kaXNhYmxlZD0ke2Rpc2FibGVkfVxuICAgICAgaWQ9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLSR7ZGlyZWN0aW9uID09PSAnbGVmdCcgPyAncHJldmlvdXMnIDogJ25leHQnfVwiXG4gICAgPjwvc2JiLXNlY29uZGFyeS1idXR0b24+YDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSBsYWJlbCB3aXRoIHRoZSB5ZWFyIHJhbmdlIGZvciB0aGUgeWVhcmx5IHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZUxhYmVsRm9yWWVhclZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGNvbnN0IGZpcnN0WWVhcjogbnVtYmVyID0gdGhpcy5feWVhcnMuZmxhdCgpWzBdO1xuICAgIGNvbnN0IGxhc3RZZWFyQXJyYXk6IG51bWJlcltdID0gKHRoaXMuX3dpZGUgPyB0aGlzLl9uZXh0TW9udGhZZWFycyA6IHRoaXMuX3llYXJzKS5mbGF0KCk7XG4gICAgY29uc3QgbGFzdFllYXI6IG51bWJlciA9IGxhc3RZZWFyQXJyYXlbbGFzdFllYXJBcnJheS5sZW5ndGggLSAxXTtcbiAgICBjb25zdCB5ZWFyTGFiZWwgPSBgJHtmaXJzdFllYXJ9IC0gJHtsYXN0WWVhcn1gO1xuICAgIHJldHVybiBodG1sYFxuICAgICAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgaWQ9XCJzYmItY2FsZW5kYXJfX3llYXItc2VsZWN0aW9uXCJcbiAgICAgICAgY2xhc3M9XCJzYmItY2FsZW5kYXJfX2NvbnRyb2xzLWNoYW5nZS1kYXRlXCJcbiAgICAgICAgYXJpYS1sYWJlbD1cIiR7aTE4bkNhbGVuZGFyRGF0ZVNlbGVjdGlvblt0aGlzLl9sYW5ndWFnZS5jdXJyZW50XX0gJHt5ZWFyTGFiZWx9XCJcbiAgICAgICAgQGNsaWNrPSR7KCkgPT4gdGhpcy5fcmVzZXRDYWxlbmRhclZpZXdBbmRFbWl0TW9udGhDaGFuZ2UodHJ1ZSl9XG4gICAgICA+XG4gICAgICAgICR7eWVhckxhYmVsfVxuICAgICAgICA8c2JiLWljb24gbmFtZT1cImNoZXZyb24tc21hbGwtdXAtc21hbGxcIj48L3NiYi1pY29uPlxuICAgICAgPC9idXR0b24+XG4gICAgICA8c3BhbiBjbGFzcz1cInNiYi1zY3JlZW4tcmVhZGVyLW9ubHlcIiByb2xlPVwic3RhdHVzXCI+ICR7eWVhckxhYmVsfSA8L3NwYW4+XG4gICAgYDtcbiAgfVxuXG4gIC8qKiBDcmVhdGVzIHRoZSB0YWJsZSBmb3IgdGhlIHllYXIgc2VsZWN0aW9uIHZpZXcuICovXG4gIHByaXZhdGUgX2NyZWF0ZVllYXJUYWJsZSh5ZWFyczogbnVtYmVyW11bXSwgc2hpZnRSaWdodCA9IGZhbHNlKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIHJldHVybiBodG1sYCA8dGFibGVcbiAgICAgIGNsYXNzPVwic2JiLWNhbGVuZGFyX190YWJsZVwiXG4gICAgICBAYW5pbWF0aW9uZW5kPSR7KGU6IEFuaW1hdGlvbkV2ZW50KSA9PiB0aGlzLl90YWJsZUFuaW1hdGlvbkVuZChlKX1cbiAgICA+XG4gICAgICA8dGJvZHkgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWJvZHlcIj5cbiAgICAgICAgJHt5ZWFycy5tYXAoXG4gICAgICAgICAgKHJvdzogbnVtYmVyW10pID0+XG4gICAgICAgICAgICBodG1sYCA8dHI+XG4gICAgICAgICAgICAgICR7cm93Lm1hcCgoeWVhcjogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIGh0bWxgXG4gICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9XCJzYmItY2FsZW5kYXJfX3RhYmxlLWRhdGFcIj5cbiAgICAgICAgICAgICAgICAgICAgPHNiYi1jYWxlbmRhci15ZWFyXG4gICAgICAgICAgICAgICAgICAgICAgLnZhbHVlPSR7U3RyaW5nKHllYXIpfVxuICAgICAgICAgICAgICAgICAgICAgIEBrZXlkb3duPSR7KGV2dDogS2V5Ym9hcmRFdmVudCkgPT4gdGhpcy5faGFuZGxlS2V5Ym9hcmRFdmVudChldnQpfVxuICAgICAgICAgICAgICAgICAgICAgIEBjbGljaz0keygpID0+IHRoaXMuX29uWWVhclNlbGVjdGlvbih5ZWFyLCBzaGlmdFJpZ2h0KX1cbiAgICAgICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAgICA8L3NiYi1jYWxlbmRhci15ZWFyPlxuICAgICAgICAgICAgICAgICAgPC90ZD5cbiAgICAgICAgICAgICAgICBgO1xuICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgIDwvdHI+YCxcbiAgICAgICAgKX1cbiAgICAgIDwvdGJvZHk+XG4gICAgPC90YWJsZT5gO1xuICB9XG5cbiAgLyoqIFNlbGVjdCB0aGUgeWVhciBhbmQgY2hhbmdlIHRoZSB2aWV3IHRvIG1vbnRoIHNlbGVjdGlvbi4gKi9cbiAgcHJpdmF0ZSBfb25ZZWFyU2VsZWN0aW9uKHllYXI6IG51bWJlciwgcmlnaHRTaWRlOiBib29sZWFuKTogdm9pZCB7XG4gICAgdGhpcy5fY2hvc2VuWWVhciA9IHJpZ2h0U2lkZSA/IHllYXIgLSAxIDogeWVhcjtcbiAgICB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3ID0gJ21vbnRoJztcbiAgICB0aGlzLl9hc3NpZ25BY3RpdmVEYXRlKFxuICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuY3JlYXRlRGF0ZShcbiAgICAgICAgdGhpcy5fY2hvc2VuWWVhcixcbiAgICAgICAgdGhpcy5fZGF0ZUFkYXB0ZXIuZ2V0TW9udGgodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICAgIHRoaXMuX2RhdGVBZGFwdGVyLmdldERhdGUodGhpcy5fYWN0aXZlRGF0ZSksXG4gICAgICApLFxuICAgICk7XG4gICAgdGhpcy5fc3RhcnRUYWJsZVRyYW5zaXRpb24oKTtcbiAgfVxuXG4gIHByaXZhdGUgX2dldFZpZXcoKTogVGVtcGxhdGVSZXN1bHQge1xuICAgIGlmIChpc1NlcnZlciB8fCB0aGlzLmh5ZHJhdGlvblJlcXVpcmVkKSB7XG4gICAgICAvLyBUT0RPOiBXZSBkaXNhYmxlIFNTUiBmb3IgY2FsZW5kYXIgZm9yIG5vdy4gRmlndXJlIG91dCwgaWYgdGhlcmUgaXMgYSB3YXlcbiAgICAgIC8vIHRvIGVuYWJsZSBpdCwgd2hpbGUgY29uc2lkZXJpbmcgaTE4biBhbmQgZGF0ZSBpbmZvcm1hdGlvbi5cbiAgICAgIHJldHVybiBodG1sYCR7bm90aGluZ31gO1xuICAgIH1cbiAgICBzd2l0Y2ggKHRoaXMuX2NhbGVuZGFyVmlldykge1xuICAgICAgY2FzZSAneWVhcic6XG4gICAgICAgIHJldHVybiB0aGlzLl9yZW5kZXJZZWFyVmlldygpO1xuICAgICAgY2FzZSAnbW9udGgnOlxuICAgICAgICByZXR1cm4gdGhpcy5fcmVuZGVyTW9udGhWaWV3KCk7XG4gICAgICBjYXNlICdkYXknOlxuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIHRoaXMuX3JlbmRlckRheVZpZXcoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF90YWJsZUFuaW1hdGlvbkVuZChldmVudDogQW5pbWF0aW9uRXZlbnQpOiB2b2lkIHtcbiAgICBjb25zdCB0YWJsZSA9IGV2ZW50LnRhcmdldCBhcyBIVE1MRWxlbWVudDtcbiAgICBpZiAoZXZlbnQuYW5pbWF0aW9uTmFtZSA9PT0gJ2hpZGUnKSB7XG4gICAgICB0YWJsZS5jbGFzc0xpc3QucmVtb3ZlKCdzYmItY2FsZW5kYXJfX3RhYmxlLWhpZGUnKTtcbiAgICAgIGlmICh0aGlzLl9jb250YWluaW5nRm9jdXMpIHtcbiAgICAgICAgdGhpcy5fcmVzZXRGb2N1cyA9IHRydWU7XG4gICAgICB9XG4gICAgICB0aGlzLl9jYWxlbmRhclZpZXcgPSB0aGlzLl9uZXh0Q2FsZW5kYXJWaWV3O1xuICAgIH0gZWxzZSBpZiAoZXZlbnQuYW5pbWF0aW9uTmFtZSA9PT0gJ3Nob3cnKSB7XG4gICAgICB0aGlzLmludGVybmFscy5zdGF0ZXMuZGVsZXRlKCd0cmFuc2l0aW9uJyk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfc3RhcnRUYWJsZVRyYW5zaXRpb24oKTogdm9pZCB7XG4gICAgdGhpcy5pbnRlcm5hbHMuc3RhdGVzLmFkZCgndHJhbnNpdGlvbicpO1xuICAgIHRoaXMuc2hhZG93Um9vdFxuICAgICAgPy5xdWVyeVNlbGVjdG9yQWxsKCd0YWJsZScpXG4gICAgICA/LmZvckVhY2goKGUpID0+IGUuY2xhc3NMaXN0LnRvZ2dsZSgnc2JiLWNhbGVuZGFyX190YWJsZS1oaWRlJykpO1xuICB9XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIHJlbmRlcigpOiBUZW1wbGF0ZVJlc3VsdCB7XG4gICAgcmV0dXJuIGh0bWxgPGRpdiBjbGFzcz1cInNiYi1jYWxlbmRhcl9fd3JhcHBlclwiPiR7dGhpcy5fZ2V0VmlldygpfTwvZGl2PmA7XG4gIH1cbn1cblxuZGVjbGFyZSBnbG9iYWwge1xuICBpbnRlcmZhY2UgSFRNTEVsZW1lbnRUYWdOYW1lTWFwIHtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25hbWluZy1jb252ZW50aW9uXG4gICAgJ3NiYi1jYWxlbmRhcic6IFNiYkNhbGVuZGFyRWxlbWVudDtcbiAgfVxuICBpbnRlcmZhY2UgSFRNTEVsZW1lbnRFdmVudE1hcCB7XG4gICAgbW9udGhjaGFuZ2U6IFNiYk1vbnRoQ2hhbmdlRXZlbnQ7XG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7O0FDdURBLElBQWEsc0JBQWIsY0FBeUMsTUFBSztDQUc1QyxJQUFXLFFBQUs7RUFDZCxPQUFPLEtBQUs7Q0FDZDtDQUVBLFlBQW1CLE9BQXFCO0VBQ3RDLE1BQU0sZUFBZTtHQUFFLFNBQVM7R0FBTSxVQUFVO0VBQUksQ0FBRTtFQUN0RCxLQUFLLFNBQVMsT0FBTyxPQUFPLFNBQVMsQ0FBQSxDQUFFO0NBQ3pDOzs7Ozs7O0lBMEVXLDRCQUFrQjttQkFBbUI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Y0FBckMsMkJBQXFDLFlBQVU7Ozt1QkFrQnpELFVBQVMsR0FDVCxTQUFTLEVBQUUsTUFBTSxRQUFPLENBQUUsQ0FBQzt1QkFJM0IsU0FBUSxDQUFFO3NCQU1WLFVBQVMsR0FDVCxTQUFRLENBQUU7c0JBT1YsVUFBUyxHQUNULFNBQVEsQ0FBRTs7SUFJVixVQUFTO0lBQ1Qsc0JBQXNCLEdBQTBCLGFBQy9DLEVBQUUsbUJBQW1CLFFBQVEsQ0FBQztJQUUvQixTQUFTLEVBQUUsTUFBTSxRQUFPLENBQUU7R0FBQzsrQkFNM0IsU0FBUSxDQUFFOzRCQThCVixNQUFLLENBQUU7NkJBR1AsU0FBUyxFQUFFLFdBQVcsY0FBYSxDQUFFLENBQUM7OEJBSXRDLFNBQVMsRUFBRSxTQUFTLEtBQUksQ0FBRSxDQUFDOzhCQUkzQixVQUFTLEdBQ1QsU0FBUztJQUFFLFdBQVc7SUFBZ0IsTUFBTTtHQUFPLENBQUUsQ0FBQzs4QkFNdEQsTUFBSyxDQUFFOzRCQUdQLE1BQUssQ0FBRTtnQ0FZUCxNQUFLLENBQUU7K0JBaUVQLE1BQUssQ0FBRTtHQWhLUixhQUFBLE1BQUEsTUFBQSxrQkFBQTtJQUFBLE1BQUE7SUFBQSxNQUFBO0lBQUEsUUFBQTtJQUFBLFNBQUE7SUFBQSxRQUFBO0tBQUEsTUFBQSxRQUFBLFVBQUE7S0FBQSxNQUFBLFFBQUEsSUFBZ0I7S0FBSSxNQUFBLEtBQUEsVUFBQTtNQUFBLElBQUosT0FBSTtLQUFBO0lBQUE7SUFBQSxVQUFBO0dBQUEsR0FBQSxvQkFBQSx1QkFBQTtHQUdSLGFBQUEsTUFBQSxNQUFBLGtCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsVUFBQTtLQUFBLE1BQUEsUUFBQSxJQUFnQjtLQUFJLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBSixPQUFJO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLG9CQUFBLHVCQUFBO0dBUWhDLGFBQUEsTUFBQSxNQUFBLGlCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsU0FBQTtLQUFBLE1BQUEsUUFBQSxJQUFnQjtLQUFHLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBSCxNQUFHO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLG1CQUFBLHNCQUFBO0dBUW5CLGFBQUEsTUFBQSxNQUFBLGlCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsU0FBQTtLQUFBLE1BQUEsUUFBQSxJQUFnQjtLQUFHLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBSCxNQUFHO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLG1CQUFBLHNCQUFBO0dBUW5CLGFBQUEsTUFBQSxNQUFBLHNCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsY0FBQTtLQUFBLE1BQUEsUUFBQSxJQUFnQjtLQUFRLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBUixXQUFRO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLHdCQUFBLDJCQUFBO0dBTXhCLGFBQUEsTUFBQSxNQUFBLDBCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsY0FBQTtLQUFBLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBVyxXQUFRO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLE1BQUEsMEJBQUE7R0E2QlYsYUFBQSxNQUFBLE1BQUEsdUJBQUE7SUFBQSxNQUFBO0lBQUEsTUFBQTtJQUFBLFFBQUE7SUFBQSxTQUFBO0lBQUEsUUFBQTtLQUFBLE1BQUEsUUFBQSxlQUFBO0tBQUEsTUFBQSxRQUFBLElBQWlCO0tBQVMsTUFBQSxLQUFBLFVBQUE7TUFBQSxJQUFULFlBQVM7S0FBQTtJQUFBO0lBQUEsVUFBQTtHQUFBLEdBQUEseUJBQUEsNEJBQUE7R0FJbkMsYUFBQSxNQUFBLE1BQUEsd0JBQUE7SUFBQSxNQUFBO0lBQUEsTUFBQTtJQUFBLFFBQUE7SUFBQSxTQUFBO0lBQUEsUUFBQTtLQUFBLE1BQUEsUUFBQSxnQkFBQTtLQUFBLE1BQUEsUUFBQSxJQUFnQjtLQUFVLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBVixhQUFVO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLDBCQUFBLDZCQUFBO0dBR0csYUFBQSxNQUFBLE1BQUEseUJBQUE7SUFBQSxNQUFBO0lBQUEsTUFBQTtJQUFBLFFBQUE7SUFBQSxTQUFBO0lBQUEsUUFBQTtLQUFBLE1BQUEsUUFBQSxpQkFBQTtLQUFBLE1BQUEsUUFBQSxJQUFnQjtLQUFXLE1BQUEsS0FBQSxVQUFBO01BQUEsSUFBWCxjQUFXO0tBQUE7SUFBQTtJQUFBLFVBQUE7R0FBQSxHQUFBLDJCQUFBLDhCQUFBO0dBTXhELGFBQUEsTUFBQSxNQUFBLHlCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsaUJBQUE7S0FBQSxNQUFBLFFBQUEsSUFBZ0I7S0FBVyxNQUFBLEtBQUEsVUFBQTtNQUFBLElBQVgsY0FBVztLQUFBO0lBQUE7SUFBQSxVQUFBO0dBQUEsR0FBQSwyQkFBQSw4QkFBQTtHQUtsQixhQUFBLE1BQUEsTUFBQSx5QkFBQTtJQUFBLE1BQUE7SUFBQSxNQUFBO0lBQUEsUUFBQTtJQUFBLFNBQUE7SUFBQSxRQUFBO0tBQUEsTUFBQSxRQUFBLGlCQUFBO0tBQUEsTUFBQSxRQUFBLElBQWlCO0tBQVcsTUFBQSxLQUFBLFVBQUE7TUFBQSxJQUFYLGNBQVc7S0FBQTtJQUFBO0lBQUEsVUFBQTtHQUFBLEdBQUEsMkJBQUEsOEJBQUE7R0FJckMsYUFBQSxNQUFBLE1BQUEsdUJBQUE7SUFBQSxNQUFBO0lBQUEsTUFBQTtJQUFBLFFBQUE7SUFBQSxTQUFBO0lBQUEsUUFBQTtLQUFBLE1BQUEsUUFBQSxXQUFBO0tBQUEsTUFBQSxLQUFBLFVBQUE7TUFBQSxJQUFZLFFBQUs7S0FBQTtJQUFBO0lBQUEsVUFBQTtHQUFBLEdBQUEsTUFBQSwwQkFBQTtHQVdSLGFBQUEsTUFBQSxNQUFBLDJCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsbUJBQUE7S0FBQSxNQUFBLFFBQUEsSUFBaUI7S0FBYSxNQUFBLEtBQUEsVUFBQTtNQUFBLElBQWIsZ0JBQWE7S0FBQTtJQUFBO0lBQUEsVUFBQTtHQUFBLEdBQUEsNkJBQUEsZ0NBQUE7R0FrRXZDLGFBQUEsTUFBQSxNQUFBLDBCQUFBO0lBQUEsTUFBQTtJQUFBLE1BQUE7SUFBQSxRQUFBO0lBQUEsU0FBQTtJQUFBLFFBQUE7S0FBQSxNQUFBLFFBQUEsa0JBQUE7S0FBQSxNQUFBLFFBQUEsSUFBaUI7S0FBWSxNQUFBLEtBQUEsVUFBQTtNQUFBLElBQVosZUFBWTtLQUFBO0lBQUE7SUFBQSxVQUFBO0dBQUEsR0FBQSw0QkFBQSwrQkFBQTs7Ozs7Ozs7O0dBcExHLEtBQUEsY0FBc0I7RUFBZTs7R0FDOUMsS0FBQSxzQkFBd0M7SUFDN0Q7SUFDQTtJQUNBO0lBQ0E7SUFDQTtJQUNBO0lBQ0E7O0VBQ0E7O0dBQ3FCLEtBQUEsU0FBeUIsQ0FBQyx3QkFBd0IsVUFBVSxnQkFBSyxDQUFDO0VBQUU7O0dBQ3BFLEtBQUEsU0FBUztJQUM5QixjQUFjO0lBQ2QsYUFBYTs7RUFDSjtFQUtYOztFQUFBLElBQWdCLE9BQUk7R0FBQSxPQUFBLEtBQUE7RUFBQTtFQUFwQixJQUFnQixLQUFJLE9BQUE7R0FBQSxLQUFBLHlCQUFBO0VBQUE7RUFHUjs7RUFBQSxJQUFnQixPQUFJO0dBQUEsT0FBQSxLQUFBO0VBQUE7RUFBcEIsSUFBZ0IsS0FBSSxPQUFBO0dBQUEsS0FBQSx5QkFBQTtFQUFBO0VBUWhDOzs7OztFQUFBLElBQWdCLE1BQUc7R0FBQSxPQUFBLEtBQUE7RUFBQTtFQUFuQixJQUFnQixJQUFHLE9BQUE7R0FBQSxLQUFBLHdCQUFBO0VBQUE7RUFRbkI7Ozs7O0VBQUEsSUFBZ0IsTUFBRztHQUFBLE9BQUEsS0FBQTtFQUFBO0VBQW5CLElBQWdCLElBQUcsT0FBQTtHQUFBLEtBQUEsd0JBQUE7RUFBQTtFQVFuQjs7RUFBQSxJQUFnQixXQUFRO0dBQUEsT0FBQSxLQUFBO0VBQUE7RUFBeEIsSUFBZ0IsU0FBUSxPQUFBO0dBQUEsS0FBQSw2QkFBQTtFQUFBOzs7O0VBTXhCLElBQVcsU0FBUyxPQUFxQjtHQUN2QyxJQUFJLE1BQU0sUUFBUSxLQUFLLEdBQ3JCLEtBQUssWUFBWSxNQUNkLEtBQUssYUFDSixLQUFLLGFBQWEsbUJBQW1CLEtBQUssYUFBYSxZQUFZLFFBQVEsQ0FBQyxDQUFDLEVBRTlFLFFBQVEsU0FBOEIsU0FBUyxJQUFJLEVBQ25ELFFBQ0UsU0FDQyxDQUFDLEtBQUssY0FBYyxLQUFLLGFBQWEsVUFBVSxJQUFJLENBQUMsS0FBSyxLQUFLLFlBQVksSUFBSSxDQUFDO1FBRWpGO0lBQ0wsTUFBTSxlQUFlLEtBQUssYUFBYSxtQkFDckMsS0FBSyxhQUFhLFlBQVksS0FBSyxDQUFDO0lBRXRDLElBQ0UsQ0FBQyxDQUFDLGlCQUNELENBQUMsS0FBSyxjQUFjLEtBQUssYUFBYSxVQUFVLFlBQVksQ0FBQyxLQUM1RCxLQUFLLFlBQVksWUFBWSxJQUUvQixLQUFLLFlBQVk7U0FFakIsS0FBSyxZQUFZO0dBRXJCO0VBQ0Y7RUFDQSxJQUFXLFdBQVE7R0FDakIsT0FBTyxLQUFLO0VBQ2Q7RUFDUztFQUFBLElBQWlCLFlBQVM7R0FBQSxPQUFBLEtBQUE7RUFBQTtFQUExQixJQUFpQixVQUFTLE9BQUE7R0FBQSxLQUFBLDhCQUFBO0VBQUE7RUFJbkM7O0VBQUEsSUFBZ0IsYUFBVTtHQUFBLE9BQUEsS0FBQTtFQUFBO0VBQTFCLElBQWdCLFdBQVUsT0FBQTtHQUFBLEtBQUEsK0JBQUE7RUFBQTtFQUdHOztFQUFBLElBQWdCLGNBQVc7R0FBQSxPQUFBLEtBQUE7RUFBQTtFQUEzQixJQUFnQixZQUFXLE9BQUE7R0FBQSxLQUFBLGdDQUFBO0VBQUE7RUFNeEQ7O0VBQUEsSUFBZ0IsY0FBVztHQUFBLE9BQUEsS0FBQTtFQUFBO0VBQTNCLElBQWdCLFlBQVcsT0FBQTtHQUFBLEtBQUEsZ0NBQUE7RUFBQTtFQUtsQjs7RUFBQSxJQUFpQixjQUFXO0dBQUEsT0FBQSxLQUFBO0VBQUE7RUFBNUIsSUFBaUIsWUFBVyxPQUFBO0dBQUEsS0FBQSxnQ0FBQTtFQUFBOztFQUlyQyxJQUFZLE1BQU0sTUFBYTtHQUM3QixLQUFLLFlBQVksUUFBUSxJQUFJO0dBQzdCLEtBQUssZ0JBQWdCO0VBQ3ZCO0VBQ0EsSUFBWSxRQUFLO0dBQ2YsT0FBTyxLQUFLO0VBQ2Q7RUFLUztFQUFBLElBQWlCLGdCQUFhO0dBQUEsT0FBQSxLQUFBO0VBQUE7RUFBOUIsSUFBaUIsY0FBYSxPQUFBO0dBQUEsS0FBQSxrQ0FBQTtFQUFBOztFQTJDdkMsSUFBWSxTQUFNO0dBQ2hCLE9BQU8sTUFBTSxNQUNWLEtBQUssa0JBQWtCLFFBQ25CLE1BQU0sS0FBSyxLQUFLLFdBQVksaUJBQWlCLE1BQU0sQ0FBQyxFQUFFLFNBQVMsTUFDOUQsRUFBRSxpQkFBaUIsRUFBRSxTQUFTLEtBQUksQ0FBRSxDQUFDLElBRXZDLEtBQUssWUFBWSxpQkFBaUIsZ0JBQWdCLEtBQUssZUFBZSxNQUFNLENBQUEsQ0FBRTtFQUV0RjtFQWVBO0VBQUEsSUFBaUIsZUFBWTtHQUFBLE9BQUEsS0FBQTtFQUFBO0VBQTdCLElBQWlCLGFBQVksT0FBQTtHQUFBLEtBQUEsaUNBQUE7RUFBQTtFQVU3QixjQUFBO0dBQ0UsTUFBSztHQTVLUyxLQUFBLDBCQXBCTCxrQkFBQSxNQUFBLDBCQUFBLEdBQWtCLGtCQUFBLE1BQUEsb0JBb0JHLEtBQUs7R0FHVCxLQUFBLDBCQUFBLGtCQUFBLE1BQUEsdUJBQUEsR0FBQSxrQkFBQSxNQUFBLG9CQUFpQyxLQUFLO0dBUWxELEtBQUEseUJBQUEsa0JBQUEsTUFBQSx1QkFBQSxHQUFBLGtCQUFBLE1BQUEsbUJBQWdCLElBQUk7R0FRcEIsS0FBQSx5QkFBQSxrQkFBQSxNQUFBLHNCQUFBLEdBQUEsa0JBQUEsTUFBQSxtQkFBZ0IsSUFBSTtHQVFwQixLQUFBLDhCQUFBLGtCQUFBLE1BQUEsc0JBQUEsR0FBQSxrQkFBQSxNQUFBLHdCQUFvQixLQUFLO0dBbUNmLEtBQUEsK0JBQUEsa0JBQUEsTUFBQSwyQkFBQSxHQUFBLGtCQUFBLE1BQUEseUJBQTRCLElBQUk7R0FJMUMsS0FBQSxnQ0FBQSxrQkFBQSxNQUFBLDRCQUFBLEdBQUEsa0JBQUEsTUFBQSwwQkFBbUQsSUFBSTtHQUcxQixLQUFBLGlDQUFBLGtCQUFBLE1BQUEsNkJBQUEsR0FBQSxrQkFBQSxNQUFBLDJCQUMzQyxZQUFZO0dBS0UsS0FBQSxpQ0FBQSxrQkFBQSxNQUFBLDhCQUFBLEdBQUEsa0JBQUEsTUFBQSwyQkFBdUIsS0FBSztHQUVwQyxLQUFBLGdCQUFZLGtCQUFBLE1BQUEsOEJBQUEsR0FBbUIsV0FBVSxFQUFHLFVBQVUsZUFBZTtHQUduRCxLQUFBLGdDQUFBLGtCQUFBLE1BQUEsMkJBQWlCLEtBQUssYUFBYSxNQUFLLENBQUU7R0FhNUQsS0FBQSxpQkFBYSxrQkFBQSxNQUFBLDhCQUFBLEdBQVk7R0FFUCxLQUFBLGtDQUFBLGtCQUFBLE1BQUEsNkJBQTRDLEtBQUs7R0FFbkUsS0FBQSxxQkFBaUIsa0JBQUEsTUFBQSxnQ0FBQSxHQUErQjs7R0FHaEQsS0FBQSx1Q0FBb0Y7SUFDMUYsZ0JBQWdCO0lBQ2hCLGVBQWU7SUFDZixrQkFBa0I7SUFDbEIsa0JBQWtCO0lBQ2xCLG1CQUFtQjs7O0dBT2IsS0FBQSxTQUFxQixDQUFBOztHQWVyQixLQUFBLGNBQXdCLEtBQUssYUFBYSxjQUFjLE1BQU07R0FROUQsS0FBQSxtQkFBNEI7O0dBb0I1QixLQUFBLGNBQWM7O0dBR2QsS0FBQSxtQkFBbUI7R0FHVixLQUFBLGlDQUFBLGtCQUFBLE1BQUEsNEJBQWUsS0FBSztHQUU3QixLQUFBLGFBQVMsa0JBQUEsTUFBQSwrQkFBQSxHQUFHLElBQUksc0JBQXNCLElBQUksRUFBRSxrQkFBaUI7SUFDbkUsS0FBSyxjQUFjLEtBQUssYUFBYSxjQUFjLE1BQU07SUFDekQsS0FBSyxpQkFBZ0I7R0FDdkIsQ0FBQztHQUNPLEtBQUEsZ0JBQWdCLElBQUksMEJBQTBCLE1BQU0sR0FDekQsNkNBQTZDLEtBQUssTUFBSyxFQUFBLENBQ3pEO0dBb0ZPLEtBQUEsc0JBQTJCO0lBQ2pDLEtBQUssbUJBQW1CLE1BQU0sS0FBSyxLQUFLLFFBQVEsRUFBRSxNQUMvQyxNQUFNLEVBQUUsY0FBYyxrQkFBa0I7SUFFM0MsS0FBSyxZQUFZLFlBQVksS0FBSyxnQkFBZ0I7SUFDbEQsS0FBSyxhQUFZO0dBQ25CO0dBdEZFLEtBQUssaUJBQWdCO0dBQ3JCLEtBQUssYUFBWTtHQUtqQixLQUFLLGlCQUFpQixpQkFBa0IsS0FBSyxtQkFBbUIsSUFBSztHQUNyRSxLQUFLLGlCQUFpQixrQkFBbUIsS0FBSyxtQkFBbUIsS0FBTTtHQUN2RSxLQUFLLGlCQUFpQixVQUFVLE1BQUs7SUFDbkMsTUFBTSxNQUFPLEVBQUUsT0FBdUIsUUFBa0Msa0JBQWtCO0lBQzFGLElBQUksS0FDRixLQUFLLFlBQVksSUFBSSxLQUFNO0dBRS9CLENBQUM7R0FDRCxLQUFLLGlCQUFpQixZQUFZLE1BQUs7SUFDckMsSUFBSyxFQUFFLE9BQXVCLGNBQWMsb0JBQzFDLEtBQUsscUJBQ0gsR0FDQSxLQUFLLGNBQWUsRUFBRSxPQUFpQyxLQUFXLENBQUM7R0FHekUsQ0FBQztFQUNIO0VBRVEsWUFBWSxNQUFPO0dBQ3pCLE9BQU8sS0FBSyxhQUFhLElBQUksS0FBSztFQUNwQzs7RUFHTyxnQkFBYTtHQUNsQixLQUFLLG1CQUFrQjtHQUN2QixLQUFLLE1BQUs7RUFDWjtFQUVnQixvQkFBaUI7R0FDL0IsTUFBTSxrQkFBaUI7R0FDdkIsS0FBSyxjQUFhO0dBQ2xCLEtBQUssWUFBWSxpQkFBaUIsY0FBYyxLQUFLLGVBQWUsRUFBRSxTQUFTLEtBQUksQ0FBRTtFQUN2RjtFQUVnQix1QkFBb0I7R0FDbEMsTUFBTSxxQkFBb0I7R0FDMUIsS0FBSyxZQUFZLG9CQUFvQixjQUFjLEtBQUssZUFBZSxFQUFFLFNBQVMsS0FBSSxDQUFFO0VBQzFGOztFQUdnQixRQUFLO0dBQ25CLEtBQUssY0FBYztHQUNuQixLQUFLLFdBQVU7RUFDakI7RUFFbUIsV0FBVyxtQkFBdUM7R0FDbkUsTUFBTSxXQUFXLGlCQUFpQjtHQUVsQyxJQUFJLENBQUMsS0FBSyxjQUNSO0dBR0YsSUFBSSxrQkFBa0IsSUFBSSxNQUFNLEtBQUssa0JBQWtCLElBQUksYUFBYSxHQUN0RSxLQUFLLGNBQWE7R0FHcEIsSUFBSSxrQkFBa0IsSUFBSSxNQUFNLEdBQUc7SUFDakMsS0FBSyxlQUFjO0lBQ25CLEtBQUssZUFBZSxLQUFBO0lBQ3BCLEtBQUssb0JBQW9CLEtBQUssZ0JBQWdCLEtBQUs7R0FDckQ7RUFDRjtFQUVtQixRQUFRLG1CQUF1QztHQUNoRSxNQUFNLFFBQVEsaUJBQWlCO0dBRy9CLEtBQUssYUFBWTtHQUlqQixLQUFLLFdBQVU7RUFDakI7Ozs7OztFQWVRLG1CQUFtQixZQUFtQjtHQUM1QyxJQUFJLGNBQWMsQ0FBQyxNQUFNLFFBQVEsS0FBSyxTQUFTLEdBQzdDLEtBQUssWUFBWSxLQUFLLFlBQVksQ0FBQyxLQUFLLFNBQWMsSUFBSSxDQUFBO0dBRTVELElBQUksQ0FBQyxjQUFjLE1BQU0sUUFBUSxLQUFLLFNBQVMsR0FDN0MsS0FBSyxZQUFhLEtBQUssVUFBa0IsU0FBVSxLQUFLLFVBQWtCLEtBQUs7RUFFbkY7O0VBR1EsTUFBTSxZQUFjO0dBRTFCLElBQUksVUFDRjtRQUNLLElBQUksS0FBSyxtQkFBbUI7SUFDakMsS0FBSyxrQkFBa0IsV0FBVyxLQUFLLE1BQUssQ0FBRTtJQUM5QztHQUNGO0dBRUEsSUFBSSxZQUNGLEtBQUssa0JBQWtCLFVBQVU7R0FFbkMsS0FBSyxTQUNGLEtBQUssY0FBYyxRQUFRLG9DQUFvQyxLQUFLLFVBQVUsS0FBSztHQUN0RixLQUFLLFNBQVMsS0FBSyxnQkFBZ0IsS0FBSyxXQUFXO0dBQ25ELEtBQUssU0FBUyxLQUFLLGdCQUFlO0dBQ2xDLEtBQUssZUFBZSxLQUFLLE9BQ3RCLEtBQUksRUFDSixNQUFNLEdBQUcsTUFBTSxFQUFFLE1BQU0sY0FBYyxFQUFFLEtBQUssQ0FBQyxFQUM3QyxLQUFLLFFBQWdCLElBQUksU0FBUyxFQUNsQyxRQUFRLEdBQUcsR0FBRyxNQUFNLEVBQUUsUUFBUSxDQUFDLE1BQU0sQ0FBQztHQUN6QyxLQUFLLGtCQUFrQixDQUFDLENBQUEsQ0FBRTtHQUMxQixLQUFLLGtCQUFrQixDQUFDLENBQUEsQ0FBRTtHQUMxQixJQUFJLEtBQUssT0FBTztJQUNkLE1BQU0sZ0JBQWdCLEtBQUssYUFBYSxrQkFBa0IsS0FBSyxhQUFhLENBQUM7SUFDN0UsS0FBSyxrQkFBa0IsS0FBSyxnQkFBZ0IsZUFBZSxJQUFJO0lBQy9ELEtBQUssa0JBQWtCLEtBQUssZ0JBQWdCLGNBQWM7SUFDMUQsS0FBSyx3QkFBd0IsS0FBSyxnQkFDL0IsS0FBSSxFQUNKLE1BQU0sR0FBRyxNQUFNLEVBQUUsTUFBTSxjQUFjLEVBQUUsS0FBSyxDQUFDLEVBQzdDLEtBQUssUUFBZ0IsSUFBSSxTQUFTLEVBQ2xDLFFBQVEsR0FBRyxHQUFHLE1BQU0sRUFBRSxRQUFRLENBQUMsTUFBTSxDQUFDO0dBQzNDO0dBQ0EsS0FBSyxlQUFlO0VBQ3RCOztFQUdRLGFBQVU7R0FDaEIsSUFBSSxLQUFLLGFBQWE7SUFDcEIsS0FBSyxtQkFBa0IsR0FBSSxNQUFLO0lBQ2hDLEtBQUssY0FBYztHQUNyQjtFQUNGOztFQUdRLGVBQVk7R0FDbEIsTUFBTSxpQkFBMkIsS0FBSyxhQUFhLGtCQUFrQixRQUFRO0dBRTdFLE1BQU0sV0FEeUIsS0FBSyxhQUFhLGtCQUFrQixNQUN2QyxFQUFhLEtBQUssTUFBYyxPQUFlO0lBQ3pFO0lBQ0EsUUFBUSxlQUFlO0tBQ3ZCO0dBR0YsTUFBTSxpQkFBeUIsS0FBSyxhQUFhLGtCQUFpQjtHQUNsRSxLQUFLLFlBQVksU0FBUyxNQUFNLGNBQWMsRUFBRSxPQUFPLFNBQVMsTUFBTSxHQUFHLGNBQWMsQ0FBQztFQUMxRjs7RUFHUSxnQkFBZ0IsT0FBVSxzQkFBc0IsT0FBSztHQUMzRCxNQUFNLGNBQXNCLEtBQUssYUFBYSxrQkFBa0IsS0FBSztHQUNyRSxNQUFNLGFBQXFCLEtBQUssYUFBYSxtQkFBbUIsS0FBSztHQUNyRSxJQUFJLENBQUMscUJBQXFCO0lBQ3hCLEtBQUsscUNBQXFDLG1CQUFtQjtJQUM3RCxLQUFLLHFDQUFxQyxtQkFBbUI7SUFDN0QsS0FBSyxxQ0FBcUMsaUJBQWlCLEtBQUssYUFBYSxVQUMzRSxLQUFLLGFBQWEsV0FDaEIsS0FBSyxhQUFhLFFBQVEsS0FBSyxHQUMvQixLQUFLLGFBQWEsU0FBUyxLQUFLLEdBQ2hDLENBQUMsQ0FDRjtJQUVILEtBQUsscUNBQXFDLGdCQUFnQixLQUFLLGFBQWEsVUFDMUUsS0FBSyxhQUFhLFdBQ2hCLEtBQUssYUFBYSxRQUFRLEtBQUssR0FDL0IsS0FBSyxhQUFhLFNBQVMsS0FBSyxHQUNoQyxXQUFXLENBQ1o7R0FFTCxPQUFPO0lBQ0wsS0FBSyxxQ0FBcUMsb0JBQW9CO0lBQzlELEtBQUsscUNBQXFDLGdCQUFnQixLQUFLLGFBQWEsVUFDMUUsS0FBSyxhQUFhLFdBQ2hCLEtBQUssYUFBYSxRQUFRLEtBQUssR0FDL0IsS0FBSyxhQUFhLFNBQVMsS0FBSyxHQUNoQyxXQUFXLENBQ1o7R0FFTDtHQUNBLE9BQU8sS0FBSyxnQkFBZ0IsZUFDeEIsS0FBSywwQkFBMEIsT0FBTyxhQUFhLFVBQVUsSUFDN0QsS0FBSyx3QkFBd0IsT0FBTyxhQUFhLFVBQVU7RUFDakU7Ozs7Ozs7OztFQVVRLDBCQUEwQixPQUFVLGFBQXFCLFlBQWtCO0dBQ2pGLE1BQU0sUUFBb0IsQ0FBQyxDQUFBLENBQUU7R0FDN0IsS0FBSyxJQUFJLElBQUksR0FBRyxPQUFPLFlBQVksSUFBSSxhQUFhLEtBQUssUUFBUTtJQUMvRCxJQUFJLFNBQVMsY0FBYztLQUN6QixNQUFNLEtBQUssQ0FBQSxDQUFFO0tBQ2IsT0FBTztJQUNUO0lBQ0EsTUFBTSxPQUFPLEtBQUssYUFBYSxXQUM3QixLQUFLLGFBQWEsUUFBUSxLQUFLLEdBQy9CLEtBQUssYUFBYSxTQUFTLEtBQUssR0FDaEMsSUFBSSxDQUFDO0lBRVAsTUFBTSxNQUFNLFNBQVMsR0FBRyxLQUFLLEtBQUssY0FBYyxJQUFJLENBQUM7R0FDdkQ7R0FDQSxPQUFPO0VBQ1Q7Ozs7Ozs7Ozs7Ozs7OztFQWdCUSx3QkFBd0IsT0FBVSxhQUFxQixZQUFrQjtHQUMvRSxNQUFNLFFBQW9CLE1BQU0sS0FBSyxFQUFFLFFBQVEsYUFBWSxTQUFVLENBQUEsQ0FBRTtHQUN2RSxLQUFLLElBQUksSUFBSSxHQUFHLE9BQU8sWUFBWSxJQUFJLGFBQWEsS0FBSyxRQUFRO0lBQy9ELElBQUksU0FBUyxjQUNYLE9BQU87SUFFVCxNQUFNLE9BQU8sS0FBSyxhQUFhLFdBQzdCLEtBQUssYUFBYSxRQUFRLEtBQUssR0FDL0IsS0FBSyxhQUFhLFNBQVMsS0FBSyxHQUNoQyxJQUFJLENBQUM7SUFFUCxNQUFNLE1BQU0sS0FBSyxLQUFLLGNBQWMsSUFBSSxDQUFDO0dBQzNDO0dBQ0EsT0FBTztFQUNUO0VBRVEsY0FBYyxNQUFPO0dBRTNCLE9BQU87SUFDTCxPQUZjLEtBQUssYUFBYSxVQUFVLElBRW5DO0lBQ1AsV0FBVztJQUNYLFVBQVUsT0FBTyxLQUFLLGFBQWEsUUFBUSxJQUFJLENBQUM7SUFDaEQsWUFBWSxPQUFPLEtBQUssYUFBYSxTQUFTLElBQUksQ0FBQztJQUNuRCxXQUFXLE9BQU8sS0FBSyxhQUFhLFFBQVEsSUFBSSxDQUFDO0lBR2pELFdBQVcsS0FBSyxTQUFTLElBQUk7SUFDN0IsY0FBYyxLQUFLLGFBQWEsYUFBYSxJQUFJOztFQUVyRDtFQUVRLFNBQVMsTUFBTztHQUN0QixNQUFNLGlCQUFpQixLQUFLLGFBQWEsV0FBVyxLQUFLLGFBQWEsUUFBUSxJQUFJLEdBQUcsR0FBRyxDQUFDO0dBQ3pGLE1BQU0sVUFBVSxLQUFLLGFBQWEsYUFBYSxjQUFjO0dBRTdELElBQUksWUFBWTtJQUFDO0lBQVE7SUFBUztJQUFXO0dBQVEsRUFBRSxTQUFTLE9BQU8sSUFBSSxJQUFJO0dBQy9FLElBQUksWUFBWSxLQUFLLGFBQWEsZ0JBQ2hDLGdCQUNBLEtBQUssYUFBYSxtQkFBbUIsY0FBYyxJQUFJLEVBQUU7R0FFM0QsT0FBTyxLQUFLLGFBQWEsWUFBWSxXQUFXLElBQUksS0FBSyxHQUFHO0lBQzFELE1BQU0sVUFBVSxLQUFLLGFBQWEsZ0JBQWdCLFdBQVcsQ0FBQztJQUM5RCxJQUFJLEtBQUssYUFBYSxZQUFZLE1BQU0sT0FBTyxLQUFLLEdBQ2xELE9BQU8sWUFBWSxJQUNmLFlBQ0EsS0FBSyxTQUFTLEtBQUssYUFBYSxnQkFBZ0IsZ0JBQWdCLEVBQUUsQ0FBQztJQUV6RSxZQUFZLEtBQUssYUFBYSxnQkFBZ0IsV0FBVyxZQUFZO0lBQ3JFO0dBQ0Y7R0FFQSxNQUFNLElBQUksTUFBTSw4QkFBOEI7RUFDaEQ7O0VBR1EsbUJBQW1CLE9BQW9CO0dBQzdDLE9BQU8sT0FBTyxVQUFVLFdBQVcsUUFBUSxLQUFLLGFBQWEsVUFBVSxLQUFVO0VBQ25GOztFQUdRLG1CQUFnQjtHQUN0QixNQUFNLFNBQXNCLElBQUksTUFBTSxFQUFFLEVBQUUsS0FBSyxJQUFJLEVBQUUsS0FDbEQsR0FBRyxPQUEwQjtJQUM1QixPQUFPLE9BQU8sSUFBSSxDQUFDLEVBQUUsU0FBUyxHQUFHLEdBQUc7SUFDcEMsWUFBWSxJQUFJO0tBQ2hCO0dBRUosTUFBTSxPQUFlLEtBQUs7R0FDMUIsTUFBTSxhQUE0QixDQUFBO0dBQ2xDLEtBQUssSUFBSSxJQUFZLEdBQUcsSUFBSSxNQUFNLEtBQ2hDLFdBQVcsS0FBSyxPQUFPLE1BQU0saUJBQWlCLEdBQUcsa0JBQWtCLElBQUksRUFBRSxDQUFDO0dBRTVFLEtBQUssVUFBVTtFQUNqQjs7RUFHUSxnQkFBZ0IsU0FBaUIsR0FBQztHQUN4QyxNQUFNLHFCQUE2QixLQUFLLHVCQUFzQjtHQUM5RCxNQUFNLFdBQXFCLElBQUksTUFBTSxjQUFjLEVBQ2hELEtBQUssQ0FBQyxFQUNOLEtBQUssR0FBRyxNQUFjLHFCQUFxQixTQUFTLENBQUM7R0FDeEQsTUFBTSxPQUFlLGlCQUFpQjtHQUN0QyxNQUFNLFlBQXdCLENBQUE7R0FDOUIsS0FBSyxJQUFJLElBQVksR0FBRyxJQUFJLE1BQU0sS0FDaEMsVUFBVSxLQUFLLFNBQVMsTUFBTSxnQkFBZ0IsR0FBRyxpQkFBaUIsSUFBSSxFQUFFLENBQUM7R0FFM0UsT0FBTztFQUNUOzs7Ozs7Ozs7OztFQVlRLHlCQUFzQjtHQUM1QixJQUFJLGVBQWU7R0FDbkIsSUFBSSxLQUFLLEtBQ1AsZUFBZSxLQUFLLGFBQWEsUUFBUSxLQUFLLEdBQUcsSUFBSSxpQkFBaUI7UUFDakUsSUFBSSxLQUFLLEtBQ2QsZUFBZSxLQUFLLGFBQWEsUUFBUSxLQUFLLEdBQUc7R0FFbkQsTUFBTSxhQUFhLEtBQUssYUFBYSxRQUFRLEtBQUssV0FBVztHQUM3RCxPQUNFLGVBQ0ksYUFBYSxnQkFBZ0IsaUJBQWtCLGtCQUFrQjtFQUV6RTs7RUFHUSxjQUFjLFlBQWtCO0dBQ3RDLElBQUksQ0FBQyxLQUFLLE9BQU8sQ0FBQyxLQUFLLEtBQ3JCLE9BQU87R0FFVCxNQUFNLE9BQU8sS0FBSyxhQUFhLFlBQVksVUFBVTtHQUNyRCxPQUFPLEtBQUssYUFBYSxTQUFTLE1BQU0sS0FBSyxhQUFhLFVBQVUsTUFBTSxLQUFLLEtBQUssS0FBSyxHQUFHLENBQUM7RUFDL0Y7O0VBR1EsWUFBWSxLQUFNO0dBQ3hCLEtBQUssZUFBZSxLQUFBO0dBQ3BCLEtBQUssZUFBYztHQUNuQixJQUFJLEtBQUssVUFBVTtJQUVqQixJQUFJLEtBQUssYUFBYyxLQUFLLFVBQWtCLFNBQVMsR0FBRztLQUN4RCxNQUFNLHFCQUE4QixLQUFLLFVBQWtCLFdBQ3hELFFBQVEsS0FBSyxhQUFhLFlBQVksS0FBSyxHQUFHLE1BQU0sQ0FBQztLQUd4RCxJQUFJLHVCQUF1QixJQUN6QixLQUFLLFlBQWEsS0FBSyxVQUFrQixRQUFRLEdBQUcsTUFBTSxNQUFNLGtCQUFrQjtVQUVsRixLQUFLLFlBQVksQ0FBQyxHQUFJLEtBQUssV0FBbUIsR0FBRztJQUVyRCxPQUVFLEtBQUssWUFBWSxDQUFDLEdBQUc7SUFFdkIsS0FBSyx1QkFBdUIsS0FBSyxVQUFVLEtBQUssTUFBTSxLQUFLLGFBQWEsWUFBWSxDQUFDLENBQUUsQ0FBQztHQUMxRixPQUVFLElBQUksQ0FBQyxLQUFLLGFBQWEsS0FBSyxhQUFhLFlBQVksS0FBSyxXQUFnQixHQUFHLE1BQU0sR0FBRztJQUNwRixLQUFLLFlBQVk7SUFDakIsS0FBSyx1QkFBdUIsS0FBSyxhQUFhLFlBQVksR0FBRyxDQUFFO0dBQ2pFO0VBRUo7Ozs7Ozs7O0VBU1EscUJBQXFCLE1BQWM7R0FHekMsTUFBTSxjQUF5QixLQUFLLE9BQ2pDLFFBQVEsTUFBTSxDQUFDLEVBQUUsUUFBUSxFQUN6QixLQUFLLE1BQU0sS0FBSyxtQkFBbUIsRUFBRSxLQUFNLENBQUM7R0FDL0MsTUFBTSxZQUFzQixLQUN6QixLQUFLLE1BQWMsRUFBRSxLQUFLLEVBQzFCLFFBQVEsWUFBb0IsWUFBWSxTQUFTLE9BQU8sQ0FBQztHQUM1RCxNQUFNLGVBQWUsSUFBSSxJQUFJLFNBQVM7R0FDdEMsTUFBTSxjQUFjLElBQUksSUFBSyxLQUFLLFVBQWtCLEtBQUssTUFBTSxLQUFLLGFBQWEsVUFBVSxDQUFDLENBQUMsQ0FBQztHQUM5RixNQUFNLGFBQWEsS0FBSyxpQ0FBaUMsV0FBVyxjQUFjLFdBQVc7R0FDN0YsS0FBSyxZQUFZLFdBQVcsS0FBSyxNQUFNLEtBQUssYUFBYSxZQUFZLENBQUMsQ0FBRTtHQUV4RSxLQUFLLHVCQUF1QixLQUFLLFVBQVUsS0FBSyxNQUFNLEtBQUssYUFBYSxZQUFZLENBQUMsQ0FBRSxDQUFDO0VBQzFGOzs7O0VBS1EsdUJBQXVCLFFBQWU7O0dBRTVDLEtBQUssY0FDSCxJQUFJLFlBQXFCLGdCQUFnQjtJQUN2QztJQUNBLFVBQVU7SUFDVixTQUFTO0lBQ1YsQ0FBQztFQUVOO0VBRVEsbUJBQWdCO0dBSXRCLE1BQU0sZUFBZSxLQUFLLE9BQU8sQ0FBQyxHQUFHLEtBQUssUUFBUSxHQUFHLEtBQUssZUFBZSxJQUFJLEtBQUssUUFDL0UsS0FBSSxFQUNKLE1BQU0sR0FBRyxNQUFNLEVBQUUsTUFBTSxjQUFjLEVBQUUsS0FBSyxDQUFDOzs7Ozs7R0FNaEQsS0FBSyxjQUFjLElBQUksb0JBQW9CLFdBQVcsQ0FBQztFQUN6RDs7Ozs7RUFNUSxpQ0FDTixXQUNBLGNBQ0EsYUFBd0I7R0FFeEIsSUFBSSxVQUFVLE9BQU8sUUFBZ0IsWUFBWSxJQUFJLEdBQUcsQ0FBQyxHQUN2RCxhQUFhLFNBQVMsUUFBZ0IsWUFBWSxPQUFPLEdBQUcsQ0FBQztRQUU3RCxhQUFhLFNBQVMsUUFBZ0IsWUFBWSxJQUFJLEdBQUcsQ0FBQztHQUU1RCxPQUFPLE1BQU0sS0FBSyxXQUFXO0VBQy9CO0VBRVEsaUJBQWM7R0FDcEIsSUFBSSxLQUFLLFNBQVMsU0FBUztJQUN6QixJQUFJO0lBQ0osSUFBSSxLQUFLLFVBQ1AsZUFBZ0IsS0FBSyxTQUFpQixHQUFHLEVBQUU7U0FFM0MsZUFBZSxLQUFLO0lBRXRCLEtBQUssY0FBYyxLQUFLLGFBQWEsUUFBUSxnQkFBZ0IsS0FBSyxhQUFhLE1BQUssQ0FBRTtHQUN4RixPQUNFLEtBQUssY0FBYyxLQUFBO0VBRXZCO0VBRVEsa0JBQWtCLE1BQU87R0FDL0IsSUFBSSxLQUFLLE9BQU8sS0FBSyxhQUFhLFlBQVksS0FBSyxLQUFLLElBQUksSUFBSSxHQUFHO0lBQ2pFLEtBQUssY0FBYyxLQUFLO0lBQ3hCO0dBQ0Y7R0FDQSxJQUFJLEtBQUssT0FBTyxLQUFLLGFBQWEsWUFBWSxLQUFLLEtBQUssSUFBSSxJQUFJLEdBQUc7SUFDakUsS0FBSyxjQUFjLEtBQUs7SUFDeEI7R0FDRjtHQUNBLEtBQUssY0FBYztFQUNyQjs7RUFHUSxvQkFBb0IsUUFBYztHQUN4QyxLQUFLLE1BQU0sS0FBSyxhQUFhLGtCQUFrQixLQUFLLGFBQWEsTUFBTSxDQUFDO0dBQ3hFLEtBQUssaUJBQWdCO0VBQ3ZCO0VBRVEsbUJBQW1CLE9BQWE7R0FDdEMsS0FBSyxlQUFnQjtHQUVyQixLQUFLLGNBQWMsS0FBSyxhQUFhLFdBQ25DLEtBQUssYUFDTCxLQUFLLGFBQWEsU0FBUyxLQUFLLFdBQVcsR0FDM0MsS0FBSyxhQUFhLFFBQVEsS0FBSyxXQUFXLENBQUM7R0FFN0MsS0FBSyxNQUFLO0VBQ1o7RUFFUSx3QkFBd0IsT0FBYTtHQUMzQyxLQUFLLE1BQU0sS0FBSyxhQUFhLGlCQUFpQixLQUFLLGFBQWEsS0FBSyxDQUFDO0VBQ3hFO0VBRVEsY0FBYyxVQUFXO0dBQy9CLElBQUksQ0FBQyxLQUFLLEtBQ1IsT0FBTztHQUVULE9BQU8sS0FBSyxhQUFhLFlBQVksVUFBVSxLQUFLLEdBQUcsSUFBSTtFQUM3RDtFQUVRLGNBQWMsVUFBVztHQUMvQixJQUFJLENBQUMsS0FBSyxLQUNSLE9BQU87R0FFVCxPQUFPLEtBQUssYUFBYSxZQUFZLFVBQVUsS0FBSyxHQUFHLElBQUk7RUFDN0Q7O0VBR1EseUJBQXNCO0dBQzVCLE1BQU0sWUFBWSxLQUFLLGFBQWEsZ0JBQ2xDLEtBQUssYUFDTCxLQUFLLGFBQWEsUUFBUSxLQUFLLFdBQVcsSUFBSSxFQUFFO0dBRWxELE9BQU8sS0FBSyxjQUFjLFNBQVM7RUFDckM7O0VBR1EscUJBQWtCO0dBQ3hCLElBQUksWUFBWSxLQUFLLGFBQWEsa0JBQWtCLEtBQUssYUFBYSxLQUFLLFFBQVEsSUFBSSxDQUFDO0dBQ3hGLFlBQVksS0FBSyxhQUFhLFdBQzVCLEtBQUssYUFBYSxRQUFRLFNBQVMsR0FDbkMsS0FBSyxhQUFhLFNBQVMsU0FBUyxHQUNwQyxDQUFDO0dBRUgsT0FBTyxLQUFLLGNBQWMsU0FBUztFQUNyQzs7RUFHUSx3QkFBcUI7R0FDM0IsTUFBTSxXQUFXLEtBQUssYUFBYSxXQUNqQyxLQUFLLGFBQWEsUUFBUSxLQUFLLFdBQVcsSUFBSSxHQUM5QyxJQUNBLEVBQUU7R0FFSixPQUFPLEtBQUssY0FBYyxRQUFRO0VBQ3BDOztFQUdRLG9CQUFpQjtHQUN2QixNQUFNLFdBQVcsS0FBSyxhQUFhLFdBQ2pDLEtBQUssYUFBYSxRQUFRLEtBQUssV0FBVyxLQUFLLEtBQUssUUFBUSxJQUFJLElBQ2hFLEdBQ0EsQ0FBQztHQUVILE9BQU8sS0FBSyxjQUFjLFFBQVE7RUFDcEM7O0VBR1EsNkJBQTBCO0dBQ2hDLE1BQU0sV0FBVyxLQUFLLGFBQWEsV0FBVyxLQUFLLE9BQU8sR0FBRyxLQUFLLEdBQUcsSUFBSSxFQUFFO0dBQzNFLE9BQU8sS0FBSyxjQUFjLFFBQVE7RUFDcEM7O0VBR1EseUJBQXNCO0dBQzVCLE1BQU0sUUFBUSxLQUFLLFFBQVEsS0FBSyxrQkFBa0IsS0FBSztHQUN2RCxNQUFNLGdCQUFnQixNQUFNLE1BQU0sU0FBUztHQUMzQyxNQUFNLFdBQVcsY0FBYyxjQUFjLFNBQVM7R0FDdEQsTUFBTSxXQUFXLEtBQUssYUFBYSxXQUFXLFdBQVcsR0FBRyxHQUFHLENBQUM7R0FDaEUsT0FBTyxLQUFLLGNBQWMsUUFBUTtFQUNwQztFQUVRLGlCQUFpQixhQUF3QjtHQUMvQyxJQUFJLGFBQWEsY0FBYyxvQkFDN0IsS0FBSyxhQUFZO0VBRXJCO0VBRVEsZUFBWTtHQUNsQixNQUFNLEtBQUssS0FBSyxPQUFPLFFBQVEsTUFBTSxFQUFFLGFBQWEsQ0FBQyxLQUFLLENBQUEsQ0FBRSxFQUFFLFNBQzNELFFBQVMsSUFBSSxXQUFXLEVBQUc7R0FFOUIsTUFBTSxpQkFBaUIsS0FBSyxtQkFBa0I7R0FDOUMsSUFBSSxnQkFDRixlQUFlLFdBQVc7RUFFOUI7O0VBR1EscUJBQWtCO0dBQ3hCLElBQUksS0FBSyxrQkFBa0IsT0FBTztJQUNoQyxNQUFNLG9CQUNKLEtBQUssT0FBTyxNQUFNLE1BQU0sRUFBRSxRQUFRLHlDQUFrQixDQUFDLEtBQ3JELEtBQUssT0FBTyxNQUFNLE1BQU0sRUFBRSxRQUFRLHVDQUFpQixDQUFDO0lBQ3RELE9BQU8scUJBQXFCLENBQUMsa0JBQWtCLFdBQzNDLG9CQUNBLEtBQUssc0JBQXFCO0dBQ2hDLE9BQU87SUFDTCxNQUFNLG9CQUNKLEtBQUssWUFBWSxjQUEwQyx5Q0FBa0IsS0FDN0UsS0FBSyxZQUFZLGNBQTBDLHVDQUFpQjtJQUM5RSxPQUFPLHFCQUFxQixDQUFDLGtCQUFrQixXQUMzQyxvQkFDQSxLQUFLLFdBQVksY0FDZixnQkFBZ0IsS0FBSyxjQUFhLGlCQUFrQjtHQUU1RDtFQUNGOzs7Ozs7OztFQVNRLHdCQUFxQjtHQUMzQixNQUFNLFFBQVEsS0FBSztHQUNuQixNQUFNLGFBQWEsTUFBTSxRQUFRLE1BQU0sQ0FBQyxFQUFFLFFBQVE7R0FDbEQsSUFBSSxDQUFDLGNBQWMsV0FBVyxXQUFXLEdBQ3ZDLE9BQU87UUFDRjtJQUNMLE1BQU0sZUFBZSxXQUNsQixLQUFLLE1BQWMsS0FBSyxhQUFhLFVBQVUsRUFBRSxLQUFXLENBQUMsRUFDN0QsS0FBSSxFQUFHO0lBQ1YsT0FBTyxNQUFNLE1BQU0sTUFBTSxFQUFFLFFBQVEsVUFBVSxhQUFZLEdBQUksQ0FBQyxLQUFNO0dBQ3RFO0VBQ0Y7RUFFUSxxQkFBcUIsT0FBc0IsS0FBWTtHQUM3RCxJQUFJLDRCQUE0QixLQUFLLEdBQ25DLE1BQU0sZUFBYztHQUt0QixNQUFNLFFBQVEsS0FBSztHQUNuQixNQUFNLFFBQWdCLE1BQU0sV0FBVyxNQUFNLE1BQU0sTUFBTSxNQUFNO0dBQy9ELElBQUk7R0FDSixJQUFJLEtBQ0YsU0FBUyxLQUFLLDJCQUEyQixPQUFPLE9BQU8sT0FBa0MsR0FBRztRQUU1RixTQUFTLEtBQUssb0JBQW9CLE9BQU8sT0FBTyxLQUFLO0dBRXZELE1BQU0sWUFBWSxLQUFLLG1CQUFtQixXQUFXLEtBQUssWUFDdkQ7R0FDSCxJQUFJLFdBQVcsVUFBVTtJQUN2QixPQUFPLFdBQVc7SUFDbEIsUUFBUSxNQUFLO0lBQ2IsU0FBUyxXQUFXO0dBQ3RCO0VBQ0Y7RUFFUSwyQkFDTixLQUNBLE9BQ0EsT0FDQSxLQUFXO0dBRVgsTUFBTSxlQUNKLEtBQUssZ0JBQWdCLGVBQ2pCO0lBQUUsV0FBVztJQUFHLFFBQVE7R0FBWSxJQUNwQztJQUFFLFdBQVc7SUFBYyxRQUFRO0dBQUM7R0FDMUMsTUFBTSxvQkFDSixRQUFRLEtBQUsscUNBQXFDLG1CQUM5QyxLQUFLLHFDQUFxQyxtQkFDMUMsS0FBSyxxQ0FBcUM7R0FFaEQsUUFBUSxJQUFJLEtBQVo7SUFDRSxLQUFLLFdBQ0gsT0FBTyxLQUFLLGVBQWUsT0FBTyxPQUFPLElBQUksV0FBVyxDQUFDLGFBQWEsTUFBTTtJQUM5RSxLQUFLLGFBQ0gsT0FBTyxLQUFLLGVBQWUsT0FBTyxPQUFPLElBQUksV0FBVyxhQUFhLE1BQU07SUFDN0UsS0FBSyxhQUNILE9BQU8sS0FBSyxlQUFlLE9BQU8sT0FBTyxJQUFJLFdBQVcsQ0FBQyxhQUFhLFNBQVM7SUFDakYsS0FBSyxjQUNILE9BQU8sS0FBSyxlQUFlLE9BQU8sT0FBTyxJQUFJLFdBQVcsYUFBYSxTQUFTO0lBQ2hGLEtBQUssVUFDSCxJQUFJLEtBQUssZ0JBQWdCLGNBQWM7S0FFckMsTUFBTSxTQURzQixDQUFDLElBQUksV0FBVyxnQkFBZ0IsZ0JBQ3hCLENBQUMsSUFBSTtLQUN6QyxPQUFPLEtBQUssbUJBQW1CLE9BQU8sT0FBTyxLQUFLLE9BQU8sYUFBYSxNQUFNO0lBQzlFLE9BQU87S0FHTCxNQUFNLFNBRnFCLEtBQUssTUFBTSxDQUFDLElBQUksV0FBVyxxQkFBcUIsWUFDOUMsSUFBYSxLQUFLLGVBQWUsb0JBQW9CLElBQzlDLENBQUMsSUFBSTtLQUN6QyxPQUFPLEtBQUssbUJBQW1CLE9BQU8sT0FBTyxLQUFLLE9BQU8sYUFBYSxNQUFNO0lBQzlFO0lBRUYsS0FBSyxZQUNILElBQUksS0FBSyxnQkFBZ0IsY0FBYztLQUNyQyxNQUFNLGdCQUFnQixDQUFDLElBQUksYUFBYSxJQUFJLEtBQUssSUFBSSxDQUFDLElBQUksYUFBYTtLQUN2RSxNQUFNLGVBQWUsQ0FBQyxJQUFJLGFBQWEsSUFBSSxLQUFLLENBQUMsSUFBSSxZQUFZLElBQUksQ0FBQyxJQUFJO0tBQzFFLE1BQU0saUJBQW9CLEtBQUssYUFBYSxXQUFXLGNBQWMsZUFBZSxDQUFDO0tBQ3JGLE1BQU0sY0FBc0IsS0FBSyxhQUFhLFFBQzVDLEtBQUssYUFBYSxnQkFBZ0IsZ0JBQWdCLEVBQUUsQ0FBQztLQUV2RCxNQUFNLFFBQ0osS0FBSyxPQUFPLGNBQWMsQ0FBQyxJQUFJLFlBQWEsWUFBWSxJQUFJO0tBQzlELE9BQU8sS0FBSyxtQkFBbUIsT0FBTyxPQUFPLEtBQUssT0FBTyxDQUFDLGFBQWEsTUFBTTtJQUMvRSxPQUFPO0tBR0wsTUFBTSxRQUZxQixLQUFLLE1BQU0sQ0FBQyxJQUFJLFdBQVcscUJBQXFCLFlBQ2hELElBQWEsZUFBZSxvQkFDcEIsQ0FBQyxJQUFJO0tBQ3hDLE9BQU8sS0FBSyxtQkFBbUIsT0FBTyxPQUFPLEtBQUssT0FBTyxDQUFDLGFBQWEsTUFBTTtJQUMvRTtJQUVGLEtBQUssUUFDSCxPQUFPLEtBQUssY0FBYyxPQUFPLE9BQU8sS0FBSyxDQUFDO0lBRWhELEtBQUssT0FBTztLQUNWLE1BQU0sZ0JBQWdCLENBQUMsSUFBSSxhQUFhLElBQUksS0FBSyxJQUFJLENBQUMsSUFBSSxhQUFhO0tBQ3ZFLE1BQU0sZUFBZSxDQUFDLElBQUksYUFBYSxJQUFJLEtBQUssQ0FBQyxJQUFJLFlBQVksSUFBSSxDQUFDLElBQUk7S0FDMUUsTUFBTSxpQkFBb0IsS0FBSyxhQUFhLFdBQVcsY0FBYyxlQUFlLENBQUM7S0FDckYsT0FBTyxLQUFLLGFBQWEsT0FBTyxPQUFPLGNBQWM7SUFDdkQ7SUFDQSxTQUNFLE9BQU8sTUFBTTtHQUNqQjtFQUNGO0VBRVEsZ0JBQWdCLE1BQVk7R0FDbEMsT0FDRSxPQUFPLEtBQUsscUNBQXFDLGtCQUNqRCxPQUFPLEtBQUsscUNBQXFDO0VBRXJEO0VBRVEsZUFDTixPQUNBLE9BQ0EsTUFDQSxPQUFhO0dBRWIsTUFBTSxlQUFlLEtBQUssYUFBYSxVQUNyQyxLQUFLLGFBQWEsZ0JBQWdCLE1BQU0sS0FBSyxDQUFDO0dBRWhELElBQUksS0FBSyxnQkFBZ0IsWUFBWSxHQUNuQyxPQUFPLE1BQU07R0FFZixNQUFNLFdBQVcsTUFBTSxNQUFNLE1BQU0sS0FBSyxtQkFBbUIsRUFBRSxLQUFNLE1BQU0sWUFBWTtHQUNyRixJQUFJLENBQUMsWUFBWSxTQUFTLFVBQ3hCLE9BQU8sS0FBSyxlQUFlLE9BQU8sT0FBTyxLQUFLLGFBQWEsWUFBWSxZQUFZLEdBQUksS0FBSztHQUU5RixPQUFPO0VBQ1Q7RUFFUSxtQkFDTixPQUNBLE9BQ0EsS0FDQSxPQUNBLGlCQUF1QjtHQUV2QixNQUFNLGVBQWUsS0FBSyxhQUFhLFVBQ3JDLEtBQUssYUFBYSxnQkFBZ0IsSUFBSSxXQUFXLEtBQUssQ0FBQztHQUV6RCxJQUFJLEtBQUssZ0JBQWdCLFlBQVksR0FDbkMsT0FBTyxNQUFNO0dBRWYsTUFBTSxXQUFXLE1BQU0sTUFBTSxNQUFNLEtBQUssbUJBQW1CLEVBQUUsS0FBTSxNQUFNLFlBQVk7R0FDckYsSUFBSSxDQUFDLFlBQVksU0FBUyxVQUN4QixPQUFPLEtBQUssbUJBQW1CLE9BQU8sT0FBTyxLQUFLLFFBQVEsaUJBQWlCLGVBQWU7R0FFNUYsT0FBTztFQUNUO0VBRVEsY0FDTixPQUNBLE9BQ0EsS0FDQSxNQUFZO0dBRVosTUFBTSxlQUFlLEtBQUssYUFBYSxVQUNyQyxLQUFLLGFBQWEsV0FBVyxDQUFDLElBQUksV0FBVyxDQUFDLElBQUksWUFBWSxJQUFJLENBQUM7R0FFckUsSUFBSSxLQUFLLGdCQUFnQixZQUFZLEdBQ25DLE9BQU8sTUFBTTtHQUVmLE1BQU0sV0FBVyxNQUFNLE1BQU0sTUFBTSxLQUFLLG1CQUFtQixFQUFFLEtBQU0sTUFBTSxZQUFZO0dBQ3JGLElBQUksQ0FBQyxZQUFZLFNBQVMsVUFDeEIsT0FBTyxLQUFLLGNBQWMsT0FBTyxPQUFPLEtBQUssT0FBTyxDQUFDO0dBRXZELE9BQU87RUFDVDtFQUVRLGFBQ04sT0FDQSxPQUNBLGdCQUFpQjtHQUVqQixNQUFNLGVBQWUsS0FBSyxhQUFhLFVBQ3JDLEtBQUssYUFBYSxnQkFBZ0IsZ0JBQWdCLEVBQUUsQ0FBQztHQUV2RCxJQUFJLEtBQUssZ0JBQWdCLFlBQVksR0FDbkMsT0FBTyxNQUFNO0dBRWYsTUFBTSxXQUFXLE1BQU0sTUFBTSxNQUFNLEtBQUssbUJBQW1CLEVBQUUsS0FBTSxNQUFNLFlBQVk7R0FDckYsSUFBSSxDQUFDLFlBQVksU0FBUyxVQUN4QixPQUFPLEtBQUssYUFBYSxPQUFPLE9BQU8sS0FBSyxhQUFhLFlBQVksWUFBWSxDQUFFO0dBRXJGLE9BQU87RUFDVDs7Ozs7OztFQVFRLG9CQUNOLEtBQ0EsT0FDQSxPQUFtQztHQUVuQyxNQUFNLEVBQ0oseUJBQ0EsbUJBQ0EsNkJBQ0EsbUJBRUEsS0FBSywwQ0FBMEMsT0FBTyxLQUFLLGtCQUFrQixNQUFNO0dBRXJGLFFBQVEsSUFBSSxLQUFaO0lBQ0UsS0FBSyxXQUNILE9BQU8sS0FBSyxVQUFVLE9BQU8sT0FBTyxDQUFDLGNBQWM7SUFDckQsS0FBSyxhQUNILE9BQU8sS0FBSyxVQUFVLE9BQU8sT0FBTyxjQUFjO0lBQ3BELEtBQUssYUFDSCxPQUFPLEtBQUssVUFBVSxPQUFPLE9BQU8sRUFBRTtJQUN4QyxLQUFLLGNBQ0gsT0FBTyxLQUFLLFVBQVUsT0FBTyxPQUFPLENBQUM7SUFDdkMsS0FBSyxRQUNILE9BQU8sS0FBSyxXQUFXLE9BQU8saUJBQWlCO0lBQ2pELEtBQUssVUFDSCxPQUFPLEtBQUssbUJBQ1YsT0FDQSx5QkFDQSxtQkFDQSxjQUFjO0lBRWxCLEtBQUssWUFDSCxPQUFPLEtBQUssa0JBQWtCLE9BQU8sT0FBTyw2QkFBNkIsY0FBYztJQUN6RixLQUFLLE9BQ0gsT0FBTyxLQUFLLFVBQVUsT0FBTyw4QkFBOEIsQ0FBQztJQUM5RCxTQUNFLE9BQU8sTUFBTTtHQUNqQjtFQUNGOzs7Ozs7RUFPUSwwQ0FDTixPQUNBLFlBQW1CO0dBRW5CLE1BQU0sa0JBQWtCLGFBQWEsaUJBQWlCO0dBQ3RELE1BQU0sU0FBaUIsS0FBSyxNQUFNLFFBQVEsZUFBZSxJQUFJO0dBQzdELE1BQU0sY0FBc0IsV0FBVyxJQUFJLFFBQVEsUUFBUTtHQUMzRCxPQUFPO0lBQ0wsZ0JBQWdCLGFBQWEsZ0JBQWdCO0lBQzdDLHlCQUF5QjtJQUN6QixtQkFBbUIsUUFBUTtJQUMzQiw2QkFBNkIsV0FBVyxJQUFJLGtCQUFrQixrQkFBa0I7O0VBRXBGOzs7OztFQU1RLFVBQ04sTUFDQSxPQUNBLE9BQWE7R0FFYixJQUFJLFlBQVksUUFBUTtHQUN4QixPQUFPLFlBQVksS0FBSyxVQUFVLEtBQUssWUFBWSxVQUNqRCxhQUFhO0dBRWYsT0FBTyxLQUFLLGNBQWMsS0FBSztFQUNqQzs7RUFHUSxXQUNOLE1BQ0EscUJBQTJCO0dBRTNCLE9BQU8sQ0FBQyxLQUFLLHFCQUFxQixXQUM5QixLQUFLLHVCQUNMLEtBQUssVUFBVSxNQUFNLHFCQUFxQixDQUFDO0VBQ2pEOztFQUdRLFVBQ04sTUFDQSxvQkFBMEI7R0FFMUIsT0FBTyxDQUFDLEtBQUssb0JBQW9CLFdBQzdCLEtBQUssc0JBQ0wsS0FBSyxVQUFVLE1BQU0sb0JBQW9CLEVBQUU7RUFDakQ7O0VBR1EsbUJBQ04sTUFDQSxPQUNBLFFBQ0EsZ0JBQXNCO0dBRXRCLE1BQU0sWUFBYSxRQUFRLGlCQUFrQjtHQUM3QyxPQUFPLENBQUMsS0FBSyxXQUFXLFdBQ3BCLEtBQUssYUFDTCxLQUFLLFVBQVUsTUFBTSxXQUFXLGNBQWM7RUFDcEQ7O0VBR1Esa0JBQ04sTUFDQSxPQUNBLFFBQ0EsZ0JBQXNCO0dBRXRCLE1BQU0sWUFBWSxRQUFRLEtBQUssT0FBTyxTQUFTLFFBQVEsS0FBSyxjQUFjLElBQUk7R0FDOUUsT0FBTyxDQUFDLEtBQUssV0FBVyxXQUNwQixLQUFLLGFBQ0wsS0FBSyxVQUFVLE1BQU0sV0FBVyxDQUFDLGNBQWM7RUFDckQ7RUFFUSxxQ0FBcUMsaUJBQWlCLE9BQUs7R0FDakUsS0FBSyxtQkFBbUIsY0FBYztHQUN0QyxLQUFLLGlCQUFnQjtFQUN2QjtFQUVRLG1CQUFtQixpQkFBaUIsT0FBSztHQUMvQyxJQUFJLEtBQUssa0JBQ1AsS0FBSyxjQUFjO0dBRXJCLEtBQUssZUFDRixLQUFLLFdBQVksS0FBSyxVQUFrQixHQUFHLEVBQUUsSUFBSyxLQUFLLGNBQ3hELEtBQUssYUFBYSxNQUFLO0dBQ3pCLEtBQUssZUFBYztHQUNuQixLQUFLLGVBQWUsS0FBQTtHQUNwQixLQUFLLE1BQUs7R0FDVixLQUFLLG9CQUFvQixLQUFLLGdCQUFnQixLQUFLO0dBRW5ELElBQUksZ0JBQ0YsS0FBSyxzQkFBcUI7RUFFOUI7O0VBR1EsaUJBQWM7R0FDcEIsTUFBTSxzQkFBc0IsS0FBSyxRQUM3QixLQUFLLGFBQWEsa0JBQWtCLEtBQUssYUFBYSxDQUFDLElBQ3ZELEtBQUE7R0FDSixPQUFPLElBQUk7O1VBRUwsS0FBSyxVQUNMLGNBQ00sS0FBSyxvQkFBb0IsRUFBRSxHQUNqQyxrQkFBa0IsS0FBSyxVQUFVLFVBQ2pDLEtBQUssdUJBQXNCLENBQUUsRUFBQTs7WUFHM0IsS0FBSyx1QkFBdUIsS0FBSyxXQUFXLEVBQUE7WUFDNUMsS0FBSyxRQUFRLEtBQUssdUJBQXVCLG1CQUFvQixJQUFJLFFBQUE7O2NBRS9ELEtBQUssMkJBQTJCLEtBQUssYUFBYSxtQkFBb0IsRUFBQTs7O1VBRzFFLEtBQUssVUFDTCxlQUNNLEtBQUssb0JBQW9CLENBQUMsR0FDaEMsY0FBYyxLQUFLLFVBQVUsVUFDN0IsS0FBSyxtQkFBa0IsQ0FBRSxFQUFBOzs7O1lBS3ZCLEtBQUssZ0JBQWdCLGVBQ25CLElBQUk7a0JBQ0EsS0FBSyxnQkFBZ0IsS0FBSyxRQUFRLEtBQUssWUFBWSxFQUFBO2tCQUNuRCxLQUFLLFFBQ0gsS0FBSyxnQkFBZ0IsS0FBSyxpQkFBaUIsS0FBSyx1QkFBdUIsSUFBSSxJQUMzRSxRQUFBO2tCQUVOLElBQUk7a0JBQ0EsS0FBSyx3QkFBd0IsS0FBSyxRQUFRLEtBQUssWUFBWSxFQUFBO2tCQUMzRCxLQUFLLFFBQ0gsS0FBSyx3QkFDSCxLQUFLLGlCQUNMLEtBQUssdUJBQ0wsbUJBQW1CLElBRXJCLFFBQUE7Ozs7O0VBS2xCOztFQUdRLHVCQUF1QixHQUFJO0dBQ2pDLE1BQU0sYUFBYSxHQUNqQixLQUFLLFlBQVksS0FBSyxhQUFhLFNBQVMsQ0FBQyxJQUFJLEdBQ25ELEdBQUksS0FBSyxhQUFhLFFBQVEsQ0FBQztHQUMvQixPQUFPLElBQUk7Ozs7c0JBSU8sdUJBQXVCLEtBQUssVUFBVSxTQUFRLEdBQUksV0FBVTt1QkFDNUQ7SUFDWixLQUFLLGNBQWM7SUFDbkIsS0FBSyxvQkFBb0I7SUFDekIsS0FBSyxzQkFBcUI7R0FDNUIsRUFBQTs7VUFFRSxXQUFBOzs7O0VBSVI7O0VBR1EsMkJBQTJCLEdBQUcsT0FBVTtHQUM5QyxJQUFJLGFBQWE7R0FDakIsS0FBSyxNQUFNLEtBQUssT0FDZCxJQUFJLEdBQ0YsY0FBYyxHQUNaLEtBQUssWUFBWSxLQUFLLGFBQWEsU0FBUyxDQUFDLElBQUksR0FDbkQsR0FBSSxLQUFLLGFBQWEsUUFBUSxDQUFDLEVBQUM7R0FHcEMsT0FBTztFQUNUOztFQUdRLGdCQUNOLE9BQ0EsYUFDQSxrQkFBMkIsT0FBSztHQUVoQyxNQUFNLHFDQUNKLEtBQUssUUFDRCxDQUFDLEdBQUcsS0FBSyxRQUFRLEdBQUcsS0FBSyxlQUFlLElBQ3hDLGtCQUNFLEtBQUssa0JBQ0wsS0FBSyxRQUNYLEtBQUk7R0FDTixNQUFNLGtDQUNKLGtCQUFrQixLQUFLLGtCQUFrQixLQUFLLFFBQzlDLEtBQUk7R0FDTixPQUFPLElBQUk7OztxQkFHTSxVQUNYLEtBQUssaUJBQWlCLE1BQU0sYUFBNEIsRUFBQTt5QkFDekMsTUFBc0IsS0FBSyxtQkFBbUIsQ0FBQyxFQUFBOzs7O2NBSTFELEtBQUssY0FBYyxJQUFJLHNEQUFzRCxRQUFBO2NBQzdFLEtBQUssVUFBVSxLQUNkLFNBQWtCLFVBQWtCLElBQUk7O29CQUVuQyxLQUFLLFdBQ0gsSUFBSTs7bUNBRVMsUUFBQTt5Q0FDSztJQUVaLE1BQU0sT0FBaUIsK0JBQStCLFFBQ25ELFFBQWdCLElBQUksa0JBQWtCLFFBQVEsS0FBSyxDQUFDO0lBRXZELEtBQUsscUJBQXFCLElBQUk7R0FDaEMsRUFBQTs7MEJBR0osSUFBSTsrREFDcUMsUUFBUSxLQUFJO21EQUN4QixRQUFRLE9BQU07OztlQUdsRCxFQUFBOzs7O1lBS0gsTUFBTSxLQUFLLE1BQWdCLGFBQW9CO0lBQy9DLE1BQU0saUJBQXlCLGVBQWUsS0FBSztJQUNuRCxJQUFJLGFBQWEsS0FBSyxnQkFDcEIsT0FBTyxJQUFJOztvQkFFTCxLQUFLLGNBQ0gsSUFBSTs7NEJBRUUsS0FBSyxXQUNILElBQUk7OzJDQUVTLFlBQVksR0FBQTtpREFDUDtLQUNaLE1BQU0sT0FBaUIsa0NBQWtDLFFBQ3RELFFBQWdCLElBQUksY0FBYyxZQUFZLEVBQUU7S0FFbkQsS0FBSyxxQkFBcUIsSUFBSTtJQUNoQyxFQUFBOztrQ0FHSixJQUFJOztxQ0FFRyxHQUFHLHVCQUF1QixLQUFLLFVBQVUsU0FBUSxHQUFJLFlBQVksS0FBSTs7MkRBRS9DLFlBQVksR0FBRTs7OzBCQUluRCxRQUFBO29CQUNGLENBQUMsR0FBRyxNQUFNLGNBQWMsRUFBRSxLQUFJLENBQUUsRUFBRSxVQUM1QixJQUFJLDRDQUE0QyxFQUFBO29CQUV0RCxLQUFLLGdCQUFnQixJQUFJLEVBQUE7OztJQUlqQyxPQUFPLElBQUk7O2tCQUVMLEtBQUssY0FDSCxJQUFJOzswQkFFRSxLQUFLLFdBQ0gsSUFBSTs7eUNBRVMsWUFBWSxVQUFBOytDQUNQO0tBQ1osTUFBTSxPQUFpQixrQ0FBa0MsUUFDdEQsUUFBZ0IsSUFBSSxjQUFjLFlBQVksU0FBUztLQUUxRCxLQUFLLHFCQUFxQixJQUFJO0lBQ2hDLEVBQUE7O2dDQUdKLElBQUk7O21DQUVHLEdBQUcsdUJBQXVCLEtBQUssVUFBVSxTQUFRLEdBQUksWUFBWSxZQUFXOzt5REFFdEQsWUFBWSxVQUFTOzs7d0JBSTFELFFBQUE7a0JBQ0YsS0FBSyxnQkFBZ0IsSUFBSSxFQUFBOzs7R0FHakMsQ0FBQyxFQUFBOzs7O0VBSVQ7O0VBR1Esd0JBQ04sT0FDQSxhQUNBLHFCQUF1QjtHQUV2QixNQUFNLGFBQWEsS0FBSyxhQUFhLG1CQUNuQyx1QkFBdUIsS0FBSyxXQUFXO0dBRXpDLE1BQU0scUNBQ0osS0FBSyxRQUNELENBQUMsR0FBRyxLQUFLLFFBQVEsR0FBRyxLQUFLLGVBQWUsSUFDeEMsc0JBQ0UsS0FBSyxrQkFDTCxLQUFLLFFBQ1gsS0FBSTtHQUNOLE9BQU8sSUFBSTs7O3FCQUdNLFVBQ1gsS0FBSyxpQkFBaUIsTUFBTSxhQUE0QixFQUFBO3lCQUN6QyxNQUFzQixLQUFLLG1CQUFtQixDQUFDLEVBQUE7O1VBRTlELEtBQUssY0FDSCxJQUFJOzs7b0JBR0ksc0JBQ0UsVUFDQSxJQUFJLDZDQUFBO29CQUNOLFlBQVksS0FDWCxlQUF1QixJQUFJOzswQkFFdEIsS0FBSyxXQUNILElBQUk7O3lDQUVTLFdBQUE7K0NBQ0s7SUFDWixNQUFNLE9BQWlCLGtDQUFrQyxRQUN0RCxRQUFnQixJQUFJLGNBQWMsVUFBVTtJQUUvQyxLQUFLLHFCQUFxQixJQUFJO0dBQ2hDLEVBQUE7O2dDQUdKLElBQUk7O21DQUVHLEdBQUcsdUJBQXVCLEtBQUssVUFBVSxTQUFRLEdBQUksYUFBWTs7eURBRTNDLFdBQVU7OztxQkFHOUMsRUFBQTs7O2dCQUtULFFBQUE7O1lBRUEsTUFBTSxLQUFLLE1BQWdCLGFBQW9CO0lBQy9DLE1BQU0sVUFBVSxLQUFLLFVBQVU7SUFDL0IsTUFBTSxpQkFBaUIsS0FBSyxRQUFRLENBQUMsR0FBRyxNQUFNLEdBQUcsS0FBSyxnQkFBZ0IsU0FBUyxJQUFJO0lBQ25GLE9BQU8sSUFBSTs7a0JBRUwsc0JBQ0UsVUFDQSxJQUFJOzswQkFFRSxLQUFLLFdBQ0gsSUFBSTs7eUNBRVMsUUFBQTsrQ0FDTSxLQUFLLHFCQUFxQixjQUFjLEVBQUE7O2tDQUVyRCxRQUFRLE9BQUE7O2dDQUdkLElBQUk7cUVBQ3FDLFFBQVEsS0FBSTt5REFDeEIsUUFBUSxPQUFNOzs7O2tCQUlyRCxXQUFXLGFBQ1QsSUFBSSwrQ0FDSixRQUFBO2tCQUNGLEtBQUssZ0JBQWdCLElBQUksRUFBQTs7O0dBR2pDLENBQUMsRUFBQTs7OztFQUlUOztFQUdRLGdCQUFnQixNQUFjO0dBQ3BDLE9BQU8sS0FBSyxLQUFLLFFBQWU7SUFDOUIsT0FBTyxJQUFJOzt1QkFFTSxJQUFJLE1BQUs7O3FCQUVYLElBQUksTUFBQTs2QkFDSSxLQUFLLFlBQVksSUFBSSxTQUFTLEVBQUE7MEJBQ2pDLFFBQXVCLEtBQUsscUJBQXFCLEtBQUssR0FBRyxFQUFBOzs7OztHQUsvRSxDQUFDO0VBQ0g7O0VBR1EsbUJBQWdCO0dBQ3RCLE9BQU8sSUFBSTs7VUFFTCxLQUFLLFVBQ0wsY0FDTSxLQUFLLG1CQUFtQixFQUFFLEdBQ2hDLGlCQUFpQixLQUFLLFVBQVUsVUFDaEMsS0FBSyxzQkFBcUIsQ0FBRSxFQUFBO29EQUVjLEtBQUsseUJBQXdCLEVBQUU7VUFDekUsS0FBSyxVQUNMLGVBQ00sS0FBSyxtQkFBbUIsQ0FBQyxHQUMvQixhQUFhLEtBQUssVUFBVSxVQUM1QixLQUFLLGtCQUFpQixDQUFFLEVBQUE7Ozs7WUFLdEIsS0FBSyxrQkFBa0IsS0FBSyxTQUFTLEtBQUssV0FBWSxFQUFBO1lBQ3RELEtBQUssUUFBUSxLQUFLLGtCQUFrQixLQUFLLFNBQVMsS0FBSyxjQUFlLENBQUMsSUFBSSxRQUFBOzs7O0VBSXJGOztFQUdRLDJCQUF3QjtHQUM5QixPQUFPLElBQUk7Ozs7cUJBSU0sR0FBRywwQkFBMEIsS0FBSyxVQUFVLFNBQVEsR0FBSSxLQUFLLGNBQUE7dUJBQzNELEtBQUsscUNBQXFDLElBQUksRUFBQTs7VUFFM0QsS0FBSyxZQUFXLEdBQUksS0FBSyxRQUFRLE1BQU0sS0FBSyxjQUFlLE1BQU0sUUFBQTs7OzREQUdmLEtBQUssWUFBVztFQUMxRTs7RUFHUSxrQkFBa0IsUUFBdUIsTUFBWTtHQUMzRCxPQUFPLElBQUk7Ozt5QkFHVSxNQUFzQixLQUFLLG1CQUFtQixDQUFDLEVBQUE7O1VBRTlELEtBQUssUUFDSCxJQUFJOztzRUFFc0QsZUFBYyxHQUFJLEtBQUk7O3dCQUdoRixRQUFBOztZQUVBLE9BQU8sS0FDTixRQUFxQixJQUFJOztrQkFFcEIsSUFBSSxLQUFLLFVBQW9CO0lBQzdCLE9BQU8sSUFBSTs7O2tDQUdLLEtBQUksR0FBSSxNQUFNLE1BQUs7dUNBQ2QsS0FBSyxrQkFBa0IsTUFBTSxZQUFZLElBQUksRUFBQTtvQ0FDaEQsUUFBdUIsS0FBSyxxQkFBcUIsR0FBRyxFQUFBOzs7OztHQUt4RSxDQUFDLEVBQUE7O2FBRUosRUFBQTs7OztFQUtYOztFQUdRLGtCQUFrQixPQUFlLE1BQVk7R0FDbkQsS0FBSyxlQUFlO0dBQ3BCLEtBQUssb0JBQW9CO0dBQ3pCLEtBQUssTUFDSCxLQUFLLGFBQWEsV0FDaEIsTUFDQSxLQUFLLGNBQ0wsS0FBSyxhQUFhLFFBQVEsS0FBSyxXQUFXLENBQUMsQ0FDNUM7R0FFSCxLQUFLLHNCQUFxQjtHQUMxQixLQUFLLGlCQUFnQjtFQUN2Qjs7RUFHUSxrQkFBZTtHQUNyQixPQUFPLElBQUk7O1VBRUwsS0FBSyxVQUNMLGNBQ00sS0FBSyx3QkFBd0IsQ0FBQyxjQUFjLEdBQ2xELHNCQUFzQixjQUFjLEVBQUUsS0FBSyxVQUFVLFVBQ3JELEtBQUssMkJBQTBCLENBQUUsRUFBQTtvREFFUyxLQUFLLHdCQUF1QixFQUFFO1VBQ3hFLEtBQUssVUFDTCxlQUNNLEtBQUssd0JBQXdCLGNBQWMsR0FDakQsa0JBQWtCLGNBQWMsRUFBRSxLQUFLLFVBQVUsVUFDakQsS0FBSyx1QkFBc0IsQ0FBRSxFQUFBOzs7O1lBSzNCLEtBQUssaUJBQWlCLEtBQUssTUFBTSxFQUFBO1lBQ2pDLEtBQUssUUFBUSxLQUFLLGlCQUFpQixLQUFLLGlCQUFpQixJQUFJLElBQUksUUFBQTs7OztFQUkzRTs7RUFHUSxVQUNOLFdBQ0EsT0FDQSxXQUNBLFVBQWlCO0dBRWpCLE9BQU8sSUFBSTs7aUNBRWtCLFVBQVM7bUJBQ3ZCLFVBQUE7ZUFDSixNQUFBO2tCQUNHLFNBQUE7bUNBQ2lCLGNBQWMsU0FBUyxhQUFhLE9BQU07O0VBRTNFOztFQUdRLDBCQUF1QjtHQUM3QixNQUFNLFlBQW9CLEtBQUssT0FBTyxLQUFJLEVBQUc7R0FDN0MsTUFBTSxpQkFBMkIsS0FBSyxRQUFRLEtBQUssa0JBQWtCLEtBQUssUUFBUSxLQUFJO0dBRXRGLE1BQU0sWUFBWSxHQUFHLFVBQVMsS0FETCxjQUFjLGNBQWMsU0FBUztHQUU5RCxPQUFPLElBQUk7Ozs7O3NCQUtPLDBCQUEwQixLQUFLLFVBQVUsU0FBUSxHQUFJLFVBQVM7dUJBQzdELEtBQUsscUNBQXFDLElBQUksRUFBQTs7VUFFM0QsVUFBQTs7OzREQUdrRCxVQUFTOztFQUVuRTs7RUFHUSxpQkFBaUIsT0FBbUIsYUFBYSxPQUFLO0dBQzVELE9BQU8sSUFBSTs7dUJBRVEsTUFBc0IsS0FBSyxtQkFBbUIsQ0FBQyxFQUFBOzs7VUFHNUQsTUFBTSxLQUNMLFFBQ0MsSUFBSTtnQkFDQSxJQUFJLEtBQUssU0FBZ0I7SUFDekIsT0FBTyxJQUFJOzs7K0JBR0ksT0FBTyxJQUFJLEVBQUE7a0NBQ1IsUUFBdUIsS0FBSyxxQkFBcUIsR0FBRyxFQUFBO3FDQUNqRCxLQUFLLGlCQUFpQixNQUFNLFVBQVUsRUFBQTs7Ozs7R0FLN0QsQ0FBQyxFQUFBO2tCQUNHLEVBQUE7OztFQUloQjs7RUFHUSxpQkFBaUIsTUFBYyxXQUFrQjtHQUN2RCxLQUFLLGNBQWMsWUFBWSxPQUFPLElBQUk7R0FDMUMsS0FBSyxvQkFBb0I7R0FDekIsS0FBSyxrQkFDSCxLQUFLLGFBQWEsV0FDaEIsS0FBSyxhQUNMLEtBQUssYUFBYSxTQUFTLEtBQUssV0FBVyxHQUMzQyxLQUFLLGFBQWEsUUFBUSxLQUFLLFdBQVcsQ0FBQyxDQUM1QztHQUVILEtBQUssc0JBQXFCO0VBQzVCO0VBRVEsV0FBUTtHQUNkLElBQUksWUFBWSxLQUFLLG1CQUduQixPQUFPLElBQUksR0FBRztHQUVoQixRQUFRLEtBQUssZUFBYjtJQUNFLEtBQUssUUFDSCxPQUFPLEtBQUssZ0JBQWU7SUFDN0IsS0FBSyxTQUNILE9BQU8sS0FBSyxpQkFBZ0I7SUFFOUIsU0FDRSxPQUFPLEtBQUssZUFBYztHQUM5QjtFQUNGO0VBRVEsbUJBQW1CLE9BQXFCO0dBQzlDLE1BQU0sUUFBUSxNQUFNO0dBQ3BCLElBQUksTUFBTSxrQkFBa0IsUUFBUTtJQUNsQyxNQUFNLFVBQVUsT0FBTywwQkFBMEI7SUFDakQsSUFBSSxLQUFLLGtCQUNQLEtBQUssY0FBYztJQUVyQixLQUFLLGdCQUFnQixLQUFLO0dBQzVCLE9BQU8sSUFBSSxNQUFNLGtCQUFrQixRQUNqQyxLQUFLLFVBQVUsT0FBTyxPQUFPLFlBQVk7RUFFN0M7RUFFUSx3QkFBcUI7R0FDM0IsS0FBSyxVQUFVLE9BQU8sSUFBSSxZQUFZO0dBQ3RDLEtBQUssWUFDRCxpQkFBaUIsT0FBTyxHQUN4QixTQUFTLE1BQU0sRUFBRSxVQUFVLE9BQU8sMEJBQTBCLENBQUM7RUFDbkU7RUFFbUIsU0FBTTtHQUN2QixPQUFPLElBQUksc0NBQXNDLEtBQUssU0FBUSxFQUFFO0VBQ2xFIn0=