react-elmish 4.4.0 → 5.0.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/Common.js CHANGED
@@ -23,13 +23,7 @@ function modelHasChanged(currentModel, model) {
23
23
 
24
24
  function execCmd(cmd, dispatch) {
25
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
- }
26
+ return call(dispatch);
33
27
  });
34
28
  }
35
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJsb2dNZXNzYWdlIiwibmFtZSIsIm1zZyIsIlNlcnZpY2VzIiwibG9nZ2VyIiwiaW5mbyIsImRlYnVnIiwiZGlzcGF0Y2hNaWRkbGV3YXJlIiwibW9kZWxIYXNDaGFuZ2VkIiwiY3VycmVudE1vZGVsIiwibW9kZWwiLCJPYmplY3QiLCJpcyIsImdldE93blByb3BlcnR5TmFtZXMiLCJsZW5ndGgiLCJleGVjQ21kIiwiY21kIiwiZGlzcGF0Y2giLCJmb3JFYWNoIiwiY2FsbCIsImV4IiwiZXJyb3IiXSwic291cmNlcyI6WyIuLi9zcmMvQ29tbW9uLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgRGlzcGF0Y2ggfSBmcm9tIFwiLi9DbWRcIjtcbmltcG9ydCB7IFNlcnZpY2VzIH0gZnJvbSBcIi4vSW5pdFwiO1xuaW1wb3J0IHsgTWVzc2FnZUJhc2UgfSBmcm9tIFwiLi9UeXBlc1wiO1xuXG5mdW5jdGlvbiBsb2dNZXNzYWdlPFRNZXNzYWdlIGV4dGVuZHMgTWVzc2FnZUJhc2U+IChuYW1lOiBzdHJpbmcsIG1zZzogVE1lc3NhZ2UpOiB2b2lkIHtcbiAgICBTZXJ2aWNlcy5sb2dnZXI/LmluZm8oXCJFbG1cIiwgXCJtZXNzYWdlIGZyb21cIiwgbmFtZSwgbXNnLm5hbWUpO1xuICAgIFNlcnZpY2VzLmxvZ2dlcj8uZGVidWcoXCJFbG1cIiwgXCJtZXNzYWdlIGZyb21cIiwgbmFtZSwgbXNnKTtcblxuICAgIFNlcnZpY2VzLmRpc3BhdGNoTWlkZGxld2FyZT8uKG1zZyk7XG59XG5cbmZ1bmN0aW9uIG1vZGVsSGFzQ2hhbmdlZDxUTW9kZWw+IChjdXJyZW50TW9kZWw6IFRNb2RlbCwgbW9kZWw6IFBhcnRpYWw8VE1vZGVsPik6IGJvb2xlYW4ge1xuICAgIHJldHVybiAhT2JqZWN0LmlzKG1vZGVsLCBjdXJyZW50TW9kZWwpICYmIE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzKG1vZGVsKS5sZW5ndGggPiAwO1xufVxuXG5mdW5jdGlvbiBleGVjQ21kPFRNZXNzYWdlPiAoY21kOiBDbWQ8VE1lc3NhZ2U+LCBkaXNwYXRjaDogRGlzcGF0Y2g8VE1lc3NhZ2U+KTogdm9pZCB7XG4gICAgY21kLmZvckVhY2goY2FsbCA9PiB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgICBjYWxsKGRpc3BhdGNoKTtcbiAgICAgICAgfSBjYXRjaCAoZXg6IHVua25vd24pIHtcbiAgICAgICAgICAgIFNlcnZpY2VzLmxvZ2dlcj8uZXJyb3IoZXgpO1xuICAgICAgICB9XG4gICAgfSk7XG59XG5cbmV4cG9ydCB7XG4gICAgbG9nTWVzc2FnZSxcbiAgICBtb2RlbEhhc0NoYW5nZWQsXG4gICAgZXhlY0NtZCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFDQTs7QUFHQSxTQUFTQSxVQUFULENBQW1EQyxJQUFuRCxFQUFpRUMsR0FBakUsRUFBc0Y7RUFBQTs7RUFDbEYsb0JBQUFDLGNBQUEsQ0FBU0MsTUFBVCxzRUFBaUJDLElBQWpCLENBQXNCLEtBQXRCLEVBQTZCLGNBQTdCLEVBQTZDSixJQUE3QyxFQUFtREMsR0FBRyxDQUFDRCxJQUF2RDtFQUNBLHFCQUFBRSxjQUFBLENBQVNDLE1BQVQsd0VBQWlCRSxLQUFqQixDQUF1QixLQUF2QixFQUE4QixjQUE5QixFQUE4Q0wsSUFBOUMsRUFBb0RDLEdBQXBEO0VBRUEseUJBQUFDLGNBQUEsQ0FBU0ksa0JBQVQscUZBQUFKLGNBQUEsRUFBOEJELEdBQTlCO0FBQ0g7O0FBRUQsU0FBU00sZUFBVCxDQUFrQ0MsWUFBbEMsRUFBd0RDLEtBQXhELEVBQXlGO0VBQ3JGLE9BQU8sQ0FBQ0MsTUFBTSxDQUFDQyxFQUFQLENBQVVGLEtBQVYsRUFBaUJELFlBQWpCLENBQUQsSUFBbUNFLE1BQU0sQ0FBQ0UsbUJBQVAsQ0FBMkJILEtBQTNCLEVBQWtDSSxNQUFsQyxHQUEyQyxDQUFyRjtBQUNIOztBQUVELFNBQVNDLE9BQVQsQ0FBNEJDLEdBQTVCLEVBQWdEQyxRQUFoRCxFQUFvRjtFQUNoRkQsR0FBRyxDQUFDRSxPQUFKLENBQVksVUFBQUMsSUFBSSxFQUFJO0lBQ2hCLElBQUk7TUFDQUEsSUFBSSxDQUFDRixRQUFELENBQUo7SUFDSCxDQUZELENBRUUsT0FBT0csRUFBUCxFQUFvQjtNQUFBOztNQUNsQixxQkFBQWpCLGNBQUEsQ0FBU0MsTUFBVCx3RUFBaUJpQixLQUFqQixDQUF1QkQsRUFBdkI7SUFDSDtFQUNKLENBTkQ7QUFPSCJ9
29
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJsb2dNZXNzYWdlIiwibmFtZSIsIm1zZyIsIlNlcnZpY2VzIiwibG9nZ2VyIiwiaW5mbyIsImRlYnVnIiwiZGlzcGF0Y2hNaWRkbGV3YXJlIiwibW9kZWxIYXNDaGFuZ2VkIiwiY3VycmVudE1vZGVsIiwibW9kZWwiLCJPYmplY3QiLCJpcyIsImdldE93blByb3BlcnR5TmFtZXMiLCJsZW5ndGgiLCJleGVjQ21kIiwiY21kIiwiZGlzcGF0Y2giLCJmb3JFYWNoIiwiY2FsbCJdLCJzb3VyY2VzIjpbIi4uL3NyYy9Db21tb24udHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgU2VydmljZXMgfSBmcm9tIFwiLi9Jbml0XCI7XG5pbXBvcnQgeyBNZXNzYWdlQmFzZSB9IGZyb20gXCIuL1R5cGVzXCI7XG5cbmZ1bmN0aW9uIGxvZ01lc3NhZ2U8VE1lc3NhZ2UgZXh0ZW5kcyBNZXNzYWdlQmFzZT4gKG5hbWU6IHN0cmluZywgbXNnOiBUTWVzc2FnZSk6IHZvaWQge1xuICAgIFNlcnZpY2VzLmxvZ2dlcj8uaW5mbyhcIkVsbVwiLCBcIm1lc3NhZ2UgZnJvbVwiLCBuYW1lLCBtc2cubmFtZSk7XG4gICAgU2VydmljZXMubG9nZ2VyPy5kZWJ1ZyhcIkVsbVwiLCBcIm1lc3NhZ2UgZnJvbVwiLCBuYW1lLCBtc2cpO1xuXG4gICAgU2VydmljZXMuZGlzcGF0Y2hNaWRkbGV3YXJlPy4obXNnKTtcbn1cblxuZnVuY3Rpb24gbW9kZWxIYXNDaGFuZ2VkPFRNb2RlbD4gKGN1cnJlbnRNb2RlbDogVE1vZGVsLCBtb2RlbDogUGFydGlhbDxUTW9kZWw+KTogYm9vbGVhbiB7XG4gICAgcmV0dXJuICFPYmplY3QuaXMobW9kZWwsIGN1cnJlbnRNb2RlbCkgJiYgT2JqZWN0LmdldE93blByb3BlcnR5TmFtZXMobW9kZWwpLmxlbmd0aCA+IDA7XG59XG5cbmZ1bmN0aW9uIGV4ZWNDbWQ8VE1lc3NhZ2U+IChjbWQ6IENtZDxUTWVzc2FnZT4sIGRpc3BhdGNoOiBEaXNwYXRjaDxUTWVzc2FnZT4pOiB2b2lkIHtcbiAgICBjbWQuZm9yRWFjaChjYWxsID0+IGNhbGwoZGlzcGF0Y2gpKTtcbn1cblxuZXhwb3J0IHtcbiAgICBsb2dNZXNzYWdlLFxuICAgIG1vZGVsSGFzQ2hhbmdlZCxcbiAgICBleGVjQ21kLFxufTsiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUNBOztBQUdBLFNBQVNBLFVBQVQsQ0FBbURDLElBQW5ELEVBQWlFQyxHQUFqRSxFQUFzRjtFQUFBOztFQUNsRixvQkFBQUMsY0FBQSxDQUFTQyxNQUFULHNFQUFpQkMsSUFBakIsQ0FBc0IsS0FBdEIsRUFBNkIsY0FBN0IsRUFBNkNKLElBQTdDLEVBQW1EQyxHQUFHLENBQUNELElBQXZEO0VBQ0EscUJBQUFFLGNBQUEsQ0FBU0MsTUFBVCx3RUFBaUJFLEtBQWpCLENBQXVCLEtBQXZCLEVBQThCLGNBQTlCLEVBQThDTCxJQUE5QyxFQUFvREMsR0FBcEQ7RUFFQSx5QkFBQUMsY0FBQSxDQUFTSSxrQkFBVCxxRkFBQUosY0FBQSxFQUE4QkQsR0FBOUI7QUFDSDs7QUFFRCxTQUFTTSxlQUFULENBQWtDQyxZQUFsQyxFQUF3REMsS0FBeEQsRUFBeUY7RUFDckYsT0FBTyxDQUFDQyxNQUFNLENBQUNDLEVBQVAsQ0FBVUYsS0FBVixFQUFpQkQsWUFBakIsQ0FBRCxJQUFtQ0UsTUFBTSxDQUFDRSxtQkFBUCxDQUEyQkgsS0FBM0IsRUFBa0NJLE1BQWxDLEdBQTJDLENBQXJGO0FBQ0g7O0FBRUQsU0FBU0MsT0FBVCxDQUE0QkMsR0FBNUIsRUFBZ0RDLFFBQWhELEVBQW9GO0VBQ2hGRCxHQUFHLENBQUNFLE9BQUosQ0FBWSxVQUFBQyxJQUFJO0lBQUEsT0FBSUEsSUFBSSxDQUFDRixRQUFELENBQVI7RUFBQSxDQUFoQjtBQUNIIn0=
@@ -107,24 +107,18 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
107
107
  while (nextMsg) {
108
108
  (0, _Common.logMessage)(_this.componentName, nextMsg);
109
109
 
110
- try {
111
- var _this$update = _this.update(_this.currentModel, nextMsg, _this.props),
112
- _this$update2 = _slicedToArray(_this$update, 2),
113
- model = _this$update2[0],
114
- cmd = _this$update2[1];
115
-
116
- if ((0, _Common.modelHasChanged)(_this.currentModel, model)) {
117
- _this.currentModel = _objectSpread(_objectSpread({}, _this.currentModel), model);
118
- modified = true;
119
- }
120
-
121
- if (cmd) {
122
- (0, _Common.execCmd)(cmd, _this.dispatch);
123
- }
124
- } catch (ex) {
125
- var _Services$logger;
126
-
127
- (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.error(ex);
110
+ var _this$update = _this.update(_this.currentModel, nextMsg, _this.props),
111
+ _this$update2 = _slicedToArray(_this$update, 2),
112
+ model = _this$update2[0],
113
+ cmd = _this$update2[1];
114
+
115
+ if ((0, _Common.modelHasChanged)(_this.currentModel, model)) {
116
+ _this.currentModel = _objectSpread(_objectSpread({}, _this.currentModel), model);
117
+ modified = true;
118
+ }
119
+
120
+ if (cmd) {
121
+ (0, _Common.execCmd)(cmd, _this.dispatch);
128
122
  }
129
123
 
130
124
  nextMsg = _this.buffer.shift();
@@ -133,9 +127,9 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
133
127
  _this.reentered = false;
134
128
 
135
129
  if (_this.mounted && modified) {
136
- var _Services$logger2;
130
+ var _Services$logger;
137
131
 
138
- (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "update model for", _this.componentName, _this.currentModel);
132
+ (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.debug("Elm", "update model for", _this.componentName, _this.currentModel);
139
133
 
140
134
  _this.forceUpdate();
141
135
  }
@@ -212,4 +206,4 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
212
206
  }(_react["default"].Component);
213
207
 
214
208
  exports.ElmComponent = ElmComponent;
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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"}
209
+ //# 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","shift","mounted","Services","logger","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                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\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":";;;;;;;;;AAAA;;AAEA;;AACA;;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;IACeA,Y;;;;;EAQX;AACJ;AACA;AACA;AACA;AACA;AACA;EACI,sBAAoBC,KAApB,EAAmCC,IAAnC,EAAiFC,IAAjF,EAA+F;IAAA;;IAAA;;IAC3F,0BAAMF,KAAN;;IAD2F;;IAAA;;IAAA,yDAZxD,EAYwD;;IAAA,4DAX3E,KAW2E;;IAAA,0DAV7E,KAU6E;;IAAA;;IAAA,2DAsDpE,UAACG,GAAD,EAAyB;MAChD,IAAI,MAAKC,SAAT,EAAoB;QAChB,MAAKC,MAAL,CAAYC,IAAZ,CAAiBH,GAAjB;MACH,CAFD,MAEO;QACH,MAAKC,SAAL,GAAiB,IAAjB;QAEA,IAAIG,OAA6B,GAAGJ,GAApC;QACA,IAAIK,QAAQ,GAAG,KAAf;;QAEA,OAAOD,OAAP,EAAgB;UACZ,IAAAE,kBAAA,EAAW,MAAKC,aAAhB,EAA+BH,OAA/B;;UAEA,mBAAqB,MAAKI,MAAL,CAAY,MAAKC,YAAjB,EAA+BL,OAA/B,EAAwC,MAAKP,KAA7C,CAArB;UAAA;UAAA,IAAOa,KAAP;UAAA,IAAcC,GAAd;;UAEA,IAAI,IAAAC,uBAAA,EAAgB,MAAKH,YAArB,EAAmCC,KAAnC,CAAJ,EAA+C;YAC3C,MAAKD,YAAL,mCAAyB,MAAKA,YAA9B,GAA+CC,KAA/C;YACAL,QAAQ,GAAG,IAAX;UACH;;UAED,IAAIM,GAAJ,EAAS;YACL,IAAAE,eAAA,EAAQF,GAAR,EAAa,MAAKG,QAAlB;UACH;;UAEDV,OAAO,GAAG,MAAKF,MAAL,CAAYa,KAAZ,EAAV;QACH;;QACD,MAAKd,SAAL,GAAiB,KAAjB;;QAEA,IAAI,MAAKe,OAAL,IAAgBX,QAApB,EAA8B;UAAA;;UAC1B,oBAAAY,cAAA,CAASC,MAAT,sEAAiBC,KAAjB,CAAuB,KAAvB,EAA8B,kBAA9B,EAAkD,MAAKZ,aAAvD,EAAsE,MAAKE,YAA3E;;UACA,MAAKW,WAAL;QACH;MACJ;IACJ,CAtF8F;;IAAA;;IAG3F,IAAMC,WAAW,GAAG,IAAAC,+BAAA,GAApB;;IAEA,IAAID,WAAJ,aAAIA,WAAJ,eAAIA,WAAW,CAAEP,QAAjB,EAA2B;MACvB,MAAKA,QAAL,GAAgBO,WAAW,CAACP,QAA5B;IACH;;IAED,WAAqBO,WAAW,SAAX,IAAAA,WAAW,WAAX,IAAAA,WAAW,CAAEX,KAAb,GAAqB,CAACW,WAAW,CAACX,KAAb,CAArB,GAAqDZ,IAAI,CAAC,MAAKD,KAAN,CAA9E;IAAA;IAAA,IAAOa,MAAP;IAAA,IAAcC,IAAd;;IAEA,MAAKJ,aAAL,GAAqBR,IAArB;IACA,MAAKU,YAAL,GAAoBC,MAApB;IACA,MAAKa,OAAL,GAAeZ,IAAf;IAb2F;EAc9F;EAED;AACJ;AACA;AACA;AACA;;;;;WACI,6BAAkC;MAC9B,KAAKK,OAAL,GAAe,IAAf;;MAEA,IAAI,KAAKO,OAAT,EAAkB;QACd,IAAAV,eAAA,EAAQ,KAAKU,OAAb,EAAsB,KAAKT,QAA3B;QACA,KAAKS,OAAL,GAAe,IAAf;MACH;IACJ;IAED;AACJ;AACA;AACA;AACA;;;;WACI,gCAAqC;MACjC,KAAKP,OAAL,GAAe,KAAf;IACH;IAED;AACJ;AACA;AACA;AACA;AACA;;;;SACI,eAAsC;MAClC,OAAO,KAAKP,YAAZ;IACH;IAED;AACJ;AACA;AACA;AACA;;;;;EApE8Ee,iBAAA,CAAMC,S"}
package/dist/useElmish.js CHANGED
@@ -60,6 +60,7 @@ function useElmish(_ref) {
60
60
  var propsRef = (0, _react.useRef)(props);
61
61
  var isMountedRef = (0, _react.useRef)(true);
62
62
  (0, _react.useEffect)(function () {
63
+ isMountedRef.current = true;
63
64
  return function () {
64
65
  isMountedRef.current = false;
65
66
  };
@@ -86,24 +87,18 @@ function useElmish(_ref) {
86
87
  while (nextMsg) {
87
88
  (0, _Common.logMessage)(name, nextMsg);
88
89
 
89
- try {
90
- 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
-
95
- if ((0, _Common.modelHasChanged)(currentModel, newModel)) {
96
- currentModel = _objectSpread(_objectSpread({}, currentModel), newModel);
97
- modified = true;
98
- }
99
-
100
- if (cmd) {
101
- (0, _Common.execCmd)(cmd, dispatch);
102
- }
103
- } catch (ex) {
104
- var _Services$logger;
90
+ 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
+
95
+ if ((0, _Common.modelHasChanged)(currentModel, newModel)) {
96
+ currentModel = _objectSpread(_objectSpread({}, currentModel), newModel);
97
+ modified = true;
98
+ }
105
99
 
106
- (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.error(ex);
100
+ if (cmd) {
101
+ (0, _Common.execCmd)(cmd, dispatch);
107
102
  }
108
103
 
109
104
  nextMsg = buffer.shift();
@@ -113,11 +108,11 @@ function useElmish(_ref) {
113
108
 
114
109
  if (isMountedRef.current && modified) {
115
110
  setModel(function (prevModel) {
116
- var _Services$logger2;
111
+ var _Services$logger;
117
112
 
118
113
  var updatedModel = _objectSpread(_objectSpread({}, prevModel), currentModel);
119
114
 
120
- (_Services$logger2 = _Init.Services.logger) === null || _Services$logger2 === void 0 ? void 0 : _Services$logger2.debug("Elm", "update model for", name, updatedModel);
115
+ (_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.debug("Elm", "update model for", name, updatedModel);
121
116
  return updatedModel;
122
117
  });
123
118
  }
@@ -169,4 +164,4 @@ function callUpdateMap(updateMap, msg, model, props) {
169
164
  var updateFn = updateMap[msg.name];
170
165
  return updateFn(msg, model, props);
171
166
  }
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":"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"}
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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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "react-elmish",
3
- "version": "4.4.0",
3
+ "version": "5.0.1",
4
4
  "description": "Elmish for React using Typescript",
5
5
  "author": "atheck",
6
6
  "license": "MIT",
@@ -19,16 +19,16 @@
19
19
  },
20
20
  "devDependencies": {
21
21
  "@babel/cli": "7.18.10",
22
- "@babel/core": "7.18.13",
22
+ "@babel/core": "7.19.0",
23
23
  "@babel/plugin-proposal-class-properties": "7.18.6",
24
- "@babel/preset-env": "7.18.10",
24
+ "@babel/preset-env": "7.19.0",
25
25
  "@babel/preset-react": "7.18.6",
26
26
  "@babel/preset-typescript": "7.18.6",
27
- "@testing-library/react": "13.3.0",
27
+ "@testing-library/react": "13.4.0",
28
28
  "@types/jest": "28.1.8",
29
- "@types/react": "18.0.17",
30
- "eslint": "8.22.0",
31
- "eslint-config-heck": "1.22.3",
29
+ "@types/react": "18.0.18",
30
+ "eslint": "8.23.0",
31
+ "eslint-config-heck": "1.23.0",
32
32
  "jest": "28.1.3",
33
33
  "jest-environment-jsdom": "28.1.3",
34
34
  "semantic-release": "19.0.5",