react-elmish 4.3.0 → 4.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,6 @@
1
+ import { Cmd, Dispatch } from "./Cmd";
2
+ import { MessageBase } from "./Types";
3
+ declare function logMessage<TMessage extends MessageBase>(name: string, msg: TMessage): void;
4
+ declare function modelHasChanged<TModel>(currentModel: TModel, model: Partial<TModel>): boolean;
5
+ declare function execCmd<TMessage>(cmd: Cmd<TMessage>, dispatch: Dispatch<TMessage>): void;
6
+ export { logMessage, modelHasChanged, execCmd, };
package/dist/Common.js ADDED
@@ -0,0 +1,35 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.execCmd = execCmd;
7
+ exports.logMessage = logMessage;
8
+ exports.modelHasChanged = modelHasChanged;
9
+
10
+ var _Init = require("./Init");
11
+
12
+ function logMessage(name, msg) {
13
+ var _Services$logger, _Services$logger2, _Services$dispatchMid;
14
+
15
+ (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.info("Elm", "message from", name, msg.name);
16
+ (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "message from", name, msg);
17
+ (_Services$dispatchMid = _Init.Services.dispatchMiddleware) === null || _Services$dispatchMid === void 0 ? void 0 : _Services$dispatchMid.call(_Init.Services, msg);
18
+ }
19
+
20
+ function modelHasChanged(currentModel, model) {
21
+ return !Object.is(model, currentModel) && Object.getOwnPropertyNames(model).length > 0;
22
+ }
23
+
24
+ function execCmd(cmd, dispatch) {
25
+ cmd.forEach(function (call) {
26
+ try {
27
+ call(dispatch);
28
+ } catch (ex) {
29
+ var _Services$logger3;
30
+
31
+ (_Services$logger3 = _Init.Services.logger) === null || _Services$logger3 === void 0 ? void 0 : _Services$logger3.error(ex);
32
+ }
33
+ });
34
+ }
35
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJsb2dNZXNzYWdlIiwibmFtZSIsIm1zZyIsIlNlcnZpY2VzIiwibG9nZ2VyIiwiaW5mbyIsImRlYnVnIiwiZGlzcGF0Y2hNaWRkbGV3YXJlIiwibW9kZWxIYXNDaGFuZ2VkIiwiY3VycmVudE1vZGVsIiwibW9kZWwiLCJPYmplY3QiLCJpcyIsImdldE93blByb3BlcnR5TmFtZXMiLCJsZW5ndGgiLCJleGVjQ21kIiwiY21kIiwiZGlzcGF0Y2giLCJmb3JFYWNoIiwiY2FsbCIsImV4IiwiZXJyb3IiXSwic291cmNlcyI6WyIuLi9zcmMvQ29tbW9uLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgRGlzcGF0Y2ggfSBmcm9tIFwiLi9DbWRcIjtcbmltcG9ydCB7IFNlcnZpY2VzIH0gZnJvbSBcIi4vSW5pdFwiO1xuaW1wb3J0IHsgTWVzc2FnZUJhc2UgfSBmcm9tIFwiLi9UeXBlc1wiO1xuXG5mdW5jdGlvbiBsb2dNZXNzYWdlPFRNZXNzYWdlIGV4dGVuZHMgTWVzc2FnZUJhc2U+IChuYW1lOiBzdHJpbmcsIG1zZzogVE1lc3NhZ2UpOiB2b2lkIHtcbiAgICBTZXJ2aWNlcy5sb2dnZXI/LmluZm8oXCJFbG1cIiwgXCJtZXNzYWdlIGZyb21cIiwgbmFtZSwgbXNnLm5hbWUpO1xuICAgIFNlcnZpY2VzLmxvZ2dlcj8uZGVidWcoXCJFbG1cIiwgXCJtZXNzYWdlIGZyb21cIiwgbmFtZSwgbXNnKTtcblxuICAgIFNlcnZpY2VzLmRpc3BhdGNoTWlkZGxld2FyZT8uKG1zZyk7XG59XG5cbmZ1bmN0aW9uIG1vZGVsSGFzQ2hhbmdlZDxUTW9kZWw+IChjdXJyZW50TW9kZWw6IFRNb2RlbCwgbW9kZWw6IFBhcnRpYWw8VE1vZGVsPik6IGJvb2xlYW4ge1xuICAgIHJldHVybiAhT2JqZWN0LmlzKG1vZGVsLCBjdXJyZW50TW9kZWwpICYmIE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzKG1vZGVsKS5sZW5ndGggPiAwO1xufVxuXG5mdW5jdGlvbiBleGVjQ21kPFRNZXNzYWdlPiAoY21kOiBDbWQ8VE1lc3NhZ2U+LCBkaXNwYXRjaDogRGlzcGF0Y2g8VE1lc3NhZ2U+KTogdm9pZCB7XG4gICAgY21kLmZvckVhY2goY2FsbCA9PiB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgICBjYWxsKGRpc3BhdGNoKTtcbiAgICAgICAgfSBjYXRjaCAoZXg6IHVua25vd24pIHtcbiAgICAgICAgICAgIFNlcnZpY2VzLmxvZ2dlcj8uZXJyb3IoZXgpO1xuICAgICAgICB9XG4gICAgfSk7XG59XG5cbmV4cG9ydCB7XG4gICAgbG9nTWVzc2FnZSxcbiAgICBtb2RlbEhhc0NoYW5nZWQsXG4gICAgZXhlY0NtZCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFDQTs7QUFHQSxTQUFTQSxVQUFULENBQW1EQyxJQUFuRCxFQUFpRUMsR0FBakUsRUFBc0Y7RUFBQTs7RUFDbEYsb0JBQUFDLGNBQUEsQ0FBU0MsTUFBVCxzRUFBaUJDLElBQWpCLENBQXNCLEtBQXRCLEVBQTZCLGNBQTdCLEVBQTZDSixJQUE3QyxFQUFtREMsR0FBRyxDQUFDRCxJQUF2RDtFQUNBLHFCQUFBRSxjQUFBLENBQVNDLE1BQVQsd0VBQWlCRSxLQUFqQixDQUF1QixLQUF2QixFQUE4QixjQUE5QixFQUE4Q0wsSUFBOUMsRUFBb0RDLEdBQXBEO0VBRUEseUJBQUFDLGNBQUEsQ0FBU0ksa0JBQVQscUZBQUFKLGNBQUEsRUFBOEJELEdBQTlCO0FBQ0g7O0FBRUQsU0FBU00sZUFBVCxDQUFrQ0MsWUFBbEMsRUFBd0RDLEtBQXhELEVBQXlGO0VBQ3JGLE9BQU8sQ0FBQ0MsTUFBTSxDQUFDQyxFQUFQLENBQVVGLEtBQVYsRUFBaUJELFlBQWpCLENBQUQsSUFBbUNFLE1BQU0sQ0FBQ0UsbUJBQVAsQ0FBMkJILEtBQTNCLEVBQWtDSSxNQUFsQyxHQUEyQyxDQUFyRjtBQUNIOztBQUVELFNBQVNDLE9BQVQsQ0FBNEJDLEdBQTVCLEVBQWdEQyxRQUFoRCxFQUFvRjtFQUNoRkQsR0FBRyxDQUFDRSxPQUFKLENBQVksVUFBQUMsSUFBSSxFQUFJO0lBQ2hCLElBQUk7TUFDQUEsSUFBSSxDQUFDRixRQUFELENBQUo7SUFDSCxDQUZELENBRUUsT0FBT0csRUFBUCxFQUFvQjtNQUFBOztNQUNsQixxQkFBQWpCLGNBQUEsQ0FBU0MsTUFBVCx3RUFBaUJpQixLQUFqQixDQUF1QkQsRUFBdkI7SUFDSDtFQUNKLENBTkQ7QUFPSCJ9
@@ -38,7 +38,6 @@ declare abstract class ElmComponent<TModel, TMessage extends Message, TProps> ex
38
38
  * @memberof ElmComponent
39
39
  */
40
40
  componentWillUnmount(): void;
41
- private execCmd;
42
41
  /**
43
42
  * Returns the current model.
44
43
  * @readonly
@@ -9,6 +9,8 @@ exports.ElmComponent = void 0;
9
9
 
10
10
  var _react = _interopRequireDefault(require("react"));
11
11
 
12
+ var _Common = require("./Common");
13
+
12
14
  var _Init = require("./Init");
13
15
 
14
16
  var _fakeOptions = require("./Testing/fakeOptions");
@@ -95,10 +97,6 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
95
97
  _defineProperty(_assertThisInitialized(_this), "currentModel", void 0);
96
98
 
97
99
  _defineProperty(_assertThisInitialized(_this), "dispatch", function (msg) {
98
- var modelHasChanged = function modelHasChanged(model) {
99
- return model !== _this.currentModel && Object.getOwnPropertyNames(model).length > 0;
100
- };
101
-
102
100
  if (_this.reentered) {
103
101
  _this.buffer.push(msg);
104
102
  } else {
@@ -107,14 +105,7 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
107
105
  var modified = false;
108
106
 
109
107
  while (nextMsg) {
110
- var _Services$logger, _Services$logger2;
111
-
112
- (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.info("Elm", "message from", _this.componentName, nextMsg.name);
113
- (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "message from", _this.componentName, nextMsg);
114
-
115
- if (_Init.Services.dispatchMiddleware) {
116
- _Init.Services.dispatchMiddleware(nextMsg);
117
- }
108
+ (0, _Common.logMessage)(_this.componentName, nextMsg);
118
109
 
119
110
  try {
120
111
  var _this$update = _this.update(_this.currentModel, nextMsg, _this.props),
@@ -122,18 +113,18 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
122
113
  model = _this$update2[0],
123
114
  cmd = _this$update2[1];
124
115
 
125
- if (modelHasChanged(model)) {
116
+ if ((0, _Common.modelHasChanged)(_this.currentModel, model)) {
126
117
  _this.currentModel = _objectSpread(_objectSpread({}, _this.currentModel), model);
127
118
  modified = true;
128
119
  }
129
120
 
130
121
  if (cmd) {
131
- _this.execCmd(cmd);
122
+ (0, _Common.execCmd)(cmd, _this.dispatch);
132
123
  }
133
124
  } catch (ex) {
134
- var _Services$logger3;
125
+ var _Services$logger;
135
126
 
136
- (_Services$logger3 = _Init.Services.logger) === null || _Services$logger3 === void 0 ? void 0 : _Services$logger3.error(ex);
127
+ (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.error(ex);
137
128
  }
138
129
 
139
130
  nextMsg = _this.buffer.shift();
@@ -142,9 +133,9 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
142
133
  _this.reentered = false;
143
134
 
144
135
  if (_this.mounted && modified) {
145
- var _Services$logger4;
136
+ var _Services$logger2;
146
137
 
147
- (_Services$logger4 = _Init.Services.logger) === null || _Services$logger4 === void 0 ? void 0 : _Services$logger4.debug("Elm", "update model for", _this.componentName, _this.currentModel);
138
+ (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "update model for", _this.componentName, _this.currentModel);
148
139
 
149
140
  _this.forceUpdate();
150
141
  }
@@ -182,7 +173,7 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
182
173
  this.mounted = true;
183
174
 
184
175
  if (this.initCmd) {
185
- this.execCmd(this.initCmd);
176
+ (0, _Common.execCmd)(this.initCmd, this.dispatch);
186
177
  this.initCmd = null;
187
178
  }
188
179
  }
@@ -197,21 +188,6 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
197
188
  value: function componentWillUnmount() {
198
189
  this.mounted = false;
199
190
  }
200
- }, {
201
- key: "execCmd",
202
- value: function execCmd(cmd) {
203
- var _this2 = this;
204
-
205
- cmd.forEach(function (call) {
206
- try {
207
- call(_this2.dispatch);
208
- } catch (ex) {
209
- var _Services$logger5;
210
-
211
- (_Services$logger5 = _Init.Services.logger) === null || _Services$logger5 === void 0 ? void 0 : _Services$logger5.error(ex);
212
- }
213
- });
214
- }
215
191
  /**
216
192
  * Returns the current model.
217
193
  * @readonly
@@ -236,4 +212,4 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
236
212
  }(_react["default"].Component);
237
213
 
238
214
  exports.ElmComponent = ElmComponent;
239
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ElmComponent","props","init","name","msg","modelHasChanged","model","currentModel","Object","getOwnPropertyNames","length","reentered","buffer","push","nextMsg","modified","Services","logger","info","componentName","debug","dispatchMiddleware","update","cmd","execCmd","ex","error","shift","mounted","forceUpdate","fakeOptions","getFakeOptionsOnce","dispatch","initCmd","forEach","call","React","Component"],"sources":["../src/ElmComponent.ts"],"sourcesContent":["import React from \"react\";\nimport { Cmd } from \"./Cmd\";\nimport { Message, Services } from \"./Init\";\nimport { getFakeOptionsOnce } from \"./Testing/fakeOptions\";\nimport { InitFunction, Nullable, UpdateFunction } from \"./Types\";\n\n/**\n * Abstract class for a react class component using the Elmish pattern.\n * @export\n * @abstract\n * @class ElmComponent\n * @extends {Component<TProps, TModel>}\n * @template TModel The type of the model.\n * @template TMessage The type of the messages.\n * @template TProps The type of the props.\n */\nabstract class ElmComponent<TModel, TMessage extends Message, TProps> extends React.Component<TProps> {\n    private initCmd: Nullable<Cmd<TMessage>> | undefined;\n    private readonly componentName: string;\n    private readonly buffer: TMessage [] = [];\n    private reentered = false;\n    private mounted = false;\n    private currentModel: TModel;\n\n    /**\n     * Creates an instance of ElmComponent.\n     * @param {TProps} props The props for the component.\n     * @param {() => TModel} init The initializer function.\n     * @param name The name of the component.\n     * @memberof ElmComponent\n     */\n    public constructor (props: TProps, init: InitFunction<TProps, TModel, TMessage>, name: string) {\n        super(props);\n\n        const fakeOptions = getFakeOptionsOnce();\n\n        if (fakeOptions?.dispatch) {\n            this.dispatch = fakeOptions.dispatch;\n        }\n\n        const [model, cmd] = fakeOptions?.model ? [fakeOptions.model as TModel] : init(this.props);\n\n        this.componentName = name;\n        this.currentModel = model;\n        this.initCmd = cmd;\n    }\n\n    /**\n     * Is called when the component is loaded.\n     * When implementing this method, the base implementation has to be called.\n     * @memberof ElmComponent\n     */\n    public componentDidMount (): void {\n        this.mounted = true;\n\n        if (this.initCmd) {\n            this.execCmd(this.initCmd);\n            this.initCmd = null;\n        }\n    }\n\n    /**\n     * Is called before unloading the component.\n     * When implementing this method, the base implementation has to be called.\n     * @memberof ElmComponent\n     */\n    public componentWillUnmount (): void {\n        this.mounted = false;\n    }\n\n    private execCmd (cmd: Cmd<TMessage>): void {\n        cmd.forEach(call => {\n            try {\n                call(this.dispatch);\n            } catch (ex: unknown) {\n                Services.logger?.error(ex);\n            }\n        });\n    }\n\n    /**\n     * Returns the current model.\n     * @readonly\n     * @type {Readonly<TModel>}\n     * @memberof ElmComponent\n     */\n    public get model (): Readonly<TModel> {\n        return this.currentModel;\n    }\n\n    /**\n     * Dispatches a message.\n     * @param {TMessage} msg The message to dispatch.\n     * @memberof ElmComponent\n     */\n    public readonly dispatch = (msg: TMessage): void => {\n        const modelHasChanged = (model: Partial<TModel>): boolean => model !== this.currentModel && Object.getOwnPropertyNames(model).length > 0;\n\n        if (this.reentered) {\n            this.buffer.push(msg);\n        } else {\n            this.reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                Services.logger?.info(\"Elm\", \"message from\", this.componentName, nextMsg.name);\n                Services.logger?.debug(\"Elm\", \"message from\", this.componentName, nextMsg);\n\n                if (Services.dispatchMiddleware) {\n                    Services.dispatchMiddleware(nextMsg);\n                }\n\n                try {\n                    const [model, cmd] = this.update(this.currentModel, nextMsg, this.props);\n\n                    if (modelHasChanged(model)) {\n                        this.currentModel = { ...this.currentModel, ...model };\n                        modified = true;\n                    }\n\n                    if (cmd) {\n                        this.execCmd(cmd);\n                    }\n                } catch (ex: unknown) {\n                    Services.logger?.error(ex);\n                }\n\n                nextMsg = this.buffer.shift();\n            }\n            this.reentered = false;\n\n            if (this.mounted && modified) {\n                Services.logger?.debug(\"Elm\", \"update model for\", this.componentName, this.currentModel);\n                this.forceUpdate();\n            }\n        }\n    };\n\n    /**\n     * Function to modify the model based on a message.\n     * @param {TModel} model The current model.\n     * @param {TMessage} msg The message to process.\n     * @param {TProps} props The props of the component.\n     * @returns The new model (can also be an empty object {}) and an optional new message to dispatch.\n     * @abstract\n     * @memberof ElmComponent\n     */\n    public abstract update: UpdateFunction<TProps, TModel, TMessage>;\n}\n\nexport {\n    ElmComponent,\n};"],"mappings":"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"}
215
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ElmComponent","props","init","name","msg","reentered","buffer","push","nextMsg","modified","logMessage","componentName","update","currentModel","model","cmd","modelHasChanged","execCmd","dispatch","ex","Services","logger","error","shift","mounted","debug","forceUpdate","fakeOptions","getFakeOptionsOnce","initCmd","React","Component"],"sources":["../src/ElmComponent.ts"],"sourcesContent":["import React from \"react\";\nimport { Cmd } from \"./Cmd\";\nimport { execCmd, logMessage, modelHasChanged } from \"./Common\";\nimport { Message, Services } from \"./Init\";\nimport { getFakeOptionsOnce } from \"./Testing/fakeOptions\";\nimport { InitFunction, Nullable, UpdateFunction } from \"./Types\";\n\n/**\n * Abstract class for a react class component using the Elmish pattern.\n * @export\n * @abstract\n * @class ElmComponent\n * @extends {Component<TProps, TModel>}\n * @template TModel The type of the model.\n * @template TMessage The type of the messages.\n * @template TProps The type of the props.\n */\nabstract class ElmComponent<TModel, TMessage extends Message, TProps> extends React.Component<TProps> {\n    private initCmd: Nullable<Cmd<TMessage>> | undefined;\n    private readonly componentName: string;\n    private readonly buffer: TMessage [] = [];\n    private reentered = false;\n    private mounted = false;\n    private currentModel: TModel;\n\n    /**\n     * Creates an instance of ElmComponent.\n     * @param {TProps} props The props for the component.\n     * @param {() => TModel} init The initializer function.\n     * @param name The name of the component.\n     * @memberof ElmComponent\n     */\n    public constructor (props: TProps, init: InitFunction<TProps, TModel, TMessage>, name: string) {\n        super(props);\n\n        const fakeOptions = getFakeOptionsOnce();\n\n        if (fakeOptions?.dispatch) {\n            this.dispatch = fakeOptions.dispatch;\n        }\n\n        const [model, cmd] = fakeOptions?.model ? [fakeOptions.model as TModel] : init(this.props);\n\n        this.componentName = name;\n        this.currentModel = model;\n        this.initCmd = cmd;\n    }\n\n    /**\n     * Is called when the component is loaded.\n     * When implementing this method, the base implementation has to be called.\n     * @memberof ElmComponent\n     */\n    public componentDidMount (): void {\n        this.mounted = true;\n\n        if (this.initCmd) {\n            execCmd(this.initCmd, this.dispatch);\n            this.initCmd = null;\n        }\n    }\n\n    /**\n     * Is called before unloading the component.\n     * When implementing this method, the base implementation has to be called.\n     * @memberof ElmComponent\n     */\n    public componentWillUnmount (): void {\n        this.mounted = false;\n    }\n\n    /**\n     * Returns the current model.\n     * @readonly\n     * @type {Readonly<TModel>}\n     * @memberof ElmComponent\n     */\n    public get model (): Readonly<TModel> {\n        return this.currentModel;\n    }\n\n    /**\n     * Dispatches a message.\n     * @param {TMessage} msg The message to dispatch.\n     * @memberof ElmComponent\n     */\n    public readonly dispatch = (msg: TMessage): void => {\n        if (this.reentered) {\n            this.buffer.push(msg);\n        } else {\n            this.reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                logMessage(this.componentName, nextMsg);\n\n                try {\n                    const [model, cmd] = this.update(this.currentModel, nextMsg, this.props);\n\n                    if (modelHasChanged(this.currentModel, model)) {\n                        this.currentModel = { ...this.currentModel, ...model };\n                        modified = true;\n                    }\n\n                    if (cmd) {\n                        execCmd(cmd, this.dispatch);\n                    }\n                } catch (ex: unknown) {\n                    Services.logger?.error(ex);\n                }\n\n                nextMsg = this.buffer.shift();\n            }\n            this.reentered = false;\n\n            if (this.mounted && modified) {\n                Services.logger?.debug(\"Elm\", \"update model for\", this.componentName, this.currentModel);\n                this.forceUpdate();\n            }\n        }\n    };\n\n    /**\n     * Function to modify the model based on a message.\n     * @param {TModel} model The current model.\n     * @param {TMessage} msg The message to process.\n     * @param {TProps} props The props of the component.\n     * @returns The new model (can also be an empty object {}) and an optional new message to dispatch.\n     * @abstract\n     * @memberof ElmComponent\n     */\n    public abstract update: UpdateFunction<TProps, TModel, TMessage>;\n}\n\nexport {\n    ElmComponent,\n};"],"mappings":"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"}
package/dist/useElmish.js CHANGED
@@ -9,6 +9,8 @@ exports.useElmish = useElmish;
9
9
 
10
10
  var _react = require("react");
11
11
 
12
+ var _Common = require("./Common");
13
+
12
14
  var _Init = require("./Init");
13
15
 
14
16
  var _fakeOptions = require("./Testing/fakeOptions");
@@ -55,34 +57,25 @@ function useElmish(_ref) {
55
57
  model = _useState2[0],
56
58
  setModel = _useState2[1];
57
59
 
58
- var initializedModel = model;
59
60
  var propsRef = (0, _react.useRef)(props);
61
+ var isMountedRef = (0, _react.useRef)(true);
62
+ (0, _react.useEffect)(function () {
63
+ return function () {
64
+ isMountedRef.current = false;
65
+ };
66
+ }, []);
67
+ var initializedModel = model;
60
68
 
61
69
  if (propsRef.current !== props) {
62
70
  propsRef.current = props;
63
71
  }
64
72
 
65
- var execCmd = (0, _react.useCallback)(function (cmd) {
66
- cmd.forEach(function (call) {
67
- try {
68
- call(dispatch);
69
- } catch (ex) {
70
- var _Services$logger;
71
-
72
- (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.error(ex);
73
- }
74
- });
75
- }, []);
76
73
  var fakeOptions = (0, _fakeOptions.getFakeOptionsOnce)();
77
74
  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) {
78
75
  if (!initializedModel) {
79
76
  return;
80
77
  }
81
78
 
82
- var modelHasChanged = function modelHasChanged(updatedModel) {
83
- return updatedModel !== initializedModel && Object.getOwnPropertyNames(updatedModel).length > 0;
84
- };
85
-
86
79
  if (reentered) {
87
80
  buffer.push(msg);
88
81
  } else {
@@ -91,14 +84,7 @@ function useElmish(_ref) {
91
84
  var modified = false;
92
85
 
93
86
  while (nextMsg) {
94
- var _Services$logger2, _Services$logger3;
95
-
96
- (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.info("Elm", "message from", name, nextMsg.name);
97
- (_Services$logger3 = _Init.Services.logger) === null || _Services$logger3 === void 0 ? void 0 : _Services$logger3.debug("Elm", "message from", name, nextMsg);
98
-
99
- if (_Init.Services.dispatchMiddleware) {
100
- _Init.Services.dispatchMiddleware(nextMsg);
101
- }
87
+ (0, _Common.logMessage)(name, nextMsg);
102
88
 
103
89
  try {
104
90
  var _callUpdate = callUpdate(update, nextMsg, _objectSpread(_objectSpread({}, initializedModel), currentModel), propsRef.current),
@@ -106,18 +92,18 @@ function useElmish(_ref) {
106
92
  newModel = _callUpdate2[0],
107
93
  cmd = _callUpdate2[1];
108
94
 
109
- if (modelHasChanged(newModel)) {
95
+ if ((0, _Common.modelHasChanged)(currentModel, newModel)) {
110
96
  currentModel = _objectSpread(_objectSpread({}, currentModel), newModel);
111
97
  modified = true;
112
98
  }
113
99
 
114
100
  if (cmd) {
115
- execCmd(cmd);
101
+ (0, _Common.execCmd)(cmd, dispatch);
116
102
  }
117
103
  } catch (ex) {
118
- var _Services$logger4;
104
+ var _Services$logger;
119
105
 
120
- (_Services$logger4 = _Init.Services.logger) === null || _Services$logger4 === void 0 ? void 0 : _Services$logger4.error(ex);
106
+ (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.error(ex);
121
107
  }
122
108
 
123
109
  nextMsg = buffer.shift();
@@ -125,13 +111,13 @@ function useElmish(_ref) {
125
111
 
126
112
  reentered = false;
127
113
 
128
- if (modified) {
114
+ if (isMountedRef.current && modified) {
129
115
  setModel(function (prevModel) {
130
- var _Services$logger5;
116
+ var _Services$logger2;
131
117
 
132
118
  var updatedModel = _objectSpread(_objectSpread({}, prevModel), currentModel);
133
119
 
134
- (_Services$logger5 = _Init.Services.logger) === null || _Services$logger5 === void 0 ? void 0 : _Services$logger5.debug("Elm", "update model for", name, updatedModel);
120
+ (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "update model for", name, updatedModel);
135
121
  return updatedModel;
136
122
  });
137
123
  }
@@ -148,7 +134,7 @@ function useElmish(_ref) {
148
134
  setModel(initializedModel);
149
135
 
150
136
  if (initCmd) {
151
- execCmd(initCmd);
137
+ (0, _Common.execCmd)(initCmd, dispatch);
152
138
  }
153
139
  }
154
140
 
@@ -159,7 +145,7 @@ function useElmish(_ref) {
159
145
  subCmd = _subscription2[0],
160
146
  destructor = _subscription2[1];
161
147
 
162
- execCmd(subCmd);
148
+ (0, _Common.execCmd)(subCmd, dispatch);
163
149
 
164
150
  if (destructor) {
165
151
  return destructor;
@@ -183,4 +169,4 @@ function callUpdateMap(updateMap, msg, model, props) {
183
169
  var updateFn = updateMap[msg.name];
184
170
  return updateFn(msg, model, props);
185
171
  }
186
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useElmish","name","props","init","update","subscription","reentered","buffer","currentModel","useState","model","setModel","initializedModel","propsRef","useRef","current","execCmd","useCallback","cmd","forEach","call","dispatch","ex","Services","logger","error","fakeOptions","getFakeOptionsOnce","msg","modelHasChanged","updatedModel","Object","getOwnPropertyNames","length","push","nextMsg","modified","info","debug","dispatchMiddleware","callUpdate","newModel","shift","prevModel","initModel","initCmd","useEffect","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 { 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    let initializedModel = model;\n\n    const propsRef = useRef(props);\n\n    if (propsRef.current !== props) {\n        propsRef.current = props;\n    }\n\n    const execCmd = useCallback((cmd: Cmd<TMessage>): void => {\n        cmd.forEach(call => {\n            try {\n                call(dispatch);\n            } catch (ex: unknown) {\n                Services.logger?.error(ex);\n            }\n        });\n    }, []);\n\n    const fakeOptions = getFakeOptionsOnce();\n    const dispatch = useCallback(fakeOptions?.dispatch ?? ((msg: TMessage): void => {\n        if (!initializedModel) {\n            return;\n        }\n\n        const modelHasChanged = (updatedModel: Partial<TModel>): boolean => updatedModel !== initializedModel && Object.getOwnPropertyNames(updatedModel).length > 0;\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                Services.logger?.info(\"Elm\", \"message from\", name, nextMsg.name);\n                Services.logger?.debug(\"Elm\", \"message from\", name, nextMsg);\n\n                if (Services.dispatchMiddleware) {\n                    Services.dispatchMiddleware(nextMsg);\n                }\n\n                try {\n                    const [newModel, cmd] = callUpdate(update, nextMsg, { ...initializedModel, ...currentModel }, propsRef.current);\n\n                    if (modelHasChanged(newModel)) {\n                        currentModel = { ...currentModel, ...newModel };\n\n                        modified = true;\n                    }\n\n                    if (cmd) {\n                        execCmd(cmd);\n                    }\n                } catch (ex: unknown) {\n                    Services.logger?.error(ex);\n                }\n\n                nextMsg = buffer.shift();\n            }\n            reentered = false;\n\n            if (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);\n        }\n    }\n\n    useEffect(() => {\n        if (subscription) {\n            const [subCmd, destructor] = subscription(initializedModel as TModel, props);\n\n            execCmd(subCmd);\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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"}
172
+ //# 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","ex","Services","logger","error","shift","prevModel","updatedModel","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 = false;\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                try {\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                } catch (ex: unknown) {\n                    Services.logger?.error(ex);\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":";;;;;;;;;AACA;;AAEA;;AACA;;AACA;;;;;;;;;;;;;;;;;;;;AA6CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAASA,SAAT,OAAyL;EAAA;;EAAA,IAArHC,IAAqH,QAArHA,IAAqH;EAAA,IAA/GC,KAA+G,QAA/GA,KAA+G;EAAA,IAAxGC,IAAwG,QAAxGA,IAAwG;EAAA,IAAlGC,MAAkG,QAAlGA,MAAkG;EAAA,IAA1FC,YAA0F,QAA1FA,YAA0F;EACrL,IAAIC,SAAS,GAAG,KAAhB;EACA,IAAMC,MAAmB,GAAG,EAA5B;EACA,IAAIC,YAA6B,GAAG,EAApC;;EAEA,gBAA0B,IAAAC,eAAA,EAA2B,IAA3B,CAA1B;EAAA;EAAA,IAAOC,KAAP;EAAA,IAAcC,QAAd;;EACA,IAAMC,QAAQ,GAAG,IAAAC,aAAA,EAAOX,KAAP,CAAjB;EACA,IAAMY,YAAY,GAAG,IAAAD,aAAA,EAAO,IAAP,CAArB;EAEA,IAAAE,gBAAA,EAAU;IAAA,OAAM,YAAM;MAClBD,YAAY,CAACE,OAAb,GAAuB,KAAvB;IACH,CAFS;EAAA,CAAV,EAEG,EAFH;EAIA,IAAIC,gBAAgB,GAAGP,KAAvB;;EAEA,IAAIE,QAAQ,CAACI,OAAT,KAAqBd,KAAzB,EAAgC;IAC5BU,QAAQ,CAACI,OAAT,GAAmBd,KAAnB;EACH;;EAED,IAAMgB,WAAW,GAAG,IAAAC,+BAAA,GAApB;EACA,IAAMC,QAAQ,GAAG,IAAAC,kBAAA,2BAAYH,WAAZ,aAAYA,WAAZ,uBAAYA,WAAW,CAAEE,QAAzB,yEAAsC,UAACE,GAAD,EAAyB;IAC5E,IAAI,CAACL,gBAAL,EAAuB;MACnB;IACH;;IAED,IAAIX,SAAJ,EAAe;MACXC,MAAM,CAACgB,IAAP,CAAYD,GAAZ;IACH,CAFD,MAEO;MACHhB,SAAS,GAAG,IAAZ;MAEA,IAAIkB,OAA6B,GAAGF,GAApC;MACA,IAAIG,QAAQ,GAAG,KAAf;;MAEA,OAAOD,OAAP,EAAgB;QACZ,IAAAE,kBAAA,EAAWzB,IAAX,EAAiBuB,OAAjB;;QAEA,IAAI;UACA,kBAAwBG,UAAU,CAACvB,MAAD,EAASoB,OAAT,kCAAuBP,gBAAvB,GAA4CT,YAA5C,GAA4DI,QAAQ,CAACI,OAArE,CAAlC;UAAA;UAAA,IAAOY,QAAP;UAAA,IAAiBC,GAAjB;;UAEA,IAAI,IAAAC,uBAAA,EAAgBtB,YAAhB,EAA8BoB,QAA9B,CAAJ,EAA6C;YACzCpB,YAAY,mCAAQA,YAAR,GAAyBoB,QAAzB,CAAZ;YAEAH,QAAQ,GAAG,IAAX;UACH;;UAED,IAAII,GAAJ,EAAS;YACL,IAAAE,eAAA,EAAQF,GAAR,EAAaT,QAAb;UACH;QACJ,CAZD,CAYE,OAAOY,EAAP,EAAoB;UAAA;;UAClB,oBAAAC,cAAA,CAASC,MAAT,sEAAiBC,KAAjB,CAAuBH,EAAvB;QACH;;QAEDR,OAAO,GAAGjB,MAAM,CAAC6B,KAAP,EAAV;MACH;;MACD9B,SAAS,GAAG,KAAZ;;MAEA,IAAIQ,YAAY,CAACE,OAAb,IAAwBS,QAA5B,EAAsC;QAClCd,QAAQ,CAAC,UAAA0B,SAAS,EAAI;UAAA;;UAClB,IAAMC,YAAY,mCAAQD,SAAR,GAAgC7B,YAAhC,CAAlB;;UAEA,qBAAAyB,cAAA,CAASC,MAAT,wEAAiBK,KAAjB,CAAuB,KAAvB,EAA8B,kBAA9B,EAAkDtC,IAAlD,EAAwDqC,YAAxD;UAEA,OAAOA,YAAP;QACH,CANO,CAAR;MAOH;IACJ;EACJ,CA9CgB,EA8Cb,EA9Ca,CAAjB;;EAgDA,IAAI,CAACrB,gBAAL,EAAuB;IACnB,YAA6BC,WAAW,SAAX,IAAAA,WAAW,WAAX,IAAAA,WAAW,CAAER,KAAb,GAAqB,CAACQ,WAAW,CAACR,KAAb,CAArB,GAAqDP,IAAI,CAACD,KAAD,CAAtF;IAAA;IAAA,IAAOsC,SAAP;IAAA,IAAkBC,OAAlB;;IAEAxB,gBAAgB,GAAGuB,SAAnB;IACA7B,QAAQ,CAACM,gBAAD,CAAR;;IAEA,IAAIwB,OAAJ,EAAa;MACT,IAAAV,eAAA,EAAQU,OAAR,EAAiBrB,QAAjB;IACH;EACJ;;EAED,IAAAL,gBAAA,EAAU,YAAM;IACZ,IAAIV,YAAJ,EAAkB;MACd,oBAA6BA,YAAY,CAACY,gBAAD,EAA6Bf,KAA7B,CAAzC;MAAA;MAAA,IAAOwC,MAAP;MAAA,IAAeC,UAAf;;MAEA,IAAAZ,eAAA,EAAQW,MAAR,EAAgBtB,QAAhB;;MAEA,IAAIuB,UAAJ,EAAgB;QACZ,OAAOA,UAAP;MACH;IACJ;EACJ,CAVD,EAUG,EAVH;EAYA,OAAO,CAAC1B,gBAAD,EAAmBG,QAAnB,CAAP;AACH;;AAED,SAASO,UAAT,CAAmEvB,MAAnE,EAA2JkB,GAA3J,EAA0KZ,KAA1K,EAAyLR,KAAzL,EAA4O;EACxO,IAAI,OAAOE,MAAP,KAAkB,UAAtB,EAAkC;IAC9B,OAAOA,MAAM,CAACM,KAAD,EAAQY,GAAR,EAAapB,KAAb,CAAb;EACH;;EAED,OAAO0C,aAAa,CAACxC,MAAD,EAASkB,GAAT,EAAcZ,KAAd,EAAqBR,KAArB,CAApB;AACH;;AAED,SAAS0C,aAAT,CAAsEC,SAAtE,EAAsHvB,GAAtH,EAAqIZ,KAArI,EAAoJR,KAApJ,EAAuM;EACnM;EACA;EACA,IAAM4C,QAAQ,GAAGD,SAAS,CAACvB,GAAG,CAACrB,IAAL,CAA1B;EAEA,OAAO6C,QAAQ,CAACxB,GAAD,EAAMZ,KAAN,EAAaR,KAAb,CAAf;AACH"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "react-elmish",
3
- "version": "4.3.0",
3
+ "version": "4.4.0",
4
4
  "description": "Elmish for React using Typescript",
5
5
  "author": "atheck",
6
6
  "license": "MIT",
@@ -19,21 +19,21 @@
19
19
  },
20
20
  "devDependencies": {
21
21
  "@babel/cli": "7.18.10",
22
- "@babel/core": "7.18.10",
22
+ "@babel/core": "7.18.13",
23
23
  "@babel/plugin-proposal-class-properties": "7.18.6",
24
24
  "@babel/preset-env": "7.18.10",
25
25
  "@babel/preset-react": "7.18.6",
26
26
  "@babel/preset-typescript": "7.18.6",
27
27
  "@testing-library/react": "13.3.0",
28
- "@types/jest": "28.1.7",
28
+ "@types/jest": "28.1.8",
29
29
  "@types/react": "18.0.17",
30
30
  "eslint": "8.22.0",
31
- "eslint-config-heck": "1.22.1",
31
+ "eslint-config-heck": "1.22.3",
32
32
  "jest": "28.1.3",
33
33
  "jest-environment-jsdom": "28.1.3",
34
- "semantic-release": "19.0.3",
34
+ "semantic-release": "19.0.5",
35
35
  "ts-jest": "28.0.8",
36
- "typescript": "4.7.4"
36
+ "typescript": "4.8.2"
37
37
  },
38
38
  "homepage": "https://github.com/atheck/react-elmish",
39
39
  "repository": {