react-elmish 5.0.1 → 5.1.1-beta.1
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/README.md +80 -105
- package/dist/Cmd.js +3 -20
- package/dist/Common.js +1 -6
- package/dist/ElmComponent.js +14 -68
- package/dist/ErrorHandling.js +3 -9
- package/dist/Init.js +2 -3
- package/dist/Testing/createUpdateArgsFactory.js +1 -5
- package/dist/Testing/execCmd.js +2 -16
- package/dist/Testing/fakeOptions.js +2 -4
- package/dist/Testing/getOfMsgParams.js +1 -4
- package/dist/Testing/getUpdateFn.js +1 -3
- package/dist/Testing/index.js +1 -6
- package/dist/Testing/renderWithModel.js +1 -4
- package/dist/index.js +1 -6
- package/dist/useElmish.js +19 -55
- package/installScript.js +4 -0
- package/package.json +13 -12
package/dist/index.js
CHANGED
|
@@ -45,14 +45,9 @@ Object.defineProperty(exports, "useElmish", {
|
|
|
45
45
|
return _useElmish.useElmish;
|
|
46
46
|
}
|
|
47
47
|
});
|
|
48
|
-
|
|
49
48
|
var _Cmd = require("./Cmd");
|
|
50
|
-
|
|
51
49
|
var _ElmComponent = require("./ElmComponent");
|
|
52
|
-
|
|
53
50
|
var _ErrorHandling = require("./ErrorHandling");
|
|
54
|
-
|
|
55
51
|
var _Init = require("./Init");
|
|
56
|
-
|
|
57
52
|
var _useElmish = require("./useElmish");
|
|
58
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
53
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgY3JlYXRlQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgRWxtQ29tcG9uZW50IH0gZnJvbSBcIi4vRWxtQ29tcG9uZW50XCI7XG5pbXBvcnQgeyBlcnJvckhhbmRsZXIsIEVycm9yTWVzc2FnZSwgZXJyb3JNc2csIGhhbmRsZUVycm9yIH0gZnJvbSBcIi4vRXJyb3JIYW5kbGluZ1wiO1xuaW1wb3J0IHsgaW5pdCwgTG9nZ2VyLCBNZXNzYWdlIH0gZnJvbSBcIi4vSW5pdFwiO1xuaW1wb3J0IHsgSW5pdFJlc3VsdCwgTXNnU291cmNlLCBVcGRhdGVNYXAsIFVwZGF0ZVJldHVyblR5cGUgfSBmcm9tIFwiLi9UeXBlc1wiO1xuaW1wb3J0IHsgU3Vic2NyaXB0aW9uUmVzdWx0LCB1c2VFbG1pc2ggfSBmcm9tIFwiLi91c2VFbG1pc2hcIjtcblxuZXhwb3J0IHR5cGUge1xuICAgIExvZ2dlcixcbiAgICBNZXNzYWdlLFxuICAgIENtZCxcbiAgICBEaXNwYXRjaCxcbiAgICBJbml0UmVzdWx0LFxuICAgIFVwZGF0ZVJldHVyblR5cGUsXG4gICAgU3Vic2NyaXB0aW9uUmVzdWx0LFxuICAgIE1zZ1NvdXJjZSxcbiAgICBVcGRhdGVNYXAsXG4gICAgRXJyb3JNZXNzYWdlLFxufTtcblxuZXhwb3J0IHtcbiAgICBpbml0LFxuICAgIGNyZWF0ZUNtZCxcbiAgICBFbG1Db21wb25lbnQsXG4gICAgZXJyb3JNc2csXG4gICAgZXJyb3JIYW5kbGVyLFxuICAgIGhhbmRsZUVycm9yLFxuICAgIHVzZUVsbWlzaCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBRUEifQ==
|
package/dist/useElmish.js
CHANGED
|
@@ -6,33 +6,19 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
6
6
|
exports.callUpdate = callUpdate;
|
|
7
7
|
exports.callUpdateMap = callUpdateMap;
|
|
8
8
|
exports.useElmish = useElmish;
|
|
9
|
-
|
|
10
9
|
var _react = require("react");
|
|
11
|
-
|
|
12
10
|
var _Common = require("./Common");
|
|
13
|
-
|
|
14
11
|
var _Init = require("./Init");
|
|
15
|
-
|
|
16
12
|
var _fakeOptions = require("./Testing/fakeOptions");
|
|
17
|
-
|
|
18
13
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
19
|
-
|
|
20
14
|
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
21
|
-
|
|
22
15
|
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
23
|
-
|
|
24
16
|
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
|
|
25
|
-
|
|
26
17
|
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
|
|
27
|
-
|
|
28
18
|
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
|
|
29
|
-
|
|
30
19
|
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
|
|
31
|
-
|
|
32
20
|
function _iterableToArrayLimit(arr, i) { var _i = arr == null ? null : typeof Symbol !== "undefined" && arr[Symbol.iterator] || arr["@@iterator"]; if (_i == null) return; var _arr = []; var _n = true; var _d = false; var _s, _e; try { for (_i = _i.call(arr); !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
|
|
33
|
-
|
|
34
21
|
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
35
|
-
|
|
36
22
|
/**
|
|
37
23
|
* Hook to use the Elm architecture pattern in a function component.
|
|
38
24
|
* @param {UseElmishOptions} options The options passed the the hook.
|
|
@@ -42,21 +28,18 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
|
42
28
|
*/
|
|
43
29
|
function useElmish(_ref) {
|
|
44
30
|
var _fakeOptions$dispatch;
|
|
45
|
-
|
|
46
31
|
var name = _ref.name,
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
32
|
+
props = _ref.props,
|
|
33
|
+
init = _ref.init,
|
|
34
|
+
update = _ref.update,
|
|
35
|
+
subscription = _ref.subscription;
|
|
51
36
|
var reentered = false;
|
|
52
37
|
var buffer = [];
|
|
53
38
|
var currentModel = {};
|
|
54
|
-
|
|
55
39
|
var _useState = (0, _react.useState)(null),
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
40
|
+
_useState2 = _slicedToArray(_useState, 2),
|
|
41
|
+
model = _useState2[0],
|
|
42
|
+
setModel = _useState2[1];
|
|
60
43
|
var propsRef = (0, _react.useRef)(props);
|
|
61
44
|
var isMountedRef = (0, _react.useRef)(true);
|
|
62
45
|
(0, _react.useEffect)(function () {
|
|
@@ -66,82 +49,66 @@ function useElmish(_ref) {
|
|
|
66
49
|
};
|
|
67
50
|
}, []);
|
|
68
51
|
var initializedModel = model;
|
|
69
|
-
|
|
70
52
|
if (propsRef.current !== props) {
|
|
71
53
|
propsRef.current = props;
|
|
72
54
|
}
|
|
73
|
-
|
|
74
55
|
var fakeOptions = (0, _fakeOptions.getFakeOptionsOnce)();
|
|
75
56
|
var dispatch = (0, _react.useCallback)((_fakeOptions$dispatch = fakeOptions === null || fakeOptions === void 0 ? void 0 : fakeOptions.dispatch) !== null && _fakeOptions$dispatch !== void 0 ? _fakeOptions$dispatch : function (msg) {
|
|
76
57
|
if (!initializedModel) {
|
|
77
58
|
return;
|
|
78
59
|
}
|
|
79
|
-
|
|
80
60
|
if (reentered) {
|
|
81
61
|
buffer.push(msg);
|
|
82
62
|
} else {
|
|
83
63
|
reentered = true;
|
|
84
64
|
var nextMsg = msg;
|
|
85
65
|
var modified = false;
|
|
86
|
-
|
|
87
66
|
while (nextMsg) {
|
|
88
67
|
(0, _Common.logMessage)(name, nextMsg);
|
|
89
|
-
|
|
90
68
|
var _callUpdate = callUpdate(update, nextMsg, _objectSpread(_objectSpread({}, initializedModel), currentModel), propsRef.current),
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
69
|
+
_callUpdate2 = _slicedToArray(_callUpdate, 2),
|
|
70
|
+
newModel = _callUpdate2[0],
|
|
71
|
+
cmd = _callUpdate2[1];
|
|
95
72
|
if ((0, _Common.modelHasChanged)(currentModel, newModel)) {
|
|
96
73
|
currentModel = _objectSpread(_objectSpread({}, currentModel), newModel);
|
|
97
74
|
modified = true;
|
|
98
75
|
}
|
|
99
|
-
|
|
100
76
|
if (cmd) {
|
|
101
77
|
(0, _Common.execCmd)(cmd, dispatch);
|
|
102
78
|
}
|
|
103
|
-
|
|
104
79
|
nextMsg = buffer.shift();
|
|
105
80
|
}
|
|
106
|
-
|
|
107
81
|
reentered = false;
|
|
108
|
-
|
|
109
82
|
if (isMountedRef.current && modified) {
|
|
110
83
|
setModel(function (prevModel) {
|
|
111
84
|
var _Services$logger;
|
|
112
|
-
|
|
113
85
|
var updatedModel = _objectSpread(_objectSpread({}, prevModel), currentModel);
|
|
114
|
-
|
|
115
86
|
(_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.debug("Elm", "update model for", name, updatedModel);
|
|
116
87
|
return updatedModel;
|
|
117
88
|
});
|
|
118
89
|
}
|
|
119
90
|
}
|
|
120
91
|
}, []);
|
|
121
|
-
|
|
122
92
|
if (!initializedModel) {
|
|
93
|
+
var _Services$logger2;
|
|
123
94
|
var _ref2 = fakeOptions !== null && fakeOptions !== void 0 && fakeOptions.model ? [fakeOptions.model] : init(props),
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
95
|
+
_ref3 = _slicedToArray(_ref2, 2),
|
|
96
|
+
initModel = _ref3[0],
|
|
97
|
+
initCmd = _ref3[1];
|
|
128
98
|
initializedModel = initModel;
|
|
129
99
|
setModel(initializedModel);
|
|
130
|
-
|
|
100
|
+
(_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "initial model for", name, initializedModel);
|
|
131
101
|
if (initCmd) {
|
|
132
102
|
(0, _Common.execCmd)(initCmd, dispatch);
|
|
133
103
|
}
|
|
134
104
|
}
|
|
135
|
-
|
|
136
105
|
(0, _react.useEffect)(function () {
|
|
137
106
|
if (subscription) {
|
|
138
107
|
var _subscription = subscription(initializedModel, props),
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
108
|
+
_subscription2 = _slicedToArray(_subscription, 2),
|
|
109
|
+
subCmd = _subscription2[0],
|
|
110
|
+
destructor = _subscription2[1];
|
|
143
111
|
(0, _Common.execCmd)(subCmd, dispatch);
|
|
144
|
-
|
|
145
112
|
if (destructor) {
|
|
146
113
|
return destructor;
|
|
147
114
|
}
|
|
@@ -149,19 +116,16 @@ function useElmish(_ref) {
|
|
|
149
116
|
}, []);
|
|
150
117
|
return [initializedModel, dispatch];
|
|
151
118
|
}
|
|
152
|
-
|
|
153
119
|
function callUpdate(update, msg, model, props) {
|
|
154
120
|
if (typeof update === "function") {
|
|
155
121
|
return update(model, msg, props);
|
|
156
122
|
}
|
|
157
|
-
|
|
158
123
|
return callUpdateMap(update, msg, model, props);
|
|
159
124
|
}
|
|
160
|
-
|
|
161
125
|
function callUpdateMap(updateMap, msg, model, props) {
|
|
162
126
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
163
127
|
// @ts-expect-error -- We know that msg fits
|
|
164
128
|
var updateFn = updateMap[msg.name];
|
|
165
129
|
return updateFn(msg, model, props);
|
|
166
130
|
}
|
|
167
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useElmish","name","props","init","update","subscription","reentered","buffer","currentModel","useState","model","setModel","propsRef","useRef","isMountedRef","useEffect","current","initializedModel","fakeOptions","getFakeOptionsOnce","dispatch","useCallback","msg","push","nextMsg","modified","logMessage","callUpdate","newModel","cmd","modelHasChanged","execCmd","shift","prevModel","updatedModel","Services","logger","debug","initModel","initCmd","subCmd","destructor","callUpdateMap","updateMap","updateFn"],"sources":["../src/useElmish.ts"],"sourcesContent":["/* eslint-disable react-hooks/exhaustive-deps */\nimport { useCallback, useEffect, useRef, useState } from \"react\";\nimport { Cmd, Dispatch } from \"./Cmd\";\nimport { execCmd, logMessage, modelHasChanged } from \"./Common\";\nimport { Services } from \"./Init\";\nimport { getFakeOptionsOnce } from \"./Testing/fakeOptions\";\nimport { InitFunction, MessageBase, Nullable, UpdateFunction, UpdateMap, UpdateReturnType } from \"./Types\";\n\n/**\n * The return type of the `subscription` function.\n * @template TMessage The type of the messages discriminated union.\n */\ntype SubscriptionResult<TMessage> = [Cmd<TMessage>, (() => void)?];\ntype Subscription<TProps, TModel, TMessage> = (model: TModel, props: TProps) => SubscriptionResult<TMessage>;\n\n/**\n * Options for the `useElmish` hook.\n * @interface UseElmishOptions\n * @template TProps The type of the props.\n * @template TModel The type of the model.\n * @template TMessage The type of the messages discriminated union.\n */\ninterface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {\n    /**\n     * The name of the component. This is used for logging only.\n     * @type {string}\n     */\n    name: string,\n    /**\n     * The props passed to the component.\n     * @type {TProps}\n     */\n    props: TProps,\n    /**\n     * The function to initialize the components model. This function is only called once.\n     * @type {InitFunction<TProps, TModel, TMessage>}\n     */\n    init: InitFunction<TProps, TModel, TMessage>,\n    /**\n     * The `update` function or update map object.\n     * @type {(UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>)}\n     */\n    update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>,\n    /**\n     * The optional `subscription` function. This function is only called once.\n     * @type {(UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>)}\n     */\n    subscription?: Subscription<TProps, TModel, TMessage>,\n}\n\n/**\n * Hook to use the Elm architecture pattern in a function component.\n * @param {UseElmishOptions} options The options passed the the hook.\n * @returns A tuple containing the current model and the dispatcher.\n * @example\n * const [model, dispatch] = useElmish({ props, init, update, name: \"MyComponent\" });\n */\nfunction useElmish<TProps, TModel, TMessage extends MessageBase> ({ name, props, init, update, subscription }: UseElmishOptions<TProps, TModel, TMessage>): [TModel, Dispatch<TMessage>] {\n    let reentered = false;\n    const buffer: TMessage [] = [];\n    let currentModel: Partial<TModel> = {};\n\n    const [model, setModel] = useState<Nullable<TModel>>(null);\n    const propsRef = useRef(props);\n    const isMountedRef = useRef(true);\n\n    useEffect(() => {\n        isMountedRef.current = true;\n\n        return () => {\n            isMountedRef.current = false;\n        };\n    }, []);\n\n    let initializedModel = model;\n\n    if (propsRef.current !== props) {\n        propsRef.current = props;\n    }\n\n    const fakeOptions = getFakeOptionsOnce();\n    const dispatch = useCallback(fakeOptions?.dispatch ?? ((msg: TMessage): void => {\n        if (!initializedModel) {\n            return;\n        }\n\n        if (reentered) {\n            buffer.push(msg);\n        } else {\n            reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                logMessage(name, nextMsg);\n\n                const [newModel, cmd] = callUpdate(update, nextMsg, { ...initializedModel, ...currentModel }, propsRef.current);\n\n                if (modelHasChanged(currentModel, newModel)) {\n                    currentModel = { ...currentModel, ...newModel };\n\n                    modified = true;\n                }\n\n                if (cmd) {\n                    execCmd(cmd, dispatch);\n                }\n\n                nextMsg = buffer.shift();\n            }\n            reentered = false;\n\n            if (isMountedRef.current && modified) {\n                setModel(prevModel => {\n                    const updatedModel = { ...prevModel as TModel, ...currentModel };\n\n                    Services.logger?.debug(\"Elm\", \"update model for\", name, updatedModel);\n\n                    return updatedModel;\n                });\n            }\n        }\n    }), []);\n\n    if (!initializedModel) {\n        const [initModel, initCmd] = fakeOptions?.model ? [fakeOptions.model as TModel] : init(props);\n\n        initializedModel = initModel;\n        setModel(initializedModel);\n\n        if (initCmd) {\n            execCmd(initCmd, dispatch);\n        }\n    }\n\n    useEffect(() => {\n        if (subscription) {\n            const [subCmd, destructor] = subscription(initializedModel as TModel, props);\n\n            execCmd(subCmd, dispatch);\n\n            if (destructor) {\n                return destructor;\n            }\n        }\n    }, []);\n\n    return [initializedModel, dispatch];\n}\n\nfunction callUpdate<TProps, TModel, TMessage extends MessageBase> (update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>, msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage> {\n    if (typeof update === \"function\") {\n        return update(model, msg, props);\n    }\n\n    return callUpdateMap(update, msg, model, props);\n}\n\nfunction callUpdateMap<TProps, TModel, TMessage extends MessageBase> (updateMap: UpdateMap<TProps, TModel, TMessage>, msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage> {\n    // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n    // @ts-expect-error -- We know that msg fits\n    const updateFn = updateMap[msg.name as TMessage[\"name\"]] as (msg: TMessage, model: TModel, props: TProps) => UpdateReturnType<TModel, TMsg>;\n\n    return updateFn(msg, model, props);\n}\n\nexport type {\n    SubscriptionResult,\n};\n\nexport {\n    useElmish,\n    callUpdate,\n    callUpdateMap,\n};"],"mappings":"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"}
|
|
131
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useElmish","name","props","init","update","subscription","reentered","buffer","currentModel","useState","model","setModel","propsRef","useRef","isMountedRef","useEffect","current","initializedModel","fakeOptions","getFakeOptionsOnce","dispatch","useCallback","msg","push","nextMsg","modified","logMessage","callUpdate","newModel","cmd","modelHasChanged","execCmd","shift","prevModel","updatedModel","Services","logger","debug","initModel","initCmd","subCmd","destructor","callUpdateMap","updateMap","updateFn"],"sources":["../src/useElmish.ts"],"sourcesContent":["/* eslint-disable react-hooks/exhaustive-deps */\nimport { useCallback, useEffect, useRef, useState } from \"react\";\nimport { Cmd, Dispatch } from \"./Cmd\";\nimport { execCmd, logMessage, modelHasChanged } from \"./Common\";\nimport { Services } from \"./Init\";\nimport { getFakeOptionsOnce } from \"./Testing/fakeOptions\";\nimport { InitFunction, MessageBase, Nullable, UpdateFunction, UpdateMap, UpdateReturnType } from \"./Types\";\n\n/**\n * The return type of the `subscription` function.\n * @template TMessage The type of the messages discriminated union.\n */\ntype SubscriptionResult<TMessage> = [Cmd<TMessage>, (() => void)?];\ntype Subscription<TProps, TModel, TMessage> = (model: TModel, props: TProps) => SubscriptionResult<TMessage>;\n\n/**\n * Options for the `useElmish` hook.\n * @interface UseElmishOptions\n * @template TProps The type of the props.\n * @template TModel The type of the model.\n * @template TMessage The type of the messages discriminated union.\n */\ninterface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {\n    /**\n     * The name of the component. This is used for logging only.\n     * @type {string}\n     */\n    name: string,\n    /**\n     * The props passed to the component.\n     * @type {TProps}\n     */\n    props: TProps,\n    /**\n     * The function to initialize the components model. This function is only called once.\n     * @type {InitFunction<TProps, TModel, TMessage>}\n     */\n    init: InitFunction<TProps, TModel, TMessage>,\n    /**\n     * The `update` function or update map object.\n     * @type {(UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>)}\n     */\n    update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>,\n    /**\n     * The optional `subscription` function. This function is only called once.\n     * @type {(UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>)}\n     */\n    subscription?: Subscription<TProps, TModel, TMessage>,\n}\n\n/**\n * Hook to use the Elm architecture pattern in a function component.\n * @param {UseElmishOptions} options The options passed the the hook.\n * @returns A tuple containing the current model and the dispatcher.\n * @example\n * const [model, dispatch] = useElmish({ props, init, update, name: \"MyComponent\" });\n */\nfunction useElmish<TProps, TModel, TMessage extends MessageBase> ({ name, props, init, update, subscription }: UseElmishOptions<TProps, TModel, TMessage>): [TModel, Dispatch<TMessage>] {\n    let reentered = false;\n    const buffer: TMessage [] = [];\n    let currentModel: Partial<TModel> = {};\n\n    const [model, setModel] = useState<Nullable<TModel>>(null);\n    const propsRef = useRef(props);\n    const isMountedRef = useRef(true);\n\n    useEffect(() => {\n        isMountedRef.current = true;\n\n        return () => {\n            isMountedRef.current = false;\n        };\n    }, []);\n\n    let initializedModel = model;\n\n    if (propsRef.current !== props) {\n        propsRef.current = props;\n    }\n\n    const fakeOptions = getFakeOptionsOnce();\n    const dispatch = useCallback(fakeOptions?.dispatch ?? ((msg: TMessage): void => {\n        if (!initializedModel) {\n            return;\n        }\n\n        if (reentered) {\n            buffer.push(msg);\n        } else {\n            reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                logMessage(name, nextMsg);\n\n                const [newModel, cmd] = callUpdate(update, nextMsg, { ...initializedModel, ...currentModel }, propsRef.current);\n\n                if (modelHasChanged(currentModel, newModel)) {\n                    currentModel = { ...currentModel, ...newModel };\n\n                    modified = true;\n                }\n\n                if (cmd) {\n                    execCmd(cmd, dispatch);\n                }\n\n                nextMsg = buffer.shift();\n            }\n            reentered = false;\n\n            if (isMountedRef.current && modified) {\n                setModel(prevModel => {\n                    const updatedModel = { ...prevModel as TModel, ...currentModel };\n\n                    Services.logger?.debug(\"Elm\", \"update model for\", name, updatedModel);\n\n                    return updatedModel;\n                });\n            }\n        }\n    }), []);\n\n    if (!initializedModel) {\n        const [initModel, initCmd] = fakeOptions?.model ? [fakeOptions.model as TModel] : init(props);\n\n        initializedModel = initModel;\n        setModel(initializedModel);\n\n        Services.logger?.debug(\"Elm\", \"initial model for\", name, initializedModel);\n\n        if (initCmd) {\n            execCmd(initCmd, dispatch);\n        }\n    }\n\n    useEffect(() => {\n        if (subscription) {\n            const [subCmd, destructor] = subscription(initializedModel as TModel, props);\n\n            execCmd(subCmd, dispatch);\n\n            if (destructor) {\n                return destructor;\n            }\n        }\n    }, []);\n\n    return [initializedModel, dispatch];\n}\n\nfunction callUpdate<TProps, TModel, TMessage extends MessageBase> (update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>, msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage> {\n    if (typeof update === \"function\") {\n        return update(model, msg, props);\n    }\n\n    return callUpdateMap(update, msg, model, props);\n}\n\nfunction callUpdateMap<TProps, TModel, TMessage extends MessageBase> (updateMap: UpdateMap<TProps, TModel, TMessage>, msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage> {\n    // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n    // @ts-expect-error -- We know that msg fits\n    const updateFn = updateMap[msg.name as TMessage[\"name\"]] as (msg: TMessage, model: TModel, props: TProps) => UpdateReturnType<TModel, TMsg>;\n\n    return updateFn(msg, model, props);\n}\n\nexport type {\n    SubscriptionResult,\n};\n\nexport {\n    useElmish,\n    callUpdate,\n    callUpdateMap,\n};"],"mappings":"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"}
|
package/installScript.js
ADDED
package/package.json
CHANGED
|
@@ -1,10 +1,11 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "react-elmish",
|
|
3
|
-
"version": "5.
|
|
3
|
+
"version": "5.1.1-beta.1",
|
|
4
4
|
"description": "Elmish for React using Typescript",
|
|
5
5
|
"author": "atheck",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"scripts": {
|
|
8
|
+
"install": "node installScript.js",
|
|
8
9
|
"build": "npm run build:types && npm run build:js",
|
|
9
10
|
"build:types": "tsc --emitDeclarationOnly --project ./src",
|
|
10
11
|
"build:js": "babel src --out-dir dist --extensions \".ts,.tsx\" --ignore \"./**/*.spec.ts\",\"./**/*.spec.tsx\" --source-maps inline",
|
|
@@ -18,22 +19,22 @@
|
|
|
18
19
|
"react": ">=16.8.0"
|
|
19
20
|
},
|
|
20
21
|
"devDependencies": {
|
|
21
|
-
"@babel/cli": "7.
|
|
22
|
-
"@babel/core": "7.19.
|
|
22
|
+
"@babel/cli": "7.19.3",
|
|
23
|
+
"@babel/core": "7.19.3",
|
|
23
24
|
"@babel/plugin-proposal-class-properties": "7.18.6",
|
|
24
|
-
"@babel/preset-env": "7.19.
|
|
25
|
+
"@babel/preset-env": "7.19.4",
|
|
25
26
|
"@babel/preset-react": "7.18.6",
|
|
26
27
|
"@babel/preset-typescript": "7.18.6",
|
|
27
28
|
"@testing-library/react": "13.4.0",
|
|
28
|
-
"@types/jest": "
|
|
29
|
-
"@types/react": "18.0.
|
|
30
|
-
"eslint": "8.
|
|
31
|
-
"eslint-config-heck": "1.
|
|
32
|
-
"jest": "
|
|
33
|
-
"jest-environment-jsdom": "
|
|
29
|
+
"@types/jest": "29.1.2",
|
|
30
|
+
"@types/react": "18.0.21",
|
|
31
|
+
"eslint": "8.25.0",
|
|
32
|
+
"eslint-config-heck": "1.24.0",
|
|
33
|
+
"jest": "29.1.2",
|
|
34
|
+
"jest-environment-jsdom": "29.1.2",
|
|
34
35
|
"semantic-release": "19.0.5",
|
|
35
|
-
"ts-jest": "
|
|
36
|
-
"typescript": "4.8.
|
|
36
|
+
"ts-jest": "29.0.3",
|
|
37
|
+
"typescript": "4.8.4"
|
|
37
38
|
},
|
|
38
39
|
"homepage": "https://github.com/atheck/react-elmish",
|
|
39
40
|
"repository": {
|