react-elmish 4.2.0-beta.4 → 4.2.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/.eslintrc +1 -1
- package/.vscode/settings.json +3 -0
- package/README.md +86 -13
- package/dist/Cmd.d.ts +19 -18
- package/dist/Cmd.js +3 -3
- package/dist/ElmComponent.d.ts +10 -10
- package/dist/ElmComponent.js +14 -10
- package/dist/ErrorHandling.d.ts +1 -1
- package/dist/ErrorHandling.js +1 -1
- package/dist/Init.d.ts +17 -0
- package/dist/Init.js +7 -1
- package/dist/Testing/fakeOptions.d.ts +24 -0
- package/dist/Testing/fakeOptions.js +26 -0
- package/dist/Testing/index.d.ts +17 -7
- package/dist/Testing/index.js +23 -6
- package/dist/Types.d.ts +8 -5
- package/dist/Types.js +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +3 -3
- package/dist/useElmish.d.ts +31 -0
- package/dist/useElmish.js +8 -8
- package/package.json +7 -7
- package/dist/Testing/fakeInitResult.d.ts +0 -7
- package/dist/Testing/fakeInitResult.js +0 -28
package/.eslintrc
CHANGED
package/README.md
CHANGED
|
@@ -5,6 +5,35 @@
|
|
|
5
5
|
|
|
6
6
|
This library brings the elmish pattern to react.
|
|
7
7
|
|
|
8
|
+
- [Installation](#installation)
|
|
9
|
+
- [Basic Usage](#basic-usage)
|
|
10
|
+
- [More on messages](#more-on-messages)
|
|
11
|
+
- [Message arguments](#message-arguments)
|
|
12
|
+
- [Symbols instead of strings](#symbols-instead-of-strings)
|
|
13
|
+
- [Dispatch commands in the update map or update function](#dispatch-commands-in-the-update-map-or-update-function)
|
|
14
|
+
- [Dispatch a message](#dispatch-a-message)
|
|
15
|
+
- [Call an async function](#call-an-async-function)
|
|
16
|
+
- [Dispatch a command from `init`](#dispatch-a-command-from-init)
|
|
17
|
+
- [Subscriptions](#subscriptions)
|
|
18
|
+
- [Working with external sources of events](#working-with-external-sources-of-events)
|
|
19
|
+
- [Cleanup subscriptions](#cleanup-subscriptions)
|
|
20
|
+
- [Setup](#setup)
|
|
21
|
+
- [Error handling](#error-handling)
|
|
22
|
+
- [React life cycle management](#react-life-cycle-management)
|
|
23
|
+
- [Composition](#composition)
|
|
24
|
+
- [With an `UpdateMap`](#with-an-updatemap)
|
|
25
|
+
- [With an update function](#with-an-update-function)
|
|
26
|
+
- [Call back parent components](#call-back-parent-components)
|
|
27
|
+
- [Testing](#testing)
|
|
28
|
+
- [Testing the model and simple message commands](#testing-the-model-and-simple-message-commands)
|
|
29
|
+
- [Testing all (async) messages](#testing-all-async-messages)
|
|
30
|
+
- [Testing with an UpdateMap](#testing-with-an-updatemap)
|
|
31
|
+
- [UI Tests](#ui-tests)
|
|
32
|
+
- [Migrations](#migrations)
|
|
33
|
+
- [From v1.x to v2.x](#from-v1x-to-v2x)
|
|
34
|
+
- [From v2.x to v3.x](#from-v2x-to-v3x)
|
|
35
|
+
- [From v3.x to v4.x](#from-v3x-to-v4x)
|
|
36
|
+
|
|
8
37
|
## Installation
|
|
9
38
|
|
|
10
39
|
`npm install react-elmish`
|
|
@@ -13,14 +42,14 @@ This library brings the elmish pattern to react.
|
|
|
13
42
|
|
|
14
43
|
An elmish component basically consists of the following parts:
|
|
15
44
|
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
45
|
+
- The **Model** holding the state of the component.
|
|
46
|
+
- The **Props** for the component.
|
|
47
|
+
- The **Init** function to create the initial model based on the props.
|
|
48
|
+
- The **Messages** to dispatch which modify the model.
|
|
49
|
+
- The **Update** function to modify the model based on a specific message.
|
|
50
|
+
- The **View** which renders the component based on the current model.
|
|
22
51
|
|
|
23
|
-
|
|
52
|
+
**App.ts:**
|
|
24
53
|
|
|
25
54
|
First import everything from `react-elmish` and declare the **Message** discriminated union type:
|
|
26
55
|
|
|
@@ -101,7 +130,7 @@ export const update = (model: Model, msg: Msg, props: Props): UpdateReturnType<M
|
|
|
101
130
|
|
|
102
131
|
> **Note:** If you are using **typescript** and **typescript-eslint** you should enable the [switch-exhaustive-check](https://github.com/typescript-eslint/typescript-eslint/blob/master/packages/eslint-plugin/docs/rules/switch-exhaustiveness-check.md) rule.
|
|
103
132
|
|
|
104
|
-
|
|
133
|
+
**App.tsx:**
|
|
105
134
|
|
|
106
135
|
To put all this together and to render our component, we need a React component.
|
|
107
136
|
|
|
@@ -922,10 +951,54 @@ const update = getUpdateFn(updateMap);
|
|
|
922
951
|
const [model, cmd] = update(msg, model, props);
|
|
923
952
|
```
|
|
924
953
|
|
|
925
|
-
|
|
954
|
+
### UI Tests
|
|
955
|
+
|
|
956
|
+
To test UI components with a fake model you can use `renderWithModel` from the Testing namespace. The first parameter is a function to render your component (e.g. with **@testing-library/react**). The second parameter is the fake model or an options object, where you can also pass a fake `dispatch` function.
|
|
957
|
+
|
|
958
|
+
```tsx
|
|
959
|
+
import { renderWithModel } from "react-elmish/dist/Testing";
|
|
960
|
+
import { fireEvent, render, screen } from "@testing-library/react";
|
|
961
|
+
|
|
962
|
+
it("renders the correct value", () => {
|
|
963
|
+
// arrange
|
|
964
|
+
const model: Model = { value: "It works" };
|
|
965
|
+
|
|
966
|
+
// act
|
|
967
|
+
renderWithModel(() => render(<TestComponent />), model);
|
|
968
|
+
|
|
969
|
+
// assert
|
|
970
|
+
expect(screen.getByText("It works")).not.toBeNull();
|
|
971
|
+
});
|
|
972
|
+
|
|
973
|
+
it("dispatches the correct message", async () => {
|
|
974
|
+
// arrange
|
|
975
|
+
const model: Model = { value: "" };
|
|
976
|
+
const mockDispatch = jest.fn();
|
|
977
|
+
|
|
978
|
+
renderWithModel(
|
|
979
|
+
() => render(<TestComponent />),
|
|
980
|
+
{
|
|
981
|
+
model,
|
|
982
|
+
dispatch: mockDispatch
|
|
983
|
+
}
|
|
984
|
+
);
|
|
985
|
+
|
|
986
|
+
// act
|
|
987
|
+
fireEvent.click(screen.getByText("Click"));
|
|
988
|
+
|
|
989
|
+
// assert
|
|
990
|
+
expect(mockDispatch).toHaveBeenCalledWith({ name: "click" });
|
|
991
|
+
});
|
|
992
|
+
```
|
|
993
|
+
|
|
994
|
+
This works for function components using the `useElmish` hook and class components.
|
|
995
|
+
|
|
996
|
+
## Migrations
|
|
997
|
+
|
|
998
|
+
### From v1.x to v2.x
|
|
926
999
|
|
|
927
|
-
|
|
928
|
-
|
|
1000
|
+
- Use `Logger` and `Message` instead of `ILogger` and `IMessage`.
|
|
1001
|
+
- The global declaration of the `Nullable` type was removed, because it is unexpected for this library to declare such a type. You can declare this type for yourself if needed:
|
|
929
1002
|
|
|
930
1003
|
```ts
|
|
931
1004
|
declare global {
|
|
@@ -933,7 +1006,7 @@ const [model, cmd] = update(msg, model, props);
|
|
|
933
1006
|
}
|
|
934
1007
|
```
|
|
935
1008
|
|
|
936
|
-
|
|
1009
|
+
### From v2.x to v3.x
|
|
937
1010
|
|
|
938
1011
|
The signature of `useElmish` has changed. It takes an options object now. Thus there is no need for the `useElmishMap` function. Use the new `useElmish` hook with an `UpdateMap` instead.
|
|
939
1012
|
|
|
@@ -946,6 +1019,6 @@ import { useElmishMap } from "react-elmish/dist/legacy/useElmishMap";
|
|
|
946
1019
|
|
|
947
1020
|
**Notice**: These functions are marked as deprecated and will be removed in a later release.
|
|
948
1021
|
|
|
949
|
-
|
|
1022
|
+
### From v3.x to v4.x
|
|
950
1023
|
|
|
951
1024
|
Because the legacy `useElmish` and `useElmishMap` have been removed, you have to convert all usages of `useElmish` to use the parameter object.
|
package/dist/Cmd.d.ts
CHANGED
|
@@ -1,37 +1,38 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Type of the dispatch function.
|
|
3
3
|
*/
|
|
4
|
-
declare type Dispatch<
|
|
4
|
+
declare type Dispatch<TMessage> = (msg: TMessage) => void;
|
|
5
5
|
declare type FallbackHandler = (error?: Error) => void;
|
|
6
6
|
declare type Sub<TMsg> = (dispatch: Dispatch<TMsg>, fallback?: FallbackHandler) => void;
|
|
7
7
|
/**
|
|
8
8
|
* Type of a command.
|
|
9
9
|
*/
|
|
10
|
-
declare type Cmd<
|
|
10
|
+
declare type Cmd<TMessage> = Sub<TMessage>[];
|
|
11
11
|
/**
|
|
12
12
|
* Contains functions to create commands.
|
|
13
|
-
* @template
|
|
13
|
+
* @template TMessage Type of the Message discriminated union.
|
|
14
14
|
*/
|
|
15
|
-
interface Command<
|
|
15
|
+
interface Command<TMessage> {
|
|
16
16
|
/**
|
|
17
17
|
* Represents an empty command.
|
|
18
|
+
* @deprecated Do return nothing (`undefined`) instead.
|
|
18
19
|
*/
|
|
19
20
|
none: [];
|
|
20
21
|
/**
|
|
21
22
|
* Creates a command out of a specific message.
|
|
22
|
-
* @param {
|
|
23
|
+
* @param {TMessage} msg The specific message.
|
|
23
24
|
*/
|
|
24
|
-
ofMsg: (msg:
|
|
25
|
+
ofMsg: (msg: TMessage) => Cmd<TMessage>;
|
|
25
26
|
/**
|
|
26
27
|
* Aggregates multiple commands.
|
|
27
|
-
* @param {Cmd<
|
|
28
|
+
* @param {Cmd<TMessage> []} commands Array of commands.
|
|
28
29
|
*/
|
|
29
|
-
batch: (...commands: Cmd<
|
|
30
|
+
batch: (...commands: Cmd<TMessage>[]) => Cmd<TMessage>;
|
|
30
31
|
/**
|
|
31
32
|
* Command to call the subscriber.
|
|
32
|
-
* @param {Sub<
|
|
33
|
+
* @param {Sub<TMessage>} sub The subscriber function.
|
|
33
34
|
*/
|
|
34
|
-
ofSub: (sub: Sub<
|
|
35
|
+
ofSub: (sub: Sub<TMessage>) => Cmd<TMessage>;
|
|
35
36
|
/**
|
|
36
37
|
* Provides functionalities to create commands from simple functions.
|
|
37
38
|
*/
|
|
@@ -43,21 +44,21 @@ interface Command<TMsg> {
|
|
|
43
44
|
* @param ofError Creates the message to dispatch when an error occurred.
|
|
44
45
|
* @param args The parameters of the task.
|
|
45
46
|
*/
|
|
46
|
-
either: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) =>
|
|
47
|
+
either: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMessage, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>;
|
|
47
48
|
/**
|
|
48
49
|
* Creates a command out of a simple function and ignores the error case.
|
|
49
50
|
* @param task The function to call.
|
|
50
51
|
* @param ofSuccess Creates the message to dispatch after a successful call of the task.
|
|
51
52
|
* @param args The parameters of the task.
|
|
52
53
|
*/
|
|
53
|
-
perform: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) =>
|
|
54
|
+
perform: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMessage, ...args: TArgs) => Cmd<TMessage>;
|
|
54
55
|
/**
|
|
55
56
|
* Creates a command out of a simple function and ignores the success case.
|
|
56
57
|
* @param task The function to call.
|
|
57
58
|
* @param ofError Creates the message to dispatch when an error occurred.
|
|
58
59
|
* @param args The parameters of the task.
|
|
59
60
|
*/
|
|
60
|
-
attempt: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => TReturn, ofError: (error: Error) =>
|
|
61
|
+
attempt: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => TReturn, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>;
|
|
61
62
|
};
|
|
62
63
|
/**
|
|
63
64
|
* Provides functionalities to create commands from async functions.
|
|
@@ -70,27 +71,27 @@ interface Command<TMsg> {
|
|
|
70
71
|
* @param ofError Creates the message to dispatch when the promise is rejected.
|
|
71
72
|
* @param args The parameters of the task.
|
|
72
73
|
*/
|
|
73
|
-
either: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) =>
|
|
74
|
+
either: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMessage, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>;
|
|
74
75
|
/**
|
|
75
76
|
* Creates a command out of an async function and ignores the error case.
|
|
76
77
|
* @param task The async function to call.
|
|
77
78
|
* @param ofSuccess Creates the message to dispatch when the promise is resolved.
|
|
78
79
|
* @param args The parameters of the task.
|
|
79
80
|
*/
|
|
80
|
-
perform: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) =>
|
|
81
|
+
perform: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMessage, ...args: TArgs) => Cmd<TMessage>;
|
|
81
82
|
/**
|
|
82
83
|
* Creates a command out of an async function and ignores the success case.
|
|
83
84
|
* @param task The async function to call.
|
|
84
85
|
* @param ofError Creates the message to dispatch when the promise is rejected.
|
|
85
86
|
* @param args The parameters of the task.
|
|
86
87
|
*/
|
|
87
|
-
attempt: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofError: (error: Error) =>
|
|
88
|
+
attempt: <TArgs extends unknown[], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>;
|
|
88
89
|
};
|
|
89
90
|
}
|
|
90
91
|
/**
|
|
91
92
|
* Creates a typed instance of the Command class.
|
|
92
|
-
* @template
|
|
93
|
+
* @template TMessage The type of the Msg discriminated union.
|
|
93
94
|
*/
|
|
94
|
-
declare function createCmd<
|
|
95
|
+
declare function createCmd<TMessage>(): Command<TMessage>;
|
|
95
96
|
export type { Dispatch, Cmd, };
|
|
96
97
|
export { createCmd, };
|
package/dist/Cmd.js
CHANGED
|
@@ -15,12 +15,12 @@ exports.createCmd = createCmd;
|
|
|
15
15
|
|
|
16
16
|
/**
|
|
17
17
|
* Contains functions to create commands.
|
|
18
|
-
* @template
|
|
18
|
+
* @template TMessage Type of the Message discriminated union.
|
|
19
19
|
*/
|
|
20
20
|
|
|
21
21
|
/**
|
|
22
22
|
* Creates a typed instance of the Command class.
|
|
23
|
-
* @template
|
|
23
|
+
* @template TMessage The type of the Msg discriminated union.
|
|
24
24
|
*/
|
|
25
25
|
function createCmd() {
|
|
26
26
|
return {
|
|
@@ -150,4 +150,4 @@ function createCmd() {
|
|
|
150
150
|
}
|
|
151
151
|
};
|
|
152
152
|
}
|
|
153
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["createCmd","none","ofMsg","msg","dispatch","batch","commands","flat","ofSub","sub","ofFunc","either","task","ofSuccess","ofError","args","bind","result","ex","perform","fallback","attempt","ofPromise","then","defaultFallbackHandler"],"sources":["../src/Cmd.ts"],"sourcesContent":["/**\n * Type of the dispatch function.\n */\ntype Dispatch<TMsg> = (msg: TMsg) => void;\n\ntype FallbackHandler = (error?: Error) => void;\ntype Sub<TMsg> = (dispatch: Dispatch<TMsg>, fallback?: FallbackHandler) => void;\n\n/**\n * Type of a command.\n */\ntype Cmd<TMsg> = Sub<TMsg> [];\n\n/**\n * Contains functions to create commands.\n * @template TMsg Type of the Message discriminated union.\n */\ninterface Command<TMsg> {\n    /**\n     * Represents an empty command.\n     */\n    none: [],\n    /**\n     * Creates a command out of a specific message.\n     * @param {TMsg} msg The specific message.\n     */\n    ofMsg: (msg: TMsg) => Cmd<TMsg>,\n    /**\n     * Aggregates multiple commands.\n     * @param {Cmd<TMsg> []} commands Array of commands.\n     */\n    batch: (...commands: Cmd<TMsg> []) => Cmd<TMsg>,\n    /**\n     * Command to call the subscriber.\n     * @param {Sub<TMsg>} sub The subscriber function.\n     */\n    ofSub: (sub: Sub<TMsg>) => Cmd<TMsg>,\n    /**\n     * Provides functionalities to create commands from simple functions.\n     */\n    ofFunc: {\n        /**\n        * Creates a command out of a simple function and maps the result.\n        * @param task The function to call.\n        * @param ofSuccess Creates the message to dispatch after a successful call of the task.\n        * @param ofError Creates the message to dispatch when an error occurred.\n        * @param args The parameters of the task.\n        */\n        either: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMsg, ofError: (error: Error) => TMsg, ...args: TArgs) => Cmd<TMsg>,\n        /**\n        * Creates a command out of a simple function and ignores the error case.\n        * @param task The function to call.\n        * @param ofSuccess Creates the message to dispatch after a successful call of the task.\n        * @param args The parameters of the task.\n        */\n        perform: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMsg, ...args: TArgs) => Cmd<TMsg>,\n        /**\n        * Creates a command out of a simple function and ignores the success case.\n        * @param task The function to call.\n        * @param ofError Creates the message to dispatch when an error occurred.\n        * @param args The parameters of the task.\n        */\n        attempt: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofError: (error: Error) => TMsg, ...args: TArgs) => Cmd<TMsg>,\n    },\n    /**\n     * Provides functionalities to create commands from async functions.\n     */\n    ofPromise: {\n        /**\n        * Creates a command out of an async function and maps the result.\n        * @param task The async function to call.\n        * @param ofSuccess Creates the message to dispatch when the promise is resolved.\n        * @param ofError Creates the message to dispatch when the promise is rejected.\n        * @param args The parameters of the task.\n        */\n        either: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMsg, ofError: (error: Error) => TMsg, ...args: TArgs) => Cmd<TMsg>,\n        /**\n        * Creates a command out of an async function and ignores the error case.\n        * @param task The async function to call.\n        * @param ofSuccess Creates the message to dispatch when the promise is resolved.\n        * @param args The parameters of the task.\n        */\n        perform: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMsg, ...args: TArgs) => Cmd<TMsg>,\n        /**\n        * Creates a command out of an async function and ignores the success case.\n        * @param task The async function to call.\n        * @param ofError Creates the message to dispatch when the promise is rejected.\n        * @param args The parameters of the task.\n        */\n        attempt: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofError: (error: Error) => TMsg, ...args: TArgs) => Cmd<TMsg>,\n    },\n}\n\n/**\n * Creates a typed instance of the Command class.\n * @template TMsg The type of the Msg discriminated union.\n */\nfunction createCmd<TMsg> (): Command<TMsg> {\n    return {\n        none: [],\n        ofMsg (msg: TMsg): Cmd<TMsg> {\n            return [dispatch => dispatch(msg)];\n        },\n        batch (...commands: Cmd<TMsg> []): Cmd<TMsg> {\n            return commands.flat();\n        },\n        ofSub (sub: Sub<TMsg>): Cmd<TMsg> {\n            return [sub];\n        },\n        ofFunc: {\n            either<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMsg, ofError: (error: Error) => TMsg, ...args: TArgs): Cmd<TMsg> {\n                const bind = (dispatch: Dispatch<TMsg>): void => {\n                    try {\n                        const result = task(...args);\n\n                        dispatch(ofSuccess(result));\n                    } catch (ex: unknown) {\n                        dispatch(ofError(ex as Error));\n                    }\n                };\n\n                return [bind];\n            },\n            perform<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMsg, ...args: TArgs): Cmd<TMsg> {\n                const bind = (dispatch: Dispatch<TMsg>, fallback?: FallbackHandler): void => {\n                    try {\n                        const result = task(...args);\n\n                        dispatch(ofSuccess(result));\n                    } catch (ex: unknown) {\n                        if (fallback) {\n                            fallback(ex as Error);\n                        }\n                    }\n                };\n\n                return [bind];\n            },\n            attempt<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofError: (error: Error) => TMsg, ...args: TArgs): Cmd<TMsg> {\n                const bind = (dispatch: Dispatch<TMsg>, fallback?: FallbackHandler): void => {\n                    try {\n                        task(...args);\n\n                        if (fallback) {\n                            fallback();\n                        }\n                    } catch (ex: unknown) {\n                        dispatch(ofError(ex as Error));\n                    }\n                };\n\n                return [bind];\n            },\n        },\n        ofPromise: {\n            either<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMsg, ofError: (error: Error) => TMsg, ...args: TArgs): Cmd<TMsg> {\n                const bind = (dispatch: Dispatch<TMsg>): void => {\n                    task(...args).then(result => dispatch(ofSuccess(result)))\n                        .catch((ex: Error) => dispatch(ofError(ex)));\n                };\n\n                return [bind];\n            },\n            perform<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMsg, ...args: TArgs): Cmd<TMsg> {\n                const defaultFallbackHandler = (): void => {\n                    // blank\n                };\n\n                const bind = (dispatch: Dispatch<TMsg>, fallback: FallbackHandler = defaultFallbackHandler): void => {\n                    task(...args).then(result => dispatch(ofSuccess(result)))\n                        .catch(fallback);\n                };\n\n                return [bind];\n            },\n            attempt<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofError: (error: Error) => TMsg, ...args: TArgs): Cmd<TMsg> {\n                const bind = (dispatch: Dispatch<TMsg>, fallback?: FallbackHandler): void => {\n                    task(...args).then(() => {\n                        if (fallback) {\n                            fallback();\n                        }\n                    })\n                        .catch((ex: Error) => dispatch(ofError(ex)));\n                };\n\n                return [bind];\n            },\n        },\n    };\n}\n\nexport type {\n    Dispatch,\n    Cmd,\n};\n\nexport {\n    createCmd,\n};"],"mappings":"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"}
|
|
153
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["createCmd","none","ofMsg","msg","dispatch","batch","commands","flat","ofSub","sub","ofFunc","either","task","ofSuccess","ofError","args","bind","result","ex","perform","fallback","attempt","ofPromise","then","defaultFallbackHandler"],"sources":["../src/Cmd.ts"],"sourcesContent":["/**\n * Type of the dispatch function.\n */\ntype Dispatch<TMessage> = (msg: TMessage) => void;\n\ntype FallbackHandler = (error?: Error) => void;\ntype Sub<TMsg> = (dispatch: Dispatch<TMsg>, fallback?: FallbackHandler) => void;\n\n/**\n * Type of a command.\n */\ntype Cmd<TMessage> = Sub<TMessage> [];\n\n/**\n * Contains functions to create commands.\n * @template TMessage Type of the Message discriminated union.\n */\ninterface Command<TMessage> {\n    /**\n     * Represents an empty command.\n     * @deprecated Do return nothing (`undefined`) instead.\n     */\n    none: [],\n    /**\n     * Creates a command out of a specific message.\n     * @param {TMessage} msg The specific message.\n     */\n    ofMsg: (msg: TMessage) => Cmd<TMessage>,\n    /**\n     * Aggregates multiple commands.\n     * @param {Cmd<TMessage> []} commands Array of commands.\n     */\n    batch: (...commands: Cmd<TMessage> []) => Cmd<TMessage>,\n    /**\n     * Command to call the subscriber.\n     * @param {Sub<TMessage>} sub The subscriber function.\n     */\n    ofSub: (sub: Sub<TMessage>) => Cmd<TMessage>,\n    /**\n     * Provides functionalities to create commands from simple functions.\n     */\n    ofFunc: {\n        /**\n        * Creates a command out of a simple function and maps the result.\n        * @param task The function to call.\n        * @param ofSuccess Creates the message to dispatch after a successful call of the task.\n        * @param ofError Creates the message to dispatch when an error occurred.\n        * @param args The parameters of the task.\n        */\n        either: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMessage, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>,\n        /**\n        * Creates a command out of a simple function and ignores the error case.\n        * @param task The function to call.\n        * @param ofSuccess Creates the message to dispatch after a successful call of the task.\n        * @param args The parameters of the task.\n        */\n        perform: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMessage, ...args: TArgs) => Cmd<TMessage>,\n        /**\n        * Creates a command out of a simple function and ignores the success case.\n        * @param task The function to call.\n        * @param ofError Creates the message to dispatch when an error occurred.\n        * @param args The parameters of the task.\n        */\n        attempt: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>,\n    },\n    /**\n     * Provides functionalities to create commands from async functions.\n     */\n    ofPromise: {\n        /**\n        * Creates a command out of an async function and maps the result.\n        * @param task The async function to call.\n        * @param ofSuccess Creates the message to dispatch when the promise is resolved.\n        * @param ofError Creates the message to dispatch when the promise is rejected.\n        * @param args The parameters of the task.\n        */\n        either: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMessage, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>,\n        /**\n        * Creates a command out of an async function and ignores the error case.\n        * @param task The async function to call.\n        * @param ofSuccess Creates the message to dispatch when the promise is resolved.\n        * @param args The parameters of the task.\n        */\n        perform: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMessage, ...args: TArgs) => Cmd<TMessage>,\n        /**\n        * Creates a command out of an async function and ignores the success case.\n        * @param task The async function to call.\n        * @param ofError Creates the message to dispatch when the promise is rejected.\n        * @param args The parameters of the task.\n        */\n        attempt: <TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofError: (error: Error) => TMessage, ...args: TArgs) => Cmd<TMessage>,\n    },\n}\n\n/**\n * Creates a typed instance of the Command class.\n * @template TMessage The type of the Msg discriminated union.\n */\nfunction createCmd<TMessage> (): Command<TMessage> {\n    return {\n        none: [],\n        ofMsg (msg: TMessage): Cmd<TMessage> {\n            return [dispatch => dispatch(msg)];\n        },\n        batch (...commands: Cmd<TMessage> []): Cmd<TMessage> {\n            return commands.flat();\n        },\n        ofSub (sub: Sub<TMessage>): Cmd<TMessage> {\n            return [sub];\n        },\n        ofFunc: {\n            either<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMessage, ofError: (error: Error) => TMessage, ...args: TArgs): Cmd<TMessage> {\n                const bind = (dispatch: Dispatch<TMessage>): void => {\n                    try {\n                        const result = task(...args);\n\n                        dispatch(ofSuccess(result));\n                    } catch (ex: unknown) {\n                        dispatch(ofError(ex as Error));\n                    }\n                };\n\n                return [bind];\n            },\n            perform<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofSuccess: (result: TReturn) => TMessage, ...args: TArgs): Cmd<TMessage> {\n                const bind = (dispatch: Dispatch<TMessage>, fallback?: FallbackHandler): void => {\n                    try {\n                        const result = task(...args);\n\n                        dispatch(ofSuccess(result));\n                    } catch (ex: unknown) {\n                        if (fallback) {\n                            fallback(ex as Error);\n                        }\n                    }\n                };\n\n                return [bind];\n            },\n            attempt<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => TReturn, ofError: (error: Error) => TMessage, ...args: TArgs): Cmd<TMessage> {\n                const bind = (dispatch: Dispatch<TMessage>, fallback?: FallbackHandler): void => {\n                    try {\n                        task(...args);\n\n                        if (fallback) {\n                            fallback();\n                        }\n                    } catch (ex: unknown) {\n                        dispatch(ofError(ex as Error));\n                    }\n                };\n\n                return [bind];\n            },\n        },\n        ofPromise: {\n            either<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMessage, ofError: (error: Error) => TMessage, ...args: TArgs): Cmd<TMessage> {\n                const bind = (dispatch: Dispatch<TMessage>): void => {\n                    task(...args).then(result => dispatch(ofSuccess(result)))\n                        .catch((ex: Error) => dispatch(ofError(ex)));\n                };\n\n                return [bind];\n            },\n            perform<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofSuccess: (result: TReturn) => TMessage, ...args: TArgs): Cmd<TMessage> {\n                const defaultFallbackHandler = (): void => {\n                    // blank\n                };\n\n                const bind = (dispatch: Dispatch<TMessage>, fallback: FallbackHandler = defaultFallbackHandler): void => {\n                    task(...args).then(result => dispatch(ofSuccess(result)))\n                        .catch(fallback);\n                };\n\n                return [bind];\n            },\n            attempt<TArgs extends unknown [], TReturn>(task: (...args: TArgs) => Promise<TReturn>, ofError: (error: Error) => TMessage, ...args: TArgs): Cmd<TMessage> {\n                const bind = (dispatch: Dispatch<TMessage>, fallback?: FallbackHandler): void => {\n                    task(...args).then(() => {\n                        if (fallback) {\n                            fallback();\n                        }\n                    })\n                        .catch((ex: Error) => dispatch(ofError(ex)));\n                };\n\n                return [bind];\n            },\n        },\n    };\n}\n\nexport type {\n    Dispatch,\n    Cmd,\n};\n\nexport {\n    createCmd,\n};"],"mappings":"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"}
|
package/dist/ElmComponent.d.ts
CHANGED
|
@@ -1,17 +1,17 @@
|
|
|
1
|
-
import { InitFunction, UpdateFunction } from "./Types";
|
|
2
|
-
import { Message } from "./Init";
|
|
3
1
|
import React from "react";
|
|
2
|
+
import { Message } from "./Init";
|
|
3
|
+
import { InitFunction, UpdateFunction } from "./Types";
|
|
4
4
|
/**
|
|
5
|
-
* Abstract class for a react class component using the
|
|
5
|
+
* Abstract class for a react class component using the Elmish pattern.
|
|
6
6
|
* @export
|
|
7
7
|
* @abstract
|
|
8
8
|
* @class ElmComponent
|
|
9
9
|
* @extends {Component<TProps, TModel>}
|
|
10
10
|
* @template TModel The type of the model.
|
|
11
|
-
* @template
|
|
11
|
+
* @template TMessage The type of the messages.
|
|
12
12
|
* @template TProps The type of the props.
|
|
13
13
|
*/
|
|
14
|
-
declare abstract class ElmComponent<TModel,
|
|
14
|
+
declare abstract class ElmComponent<TModel, TMessage extends Message, TProps> extends React.Component<TProps> {
|
|
15
15
|
private initCmd;
|
|
16
16
|
private readonly componentName;
|
|
17
17
|
private readonly buffer;
|
|
@@ -25,7 +25,7 @@ declare abstract class ElmComponent<TModel, TMsg extends Message, TProps> extend
|
|
|
25
25
|
* @param name The name of the component.
|
|
26
26
|
* @memberof ElmComponent
|
|
27
27
|
*/
|
|
28
|
-
constructor(props: TProps, init: InitFunction<TProps, TModel,
|
|
28
|
+
constructor(props: TProps, init: InitFunction<TProps, TModel, TMessage>, name: string);
|
|
29
29
|
/**
|
|
30
30
|
* Is called when the component is loaded.
|
|
31
31
|
* When implementing this method, the base implementation has to be called.
|
|
@@ -48,19 +48,19 @@ declare abstract class ElmComponent<TModel, TMsg extends Message, TProps> extend
|
|
|
48
48
|
get model(): Readonly<TModel>;
|
|
49
49
|
/**
|
|
50
50
|
* Dispatches a message.
|
|
51
|
-
* @param {
|
|
51
|
+
* @param {TMessage} msg The message to dispatch.
|
|
52
52
|
* @memberof ElmComponent
|
|
53
53
|
*/
|
|
54
|
-
readonly dispatch: (msg:
|
|
54
|
+
readonly dispatch: (msg: TMessage) => void;
|
|
55
55
|
/**
|
|
56
56
|
* Function to modify the model based on a message.
|
|
57
57
|
* @param {TModel} model The current model.
|
|
58
|
-
* @param {
|
|
58
|
+
* @param {TMessage} msg The message to process.
|
|
59
59
|
* @param {TProps} props The props of the component.
|
|
60
60
|
* @returns The new model (can also be an empty object {}) and an optional new message to dispatch.
|
|
61
61
|
* @abstract
|
|
62
62
|
* @memberof ElmComponent
|
|
63
63
|
*/
|
|
64
|
-
abstract update: UpdateFunction<TProps, TModel,
|
|
64
|
+
abstract update: UpdateFunction<TProps, TModel, TMessage>;
|
|
65
65
|
}
|
|
66
66
|
export { ElmComponent, };
|
package/dist/ElmComponent.js
CHANGED
|
@@ -7,11 +7,11 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
7
7
|
});
|
|
8
8
|
exports.ElmComponent = void 0;
|
|
9
9
|
|
|
10
|
-
var
|
|
10
|
+
var _react = _interopRequireDefault(require("react"));
|
|
11
11
|
|
|
12
|
-
var
|
|
12
|
+
var _Init = require("./Init");
|
|
13
13
|
|
|
14
|
-
var
|
|
14
|
+
var _fakeOptions = require("./Testing/fakeOptions");
|
|
15
15
|
|
|
16
16
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
|
|
17
17
|
|
|
@@ -54,13 +54,13 @@ function _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.g
|
|
|
54
54
|
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
55
55
|
|
|
56
56
|
/**
|
|
57
|
-
* Abstract class for a react class component using the
|
|
57
|
+
* Abstract class for a react class component using the Elmish pattern.
|
|
58
58
|
* @export
|
|
59
59
|
* @abstract
|
|
60
60
|
* @class ElmComponent
|
|
61
61
|
* @extends {Component<TProps, TModel>}
|
|
62
62
|
* @template TModel The type of the model.
|
|
63
|
-
* @template
|
|
63
|
+
* @template TMessage The type of the messages.
|
|
64
64
|
* @template TProps The type of the props.
|
|
65
65
|
*/
|
|
66
66
|
var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
@@ -76,8 +76,6 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
76
76
|
* @memberof ElmComponent
|
|
77
77
|
*/
|
|
78
78
|
function ElmComponent(props, init, name) {
|
|
79
|
-
var _getFakeInitResultOnc;
|
|
80
|
-
|
|
81
79
|
var _this;
|
|
82
80
|
|
|
83
81
|
_classCallCheck(this, ElmComponent);
|
|
@@ -155,7 +153,13 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
155
153
|
|
|
156
154
|
_defineProperty(_assertThisInitialized(_this), "update", void 0);
|
|
157
155
|
|
|
158
|
-
var
|
|
156
|
+
var fakeOptions = (0, _fakeOptions.getFakeOptionsOnce)();
|
|
157
|
+
|
|
158
|
+
if (fakeOptions !== null && fakeOptions !== void 0 && fakeOptions.dispatch) {
|
|
159
|
+
_this.dispatch = fakeOptions.dispatch;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
var _ref = fakeOptions !== null && fakeOptions !== void 0 && fakeOptions.model ? [fakeOptions.model] : init(_this.props),
|
|
159
163
|
_ref2 = _slicedToArray(_ref, 2),
|
|
160
164
|
_model = _ref2[0],
|
|
161
165
|
_cmd = _ref2[1];
|
|
@@ -222,7 +226,7 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
222
226
|
}
|
|
223
227
|
/**
|
|
224
228
|
* Dispatches a message.
|
|
225
|
-
* @param {
|
|
229
|
+
* @param {TMessage} msg The message to dispatch.
|
|
226
230
|
* @memberof ElmComponent
|
|
227
231
|
*/
|
|
228
232
|
|
|
@@ -232,4 +236,4 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
232
236
|
}(_react["default"].Component);
|
|
233
237
|
|
|
234
238
|
exports.ElmComponent = ElmComponent;
|
|
235
|
-
//# 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","getFakeInitResultOnce","initCmd","forEach","call","dispatch","React","Component"],"sources":["../src/ElmComponent.ts"],"sourcesContent":["import { InitFunction, Nullable, UpdateFunction } from \"./Types\";\nimport { Message, Services } from \"./Init\";\nimport { Cmd } from \"./Cmd\";\nimport { getFakeInitResultOnce } from \"./Testing/fakeInitResult\";\nimport React from \"react\";\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 TMsg The type of the messages.\n * @template TProps The type of the props.\n */\nabstract class ElmComponent<TModel, TMsg extends Message, TProps> extends React.Component<TProps> {\n    private initCmd: Nullable<Cmd<TMsg>> | undefined;\n    private readonly componentName: string;\n    private readonly buffer: TMsg [] = [];\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, TMsg>, name: string) {\n        super(props);\n\n        const [model, cmd] = getFakeInitResultOnce<TModel, TMsg>() ?? 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<TMsg>): 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 {TMsg} msg The message to dispatch.\n     * @memberof ElmComponent\n     */\n    public readonly dispatch = (msg: TMsg): 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: TMsg | 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 {TMsg} 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, TMsg>;\n}\n\nexport {\n    ElmComponent,\n};"],"mappings":"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"}
|
|
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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"}
|
package/dist/ErrorHandling.d.ts
CHANGED
|
@@ -28,6 +28,6 @@ declare function errorHandler<TModel, TMessage>(): {
|
|
|
28
28
|
* Calls the error handling middleware if specified.
|
|
29
29
|
* @param {Error} error The error.
|
|
30
30
|
*/
|
|
31
|
-
declare function handleError<TModel,
|
|
31
|
+
declare function handleError<TModel, TMessage>(error: Error): UpdateReturnType<TModel, TMessage>;
|
|
32
32
|
export type { ErrorMessage };
|
|
33
33
|
export { errorMsg, errorHandler, handleError, };
|
package/dist/ErrorHandling.js
CHANGED
|
@@ -55,4 +55,4 @@ function handleError(error) {
|
|
|
55
55
|
(_Services$logger = _Init.Services.logger) === null || _Services$logger === void 0 ? void 0 : _Services$logger.error(error);
|
|
56
56
|
return [{}];
|
|
57
57
|
}
|
|
58
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJlcnJvck1zZyIsImVycm9yIiwibmFtZSIsImVycm9ySGFuZGxlciIsImhhbmRsZUVycm9yIiwiU2VydmljZXMiLCJlcnJvck1pZGRsZXdhcmUiLCJsb2dnZXIiXSwic291cmNlcyI6WyIuLi9zcmMvRXJyb3JIYW5kbGluZy50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTZXJ2aWNlcyB9IGZyb20gXCIuL0luaXRcIjtcbmltcG9ydCB7IFVwZGF0ZVJldHVyblR5cGUgfSBmcm9tIFwiLi9UeXBlc1wiO1xuXG4vKipcbiAqIEVycm9yIG1lc3NhZ2Ugb2JqZWN0LlxuICogQWRkIHRoaXMgdG8geW91ciBNZXNzYWdlIHR5cGUuXG4gKi9cbmludGVyZmFjZSBFcnJvck1lc3NhZ2Uge1xuICAgIG5hbWU6IFwiZXJyb3JcIixcbiAgICBlcnJvcjogRXJyb3IsXG59XG5cbi8qKlxuICogVGhpcyBvYmplY3QgY29udGFpbnMgdGhlIGZ1bmN0aW9uIHRvIGNyZWF0ZSBhbiBlcnJvciBtZXNzYWdlLlxuICogU3ByZWFkIHRoaXMgaW50byB5b3VyIE1zZyBvYmplY3QuXG4gKi9cbmNvbnN0IGVycm9yTXNnID0ge1xuICAgIGVycm9yOiAoZXJyb3I6IEVycm9yKTogRXJyb3JNZXNzYWdlID0+ICh7IG5hbWU6IFwiZXJyb3JcIiwgZXJyb3IgfSksXG59O1xuXG4vKipcbiAqIENyZWF0ZXMgYW4gb2JqZWN0IHRvIGhhbmRsZSBlcnJvciBtZXNzYWdlcyBpbiBhbiB1cGRhdGUgbWFwLlxuICogU3ByZWFkIHRoaXMgaW50byB5b3VyIGBVcGRhdGVNYXBgLlxuICogQHJldHVybnMgQW4gb2JqZWN0IGNvbnRhaW5pbmcgYW4gZXJyb3IgaGFuZGxlciBmdW5jdGlvbi5cbiAqL1xuZnVuY3Rpb24gZXJyb3JIYW5kbGVyPFRNb2RlbCwgVE1lc3NhZ2U+ICgpOiB7IGVycm9yOiAobXNnOiBFcnJvck1lc3NhZ2UpID0+
|
|
58
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJlcnJvck1zZyIsImVycm9yIiwibmFtZSIsImVycm9ySGFuZGxlciIsImhhbmRsZUVycm9yIiwiU2VydmljZXMiLCJlcnJvck1pZGRsZXdhcmUiLCJsb2dnZXIiXSwic291cmNlcyI6WyIuLi9zcmMvRXJyb3JIYW5kbGluZy50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTZXJ2aWNlcyB9IGZyb20gXCIuL0luaXRcIjtcbmltcG9ydCB7IFVwZGF0ZVJldHVyblR5cGUgfSBmcm9tIFwiLi9UeXBlc1wiO1xuXG4vKipcbiAqIEVycm9yIG1lc3NhZ2Ugb2JqZWN0LlxuICogQWRkIHRoaXMgdG8geW91ciBNZXNzYWdlIHR5cGUuXG4gKi9cbmludGVyZmFjZSBFcnJvck1lc3NhZ2Uge1xuICAgIG5hbWU6IFwiZXJyb3JcIixcbiAgICBlcnJvcjogRXJyb3IsXG59XG5cbi8qKlxuICogVGhpcyBvYmplY3QgY29udGFpbnMgdGhlIGZ1bmN0aW9uIHRvIGNyZWF0ZSBhbiBlcnJvciBtZXNzYWdlLlxuICogU3ByZWFkIHRoaXMgaW50byB5b3VyIE1zZyBvYmplY3QuXG4gKi9cbmNvbnN0IGVycm9yTXNnID0ge1xuICAgIGVycm9yOiAoZXJyb3I6IEVycm9yKTogRXJyb3JNZXNzYWdlID0+ICh7IG5hbWU6IFwiZXJyb3JcIiwgZXJyb3IgfSksXG59O1xuXG4vKipcbiAqIENyZWF0ZXMgYW4gb2JqZWN0IHRvIGhhbmRsZSBlcnJvciBtZXNzYWdlcyBpbiBhbiB1cGRhdGUgbWFwLlxuICogU3ByZWFkIHRoaXMgaW50byB5b3VyIGBVcGRhdGVNYXBgLlxuICogQHJldHVybnMgQW4gb2JqZWN0IGNvbnRhaW5pbmcgYW4gZXJyb3IgaGFuZGxlciBmdW5jdGlvbi5cbiAqL1xuZnVuY3Rpb24gZXJyb3JIYW5kbGVyPFRNb2RlbCwgVE1lc3NhZ2U+ICgpOiB7IGVycm9yOiAobXNnOiBFcnJvck1lc3NhZ2UpID0+IFVwZGF0ZVJldHVyblR5cGU8VE1vZGVsLCBUTWVzc2FnZT4gfSB7XG4gICAgcmV0dXJuIHtcbiAgICAgICAgZXJyb3IgKHsgZXJyb3IgfSkge1xuICAgICAgICAgICAgcmV0dXJuIGhhbmRsZUVycm9yKGVycm9yKTtcbiAgICAgICAgfSxcbiAgICB9O1xufVxuXG4vKipcbiAqIEhhbmRsZXMgYW4gZXJyb3IuXG4gKiBMb2dzIHRoZSBlcnJvciBpZiBhIExvZ2dlciB3YXMgc3BlY2lmaWVkLlxuICogQ2FsbHMgdGhlIGVycm9yIGhhbmRsaW5nIG1pZGRsZXdhcmUgaWYgc3BlY2lmaWVkLlxuICogQHBhcmFtIHtFcnJvcn0gZXJyb3IgVGhlIGVycm9yLlxuICovXG5mdW5jdGlvbiBoYW5kbGVFcnJvcjxUTW9kZWwsIFRNZXNzYWdlPiAoZXJyb3I6IEVycm9yKTogVXBkYXRlUmV0dXJuVHlwZTxUTW9kZWwsIFRNZXNzYWdlPiB7XG4gICAgaWYgKFNlcnZpY2VzLmVycm9yTWlkZGxld2FyZSkge1xuICAgICAgICBTZXJ2aWNlcy5lcnJvck1pZGRsZXdhcmUoZXJyb3IpO1xuICAgIH1cbiAgICBTZXJ2aWNlcy5sb2dnZXI/LmVycm9yKGVycm9yKTtcblxuICAgIHJldHVybiBbe31dO1xufVxuXG5leHBvcnQgdHlwZSB7IEVycm9yTWVzc2FnZSB9O1xuXG5leHBvcnQge1xuICAgIGVycm9yTXNnLFxuICAgIGVycm9ySGFuZGxlcixcbiAgICBoYW5kbGVFcnJvcixcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFBQTs7QUFZQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLElBQU1BLFFBQVEsR0FBRztFQUNiQyxLQUFLLEVBQUUsZUFBQ0EsTUFBRDtJQUFBLE9BQWlDO01BQUVDLElBQUksRUFBRSxPQUFSO01BQWlCRCxLQUFLLEVBQUxBO0lBQWpCLENBQWpDO0VBQUE7QUFETSxDQUFqQjtBQUlBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Ozs7QUFDQSxTQUFTRSxZQUFULEdBQWlIO0VBQzdHLE9BQU87SUFDSEYsS0FERyx1QkFDZTtNQUFBLElBQVRBLEtBQVMsUUFBVEEsS0FBUztNQUNkLE9BQU9HLFdBQVcsQ0FBQ0gsS0FBRCxDQUFsQjtJQUNIO0VBSEUsQ0FBUDtBQUtIO0FBRUQ7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7QUFDQSxTQUFTRyxXQUFULENBQXdDSCxLQUF4QyxFQUEwRjtFQUFBOztFQUN0RixJQUFJSSxjQUFBLENBQVNDLGVBQWIsRUFBOEI7SUFDMUJELGNBQUEsQ0FBU0MsZUFBVCxDQUF5QkwsS0FBekI7RUFDSDs7RUFDRCxvQkFBQUksY0FBQSxDQUFTRSxNQUFULHNFQUFpQk4sS0FBakIsQ0FBdUJBLEtBQXZCO0VBRUEsT0FBTyxDQUFDLEVBQUQsQ0FBUDtBQUNIIn0=
|
package/dist/Init.d.ts
CHANGED
|
@@ -9,11 +9,28 @@ interface Message {
|
|
|
9
9
|
declare type ErrorMiddlewareFunc = (error: Error) => void;
|
|
10
10
|
declare type DispatchMiddlewareFunc = (msg: Message) => void;
|
|
11
11
|
interface ElmOptions {
|
|
12
|
+
/**
|
|
13
|
+
* The logger to use for logging called messages and updated models.
|
|
14
|
+
* @type {Logger}
|
|
15
|
+
*/
|
|
12
16
|
logger?: Logger;
|
|
17
|
+
/**
|
|
18
|
+
* Middleware to call when error messages are handled by the `handleError` function.
|
|
19
|
+
* @type {ErrorMiddlewareFunc}
|
|
20
|
+
*/
|
|
13
21
|
errorMiddleware?: ErrorMiddlewareFunc;
|
|
22
|
+
/**
|
|
23
|
+
* Middleware to call for every processed message.
|
|
24
|
+
* @type {DispatchMiddlewareFunc}
|
|
25
|
+
*/
|
|
14
26
|
dispatchMiddleware?: DispatchMiddlewareFunc;
|
|
15
27
|
}
|
|
16
28
|
declare const Services: ElmOptions;
|
|
29
|
+
/**
|
|
30
|
+
* This initializes the Elmish module.
|
|
31
|
+
* You only need to call this function if you want to set some of the options.
|
|
32
|
+
* @param {ElmOptions} options
|
|
33
|
+
*/
|
|
17
34
|
declare function init(options: ElmOptions): void;
|
|
18
35
|
export type { Logger, Message, ErrorMiddlewareFunc, DispatchMiddlewareFunc, ElmOptions, };
|
|
19
36
|
export { Services, init, };
|
package/dist/Init.js
CHANGED
|
@@ -10,6 +10,12 @@ var Services = {
|
|
|
10
10
|
errorMiddleware: undefined,
|
|
11
11
|
dispatchMiddleware: undefined
|
|
12
12
|
};
|
|
13
|
+
/**
|
|
14
|
+
* This initializes the Elmish module.
|
|
15
|
+
* You only need to call this function if you want to set some of the options.
|
|
16
|
+
* @param {ElmOptions} options
|
|
17
|
+
*/
|
|
18
|
+
|
|
13
19
|
exports.Services = Services;
|
|
14
20
|
|
|
15
21
|
function init(options) {
|
|
@@ -17,4 +23,4 @@ function init(options) {
|
|
|
17
23
|
Services.errorMiddleware = options.errorMiddleware;
|
|
18
24
|
Services.dispatchMiddleware = options.dispatchMiddleware;
|
|
19
25
|
}
|
|
20
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJTZXJ2aWNlcyIsImxvZ2dlciIsInVuZGVmaW5lZCIsImVycm9yTWlkZGxld2FyZSIsImRpc3BhdGNoTWlkZGxld2FyZSIsImluaXQiLCJvcHRpb25zIl0sInNvdXJjZXMiOlsiLi4vc3JjL0luaXQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW50ZXJmYWNlIExvZ2dlciB7XG4gICAgZGVidWc6ICguLi5hcmdzOiB1bmtub3duIFtdKSA9PiB2b2lkLFxuICAgIGluZm86ICguLi5hcmdzOiB1bmtub3duIFtdKSA9PiB2b2lkLFxuICAgIGVycm9yOiAoLi4uYXJnczogdW5rbm93biBbXSkgPT4gdm9pZCxcbn1cblxuaW50ZXJmYWNlIE1lc3NhZ2Uge1xuICAgIG5hbWU6IHN0cmluZyB8IHN5bWJvbCxcbn1cblxudHlwZSBFcnJvck1pZGRsZXdhcmVGdW5jID0gKGVycm9yOiBFcnJvcikgPT4gdm9pZDtcbnR5cGUgRGlzcGF0Y2hNaWRkbGV3YXJlRnVuYyA9IChtc2c6IE1lc3NhZ2UpID0+
|
|
26
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJTZXJ2aWNlcyIsImxvZ2dlciIsInVuZGVmaW5lZCIsImVycm9yTWlkZGxld2FyZSIsImRpc3BhdGNoTWlkZGxld2FyZSIsImluaXQiLCJvcHRpb25zIl0sInNvdXJjZXMiOlsiLi4vc3JjL0luaXQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW50ZXJmYWNlIExvZ2dlciB7XG4gICAgZGVidWc6ICguLi5hcmdzOiB1bmtub3duIFtdKSA9PiB2b2lkLFxuICAgIGluZm86ICguLi5hcmdzOiB1bmtub3duIFtdKSA9PiB2b2lkLFxuICAgIGVycm9yOiAoLi4uYXJnczogdW5rbm93biBbXSkgPT4gdm9pZCxcbn1cblxuaW50ZXJmYWNlIE1lc3NhZ2Uge1xuICAgIG5hbWU6IHN0cmluZyB8IHN5bWJvbCxcbn1cblxudHlwZSBFcnJvck1pZGRsZXdhcmVGdW5jID0gKGVycm9yOiBFcnJvcikgPT4gdm9pZDtcbnR5cGUgRGlzcGF0Y2hNaWRkbGV3YXJlRnVuYyA9IChtc2c6IE1lc3NhZ2UpID0+IHZvaWQ7XG5cbmludGVyZmFjZSBFbG1PcHRpb25zIHtcbiAgICAvKipcbiAgICAgKiBUaGUgbG9nZ2VyIHRvIHVzZSBmb3IgbG9nZ2luZyBjYWxsZWQgbWVzc2FnZXMgYW5kIHVwZGF0ZWQgbW9kZWxzLlxuICAgICAqIEB0eXBlIHtMb2dnZXJ9XG4gICAgICovXG4gICAgbG9nZ2VyPzogTG9nZ2VyLFxuXG4gICAgLyoqXG4gICAgICogTWlkZGxld2FyZSB0byBjYWxsIHdoZW4gZXJyb3IgbWVzc2FnZXMgYXJlIGhhbmRsZWQgYnkgdGhlIGBoYW5kbGVFcnJvcmAgZnVuY3Rpb24uXG4gICAgICogQHR5cGUge0Vycm9yTWlkZGxld2FyZUZ1bmN9XG4gICAgICovXG4gICAgZXJyb3JNaWRkbGV3YXJlPzogRXJyb3JNaWRkbGV3YXJlRnVuYyxcblxuICAgIC8qKlxuICAgICAqIE1pZGRsZXdhcmUgdG8gY2FsbCBmb3IgZXZlcnkgcHJvY2Vzc2VkIG1lc3NhZ2UuXG4gICAgICogQHR5cGUge0Rpc3BhdGNoTWlkZGxld2FyZUZ1bmN9XG4gICAgICovXG4gICAgZGlzcGF0Y2hNaWRkbGV3YXJlPzogRGlzcGF0Y2hNaWRkbGV3YXJlRnVuYyxcbn1cblxuY29uc3QgU2VydmljZXM6IEVsbU9wdGlvbnMgPSB7XG4gICAgbG9nZ2VyOiB1bmRlZmluZWQsXG4gICAgZXJyb3JNaWRkbGV3YXJlOiB1bmRlZmluZWQsXG4gICAgZGlzcGF0Y2hNaWRkbGV3YXJlOiB1bmRlZmluZWQsXG59O1xuXG4vKipcbiAqIFRoaXMgaW5pdGlhbGl6ZXMgdGhlIEVsbWlzaCBtb2R1bGUuXG4gKiBZb3Ugb25seSBuZWVkIHRvIGNhbGwgdGhpcyBmdW5jdGlvbiBpZiB5b3Ugd2FudCB0byBzZXQgc29tZSBvZiB0aGUgb3B0aW9ucy5cbiAqIEBwYXJhbSB7RWxtT3B0aW9uc30gb3B0aW9uc1xuICovXG5mdW5jdGlvbiBpbml0IChvcHRpb25zOiBFbG1PcHRpb25zKTogdm9pZCB7XG4gICAgU2VydmljZXMubG9nZ2VyID0gb3B0aW9ucy5sb2dnZXI7XG4gICAgU2VydmljZXMuZXJyb3JNaWRkbGV3YXJlID0gb3B0aW9ucy5lcnJvck1pZGRsZXdhcmU7XG4gICAgU2VydmljZXMuZGlzcGF0Y2hNaWRkbGV3YXJlID0gb3B0aW9ucy5kaXNwYXRjaE1pZGRsZXdhcmU7XG59XG5cbmV4cG9ydCB0eXBlIHtcbiAgICBMb2dnZXIsXG4gICAgTWVzc2FnZSxcbiAgICBFcnJvck1pZGRsZXdhcmVGdW5jLFxuICAgIERpc3BhdGNoTWlkZGxld2FyZUZ1bmMsXG4gICAgRWxtT3B0aW9ucyxcbn07XG5cbmV4cG9ydCB7XG4gICAgU2VydmljZXMsXG4gICAgaW5pdCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBaUNBLElBQU1BLFFBQW9CLEdBQUc7RUFDekJDLE1BQU0sRUFBRUMsU0FEaUI7RUFFekJDLGVBQWUsRUFBRUQsU0FGUTtFQUd6QkUsa0JBQWtCLEVBQUVGO0FBSEssQ0FBN0I7QUFNQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7O0FBQ0EsU0FBU0csSUFBVCxDQUFlQyxPQUFmLEVBQTBDO0VBQ3RDTixRQUFRLENBQUNDLE1BQVQsR0FBa0JLLE9BQU8sQ0FBQ0wsTUFBMUI7RUFDQUQsUUFBUSxDQUFDRyxlQUFULEdBQTJCRyxPQUFPLENBQUNILGVBQW5DO0VBQ0FILFFBQVEsQ0FBQ0ksa0JBQVQsR0FBOEJFLE9BQU8sQ0FBQ0Ysa0JBQXRDO0FBQ0gifQ==
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { Dispatch } from "../Cmd";
|
|
2
|
+
import { MessageBase, Nullable } from "../Types";
|
|
3
|
+
/**
|
|
4
|
+
* Options for the `renderWithModel` function.
|
|
5
|
+
* @interface RenderWithModelOptions
|
|
6
|
+
* @template TModel The type of the model.
|
|
7
|
+
* @template TMessage The type of the messages discriminated union.
|
|
8
|
+
*/
|
|
9
|
+
interface RenderWithModelOptions<TModel, TMessage extends MessageBase> {
|
|
10
|
+
/**
|
|
11
|
+
* The model to use when rendering the component.
|
|
12
|
+
* @type {TModel}
|
|
13
|
+
*/
|
|
14
|
+
model: TModel;
|
|
15
|
+
/**
|
|
16
|
+
* A fake dispatch function to use when processing messages.
|
|
17
|
+
* @type {Dispatch<TMessage>}
|
|
18
|
+
*/
|
|
19
|
+
dispatch?: Dispatch<TMessage>;
|
|
20
|
+
}
|
|
21
|
+
declare function setFakeOptions(options: Nullable<RenderWithModelOptions<unknown, MessageBase>>): void;
|
|
22
|
+
declare function getFakeOptionsOnce<TModel, TMessage extends MessageBase>(): Nullable<RenderWithModelOptions<TModel, TMessage>>;
|
|
23
|
+
export type { RenderWithModelOptions, };
|
|
24
|
+
export { setFakeOptions, getFakeOptionsOnce, };
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.getFakeOptionsOnce = getFakeOptionsOnce;
|
|
7
|
+
exports.setFakeOptions = setFakeOptions;
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Options for the `renderWithModel` function.
|
|
11
|
+
* @interface RenderWithModelOptions
|
|
12
|
+
* @template TModel The type of the model.
|
|
13
|
+
* @template TMessage The type of the messages discriminated union.
|
|
14
|
+
*/
|
|
15
|
+
var currentFakeOptions;
|
|
16
|
+
|
|
17
|
+
function setFakeOptions(options) {
|
|
18
|
+
currentFakeOptions = options;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function getFakeOptionsOnce() {
|
|
22
|
+
var temp = currentFakeOptions;
|
|
23
|
+
currentFakeOptions = null;
|
|
24
|
+
return temp;
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjdXJyZW50RmFrZU9wdGlvbnMiLCJzZXRGYWtlT3B0aW9ucyIsIm9wdGlvbnMiLCJnZXRGYWtlT3B0aW9uc09uY2UiLCJ0ZW1wIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL1Rlc3RpbmcvZmFrZU9wdGlvbnMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlzcGF0Y2ggfSBmcm9tIFwiLi4vQ21kXCI7XG5pbXBvcnQgeyBNZXNzYWdlQmFzZSwgTnVsbGFibGUgfSBmcm9tIFwiLi4vVHlwZXNcIjtcblxuLyoqXG4gKiBPcHRpb25zIGZvciB0aGUgYHJlbmRlcldpdGhNb2RlbGAgZnVuY3Rpb24uXG4gKiBAaW50ZXJmYWNlIFJlbmRlcldpdGhNb2RlbE9wdGlvbnNcbiAqIEB0ZW1wbGF0ZSBUTW9kZWwgVGhlIHR5cGUgb2YgdGhlIG1vZGVsLlxuICogQHRlbXBsYXRlIFRNZXNzYWdlIFRoZSB0eXBlIG9mIHRoZSBtZXNzYWdlcyBkaXNjcmltaW5hdGVkIHVuaW9uLlxuICovXG5pbnRlcmZhY2UgUmVuZGVyV2l0aE1vZGVsT3B0aW9uczxUTW9kZWwsIFRNZXNzYWdlIGV4dGVuZHMgTWVzc2FnZUJhc2U+IHtcbiAgICAvKipcbiAgICAgKiBUaGUgbW9kZWwgdG8gdXNlIHdoZW4gcmVuZGVyaW5nIHRoZSBjb21wb25lbnQuXG4gICAgICogQHR5cGUge1RNb2RlbH1cbiAgICAgKi9cbiAgICBtb2RlbDogVE1vZGVsLFxuXG4gICAgLyoqXG4gICAgICogQSBmYWtlIGRpc3BhdGNoIGZ1bmN0aW9uIHRvIHVzZSB3aGVuIHByb2Nlc3NpbmcgbWVzc2FnZXMuXG4gICAgICogQHR5cGUge0Rpc3BhdGNoPFRNZXNzYWdlPn1cbiAgICAgKi9cbiAgICBkaXNwYXRjaD86IERpc3BhdGNoPFRNZXNzYWdlPixcbn1cblxubGV0IGN1cnJlbnRGYWtlT3B0aW9uczogTnVsbGFibGU8UmVuZGVyV2l0aE1vZGVsT3B0aW9uczx1bmtub3duLCBNZXNzYWdlQmFzZT4+O1xuXG5mdW5jdGlvbiBzZXRGYWtlT3B0aW9ucyAob3B0aW9uczogTnVsbGFibGU8UmVuZGVyV2l0aE1vZGVsT3B0aW9uczx1bmtub3duLCBNZXNzYWdlQmFzZT4+KTogdm9pZCB7XG4gICAgY3VycmVudEZha2VPcHRpb25zID0gb3B0aW9ucztcbn1cblxuZnVuY3Rpb24gZ2V0RmFrZU9wdGlvbnNPbmNlPFRNb2RlbCwgVE1lc3NhZ2UgZXh0ZW5kcyBNZXNzYWdlQmFzZT4gKCk6IE51bGxhYmxlPFJlbmRlcldpdGhNb2RlbE9wdGlvbnM8VE1vZGVsLCBUTWVzc2FnZT4+IHtcbiAgICBjb25zdCB0ZW1wID0gY3VycmVudEZha2VPcHRpb25zO1xuXG4gICAgY3VycmVudEZha2VPcHRpb25zID0gbnVsbDtcblxuICAgIHJldHVybiB0ZW1wIGFzIFJlbmRlcldpdGhNb2RlbE9wdGlvbnM8VE1vZGVsLCBUTWVzc2FnZT47XG59XG5cbmV4cG9ydCB0eXBlIHtcbiAgICBSZW5kZXJXaXRoTW9kZWxPcHRpb25zLFxufTtcblxuZXhwb3J0IHtcbiAgICBzZXRGYWtlT3B0aW9ucyxcbiAgICBnZXRGYWtlT3B0aW9uc09uY2UsXG59OyJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7QUFHQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFlQSxJQUFJQSxrQkFBSjs7QUFFQSxTQUFTQyxjQUFULENBQXlCQyxPQUF6QixFQUFnRztFQUM1RkYsa0JBQWtCLEdBQUdFLE9BQXJCO0FBQ0g7O0FBRUQsU0FBU0Msa0JBQVQsR0FBeUg7RUFDckgsSUFBTUMsSUFBSSxHQUFHSixrQkFBYjtFQUVBQSxrQkFBa0IsR0FBRyxJQUFyQjtFQUVBLE9BQU9JLElBQVA7QUFDSCJ9
|
package/dist/Testing/index.d.ts
CHANGED
|
@@ -1,17 +1,18 @@
|
|
|
1
|
-
import { Cmd
|
|
2
|
-
import {
|
|
1
|
+
import { Cmd } from "../Cmd";
|
|
2
|
+
import { MessageBase, Nullable, UpdateMap, UpdateReturnType } from "../Types";
|
|
3
|
+
import { RenderWithModelOptions } from "./fakeOptions";
|
|
3
4
|
/**
|
|
4
5
|
* Extracts the messages out of a command.
|
|
5
6
|
* @param cmd The command to process.
|
|
6
7
|
* @returns The array of messages.
|
|
7
8
|
*/
|
|
8
|
-
declare function getOfMsgParams<
|
|
9
|
+
declare function getOfMsgParams<TMessage>(cmd?: Cmd<TMessage>): TMessage[];
|
|
9
10
|
/**
|
|
10
11
|
* Executes all commands and resolves the messages.
|
|
11
12
|
* @param cmd The command to process.
|
|
12
13
|
* @returns The array of processed messages.
|
|
13
14
|
*/
|
|
14
|
-
declare function execCmd<
|
|
15
|
+
declare function execCmd<TMessage>(cmd?: Cmd<TMessage>): Promise<Nullable<TMessage>[]>;
|
|
15
16
|
/**
|
|
16
17
|
* Creates an update function out of an UpdateMap.
|
|
17
18
|
* @param {UpdateMap<TProps, TModel, TMessage>} updateMap The UpdateMap.
|
|
@@ -31,6 +32,15 @@ declare type UpdateArgsFactory<TProps, TModel, TMessage extends MessageBase> = (
|
|
|
31
32
|
* const [msg, model, props] = createUpdateArgs(Msg.myMessage(), { ... }, , { ... });
|
|
32
33
|
*/
|
|
33
34
|
declare function createUpdateArgsFactory<TProps, TModel, TMessage extends MessageBase>(initModel: () => TModel, initProps: () => TProps): UpdateArgsFactory<TProps, TModel, TMessage>;
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
35
|
+
/**
|
|
36
|
+
* Renders a component with a provided model.
|
|
37
|
+
* @template TModel The type of the model.
|
|
38
|
+
* @template TMessage The type of the message discriminated union.
|
|
39
|
+
* @template TResult The return type of the render function.
|
|
40
|
+
* @param {() => TResult} render A function to render the component. Use the `render` function of the used testing library.
|
|
41
|
+
* @param {(TModel | RenderWithModelOptions<TModel, TMessage>)} options The model or an options object.
|
|
42
|
+
* @returns {TResult} The returned value of the `render` function.
|
|
43
|
+
*/
|
|
44
|
+
declare function renderWithModel<TModel, TMessage extends MessageBase, TResult>(render: () => TResult, options: TModel | RenderWithModelOptions<TModel, TMessage>): TResult;
|
|
45
|
+
export type { UpdateArgsFactory, RenderWithModelOptions, };
|
|
46
|
+
export { getOfMsgParams, execCmd, getUpdateFn, createUpdateArgsFactory, renderWithModel, };
|
package/dist/Testing/index.js
CHANGED
|
@@ -9,11 +9,11 @@ exports.createUpdateArgsFactory = createUpdateArgsFactory;
|
|
|
9
9
|
exports.execCmd = execCmd;
|
|
10
10
|
exports.getOfMsgParams = getOfMsgParams;
|
|
11
11
|
exports.getUpdateFn = getUpdateFn;
|
|
12
|
-
exports.
|
|
12
|
+
exports.renderWithModel = renderWithModel;
|
|
13
13
|
|
|
14
14
|
var _useElmish = require("../useElmish");
|
|
15
15
|
|
|
16
|
-
var
|
|
16
|
+
var _fakeOptions = require("./fakeOptions");
|
|
17
17
|
|
|
18
18
|
function _regeneratorRuntime() { "use strict"; /*! regenerator-runtime -- Copyright (c) 2014-present, Facebook, Inc. -- license (MIT): https://github.com/facebook/regenerator/blob/main/LICENSE */ _regeneratorRuntime = function _regeneratorRuntime() { return exports; }; var exports = {}, Op = Object.prototype, hasOwn = Op.hasOwnProperty, $Symbol = "function" == typeof Symbol ? Symbol : {}, iteratorSymbol = $Symbol.iterator || "@@iterator", asyncIteratorSymbol = $Symbol.asyncIterator || "@@asyncIterator", toStringTagSymbol = $Symbol.toStringTag || "@@toStringTag"; function define(obj, key, value) { return Object.defineProperty(obj, key, { value: value, enumerable: !0, configurable: !0, writable: !0 }), obj[key]; } try { define({}, ""); } catch (err) { define = function define(obj, key, value) { return obj[key] = value; }; } function wrap(innerFn, outerFn, self, tryLocsList) { var protoGenerator = outerFn && outerFn.prototype instanceof Generator ? outerFn : Generator, generator = Object.create(protoGenerator.prototype), context = new Context(tryLocsList || []); return generator._invoke = function (innerFn, self, context) { var state = "suspendedStart"; return function (method, arg) { if ("executing" === state) throw new Error("Generator is already running"); if ("completed" === state) { if ("throw" === method) throw arg; return doneResult(); } for (context.method = method, context.arg = arg;;) { var delegate = context.delegate; if (delegate) { var delegateResult = maybeInvokeDelegate(delegate, context); if (delegateResult) { if (delegateResult === ContinueSentinel) continue; return delegateResult; } } if ("next" === context.method) context.sent = context._sent = context.arg;else if ("throw" === context.method) { if ("suspendedStart" === state) throw state = "completed", context.arg; context.dispatchException(context.arg); } else "return" === context.method && context.abrupt("return", context.arg); state = "executing"; var record = tryCatch(innerFn, self, context); if ("normal" === record.type) { if (state = context.done ? "completed" : "suspendedYield", record.arg === ContinueSentinel) continue; return { value: record.arg, done: context.done }; } "throw" === record.type && (state = "completed", context.method = "throw", context.arg = record.arg); } }; }(innerFn, self, context), generator; } function tryCatch(fn, obj, arg) { try { return { type: "normal", arg: fn.call(obj, arg) }; } catch (err) { return { type: "throw", arg: err }; } } exports.wrap = wrap; var ContinueSentinel = {}; function Generator() {} function GeneratorFunction() {} function GeneratorFunctionPrototype() {} var IteratorPrototype = {}; define(IteratorPrototype, iteratorSymbol, function () { return this; }); var getProto = Object.getPrototypeOf, NativeIteratorPrototype = getProto && getProto(getProto(values([]))); NativeIteratorPrototype && NativeIteratorPrototype !== Op && hasOwn.call(NativeIteratorPrototype, iteratorSymbol) && (IteratorPrototype = NativeIteratorPrototype); var Gp = GeneratorFunctionPrototype.prototype = Generator.prototype = Object.create(IteratorPrototype); function defineIteratorMethods(prototype) { ["next", "throw", "return"].forEach(function (method) { define(prototype, method, function (arg) { return this._invoke(method, arg); }); }); } function AsyncIterator(generator, PromiseImpl) { function invoke(method, arg, resolve, reject) { var record = tryCatch(generator[method], generator, arg); if ("throw" !== record.type) { var result = record.arg, value = result.value; return value && "object" == _typeof(value) && hasOwn.call(value, "__await") ? PromiseImpl.resolve(value.__await).then(function (value) { invoke("next", value, resolve, reject); }, function (err) { invoke("throw", err, resolve, reject); }) : PromiseImpl.resolve(value).then(function (unwrapped) { result.value = unwrapped, resolve(result); }, function (error) { return invoke("throw", error, resolve, reject); }); } reject(record.arg); } var previousPromise; this._invoke = function (method, arg) { function callInvokeWithMethodAndArg() { return new PromiseImpl(function (resolve, reject) { invoke(method, arg, resolve, reject); }); } return previousPromise = previousPromise ? previousPromise.then(callInvokeWithMethodAndArg, callInvokeWithMethodAndArg) : callInvokeWithMethodAndArg(); }; } function maybeInvokeDelegate(delegate, context) { var method = delegate.iterator[context.method]; if (undefined === method) { if (context.delegate = null, "throw" === context.method) { if (delegate.iterator["return"] && (context.method = "return", context.arg = undefined, maybeInvokeDelegate(delegate, context), "throw" === context.method)) return ContinueSentinel; context.method = "throw", context.arg = new TypeError("The iterator does not provide a 'throw' method"); } return ContinueSentinel; } var record = tryCatch(method, delegate.iterator, context.arg); if ("throw" === record.type) return context.method = "throw", context.arg = record.arg, context.delegate = null, ContinueSentinel; var info = record.arg; return info ? info.done ? (context[delegate.resultName] = info.value, context.next = delegate.nextLoc, "return" !== context.method && (context.method = "next", context.arg = undefined), context.delegate = null, ContinueSentinel) : info : (context.method = "throw", context.arg = new TypeError("iterator result is not an object"), context.delegate = null, ContinueSentinel); } function pushTryEntry(locs) { var entry = { tryLoc: locs[0] }; 1 in locs && (entry.catchLoc = locs[1]), 2 in locs && (entry.finallyLoc = locs[2], entry.afterLoc = locs[3]), this.tryEntries.push(entry); } function resetTryEntry(entry) { var record = entry.completion || {}; record.type = "normal", delete record.arg, entry.completion = record; } function Context(tryLocsList) { this.tryEntries = [{ tryLoc: "root" }], tryLocsList.forEach(pushTryEntry, this), this.reset(!0); } function values(iterable) { if (iterable) { var iteratorMethod = iterable[iteratorSymbol]; if (iteratorMethod) return iteratorMethod.call(iterable); if ("function" == typeof iterable.next) return iterable; if (!isNaN(iterable.length)) { var i = -1, next = function next() { for (; ++i < iterable.length;) { if (hasOwn.call(iterable, i)) return next.value = iterable[i], next.done = !1, next; } return next.value = undefined, next.done = !0, next; }; return next.next = next; } } return { next: doneResult }; } function doneResult() { return { value: undefined, done: !0 }; } return GeneratorFunction.prototype = GeneratorFunctionPrototype, define(Gp, "constructor", GeneratorFunctionPrototype), define(GeneratorFunctionPrototype, "constructor", GeneratorFunction), GeneratorFunction.displayName = define(GeneratorFunctionPrototype, toStringTagSymbol, "GeneratorFunction"), exports.isGeneratorFunction = function (genFun) { var ctor = "function" == typeof genFun && genFun.constructor; return !!ctor && (ctor === GeneratorFunction || "GeneratorFunction" === (ctor.displayName || ctor.name)); }, exports.mark = function (genFun) { return Object.setPrototypeOf ? Object.setPrototypeOf(genFun, GeneratorFunctionPrototype) : (genFun.__proto__ = GeneratorFunctionPrototype, define(genFun, toStringTagSymbol, "GeneratorFunction")), genFun.prototype = Object.create(Gp), genFun; }, exports.awrap = function (arg) { return { __await: arg }; }, defineIteratorMethods(AsyncIterator.prototype), define(AsyncIterator.prototype, asyncIteratorSymbol, function () { return this; }), exports.AsyncIterator = AsyncIterator, exports.async = function (innerFn, outerFn, self, tryLocsList, PromiseImpl) { void 0 === PromiseImpl && (PromiseImpl = Promise); var iter = new AsyncIterator(wrap(innerFn, outerFn, self, tryLocsList), PromiseImpl); return exports.isGeneratorFunction(outerFn) ? iter : iter.next().then(function (result) { return result.done ? result.value : iter.next(); }); }, defineIteratorMethods(Gp), define(Gp, toStringTagSymbol, "Generator"), define(Gp, iteratorSymbol, function () { return this; }), define(Gp, "toString", function () { return "[object Generator]"; }), exports.keys = function (object) { var keys = []; for (var key in object) { keys.push(key); } return keys.reverse(), function next() { for (; keys.length;) { var key = keys.pop(); if (key in object) return next.value = key, next.done = !1, next; } return next.done = !0, next; }; }, exports.values = values, Context.prototype = { constructor: Context, reset: function reset(skipTempReset) { if (this.prev = 0, this.next = 0, this.sent = this._sent = undefined, this.done = !1, this.delegate = null, this.method = "next", this.arg = undefined, this.tryEntries.forEach(resetTryEntry), !skipTempReset) for (var name in this) { "t" === name.charAt(0) && hasOwn.call(this, name) && !isNaN(+name.slice(1)) && (this[name] = undefined); } }, stop: function stop() { this.done = !0; var rootRecord = this.tryEntries[0].completion; if ("throw" === rootRecord.type) throw rootRecord.arg; return this.rval; }, dispatchException: function dispatchException(exception) { if (this.done) throw exception; var context = this; function handle(loc, caught) { return record.type = "throw", record.arg = exception, context.next = loc, caught && (context.method = "next", context.arg = undefined), !!caught; } for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i], record = entry.completion; if ("root" === entry.tryLoc) return handle("end"); if (entry.tryLoc <= this.prev) { var hasCatch = hasOwn.call(entry, "catchLoc"), hasFinally = hasOwn.call(entry, "finallyLoc"); if (hasCatch && hasFinally) { if (this.prev < entry.catchLoc) return handle(entry.catchLoc, !0); if (this.prev < entry.finallyLoc) return handle(entry.finallyLoc); } else if (hasCatch) { if (this.prev < entry.catchLoc) return handle(entry.catchLoc, !0); } else { if (!hasFinally) throw new Error("try statement without catch or finally"); if (this.prev < entry.finallyLoc) return handle(entry.finallyLoc); } } } }, abrupt: function abrupt(type, arg) { for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; if (entry.tryLoc <= this.prev && hasOwn.call(entry, "finallyLoc") && this.prev < entry.finallyLoc) { var finallyEntry = entry; break; } } finallyEntry && ("break" === type || "continue" === type) && finallyEntry.tryLoc <= arg && arg <= finallyEntry.finallyLoc && (finallyEntry = null); var record = finallyEntry ? finallyEntry.completion : {}; return record.type = type, record.arg = arg, finallyEntry ? (this.method = "next", this.next = finallyEntry.finallyLoc, ContinueSentinel) : this.complete(record); }, complete: function complete(record, afterLoc) { if ("throw" === record.type) throw record.arg; return "break" === record.type || "continue" === record.type ? this.next = record.arg : "return" === record.type ? (this.rval = this.arg = record.arg, this.method = "return", this.next = "end") : "normal" === record.type && afterLoc && (this.next = afterLoc), ContinueSentinel; }, finish: function finish(finallyLoc) { for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; if (entry.finallyLoc === finallyLoc) return this.complete(entry.completion, entry.afterLoc), resetTryEntry(entry), ContinueSentinel; } }, "catch": function _catch(tryLoc) { for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; if (entry.tryLoc === tryLoc) { var record = entry.completion; if ("throw" === record.type) { var thrown = record.arg; resetTryEntry(entry); } return thrown; } } throw new Error("illegal catch attempt"); }, delegateYield: function delegateYield(iterable, resultName, nextLoc) { return this.delegate = { iterator: values(iterable), resultName: resultName, nextLoc: nextLoc }, "next" === this.method && (this.arg = undefined), ContinueSentinel; } }, exports; }
|
|
19
19
|
|
|
@@ -147,11 +147,28 @@ function createUpdateArgsFactory(initModel, initProps) {
|
|
|
147
147
|
return [msg, _objectSpread(_objectSpread({}, initModel()), modelTemplate), _objectSpread(_objectSpread({}, initProps()), propsTemplate)];
|
|
148
148
|
};
|
|
149
149
|
}
|
|
150
|
+
/**
|
|
151
|
+
* Renders a component with a provided model.
|
|
152
|
+
* @template TModel The type of the model.
|
|
153
|
+
* @template TMessage The type of the message discriminated union.
|
|
154
|
+
* @template TResult The return type of the render function.
|
|
155
|
+
* @param {() => TResult} render A function to render the component. Use the `render` function of the used testing library.
|
|
156
|
+
* @param {(TModel | RenderWithModelOptions<TModel, TMessage>)} options The model or an options object.
|
|
157
|
+
* @returns {TResult} The returned value of the `render` function.
|
|
158
|
+
*/
|
|
159
|
+
|
|
160
|
+
|
|
161
|
+
function renderWithModel(render, options) {
|
|
162
|
+
if ("model" in options && "dispatch" in options) {
|
|
163
|
+
(0, _fakeOptions.setFakeOptions)(options);
|
|
164
|
+
} else {
|
|
165
|
+
(0, _fakeOptions.setFakeOptions)({
|
|
166
|
+
model: options
|
|
167
|
+
});
|
|
168
|
+
}
|
|
150
169
|
|
|
151
|
-
function renderWithFake(initResult, render, fakeDispatch) {
|
|
152
|
-
(0, _fakeInitResult.setFakes)(initResult, fakeDispatch !== null && fakeDispatch !== void 0 ? fakeDispatch : null);
|
|
153
170
|
var result = render();
|
|
154
|
-
(0,
|
|
171
|
+
(0, _fakeOptions.setFakeOptions)(null);
|
|
155
172
|
return result;
|
|
156
173
|
}
|
|
157
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
174
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["getOfMsgParams","cmd","msgNames","dispatch","msg","push","map","currentCmd","execCmd","callers","Promise","resolve","reject","error","all","results","getUpdateFn","updateMap","model","props","callUpdateMap","createUpdateArgsFactory","initModel","initProps","modelTemplate","propsTemplate","renderWithModel","render","options","setFakeOptions","result"],"sources":["../../src/Testing/index.ts"],"sourcesContent":["import { Cmd } from \"../Cmd\";\nimport { MessageBase, Nullable, UpdateMap, UpdateReturnType } from \"../Types\";\nimport { callUpdateMap } from \"../useElmish\";\nimport { RenderWithModelOptions, setFakeOptions } from \"./fakeOptions\";\n\n/**\n * Extracts the messages out of a command.\n * @param cmd The command to process.\n * @returns The array of messages.\n */\nfunction getOfMsgParams<TMessage> (cmd?: Cmd<TMessage>): TMessage [] {\n    const msgNames: TMessage [] = [];\n\n    const dispatch = (msg: TMessage): void => {\n        msgNames.push(msg);\n    };\n\n    cmd?.map(currentCmd => currentCmd(dispatch));\n\n    return msgNames;\n}\n\n/**\n * Executes all commands and resolves the messages.\n * @param cmd The command to process.\n * @returns The array of processed messages.\n */\nasync function execCmd<TMessage> (cmd?: Cmd<TMessage>): Promise<Nullable<TMessage> []> {\n    if (!cmd) {\n        return [];\n    }\n\n    const callers = cmd.map(async currentCmd => new Promise<Nullable<TMessage>>((resolve, reject) => {\n        const dispatch = (msg: TMessage): void => resolve(msg);\n\n        currentCmd(dispatch, error => {\n            if (error) {\n                reject(error);\n            } else {\n                resolve(null);\n            }\n        });\n    }));\n\n    const results = await Promise.all(callers);\n\n    return results;\n}\n\n/**\n * Creates an update function out of an UpdateMap.\n * @param {UpdateMap<TProps, TModel, TMessage>} updateMap The UpdateMap.\n * @returns {(msg: TMessage, model: TModel, props: TProps) => UpdateReturnType<TModel, TMessage>} The created update function which can be used in tests.\n */\nfunction getUpdateFn<TProps, TModel, TMessage extends MessageBase> (updateMap: UpdateMap<TProps, TModel, TMessage>): (msg: TMessage, model: TModel, props: TProps) => UpdateReturnType<TModel, TMessage> {\n    return function (msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage> {\n        return callUpdateMap(updateMap, msg, model, props);\n    };\n}\n\ntype UpdateArgsFactory<TProps, TModel, TMessage extends MessageBase> = (msg: TMessage, modelTemplate?: Partial<TModel>, propsTemplate?: Partial<TProps>) => [TMessage, TModel, TProps];\n\n/**\n * Creates a factory function to create a message, a model, and props which can be passed to an update function in tests.\n * @param {() => TModel} initModel A function to create an initial model.\n * @param {() => TProps} initProps A function to create initial props.\n * @returns {UpdateArgsFactory<TProps, TModel, TMessage>} A function to create a message, a model, and props.\n * @example\n * // one time\n * const createUpdateArgs = createUpdateArgsFactory(() => ({ ... }), () => ({ ... }));\n * // in tests\n * const [msg, model, props] = createUpdateArgs(Msg.myMessage(), { ... }, , { ... });\n */\nfunction createUpdateArgsFactory<TProps, TModel, TMessage extends MessageBase> (initModel: () => TModel, initProps: () => TProps): UpdateArgsFactory<TProps, TModel, TMessage> {\n    return function (msg: TMessage, modelTemplate?: Partial<TModel>, propsTemplate?: Partial<TProps>): [TMessage, TModel, TProps] {\n        return [\n            msg,\n            {\n                ...initModel(),\n                ...modelTemplate,\n            },\n            {\n                ...initProps(),\n                ...propsTemplate,\n            },\n        ];\n    };\n}\n\n/**\n * Renders a component with a provided model.\n * @template TModel The type of the model.\n * @template TMessage The type of the message discriminated union.\n * @template TResult The return type of the render function.\n * @param {() => TResult} render A function to render the component. Use the `render` function of the used testing library.\n * @param {(TModel | RenderWithModelOptions<TModel, TMessage>)} options The model or an options object.\n * @returns {TResult} The returned value of the `render` function.\n */\nfunction renderWithModel<TModel, TMessage extends MessageBase, TResult> (render: () => TResult, options: TModel | RenderWithModelOptions<TModel, TMessage>): TResult {\n    if (\"model\" in options && \"dispatch\" in options) {\n        setFakeOptions(options as RenderWithModelOptions<unknown, MessageBase>);\n    } else {\n        setFakeOptions({\n            model: options,\n        });\n    }\n\n    const result = render();\n\n    setFakeOptions(null);\n\n    return result;\n}\n\nexport type {\n    UpdateArgsFactory,\n    RenderWithModelOptions,\n};\n\nexport {\n    getOfMsgParams,\n    execCmd,\n    getUpdateFn,\n    createUpdateArgsFactory,\n    renderWithModel,\n};"],"mappings":";;;;;;;;;;;;;AAEA;;AACA;;+CAFA,oJ;;;;;;;;;;;;AAIA;AACA;AACA;AACA;AACA;AACA,SAASA,cAAT,CAAmCC,GAAnC,EAAqE;EACjE,IAAMC,QAAqB,GAAG,EAA9B;;EAEA,IAAMC,QAAQ,GAAG,SAAXA,QAAW,CAACC,GAAD,EAAyB;IACtCF,QAAQ,CAACG,IAAT,CAAcD,GAAd;EACH,CAFD;;EAIAH,GAAG,SAAH,IAAAA,GAAG,WAAH,YAAAA,GAAG,CAAEK,GAAL,CAAS,UAAAC,UAAU;IAAA,OAAIA,UAAU,CAACJ,QAAD,CAAd;EAAA,CAAnB;EAEA,OAAOD,QAAP;AACH;AAED;AACA;AACA;AACA;AACA;;;SACeM,O;;;AAsBf;AACA;AACA;AACA;AACA;;;;wEA1BA,kBAAkCP,GAAlC;IAAA;IAAA;MAAA;QAAA;UAAA;YAAA,IACSA,GADT;cAAA;cAAA;YAAA;;YAAA,kCAEe,EAFf;;UAAA;YAKUQ,OALV,GAKoBR,GAAG,CAACK,GAAJ;cAAA,sEAAQ,iBAAMC,UAAN;gBAAA;kBAAA;oBAAA;sBAAA;wBAAA,iCAAoB,IAAIG,OAAJ,CAAgC,UAACC,OAAD,EAAUC,MAAV,EAAqB;0BAC7F,IAAMT,QAAQ,GAAG,SAAXA,QAAW,CAACC,GAAD;4BAAA,OAAyBO,OAAO,CAACP,GAAD,CAAhC;0BAAA,CAAjB;;0BAEAG,UAAU,CAACJ,QAAD,EAAW,UAAAU,KAAK,EAAI;4BAC1B,IAAIA,KAAJ,EAAW;8BACPD,MAAM,CAACC,KAAD,CAAN;4BACH,CAFD,MAEO;8BACHF,OAAO,CAAC,IAAD,CAAP;4BACH;0BACJ,CANS,CAAV;wBAOH,CAV2C,CAApB;;sBAAA;sBAAA;wBAAA;oBAAA;kBAAA;gBAAA;cAAA,CAAR;;cAAA;gBAAA;cAAA;YAAA,IALpB;YAAA;YAAA,OAiB0BD,OAAO,CAACI,GAAR,CAAYL,OAAZ,CAjB1B;;UAAA;YAiBUM,OAjBV;YAAA,kCAmBWA,OAnBX;;UAAA;UAAA;YAAA;QAAA;MAAA;IAAA;EAAA,C;;;;AA2BA,SAASC,WAAT,CAAoEC,SAApE,EAAyM;EACrM,OAAO,UAAUb,GAAV,EAAyBc,KAAzB,EAAwCC,KAAxC,EAA2F;IAC9F,OAAO,IAAAC,wBAAA,EAAcH,SAAd,EAAyBb,GAAzB,EAA8Bc,KAA9B,EAAqCC,KAArC,CAAP;EACH,CAFD;AAGH;;AAID;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAASE,uBAAT,CAAgFC,SAAhF,EAAyGC,SAAzG,EAA+K;EAC3K,OAAO,UAAUnB,GAAV,EAAyBoB,aAAzB,EAA0DC,aAA1D,EAAuH;IAC1H,OAAO,CACHrB,GADG,kCAGIkB,SAAS,EAHb,GAIIE,aAJJ,mCAOID,SAAS,EAPb,GAQIE,aARJ,EAAP;EAWH,CAZD;AAaH;AAED;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AACA,SAASC,eAAT,CAAyEC,MAAzE,EAAgGC,OAAhG,EAAqK;EACjK,IAAI,WAAWA,OAAX,IAAsB,cAAcA,OAAxC,EAAiD;IAC7C,IAAAC,2BAAA,EAAeD,OAAf;EACH,CAFD,MAEO;IACH,IAAAC,2BAAA,EAAe;MACXX,KAAK,EAAEU;IADI,CAAf;EAGH;;EAED,IAAME,MAAM,GAAGH,MAAM,EAArB;EAEA,IAAAE,2BAAA,EAAe,IAAf;EAEA,OAAOC,MAAP;AACH"}
|
package/dist/Types.d.ts
CHANGED
|
@@ -9,18 +9,21 @@ interface MessageBase {
|
|
|
9
9
|
interface MsgSource<T extends string> {
|
|
10
10
|
source: T;
|
|
11
11
|
}
|
|
12
|
+
/**
|
|
13
|
+
* The return type of the `init` function.
|
|
14
|
+
*/
|
|
12
15
|
declare type InitResult<TModel, TMessage> = [TModel, Cmd<TMessage>?];
|
|
13
16
|
declare type InitFunction<TProps, TModel, TMessage> = (props: TProps) => InitResult<TModel, TMessage>;
|
|
14
17
|
/**
|
|
15
|
-
* Type for the return value of the update function.
|
|
18
|
+
* Type for the return value of the `update` function.
|
|
16
19
|
*/
|
|
17
|
-
declare type UpdateReturnType<TModel,
|
|
18
|
-
declare type UpdateFunction<TProps, TModel,
|
|
20
|
+
declare type UpdateReturnType<TModel, TMessage> = [Partial<TModel>, Cmd<TMessage>?];
|
|
21
|
+
declare type UpdateFunction<TProps, TModel, TMessage> = (model: TModel, msg: TMessage, props: TProps) => UpdateReturnType<TModel, TMessage>;
|
|
19
22
|
/**
|
|
20
23
|
* Type for mapping messages to functions.
|
|
21
24
|
* Use this type to create your update logic for the useElmish hook.
|
|
22
25
|
*/
|
|
23
|
-
declare type UpdateMap<TProps, TModel,
|
|
24
|
-
[M in
|
|
26
|
+
declare type UpdateMap<TProps, TModel, TMessage extends MessageBase> = {
|
|
27
|
+
[M in TMessage as M["name"]]: (msg: M, model: TModel, props: TProps) => UpdateReturnType<TModel, TMessage>;
|
|
25
28
|
};
|
|
26
29
|
export type { Nullable, MessageBase, MsgSource, InitResult, InitFunction, UpdateReturnType, UpdateFunction, UpdateMap, };
|
package/dist/Types.js
CHANGED
|
@@ -3,4 +3,4 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL1R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCB9IGZyb20gXCIuL0NtZFwiO1xuXG50eXBlIE51bGxhYmxlPFQ+
|
|
6
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL1R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCB9IGZyb20gXCIuL0NtZFwiO1xuXG50eXBlIE51bGxhYmxlPFQ+ID0gVCB8IG51bGw7XG5cbmludGVyZmFjZSBNZXNzYWdlQmFzZSB7XG4gICAgbmFtZTogc3RyaW5nIHwgc3ltYm9sLFxufVxuXG4vKipcbiAqIENyZWF0ZXMgYSBNc2dTb3VyY2UgdHlwZS5cbiAqL1xuaW50ZXJmYWNlIE1zZ1NvdXJjZTxUIGV4dGVuZHMgc3RyaW5nPiB7XG4gICAgc291cmNlOiBULFxufVxuXG4vKipcbiAqIFRoZSByZXR1cm4gdHlwZSBvZiB0aGUgYGluaXRgIGZ1bmN0aW9uLlxuICovXG50eXBlIEluaXRSZXN1bHQ8VE1vZGVsLCBUTWVzc2FnZT4gPSBbVE1vZGVsLCBDbWQ8VE1lc3NhZ2U+P107XG5cbnR5cGUgSW5pdEZ1bmN0aW9uPFRQcm9wcywgVE1vZGVsLCBUTWVzc2FnZT4gPSAocHJvcHM6IFRQcm9wcykgPT4gSW5pdFJlc3VsdDxUTW9kZWwsIFRNZXNzYWdlPjtcblxuLyoqXG4gKiBUeXBlIGZvciB0aGUgcmV0dXJuIHZhbHVlIG9mIHRoZSBgdXBkYXRlYCBmdW5jdGlvbi5cbiAqL1xudHlwZSBVcGRhdGVSZXR1cm5UeXBlPFRNb2RlbCwgVE1lc3NhZ2U+ID0gW1BhcnRpYWw8VE1vZGVsPiwgQ21kPFRNZXNzYWdlPj9dO1xuXG50eXBlIFVwZGF0ZUZ1bmN0aW9uPFRQcm9wcywgVE1vZGVsLCBUTWVzc2FnZT4gPSAobW9kZWw6IFRNb2RlbCwgbXNnOiBUTWVzc2FnZSwgcHJvcHM6IFRQcm9wcykgPT4gVXBkYXRlUmV0dXJuVHlwZTxUTW9kZWwsIFRNZXNzYWdlPjtcblxuLyoqXG4gKiBUeXBlIGZvciBtYXBwaW5nIG1lc3NhZ2VzIHRvIGZ1bmN0aW9ucy5cbiAqIFVzZSB0aGlzIHR5cGUgdG8gY3JlYXRlIHlvdXIgdXBkYXRlIGxvZ2ljIGZvciB0aGUgdXNlRWxtaXNoIGhvb2suXG4gKi9cbnR5cGUgVXBkYXRlTWFwPFRQcm9wcywgVE1vZGVsLCBUTWVzc2FnZSBleHRlbmRzIE1lc3NhZ2VCYXNlPiA9IHtcbiAgICBbTSBpbiBUTWVzc2FnZSBhcyBNW1wibmFtZVwiXV06IChtc2c6IE0sIG1vZGVsOiBUTW9kZWwsIHByb3BzOiBUUHJvcHMpID0+IFVwZGF0ZVJldHVyblR5cGU8VE1vZGVsLCBUTWVzc2FnZT47XG59O1xuXG5leHBvcnQgdHlwZSB7XG4gICAgTnVsbGFibGUsXG4gICAgTWVzc2FnZUJhc2UsXG4gICAgTXNnU291cmNlLFxuICAgIEluaXRSZXN1bHQsXG4gICAgSW5pdEZ1bmN0aW9uLFxuICAgIFVwZGF0ZVJldHVyblR5cGUsXG4gICAgVXBkYXRlRnVuY3Rpb24sXG4gICAgVXBkYXRlTWFwLFxufTsiXSwibWFwcGluZ3MiOiIifQ==
|
package/dist/index.d.ts
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { Cmd, createCmd, Dispatch } from "./Cmd";
|
|
2
|
+
import { ElmComponent } from "./ElmComponent";
|
|
2
3
|
import { errorHandler, ErrorMessage, errorMsg, handleError } from "./ErrorHandling";
|
|
3
4
|
import { init, Logger, Message } from "./Init";
|
|
4
5
|
import { InitResult, MsgSource, UpdateMap, UpdateReturnType } from "./Types";
|
|
5
6
|
import { SubscriptionResult, useElmish } from "./useElmish";
|
|
6
|
-
import { ElmComponent } from "./ElmComponent";
|
|
7
7
|
export type { Logger, Message, Cmd, Dispatch, InitResult, UpdateReturnType, SubscriptionResult, MsgSource, UpdateMap, ErrorMessage, };
|
|
8
8
|
export { init, createCmd, ElmComponent, errorMsg, errorHandler, handleError, useElmish, };
|
package/dist/index.js
CHANGED
|
@@ -48,11 +48,11 @@ Object.defineProperty(exports, "useElmish", {
|
|
|
48
48
|
|
|
49
49
|
var _Cmd = require("./Cmd");
|
|
50
50
|
|
|
51
|
+
var _ElmComponent = require("./ElmComponent");
|
|
52
|
+
|
|
51
53
|
var _ErrorHandling = require("./ErrorHandling");
|
|
52
54
|
|
|
53
55
|
var _Init = require("./Init");
|
|
54
56
|
|
|
55
57
|
var _useElmish = require("./useElmish");
|
|
56
|
-
|
|
57
|
-
var _ElmComponent = require("./ElmComponent");
|
|
58
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgY3JlYXRlQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgZXJyb3JIYW5kbGVyLCBFcnJvck1lc3NhZ2UsIGVycm9yTXNnLCBoYW5kbGVFcnJvciB9IGZyb20gXCIuL0Vycm9ySGFuZGxpbmdcIjtcbmltcG9ydCB7IGluaXQsIExvZ2dlciwgTWVzc2FnZSB9IGZyb20gXCIuL0luaXRcIjtcbmltcG9ydCB7IEluaXRSZXN1bHQsIE1zZ1NvdXJjZSwgVXBkYXRlTWFwLCBVcGRhdGVSZXR1cm5UeXBlIH0gZnJvbSBcIi4vVHlwZXNcIjtcbmltcG9ydCB7IFN1YnNjcmlwdGlvblJlc3VsdCwgdXNlRWxtaXNoIH0gZnJvbSBcIi4vdXNlRWxtaXNoXCI7XG5pbXBvcnQgeyBFbG1Db21wb25lbnQgfSBmcm9tIFwiLi9FbG1Db21wb25lbnRcIjtcblxuZXhwb3J0IHR5cGUge1xuICAgIExvZ2dlcixcbiAgICBNZXNzYWdlLFxuICAgIENtZCxcbiAgICBEaXNwYXRjaCxcbiAgICBJbml0UmVzdWx0LFxuICAgIFVwZGF0ZVJldHVyblR5cGUsXG4gICAgU3Vic2NyaXB0aW9uUmVzdWx0LFxuICAgIE1zZ1NvdXJjZSxcbiAgICBVcGRhdGVNYXAsXG4gICAgRXJyb3JNZXNzYWdlLFxufTtcblxuZXhwb3J0IHtcbiAgICBpbml0LFxuICAgIGNyZWF0ZUNtZCxcbiAgICBFbG1Db21wb25lbnQsXG4gICAgZXJyb3JNc2csXG4gICAgZXJyb3JIYW5kbGVyLFxuICAgIGhhbmRsZUVycm9yLFxuICAgIHVzZUVsbWlzaCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFDQTs7QUFFQTs7QUFDQSJ9
|
|
58
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgY3JlYXRlQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgRWxtQ29tcG9uZW50IH0gZnJvbSBcIi4vRWxtQ29tcG9uZW50XCI7XG5pbXBvcnQgeyBlcnJvckhhbmRsZXIsIEVycm9yTWVzc2FnZSwgZXJyb3JNc2csIGhhbmRsZUVycm9yIH0gZnJvbSBcIi4vRXJyb3JIYW5kbGluZ1wiO1xuaW1wb3J0IHsgaW5pdCwgTG9nZ2VyLCBNZXNzYWdlIH0gZnJvbSBcIi4vSW5pdFwiO1xuaW1wb3J0IHsgSW5pdFJlc3VsdCwgTXNnU291cmNlLCBVcGRhdGVNYXAsIFVwZGF0ZVJldHVyblR5cGUgfSBmcm9tIFwiLi9UeXBlc1wiO1xuaW1wb3J0IHsgU3Vic2NyaXB0aW9uUmVzdWx0LCB1c2VFbG1pc2ggfSBmcm9tIFwiLi91c2VFbG1pc2hcIjtcblxuZXhwb3J0IHR5cGUge1xuICAgIExvZ2dlcixcbiAgICBNZXNzYWdlLFxuICAgIENtZCxcbiAgICBEaXNwYXRjaCxcbiAgICBJbml0UmVzdWx0LFxuICAgIFVwZGF0ZVJldHVyblR5cGUsXG4gICAgU3Vic2NyaXB0aW9uUmVzdWx0LFxuICAgIE1zZ1NvdXJjZSxcbiAgICBVcGRhdGVNYXAsXG4gICAgRXJyb3JNZXNzYWdlLFxufTtcblxuZXhwb3J0IHtcbiAgICBpbml0LFxuICAgIGNyZWF0ZUNtZCxcbiAgICBFbG1Db21wb25lbnQsXG4gICAgZXJyb3JNc2csXG4gICAgZXJyb3JIYW5kbGVyLFxuICAgIGhhbmRsZUVycm9yLFxuICAgIHVzZUVsbWlzaCxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFFQSJ9
|
package/dist/useElmish.d.ts
CHANGED
|
@@ -1,12 +1,43 @@
|
|
|
1
1
|
import { Cmd, Dispatch } from "./Cmd";
|
|
2
2
|
import { InitFunction, MessageBase, UpdateFunction, UpdateMap, UpdateReturnType } from "./Types";
|
|
3
|
+
/**
|
|
4
|
+
* The return type of the `subscription` function.
|
|
5
|
+
* @template TMessage The type of the messages discriminated union.
|
|
6
|
+
*/
|
|
3
7
|
declare type SubscriptionResult<TMessage> = [Cmd<TMessage>, (() => void)?];
|
|
4
8
|
declare type Subscription<TProps, TModel, TMessage> = (model: TModel, props: TProps) => SubscriptionResult<TMessage>;
|
|
9
|
+
/**
|
|
10
|
+
* Options for the `useElmish` hook.
|
|
11
|
+
* @interface UseElmishOptions
|
|
12
|
+
* @template TProps The type of the props.
|
|
13
|
+
* @template TModel The type of the model.
|
|
14
|
+
* @template TMessage The type of the messages discriminated union.
|
|
15
|
+
*/
|
|
5
16
|
interface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {
|
|
17
|
+
/**
|
|
18
|
+
* The name of the component. This is used for logging only.
|
|
19
|
+
* @type {string}
|
|
20
|
+
*/
|
|
6
21
|
name: string;
|
|
22
|
+
/**
|
|
23
|
+
* The props passed to the component.
|
|
24
|
+
* @type {TProps}
|
|
25
|
+
*/
|
|
7
26
|
props: TProps;
|
|
27
|
+
/**
|
|
28
|
+
* The function to initialize the components model. This function is only called once.
|
|
29
|
+
* @type {InitFunction<TProps, TModel, TMessage>}
|
|
30
|
+
*/
|
|
8
31
|
init: InitFunction<TProps, TModel, TMessage>;
|
|
32
|
+
/**
|
|
33
|
+
* The `update` function or update map object.
|
|
34
|
+
* @type {(UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>)}
|
|
35
|
+
*/
|
|
9
36
|
update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>;
|
|
37
|
+
/**
|
|
38
|
+
* The optional `subscription` function. This function is only called once.
|
|
39
|
+
* @type {(UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>)}
|
|
40
|
+
*/
|
|
10
41
|
subscription?: Subscription<TProps, TModel, TMessage>;
|
|
11
42
|
}
|
|
12
43
|
/**
|
package/dist/useElmish.js
CHANGED
|
@@ -7,12 +7,12 @@ exports.callUpdate = callUpdate;
|
|
|
7
7
|
exports.callUpdateMap = callUpdateMap;
|
|
8
8
|
exports.useElmish = useElmish;
|
|
9
9
|
|
|
10
|
-
var _fakeInitResult = require("./Testing/fakeInitResult");
|
|
11
|
-
|
|
12
10
|
var _react = require("react");
|
|
13
11
|
|
|
14
12
|
var _Init = require("./Init");
|
|
15
13
|
|
|
14
|
+
var _fakeOptions = require("./Testing/fakeOptions");
|
|
15
|
+
|
|
16
16
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
17
17
|
|
|
18
18
|
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
@@ -39,6 +39,8 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
|
39
39
|
* const [model, dispatch] = useElmish({ props, init, update, name: "MyComponent" });
|
|
40
40
|
*/
|
|
41
41
|
function useElmish(_ref) {
|
|
42
|
+
var _fakeOptions$dispatch;
|
|
43
|
+
|
|
42
44
|
var name = _ref.name,
|
|
43
45
|
props = _ref.props,
|
|
44
46
|
init = _ref.init,
|
|
@@ -71,8 +73,8 @@ function useElmish(_ref) {
|
|
|
71
73
|
}
|
|
72
74
|
});
|
|
73
75
|
}, []);
|
|
74
|
-
var
|
|
75
|
-
var dispatch = (0, _react.useCallback)(
|
|
76
|
+
var fakeOptions = (0, _fakeOptions.getFakeOptionsOnce)();
|
|
77
|
+
var dispatch = (0, _react.useCallback)((_fakeOptions$dispatch = fakeOptions === null || fakeOptions === void 0 ? void 0 : fakeOptions.dispatch) !== null && _fakeOptions$dispatch !== void 0 ? _fakeOptions$dispatch : function (msg) {
|
|
76
78
|
if (!initializedModel) {
|
|
77
79
|
return;
|
|
78
80
|
}
|
|
@@ -137,9 +139,7 @@ function useElmish(_ref) {
|
|
|
137
139
|
}, []);
|
|
138
140
|
|
|
139
141
|
if (!initializedModel) {
|
|
140
|
-
var
|
|
141
|
-
|
|
142
|
-
var _ref2 = (_getFakeInitResultOnc = (0, _fakeInitResult.getFakeInitResultOnce)()) !== null && _getFakeInitResultOnc !== void 0 ? _getFakeInitResultOnc : init(props),
|
|
142
|
+
var _ref2 = fakeOptions !== null && fakeOptions !== void 0 && fakeOptions.model ? [fakeOptions.model] : init(props),
|
|
143
143
|
_ref3 = _slicedToArray(_ref2, 2),
|
|
144
144
|
initModel = _ref3[0],
|
|
145
145
|
initCmd = _ref3[1];
|
|
@@ -183,4 +183,4 @@ function callUpdateMap(updateMap, msg, model, props) {
|
|
|
183
183
|
var updateFn = updateMap[msg.name];
|
|
184
184
|
return updateFn(msg, model, props);
|
|
185
185
|
}
|
|
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","fakeDispatch","getFakeDispatchOnce","msg","modelHasChanged","updatedModel","Object","getOwnPropertyNames","length","push","nextMsg","modified","info","debug","dispatchMiddleware","callUpdate","newModel","shift","prevModel","getFakeInitResultOnce","initModel","initCmd","useEffect","subCmd","destructor","callUpdateMap","updateMap","updateFn"],"sources":["../src/useElmish.ts"],"sourcesContent":["/* eslint-disable react-hooks/exhaustive-deps */\nimport { Cmd, Dispatch } from \"./Cmd\";\nimport { getFakeDispatchOnce, getFakeInitResultOnce } from \"./Testing/fakeInitResult\";\nimport { InitFunction, MessageBase, Nullable, UpdateFunction, UpdateMap, UpdateReturnType } from \"./Types\";\nimport { useCallback, useEffect, useRef, useState } from \"react\";\nimport { Services } from \"./Init\";\n\ntype SubscriptionResult<TMessage> = [Cmd<TMessage>, (() => void)?];\ntype Subscription<TProps, TModel, TMessage> = (model: TModel, props: TProps) => SubscriptionResult<TMessage>;\n\ninterface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {\n    name: string,\n    props: TProps,\n    init: InitFunction<TProps, TModel, TMessage>,\n    update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>,\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 fakeDispatch = getFakeDispatchOnce();\n    const dispatch = useCallback(fakeDispatch ?? ((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] = getFakeInitResultOnce<TModel, TMessage>() ?? 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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"}
|
|
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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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "react-elmish",
|
|
3
|
-
"version": "4.2.
|
|
3
|
+
"version": "4.2.1",
|
|
4
4
|
"description": "Elmish for React using Typescript",
|
|
5
5
|
"author": "atheck",
|
|
6
6
|
"license": "MIT",
|
|
@@ -10,7 +10,7 @@
|
|
|
10
10
|
"build:js": "babel src --out-dir dist --extensions \".ts,.tsx\" --source-maps inline",
|
|
11
11
|
"test": "jest --coverage",
|
|
12
12
|
"test:watch": "jest --watch --coverage",
|
|
13
|
-
"lint": "npx eslint
|
|
13
|
+
"lint": "npx eslint \"{src,tests}/**/*.ts[x]\"",
|
|
14
14
|
"update": "npx npm-check-updates -i",
|
|
15
15
|
"semantic-release": "semantic-release"
|
|
16
16
|
},
|
|
@@ -25,14 +25,14 @@
|
|
|
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.
|
|
28
|
+
"@types/jest": "28.1.7",
|
|
29
|
+
"@types/react": "18.0.17",
|
|
30
|
+
"eslint": "8.22.0",
|
|
31
|
+
"eslint-config-heck": "1.22.1",
|
|
32
32
|
"jest": "28.1.3",
|
|
33
33
|
"jest-environment-jsdom": "28.1.3",
|
|
34
34
|
"semantic-release": "19.0.3",
|
|
35
|
-
"ts-jest": "28.0.
|
|
35
|
+
"ts-jest": "28.0.8",
|
|
36
36
|
"typescript": "4.7.4"
|
|
37
37
|
},
|
|
38
38
|
"homepage": "https://github.com/atheck/react-elmish",
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
import { InitResult, MessageBase, Nullable } from "../Types";
|
|
2
|
-
import { Dispatch } from "../Cmd";
|
|
3
|
-
declare type FakeInitResult = Nullable<[unknown, unknown?]>;
|
|
4
|
-
declare function setFakes(fakeInitResult: FakeInitResult, fakeDispatch?: Nullable<Dispatch<MessageBase>>): void;
|
|
5
|
-
declare function getFakeInitResultOnce<TModel, TMessage extends MessageBase>(): Nullable<InitResult<TModel, TMessage>>;
|
|
6
|
-
declare function getFakeDispatchOnce<TMessage extends MessageBase>(): Nullable<Dispatch<TMessage>>;
|
|
7
|
-
export { setFakes, getFakeInitResultOnce, getFakeDispatchOnce, };
|
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
|
|
3
|
-
Object.defineProperty(exports, "__esModule", {
|
|
4
|
-
value: true
|
|
5
|
-
});
|
|
6
|
-
exports.getFakeDispatchOnce = getFakeDispatchOnce;
|
|
7
|
-
exports.getFakeInitResultOnce = getFakeInitResultOnce;
|
|
8
|
-
exports.setFakes = setFakes;
|
|
9
|
-
var currentFakeInitResult;
|
|
10
|
-
var currentFakeDispatch;
|
|
11
|
-
|
|
12
|
-
function setFakes(fakeInitResult, fakeDispatch) {
|
|
13
|
-
currentFakeInitResult = fakeInitResult;
|
|
14
|
-
currentFakeDispatch = fakeDispatch !== null && fakeDispatch !== void 0 ? fakeDispatch : null;
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
function getFakeInitResultOnce() {
|
|
18
|
-
var temp = currentFakeInitResult;
|
|
19
|
-
currentFakeInitResult = null;
|
|
20
|
-
return temp;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
function getFakeDispatchOnce() {
|
|
24
|
-
var temp = currentFakeDispatch;
|
|
25
|
-
currentFakeDispatch = null;
|
|
26
|
-
return temp;
|
|
27
|
-
}
|
|
28
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjdXJyZW50RmFrZUluaXRSZXN1bHQiLCJjdXJyZW50RmFrZURpc3BhdGNoIiwic2V0RmFrZXMiLCJmYWtlSW5pdFJlc3VsdCIsImZha2VEaXNwYXRjaCIsImdldEZha2VJbml0UmVzdWx0T25jZSIsInRlbXAiLCJnZXRGYWtlRGlzcGF0Y2hPbmNlIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL1Rlc3RpbmcvZmFrZUluaXRSZXN1bHQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5pdFJlc3VsdCwgTWVzc2FnZUJhc2UsIE51bGxhYmxlIH0gZnJvbSBcIi4uL1R5cGVzXCI7XG5pbXBvcnQgeyBEaXNwYXRjaCB9IGZyb20gXCIuLi9DbWRcIjtcblxudHlwZSBGYWtlSW5pdFJlc3VsdCA9IE51bGxhYmxlPFt1bmtub3duLCB1bmtub3duP10+O1xuXG5sZXQgY3VycmVudEZha2VJbml0UmVzdWx0OiBGYWtlSW5pdFJlc3VsdDtcbmxldCBjdXJyZW50RmFrZURpc3BhdGNoOiBOdWxsYWJsZTxEaXNwYXRjaDxNZXNzYWdlQmFzZT4+O1xuXG5mdW5jdGlvbiBzZXRGYWtlcyAoZmFrZUluaXRSZXN1bHQ6IEZha2VJbml0UmVzdWx0LCBmYWtlRGlzcGF0Y2g/OiBOdWxsYWJsZTxEaXNwYXRjaDxNZXNzYWdlQmFzZT4+KTogdm9pZCB7XG4gICAgY3VycmVudEZha2VJbml0UmVzdWx0ID0gZmFrZUluaXRSZXN1bHQ7XG4gICAgY3VycmVudEZha2VEaXNwYXRjaCA9IGZha2VEaXNwYXRjaCA/PyBudWxsO1xufVxuXG5mdW5jdGlvbiBnZXRGYWtlSW5pdFJlc3VsdE9uY2U8VE1vZGVsLCBUTWVzc2FnZSBleHRlbmRzIE1lc3NhZ2VCYXNlPiAoKTogTnVsbGFibGU8SW5pdFJlc3VsdDxUTW9kZWwsIFRNZXNzYWdlPj4ge1xuICAgIGNvbnN0IHRlbXAgPSBjdXJyZW50RmFrZUluaXRSZXN1bHQgYXMgTnVsbGFibGU8SW5pdFJlc3VsdDxUTW9kZWwsIFRNZXNzYWdlPj47XG5cbiAgICBjdXJyZW50RmFrZUluaXRSZXN1bHQgPSBudWxsO1xuXG4gICAgcmV0dXJuIHRlbXA7XG59XG5cbmZ1bmN0aW9uIGdldEZha2VEaXNwYXRjaE9uY2U8VE1lc3NhZ2UgZXh0ZW5kcyBNZXNzYWdlQmFzZT4gKCk6IE51bGxhYmxlPERpc3BhdGNoPFRNZXNzYWdlPj4ge1xuICAgIGNvbnN0IHRlbXAgPSBjdXJyZW50RmFrZURpc3BhdGNoO1xuXG4gICAgY3VycmVudEZha2VEaXNwYXRjaCA9IG51bGw7XG5cbiAgICByZXR1cm4gdGVtcDtcbn1cblxuZXhwb3J0IHtcbiAgICBzZXRGYWtlcyxcbiAgICBnZXRGYWtlSW5pdFJlc3VsdE9uY2UsXG4gICAgZ2V0RmFrZURpc3BhdGNoT25jZSxcbn07Il0sIm1hcHBpbmdzIjoiOzs7Ozs7OztBQUtBLElBQUlBLHFCQUFKO0FBQ0EsSUFBSUMsbUJBQUo7O0FBRUEsU0FBU0MsUUFBVCxDQUFtQkMsY0FBbkIsRUFBbURDLFlBQW5ELEVBQXlHO0VBQ3JHSixxQkFBcUIsR0FBR0csY0FBeEI7RUFDQUYsbUJBQW1CLEdBQUdHLFlBQUgsYUFBR0EsWUFBSCxjQUFHQSxZQUFILEdBQW1CLElBQXRDO0FBQ0g7O0FBRUQsU0FBU0MscUJBQVQsR0FBZ0g7RUFDNUcsSUFBTUMsSUFBSSxHQUFHTixxQkFBYjtFQUVBQSxxQkFBcUIsR0FBRyxJQUF4QjtFQUVBLE9BQU9NLElBQVA7QUFDSDs7QUFFRCxTQUFTQyxtQkFBVCxHQUE0RjtFQUN4RixJQUFNRCxJQUFJLEdBQUdMLG1CQUFiO0VBRUFBLG1CQUFtQixHQUFHLElBQXRCO0VBRUEsT0FBT0ssSUFBUDtBQUNIIn0=
|