cx 26.0.12 → 26.0.13

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 (363) hide show
  1. package/build/charts/ScatterGraph.d.ts +4 -4
  2. package/build/charts/ScatterGraph.js +2 -2
  3. package/build/data/ArrayElementView.spec.d.ts +1 -0
  4. package/build/data/ArrayElementView.spec.js +81 -0
  5. package/build/data/Binding.spec.d.ts +1 -0
  6. package/build/data/Binding.spec.js +61 -0
  7. package/build/data/Expression.spec.d.ts +1 -0
  8. package/build/data/Expression.spec.js +196 -0
  9. package/build/data/Grouper.spec.d.ts +1 -0
  10. package/build/data/Grouper.spec.js +48 -0
  11. package/build/data/Ref.spec.d.ts +1 -0
  12. package/build/data/Ref.spec.js +72 -0
  13. package/build/data/Selector.d.ts +1 -1
  14. package/build/data/Store.spec.d.ts +1 -0
  15. package/build/data/Store.spec.js +19 -0
  16. package/build/data/StoreRef.spec.d.ts +1 -0
  17. package/build/data/StoreRef.spec.js +22 -0
  18. package/build/data/StringTemplate.spec.d.ts +1 -0
  19. package/build/data/StringTemplate.spec.js +112 -0
  20. package/build/data/StructuredSelector.spec.d.ts +1 -0
  21. package/build/data/StructuredSelector.spec.js +102 -0
  22. package/build/data/View.spec.d.ts +1 -0
  23. package/build/data/View.spec.js +44 -0
  24. package/build/data/ZoomIntoPropertyView.spec.d.ts +1 -0
  25. package/build/data/ZoomIntoPropertyView.spec.js +54 -0
  26. package/build/data/comparer.spec.d.ts +1 -0
  27. package/build/data/comparer.spec.js +50 -0
  28. package/build/data/computable.d.ts +3 -6
  29. package/build/data/computable.spec.d.ts +1 -0
  30. package/build/data/computable.spec.js +56 -0
  31. package/build/data/createAccessorModelProxy.d.ts +5 -3
  32. package/build/data/createAccessorModelProxy.spec.d.ts +1 -0
  33. package/build/data/createAccessorModelProxy.spec.js +30 -0
  34. package/build/data/createStructuredSelector.spec.d.ts +1 -0
  35. package/build/data/createStructuredSelector.spec.js +42 -0
  36. package/build/data/diff/diffs.spec.d.ts +1 -0
  37. package/build/data/diff/diffs.spec.js +45 -0
  38. package/build/data/getAccessor.spec.d.ts +1 -0
  39. package/build/data/getAccessor.spec.js +10 -0
  40. package/build/data/getSelector.spec.d.ts +1 -0
  41. package/build/data/getSelector.spec.js +36 -0
  42. package/build/data/ops/append.spec.d.ts +1 -0
  43. package/build/data/ops/append.spec.js +24 -0
  44. package/build/data/ops/filter.spec.d.ts +1 -0
  45. package/build/data/ops/filter.spec.js +25 -0
  46. package/build/data/ops/findTreeNode.d.ts +1 -1
  47. package/build/data/ops/findTreeNode.js +1 -1
  48. package/build/data/ops/findTreeNode.spec.d.ts +1 -0
  49. package/build/data/ops/findTreeNode.spec.js +20 -0
  50. package/build/data/ops/findTreePath.d.ts +1 -1
  51. package/build/data/ops/merge.spec.d.ts +1 -0
  52. package/build/data/ops/merge.spec.js +23 -0
  53. package/build/data/ops/removeTreeNodes.d.ts +1 -1
  54. package/build/data/ops/removeTreeNodes.js +1 -1
  55. package/build/data/ops/removeTreeNodes.spec.d.ts +1 -0
  56. package/build/data/ops/removeTreeNodes.spec.js +35 -0
  57. package/build/data/ops/updateArray.spec.d.ts +1 -0
  58. package/build/data/ops/updateArray.spec.js +33 -0
  59. package/build/data/ops/updateTree.d.ts +1 -1
  60. package/build/data/ops/updateTree.spec.d.ts +1 -0
  61. package/build/data/ops/updateTree.spec.js +44 -0
  62. package/build/hooks/invokeCallback.spec.d.ts +1 -0
  63. package/build/hooks/invokeCallback.spec.js +44 -0
  64. package/build/hooks/resolveCallback.spec.d.ts +1 -0
  65. package/build/hooks/resolveCallback.spec.js +35 -0
  66. package/build/hooks/store.spec.d.ts +1 -0
  67. package/build/hooks/store.spec.js +48 -0
  68. package/build/hooks/useTrigger.spec.d.ts +1 -0
  69. package/build/hooks/useTrigger.spec.js +59 -0
  70. package/build/jsx-runtime.d.ts +10 -10
  71. package/build/jsx-runtime.js +6 -0
  72. package/build/svg/util/Rect.d.ts +1 -1
  73. package/build/ui/ContentResolver.d.ts +16 -6
  74. package/build/ui/Controller.d.ts +7 -0
  75. package/build/ui/Controller.js +2 -1
  76. package/build/ui/Controller.spec.d.ts +1 -0
  77. package/build/ui/Controller.spec.js +247 -0
  78. package/build/ui/Cx.spec.d.ts +1 -0
  79. package/build/ui/Cx.spec.js +153 -0
  80. package/build/ui/DataProxy.spec.d.ts +1 -0
  81. package/build/ui/DataProxy.spec.js +208 -0
  82. package/build/ui/Instance.d.ts +1 -1
  83. package/build/ui/Instance.js +10 -10
  84. package/build/ui/IsolatedScope.spec.d.ts +1 -0
  85. package/build/ui/IsolatedScope.spec.js +42 -0
  86. package/build/ui/Prop.d.ts +12 -1
  87. package/build/ui/PureContainer.spec.d.ts +1 -0
  88. package/build/ui/PureContainer.spec.js +149 -0
  89. package/build/ui/Repeater.d.ts +3 -3
  90. package/build/ui/Repeater.spec.d.ts +1 -0
  91. package/build/ui/Repeater.spec.js +109 -0
  92. package/build/ui/Rescope.spec.d.ts +1 -0
  93. package/build/ui/Rescope.spec.js +134 -0
  94. package/build/ui/Restate.spec.d.ts +1 -0
  95. package/build/ui/Restate.spec.js +257 -0
  96. package/build/ui/Text.d.ts +14 -2
  97. package/build/ui/Text.js +3 -0
  98. package/build/ui/adapter/ArrayAdapter.js +4 -1
  99. package/build/ui/adapter/ArrayAdapter.spec.d.ts +1 -0
  100. package/build/ui/adapter/ArrayAdapter.spec.js +44 -0
  101. package/build/ui/adapter/TreeAdapter.spec.d.ts +1 -0
  102. package/build/ui/adapter/TreeAdapter.spec.js +71 -0
  103. package/build/ui/app/Url.spec.d.ts +1 -0
  104. package/build/ui/app/Url.spec.js +43 -0
  105. package/build/ui/app/startHotAppLoop.js +1 -1
  106. package/build/ui/createFunctionalComponent.d.ts +14 -1
  107. package/build/ui/createFunctionalComponent.js +7 -4
  108. package/build/ui/createFunctionalComponent.spec.d.ts +1 -0
  109. package/build/ui/createFunctionalComponent.spec.js +272 -0
  110. package/build/ui/expr.d.ts +3 -1
  111. package/build/ui/exprHelpers.d.ts +32 -0
  112. package/build/ui/exprHelpers.js +61 -0
  113. package/build/ui/index.d.ts +1 -0
  114. package/build/ui/index.js +1 -0
  115. package/build/ui/layout/ContentPlaceholder.spec.d.ts +1 -0
  116. package/build/ui/layout/ContentPlaceholder.spec.js +333 -0
  117. package/build/ui/layout/FirstVisibleChildLayout.spec.d.ts +1 -0
  118. package/build/ui/layout/FirstVisibleChildLayout.spec.js +101 -0
  119. package/build/util/Console.d.ts +1 -0
  120. package/build/util/Console.js +7 -3
  121. package/build/util/Format.spec.d.ts +1 -0
  122. package/build/util/Format.spec.js +58 -0
  123. package/build/util/TraversalStack.spec.d.ts +1 -0
  124. package/build/util/TraversalStack.spec.js +43 -0
  125. package/build/util/date/upperBoundCheck.spec.d.ts +1 -0
  126. package/build/util/date/upperBoundCheck.spec.js +22 -0
  127. package/build/util/getSearchQueryPredicate.spec.d.ts +1 -0
  128. package/build/util/getSearchQueryPredicate.spec.js +33 -0
  129. package/build/util/isValidIdentifierName.spec.d.ts +1 -0
  130. package/build/util/isValidIdentifierName.spec.js +28 -0
  131. package/build/util/routeAppend.spec.d.ts +1 -0
  132. package/build/util/routeAppend.spec.js +14 -0
  133. package/build/widgets/AccessorBindings.spec.d.ts +1 -0
  134. package/build/widgets/AccessorBindings.spec.js +40 -0
  135. package/build/widgets/Button.d.ts +3 -6
  136. package/build/widgets/Button.js +1 -1
  137. package/build/widgets/DocumentTitle.d.ts +2 -0
  138. package/build/widgets/Heading.d.ts +2 -2
  139. package/build/widgets/HtmlElement.d.ts +31 -8
  140. package/build/widgets/HtmlElement.js +7 -9
  141. package/build/widgets/HtmlElement.spec.d.ts +1 -0
  142. package/build/widgets/HtmlElement.spec.js +38 -0
  143. package/build/widgets/Icon.d.ts +3 -13
  144. package/build/widgets/List.d.ts +4 -0
  145. package/build/widgets/ReactElementWrapper.d.ts +29 -0
  146. package/build/widgets/ReactElementWrapper.js +59 -0
  147. package/build/widgets/drag-drop/DragSource.d.ts +3 -3
  148. package/build/widgets/drag-drop/DragSource.js +2 -3
  149. package/build/widgets/drag-drop/DropZone.d.ts +3 -3
  150. package/build/widgets/drag-drop/DropZone.js +2 -3
  151. package/build/widgets/form/Checkbox.d.ts +2 -0
  152. package/build/widgets/form/ColorField.d.ts +2 -0
  153. package/build/widgets/form/DateTimeField.d.ts +2 -0
  154. package/build/widgets/form/Field.d.ts +0 -2
  155. package/build/widgets/form/LabeledContainer.d.ts +9 -8
  156. package/build/widgets/form/LabeledContainer.js +9 -9
  157. package/build/widgets/form/LookupField.d.ts +57 -9
  158. package/build/widgets/form/MonthField.d.ts +2 -0
  159. package/build/widgets/form/NumberField.d.ts +2 -0
  160. package/build/widgets/form/Radio.d.ts +2 -0
  161. package/build/widgets/form/Select.d.ts +2 -0
  162. package/build/widgets/form/Slider.d.ts +3 -0
  163. package/build/widgets/form/Switch.d.ts +2 -0
  164. package/build/widgets/form/TextField.d.ts +34 -0
  165. package/build/widgets/form/TimeList.d.ts +16 -1
  166. package/build/widgets/form/TimeList.js +34 -62
  167. package/build/widgets/form/UploadButton.d.ts +34 -2
  168. package/build/widgets/form/UploadButton.js +3 -1
  169. package/build/widgets/form/ValidationGroup.spec.d.ts +1 -0
  170. package/build/widgets/form/ValidationGroup.spec.js +62 -0
  171. package/build/widgets/form/Validator.d.ts +33 -2
  172. package/build/widgets/form/Validator.js +3 -0
  173. package/build/widgets/grid/Grid.d.ts +9 -9
  174. package/build/widgets/grid/TreeNode.d.ts +6 -0
  175. package/build/widgets/index.d.ts +1 -0
  176. package/build/widgets/index.js +1 -0
  177. package/build/widgets/nav/MenuItem.d.ts +3 -2
  178. package/build/widgets/nav/Route.spec.d.ts +1 -0
  179. package/build/widgets/nav/Route.spec.js +15 -0
  180. package/build/widgets/nav/Scroller.d.ts +4 -6
  181. package/build/widgets/nav/Scroller.js +6 -3
  182. package/build/widgets/nav/Tab.d.ts +2 -2
  183. package/build/widgets/overlay/ContextMenu.d.ts +3 -3
  184. package/build/widgets/overlay/Overlay.d.ts +2 -1
  185. package/build/widgets/overlay/Overlay.js +1 -1
  186. package/build.js +133 -133
  187. package/dist/data.js +2 -2
  188. package/dist/jsx-runtime.js +6 -1
  189. package/dist/manifest.d.ts +1443 -0
  190. package/dist/manifest.js +855 -807
  191. package/dist/ui.js +91 -5
  192. package/dist/util.js +3 -0
  193. package/dist/widgets.js +520 -161
  194. package/package.json +46 -20
  195. package/src/charts/Chart.ts +108 -108
  196. package/src/charts/ScatterGraph.tsx +6 -6
  197. package/src/data/ArrayElementView.ts +90 -90
  198. package/src/data/AugmentedViewBase.ts +88 -88
  199. package/src/data/Binding.ts +104 -104
  200. package/src/data/ExposedRecordView.ts +95 -95
  201. package/src/data/ExposedValueView.ts +89 -89
  202. package/src/data/Expression.spec.ts +229 -229
  203. package/src/data/Expression.ts +233 -233
  204. package/src/data/Grouper.spec.ts +57 -57
  205. package/src/data/Grouper.ts +158 -158
  206. package/src/data/NestedDataView.ts +43 -43
  207. package/src/data/ReadOnlyDataView.ts +39 -39
  208. package/src/data/Ref.ts +104 -104
  209. package/src/data/Selector.ts +10 -10
  210. package/src/data/Store.ts +52 -52
  211. package/src/data/StoreProxy.ts +19 -19
  212. package/src/data/StoreRef.ts +66 -66
  213. package/src/data/StringTemplate.spec.ts +132 -132
  214. package/src/data/StringTemplate.ts +93 -93
  215. package/src/data/StructuredSelector.spec.ts +113 -113
  216. package/src/data/StructuredSelector.ts +146 -146
  217. package/src/data/SubscribableView.ts +63 -63
  218. package/src/data/ZoomIntoPropertyView.spec.ts +64 -64
  219. package/src/data/ZoomIntoPropertyView.ts +45 -45
  220. package/src/data/computable.spec.ts +87 -62
  221. package/src/data/computable.ts +3 -6
  222. package/src/data/createAccessorModelProxy.spec.tsx +102 -1
  223. package/src/data/createAccessorModelProxy.ts +9 -3
  224. package/src/data/createStructuredSelector.ts +62 -62
  225. package/src/data/getAccessor.spec.ts +11 -11
  226. package/src/data/getAccessor.ts +74 -74
  227. package/src/data/getSelector.spec.ts +43 -43
  228. package/src/data/getSelector.ts +66 -66
  229. package/src/data/ops/filter.spec.ts +35 -35
  230. package/src/data/ops/filter.ts +9 -9
  231. package/src/data/ops/findTreeNode.ts +1 -5
  232. package/src/data/ops/findTreePath.ts +1 -1
  233. package/src/data/ops/merge.ts +13 -13
  234. package/src/data/ops/removeTreeNodes.spec.ts +37 -37
  235. package/src/data/ops/removeTreeNodes.ts +2 -2
  236. package/src/data/ops/updateArray.spec.ts +69 -69
  237. package/src/data/ops/updateArray.ts +31 -31
  238. package/src/data/ops/updateTree.ts +1 -1
  239. package/src/data/test-types.ts +7 -7
  240. package/src/hooks/resolveCallback.spec.tsx +30 -7
  241. package/src/hooks/useTrigger.ts +26 -26
  242. package/src/index.scss +6 -6
  243. package/src/jsx-dev-runtime.ts +4 -4
  244. package/src/jsx-runtime.spec.tsx +402 -0
  245. package/src/jsx-runtime.ts +26 -22
  246. package/src/svg/BoundedObject.ts +101 -101
  247. package/src/svg/util/Rect.ts +105 -105
  248. package/src/ui/CSSHelper.ts +17 -17
  249. package/src/ui/ContentResolver.spec.tsx +172 -19
  250. package/src/ui/ContentResolver.ts +16 -8
  251. package/src/ui/Controller.ts +15 -2
  252. package/src/ui/Culture.ts +159 -159
  253. package/src/ui/DataProxy.ts +55 -55
  254. package/src/ui/FocusManager.ts +171 -171
  255. package/src/ui/Instance.ts +866 -868
  256. package/src/ui/Prop.ts +140 -112
  257. package/src/ui/RenderingContext.ts +99 -99
  258. package/src/ui/Repeater.ts +3 -12
  259. package/src/ui/Rescope.ts +49 -49
  260. package/src/ui/StructuredInstanceDataAccessor.ts +32 -32
  261. package/src/ui/Text.ts +21 -2
  262. package/src/ui/VDOM.ts +34 -34
  263. package/src/ui/adapter/ArrayAdapter.spec.ts +55 -55
  264. package/src/ui/adapter/ArrayAdapter.ts +4 -1
  265. package/src/ui/adapter/TreeAdapter.spec.ts +76 -76
  266. package/src/ui/adapter/TreeAdapter.ts +185 -185
  267. package/src/ui/app/History.ts +133 -133
  268. package/src/ui/app/Url.spec.ts +50 -50
  269. package/src/ui/app/startHotAppLoop.ts +41 -41
  270. package/src/ui/createFunctionalComponent.spec.tsx +53 -0
  271. package/src/ui/createFunctionalComponent.ts +86 -65
  272. package/src/ui/expr.ts +4 -1
  273. package/src/ui/exprHelpers.spec.ts +379 -0
  274. package/src/ui/exprHelpers.ts +78 -0
  275. package/src/ui/index.ts +47 -46
  276. package/src/ui/layout/Content.ts +30 -30
  277. package/src/ui/layout/FirstVisibleChildLayout.spec.tsx +1 -1
  278. package/src/ui/layout/FirstVisibleChildLayout.ts +60 -60
  279. package/src/ui/selection/KeySelection.ts +153 -153
  280. package/src/ui/selection/Selection.ts +128 -128
  281. package/src/util/Console.ts +13 -11
  282. package/src/util/Format.ts +267 -267
  283. package/src/util/addEventListenerWithOptions.ts +41 -41
  284. package/src/util/browserSupportsPassiveEventHandlers.ts +20 -20
  285. package/src/util/color/rgbToHsl.ts +35 -35
  286. package/src/util/getActiveElement.ts +4 -4
  287. package/src/util/hasKey.ts +18 -18
  288. package/src/util/index.ts +55 -55
  289. package/src/util/innerTextTrim.ts +10 -10
  290. package/src/util/isArray.ts +3 -3
  291. package/src/util/isDataRecord.ts +5 -5
  292. package/src/util/isDefined.ts +3 -3
  293. package/src/util/isString.ts +3 -3
  294. package/src/widgets/AccessorBindings.spec.tsx +26 -0
  295. package/src/widgets/Button.tsx +5 -17
  296. package/src/widgets/DocumentTitle.ts +95 -92
  297. package/src/widgets/Heading.ts +2 -2
  298. package/src/widgets/HtmlElement.spec.helpers.tsx +108 -0
  299. package/src/widgets/HtmlElement.spec.tsx +20 -12
  300. package/src/widgets/HtmlElement.tsx +77 -23
  301. package/src/widgets/Icon.ts +3 -17
  302. package/src/widgets/List.tsx +6 -0
  303. package/src/widgets/ReactElementWrapper.spec.tsx +452 -0
  304. package/src/widgets/ReactElementWrapper.tsx +108 -0
  305. package/src/widgets/Sandbox.ts +103 -103
  306. package/src/widgets/autoFocus.ts +9 -9
  307. package/src/widgets/cx.ts +63 -63
  308. package/src/widgets/drag-drop/DragSource.tsx +3 -4
  309. package/src/widgets/drag-drop/DropZone.tsx +3 -4
  310. package/src/widgets/form/Checkbox.tsx +3 -0
  311. package/src/widgets/form/ColorField.tsx +3 -0
  312. package/src/widgets/form/DateTimeField.tsx +5 -0
  313. package/src/widgets/form/Field.tsx +0 -3
  314. package/src/widgets/form/Label.tsx +1 -0
  315. package/src/widgets/form/LabeledContainer.ts +22 -26
  316. package/src/widgets/form/LookupField.spec.tsx +93 -0
  317. package/src/widgets/form/LookupField.tsx +104 -9
  318. package/src/widgets/form/MonthField.tsx +5 -0
  319. package/src/widgets/form/NumberField.tsx +3 -0
  320. package/src/widgets/form/Radio.tsx +5 -0
  321. package/src/widgets/form/Select.tsx +5 -0
  322. package/src/widgets/form/Slider.tsx +4 -0
  323. package/src/widgets/form/Switch.tsx +3 -0
  324. package/src/widgets/form/TextField.tsx +62 -0
  325. package/src/widgets/form/TimeList.tsx +84 -73
  326. package/src/widgets/form/UploadButton.tsx +53 -2
  327. package/src/widgets/form/Validator.ts +40 -3
  328. package/src/widgets/grid/Grid.tsx +9 -12
  329. package/src/widgets/grid/GridCell.ts +143 -143
  330. package/src/widgets/grid/TreeNode.tsx +9 -0
  331. package/src/widgets/icons/calendar.tsx +17 -17
  332. package/src/widgets/icons/check.tsx +13 -13
  333. package/src/widgets/icons/clear.tsx +15 -15
  334. package/src/widgets/icons/close.tsx +20 -20
  335. package/src/widgets/icons/cx.tsx +38 -38
  336. package/src/widgets/icons/drop-down.tsx +15 -15
  337. package/src/widgets/icons/file.tsx +13 -13
  338. package/src/widgets/icons/folder-open.tsx +15 -15
  339. package/src/widgets/icons/folder.tsx +13 -13
  340. package/src/widgets/icons/forward.tsx +22 -22
  341. package/src/widgets/icons/loading.tsx +24 -24
  342. package/src/widgets/icons/menu.tsx +17 -17
  343. package/src/widgets/icons/pixel-picker.tsx +18 -18
  344. package/src/widgets/icons/search.tsx +13 -13
  345. package/src/widgets/icons/sort-asc.tsx +14 -14
  346. package/src/widgets/icons/square.tsx +18 -18
  347. package/src/widgets/index.ts +1 -0
  348. package/src/widgets/nav/MenuItem.tsx +3 -2
  349. package/src/widgets/nav/Route.ts +142 -142
  350. package/src/widgets/nav/Scroller.tsx +8 -9
  351. package/src/widgets/nav/Tab.ts +2 -2
  352. package/src/widgets/overlay/ContextMenu.ts +42 -42
  353. package/src/widgets/overlay/Dropdown.tsx +762 -762
  354. package/src/widgets/overlay/MsgBox.tsx +141 -141
  355. package/src/widgets/overlay/Overlay.tsx +5 -4
  356. package/src/widgets/overlay/Toast.ts +111 -111
  357. package/src/widgets/overlay/Window.tsx +299 -299
  358. package/src/widgets/overlay/alerts.ts +46 -46
  359. package/src/widgets/overlay/captureMouse.ts +195 -195
  360. package/src/widgets/overlay/createHotPromiseWindowFactory.ts +72 -72
  361. package/src/widgets/overlay/index.d.ts +11 -11
  362. package/src/widgets/overlay/index.ts +11 -11
  363. package/src/widgets/overlay/tooltip-ops.ts +173 -173
