@@ -12,7 +12,7 @@ import type {
12
12
Params ,
13
13
Service ,
14
14
} from "@zag-js/core"
15
- import { createScope } from "@zag-js/core"
15
+ import { createScope , INIT_STATE , MachineStatus } from "@zag-js/core"
16
16
import { ensure , isFunction , isString , toArray , warn } from "@zag-js/utils"
17
17
import { useLayoutEffect , useMemo , useRef } from "preact/hooks"
18
18
import { flushSync } from "react-dom"
@@ -29,6 +29,10 @@ export function useMachine<T extends MachineSchema>(
29
29
return createScope ( { id, ids, getRootNode } )
30
30
} , [ userProps ] )
31
31
32
+ const debug = ( ...args : any [ ] ) => {
33
+ if ( machine . debug ) console . log ( ...args )
34
+ }
35
+
32
36
const props : any = machine . props ?.( { props : userProps , scope } ) ?? userProps
33
37
const prop = useProp ( props )
34
38
@@ -194,10 +198,10 @@ export function useMachine<T extends MachineSchema>(
194
198
if ( cleanup ) effects . current . set ( nextState as string , cleanup )
195
199
196
200
// root entry actions
197
- if ( prevState === "__init__" ) {
201
+ if ( prevState === INIT_STATE ) {
198
202
action ( machine . entry )
199
203
const cleanup = effect ( machine . effects )
200
- if ( cleanup ) effects . current . set ( "__init__" , cleanup )
204
+ if ( cleanup ) effects . current . set ( INIT_STATE , cleanup )
201
205
}
202
206
203
207
// enter actions
@@ -206,13 +210,30 @@ export function useMachine<T extends MachineSchema>(
206
210
} ,
207
211
} ) )
208
212
213
+ // improve HMR (to restart effects)
214
+ const hydratedStateRef = useRef < string | undefined > ( undefined )
215
+ const statusRef = useRef ( MachineStatus . NotStarted )
216
+
209
217
useLayoutEffect ( ( ) => {
210
- state . invoke ( state . initial ! , "__init__" )
218
+ const started = statusRef . current === MachineStatus . Started
219
+ statusRef . current = MachineStatus . Started
220
+ debug ( started ? "rehydrating..." : "initializing..." )
221
+
222
+ // start the transition
223
+ const initialState = hydratedStateRef . current ?? state . initial !
224
+ state . invoke ( initialState , started ? state . get ( ) : INIT_STATE )
225
+
211
226
const fns = effects . current
227
+ const currentState = state . ref . current
212
228
return ( ) => {
229
+ debug ( "unmounting..." )
230
+ hydratedStateRef . current = currentState
231
+ statusRef . current = MachineStatus . Stopped
232
+
213
233
fns . forEach ( ( fn ) => fn ?.( ) )
214
234
effects . current = new Map ( )
215
235
transitionRef . current = null
236
+
216
237
action ( machine . exit )
217
238
}
218
239
} , [ ] )
@@ -224,6 +245,8 @@ export function useMachine<T extends MachineSchema>(
224
245
225
246
const send = ( event : any ) => {
226
247
queueMicrotask ( ( ) => {
248
+ if ( statusRef . current !== MachineStatus . Started ) return
249
+
227
250
previousEventRef . current = eventRef . current
228
251
eventRef . current = event
229
252
@@ -264,6 +287,7 @@ export function useMachine<T extends MachineSchema>(
264
287
refs,
265
288
computed,
266
289
event : getEvent ( ) ,
290
+ getStatus : ( ) => statusRef . current ,
267
291
} as Service < T >
268
292
}
269
293
0 commit comments