fix(settings/user/snake): personnalMessage -> snake

This commit is contained in:
2026-01-31 00:37:07 +01:00
parent 10f7d16805
commit 742e6363a4
9 changed files with 17 additions and 22 deletions

View File

@@ -0,0 +1,257 @@
<script setup lang="ts">
import * as THREE from "three";
import { useIntervalFn, useLocalStorage } from "@vueuse/core";
const store = useSettingsStore();
const achievements = useAchievementsStore();
const confirmationModal = useConfirmationModal();
const { onRender } = useScreen();
const { assets: atlas } = useAssets();
const assets = atlas.images.settings.bottomScreen;
const highScore = useLocalStorage("snake_high_score", 0);
const handleCancel = () => {
switch (state.value) {
case "alive": {
state.value = "pause";
confirmationModal.open({
text: $t("settings.user.snake.quitConfirmation"),
onConfirm: () => {},
onClosed: (choice) => {
if (choice === "confirm") store.closeSubMenu();
},
onCancel: () => {
state.value = "alive";
},
});
break;
}
case "dead":
case "waiting": {
store.closeSubMenu();
break;
}
}
};
const handleConfirm = () => {
switch (state.value) {
case "alive": {
state.value = "pause";
confirmationModal.open({
text: $t("settings.user.snake.restartConfirmation"),
onConfirm: () => {
spawn();
},
onCancel: () => {
state.value = "alive";
},
});
break;
}
case "waiting": {
achievements.unlock("snake_play");
spawn();
break;
}
case "dead": {
spawn();
break;
}
}
};
const BOARD_X = 15;
const BOARD_Y = 48;
const BOARD_WIDTH = 13;
const BOARD_HEIGHT = 7;
const CELL_SIZE = 15;
const CELL_PADDING = 1;
const position = new THREE.Vector2();
const direction = new THREE.Vector2();
const nextDirection = new THREE.Vector2();
const tail: THREE.Vector2[] = [];
const food = new THREE.Vector2();
let score = 0;
const state = ref<"waiting" | "alive" | "pause" | "dead">("waiting");
const randomFoodPos = (): THREE.Vector2 => {
// can't spawn on the head or tail
const occupiedPositions = [position, ...tail];
const emptyPositions: THREE.Vector2[] = [];
for (let x = 0; x < BOARD_WIDTH; x++) {
for (let y = 0; y < BOARD_HEIGHT; y++) {
const candidate = new THREE.Vector2(x, y);
if (!occupiedPositions.find((part) => part.equals(candidate))) {
emptyPositions.push(candidate);
}
}
}
return emptyPositions[THREE.MathUtils.randInt(0, emptyPositions.length - 1)]!;
};
const eat = () => {
highScore.value = Math.max(highScore.value, score);
food.copy(randomFoodPos());
score += 1;
if (score === 40) {
achievements.unlock("snake_score_40");
}
};
const die = () => {
state.value = "dead";
};
const spawn = () => {
state.value = "alive";
score = 0;
tail.length = 0;
position.set(0, 0);
direction.set(1, 0);
nextDirection.set(1, 0);
food.copy(randomFoodPos());
};
useIntervalFn(() => {
if (state.value !== "alive") return;
direction.copy(nextDirection);
const previousPosition = position.clone();
position.set(
(position.x + direction.x + BOARD_WIDTH) % BOARD_WIDTH,
(position.y + direction.y + BOARD_HEIGHT) % BOARD_HEIGHT,
);
if (!position.equals(previousPosition)) {
if (tail.find((part) => part.equals(position))) {
die();
}
tail.push(previousPosition);
if (food.equals(position)) {
eat();
} else {
tail.shift();
}
}
}, 200);
const cellToBoardPos = (pos: THREE.Vector2) => ({
x: BOARD_X + CELL_SIZE + 1 + pos.x * (CELL_SIZE + CELL_PADDING),
y: BOARD_Y + pos.y * (CELL_SIZE + CELL_PADDING),
});
onRender((ctx) => {
assets.background.draw(ctx, 0, 0);
// score
assets.user.snakeScore.draw(ctx, 27, 32);
ctx.textBaseline = "top";
ctx.font = "10px NDS10";
ctx.fillStyle = "#282828";
fillTextHCentered(
ctx,
$t("settings.user.snake.score", { score }),
27,
36,
108,
);
fillTextHCentered(
ctx,
$t("settings.user.snake.best", { best: highScore.value }),
135,
36,
108,
);
// board
assets.user.snakeBoard.draw(ctx, 15, 48);
if (state.value === "waiting") {
ctx.fillStyle = "#fbfbfb";
const text = `\n\n\n ${$t("settings.user.snake.startPrompt")}`;
let x = 15;
let y = 52;
for (let i = 0; i < text.length; i += 1, x += 16) {
while (text[i] === "\n") {
x = 15;
y += 16;
i += 1;
}
ctx.fillText(text[i]!, x + 20, y);
}
} else {
// food
ctx.fillStyle = "#ff2020";
ctx.fillRect(
cellToBoardPos(food).x + 1,
cellToBoardPos(food).y + 1,
CELL_SIZE,
CELL_SIZE,
);
// snake
for (const part of tail) {
ctx.fillStyle = state.value === "dead" ? "#991010" : "#20ff20";
ctx.fillRect(
cellToBoardPos(part).x + 1,
cellToBoardPos(part).y + 1,
CELL_SIZE,
CELL_SIZE,
);
}
assets.user.snakeHead.draw(
ctx,
cellToBoardPos(position).x + 1,
cellToBoardPos(position).y,
);
}
});
useKeyDown((key) => {
if (state.value !== "alive") return;
const newDirection = direction.clone();
switch (key) {
case "NDS_UP":
newDirection.set(0, -1);
break;
case "NDS_RIGHT":
newDirection.set(1, 0);
break;
case "NDS_DOWN":
newDirection.set(0, 1);
break;
case "NDS_LEFT":
newDirection.set(-1, 0);
break;
}
if (newDirection.clone().dot(direction) === 0) {
nextDirection.copy(newDirection);
}
});
defineOptions({ render: () => null });
</script>
<template>
<CommonButtons
:y-offset="confirmationModal.buttonsYOffset"
:b-label="$t('common.quit')"
:a-label="state === 'waiting' ? $t('common.start') : $t('common.restart')"
@activate-b="handleCancel"
@activate-a="handleConfirm"
/>
</template>