Compare commits

..

No commits in common. "c29b9a43b3ce95ab2dfc0a9510808661e1100c2f" and "a181932fb31fe44040c43d7ba661adbe46079bf0" have entirely different histories.

13 changed files with 141 additions and 371 deletions

View File

@ -5,11 +5,10 @@ export type { IDisposable, DisposableItem } from "./utils";
// Drag & drop utilities // Drag & drop utilities
export { dragDropEventEffect, DragDropEventType } from "./utils"; export { dragDropEventEffect, DragDropEventType } from "./utils";
export type { DragDropEvent, DragDropCallback } from "./utils"; export type { DragDropEvent, DragDropCallback } from "./utils";
export { hoverEffect } from "./utils";
// Data-driven object spawning // Data-driven object spawning
export { spawnEffect, createSpawnUpdate } from "./spawner"; export { spawnEffect } from "./spawner";
export type { Spawner, SpawnerCallback } from "./spawner"; export type { Spawner } from "./spawner";
// Scene base classes // Scene base classes
export { export {

View File

@ -4,7 +4,9 @@ import type Phaser from "phaser";
type GO = Phaser.GameObjects.GameObject; type GO = Phaser.GameObjects.GameObject;
export interface SpawnerCallback<TData, TObject extends GO = GO> { export interface Spawner<TData, TObject extends GO = GO> {
/** 数据源迭代器 */
getData(): Iterable<TData>;
/** 获取数据的唯一键 */ /** 获取数据的唯一键 */
getKey(t: TData): string; getKey(t: TData): string;
/** 创建新对象 */ /** 创建新对象 */
@ -15,33 +17,15 @@ export interface SpawnerCallback<TData, TObject extends GO = GO> {
onUpdate(t: TData, obj: TObject): void; onUpdate(t: TData, obj: TObject): void;
} }
export interface Spawner<
TData,
TObject extends GO = GO,
> extends SpawnerCallback<TData, TObject> {
/** 数据源迭代器 */
getData(): Iterable<TData>;
}
export function spawnEffect<TData, TObject extends GO = GO>( export function spawnEffect<TData, TObject extends GO = GO>(
spawner: Spawner<TData, TObject>, spawner: Spawner<TData, TObject>,
) { ): () => void {
const update = createSpawnUpdate(spawner);
return effect(() => {
update(spawner.getData());
});
}
export function createSpawnUpdate<TData, TObject extends GO = GO>(
spawner: SpawnerCallback<TData, TObject>,
) {
const entries = new Map<string, { object: TObject; data: TData }>(); const entries = new Map<string, { object: TObject; data: TData }>();
function update(data: Iterable<TData>) { return effect(() => {
const current = new Set<string>(); const current = new Set<string>();
for (const t of data) { for (const t of spawner.getData()) {
const key = spawner.getKey(t); const key = spawner.getKey(t);
current.add(key); current.add(key);
@ -64,7 +48,5 @@ export function createSpawnUpdate<TData, TObject extends GO = GO>(
entries.delete(key); entries.delete(key);
} }
} }
} });
return update;
} }

View File

@ -1,28 +0,0 @@
type HoverCallback = (hovering: boolean) => void;
export function hoverEffect(
gameObject: Phaser.GameObjects.GameObject,
callback: HoverCallback,
) {
let isHovering = false;
const onPointerOver = () => {
if (isHovering) return;
isHovering = true;
callback(true);
};
const onPointerOut = () => {
if (!isHovering) return;
isHovering = false;
callback(false);
};
gameObject.on("pointerover", onPointerOver);
gameObject.on("pointerout", onPointerOut);
const cleanup = () => {
gameObject.off("pointerover", onPointerOver);
gameObject.off("pointerout", onPointerOut);
gameObject.off("destroy", cleanup);
};
gameObject.once("destroy", cleanup);
return cleanup;
}

View File

@ -6,4 +6,3 @@ export {
type DragDropEvent, type DragDropEvent,
type DragDropCallback, type DragDropCallback,
} from "./dnd"; } from "./dnd";
export * from "./hover";

View File

@ -19,7 +19,6 @@
"@preact/preset-vite": "^2.8.1", "@preact/preset-vite": "^2.8.1",
"@preact/signals": "^2.9.0", "@preact/signals": "^2.9.0",
"@tailwindcss/vite": "^4.0.0", "@tailwindcss/vite": "^4.0.0",
"@types/node": "^25.6.0",
"tailwindcss": "^4.0.0", "tailwindcss": "^4.0.0",
"typescript": "^5.3.3", "typescript": "^5.3.3",
"vite": "^5.1.0" "vite": "^5.1.0"

View File

@ -156,12 +156,6 @@ export const NEGATIVE_EFFECTS = new Set([
"poison", "poison",
]); ]);
export const CARD_CONFIG = {
WIDTH: 140,
HEIGHT: 200,
CORNER_RADIUS: 8,
} as const;
export const GAME_CONFIG: Phaser.Types.Core.GameConfig = { export const GAME_CONFIG: Phaser.Types.Core.GameConfig = {
width: 1920, width: 1920,
height: 1080, height: 1080,

View File

@ -1,132 +0,0 @@
import Phaser from "phaser";
export type BuffData = {
id: string;
stacks: number;
emoji: string;
isPositive: boolean;
description?: string;
};
const ICON_SIZE = 28;
const ICON_COLOR_POSITIVE = 0x44aa44;
const ICON_COLOR_NEGATIVE = 0xaa4444;
const TOOLTIP_WIDTH = 200;
const TOOLTIP_PADDING = 8;
export class Buff extends Phaser.GameObjects.Container {
private bg!: Phaser.GameObjects.Rectangle;
private label!: Phaser.GameObjects.Text;
private tooltipContainer!: Phaser.GameObjects.Container;
private tooltipBg!: Phaser.GameObjects.Rectangle;
private tooltipText!: Phaser.GameObjects.Text;
private currentData!: BuffData;
constructor(scene: Phaser.Scene, x: number, y: number, data: BuffData) {
super(scene, x, y);
scene.add.existing(this);
this.currentData = data;
this.createVisuals();
this.createTooltip();
this.setupInteraction();
this.updateFromData(data);
}
private createVisuals(): void {
const color = this.currentData.isPositive
? ICON_COLOR_POSITIVE
: ICON_COLOR_NEGATIVE;
this.bg = this.scene.add
.rectangle(0, 0, ICON_SIZE, ICON_SIZE, color)
.setStrokeStyle(1, 0xffffff);
this.label = this.scene.add
.text(0, 0, "", {
fontSize: "12px",
color: "#ffffff",
fontStyle: "bold",
})
.setOrigin(0.5);
this.add([this.bg, this.label]);
}
private createTooltip(): void {
this.tooltipContainer = this.scene.add.container(0, 0).setVisible(false);
this.tooltipBg = this.scene.add
.rectangle(0, 0, 0, 0, 0x000000, 0.9)
.setStrokeStyle(1, 0xffffff);
this.tooltipText = this.scene.add
.text(0, 0, "", {
fontSize: "12px",
color: "#ffffff",
align: "center",
wordWrap: { width: TOOLTIP_WIDTH - TOOLTIP_PADDING * 2 },
})
.setOrigin(0.5);
this.tooltipContainer.add([this.tooltipBg, this.tooltipText]);
this.tooltipContainer.setDepth(1000);
}
private setupInteraction(): void {
this.setInteractive(
new Phaser.Geom.Rectangle(
-ICON_SIZE / 2,
-ICON_SIZE / 2,
ICON_SIZE,
ICON_SIZE,
),
Phaser.Geom.Rectangle.Contains,
);
this.on("pointerover", this.onPointerOver, this);
this.on("pointerout", this.onPointerOut, this);
}
private onPointerOver(): void {
const desc = this.currentData.description ?? "";
const content = desc;
this.tooltipText.setText(content);
const bounds = this.tooltipText.getBounds();
const width = Math.min(bounds.width + TOOLTIP_PADDING * 2, TOOLTIP_WIDTH);
const height = bounds.height + TOOLTIP_PADDING * 2;
this.tooltipBg.setDisplaySize(width, height);
const worldPos = this.getBounds();
this.tooltipContainer.setPosition(
worldPos.x + worldPos.width / 2,
worldPos.y - height / 2 - 4,
);
this.tooltipContainer.setDepth(1000);
this.scene.children.bringToTop(this.tooltipContainer);
this.tooltipContainer.setVisible(true);
}
private onPointerOut(): void {
this.tooltipContainer.setVisible(false);
}
updateFromData(data: BuffData): void {
this.currentData = data;
this.label.setText(`${data.emoji}${data.stacks}`);
const color = data.isPositive ? ICON_COLOR_POSITIVE : ICON_COLOR_NEGATIVE;
this.bg.setFillStyle(color);
}
destroy(fromScene?: boolean): void {
this.off("pointerover", this.onPointerOver, this);
this.off("pointerout", this.onPointerOut, this);
this.tooltipContainer.destroy(fromScene);
super.destroy(fromScene);
}
}

View File

@ -1,25 +0,0 @@
import { BuffData, Buff } from "./Buff";
import { SpawnerCallback } from "boardgame-phaser";
export type BuffDataWithIndex = BuffData & { x: number };
export class BuffSpawner implements SpawnerCallback<BuffDataWithIndex, Buff> {
constructor(
public scene: Phaser.Scene,
public buffContainer: Phaser.GameObjects.Container,
) {}
getKey(t: BuffDataWithIndex) {
return t.id;
}
onSpawn(t: BuffDataWithIndex) {
const buff = new Buff(this.scene, t.x, 0, t);
this.buffContainer.add(buff);
return buff;
}
onDespawn(obj: Buff) {
obj.destroy();
}
onUpdate(t: BuffDataWithIndex, obj: Buff) {
obj.setPosition(t.x, 0);
obj.updateFromData(t);
}
}

View File

@ -1,7 +1,8 @@
import Phaser, { Tweens } from "phaser"; import Phaser from "phaser";
import type { GameCard } from "boardgame-core/samples/slay-the-spire-like"; import type {
import { CARD_CONFIG } from "@/config"; CardData,
import { hoverEffect } from "boardgame-phaser"; GameCard,
} from "boardgame-core/samples/slay-the-spire-like";
export interface CardContainerOptions { export interface CardContainerOptions {
card: GameCard; card: GameCard;
@ -9,9 +10,9 @@ export interface CardContainerOptions {
playable?: boolean; playable?: boolean;
} }
const CARD_WIDTH = CARD_CONFIG.WIDTH; const CARD_WIDTH = 140;
const CARD_HEIGHT = CARD_CONFIG.HEIGHT; const CARD_HEIGHT = 200;
const CORNER_RADIUS = CARD_CONFIG.CORNER_RADIUS; const CORNER_RADIUS = 8;
export class CardContainer extends Phaser.GameObjects.Container { export class CardContainer extends Phaser.GameObjects.Container {
private bg!: Phaser.GameObjects.Rectangle; private bg!: Phaser.GameObjects.Rectangle;
@ -86,18 +87,8 @@ export class CardContainer extends Phaser.GameObjects.Container {
this.add(this.descText); this.add(this.descText);
// Target indicator // Target indicator
const targetLabel = (() => { const targetLabel =
switch (cardData.targetType) { cardData.targetType === "single" ? "🎯 Single" : "✨ Self";
case "enemy":
return "⚔️ Enemy";
case "enemies":
return "⚔️ Enemies";
case "player":
return "🛡️ Player";
default:
return "❓ Unknown";
}
})();
const targetText = this.scene.add const targetText = this.scene.add
.text(0, CARD_HEIGHT / 2 - 20, targetLabel, { .text(0, CARD_HEIGHT / 2 - 20, targetLabel, {
fontSize: "10px", fontSize: "10px",
@ -119,9 +110,16 @@ export class CardContainer extends Phaser.GameObjects.Container {
this.input.cursor = options.playable ? "pointer" : "not-allowed"; this.input.cursor = options.playable ? "pointer" : "not-allowed";
} }
hoverEffect(this, (hovering) => { this.on("pointerover", () => {
if (!options.playable) return; if (options.playable && !this._selected) {
this.playHoverTween(hovering); this.hoverIn();
}
});
this.on("pointerout", () => {
if (!this._selected) {
this.hoverOut();
}
}); });
this.on("pointerdown", () => { this.on("pointerdown", () => {
@ -131,6 +129,28 @@ export class CardContainer extends Phaser.GameObjects.Container {
}); });
} }
private hoverIn(): void {
this.scene.tweens.add({
targets: this,
y: this.y - 10,
scale: 1.08,
duration: 150,
ease: "Power2",
});
this.bg.setStrokeStyle(3, 0x88aaff);
}
private hoverOut(): void {
this.scene.tweens.add({
targets: this,
y: this.y + 10,
scale: 1,
duration: 150,
ease: "Power2",
});
this.bg.setStrokeStyle(2, 0x555577);
}
setSelected(selected: boolean): void { setSelected(selected: boolean): void {
this._selected = selected; this._selected = selected;
if (selected) { if (selected) {
@ -192,20 +212,6 @@ export class CardContainer extends Phaser.GameObjects.Container {
}); });
} }
private playHoverTween(isHovering: boolean) {
this.scene.tweens.add({
targets: this,
y: isHovering ? this.y - 10 : this.y + 10,
scale: isHovering ? 1.08 : 1,
duration: 150,
ease: "Power2",
});
this.bg.setStrokeStyle(
isHovering ? 3 : 2,
isHovering ? 0x88aaff : 0x555577,
);
}
destroy(fromScene?: boolean): void { destroy(fromScene?: boolean): void {
this.highlight = null; this.highlight = null;
super.destroy(fromScene); super.destroy(fromScene);

View File

@ -95,7 +95,7 @@ export class CardSpawner implements Spawner<CardSpawnData, CardContainer> {
type: "item", type: "item",
costType: "none", costType: "none",
costCount: 0, costCount: 0,
targetType: "player", targetType: "none",
effects: [], effects: [],
}, },
itemId: "", itemId: "",

View File

@ -1,12 +1,5 @@
import Phaser from "phaser"; import Phaser from "phaser";
import type { import type { CombatEntity, EffectTable, EnemyEntity } from "boardgame-core/samples/slay-the-spire-like";
CombatEntity,
EffectTable,
EnemyEntity,
} from "boardgame-core/samples/slay-the-spire-like";
import { BuffData } from "./Buff";
import { createSpawnUpdate } from "boardgame-phaser";
import { BuffDataWithIndex, BuffSpawner } from "./BuffSpawner";
export type CombatUnitData = { export type CombatUnitData = {
key: string; key: string;
@ -30,7 +23,6 @@ export class CombatUnitContainer extends Phaser.GameObjects.Container {
private hpText!: Phaser.GameObjects.Text; private hpText!: Phaser.GameObjects.Text;
private buffContainer!: Phaser.GameObjects.Container; private buffContainer!: Phaser.GameObjects.Container;
private intentText!: Phaser.GameObjects.Text | null; private intentText!: Phaser.GameObjects.Text | null;
private updateBuffs!: (buffs: Iterable<BuffData & { x: number }>) => void;
private currentEntity!: CombatEntity; private currentEntity!: CombatEntity;
private currentName: string; private currentName: string;
@ -64,22 +56,11 @@ export class CombatUnitContainer extends Phaser.GameObjects.Container {
}) })
.setOrigin(0.5); .setOrigin(0.5);
this.hpBarBg = this.scene.add.rectangle( this.hpBarBg = this.scene.add
0, .rectangle(0, -CONTAINER_HEIGHT / 2 + 60, HP_BAR_WIDTH, HP_BAR_HEIGHT, 0x333333);
-CONTAINER_HEIGHT / 2 + 60,
HP_BAR_WIDTH,
HP_BAR_HEIGHT,
0x333333,
);
this.hpBarFill = this.scene.add this.hpBarFill = this.scene.add
.rectangle( .rectangle(-HP_BAR_WIDTH / 2, -CONTAINER_HEIGHT / 2 + 60, HP_BAR_WIDTH, HP_BAR_HEIGHT, 0x22c55e)
-HP_BAR_WIDTH / 2,
-CONTAINER_HEIGHT / 2 + 60,
HP_BAR_WIDTH,
HP_BAR_HEIGHT,
0x22c55e,
)
.setOrigin(0, 0.5); .setOrigin(0, 0.5);
this.hpText = this.scene.add this.hpText = this.scene.add
@ -91,9 +72,6 @@ export class CombatUnitContainer extends Phaser.GameObjects.Container {
.setOrigin(0.5); .setOrigin(0.5);
this.buffContainer = this.scene.add.container(0, CONTAINER_HEIGHT / 2 - 40); this.buffContainer = this.scene.add.container(0, CONTAINER_HEIGHT / 2 - 40);
this.updateBuffs = createSpawnUpdate(
new BuffSpawner(this.scene, this.buffContainer),
);
if (!this.currentIsPlayer) { if (!this.currentIsPlayer) {
this.intentText = this.scene.add this.intentText = this.scene.add
@ -123,23 +101,19 @@ export class CombatUnitContainer extends Phaser.GameObjects.Container {
this.nameText.setText(this.currentName); this.nameText.setText(this.currentName);
const hpPercent = const hpPercent = this.currentEntity.maxHp > 0
this.currentEntity.maxHp > 0 ? this.currentEntity.hp / this.currentEntity.maxHp
? this.currentEntity.hp / this.currentEntity.maxHp : 0;
: 0;
const fillWidth = Math.max(0, HP_BAR_WIDTH * hpPercent); const fillWidth = Math.max(0, HP_BAR_WIDTH * hpPercent);
this.hpBarFill.setDisplaySize(fillWidth, HP_BAR_HEIGHT); this.hpBarFill.setDisplaySize(fillWidth, HP_BAR_HEIGHT);
const hpColor = const hpColor = hpPercent > 0.5 ? 0x22c55e : hpPercent > 0.25 ? 0xf59e0b : 0xef4444;
hpPercent > 0.5 ? 0x22c55e : hpPercent > 0.25 ? 0xf59e0b : 0xef4444;
this.hpBarFill.setFillStyle(hpColor); this.hpBarFill.setFillStyle(hpColor);
this.hpText.setText( this.hpText.setText(`${this.currentEntity.hp} / ${this.currentEntity.maxHp} HP`);
`${this.currentEntity.hp} / ${this.currentEntity.maxHp} HP`,
);
this.updateBuffs(this.getBuffEntries()); this.renderBuffs(this.currentEntity.effects);
if (!this.currentIsPlayer && this.intentText) { if (!this.currentIsPlayer && this.intentText) {
const enemyEntity = data.entity as EnemyEntity; const enemyEntity = data.entity as EnemyEntity;
@ -149,20 +123,47 @@ export class CombatUnitContainer extends Phaser.GameObjects.Container {
} }
} }
private *getBuffEntries() { private renderBuffs(effects: EffectTable): void {
const entries = Object.values(this.currentEntity.effects); this.buffContainer.removeAll(true);
const totalWidth =
entries.length * BUFF_ICON_SIZE + (entries.length - 1) * BUFF_ICON_GAP; const entries = Object.entries(effects);
const totalWidth = entries.length * BUFF_ICON_SIZE + (entries.length - 1) * BUFF_ICON_GAP;
const startX = -totalWidth / 2 + BUFF_ICON_SIZE / 2; const startX = -totalWidth / 2 + BUFF_ICON_SIZE / 2;
for (let i = 0; i < entries.length; i++) {
const buff = entries[i]; entries.forEach(([key, entry], index) => {
yield { const x = startX + index * (BUFF_ICON_SIZE + BUFF_ICON_GAP);
...buff.data, const stacks = entry.stacks;
stacks: buff.stacks,
isPositive: true, const isPositive = this.isPositiveEffect(key);
x: startX + i * BUFF_ICON_SIZE, const iconColor = isPositive ? 0x44aa44 : 0xaa4444;
} as BuffDataWithIndex;
} const bg = this.scene.add
.rectangle(x, 0, BUFF_ICON_SIZE, BUFF_ICON_SIZE, iconColor)
.setStrokeStyle(1, 0xffffff);
const text = this.scene.add
.text(x, 0, `${stacks}`, {
fontSize: "12px",
color: "#ffffff",
fontStyle: "bold",
})
.setOrigin(0.5);
this.buffContainer.add([bg, text]);
});
}
private isPositiveEffect(effectId: string): boolean {
const positive = new Set([
"block",
"strength",
"dexterity",
"regen",
"armor",
"barrier",
"momentum",
]);
return positive.has(effectId.toLowerCase());
} }
playSpawnEffect(): void { playSpawnEffect(): void {
@ -180,14 +181,7 @@ export class CombatUnitContainer extends Phaser.GameObjects.Container {
playDamageEffect(): void { playDamageEffect(): void {
const flash = this.scene.add const flash = this.scene.add
.rectangle( .rectangle(this.x, this.y, CONTAINER_WIDTH, CONTAINER_HEIGHT, 0xff0000, 0.4)
this.x,
this.y,
CONTAINER_WIDTH,
CONTAINER_HEIGHT,
0xff0000,
0.4,
)
.setDepth(200); .setDepth(200);
this.scene.tweens.add({ this.scene.tweens.add({

View File

@ -129,7 +129,7 @@ export class CombatTestScene extends GameHostScene<CombatState> {
const targetType = card.cardData.targetType; const targetType = card.cardData.targetType;
if (targetType === "enemy") { if (targetType === "single") {
this.selectedCardId = cardId; this.selectedCardId = cardId;
this.isTargeting = true; this.isTargeting = true;
this.targetingText.setText("Select a target!"); this.targetingText.setText("Select a target!");

View File

@ -25,7 +25,7 @@ importers:
version: 8.58.2(eslint@9.39.4(jiti@2.6.1))(typescript@5.9.3) version: 8.58.2(eslint@9.39.4(jiti@2.6.1))(typescript@5.9.3)
vitest: vitest:
specifier: ^3.2.4 specifier: ^3.2.4
version: 3.2.4(@types/node@25.6.0)(lightningcss@1.32.0) version: 3.2.4(lightningcss@1.32.0)
packages/boop-game: packages/boop-game:
dependencies: dependencies:
@ -53,13 +53,13 @@ importers:
devDependencies: devDependencies:
'@preact/preset-vite': '@preact/preset-vite':
specifier: ^2.8.1 specifier: ^2.8.1
version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(lightningcss@1.32.0))
'@preact/signals': '@preact/signals':
specifier: ^2.9.0 specifier: ^2.9.0
version: 2.9.0(preact@10.29.0) version: 2.9.0(preact@10.29.0)
'@tailwindcss/vite': '@tailwindcss/vite':
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 4.2.2(vite@5.4.21(lightningcss@1.32.0))
tailwindcss: tailwindcss:
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2 version: 4.2.2
@ -68,7 +68,7 @@ importers:
version: 5.9.3 version: 5.9.3
vite: vite:
specifier: ^5.1.0 specifier: ^5.1.0
version: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) version: 5.4.21(lightningcss@1.32.0)
packages/framework: packages/framework:
devDependencies: devDependencies:
@ -98,7 +98,7 @@ importers:
version: 5.9.3 version: 5.9.3
vitest: vitest:
specifier: ^3.2.4 specifier: ^3.2.4
version: 3.2.4(@types/node@25.6.0)(lightningcss@1.32.0) version: 3.2.4(lightningcss@1.32.0)
packages/onitama-game: packages/onitama-game:
dependencies: dependencies:
@ -123,13 +123,13 @@ importers:
devDependencies: devDependencies:
'@preact/preset-vite': '@preact/preset-vite':
specifier: ^2.8.1 specifier: ^2.8.1
version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(lightningcss@1.32.0))
'@preact/signals': '@preact/signals':
specifier: ^2.9.0 specifier: ^2.9.0
version: 2.9.0(preact@10.29.0) version: 2.9.0(preact@10.29.0)
'@tailwindcss/vite': '@tailwindcss/vite':
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 4.2.2(vite@5.4.21(lightningcss@1.32.0))
tailwindcss: tailwindcss:
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2 version: 4.2.2
@ -138,7 +138,7 @@ importers:
version: 5.9.3 version: 5.9.3
vite: vite:
specifier: ^5.1.0 specifier: ^5.1.0
version: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) version: 5.4.21(lightningcss@1.32.0)
packages/regicide-game: packages/regicide-game:
dependencies: dependencies:
@ -163,13 +163,13 @@ importers:
devDependencies: devDependencies:
'@preact/preset-vite': '@preact/preset-vite':
specifier: ^2.8.1 specifier: ^2.8.1
version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(lightningcss@1.32.0))
'@preact/signals': '@preact/signals':
specifier: ^2.9.0 specifier: ^2.9.0
version: 2.9.0(preact@10.29.0) version: 2.9.0(preact@10.29.0)
'@tailwindcss/vite': '@tailwindcss/vite':
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 4.2.2(vite@5.4.21(lightningcss@1.32.0))
tailwindcss: tailwindcss:
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2 version: 4.2.2
@ -178,10 +178,10 @@ importers:
version: 5.9.3 version: 5.9.3
vite: vite:
specifier: ^5.1.0 specifier: ^5.1.0
version: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) version: 5.4.21(lightningcss@1.32.0)
vitest: vitest:
specifier: ^3.2.4 specifier: ^3.2.4
version: 3.2.4(@types/node@25.6.0)(lightningcss@1.32.0) version: 3.2.4(lightningcss@1.32.0)
packages/sample-game: packages/sample-game:
dependencies: dependencies:
@ -206,13 +206,13 @@ importers:
devDependencies: devDependencies:
'@preact/preset-vite': '@preact/preset-vite':
specifier: ^2.8.1 specifier: ^2.8.1
version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(lightningcss@1.32.0))
'@preact/signals': '@preact/signals':
specifier: ^2.9.0 specifier: ^2.9.0
version: 2.9.0(preact@10.29.0) version: 2.9.0(preact@10.29.0)
'@tailwindcss/vite': '@tailwindcss/vite':
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 4.2.2(vite@5.4.21(lightningcss@1.32.0))
tailwindcss: tailwindcss:
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2 version: 4.2.2
@ -221,7 +221,7 @@ importers:
version: 5.9.3 version: 5.9.3
vite: vite:
specifier: ^5.1.0 specifier: ^5.1.0
version: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) version: 5.4.21(lightningcss@1.32.0)
packages/sts-like-viewer: packages/sts-like-viewer:
dependencies: dependencies:
@ -246,16 +246,13 @@ importers:
devDependencies: devDependencies:
'@preact/preset-vite': '@preact/preset-vite':
specifier: ^2.8.1 specifier: ^2.8.1
version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(lightningcss@1.32.0))
'@preact/signals': '@preact/signals':
specifier: ^2.9.0 specifier: ^2.9.0
version: 2.9.0(preact@10.29.0) version: 2.9.0(preact@10.29.0)
'@tailwindcss/vite': '@tailwindcss/vite':
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) version: 4.2.2(vite@5.4.21(lightningcss@1.32.0))
'@types/node':
specifier: ^25.6.0
version: 25.6.0
tailwindcss: tailwindcss:
specifier: ^4.0.0 specifier: ^4.0.0
version: 4.2.2 version: 4.2.2
@ -264,7 +261,7 @@ importers:
version: 5.9.3 version: 5.9.3
vite: vite:
specifier: ^5.1.0 specifier: ^5.1.0
version: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) version: 5.4.21(lightningcss@1.32.0)
packages: packages:
@ -1023,9 +1020,6 @@ packages:
'@types/json5@0.0.29': '@types/json5@0.0.29':
resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==} resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==}
'@types/node@25.6.0':
resolution: {integrity: sha512-+qIYRKdNYJwY3vRCZMdJbPLJAtGjQBudzZzdzwQYkEPQd+PJGixUL5QfvCLDaULoLv+RhT3LDkwEfKaAkgSmNQ==}
'@typescript-eslint/eslint-plugin@8.58.2': '@typescript-eslint/eslint-plugin@8.58.2':
resolution: {integrity: sha512-aC2qc5thQahutKjP+cl8cgN9DWe3ZUqVko30CMSZHnFEHyhOYoZSzkGtAI2mcwZ38xeImDucI4dnqsHiOYuuCw==} resolution: {integrity: sha512-aC2qc5thQahutKjP+cl8cgN9DWe3ZUqVko30CMSZHnFEHyhOYoZSzkGtAI2mcwZ38xeImDucI4dnqsHiOYuuCw==}
engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0}
@ -2343,9 +2337,6 @@ packages:
resolution: {integrity: sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==} resolution: {integrity: sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
undici-types@7.19.2:
resolution: {integrity: sha512-qYVnV5OEm2AW8cJMCpdV20CDyaN3g0AjDlOGf1OW4iaDEx8MwdtChUp4zu4H0VP3nDRF/8RKWH+IPp9uW0YGZg==}
update-browserslist-db@1.2.3: update-browserslist-db@1.2.3:
resolution: {integrity: sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==} resolution: {integrity: sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==}
hasBin: true hasBin: true
@ -2823,19 +2814,19 @@ snapshots:
'@jridgewell/resolve-uri': 3.1.2 '@jridgewell/resolve-uri': 3.1.2
'@jridgewell/sourcemap-codec': 1.5.5 '@jridgewell/sourcemap-codec': 1.5.5
'@preact/preset-vite@2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0))': '@preact/preset-vite@2.10.5(@babel/core@7.29.0)(preact@10.29.0)(rollup@4.60.1)(vite@5.4.21(lightningcss@1.32.0))':
dependencies: dependencies:
'@babel/core': 7.29.0 '@babel/core': 7.29.0
'@babel/plugin-transform-react-jsx': 7.28.6(@babel/core@7.29.0) '@babel/plugin-transform-react-jsx': 7.28.6(@babel/core@7.29.0)
'@babel/plugin-transform-react-jsx-development': 7.27.1(@babel/core@7.29.0) '@babel/plugin-transform-react-jsx-development': 7.27.1(@babel/core@7.29.0)
'@prefresh/vite': 2.4.12(preact@10.29.0)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) '@prefresh/vite': 2.4.12(preact@10.29.0)(vite@5.4.21(lightningcss@1.32.0))
'@rollup/pluginutils': 5.3.0(rollup@4.60.1) '@rollup/pluginutils': 5.3.0(rollup@4.60.1)
babel-plugin-transform-hook-names: 1.0.2(@babel/core@7.29.0) babel-plugin-transform-hook-names: 1.0.2(@babel/core@7.29.0)
debug: 4.4.3 debug: 4.4.3
magic-string: 0.30.21 magic-string: 0.30.21
picocolors: 1.1.1 picocolors: 1.1.1
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
vite-prerender-plugin: 0.5.13(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) vite-prerender-plugin: 0.5.13(vite@5.4.21(lightningcss@1.32.0))
zimmerframe: 1.1.4 zimmerframe: 1.1.4
transitivePeerDependencies: transitivePeerDependencies:
- preact - preact
@ -2857,7 +2848,7 @@ snapshots:
'@prefresh/utils@1.2.1': {} '@prefresh/utils@1.2.1': {}
'@prefresh/vite@2.4.12(preact@10.29.0)(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0))': '@prefresh/vite@2.4.12(preact@10.29.0)(vite@5.4.21(lightningcss@1.32.0))':
dependencies: dependencies:
'@babel/core': 7.29.0 '@babel/core': 7.29.0
'@prefresh/babel-plugin': 0.5.3 '@prefresh/babel-plugin': 0.5.3
@ -2865,7 +2856,7 @@ snapshots:
'@prefresh/utils': 1.2.1 '@prefresh/utils': 1.2.1
'@rollup/pluginutils': 4.2.1 '@rollup/pluginutils': 4.2.1
preact: 10.29.0 preact: 10.29.0
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
transitivePeerDependencies: transitivePeerDependencies:
- supports-color - supports-color
@ -3020,12 +3011,12 @@ snapshots:
'@tailwindcss/oxide-win32-arm64-msvc': 4.2.2 '@tailwindcss/oxide-win32-arm64-msvc': 4.2.2
'@tailwindcss/oxide-win32-x64-msvc': 4.2.2 '@tailwindcss/oxide-win32-x64-msvc': 4.2.2
'@tailwindcss/vite@4.2.2(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0))': '@tailwindcss/vite@4.2.2(vite@5.4.21(lightningcss@1.32.0))':
dependencies: dependencies:
'@tailwindcss/node': 4.2.2 '@tailwindcss/node': 4.2.2
'@tailwindcss/oxide': 4.2.2 '@tailwindcss/oxide': 4.2.2
tailwindcss: 4.2.2 tailwindcss: 4.2.2
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
'@types/chai@5.2.3': '@types/chai@5.2.3':
dependencies: dependencies:
@ -3040,10 +3031,6 @@ snapshots:
'@types/json5@0.0.29': {} '@types/json5@0.0.29': {}
'@types/node@25.6.0':
dependencies:
undici-types: 7.19.2
'@typescript-eslint/eslint-plugin@8.58.2(@typescript-eslint/parser@8.58.2(eslint@9.39.4(jiti@2.6.1))(typescript@5.9.3))(eslint@9.39.4(jiti@2.6.1))(typescript@5.9.3)': '@typescript-eslint/eslint-plugin@8.58.2(@typescript-eslint/parser@8.58.2(eslint@9.39.4(jiti@2.6.1))(typescript@5.9.3))(eslint@9.39.4(jiti@2.6.1))(typescript@5.9.3)':
dependencies: dependencies:
'@eslint-community/regexpp': 4.12.2 '@eslint-community/regexpp': 4.12.2
@ -3143,13 +3130,13 @@ snapshots:
chai: 5.3.3 chai: 5.3.3
tinyrainbow: 2.0.0 tinyrainbow: 2.0.0
'@vitest/mocker@3.2.4(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0))': '@vitest/mocker@3.2.4(vite@5.4.21(lightningcss@1.32.0))':
dependencies: dependencies:
'@vitest/spy': 3.2.4 '@vitest/spy': 3.2.4
estree-walker: 3.0.3 estree-walker: 3.0.3
magic-string: 0.30.21 magic-string: 0.30.21
optionalDependencies: optionalDependencies:
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
'@vitest/pretty-format@3.2.4': '@vitest/pretty-format@3.2.4':
dependencies: dependencies:
@ -4580,8 +4567,6 @@ snapshots:
has-symbols: 1.1.0 has-symbols: 1.1.0
which-boxed-primitive: 1.1.1 which-boxed-primitive: 1.1.1
undici-types@7.19.2: {}
update-browserslist-db@1.2.3(browserslist@4.28.2): update-browserslist-db@1.2.3(browserslist@4.28.2):
dependencies: dependencies:
browserslist: 4.28.2 browserslist: 4.28.2
@ -4592,13 +4577,13 @@ snapshots:
dependencies: dependencies:
punycode: 2.3.1 punycode: 2.3.1
vite-node@3.2.4(@types/node@25.6.0)(lightningcss@1.32.0): vite-node@3.2.4(lightningcss@1.32.0):
dependencies: dependencies:
cac: 6.7.14 cac: 6.7.14
debug: 4.4.3 debug: 4.4.3
es-module-lexer: 1.7.0 es-module-lexer: 1.7.0
pathe: 2.0.3 pathe: 2.0.3
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
transitivePeerDependencies: transitivePeerDependencies:
- '@types/node' - '@types/node'
- less - less
@ -4610,7 +4595,7 @@ snapshots:
- supports-color - supports-color
- terser - terser
vite-prerender-plugin@0.5.13(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)): vite-prerender-plugin@0.5.13(vite@5.4.21(lightningcss@1.32.0)):
dependencies: dependencies:
kolorist: 1.8.0 kolorist: 1.8.0
magic-string: 0.30.21 magic-string: 0.30.21
@ -4618,23 +4603,22 @@ snapshots:
simple-code-frame: 1.3.0 simple-code-frame: 1.3.0
source-map: 0.7.6 source-map: 0.7.6
stack-trace: 1.0.0-pre2 stack-trace: 1.0.0-pre2
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0): vite@5.4.21(lightningcss@1.32.0):
dependencies: dependencies:
esbuild: 0.21.5 esbuild: 0.21.5
postcss: 8.5.8 postcss: 8.5.8
rollup: 4.60.1 rollup: 4.60.1
optionalDependencies: optionalDependencies:
'@types/node': 25.6.0
fsevents: 2.3.3 fsevents: 2.3.3
lightningcss: 1.32.0 lightningcss: 1.32.0
vitest@3.2.4(@types/node@25.6.0)(lightningcss@1.32.0): vitest@3.2.4(lightningcss@1.32.0):
dependencies: dependencies:
'@types/chai': 5.2.3 '@types/chai': 5.2.3
'@vitest/expect': 3.2.4 '@vitest/expect': 3.2.4
'@vitest/mocker': 3.2.4(vite@5.4.21(@types/node@25.6.0)(lightningcss@1.32.0)) '@vitest/mocker': 3.2.4(vite@5.4.21(lightningcss@1.32.0))
'@vitest/pretty-format': 3.2.4 '@vitest/pretty-format': 3.2.4
'@vitest/runner': 3.2.4 '@vitest/runner': 3.2.4
'@vitest/snapshot': 3.2.4 '@vitest/snapshot': 3.2.4
@ -4652,11 +4636,9 @@ snapshots:
tinyglobby: 0.2.15 tinyglobby: 0.2.15
tinypool: 1.1.1 tinypool: 1.1.1
tinyrainbow: 2.0.0 tinyrainbow: 2.0.0
vite: 5.4.21(@types/node@25.6.0)(lightningcss@1.32.0) vite: 5.4.21(lightningcss@1.32.0)
vite-node: 3.2.4(@types/node@25.6.0)(lightningcss@1.32.0) vite-node: 3.2.4(lightningcss@1.32.0)
why-is-node-running: 2.3.0 why-is-node-running: 2.3.0
optionalDependencies:
'@types/node': 25.6.0
transitivePeerDependencies: transitivePeerDependencies:
- less - less
- lightningcss - lightningcss