@fluidframework/tree 2.41.0 → 2.42.0

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 (416) hide show
  1. package/.vscode/settings.json +1 -0
  2. package/CHANGELOG.md +33 -0
  3. package/api-report/tree.alpha.api.md +11 -7
  4. package/dist/alpha.d.ts +1 -0
  5. package/dist/codec/codec.d.ts +23 -3
  6. package/dist/codec/codec.d.ts.map +1 -1
  7. package/dist/codec/codec.js.map +1 -1
  8. package/dist/codec/index.d.ts +1 -1
  9. package/dist/codec/index.d.ts.map +1 -1
  10. package/dist/codec/index.js.map +1 -1
  11. package/dist/codec/versioned/codec.d.ts +35 -2
  12. package/dist/codec/versioned/codec.d.ts.map +1 -1
  13. package/dist/codec/versioned/codec.js +38 -3
  14. package/dist/codec/versioned/codec.js.map +1 -1
  15. package/dist/core/tree/detachedFieldIndex.d.ts +2 -2
  16. package/dist/core/tree/detachedFieldIndex.d.ts.map +1 -1
  17. package/dist/core/tree/detachedFieldIndex.js +5 -1
  18. package/dist/core/tree/detachedFieldIndex.js.map +1 -1
  19. package/dist/core/tree/mapTree.d.ts +2 -1
  20. package/dist/core/tree/mapTree.d.ts.map +1 -1
  21. package/dist/core/tree/mapTree.js +11 -5
  22. package/dist/core/tree/mapTree.js.map +1 -1
  23. package/dist/core/tree/visitorUtils.d.ts +2 -2
  24. package/dist/core/tree/visitorUtils.d.ts.map +1 -1
  25. package/dist/core/tree/visitorUtils.js.map +1 -1
  26. package/dist/feature-libraries/default-schema/defaultEditBuilder.d.ts +3 -2
  27. package/dist/feature-libraries/default-schema/defaultEditBuilder.d.ts.map +1 -1
  28. package/dist/feature-libraries/default-schema/defaultEditBuilder.js.map +1 -1
  29. package/dist/feature-libraries/default-schema/schemaChecker.d.ts +4 -3
  30. package/dist/feature-libraries/default-schema/schemaChecker.d.ts.map +1 -1
  31. package/dist/feature-libraries/default-schema/schemaChecker.js +4 -3
  32. package/dist/feature-libraries/default-schema/schemaChecker.js.map +1 -1
  33. package/dist/feature-libraries/flex-tree/context.d.ts +14 -6
  34. package/dist/feature-libraries/flex-tree/context.d.ts.map +1 -1
  35. package/dist/feature-libraries/flex-tree/context.js.map +1 -1
  36. package/dist/feature-libraries/flex-tree/flexTreeTypes.d.ts +34 -14
  37. package/dist/feature-libraries/flex-tree/flexTreeTypes.d.ts.map +1 -1
  38. package/dist/feature-libraries/flex-tree/flexTreeTypes.js +4 -0
  39. package/dist/feature-libraries/flex-tree/flexTreeTypes.js.map +1 -1
  40. package/dist/feature-libraries/flex-tree/index.d.ts +2 -2
  41. package/dist/feature-libraries/flex-tree/index.d.ts.map +1 -1
  42. package/dist/feature-libraries/flex-tree/index.js.map +1 -1
  43. package/dist/feature-libraries/flex-tree/lazyField.d.ts +6 -6
  44. package/dist/feature-libraries/flex-tree/lazyField.d.ts.map +1 -1
  45. package/dist/feature-libraries/flex-tree/lazyField.js.map +1 -1
  46. package/dist/feature-libraries/flex-tree/lazyNode.d.ts +3 -2
  47. package/dist/feature-libraries/flex-tree/lazyNode.d.ts.map +1 -1
  48. package/dist/feature-libraries/flex-tree/lazyNode.js +3 -0
  49. package/dist/feature-libraries/flex-tree/lazyNode.js.map +1 -1
  50. package/dist/feature-libraries/forest-summary/forestSummarizer.d.ts +2 -2
  51. package/dist/feature-libraries/forest-summary/forestSummarizer.d.ts.map +1 -1
  52. package/dist/feature-libraries/forest-summary/forestSummarizer.js +7 -7
  53. package/dist/feature-libraries/forest-summary/forestSummarizer.js.map +1 -1
  54. package/dist/feature-libraries/index.d.ts +2 -2
  55. package/dist/feature-libraries/index.d.ts.map +1 -1
  56. package/dist/feature-libraries/index.js +4 -2
  57. package/dist/feature-libraries/index.js.map +1 -1
  58. package/dist/feature-libraries/mapTreeCursor.d.ts +39 -3
  59. package/dist/feature-libraries/mapTreeCursor.d.ts.map +1 -1
  60. package/dist/feature-libraries/mapTreeCursor.js +45 -7
  61. package/dist/feature-libraries/mapTreeCursor.js.map +1 -1
  62. package/dist/feature-libraries/modular-schema/modularChangeCodecs.d.ts +1 -1
  63. package/dist/feature-libraries/modular-schema/modularChangeCodecs.d.ts.map +1 -1
  64. package/dist/feature-libraries/modular-schema/modularChangeCodecs.js +5 -5
  65. package/dist/feature-libraries/modular-schema/modularChangeCodecs.js.map +1 -1
  66. package/dist/feature-libraries/treeCursorUtils.d.ts +5 -1
  67. package/dist/feature-libraries/treeCursorUtils.d.ts.map +1 -1
  68. package/dist/feature-libraries/treeCursorUtils.js +8 -2
  69. package/dist/feature-libraries/treeCursorUtils.js.map +1 -1
  70. package/dist/index.d.ts +1 -1
  71. package/dist/index.d.ts.map +1 -1
  72. package/dist/index.js.map +1 -1
  73. package/dist/packageVersion.d.ts +1 -1
  74. package/dist/packageVersion.js +1 -1
  75. package/dist/packageVersion.js.map +1 -1
  76. package/dist/shared-tree/schematizingTreeView.d.ts +11 -1
  77. package/dist/shared-tree/schematizingTreeView.d.ts.map +1 -1
  78. package/dist/shared-tree/schematizingTreeView.js +36 -22
  79. package/dist/shared-tree/schematizingTreeView.js.map +1 -1
  80. package/dist/shared-tree/sharedTree.d.ts +3 -3
  81. package/dist/shared-tree/sharedTree.d.ts.map +1 -1
  82. package/dist/shared-tree/sharedTree.js +1 -0
  83. package/dist/shared-tree/sharedTree.js.map +1 -1
  84. package/dist/shared-tree/tree.d.ts.map +1 -1
  85. package/dist/shared-tree/tree.js +8 -24
  86. package/dist/shared-tree/tree.js.map +1 -1
  87. package/dist/shared-tree/treeAlpha.d.ts +2 -3
  88. package/dist/shared-tree/treeAlpha.d.ts.map +1 -1
  89. package/dist/shared-tree/treeAlpha.js +13 -15
  90. package/dist/shared-tree/treeAlpha.js.map +1 -1
  91. package/dist/shared-tree/treeCheckout.d.ts.map +1 -1
  92. package/dist/shared-tree/treeCheckout.js +5 -2
  93. package/dist/shared-tree/treeCheckout.js.map +1 -1
  94. package/dist/simple-tree/api/configuration.d.ts +2 -2
  95. package/dist/simple-tree/api/configuration.js +1 -1
  96. package/dist/simple-tree/api/configuration.js.map +1 -1
  97. package/dist/simple-tree/api/create.d.ts +9 -4
  98. package/dist/simple-tree/api/create.d.ts.map +1 -1
  99. package/dist/simple-tree/api/create.js +29 -16
  100. package/dist/simple-tree/api/create.js.map +1 -1
  101. package/dist/simple-tree/api/customTree.d.ts +4 -0
  102. package/dist/simple-tree/api/customTree.d.ts.map +1 -1
  103. package/dist/simple-tree/api/customTree.js +9 -1
  104. package/dist/simple-tree/api/customTree.js.map +1 -1
  105. package/dist/simple-tree/api/index.d.ts +1 -1
  106. package/dist/simple-tree/api/index.d.ts.map +1 -1
  107. package/dist/simple-tree/api/index.js +1 -2
  108. package/dist/simple-tree/api/index.js.map +1 -1
  109. package/dist/simple-tree/api/schemaFactory.d.ts +10 -2
  110. package/dist/simple-tree/api/schemaFactory.d.ts.map +1 -1
  111. package/dist/simple-tree/api/schemaFactory.js +38 -9
  112. package/dist/simple-tree/api/schemaFactory.js.map +1 -1
  113. package/dist/simple-tree/api/treeNodeApi.d.ts +14 -3
  114. package/dist/simple-tree/api/treeNodeApi.d.ts.map +1 -1
  115. package/dist/simple-tree/api/treeNodeApi.js +32 -17
  116. package/dist/simple-tree/api/treeNodeApi.js.map +1 -1
  117. package/dist/simple-tree/api/verboseTree.d.ts.map +1 -1
  118. package/dist/simple-tree/api/verboseTree.js +12 -9
  119. package/dist/simple-tree/api/verboseTree.js.map +1 -1
  120. package/dist/simple-tree/core/getOrCreateNode.d.ts.map +1 -1
  121. package/dist/simple-tree/core/getOrCreateNode.js +2 -1
  122. package/dist/simple-tree/core/getOrCreateNode.js.map +1 -1
  123. package/dist/simple-tree/core/index.d.ts +2 -2
  124. package/dist/simple-tree/core/index.d.ts.map +1 -1
  125. package/dist/simple-tree/core/index.js +3 -4
  126. package/dist/simple-tree/core/index.js.map +1 -1
  127. package/dist/simple-tree/core/treeNodeKernel.d.ts +15 -25
  128. package/dist/simple-tree/core/treeNodeKernel.d.ts.map +1 -1
  129. package/dist/simple-tree/core/treeNodeKernel.js +26 -33
  130. package/dist/simple-tree/core/treeNodeKernel.js.map +1 -1
  131. package/dist/simple-tree/core/unhydratedFlexTree.d.ts +128 -59
  132. package/dist/simple-tree/core/unhydratedFlexTree.d.ts.map +1 -1
  133. package/dist/simple-tree/core/unhydratedFlexTree.js +169 -182
  134. package/dist/simple-tree/core/unhydratedFlexTree.js.map +1 -1
  135. package/dist/simple-tree/index.d.ts +3 -3
  136. package/dist/simple-tree/index.d.ts.map +1 -1
  137. package/dist/simple-tree/index.js +5 -5
  138. package/dist/simple-tree/index.js.map +1 -1
  139. package/dist/simple-tree/node-kinds/array/arrayNode.d.ts.map +1 -1
  140. package/dist/simple-tree/node-kinds/array/arrayNode.js +5 -6
  141. package/dist/simple-tree/node-kinds/array/arrayNode.js.map +1 -1
  142. package/dist/simple-tree/node-kinds/index.d.ts +1 -1
  143. package/dist/simple-tree/node-kinds/index.d.ts.map +1 -1
  144. package/dist/simple-tree/node-kinds/index.js +1 -2
  145. package/dist/simple-tree/node-kinds/index.js.map +1 -1
  146. package/dist/simple-tree/node-kinds/map/mapNode.d.ts.map +1 -1
  147. package/dist/simple-tree/node-kinds/map/mapNode.js +2 -2
  148. package/dist/simple-tree/node-kinds/map/mapNode.js.map +1 -1
  149. package/dist/simple-tree/node-kinds/object/index.d.ts +1 -1
  150. package/dist/simple-tree/node-kinds/object/index.d.ts.map +1 -1
  151. package/dist/simple-tree/node-kinds/object/index.js +1 -2
  152. package/dist/simple-tree/node-kinds/object/index.js.map +1 -1
  153. package/dist/simple-tree/node-kinds/object/objectNode.d.ts +3 -14
  154. package/dist/simple-tree/node-kinds/object/objectNode.d.ts.map +1 -1
  155. package/dist/simple-tree/node-kinds/object/objectNode.js +12 -43
  156. package/dist/simple-tree/node-kinds/object/objectNode.js.map +1 -1
  157. package/dist/simple-tree/prepareForInsertion.d.ts +20 -6
  158. package/dist/simple-tree/prepareForInsertion.d.ts.map +1 -1
  159. package/dist/simple-tree/prepareForInsertion.js +26 -19
  160. package/dist/simple-tree/prepareForInsertion.js.map +1 -1
  161. package/dist/simple-tree/schemaTypes.d.ts +8 -8
  162. package/dist/simple-tree/schemaTypes.d.ts.map +1 -1
  163. package/dist/simple-tree/schemaTypes.js.map +1 -1
  164. package/dist/simple-tree/toStoredSchema.d.ts +6 -1
  165. package/dist/simple-tree/toStoredSchema.d.ts.map +1 -1
  166. package/dist/simple-tree/toStoredSchema.js +6 -3
  167. package/dist/simple-tree/toStoredSchema.js.map +1 -1
  168. package/dist/simple-tree/{toMapTree.d.ts → unhydratedFlexTreeFromInsertable.d.ts} +12 -23
  169. package/dist/simple-tree/unhydratedFlexTreeFromInsertable.d.ts.map +1 -0
  170. package/dist/simple-tree/{toMapTree.js → unhydratedFlexTreeFromInsertable.js} +103 -185
  171. package/dist/simple-tree/unhydratedFlexTreeFromInsertable.js.map +1 -0
  172. package/dist/treeFactory.d.ts.map +1 -1
  173. package/dist/treeFactory.js +6 -1
  174. package/dist/treeFactory.js.map +1 -1
  175. package/dist/util/index.d.ts +1 -1
  176. package/dist/util/index.d.ts.map +1 -1
  177. package/dist/util/index.js +2 -1
  178. package/dist/util/index.js.map +1 -1
  179. package/dist/util/utils.d.ts +4 -0
  180. package/dist/util/utils.d.ts.map +1 -1
  181. package/dist/util/utils.js +8 -1
  182. package/dist/util/utils.js.map +1 -1
  183. package/docs/user-facing/schema-evolution.md +1 -1
  184. package/lib/alpha.d.ts +1 -0
  185. package/lib/codec/codec.d.ts +23 -3
  186. package/lib/codec/codec.d.ts.map +1 -1
  187. package/lib/codec/codec.js.map +1 -1
  188. package/lib/codec/index.d.ts +1 -1
  189. package/lib/codec/index.d.ts.map +1 -1
  190. package/lib/codec/index.js.map +1 -1
  191. package/lib/codec/versioned/codec.d.ts +35 -2
  192. package/lib/codec/versioned/codec.d.ts.map +1 -1
  193. package/lib/codec/versioned/codec.js +36 -2
  194. package/lib/codec/versioned/codec.js.map +1 -1
  195. package/lib/core/tree/detachedFieldIndex.d.ts +2 -2
  196. package/lib/core/tree/detachedFieldIndex.d.ts.map +1 -1
  197. package/lib/core/tree/detachedFieldIndex.js +6 -2
  198. package/lib/core/tree/detachedFieldIndex.js.map +1 -1
  199. package/lib/core/tree/mapTree.d.ts +2 -1
  200. package/lib/core/tree/mapTree.d.ts.map +1 -1
  201. package/lib/core/tree/mapTree.js +11 -5
  202. package/lib/core/tree/mapTree.js.map +1 -1
  203. package/lib/core/tree/visitorUtils.d.ts +2 -2
  204. package/lib/core/tree/visitorUtils.d.ts.map +1 -1
  205. package/lib/core/tree/visitorUtils.js.map +1 -1
  206. package/lib/feature-libraries/default-schema/defaultEditBuilder.d.ts +3 -2
  207. package/lib/feature-libraries/default-schema/defaultEditBuilder.d.ts.map +1 -1
  208. package/lib/feature-libraries/default-schema/defaultEditBuilder.js.map +1 -1
  209. package/lib/feature-libraries/default-schema/schemaChecker.d.ts +4 -3
  210. package/lib/feature-libraries/default-schema/schemaChecker.d.ts.map +1 -1
  211. package/lib/feature-libraries/default-schema/schemaChecker.js +4 -3
  212. package/lib/feature-libraries/default-schema/schemaChecker.js.map +1 -1
  213. package/lib/feature-libraries/flex-tree/context.d.ts +14 -6
  214. package/lib/feature-libraries/flex-tree/context.d.ts.map +1 -1
  215. package/lib/feature-libraries/flex-tree/context.js.map +1 -1
  216. package/lib/feature-libraries/flex-tree/flexTreeTypes.d.ts +34 -14
  217. package/lib/feature-libraries/flex-tree/flexTreeTypes.d.ts.map +1 -1
  218. package/lib/feature-libraries/flex-tree/flexTreeTypes.js +4 -0
  219. package/lib/feature-libraries/flex-tree/flexTreeTypes.js.map +1 -1
  220. package/lib/feature-libraries/flex-tree/index.d.ts +2 -2
  221. package/lib/feature-libraries/flex-tree/index.d.ts.map +1 -1
  222. package/lib/feature-libraries/flex-tree/index.js.map +1 -1
  223. package/lib/feature-libraries/flex-tree/lazyField.d.ts +6 -6
  224. package/lib/feature-libraries/flex-tree/lazyField.d.ts.map +1 -1
  225. package/lib/feature-libraries/flex-tree/lazyField.js.map +1 -1
  226. package/lib/feature-libraries/flex-tree/lazyNode.d.ts +3 -2
  227. package/lib/feature-libraries/flex-tree/lazyNode.d.ts.map +1 -1
  228. package/lib/feature-libraries/flex-tree/lazyNode.js +3 -0
  229. package/lib/feature-libraries/flex-tree/lazyNode.js.map +1 -1
  230. package/lib/feature-libraries/forest-summary/forestSummarizer.d.ts +2 -2
  231. package/lib/feature-libraries/forest-summary/forestSummarizer.d.ts.map +1 -1
  232. package/lib/feature-libraries/forest-summary/forestSummarizer.js +2 -2
  233. package/lib/feature-libraries/forest-summary/forestSummarizer.js.map +1 -1
  234. package/lib/feature-libraries/index.d.ts +2 -2
  235. package/lib/feature-libraries/index.d.ts.map +1 -1
  236. package/lib/feature-libraries/index.js +1 -1
  237. package/lib/feature-libraries/index.js.map +1 -1
  238. package/lib/feature-libraries/mapTreeCursor.d.ts +39 -3
  239. package/lib/feature-libraries/mapTreeCursor.d.ts.map +1 -1
  240. package/lib/feature-libraries/mapTreeCursor.js +43 -7
  241. package/lib/feature-libraries/mapTreeCursor.js.map +1 -1
  242. package/lib/feature-libraries/modular-schema/modularChangeCodecs.d.ts +1 -1
  243. package/lib/feature-libraries/modular-schema/modularChangeCodecs.d.ts.map +1 -1
  244. package/lib/feature-libraries/modular-schema/modularChangeCodecs.js +5 -5
  245. package/lib/feature-libraries/modular-schema/modularChangeCodecs.js.map +1 -1
  246. package/lib/feature-libraries/treeCursorUtils.d.ts +5 -1
  247. package/lib/feature-libraries/treeCursorUtils.d.ts.map +1 -1
  248. package/lib/feature-libraries/treeCursorUtils.js +8 -2
  249. package/lib/feature-libraries/treeCursorUtils.js.map +1 -1
  250. package/lib/index.d.ts +1 -1
  251. package/lib/index.d.ts.map +1 -1
  252. package/lib/index.js.map +1 -1
  253. package/lib/packageVersion.d.ts +1 -1
  254. package/lib/packageVersion.js +1 -1
  255. package/lib/packageVersion.js.map +1 -1
  256. package/lib/shared-tree/schematizingTreeView.d.ts +11 -1
  257. package/lib/shared-tree/schematizingTreeView.d.ts.map +1 -1
  258. package/lib/shared-tree/schematizingTreeView.js +34 -21
  259. package/lib/shared-tree/schematizingTreeView.js.map +1 -1
  260. package/lib/shared-tree/sharedTree.d.ts +3 -3
  261. package/lib/shared-tree/sharedTree.d.ts.map +1 -1
  262. package/lib/shared-tree/sharedTree.js +2 -1
  263. package/lib/shared-tree/sharedTree.js.map +1 -1
  264. package/lib/shared-tree/tree.d.ts.map +1 -1
  265. package/lib/shared-tree/tree.js +2 -18
  266. package/lib/shared-tree/tree.js.map +1 -1
  267. package/lib/shared-tree/treeAlpha.d.ts +2 -3
  268. package/lib/shared-tree/treeAlpha.d.ts.map +1 -1
  269. package/lib/shared-tree/treeAlpha.js +4 -6
  270. package/lib/shared-tree/treeAlpha.js.map +1 -1
  271. package/lib/shared-tree/treeCheckout.d.ts.map +1 -1
  272. package/lib/shared-tree/treeCheckout.js +6 -3
  273. package/lib/shared-tree/treeCheckout.js.map +1 -1
  274. package/lib/simple-tree/api/configuration.d.ts +2 -2
  275. package/lib/simple-tree/api/configuration.js +1 -1
  276. package/lib/simple-tree/api/configuration.js.map +1 -1
  277. package/lib/simple-tree/api/create.d.ts +9 -4
  278. package/lib/simple-tree/api/create.d.ts.map +1 -1
  279. package/lib/simple-tree/api/create.js +22 -9
  280. package/lib/simple-tree/api/create.js.map +1 -1
  281. package/lib/simple-tree/api/customTree.d.ts +4 -0
  282. package/lib/simple-tree/api/customTree.d.ts.map +1 -1
  283. package/lib/simple-tree/api/customTree.js +7 -0
  284. package/lib/simple-tree/api/customTree.js.map +1 -1
  285. package/lib/simple-tree/api/index.d.ts +1 -1
  286. package/lib/simple-tree/api/index.d.ts.map +1 -1
  287. package/lib/simple-tree/api/index.js +1 -1
  288. package/lib/simple-tree/api/index.js.map +1 -1
  289. package/lib/simple-tree/api/schemaFactory.d.ts +10 -2
  290. package/lib/simple-tree/api/schemaFactory.d.ts.map +1 -1
  291. package/lib/simple-tree/api/schemaFactory.js +40 -11
  292. package/lib/simple-tree/api/schemaFactory.js.map +1 -1
  293. package/lib/simple-tree/api/treeNodeApi.d.ts +14 -3
  294. package/lib/simple-tree/api/treeNodeApi.d.ts.map +1 -1
  295. package/lib/simple-tree/api/treeNodeApi.js +35 -20
  296. package/lib/simple-tree/api/treeNodeApi.js.map +1 -1
  297. package/lib/simple-tree/api/verboseTree.d.ts.map +1 -1
  298. package/lib/simple-tree/api/verboseTree.js +13 -10
  299. package/lib/simple-tree/api/verboseTree.js.map +1 -1
  300. package/lib/simple-tree/core/getOrCreateNode.d.ts.map +1 -1
  301. package/lib/simple-tree/core/getOrCreateNode.js +3 -2
  302. package/lib/simple-tree/core/getOrCreateNode.js.map +1 -1
  303. package/lib/simple-tree/core/index.d.ts +2 -2
  304. package/lib/simple-tree/core/index.d.ts.map +1 -1
  305. package/lib/simple-tree/core/index.js +2 -2
  306. package/lib/simple-tree/core/index.js.map +1 -1
  307. package/lib/simple-tree/core/treeNodeKernel.d.ts +15 -25
  308. package/lib/simple-tree/core/treeNodeKernel.d.ts.map +1 -1
  309. package/lib/simple-tree/core/treeNodeKernel.js +24 -32
  310. package/lib/simple-tree/core/treeNodeKernel.js.map +1 -1
  311. package/lib/simple-tree/core/unhydratedFlexTree.d.ts +128 -59
  312. package/lib/simple-tree/core/unhydratedFlexTree.d.ts.map +1 -1
  313. package/lib/simple-tree/core/unhydratedFlexTree.js +166 -181
  314. package/lib/simple-tree/core/unhydratedFlexTree.js.map +1 -1
  315. package/lib/simple-tree/index.d.ts +3 -3
  316. package/lib/simple-tree/index.d.ts.map +1 -1
  317. package/lib/simple-tree/index.js +3 -3
  318. package/lib/simple-tree/index.js.map +1 -1
  319. package/lib/simple-tree/node-kinds/array/arrayNode.d.ts.map +1 -1
  320. package/lib/simple-tree/node-kinds/array/arrayNode.js +4 -5
  321. package/lib/simple-tree/node-kinds/array/arrayNode.js.map +1 -1
  322. package/lib/simple-tree/node-kinds/index.d.ts +1 -1
  323. package/lib/simple-tree/node-kinds/index.d.ts.map +1 -1
  324. package/lib/simple-tree/node-kinds/index.js +1 -1
  325. package/lib/simple-tree/node-kinds/index.js.map +1 -1
  326. package/lib/simple-tree/node-kinds/map/mapNode.d.ts.map +1 -1
  327. package/lib/simple-tree/node-kinds/map/mapNode.js +3 -3
  328. package/lib/simple-tree/node-kinds/map/mapNode.js.map +1 -1
  329. package/lib/simple-tree/node-kinds/object/index.d.ts +1 -1
  330. package/lib/simple-tree/node-kinds/object/index.d.ts.map +1 -1
  331. package/lib/simple-tree/node-kinds/object/index.js +1 -1
  332. package/lib/simple-tree/node-kinds/object/index.js.map +1 -1
  333. package/lib/simple-tree/node-kinds/object/objectNode.d.ts +3 -14
  334. package/lib/simple-tree/node-kinds/object/objectNode.d.ts.map +1 -1
  335. package/lib/simple-tree/node-kinds/object/objectNode.js +3 -33
  336. package/lib/simple-tree/node-kinds/object/objectNode.js.map +1 -1
  337. package/lib/simple-tree/prepareForInsertion.d.ts +20 -6
  338. package/lib/simple-tree/prepareForInsertion.d.ts.map +1 -1
  339. package/lib/simple-tree/prepareForInsertion.js +25 -19
  340. package/lib/simple-tree/prepareForInsertion.js.map +1 -1
  341. package/lib/simple-tree/schemaTypes.d.ts +8 -8
  342. package/lib/simple-tree/schemaTypes.d.ts.map +1 -1
  343. package/lib/simple-tree/schemaTypes.js.map +1 -1
  344. package/lib/simple-tree/toStoredSchema.d.ts +6 -1
  345. package/lib/simple-tree/toStoredSchema.d.ts.map +1 -1
  346. package/lib/simple-tree/toStoredSchema.js +4 -1
  347. package/lib/simple-tree/toStoredSchema.js.map +1 -1
  348. package/lib/simple-tree/{toMapTree.d.ts → unhydratedFlexTreeFromInsertable.d.ts} +12 -23
  349. package/lib/simple-tree/unhydratedFlexTreeFromInsertable.d.ts.map +1 -0
  350. package/lib/simple-tree/{toMapTree.js → unhydratedFlexTreeFromInsertable.js} +105 -186
  351. package/lib/simple-tree/unhydratedFlexTreeFromInsertable.js.map +1 -0
  352. package/lib/treeFactory.d.ts.map +1 -1
  353. package/lib/treeFactory.js +6 -1
  354. package/lib/treeFactory.js.map +1 -1
  355. package/lib/util/index.d.ts +1 -1
  356. package/lib/util/index.d.ts.map +1 -1
  357. package/lib/util/index.js +1 -1
  358. package/lib/util/index.js.map +1 -1
  359. package/lib/util/utils.d.ts +4 -0
  360. package/lib/util/utils.d.ts.map +1 -1
  361. package/lib/util/utils.js +6 -0
  362. package/lib/util/utils.js.map +1 -1
  363. package/package.json +21 -21
  364. package/src/codec/codec.ts +24 -3
  365. package/src/codec/index.ts +1 -0
  366. package/src/codec/versioned/codec.ts +42 -5
  367. package/src/core/tree/detachedFieldIndex.ts +13 -4
  368. package/src/core/tree/mapTree.ts +22 -7
  369. package/src/core/tree/visitorUtils.ts +2 -2
  370. package/src/feature-libraries/default-schema/defaultEditBuilder.ts +3 -2
  371. package/src/feature-libraries/default-schema/schemaChecker.ts +7 -6
  372. package/src/feature-libraries/flex-tree/context.ts +17 -7
  373. package/src/feature-libraries/flex-tree/flexTreeTypes.ts +36 -15
  374. package/src/feature-libraries/flex-tree/index.ts +4 -0
  375. package/src/feature-libraries/flex-tree/lazyField.ts +8 -6
  376. package/src/feature-libraries/flex-tree/lazyNode.ts +6 -2
  377. package/src/feature-libraries/forest-summary/forestSummarizer.ts +3 -2
  378. package/src/feature-libraries/index.ts +9 -0
  379. package/src/feature-libraries/mapTreeCursor.ts +103 -16
  380. package/src/feature-libraries/modular-schema/modularChangeCodecs.ts +9 -5
  381. package/src/feature-libraries/treeCursorUtils.ts +21 -10
  382. package/src/index.ts +1 -0
  383. package/src/packageVersion.ts +1 -1
  384. package/src/shared-tree/schematizingTreeView.ts +40 -22
  385. package/src/shared-tree/sharedTree.ts +9 -3
  386. package/src/shared-tree/tree.ts +5 -20
  387. package/src/shared-tree/treeAlpha.ts +17 -11
  388. package/src/shared-tree/treeCheckout.ts +6 -3
  389. package/src/simple-tree/api/configuration.ts +3 -3
  390. package/src/simple-tree/api/create.ts +49 -19
  391. package/src/simple-tree/api/customTree.ts +10 -0
  392. package/src/simple-tree/api/index.ts +1 -4
  393. package/src/simple-tree/api/schemaFactory.ts +62 -13
  394. package/src/simple-tree/api/treeNodeApi.ts +48 -27
  395. package/src/simple-tree/api/verboseTree.ts +15 -12
  396. package/src/simple-tree/core/getOrCreateNode.ts +4 -2
  397. package/src/simple-tree/core/index.ts +2 -3
  398. package/src/simple-tree/core/treeNodeKernel.ts +37 -54
  399. package/src/simple-tree/core/unhydratedFlexTree.ts +222 -261
  400. package/src/simple-tree/index.ts +3 -3
  401. package/src/simple-tree/node-kinds/array/arrayNode.ts +12 -13
  402. package/src/simple-tree/node-kinds/index.ts +0 -1
  403. package/src/simple-tree/node-kinds/map/mapNode.ts +6 -9
  404. package/src/simple-tree/node-kinds/object/index.ts +0 -1
  405. package/src/simple-tree/node-kinds/object/objectNode.ts +7 -49
  406. package/src/simple-tree/prepareForInsertion.ts +49 -42
  407. package/src/simple-tree/schemaTypes.ts +9 -8
  408. package/src/simple-tree/toStoredSchema.ts +7 -1
  409. package/src/simple-tree/{toMapTree.ts → unhydratedFlexTreeFromInsertable.ts} +134 -226
  410. package/src/treeFactory.ts +6 -1
  411. package/src/util/index.ts +1 -0
  412. package/src/util/utils.ts +7 -0
  413. package/dist/simple-tree/toMapTree.d.ts.map +0 -1
  414. package/dist/simple-tree/toMapTree.js.map +0 -1
  415. package/lib/simple-tree/toMapTree.d.ts.map +0 -1
  416. package/lib/simple-tree/toMapTree.js.map +0 -1
