@scion/workbench 18.0.0-beta.1 → 18.0.0-beta.3

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 (111) hide show
  1. package/_index.scss +3 -3
  2. package/design/_workbench-icon-font.scss +1 -1
  3. package/esm2022/lib/common/objects.util.mjs +7 -1
  4. package/esm2022/lib/common/uid.util.mjs +22 -0
  5. package/esm2022/lib/content-projection/content-projection.directive.mjs +16 -17
  6. package/esm2022/lib/dialog//311/265workbench-dialog.mjs +3 -3
  7. package/esm2022/lib/filter-field/filter-field.component.mjs +5 -5
  8. package/esm2022/lib/layout/grid-element/grid-element.component.mjs +6 -16
  9. package/esm2022/lib/layout/migration/model/workbench-layout-migration-v5.model.mjs +11 -0
  10. package/esm2022/lib/layout/migration/workbench-layout-migration-v3.service.mjs +2 -2
  11. package/esm2022/lib/layout/migration/workbench-layout-migration-v5.service.mjs +67 -0
  12. package/esm2022/lib/layout/stringifier.mjs +70 -0
  13. package/esm2022/lib/layout/workbench-layout.model.mjs +1 -1
  14. package/esm2022/lib/layout/workench-layout-serializer.service.mjs +16 -25
  15. package/esm2022/lib/layout//311/265workbench-layout.mjs +14 -18
  16. package/esm2022/lib/microfrontend-platform/common/microfrontend.util.mjs +18 -1
  17. package/esm2022/lib/microfrontend-platform/initialization/microfrontend-platform-initializer.service.mjs +28 -21
  18. package/esm2022/lib/microfrontend-platform/initialization/workbench-host-manifest-interceptor.service.mjs +11 -1
  19. package/esm2022/lib/microfrontend-platform/manifest-object-cache.service.mjs +63 -0
  20. package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog-capability-validator.interceptor.mjs +5 -4
  21. package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog.component.mjs +2 -2
  22. package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/text-message/text-message.component.mjs +3 -3
  23. package/esm2022/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.mjs +3 -2
  24. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-capability-validator.interceptor.mjs +3 -2
  25. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box.component.mjs +2 -2
  26. package/esm2022/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-capability-validator.interceptor.mjs +39 -0
  27. package/esm2022/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-installer.service.mjs +120 -0
  28. package/esm2022/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-intent-handler.interceptor.mjs +55 -0
  29. package/esm2022/lib/microfrontend-platform/microfrontend-perspective/workbench-perspective-data.mjs +19 -0
  30. package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup-capability-validator.interceptor.mjs +4 -3
  31. package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.mjs +2 -13
  32. package/esm2022/lib/microfrontend-platform/microfrontend-view/microfrontend-view-command-handler.service.mjs +1 -16
  33. package/esm2022/lib/microfrontend-platform/microfrontend-view/microfrontend-view.component.mjs +42 -17
  34. package/esm2022/lib/microfrontend-platform/public_api.mjs +2 -1
  35. package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-capability-validator.interceptor.mjs +5 -4
  36. package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-intent-handler.interceptor.mjs +7 -4
  37. package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-routes.mjs +26 -4
  38. package/esm2022/lib/microfrontend-platform/stable-capability-id-assigner.interceptor.mjs +32 -0
  39. package/esm2022/lib/microfrontend-platform/workbench-microfrontend-support.mjs +23 -5
  40. package/esm2022/lib/page-not-found/format-url.pipe.mjs +2 -2
  41. package/esm2022/lib/page-not-found/page-not-found.component.mjs +3 -3
  42. package/esm2022/lib/part/part.component.mjs +3 -5
  43. package/esm2022/lib/perspective/workbench-grid-merger.service.mjs +3 -3
  44. package/esm2022/lib/perspective/workbench-perspective.model.mjs +1 -1
  45. package/esm2022/lib/perspective/workbench-perspective.service.mjs +61 -51
  46. package/esm2022/lib/perspective//311/265workbench-perspective.model.mjs +11 -2
  47. package/esm2022/lib/popup/popup.config.mjs +3 -3
  48. package/esm2022/lib/portal/wb-component-portal.mjs +3 -1
  49. package/esm2022/lib/routing/public_api.mjs +1 -2
  50. package/esm2022/lib/routing/routing.model.mjs +1 -1
  51. package/esm2022/lib/routing/workbench-auxiliary-route-installer.service.mjs +94 -0
  52. package/esm2022/lib/routing/workbench-layout-differ.mjs +3 -10
  53. package/esm2022/lib/routing/workbench-url-observer.service.mjs +31 -27
  54. package/esm2022/lib/routing/workbench-view-outlet-differ.mjs +58 -0
  55. package/esm2022/lib/routing//311/265workbench-router.service.mjs +2 -2
  56. package/esm2022/lib/view/view-move-handler.service.mjs +5 -5
  57. package/esm2022/lib/view/view.component.mjs +38 -25
  58. package/esm2022/lib/view/workbench-view-route-guards.mjs +2 -2
  59. package/esm2022/lib/view//311/265workbench-view.model.mjs +25 -19
  60. package/esm2022/lib/view-dnd/grid-drop-targets.util.mjs +2 -2
  61. package/esm2022/lib/workbench-config.mjs +1 -1
  62. package/esm2022/lib/workbench-id.mjs +3 -3
  63. package/esm2022/lib/workbench.component.mjs +2 -2
  64. package/esm2022/lib/workbench.constants.mjs +1 -5
  65. package/esm2022/lib/workbench.provider.mjs +3 -9
  66. package/fesm2022/scion-workbench.mjs +855 -326
  67. package/fesm2022/scion-workbench.mjs.map +1 -1
  68. package/lib/common/objects.util.d.ts +4 -0
  69. package/lib/common/uid.util.d.ts +9 -0
  70. package/lib/dialog//311/265workbench-dialog.d.ts +1 -1
  71. package/lib/filter-field/filter-field.component.d.ts +1 -1
  72. package/lib/layout/grid-element/grid-element.component.d.ts +1 -8
  73. package/lib/layout/migration/model/workbench-layout-migration-v5.model.d.ts +32 -0
  74. package/lib/layout/migration/workbench-layout-migration-v5.service.d.ts +12 -0
  75. package/lib/layout/stringifier.d.ts +26 -0
  76. package/lib/layout/workbench-layout.model.d.ts +3 -3
  77. package/lib/layout/workench-layout-serializer.service.d.ts +13 -15
  78. package/lib/layout//311/265workbench-layout.d.ts +3 -3
  79. package/lib/microfrontend-platform/common/microfrontend.util.d.ts +5 -1
  80. package/lib/microfrontend-platform/initialization/microfrontend-platform-initializer.service.d.ts +7 -3
  81. package/lib/microfrontend-platform/manifest-object-cache.service.d.ts +33 -0
  82. package/lib/microfrontend-platform/microfrontend-host-message-box/text-message/text-message.component.d.ts +1 -1
  83. package/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-capability-validator.interceptor.d.ts +10 -0
  84. package/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-installer.service.d.ts +24 -0
  85. package/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-intent-handler.interceptor.d.ts +20 -0
  86. package/lib/microfrontend-platform/microfrontend-perspective/workbench-perspective-data.d.ts +9 -0
  87. package/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.d.ts +0 -4
  88. package/lib/microfrontend-platform/microfrontend-view/microfrontend-view-command-handler.service.d.ts +0 -4
  89. package/lib/microfrontend-platform/microfrontend-view/microfrontend-view.component.d.ts +7 -2
  90. package/lib/microfrontend-platform/public_api.d.ts +1 -0
  91. package/lib/microfrontend-platform/routing/microfrontend-view-capability-validator.interceptor.d.ts +1 -1
  92. package/lib/microfrontend-platform/routing/microfrontend-view-routes.d.ts +9 -2
  93. package/lib/microfrontend-platform/stable-capability-id-assigner.interceptor.d.ts +11 -0
  94. package/lib/perspective/workbench-perspective.model.d.ts +16 -17
  95. package/lib/perspective/workbench-perspective.service.d.ts +15 -7
  96. package/lib/perspective//311/265workbench-perspective.model.d.ts +4 -0
  97. package/lib/routing/public_api.d.ts +0 -1
  98. package/lib/routing/{workbench-auxiliary-routes-registrator.service.d.ts → workbench-auxiliary-route-installer.service.d.ts} +3 -3
  99. package/lib/routing/workbench-layout-differ.d.ts +1 -2
  100. package/lib/routing/workbench-url-observer.service.d.ts +5 -3
  101. package/lib/routing/workbench-view-outlet-differ.d.ts +32 -0
  102. package/lib/view/view.component.d.ts +8 -7
  103. package/lib/view//311/265workbench-view.model.d.ts +11 -7
  104. package/lib/workbench-config.d.ts +2 -2
  105. package/lib/workbench.constants.d.ts +0 -5
  106. package/package.json +3 -3
  107. package/esm2022/lib/common/uuid.util.mjs +0 -17
  108. package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-capability-id-assigner.interceptor.mjs +0 -41
  109. package/esm2022/lib/routing/workbench-auxiliary-routes-registrator.service.mjs +0 -94
  110. package/lib/common/uuid.util.d.ts +0 -8
  111. package/lib/microfrontend-platform/routing/microfrontend-view-capability-id-assigner.interceptor.d.ts +0 -10
package/_index.scss CHANGED
@@ -99,7 +99,7 @@
99
99
  * ```scss
100
100
  * @use '@scion/workbench' with (
101
101
  * $icon-font: (
102
- * directory: '/path/to/font', // defaults to '/assets/fonts' if omitted
102
+ * directory: '/path/to/font', // defaults to '/fonts' if omitted
103
103
  * filename: 'scion-workbench-icons' // defaults to 'scion-workbench-icons' if omitted
104
104
  * version: '1.0.0' // appended to the HTTP request to load the icon font, enabling browser cache invalidation when icons change
105
105
  * );
@@ -116,7 +116,7 @@
116
116
  * ```scss
117
117
  * @use '@scion/workbench' with (
118
118
  * $icon-font: (
119
- * directory: 'path/to/font' // no leading slash, typically `assets/fonts`
119
+ * directory: 'path/to/font' // no leading slash, typically `fonts`
120
120
  * )
121
121
  * );
122
122
  * ```
@@ -134,7 +134,7 @@
134
134
  * ```scss
135
135
  * @use '@scion/workbench' with (
136
136
  * $icon-font: (
137
- * directory: '^path/to/font' // no leading slash but with a caret (^), typically `^assets/fonts`
137
+ * directory: '^path/to/font' // no leading slash but with a caret (^), typically `^fonts`
138
138
  * )
139
139
  * );