@@ -1,62 +1,87 @@
1
- import { computable } from "./computable";
2
- import assert from "assert";
3
- import { createAccessorModelProxy } from "./createAccessorModelProxy";
4
-
5
- describe("computable", function () {
6
- it("creates a selector", function () {
7
- let state = { person: { name: "Joe" } };
8
- let nameLength = computable("person.name", (name: string) => name.length);
9
- assert.equal(nameLength(state), 3);
10
- });
11
-
12
- it("fires every time if not memoized", function () {
13
- let state = { person: { name: "Joe" } };
14
- let fired = 0;
15
- let nameLength = computable("person.name", (name: string) => {
16
- fired++;
17
- return name.length;
18
- });
19
- assert.equal(nameLength(state), 3);
20
- assert.equal(nameLength(state), 3);
21
- assert.equal(fired, 2);
22
- });
23
-
24
- it("fires once if memoized and data has not changed", function () {
25
- let state = { person: { name: "Joe" } };
26
- let fired = 0;
27
- let nameLength = computable("person.name", (name: string) => {
28
- fired++;
29
- return name.length;
30
- }).memoize();
31
- assert.equal(nameLength(state), 3);
32
- assert.equal(nameLength(state), 3);
33
- assert.equal(fired, 1);
34
- });
35
-
36
- //weird but that's how triggers work and
37
- it("memoize with warmup data will not call compute", function () {
38
- let state = { person: { name: "Joe" } };
39
- let fired = 0;
40
- let nameLength = computable("person.name", (name: string) => {
41
- fired++;
42
- return name.length;
43
- }).memoize(state);
44
- assert.equal(nameLength(state), undefined);
45
- assert.equal(nameLength(state), undefined);
46
- assert.equal(fired, 0);
47
- });
48
-
49
- it("works with accessors", function () {
50
- var m = createAccessorModelProxy<{ person: { name: string } }>();
51
- let state = { person: { name: "Joe" } };
52
- let nameLength = computable(m.person.name, (name) => name.length);
53
- assert.equal(nameLength(state), 3);
54
- });
55
-
56
- it("works with array length accessor", function () {
57
- var m = createAccessorModelProxy<{ items: string[] }>();
58
- let state = { items: ["a", "b", "c"] };
59
- let itemCount = computable(m.items.length, (length) => length);
60
- assert.equal(itemCount(state), 3);
61
- });
62
- });
1
+ import { computable } from "./computable";
2
+ import assert from "assert";
3
+ import { createAccessorModelProxy } from "./createAccessorModelProxy";
4
+
5
+ describe("computable", function () {
6
+ it("creates a selector", function () {
7
+ let state = { person: { name: "Joe" } };
8
+ let nameLength = computable("person.name", (name: string) => name.length);
9
+ assert.equal(nameLength(state), 3);
10
+ });
11
+
12
+ it("fires every time if not memoized", function () {
13
+ let state = { person: { name: "Joe" } };
14
+ let fired = 0;
15
+ let nameLength = computable("person.name", (name: string) => {
16
+ fired++;
17
+ return name.length;
18
+ });
19
+ assert.equal(nameLength(state), 3);
20
+ assert.equal(nameLength(state), 3);
21
+ assert.equal(fired, 2);
22
+ });
23
+
24
+ it("fires once if memoized and data has not changed", function () {
25
+ let state = { person: { name: "Joe" } };
26
+ let fired = 0;
27
+ let nameLength = computable("person.name", (name: string) => {
28
+ fired++;
29
+ return name.length;
30
+ }).memoize();
31
+ assert.equal(nameLength(state), 3);
32
+ assert.equal(nameLength(state), 3);
33
+ assert.equal(fired, 1);
34
+ });
35
+
36
+ //weird but that's how triggers work and
37
+ it("memoize with warmup data will not call compute", function () {
38
+ let state = { person: { name: "Joe" } };
39
+ let fired = 0;
40
+ let nameLength = computable("person.name", (name: string) => {
41
+ fired++;
42
+ return name.length;
43
+ }).memoize(state);
44
+ assert.equal(nameLength(state), undefined);
45
+ assert.equal(nameLength(state), undefined);
46
+ assert.equal(fired, 0);
47
+ });
48
+
49
+ it("works with accessors", function () {
50
+ var m = createAccessorModelProxy<{ person: { name: string } }>();
51
+ let state = { person: { name: "Joe" } };
52
+ let nameLength = computable(m.person.name, (name) => {
53
+ // name should be inferred as string
54
+ const typedName: string = name;
55
+ // @ts-expect-error - name should not be number
56
+ const wrongType: number = name;
57
+ return typedName.length;
58
+ });
59
+ assert.equal(nameLength(state), 3);
60
+ });
61
+
62
+ it("works with array length accessor", function () {
63
+ var m = createAccessorModelProxy<{ items: string[] }>();
64
+ let state = { items: ["a", "b", "c"] };
65
+ let itemCount = computable(m.items.length, (length) => {
66
+ // length should be inferred as number
67
+ const typedLength: number = length;
68
+ // @ts-expect-error - length should not be string
69
+ const wrongType: string = length;
70
+ return typedLength;
71
+ });
72
+ assert.equal(itemCount(state), 3);
73
+ });
74
+
75
+ it("resolves AccessorChain<any> and nested properties to any", function () {
76
+ var m = createAccessorModelProxy<{ data: any }>();
77
+ let state = { data: { nested: { value: 42 } } };
78
+ let result = computable(m.data.nested.value, (value) => {
79
+ // value should be any, so all assignments should work
80
+ const asString: string = value;
81
+ const asNumber: number = value;
82
+ const asBoolean: boolean = value;
83
+ return value;
84
+ });
85
+ assert.equal(result(state), 42);
86
+ });
87
+ });
@@ -11,12 +11,9 @@ export type ComputableSelector<T = any> = string | Selector<T> | AccessorChain<T
11
11
  export type InferSelectorValue<T> =
