@elderbyte/ngx-starter 13.5.2 → 13.7.1

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 (245) hide show
  1. package/esm2020/lib/common/enums/elder-enum-translation.service.mjs +3 -3
  2. package/esm2020/lib/common/forms/form-field-base.component.mjs +3 -3
  3. package/esm2020/lib/common/forms/multi-model-base.component.mjs +3 -3
  4. package/esm2020/lib/common/forms/template-composite-control.mjs +3 -3
  5. package/esm2020/lib/common/forms/value-accessor-base.mjs +3 -3
  6. package/esm2020/lib/common/http/http-client-builder.service.mjs +3 -3
  7. package/esm2020/lib/common/http/http-client-pristine.mjs +3 -3
  8. package/esm2020/lib/common/http/transfer/data-transfer-factory.mjs +3 -3
  9. package/esm2020/lib/common/url/elder-url-fragment-params.service.mjs +3 -3
  10. package/esm2020/lib/common/utils/collection-util.mjs +4 -1
  11. package/esm2020/lib/components/access-denied/elder-access-denied.component.mjs +3 -3
  12. package/esm2020/lib/components/access-denied/elder-access-denied.module.mjs +4 -4
  13. package/esm2020/lib/components/auditing/audited-entity/elder-audited-entity.component.mjs +3 -3
  14. package/esm2020/lib/components/auditing/elder-audit.module.mjs +4 -4
  15. package/esm2020/lib/components/button-group/elder-button-group/elder-button-group.component.mjs +3 -3
  16. package/esm2020/lib/components/button-group/elder-button-group.module.mjs +4 -4
  17. package/esm2020/lib/components/card-organizer/card-organizer/elder-card-organizer.component.mjs +3 -3
  18. package/esm2020/lib/components/card-organizer/card-organizer/elder-stack-card.directive.mjs +3 -3
  19. package/esm2020/lib/components/card-organizer/card-stack/elder-card-stack.component.mjs +3 -3
  20. package/esm2020/lib/components/card-organizer/elder-card-organizer.module.mjs +4 -4
  21. package/esm2020/lib/components/cards/elder-card/elder-card-header/elder-card-header.component.mjs +3 -3
  22. package/esm2020/lib/components/cards/elder-card/elder-card.component.mjs +18 -18
  23. package/esm2020/lib/components/cards/elder-card/elder-card.module.mjs +4 -4
  24. package/esm2020/lib/components/connectivity/elder-connectivity.module.mjs +4 -4
  25. package/esm2020/lib/components/connectivity/elder-connectivity.service.mjs +3 -3
  26. package/esm2020/lib/components/connectivity/offline-indicator/elder-offline-indicator.component.mjs +3 -3
  27. package/esm2020/lib/components/containers/elder-containers.module.mjs +4 -4
  28. package/esm2020/lib/components/containers/elder-scroll-container/elder-scroll-container.component.mjs +3 -3
  29. package/esm2020/lib/components/csv/elder-csv-export-btn/elder-csv-export-btn.component.mjs +3 -3
  30. package/esm2020/lib/components/csv/elder-csv-stream-exporter-builder.service.mjs +3 -3
  31. package/esm2020/lib/components/csv/elder-csv.module.mjs +4 -4
  32. package/esm2020/lib/components/currency/elder-currency.module.mjs +4 -4
  33. package/esm2020/lib/components/currency/elder-currency.pipe.mjs +3 -3
  34. package/esm2020/lib/components/data-transfer/elder-data-transfer.module.mjs +4 -4
  35. package/esm2020/lib/components/data-transfer/elder-data-transfer.service.mjs +3 -3
  36. package/esm2020/lib/components/data-transfer/http-data-transfer/http-data-transfer.component.mjs +3 -3
  37. package/esm2020/lib/components/data-transfer/http-data-transfer-aggregate/http-data-transfer-aggregate.component.mjs +3 -3
  38. package/esm2020/lib/components/data-transfer/http-data-transfer-indicator/http-data-transfer-indicator.component.mjs +3 -3
  39. package/esm2020/lib/components/data-transfer/http-data-transfer-overview/http-data-transfer-overview.component.mjs +3 -3
  40. package/esm2020/lib/components/data-view/base/elder-data-view-base.mjs +3 -3
  41. package/esm2020/lib/components/data-view/base/elder-data-view-base.module.mjs +4 -4
  42. package/esm2020/lib/components/data-view/common/data-context-state-indicator/data-context-state-indicator.component.mjs +3 -3
  43. package/esm2020/lib/components/data-view/common/elder-data-common.module.mjs +4 -4
  44. package/esm2020/lib/components/data-view/common/elder-data-toolbar/elder-data-toolbar.component.mjs +6 -6
  45. package/esm2020/lib/components/data-view/common/elder-single-sort/elder-single-sort.component.mjs +3 -3
  46. package/esm2020/lib/components/data-view/common/selection/data-context-selection.directive.mjs +3 -3
  47. package/esm2020/lib/components/data-view/common/selection/elder-selection-master-checkbox/elder-selection-master-checkbox.component.mjs +3 -3
  48. package/esm2020/lib/components/data-view/grid/elder-grid/elder-grid.component.mjs +9 -9
  49. package/esm2020/lib/components/data-view/grid/elder-grid.module.mjs +4 -4
  50. package/esm2020/lib/components/data-view/simple/elder-simple-selection-view/elder-simple-selection-view.component.mjs +3 -3
  51. package/esm2020/lib/components/data-view/simple/elder-simple-selection-view/elder-simple-selection-view.module.mjs +4 -4
  52. package/esm2020/lib/components/data-view/table/activation/elder-table-activation.directive.mjs +3 -3
  53. package/esm2020/lib/components/data-view/table/elder-number-cell.directive.mjs +3 -3
  54. package/esm2020/lib/components/data-view/table/elder-paginator-intl.mjs +3 -3
  55. package/esm2020/lib/components/data-view/table/elder-table/elder-table-toolbar.directive.mjs +3 -3
  56. package/esm2020/lib/components/data-view/table/elder-table/elder-table.component.mjs +3 -3
  57. package/esm2020/lib/components/data-view/table/elder-table-column.directive.mjs +3 -3
  58. package/esm2020/lib/components/data-view/table/elder-table-extension.directive.mjs +3 -3
  59. package/esm2020/lib/components/data-view/table/elder-table-root.directive.mjs +3 -3
  60. package/esm2020/lib/components/data-view/table/elder-table-row.directive.mjs +3 -3
  61. package/esm2020/lib/components/data-view/table/elder-table-sort.directive.mjs +3 -3
  62. package/esm2020/lib/components/data-view/table/elder-table.module.mjs +4 -4
  63. package/esm2020/lib/components/data-view/table/model/elder-table-model.mjs +3 -3
  64. package/esm2020/lib/components/dialogs/confirm-dialog/elder-confirm-dialog.component.mjs +3 -3
  65. package/esm2020/lib/components/dialogs/elder-dialog.module.mjs +7 -7
  66. package/esm2020/lib/components/dialogs/elder-dialog.service.mjs +3 -3
  67. package/esm2020/lib/components/dialogs/question-dialog/elder-question-dialog.component.mjs +3 -3
  68. package/esm2020/lib/components/dialogs/selection-dialog/elder-selection-dialog/elder-selection-dialog.component.mjs +3 -3
  69. package/esm2020/lib/components/dialogs/selection-dialog/elder-selection-dialog.directive.mjs +3 -3
  70. package/esm2020/lib/components/errors/elder-error.module.mjs +4 -4
  71. package/esm2020/lib/components/errors/exception-detail/elder-exception-detail.component.mjs +3 -3
  72. package/esm2020/lib/components/expand-toggle-button/elder-expand-toggle-button.component.mjs +3 -3
  73. package/esm2020/lib/components/expand-toggle-button/elder-expand-toggle-button.module.mjs +4 -4
  74. package/esm2020/lib/components/files/blob-viewer/elder-blob-viewer.component.mjs +3 -3
  75. package/esm2020/lib/components/files/elder-file-drop-zone.directive.mjs +3 -3
  76. package/esm2020/lib/components/files/elder-file-select.directive.mjs +3 -3
  77. package/esm2020/lib/components/files/elder-file.module.mjs +4 -4
  78. package/esm2020/lib/components/files/file-select/file-select.component.mjs +3 -3
  79. package/esm2020/lib/components/files/file-upload/file-upload.component.mjs +3 -3
  80. package/esm2020/lib/components/forms/clipboard/elder-clipboard.service.mjs +3 -3
  81. package/esm2020/lib/components/forms/directives/elder-clipboard-put.directive.mjs +3 -3
  82. package/esm2020/lib/components/forms/directives/elder-delayed-focus.directive.mjs +3 -3
  83. package/esm2020/lib/components/forms/directives/elder-form-field-dense.directive.mjs +3 -3
  84. package/esm2020/lib/components/forms/directives/elder-form-field-label.directive.mjs +3 -3
  85. package/esm2020/lib/components/forms/directives/elder-form-field-no-spinner.directive.mjs +3 -3
  86. package/esm2020/lib/components/forms/directives/elder-forms-directives.module.mjs +13 -13
  87. package/esm2020/lib/components/forms/directives/elder-key-event.directive.mjs +3 -3
  88. package/esm2020/lib/components/forms/directives/elder-max.directive.mjs +3 -3
  89. package/esm2020/lib/components/forms/directives/elder-min.directive.mjs +3 -3
  90. package/esm2020/lib/components/forms/directives/elder-next-focusable.directive.mjs +3 -3
  91. package/esm2020/lib/components/forms/directives/elder-plug-parent-form.directive.mjs +3 -3
  92. package/esm2020/lib/components/forms/directives/elder-stop-event-propagation.directive.mjs +3 -3
  93. package/esm2020/lib/components/forms/directives/elder-tab-focus-trap.directive.mjs +3 -3
  94. package/esm2020/lib/components/forms/directives/elder-touched.directive.mjs +3 -3
  95. package/esm2020/lib/components/forms/directives/elder-triple-state-checkbox.directive.mjs +3 -3
  96. package/esm2020/lib/components/forms/directives/validation/elder-multiple-of.validator.mjs +3 -3
  97. package/esm2020/lib/components/forms/directives/validation/elder-required-ignore-zero.validator.mjs +3 -3
  98. package/esm2020/lib/components/forms/elder-forms.module.mjs +4 -4
  99. package/esm2020/lib/components/forms/search/elder-search-context.directive.mjs +3 -3
  100. package/esm2020/lib/components/forms/search/elder-search-input.directive.mjs +3 -3
  101. package/esm2020/lib/components/forms/search/elder-search.module.mjs +4 -4
  102. package/esm2020/lib/components/forms/search/search-box/elder-search-box.component.mjs +3 -3
  103. package/esm2020/lib/components/forms/search/search-box/elder-search-panel.component.mjs +3 -3
  104. package/esm2020/lib/components/global-search/elder-global-search.component.mjs +3 -3
  105. package/esm2020/lib/components/global-search/elder-global-search.module.mjs +4 -4
  106. package/esm2020/lib/components/global-search/elder-global-search.service.mjs +3 -3
  107. package/esm2020/lib/components/graph/elder-progress-bar/elder-progress-bar.component.mjs +3 -3
  108. package/esm2020/lib/components/graph/elder-progress-bar/elder-progress-bar.module.mjs +4 -4
  109. package/esm2020/lib/components/headers/elder-header/elder-header.component.mjs +3 -3
  110. package/esm2020/lib/components/headers/elder-header.module.mjs +4 -4
  111. package/esm2020/lib/components/http-support/elder-http-client.service.mjs +3 -3
  112. package/esm2020/lib/components/i18n/entities/elder-i18n-entities.module.mjs +4 -4
  113. package/esm2020/lib/components/i18n/entities/elder-localized-input/elder-localized-input.component.mjs +3 -3
  114. package/esm2020/lib/components/i18n/entities/elder-localized-input-dialog/elder-localized-input-dialog.component.mjs +3 -3
  115. package/esm2020/lib/components/i18n/entities/elder-localized-input-dialog.service.mjs +3 -3
  116. package/esm2020/lib/components/i18n/entities/elder-localized-input-table/elder-localized-input-table.component.mjs +3 -3
  117. package/esm2020/lib/components/i18n/entities/elder-localized-text-column.directive.mjs +3 -3
  118. package/esm2020/lib/components/i18n/entities/elder-localized-texts.directive.mjs +3 -3
  119. package/esm2020/lib/components/i18n/entities/picker/i18n-pick-async.pipe.mjs +3 -3
  120. package/esm2020/lib/components/i18n/entities/picker/i18n-pick.pipe.mjs +3 -3
  121. package/esm2020/lib/components/i18n/entities/picker/localisation-picker.service.mjs +3 -3
  122. package/esm2020/lib/components/i18n/language/elder-language-interceptor.mjs +3 -3
  123. package/esm2020/lib/components/i18n/language/elder-language.module.mjs +4 -4
  124. package/esm2020/lib/components/i18n/language/elder-language.service.mjs +3 -3
  125. package/esm2020/lib/components/i18n/language/language-switcher/elder-language-switcher.component.mjs +3 -3
  126. package/esm2020/lib/components/i18n/locales/elder-locales-de-ch.module.mjs +4 -4
  127. package/esm2020/lib/components/iframes/elder-iframe.module.mjs +4 -4
  128. package/esm2020/lib/components/iframes/iframe-close.directive.mjs +3 -3
  129. package/esm2020/lib/components/iframes/iframe-dialog/iframe-dialog.component.mjs +3 -3
  130. package/esm2020/lib/components/iframes/iframe-host/iframe-host.component.mjs +3 -3
  131. package/esm2020/lib/components/iframes/iframe-side-content/iframe-side-content.component.mjs +3 -3
  132. package/esm2020/lib/components/iframes/iframe.service.mjs +3 -3
  133. package/esm2020/lib/components/infinitescroll/elder-infinite-autocomplete.directive.mjs +3 -3
  134. package/esm2020/lib/components/infinitescroll/elder-infinite-scroll.directive.mjs +3 -3
  135. package/esm2020/lib/components/infinitescroll/elder-infinite-scroll.module.mjs +7 -7
  136. package/esm2020/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.mjs +3 -3
  137. package/esm2020/lib/components/input/autocomplete/elder-autocomplete.directive.mjs +3 -3
  138. package/esm2020/lib/components/input/autocomplete/elder-autocomplete.module.mjs +4 -4
  139. package/esm2020/lib/components/labels/elder-labels.module.mjs +4 -4
  140. package/esm2020/lib/components/labels/labels-input/labels-input.component.mjs +3 -3
  141. package/esm2020/lib/components/measures/dimensions-input/elder-dimensions-input.component.mjs +3 -3
  142. package/esm2020/lib/components/measures/directives/elder-unit-select.directive.mjs +3 -3
  143. package/esm2020/lib/components/measures/elder-measures.module.mjs +8 -8
  144. package/esm2020/lib/components/measures/elder-quantity-transform.pipe.mjs +3 -3
  145. package/esm2020/lib/components/measures/elder-quantity.pipe.mjs +3 -3
  146. package/esm2020/lib/components/measures/elder-unit.service.mjs +3 -3
  147. package/esm2020/lib/components/measures/quantity-input/elder-quantity-input.component.mjs +3 -3
  148. package/esm2020/lib/components/measures/util/elder-quantity.service.mjs +3 -3
  149. package/esm2020/lib/components/multi-autocomplete/elder-multi-auto-complete.module.mjs +4 -4
  150. package/esm2020/lib/components/multi-autocomplete/elder-multi-autocomplete.component.mjs +3 -3
  151. package/esm2020/lib/components/navigation/bread-crumbs/bread-crumbs/elder-bread-crumbs.component.mjs +3 -3
  152. package/esm2020/lib/components/navigation/bread-crumbs/elder-bread-crumbs.module.mjs +4 -4
  153. package/esm2020/lib/components/navigation/nav/elder-nav.module.mjs +4 -4
  154. package/esm2020/lib/components/navigation/nav/nav-group/elder-nav-group.component.mjs +3 -3
  155. package/esm2020/lib/components/navigation/nav/nav-link/elder-nav-link.component.mjs +3 -3
  156. package/esm2020/lib/components/navigation/nav/nav-list/elder-nav-list.component.mjs +3 -3
  157. package/esm2020/lib/components/navigation/toolbar/elder-toolbar-column.directive.mjs +3 -3
  158. package/esm2020/lib/components/navigation/toolbar/elder-toolbar.module.mjs +4 -4
  159. package/esm2020/lib/components/navigation/toolbar/elder-toolbar.service.mjs +3 -3
  160. package/esm2020/lib/components/navigation/toolbar/toolbar/elder-toolbar.component.mjs +3 -3
  161. package/esm2020/lib/components/navigation/toolbar/toolbar-title/elder-toolbar-title.component.mjs +3 -3
  162. package/esm2020/lib/components/navigation/toolbar/toolbar-title/elder-toolbar-title.service.mjs +3 -3
  163. package/esm2020/lib/components/overlays/elder-overlay-origin.directive.mjs +3 -3
  164. package/esm2020/lib/components/overlays/elder-overlay-trigger.directive.mjs +3 -3
  165. package/esm2020/lib/components/overlays/elder-overlay.component.mjs +3 -3
  166. package/esm2020/lib/components/overlays/elder-overlay.module.mjs +4 -4
  167. package/esm2020/lib/components/panels/card-panel/elder-card-panel.component.mjs +3 -3
  168. package/esm2020/lib/components/panels/elder-panel.module.mjs +4 -4
  169. package/esm2020/lib/components/panels/flat/elder-panel.component.mjs +3 -3
  170. package/esm2020/lib/components/select/auto-select-first.directive.mjs +126 -0
  171. package/esm2020/lib/components/select/elder-entity-value-accessor.mjs +12 -0
  172. package/esm2020/lib/components/select/elder-select/elder-select.component.mjs +109 -159
  173. package/esm2020/lib/components/select/elder-select-base.mjs +22 -10
  174. package/esm2020/lib/components/select/elder-select-chip.directive.mjs +3 -3
  175. package/esm2020/lib/components/select/elder-select-value.directive.mjs +3 -3
  176. package/esm2020/lib/components/select/elder-select.module.mjs +15 -9
  177. package/esm2020/lib/components/select/multi/elder-multi-select-base.mjs +102 -44
  178. package/esm2020/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.mjs +15 -15
  179. package/esm2020/lib/components/select/popup/selection-model-popup.directive.mjs +3 -3
  180. package/esm2020/lib/components/select/popup/templated-selection-dialog/templated-selection-dialog.component.mjs +3 -3
  181. package/esm2020/lib/components/select/select-on-tab.directive.mjs +22 -11
  182. package/esm2020/lib/components/select-chip-list/chip-list-select/elder-chip-list-select.component.mjs +3 -3
  183. package/esm2020/lib/components/select-chip-list/elder-chip-list-select.module.mjs +4 -4
  184. package/esm2020/lib/components/select-list/elder-select-list.module.mjs +4 -4
  185. package/esm2020/lib/components/select-list/select-list/elder-select-list.component.mjs +3 -3
  186. package/esm2020/lib/components/select-list/select-list-item/elder-select-list-item.component.mjs +3 -3
  187. package/esm2020/lib/components/shell/drawers/elder-route-outlet-drawer.service.mjs +3 -3
  188. package/esm2020/lib/components/shell/drawers/elder-router-outlet.service.mjs +3 -3
  189. package/esm2020/lib/components/shell/elder-shell-slot.directive.mjs +3 -3
  190. package/esm2020/lib/components/shell/elder-shell.module.mjs +4 -4
  191. package/esm2020/lib/components/shell/elder-shell.service.mjs +3 -3
  192. package/esm2020/lib/components/shell/header/elder-app-header/elder-app-header.component.mjs +3 -3
  193. package/esm2020/lib/components/shell/shell/elder-shell.component.mjs +12 -12
  194. package/esm2020/lib/components/shell/shell-navigation-toggle/elder-shell-navigation-toggle.component.mjs +3 -3
  195. package/esm2020/lib/components/tabs/elder-tab/elder-tab.directive.mjs +3 -3
  196. package/esm2020/lib/components/tabs/elder-tab-group-routing/elder-tab-group-routing.directive.mjs +3 -3
  197. package/esm2020/lib/components/tabs/elder-tab.module.mjs +4 -4
  198. package/esm2020/lib/components/time/date-adapters/custom-date-adapter.mjs +3 -3
  199. package/esm2020/lib/components/time/duration/elder-duration-input/elder-duration-input.component.mjs +3 -3
  200. package/esm2020/lib/components/time/elder-date-switcher/elder-date-switcher.component.mjs +3 -3
  201. package/esm2020/lib/components/time/elder-date-time-input/elder-date-time-input.component.mjs +3 -3
  202. package/esm2020/lib/components/time/elder-interval-input/elder-interval-input.component.mjs +3 -3
  203. package/esm2020/lib/components/time/elder-local-date-input/elder-local-date-input.component.mjs +3 -3
  204. package/esm2020/lib/components/time/elder-local-time-input/elder-local-time-input.component.mjs +3 -3
  205. package/esm2020/lib/components/time/elder-time.module.mjs +8 -8
  206. package/esm2020/lib/components/time/period/elder-period-input/elder-period-input.component.mjs +3 -3
  207. package/esm2020/lib/components/toasts/elder-toast.module.mjs +4 -4
  208. package/esm2020/lib/components/toasts/elder-toast.service.mjs +9 -9
  209. package/esm2020/lib/components/toasts/standard-toast/standard-toast.component.mjs +3 -3
  210. package/esm2020/lib/components/url-fragment/elder-url-fragment.module.mjs +4 -4
  211. package/esm2020/lib/components/url-fragment/url-fragment-switcher/elder-url-fragment-switcher.component.mjs +3 -3
  212. package/esm2020/lib/components/viewers/elder-svg-viewer/elder-svg-viewer.component.mjs +3 -3
  213. package/esm2020/lib/components/viewers/elder-viewers.module.mjs +4 -4
  214. package/esm2020/lib/features/event-source/elder-event-source.service.mjs +3 -3
  215. package/esm2020/lib/features/kafent/access-token-provider.mjs +3 -3
  216. package/esm2020/lib/features/kafent/kafent-config.mjs +3 -3
  217. package/esm2020/lib/features/kafent/kafent-event-stream.mjs +3 -3
  218. package/esm2020/lib/features/kafent/kafent-event.service.mjs +3 -3
  219. package/esm2020/lib/features/kafent/kafent.module.mjs +4 -4
  220. package/esm2020/lib/features/kafent/sse/kafent-event-stream-sse.service.mjs +3 -3
  221. package/esm2020/lib/pipes/bytes.pipe.mjs +3 -3
  222. package/esm2020/lib/pipes/elder-pipes.module.mjs +7 -7
  223. package/esm2020/lib/pipes/elder-repeat.pipe.mjs +3 -3
  224. package/esm2020/lib/pipes/elder-round.pipe.mjs +3 -3
  225. package/esm2020/lib/pipes/elder-safe-url.pipe.mjs +3 -3
  226. package/esm2020/lib/pipes/iso-duration.pipe.mjs +3 -3
  227. package/esm2020/lib/pipes/iso-interval-parse.pipe.mjs +3 -3
  228. package/esm2020/lib/pipes/iso-interval.pipe.mjs +3 -3
  229. package/esm2020/lib/pipes/time-ago.pipe.mjs +3 -3
  230. package/esm2020/lib/pipes/time-duration.pipe.mjs +3 -3
  231. package/esm2020/lib/pipes/weight.pipe.mjs +3 -3
  232. package/fesm2015/elderbyte-ngx-starter.mjs +1185 -1022
  233. package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
  234. package/fesm2020/elderbyte-ngx-starter.mjs +1185 -1020
  235. package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
  236. package/lib/common/utils/collection-util.d.ts +1 -0
  237. package/lib/components/select/auto-select-first.directive.d.ts +54 -0
  238. package/lib/components/select/elder-entity-value-accessor.d.ts +27 -0
  239. package/lib/components/select/elder-select/elder-select.component.d.ts +42 -48
  240. package/lib/components/select/elder-select-base.d.ts +21 -15
  241. package/lib/components/select/elder-select.module.d.ts +22 -20
  242. package/lib/components/select/multi/elder-multi-select-base.d.ts +29 -15
  243. package/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.d.ts +11 -11
  244. package/lib/components/select/select-on-tab.directive.d.ts +7 -6
  245. package/package.json +3 -3
