@itwin/tree-widget-react 4.0.0-alpha.6 → 4.0.0-alpha.8

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 (206) hide show
  1. package/CHANGELOG.md +22 -1
  2. package/README.md +2 -1
  3. package/lib/esm/tree-widget-react/TreeWidget.d.ts +6 -1
  4. package/lib/esm/tree-widget-react/TreeWidget.js +19 -2
  5. package/lib/esm/tree-widget-react/TreeWidget.js.map +1 -1
  6. package/lib/esm/tree-widget-react/components/TreeWidgetComponentImpl.d.ts +6 -0
  7. package/lib/esm/tree-widget-react/components/TreeWidgetComponentImpl.js +22 -23
  8. package/lib/esm/tree-widget-react/components/TreeWidgetComponentImpl.js.map +1 -1
  9. package/lib/esm/tree-widget-react/components/TreeWidgetUiItemsProvider.js +1 -1
  10. package/lib/esm/tree-widget-react/components/TreeWidgetUiItemsProvider.js.map +1 -1
  11. package/lib/esm/tree-widget-react/components/tree-header/ErrorState.js +1 -1
  12. package/lib/esm/tree-widget-react/components/tree-header/ErrorState.js.map +1 -1
  13. package/lib/esm/tree-widget-react/components/tree-header/SearchBox.d.ts +1 -1
  14. package/lib/esm/tree-widget-react/components/tree-header/SearchBox.js +4 -4
  15. package/lib/esm/tree-widget-react/components/tree-header/SearchBox.js.map +1 -1
  16. package/lib/esm/tree-widget-react/components/tree-header/SelectableTree.css +2 -0
  17. package/lib/esm/tree-widget-react/components/tree-header/WidgetHeader.css +1 -0
  18. package/lib/esm/tree-widget-react/components/tree-header/WidgetHeader.d.ts +12 -4
  19. package/lib/esm/tree-widget-react/components/tree-header/WidgetHeader.js +13 -9
  20. package/lib/esm/tree-widget-react/components/tree-header/WidgetHeader.js.map +1 -1
  21. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTree.d.ts +2 -2
  22. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTree.js +2 -2
  23. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTree.js.map +1 -1
  24. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeButtons.d.ts +9 -2
  25. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeButtons.js +53 -7
  26. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeButtons.js.map +1 -1
  27. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeComponent.d.ts +1 -1
  28. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeComponent.js +1 -1
  29. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeComponent.js.map +1 -1
  30. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeDefinition.d.ts +17 -3
  31. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeDefinition.js +561 -161
  32. package/lib/esm/tree-widget-react/components/trees/categories-tree/CategoriesTreeDefinition.js.map +1 -1
  33. package/lib/esm/tree-widget-react/components/trees/categories-tree/UseCategoriesTree.d.ts +6 -2
  34. package/lib/esm/tree-widget-react/components/trees/categories-tree/UseCategoriesTree.js +46 -117
  35. package/lib/esm/tree-widget-react/components/trees/categories-tree/UseCategoriesTree.js.map +1 -1
  36. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeIdsCache.d.ts +30 -15
  37. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeIdsCache.js +188 -38
  38. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeIdsCache.js.map +1 -1
  39. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeNode.d.ts +18 -3
  40. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeNode.js +30 -3
  41. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeNode.js.map +1 -1
  42. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeVisibilityHandler.d.ts +24 -0
  43. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeVisibilityHandler.js +701 -0
  44. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesTreeVisibilityHandler.js.map +1 -0
  45. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/FilteredTree.d.ts +39 -0
  46. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/FilteredTree.js +221 -0
  47. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/FilteredTree.js.map +1 -0
  48. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseFilteredPaths.d.ts +25 -0
  49. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseFilteredPaths.js +133 -0
  50. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseFilteredPaths.js.map +1 -0
  51. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseIdsCache.d.ts +8 -0
  52. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseIdsCache.js +48 -0
  53. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/UseIdsCache.js.map +1 -0
  54. package/lib/esm/tree-widget-react/components/trees/common/CategoriesVisibilityUtils.d.ts +5 -17
  55. package/lib/esm/tree-widget-react/components/trees/common/CategoriesVisibilityUtils.js +5 -62
  56. package/lib/esm/tree-widget-react/components/trees/common/CategoriesVisibilityUtils.js.map +1 -1
  57. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyVisibility.d.ts +2 -2
  58. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyVisibility.js +11 -5
  59. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyVisibility.js.map +1 -1
  60. package/lib/esm/tree-widget-react/components/trees/common/UseNodeHighlighting.d.ts +1 -0
  61. package/lib/esm/tree-widget-react/components/trees/common/UseNodeHighlighting.js +13 -11
  62. package/lib/esm/tree-widget-react/components/trees/common/UseNodeHighlighting.js.map +1 -1
  63. package/lib/esm/tree-widget-react/components/trees/common/UseTelemetryContext.d.ts +1 -0
  64. package/lib/esm/tree-widget-react/components/trees/common/UseTelemetryContext.js +2 -1
  65. package/lib/esm/tree-widget-react/components/trees/common/UseTelemetryContext.js.map +1 -1
  66. package/lib/esm/tree-widget-react/components/trees/common/Utils.d.ts +38 -23
  67. package/lib/esm/tree-widget-react/components/trees/common/Utils.js +65 -47
  68. package/lib/esm/tree-widget-react/components/trees/common/Utils.js.map +1 -1
  69. package/lib/esm/tree-widget-react/components/trees/common/components/BaseTreeRenderer.d.ts +1 -1
  70. package/lib/esm/tree-widget-react/components/trees/common/components/BaseTreeRenderer.js +4 -4
  71. package/lib/esm/tree-widget-react/components/trees/common/components/BaseTreeRenderer.js.map +1 -1
  72. package/lib/esm/tree-widget-react/components/trees/common/components/Delayed.d.ts +1 -0
  73. package/lib/esm/tree-widget-react/components/trees/common/components/Delayed.js +1 -0
  74. package/lib/esm/tree-widget-react/components/trees/common/components/Delayed.js.map +1 -1
  75. package/lib/esm/tree-widget-react/components/trees/common/components/ProgressOverlay.d.ts +1 -0
  76. package/lib/esm/tree-widget-react/components/trees/common/components/ProgressOverlay.js +1 -0
  77. package/lib/esm/tree-widget-react/components/trees/common/components/ProgressOverlay.js.map +1 -1
  78. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.css +2 -1
  79. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.d.ts +3 -3
  80. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.js +15 -6
  81. package/lib/esm/tree-widget-react/components/trees/common/components/Tree.js.map +1 -1
  82. package/lib/esm/tree-widget-react/components/trees/common/components/TreeNodeVisibilityButton.d.ts +6 -4
  83. package/lib/esm/tree-widget-react/components/trees/common/components/TreeNodeVisibilityButton.js +20 -25
  84. package/lib/esm/tree-widget-react/components/trees/common/components/TreeNodeVisibilityButton.js.map +1 -1
  85. package/lib/esm/tree-widget-react/components/trees/common/components/TreeRenderer.d.ts +1 -1
  86. package/lib/esm/tree-widget-react/components/trees/common/components/TreeRenderer.js +10 -4
  87. package/lib/esm/tree-widget-react/components/trees/common/components/TreeRenderer.js.map +1 -1
  88. package/lib/esm/tree-widget-react/components/trees/common/components/UseVisibilityButtonHandler.d.ts +1 -0
  89. package/lib/esm/tree-widget-react/components/trees/common/components/UseVisibilityButtonHandler.js +7 -3
  90. package/lib/esm/tree-widget-react/components/trees/common/components/UseVisibilityButtonHandler.js.map +1 -1
  91. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTree.js +1 -1
  92. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTree.js.map +1 -1
  93. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTreeRenderer.d.ts +1 -1
  94. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTreeRenderer.js +12 -11
  95. package/lib/esm/tree-widget-react/components/trees/common/components/VisibilityTreeRenderer.js.map +1 -1
  96. package/lib/esm/tree-widget-react/components/trees/common/internal/AlwaysAndNeverDrawnElementInfo.d.ts +44 -0
  97. package/lib/esm/tree-widget-react/components/trees/{models-tree → common}/internal/AlwaysAndNeverDrawnElementInfo.js +81 -31
  98. package/lib/esm/tree-widget-react/components/trees/common/internal/AlwaysAndNeverDrawnElementInfo.js.map +1 -0
  99. package/lib/esm/tree-widget-react/components/trees/common/internal/ClassNameDefinitions.d.ts +33 -0
  100. package/lib/esm/tree-widget-react/components/trees/common/internal/ClassNameDefinitions.js +37 -0
  101. package/lib/esm/tree-widget-react/components/trees/common/internal/ClassNameDefinitions.js.map +1 -0
  102. package/lib/esm/tree-widget-react/components/trees/common/internal/ModelCategoryElementsCountCache.d.ts +16 -0
  103. package/lib/esm/tree-widget-react/components/trees/common/internal/ModelCategoryElementsCountCache.js +72 -0
  104. package/lib/esm/tree-widget-react/components/trees/common/internal/ModelCategoryElementsCountCache.js.map +1 -0
  105. package/lib/esm/tree-widget-react/components/trees/common/{Rxjs.d.ts → internal/Rxjs.d.ts} +9 -2
  106. package/lib/esm/tree-widget-react/components/trees/common/{Rxjs.js → internal/Rxjs.js} +9 -2
  107. package/lib/esm/tree-widget-react/components/trees/common/internal/Rxjs.js.map +1 -0
  108. package/lib/esm/tree-widget-react/components/trees/common/{Tooltip.d.ts → internal/Tooltip.d.ts} +4 -8
  109. package/lib/esm/tree-widget-react/components/trees/common/{Tooltip.js → internal/Tooltip.js} +4 -10
  110. package/lib/esm/tree-widget-react/components/trees/common/internal/Tooltip.js.map +1 -0
  111. package/lib/esm/tree-widget-react/components/trees/common/internal/Types.d.ts +14 -0
  112. package/lib/esm/tree-widget-react/components/{tree-header/SearchBox.css → trees/common/internal/Types.js} +2 -4
  113. package/lib/esm/tree-widget-react/components/trees/common/internal/Types.js.map +1 -0
  114. package/lib/esm/tree-widget-react/components/trees/common/internal/UseActiveViewport.js.map +1 -0
  115. package/lib/esm/tree-widget-react/components/trees/common/{UseHierarchiesLocalization.d.ts → internal/UseHierarchiesLocalization.d.ts} +1 -0
  116. package/lib/esm/tree-widget-react/components/trees/common/{UseHierarchiesLocalization.js → internal/UseHierarchiesLocalization.js} +6 -2
  117. package/lib/esm/tree-widget-react/components/trees/common/internal/UseHierarchiesLocalization.js.map +1 -0
  118. package/lib/esm/tree-widget-react/components/trees/common/{UseHierarchyFiltering.d.ts → internal/UseHierarchyFiltering.d.ts} +1 -0
  119. package/lib/esm/tree-widget-react/components/trees/common/{UseHierarchyFiltering.js → internal/UseHierarchyFiltering.js} +4 -3
  120. package/lib/esm/tree-widget-react/components/trees/common/internal/UseHierarchyFiltering.js.map +1 -0
  121. package/lib/esm/tree-widget-react/components/trees/common/{UseIModelChangeListener.d.ts → internal/UseIModelChangeListener.d.ts} +1 -0
  122. package/lib/esm/tree-widget-react/components/trees/common/{UseIModelChangeListener.js → internal/UseIModelChangeListener.js} +1 -0
  123. package/lib/esm/tree-widget-react/components/trees/common/internal/UseIModelChangeListener.js.map +1 -0
  124. package/lib/esm/tree-widget-react/components/trees/common/internal/Utils.d.ts +47 -0
  125. package/lib/esm/tree-widget-react/components/trees/common/internal/Utils.js +101 -0
  126. package/lib/esm/tree-widget-react/components/trees/common/internal/Utils.js.map +1 -0
  127. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityChangeEventListener.d.ts +20 -0
  128. package/lib/esm/tree-widget-react/components/trees/{models-tree → common}/internal/VisibilityChangeEventListener.js +17 -9
  129. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityChangeEventListener.js.map +1 -0
  130. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityUtils.d.ts +63 -0
  131. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityUtils.js +210 -0
  132. package/lib/esm/tree-widget-react/components/trees/common/internal/VisibilityUtils.js.map +1 -0
  133. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTree.d.ts +2 -2
  134. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTree.js +8 -8
  135. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTree.js.map +1 -1
  136. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeComponent.d.ts +1 -1
  137. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeComponent.js.map +1 -1
  138. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeDefinition.d.ts +1 -0
  139. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeDefinition.js +1 -0
  140. package/lib/esm/tree-widget-react/components/trees/external-sources-tree/ExternalSourcesTreeDefinition.js.map +1 -1
  141. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTree.d.ts +4 -2
  142. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTree.js +22 -18
  143. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTree.js.map +1 -1
  144. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeComponent.d.ts +1 -1
  145. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeComponent.js +1 -1
  146. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeComponent.js.map +1 -1
  147. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeDefinition.d.ts +13 -0
  148. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeDefinition.js +39 -38
  149. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/IModelContentTreeDefinition.js.map +1 -1
  150. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/internal/IModelContentTreeIdsCache.d.ts +5 -5
  151. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/internal/IModelContentTreeIdsCache.js +25 -24
  152. package/lib/esm/tree-widget-react/components/trees/imodel-content-tree/internal/IModelContentTreeIdsCache.js.map +1 -1
  153. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTree.d.ts +2 -2
  154. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTree.js +2 -2
  155. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTree.js.map +1 -1
  156. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeButtons.js +15 -11
  157. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeButtons.js.map +1 -1
  158. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeComponent.d.ts +1 -1
  159. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeComponent.js +1 -1
  160. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeComponent.js.map +1 -1
  161. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeDefinition.d.ts +7 -2
  162. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeDefinition.js +73 -70
  163. package/lib/esm/tree-widget-react/components/trees/models-tree/ModelsTreeDefinition.js.map +1 -1
  164. package/lib/esm/tree-widget-react/components/trees/models-tree/UseModelsTree.js +36 -202
  165. package/lib/esm/tree-widget-react/components/trees/models-tree/UseModelsTree.js.map +1 -1
  166. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/FilteredTree.d.ts +7 -9
  167. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/FilteredTree.js +4 -6
  168. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/FilteredTree.js.map +1 -1
  169. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeIdsCache.d.ts +4 -5
  170. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeIdsCache.js +54 -104
  171. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeIdsCache.js.map +1 -1
  172. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeVisibilityHandler.d.ts +0 -26
  173. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeVisibilityHandler.js +80 -359
  174. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/ModelsTreeVisibilityHandler.js.map +1 -1
  175. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseFilteredPaths.d.ts +27 -0
  176. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseFilteredPaths.js +170 -0
  177. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseFilteredPaths.js.map +1 -0
  178. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseIdsCache.d.ts +8 -0
  179. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseIdsCache.js +47 -0
  180. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/UseIdsCache.js.map +1 -0
  181. package/lib/public/locales/en/TreeWidget.json +13 -59
  182. package/package.json +19 -20
  183. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesVisibilityHandler.d.ts +0 -37
  184. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesVisibilityHandler.js +0 -214
  185. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/CategoriesVisibilityHandler.js.map +0 -1
  186. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/ClassNameDefinitions.d.ts +0 -7
  187. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/ClassNameDefinitions.js +0 -11
  188. package/lib/esm/tree-widget-react/components/trees/categories-tree/internal/ClassNameDefinitions.js.map +0 -1
  189. package/lib/esm/tree-widget-react/components/trees/common/Rxjs.js.map +0 -1
  190. package/lib/esm/tree-widget-react/components/trees/common/Tooltip.js.map +0 -1
  191. package/lib/esm/tree-widget-react/components/trees/common/UseActiveViewport.js.map +0 -1
  192. package/lib/esm/tree-widget-react/components/trees/common/UseFiltering.d.ts +0 -11
  193. package/lib/esm/tree-widget-react/components/trees/common/UseFiltering.js +0 -24
  194. package/lib/esm/tree-widget-react/components/trees/common/UseFiltering.js.map +0 -1
  195. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchiesLocalization.js.map +0 -1
  196. package/lib/esm/tree-widget-react/components/trees/common/UseHierarchyFiltering.js.map +0 -1
  197. package/lib/esm/tree-widget-react/components/trees/common/UseIModelChangeListener.js.map +0 -1
  198. package/lib/esm/tree-widget-react/components/trees/models-tree/Utils.d.ts +0 -7
  199. package/lib/esm/tree-widget-react/components/trees/models-tree/Utils.js +0 -21
  200. package/lib/esm/tree-widget-react/components/trees/models-tree/Utils.js.map +0 -1
  201. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/AlwaysAndNeverDrawnElementInfo.d.ts +0 -29
  202. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/AlwaysAndNeverDrawnElementInfo.js.map +0 -1
  203. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/VisibilityChangeEventListener.d.ts +0 -12
  204. package/lib/esm/tree-widget-react/components/trees/models-tree/internal/VisibilityChangeEventListener.js.map +0 -1
  205. /package/lib/esm/tree-widget-react/components/trees/common/{UseActiveViewport.d.ts → internal/UseActiveViewport.d.ts} +0 -0
  206. /package/lib/esm/tree-widget-react/components/trees/common/{UseActiveViewport.js → internal/UseActiveViewport.js} +0 -0
