@itwin/tree-widget-react 4.0.0-alpha.14 → 4.0.0-alpha.16

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 (194) hide show
  1. package/CHANGELOG.md +116 -1
  2. package/README.md +88 -11
  3. package/lib/esm/tree-widget-react/components/tree-header/SelectableTree.d.ts +4 -4
  4. package/lib/esm/tree-widget-react/components/tree-header/SelectableTree.js.map +1 -1
  5. package/lib/esm/tree-widget-react/components/tree-header/WidgetHeader.js.map +1 -1
  6. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTree.d.ts +2 -2
  7. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTree.js +2 -2
  8. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTree.js.map +1 -1
  9. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeButtons.d.ts +5 -5
  10. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeButtons.js +4 -2
  11. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeButtons.js.map +1 -1
  12. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeComponent.d.ts +8 -1
  13. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeComponent.js +2 -2
  14. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeComponent.js.map +1 -1
  15. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeDefinition.d.ts +4 -2
  16. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeDefinition.js +26 -20
  17. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeDefinition.js.map +1 -1
  18. package/lib/esm/tree-widget-react/components/trees/categories-tree/UseCategoriesTree.d.ts +4 -4
  19. package/lib/esm/tree-widget-react/components/trees/categories-tree/UseCategoriesTree.js +28 -17
  20. package/lib/esm/tree-widget-react/components/trees/categories-tree/UseCategoriesTree.js.map +1 -1
  21. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeIdsCache.d.ts +8 -7
  22. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeIdsCache.js +27 -15
  23. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeIdsCache.js.map +1 -1
  24. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseFilteredPaths.js +4 -3
  25. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseFilteredPaths.js.map +1 -1
  26. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/CategoriesTreeVisibilityHandler.d.ts +51 -0
  27. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/CategoriesTreeVisibilityHandler.js +283 -0
  28. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/CategoriesTreeVisibilityHandler.js.map +1 -0
  29. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/CategoriesTreeVisibilityHelper.d.ts +62 -0
  30. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/CategoriesTreeVisibilityHelper.js +84 -0
  31. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/CategoriesTreeVisibilityHelper.js.map +1 -0
  32. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/FilteredTree.d.ts +33 -0
  33. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/FilteredTree.js +218 -0
  34. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/visibility/FilteredTree.js.map +1 -0
  35. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTree.d.ts +2 -2
  36. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTree.js +2 -2
  37. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTree.js.map +1 -1
  38. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTreeComponent.d.ts +9 -2
  39. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTreeComponent.js +2 -2
  40. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTreeComponent.js.map +1 -1
  41. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTreeDefinition.d.ts +3 -2
  42. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTreeDefinition.js +1 -1
  43. package/lib/esm/tree-widget-react/components/trees/classifications-tree/ClassificationsTreeDefinition.js.map +1 -1
  44. package/lib/esm/tree-widget-react/components/trees/classifications-tree/UseClassificationsTree.d.ts +4 -4
  45. package/lib/esm/tree-widget-react/components/trees/classifications-tree/UseClassificationsTree.js +21 -8
  46. package/lib/esm/tree-widget-react/components/trees/classifications-tree/UseClassificationsTree.js.map +1 -1
  47. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeIdsCache.d.ts +12 -8
  48. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeIdsCache.js +41 -21
  49. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeIdsCache.js.map +1 -1
  50. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeNode.js.map +1 -1
  51. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/ClassificationsTreeVisibilityHandler.d.ts +37 -0
  52. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/ClassificationsTreeVisibilityHandler.js +170 -0
  53. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/ClassificationsTreeVisibilityHandler.js.map +1 -0
  54. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/ClassificationsTreeVisibilityHelper.d.ts +55 -0
  55. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/ClassificationsTreeVisibilityHelper.js +53 -0
  56. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/ClassificationsTreeVisibilityHelper.js.map +1 -0
  57. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/FilteredTree.d.ts +27 -0
  58. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/FilteredTree.js +165 -0
  59. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/visibility/FilteredTree.js.map +1 -0
  60. package/lib/esm/tree-widget-react/components/trees/common/CategoriesVisibilityUtils.d.ts +8 -4
  61. package/lib/esm/tree-widget-react/components/trees/common/CategoriesVisibilityUtils.js +22 -19
  62. package/lib/esm/tree-widget-react/components/trees/common/CategoriesVisibilityUtils.js.map +1 -1
  63. package/lib/esm/tree-widget-react/components/trees/common/TreeWidgetViewport.d.ts +186 -0
  64. package/lib/esm/tree-widget-react/components/trees/common/TreeWidgetViewport.js +79 -0
  65. package/lib/esm/tree-widget-react/components/trees/common/TreeWidgetViewport.js.map +1 -0
  66. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyVisibility.d.ts +11 -3
  67. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyVisibility.js +24 -19
  68. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyVisibility.js.map +1 -1
  69. package/lib/esm/tree-widget-react/components/trees/common/Utils.d.ts +12 -6
  70. package/lib/esm/tree-widget-react/components/trees/common/Utils.js +56 -13
  71. package/lib/esm/tree-widget-react/components/trees/common/Utils.js.map +1 -1
  72. package/lib/esm/tree-widget-react/components/trees/common/components/BaseTreeRenderer.d.ts +1 -1
  73. package/lib/esm/tree-widget-react/components/trees/common/components/BaseTreeRenderer.js +2 -2
  74. package/lib/esm/tree-widget-react/components/trees/common/components/BaseTreeRenderer.js.map +1 -1
  75. package/lib/esm/tree-widget-react/components/trees/common/components/EmptyTree.d.ts +6 -0
  76. package/lib/esm/tree-widget-react/components/trees/common/components/EmptyTree.js +4 -0
  77. package/lib/esm/tree-widget-react/components/trees/common/components/EmptyTree.js.map +1 -1
  78. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.d.ts +2 -3
  79. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.js +4 -4
  80. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.js.map +1 -1
  81. package/lib/esm/tree-widget-react/components/trees/common/components/TreeNodeVisibilityButton.d.ts +34 -8
  82. package/lib/esm/tree-widget-react/components/trees/common/components/TreeNodeVisibilityButton.js +26 -5
  83. package/lib/esm/tree-widget-react/components/trees/common/components/TreeNodeVisibilityButton.js.map +1 -1
  84. package/lib/esm/tree-widget-react/components/trees/common/components/TreeRenderer.d.ts +1 -1
  85. package/lib/esm/tree-widget-react/components/trees/common/components/TreeRenderer.js +7 -8
  86. package/lib/esm/tree-widget-react/components/trees/common/components/TreeRenderer.js.map +1 -1
  87. package/lib/esm/tree-widget-react/components/trees/common/components/UseVisibilityButtonHandler.d.ts +3 -3
  88. package/lib/esm/tree-widget-react/components/trees/common/components/UseVisibilityButtonHandler.js.map +1 -1
  89. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTree.d.ts +2 -2
  90. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTree.js.map +1 -1
  91. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTreeRenderer.d.ts +3 -3
  92. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTreeRenderer.js +10 -9
  93. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTreeRenderer.js.map +1 -1
  94. package/lib/esm/tree-widget-react/components/trees/common/internal/AlwaysAndNeverDrawnElementInfo.d.ts +6 -12
  95. package/lib/esm/tree-widget-react/components/trees/common/internal/AlwaysAndNeverDrawnElementInfo.js +53 -61
  96. package/lib/esm/tree-widget-react/components/trees/common/internal/AlwaysAndNeverDrawnElementInfo.js.map +1 -1
  97. package/lib/esm/tree-widget-react/components/trees/common/internal/ModelCategoryElementsCountCache.js +54 -34
  98. package/lib/esm/tree-widget-react/components/trees/common/internal/ModelCategoryElementsCountCache.js.map +1 -1
  99. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveTreeWidgetViewport.d.ts +6 -0
  100. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveTreeWidgetViewport.js +24 -0
  101. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveTreeWidgetViewport.js.map +1 -0
  102. package/lib/esm/tree-widget-react/components/trees/common/internal/Utils.d.ts +7 -17
  103. package/lib/esm/tree-widget-react/components/trees/common/internal/Utils.js +26 -16
  104. package/lib/esm/tree-widget-react/components/trees/common/internal/Utils.js.map +1 -1
  105. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityChangeEventListener.d.ts +3 -2
  106. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityChangeEventListener.js +10 -6
  107. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityChangeEventListener.js.map +1 -1
  108. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityUtils.d.ts +10 -29
  109. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityUtils.js +28 -64
  110. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityUtils.js.map +1 -1
  111. package/lib/esm/tree-widget-react/components/trees/common/internal/useTreeHooks/UseCachedVisibility.d.ts +53 -15
  112. package/lib/esm/tree-widget-react/components/trees/common/internal/useTreeHooks/UseCachedVisibility.js +142 -4
  113. package/lib/esm/tree-widget-react/components/trees/common/internal/useTreeHooks/UseCachedVisibility.js.map +1 -1
  114. package/lib/esm/tree-widget-react/components/trees/common/internal/visibility/BaseFilteredTree.d.ts +87 -0
  115. package/lib/esm/tree-widget-react/components/trees/common/internal/visibility/BaseFilteredTree.js +107 -0
  116. package/lib/esm/tree-widget-react/components/trees/common/internal/visibility/BaseFilteredTree.js.map +1 -0
  117. package/lib/esm/tree-widget-react/components/trees/common/internal/visibility/BaseVisibilityHelper.d.ts +235 -0
  118. package/lib/esm/tree-widget-react/components/trees/common/internal/visibility/BaseVisibilityHelper.js +584 -0
  119. package/lib/esm/tree-widget-react/components/trees/common/internal/visibility/BaseVisibilityHelper.js.map +1 -0
  120. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTree.d.ts +2 -2
  121. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTree.js +2 -2
  122. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTree.js.map +1 -1
  123. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeComponent.d.ts +1 -1
  124. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeComponent.js.map +1 -1
  125. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeDefinition.js +4 -0
  126. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeDefinition.js.map +1 -1
  127. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTree.d.ts +2 -2
  128. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTree.js +2 -2
  129. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTree.js.map +1 -1
  130. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeComponent.d.ts +1 -1
  131. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeComponent.js +1 -1
  132. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeComponent.js.map +1 -1
  133. package/lib/esm/tree-widget-react/components/trees/index.d.ts +4 -1
  134. package/lib/esm/tree-widget-react/components/trees/index.js +4 -0
  135. package/lib/esm/tree-widget-react/components/trees/index.js.map +1 -1
  136. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTree.d.ts +2 -2
  137. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTree.js +3 -2
  138. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTree.js.map +1 -1
  139. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeButtons.d.ts +3 -2
  140. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeButtons.js +12 -9
  141. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeButtons.js.map +1 -1
  142. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeComponent.d.ts +8 -1
  143. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeComponent.js +2 -2
  144. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeComponent.js.map +1 -1
  145. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeDefinition.d.ts +9 -9
  146. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeDefinition.js +37 -33
  147. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeDefinition.js.map +1 -1
  148. package/lib/esm/tree-widget-react/components/trees/models-tree/UseModelsTree.d.ts +28 -9
  149. package/lib/esm/tree-widget-react/components/trees/models-tree/UseModelsTree.js +31 -16
  150. package/lib/esm/tree-widget-react/components/trees/models-tree/UseModelsTree.js.map +1 -1
  151. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeIdsCache.d.ts +9 -7
  152. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeIdsCache.js +51 -51
  153. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeIdsCache.js.map +1 -1
  154. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseFilteredPaths.d.ts +14 -4
  155. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseFilteredPaths.js +112 -36
  156. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseFilteredPaths.js.map +1 -1
  157. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/FilteredTree.d.ts +24 -0
  158. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/FilteredTree.js +148 -0
  159. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/FilteredTree.js.map +1 -0
  160. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/ModelsTreeVisibilityHandler.d.ts +76 -0
  161. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/ModelsTreeVisibilityHandler.js +263 -0
  162. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/ModelsTreeVisibilityHandler.js.map +1 -0
  163. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/ModelsTreeVisibilityHelper.d.ts +53 -0
  164. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/ModelsTreeVisibilityHelper.js +71 -0
  165. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/visibility/ModelsTreeVisibilityHelper.js.map +1 -0
  166. package/lib/esm/tree-widget-react-internal.d.ts +2 -2
  167. package/lib/esm/tree-widget-react-internal.js +2 -2
  168. package/lib/esm/tree-widget-react-internal.js.map +1 -1
  169. package/lib/public/locales/en/TreeWidget.json +3 -0
  170. package/package.json +19 -18
  171. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeVisibilityHandler.d.ts +0 -23
  172. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeVisibilityHandler.js +0 -713
  173. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeVisibilityHandler.js.map +0 -1
  174. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/FilteredTree.d.ts +0 -39
  175. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/FilteredTree.js +0 -221
  176. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/FilteredTree.js.map +0 -1
  177. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeVisibilityHandler.d.ts +0 -21
  178. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeVisibilityHandler.js +0 -551
  179. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/ClassificationsTreeVisibilityHandler.js.map +0 -1
  180. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/FilteredTree.d.ts +0 -37
  181. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/FilteredTree.js +0 -193
  182. package/lib/esm/tree-widget-react/components/trees/classifications-tree/internal/FilteredTree.js.map +0 -1
  183. package/lib/esm/tree-widget-react/components/trees/common/UseNodeHighlighting.d.ts +0 -20
  184. package/lib/esm/tree-widget-react/components/trees/common/UseNodeHighlighting.js +0 -125
  185. package/lib/esm/tree-widget-react/components/trees/common/UseNodeHighlighting.js.map +0 -1
  186. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveViewport.d.ts +0 -7
  187. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveViewport.js +0 -21
  188. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveViewport.js.map +0 -1
  189. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/FilteredTree.d.ts +0 -23
  190. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/FilteredTree.js +0 -171
  191. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/FilteredTree.js.map +0 -1
  192. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeVisibilityHandler.d.ts +0 -80
  193. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeVisibilityHandler.js +0 -507
  194. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeVisibilityHandler.js.map +0 -1
