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.
- package/build/charts/ScatterGraph.d.ts +4 -4
- package/build/charts/ScatterGraph.js +2 -2
- package/build/data/ArrayElementView.spec.d.ts +1 -0
- package/build/data/ArrayElementView.spec.js +81 -0
- package/build/data/Binding.spec.d.ts +1 -0
- package/build/data/Binding.spec.js +61 -0
- package/build/data/Expression.spec.d.ts +1 -0
- package/build/data/Expression.spec.js +196 -0
- package/build/data/Grouper.spec.d.ts +1 -0
- package/build/data/Grouper.spec.js +48 -0
- package/build/data/Ref.spec.d.ts +1 -0
- package/build/data/Ref.spec.js +72 -0
- package/build/data/Selector.d.ts +1 -1
- package/build/data/Store.spec.d.ts +1 -0
- package/build/data/Store.spec.js +19 -0
- package/build/data/StoreRef.spec.d.ts +1 -0
- package/build/data/StoreRef.spec.js +22 -0
- package/build/data/StringTemplate.spec.d.ts +1 -0
- package/build/data/StringTemplate.spec.js +112 -0
- package/build/data/StructuredSelector.spec.d.ts +1 -0
- package/build/data/StructuredSelector.spec.js +102 -0
- package/build/data/View.spec.d.ts +1 -0
- package/build/data/View.spec.js +44 -0
- package/build/data/ZoomIntoPropertyView.spec.d.ts +1 -0
- package/build/data/ZoomIntoPropertyView.spec.js +54 -0
- package/build/data/comparer.spec.d.ts +1 -0
- package/build/data/comparer.spec.js +50 -0
- package/build/data/computable.d.ts +3 -6
- package/build/data/computable.spec.d.ts +1 -0
- package/build/data/computable.spec.js +56 -0
- package/build/data/createAccessorModelProxy.d.ts +5 -3
- package/build/data/createAccessorModelProxy.spec.d.ts +1 -0
- package/build/data/createAccessorModelProxy.spec.js +30 -0
- package/build/data/createStructuredSelector.spec.d.ts +1 -0
- package/build/data/createStructuredSelector.spec.js +42 -0
- package/build/data/diff/diffs.spec.d.ts +1 -0
- package/build/data/diff/diffs.spec.js +45 -0
- package/build/data/getAccessor.spec.d.ts +1 -0
- package/build/data/getAccessor.spec.js +10 -0
- package/build/data/getSelector.spec.d.ts +1 -0
- package/build/data/getSelector.spec.js +36 -0
- package/build/data/ops/append.spec.d.ts +1 -0
- package/build/data/ops/append.spec.js +24 -0
- package/build/data/ops/filter.spec.d.ts +1 -0
- package/build/data/ops/filter.spec.js +25 -0
- package/build/data/ops/findTreeNode.d.ts +1 -1
- package/build/data/ops/findTreeNode.js +1 -1
- package/build/data/ops/findTreeNode.spec.d.ts +1 -0
- package/build/data/ops/findTreeNode.spec.js +20 -0
- package/build/data/ops/findTreePath.d.ts +1 -1
- package/build/data/ops/merge.spec.d.ts +1 -0
- package/build/data/ops/merge.spec.js +23 -0
- package/build/data/ops/removeTreeNodes.d.ts +1 -1
- package/build/data/ops/removeTreeNodes.js +1 -1
- package/build/data/ops/removeTreeNodes.spec.d.ts +1 -0
- package/build/data/ops/removeTreeNodes.spec.js +35 -0
- package/build/data/ops/updateArray.spec.d.ts +1 -0
- package/build/data/ops/updateArray.spec.js +33 -0
- package/build/data/ops/updateTree.d.ts +1 -1
- package/build/data/ops/updateTree.spec.d.ts +1 -0
- package/build/data/ops/updateTree.spec.js +44 -0
- package/build/hooks/invokeCallback.spec.d.ts +1 -0
- package/build/hooks/invokeCallback.spec.js +44 -0
- package/build/hooks/resolveCallback.spec.d.ts +1 -0
- package/build/hooks/resolveCallback.spec.js +35 -0
- package/build/hooks/store.spec.d.ts +1 -0
- package/build/hooks/store.spec.js +48 -0
- package/build/hooks/useTrigger.spec.d.ts +1 -0
- package/build/hooks/useTrigger.spec.js +59 -0
- package/build/jsx-runtime.d.ts +10 -10
- package/build/jsx-runtime.js +6 -0
- package/build/svg/util/Rect.d.ts +1 -1
- package/build/ui/ContentResolver.d.ts +16 -6
- package/build/ui/Controller.d.ts +7 -0
- package/build/ui/Controller.js +2 -1
- package/build/ui/Controller.spec.d.ts +1 -0
- package/build/ui/Controller.spec.js +247 -0
- package/build/ui/Cx.spec.d.ts +1 -0
- package/build/ui/Cx.spec.js +153 -0
- package/build/ui/DataProxy.spec.d.ts +1 -0
- package/build/ui/DataProxy.spec.js +208 -0
- package/build/ui/Instance.d.ts +1 -1
- package/build/ui/Instance.js +10 -10
- package/build/ui/IsolatedScope.spec.d.ts +1 -0
- package/build/ui/IsolatedScope.spec.js +42 -0
- package/build/ui/Prop.d.ts +12 -1
- package/build/ui/PureContainer.spec.d.ts +1 -0
- package/build/ui/PureContainer.spec.js +149 -0
- package/build/ui/Repeater.d.ts +3 -3
- package/build/ui/Repeater.spec.d.ts +1 -0
- package/build/ui/Repeater.spec.js +109 -0
- package/build/ui/Rescope.spec.d.ts +1 -0
- package/build/ui/Rescope.spec.js +134 -0
- package/build/ui/Restate.spec.d.ts +1 -0
- package/build/ui/Restate.spec.js +257 -0
- package/build/ui/Text.d.ts +14 -2
- package/build/ui/Text.js +3 -0
- package/build/ui/adapter/ArrayAdapter.js +4 -1
- package/build/ui/adapter/ArrayAdapter.spec.d.ts +1 -0
- package/build/ui/adapter/ArrayAdapter.spec.js +44 -0
- package/build/ui/adapter/TreeAdapter.spec.d.ts +1 -0
- package/build/ui/adapter/TreeAdapter.spec.js +71 -0
- package/build/ui/app/Url.spec.d.ts +1 -0
- package/build/ui/app/Url.spec.js +43 -0
- package/build/ui/app/startHotAppLoop.js +1 -1
- package/build/ui/createFunctionalComponent.d.ts +14 -1
- package/build/ui/createFunctionalComponent.js +7 -4
- package/build/ui/createFunctionalComponent.spec.d.ts +1 -0
- package/build/ui/createFunctionalComponent.spec.js +272 -0
- package/build/ui/expr.d.ts +3 -1
- package/build/ui/exprHelpers.d.ts +32 -0
- package/build/ui/exprHelpers.js +61 -0
- package/build/ui/index.d.ts +1 -0
- package/build/ui/index.js +1 -0
- package/build/ui/layout/ContentPlaceholder.spec.d.ts +1 -0
- package/build/ui/layout/ContentPlaceholder.spec.js +333 -0
- package/build/ui/layout/FirstVisibleChildLayout.spec.d.ts +1 -0
- package/build/ui/layout/FirstVisibleChildLayout.spec.js +101 -0
- package/build/util/Console.d.ts +1 -0
- package/build/util/Console.js +7 -3
- package/build/util/Format.spec.d.ts +1 -0
- package/build/util/Format.spec.js +58 -0
- package/build/util/TraversalStack.spec.d.ts +1 -0
- package/build/util/TraversalStack.spec.js +43 -0
- package/build/util/date/upperBoundCheck.spec.d.ts +1 -0
- package/build/util/date/upperBoundCheck.spec.js +22 -0
- package/build/util/getSearchQueryPredicate.spec.d.ts +1 -0
- package/build/util/getSearchQueryPredicate.spec.js +33 -0
- package/build/util/isValidIdentifierName.spec.d.ts +1 -0
- package/build/util/isValidIdentifierName.spec.js +28 -0
- package/build/util/routeAppend.spec.d.ts +1 -0
- package/build/util/routeAppend.spec.js +14 -0
- package/build/widgets/AccessorBindings.spec.d.ts +1 -0
- package/build/widgets/AccessorBindings.spec.js +40 -0
- package/build/widgets/Button.d.ts +3 -6
- package/build/widgets/Button.js +1 -1
- package/build/widgets/DocumentTitle.d.ts +2 -0
- package/build/widgets/Heading.d.ts +2 -2
- package/build/widgets/HtmlElement.d.ts +31 -8
- package/build/widgets/HtmlElement.js +7 -9
- package/build/widgets/HtmlElement.spec.d.ts +1 -0
- package/build/widgets/HtmlElement.spec.js +38 -0
- package/build/widgets/Icon.d.ts +3 -13
- package/build/widgets/List.d.ts +4 -0
- package/build/widgets/ReactElementWrapper.d.ts +29 -0
- package/build/widgets/ReactElementWrapper.js +59 -0
- package/build/widgets/drag-drop/DragSource.d.ts +3 -3
- package/build/widgets/drag-drop/DragSource.js +2 -3
- package/build/widgets/drag-drop/DropZone.d.ts +3 -3
- package/build/widgets/drag-drop/DropZone.js +2 -3
- package/build/widgets/form/Checkbox.d.ts +2 -0
- package/build/widgets/form/ColorField.d.ts +2 -0
- package/build/widgets/form/DateTimeField.d.ts +2 -0
- package/build/widgets/form/Field.d.ts +0 -2
- package/build/widgets/form/LabeledContainer.d.ts +9 -8
- package/build/widgets/form/LabeledContainer.js +9 -9
- package/build/widgets/form/LookupField.d.ts +57 -9
- package/build/widgets/form/MonthField.d.ts +2 -0
- package/build/widgets/form/NumberField.d.ts +2 -0
- package/build/widgets/form/Radio.d.ts +2 -0
- package/build/widgets/form/Select.d.ts +2 -0
- package/build/widgets/form/Slider.d.ts +3 -0
- package/build/widgets/form/Switch.d.ts +2 -0
- package/build/widgets/form/TextField.d.ts +34 -0
- package/build/widgets/form/TimeList.d.ts +16 -1
- package/build/widgets/form/TimeList.js +34 -62
- package/build/widgets/form/UploadButton.d.ts +34 -2
- package/build/widgets/form/UploadButton.js +3 -1
- package/build/widgets/form/ValidationGroup.spec.d.ts +1 -0
- package/build/widgets/form/ValidationGroup.spec.js +62 -0
- package/build/widgets/form/Validator.d.ts +33 -2
- package/build/widgets/form/Validator.js +3 -0
- package/build/widgets/grid/Grid.d.ts +9 -9
- package/build/widgets/grid/TreeNode.d.ts +6 -0
- package/build/widgets/index.d.ts +1 -0
- package/build/widgets/index.js +1 -0
- package/build/widgets/nav/MenuItem.d.ts +3 -2
- package/build/widgets/nav/Route.spec.d.ts +1 -0
- package/build/widgets/nav/Route.spec.js +15 -0
- package/build/widgets/nav/Scroller.d.ts +4 -6
- package/build/widgets/nav/Scroller.js +6 -3
- package/build/widgets/nav/Tab.d.ts +2 -2
- package/build/widgets/overlay/ContextMenu.d.ts +3 -3
- package/build/widgets/overlay/Overlay.d.ts +2 -1
- package/build/widgets/overlay/Overlay.js +1 -1
- package/build.js +133 -133
- package/dist/data.js +2 -2
- package/dist/jsx-runtime.js +6 -1
- package/dist/manifest.d.ts +1443 -0
- package/dist/manifest.js +855 -807
- package/dist/ui.js +91 -5
- package/dist/util.js +3 -0
- package/dist/widgets.js +520 -161
- package/package.json +46 -20
- package/src/charts/Chart.ts +108 -108
- package/src/charts/ScatterGraph.tsx +6 -6
- package/src/data/ArrayElementView.ts +90 -90
- package/src/data/AugmentedViewBase.ts +88 -88
- package/src/data/Binding.ts +104 -104
- package/src/data/ExposedRecordView.ts +95 -95
- package/src/data/ExposedValueView.ts +89 -89
- package/src/data/Expression.spec.ts +229 -229
- package/src/data/Expression.ts +233 -233
- package/src/data/Grouper.spec.ts +57 -57
- package/src/data/Grouper.ts +158 -158
- package/src/data/NestedDataView.ts +43 -43
- package/src/data/ReadOnlyDataView.ts +39 -39
- package/src/data/Ref.ts +104 -104
- package/src/data/Selector.ts +10 -10
- package/src/data/Store.ts +52 -52
- package/src/data/StoreProxy.ts +19 -19
- package/src/data/StoreRef.ts +66 -66
- package/src/data/StringTemplate.spec.ts +132 -132
- package/src/data/StringTemplate.ts +93 -93
- package/src/data/StructuredSelector.spec.ts +113 -113
- package/src/data/StructuredSelector.ts +146 -146
- package/src/data/SubscribableView.ts +63 -63
- package/src/data/ZoomIntoPropertyView.spec.ts +64 -64
- package/src/data/ZoomIntoPropertyView.ts +45 -45
- package/src/data/computable.spec.ts +87 -62
- package/src/data/computable.ts +3 -6
- package/src/data/createAccessorModelProxy.spec.tsx +102 -1
- package/src/data/createAccessorModelProxy.ts +9 -3
- package/src/data/createStructuredSelector.ts +62 -62
- package/src/data/getAccessor.spec.ts +11 -11
- package/src/data/getAccessor.ts +74 -74
- package/src/data/getSelector.spec.ts +43 -43
- package/src/data/getSelector.ts +66 -66
- package/src/data/ops/filter.spec.ts +35 -35
- package/src/data/ops/filter.ts +9 -9
- package/src/data/ops/findTreeNode.ts +1 -5
- package/src/data/ops/findTreePath.ts +1 -1
- package/src/data/ops/merge.ts +13 -13
- package/src/data/ops/removeTreeNodes.spec.ts +37 -37
- package/src/data/ops/removeTreeNodes.ts +2 -2
- package/src/data/ops/updateArray.spec.ts +69 -69
- package/src/data/ops/updateArray.ts +31 -31
- package/src/data/ops/updateTree.ts +1 -1
- package/src/data/test-types.ts +7 -7
- package/src/hooks/resolveCallback.spec.tsx +30 -7
- package/src/hooks/useTrigger.ts +26 -26
- package/src/index.scss +6 -6
- package/src/jsx-dev-runtime.ts +4 -4
- package/src/jsx-runtime.spec.tsx +402 -0
- package/src/jsx-runtime.ts +26 -22
- package/src/svg/BoundedObject.ts +101 -101
- package/src/svg/util/Rect.ts +105 -105
- package/src/ui/CSSHelper.ts +17 -17
- package/src/ui/ContentResolver.spec.tsx +172 -19
- package/src/ui/ContentResolver.ts +16 -8
- package/src/ui/Controller.ts +15 -2
- package/src/ui/Culture.ts +159 -159
- package/src/ui/DataProxy.ts +55 -55
- package/src/ui/FocusManager.ts +171 -171
- package/src/ui/Instance.ts +866 -868
- package/src/ui/Prop.ts +140 -112
- package/src/ui/RenderingContext.ts +99 -99
- package/src/ui/Repeater.ts +3 -12
- package/src/ui/Rescope.ts +49 -49
- package/src/ui/StructuredInstanceDataAccessor.ts +32 -32
- package/src/ui/Text.ts +21 -2
- package/src/ui/VDOM.ts +34 -34
- package/src/ui/adapter/ArrayAdapter.spec.ts +55 -55
- package/src/ui/adapter/ArrayAdapter.ts +4 -1
- package/src/ui/adapter/TreeAdapter.spec.ts +76 -76
- package/src/ui/adapter/TreeAdapter.ts +185 -185
- package/src/ui/app/History.ts +133 -133
- package/src/ui/app/Url.spec.ts +50 -50
- package/src/ui/app/startHotAppLoop.ts +41 -41
- package/src/ui/createFunctionalComponent.spec.tsx +53 -0
- package/src/ui/createFunctionalComponent.ts +86 -65
- package/src/ui/expr.ts +4 -1
- package/src/ui/exprHelpers.spec.ts +379 -0
- package/src/ui/exprHelpers.ts +78 -0
- package/src/ui/index.ts +47 -46
- package/src/ui/layout/Content.ts +30 -30
- package/src/ui/layout/FirstVisibleChildLayout.spec.tsx +1 -1
- package/src/ui/layout/FirstVisibleChildLayout.ts +60 -60
- package/src/ui/selection/KeySelection.ts +153 -153
- package/src/ui/selection/Selection.ts +128 -128
- package/src/util/Console.ts +13 -11
- package/src/util/Format.ts +267 -267
- package/src/util/addEventListenerWithOptions.ts +41 -41
- package/src/util/browserSupportsPassiveEventHandlers.ts +20 -20
- package/src/util/color/rgbToHsl.ts +35 -35
- package/src/util/getActiveElement.ts +4 -4
- package/src/util/hasKey.ts +18 -18
- package/src/util/index.ts +55 -55
- package/src/util/innerTextTrim.ts +10 -10
- package/src/util/isArray.ts +3 -3
- package/src/util/isDataRecord.ts +5 -5
- package/src/util/isDefined.ts +3 -3
- package/src/util/isString.ts +3 -3
- package/src/widgets/AccessorBindings.spec.tsx +26 -0
- package/src/widgets/Button.tsx +5 -17
- package/src/widgets/DocumentTitle.ts +95 -92
- package/src/widgets/Heading.ts +2 -2
- package/src/widgets/HtmlElement.spec.helpers.tsx +108 -0
- package/src/widgets/HtmlElement.spec.tsx +20 -12
- package/src/widgets/HtmlElement.tsx +77 -23
- package/src/widgets/Icon.ts +3 -17
- package/src/widgets/List.tsx +6 -0
- package/src/widgets/ReactElementWrapper.spec.tsx +452 -0
- package/src/widgets/ReactElementWrapper.tsx +108 -0
- package/src/widgets/Sandbox.ts +103 -103
- package/src/widgets/autoFocus.ts +9 -9
- package/src/widgets/cx.ts +63 -63
- package/src/widgets/drag-drop/DragSource.tsx +3 -4
- package/src/widgets/drag-drop/DropZone.tsx +3 -4
- package/src/widgets/form/Checkbox.tsx +3 -0
- package/src/widgets/form/ColorField.tsx +3 -0
- package/src/widgets/form/DateTimeField.tsx +5 -0
- package/src/widgets/form/Field.tsx +0 -3
- package/src/widgets/form/Label.tsx +1 -0
- package/src/widgets/form/LabeledContainer.ts +22 -26
- package/src/widgets/form/LookupField.spec.tsx +93 -0
- package/src/widgets/form/LookupField.tsx +104 -9
- package/src/widgets/form/MonthField.tsx +5 -0
- package/src/widgets/form/NumberField.tsx +3 -0
- package/src/widgets/form/Radio.tsx +5 -0
- package/src/widgets/form/Select.tsx +5 -0
- package/src/widgets/form/Slider.tsx +4 -0
- package/src/widgets/form/Switch.tsx +3 -0
- package/src/widgets/form/TextField.tsx +62 -0
- package/src/widgets/form/TimeList.tsx +84 -73
- package/src/widgets/form/UploadButton.tsx +53 -2
- package/src/widgets/form/Validator.ts +40 -3
- package/src/widgets/grid/Grid.tsx +9 -12
- package/src/widgets/grid/GridCell.ts +143 -143
- package/src/widgets/grid/TreeNode.tsx +9 -0
- package/src/widgets/icons/calendar.tsx +17 -17
- package/src/widgets/icons/check.tsx +13 -13
- package/src/widgets/icons/clear.tsx +15 -15
- package/src/widgets/icons/close.tsx +20 -20
- package/src/widgets/icons/cx.tsx +38 -38
- package/src/widgets/icons/drop-down.tsx +15 -15
- package/src/widgets/icons/file.tsx +13 -13
- package/src/widgets/icons/folder-open.tsx +15 -15
- package/src/widgets/icons/folder.tsx +13 -13
- package/src/widgets/icons/forward.tsx +22 -22
- package/src/widgets/icons/loading.tsx +24 -24
- package/src/widgets/icons/menu.tsx +17 -17
- package/src/widgets/icons/pixel-picker.tsx +18 -18
- package/src/widgets/icons/search.tsx +13 -13
- package/src/widgets/icons/sort-asc.tsx +14 -14
- package/src/widgets/icons/square.tsx +18 -18
- package/src/widgets/index.ts +1 -0
- package/src/widgets/nav/MenuItem.tsx +3 -2
- package/src/widgets/nav/Route.ts +142 -142
- package/src/widgets/nav/Scroller.tsx +8 -9
- package/src/widgets/nav/Tab.ts +2 -2
- package/src/widgets/overlay/ContextMenu.ts +42 -42
- package/src/widgets/overlay/Dropdown.tsx +762 -762
- package/src/widgets/overlay/MsgBox.tsx +141 -141
- package/src/widgets/overlay/Overlay.tsx +5 -4
- package/src/widgets/overlay/Toast.ts +111 -111
- package/src/widgets/overlay/Window.tsx +299 -299
- package/src/widgets/overlay/alerts.ts +46 -46
- package/src/widgets/overlay/captureMouse.ts +195 -195
- package/src/widgets/overlay/createHotPromiseWindowFactory.ts +72 -72
- package/src/widgets/overlay/index.d.ts +11 -11
- package/src/widgets/overlay/index.ts +11 -11
- 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) =>
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
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
|
+
});
|
package/src/data/computable.ts
CHANGED
|
@@ -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
|
|
15
|
-
export function computable<
|
|
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():
|
|
16
|
+
valueOf(): M | undefined;
|
|
14
17
|
nameOf(): string;
|
|
15
|
-
|
|
16
|
-
}
|
|
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
|
+
});
|
package/src/data/getAccessor.ts
CHANGED
|
@@ -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
|
+
}
|