2026-04-04 13:08:28 +08:00
|
|
|
import Phaser from 'phaser';
|
2026-04-12 16:52:53 +08:00
|
|
|
import { computed, signal, useSignal, useSignalEffect } from '@preact/signals';
|
2026-04-04 13:08:28 +08:00
|
|
|
import { createContext, h } from 'preact';
|
|
|
|
|
import { useContext } from 'preact/hooks';
|
2026-04-04 16:19:18 +08:00
|
|
|
import {ReadonlySignal} from "@preact/signals-core";
|
2026-04-12 16:26:52 +08:00
|
|
|
import type { ReactiveScene, ReactiveScenePhaserData } from '../scenes';
|
2026-04-12 16:52:53 +08:00
|
|
|
import { FadeScene as FadeSceneClass, FADE_SCENE_KEY } from '../scenes/FadeScene';
|
|
|
|
|
|
|
|
|
|
export interface SceneController {
|
|
|
|
|
/** 启动场景(带淡入淡出过渡) */
|
|
|
|
|
launch(sceneKey: string, data?: Record<string, unknown>): Promise<void>;
|
|
|
|
|
/** 当前活跃场景 key */
|
|
|
|
|
currentScene: ReadonlySignal<string | null>;
|
|
|
|
|
/** 是否正在过渡 */
|
|
|
|
|
isTransitioning: ReadonlySignal<boolean>;
|
|
|
|
|
}
|
2026-04-04 12:14:26 +08:00
|
|
|
|
2026-04-12 16:26:52 +08:00
|
|
|
export interface PhaserGameContext {
|
|
|
|
|
game: Phaser.Game;
|
2026-04-12 16:52:53 +08:00
|
|
|
sceneController: SceneController;
|
2026-04-12 16:26:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export const phaserContext = createContext<ReadonlySignal<PhaserGameContext> | null>(null);
|
2026-04-04 12:14:26 +08:00
|
|
|
|
|
|
|
|
export const defaultPhaserConfig: Phaser.Types.Core.GameConfig = {
|
|
|
|
|
type: Phaser.AUTO,
|
|
|
|
|
width: 560,
|
|
|
|
|
height: 560,
|
|
|
|
|
parent: 'phaser-container',
|
|
|
|
|
backgroundColor: '#f9fafb',
|
|
|
|
|
scene: [],
|
|
|
|
|
};
|
|
|
|
|
|
2026-04-04 13:08:28 +08:00
|
|
|
export interface PhaserGameProps {
|
|
|
|
|
config?: Partial<Phaser.Types.Core.GameConfig>;
|
|
|
|
|
children?: any;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function PhaserGame(props: PhaserGameProps) {
|
2026-04-12 16:52:53 +08:00
|
|
|
const gameSignal = useSignal<PhaserGameContext>({ game: undefined!, sceneController: undefined! });
|
|
|
|
|
const registeredScenes = useSignal<Map<string, ReactiveScene>>(new Map());
|
2026-04-04 13:08:28 +08:00
|
|
|
|
2026-04-04 12:14:26 +08:00
|
|
|
useSignalEffect(() => {
|
2026-04-04 14:47:03 +08:00
|
|
|
const config: Phaser.Types.Core.GameConfig = {
|
|
|
|
|
...defaultPhaserConfig,
|
|
|
|
|
...props.config,
|
|
|
|
|
};
|
|
|
|
|
const phaserGame = new Phaser.Game(config);
|
2026-04-12 16:52:53 +08:00
|
|
|
|
|
|
|
|
// 添加 FadeScene
|
|
|
|
|
const fadeScene = new FadeSceneClass();
|
|
|
|
|
phaserGame.scene.add(FADE_SCENE_KEY, fadeScene, false);
|
|
|
|
|
|
|
|
|
|
// 创建 SceneController
|
|
|
|
|
const currentScene = signal<string | null>(null);
|
|
|
|
|
const isTransitioning = signal(false);
|
|
|
|
|
|
|
|
|
|
const sceneController: SceneController = {
|
|
|
|
|
async launch(sceneKey: string, data?: Record<string, unknown>) {
|
|
|
|
|
if (isTransitioning.value) {
|
|
|
|
|
console.warn('SceneController: 正在进行场景切换');
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
isTransitioning.value = true;
|
|
|
|
|
const fade = phaserGame.scene.getScene(FADE_SCENE_KEY) as FadeSceneClass;
|
|
|
|
|
|
|
|
|
|
// 淡出到黑色
|
|
|
|
|
await fade.fadeOut(300);
|
|
|
|
|
|
|
|
|
|
// 停止当前场景
|
|
|
|
|
if (currentScene.value) {
|
|
|
|
|
phaserGame.scene.stop(currentScene.value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 启动新场景
|
|
|
|
|
phaserGame.scene.start(sceneKey, data);
|
|
|
|
|
currentScene.value = sceneKey;
|
|
|
|
|
|
|
|
|
|
// 淡入
|
|
|
|
|
await fade.fadeIn(300);
|
|
|
|
|
isTransitioning.value = false;
|
|
|
|
|
},
|
|
|
|
|
currentScene,
|
|
|
|
|
isTransitioning,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
gameSignal.value = { game: phaserGame, sceneController };
|
2026-04-04 13:08:28 +08:00
|
|
|
|
2026-04-04 12:14:26 +08:00
|
|
|
return () => {
|
2026-04-12 16:52:53 +08:00
|
|
|
gameSignal.value = { game: undefined!, sceneController: undefined! };
|
|
|
|
|
registeredScenes.value.clear();
|
2026-04-04 12:14:26 +08:00
|
|
|
phaserGame.destroy(true);
|
2026-04-04 13:08:28 +08:00
|
|
|
};
|
2026-04-04 12:14:26 +08:00
|
|
|
});
|
2026-04-04 13:08:28 +08:00
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<div id="phaser-container" className="w-full h-full">
|
|
|
|
|
<phaserContext.Provider value={gameSignal}>
|
|
|
|
|
{props.children}
|
|
|
|
|
</phaserContext.Provider>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
2026-04-04 12:14:26 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-12 16:26:52 +08:00
|
|
|
export interface PhaserSceneProps<TData extends Record<string, unknown> = {}> {
|
2026-04-04 13:08:28 +08:00
|
|
|
sceneKey: string;
|
2026-04-12 16:26:52 +08:00
|
|
|
scene: ReactiveScene<TData>;
|
|
|
|
|
data?: TData;
|
2026-04-04 16:19:18 +08:00
|
|
|
children?: any;
|
2026-04-04 13:08:28 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-12 16:31:10 +08:00
|
|
|
export const phaserSceneContext = createContext<ReadonlySignal<ReactiveScene> | null>(null);
|
2026-04-12 16:26:52 +08:00
|
|
|
export function PhaserScene<TData extends Record<string, unknown> = {}>(props: PhaserSceneProps<TData>) {
|
|
|
|
|
const phaserGameSignal = useContext(phaserContext);
|
2026-04-12 16:52:53 +08:00
|
|
|
const sceneSignal = useSignal<ReactiveScene<TData>>();
|
2026-04-04 13:08:28 +08:00
|
|
|
|
2026-04-04 12:14:26 +08:00
|
|
|
useSignalEffect(() => {
|
2026-04-12 16:26:52 +08:00
|
|
|
if (!phaserGameSignal) return;
|
|
|
|
|
const ctx = phaserGameSignal.value;
|
|
|
|
|
if (!ctx?.game) return;
|
|
|
|
|
|
|
|
|
|
const game = ctx.game;
|
|
|
|
|
const initData = {
|
|
|
|
|
...props.data,
|
2026-04-12 16:52:53 +08:00
|
|
|
phaserGame: phaserGameSignal,
|
|
|
|
|
sceneController: ctx.sceneController,
|
|
|
|
|
};
|
2026-04-04 13:08:28 +08:00
|
|
|
|
2026-04-12 16:52:53 +08:00
|
|
|
// 注册场景但不启动
|
|
|
|
|
if (!game.scene.getScene(props.sceneKey)) {
|
|
|
|
|
game.scene.add(props.sceneKey, props.scene, false, initData);
|
|
|
|
|
}
|
|
|
|
|
sceneSignal.value = props.scene;
|
|
|
|
|
|
2026-04-04 12:14:26 +08:00
|
|
|
return () => {
|
2026-04-04 16:19:18 +08:00
|
|
|
sceneSignal.value = undefined;
|
2026-04-12 16:52:53 +08:00
|
|
|
// 不在这里移除场景,让 SceneController 管理生命周期
|
2026-04-04 13:08:28 +08:00
|
|
|
};
|
2026-04-04 12:14:26 +08:00
|
|
|
});
|
2026-04-04 13:08:28 +08:00
|
|
|
|
2026-04-12 16:31:10 +08:00
|
|
|
return <phaserSceneContext.Provider value={sceneSignal as ReadonlySignal<ReactiveScene>}>{props.children}</phaserSceneContext.Provider>;
|
2026-04-04 13:08:28 +08:00
|
|
|
}
|