@@ -0,0 +1,584 @@
1
+ /*---------------------------------------------------------------------------------------------
2
+ * Copyright (c) Bentley Systems, Incorporated. All rights reserved.
3
+ * See LICENSE.md in the project root for license terms and full copyright notice.
4
+ *--------------------------------------------------------------------------------------------*/
5
+ import { concat, concatAll, defaultIfEmpty, defer, EMPTY, filter, forkJoin, from, map, merge, mergeMap, of, reduce, shareReplay, startWith, Subject, take, takeUntil, tap, } from "rxjs";
6
+ import { assert, Id64 } from "@itwin/core-bentley";
7
+ import { createVisibilityStatus } from "../Tooltip.js";
8
+ import { getSetFromId64Arg, setDifference, setIntersection } from "../Utils.js";
9
+ import { changeElementStateNoChildrenOperator, enableCategoryDisplay, getVisibilityFromAlwaysAndNeverDrawnElementsImpl, mergeVisibilityStatuses, } from "../VisibilityUtils.js";
10
+ /**
11
+ * Base class for visibility status getters and modifiers.
12
+ *
13
+ * It provides methods that help retrieve and change visibility status of models, categories, elements.
14
+ * @internal
15
+ */
16
+ export class BaseVisibilityHelper {
17
+ #props;
18
+ #alwaysAndNeverDrawnElements;
19
+ #elementChangeQueue = new Subject();
20
+ #subscriptions = [];
21
+ constructor(props) {
22
+ this.#props = props;
23
+ this.#alwaysAndNeverDrawnElements = this.#props.alwaysAndNeverDrawnElementInfo;
24
+ this.#subscriptions.push(this.#elementChangeQueue.pipe(concatAll()).subscribe());
25
+ }
26
+ [Symbol.dispose]() {
27
+ this.#subscriptions.forEach((x) => x.unsubscribe());
28
+ }
29
+ /**
30
+ * Removes "always drawn exclusive" mode from the viewport without affecting any visibilities.
31
+ *
32
+ * This is achieved by:
33
+ * - Resets `alwaysDrawn` exclusive flag to `false`;
34
+ * - Turns off all categories;
35
+ * - Clears always drawn list;
36
+ * - Removes all per-model category overrides. */
37
+ removeAlwaysDrawnExclusive() {
38
+ return from(this.#props.baseIdsCache.getAllCategories()).pipe(map(({ drawingCategories, spatialCategories }) => {
39
+ const categoriesToTurnOff = this.#props.viewport.viewType === "2d" ? drawingCategories : spatialCategories;
40
+ if (categoriesToTurnOff) {
41
+ this.#props.viewport.changeCategoryDisplay({ categoryIds: categoriesToTurnOff, display: false, enableAllSubCategories: false });
42
+ }
43
+ this.#props.viewport.clearNeverDrawn();
44
+ this.#props.viewport.clearPerModelCategoryOverrides();
45
+ this.#props.viewport.setAlwaysDrawn({ elementIds: this.#props.viewport.alwaysDrawn ? new Set([...this.#props.viewport.alwaysDrawn]) : new Set() });
46
+ }));
47
+ }
48
+ /**
49
+ * Returns visibility status of models.
50
+ *
51
+ * Determines visibility status by checking:
52
+ * - Models visibility in the viewport;
53
+ * - Models' subModels visibility (if elements' modelId is in the provided modelIds, and element is itself a model, then it is considered a subModel);
54
+ * - Categories visibility in the viewport (if elements' modelId is in the provided modelIds, then its' category gets checked).
55
+ */
56
+ getModelsVisibilityStatus(props) {
57
+ const result = defer(() => {
58
+ const { modelIds, type } = props;
59
+ if ((type === "GeometricModel3d" && this.#props.viewport.viewType !== "3d") || (type === "GeometricModel2d" && this.#props.viewport.viewType !== "2d")) {
60
+ return of(createVisibilityStatus("disabled"));
61
+ }
62
+ return from(Id64.iterable(modelIds)).pipe(mergeMap((modelId) => {
63
+ // For hidden models we only need to check subModels
64
+ if (!this.#props.viewport.viewsModel(modelId)) {
65
+ return this.#props.baseIdsCache.getSubModels({ modelIds: modelId }).pipe(mergeMap(({ subModels }) => {
66
+ if (subModels && Id64.sizeOf(subModels) > 0) {
67
+ return this.getModelsVisibilityStatus({ modelIds: subModels, type }).pipe(map((subModelsVisibilityStatus) => subModelsVisibilityStatus.state !== "hidden" ? createVisibilityStatus("partial") : createVisibilityStatus("hidden")));
68
+ }
69
+ return of(createVisibilityStatus("hidden"));
70
+ }));
71
+ }
72
+ // For visible models we need to check all categories
73
+ return this.#props.baseIdsCache.getCategories({ modelIds: modelId }).pipe(mergeMap(({ drawingCategories, spatialCategories }) => merge(drawingCategories
74
+ ? of(drawingCategories).pipe(mergeMap((categoryIds) => this.getCategoriesVisibilityStatus({ modelId, categoryIds, type: "DrawingCategory" })))
75
+ : EMPTY, spatialCategories
76
+ ? of(spatialCategories).pipe(mergeMap((categoryIds) => this.getCategoriesVisibilityStatus({ modelId, categoryIds, type: "SpatialCategory" })))
77
+ : EMPTY)), defaultIfEmpty(createVisibilityStatus("visible")));
78
+ }), mergeVisibilityStatuses);
79
+ });
80
+ return this.#props.overrideHandler
81
+ ? this.#props.overrideHandler.createVisibilityHandlerResult({
82
+ overrideProps: { modelIds: props.modelIds },
83
+ nonOverriddenResult: result,
84
+ override: this.#props.overrides?.getModelsVisibilityStatus,
85
+ })
86
+ : result;
87
+ }
88
+ /**
89
+ * Gets visibility status of a model's categories, assuming model is visible.
90
+ *
91
+ * Determines visibility status by checking:
92
+ * - Elements in the viewports' always/never drawn lists;
93
+ * - Default categories visibility status in the viewport;
94
+ * - SubModels that are related to the modelId and categoryIds
95
+ */
96
+ getVisibleModelCategoriesVisibilityStatus({ modelId, categoryIds, type, }) {
97
+ return merge(this.getVisibilityFromAlwaysAndNeverDrawnElements({
98
+ queryProps: { modelId, categoryIds },
99
+ defaultStatus: () => this.getVisibleModelCategoriesDirectVisibilityStatus({ modelId, categoryIds }),
100
+ }), this.#props.baseIdsCache.getSubModels({ modelId, categoryIds }).pipe(mergeMap(({ subModels }) => {
101
+ if (subModels && Id64.sizeOf(subModels) > 0) {
102
+ return this.getModelsVisibilityStatus({ modelIds: subModels, type });
103
+ }
104
+ return EMPTY;
105
+ }))).pipe(mergeVisibilityStatuses);
106
+ }
107
+ /** Gets visibility status of sub-categories, assuming category is visible. */
108
+ getVisibleCategorySubCategoriesVisibilityStatus(props) {
109
+ const { subCategoryIds } = props;
110
+ let subCategoryVisibility = "unknown";
111
+ for (const subCategoryId of Id64.iterable(subCategoryIds)) {
112
+ const isSubCategoryVisible = this.#props.viewport.viewsSubCategory(subCategoryId);
113
+ if (isSubCategoryVisible && subCategoryVisibility === "hidden") {
114
+ return createVisibilityStatus("partial");
115
+ }
116
+ if (!isSubCategoryVisible && subCategoryVisibility === "visible") {
117
+ return createVisibilityStatus("partial");
118
+ }
119
+ subCategoryVisibility = isSubCategoryVisible ? "visible" : "hidden";
120
+ }
121
+ // If visibility is unknown, no subCategories were provided,
122
+ // Since category is visible we return visible
123
+ return createVisibilityStatus(subCategoryVisibility === "unknown" ? "visible" : subCategoryVisibility);
124
+ }
125
+ /**
126
+ * Gets visibility status of sub-categories.
127
+ *
128
+ * Determines visibility status by checking:
129
+ * - Models that contain the category visibility;
130
+ * - Per model category visibility overrides;
131
+ * - Category selector visibility in the viewport.
132
+ * - Sub-categories visibility in the viewport.
133
+ */
134
+ getSubCategoriesVisibilityStatus(props) {
135
+ return (props.modelId ? of({ id: props.categoryId, models: props.modelId }) : from(this.#props.baseIdsCache.getModels({ categoryIds: props.categoryId }))).pipe(map(({ models }) => {
136
+ let visibility = "unknown";
137
+ let nonDefaultModelDisplayStatesCount = 0;
138
+ for (const modelId of Id64.iterable(models ?? [])) {
139
+ if (!this.#props.viewport.viewsModel(modelId)) {
140
+ if (visibility === "visible") {
141
+ return createVisibilityStatus("partial");
142
+ }
143
+ visibility = "hidden";
144
+ ++nonDefaultModelDisplayStatesCount;
145
+ continue;
146
+ }
147
+ const override = this.#props.viewport.getPerModelCategoryOverride({ modelId, categoryId: props.categoryId });
148
+ if (override === "show") {
149
+ if (visibility === "hidden") {
150
+ return createVisibilityStatus("partial");
151
+ }
152
+ visibility = "visible";
153
+ ++nonDefaultModelDisplayStatesCount;
154
+ continue;
155
+ }
156
+ if (override === "hide") {
157
+ if (visibility === "visible") {
158
+ return createVisibilityStatus("partial");
159
+ }
160
+ visibility = "hidden";
161
+ ++nonDefaultModelDisplayStatesCount;
162
+ continue;
163
+ }
164
+ }
165
+ if (models && Id64.sizeOf(models) > 0 && nonDefaultModelDisplayStatesCount === Id64.sizeOf(models)) {
166
+ assert(visibility === "visible" || visibility === "hidden");
167
+ return createVisibilityStatus(visibility);
168
+ }
169
+ if (!this.#props.viewport.viewsCategory(props.categoryId)) {
170
+ return createVisibilityStatus(visibility === "visible" ? "partial" : "hidden");
171
+ }
172
+ if (Id64.sizeOf(props.subCategoryIds) === 0) {
173
+ if (visibility === "hidden") {
174
+ return createVisibilityStatus("partial");
175
+ }
176
+ return createVisibilityStatus("visible");
177
+ }
178
+ const subCategoriesVisibility = this.getVisibleCategorySubCategoriesVisibilityStatus({ subCategoryIds: props.subCategoryIds });
179
+ return subCategoriesVisibility.state === visibility || visibility === "unknown" ? subCategoriesVisibility : createVisibilityStatus("partial");
180
+ }), mergeVisibilityStatuses);
181
+ }
182
+ /**
183
+ * Gets visibility status of categories.
184
+ *
185
+ * Determines visibility status by checking:
186
+ * - Categories visibility;
187
+ * - Visibility of models that are related to the categories;
188
+ * - sub-categories visibility.
189
+ */
190
+ getCategoriesVisibilityStatus(props) {
191
+ const result = defer(() => {
192
+ const { categoryIds, modelId: modelIdFromProps, type } = props;
193
+ if (Id64.sizeOf(categoryIds) === 0 || this.#props.viewport.viewType === "other") {
194
+ return EMPTY;
195
+ }
196
+ const isSupportedInView = (this.#props.viewport.viewType === "3d" && type === "SpatialCategory") || (this.#props.viewport.viewType === "2d" && type === "DrawingCategory");
197
+ if (!isSupportedInView) {
198
+ return of(createVisibilityStatus("disabled"));
199
+ }
200
+ return (modelIdFromProps
201
+ ? from(Id64.iterable(categoryIds)).pipe(map((categoryId) => ({ id: categoryId, models: modelIdFromProps })))
202
+ : this.#props.baseIdsCache.getModels({ categoryIds })).pipe(map(({ id, models }) => {
203
+ const acc = { categoryId: id, visibleModels: new Array(), hiddenModels: new Array() };
204
+ if (!models) {
205
+ return acc;
206
+ }
207
+ for (const modelId of Id64.iterable(models)) {
208
+ if (this.#props.viewport.viewsModel(modelId)) {
209
+ acc.visibleModels.push(modelId);
210
+ }
211
+ else {
212
+ acc.hiddenModels.push(modelId);
213
+ }
214
+ }
215
+ return acc;
216
+ }), mergeMap(({ categoryId, visibleModels, hiddenModels }) => {
217
+ return merge(
218
+ // For hidden models we only need to check subModels
219
+ hiddenModels.length > 0
220
+ ? this.#props.baseIdsCache.getSubModels({ modelIds: hiddenModels }).pipe(mergeMap(({ subModels }) => {
221
+ if (subModels && Id64.sizeOf(subModels) > 0) {
222
+ return this.getModelsVisibilityStatus({
223
+ modelIds: subModels,
224
+ type: this.#props.viewport.viewType === "2d" ? "GeometricModel2d" : "GeometricModel3d",
225
+ }).pipe(map((subModelsVisibilityStatus) => subModelsVisibilityStatus.state !== "hidden" ? createVisibilityStatus("partial") : createVisibilityStatus("hidden")));
226
+ }
227
+ return of(createVisibilityStatus("hidden"));
228
+ }))
229
+ : EMPTY,
230
+ // For visible models we need to check all categories
231
+ visibleModels.length > 0
232
+ ? from(visibleModels).pipe(mergeMap((modelId) => this.getVisibleModelCategoriesVisibilityStatus({
233
+ modelId,
234
+ categoryIds: categoryId,
235
+ type: this.#props.viewport.viewType === "2d" ? "GeometricModel2d" : "GeometricModel3d",
236
+ })))
237
+ : EMPTY,
238
+ // We need to check subCategories as well
239
+ this.#props.baseIdsCache.getSubCategories({ categoryIds: categoryId }).pipe(mergeMap(({ subCategories }) => {
240
+ if (subCategories && Id64.sizeOf(subCategories) > 0) {
241
+ return this.getSubCategoriesVisibilityStatus({ categoryId, modelId: modelIdFromProps, subCategoryIds: subCategories });
242
+ }
243
+ return EMPTY;
244
+ }))).pipe(defaultIfEmpty(createVisibilityStatus(this.#props.viewport.viewsCategory(categoryId) ? "visible" : "hidden")));
245
+ }), mergeVisibilityStatuses);
246
+ });
247
+ return this.#props.overrideHandler
248
+ ? this.#props.overrideHandler.createVisibilityHandlerResult({
249
+ overrideProps: props,
250
+ nonOverriddenResult: result,
251
+ override: this.#props.overrides?.getCategoriesVisibilityStatus,
252
+ })
253
+ : result;
254
+ }
255
+ /**
256
+ * Gets visibility status of categories, assuming model is visible.
257
+ *
258
+ * Determines visibility status by checking:
259
+ * - Per model category visibility overrides;
260
+ * - Category selector visibility in the viewport.
261
+ */
262
+ getVisibleModelCategoriesDirectVisibilityStatus({ modelId, categoryIds }) {
263
+ const viewport = this.#props.viewport;
264
+ let visibleCount = 0;
265
+ for (const categoryId of Id64.iterable(categoryIds)) {
266
+ const override = this.#props.viewport.getPerModelCategoryOverride({ modelId, categoryId });
267
+ if (override === "show" || (override === "none" && viewport.viewsCategory(categoryId))) {
268
+ ++visibleCount;
269
+ continue;
270
+ }
271
+ if (visibleCount > 0) {
272
+ return createVisibilityStatus("partial");
273
+ }
274
+ }
275
+ return visibleCount > 0 ? createVisibilityStatus("visible") : createVisibilityStatus("hidden");
276
+ }
277
+ /**
278
+ * Gets visibility status of elements.
279
+ *
280
+ * Determines visibility status by checking:
281
+ * - Elements in the viewports' always/never drawn lists;
282
+ * - Related categories and models visibility status;
283
+ * - Sub-models that are related to the specified elements.
284
+ */
285
+ getElementsVisibilityStatus(props) {
286
+ const result = defer(() => {
287
+ const { elementIds, modelId, categoryId, type } = props;
288
+ if (this.#props.viewport.viewType === "other") {
289
+ return EMPTY;
290
+ }
291
+ const isSupportedInView = (this.#props.viewport.viewType === "3d" && type === "GeometricElement3d") || (this.#props.viewport.viewType === "2d" && type === "GeometricElement2d");
292
+ if (!isSupportedInView) {
293
+ return of(createVisibilityStatus("disabled"));
294
+ }
295
+ // TODO: check child elements that are subModels
296
+ if (!this.#props.viewport.viewsModel(modelId)) {
297
+ return from(elementIds).pipe(mergeMap((elementId) => from(this.#props.baseIdsCache.hasSubModel(elementId)).pipe(mergeMap((isSubModel) => {
298
+ if (isSubModel) {
299
+ return this.getModelsVisibilityStatus({
300
+ modelIds: elementId,
301
+ type: this.#props.viewport.viewType === "2d" ? "GeometricModel2d" : "GeometricModel3d",
302
+ }).pipe(map((subModelVisibilityStatus) => subModelVisibilityStatus.state !== "hidden" ? createVisibilityStatus("partial") : createVisibilityStatus("hidden")));
303
+ }
304
+ return of(createVisibilityStatus("hidden"));
305
+ }))), mergeVisibilityStatuses);
306
+ }
307
+ // TODO: check child elements
308
+ // TODO: check child element categories
309
+ // TODO: check child elements that are subModels
310
+ return this.getVisibilityFromAlwaysAndNeverDrawnElements({
311
+ elements: elementIds,
312
+ defaultStatus: () => this.getVisibleModelCategoriesDirectVisibilityStatus({ categoryIds: categoryId, modelId }),
313
+ }).pipe(mergeMap((visibilityStatusAlwaysAndNeverDraw) => {
314
+ return from(Id64.iterable(elementIds)).pipe(mergeMap((elementId) => from(this.#props.baseIdsCache.hasSubModel(elementId)).pipe(mergeMap((isSubModel) => {
315
+ if (isSubModel) {
316
+ return this.getModelsVisibilityStatus({
317
+ modelIds: elementId,
318
+ type: this.#props.viewport.viewType === "2d" ? "GeometricModel2d" : "GeometricModel3d",
319
+ }).pipe(map((subModelVisibilityStatus) => subModelVisibilityStatus.state !== visibilityStatusAlwaysAndNeverDraw.state
320
+ ? createVisibilityStatus("partial")
321
+ : visibilityStatusAlwaysAndNeverDraw));
322
+ }
323
+ return of(visibilityStatusAlwaysAndNeverDraw);
324
+ }))), mergeVisibilityStatuses);
325
+ }));
326
+ });
327
+ return this.#props.overrideHandler
328
+ ? this.#props.overrideHandler.createVisibilityHandlerResult({
329
+ overrideProps: props,
330
+ nonOverriddenResult: result,
331
+ override: this.#props.overrides?.getElementsVisibilityStatus,
332
+ })
333
+ : result;
334
+ }
335
+ /** Gets visibility status of elements based on viewport's always/never drawn elements and related categories and models. */
336
+ getVisibilityFromAlwaysAndNeverDrawnElements(props) {
337
+ const viewport = this.#props.viewport;
338
+ if (viewport.isAlwaysDrawnExclusive) {
339
+ if (!viewport?.alwaysDrawn?.size) {
340
+ return of(createVisibilityStatus("hidden"));
341
+ }
342
+ }
343
+ else if (!viewport?.neverDrawn?.size && !viewport?.alwaysDrawn?.size) {
344
+ return of(props.defaultStatus());
345
+ }
346
+ if ("elements" in props) {
347
+ return of(getVisibilityFromAlwaysAndNeverDrawnElementsImpl({
348
+ ...props,
349
+ alwaysDrawn: viewport.alwaysDrawn?.size ? setIntersection(Id64.iterable(props.elements), viewport.alwaysDrawn) : undefined,
350
+ neverDrawn: viewport.neverDrawn?.size ? setIntersection(Id64.iterable(props.elements), viewport.neverDrawn) : undefined,
351
+ totalCount: Id64.sizeOf(props.elements),
352
+ viewport,
353
+ }));
354
+ }
355
+ const { modelId, categoryIds } = props.queryProps;
356
+ const totalCount = from(Id64.iterable(categoryIds)).pipe(mergeMap((categoryId) => this.#props.baseIdsCache.getElementsCount({ modelId, categoryId })), reduce((acc, specificModelCategoryCount) => {
357
+ return acc + specificModelCategoryCount;
358
+ }, 0));
359
+ return forkJoin({
360
+ totalCount,
361
+ alwaysDrawn: this.#alwaysAndNeverDrawnElements.getAlwaysDrawnElements(props.queryProps),
362
+ neverDrawn: this.#alwaysAndNeverDrawnElements.getNeverDrawnElements(props.queryProps),
363
+ }).pipe(
364
+ // There is a known bug:
365
+ // Categories that don't have root elements will make visibility result incorrect
366
+ // E.g.:
367
+ // - CategoryA
368
+ // - ElementA (CategoryA is visible)
369
+ // - ChildElementB (CategoryB is hidden) ChildElementB is in always drawn list
370
+ // Result will be "partial" because CategoryB will return hidden visibility, even though all elements are visible
371
+ // TODO fix with: https://github.com/iTwin/viewer-components-react/issues/1100
372
+ map((state) => {
373
+ return getVisibilityFromAlwaysAndNeverDrawnElementsImpl({
374
+ ...props,
375
+ ...state,
376
+ viewport,
377
+ });
378
+ }));
379
+ }
380
+ /**
381
+ * Changes visibility status of models.
382
+ *
383
+ * Also, changes visibility status of related categories and sub-models.
384
+ */
385
+ changeModelsVisibilityStatus(props) {
386
+ const result = defer(() => {
387
+ const { modelIds, on } = props;
388
+ if (Id64.sizeOf(modelIds) === 0) {
389
+ return EMPTY;
390
+ }
391
+ const viewport = this.#props.viewport;
392
+ viewport.clearPerModelCategoryOverrides({ modelIds });
393
+ if (!on) {
394
+ viewport.changeModelDisplay({ modelIds, display: false });
395
+ return this.#props.baseIdsCache
396
+ .getSubModels({ modelIds })
397
+ .pipe(mergeMap(({ subModels }) => (subModels ? this.changeModelsVisibilityStatus({ modelIds: subModels, on }) : EMPTY)));
398
+ }
399
+ viewport.changeModelDisplay({ modelIds, display: true });
400
+ return this.#props.baseIdsCache.getCategories({ modelIds }).pipe(mergeMap(({ id, drawingCategories, spatialCategories }) => {
401
+ return merge(drawingCategories ? this.changeCategoriesVisibilityStatus({ categoryIds: drawingCategories, modelId: id, on }) : EMPTY, spatialCategories ? this.changeCategoriesVisibilityStatus({ categoryIds: spatialCategories, modelId: id, on }) : EMPTY);
402
+ }));
403
+ });
404
+ return this.#props.overrideHandler
405
+ ? this.#props.overrideHandler.createVisibilityHandlerResult({
406
+ overrideProps: props,
407
+ nonOverriddenResult: result,
408
+ override: this.#props.overrides?.changeModelsVisibilityStatus,
409
+ })
410
+ : result;
411
+ }
412
+ /** Turns model on and turns off elements with categories related to that model. */
413
+ showModelWithoutAnyCategoriesOrElements(modelId, categoriesToNotOverride) {
414
+ const viewport = this.#props.viewport;
415
+ return forkJoin({
416
+ allModelCategories: this.#props.baseIdsCache.getCategories({ modelIds: modelId }).pipe(reduce((acc, { drawingCategories, spatialCategories }) => {
417
+ for (const category of Id64.iterable(drawingCategories ?? [])) {
418
+ acc.add(category);
419
+ }
420
+ for (const category of Id64.iterable(spatialCategories ?? [])) {
421
+ acc.add(category);
422
+ }
423
+ return acc;
424
+ }, new Set())),
425
+ modelAlwaysDrawnElements: this.#alwaysAndNeverDrawnElements.getAlwaysDrawnElements({ modelId }),
426
+ }).pipe(mergeMap(async ({ allModelCategories, modelAlwaysDrawnElements }) => {
427
+ const alwaysDrawn = this.#props.viewport.alwaysDrawn;
428
+ if (alwaysDrawn && modelAlwaysDrawnElements) {
429
+ viewport.setAlwaysDrawn({ elementIds: setDifference(alwaysDrawn, modelAlwaysDrawnElements) });
430
+ }
431
+ const categoriesToOverride = categoriesToNotOverride
432
+ ? setDifference(allModelCategories, getSetFromId64Arg(categoriesToNotOverride))
433
+ : allModelCategories;
434
+ categoriesToOverride.forEach((categoryId) => {
435
+ this.changeCategoryStateInViewportAccordingToModelVisibility(modelId, categoryId, false, false);
436
+ });
437
+ viewport.changeModelDisplay({ modelIds: modelId, display: true });
438
+ }));
439
+ }
440
+ /** Adds per-model category overrides based on category visibility in category selector. */
441
+ changeCategoryStateInViewportAccordingToModelVisibility(modelId, categoryId, on, changeSubCategories) {
442
+ const viewport = this.#props.viewport;
443
+ const isDisplayedInSelector = viewport.viewsCategory(categoryId);
444
+ const override = on === isDisplayedInSelector ? "none" : on ? "show" : "hide";
445
+ viewport.setPerModelCategoryOverride({ modelIds: modelId, categoryIds: categoryId, override });
446
+ if (override === "none" && on) {
447
+ // we took off the override which means the category is displayed in selector, but
448
+ // doesn't mean all its subcategories are displayed - this call ensures that
449
+ viewport.changeCategoryDisplay({ categoryIds: categoryId, display: true, enableAllSubCategories: changeSubCategories });
450
+ }
451
+ }
452
+ /**
453
+ * Changes categories visibility status.
454
+ *
455
+ * Also:
456
+ * - Turns on models in cases where categories need to be turned on and models are not already on.
457
+ * - Removed related elements from always/never drawn elements.
458
+ * - changes visibility of sub-models that are related to the specified categories.
459
+ */
460
+ changeCategoriesVisibilityStatus(props) {
461
+ const result = defer(() => {
462
+ const { modelId: modelIdFromProps, categoryIds, on } = props;
463
+ const viewport = this.#props.viewport;
464
+ const modelIdsObservable = (modelIdFromProps
465
+ ? of(new Map([[modelIdFromProps, getSetFromId64Arg(categoryIds)]]))
466
+ : this.#props.baseIdsCache.getModels({ categoryIds }).pipe(reduce((acc, { id, models }) => {
467
+ if (!models) {
468
+ return acc;
469
+ }
470
+ for (const modelId of Id64.iterable(models)) {
471
+ let entry = acc.get(modelId);
472
+ if (!entry) {
473
+ entry = new Set();
474
+ acc.set(modelId, entry);
475
+ }
476
+ entry.add(id);
477
+ }
478
+ return acc;
479
+ }, new Map()))).pipe(mergeMap((modelCategoriesMap) => modelCategoriesMap.entries()), shareReplay());
480
+ return concat(
481
+ // If modelId was provided: add override
482
+ // If modelId was not provided: change categoryDisplay and remove categories per model overrides
483
+ modelIdFromProps
484
+ ? of(viewport.setPerModelCategoryOverride({
485
+ modelIds: modelIdFromProps,
486
+ categoryIds,
487
+ override: on ? "show" : "hide",
488
+ }))
489
+ : concat(from(enableCategoryDisplay(viewport, categoryIds, on, on)), modelIdsObservable.pipe(map(([modelId, modelCategories]) => {
490
+ viewport.setPerModelCategoryOverride({ modelIds: modelId, categoryIds: modelCategories, override: "none" });
491
+ }))),
492
+ // If categories visibility needs to be turned on, we need to turn on models without turning on unrelated elements or categories for that model
493
+ on
494
+ ? modelIdsObservable.pipe(mergeMap(([modelId, categories]) => {
495
+ if (!viewport.viewsModel(modelId)) {
496
+ return this.showModelWithoutAnyCategoriesOrElements(modelId, categories);
497
+ }
498
+ return EMPTY;
499
+ }))
500
+ : EMPTY, this.#alwaysAndNeverDrawnElements.clearAlwaysAndNeverDrawnElements({ categoryIds, modelId: modelIdFromProps }), this.#props.baseIdsCache
501
+ .getSubModels({ categoryIds, modelId: modelIdFromProps })
502
+ .pipe(mergeMap(({ subModels }) => (subModels ? this.changeModelsVisibilityStatus({ modelIds: subModels, on }) : EMPTY))));
503
+ });
504
+ return this.#props.overrideHandler
505
+ ? this.#props.overrideHandler.createVisibilityHandlerResult({
506
+ overrideProps: props,
507
+ nonOverriddenResult: result,
508
+ override: this.#props.overrides?.changeCategoriesVisibilityStatus,
509
+ })
510
+ : result;
511
+ }
512
+ /**
513
+ * Changes visibility status of elements by adding them to the viewport's always/never drawn elements.
514
+ *
515
+ * Also, changes visibility status of specified elements that are models.
516
+ */
517
+ changeElementsVisibilityStatus(props) {
518
+ const result = defer(() => {
519
+ const { modelId, categoryId, elementIds, on } = props;
520
+ const viewport = this.#props.viewport;
521
+ // TODO: change child elements
522
+ // TODO: change child element categories
523
+ // TODO: change child subModels
524
+ return concat(
525
+ // Change elements state
526
+ defer(() => {
527
+ if (!viewport.viewsModel(modelId)) {
528
+ if (!on) {
529
+ return this.queueElementsVisibilityChange(elementIds, on, false);
530
+ }
531
+ return this.showModelWithoutAnyCategoriesOrElements(modelId).pipe(mergeMap(() => {
532
+ const defaultVisibility = this.getVisibleModelCategoriesDirectVisibilityStatus({
533
+ categoryIds: categoryId,
534
+ modelId,
535
+ });
536
+ const displayedByDefault = defaultVisibility.state === "visible";
537
+ return this.queueElementsVisibilityChange(elementIds, on, displayedByDefault);
538
+ }));
539
+ }
540
+ const categoryVisibility = this.getVisibleModelCategoriesDirectVisibilityStatus({ categoryIds: categoryId, modelId });
541
+ const isDisplayedByDefault = categoryVisibility.state === "visible";
542
+ return this.queueElementsVisibilityChange(elementIds, on, isDisplayedByDefault);
543
+ }),
544
+ // Change visibility of elements that are models
545
+ from(Id64.iterable(elementIds)).pipe(mergeMap((elementId) => from(this.#props.baseIdsCache.hasSubModel(elementId)).pipe(mergeMap((isSubModel) => {
546
+ if (isSubModel) {
547
+ return this.changeModelsVisibilityStatus({ modelIds: elementId, on });
548
+ }
549
+ return EMPTY;
550
+ })))));
551
+ });
552
+ return this.#props.overrideHandler
553
+ ? this.#props.overrideHandler.createVisibilityHandlerResult({
554
+ overrideProps: props,
555
+ nonOverriddenResult: result,
556
+ override: this.#props.overrides?.changeElementsVisibilityStatus,
557
+ })
558
+ : result;
559
+ }
560
+ /** Queues visibility change for elements. */
561
+ queueElementsVisibilityChange(elementIds, on, visibleByDefault) {
562
+ const finishedSubject = new Subject();
563
+ // observable to track if visibility change is finished/cancelled
564
+ const changeFinished = finishedSubject.pipe(startWith(false), shareReplay(1), filter((finished) => finished));
565
+ const changeObservable = from(Id64.iterable(elementIds)).pipe(
566
+ // check if visibility change is not finished (cancelled) due to change overall change request being cancelled
567
+ takeUntil(changeFinished), changeElementStateNoChildrenOperator({ on, isDisplayedByDefault: visibleByDefault, viewport: this.#props.viewport }), tap({
568
+ next: () => {
569
+ // notify that visibility change is finished
570
+ finishedSubject.next(true);
571
+ },
572
+ }));
573
+ // queue visibility change. `changeObservable` will be subscribed to when other queue changes are finished
574
+ this.#elementChangeQueue.next(changeObservable);
575
+ // return observable that will emit when visibility change is finished
576
+ return changeFinished.pipe(take(1), tap({
577
+ unsubscribe: () => {
578
+ // if this observable is unsubscribed before visibility change is finished, we have to notify that it queued change request is cancelled
579
+ finishedSubject.next(true);
580
+ },
581
+ }), map(() => undefined));
582
+ }
583
+ }
584
+ //# sourceMappingURL=BaseVisibilityHelper.js.map