12
12
  T extends Selector<infer R> ? R : T extends AccessorChain<infer R> ? R : T extends string ? any : never;
13
13
 
14
- // Generic overload - handles all cases with proper type inference
15
- export function computable<Selectors extends readonly any[], R>(
16
- ...args: [
17
- ...selectors: ComputableSelector[],
18
- compute: (...values: { [K in keyof Selectors]: InferSelectorValue<Selectors[K]> }) => R,
19
- ]
14
+ // Generic function with proper type inference for selectors
15
+ export function computable<T extends ComputableSelector[], R>(
16
+ ...args: [...T, (...values: { [K in keyof T]: InferSelectorValue<T[K]> }) => R]
20
17
  ): MemoSelector<R>;
21
18
 
22
19
  export function computable(...selectorsAndCompute: any[]): MemoSelector {
@@ -1,4 +1,6 @@
1
- import { createAccessorModelProxy } from "./createAccessorModelProxy";
1
+ import { createAccessorModelProxy, AccessorChain } from "./createAccessorModelProxy";
2
+ import { Prop, ResolvePropType } from "../ui/Prop";
3
+ import { Widget, WidgetConfig } from "../ui/Widget";
2
4
  import assert from "assert";
3
5
 
4
6
  interface Model {
@@ -8,6 +10,13 @@ interface Model {
8
10
  streetNumber: number;
9
11
  };
10
12
  "@crazy": string;
13
+ users: User[];
14
+ }
15
+
16
+ interface User {
17
+ id: number;
18
+ name: string;
19
+ email: string;
11
20
  }
12
21
 
13
22
  describe("createAccessorModelProxy", () => {
@@ -41,4 +50,96 @@ describe("createAccessorModelProxy", () => {
41
50
  let model = createAccessorModelProxy<Model>();
42
51
  assert.strictEqual(model["@crazy"].nameOf(), "@crazy");
43
52
  });
53
+
54
+ it("AccessorChain<any> allows access to any property", () => {
55
+ // When using an untyped model (any), all property access should be allowed
56
+ let model = createAccessorModelProxy<any>();
57
+
58
+ // These should all be valid - no TypeScript errors
59
+ assert.strictEqual(model.foo.toString(), "foo");
60
+ assert.strictEqual(model.bar.baz.toString(), "bar.baz");
61
+ assert.strictEqual(model.deeply.nested.property.toString(), "deeply.nested.property");
62
+ assert.strictEqual(model.anyName.anyChild.anyGrandchild.toString(), "anyName.anyChild.anyGrandchild");
63
+ });
64
+
65
+ it("ResolvePropType extracts correct types from AccessorChain", () => {
66
+ let model = createAccessorModelProxy<Model>();
67
+
68
+ // ResolvePropType should extract the inner type from AccessorChain
69
+ type ResolvedString = ResolvePropType<typeof model.firstName>;
70
+ type ResolvedNumber = ResolvePropType<typeof model.address.streetNumber>;
71
+ type ResolvedUsers = ResolvePropType<typeof model.users>;
72
+
73
+ // These assignments verify the types are correctly inferred
74
+ // If types are wrong, TypeScript will error
75
+ const str: ResolvedString = "test";
76
+ const num: ResolvedNumber = 42;
77
+ const users: ResolvedUsers = [{ id: 1, name: "John", email: "john@example.com" }];
78
+
79
+ // Verify array element type is preserved
80
+ type UserArray = ResolvePropType<AccessorChain<User[]>>;
81
+ const userArray: UserArray = [{ id: 1, name: "Test", email: "test@test.com" }];
82
+
83
+ // Runtime assertion to make the test meaningful
84
+ assert.ok(true);
85
+ });
86
+
87
+ it("generic widget infers type from AccessorChain in JSX", () => {
88
+ let model = createAccessorModelProxy<Model>();
89
+
90
+ // Simulates a generic widget config like LookupFieldConfig<TOption>
91
+ interface GenericWidgetConfig<T = unknown> extends WidgetConfig {
92
+ items?: AccessorChain<T[]>;
93
+ onProcess?: (item: T) => void;
94
+ }
95
+
96
+ // Simulates a CxJS widget class
97
+ class GenericWidget<T = unknown> extends Widget<GenericWidgetConfig<T>> {}
98
+
99
+ // T should be inferred as User from model.users (AccessorChain<User[]>)
100
+ let widget = (
101
+ <cx>
102
+ <GenericWidget
103
+ items={model.users}
104
+ onProcess={(user) => {
105
+ // user should be typed as User
106
+ const id: number = user.id;
107
+ const name: string = user.name;
108
+ const email: string = user.email;
109
+ }}
110
+ />
111
+ </cx>
112
+ );
113
+
114
+ assert.ok(true);
115
+ });
116
+
117
+ it("generic widget infers type from Prop<T[]> in JSX", () => {
118
+ let model = createAccessorModelProxy<Model>();
119
+
120
+ // Uses Prop<T[]> like LookupFieldConfig does
121
+ interface GenericWidgetConfig<T = unknown> extends WidgetConfig {
122
+ items?: Prop<T[]>;
123
+ onProcess?: (item: T) => void;
124
+ }
125
+
126
+ class GenericWidget<T = unknown> extends Widget<GenericWidgetConfig<T>> {}
127
+
128
+ // T should be inferred as User from model.users (AccessorChain<User[]>)
129
+ let widget = (
130
+ <cx>
131
+ <GenericWidget
132
+ items={model.users}
133
+ onProcess={(user) => {
134
+ // user should be typed as User
135
+ const id: number = user.id;
136
+ const name: string = user.name;
137
+ const email: string = user.email;
138
+ }}
139
+ />
140
+ </cx>
141
+ );
142
+
143
+ assert.ok(true);
144
+ });
44
145
  });
@@ -8,12 +8,18 @@ type AccessorChainMap<M extends object> = {
8
8
  [prop in Exclude<keyof M, keyof AccessorChainMethods>]: AccessorChain<M[prop]>;
9
9
  };
10
10
 
11
+ // Check if a type is `any` using the intersection trick
12
+ type IsAny<T> = 0 extends 1 & T ? true : false;
13
+
11
14
  export type AccessorChain<M> = {
12
15
  toString(): string;
13
- valueOf(): string;
16
+ valueOf(): M | undefined;
14
17
  nameOf(): string;
15
- __accessorChainType?: M; // Type-only marker for inference
16
- } & (M extends object ? AccessorChainMap<M> : {});
18
+ } & (IsAny<M> extends true
19
+ ? { [key: string]: any } // Allow any property access for `any` type
20
+ : M extends object
21
+ ? AccessorChainMap<M>
22
+ : {});
17
23
 
18
24
  const emptyFn = () => {};
19
25
 
@@ -1,62 +1,62 @@
1
- import { MemoSelector, Selector } from "./Selector";
2
-
3
- export interface StructuredSelectorConfig {
4
- [prop: string]: Selector;
5
- }
6
-
7
- // Helper type to infer result type from selector config
8
- type InferStructuredSelectorResult<T extends StructuredSelectorConfig, C = {}> = {
9
- [K in keyof T]: T[K] extends Selector<infer R> ? R : any;
10
- } & C;
11
-
12
- export function createStructuredSelector<S extends StructuredSelectorConfig, C extends Record<string, any> = {}>(
13
- selector: S,
14
- constants?: C,
15
- ): MemoSelector<InferStructuredSelectorResult<S, C>> {
16
- let keys = Object.keys(selector);
17
- constants = constants ?? ({} as C);
18
- if (keys.length == 0) {
19
- let result: Selector = () => constants;
20
- result.memoize = () => result;
21
- return result as MemoSelector<InferStructuredSelectorResult<S, C>>;
22
- }
23
-
24
- function memoize() {
25
- let lastResult: Record<string, any> = Object.assign({}, constants);
26
-
27
- let memoizedSelectors: Record<string, Selector> = {};
28
-
29
- keys.forEach((key) => {
30
- memoizedSelectors[key] = selector[key].memoize ? selector[key].memoize() : selector[key];
31
- lastResult[key] = undefined;
32
- });
33
-
34
- return function (data: any) {
35
- let result = lastResult,
36
- k,
37
- v,
38
- i;
39
- for (i = 0; i < keys.length; i++) {
40
- k = keys[i];
41
- v = memoizedSelectors[k](data);
42
- if (result === lastResult) {
43
- if (v === lastResult[k]) continue;
44
- result = Object.assign({}, lastResult);
45
- }
46
- result[k] = v;
47
- }
48
- return (lastResult = result);
49
- };
50
- }
51
-
52
- let result: Selector = function evaluate(data: any) {
53
- let result: Record<string, any> = Object.assign({}, constants);
54
- for (let i = 0; i < keys.length; i++) {
55
- result[keys[i]] = selector[keys[i]](data);
56
- }
57
- return result;
58
- };
59
-
60
- result.memoize = memoize;
61
- return result as MemoSelector<InferStructuredSelectorResult<S, C>>;
62
- }
1
+ import { MemoSelector, Selector } from "./Selector";
2
+
3
+ export interface StructuredSelectorConfig {
4
+ [prop: string]: Selector;
5
+ }
6
+
7
+ // Helper type to infer result type from selector config
8
+ type InferStructuredSelectorResult<T extends StructuredSelectorConfig, C = {}> = {
9
+ [K in keyof T]: T[K] extends Selector<infer R> ? R : any;
10
+ } & C;
11
+
12
+ export function createStructuredSelector<S extends StructuredSelectorConfig, C extends Record<string, any> = {}>(
13
+ selector: S,
14
+ constants?: C,
15
+ ): MemoSelector<InferStructuredSelectorResult<S, C>> {
16
+ let keys = Object.keys(selector);
17
+ constants = constants ?? ({} as C);
18
+ if (keys.length == 0) {
19
+ let result: Selector = () => constants;
20
+ result.memoize = () => result;
21
+ return result as MemoSelector<InferStructuredSelectorResult<S, C>>;
22
+ }
23
+
24
+ function memoize() {
25
+ let lastResult: Record<string, any> = Object.assign({}, constants);
26
+
27
+ let memoizedSelectors: Record<string, Selector> = {};
28
+
29
+ keys.forEach((key) => {
30
+ memoizedSelectors[key] = selector[key].memoize ? selector[key].memoize() : selector[key];
31
+ lastResult[key] = undefined;
32
+ });
33
+
34
+ return function (data: any) {
35
+ let result = lastResult,
36
+ k,
37
+ v,
38
+ i;
39
+ for (i = 0; i < keys.length; i++) {
40
+ k = keys[i];
41
+ v = memoizedSelectors[k](data);
42
+ if (result === lastResult) {
43
+ if (v === lastResult[k]) continue;
44
+ result = Object.assign({}, lastResult);
45
+ }
46
+ result[k] = v;
47
+ }
48
+ return (lastResult = result);
49
+ };
50
+ }
51
+
52
+ let result: Selector = function evaluate(data: any) {
53
+ let result: Record<string, any> = Object.assign({}, constants);
54
+ for (let i = 0; i < keys.length; i++) {
55
+ result[keys[i]] = selector[keys[i]](data);
56
+ }
57
+ return result;
58
+ };
59
+
60
+ result.memoize = memoize;
61
+ return result as MemoSelector<InferStructuredSelectorResult<S, C>>;
62
+ }
@@ -1,11 +1,11 @@
1
- import assert from "assert";
2
- import { createAccessorModelProxy } from "./createAccessorModelProxy";
3
- import { getAccessor } from "./getAccessor";
4
-
5
- describe("getAccessor", function () {
6
- it("works with accessor chains", function () {
7
- let m = createAccessorModelProxy<{ a: { b: any } }>();
8
- let accessor = getAccessor(m.a.b);
9
- assert(typeof accessor.set == "function");
10
- });
11
- });
1
+ import assert from "assert";
2
+ import { createAccessorModelProxy } from "./createAccessorModelProxy";
3
+ import { getAccessor } from "./getAccessor";
4
+
5
+ describe("getAccessor", function () {
6
+ it("works with accessor chains", function () {
7
+ let m = createAccessorModelProxy<{ a: { b: any } }>();
8
+ let accessor = getAccessor(m.a.b);
9
+ assert(typeof accessor.set == "function");
10
+ });
11
+ });
@@ -1,74 +1,74 @@
1
- import { Binding, isBinding } from "./Binding";
2
- import { isSelector } from "./isSelector";
3
- import { getSelector } from "./getSelector";
4
- import { isObject } from "../util/isObject";
5
- import { AccessorChain, isAccessorChain } from "./createAccessorModelProxy";
6
- import { Prop } from "../ui/Prop";
7
- import { View } from "./View";
8
-
9
- /*
10
- Accessor provides a common ground between refs and bindings.
11
- Refs offer simplicity, bindings have better performance with more arguments.
12
- Accessor works as a common interface which works with both patterns.
13
- */
14
-
15
- export interface Accessor<T = any> {
16
- get: (data: any) => T;
17
- set?: (value: T, store: View) => boolean;
18
- bindInstance?(instance: any): Accessor;
19
- isAccessor?: boolean;
20
- isRef?: boolean;
21
- }
22
-
23
- export function getAccessor<T = any>(accessor: Prop<T>): Accessor;
24
- export function getAccessor(accessor: Accessor): Accessor;
25
-
26
- export function getAccessor(accessor: any): Accessor | undefined {
27
- if (accessor == null) return undefined;
28
-
29
- if (isObject(accessor)) {
30
- if ("isAccessor" in accessor || "isRef" in accessor) return accessor as Accessor;
31
- if (isBinding(accessor)) {
32
- let binding = Binding.get(accessor);
33
- return {
34
- get: binding.value,
35
- set: (v, store) => store.set(binding.path, v),
36
- isAccessor: true,
37
- };
38
- }
39
- }
40
-
41
- if (isAccessorChain(accessor)) {
42
- let binding = Binding.get(accessor);
43
- return {
44
- get: binding.value,
45
- set: (v, store) => store.set(binding.path, v),
46
- isAccessor: true,
47
- };
48
- }
49
-
50
- if (isSelector(accessor)) {
51
- let selector = getSelector(accessor);
52
- if (accessor && accessor.set)
53
- return {
54
- get: selector,
55
- isAccessor: true,
56
- bindInstance(instance) {
57
- return {
58
- get: selector,
59
- set: (value) => accessor.set(value, instance),
60
- isAccessor: true,
61
- };
62
- },
63
- };
64
-
65
- return {
66
- get: selector,
67
- isAccessor: true,
68
- };
69
- }
70
-
71
- return {
72
- get: () => accessor,
73
- };
74
- }
1
+ import { Binding, isBinding } from "./Binding";
2
+ import { isSelector } from "./isSelector";
3
+ import { getSelector } from "./getSelector";
4
+ import { isObject } from "../util/isObject";
5
+ import { AccessorChain, isAccessorChain } from "./createAccessorModelProxy";
6
+ import { Prop } from "../ui/Prop";
7
+ import { View } from "./View";
8
+
9
+ /*
10
+ Accessor provides a common ground between refs and bindings.
11
+ Refs offer simplicity, bindings have better performance with more arguments.
12
+ Accessor works as a common interface which works with both patterns.
13
+ */
14
+
15
+ export interface Accessor<T = any> {
16
+ get: (data: any) => T;
17
+ set?: (value: T, store: View) => boolean;
18
+ bindInstance?(instance: any): Accessor;
19
+ isAccessor?: boolean;
20
+ isRef?: boolean;
21
+ }
22
+
23
+ export function getAccessor<T = any>(accessor: Prop<T>): Accessor;
24
+ export function getAccessor(accessor: Accessor): Accessor;
25
+
26
+ export function getAccessor(accessor: any): Accessor | undefined {
27
+ if (accessor == null) return undefined;
28
+
29
+ if (isObject(accessor)) {
30
+ if ("isAccessor" in accessor || "isRef" in accessor) return accessor as Accessor;
31
+ if (isBinding(accessor)) {
32
+ let binding = Binding.get(accessor);
33
+ return {
34
+ get: binding.value,
35
+ set: (v, store) => store.set(binding.path, v),
36
+ isAccessor: true,
37
+ };
38
+ }
39
+ }
40
+
41
+ if (isAccessorChain(accessor)) {
42
+ let binding = Binding.get(accessor);
43
+ return {
44
+ get: binding.value,
45
+ set: (v, store) => store.set(binding.path, v),
46
+ isAccessor: true,
47
+ };
48
+ }
49
+
50
+ if (isSelector(accessor)) {
51
+ let selector = getSelector(accessor);
52
+ if (accessor && accessor.set)
53
+ return {
54
+ get: selector,
55
+ isAccessor: true,
56
+ bindInstance(instance) {
57
+ return {
58
+ get: selector,
59
+ set: (value) => accessor.set(value, instance),
60
+ isAccessor: true,
61
+ };
62
+ },
63
+ };
64
+
65
+ return {
66
+ get: selector,
67
+ isAccessor: true,
68
+ };
69
+ }
70
+
71
+ return {
72
+ get: () => accessor,
73
+ };
74
+ }