angular-slickgrid 5.6.3 → 6.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (234) hide show
  1. package/README.md +26 -27
  2. package/app/modules/angular-slickgrid/components/angular-slickgrid.component.d.ts +198 -198
  3. package/app/modules/angular-slickgrid/constants.d.ts +30 -30
  4. package/app/modules/angular-slickgrid/extensions/index.d.ts +1 -1
  5. package/app/modules/angular-slickgrid/extensions/slickRowDetailView.d.ts +87 -88
  6. package/app/modules/angular-slickgrid/global-grid-options.d.ts +3 -3
  7. package/app/modules/angular-slickgrid/index.d.ts +7 -7
  8. package/app/modules/angular-slickgrid/models/angularComponentOutput.interface.d.ts +5 -5
  9. package/app/modules/angular-slickgrid/models/angularGridInstance.interface.d.ts +36 -36
  10. package/app/modules/angular-slickgrid/models/externalTestingDependencies.interface.d.ts +20 -20
  11. package/app/modules/angular-slickgrid/models/gridOption.interface.d.ts +9 -9
  12. package/app/modules/angular-slickgrid/models/index.d.ts +6 -6
  13. package/app/modules/angular-slickgrid/models/rowDetailView.interface.d.ts +25 -25
  14. package/app/modules/angular-slickgrid/models/slickGrid.interface.d.ts +6 -6
  15. package/app/modules/angular-slickgrid/modules/angular-slickgrid.module.d.ts +12 -12
  16. package/app/modules/angular-slickgrid/services/angularUtil.service.d.ts +11 -13
  17. package/app/modules/angular-slickgrid/services/container.service.d.ts +10 -11
  18. package/app/modules/angular-slickgrid/services/index.d.ts +4 -4
  19. package/app/modules/angular-slickgrid/services/translater.service.d.ts +30 -30
  20. package/app/modules/angular-slickgrid/services/utilities.d.ts +8 -8
  21. package/app/modules/angular-slickgrid/slickgrid-config.d.ts +5 -5
  22. package/{esm2020 → esm2022}/angular-slickgrid.mjs +4 -4
  23. package/esm2022/app/modules/angular-slickgrid/components/angular-slickgrid.component.mjs +1245 -0
  24. package/esm2022/app/modules/angular-slickgrid/constants.mjs +96 -0
  25. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/extensions/index.mjs +1 -1
  26. package/esm2022/app/modules/angular-slickgrid/extensions/slickRowDetailView.mjs +307 -0
  27. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/global-grid-options.mjs +246 -246
  28. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/index.mjs +7 -7
  29. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/angularComponentOutput.interface.mjs +1 -1
  30. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/angularGridInstance.interface.mjs +1 -1
  31. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/externalTestingDependencies.interface.mjs +1 -1
  32. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/gridOption.interface.mjs +1 -1
  33. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/index.mjs +6 -6
  34. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/rowDetailView.interface.mjs +1 -1
  35. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/models/slickGrid.interface.mjs +1 -1
  36. package/esm2022/app/modules/angular-slickgrid/modules/angular-slickgrid.module.mjs +42 -0
  37. package/esm2022/app/modules/angular-slickgrid/services/angularUtil.service.mjs +40 -0
  38. package/esm2022/app/modules/angular-slickgrid/services/container.service.mjs +28 -0
  39. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/services/index.mjs +4 -4
  40. package/esm2022/app/modules/angular-slickgrid/services/translater.service.mjs +45 -0
  41. package/{esm2020 → esm2022}/app/modules/angular-slickgrid/services/utilities.mjs +18 -18
  42. package/esm2022/app/modules/angular-slickgrid/slickgrid-config.mjs +8 -0
  43. package/{esm2020 → esm2022}/public_api.mjs +1 -1
  44. package/{fesm2020 → fesm2022}/angular-slickgrid.mjs +2006 -1960
  45. package/fesm2022/angular-slickgrid.mjs.map +1 -0
  46. package/index.d.ts +5 -5
  47. package/package.json +16 -24
  48. package/public_api.d.ts +1 -1
  49. package/esm2020/app/modules/angular-slickgrid/components/angular-slickgrid.component.mjs +0 -1200
  50. package/esm2020/app/modules/angular-slickgrid/constants.mjs +0 -95
  51. package/esm2020/app/modules/angular-slickgrid/extensions/slickRowDetailView.mjs +0 -298
  52. package/esm2020/app/modules/angular-slickgrid/modules/angular-slickgrid.module.mjs +0 -41
  53. package/esm2020/app/modules/angular-slickgrid/services/angularUtil.service.mjs +0 -46
  54. package/esm2020/app/modules/angular-slickgrid/services/container.service.mjs +0 -29
  55. package/esm2020/app/modules/angular-slickgrid/services/translater.service.mjs +0 -43
  56. package/esm2020/app/modules/angular-slickgrid/slickgrid-config.mjs +0 -7
  57. package/fesm2015/angular-slickgrid.mjs +0 -2044
  58. package/fesm2015/angular-slickgrid.mjs.map +0 -1
  59. package/fesm2020/angular-slickgrid.mjs.map +0 -1
  60. package/test/cypress/node_modules/@4tw/cypress-drag-drop/README.md +0 -169
  61. package/test/cypress/node_modules/@cypress/request/README.md +0 -1038
  62. package/test/cypress/node_modules/@cypress/xvfb/README.md +0 -48
  63. package/test/cypress/node_modules/@cypress/xvfb/node_modules/debug/README.md +0 -437
  64. package/test/cypress/node_modules/@types/node/README.md +0 -16
  65. package/test/cypress/node_modules/@types/sinonjs__fake-timers/README.md +0 -16
  66. package/test/cypress/node_modules/@types/sizzle/README.md +0 -16
  67. package/test/cypress/node_modules/@types/yauzl/README.md +0 -16
  68. package/test/cypress/node_modules/@types/yauzl/node_modules/@types/node/README.md +0 -16
  69. package/test/cypress/node_modules/aggregate-error/README.md +0 -61
  70. package/test/cypress/node_modules/ansi-colors/README.md +0 -315
  71. package/test/cypress/node_modules/ansi-escapes/README.md +0 -245
  72. package/test/cypress/node_modules/ansi-regex/README.md +0 -78
  73. package/test/cypress/node_modules/ansi-styles/README.md +0 -152
  74. package/test/cypress/node_modules/arch/README.md +0 -71
  75. package/test/cypress/node_modules/asn1/README.md +0 -50
  76. package/test/cypress/node_modules/assert-plus/README.md +0 -162
  77. package/test/cypress/node_modules/astral-regex/README.md +0 -46
  78. package/test/cypress/node_modules/async/README.md +0 -60
  79. package/test/cypress/node_modules/asynckit/README.md +0 -233
  80. package/test/cypress/node_modules/at-least-node/README.md +0 -25
  81. package/test/cypress/node_modules/aws-sign2/README.md +0 -4
  82. package/test/cypress/node_modules/aws4/README.md +0 -183
  83. package/test/cypress/node_modules/balanced-match/README.md +0 -91
  84. package/test/cypress/node_modules/base64-js/README.md +0 -34
  85. package/test/cypress/node_modules/bcrypt-pbkdf/README.md +0 -45
  86. package/test/cypress/node_modules/blob-util/README.md +0 -623
  87. package/test/cypress/node_modules/bluebird/README.md +0 -57
  88. package/test/cypress/node_modules/brace-expansion/README.md +0 -129
  89. package/test/cypress/node_modules/buffer/README.md +0 -410
  90. package/test/cypress/node_modules/buffer-crc32/README.md +0 -47
  91. package/test/cypress/node_modules/cachedir/README.md +0 -27
  92. package/test/cypress/node_modules/caseless/README.md +0 -45
  93. package/test/cypress/node_modules/chalk/README.md +0 -293
  94. package/test/cypress/node_modules/chalk/node_modules/supports-color/README.md +0 -76
  95. package/test/cypress/node_modules/check-more-types/README.md +0 -952
  96. package/test/cypress/node_modules/ci-info/README.md +0 -114
  97. package/test/cypress/node_modules/clean-stack/README.md +0 -76
  98. package/test/cypress/node_modules/cli-cursor/README.md +0 -55
  99. package/test/cypress/node_modules/cli-table3/README.md +0 -218
  100. package/test/cypress/node_modules/cli-truncate/README.md +0 -139
  101. package/test/cypress/node_modules/color-convert/README.md +0 -68
  102. package/test/cypress/node_modules/color-name/README.md +0 -11
  103. package/test/cypress/node_modules/colorette/README.md +0 -102
  104. package/test/cypress/node_modules/colors/README.md +0 -221
  105. package/test/cypress/node_modules/combined-stream/README.md +0 -138
  106. package/test/cypress/node_modules/commander/README.md +0 -737
  107. package/test/cypress/node_modules/common-tags/README.md +0 -687
  108. package/test/cypress/node_modules/core-util-is/README.md +0 -3
  109. package/test/cypress/node_modules/cross-spawn/README.md +0 -96
  110. package/test/cypress/node_modules/cypress/README.md +0 -25
  111. package/test/cypress/node_modules/cypress/angular/README.md +0 -10
  112. package/test/cypress/node_modules/cypress/mount-utils/README.md +0 -140
  113. package/test/cypress/node_modules/cypress/react/README.md +0 -14
  114. package/test/cypress/node_modules/cypress/react18/README.md +0 -7
  115. package/test/cypress/node_modules/cypress/svelte/README.md +0 -15
  116. package/test/cypress/node_modules/cypress/vue/README.md +0 -14
  117. package/test/cypress/node_modules/cypress/vue2/README.md +0 -7
  118. package/test/cypress/node_modules/dashdash/README.md +0 -574
  119. package/test/cypress/node_modules/dayjs/README.md +0 -128
  120. package/test/cypress/node_modules/debug/README.md +0 -481
  121. package/test/cypress/node_modules/delayed-stream/README.md +0 -141
  122. package/test/cypress/node_modules/ecc-jsbn/README.md +0 -8
  123. package/test/cypress/node_modules/emoji-regex/README.md +0 -73
  124. package/test/cypress/node_modules/end-of-stream/README.md +0 -54
  125. package/test/cypress/node_modules/enquirer/README.md +0 -1752
  126. package/test/cypress/node_modules/escape-string-regexp/README.md +0 -27
  127. package/test/cypress/node_modules/eventemitter2/README.md +0 -809
  128. package/test/cypress/node_modules/execa/README.md +0 -656
  129. package/test/cypress/node_modules/executable/README.md +0 -64
  130. package/test/cypress/node_modules/extend/README.md +0 -81
  131. package/test/cypress/node_modules/extract-zip/README.md +0 -57
  132. package/test/cypress/node_modules/extract-zip/node_modules/debug/README.md +0 -455
  133. package/test/cypress/node_modules/extsprintf/README.md +0 -46
  134. package/test/cypress/node_modules/fd-slicer/README.md +0 -199
  135. package/test/cypress/node_modules/figures/README.md +0 -139
  136. package/test/cypress/node_modules/forever-agent/README.md +0 -4
  137. package/test/cypress/node_modules/form-data/README.md +0 -234
  138. package/test/cypress/node_modules/fs-extra/README.md +0 -264
  139. package/test/cypress/node_modules/fs.realpath/README.md +0 -33
  140. package/test/cypress/node_modules/get-stream/README.md +0 -124
  141. package/test/cypress/node_modules/getos/README.md +0 -79
  142. package/test/cypress/node_modules/getpass/README.md +0 -32
  143. package/test/cypress/node_modules/glob/README.md +0 -375
  144. package/test/cypress/node_modules/global-dirs/README.md +0 -72
  145. package/test/cypress/node_modules/graceful-fs/README.md +0 -133
  146. package/test/cypress/node_modules/has-flag/README.md +0 -89
  147. package/test/cypress/node_modules/http-signature/README.md +0 -80
  148. package/test/cypress/node_modules/human-signals/README.md +0 -155
  149. package/test/cypress/node_modules/ieee754/README.md +0 -51
  150. package/test/cypress/node_modules/indent-string/README.md +0 -70
  151. package/test/cypress/node_modules/inflight/README.md +0 -37
  152. package/test/cypress/node_modules/inherits/README.md +0 -42
  153. package/test/cypress/node_modules/ini/README.md +0 -102
  154. package/test/cypress/node_modules/is-ci/README.md +0 -50
  155. package/test/cypress/node_modules/is-fullwidth-code-point/README.md +0 -39
  156. package/test/cypress/node_modules/is-installed-globally/README.md +0 -31
  157. package/test/cypress/node_modules/is-path-inside/README.md +0 -63
  158. package/test/cypress/node_modules/is-stream/README.md +0 -57
  159. package/test/cypress/node_modules/is-typedarray/README.md +0 -16
  160. package/test/cypress/node_modules/isexe/README.md +0 -51
  161. package/test/cypress/node_modules/isstream/README.md +0 -66
  162. package/test/cypress/node_modules/jsbn/README.md +0 -175
  163. package/test/cypress/node_modules/json-schema/README.md +0 -3
  164. package/test/cypress/node_modules/json-stringify-safe/README.md +0 -52
  165. package/test/cypress/node_modules/jsonfile/README.md +0 -205
  166. package/test/cypress/node_modules/jsonfile/node_modules/universalify/README.md +0 -76
  167. package/test/cypress/node_modules/jsprim/README.md +0 -287
  168. package/test/cypress/node_modules/lazy-ass/README.md +0 -269
  169. package/test/cypress/node_modules/listr2/README.md +0 -17
  170. package/test/cypress/node_modules/lodash/README.md +0 -39
  171. package/test/cypress/node_modules/lodash.once/README.md +0 -18
  172. package/test/cypress/node_modules/log-symbols/README.md +0 -51
  173. package/test/cypress/node_modules/log-update/README.md +0 -97
  174. package/test/cypress/node_modules/log-update/node_modules/slice-ansi/README.md +0 -66
  175. package/test/cypress/node_modules/log-update/node_modules/wrap-ansi/README.md +0 -97
  176. package/test/cypress/node_modules/lru-cache/README.md +0 -166
  177. package/test/cypress/node_modules/merge-stream/README.md +0 -78
  178. package/test/cypress/node_modules/mime-db/README.md +0 -102
  179. package/test/cypress/node_modules/mime-types/README.md +0 -123
  180. package/test/cypress/node_modules/mimic-fn/README.md +0 -69
  181. package/test/cypress/node_modules/minimatch/README.md +0 -209
  182. package/test/cypress/node_modules/ms/README.md +0 -60
  183. package/test/cypress/node_modules/npm-run-path/README.md +0 -115
  184. package/test/cypress/node_modules/once/README.md +0 -79
  185. package/test/cypress/node_modules/onetime/README.md +0 -94
  186. package/test/cypress/node_modules/ospath/README.md +0 -53
  187. package/test/cypress/node_modules/p-map/README.md +0 -89
  188. package/test/cypress/node_modules/path-is-absolute/README.md +0 -59
  189. package/test/cypress/node_modules/path-key/README.md +0 -61
  190. package/test/cypress/node_modules/pend/README.md +0 -41
  191. package/test/cypress/node_modules/performance-now/README.md +0 -30
  192. package/test/cypress/node_modules/pify/README.md +0 -119
  193. package/test/cypress/node_modules/pretty-bytes/README.md +0 -131
  194. package/test/cypress/node_modules/proxy-from-env/README.md +0 -131
  195. package/test/cypress/node_modules/psl/README.md +0 -215
  196. package/test/cypress/node_modules/pump/README.md +0 -65
  197. package/test/cypress/node_modules/punycode/README.md +0 -122
  198. package/test/cypress/node_modules/qs/README.md +0 -510
  199. package/test/cypress/node_modules/request-progress/README.md +0 -78
  200. package/test/cypress/node_modules/restore-cursor/README.md +0 -26
  201. package/test/cypress/node_modules/rimraf/README.md +0 -101
  202. package/test/cypress/node_modules/rxjs/README.md +0 -147
  203. package/test/cypress/node_modules/rxjs/_esm2015/README.md +0 -147
  204. package/test/cypress/node_modules/rxjs/_esm5/README.md +0 -147
  205. package/test/cypress/node_modules/rxjs/src/README.md +0 -147
  206. package/test/cypress/node_modules/safe-buffer/README.md +0 -584
  207. package/test/cypress/node_modules/safer-buffer/README.md +0 -156
  208. package/test/cypress/node_modules/semver/README.md +0 -566
  209. package/test/cypress/node_modules/shebang-command/README.md +0 -34
  210. package/test/cypress/node_modules/shebang-regex/README.md +0 -33
  211. package/test/cypress/node_modules/signal-exit/README.md +0 -39
  212. package/test/cypress/node_modules/slice-ansi/README.md +0 -72
  213. package/test/cypress/node_modules/sshpk/README.md +0 -804
  214. package/test/cypress/node_modules/string-width/README.md +0 -50
  215. package/test/cypress/node_modules/strip-ansi/README.md +0 -46
  216. package/test/cypress/node_modules/strip-final-newline/README.md +0 -30
  217. package/test/cypress/node_modules/supports-color/README.md +0 -77
  218. package/test/cypress/node_modules/throttleit/README.md +0 -32
  219. package/test/cypress/node_modules/tmp/README.md +0 -365
  220. package/test/cypress/node_modules/tough-cookie/README.md +0 -527
  221. package/test/cypress/node_modules/tslib/README.md +0 -142
  222. package/test/cypress/node_modules/tunnel-agent/README.md +0 -4
  223. package/test/cypress/node_modules/tweetnacl/README.md +0 -459
  224. package/test/cypress/node_modules/type-fest/README.md +0 -760
  225. package/test/cypress/node_modules/universalify/README.md +0 -76
  226. package/test/cypress/node_modules/untildify/README.md +0 -30
  227. package/test/cypress/node_modules/uuid/README.md +0 -505
  228. package/test/cypress/node_modules/verror/README.md +0 -528
  229. package/test/cypress/node_modules/verror/node_modules/extsprintf/README.md +0 -46
  230. package/test/cypress/node_modules/which/README.md +0 -54
  231. package/test/cypress/node_modules/wrap-ansi/README.md +0 -91
  232. package/test/cypress/node_modules/wrappy/README.md +0 -36
  233. package/test/cypress/node_modules/yallist/README.md +0 -204
  234. package/test/cypress/node_modules/yauzl/README.md +0 -658
