Commit 91dcb148 authored by nanahira's avatar nanahira

add menu

parent 4af43c04
......@@ -63,6 +63,17 @@ randomDuelDisableChat: 0
randomDuelReadyTime: 20
randomDuelHangTimeout: 90
sideTimeoutMinutes: 3
enableMenu: 0
menu:
"#{menu_random_duel}": ""
"#{menu_random_duel_match}": M
"#{menu_ai_duel}": AI
"#{menu_more}":
"#{menu_random_duel_single}": S
"#{menu_random_duel_tag}": T
"#{menu_ai_duel_match}": AI,M
"#{menu_ai_duel_tag}": AI,T
"#{menu_return}": {}
hostinfoLflist: 0
hostinfoRule: 0
hostinfoMode: 0
......
......@@ -17,7 +17,7 @@
"https-proxy-agent": "^7.0.6",
"ipaddr.js": "^2.3.0",
"koishipro-core.js": "^1.3.4",
"nfkit": "^1.0.32",
"nfkit": "^1.0.33",
"p-queue": "6.6.2",
"pg": "^8.18.0",
"pino": "^10.3.1",
......@@ -5330,9 +5330,9 @@
"license": "MIT"
},
"node_modules/nfkit": {
"version": "1.0.32",
"resolved": "https://registry.npmjs.org/nfkit/-/nfkit-1.0.32.tgz",
"integrity": "sha512-y+UoxDBs6JV4CSBZkidBGK4GfzJ1Qev8uU4m4oClWGs09oxOCh6TQqnOGRaZY1yCmD8yzYcED+8waSMU4WS5fg==",
"version": "1.0.33",
"resolved": "https://registry.npmjs.org/nfkit/-/nfkit-1.0.33.tgz",
"integrity": "sha512-mhF4ZAoGUD3cI0sB/+qH2AothZG2j5y18FkyTKF6etR6nod8jBJWQ5hAr3Q6HnaWlG3HpUKN5i1wfZqQP6hyZw==",
"license": "MIT"
},
"node_modules/node-int64": {
......
......@@ -52,6 +52,14 @@ export class ClientHandler {
.middleware(
YGOProCtosBase,
async (msg, client, next) => {
const bypassEstablished =
msg instanceof YGOProCtosJoinGame &&
msg.bypassEstablished;
if (bypassEstablished) {
delete msg.bypassEstablished;
return next();
}
const isPreHandshakeMsg = [
YGOProCtosExternalAddress,
YGOProCtosPlayerInfo,
......@@ -125,3 +133,9 @@ export class ClientHandler {
});
}
}
declare module 'ygopro-msg-encode' {
interface YGOProCtosJoinGame {
bypassEstablished?: boolean;
}
}
export * from './client';
export * from './client-handler';
export * from './chnroute';
export * from './i18n';
......@@ -152,6 +152,15 @@ export const defaultConfig = {
// Room hostinfo defaults expanded into HOSTINFO_* keys.
// Format: each HOSTINFO_* value is a string; numeric fields use integer strings.
// Unit note: HOSTINFO_TIME_LIMIT is in seconds (s).
// Enable blank-pass panel menu.
// Boolean parse rule (default false): ''/'0'/'false'/'null' => false, otherwise true.
ENABLE_MENU: '0',
// Blank-pass panel definition in JSON object format.
// Format: {"Display Text": "ROOM_PASS"}.
// - key: text shown to client; supports i18n placeholder like "#{menu_random_duel}".
// - value(string): equivalent room password, then redispatches CTOS_JOIN_GAME with this pass.
// - value(object): submenu; empty object {} means "return to previous level".
MENU: '{"#{menu_random_duel}":"","#{menu_random_duel_match}":"M","#{menu_ai_duel}":"AI","#{menu_more}":{"#{menu_random_duel_single}":"S","#{menu_random_duel_tag}":"T","#{menu_ai_duel_match}":"AI,M","#{menu_ai_duel_tag}":"AI,T","#{menu_return}":{}}}',
...(Object.fromEntries(
Object.entries(DefaultHostinfo).map(([key, value]) => [
`HOSTINFO_${key.toUpperCase()}`,
......
......@@ -82,6 +82,19 @@ export const TRANSLATIONS = {
chat_disabled: 'Chat is disabled in this room.',
chat_warn_level1: 'Please avoid sensitive words.',
chat_warn_level2: 'Your message contains blocked words.',
menu_random_duel: 'Random Duel',
menu_random_duel_match: 'Random Duel (Match)',
menu_ai_duel: 'AI Duel',
menu_more: 'More',
menu_random_duel_single: 'Random Duel (Single)',
menu_random_duel_tag: 'Random Duel (Tag)',
menu_ai_duel_match: 'AI Duel (Match)',
menu_ai_duel_tag: 'AI Duel (Tag)',
menu_return: 'Return',
menu_match_random_duel: 'Match Random Duel',
menu_single_random_duel: 'Single Random Duel',
menu_prev_page: 'Previous Page',
menu_next_page: 'Next Page',
},
'zh-CN': {
update_required: '请更新你的客户端版本',
......@@ -157,5 +170,18 @@ export const TRANSLATIONS = {
chat_disabled: '本房间禁止聊天。',
chat_warn_level1: '请注意发言,敏感词已被替换。',
chat_warn_level2: '消息包含敏感词,已被拦截。',
menu_random_duel: '随机对战',
menu_random_duel_match: '随机对战(比赛)',
menu_ai_duel: '人机对战',
menu_more: '更多',
menu_random_duel_single: '随机对战(单局)',
menu_random_duel_tag: '随机对战(双打)',
menu_ai_duel_match: '人机对战(比赛)',
menu_ai_duel_tag: '人机对战(双打)',
menu_return: '返回',
menu_match_random_duel: '随机对战(比赛)',
menu_single_random_duel: '随机对战(单局)',
menu_prev_page: '上一页',
menu_next_page: '下一页',
},
};
export * from './client-version-check';
export * from './welcome';
export * from './random-duel';
export * from './reconnect';
export * from './wait-for-player-provider';
......
import { ChatColor } from 'ygopro-msg-encode';
import { Context } from '../app';
import { Client } from '../client';
import { OnRoomJoin } from '../room/room-event/on-room-join';
declare module '../room' {
......@@ -9,15 +10,19 @@ declare module '../room' {
}
}
declare module '../client' {
interface Client {
configWelcomeSent?: boolean;
}
}
export class Welcome {
private welcomeMessage = this.ctx.config.getString('WELCOME');
constructor(private ctx: Context) {
this.ctx.middleware(OnRoomJoin, async (event, client, next) => {
const room = event.room;
if (this.welcomeMessage) {
await client.sendChat(this.welcomeMessage, ChatColor.GREEN);
}
await this.sendConfigWelcome(client);
if (room.welcome) {
await client.sendChat(room.welcome, ChatColor.BABYBLUE);
}
......@@ -27,4 +32,12 @@ export class Welcome {
return next();
});
}
async sendConfigWelcome(client: Client) {
if (!this.welcomeMessage || client.configWelcomeSent) {
return;
}
client.configWelcomeSent = true;
await client.sendChat(this.welcomeMessage, ChatColor.GREEN);
}
}
This diff is collapsed.
......@@ -9,6 +9,7 @@ import { RandomDuelJoinHandler } from './random-duel-join-handler';
import { BadwordPlayerInfoChecker } from './badword-player-info-checker';
import { JoinBlankPassRandomDuel } from './join-blank-pass-random-duel';
import { JoinBlankPassWindbotAi } from './join-blank-pass-windbot-ai';
import { JoinBlankPassMenu } from './join-blank-pass-menu';
export const JoinHandlerModule = createAppContext<ContextState>()
.provide(ClientVersionCheck)
......@@ -18,6 +19,7 @@ export const JoinHandlerModule = createAppContext<ContextState>()
.provide(RandomDuelJoinHandler)
.provide(JoinWindbotAi)
.provide(JoinRoom)
.provide(JoinBlankPassMenu)
.provide(JoinBlankPassRandomDuel)
.provide(JoinBlankPassWindbotAi)
.provide(JoinFallback)
......
......@@ -17,3 +17,4 @@ export * from './room-event/on-room-siding-ready';
export * from './room-event/on-room-siding-start';
export * from './room-event/on-room-win';
export * from './default-hostinfo-provder';
export * from './default-hostinfo';
export * from './panel-pagination';
export type PanelPageLayout = {
pageStarts: number[];
pageIndex: number;
pageStart: number;
isFirstPage: boolean;
isLastPage: boolean;
};
function resolvePageIndex(pageStarts: number[], requestedStart: number) {
if (!pageStarts.length) {
return 0;
}
if (requestedStart <= pageStarts[0]) {
return 0;
}
for (let i = pageStarts.length - 1; i >= 0; i--) {
if (requestedStart >= pageStarts[i]) {
return i;
}
}
return 0;
}
export function buildPanelPageStarts(totalEntries: number) {
if (totalEntries <= 4) {
return [0];
}
const pageStarts = [0];
const lastPageStart = totalEntries - 3;
let cursor = 3;
while (cursor < lastPageStart) {
pageStarts.push(cursor);
cursor += 2;
}
if (pageStarts[pageStarts.length - 1] !== lastPageStart) {
pageStarts.push(lastPageStart);
}
return pageStarts;
}
export function resolvePanelPageLayout(
totalEntries: number,
requestedStart: number,
): PanelPageLayout {
const pageStarts = buildPanelPageStarts(totalEntries);
const pageIndex = resolvePageIndex(pageStarts, requestedStart);
const pageStart = pageStarts[pageIndex] || 0;
return {
pageStarts,
pageIndex,
pageStart,
isFirstPage: pageIndex === 0,
isLastPage: pageIndex === pageStarts.length - 1,
};
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment