PKHeX/PKHeX.Core/PersonalInfo/PersonalInfo.cs

368 lines
12 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
namespace PKHeX.Core
{
/// <summary>
/// Stat/misc data for individual species or their associated alternate form data.
/// </summary>
public abstract class PersonalInfo
{
2018-07-04 15:43:19 +00:00
/// <summary>
/// Raw Data
/// </summary>
PKHeX.Core Nullable cleanup (#2401) * Handle some nullable cases Refactor MysteryGift into a second abstract class (backed by a byte array, or fake data) Make some classes have explicit constructors instead of { } initialization * Handle bits more obviously without null * Make SaveFile.BAK explicitly readonly again * merge constructor methods to have readonly fields * Inline some properties * More nullable handling * Rearrange box actions define straightforward classes to not have any null properties * Make extrabyte reference array immutable * Move tooltip creation to designer * Rearrange some logic to reduce nesting * Cache generated fonts * Split mystery gift album purpose * Handle more tooltips * Disallow null setters * Don't capture RNG object, only type enum * Unify learnset objects Now have readonly properties which are never null don't new() empty learnsets (>800 Learnset objects no longer created, total of 2400 objects since we also new() a move & level array) optimize g1/2 reader for early abort case * Access rewrite Initialize blocks in a separate object, and get via that object removes a couple hundred "might be null" warnings since blocks are now readonly getters some block references have been relocated, but interfaces should expose all that's needed put HoF6 controls in a groupbox, and disable * Readonly personal data * IVs non nullable for mystery gift * Explicitly initialize forced encounter moves * Make shadow objects readonly & non-null Put murkrow fix in binary data resource, instead of on startup * Assign dex form fetch on constructor Fixes legality parsing edge cases also handle cxd parse for valid; exit before exception is thrown in FrameGenerator * Remove unnecessary null checks * Keep empty value until init SetPouch sets the value to an actual one during load, but whatever * Readonly team lock data * Readonly locks Put locked encounters at bottom (favor unlocked) * Mail readonly data / offset Rearrange some call flow and pass defaults Add fake classes for SaveDataEditor mocking Always party size, no need to check twice in stat editor use a fake save file as initial data for savedata editor, and for gamedata (wow i found a usage) constrain eventwork editor to struct variable types (uint, int, etc), thus preventing null assignment errors
2019-10-17 01:47:31 +00:00
protected readonly byte[] Data;
protected PersonalInfo(byte[] data) => Data = data;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Writes entry to raw bytes.
/// </summary>
/// <returns></returns>
public abstract byte[] Write();
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base HP
/// </summary>
public abstract int HP { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Attack
/// </summary>
public abstract int ATK { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Defense
/// </summary>
public abstract int DEF { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Speed
/// </summary>
public abstract int SPE { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Special Attack
/// </summary>
public abstract int SPA { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Special Defense
/// </summary>
public abstract int SPD { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Stat values
/// </summary>
public IReadOnlyList<int> Stats => new[] { HP, ATK, DEF, SPE, SPA, SPD };
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of HP Effort Values to yield when defeating this entry.
/// </summary>
public abstract int EV_HP { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of Attack Effort Values to yield when defeating this entry.
/// </summary>
public abstract int EV_ATK { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of Defense Effort Values to yield when defeating this entry.
/// </summary>
public abstract int EV_DEF { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of Speed Effort Values to yield when defeating this entry.
/// </summary>
public abstract int EV_SPE { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of Special Attack Effort Values to yield when defeating this entry.
/// </summary>
public abstract int EV_SPA { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of Special Defense Effort Values to yield when defeating this entry.
/// </summary>
public abstract int EV_SPD { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Primary Type
/// </summary>
public abstract int Type1 { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Secondary Type
/// </summary>
public abstract int Type2 { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// First Egg Group
/// </summary>
public abstract int EggGroup1 { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Second Egg Group
/// </summary>
public abstract int EggGroup2 { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Catch Rate
/// </summary>
public abstract int CatchRate { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Evolution Stage value (or equivalent for un-evolved).
2018-07-04 15:43:19 +00:00
/// </summary>
public virtual int EvoStage { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Held Items the entry can be randomly encountered with.
/// </summary>
public abstract IReadOnlyList<int> Items { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Gender Ratio value determining if the entry is a fixed gender or bi-gendered.
2018-07-04 15:43:19 +00:00
/// </summary>
public abstract int Gender { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Amount of Hatching Step Cycles required to hatch if in an egg.
/// </summary>
public abstract int HatchCycles { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Initial Friendship when captured or received.
/// </summary>
public abstract int BaseFriendship { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Experience-Level Growth Rate type
/// </summary>
public abstract int EXPGrowth { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Full list of <see cref="PKM.Ability"/> values the entry can have.
/// </summary>
public abstract IReadOnlyList<int> Abilities { get; set; }
/// <summary>
/// Gets the ability index without creating an array and looking through it.
/// </summary>
/// <param name="abilityID">Ability ID</param>
/// <returns>Ability Index</returns>
public abstract int GetAbilityIndex(int abilityID);
2018-07-04 15:43:19 +00:00
/// <summary>
/// Escape factor used for fleeing the Safari Zone or calling for help in SOS Battles.
/// </summary>
public abstract int EscapeRate { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Count of <see cref="PKM.Form"/> values the entry can have.
2018-07-04 15:43:19 +00:00
/// </summary>
public virtual int FormCount { get; set; } = 1;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Pointer to the first <see cref="PKM.Form"/> <see cref="PersonalInfo"/> index
2018-07-04 15:43:19 +00:00
/// </summary>
protected internal virtual int FormStatsIndex { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Pointer to the <see cref="PKM.Form"/> sprite index.
2018-07-04 15:43:19 +00:00
/// </summary>
public virtual int FormSprite { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Experience Yield factor
/// </summary>
public abstract int BaseEXP { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Main color ID of the entry. The majority of the Pokémon's color is of this color, usually.
2018-07-04 15:43:19 +00:00
/// </summary>
public abstract int Color { get; set; }
2018-07-04 15:43:19 +00:00
/// <summary>
/// Height of the entry in meters (m).
/// </summary>
public virtual int Height { get; set; } = 0;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Mass of the entry in kilograms (kg).
/// </summary>
public virtual int Weight { get; set; } = 0;
2017-11-07 06:44:51 +00:00
/// <summary>
/// TM/HM learn compatibility flags for individual moves.
/// </summary>
public bool[] TMHM = Array.Empty<bool>();
2018-07-04 15:43:19 +00:00
2017-11-07 06:44:51 +00:00
/// <summary>
/// Grass-Fire-Water-Etc typed learn compatibility flags for individual moves.
/// </summary>
public bool[] TypeTutors = Array.Empty<bool>();
2018-07-04 15:43:19 +00:00
2017-11-07 06:44:51 +00:00
/// <summary>
/// Special tutor learn compatibility flags for individual moves.
/// </summary>
public bool[][] SpecialTutors = Array.Empty<bool[]>();
protected static bool[] GetBits(byte[] data, int start = 0, int length = -1)
{
if (length < 0)
length = data.Length;
bool[] result = new bool[length << 3];
for (int i = 0; i < result.Length; i++)
result[i] = (data[start + (i >> 3)] >> (i & 7) & 0x1) == 1;
return result;
}
protected static byte[] SetBits(bool[] bits)
{
byte[] data = new byte[bits.Length>>3];
for (int i = 0; i < bits.Length; i++)
data[i>>3] |= (byte)(bits[i] ? 1 << (i&0x7) : 0);
return data;
}
/// <summary>
2017-11-07 06:44:51 +00:00
/// Injects supplementary TM/HM compatibility which is not present in the generation specific <see cref="PersonalInfo"/> format.
/// </summary>
/// <param name="data">Data to read from</param>
/// <param name="start">Starting offset to read at</param>
/// <param name="length">Amount of bytes to decompose into bits</param>
internal void AddTMHM(byte[] data, int start = 0, int length = -1) => TMHM = GetBits(data, start, length);
2018-07-04 15:43:19 +00:00
/// <summary>
2017-11-07 06:44:51 +00:00
/// Injects supplementary Type Tutor compatibility which is not present in the generation specific <see cref="PersonalInfo"/> format.
/// </summary>
/// <param name="data">Data to read from</param>
/// <param name="start">Starting offset to read at</param>
/// <param name="length">Amount of bytes to decompose into bits</param>
internal void AddTypeTutors(byte[] data, int start = 0, int length = -1) => TypeTutors = GetBits(data, start, length);
2017-11-07 06:44:51 +00:00
/// <summary>
/// Gets the <see cref="PersonalTable"/> <see cref="PKM.Form"/> entry index for the input criteria, with fallback for the original species entry.
2017-11-07 06:44:51 +00:00
/// </summary>
/// <param name="species"><see cref="PKM.Species"/> to retrieve for</param>
/// <param name="form"><see cref="PKM.Form"/> to retrieve for</param>
/// <returns>Index the <see cref="PKM.Form"/> exists as in the <see cref="PersonalTable"/>.</returns>
public int FormIndex(int species, int form)
{
if (!HasForm(form))
return species;
return FormStatsIndex + form - 1;
}
/// <summary>
/// Checks if the <see cref="PersonalInfo"/> has the requested <see cref="PKM.Form"/> entry index available.
/// </summary>
/// <param name="form"><see cref="PKM.Form"/> to retrieve for</param>
public bool HasForm(int form)
{
if (form <= 0) // no forme requested
return false;
if (FormStatsIndex <= 0) // no formes present
return false;
if (form >= FormCount) // beyond range of species' formes
return false;
return true;
}
2018-07-04 15:43:19 +00:00
/// <summary>
/// Gets a random valid gender for the entry.
/// </summary>
public int RandomGender()
{
var fix = FixedGender;
return fix >= 0 ? fix : Util.Rand.Next(2);
}
public bool IsDualGender => FixedGender < 0;
public int FixedGender
{
get
{
if (Genderless)
return 2;
if (OnlyFemale)
return 1;
if (OnlyMale)
return 0;
return -1;
}
}
2018-07-04 15:43:19 +00:00
/// <summary>
/// Indicates that the entry is exclusively Genderless.
/// </summary>
public bool Genderless => Gender == 255;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Indicates that the entry is exclusively Female gendered.
/// </summary>
public bool OnlyFemale => Gender == 254;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Indicates that the entry is exclusively Male gendered.
/// </summary>
public bool OnlyMale => Gender == 0;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Indicates if the entry has Formes or not.
/// </summary>
public bool HasForms => FormCount > 1;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Base Stat Total sum of all stats.
/// </summary>
public int BST => HP + ATK + DEF + SPE + SPA + SPD;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Checks to see if the <see cref="PKM.Form"/> is valid within the <see cref="FormCount"/>
2018-07-04 15:43:19 +00:00
/// </summary>
/// <param name="form"></param>
2018-07-04 15:43:19 +00:00
/// <returns></returns>
public bool IsFormWithinRange(int form)
{
if (form == 0)
return true;
return form < FormCount;
}
2018-07-04 15:43:19 +00:00
/// <summary>
/// Checks to see if the provided Types match the entry's types.
/// </summary>
/// <remarks>Input order matters! If input order does not matter, use <see cref="o:IsType(type1, type2)"/>.</remarks>
/// <param name="type1">First type</param>
/// <param name="type2">Second type</param>
/// <returns>Typing is an exact match</returns>
public bool IsValidTypeCombination(int type1, int type2) => Type1 == type1 && Type2 == type2;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Checks if the entry has either type equal to the input type.
/// </summary>
/// <param name="type1">Type</param>
/// <returns>Typing is present in entry</returns>
public bool IsType(int type1) => Type1 == type1 || Type2 == type1;
2018-07-04 15:43:19 +00:00
/// <summary>
/// Checks if the entry has either type equal to both input types.
/// </summary>
/// <remarks>Input order does not matter.</remarks>
/// <param name="type1">Type 1</param>
/// <param name="type2">Type 2</param>
/// <returns>Typing is present in entry</returns>
public bool IsType(int type1, int type2) => (Type1 == type1 || Type2 == type1) && (Type1 == type2 || Type2 == type2);
2018-07-04 15:43:19 +00:00
/// <summary>
/// Checks if the entry has either egg group equal to the input type.
/// </summary>
/// <param name="group">Egg group</param>
/// <returns>Egg is present in entry</returns>
public bool IsEggGroup(int group) => EggGroup1 == group || EggGroup2 == group;
}
}