@@ -0,0 +1,1245 @@
1
+ // import 3rd party vendor libs
2
+ import 'slickgrid/slick.core';
3
+ import 'slickgrid/slick.interactions';
4
+ import 'slickgrid/slick.grid';
5
+ import 'slickgrid/slick.dataview';
6
+ // ...then everything else...
7
+ import { ApplicationRef, Component, EventEmitter, Inject, Input, Optional, Output, } from '@angular/core';
8
+ import { Observable } from 'rxjs';
9
+ import { ExtensionName,
10
+ // services
11
+ BackendUtilityService, CollectionService, EventNamingStyle, ExtensionService, ExtensionUtility, FilterFactory, FilterService, GridEventService, GridService, GridStateService, GroupingAndColspanService, PaginationService, ResizerService, SharedService, SlickgridConfig, SlickGroupItemMetadataProvider, SortService, TreeDataService,
12
+ // utilities
13
+ autoAddEditorFormatterToColumnsWithEditor, emptyElement, GridStateType, unsubscribeAll, } from '@slickgrid-universal/common';
14
+ import { EventPubSubService } from '@slickgrid-universal/event-pub-sub';
15
+ import { SlickEmptyWarningComponent } from '@slickgrid-universal/empty-warning-component';
16
+ import { SlickFooterComponent } from '@slickgrid-universal/custom-footer-component';
17
+ import { SlickPaginationComponent } from '@slickgrid-universal/pagination-component';
18
+ import { RxJsResource } from '@slickgrid-universal/rxjs-observable';
19
+ import { dequal } from 'dequal/lite';
20
+ import { Constants } from '../constants';
21
+ import { GlobalGridOptions } from './../global-grid-options';
22
+ import { TranslaterService } from '../services/translater.service';
23
+ // Services
24
+ import { AngularUtilService } from '../services/angularUtil.service';
25
+ import { SlickRowDetailView } from '../extensions/slickRowDetailView';
26
+ import * as i0 from "@angular/core";
27
+ import * as i1 from "../services/angularUtil.service";
28
+ import * as i2 from "../services/container.service";
29
+ import * as i3 from "@ngx-translate/core";
30
+ import * as i4 from "../services/translater.service";
31
+ class AngularSlickgridComponent {
32
+ angularUtilService;
33
+ appRef;
34
+ cd;
35
+ containerService;
36
+ elm;
37
+ translate;
38
+ translaterService;
39
+ forRootConfig;
40
+ _dataset;
41
+ _columnDefinitions;
42
+ _currentDatasetLength = 0;
43
+ _eventHandler = new Slick.EventHandler();
44
+ _eventPubSubService;
45
+ _angularGridInstances;
46
+ _hideHeaderRowAfterPageLoad = false;
47
+ _isGridInitialized = false;
48
+ _isDatasetInitialized = false;
49
+ _isDatasetHierarchicalInitialized = false;
50
+ _isPaginationInitialized = false;
51
+ _isLocalGrid = true;
52
+ _paginationOptions;
53
+ _registeredResources = [];
54
+ dataView;
55
+ slickGrid;
56
+ groupingDefinition = {};
57
+ groupItemMetadataProvider;
58
+ backendServiceApi;
59
+ locales;
60
+ metrics;
61
+ showPagination = false;
62
+ serviceList = [];
63
+ totalItems = 0;
64
+ paginationData;
65
+ subscriptions = [];
66
+ // components / plugins
67
+ slickEmptyWarning;
68
+ slickFooter;
69
+ slickPagination;
70
+ slickRowDetailView;
71
+ // services
72
+ backendUtilityService;
73
+ collectionService;
74
+ extensionService;
75
+ extensionUtility;
76
+ filterFactory;
77
+ filterService;
78
+ gridEventService;
79
+ gridService;
80
+ gridStateService;
81
+ groupingService;
82
+ paginationService;
83
+ resizerService;
84
+ rxjs;
85
+ sharedService;
86
+ sortService;
87
+ treeDataService;
88
+ customDataView;
89
+ gridId = '';
90
+ gridOptions;
91
+ get paginationOptions() {
92
+ return this._paginationOptions;
93
+ }
94
+ set paginationOptions(newPaginationOptions) {
95
+ if (newPaginationOptions && this._paginationOptions) {
96
+ this._paginationOptions = { ...this.gridOptions.pagination, ...this._paginationOptions, ...newPaginationOptions };
97
+ }
98
+ else {
99
+ this._paginationOptions = newPaginationOptions;
100
+ }
101
+ this.gridOptions.pagination = this._paginationOptions ?? this.gridOptions.pagination;
102
+ this.paginationService.updateTotalItems(this.gridOptions.pagination?.totalItems ?? 0, true);
103
+ }
104
+ set columnDefinitions(columnDefinitions) {
105
+ this._columnDefinitions = columnDefinitions;
106
+ if (this._isGridInitialized) {
107
+ this.updateColumnDefinitionsList(columnDefinitions);
108
+ }
109
+ if (columnDefinitions.length > 0) {
110
+ this.copyColumnWidthsReference(columnDefinitions);
111
+ }
112
+ }
113
+ get columnDefinitions() {
114
+ return this._columnDefinitions;
115
+ }
116
+ // make the columnDefinitions a 2-way binding so that plugin adding cols
117
+ // are synched on user's side as well (RowMove, RowDetail, RowSelections)
118
+ columnDefinitionsChange = new EventEmitter(true);
119
+ get dataset() {
120
+ return (this.customDataView ? this.slickGrid?.getData?.() : this.dataView?.getItems?.()) || [];
121
+ }
122
+ set dataset(newDataset) {
123
+ const prevDatasetLn = this._currentDatasetLength;
124
+ const isDatasetEqual = dequal(newDataset, this._dataset || []);
125
+ let data = newDataset;
126
+ // when Tree Data is enabled and we don't yet have the hierarchical dataset filled, we can force a convert+sort of the array
127
+ if (this.slickGrid && this.gridOptions?.enableTreeData && Array.isArray(newDataset) && (newDataset.length > 0 || newDataset.length !== prevDatasetLn || !isDatasetEqual)) {
128
+ this._isDatasetHierarchicalInitialized = false;
129
+ data = this.sortTreeDataset(newDataset, !isDatasetEqual); // if dataset changed, then force a refresh anyway
130
+ }
131
+ this._dataset = data;
132
+ this.refreshGridData(data || []);
133
+ this._currentDatasetLength = (newDataset || []).length;
134
+ // expand/autofit columns on first page load
135
+ // we can assume that if the prevDataset was empty then we are on first load
136
+ if (this.gridOptions?.autoFitColumnsOnFirstLoad && prevDatasetLn === 0) {
137
+ this.slickGrid.autosizeColumns();
138
+ }
139
+ }
140
+ get datasetHierarchical() {
141
+ return this.sharedService.hierarchicalDataset;
142
+ }
143
+ set datasetHierarchical(newHierarchicalDataset) {
144
+ const isDatasetEqual = dequal(newHierarchicalDataset, this.sharedService?.hierarchicalDataset ?? []);
145
+ const prevFlatDatasetLn = this._currentDatasetLength;
146
+ this.sharedService.hierarchicalDataset = newHierarchicalDataset;
147
+ if (newHierarchicalDataset && this.columnDefinitions && this.filterService?.clearFilters) {
148
+ this.filterService.clearFilters();
149
+ }
150
+ // when a hierarchical dataset is set afterward, we can reset the flat dataset and call a tree data sort that will overwrite the flat dataset
151
+ if (newHierarchicalDataset && this.slickGrid && this.sortService?.processTreeDataInitialSort) {
152
+ this.dataView.setItems([], this.gridOptions.datasetIdPropertyName ?? 'id');
153
+ this.sortService.processTreeDataInitialSort();
154
+ // we also need to reset/refresh the Tree Data filters because if we inserted new item(s) then it might not show up without doing this refresh
155
+ // however we need 1 cpu cycle before having the DataView refreshed, so we need to wrap this check in a setTimeout
156
+ setTimeout(() => {
157
+ const flatDatasetLn = this.dataView.getItemCount();
158
+ if (flatDatasetLn > 0 && (flatDatasetLn !== prevFlatDatasetLn || !isDatasetEqual)) {
159
+ this.filterService.refreshTreeDataFilters();
160
+ }
161
+ });
162
+ this._isDatasetHierarchicalInitialized = true;
163
+ }
164
+ }
165
+ get elementRef() {
166
+ return this.elm;
167
+ }
168
+ get eventHandler() {
169
+ return this._eventHandler;
170
+ }
171
+ get gridContainerElement() {
172
+ return document.querySelector(`#${this.gridOptions.gridContainerId || ''}`);
173
+ }
174
+ /** GETTER to know if dataset was initialized or not */
175
+ get isDatasetInitialized() {
176
+ return this._isDatasetInitialized;
177
+ }
178
+ /** SETTER to change if dataset was initialized or not (stringly used for unit testing purposes) */
179
+ set isDatasetInitialized(isInitialized) {
180
+ this._isDatasetInitialized = isInitialized;
181
+ }
182
+ set isDatasetHierarchicalInitialized(isInitialized) {
183
+ this._isDatasetHierarchicalInitialized = isInitialized;
184
+ }
185
+ get registeredResources() {
186
+ return this._registeredResources;
187
+ }
188
+ constructor(angularUtilService, appRef, cd, containerService, elm, translate, translaterService, forRootConfig, externalServices) {
189
+ this.angularUtilService = angularUtilService;
190
+ this.appRef = appRef;
191
+ this.cd = cd;
192
+ this.containerService = containerService;
193
+ this.elm = elm;
194
+ this.translate = translate;
195
+ this.translaterService = translaterService;
196
+ this.forRootConfig = forRootConfig;
197
+ const slickgridConfig = new SlickgridConfig();
198
+ // initialize and assign all Service Dependencies
199
+ this._eventPubSubService = externalServices?.eventPubSubService ?? new EventPubSubService(this.elm.nativeElement);
200
+ this._eventPubSubService.eventNamingStyle = EventNamingStyle.camelCase;
201
+ this.backendUtilityService = externalServices?.backendUtilityService ?? new BackendUtilityService();
202
+ this.gridEventService = externalServices?.gridEventService ?? new GridEventService();
203
+ this.sharedService = externalServices?.sharedService ?? new SharedService();
204
+ this.collectionService = externalServices?.collectionService ?? new CollectionService(this.translaterService);
205
+ this.extensionUtility = externalServices?.extensionUtility ?? new ExtensionUtility(this.sharedService, this.backendUtilityService, this.translaterService);
206
+ this.filterFactory = new FilterFactory(slickgridConfig, this.translaterService, this.collectionService);
207
+ this.filterService = externalServices?.filterService ?? new FilterService(this.filterFactory, this._eventPubSubService, this.sharedService, this.backendUtilityService);
208
+ this.resizerService = externalServices?.resizerService ?? new ResizerService(this._eventPubSubService);
209
+ this.sortService = externalServices?.sortService ?? new SortService(this.sharedService, this._eventPubSubService, this.backendUtilityService);
210
+ this.treeDataService = externalServices?.treeDataService ?? new TreeDataService(this._eventPubSubService, this.sharedService, this.sortService);
211
+ this.paginationService = externalServices?.paginationService ?? new PaginationService(this._eventPubSubService, this.sharedService, this.backendUtilityService);
212
+ this.extensionService = externalServices?.extensionService ?? new ExtensionService(this.extensionUtility, this.filterService, this._eventPubSubService, this.sharedService, this.sortService, this.treeDataService, this.translaterService);
213
+ this.gridStateService = externalServices?.gridStateService ?? new GridStateService(this.extensionService, this.filterService, this._eventPubSubService, this.sharedService, this.sortService, this.treeDataService);
214
+ this.gridService = externalServices?.gridService ?? new GridService(this.gridStateService, this.filterService, this._eventPubSubService, this.paginationService, this.sharedService, this.sortService, this.treeDataService);
215
+ this.groupingService = externalServices?.groupingAndColspanService ?? new GroupingAndColspanService(this.extensionUtility, this._eventPubSubService);
216
+ this.serviceList = [
217
+ this.containerService,
218
+ this.extensionService,
219
+ this.filterService,
220
+ this.gridEventService,
221
+ this.gridService,
222
+ this.gridStateService,
223
+ this.groupingService,
224
+ this.paginationService,
225
+ this.resizerService,
226
+ this.sortService,
227
+ this.treeDataService,
228
+ ];
229
+ // register all Service instances in the container
230
+ this.containerService.registerInstance('ExtensionUtility', this.extensionUtility);
231
+ this.containerService.registerInstance('FilterService', this.filterService);
232
+ this.containerService.registerInstance('CollectionService', this.collectionService);
233
+ this.containerService.registerInstance('ExtensionService', this.extensionService);
234
+ this.containerService.registerInstance('GridEventService', this.gridEventService);
235
+ this.containerService.registerInstance('GridService', this.gridService);
236
+ this.containerService.registerInstance('GridStateService', this.gridStateService);
237
+ this.containerService.registerInstance('GroupingAndColspanService', this.groupingService);
238
+ this.containerService.registerInstance('PaginationService', this.paginationService);
239
+ this.containerService.registerInstance('ResizerService', this.resizerService);
240
+ this.containerService.registerInstance('SharedService', this.sharedService);
241
+ this.containerService.registerInstance('SortService', this.sortService);
242
+ this.containerService.registerInstance('EventPubSubService', this._eventPubSubService);
243
+ this.containerService.registerInstance('PubSubService', this._eventPubSubService);
244
+ this.containerService.registerInstance('TranslaterService', this.translaterService);
245
+ this.containerService.registerInstance('TreeDataService', this.treeDataService);
246
+ }
247
+ ngAfterViewInit() {
248
+ if (!this.gridOptions || !this.columnDefinitions) {
249
+ throw new Error('Using `<angular-slickgrid>` requires [gridOptions] and [columnDefinitions], it seems that you might have forgot to provide them since at least of them is undefined.');
250
+ }
251
+ this.initialization(this._eventHandler);
252
+ this._isGridInitialized = true;
253
+ // recheck the empty warning message after grid is shown so that it works in every use case
254
+ if (this.gridOptions && this.gridOptions.enableEmptyDataWarningMessage && Array.isArray(this.dataset)) {
255
+ const finalTotalCount = this.dataset.length;
256
+ this.displayEmptyDataWarning(finalTotalCount < 1);
257
+ }
258
+ }
259
+ ngOnDestroy() {
260
+ this._eventPubSubService.publish('onBeforeGridDestroy', this.slickGrid);
261
+ this.destroy();
262
+ this._eventPubSubService.publish('onAfterGridDestroyed', true);
263
+ }
264
+ destroy(shouldEmptyDomElementContainer = false) {
265
+ // dispose of all Services
266
+ this.serviceList.forEach((service) => {
267
+ if (service && service.dispose) {
268
+ service.dispose();
269
+ }
270
+ });
271
+ this.serviceList = [];
272
+ // dispose all registered external resources
273
+ if (Array.isArray(this._registeredResources)) {
274
+ while (this._registeredResources.length > 0) {
275
+ const resource = this._registeredResources.pop();
276
+ if (resource?.dispose) {
277
+ resource.dispose();
278
+ }
279
+ }
280
+ this._registeredResources = [];
281
+ }
282
+ // dispose the Components
283
+ this.slickEmptyWarning?.dispose();
284
+ this.slickFooter?.dispose();
285
+ this.slickPagination?.dispose();
286
+ if (this._eventHandler?.unsubscribeAll) {
287
+ this._eventHandler.unsubscribeAll();
288
+ }
289
+ this._eventPubSubService?.unsubscribeAll();
290
+ if (this.dataView) {
291
+ if (this.dataView?.setItems) {
292
+ this.dataView.setItems([]);
293
+ }
294
+ if (this.dataView.destroy) {
295
+ this.dataView.destroy();
296
+ }
297
+ }
298
+ if (this.slickGrid?.destroy) {
299
+ this.slickGrid.destroy(shouldEmptyDomElementContainer);
300
+ }
301
+ if (this.backendServiceApi) {
302
+ for (const prop of Object.keys(this.backendServiceApi)) {
303
+ delete this.backendServiceApi[prop];
304
+ }
305
+ this.backendServiceApi = undefined;
306
+ }
307
+ for (const prop of Object.keys(this.columnDefinitions)) {
308
+ this.columnDefinitions[prop] = null;
309
+ }
310
+ for (const prop of Object.keys(this.sharedService)) {
311
+ this.sharedService[prop] = null;
312
+ }
313
+ // we could optionally also empty the content of the grid container DOM element
314
+ if (shouldEmptyDomElementContainer) {
315
+ this.emptyGridContainerElm();
316
+ }
317
+ // also unsubscribe all RxJS subscriptions
318
+ this.subscriptions = unsubscribeAll(this.subscriptions);
319
+ this._dataset = null;
320
+ this.datasetHierarchical = undefined;
321
+ this._columnDefinitions = [];
322
+ this._angularGridInstances = undefined;
323
+ this.slickGrid = undefined;
324
+ }
325
+ emptyGridContainerElm() {
326
+ const gridContainerId = this.gridOptions?.gridContainerId ?? 'grid1';
327
+ const gridContainerElm = document.querySelector(`#${gridContainerId}`);
328
+ emptyElement(gridContainerElm);
329
+ }
330
+ /**
331
+ * Define our internal Post Process callback, it will execute internally after we get back result from the Process backend call
332
+ * For now, this is GraphQL Service ONLY feature and it will basically refresh the Dataset & Pagination without having the user to create his own PostProcess every time
333
+ */
334
+ createBackendApiInternalPostProcessCallback(gridOptions) {
335
+ const backendApi = gridOptions && gridOptions.backendServiceApi;
336
+ if (backendApi && backendApi.service) {
337
+ const backendApiService = backendApi.service;
338
+ // internalPostProcess only works (for now) with a GraphQL Service, so make sure it is of that type
339
+ if (typeof backendApiService.getDatasetName === 'function') {
340
+ backendApi.internalPostProcess = (processResult) => {
341
+ const datasetName = (backendApi && backendApiService && typeof backendApiService.getDatasetName === 'function') ? backendApiService.getDatasetName() : '';
342
+ if (processResult?.data[datasetName]) {
343
+ const data = processResult.data[datasetName].hasOwnProperty('nodes') ? processResult.data[datasetName].nodes : processResult.data[datasetName];
344
+ const totalCount = processResult.data[datasetName].hasOwnProperty('totalCount') ? processResult.data[datasetName].totalCount : processResult.data[datasetName].length;
345
+ this.refreshGridData(data, totalCount || 0);
346
+ }
347
+ };
348
+ }
349
+ }
350
+ }
351
+ initialization(eventHandler) {
352
+ this.gridOptions.translater = this.translaterService;
353
+ this._eventHandler = eventHandler;
354
+ // when detecting a frozen grid, we'll automatically enable the mousewheel scroll handler so that we can scroll from both left/right frozen containers
355
+ if (this.gridOptions && ((this.gridOptions.frozenRow !== undefined && this.gridOptions.frozenRow >= 0) || this.gridOptions.frozenColumn !== undefined && this.gridOptions.frozenColumn >= 0) && this.gridOptions.enableMouseWheelScrollHandler === undefined) {
356
+ this.gridOptions.enableMouseWheelScrollHandler = true;
357
+ }
358
+ this._eventPubSubService.eventNamingStyle = this.gridOptions?.eventNamingStyle ?? EventNamingStyle.camelCase;
359
+ this._eventPubSubService.publish('onBeforeGridCreate', true);
360
+ // make sure the dataset is initialized (if not it will throw an error that it cannot getLength of null)
361
+ this._dataset = this._dataset || [];
362
+ this.gridOptions = this.mergeGridOptions(this.gridOptions);
363
+ this._paginationOptions = this.gridOptions?.pagination;
364
+ this.locales = this.gridOptions?.locales ?? Constants.locales;
365
+ this.backendServiceApi = this.gridOptions?.backendServiceApi;
366
+ this._isLocalGrid = !this.backendServiceApi; // considered a local grid if it doesn't have a backend service set
367
+ this.createBackendApiInternalPostProcessCallback(this.gridOptions);
368
+ if (!this.customDataView) {
369
+ const dataviewInlineFilters = this.gridOptions.dataView && this.gridOptions.dataView.inlineFilters || false;
370
+ let dataViewOptions = { inlineFilters: dataviewInlineFilters };
371
+ if (this.gridOptions.draggableGrouping || this.gridOptions.enableGrouping) {
372
+ this.groupItemMetadataProvider = new SlickGroupItemMetadataProvider();
373
+ this.sharedService.groupItemMetadataProvider = this.groupItemMetadataProvider;
374
+ dataViewOptions = { ...dataViewOptions, groupItemMetadataProvider: this.groupItemMetadataProvider };
375
+ }
376
+ this.dataView = new Slick.Data.DataView(dataViewOptions);
377
+ this._eventPubSubService.publish('onDataviewCreated', this.dataView);
378
+ }
379
+ // get any possible Services that user want to register which don't require SlickGrid to be instantiated
380
+ // RxJS Resource is in this lot because it has to be registered before anything else and doesn't require SlickGrid to be initialized
381
+ this.preRegisterResources();
382
+ // for convenience to the user, we provide the property "editor" as an Angular-Slickgrid editor complex object
383
+ // however "editor" is used internally by SlickGrid for it's own Editor Factory
384
+ // so in our lib we will swap "editor" and copy it into a new property called "internalColumnEditor"
385
+ // then take back "editor.model" and make it the new "editor" so that SlickGrid Editor Factory still works
386
+ this._columnDefinitions = this.swapInternalEditorToSlickGridFactoryEditor(this._columnDefinitions);
387
+ // if the user wants to automatically add a Custom Editor Formatter, we need to call the auto add function again
388
+ if (this.gridOptions.autoAddCustomEditorFormatter) {
389
+ autoAddEditorFormatterToColumnsWithEditor(this._columnDefinitions, this.gridOptions.autoAddCustomEditorFormatter);
390
+ }
391
+ // save reference for all columns before they optionally become hidden/visible
392
+ this.sharedService.allColumns = this._columnDefinitions;
393
+ this.sharedService.visibleColumns = this._columnDefinitions;
394
+ // before certain extentions/plugins potentially adds extra columns not created by the user itself (RowMove, RowDetail, RowSelections)
395
+ // we'll subscribe to the event and push back the change to the user so they always use full column defs array including extra cols
396
+ this.subscriptions.push(this._eventPubSubService.subscribe('onPluginColumnsChanged', data => {
397
+ this._columnDefinitions = data.columns;
398
+ this.columnDefinitionsChange.emit(this._columnDefinitions);
399
+ }));
400
+ // after subscribing to potential columns changed, we are ready to create these optional extensions
401
+ // when we did find some to create (RowMove, RowDetail, RowSelections), it will automatically modify column definitions (by previous subscribe)
402
+ this.extensionService.createExtensionsBeforeGridCreation(this._columnDefinitions, this.gridOptions);
403
+ // if user entered some Pinning/Frozen "presets", we need to apply them in the grid options
404
+ if (this.gridOptions.presets?.pinning) {
405
+ this.gridOptions = { ...this.gridOptions, ...this.gridOptions.presets.pinning };
406
+ }
407
+ // build SlickGrid Grid, also user might optionally pass a custom dataview (e.g. remote model)
408
+ this.slickGrid = new Slick.Grid(`#${this.gridId}`, this.customDataView || this.dataView, this._columnDefinitions, this.gridOptions);
409
+ this.sharedService.dataView = this.dataView;
410
+ this.sharedService.slickGrid = this.slickGrid;
411
+ this.sharedService.gridContainerElement = this.elm.nativeElement;
412
+ this.extensionService.bindDifferentExtensions();
413
+ this.bindDifferentHooks(this.slickGrid, this.gridOptions, this.dataView);
414
+ // when it's a frozen grid, we need to keep the frozen column id for reference if we ever show/hide column from ColumnPicker/GridMenu afterward
415
+ const frozenColumnIndex = this.gridOptions.frozenColumn !== undefined ? this.gridOptions.frozenColumn : -1;
416
+ if (frozenColumnIndex >= 0 && frozenColumnIndex <= this._columnDefinitions.length) {
417
+ this.sharedService.frozenVisibleColumnId = this._columnDefinitions[frozenColumnIndex].id || '';
418
+ }
419
+ // get any possible Services that user want to register
420
+ this.registerResources();
421
+ // initialize the SlickGrid grid
422
+ this.slickGrid.init();
423
+ // initialized the resizer service only after SlickGrid is initialized
424
+ // if we don't we end up binding our resize to a grid element that doesn't yet exist in the DOM and the resizer service will fail silently (because it has a try/catch that unbinds the resize without throwing back)
425
+ if (this.gridContainerElement) {
426
+ this.resizerService.init(this.slickGrid, this.gridContainerElement);
427
+ }
428
+ // user could show a custom footer with the data metrics (dataset length and last updated timestamp)
429
+ if (!this.gridOptions.enablePagination && this.gridOptions.showCustomFooter && this.gridOptions.customFooterOptions && this.gridContainerElement) {
430
+ this.slickFooter = new SlickFooterComponent(this.slickGrid, this.gridOptions.customFooterOptions, this._eventPubSubService, this.translaterService);
431
+ this.slickFooter.renderFooter(this.gridContainerElement);
432
+ }
433
+ if (!this.customDataView && this.dataView) {
434
+ // load the data in the DataView (unless it's a hierarchical dataset, if so it will be loaded after the initial tree sort)
435
+ const initialDataset = this.gridOptions?.enableTreeData ? this.sortTreeDataset(this._dataset) : this._dataset;
436
+ this.dataView.beginUpdate();
437
+ this.dataView.setItems(initialDataset || [], this.gridOptions.datasetIdPropertyName ?? 'id');
438
+ this.dataView.endUpdate();
439
+ // if you don't want the items that are not visible (due to being filtered out or being on a different page)
440
+ // to stay selected, pass 'false' to the second arg
441
+ if (this.slickGrid?.getSelectionModel() && this.gridOptions && this.gridOptions.dataView && this.gridOptions.dataView.hasOwnProperty('syncGridSelection')) {
442
+ // if we are using a Backend Service, we will do an extra flag check, the reason is because it might have some unintended behaviors
443
+ // with the BackendServiceApi because technically the data in the page changes the DataView on every page change.
444
+ let preservedRowSelectionWithBackend = false;
445
+ if (this.gridOptions.backendServiceApi && this.gridOptions.dataView.hasOwnProperty('syncGridSelectionWithBackendService')) {
446
+ preservedRowSelectionWithBackend = this.gridOptions.dataView.syncGridSelectionWithBackendService;
447
+ }
448
+ const syncGridSelection = this.gridOptions.dataView.syncGridSelection;
449
+ if (typeof syncGridSelection === 'boolean') {
450
+ let preservedRowSelection = syncGridSelection;
451
+ if (!this._isLocalGrid) {
452
+ // when using BackendServiceApi, we'll be using the "syncGridSelectionWithBackendService" flag BUT "syncGridSelection" must also be set to True
453
+ preservedRowSelection = syncGridSelection && preservedRowSelectionWithBackend;
454
+ }
455
+ this.dataView.syncGridSelection(this.slickGrid, preservedRowSelection);
456
+ }
457
+ else if (typeof syncGridSelection === 'object') {
458
+ this.dataView.syncGridSelection(this.slickGrid, syncGridSelection.preserveHidden, syncGridSelection.preserveHiddenOnSelectionChange);
459
+ }
460
+ }
461
+ const datasetLn = this.dataView.getLength() || this._dataset && this._dataset.length || 0;
462
+ if (datasetLn > 0) {
463
+ if (!this._isDatasetInitialized && (this.gridOptions.enableCheckboxSelector || this.gridOptions.enableRowSelection)) {
464
+ this.loadRowSelectionPresetWhenExists();
465
+ }
466
+ this.loadFilterPresetsWhenDatasetInitialized();
467
+ this._isDatasetInitialized = true;
468
+ }
469
+ }
470
+ // user might want to hide the header row on page load but still have `enableFiltering: true`
471
+ // if that is the case, we need to hide the headerRow ONLY AFTER all filters got created & dataView exist
472
+ if (this._hideHeaderRowAfterPageLoad) {
473
+ this.showHeaderRow(false);
474
+ this.sharedService.hideHeaderRowAfterPageLoad = this._hideHeaderRowAfterPageLoad;
475
+ }
476
+ // publish & dispatch certain events
477
+ this._eventPubSubService.publish('onGridCreated', this.slickGrid);
478
+ // after the DataView is created & updated execute some processes
479
+ if (!this.customDataView) {
480
+ this.executeAfterDataviewCreated(this.slickGrid, this.gridOptions);
481
+ }
482
+ // bind resize ONLY after the dataView is ready
483
+ this.bindResizeHook(this.slickGrid, this.gridOptions);
484
+ // bind the Backend Service API callback functions only after the grid is initialized
485
+ // because the preProcess() and onInit() might get triggered
486
+ if (this.gridOptions?.backendServiceApi) {
487
+ this.bindBackendCallbackFunctions(this.gridOptions);
488
+ }
489
+ // local grid, check if we need to show the Pagination
490
+ // if so then also check if there's any presets and finally initialize the PaginationService
491
+ // a local grid with Pagination presets will potentially have a different total of items, we'll need to get it from the DataView and update our total
492
+ if (this.gridOptions?.enablePagination && this._isLocalGrid) {
493
+ this.showPagination = true;
494
+ this.loadLocalGridPagination(this.dataset);
495
+ }
496
+ this._angularGridInstances = {
497
+ // Slick Grid & DataView objects
498
+ dataView: this.dataView,
499
+ slickGrid: this.slickGrid,
500
+ extensions: this.extensionService?.extensionList,
501
+ // public methods
502
+ destroy: this.destroy.bind(this),
503
+ // return all available Services (non-singleton)
504
+ backendService: this.gridOptions?.backendServiceApi?.service,
505
+ eventPubSubService: this._eventPubSubService,
506
+ filterService: this.filterService,
507
+ gridEventService: this.gridEventService,
508
+ gridStateService: this.gridStateService,
509
+ gridService: this.gridService,
510
+ groupingService: this.groupingService,
511
+ extensionService: this.extensionService,
512
+ paginationService: this.paginationService,
513
+ resizerService: this.resizerService,
514
+ sortService: this.sortService,
515
+ treeDataService: this.treeDataService,
516
+ };
517
+ // all instances (SlickGrid, DataView & all Services)
518
+ this._eventPubSubService.publish('onAngularGridCreated', this._angularGridInstances);
519
+ }
520
+ /**
521
+ * On a Pagination changed, we will trigger a Grid State changed with the new pagination info
522
+ * Also if we use Row Selection or the Checkbox Selector with a Backend Service (Odata, GraphQL), we need to reset any selection
523
+ */
524
+ paginationChanged(pagination) {
525
+ const isSyncGridSelectionEnabled = this.gridStateService?.needToPreserveRowSelection() ?? false;
526
+ if (this.slickGrid && !isSyncGridSelectionEnabled && this.gridOptions?.backendServiceApi && (this.gridOptions.enableRowSelection || this.gridOptions.enableCheckboxSelector)) {
527
+ this.slickGrid.setSelectedRows([]);
528
+ }
529
+ const { pageNumber, pageSize } = pagination;
530
+ if (this.sharedService) {
531
+ if (pageSize !== undefined && pageNumber !== undefined) {
532
+ this.sharedService.currentPagination = { pageNumber, pageSize };
533
+ }
534
+ }
535
+ this._eventPubSubService.publish('onGridStateChanged', {
536
+ change: { newValues: { pageNumber, pageSize }, type: GridStateType.pagination },
537
+ gridState: this.gridStateService.getCurrentGridState()
538
+ });
539
+ this.cd.markForCheck();
540
+ }
541
+ /**
542
+ * When dataset changes, we need to refresh the entire grid UI & possibly resize it as well
543
+ * @param dataset
544
+ */
545
+ refreshGridData(dataset, totalCount) {
546
+ if (this.gridOptions && this.gridOptions.enableEmptyDataWarningMessage && Array.isArray(dataset)) {
547
+ const finalTotalCount = totalCount || dataset.length;
548
+ this.displayEmptyDataWarning(finalTotalCount < 1);
549
+ }
550
+ if (Array.isArray(dataset) && this.slickGrid && this.dataView?.setItems) {
551
+ this.dataView.setItems(dataset, this.gridOptions.datasetIdPropertyName ?? 'id');
552
+ if (!this.gridOptions.backendServiceApi && !this.gridOptions.enableTreeData) {
553
+ this.dataView.reSort();
554
+ }
555
+ if (dataset.length > 0) {
556
+ if (!this._isDatasetInitialized) {
557
+ this.loadFilterPresetsWhenDatasetInitialized();
558
+ if (this.gridOptions.enableCheckboxSelector) {
559
+ this.loadRowSelectionPresetWhenExists();
560
+ }
561
+ }
562
+ this._isDatasetInitialized = true;
563
+ }
564
+ if (dataset) {
565
+ this.slickGrid.invalidate();
566
+ }
567
+ // display the Pagination component only after calling this refresh data first, we call it here so that if we preset pagination page number it will be shown correctly
568
+ this.showPagination = (this.gridOptions && (this.gridOptions.enablePagination || (this.gridOptions.backendServiceApi && this.gridOptions.enablePagination === undefined))) ? true : false;
569
+ if (this._paginationOptions && this.gridOptions?.pagination && this.gridOptions?.backendServiceApi) {
570
+ const paginationOptions = this.setPaginationOptionsWhenPresetDefined(this.gridOptions, this._paginationOptions);
571
+ // when we have a totalCount use it, else we'll take it from the pagination object
572
+ // only update the total items if it's different to avoid refreshing the UI
573
+ const totalRecords = (totalCount !== undefined) ? totalCount : (this.gridOptions?.pagination?.totalItems);
574
+ if (totalRecords !== undefined && totalRecords !== this.totalItems) {
575
+ this.totalItems = +totalRecords;
576
+ }
577
+ // initialize the Pagination Service with new pagination options (which might have presets)
578
+ if (!this._isPaginationInitialized) {
579
+ this.initializePaginationService(paginationOptions);
580
+ }
581
+ else {
582
+ // update the pagination service with the new total
583
+ this.paginationService.updateTotalItems(this.totalItems);
584
+ }
585
+ }
586
+ // resize the grid inside a slight timeout, in case other DOM element changed prior to the resize (like a filter/pagination changed)
587
+ if (this.slickGrid && this.gridOptions.enableAutoResize) {
588
+ const delay = this.gridOptions.autoResize && this.gridOptions.autoResize.delay;
589
+ this.resizerService.resizeGrid(delay || 10);
590
+ }
591
+ }
592
+ }
593
+ /**
594
+ * Check if there's any Pagination Presets defined in the Grid Options,
595
+ * if there are then load them in the paginationOptions object
596
+ */
597
+ setPaginationOptionsWhenPresetDefined(gridOptions, paginationOptions) {
598
+ if (gridOptions.presets?.pagination && paginationOptions && !this._isPaginationInitialized) {
599
+ paginationOptions.pageSize = gridOptions.presets.pagination.pageSize;
600
+ paginationOptions.pageNumber = gridOptions.presets.pagination.pageNumber;
601
+ }
602
+ return paginationOptions;
603
+ }
604
+ /**
605
+ * Dynamically change or update the column definitions list.
606
+ * We will re-render the grid so that the new header and data shows up correctly.
607
+ * If using i18n, we also need to trigger a re-translate of the column headers
608
+ */
609
+ updateColumnDefinitionsList(newColumnDefinitions) {
610
+ // map/swap the internal library Editor to the SlickGrid Editor factory
611
+ newColumnDefinitions = this.swapInternalEditorToSlickGridFactoryEditor(newColumnDefinitions);
612
+ if (this.gridOptions.enableTranslate) {
613
+ this.extensionService.translateColumnHeaders(false, newColumnDefinitions);
614
+ }
615
+ else {
616
+ this.extensionService.renderColumnHeaders(newColumnDefinitions, true);
617
+ }
618
+ if (this.gridOptions?.enableAutoSizeColumns) {
619
+ this.slickGrid.autosizeColumns();
620
+ }
621
+ else if (this.gridOptions?.enableAutoResizeColumnsByCellContent && this.resizerService?.resizeColumnsByCellContent) {
622
+ this.resizerService.resizeColumnsByCellContent();
623
+ }
624
+ }
625
+ /**
626
+ * Show the filter row displayed on first row, we can optionally pass false to hide it.
627
+ * @param showing
628
+ */
629
+ showHeaderRow(showing = true) {
630
+ this.slickGrid.setHeaderRowVisibility(showing, false);
631
+ if (showing === true && this._isGridInitialized) {
632
+ this.slickGrid.setColumns(this.columnDefinitions);
633
+ }
634
+ return showing;
635
+ }
636
+ //
637
+ // private functions
638
+ // ------------------
639
+ /**
640
+ * Loop through all column definitions and copy the original optional `width` properties optionally provided by the user.
641
+ * We will use this when doing a resize by cell content, if user provided a `width` it won't override it.
642
+ */
643
+ copyColumnWidthsReference(columnDefinitions) {
644
+ columnDefinitions.forEach(col => col.originalWidth = col.width);
645
+ }
646
+ displayEmptyDataWarning(showWarning = true) {
647
+ this.slickEmptyWarning?.showEmptyDataMessage(showWarning);
648
+ }
649
+ bindDifferentHooks(grid, gridOptions, dataView) {
650
+ // on locale change, we have to manually translate the Headers, GridMenu
651
+ if (this.translate?.onLangChange) {
652
+ // translate some of them on first load, then on each language change
653
+ if (gridOptions.enableTranslate) {
654
+ this.extensionService.translateAllExtensions();
655
+ this.translateColumnHeaderTitleKeys();
656
+ this.translateColumnGroupKeys();
657
+ }
658
+ this.subscriptions.push(this.translate.onLangChange.subscribe(() => {
659
+ // publish event of the same name that Slickgrid-Universal uses on a language change event
660
+ this._eventPubSubService.publish('onLanguageChange');
661
+ if (gridOptions.enableTranslate) {
662
+ this.extensionService.translateAllExtensions();
663
+ this.translateColumnHeaderTitleKeys();
664
+ this.translateColumnGroupKeys();
665
+ if (gridOptions.createPreHeaderPanel && !gridOptions.enableDraggableGrouping) {
666
+ this.groupingService.translateGroupingAndColSpan();
667
+ }
668
+ }
669
+ }));
670
+ }
671
+ // if user set an onInit Backend, we'll run it right away (and if so, we also need to run preProcess, internalPostProcess & postProcess)
672
+ if (gridOptions.backendServiceApi) {
673
+ const backendApi = gridOptions.backendServiceApi;
674
+ if (backendApi?.service?.init) {
675
+ backendApi.service.init(backendApi.options, gridOptions.pagination, this.slickGrid, this.sharedService);
676
+ }
677
+ }
678
+ if (dataView && grid) {
679
+ const slickgridEventPrefix = this.gridOptions?.defaultSlickgridEventPrefix ?? '';
680
+ // expose all Slick Grid Events through dispatch
681
+ for (const prop in grid) {
682
+ if (grid.hasOwnProperty(prop) && prop.startsWith('on')) {
683
+ const gridEventName = this._eventPubSubService.getEventNameByNamingConvention(prop, slickgridEventPrefix);
684
+ this._eventHandler.subscribe(grid[prop], (event, args) => {
685
+ return this._eventPubSubService.dispatchCustomEvent(gridEventName, { eventData: event, args });
686
+ });
687
+ }
688
+ }
689
+ // expose all Slick DataView Events through dispatch
690
+ for (const prop in dataView) {
691
+ if (dataView.hasOwnProperty(prop) && prop.startsWith('on')) {
692
+ this._eventHandler.subscribe(dataView[prop], (event, args) => {
693
+ const dataViewEventName = this._eventPubSubService.getEventNameByNamingConvention(prop, slickgridEventPrefix);
694
+ return this._eventPubSubService.dispatchCustomEvent(dataViewEventName, { eventData: event, args });
695
+ });
696
+ }
697
+ }
698
+ // on cell click, mainly used with the columnDef.action callback
699
+ this.gridEventService.bindOnCellChange(grid);
700
+ this.gridEventService.bindOnClick(grid);
701
+ if (dataView && grid) {
702
+ // bind external sorting (backend) when available or default onSort (dataView)
703
+ if (gridOptions.enableSorting) {
704
+ // bind external sorting (backend) unless specified to use the local one
705
+ if (gridOptions.backendServiceApi && !gridOptions.backendServiceApi.useLocalSorting) {
706
+ this.sortService.bindBackendOnSort(grid);
707
+ }
708
+ else {
709
+ this.sortService.bindLocalOnSort(grid);
710
+ }
711
+ }
712
+ // bind external filter (backend) when available or default onFilter (dataView)
713
+ if (gridOptions.enableFiltering) {
714
+ this.filterService.init(grid);
715
+ // bind external filter (backend) unless specified to use the local one
716
+ if (gridOptions.backendServiceApi && !gridOptions.backendServiceApi.useLocalFiltering) {
717
+ this.filterService.bindBackendOnFilter(grid);
718
+ }
719
+ else {
720
+ this.filterService.bindLocalOnFilter(grid);
721
+ }
722
+ }
723
+ // when column are reordered, we need to update the visibleColumn array
724
+ this._eventHandler.subscribe(grid.onColumnsReordered, (_e, args) => {
725
+ this.sharedService.hasColumnsReordered = true;
726
+ this.sharedService.visibleColumns = args.impactedColumns;
727
+ });
728
+ // load any presets if any (after dataset is initialized)
729
+ this.loadColumnPresetsWhenDatasetInitialized();
730
+ this.loadFilterPresetsWhenDatasetInitialized();
731
+ // When data changes in the DataView, we need to refresh the metrics and/or display a warning if the dataset is empty
732
+ this._eventHandler.subscribe(dataView.onRowCountChanged, () => {
733
+ grid.invalidate();
734
+ this.handleOnItemCountChanged(dataView.getFilteredItemCount() || 0, dataView.getItemCount() || 0);
735
+ });
736
+ this._eventHandler.subscribe(dataView.onSetItemsCalled, (_e, args) => {
737
+ this.handleOnItemCountChanged(dataView.getFilteredItemCount() || 0, args.itemCount);
738
+ // when user has resize by content enabled, we'll force a full width calculation since we change our entire dataset
739
+ if (args.itemCount > 0 && (this.gridOptions.autosizeColumnsByCellContentOnFirstLoad || this.gridOptions.enableAutoResizeColumnsByCellContent)) {
740
+ this.resizerService.resizeColumnsByCellContent(!this.gridOptions?.resizeByContentOnlyOnFirstLoad);
741
+ }
742
+ });
743
+ if (gridOptions?.enableFiltering && !gridOptions.enableRowDetailView) {
744
+ this._eventHandler.subscribe(dataView.onRowsChanged, (_e, args) => {
745
+ // filtering data with local dataset will not always show correctly unless we call this updateRow/render
746
+ // also don't use "invalidateRows" since it destroys the entire row and as bad user experience when updating a row
747
+ // see commit: https://github.com/ghiscoding/aurelia-slickgrid/commit/8c503a4d45fba11cbd8d8cc467fae8d177cc4f60
748
+ if (args?.rows && Array.isArray(args.rows)) {
749
+ args.rows.forEach((row) => grid.updateRow(row));
750
+ grid.render();
751
+ }
752
+ });
753
+ }
754
+ }
755
+ }
756
+ // did the user add a colspan callback? If so, hook it into the DataView getItemMetadata
757
+ if (gridOptions && gridOptions.colspanCallback && dataView && dataView.getItem && dataView.getItemMetadata) {
758
+ dataView.getItemMetadata = (rowNumber) => {
759
+ let callbackResult = null;
760
+ if (gridOptions.colspanCallback && gridOptions.colspanCallback) {
761
+ callbackResult = gridOptions.colspanCallback(dataView.getItem(rowNumber));
762
+ }
763
+ return callbackResult;
764
+ };
765
+ }
766
+ }
767
+ bindBackendCallbackFunctions(gridOptions) {
768
+ const backendApi = gridOptions.backendServiceApi;
769
+ const backendApiService = backendApi && backendApi.service;
770
+ const serviceOptions = backendApiService?.options ?? {};
771
+ const isExecuteCommandOnInit = (!serviceOptions) ? false : ((serviceOptions && serviceOptions.hasOwnProperty('executeProcessCommandOnInit')) ? serviceOptions['executeProcessCommandOnInit'] : true);
772
+ if (backendApiService) {
773
+ // update backend filters (if need be) BEFORE the query runs (via the onInit command a few lines below)
774
+ // if user entered some any "presets", we need to reflect them all in the grid
775
+ if (gridOptions && gridOptions.presets) {
776
+ // Filters "presets"
777
+ if (backendApiService.updateFilters && Array.isArray(gridOptions.presets.filters) && gridOptions.presets.filters.length > 0) {
778
+ backendApiService.updateFilters(gridOptions.presets.filters, true);
779
+ }
780
+ // Sorters "presets"
781
+ if (backendApiService.updateSorters && Array.isArray(gridOptions.presets.sorters) && gridOptions.presets.sorters.length > 0) {
782
+ // when using multi-column sort, we can have multiple but on single sort then only grab the first sort provided
783
+ const sortColumns = this.gridOptions.multiColumnSort ? gridOptions.presets.sorters : gridOptions.presets.sorters.slice(0, 1);
784
+ backendApiService.updateSorters(undefined, sortColumns);
785
+ }
786
+ // Pagination "presets"
787
+ if (backendApiService.updatePagination && gridOptions.presets.pagination) {
788
+ const { pageNumber, pageSize } = gridOptions.presets.pagination;
789
+ backendApiService.updatePagination(pageNumber, pageSize);
790
+ }
791
+ }
792
+ else {
793
+ const columnFilters = this.filterService.getColumnFilters();
794
+ if (columnFilters && backendApiService.updateFilters) {
795
+ backendApiService.updateFilters(columnFilters, false);
796
+ }
797
+ }
798
+ // execute onInit command when necessary
799
+ if (backendApi && backendApiService && (backendApi.onInit || isExecuteCommandOnInit)) {
800
+ const query = (typeof backendApiService.buildQuery === 'function') ? backendApiService.buildQuery() : '';
801
+ const process = (isExecuteCommandOnInit) ? (backendApi.process && backendApi.process(query) || null) : (backendApi.onInit && backendApi.onInit(query) || null);
802
+ // wrap this inside a setTimeout to avoid timing issue since the gridOptions needs to be ready before running this onInit
803
+ setTimeout(() => {
804
+ const backendUtilityService = this.backendUtilityService;
805
+ // keep start time & end timestamps & return it after process execution
806
+ const startTime = new Date();
807
+ // run any pre-process, if defined, for example a spinner
808
+ if (backendApi.preProcess) {
809
+ backendApi.preProcess();
810
+ }
811
+ // the processes can be a Promise (like Http)
812
+ const totalItems = this.gridOptions?.pagination?.totalItems ?? 0;
813
+ if (process instanceof Promise) {
814
+ process
815
+ .then((processResult) => backendUtilityService.executeBackendProcessesCallback(startTime, processResult, backendApi, totalItems))
816
+ .catch((error) => backendUtilityService.onBackendError(error, backendApi));
817
+ }
818
+ else if (process && this.rxjs?.isObservable(process)) {
819
+ this.subscriptions.push(process.subscribe({
820
+ next: (processResult) => backendUtilityService.executeBackendProcessesCallback(startTime, processResult, backendApi, totalItems),
821
+ error: (error) => backendUtilityService.onBackendError(error, backendApi)
822
+ }));
823
+ }
824
+ });
825
+ }
826
+ }
827
+ }
828
+ bindResizeHook(grid, options) {
829
+ if ((options.autoFitColumnsOnFirstLoad && options.autosizeColumnsByCellContentOnFirstLoad) || (options.enableAutoSizeColumns && options.enableAutoResizeColumnsByCellContent)) {
830
+ throw new Error(`[Angular-Slickgrid] You cannot enable both autosize/fit viewport & resize by content, you must choose which resize technique to use. You can enable these 2 options ("autoFitColumnsOnFirstLoad" and "enableAutoSizeColumns") OR these other 2 options ("autosizeColumnsByCellContentOnFirstLoad" and "enableAutoResizeColumnsByCellContent").`);
831
+ }
832
+ // expand/autofit columns on first page load
833
+ if (grid && options.autoFitColumnsOnFirstLoad && options.enableAutoSizeColumns) {
834
+ grid.autosizeColumns();
835
+ }
836
+ // auto-resize grid on browser resize
837
+ if (options.gridHeight || options.gridWidth) {
838
+ this.resizerService.resizeGrid(0, { height: options.gridHeight, width: options.gridWidth });
839
+ }
840
+ else {
841
+ this.resizerService.resizeGrid();
842
+ }
843
+ if (options.enableAutoResize) {
844
+ if (grid && options.autoFitColumnsOnFirstLoad && options.enableAutoSizeColumns) {
845
+ grid.autosizeColumns();
846
+ }
847
+ }
848
+ }
849
+ executeAfterDataviewCreated(_grid, gridOptions) {
850
+ // if user entered some Sort "presets", we need to reflect them all in the DOM
851
+ if (gridOptions.enableSorting) {
852
+ if (gridOptions.presets && Array.isArray(gridOptions.presets.sorters)) {
853
+ // when using multi-column sort, we can have multiple but on single sort then only grab the first sort provided
854
+ const sortColumns = this.gridOptions.multiColumnSort ? gridOptions.presets.sorters : gridOptions.presets.sorters.slice(0, 1);
855
+ this.sortService.loadGridSorters(sortColumns);
856
+ }
857
+ }
858
+ }
859
+ /** When data changes in the DataView, we'll refresh the metrics and/or display a warning if the dataset is empty */
860
+ handleOnItemCountChanged(currentPageRowItemCount, totalItemCount) {
861
+ this._currentDatasetLength = totalItemCount;
862
+ this.metrics = {
863
+ startTime: new Date(),
864
+ endTime: new Date(),
865
+ itemCount: currentPageRowItemCount,
866
+ totalItemCount
867
+ };
868
+ // if custom footer is enabled, then we'll update its metrics
869
+ if (this.slickFooter) {
870
+ this.slickFooter.metrics = this.metrics;
871
+ }
872
+ // when using local (in-memory) dataset, we'll display a warning message when filtered data is empty
873
+ if (this._isLocalGrid && this.gridOptions?.enableEmptyDataWarningMessage) {
874
+ this.displayEmptyDataWarning(currentPageRowItemCount === 0);
875
+ }
876
+ }
877
+ initializePaginationService(paginationOptions) {
878
+ if (this.gridOptions) {
879
+ this.paginationData = {
880
+ gridOptions: this.gridOptions,
881
+ paginationService: this.paginationService,
882
+ };
883
+ this.paginationService.totalItems = this.totalItems;
884
+ this.paginationService.init(this.slickGrid, paginationOptions, this.backendServiceApi);
885
+ this.subscriptions.push(this._eventPubSubService.subscribe('onPaginationChanged', (paginationChanges) => {
886
+ this.paginationChanged(paginationChanges);
887
+ }), this._eventPubSubService.subscribe('onPaginationVisibilityChanged', (visibility) => {
888
+ this.showPagination = visibility?.visible ?? false;
889
+ if (this.gridOptions?.backendServiceApi) {
890
+ this.backendUtilityService?.refreshBackendDataset(this.gridOptions);
891
+ }
892
+ this.renderPagination(this.showPagination);
893
+ }));
894
+ // also initialize (render) the pagination component
895
+ this.renderPagination();
896
+ this._isPaginationInitialized = true;
897
+ }
898
+ this.cd.detectChanges();
899
+ }
900
+ /** Load the Editor Collection asynchronously and replace the "collection" property when Observable resolves */
901
+ loadEditorCollectionAsync(column) {
902
+ const collectionAsync = column && column.editor && column.editor.collectionAsync;
903
+ if (collectionAsync instanceof Observable) {
904
+ this.subscriptions.push(collectionAsync.subscribe((resolvedCollection) => this.updateEditorCollection(column, resolvedCollection)));
905
+ }
906
+ else if (collectionAsync instanceof Promise) {
907
+ // wait for the "collectionAsync", once resolved we will save it into the "collection"
908
+ // the collectionAsync can be of 3 types HttpClient, HttpFetch or a Promise
909
+ collectionAsync.then((response) => {
910
+ if (Array.isArray(response)) {
911
+ this.updateEditorCollection(column, response); // from Promise
912
+ }
913
+ });
914
+ }
915
+ }
916
+ insertDynamicPresetColumns(columnId, gridPresetColumns) {
917
+ if (this._columnDefinitions) {
918
+ const columnPosition = this._columnDefinitions.findIndex(c => c.id === columnId);
919
+ if (columnPosition >= 0) {
920
+ const dynColumn = this._columnDefinitions[columnPosition];
921
+ if (dynColumn?.id === columnId && !gridPresetColumns.some(c => c.id === columnId)) {
922
+ columnPosition > 0
923
+ ? gridPresetColumns.splice(columnPosition, 0, dynColumn)
924
+ : gridPresetColumns.unshift(dynColumn);
925
+ }
926
+ }
927
+ }
928
+ }
929
+ /** Load any possible Columns Grid Presets */
930
+ loadColumnPresetsWhenDatasetInitialized() {
931
+ // if user entered some Columns "presets", we need to reflect them all in the grid
932
+ if (this.gridOptions.presets && Array.isArray(this.gridOptions.presets.columns) && this.gridOptions.presets.columns.length > 0) {
933
+ const gridPresetColumns = this.gridStateService.getAssociatedGridColumns(this.slickGrid, this.gridOptions.presets.columns);
934
+ if (gridPresetColumns && Array.isArray(gridPresetColumns) && gridPresetColumns.length > 0 && Array.isArray(this._columnDefinitions)) {
935
+ // make sure that the dynamic columns are included in presets (1.Row Move, 2. Row Selection, 3. Row Detail)
936
+ if (this.gridOptions.enableRowMoveManager) {
937
+ const rmmColId = this.gridOptions?.rowMoveManager?.columnId ?? '_move';
938
+ this.insertDynamicPresetColumns(rmmColId, gridPresetColumns);
939
+ }
940
+ if (this.gridOptions.enableCheckboxSelector) {
941
+ const chkColId = this.gridOptions?.checkboxSelector?.columnId ?? '_checkbox_selector';
942
+ this.insertDynamicPresetColumns(chkColId, gridPresetColumns);
943
+ }
944
+ if (this.gridOptions.enableRowDetailView) {
945
+ const rdvColId = this.gridOptions?.rowDetailView?.columnId ?? '_detail_selector';
946
+ this.insertDynamicPresetColumns(rdvColId, gridPresetColumns);
947
+ }
948
+ // keep copy the original optional `width` properties optionally provided by the user.
949
+ // We will use this when doing a resize by cell content, if user provided a `width` it won't override it.
950
+ gridPresetColumns.forEach(col => col.originalWidth = col.width);
951
+ // finally set the new presets columns (including checkbox selector if need be)
952
+ this.slickGrid.setColumns(gridPresetColumns);
953
+ this.sharedService.visibleColumns = gridPresetColumns;
954
+ }
955
+ }
956
+ }
957
+ /** Load any possible Filters Grid Presets */
958
+ loadFilterPresetsWhenDatasetInitialized() {
959
+ if (this.gridOptions && !this.customDataView) {
960
+ // if user entered some Filter "presets", we need to reflect them all in the DOM
961
+ // also note that a presets of Tree Data Toggling will also call this method because Tree Data toggling does work with data filtering
962
+ // (collapsing a parent will basically use Filter for hidding (aka collapsing) away the child underneat it)
963
+ if (this.gridOptions.presets && (Array.isArray(this.gridOptions.presets.filters) || Array.isArray(this.gridOptions.presets?.treeData?.toggledItems))) {
964
+ this.filterService.populateColumnFilterSearchTermPresets(this.gridOptions.presets?.filters || []);
965
+ }
966
+ }
967
+ }
968
+ /**
969
+ * local grid, check if we need to show the Pagination
970
+ * if so then also check if there's any presets and finally initialize the PaginationService
971
+ * a local grid with Pagination presets will potentially have a different total of items, we'll need to get it from the DataView and update our total
972
+ */
973
+ loadLocalGridPagination(dataset) {
974
+ if (this.gridOptions && this._paginationOptions) {
975
+ this.totalItems = Array.isArray(dataset) ? dataset.length : 0;
976
+ if (this._paginationOptions && this.dataView?.getPagingInfo) {
977
+ const slickPagingInfo = this.dataView.getPagingInfo();
978
+ if (slickPagingInfo?.hasOwnProperty('totalRows') && this._paginationOptions.totalItems !== slickPagingInfo.totalRows) {
979
+ this.totalItems = slickPagingInfo.totalRows || 0;
980
+ }
981
+ }
982
+ this._paginationOptions.totalItems = this.totalItems;
983
+ const paginationOptions = this.setPaginationOptionsWhenPresetDefined(this.gridOptions, this._paginationOptions);
984
+ this.initializePaginationService(paginationOptions);
985
+ }
986
+ }
987
+ /** Load any Row Selections into the DataView that were presets by the user */
988
+ loadRowSelectionPresetWhenExists() {
989
+ // if user entered some Row Selections "presets"
990
+ const presets = this.gridOptions?.presets;
991
+ const enableRowSelection = this.gridOptions && (this.gridOptions.enableCheckboxSelector || this.gridOptions.enableRowSelection);
992
+ if (enableRowSelection && this.slickGrid?.getSelectionModel() && presets?.rowSelection && (Array.isArray(presets.rowSelection.gridRowIndexes) || Array.isArray(presets.rowSelection.dataContextIds))) {
993
+ let dataContextIds = presets.rowSelection.dataContextIds;
994
+ let gridRowIndexes = presets.rowSelection.gridRowIndexes;
995
+ // maps the IDs to the Grid Rows and vice versa, the "dataContextIds" has precedence over the other
996
+ if (Array.isArray(dataContextIds) && dataContextIds.length > 0) {
997
+ gridRowIndexes = this.dataView.mapIdsToRows(dataContextIds) || [];
998
+ }
999
+ else if (Array.isArray(gridRowIndexes) && gridRowIndexes.length > 0) {
1000
+ dataContextIds = this.dataView.mapRowsToIds(gridRowIndexes) || [];
1001
+ }
1002
+ // apply row selection when defined as grid presets
1003
+ if (this.slickGrid && Array.isArray(gridRowIndexes)) {
1004
+ this.slickGrid.setSelectedRows(gridRowIndexes);
1005
+ this.dataView.setSelectedIds(dataContextIds || [], {
1006
+ isRowBeingAdded: true,
1007
+ shouldTriggerEvent: false,
1008
+ applyRowSelectionToGrid: true
1009
+ });
1010
+ }
1011
+ }
1012
+ }
1013
+ mergeGridOptions(gridOptions) {
1014
+ gridOptions.gridId = this.gridId;
1015
+ gridOptions.gridContainerId = `slickGridContainer-${this.gridId}`;
1016
+ // if we have a backendServiceApi and the enablePagination is undefined, we'll assume that we do want to see it, else get that defined value
1017
+ gridOptions.enablePagination = ((gridOptions.backendServiceApi && gridOptions.enablePagination === undefined) ? true : gridOptions.enablePagination) || false;
1018
+ // use extend to deep merge & copy to avoid immutable properties being changed in GlobalGridOptions after a route change
1019
+ const options = Slick.Utils.extend(true, {}, GlobalGridOptions, this.forRootConfig, gridOptions);
1020
+ // using copy extend to do a deep clone has an unwanted side on objects and pageSizes but ES6 spread has other worst side effects
1021
+ // so we will just overwrite the pageSizes when needed, this is the only one causing issues so far.
1022
+ // On a deep extend, Object and Array are extended, but object wrappers on primitive types such as String, Boolean, and Number are not.
1023
+ if (options?.pagination && (gridOptions.enablePagination || gridOptions.backendServiceApi) && (this.forRootConfig.pagination || gridOptions.pagination)) {
1024
+ options.pagination.pageSize = gridOptions.pagination?.pageSize ?? this.forRootConfig.pagination?.pageSize ?? GlobalGridOptions.pagination.pageSize;
1025
+ options.pagination.pageSizes = gridOptions.pagination?.pageSizes ?? this.forRootConfig.pagination?.pageSizes ?? GlobalGridOptions.pagination.pageSizes;
1026
+ }
1027
+ // also make sure to show the header row if user have enabled filtering
1028
+ this._hideHeaderRowAfterPageLoad = (options.showHeaderRow === false);
1029
+ if (options.enableFiltering && !options.showHeaderRow) {
1030
+ options.showHeaderRow = options.enableFiltering;
1031
+ }
1032
+ // when we use Pagination on Local Grid, it doesn't seem to work without enableFiltering
1033
+ // so we'll enable the filtering but we'll keep the header row hidden
1034
+ if (options && !options.enableFiltering && options.enablePagination && this._isLocalGrid) {
1035
+ options.enableFiltering = true;
1036
+ options.showHeaderRow = false;
1037
+ this._hideHeaderRowAfterPageLoad = true;
1038
+ if (this.sharedService) {
1039
+ this.sharedService.hideHeaderRowAfterPageLoad = true;
1040
+ }
1041
+ }
1042
+ return options;
1043
+ }
1044
+ /** Pre-Register any Resource that don't require SlickGrid to be instantiated (for example RxJS Resource & RowDetail) */
1045
+ preRegisterResources() {
1046
+ this._registeredResources = this.gridOptions.registerExternalResources || [];
1047
+ // Angular-Slickgrid requires RxJS, so we'll register it as the first resource
1048
+ this.registerRxJsResource(new RxJsResource());
1049
+ if (this.gridOptions.enableRowDetailView) {
1050
+ this.slickRowDetailView = new SlickRowDetailView(this.angularUtilService, this.appRef, this._eventPubSubService, this.elm.nativeElement, this.rxjs);
1051
+ this.slickRowDetailView.create(this.columnDefinitions, this.gridOptions);
1052
+ this._registeredResources.push(this.slickRowDetailView);
1053
+ this.extensionService.addExtensionToList(ExtensionName.rowDetailView, { name: ExtensionName.rowDetailView, instance: this.slickRowDetailView });
1054
+ }
1055
+ }
1056
+ registerResources() {
1057
+ // at this point, we consider all the registered services as external services, anything else registered afterward aren't external
1058
+ if (Array.isArray(this._registeredResources)) {
1059
+ this.sharedService.externalRegisteredResources = this._registeredResources;
1060
+ }
1061
+ // push all other Services that we want to be registered
1062
+ this._registeredResources.push(this.gridService, this.gridStateService);
1063
+ // when using Grouping/DraggableGrouping/Colspan register its Service
1064
+ if (this.gridOptions.createPreHeaderPanel && !this.gridOptions.enableDraggableGrouping) {
1065
+ this._registeredResources.push(this.groupingService);
1066
+ }
1067
+ // when using Tree Data View, register its Service
1068
+ if (this.gridOptions.enableTreeData) {
1069
+ this._registeredResources.push(this.treeDataService);
1070
+ }
1071
+ // when user enables translation, we need to translate Headers on first pass & subsequently in the bindDifferentHooks
1072
+ if (this.gridOptions.enableTranslate) {
1073
+ this.extensionService.translateColumnHeaders();
1074
+ }
1075
+ // also initialize (render) the empty warning component
1076
+ this.slickEmptyWarning = new SlickEmptyWarningComponent();
1077
+ this._registeredResources.push(this.slickEmptyWarning);
1078
+ // bind & initialize all Components/Services that were tagged as enabled
1079
+ // register all services by executing their init method and providing them with the Grid object
1080
+ if (Array.isArray(this._registeredResources)) {
1081
+ for (const resource of this._registeredResources) {
1082
+ if (this.slickGrid && typeof resource.init === 'function') {
1083
+ resource.init(this.slickGrid, this.containerService);
1084
+ }
1085
+ }
1086
+ }
1087
+ }
1088
+ /** Register the RxJS Resource in all necessary services which uses */
1089
+ registerRxJsResource(resource) {
1090
+ this.rxjs = resource;
1091
+ this.backendUtilityService.addRxJsResource(this.rxjs);
1092
+ this.filterFactory.addRxJsResource(this.rxjs);
1093
+ this.filterService.addRxJsResource(this.rxjs);
1094
+ this.sortService.addRxJsResource(this.rxjs);
1095
+ this.paginationService.addRxJsResource(this.rxjs);
1096
+ this.containerService.registerInstance('RxJsResource', this.rxjs);
1097
+ }
1098
+ /**
1099
+ * Render (or dispose) the Pagination Component, user can optionally provide False (to not show it) which will in term dispose of the Pagination,
1100
+ * also while disposing we can choose to omit the disposable of the Pagination Service (if we are simply toggling the Pagination, we want to keep the Service alive)
1101
+ * @param {Boolean} showPagination - show (new render) or not (dispose) the Pagination
1102
+ * @param {Boolean} shouldDisposePaginationService - when disposing the Pagination, do we also want to dispose of the Pagination Service? (defaults to True)
1103
+ */
1104
+ renderPagination(showPagination = true) {
1105
+ if (this.gridOptions?.enablePagination && !this._isPaginationInitialized && showPagination) {
1106
+ this.slickPagination = new SlickPaginationComponent(this.paginationService, this._eventPubSubService, this.sharedService, this.translaterService);
1107
+ this.slickPagination.renderPagination(this.gridContainerElement);
1108
+ this._isPaginationInitialized = true;
1109
+ }
1110
+ else if (!showPagination) {
1111
+ if (this.slickPagination) {
1112
+ this.slickPagination.dispose();
1113
+ }
1114
+ this._isPaginationInitialized = false;
1115
+ }
1116
+ }
1117
+ /**
1118
+ * Takes a flat dataset with parent/child relationship, sort it (via its tree structure) and return the sorted flat array
1119
+ * @param {Array<Object>} flatDatasetInput - flat dataset input
1120
+ * @param {Boolean} forceGridRefresh - optionally force a full grid refresh
1121
+ * @returns {Array<Object>} sort flat parent/child dataset
1122
+ */
1123
+ sortTreeDataset(flatDatasetInput, forceGridRefresh = false) {
1124
+ const prevDatasetLn = this._currentDatasetLength;
1125
+ let sortedDatasetResult;
1126
+ let flatDatasetOutput = [];
1127
+ // if the hierarchical dataset was already initialized then no need to re-convert it, we can use it directly from the shared service ref
1128
+ if (this._isDatasetHierarchicalInitialized && this.datasetHierarchical) {
1129
+ sortedDatasetResult = this.treeDataService.sortHierarchicalDataset(this.datasetHierarchical);
1130
+ flatDatasetOutput = sortedDatasetResult.flat;
1131
+ }
1132
+ else if (Array.isArray(flatDatasetInput) && flatDatasetInput.length > 0) {
1133
+ if (this.gridOptions?.treeDataOptions?.initialSort) {
1134
+ // else we need to first convert the flat dataset to a hierarchical dataset and then sort
1135
+ sortedDatasetResult = this.treeDataService.convertFlatParentChildToTreeDatasetAndSort(flatDatasetInput, this._columnDefinitions, this.gridOptions);
1136
+ this.sharedService.hierarchicalDataset = sortedDatasetResult.hierarchical;
1137
+ flatDatasetOutput = sortedDatasetResult.flat;
1138
+ }
1139
+ else {
1140
+ // else we assume that the user provided an array that is already sorted (user's responsability)
1141
+ // and so we can simply convert the array to a tree structure and we're done, no need to sort
1142
+ this.sharedService.hierarchicalDataset = this.treeDataService.convertFlatParentChildToTreeDataset(flatDatasetInput, this.gridOptions);
1143
+ flatDatasetOutput = flatDatasetInput || [];
1144
+ }
1145
+ }
1146
+ // if we add/remove item(s) from the dataset, we need to also refresh our tree data filters
1147
+ if (flatDatasetInput.length > 0 && (forceGridRefresh || flatDatasetInput.length !== prevDatasetLn)) {
1148
+ this.filterService.refreshTreeDataFilters(flatDatasetOutput);
1149
+ }
1150
+ return flatDatasetOutput;
1151
+ }
1152
+ /**
1153
+ * For convenience to the user, we provide the property "editor" as an Angular-Slickgrid editor complex object
1154
+ * however "editor" is used internally by SlickGrid for it's own Editor Factory
1155
+ * so in our lib we will swap "editor" and copy it into a new property called "internalColumnEditor"
1156
+ * then take back "editor.model" and make it the new "editor" so that SlickGrid Editor Factory still works
1157
+ */
1158
+ swapInternalEditorToSlickGridFactoryEditor(columnDefinitions) {
1159
+ if (columnDefinitions.some(col => `${col.id}`.includes('.'))) {
1160
+ console.error('[Angular-Slickgrid] Make sure that none of your Column Definition "id" property includes a dot in its name because that will cause some problems with the Editors. For example if your column definition "field" property is "user.firstName" then use "firstName" as the column "id".');
1161
+ }
1162
+ return columnDefinitions.map((column) => {
1163
+ // on every Editor that have a "collectionAsync", resolve the data and assign it to the "collection" property
1164
+ if (column?.editor?.collectionAsync) {
1165
+ this.loadEditorCollectionAsync(column);
1166
+ }
1167
+ return { ...column, editor: column.editor?.model, internalColumnEditor: { ...column.editor } };
1168
+ });
1169
+ }
1170
+ translateColumnHeaderTitleKeys() {
1171
+ // translate all columns (including hidden columns)
1172
+ this.extensionUtility.translateItems(this.sharedService.allColumns, 'nameKey', 'name');
1173
+ }
1174
+ translateColumnGroupKeys() {
1175
+ // translate all column groups (including hidden columns)
1176
+ this.extensionUtility.translateItems(this.sharedService.allColumns, 'columnGroupKey', 'columnGroup');
1177
+ }
1178
+ /**
1179
+ * Update the "internalColumnEditor.collection" property.
1180
+ * Since this is called after the async call resolves, the pointer will not be the same as the "column" argument passed.
1181
+ * Once we found the new pointer, we will reassign the "editor" and "collection" to the "internalColumnEditor" so it has newest collection
1182
+ */
1183
+ updateEditorCollection(column, newCollection) {
1184
+ column.editor.collection = newCollection;
1185
+ column.editor.disabled = false;
1186
+ // find the new column reference pointer & re-assign the new editor to the internalColumnEditor
1187
+ if (Array.isArray(this.columnDefinitions)) {
1188
+ const columnRef = this.columnDefinitions.find((col) => col.id === column.id);
1189
+ if (columnRef) {
1190
+ columnRef.internalColumnEditor = column.editor;
1191
+ }
1192
+ }
1193
+ // get current Editor, remove it from the DOM then re-enable it and re-render it with the new collection.
1194
+ const currentEditor = this.slickGrid.getCellEditor();
1195
+ if (currentEditor?.disable && currentEditor?.renderDomElement) {
1196
+ currentEditor.destroy();
1197
+ currentEditor.disable(false);
1198
+ currentEditor.renderDomElement(newCollection);
1199
+ }
1200
+ }
1201
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.3", ngImport: i0, type: AngularSlickgridComponent, deps: [{ token: i1.AngularUtilService }, { token: i0.ApplicationRef }, { token: i0.ChangeDetectorRef }, { token: i2.ContainerService }, { token: i0.ElementRef }, { token: i3.TranslateService, optional: true }, { token: i4.TranslaterService, optional: true }, { token: 'config' }, { token: 'externalService' }], target: i0.ɵɵFactoryTarget.Component });
1202
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.3", type: AngularSlickgridComponent, selector: "angular-slickgrid", inputs: { customDataView: "customDataView", gridId: "gridId", gridOptions: "gridOptions", paginationOptions: "paginationOptions", columnDefinitions: "columnDefinitions", dataset: "dataset", datasetHierarchical: "datasetHierarchical" }, outputs: { columnDefinitionsChange: "columnDefinitionsChange" }, providers: [
1203
+ // make everything transient (non-singleton)
1204
+ AngularUtilService,
1205
+ ApplicationRef,
1206
+ TranslaterService,
1207
+ ], ngImport: i0, template: "<div id=\"slickGridContainer-{{gridId}}\" class=\"gridPane\">\n <div attr.id='{{gridId}}' class=\"slickgrid-container\" style=\"width: 100%\">\n </div>\n</div>" });
1208
+ }
1209
+ export { AngularSlickgridComponent };
1210
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.3", ngImport: i0, type: AngularSlickgridComponent, decorators: [{
1211
+ type: Component,
1212
+ args: [{ selector: 'angular-slickgrid', providers: [
1213
+ // make everything transient (non-singleton)
1214
+ AngularUtilService,
1215
+ ApplicationRef,
1216
+ TranslaterService,
1217
+ ], template: "<div id=\"slickGridContainer-{{gridId}}\" class=\"gridPane\">\n <div attr.id='{{gridId}}' class=\"slickgrid-container\" style=\"width: 100%\">\n </div>\n</div>" }]
1218
+ }], ctorParameters: function () { return [{ type: i1.AngularUtilService }, { type: i0.ApplicationRef }, { type: i0.ChangeDetectorRef }, { type: i2.ContainerService }, { type: i0.ElementRef }, { type: i3.TranslateService, decorators: [{
1219
+ type: Optional
1220
+ }] }, { type: i4.TranslaterService, decorators: [{
1221
+ type: Optional
1222
+ }] }, { type: undefined, decorators: [{
1223
+ type: Inject,
1224
+ args: ['config']
1225
+ }] }, { type: undefined, decorators: [{
1226
+ type: Inject,
1227
+ args: ['externalService']
1228
+ }] }]; }, propDecorators: { customDataView: [{
1229
+ type: Input
1230
+ }], gridId: [{
1231
+ type: Input
1232
+ }], gridOptions: [{
1233
+ type: Input
1234
+ }], paginationOptions: [{
1235
+ type: Input
1236
+ }], columnDefinitions: [{
1237
+ type: Input
1238
+ }], columnDefinitionsChange: [{
1239
+ type: Output
1240
+ }], dataset: [{
1241
+ type: Input
1242
+ }], datasetHierarchical: [{
1243
+ type: Input
1244
+ }] } });
1245
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5ndWxhci1zbGlja2dyaWQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9tb2R1bGVzL2FuZ3VsYXItc2xpY2tncmlkL2NvbXBvbmVudHMvYW5ndWxhci1zbGlja2dyaWQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2FwcC9tb2R1bGVzL2FuZ3VsYXItc2xpY2tncmlkL2NvbXBvbmVudHMvYW5ndWxhci1zbGlja2dyaWQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsK0JBQStCO0FBQy9CLE9BQU8sc0JBQXNCLENBQUM7QUFDOUIsT0FBTyw4QkFBOEIsQ0FBQztBQUN0QyxPQUFPLHNCQUFzQixDQUFDO0FBQzlCLE9BQU8sMEJBQTBCLENBQUM7QUFFbEMsNkJBQTZCO0FBQzdCLE9BQU8sRUFBaUIsY0FBYyxFQUFxQixTQUFTLEVBQWMsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQWEsUUFBUSxFQUFFLE1BQU0sR0FBRyxNQUFNLGVBQWUsQ0FBQztBQUVuSyxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRWxDLE9BQU8sRUFTTCxhQUFhO0FBWWIsV0FBVztBQUNYLHFCQUFxQixFQUNyQixpQkFBaUIsRUFDakIsZ0JBQWdCLEVBQ2hCLGdCQUFnQixFQUNoQixnQkFBZ0IsRUFDaEIsYUFBYSxFQUNiLGFBQWEsRUFDYixnQkFBZ0IsRUFDaEIsV0FBVyxFQUNYLGdCQUFnQixFQUNoQix5QkFBeUIsRUFDekIsaUJBQWlCLEVBQ2pCLGNBQWMsRUFFZCxhQUFhLEVBQ2IsZUFBZSxFQUNmLDhCQUE4QixFQUM5QixXQUFXLEVBQ1gsZUFBZTtBQUVmLFlBQVk7QUFDWix5Q0FBeUMsRUFDekMsWUFBWSxFQUNaLGFBQWEsRUFDYixjQUFjLEdBQ2YsTUFBTSw2QkFBNkIsQ0FBQztBQUNyQyxPQUFPLEVBQUUsa0JBQWtCLEVB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dGVudCI6WyIvLyBpbXBvcnQgM3JkIHBhcnR5IHZlbmRvciBsaWJzXG5pbXBvcnQgJ3NsaWNrZ3JpZC9zbGljay5jb3JlJztcbmltcG9ydCAnc2xpY2tncmlkL3NsaWNrLmludGVyYWN0aW9ucyc7XG5pbXBvcnQgJ3NsaWNrZ3JpZC9zbGljay5ncmlkJztcbmltcG9ydCAnc2xpY2tncmlkL3NsaWNrLmRhdGF2aWV3JztcblxuLy8gLi4udGhlbiBldmVyeXRoaW5nIGVsc2UuLi5cbmltcG9ydCB7IEFmdGVyVmlld0luaXQsIEFwcGxpY2F0aW9uUmVmLCBDaGFuZ2VEZXRlY3RvclJlZiwgQ29tcG9uZW50LCBFbGVtZW50UmVmLCBFdmVudEVtaXR0ZXIsIEluamVjdCwgSW5wdXQsIE9uRGVzdHJveSwgT3B0aW9uYWwsIE91dHB1dCwgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFRyYW5zbGF0ZVNlcnZpY2UgfSBmcm9tICdAbmd4LXRyYW5zbGF0ZS9jb3JlJztcbmltcG9ydCB7IE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcblxuaW1wb3J0IHtcbiAgLy8gaW50ZXJmYWNlcy90eXBlc1xuICBBdXRvY29tcGxldGVyRWRpdG9yLFxuICBCYWNrZW5kU2VydmljZUFwaSxcbiAgQmFja2VuZFNlcnZpY2VPcHRpb24sXG4gIENvbHVtbixcbiAgQ29sdW1uRWRpdG9yLFxuICBEYXRhVmlld09wdGlvbixcbiAgRXZlbnRTdWJzY3JpcHRpb24sXG4gIEV4dGVuc2lvbk5hbWUsXG4gIEV4dGVybmFsUmVzb3VyY2UsXG4gIExvY2FsZSxcbiAgTWV0cmljcyxcbiAgUGFnaW5hdGlvbixcbiAgU2VsZWN0RWRpdG9yLFxuICBTZXJ2aWNlUGFnaW5hdGlvbixcbiAgU2xpY2tEYXRhVmlldyxcbiAgU2xpY2tFdmVudEhhbmRsZXIsXG4gIFNsaWNrR3JpZCxcbiAgU2xpY2tOYW1lc3BhY2UsXG5cbiAgLy8gc2VydmljZXNcbiAgQmFja2VuZFV0aWxpdHlTZXJ2aWNlLFxuICBDb2xsZWN0aW9uU2VydmljZSxcbiAgRXZlbnROYW1pbmdTdHlsZSxcbiAgRXh0ZW5zaW9uU2VydmljZSxcbiAgRXh0ZW5zaW9uVXRpbGl0eSxcbiAgRmlsdGVyRmFjdG9yeSxcbiAgRmlsdGVyU2VydmljZSxcbiAgR3JpZEV2ZW50U2VydmljZSxcbiAgR3JpZFNlcnZpY2UsXG4gIEdyaWRTdGF0ZVNlcnZpY2UsXG4gIEdyb3VwaW5nQW5kQ29sc3BhblNlcnZpY2UsXG4gIFBhZ2luYXRpb25TZXJ2aWNlLFxuICBSZXNpemVyU2VydmljZSxcbiAgUnhKc0ZhY2FkZSxcbiAgU2hhcmVkU2VydmljZSxcbiAgU2xpY2tncmlkQ29uZmlnLFxuICBTbGlja0dyb3VwSXRlbU1ldGFkYXRhUHJvdmlkZXIsXG4gIFNvcnRTZXJ2aWNlLFxuICBUcmVlRGF0YVNlcnZpY2UsXG5cbiAgLy8gdXRpbGl0aWVzXG4gIGF1dG9BZGRFZGl0b3JGb3JtYXR0ZXJUb0NvbHVtbnNXaXRoRWRpdG9yLFxuICBlbXB0eUVsZW1lbnQsXG4gIEdyaWRTdGF0ZVR5cGUsXG4gIHVuc3Vic2NyaWJlQWxsLFxufSBmcm9tICdAc2xpY2tncmlkLXVuaXZlcnNhbC9jb21tb24nO1xuaW1wb3J0IHsgRXZlbnRQdWJTdWJTZXJ2aWNlIH0gZnJvbSAnQHNsaWNrZ3JpZC11bml2ZXJzYWwvZXZlbnQtcHViLXN1Yic7XG5pbXBvcnQgeyBTbGlja0VtcHR5V2FybmluZ0NvbXBvbmVudCB9IGZyb20gJ0BzbGlja2dyaWQtdW5pdmVyc2FsL2VtcHR5LXdhcm5pbmctY29tcG9uZW50JztcbmltcG9ydCB7IFNsaWNrRm9vdGVyQ29tcG9uZW50IH0gZnJvbSAnQHNsaWNrZ3JpZC11bml2ZXJzYWwvY3VzdG9tLWZvb3Rlci1jb21wb25lbnQnO1xuaW1wb3J0IHsgU2xpY2tQYWdpbmF0aW9uQ29tcG9uZW50IH0gZnJvbSAnQHNsaWNrZ3JpZC11bml2ZXJzYWwvcGFnaW5hdGlvbi1jb21wb25lbnQnO1xuaW1wb3J0IHsgUnhKc1Jlc291cmNlIH0gZnJvbSAnQHNsaWNrZ3JpZC11bml2ZXJzYWwvcnhqcy1vYnNlcnZhYmxlJztcbmltcG9ydCB7IGRlcXVhbCB9IGZyb20gJ2RlcXVhbC9saXRlJztcblxuaW1wb3J0IHsgQ29uc3RhbnRzIH0gZnJvbSAnLi4vY29uc3RhbnRzJztcbmltcG9ydCB7IEFuZ3VsYXJHcmlkSW5zdGFuY2UsIEV4dGVybmFsVGVzdGluZ0RlcGVuZGVuY2llcywgR3JpZE9wdGlvbiwgfSBmcm9tICcuLy4uL21vZGVscy9pbmRleCc7XG5pbXBvcnQgeyBHbG9iYWxHcmlkT3B0aW9ucyB9IGZyb20gJy4vLi4vZ2xvYmFsLWdyaWQtb3B0aW9ucyc7XG5pbXBvcnQgeyBUcmFuc2xhdGVyU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL3RyYW5zbGF0ZXIuc2VydmljZSc7XG5cbi8vIFNlcnZpY2VzXG5pbXBvcnQgeyBBbmd1bGFyVXRpbFNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9hbmd1bGFyVXRpbC5zZXJ2aWNlJztcbmltcG9ydCB7IFNsaWNrUm93RGV0YWlsVmlldyB9IGZyb20gJy4uL2V4dGVuc2lvbnMvc2xpY2tSb3dEZXRhaWxWaWV3JztcbmltcG9ydCB7IENvbnRhaW5lclNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9jb250YWluZXIuc2VydmljZSc7XG5cbi8vIHVzaW5nIGV4dGVybmFsIG5vbi10eXBlZCBqcyBsaWJyYXJpZXNcbmRlY2xhcmUgY29uc3QgU2xpY2s6IFNsaWNrTmFtZXNwYWNlO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdhbmd1bGFyLXNsaWNrZ3JpZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9hbmd1bGFyLXNsaWNrZ3JpZC5jb21wb25lbnQuaHRtbCcsXG4gIHByb3ZpZGVyczogW1xuICAgIC8vIG1ha2UgZXZlcnl0aGluZyB0cmFuc2llbnQgKG5vbi1zaW5nbGV0b24pXG4gICAgQW5ndWxhclV0aWxTZXJ2aWNlLFxuICAgIEFwcGxpY2F0aW9uUmVmLFxuICAgIFRyYW5zbGF0ZXJTZXJ2aWNlLFxuICBdXG59KVxuZXhwb3J0IGNsYXNzIEFuZ3VsYXJTbGlja2dyaWRDb21wb25lbnQgaW1wbGVtZW50cyBBZnRlclZpZXdJbml0LCBPbkRlc3Ryb3kge1xuICBwcml2YXRlIF9kYXRhc2V0PzogYW55W10gfCBudWxsO1xuICBwcml2YXRlIF9jb2x1bW5EZWZpbml0aW9ucyE6IENvbHVtbltdO1xuICBwcml2YXRlIF9jdXJyZW50RGF0YXNldExlbmd0aCA9IDA7XG4gIHByaXZhdGUgX2V2ZW50SGFuZGxlcjogU2xpY2tFdmVudEhhbmRsZXIgPSBuZXcgU2xpY2suRXZlbnRIYW5kbGVyKCk7XG4gIHByaXZhdGUgX2V2ZW50UHViU3ViU2VydmljZSE6IEV2ZW50UHViU3ViU2VydmljZTtcbiAgcHJpdmF0ZSBfYW5ndWxhckdyaWRJbnN0YW5jZXM6IEFuZ3VsYXJHcmlkSW5zdGFuY2UgfCB1bmRlZmluZWQ7XG4gIHByaXZhdGUgX2hpZGVIZWFkZXJSb3dBZnRlclBhZ2VMb2FkID0gZmFsc2U7XG4gIHByaXZhdGUgX2lzR3JpZEluaXRpYWxpemVkID0gZmFsc2U7XG4gIHByaXZhdGUgX2lzRGF0YXNldEluaXRpYWxpemVkID0gZmFsc2U7XG4gIHByaXZhdGUgX2lzRGF0YXNldEhpZXJhcmNoaWNhbEluaXRpYWxpemVkID0gZmFsc2U7XG4gIHByaXZhdGUgX2lzUGFnaW5hdGlvbkluaXRpYWxpemVkID0gZmFsc2U7XG4gIHByaXZhdGUgX2lzTG9jYWxHcmlkID0gdHJ1ZTtcbiAgcHJpdmF0ZSBfcGFnaW5hdGlvbk9wdGlvbnM6IFBhZ2luYXRpb24gfCB1bmRlZmluZWQ7XG4gIHByaXZhdGUgX3JlZ2lzdGVyZWRSZXNvdXJjZXM6IEV4dGVybmFsUmVzb3VyY2VbXSA9IFtdO1xuICBkYXRhVmlldyE6IFNsaWNrRGF0YVZpZXc7XG4gIHNsaWNrR3JpZCE6IFNsaWNrR3JpZDtcbiAgZ3JvdXBpbmdEZWZpbml0aW9uOiBhbnkgPSB7fTtcbiAgZ3JvdXBJdGVtTWV0YWRhdGFQcm92aWRlcj86IFNsaWNrR3JvdXBJdGVtTWV0YWRhdGFQcm92aWRlcjtcbiAgYmFja2VuZFNlcnZpY2VBcGk/OiBCYWNrZW5kU2VydmljZUFwaTtcbiAgbG9jYWxlcyE6IExvY2FsZTtcbiAgbWV0cmljcz86IE1ldHJpY3M7XG4gIHNob3dQYWdpbmF0aW9uID0gZmFsc2U7XG4gIHNlcnZpY2VMaXN0OiBhbnlbXSA9IFtdO1xuICB0b3RhbEl0ZW1zID0gMDtcbiAgcGFnaW5hdGlvbkRhdGE/OiB7XG4gICAgZ3JpZE9wdGlvbnM6IEdyaWRPcHRpb247XG4gICAgcGFnaW5hdGlvblNlcnZpY2U6IFBhZ2luYXRpb25TZXJ2aWNlO1xuICB9O1xuICBzdWJzY3JpcHRpb25zOiBFdmVudFN1YnNjcmlwdGlvbltdID0gW107XG5cbiAgLy8gY29tcG9uZW50cyAvIHBsdWdpbnNcbiAgc2xpY2tFbXB0eVdhcm5pbmc/OiBTbGlja0VtcHR5V2FybmluZ0NvbXBvbmVudDtcbiAgc2xpY2tGb290ZXI/OiBTbGlja0Zvb3RlckNvbXBvbmVudDtcbiAgc2xpY2tQYWdpbmF0aW9uPzogU2xpY2tQYWdpbmF0aW9uQ29tcG9uZW50O1xuICBzbGlja1Jvd0RldGFpbFZpZXc/OiBTbGlja1Jvd0RldGFpbFZpZXc7XG5cbiAgLy8gc2VydmljZXNcbiAgYmFja2VuZFV0aWxpdHlTZXJ2aWNlITogQmFja2VuZFV0aWxpdHlTZXJ2aWNlO1xuICBjb2xsZWN0aW9uU2VydmljZTogQ29sbGVjdGlvblNlcnZpY2U7XG4gIGV4dGVuc2lvblNlcnZpY2U6IEV4dGVuc2lvblNlcnZpY2U7XG4gIGV4dGVuc2lvblV0aWxpdHk6IEV4dGVuc2lvblV0aWxpdHk7XG4gIGZpbHRlckZhY3RvcnkhOiBGaWx0ZXJGYWN0b3J5O1xuICBmaWx0ZXJTZXJ2aWNlOiBGaWx0ZXJTZXJ2aWNlO1xuICBncmlkRXZlbnRTZXJ2aWNlOiBHcmlkRXZlbnRTZXJ2aWNlO1xuICBncmlkU2VydmljZTogR3JpZFNlcnZpY2U7XG4gIGdyaWRTdGF0ZVNlcnZpY2U6IEdyaWRTdGF0ZVNlcnZpY2U7XG4gIGdyb3VwaW5nU2VydmljZTogR3JvdXBpbmdBbmRDb2xzcGFuU2VydmljZTtcbiAgcGFnaW5hdGlvblNlcnZpY2U6IFBhZ2luYXRpb25TZXJ2aWNlO1xuICByZXNpemVyU2VydmljZSE6IFJlc2l6ZXJTZXJ2aWNlO1xuICByeGpzPzogUnhKc0ZhY2FkZTtcbiAgc2hhcmVkU2VydmljZTogU2hhcmVkU2VydmljZTtcbiAgc29ydFNlcnZpY2U6IFNvcnRTZXJ2aWNlO1xuICB0cmVlRGF0YVNlcnZpY2U6IFRyZWVEYXRhU2VydmljZTtcblxuICBASW5wdXQoKSBjdXN0b21EYXRhVmlldzogYW55O1xuICBASW5wdXQoKSBncmlkSWQ6IHN0cmluZyA9ICcnO1xuICBASW5wdXQoKSBncmlkT3B0aW9ucyE6IEdyaWRPcHRpb247XG5cbiAgQElucHV0KClcbiAgZ2V0IHBhZ2luYXRpb25PcHRpb25zKCk6IFBhZ2luYXRpb24gfCB1bmRlZmluZWQge1xuICAgIHJldHVybiB0aGlzLl9wYWdpbmF0aW9uT3B0aW9ucztcbiAgfVxuICBzZXQgcGFnaW5hdGlvbk9wdGlvbnMobmV3UGFnaW5hdGlvbk9wdGlvbnM6IFBhZ2luYXRpb24gfCB1bmRlZmluZWQpIHtcbiAgICBpZiAobmV3UGFnaW5hdGlvbk9wdGlvbnMgJiYgdGhpcy5fcGFnaW5hdGlvbk9wdGlvbnMpIHtcbiAgICAgIHRoaXMuX3BhZ2luYXRpb25PcHRpb25zID0geyAuLi50aGlzLmdyaWRPcHRpb25zLnBhZ2luYXRpb24sIC4uLnRoaXMuX3BhZ2luYXRpb25PcHRpb25zLCAuLi5uZXdQYWdpbmF0aW9uT3B0aW9ucyB9O1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLl9wYWdpbmF0aW9uT3B0aW9ucyA9IG5ld1BhZ2luYXRpb25PcHRpb25zO1xuICAgIH1cbiAgICB0aGlzLmdyaWRPcHRpb25zLnBhZ2luYXRpb24gPSB0aGlzLl9wYWdpbmF0aW9uT3B0aW9ucyA/PyB0aGlzLmdyaWRPcHRpb25zLnBhZ2luYXRpb247XG4gICAgdGhpcy5wYWdpbmF0aW9uU2VydmljZS51cGRhdGVUb3RhbEl0ZW1zKHRoaXMuZ3JpZE9wdGlvbnMucGFnaW5hdGlvbj8udG90YWxJdGVtcyA/PyAwLCB0cnVlKTtcbiAgfVxuXG4gIEBJbnB1dCgpXG4gIHNldCBjb2x1bW5EZWZpbml0aW9ucyhjb2x1bW5EZWZpbml0aW9uczogQ29sdW1uW10pIHtcbiAgICB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucyA9IGNvbHVtbkRlZmluaXRpb25zO1xuICAgIGlmICh0aGlzLl9pc0dyaWRJbml0aWFsaXplZCkge1xuICAgICAgdGhpcy51cGRhdGVDb2x1bW5EZWZpbml0aW9uc0xpc3QoY29sdW1uRGVmaW5pdGlvbnMpO1xuICAgIH1cbiAgICBpZiAoY29sdW1uRGVmaW5pdGlvbnMubGVuZ3RoID4gMCkge1xuICAgICAgdGhpcy5jb3B5Q29sdW1uV2lkdGhzUmVmZXJlbmNlKGNvbHVtbkRlZmluaXRpb25zKTtcbiAgICB9XG4gIH1cbiAgZ2V0IGNvbHVtbkRlZmluaXRpb25zKCk6IENvbHVtbltdIHtcbiAgICByZXR1cm4gdGhpcy5fY29sdW1uRGVmaW5pdGlvbnM7XG4gIH1cblxuICAvLyBtYWtlIHRoZSBjb2x1bW5EZWZpbml0aW9ucyBhIDItd2F5IGJpbmRpbmcgc28gdGhhdCBwbHVnaW4gYWRkaW5nIGNvbHNcbiAgLy8gYXJlIHN5bmNoZWQgb24gdXNlcidzIHNpZGUgYXMgd2VsbCAoUm93TW92ZSwgUm93RGV0YWlsLCBSb3dTZWxlY3Rpb25zKVxuICBAT3V0cHV0KCkgY29sdW1uRGVmaW5pdGlvbnNDaGFuZ2UgPSBuZXcgRXZlbnRFbWl0dGVyKHRydWUpO1xuXG4gIEBJbnB1dCgpXG4gIGdldCBkYXRhc2V0KCk6IGFueVtdIHtcbiAgICByZXR1cm4gKHRoaXMuY3VzdG9tRGF0YVZpZXcgPyB0aGlzLnNsaWNrR3JpZD8uZ2V0RGF0YT8uKCkgOiB0aGlzLmRhdGFWaWV3Py5nZXRJdGVtcz8uKCkpIHx8IFtdO1xuICB9XG4gIHNldCBkYXRhc2V0KG5ld0RhdGFzZXQ6IGFueVtdKSB7XG4gICAgY29uc3QgcHJldkRhdGFzZXRMbiA9IHRoaXMuX2N1cnJlbnREYXRhc2V0TGVuZ3RoO1xuICAgIGNvbnN0IGlzRGF0YXNldEVxdWFsID0gZGVxdWFsKG5ld0RhdGFzZXQsIHRoaXMuX2RhdGFzZXQgfHwgW10pO1xuICAgIGxldCBkYXRhID0gbmV3RGF0YXNldDtcblxuICAgIC8vIHdoZW4gVHJlZSBEYXRhIGlzIGVuYWJsZWQgYW5kIHdlIGRvbid0IHlldCBoYXZlIHRoZSBoaWVyYXJjaGljYWwgZGF0YXNldCBmaWxsZWQsIHdlIGNhbiBmb3JjZSBhIGNvbnZlcnQrc29ydCBvZiB0aGUgYXJyYXlcbiAgICBpZiAodGhpcy5zbGlja0dyaWQgJiYgdGhpcy5ncmlkT3B0aW9ucz8uZW5hYmxlVHJlZURhdGEgJiYgQXJyYXkuaXNBcnJheShuZXdEYXRhc2V0KSAmJiAobmV3RGF0YXNldC5sZW5ndGggPiAwIHx8IG5ld0RhdGFzZXQubGVuZ3RoICE9PSBwcmV2RGF0YXNldExuIHx8ICFpc0RhdGFzZXRFcXVhbCkpIHtcbiAgICAgIHRoaXMuX2lzRGF0YXNldEhpZXJhcmNoaWNhbEluaXRpYWxpemVkID0gZmFsc2U7XG4gICAgICBkYXRhID0gdGhpcy5zb3J0VHJlZURhdGFzZXQobmV3RGF0YXNldCwgIWlzRGF0YXNldEVxdWFsKTsgLy8gaWYgZGF0YXNldCBjaGFuZ2VkLCB0aGVuIGZvcmNlIGEgcmVmcmVzaCBhbnl3YXlcbiAgICB9XG4gICAgdGhpcy5fZGF0YXNldCA9IGRhdGE7XG4gICAgdGhpcy5yZWZyZXNoR3JpZERhdGEoZGF0YSB8fCBbXSk7XG4gICAgdGhpcy5fY3VycmVudERhdGFzZXRMZW5ndGggPSAobmV3RGF0YXNldCB8fCBbXSkubGVuZ3RoO1xuXG4gICAgLy8gZXhwYW5kL2F1dG9maXQgY29sdW1ucyBvbiBmaXJzdCBwYWdlIGxvYWRcbiAgICAvLyB3ZSBjYW4gYXNzdW1lIHRoYXQgaWYgdGhlIHByZXZEYXRhc2V0IHdhcyBlbXB0eSB0aGVuIHdlIGFyZSBvbiBmaXJzdCBsb2FkXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnM/LmF1dG9GaXRDb2x1bW5zT25GaXJzdExvYWQgJiYgcHJldkRhdGFzZXRMbiA9PT0gMCkge1xuICAgICAgdGhpcy5zbGlja0dyaWQuYXV0b3NpemVDb2x1bW5zKCk7XG4gICAgfVxuICB9XG5cbiAgQElucHV0KClcbiAgZ2V0IGRhdGFzZXRIaWVyYXJjaGljYWwoKTogYW55W10gfCB1bmRlZmluZWQge1xuICAgIHJldHVybiB0aGlzLnNoYXJlZFNlcnZpY2UuaGllcmFyY2hpY2FsRGF0YXNldDtcbiAgfVxuICBzZXQgZGF0YXNldEhpZXJhcmNoaWNhbChuZXdIaWVyYXJjaGljYWxEYXRhc2V0OiBhbnlbXSB8IHVuZGVmaW5lZCkge1xuICAgIGNvbnN0IGlzRGF0YXNldEVxdWFsID0gZGVxdWFsKG5ld0hpZXJhcmNoaWNhbERhdGFzZXQsIHRoaXMuc2hhcmVkU2VydmljZT8uaGllcmFyY2hpY2FsRGF0YXNldCA/PyBbXSk7XG4gICAgY29uc3QgcHJldkZsYXREYXRhc2V0TG4gPSB0aGlzLl9jdXJyZW50RGF0YXNldExlbmd0aDtcbiAgICB0aGlzLnNoYXJlZFNlcnZpY2UuaGllcmFyY2hpY2FsRGF0YXNldCA9IG5ld0hpZXJhcmNoaWNhbERhdGFzZXQ7XG5cbiAgICBpZiAobmV3SGllcmFyY2hpY2FsRGF0YXNldCAmJiB0aGlzLmNvbHVtbkRlZmluaXRpb25zICYmIHRoaXMuZmlsdGVyU2VydmljZT8uY2xlYXJGaWx0ZXJzKSB7XG4gICAgICB0aGlzLmZpbHRlclNlcnZpY2UuY2xlYXJGaWx0ZXJzKCk7XG4gICAgfVxuXG4gICAgLy8gd2hlbiBhIGhpZXJhcmNoaWNhbCBkYXRhc2V0IGlzIHNldCBhZnRlcndhcmQsIHdlIGNhbiByZXNldCB0aGUgZmxhdCBkYXRhc2V0IGFuZCBjYWxsIGEgdHJlZSBkYXRhIHNvcnQgdGhhdCB3aWxsIG92ZXJ3cml0ZSB0aGUgZmxhdCBkYXRhc2V0XG4gICAgaWYgKG5ld0hpZXJhcmNoaWNhbERhdGFzZXQgJiYgdGhpcy5zbGlja0dyaWQgJiYgdGhpcy5zb3J0U2VydmljZT8ucHJvY2Vzc1RyZWVEYXRhSW5pdGlhbFNvcnQpIHtcbiAgICAgIHRoaXMuZGF0YVZpZXcuc2V0SXRlbXMoW10sIHRoaXMuZ3JpZE9wdGlvbnMuZGF0YXNldElkUHJvcGVydHlOYW1lID8/ICdpZCcpO1xuICAgICAgdGhpcy5zb3J0U2VydmljZS5wcm9jZXNzVHJlZURhdGFJbml0aWFsU29ydCgpO1xuXG4gICAgICAvLyB3ZSBhbHNvIG5lZWQgdG8gcmVzZXQvcmVmcmVzaCB0aGUgVHJlZSBEYXRhIGZpbHRlcnMgYmVjYXVzZSBpZiB3ZSBpbnNlcnRlZCBuZXcgaXRlbShzKSB0aGVuIGl0IG1pZ2h0IG5vdCBzaG93IHVwIHdpdGhvdXQgZG9pbmcgdGhpcyByZWZyZXNoXG4gICAgICAvLyBob3dldmVyIHdlIG5lZWQgMSBjcHUgY3ljbGUgYmVmb3JlIGhhdmluZyB0aGUgRGF0YVZpZXcgcmVmcmVzaGVkLCBzbyB3ZSBuZWVkIHRvIHdyYXAgdGhpcyBjaGVjayBpbiBhIHNldFRpbWVvdXRcbiAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICBjb25zdCBmbGF0RGF0YXNldExuID0gdGhpcy5kYXRhVmlldy5nZXRJdGVtQ291bnQoKTtcbiAgICAgICAgaWYgKGZsYXREYXRhc2V0TG4gPiAwICYmIChmbGF0RGF0YXNldExuICE9PSBwcmV2RmxhdERhdGFzZXRMbiB8fCAhaXNEYXRhc2V0RXF1YWwpKSB7XG4gICAgICAgICAgdGhpcy5maWx0ZXJTZXJ2aWNlLnJlZnJlc2hUcmVlRGF0YUZpbHRlcnMoKTtcbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgICB0aGlzLl9pc0RhdGFzZXRIaWVyYXJjaGljYWxJbml0aWFsaXplZCA9IHRydWU7XG4gICAgfVxuICB9XG5cbiAgZ2V0IGVsZW1lbnRSZWYoKTogRWxlbWVudFJlZiB7XG4gICAgcmV0dXJuIHRoaXMuZWxtO1xuICB9XG5cbiAgZ2V0IGV2ZW50SGFuZGxlcigpOiBTbGlja0V2ZW50SGFuZGxlciB7XG4gICAgcmV0dXJuIHRoaXMuX2V2ZW50SGFuZGxlcjtcbiAgfVxuXG4gIGdldCBncmlkQ29udGFpbmVyRWxlbWVudCgpOiBIVE1MRWxlbWVudCB8IG51bGwge1xuICAgIHJldHVybiBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKGAjJHt0aGlzLmdyaWRPcHRpb25zLmdyaWRDb250YWluZXJJZCB8fCAnJ31gKTtcbiAgfVxuXG4gIC8qKiBHRVRURVIgdG8ga25vdyBpZiBkYXRhc2V0IHdhcyBpbml0aWFsaXplZCBvciBub3QgKi9cbiAgZ2V0IGlzRGF0YXNldEluaXRpYWxpemVkKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl9pc0RhdGFzZXRJbml0aWFsaXplZDtcbiAgfVxuICAvKiogU0VUVEVSIHRvIGNoYW5nZSBpZiBkYXRhc2V0IHdhcyBpbml0aWFsaXplZCBvciBub3QgKHN0cmluZ2x5IHVzZWQgZm9yIHVuaXQgdGVzdGluZyBwdXJwb3NlcykgKi9cbiAgc2V0IGlzRGF0YXNldEluaXRpYWxpemVkKGlzSW5pdGlhbGl6ZWQ6IGJvb2xlYW4pIHtcbiAgICB0aGlzLl9pc0RhdGFzZXRJbml0aWFsaXplZCA9IGlzSW5pdGlhbGl6ZWQ7XG4gIH1cbiAgc2V0IGlzRGF0YXNldEhpZXJhcmNoaWNhbEluaXRpYWxpemVkKGlzSW5pdGlhbGl6ZWQ6IGJvb2xlYW4pIHtcbiAgICB0aGlzLl9pc0RhdGFzZXRIaWVyYXJjaGljYWxJbml0aWFsaXplZCA9IGlzSW5pdGlhbGl6ZWQ7XG4gIH1cblxuICBnZXQgcmVnaXN0ZXJlZFJlc291cmNlcygpOiBFeHRlcm5hbFJlc291cmNlW10ge1xuICAgIHJldHVybiB0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzO1xuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSByZWFkb25seSBhbmd1bGFyVXRpbFNlcnZpY2U6IEFuZ3VsYXJVdGlsU2VydmljZSxcbiAgICBwcml2YXRlIHJlYWRvbmx5IGFwcFJlZjogQXBwbGljYXRpb25SZWYsXG4gICAgcHJpdmF0ZSByZWFkb25seSBjZDogQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gICAgcHJpdmF0ZSByZWFkb25seSBjb250YWluZXJTZXJ2aWNlOiBDb250YWluZXJTZXJ2aWNlLFxuICAgIHByaXZhdGUgcmVhZG9ubHkgZWxtOiBFbGVtZW50UmVmLFxuICAgIEBPcHRpb25hbCgpIHByaXZhdGUgcmVhZG9ubHkgdHJhbnNsYXRlOiBUcmFuc2xhdGVTZXJ2aWNlLFxuICAgIEBPcHRpb25hbCgpIHByaXZhdGUgcmVhZG9ubHkgdHJhbnNsYXRlclNlcnZpY2U6IFRyYW5zbGF0ZXJTZXJ2aWNlLFxuICAgIEBJbmplY3QoJ2NvbmZpZycpIHByaXZhdGUgZm9yUm9vdENvbmZpZzogR3JpZE9wdGlvbixcbiAgICBASW5qZWN0KCdleHRlcm5hbFNlcnZpY2UnKSBleHRlcm5hbFNlcnZpY2VzOiBFeHRlcm5hbFRlc3RpbmdEZXBlbmRlbmNpZXNcbiAgKSB7XG4gICAgY29uc3Qgc2xpY2tncmlkQ29uZmlnID0gbmV3IFNsaWNrZ3JpZENvbmZpZygpO1xuXG4gICAgLy8gaW5pdGlhbGl6ZSBhbmQgYXNzaWduIGFsbCBTZXJ2aWNlIERlcGVuZGVuY2llc1xuICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSA9IGV4dGVybmFsU2VydmljZXM/LmV2ZW50UHViU3ViU2VydmljZSA/PyBuZXcgRXZlbnRQdWJTdWJTZXJ2aWNlKHRoaXMuZWxtLm5hdGl2ZUVsZW1lbnQpO1xuICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5ldmVudE5hbWluZ1N0eWxlID0gRXZlbnROYW1pbmdTdHlsZS5jYW1lbENhc2U7XG5cbiAgICB0aGlzLmJhY2tlbmRVdGlsaXR5U2VydmljZSA9IGV4dGVybmFsU2VydmljZXM/LmJhY2tlbmRVdGlsaXR5U2VydmljZSA/PyBuZXcgQmFja2VuZFV0aWxpdHlTZXJ2aWNlKCk7XG4gICAgdGhpcy5ncmlkRXZlbnRTZXJ2aWNlID0gZXh0ZXJuYWxTZXJ2aWNlcz8uZ3JpZEV2ZW50U2VydmljZSA/PyBuZXcgR3JpZEV2ZW50U2VydmljZSgpO1xuICAgIHRoaXMuc2hhcmVkU2VydmljZSA9IGV4dGVybmFsU2VydmljZXM/LnNoYXJlZFNlcnZpY2UgPz8gbmV3IFNoYXJlZFNlcnZpY2UoKTtcbiAgICB0aGlzLmNvbGxlY3Rpb25TZXJ2aWNlID0gZXh0ZXJuYWxTZXJ2aWNlcz8uY29sbGVjdGlvblNlcnZpY2UgPz8gbmV3IENvbGxlY3Rpb25TZXJ2aWNlKHRoaXMudHJhbnNsYXRlclNlcnZpY2UpO1xuICAgIHRoaXMuZXh0ZW5zaW9uVXRpbGl0eSA9IGV4dGVybmFsU2VydmljZXM/LmV4dGVuc2lvblV0aWxpdHkgPz8gbmV3IEV4dGVuc2lvblV0aWxpdHkodGhpcy5zaGFyZWRTZXJ2aWNlLCB0aGlzLmJhY2tlbmRVdGlsaXR5U2VydmljZSwgdGhpcy50cmFuc2xhdGVyU2VydmljZSk7XG4gICAgdGhpcy5maWx0ZXJGYWN0b3J5ID0gbmV3IEZpbHRlckZhY3Rvcnkoc2xpY2tncmlkQ29uZmlnLCB0aGlzLnRyYW5zbGF0ZXJTZXJ2aWNlLCB0aGlzLmNvbGxlY3Rpb25TZXJ2aWNlKTtcbiAgICB0aGlzLmZpbHRlclNlcnZpY2UgPSBleHRlcm5hbFNlcnZpY2VzPy5maWx0ZXJTZXJ2aWNlID8/IG5ldyBGaWx0ZXJTZXJ2aWNlKHRoaXMuZmlsdGVyRmFjdG9yeSBhcyBhbnksIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSwgdGhpcy5zaGFyZWRTZXJ2aWNlLCB0aGlzLmJhY2tlbmRVdGlsaXR5U2VydmljZSk7XG4gICAgdGhpcy5yZXNpemVyU2VydmljZSA9IGV4dGVybmFsU2VydmljZXM/LnJlc2l6ZXJTZXJ2aWNlID8/IG5ldyBSZXNpemVyU2VydmljZSh0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UpO1xuICAgIHRoaXMuc29ydFNlcnZpY2UgPSBleHRlcm5hbFNlcnZpY2VzPy5zb3J0U2VydmljZSA/PyBuZXcgU29ydFNlcnZpY2UodGhpcy5zaGFyZWRTZXJ2aWNlLCB0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UsIHRoaXMuYmFja2VuZFV0aWxpdHlTZXJ2aWNlKTtcbiAgICB0aGlzLnRyZWVEYXRhU2VydmljZSA9IGV4dGVybmFsU2VydmljZXM/LnRyZWVEYXRhU2VydmljZSA/PyBuZXcgVHJlZURhdGFTZXJ2aWNlKHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSwgdGhpcy5zaGFyZWRTZXJ2aWNlLCB0aGlzLnNvcnRTZXJ2aWNlKTtcbiAgICB0aGlzLnBhZ2luYXRpb25TZXJ2aWNlID0gZXh0ZXJuYWxTZXJ2aWNlcz8ucGFnaW5hdGlvblNlcnZpY2UgPz8gbmV3IFBhZ2luYXRpb25TZXJ2aWNlKHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSwgdGhpcy5zaGFyZWRTZXJ2aWNlLCB0aGlzLmJhY2tlbmRVdGlsaXR5U2VydmljZSk7XG5cbiAgICB0aGlzLmV4dGVuc2lvblNlcnZpY2UgPSBleHRlcm5hbFNlcnZpY2VzPy5leHRlbnNpb25TZXJ2aWNlID8/IG5ldyBFeHRlbnNpb25TZXJ2aWNlKFxuICAgICAgdGhpcy5leHRlbnNpb25VdGlsaXR5LFxuICAgICAgdGhpcy5maWx0ZXJTZXJ2aWNlLFxuICAgICAgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLFxuICAgICAgdGhpcy5zaGFyZWRTZXJ2aWNlLFxuICAgICAgdGhpcy5zb3J0U2VydmljZSxcbiAgICAgIHRoaXMudHJlZURhdGFTZXJ2aWNlLFxuICAgICAgdGhpcy50cmFuc2xhdGVyU2VydmljZSxcbiAgICApO1xuXG4gICAgdGhpcy5ncmlkU3RhdGVTZXJ2aWNlID0gZXh0ZXJuYWxTZXJ2aWNlcz8uZ3JpZFN0YXRlU2VydmljZSA/PyBuZXcgR3JpZFN0YXRlU2VydmljZSh0aGlzLmV4dGVuc2lvblNlcnZpY2UsIHRoaXMuZmlsdGVyU2VydmljZSwgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLCB0aGlzLnNoYXJlZFNlcnZpY2UsIHRoaXMuc29ydFNlcnZpY2UsIHRoaXMudHJlZURhdGFTZXJ2aWNlKTtcbiAgICB0aGlzLmdyaWRTZXJ2aWNlID0gZXh0ZXJuYWxTZXJ2aWNlcz8uZ3JpZFNlcnZpY2UgPz8gbmV3IEdyaWRTZXJ2aWNlKHRoaXMuZ3JpZFN0YXRlU2VydmljZSwgdGhpcy5maWx0ZXJTZXJ2aWNlLCB0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UsIHRoaXMucGFnaW5hdGlvblNlcnZpY2UsIHRoaXMuc2hhcmVkU2VydmljZSwgdGhpcy5zb3J0U2VydmljZSwgdGhpcy50cmVlRGF0YVNlcnZpY2UpO1xuICAgIHRoaXMuZ3JvdXBpbmdTZXJ2aWNlID0gZXh0ZXJuYWxTZXJ2aWNlcz8uZ3JvdXBpbmdBbmRDb2xzcGFuU2VydmljZSA/PyBuZXcgR3JvdXBpbmdBbmRDb2xzcGFuU2VydmljZSh0aGlzLmV4dGVuc2lvblV0aWxpdHksIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSk7XG5cbiAgICB0aGlzLnNlcnZpY2VMaXN0ID0gW1xuICAgICAgdGhpcy5jb250YWluZXJTZXJ2aWNlLFxuICAgICAgdGhpcy5leHRlbnNpb25TZXJ2aWNlLFxuICAgICAgdGhpcy5maWx0ZXJTZXJ2aWNlLFxuICAgICAgdGhpcy5ncmlkRXZlbnRTZXJ2aWNlLFxuICAgICAgdGhpcy5ncmlkU2VydmljZSxcbiAgICAgIHRoaXMuZ3JpZFN0YXRlU2VydmljZSxcbiAgICAgIHRoaXMuZ3JvdXBpbmdTZXJ2aWNlLFxuICAgICAgdGhpcy5wYWdpbmF0aW9uU2VydmljZSxcbiAgICAgIHRoaXMucmVzaXplclNlcnZpY2UsXG4gICAgICB0aGlzLnNvcnRTZXJ2aWNlLFxuICAgICAgdGhpcy50cmVlRGF0YVNlcnZpY2UsXG4gICAgXTtcblxuICAgIC8vIHJlZ2lzdGVyIGFsbCBTZXJ2aWNlIGluc3RhbmNlcyBpbiB0aGUgY29udGFpbmVyXG4gICAgdGhpcy5jb250YWluZXJTZXJ2aWNlLnJlZ2lzdGVySW5zdGFuY2UoJ0V4dGVuc2lvblV0aWxpdHknLCB0aGlzLmV4dGVuc2lvblV0aWxpdHkpO1xuICAgIHRoaXMuY29udGFpbmVyU2VydmljZS5yZWdpc3Rlckluc3RhbmNlKCdGaWx0ZXJTZXJ2aWNlJywgdGhpcy5maWx0ZXJTZXJ2aWNlKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnQ29sbGVjdGlvblNlcnZpY2UnLCB0aGlzLmNvbGxlY3Rpb25TZXJ2aWNlKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnRXh0ZW5zaW9uU2VydmljZScsIHRoaXMuZXh0ZW5zaW9uU2VydmljZSk7XG4gICAgdGhpcy5jb250YWluZXJTZXJ2aWNlLnJlZ2lzdGVySW5zdGFuY2UoJ0dyaWRFdmVudFNlcnZpY2UnLCB0aGlzLmdyaWRFdmVudFNlcnZpY2UpO1xuICAgIHRoaXMuY29udGFpbmVyU2VydmljZS5yZWdpc3Rlckluc3RhbmNlKCdHcmlkU2VydmljZScsIHRoaXMuZ3JpZFNlcnZpY2UpO1xuICAgIHRoaXMuY29udGFpbmVyU2VydmljZS5yZWdpc3Rlckluc3RhbmNlKCdHcmlkU3RhdGVTZXJ2aWNlJywgdGhpcy5ncmlkU3RhdGVTZXJ2aWNlKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnR3JvdXBpbmdBbmRDb2xzcGFuU2VydmljZScsIHRoaXMuZ3JvdXBpbmdTZXJ2aWNlKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnUGFnaW5hdGlvblNlcnZpY2UnLCB0aGlzLnBhZ2luYXRpb25TZXJ2aWNlKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnUmVzaXplclNlcnZpY2UnLCB0aGlzLnJlc2l6ZXJTZXJ2aWNlKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnU2hhcmVkU2VydmljZScsIHRoaXMuc2hhcmVkU2VydmljZSk7XG4gICAgdGhpcy5jb250YWluZXJTZXJ2aWNlLnJlZ2lzdGVySW5zdGFuY2UoJ1NvcnRTZXJ2aWNlJywgdGhpcy5zb3J0U2VydmljZSk7XG4gICAgdGhpcy5jb250YWluZXJTZXJ2aWNlLnJlZ2lzdGVySW5zdGFuY2UoJ0V2ZW50UHViU3ViU2VydmljZScsIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSk7XG4gICAgdGhpcy5jb250YWluZXJTZXJ2aWNlLnJlZ2lzdGVySW5zdGFuY2UoJ1B1YlN1YlNlcnZpY2UnLCB0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UpO1xuICAgIHRoaXMuY29udGFpbmVyU2VydmljZS5yZWdpc3Rlckluc3RhbmNlKCdUcmFuc2xhdGVyU2VydmljZScsIHRoaXMudHJhbnNsYXRlclNlcnZpY2UpO1xuICAgIHRoaXMuY29udGFpbmVyU2VydmljZS5yZWdpc3Rlckluc3RhbmNlKCdUcmVlRGF0YVNlcnZpY2UnLCB0aGlzLnRyZWVEYXRhU2VydmljZSk7XG4gIH1cblxuICBuZ0FmdGVyVmlld0luaXQoKSB7XG4gICAgaWYgKCF0aGlzLmdyaWRPcHRpb25zIHx8ICF0aGlzLmNvbHVtbkRlZmluaXRpb25zKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ1VzaW5nIGA8YW5ndWxhci1zbGlja2dyaWQ+YCByZXF1aXJlcyBbZ3JpZE9wdGlvbnNdIGFuZCBbY29sdW1uRGVmaW5pdGlvbnNdLCBpdCBzZWVtcyB0aGF0IHlvdSBtaWdodCBoYXZlIGZvcmdvdCB0byBwcm92aWRlIHRoZW0gc2luY2UgYXQgbGVhc3Qgb2YgdGhlbSBpcyB1bmRlZmluZWQuJyk7XG4gICAgfVxuICAgIHRoaXMuaW5pdGlhbGl6YXRpb24odGhpcy5fZXZlbnRIYW5kbGVyKTtcbiAgICB0aGlzLl9pc0dyaWRJbml0aWFsaXplZCA9IHRydWU7XG5cbiAgICAvLyByZWNoZWNrIHRoZSBlbXB0eSB3YXJuaW5nIG1lc3NhZ2UgYWZ0ZXIgZ3JpZCBpcyBzaG93biBzbyB0aGF0IGl0IHdvcmtzIGluIGV2ZXJ5IHVzZSBjYXNlXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMgJiYgdGhpcy5ncmlkT3B0aW9ucy5lbmFibGVFbXB0eURhdGFXYXJuaW5nTWVzc2FnZSAmJiBBcnJheS5pc0FycmF5KHRoaXMuZGF0YXNldCkpIHtcbiAgICAgIGNvbnN0IGZpbmFsVG90YWxDb3VudCA9IHRoaXMuZGF0YXNldC5sZW5ndGg7XG4gICAgICB0aGlzLmRpc3BsYXlFbXB0eURhdGFXYXJuaW5nKGZpbmFsVG90YWxDb3VudCA8IDEpO1xuICAgIH1cbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5wdWJsaXNoKCdvbkJlZm9yZUdyaWREZXN0cm95JywgdGhpcy5zbGlja0dyaWQpO1xuICAgIHRoaXMuZGVzdHJveSgpO1xuICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5wdWJsaXNoKCdvbkFmdGVyR3JpZERlc3Ryb3llZCcsIHRydWUpO1xuICB9XG5cbiAgZGVzdHJveShzaG91bGRFbXB0eURvbUVsZW1lbnRDb250YWluZXIgPSBmYWxzZSkge1xuICAgIC8vIGRpc3Bvc2Ugb2YgYWxsIFNlcnZpY2VzXG4gICAgdGhpcy5zZXJ2aWNlTGlzdC5mb3JFYWNoKChzZXJ2aWNlOiBhbnkpID0+IHtcbiAgICAgIGlmIChzZXJ2aWNlICYmIHNlcnZpY2UuZGlzcG9zZSkge1xuICAgICAgICBzZXJ2aWNlLmRpc3Bvc2UoKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgICB0aGlzLnNlcnZpY2VMaXN0ID0gW107XG5cbiAgICAvLyBkaXNwb3NlIGFsbCByZWdpc3RlcmVkIGV4dGVybmFsIHJlc291cmNlc1xuICAgIGlmIChBcnJheS5pc0FycmF5KHRoaXMuX3JlZ2lzdGVyZWRSZXNvdXJjZXMpKSB7XG4gICAgICB3aGlsZSAodGhpcy5fcmVnaXN0ZXJlZFJlc291cmNlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgIGNvbnN0IHJlc291cmNlID0gdGhpcy5fcmVnaXN0ZXJlZFJlc291cmNlcy5wb3AoKTtcbiAgICAgICAgaWYgKHJlc291cmNlPy5kaXNwb3NlKSB7XG4gICAgICAgICAgcmVzb3VyY2UuZGlzcG9zZSgpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICB0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzID0gW107XG4gICAgfVxuXG4gICAgLy8gZGlzcG9zZSB0aGUgQ29tcG9uZW50c1xuICAgIHRoaXMuc2xpY2tFbXB0eVdhcm5pbmc/LmRpc3Bvc2UoKTtcbiAgICB0aGlzLnNsaWNrRm9vdGVyPy5kaXNwb3NlKCk7XG4gICAgdGhpcy5zbGlja1BhZ2luYXRpb24/LmRpc3Bvc2UoKTtcblxuICAgIGlmICh0aGlzLl9ldmVudEhhbmRsZXI/LnVuc3Vic2NyaWJlQWxsKSB7XG4gICAgICB0aGlzLl9ldmVudEhhbmRsZXIudW5zdWJzY3JpYmVBbGwoKTtcbiAgICB9XG4gICAgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlPy51bnN1YnNjcmliZUFsbCgpO1xuICAgIGlmICh0aGlzLmRhdGFWaWV3KSB7XG4gICAgICBpZiAodGhpcy5kYXRhVmlldz8uc2V0SXRlbXMpIHtcbiAgICAgICAgdGhpcy5kYXRhVmlldy5zZXRJdGVtcyhbXSk7XG4gICAgICB9XG4gICAgICBpZiAodGhpcy5kYXRhVmlldy5kZXN0cm95KSB7XG4gICAgICAgIHRoaXMuZGF0YVZpZXcuZGVzdHJveSgpO1xuICAgICAgfVxuICAgIH1cbiAgICBpZiAodGhpcy5zbGlja0dyaWQ/LmRlc3Ryb3kpIHtcbiAgICAgIHRoaXMuc2xpY2tHcmlkLmRlc3Ryb3koc2hvdWxkRW1wdHlEb21FbGVtZW50Q29udGFpbmVyKTtcbiAgICB9XG5cbiAgICBpZiAodGhpcy5iYWNrZW5kU2VydmljZUFwaSkge1xuICAgICAgZm9yIChjb25zdCBwcm9wIG9mIE9iamVjdC5rZXlzKHRoaXMuYmFja2VuZFNlcnZpY2VBcGkpKSB7XG4gICAgICAgIGRlbGV0ZSB0aGlzLmJhY2tlbmRTZXJ2aWNlQXBpW3Byb3AgYXMga2V5b2YgQmFja2VuZFNlcnZpY2VBcGldO1xuICAgICAgfVxuICAgICAgdGhpcy5iYWNrZW5kU2VydmljZUFwaSA9IHVuZGVmaW5lZDtcbiAgICB9XG4gICAgZm9yIChjb25zdCBwcm9wIG9mIE9iamVjdC5rZXlzKHRoaXMuY29sdW1uRGVmaW5pdGlvbnMpKSB7XG4gICAgICAodGhpcy5jb2x1bW5EZWZpbml0aW9ucyBhcyBhbnkpW3Byb3BdID0gbnVsbDtcbiAgICB9XG4gICAgZm9yIChjb25zdCBwcm9wIG9mIE9iamVjdC5rZXlzKHRoaXMuc2hhcmVkU2VydmljZSkpIHtcbiAgICAgICh0aGlzLnNoYXJlZFNlcnZpY2UgYXMgYW55KVtwcm9wXSA9IG51bGw7XG4gICAgfVxuXG4gICAgLy8gd2UgY291bGQgb3B0aW9uYWxseSBhbHNvIGVtcHR5IHRoZSBjb250ZW50IG9mIHRoZSBncmlkIGNvbnRhaW5lciBET00gZWxlbWVudFxuICAgIGlmIChzaG91bGRFbXB0eURvbUVsZW1lbnRDb250YWluZXIpIHtcbiAgICAgIHRoaXMuZW1wdHlHcmlkQ29udGFpbmVyRWxtKCk7XG4gICAgfVxuXG4gICAgLy8gYWxzbyB1bnN1YnNjcmliZSBhbGwgUnhKUyBzdWJzY3JpcHRpb25zXG4gICAgdGhpcy5zdWJzY3JpcHRpb25zID0gdW5zdWJzY3JpYmVBbGwodGhpcy5zdWJzY3JpcHRpb25zKTtcblxuICAgIHRoaXMuX2RhdGFzZXQgPSBudWxsO1xuICAgIHRoaXMuZGF0YXNldEhpZXJhcmNoaWNhbCA9IHVuZGVmaW5lZDtcbiAgICB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucyA9IFtdO1xuICAgIHRoaXMuX2FuZ3VsYXJHcmlkSW5zdGFuY2VzID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuc2xpY2tHcmlkID0gdW5kZWZpbmVkIGFzIGFueTtcbiAgfVxuXG4gIGVtcHR5R3JpZENvbnRhaW5lckVsbSgpIHtcbiAgICBjb25zdCBncmlkQ29udGFpbmVySWQgPSB0aGlzLmdyaWRPcHRpb25zPy5ncmlkQ29udGFpbmVySWQgPz8gJ2dyaWQxJztcbiAgICBjb25zdCBncmlkQ29udGFpbmVyRWxtID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvcihgIyR7Z3JpZENvbnRhaW5lcklkfWApO1xuICAgIGVtcHR5RWxlbWVudChncmlkQ29udGFpbmVyRWxtKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZWZpbmUgb3VyIGludGVybmFsIFBvc3QgUHJvY2VzcyBjYWxsYmFjaywgaXQgd2lsbCBleGVjdXRlIGludGVybmFsbHkgYWZ0ZXIgd2UgZ2V0IGJhY2sgcmVzdWx0IGZyb20gdGhlIFByb2Nlc3MgYmFja2VuZCBjYWxsXG4gICAqIEZvciBub3csIHRoaXMgaXMgR3JhcGhRTCBTZXJ2aWNlIE9OTFkgZmVhdHVyZSBhbmQgaXQgd2lsbCBiYXNpY2FsbHkgcmVmcmVzaCB0aGUgRGF0YXNldCAmIFBhZ2luYXRpb24gd2l0aG91dCBoYXZpbmcgdGhlIHVzZXIgdG8gY3JlYXRlIGhpcyBvd24gUG9zdFByb2Nlc3MgZXZlcnkgdGltZVxuICAgKi9cbiAgY3JlYXRlQmFja2VuZEFwaUludGVybmFsUG9zdFByb2Nlc3NDYWxsYmFjayhncmlkT3B0aW9uczogR3JpZE9wdGlvbikge1xuICAgIGNvbnN0IGJhY2tlbmRBcGkgPSBncmlkT3B0aW9ucyAmJiBncmlkT3B0aW9ucy5iYWNrZW5kU2VydmljZUFwaTtcbiAgICBpZiAoYmFja2VuZEFwaSAmJiBiYWNrZW5kQXBpLnNlcnZpY2UpIHtcbiAgICAgIGNvbnN0IGJhY2tlbmRBcGlTZXJ2aWNlID0gYmFja2VuZEFwaS5zZXJ2aWNlO1xuXG4gICAgICAvLyBpbnRlcm5hbFBvc3RQcm9jZXNzIG9ubHkgd29ya3MgKGZvciBub3cpIHdpdGggYSBHcmFwaFFMIFNlcnZpY2UsIHNvIG1ha2Ugc3VyZSBpdCBpcyBvZiB0aGF0IHR5cGVcbiAgICAgIGlmICh0eXBlb2YgYmFja2VuZEFwaVNlcnZpY2UuZ2V0RGF0YXNldE5hbWUgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgYmFja2VuZEFwaS5pbnRlcm5hbFBvc3RQcm9jZXNzID0gKHByb2Nlc3NSZXN1bHQ6IGFueSkgPT4ge1xuICAgICAgICAgIGNvbnN0IGRhdGFzZXROYW1lID0gKGJhY2tlbmRBcGkgJiYgYmFja2VuZEFwaVNlcnZpY2UgJiYgdHlwZW9mIGJhY2tlbmRBcGlTZXJ2aWNlLmdldERhdGFzZXROYW1lID09PSAnZnVuY3Rpb24nKSA/IGJhY2tlbmRBcGlTZXJ2aWNlLmdldERhdGFzZXROYW1lKCkgOiAnJztcbiAgICAgICAgICBpZiAocHJvY2Vzc1Jlc3VsdD8uZGF0YVtkYXRhc2V0TmFtZV0pIHtcbiAgICAgICAgICAgIGNvbnN0IGRhdGEgPSBwcm9jZXNzUmVzdWx0LmRhdGFbZGF0YXNldE5hbWVdLmhhc093blByb3BlcnR5KCdub2RlcycpID8gKHByb2Nlc3NSZXN1bHQgYXMgYW55KS5kYXRhW2RhdGFzZXROYW1lXS5ub2RlcyA6IChwcm9jZXNzUmVzdWx0IGFzIGFueSkuZGF0YVtkYXRhc2V0TmFtZV07XG4gICAgICAgICAgICBjb25zdCB0b3RhbENvdW50ID0gcHJvY2Vzc1Jlc3VsdC5kYXRhW2RhdGFzZXROYW1lXS5oYXNPd25Qcm9wZXJ0eSgndG90YWxDb3VudCcpID8gKHByb2Nlc3NSZXN1bHQgYXMgYW55KS5kYXRhW2RhdGFzZXROYW1lXS50b3RhbENvdW50IDogKHByb2Nlc3NSZXN1bHQgYXMgYW55KS5kYXRhW2RhdGFzZXROYW1lXS5sZW5ndGg7XG4gICAgICAgICAgICB0aGlzLnJlZnJlc2hHcmlkRGF0YShkYXRhLCB0b3RhbENvdW50IHx8IDApO1xuICAgICAgICAgIH1cbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICBpbml0aWFsaXphdGlvbihldmVudEhhbmRsZXI6IFNsaWNrRXZlbnRIYW5kbGVyKSB7XG4gICAgdGhpcy5ncmlkT3B0aW9ucy50cmFuc2xhdGVyID0gdGhpcy50cmFuc2xhdGVyU2VydmljZTtcbiAgICB0aGlzLl9ldmVudEhhbmRsZXIgPSBldmVudEhhbmRsZXI7XG5cbiAgICAvLyB3aGVuIGRldGVjdGluZyBhIGZyb3plbiBncmlkLCB3ZSdsbCBhdXRvbWF0aWNhbGx5IGVuYWJsZSB0aGUgbW91c2V3aGVlbCBzY3JvbGwgaGFuZGxlciBzbyB0aGF0IHdlIGNhbiBzY3JvbGwgZnJvbSBib3RoIGxlZnQvcmlnaHQgZnJvemVuIGNvbnRhaW5lcnNcbiAgICBpZiAodGhpcy5ncmlkT3B0aW9ucyAmJiAoKHRoaXMuZ3JpZE9wdGlvbnMuZnJvemVuUm93ICE9PSB1bmRlZmluZWQgJiYgdGhpcy5ncmlkT3B0aW9ucy5mcm96ZW5Sb3cgPj0gMCkgfHwgdGhpcy5ncmlkT3B0aW9ucy5mcm96ZW5Db2x1bW4gIT09IHVuZGVmaW5lZCAmJiB0aGlzLmdyaWRPcHRpb25zLmZyb3plbkNvbHVtbiA+PSAwKSAmJiB0aGlzLmdyaWRPcHRpb25zLmVuYWJsZU1vdXNlV2hlZWxTY3JvbGxIYW5kbGVyID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlTW91c2VXaGVlbFNjcm9sbEhhbmRsZXIgPSB0cnVlO1xuICAgIH1cblxuICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5ldmVudE5hbWluZ1N0eWxlID0gdGhpcy5ncmlkT3B0aW9ucz8uZXZlbnROYW1pbmdTdHlsZSA/PyBFdmVudE5hbWluZ1N0eWxlLmNhbWVsQ2FzZTtcbiAgICB0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UucHVibGlzaCgnb25CZWZvcmVHcmlkQ3JlYXRlJywgdHJ1ZSk7XG5cbiAgICAvLyBtYWtlIHN1cmUgdGhlIGRhdGFzZXQgaXMgaW5pdGlhbGl6ZWQgKGlmIG5vdCBpdCB3aWxsIHRocm93IGFuIGVycm9yIHRoYXQgaXQgY2Fubm90IGdldExlbmd0aCBvZiBudWxsKVxuICAgIHRoaXMuX2RhdGFzZXQgPSB0aGlzLl9kYXRhc2V0IHx8IFtdO1xuICAgIHRoaXMuZ3JpZE9wdGlvbnMgPSB0aGlzLm1lcmdlR3JpZE9wdGlvbnModGhpcy5ncmlkT3B0aW9ucyk7XG4gICAgdGhpcy5fcGFnaW5hdGlvbk9wdGlvbnMgPSB0aGlzLmdyaWRPcHRpb25zPy5wYWdpbmF0aW9uO1xuICAgIHRoaXMubG9jYWxlcyA9IHRoaXMuZ3JpZE9wdGlvbnM/LmxvY2FsZXMgPz8gQ29uc3RhbnRzLmxvY2FsZXM7XG4gICAgdGhpcy5iYWNrZW5kU2VydmljZUFwaSA9IHRoaXMuZ3JpZE9wdGlvbnM/LmJhY2tlbmRTZXJ2aWNlQXBpO1xuICAgIHRoaXMuX2lzTG9jYWxHcmlkID0gIXRoaXMuYmFja2VuZFNlcnZpY2VBcGk7IC8vIGNvbnNpZGVyZWQgYSBsb2NhbCBncmlkIGlmIGl0IGRvZXNuJ3QgaGF2ZSBhIGJhY2tlbmQgc2VydmljZSBzZXRcblxuICAgIHRoaXMuY3JlYXRlQmFja2VuZEFwaUludGVybmFsUG9zdFByb2Nlc3NDYWxsYmFjayh0aGlzLmdyaWRPcHRpb25zKTtcblxuICAgIGlmICghdGhpcy5jdXN0b21EYXRhVmlldykge1xuICAgICAgY29uc3QgZGF0YXZpZXdJbmxpbmVGaWx0ZXJzID0gdGhpcy5ncmlkT3B0aW9ucy5kYXRhVmlldyAmJiB0aGlzLmdyaWRPcHRpb25zLmRhdGFWaWV3LmlubGluZUZpbHRlcnMgfHwgZmFsc2U7XG4gICAgICBsZXQgZGF0YVZpZXdPcHRpb25zOiBEYXRhVmlld09wdGlvbiA9IHsgaW5saW5lRmlsdGVyczogZGF0YXZpZXdJbmxpbmVGaWx0ZXJzIH07XG5cbiAgICAgIGlmICh0aGlzLmdyaWRPcHRpb25zLmRyYWdnYWJsZUdyb3VwaW5nIHx8IHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlR3JvdXBpbmcpIHtcbiAgICAgICAgdGhpcy5ncm91cEl0ZW1NZXRhZGF0YVByb3ZpZGVyID0gbmV3IFNsaWNrR3JvdXBJdGVtTWV0YWRhdGFQcm92aWRlcigpO1xuICAgICAgICB0aGlzLnNoYXJlZFNlcnZpY2UuZ3JvdXBJdGVtTWV0YWRhdGFQcm92aWRlciA9IHRoaXMuZ3JvdXBJdGVtTWV0YWRhdGFQcm92aWRlcjtcbiAgICAgICAgZGF0YVZpZXdPcHRpb25zID0geyAuLi5kYXRhVmlld09wdGlvbnMsIGdyb3VwSXRlbU1ldGFkYXRhUHJvdmlkZXI6IHRoaXMuZ3JvdXBJdGVtTWV0YWRhdGFQcm92aWRlciB9O1xuICAgICAgfVxuICAgICAgdGhpcy5kYXRhVmlldyA9IG5ldyBTbGljay5EYXRhLkRhdGFWaWV3KGRhdGFWaWV3T3B0aW9ucyk7XG4gICAgICB0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UucHVibGlzaCgnb25EYXRhdmlld0NyZWF0ZWQnLCB0aGlzLmRhdGFWaWV3KTtcbiAgICB9XG5cbiAgICAvLyBnZXQgYW55IHBvc3NpYmxlIFNlcnZpY2VzIHRoYXQgdXNlciB3YW50IHRvIHJlZ2lzdGVyIHdoaWNoIGRvbid0IHJlcXVpcmUgU2xpY2tHcmlkIHRvIGJlIGluc3RhbnRpYXRlZFxuICAgIC8vIFJ4SlMgUmVzb3VyY2UgaXMgaW4gdGhpcyBsb3QgYmVjYXVzZSBpdCBoYXMgdG8gYmUgcmVnaXN0ZXJlZCBiZWZvcmUgYW55dGhpbmcgZWxzZSBhbmQgZG9lc24ndCByZXF1aXJlIFNsaWNrR3JpZCB0byBiZSBpbml0aWFsaXplZFxuICAgIHRoaXMucHJlUmVnaXN0ZXJSZXNvdXJjZXMoKTtcblxuICAgIC8vIGZvciBjb252ZW5pZW5jZSB0byB0aGUgdXNlciwgd2UgcHJvdmlkZSB0aGUgcHJvcGVydHkgXCJlZGl0b3JcIiBhcyBhbiBBbmd1bGFyLVNsaWNrZ3JpZCBlZGl0b3IgY29tcGxleCBvYmplY3RcbiAgICAvLyBob3dldmVyIFwiZWRpdG9yXCIgaXMgdXNlZCBpbnRlcm5hbGx5IGJ5IFNsaWNrR3JpZCBmb3IgaXQncyBvd24gRWRpdG9yIEZhY3RvcnlcbiAgICAvLyBzbyBpbiBvdXIgbGliIHdlIHdpbGwgc3dhcCBcImVkaXRvclwiIGFuZCBjb3B5IGl0IGludG8gYSBuZXcgcHJvcGVydHkgY2FsbGVkIFwiaW50ZXJuYWxDb2x1bW5FZGl0b3JcIlxuICAgIC8vIHRoZW4gdGFrZSBiYWNrIFwiZWRpdG9yLm1vZGVsXCIgYW5kIG1ha2UgaXQgdGhlIG5ldyBcImVkaXRvclwiIHNvIHRoYXQgU2xpY2tHcmlkIEVkaXRvciBGYWN0b3J5IHN0aWxsIHdvcmtzXG4gICAgdGhpcy5fY29sdW1uRGVmaW5pdGlvbnMgPSB0aGlzLnN3YXBJbnRlcm5hbEVkaXRvclRvU2xpY2tHcmlkRmFjdG9yeUVkaXRvcih0aGlzLl9jb2x1bW5EZWZpbml0aW9ucyk7XG5cbiAgICAvLyBpZiB0aGUgdXNlciB3YW50cyB0byBhdXRvbWF0aWNhbGx5IGFkZCBhIEN1c3RvbSBFZGl0b3IgRm9ybWF0dGVyLCB3ZSBuZWVkIHRvIGNhbGwgdGhlIGF1dG8gYWRkIGZ1bmN0aW9uIGFnYWluXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMuYXV0b0FkZEN1c3RvbUVkaXRvckZvcm1hdHRlcikge1xuICAgICAgYXV0b0FkZEVkaXRvckZvcm1hdHRlclRvQ29sdW1uc1dpdGhFZGl0b3IodGhpcy5fY29sdW1uRGVmaW5pdGlvbnMsIHRoaXMuZ3JpZE9wdGlvbnMuYXV0b0FkZEN1c3RvbUVkaXRvckZvcm1hdHRlcik7XG4gICAgfVxuXG4gICAgLy8gc2F2ZSByZWZlcmVuY2UgZm9yIGFsbCBjb2x1bW5zIGJlZm9yZSB0aGV5IG9wdGlvbmFsbHkgYmVjb21lIGhpZGRlbi92aXNpYmxlXG4gICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmFsbENvbHVtbnMgPSB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucztcbiAgICB0aGlzLnNoYXJlZFNlcnZpY2UudmlzaWJsZUNvbHVtbnMgPSB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucztcblxuICAgIC8vIGJlZm9yZSBjZXJ0YWluIGV4dGVudGlvbnMvcGx1Z2lucyBwb3RlbnRpYWxseSBhZGRzIGV4dHJhIGNvbHVtbnMgbm90IGNyZWF0ZWQgYnkgdGhlIHVzZXIgaXRzZWxmIChSb3dNb3ZlLCBSb3dEZXRhaWwsIFJvd1NlbGVjdGlvbnMpXG4gICAgLy8gd2UnbGwgc3Vic2NyaWJlIHRvIHRoZSBldmVudCBhbmQgcHVzaCBiYWNrIHRoZSBjaGFuZ2UgdG8gdGhlIHVzZXIgc28gdGhleSBhbHdheXMgdXNlIGZ1bGwgY29sdW1uIGRlZnMgYXJyYXkgaW5jbHVkaW5nIGV4dHJhIGNvbHNcbiAgICB0aGlzLnN1YnNjcmlwdGlvbnMucHVzaChcbiAgICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5zdWJzY3JpYmU8eyBjb2x1bW5zOiBDb2x1bW5bXTsgZ3JpZDogU2xpY2tHcmlkOyB9Pignb25QbHVnaW5Db2x1bW5zQ2hhbmdlZCcsIGRhdGEgPT4ge1xuICAgICAgICB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucyA9IGRhdGEuY29sdW1ucztcbiAgICAgICAgdGhpcy5jb2x1bW5EZWZpbml0aW9uc0NoYW5nZS5lbWl0KHRoaXMuX2NvbHVtbkRlZmluaXRpb25zKTtcbiAgICAgIH0pXG4gICAgKTtcblxuICAgIC8vIGFmdGVyIHN1YnNjcmliaW5nIHRvIHBvdGVudGlhbCBjb2x1bW5zIGNoYW5nZWQsIHdlIGFyZSByZWFkeSB0byBjcmVhdGUgdGhlc2Ugb3B0aW9uYWwgZXh0ZW5zaW9uc1xuICAgIC8vIHdoZW4gd2UgZGlkIGZpbmQgc29tZSB0byBjcmVhdGUgKFJvd01vdmUsIFJvd0RldGFpbCwgUm93U2VsZWN0aW9ucyksIGl0IHdpbGwgYXV0b21hdGljYWxseSBtb2RpZnkgY29sdW1uIGRlZmluaXRpb25zIChieSBwcmV2aW91cyBzdWJzY3JpYmUpXG4gICAgdGhpcy5leHRlbnNpb25TZXJ2aWNlLmNyZWF0ZUV4dGVuc2lvbnNCZWZvcmVHcmlkQ3JlYXRpb24odGhpcy5fY29sdW1uRGVmaW5pdGlvbnMsIHRoaXMuZ3JpZE9wdGlvbnMpO1xuXG4gICAgLy8gaWYgdXNlciBlbnRlcmVkIHNvbWUgUGlubmluZy9Gcm96ZW4gXCJwcmVzZXRzXCIsIHdlIG5lZWQgdG8gYXBwbHkgdGhlbSBpbiB0aGUgZ3JpZCBvcHRpb25zXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMucHJlc2V0cz8ucGlubmluZykge1xuICAgICAgdGhpcy5ncmlkT3B0aW9ucyA9IHsgLi4udGhpcy5ncmlkT3B0aW9ucywgLi4udGhpcy5ncmlkT3B0aW9ucy5wcmVzZXRzLnBpbm5pbmcgfTtcbiAgICB9XG5cbiAgICAvLyBidWlsZCBTbGlja0dyaWQgR3JpZCwgYWxzbyB1c2VyIG1pZ2h0IG9wdGlvbmFsbHkgcGFzcyBhIGN1c3RvbSBkYXRhdmlldyAoZS5nLiByZW1vdGUgbW9kZWwpXG4gICAgdGhpcy5zbGlja0dyaWQgPSBuZXcgU2xpY2suR3JpZChgIyR7dGhpcy5ncmlkSWR9YCwgdGhpcy5jdXN0b21EYXRhVmlldyB8fCB0aGlzLmRhdGFWaWV3LCB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucywgdGhpcy5ncmlkT3B0aW9ucyk7XG4gICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmRhdGFWaWV3ID0gdGhpcy5kYXRhVmlldztcbiAgICB0aGlzLnNoYXJlZFNlcnZpY2Uuc2xpY2tHcmlkID0gdGhpcy5zbGlja0dyaWQ7XG4gICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmdyaWRDb250YWluZXJFbGVtZW50ID0gdGhpcy5lbG0ubmF0aXZlRWxlbWVudCBhcyBIVE1MRGl2RWxlbWVudDtcblxuICAgIHRoaXMuZXh0ZW5zaW9uU2VydmljZS5iaW5kRGlmZmVyZW50RXh0ZW5zaW9ucygpO1xuICAgIHRoaXMuYmluZERpZmZlcmVudEhvb2tzKHRoaXMuc2xpY2tHcmlkLCB0aGlzLmdyaWRPcHRpb25zLCB0aGlzLmRhdGFWaWV3KTtcblxuICAgIC8vIHdoZW4gaXQncyBhIGZyb3plbiBncmlkLCB3ZSBuZWVkIHRvIGtlZXAgdGhlIGZyb3plbiBjb2x1bW4gaWQgZm9yIHJlZmVyZW5jZSBpZiB3ZSBldmVyIHNob3cvaGlkZSBjb2x1bW4gZnJvbSBDb2x1bW5QaWNrZXIvR3JpZE1lbnUgYWZ0ZXJ3YXJkXG4gICAgY29uc3QgZnJvemVuQ29sdW1uSW5kZXggPSB0aGlzLmdyaWRPcHRpb25zLmZyb3plbkNvbHVtbiAhPT0gdW5kZWZpbmVkID8gdGhpcy5ncmlkT3B0aW9ucy5mcm96ZW5Db2x1bW4gOiAtMTtcbiAgICBpZiAoZnJvemVuQ29sdW1uSW5kZXggPj0gMCAmJiBmcm96ZW5Db2x1bW5JbmRleCA8PSB0aGlzLl9jb2x1bW5EZWZpbml0aW9ucy5sZW5ndGgpIHtcbiAgICAgIHRoaXMuc2hhcmVkU2VydmljZS5mcm96ZW5WaXNpYmxlQ29sdW1uSWQgPSB0aGlzLl9jb2x1bW5EZWZpbml0aW9uc1tmcm96ZW5Db2x1bW5JbmRleF0uaWQgfHwgJyc7XG4gICAgfVxuXG4gICAgLy8gZ2V0IGFueSBwb3NzaWJsZSBTZXJ2aWNlcyB0aGF0IHVzZXIgd2FudCB0byByZWdpc3RlclxuICAgIHRoaXMucmVnaXN0ZXJSZXNvdXJjZXMoKTtcblxuICAgIC8vIGluaXRpYWxpemUgdGhlIFNsaWNrR3JpZCBncmlkXG4gICAgdGhpcy5zbGlja0dyaWQuaW5pdCgpO1xuXG4gICAgLy8gaW5pdGlhbGl6ZWQgdGhlIHJlc2l6ZXIgc2VydmljZSBvbmx5IGFmdGVyIFNsaWNrR3JpZCBpcyBpbml0aWFsaXplZFxuICAgIC8vIGlmIHdlIGRvbid0IHdlIGVuZCB1cCBiaW5kaW5nIG91ciByZXNpemUgdG8gYSBncmlkIGVsZW1lbnQgdGhhdCBkb2Vzbid0IHlldCBleGlzdCBpbiB0aGUgRE9NIGFuZCB0aGUgcmVzaXplciBzZXJ2aWNlIHdpbGwgZmFpbCBzaWxlbnRseSAoYmVjYXVzZSBpdCBoYXMgYSB0cnkvY2F0Y2ggdGhhdCB1bmJpbmRzIHRoZSByZXNpemUgd2l0aG91dCB0aHJvd2luZyBiYWNrKVxuICAgIGlmICh0aGlzLmdyaWRDb250YWluZXJFbGVtZW50KSB7XG4gICAgICB0aGlzLnJlc2l6ZXJTZXJ2aWNlLmluaXQodGhpcy5zbGlja0dyaWQsIHRoaXMuZ3JpZENvbnRhaW5lckVsZW1lbnQgYXMgSFRNTERpdkVsZW1lbnQpO1xuICAgIH1cblxuICAgIC8vIHVzZXIgY291bGQgc2hvdyBhIGN1c3RvbSBmb290ZXIgd2l0aCB0aGUgZGF0YSBtZXRyaWNzIChkYXRhc2V0IGxlbmd0aCBhbmQgbGFzdCB1cGRhdGVkIHRpbWVzdGFtcClcbiAgICBpZiAoIXRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlUGFnaW5hdGlvbiAmJiB0aGlzLmdyaWRPcHRpb25zLnNob3dDdXN0b21Gb290ZXIgJiYgdGhpcy5ncmlkT3B0aW9ucy5jdXN0b21Gb290ZXJPcHRpb25zICYmIHRoaXMuZ3JpZENvbnRhaW5lckVsZW1lbnQpIHtcbiAgICAgIHRoaXMuc2xpY2tGb290ZXIgPSBuZXcgU2xpY2tGb290ZXJDb21wb25lbnQodGhpcy5zbGlja0dyaWQsIHRoaXMuZ3JpZE9wdGlvbnMuY3VzdG9tRm9vdGVyT3B0aW9ucywgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLCB0aGlzLnRyYW5zbGF0ZXJTZXJ2aWNlKTtcbiAgICAgIHRoaXMuc2xpY2tGb290ZXIucmVuZGVyRm9vdGVyKHRoaXMuZ3JpZENvbnRhaW5lckVsZW1lbnQpO1xuICAgIH1cblxuICAgIGlmICghdGhpcy5jdXN0b21EYXRhVmlldyAmJiB0aGlzLmRhdGFWaWV3KSB7XG4gICAgICAvLyBsb2FkIHRoZSBkYXRhIGluIHRoZSBEYXRhVmlldyAodW5sZXNzIGl0J3MgYSBoaWVyYXJjaGljYWwgZGF0YXNldCwgaWYgc28gaXQgd2lsbCBiZSBsb2FkZWQgYWZ0ZXIgdGhlIGluaXRpYWwgdHJlZSBzb3J0KVxuICAgICAgY29uc3QgaW5pdGlhbERhdGFzZXQgPSB0aGlzLmdyaWRPcHRpb25zPy5lbmFibGVUcmVlRGF0YSA/IHRoaXMuc29ydFRyZWVEYXRhc2V0KHRoaXMuX2RhdGFzZXQpIDogdGhpcy5fZGF0YXNldDtcbiAgICAgIHRoaXMuZGF0YVZpZXcuYmVnaW5VcGRhdGUoKTtcbiAgICAgIHRoaXMuZGF0YVZpZXcuc2V0SXRlbXMoaW5pdGlhbERhdGFzZXQgfHwgW10sIHRoaXMuZ3JpZE9wdGlvbnMuZGF0YXNldElkUHJvcGVydHlOYW1lID8/ICdpZCcpO1xuICAgICAgdGhpcy5kYXRhVmlldy5lbmRVcGRhdGUoKTtcblxuICAgICAgLy8gaWYgeW91IGRvbid0IHdhbnQgdGhlIGl0ZW1zIHRoYXQgYXJlIG5vdCB2aXNpYmxlIChkdWUgdG8gYmVpbmcgZmlsdGVyZWQgb3V0IG9yIGJlaW5nIG9uIGEgZGlmZmVyZW50IHBhZ2UpXG4gICAgICAvLyB0byBzdGF5IHNlbGVjdGVkLCBwYXNzICdmYWxzZScgdG8gdGhlIHNlY29uZCBhcmdcbiAgICAgIGlmICh0aGlzLnNsaWNrR3JpZD8uZ2V0U2VsZWN0aW9uTW9kZWwoKSAmJiB0aGlzLmdyaWRPcHRpb25zICYmIHRoaXMuZ3JpZE9wdGlvbnMuZGF0YVZpZXcgJiYgdGhpcy5ncmlkT3B0aW9ucy5kYXRhVmlldy5oYXNPd25Qcm9wZXJ0eSgnc3luY0dyaWRTZWxlY3Rpb24nKSkge1xuICAgICAgICAvLyBpZiB3ZSBhcmUgdXNpbmcgYSBCYWNrZW5kIFNlcnZpY2UsIHdlIHdpbGwgZG8gYW4gZXh0cmEgZmxhZyBjaGVjaywgdGhlIHJlYXNvbiBpcyBiZWNhdXNlIGl0IG1pZ2h0IGhhdmUgc29tZSB1bmludGVuZGVkIGJlaGF2aW9yc1xuICAgICAgICAvLyB3aXRoIHRoZSBCYWNrZW5kU2VydmljZUFwaSBiZWNhdXNlIHRlY2huaWNhbGx5IHRoZSBkYXRhIGluIHRoZSBwYWdlIGNoYW5nZXMgdGhlIERhdGFWaWV3IG9uIGV2ZXJ5IHBhZ2UgY2hhbmdlLlxuICAgICAgICBsZXQgcHJlc2VydmVkUm93U2VsZWN0aW9uV2l0aEJhY2tlbmQgPSBmYWxzZTtcbiAgICAgICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMuYmFja2VuZFNlcnZpY2VBcGkgJiYgdGhpcy5ncmlkT3B0aW9ucy5kYXRhVmlldy5oYXNPd25Qcm9wZXJ0eSgnc3luY0dyaWRTZWxlY3Rpb25XaXRoQmFja2VuZFNlcnZpY2UnKSkge1xuICAgICAgICAgIHByZXNlcnZlZFJvd1NlbGVjdGlvbldpdGhCYWNrZW5kID0gdGhpcy5ncmlkT3B0aW9ucy5kYXRhVmlldy5zeW5jR3JpZFNlbGVjdGlvbldpdGhCYWNrZW5kU2VydmljZSBhcyBib29sZWFuO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3Qgc3luY0dyaWRTZWxlY3Rpb24gPSB0aGlzLmdyaWRPcHRpb25zLmRhdGFWaWV3LnN5bmNHcmlkU2VsZWN0aW9uO1xuICAgICAgICBpZiAodHlwZW9mIHN5bmNHcmlkU2VsZWN0aW9uID09PSAnYm9vbGVhbicpIHtcbiAgICAgICAgICBsZXQgcHJlc2VydmVkUm93U2VsZWN0aW9uID0gc3luY0dyaWRTZWxlY3Rpb247XG4gICAgICAgICAgaWYgKCF0aGlzLl9pc0xvY2FsR3JpZCkge1xuICAgICAgICAgICAgLy8gd2hlbiB1c2luZyBCYWNrZW5kU2VydmljZUFwaSwgd2UnbGwgYmUgdXNpbmcgdGhlIFwic3luY0dyaWRTZWxlY3Rpb25XaXRoQmFja2VuZFNlcnZpY2VcIiBmbGFnIEJVVCBcInN5bmNHcmlkU2VsZWN0aW9uXCIgbXVzdCBhbHNvIGJlIHNldCB0byBUcnVlXG4gICAgICAgICAgICBwcmVzZXJ2ZWRSb3dTZWxlY3Rpb24gPSBzeW5jR3JpZFNlbGVjdGlvbiAmJiBwcmVzZXJ2ZWRSb3dTZWxlY3Rpb25XaXRoQmFja2VuZDtcbiAgICAgICAgICB9XG4gICAgICAgICAgdGhpcy5kYXRhVmlldy5zeW5jR3JpZFNlbGVjdGlvbih0aGlzLnNsaWNrR3JpZCwgcHJlc2VydmVkUm93U2VsZWN0aW9uKTtcbiAgICAgICAgfSBlbHNlIGlmICh0eXBlb2Ygc3luY0dyaWRTZWxlY3Rpb24gPT09ICdvYmplY3QnKSB7XG4gICAgICAgICAgdGhpcy5kYXRhVmlldy5zeW5jR3JpZFNlbGVjdGlvbih0aGlzLnNsaWNrR3JpZCwgc3luY0dyaWRTZWxlY3Rpb24ucHJlc2VydmVIaWRkZW4sIHN5bmNHcmlkU2VsZWN0aW9uLnByZXNlcnZlSGlkZGVuT25TZWxlY3Rpb25DaGFuZ2UpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGRhdGFzZXRMbiA9IHRoaXMuZGF0YVZpZXcuZ2V0TGVuZ3RoKCkgfHwgdGhpcy5fZGF0YXNldCAmJiB0aGlzLl9kYXRhc2V0Lmxlbmd0aCB8fCAwO1xuICAgICAgaWYgKGRhdGFzZXRMbiA+IDApIHtcbiAgICAgICAgaWYgKCF0aGlzLl9pc0RhdGFzZXRJbml0aWFsaXplZCAmJiAodGhpcy5ncmlkT3B0aW9ucy5lbmFibGVDaGVja2JveFNlbGVjdG9yIHx8IHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlUm93U2VsZWN0aW9uKSkge1xuICAgICAgICAgIHRoaXMubG9hZFJvd1NlbGVjdGlvblByZXNldFdoZW5FeGlzdHMoKTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmxvYWRGaWx0ZXJQcmVzZXRzV2hlbkRhdGFzZXRJbml0aWFsaXplZCgpO1xuICAgICAgICB0aGlzLl9pc0RhdGFzZXRJbml0aWFsaXplZCA9IHRydWU7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gdXNlciBtaWdodCB3YW50IHRvIGhpZGUgdGhlIGhlYWRlciByb3cgb24gcGFnZSBsb2FkIGJ1dCBzdGlsbCBoYXZlIGBlbmFibGVGaWx0ZXJpbmc6IHRydWVgXG4gICAgLy8gaWYgdGhhdCBpcyB0aGUgY2FzZSwgd2UgbmVlZCB0byBoaWRlIHRoZSBoZWFkZXJSb3cgT05MWSBBRlRFUiBhbGwgZmlsdGVycyBnb3QgY3JlYXRlZCAmIGRhdGFWaWV3IGV4aXN0XG4gICAgaWYgKHRoaXMuX2hpZGVIZWFkZXJSb3dBZnRlclBhZ2VMb2FkKSB7XG4gICAgICB0aGlzLnNob3dIZWFkZXJSb3coZmFsc2UpO1xuICAgICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmhpZGVIZWFkZXJSb3dBZnRlclBhZ2VMb2FkID0gdGhpcy5faGlkZUhlYWRlclJvd0FmdGVyUGFnZUxvYWQ7XG4gICAgfVxuXG4gICAgLy8gcHVibGlzaCAmIGRpc3BhdGNoIGNlcnRhaW4gZXZlbnRzXG4gICAgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLnB1Ymxpc2goJ29uR3JpZENyZWF0ZWQnLCB0aGlzLnNsaWNrR3JpZCk7XG5cbiAgICAvLyBhZnRlciB0aGUgRGF0YVZpZXcgaXMgY3JlYXRlZCAmIHVwZGF0ZWQgZXhlY3V0ZSBzb21lIHByb2Nlc3Nlc1xuICAgIGlmICghdGhpcy5jdXN0b21EYXRhVmlldykge1xuICAgICAgdGhpcy5leGVjdXRlQWZ0ZXJEYXRhdmlld0NyZWF0ZWQodGhpcy5zbGlja0dyaWQsIHRoaXMuZ3JpZE9wdGlvbnMpO1xuICAgIH1cblxuICAgIC8vIGJpbmQgcmVzaXplIE9OTFkgYWZ0ZXIgdGhlIGRhdGFWaWV3IGlzIHJlYWR5XG4gICAgdGhpcy5iaW5kUmVzaXplSG9vayh0aGlzLnNsaWNrR3JpZCwgdGhpcy5ncmlkT3B0aW9ucyk7XG5cbiAgICAvLyBiaW5kIHRoZSBCYWNrZW5kIFNlcnZpY2UgQVBJIGNhbGxiYWNrIGZ1bmN0aW9ucyBvbmx5IGFmdGVyIHRoZSBncmlkIGlzIGluaXRpYWxpemVkXG4gICAgLy8gYmVjYXVzZSB0aGUgcHJlUHJvY2VzcygpIGFuZCBvbkluaXQoKSBtaWdodCBnZXQgdHJpZ2dlcmVkXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnM/LmJhY2tlbmRTZXJ2aWNlQXBpKSB7XG4gICAgICB0aGlzLmJpbmRCYWNrZW5kQ2FsbGJhY2tGdW5jdGlvbnModGhpcy5ncmlkT3B0aW9ucyk7XG4gICAgfVxuXG4gICAgLy8gbG9jYWwgZ3JpZCwgY2hlY2sgaWYgd2UgbmVlZCB0byBzaG93IHRoZSBQYWdpbmF0aW9uXG4gICAgLy8gaWYgc28gdGhlbiBhbHNvIGNoZWNrIGlmIHRoZXJlJ3MgYW55IHByZXNldHMgYW5kIGZpbmFsbHkgaW5pdGlhbGl6ZSB0aGUgUGFnaW5hdGlvblNlcnZpY2VcbiAgICAvLyBhIGxvY2FsIGdyaWQgd2l0aCBQYWdpbmF0aW9uIHByZXNldHMgd2lsbCBwb3RlbnRpYWxseSBoYXZlIGEgZGlmZmVyZW50IHRvdGFsIG9mIGl0ZW1zLCB3ZSdsbCBuZWVkIHRvIGdldCBpdCBmcm9tIHRoZSBEYXRhVmlldyBhbmQgdXBkYXRlIG91ciB0b3RhbFxuICAgIGlmICh0aGlzLmdyaWRPcHRpb25zPy5lbmFibGVQYWdpbmF0aW9uICYmIHRoaXMuX2lzTG9jYWxHcmlkKSB7XG4gICAgICB0aGlzLnNob3dQYWdpbmF0aW9uID0gdHJ1ZTtcbiAgICAgIHRoaXMubG9hZExvY2FsR3JpZFBhZ2luYXRpb24odGhpcy5kYXRhc2V0KTtcbiAgICB9XG5cbiAgICB0aGlzLl9hbmd1bGFyR3JpZEluc3RhbmNlcyA9IHtcbiAgICAgIC8vIFNsaWNrIEdyaWQgJiBEYXRhVmlldyBvYmplY3RzXG4gICAgICBkYXRhVmlldzogdGhpcy5kYXRhVmlldyxcbiAgICAgIHNsaWNrR3JpZDogdGhpcy5zbGlja0dyaWQsXG4gICAgICBleHRlbnNpb25zOiB0aGlzLmV4dGVuc2lvblNlcnZpY2U/LmV4dGVuc2lvbkxpc3QsXG5cbiAgICAgIC8vIHB1YmxpYyBtZXRob2RzXG4gICAgICBkZXN0cm95OiB0aGlzLmRlc3Ryb3kuYmluZCh0aGlzKSxcblxuICAgICAgLy8gcmV0dXJuIGFsbCBhdmFpbGFibGUgU2VydmljZXMgKG5vbi1zaW5nbGV0b24pXG4gICAgICBiYWNrZW5kU2VydmljZTogdGhpcy5ncmlkT3B0aW9ucz8uYmFja2VuZFNlcnZpY2VBcGk/LnNlcnZpY2UsXG4gICAgICBldmVudFB1YlN1YlNlcnZpY2U6IHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSxcbiAgICAgIGZpbHRlclNlcnZpY2U6IHRoaXMuZmlsdGVyU2VydmljZSxcbiAgICAgIGdyaWRFdmVudFNlcnZpY2U6IHRoaXMuZ3JpZEV2ZW50U2VydmljZSxcbiAgICAgIGdyaWRTdGF0ZVNlcnZpY2U6IHRoaXMuZ3JpZFN0YXRlU2VydmljZSxcbiAgICAgIGdyaWRTZXJ2aWNlOiB0aGlzLmdyaWRTZXJ2aWNlLFxuICAgICAgZ3JvdXBpbmdTZXJ2aWNlOiB0aGlzLmdyb3VwaW5nU2VydmljZSxcbiAgICAgIGV4dGVuc2lvblNlcnZpY2U6IHRoaXMuZXh0ZW5zaW9uU2VydmljZSxcbiAgICAgIHBhZ2luYXRpb25TZXJ2aWNlOiB0aGlzLnBhZ2luYXRpb25TZXJ2aWNlLFxuICAgICAgcmVzaXplclNlcnZpY2U6IHRoaXMucmVzaXplclNlcnZpY2UsXG4gICAgICBzb3J0U2VydmljZTogdGhpcy5zb3J0U2VydmljZSxcbiAgICAgIHRyZWVEYXRhU2VydmljZTogdGhpcy50cmVlRGF0YVNlcnZpY2UsXG4gICAgfTtcblxuICAgIC8vIGFsbCBpbnN0YW5jZXMgKFNsaWNrR3JpZCwgRGF0YVZpZXcgJiBhbGwgU2VydmljZXMpXG4gICAgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLnB1Ymxpc2goJ29uQW5ndWxhckdyaWRDcmVhdGVkJywgdGhpcy5fYW5ndWxhckdyaWRJbnN0YW5jZXMpO1xuICB9XG5cbiAgLyoqXG4gICAqIE9uIGEgUGFnaW5hdGlvbiBjaGFuZ2VkLCB3ZSB3aWxsIHRyaWdnZXIgYSBHcmlkIFN0YXRlIGNoYW5nZWQgd2l0aCB0aGUgbmV3IHBhZ2luYXRpb24gaW5mb1xuICAgKiBBbHNvIGlmIHdlIHVzZSBSb3cgU2VsZWN0aW9uIG9yIHRoZSBDaGVja2JveCBTZWxlY3RvciB3aXRoIGEgQmFja2VuZCBTZXJ2aWNlIChPZGF0YSwgR3JhcGhRTCksIHdlIG5lZWQgdG8gcmVzZXQgYW55IHNlbGVjdGlvblxuICAgKi9cbiAgcGFnaW5hdGlvbkNoYW5nZWQocGFnaW5hdGlvbjogU2VydmljZVBhZ2luYXRpb24pIHtcbiAgICBjb25zdCBpc1N5bmNHcmlkU2VsZWN0aW9uRW5hYmxlZCA9IHRoaXMuZ3JpZFN0YXRlU2VydmljZT8ubmVlZFRvUHJlc2VydmVSb3dTZWxlY3Rpb24oKSA/PyBmYWxzZTtcbiAgICBpZiAodGhpcy5zbGlja0dyaWQgJiYgIWlzU3luY0dyaWRTZWxlY3Rpb25FbmFibGVkICYmIHRoaXMuZ3JpZE9wdGlvbnM/LmJhY2tlbmRTZXJ2aWNlQXBpICYmICh0aGlzLmdyaWRPcHRpb25zLmVuYWJsZVJvd1NlbGVjdGlvbiB8fCB0aGlzLmdyaWRPcHRpb25zLmVuYWJsZUNoZWNrYm94U2VsZWN0b3IpKSB7XG4gICAgICB0aGlzLnNsaWNrR3JpZC5zZXRTZWxlY3RlZFJvd3MoW10pO1xuICAgIH1cbiAgICBjb25zdCB7IHBhZ2VOdW1iZXIsIHBhZ2VTaXplIH0gPSBwYWdpbmF0aW9uO1xuICAgIGlmICh0aGlzLnNoYXJlZFNlcnZpY2UpIHtcbiAgICAgIGlmIChwYWdlU2l6ZSAhPT0gdW5kZWZpbmVkICYmIHBhZ2VOdW1iZXIgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICB0aGlzLnNoYXJlZFNlcnZpY2UuY3VycmVudFBhZ2luYXRpb24gPSB7IHBhZ2VOdW1iZXIsIHBhZ2VTaXplIH07XG4gICAgICB9XG4gICAgfVxuICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5wdWJsaXNoKCdvbkdyaWRTdGF0ZUNoYW5nZWQnLCB7XG4gICAgICBjaGFuZ2U6IHsgbmV3VmFsdWVzOiB7IHBhZ2VOdW1iZXIsIHBhZ2VTaXplIH0sIHR5cGU6IEdyaWRTdGF0ZVR5cGUucGFnaW5hdGlvbiB9LFxuICAgICAgZ3JpZFN0YXRlOiB0aGlzLmdyaWRTdGF0ZVNlcnZpY2UuZ2V0Q3VycmVudEdyaWRTdGF0ZSgpXG4gICAgfSk7XG4gICAgdGhpcy5jZC5tYXJrRm9yQ2hlY2soKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBXaGVuIGRhdGFzZXQgY2hhbmdlcywgd2UgbmVlZCB0byByZWZyZXNoIHRoZSBlbnRpcmUgZ3JpZCBVSSAmIHBvc3NpYmx5IHJlc2l6ZSBpdCBhcyB3ZWxsXG4gICAqIEBwYXJhbSBkYXRhc2V0XG4gICAqL1xuICByZWZyZXNoR3JpZERhdGEoZGF0YXNldDogYW55W10sIHRvdGFsQ291bnQ/OiBudW1iZXIpIHtcbiAgICBpZiAodGhpcy5ncmlkT3B0aW9ucyAmJiB0aGlzLmdyaWRPcHRpb25zLmVuYWJsZUVtcHR5RGF0YVdhcm5pbmdNZXNzYWdlICYmIEFycmF5LmlzQXJyYXkoZGF0YXNldCkpIHtcbiAgICAgIGNvbnN0IGZpbmFsVG90YWxDb3VudCA9IHRvdGFsQ291bnQgfHwgZGF0YXNldC5sZW5ndGg7XG4gICAgICB0aGlzLmRpc3BsYXlFbXB0eURhdGFXYXJuaW5nKGZpbmFsVG90YWxDb3VudCA8IDEpO1xuICAgIH1cblxuICAgIGlmIChBcnJheS5pc0FycmF5KGRhdGFzZXQpICYmIHRoaXMuc2xpY2tHcmlkICYmIHRoaXMuZGF0YVZpZXc/LnNldEl0ZW1zKSB7XG4gICAgICB0aGlzLmRhdGFWaWV3LnNldEl0ZW1zKGRhdGFzZXQsIHRoaXMuZ3JpZE9wdGlvbnMuZGF0YXNldElkUHJvcGVydHlOYW1lID8/ICdpZCcpO1xuICAgICAgaWYgKCF0aGlzLmdyaWRPcHRpb25zLmJhY2tlbmRTZXJ2aWNlQXBpICYmICF0aGlzLmdyaWRPcHRpb25zLmVuYWJsZVRyZWVEYXRhKSB7XG4gICAgICAgIHRoaXMuZGF0YVZpZXcucmVTb3J0KCk7XG4gICAgICB9XG5cbiAgICAgIGlmIChkYXRhc2V0Lmxlbmd0aCA+IDApIHtcbiAgICAgICAgaWYgKCF0aGlzLl9pc0RhdGFzZXRJbml0aWFsaXplZCkge1xuICAgICAgICAgIHRoaXMubG9hZEZpbHRlclByZXNldHNXaGVuRGF0YXNldEluaXRpYWxpemVkKCk7XG5cbiAgICAgICAgICBpZiAodGhpcy5ncmlkT3B0aW9ucy5lbmFibGVDaGVja2JveFNlbGVjdG9yKSB7XG4gICAgICAgICAgICB0aGlzLmxvYWRSb3dTZWxlY3Rpb25QcmVzZXRXaGVuRXhpc3RzKCk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHRoaXMuX2lzRGF0YXNldEluaXRpYWxpemVkID0gdHJ1ZTtcbiAgICAgIH1cblxuICAgICAgaWYgKGRhdGFzZXQpIHtcbiAgICAgICAgdGhpcy5zbGlja0dyaWQuaW52YWxpZGF0ZSgpO1xuICAgICAgfVxuXG4gICAgICAvLyBkaXNwbGF5IHRoZSBQYWdpbmF0aW9uIGNvbXBvbmVudCBvbmx5IGFmdGVyIGNhbGxpbmcgdGhpcyByZWZyZXNoIGRhdGEgZmlyc3QsIHdlIGNhbGwgaXQgaGVyZSBzbyB0aGF0IGlmIHdlIHByZXNldCBwYWdpbmF0aW9uIHBhZ2UgbnVtYmVyIGl0IHdpbGwgYmUgc2hvd24gY29ycmVjdGx5XG4gICAgICB0aGlzLnNob3dQYWdpbmF0aW9uID0gKHRoaXMuZ3JpZE9wdGlvbnMgJiYgKHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlUGFnaW5hdGlvbiB8fCAodGhpcy5ncmlkT3B0aW9ucy5iYWNrZW5kU2VydmljZUFwaSAmJiB0aGlzLmdyaWRPcHRpb25zLmVuYWJsZVBhZ2luYXRpb24gPT09IHVuZGVmaW5lZCkpKSA/IHRydWUgOiBmYWxzZTtcblxuICAgICAgaWYgKHRoaXMuX3BhZ2luYXRpb25PcHRpb25zICYmIHRoaXMuZ3JpZE9wdGlvbnM/LnBhZ2luYXRpb24gJiYgdGhpcy5ncmlkT3B0aW9ucz8uYmFja2VuZFNlcnZpY2VBcGkpIHtcbiAgICAgICAgY29uc3QgcGFnaW5hdGlvbk9wdGlvbnMgPSB0aGlzLnNldFBhZ2luYXRpb25PcHRpb25zV2hlblByZXNldERlZmluZWQodGhpcy5ncmlkT3B0aW9ucywgdGhpcy5fcGFnaW5hdGlvbk9wdGlvbnMgYXMgUGFnaW5hdGlvbik7XG4gICAgICAgIC8vIHdoZW4gd2UgaGF2ZSBhIHRvdGFsQ291bnQgdXNlIGl0LCBlbHNlIHdlJ2xsIHRha2UgaXQgZnJvbSB0aGUgcGFnaW5hdGlvbiBvYmplY3RcbiAgICAgICAgLy8gb25seSB1cGRhdGUgdGhlIHRvdGFsIGl0ZW1zIGlmIGl0J3MgZGlmZmVyZW50IHRvIGF2b2lkIHJlZnJlc2hpbmcgdGhlIFVJXG4gICAgICAgIGNvbnN0IHRvdGFsUmVjb3JkcyA9ICh0b3RhbENvdW50ICE9PSB1bmRlZmluZWQpID8gdG90YWxDb3VudCA6ICh0aGlzLmdyaWRPcHRpb25zPy5wYWdpbmF0aW9uPy50b3RhbEl0ZW1zKTtcbiAgICAgICAgaWYgKHRvdGFsUmVjb3JkcyAhPT0gdW5kZWZpbmVkICYmIHRvdGFsUmVjb3JkcyAhPT0gdGhpcy50b3RhbEl0ZW1zKSB7XG4gICAgICAgICAgdGhpcy50b3RhbEl0ZW1zID0gK3RvdGFsUmVjb3JkcztcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIGluaXRpYWxpemUgdGhlIFBhZ2luYXRpb24gU2VydmljZSB3aXRoIG5ldyBwYWdpbmF0aW9uIG9wdGlvbnMgKHdoaWNoIG1pZ2h0IGhhdmUgcHJlc2V0cylcbiAgICAgICAgaWYgKCF0aGlzLl9pc1BhZ2luYXRpb25Jbml0aWFsaXplZCkge1xuICAgICAgICAgIHRoaXMuaW5pdGlhbGl6ZVBhZ2luYXRpb25TZXJ2aWNlKHBhZ2luYXRpb25PcHRpb25zKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyB1cGRhdGUgdGhlIHBhZ2luYXRpb24gc2VydmljZSB3aXRoIHRoZSBuZXcgdG90YWxcbiAgICAgICAgICB0aGlzLnBhZ2luYXRpb25TZXJ2aWNlLnVwZGF0ZVRvdGFsSXRlbXModGhpcy50b3RhbEl0ZW1zKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyByZXNpemUgdGhlIGdyaWQgaW5zaWRlIGEgc2xpZ2h0IHRpbWVvdXQsIGluIGNhc2Ugb3RoZXIgRE9NIGVsZW1lbnQgY2hhbmdlZCBwcmlvciB0byB0aGUgcmVzaXplIChsaWtlIGEgZmlsdGVyL3BhZ2luYXRpb24gY2hhbmdlZClcbiAgICAgIGlmICh0aGlzLnNsaWNrR3JpZCAmJiB0aGlzLmdyaWRPcHRpb25zLmVuYWJsZUF1dG9SZXNpemUpIHtcbiAgICAgICAgY29uc3QgZGVsYXkgPSB0aGlzLmdyaWRPcHRpb25zLmF1dG9SZXNpemUgJiYgdGhpcy5ncmlkT3B0aW9ucy5hdXRvUmVzaXplLmRlbGF5O1xuICAgICAgICB0aGlzLnJlc2l6ZXJTZXJ2aWNlLnJlc2l6ZUdyaWQoZGVsYXkgfHwgMTApO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDaGVjayBpZiB0aGVyZSdzIGFueSBQYWdpbmF0aW9uIFByZXNldHMgZGVmaW5lZCBpbiB0aGUgR3JpZCBPcHRpb25zLFxuICAgKiBpZiB0aGVyZSBhcmUgdGhlbiBsb2FkIHRoZW0gaW4gdGhlIHBhZ2luYXRpb25PcHRpb25zIG9iamVjdFxuICAgKi9cbiAgc2V0UGFnaW5hdGlvbk9wdGlvbnNXaGVuUHJlc2V0RGVmaW5lZChncmlkT3B0aW9uczogR3JpZE9wdGlvbiwgcGFnaW5hdGlvbk9wdGlvbnM6IFBhZ2luYXRpb24pOiBQYWdpbmF0aW9uIHtcbiAgICBpZiAoZ3JpZE9wdGlvbnMucHJlc2V0cz8ucGFnaW5hdGlvbiAmJiBwYWdpbmF0aW9uT3B0aW9ucyAmJiAhdGhpcy5faXNQYWdpbmF0aW9uSW5pdGlhbGl6ZWQpIHtcbiAgICAgIHBhZ2luYXRpb25PcHRpb25zLnBhZ2VTaXplID0gZ3JpZE9wdGlvbnMucHJlc2V0cy5wYWdpbmF0aW9uLnBhZ2VTaXplO1xuICAgICAgcGFnaW5hdGlvbk9wdGlvbnMucGFnZU51bWJlciA9IGdyaWRPcHRpb25zLnByZXNldHMucGFnaW5hdGlvbi5wYWdlTnVtYmVyO1xuICAgIH1cbiAgICByZXR1cm4gcGFnaW5hdGlvbk9wdGlvbnM7XG4gIH1cblxuICAvKipcbiAgICogRHluYW1pY2FsbHkgY2hhbmdlIG9yIHVwZGF0ZSB0aGUgY29sdW1uIGRlZmluaXRpb25zIGxpc3QuXG4gICAqIFdlIHdpbGwgcmUtcmVuZGVyIHRoZSBncmlkIHNvIHRoYXQgdGhlIG5ldyBoZWFkZXIgYW5kIGRhdGEgc2hvd3MgdXAgY29ycmVjdGx5LlxuICAgKiBJZiB1c2luZyBpMThuLCB3ZSBhbHNvIG5lZWQgdG8gdHJpZ2dlciBhIHJlLXRyYW5zbGF0ZSBvZiB0aGUgY29sdW1uIGhlYWRlcnNcbiAgICovXG4gIHVwZGF0ZUNvbHVtbkRlZmluaXRpb25zTGlzdChuZXdDb2x1bW5EZWZpbml0aW9uczogQ29sdW1uW10pIHtcbiAgICAvLyBtYXAvc3dhcCB0aGUgaW50ZXJuYWwgbGlicmFyeSBFZGl0b3IgdG8gdGhlIFNsaWNrR3JpZCBFZGl0b3IgZmFjdG9yeVxuICAgIG5ld0NvbHVtbkRlZmluaXRpb25zID0gdGhpcy5zd2FwSW50ZXJuYWxFZGl0b3JUb1NsaWNrR3JpZEZhY3RvcnlFZGl0b3IobmV3Q29sdW1uRGVmaW5pdGlvbnMpO1xuXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlVHJhbnNsYXRlKSB7XG4gICAgICB0aGlzLmV4dGVuc2lvblNlcnZpY2UudHJhbnNsYXRlQ29sdW1uSGVhZGVycyhmYWxzZSwgbmV3Q29sdW1uRGVmaW5pdGlvbnMpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLmV4dGVuc2lvblNlcnZpY2UucmVuZGVyQ29sdW1uSGVhZGVycyhuZXdDb2x1bW5EZWZpbml0aW9ucywgdHJ1ZSk7XG4gICAgfVxuXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnM/LmVuYWJsZUF1dG9TaXplQ29sdW1ucykge1xuICAgICAgdGhpcy5zbGlja0dyaWQuYXV0b3NpemVDb2x1bW5zKCk7XG4gICAgfSBlbHNlIGlmICh0aGlzLmdyaWRPcHRpb25zPy5lbmFibGVBdXRvUmVzaXplQ29sdW1uc0J5Q2VsbENvbnRlbnQgJiYgdGhpcy5yZXNpemVyU2VydmljZT8ucmVzaXplQ29sdW1uc0J5Q2VsbENvbnRlbnQpIHtcbiAgICAgIHRoaXMucmVzaXplclNlcnZpY2UucmVzaXplQ29sdW1uc0J5Q2VsbENvbnRlbnQoKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogU2hvdyB0aGUgZmlsdGVyIHJvdyBkaXNwbGF5ZWQgb24gZmlyc3Qgcm93LCB3ZSBjYW4gb3B0aW9uYWxseSBwYXNzIGZhbHNlIHRvIGhpZGUgaXQuXG4gICAqIEBwYXJhbSBzaG93aW5nXG4gICAqL1xuICBzaG93SGVhZGVyUm93KHNob3dpbmcgPSB0cnVlKSB7XG4gICAgdGhpcy5zbGlja0dyaWQuc2V0SGVhZGVyUm93VmlzaWJpbGl0eShzaG93aW5nLCBmYWxzZSk7XG4gICAgaWYgKHNob3dpbmcgPT09IHRydWUgJiYgdGhpcy5faXNHcmlkSW5pdGlhbGl6ZWQpIHtcbiAgICAgIHRoaXMuc2xpY2tHcmlkLnNldENvbHVtbnModGhpcy5jb2x1bW5EZWZpbml0aW9ucyk7XG4gICAgfVxuICAgIHJldHVybiBzaG93aW5nO1xuICB9XG5cbiAgLy9cbiAgLy8gcHJpdmF0ZSBmdW5jdGlvbnNcbiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tXG5cbiAgLyoqXG4gICAqIExvb3AgdGhyb3VnaCBhbGwgY29sdW1uIGRlZmluaXRpb25zIGFuZCBjb3B5IHRoZSBvcmlnaW5hbCBvcHRpb25hbCBgd2lkdGhgIHByb3BlcnRpZXMgb3B0aW9uYWxseSBwcm92aWRlZCBieSB0aGUgdXNlci5cbiAgICogV2Ugd2lsbCB1c2UgdGhpcyB3aGVuIGRvaW5nIGEgcmVzaXplIGJ5IGNlbGwgY29udGVudCwgaWYgdXNlciBwcm92aWRlZCBhIGB3aWR0aGAgaXQgd29uJ3Qgb3ZlcnJpZGUgaXQuXG4gICAqL1xuICBwcml2YXRlIGNvcHlDb2x1bW5XaWR0aHNSZWZlcmVuY2UoY29sdW1uRGVmaW5pdGlvbnM6IENvbHVtbltdKSB7XG4gICAgY29sdW1uRGVmaW5pdGlvbnMuZm9yRWFjaChjb2wgPT4gY29sLm9yaWdpbmFsV2lkdGggPSBjb2wud2lkdGgpO1xuICB9XG5cbiAgcHJpdmF0ZSBkaXNwbGF5RW1wdHlEYXRhV2FybmluZyhzaG93V2FybmluZyA9IHRydWUpIHtcbiAgICB0aGlzLnNsaWNrRW1wdHlXYXJuaW5nPy5zaG93RW1wdHlEYXRhTWVzc2FnZShzaG93V2FybmluZyk7XG4gIH1cblxuICBwcml2YXRlIGJpbmREaWZmZXJlbnRIb29rcyhncmlkOiBTbGlja0dyaWQsIGdyaWRPcHRpb25zOiBHcmlkT3B0aW9uLCBkYXRhVmlldzogU2xpY2tEYXRhVmlldykge1xuICAgIC8vIG9uIGxvY2FsZSBjaGFuZ2UsIHdlIGhhdmUgdG8gbWFudWFsbHkgdHJhbnNsYXRlIHRoZSBIZWFkZXJzLCBHcmlkTWVudVxuICAgIGlmICh0aGlzLnRyYW5zbGF0ZT8ub25MYW5nQ2hhbmdlKSB7XG4gICAgICAvLyB0cmFuc2xhdGUgc29tZSBvZiB0aGVtIG9uIGZpcnN0IGxvYWQsIHRoZW4gb24gZWFjaCBsYW5ndWFnZSBjaGFuZ2VcbiAgICAgIGlmIChncmlkT3B0aW9ucy5lbmFibGVUcmFuc2xhdGUpIHtcbiAgICAgICAgdGhpcy5leHRlbnNpb25TZXJ2aWNlLnRyYW5zbGF0ZUFsbEV4dGVuc2lvbnMoKTtcbiAgICAgICAgdGhpcy50cmFuc2xhdGVDb2x1bW5IZWFkZXJUaXRsZUtleXMoKTtcbiAgICAgICAgdGhpcy50cmFuc2xhdGVDb2x1bW5Hcm91cEtleXMoKTtcbiAgICAgIH1cblxuICAgICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2goXG4gICAgICAgIHRoaXMudHJhbnNsYXRlLm9uTGFuZ0NoYW5nZS5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgICAgIC8vIHB1Ymxpc2ggZXZlbnQgb2YgdGhlIHNhbWUgbmFtZSB0aGF0IFNsaWNrZ3JpZC1Vbml2ZXJzYWwgdXNlcyBvbiBhIGxhbmd1YWdlIGNoYW5nZSBldmVudFxuICAgICAgICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5wdWJsaXNoKCdvbkxhbmd1YWdlQ2hhbmdlJyk7XG5cbiAgICAgICAgICBpZiAoZ3JpZE9wdGlvbnMuZW5hYmxlVHJhbnNsYXRlKSB7XG4gICAgICAgICAgICB0aGlzLmV4dGVuc2lvblNlcnZpY2UudHJhbnNsYXRlQWxsRXh0ZW5zaW9ucygpO1xuICAgICAgICAgICAgdGhpcy50cmFuc2xhdGVDb2x1bW5IZWFkZXJUaXRsZUtleXMoKTtcbiAgICAgICAgICAgIHRoaXMudHJhbnNsYXRlQ29sdW1uR3JvdXBLZXlzKCk7XG4gICAgICAgICAgICBpZiAoZ3JpZE9wdGlvbnMuY3JlYXRlUHJlSGVhZGVyUGFuZWwgJiYgIWdyaWRPcHRpb25zLmVuYWJsZURyYWdnYWJsZUdyb3VwaW5nKSB7XG4gICAgICAgICAgICAgIHRoaXMuZ3JvdXBpbmdTZXJ2aWNlLnRyYW5zbGF0ZUdyb3VwaW5nQW5kQ29sU3BhbigpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICk7XG4gICAgfVxuXG4gICAgLy8gaWYgdXNlciBzZXQgYW4gb25Jbml0IEJhY2tlbmQsIHdlJ2xsIHJ1biBpdCByaWdodCBhd2F5IChhbmQgaWYgc28sIHdlIGFsc28gbmVlZCB0byBydW4gcHJlUHJvY2VzcywgaW50ZXJuYWxQb3N0UHJvY2VzcyAmIHBvc3RQcm9jZXNzKVxuICAgIGlmIChncmlkT3B0aW9ucy5iYWNrZW5kU2VydmljZUFwaSkge1xuICAgICAgY29uc3QgYmFja2VuZEFwaSA9IGdyaWRPcHRpb25zLmJhY2tlbmRTZXJ2aWNlQXBpO1xuXG4gICAgICBpZiAoYmFja2VuZEFwaT8uc2VydmljZT8uaW5pdCkge1xuICAgICAgICBiYWNrZW5kQXBpLnNlcnZpY2UuaW5pdChiYWNrZW5kQXBpLm9wdGlvbnMsIGdyaWRPcHRpb25zLnBhZ2luYXRpb24sIHRoaXMuc2xpY2tHcmlkLCB0aGlzLnNoYXJlZFNlcnZpY2UpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGlmIChkYXRhVmlldyAmJiBncmlkKSB7XG4gICAgICBjb25zdCBzbGlja2dyaWRFdmVudFByZWZpeCA9IHRoaXMuZ3JpZE9wdGlvbnM/LmRlZmF1bHRTbGlja2dyaWRFdmVudFByZWZpeCA/PyAnJztcblxuICAgICAgLy8gZXhwb3NlIGFsbCBTbGljayBHcmlkIEV2ZW50cyB0aHJvdWdoIGRpc3BhdGNoXG4gICAgICBmb3IgKGNvbnN0IHByb3AgaW4gZ3JpZCkge1xuICAgICAgICBpZiAoZ3JpZC5oYXNPd25Qcm9wZXJ0eShwcm9wKSAmJiBwcm9wLnN0YXJ0c1dpdGgoJ29uJykpIHtcbiAgICAgICAgICBjb25zdCBncmlkRXZlbnROYW1lID0gdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLmdldEV2ZW50TmFtZUJ5TmFtaW5nQ29udmVudGlvbihwcm9wLCBzbGlja2dyaWRFdmVudFByZWZpeCk7XG4gICAgICAgICAgdGhpcy5fZXZlbnRIYW5kbGVyLnN1YnNjcmliZSgoZ3JpZCBhcyBhbnkpW3Byb3BdLCAoZXZlbnQsIGFyZ3MpID0+IHtcbiAgICAgICAgICAgIHJldHVybiB0aGlzLl9ldmVudFB1YlN1YlNlcnZpY2UuZGlzcGF0Y2hDdXN0b21FdmVudChncmlkRXZlbnROYW1lLCB7IGV2ZW50RGF0YTogZXZlbnQsIGFyZ3MgfSk7XG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLy8gZXhwb3NlIGFsbCBTbGljayBEYXRhVmlldyBFdmVudHMgdGhyb3VnaCBkaXNwYXRjaFxuICAgICAgZm9yIChjb25zdCBwcm9wIGluIGRhdGFWaWV3KSB7XG4gICAgICAgIGlmIChkYXRhVmlldy5oYXNPd25Qcm9wZXJ0eShwcm9wKSAmJiBwcm9wLnN0YXJ0c1dpdGgoJ29uJykpIHtcbiAgICAgICAgICB0aGlzLl9ldmVudEhhbmRsZXIuc3Vic2NyaWJlKChkYXRhVmlldyBhcyBhbnkpW3Byb3BdLCAoZXZlbnQsIGFyZ3MpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGRhdGFWaWV3RXZlbnROYW1lID0gdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLmdldEV2ZW50TmFtZUJ5TmFtaW5nQ29udmVudGlvbihwcm9wLCBzbGlja2dyaWRFdmVudFByZWZpeCk7XG4gICAgICAgICAgICByZXR1cm4gdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLmRpc3BhdGNoQ3VzdG9tRXZlbnQoZGF0YVZpZXdFdmVudE5hbWUsIHsgZXZlbnREYXRhOiBldmVudCwgYXJncyB9KTtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyBvbiBjZWxsIGNsaWNrLCBtYWlubHkgdXNlZCB3aXRoIHRoZSBjb2x1bW5EZWYuYWN0aW9uIGNhbGxiYWNrXG4gICAgICB0aGlzLmdyaWRFdmVudFNlcnZpY2UuYmluZE9uQ2VsbENoYW5nZShncmlkKTtcbiAgICAgIHRoaXMuZ3JpZEV2ZW50U2VydmljZS5iaW5kT25DbGljayhncmlkKTtcblxuICAgICAgaWYgKGRhdGFWaWV3ICYmIGdyaWQpIHtcbiAgICAgICAgLy8gYmluZCBleHRlcm5hbCBzb3J0aW5nIChiYWNrZW5kKSB3aGVuIGF2YWlsYWJsZSBvciBkZWZhdWx0IG9uU29ydCAoZGF0YVZpZXcpXG4gICAgICAgIGlmIChncmlkT3B0aW9ucy5lbmFibGVTb3J0aW5nKSB7XG4gICAgICAgICAgLy8gYmluZCBleHRlcm5hbCBzb3J0aW5nIChiYWNrZW5kKSB1bmxlc3Mgc3BlY2lmaWVkIHRvIHVzZSB0aGUgbG9jYWwgb25lXG4gICAgICAgICAgaWYgKGdyaWRPcHRpb25zLmJhY2tlbmRTZXJ2aWNlQXBpICYmICFncmlkT3B0aW9ucy5iYWNrZW5kU2VydmljZUFwaS51c2VMb2NhbFNvcnRpbmcpIHtcbiAgICAgICAgICAgIHRoaXMuc29ydFNlcnZpY2UuYmluZEJhY2tlbmRPblNvcnQoZ3JpZCk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHRoaXMuc29ydFNlcnZpY2UuYmluZExvY2FsT25Tb3J0KGdyaWQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIGJpbmQgZXh0ZXJuYWwgZmlsdGVyIChiYWNrZW5kKSB3aGVuIGF2YWlsYWJsZSBvciBkZWZhdWx0IG9uRmlsdGVyIChkYXRhVmlldylcbiAgICAgICAgaWYgKGdyaWRPcHRpb25zLmVuYWJsZUZpbHRlcmluZykge1xuICAgICAgICAgIHRoaXMuZmlsdGVyU2VydmljZS5pbml0KGdyaWQpO1xuXG4gICAgICAgICAgLy8gYmluZCBleHRlcm5hbCBmaWx0ZXIgKGJhY2tlbmQpIHVubGVzcyBzcGVjaWZpZWQgdG8gdXNlIHRoZSBsb2NhbCBvbmVcbiAgICAgICAgICBpZiAoZ3JpZE9wdGlvbnMuYmFja2VuZFNlcnZpY2VBcGkgJiYgIWdyaWRPcHRpb25zLmJhY2tlbmRTZXJ2aWNlQXBpLnVzZUxvY2FsRmlsdGVyaW5nKSB7XG4gICAgICAgICAgICB0aGlzLmZpbHRlclNlcnZpY2UuYmluZEJhY2tlbmRPbkZpbHRlcihncmlkKTtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgdGhpcy5maWx0ZXJTZXJ2aWNlLmJpbmRMb2NhbE9uRmlsdGVyKGdyaWQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIHdoZW4gY29sdW1uIGFyZSByZW9yZGVyZWQsIHdlIG5lZWQgdG8gdXBkYXRlIHRoZSB2aXNpYmxlQ29sdW1uIGFycmF5XG4gICAgICAgIHRoaXMuX2V2ZW50SGFuZGxlci5zdWJzY3JpYmUoZ3JpZC5vbkNvbHVtbnNSZW9yZGVyZWQsIChfZSwgYXJncykgPT4ge1xuICAgICAgICAgIHRoaXMuc2hhcmVkU2VydmljZS5oYXNDb2x1bW5zUmVvcmRlcmVkID0gdHJ1ZTtcbiAgICAgICAgICB0aGlzLnNoYXJlZFNlcnZpY2UudmlzaWJsZUNvbHVtbnMgPSBhcmdzLmltcGFjdGVkQ29sdW1ucztcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gbG9hZCBhbnkgcHJlc2V0cyBpZiBhbnkgKGFmdGVyIGRhdGFzZXQgaXMgaW5pdGlhbGl6ZWQpXG4gICAgICAgIHRoaXMubG9hZENvbHVtblByZXNldHNXaGVuRGF0YXNldEluaXRpYWxpemVkKCk7XG4gICAgICAgIHRoaXMubG9hZEZpbHRlclByZXNldHNXaGVuRGF0YXNldEluaXRpYWxpemVkKCk7XG5cbiAgICAgICAgLy8gV2hlbiBkYXRhIGNoYW5nZXMgaW4gdGhlIERhdGFWaWV3LCB3ZSBuZWVkIHRvIHJlZnJlc2ggdGhlIG1ldHJpY3MgYW5kL29yIGRpc3BsYXkgYSB3YXJuaW5nIGlmIHRoZSBkYXRhc2V0IGlzIGVtcHR5XG4gICAgICAgIHRoaXMuX2V2ZW50SGFuZGxlci5zdWJzY3JpYmUoZGF0YVZpZXcub25Sb3dDb3VudENoYW5nZWQsICgpID0+IHtcbiAgICAgICAgICBncmlkLmludmFsaWRhdGUoKTtcbiAgICAgICAgICB0aGlzLmhhbmRsZU9uSXRlbUNvdW50Q2hhbmdlZChkYXRhVmlldy5nZXRGaWx0ZXJlZEl0ZW1Db3VudCgpIHx8IDAsIGRhdGFWaWV3LmdldEl0ZW1Db3VudCgpIHx8IDApO1xuICAgICAgICB9KTtcbiAgICAgICAgdGhpcy5fZXZlbnRIYW5kbGVyLnN1YnNjcmliZShkYXRhVmlldy5vblNldEl0ZW1zQ2FsbGVkLCAoX2UsIGFyZ3MpID0+IHtcbiAgICAgICAgICB0aGlzLmhhbmRsZU9uSXRlbUNvdW50Q2hhbmdlZChkYXRhVmlldy5nZXRGaWx0ZXJlZEl0ZW1Db3VudCgpIHx8IDAsIGFyZ3MuaXRlbUNvdW50KTtcblxuICAgICAgICAgIC8vIHdoZW4gdXNlciBoYXMgcmVzaXplIGJ5IGNvbnRlbnQgZW5hYmxlZCwgd2UnbGwgZm9yY2UgYSBmdWxsIHdpZHRoIGNhbGN1bGF0aW9uIHNpbmNlIHdlIGNoYW5nZSBvdXIgZW50aXJlIGRhdGFzZXRcbiAgICAgICAgICBpZiAoYXJncy5pdGVtQ291bnQgPiAwICYmICh0aGlzLmdyaWRPcHRpb25zLmF1dG9zaXplQ29sdW1uc0J5Q2VsbENvbnRlbnRPbkZpcnN0TG9hZCB8fCB0aGlzLmdyaWRPcHRpb25zLmVuYWJsZUF1dG9SZXNpemVDb2x1bW5zQnlDZWxsQ29udGVudCkpIHtcbiAgICAgICAgICAgIHRoaXMucmVzaXplclNlcnZpY2UucmVzaXplQ29sdW1uc0J5Q2VsbENvbnRlbnQoIXRoaXMuZ3JpZE9wdGlvbnM/LnJlc2l6ZUJ5Q29udGVudE9ubHlPbkZpcnN0TG9hZCk7XG4gICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICBpZiAoZ3JpZE9wdGlvbnM/LmVuYWJsZUZpbHRlcmluZyAmJiAhZ3JpZE9wdGlvbnMuZW5hYmxlUm93RGV0YWlsVmlldykge1xuICAgICAgICAgIHRoaXMuX2V2ZW50SGFuZGxlci5zdWJzY3JpYmUoZGF0YVZpZXcub25Sb3dzQ2hhbmdlZCwgKF9lLCBhcmdzKSA9PiB7XG4gICAgICAgICAgICAvLyBmaWx0ZXJpbmcgZGF0YSB3aXRoIGxvY2FsIGRhdGFzZXQgd2lsbCBub3QgYWx3YXlzIHNob3cgY29ycmVjdGx5IHVubGVzcyB3ZSBjYWxsIHRoaXMgdXBkYXRlUm93L3JlbmRlclxuICAgICAgICAgICAgLy8gYWxzbyBkb24ndCB1c2UgXCJpbnZhbGlkYXRlUm93c1wiIHNpbmNlIGl0IGRlc3Ryb3lzIHRoZSBlbnRpcmUgcm93IGFuZCBhcyBiYWQgdXNlciBleHBlcmllbmNlIHdoZW4gdXBkYXRpbmcgYSByb3dcbiAgICAgICAgICAgIC8vIHNlZSBjb21taXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9naGlzY29kaW5nL2F1cmVsaWEtc2xpY2tncmlkL2NvbW1pdC84YzUwM2E0ZDQ1ZmJhMTFjYmQ4ZDhjYzQ2N2ZhZThkMTc3Y2M0ZjYwXG4gICAgICAgICAgICBpZiAoYXJncz8ucm93cyAmJiBBcnJheS5pc0FycmF5KGFyZ3Mucm93cykpIHtcbiAgICAgICAgICAgICAgYXJncy5yb3dzLmZvckVhY2goKHJvdzogbnVtYmVyKSA9PiBncmlkLnVwZGF0ZVJvdyhyb3cpKTtcbiAgICAgICAgICAgICAgZ3JpZC5yZW5kZXIoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIGRpZCB0aGUgdXNlciBhZGQgYSBjb2xzcGFuIGNhbGxiYWNrPyBJZiBzbywgaG9vayBpdCBpbnRvIHRoZSBEYXRhVmlldyBnZXRJdGVtTWV0YWRhdGFcbiAgICBpZiAoZ3JpZE9wdGlvbnMgJiYgZ3JpZE9wdGlvbnMuY29sc3BhbkNhbGxiYWNrICYmIGRhdGFWaWV3ICYmIGRhdGFWaWV3LmdldEl0ZW0gJiYgZGF0YVZpZXcuZ2V0SXRlbU1ldGFkYXRhKSB7XG4gICAgICBkYXRhVmlldy5nZXRJdGVtTWV0YWRhdGEgPSAocm93TnVtYmVyOiBudW1iZXIpID0+IHtcbiAgICAgICAgbGV0IGNhbGxiYWNrUmVzdWx0ID0gbnVsbDtcbiAgICAgICAgaWYgKGdyaWRPcHRpb25zLmNvbHNwYW5DYWxsYmFjayAmJiBncmlkT3B0aW9ucy5jb2xzcGFuQ2FsbGJhY2spIHtcbiAgICAgICAgICBjYWxsYmFja1Jlc3VsdCA9IGdyaWRPcHRpb25zLmNvbHNwYW5DYWxsYmFjayhkYXRhVmlldy5nZXRJdGVtKHJvd051bWJlcikpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBjYWxsYmFja1Jlc3VsdDtcbiAgICAgIH07XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBiaW5kQmFja2VuZENhbGxiYWNrRnVuY3Rpb25zKGdyaWRPcHRpb25zOiBHcmlkT3B0aW9uKSB7XG4gICAgY29uc3QgYmFja2VuZEFwaSA9IGdyaWRPcHRpb25zLmJhY2tlbmRTZXJ2aWNlQXBpO1xuICAgIGNvbnN0IGJhY2tlbmRBcGlTZXJ2aWNlID0gYmFja2VuZEFwaSAmJiBiYWNrZW5kQXBpLnNlcnZpY2U7XG4gICAgY29uc3Qgc2VydmljZU9wdGlvbnM6IEJhY2tlbmRTZXJ2aWNlT3B0aW9uID0gYmFja2VuZEFwaVNlcnZpY2U/Lm9wdGlvbnMgPz8ge307XG4gICAgY29uc3QgaXNFeGVjdXRlQ29tbWFuZE9uSW5pdCA9ICghc2VydmljZU9wdGlvbnMpID8gZmFsc2UgOiAoKHNlcnZpY2VPcHRpb25zICYmIHNlcnZpY2VPcHRpb25zLmhhc093blByb3BlcnR5KCdleGVjdXRlUHJvY2Vzc0NvbW1hbmRPbkluaXQnKSkgPyBzZXJ2aWNlT3B0aW9uc1snZXhlY3V0ZVByb2Nlc3NDb21tYW5kT25Jbml0J10gOiB0cnVlKTtcblxuICAgIGlmIChiYWNrZW5kQXBpU2VydmljZSkge1xuICAgICAgLy8gdXBkYXRlIGJhY2tlbmQgZmlsdGVycyAoaWYgbmVlZCBiZSkgQkVGT1JFIHRoZSBxdWVyeSBydW5zICh2aWEgdGhlIG9uSW5pdCBjb21tYW5kIGEgZmV3IGxpbmVzIGJlbG93KVxuICAgICAgLy8gaWYgdXNlciBlbnRlcmVkIHNvbWUgYW55IFwicHJlc2V0c1wiLCB3ZSBuZWVkIHRvIHJlZmxlY3QgdGhlbSBhbGwgaW4gdGhlIGdyaWRcbiAgICAgIGlmIChncmlkT3B0aW9ucyAmJiBncmlkT3B0aW9ucy5wcmVzZXRzKSB7XG4gICAgICAgIC8vIEZpbHRlcnMgXCJwcmVzZXRzXCJcbiAgICAgICAgaWYgKGJhY2tlbmRBcGlTZXJ2aWNlLnVwZGF0ZUZpbHRlcnMgJiYgQXJyYXkuaXNBcnJheShncmlkT3B0aW9ucy5wcmVzZXRzLmZpbHRlcnMpICYmIGdyaWRPcHRpb25zLnByZXNldHMuZmlsdGVycy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgYmFja2VuZEFwaVNlcnZpY2UudXBkYXRlRmlsdGVycyhncmlkT3B0aW9ucy5wcmVzZXRzLmZpbHRlcnMsIHRydWUpO1xuICAgICAgICB9XG4gICAgICAgIC8vIFNvcnRlcnMgXCJwcmVzZXRzXCJcbiAgICAgICAgaWYgKGJhY2tlbmRBcGlTZXJ2aWNlLnVwZGF0ZVNvcnRlcnMgJiYgQXJyYXkuaXNBcnJheShncmlkT3B0aW9ucy5wcmVzZXRzLnNvcnRlcnMpICYmIGdyaWRPcHRpb25zLnByZXNldHMuc29ydGVycy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgLy8gd2hlbiB1c2luZyBtdWx0aS1jb2x1bW4gc29ydCwgd2UgY2FuIGhhdmUgbXVsdGlwbGUgYnV0IG9uIHNpbmdsZSBzb3J0IHRoZW4gb25seSBncmFiIHRoZSBmaXJzdCBzb3J0IHByb3ZpZGVkXG4gICAgICAgICAgY29uc3Qgc29ydENvbHVtbnMgPSB0aGlzLmdyaWRPcHRpb25zLm11bHRpQ29sdW1uU29ydCA/IGdyaWRPcHRpb25zLnByZXNldHMuc29ydGVycyA6IGdyaWRPcHRpb25zLnByZXNldHMuc29ydGVycy5zbGljZSgwLCAxKTtcbiAgICAgICAgICBiYWNrZW5kQXBpU2VydmljZS51cGRhdGVTb3J0ZXJzKHVuZGVmaW5lZCwgc29ydENvbHVtbnMpO1xuICAgICAgICB9XG4gICAgICAgIC8vIFBhZ2luYXRpb24gXCJwcmVzZXRzXCJcbiAgICAgICAgaWYgKGJhY2tlbmRBcGlTZXJ2aWNlLnVwZGF0ZVBhZ2luYXRpb24gJiYgZ3JpZE9wdGlvbnMucHJlc2V0cy5wYWdpbmF0aW9uKSB7XG4gICAgICAgICAgY29uc3QgeyBwYWdlTnVtYmVyLCBwYWdlU2l6ZSB9ID0gZ3JpZE9wdGlvbnMucHJlc2V0cy5wYWdpbmF0aW9uO1xuICAgICAgICAgIGJhY2tlbmRBcGlTZXJ2aWNlLnVwZGF0ZVBhZ2luYXRpb24ocGFnZU51bWJlciwgcGFnZVNpemUpO1xuICAgICAgICB9XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zdCBjb2x1bW5GaWx0ZXJzID0gdGhpcy5maWx0ZXJTZXJ2aWNlLmdldENvbHVtbkZpbHRlcnMoKTtcbiAgICAgICAgaWYgKGNvbHVtbkZpbHRlcnMgJiYgYmFja2VuZEFwaVNlcnZpY2UudXBkYXRlRmlsdGVycykge1xuICAgICAgICAgIGJhY2tlbmRBcGlTZXJ2aWNlLnVwZGF0ZUZpbHRlcnMoY29sdW1uRmlsdGVycywgZmFsc2UpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIGV4ZWN1dGUgb25Jbml0IGNvbW1hbmQgd2hlbiBuZWNlc3NhcnlcbiAgICAgIGlmIChiYWNrZW5kQXBpICYmIGJhY2tlbmRBcGlTZXJ2aWNlICYmIChiYWNrZW5kQXBpLm9uSW5pdCB8fCBpc0V4ZWN1dGVDb21tYW5kT25Jbml0KSkge1xuICAgICAgICBjb25zdCBxdWVyeSA9ICh0eXBlb2YgYmFja2VuZEFwaVNlcnZpY2UuYnVpbGRRdWVyeSA9PT0gJ2Z1bmN0aW9uJykgPyBiYWNrZW5kQXBpU2VydmljZS5idWlsZFF1ZXJ5KCkgOiAnJztcbiAgICAgICAgY29uc3QgcHJvY2VzcyA9IChpc0V4ZWN1dGVDb21tYW5kT25Jbml0KSA/IChiYWNrZW5kQXBpLnByb2Nlc3MgJiYgYmFja2VuZEFwaS5wcm9jZXNzKHF1ZXJ5KSB8fCBudWxsKSA6IChiYWNrZW5kQXBpLm9uSW5pdCAmJiBiYWNrZW5kQXBpLm9uSW5pdChxdWVyeSkgfHwgbnVsbCk7XG5cbiAgICAgICAgLy8gd3JhcCB0aGlzIGluc2lkZSBhIHNldFRpbWVvdXQgdG8gYXZvaWQgdGltaW5nIGlzc3VlIHNpbmNlIHRoZSBncmlkT3B0aW9ucyBuZWVkcyB0byBiZSByZWFkeSBiZWZvcmUgcnVubmluZyB0aGlzIG9uSW5pdFxuICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICBjb25zdCBiYWNrZW5kVXRpbGl0eVNlcnZpY2UgPSB0aGlzLmJhY2tlbmRVdGlsaXR5U2VydmljZSBhcyBCYWNrZW5kVXRpbGl0eVNlcnZpY2U7XG5cbiAgICAgICAgICAvLyBrZWVwIHN0YXJ0IHRpbWUgJiBlbmQgdGltZXN0YW1wcyAmIHJldHVybiBpdCBhZnRlciBwcm9jZXNzIGV4ZWN1dGlvblxuICAgICAgICAgIGNvbnN0IHN0YXJ0VGltZSA9IG5ldyBEYXRlKCk7XG5cbiAgICAgICAgICAvLyBydW4gYW55IHByZS1wcm9jZXNzLCBpZiBkZWZpbmVkLCBmb3IgZXhhbXBsZSBhIHNwaW5uZXJcbiAgICAgICAgICBpZiAoYmFja2VuZEFwaS5wcmVQcm9jZXNzKSB7XG4gICAgICAgICAgICBiYWNrZW5kQXBpLnByZVByb2Nlc3MoKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyB0aGUgcHJvY2Vzc2VzIGNhbiBiZSBhIFByb21pc2UgKGxpa2UgSHR0cClcbiAgICAgICAgICBjb25zdCB0b3RhbEl0ZW1zID0gdGhpcy5ncmlkT3B0aW9ucz8ucGFnaW5hdGlvbj8udG90YWxJdGVtcyA/PyAwO1xuICAgICAgICAgIGlmIChwcm9jZXNzIGluc3RhbmNlb2YgUHJvbWlzZSkge1xuICAgICAgICAgICAgcHJvY2Vzc1xuICAgICAgICAgICAgICAudGhlbigocHJvY2Vzc1Jlc3VsdDogYW55KSA9PiBiYWNrZW5kVXRpbGl0eVNlcnZpY2UuZXhlY3V0ZUJhY2tlbmRQcm9jZXNzZXNDYWxsYmFjayhzdGFydFRpbWUsIHByb2Nlc3NSZXN1bHQsIGJhY2tlbmRBcGksIHRvdGFsSXRlbXMpKVxuICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yKSA9PiBiYWNrZW5kVXRpbGl0eVNlcnZpY2Uub25CYWNrZW5kRXJyb3IoZXJyb3IsIGJhY2tlbmRBcGkpKTtcbiAgICAgICAgICB9IGVsc2UgaWYgKHByb2Nlc3MgJiYgdGhpcy5yeGpzPy5pc09ic2VydmFibGUocHJvY2VzcykpIHtcbiAgICAgICAgICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgICAgICAgICAocHJvY2VzcyBhcyBPYnNlcnZhYmxlPGFueT4pLnN1YnNjcmliZSh7XG4gICAgICAgICAgICAgICAgbmV4dDogKHByb2Nlc3NSZXN1bHQ6IGFueSkgPT4gYmFja2VuZFV0aWxpdHlTZXJ2aWNlLmV4ZWN1dGVCYWNrZW5kUHJvY2Vzc2VzQ2FsbGJhY2soc3RhcnRUaW1lLCBwcm9jZXNzUmVzdWx0LCBiYWNrZW5kQXBpLCB0b3RhbEl0ZW1zKSxcbiAgICAgICAgICAgICAgICBlcnJvcjogKGVycm9yOiBhbnkpID0+IGJhY2tlbmRVdGlsaXR5U2VydmljZS5vbkJhY2tlbmRFcnJvcihlcnJvciwgYmFja2VuZEFwaSlcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGJpbmRSZXNpemVIb29rKGdyaWQ6IFNsaWNrR3JpZCwgb3B0aW9uczogR3JpZE9wdGlvbikge1xuICAgIGlmICgob3B0aW9ucy5hdXRvRml0Q29sdW1uc09uRmlyc3RMb2FkICYmIG9wdGlvbnMuYXV0b3NpemVDb2x1bW5zQnlDZWxsQ29udGVudE9uRmlyc3RMb2FkKSB8fCAob3B0aW9ucy5lbmFibGVBdXRvU2l6ZUNvbHVtbnMgJiYgb3B0aW9ucy5lbmFibGVBdXRvUmVzaXplQ29sdW1uc0J5Q2VsbENvbnRlbnQpKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFtBbmd1bGFyLVNsaWNrZ3JpZF0gWW91IGNhbm5vdCBlbmFibGUgYm90aCBhdXRvc2l6ZS9maXQgdmlld3BvcnQgJiByZXNpemUgYnkgY29udGVudCwgeW91IG11c3QgY2hvb3NlIHdoaWNoIHJlc2l6ZSB0ZWNobmlxdWUgdG8gdXNlLiBZb3UgY2FuIGVuYWJsZSB0aGVzZSAyIG9wdGlvbnMgKFwiYXV0b0ZpdENvbHVtbnNPbkZpcnN0TG9hZFwiIGFuZCBcImVuYWJsZUF1dG9TaXplQ29sdW1uc1wiKSBPUiB0aGVzZSBvdGhlciAyIG9wdGlvbnMgKFwiYXV0b3NpemVDb2x1bW5zQnlDZWxsQ29udGVudE9uRmlyc3RMb2FkXCIgYW5kIFwiZW5hYmxlQXV0b1Jlc2l6ZUNvbHVtbnNCeUNlbGxDb250ZW50XCIpLmApO1xuICAgIH1cblxuICAgIC8vIGV4cGFuZC9hdXRvZml0IGNvbHVtbnMgb24gZmlyc3QgcGFnZSBsb2FkXG4gICAgaWYgKGdyaWQgJiYgb3B0aW9ucy5hdXRvRml0Q29sdW1uc09uRmlyc3RMb2FkICYmIG9wdGlvbnMuZW5hYmxlQXV0b1NpemVDb2x1bW5zKSB7XG4gICAgICBncmlkLmF1dG9zaXplQ29sdW1ucygpO1xuICAgIH1cblxuICAgIC8vIGF1dG8tcmVzaXplIGdyaWQgb24gYnJvd3NlciByZXNpemVcbiAgICBpZiAob3B0aW9ucy5ncmlkSGVpZ2h0IHx8IG9wdGlvbnMuZ3JpZFdpZHRoKSB7XG4gICAgICB0aGlzLnJlc2l6ZXJTZXJ2aWNlLnJlc2l6ZUdyaWQoMCwgeyBoZWlnaHQ6IG9wdGlvbnMuZ3JpZEhlaWdodCwgd2lkdGg6IG9wdGlvbnMuZ3JpZFdpZHRoIH0pO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLnJlc2l6ZXJTZXJ2aWNlLnJlc2l6ZUdyaWQoKTtcbiAgICB9XG4gICAgaWYgKG9wdGlvbnMuZW5hYmxlQXV0b1Jlc2l6ZSkge1xuICAgICAgaWYgKGdyaWQgJiYgb3B0aW9ucy5hdXRvRml0Q29sdW1uc09uRmlyc3RMb2FkICYmIG9wdGlvbnMuZW5hYmxlQXV0b1NpemVDb2x1bW5zKSB7XG4gICAgICAgIGdyaWQuYXV0b3NpemVDb2x1bW5zKCk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBleGVjdXRlQWZ0ZXJEYXRhdmlld0NyZWF0ZWQoX2dyaWQ6IFNsaWNrR3JpZCwgZ3JpZE9wdGlvbnM6IEdyaWRPcHRpb24pIHtcbiAgICAvLyBpZiB1c2VyIGVudGVyZWQgc29tZSBTb3J0IFwicHJlc2V0c1wiLCB3ZSBuZWVkIHRvIHJlZmxlY3QgdGhlbSBhbGwgaW4gdGhlIERPTVxuICAgIGlmIChncmlkT3B0aW9ucy5lbmFibGVTb3J0aW5nKSB7XG4gICAgICBpZiAoZ3JpZE9wdGlvbnMucHJlc2V0cyAmJiBBcnJheS5pc0FycmF5KGdyaWRPcHRpb25zLnByZXNldHMuc29ydGVycykpIHtcbiAgICAgICAgLy8gd2hlbiB1c2luZyBtdWx0aS1jb2x1bW4gc29ydCwgd2UgY2FuIGhhdmUgbXVsdGlwbGUgYnV0IG9uIHNpbmdsZSBzb3J0IHRoZW4gb25seSBncmFiIHRoZSBmaXJzdCBzb3J0IHByb3ZpZGVkXG4gICAgICAgIGNvbnN0IHNvcnRDb2x1bW5zID0gdGhpcy5ncmlkT3B0aW9ucy5tdWx0aUNvbHVtblNvcnQgPyBncmlkT3B0aW9ucy5wcmVzZXRzLnNvcnRlcnMgOiBncmlkT3B0aW9ucy5wcmVzZXRzLnNvcnRlcnMuc2xpY2UoMCwgMSk7XG4gICAgICAgIHRoaXMuc29ydFNlcnZpY2UubG9hZEdyaWRTb3J0ZXJzKHNvcnRDb2x1bW5zKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvKiogV2hlbiBkYXRhIGNoYW5nZXMgaW4gdGhlIERhdGFWaWV3LCB3ZSdsbCByZWZyZXNoIHRoZSBtZXRyaWNzIGFuZC9vciBkaXNwbGF5IGEgd2FybmluZyBpZiB0aGUgZGF0YXNldCBpcyBlbXB0eSAqL1xuICBwcml2YXRlIGhhbmRsZU9uSXRlbUNvdW50Q2hhbmdlZChjdXJyZW50UGFnZVJvd0l0ZW1Db3VudDogbnVtYmVyLCB0b3RhbEl0ZW1Db3VudDogbnVtYmVyKSB7XG4gICAgdGhpcy5fY3VycmVudERhdGFzZXRMZW5ndGggPSB0b3RhbEl0ZW1Db3VudDtcbiAgICB0aGlzLm1ldHJpY3MgPSB7XG4gICAgICBzdGFydFRpbWU6IG5ldyBEYXRlKCksXG4gICAgICBlbmRUaW1lOiBuZXcgRGF0ZSgpLFxuICAgICAgaXRlbUNvdW50OiBjdXJyZW50UGFnZVJvd0l0ZW1Db3VudCxcbiAgICAgIHRvdGFsSXRlbUNvdW50XG4gICAgfTtcbiAgICAvLyBpZiBjdXN0b20gZm9vdGVyIGlzIGVuYWJsZWQsIHRoZW4gd2UnbGwgdXBkYXRlIGl0cyBtZXRyaWNzXG4gICAgaWYgKHRoaXMuc2xpY2tGb290ZXIpIHtcbiAgICAgIHRoaXMuc2xpY2tGb290ZXIubWV0cmljcyA9IHRoaXMubWV0cmljcztcbiAgICB9XG5cbiAgICAvLyB3aGVuIHVzaW5nIGxvY2FsIChpbi1tZW1vcnkpIGRhdGFzZXQsIHdlJ2xsIGRpc3BsYXkgYSB3YXJuaW5nIG1lc3NhZ2Ugd2hlbiBmaWx0ZXJlZCBkYXRhIGlzIGVtcHR5XG4gICAgaWYgKHRoaXMuX2lzTG9jYWxHcmlkICYmIHRoaXMuZ3JpZE9wdGlvbnM/LmVuYWJsZUVtcHR5RGF0YVdhcm5pbmdNZXNzYWdlKSB7XG4gICAgICB0aGlzLmRpc3BsYXlFbXB0eURhdGFXYXJuaW5nKGN1cnJlbnRQYWdlUm93SXRlbUNvdW50ID09PSAwKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGluaXRpYWxpemVQYWdpbmF0aW9uU2VydmljZShwYWdpbmF0aW9uT3B0aW9uczogUGFnaW5hdGlvbikge1xuICAgIGlmICh0aGlzLmdyaWRPcHRpb25zKSB7XG4gICAgICB0aGlzLnBhZ2luYXRpb25EYXRhID0ge1xuICAgICAgICBncmlkT3B0aW9uczogdGhpcy5ncmlkT3B0aW9ucyxcbiAgICAgICAgcGFnaW5hdGlvblNlcnZpY2U6IHRoaXMucGFnaW5hdGlvblNlcnZpY2UsXG4gICAgICB9O1xuICAgICAgdGhpcy5wYWdpbmF0aW9uU2VydmljZS50b3RhbEl0ZW1zID0gdGhpcy50b3RhbEl0ZW1zO1xuICAgICAgdGhpcy5wYWdpbmF0aW9uU2VydmljZS5pbml0KHRoaXMuc2xpY2tHcmlkLCBwYWdpbmF0aW9uT3B0aW9ucywgdGhpcy5iYWNrZW5kU2VydmljZUFwaSk7XG4gICAgICB0aGlzLnN1YnNjcmlwdGlvbnMucHVzaChcbiAgICAgICAgdGhpcy5fZXZlbnRQdWJTdWJTZXJ2aWNlLnN1YnNjcmliZSgnb25QYWdpbmF0aW9uQ2hhbmdlZCcsIChwYWdpbmF0aW9uQ2hhbmdlczogU2VydmljZVBhZ2luYXRpb24pID0+IHtcbiAgICAgICAgICB0aGlzLnBhZ2luYXRpb25DaGFuZ2VkKHBhZ2luYXRpb25DaGFuZ2VzKTtcbiAgICAgICAgfSksXG4gICAgICAgIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZS5zdWJzY3JpYmUoJ29uUGFnaW5hdGlvblZpc2liaWxpdHlDaGFuZ2VkJywgKHZpc2liaWxpdHk6IHsgdmlzaWJsZTogYm9vbGVhbjsgfSkgPT4ge1xuICAgICAgICAgIHRoaXMuc2hvd1BhZ2luYXRpb24gPSB2aXNpYmlsaXR5Py52aXNpYmxlID8/IGZhbHNlO1xuICAgICAgICAgIGlmICh0aGlzLmdyaWRPcHRpb25zPy5iYWNrZW5kU2VydmljZUFwaSkge1xuICAgICAgICAgICAgdGhpcy5iYWNrZW5kVXRpbGl0eVNlcnZpY2U/LnJlZnJlc2hCYWNrZW5kRGF0YXNldCh0aGlzLmdyaWRPcHRpb25zKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgdGhpcy5yZW5kZXJQYWdpbmF0aW9uKHRoaXMuc2hvd1BhZ2luYXRpb24pO1xuICAgICAgICB9KVxuICAgICAgKTtcbiAgICAgIC8vIGFsc28gaW5pdGlhbGl6ZSAocmVuZGVyKSB0aGUgcGFnaW5hdGlvbiBjb21wb25lbnRcbiAgICAgIHRoaXMucmVuZGVyUGFnaW5hdGlvbigpO1xuICAgICAgdGhpcy5faXNQYWdpbmF0aW9uSW5pdGlhbGl6ZWQgPSB0cnVlO1xuICAgIH1cbiAgICB0aGlzLmNkLmRldGVjdENoYW5nZXMoKTtcbiAgfVxuXG4gIC8qKiBMb2FkIHRoZSBFZGl0b3IgQ29sbGVjdGlvbiBhc3luY2hyb25vdXNseSBhbmQgcmVwbGFjZSB0aGUgXCJjb2xsZWN0aW9uXCIgcHJvcGVydHkgd2hlbiBPYnNlcnZhYmxlIHJlc29sdmVzICovXG4gIHByaXZhdGUgbG9hZEVkaXRvckNvbGxlY3Rpb25Bc3luYyhjb2x1bW46IENvbHVtbikge1xuICAgIGNvbnN0IGNvbGxlY3Rpb25Bc3luYyA9IGNvbHVtbiAmJiBjb2x1bW4uZWRpdG9yICYmIChjb2x1bW4uZWRpdG9yIGFzIENvbHVtbkVkaXRvcikuY29sbGVjdGlvbkFzeW5jO1xuICAgIGlmIChjb2xsZWN0aW9uQXN5bmMgaW5zdGFuY2VvZiBPYnNlcnZhYmxlKSB7XG4gICAgICB0aGlzLnN1YnNjcmlwdGlvbnMucHVzaChcbiAgICAgICAgY29sbGVjdGlvbkFzeW5jLnN1YnNjcmliZSgocmVzb2x2ZWRDb2xsZWN0aW9uKSA9PiB0aGlzLnVwZGF0ZUVkaXRvckNvbGxlY3Rpb24oY29sdW1uLCByZXNvbHZlZENvbGxlY3Rpb24pKVxuICAgICAgKTtcbiAgICB9IGVsc2UgaWYgKGNvbGxlY3Rpb25Bc3luYyBpbnN0YW5jZW9mIFByb21pc2UpIHtcbiAgICAgIC8vIHdhaXQgZm9yIHRoZSBcImNvbGxlY3Rpb25Bc3luY1wiLCBvbmNlIHJlc29sdmVkIHdlIHdpbGwgc2F2ZSBpdCBpbnRvIHRoZSBcImNvbGxlY3Rpb25cIlxuICAgICAgLy8gdGhlIGNvbGxlY3Rpb25Bc3luYyBjYW4gYmUgb2YgMyB0eXBlcyBIdHRwQ2xpZW50LCBIdHRwRmV0Y2ggb3IgYSBQcm9taXNlXG4gICAgICBjb2xsZWN0aW9uQXN5bmMudGhlbigocmVzcG9uc2U6IGFueSB8IGFueVtdKSA9PiB7XG4gICAgICAgIGlmIChBcnJheS5pc0FycmF5KHJlc3BvbnNlKSkge1xuICAgICAgICAgIHRoaXMudXBkYXRlRWRpdG9yQ29sbGVjdGlvbihjb2x1bW4sIHJlc3BvbnNlKTsgLy8gZnJvbSBQcm9taXNlXG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBpbnNlcnREeW5hbWljUHJlc2V0Q29sdW1ucyhjb2x1bW5JZDogc3RyaW5nLCBncmlkUHJlc2V0Q29sdW1uczogQ29sdW1uW10pIHtcbiAgICBpZiAodGhpcy5fY29sdW1uRGVmaW5pdGlvbnMpIHtcbiAgICAgIGNvbnN0IGNvbHVtblBvc2l0aW9uID0gdGhpcy5fY29sdW1uRGVmaW5pdGlvbnMuZmluZEluZGV4KGMgPT4gYy5pZCA9PT0gY29sdW1uSWQpO1xuICAgICAgaWYgKGNvbHVtblBvc2l0aW9uID49IDApIHtcbiAgICAgICAgY29uc3QgZHluQ29sdW1uID0gdGhpcy5fY29sdW1uRGVmaW5pdGlvbnNbY29sdW1uUG9zaXRpb25dO1xuICAgICAgICBpZiAoZHluQ29sdW1uPy5pZCA9PT0gY29sdW1uSWQgJiYgIWdyaWRQcmVzZXRDb2x1bW5zLnNvbWUoYyA9PiBjLmlkID09PSBjb2x1bW5JZCkpIHtcbiAgICAgICAgICBjb2x1bW5Qb3NpdGlvbiA+IDBcbiAgICAgICAgICAgID8gZ3JpZFByZXNldENvbHVtbnMuc3BsaWNlKGNvbHVtblBvc2l0aW9uLCAwLCBkeW5Db2x1bW4pXG4gICAgICAgICAgICA6IGdyaWRQcmVzZXRDb2x1bW5zLnVuc2hpZnQoZHluQ29sdW1uKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKiBMb2FkIGFueSBwb3NzaWJsZSBDb2x1bW5zIEdyaWQgUHJlc2V0cyAqL1xuICBwcml2YXRlIGxvYWRDb2x1bW5QcmVzZXRzV2hlbkRhdGFzZXRJbml0aWFsaXplZCgpIHtcbiAgICAvLyBpZiB1c2VyIGVudGVyZWQgc29tZSBDb2x1bW5zIFwicHJlc2V0c1wiLCB3ZSBuZWVkIHRvIHJlZmxlY3QgdGhlbSBhbGwgaW4gdGhlIGdyaWRcbiAgICBpZiAodGhpcy5ncmlkT3B0aW9ucy5wcmVzZXRzICYmIEFycmF5LmlzQXJyYXkodGhpcy5ncmlkT3B0aW9ucy5wcmVzZXRzLmNvbHVtbnMpICYmIHRoaXMuZ3JpZE9wdGlvbnMucHJlc2V0cy5jb2x1bW5zLmxlbmd0aCA+IDApIHtcbiAgICAgIGNvbnN0IGdyaWRQcmVzZXRDb2x1bW5zOiBDb2x1bW5bXSA9IHRoaXMuZ3JpZFN0YXRlU2VydmljZS5nZXRBc3NvY2lhdGVkR3JpZENvbHVtbnModGhpcy5zbGlja0dyaWQsIHRoaXMuZ3JpZE9wdGlvbnMucHJlc2V0cy5jb2x1bW5zKTtcbiAgICAgIGlmIChncmlkUHJlc2V0Q29sdW1ucyAmJiBBcnJheS5pc0FycmF5KGdyaWRQcmVzZXRDb2x1bW5zKSAmJiBncmlkUHJlc2V0Q29sdW1ucy5sZW5ndGggPiAwICYmIEFycmF5LmlzQXJyYXkodGhpcy5fY29sdW1uRGVmaW5pdGlvbnMpKSB7XG4gICAgICAgIC8vIG1ha2Ugc3VyZSB0aGF0IHRoZSBkeW5hbWljIGNvbHVtbnMgYXJlIGluY2x1ZGVkIGluIHByZXNldHMgKDEuUm93IE1vdmUsIDIuIFJvdyBTZWxlY3Rpb24sIDMuIFJvdyBEZXRhaWwpXG4gICAgICAgIGlmICh0aGlzLmdyaWRPcHRpb25zLmVuYWJsZVJvd01vdmVNYW5hZ2VyKSB7XG4gICAgICAgICAgY29uc3Qgcm1tQ29sSWQgPSB0aGlzLmdyaWRPcHRpb25zPy5yb3dNb3ZlTWFuYWdlcj8uY29sdW1uSWQgPz8gJ19tb3ZlJztcbiAgICAgICAgICB0aGlzLmluc2VydER5bmFtaWNQcmVzZXRDb2x1bW5zKHJtbUNvbElkLCBncmlkUHJlc2V0Q29sdW1ucyk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlQ2hlY2tib3hTZWxlY3Rvcikge1xuICAgICAgICAgIGNvbnN0IGNoa0NvbElkID0gdGhpcy5ncmlkT3B0aW9ucz8uY2hlY2tib3hTZWxlY3Rvcj8uY29sdW1uSWQgPz8gJ19jaGVja2JveF9zZWxlY3Rvcic7XG4gICAgICAgICAgdGhpcy5pbnNlcnREeW5hbWljUHJlc2V0Q29sdW1ucyhjaGtDb2xJZCwgZ3JpZFByZXNldENvbHVtbnMpO1xuICAgICAgICB9XG4gICAgICAgIGlmICh0aGlzLmdyaWRPcHRpb25zLmVuYWJsZVJvd0RldGFpbFZpZXcpIHtcbiAgICAgICAgICBjb25zdCByZHZDb2xJZCA9IHRoaXMuZ3JpZE9wdGlvbnM/LnJvd0RldGFpbFZpZXc/LmNvbHVtbklkID8/ICdfZGV0YWlsX3NlbGVjdG9yJztcbiAgICAgICAgICB0aGlzLmluc2VydER5bmFtaWNQcmVzZXRDb2x1bW5zKHJkdkNvbElkLCBncmlkUHJlc2V0Q29sdW1ucyk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBrZWVwIGNvcHkgdGhlIG9yaWdpbmFsIG9wdGlvbmFsIGB3aWR0aGAgcHJvcGVydGllcyBvcHRpb25hbGx5IHByb3ZpZGVkIGJ5IHRoZSB1c2VyLlxuICAgICAgICAvLyBXZSB3aWxsIHVzZSB0aGlzIHdoZW4gZG9pbmcgYSByZXNpemUgYnkgY2VsbCBjb250ZW50LCBpZiB1c2VyIHByb3ZpZGVkIGEgYHdpZHRoYCBpdCB3b24ndCBvdmVycmlkZSBpdC5cbiAgICAgICAgZ3JpZFByZXNldENvbHVtbnMuZm9yRWFjaChjb2wgPT4gY29sLm9yaWdpbmFsV2lkdGggPSBjb2wud2lkdGgpO1xuXG4gICAgICAgIC8vIGZpbmFsbHkgc2V0IHRoZSBuZXcgcHJlc2V0cyBjb2x1bW5zIChpbmNsdWRpbmcgY2hlY2tib3ggc2VsZWN0b3IgaWYgbmVlZCBiZSlcbiAgICAgICAgdGhpcy5zbGlja0dyaWQuc2V0Q29sdW1ucyhncmlkUHJlc2V0Q29sdW1ucyk7XG4gICAgICAgIHRoaXMuc2hhcmVkU2VydmljZS52aXNpYmxlQ29sdW1ucyA9IGdyaWRQcmVzZXRDb2x1bW5zO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8qKiBMb2FkIGFueSBwb3NzaWJsZSBGaWx0ZXJzIEdyaWQgUHJlc2V0cyAqL1xuICBwcml2YXRlIGxvYWRGaWx0ZXJQcmVzZXRzV2hlbkRhdGFzZXRJbml0aWFsaXplZCgpIHtcbiAgICBpZiAodGhpcy5ncmlkT3B0aW9ucyAmJiAhdGhpcy5jdXN0b21EYXRhVmlldykge1xuICAgICAgLy8gaWYgdXNlciBlbnRlcmVkIHNvbWUgRmlsdGVyIFwicHJlc2V0c1wiLCB3ZSBuZWVkIHRvIHJlZmxlY3QgdGhlbSBhbGwgaW4gdGhlIERPTVxuICAgICAgLy8gYWxzbyBub3RlIHRoYXQgYSBwcmVzZXRzIG9mIFRyZWUgRGF0YSBUb2dnbGluZyB3aWxsIGFsc28gY2FsbCB0aGlzIG1ldGhvZCBiZWNhdXNlIFRyZWUgRGF0YSB0b2dnbGluZyBkb2VzIHdvcmsgd2l0aCBkYXRhIGZpbHRlcmluZ1xuICAgICAgLy8gKGNvbGxhcHNpbmcgYSBwYXJlbnQgd2lsbCBiYXNpY2FsbHkgdXNlIEZpbHRlciBmb3IgaGlkZGluZyAoYWthIGNvbGxhcHNpbmcpIGF3YXkgdGhlIGNoaWxkIHVuZGVybmVhdCBpdClcbiAgICAgIGlmICh0aGlzLmdyaWRPcHRpb25zLnByZXNldHMgJiYgKEFycmF5LmlzQXJyYXkodGhpcy5ncmlkT3B0aW9ucy5wcmVzZXRzLmZpbHRlcnMpIHx8IEFycmF5LmlzQXJyYXkodGhpcy5ncmlkT3B0aW9ucy5wcmVzZXRzPy50cmVlRGF0YT8udG9nZ2xlZEl0ZW1zKSkpIHtcbiAgICAgICAgdGhpcy5maWx0ZXJTZXJ2aWNlLnBvcHVsYXRlQ29sdW1uRmlsdGVyU2VhcmNoVGVybVByZXNldHModGhpcy5ncmlkT3B0aW9ucy5wcmVzZXRzPy5maWx0ZXJzIHx8IFtdKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogbG9jYWwgZ3JpZCwgY2hlY2sgaWYgd2UgbmVlZCB0byBzaG93IHRoZSBQYWdpbmF0aW9uXG4gICAqIGlmIHNvIHRoZW4gYWxzbyBjaGVjayBpZiB0aGVyZSdzIGFueSBwcmVzZXRzIGFuZCBmaW5hbGx5IGluaXRpYWxpemUgdGhlIFBhZ2luYXRpb25TZXJ2aWNlXG4gICAqIGEgbG9jYWwgZ3JpZCB3aXRoIFBhZ2luYXRpb24gcHJlc2V0cyB3aWxsIHBvdGVudGlhbGx5IGhhdmUgYSBkaWZmZXJlbnQgdG90YWwgb2YgaXRlbXMsIHdlJ2xsIG5lZWQgdG8gZ2V0IGl0IGZyb20gdGhlIERhdGFWaWV3IGFuZCB1cGRhdGUgb3VyIHRvdGFsXG4gICAqL1xuICBwcml2YXRlIGxvYWRMb2NhbEdyaWRQYWdpbmF0aW9uKGRhdGFzZXQ/OiBhbnlbXSkge1xuICAgIGlmICh0aGlzLmdyaWRPcHRpb25zICYmIHRoaXMuX3BhZ2luYXRpb25PcHRpb25zKSB7XG4gICAgICB0aGlzLnRvdGFsSXRlbXMgPSBBcnJheS5pc0FycmF5KGRhdGFzZXQpID8gZGF0YXNldC5sZW5ndGggOiAwO1xuICAgICAgaWYgKHRoaXMuX3BhZ2luYXRpb25PcHRpb25zICYmIHRoaXMuZGF0YVZpZXc/LmdldFBhZ2luZ0luZm8pIHtcbiAgICAgICAgY29uc3Qgc2xpY2tQYWdpbmdJbmZvID0gdGhpcy5kYXRhVmlldy5nZXRQYWdpbmdJbmZvKCk7XG4gICAgICAgIGlmIChzbGlja1BhZ2luZ0luZm8/Lmhhc093blByb3BlcnR5KCd0b3RhbFJvd3MnKSAmJiB0aGlzLl9wYWdpbmF0aW9uT3B0aW9ucy50b3RhbEl0ZW1zICE9PSBzbGlja1BhZ2luZ0luZm8udG90YWxSb3dzKSB7XG4gICAgICAgICAgdGhpcy50b3RhbEl0ZW1zID0gc2xpY2tQYWdpbmdJbmZvLnRvdGFsUm93cyB8fCAwO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICB0aGlzLl9wYWdpbmF0aW9uT3B0aW9ucy50b3RhbEl0ZW1zID0gdGhpcy50b3RhbEl0ZW1zO1xuICAgICAgY29uc3QgcGFnaW5hdGlvbk9wdGlvbnMgPSB0aGlzLnNldFBhZ2luYXRpb25PcHRpb25zV2hlblByZXNldERlZmluZWQodGhpcy5ncmlkT3B0aW9ucywgdGhpcy5fcGFnaW5hdGlvbk9wdGlvbnMpO1xuICAgICAgdGhpcy5pbml0aWFsaXplUGFnaW5hdGlvblNlcnZpY2UocGFnaW5hdGlvbk9wdGlvbnMpO1xuICAgIH1cbiAgfVxuXG4gIC8qKiBMb2FkIGFueSBSb3cgU2VsZWN0aW9ucyBpbnRvIHRoZSBEYXRhVmlldyB0aGF0IHdlcmUgcHJlc2V0cyBieSB0aGUgdXNlciAqL1xuICBwcml2YXRlIGxvYWRSb3dTZWxlY3Rpb25QcmVzZXRXaGVuRXhpc3RzKCkge1xuICAgIC8vIGlmIHVzZXIgZW50ZXJlZCBzb21lIFJvdyBTZWxlY3Rpb25zIFwicHJlc2V0c1wiXG4gICAgY29uc3QgcHJlc2V0cyA9IHRoaXMuZ3JpZE9wdGlvbnM/LnByZXNldHM7XG4gICAgY29uc3QgZW5hYmxlUm93U2VsZWN0aW9uID0gdGhpcy5ncmlkT3B0aW9ucyAmJiAodGhpcy5ncmlkT3B0aW9ucy5lbmFibGVDaGVja2JveFNlbGVjdG9yIHx8IHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlUm93U2VsZWN0aW9uKTtcbiAgICBpZiAoZW5hYmxlUm93U2VsZWN0aW9uICYmIHRoaXMuc2xpY2tHcmlkPy5nZXRTZWxlY3Rpb25Nb2RlbCgpICYmIHByZXNldHM/LnJvd1NlbGVjdGlvbiAmJiAoQXJyYXkuaXNBcnJheShwcmVzZXRzLnJvd1NlbGVjdGlvbi5ncmlkUm93SW5kZXhlcykgfHwgQXJyYXkuaXNBcnJheShwcmVzZXRzLnJvd1NlbGVjdGlvbi5kYXRhQ29udGV4dElkcykpKSB7XG4gICAgICBsZXQgZGF0YUNvbnRleHRJZHMgPSBwcmVzZXRzLnJvd1NlbGVjdGlvbi5kYXRhQ29udGV4dElkcztcbiAgICAgIGxldCBncmlkUm93SW5kZXhlcyA9IHByZXNldHMucm93U2VsZWN0aW9uLmdyaWRSb3dJbmRleGVzO1xuXG4gICAgICAvLyBtYXBzIHRoZSBJRHMgdG8gdGhlIEdyaWQgUm93cyBhbmQgdmljZSB2ZXJzYSwgdGhlIFwiZGF0YUNvbnRleHRJZHNcIiBoYXMgcHJlY2VkZW5jZSBvdmVyIHRoZSBvdGhlclxuICAgICAgaWYgKEFycmF5LmlzQXJyYXkoZGF0YUNvbnRleHRJZHMpICYmIGRhdGFDb250ZXh0SWRzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgZ3JpZFJvd0luZGV4ZXMgPSB0aGlzLmRhdGFWaWV3Lm1hcElkc1RvUm93cyhkYXRhQ29udGV4dElkcykgfHwgW107XG4gICAgICB9IGVsc2UgaWYgKEFycmF5LmlzQXJyYXkoZ3JpZFJvd0luZGV4ZXMpICYmIGdyaWRSb3dJbmRleGVzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgZGF0YUNvbnRleHRJZHMgPSB0aGlzLmRhdGFWaWV3Lm1hcFJvd3NUb0lkcyhncmlkUm93SW5kZXhlcykgfHwgW107XG4gICAgICB9XG5cbiAgICAgIC8vIGFwcGx5IHJvdyBzZWxlY3Rpb24gd2hlbiBkZWZpbmVkIGFzIGdyaWQgcHJlc2V0c1xuICAgICAgaWYgKHRoaXMuc2xpY2tHcmlkICYmIEFycmF5LmlzQXJyYXkoZ3JpZFJvd0luZGV4ZXMpKSB7XG4gICAgICAgIHRoaXMuc2xpY2tHcmlkLnNldFNlbGVjdGVkUm93cyhncmlkUm93SW5kZXhlcyk7XG4gICAgICAgIHRoaXMuZGF0YVZpZXchLnNldFNlbGVjdGVkSWRzKGRhdGFDb250ZXh0SWRzIHx8IFtdLCB7XG4gICAgICAgICAgaXNSb3dCZWluZ0FkZGVkOiB0cnVlLFxuICAgICAgICAgIHNob3VsZFRyaWdnZXJFdmVudDogZmFsc2UsIC8vIGRvIG5vdCB0cmlnZ2VyIHdoZW4gcHJlc2V0dGluZyB0aGUgZ3JpZFxuICAgICAgICAgIGFwcGx5Um93U2VsZWN0aW9uVG9HcmlkOiB0cnVlXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgbWVyZ2VHcmlkT3B0aW9ucyhncmlkT3B0aW9uczogR3JpZE9wdGlvbik6IEdyaWRPcHRpb24ge1xuICAgIGdyaWRPcHRpb25zLmdyaWRJZCA9IHRoaXMuZ3JpZElkO1xuICAgIGdyaWRPcHRpb25zLmdyaWRDb250YWluZXJJZCA9IGBzbGlja0dyaWRDb250YWluZXItJHt0aGlzLmdyaWRJZH1gO1xuXG4gICAgLy8gaWYgd2UgaGF2ZSBhIGJhY2tlbmRTZXJ2aWNlQXBpIGFuZCB0aGUgZW5hYmxlUGFnaW5hdGlvbiBpcyB1bmRlZmluZWQsIHdlJ2xsIGFzc3VtZSB0aGF0IHdlIGRvIHdhbnQgdG8gc2VlIGl0LCBlbHNlIGdldCB0aGF0IGRlZmluZWQgdmFsdWVcbiAgICBncmlkT3B0aW9ucy5lbmFibGVQYWdpbmF0aW9uID0gKChncmlkT3B0aW9ucy5iYWNrZW5kU2VydmljZUFwaSAmJiBncmlkT3B0aW9ucy5lbmFibGVQYWdpbmF0aW9uID09PSB1bmRlZmluZWQpID8gdHJ1ZSA6IGdyaWRPcHRpb25zLmVuYWJsZVBhZ2luYXRpb24pIHx8IGZhbHNlO1xuXG4gICAgLy8gdXNlIGV4dGVuZCB0byBkZWVwIG1lcmdlICYgY29weSB0byBhdm9pZCBpbW11dGFibGUgcHJvcGVydGllcyBiZWluZyBjaGFuZ2VkIGluIEdsb2JhbEdyaWRPcHRpb25zIGFmdGVyIGEgcm91dGUgY2hhbmdlXG4gICAgY29uc3Qgb3B0aW9ucyA9IFNsaWNrLlV0aWxzLmV4dGVuZCh0cnVlLCB7fSwgR2xvYmFsR3JpZE9wdGlvbnMsIHRoaXMuZm9yUm9vdENvbmZpZywgZ3JpZE9wdGlvbnMpIGFzIEdyaWRPcHRpb247XG5cbiAgICAvLyB1c2luZyBjb3B5IGV4dGVuZCB0byBkbyBhIGRlZXAgY2xvbmUgaGFzIGFuIHVud2FudGVkIHNpZGUgb24gb2JqZWN0cyBhbmQgcGFnZVNpemVzIGJ1dCBFUzYgc3ByZWFkIGhhcyBvdGhlciB3b3JzdCBzaWRlIGVmZmVjdHNcbiAgICAvLyBzbyB3ZSB3aWxsIGp1c3Qgb3ZlcndyaXRlIHRoZSBwYWdlU2l6ZXMgd2hlbiBuZWVkZWQsIHRoaXMgaXMgdGhlIG9ubHkgb25lIGNhdXNpbmcgaXNzdWVzIHNvIGZhci5cbiAgICAvLyBPbiBhIGRlZXAgZXh0ZW5kLCBPYmplY3QgYW5kIEFycmF5IGFyZSBleHRlbmRlZCwgYnV0IG9iamVjdCB3cmFwcGVycyBvbiBwcmltaXRpdmUgdHlwZXMgc3VjaCBhcyBTdHJpbmcsIEJvb2xlYW4sIGFuZCBOdW1iZXIgYXJlIG5vdC5cbiAgICBpZiAob3B0aW9ucz8ucGFnaW5hdGlvbiAmJiAoZ3JpZE9wdGlvbnMuZW5hYmxlUGFnaW5hdGlvbiB8fCBncmlkT3B0aW9ucy5iYWNrZW5kU2VydmljZUFwaSkgJiYgKHRoaXMuZm9yUm9vdENvbmZpZy5wYWdpbmF0aW9uIHx8IGdyaWRPcHRpb25zLnBhZ2luYXRpb24pKSB7XG4gICAgICBvcHRpb25zLnBhZ2luYXRpb24ucGFnZVNpemUgPSBncmlkT3B0aW9ucy5wYWdpbmF0aW9uPy5wYWdlU2l6ZSA/PyB0aGlzLmZvclJvb3RDb25maWcucGFnaW5hdGlvbj8ucGFnZVNpemUgPz8gR2xvYmFsR3JpZE9wdGlvbnMucGFnaW5hdGlvbiEucGFnZVNpemU7XG4gICAgICBvcHRpb25zLnBhZ2luYXRpb24ucGFnZVNpemVzID0gZ3JpZE9wdGlvbnMucGFnaW5hdGlvbj8ucGFnZVNpemVzID8/IHRoaXMuZm9yUm9vdENvbmZpZy5wYWdpbmF0aW9uPy5wYWdlU2l6ZXMgPz8gR2xvYmFsR3JpZE9wdGlvbnMucGFnaW5hdGlvbiEucGFnZVNpemVzO1xuICAgIH1cblxuICAgIC8vIGFsc28gbWFrZSBzdXJlIHRvIHNob3cgdGhlIGhlYWRlciByb3cgaWYgdXNlciBoYXZlIGVuYWJsZWQgZmlsdGVyaW5nXG4gICAgdGhpcy5faGlkZUhlYWRlclJvd0FmdGVyUGFnZUxvYWQgPSAob3B0aW9ucy5zaG93SGVhZGVyUm93ID09PSBmYWxzZSk7XG4gICAgaWYgKG9wdGlvbnMuZW5hYmxlRmlsdGVyaW5nICYmICFvcHRpb25zLnNob3dIZWFkZXJSb3cpIHtcbiAgICAgIG9wdGlvbnMuc2hvd0hlYWRlclJvdyA9IG9wdGlvbnMuZW5hYmxlRmlsdGVyaW5nO1xuICAgIH1cblxuICAgIC8vIHdoZW4gd2UgdXNlIFBhZ2luYXRpb24gb24gTG9jYWwgR3JpZCwgaXQgZG9lc24ndCBzZWVtIHRvIHdvcmsgd2l0aG91dCBlbmFibGVGaWx0ZXJpbmdcbiAgICAvLyBzbyB3ZSdsbCBlbmFibGUgdGhlIGZpbHRlcmluZyBidXQgd2UnbGwga2VlcCB0aGUgaGVhZGVyIHJvdyBoaWRkZW5cbiAgICBpZiAob3B0aW9ucyAmJiAhb3B0aW9ucy5lbmFibGVGaWx0ZXJpbmcgJiYgb3B0aW9ucy5lbmFibGVQYWdpbmF0aW9uICYmIHRoaXMuX2lzTG9jYWxHcmlkKSB7XG4gICAgICBvcHRpb25zLmVuYWJsZUZpbHRlcmluZyA9IHRydWU7XG4gICAgICBvcHRpb25zLnNob3dIZWFkZXJSb3cgPSBmYWxzZTtcbiAgICAgIHRoaXMuX2hpZGVIZWFkZXJSb3dBZnRlclBhZ2VMb2FkID0gdHJ1ZTtcbiAgICAgIGlmICh0aGlzLnNoYXJlZFNlcnZpY2UpIHtcbiAgICAgICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmhpZGVIZWFkZXJSb3dBZnRlclBhZ2VMb2FkID0gdHJ1ZTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gb3B0aW9ucztcbiAgfVxuXG4gIC8qKiBQcmUtUmVnaXN0ZXIgYW55IFJlc291cmNlIHRoYXQgZG9uJ3QgcmVxdWlyZSBTbGlja0dyaWQgdG8gYmUgaW5zdGFudGlhdGVkIChmb3IgZXhhbXBsZSBSeEpTIFJlc291cmNlICYgUm93RGV0YWlsKSAqL1xuICBwcml2YXRlIHByZVJlZ2lzdGVyUmVzb3VyY2VzKCkge1xuICAgIHRoaXMuX3JlZ2lzdGVyZWRSZXNvdXJjZXMgPSB0aGlzLmdyaWRPcHRpb25zLnJlZ2lzdGVyRXh0ZXJuYWxSZXNvdXJjZXMgfHwgW107XG5cbiAgICAvLyBBbmd1bGFyLVNsaWNrZ3JpZCByZXF1aXJlcyBSeEpTLCBzbyB3ZSdsbCByZWdpc3RlciBpdCBhcyB0aGUgZmlyc3QgcmVzb3VyY2VcbiAgICB0aGlzLnJlZ2lzdGVyUnhKc1Jlc291cmNlKG5ldyBSeEpzUmVzb3VyY2UoKSBhcyBSeEpzRmFjYWRlKTtcblxuICAgIGlmICh0aGlzLmdyaWRPcHRpb25zLmVuYWJsZVJvd0RldGFpbFZpZXcpIHtcbiAgICAgIHRoaXMuc2xpY2tSb3dEZXRhaWxWaWV3ID0gbmV3IFNsaWNrUm93RGV0YWlsVmlldyh0aGlzLmFuZ3VsYXJVdGlsU2VydmljZSwgdGhpcy5hcHBSZWYsIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSwgdGhpcy5lbG0ubmF0aXZlRWxlbWVudCwgdGhpcy5yeGpzKTtcbiAgICAgIHRoaXMuc2xpY2tSb3dEZXRhaWxWaWV3LmNyZWF0ZSh0aGlzLmNvbHVtbkRlZmluaXRpb25zLCB0aGlzLmdyaWRPcHRpb25zKTtcbiAgICAgIHRoaXMuX3JlZ2lzdGVyZWRSZXNvdXJjZXMucHVzaCh0aGlzLnNsaWNrUm93RGV0YWlsVmlldyk7XG4gICAgICB0aGlzLmV4dGVuc2lvblNlcnZpY2UuYWRkRXh0ZW5zaW9uVG9MaXN0KEV4dGVuc2lvbk5hbWUucm93RGV0YWlsVmlldywgeyBuYW1lOiBFeHRlbnNpb25OYW1lLnJvd0RldGFpbFZpZXcsIGluc3RhbmNlOiB0aGlzLnNsaWNrUm93RGV0YWlsVmlldyB9KTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIHJlZ2lzdGVyUmVzb3VyY2VzKCkge1xuICAgIC8vIGF0IHRoaXMgcG9pbnQsIHdlIGNvbnNpZGVyIGFsbCB0aGUgcmVnaXN0ZXJlZCBzZXJ2aWNlcyBhcyBleHRlcm5hbCBzZXJ2aWNlcywgYW55dGhpbmcgZWxzZSByZWdpc3RlcmVkIGFmdGVyd2FyZCBhcmVuJ3QgZXh0ZXJuYWxcbiAgICBpZiAoQXJyYXkuaXNBcnJheSh0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzKSkge1xuICAgICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmV4dGVybmFsUmVnaXN0ZXJlZFJlc291cmNlcyA9IHRoaXMuX3JlZ2lzdGVyZWRSZXNvdXJjZXM7XG4gICAgfVxuXG4gICAgLy8gcHVzaCBhbGwgb3RoZXIgU2VydmljZXMgdGhhdCB3ZSB3YW50IHRvIGJlIHJlZ2lzdGVyZWRcbiAgICB0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzLnB1c2godGhpcy5ncmlkU2VydmljZSwgdGhpcy5ncmlkU3RhdGVTZXJ2aWNlKTtcblxuICAgIC8vIHdoZW4gdXNpbmcgR3JvdXBpbmcvRHJhZ2dhYmxlR3JvdXBpbmcvQ29sc3BhbiByZWdpc3RlciBpdHMgU2VydmljZVxuICAgIGlmICh0aGlzLmdyaWRPcHRpb25zLmNyZWF0ZVByZUhlYWRlclBhbmVsICYmICF0aGlzLmdyaWRPcHRpb25zLmVuYWJsZURyYWdnYWJsZUdyb3VwaW5nKSB7XG4gICAgICB0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzLnB1c2godGhpcy5ncm91cGluZ1NlcnZpY2UpO1xuICAgIH1cblxuICAgIC8vIHdoZW4gdXNpbmcgVHJlZSBEYXRhIFZpZXcsIHJlZ2lzdGVyIGl0cyBTZXJ2aWNlXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlVHJlZURhdGEpIHtcbiAgICAgIHRoaXMuX3JlZ2lzdGVyZWRSZXNvdXJjZXMucHVzaCh0aGlzLnRyZWVEYXRhU2VydmljZSk7XG4gICAgfVxuXG4gICAgLy8gd2hlbiB1c2VyIGVuYWJsZXMgdHJhbnNsYXRpb24sIHdlIG5lZWQgdG8gdHJhbnNsYXRlIEhlYWRlcnMgb24gZmlyc3QgcGFzcyAmIHN1YnNlcXVlbnRseSBpbiB0aGUgYmluZERpZmZlcmVudEhvb2tzXG4gICAgaWYgKHRoaXMuZ3JpZE9wdGlvbnMuZW5hYmxlVHJhbnNsYXRlKSB7XG4gICAgICB0aGlzLmV4dGVuc2lvblNlcnZpY2UudHJhbnNsYXRlQ29sdW1uSGVhZGVycygpO1xuICAgIH1cblxuICAgIC8vIGFsc28gaW5pdGlhbGl6ZSAocmVuZGVyKSB0aGUgZW1wdHkgd2FybmluZyBjb21wb25lbnRcbiAgICB0aGlzLnNsaWNrRW1wdHlXYXJuaW5nID0gbmV3IFNsaWNrRW1wdHlXYXJuaW5nQ29tcG9uZW50KCk7XG4gICAgdGhpcy5fcmVnaXN0ZXJlZFJlc291cmNlcy5wdXNoKHRoaXMuc2xpY2tFbXB0eVdhcm5pbmcpO1xuXG4gICAgLy8gYmluZCAmIGluaXRpYWxpemUgYWxsIENvbXBvbmVudHMvU2VydmljZXMgdGhhdCB3ZXJlIHRhZ2dlZCBhcyBlbmFibGVkXG4gICAgLy8gcmVnaXN0ZXIgYWxsIHNlcnZpY2VzIGJ5IGV4ZWN1dGluZyB0aGVpciBpbml0IG1ldGhvZCBhbmQgcHJvdmlkaW5nIHRoZW0gd2l0aCB0aGUgR3JpZCBvYmplY3RcbiAgICBpZiAoQXJyYXkuaXNBcnJheSh0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzKSkge1xuICAgICAgZm9yIChjb25zdCByZXNvdXJjZSBvZiB0aGlzLl9yZWdpc3RlcmVkUmVzb3VyY2VzKSB7XG4gICAgICAgIGlmICh0aGlzLnNsaWNrR3JpZCAmJiB0eXBlb2YgcmVzb3VyY2UuaW5pdCA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgICAgIHJlc291cmNlLmluaXQodGhpcy5zbGlja0dyaWQsIHRoaXMuY29udGFpbmVyU2VydmljZSk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvKiogUmVnaXN0ZXIgdGhlIFJ4SlMgUmVzb3VyY2UgaW4gYWxsIG5lY2Vzc2FyeSBzZXJ2aWNlcyB3aGljaCB1c2VzICovXG4gIHByaXZhdGUgcmVnaXN0ZXJSeEpzUmVzb3VyY2UocmVzb3VyY2U6IFJ4SnNGYWNhZGUpIHtcbiAgICB0aGlzLnJ4anMgPSByZXNvdXJjZTtcbiAgICB0aGlzLmJhY2tlbmRVdGlsaXR5U2VydmljZS5hZGRSeEpzUmVzb3VyY2UodGhpcy5yeGpzKTtcbiAgICB0aGlzLmZpbHRlckZhY3RvcnkuYWRkUnhKc1Jlc291cmNlKHRoaXMucnhqcyk7XG4gICAgdGhpcy5maWx0ZXJTZXJ2aWNlLmFkZFJ4SnNSZXNvdXJjZSh0aGlzLnJ4anMpO1xuICAgIHRoaXMuc29ydFNlcnZpY2UuYWRkUnhKc1Jlc291cmNlKHRoaXMucnhqcyk7XG4gICAgdGhpcy5wYWdpbmF0aW9uU2VydmljZS5hZGRSeEpzUmVzb3VyY2UodGhpcy5yeGpzKTtcbiAgICB0aGlzLmNvbnRhaW5lclNlcnZpY2UucmVnaXN0ZXJJbnN0YW5jZSgnUnhKc1Jlc291cmNlJywgdGhpcy5yeGpzKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW5kZXIgKG9yIGRpc3Bvc2UpIHRoZSBQYWdpbmF0aW9uIENvbXBvbmVudCwgdXNlciBjYW4gb3B0aW9uYWxseSBwcm92aWRlIEZhbHNlICh0byBub3Qgc2hvdyBpdCkgd2hpY2ggd2lsbCBpbiB0ZXJtIGRpc3Bvc2Ugb2YgdGhlIFBhZ2luYXRpb24sXG4gICAqIGFsc28gd2hpbGUgZGlzcG9zaW5nIHdlIGNhbiBjaG9vc2UgdG8gb21pdCB0aGUgZGlzcG9zYWJsZSBvZiB0aGUgUGFnaW5hdGlvbiBTZXJ2aWNlIChpZiB3ZSBhcmUgc2ltcGx5IHRvZ2dsaW5nIHRoZSBQYWdpbmF0aW9uLCB3ZSB3YW50IHRvIGtlZXAgdGhlIFNlcnZpY2UgYWxpdmUpXG4gICAqIEBwYXJhbSB7Qm9vbGVhbn0gc2hvd1BhZ2luYXRpb24gLSBzaG93IChuZXcgcmVuZGVyKSBvciBub3QgKGRpc3Bvc2UpIHRoZSBQYWdpbmF0aW9uXG4gICAqIEBwYXJhbSB7Qm9vbGVhbn0gc2hvdWxkRGlzcG9zZVBhZ2luYXRpb25TZXJ2aWNlIC0gd2hlbiBkaXNwb3NpbmcgdGhlIFBhZ2luYXRpb24sIGRvIHdlIGFsc28gd2FudCB0byBkaXNwb3NlIG9mIHRoZSBQYWdpbmF0aW9uIFNlcnZpY2U/IChkZWZhdWx0cyB0byBUcnVlKVxuICAgKi9cbiAgcHJpdmF0ZSByZW5kZXJQYWdpbmF0aW9uKHNob3dQYWdpbmF0aW9uID0gdHJ1ZSkge1xuICAgIGlmICh0aGlzLmdyaWRPcHRpb25zPy5lbmFibGVQYWdpbmF0aW9uICYmICF0aGlzLl9pc1BhZ2luYXRpb25Jbml0aWFsaXplZCAmJiBzaG93UGFnaW5hdGlvbikge1xuICAgICAgdGhpcy5zbGlja1BhZ2luYXRpb24gPSBuZXcgU2xpY2tQYWdpbmF0aW9uQ29tcG9uZW50KHRoaXMucGFnaW5hdGlvblNlcnZpY2UsIHRoaXMuX2V2ZW50UHViU3ViU2VydmljZSwgdGhpcy5zaGFyZWRTZXJ2aWNlLCB0aGlzLnRyYW5zbGF0ZXJTZXJ2aWNlKTtcbiAgICAgIHRoaXMuc2xpY2tQYWdpbmF0aW9uLnJlbmRlclBhZ2luYXRpb24odGhpcy5ncmlkQ29udGFpbmVyRWxlbWVudCBhcyBIVE1MRWxlbWVudCk7XG4gICAgICB0aGlzLl9pc1BhZ2luYXRpb25Jbml0aWFsaXplZCA9IHRydWU7XG4gICAgfSBlbHNlIGlmICghc2hvd1BhZ2luYXRpb24pIHtcbiAgICAgIGlmICh0aGlzLnNsaWNrUGFnaW5hdGlvbikge1xuICAgICAgICB0aGlzLnNsaWNrUGFnaW5hdGlvbi5kaXNwb3NlKCk7XG4gICAgICB9XG4gICAgICB0aGlzLl9pc1BhZ2luYXRpb25Jbml0aWFsaXplZCA9IGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBUYWtlcyBhIGZsYXQgZGF0YXNldCB3aXRoIHBhcmVudC9jaGlsZCByZWxhdGlvbnNoaXAsIHNvcnQgaXQgKHZpYSBpdHMgdHJlZSBzdHJ1Y3R1cmUpIGFuZCByZXR1cm4gdGhlIHNvcnRlZCBmbGF0IGFycmF5XG4gICAqIEBwYXJhbSB7QXJyYXk8T2JqZWN0Pn0gZmxhdERhdGFzZXRJbnB1dCAtIGZsYXQgZGF0YXNldCBpbnB1dFxuICAgKiBAcGFyYW0ge0Jvb2xlYW59IGZvcmNlR3JpZFJlZnJlc2ggLSBvcHRpb25hbGx5IGZvcmNlIGEgZnVsbCBncmlkIHJlZnJlc2hcbiAgICogQHJldHVybnMge0FycmF5PE9iamVjdD59IHNvcnQgZmxhdCBwYXJlbnQvY2hpbGQgZGF0YXNldFxuICAgKi9cbiAgcHJpdmF0ZSBzb3J0VHJlZURhdGFzZXQ8VD4oZmxhdERhdGFzZXRJbnB1dDogVFtdLCBmb3JjZUdyaWRSZWZyZXNoID0gZmFsc2UpOiBUW10ge1xuICAgIGNvbnN0IHByZXZEYXRhc2V0TG4gPSB0aGlzLl9jdXJyZW50RGF0YXNldExlbmd0aDtcbiAgICBsZXQgc29ydGVkRGF0YXNldFJlc3VsdDtcbiAgICBsZXQgZmxhdERhdGFzZXRPdXRwdXQ6IGFueVtdID0gW107XG5cbiAgICAvLyBpZiB0aGUgaGllcmFyY2hpY2FsIGRhdGFzZXQgd2FzIGFscmVhZHkgaW5pdGlhbGl6ZWQgdGhlbiBubyBuZWVkIHRvIHJlLWNvbnZlcnQgaXQsIHdlIGNhbiB1c2UgaXQgZGlyZWN0bHkgZnJvbSB0aGUgc2hhcmVkIHNlcnZpY2UgcmVmXG4gICAgaWYgKHRoaXMuX2lzRGF0YXNldEhpZXJhcmNoaWNhbEluaXRpYWxpemVkICYmIHRoaXMuZGF0YXNldEhpZXJhcmNoaWNhbCkge1xuICAgICAgc29ydGVkRGF0YXNldFJlc3VsdCA9IHRoaXMudHJlZURhdGFTZXJ2aWNlLnNvcnRIaWVyYXJjaGljYWxEYXRhc2V0KHRoaXMuZGF0YXNldEhpZXJhcmNoaWNhbCk7XG4gICAgICBmbGF0RGF0YXNldE91dHB1dCA9IHNvcnRlZERhdGFzZXRSZXN1bHQuZmxhdDtcbiAgICB9IGVsc2UgaWYgKEFycmF5LmlzQXJyYXkoZmxhdERhdGFzZXRJbnB1dCkgJiYgZmxhdERhdGFzZXRJbnB1dC5sZW5ndGggPiAwKSB7XG4gICAgICBpZiAodGhpcy5ncmlkT3B0aW9ucz8udHJlZURhdGFPcHRpb25zPy5pbml0aWFsU29ydCkge1xuICAgICAgICAvLyBlbHNlIHdlIG5lZWQgdG8gZmlyc3QgY29udmVydCB0aGUgZmxhdCBkYXRhc2V0IHRvIGEgaGllcmFyY2hpY2FsIGRhdGFzZXQgYW5kIHRoZW4gc29ydFxuICAgICAgICBzb3J0ZWREYXRhc2V0UmVzdWx0ID0gdGhpcy50cmVlRGF0YVNlcnZpY2UuY29udmVydEZsYXRQYXJlbnRDaGlsZFRvVHJlZURhdGFzZXRBbmRTb3J0KGZsYXREYXRhc2V0SW5wdXQsIHRoaXMuX2NvbHVtbkRlZmluaXRpb25zLCB0aGlzLmdyaWRPcHRpb25zKTtcbiAgICAgICAgdGhpcy5zaGFyZWRTZXJ2aWNlLmhpZXJhcmNoaWNhbERhdGFzZXQgPSBzb3J0ZWREYXRhc2V0UmVzdWx0LmhpZXJhcmNoaWNhbDtcbiAgICAgICAgZmxhdERhdGFzZXRPdXRwdXQgPSBzb3J0ZWREYXRhc2V0UmVzdWx0LmZsYXQ7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBlbHNlIHdlIGFzc3VtZSB0aGF0IHRoZSB1c2VyIHByb3ZpZGVkIGFuIGFycmF5IHRoYXQgaXMgYWxyZWFkeSBzb3J0ZWQgKHVzZXIncyByZXNwb25zYWJpbGl0eSlcbiAgICAgICAgLy8gYW5kIHNvIHdlIGNhbiBzaW1wbHkgY29udmVydCB0aGUgYXJyYXkgdG8gYSB0cmVlIHN0cnVjdHVyZSBhbmQgd2UncmUgZG9uZSwgbm8gbmVlZCB0byBzb3J0XG4gICAgICAgIHRoaXMuc2hhcmVkU2VydmljZS5oaWVyYXJjaGljYWxEYXRhc2V0ID0gdGhpcy50cmVlRGF0YVNlcnZpY2UuY29udmVydEZsYXRQYXJlbnRDaGlsZFRvVHJlZURhdGFzZXQoZmxhdERhdGFzZXRJbnB1dCwgdGhpcy5ncmlkT3B0aW9ucyk7XG4gICAgICAgIGZsYXREYXRhc2V0T3V0cHV0ID0gZmxhdERhdGFzZXRJbnB1dCB8fCBbXTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBpZiB3ZSBhZGQvcmVtb3ZlIGl0ZW0ocykgZnJvbSB0aGUgZGF0YXNldCwgd2UgbmVlZCB0byBhbHNvIHJlZnJlc2ggb3VyIHRyZWUgZGF0YSBmaWx0ZXJzXG4gICAgaWYgKGZsYXREYXRhc2V0SW5wdXQubGVuZ3RoID4gMCAmJiAoZm9yY2VHcmlkUmVmcmVzaCB8fCBmbGF0RGF0YXNldElucHV0Lmxlbmd0aCAhPT0gcHJldkRhdGFzZXRMbikpIHtcbiAgICAgIHRoaXMuZmlsdGVyU2VydmljZS5yZWZyZXNoVHJlZURhdGFGaWx0ZXJzKGZsYXREYXRhc2V0T3V0cHV0KTtcbiAgICB9XG5cbiAgICByZXR1cm4gZmxhdERhdGFzZXRPdXRwdXQ7XG4gIH1cblxuICAvKipcbiAgICogRm9yIGNvbnZlbmllbmNlIHRvIHRoZSB1c2VyLCB3ZSBwcm92aWRlIHRoZSBwcm9wZXJ0eSBcImVkaXRvclwiIGFzIGFuIEFuZ3VsYXItU2xpY2tncmlkIGVkaXRvciBjb21wbGV4IG9iamVjdFxuICAgKiBob3dldmVyIFwiZWRpdG9yXCIgaXMgdXNlZCBpbnRlcm5hbGx5IGJ5IFNsaWNrR3JpZCBmb3IgaXQncyBvd24gRWRpdG9yIEZhY3RvcnlcbiAgICogc28gaW4gb3VyIGxpYiB3ZSB3aWxsIHN3YXAgXCJlZGl0b3JcIiBhbmQgY29weSBpdCBpbnRvIGEgbmV3IHByb3BlcnR5IGNhbGxlZCBcImludGVybmFsQ29sdW1uRWRpdG9yXCJcbiAgICogdGhlbiB0YWtlIGJhY2sgXCJlZGl0b3IubW9kZWxcIiBhbmQgbWFrZSBpdCB0aGUgbmV3IFwiZWRpdG9yXCIgc28gdGhhdCBTbGlja0dyaWQgRWRpdG9yIEZhY3Rvcnkgc3RpbGwgd29ya3NcbiAgICovXG4gIHByaXZhdGUgc3dhcEludGVybmFsRWRpdG9yVG9TbGlja0dyaWRGYWN0b3J5RWRpdG9yKGNvbHVtbkRlZmluaXRpb25zOiBDb2x1bW5bXSkge1xuICAgIGlmIChjb2x1bW5EZWZpbml0aW9ucy5zb21lKGNvbCA9PiBgJHtjb2wuaWR9YC5pbmNsdWRlcygnLicpKSkge1xuICAgICAgY29uc29sZS5lcnJvcignW0FuZ3VsYXItU2xpY2tncmlkXSBNYWtlIHN1cmUgdGhhdCBub25lIG9mIHlvdXIgQ29sdW1uIERlZmluaXRpb24gXCJpZFwiIHByb3BlcnR5IGluY2x1ZGVzIGEgZG90IGluIGl0cyBuYW1lIGJlY2F1c2UgdGhhdCB3aWxsIGNhdXNlIHNvbWUgcHJvYmxlbXMgd2l0aCB0aGUgRWRpdG9ycy4gRm9yIGV4YW1wbGUgaWYgeW91ciBjb2x1bW4gZGVmaW5pdGlvbiBcImZpZWxkXCIgcHJvcGVydHkgaXMgXCJ1c2VyLmZpcnN0TmFtZVwiIHRoZW4gdXNlIFwiZmlyc3ROYW1lXCIgYXMgdGhlIGNvbHVtbiBcImlkXCIuJyk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGNvbHVtbkRlZmluaXRpb25zLm1hcCgoY29sdW1uOiBDb2x1bW4gfCBhbnkpID0+IHtcbiAgICAgIC8vIG9uIGV2ZXJ5IEVkaXRvciB0aGF0IGhhdmUgYSBcImNvbGxlY3Rpb25Bc3luY1wiLCByZXNvbHZlIHRoZSBkYXRhIGFuZCBhc3NpZ24gaXQgdG8gdGhlIFwiY29sbGVjdGlvblwiIHByb3BlcnR5XG4gICAgICBpZiAoY29sdW1uPy5lZGl0b3I/LmNvbGxlY3Rpb25Bc3luYykge1xuICAgICAgICB0aGlzLmxvYWRFZGl0b3JDb2xsZWN0aW9uQXN5bmMoY29sdW1uKTtcbiAgICAgIH1cbiAgICAgIHJldHVybiB7IC4uLmNvbHVtbiwgZWRpdG9yOiBjb2x1bW4uZWRpdG9yPy5tb2RlbCwgaW50ZXJuYWxDb2x1bW5FZGl0b3I6IHsgLi4uY29sdW1uLmVkaXRvciB9IH07XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIHRyYW5zbGF0ZUNvbHVtbkhlYWRlclRpdGxlS2V5cygpIHtcbiAgICAvLyB0cmFuc2xhdGUgYWxsIGNvbHVtbnMgKGluY2x1ZGluZyBoaWRkZW4gY29sdW1ucylcbiAgICB0aGlzLmV4dGVuc2lvblV0aWxpdHkudHJhbnNsYXRlSXRlbXModGhpcy5zaGFyZWRTZXJ2aWNlLmFsbENvbHVtbnMsICduYW1lS2V5JywgJ25hbWUnKTtcbiAgfVxuXG4gIHByaXZhdGUgdHJhbnNsYXRlQ29sdW1uR3JvdXBLZXlzKCkge1xuICAgIC8vIHRyYW5zbGF0ZSBhbGwgY29sdW1uIGdyb3VwcyAoaW5jbHVkaW5nIGhpZGRlbiBjb2x1bW5zKVxuICAgIHRoaXMuZXh0ZW5zaW9uVXRpbGl0eS50cmFuc2xhdGVJdGVtcyh0aGlzLnNoYXJlZFNlcnZpY2UuYWxsQ29sdW1ucywgJ2NvbHVtbkdyb3VwS2V5JywgJ2NvbHVtbkdyb3VwJyk7XG4gIH1cblxuICAvKipcbiAgICogVXBkYXRlIHRoZSBcImludGVybmFsQ29sdW1uRWRpdG9yLmNvbGxlY3Rpb25cIiBwcm9wZXJ0eS5cbiAgICogU2luY2UgdGhpcyBpcyBjYWxsZWQgYWZ0ZXIgdGhlIGFzeW5jIGNhbGwgcmVzb2x2ZXMsIHRoZSBwb2ludGVyIHdpbGwgbm90IGJlIHRoZSBzYW1lIGFzIHRoZSBcImNvbHVtblwiIGFyZ3VtZW50IHBhc3NlZC5cbiAgICogT25jZSB3ZSBmb3VuZCB0aGUgbmV3IHBvaW50ZXIsIHdlIHdpbGwgcmVhc3NpZ24gdGhlIFwiZWRpdG9yXCIgYW5kIFwiY29sbGVjdGlvblwiIHRvIHRoZSBcImludGVybmFsQ29sdW1uRWRpdG9yXCIgc28gaXQgaGFzIG5ld2VzdCBjb2xsZWN0aW9uXG4gICAqL1xuICBwcml2YXRlIHVwZGF0ZUVkaXRvckNvbGxlY3Rpb248VCA9IGFueT4oY29sdW1uOiBDb2x1bW48VD4sIG5ld0NvbGxlY3Rpb246IFRbXSkge1xuICAgIChjb2x1bW4uZWRpdG9yIGFzIENvbHVtbkVkaXRvcikuY29sbGVjdGlvbiA9IG5ld0NvbGxlY3Rpb247XG4gICAgKGNvbHVtbi5lZGl0b3IgYXMgQ29sdW1uRWRpdG9yKS5kaXNhYmxlZCA9IGZhbHNlO1xuXG4gICAgLy8gZmluZCB0aGUgbmV3IGNvbHVtbiByZWZlcmVuY2UgcG9pbnRlciAmIHJlLWFzc2lnbiB0aGUgbmV3IGVkaXRvciB0byB0aGUgaW50ZXJuYWxDb2x1bW5FZGl0b3JcbiAgICBpZiAoQXJyYXkuaXNBcnJheSh0aGlzLmNvbHVtbkRlZmluaXRpb25zKSkge1xuICAgICAgY29uc3QgY29sdW1uUmVmID0gdGhpcy5jb2x1bW5EZWZpbml0aW9ucy5maW5kKChjb2w6IENvbHVtbikgPT4gY29sLmlkID09PSBjb2x1bW4uaWQpO1xuICAgICAgaWYgKGNvbHVtblJlZikge1xuICAgICAgICBjb2x1bW5SZWYuaW50ZXJuYWxDb2x1bW5FZGl0b3IgPSBjb2x1bW4uZWRpdG9yIGFzIENvbHVtbkVkaXRvcjtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBnZXQgY3VycmVudCBFZGl0b3IsIHJlbW92ZSBpdCBmcm9tIHRoZSBET00gdGhlbiByZS1lbmFibGUgaXQgYW5kIHJlLXJlbmRlciBpdCB3aXRoIHRoZSBuZXcgY29sbGVjdGlvbi5cbiAgICBjb25zdCBjdXJyZW50RWRpdG9yID0gdGhpcy5zbGlja0dyaWQuZ2V0Q2VsbEVkaXRvcigpIGFzIEF1dG9jb21wbGV0ZXJFZGl0b3IgfCBTZWxlY3RFZGl0b3I7XG4gICAgaWYgKGN1cnJlbnRFZGl0b3I/LmRpc2FibGUgJiYgY3VycmVudEVkaXRvcj8ucmVuZGVyRG9tRWxlbWVudCkge1xuICAgICAgY3VycmVudEVkaXRvci5kZXN0cm95KCk7XG4gICAgICBjdXJyZW50RWRpdG9yLmRpc2FibGUoZmFsc2UpO1xuICAgICAgY3VycmVudEVkaXRvci5yZW5kZXJEb21FbGVtZW50KG5ld0NvbGxlY3Rpb24pO1xuICAgIH1cbiAgfVxufVxuIiwiPGRpdiBpZD1cInNsaWNrR3JpZENvbnRhaW5lci17e2dyaWRJZH19XCIgY2xhc3M9XCJncmlkUGFuZVwiPlxuICA8ZGl2IGF0dHIuaWQ9J3t7Z3JpZElkfX0nIGNsYXNzPVwic2xpY2tncmlkLWNvbnRhaW5lclwiIHN0eWxlPVwid2lkdGg6IDEwMCVcIj5cbiAgPC9kaXY+XG48L2Rpdj4iXX0=