2024-01-29 10:46:23 +09:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: syuilo and other misskey contributors
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
import CRC32 from 'crc-32';
|
2024-01-29 17:15:09 +09:00
|
|
|
import { Tile, House, Huro, TILE_TYPES, YAKU_DEFINITIONS } from './common.js';
|
2024-01-29 10:46:23 +09:00
|
|
|
import * as Utils from './utils.js';
|
|
|
|
|
|
|
|
export type PlayerState = {
|
|
|
|
user1House: House;
|
|
|
|
user2House: House;
|
|
|
|
user3House: House;
|
|
|
|
user4House: House;
|
2024-01-29 14:14:00 +09:00
|
|
|
|
|
|
|
round: 'e' | 's' | 'w' | 'n';
|
|
|
|
kyoku: number;
|
|
|
|
|
2024-01-29 10:46:23 +09:00
|
|
|
tilesCount: number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 副露した牌を含まない手牌
|
|
|
|
*/
|
|
|
|
handTiles: {
|
|
|
|
e: Tile[] | null[];
|
|
|
|
s: Tile[] | null[];
|
|
|
|
w: Tile[] | null[];
|
|
|
|
n: Tile[] | null[];
|
|
|
|
};
|
|
|
|
|
|
|
|
hoTiles: {
|
|
|
|
e: Tile[];
|
|
|
|
s: Tile[];
|
|
|
|
w: Tile[];
|
|
|
|
n: Tile[];
|
|
|
|
};
|
|
|
|
huros: {
|
|
|
|
e: Huro[];
|
|
|
|
s: Huro[];
|
|
|
|
w: Huro[];
|
|
|
|
n: Huro[];
|
|
|
|
};
|
|
|
|
riichis: {
|
|
|
|
e: boolean;
|
|
|
|
s: boolean;
|
|
|
|
w: boolean;
|
|
|
|
n: boolean;
|
|
|
|
};
|
|
|
|
points: {
|
|
|
|
e: number;
|
|
|
|
s: number;
|
|
|
|
w: number;
|
|
|
|
n: number;
|
|
|
|
};
|
|
|
|
latestDahaiedTile: Tile | null;
|
|
|
|
turn: House | null;
|
|
|
|
canPonSource: House | null;
|
|
|
|
canCiiSource: House | null;
|
|
|
|
canKanSource: House | null;
|
|
|
|
canRonSource: House | null;
|
|
|
|
canCiiTo: House | null;
|
|
|
|
canKanTo: House | null;
|
|
|
|
canRonTo: House | null;
|
|
|
|
};
|
|
|
|
|
|
|
|
export class PlayerGameEngine {
|
|
|
|
/**
|
|
|
|
* このエラーが発生したときはdesyncが疑われる
|
|
|
|
*/
|
|
|
|
public static InvalidOperationError = class extends Error {};
|
|
|
|
|
|
|
|
private myUserNumber: 1 | 2 | 3 | 4;
|
|
|
|
public state: PlayerState;
|
|
|
|
|
|
|
|
constructor(myUserNumber: PlayerGameEngine['myUserNumber'], state: PlayerState) {
|
|
|
|
this.myUserNumber = myUserNumber;
|
|
|
|
this.state = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
public get myHouse(): House {
|
|
|
|
switch (this.myUserNumber) {
|
|
|
|
case 1: return this.state.user1House;
|
|
|
|
case 2: return this.state.user2House;
|
|
|
|
case 3: return this.state.user3House;
|
|
|
|
case 4: return this.state.user4House;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public get myHandTiles(): Tile[] {
|
|
|
|
return this.state.handTiles[this.myHouse] as Tile[];
|
|
|
|
}
|
|
|
|
|
|
|
|
public get isMeRiichi(): boolean {
|
|
|
|
return this.state.riichis[this.myHouse];
|
|
|
|
}
|
|
|
|
|
|
|
|
public commit_tsumo(house: House, tile: Tile) {
|
|
|
|
console.log('commit_tsumo', this.state.turn, house, tile);
|
2024-01-29 17:15:09 +09:00
|
|
|
this.state.tilesCount--;
|
2024-01-29 10:46:23 +09:00
|
|
|
this.state.turn = house;
|
|
|
|
if (house === this.myHouse) {
|
|
|
|
this.myHandTiles.push(tile);
|
|
|
|
} else {
|
|
|
|
this.state.handTiles[house].push(null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public commit_dahai(house: House, tile: Tile, riichi = false) {
|
|
|
|
console.log('commit_dahai', this.state.turn, house, tile, riichi);
|
|
|
|
if (this.state.turn !== house) throw new PlayerGameEngine.InvalidOperationError();
|
|
|
|
|
|
|
|
if (riichi) {
|
|
|
|
this.state.riichis[house] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (house === this.myHouse) {
|
|
|
|
this.myHandTiles.splice(this.myHandTiles.indexOf(tile), 1);
|
|
|
|
this.state.hoTiles[this.myHouse].push(tile);
|
|
|
|
} else {
|
|
|
|
this.state.handTiles[house].pop();
|
|
|
|
this.state.hoTiles[house].push(tile);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.state.turn = null;
|
|
|
|
|
|
|
|
if (house === this.myHouse) {
|
|
|
|
} else {
|
|
|
|
const canRon = Utils.getHoraSets(this.myHandTiles.concat(tile)).length > 0;
|
|
|
|
const canPon = this.myHandTiles.filter(t => t === tile).length === 2;
|
|
|
|
|
|
|
|
// TODO: canCii
|
|
|
|
|
|
|
|
if (canRon) this.state.canRonSource = house;
|
|
|
|
if (canPon) this.state.canPonSource = house;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-29 14:14:00 +09:00
|
|
|
public commit_kakan(house: House, tile: Tile) {
|
|
|
|
console.log('commit_kakan', this.state.turn, house, tile);
|
|
|
|
if (this.state.turn !== house) throw new PlayerGameEngine.InvalidOperationError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public commit_hora(house: House) {
|
|
|
|
console.log('commit_hora', this.state.turn, house);
|
2024-01-29 17:15:09 +09:00
|
|
|
|
|
|
|
// TODO: ツモした人の手牌情報を貰う必要がある
|
2024-01-29 14:14:00 +09:00
|
|
|
}
|
|
|
|
|
2024-01-29 10:46:23 +09:00
|
|
|
/**
|
|
|
|
* ロンします
|
2024-01-29 17:15:09 +09:00
|
|
|
* @param callers ロンした人
|
|
|
|
* @param callee 牌を捨てた人
|
2024-01-29 10:46:23 +09:00
|
|
|
*/
|
2024-01-29 17:15:09 +09:00
|
|
|
public commit_ron(callers: House[], callee: House) {
|
|
|
|
console.log('commit_ron', this.state.turn, callers, callee);
|
|
|
|
|
2024-01-29 10:46:23 +09:00
|
|
|
this.state.canRonSource = null;
|
|
|
|
|
2024-01-29 17:15:09 +09:00
|
|
|
// TODO: ロンした人の手牌情報を貰う必要がある
|
|
|
|
|
|
|
|
for (const house of callers) {
|
|
|
|
const yakus = YAKU_DEFINITIONS.filter(yaku => yaku.calc({
|
|
|
|
house: house,
|
|
|
|
handTiles: this.state.handTiles[house],
|
|
|
|
huros: this.state.huros[house],
|
|
|
|
tsumoTile: null,
|
|
|
|
ronTile: this.state.hoTiles[callee].at(-1)!,
|
|
|
|
riichi: this.state.riichis[house],
|
|
|
|
}));
|
|
|
|
console.log('yakus', yakus);
|
2024-01-29 10:46:23 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ポンします
|
2024-01-29 17:15:09 +09:00
|
|
|
* @param caller ポンした人
|
|
|
|
* @param callee 牌を捨てた人
|
2024-01-29 10:46:23 +09:00
|
|
|
*/
|
2024-01-29 17:15:09 +09:00
|
|
|
public commit_pon(caller: House, callee: House) {
|
2024-01-29 10:46:23 +09:00
|
|
|
this.state.canPonSource = null;
|
|
|
|
|
2024-01-29 17:15:09 +09:00
|
|
|
const lastTile = this.state.hoTiles[callee].pop();
|
2024-01-29 10:46:23 +09:00
|
|
|
if (lastTile == null) throw new PlayerGameEngine.InvalidOperationError();
|
2024-01-29 17:15:09 +09:00
|
|
|
if (caller === this.myHouse) {
|
2024-01-29 10:46:23 +09:00
|
|
|
this.myHandTiles.splice(this.myHandTiles.indexOf(lastTile), 1);
|
|
|
|
this.myHandTiles.splice(this.myHandTiles.indexOf(lastTile), 1);
|
|
|
|
} else {
|
2024-01-29 17:15:09 +09:00
|
|
|
this.state.handTiles[caller].unshift();
|
|
|
|
this.state.handTiles[caller].unshift();
|
2024-01-29 10:46:23 +09:00
|
|
|
}
|
2024-01-29 17:15:09 +09:00
|
|
|
this.state.huros[caller].push({ type: 'pon', tile: lastTile, from: callee });
|
2024-01-29 10:46:23 +09:00
|
|
|
|
2024-01-29 17:15:09 +09:00
|
|
|
this.state.turn = caller;
|
2024-01-29 10:46:23 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
public commit_nop() {
|
|
|
|
this.state.canRonSource = null;
|
|
|
|
this.state.canPonSource = null;
|
|
|
|
}
|
2024-01-29 17:15:09 +09:00
|
|
|
|
|
|
|
public get isMenzen(): boolean {
|
|
|
|
const calls = ['pon', 'cii', 'minkan'];
|
|
|
|
return this.state.huros[this.myHouse].filter(h => calls.includes(h.type)).length === 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public canRiichi(): boolean {
|
|
|
|
if (this.state.turn !== this.myHouse) return false;
|
|
|
|
if (this.state.riichis[this.myHouse]) return false;
|
|
|
|
if (this.state.points[this.myHouse] < 1000) return false;
|
|
|
|
if (!this.isMenzen) return false;
|
|
|
|
if (Utils.getTilesForRiichi(this.myHandTiles).length === 0) return false;
|
|
|
|
return true;
|
|
|
|
}
|
2024-01-29 10:46:23 +09:00
|
|
|
}
|