Commit 91dcb148 authored by nanahira's avatar nanahira

add menu

parent 4af43c04
...@@ -63,6 +63,17 @@ randomDuelDisableChat: 0 ...@@ -63,6 +63,17 @@ randomDuelDisableChat: 0
randomDuelReadyTime: 20 randomDuelReadyTime: 20
randomDuelHangTimeout: 90 randomDuelHangTimeout: 90
sideTimeoutMinutes: 3 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 hostinfoLflist: 0
hostinfoRule: 0 hostinfoRule: 0
hostinfoMode: 0 hostinfoMode: 0
......
...@@ -17,7 +17,7 @@ ...@@ -17,7 +17,7 @@
"https-proxy-agent": "^7.0.6", "https-proxy-agent": "^7.0.6",
"ipaddr.js": "^2.3.0", "ipaddr.js": "^2.3.0",
"koishipro-core.js": "^1.3.4", "koishipro-core.js": "^1.3.4",
"nfkit": "^1.0.32", "nfkit": "^1.0.33",
"p-queue": "6.6.2", "p-queue": "6.6.2",
"pg": "^8.18.0", "pg": "^8.18.0",
"pino": "^10.3.1", "pino": "^10.3.1",
...@@ -5330,9 +5330,9 @@ ...@@ -5330,9 +5330,9 @@
"license": "MIT" "license": "MIT"
}, },
"node_modules/nfkit": { "node_modules/nfkit": {
"version": "1.0.32", "version": "1.0.33",
"resolved": "https://registry.npmjs.org/nfkit/-/nfkit-1.0.32.tgz", "resolved": "https://registry.npmjs.org/nfkit/-/nfkit-1.0.33.tgz",
"integrity": "sha512-y+UoxDBs6JV4CSBZkidBGK4GfzJ1Qev8uU4m4oClWGs09oxOCh6TQqnOGRaZY1yCmD8yzYcED+8waSMU4WS5fg==", "integrity": "sha512-mhF4ZAoGUD3cI0sB/+qH2AothZG2j5y18FkyTKF6etR6nod8jBJWQ5hAr3Q6HnaWlG3HpUKN5i1wfZqQP6hyZw==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/node-int64": { "node_modules/node-int64": {
......
...@@ -52,6 +52,14 @@ export class ClientHandler { ...@@ -52,6 +52,14 @@ export class ClientHandler {
.middleware( .middleware(
YGOProCtosBase, YGOProCtosBase,
async (msg, client, next) => { async (msg, client, next) => {
const bypassEstablished =
msg instanceof YGOProCtosJoinGame &&
msg.bypassEstablished;
if (bypassEstablished) {
delete msg.bypassEstablished;
return next();
}
const isPreHandshakeMsg = [ const isPreHandshakeMsg = [
YGOProCtosExternalAddress, YGOProCtosExternalAddress,
YGOProCtosPlayerInfo, YGOProCtosPlayerInfo,
...@@ -125,3 +133,9 @@ export class ClientHandler { ...@@ -125,3 +133,9 @@ export class ClientHandler {
}); });
} }
} }
declare module 'ygopro-msg-encode' {
interface YGOProCtosJoinGame {
bypassEstablished?: boolean;
}
}
export * from './client'; export * from './client';
export * from './client-handler'; export * from './client-handler';
export * from './chnroute'; export * from './chnroute';
export * from './i18n';
...@@ -152,6 +152,15 @@ export const defaultConfig = { ...@@ -152,6 +152,15 @@ export const defaultConfig = {
// Room hostinfo defaults expanded into HOSTINFO_* keys. // Room hostinfo defaults expanded into HOSTINFO_* keys.
// Format: each HOSTINFO_* value is a string; numeric fields use integer strings. // Format: each HOSTINFO_* value is a string; numeric fields use integer strings.
// Unit note: HOSTINFO_TIME_LIMIT is in seconds (s). // 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.fromEntries(
Object.entries(DefaultHostinfo).map(([key, value]) => [ Object.entries(DefaultHostinfo).map(([key, value]) => [
`HOSTINFO_${key.toUpperCase()}`, `HOSTINFO_${key.toUpperCase()}`,
......
...@@ -82,6 +82,19 @@ export const TRANSLATIONS = { ...@@ -82,6 +82,19 @@ export const TRANSLATIONS = {
chat_disabled: 'Chat is disabled in this room.', chat_disabled: 'Chat is disabled in this room.',
chat_warn_level1: 'Please avoid sensitive words.', chat_warn_level1: 'Please avoid sensitive words.',
chat_warn_level2: 'Your message contains blocked 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': { 'zh-CN': {
update_required: '请更新你的客户端版本', update_required: '请更新你的客户端版本',
...@@ -157,5 +170,18 @@ export const TRANSLATIONS = { ...@@ -157,5 +170,18 @@ export const TRANSLATIONS = {
chat_disabled: '本房间禁止聊天。', chat_disabled: '本房间禁止聊天。',
chat_warn_level1: '请注意发言,敏感词已被替换。', chat_warn_level1: '请注意发言,敏感词已被替换。',
chat_warn_level2: '消息包含敏感词,已被拦截。', 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 './client-version-check';
export * from './welcome';
export * from './random-duel'; export * from './random-duel';
export * from './reconnect'; export * from './reconnect';
export * from './wait-for-player-provider'; export * from './wait-for-player-provider';
......
import { ChatColor } from 'ygopro-msg-encode'; import { ChatColor } from 'ygopro-msg-encode';
import { Context } from '../app'; import { Context } from '../app';
import { Client } from '../client';
import { OnRoomJoin } from '../room/room-event/on-room-join'; import { OnRoomJoin } from '../room/room-event/on-room-join';
declare module '../room' { declare module '../room' {
...@@ -9,15 +10,19 @@ declare module '../room' { ...@@ -9,15 +10,19 @@ declare module '../room' {
} }
} }
declare module '../client' {
interface Client {
configWelcomeSent?: boolean;
}
}
export class Welcome { export class Welcome {
private welcomeMessage = this.ctx.config.getString('WELCOME'); private welcomeMessage = this.ctx.config.getString('WELCOME');
constructor(private ctx: Context) { constructor(private ctx: Context) {
this.ctx.middleware(OnRoomJoin, async (event, client, next) => { this.ctx.middleware(OnRoomJoin, async (event, client, next) => {
const room = event.room; const room = event.room;
if (this.welcomeMessage) { await this.sendConfigWelcome(client);
await client.sendChat(this.welcomeMessage, ChatColor.GREEN);
}
if (room.welcome) { if (room.welcome) {
await client.sendChat(room.welcome, ChatColor.BABYBLUE); await client.sendChat(room.welcome, ChatColor.BABYBLUE);
} }
...@@ -27,4 +32,12 @@ export class Welcome { ...@@ -27,4 +32,12 @@ export class Welcome {
return next(); 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'; ...@@ -9,6 +9,7 @@ import { RandomDuelJoinHandler } from './random-duel-join-handler';
import { BadwordPlayerInfoChecker } from './badword-player-info-checker'; import { BadwordPlayerInfoChecker } from './badword-player-info-checker';
import { JoinBlankPassRandomDuel } from './join-blank-pass-random-duel'; import { JoinBlankPassRandomDuel } from './join-blank-pass-random-duel';
import { JoinBlankPassWindbotAi } from './join-blank-pass-windbot-ai'; import { JoinBlankPassWindbotAi } from './join-blank-pass-windbot-ai';
import { JoinBlankPassMenu } from './join-blank-pass-menu';
export const JoinHandlerModule = createAppContext<ContextState>() export const JoinHandlerModule = createAppContext<ContextState>()
.provide(ClientVersionCheck) .provide(ClientVersionCheck)
...@@ -18,6 +19,7 @@ export const JoinHandlerModule = createAppContext<ContextState>() ...@@ -18,6 +19,7 @@ export const JoinHandlerModule = createAppContext<ContextState>()
.provide(RandomDuelJoinHandler) .provide(RandomDuelJoinHandler)
.provide(JoinWindbotAi) .provide(JoinWindbotAi)
.provide(JoinRoom) .provide(JoinRoom)
.provide(JoinBlankPassMenu)
.provide(JoinBlankPassRandomDuel) .provide(JoinBlankPassRandomDuel)
.provide(JoinBlankPassWindbotAi) .provide(JoinBlankPassWindbotAi)
.provide(JoinFallback) .provide(JoinFallback)
......
...@@ -17,3 +17,4 @@ export * from './room-event/on-room-siding-ready'; ...@@ -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-siding-start';
export * from './room-event/on-room-win'; export * from './room-event/on-room-win';
export * from './default-hostinfo-provder'; 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