react-elmish 4.3.0 → 5.0.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.
- package/dist/Common.d.ts +6 -0
- package/dist/Common.js +29 -0
- package/dist/ElmComponent.d.ts +0 -1
- package/dist/ElmComponent.js +16 -46
- package/dist/useElmish.js +25 -45
- package/package.json +8 -8
package/dist/Common.d.ts
ADDED
|
@@ -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,29 @@
|
|
|
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
|
+
return call(dispatch);
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJsb2dNZXNzYWdlIiwibmFtZSIsIm1zZyIsIlNlcnZpY2VzIiwibG9nZ2VyIiwiaW5mbyIsImRlYnVnIiwiZGlzcGF0Y2hNaWRkbGV3YXJlIiwibW9kZWxIYXNDaGFuZ2VkIiwiY3VycmVudE1vZGVsIiwibW9kZWwiLCJPYmplY3QiLCJpcyIsImdldE93blByb3BlcnR5TmFtZXMiLCJsZW5ndGgiLCJleGVjQ21kIiwiY21kIiwiZGlzcGF0Y2giLCJmb3JFYWNoIiwiY2FsbCJdLCJzb3VyY2VzIjpbIi4uL3NyYy9Db21tb24udHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgU2VydmljZXMgfSBmcm9tIFwiLi9Jbml0XCI7XG5pbXBvcnQgeyBNZXNzYWdlQmFzZSB9IGZyb20gXCIuL1R5cGVzXCI7XG5cbmZ1bmN0aW9uIGxvZ01lc3NhZ2U8VE1lc3NhZ2UgZXh0ZW5kcyBNZXNzYWdlQmFzZT4gKG5hbWU6IHN0cmluZywgbXNnOiBUTWVzc2FnZSk6IHZvaWQge1xuICAgIFNlcnZpY2VzLmxvZ2dlcj8uaW5mbyhcIkVsbVwiLCBcIm1lc3NhZ2UgZnJvbVwiLCBuYW1lLCBtc2cubmFtZSk7XG4gICAgU2VydmljZXMubG9nZ2VyPy5kZWJ1ZyhcIkVsbVwiLCBcIm1lc3NhZ2UgZnJvbVwiLCBuYW1lLCBtc2cpO1xuXG4gICAgU2VydmljZXMuZGlzcGF0Y2hNaWRkbGV3YXJlPy4obXNnKTtcbn1cblxuZnVuY3Rpb24gbW9kZWxIYXNDaGFuZ2VkPFRNb2RlbD4gKGN1cnJlbnRNb2RlbDogVE1vZGVsLCBtb2RlbDogUGFydGlhbDxUTW9kZWw+KTogYm9vbGVhbiB7XG4gICAgcmV0dXJuICFPYmplY3QuaXMobW9kZWwsIGN1cnJlbnRNb2RlbCkgJiYgT2JqZWN0LmdldE93blByb3BlcnR5TmFtZXMobW9kZWwpLmxlbmd0aCA+IDA7XG59XG5cbmZ1bmN0aW9uIGV4ZWNDbWQ8VE1lc3NhZ2U+IChjbWQ6IENtZDxUTWVzc2FnZT4sIGRpc3BhdGNoOiBEaXNwYXRjaDxUTWVzc2FnZT4pOiB2b2lkIHtcbiAgICBjbWQuZm9yRWFjaChjYWxsID0+IGNhbGwoZGlzcGF0Y2gpKTtcbn1cblxuZXhwb3J0IHtcbiAgICBsb2dNZXNzYWdlLFxuICAgIG1vZGVsSGFzQ2hhbmdlZCxcbiAgICBleGVjQ21kLFxufTsiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUNBOztBQUdBLFNBQVNBLFVBQVQsQ0FBbURDLElBQW5ELEVBQWlFQyxHQUFqRSxFQUFzRjtFQUFBOztFQUNsRixvQkFBQUMsY0FBQSxDQUFTQyxNQUFULHNFQUFpQkMsSUFBakIsQ0FBc0IsS0FBdEIsRUFBNkIsY0FBN0IsRUFBNkNKLElBQTdDLEVBQW1EQyxHQUFHLENBQUNELElBQXZEO0VBQ0EscUJBQUFFLGNBQUEsQ0FBU0MsTUFBVCx3RUFBaUJFLEtBQWpCLENBQXVCLEtBQXZCLEVBQThCLGNBQTlCLEVBQThDTCxJQUE5QyxFQUFvREMsR0FBcEQ7RUFFQSx5QkFBQUMsY0FBQSxDQUFTSSxrQkFBVCxxRkFBQUosY0FBQSxFQUE4QkQsR0FBOUI7QUFDSDs7QUFFRCxTQUFTTSxlQUFULENBQWtDQyxZQUFsQyxFQUF3REMsS0FBeEQsRUFBeUY7RUFDckYsT0FBTyxDQUFDQyxNQUFNLENBQUNDLEVBQVAsQ0FBVUYsS0FBVixFQUFpQkQsWUFBakIsQ0FBRCxJQUFtQ0UsTUFBTSxDQUFDRSxtQkFBUCxDQUEyQkgsS0FBM0IsRUFBa0NJLE1BQWxDLEdBQTJDLENBQXJGO0FBQ0g7O0FBRUQsU0FBU0MsT0FBVCxDQUE0QkMsR0FBNUIsRUFBZ0RDLFFBQWhELEVBQW9GO0VBQ2hGRCxHQUFHLENBQUNFLE9BQUosQ0FBWSxVQUFBQyxJQUFJO0lBQUEsT0FBSUEsSUFBSSxDQUFDRixRQUFELENBQVI7RUFBQSxDQUFoQjtBQUNIIn0=
|
package/dist/ElmComponent.d.ts
CHANGED
package/dist/ElmComponent.js
CHANGED
|
@@ -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,33 +105,20 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
107
105
|
var modified = false;
|
|
108
106
|
|
|
109
107
|
while (nextMsg) {
|
|
110
|
-
|
|
108
|
+
(0, _Common.logMessage)(_this.componentName, nextMsg);
|
|
111
109
|
|
|
112
|
-
|
|
113
|
-
|
|
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
114
|
|
|
115
|
-
if (
|
|
116
|
-
|
|
115
|
+
if ((0, _Common.modelHasChanged)(_this.currentModel, model)) {
|
|
116
|
+
_this.currentModel = _objectSpread(_objectSpread({}, _this.currentModel), model);
|
|
117
|
+
modified = true;
|
|
117
118
|
}
|
|
118
119
|
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
_this$update2 = _slicedToArray(_this$update, 2),
|
|
122
|
-
model = _this$update2[0],
|
|
123
|
-
cmd = _this$update2[1];
|
|
124
|
-
|
|
125
|
-
if (modelHasChanged(model)) {
|
|
126
|
-
_this.currentModel = _objectSpread(_objectSpread({}, _this.currentModel), model);
|
|
127
|
-
modified = true;
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
if (cmd) {
|
|
131
|
-
_this.execCmd(cmd);
|
|
132
|
-
}
|
|
133
|
-
} catch (ex) {
|
|
134
|
-
var _Services$logger3;
|
|
135
|
-
|
|
136
|
-
(_Services$logger3 = _Init.Services.logger) === null || _Services$logger3 === void 0 ? void 0 : _Services$logger3.error(ex);
|
|
120
|
+
if (cmd) {
|
|
121
|
+
(0, _Common.execCmd)(cmd, _this.dispatch);
|
|
137
122
|
}
|
|
138
123
|
|
|
139
124
|
nextMsg = _this.buffer.shift();
|
|
@@ -142,9 +127,9 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
142
127
|
_this.reentered = false;
|
|
143
128
|
|
|
144
129
|
if (_this.mounted && modified) {
|
|
145
|
-
var _Services$
|
|
130
|
+
var _Services$logger;
|
|
146
131
|
|
|
147
|
-
(_Services$
|
|
132
|
+
(_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.debug("Elm", "update model for", _this.componentName, _this.currentModel);
|
|
148
133
|
|
|
149
134
|
_this.forceUpdate();
|
|
150
135
|
}
|
|
@@ -182,7 +167,7 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
182
167
|
this.mounted = true;
|
|
183
168
|
|
|
184
169
|
if (this.initCmd) {
|
|
185
|
-
|
|
170
|
+
(0, _Common.execCmd)(this.initCmd, this.dispatch);
|
|
186
171
|
this.initCmd = null;
|
|
187
172
|
}
|
|
188
173
|
}
|
|
@@ -197,21 +182,6 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
197
182
|
value: function componentWillUnmount() {
|
|
198
183
|
this.mounted = false;
|
|
199
184
|
}
|
|
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
185
|
/**
|
|
216
186
|
* Returns the current model.
|
|
217
187
|
* @readonly
|
|
@@ -236,4 +206,4 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
236
206
|
}(_react["default"].Component);
|
|
237
207
|
|
|
238
208
|
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":";;;;;;;;;AAAA;;AAEA;;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,2DAgEpE,UAACG,GAAD,EAAyB;MAChD,IAAMC,eAAe,GAAG,SAAlBA,eAAkB,CAACC,KAAD;QAAA,OAAqCA,KAAK,KAAK,MAAKC,YAAf,IAA+BC,MAAM,CAACC,mBAAP,CAA2BH,KAA3B,EAAkCI,MAAlC,GAA2C,CAA/G;MAAA,CAAxB;;MAEA,IAAI,MAAKC,SAAT,EAAoB;QAChB,MAAKC,MAAL,CAAYC,IAAZ,CAAiBT,GAAjB;MACH,CAFD,MAEO;QACH,MAAKO,SAAL,GAAiB,IAAjB;QAEA,IAAIG,OAA6B,GAAGV,GAApC;QACA,IAAIW,QAAQ,GAAG,KAAf;;QAEA,OAAOD,OAAP,EAAgB;UAAA;;UACZ,oBAAAE,cAAA,CAASC,MAAT,sEAAiBC,IAAjB,CAAsB,KAAtB,EAA6B,cAA7B,EAA6C,MAAKC,aAAlD,EAAiEL,OAAO,CAACX,IAAzE;UACA,qBAAAa,cAAA,CAASC,MAAT,wEAAiBG,KAAjB,CAAuB,KAAvB,EAA8B,cAA9B,EAA8C,MAAKD,aAAnD,EAAkEL,OAAlE;;UAEA,IAAIE,cAAA,CAASK,kBAAb,EAAiC;YAC7BL,cAAA,CAASK,kBAAT,CAA4BP,OAA5B;UACH;;UAED,IAAI;YACA,mBAAqB,MAAKQ,MAAL,CAAY,MAAKf,YAAjB,EAA+BO,OAA/B,EAAwC,MAAKb,KAA7C,CAArB;YAAA;YAAA,IAAOK,KAAP;YAAA,IAAciB,GAAd;;YAEA,IAAIlB,eAAe,CAACC,KAAD,CAAnB,EAA4B;cACxB,MAAKC,YAAL,mCAAyB,MAAKA,YAA9B,GAA+CD,KAA/C;cACAS,QAAQ,GAAG,IAAX;YACH;;YAED,IAAIQ,GAAJ,EAAS;cACL,MAAKC,OAAL,CAAaD,GAAb;YACH;UACJ,CAXD,CAWE,OAAOE,EAAP,EAAoB;YAAA;;YAClB,qBAAAT,cAAA,CAASC,MAAT,wEAAiBS,KAAjB,CAAuBD,EAAvB;UACH;;UAEDX,OAAO,GAAG,MAAKF,MAAL,CAAYe,KAAZ,EAAV;QACH;;QACD,MAAKhB,SAAL,GAAiB,KAAjB;;QAEA,IAAI,MAAKiB,OAAL,IAAgBb,QAApB,EAA8B;UAAA;;UAC1B,qBAAAC,cAAA,CAASC,MAAT,wEAAiBG,KAAjB,CAAuB,KAAvB,EAA8B,kBAA9B,EAAkD,MAAKD,aAAvD,EAAsE,MAAKZ,YAA3E;;UACA,MAAKsB,WAAL;QACH;MACJ;IACJ,CA3G8F;;IAAA;;IAG3F,IAAMC,WAAW,GAAG,IAAAC,+BAAA,GAApB;;IAEA,IAAID,WAAJ,aAAIA,WAAJ,eAAIA,WAAW,CAAEE,QAAjB,EAA2B;MACvB,MAAKA,QAAL,GAAgBF,WAAW,CAACE,QAA5B;IACH;;IAED,WAAqBF,WAAW,SAAX,IAAAA,WAAW,WAAX,IAAAA,WAAW,CAAExB,KAAb,GAAqB,CAACwB,WAAW,CAACxB,KAAb,CAArB,GAAqDJ,IAAI,CAAC,MAAKD,KAAN,CAA9E;IAAA;IAAA,IAAOK,MAAP;IAAA,IAAciB,IAAd;;IAEA,MAAKJ,aAAL,GAAqBhB,IAArB;IACA,MAAKI,YAAL,GAAoBD,MAApB;IACA,MAAK2B,OAAL,GAAeV,IAAf;IAb2F;EAc9F;EAED;AACJ;AACA;AACA;AACA;;;;;WACI,6BAAkC;MAC9B,KAAKK,OAAL,GAAe,IAAf;;MAEA,IAAI,KAAKK,OAAT,EAAkB;QACd,KAAKT,OAAL,CAAa,KAAKS,OAAlB;QACA,KAAKA,OAAL,GAAe,IAAf;MACH;IACJ;IAED;AACJ;AACA;AACA;AACA;;;;WACI,gCAAqC;MACjC,KAAKL,OAAL,GAAe,KAAf;IACH;;;WAED,iBAAiBL,GAAjB,EAA2C;MAAA;;MACvCA,GAAG,CAACW,OAAJ,CAAY,UAAAC,IAAI,EAAI;QAChB,IAAI;UACAA,IAAI,CAAC,MAAI,CAACH,QAAN,CAAJ;QACH,CAFD,CAEE,OAAOP,EAAP,EAAoB;UAAA;;UAClB,qBAAAT,cAAA,CAASC,MAAT,wEAAiBS,KAAjB,CAAuBD,EAAvB;QACH;MACJ,CAND;IAOH;IAED;AACJ;AACA;AACA;AACA;AACA;;;;SACI,eAAsC;MAClC,OAAO,KAAKlB,YAAZ;IACH;IAED;AACJ;AACA;AACA;AACA;;;;;EA9E8E6B,iBAAA,CAAMC,S"}
|
|
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
|
@@ -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,33 +84,20 @@ function useElmish(_ref) {
|
|
|
91
84
|
var modified = false;
|
|
92
85
|
|
|
93
86
|
while (nextMsg) {
|
|
94
|
-
|
|
87
|
+
(0, _Common.logMessage)(name, nextMsg);
|
|
95
88
|
|
|
96
|
-
|
|
97
|
-
|
|
89
|
+
var _callUpdate = callUpdate(update, nextMsg, _objectSpread(_objectSpread({}, initializedModel), currentModel), propsRef.current),
|
|
90
|
+
_callUpdate2 = _slicedToArray(_callUpdate, 2),
|
|
91
|
+
newModel = _callUpdate2[0],
|
|
92
|
+
cmd = _callUpdate2[1];
|
|
98
93
|
|
|
99
|
-
if (
|
|
100
|
-
|
|
94
|
+
if ((0, _Common.modelHasChanged)(currentModel, newModel)) {
|
|
95
|
+
currentModel = _objectSpread(_objectSpread({}, currentModel), newModel);
|
|
96
|
+
modified = true;
|
|
101
97
|
}
|
|
102
98
|
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
_callUpdate2 = _slicedToArray(_callUpdate, 2),
|
|
106
|
-
newModel = _callUpdate2[0],
|
|
107
|
-
cmd = _callUpdate2[1];
|
|
108
|
-
|
|
109
|
-
if (modelHasChanged(newModel)) {
|
|
110
|
-
currentModel = _objectSpread(_objectSpread({}, currentModel), newModel);
|
|
111
|
-
modified = true;
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
if (cmd) {
|
|
115
|
-
execCmd(cmd);
|
|
116
|
-
}
|
|
117
|
-
} catch (ex) {
|
|
118
|
-
var _Services$logger4;
|
|
119
|
-
|
|
120
|
-
(_Services$logger4 = _Init.Services.logger) === null || _Services$logger4 === void 0 ? void 0 : _Services$logger4.error(ex);
|
|
99
|
+
if (cmd) {
|
|
100
|
+
(0, _Common.execCmd)(cmd, dispatch);
|
|
121
101
|
}
|
|
122
102
|
|
|
123
103
|
nextMsg = buffer.shift();
|
|
@@ -125,13 +105,13 @@ function useElmish(_ref) {
|
|
|
125
105
|
|
|
126
106
|
reentered = false;
|
|
127
107
|
|
|
128
|
-
if (modified) {
|
|
108
|
+
if (isMountedRef.current && modified) {
|
|
129
109
|
setModel(function (prevModel) {
|
|
130
|
-
var _Services$
|
|
110
|
+
var _Services$logger;
|
|
131
111
|
|
|
132
112
|
var updatedModel = _objectSpread(_objectSpread({}, prevModel), currentModel);
|
|
133
113
|
|
|
134
|
-
(_Services$
|
|
114
|
+
(_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.debug("Elm", "update model for", name, updatedModel);
|
|
135
115
|
return updatedModel;
|
|
136
116
|
});
|
|
137
117
|
}
|
|
@@ -148,7 +128,7 @@ function useElmish(_ref) {
|
|
|
148
128
|
setModel(initializedModel);
|
|
149
129
|
|
|
150
130
|
if (initCmd) {
|
|
151
|
-
execCmd(initCmd);
|
|
131
|
+
(0, _Common.execCmd)(initCmd, dispatch);
|
|
152
132
|
}
|
|
153
133
|
}
|
|
154
134
|
|
|
@@ -159,7 +139,7 @@ function useElmish(_ref) {
|
|
|
159
139
|
subCmd = _subscription2[0],
|
|
160
140
|
destructor = _subscription2[1];
|
|
161
141
|
|
|
162
|
-
execCmd(subCmd);
|
|
142
|
+
(0, _Common.execCmd)(subCmd, dispatch);
|
|
163
143
|
|
|
164
144
|
if (destructor) {
|
|
165
145
|
return destructor;
|
|
@@ -183,4 +163,4 @@ function callUpdateMap(updateMap, msg, model, props) {
|
|
|
183
163
|
var updateFn = updateMap[msg.name];
|
|
184
164
|
return updateFn(msg, model, props);
|
|
185
165
|
}
|
|
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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"}
|
|
166
|
+
//# 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 = 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                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":";;;;;;;;;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,kBAAwBG,UAAU,CAACvB,MAAD,EAASoB,OAAT,kCAAuBP,gBAAvB,GAA4CT,YAA5C,GAA4DI,QAAQ,CAACI,OAArE,CAAlC;QAAA;QAAA,IAAOY,QAAP;QAAA,IAAiBC,GAAjB;;QAEA,IAAI,IAAAC,uBAAA,EAAgBtB,YAAhB,EAA8BoB,QAA9B,CAAJ,EAA6C;UACzCpB,YAAY,mCAAQA,YAAR,GAAyBoB,QAAzB,CAAZ;UAEAH,QAAQ,GAAG,IAAX;QACH;;QAED,IAAII,GAAJ,EAAS;UACL,IAAAE,eAAA,EAAQF,GAAR,EAAaT,QAAb;QACH;;QAEDI,OAAO,GAAGjB,MAAM,CAACyB,KAAP,EAAV;MACH;;MACD1B,SAAS,GAAG,KAAZ;;MAEA,IAAIQ,YAAY,CAACE,OAAb,IAAwBS,QAA5B,EAAsC;QAClCd,QAAQ,CAAC,UAAAsB,SAAS,EAAI;UAAA;;UAClB,IAAMC,YAAY,mCAAQD,SAAR,GAAgCzB,YAAhC,CAAlB;;UAEA,oBAAA2B,cAAA,CAASC,MAAT,sEAAiBC,KAAjB,CAAuB,KAAvB,EAA8B,kBAA9B,EAAkDpC,IAAlD,EAAwDiC,YAAxD;UAEA,OAAOA,YAAP;QACH,CANO,CAAR;MAOH;IACJ;EACJ,CA1CgB,EA0Cb,EA1Ca,CAAjB;;EA4CA,IAAI,CAACjB,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,IAAOoC,SAAP;IAAA,IAAkBC,OAAlB;;IAEAtB,gBAAgB,GAAGqB,SAAnB;IACA3B,QAAQ,CAACM,gBAAD,CAAR;;IAEA,IAAIsB,OAAJ,EAAa;MACT,IAAAR,eAAA,EAAQQ,OAAR,EAAiBnB,QAAjB;IACH;EACJ;;EAED,IAAAL,gBAAA,EAAU,YAAM;IACZ,IAAIV,YAAJ,EAAkB;MACd,oBAA6BA,YAAY,CAACY,gBAAD,EAA6Bf,KAA7B,CAAzC;MAAA;MAAA,IAAOsC,MAAP;MAAA,IAAeC,UAAf;;MAEA,IAAAV,eAAA,EAAQS,MAAR,EAAgBpB,QAAhB;;MAEA,IAAIqB,UAAJ,EAAgB;QACZ,OAAOA,UAAP;MACH;IACJ;EACJ,CAVD,EAUG,EAVH;EAYA,OAAO,CAACxB,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,OAAOwC,aAAa,CAACtC,MAAD,EAASkB,GAAT,EAAcZ,KAAd,EAAqBR,KAArB,CAApB;AACH;;AAED,SAASwC,aAAT,CAAsEC,SAAtE,EAAsHrB,GAAtH,EAAqIZ,KAArI,EAAoJR,KAApJ,EAAuM;EACnM;EACA;EACA,IAAM0C,QAAQ,GAAGD,SAAS,CAACrB,GAAG,CAACrB,IAAL,CAA1B;EAEA,OAAO2C,QAAQ,CAACtB,GAAD,EAAMZ,KAAN,EAAaR,KAAb,CAAf;AACH"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "react-elmish",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "5.0.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.
|
|
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.
|
|
29
|
-
"@types/react": "18.0.
|
|
30
|
-
"eslint": "8.
|
|
31
|
-
"eslint-config-heck": "1.22.
|
|
28
|
+
"@types/jest": "28.1.8",
|
|
29
|
+
"@types/react": "18.0.18",
|
|
30
|
+
"eslint": "8.23.0",
|
|
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.
|
|
34
|
+
"semantic-release": "19.0.5",
|
|
35
35
|
"ts-jest": "28.0.8",
|
|
36
|
-
"typescript": "4.
|
|
36
|
+
"typescript": "4.8.2"
|
|
37
37
|
},
|
|
38
38
|
"homepage": "https://github.com/atheck/react-elmish",
|
|
39
39
|
"repository": {
|