@@ -8,72 +8,92 @@ import { assert, oob, fail } from "@fluidframework/core-utils/internal";
8
8
  import { UsageError } from "@fluidframework/telemetry-utils/internal";
9
9
  import { EmptyKey, forbiddenFieldKindIdentifier, } from "../../core/index.js";
10
10
  import { FlexTreeEntityKind, flexTreeMarker, indexForAt, FieldKinds, cursorForMapTreeNode, } from "../../feature-libraries/index.js";
11
- import { brand, getOrCreate, mapIterable } from "../../util/index.js";
11
+ import { brand, filterIterable, getOrCreate } from "../../util/index.js";
12
12
  /**
13
- * An unhydrated implementation of {@link FlexTreeNode} which wraps a {@link MapTree}.
14
- * @remarks
15
- * MapTreeNodes are unconditionally cached -
16
- * when retrieved via {@link getOrCreateNodeFromInnerNode}, the same {@link MapTree} object will always produce the same `UnhydratedFlexTreeNode` object.
17
- *
18
- * Create a `UnhydratedFlexTreeNode` by calling {@link getOrCreate}.
13
+ * The {@link Unhydrated} implementation of {@link FlexTreeNode}.
19
14
  */
20
15
  export class UnhydratedFlexTreeNode {
21
- get schema() {
22
- return this.mapTree.type;
16
+ isHydrated() {
17
+ return false;
23
18
  }
24
19
  get storedSchema() {
25
- return (this.context.schema.nodeSchema.get(this.mapTree.type) ?? fail(0xb46 /* missing schema */));
20
+ return (this.context.schema.nodeSchema.get(this.data.type) ?? fail(0xb46 /* missing schema */));
26
21
  }
27
22
  get events() {
28
23
  return this._events;
29
24
  }
30
- /**
31
- * Create a {@link UnhydratedFlexTreeNode} that wraps the given {@link MapTree}, or get the node that already exists for that {@link MapTree} if there is one.
32
- * @param nodeSchema - the {@link FlexTreeNodeSchema | schema} that the node conforms to
33
- * @param mapTree - the {@link MapTree} containing the data for this node.
34
- * @remarks It must conform to the `nodeSchema`.
35
- */
36
- static getOrCreate(context, mapTree) {
37
- return nodeCache.get(mapTree) ?? new UnhydratedFlexTreeNode(context, mapTree, undefined);
38
- }
39
25
  get context() {
40
26
  return this.simpleContext.flexContext;
41
27
  }
42
28
  /**
43
29
  * Create a new UnhydratedFlexTreeNode.
44
- * @param location - the parentage of this node, if it is being created underneath an existing node and field, or undefined if not
45
- * @remarks This class (and its subclasses) should not be directly constructed outside of this module.
46
- * Instead, use {@link getOrCreateNodeFromInnerNode} to create a UnhydratedFlexTreeNode from a {@link MapTree}.
47
- * A `UnhydratedFlexTreeNode` may never be constructed more than once for the same {@link MapTree} object.
48
- * Instead, it should always be acquired via {@link getOrCreateNodeFromInnerNode}.
49
30
  */
50
- constructor(simpleContext,
51
- /** The underlying {@link MapTree} that this `UnhydratedFlexTreeNode` reads its data from */
52
- mapTree, location = unparentedLocation) {
31
+ constructor(
32
+ /**
33
+ * The {@link NodeData} for this node.
34
+ */
35
+ data,
36
+ /**
37
+ * All {@link UnhydratedFlexTreeField} for this node that have been created so far.
38
+ * @remarks
39
+ * This includes all non-empty fields, but also any empty fields which have been previously requested.
40
+ */
41
+ fieldsAll,
42
+ /**
43
+ * The {@link Context} for this node.
44
+ * @remarks
45
+ * Provides access to all schema reachable from this node.
46
+ * See {@link getUnhydratedContext}.
47
+ */
48
+ simpleContext) {
49
+ this.data = data;
50
+ this.fieldsAll = fieldsAll;
53
51
  this.simpleContext = simpleContext;
54
- this.mapTree = mapTree;
55
- this.location = location;
52
+ this.location = unparentedLocation;
56
53
  this[_a] = FlexTreeEntityKind.Node;
57
54
  this._events = createEmitter();
58
- assert(!nodeCache.has(mapTree), 0x98b /* A node already exists for the given MapTree */);
59
- nodeCache.set(mapTree, this);
60
- // Fully demand the tree to ensure that parent pointers are present and accurate on all nodes.
61
- // When a UnhydratedFlexTreeNode is constructed, its MapTree may contain nodes (anywhere below) that map (via the `nodeCache`) to pre-existing UnhydratedFlexTreeNodes.
62
- // Put another way, for a given MapTree, some ancestor UnhydratedFlexTreeNode can be created after any number of its descendant UnhydratedFlexTreeNodes already exist.
63
- // In such a case, the spine of nodes between the descendant and ancestor need to exist in order for the ancestor to be able to walk upwards via the `parentField` property.
64
- // This needs to happen for all UnhydratedFlexTreeNodes that are descendants of the ancestor UnhydratedFlexTreeNode.
65
- // Demanding the entire tree is overkill to solve this problem since not all descendant MapTree nodes will have corresponding UnhydratedFlexTreeNodes.
66
- // However, demanding the full tree also lets us eagerly validate that there are no duplicate MapTrees (i.e. same MapTree object) anywhere in the tree.
67
- this.walkTree();
55
+ /**
56
+ * The non-empty fields on this node.
57
+ * @remarks
58
+ * This is needed to implement {@link MapTreeNodeViewGeneric.fields}, which must omit empty fields.
59
+ * Due to having to detect if a field is empty, this forces the evaluation of any pending defaults in the fields.
60
+ * Use {@link allFieldsLazy} to avoid evaluating pending defaults.
61
+ */
62
+ this.fields = {
63
+ get: (key) => this.tryGetField(key),
64
+ [Symbol.iterator]: () => filterIterable(this.fieldsAll, ([, field]) => field.length > 0),
65
+ };
66
+ for (const [_key, field] of this.fieldsAll) {
67
+ field.parent = this;
68
+ }
69
+ }
70
+ /**
71
+ * Gets all fields, without filtering out empty ones.
72
+ * @remarks
73
+ * This avoids forcing the evaluating of pending defaults in the fields, and also saves a copy on access.
74
+ */
75
+ get allFieldsLazy() {
76
+ return this.fieldsAll;
68
77
  }
69
78
  get type() {
70
- return this.mapTree.type;
79
+ return this.data.type;
80
+ }
81
+ get schema() {
82
+ return this.data.type;
83
+ }
84
+ getOrCreateField(key) {
85
+ return getOrCreate(this.fieldsAll, key, () => {
86
+ const stored = this.storedSchema.getFieldSchema(key).kind;
87
+ const field = createField(this.context, stored, key, []);
88
+ field.parent = this;
89
+ return field;
90
+ });
71
91
  }
72
92
  adoptBy(parent, index) {
73
93
  if (parent !== undefined) {
74
94
  assert(index !== undefined, 0xa08 /* Expected index */);
75
95
  if (this.location !== unparentedLocation) {
76
- throw new UsageError("A node may not be inserted if it's already in a tree");
96
+ throw new UsageError("A node may not be in more than one place in the tree");
77
97
  }
78
98
  let unhydratedNode = parent.parent;
79
99
  while (unhydratedNode !== undefined) {
@@ -99,49 +119,27 @@ export class UnhydratedFlexTreeNode {
99
119
  return this.location;
100
120
  }
101
121
  borrowCursor() {
102
- return cursorForMapTreeNode(this.mapTree);
122
+ return cursorForMapTreeNode(this);
103
123
  }
104
124
  tryGetField(key) {
105
- const field = this.mapTree.fields.get(key);
125
+ const field = this.fieldsAll.get(key);
106
126
  // Only return the field if it is not empty, in order to fulfill the contract of `tryGetField`.
107
127
  if (field !== undefined && field.length > 0) {
108
- return getOrCreateField(this, key, this.storedSchema.getFieldSchema(key).kind, () => this.emitChangedEvent(key));
128
+ return field;
109
129
  }
110
130
  }
111
131
  getBoxed(key) {
112
132
  const fieldKey = brand(key);
113
- return getOrCreateField(this, fieldKey, this.storedSchema.getFieldSchema(fieldKey).kind, () => this.emitChangedEvent(fieldKey));
133
+ return this.getOrCreateField(fieldKey);
114
134
  }
115
135
  boxedIterator() {
116
- return mapIterable(this.mapTree.fields.entries(), ([key]) => getOrCreateField(this, key, this.storedSchema.getFieldSchema(key).kind, () => this.emitChangedEvent(key)));
136
+ return Array.from(this.fields, ([key, field]) => field)[Symbol.iterator]();
117
137
  }
118
138
  keys() {
119
- // TODO: how this should handle missing defaults (and empty keys if they end up being allowed) needs to be determined.
120
- return this.mapTree.fields.keys();
139
+ return Array.from(this.fields, ([key]) => key)[Symbol.iterator]();
121
140
  }
122
141
  get value() {
123
- return this.mapTree.value;
124
- }
125
- get anchorNode() {
126
- // This API is relevant to `LazyTreeNode`s, but not `UnhydratedFlexTreeNode`s.
127
- // TODO: Refactor the FlexTreeNode interface so that stubbing this out isn't necessary.
128
- return fail(0xb47 /* UnhydratedFlexTreeNode does not implement anchorNode */);
129
- }
130
- walkTree() {
131
- for (const [key, mapTrees] of this.mapTree.fields) {
132
- const field = getOrCreateField(this, key, this.storedSchema.getFieldSchema(key).kind, () => this.emitChangedEvent(key));
133
- for (let index = 0; index < field.length; index++) {
134
- const child = getOrCreateChild(this.simpleContext, mapTrees[index] ?? oob(), {
135
- parent: field,
136
- index,
137
- });
138
- // These next asserts detect the case where `getOrCreateChild` gets a cache hit of a different node than the one we're trying to create
139
- assert(child.location !== undefined, 0x98d /* Expected node to have parent */);
140
- assert(child.location.parent.parent === this, 0x98e /* Node may not be multi-parented */);
141
- assert(child.location.index === index, 0x98f /* Node may not be multi-parented */);
142
- child.walkTree();
143
- }
144
- }
142
+ return this.data.value;
145
143
  }
146
144
  emitChangedEvent(key) {
147
145
  this._events.emit("childrenChangedAfterBatch", { changedFields: new Set([key]) });
@@ -202,61 +200,82 @@ const unparentedLocation = {
202
200
  },
203
201
  index: -1,
204
202
  };
205
- class UnhydratedFlexTreeField {
206
- get context() {
207
- return this.simpleContext.flexContext;
208
- }
209
- constructor(simpleContext, schema, key, parent, onEdit) {
210
- this.simpleContext = simpleContext;
203
+ /**
204
+ * The {@link Unhydrated} implementation of {@link FlexTreeField}.
205
+ */
206
+ export class UnhydratedFlexTreeField {
207
+ constructor(context, schema, key,
208
+ /**
209
+ * The children of this field.
210
+ * @remarks
211
+ * This is either an array of {@link UnhydratedFlexTreeNode}s or a {@link ContextualFieldProvider} that will be used to populate the children lazily (after which it will become an array).
212
+ * See {@link fillPendingDefaults}.
213
+ * Note that any fields using a {@link ConstantFieldProvider} should be evaluated before constructing the UnhydratedFlexTreeField.
214
+ */
215
+ lazyChildren) {
216
+ this.context = context;
211
217
  this.schema = schema;
212
218
  this.key = key;
213
- this.parent = parent;
214
- this.onEdit = onEdit;
219
+ this.lazyChildren = lazyChildren;
215
220
  this[_b] = FlexTreeEntityKind.Field;
216
- const fieldKeyCache = getFieldKeyCache(parent);
217
- assert(!fieldKeyCache.has(key), 0x990 /* A field already exists for the given MapTrees */);
218
- fieldKeyCache.set(key, this);
221
+ this.parent = undefined;
219
222
  // When this field is created (which only happens one time, because it is cached), all the children become parented for the first time.
220
223
  // "Adopt" each child by updating its parent information to point to this field.
221
- for (const [i, mapTree] of this.mapTrees.entries()) {
222
- const mapTreeNodeChild = nodeCache.get(mapTree);
223
- if (mapTreeNodeChild !== undefined) {
224
- if (mapTreeNodeChild.parentField !== unparentedLocation) {
225
- throw new UsageError("A node may not be in more than one place in the tree");
226
- }
227
- mapTreeNodeChild.adoptBy(this, i);
224
+ if (Array.isArray(lazyChildren)) {
225
+ for (const [i, child] of lazyChildren.entries()) {
226
+ child.adoptBy(this, i);
228
227
  }
229
228
  }
230
229
  }
231
- get mapTrees() {
232
- return this.parent.mapTree.fields.get(this.key) ?? [];
230
+ getPendingDefault() {
231
+ return !Array.isArray(this.lazyChildren) ? this.lazyChildren : undefined;
232
+ }
233
+ /**
234
+ * Populate pending default (if present) using the provided context.
235
+ * @remarks
236
+ * This apply to just this field: caller will likely want to recursively walk the tree.
237
+ * @see {@link pendingDefault}.
238
+ */
239
+ fillPendingDefaults(context) {
240
+ const provider = this.getPendingDefault();
241
+ if (provider) {
242
+ const content = provider(context);
243
+ this.lazyChildren = content;
244
+ }
245
+ }
246
+ /**
247
+ * Returns true if this field has a pending default due to defined defined using a {@link ContextualFieldProvider}.
248
+ */
249
+ get pendingDefault() {
250
+ return this.getPendingDefault() !== undefined;
251
+ }
252
+ get children() {
253
+ const provider = this.getPendingDefault();
254
+ if (provider) {
255
+ const content = provider("UseGlobalContext");
256
+ this.lazyChildren = content;
257
+ }
258
+ return this.lazyChildren;
233
259
  }
234
260
  get length() {
235
- return this.mapTrees.length;
261
+ return this.children.length;
236
262
  }
237
263
  is(kind) {
238
264
  return this.schema === kind.identifier;
239
265
  }
240
266
  boxedIterator() {
241
- return this.mapTrees
242
- .map((m, index) => getOrCreateChild(this.simpleContext, m, {
243
- parent: this,
244
- index,
245
- }))
246
- .values();
267
+ return this.children[Symbol.iterator]();
247
268
  }
248
269
  boxedAt(index) {
249
270
  const i = indexForAt(index, this.length);
250
271
  if (i === undefined) {
251
272
  return undefined;
252
273
  }
253
- const m = this.mapTrees[i];
254
- if (m !== undefined) {
255
- return getOrCreateChild(this.simpleContext, m, {
256
- parent: this,
257
- index: i,
258
- });
259
- }
274
+ const m = this.children[i];
275
+ return m;
276
+ }
277
+ [(_b = flexTreeMarker, Symbol.iterator)]() {
278
+ return this.boxedIterator();
260
279
  }
261
280
  /**
262
281
  * Mutate this field.
@@ -267,43 +286,40 @@ class UnhydratedFlexTreeField {
267
286
  * This function ensures that the parent MapTree has no empty fields (which is an invariant of `MapTree`) after the mutation.
268
287
  */
269
288
  edit(edit) {
270
- const oldMapTrees = this.parent.mapTree.fields.get(this.key) ?? [];
271
- const newMapTrees = edit(oldMapTrees) ?? oldMapTrees;
272
- if (newMapTrees.length > 0) {
273
- this.parent.mapTree.fields.set(this.key, newMapTrees);
289
+ // Clear parents for all old map trees.
290
+ for (const tree of this.children) {
291
+ tree.adoptBy(undefined);
274
292
  }
275
- else {
276
- this.parent.mapTree.fields.delete(this.key);
293
+ this.lazyChildren = edit(this.children) ?? this.children;
294
+ // Set parents for all new map trees.
295
+ for (const [index, tree] of this.children.entries()) {
296
+ tree.adoptBy(this, index);
277
297
  }
278
- this.onEdit?.();
298
+ this.parent?.emitChangedEvent(this.key);
279
299
  }
280
300
  getFieldPath() {
281
301
  throw unsupportedUsageError("Editing an array");
282
302
  }
283
303
  /** Unboxes leaf nodes to their values */
284
304
  unboxed(index) {
285
- const mapTree = this.mapTrees[index] ?? oob();
286
- const value = mapTree.value;
305
+ const child = this.children[index] ?? oob();
306
+ const value = child.value;
287
307
  if (value !== undefined) {
288
308
  return value;
289
309
  }
290
- return getOrCreateChild(this.simpleContext, mapTree, { parent: this, index });
310
+ return child;
291
311
  }
292
312
  }
293
- _b = flexTreeMarker;
294
- class EagerMapTreeOptionalField extends UnhydratedFlexTreeField {
313
+ /**
314
+ * The {@link Unhydrated} implementation of {@link FlexTreeOptionalField}.
315
+ */
316
+ export class UnhydratedOptionalField extends UnhydratedFlexTreeField {
295
317
  constructor() {
296
318
  super(...arguments);
297
319
  this.editor = {
298
320
  set: (newContent) => {
299
- // If the new content is a UnhydratedFlexTreeNode, it needs to have its parent pointer updated
300
321
  if (newContent !== undefined) {
301
- nodeCache.get(newContent)?.adoptBy(this, 0);
302
- }
303
- // If the old content is a UnhydratedFlexTreeNode, it needs to have its parent pointer unset
304
- const oldContent = this.mapTrees[0];
305
- if (oldContent !== undefined) {
306
- nodeCache.get(oldContent)?.adoptBy(undefined);
322
+ assert(newContent instanceof UnhydratedFlexTreeNode, 0xbb7 /* Expected unhydrated node */);
307
323
  }
308
324
  this.edit((mapTrees) => {
309
325
  if (newContent !== undefined) {
@@ -317,46 +333,48 @@ class EagerMapTreeOptionalField extends UnhydratedFlexTreeField {
317
333
  };
318
334
  }
319
335
  get content() {
320
- const value = this.mapTrees[0];
336
+ const value = this.children[0];
321
337
  if (value !== undefined) {
322
338
  return this.unboxed(0);
323
339
  }
324
340
  return undefined;
325
341
  }
326
342
  }
327
- class EagerMapTreeRequiredField extends EagerMapTreeOptionalField {
343
+ class UnhydratedRequiredField extends UnhydratedOptionalField {
328
344
  get content() {
329
345
  // This cannot use ?? since null is a legal value here.
330
346
  assert(super.content !== undefined, 0xa57 /* Expected EagerMapTree required field to have a value */);
331
347
  return super.content;
332
348
  }
333
349
  }
334
- export class UnhydratedTreeSequenceField extends UnhydratedFlexTreeField {
350
+ /**
351
+ * The {@link Unhydrated} implementation of {@link FlexTreeSequenceField}.
352
+ */
353
+ export class UnhydratedSequenceField extends UnhydratedFlexTreeField {
335
354
  constructor() {
336
355
  super(...arguments);
337
356
  this.editor = {
338
357
  insert: (index, newContent) => {
339
- for (let i = 0; i < newContent.length; i++) {
340
- const c = newContent[i];
358
+ for (const c of newContent) {
341
359
  assert(c !== undefined, 0xa0a /* Unexpected sparse array content */);
342
- nodeCache.get(c)?.adoptBy(this, index + i);
360
+ assert(c instanceof UnhydratedFlexTreeNode, 0xbb8 /* Expected unhydrated node */);
343
361
  }
362
+ const newContentChecked = newContent;
344
363
  this.edit((mapTrees) => {
345
364
  if (newContent.length < 1000) {
346
365
  // For "smallish arrays" (`1000` is not empirically derived), the `splice` function is appropriate...
347
- mapTrees.splice(index, 0, ...newContent);
366
+ mapTrees.splice(index, 0, ...newContentChecked);
348
367
  }
349
368
  else {
350
369
  // ...but we avoid using `splice` + spread for very large input arrays since there is a limit on how many elements can be spread (too many will overflow the stack).
351
- return mapTrees.slice(0, index).concat(newContent, mapTrees.slice(index));
370
+ return mapTrees.slice(0, index).concat(newContentChecked, mapTrees.slice(index));
352
371
  }
353
372
  });
354
373
  },
355
374
  remove: (index, count) => {
356
375
  for (let i = index; i < index + count; i++) {
357
- const c = this.mapTrees[i];
376
+ const c = this.children[i];
358
377
  assert(c !== undefined, 0xa0b /* Unexpected sparse array */);
359
- nodeCache.get(c)?.adoptBy(undefined);
360
378
  }
361
379
  let removed;
362
380
  this.edit((mapTrees) => {
@@ -376,58 +394,25 @@ export class UnhydratedTreeSequenceField extends UnhydratedFlexTreeField {
376
394
  map(callbackfn) {
377
395
  return Array.from(this, callbackfn);
378
396
  }
379
- *[Symbol.iterator]() {
380
- for (const [i] of this.mapTrees.entries()) {
381
- yield this.unboxed(i);
382
- }
383
- }
384
397
  }
385
398
  // #endregion Fields
386
- // #region Caching and unboxing utilities
387
- const nodeCache = new WeakMap();
388
- /** Node Parent -\> Field Key -\> Field */
389
- const fieldCache = new WeakMap();
390
- function getFieldKeyCache(parent) {
391
- return getOrCreate(fieldCache, parent, () => new Map());
392
- }
393
- /**
394
- * If there exists a {@link UnhydratedFlexTreeNode} for the given {@link MapTree}, returns it, otherwise returns `undefined`.
395
- * @remarks {@link UnhydratedFlexTreeNode | UnhydratedFlexTreeNodes} are created via {@link getOrCreateNodeFromInnerNode}.
396
- */
397
- export function tryUnhydratedFlexTreeNode(mapTree) {
398
- return nodeCache.get(mapTree);
399
- }
400
- /** Helper for creating a `UnhydratedFlexTreeNode` given the parent field (e.g. when "walking down") */
401
- function getOrCreateChild(context, mapTree, parent) {
402
- const cached = nodeCache.get(mapTree);
403
- if (cached !== undefined) {
404
- return cached;
405
- }
406
- return new UnhydratedFlexTreeNode(context, mapTree, parent);
407
- }
408
- /** Creates a field with the given attributes, or returns a cached field if there is one */
409
- function getOrCreateField(parent, key, schema, onEdit) {
410
- const cached = getFieldKeyCache(parent).get(key);
411
- if (cached !== undefined) {
412
- return cached;
413
- }
414
- if (schema === FieldKinds.required.identifier ||
415
- schema === FieldKinds.identifier.identifier) {
416
- return new EagerMapTreeRequiredField(parent.simpleContext, schema, key, parent, onEdit);
417
- }
418
- if (schema === FieldKinds.optional.identifier) {
419
- return new EagerMapTreeOptionalField(parent.simpleContext, schema, key, parent, onEdit);
420
- }
421
- if (schema === FieldKinds.sequence.identifier) {
422
- return new UnhydratedTreeSequenceField(parent.simpleContext, schema, key, parent, onEdit);
423
- }
424
- // TODO: this seems to used by unknown optional fields. They should probably use "optional" not "Forbidden" schema.
425
- if (schema === FieldKinds.forbidden.identifier) {
426
- return new UnhydratedFlexTreeField(parent.simpleContext, schema, key, parent, onEdit);
399
+ /** Creates a field with the given attributes */
400
+ export function createField(...args) {
401
+ switch (args[1]) {
402
+ case FieldKinds.required.identifier:
403
+ case FieldKinds.identifier.identifier:
404
+ return new UnhydratedRequiredField(...args);
405
+ case FieldKinds.optional.identifier:
406
+ return new UnhydratedOptionalField(...args);
407
+ case FieldKinds.sequence.identifier:
408
+ return new UnhydratedSequenceField(...args);
409
+ case FieldKinds.forbidden.identifier:
410
+ // TODO: this seems to used by unknown optional fields. They should probably use "optional" not "Forbidden" schema.
411
+ return new UnhydratedFlexTreeField(...args);
412
+ default:
413
+ return fail(0xb9d /* unsupported field kind */);
427
414
  }
428
- return fail(0xb9d /* unsupported field kind */);
429
415
  }
430
- // #endregion Caching and unboxing utilities
431
416
  export function unsupportedUsageError(message) {
432
417
  return new UsageError(`${message ?? "Operation"} is not supported for content that has not yet been inserted into the tree`);
433
418
  }