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/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,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgY3JlYXRlQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgRWxtQ29tcG9uZW50IH0gZnJvbSBcIi4vRWxtQ29tcG9uZW50XCI7XG5pbXBvcnQgeyBlcnJvckhhbmRsZXIsIEVycm9yTWVzc2FnZSwgZXJyb3JNc2csIGhhbmRsZUVycm9yIH0gZnJvbSBcIi4vRXJyb3JIYW5kbGluZ1wiO1xuaW1wb3J0IHsgaW5pdCwgTG9nZ2VyLCBNZXNzYWdlIH0gZnJvbSBcIi4vSW5pdFwiO1xuaW1wb3J0IHsgSW5pdFJlc3VsdCwgTXNnU291cmNlLCBVcGRhdGVNYXAsIFVwZGF0ZVJldHVyblR5cGUgfSBmcm9tIFwiLi9UeXBlc1wiO1xuaW1wb3J0IHsgU3Vic2NyaXB0aW9uUmVzdWx0LCB1c2VFbG1pc2ggfSBmcm9tIFwiLi91c2VFbG1pc2hcIjtcblxuZXhwb3J0IHR5cGUge1xuICAgIExvZ2dlcixcbiAgICBNZXNzYWdlLFxuICAgIENtZCxcbiAgICBEaXNwYXRjaCxcbiAgICBJbml0UmVzdWx0LFxuICAgIFVwZGF0ZVJldHVyblR5cGUsXG4gICAgU3Vic2NyaXB0aW9uUmVzdWx0LFxuICAgIE1zZ1NvdXJjZSxcbiAgICBVcGRhdGVNYXAsXG4gICAgRXJyb3JNZXNzYWdlLFxufTtcblxuZXhwb3J0IHtcbiAgICBpbml0LFxuICAgIGNyZWF0ZUNtZCxcbiAgICBFbG1Db21wb25lbnQsXG4gICAgZXJyb3JNc2csXG4gICAgZXJyb3JIYW5kbGVyLFxuICAgIGhhbmRsZUVycm9yLFxuICAgIHVzZUVsbWlzaCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFFQSJ9
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
- props = _ref.props,
48
- init = _ref.init,
49
- update = _ref.update,
50
- subscription = _ref.subscription;
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
- _useState2 = _slicedToArray(_useState, 2),
57
- model = _useState2[0],
58
- setModel = _useState2[1];
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
- _callUpdate2 = _slicedToArray(_callUpdate, 2),
92
- newModel = _callUpdate2[0],
93
- cmd = _callUpdate2[1];
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
- _ref3 = _slicedToArray(_ref2, 2),
125
- initModel = _ref3[0],
126
- initCmd = _ref3[1];
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
- _subscription2 = _slicedToArray(_subscription, 2),
140
- subCmd = _subscription2[0],
141
- destructor = _subscription2[1];
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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"}
@@ -0,0 +1,4 @@
1
+ "use strict";
2
+
3
+ // eslint-disable-next-line no-console
4
+ console.warn("react-elmish: Symbols for message names are deprecated now and will be removed in the next major version. Please convert all message names into strings.");
package/package.json CHANGED
@@ -1,10 +1,11 @@
1
1
  {
2
2
  "name": "react-elmish",
3
- "version": "5.0.1",
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.18.10",
22
- "@babel/core": "7.19.0",
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.0",
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": "28.1.8",
29
- "@types/react": "18.0.18",
30
- "eslint": "8.23.0",
31
- "eslint-config-heck": "1.23.0",
32
- "jest": "28.1.3",
33
- "jest-environment-jsdom": "28.1.3",
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": "28.0.8",
36
- "typescript": "4.8.2"
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": {