mirror of
https://github.com/thib8956/tic-tac-toe-ws.git
synced 2025-01-12 04:41:06 +00:00
keep colors and refactor
This commit is contained in:
parent
2136159808
commit
a636abddbe
86
client.mts
86
client.mts
@ -12,6 +12,10 @@ interface Point {
|
|||||||
y: number;
|
y: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Empty = undefined;
|
||||||
|
|
||||||
|
type Cell = Empty | Shape;
|
||||||
|
|
||||||
interface Shape {
|
interface Shape {
|
||||||
kind: "o" | "x";
|
kind: "o" | "x";
|
||||||
pos: Point;
|
pos: Point;
|
||||||
@ -19,9 +23,8 @@ interface Shape {
|
|||||||
time: number | null;
|
time: number | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
let grid = [0, 0, 0, 0, 0, 0, 0, 0, 0];
|
let grid: Cell[] = new Array(9);
|
||||||
let pendingEvts: Point[] = [];
|
let pendingEvts: Point[] = [];
|
||||||
let pendingShapes: Shape[] = [];
|
|
||||||
let myId: number | null = null;
|
let myId: number | null = null;
|
||||||
let mySymbol: "x" | "o" | null = null;
|
let mySymbol: "x" | "o" | null = null;
|
||||||
let canvasMsg: string = "Offline";
|
let canvasMsg: string = "Offline";
|
||||||
@ -72,47 +75,6 @@ function handlePendingEvts(ws: WebSocket, gridOrigin: Point) {
|
|||||||
pendingEvts = [];
|
pendingEvts = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function handlePendingShapes(ctx: CanvasRenderingContext2D, gridOrigin: Point, time: number) {
|
|
||||||
const shapes = [];
|
|
||||||
for (let shape of pendingShapes) {
|
|
||||||
if (shape.time === null) {
|
|
||||||
shape.time = time;
|
|
||||||
}
|
|
||||||
const dt = time - shape.time;
|
|
||||||
const p = gridIndexToCoords(gridOrigin, shape.pos.x, shape.pos.y);
|
|
||||||
switch (shape.kind) {
|
|
||||||
case "o":
|
|
||||||
drawAnimatedCircle(ctx, dt, p.x, p.y, shape.hue);
|
|
||||||
break;
|
|
||||||
case "x":
|
|
||||||
drawAnimatedCross(ctx, dt, p.x, p.y, shape.hue);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (dt <= ANIMATE_DURATION) {
|
|
||||||
shapes.push(shape);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pendingShapes = shapes;
|
|
||||||
}
|
|
||||||
|
|
||||||
function drawCircle(ctx: CanvasRenderingContext2D, center: Point) {
|
|
||||||
const radius = SHAPE_SIZE/2;
|
|
||||||
ctx.beginPath();
|
|
||||||
ctx.arc(center.x, center.y, radius, 0, Math.PI * 2);
|
|
||||||
ctx.stroke();
|
|
||||||
}
|
|
||||||
|
|
||||||
function drawCross(ctx: CanvasRenderingContext2D, center: Point) {
|
|
||||||
const startPoint = { x: center.x-SHAPE_SIZE/2, y: center.y-SHAPE_SIZE/2 };
|
|
||||||
ctx.beginPath();
|
|
||||||
ctx.moveTo(startPoint.x, startPoint.y);
|
|
||||||
ctx.lineTo(startPoint.x + 100, startPoint.y + 100);
|
|
||||||
ctx.moveTo(startPoint.x + 100, startPoint.y);
|
|
||||||
ctx.lineTo(startPoint.x, startPoint.y + 100);
|
|
||||||
ctx.stroke();
|
|
||||||
}
|
|
||||||
|
|
||||||
function drawAnimatedCircle(ctx: CanvasRenderingContext2D, dt: number, x: number, y: number, hue: number) {
|
function drawAnimatedCircle(ctx: CanvasRenderingContext2D, dt: number, x: number, y: number, hue: number) {
|
||||||
const radius = SHAPE_SIZE / 2;
|
const radius = SHAPE_SIZE / 2;
|
||||||
const end = dt*2*Math.PI/ANIMATE_DURATION;
|
const end = dt*2*Math.PI/ANIMATE_DURATION;
|
||||||
@ -160,26 +122,29 @@ function gridIndexToCoords(gridOrigin: Point, x: number, y: number): Point {
|
|||||||
return center;
|
return center;
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateGridState(ctx: CanvasRenderingContext2D, gridOrigin: Point) {
|
function updateGridState(ctx: CanvasRenderingContext2D, time: number, gridOrigin: Point) {
|
||||||
for (let y = 0; y < 3; ++y) {
|
for (let y = 0; y < 3; ++y) {
|
||||||
for (let x = 0; x < 3; ++x) {
|
for (let x = 0; x < 3; ++x) {
|
||||||
if (pendingShapes.some(s => s.pos.x === x && s.pos.y === y)) {
|
const shape = grid[y*3+x]
|
||||||
continue;
|
if (shape) {
|
||||||
|
if (shape.time === null) {
|
||||||
|
shape.time = time;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (grid[y*3+x]) {
|
const p = gridIndexToCoords(gridOrigin, shape.pos.x, shape.pos.y);
|
||||||
case 0: break;
|
const dt = time - shape.time;
|
||||||
case myId: {
|
|
||||||
const p = gridIndexToCoords(gridOrigin, x, y);
|
|
||||||
mySymbol == "o" ? drawCircle(ctx, p) : drawCross(ctx, p);
|
|
||||||
break;
|
|
||||||
|
|
||||||
}
|
switch (shape.kind) {
|
||||||
default: {
|
case "o": {
|
||||||
const p = gridIndexToCoords(gridOrigin, x, y);
|
drawAnimatedCircle(ctx, dt, p.x, p.y, shape.hue);
|
||||||
mySymbol == "o" ? drawCross(ctx, p) : drawCircle(ctx, p);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case "x": {
|
||||||
|
drawAnimatedCross(ctx, dt, p.x, p.y, shape.hue);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -199,8 +164,7 @@ function update(ctx: CanvasRenderingContext2D, time: number, ws: WebSocket) {
|
|||||||
|
|
||||||
drawGridBackground(ctx, gridOrigin);
|
drawGridBackground(ctx, gridOrigin);
|
||||||
handlePendingEvts(ws, gridOrigin);
|
handlePendingEvts(ws, gridOrigin);
|
||||||
handlePendingShapes(ctx, gridOrigin, time);
|
updateGridState(ctx, time, gridOrigin);
|
||||||
updateGridState(ctx, gridOrigin);
|
|
||||||
|
|
||||||
window.requestAnimationFrame(t => update(ctx, t, ws));
|
window.requestAnimationFrame(t => update(ctx, t, ws));
|
||||||
}
|
}
|
||||||
@ -236,14 +200,14 @@ function init() {
|
|||||||
}
|
}
|
||||||
case "update": {
|
case "update": {
|
||||||
const res = msg.data as Response;
|
const res = msg.data as Response;
|
||||||
|
const { x, y } = res.last;
|
||||||
const shape: Shape = {
|
const shape: Shape = {
|
||||||
kind: res.last.symbol,
|
kind: res.last.symbol,
|
||||||
pos: { x: res.last.x, y: res.last.y },
|
pos: { x, y },
|
||||||
hue: Math.floor(Math.random() * 255),
|
hue: Math.floor(Math.random() * 255),
|
||||||
time: null,
|
time: null,
|
||||||
};
|
};
|
||||||
grid = res.grid;
|
grid[y*3+x] = shape;
|
||||||
pendingShapes.push(shape);
|
|
||||||
console.log(grid);
|
console.log(grid);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user