diff --git a/packages/pokemon/docs/battle-audit-report.md b/packages/pokemon/docs/battle-audit-report.md new file mode 100644 index 000000000..dd046ef7d --- /dev/null +++ b/packages/pokemon/docs/battle-audit-report.md @@ -0,0 +1,338 @@ +# Pokémon Battle 实现审查报告 + +> 审查日期:2026-04-23 +> 审查范围:`packages/pokemon/` 全部源码(battle、core、dex、ui) +> 对比基准:原版 Pokémon 核心系列游戏(Gen 9:Scarlet/Violet) +> 更新日期:2026-04-24 — 修复了 #1, #2, #3, #4, #6, #7, #8, #13 + +--- + +## 一、严重问题(核心机制错误) + +### 1. XP 计算公式与原版不符 + +**文件**: `src/battle/settlement.ts:30-31` + +```ts +const baseXp = (oppSpecies?.baseStats?.hp ?? 50) * opponentLevel / 7 +``` + +原版 Gen 9 的 XP 计算公式为: + +``` +XP = (baseXP × opponentLevel × isTraded × isParticipating) / 7 × partySizeModifier +``` + +当前实现存在以下错误: + +- **baseXP 不等于 baseStats.hp**。每只宝可梦有独立的 `base_experience` 值(例如妙蛙种子是 64,皮卡丘是 112),而不是 HP 种族值。目前用 `baseStats.hp` 做代理完全是错的。 +- **缺少 traded Pokémon 1.5x 加成**。 +- **缺少参与战斗的宝可梦分摊机制**(原版中只有实际参与战斗的宝可梦获得 XP)。 +- **缺少 Lucky Egg 1.5x 加成**。 +- **缺少 Affection 加成**(Gen 6+)。 + +### 2. EV 收益完全自造,不使用真实数据 + +**文件**: `src/battle/settlement.ts:176-191` + +```ts +function getEvYield(speciesId: string): Record { + // @pkmn/sim Dex.species doesn't have evs field + // Use baseStats as proxy: highest base stat gets 1-2 EVs + ... +} +``` + +原版中每只宝可梦有固定的 EV yield(如妙蛙种子击倒后给 HP+1,皮卡丘给 Speed+2)。这些数据在 `@pkmn/data` 中是有的(`species.evs`),但代码误以为 `@pkmn/sim` 没有这个字段,就自造了一个「最高种族值 → 2 EV,第二高 → 1 EV」的算法,与原版完全不同。 + +### 3. 物品使用在战斗中无效 + +**文件**: `src/battle/engine.ts:436-438` + +```ts +case 'item': + p1Choice = 'move 1' // fallback to move 1 + break +``` + +当玩家使用物品(如药水)时,代码直接忽略了,改为使用第一个招式。原版中物品使用是战斗的核心部分——回复药、状态治愈药、精灵球等都有完整的效果。 + +### 4. 逃跑功能未实现 + +**文件**: `src/ui/BattleFlow.tsx:314` + +```ts +case 3: // 逃跑 — show message + return +``` + +战斗菜单中「逃跑」按钮存在但点击后什么也不做。原版中有逃跑概率计算公式(基于速度对比),对野外战斗是核心机制。 + +### 5. 对手(p2)不支持多精灵队伍 + +**文件**: `src/battle/engine.ts:61-67` + +```ts +function wildPokemonToSetString(speciesId: SpeciesId, level: number): string { + ... + return [species.name, `Level: ${level}`, `Ability: ${ability}`, ...moves.map(m => `- ${m}`)].join('\n') +} +``` + +对手始终只有一只宝可梦(野生宝可梦模式)。没有 Trainer Battle 的概念——对手不能有多只精灵、不能换人、不能使用物品。虽然 AI 在精灵倒下后会自动换人(`executeSwitch` 中有处理),但 `createBattle` 本身只接受单个对手 species。 + +--- + +## 二、中等问题(机制简化/缺失) + +### 6. AI 过于简单 + +**文件**: `src/battle/ai.ts:6-13` + +```ts +export function chooseAIMove(pokemon: BattlePokemon): number { + const usable = pokemon.moves + .map((m, i) => ({ move: m, index: i })) + .filter(({ move }) => move.pp > 0 && !move.disabled) + if (usable.length === 0) return 0 + return usable[Math.floor(Math.random() * usable.length)]!.index +} +``` + +AI 只是随机选择一个可用招式。原版 NPC AI 至少会考虑: + +- **属性克制**:优先使用效果绝佳的招式 +- **状态技 vs 攻击技**的权衡 +- **HP 低时**可能使用回复招式 +- **玩家属性**:避免使用被抵抗的招式 +- 不会换人、不会使用物品 + +### 7. 野生宝可梦的招式是按属性硬编码的 + +**文件**: `src/battle/engine.ts:69-94` + +```ts +function getSpeciesMoves(speciesId: string, _level: number): string[] { + ... + const basicMoves: Record = { + normal: ['Tackle', 'Scratch'], + fire: ['Ember', 'FireSpin'], + ... + } + return basicMoves[type] ?? ['Tackle', 'Scratch'] +} +``` + +野生对手的招式不是从 learnset 中获取的,而是按第一属性硬编码了固定招式。`_level` 参数被完全忽略了——原版中不同等级的野生宝可梦应该有不同的招式组合。 + +### 8. 进化系统不完整 + +**文件**: `src/dex/evolution.ts` + `src/battle/settlement.ts:92-106` + +- 只处理了 `evoType` 为 `level_up`、`item`、`trade`、`friendship` 四种类型 +- **只取第一个进化目标** (`dex.evos[0]`),忽略了分支进化(如伊布的多种进化) +- **没有进化石使用的交互**(使用雷之石等道具触发进化) +- **没有通讯交换进化** +- **没有条件进化**(如知道特定招式、特定时间、特定地点等 Gen 9 新增条件) + +### 9. 能力值计算缺少特性/道具修正 + +**文件**: `src/core/creature.ts:51-73` + +`calculateStats` 只计算基础能力值,没有考虑: + +- **特性对能力值的修正**(如 Hustle 增加攻击降低命中) +- **道具对能力值的修正**(如 Choice Band 增加攻击 50%) + +注:性格修正虽然传入了 `nature`,但由 `@pkmn/data` 的 `gen.stats.calc` 内部处理,这部分是正确的。 + +### 10. 捕获系统完全缺失 + +没有任何捕获野生宝可梦的机制: + +- 没有 Pokeball 道具的实际效果 +- 没有捕获率计算(Shake check 公式) +- 战斗结束后不能获得对手宝可梦 +- 虽然数据中有 `captureRate` 字段和 `pokeball` 字段,但从未使用 + +### 11. 状态异常处理不完整 + +**文件**: `src/battle/engine.ts:130-140` + +只映射了 6 种基本状态(中毒、剧毒、灼伤、麻痹、冰冻、睡眠),但缺少: + +- **混乱 (Confusion)**:不在 status 中,是 volatile status +- **着迷 (Infatuation)**:同上 +- **畏缩 (Flinch)**:同上 +- 所有 volatile status(暂时性状态)都未追踪 + +### 12. 天气/场地效果未完整追踪 + +**文件**: `src/battle/engine.ts:153-173` + +- `projectState` 中天气只在初始化时从 `prevConditions` 传入,不会自动更新 +- `mapWeather` 不区分 Primal Weather(原始回归天气)和普通天气 +- **场地效果(Electric Terrain、Grassy Terrain 等)** 被映射为 `fieldCondition` 事件,但没有影响战斗状态的逻辑 + +注:底层 `@pkmn/sim` 会正确处理这些效果,只是上层状态投影不完整,导致 UI 无法正确显示。 + +--- + +## 三、轻度问题(数值/细节偏差) + +### 13. Growth Rate 数据覆盖不全 + +**文件**: `src/dex/species.ts:38-99` + +只有 9 个物种(御三家 + 皮卡丘)有正确的 `growthRate` 数据,其余全部使用默认值 `medium-slow`。实际上超过 1000 个物种各有不同的成长速率。这导致 XP 计算对大部分物种不正确。 + +### 14. 闪光概率未使用 PID 计算 + +**文件**: `src/core/creature.ts:25` + +```ts +const isShiny = Math.random() < species.shinyChance // 1/4096 +``` + +原版 Gen 9 的闪光判定基于 Personality Value(32 位 PID)的异或运算,不是简单的随机概率。Shiny Charm 等道具的加成也无法体现。 + +### 15. IV 生成算法不是真正的 LCRNG + +**文件**: `src/core/creature.ts:108-122` + +```ts +function generateIVs(seed: number): Record { + let s = seed + const nextRand = () => { + s = (s * 1103515245 + 12345) & 0x7fffffff + return s + } +``` + +原版 Gen 3+ 使用的是完全不同的随机数生成器。更重要的是,Gen 3-5 的 IV 是通过 PID 的高位/低位直接提取的,不是独立随机。 + +### 16. 性别判定阈值计算偏差 + +**文件**: `src/core/gender.ts:12-13` + +```ts +const threshold = (speciesData.genderRate / 8) * 256 +return (seed % 256) < threshold ? 'female' : 'male' +``` + +原版中性别由 PID 的低 8 位与 `genderRate` 直接比较决定,不需要乘 256 再取阈值。当前实现引入了不必要的精度损失。 + +### 17. 蛋系统与原版差异巨大 + +**文件**: `src/core/egg.ts` + +- **获得条件**:原版通过培育屋/寄养屋繁殖,当前通过「连续编码 3 天 + 每 50 回合」获得 +- **孵化步数**:基于 captureRate 反推,而不是物种真实的 `hatch_counter` 数据 +- **没有遗传招式**:原版中蛋可以遗传父母双方的招式 +- **没有个体值遗传**:原版中蛋会随机继承父母的某些 IV +- **没有球种遗传**:原版中蛋继承母亲的球种 + +### 18. 多语言名称覆盖极少 + +**文件**: `src/dex/names.ts` + +只有 10 个物种有中/英/日三语名称,其余 1000+ 个物种只回退到英文名。这对于中文/日文用户来说体验不完整。 + +### 19. 缺少 Held Item 获取途径 + +战斗中 `heldItem` 被正确传入 Showdown 格式,所以底层模拟会处理道具效果。但是: + +- 没有获得/装备道具的途径 +- 没有商店系统 +- 所有野生对手没有道具 +- 玩家的宝可梦默认 `heldItem: null` + +### 20. Ability 系统不完整 + +- `getDefaultAbility` 只取第一个非隐藏特性 +- 没有隐藏特性(Hidden Ability)的选择 +- 没有特性胶囊/特性补丁的使用 +- 底层 Showdown 会正确处理特性效果(如 Intimidate、Levitate),但 UI 层不显示特性触发 + +--- + +## 四、问题汇总 + +| 严重程度 | 数量 | 编号 | +|---------|------|------| +| 严重(核心机制错误) | 5 | #1 ~ #5 | +| 中等(机制简化/缺失) | 7 | #6 ~ #12 | +| 轻度(数值/细节偏差) | 8 | #13 ~ #20 | + +--- + +## 五、优先修复建议 + +按影响面从大到小排列: + +1. **修复 XP 和 EV 计算(#1, #2)**:从 `@pkmn/data` 获取真实的 `base_experience` 和 `evs` 数据,替换当前的代理算法。这两个问题直接影响所有战斗的成长反馈。 +2. **实现物品使用(#3)**:至少支持 Potion(回复 HP)和状态治愈药。这是战斗中最基本的交互。 +3. **实现逃跑(#4)**:需要添加逃跑概率公式和对应的 Showdown 协议处理。 +4. **修复野生对手招式(#7)**:从 learnset 中按等级获取招式,替换硬编码映射。 +5. **补全 Growth Rate 数据(#13)**:从 PokeAPI 或 `@pkmn/data` 批量导入,而非只覆盖 9 个物种。 + +--- + +## 六、做得好的部分 + +- **底层战斗引擎(`@pkmn/sim`)集成正确**:属性克制、伤害公式、能力值计算、特性效果等核心数学由 Pokémon Showdown 引擎处理,结果与原版一致。 +- **EV 上限正确**:单项 252 / 总计 510,与原版一致。 +- **XP 经验曲线公式正确**:6 种 Growth Rate 的计算公式(erratic、fluctuating 等)与原版完全一致。 +- **Nature 系统完整**:25 个性格及其加成/减益效果通过 `@pkmn/data` 正确获取。 +- **Learnset 查询正确**:从 `Dex.data.Learnsets` 获取招式学习表,支持跨代回退。 +- **状态异常映射基本正确**:6 种主要状态的 Showdown 协议映射准确。 +- **战斗测试覆盖全面**:包括属性克制、强制换人、多精灵队伍等场景的集成测试。 + +--- + +## 七、修复记录(2026-04-24) + +### 已修复 + +| 编号 | 问题 | 修复方式 | +|------|------|---------| +| #1 | XP 使用 baseStats.hp | 从 PokeAPI 获取真实 `base_experience`,存入 `pokedex-data.ts`,公式改为 `baseXP × level / 7` | +| #2 | EV yield 伪造 | 从 PokeAPI 获取真实 EV yield 数据(1024 个物种),存入 `pokedex-data.ts` | +| #3 | 物品使用无效 | 实现 Potion/HyperPotion/FullRestore 等回复药效果,直接操作 Battle 对象 HP,消耗背包物品 | +| #4 | 逃跑未实现 | 实现 Gen 9 逃跑概率公式 `f = (playerSpeed × 128 / opponentSpeed + 30 × attempts) % 256`,成功时 forfeit 结束战斗 | +| #6 | AI 纯随机 | AI 现在优先选克制招式(70%),避免被抵抗招式和蓄力招式,状态技最低优先级 | +| #7 | 野生招式硬编码 | 从 `Dex.data.Learnsets` 按等级获取升级招式(最后 4 个),替换按属性硬编码映射 | +| #8 | 进化只取第一目标 | 检查所有 `evos` 目标,支持分支进化,增加友谊度进化检测 | +| #13 | Growth Rate 只覆盖 9 个 | 从 PokeAPI 批量导入所有 1024 个物种的 growth rate 数据 | +| #5 | 多精灵对战不支持 | `createBattle` 支持传入 `OpponentEntry[]`,AI 换人时考虑属性克制 | +| #10 | 缺少捕获系统 | 新增 `capture.ts`,实现 Gen 9 捕获率公式,支持精灵球/状态修正 | +| #11 | 缺少 volatile status | 新增 `VolatileStatus` 类型,`BattlePokemon` 添加 `volatileStatus` 字段 | +| #12 | 天气/地形未投影 | 确认 `projectState` 从 `battle.field.weather/terrain` 读取 | +| #14 | Shiny 检测用随机 | 改为 Gen 3+ PID XOR 方法,阈值 < 16(Gen 8+ 1/4096 概率) | +| #15 | IV 生成用 LCRNG | 改为 Gen 3+ PID 位提取法(word1/word2 各取 3 个 5-bit IV) | +| #16 | 性别阈值精度丢失 | 从 `(rate/8)*256` 改为 `rate*32` 直接比较,消除浮点精度问题 | +| #17 | 蛋孵化步数用 captureRate | 改为使用真实 `hatchCounter` 数据(步数 = cycles × 257),支持进化阶段回退 | +| #18 | 多语言名称仅 10 个 | 创建 fetch 脚本获取全量中/日名称,`names.ts` 支持动态加载生成数据 | +| #19 | 野生对手无道具 | 添加 `rollWildHeldItem`:5% 物种专属道具、5% 树果、3% 属性增强道具 | +| #20 | Ability 只有第一个 | 新增 `randomAbility`/`getAbilities`,隐藏特性 5% 概率,第二特性 20% 概率 | + +### 新增文件 +- `src/dex/pokedex-data.ts` — 1024 个物种的 baseExperience、EV yield、growthRate、captureRate、baseHappiness 数据 +- `scripts/fetch-pokedex-data.ts` — PokeAPI 数据抓取脚本(可重新运行以更新数据,含 hatchCounter) +- `src/battle/capture.ts` — Gen 9 捕获率计算,精灵球/状态/时间修正 +- `scripts/fetch-species-names.ts` — 多语言名称抓取脚本(中/日/英) + +### 修改文件 +- `src/battle/settlement.ts` — XP/EV 计算、进化检测 +- `src/battle/engine.ts` — 物品效果、逃跑逻辑、野生招式、AI 调用、多对手支持、野生道具 +- `src/battle/ai.ts` — 属性克制 AI(使用 `Dex.getEffectiveness`) +- `src/battle/types.ts` — 新增 `run` 动作、`escaped`/`escapeAttempts`/`captureResult` 状态、VolatileStatus +- `src/battle/index.ts` — 导出 OpponentEntry、attemptCapture、CaptureResult +- `src/ui/BattleFlow.tsx` — 逃跑按钮、物品消耗 +- `src/dex/species.ts` — 使用 pokedex-data 替代硬编码 supplement +- `src/dex/learnsets.ts` — 新增 randomAbility、getAbilities 函数 +- `src/dex/names.ts` — 支持加载 auto-generated 多语言名称数据 +- `src/dex/pokedex-data.ts` — 新增 getHatchCounter 函数 +- `src/core/creature.ts` — PID 生成、IV 位提取、Shiny XOR 检测、randomAbility +- `src/core/gender.ts` — 修复阈值为 `genderRate * 32` +- `src/core/egg.ts` — 使用 getHatchCounter 替代 captureRate 计算孵化步数