@@ -1,12 +1,11 @@
1
1
  import { ChangeDetectionStrategy, Component, forwardRef, Input, Output, } from '@angular/core';
2
2
  import { buildFormIntegrationProviders } from '../../../common/forms/template-composite-control';
3
3
  import { LoggerFactory } from '@elderbyte/ts-logger';
4
- import { BehaviorSubject, combineLatest, EMPTY, of, Subject } from 'rxjs';
5
- import { catchError, filter, map, switchMap, takeUntil, timeout, first, tap, } from 'rxjs/operators';
4
+ import { BehaviorSubject, combineLatest, Subject } from 'rxjs';
5
+ import { map, takeUntil, tap, skip, filter, } from 'rxjs/operators';
6
6
  import { coerceBooleanProperty } from '@angular/cdk/coercion';
7
- import { NgControl } from '@angular/forms';
8
- import { DataSourceAdapter } from '../../../common/data/datasource/data-source-adapter';
9
7
  import { ELDER_SELECT_BASE, ElderSelectBase, ElderSelectComponentState } from '../elder-select-base';
8
+ import { AutoSelectFirstDirective } from '../auto-select-first.directive';
10
9
  import * as i0 from "@angular/core";
11
10
  import * as i1 from "@angular/material/form-field";
12
11
  import * as i2 from "@angular/material/icon";
