Compare commits

...

2 Commits

Author SHA1 Message Date
hypercross 71ccb8732e refactor: some manual changes 2026-04-08 14:21:33 +08:00
hypercross 92e1a5bec1 feat: rotation and more 2026-04-08 11:49:25 +08:00
4 changed files with 74 additions and 49 deletions

View File

@ -57,15 +57,15 @@ export class OnitamaScene extends GameHostScene<OnitamaState> {
// Info text // Info text
this.infoText = this.add.text( this.infoText = this.add.text(
BOARD_OFFSET.x + (BOARD_SIZE * CELL_SIZE) / 2, 20,
BOARD_OFFSET.y + BOARD_SIZE * CELL_SIZE + 30, BOARD_OFFSET.y,
'', '',
{ {
fontSize: '20px', fontSize: '15px',
fontFamily: 'Arial', fontFamily: 'Arial',
color: '#4b5563', color: '#4b5563',
} }
).setOrigin(0.5); );
// Update info text when UI state changes // Update info text when UI state changes
this.addEffect(() => { this.addEffect(() => {
@ -80,12 +80,13 @@ export class OnitamaScene extends GameHostScene<OnitamaState> {
const boardLeft = BOARD_OFFSET.x; const boardLeft = BOARD_OFFSET.x;
const boardTop = BOARD_OFFSET.y; const boardTop = BOARD_OFFSET.y;
const boardRight = BOARD_OFFSET.x + BOARD_SIZE * CELL_SIZE; const boardRight = BOARD_OFFSET.x + BOARD_SIZE * CELL_SIZE;
const boardBottom = BOARD_OFFSET.y + BOARD_SIZE * CELL_SIZE;
// Red cards label // Red cards label - 棋盘下方
const redLabel = this.add.text( const redLabel = this.add.text(
boardLeft - CARD_WIDTH - 60 + CARD_WIDTH / 2, boardLeft + (BOARD_SIZE * CELL_SIZE) / 2,
boardTop + 50 + 2 * (CARD_HEIGHT + 15) + 10, boardBottom + 10,
"RED's Cards", "RED",
{ {
fontSize: '16px', fontSize: '16px',
fontFamily: 'Arial', fontFamily: 'Arial',
@ -94,47 +95,27 @@ export class OnitamaScene extends GameHostScene<OnitamaState> {
).setOrigin(0.5, 0); ).setOrigin(0.5, 0);
this.cardLabelContainers.set('red', redLabel); this.cardLabelContainers.set('red', redLabel);
// Black cards label // Black cards label - 棋盘上方
const blackLabel = this.add.text( const blackLabel = this.add.text(
boardRight + 60 + CARD_WIDTH / 2, boardLeft + (BOARD_SIZE * CELL_SIZE) / 2,
boardTop + 50 + 2 * (CARD_HEIGHT + 15) + 10, boardTop - 10,
"BLACK's Cards", "BLACK",
{ {
fontSize: '16px', fontSize: '16px',
fontFamily: 'Arial', fontFamily: 'Arial',
color: '#3b82f6', color: '#3b82f6',
} }
).setOrigin(0.5, 0); ).setOrigin(0.5, 1);
this.cardLabelContainers.set('black', blackLabel); this.cardLabelContainers.set('black', blackLabel);
// Spare card label
const boardCenterX = boardLeft + (BOARD_SIZE * CELL_SIZE) / 2;
const spareLabel = this.add.text(
boardCenterX,
boardTop - 50,
'Spare Card',
{
fontSize: '16px',
fontFamily: 'Arial',
color: '#6b7280',
}
).setOrigin(0.5, 0);
this.cardLabelContainers.set('spare', spareLabel);
} }
private updateInfoText(): void { private updateInfoText(): void {
const currentPlayer = this.state.currentPlayer; const currentPlayer = this.state.currentPlayer;
const selectedCard = this.uiState.value.selectedCard;
const selectedPiece = this.uiState.value.selectedPiece;
if (this.state.winner) { if (this.state.winner) {
this.infoText.setText(`${this.state.winner} wins!`); this.infoText.setText(`${this.state.winner} wins!`);
} else if (!selectedCard) {
this.infoText.setText(`${currentPlayer}'s turn - Select a card first`);
} else if (!selectedPiece) {
this.infoText.setText(`Card: ${selectedCard} - Select a piece to move`);
} else { } else {
this.infoText.setText(`${currentPlayer}'s turn (Turn ${this.state.turn + 1})`); this.infoText.setText(`Turn ${this.state.turn + 1}\n\n${currentPlayer}`);
} }
} }
@ -160,15 +141,15 @@ export class OnitamaScene extends GameHostScene<OnitamaState> {
g.strokePath(); g.strokePath();
this.add.text( this.add.text(
BOARD_OFFSET.x + (BOARD_SIZE * CELL_SIZE) / 2, 20,
BOARD_OFFSET.y - 40, 20,
'Onitama', 'Onitama',
{ {
fontSize: '28px', fontSize: '28px',
fontFamily: 'Arial', fontFamily: 'Arial',
color: '#1f2937', color: '#1f2937',
} }
).setOrigin(0.5); );
} }
private setupInput(): void { private setupInput(): void {

View File

@ -23,14 +23,19 @@ export class CardContainer extends Phaser.GameObjects.Container {
private highlightRect: Phaser.GameObjects.Rectangle | null = null; private highlightRect: Phaser.GameObjects.Rectangle | null = null;
private highlightTween: Phaser.Tweens.Tween | null = null; private highlightTween: Phaser.Tweens.Tween | null = null;
private _cardId: string; private _cardId: string;
private _rotation: number = 0;
constructor(scene: OnitamaScene, cardId: string, card: Card) { constructor(scene: OnitamaScene, cardId: string, card: Card, rotation: number = 0) {
super(scene, 0, 0); super(scene, 0, 0);
this._cardId = cardId; this._cardId = cardId;
this._rotation = rotation;
// 将容器添加到场景 // 将容器添加到场景
scene.add.existing(this); scene.add.existing(this);
// 应用旋转
this.angle = rotation;
// 创建卡牌视觉 // 创建卡牌视觉
this.createCardVisual(card); this.createCardVisual(card);
@ -42,6 +47,26 @@ export class CardContainer extends Phaser.GameObjects.Container {
this.addHighlightEffect(scene); this.addHighlightEffect(scene);
} }
/**
*
*/
setRotation(rotation: number, animated: boolean = false): this {
if (animated) {
const currentAngle = this.angle;
const tween = this.scene.tweens.add({
targets: this,
angle: { from: currentAngle, to: rotation },
duration: 400,
ease: 'Back.easeOut',
});
(this.scene as OnitamaScene).addTweenInterruption(tween);
} else {
this.angle = rotation;
}
this._rotation = rotation;
return this;
}
/** /**
* *
*/ */
@ -225,21 +250,26 @@ export class CardSpawner implements Spawner<CardSpawnData, CardContainer> {
const boardTop = BOARD_OFFSET.y; const boardTop = BOARD_OFFSET.y;
const boardRight = BOARD_OFFSET.x + BOARD_SIZE * CELL_SIZE; const boardRight = BOARD_OFFSET.x + BOARD_SIZE * CELL_SIZE;
const boardCenterX = boardLeft + (BOARD_SIZE * CELL_SIZE) / 2; const boardCenterX = boardLeft + (BOARD_SIZE * CELL_SIZE) / 2;
const boardBottom = BOARD_OFFSET.y + BOARD_SIZE * CELL_SIZE;
if (data.position === 'red') { if (data.position === 'red') {
// 红方卡牌在棋盘下方,水平排列
return { return {
x: boardLeft - CARD_WIDTH - 60 + 60, x: boardLeft + BOARD_SIZE * CELL_SIZE / 2 - (data.index - 0.5) * (CARD_WIDTH + 15),
y: boardTop + 80 + data.index * (CARD_HEIGHT + 15), y: boardBottom + CARD_HEIGHT - 30,
}; };
} else if (data.position === 'black') { } else if (data.position === 'black') {
// 黑方卡牌在棋盘上方,水平排列
return { return {
x: boardRight + 60 + 40, x: boardLeft + BOARD_SIZE * CELL_SIZE / 2 - (data.index - 0.5) * (CARD_WIDTH + 15),
y: boardTop + 80 + data.index * (CARD_HEIGHT + 15), y: boardTop - CARD_HEIGHT + 30,
}; };
} else { } else {
// 备用卡牌在棋盘左侧,垂直居中
const boardCenterY = boardTop + (BOARD_SIZE * CELL_SIZE) / 2;
return { return {
x: boardCenterX, x: boardLeft - CARD_WIDTH,
y: boardTop - CARD_HEIGHT - 20, y: boardCenterY,
}; };
} }
} }
@ -251,6 +281,20 @@ export class CardSpawner implements Spawner<CardSpawnData, CardContainer> {
} }
onUpdate(data: CardSpawnData, obj: CardContainer): void { onUpdate(data: CardSpawnData, obj: CardContainer): void {
const isBlackTurn = this.scene.state.currentPlayer === 'black';
// 检查卡牌是否需要旋转
let targetRotation = 0;
if (data.position === 'black') {
targetRotation = -180; // 黑方卡牌始终旋转
} else if (data.position === 'spare' && isBlackTurn) {
targetRotation = -180; // 备用卡牌在黑方回合旋转
}
if (obj.angle !== targetRotation) {
obj.setRotation(targetRotation, true);
}
// 只在位置实际变化时才播放移动动画 // 只在位置实际变化时才播放移动动画
if (!this.hasPositionChanged(data)) { if (!this.hasPositionChanged(data)) {
this.previousData.set(data.cardId, { ...data }); this.previousData.set(data.cardId, { ...data });
@ -285,7 +329,8 @@ export class CardSpawner implements Spawner<CardSpawnData, CardContainer> {
return emptyContainer; return emptyContainer;
} }
const container = new CardContainer(this.scene, data.cardId, card); // 计算旋转角度 - 初始不旋转,后续由 onUpdate 处理
const container = new CardContainer(this.scene, data.cardId, card, 0);
const pos = this.getCardPosition(data); const pos = this.getCardPosition(data);
container.x = pos.x; container.x = pos.x;
container.y = pos.y; container.y = pos.y;

View File

@ -6,7 +6,7 @@ import type { OnitamaUIState } from '@/state';
import { effect } from "@preact/signals-core"; import { effect } from "@preact/signals-core";
export const CELL_SIZE = 80; export const CELL_SIZE = 80;
export const BOARD_OFFSET = { x: 200, y: 180 }; export const BOARD_OFFSET = { x: 240, y: 200 };
export const BOARD_SIZE = 5; export const BOARD_SIZE = 5;
export function boardToScreen(boardX: number, boardY: number): { x: number; y: number } { export function boardToScreen(boardX: number, boardY: number): { x: number; y: number } {
@ -228,7 +228,6 @@ export class PawnSpawner implements Spawner<Pawn, PawnContainer> {
targets: obj, targets: obj,
scale: 0, scale: 0,
alpha: 0, alpha: 0,
y: obj.y - 30,
duration: 300, duration: 300,
ease: 'Back.easeIn', ease: 'Back.easeIn',
onComplete: () => obj.destroy(), onComplete: () => obj.destroy(),

View File

@ -19,8 +19,8 @@ export default function App(props: { gameModule: any, gameScene: { new(): Phaser
const label = useComputed(() => gameHost.value.gameHost.status.value === 'running' ? 'Restart' : 'Start'); const label = useComputed(() => gameHost.value.gameHost.status.value === 'running' ? 'Restart' : 'Start');
const phaserConfig: Partial<Phaser.Types.Core.GameConfig> = { const phaserConfig: Partial<Phaser.Types.Core.GameConfig> = {
width: 800, width: 700,
height: 700, height: 800,
}; };
return ( return (