-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathProgram.fs
65 lines (56 loc) · 2.43 KB
/
Program.fs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
namespace Fable.Elmish.Nile
open Elmish
open FSharp.Control
open FSharp.Control.Core
[<RequireQualifiedAccess>]
module Program =
/// Uses `stream` to transform a stream of messages dispatched from the view
/// and a stream of updates to a stream of messages dispatched to `update`.
/// `stream` is only called once.
let withStream (stream: IAsyncObservable<_> -> IAsyncObservable<_> -> IAsyncObservable<_>) (program: Elmish.Program<_,_,_,_>) =
let (modelObserver, modelObservable) = AsyncRx.subject ()
let (messageObserver, messageObservable) = AsyncRx.subject ()
let messages = stream modelObservable messageObservable
let mutable dispatch = ignore
let msgObserver =
{ new IAsyncObserver<'msg> with
member __.OnNextAsync x = async {
dispatch x
}
member __.OnErrorAsync err = async {
Browser.Dom.console.error ("[Fable.Elmish.Nile] Stream error", err)
}
member __.OnCompletedAsync () = async {
Browser.Dom.console.log ("[Fable.Elmish.Nile] Stream completed.")
}
}
let mutable initState = None
let init' fn arg =
let (model, cmd) = fn arg
initState <- Some (None, model)
(model, cmd)
let update' fn msg model =
let (model, cmd) = fn msg model
modelObserver.OnNextAsync (Some msg, model) |> Async.StartImmediate
(model, cmd)
let mutable hasSubscription = 0
let mutable subscription = AsyncDisposable.Empty
let view' fn model dispatch' =
dispatch <- dispatch'
// Wait with subscribing until we have a `dispatch`, otherwise `startWith` messages would get lost
#if FABLE_COMPILER
if hasSubscription = 0 then
hasSubscription <- 1
#else
if System.Threading.Interlocked.Exchange(&hasSubscription, 1) = 0 then
#endif
async {
let! sub = messages.SubscribeAsync msgObserver
subscription <- sub
}
|> Async.Start'
initState |> Option.iter (modelObserver.OnNextAsync >> Async.StartImmediate)
initState <- None
fn model (messageObserver.OnNextAsync >> Async.StartImmediate)
program
|> Program.map init' update' view' id id