Spaces:
Running
Running
import type { PokemonEventMethods, ConditionData } from './dex-conditions'; | |
import { assignMissingFields, BasicEffect, toID } from './dex-data'; | |
import { Utils } from '../lib/utils'; | |
interface FlingData { | |
basePower: number; | |
status?: string; | |
volatileStatus?: string; | |
effect?: CommonHandlers['ResultMove']; | |
} | |
export interface ItemData extends Partial<Item>, PokemonEventMethods { | |
name: string; | |
} | |
export type ModdedItemData = ItemData | Partial<Omit<ItemData, 'name'>> & { | |
inherit: true, | |
onCustap?: (this: Battle, pokemon: Pokemon) => void, | |
}; | |
export interface ItemDataTable { [itemid: IDEntry]: ItemData } | |
export interface ModdedItemDataTable { [itemid: IDEntry]: ModdedItemData } | |
export class Item extends BasicEffect implements Readonly<BasicEffect> { | |
declare readonly effectType: 'Item'; | |
/** just controls location on the item spritesheet */ | |
declare readonly num: number; | |
/** | |
* A Move-like object depicting what happens when Fling is used on | |
* this item. | |
*/ | |
readonly fling?: FlingData; | |
/** | |
* If this is a Drive: The type it turns Techno Blast into. | |
* undefined, if not a Drive. | |
*/ | |
readonly onDrive?: string; | |
/** | |
* If this is a Memory: The type it turns Multi-Attack into. | |
* undefined, if not a Memory. | |
*/ | |
readonly onMemory?: string; | |
/** | |
* If this is a mega stone: The name (e.g. Charizard-Mega-X) of the | |
* forme this allows transformation into. | |
* undefined, if not a mega stone. | |
*/ | |
readonly megaStone?: string; | |
/** | |
* If this is a mega stone: The name (e.g. Charizard) of the | |
* forme this allows transformation from. | |
* undefined, if not a mega stone. | |
*/ | |
readonly megaEvolves?: string; | |
/** | |
* If this is a Z crystal: true if the Z Crystal is generic | |
* (e.g. Firium Z). If species-specific, the name | |
* (e.g. Inferno Overdrive) of the Z Move this crystal allows | |
* the use of. | |
* undefined, if not a Z crystal. | |
*/ | |
readonly zMove?: true | string; | |
/** | |
* If this is a generic Z crystal: The type (e.g. Fire) of the | |
* Z Move this crystal allows the use of (e.g. Fire) | |
* undefined, if not a generic Z crystal | |
*/ | |
readonly zMoveType?: string; | |
/** | |
* If this is a species-specific Z crystal: The name | |
* (e.g. Play Rough) of the move this crystal requires its | |
* holder to know to use its Z move. | |
* undefined, if not a species-specific Z crystal | |
*/ | |
readonly zMoveFrom?: string; | |
/** | |
* If this is a species-specific Z crystal: An array of the | |
* species of Pokemon that can use this crystal's Z move. | |
* Note that these are the full names, e.g. 'Mimikyu-Busted' | |
* undefined, if not a species-specific Z crystal | |
*/ | |
readonly itemUser?: string[]; | |
/** Is this item a Berry? */ | |
readonly isBerry: boolean; | |
/** Whether or not this item ignores the Klutz ability. */ | |
readonly ignoreKlutz: boolean; | |
/** The type the holder will change into if it is an Arceus. */ | |
readonly onPlate?: string; | |
/** Is this item a Gem? */ | |
readonly isGem: boolean; | |
/** Is this item a Pokeball? */ | |
readonly isPokeball: boolean; | |
/** Is this item a Red or Blue Orb? */ | |
readonly isPrimalOrb: boolean; | |
declare readonly condition?: ConditionData; | |
declare readonly forcedForme?: string; | |
declare readonly isChoice?: boolean; | |
declare readonly naturalGift?: { basePower: number, type: string }; | |
declare readonly spritenum?: number; | |
declare readonly boosts?: SparseBoostsTable | false; | |
declare readonly onEat?: ((this: Battle, pokemon: Pokemon) => void) | false; | |
declare readonly onUse?: ((this: Battle, pokemon: Pokemon) => void) | false; | |
declare readonly onStart?: (this: Battle, target: Pokemon) => void; | |
declare readonly onEnd?: (this: Battle, target: Pokemon) => void; | |
constructor(data: AnyObject) { | |
super(data); | |
this.fullname = `item: ${this.name}`; | |
this.effectType = 'Item'; | |
this.fling = data.fling || undefined; | |
this.onDrive = data.onDrive || undefined; | |
this.onMemory = data.onMemory || undefined; | |
this.megaStone = data.megaStone || undefined; | |
this.megaEvolves = data.megaEvolves || undefined; | |
this.zMove = data.zMove || undefined; | |
this.zMoveType = data.zMoveType || undefined; | |
this.zMoveFrom = data.zMoveFrom || undefined; | |
this.itemUser = data.itemUser || undefined; | |
this.isBerry = !!data.isBerry; | |
this.ignoreKlutz = !!data.ignoreKlutz; | |
this.onPlate = data.onPlate || undefined; | |
this.isGem = !!data.isGem; | |
this.isPokeball = !!data.isPokeball; | |
this.isPrimalOrb = !!data.isPrimalOrb; | |
if (!this.gen) { | |
if (this.num >= 1124) { | |
this.gen = 9; | |
} else if (this.num >= 927) { | |
this.gen = 8; | |
} else if (this.num >= 689) { | |
this.gen = 7; | |
} else if (this.num >= 577) { | |
this.gen = 6; | |
} else if (this.num >= 537) { | |
this.gen = 5; | |
} else if (this.num >= 377) { | |
this.gen = 4; | |
} else { | |
this.gen = 3; | |
} | |
// Due to difference in gen 2 item numbering, gen 2 items must be | |
// specified manually | |
} | |
if (this.isBerry) this.fling = { basePower: 10 }; | |
if (this.id.endsWith('plate')) this.fling = { basePower: 90 }; | |
if (this.onDrive) this.fling = { basePower: 70 }; | |
if (this.megaStone) this.fling = { basePower: 80 }; | |
if (this.onMemory) this.fling = { basePower: 50 }; | |
assignMissingFields(this, data); | |
} | |
} | |
const EMPTY_ITEM = Utils.deepFreeze(new Item({ name: '', exists: false })); | |
export class DexItems { | |
readonly dex: ModdedDex; | |
readonly itemCache = new Map<ID, Item>(); | |
allCache: readonly Item[] | null = null; | |
constructor(dex: ModdedDex) { | |
this.dex = dex; | |
} | |
get(name?: string | Item): Item { | |
if (name && typeof name !== 'string') return name; | |
const id = name ? toID(name.trim()) : '' as ID; | |
return this.getByID(id); | |
} | |
getByID(id: ID): Item { | |
if (id === '') return EMPTY_ITEM; | |
let item = this.itemCache.get(id); | |
if (item) return item; | |
if (this.dex.data.Aliases.hasOwnProperty(id)) { | |
item = this.get(this.dex.data.Aliases[id]); | |
if (item.exists) { | |
this.itemCache.set(id, item); | |
} | |
return item; | |
} | |
if (id && !this.dex.data.Items[id] && this.dex.data.Items[id + 'berry']) { | |
item = this.getByID(id + 'berry' as ID); | |
this.itemCache.set(id, item); | |
return item; | |
} | |
if (id && this.dex.data.Items.hasOwnProperty(id)) { | |
const itemData = this.dex.data.Items[id] as any; | |
const itemTextData = this.dex.getDescs('Items', id, itemData); | |
item = new Item({ | |
name: id, | |
...itemData, | |
...itemTextData, | |
}); | |
if (item.gen > this.dex.gen) { | |
(item as any).isNonstandard = 'Future'; | |
} | |
if (this.dex.parentMod) { | |
// If this item is exactly identical to parentMod's item, reuse parentMod's copy | |
const parent = this.dex.mod(this.dex.parentMod); | |
if (itemData === parent.data.Items[id]) { | |
const parentItem = parent.items.getByID(id); | |
if ( | |
item.isNonstandard === parentItem.isNonstandard && | |
item.desc === parentItem.desc && | |
item.shortDesc === parentItem.shortDesc | |
) { | |
item = parentItem; | |
} | |
} | |
} | |
} else { | |
item = new Item({ name: id, exists: false }); | |
} | |
if (item.exists) this.itemCache.set(id, this.dex.deepFreeze(item)); | |
return item; | |
} | |
all(): readonly Item[] { | |
if (this.allCache) return this.allCache; | |
const items = []; | |
for (const id in this.dex.data.Items) { | |
items.push(this.getByID(id as ID)); | |
} | |
this.allCache = Object.freeze(items); | |
return this.allCache; | |
} | |
} | |