boardgame-phaser/packages/framework/src/ui/PhaserBridge.tsx

88 lines
2.7 KiB
TypeScript
Raw Normal View History

2026-04-04 13:08:28 +08:00
import Phaser from 'phaser';
2026-04-06 11:22:33 +08:00
import { 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:26:52 +08:00
export interface PhaserGameContext {
game: Phaser.Game;
}
export const phaserContext = createContext<ReadonlySignal<PhaserGameContext> | null>(null);
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:26:52 +08:00
const gameSignal = useSignal<PhaserGameContext>({ game: undefined! });
2026-04-04 13:08:28 +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:26:52 +08:00
gameSignal.value = { game: phaserGame };
2026-04-04 13:08:28 +08:00
return () => {
2026-04-12 16:26:52 +08:00
gameSignal.value = { game: undefined! };
phaserGame.destroy(true);
2026-04-04 13:08:28 +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-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>;
2026-04-04 13:08:28 +08:00
autoStart: boolean;
2026-04-12 16:26:52 +08:00
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-04 16:19:18 +08:00
const sceneSignal = useSignal<Phaser.Scene>();
2026-04-04 13:08:28 +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,
phaserGame: phaserGameSignal.value,
} as TData & ReactiveScenePhaserData;
2026-04-04 13:08:28 +08:00
2026-04-12 16:26:52 +08:00
game.scene.add(props.sceneKey, props.scene, props.autoStart, initData);
2026-04-04 16:19:18 +08:00
sceneSignal.value = game.scene.getScene(props.sceneKey);
return () => {
2026-04-04 16:19:18 +08:00
sceneSignal.value = undefined;
game.scene.remove(props.sceneKey);
2026-04-04 13:08:28 +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
}