react-elmish 3.0.0 → 3.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +177 -73
- package/dist/Cmd.d.ts +5 -0
- package/dist/Cmd.js +12 -1
- package/dist/ElmComponent.d.ts +4 -2
- package/dist/ElmComponent.js +1 -1
- package/dist/index.d.ts +5 -5
- package/dist/index.js +7 -1
- package/dist/legacy/useElmish.d.ts +3 -3
- package/dist/legacy/useElmish.js +5 -2
- package/dist/legacy/useElmishMap.d.ts +3 -2
- package/dist/legacy/useElmishMap.js +5 -2
- package/dist/useElmish.d.ts +7 -4
- package/dist/useElmish.js +22 -4
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -25,7 +25,7 @@ An elmish component basically consists of the following parts:
|
|
|
25
25
|
First import everything from `react-elmish` and declare the **Message** discriminated union type:
|
|
26
26
|
|
|
27
27
|
```ts
|
|
28
|
-
import { Cmd, createCmd, UpdateReturnType, UpdateMap } from "react-elmish";
|
|
28
|
+
import { Cmd, createCmd, InitResult, UpdateReturnType, UpdateMap } from "react-elmish";
|
|
29
29
|
|
|
30
30
|
export type Message =
|
|
31
31
|
| { name: "increment" }
|
|
@@ -57,21 +57,14 @@ export interface Props {
|
|
|
57
57
|
}
|
|
58
58
|
```
|
|
59
59
|
|
|
60
|
-
Now we create a `cmd` object for our messages type:
|
|
61
|
-
|
|
62
|
-
```ts
|
|
63
|
-
const cmd = createCmd<Message>();
|
|
64
|
-
```
|
|
65
|
-
|
|
66
60
|
To create the initial model we need an **init** function:
|
|
67
61
|
|
|
68
62
|
```ts
|
|
69
|
-
export function init (props: Props):
|
|
63
|
+
export function init (props: Props): InitResult {
|
|
70
64
|
return [
|
|
71
65
|
{
|
|
72
66
|
value: props.initialValue,
|
|
73
|
-
}
|
|
74
|
-
cmd.none
|
|
67
|
+
}
|
|
75
68
|
];
|
|
76
69
|
};
|
|
77
70
|
```
|
|
@@ -251,75 +244,19 @@ You can also use **Symbols** for the message type instead of strings:
|
|
|
251
244
|
...
|
|
252
245
|
```
|
|
253
246
|
|
|
254
|
-
##
|
|
247
|
+
## Dispatch commands in the update map or update function
|
|
255
248
|
|
|
256
|
-
|
|
249
|
+
In addition to modifying the model, you can dispatch new commands here.
|
|
257
250
|
|
|
258
|
-
|
|
259
|
-
import * as Elm from "react-elmish";
|
|
251
|
+
To do so, you have to create a `cmd` object:
|
|
260
252
|
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
console.debug(...args);
|
|
264
|
-
},
|
|
265
|
-
info(...args: unknown []) {
|
|
266
|
-
console.info(...args);
|
|
267
|
-
},
|
|
268
|
-
error(...args: unknown []) {
|
|
269
|
-
console.error(...args);
|
|
270
|
-
},
|
|
271
|
-
}
|
|
253
|
+
```ts
|
|
254
|
+
import { createCmd } from "react-elmish";
|
|
272
255
|
|
|
273
|
-
|
|
274
|
-
logger: myLogger,
|
|
275
|
-
errorMiddleware: error => Toast.error(error.message),
|
|
276
|
-
dispatchMiddleware: msg => console.log(msg),
|
|
277
|
-
});
|
|
256
|
+
const cmd = createCmd<Message>();
|
|
278
257
|
```
|
|
279
258
|
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
The dispatch middleware function is called whenever a Message is dispatched.
|
|
283
|
-
|
|
284
|
-
## Error handling
|
|
285
|
-
|
|
286
|
-
You can handle errors easily with the following pattern.
|
|
287
|
-
|
|
288
|
-
1. Add an error message:
|
|
289
|
-
|
|
290
|
-
```ts
|
|
291
|
-
import { ErrorMessage, errorMsg, handleError } from "react-elmish";
|
|
292
|
-
|
|
293
|
-
export type Message =
|
|
294
|
-
| ...
|
|
295
|
-
| ErrorMessage;
|
|
296
|
-
```
|
|
297
|
-
|
|
298
|
-
1. Optionally add the convenient function to the **Msg** object:
|
|
299
|
-
|
|
300
|
-
```ts
|
|
301
|
-
export const Msg = {
|
|
302
|
-
...
|
|
303
|
-
...errorMsg,
|
|
304
|
-
}
|
|
305
|
-
```
|
|
306
|
-
|
|
307
|
-
1. Handle the error message in the **update** function:
|
|
308
|
-
|
|
309
|
-
```ts
|
|
310
|
-
...
|
|
311
|
-
case "error":
|
|
312
|
-
return handleError(msg.error);
|
|
313
|
-
...
|
|
314
|
-
```
|
|
315
|
-
|
|
316
|
-
The **handleError** function then calls your error handling middleware.
|
|
317
|
-
|
|
318
|
-
## Dispatch commands in the update map or update function
|
|
319
|
-
|
|
320
|
-
In addition to modifying the model, you can dispatch new commands here.
|
|
321
|
-
|
|
322
|
-
To do so, you can call one of the functions in the `cmd` object:
|
|
259
|
+
Then you can call one of the functions of that object:
|
|
323
260
|
|
|
324
261
|
| Function | Description |
|
|
325
262
|
|---|---|
|
|
@@ -332,6 +269,7 @@ To do so, you can call one of the functions in the `cmd` object:
|
|
|
332
269
|
| `cmd.ofPromise.either` | Calls an async function and maps the result into a message. |
|
|
333
270
|
| `cmd.ofPromise.attempt` | Like `either` but ignores the success case. |
|
|
334
271
|
| `cmd.ofPromise.perform` | Like `either` but ignores the error case. |
|
|
272
|
+
| `cmd.ofSub` | Use this function to trigger a command in a subscription. |
|
|
335
273
|
|
|
336
274
|
### Dispatch a message
|
|
337
275
|
|
|
@@ -348,6 +286,8 @@ export const Msg = {
|
|
|
348
286
|
printLastMessage: (message: string): Message => ({ name: "printLastMessage", message }),
|
|
349
287
|
...
|
|
350
288
|
}
|
|
289
|
+
|
|
290
|
+
const cmd = createCmd<Message>();
|
|
351
291
|
```
|
|
352
292
|
|
|
353
293
|
In the **update** function you can dispatch that message like this:
|
|
@@ -408,6 +348,170 @@ case "error":
|
|
|
408
348
|
...
|
|
409
349
|
```
|
|
410
350
|
|
|
351
|
+
### Dispatch a command from `init`
|
|
352
|
+
|
|
353
|
+
The same way as in the `update` map or function, you can also dispatch an initial command in the `init` function:
|
|
354
|
+
|
|
355
|
+
```ts
|
|
356
|
+
export function init (props: Props): InitResult {
|
|
357
|
+
return [
|
|
358
|
+
{
|
|
359
|
+
value: props.initialValue,
|
|
360
|
+
},
|
|
361
|
+
cmd.ofMsg(Msg.loadData())
|
|
362
|
+
];
|
|
363
|
+
};
|
|
364
|
+
```
|
|
365
|
+
|
|
366
|
+
## Subscriptions
|
|
367
|
+
|
|
368
|
+
### Working with external sources of events
|
|
369
|
+
|
|
370
|
+
If you want to use external sources of events (e.g. a timer), you can use a `subscription`. With this those events can be processed by our `update` handler.
|
|
371
|
+
|
|
372
|
+
Let's define a `Model` and a `Message`:
|
|
373
|
+
|
|
374
|
+
```ts
|
|
375
|
+
type Message =
|
|
376
|
+
| { name: "timer", date: Date };
|
|
377
|
+
|
|
378
|
+
interface Model {
|
|
379
|
+
date: Date,
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
const Msg = {
|
|
383
|
+
timer: (date: Date): Message => ({ name: "timer", date }),
|
|
384
|
+
};
|
|
385
|
+
```
|
|
386
|
+
|
|
387
|
+
Now we define the `init` function and the `update` object:
|
|
388
|
+
|
|
389
|
+
```ts
|
|
390
|
+
const cmd = createCmd<Message>();
|
|
391
|
+
|
|
392
|
+
function init (props: Props): InitResult<Model, Message> {
|
|
393
|
+
return [{
|
|
394
|
+
date: new Date(),
|
|
395
|
+
}];
|
|
396
|
+
}
|
|
397
|
+
|
|
398
|
+
const update: UpdateMap<Props, Model, Message> = {
|
|
399
|
+
timer ({ date }) {
|
|
400
|
+
return [{ date }];
|
|
401
|
+
},
|
|
402
|
+
};
|
|
403
|
+
```
|
|
404
|
+
|
|
405
|
+
Then we write our `subscription` function:
|
|
406
|
+
|
|
407
|
+
```ts
|
|
408
|
+
function subscription (model: Model): SubscriptionResult<Message> {
|
|
409
|
+
const sub = (dispatch: Dispatch<Message>): void => {
|
|
410
|
+
setInterval(() => dispatch(Msg.timer(new Date())), 1000) as unknown as number;
|
|
411
|
+
}
|
|
412
|
+
|
|
413
|
+
return [cmd.ofSub(sub)];
|
|
414
|
+
}
|
|
415
|
+
```
|
|
416
|
+
|
|
417
|
+
This function gets the initialized model as parameter and returns a command.
|
|
418
|
+
|
|
419
|
+
In the function component we call `useElmish` and pass the subscription to it:
|
|
420
|
+
|
|
421
|
+
```ts
|
|
422
|
+
const [{ date }] = useElmish({ name: "Subscriptions", props, init, update, subscription })
|
|
423
|
+
```
|
|
424
|
+
|
|
425
|
+
You can define and aggregate multiple subscriptions with a call to `cmd.batch(...)`.
|
|
426
|
+
|
|
427
|
+
### Cleanup subscriptions
|
|
428
|
+
|
|
429
|
+
In the solution above `setInterval` will trigger events even if the component is removed from the DOM. To cleanup subscriptions, we can return a `destructor` function from the subscription the same as in the `useEffect` hook.
|
|
430
|
+
|
|
431
|
+
Let's rewrite our `subscription` function:
|
|
432
|
+
|
|
433
|
+
```ts
|
|
434
|
+
function subscription (model: Model): SubscriptionResult<Message> {
|
|
435
|
+
let timer: NodeJS.Timer;
|
|
436
|
+
|
|
437
|
+
const sub = (dispatch: Dispatch<Message>): void => {
|
|
438
|
+
timer = setInterval(() => dispatch(Msg.timer(new Date())), 1000);
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
const destructor = () => {
|
|
442
|
+
clearInterval(timer1);
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
return [cmd.ofSub(sub), destructor];
|
|
446
|
+
}
|
|
447
|
+
```
|
|
448
|
+
|
|
449
|
+
Here we save the return value of `setInterval` and clear that interval in the returned `destructor` function.
|
|
450
|
+
|
|
451
|
+
## Setup
|
|
452
|
+
|
|
453
|
+
**react-elmish** works without a setup. But if you want to use logging or some middleware, you can setup **react-elmish** at the start of your program.
|
|
454
|
+
|
|
455
|
+
```ts
|
|
456
|
+
import { init } from "react-elmish";
|
|
457
|
+
|
|
458
|
+
const myLogger = {
|
|
459
|
+
debug(...args: unknown []) {
|
|
460
|
+
console.debug(...args);
|
|
461
|
+
},
|
|
462
|
+
info(...args: unknown []) {
|
|
463
|
+
console.info(...args);
|
|
464
|
+
},
|
|
465
|
+
error(...args: unknown []) {
|
|
466
|
+
console.error(...args);
|
|
467
|
+
},
|
|
468
|
+
}
|
|
469
|
+
|
|
470
|
+
init({
|
|
471
|
+
logger: myLogger,
|
|
472
|
+
errorMiddleware: error => Toast.error(error.message),
|
|
473
|
+
dispatchMiddleware: msg => myLogger.debug(msg),
|
|
474
|
+
});
|
|
475
|
+
```
|
|
476
|
+
|
|
477
|
+
The error middleware function is called by the `handleError` function (see [Error handling](#error-handling)).
|
|
478
|
+
|
|
479
|
+
The dispatch middleware function is called whenever a Message is dispatched.
|
|
480
|
+
|
|
481
|
+
## Error handling
|
|
482
|
+
|
|
483
|
+
You can handle errors easily with the following pattern.
|
|
484
|
+
|
|
485
|
+
1. Add an error message:
|
|
486
|
+
|
|
487
|
+
```ts
|
|
488
|
+
import { ErrorMessage, errorMsg, handleError } from "react-elmish";
|
|
489
|
+
|
|
490
|
+
export type Message =
|
|
491
|
+
| ...
|
|
492
|
+
| ErrorMessage;
|
|
493
|
+
```
|
|
494
|
+
|
|
495
|
+
1. Optionally add the convenient function to the **Msg** object:
|
|
496
|
+
|
|
497
|
+
```ts
|
|
498
|
+
export const Msg = {
|
|
499
|
+
...
|
|
500
|
+
...errorMsg,
|
|
501
|
+
}
|
|
502
|
+
```
|
|
503
|
+
|
|
504
|
+
1. Handle the error message in the **update** function:
|
|
505
|
+
|
|
506
|
+
```ts
|
|
507
|
+
...
|
|
508
|
+
case "error":
|
|
509
|
+
return handleError(msg.error);
|
|
510
|
+
...
|
|
511
|
+
```
|
|
512
|
+
|
|
513
|
+
The **handleError** function then calls your error handling middleware.
|
|
514
|
+
|
|
411
515
|
## React life cycle management
|
|
412
516
|
|
|
413
517
|
If you want to use `componentDidMount` or `componentWillUnmount` in a class component, don't forget to call the base class implementation of it as the **ElmComponent** is using them internally.
|
package/dist/Cmd.d.ts
CHANGED
|
@@ -28,6 +28,11 @@ declare class Command<TMsg> {
|
|
|
28
28
|
* @param {Cmd<TMsg> []} commands Array of commands.
|
|
29
29
|
*/
|
|
30
30
|
batch(...commands: Cmd<TMsg>[]): Cmd<TMsg>;
|
|
31
|
+
/**
|
|
32
|
+
* Command to call the subscriber.
|
|
33
|
+
* @param {Sub<TMsg>} sub The subscriber function.
|
|
34
|
+
*/
|
|
35
|
+
ofSub(sub: Sub<TMsg>): Cmd<TMsg>;
|
|
31
36
|
/**
|
|
32
37
|
* Provides functionalities to create commands from simple functions.
|
|
33
38
|
*/
|
package/dist/Cmd.js
CHANGED
|
@@ -213,6 +213,17 @@ var Command = /*#__PURE__*/function () {
|
|
|
213
213
|
|
|
214
214
|
return commands.flat();
|
|
215
215
|
}
|
|
216
|
+
/**
|
|
217
|
+
* Command to call the subscriber.
|
|
218
|
+
* @param {Sub<TMsg>} sub The subscriber function.
|
|
219
|
+
*/
|
|
220
|
+
// eslint-disable-next-line class-methods-use-this
|
|
221
|
+
|
|
222
|
+
}, {
|
|
223
|
+
key: "ofSub",
|
|
224
|
+
value: function ofSub(sub) {
|
|
225
|
+
return [sub];
|
|
226
|
+
}
|
|
216
227
|
/**
|
|
217
228
|
* Provides functionalities to create commands from simple functions.
|
|
218
229
|
*/
|
|
@@ -231,4 +242,4 @@ var Command = /*#__PURE__*/function () {
|
|
|
231
242
|
function createCmd() {
|
|
232
243
|
return new Command();
|
|
233
244
|
}
|
|
234
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Cmd.ts"],"names":["Command","either","task","ofSuccess","ofError","args","bind","dispatch","result","ex","perform","fallback","attempt","then","defaultFallbackHandler","msg","commands","flat","createCmd"],"mappings":"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","sourcesContent":["/**\n * Type of the dispatch function.\n */\nexport type 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 */\nexport type Cmd<TMsg> = Sub<TMsg> [];\n\n/**\n * Class to create commands.\n * @class Command\n * @template TMsg Type of the Msg discriminated union.\n */\nclass Command<TMsg> {\n    /**\n     * Represents an empty command.\n     */\n    public none = [];\n\n    /**\n     * Creates a command out of a specific message.\n     * @param {TMsg} msg The specific message.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public ofMsg (msg: TMsg): Cmd<TMsg> {\n        return [dispatch => dispatch(msg)];\n    }\n\n    /**\n     * Aggregates multiple commands.\n     * @param {Cmd<TMsg> []} commands Array of commands.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public batch (...commands: Cmd<TMsg> []): Cmd<TMsg> {\n        return commands.flat();\n    }\n\n    /**\n     * Provides functionalities to create commands from simple functions.\n     */\n    public 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            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\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            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\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            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\n    /**\n     * Provides functionalities to create commands from async functions.\n     */\n    public 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            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\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            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\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            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/**\n * Creates a typed instance of the Command class.\n * @template TMsg The type of the Msg discriminated union.\n * @see Command\n */\nexport function createCmd<TMsg> (): Command<TMsg> {\n    return new Command<TMsg>();\n}"]}
|
|
245
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Cmd.ts"],"names":["Command","either","task","ofSuccess","ofError","args","bind","dispatch","result","ex","perform","fallback","attempt","then","defaultFallbackHandler","msg","commands","flat","sub","createCmd"],"mappings":"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","sourcesContent":["/**\n * Type of the dispatch function.\n */\nexport type 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 */\nexport type Cmd<TMsg> = Sub<TMsg> [];\n\n/**\n * Class to create commands.\n * @class Command\n * @template TMsg Type of the Msg discriminated union.\n */\nclass Command<TMsg> {\n    /**\n     * Represents an empty command.\n     */\n    public none = [];\n\n    /**\n     * Creates a command out of a specific message.\n     * @param {TMsg} msg The specific message.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public ofMsg (msg: TMsg): Cmd<TMsg> {\n        return [dispatch => dispatch(msg)];\n    }\n\n    /**\n     * Aggregates multiple commands.\n     * @param {Cmd<TMsg> []} commands Array of commands.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public batch (...commands: Cmd<TMsg> []): Cmd<TMsg> {\n        return commands.flat();\n    }\n\n    /**\n     * Command to call the subscriber.\n     * @param {Sub<TMsg>} sub The subscriber function.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public ofSub (sub: Sub<TMsg>): Cmd<TMsg> {\n        return [sub];\n    }\n\n    /**\n     * Provides functionalities to create commands from simple functions.\n     */\n    public 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            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\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            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\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            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\n    /**\n     * Provides functionalities to create commands from async functions.\n     */\n    public 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            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\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            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\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            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/**\n * Creates a typed instance of the Command class.\n * @template TMsg The type of the Msg discriminated union.\n * @see Command\n */\nexport function createCmd<TMsg> (): Command<TMsg> {\n    return new Command<TMsg>();\n}"]}
|
package/dist/ElmComponent.d.ts
CHANGED
|
@@ -25,7 +25,7 @@ export declare abstract class ElmComponent<TModel, TMsg extends Message, TProps>
|
|
|
25
25
|
* @param name The name of the component.
|
|
26
26
|
* @memberof ElmComponent
|
|
27
27
|
*/
|
|
28
|
-
constructor(props: TProps, init:
|
|
28
|
+
constructor(props: TProps, init: InitFunction<TProps, TModel, TMsg>, 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.
|
|
@@ -63,8 +63,10 @@ export declare abstract class ElmComponent<TModel, TMsg extends Message, TProps>
|
|
|
63
63
|
*/
|
|
64
64
|
abstract update: UpdateFunction<TProps, TModel, TMsg>;
|
|
65
65
|
}
|
|
66
|
-
export declare type
|
|
66
|
+
export declare type InitResult<TModel, TMessage> = [TModel, Cmd<TMessage>?];
|
|
67
|
+
export declare type InitFunction<TProps, TModel, TMessage> = (props: TProps) => InitResult<TModel, TMessage>;
|
|
67
68
|
/**
|
|
68
69
|
* Type for the return value of the update function.
|
|
69
70
|
*/
|
|
70
71
|
export declare type UpdateReturnType<TModel, TMsg> = [Partial<TModel>, Cmd<TMsg>?];
|
|
72
|
+
export declare type UpdateFunction<TProps, TModel, TMsg> = (model: TModel, msg: TMsg, props: TProps) => UpdateReturnType<TModel, TMsg>;
|
package/dist/ElmComponent.js
CHANGED
|
@@ -222,4 +222,4 @@ var ElmComponent = /*#__PURE__*/function (_React$Component) {
|
|
|
222
222
|
}(_react["default"].Component);
|
|
223
223
|
|
|
224
224
|
exports.ElmComponent = ElmComponent;
|
|
225
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ElmComponent.ts"],"names":["ElmComponent","props","init","name","msg","modelHasChanged","model","currentModel","Object","getOwnPropertyNames","length","dispatchMiddleware","reentered","buffer","push","nextMsg","modified","LoggerService","info","componentName","debug","update","cmd","execCmd","ex","error","shift","mounted","forceUpdate","initCmd","forEach","call","dispatch","React","Component"],"mappings":"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","sourcesContent":["import { dispatchMiddleware, LoggerService, Message } from \"./Init\";\nimport { Cmd } from \"./Cmd\";\nimport { Nullable } from \"./ElmUtilities\";\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 */\nexport abstract 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: (arg: TProps) => [TModel, Cmd<TMsg> | undefined], name: string) {\n        super(props);\n\n        const [model, cmd] = 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                LoggerService?.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 (dispatchMiddleware) {\n            dispatchMiddleware(msg);\n        }\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                LoggerService?.info(\"Elm\", \"message from\", this.componentName, nextMsg.name);\n                LoggerService?.debug(\"Elm\", \"message from\", this.componentName, nextMsg);\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                    LoggerService?.error(ex);\n                }\n\n                nextMsg = this.buffer.shift();\n            }\n            this.reentered = false;\n\n            if (this.mounted && modified) {\n                LoggerService?.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 type UpdateFunction<TProps, TModel, TMsg> = (model: TModel, msg: TMsg, props: TProps) => UpdateReturnType<TModel, TMsg>;\n\n/**\n * Type for the return value of the update function.\n */\nexport type UpdateReturnType<TModel, TMsg> = [Partial<TModel>, Cmd<TMsg>?];"]}
|
|
225
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ElmComponent.ts"],"names":["ElmComponent","props","init","name","msg","modelHasChanged","model","currentModel","Object","getOwnPropertyNames","length","dispatchMiddleware","reentered","buffer","push","nextMsg","modified","LoggerService","info","componentName","debug","update","cmd","execCmd","ex","error","shift","mounted","forceUpdate","initCmd","forEach","call","dispatch","React","Component"],"mappings":"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","sourcesContent":["import { dispatchMiddleware, LoggerService, Message } from \"./Init\";\nimport { Cmd } from \"./Cmd\";\nimport { Nullable } from \"./ElmUtilities\";\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 */\nexport abstract 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] = 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                LoggerService?.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 (dispatchMiddleware) {\n            dispatchMiddleware(msg);\n        }\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                LoggerService?.info(\"Elm\", \"message from\", this.componentName, nextMsg.name);\n                LoggerService?.debug(\"Elm\", \"message from\", this.componentName, nextMsg);\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                    LoggerService?.error(ex);\n                }\n\n                nextMsg = this.buffer.shift();\n            }\n            this.reentered = false;\n\n            if (this.mounted && modified) {\n                LoggerService?.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 type InitResult<TModel, TMessage> = [TModel, Cmd<TMessage>?];\n\nexport type InitFunction<TProps, TModel, TMessage> = (props: TProps) => InitResult<TModel, TMessage>;\n\n/**\n * Type for the return value of the update function.\n */\nexport type UpdateReturnType<TModel, TMsg> = [Partial<TModel>, Cmd<TMsg>?];\n\nexport type UpdateFunction<TProps, TModel, TMsg> = (model: TModel, msg: TMsg, props: TProps) => UpdateReturnType<TModel, TMsg>;"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { Cmd, createCmd, Dispatch } from "./Cmd";
|
|
2
|
-
import { ElmComponent, UpdateReturnType } from "./ElmComponent";
|
|
3
|
-
import { ErrorMessage, handleError, MsgSource, UpdateMap } from "./ElmUtilities";
|
|
2
|
+
import { ElmComponent, InitResult, UpdateReturnType } from "./ElmComponent";
|
|
3
|
+
import { ErrorMessage, errorMsg, handleError, MsgSource, UpdateMap } from "./ElmUtilities";
|
|
4
4
|
import { init, Logger, Message } from "./Init";
|
|
5
|
-
import { useElmish } from "./useElmish";
|
|
6
|
-
export type { Logger, Message, Cmd, Dispatch, UpdateReturnType, MsgSource, UpdateMap, ErrorMessage, };
|
|
7
|
-
export { init, createCmd, ElmComponent, handleError, useElmish, };
|
|
5
|
+
import { SubscriptionResult, useElmish } from "./useElmish";
|
|
6
|
+
export type { Logger, Message, Cmd, Dispatch, InitResult, UpdateReturnType, SubscriptionResult, MsgSource, UpdateMap, ErrorMessage, };
|
|
7
|
+
export { init, createCmd, ElmComponent, errorMsg, handleError, useElmish, };
|
package/dist/index.js
CHANGED
|
@@ -15,6 +15,12 @@ Object.defineProperty(exports, "createCmd", {
|
|
|
15
15
|
return _Cmd.createCmd;
|
|
16
16
|
}
|
|
17
17
|
});
|
|
18
|
+
Object.defineProperty(exports, "errorMsg", {
|
|
19
|
+
enumerable: true,
|
|
20
|
+
get: function get() {
|
|
21
|
+
return _ElmUtilities.errorMsg;
|
|
22
|
+
}
|
|
23
|
+
});
|
|
18
24
|
Object.defineProperty(exports, "handleError", {
|
|
19
25
|
enumerable: true,
|
|
20
26
|
get: function get() {
|
|
@@ -43,4 +49,4 @@ var _ElmUtilities = require("./ElmUtilities");
|
|
|
43
49
|
var _Init = require("./Init");
|
|
44
50
|
|
|
45
51
|
var _useElmish = require("./useElmish");
|
|
46
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
52
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFDQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENtZCwgY3JlYXRlQ21kLCBEaXNwYXRjaCB9IGZyb20gXCIuL0NtZFwiO1xuaW1wb3J0IHsgRWxtQ29tcG9uZW50LCBJbml0UmVzdWx0LCBVcGRhdGVSZXR1cm5UeXBlIH0gZnJvbSBcIi4vRWxtQ29tcG9uZW50XCI7XG5pbXBvcnQgeyBFcnJvck1lc3NhZ2UsIGVycm9yTXNnLCBoYW5kbGVFcnJvciwgTXNnU291cmNlLCBVcGRhdGVNYXAgfSBmcm9tIFwiLi9FbG1VdGlsaXRpZXNcIjtcbmltcG9ydCB7IGluaXQsIExvZ2dlciwgTWVzc2FnZSB9IGZyb20gXCIuL0luaXRcIjtcbmltcG9ydCB7IFN1YnNjcmlwdGlvblJlc3VsdCwgdXNlRWxtaXNoIH0gZnJvbSBcIi4vdXNlRWxtaXNoXCI7XG5cbmV4cG9ydCB0eXBlIHtcbiAgICBMb2dnZXIsXG4gICAgTWVzc2FnZSxcbiAgICBDbWQsXG4gICAgRGlzcGF0Y2gsXG4gICAgSW5pdFJlc3VsdCxcbiAgICBVcGRhdGVSZXR1cm5UeXBlLFxuICAgIFN1YnNjcmlwdGlvblJlc3VsdCxcbiAgICBNc2dTb3VyY2UsXG4gICAgVXBkYXRlTWFwLFxuICAgIEVycm9yTWVzc2FnZSxcbn07XG5cbmV4cG9ydCB7XG4gICAgaW5pdCxcbiAgICBjcmVhdGVDbWQsXG4gICAgRWxtQ29tcG9uZW50LFxuICAgIGVycm9yTXNnLFxuICAgIGhhbmRsZUVycm9yLFxuICAgIHVzZUVsbWlzaCxcbn07Il19
|
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { Dispatch } from "../Cmd";
|
|
2
|
+
import { InitFunction, UpdateFunction } from "../ElmComponent";
|
|
2
3
|
import { MessageBase } from "../ElmUtilities";
|
|
3
|
-
import { UpdateFunction } from "../ElmComponent";
|
|
4
4
|
/**
|
|
5
5
|
* Hook to use the Elm architecture pattern in a function component.
|
|
6
6
|
* @param props The props of the component.
|
|
@@ -12,4 +12,4 @@ import { UpdateFunction } from "../ElmComponent";
|
|
|
12
12
|
* const [model, dispatch] = useElmish(props, init, update, "MyComponent");
|
|
13
13
|
* @deprecated Use `useElmish` with an options object instead.
|
|
14
14
|
*/
|
|
15
|
-
export declare function useElmish<TProps, TModel, TMsg extends MessageBase>(props: TProps, init:
|
|
15
|
+
export declare function useElmish<TProps, TModel, TMsg extends MessageBase>(props: TProps, init: InitFunction<TProps, TModel, TMsg>, update: UpdateFunction<TProps, TModel, TMsg>, name: string): [TModel, Dispatch<TMsg>];
|
package/dist/legacy/useElmish.js
CHANGED
|
@@ -124,9 +124,12 @@ function useElmish(props, init, update, name) {
|
|
|
124
124
|
|
|
125
125
|
initializedModel = initModel;
|
|
126
126
|
setModel(initializedModel);
|
|
127
|
-
|
|
127
|
+
|
|
128
|
+
if (initCmd) {
|
|
129
|
+
execCmd(initCmd);
|
|
130
|
+
}
|
|
128
131
|
}
|
|
129
132
|
|
|
130
133
|
return [initializedModel, dispatch];
|
|
131
134
|
}
|
|
132
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
135
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/legacy/useElmish.ts"],"names":["useElmish","props","init","update","name","reentered","buffer","currentModel","model","setModel","initializedModel","execCmd","cmd","forEach","call","dispatch","ex","LoggerService","error","msg","modelHasChanged","updatedModel","Object","getOwnPropertyNames","length","dispatchMiddleware","push","nextMsg","modified","info","debug","newModel","shift","prevModel","initModel","initCmd"],"mappings":";;;;;;;AACA;;AAGA;;;;;;;;;;;;;;;;;;;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAASA,SAAT,CAA8DC,KAA9D,EAA6EC,IAA7E,EAAuHC,MAAvH,EAAqKC,IAArK,EAA6M;AAChN,MAAIC,SAAS,GAAG,KAAhB;AACA,MAAMC,MAAe,GAAG,EAAxB;AACA,MAAIC,YAA6B,GAAG,EAApC;;AAEA,kBAA0B,qBAA2B,IAA3B,CAA1B;AAAA;AAAA,MAAOC,KAAP;AAAA,MAAcC,QAAd;;AACA,MAAIC,gBAAgB,GAAGF,KAAvB;AAEA,MAAMG,OAAO,GAAG,wBAAY,UAACC,GAAD,EAA0B;AAClDA,IAAAA,GAAG,CAACC,OAAJ,CAAY,UAAAC,IAAI,EAAI;AAChB,UAAI;AACAA,QAAAA,IAAI,CAACC,QAAD,CAAJ;AACH,OAFD,CAEE,OAAOC,EAAP,EAAoB;AAClBC,sGAAeC,KAAf,CAAqBF,EAArB;AACH;AACJ,KAND;AAOH,GARe,EAQb,EARa,CAAhB;AAUA,MAAMD,QAAQ,GAAG,wBAAY,UAACI,GAAD,EAAqB;AAC9C,QAAI,CAACT,gBAAL,EAAuB;AACnB;AACH;;AAED,QAAMU,eAAe,GAAG,SAAlBA,eAAkB,CAACC,YAAD;AAAA,aAA4CA,YAAY,KAAKX,gBAAjB,IAAqCY,MAAM,CAACC,mBAAP,CAA2BF,YAA3B,EAAyCG,MAAzC,GAAkD,CAAnI;AAAA,KAAxB;;AAEA,QAAIC,wBAAJ,EAAwB;AACpB,oCAAmBN,GAAnB;AACH;;AAED,QAAId,SAAJ,EAAe;AACXC,MAAAA,MAAM,CAACoB,IAAP,CAAYP,GAAZ;AACH,KAFD,MAEO;AACHd,MAAAA,SAAS,GAAG,IAAZ;AAEA,UAAIsB,OAAyB,GAAGR,GAAhC;AACA,UAAIS,QAAQ,GAAG,KAAf;;AAEA,aAAOD,OAAP,EAAgB;AACZV,sGAAeY,IAAf,CAAoB,KAApB,EAA2B,cAA3B,EAA2CzB,IAA3C,EAAiDuB,OAAO,CAACvB,IAAzD;AACAa,sGAAea,KAAf,CAAqB,KAArB,EAA4B,cAA5B,EAA4C1B,IAA5C,EAAkDuB,OAAlD;;AAEA,YAAI;AACA,wBAAwBxB,MAAM,iCAAMO,gBAAN,GAA2BH,YAA3B,GAA2CoB,OAA3C,EAAoD1B,KAApD,CAA9B;AAAA;AAAA,cAAO8B,QAAP;AAAA,cAAiBnB,GAAjB;;AAEA,cAAIQ,eAAe,CAACW,QAAD,CAAnB,EAA+B;AAC3BxB,YAAAA,YAAY,mCAAQA,YAAR,GAAyBwB,QAAzB,CAAZ;AAEAH,YAAAA,QAAQ,GAAG,IAAX;AACH;;AAED,cAAIhB,GAAJ,EAAS;AACLD,YAAAA,OAAO,CAACC,GAAD,CAAP;AACH;AACJ,SAZD,CAYE,OAAOI,EAAP,EAAoB;AAClBC,wGAAeC,KAAf,CAAqBF,EAArB;AACH;;AAEDW,QAAAA,OAAO,GAAGrB,MAAM,CAAC0B,KAAP,EAAV;AACH;;AACD3B,MAAAA,SAAS,GAAG,KAAZ;;AAEA,UAAIuB,QAAJ,EAAc;AACVnB,QAAAA,QAAQ,CAAC,UAAAwB,SAAS,EAAI;AAClB,cAAMZ,YAAY,mCAAQY,SAAR,GAAgC1B,YAAhC,CAAlB;;AAEAU,wGAAea,KAAf,CAAqB,KAArB,EAA4B,kBAA5B,EAAgD1B,IAAhD,EAAsDiB,YAAtD;AAEA,iBAAOA,YAAP;AACH,SANO,CAAR;AAOH;AACJ;AACJ,GArDgB,EAqDd,EArDc,CAAjB;;AAuDA,MAAI,CAACX,gBAAL,EAAuB;AACnB,gBAA6BR,IAAI,CAACD,KAAD,CAAjC;AAAA;AAAA,QAAOiC,SAAP;AAAA,QAAkBC,OAAlB;;AAEAzB,IAAAA,gBAAgB,GAAGwB,SAAnB;AACAzB,IAAAA,QAAQ,CAACC,gBAAD,CAAR;;AAEA,QAAIyB,OAAJ,EAAa;AACTxB,MAAAA,OAAO,CAACwB,OAAD,CAAP;AACH;AACJ;;AAED,SAAO,CAACzB,gBAAD,EAAmBK,QAAnB,CAAP;AACH","sourcesContent":["import { Cmd, Dispatch } from \"../Cmd\";\nimport { dispatchMiddleware, LoggerService } from \"../Init\";\nimport { InitFunction, UpdateFunction } from \"../ElmComponent\";\nimport { MessageBase, Nullable } from \"../ElmUtilities\";\nimport { useCallback, useState } from \"react\";\n\n/**\n * Hook to use the Elm architecture pattern in a function component.\n * @param props The props of the component.\n * @param init Function to initialize the model.\n * @param update The update function.\n * @param name The name of the component.\n * @returns A tuple containing the current model and the dispatcher.\n * @example\n * const [model, dispatch] = useElmish(props, init, update, \"MyComponent\");\n * @deprecated Use `useElmish` with an options object instead.\n */\nexport function useElmish<TProps, TModel, TMsg extends MessageBase> (props: TProps, init: InitFunction<TProps, TModel, TMsg>, update: UpdateFunction<TProps, TModel, TMsg>, name: string): [TModel, Dispatch<TMsg>] {\n    let reentered = false;\n    const buffer: TMsg [] = [];\n    let currentModel: Partial<TModel> = {};\n\n    const [model, setModel] = useState<Nullable<TModel>>(null);\n    let initializedModel = model;\n\n    const execCmd = useCallback((cmd: Cmd<TMsg>): void => {\n        cmd.forEach(call => {\n            try {\n                call(dispatch);\n            } catch (ex: unknown) {\n                LoggerService?.error(ex);\n            }\n        });\n    }, []);\n\n    const dispatch = useCallback((msg: TMsg): 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 (dispatchMiddleware) {\n            dispatchMiddleware(msg);\n        }\n\n        if (reentered) {\n            buffer.push(msg);\n        } else {\n            reentered = true;\n\n            let nextMsg: TMsg | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                LoggerService?.info(\"Elm\", \"message from\", name, nextMsg.name);\n                LoggerService?.debug(\"Elm\", \"message from\", name, nextMsg);\n\n                try {\n                    const [newModel, cmd] = update({ ...initializedModel, ...currentModel }, nextMsg, props);\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                    LoggerService?.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                    LoggerService?.debug(\"Elm\", \"update model for\", name, updatedModel);\n\n                    return updatedModel;\n                });\n            }\n        }\n    }, []);\n\n    if (!initializedModel) {\n        const [initModel, initCmd] = init(props);\n\n        initializedModel = initModel;\n        setModel(initializedModel);\n\n        if (initCmd) {\n            execCmd(initCmd);\n        }\n    }\n\n    return [initializedModel, dispatch];\n}"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { Dispatch } from "../Cmd";
|
|
2
2
|
import { MessageBase, UpdateMap } from "../ElmUtilities";
|
|
3
|
+
import { InitFunction } from "../ElmComponent";
|
|
3
4
|
/**
|
|
4
5
|
* Hook to use the Elm architecture pattern in a function component.
|
|
5
6
|
* @param props The props of the component.
|
|
@@ -11,4 +12,4 @@ import { MessageBase, UpdateMap } from "../ElmUtilities";
|
|
|
11
12
|
* const [model, dispatch] = useElmishMap(props, init, updateMap, "MyComponent");
|
|
12
13
|
* @deprecated Use `useElmish` with an options object instead.
|
|
13
14
|
*/
|
|
14
|
-
export declare function useElmishMap<TProps, TModel, TMessage extends MessageBase>(props: TProps, init:
|
|
15
|
+
export declare function useElmishMap<TProps, TModel, TMessage extends MessageBase>(props: TProps, init: InitFunction<TProps, TModel, TMessage>, updateMap: UpdateMap<TProps, TModel, TMessage>, name: string): [TModel, Dispatch<TMessage>];
|
|
@@ -126,9 +126,12 @@ function useElmishMap(props, init, updateMap, name) {
|
|
|
126
126
|
|
|
127
127
|
initializedModel = initModel;
|
|
128
128
|
setModel(initializedModel);
|
|
129
|
-
|
|
129
|
+
|
|
130
|
+
if (initCmd) {
|
|
131
|
+
execCmd(initCmd);
|
|
132
|
+
}
|
|
130
133
|
}
|
|
131
134
|
|
|
132
135
|
return [initializedModel, dispatch];
|
|
133
136
|
}
|
|
134
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
137
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/legacy/useElmishMap.ts"],"names":["useElmishMap","props","init","updateMap","name","reentered","buffer","currentModel","model","setModel","initializedModel","execCmd","cmd","forEach","call","dispatch","ex","LoggerService","error","msg","modelHasChanged","updatedModel","Object","getOwnPropertyNames","length","dispatchMiddleware","push","nextMsg","modified","info","debug","newModel","shift","prevModel","initModel","initCmd"],"mappings":"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","sourcesContent":["import { Cmd, Dispatch } from \"../Cmd\";\nimport { dispatchMiddleware, LoggerService } from \"../Init\";\nimport { MessageBase, Nullable, UpdateMap } from \"../ElmUtilities\";\nimport { useCallback, useState } from \"react\";\nimport { callUpdateMap } from \"../useElmish\";\nimport { InitFunction } from \"../ElmComponent\";\n\n/**\n * Hook to use the Elm architecture pattern in a function component.\n * @param props The props of the component.\n * @param init Function to initialize the model.\n * @param updateMap The update map object.\n * @param name The name of the component.\n * @returns A tuple containing the current model and the dispatcher.\n * @example\n * const [model, dispatch] = useElmishMap(props, init, updateMap, \"MyComponent\");\n * @deprecated Use `useElmish` with an options object instead.\n */\nexport function useElmishMap<TProps, TModel, TMessage extends MessageBase> (props: TProps, init: InitFunction<TProps, TModel, TMessage>, updateMap: UpdateMap<TProps, TModel, TMessage>, name: string): [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 execCmd = useCallback((cmd: Cmd<TMessage>): void => {\n        cmd.forEach(call => {\n            try {\n                call(dispatch);\n            } catch (ex: unknown) {\n                LoggerService?.error(ex);\n            }\n        });\n    }, []);\n\n    const dispatch = useCallback((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 (dispatchMiddleware) {\n            dispatchMiddleware(msg);\n        }\n\n        if (reentered) {\n            buffer.push(msg);\n        } else {\n            reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                LoggerService?.info(\"Elm\", \"message from\", name, nextMsg.name);\n                LoggerService?.debug(\"Elm\", \"message from\", name, nextMsg);\n\n                try {\n                    const [newModel, cmd] = callUpdateMap(updateMap, nextMsg, { ...initializedModel, ...currentModel }, props);\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                    LoggerService?.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                    LoggerService?.debug(\"Elm\", \"update model for\", name, updatedModel);\n\n                    return updatedModel;\n                });\n            }\n        }\n    }, []);\n\n    if (!initializedModel) {\n        const [initModel, initCmd] = init(props);\n\n        initializedModel = initModel;\n        setModel(initializedModel);\n\n        if (initCmd) {\n            execCmd(initCmd);\n        }\n    }\n\n    return [initializedModel, dispatch];\n}"]}
|
package/dist/useElmish.d.ts
CHANGED
|
@@ -1,11 +1,14 @@
|
|
|
1
1
|
import { Cmd, Dispatch } from "./Cmd";
|
|
2
|
+
import { InitFunction, UpdateFunction, UpdateReturnType } from "./ElmComponent";
|
|
2
3
|
import { MessageBase, UpdateMap } from "./ElmUtilities";
|
|
3
|
-
|
|
4
|
+
export declare type SubscriptionResult<TMessage> = [Cmd<TMessage>, (() => void)?];
|
|
5
|
+
declare type Subscription<TModel, TMessage> = (model: TModel) => SubscriptionResult<TMessage>;
|
|
4
6
|
interface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {
|
|
7
|
+
name: string;
|
|
5
8
|
props: TProps;
|
|
6
|
-
init:
|
|
9
|
+
init: InitFunction<TProps, TModel, TMessage>;
|
|
7
10
|
update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>;
|
|
8
|
-
|
|
11
|
+
subscription?: Subscription<TModel, TMessage>;
|
|
9
12
|
}
|
|
10
13
|
/**
|
|
11
14
|
* Hook to use the Elm architecture pattern in a function component.
|
|
@@ -14,7 +17,7 @@ interface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {
|
|
|
14
17
|
* @example
|
|
15
18
|
* const [model, dispatch] = useElmish({ props, init, update, name: "MyComponent" });
|
|
16
19
|
*/
|
|
17
|
-
export declare function useElmish<TProps, TModel, TMessage extends MessageBase>({ props, init, update,
|
|
20
|
+
export declare function useElmish<TProps, TModel, TMessage extends MessageBase>({ name, props, init, update, subscription }: UseElmishOptions<TProps, TModel, TMessage>): [TModel, Dispatch<TMessage>];
|
|
18
21
|
export declare function callUpdate<TProps, TModel, TMessage extends MessageBase>(update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>, msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage>;
|
|
19
22
|
export declare function callUpdateMap<TProps, TModel, TMessage extends MessageBase>(updateMap: UpdateMap<TProps, TModel, TMessage>, msg: TMessage, model: TModel, props: TProps): UpdateReturnType<TModel, TMessage>;
|
|
20
23
|
export {};
|
package/dist/useElmish.js
CHANGED
|
@@ -37,10 +37,11 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
|
37
37
|
* const [model, dispatch] = useElmish({ props, init, update, name: "MyComponent" });
|
|
38
38
|
*/
|
|
39
39
|
function useElmish(_ref) {
|
|
40
|
-
var
|
|
40
|
+
var name = _ref.name,
|
|
41
|
+
props = _ref.props,
|
|
41
42
|
init = _ref.init,
|
|
42
43
|
update = _ref.update,
|
|
43
|
-
|
|
44
|
+
subscription = _ref.subscription;
|
|
44
45
|
var reentered = false;
|
|
45
46
|
var buffer = [];
|
|
46
47
|
var currentModel = {};
|
|
@@ -126,9 +127,26 @@ function useElmish(_ref) {
|
|
|
126
127
|
|
|
127
128
|
initializedModel = initModel;
|
|
128
129
|
setModel(initializedModel);
|
|
129
|
-
|
|
130
|
+
|
|
131
|
+
if (initCmd) {
|
|
132
|
+
execCmd(initCmd);
|
|
133
|
+
}
|
|
130
134
|
}
|
|
131
135
|
|
|
136
|
+
(0, _react.useEffect)(function () {
|
|
137
|
+
if (subscription) {
|
|
138
|
+
var _subscription = subscription(initializedModel),
|
|
139
|
+
_subscription2 = _slicedToArray(_subscription, 2),
|
|
140
|
+
subCmd = _subscription2[0],
|
|
141
|
+
destructor = _subscription2[1];
|
|
142
|
+
|
|
143
|
+
execCmd(subCmd);
|
|
144
|
+
|
|
145
|
+
if (destructor) {
|
|
146
|
+
return destructor;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
}, []);
|
|
132
150
|
return [initializedModel, dispatch];
|
|
133
151
|
}
|
|
134
152
|
|
|
@@ -145,4 +163,4 @@ function callUpdateMap(updateMap, msg, model, props) {
|
|
|
145
163
|
// @ts-expect-error -- We know that nextMsg fits
|
|
146
164
|
return updateMap[msg.name](msg, model, props);
|
|
147
165
|
}
|
|
148
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/useElmish.ts"],"names":["useElmish","props","init","update","name","reentered","buffer","currentModel","model","setModel","initializedModel","execCmd","cmd","forEach","call","dispatch","ex","LoggerService","error","msg","modelHasChanged","updatedModel","Object","getOwnPropertyNames","length","dispatchMiddleware","push","nextMsg","modified","info","debug","callUpdate","newModel","shift","prevModel","initModel","initCmd","callUpdateMap","updateMap"],"mappings":"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","sourcesContent":["import { Cmd, Dispatch } from \"./Cmd\";\nimport { dispatchMiddleware, LoggerService } from \"./Init\";\nimport { MessageBase, Nullable, UpdateMap } from \"./ElmUtilities\";\nimport { UpdateFunction, UpdateReturnType } from \"./ElmComponent\";\nimport { useCallback, useState } from \"react\";\n\ninterface UseElmishOptions<TProps, TModel, TMessage extends MessageBase> {\n    props: TProps,\n    init: (props: TProps) => [TModel, Cmd<TMessage>],\n    update: UpdateFunction<TProps, TModel, TMessage> | UpdateMap<TProps, TModel, TMessage>,\n    name: string,\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 */\nexport function useElmish<TProps, TModel, TMessage extends MessageBase> ({ props, init, update, name }: 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 execCmd = useCallback((cmd: Cmd<TMessage>): void => {\n        cmd.forEach(call => {\n            try {\n                call(dispatch);\n            } catch (ex: unknown) {\n                LoggerService?.error(ex);\n            }\n        });\n    }, []);\n\n    const dispatch = useCallback((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 (dispatchMiddleware) {\n            dispatchMiddleware(msg);\n        }\n\n        if (reentered) {\n            buffer.push(msg);\n        } else {\n            reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                LoggerService?.info(\"Elm\", \"message from\", name, nextMsg.name);\n                LoggerService?.debug(\"Elm\", \"message from\", name, nextMsg);\n\n                try {\n                    const [newModel, cmd] = callUpdate(update, nextMsg, { ...initializedModel, ...currentModel }, props);\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                    LoggerService?.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                    LoggerService?.debug(\"Elm\", \"update model for\", name, updatedModel);\n\n                    return updatedModel;\n                });\n            }\n        }\n    }, []);\n\n    if (!initializedModel) {\n        const [initModel, initCmd] = init(props);\n\n        initializedModel = initModel;\n        setModel(initializedModel);\n\n        execCmd(initCmd);\n    }\n\n    return [initializedModel, dispatch];\n}\n\nexport function 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\nexport function 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 nextMsg fits\n    return updateMap[msg.name as TMessage[\"name\"]](msg, model, props);\n}"]}
|
|
166
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/useElmish.ts"],"names":["useElmish","name","props","init","update","subscription","reentered","buffer","currentModel","model","setModel","initializedModel","execCmd","cmd","forEach","call","dispatch","ex","LoggerService","error","msg","modelHasChanged","updatedModel","Object","getOwnPropertyNames","length","dispatchMiddleware","push","nextMsg","modified","info","debug","callUpdate","newModel","shift","prevModel","initModel","initCmd","subCmd","destructor","callUpdateMap","updateMap"],"mappings":"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","sourcesContent":["import { Cmd, Dispatch } from \"./Cmd\";\nimport { dispatchMiddleware, LoggerService } from \"./Init\";\nimport { InitFunction, UpdateFunction, UpdateReturnType } from \"./ElmComponent\";\nimport { MessageBase, Nullable, UpdateMap } from \"./ElmUtilities\";\nimport { useCallback, useEffect, useState } from \"react\";\n\nexport type SubscriptionResult<TMessage> = [Cmd<TMessage>, (() => void)?];\ntype Subscription<TModel, TMessage> = (model: TModel) => 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<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 */\nexport function 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 execCmd = useCallback((cmd: Cmd<TMessage>): void => {\n        cmd.forEach(call => {\n            try {\n                call(dispatch);\n            } catch (ex: unknown) {\n                LoggerService?.error(ex);\n            }\n        });\n    }, []);\n\n    const dispatch = useCallback((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 (dispatchMiddleware) {\n            dispatchMiddleware(msg);\n        }\n\n        if (reentered) {\n            buffer.push(msg);\n        } else {\n            reentered = true;\n\n            let nextMsg: TMessage | undefined = msg;\n            let modified = false;\n\n            while (nextMsg) {\n                LoggerService?.info(\"Elm\", \"message from\", name, nextMsg.name);\n                LoggerService?.debug(\"Elm\", \"message from\", name, nextMsg);\n\n                try {\n                    const [newModel, cmd] = callUpdate(update, nextMsg, { ...initializedModel, ...currentModel }, props);\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                    LoggerService?.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                    LoggerService?.debug(\"Elm\", \"update model for\", name, updatedModel);\n\n                    return updatedModel;\n                });\n            }\n        }\n    }, []);\n\n    if (!initializedModel) {\n        const [initModel, initCmd] = 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);\n\n            execCmd(subCmd);\n\n            if (destructor) {\n                return destructor;\n            }\n        }\n    }, []);\n\n    return [initializedModel, dispatch];\n}\n\nexport function 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\nexport function 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 nextMsg fits\n    return updateMap[msg.name as TMessage[\"name\"]](msg, model, props);\n}"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "react-elmish",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.1.0",
|
|
4
4
|
"description": "Elmish for React using Typescript",
|
|
5
5
|
"author": "atheck",
|
|
6
6
|
"license": "MIT",
|
|
@@ -14,7 +14,7 @@
|
|
|
14
14
|
"@babel/preset-env": "7.16.11",
|
|
15
15
|
"@babel/preset-react": "7.16.7",
|
|
16
16
|
"@babel/preset-typescript": "7.16.7",
|
|
17
|
-
"@testing-library/react": "
|
|
17
|
+
"@testing-library/react": "13.0.0",
|
|
18
18
|
"@types/jest": "27.4.1",
|
|
19
19
|
"@types/react": "17.0.43",
|
|
20
20
|
"eslint": "8.12.0",
|