140
140
  * ```
@@ -21,7 +21,7 @@ $-default-icon-font-config: (
21
21
  // To keep the workbench configuration minimal for Angular projects deployed in the context root, we reference font files by
22
22
  // their root relative path. Consequently, applications deployed in a subdirectory require additional configuration depending
23
23
  // on whether the application is built with esbuild or webpack.
24
- directory: '/assets/fonts',
24
+ directory: '/fonts',
25
25
  filename: 'scion-workbench-icons',
26
26
  version: $-version,
27
27
  );
@@ -30,5 +30,11 @@ export const Objects = {
30
30
  withoutUndefinedEntries: (object) => {
31
31
  return Dictionaries.withoutUndefinedEntries(object);
32
32
  },
33
+ /**
34
+ * Stringifies given object to matrix notation: a=b;c=d;e=f
35
+ */
36
+ toMatrixNotation: (object) => {
37
+ return Object.entries(object ?? {}).map(([key, value]) => `${key}=${value}`).join(';');
38
+ },
33
39
  };
34
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib2JqZWN0cy51dGlsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvY29tbW9uL29iamVjdHMudXRpbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRztBQUVILE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUVqRDs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRztJQUVyQjs7T0FFRztJQUNILElBQUksRUFBRSxDQUFJLE1BQVMsRUFBa0IsRUFBRTtRQUNyQyxPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBa0MsQ0FBbUIsQ0FBQztJQUMzRSxDQUFDO0lBRUQ7O09BRUc7SUFDSCxPQUFPLEVBQUUsQ0FBZ0IsTUFBd0MsRUFBaUIsRUFBRTtRQUNsRixPQUFPLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFrQixDQUFDO0lBQ2pELENBQUM7SUFFRDs7T0FFRztJQUNILHVCQUF1QixFQUFFLENBQUksTUFBbUMsRUFBSyxFQUFFO1FBQ3JFLE9BQU8sWUFBWSxDQUFDLHVCQUF1QixDQUFDLE1BQU0sQ0FBTSxDQUFDO0lBQzNELENBQUM7Q0FDTyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAxOC0yMDI0IFN3aXNzIEZlZGVyYWwgUmFpbHdheXNcbiAqXG4gKiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzIGFyZSBtYWRlXG4gKiBhdmFpbGFibGUgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBFY2xpcHNlIFB1YmxpYyBMaWNlbnNlIDIuMFxuICogd2hpY2ggaXMgYXZhaWxhYmxlIGF0IGh0dHBzOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC0yLjAvXG4gKlxuICogU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEVQTC0yLjBcbiAqL1xuXG5pbXBvcnQge0RpY3Rpb25hcmllc30gZnJvbSAnQHNjaW9uL3Rvb2xraXQvdXRpbCc7XG5cbi8qKlxuICogUHJvdmlkZXMgaGVscGVyIGZ1bmN0aW9ucyBmb3Igd29ya2luZyB3aXRoIG9iamVjdHMuXG4gKi9cbmV4cG9ydCBjb25zdCBPYmplY3RzID0ge1xuXG4gIC8qKlxuICAgKiBMaWtlIHtAbGluayBPYmplY3Qua2V5c30sIGJ1dCBwcmVzZXJ2aW5nIHRoZSBkYXRhIHR5cGUgb2Yga2V5cy5cbiAgICovXG4gIGtleXM6IDxUPihvYmplY3Q6IFQpOiBBcnJheTxrZXlvZiBUPiA9PiB7XG4gICAgcmV0dXJuIE9iamVjdC5rZXlzKG9iamVjdCBhcyBSZWNvcmQ8a2V5b2YgVCwgdW5rbm93bj4pIGFzIEFycmF5PGtleW9mIFQ+O1xuICB9LFxuXG4gIC8qKlxuICAgKiBMaWtlIHtAbGluayBPYmplY3QuZW50cmllc30sIGJ1dCBwcmVzZXJ2aW5nIHRoZSBkYXRhIHR5cGUgb2Yga2V5cy5cbiAgICovXG4gIGVudHJpZXM6IDxWLCBLID0gc3RyaW5nPihvYmplY3Q6IFJlY29yZDxzdHJpbmcsIFY+IHwgQXJyYXlMaWtlPFY+KTogQXJyYXk8W0ssIFZdPiA9PiB7XG4gICAgcmV0dXJuIE9iamVjdC5lbnRyaWVzKG9iamVjdCkgYXMgQXJyYXk8W0ssIFZdPjtcbiAgfSxcblxuICAvKipcbiAgICogTGlrZSB7QGxpbmsgRGljdGlvbmFyaWVzLndpdGhvdXRVbmRlZmluZWRFbnRyaWVzfSwgYnV0IHByZXNlcnZpbmcgdGhlIG9iamVjdCBkYXRhIHR5cGUuXG4gICAqL1xuICB3aXRob3V0VW5kZWZpbmVkRW50cmllczogPFQ+KG9iamVjdDogVCAmIFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogVCA9PiB7XG4gICAgcmV0dXJuIERpY3Rpb25hcmllcy53aXRob3V0VW5kZWZpbmVkRW50cmllcyhvYmplY3QpIGFzIFQ7XG4gIH0sXG59IGFzIGNvbnN0O1xuIl19
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib2JqZWN0cy51dGlsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvY29tbW9uL29iamVjdHMudXRpbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRztBQUVILE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUVqRDs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRztJQUVyQjs7T0FFRztJQUNILElBQUksRUFBRSxDQUFJLE1BQVMsRUFBa0IsRUFBRTtRQUNyQyxPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBa0MsQ0FBbUIsQ0FBQztJQUMzRSxDQUFDO0lBRUQ7O09BRUc7SUFDSCxPQUFPLEVBQUUsQ0FBZ0IsTUFBd0MsRUFBaUIsRUFBRTtRQUNsRixPQUFPLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFrQixDQUFDO0lBQ2pELENBQUM7SUFFRDs7T0FFRztJQUNILHVCQUF1QixFQUFFLENBQUksTUFBbUMsRUFBSyxFQUFFO1FBQ3JFLE9BQU8sWUFBWSxDQUFDLHVCQUF1QixDQUFDLE1BQU0sQ0FBTSxDQUFDO0lBQzNELENBQUM7SUFDRDs7T0FFRztJQUNILGdCQUFnQixFQUFFLENBQUMsTUFBa0QsRUFBVSxFQUFFO1FBQy9FLE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNLElBQUksRUFBRSxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLEVBQUUsRUFBRSxDQUFDLEdBQUcsR0FBRyxJQUFJLEtBQUssRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ3pGLENBQUM7Q0FDTyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAxOC0yMDI0IFN3aXNzIEZlZGVyYWwgUmFpbHdheXNcbiAqXG4gKiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzIGFyZSBtYWRlXG4gKiBhdmFpbGFibGUgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBFY2xpcHNlIFB1YmxpYyBMaWNlbnNlIDIuMFxuICogd2hpY2ggaXMgYXZhaWxhYmxlIGF0IGh0dHBzOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC0yLjAvXG4gKlxuICogU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEVQTC0yLjBcbiAqL1xuXG5pbXBvcnQge0RpY3Rpb25hcmllc30gZnJvbSAnQHNjaW9uL3Rvb2xraXQvdXRpbCc7XG5cbi8qKlxuICogUHJvdmlkZXMgaGVscGVyIGZ1bmN0aW9ucyBmb3Igd29ya2luZyB3aXRoIG9iamVjdHMuXG4gKi9cbmV4cG9ydCBjb25zdCBPYmplY3RzID0ge1xuXG4gIC8qKlxuICAgKiBMaWtlIHtAbGluayBPYmplY3Qua2V5c30sIGJ1dCBwcmVzZXJ2aW5nIHRoZSBkYXRhIHR5cGUgb2Yga2V5cy5cbiAgICovXG4gIGtleXM6IDxUPihvYmplY3Q6IFQpOiBBcnJheTxrZXlvZiBUPiA9PiB7XG4gICAgcmV0dXJuIE9iamVjdC5rZXlzKG9iamVjdCBhcyBSZWNvcmQ8a2V5b2YgVCwgdW5rbm93bj4pIGFzIEFycmF5PGtleW9mIFQ+O1xuICB9LFxuXG4gIC8qKlxuICAgKiBMaWtlIHtAbGluayBPYmplY3QuZW50cmllc30sIGJ1dCBwcmVzZXJ2aW5nIHRoZSBkYXRhIHR5cGUgb2Yga2V5cy5cbiAgICovXG4gIGVudHJpZXM6IDxWLCBLID0gc3RyaW5nPihvYmplY3Q6IFJlY29yZDxzdHJpbmcsIFY+IHwgQXJyYXlMaWtlPFY+KTogQXJyYXk8W0ssIFZdPiA9PiB7XG4gICAgcmV0dXJuIE9iamVjdC5lbnRyaWVzKG9iamVjdCkgYXMgQXJyYXk8W0ssIFZdPjtcbiAgfSxcblxuICAvKipcbiAgICogTGlrZSB7QGxpbmsgRGljdGlvbmFyaWVzLndpdGhvdXRVbmRlZmluZWRFbnRyaWVzfSwgYnV0IHByZXNlcnZpbmcgdGhlIG9iamVjdCBkYXRhIHR5cGUuXG4gICAqL1xuICB3aXRob3V0VW5kZWZpbmVkRW50cmllczogPFQ+KG9iamVjdDogVCAmIFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogVCA9PiB7XG4gICAgcmV0dXJuIERpY3Rpb25hcmllcy53aXRob3V0VW5kZWZpbmVkRW50cmllcyhvYmplY3QpIGFzIFQ7XG4gIH0sXG4gIC8qKlxuICAgKiBTdHJpbmdpZmllcyBnaXZlbiBvYmplY3QgdG8gbWF0cml4IG5vdGF0aW9uOiBhPWI7Yz1kO2U9ZlxuICAgKi9cbiAgdG9NYXRyaXhOb3RhdGlvbjogKG9iamVjdDogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCBudWxsIHwgdW5kZWZpbmVkKTogc3RyaW5nID0+IHtcbiAgICByZXR1cm4gT2JqZWN0LmVudHJpZXMob2JqZWN0ID8/IHt9KS5tYXAoKFtrZXksIHZhbHVlXSkgPT4gYCR7a2V5fT0ke3ZhbHVlfWApLmpvaW4oJzsnKTtcbiAgfSxcbn0gYXMgY29uc3Q7XG4iXX0=
@@ -0,0 +1,22 @@
1
+ /*
2
+ * Copyright (c) 2018-2024 Swiss Federal Railways
3
+ *
4
+ * This program and the accompanying materials are made
5
+ * available under the terms of the Eclipse Public License 2.0
6
+ * which is available at https://www.eclipse.org/legal/epl-2.0/
7
+ *
8
+ * SPDX-License-Identifier: EPL-2.0
9
+ */
10
+ import { UUID } from '@scion/toolkit/uuid';
11
+ /**
12
+ * Generates a UID (unique identifier).
13
+ */
14
+ export const UID = {
15
+ /**
16
+ * Generates a UID (unique identifier) with length 8.
17
+ */
18
+ randomUID: () => {
19
+ return UUID.randomUUID().substring(0, 8);
20
+ },
21
+ };
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidWlkLnV0aWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9jb21tb24vdWlkLnV0aWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7O0dBUUc7QUFFSCxPQUFPLEVBQUMsSUFBSSxFQUFDLE1BQU0scUJBQXFCLENBQUM7QUFFekM7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxHQUFHLEdBQUc7SUFDakI7O09BRUc7SUFDSCxTQUFTLEVBQUUsR0FBVyxFQUFFO1FBQ3RCLE9BQU8sSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDM0MsQ0FBQztDQUNGLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjQgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7VVVJRH0gZnJvbSAnQHNjaW9uL3Rvb2xraXQvdXVpZCc7XG5cbi8qKlxuICogR2VuZXJhdGVzIGEgVUlEICh1bmlxdWUgaWRlbnRpZmllcikuXG4gKi9cbmV4cG9ydCBjb25zdCBVSUQgPSB7XG4gIC8qKlxuICAgKiBHZW5lcmF0ZXMgYSBVSUQgKHVuaXF1ZSBpZGVudGlmaWVyKSB3aXRoIGxlbmd0aCA4LlxuICAgKi9cbiAgcmFuZG9tVUlEOiAoKTogc3RyaW5nID0+IHtcbiAgICByZXR1cm4gVVVJRC5yYW5kb21VVUlEKCkuc3Vic3RyaW5nKDAsIDgpO1xuICB9LFxufTtcblxuXG4iXX0=
@@ -10,8 +10,8 @@
10
10
  import { Directive, Input, Optional } from '@angular/core';
11
11
  import { fromDimension$ } from '@scion/toolkit/observable';
12
12
  import { setStyle } from '../common/dom.util';
13
- import { Subject } from 'rxjs';
14
- import { takeUntil } from 'rxjs/operators';
13
+ import { animationFrameScheduler, EMPTY, merge, Subject } from 'rxjs';
14
+ import { filter, observeOn, takeUntil } from 'rxjs/operators';
15
15
  import * as i0 from "@angular/core";
16
16
  import * as i1 from "../view/\u0275workbench-view.model";
17
17
  /**
@@ -35,17 +35,12 @@ export class ContentProjectionDirective {
35
35
  this._contentViewRef.onDestroy(() => dispose$.next());
36
36
  // Position projected content out of the document flow relative to the page viewport.
37
37
  this.styleContent({ position: 'fixed' });
38
- // Align content each time the bounding box element changes its size.
39
- fromDimension$(this._host.nativeElement)
40
- .pipe(takeUntil(dispose$))
41
- .subscribe(dimension => {
42
- if (isNullDimension(dimension)) {
43
- // When removing the bounding box element (this directive's host) from the DOM, its dimension drops to 0.
44
- // We ignore this event to preserve the dimension of projected content, crucial, for example, if projected
45
- // content implements virtual scrolling. Otherwise, its content would reload when adding the host to the DOM again.
46
- // For example, inactive views are removed from the DOM.
47
- return;
48
- }
38
+ // Align content each time the size of the host element changes, or when the content is attached to the DOM.
39
+ // For example, moving a view to another part of the same size will not trigger a dimension change event.
40
+ merge(fromDimension$(this._host.nativeElement), this._view?.portal.attached$.pipe(filter(Boolean)) ?? EMPTY)
41
+ .pipe(observeOn(animationFrameScheduler), // Align to host boundaries right before the next repaint.
42
+ takeUntil(dispose$))
43
+ .subscribe(() => {
49
44
  this.alignContentToHostBoundaries();
50
45
  });
51
46
  // Hide content when contextual view is detached, e.g., if not active, or located in the peripheral area and the main area is maximized.
@@ -60,6 +55,13 @@ export class ContentProjectionDirective {
60
55
  */
61
56
  alignContentToHostBoundaries() {
62
57
  const hostPosition = this._host.nativeElement.getBoundingClientRect();
58
+ if (!hostPosition.width && !hostPosition.height) {
59
+ // When removing the bounding box element (this directive's host) from the DOM, its dimension drops to 0.
60
+ // We ignore this event to preserve the dimension of projected content, crucial, for example, if projected
61
+ // content implements virtual scrolling. Otherwise, its content would reload when adding the host to the DOM again.
62
+ // For example, inactive views are removed from the DOM.
63
+ return;
64
+ }
63
65
  this.styleContent({
64
66
  top: `${hostPosition.top}px`,
65
67
  left: `${hostPosition.left}px`,
@@ -95,7 +97,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImpor
95
97
  type: Input,
96
98
  args: [{ alias: 'wbContentProjectionContent', required: true }]
97
99
  }] } });
98
- function isNullDimension(dimension) {
99
- return dimension.offsetWidth === 0 && dimension.offsetHeight === 0;
100
- }
101
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"content-projection.directive.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/content-projection/content-projection.directive.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAC,SAAS,EAA+B,KAAK,EAAwB,QAAQ,EAA+C,MAAM,eAAe,CAAC;AAE1J,OAAO,EAAY,cAAc,EAAC,MAAM,2BAA2B,CAAC;AACpE,OAAO,EAAC,QAAQ,EAAC,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAC,OAAO,EAAC,MAAM,MAAM,CAAC;AAC7B,OAAO,EAAC,SAAS,EAAC,MAAM,gBAAgB,CAAC;;;AAEzC;;GAEG;AAEH,MAAM,OAAO,0BAA0B;IAgBrC,YAAoB,KAA8B,EAAsB,KAAqB;QAAzE,UAAK,GAAL,KAAK,CAAyB;QAAsB,UAAK,GAAL,KAAK,CAAgB;IAC7F,CAAC;IAEM,WAAW,CAAC,OAAsB;QACvC,IAAI,CAAC,eAAe,EAAE,OAAO,EAAE,CAAC;QAEhC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,OAAO;QACT,CAAC;QAED,6FAA6F;QAC7F,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,WAAW,CAAC,kBAAkB,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAC;QAC1F,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,CAAC;QAErC,6BAA6B;QAC7B,MAAM,QAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QACrC,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;QAEtD,qFAAqF;QACrF,IAAI,CAAC,YAAY,CAAC,EAAC,QAAQ,EAAE,OAAO,EAAC,CAAC,CAAC;QAEvC,qEAAqE;QACrE,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC;aACrC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;aACzB,SAAS,CAAC,SAAS,CAAC,EAAE;YACrB,IAAI,eAAe,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC/B,yGAAyG;gBACzG,0GAA0G;gBAC1G,mHAAmH;gBACnH,wDAAwD;gBACxD,OAAO;YACT,CAAC;YACD,IAAI,CAAC,4BAA4B,EAAE,CAAC;QACtC,CAAC,CAAC,CAAC;QAEL,wIAAwI;QACxI,IAAI,CAAC,KAAK,EAAE,MAAM,CAAC,SAAS;aACzB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;aACzB,SAAS,CAAC,QAAQ,CAAC,EAAE;YACpB,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QAC5B,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;OAEG;IACK,4BAA4B;QAClC,MAAM,YAAY,GAAY,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC;QAC/E,IAAI,CAAC,YAAY,CAAC;YAChB,GAAG,EAAE,GAAG,YAAY,CAAC,GAAG,IAAI;YAC5B,IAAI,EAAE,GAAG,YAAY,CAAC,IAAI,IAAI;YAC9B,KAAK,EAAE,GAAG,YAAY,CAAC,KAAK,IAAI;YAChC,MAAM,EAAE,GAAG,YAAY,CAAC,MAAM,IAAI;SACnC,CAAC,CAAC;IACL,CAAC;IAEO,UAAU,CAAC,OAAgB;QACjC,sGAAsG;QACtG,aAAa;QACb,oIAAoI;QACpI,mHAAmH;QACnH,IAAI,CAAC,YAAY,CAAC,EAAC,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,EAAC,CAAC,CAAC;IAC7D,CAAC;IAEO,YAAY,CAAC,KAA6B;QAChD,IAAI,CAAC,eAAgB,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;IACzE,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,eAAe,EAAE,OAAO,EAAE,CAAC;IAClC,CAAC;8GAtFU,0BAA0B;kGAA1B,0BAA0B;;2FAA1B,0BAA0B;kBADtC,SAAS;mBAAC,EAAC,QAAQ,EAAE,uBAAuB,EAAE,UAAU,EAAE,IAAI,EAAC;;0BAiBT,QAAQ;yCARtD,WAAW;sBADjB,KAAK;uBAAC,EAAC,KAAK,EAAE,gCAAgC,EAAE,QAAQ,EAAE,IAAI,EAAC;gBAOzD,kBAAkB;sBADxB,KAAK;uBAAC,EAAC,KAAK,EAAE,4BAA4B,EAAE,QAAQ,EAAE,IAAI,EAAC;;AA4E9D,SAAS,eAAe,CAAC,SAAoB;IAC3C,OAAO,SAAS,CAAC,WAAW,KAAK,CAAC,IAAI,SAAS,CAAC,YAAY,KAAK,CAAC,CAAC;AACrE,CAAC","sourcesContent":["/*\n * Copyright (c) 2018-2022 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {Directive, ElementRef, EmbeddedViewRef, Input, OnChanges, OnDestroy, Optional, SimpleChanges, TemplateRef, ViewContainerRef} from '@angular/core';\nimport {ɵWorkbenchView} from '../view/ɵworkbench-view.model';\nimport {Dimension, fromDimension$} from '@scion/toolkit/observable';\nimport {setStyle} from '../common/dom.util';\nimport {Subject} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\n\n/**\n * Renders a given template as overlay. The template will stick to the bounding box of the host element of this directive.\n */\n@Directive({selector: '[wbContentProjection]', standalone: true})\nexport class ContentProjectionDirective implements OnChanges, OnDestroy {\n\n  private _contentViewRef: EmbeddedViewRef<unknown> | undefined;\n\n  /**\n   * Reference to the view container where to insert the overlay.\n   */\n  @Input({alias: 'wbContentProjectionOverlayHost', required: true}) // eslint-disable-line @angular-eslint/no-input-rename\n  public overlayHost: ViewContainerRef | undefined | null;\n\n  /**\n   * Template which to render as overlay. The template will stick to the bounding box of the host element of this directive.\n   */\n  @Input({alias: 'wbContentProjectionContent', required: true}) // eslint-disable-line @angular-eslint/no-input-rename\n  public contentTemplateRef!: TemplateRef<void>;\n\n  constructor(private _host: ElementRef<HTMLElement>, @Optional() private _view: ɵWorkbenchView) {\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this._contentViewRef?.destroy();\n\n    if (!this.overlayHost) {\n      return;\n    }\n\n    // Create embedded view from content template and align it to the bounds of the host element.\n    this._contentViewRef = this.overlayHost.createEmbeddedView(this.contentTemplateRef, null);\n    this._contentViewRef.detectChanges();\n\n    // Register dispose notifier.\n    const dispose$ = new Subject<void>();\n    this._contentViewRef.onDestroy(() => dispose$.next());\n\n    // Position projected content out of the document flow relative to the page viewport.\n    this.styleContent({position: 'fixed'});\n\n    // Align content each time the bounding box element changes its size.\n    fromDimension$(this._host.nativeElement)\n      .pipe(takeUntil(dispose$))\n      .subscribe(dimension => {\n        if (isNullDimension(dimension)) {\n          // When removing the bounding box element (this directive's host) from the DOM, its dimension drops to 0.\n          // We ignore this event to preserve the dimension of projected content, crucial, for example, if projected\n          // content implements virtual scrolling. Otherwise, its content would reload when adding the host to the DOM again.\n          // For example, inactive views are removed from the DOM.\n          return;\n        }\n        this.alignContentToHostBoundaries();\n      });\n\n    // Hide content when contextual view is detached, e.g., if not active, or located in the peripheral area and the main area is maximized.\n    this._view?.portal.attached$\n      .pipe(takeUntil(dispose$))\n      .subscribe(attached => {\n        this.setVisible(attached);\n      });\n  }\n\n  /**\n   * Aligns the content of the projection to the boundaries of the host element.\n   */\n  private alignContentToHostBoundaries(): void {\n    const hostPosition: DOMRect = this._host.nativeElement.getBoundingClientRect();\n    this.styleContent({\n      top: `${hostPosition.top}px`,\n      left: `${hostPosition.left}px`,\n      width: `${hostPosition.width}px`,\n      height: `${hostPosition.height}px`,\n    });\n  }\n\n  private setVisible(visible: boolean): void {\n    // We use `visibility: hidden` and not `display: none` to preserve the dimension of projected content.\n    // Otherwise:\n    // - Projected content would flicker when attaching the contextual view, most noticeable with content that displays a microfrontend.\n    // - Projected content would not retain virtual scrollable content since `display:none` sets width and height to 0.\n    this.styleContent({visibility: visible ? null : 'hidden'});\n  }\n\n  private styleContent(style: {[style: string]: any}): void {\n    this._contentViewRef!.rootNodes.forEach(node => setStyle(node, style));\n  }\n\n  public ngOnDestroy(): void {\n    this._contentViewRef?.destroy();\n  }\n}\n\nfunction isNullDimension(dimension: Dimension): boolean {\n  return dimension.offsetWidth === 0 && dimension.offsetHeight === 0;\n}\n"]}
100
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"content-projection.directive.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/content-projection/content-projection.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2022 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {Directive, ElementRef, EmbeddedViewRef, Input, OnChanges, OnDestroy, Optional, SimpleChanges, TemplateRef, ViewContainerRef} from '@angular/core';\nimport {ɵWorkbenchView} from '../view/ɵworkbench-view.model';\nimport {fromDimension$} from '@scion/toolkit/observable';\nimport {setStyle} from '../common/dom.util';\nimport {animationFrameScheduler, EMPTY, merge, Subject} from 'rxjs';\nimport {filter, observeOn, takeUntil} from 'rxjs/operators';\n\n/**\n * Renders a given template as overlay. The template will stick to the bounding box of the host element of this directive.\n */\n@Directive({selector: '[wbContentProjection]', standalone: true})\nexport class ContentProjectionDirective implements OnChanges, OnDestroy {\n\n  private _contentViewRef: EmbeddedViewRef<unknown> | undefined;\n\n  /**\n   * Reference to the view container where to insert the overlay.\n   */\n  @Input({alias: 'wbContentProjectionOverlayHost', required: true}) // eslint-disable-line @angular-eslint/no-input-rename\n  public overlayHost: ViewContainerRef | undefined | null;\n\n  /**\n   * Template which to render as overlay. The template will stick to the bounding box of the host element of this directive.\n   */\n  @Input({alias: 'wbContentProjectionContent', required: true}) // eslint-disable-line @angular-eslint/no-input-rename\n  public contentTemplateRef!: TemplateRef<void>;\n\n  constructor(private _host: ElementRef<HTMLElement>, @Optional() private _view: ɵWorkbenchView) {\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this._contentViewRef?.destroy();\n\n    if (!this.overlayHost) {\n      return;\n    }\n\n    // Create embedded view from content template and align it to the bounds of the host element.\n    this._contentViewRef = this.overlayHost.createEmbeddedView(this.contentTemplateRef, null);\n    this._contentViewRef.detectChanges();\n\n    // Register dispose notifier.\n    const dispose$ = new Subject<void>();\n    this._contentViewRef.onDestroy(() => dispose$.next());\n\n    // Position projected content out of the document flow relative to the page viewport.\n    this.styleContent({position: 'fixed'});\n\n    // Align content each time the size of the host element changes, or when the content is attached to the DOM.\n    // For example, moving a view to another part of the same size will not trigger a dimension change event.\n    merge(fromDimension$(this._host.nativeElement), this._view?.portal.attached$.pipe(filter(Boolean)) ?? EMPTY)\n      .pipe(\n        observeOn(animationFrameScheduler), // Align to host boundaries right before the next repaint.\n        takeUntil(dispose$),\n      )\n      .subscribe(() => {\n        this.alignContentToHostBoundaries();\n      });\n\n    // Hide content when contextual view is detached, e.g., if not active, or located in the peripheral area and the main area is maximized.\n    this._view?.portal.attached$\n      .pipe(takeUntil(dispose$))\n      .subscribe(attached => {\n        this.setVisible(attached);\n      });\n  }\n\n  /**\n   * Aligns the content of the projection to the boundaries of the host element.\n   */\n  private alignContentToHostBoundaries(): void {\n    const hostPosition: DOMRect = this._host.nativeElement.getBoundingClientRect();\n    if (!hostPosition.width && !hostPosition.height) {\n      // When removing the bounding box element (this directive's host) from the DOM, its dimension drops to 0.\n      // We ignore this event to preserve the dimension of projected content, crucial, for example, if projected\n      // content implements virtual scrolling. Otherwise, its content would reload when adding the host to the DOM again.\n      // For example, inactive views are removed from the DOM.\n      return;\n    }\n\n    this.styleContent({\n      top: `${hostPosition.top}px`,\n      left: `${hostPosition.left}px`,\n      width: `${hostPosition.width}px`,\n      height: `${hostPosition.height}px`,\n    });\n  }\n\n  private setVisible(visible: boolean): void {\n    // We use `visibility: hidden` and not `display: none` to preserve the dimension of projected content.\n    // Otherwise:\n    // - Projected content would flicker when attaching the contextual view, most noticeable with content that displays a microfrontend.\n    // - Projected content would not retain virtual scrollable content since `display:none` sets width and height to 0.\n    this.styleContent({visibility: visible ? null : 'hidden'});\n  }\n\n  private styleContent(style: {[style: string]: any}): void {\n    this._contentViewRef!.rootNodes.forEach(node => setStyle(node, style));\n  }\n\n  public ngOnDestroy(): void {\n    this._contentViewRef?.destroy();\n  }\n}\n"]}
@@ -25,7 +25,7 @@ import { WORKBENCH_ELEMENT_REF } from '../content-projection/view-container.refe
25
25
  import { Arrays } from '@scion/toolkit/util';
26
26
  import { WorkbenchConfig } from '../workbench-config';
27
27
  import { distinctUntilChanged, filter } from 'rxjs/operators';
28
- import { randomUUID } from '../common/uuid.util';
28
+ import { UUID } from '@scion/toolkit/uuid';
29
29
  /** @inheritDoc */
30
30
  export class ɵWorkbenchDialog {
31
31
  constructor(component, _options) {
@@ -39,7 +39,7 @@ export class ɵWorkbenchDialog {
39
39
  /**
40
40
  * Unique identity of this dialog.
41
41
  */
42
- this.id = randomUUID();
42
+ this.id = UUID.randomUUID();
43
43
  /**
44
44
  * Indicates whether this dialog is blocked by other dialog(s) that overlay this dialog.
45
45
  */
@@ -269,4 +269,4 @@ export class ɵWorkbenchDialog {
269
269
  }
270
270
  }
271
271
  }
272
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ɵworkbench-dialog.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/dialog/ɵworkbench-dialog.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {BehaviorSubject, combineLatest, concatWith, delay, EMPTY, firstValueFrom, map, merge, Observable, of, Subject, switchMap} from 'rxjs';\nimport {ComponentRef, inject, Injector, NgZone} from '@angular/core';\nimport {WorkbenchDialog, WorkbenchDialogSize} from './workbench-dialog';\nimport {WorkbenchDialogOptions} from './workbench-dialog.options';\nimport {ComponentPortal, ComponentType} from '@angular/cdk/portal';\nimport {Overlay, OverlayRef} from '@angular/cdk/overlay';\nimport {WorkbenchDialogComponent} from './workbench-dialog.component';\nimport {ɵWorkbenchView} from '../view/ɵworkbench-view.model';\nimport {WorkbenchDialogRegistry} from './workbench-dialog.registry';\nimport {ɵDestroyRef} from '../common/ɵdestroy-ref';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {setStyle} from '../common/dom.util';\nimport {fromDimension$} from '@scion/toolkit/observable';\nimport {subscribeInside} from '@scion/toolkit/operators';\nimport {ViewDragService} from '../view-dnd/view-drag.service';\nimport {WORKBENCH_ELEMENT_REF} from '../content-projection/view-container.reference';\nimport {Arrays} from '@scion/toolkit/util';\nimport {WorkbenchConfig} from '../workbench-config';\nimport {distinctUntilChanged, filter} from 'rxjs/operators';\nimport {WorkbenchDialogActionDirective} from './dialog-footer/workbench-dialog-action.directive';\nimport {WorkbenchDialogFooterDirective} from './dialog-footer/workbench-dialog-footer.directive';\nimport {WorkbenchDialogHeaderDirective} from './dialog-header/workbench-dialog-header.directive';\nimport {Disposable} from '../common/disposable';\nimport {Blockable} from '../glass-pane/blockable';\nimport {Blocking} from '../glass-pane/blocking';\nimport {randomUUID} from '../common/uuid.util';\n\n/** @inheritDoc */\nexport class ɵWorkbenchDialog<R = unknown> implements WorkbenchDialog<R>, Blockable, Blocking {\n\n  private readonly _overlayRef: OverlayRef;\n  private readonly _portal: ComponentPortal<WorkbenchDialogComponent>;\n  private readonly _workbenchDialogRegistry = inject(WorkbenchDialogRegistry);\n  private readonly _zone = inject(NgZone);\n  private readonly _workbenchConfig = inject(WorkbenchConfig);\n  private readonly _destroyRef = new ɵDestroyRef();\n  private readonly _attached$: Observable<boolean>;\n  private _blink$ = new Subject<void>();\n\n  /**\n   * Result (or error) to be passed to the dialog opener.\n   */\n  private _result: R | Error | undefined;\n  private _componentRef: ComponentRef<WorkbenchDialogComponent> | undefined;\n  private _cssClass: string;\n\n  /**\n   * Unique identity of this dialog.\n   */\n  public readonly id = randomUUID();\n  /**\n   * Indicates whether this dialog is blocked by other dialog(s) that overlay this dialog.\n   */\n  public readonly blockedBy$ = new BehaviorSubject<ɵWorkbenchDialog | null>(null);\n  public readonly size: WorkbenchDialogSize = {};\n  public title: string | Observable<string | undefined> | undefined;\n  public closable = true;\n  public resizable = true;\n  public padding = true;\n  public header: WorkbenchDialogHeaderDirective | undefined;\n  public footer: WorkbenchDialogFooterDirective | undefined;\n  public actions = new Array<WorkbenchDialogActionDirective>();\n  public blinking$ = new BehaviorSubject(false);\n  public context = {\n    view: inject(ɵWorkbenchView, {optional: true}),\n  };\n\n  constructor(public component: ComponentType<unknown>, public _options: WorkbenchDialogOptions) {\n    this._overlayRef = this.createOverlay();\n    this._portal = this.createPortal();\n    this._cssClass = Arrays.coerce(this._options.cssClass).join(' ');\n    this._attached$ = this.monitorHostElementAttached$();\n\n    this.hideOnHostElementDetachOrViewDrag();\n    this.stickToHostElement();\n    this.blockWhenNotOnTop();\n    this.restoreFocusOnAttach();\n    this.restoreFocusOnUnblock();\n    this.blinkOnRequest();\n  }\n\n  public async open(): Promise<R | undefined> {\n    // Wait for the overlay to be initially positioned to have a smooth slide-in animation.\n    if (this.animate) {\n      await firstValueFrom(fromDimension$(this._overlayRef.hostElement));\n    }\n\n    // Attach the dialog portal to the overlay.\n    this._componentRef = this._overlayRef.attach(this._portal);\n\n    // Ensure to destroy this handle on browser back/forward navigation.\n    this._componentRef.onDestroy(() => this.destroy());\n\n    // Trigger a manual change detection cycle to avoid 'ExpressionChangedAfterItHasBeenCheckedError'\n    // when the dialog sets dialog-specific properties such as title or size during construction.\n    this._componentRef.changeDetectorRef.detectChanges();\n\n    // Wait for the dialog to close, resolving to its result or rejecting if closed with an error.\n    return new Promise<R | undefined>((resolve, reject) => {\n      this._destroyRef.onDestroy(() => {\n        this._result instanceof Error ? reject(this._result) : resolve(this._result);\n      });\n    });\n  }\n\n  /** @inheritDoc */\n  public close(result?: R | Error): void {\n    this._result = result;\n    this.destroy();\n  }\n\n  /**\n   * Inputs passed to the dialog.\n   */\n  public get inputs(): {[name: string]: unknown} | undefined {\n    return this._options.inputs;\n  }\n\n  /**\n   * Indicates if to animate the dialog.\n   */\n  public get animate(): boolean {\n    return this._options.animate ?? false;\n  }\n\n  /** @inheritDoc */\n  public focus(): void {\n    if (!this.blockedBy$.value) {\n      this._componentRef?.instance.focus();\n    }\n  }\n\n  public registerHeader(header: WorkbenchDialogHeaderDirective): Disposable {\n    this.header = header;\n    return {\n      dispose: () => {\n        if (this.header === header) {\n          this.header = undefined;\n        }\n      },\n    };\n  }\n\n  public registerFooter(footer: WorkbenchDialogFooterDirective): Disposable {\n    if (this.actions.length) {\n      throw Error('[DialogInitError] Custom dialog footer not supported if using dialog actions.');\n    }\n    this.footer = footer;\n    return {\n      dispose: () => {\n        if (this.footer === footer) {\n          this.footer = undefined;\n        }\n      },\n    };\n  }\n\n  public registerAction(action: WorkbenchDialogActionDirective): Disposable {\n    if (this.footer) {\n      throw Error('[DialogInitError] Dialog actions not supported if using a custom dialog footer.');\n    }\n    this.actions = this.actions.concat(action);\n    return {\n      dispose: () => this.actions = this.actions.filter(candidate => candidate !== action),\n    };\n  }\n\n  /**\n   * Returns the position of the dialog in the dialog stack.\n   */\n  public getPositionInDialogStack(): number {\n    return this._workbenchDialogRegistry.indexOf(this);\n  }\n\n  /** @inheritDoc */\n  public set cssClass(cssClass: string | string[]) {\n    this._cssClass = new Array<string>().concat(this._options.cssClass ?? []).concat(cssClass).join(' ');\n  }\n\n  /** @inheritDoc */\n  public get cssClass(): string {\n    return this._cssClass;\n  }\n\n  /** @inheritDoc */\n  public blink(): void {\n    this._blink$.next();\n  }\n\n  private createPortal(): ComponentPortal<WorkbenchDialogComponent> {\n    return new ComponentPortal(WorkbenchDialogComponent, null, Injector.create({\n      parent: inject(Injector),\n      providers: [\n        {provide: ɵWorkbenchDialog, useValue: this},\n        {provide: WorkbenchDialog, useExisting: ɵWorkbenchDialog},\n      ],\n    }));\n  }\n\n  /**\n   * Creates a dedicated overlay per dialog to place it on top of previously created overlays, such as dialogs, popups, dropdowns, etc.\n   */\n  private createOverlay(): OverlayRef {\n    const overlay = inject(Overlay);\n    return overlay.create({\n      disposeOnNavigation: true, // dispose dialog on browser back/forward navigation\n      panelClass: ['wb-dialog-modality-context'],\n      positionStrategy: overlay.position().global(),\n      scrollStrategy: overlay.scrollStrategies.noop(),\n    });\n  }\n\n  /**\n   * Restores focus when re-attaching this dialog.\n   */\n  private restoreFocusOnAttach(): void {\n    this._attached$\n      .pipe(\n        filter(Boolean),\n        takeUntilDestroyed(),\n      )\n      .subscribe(() => {\n        this.focus();\n      });\n  }\n\n  /**\n   * Restores focus when unblocking this dialog.\n   */\n  private restoreFocusOnUnblock(): void {\n    this.blockedBy$\n      .pipe(\n        filter(blockedBy => !blockedBy),\n        takeUntilDestroyed(),\n      )\n      .subscribe(() => {\n        this.focus();\n      });\n  }\n\n  /**\n   * Monitors attachment of the host element.\n   */\n  private monitorHostElementAttached$(): Observable<boolean> {\n    if (this.context.view) {\n      return this.context.view.portal.attached$;\n    }\n    if (this._workbenchConfig.dialog?.modalityScope === 'viewport') {\n      return of(true);\n    }\n    return inject(WORKBENCH_ELEMENT_REF).ref$.pipe(map(ref => !!ref));\n  }\n\n  /**\n   * Hides this dialog when either its host element is detached or during view drag and drop operation.\n   */\n  private hideOnHostElementDetachOrViewDrag(): void {\n    const viewDragService = inject(ViewDragService);\n    const viewDrag$ = merge(\n      viewDragService.viewDragStart$.pipe(map(() => true)),\n      viewDragService.viewDragEnd$.pipe(map(() => false)),\n      of(false), // to initialize `combineLatest`\n    );\n\n    combineLatest([this._attached$, viewDrag$])\n      .pipe(\n        subscribeInside(fn => this._zone.runOutsideAngular(fn)),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(([attached, dragging]) => {\n        const hideDialog = !attached || dragging;\n\n        // Hide via `visibility: hidden` instead of `display: none` in order to preserve the dimension of the dialog.\n        setStyle(this._overlayRef.overlayElement, {visibility: hideDialog ? 'hidden' : null});\n      });\n  }\n\n  /**\n   * Aligns this dialog with the boundaries of the host element.\n   */\n  private stickToHostElement(): void {\n    if (this._options.modality === 'application' && this._workbenchConfig.dialog?.modalityScope === 'viewport') {\n      setStyle(this._overlayRef.hostElement, {inset: '0'});\n    }\n    else {\n      const workbenchViewElement$ = (view: ɵWorkbenchView): Observable<HTMLElement> => of(view.portal.componentRef.location.nativeElement);\n      const workbenchRootElement$ = (): Observable<HTMLElement> => inject(WORKBENCH_ELEMENT_REF).ref$.pipe(map(ref => ref?.element.nativeElement));\n      const hostElement$ = this.context.view ? workbenchViewElement$(this.context.view) : workbenchRootElement$();\n\n      hostElement$\n        .pipe(\n          switchMap(hostElement => hostElement ? fromDimension$(hostElement) : EMPTY),\n          map(({element: hostElement}) => hostElement.getBoundingClientRect()),\n          subscribeInside(fn => this._zone.runOutsideAngular(fn)),\n          takeUntilDestroyed(this._destroyRef),\n        )\n        .subscribe(({top, left, width, height}) => {\n          setStyle(this._overlayRef.hostElement, {\n            top: `${top}px`,\n            left: `${left}px`,\n            width: `${width}px`,\n            height: `${height}px`,\n          });\n        });\n    }\n  }\n\n  /**\n   * Blocks this dialog if not the topmost dialog in its context.\n   */\n  private blockWhenNotOnTop(): void {\n    this._workbenchDialogRegistry.top$({viewId: this.context.view?.id})\n      .pipe(\n        map(top => top === this ? null : top),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(this.blockedBy$);\n  }\n\n  private blinkOnRequest(): void {\n    this._blink$\n      .pipe(\n        switchMap(() => of(true).pipe(concatWith(of(false).pipe(delay(300))))),\n        distinctUntilChanged(),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(this.blinking$);\n  }\n\n  /**\n   * Destroys this dialog and associated resources.\n   */\n  public destroy(): void {\n    if (!this._destroyRef.destroyed) {\n      this._destroyRef.destroy();\n      this._overlayRef.dispose();\n    }\n  }\n}\n"]}
272
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ɵworkbench-dialog.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/dialog/ɵworkbench-dialog.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAC,eAAe,EAAE,aAAa,EAAE,UAAU,EAAE,KAAK,EAAE,KAAK,EAAE,cAAc,EAAE,GAAG,EAAE,KAAK,EAAc,EAAE,EAAE,OAAO,EAAE,SAAS,EAAC,MAAM,MAAM,CAAC;AAC9I,OAAO,EAAe,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAC,MAAM,eAAe,CAAC;AACrE,OAAO,EAAC,eAAe,EAAsB,MAAM,oBAAoB,CAAC;AAExE,OAAO,EAAC,eAAe,EAAgB,MAAM,qBAAqB,CAAC;AACnE,OAAO,EAAC,OAAO,EAAa,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAC,wBAAwB,EAAC,MAAM,8BAA8B,CAAC;AACtE,OAAO,EAAC,cAAc,EAAC,MAAM,+BAA+B,CAAC;AAC7D,OAAO,EAAC,uBAAuB,EAAC,MAAM,6BAA6B,CAAC;AACpE,OAAO,EAAC,WAAW,EAAC,MAAM,wBAAwB,CAAC;AACnD,OAAO,EAAC,kBAAkB,EAAC,MAAM,4BAA4B,CAAC;AAC9D,OAAO,EAAC,QAAQ,EAAC,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAC,cAAc,EAAC,MAAM,2BAA2B,CAAC;AACzD,OAAO,EAAC,eAAe,EAAC,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAC,eAAe,EAAC,MAAM,+BAA+B,CAAC;AAC9D,OAAO,EAAC,qBAAqB,EAAC,MAAM,gDAAgD,CAAC;AACrF,OAAO,EAAC,MAAM,EAAC,MAAM,qBAAqB,CAAC;AAC3C,OAAO,EAAC,eAAe,EAAC,MAAM,qBAAqB,CAAC;AACpD,OAAO,EAAC,oBAAoB,EAAE,MAAM,EAAC,MAAM,gBAAgB,CAAC;AAO5D,OAAO,EAAC,IAAI,EAAC,MAAM,qBAAqB,CAAC;AAEzC,kBAAkB;AAClB,MAAM,OAAO,gBAAgB;IAuC3B,YAAmB,SAAiC,EAAS,QAAgC;QAA1E,cAAS,GAAT,SAAS,CAAwB;QAAS,aAAQ,GAAR,QAAQ,CAAwB;QAnC5E,6BAAwB,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAC3D,UAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QACvB,qBAAgB,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QAC3C,gBAAW,GAAG,IAAI,WAAW,EAAE,CAAC;QAEzC,YAAO,GAAG,IAAI,OAAO,EAAQ,CAAC;QAStC;;WAEG;QACa,OAAE,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QACvC;;WAEG;QACa,eAAU,GAAG,IAAI,eAAe,CAA0B,IAAI,CAAC,CAAC;QAChE,SAAI,GAAwB,EAAE,CAAC;QAExC,aAAQ,GAAG,IAAI,CAAC;QAChB,cAAS,GAAG,IAAI,CAAC;QACjB,YAAO,GAAG,IAAI,CAAC;QAGf,YAAO,GAAG,IAAI,KAAK,EAAkC,CAAC;QACtD,cAAS,GAAG,IAAI,eAAe,CAAC,KAAK,CAAC,CAAC;QACvC,YAAO,GAAG;YACf,IAAI,EAAE,MAAM,CAAC,cAAc,EAAE,EAAC,QAAQ,EAAE,IAAI,EAAC,CAAC;SAC/C,CAAC;QAGA,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACxC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QACnC,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACjE,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,2BAA2B,EAAE,CAAC;QAErD,IAAI,CAAC,iCAAiC,EAAE,CAAC;QACzC,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5B,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC7B,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAEM,KAAK,CAAC,IAAI;QACf,uFAAuF;QACvF,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,MAAM,cAAc,CAAC,cAAc,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC,CAAC;QACrE,CAAC;QAED,2CAA2C;QAC3C,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAE3D,oEAAoE;QACpE,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAEnD,iGAAiG;QACjG,6FAA6F;QAC7F,IAAI,CAAC,aAAa,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;QAErD,8FAA8F;QAC9F,OAAO,IAAI,OAAO,CAAgB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACpD,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,GAAG,EAAE;gBAC9B,IAAI,CAAC,OAAO,YAAY,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC/E,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,kBAAkB;IACX,KAAK,CAAC,MAAkB;QAC7B,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC;QACtB,IAAI,CAAC,OAAO,EAAE,CAAC;IACjB,CAAC;IAED;;OAEG;IACH,IAAW,MAAM;QACf,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC;IAC9B,CAAC;IAED;;OAEG;IACH,IAAW,OAAO;QAChB,OAAO,IAAI,CAAC,QAAQ,CAAC,OAAO,IAAI,KAAK,CAAC;IACxC,CAAC;IAED,kBAAkB;IACX,KAAK;QACV,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YAC3B,IAAI,CAAC,aAAa,EAAE,QAAQ,CAAC,KAAK,EAAE,CAAC;QACvC,CAAC;IACH,CAAC;IAEM,cAAc,CAAC,MAAsC;QAC1D,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,OAAO;YACL,OAAO,EAAE,GAAG,EAAE;gBACZ,IAAI,IAAI,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;oBAC3B,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;gBAC1B,CAAC;YACH,CAAC;SACF,CAAC;IACJ,CAAC;IAEM,cAAc,CAAC,MAAsC;QAC1D,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;YACxB,MAAM,KAAK,CAAC,+EAA+E,CAAC,CAAC;QAC/F,CAAC;QACD,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,OAAO;YACL,OAAO,EAAE,GAAG,EAAE;gBACZ,IAAI,IAAI,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;oBAC3B,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;gBAC1B,CAAC;YACH,CAAC;SACF,CAAC;IACJ,CAAC;IAEM,cAAc,CAAC,MAAsC;QAC1D,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,MAAM,KAAK,CAAC,iFAAiF,CAAC,CAAC;QACjG,CAAC;QACD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC3C,OAAO;YACL,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,SAAS,KAAK,MAAM,CAAC;SACrF,CAAC;IACJ,CAAC;IAED;;OAEG;IACI,wBAAwB;QAC7B,OAAO,IAAI,CAAC,wBAAwB,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IACrD,CAAC;IAED,kBAAkB;IAClB,IAAW,QAAQ,CAAC,QAA2B;QAC7C,IAAI,CAAC,SAAS,GAAG,IAAI,KAAK,EAAU,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACvG,CAAC;IAED,kBAAkB;IAClB,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED,kBAAkB;IACX,KAAK;QACV,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;IACtB,CAAC;IAEO,YAAY;QAClB,OAAO,IAAI,eAAe,CAAC,wBAAwB,EAAE,IAAI,EAAE,QAAQ,CAAC,MAAM,CAAC;YACzE,MAAM,EAAE,MAAM,CAAC,QAAQ,CAAC;YACxB,SAAS,EAAE;gBACT,EAAC,OAAO,EAAE,gBAAgB,EAAE,QAAQ,EAAE,IAAI,EAAC;gBAC3C,EAAC,OAAO,EAAE,eAAe,EAAE,WAAW,EAAE,gBAAgB,EAAC;aAC1D;SACF,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;OAEG;IACK,aAAa;QACnB,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;QAChC,OAAO,OAAO,CAAC,MAAM,CAAC;YACpB,mBAAmB,EAAE,IAAI,EAAE,oDAAoD;YAC/E,UAAU,EAAE,CAAC,4BAA4B,CAAC;YAC1C,gBAAgB,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE;YAC7C,cAAc,EAAE,OAAO,CAAC,gBAAgB,CAAC,IAAI,EAAE;SAChD,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,oBAAoB;QAC1B,IAAI,CAAC,UAAU;aACZ,IAAI,CACH,MAAM,CAAC,OAAO,CAAC,EACf,kBAAkB,EAAE,CACrB;aACA,SAAS,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,KAAK,EAAE,CAAC;QACf,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;OAEG;IACK,qBAAqB;QAC3B,IAAI,CAAC,UAAU;aACZ,IAAI,CACH,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,EAC/B,kBAAkB,EAAE,CACrB;aACA,SAAS,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,KAAK,EAAE,CAAC;QACf,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;OAEG;IACK,2BAA2B;QACjC,IAAI,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC;QAC5C,CAAC;QACD,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,aAAa,KAAK,UAAU,EAAE,CAAC;YAC/D,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC;QAClB,CAAC;QACD,OAAO,MAAM,CAAC,qBAAqB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IACpE,CAAC;IAED;;OAEG;IACK,iCAAiC;QACvC,MAAM,eAAe,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QAChD,MAAM,SAAS,GAAG,KAAK,CACrB,eAAe,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,CAAC,EACpD,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC,EACnD,EAAE,CAAC,KAAK,CAAC,CACV,CAAC;QAEF,aAAa,CAAC,CAAC,IAAI,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;aACxC,IAAI,CACH,eAAe,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,EACvD,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CACrC;aACA,SAAS,CAAC,CAAC,CAAC,QAAQ,EAAE,QAAQ,CAAC,EAAE,EAAE;YAClC,MAAM,UAAU,GAAG,CAAC,QAAQ,IAAI,QAAQ,CAAC;YAEzC,6GAA6G;YAC7G,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,cAAc,EAAE,EAAC,UAAU,EAAE,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,EAAC,CAAC,CAAC;QACxF,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;OAEG;IACK,kBAAkB;QACxB,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,KAAK,aAAa,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,aAAa,KAAK,UAAU,EAAE,CAAC;YAC3G,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE,EAAC,KAAK,EAAE,GAAG,EAAC,CAAC,CAAC;QACvD,CAAC;aACI,CAAC;YACJ,MAAM,qBAAqB,GAAG,CAAC,IAAoB,EAA2B,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;YACrI,MAAM,qBAAqB,GAAG,GAA4B,EAAE,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC;YAC7I,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,qBAAqB,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,qBAAqB,EAAE,CAAC;YAE5G,YAAY;iBACT,IAAI,CACH,SAAS,CAAC,WAAW,CAAC,EAAE,CAAC,WAAW,CAAC,CAAC,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,EAC3E,GAAG,CAAC,CAAC,EAAC,OAAO,EAAE,WAAW,EAAC,EAAE,EAAE,CAAC,WAAW,CAAC,qBAAqB,EAAE,CAAC,EACpE,eAAe,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,EACvD,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CACrC;iBACA,SAAS,CAAC,CAAC,EAAC,GAAG,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,EAAC,EAAE,EAAE;gBACxC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE;oBACrC,GAAG,EAAE,GAAG,GAAG,IAAI;oBACf,IAAI,EAAE,GAAG,IAAI,IAAI;oBACjB,KAAK,EAAE,GAAG,KAAK,IAAI;oBACnB,MAAM,EAAE,GAAG,MAAM,IAAI;iBACtB,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC;IACH,CAAC;IAED;;OAEG;IACK,iBAAiB;QACvB,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,EAAC,CAAC;aAChE,IAAI,CACH,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,EACrC,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CACrC;aACA,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAChC,CAAC;IAEO,cAAc;QACpB,IAAI,CAAC,OAAO;aACT,IAAI,CACH,SAAS,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EACtE,oBAAoB,EAAE,EACtB,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CACrC;aACA,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC/B,CAAC;IAED;;OAEG;IACI,OAAO;QACZ,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC;YAChC,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;YAC3B,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;CACF","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {BehaviorSubject, combineLatest, concatWith, delay, EMPTY, firstValueFrom, map, merge, Observable, of, Subject, switchMap} from 'rxjs';\nimport {ComponentRef, inject, Injector, NgZone} from '@angular/core';\nimport {WorkbenchDialog, WorkbenchDialogSize} from './workbench-dialog';\nimport {WorkbenchDialogOptions} from './workbench-dialog.options';\nimport {ComponentPortal, ComponentType} from '@angular/cdk/portal';\nimport {Overlay, OverlayRef} from '@angular/cdk/overlay';\nimport {WorkbenchDialogComponent} from './workbench-dialog.component';\nimport {ɵWorkbenchView} from '../view/ɵworkbench-view.model';\nimport {WorkbenchDialogRegistry} from './workbench-dialog.registry';\nimport {ɵDestroyRef} from '../common/ɵdestroy-ref';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {setStyle} from '../common/dom.util';\nimport {fromDimension$} from '@scion/toolkit/observable';\nimport {subscribeInside} from '@scion/toolkit/operators';\nimport {ViewDragService} from '../view-dnd/view-drag.service';\nimport {WORKBENCH_ELEMENT_REF} from '../content-projection/view-container.reference';\nimport {Arrays} from '@scion/toolkit/util';\nimport {WorkbenchConfig} from '../workbench-config';\nimport {distinctUntilChanged, filter} from 'rxjs/operators';\nimport {WorkbenchDialogActionDirective} from './dialog-footer/workbench-dialog-action.directive';\nimport {WorkbenchDialogFooterDirective} from './dialog-footer/workbench-dialog-footer.directive';\nimport {WorkbenchDialogHeaderDirective} from './dialog-header/workbench-dialog-header.directive';\nimport {Disposable} from '../common/disposable';\nimport {Blockable} from '../glass-pane/blockable';\nimport {Blocking} from '../glass-pane/blocking';\nimport {UUID} from '@scion/toolkit/uuid';\n\n/** @inheritDoc */\nexport class ɵWorkbenchDialog<R = unknown> implements WorkbenchDialog<R>, Blockable, Blocking {\n\n  private readonly _overlayRef: OverlayRef;\n  private readonly _portal: ComponentPortal<WorkbenchDialogComponent>;\n  private readonly _workbenchDialogRegistry = inject(WorkbenchDialogRegistry);\n  private readonly _zone = inject(NgZone);\n  private readonly _workbenchConfig = inject(WorkbenchConfig);\n  private readonly _destroyRef = new ɵDestroyRef();\n  private readonly _attached$: Observable<boolean>;\n  private _blink$ = new Subject<void>();\n\n  /**\n   * Result (or error) to be passed to the dialog opener.\n   */\n  private _result: R | Error | undefined;\n  private _componentRef: ComponentRef<WorkbenchDialogComponent> | undefined;\n  private _cssClass: string;\n\n  /**\n   * Unique identity of this dialog.\n   */\n  public readonly id = UUID.randomUUID();\n  /**\n   * Indicates whether this dialog is blocked by other dialog(s) that overlay this dialog.\n   */\n  public readonly blockedBy$ = new BehaviorSubject<ɵWorkbenchDialog | null>(null);\n  public readonly size: WorkbenchDialogSize = {};\n  public title: string | Observable<string | undefined> | undefined;\n  public closable = true;\n  public resizable = true;\n  public padding = true;\n  public header: WorkbenchDialogHeaderDirective | undefined;\n  public footer: WorkbenchDialogFooterDirective | undefined;\n  public actions = new Array<WorkbenchDialogActionDirective>();\n  public blinking$ = new BehaviorSubject(false);\n  public context = {\n    view: inject(ɵWorkbenchView, {optional: true}),\n  };\n\n  constructor(public component: ComponentType<unknown>, public _options: WorkbenchDialogOptions) {\n    this._overlayRef = this.createOverlay();\n    this._portal = this.createPortal();\n    this._cssClass = Arrays.coerce(this._options.cssClass).join(' ');\n    this._attached$ = this.monitorHostElementAttached$();\n\n    this.hideOnHostElementDetachOrViewDrag();\n    this.stickToHostElement();\n    this.blockWhenNotOnTop();\n    this.restoreFocusOnAttach();\n    this.restoreFocusOnUnblock();\n    this.blinkOnRequest();\n  }\n\n  public async open(): Promise<R | undefined> {\n    // Wait for the overlay to be initially positioned to have a smooth slide-in animation.\n    if (this.animate) {\n      await firstValueFrom(fromDimension$(this._overlayRef.hostElement));\n    }\n\n    // Attach the dialog portal to the overlay.\n    this._componentRef = this._overlayRef.attach(this._portal);\n\n    // Ensure to destroy this handle on browser back/forward navigation.\n    this._componentRef.onDestroy(() => this.destroy());\n\n    // Trigger a manual change detection cycle to avoid 'ExpressionChangedAfterItHasBeenCheckedError'\n    // when the dialog sets dialog-specific properties such as title or size during construction.\n    this._componentRef.changeDetectorRef.detectChanges();\n\n    // Wait for the dialog to close, resolving to its result or rejecting if closed with an error.\n    return new Promise<R | undefined>((resolve, reject) => {\n      this._destroyRef.onDestroy(() => {\n        this._result instanceof Error ? reject(this._result) : resolve(this._result);\n      });\n    });\n  }\n\n  /** @inheritDoc */\n  public close(result?: R | Error): void {\n    this._result = result;\n    this.destroy();\n  }\n\n  /**\n   * Inputs passed to the dialog.\n   */\n  public get inputs(): {[name: string]: unknown} | undefined {\n    return this._options.inputs;\n  }\n\n  /**\n   * Indicates if to animate the dialog.\n   */\n  public get animate(): boolean {\n    return this._options.animate ?? false;\n  }\n\n  /** @inheritDoc */\n  public focus(): void {\n    if (!this.blockedBy$.value) {\n      this._componentRef?.instance.focus();\n    }\n  }\n\n  public registerHeader(header: WorkbenchDialogHeaderDirective): Disposable {\n    this.header = header;\n    return {\n      dispose: () => {\n        if (this.header === header) {\n          this.header = undefined;\n        }\n      },\n    };\n  }\n\n  public registerFooter(footer: WorkbenchDialogFooterDirective): Disposable {\n    if (this.actions.length) {\n      throw Error('[DialogInitError] Custom dialog footer not supported if using dialog actions.');\n    }\n    this.footer = footer;\n    return {\n      dispose: () => {\n        if (this.footer === footer) {\n          this.footer = undefined;\n        }\n      },\n    };\n  }\n\n  public registerAction(action: WorkbenchDialogActionDirective): Disposable {\n    if (this.footer) {\n      throw Error('[DialogInitError] Dialog actions not supported if using a custom dialog footer.');\n    }\n    this.actions = this.actions.concat(action);\n    return {\n      dispose: () => this.actions = this.actions.filter(candidate => candidate !== action),\n    };\n  }\n\n  /**\n   * Returns the position of the dialog in the dialog stack.\n   */\n  public getPositionInDialogStack(): number {\n    return this._workbenchDialogRegistry.indexOf(this);\n  }\n\n  /** @inheritDoc */\n  public set cssClass(cssClass: string | string[]) {\n    this._cssClass = new Array<string>().concat(this._options.cssClass ?? []).concat(cssClass).join(' ');\n  }\n\n  /** @inheritDoc */\n  public get cssClass(): string {\n    return this._cssClass;\n  }\n\n  /** @inheritDoc */\n  public blink(): void {\n    this._blink$.next();\n  }\n\n  private createPortal(): ComponentPortal<WorkbenchDialogComponent> {\n    return new ComponentPortal(WorkbenchDialogComponent, null, Injector.create({\n      parent: inject(Injector),\n      providers: [\n        {provide: ɵWorkbenchDialog, useValue: this},\n        {provide: WorkbenchDialog, useExisting: ɵWorkbenchDialog},\n      ],\n    }));\n  }\n\n  /**\n   * Creates a dedicated overlay per dialog to place it on top of previously created overlays, such as dialogs, popups, dropdowns, etc.\n   */\n  private createOverlay(): OverlayRef {\n    const overlay = inject(Overlay);\n    return overlay.create({\n      disposeOnNavigation: true, // dispose dialog on browser back/forward navigation\n      panelClass: ['wb-dialog-modality-context'],\n      positionStrategy: overlay.position().global(),\n      scrollStrategy: overlay.scrollStrategies.noop(),\n    });\n  }\n\n  /**\n   * Restores focus when re-attaching this dialog.\n   */\n  private restoreFocusOnAttach(): void {\n    this._attached$\n      .pipe(\n        filter(Boolean),\n        takeUntilDestroyed(),\n      )\n      .subscribe(() => {\n        this.focus();\n      });\n  }\n\n  /**\n   * Restores focus when unblocking this dialog.\n   */\n  private restoreFocusOnUnblock(): void {\n    this.blockedBy$\n      .pipe(\n        filter(blockedBy => !blockedBy),\n        takeUntilDestroyed(),\n      )\n      .subscribe(() => {\n        this.focus();\n      });\n  }\n\n  /**\n   * Monitors attachment of the host element.\n   */\n  private monitorHostElementAttached$(): Observable<boolean> {\n    if (this.context.view) {\n      return this.context.view.portal.attached$;\n    }\n    if (this._workbenchConfig.dialog?.modalityScope === 'viewport') {\n      return of(true);\n    }\n    return inject(WORKBENCH_ELEMENT_REF).ref$.pipe(map(ref => !!ref));\n  }\n\n  /**\n   * Hides this dialog when either its host element is detached or during view drag and drop operation.\n   */\n  private hideOnHostElementDetachOrViewDrag(): void {\n    const viewDragService = inject(ViewDragService);\n    const viewDrag$ = merge(\n      viewDragService.viewDragStart$.pipe(map(() => true)),\n      viewDragService.viewDragEnd$.pipe(map(() => false)),\n      of(false), // to initialize `combineLatest`\n    );\n\n    combineLatest([this._attached$, viewDrag$])\n      .pipe(\n        subscribeInside(fn => this._zone.runOutsideAngular(fn)),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(([attached, dragging]) => {\n        const hideDialog = !attached || dragging;\n\n        // Hide via `visibility: hidden` instead of `display: none` in order to preserve the dimension of the dialog.\n        setStyle(this._overlayRef.overlayElement, {visibility: hideDialog ? 'hidden' : null});\n      });\n  }\n\n  /**\n   * Aligns this dialog with the boundaries of the host element.\n   */\n  private stickToHostElement(): void {\n    if (this._options.modality === 'application' && this._workbenchConfig.dialog?.modalityScope === 'viewport') {\n      setStyle(this._overlayRef.hostElement, {inset: '0'});\n    }\n    else {\n      const workbenchViewElement$ = (view: ɵWorkbenchView): Observable<HTMLElement> => of(view.portal.componentRef.location.nativeElement);\n      const workbenchRootElement$ = (): Observable<HTMLElement> => inject(WORKBENCH_ELEMENT_REF).ref$.pipe(map(ref => ref?.element.nativeElement));\n      const hostElement$ = this.context.view ? workbenchViewElement$(this.context.view) : workbenchRootElement$();\n\n      hostElement$\n        .pipe(\n          switchMap(hostElement => hostElement ? fromDimension$(hostElement) : EMPTY),\n          map(({element: hostElement}) => hostElement.getBoundingClientRect()),\n          subscribeInside(fn => this._zone.runOutsideAngular(fn)),\n          takeUntilDestroyed(this._destroyRef),\n        )\n        .subscribe(({top, left, width, height}) => {\n          setStyle(this._overlayRef.hostElement, {\n            top: `${top}px`,\n            left: `${left}px`,\n            width: `${width}px`,\n            height: `${height}px`,\n          });\n        });\n    }\n  }\n\n  /**\n   * Blocks this dialog if not the topmost dialog in its context.\n   */\n  private blockWhenNotOnTop(): void {\n    this._workbenchDialogRegistry.top$({viewId: this.context.view?.id})\n      .pipe(\n        map(top => top === this ? null : top),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(this.blockedBy$);\n  }\n\n  private blinkOnRequest(): void {\n    this._blink$\n      .pipe(\n        switchMap(() => of(true).pipe(concatWith(of(false).pipe(delay(300))))),\n        distinctUntilChanged(),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(this.blinking$);\n  }\n\n  /**\n   * Destroys this dialog and associated resources.\n   */\n  public destroy(): void {\n    if (!this._destroyRef.destroyed) {\n      this._destroyRef.destroy();\n      this._overlayRef.dispose();\n    }\n  }\n}\n"]}
@@ -12,7 +12,7 @@ import { FormControl, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/fo
12
12
  import { noop } from 'rxjs';
13
13
  import { coerceBooleanProperty } from '@angular/cdk/coercion';
14
14
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
15
- import { randomUUID } from '../common/uuid.util';
15
+ import { UUID } from '@scion/toolkit/uuid';
16
16
  import * as i0 from "@angular/core";
17
17
  import * as i1 from "@angular/cdk/a11y";
18
18
  import * as i2 from "@angular/forms";
@@ -35,7 +35,7 @@ export class FilterFieldComponent {
35
35
  this._cd = _cd;
36
36
  this._cvaChangeFn = noop;
37
37
  this._cvaTouchedFn = noop;
38
- this.id = randomUUID();
38
+ this.id = UUID.randomUUID();
39
39
  /**
40
40
  * Emits on filter change.
41
41
  */
@@ -99,13 +99,13 @@ export class FilterFieldComponent {
99
99
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: FilterFieldComponent, deps: [{ token: i0.ElementRef }, { token: i1.FocusMonitor }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
100
100
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.2", type: FilterFieldComponent, isStandalone: true, selector: "wb-filter-field", inputs: { tabindex: "tabindex", placeholder: "placeholder", disabled: "disabled" }, outputs: { filter: "filter" }, host: { listeners: { "focus": "focus()" }, properties: { "attr.tabindex": "this.componentTabindex", "class.empty": "this.empty" } }, providers: [
101
101
  { provide: NG_VALUE_ACCESSOR, multi: true, useExisting: forwardRef(() => FilterFieldComponent) },
102
- ], viewQueries: [{ propertyName: "_inputElement", first: true, predicate: ["input"], descendants: true, static: true }], ngImport: i0, template: "<label [for]=\"id\" class=\"filter-icon scion-workbench-icons\">search</label>\n<input #input\n [attr.id]=\"id\"\n autocomplete=\"off\"\n [formControl]=\"formControl\"\n [tabindex]=\"tabindex ?? 0\"\n [placeholder]=\"placeholder ?? ''\">\n<button class=\"clear scion-workbench-icons\" tabindex=\"-1\" (click)=\"onClear()\">\n clear\n</button>\n", styles: [":host{display:inline-flex;flex-direction:row;gap:.5em;padding:.25em .5em}:host>label.filter-icon{flex:none;align-self:center;-webkit-user-select:none;user-select:none;font-size:1.5em}:host>input{all:unset;flex:auto;min-width:0}:host>button.clear{all:unset;flex:none;align-self:center;opacity:.75;cursor:pointer}:host>button.clear:hover{opacity:1}:host:not(:focus-within):not(:hover)>button.clear,:host:has(>input:disabled)>button.clear,:host.empty>button.clear{visibility:hidden}:host:has(>input:disabled)>label.filter-icon{color:var(--sci-color-text-subtlest)}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
102
+ ], viewQueries: [{ propertyName: "_inputElement", first: true, predicate: ["input"], descendants: true, static: true }], ngImport: i0, template: "<label [for]=\"id\" class=\"filter-icon scion-workbench-icons\">search</label>\n<input #input\n [attr.id]=\"id\"\n autocomplete=\"off\"\n [formControl]=\"formControl\"\n [tabindex]=\"tabindex ?? 0\"\n [placeholder]=\"placeholder ?? ''\">\n<button class=\"clear scion-workbench-icons\" tabindex=\"-1\" (click)=\"onClear()\">\n clear\n</button>\n", styles: [":host{display:inline-flex;flex-direction:row;gap:.5em;padding:.25em .5em}:host>label.filter-icon{flex:none;align-self:center;-webkit-user-select:none;user-select:none;font-size:1.5em}:host>input{all:unset;flex:auto;min-width:0}:host>input::placeholder{color:var(--sci-color-gray-400)}:host>button.clear{all:unset;flex:none;align-self:center;opacity:.75;cursor:pointer}:host>button.clear:hover{opacity:1}:host:not(:focus-within):not(:hover)>button.clear,:host:has(>input:disabled)>button.clear,:host.empty>button.clear{visibility:hidden}:host:has(>input:disabled)>label.filter-icon{color:var(--sci-color-text-subtlest)}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
103
103
  }
104
104
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: FilterFieldComponent, decorators: [{
105
105
  type: Component,
106
106
  args: [{ selector: 'wb-filter-field', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [ReactiveFormsModule], providers: [
107
107
  { provide: NG_VALUE_ACCESSOR, multi: true, useExisting: forwardRef(() => FilterFieldComponent) },
108
- ], template: "<label [for]=\"id\" class=\"filter-icon scion-workbench-icons\">search</label>\n<input #input\n [attr.id]=\"id\"\n autocomplete=\"off\"\n [formControl]=\"formControl\"\n [tabindex]=\"tabindex ?? 0\"\n [placeholder]=\"placeholder ?? ''\">\n<button class=\"clear scion-workbench-icons\" tabindex=\"-1\" (click)=\"onClear()\">\n clear\n</button>\n", styles: [":host{display:inline-flex;flex-direction:row;gap:.5em;padding:.25em .5em}:host>label.filter-icon{flex:none;align-self:center;-webkit-user-select:none;user-select:none;font-size:1.5em}:host>input{all:unset;flex:auto;min-width:0}:host>button.clear{all:unset;flex:none;align-self:center;opacity:.75;cursor:pointer}:host>button.clear:hover{opacity:1}:host:not(:focus-within):not(:hover)>button.clear,:host:has(>input:disabled)>button.clear,:host.empty>button.clear{visibility:hidden}:host:has(>input:disabled)>label.filter-icon{color:var(--sci-color-text-subtlest)}\n"] }]
108
+ ], template: "<label [for]=\"id\" class=\"filter-icon scion-workbench-icons\">search</label>\n<input #input\n [attr.id]=\"id\"\n autocomplete=\"off\"\n [formControl]=\"formControl\"\n [tabindex]=\"tabindex ?? 0\"\n [placeholder]=\"placeholder ?? ''\">\n<button class=\"clear scion-workbench-icons\" tabindex=\"-1\" (click)=\"onClear()\">\n clear\n</button>\n", styles: [":host{display:inline-flex;flex-direction:row;gap:.5em;padding:.25em .5em}:host>label.filter-icon{flex:none;align-self:center;-webkit-user-select:none;user-select:none;font-size:1.5em}:host>input{all:unset;flex:auto;min-width:0}:host>input::placeholder{color:var(--sci-color-gray-400)}:host>button.clear{all:unset;flex:none;align-self:center;opacity:.75;cursor:pointer}:host>button.clear:hover{opacity:1}:host:not(:focus-within):not(:hover)>button.clear,:host:has(>input:disabled)>button.clear,:host.empty>button.clear{visibility:hidden}:host:has(>input:disabled)>label.filter-icon{color:var(--sci-color-text-subtlest)}\n"] }]
109
109
  }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.FocusMonitor }, { type: i0.ChangeDetectorRef }], propDecorators: { tabindex: [{
110
110
  type: Input
111
111
  }], placeholder: [{
@@ -127,4 +127,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImpor
127
127
  type: HostListener,
128
128
  args: ['focus']
129
129
  }] } });
130
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-field.component.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/filter-field/filter-field.component.ts","../../../../../../projects/scion/workbench/src/lib/filter-field/filter-field.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n *  SPDX-License-Identifier: EPL-2.0\n */\n\nimport {ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, HostBinding, HostListener, Input, OnDestroy, Output, ViewChild} from '@angular/core';\nimport {ControlValueAccessor, FormControl, NG_VALUE_ACCESSOR, ReactiveFormsModule} from '@angular/forms';\nimport {noop} from 'rxjs';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {randomUUID} from '../common/uuid.util';\n\n/**\n * Provides a filter control.\n */\n@Component({\n  selector: 'wb-filter-field',\n  templateUrl: './filter-field.component.html',\n  styleUrls: ['./filter-field.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [ReactiveFormsModule],\n  providers: [\n    {provide: NG_VALUE_ACCESSOR, multi: true, useExisting: forwardRef(() => FilterFieldComponent)},\n  ],\n})\nexport class FilterFieldComponent implements ControlValueAccessor, OnDestroy {\n\n  private _cvaChangeFn: (value: any) => void = noop;\n  private _cvaTouchedFn: () => void = noop;\n\n  public readonly id = randomUUID();\n\n  /**\n   * Sets focus order in sequential keyboard navigation.\n   * If not specified, the focus order is according to the position in the document (tabindex=0).\n   */\n  @Input()\n  public tabindex?: number | undefined;\n\n  /**\n   * Specifies the hint displayed when this field is empty.\n   */\n  @Input()\n  public placeholder?: string | undefined;\n\n  @Input()\n  public set disabled(disabled: boolean | string | undefined | null) {\n    coerceBooleanProperty(disabled) ? this.formControl.disable() : this.formControl.enable();\n  }\n\n  public get disabled(): boolean {\n    return this.formControl.disabled;\n  }\n\n  /**\n   * Emits on filter change.\n   */\n  @Output()\n  public filter = new EventEmitter<string>();\n\n  @ViewChild('input', {static: true})\n  private _inputElement!: ElementRef<HTMLInputElement>;\n\n  @HostBinding('attr.tabindex')\n  public componentTabindex = -1; // component is not focusable in sequential keyboard navigation, but tabindex (if any) is installed on input field\n\n  @HostBinding('class.empty')\n  public get empty(): boolean {\n    return !this.formControl.value;\n  }\n\n  /* @docs-private */\n  public formControl: FormControl;\n\n  constructor(private _host: ElementRef,\n              private _focusManager: FocusMonitor,\n              private _cd: ChangeDetectorRef) {\n    this.formControl = new FormControl('', {updateOn: 'change', nonNullable: true});\n    this.formControl.valueChanges\n      .pipe(takeUntilDestroyed())\n      .subscribe(value => {\n        this._cvaChangeFn(value);\n        this.filter.emit(value);\n      });\n\n    this._focusManager.monitor(this._host.nativeElement, true)\n      .pipe(takeUntilDestroyed())\n      .subscribe((focusOrigin: FocusOrigin) => {\n        if (!focusOrigin) {\n          this._cvaTouchedFn(); // triggers form field validation and signals a blur event\n        }\n      });\n  }\n\n  @HostListener('focus')\n  public focus(): void {\n    this._inputElement.nativeElement.focus();\n  }\n\n  public onClear(): void {\n    this.formControl.setValue('');\n    this.focus(); // restore the focus\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public registerOnChange(fn: any): void {\n    this._cvaChangeFn = fn;\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public registerOnTouched(fn: any): void {\n    this._cvaTouchedFn = fn;\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this._cd.markForCheck();\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public writeValue(value: any): void {\n    this.formControl.setValue(value, {emitEvent: false});\n    this._cd.markForCheck();\n  }\n\n  public ngOnDestroy(): void {\n    this._focusManager.stopMonitoring(this._host.nativeElement);\n  }\n}\n","<label [for]=\"id\" class=\"filter-icon scion-workbench-icons\">search</label>\n<input #input\n       [attr.id]=\"id\"\n       autocomplete=\"off\"\n       [formControl]=\"formControl\"\n       [tabindex]=\"tabindex ?? 0\"\n       [placeholder]=\"placeholder ?? ''\">\n<button class=\"clear scion-workbench-icons\" tabindex=\"-1\" (click)=\"onClear()\">\n  clear\n</button>\n"]}
130
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-field.component.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/filter-field/filter-field.component.ts","../../../../../../projects/scion/workbench/src/lib/filter-field/filter-field.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n *  SPDX-License-Identifier: EPL-2.0\n */\n\nimport {ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, HostBinding, HostListener, Input, OnDestroy, Output, ViewChild} from '@angular/core';\nimport {ControlValueAccessor, FormControl, NG_VALUE_ACCESSOR, ReactiveFormsModule} from '@angular/forms';\nimport {noop} from 'rxjs';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {UUID} from '@scion/toolkit/uuid';\n\n/**\n * Provides a filter control.\n */\n@Component({\n  selector: 'wb-filter-field',\n  templateUrl: './filter-field.component.html',\n  styleUrls: ['./filter-field.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [ReactiveFormsModule],\n  providers: [\n    {provide: NG_VALUE_ACCESSOR, multi: true, useExisting: forwardRef(() => FilterFieldComponent)},\n  ],\n})\nexport class FilterFieldComponent implements ControlValueAccessor, OnDestroy {\n\n  private _cvaChangeFn: (value: any) => void = noop;\n  private _cvaTouchedFn: () => void = noop;\n\n  public readonly id = UUID.randomUUID();\n\n  /**\n   * Sets focus order in sequential keyboard navigation.\n   * If not specified, the focus order is according to the position in the document (tabindex=0).\n   */\n  @Input()\n  public tabindex?: number | undefined;\n\n  /**\n   * Specifies the hint displayed when this field is empty.\n   */\n  @Input()\n  public placeholder?: string | undefined;\n\n  @Input()\n  public set disabled(disabled: boolean | string | undefined | null) {\n    coerceBooleanProperty(disabled) ? this.formControl.disable() : this.formControl.enable();\n  }\n\n  public get disabled(): boolean {\n    return this.formControl.disabled;\n  }\n\n  /**\n   * Emits on filter change.\n   */\n  @Output()\n  public filter = new EventEmitter<string>();\n\n  @ViewChild('input', {static: true})\n  private _inputElement!: ElementRef<HTMLInputElement>;\n\n  @HostBinding('attr.tabindex')\n  public componentTabindex = -1; // component is not focusable in sequential keyboard navigation, but tabindex (if any) is installed on input field\n\n  @HostBinding('class.empty')\n  public get empty(): boolean {\n    return !this.formControl.value;\n  }\n\n  /* @docs-private */\n  public formControl: FormControl;\n\n  constructor(private _host: ElementRef,\n              private _focusManager: FocusMonitor,\n              private _cd: ChangeDetectorRef) {\n    this.formControl = new FormControl('', {updateOn: 'change', nonNullable: true});\n    this.formControl.valueChanges\n      .pipe(takeUntilDestroyed())\n      .subscribe(value => {\n        this._cvaChangeFn(value);\n        this.filter.emit(value);\n      });\n\n    this._focusManager.monitor(this._host.nativeElement, true)\n      .pipe(takeUntilDestroyed())\n      .subscribe((focusOrigin: FocusOrigin) => {\n        if (!focusOrigin) {\n          this._cvaTouchedFn(); // triggers form field validation and signals a blur event\n        }\n      });\n  }\n\n  @HostListener('focus')\n  public focus(): void {\n    this._inputElement.nativeElement.focus();\n  }\n\n  public onClear(): void {\n    this.formControl.setValue('');\n    this.focus(); // restore the focus\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public registerOnChange(fn: any): void {\n    this._cvaChangeFn = fn;\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public registerOnTouched(fn: any): void {\n    this._cvaTouchedFn = fn;\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this._cd.markForCheck();\n  }\n\n  /**\n   * Method implemented as part of `ControlValueAccessor` to work with Angular forms API\n   * @docs-private\n   */\n  public writeValue(value: any): void {\n    this.formControl.setValue(value, {emitEvent: false});\n    this._cd.markForCheck();\n  }\n\n  public ngOnDestroy(): void {\n    this._focusManager.stopMonitoring(this._host.nativeElement);\n  }\n}\n","<label [for]=\"id\" class=\"filter-icon scion-workbench-icons\">search</label>\n<input #input\n       [attr.id]=\"id\"\n       autocomplete=\"off\"\n       [formControl]=\"formControl\"\n       [tabindex]=\"tabindex ?? 0\"\n       [placeholder]=\"placeholder ?? ''\">\n<button class=\"clear scion-workbench-icons\" tabindex=\"-1\" (click)=\"onClear()\">\n  clear\n</button>\n"]}
@@ -12,7 +12,6 @@ import { MPart, MTreeNode } from '../workbench-layout.model';
12
12
  import { InstanceofPipe } from '../../common/instanceof.pipe';
13
13
  import { PortalModule } from '@angular/cdk/portal';
14
14
  import { PartPortalPipe } from '../../part/part-portal.pipe';
15
- import { NgFor, NgIf } from '@angular/common';
16
15
  import { SciSashboxComponent, SciSashDirective } from '@scion/components/sashbox';
17
16
  import { WorkbenchLayouts } from '../workbench-layouts.util';
18
17
  import * as i0 from "@angular/core";
@@ -38,18 +37,11 @@ export class GridElementComponent {
38
37
  this.MTreeNode = MTreeNode;
39
38
  this.MPart = MPart;
40
39
  this.children = new Array();
41
- /**
42
- * Each layout change creates new model object instances since the layout is deserialized from the URL.
43
- * Therefore, the "correct" track-by function is critical to help Angular identify DOM elements for reuse, which significantly can
44
- * improve performance. Note that we use the index instead of the object identity because a different identity may be computed for
45
- * nodes when re-arranging parts. Using the index is like not using *ngFor at all.
46
- */
47
- this.indexTrackByFn = (index) => index;
48
40
  }
49
41
  ngOnChanges(changes) {
50
42
  this.children = this.element instanceof MTreeNode ? this.computeChildren(this.element) : [];
51
- this.parentNodeId = this.element.parent?.nodeId;
52
- this.nodeId = this.element instanceof MTreeNode ? this.element.nodeId : undefined;
43
+ this.parentNodeId = this.element.parent?.id;
44
+ this.nodeId = this.element instanceof MTreeNode ? this.element.id : undefined;
53
45
  this.partId = this.element instanceof MPart ? this.element.id : undefined;
54
46
  }
55
47
  onSashStart() {
@@ -58,7 +50,7 @@ export class GridElementComponent {
58
50
  onSashEnd(treeNode, [sashSize1, sashSize2]) {
59
51
  const ratio = sashSize1 / (sashSize1 + sashSize2);
60
52
  this._workbenchLayoutService.notifyDragEnding();
61
- this._workbenchRouter.navigate(layout => layout.setSplitRatio(treeNode.nodeId, ratio)).then();
53
+ this._workbenchRouter.navigate(layout => layout.setSplitRatio(treeNode.id, ratio)).then();
62
54
  }
63
55
  computeChildren(treeNode) {
64
56
  const child1Visible = WorkbenchLayouts.isGridElementVisible(treeNode.child1);
@@ -79,19 +71,17 @@ export class GridElementComponent {
79
71
  return [];
80
72
  }
81
73
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: GridElementComponent, deps: [{ token: i1.ɵWorkbenchRouter }, { token: i2.WorkbenchLayoutService }], target: i0.ɵɵFactoryTarget.Component }); }
82
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.2", type: GridElementComponent, isStandalone: true, selector: "wb-grid-element", inputs: { element: "element" }, host: { properties: { "attr.data-parentnodeid": "this.parentNodeId", "attr.data-nodeid": "this.nodeId", "attr.data-partid": "this.partId" } }, usesOnChanges: true, ngImport: i0, template: "<!-- MPart (leaf) -->\n<ng-container *ngIf=\"element | wbInstanceof:MPart as part\" [cdkPortalOutlet]=\"part.id | wbPartPortal\"></ng-container>\n\n<!-- MTreeNode -->\n<ng-container *ngIf=\"element | wbInstanceof:MTreeNode as treeNode\">\n <!-- Node with a single visible child. -->\n <wb-grid-element *ngIf=\"children.length === 1\" [element]=\"children[0].element\"></wb-grid-element>\n\n <!-- Node with multiple visible children. -->\n <sci-sashbox *ngIf=\"children.length > 1\"\n [direction]=\"treeNode.direction\"\n [attr.data-nodeid]=\"treeNode.nodeId\"\n (sashStart)=\"onSashStart()\"\n (sashEnd)=\"onSashEnd(treeNode, $event)\">\n <ng-template *ngFor=\"let child of children; index as i; trackBy: indexTrackByFn\" sciSash [size]=\"child.size\">\n <wb-grid-element [element]=\"child.element\" [class]=\"'sash-' + (i + 1)\"></wb-grid-element>\n </ng-template>\n </sci-sashbox>\n</ng-container>\n", styles: [":host{display:grid}:host>sci-sashbox{z-index:auto;--sci-sashbox-gap: 0;--sci-sashbox-splitter-background-color: var(--sci-workbench-part-divider-color);--sci-sashbox-splitter-background-color-hover: var(--sci-workbench-part-divider-color-hover);--sci-sashbox-splitter-size: var(--sci-workbench-part-divider-size);--sci-sashbox-splitter-size-hover: var(--sci-workbench-part-divider-size-hover);--sci-sashbox-splitter-touch-target-size: var(--sci-workbench-part-divider-touch-target-size);--sci-sashbox-splitter-border-radius: 0;--sci-sashbox-splitter-opacity-active: var(--sci-workbench-part-divider-opacity-active);--sci-sashbox-splitter-opacity-hover: var(--sci-workbench-part-divider-opacity-hover)}\n"], dependencies: [{ kind: "component", type: GridElementComponent, selector: "wb-grid-element", inputs: ["element"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgFor, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "pipe", type: InstanceofPipe, name: "wbInstanceof" }, { kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { kind: "pipe", type: PartPortalPipe, name: "wbPartPortal" }, { kind: "component", type: SciSashboxComponent, selector: "sci-sashbox", inputs: ["direction"], outputs: ["sashStart", "sashEnd"] }, { kind: "directive", type: SciSashDirective, selector: "ng-template[sciSash]", inputs: ["size", "minSize"], exportAs: ["sciSash"] }] }); }
74
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.2", type: GridElementComponent, isStandalone: true, selector: "wb-grid-element", inputs: { element: "element" }, host: { properties: { "attr.data-parentnodeid": "this.parentNodeId", "attr.data-nodeid": "this.nodeId", "attr.data-partid": "this.partId" } }, usesOnChanges: true, ngImport: i0, template: "<!-- MPart (leaf) -->\n@if (element | wbInstanceof:MPart; as part) {\n <ng-container [cdkPortalOutlet]=\"part.id | wbPartPortal\"/>\n}\n\n<!-- MTreeNode -->\n@if (element | wbInstanceof:MTreeNode; as treeNode) {\n @if (children.length === 1) {\n <!-- Node with a single visible child. -->\n <wb-grid-element [element]=\"children[0].element\"/>\n }\n @if (children.length > 1) {\n <!-- Node with multiple visible children. -->\n <sci-sashbox [direction]=\"treeNode.direction\"\n [attr.data-nodeid]=\"treeNode.id\"\n (sashStart)=\"onSashStart()\"\n (sashEnd)=\"onSashEnd(treeNode, $event)\">\n @for (child of children; track child.element.id) {\n <ng-template sciSash [size]=\"child.size\">\n <wb-grid-element [element]=\"child.element\" [class]=\"'sash-' + ($index + 1)\"/>\n </ng-template>\n }\n </sci-sashbox>\n }\n}\n", styles: [":host{display:grid}:host>sci-sashbox{z-index:auto;--sci-sashbox-gap: 0;--sci-sashbox-splitter-background-color: var(--sci-workbench-part-divider-color);--sci-sashbox-splitter-background-color-hover: var(--sci-workbench-part-divider-color-hover);--sci-sashbox-splitter-size: var(--sci-workbench-part-divider-size);--sci-sashbox-splitter-size-hover: var(--sci-workbench-part-divider-size-hover);--sci-sashbox-splitter-touch-target-size: var(--sci-workbench-part-divider-touch-target-size);--sci-sashbox-splitter-border-radius: 0;--sci-sashbox-splitter-opacity-active: var(--sci-workbench-part-divider-opacity-active);--sci-sashbox-splitter-opacity-hover: var(--sci-workbench-part-divider-opacity-hover)}\n"], dependencies: [{ kind: "component", type: GridElementComponent, selector: "wb-grid-element", inputs: ["element"] }, { kind: "pipe", type: InstanceofPipe, name: "wbInstanceof" }, { kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { kind: "pipe", type: PartPortalPipe, name: "wbPartPortal" }, { kind: "component", type: SciSashboxComponent, selector: "sci-sashbox", inputs: ["direction"], outputs: ["sashStart", "sashEnd"] }, { kind: "directive", type: SciSashDirective, selector: "ng-template[sciSash]", inputs: ["size", "minSize"], exportAs: ["sciSash"] }] }); }
83
75
  }
84
76
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: GridElementComponent, decorators: [{
85
77
  type: Component,
86
78
  args: [{ selector: 'wb-grid-element', standalone: true, imports: [
87
- NgIf,
88
- NgFor,
89
79
  InstanceofPipe,
90
80
  PortalModule,
91
81
  PartPortalPipe,
92
82
  SciSashboxComponent,
93
83
  SciSashDirective,
94
- ], template: "<!-- MPart (leaf) -->\n<ng-container *ngIf=\"element | wbInstanceof:MPart as part\" [cdkPortalOutlet]=\"part.id | wbPartPortal\"></ng-container>\n\n<!-- MTreeNode -->\n<ng-container *ngIf=\"element | wbInstanceof:MTreeNode as treeNode\">\n <!-- Node with a single visible child. -->\n <wb-grid-element *ngIf=\"children.length === 1\" [element]=\"children[0].element\"></wb-grid-element>\n\n <!-- Node with multiple visible children. -->\n <sci-sashbox *ngIf=\"children.length > 1\"\n [direction]=\"treeNode.direction\"\n [attr.data-nodeid]=\"treeNode.nodeId\"\n (sashStart)=\"onSashStart()\"\n (sashEnd)=\"onSashEnd(treeNode, $event)\">\n <ng-template *ngFor=\"let child of children; index as i; trackBy: indexTrackByFn\" sciSash [size]=\"child.size\">\n <wb-grid-element [element]=\"child.element\" [class]=\"'sash-' + (i + 1)\"></wb-grid-element>\n </ng-template>\n </sci-sashbox>\n</ng-container>\n", styles: [":host{display:grid}:host>sci-sashbox{z-index:auto;--sci-sashbox-gap: 0;--sci-sashbox-splitter-background-color: var(--sci-workbench-part-divider-color);--sci-sashbox-splitter-background-color-hover: var(--sci-workbench-part-divider-color-hover);--sci-sashbox-splitter-size: var(--sci-workbench-part-divider-size);--sci-sashbox-splitter-size-hover: var(--sci-workbench-part-divider-size-hover);--sci-sashbox-splitter-touch-target-size: var(--sci-workbench-part-divider-touch-target-size);--sci-sashbox-splitter-border-radius: 0;--sci-sashbox-splitter-opacity-active: var(--sci-workbench-part-divider-opacity-active);--sci-sashbox-splitter-opacity-hover: var(--sci-workbench-part-divider-opacity-hover)}\n"] }]
84
+ ], template: "<!-- MPart (leaf) -->\n@if (element | wbInstanceof:MPart; as part) {\n <ng-container [cdkPortalOutlet]=\"part.id | wbPartPortal\"/>\n}\n\n<!-- MTreeNode -->\n@if (element | wbInstanceof:MTreeNode; as treeNode) {\n @if (children.length === 1) {\n <!-- Node with a single visible child. -->\n <wb-grid-element [element]=\"children[0].element\"/>\n }\n @if (children.length > 1) {\n <!-- Node with multiple visible children. -->\n <sci-sashbox [direction]=\"treeNode.direction\"\n [attr.data-nodeid]=\"treeNode.id\"\n (sashStart)=\"onSashStart()\"\n (sashEnd)=\"onSashEnd(treeNode, $event)\">\n @for (child of children; track child.element.id) {\n <ng-template sciSash [size]=\"child.size\">\n <wb-grid-element [element]=\"child.element\" [class]=\"'sash-' + ($index + 1)\"/>\n </ng-template>\n }\n </sci-sashbox>\n }\n}\n", styles: [":host{display:grid}:host>sci-sashbox{z-index:auto;--sci-sashbox-gap: 0;--sci-sashbox-splitter-background-color: var(--sci-workbench-part-divider-color);--sci-sashbox-splitter-background-color-hover: var(--sci-workbench-part-divider-color-hover);--sci-sashbox-splitter-size: var(--sci-workbench-part-divider-size);--sci-sashbox-splitter-size-hover: var(--sci-workbench-part-divider-size-hover);--sci-sashbox-splitter-touch-target-size: var(--sci-workbench-part-divider-touch-target-size);--sci-sashbox-splitter-border-radius: 0;--sci-sashbox-splitter-opacity-active: var(--sci-workbench-part-divider-opacity-active);--sci-sashbox-splitter-opacity-hover: var(--sci-workbench-part-divider-opacity-hover)}\n"] }]
95
85
  }], ctorParameters: () => [{ type: i1.ɵWorkbenchRouter }, { type: i2.WorkbenchLayoutService }], propDecorators: { parentNodeId: [{
96
86
  type: HostBinding,
97
87
  args: ['attr.data-parentnodeid']
@@ -118,4 +108,4 @@ function calculateSashSizes(ratio) {
118
108
  }
119
109
  return [`${1 / (1 - ratio)}`, `${1 / ratio}`];
120
110
  }
121
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grid-element.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/layout/grid-element/grid-element.component.ts","../../../../../../../projects/scion/workbench/src/lib/layout/grid-element/grid-element.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {Component, HostBinding, Input, OnChanges, SimpleChanges, TrackByFunction} from '@angular/core';\nimport {MPart, MTreeNode} from '../workbench-layout.model';\nimport {ɵWorkbenchRouter} from '../../routing/ɵworkbench-router.service';\nimport {WorkbenchLayoutService} from '../workbench-layout.service';\nimport {InstanceofPipe} from '../../common/instanceof.pipe';\nimport {PortalModule} from '@angular/cdk/portal';\nimport {PartPortalPipe} from '../../part/part-portal.pipe';\nimport {NgFor, NgIf} from '@angular/common';\nimport {SciSashboxComponent, SciSashDirective} from '@scion/components/sashbox';\nimport {WorkbenchLayouts} from '../workbench-layouts.util';\n\n/**\n * Renders a {@link MTreeNode} or {@link MPart}.\n *\n * The workbench layout is modeled as a tree of nodes `{@link MTreeNode}` and parts `{@link MPart}`.\n * Each node has two children, which can be either another node or a part (leaf). A node defines\n * a split layout in which the two children are arranged vertically or horizontally.\n *\n * Nodes are rendered as {@link SciSashboxComponent} and parts as {@link PartComponent} or\n * {@link MainAreaLayoutComponent}.\n *\n * @see WorkbenchLayoutComponent\n */\n@Component({\n  selector: 'wb-grid-element',\n  templateUrl: './grid-element.component.html',\n  styleUrls: ['./grid-element.component.scss'],\n  standalone: true,\n  imports: [\n    NgIf,\n    NgFor,\n    InstanceofPipe,\n    PortalModule,\n    PartPortalPipe,\n    SciSashboxComponent,\n    SciSashDirective,\n  ],\n})\nexport class GridElementComponent implements OnChanges {\n\n  public MTreeNode = MTreeNode;\n  public MPart = MPart;\n\n  public children = new Array<ChildElement>();\n\n  @HostBinding('attr.data-parentnodeid')\n  public parentNodeId: string | undefined;\n\n  @HostBinding('attr.data-nodeid')\n  public nodeId: string | undefined;\n\n  @HostBinding('attr.data-partid')\n  public partId: string | undefined;\n\n  @Input({required: true})\n  public element!: MTreeNode | MPart;\n\n  constructor(private _workbenchRouter: ɵWorkbenchRouter, private _workbenchLayoutService: WorkbenchLayoutService) {\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this.children = this.element instanceof MTreeNode ? this.computeChildren(this.element) : [];\n    this.parentNodeId = this.element.parent?.nodeId;\n    this.nodeId = this.element instanceof MTreeNode ? this.element.nodeId : undefined;\n    this.partId = this.element instanceof MPart ? this.element.id : undefined;\n  }\n\n  public onSashStart(): void {\n    this._workbenchLayoutService.notifyDragStarting();\n  }\n\n  public onSashEnd(treeNode: MTreeNode, [sashSize1, sashSize2]: number[]): void {\n    const ratio = sashSize1 / (sashSize1 + sashSize2);\n    this._workbenchLayoutService.notifyDragEnding();\n    this._workbenchRouter.navigate(layout => layout.setSplitRatio(treeNode.nodeId, ratio)).then();\n  }\n\n  private computeChildren(treeNode: MTreeNode): ChildElement[] {\n    const child1Visible = WorkbenchLayouts.isGridElementVisible(treeNode.child1);\n    const child2Visible = WorkbenchLayouts.isGridElementVisible(treeNode.child2);\n\n    if (child1Visible && child2Visible) {\n      const [size1, size2] = calculateSashSizes(treeNode.ratio);\n      return [\n        {element: treeNode.child1, size: size1},\n        {element: treeNode.child2, size: size2},\n      ];\n    }\n    else if (child1Visible) {\n      return [{element: treeNode.child1}];\n    }\n    else if (child2Visible) {\n      return [{element: treeNode.child2}];\n    }\n    return [];\n  }\n\n  /**\n   * Each layout change creates new model object instances since the layout is deserialized from the URL.\n   * Therefore, the \"correct\" track-by function is critical to help Angular identify DOM elements for reuse, which significantly can\n   * improve performance. Note that we use the index instead of the object identity because a different identity may be computed for\n   * nodes when re-arranging parts. Using the index is like not using *ngFor at all.\n   */\n  public indexTrackByFn: TrackByFunction<ChildElement> = (index: number): number => index;\n}\n\n/**\n * Calculates the two sash proportions for the given ratio. Each proportion is >= 1.\n */\nfunction calculateSashSizes(ratio: number): [string, string] {\n  // Important: `SciSashboxComponent` requires proportions to be >= 1. For this reason we cannot simply calculate [ratio, 1 - ratio].\n  if (ratio === 0) {\n    return ['0px', '1'];\n  }\n  if (ratio === 1) {\n    return ['1', '0px'];\n  }\n\n  return [`${1 / (1 - ratio)}`, `${1 / ratio}`];\n}\n\n/**\n * Represents a visible child of a {@link MTreeNode}.\n */\ninterface ChildElement {\n  element: MTreeNode | MPart;\n  size?: string;\n}\n","<!-- MPart (leaf) -->\n<ng-container *ngIf=\"element | wbInstanceof:MPart as part\" [cdkPortalOutlet]=\"part.id | wbPartPortal\"></ng-container>\n\n<!-- MTreeNode -->\n<ng-container *ngIf=\"element | wbInstanceof:MTreeNode as treeNode\">\n  <!-- Node with a single visible child. -->\n  <wb-grid-element *ngIf=\"children.length === 1\" [element]=\"children[0].element\"></wb-grid-element>\n\n  <!-- Node with multiple visible children. -->\n  <sci-sashbox *ngIf=\"children.length > 1\"\n               [direction]=\"treeNode.direction\"\n               [attr.data-nodeid]=\"treeNode.nodeId\"\n               (sashStart)=\"onSashStart()\"\n               (sashEnd)=\"onSashEnd(treeNode, $event)\">\n    <ng-template *ngFor=\"let child of children; index as i; trackBy: indexTrackByFn\" sciSash [size]=\"child.size\">\n      <wb-grid-element [element]=\"child.element\" [class]=\"'sash-' + (i + 1)\"></wb-grid-element>\n    </ng-template>\n  </sci-sashbox>\n</ng-container>\n"]}
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grid-element.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/layout/grid-element/grid-element.component.ts","../../../../../../../projects/scion/workbench/src/lib/layout/grid-element/grid-element.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {Component, HostBinding, Input, OnChanges, SimpleChanges} from '@angular/core';\nimport {MPart, MTreeNode} from '../workbench-layout.model';\nimport {ɵWorkbenchRouter} from '../../routing/ɵworkbench-router.service';\nimport {WorkbenchLayoutService} from '../workbench-layout.service';\nimport {InstanceofPipe} from '../../common/instanceof.pipe';\nimport {PortalModule} from '@angular/cdk/portal';\nimport {PartPortalPipe} from '../../part/part-portal.pipe';\nimport {SciSashboxComponent, SciSashDirective} from '@scion/components/sashbox';\nimport {WorkbenchLayouts} from '../workbench-layouts.util';\n\n/**\n * Renders a {@link MTreeNode} or {@link MPart}.\n *\n * The workbench layout is modeled as a tree of nodes `{@link MTreeNode}` and parts `{@link MPart}`.\n * Each node has two children, which can be either another node or a part (leaf). A node defines\n * a split layout in which the two children are arranged vertically or horizontally.\n *\n * Nodes are rendered as {@link SciSashboxComponent} and parts as {@link PartComponent} or\n * {@link MainAreaLayoutComponent}.\n *\n * @see WorkbenchLayoutComponent\n */\n@Component({\n  selector: 'wb-grid-element',\n  templateUrl: './grid-element.component.html',\n  styleUrls: ['./grid-element.component.scss'],\n  standalone: true,\n  imports: [\n    InstanceofPipe,\n    PortalModule,\n    PartPortalPipe,\n    SciSashboxComponent,\n    SciSashDirective,\n  ],\n})\nexport class GridElementComponent implements OnChanges {\n\n  public MTreeNode = MTreeNode;\n  public MPart = MPart;\n\n  public children = new Array<ChildElement>();\n\n  @HostBinding('attr.data-parentnodeid')\n  public parentNodeId: string | undefined;\n\n  @HostBinding('attr.data-nodeid')\n  public nodeId: string | undefined;\n\n  @HostBinding('attr.data-partid')\n  public partId: string | undefined;\n\n  @Input({required: true})\n  public element!: MTreeNode | MPart;\n\n  constructor(private _workbenchRouter: ɵWorkbenchRouter, private _workbenchLayoutService: WorkbenchLayoutService) {\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this.children = this.element instanceof MTreeNode ? this.computeChildren(this.element) : [];\n    this.parentNodeId = this.element.parent?.id;\n    this.nodeId = this.element instanceof MTreeNode ? this.element.id : undefined;\n    this.partId = this.element instanceof MPart ? this.element.id : undefined;\n  }\n\n  public onSashStart(): void {\n    this._workbenchLayoutService.notifyDragStarting();\n  }\n\n  public onSashEnd(treeNode: MTreeNode, [sashSize1, sashSize2]: number[]): void {\n    const ratio = sashSize1 / (sashSize1 + sashSize2);\n    this._workbenchLayoutService.notifyDragEnding();\n    this._workbenchRouter.navigate(layout => layout.setSplitRatio(treeNode.id, ratio)).then();\n  }\n\n  private computeChildren(treeNode: MTreeNode): ChildElement[] {\n    const child1Visible = WorkbenchLayouts.isGridElementVisible(treeNode.child1);\n    const child2Visible = WorkbenchLayouts.isGridElementVisible(treeNode.child2);\n\n    if (child1Visible && child2Visible) {\n      const [size1, size2] = calculateSashSizes(treeNode.ratio);\n      return [\n        {element: treeNode.child1, size: size1},\n        {element: treeNode.child2, size: size2},\n      ];\n    }\n    else if (child1Visible) {\n      return [{element: treeNode.child1}];\n    }\n    else if (child2Visible) {\n      return [{element: treeNode.child2}];\n    }\n    return [];\n  }\n}\n\n/**\n * Calculates the two sash proportions for the given ratio. Each proportion is >= 1.\n */\nfunction calculateSashSizes(ratio: number): [string, string] {\n  // Important: `SciSashboxComponent` requires proportions to be >= 1. For this reason we cannot simply calculate [ratio, 1 - ratio].\n  if (ratio === 0) {\n    return ['0px', '1'];\n  }\n  if (ratio === 1) {\n    return ['1', '0px'];\n  }\n\n  return [`${1 / (1 - ratio)}`, `${1 / ratio}`];\n}\n\n/**\n * Represents a visible child of a {@link MTreeNode}.\n */\ninterface ChildElement {\n  element: MTreeNode | MPart;\n  size?: string;\n}\n","<!-- MPart (leaf) -->\n@if (element | wbInstanceof:MPart; as part) {\n  <ng-container [cdkPortalOutlet]=\"part.id | wbPartPortal\"/>\n}\n\n<!-- MTreeNode -->\n@if (element | wbInstanceof:MTreeNode; as treeNode) {\n  @if (children.length === 1) {\n    <!-- Node with a single visible child. -->\n    <wb-grid-element [element]=\"children[0].element\"/>\n  }\n  @if (children.length > 1) {\n    <!-- Node with multiple visible children. -->\n    <sci-sashbox [direction]=\"treeNode.direction\"\n                 [attr.data-nodeid]=\"treeNode.id\"\n                 (sashStart)=\"onSashStart()\"\n                 (sashEnd)=\"onSashEnd(treeNode, $event)\">\n      @for (child of children; track child.element.id) {\n        <ng-template sciSash [size]=\"child.size\">\n          <wb-grid-element [element]=\"child.element\" [class]=\"'sash-' + ($index + 1)\"/>\n        </ng-template>\n      }\n    </sci-sashbox>\n  }\n}\n"]}