2025-11-25 22:23:19 +08:00
|
|
|
|
import { useState, useEffect, useMemo, useRef, createLogger } from '@esengine/editor-runtime';
|
2025-11-18 14:46:51 +08:00
|
|
|
|
import { ExecutionController, ExecutionMode } from '../application/services/ExecutionController';
|
|
|
|
|
|
import { BlackboardManager } from '../application/services/BlackboardManager';
|
|
|
|
|
|
import { BehaviorTreeNode, Connection, useBehaviorTreeDataStore } from '../stores';
|
|
|
|
|
|
import { ExecutionLog } from '../utils/BehaviorTreeExecutor';
|
|
|
|
|
|
import { BlackboardValue } from '../domain/models/Blackboard';
|
|
|
|
|
|
|
|
|
|
|
|
const logger = createLogger('useExecutionController');
|
2025-11-03 21:22:16 +08:00
|
|
|
|
|
|
|
|
|
|
type BlackboardVariables = Record<string, BlackboardValue>;
|
|
|
|
|
|
|
|
|
|
|
|
interface UseExecutionControllerParams {
|
|
|
|
|
|
rootNodeId: string;
|
|
|
|
|
|
projectPath: string | null;
|
|
|
|
|
|
blackboardVariables: BlackboardVariables;
|
|
|
|
|
|
nodes: BehaviorTreeNode[];
|
|
|
|
|
|
connections: Connection[];
|
|
|
|
|
|
initialBlackboardVariables: BlackboardVariables;
|
|
|
|
|
|
onBlackboardUpdate: (variables: BlackboardVariables) => void;
|
|
|
|
|
|
onInitialBlackboardSave: (variables: BlackboardVariables) => void;
|
|
|
|
|
|
onExecutingChange: (isExecuting: boolean) => void;
|
2025-11-04 18:29:28 +08:00
|
|
|
|
onSaveNodesDataSnapshot: () => void;
|
|
|
|
|
|
onRestoreNodesData: () => void;
|
2025-11-18 14:46:51 +08:00
|
|
|
|
sortChildrenByPosition: () => void;
|
2025-11-03 21:22:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function useExecutionController(params: UseExecutionControllerParams) {
|
|
|
|
|
|
const {
|
|
|
|
|
|
rootNodeId,
|
|
|
|
|
|
projectPath,
|
|
|
|
|
|
blackboardVariables,
|
|
|
|
|
|
nodes,
|
|
|
|
|
|
connections,
|
|
|
|
|
|
onBlackboardUpdate,
|
|
|
|
|
|
onInitialBlackboardSave,
|
2025-11-04 18:29:28 +08:00
|
|
|
|
onExecutingChange,
|
|
|
|
|
|
onSaveNodesDataSnapshot,
|
2025-11-18 14:46:51 +08:00
|
|
|
|
onRestoreNodesData,
|
|
|
|
|
|
sortChildrenByPosition
|
2025-11-03 21:22:16 +08:00
|
|
|
|
} = params;
|
|
|
|
|
|
|
|
|
|
|
|
const [executionMode, setExecutionMode] = useState<ExecutionMode>('idle');
|
|
|
|
|
|
const [executionLogs, setExecutionLogs] = useState<ExecutionLog[]>([]);
|
|
|
|
|
|
const [executionSpeed, setExecutionSpeed] = useState<number>(1.0);
|
|
|
|
|
|
const [tickCount, setTickCount] = useState(0);
|
|
|
|
|
|
|
|
|
|
|
|
const controller = useMemo(() => {
|
|
|
|
|
|
return new ExecutionController({
|
|
|
|
|
|
rootNodeId,
|
|
|
|
|
|
projectPath,
|
|
|
|
|
|
onLogsUpdate: setExecutionLogs,
|
|
|
|
|
|
onBlackboardUpdate,
|
2025-11-04 18:29:28 +08:00
|
|
|
|
onTickCountUpdate: setTickCount,
|
|
|
|
|
|
onExecutionStatusUpdate: (statuses, orders) => {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
const store = useBehaviorTreeDataStore.getState();
|
2025-11-04 18:29:28 +08:00
|
|
|
|
store.updateNodeExecutionStatuses(statuses, orders);
|
|
|
|
|
|
}
|
2025-11-18 14:46:51 +08:00
|
|
|
|
// 不在这里传递 onBreakpointHit,避免频繁重建
|
2025-11-03 21:22:16 +08:00
|
|
|
|
});
|
2025-11-04 18:29:28 +08:00
|
|
|
|
}, [rootNodeId, projectPath, onBlackboardUpdate]);
|
2025-11-03 21:22:16 +08:00
|
|
|
|
|
|
|
|
|
|
const blackboardManager = useMemo(() => new BlackboardManager(), []);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
// 保存当前 controller 的引用,确保清理时使用正确的实例
|
|
|
|
|
|
const currentController = controller;
|
2025-11-03 21:22:16 +08:00
|
|
|
|
return () => {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
currentController.destroy();
|
2025-11-03 21:22:16 +08:00
|
|
|
|
};
|
|
|
|
|
|
}, [controller]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
controller.setConnections(connections);
|
|
|
|
|
|
}, [connections, controller]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (executionMode === 'idle') return;
|
|
|
|
|
|
|
|
|
|
|
|
const executorVars = controller.getBlackboardVariables();
|
|
|
|
|
|
|
|
|
|
|
|
Object.entries(blackboardVariables).forEach(([key, value]) => {
|
|
|
|
|
|
if (executorVars[key] !== value) {
|
|
|
|
|
|
controller.updateBlackboardVariable(key, value);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
}, [blackboardVariables, executionMode, controller]);
|
|
|
|
|
|
|
2025-11-04 18:29:28 +08:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (executionMode === 'idle') return;
|
|
|
|
|
|
|
|
|
|
|
|
controller.updateNodes(nodes);
|
|
|
|
|
|
}, [nodes, executionMode, controller]);
|
|
|
|
|
|
|
2025-11-03 21:22:16 +08:00
|
|
|
|
const handlePlay = async () => {
|
|
|
|
|
|
try {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
sortChildrenByPosition();
|
|
|
|
|
|
logger.info('[Execute] Sorted children by position before execution');
|
|
|
|
|
|
|
2025-11-03 21:22:16 +08:00
|
|
|
|
blackboardManager.setInitialVariables(blackboardVariables);
|
|
|
|
|
|
blackboardManager.setCurrentVariables(blackboardVariables);
|
|
|
|
|
|
onInitialBlackboardSave(blackboardManager.getInitialVariables());
|
2025-11-04 18:29:28 +08:00
|
|
|
|
onSaveNodesDataSnapshot();
|
2025-11-03 21:22:16 +08:00
|
|
|
|
onExecutingChange(true);
|
|
|
|
|
|
|
|
|
|
|
|
setExecutionMode('running');
|
|
|
|
|
|
await controller.play(nodes, blackboardVariables, connections);
|
|
|
|
|
|
} catch (error) {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
logger.error('Failed to start execution:', error);
|
2025-11-03 21:22:16 +08:00
|
|
|
|
setExecutionMode('idle');
|
|
|
|
|
|
onExecutingChange(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handlePause = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await controller.pause();
|
|
|
|
|
|
const newMode = controller.getMode();
|
|
|
|
|
|
setExecutionMode(newMode);
|
|
|
|
|
|
} catch (error) {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
logger.error('Failed to pause/resume execution:', error);
|
2025-11-03 21:22:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleStop = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await controller.stop();
|
|
|
|
|
|
setExecutionMode('idle');
|
|
|
|
|
|
setTickCount(0);
|
|
|
|
|
|
|
|
|
|
|
|
const restoredVars = blackboardManager.restoreInitialVariables();
|
|
|
|
|
|
onBlackboardUpdate(restoredVars);
|
2025-11-04 18:29:28 +08:00
|
|
|
|
onRestoreNodesData();
|
2025-11-18 14:46:51 +08:00
|
|
|
|
useBehaviorTreeDataStore.getState().clearNodeExecutionStatuses();
|
2025-11-03 21:22:16 +08:00
|
|
|
|
onExecutingChange(false);
|
|
|
|
|
|
} catch (error) {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
logger.error('Failed to stop execution:', error);
|
2025-11-03 21:22:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleStep = () => {
|
|
|
|
|
|
controller.step();
|
2025-11-18 14:46:51 +08:00
|
|
|
|
// 单步执行后保持idle状态,不需要专门的step状态
|
2025-11-03 21:22:16 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleReset = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await controller.reset();
|
|
|
|
|
|
setExecutionMode('idle');
|
|
|
|
|
|
setTickCount(0);
|
|
|
|
|
|
} catch (error) {
|
2025-11-18 14:46:51 +08:00
|
|
|
|
logger.error('Failed to reset execution:', error);
|
2025-11-03 21:22:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleSpeedChange = (speed: number) => {
|
|
|
|
|
|
setExecutionSpeed(speed);
|
|
|
|
|
|
controller.setSpeed(speed);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
executionMode,
|
|
|
|
|
|
executionLogs,
|
|
|
|
|
|
executionSpeed,
|
|
|
|
|
|
tickCount,
|
|
|
|
|
|
handlePlay,
|
|
|
|
|
|
handlePause,
|
|
|
|
|
|
handleStop,
|
|
|
|
|
|
handleStep,
|
|
|
|
|
|
handleReset,
|
|
|
|
|
|
handleSpeedChange,
|
|
|
|
|
|
setExecutionLogs,
|
|
|
|
|
|
controller,
|
|
|
|
|
|
blackboardManager
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|