@@ -21,7 +20,7 @@ import * as i10 from "../select-on-tab.directive";
21
20
  import * as i11 from "../../input/autocomplete/elder-autocomplete.directive";
22
21
  import * as i12 from "../../forms/directives/elder-stop-event-propagation.directive";
23
22
  import * as i13 from "@ngx-translate/core";
24
- class ValueContext {
23
+ class EntityContext {
25
24
  constructor(value, hintText, displayText) {
26
25
  this.value = value;
27
26
  this.hintText = hintText;
@@ -44,9 +43,8 @@ export class ElderSelectComponent extends ElderSelectBase {
44
43
  * Constructor *
45
44
  * *
46
45
  **************************************************************************/
47
- constructor(zone, injector) {
46
+ constructor(zone) {
48
47
  super(zone);
49
- this.injector = injector;
50
48
  /***************************************************************************
51
49
  * *
52
50
  * Fields *
@@ -59,18 +57,20 @@ export class ElderSelectComponent extends ElderSelectBase {
59
57
  * String to display for the 'nothing' / null value.
60
58
  */
61
59
  this.nullDisplay = '-';
62
- this._autoSelectFirst = false;
63
- this._valueId = null;
60
+ this.entity$ = new BehaviorSubject(undefined);
64
61
  this._autocomplete = false;
65
62
  this.unsubscribe$ = new Subject();
66
- this.ngModelInitialized = false;
67
- this.valueIdChange = this.valueChange.pipe(map(v => this.getValueId(v)));
68
- this.valueWrapped$ = combineLatest([
69
- this.value$,
63
+ this.entityIdChange = this.valueChange.pipe(map(v => this.entityIdFromValue(v)));
64
+ this.entityChange = this.entity$.pipe(skip(1) // Skip the initial NULL value
65
+ );
66
+ this.entityIdUpdated = this.valueUpdated.pipe(map(value => this.entityIdFromValue(value)));
67
+ this.entityUpdated = combineLatest([this.entityIdUpdated, this.entity$]).pipe(filter(([updatedId, entity]) => this.getEntityId(entity) === updatedId), map(([updatedId, entity]) => entity));
68
+ this.entityWrapped$ = combineLatest([
69
+ this.entity$,
70
70
  this.displayPropertyResolver$,
71
71
  this.hintPropertyResolver$
72
- ]).pipe(map(([value, dPR, hPR]) => new ValueContext(value, hPR(value), dPR(value))));
73
- this.inputText$ = this.valueWrapped$.pipe(takeUntil(this.unsubscribe$), tap(valueContext => this.valueContext = valueContext), map(valueContext => {
72
+ ]).pipe(map(([value, dPR, hPR]) => new EntityContext(value, hPR(value), dPR(value))));
73
+ this.inputText$ = this.entityWrapped$.pipe(takeUntil(this.unsubscribe$), tap(valueContext => this.entityContext = valueContext), map(valueContext => {
74
74
  if (valueContext.value) {
75
75
  return valueContext.displayText;
76
76
  }
@@ -85,17 +85,6 @@ export class ElderSelectComponent extends ElderSelectBase {
85
85
  * *
86
86
  **************************************************************************/
87
87
  ngOnInit() {
88
- // Inject the ngControl/ngModel
89
- this.ngControl = this.injector.get(NgControl, null);
90
- this.ensureExistingValueNgControl(this.value);
91
- if (this.autoSelectFirst) {
92
- this.handleAutoSelectFirst();
93
- }
94
- }
95
- ngAfterViewInit() {
96
- setTimeout(() => {
97
- this.ngModelInitialized = true;
98
- }, 10);
99
88
  }
100
89
  ngOnDestroy() {
101
90
  this.unsubscribe$.next();
@@ -122,39 +111,29 @@ export class ElderSelectComponent extends ElderSelectBase {
122
111
  get allowNull() {
123
112
  return this._allowNull;
124
113
  }
125
- /**
126
- * After data is loaded into this select, ensure that
127
- * a value is selected or select the first automatically.
128
- */
129
- set autoSelectFirst(val) {
130
- this._autoSelectFirst = coerceBooleanProperty(val);
131
- }
132
- get autoSelectFirst() {
133
- return this._autoSelectFirst;
134
- }
135
114
  /***************************************************************************
136
115
  * *
137
116
  * Properties *
138
117
  * *
139
118
  **************************************************************************/
140
- /**
141
- * Select the value with the given id.
142
- * @param id
143
- */
144
- set valueId(id) {
145
- if (id !== this._valueId) {
146
- this._valueId = id;
147
- this.selectValueById(id);
148
- }
119
+ set entity(entity) {
120
+ this.entity$.next(entity);
121
+ this.writeValueInternal(this.entityToValue(entity));
149
122
  }
150
- get valueId() {
151
- if (this.value) {
152
- return this.getValueId(this.value);
123
+ get entity() {
124
+ return this.entity$.getValue();
125
+ }
126
+ set entityId(id) {
127
+ if (this.valueAsId) {
128
+ this.writeValueInternal(id);
153
129
  }
154
130
  else {
155
- return this._valueId;
131
+ this.selectEntityById(id);
156
132
  }
157
133
  }
134
+ get entityId() {
135
+ return this.entityIdFromValue(this.value);
136
+ }
158
137
  set hintProperty(property) {
159
138
  this.hintPropertyResolver = (value => this.propertyStringValue(value, property));
160
139
  }
@@ -168,8 +147,8 @@ export class ElderSelectComponent extends ElderSelectBase {
168
147
  **************************************************************************/
169
148
  onInputBlur(event) {
170
149
  const input = event.target;
171
- if (this.valueContext.value) {
172
- input.value = this.valueContext.displayText;
150
+ if (this.entityContext.value) {
151
+ input.value = this.entityContext.displayText;
173
152
  }
174
153
  else {
175
154
  input.value = null;
@@ -177,7 +156,7 @@ export class ElderSelectComponent extends ElderSelectBase {
177
156
  }
178
157
  get isOptionDisabledInternalFn() {
179
158
  return (option) => {
180
- if (!this.valuesEquals(option, this.value)) {
159
+ if (!this.isEntitySelected(option)) {
181
160
  if (this.isOptionDisabledFn) {
182
161
  return this.isOptionDisabledFn(option);
183
162
  }
@@ -224,15 +203,15 @@ export class ElderSelectComponent extends ElderSelectBase {
224
203
  this.togglePanel(autoTrigger);
225
204
  }
226
205
  onOptionSelected(selectedValue) {
227
- if (this.valuesEquals(selectedValue, this.value)) {
206
+ if (this.isEntitySelected(selectedValue)) {
228
207
  this.writeValueInternal(null); // HACK: Ensure we trigger view redraw
229
208
  }
230
- this.updateValue(selectedValue);
209
+ this.updateValueByEntity(selectedValue);
231
210
  }
232
211
  openSelectionPopup(event) {
233
212
  if (this.selectionPopup) {
234
- this.selectionPopup.choose(this.filters, this.sorts).subscribe(selection => {
235
- this.updateValue(selection[0]);
213
+ this.selectionPopup.choose(this.filters, this.sorts).subscribe((selection) => {
214
+ this.updateValueByEntity(selection[0]);
236
215
  });
237
216
  }
238
217
  else {
@@ -244,58 +223,11 @@ export class ElderSelectComponent extends ElderSelectBase {
244
223
  }
245
224
  forceReloadFirst() {
246
225
  this.value = null; // valueId is set to null automatically
247
- this.handleAutoSelectFirst();
248
- }
249
- /***************************************************************************
250
- * *
251
- * Private Methods (Auto Select First) *
252
- * *
253
- **************************************************************************/
254
- handleAutoSelectFirst() {
255
- if (!this._valueId) {
256
- this.logger.debug('Attempting to auto select first entity...');
257
- this.trySelectFirst();
258
- }
259
- else {
260
- this.logger.warn('Conflict avoided: Aborted auto selection because value id already present!');
261
- }
262
- }
263
- trySelectFirst() {
264
- const dataContextFirstItem = this.suggestionsDc?.isStarted
265
- ? this.firstItemInDataContextSoon(this.suggestionsDc)
266
- : of(null);
267
- // TODO takeUntil value-id is set, after that we can cancel.
268
- dataContextFirstItem.pipe(takeUntil(this.valueIdChange.pipe(filter(id => !!id))), switchMap(firstItem => firstItem ? of(firstItem) : this.firstSuggestionItem())).subscribe(firstItem => {
269
- if (firstItem && !this._valueId) {
270
- this.updateValue(firstItem);
271
- }
272
- }, err => {
273
- this.updateState(ElderSelectComponentState.error(err));
274
- this.logger.error('Failed to load first entity for auto-select-first flow!', err);
275
- });
226
+ const autoReloadFirst = new AutoSelectFirstDirective(this);
227
+ autoReloadFirst.trySelectFirst();
276
228
  }
277
- firstItemInDataContextSoon(dataContext) {
278
- return dataContext.data.pipe(filter(data => data && data.length > 0), map(data => data[0]), timeout(250), catchError(timeoutError => {
279
- this.logger.warn('Gave up waiting for DataContext items, timeout reached. Falling back to sugestion lookup.', timeoutError);
280
- return of(null);
281
- }), first());
282
- }
283
- firstSuggestionItem() {
284
- const dataSource = this.suggestionsDc?.dataSource;
285
- if (dataSource) {
286
- this.updateState(ElderSelectComponentState.loading());
287
- return DataSourceAdapter.from(dataSource)
288
- .findFirst(this.filters, this.sorts).pipe(tap(() => this.updateState(ElderSelectComponentState.idle())), catchError(err => {
289
- this.logger.error('Failed to load suggestions!', err);
290
- this.updateState(ElderSelectComponentState.error(err));
291
- return of(null);
292
- }));
293
- }
294
- else {
295
- this.logger.warn('Failed to load first suggestion!');
296
- this.updateState(ElderSelectComponentState.idle());
297
- return EMPTY;
298
- }
229
+ updateValueByEntity(entity) {
230
+ this.updateValue(this.entityToValue(entity));
299
231
  }
300
232
  /***************************************************************************
301
233
  * *
@@ -306,8 +238,8 @@ export class ElderSelectComponent extends ElderSelectBase {
306
238
  * Occurs when the suggestions data-context has changed
307
239
  */
308
240
  onSuggestionsDcChanged(data) {
309
- if (this._valueId) {
310
- this.selectValueById(this._valueId);
241
+ if (this.valueAsId) {
242
+ this.selectEntityById(this.entityId);
311
243
  }
312
244
  }
313
245
  /***************************************************************************
@@ -316,22 +248,35 @@ export class ElderSelectComponent extends ElderSelectBase {
316
248
  * *
317
249
  **************************************************************************/
318
250
  valuesEquals(a, b) {
319
- return this.isEqual(a, b);
251
+ return this.entityIdFromValue(a) === this.entityIdFromValue(b);
252
+ }
253
+ isEntitySelected(entity) {
254
+ return this.getEntityId(entity) === this.entityId;
320
255
  }
321
- selectValueById(id) {
322
- if (this.value && this.getValueId(this.value) === id) {
323
- return; // Already selected
256
+ entityIdFromValue(value) {
257
+ if (this.valueAsId) {
258
+ return value;
259
+ }
260
+ else {
261
+ const entity = value;
262
+ return this.getEntityId(entity);
263
+ }
264
+ }
265
+ selectEntityById(id) {
266
+ const currentEntity = this.entity;
267
+ if (currentEntity && this.getEntityId(currentEntity) === id) {
268
+ return; // Entity already loaded
324
269
  }
325
270
  if (id === null || id === undefined) {
326
- if (this.value !== null && this.value !== undefined) {
327
- this.writeValueInternal(null);
271
+ if (currentEntity !== null && currentEntity !== undefined) {
272
+ this.entity = null;
328
273
  }
329
274
  }
330
275
  else {
331
276
  if (this.suggestionsDc) {
332
277
  const existing = this.findInDataContext(id);
333
278
  if (existing) {
334
- this.writeValueInternal(existing);
279
+ this.entity = existing;
335
280
  }
336
281
  else {
337
282
  this.loadEntityById(id);
@@ -346,39 +291,49 @@ export class ElderSelectComponent extends ElderSelectBase {
346
291
  this.suggestionsDc.dataSource.findById(id)
347
292
  .subscribe(entity => {
348
293
  this.logger.debug('Loaded missing entity by id ' + id + ':', entity);
349
- this.writeValueInternalDelayed(entity);
294
+ this.entity = entity;
350
295
  }, err => {
351
296
  this.logger.error('Failed to load entity by id ' + id, err);
352
297
  this.updateState(ElderSelectComponentState.error(err));
353
298
  });
354
299
  }
355
- writeValueInternalDelayed(value) {
356
- setTimeout(// HACK: Counter ngModel race condition
357
- () => this.writeValueInternal(value), 10);
358
- }
359
300
  findInDataContext(id) {
360
301
  return this.suggestionsDc.snapshot.data
361
- .find(v => this.getValueId(v) === id);
362
- }
363
- writeToControl(value) {
364
- this._valueId = this.getValueId(value);
365
- this.ensureExistingValueNgControl(value);
366
- super.writeToControl(value);
302
+ .find(v => this.getEntityId(v) === id);
367
303
  }
368
- ensureExistingValueNgControl(value) {
369
- if (value) {
370
- // Patch ngModel value (especially useful when using ngModel & value-id binding)
371
- this.tryUpdateNgControl(value);
304
+ entityToValue(entity) {
305
+ if (this.valueAsId) {
306
+ const id = this.getEntityId(entity);
307
+ return id;
308
+ }
309
+ else {
310
+ return entity;
372
311
  }
373
312
  }
374
- tryUpdateNgControl(value) {
375
- if (this.ngControl) {
376
- const ngModel = this.ngControl;
377
- if (ngModel) {
378
- ngModel.model = value;
379
- this.logger.debug('Wrote value directly into ngModel:', value);
313
+ /**
314
+ * This method is invoked after a value has been written to this control.
315
+ *
316
+ */
317
+ writeToControl(value) {
318
+ if (this.valueAsId) {
319
+ const entityId = value;
320
+ // Value was written as entity id, ensure we select entity by id
321
+ this.logger.debug('writeToControl: value was written as id: ' + entityId, value);
322
+ this.selectEntityById(entityId);
323
+ }
324
+ else {
325
+ // Value was written as entity, ensure entity is updated
326
+ const currentEntity = this.entity;
327
+ const newEntity = value;
328
+ this.logger.debug('writeToControl: value was written as entity: ' + JSON.stringify(newEntity), value);
329
+ if (!this.isEqual(currentEntity, newEntity)) {
330
+ this.entity$.next(newEntity);
331
+ }
332
+ else {
333
+ this.logger.warn('Ignored written entity as it is already set to entity$!');
380
334
  }
381
335
  }
336
+ super.writeToControl(value);
382
337
  }
383
338
  /**
384
339
  * @description
@@ -391,32 +346,21 @@ export class ElderSelectComponent extends ElderSelectBase {
391
346
  */
392
347
  writeValue(value) {
393
348
  // this.logger.debug('ElderSelectBase.writeValue(): Current value-id ' + this._valueId + ', new value: ', value);
394
- if (this.isNgModelInitializing() && this.isNullOrEmpty(value)) {
395
- this.logger.debug('Prevented null/empty write while ngModel is still initializing!');
396
- }
397
- else {
398
- super.writeValue(value);
399
- }
349
+ super.writeValue(value);
400
350
  }
401
351
  isNullOrEmpty(value) {
402
352
  return (value == null || value === '');
403
353
  }
404
- isNgModelInitializing() {
405
- if (this.ngControl) {
406
- return !this.ngModelInitialized;
407
- }
408
- return false;
409
- }
410
354
  }
411
- ElderSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.2", ngImport: i0, type: ElderSelectComponent, deps: [{ token: i0.NgZone }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Component });
412
- ElderSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.2.2", type: ElderSelectComponent, selector: "elder-select", inputs: { nullDisplay: "nullDisplay", autocomplete: "autocomplete", allowNull: "allowNull", autoSelectFirst: "autoSelectFirst", valueId: "valueId", hintProperty: "hintProperty", hintPropertyResolver: "hintPropertyResolver" }, outputs: { valueIdChange: "valueIdChange" }, providers: [
355
+ ElderSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
356
+ ElderSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.1", type: ElderSelectComponent, selector: "elder-select", inputs: { nullDisplay: "nullDisplay", autocomplete: "autocomplete", allowNull: "allowNull", entity: "entity", entityId: "entityId", hintProperty: "hintProperty", hintPropertyResolver: "hintPropertyResolver" }, outputs: { entityIdChange: "entityIdChange", entityChange: "entityChange", entityIdUpdated: "entityIdUpdated", entityUpdated: "entityUpdated" }, providers: [
413
357
  {
414
358
  provide: ELDER_SELECT_BASE,
415
359
  useExisting: forwardRef(() => ElderSelectComponent)
416
360
  },
417
361
  ...buildFormIntegrationProviders(ElderSelectComponent)
418
- ], usesInheritance: true, ngImport: i0, template: "<mat-form-field *ngIf=\"(valueWrapped$ | async) as valueWrapper\" fxFlex\n class=\"elder-std-form-field\"\n [appearance]=\"appearance\"\n [floatLabel]=\"floatLabel\"\n [color]=\"color\"\n>\n\n <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n <mat-icon *ngIf=\"icon\" disabled\n class=\"leading-icon prefix-padding noselect\"\n [class.loading]=\"state.loading\"\n [color]=\"state?.error ? 'warn' : color\">\n {{icon}}\n </mat-icon>\n\n\n <mat-icon *ngIf=\"!icon && state?.error\"\n class=\"leading-icon prefix-padding noselect\"\n color=\"warn\">\n warning\n </mat-icon>\n </ng-container>\n\n <!-- A dynamic input -->\n <input\n matInput type=\"text\" fxFlex=\"grow\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [readonly]=\"readonly || !autocomplete\"\n [name]=\"name + '-inner-input'\"\n [placeholder]=\"placeholder | translate\"\n [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n elderSelectOnTab\n [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n (blur)=\"onInputBlur($event)\"\n >\n\n <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n <elder-autocomplete\n #elderAuto=\"elderAutocomplete\"\n [suggestionsDc]=\"suggestionsDc$ | async\"\n [valueTemplate]=\"valueTemplate\"\n [enabled]=\"autocomplete\"\n [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n (optionSelected)=\"onOptionSelected($any($event))\"\n ></elder-autocomplete>\n\n\n <ng-container matSuffix>\n\n <mat-icon\n *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n (click)=\"onInputClicked(autoTrigger)\">\n arrow_drop_down\n </mat-icon>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"selectionPopup && (!valueWrapper.value || !allowNull)\"\n [disabled]=\"isLocked\"\n (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>search</mat-icon>\n </button>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"valueWrapper.value && allowNull\"\n [disabled]=\"isLocked\"\n (click)=\"clear($event)\" aria-label=\"Clear\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>close</mat-icon>\n </button>\n\n </ng-container>\n\n <mat-hint *ngIf=\"valueWrapper.hintText\">{{valueWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n", styles: [".prefix-container{width:24px;height:16px}@-webkit-keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}@keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}.loading{animation:shrink .3s ease-in-out infinite alternate;-webkit-animation:shrink .3s ease-in-out infinite alternate}.prefix-padding{padding-right:4px}.leading-icon{font-size:16px;width:16px;height:16px}.full-width{width:100%}.select{cursor:pointer;width:162px}.select-arrow{font-size:18px;width:18px;height:18px;cursor:pointer}\n"], components: [{ type: i1.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.ElderAutocompleteComponent, selector: "elder-autocomplete", inputs: ["isOptionDisabledFn", "isOptionHiddenFn", "enabled", "valueTemplate", "suggestionsDc", "displayPropertyResolver"], outputs: ["optionSelected"], exportAs: ["elderAutocomplete"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { type: i1.MatLabel, selector: "mat-label" }, { type: i1.MatPrefix, selector: "[matPrefix]" }, { type: i7.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"] }, { type: i8.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { type: i9.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i10.SelectOnTabDirective, selector: "[elderSelectOnTab]" }, { type: i9.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { type: i11.ElderAutocompleteDirective, selector: "[elderElderAutocomplete]", inputs: ["queryFilter", "filters", "sorts", "elderElderAutocomplete"] }, { type: i9.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i9.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i1.MatSuffix, selector: "[matSuffix]" }, { type: i12.ElderStopEventPropagationDirective, selector: "[elderStopEventPropagation]" }, { type: i1.MatHint, selector: "mat-hint", inputs: ["align", "id"] }], pipes: { "async": i5.AsyncPipe, "translate": i13.TranslatePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
419
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.2", ngImport: i0, type: ElderSelectComponent, decorators: [{
362
+ ], usesInheritance: true, ngImport: i0, template: "<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n class=\"elder-std-form-field\"\n [appearance]=\"appearance\"\n [floatLabel]=\"floatLabel\"\n [color]=\"color\"\n>\n\n <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n <mat-icon *ngIf=\"icon\" disabled\n class=\"leading-icon prefix-padding noselect\"\n [class.loading]=\"state.loading\"\n [color]=\"state?.error ? 'warn' : color\">\n {{icon}}\n </mat-icon>\n\n\n <mat-icon *ngIf=\"!icon && state?.error\"\n class=\"leading-icon prefix-padding noselect\"\n color=\"warn\">\n warning\n </mat-icon>\n </ng-container>\n\n <!-- A dynamic input -->\n <input\n matInput type=\"text\" fxFlex=\"grow\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [readonly]=\"readonly || !autocomplete\"\n [name]=\"name + '-inner-input'\"\n [placeholder]=\"placeholder | translate\"\n [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n elderSelectOnTab\n [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n (blur)=\"onInputBlur($event)\"\n >\n\n <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n <elder-autocomplete\n #elderAuto=\"elderAutocomplete\"\n [suggestionsDc]=\"suggestionsDc$ | async\"\n [valueTemplate]=\"valueTemplate\"\n [enabled]=\"autocomplete\"\n [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n (optionSelected)=\"onOptionSelected($any($event))\"\n ></elder-autocomplete>\n\n\n <ng-container matSuffix>\n\n <mat-icon\n *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n (click)=\"onInputClicked(autoTrigger)\">\n arrow_drop_down\n </mat-icon>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n [disabled]=\"isLocked\"\n (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>search</mat-icon>\n </button>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"entityWrapper.value && allowNull\"\n [disabled]=\"isLocked\"\n (click)=\"clear($event)\" aria-label=\"Clear\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>close</mat-icon>\n </button>\n\n </ng-container>\n\n <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n", styles: [".prefix-container{width:24px;height:16px}@-webkit-keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}@keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}.loading{animation:shrink .3s ease-in-out infinite alternate;-webkit-animation:shrink .3s ease-in-out infinite alternate}.prefix-padding{padding-right:4px}.leading-icon{font-size:16px;width:16px;height:16px}.full-width{width:100%}.select{cursor:pointer;width:162px}.select-arrow{font-size:18px;width:18px;height:18px;cursor:pointer}\n"], components: [{ type: i1.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.ElderAutocompleteComponent, selector: "elder-autocomplete", inputs: ["isOptionDisabledFn", "isOptionHiddenFn", "enabled", "valueTemplate", "suggestionsDc", "displayPropertyResolver"], outputs: ["optionSelected"], exportAs: ["elderAutocomplete"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { type: i1.MatLabel, selector: "mat-label" }, { type: i1.MatPrefix, selector: "[matPrefix]" }, { type: i7.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"] }, { type: i8.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { type: i9.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i10.SelectOnTabDirective, selector: "[elderSelectOnTab]" }, { type: i9.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { type: i11.ElderAutocompleteDirective, selector: "[elderElderAutocomplete]", inputs: ["queryFilter", "filters", "sorts", "elderElderAutocomplete"] }, { type: i9.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i9.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i1.MatSuffix, selector: "[matSuffix]" }, { type: i12.ElderStopEventPropagationDirective, selector: "[elderStopEventPropagation]" }, { type: i1.MatHint, selector: "mat-hint", inputs: ["align", "id"] }], pipes: { "async": i5.AsyncPipe, "translate": i13.TranslatePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
363
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectComponent, decorators: [{
420
364
  type: Component,
421
365
  args: [{ selector: 'elder-select', changeDetection: ChangeDetectionStrategy.OnPush, providers: [
422
366
  {
@@ -424,22 +368,28 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.2", ngImpor
424
368
  useExisting: forwardRef(() => ElderSelectComponent)
425
369
  },
426
370
  ...buildFormIntegrationProviders(ElderSelectComponent)
427
- ], template: "<mat-form-field *ngIf=\"(valueWrapped$ | async) as valueWrapper\" fxFlex\n class=\"elder-std-form-field\"\n [appearance]=\"appearance\"\n [floatLabel]=\"floatLabel\"\n [color]=\"color\"\n>\n\n <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n <mat-icon *ngIf=\"icon\" disabled\n class=\"leading-icon prefix-padding noselect\"\n [class.loading]=\"state.loading\"\n [color]=\"state?.error ? 'warn' : color\">\n {{icon}}\n </mat-icon>\n\n\n <mat-icon *ngIf=\"!icon && state?.error\"\n class=\"leading-icon prefix-padding noselect\"\n color=\"warn\">\n warning\n </mat-icon>\n </ng-container>\n\n <!-- A dynamic input -->\n <input\n matInput type=\"text\" fxFlex=\"grow\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [readonly]=\"readonly || !autocomplete\"\n [name]=\"name + '-inner-input'\"\n [placeholder]=\"placeholder | translate\"\n [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n elderSelectOnTab\n [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n (blur)=\"onInputBlur($event)\"\n >\n\n <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n <elder-autocomplete\n #elderAuto=\"elderAutocomplete\"\n [suggestionsDc]=\"suggestionsDc$ | async\"\n [valueTemplate]=\"valueTemplate\"\n [enabled]=\"autocomplete\"\n [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n (optionSelected)=\"onOptionSelected($any($event))\"\n ></elder-autocomplete>\n\n\n <ng-container matSuffix>\n\n <mat-icon\n *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n (click)=\"onInputClicked(autoTrigger)\">\n arrow_drop_down\n </mat-icon>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"selectionPopup && (!valueWrapper.value || !allowNull)\"\n [disabled]=\"isLocked\"\n (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>search</mat-icon>\n </button>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"valueWrapper.value && allowNull\"\n [disabled]=\"isLocked\"\n (click)=\"clear($event)\" aria-label=\"Clear\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>close</mat-icon>\n </button>\n\n </ng-container>\n\n <mat-hint *ngIf=\"valueWrapper.hintText\">{{valueWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n", styles: [".prefix-container{width:24px;height:16px}@-webkit-keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}@keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}.loading{animation:shrink .3s ease-in-out infinite alternate;-webkit-animation:shrink .3s ease-in-out infinite alternate}.prefix-padding{padding-right:4px}.leading-icon{font-size:16px;width:16px;height:16px}.full-width{width:100%}.select{cursor:pointer;width:162px}.select-arrow{font-size:18px;width:18px;height:18px;cursor:pointer}\n"] }]
428
- }], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.Injector }]; }, propDecorators: { nullDisplay: [{
371
+ ], template: "<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n class=\"elder-std-form-field\"\n [appearance]=\"appearance\"\n [floatLabel]=\"floatLabel\"\n [color]=\"color\"\n>\n\n <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n <mat-icon *ngIf=\"icon\" disabled\n class=\"leading-icon prefix-padding noselect\"\n [class.loading]=\"state.loading\"\n [color]=\"state?.error ? 'warn' : color\">\n {{icon}}\n </mat-icon>\n\n\n <mat-icon *ngIf=\"!icon && state?.error\"\n class=\"leading-icon prefix-padding noselect\"\n color=\"warn\">\n warning\n </mat-icon>\n </ng-container>\n\n <!-- A dynamic input -->\n <input\n matInput type=\"text\" fxFlex=\"grow\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [readonly]=\"readonly || !autocomplete\"\n [name]=\"name + '-inner-input'\"\n [placeholder]=\"placeholder | translate\"\n [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n elderSelectOnTab\n [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n (blur)=\"onInputBlur($event)\"\n >\n\n <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n <elder-autocomplete\n #elderAuto=\"elderAutocomplete\"\n [suggestionsDc]=\"suggestionsDc$ | async\"\n [valueTemplate]=\"valueTemplate\"\n [enabled]=\"autocomplete\"\n [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n (optionSelected)=\"onOptionSelected($any($event))\"\n ></elder-autocomplete>\n\n\n <ng-container matSuffix>\n\n <mat-icon\n *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n (click)=\"onInputClicked(autoTrigger)\">\n arrow_drop_down\n </mat-icon>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n [disabled]=\"isLocked\"\n (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>search</mat-icon>\n </button>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"entityWrapper.value && allowNull\"\n [disabled]=\"isLocked\"\n (click)=\"clear($event)\" aria-label=\"Clear\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>close</mat-icon>\n </button>\n\n </ng-container>\n\n <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n", styles: [".prefix-container{width:24px;height:16px}@-webkit-keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}@keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}.loading{animation:shrink .3s ease-in-out infinite alternate;-webkit-animation:shrink .3s ease-in-out infinite alternate}.prefix-padding{padding-right:4px}.leading-icon{font-size:16px;width:16px;height:16px}.full-width{width:100%}.select{cursor:pointer;width:162px}.select-arrow{font-size:18px;width:18px;height:18px;cursor:pointer}\n"] }]
372
+ }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { nullDisplay: [{
429
373
  type: Input
430
- }], valueIdChange: [{
374
+ }], entityIdChange: [{
375
+ type: Output
376
+ }], entityChange: [{
377
+ type: Output
378
+ }], entityIdUpdated: [{
379
+ type: Output
380
+ }], entityUpdated: [{
431
381
  type: Output
432
382
  }], autocomplete: [{
433
383
  type: Input
434
384
  }], allowNull: [{
435
385
  type: Input
436
- }], autoSelectFirst: [{
386
+ }], entity: [{
437
387
  type: Input
438
- }], valueId: [{
388
+ }], entityId: [{
439
389
  type: Input
440
390
  }], hintProperty: [{
441
391
  type: Input
442
392
  }], hintPropertyResolver: [{
443
393
  type: Input
444
394
  }] } });
445
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component, forwardRef, InjectionToken,\n  Injector,\n  Input, NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport {buildFormIntegrationProviders} from '../../../common/forms/template-composite-control';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {BehaviorSubject, combineLatest, EMPTY, Observable, of, Subject} from 'rxjs';\nimport {\n  catchError,\n  filter,\n  map,\n  switchMap,\n  takeUntil,\n  timeout,\n  first,\n  tap,\n} from 'rxjs/operators';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {NgControl, NgModel} from '@angular/forms';\nimport {DataSourceAdapter} from '../../../common/data/datasource/data-source-adapter';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {ELDER_SELECT_BASE, ElderSelectBase, ElderSelectComponentState, TextResolverFn} from '../elder-select-base';\n\nclass ValueContext<T> {\n  constructor(\n    public readonly value: T,\n    public readonly hintText: string,\n    public readonly displayText: string\n  ) {\n  }\n}\n\n\n/**\n * The elder-select control allows to select a single\n * entity.\n *\n * Primary focus is on easy usage for common scenarios:\n *\n * - Integrated into angular forms and validation\n * - Support entity id handling, input / output is the entity id vs full value.\n *\n */\n@Component({\n  selector: 'elder-select',\n  templateUrl: './elder-select.component.html',\n  styleUrls: ['./elder-select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: ELDER_SELECT_BASE,\n      useExisting: forwardRef(() => ElderSelectComponent)\n    },\n    ...buildFormIntegrationProviders(ElderSelectComponent)\n  ]\n})\nexport class ElderSelectComponent<T = any, TId = any> extends ElderSelectBase<TId, T, T> implements OnInit, OnDestroy, AfterViewInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly hintPropertyResolver$ = new BehaviorSubject<TextResolverFn>((value: T) => null);\n\n  private _allowNull = false;\n\n  /**\n   * String to display for the 'nothing' / null value.\n   */\n  @Input()\n  public nullDisplay = '-';\n\n  private _autoSelectFirst = false;\n\n  @Output()\n  public readonly valueIdChange: Observable<TId>;\n\n  private _valueId: TId = null;\n\n  private _autocomplete = false;\n\n  public readonly valueWrapped$: Observable<ValueContext<T>>;\n\n  private readonly unsubscribe$ = new Subject();\n\n  public readonly inputText$: Observable<string>;\n\n  private ngControl: NgControl;\n  private valueContext: ValueContext<T>;\n\n  private ngModelInitialized = false;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    zone: NgZone,\n    private readonly injector: Injector\n  ) {\n    super(zone);\n\n    this.valueIdChange = this.valueChange.pipe(\n      map(v => this.getValueId(v))\n    );\n\n    this.valueWrapped$ = combineLatest(\n      [\n        this.value$,\n        this.displayPropertyResolver$,\n        this.hintPropertyResolver$\n      ]\n    ).pipe(\n      map(([value, dPR, hPR]) => new ValueContext<T>(value, hPR(value), dPR(value)))\n    );\n\n    this.inputText$ = this.valueWrapped$.pipe(\n      takeUntil(this.unsubscribe$),\n      tap(valueContext => this.valueContext = valueContext),\n      map(valueContext => {\n        if (valueContext.value) {\n          return valueContext.displayText;\n        } else {\n          return null;\n        }\n      })\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n\n    // Inject the ngControl/ngModel\n    this.ngControl = this.injector.get(NgControl, null);\n    this.ensureExistingValueNgControl(this.value);\n\n    if (this.autoSelectFirst) {\n      this.handleAutoSelectFirst();\n    }\n  }\n\n  public ngAfterViewInit(): void {\n    setTimeout(\n      () => {\n        this.ngModelInitialized = true;\n      }, 10\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.unsubscribe$.next();\n    this.unsubscribe$.complete();\n    this.autoCleanUp();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Behaviour Properties                                                    *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set autocomplete(value: boolean) {\n    this._autocomplete = coerceBooleanProperty(value);\n  }\n\n  public get autocomplete(): boolean {\n    return this._autocomplete;\n  }\n\n  /**\n   * Allows the user to select 'nothing' / null.\n   */\n  @Input()\n  public set allowNull(val: boolean) {\n    this._allowNull = coerceBooleanProperty(val);\n  }\n\n  public get allowNull(): boolean {\n    return this._allowNull;\n  }\n\n  /**\n   * After data is loaded into this select, ensure that\n   * a value is selected or select the first automatically.\n   */\n  @Input()\n  public set autoSelectFirst(val: boolean) {\n    this._autoSelectFirst = coerceBooleanProperty(val);\n  }\n\n  public get autoSelectFirst(): boolean {\n    return this._autoSelectFirst;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Select the value with the given id.\n   * @param id\n   */\n  @Input()\n  public set valueId(id: TId) {\n    if (id !== this._valueId) {\n      this._valueId = id;\n      this.selectValueById(id);\n    }\n  }\n\n  public get valueId(): TId {\n    if (this.value) {\n      return this.getValueId(this.value);\n    } else {\n      return this._valueId;\n    }\n  }\n\n  @Input()\n  public set hintProperty(property: string) {\n    this.hintPropertyResolver = (value => this.propertyStringValue(value, property));\n  }\n\n  @Input()\n  public set hintPropertyResolver(resolver: TextResolverFn) {\n    this.hintPropertyResolver$.next(resolver);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onInputBlur(event: FocusEvent): void {\n    const input = event.target as HTMLInputElement;\n    if (this.valueContext.value) {\n      input.value = this.valueContext.displayText;\n    } else {\n      input.value = null;\n    }\n  }\n\n  public get isOptionDisabledInternalFn(): (option: T) => boolean {\n    return (option: T) => {\n      if (!this.valuesEquals(option, this.value)) {\n        if (this.isOptionDisabledFn) {\n          return this.isOptionDisabledFn(option);\n        } else {\n          return false;\n        }\n      }\n      return true;\n    };\n  }\n\n  public get isOptionHiddenInternalFn(): (option: T) => boolean {\n    return (option: T) => {\n      if (this.isOptionHiddenFn) {\n        return this.isOptionHiddenFn(option);\n      } else {\n        return false;\n      }\n    };\n  }\n\n\n  public togglePanel(autoTrigger: MatAutocompleteTrigger): void {\n    if (autoTrigger.autocomplete.isOpen) {\n      autoTrigger.closePanel();\n    } else {\n      this.logger.debug('Requesting to open auto-complete panel...', autoTrigger);\n      if (this.suggestionsDc) {\n        try {\n          this.suggestionsDc.start(this.sorts, this.filters);\n        } catch (err) {\n          this.logger.error('Failed to start DC!', err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      }\n      autoTrigger.openPanel();\n    }\n  }\n\n  public onInputClicked(autoTrigger: MatAutocompleteTrigger): void {\n\n    // this.logger.debug('onInputClicked, locked: ' + this.isLocked + ', autocomplete: ' + this.autocomplete, autoTrigger);\n    if (this.isLocked || this.autocomplete) {\n      return;\n    }\n    this.togglePanel(autoTrigger);\n  }\n\n  public onOptionSelected(selectedValue: T): void {\n    if (this.valuesEquals(selectedValue, this.value)) {\n      this.writeValueInternal(null); // HACK: Ensure we trigger view redraw\n    }\n    this.updateValue(selectedValue);\n  }\n\n  public openSelectionPopup(event: Event): void {\n    if (this.selectionPopup) {\n      this.selectionPopup.choose(\n        this.filters,\n        this.sorts\n      ).subscribe(\n        selection => {\n          this.updateValue(selection[0]);\n        }\n      );\n    } else {\n      this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');\n    }\n  }\n\n  public clear(event: Event): void {\n    this.updateValue(null);\n  }\n\n  public forceReloadFirst() {\n    this.value = null; // valueId is set to null automatically\n    this.handleAutoSelectFirst();\n  }\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods (Auto Select First)                                     *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleAutoSelectFirst(): void {\n    if (!this._valueId) {\n      this.logger.debug('Attempting to auto select first entity...');\n      this.trySelectFirst();\n    } else {\n      this.logger.warn('Conflict avoided: Aborted auto selection because value id already present!');\n    }\n  }\n\n  private trySelectFirst(): void {\n\n    const dataContextFirstItem = this.suggestionsDc?.isStarted\n      ? this.firstItemInDataContextSoon(this.suggestionsDc)\n      : of(null);\n\n    // TODO takeUntil value-id is set, after that we can cancel.\n\n    dataContextFirstItem.pipe(\n      takeUntil(this.valueIdChange.pipe(filter(id => !!id))),\n      switchMap(firstItem => firstItem ? of(firstItem) : this.firstSuggestionItem())\n    ).subscribe(\n      firstItem => {\n        if (firstItem && !this._valueId) {\n          this.updateValue(firstItem);\n        }\n      },\n      err => {\n        this.updateState(ElderSelectComponentState.error(err));\n        this.logger.error('Failed to load first entity for auto-select-first flow!', err);\n      }\n    );\n  }\n\n  private firstItemInDataContextSoon(dataContext: IDataContext<T>): Observable<T | null> {\n    return dataContext.data.pipe(\n      filter(data => data && data.length > 0),\n      map(data => data[0]),\n      timeout(250),\n      catchError(timeoutError => {\n        this.logger.warn('Gave up waiting for DataContext items, timeout reached. Falling back to sugestion lookup.', timeoutError);\n        return of(null);\n      }),\n      first()\n    );\n  }\n\n  private firstSuggestionItem(): Observable<T | null> {\n\n    const dataSource = this.suggestionsDc?.dataSource;\n\n    if (dataSource) {\n\n      this.updateState(ElderSelectComponentState.loading());\n\n      return DataSourceAdapter.from(dataSource)\n        .findFirst(this.filters, this.sorts).pipe(\n          tap(() => this.updateState(ElderSelectComponentState.idle())),\n          catchError(err => {\n            this.logger.error('Failed to load suggestions!', err);\n            this.updateState(ElderSelectComponentState.error(err));\n            return of(null);\n          }),\n        );\n    } else {\n      this.logger.warn('Failed to load first suggestion!');\n      this.updateState(ElderSelectComponentState.idle());\n      return EMPTY;\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Abstract Implementation                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Occurs when the suggestions data-context has changed\n   */\n  protected onSuggestionsDcChanged(data: IDataContext<T>): void {\n    if (this._valueId) {\n      this.selectValueById(this._valueId);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  protected valuesEquals(a: T, b: T): boolean {\n    return this.isEqual(a, b);\n  }\n\n  private selectValueById(id: TId): void {\n\n    if (this.value && this.getValueId(this.value) === id) {\n      return; // Already selected\n    }\n\n    if (id === null || id === undefined) {\n      if (this.value !== null && this.value !== undefined) {\n        this.writeValueInternal(null);\n      }\n    } else {\n      if (this.suggestionsDc) {\n        const existing = this.findInDataContext(id);\n        if (existing) {\n          this.writeValueInternal(existing);\n        } else {\n          this.loadEntityById(id);\n        }\n      } else {\n        this.logger.warn('Failed to select value by Id: ' + id + ' - DataContext not available.');\n      }\n    }\n  }\n\n  private loadEntityById(id: TId): void {\n    this.suggestionsDc.dataSource.findById(<any>id)\n      .subscribe(\n        entity => {\n          this.logger.debug('Loaded missing entity by id ' + id + ':', entity);\n          this.writeValueInternalDelayed(entity);\n        },\n        err => {\n          this.logger.error('Failed to load entity by id ' + id, err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      );\n  }\n\n  private writeValueInternalDelayed(value: T): void {\n    setTimeout( // HACK: Counter ngModel race condition\n      () => this.writeValueInternal(value),\n      10\n    );\n  }\n\n  private findInDataContext(id: TId): T | null {\n    return this.suggestionsDc.snapshot.data\n      .find(v => this.getValueId(v) === id);\n  }\n\n  protected writeToControl(value: T): void {\n    this._valueId = this.getValueId(value);\n    this.ensureExistingValueNgControl(value);\n    super.writeToControl(value);\n  }\n\n  private ensureExistingValueNgControl(value: T): void {\n    if (value) {\n      // Patch ngModel value (especially useful when using ngModel & value-id binding)\n      this.tryUpdateNgControl(value);\n    }\n  }\n\n  private tryUpdateNgControl(value: T): void {\n    if (this.ngControl) {\n      const ngModel = this.ngControl as NgModel;\n      if (ngModel) {\n        ngModel.model = value;\n        this.logger.debug('Wrote value directly into ngModel:', value);\n      }\n    }\n  }\n\n  /**\n   * @description\n   * Writes a new value to the element.\n   *\n   * This method is called by the forms API to write to the view when programmatic\n   * changes from model to view are requested.\n   *\n   * @param value The new value for the element\n   */\n  public writeValue(value: T): void {\n    // this.logger.debug('ElderSelectBase.writeValue(): Current value-id ' + this._valueId + ', new value: ', value);\n    if (this.isNgModelInitializing() && this.isNullOrEmpty(value)) {\n      this.logger.debug('Prevented null/empty write while ngModel is still initializing!');\n    } else {\n      super.writeValue(value);\n    }\n  }\n\n  private isNullOrEmpty(value: T): boolean {\n    return (value == null || (<any>value) === '');\n  }\n\n  private isNgModelInitializing(): boolean {\n    if (this.ngControl) {\n      return !this.ngModelInitialized;\n    }\n    return false;\n  }\n\n}\n","<mat-form-field *ngIf=\"(valueWrapped$ | async) as valueWrapper\" fxFlex\n                class=\"elder-std-form-field\"\n                [appearance]=\"appearance\"\n                [floatLabel]=\"floatLabel\"\n                [color]=\"color\"\n>\n\n  <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n  <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n    <mat-icon *ngIf=\"icon\" disabled\n              class=\"leading-icon prefix-padding noselect\"\n              [class.loading]=\"state.loading\"\n              [color]=\"state?.error ? 'warn' : color\">\n      {{icon}}\n    </mat-icon>\n\n\n    <mat-icon *ngIf=\"!icon && state?.error\"\n              class=\"leading-icon prefix-padding noselect\"\n              color=\"warn\">\n      warning\n    </mat-icon>\n  </ng-container>\n\n  <!-- A dynamic input -->\n  <input\n    matInput type=\"text\" fxFlex=\"grow\"\n    [disabled]=\"disabled\"\n    [required]=\"required\"\n    [readonly]=\"readonly || !autocomplete\"\n    [name]=\"name + '-inner-input'\"\n    [placeholder]=\"placeholder | translate\"\n    [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n    [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n    elderSelectOnTab\n    [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n    [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n    (blur)=\"onInputBlur($event)\"\n  >\n\n  <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n  <elder-autocomplete\n    #elderAuto=\"elderAutocomplete\"\n    [suggestionsDc]=\"suggestionsDc$ | async\"\n    [valueTemplate]=\"valueTemplate\"\n    [enabled]=\"autocomplete\"\n    [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n    [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n    [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n    (optionSelected)=\"onOptionSelected($any($event))\"\n  ></elder-autocomplete>\n\n\n  <ng-container matSuffix>\n\n    <mat-icon\n      *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n      (click)=\"onInputClicked(autoTrigger)\">\n      arrow_drop_down\n    </mat-icon>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"selectionPopup && (!valueWrapper.value || !allowNull)\"\n            [disabled]=\"isLocked\"\n            (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>search</mat-icon>\n    </button>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"valueWrapper.value && allowNull\"\n            [disabled]=\"isLocked\"\n            (click)=\"clear($event)\" aria-label=\"Clear\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>close</mat-icon>\n    </button>\n\n  </ng-container>\n\n  <mat-hint *ngIf=\"valueWrapper.hintText\">{{valueWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n"]}
395
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EAAE,UAAU,EACrB,KAAK,EAGL,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAC,6BAA6B,EAAC,MAAM,kDAAkD,CAAC;AAC/F,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AAEnD,OAAO,EAAC,eAAe,EAAE,aAAa,EAAc,OAAO,EAAC,MAAM,MAAM,CAAC;AACzE,OAAO,EACL,GAAG,EACH,SAAS,EACT,GAAG,EAAE,IAAI,EAAE,MAAM,GAClB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAC,qBAAqB,EAAC,MAAM,uBAAuB,CAAC;AAE5D,OAAO,EAAC,iBAAiB,EAAE,eAAe,EAAE,yBAAyB,EAAiB,MAAM,sBAAsB,CAAC;AACnH,OAAO,EAAC,wBAAwB,EAAC,MAAM,gCAAgC,CAAC;;;;;;;;;;;;;;;AAGxE,MAAM,aAAa;IACjB,YACkB,KAAQ,EACR,QAAgB,EAChB,WAAmB;QAFnB,UAAK,GAAL,KAAK,CAAG;QACR,aAAQ,GAAR,QAAQ,CAAQ;QAChB,gBAAW,GAAX,WAAW,CAAQ;IAErC,CAAC;CACF;AAGD;;;;;;;;;GASG;AAcH,MAAM,OAAO,oBACX,SAAQ,eAAqC;IAqD7C;;;;gFAI4E;IAE5E,YACE,IAAY;QAEZ,KAAK,CAAC,IAAI,CAAC,CAAC;QA3Dd;;;;oFAI4E;QAE3D,WAAM,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAExD,0BAAqB,GAAG,IAAI,eAAe,CAAiB,CAAC,KAAc,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC;QAE/F,eAAU,GAAG,KAAK,CAAC;QAE3B;;WAEG;QAEI,gBAAW,GAAG,GAAG,CAAC;QAsBR,YAAO,GAAG,IAAI,eAAe,CAAU,SAAS,CAAC,CAAC;QAE3D,kBAAa,GAAG,KAAK,CAAC;QAIb,iBAAY,GAAG,IAAI,OAAO,EAAE,CAAC;QAiB5C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CACzC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CACpC,CAAC;QAEF,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CACnC,IAAI,CAAC,CAAC,CAAC,CAAC,8BAA8B;SACvC,CAAC;QAEF,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAC3C,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,CAC5C,CAAC;QAEF,IAAI,CAAC,aAAa,GAAG,aAAa,CAChC,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,OAAO,CAAC,CACrC,CAAC,IAAI,CACH,MAAM,CAAC,CAAC,CAAC,SAAS,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,KAAK,SAAS,CAAC,EACtE,GAAG,CAAC,CAAC,CAAC,SAAS,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,CACvC,CAAC;QAEF,IAAI,CAAC,cAAc,GAAG,aAAa,CACjC;YACE,IAAI,CAAC,OAAO;YACZ,IAAI,CAAC,wBAAwB;YAC7B,IAAI,CAAC,qBAAqB;SAC3B,CACF,CAAC,IAAI,CACJ,GAAG,CAAC,CAAC,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,IAAI,aAAa,CAAU,KAAK,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CACtF,CAAC;QAEF,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CACxC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,EAC5B,GAAG,CAAC,YAAY,CAAC,EAAE,CAAC,IAAI,CAAC,aAAa,GAAG,YAAY,CAAC,EACtD,GAAG,CAAC,YAAY,CAAC,EAAE;YACjB,IAAI,YAAY,CAAC,KAAK,EAAE;gBACtB,OAAO,YAAY,CAAC,WAAW,CAAC;aACjC;iBAAM;gBACL,OAAO,IAAI,CAAC;aACb;QACH,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED;;;;gFAI4E;IAErE,QAAQ;IAEf,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;QACzB,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC;QAC7B,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAED;;;;gFAI4E;IAE5E,IACW,YAAY,CAAC,KAAc;QACpC,IAAI,CAAC,aAAa,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;IACpD,CAAC;IAED,IAAW,YAAY;QACrB,OAAO,IAAI,CAAC,aAAa,CAAC;IAC5B,CAAC;IAED;;OAEG;IACH,IACW,SAAS,CAAC,GAAY;QAC/B,IAAI,CAAC,UAAU,GAAG,qBAAqB,CAAC,GAAG,CAAC,CAAC;IAC/C,CAAC;IAED,IAAW,SAAS;QAClB,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED;;;;gFAI4E;IAE5E,IACW,MAAM,CAAC,MAAe;QAC/B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC1B,IAAI,CAAC,kBAAkB,CACrB,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAC3B,CAAC;IACJ,CAAC;IAED,IAAW,MAAM;QACf,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;IACjC,CAAC;IAED,IACW,QAAQ,CAAC,EAAO;QACzB,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,kBAAkB,CAAM,EAAY,CAAC,CAAC;SAC5C;aAAM;YACL,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC;SAC3B;IACH,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC5C,CAAC;IAED,IACW,YAAY,CAAC,QAAgB;QACtC,IAAI,CAAC,oBAAoB,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC,CAAC;IACnF,CAAC;IAED,IACW,oBAAoB,CAAC,QAAwB;QACtD,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC5C,CAAC;IAED;;;;gFAI4E;IAErE,WAAW,CAAC,KAAiB;QAClC,MAAM,KAAK,GAAG,KAAK,CAAC,MAA0B,CAAC;QAC/C,IAAI,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE;YAC5B,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC;SAC9C;aAAM;YACL,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC;SACpB;IACH,CAAC;IAED,IAAW,0BAA0B;QACnC,OAAO,CAAC,MAAe,EAAE,EAAE;YACzB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,EAAE;gBAClC,IAAI,IAAI,CAAC,kBAAkB,EAAE;oBAC3B,OAAO,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAC;iBACxC;qBAAM;oBACL,OAAO,KAAK,CAAC;iBACd;aACF;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAC;IACJ,CAAC;IAED,IAAW,wBAAwB;QACjC,OAAO,CAAC,MAAe,EAAE,EAAE;YACzB,IAAI,IAAI,CAAC,gBAAgB,EAAE;gBACzB,OAAO,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC;aACtC;iBAAM;gBACL,OAAO,KAAK,CAAC;aACd;QACH,CAAC,CAAC;IACJ,CAAC;IAGM,WAAW,CAAC,WAAmC;QACpD,IAAI,WAAW,CAAC,YAAY,CAAC,MAAM,EAAE;YACnC,WAAW,CAAC,UAAU,EAAE,CAAC;SAC1B;aAAM;YACL,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,2CAA2C,EAAE,WAAW,CAAC,CAAC;YAC5E,IAAI,IAAI,CAAC,aAAa,EAAE;gBACtB,IAAI;oBACF,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;iBACpD;gBAAC,OAAO,GAAG,EAAE;oBACZ,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,qBAAqB,EAAE,GAAG,CAAC,CAAC;oBAC9C,IAAI,CAAC,WAAW,CAAC,yBAAyB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;iBACxD;aACF;YACD,WAAW,CAAC,SAAS,EAAE,CAAC;SACzB;IACH,CAAC;IAEM,cAAc,CAAC,WAAmC;QAEvD,uHAAuH;QACvH,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,YAAY,EAAE;YACtC,OAAO;SACR;QACD,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;IAChC,CAAC;IAEM,gBAAgB,CAAC,aAAsB;QAC5C,IAAI,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC,EAAE;YACxC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC,CAAC,sCAAsC;SACtE;QACD,IAAI,CAAC,mBAAmB,CAAC,aAAa,CAAC,CAAC;IAC1C,CAAC;IAEM,kBAAkB,CAAC,KAAY;QACpC,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,IAAI,CAAC,cAAc,CAAC,MAAM,CACxB,IAAI,CAAC,OAAO,EACZ,IAAI,CAAC,KAAK,CACX,CAAC,SAAS,CACT,CAAC,SAAoB,EAAE,EAAE;gBACvB,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;YACzC,CAAC,CACF,CAAC;SACH;aAAM;YACL,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,wFAAwF,CAAC,CAAC;SAC5G;IACH,CAAC;IAEM,KAAK,CAAC,KAAY;QACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IAEM,gBAAgB;QACrB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,uCAAuC;QAC1D,MAAM,eAAe,GAAG,IAAI,wBAAwB,CAAC,IAAI,CAAC,CAAC;QAC3D,eAAe,CAAC,cAAc,EAAE,CAAC;IACnC,CAAC;IAEM,mBAAmB,CAAC,MAAe;QACxC,IAAI,CAAC,WAAW,CACd,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAC3B,CAAC;IACJ,CAAC;IAED;;;;gFAI4E;IAE5E;;OAEG;IACO,sBAAsB,CAAC,IAA2B;QAC1D,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SACtC;IACH,CAAC;IAED;;;;gFAI4E;IAElE,YAAY,CAAC,CAAS,EAAE,CAAS;QACzC,OAAO,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC;IACjE,CAAC;IAEO,gBAAgB,CAAC,MAAe;QACtC,OAAO,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,KAAK,IAAI,CAAC,QAAQ,CAAC;IACpD,CAAC;IAEO,iBAAiB,CAAC,KAAa;QACrC,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,OAAY,KAAY,CAAC;SAC1B;aAAM;YACL,MAAM,MAAM,GAAS,KAAiB,CAAC;YACvC,OAAO,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;SACjC;IACH,CAAC;IAEO,gBAAgB,CAAC,EAAO;QAE9B,MAAM,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC;QAClC,IAAI,aAAa,IAAI,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,KAAK,EAAE,EAAE;YAC3D,OAAO,CAAC,wBAAwB;SACjC;QAED,IAAI,EAAE,KAAK,IAAI,IAAI,EAAE,KAAK,SAAS,EAAE;YACnC,IAAI,aAAa,KAAK,IAAI,IAAI,aAAa,KAAK,SAAS,EAAE;gBACzD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;aACpB;SACF;aAAM;YACL,IAAI,IAAI,CAAC,aAAa,EAAE;gBACtB,MAAM,QAAQ,GAAG,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC;gBAC5C,IAAI,QAAQ,EAAE;oBACZ,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAC;iBACxB;qBAAM;oBACL,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC;iBACzB;aACF;iBAAM;gBACL,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gCAAgC,GAAG,EAAE,GAAG,+BAA+B,CAAC,CAAC;aAC3F;SACF;IACH,CAAC;IAEO,cAAc,CAAC,EAAO;QAC5B,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,QAAQ,CAAM,EAAE,CAAC;aAC5C,SAAS,CACR,MAAM,CAAC,EAAE;YACP,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,8BAA8B,GAAG,EAAE,GAAG,GAAG,EAAE,MAAM,CAAC,CAAC;YACrE,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACvB,CAAC,EACD,GAAG,CAAC,EAAE;YACJ,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,8BAA8B,GAAG,EAAE,EAAE,GAAG,CAAC,CAAC;YAC5D,IAAI,CAAC,WAAW,CAAC,yBAAyB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;QACzD,CAAC,CACF,CAAC;IACN,CAAC;IAEO,iBAAiB,CAAC,EAAO;QAC/B,OAAO,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI;aACpC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;IAC3C,CAAC;IAEO,aAAa,CAAC,MAAe;QACnC,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YACpC,OAAY,EAAY,CAAC;SAC1B;aAAM;YACL,OAAY,MAAgB,CAAC;SAC9B;IACH,CAAC;IAED;;;OAGG;IACO,cAAc,CAAC,KAAa;QACpC,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,MAAM,QAAQ,GAAQ,KAAY,CAAC;YACnC,gEAAgE;YAChE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,2CAA2C,GAAG,QAAQ,EAAE,KAAK,CAAC,CAAC;YACjF,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;SACjC;aAAM;YACL,wDAAwD;YACxD,MAAM,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC;YAClC,MAAM,SAAS,GAAQ,KAAgB,CAAC;YACxC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,+CAA+C,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,EAAE,KAAK,CAAC,CAAC;YACtG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,SAAS,CAAC,EAAE;gBAC3C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;aAC9B;iBAAM;gBACL,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,yDAAyD,CAAC,CAAC;aAC7E;SACF;QACD,KAAK,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC;IAED;;;;;;;;OAQG;IACI,UAAU,CAAC,KAAa;QAC7B,iHAAiH;QACjH,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;IAC1B,CAAC;IAEO,aAAa,CAAC,KAAa;QACjC,OAAO,CAAC,KAAK,IAAI,IAAI,IAAU,KAAM,KAAK,EAAE,CAAC,CAAC;IAChD,CAAC;;iHAzaU,oBAAoB;qGAApB,oBAAoB,0YARpB;QACT;YACE,OAAO,EAAE,iBAAiB;YAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,oBAAoB,CAAC;SACpD;QACD,GAAG,6BAA6B,CAAC,oBAAoB,CAAC;KACvD,iDCtDH,2/FA6FA;2FDrCa,oBAAoB;kBAbhC,SAAS;+BACE,cAAc,mBAGP,uBAAuB,CAAC,MAAM,aACpC;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,qBAAqB,CAAC;yBACpD;wBACD,GAAG,6BAA6B,sBAAsB;qBACvD;6FAsBM,WAAW;sBADjB,KAAK;gBAIU,cAAc;sBAD7B,MAAM;gBAIS,YAAY;sBAD3B,MAAM;gBAQS,eAAe;sBAD9B,MAAM;gBAQS,aAAa;sBAD5B,MAAM;gBA2FI,YAAY;sBADtB,KAAK;gBAaK,SAAS;sBADnB,KAAK;gBAgBK,MAAM;sBADhB,KAAK;gBAaK,QAAQ;sBADlB,KAAK;gBAcK,YAAY;sBADtB,KAAK;gBAMK,oBAAoB;sBAD9B,KAAK","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component, forwardRef,\n  Input, NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport {buildFormIntegrationProviders} from '../../../common/forms/template-composite-control';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {\n  map,\n  takeUntil,\n  tap, skip, filter,\n} from 'rxjs/operators';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {ELDER_SELECT_BASE, ElderSelectBase, ElderSelectComponentState, TextResolverFn} from '../elder-select-base';\nimport {AutoSelectFirstDirective} from '../auto-select-first.directive';\nimport {IElderEntityValueAccessor} from '../elder-entity-value-accessor';\n\nclass EntityContext<T> {\n  constructor(\n    public readonly value: T,\n    public readonly hintText: string,\n    public readonly displayText: string\n  ) {\n  }\n}\n\n\n/**\n * The elder-select control allows to select a single\n * entity.\n *\n * Primary focus is on easy usage for common scenarios:\n *\n * - Integrated into angular forms and validation\n * - Support entity id handling, input / output is the entity id vs full value.\n *\n */\n@Component({\n  selector: 'elder-select',\n  templateUrl: './elder-select.component.html',\n  styleUrls: ['./elder-select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: ELDER_SELECT_BASE,\n      useExisting: forwardRef(() => ElderSelectComponent)\n    },\n    ...buildFormIntegrationProviders(ElderSelectComponent)\n  ]\n})\nexport class ElderSelectComponent<TEntity = any, TId = any, TValue = TEntity | TId>\n  extends ElderSelectBase<TId, TEntity, TValue>\n  implements IElderEntityValueAccessor<TEntity, TId>, OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly hintPropertyResolver$ = new BehaviorSubject<TextResolverFn>((value: TEntity) => null);\n\n  private _allowNull = false;\n\n  /**\n   * String to display for the 'nothing' / null value.\n   */\n  @Input()\n  public nullDisplay = '-';\n\n  @Output()\n  public readonly entityIdChange: Observable<TId>;\n\n  @Output()\n  public readonly entityChange: Observable<TEntity>;\n\n  /**\n   * Similar to entity-id change, but emits only when the user\n   * has updated the value.\n   */\n  @Output()\n  public readonly entityIdUpdated: Observable<TId>;\n\n  /**\n   * Similar to entity change, but emits only when the user\n   * has updated the value.\n   */\n  @Output()\n  public readonly entityUpdated: Observable<TEntity>;\n\n  private readonly entity$ = new BehaviorSubject<TEntity>(undefined);\n\n  private _autocomplete = false;\n\n  public readonly entityWrapped$: Observable<EntityContext<TEntity>>;\n\n  private readonly unsubscribe$ = new Subject();\n\n  public readonly inputText$: Observable<string>;\n\n  private entityContext: EntityContext<TEntity>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    zone: NgZone,\n  ) {\n    super(zone);\n\n    this.entityIdChange = this.valueChange.pipe(\n      map(v => this.entityIdFromValue(v))\n    );\n\n    this.entityChange = this.entity$.pipe(\n      skip(1) // Skip the initial NULL value\n    );\n\n    this.entityIdUpdated = this.valueUpdated.pipe(\n      map(value => this.entityIdFromValue(value))\n    );\n\n    this.entityUpdated = combineLatest(\n      [this.entityIdUpdated, this.entity$]\n    ).pipe(\n       filter(([updatedId, entity]) => this.getEntityId(entity) === updatedId),\n        map(([updatedId, entity]) => entity)\n    );\n\n    this.entityWrapped$ = combineLatest(\n      [\n        this.entity$,\n        this.displayPropertyResolver$,\n        this.hintPropertyResolver$\n      ]\n    ).pipe(\n      map(([value, dPR, hPR]) => new EntityContext<TEntity>(value, hPR(value), dPR(value)))\n    );\n\n    this.inputText$ = this.entityWrapped$.pipe(\n      takeUntil(this.unsubscribe$),\n      tap(valueContext => this.entityContext = valueContext),\n      map(valueContext => {\n        if (valueContext.value) {\n          return valueContext.displayText;\n        } else {\n          return null;\n        }\n      })\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n\n  }\n\n  public ngOnDestroy(): void {\n    this.unsubscribe$.next();\n    this.unsubscribe$.complete();\n    this.autoCleanUp();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Behaviour Properties                                                    *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set autocomplete(value: boolean) {\n    this._autocomplete = coerceBooleanProperty(value);\n  }\n\n  public get autocomplete(): boolean {\n    return this._autocomplete;\n  }\n\n  /**\n   * Allows the user to select 'nothing' / null.\n   */\n  @Input()\n  public set allowNull(val: boolean) {\n    this._allowNull = coerceBooleanProperty(val);\n  }\n\n  public get allowNull(): boolean {\n    return this._allowNull;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set entity(entity: TEntity) {\n    this.entity$.next(entity);\n    this.writeValueInternal(\n      this.entityToValue(entity)\n    );\n  }\n\n  public get entity(): TEntity {\n    return this.entity$.getValue();\n  }\n\n  @Input()\n  public set entityId(id: TId) {\n    if (this.valueAsId) {\n      this.writeValueInternal(<any>id as TValue);\n    } else {\n      this.selectEntityById(id);\n    }\n  }\n\n  public get entityId(): TId {\n    return this.entityIdFromValue(this.value);\n  }\n\n  @Input()\n  public set hintProperty(property: string) {\n    this.hintPropertyResolver = (value => this.propertyStringValue(value, property));\n  }\n\n  @Input()\n  public set hintPropertyResolver(resolver: TextResolverFn) {\n    this.hintPropertyResolver$.next(resolver);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onInputBlur(event: FocusEvent): void {\n    const input = event.target as HTMLInputElement;\n    if (this.entityContext.value) {\n      input.value = this.entityContext.displayText;\n    } else {\n      input.value = null;\n    }\n  }\n\n  public get isOptionDisabledInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (!this.isEntitySelected(option)) {\n        if (this.isOptionDisabledFn) {\n          return this.isOptionDisabledFn(option);\n        } else {\n          return false;\n        }\n      }\n      return true;\n    };\n  }\n\n  public get isOptionHiddenInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (this.isOptionHiddenFn) {\n        return this.isOptionHiddenFn(option);\n      } else {\n        return false;\n      }\n    };\n  }\n\n\n  public togglePanel(autoTrigger: MatAutocompleteTrigger): void {\n    if (autoTrigger.autocomplete.isOpen) {\n      autoTrigger.closePanel();\n    } else {\n      this.logger.debug('Requesting to open auto-complete panel...', autoTrigger);\n      if (this.suggestionsDc) {\n        try {\n          this.suggestionsDc.start(this.sorts, this.filters);\n        } catch (err) {\n          this.logger.error('Failed to start DC!', err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      }\n      autoTrigger.openPanel();\n    }\n  }\n\n  public onInputClicked(autoTrigger: MatAutocompleteTrigger): void {\n\n    // this.logger.debug('onInputClicked, locked: ' + this.isLocked + ', autocomplete: ' + this.autocomplete, autoTrigger);\n    if (this.isLocked || this.autocomplete) {\n      return;\n    }\n    this.togglePanel(autoTrigger);\n  }\n\n  public onOptionSelected(selectedValue: TEntity): void {\n    if (this.isEntitySelected(selectedValue)) {\n      this.writeValueInternal(null); // HACK: Ensure we trigger view redraw\n    }\n    this.updateValueByEntity(selectedValue);\n  }\n\n  public openSelectionPopup(event: Event): void {\n    if (this.selectionPopup) {\n      this.selectionPopup.choose(\n        this.filters,\n        this.sorts\n      ).subscribe(\n        (selection: TEntity[]) => {\n          this.updateValueByEntity(selection[0]);\n        }\n      );\n    } else {\n      this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');\n    }\n  }\n\n  public clear(event: Event): void {\n    this.updateValue(null);\n  }\n\n  public forceReloadFirst() {\n    this.value = null; // valueId is set to null automatically\n    const autoReloadFirst = new AutoSelectFirstDirective(this);\n    autoReloadFirst.trySelectFirst();\n  }\n\n  public updateValueByEntity(entity: TEntity): void {\n    this.updateValue(\n      this.entityToValue(entity)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Abstract Implementation                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Occurs when the suggestions data-context has changed\n   */\n  protected onSuggestionsDcChanged(data: IDataContext<TEntity>): void {\n    if (this.valueAsId) {\n      this.selectEntityById(this.entityId);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  protected valuesEquals(a: TValue, b: TValue): boolean {\n    return this.entityIdFromValue(a) === this.entityIdFromValue(b);\n  }\n\n  private isEntitySelected(entity: TEntity): boolean {\n    return this.getEntityId(entity) === this.entityId;\n  }\n\n  private entityIdFromValue(value: TValue): TId {\n    if (this.valueAsId) {\n      return <any>value as TId;\n    } else {\n      const entity = (<any>value) as TEntity;\n      return this.getEntityId(entity);\n    }\n  }\n\n  private selectEntityById(id: TId): void {\n\n    const currentEntity = this.entity;\n    if (currentEntity && this.getEntityId(currentEntity) === id) {\n      return; // Entity already loaded\n    }\n\n    if (id === null || id === undefined) {\n      if (currentEntity !== null && currentEntity !== undefined) {\n        this.entity = null;\n      }\n    } else {\n      if (this.suggestionsDc) {\n        const existing = this.findInDataContext(id);\n        if (existing) {\n          this.entity = existing;\n        } else {\n          this.loadEntityById(id);\n        }\n      } else {\n        this.logger.warn('Failed to select value by Id: ' + id + ' - DataContext not available.');\n      }\n    }\n  }\n\n  private loadEntityById(id: TId): void {\n    this.suggestionsDc.dataSource.findById(<any>id)\n      .subscribe(\n        entity => {\n          this.logger.debug('Loaded missing entity by id ' + id + ':', entity);\n          this.entity = entity;\n        },\n        err => {\n          this.logger.error('Failed to load entity by id ' + id, err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      );\n  }\n\n  private findInDataContext(id: TId): TEntity | null {\n    return this.suggestionsDc.snapshot.data\n      .find(v => this.getEntityId(v) === id);\n  }\n\n  private entityToValue(entity: TEntity): TValue {\n    if (this.valueAsId) {\n      const id = this.getEntityId(entity);\n      return <any>id as TValue;\n    } else {\n      return <any>entity as TValue;\n    }\n  }\n\n  /**\n   * This method is invoked after a value has been written to this control.\n   *\n   */\n  protected writeToControl(value: TValue): void {\n    if (this.valueAsId) {\n      const entityId = <any>value as TId;\n      // Value was written as entity id, ensure we select entity by id\n      this.logger.debug('writeToControl: value was written as id: ' + entityId, value);\n      this.selectEntityById(entityId);\n    } else {\n      // Value was written as entity, ensure entity is updated\n      const currentEntity = this.entity;\n      const newEntity = <any>value as TEntity;\n      this.logger.debug('writeToControl: value was written as entity: ' + JSON.stringify(newEntity), value);\n      if (!this.isEqual(currentEntity, newEntity)) {\n        this.entity$.next(newEntity);\n      } else {\n        this.logger.warn('Ignored written entity as it is already set to entity$!');\n      }\n    }\n    super.writeToControl(value);\n  }\n\n  /**\n   * @description\n   * Writes a new value to the element.\n   *\n   * This method is called by the forms API to write to the view when programmatic\n   * changes from model to view are requested.\n   *\n   * @param value The new value for the element\n   */\n  public writeValue(value: TValue): void {\n    // this.logger.debug('ElderSelectBase.writeValue(): Current value-id ' + this._valueId + ', new value: ', value);\n    super.writeValue(value);\n  }\n\n  private isNullOrEmpty(value: TValue): boolean {\n    return (value == null || (<any>value) === '');\n  }\n\n}\n","<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n                class=\"elder-std-form-field\"\n                [appearance]=\"appearance\"\n                [floatLabel]=\"floatLabel\"\n                [color]=\"color\"\n>\n\n  <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n  <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n    <mat-icon *ngIf=\"icon\" disabled\n              class=\"leading-icon prefix-padding noselect\"\n              [class.loading]=\"state.loading\"\n              [color]=\"state?.error ? 'warn' : color\">\n      {{icon}}\n    </mat-icon>\n\n\n    <mat-icon *ngIf=\"!icon && state?.error\"\n              class=\"leading-icon prefix-padding noselect\"\n              color=\"warn\">\n      warning\n    </mat-icon>\n  </ng-container>\n\n  <!-- A dynamic input -->\n  <input\n    matInput type=\"text\" fxFlex=\"grow\"\n    [disabled]=\"disabled\"\n    [required]=\"required\"\n    [readonly]=\"readonly || !autocomplete\"\n    [name]=\"name + '-inner-input'\"\n    [placeholder]=\"placeholder | translate\"\n    [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n    [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n    elderSelectOnTab\n    [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n    [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n    (blur)=\"onInputBlur($event)\"\n  >\n\n  <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n  <elder-autocomplete\n    #elderAuto=\"elderAutocomplete\"\n    [suggestionsDc]=\"suggestionsDc$ | async\"\n    [valueTemplate]=\"valueTemplate\"\n    [enabled]=\"autocomplete\"\n    [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n    [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n    [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n    (optionSelected)=\"onOptionSelected($any($event))\"\n  ></elder-autocomplete>\n\n\n  <ng-container matSuffix>\n\n    <mat-icon\n      *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n      (click)=\"onInputClicked(autoTrigger)\">\n      arrow_drop_down\n    </mat-icon>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n            [disabled]=\"isLocked\"\n            (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>search</mat-icon>\n    </button>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"entityWrapper.value && allowNull\"\n            [disabled]=\"isLocked\"\n            (click)=\"clear($event)\" aria-label=\"Clear\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>close</mat-icon>\n    </button>\n\n  </ng-container>\n\n  <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n"]}