@@ -0,0 +1,701 @@
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, firstValueFrom, forkJoin, from, fromEventPattern, map, merge, mergeAll, mergeMap, of, reduce, shareReplay, startWith, Subject, take, takeUntil, tap, toArray, } from "rxjs";
6
+ import { assert, Id64 } from "@itwin/core-bentley";
7
+ import { PerModelCategoryVisibility } from "@itwin/core-frontend";
8
+ import { HierarchyNode, HierarchyNodeKey } from "@itwin/presentation-hierarchies";
9
+ import { AlwaysAndNeverDrawnElementInfo } from "../../common/internal/AlwaysAndNeverDrawnElementInfo.js";
10
+ import { toVoidPromise } from "../../common/internal/Rxjs.js";
11
+ import { createVisibilityStatus } from "../../common/internal/Tooltip.js";
12
+ import { getClassesByView, releaseMainThreadOnItemsCount, setDifference, setIntersection } from "../../common/internal/Utils.js";
13
+ import { createVisibilityChangeEventListener } from "../../common/internal/VisibilityChangeEventListener.js";
14
+ import { changeElementStateNoChildrenOperator, enableCategoryDisplay, enableSubCategoryDisplay, filterSubModeledElementIds, getElementOverriddenVisibility, getElementVisibility, getSubModeledElementsVisibilityStatus, getVisibilityFromAlwaysAndNeverDrawnElementsImpl, mergeVisibilityStatuses, } from "../../common/internal/VisibilityUtils.js";
15
+ import { createVisibilityHandlerResult } from "../../common/UseHierarchyVisibility.js";
16
+ import { CategoriesTreeNode } from "./CategoriesTreeNode.js";
17
+ import { createFilteredTree, parseCategoryKey, parseSubCategoryKey } from "./FilteredTree.js";
18
+ /**
19
+ * Creates an instance if `CategoriesTreeVisibilityHandler`.
20
+ * @internal
21
+ */
22
+ export function createCategoriesTreeVisibilityHandler(props) {
23
+ return new CategoriesTreeVisibilityHandlerImpl(props);
24
+ }
25
+ class CategoriesTreeVisibilityHandlerImpl {
26
+ constructor(_props) {
27
+ this._props = _props;
28
+ this._elementChangeQueue = new Subject();
29
+ this._subscriptions = [];
30
+ this._changeRequest = new Subject();
31
+ this._eventListener = createVisibilityChangeEventListener({
32
+ viewport: _props.viewport,
33
+ listeners: {
34
+ models: true,
35
+ categories: true,
36
+ elements: _props.hierarchyConfig.showElements,
37
+ displayStyle: true,
38
+ },
39
+ });
40
+ this._alwaysAndNeverDrawnElements = new AlwaysAndNeverDrawnElementInfo(_props.viewport);
41
+ this._idsCache = this._props.idsCache;
42
+ const { categoryClass, elementClass, modelClass } = getClassesByView(_props.viewport.view.is2d() ? "2d" : "3d");
43
+ if (_props.filteredPaths) {
44
+ this._filteredTree = createFilteredTree({
45
+ idsCache: this._idsCache,
46
+ filteringPaths: _props.filteredPaths,
47
+ categoryClassName: categoryClass,
48
+ categoryElementClassName: elementClass,
49
+ categoryModelClassName: modelClass,
50
+ imodelAccess: this._props.imodelAccess,
51
+ });
52
+ }
53
+ this._subscriptions.push(this._elementChangeQueue.pipe(concatAll()).subscribe());
54
+ }
55
+ get onVisibilityChange() {
56
+ return this._eventListener.onVisibilityChange;
57
+ }
58
+ async getVisibilityStatus(node) {
59
+ return firstValueFrom(this.getVisibilityStatusObs(node).pipe(
60
+ // unsubscribe from the observable if the change request for this node is received
61
+ takeUntil(this._changeRequest.pipe(filter(({ key, depth }) => depth === node.parentKeys.length && HierarchyNodeKey.equals(node.key, key)))),
62
+ // unsubscribe if visibility changes
63
+ takeUntil(fromEventPattern((handler) => {
64
+ this._eventListener.onVisibilityChange.addListener(handler);
65
+ }, (handler) => {
66
+ this._eventListener.onVisibilityChange.removeListener(handler);
67
+ })), defaultIfEmpty(createVisibilityStatus("hidden"))));
68
+ }
69
+ async changeVisibility(node, shouldDisplay) {
70
+ // notify about new change request
71
+ this._changeRequest.next({ key: node.key, depth: node.parentKeys.length });
72
+ const changeObservable = this.changeVisibilityObs(node, shouldDisplay).pipe(
73
+ // unsubscribe from the observable if the change request for this node is received
74
+ takeUntil(this._changeRequest.pipe(filter(({ key, depth }) => depth === node.parentKeys.length && HierarchyNodeKey.equals(node.key, key)))), tap({
75
+ subscribe: () => {
76
+ this._eventListener.suppressChangeEvents();
77
+ this._alwaysAndNeverDrawnElements.suppressChangeEvents();
78
+ },
79
+ finalize: () => {
80
+ this._eventListener.resumeChangeEvents();
81
+ this._alwaysAndNeverDrawnElements.resumeChangeEvents();
82
+ },
83
+ }));
84
+ return toVoidPromise(changeObservable);
85
+ }
86
+ dispose() {
87
+ this[Symbol.dispose]();
88
+ }
89
+ [Symbol.dispose]() {
90
+ this._eventListener[Symbol.dispose]();
91
+ this._alwaysAndNeverDrawnElements[Symbol.dispose]();
92
+ this._subscriptions.forEach((x) => x.unsubscribe());
93
+ }
94
+ getVisibilityStatusObs(node) {
95
+ if (node.filtering?.filteredChildrenIdentifierPaths?.length && !node.filtering.isFilterTarget) {
96
+ return this.getFilteredNodeVisibility({ node });
97
+ }
98
+ if (HierarchyNode.isClassGroupingNode(node)) {
99
+ return this.getClassGroupingNodeDisplayStatus(node);
100
+ }
101
+ if (!HierarchyNode.isInstancesNode(node)) {
102
+ return of(createVisibilityStatus("disabled"));
103
+ }
104
+ if (CategoriesTreeNode.isDefinitionContainerNode(node)) {
105
+ return this.getDefinitionContainerDisplayStatus({
106
+ definitionContainerIds: node.key.instanceKeys.map((instanceKey) => instanceKey.id),
107
+ });
108
+ }
109
+ if (CategoriesTreeNode.isModelNode(node)) {
110
+ return this.getModelVisibilityStatus({
111
+ modelId: node.key.instanceKeys[0].id,
112
+ });
113
+ }
114
+ if (CategoriesTreeNode.isCategoryNode(node)) {
115
+ return this.getCategoryDisplayStatus({
116
+ categoryIds: node.key.instanceKeys.map((instanceKey) => instanceKey.id),
117
+ modelId: CategoriesTreeNode.getModelId(node),
118
+ ignoreSubCategories: node.extendedData?.isCategoryOfSubModel,
119
+ });
120
+ }
121
+ const categoryId = CategoriesTreeNode.getCategoryId(node);
122
+ if (!categoryId) {
123
+ return of(createVisibilityStatus("disabled"));
124
+ }
125
+ if (CategoriesTreeNode.isSubCategoryNode(node)) {
126
+ return this.getSubCategoryDisplayStatus({
127
+ categoryId,
128
+ subCategoryIds: node.key.instanceKeys.map((instanceKey) => instanceKey.id),
129
+ });
130
+ }
131
+ const modelId = CategoriesTreeNode.getModelId(node);
132
+ if (!modelId) {
133
+ return of(createVisibilityStatus("disabled"));
134
+ }
135
+ return this.getElementDisplayStatus({
136
+ elementId: node.key.instanceKeys[0].id,
137
+ modelId,
138
+ categoryId,
139
+ });
140
+ }
141
+ getFilteredNodeVisibility(props) {
142
+ return from(this.getVisibilityChangeTargets(props)).pipe(mergeMap(({ definitionContainerIds: definitionContainers, subCategories, modelIds: models, categories, elements }) => {
143
+ const observables = new Array();
144
+ if (definitionContainers?.size) {
145
+ observables.push(from(definitionContainers).pipe(mergeMap((definitionContainerId) => this.getDefinitionContainerDisplayStatus({ definitionContainerIds: [definitionContainerId] }))));
146
+ }
147
+ if (models?.size) {
148
+ observables.push(from(models).pipe(mergeMap((modelId) => this.getModelVisibilityStatus({ modelId }))));
149
+ }
150
+ if (categories?.size) {
151
+ observables.push(from(categories).pipe(mergeMap((key) => {
152
+ const { modelId, categoryId } = parseCategoryKey(key);
153
+ return this.getCategoryDisplayStatus({ ignoreSubCategories: !!modelId, modelId, categoryIds: [categoryId] });
154
+ })));
155
+ }
156
+ if (subCategories?.size) {
157
+ observables.push(from(subCategories).pipe(mergeMap((key) => {
158
+ const { subCategoryId, categoryId } = parseSubCategoryKey(key);
159
+ return this.getSubCategoryDisplayStatus({ subCategoryIds: [subCategoryId], categoryId });
160
+ })));
161
+ }
162
+ if (elements?.size) {
163
+ observables.push(from(elements).pipe(releaseMainThreadOnItemsCount(50), mergeMap(([categoryKey, elementIds]) => {
164
+ const { modelId, categoryId } = parseCategoryKey(categoryKey);
165
+ assert(modelId !== undefined);
166
+ return from(elementIds).pipe(releaseMainThreadOnItemsCount(1000), mergeMap((elementId) => this.getElementDisplayStatus({ modelId, categoryId, elementId })));
167
+ })));
168
+ }
169
+ return merge(...observables);
170
+ }), mergeVisibilityStatuses);
171
+ }
172
+ getModelVisibilityStatus({ modelId }) {
173
+ const result = defer(() => {
174
+ const viewport = this._props.viewport;
175
+ if (!viewport.view.viewsModel(modelId)) {
176
+ return from(this._idsCache.getModelCategoryIds(modelId)).pipe(mergeMap((categoryIds) => from(this._idsCache.getCategoriesModeledElements(modelId, categoryIds))), getSubModeledElementsVisibilityStatus({
177
+ parentNodeVisibilityStatus: createVisibilityStatus("hidden"),
178
+ getModelVisibilityStatus: (modelProps) => this.getModelVisibilityStatus(modelProps),
179
+ }));
180
+ }
181
+ return from(this._idsCache.getModelCategoryIds(modelId)).pipe(mergeAll(), mergeMap((categoryId) => this.getCategoryDisplayStatus({ modelId, categoryIds: [categoryId], ignoreSubCategories: true })), mergeVisibilityStatuses);
182
+ });
183
+ return createVisibilityHandlerResult(this, { id: modelId }, result, undefined);
184
+ }
185
+ getDefinitionContainerDisplayStatus(props) {
186
+ const result = defer(() => {
187
+ return from(this._idsCache.getAllContainedCategories(props.definitionContainerIds)).pipe(mergeAll(), mergeMap((categoryId) => this.getCategoryDisplayStatus({ categoryIds: [categoryId] })), mergeVisibilityStatuses);
188
+ });
189
+ return createVisibilityHandlerResult(this, props, result, undefined);
190
+ }
191
+ getSubCategoryDisplayStatus(props) {
192
+ const result = defer(async () => {
193
+ const { categoryId, subCategoryIds } = props;
194
+ let visibility = "unknown";
195
+ const categoryModels = [...(await this._idsCache.getCategoriesElementModels([categoryId], true)).values()].flat();
196
+ let nonDefaultModelDisplayStatesCount = 0;
197
+ for (const modelId of categoryModels) {
198
+ if (!this._props.viewport.view.viewsModel(modelId)) {
199
+ if (visibility === "visible") {
200
+ return createVisibilityStatus("partial");
201
+ }
202
+ visibility = "hidden";
203
+ ++nonDefaultModelDisplayStatesCount;
204
+ continue;
205
+ }
206
+ const override = this._props.viewport.perModelCategoryVisibility.getOverride(modelId, categoryId);
207
+ if (override === PerModelCategoryVisibility.Override.Show) {
208
+ if (visibility === "hidden") {
209
+ return createVisibilityStatus("partial");
210
+ }
211
+ visibility = "visible";
212
+ ++nonDefaultModelDisplayStatesCount;
213
+ continue;
214
+ }
215
+ if (override === PerModelCategoryVisibility.Override.Hide) {
216
+ if (visibility === "visible") {
217
+ return createVisibilityStatus("partial");
218
+ }
219
+ visibility = "hidden";
220
+ ++nonDefaultModelDisplayStatesCount;
221
+ continue;
222
+ }
223
+ }
224
+ if (categoryModels.length > 0 && nonDefaultModelDisplayStatesCount === categoryModels.length) {
225
+ assert(visibility === "visible" || visibility === "hidden");
226
+ return createVisibilityStatus(visibility);
227
+ }
228
+ if (!this._props.viewport.view.viewsCategory(categoryId)) {
229
+ return createVisibilityStatus(visibility === "visible" ? "partial" : "hidden");
230
+ }
231
+ if (subCategoryIds.length === 0) {
232
+ if (visibility === "hidden") {
233
+ return createVisibilityStatus("partial");
234
+ }
235
+ return createVisibilityStatus("visible");
236
+ }
237
+ for (const subCategoryId of subCategoryIds) {
238
+ const isSubCategoryVisible = this._props.viewport.isSubCategoryVisible(subCategoryId);
239
+ if (isSubCategoryVisible && visibility === "hidden") {
240
+ return createVisibilityStatus("partial");
241
+ }
242
+ if (!isSubCategoryVisible && visibility === "visible") {
243
+ return createVisibilityStatus("partial");
244
+ }
245
+ visibility = isSubCategoryVisible ? "visible" : "hidden";
246
+ }
247
+ assert(visibility === "visible" || visibility === "hidden");
248
+ return createVisibilityStatus(visibility);
249
+ });
250
+ return createVisibilityHandlerResult(this, props, result, undefined);
251
+ }
252
+ getDefaultModelsCategoryVisibilityStatus({ modelId, categoryIds }) {
253
+ const viewport = this._props.viewport;
254
+ if (!viewport.view.viewsModel(modelId)) {
255
+ return createVisibilityStatus("hidden");
256
+ }
257
+ let visibleCount = 0;
258
+ let hiddenCount = 0;
259
+ let visibleThroughCategorySelectorCount = 0;
260
+ for (const categoryId of categoryIds) {
261
+ if (viewport.view.viewsCategory(categoryId)) {
262
+ ++visibleThroughCategorySelectorCount;
263
+ }
264
+ const override = this._props.viewport.perModelCategoryVisibility.getOverride(modelId, categoryId);
265
+ if (override === PerModelCategoryVisibility.Override.Show) {
266
+ ++visibleCount;
267
+ continue;
268
+ }
269
+ if (override === PerModelCategoryVisibility.Override.Hide) {
270
+ ++hiddenCount;
271
+ continue;
272
+ }
273
+ if (visibleCount > 0 && hiddenCount > 0) {
274
+ return createVisibilityStatus("partial");
275
+ }
276
+ }
277
+ if (hiddenCount + visibleCount > 0) {
278
+ return createVisibilityStatus(hiddenCount > 0 ? "hidden" : "visible");
279
+ }
280
+ return createVisibilityStatus(visibleThroughCategorySelectorCount > 0 ? "visible" : "hidden");
281
+ }
282
+ getDefaultCategoryVisibilityStatus({ categoryIds }) {
283
+ return from(this._idsCache.getSubCategories(categoryIds)).pipe(mergeMap((categoriesSubCategoriesMap) => {
284
+ return from(categoryIds).pipe(mergeMap((categoryId) => {
285
+ const subCategoryIds = categoriesSubCategoriesMap.get(categoryId);
286
+ return this.getSubCategoryDisplayStatus({ subCategoryIds: subCategoryIds ?? [], categoryId });
287
+ }), mergeVisibilityStatuses);
288
+ }));
289
+ }
290
+ getCategoryDisplayStatus(props) {
291
+ const result = defer(() => {
292
+ if (!this._props.hierarchyConfig.showElements) {
293
+ return from(this.getDefaultCategoryVisibilityStatus({ categoryIds: props.categoryIds }));
294
+ }
295
+ const modelsObservable = props.modelId
296
+ ? of(new Map(props.categoryIds.map((id) => [id, [props.modelId]])))
297
+ : from(this._idsCache.getCategoriesElementModels(props.categoryIds));
298
+ return merge(
299
+ // get visibility status from always and never drawn elements
300
+ this._props.hierarchyConfig.showElements
301
+ ? modelsObservable.pipe(mergeMap((categoryModelsMap) => {
302
+ if (categoryModelsMap.size === 0) {
303
+ return props.modelId
304
+ ? of(this.getDefaultModelsCategoryVisibilityStatus({ modelId: props.modelId, categoryIds: props.categoryIds }))
305
+ : from(this.getDefaultCategoryVisibilityStatus({ categoryIds: props.categoryIds }));
306
+ }
307
+ return from(categoryModelsMap).pipe(mergeMap(([category, models]) => from(models).pipe(mergeMap((model) => {
308
+ if (this._props.viewport.view.viewsModel(model)) {
309
+ return this.getVisibilityFromAlwaysAndNeverDrawnElements({
310
+ queryProps: props,
311
+ defaultStatus: () => this.getDefaultModelsCategoryVisibilityStatus({ modelId: model, categoryIds: [category] }),
312
+ }).pipe(mergeMap((visibilityStatusAlwaysAndNeverDraw) => {
313
+ return from(this._idsCache.getCategoriesModeledElements(model, [category])).pipe(getSubModeledElementsVisibilityStatus({
314
+ parentNodeVisibilityStatus: visibilityStatusAlwaysAndNeverDraw,
315
+ getModelVisibilityStatus: (modelProps) => this.getModelVisibilityStatus(modelProps),
316
+ }));
317
+ }));
318
+ }
319
+ return from(this._idsCache.getCategoriesModeledElements(model, [category])).pipe(getSubModeledElementsVisibilityStatus({
320
+ parentNodeVisibilityStatus: createVisibilityStatus("hidden"),
321
+ getModelVisibilityStatus: (modelProps) => this.getModelVisibilityStatus(modelProps),
322
+ }));
323
+ }), mergeVisibilityStatuses)), mergeVisibilityStatuses);
324
+ }), map((visibilityStatus) => {
325
+ return { visibilityStatus, type: 0 };
326
+ }))
327
+ : EMPTY,
328
+ // get category status
329
+ (props.modelId
330
+ ? of(this.getDefaultModelsCategoryVisibilityStatus({ modelId: props.modelId, categoryIds: props.categoryIds }))
331
+ : from(this.getDefaultCategoryVisibilityStatus({ categoryIds: props.categoryIds }))).pipe(map((visibilityStatus) => {
332
+ return { visibilityStatus, type: 1 };
333
+ }))).pipe(toArray(), mergeMap(async (visibilityStatusesInfo) => {
334
+ let defaultStatus;
335
+ let alwaysNeverDrawStatus;
336
+ visibilityStatusesInfo.forEach((visibilityStatusInfo) => {
337
+ switch (visibilityStatusInfo.type) {
338
+ case 0:
339
+ alwaysNeverDrawStatus = visibilityStatusInfo.visibilityStatus;
340
+ break;
341
+ case 1:
342
+ defaultStatus = visibilityStatusInfo.visibilityStatus;
343
+ break;
344
+ }
345
+ });
346
+ assert(defaultStatus !== undefined);
347
+ if (defaultStatus.state === "partial") {
348
+ return defaultStatus;
349
+ }
350
+ // This can happen if:
351
+ // a) showElements is set to false
352
+ // b) root category does not have any elements (that dont have Parent)
353
+ // In both cases we don't need to look at modeled elements visibility
354
+ if (alwaysNeverDrawStatus === undefined) {
355
+ return defaultStatus;
356
+ }
357
+ // In cases where
358
+ // a) SubCategories are hidden
359
+ // b) Category has model (it means that category is under hidden subModel)
360
+ // We dont need to look at default category status, it is already accounted for in always/never drawn visibility
361
+ if (this._props.hierarchyConfig.hideSubCategories || props.modelId) {
362
+ return alwaysNeverDrawStatus;
363
+ }
364
+ if ((await this._idsCache.getSubCategories(props.categoryIds)).size === 0) {
365
+ return alwaysNeverDrawStatus;
366
+ }
367
+ if (alwaysNeverDrawStatus.state === "partial" || alwaysNeverDrawStatus.state !== defaultStatus.state) {
368
+ return createVisibilityStatus("partial");
369
+ }
370
+ return alwaysNeverDrawStatus;
371
+ }));
372
+ });
373
+ return createVisibilityHandlerResult(this, props, result, undefined);
374
+ }
375
+ getClassGroupingNodeDisplayStatus(node) {
376
+ const result = defer(() => {
377
+ const info = this.getGroupingNodeInfo(node);
378
+ const { modelElementsMap, categoryId } = info;
379
+ return from(modelElementsMap).pipe(mergeMap(([modelId, elementIds]) => {
380
+ if (!this._props.viewport.view.viewsModel(modelId)) {
381
+ return of([...elementIds]).pipe(filterSubModeledElementIds({ doesSubModelExist: async (id) => this._idsCache.hasSubModel(id) }), getSubModeledElementsVisibilityStatus({
382
+ parentNodeVisibilityStatus: createVisibilityStatus("hidden"),
383
+ getModelVisibilityStatus: (modelProps) => this.getModelVisibilityStatus(modelProps),
384
+ }));
385
+ }
386
+ return this.getVisibilityFromAlwaysAndNeverDrawnElements({
387
+ elements: elementIds,
388
+ defaultStatus: () => this.getDefaultModelsCategoryVisibilityStatus({ categoryIds: [categoryId], modelId }),
389
+ }).pipe(mergeMap((visibilityStatusAlwaysAndNeverDraw) => {
390
+ return of([...elementIds]).pipe(filterSubModeledElementIds({ doesSubModelExist: async (id) => this._idsCache.hasSubModel(id) }), getSubModeledElementsVisibilityStatus({
391
+ parentNodeVisibilityStatus: visibilityStatusAlwaysAndNeverDraw,
392
+ getModelVisibilityStatus: (modelProps) => this.getModelVisibilityStatus(modelProps),
393
+ }));
394
+ }));
395
+ }), mergeVisibilityStatuses);
396
+ });
397
+ return createVisibilityHandlerResult(this, { node }, result, undefined);
398
+ }
399
+ getElementDisplayStatus(props) {
400
+ const result = defer(() => {
401
+ const viewport = this._props.viewport;
402
+ const { elementId, modelId, categoryId } = props;
403
+ const viewsModel = viewport.view.viewsModel(modelId);
404
+ const elementStatus = getElementOverriddenVisibility({
405
+ elementId,
406
+ viewport,
407
+ });
408
+ return from(this._idsCache.hasSubModel(elementId)).pipe(mergeMap((hasSubModel) => (hasSubModel ? this.getModelVisibilityStatus({ modelId: elementId }) : of(undefined))), map((subModelVisibilityStatus) => getElementVisibility(viewsModel, elementStatus, this.getDefaultModelsCategoryVisibilityStatus({ categoryIds: [categoryId], modelId }), subModelVisibilityStatus)));
409
+ });
410
+ return createVisibilityHandlerResult(this, props, result, undefined);
411
+ }
412
+ /** Changes visibility of the items represented by the tree node. */
413
+ changeVisibilityObs(node, on) {
414
+ if (node.filtering?.filteredChildrenIdentifierPaths?.length && !node.filtering.isFilterTarget) {
415
+ return this.changeFilteredNodeVisibility({ node, on });
416
+ }
417
+ if (HierarchyNode.isClassGroupingNode(node)) {
418
+ return this.changeElementGroupingNodeState(node, on);
419
+ }
420
+ if (!HierarchyNode.isInstancesNode(node)) {
421
+ return EMPTY;
422
+ }
423
+ if (CategoriesTreeNode.isDefinitionContainerNode(node)) {
424
+ return this.changeDefinitionContainerState({
425
+ definitionContainerIds: node.key.instanceKeys.map((instanceKey) => instanceKey.id),
426
+ on,
427
+ });
428
+ }
429
+ if (CategoriesTreeNode.isModelNode(node)) {
430
+ return this.changeModelState({
431
+ ids: node.key.instanceKeys.map(({ id }) => id),
432
+ on,
433
+ });
434
+ }
435
+ if (CategoriesTreeNode.isCategoryNode(node)) {
436
+ return this.changeCategoryState({
437
+ categoryIds: node.key.instanceKeys.map((instanceKey) => instanceKey.id),
438
+ modelId: CategoriesTreeNode.getModelId(node),
439
+ on,
440
+ ignoreSubCategories: node.extendedData?.isCategoryOfSubModel,
441
+ });
442
+ }
443
+ const categoryId = CategoriesTreeNode.getCategoryId(node);
444
+ if (!categoryId) {
445
+ return EMPTY;
446
+ }
447
+ if (CategoriesTreeNode.isSubCategoryNode(node)) {
448
+ if (this._props.hierarchyConfig.hideSubCategories) {
449
+ return EMPTY;
450
+ }
451
+ return this.changeSubCategoryState({
452
+ categoryId,
453
+ subCategoryIds: node.key.instanceKeys.map((instanceKey) => instanceKey.id),
454
+ on,
455
+ });
456
+ }
457
+ if (!this._props.hierarchyConfig.showElements) {
458
+ return EMPTY;
459
+ }
460
+ const modelId = CategoriesTreeNode.getModelId(node);
461
+ if (!modelId) {
462
+ return EMPTY;
463
+ }
464
+ return this.changeElementsState({
465
+ elementIds: new Set([...node.key.instanceKeys.map(({ id }) => id)]),
466
+ modelId,
467
+ categoryId,
468
+ on,
469
+ });
470
+ }
471
+ async getVisibilityChangeTargets({ node }) {
472
+ const filteredTree = await this._filteredTree;
473
+ return filteredTree ? filteredTree.getVisibilityChangeTargets(node) : {};
474
+ }
475
+ changeFilteredNodeVisibility({ on, ...props }) {
476
+ return from(this.getVisibilityChangeTargets(props)).pipe(mergeMap(({ definitionContainerIds: definitionContainers, subCategories, modelIds: models, categories, elements }) => {
477
+ const observables = new Array();
478
+ if (definitionContainers?.size) {
479
+ observables.push(this.changeDefinitionContainerState({ definitionContainerIds: [...definitionContainers], on }));
480
+ }
481
+ if (models?.size) {
482
+ observables.push(this.changeModelState({ ids: models, on }));
483
+ }
484
+ if (categories?.size) {
485
+ observables.push(from(categories).pipe(mergeMap((key) => {
486
+ const { modelId, categoryId } = parseCategoryKey(key);
487
+ return this.changeCategoryState({ modelId, categoryIds: [categoryId], ignoreSubCategories: false, on });
488
+ })));
489
+ }
490
+ if (subCategories?.size) {
491
+ observables.push(from(subCategories).pipe(mergeMap((key) => {
492
+ const { subCategoryId, categoryId } = parseSubCategoryKey(key);
493
+ return this.changeSubCategoryState({ categoryId, subCategoryIds: [subCategoryId], on });
494
+ })));
495
+ }
496
+ if (elements?.size) {
497
+ observables.push(from(elements).pipe(mergeMap(([categoryKey, elementIds]) => {
498
+ const { modelId, categoryId } = parseCategoryKey(categoryKey);
499
+ assert(modelId !== undefined);
500
+ return this.changeElementsState({ modelId, categoryId, elementIds, on });
501
+ })));
502
+ }
503
+ return merge(...observables);
504
+ }));
505
+ }
506
+ changeModelState(props) {
507
+ const { ids, on } = props;
508
+ if (Id64.sizeOf(ids) === 0) {
509
+ return EMPTY;
510
+ }
511
+ const result = defer(() => {
512
+ const viewport = this._props.viewport;
513
+ viewport.perModelCategoryVisibility.clearOverrides(ids);
514
+ const idsObs = from(Id64.iterable(ids));
515
+ if (!on) {
516
+ viewport.changeModelDisplay(ids, false);
517
+ return idsObs.pipe(mergeMap(async (modelId) => ({ modelId, categoryIds: await this._idsCache.getModelCategoryIds(modelId) })), mergeMap(({ modelId, categoryIds }) => from(this._idsCache.getCategoriesModeledElements(modelId, categoryIds))), mergeMap((modeledElementIds) => this.changeModelState({ ids: modeledElementIds, on })));
518
+ }
519
+ return concat(from(viewport.addViewedModels(ids)), idsObs.pipe(mergeMap((modelId) => {
520
+ return from(this._idsCache.getModelCategoryIds(modelId)).pipe(mergeAll(), mergeMap((categoryId) => this.changeCategoryState({ categoryIds: [categoryId], modelId, on, ignoreSubCategories: true })));
521
+ })));
522
+ });
523
+ return createVisibilityHandlerResult(this, props, result, undefined);
524
+ }
525
+ showModelWithoutAnyCategoriesOrElements(modelId) {
526
+ const viewport = this._props.viewport;
527
+ return forkJoin({
528
+ categories: this._idsCache.getModelCategoryIds(modelId),
529
+ alwaysDrawnElements: this._alwaysAndNeverDrawnElements.getAlwaysDrawnElements({ modelId }),
530
+ }).pipe(mergeMap(async ({ categories, alwaysDrawnElements }) => {
531
+ const alwaysDrawn = this._props.viewport.alwaysDrawn;
532
+ if (alwaysDrawn && alwaysDrawnElements) {
533
+ viewport.setAlwaysDrawn(setDifference(alwaysDrawn, alwaysDrawnElements));
534
+ }
535
+ categories.forEach((categoryId) => {
536
+ this.changeCategoryStateInViewportAccordingToModelVisibility(modelId, categoryId, false, false);
537
+ });
538
+ await viewport.addViewedModels(modelId);
539
+ }));
540
+ }
541
+ changeCategoryStateInViewportAccordingToModelVisibility(modelId, categoryId, on, changeSubCategories) {
542
+ const viewport = this._props.viewport;
543
+ const isDisplayedInSelector = viewport.view.viewsCategory(categoryId);
544
+ const override = on === isDisplayedInSelector
545
+ ? PerModelCategoryVisibility.Override.None
546
+ : on
547
+ ? PerModelCategoryVisibility.Override.Show
548
+ : PerModelCategoryVisibility.Override.Hide;
549
+ viewport.perModelCategoryVisibility.setOverride(modelId, categoryId, override);
550
+ if (override === PerModelCategoryVisibility.Override.None && on) {
551
+ // we took off the override which means the category is displayed in selector, but
552
+ // doesn't mean all its subcategories are displayed - this call ensures that
553
+ viewport.changeCategoryDisplay(categoryId, true, changeSubCategories);
554
+ }
555
+ }
556
+ changeSubCategoryState(props) {
557
+ const result = defer(() => {
558
+ return concat(
559
+ // make sure parent category and models are enabled
560
+ props.on
561
+ ? concat(from(enableCategoryDisplay(this._props.viewport, [props.categoryId], props.on, false)), from(this.enableCategoriesElementModelsVisibility([props.categoryId])))
562
+ : EMPTY, from(props.subCategoryIds).pipe(map((subCategoryId) => enableSubCategoryDisplay(this._props.viewport, subCategoryId, props.on))));
563
+ });
564
+ return createVisibilityHandlerResult(this, props, result, undefined);
565
+ }
566
+ async enableCategoriesElementModelsVisibility(categoryIds) {
567
+ const categoriesModelsMap = await this._idsCache.getCategoriesElementModels(categoryIds, true);
568
+ const modelIds = [...categoriesModelsMap.values()].flat();
569
+ const hiddenModels = modelIds.filter((modelId) => !this._props.viewport.view.viewsModel(modelId));
570
+ if (hiddenModels.length > 0) {
571
+ this._props.viewport.changeModelDisplay(hiddenModels, true);
572
+ }
573
+ }
574
+ changeDefinitionContainerState(props) {
575
+ const result = defer(() => {
576
+ return from(this._idsCache.getAllContainedCategories(props.definitionContainerIds)).pipe(mergeMap((categoryIds) => {
577
+ return this.changeCategoryState({ categoryIds, on: props.on });
578
+ }));
579
+ });
580
+ return createVisibilityHandlerResult(this, props, result, undefined);
581
+ }
582
+ changeCategoryState(props) {
583
+ const result = defer(() => {
584
+ const { modelId, categoryIds, on, ignoreSubCategories } = props;
585
+ const viewport = this._props.viewport;
586
+ if (!this._props.hierarchyConfig.showElements) {
587
+ return concat(from(enableCategoryDisplay(viewport, categoryIds, on, on)), on ? from(this.enableCategoriesElementModelsVisibility(categoryIds)) : EMPTY);
588
+ }
589
+ const modelIdsObservable = modelId
590
+ ? of(new Map(categoryIds.map((id) => [id, [modelId]])))
591
+ : from(this._idsCache.getCategoriesElementModels(categoryIds, true));
592
+ return concat(modelId ? EMPTY : from(enableCategoryDisplay(viewport, categoryIds, on, ignoreSubCategories ? false : on)), on
593
+ ? modelIdsObservable.pipe(mergeMap((categoriesMap) => from(categoriesMap.values())), mergeAll(), filter((modelIdToCheck) => !viewport.view.viewsModel(modelIdToCheck)), mergeMap((modelIdToCheck) => this.showModelWithoutAnyCategoriesOrElements(modelIdToCheck)))
594
+ : EMPTY, modelIdsObservable.pipe(mergeMap((categoriesMap) => from(categoriesMap.entries())), mergeMap(([categoryId, modelIds]) => {
595
+ return from(modelIds).pipe(mergeMap((modelOfCategory) => {
596
+ this.changeCategoryStateInViewportAccordingToModelVisibility(modelOfCategory, categoryId, on, !!ignoreSubCategories);
597
+ return this._alwaysAndNeverDrawnElements.clearAlwaysAndNeverDrawnElements({ modelId: modelOfCategory, categoryIds: [categoryId] });
598
+ }));
599
+ })), modelIdsObservable.pipe(mergeMap((categoriesMap) => from(categoriesMap.entries())), mergeMap(([categoryId, modelIds]) => {
600
+ return from(modelIds).pipe(mergeMap((modelOfCategory) => from(this._idsCache.getCategoriesModeledElements(modelOfCategory, [categoryId])).pipe(mergeMap((modeledElementIds) => this.changeModelState({ ids: modeledElementIds, on })))));
601
+ })));
602
+ });
603
+ return createVisibilityHandlerResult(this, props, result, undefined);
604
+ }
605
+ doChangeElementsState(props) {
606
+ return defer(() => {
607
+ const { modelId, categoryId, elementIds, on } = props;
608
+ const viewport = this._props.viewport;
609
+ return concat(on && !viewport.view.viewsModel(modelId) ? this.showModelWithoutAnyCategoriesOrElements(modelId) : EMPTY, defer(() => {
610
+ const categoryVisibility = this.getDefaultModelsCategoryVisibilityStatus({ categoryIds: [categoryId], modelId });
611
+ const isDisplayedByDefault = categoryVisibility.state === "visible";
612
+ return this.queueElementsVisibilityChange(elementIds, on, isDisplayedByDefault);
613
+ }), from(elementIds).pipe(mergeMap(async (elementId) => ({ elementId, isSubModel: await this._idsCache.hasSubModel(elementId) })), filter(({ isSubModel }) => isSubModel), map(({ elementId }) => elementId), toArray(), mergeMap((subModelIds) => this.changeModelState({ ids: subModelIds, on }))));
614
+ });
615
+ }
616
+ /**
617
+ * Updates visibility of all grouping node's elements.
618
+ * @see `changeElementState`
619
+ */
620
+ changeElementGroupingNodeState(node, on) {
621
+ const info = this.getGroupingNodeInfo(node);
622
+ const result = from(info.modelElementsMap).pipe(mergeMap(([modelId, elementIds]) => {
623
+ return this.doChangeElementsState({ modelId, elementIds, categoryId: info.categoryId, on });
624
+ }));
625
+ return createVisibilityHandlerResult(this, { node, on }, result, undefined);
626
+ }
627
+ /**
628
+ * Updates visibility of an element and all its child elements by adding them to the always/never drawn list.
629
+ * @note If element is to be enabled and model is hidden, it will be enabled.
630
+ */
631
+ changeElementsState(props) {
632
+ const result = this.doChangeElementsState(props);
633
+ return createVisibilityHandlerResult(this, props, result, undefined);
634
+ }
635
+ queueElementsVisibilityChange(elementIds, on, visibleByDefault) {
636
+ const finishedSubject = new Subject();
637
+ // observable to track if visibility change is finished/cancelled
638
+ const changeFinished = finishedSubject.pipe(startWith(false), shareReplay(1), filter((finished) => finished));
639
+ const changeObservable = from(elementIds).pipe(
640
+ // check if visibility change is not finished (cancelled) due to change overall change request being cancelled
641
+ takeUntil(changeFinished), changeElementStateNoChildrenOperator({ on, isDisplayedByDefault: visibleByDefault, viewport: this._props.viewport }), tap({
642
+ next: () => {
643
+ // notify that visibility change is finished
644
+ finishedSubject.next(true);
645
+ },
646
+ }));
647
+ // queue visibility change. `changeObservable` will be subscribed to when other queue changes are finished
648
+ this._elementChangeQueue.next(changeObservable);
649
+ // return observable that will emit when visibility change is finished
650
+ return changeFinished.pipe(take(1), tap({
651
+ unsubscribe: () => {
652
+ // if this observable is unsubscribed before visibility change is finished, we have to notify that it queued change request is cancelled
653
+ finishedSubject.next(true);
654
+ },
655
+ }), map(() => undefined));
656
+ }
657
+ getVisibilityFromAlwaysAndNeverDrawnElements(props) {
658
+ const viewport = this._props.viewport;
659
+ if (viewport.isAlwaysDrawnExclusive) {
660
+ if (!viewport?.alwaysDrawn?.size) {
661
+ return of(createVisibilityStatus("hidden"));
662
+ }
663
+ }
664
+ else if (!viewport?.neverDrawn?.size && !viewport?.alwaysDrawn?.size) {
665
+ return of(props.defaultStatus());
666
+ }
667
+ if ("elements" in props) {
668
+ return of(getVisibilityFromAlwaysAndNeverDrawnElementsImpl({
669
+ ...props,
670
+ alwaysDrawn: viewport.alwaysDrawn?.size ? setIntersection(props.elements, viewport.alwaysDrawn) : undefined,
671
+ neverDrawn: viewport.neverDrawn?.size ? setIntersection(props.elements, viewport.neverDrawn) : undefined,
672
+ totalCount: props.elements.size,
673
+ viewport,
674
+ }));
675
+ }
676
+ const { modelId, categoryIds } = props.queryProps;
677
+ const totalCount = (modelId ? of(new Map(categoryIds.map((categoryId) => [categoryId, [modelId]]))) : from(this._idsCache.getCategoriesElementModels(categoryIds))).pipe(mergeMap((categoriesMap) => from(categoriesMap)), mergeMap(([categoryId, modelIds]) => {
678
+ return from(modelIds).pipe(mergeMap((modelOfCategory) => from(this._idsCache.getCategoryElementsCount(modelOfCategory, categoryId))), reduce((acc, specificModelCategoryCount) => {
679
+ return acc + specificModelCategoryCount;
680
+ }, 0));
681
+ }));
682
+ return forkJoin({
683
+ totalCount,
684
+ alwaysDrawn: this._alwaysAndNeverDrawnElements.getAlwaysDrawnElements(props.queryProps),
685
+ neverDrawn: this._alwaysAndNeverDrawnElements.getNeverDrawnElements(props.queryProps),
686
+ }).pipe(map((state) => {
687
+ return getVisibilityFromAlwaysAndNeverDrawnElementsImpl({
688
+ ...props,
689
+ ...state,
690
+ viewport,
691
+ });
692
+ }));
693
+ }
694
+ getGroupingNodeInfo(node) {
695
+ const modelElementsMap = node.extendedData?.modelElementsMap;
696
+ const categoryId = node.extendedData?.categoryId;
697
+ assert(!!modelElementsMap && !!categoryId);
698
+ return { modelElementsMap, categoryId };
699
+ }
700
+ }
701
+ //# sourceMappingURL=CategoriesTreeVisibilityHandler.js.map