@firestitch/filter 14.0.0 → 14.0.2

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 (269) hide show
  1. package/{package/app → app}/classes/actions-controller.d.ts +41 -41
  2. package/{package/app → app}/components/action-button/action-button.component.d.ts +9 -9
  3. package/{package/app → app}/components/action-kebab-actions/action-kebab-actions.component.d.ts +7 -7
  4. package/{package/app → app}/components/actions/actions.component.d.ts +13 -13
  5. package/{package/app → app}/components/filter/filter.component.d.ts +204 -204
  6. package/{package/app → app}/components/filter-chip/filter-chip.component.d.ts +26 -26
  7. package/{package/app → app}/components/filter-chip-content/filter-chip-content.component.d.ts +18 -18
  8. package/{package/app → app}/components/filter-chips/filter-chips.component.d.ts +9 -9
  9. package/{package/app → app}/components/filter-drawer/filter-drawer.component.d.ts +30 -30
  10. package/{package/app → app}/components/filter-drawer-actions/filter-drawer-actions.component.d.ts +15 -15
  11. package/{package/app → app}/components/filters-item/autocomplete/autocomplete.component.d.ts +14 -14
  12. package/{package/app → app}/components/filters-item/autocompletechips/autocompletechips.component.d.ts +16 -16
  13. package/{package/app → app}/components/filters-item/base-item/base-item.component.d.ts +25 -25
  14. package/{package/app → app}/components/filters-item/checkbox/checkbox.component.d.ts +11 -11
  15. package/{package/app → app}/components/filters-item/chips/chips.component.d.ts +13 -13
  16. package/{package/app → app}/components/filters-item/date/date.component.d.ts +20 -20
  17. package/{package/app → app}/components/filters-item/date-range/date-range.component.d.ts +15 -15
  18. package/{package/app → app}/components/filters-item/filter-item.component.d.ts +39 -39
  19. package/{package/app → app}/components/filters-item/range/range.component.d.ts +15 -15
  20. package/{package/app → app}/components/filters-item/select/backdrop/backdrop.component.d.ts +6 -6
  21. package/{package/app → app}/components/filters-item/select/groups/groups.component.d.ts +12 -12
  22. package/{package/app → app}/components/filters-item/select/multiple/multiple.component.d.ts +15 -15
  23. package/{package/app → app}/components/filters-item/select/select.component.d.ts +23 -23
  24. package/{package/app → app}/components/filters-item/select/simple/simple.component.d.ts +14 -14
  25. package/{package/app → app}/components/filters-item/text/text.component.d.ts +19 -19
  26. package/{package/app → app}/components/filters-item/week/week.component.d.ts +12 -12
  27. package/{package/app → app}/components/saved-filter/index.d.ts +3 -3
  28. package/{package/app → app}/components/saved-filter/saved-filter-edit/index.d.ts +1 -1
  29. package/{package/app → app}/components/saved-filter/saved-filter-edit/saved-filter-edit.component.d.ts +18 -18
  30. package/{package/app → app}/components/saved-filter/saved-filter-manage/index.d.ts +1 -1
  31. package/{package/app → app}/components/saved-filter/saved-filter-manage/saved-filter-manage.component.d.ts +16 -16
  32. package/{package/app → app}/components/saved-filter/saved-filters-menu/index.d.ts +1 -1
  33. package/{package/app → app}/components/saved-filter/saved-filters-menu/saved-filters-menu.component.d.ts +23 -23
  34. package/{package/app → app}/consts/query-param-delimiter.d.ts +1 -1
  35. package/{package/app → app}/directives/focus-to-item/focus-to-item.directive.d.ts +27 -27
  36. package/{package/app → app}/directives/status-bar/status-bar.directive.d.ts +8 -8
  37. package/{package/app → app}/enums/action-mode.enum.d.ts +6 -6
  38. package/{package/app → app}/enums/action-type.enum.d.ts +9 -9
  39. package/{package/app → app}/enums/button-style.d.ts +9 -9
  40. package/{package/app → app}/enums/index.d.ts +6 -6
  41. package/{package/app → app}/enums/item-date-mode.enum.d.ts +5 -5
  42. package/{package/app → app}/enums/item-type.enum.d.ts +15 -15
  43. package/{package/app → app}/enums/picker-view-type.enum.d.ts +7 -7
  44. package/{package/app → app}/fs-filter.module.d.ts +70 -70
  45. package/{package/app → app}/helpers/build-query-params.d.ts +3 -3
  46. package/{package/app → app}/helpers/compare.d.ts +2 -2
  47. package/{package/app → app}/helpers/create-filter-item.d.ts +14 -14
  48. package/{package/app → app}/helpers/find-value.d.ts +1 -1
  49. package/{package/app → app}/helpers/get-range-name.d.ts +1 -1
  50. package/{package/app → app}/helpers/parse-date.d.ts +1 -1
  51. package/{package/app → app}/helpers/parse-item-value-from-stored.d.ts +1 -1
  52. package/{package/app → app}/helpers/query-param-transformers.d.ts +2 -2
  53. package/{package/app → app}/helpers/restore-items.d.ts +12 -12
  54. package/{package/app → app}/injectors/filter-config.d.ts +2 -2
  55. package/{package/app → app}/injectors/filter-drawer-data.d.ts +2 -2
  56. package/{package/app → app}/injectors/filter-drawer-overlay.d.ts +2 -2
  57. package/{package/app → app}/interfaces/action.interface.d.ts +81 -81
  58. package/{package/app → app}/interfaces/config.interface.d.ts +66 -66
  59. package/{package/app → app}/interfaces/external-params.interface.d.ts +3 -3
  60. package/{package/app → app}/interfaces/filter.interface.d.ts +4 -4
  61. package/{package/app → app}/interfaces/index.d.ts +6 -6
  62. package/{package/app → app}/interfaces/items/autocomplete-chips.interface.d.ts +15 -15
  63. package/{package/app → app}/interfaces/items/autocomplete.interface.d.ts +9 -9
  64. package/{package/app → app}/interfaces/items/base.interface.d.ts +29 -29
  65. package/{package/app → app}/interfaces/items/checkbox.interface.d.ts +7 -7
  66. package/{package/app → app}/interfaces/items/chips.interface.d.ts +6 -6
  67. package/{package/app → app}/interfaces/items/date-range.interface.d.ts +8 -8
  68. package/{package/app → app}/interfaces/items/date.interface.d.ts +7 -7
  69. package/{package/app → app}/interfaces/items/range.interface.d.ts +13 -13
  70. package/{package/app → app}/interfaces/items/select.interface.d.ts +20 -20
  71. package/{package/app → app}/interfaces/items/text.interface.d.ts +9 -9
  72. package/{package/app → app}/interfaces/items/week.interface.d.ts +7 -7
  73. package/{package/app → app}/interfaces/saved-filters.interface.d.ts +18 -18
  74. package/{package/app → app}/interfaces/update-filter-item.interface.d.ts +4 -4
  75. package/{package/app → app}/models/action-menu-item.model.d.ts +23 -23
  76. package/{package/app → app}/models/action.model.d.ts +54 -54
  77. package/{package/app → app}/models/filter-config.d.ts +31 -31
  78. package/{package/app → app}/models/items/autocomplete/base-autocomplete-item.d.ts +12 -12
  79. package/{package/app → app}/models/items/autocomplete-chips-item.d.ts +20 -20
  80. package/{package/app → app}/models/items/autocomplete-item.d.ts +11 -11
  81. package/{package/app → app}/models/items/base-item.d.ts +80 -80
  82. package/{package/app → app}/models/items/checkbox-item.d.ts +17 -17
  83. package/{package/app → app}/models/items/chips-item.d.ts +18 -18
  84. package/{package/app → app}/models/items/date/base-date-item.d.ts +14 -14
  85. package/{package/app → app}/models/items/date-item.d.ts +8 -8
  86. package/{package/app → app}/models/items/date-range/base-date-range-item.d.ts +21 -21
  87. package/{package/app → app}/models/items/date-range-item.d.ts +6 -6
  88. package/{package/app → app}/models/items/date-time-item.d.ts +7 -7
  89. package/{package/app → app}/models/items/date-time-range-item.d.ts +6 -6
  90. package/{package/app → app}/models/items/range-item.d.ts +23 -23
  91. package/{package/app → app}/models/items/select/base-select-item.d.ts +13 -13
  92. package/{package/app → app}/models/items/select/multiple-select-item.d.ts +14 -14
  93. package/{package/app → app}/models/items/select/simple-select-item.d.ts +14 -14
  94. package/{package/app → app}/models/items/select-item.d.ts +8 -8
  95. package/{package/app → app}/models/items/text-item.d.ts +15 -15
  96. package/{package/app → app}/models/items/week-item.d.ts +18 -18
  97. package/{package/app → app}/pipes/remove-isolate-value.pipe.d.ts +8 -8
  98. package/{package/app → app}/providers/filter-meta.d.ts +5 -5
  99. package/{package/app → app}/services/external-params/persistance-params-controller.service.d.ts +22 -22
  100. package/{package/app → app}/services/external-params/query-params-controller.service.d.ts +22 -22
  101. package/{package/app → app}/services/external-params/saved-filters-controller.service.d.ts +40 -40
  102. package/{package/app → app}/services/external-params-controller.service.d.ts +42 -42
  103. package/{package/app → app}/services/filter-overlay.service.d.ts +36 -36
  104. package/{package/app → app}/services/focus-controller.service.d.ts +18 -18
  105. package/{package/app → app}/services/items-store.service.d.ts +69 -69
  106. package/{package/esm2020 → esm2020}/app/classes/actions-controller.mjs +126 -126
  107. package/{package/esm2020 → esm2020}/app/components/action-button/action-button.component.mjs +21 -21
  108. package/{package/esm2020 → esm2020}/app/components/action-kebab-actions/action-kebab-actions.component.mjs +16 -16
  109. package/{package/esm2020 → esm2020}/app/components/actions/actions.component.mjs +41 -41
  110. package/{package/esm2020 → esm2020}/app/components/filter/filter.component.mjs +613 -613
  111. package/{package/esm2020 → esm2020}/app/components/filter-chip/filter-chip.component.mjs +85 -85
  112. package/{package/esm2020 → esm2020}/app/components/filter-chip-content/filter-chip-content.component.mjs +78 -78
  113. package/{package/esm2020 → esm2020}/app/components/filter-chips/filter-chips.component.mjs +18 -18
  114. package/{package/esm2020 → esm2020}/app/components/filter-drawer/filter-drawer.component.mjs +70 -70
  115. package/{package/esm2020 → esm2020}/app/components/filter-drawer-actions/filter-drawer-actions.component.mjs +40 -40
  116. package/{package/esm2020 → esm2020}/app/components/filters-item/autocomplete/autocomplete.component.mjs +26 -26
  117. package/{package/esm2020 → esm2020}/app/components/filters-item/autocompletechips/autocompletechips.component.mjs +41 -41
  118. package/{package/esm2020 → esm2020}/app/components/filters-item/base-item/base-item.component.mjs +62 -62
  119. package/{package/esm2020 → esm2020}/app/components/filters-item/checkbox/checkbox.component.mjs +20 -20
  120. package/{package/esm2020 → esm2020}/app/components/filters-item/chips/chips.component.mjs +27 -27
  121. package/{package/esm2020 → esm2020}/app/components/filters-item/date/date.component.mjs +42 -42
  122. package/{package/esm2020 → esm2020}/app/components/filters-item/date-range/date-range.component.mjs +33 -33
  123. package/{package/esm2020 → esm2020}/app/components/filters-item/filter-item.component.mjs +79 -79
  124. package/{package/esm2020 → esm2020}/app/components/filters-item/range/range.component.mjs +44 -44
  125. package/{package/esm2020 → esm2020}/app/components/filters-item/select/backdrop/backdrop.component.mjs +11 -11
  126. package/{package/esm2020 → esm2020}/app/components/filters-item/select/groups/groups.component.mjs +29 -29
  127. package/{package/esm2020 → esm2020}/app/components/filters-item/select/multiple/multiple.component.mjs +56 -56
  128. package/{package/esm2020 → esm2020}/app/components/filters-item/select/select.component.mjs +50 -50
  129. package/{package/esm2020 → esm2020}/app/components/filters-item/select/simple/simple.component.mjs +42 -42
  130. package/{package/esm2020 → esm2020}/app/components/filters-item/text/text.component.mjs +48 -48
  131. package/{package/esm2020 → esm2020}/app/components/filters-item/week/week.component.mjs +24 -24
  132. package/{package/esm2020 → esm2020}/app/components/saved-filter/index.mjs +3 -3
  133. package/{package/esm2020 → esm2020}/app/components/saved-filter/saved-filter-edit/index.mjs +1 -1
  134. package/{package/esm2020 → esm2020}/app/components/saved-filter/saved-filter-edit/saved-filter-edit.component.mjs +49 -49
  135. package/{package/esm2020 → esm2020}/app/components/saved-filter/saved-filter-manage/index.mjs +1 -1
  136. package/{package/esm2020 → esm2020}/app/components/saved-filter/saved-filter-manage/saved-filter-manage.component.mjs +39 -39
  137. package/{package/esm2020 → esm2020}/app/components/saved-filter/saved-filters-menu/index.mjs +1 -1
  138. package/{package/esm2020 → esm2020}/app/components/saved-filter/saved-filters-menu/saved-filters-menu.component.mjs +51 -51
  139. package/{package/esm2020 → esm2020}/app/consts/query-param-delimiter.mjs +1 -1
  140. package/{package/esm2020 → esm2020}/app/directives/focus-to-item/focus-to-item.directive.mjs +129 -129
  141. package/{package/esm2020 → esm2020}/app/directives/status-bar/status-bar.directive.mjs +15 -15
  142. package/{package/esm2020 → esm2020}/app/enums/action-mode.enum.mjs +7 -7
  143. package/{package/esm2020 → esm2020}/app/enums/action-type.enum.mjs +11 -11
  144. package/{package/esm2020 → esm2020}/app/enums/button-style.mjs +10 -10
  145. package/{package/esm2020 → esm2020}/app/enums/index.mjs +6 -6
  146. package/{package/esm2020 → esm2020}/app/enums/item-date-mode.enum.mjs +6 -6
  147. package/{package/esm2020 → esm2020}/app/enums/item-type.enum.mjs +16 -16
  148. package/{package/esm2020 → esm2020}/app/enums/picker-view-type.enum.mjs +8 -8
  149. package/{package/esm2020 → esm2020}/app/fs-filter.module.mjs +260 -260
  150. package/{package/esm2020 → esm2020}/app/helpers/build-query-params.mjs +32 -32
  151. package/{package/esm2020 → esm2020}/app/helpers/compare.mjs +37 -37
  152. package/{package/esm2020 → esm2020}/app/helpers/create-filter-item.mjs +54 -54
  153. package/{package/esm2020 → esm2020}/app/helpers/find-value.mjs +12 -12
  154. package/{package/esm2020 → esm2020}/app/helpers/get-range-name.mjs +8 -8
  155. package/{package/esm2020 → esm2020}/app/helpers/parse-date.mjs +7 -7
  156. package/{package/esm2020 → esm2020}/app/helpers/parse-item-value-from-stored.mjs +80 -80
  157. package/{package/esm2020 → esm2020}/app/helpers/query-param-transformers.mjs +8 -8
  158. package/{package/esm2020 → esm2020}/app/helpers/restore-items.mjs +48 -48
  159. package/{package/esm2020 → esm2020}/app/injectors/filter-config.mjs +2 -2
  160. package/{package/esm2020 → esm2020}/app/injectors/filter-drawer-data.mjs +2 -2
  161. package/{package/esm2020 → esm2020}/app/injectors/filter-drawer-overlay.mjs +2 -2
  162. package/{package/esm2020 → esm2020}/app/interfaces/action.interface.mjs +1 -1
  163. package/{package/esm2020 → esm2020}/app/interfaces/config.interface.mjs +1 -1
  164. package/{package/esm2020 → esm2020}/app/interfaces/external-params.interface.mjs +1 -1
  165. package/{package/esm2020 → esm2020}/app/interfaces/filter.interface.mjs +1 -1
  166. package/{package/esm2020 → esm2020}/app/interfaces/index.mjs +6 -6
  167. package/{package/esm2020 → esm2020}/app/interfaces/items/autocomplete-chips.interface.mjs +1 -1
  168. package/{package/esm2020 → esm2020}/app/interfaces/items/autocomplete.interface.mjs +1 -1
  169. package/{package/esm2020 → esm2020}/app/interfaces/items/base.interface.mjs +1 -1
  170. package/{package/esm2020 → esm2020}/app/interfaces/items/checkbox.interface.mjs +1 -1
  171. package/{package/esm2020 → esm2020}/app/interfaces/items/chips.interface.mjs +1 -1
  172. package/{package/esm2020 → esm2020}/app/interfaces/items/date-range.interface.mjs +1 -1
  173. package/{package/esm2020 → esm2020}/app/interfaces/items/date.interface.mjs +1 -1
  174. package/{package/esm2020 → esm2020}/app/interfaces/items/range.interface.mjs +1 -1
  175. package/{package/esm2020 → esm2020}/app/interfaces/items/select.interface.mjs +1 -1
  176. package/{package/esm2020 → esm2020}/app/interfaces/items/text.interface.mjs +1 -1
  177. package/{package/esm2020 → esm2020}/app/interfaces/items/week.interface.mjs +1 -1
  178. package/{package/esm2020 → esm2020}/app/interfaces/saved-filters.interface.mjs +1 -1
  179. package/{package/esm2020 → esm2020}/app/interfaces/update-filter-item.interface.mjs +1 -1
  180. package/{package/esm2020 → esm2020}/app/models/action-menu-item.model.mjs +66 -66
  181. package/{package/esm2020 → esm2020}/app/models/action.model.mjs +129 -129
  182. package/{package/esm2020 → esm2020}/app/models/filter-config.mjs +66 -66
  183. package/{package/esm2020 → esm2020}/app/models/items/autocomplete/base-autocomplete-item.mjs +13 -13
  184. package/{package/esm2020 → esm2020}/app/models/items/autocomplete-chips-item.mjs +60 -60
  185. package/{package/esm2020 → esm2020}/app/models/items/autocomplete-item.mjs +32 -32
  186. package/{package/esm2020 → esm2020}/app/models/items/base-item.mjs +233 -233
  187. package/{package/esm2020 → esm2020}/app/models/items/checkbox-item.mjs +49 -49
  188. package/{package/esm2020 → esm2020}/app/models/items/chips-item.mjs +88 -88
  189. package/{package/esm2020 → esm2020}/app/models/items/date/base-date-item.mjs +47 -47
  190. package/{package/esm2020 → esm2020}/app/models/items/date-item.mjs +18 -18
  191. package/{package/esm2020 → esm2020}/app/models/items/date-range/base-date-range-item.mjs +134 -134
  192. package/{package/esm2020 → esm2020}/app/models/items/date-range-item.mjs +6 -6
  193. package/{package/esm2020 → esm2020}/app/models/items/date-time-item.mjs +9 -9
  194. package/{package/esm2020 → esm2020}/app/models/items/date-time-range-item.mjs +6 -6
  195. package/{package/esm2020 → esm2020}/app/models/items/range-item.mjs +82 -82
  196. package/{package/esm2020 → esm2020}/app/models/items/select/base-select-item.mjs +36 -36
  197. package/{package/esm2020 → esm2020}/app/models/items/select/multiple-select-item.mjs +87 -87
  198. package/{package/esm2020 → esm2020}/app/models/items/select/simple-select-item.mjs +65 -65
  199. package/{package/esm2020 → esm2020}/app/models/items/select-item.mjs +10 -10
  200. package/{package/esm2020 → esm2020}/app/models/items/text-item.mjs +33 -33
  201. package/{package/esm2020 → esm2020}/app/models/items/week-item.mjs +93 -93
  202. package/{package/esm2020 → esm2020}/app/pipes/remove-isolate-value.pipe.mjs +20 -20
  203. package/{package/esm2020 → esm2020}/app/providers/filter-meta.mjs +9 -9
  204. package/{package/esm2020 → esm2020}/app/services/external-params/persistance-params-controller.service.mjs +57 -57
  205. package/{package/esm2020 → esm2020}/app/services/external-params/query-params-controller.service.mjs +61 -61
  206. package/{package/esm2020 → esm2020}/app/services/external-params/saved-filters-controller.service.mjs +162 -162
  207. package/{package/esm2020 → esm2020}/app/services/external-params-controller.service.mjs +178 -178
  208. package/{package/esm2020 → esm2020}/app/services/filter-overlay.service.mjs +121 -121
  209. package/{package/esm2020 → esm2020}/app/services/focus-controller.service.mjs +29 -29
  210. package/{package/esm2020 → esm2020}/app/services/items-store.service.mjs +344 -344
  211. package/{package/esm2020 → esm2020}/firestitch-filter.mjs +4 -4
  212. package/{package/esm2020 → esm2020}/public_api.mjs +35 -35
  213. package/{package/fesm2015 → fesm2015}/firestitch-filter.mjs +4349 -4349
  214. package/{package/fesm2015 → fesm2015}/firestitch-filter.mjs.map +1 -1
  215. package/{package/fesm2020 → fesm2020}/firestitch-filter.mjs +4336 -4336
  216. package/{package/fesm2020 → fesm2020}/firestitch-filter.mjs.map +1 -1
  217. package/{package/index.d.ts → index.d.ts} +5 -5
  218. package/package.json +22 -79
  219. package/{package/public_api.d.ts → public_api.d.ts} +40 -40
  220. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/16/25/214635fa124a83a4fe3a52bde70e7124d9d011a5d176790a9c607bf1f5ed27cadefc4a1b5a225c0bda11250a180ca5798773e0c108e8d5e2d8559d2f9eed +0 -1
  221. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/44/f9/ee834587d4ac4a429d8c4af2e1562028c7432ac179f9699f2990ab3dadc8e4f66ae45bf22625aedf47fad137e2de39a9b99141784e56d5123544f90c922a +0 -1
  222. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/4d/89/993c2eaff9e0cfca023676bf3f2d3d446f6b6cc90eec839075529d155102b2808a8b44e259d357704acd43143c12af5a5916b4e291a799164b722142ee8e +0 -1
  223. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/52/cb/71f80b48163c432fcb05ac0f736ae7cae10318713c98c74739b3eb5d83ebce635b1a7c77d357b1a4b13a9ae9cfe43348538e4336c25c234551f1e80f0567 +0 -1
  224. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/54/f6/73f0d8ea64a18aa71d80a2cf99d6eb4a2cd345e943127e018913d0f76e9575fabe3aee5002ee4da996e661662d474ad31e9a77f2858c68eed532c2d45d47 +0 -1
  225. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/61/84/7f56389ef39faf5c5b6ad292f668ec6cc5cb6e476d94b1740bdadb9832e3de22a0032b90811f02c80d3d56be5f5e32d8bb900e17ebcb09db379befddc221 +0 -1
  226. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/69/d3/d48ad36dd0a576c4d6f8af3a2ee671037639b0a8d91679992dcbd9dabc284bf316c3126201837e931a162f23cdc4a505608bb64df8e87f59a17364890873 +0 -1
  227. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/70/92/ac80dcd48732b949f163abf36c48b0912598d2bb66e5c1f2056fef872f59fb1b591abbf3bfdbf461a46cbd507eea05a6bd0f14ba3d573f1684cbc78bd094 +0 -1
  228. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/73/10/12664696206dcdab6b084331532b234c439acef8a6428af667a2e1bb1c15e2055b3ac70f44fd8bf508396d3e11a798b40849fd66dbdbd0ff9d9d7e780edd +0 -1
  229. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/74/f6/4c2db73499d08279895e1ab6e2fb65baa613d31218b6968f74143ba03819c72a2720b927ca5a708dd484c0bc6763124e62ce9c2e652f71b740cb2d29ae03 +0 -1
  230. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/78/30/8eedd8d752a08df898806b01abde5b4c1809f6edcba95fcf112c006833b34986c295b62a7e47cd73297c3da78ca6072ef18ba502b6efdd6048dbdbbcd257 +0 -1
  231. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/8c/f8/d9f30e3e898c99e8c3157d4c9157dead1b1115ea4db50aced0c33713cd394bb770b182d1d5087c821d22b2abb99e288191ba9e30954ea4ed9ab6ff70d391 +0 -1
  232. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/8d/53/ad58764bc240411378ae1c47c0c20525da378a3b143795a11bc3fdbf6c9cde6fb2de6c3947d673d5872b9ce205d6274114b3b4dfd27609a636df7fd0b600 +0 -1
  233. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/8f/74/3ed29367f32f9c909f5898704561eb1ebe5cdabba507ba08cbeeaeecd274d020b1401ba871938be899025a265299f8be66c59d59b574a996a74b2987d98e +0 -1
  234. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/a1/ac/1dbcf215585477c3f70180cb40186b5119d9ad1d27b2e0dcde66034c950e334ce0577409535c29bb15d2e6c9238eb9e841177fc62d3949a58bbbb397238d +0 -1
  235. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/a3/08/068a7a29e9cc2385b62d4b0fc14b04a7f09ed469d9b79bb754e37d4933b0ab8aa171b286080e20a11e67f42b8558a4bda64aa03ca097001ecda5c25a9015 +0 -1
  236. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/ae/6c/352ab77e603fa53f25ab95cd6fd19a230095b212464689ab76c89db8d97bd0f221c27fbd84d0767fb5ce7e33f5feb175a9256ab35cf216d30dd3263b6b14 +0 -1
  237. package/.angular/cache/14.2.13/ng-packagr/content-v2/sha512/b6/38/eb8a1b6bd6265968335935cbc2763fbcbb9cd4e32dd58a5234bfd87379ee840044338eeae5f126da9936d21ed423071cfbca6748ee153235addf874824f7 +0 -1
  238. package/.angular/cache/14.2.13/ng-packagr/index-v5/01/3c/d4dca899e5bd80c43ed2e78ec79687adc272ff835d03f0ae48cb881ddd83 +0 -2
  239. package/.angular/cache/14.2.13/ng-packagr/index-v5/05/a7/da2d9625c1bc294409dbf750062e363a1abbe39b51177bfb7e5a9aa9ae81 +0 -2
  240. package/.angular/cache/14.2.13/ng-packagr/index-v5/08/f1/b2e705e45544df7c77f4238b558d569a1215422c2086df9140963f960eef +0 -2
  241. package/.angular/cache/14.2.13/ng-packagr/index-v5/21/1d/398c87beb2402502d4e706beedd98ea707f31b441d6aa7711eef62a0439b +0 -2
  242. package/.angular/cache/14.2.13/ng-packagr/index-v5/26/0b/ee2b1cad7f114e99f8d4da848bd6ceb4e849178b50723d9dcd10f39fb05c +0 -2
  243. package/.angular/cache/14.2.13/ng-packagr/index-v5/26/98/b8614f38ea6fefe80f96f06adb98d925eccfae9eec0937c645a42034614c +0 -2
  244. package/.angular/cache/14.2.13/ng-packagr/index-v5/64/4d/18a7b240783e7b3de33ad33724d2b43e8f3349e840ccab571ca3c6c51c00 +0 -2
  245. package/.angular/cache/14.2.13/ng-packagr/index-v5/64/70/77036e049a76cab5a64caa9cba4b603631c7e9e2f47e6c5fbc7f39d67a2f +0 -2
  246. package/.angular/cache/14.2.13/ng-packagr/index-v5/6c/5d/700bf777ddebfd9b8f15e2709674069747b6fcd37376049712828252ca92 +0 -3
  247. package/.angular/cache/14.2.13/ng-packagr/index-v5/71/89/4e4e3808e28a4a95a98391ab38f87f165ae234aefd4231092d1f531f56b3 +0 -2
  248. package/.angular/cache/14.2.13/ng-packagr/index-v5/7c/c2/bd7b21a07f8b1da7797f40df86dc4289416305c26464c6b965a336fd3975 +0 -2
  249. package/.angular/cache/14.2.13/ng-packagr/index-v5/85/64/cb2f22b4df13a70eb519564e382de3d7dd3747bc0cab1109017f8934cff8 +0 -2
  250. package/.angular/cache/14.2.13/ng-packagr/index-v5/96/b1/2730596477eee53fc70082999f48e373f6795e68d21e04234a45131added +0 -2
  251. package/.angular/cache/14.2.13/ng-packagr/index-v5/af/e4/165326aa8daa8e07ac9e02bde3d64a31b00d1b2b59a797a51a89456d2ad6 +0 -2
  252. package/.angular/cache/14.2.13/ng-packagr/index-v5/cd/96/32e83b96bce723e8ad58fdb7c43cf0002b3e0561309899e8b2a75204cbff +0 -2
  253. package/.angular/cache/14.2.13/ng-packagr/index-v5/dd/38/1f423a0e17cf4e99ef3c880daeb533cdce9275226cb2f617aed52e881e59 +0 -2
  254. package/.angular/cache/14.2.13/ng-packagr/index-v5/e4/e8/f48bdfbec024c932fd928e4db790b6454d090d4db31e9d69201a5e1ea466 +0 -2
  255. package/.angular/cache/14.2.13/ng-packagr/index-v5/ee/59/16a63690de125da5067876d4ccf122230175174f3489e48893448bcbdcd3 +0 -2
  256. package/.angular/cache/14.2.13/ng-packagr/index-v5/fd/46/71e9d6d1a123ed68ce03aa617dcaaad4ad9c5a9193f1281166766c453621 +0 -3
  257. package/.angulardoc.json +0 -4
  258. package/.editorconfig +0 -15
  259. package/.eslintrc.json +0 -9
  260. package/.gitmodules +0 -3
  261. package/README.MD +0 -0
  262. package/angular.json +0 -163
  263. package/babel.config.js +0 -11
  264. package/jest.config.js +0 -24
  265. package/package/package.json +0 -56
  266. package/tsconfig.compodoc.json +0 -12
  267. package/tsconfig.json +0 -10
  268. package/tslint.json +0 -8
  269. /package/{package/styles.scss → styles.scss} +0 -0
