* Convert commandBarMachine to standalone actor * Switch all uses of CommandBarProvider pattern to use actor and selector snapshots directly
91 lines
2.7 KiB
TypeScript
91 lines
2.7 KiB
TypeScript
import { useEffect } from 'react'
|
|
import { AnyStateMachine, Actor, StateFrom, EventFrom } from 'xstate'
|
|
import { createMachineCommand } from '../lib/createMachineCommand'
|
|
import { modelingMachine } from 'machines/modelingMachine'
|
|
import { authMachine } from 'machines/authMachine'
|
|
import { settingsMachine } from 'machines/settingsMachine'
|
|
import { projectsMachine } from 'machines/projectsMachine'
|
|
import {
|
|
Command,
|
|
StateMachineCommandSetConfig,
|
|
StateMachineCommandSetSchema,
|
|
} from 'lib/commandTypes'
|
|
import { useKclContext } from 'lang/KclProvider'
|
|
import { useNetworkContext } from 'hooks/useNetworkContext'
|
|
import { NetworkHealthState } from 'hooks/useNetworkStatus'
|
|
import { useAppState } from 'AppState'
|
|
import { commandBarActor } from 'machines/commandBarMachine'
|
|
|
|
// This might not be necessary, AnyStateMachine from xstate is working
|
|
export type AllMachines =
|
|
| typeof modelingMachine
|
|
| typeof settingsMachine
|
|
| typeof authMachine
|
|
| typeof projectsMachine
|
|
|
|
interface UseStateMachineCommandsArgs<
|
|
T extends AllMachines,
|
|
S extends StateMachineCommandSetSchema<T>
|
|
> {
|
|
machineId: T['id']
|
|
state: StateFrom<T>
|
|
send: Function
|
|
actor: Actor<T>
|
|
commandBarConfig?: StateMachineCommandSetConfig<T, S>
|
|
allCommandsRequireNetwork?: boolean
|
|
onCancel?: () => void
|
|
}
|
|
|
|
export default function useStateMachineCommands<
|
|
T extends AnyStateMachine,
|
|
S extends StateMachineCommandSetSchema<T>
|
|
>({
|
|
machineId,
|
|
state,
|
|
send,
|
|
actor,
|
|
commandBarConfig,
|
|
allCommandsRequireNetwork = false,
|
|
onCancel,
|
|
}: UseStateMachineCommandsArgs<T, S>) {
|
|
const { overallState } = useNetworkContext()
|
|
const { isExecuting } = useKclContext()
|
|
const { isStreamReady } = useAppState()
|
|
|
|
useEffect(() => {
|
|
const disableAllButtons =
|
|
(overallState !== NetworkHealthState.Ok &&
|
|
overallState !== NetworkHealthState.Weak) ||
|
|
isExecuting ||
|
|
!isStreamReady
|
|
const newCommands = Object.keys(commandBarConfig || {})
|
|
.filter((_) => !allCommandsRequireNetwork || !disableAllButtons)
|
|
.flatMap((type) => {
|
|
const typeWithProperType = type as EventFrom<T>['type']
|
|
return createMachineCommand<T, S>({
|
|
// The group is the owner machine's ID.
|
|
groupId: machineId,
|
|
type: typeWithProperType,
|
|
state,
|
|
send,
|
|
actor,
|
|
commandBarConfig,
|
|
onCancel,
|
|
})
|
|
})
|
|
.filter((c) => c !== null) as Command[] // TS isn't smart enough to know this filter removes nulls
|
|
|
|
commandBarActor.send({
|
|
type: 'Add commands',
|
|
data: { commands: newCommands },
|
|
})
|
|
|
|
return () => {
|
|
commandBarActor.send({
|
|
type: 'Remove commands',
|
|
data: { commands: newCommands },
|
|
})
|
|
}
|
|
}, [overallState, isExecuting, isStreamReady])
|
|
}
|