@@ -1,614 +1,614 @@
1
- import { ChangeDetectionStrategy, Component, ContentChild, ElementRef, EventEmitter, HostBinding, Inject, Input, NgZone, Optional, Output, ViewChild, } from '@angular/core';
2
- import { MatInput } from '@angular/material/input';
3
- import { BehaviorSubject, combineLatest, fromEvent, interval, Subject } from 'rxjs';
4
- import { debounceTime, distinctUntilChanged, filter, map, takeUntil } from 'rxjs/operators';
5
- import { ActionsController } from '../../classes/actions-controller';
6
- import { objectsAreEquals } from '../../helpers/compare';
7
- import { FsFilterConfig } from '../../models/filter-config';
8
- import { ExternalParamsController } from '../../services/external-params-controller.service';
9
- import { PersistanceParamsController } from '../../services/external-params/persistance-params-controller.service';
10
- import { QueryParamsController } from '../../services/external-params/query-params-controller.service';
11
- import { SavedFiltersController } from '../../services/external-params/saved-filters-controller.service';
12
- import { FsFilterOverlayService } from '../../services/filter-overlay.service';
13
- import { FocusControllerService } from '../../services/focus-controller.service';
14
- import { FsFilterItemsStore } from '../../services/items-store.service';
15
- import { FilterStatusBarDirective } from './../../directives/status-bar/status-bar.directive';
16
- import { FS_FILTER_CONFIG } from './../../injectors/filter-config';
17
- import * as i0 from "@angular/core";
18
- import * as i1 from "../../services/filter-overlay.service";
19
- import * as i2 from "../../services/external-params-controller.service";
20
- import * as i3 from "../../services/items-store.service";
21
- import * as i4 from "../../classes/actions-controller";
22
- import * as i5 from "../../services/external-params/saved-filters-controller.service";
23
- import * as i6 from "@angular/common";
24
- import * as i7 from "@angular/forms";
25
- import * as i8 from "@angular/material/icon";
26
- import * as i9 from "@angular/material/form-field";
27
- import * as i10 from "@angular/material/input";
28
- import * as i11 from "@angular/material/button";
29
- import * as i12 from "@angular/material/slide-toggle";
30
- import * as i13 from "@firestitch/form";
31
- import * as i14 from "@firestitch/clear";
32
- import * as i15 from "../filter-chips/filter-chips.component";
33
- import * as i16 from "../actions/actions.component";
34
- import * as i17 from "../../models/filter-config";
35
- export class FilterComponent {
36
- constructor(_defaultConfig, _filterOverlay, _zone, _externalParams, _filterItems, _actions, _savedFiltersController, _externalParamsController) {
37
- this._defaultConfig = _defaultConfig;
38
- this._filterOverlay = _filterOverlay;
39
- this._zone = _zone;
40
- this._externalParams = _externalParams;
41
- this._filterItems = _filterItems;
42
- this._actions = _actions;
43
- this._savedFiltersController = _savedFiltersController;
44
- this._externalParamsController = _externalParamsController;
45
- this.showSortBy = true;
46
- this.showFilterInput = true;
47
- this.closed = new EventEmitter();
48
- this.opened = new EventEmitter();
49
- this.ready = new EventEmitter();
50
- this.showFilterMenu = false;
51
- this.windowDesktop = false;
52
- this.fsFilterClass = true;
53
- this.searchPlaceholder = 'Search';
54
- this.keyword = '';
55
- this.autoReload = true;
56
- this._config = null;
57
- this._filtersBtnVisible$ = new BehaviorSubject(true);
58
- this._keywordVisible$ = new BehaviorSubject(true);
59
- this._hasFilterChips$ = new BehaviorSubject(false);
60
- this._keyword$ = new Subject();
61
- this._destroy$ = new Subject();
62
- this._filterItems.filter = this;
63
- this._listenWhenFilterReady();
64
- this._updateWindowWidth();
65
- this._filterOverlay.attach$
66
- .pipe(takeUntil(this._destroy$))
67
- .subscribe(() => {
68
- this.showFilterMenu = true;
69
- });
70
- this._filterOverlay.detach$
71
- .pipe(takeUntil(this._destroy$))
72
- .subscribe(() => {
73
- this.showFilterMenu = false;
74
- });
75
- this._listenWindowResize();
76
- }
77
- set setConfig(config) {
78
- this._initFilterWithConfig(config);
79
- }
80
- set setFilter(config) {
81
- this._initFilterWithConfig(config);
82
- }
83
- get hasKeyword() {
84
- return this._filterItems.hasKeyword;
85
- }
86
- get config() {
87
- return this._config;
88
- }
89
- get filterParams() {
90
- return this._filterItems.values();
91
- }
92
- get filterParamsQuery() {
93
- return this._filterItems.valuesAsQuery();
94
- }
95
- get items() {
96
- return this._filterItems.items;
97
- }
98
- get visibleItems() {
99
- return this._filterItems.visibleItems;
100
- }
101
- get keywordItem() {
102
- return this._filterItems.keywordItem;
103
- }
104
- get itemsReady$() {
105
- return this._filterItems.ready$;
106
- }
107
- get hasFilterChips$() {
108
- return this._hasFilterChips$.asObservable();
109
- }
110
- get hasVisibleItemOrSorting() {
111
- return this.visibleItems.length > 0 || !!this._filterItems.sortByItem;
112
- }
113
- get filtersBtnVisible$() {
114
- return this._filtersBtnVisible$.asObservable();
115
- }
116
- get keywordVisible$() {
117
- return this._keywordVisible$.asObservable();
118
- }
119
- get actionsVisible$() {
120
- return this._actions.visible$;
121
- }
122
- get actions$() {
123
- return this._actions.actions$;
124
- }
125
- get menuActions$() {
126
- return this._actions.menuActions$;
127
- }
128
- set activeSavedFilter(savedFilter) {
129
- this._externalParams.setActiveSavedFilter(savedFilter);
130
- }
131
- get activeSavedFilter() {
132
- return this._savedFiltersController.activeFilter;
133
- }
134
- get savedFilters() {
135
- return this._savedFiltersController.savedFilters;
136
- }
137
- ngOnInit() {
138
- // Avoid ngChanges error
139
- setTimeout(() => {
140
- if (this.config.autofocus) {
141
- this.focus();
142
- }
143
- });
144
- this._initAutoReload();
145
- this._listenInputChanges();
146
- this._listenInternalItemsChange();
147
- this._initOverlay();
148
- }
149
- ngOnDestroy() {
150
- this._destroyFilterDrawer();
151
- this._destroy$.next();
152
- this._destroy$.complete();
153
- }
154
- focus() {
155
- this.keywordMatInput?.focus();
156
- }
157
- updateSort(sort) {
158
- this._filterItems.updateSort(sort);
159
- }
160
- /**
161
- *
162
- * Do update value of some field
163
- *
164
- * @param values - values for update
165
- *
166
- * To update text value just pass new text value
167
- *
168
- * public updateSelectValue(val) {
169
- * this.filterEl.updateValues({ keyword: val });
170
- * }
171
- *
172
- * To update select or observable select you could pass suitable value
173
- *
174
- * public updateSelectValue(val: number) {
175
- * this.filterEl.updateValues({ simple_select: val }, { observable_select: val });
176
- * }
177
- *
178
- * To update checkbox value just pass true/false as value
179
- *
180
- * public updateCheckox(val: boolean) {
181
- * this.filterEl.updateValues({ checkbox: val });
182
- * }
183
- *
184
- * To update range value just pass object with min&max object or just with one of targets
185
- *
186
- * Ex.: { min: 10, max 15 }, { min: 5 }, { max 5 }
187
- *
188
- * public updateRange(val) {
189
- * this.filterEl.updateValues({ range: val });
190
- * }
191
- *
192
- * To update autocomplete just pass object with name/value fields
193
- *
194
- * Ex.: { name: 'John Doe', value: 1 }
195
- *
196
- * public updateAutocomplete(val) {
197
- * this.filterEl.updateValues({ autocomplete_user_id: val });
198
- * }
199
- *
200
- * To update autocompletechips just pass:
201
- *
202
- * 1) object with name/value fields - will be appended to existing set of values
203
- *
204
- * { name: 'John Doe', value: 1 }
205
- *
206
- * 2) array of objects - will be appended to existing set of values
207
- *
208
- * [{ name: 'John Doe', value: 1 }, { name: 'Darya Filipova', value: 2 }]
209
- *
210
- * 3) null - clear existing set of values
211
- *
212
- * public updateAutocomplete(val) {
213
- * this.filterEl.updateValues({ autocompletechips_user_id: val });
214
- * }
215
- *
216
- */
217
- updateValues(values) {
218
- Object.keys(values).forEach((key) => {
219
- const filterItem = this.items
220
- .find((item) => item.name === key);
221
- if (!filterItem) {
222
- return;
223
- }
224
- filterItem.model = values[key];
225
- });
226
- }
227
- hide() {
228
- this.changeVisibility(false);
229
- }
230
- show() {
231
- this.changeVisibility(true);
232
- }
233
- changeVisibilityClick(value, event = null) {
234
- if (event) {
235
- event.stopPropagation();
236
- }
237
- this.changeVisibility(value);
238
- }
239
- get itemValues() {
240
- return this.items
241
- .map((item) => item.value);
242
- }
243
- get nonEmptyItemValues() {
244
- return this.items
245
- .filter((item) => item.value !== undefined)
246
- .map((item) => item.value);
247
- }
248
- get hasItemValues() {
249
- return this.items
250
- .some((item) => item.value !== undefined);
251
- }
252
- getItemValue(name) {
253
- const item = this.items
254
- .find((_item) => _item.name === name);
255
- return item?.value;
256
- }
257
- showItem(name) {
258
- const item = this.getItem(name);
259
- if (item) {
260
- item.hide = false;
261
- this._filterItems.updateItemsVisiblity();
262
- }
263
- }
264
- hideItem(name) {
265
- const item = this.getItem(name);
266
- if (!item) {
267
- return;
268
- }
269
- item.hide = true;
270
- this._filterItems.updateItemsVisiblity();
271
- }
272
- clearItem(name) {
273
- const item = this.getItem(name);
274
- if (!item) {
275
- return;
276
- }
277
- item.clear();
278
- }
279
- updateItemConfig(name, params) {
280
- const item = this.getItem(name);
281
- if (!item) {
282
- return;
283
- }
284
- item.label = params.label ?? item.label;
285
- item.chipLabel = params.chipLabel ?? item.chipLabel;
286
- this._filterItems.updateItemsVisiblity();
287
- }
288
- getItemValueChange$(name) {
289
- const item = this.items.find((i) => i.name === name);
290
- if (item) {
291
- return item.value$
292
- .pipe(map(() => {
293
- return item.model;
294
- }));
295
- }
296
- return null;
297
- }
298
- changeVisibility(state) {
299
- if (state === this.showFilterMenu) {
300
- return;
301
- }
302
- if (!state) {
303
- this.closed.emit();
304
- return this._destroyFilterDrawer();
305
- }
306
- if (!this.hasVisibleItemOrSorting) {
307
- return;
308
- }
309
- this._listenEscButton();
310
- this.opened.emit();
311
- this._filterOverlay.open();
312
- }
313
- init() {
314
- const data = this._filterItems.valuesAsQuery();
315
- this._sort = this._filterItems.getSort();
316
- if (this.config.init) {
317
- this.config.init(data, this._sort, this);
318
- }
319
- this._updateChipsVisibility();
320
- this.items
321
- .forEach((item) => {
322
- item.init(item, this);
323
- });
324
- }
325
- clear(event = null) {
326
- if (event) {
327
- event.stopPropagation();
328
- }
329
- this._filterItems.filtersClear();
330
- if (this.config.clear) {
331
- this.config.clear();
332
- }
333
- this.keyword = '';
334
- }
335
- /**
336
- * Close filter window and do change callback
337
- */
338
- search(event) {
339
- this.changeVisibilityClick(false, event);
340
- }
341
- reload(event = null) {
342
- if (event) {
343
- event.preventDefault();
344
- event.stopPropagation();
345
- }
346
- const data = this._filterItems.valuesAsQuery();
347
- const el = this.reloadEl?.nativeElement;
348
- if (el) {
349
- el.style.transition = 'all 0.75s 0.0s';
350
- el.style.transform = 'rotate(360deg)';
351
- setTimeout(() => {
352
- el.style.transition = null;
353
- el.style.transform = null;
354
- }, 1000);
355
- }
356
- if (this.config.reload) {
357
- this.config.reload(data, this._filterItems.getSort());
358
- }
359
- }
360
- getItem(name) {
361
- return this.items
362
- .find((item) => item.name === name);
363
- }
364
- fetchQueryParams() {
365
- this._externalParams.fetchQueryParams();
366
- }
367
- /**
368
- * Call change callback and apply new filter values
369
- */
370
- change() {
371
- const data = this._filterItems.valuesAsQuery();
372
- const sort = this._filterItems.getSort();
373
- const sortingChanged = ((!sort || !this._sort) && sort !== this._sort)
374
- || (sort && this._sort && !objectsAreEquals(this._sort, sort));
375
- if (sortingChanged) {
376
- this._sort = sort;
377
- if (this.config.sortChange) {
378
- this.config.sortChange(data, sort);
379
- }
380
- }
381
- if (this.config.change) {
382
- this.config.change(data, sort);
383
- }
384
- this._updateChipsVisibility();
385
- // visibility for actions can depend on filters state
386
- this._actions.updateActionsVisibility();
387
- }
388
- /**
389
- * Update filter actions config
390
- *
391
- * @param actions
392
- */
393
- updateActions(actions) {
394
- this._actions.initActions(actions);
395
- }
396
- /**
397
- * Show "Filters" button
398
- */
399
- showFiltersBtn() {
400
- this._filtersBtnVisible$.next(true);
401
- }
402
- /**
403
- * Hide "Filters" button
404
- */
405
- hideFiltersBtn() {
406
- this._filtersBtnVisible$.next(false);
407
- }
408
- /**
409
- * Show "Keyword" field if it present
410
- */
411
- showKeywordField() {
412
- this._keywordVisible$.next(true);
413
- }
414
- /**
415
- * Hide "Keyword" field if it present
416
- */
417
- hideKeywordField() {
418
- this._keywordVisible$.next(false);
419
- }
420
- /**
421
- * Go through actions and check show() callback and update visible actions
422
- */
423
- updateActionsVisibility() {
424
- this._actions.updateActionsVisibility();
425
- }
426
- /**
427
- * Go through actions and check disabled() callback and update disabled state
428
- */
429
- updateDisabledState() {
430
- this._actions.updateDisabledState();
431
- }
432
- setItems(items) {
433
- this._filterItems.destroyItems();
434
- this.config.items = items;
435
- this._filterItems.setConfig(this._config);
436
- this._externalParams.initItems();
437
- this._syncSearchInputWithKeyword();
438
- }
439
- keywordChange(keyword) {
440
- this._keyword$.next(keyword);
441
- }
442
- updateSortings(items) {
443
- this._filterItems.updateSortingItemsValues(items);
444
- }
445
- _initFilterWithConfig(config) {
446
- if (this.config) {
447
- this._filterItems.destroyItems();
448
- }
449
- config = {
450
- ...(this._defaultConfig || {}),
451
- ...config,
452
- };
453
- this._config = new FsFilterConfig(config);
454
- this._actions.setConfig(this._config);
455
- this._filterItems.setConfig(this._config);
456
- this._externalParams.setConfig(this._config);
457
- this._syncSearchInputWithKeyword();
458
- if (this.config.reloadWhenConfigChanged) {
459
- this.change();
460
- }
461
- }
462
- _destroyFilterDrawer() {
463
- this._filterOverlay.close();
464
- }
465
- _updateWindowWidth() {
466
- this.windowDesktop = window.innerWidth > 1200;
467
- }
468
- _listenEscButton() {
469
- this._zone.runOutsideAngular(() => {
470
- fromEvent(window, 'keyup')
471
- .pipe(filter((event) => event.code === 'Escape'), takeUntil(this.closed), takeUntil(this._destroy$))
472
- .subscribe(() => {
473
- this._zone.run(() => {
474
- this.changeVisibility(false);
475
- });
476
- });
477
- });
478
- }
479
- _listenWindowResize() {
480
- this._zone.runOutsideAngular(() => {
481
- fromEvent(window, 'resize')
482
- .pipe(debounceTime(100), takeUntil(this._destroy$))
483
- .subscribe(() => {
484
- this._zone.run(() => {
485
- this._updateWindowWidth();
486
- });
487
- });
488
- });
489
- }
490
- _initAutoReload() {
491
- if (this.config.autoReload) {
492
- interval(this.config.autoReload.seconds * 1000)
493
- .pipe(filter(() => this.autoReload), takeUntil(this._destroy$))
494
- .subscribe(() => {
495
- this.reload(null);
496
- });
497
- }
498
- }
499
- _listenInputChanges() {
500
- this._keyword$
501
- .pipe(debounceTime(200), distinctUntilChanged(), takeUntil(this._destroy$))
502
- .subscribe((value) => {
503
- const keywordItem = this._filterItems.keywordItem;
504
- keywordItem.model = value;
505
- this.change();
506
- });
507
- }
508
- _syncSearchInputWithKeyword() {
509
- const keywordItem = this._filterItems.keywordItem;
510
- if (keywordItem) {
511
- this.keyword = keywordItem.model;
512
- this.searchPlaceholder = keywordItem.label || 'Search';
513
- }
514
- }
515
- _listenInternalItemsChange() {
516
- this._filterItems
517
- .itemsChange$
518
- .pipe(takeUntil(this._destroy$))
519
- .subscribe(() => {
520
- this.change();
521
- });
522
- }
523
- _initOverlay() {
524
- this._filterOverlay.setClearFn(this.clear.bind(this));
525
- this._filterOverlay.setDoneFn(this.hide.bind(this));
526
- }
527
- // We may need some time to recieve external params and after that ready can be emitted
528
- _listenWhenFilterReady() {
529
- combineLatest([
530
- this._externalParams.pending$,
531
- this.itemsReady$,
532
- ])
533
- .pipe(filter(([pendingParams, itemsReady]) => !pendingParams && itemsReady), takeUntil(this._destroy$))
534
- .subscribe(() => {
535
- this.init();
536
- this._syncSearchInputWithKeyword();
537
- this.ready.emit();
538
- });
539
- }
540
- _updateChipsVisibility() {
541
- const hasFilterChips = this._filterItems.items
542
- .some((item) => {
543
- return item.isChipVisible;
544
- });
545
- this._hasFilterChips$.next(hasFilterChips);
546
- }
547
- }
548
- FilterComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterComponent, deps: [{ token: FS_FILTER_CONFIG, optional: true }, { token: i1.FsFilterOverlayService }, { token: i0.NgZone }, { token: i2.ExternalParamsController }, { token: i3.FsFilterItemsStore }, { token: i4.ActionsController }, { token: i5.SavedFiltersController }, { token: i2.ExternalParamsController }], target: i0.ɵɵFactoryTarget.Component });
549
- FilterComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: FilterComponent, selector: "fs-filter", inputs: { setConfig: ["config", "setConfig"], setFilter: ["filter", "setFilter"], showSortBy: "showSortBy", showFilterInput: "showFilterInput" }, outputs: { closed: "closed", opened: "opened", ready: "ready" }, host: { properties: { "class.filters-open": "this.showFilterMenu", "class.window-desktop": "this.windowDesktop", "class.fs-filter": "this.fsFilterClass", "class.has-keyword": "this.hasKeyword" } }, providers: [
550
- FsFilterOverlayService,
551
- ExternalParamsController,
552
- PersistanceParamsController,
553
- QueryParamsController,
554
- FocusControllerService,
555
- FsFilterItemsStore,
556
- SavedFiltersController,
557
- ActionsController,
558
- ], queries: [{ propertyName: "statusBar", first: true, predicate: FilterStatusBarDirective, descendants: true }], viewQueries: [{ propertyName: "keywordMatInput", first: true, predicate: ["keywordMatInput"], descendants: true, read: MatInput }, { propertyName: "reloadEl", first: true, predicate: ["reloadEl"], descendants: true, read: ElementRef }], ngImport: i0, template: "<div class=\"filter-container\">\n <div class=\"filter-inner-container\">\n <ng-container *ngIf=\"!hasKeyword; else filterKeyword\">\n <div>\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </div>\n </ng-container>\n\n <ng-container [ngTemplateOutlet]=\"filterToolbar\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"filterActions\"></ng-container>\n </div>\n\n <ng-container *ngIf=\"hasKeyword\">\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </ng-container>\n</div>\n\n<ng-template #filterStatusBarChips>\n <ng-container *ngIf=\"statusBar\">\n <div class=\"filter-status-container\" [ngClass]=\"{ 'has-status': !!filterStatus.textContent }\">\n <div class=\"filter-status\" #filterStatus>\n <ng-container *ngTemplateOutlet=\"statusBar.templateRef\"></ng-container>\n </div>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"config.chips && hasFilterChips$ | async\">\n <fs-filter-chips\n class=\"filter-chips\"\n [filters]=\"items\">\n </fs-filter-chips>\n </ng-container>\n</ng-template>\n\n<ng-template #filterKeyword>\n <div class=\"filter-keyword\">\n <ng-container *ngIf=\"(keywordVisible$ | async) && !keywordItem?.hide \">\n <mat-form-field\n [floatLabel]=\"'never'\"\n class=\"form-field-padless\"\n [ngClass]=\"search\"\n appearance=\"outline\">\n <span matPrefix>\n <mat-icon matPrefix>search</mat-icon>\n </span>\n <input\n #keywordMatInput\n matInput\n [(ngModel)]=\"keyword\"\n (ngModelChange)=\"keywordChange($event)\"\n name=\"filter-input\"\n [fsClear]=\"true\"\n [placeholder]=\"searchPlaceholder\">\n </mat-form-field>\n </ng-container>\n </div>\n</ng-template>\n\n<ng-template #filterActions>\n <div class=\"filter-actions\">\n <fs-filter-actions\n *ngIf=\"actionsVisible$ | async\"\n [actions]=\"actions$ | async\"\n [kebabActions]=\"menuActions$ | async\">\n </fs-filter-actions>\n </div>\n</ng-template>\n\n<ng-template #filterToolbar>\n <div class=\"filter-toobar\">\n <ng-container *ngIf=\"filtersBtnVisible$ | async\">\n <ng-container *ngIf=\"hasVisibleItemOrSorting\">\n <a\n mat-button\n class=\"filters-button\"\n [ngClass]=\"{\n 'mat-raised-button': config.button.style === 'raised' && config.button.label,\n 'mat-flat-button': config.button.style === 'flat' && config.button.label,\n 'mat-stroked-button': config.button.style === 'stroked' && config.button.label,\n 'mat-button': config.button.style === 'basic' && config.button.label,\n 'mat-icon-button': config.button.style === 'icon' || !config.button.label\n }\"\n (click)=\"changeVisibilityClick(!showFilterMenu, $event)\"\n [color]=\"config.button.color\">\n <mat-icon *ngIf=\"config.button.icon\">{{config.button.icon}}</mat-icon>\n {{ config.button.label }}\n </a>\n </ng-container>\n </ng-container> \n <ng-container *ngIf=\"config.reload\">\n <a\n mat-icon-button\n (click)=\"reload($event)\"\n class=\"reload\">\n <mat-icon #reloadEl>refresh</mat-icon>\n </a>\n </ng-container>\n <ng-container *ngIf=\"config.autoReload\">\n <mat-slide-toggle \n name=\"autoReload\" \n class=\"auto-reload\"\n [(ngModel)]=\"autoReload\">\n Auto refresh\n </mat-slide-toggle>\n </ng-container>\n </div>\n</ng-template>\n", styles: [":host{margin-bottom:20px;display:block}:host.has-keyword .filter-status-container.has-status{margin-top:4px}:host:not(.has-keyword) .filter-status-container:not(.has-status)+fs-filter-chips{margin-top:0}:host:not(.has-keyword) .filter-inner-container{display:flex}:host:not(.has-keyword) .filter-toobar{justify-content:flex-end}.filter-status-container{flex-grow:1;display:flex;justify-content:center;flex-direction:column;align-self:flex-end}.filter-status-container .filter-status{overflow:hidden;text-overflow:ellipsis;line-height:17px}.filter-container{width:100%}.filter-inner-container{flex-direction:row;box-sizing:border-box;display:flex;position:relative;align-items:center}.filter-inner-container .filter-keyword{flex-direction:row;box-sizing:border-box;display:flex;align-items:center;min-width:0}.filter-inner-container .filter-keyword mat-form-field{max-width:100%;min-width:100px;margin-right:6px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-wrapper{margin-bottom:0}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix .mat-icon{font-size:22px;color:#626262}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-infix{width:250px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix,.filter-inner-container .filter-keyword mat-form-field .mat-form-field-suffix{top:.45em}.filter-inner-container .filter-keyword .mat-form-field-suffix{display:flex}.filter-inner-container .filter-keyword .mat-form-field-suffix .mat-icon-button{height:24px;width:24px}.filter-actions{display:flex;align-items:center}.filter-toobar{flex:1;white-space:nowrap;display:flex;align-items:center}.filter-toobar .reload{margin-left:-6px}.results{min-height:90px;position:relative;overflow-x:auto;overflow-y:hidden}fs-filter-chips{margin-top:4px;display:block}@media screen and (min-width: 599px){.filter-actions{margin-left:10px}.filters-button+fs-filter-actions:not(:empty){margin-left:5px}}@media screen and (max-width: 600px){.filter-actions{margin-left:5px}.filter-actions .filters-button{font-size:0;padding:0;min-width:unset;width:36px;height:36px;border-radius:50%;box-shadow:unset!important}}@media screen and (min-width: 1200px){html.fs-filter-open body{margin-right:350px}.fs-filter-backdrop{display:none}}html.fs-filter-open{scrollbar-width:none}:host ::ng-deep .auto-reload.mat-checked .mat-slide-toggle-thumb-container{transform:translate3d(12px,0,0)}:host ::ng-deep .auto-reload:not(.mat-checked) .mat-slide-toggle-content{color:#ccc}:host ::ng-deep .auto-reload .mat-slide-toggle-thumb,:host ::ng-deep .auto-reload .mat-slide-toggle-thumb-container{height:15px;width:15px}:host ::ng-deep .auto-reload .mat-slide-toggle-content{font-size:90%}:host ::ng-deep .auto-reload .mat-slide-toggle-bar{width:26px;height:10px;border-radius:10px}\n"], dependencies: [{ kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i8.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i9.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i9.MatPrefix, selector: "[matPrefix]" }, { kind: "directive", type: i10.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i11.MatAnchor, selector: "a[mat-button], a[mat-raised-button], a[mat-icon-button], a[mat-fab], a[mat-mini-fab], a[mat-stroked-button], a[mat-flat-button]", inputs: ["disabled", "disableRipple", "color", "tabIndex"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i12.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["disabled", "disableRipple", "color", "tabIndex"], exportAs: ["matSlideToggle"] }, { kind: "directive", type: i13.FsFormNoFsValidatorsDirective, selector: "[ngModel]:not([required]):not([fsFormRequired]):not([fsFormCompare]):not([fsFormDateRange]):not([fsFormEmail]):not([fsFormEmails]):not([fsFormFunction]):not([fsFormGreater]):not([fsFormInteger]):not([fsFormLesser]):not([fsFormMax]):not([fsFormMaxLength]):not([fsFormMin]):not([fsFormMinLength]):not([fsFormNumeric]):not([fsFormPattern]):not([fsFormPhone]):not([fsFormUrl]):not([validate])" }, { kind: "directive", type: i13.FsButtonDirective, selector: "[mat-raised-button],[mat-button],[mat-flat-button],[mat-stroked-button]", inputs: ["name", "dirtySubmit", "form"] }, { kind: "component", type: i14.FsClearComponent, selector: "[fsClear]", inputs: ["ngModel", "fsClear"], outputs: ["ngModelChange", "cleared"] }, { kind: "component", type: i15.FsFilterChipsComponent, selector: "fs-filter-chips", inputs: ["filters"] }, { kind: "component", type: i16.FsFilterActionsComponent, selector: "fs-filter-actions", inputs: ["kebabActions", "actions"] }, { kind: "pipe", type: i6.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
559
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterComponent, decorators: [{
560
- type: Component,
561
- args: [{ selector: 'fs-filter', providers: [
562
- FsFilterOverlayService,
563
- ExternalParamsController,
564
- PersistanceParamsController,
565
- QueryParamsController,
566
- FocusControllerService,
567
- FsFilterItemsStore,
568
- SavedFiltersController,
569
- ActionsController,
570
- ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"filter-container\">\n <div class=\"filter-inner-container\">\n <ng-container *ngIf=\"!hasKeyword; else filterKeyword\">\n <div>\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </div>\n </ng-container>\n\n <ng-container [ngTemplateOutlet]=\"filterToolbar\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"filterActions\"></ng-container>\n </div>\n\n <ng-container *ngIf=\"hasKeyword\">\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </ng-container>\n</div>\n\n<ng-template #filterStatusBarChips>\n <ng-container *ngIf=\"statusBar\">\n <div class=\"filter-status-container\" [ngClass]=\"{ 'has-status': !!filterStatus.textContent }\">\n <div class=\"filter-status\" #filterStatus>\n <ng-container *ngTemplateOutlet=\"statusBar.templateRef\"></ng-container>\n </div>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"config.chips && hasFilterChips$ | async\">\n <fs-filter-chips\n class=\"filter-chips\"\n [filters]=\"items\">\n </fs-filter-chips>\n </ng-container>\n</ng-template>\n\n<ng-template #filterKeyword>\n <div class=\"filter-keyword\">\n <ng-container *ngIf=\"(keywordVisible$ | async) && !keywordItem?.hide \">\n <mat-form-field\n [floatLabel]=\"'never'\"\n class=\"form-field-padless\"\n [ngClass]=\"search\"\n appearance=\"outline\">\n <span matPrefix>\n <mat-icon matPrefix>search</mat-icon>\n </span>\n <input\n #keywordMatInput\n matInput\n [(ngModel)]=\"keyword\"\n (ngModelChange)=\"keywordChange($event)\"\n name=\"filter-input\"\n [fsClear]=\"true\"\n [placeholder]=\"searchPlaceholder\">\n </mat-form-field>\n </ng-container>\n </div>\n</ng-template>\n\n<ng-template #filterActions>\n <div class=\"filter-actions\">\n <fs-filter-actions\n *ngIf=\"actionsVisible$ | async\"\n [actions]=\"actions$ | async\"\n [kebabActions]=\"menuActions$ | async\">\n </fs-filter-actions>\n </div>\n</ng-template>\n\n<ng-template #filterToolbar>\n <div class=\"filter-toobar\">\n <ng-container *ngIf=\"filtersBtnVisible$ | async\">\n <ng-container *ngIf=\"hasVisibleItemOrSorting\">\n <a\n mat-button\n class=\"filters-button\"\n [ngClass]=\"{\n 'mat-raised-button': config.button.style === 'raised' && config.button.label,\n 'mat-flat-button': config.button.style === 'flat' && config.button.label,\n 'mat-stroked-button': config.button.style === 'stroked' && config.button.label,\n 'mat-button': config.button.style === 'basic' && config.button.label,\n 'mat-icon-button': config.button.style === 'icon' || !config.button.label\n }\"\n (click)=\"changeVisibilityClick(!showFilterMenu, $event)\"\n [color]=\"config.button.color\">\n <mat-icon *ngIf=\"config.button.icon\">{{config.button.icon}}</mat-icon>\n {{ config.button.label }}\n </a>\n </ng-container>\n </ng-container> \n <ng-container *ngIf=\"config.reload\">\n <a\n mat-icon-button\n (click)=\"reload($event)\"\n class=\"reload\">\n <mat-icon #reloadEl>refresh</mat-icon>\n </a>\n </ng-container>\n <ng-container *ngIf=\"config.autoReload\">\n <mat-slide-toggle \n name=\"autoReload\" \n class=\"auto-reload\"\n [(ngModel)]=\"autoReload\">\n Auto refresh\n </mat-slide-toggle>\n </ng-container>\n </div>\n</ng-template>\n", styles: [":host{margin-bottom:20px;display:block}:host.has-keyword .filter-status-container.has-status{margin-top:4px}:host:not(.has-keyword) .filter-status-container:not(.has-status)+fs-filter-chips{margin-top:0}:host:not(.has-keyword) .filter-inner-container{display:flex}:host:not(.has-keyword) .filter-toobar{justify-content:flex-end}.filter-status-container{flex-grow:1;display:flex;justify-content:center;flex-direction:column;align-self:flex-end}.filter-status-container .filter-status{overflow:hidden;text-overflow:ellipsis;line-height:17px}.filter-container{width:100%}.filter-inner-container{flex-direction:row;box-sizing:border-box;display:flex;position:relative;align-items:center}.filter-inner-container .filter-keyword{flex-direction:row;box-sizing:border-box;display:flex;align-items:center;min-width:0}.filter-inner-container .filter-keyword mat-form-field{max-width:100%;min-width:100px;margin-right:6px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-wrapper{margin-bottom:0}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix .mat-icon{font-size:22px;color:#626262}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-infix{width:250px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix,.filter-inner-container .filter-keyword mat-form-field .mat-form-field-suffix{top:.45em}.filter-inner-container .filter-keyword .mat-form-field-suffix{display:flex}.filter-inner-container .filter-keyword .mat-form-field-suffix .mat-icon-button{height:24px;width:24px}.filter-actions{display:flex;align-items:center}.filter-toobar{flex:1;white-space:nowrap;display:flex;align-items:center}.filter-toobar .reload{margin-left:-6px}.results{min-height:90px;position:relative;overflow-x:auto;overflow-y:hidden}fs-filter-chips{margin-top:4px;display:block}@media screen and (min-width: 599px){.filter-actions{margin-left:10px}.filters-button+fs-filter-actions:not(:empty){margin-left:5px}}@media screen and (max-width: 600px){.filter-actions{margin-left:5px}.filter-actions .filters-button{font-size:0;padding:0;min-width:unset;width:36px;height:36px;border-radius:50%;box-shadow:unset!important}}@media screen and (min-width: 1200px){html.fs-filter-open body{margin-right:350px}.fs-filter-backdrop{display:none}}html.fs-filter-open{scrollbar-width:none}:host ::ng-deep .auto-reload.mat-checked .mat-slide-toggle-thumb-container{transform:translate3d(12px,0,0)}:host ::ng-deep .auto-reload:not(.mat-checked) .mat-slide-toggle-content{color:#ccc}:host ::ng-deep .auto-reload .mat-slide-toggle-thumb,:host ::ng-deep .auto-reload .mat-slide-toggle-thumb-container{height:15px;width:15px}:host ::ng-deep .auto-reload .mat-slide-toggle-content{font-size:90%}:host ::ng-deep .auto-reload .mat-slide-toggle-bar{width:26px;height:10px;border-radius:10px}\n"] }]
571
- }], ctorParameters: function () { return [{ type: i17.FsFilterConfig, decorators: [{
572
- type: Optional
573
- }, {
574
- type: Inject,
575
- args: [FS_FILTER_CONFIG]
576
- }] }, { type: i1.FsFilterOverlayService }, { type: i0.NgZone }, { type: i2.ExternalParamsController }, { type: i3.FsFilterItemsStore }, { type: i4.ActionsController }, { type: i5.SavedFiltersController }, { type: i2.ExternalParamsController }]; }, propDecorators: { setConfig: [{
577
- type: Input,
578
- args: ['config']
579
- }], setFilter: [{
580
- type: Input,
581
- args: ['filter']
582
- }], showSortBy: [{
583
- type: Input
584
- }], showFilterInput: [{
585
- type: Input
586
- }], closed: [{
587
- type: Output
588
- }], opened: [{
589
- type: Output
590
- }], ready: [{
591
- type: Output
592
- }], statusBar: [{
593
- type: ContentChild,
594
- args: [FilterStatusBarDirective]
595
- }], keywordMatInput: [{
596
- type: ViewChild,
597
- args: ['keywordMatInput', { read: MatInput }]
598
- }], reloadEl: [{
599
- type: ViewChild,
600
- args: ['reloadEl', { read: ElementRef }]
601
- }], showFilterMenu: [{
602
- type: HostBinding,
603
- args: ['class.filters-open']
604
- }], windowDesktop: [{
605
- type: HostBinding,
606
- args: ['class.window-desktop']
607
- }], fsFilterClass: [{
608
- type: HostBinding,
609
- args: ['class.fs-filter']
610
- }], hasKeyword: [{
611
- type: HostBinding,
612
- args: ['class.has-keyword']
613
- }] } });
1
+ import { ChangeDetectionStrategy, Component, ContentChild, ElementRef, EventEmitter, HostBinding, Inject, Input, NgZone, Optional, Output, ViewChild, } from '@angular/core';
2
+ import { MatInput } from '@angular/material/input';
3
+ import { BehaviorSubject, combineLatest, fromEvent, interval, Subject } from 'rxjs';
4
+ import { debounceTime, distinctUntilChanged, filter, map, takeUntil } from 'rxjs/operators';
5
+ import { ActionsController } from '../../classes/actions-controller';
6
+ import { objectsAreEquals } from '../../helpers/compare';
7
+ import { FsFilterConfig } from '../../models/filter-config';
8
+ import { ExternalParamsController } from '../../services/external-params-controller.service';
9
+ import { PersistanceParamsController } from '../../services/external-params/persistance-params-controller.service';
10
+ import { QueryParamsController } from '../../services/external-params/query-params-controller.service';
11
+ import { SavedFiltersController } from '../../services/external-params/saved-filters-controller.service';
12
+ import { FsFilterOverlayService } from '../../services/filter-overlay.service';
13
+ import { FocusControllerService } from '../../services/focus-controller.service';
14
+ import { FsFilterItemsStore } from '../../services/items-store.service';
15
+ import { FilterStatusBarDirective } from './../../directives/status-bar/status-bar.directive';
16
+ import { FS_FILTER_CONFIG } from './../../injectors/filter-config';
17
+ import * as i0 from "@angular/core";
18
+ import * as i1 from "../../services/filter-overlay.service";
19
+ import * as i2 from "../../services/external-params-controller.service";
20
+ import * as i3 from "../../services/items-store.service";
21
+ import * as i4 from "../../classes/actions-controller";
22
+ import * as i5 from "../../services/external-params/saved-filters-controller.service";
23
+ import * as i6 from "@angular/common";
24
+ import * as i7 from "@angular/forms";
25
+ import * as i8 from "@angular/material/icon";
26
+ import * as i9 from "@angular/material/form-field";
27
+ import * as i10 from "@angular/material/input";
28
+ import * as i11 from "@angular/material/button";
29
+ import * as i12 from "@angular/material/slide-toggle";
30
+ import * as i13 from "@firestitch/form";
31
+ import * as i14 from "@firestitch/clear";
32
+ import * as i15 from "../filter-chips/filter-chips.component";
33
+ import * as i16 from "../actions/actions.component";
34
+ import * as i17 from "../../models/filter-config";
35
+ export class FilterComponent {
36
+ constructor(_defaultConfig, _filterOverlay, _zone, _externalParams, _filterItems, _actions, _savedFiltersController, _externalParamsController) {
37
+ this._defaultConfig = _defaultConfig;
38
+ this._filterOverlay = _filterOverlay;
39
+ this._zone = _zone;
40
+ this._externalParams = _externalParams;
41
+ this._filterItems = _filterItems;
42
+ this._actions = _actions;
43
+ this._savedFiltersController = _savedFiltersController;
44
+ this._externalParamsController = _externalParamsController;
45
+ this.showSortBy = true;
46
+ this.showFilterInput = true;
47
+ this.closed = new EventEmitter();
48
+ this.opened = new EventEmitter();
49
+ this.ready = new EventEmitter();
50
+ this.showFilterMenu = false;
51
+ this.windowDesktop = false;
52
+ this.fsFilterClass = true;
53
+ this.searchPlaceholder = 'Search';
54
+ this.keyword = '';
55
+ this.autoReload = true;
56
+ this._config = null;
57
+ this._filtersBtnVisible$ = new BehaviorSubject(true);
58
+ this._keywordVisible$ = new BehaviorSubject(true);
59
+ this._hasFilterChips$ = new BehaviorSubject(false);
60
+ this._keyword$ = new Subject();
61
+ this._destroy$ = new Subject();
62
+ this._filterItems.filter = this;
63
+ this._listenWhenFilterReady();
64
+ this._updateWindowWidth();
65
+ this._filterOverlay.attach$
66
+ .pipe(takeUntil(this._destroy$))
67
+ .subscribe(() => {
68
+ this.showFilterMenu = true;
69
+ });
70
+ this._filterOverlay.detach$
71
+ .pipe(takeUntil(this._destroy$))
72
+ .subscribe(() => {
73
+ this.showFilterMenu = false;
74
+ });
75
+ this._listenWindowResize();
76
+ }
77
+ set setConfig(config) {
78
+ this._initFilterWithConfig(config);
79
+ }
80
+ set setFilter(config) {
81
+ this._initFilterWithConfig(config);
82
+ }
83
+ get hasKeyword() {
84
+ return this._filterItems.hasKeyword;
85
+ }
86
+ get config() {
87
+ return this._config;
88
+ }
89
+ get filterParams() {
90
+ return this._filterItems.values();
91
+ }
92
+ get filterParamsQuery() {
93
+ return this._filterItems.valuesAsQuery();
94
+ }
95
+ get items() {
96
+ return this._filterItems.items;
97
+ }
98
+ get visibleItems() {
99
+ return this._filterItems.visibleItems;
100
+ }
101
+ get keywordItem() {
102
+ return this._filterItems.keywordItem;
103
+ }
104
+ get itemsReady$() {
105
+ return this._filterItems.ready$;
106
+ }
107
+ get hasFilterChips$() {
108
+ return this._hasFilterChips$.asObservable();
109
+ }
110
+ get hasVisibleItemOrSorting() {
111
+ return this.visibleItems.length > 0 || !!this._filterItems.sortByItem;
112
+ }
113
+ get filtersBtnVisible$() {
114
+ return this._filtersBtnVisible$.asObservable();
115
+ }
116
+ get keywordVisible$() {
117
+ return this._keywordVisible$.asObservable();
118
+ }
119
+ get actionsVisible$() {
120
+ return this._actions.visible$;
121
+ }
122
+ get actions$() {
123
+ return this._actions.actions$;
124
+ }
125
+ get menuActions$() {
126
+ return this._actions.menuActions$;
127
+ }
128
+ set activeSavedFilter(savedFilter) {
129
+ this._externalParams.setActiveSavedFilter(savedFilter);
130
+ }
131
+ get activeSavedFilter() {
132
+ return this._savedFiltersController.activeFilter;
133
+ }
134
+ get savedFilters() {
135
+ return this._savedFiltersController.savedFilters;
136
+ }
137
+ ngOnInit() {
138
+ // Avoid ngChanges error
139
+ setTimeout(() => {
140
+ if (this.config.autofocus) {
141
+ this.focus();
142
+ }
143
+ });
144
+ this._initAutoReload();
145
+ this._listenInputChanges();
146
+ this._listenInternalItemsChange();
147
+ this._initOverlay();
148
+ }
149
+ ngOnDestroy() {
150
+ this._destroyFilterDrawer();
151
+ this._destroy$.next();
152
+ this._destroy$.complete();
153
+ }
154
+ focus() {
155
+ this.keywordMatInput?.focus();
156
+ }
157
+ updateSort(sort) {
158
+ this._filterItems.updateSort(sort);
159
+ }
160
+ /**
161
+ *
162
+ * Do update value of some field
163
+ *
164
+ * @param values - values for update
165
+ *
166
+ * To update text value just pass new text value
167
+ *
168
+ * public updateSelectValue(val) {
169
+ * this.filterEl.updateValues({ keyword: val });
170
+ * }
171
+ *
172
+ * To update select or observable select you could pass suitable value
173
+ *
174
+ * public updateSelectValue(val: number) {
175
+ * this.filterEl.updateValues({ simple_select: val }, { observable_select: val });
176
+ * }
177
+ *
178
+ * To update checkbox value just pass true/false as value
179
+ *
180
+ * public updateCheckox(val: boolean) {
181
+ * this.filterEl.updateValues({ checkbox: val });
182
+ * }
183
+ *
184
+ * To update range value just pass object with min&max object or just with one of targets
185
+ *
186
+ * Ex.: { min: 10, max 15 }, { min: 5 }, { max 5 }
187
+ *
188
+ * public updateRange(val) {
189
+ * this.filterEl.updateValues({ range: val });
190
+ * }
191
+ *
192
+ * To update autocomplete just pass object with name/value fields
193
+ *
194
+ * Ex.: { name: 'John Doe', value: 1 }
195
+ *
196
+ * public updateAutocomplete(val) {
197
+ * this.filterEl.updateValues({ autocomplete_user_id: val });
198
+ * }
199
+ *
200
+ * To update autocompletechips just pass:
201
+ *
202
+ * 1) object with name/value fields - will be appended to existing set of values
203
+ *
204
+ * { name: 'John Doe', value: 1 }
205
+ *
206
+ * 2) array of objects - will be appended to existing set of values
207
+ *
208
+ * [{ name: 'John Doe', value: 1 }, { name: 'Darya Filipova', value: 2 }]
209
+ *
210
+ * 3) null - clear existing set of values
211
+ *
212
+ * public updateAutocomplete(val) {
213
+ * this.filterEl.updateValues({ autocompletechips_user_id: val });
214
+ * }
215
+ *
216
+ */
217
+ updateValues(values) {
218
+ Object.keys(values).forEach((key) => {
219
+ const filterItem = this.items
220
+ .find((item) => item.name === key);
221
+ if (!filterItem) {
222
+ return;
223
+ }
224
+ filterItem.model = values[key];
225
+ });
226
+ }
227
+ hide() {
228
+ this.changeVisibility(false);
229
+ }
230
+ show() {
231
+ this.changeVisibility(true);
232
+ }
233
+ changeVisibilityClick(value, event = null) {
234
+ if (event) {
235
+ event.stopPropagation();
236
+ }
237
+ this.changeVisibility(value);
238
+ }
239
+ get itemValues() {
240
+ return this.items
241
+ .map((item) => item.value);
242
+ }
243
+ get nonEmptyItemValues() {
244
+ return this.items
245
+ .filter((item) => item.value !== undefined)
246
+ .map((item) => item.value);
247
+ }
248
+ get hasItemValues() {
249
+ return this.items
250
+ .some((item) => item.value !== undefined);
251
+ }
252
+ getItemValue(name) {
253
+ const item = this.items
254
+ .find((_item) => _item.name === name);
255
+ return item?.value;
256
+ }
257
+ showItem(name) {
258
+ const item = this.getItem(name);
259
+ if (item) {
260
+ item.hide = false;
261
+ this._filterItems.updateItemsVisiblity();
262
+ }
263
+ }
264
+ hideItem(name) {
265
+ const item = this.getItem(name);
266
+ if (!item) {
267
+ return;
268
+ }
269
+ item.hide = true;
270
+ this._filterItems.updateItemsVisiblity();
271
+ }
272
+ clearItem(name) {
273
+ const item = this.getItem(name);
274
+ if (!item) {
275
+ return;
276
+ }
277
+ item.clear();
278
+ }
279
+ updateItemConfig(name, params) {
280
+ const item = this.getItem(name);
281
+ if (!item) {
282
+ return;
283
+ }
284
+ item.label = params.label ?? item.label;
285
+ item.chipLabel = params.chipLabel ?? item.chipLabel;
286
+ this._filterItems.updateItemsVisiblity();
287
+ }
288
+ getItemValueChange$(name) {
289
+ const item = this.items.find((i) => i.name === name);
290
+ if (item) {
291
+ return item.value$
292
+ .pipe(map(() => {
293
+ return item.model;
294
+ }));
295
+ }
296
+ return null;
297
+ }
298
+ changeVisibility(state) {
299
+ if (state === this.showFilterMenu) {
300
+ return;
301
+ }
302
+ if (!state) {
303
+ this.closed.emit();
304
+ return this._destroyFilterDrawer();
305
+ }
306
+ if (!this.hasVisibleItemOrSorting) {
307
+ return;
308
+ }
309
+ this._listenEscButton();
310
+ this.opened.emit();
311
+ this._filterOverlay.open();
312
+ }
313
+ init() {
314
+ const data = this._filterItems.valuesAsQuery();
315
+ this._sort = this._filterItems.getSort();
316
+ if (this.config.init) {
317
+ this.config.init(data, this._sort, this);
318
+ }
319
+ this._updateChipsVisibility();
320
+ this.items
321
+ .forEach((item) => {
322
+ item.init(item, this);
323
+ });
324
+ }
325
+ clear(event = null) {
326
+ if (event) {
327
+ event.stopPropagation();
328
+ }
329
+ this._filterItems.filtersClear();
330
+ if (this.config.clear) {
331
+ this.config.clear();
332
+ }
333
+ this.keyword = '';
334
+ }
335
+ /**
336
+ * Close filter window and do change callback
337
+ */
338
+ search(event) {
339
+ this.changeVisibilityClick(false, event);
340
+ }
341
+ reload(event = null) {
342
+ if (event) {
343
+ event.preventDefault();
344
+ event.stopPropagation();
345
+ }
346
+ const data = this._filterItems.valuesAsQuery();
347
+ const el = this.reloadEl?.nativeElement;
348
+ if (el) {
349
+ el.style.transition = 'all 0.75s 0.0s';
350
+ el.style.transform = 'rotate(360deg)';
351
+ setTimeout(() => {
352
+ el.style.transition = null;
353
+ el.style.transform = null;
354
+ }, 1000);
355
+ }
356
+ if (this.config.reload) {
357
+ this.config.reload(data, this._filterItems.getSort());
358
+ }
359
+ }
360
+ getItem(name) {
361
+ return this.items
362
+ .find((item) => item.name === name);
363
+ }
364
+ fetchQueryParams() {
365
+ this._externalParams.fetchQueryParams();
366
+ }
367
+ /**
368
+ * Call change callback and apply new filter values
369
+ */
370
+ change() {
371
+ const data = this._filterItems.valuesAsQuery();
372
+ const sort = this._filterItems.getSort();
373
+ const sortingChanged = ((!sort || !this._sort) && sort !== this._sort)
374
+ || (sort && this._sort && !objectsAreEquals(this._sort, sort));
375
+ if (sortingChanged) {
376
+ this._sort = sort;
377
+ if (this.config.sortChange) {
378
+ this.config.sortChange(data, sort);
379
+ }
380
+ }
381
+ if (this.config.change) {
382
+ this.config.change(data, sort);
383
+ }
384
+ this._updateChipsVisibility();
385
+ // visibility for actions can depend on filters state
386
+ this._actions.updateActionsVisibility();
387
+ }
388
+ /**
389
+ * Update filter actions config
390
+ *
391
+ * @param actions
392
+ */
393
+ updateActions(actions) {
394
+ this._actions.initActions(actions);
395
+ }
396
+ /**
397
+ * Show "Filters" button
398
+ */
399
+ showFiltersBtn() {
400
+ this._filtersBtnVisible$.next(true);
401
+ }
402
+ /**
403
+ * Hide "Filters" button
404
+ */
405
+ hideFiltersBtn() {
406
+ this._filtersBtnVisible$.next(false);
407
+ }
408
+ /**
409
+ * Show "Keyword" field if it present
410
+ */
411
+ showKeywordField() {
412
+ this._keywordVisible$.next(true);
413
+ }
414
+ /**
415
+ * Hide "Keyword" field if it present
416
+ */
417
+ hideKeywordField() {
418
+ this._keywordVisible$.next(false);
419
+ }
420
+ /**
421
+ * Go through actions and check show() callback and update visible actions
422
+ */
423
+ updateActionsVisibility() {
424
+ this._actions.updateActionsVisibility();
425
+ }
426
+ /**
427
+ * Go through actions and check disabled() callback and update disabled state
428
+ */
429
+ updateDisabledState() {
430
+ this._actions.updateDisabledState();
431
+ }
432
+ setItems(items) {
433
+ this._filterItems.destroyItems();
434
+ this.config.items = items;
435
+ this._filterItems.setConfig(this._config);
436
+ this._externalParams.initItems();
437
+ this._syncSearchInputWithKeyword();
438
+ }
439
+ keywordChange(keyword) {
440
+ this._keyword$.next(keyword);
441
+ }
442
+ updateSortings(items) {
443
+ this._filterItems.updateSortingItemsValues(items);
444
+ }
445
+ _initFilterWithConfig(config) {
446
+ if (this.config) {
447
+ this._filterItems.destroyItems();
448
+ }
449
+ config = {
450
+ ...(this._defaultConfig || {}),
451
+ ...config,
452
+ };
453
+ this._config = new FsFilterConfig(config);
454
+ this._actions.setConfig(this._config);
455
+ this._filterItems.setConfig(this._config);
456
+ this._externalParams.setConfig(this._config);
457
+ this._syncSearchInputWithKeyword();
458
+ if (this.config.reloadWhenConfigChanged) {
459
+ this.change();
460
+ }
461
+ }
462
+ _destroyFilterDrawer() {
463
+ this._filterOverlay.close();
464
+ }
465
+ _updateWindowWidth() {
466
+ this.windowDesktop = window.innerWidth > 1200;
467
+ }
468
+ _listenEscButton() {
469
+ this._zone.runOutsideAngular(() => {
470
+ fromEvent(window, 'keyup')
471
+ .pipe(filter((event) => event.code === 'Escape'), takeUntil(this.closed), takeUntil(this._destroy$))
472
+ .subscribe(() => {
473
+ this._zone.run(() => {
474
+ this.changeVisibility(false);
475
+ });
476
+ });
477
+ });
478
+ }
479
+ _listenWindowResize() {
480
+ this._zone.runOutsideAngular(() => {
481
+ fromEvent(window, 'resize')
482
+ .pipe(debounceTime(100), takeUntil(this._destroy$))
483
+ .subscribe(() => {
484
+ this._zone.run(() => {
485
+ this._updateWindowWidth();
486
+ });
487
+ });
488
+ });
489
+ }
490
+ _initAutoReload() {
491
+ if (this.config.autoReload) {
492
+ interval(this.config.autoReload.seconds * 1000)
493
+ .pipe(filter(() => this.autoReload), takeUntil(this._destroy$))
494
+ .subscribe(() => {
495
+ this.reload(null);
496
+ });
497
+ }
498
+ }
499
+ _listenInputChanges() {
500
+ this._keyword$
501
+ .pipe(debounceTime(200), distinctUntilChanged(), takeUntil(this._destroy$))
502
+ .subscribe((value) => {
503
+ const keywordItem = this._filterItems.keywordItem;
504
+ keywordItem.model = value;
505
+ this.change();
506
+ });
507
+ }
508
+ _syncSearchInputWithKeyword() {
509
+ const keywordItem = this._filterItems.keywordItem;
510
+ if (keywordItem) {
511
+ this.keyword = keywordItem.model;
512
+ this.searchPlaceholder = keywordItem.label || 'Search';
513
+ }
514
+ }
515
+ _listenInternalItemsChange() {
516
+ this._filterItems
517
+ .itemsChange$
518
+ .pipe(takeUntil(this._destroy$))
519
+ .subscribe(() => {
520
+ this.change();
521
+ });
522
+ }
523
+ _initOverlay() {
524
+ this._filterOverlay.setClearFn(this.clear.bind(this));
525
+ this._filterOverlay.setDoneFn(this.hide.bind(this));
526
+ }
527
+ // We may need some time to recieve external params and after that ready can be emitted
528
+ _listenWhenFilterReady() {
529
+ combineLatest([
530
+ this._externalParams.pending$,
531
+ this.itemsReady$,
532
+ ])
533
+ .pipe(filter(([pendingParams, itemsReady]) => !pendingParams && itemsReady), takeUntil(this._destroy$))
534
+ .subscribe(() => {
535
+ this.init();
536
+ this._syncSearchInputWithKeyword();
537
+ this.ready.emit();
538
+ });
539
+ }
540
+ _updateChipsVisibility() {
541
+ const hasFilterChips = this._filterItems.items
542
+ .some((item) => {
543
+ return item.isChipVisible;
544
+ });
545
+ this._hasFilterChips$.next(hasFilterChips);
546
+ }
547
+ }
548
+ FilterComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterComponent, deps: [{ token: FS_FILTER_CONFIG, optional: true }, { token: i1.FsFilterOverlayService }, { token: i0.NgZone }, { token: i2.ExternalParamsController }, { token: i3.FsFilterItemsStore }, { token: i4.ActionsController }, { token: i5.SavedFiltersController }, { token: i2.ExternalParamsController }], target: i0.ɵɵFactoryTarget.Component });
549
+ FilterComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: FilterComponent, selector: "fs-filter", inputs: { setConfig: ["config", "setConfig"], setFilter: ["filter", "setFilter"], showSortBy: "showSortBy", showFilterInput: "showFilterInput" }, outputs: { closed: "closed", opened: "opened", ready: "ready" }, host: { properties: { "class.filters-open": "this.showFilterMenu", "class.window-desktop": "this.windowDesktop", "class.fs-filter": "this.fsFilterClass", "class.has-keyword": "this.hasKeyword" } }, providers: [
550
+ FsFilterOverlayService,
551
+ ExternalParamsController,
552
+ PersistanceParamsController,
553
+ QueryParamsController,
554
+ FocusControllerService,
555
+ FsFilterItemsStore,
556
+ SavedFiltersController,
557
+ ActionsController,
558
+ ], queries: [{ propertyName: "statusBar", first: true, predicate: FilterStatusBarDirective, descendants: true }], viewQueries: [{ propertyName: "keywordMatInput", first: true, predicate: ["keywordMatInput"], descendants: true, read: MatInput }, { propertyName: "reloadEl", first: true, predicate: ["reloadEl"], descendants: true, read: ElementRef }], ngImport: i0, template: "<div class=\"filter-container\">\n <div class=\"filter-inner-container\">\n <ng-container *ngIf=\"!hasKeyword; else filterKeyword\">\n <div>\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </div>\n </ng-container>\n\n <ng-container [ngTemplateOutlet]=\"filterToolbar\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"filterActions\"></ng-container>\n </div>\n\n <ng-container *ngIf=\"hasKeyword\">\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </ng-container>\n</div>\n\n<ng-template #filterStatusBarChips>\n <ng-container *ngIf=\"statusBar\">\n <div class=\"filter-status-container\" [ngClass]=\"{ 'has-status': !!filterStatus.textContent }\">\n <div class=\"filter-status\" #filterStatus>\n <ng-container *ngTemplateOutlet=\"statusBar.templateRef\"></ng-container>\n </div>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"config.chips && hasFilterChips$ | async\">\n <fs-filter-chips\n class=\"filter-chips\"\n [filters]=\"items\">\n </fs-filter-chips>\n </ng-container>\n</ng-template>\n\n<ng-template #filterKeyword>\n <div class=\"filter-keyword\">\n <ng-container *ngIf=\"(keywordVisible$ | async) && !keywordItem?.hide \">\n <mat-form-field\n [floatLabel]=\"'never'\"\n class=\"form-field-padless\"\n [ngClass]=\"search\"\n appearance=\"outline\">\n <span matPrefix>\n <mat-icon matPrefix>search</mat-icon>\n </span>\n <input\n #keywordMatInput\n matInput\n [(ngModel)]=\"keyword\"\n (ngModelChange)=\"keywordChange($event)\"\n name=\"filter-input\"\n [fsClear]=\"true\"\n [placeholder]=\"searchPlaceholder\">\n </mat-form-field>\n </ng-container>\n </div>\n</ng-template>\n\n<ng-template #filterActions>\n <div class=\"filter-actions\">\n <fs-filter-actions\n *ngIf=\"actionsVisible$ | async\"\n [actions]=\"actions$ | async\"\n [kebabActions]=\"menuActions$ | async\">\n </fs-filter-actions>\n </div>\n</ng-template>\n\n<ng-template #filterToolbar>\n <div class=\"filter-toobar\">\n <ng-container *ngIf=\"filtersBtnVisible$ | async\">\n <ng-container *ngIf=\"hasVisibleItemOrSorting\">\n <a\n mat-button\n class=\"filters-button\"\n [ngClass]=\"{\n 'mat-raised-button': config.button.style === 'raised' && config.button.label,\n 'mat-flat-button': config.button.style === 'flat' && config.button.label,\n 'mat-stroked-button': config.button.style === 'stroked' && config.button.label,\n 'mat-button': config.button.style === 'basic' && config.button.label,\n 'mat-icon-button': config.button.style === 'icon' || !config.button.label\n }\"\n (click)=\"changeVisibilityClick(!showFilterMenu, $event)\"\n [color]=\"config.button.color\">\n <mat-icon *ngIf=\"config.button.icon\">{{config.button.icon}}</mat-icon>\n {{ config.button.label }}\n </a>\n </ng-container>\n </ng-container> \n <ng-container *ngIf=\"config.reload\">\n <a\n mat-icon-button\n (click)=\"reload($event)\"\n class=\"reload\">\n <mat-icon #reloadEl>refresh</mat-icon>\n </a>\n </ng-container>\n <ng-container *ngIf=\"config.autoReload\">\n <mat-slide-toggle \n name=\"autoReload\" \n class=\"auto-reload\"\n [(ngModel)]=\"autoReload\">\n Auto refresh\n </mat-slide-toggle>\n </ng-container>\n </div>\n</ng-template>\n", styles: [":host{margin-bottom:20px;display:block}:host.has-keyword .filter-status-container.has-status{margin-top:4px}:host:not(.has-keyword) .filter-status-container:not(.has-status)+fs-filter-chips{margin-top:0}:host:not(.has-keyword) .filter-inner-container{display:flex}:host:not(.has-keyword) .filter-toobar{justify-content:flex-end}.filter-status-container{flex-grow:1;display:flex;justify-content:center;flex-direction:column;align-self:flex-end}.filter-status-container .filter-status{overflow:hidden;text-overflow:ellipsis;line-height:17px}.filter-container{width:100%}.filter-inner-container{flex-direction:row;box-sizing:border-box;display:flex;position:relative;align-items:center}.filter-inner-container .filter-keyword{flex-direction:row;box-sizing:border-box;display:flex;align-items:center;min-width:0}.filter-inner-container .filter-keyword mat-form-field{max-width:100%;min-width:100px;margin-right:6px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-wrapper{margin-bottom:0}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix .mat-icon{font-size:22px;color:#626262}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-infix{width:250px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix,.filter-inner-container .filter-keyword mat-form-field .mat-form-field-suffix{top:.45em}.filter-inner-container .filter-keyword .mat-form-field-suffix{display:flex}.filter-inner-container .filter-keyword .mat-form-field-suffix .mat-icon-button{height:24px;width:24px}.filter-actions{display:flex;align-items:center}.filter-toobar{flex:1;white-space:nowrap;display:flex;align-items:center}.filter-toobar .reload{margin-left:-6px}.results{min-height:90px;position:relative;overflow-x:auto;overflow-y:hidden}fs-filter-chips{margin-top:4px;display:block}@media screen and (min-width: 599px){.filter-actions{margin-left:10px}.filters-button+fs-filter-actions:not(:empty){margin-left:5px}}@media screen and (max-width: 600px){.filter-actions{margin-left:5px}.filter-actions .filters-button{font-size:0;padding:0;min-width:unset;width:36px;height:36px;border-radius:50%;box-shadow:unset!important}}@media screen and (min-width: 1200px){html.fs-filter-open body{margin-right:350px}.fs-filter-backdrop{display:none}}html.fs-filter-open{scrollbar-width:none}:host ::ng-deep .auto-reload.mat-checked .mat-slide-toggle-thumb-container{transform:translate3d(12px,0,0)}:host ::ng-deep .auto-reload:not(.mat-checked) .mat-slide-toggle-content{color:#ccc}:host ::ng-deep .auto-reload .mat-slide-toggle-thumb,:host ::ng-deep .auto-reload .mat-slide-toggle-thumb-container{height:15px;width:15px}:host ::ng-deep .auto-reload .mat-slide-toggle-content{font-size:90%}:host ::ng-deep .auto-reload .mat-slide-toggle-bar{width:26px;height:10px;border-radius:10px}\n"], dependencies: [{ kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i8.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i9.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i9.MatPrefix, selector: "[matPrefix]" }, { kind: "directive", type: i10.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i11.MatAnchor, selector: "a[mat-button], a[mat-raised-button], a[mat-icon-button], a[mat-fab], a[mat-mini-fab], a[mat-stroked-button], a[mat-flat-button]", inputs: ["disabled", "disableRipple", "color", "tabIndex"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i12.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["disabled", "disableRipple", "color", "tabIndex"], exportAs: ["matSlideToggle"] }, { kind: "directive", type: i13.FsFormNoFsValidatorsDirective, selector: "[ngModel]:not([required]):not([fsFormRequired]):not([fsFormCompare]):not([fsFormDateRange]):not([fsFormEmail]):not([fsFormEmails]):not([fsFormFunction]):not([fsFormGreater]):not([fsFormGreaterEqual]):not([fsFormInteger]):not([fsFormLesser]):not([fsFormMax]):not([fsFormMaxLength]):not([fsFormMin]):not([fsFormMinLength]):not([fsFormNumeric]):not([fsFormPattern]):not([fsFormPhone]):not([fsFormUrl]):not([validate])" }, { kind: "directive", type: i13.FsButtonDirective, selector: "[mat-raised-button]:not([fsFormButtonStandalone]),[mat-button]:not([fsFormButtonStandalone]),[mat-flat-button]:not([fsFormButtonStandalone]),[mat-stroked-button]:not([fsFormButtonStandalone])", inputs: ["name", "dirtySubmit", "form"] }, { kind: "component", type: i14.FsClearComponent, selector: "[fsClear]", inputs: ["ngModel", "fsClear"], outputs: ["ngModelChange", "cleared"] }, { kind: "component", type: i15.FsFilterChipsComponent, selector: "fs-filter-chips", inputs: ["filters"] }, { kind: "component", type: i16.FsFilterActionsComponent, selector: "fs-filter-actions", inputs: ["kebabActions", "actions"] }, { kind: "pipe", type: i6.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
559
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: FilterComponent, decorators: [{
560
+ type: Component,
561
+ args: [{ selector: 'fs-filter', providers: [
562
+ FsFilterOverlayService,
563
+ ExternalParamsController,
564
+ PersistanceParamsController,
565
+ QueryParamsController,
566
+ FocusControllerService,
567
+ FsFilterItemsStore,
568
+ SavedFiltersController,
569
+ ActionsController,
570
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"filter-container\">\n <div class=\"filter-inner-container\">\n <ng-container *ngIf=\"!hasKeyword; else filterKeyword\">\n <div>\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </div>\n </ng-container>\n\n <ng-container [ngTemplateOutlet]=\"filterToolbar\"></ng-container>\n <ng-container [ngTemplateOutlet]=\"filterActions\"></ng-container>\n </div>\n\n <ng-container *ngIf=\"hasKeyword\">\n <ng-container *ngTemplateOutlet=\"filterStatusBarChips\"></ng-container>\n </ng-container>\n</div>\n\n<ng-template #filterStatusBarChips>\n <ng-container *ngIf=\"statusBar\">\n <div class=\"filter-status-container\" [ngClass]=\"{ 'has-status': !!filterStatus.textContent }\">\n <div class=\"filter-status\" #filterStatus>\n <ng-container *ngTemplateOutlet=\"statusBar.templateRef\"></ng-container>\n </div>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"config.chips && hasFilterChips$ | async\">\n <fs-filter-chips\n class=\"filter-chips\"\n [filters]=\"items\">\n </fs-filter-chips>\n </ng-container>\n</ng-template>\n\n<ng-template #filterKeyword>\n <div class=\"filter-keyword\">\n <ng-container *ngIf=\"(keywordVisible$ | async) && !keywordItem?.hide \">\n <mat-form-field\n [floatLabel]=\"'never'\"\n class=\"form-field-padless\"\n [ngClass]=\"search\"\n appearance=\"outline\">\n <span matPrefix>\n <mat-icon matPrefix>search</mat-icon>\n </span>\n <input\n #keywordMatInput\n matInput\n [(ngModel)]=\"keyword\"\n (ngModelChange)=\"keywordChange($event)\"\n name=\"filter-input\"\n [fsClear]=\"true\"\n [placeholder]=\"searchPlaceholder\">\n </mat-form-field>\n </ng-container>\n </div>\n</ng-template>\n\n<ng-template #filterActions>\n <div class=\"filter-actions\">\n <fs-filter-actions\n *ngIf=\"actionsVisible$ | async\"\n [actions]=\"actions$ | async\"\n [kebabActions]=\"menuActions$ | async\">\n </fs-filter-actions>\n </div>\n</ng-template>\n\n<ng-template #filterToolbar>\n <div class=\"filter-toobar\">\n <ng-container *ngIf=\"filtersBtnVisible$ | async\">\n <ng-container *ngIf=\"hasVisibleItemOrSorting\">\n <a\n mat-button\n class=\"filters-button\"\n [ngClass]=\"{\n 'mat-raised-button': config.button.style === 'raised' && config.button.label,\n 'mat-flat-button': config.button.style === 'flat' && config.button.label,\n 'mat-stroked-button': config.button.style === 'stroked' && config.button.label,\n 'mat-button': config.button.style === 'basic' && config.button.label,\n 'mat-icon-button': config.button.style === 'icon' || !config.button.label\n }\"\n (click)=\"changeVisibilityClick(!showFilterMenu, $event)\"\n [color]=\"config.button.color\">\n <mat-icon *ngIf=\"config.button.icon\">{{config.button.icon}}</mat-icon>\n {{ config.button.label }}\n </a>\n </ng-container>\n </ng-container> \n <ng-container *ngIf=\"config.reload\">\n <a\n mat-icon-button\n (click)=\"reload($event)\"\n class=\"reload\">\n <mat-icon #reloadEl>refresh</mat-icon>\n </a>\n </ng-container>\n <ng-container *ngIf=\"config.autoReload\">\n <mat-slide-toggle \n name=\"autoReload\" \n class=\"auto-reload\"\n [(ngModel)]=\"autoReload\">\n Auto refresh\n </mat-slide-toggle>\n </ng-container>\n </div>\n</ng-template>\n", styles: [":host{margin-bottom:20px;display:block}:host.has-keyword .filter-status-container.has-status{margin-top:4px}:host:not(.has-keyword) .filter-status-container:not(.has-status)+fs-filter-chips{margin-top:0}:host:not(.has-keyword) .filter-inner-container{display:flex}:host:not(.has-keyword) .filter-toobar{justify-content:flex-end}.filter-status-container{flex-grow:1;display:flex;justify-content:center;flex-direction:column;align-self:flex-end}.filter-status-container .filter-status{overflow:hidden;text-overflow:ellipsis;line-height:17px}.filter-container{width:100%}.filter-inner-container{flex-direction:row;box-sizing:border-box;display:flex;position:relative;align-items:center}.filter-inner-container .filter-keyword{flex-direction:row;box-sizing:border-box;display:flex;align-items:center;min-width:0}.filter-inner-container .filter-keyword mat-form-field{max-width:100%;min-width:100px;margin-right:6px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-wrapper{margin-bottom:0}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix .mat-icon{font-size:22px;color:#626262}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-infix{width:250px}.filter-inner-container .filter-keyword mat-form-field .mat-form-field-prefix,.filter-inner-container .filter-keyword mat-form-field .mat-form-field-suffix{top:.45em}.filter-inner-container .filter-keyword .mat-form-field-suffix{display:flex}.filter-inner-container .filter-keyword .mat-form-field-suffix .mat-icon-button{height:24px;width:24px}.filter-actions{display:flex;align-items:center}.filter-toobar{flex:1;white-space:nowrap;display:flex;align-items:center}.filter-toobar .reload{margin-left:-6px}.results{min-height:90px;position:relative;overflow-x:auto;overflow-y:hidden}fs-filter-chips{margin-top:4px;display:block}@media screen and (min-width: 599px){.filter-actions{margin-left:10px}.filters-button+fs-filter-actions:not(:empty){margin-left:5px}}@media screen and (max-width: 600px){.filter-actions{margin-left:5px}.filter-actions .filters-button{font-size:0;padding:0;min-width:unset;width:36px;height:36px;border-radius:50%;box-shadow:unset!important}}@media screen and (min-width: 1200px){html.fs-filter-open body{margin-right:350px}.fs-filter-backdrop{display:none}}html.fs-filter-open{scrollbar-width:none}:host ::ng-deep .auto-reload.mat-checked .mat-slide-toggle-thumb-container{transform:translate3d(12px,0,0)}:host ::ng-deep .auto-reload:not(.mat-checked) .mat-slide-toggle-content{color:#ccc}:host ::ng-deep .auto-reload .mat-slide-toggle-thumb,:host ::ng-deep .auto-reload .mat-slide-toggle-thumb-container{height:15px;width:15px}:host ::ng-deep .auto-reload .mat-slide-toggle-content{font-size:90%}:host ::ng-deep .auto-reload .mat-slide-toggle-bar{width:26px;height:10px;border-radius:10px}\n"] }]
571
+ }], ctorParameters: function () { return [{ type: i17.FsFilterConfig, decorators: [{
572
+ type: Optional
573
+ }, {
574
+ type: Inject,
575
+ args: [FS_FILTER_CONFIG]
576
+ }] }, { type: i1.FsFilterOverlayService }, { type: i0.NgZone }, { type: i2.ExternalParamsController }, { type: i3.FsFilterItemsStore }, { type: i4.ActionsController }, { type: i5.SavedFiltersController }, { type: i2.ExternalParamsController }]; }, propDecorators: { setConfig: [{
577
+ type: Input,
578
+ args: ['config']
579
+ }], setFilter: [{
580
+ type: Input,
581
+ args: ['filter']
582
+ }], showSortBy: [{
583
+ type: Input
584
+ }], showFilterInput: [{
585
+ type: Input
586
+ }], closed: [{
587
+ type: Output
588
+ }], opened: [{
589
+ type: Output
590
+ }], ready: [{
591
+ type: Output
592
+ }], statusBar: [{
593
+ type: ContentChild,
594
+ args: [FilterStatusBarDirective]
595
+ }], keywordMatInput: [{
596
+ type: ViewChild,
597
+ args: ['keywordMatInput', { read: MatInput }]
598
+ }], reloadEl: [{
599
+ type: ViewChild,
600
+ args: ['reloadEl', { read: ElementRef }]
601
+ }], showFilterMenu: [{
602
+ type: HostBinding,
603
+ args: ['class.filters-open']
604
+ }], windowDesktop: [{
605
+ type: HostBinding,
606
+ args: ['class.window-desktop']
607
+ }], fsFilterClass: [{
608
+ type: HostBinding,
609
+ args: ['class.fs-filter']
610
+ }], hasKeyword: [{
611
+ type: HostBinding,
612
+ args: ['class.has-keyword']
613
+ }] } });
614
614
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsdGVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9hcHAvY29tcG9uZW50cy9maWx0ZXIvZmlsdGVyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9hcHAvY29tcG9uZW50cy9maWx0ZXIvZmlsdGVyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCx1QkFBdUIsRUFDdkIsU0FBUyxFQUNULFlBQVksRUFDWixVQUFVLEVBQ1YsWUFBWSxFQUNaLFdBQVcsRUFDWCxNQUFNLEVBQ04sS0FBSyxFQUNMLE1BQU0sRUFHTixRQUFRLEVBQ1IsTUFBTSxFQUNOLFNBQVMsR0FDVixNQUFNLGVBQWUsQ0FBQztBQUV2QixPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFFbkQsT0FBTyxFQUFFLGVBQWUsRUFBRSxhQUFhLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBYyxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDaEcsT0FBTyxFQUFFLFlBQVksRUFBRSxvQkFBb0IsRUFBRSxNQUFNLEVBQUUsR0FBRyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBRTVGLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGtDQUFrQyxDQUFDO0FBQ3JFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBS3pELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUc1RCxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSxtREFBbUQsQ0FBQztBQUM3RixPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSxzRUFBc0UsQ0FBQztBQUNuSCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxnRUFBZ0UsQ0FBQztBQUN2RyxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxpRUFBaUUsQ0FBQztBQUN6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QkFBaUI7cUJBQ2xCLG1CQUNnQix1QkFBdUIsQ0FBQyxNQUFNOzswQkF5RDVDLFFBQVE7OzBCQUFJLE1BQU07MkJBQUMsZ0JBQWdCOzBSQXBEM0IsU0FBUztzQkFEbkIsS0FBSzt1QkFBQyxRQUFRO2dCQU1KLFNBQVM7c0JBRG5CLEtBQUs7dUJBQUMsUUFBUTtnQkFLQyxVQUFVO3NCQUF6QixLQUFLO2dCQUNVLGVBQWU7c0JBQTlCLEtBQUs7Z0JBQ1csTUFBTTtzQkFBdEIsTUFBTTtnQkFDVSxNQUFNO3NCQUF0QixNQUFNO2dCQUNVLEtBQUs7c0JBQXJCLE1BQU07Z0JBR0EsU0FBUztzQkFEZixZQUFZO3VCQUFDLHdCQUF3QjtnQkFJL0IsZUFBZTtzQkFEckIsU0FBUzt1QkFBQyxpQkFBaUIsRUFBRSxFQUFFLElBQUksRUFBRSxRQUFRLEVBQUU7Z0JBSXpDLFFBQVE7c0JBRGQsU0FBUzt1QkFBQyxVQUFVLEVBQUUsRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFO2dCQUlwQyxjQUFjO3NCQURwQixXQUFXO3VCQUFDLG9CQUFvQjtnQkFJMUIsYUFBYTtzQkFEbkIsV0FBVzt1QkFBQyxzQkFBc0I7Z0JBSTVCLGFBQWE7c0JBRG5CLFdBQVc7dUJBQUMsaUJBQWlCO2dCQUluQixVQUFVO3NCQURwQixXQUFXO3VCQUFDLG1CQUFtQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBDb21wb25lbnQsXG4gIENvbnRlbnRDaGlsZCxcbiAgRWxlbWVudFJlZixcbiAgRXZlbnRFbWl0dGVyLFxuICBIb3N0QmluZGluZyxcbiAgSW5qZWN0LFxuICBJbnB1dCxcbiAgTmdab25lLFxuICBPbkRlc3Ryb3ksXG4gIE9uSW5pdCxcbiAgT3B0aW9uYWwsXG4gIE91dHB1dCxcbiAgVmlld0NoaWxkLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuaW1wb3J0IHsgTWF0SW5wdXQgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9pbnB1dCc7XG5cbmltcG9ydCB7IEJlaGF2aW9yU3ViamVjdCwgY29tYmluZUxhdGVzdCwgZnJvbUV2ZW50LCBpbnRlcnZhbCwgT2JzZXJ2YWJsZSwgU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgZGVib3VuY2VUaW1lLCBkaXN0aW5jdFVudGlsQ2hhbmdlZCwgZmlsdGVyLCBtYXAsIHRha2VVbnRpbCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuaW1wb3J0IHsgQWN0aW9uc0NvbnRyb2xsZXIgfSBmcm9tICcuLi8uLi9jbGFzc2VzL2FjdGlvbnMtY29udHJvbGxlcic7XG5pbXBvcnQgeyBvYmplY3RzQXJlRXF1YWxzIH0gZnJvbSAnLi4vLi4vaGVscGVycy9jb21wYXJlJztcbmltcG9ydCB7IElGaWx0ZXJTYXZlZEZpbHRlciwgSVNvcnRpbmdDaGFuZ2VFdmVudCB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgRnNGaWx0ZXJBY3Rpb24gfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL2FjdGlvbi5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgRmlsdGVyQ29uZmlnLCBGaWx0ZXJTb3J0LCBJRmlsdGVyQ29uZmlnSXRlbSwgU29ydEl0ZW0gfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL2NvbmZpZy5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgSVVwZGF0ZUZpbHRlckl0ZW1Db25maWcgfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL3VwZGF0ZS1maWx0ZXItaXRlbS5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgRnNGaWx0ZXJDb25maWcgfSBmcm9tICcuLi8uLi9tb2RlbHMvZmlsdGVyLWNvbmZpZyc7XG5pbXBvcnQgeyBCYXNlSXRlbSB9IGZyb20gJy4uLy4uL21vZGVscy9pdGVtcy9iYXNlLWl0ZW0nO1xuaW1wb3J0IHsgVGV4dEl0ZW0gfSBmcm9tICcuLi8uLi9tb2RlbHMvaXRlbXMvdGV4dC1pdGVtJztcbmltcG9ydCB7IEV4dGVybmFsUGFyYW1zQ29udHJvbGxlciB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL2V4dGVybmFsLXBhcmFtcy1jb250cm9sbGVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgUGVyc2lzdGFuY2VQYXJhbXNDb250cm9sbGVyIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvZXh0ZXJuYWwtcGFyYW1zL3BlcnNpc3RhbmNlLXBhcmFtcy1jb250cm9sbGVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgUXVlcnlQYXJhbXNDb250cm9sbGVyIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvZXh0ZXJuYWwtcGFyYW1zL3F1ZXJ5LXBhcmFtcy1jb250cm9sbGVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgU2F2ZWRGaWx0ZXJzQ29udHJvbGxlciB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL2V4dGVybmFsLXBhcmFtcy9zYXZlZC1maWx0ZXJzLWNvbnRyb2xsZXIuc2VydmljZSc7XG5pbXBvcnQgeyBGc0ZpbHRlck92ZXJsYXlTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvZmlsdGVyLW92ZXJsYXkuc2VydmljZSc7XG5pbXBvcnQgeyBGb2N1c0NvbnRyb2xsZXJTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvZm9jdXMtY29udHJvbGxlci5zZXJ2aWNlJztcbmltcG9ydCB7IEZzRmlsdGVySXRlbXNTdG9yZSB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL2l0ZW1zLXN0b3JlLnNlcnZpY2UnO1xuXG5pbXBvcnQgeyBGaWx0ZXJTdGF0dXNCYXJEaXJlY3RpdmUgfSBmcm9tICcuLy4uLy4uL2RpcmVjdGl2ZXMvc3RhdHVzLWJhci9zdGF0dXMtYmFyLmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBGU19GSUxURVJfQ09ORklHIH0gZnJvbSAnLi8uLi8uLi9pbmplY3RvcnMvZmlsdGVyLWNvbmZpZyc7XG5cblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnZnMtZmlsdGVyJyxcbiAgc3R5bGVVcmxzOiBbJy4vZmlsdGVyLmNvbXBvbmVudC5zY3NzJ10sXG4gIHRlbXBsYXRlVXJsOiAnLi9maWx0ZXIuY29tcG9uZW50Lmh0bWwnLFxuICBwcm92aWRlcnM6IFtcbiAgICBGc0ZpbHRlck92ZXJsYXlTZXJ2aWNlLFxuICAgIEV4dGVybmFsUGFyYW1zQ29udHJvbGxlcixcbiAgICBQZXJzaXN0YW5jZVBhcmFtc0NvbnRyb2xsZXIsXG4gICAgUXVlcnlQYXJhbXNDb250cm9sbGVyLFxuICAgIEZvY3VzQ29udHJvbGxlclNlcnZpY2UsXG4gICAgRnNGaWx0ZXJJdGVtc1N0b3JlLFxuICAgIFNhdmVkRmlsdGVyc0NvbnRyb2xsZXIsXG4gICAgQWN0aW9uc0NvbnRyb2xsZXIsXG4gIF0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBGaWx0ZXJDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSB7XG5cbiAgQElucHV0KCdjb25maWcnKVxuICBwdWJsaWMgc2V0IHNldENvbmZpZyhjb25maWcpIHtcbiAgICB0aGlzLl9pbml0RmlsdGVyV2l0aENvbmZpZyhjb25maWcpO1xuICB9XG5cbiAgQElucHV0KCdmaWx0ZXInKVxuICBwdWJsaWMgc2V0IHNldEZpbHRlcihjb25maWcpIHtcbiAgICB0aGlzLl9pbml0RmlsdGVyV2l0aENvbmZpZyhjb25maWcpO1xuICB9XG5cbiAgQElucHV0KCkgcHVibGljIHNob3dTb3J0Qnk6IGFueSA9IHRydWU7XG4gIEBJbnB1dCgpIHB1YmxpYyBzaG93RmlsdGVySW5wdXQgPSB0cnVlO1xuICBAT3V0cHV0KCkgcHVibGljIGNsb3NlZCA9IG5ldyBFdmVudEVtaXR0ZXI8YW55PigpO1xuICBAT3V0cHV0KCkgcHVibGljIG9wZW5lZCA9IG5ldyBFdmVudEVtaXR0ZXI8YW55PigpO1xuICBAT3V0cHV0KCkgcHVibGljIHJlYWR5ID0gbmV3IEV2ZW50RW1pdHRlcjx2b2lkPigpO1xuXG4gIEBDb250ZW50Q2hpbGQoRmlsdGVyU3RhdHVzQmFyRGlyZWN0aXZlKVxuICBwdWJsaWMgc3RhdHVzQmFyOiBGaWx0ZXJTdGF0dXNCYXJEaXJlY3RpdmU7XG5cbiAgQFZpZXdDaGlsZCgna2V5d29yZE1hdElucHV0JywgeyByZWFkOiBNYXRJbnB1dCB9KVxuICBwdWJsaWMga2V5d29yZE1hdElucHV0OiBNYXRJbnB1dDtcblxuICBAVmlld0NoaWxkKCdyZWxvYWRFbCcsIHsgcmVhZDogRWxlbWVudFJlZiB9KVxuICBwdWJsaWMgcmVsb2FkRWw6IEVsZW1lbnRSZWY7XG5cbiAgQEhvc3RCaW5kaW5nKCdjbGFzcy5maWx0ZXJzLW9wZW4nKVxuICBwdWJsaWMgc2hvd0ZpbHRlck1lbnUgPSBmYWxzZTtcblxuICBASG9zdEJpbmRpbmcoJ2NsYXNzLndpbmRvdy1kZXNrdG9wJylcbiAgcHVibGljIHdpbmRvd0Rlc2t0b3AgPSBmYWxzZTtcblxuICBASG9zdEJpbmRpbmcoJ2NsYXNzLmZzLWZpbHRlcicpXG4gIHB1YmxpYyBmc0ZpbHRlckNsYXNzID0gdHJ1ZTtcblxuICBASG9zdEJpbmRpbmcoJ2NsYXNzLmhhcy1rZXl3b3JkJylcbiAgcHVibGljIGdldCBoYXNLZXl3b3JkKCkge1xuICAgIHJldHVybiB0aGlzLl9maWx0ZXJJdGVtcy5oYXNLZXl3b3JkO1xuICB9XG5cbiAgcHVibGljIHNlYXJjaFBsYWNlaG9sZGVyID0gJ1NlYXJjaCc7XG4gIHB1YmxpYyBrZXl3b3JkID0gJyc7XG4gIHB1YmxpYyBhdXRvUmVsb2FkID0gdHJ1ZTtcblxuICBwcm90ZWN0ZWQgX2NvbmZpZzogRnNGaWx0ZXJDb25maWcgPSBudWxsO1xuXG4gIHByaXZhdGUgX3NvcnQ6IEZpbHRlclNvcnQ7XG4gIHByaXZhdGUgX2ZpbHRlcnNCdG5WaXNpYmxlJCA9IG5ldyBCZWhhdmlvclN1YmplY3QodHJ1ZSk7XG4gIHByaXZhdGUgX2tleXdvcmRWaXNpYmxlJCA9IG5ldyBCZWhhdmlvclN1YmplY3QodHJ1ZSk7XG4gIHByaXZhdGUgX2hhc0ZpbHRlckNoaXBzJCA9IG5ldyBCZWhhdmlvclN1YmplY3QoZmFsc2UpO1xuICBwcml2YXRlIF9rZXl3b3JkJCA9IG5ldyBTdWJqZWN0KCk7XG4gIHByaXZhdGUgX2Rlc3Ryb3kkID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBAT3B0aW9uYWwoKSBASW5qZWN0KEZTX0ZJTFRFUl9DT05GSUcpIHByaXZhdGUgX2RlZmF1bHRDb25maWc6IEZzRmlsdGVyQ29uZmlnLFxuICAgIHByaXZhdGUgX2ZpbHRlck92ZXJsYXk6IEZzRmlsdGVyT3ZlcmxheVNlcnZpY2UsXG4gICAgcHJpdmF0ZSBfem9uZTogTmdab25lLFxuICAgIHByaXZhdGUgX2V4dGVybmFsUGFyYW1zOiBFeHRlcm5hbFBhcmFtc0NvbnRyb2xsZXIsXG4gICAgcHJpdmF0ZSBfZmlsdGVySXRlbXM6IEZzRmlsdGVySXRlbXNTdG9yZSxcbiAgICBwcml2YXRlIF9hY3Rpb25zOiBBY3Rpb25zQ29udHJvbGxlcixcbiAgICBwcml2YXRlIF9zYXZlZEZpbHRlcnNDb250cm9sbGVyOiBTYXZlZEZpbHRlcnNDb250cm9sbGVyLFxuICAgIHByaXZhdGUgX2V4dGVybmFsUGFyYW1zQ29udHJvbGxlcjogRXh0ZXJuYWxQYXJhbXNDb250cm9sbGVyLFxuICApIHtcbiAgICB0aGlzLl9maWx0ZXJJdGVtcy5maWx0ZXIgPSB0aGlzO1xuICAgIHRoaXMuX2xpc3RlbldoZW5GaWx0ZXJSZWFkeSgpO1xuICAgIHRoaXMuX3VwZGF0ZVdpbmRvd1dpZHRoKCk7XG5cbiAgICB0aGlzLl9maWx0ZXJPdmVybGF5LmF0dGFjaCRcbiAgICAgIC5waXBlKFxuICAgICAgICB0YWtlVW50aWwodGhpcy5fZGVzdHJveSQpLFxuICAgICAgKVxuICAgICAgLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgIHRoaXMuc2hvd0ZpbHRlck1lbnUgPSB0cnVlO1xuICAgICAgfSk7XG5cbiAgICB0aGlzLl9maWx0ZXJPdmVybGF5LmRldGFjaCRcbiAgICAgIC5waXBlKFxuICAgICAgICB0YWtlVW50aWwodGhpcy5fZGVzdHJveSQpLFxuICAgICAgKVxuICAgICAgLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgIHRoaXMuc2hvd0ZpbHRlck1lbnUgPSBmYWxzZTtcbiAgICAgIH0pO1xuXG4gICAgdGhpcy5fbGlzdGVuV2luZG93UmVzaXplKCk7XG4gIH1cblxuICBwdWJsaWMgZ2V0IGNvbmZpZygpOiBGc0ZpbHRlckNvbmZpZyB7XG4gICAgcmV0dXJuIHRoaXMuX2NvbmZpZztcbiAgfVxuXG4gIHB1YmxpYyBnZXQgZmlsdGVyUGFyYW1zKCkge1xuICAgIHJldHVybiB0aGlzLl9maWx0ZXJJdGVtcy52YWx1ZXMoKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgZmlsdGVyUGFyYW1zUXVlcnkoKTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4ge1xuICAgIHJldHVybiB0aGlzLl9maWx0ZXJJdGVtcy52YWx1ZXNBc1F1ZXJ5KCk7XG4gIH1cblxuICBwdWJsaWMgZ2V0IGl0ZW1zKCkge1xuICAgIHJldHVybiB0aGlzLl9maWx0ZXJJdGVtcy5pdGVtcztcbiAgfVxuXG4gIHB1YmxpYyBnZXQgdmlzaWJsZUl0ZW1zKCkge1xuICAgIHJldHVybiB0aGlzLl9maWx0ZXJJdGVtcy52aXNpYmxlSXRlbXM7XG4gIH1cblxuICBwdWJsaWMgZ2V0IGtleXdvcmRJdGVtKCk6IFRleHRJdGVtIHwgbnVsbCB7XG4gICAgcmV0dXJuIHRoaXMuX2ZpbHRlckl0ZW1zLmtleXdvcmRJdGVtO1xuICB9XG5cbiAgcHVibGljIGdldCBpdGVtc1JlYWR5JCgpOiBPYnNlcnZhYmxlPGJvb2xlYW4+IHtcbiAgICByZXR1cm4gdGhpcy5fZmlsdGVySXRlbXMucmVhZHkkO1xuICB9XG5cbiAgcHVibGljIGdldCBoYXNGaWx0ZXJDaGlwcyQoKTogT2JzZXJ2YWJsZTxib29sZWFuPiB7XG4gICAgcmV0dXJuIHRoaXMuX2hhc0ZpbHRlckNoaXBzJC5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgaGFzVmlzaWJsZUl0ZW1PclNvcnRpbmcoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMudmlzaWJsZUl0ZW1zLmxlbmd0aCA+IDAgfHwgISF0aGlzLl9maWx0ZXJJdGVtcy5zb3J0QnlJdGVtO1xuICB9XG5cbiAgcHVibGljIGdldCBmaWx0ZXJzQnRuVmlzaWJsZSQoKTogT2JzZXJ2YWJsZTxib29sZWFuPiB7XG4gICAgcmV0dXJuIHRoaXMuX2ZpbHRlcnNCdG5WaXNpYmxlJC5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQga2V5d29yZFZpc2libGUkKCk6IE9ic2VydmFibGU8Ym9vbGVhbj4ge1xuICAgIHJldHVybiB0aGlzLl9rZXl3b3JkVmlzaWJsZSQuYXNPYnNlcnZhYmxlKCk7XG4gIH1cblxuICBwdWJsaWMgZ2V0IGFjdGlvbnNWaXNpYmxlJCgpIHtcbiAgICByZXR1cm4gdGhpcy5fYWN0aW9ucy52aXNpYmxlJDtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgYWN0aW9ucyQoKSB7XG4gICAgcmV0dXJuIHRoaXMuX2FjdGlvbnMuYWN0aW9ucyQ7XG4gIH1cblxuICBwdWJsaWMgZ2V0IG1lbnVBY3Rpb25zJCgpIHtcbiAgICByZXR1cm4gdGhpcy5fYWN0aW9ucy5tZW51QWN0aW9ucyQ7XG4gIH1cblxuICBwdWJsaWMgc2V0IGFjdGl2ZVNhdmVkRmlsdGVyKHNhdmVkRmlsdGVyOiBJRmlsdGVyU2F2ZWRGaWx0ZXIpIHtcbiAgICB0aGlzLl9leHRlcm5hbFBhcmFtcy5zZXRBY3RpdmVTYXZlZEZpbHRlcihzYXZlZEZpbHRlcik7XG4gIH1cblxuICBwdWJsaWMgZ2V0IGFjdGl2ZVNhdmVkRmlsdGVyKCk6IElGaWx0ZXJTYXZlZEZpbHRlciB7XG4gICAgcmV0dXJuIHRoaXMuX3NhdmVkRmlsdGVyc0NvbnRyb2xsZXIuYWN0aXZlRmlsdGVyO1xuICB9XG5cbiAgcHVibGljIGdldCBzYXZlZEZpbHRlcnMoKTogSUZpbHRlclNhdmVkRmlsdGVyW10ge1xuICAgIHJldHVybiB0aGlzLl9zYXZlZEZpbHRlcnNDb250cm9sbGVyLnNhdmVkRmlsdGVycztcbiAgfVxuXG4gIHB1YmxpYyBuZ09uSW5pdCgpIHtcbiAgICAvLyBBdm9pZCBuZ0NoYW5nZXMgZXJyb3JcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGlmICh0aGlzLmNvbmZpZy5hdXRvZm9jdXMpIHtcbiAgICAgICAgdGhpcy5mb2N1cygpO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgdGhpcy5faW5pdEF1dG9SZWxvYWQoKTtcbiAgICB0aGlzLl9saXN0ZW5JbnB1dENoYW5nZXMoKTtcbiAgICB0aGlzLl9saXN0ZW5JbnRlcm5hbEl0ZW1zQ2hhbmdlKCk7XG4gICAgdGhpcy5faW5pdE92ZXJsYXkoKTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpIHtcbiAgICB0aGlzLl9kZXN0cm95RmlsdGVyRHJhd2VyKCk7XG5cbiAgICB0aGlzLl9kZXN0cm95JC5uZXh0KCk7XG4gICAgdGhpcy5fZGVzdHJveSQuY29tcGxldGUoKTtcbiAgfVxuXG4gIHB1YmxpYyBmb2N1cygpIHtcbiAgICB0aGlzLmtleXdvcmRNYXRJbnB1dD8uZm9jdXMoKTtcbiAgfVxuXG4gIHB1YmxpYyB1cGRhdGVTb3J0KHNvcnQ6IElTb3J0aW5nQ2hhbmdlRXZlbnQpIHtcbiAgICB0aGlzLl9maWx0ZXJJdGVtcy51cGRhdGVTb3J0KHNvcnQpO1xuICB9XG5cbiAgLyoqXG4gICAqXG4gICAqIERvIHVwZGF0ZSB2YWx1ZSBvZiBzb21lIGZpZWxkXG4gICAqXG4gICAqIEBwYXJhbSB2YWx1ZXMgLSB2YWx1ZXMgZm9yIHVwZGF0ZVxuICAgKlxuICAgKiBUbyB1cGRhdGUgdGV4dCB2YWx1ZSBqdXN0IHBhc3MgbmV3IHRleHQgdmFsdWVcbiAgICpcbiAgICogcHVibGljIHVwZGF0ZVNlbGVjdFZhbHVlKHZhbCkge1xuICAgKiAgIHRoaXMuZmlsdGVyRWwudXBkYXRlVmFsdWVzKHsga2V5d29yZDogdmFsIH0pO1xuICAgKiB9XG4gICAqXG4gICAqIFRvIHVwZGF0ZSBzZWxlY3Qgb3Igb2JzZXJ2YWJsZSBzZWxlY3QgeW91IGNvdWxkIHBhc3Mgc3VpdGFibGUgdmFsdWVcbiAgICpcbiAgICogcHVibGljIHVwZGF0ZVNlbGVjdFZhbHVlKHZhbDogbnVtYmVyKSB7XG4gICAqICAgdGhpcy5maWx0ZXJFbC51cGRhdGVWYWx1ZXMoeyBzaW1wbGVfc2VsZWN0OiB2YWwgfSwgeyBvYnNlcnZhYmxlX3NlbGVjdDogdmFsIH0pO1xuICAgKiB9XG4gICAqXG4gICAqIFRvIHVwZGF0ZSBjaGVja2JveCB2YWx1ZSBqdXN0IHBhc3MgdHJ1ZS9mYWxzZSBhcyB2YWx1ZVxuICAgKlxuICAgKiBwdWJsaWMgdXBkYXRlQ2hlY2tveCh2YWw6IGJvb2xlYW4pIHtcbiAgICogICB0aGlzLmZpbHRlckVsLnVwZGF0ZVZhbHVlcyh7IGNoZWNrYm94OiB2YWwgfSk7XG4gICAqIH1cbiAgICpcbiAgICogVG8gdXBkYXRlIHJhbmdlIHZhbHVlIGp1c3QgcGFzcyBvYmplY3Qgd2l0aCBtaW4mbWF4IG9iamVjdCBvciBqdXN0IHdpdGggb25lIG9mIHRhcmdldHNcbiAgICpcbiAgICogRXguOiB7IG1pbjogMTAsIG1heCAxNSB9LCB7IG1pbjogNSB9LCB7IG1heCA1IH1cbiAgICpcbiAgICogcHVibGljIHVwZGF0ZVJhbmdlKHZhbCkge1xuICAgKiAgIHRoaXMuZmlsdGVyRWwudXBkYXRlVmFsdWVzKHsgcmFuZ2U6IHZhbCB9KTtcbiAgICogfVxuICAgKlxuICAgKiBUbyB1cGRhdGUgYXV0b2NvbXBsZXRlIGp1c3QgcGFzcyBvYmplY3Qgd2l0aCBuYW1lL3ZhbHVlIGZpZWxkc1xuICAgKlxuICAgKiBFeC46IHsgbmFtZTogJ0pvaG4gRG9lJywgdmFsdWU6IDEgfVxuICAgKlxuICAgKiBwdWJsaWMgdXBkYXRlQXV0b2NvbXBsZXRlKHZhbCkge1xuICAgKiAgIHRoaXMuZmlsdGVyRWwudXBkYXRlVmFsdWVzKHsgYXV0b2NvbXBsZXRlX3VzZXJfaWQ6IHZhbCB9KTtcbiAgICogfVxuICAgKlxuICAgKiBUbyB1cGRhdGUgYXV0b2NvbXBsZXRlY2hpcHMganVzdCBwYXNzOlxuICAgKlxuICAgKiAxKSBvYmplY3Qgd2l0aCBuYW1lL3ZhbHVlIGZpZWxkcyAtIHdpbGwgYmUgYXBwZW5kZWQgdG8gZXhpc3Rpbmcgc2V0IG9mIHZhbHVlc1xuICAgKlxuICAgKiB7IG5hbWU6ICdKb2huIERvZScsIHZhbHVlOiAxIH1cbiAgICpcbiAgICogMikgYXJyYXkgb2Ygb2JqZWN0cyAtIHdpbGwgYmUgYXBwZW5kZWQgdG8gZXhpc3Rpbmcgc2V0IG9mIHZhbHVlc1xuICAgKlxuICAgKiBbeyBuYW1lOiAnSm9obiBEb2UnLCB2YWx1ZTogMSB9LCB7IG5hbWU6ICdEYXJ5YSBGaWxpcG92YScsIHZhbHVlOiAyIH1dXG4gICAqXG4gICAqIDMpIG51bGwgLSBjbGVhciBleGlzdGluZyBzZXQgb2YgdmFsdWVzXG4gICAqXG4gICAqIHB1YmxpYyB1cGRhdGVBdXRvY29tcGxldGUodmFsKSB7XG4gICAqICAgdGhpcy5maWx0ZXJFbC51cGRhdGVWYWx1ZXMoeyBhdXRvY29tcGxldGVjaGlwc191c2VyX2lkOiB2YWwgfSk7XG4gICAqIH1cbiAgICpcbiAgICovXG4gIHB1YmxpYyB1cGRhdGVWYWx1ZXModmFsdWVzKSB7XG4gICAgT2JqZWN0LmtleXModmFsdWVzKS5mb3JFYWNoKChrZXkpID0+IHtcbiAgICAgIGNvbnN0IGZpbHRlckl0ZW0gPSB0aGlzLml0ZW1zXG4gICAgICAgIC5maW5kKChpdGVtKSA9PiBpdGVtLm5hbWUgPT09IGtleSk7XG5cbiAgICAgIGlmICghZmlsdGVySXRlbSkge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIGZpbHRlckl0ZW0ubW9kZWwgPSB2YWx1ZXNba2V5XTtcbiAgICB9KTtcbiAgfVxuXG4gIHB1YmxpYyBoaWRlKCkge1xuICAgIHRoaXMuY2hhbmdlVmlzaWJpbGl0eShmYWxzZSk7XG4gIH1cblxuICBwdWJsaWMgc2hvdygpIHtcbiAgICB0aGlzLmNoYW5nZVZpc2liaWxpdHkodHJ1ZSk7XG4gIH1cblxuICBwdWJsaWMgY2hhbmdlVmlzaWJpbGl0eUNsaWNrKHZhbHVlLCBldmVudCA9IG51bGwpIHtcblxuICAgIGlmIChldmVudCkge1xuICAgICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgfVxuXG4gICAgdGhpcy5jaGFuZ2VWaXNpYmlsaXR5KHZhbHVlKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgaXRlbVZhbHVlcygpOiBhbnlbXSB7XG4gICAgcmV0dXJuIHRoaXMuaXRlbXNcbiAgICAgIC5tYXAoKGl0ZW0pID0+IGl0ZW0udmFsdWUpO1xuICB9XG5cbiAgcHVibGljIGdldCBub25FbXB0eUl0ZW1WYWx1ZXMoKTogYW55W10ge1xuICAgIHJldHVybiB0aGlzLml0ZW1zXG4gICAgICAuZmlsdGVyKChpdGVtKSA9PiBpdGVtLnZhbHVlICE9PSB1bmRlZmluZWQpXG4gICAgICAubWFwKChpdGVtKSA9PiBpdGVtLnZhbHVlKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgaGFzSXRlbVZhbHVlcygpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5pdGVtc1xuICAgICAgLnNvbWUoKGl0ZW0pID0+IGl0ZW0udmFsdWUgIT09IHVuZGVmaW5lZCk7XG4gIH1cblxuICBwdWJsaWMgZ2V0SXRlbVZhbHVlKG5hbWU6IHN0cmluZykge1xuICAgIGNvbnN0IGl0ZW0gPSB0aGlzLml0ZW1zXG4gICAgICAuZmluZCgoX2l0ZW0pID0+IF9pdGVtLm5hbWUgPT09IG5hbWUpO1xuXG4gICAgcmV0dXJuIGl0ZW0/LnZhbHVlO1xuICB9XG5cbiAgcHVibGljIHNob3dJdGVtKG5hbWU6IHN0cmluZykge1xuICAgIGNvbnN0IGl0ZW0gPSB0aGlzLmdldEl0ZW0obmFtZSk7XG5cbiAgICBpZiAoaXRlbSkge1xuICAgICAgaXRlbS5oaWRlID0gZmFsc2U7XG5cbiAgICAgIHRoaXMuX2ZpbHRlckl0ZW1zLnVwZGF0ZUl0ZW1zVmlzaWJsaXR5KCk7XG4gICAgfVxuICB9XG5cbiAgcHVibGljIGhpZGVJdGVtKG5hbWU6IHN0cmluZykge1xuICAgIGNvbnN0IGl0ZW0gPSB0aGlzLmdldEl0ZW0obmFtZSk7XG5cbiAgICBpZiAoIWl0ZW0pIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpdGVtLmhpZGUgPSB0cnVlO1xuXG4gICAgdGhpcy5fZmlsdGVySXRlbXMudXBkYXRlSXRlbXNWaXNpYmxpdHkoKTtcbiAgfVxuXG4gIHB1YmxpYyBjbGVhckl0ZW0obmFtZTogc3RyaW5nKSB7XG4gICAgY29uc3QgaXRlbSA9IHRoaXMuZ2V0SXRlbShuYW1lKTtcblxuICAgIGlmICghaXRlbSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGl0ZW0uY2xlYXIoKTtcbiAgfVxuXG4gIHB1YmxpYyB1cGRhdGVJdGVtQ29uZmlnKFxuICAgIG5hbWU6IHN0cmluZyxcbiAgICBwYXJhbXM6IElVcGRhdGVGaWx0ZXJJdGVtQ29uZmlnLFxuICApOiB2b2lkIHtcbiAgICBjb25zdCBpdGVtID0gdGhpcy5nZXRJdGVtKG5hbWUpO1xuXG4gICAgaWYgKCFpdGVtKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaXRlbS5sYWJlbCA9IHBhcmFtcy5sYWJlbCA/PyBpdGVtLmxhYmVsO1xuICAgIGl0ZW0uY2hpcExhYmVsID0gcGFyYW1zLmNoaXBMYWJlbCA/PyBpdGVtLmNoaXBMYWJlbDtcblxuICAgIHRoaXMuX2ZpbHRlckl0ZW1zLnVwZGF0ZUl0ZW1zVmlzaWJsaXR5KCk7XG4gIH1cblxuICBwdWJsaWMgZ2V0SXRlbVZhbHVlQ2hhbmdlJChuYW1lOiBzdHJpbmcpOiBPYnNlcnZhYmxlPGFueT4gfCBudWxsIHtcbiAgICBjb25zdCBpdGVtID0gdGhpcy5pdGVtcy5maW5kKChpKSA9PiBpLm5hbWUgPT09IG5hbWUpO1xuXG4gICAgaWYgKGl0ZW0pIHtcbiAgICAgIHJldHVybiBpdGVtLnZhbHVlJFxuICAgICAgICAucGlwZShcbiAgICAgICAgICBtYXAoKCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuIGl0ZW0ubW9kZWw7XG4gICAgICAgICAgfSksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgcmV0dXJuIG51bGw7XG5cbiAgfVxuXG4gIHB1YmxpYyBjaGFuZ2VWaXNpYmlsaXR5KHN0YXRlOiBib29sZWFuKSB7XG4gICAgaWYgKHN0YXRlID09PSB0aGlzLnNob3dGaWx0ZXJNZW51KSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCFzdGF0ZSkge1xuICAgICAgdGhpcy5jbG9zZWQuZW1pdCgpO1xuXG4gICAgICByZXR1cm4gdGhpcy5fZGVzdHJveUZpbHRlckRyYXdlcigpO1xuICAgIH1cblxuICAgIGlmICghdGhpcy5oYXNWaXNpYmxlSXRlbU9yU29ydGluZykge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHRoaXMuX2xpc3RlbkVzY0J1dHRvbigpO1xuXG4gICAgdGhpcy5vcGVuZWQuZW1pdCgpO1xuXG4gICAgdGhpcy5fZmlsdGVyT3ZlcmxheS5vcGVuKCk7XG4gIH1cblxuICBwdWJsaWMgaW5pdCgpIHtcbiAgICBjb25zdCBkYXRhID0gdGhpcy5fZmlsdGVySXRlbXMudmFsdWVzQXNRdWVyeSgpO1xuICAgIHRoaXMuX3NvcnQgPSB0aGlzLl9maWx0ZXJJdGVtcy5nZXRTb3J0KCk7XG5cbiAgICBpZiAodGhpcy5jb25maWcuaW5pdCkge1xuICAgICAgdGhpcy5jb25maWcuaW5pdChkYXRhLCB0aGlzLl9zb3J0LCB0aGlzKTtcbiAgICB9XG5cbiAgICB0aGlzLl91cGRhdGVDaGlwc1Zpc2liaWxpdHkoKTtcblxuICAgIHRoaXMuaXRlbXNcbiAgICAgIC5mb3JFYWNoKChpdGVtKSA9PiB7XG4gICAgICAgIGl0ZW0uaW5pdChpdGVtLCB0aGlzKTtcbiAgICAgIH0pO1xuICB9XG5cbiAgcHVibGljIGNsZWFyKGV2ZW50ID0gbnVsbCkge1xuICAgIGlmIChldmVudCkge1xuICAgICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgfVxuXG4gICAgdGhpcy5fZmlsdGVySXRlbXMuZmlsdGVyc0NsZWFyKCk7XG5cbiAgICBpZiAodGhpcy5jb25maWcuY2xlYXIpIHtcbiAgICAgIHRoaXMuY29uZmlnLmNsZWFyKCk7XG4gICAgfVxuXG4gICAgdGhpcy5rZXl3b3JkID0gJyc7XG4gIH1cblxuICAvKipcbiAgICogQ2xvc2UgZmlsdGVyIHdpbmRvdyBhbmQgZG8gY2hhbmdlIGNhbGxiYWNrXG4gICAqL1xuICBwdWJsaWMgc2VhcmNoKGV2ZW50KSB7XG4gICAgdGhpcy5jaGFuZ2VWaXNpYmlsaXR5Q2xpY2soZmFsc2UsIGV2ZW50KTtcbiAgfVxuXG4gIHB1YmxpYyByZWxvYWQoZXZlbnQgPSBudWxsKSB7XG4gICAgaWYgKGV2ZW50KSB7XG4gICAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpO1xuICAgICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgfVxuXG4gICAgY29uc3QgZGF0YSA9IHRoaXMuX2ZpbHRlckl0ZW1zLnZhbHVlc0FzUXVlcnkoKTtcbiAgICBjb25zdCBlbCA9IHRoaXMucmVsb2FkRWw/Lm5hdGl2ZUVsZW1lbnQ7XG5cbiAgICBpZihlbCkge1xuICAgICAgZWwuc3R5bGUudHJhbnNpdGlvbiA9ICdhbGwgMC43NXMgMC4wcyc7XG4gICAgICBlbC5zdHlsZS50cmFuc2Zvcm0gPSAncm90YXRlKDM2MGRlZyknO1xuXG4gICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgZWwuc3R5bGUudHJhbnNpdGlvbiA9IG51bGw7XG4gICAgICAgIGVsLnN0eWxlLnRyYW5zZm9ybSA9IG51bGw7XG4gICAgICB9LCAxMDAwKTtcbiAgICB9XG5cbiAgICBpZiAodGhpcy5jb25maWcucmVsb2FkKSB7XG4gICAgICB0aGlzLmNvbmZpZy5yZWxvYWQoZGF0YSwgdGhpcy5fZmlsdGVySXRlbXMuZ2V0U29ydCgpKTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgZ2V0SXRlbShuYW1lKTogQmFzZUl0ZW08YW55PiB7XG4gICAgcmV0dXJuIHRoaXMuaXRlbXNcbiAgICAgIC5maW5kKChpdGVtKSA9PiBpdGVtLm5hbWUgPT09IG5hbWUpO1xuICB9XG5cbiAgcHVibGljIGZldGNoUXVlcnlQYXJhbXMoKTogdm9pZCB7XG4gICAgdGhpcy5fZXh0ZXJuYWxQYXJhbXMuZmV0Y2hRdWVyeVBhcmFtcygpO1xuICB9XG5cbiAgLyoqXG4gICAqIENhbGwgY2hhbmdlIGNhbGxiYWNrIGFuZCBhcHBseSBuZXcgZmlsdGVyIHZhbHVlc1xuICAgKi9cbiAgcHVibGljIGNoYW5nZSgpIHtcbiAgICBjb25zdCBkYXRhID0gdGhpcy5fZmlsdGVySXRlbXMudmFsdWVzQXNRdWVyeSgpO1xuICAgIGNvbnN0IHNvcnQgPSB0aGlzLl9maWx0ZXJJdGVtcy5nZXRTb3J0KCk7XG5cbiAgICBjb25zdCBzb3J0aW5nQ2hhbmdlZCA9ICgoIXNvcnQgfHwgIXRoaXMuX3NvcnQpICYmIHNvcnQgIT09IHRoaXMuX3NvcnQpXG4gICAgICB8fCAoc29ydCAmJiB0aGlzLl9zb3J0ICYmICFvYmplY3RzQXJlRXF1YWxzKHRoaXMuX3NvcnQsIHNvcnQpKTtcblxuICAgIGlmIChzb3J0aW5nQ2hhbmdlZCkge1xuICAgICAgdGhpcy5fc29ydCA9IHNvcnQ7XG5cbiAgICAgIGlmICh0aGlzLmNvbmZpZy5zb3J0Q2hhbmdlKSB7XG4gICAgICAgIHRoaXMuY29uZmlnLnNvcnRDaGFuZ2UoZGF0YSwgc29ydCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYgKHRoaXMuY29uZmlnLmNoYW5nZSkge1xuICAgICAgdGhpcy5jb25maWcuY2hhbmdlKGRhdGEsIHNvcnQpO1xuICAgIH1cblxuICAgIHRoaXMuX3VwZGF0ZUNoaXBzVmlzaWJpbGl0eSgpO1xuXG4gICAgLy8gdmlzaWJpbGl0eSBmb3IgYWN0aW9ucyBjYW4gZGVwZW5kIG9uIGZpbHRlcnMgc3RhdGVcbiAgICB0aGlzLl9hY3Rpb25zLnVwZGF0ZUFjdGlvbnNWaXNpYmlsaXR5KCk7XG4gIH1cblxuICAvKipcbiAgICogVXBkYXRlIGZpbHRlciBhY3Rpb25zIGNvbmZpZ1xuICAgKlxuICAgKiBAcGFyYW0gYWN0aW9uc1xuICAgKi9cbiAgcHVibGljIHVwZGF0ZUFjdGlvbnMoYWN0aW9uczogRnNGaWx0ZXJBY3Rpb25bXSk6IHZvaWQge1xuICAgIHRoaXMuX2FjdGlvbnMuaW5pdEFjdGlvbnMoYWN0aW9ucyk7XG4gIH1cblxuICAvKipcbiAgICogU2hvdyBcIkZpbHRlcnNcIiBidXR0b25cbiAgICovXG4gIHB1YmxpYyBzaG93RmlsdGVyc0J0bigpOiB2b2lkIHtcbiAgICB0aGlzLl9maWx0ZXJzQnRuVmlzaWJsZSQubmV4dCh0cnVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBIaWRlIFwiRmlsdGVyc1wiIGJ1dHRvblxuICAgKi9cbiAgcHVibGljIGhpZGVGaWx0ZXJzQnRuKCk6IHZvaWQge1xuICAgIHRoaXMuX2ZpbHRlcnNCdG5WaXNpYmxlJC5uZXh0KGZhbHNlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTaG93IFwiS2V5d29yZFwiIGZpZWxkIGlmIGl0IHByZXNlbnRcbiAgICovXG4gIHB1YmxpYyBzaG93S2V5d29yZEZpZWxkKCk6IHZvaWQge1xuICAgIHRoaXMuX2tleXdvcmRWaXNpYmxlJC5uZXh0KHRydWUpO1xuICB9XG5cbiAgLyoqXG4gICAqIEhpZGUgXCJLZXl3b3JkXCIgZmllbGQgaWYgaXQgcHJlc2VudFxuICAgKi9cbiAgcHVibGljIGhpZGVLZXl3b3JkRmllbGQoKTogdm9pZCB7XG4gICAgdGhpcy5fa2V5d29yZFZpc2libGUkLm5leHQoZmFsc2UpO1xuICB9XG5cbiAgLyoqXG4gICAqIEdvIHRocm91Z2ggYWN0aW9ucyBhbmQgY2hlY2sgc2hvdygpIGNhbGxiYWNrIGFuZCB1cGRhdGUgdmlzaWJsZSBhY3Rpb25zXG4gICAqL1xuICBwdWJsaWMgdXBkYXRlQWN0aW9uc1Zpc2liaWxpdHkoKTogdm9pZCB7XG4gICAgdGhpcy5fYWN0aW9ucy51cGRhdGVBY3Rpb25zVmlzaWJpbGl0eSgpO1xuICB9XG5cbiAgLyoqXG4gICAqIEdvIHRocm91Z2ggYWN0aW9ucyBhbmQgY2hlY2sgZGlzYWJsZWQoKSBjYWxsYmFjayBhbmQgdXBkYXRlIGRpc2FibGVkIHN0YXRlXG4gICAqL1xuICBwdWJsaWMgdXBkYXRlRGlzYWJsZWRTdGF0ZSgpOiB2b2lkIHtcbiAgICB0aGlzLl9hY3Rpb25zLnVwZGF0ZURpc2FibGVkU3RhdGUoKTtcbiAgfVxuXG4gIHB1YmxpYyBzZXRJdGVtcyhpdGVtczogSUZpbHRlckNvbmZpZ0l0ZW1bXSkge1xuICAgIHRoaXMuX2ZpbHRlckl0ZW1zLmRlc3Ryb3lJdGVtcygpO1xuXG4gICAgdGhpcy5jb25maWcuaXRlbXMgPSBpdGVtcztcbiAgICB0aGlzLl9maWx0ZXJJdGVtcy5zZXRDb25maWcodGhpcy5fY29uZmlnKTtcbiAgICB0aGlzLl9leHRlcm5hbFBhcmFtcy5pbml0SXRlbXMoKTtcblxuICAgIHRoaXMuX3N5bmNTZWFyY2hJbnB1dFdpdGhLZXl3b3JkKCk7XG4gIH1cblxuICBwdWJsaWMga2V5d29yZENoYW5nZShrZXl3b3JkKSB7XG4gICAgdGhpcy5fa2V5d29yZCQubmV4dChrZXl3b3JkKTtcbiAgfVxuXG4gIHB1YmxpYyB1cGRhdGVTb3J0aW5ncyhpdGVtczogU29ydEl0ZW1bXSk6IHZvaWQge1xuICAgIHRoaXMuX2ZpbHRlckl0ZW1zLnVwZGF0ZVNvcnRpbmdJdGVtc1ZhbHVlcyhpdGVtcyk7XG4gIH1cblxuICBwcml2YXRlIF9pbml0RmlsdGVyV2l0aENvbmZpZyhjb25maWc6IEZpbHRlckNvbmZpZykge1xuICAgIGlmICh0aGlzLmNvbmZpZykge1xuICAgICAgdGhpcy5fZmlsdGVySXRlbXMuZGVzdHJveUl0ZW1zKCk7XG4gICAgfVxuXG4gICAgY29uZmlnID0ge1xuICAgICAgLi4uKHRoaXMuX2RlZmF1bHRDb25maWcgfHwge30pLFxuICAgICAgLi4uY29uZmlnLFxuICAgIH07XG5cbiAgICB0aGlzLl9jb25maWcgPSBuZXcgRnNGaWx0ZXJDb25maWcoY29uZmlnKTtcbiAgICB0aGlzLl9hY3Rpb25zLnNldENvbmZpZyh0aGlzLl9jb25maWcpO1xuICAgIHRoaXMuX2ZpbHRlckl0ZW1zLnNldENvbmZpZyh0aGlzLl9jb25maWcpO1xuICAgIHRoaXMuX2V4dGVybmFsUGFyYW1zLnNldENvbmZpZyh0aGlzLl9jb25maWcpO1xuXG4gICAgdGhpcy5fc3luY1NlYXJjaElucHV0V2l0aEtleXdvcmQoKTtcblxuICAgIGlmICh0aGlzLmNvbmZpZy5yZWxvYWRXaGVuQ29uZmlnQ2hhbmdlZCkge1xuICAgICAgdGhpcy5jaGFuZ2UoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9kZXN0cm95RmlsdGVyRHJhd2VyKCkge1xuICAgIHRoaXMuX2ZpbHRlck92ZXJsYXkuY2xvc2UoKTtcbiAgfVxuXG4gIHByaXZhdGUgX3VwZGF0ZVdpbmRvd1dpZHRoKCkge1xuICAgIHRoaXMud2luZG93RGVza3RvcCA9IHdpbmRvdy5pbm5lcldpZHRoID4gMTIwMDtcbiAgfVxuXG4gIHByaXZhdGUgX2xpc3RlbkVzY0J1dHRvbigpIHtcbiAgICB0aGlzLl96b25lLnJ1bk91dHNpZGVBbmd1bGFyKCgpID0+IHtcbiAgICAgIGZyb21FdmVudCh3aW5kb3csICdrZXl1cCcpXG4gICAgICAgIC5waXBlKFxuICAgICAgICAgIGZpbHRlcigoZXZlbnQ6IEtleWJvYXJkRXZlbnQpID0+IGV2ZW50LmNvZGUgPT09ICdFc2NhcGUnKSxcbiAgICAgICAgICB0YWtlVW50aWwodGhpcy5jbG9zZWQpLFxuICAgICAgICAgIHRha2VVbnRpbCh0aGlzLl9kZXN0cm95JCksXG4gICAgICAgIClcbiAgICAgICAgLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgICAgdGhpcy5fem9uZS5ydW4oKCkgPT4ge1xuICAgICAgICAgICAgdGhpcy5jaGFuZ2VWaXNpYmlsaXR5KGZhbHNlKTtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIF9saXN0ZW5XaW5kb3dSZXNpemUoKSB7XG4gICAgdGhpcy5fem9uZS5ydW5PdXRzaWRlQW5ndWxhcigoKSA9PiB7XG4gICAgICBmcm9tRXZlbnQod2luZG93LCAncmVzaXplJylcbiAgICAgICAgLnBpcGUoXG4gICAgICAgICAgZGVib3VuY2VUaW1lKDEwMCksXG4gICAgICAgICAgdGFrZVVudGlsKHRoaXMuX2Rlc3Ryb3kkKSxcbiAgICAgICAgKVxuICAgICAgICAuc3Vic2NyaWJlKCgpID0+IHtcbiAgICAgICAgICB0aGlzLl96b25lLnJ1bigoKSA9PiB7XG4gICAgICAgICAgICB0aGlzLl91cGRhdGVXaW5kb3dXaWR0aCgpO1xuICAgICAgICAgIH0pO1xuICAgICAgICB9KTtcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgX2luaXRBdXRvUmVsb2FkKCkge1xuICAgIGlmKHRoaXMuY29uZmlnLmF1dG9SZWxvYWQpIHtcbiAgICAgIGludGVydmFsKHRoaXMuY29uZmlnLmF1dG9SZWxvYWQuc2Vjb25kcyAqIDEwMDApXG4gICAgICAgIC5waXBlKFxuICAgICAgICAgIGZpbHRlcigoKSA9PiB0aGlzLmF1dG9SZWxvYWQpLFxuICAgICAgICAgIHRha2VVbnRpbCh0aGlzLl9kZXN0cm95JCksXG4gICAgICAgIClcbiAgICAgICAgLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgICAgdGhpcy5yZWxvYWQobnVsbCk7XG4gICAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgX2xpc3RlbklucHV0Q2hhbmdlcygpIHtcbiAgICB0aGlzLl9rZXl3b3JkJFxuICAgICAgLnBpcGUoXG4gICAgICAgIGRlYm91bmNlVGltZSgyMDApLFxuICAgICAgICBkaXN0aW5jdFVudGlsQ2hhbmdlZCgpLFxuICAgICAgICB0YWtlVW50aWwodGhpcy5fZGVzdHJveSQpLFxuICAgICAgKVxuICAgICAgLnN1YnNjcmliZSgodmFsdWUpID0+IHtcbiAgICAgICAgY29uc3Qga2V5d29yZEl0ZW0gPSB0aGlzLl9maWx0ZXJJdGVtcy5rZXl3b3JkSXRlbTtcbiAgICAgICAga2V5d29yZEl0ZW0ubW9kZWwgPSB2YWx1ZTtcbiAgICAgICAgdGhpcy5jaGFuZ2UoKTtcbiAgICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSBfc3luY1NlYXJjaElucHV0V2l0aEtleXdvcmQoKTogdm9pZCB7XG4gICAgY29uc3Qga2V5d29yZEl0ZW0gPSB0aGlzLl9maWx0ZXJJdGVtcy5rZXl3b3JkSXRlbTtcbiAgICBpZiAoa2V5d29yZEl0ZW0pIHtcbiAgICAgIHRoaXMua2V5d29yZCA9IGtleXdvcmRJdGVtLm1vZGVsO1xuICAgICAgdGhpcy5zZWFyY2hQbGFjZWhvbGRlciA9IGtleXdvcmRJdGVtLmxhYmVsIGFzIHN0cmluZyB8fCAnU2VhcmNoJztcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9saXN0ZW5JbnRlcm5hbEl0ZW1zQ2hhbmdlKCkge1xuICAgIHRoaXMuX2ZpbHRlckl0ZW1zXG4gICAgICAuaXRlbXNDaGFuZ2UkXG4gICAgICAucGlwZShcbiAgICAgICAgdGFrZVVudGlsKHRoaXMuX2Rlc3Ryb3kkKSxcbiAgICAgIClcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgICB0aGlzLmNoYW5nZSgpO1xuICAgICAgfSk7XG4gIH1cblxuICBwcml2YXRlIF9pbml0T3ZlcmxheSgpIHtcbiAgICB0aGlzLl9maWx0ZXJPdmVybGF5LnNldENsZWFyRm4odGhpcy5jbGVhci5iaW5kKHRoaXMpKTtcbiAgICB0aGlzLl9maWx0ZXJPdmVybGF5LnNldERvbmVGbih0aGlzLmhpZGUuYmluZCh0aGlzKSk7XG4gIH1cblxuICAvLyBXZSBtYXkgbmVlZCBzb21lIHRpbWUgdG8gcmVjaWV2ZSBleHRlcm5hbCBwYXJhbXMgYW5kIGFmdGVyIHRoYXQgcmVhZHkgY2FuIGJlIGVtaXR0ZWRcbiAgcHJpdmF0ZSBfbGlzdGVuV2hlbkZpbHRlclJlYWR5KCkge1xuICAgIGNvbWJpbmVMYXRlc3QoXG4gICAgICBbXG4gICAgICAgIHRoaXMuX2V4dGVybmFsUGFyYW1zLnBlbmRpbmckLFxuICAgICAgICB0aGlzLml0ZW1zUmVhZHkkLFxuICAgICAgXSlcbiAgICAgIC5waXBlKFxuICAgICAgICBmaWx0ZXIoKFtwZW5kaW5nUGFyYW1zLCBpdGVtc1JlYWR5XSkgPT4gIXBlbmRpbmdQYXJhbXMgJiYgaXRlbXNSZWFkeSksXG4gICAgICAgIHRha2VVbnRpbCh0aGlzLl9kZXN0cm95JCksXG4gICAgICApXG4gICAgICAuc3Vic2NyaWJlKCgpID0+IHtcbiAgICAgICAgdGhpcy5pbml0KCk7XG4gICAgICAgIHRoaXMuX3N5bmNTZWFyY2hJbnB1dFdpdGhLZXl3b3JkKCk7XG5cbiAgICAgICAgdGhpcy5yZWFkeS5lbWl0KCk7XG4gICAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgX3VwZGF0ZUNoaXBzVmlzaWJpbGl0eSgpIHtcbiAgICBjb25zdCBoYXNGaWx0ZXJDaGlwcyA9IHRoaXMuX2ZpbHRlckl0ZW1zLml0ZW1zXG4gICAgICAuc29tZSgoaXRlbTogQmFzZUl0ZW08YW55PikgPT4ge1xuICAgICAgICByZXR1cm4gaXRlbS5pc0NoaXBWaXNpYmxlO1xuICAgICAgfSk7XG5cbiAgICB0aGlzLl9oYXNGaWx0ZXJDaGlwcyQubmV4dChoYXNGaWx0ZXJDaGlwcyk7XG4gIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJmaWx0ZXItY29udGFpbmVyXCI+XG4gIDxkaXYgY2xhc3M9XCJmaWx0ZXItaW5uZXItY29udGFpbmVyXCI+XG4gICAgPG5nLWNvbnRhaW5lciAqbmdJZj1cIiFoYXNLZXl3b3JkOyBlbHNlIGZpbHRlcktleXdvcmRcIj5cbiAgICAgIDxkaXY+XG4gICAgICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJmaWx0ZXJTdGF0dXNCYXJDaGlwc1wiPjwvbmctY29udGFpbmVyPlxuICAgICAgPC9kaXY+XG4gICAgPC9uZy1jb250YWluZXI+XG5cbiAgICA8bmctY29udGFpbmVyIFtuZ1RlbXBsYXRlT3V0bGV0XT1cImZpbHRlclRvb2xiYXJcIj48L25nLWNvbnRhaW5lcj5cbiAgICA8bmctY29udGFpbmVyIFtuZ1RlbXBsYXRlT3V0bGV0XT1cImZpbHRlckFjdGlvbnNcIj48L25nLWNvbnRhaW5lcj5cbiAgPC9kaXY+XG5cbiAgPG5nLWNvbnRhaW5lciAqbmdJZj1cImhhc0tleXdvcmRcIj5cbiAgICA8bmctY29udGFpbmVyICpuZ1RlbXBsYXRlT3V0bGV0PVwiZmlsdGVyU3RhdHVzQmFyQ2hpcHNcIj48L25nLWNvbnRhaW5lcj5cbiAgPC9uZy1jb250YWluZXI+XG48L2Rpdj5cblxuPG5nLXRlbXBsYXRlICNmaWx0ZXJTdGF0dXNCYXJDaGlwcz5cbiAgPG5nLWNvbnRhaW5lciAqbmdJZj1cInN0YXR1c0JhclwiPlxuICAgIDxkaXYgY2xhc3M9XCJmaWx0ZXItc3RhdHVzLWNvbnRhaW5lclwiIFtuZ0NsYXNzXT1cInsgJ2hhcy1zdGF0dXMnOiAhIWZpbHRlclN0YXR1cy50ZXh0Q29udGVudCB9XCI+XG4gICAgICA8ZGl2IGNsYXNzPVwiZmlsdGVyLXN0YXR1c1wiICNmaWx0ZXJTdGF0dXM+XG4gICAgICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJzdGF0dXNCYXIudGVtcGxhdGVSZWZcIj48L25nLWNvbnRhaW5lcj5cbiAgICAgIDwvZGl2PlxuICAgIDwvZGl2PlxuICA8L25nLWNvbnRhaW5lcj5cblxuICA8bmctY29udGFpbmVyICpuZ0lmPVwiY29uZmlnLmNoaXBzICYmIGhhc0ZpbHRlckNoaXBzJCB8IGFzeW5jXCI+XG4gICAgPGZzLWZpbHRlci1jaGlwc1xuICAgICAgY2xhc3M9XCJmaWx0ZXItY2hpcHNcIlxuICAgICAgW2ZpbHRlcnNdPVwiaXRlbXNcIj5cbiAgICA8L2ZzLWZpbHRlci1jaGlwcz5cbiAgPC9uZy1jb250YWluZXI+XG48L25nLXRlbXBsYXRlPlxuXG48bmctdGVtcGxhdGUgI2ZpbHRlcktleXdvcmQ+XG4gIDxkaXYgY2xhc3M9XCJmaWx0ZXIta2V5d29yZFwiPlxuICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCIoa2V5d29yZFZpc2libGUkIHwgYXN5bmMpICYmICFrZXl3b3JkSXRlbT8uaGlkZSBcIj5cbiAgICAgIDxtYXQtZm9ybS1maWVsZFxuICAgICAgICAgIFtmbG9hdExhYmVsXT1cIiduZXZlcidcIlxuICAgICAgICAgIGNsYXNzPVwiZm9ybS1maWVsZC1wYWRsZXNzXCJcbiAgICAgICAgICBbbmdDbGFzc109XCJzZWFyY2hcIlxuICAgICAgICAgIGFwcGVhcmFuY2U9XCJvdXRsaW5lXCI+XG4gICAgICAgIDxzcGFuIG1hdFByZWZpeD5cbiAgICAgICAgICA8bWF0LWljb24gbWF0UHJlZml4PnNlYXJjaDwvbWF0LWljb24+XG4gICAgICAgIDwvc3Bhbj5cbiAgICAgICAgPGlucHV0XG4gICAgICAgICAgI2tleXdvcmRNYXRJbnB1dFxuICAgICAgICAgIG1hdElucHV0XG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJrZXl3b3JkXCJcbiAgICAgICAgICAobmdNb2RlbENoYW5nZSk9XCJrZXl3b3JkQ2hhbmdlKCRldmVudClcIlxuICAgICAgICAgIG5hbWU9XCJmaWx0ZXItaW5wdXRcIlxuICAgICAgICAgIFtmc0NsZWFyXT1cInRydWVcIlxuICAgICAgICAgIFtwbGFjZWhvbGRlcl09XCJzZWFyY2hQbGFjZWhvbGRlclwiPlxuICAgICAgPC9tYXQtZm9ybS1maWVsZD5cbiAgICA8L25nLWNvbnRhaW5lcj5cbiAgPC9kaXY+XG48L25nLXRlbXBsYXRlPlxuXG48bmctdGVtcGxhdGUgI2ZpbHRlckFjdGlvbnM+XG4gIDxkaXYgY2xhc3M9XCJmaWx0ZXItYWN0aW9uc1wiPlxuICAgIDxmcy1maWx0ZXItYWN0aW9uc1xuICAgICAgKm5nSWY9XCJhY3Rpb25zVmlzaWJsZSQgfCBhc3luY1wiXG4gICAgICBbYWN0aW9uc109XCJhY3Rpb25zJCB8IGFzeW5jXCJcbiAgICAgIFtrZWJhYkFjdGlvbnNdPVwibWVudUFjdGlvbnMkIHwgYXN5bmNcIj5cbiAgICA8L2ZzLWZpbHRlci1hY3Rpb25zPlxuICA8L2Rpdj5cbjwvbmctdGVtcGxhdGU+XG5cbjxuZy10ZW1wbGF0ZSAjZmlsdGVyVG9vbGJhcj5cbiAgPGRpdiBjbGFzcz1cImZpbHRlci10b29iYXJcIj5cbiAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiZmlsdGVyc0J0blZpc2libGUkIHwgYXN5bmNcIj5cbiAgICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCJoYXNWaXNpYmxlSXRlbU9yU29ydGluZ1wiPlxuICAgICAgICA8YVxuICAgICAgICAgICAgbWF0LWJ1dHRvblxuICAgICAgICAgICAgY2xhc3M9XCJmaWx0ZXJzLWJ1dHRvblwiXG4gICAgICAgICAgICBbbmdDbGFzc109XCJ7XG4gICAgICAgICAgICAgICdtYXQtcmFpc2VkLWJ1dHRvbic6IGNvbmZpZy5idXR0b24uc3R5bGUgPT09ICdyYWlzZWQnICYmIGNvbmZpZy5idXR0b24ubGFiZWwsXG4gICAgICAgICAgICAgICdtYXQtZmxhdC1idXR0b24nOiBjb25maWcuYnV0dG9uLnN0eWxlID09PSAnZmxhdCcgJiYgY29uZmlnLmJ1dHRvbi5sYWJlbCxcbiAgICAgICAgICAgICAgJ21hdC1zdHJva2VkLWJ1dHRvbic6IGNvbmZpZy5idXR0b24uc3R5bGUgPT09ICdzdHJva2VkJyAmJiBjb25maWcuYnV0dG9uLmxhYmVsLFxuICAgICAgICAgICAgICAnbWF0LWJ1dHRvbic6IGNvbmZpZy5idXR0b24uc3R5bGUgPT09ICdiYXNpYycgJiYgY29uZmlnLmJ1dHRvbi5sYWJlbCxcbiAgICAgICAgICAgICAgJ21hdC1pY29uLWJ1dHRvbic6IGNvbmZpZy5idXR0b24uc3R5bGUgPT09ICdpY29uJyB8fCAhY29uZmlnLmJ1dHRvbi5sYWJlbFxuICAgICAgICAgICAgfVwiXG4gICAgICAgICAgICAoY2xpY2spPVwiY2hhbmdlVmlzaWJpbGl0eUNsaWNrKCFzaG93RmlsdGVyTWVudSwgJGV2ZW50KVwiXG4gICAgICAgICAgICBbY29sb3JdPVwiY29uZmlnLmJ1dHRvbi5jb2xvclwiPlxuICAgICAgICAgIDxtYXQtaWNvbiAqbmdJZj1cImNvbmZpZy5idXR0b24uaWNvblwiPnt7Y29uZmlnLmJ1dHRvbi5pY29ufX08L21hdC1pY29uPlxuICAgICAgICAgIHt7IGNvbmZpZy5idXR0b24ubGFiZWwgfX1cbiAgICAgICAgPC9hPlxuICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgPC9uZy1jb250YWluZXI+ICAgIFxuICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCJjb25maWcucmVsb2FkXCI+XG4gICAgICA8YVxuICAgICAgICAgIG1hdC1pY29uLWJ1dHRvblxuICAgICAgICAgIChjbGljayk9XCJyZWxvYWQoJGV2ZW50KVwiXG4gICAgICAgICAgY2xhc3M9XCJyZWxvYWRcIj5cbiAgICAgICAgPG1hdC1pY29uICNyZWxvYWRFbD5yZWZyZXNoPC9tYXQtaWNvbj5cbiAgICAgIDwvYT5cbiAgICA8L25nLWNvbnRhaW5lcj5cbiAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiY29uZmlnLmF1dG9SZWxvYWRcIj5cbiAgICAgIDxtYXQtc2xpZGUtdG9nZ2xlIFxuICAgICAgICAgIG5hbWU9XCJhdXRvUmVsb2FkXCIgXG4gICAgICAgICAgY2xhc3M9XCJhdXRvLXJlbG9hZFwiXG4gICAgICAgICAgWyhuZ01vZGVsKV09XCJhdXRvUmVsb2FkXCI+XG4gICAgICAgIEF1dG8gcmVmcmVzaFxuICAgICAgPC9tYXQtc2xpZGUtdG9nZ2xlPlxuICAgIDwvbmctY29udGFpbmVyPlxuICA8L2Rpdj5cbjwvbmctdGVtcGxhdGU+XG4iXX0=