using System;
using System.Collections.Generic;
using System.Linq;
namespace PKHeX.Core
{
public abstract class DataMysteryGift : MysteryGift
{
public readonly byte[] Data;
protected DataMysteryGift(byte[] data) => Data = data;
///
/// Returns an array for exporting outside the program (to disk, etc).
///
public virtual byte[] Write() => Data;
public override int GetHashCode()
{
int hash = 17;
foreach (var b in Data)
hash = (hash * 31) + b;
return hash;
}
///
/// Creates a deep copy of the object data.
///
///
public override MysteryGift Clone()
{
byte[] data = (byte[])Data.Clone();
var result = GetMysteryGift(data);
if (result == null)
throw new ArgumentException(nameof(MysteryGift));
return result;
}
public override bool Empty => Data.IsRangeAll((byte)0, 0, Data.Length);
}
///
/// Mystery Gift Template File
///
public abstract class MysteryGift : IEncounterable, IMoveset, IRelearn, ILocation
{
///
/// Determines whether or not the given length of bytes is valid for a mystery gift.
///
/// Length, in bytes, of the data of which to determine validity.
/// A boolean indicating whether or not the given length is valid for a mystery gift.
public static bool IsMysteryGift(long len) => Sizes.Contains((int)len);
private static readonly HashSet Sizes = new() { WC8.Size, WC6Full.Size, WC6.Size, PGF.Size, PGT.Size, PCD.Size };
///
/// Converts the given data to a .
///
/// Raw data of the mystery gift.
/// Extension of the file from which the was retrieved.
/// An instance of representing the given data, or null if or is invalid.
/// This overload differs from by checking the / combo for validity. If either is invalid, a null reference is returned.
public static DataMysteryGift? GetMysteryGift(byte[] data, string ext) => data.Length switch
{
PGT.Size when ext == ".pgt" => new PGT(data),
PCD.Size when ext is ".pcd" or ".wc4" => new PCD(data),
PGF.Size when ext == ".pgf" => new PGF(data),
WC6.Size when ext == ".wc6" => new WC6(data),
WC7.Size when ext == ".wc7" => new WC7(data),
WB7.Size when ext == ".wb7" => new WB7(data),
WR7.Size when ext == ".wr7" => new WR7(data),
WC8.Size when ext is ".wc8" or ".wc8full" => new WC8(data),
WB7.SizeFull when ext == ".wb7full" => new WB7(data),
WC6Full.Size when ext == ".wc6full" => new WC6Full(data).Gift,
WC7Full.Size when ext == ".wc7full" => new WC7Full(data).Gift,
_ => null
};
///
/// Converts the given data to a .
///
/// Raw data of the mystery gift.
/// An instance of representing the given data, or null if is invalid.
public static DataMysteryGift? GetMysteryGift(byte[] data) => data.Length switch
{
PGT.Size => new PGT(data),
PCD.Size => new PCD(data),
PGF.Size => new PGF(data),
WR7.Size => new WR7(data),
WC8.Size => new WC8(data),
// WC6/WC7: Check year
WC6.Size => BitConverter.ToUInt32(data, 0x4C) / 10000 < 2000 ? new WC7(data) : new WC6(data),
// WC6Full/WC7Full: 0x205 has 3 * 0x46 for gen6, now only 2.
WC6Full.Size => data[0x205] == 0 ? new WC7Full(data).Gift : new WC6Full(data).Gift,
_ => null
};
public string Extension => GetType().Name.ToLower();
public string FileName => $"{CardHeader}.{Extension}";
public abstract int Generation { get; }
public PKM ConvertToPKM(ITrainerInfo sav) => ConvertToPKM(sav, EncounterCriteria.Unrestricted);
public abstract PKM ConvertToPKM(ITrainerInfo sav, EncounterCriteria criteria);
protected abstract bool IsMatchExact(PKM pkm, DexLevel evo);
protected abstract bool IsMatchDeferred(PKM pkm);
public EncounterMatchRating IsMatch(PKM pkm, DexLevel evo)
{
if (!IsMatchExact(pkm, evo))
return EncounterMatchRating.None;
if (IsMatchDeferred(pkm))
return EncounterMatchRating.Deferred;
return EncounterMatchRating.Match;
}
///
/// Creates a deep copy of the object data.
///
///
public abstract MysteryGift Clone();
///
/// Gets a friendly name for the underlying type.
///
public string Type => GetType().Name;
///
/// Gets a friendly name for the underlying type for the interface.
///
public string Name => "Event Gift";
///
/// Gets a friendly name for the underlying type for the interface.
///
public string LongName => $"{Name} ({Type})";
public virtual GameVersion Version
{
get => GameUtil.GetVersion(Generation);
set { }
}
// Properties
public virtual int Species { get => -1; set { } }
public abstract bool GiftUsed { get; set; }
public abstract string CardTitle { get; set; }
public abstract int CardID { get; set; }
public abstract bool IsItem { get; set; }
public abstract int ItemID { get; set; }
public abstract bool IsPokémon { get; set; }
public virtual int Quantity { get => 1; set { } }
public virtual bool Empty => false;
public virtual bool IsBP { get => false; set { } }
public virtual int BP { get => 0; set { } }
public virtual bool IsBean { get => false; set { } }
public virtual int Bean { get => 0; set { } }
public virtual string CardHeader => (CardID > 0 ? $"Card #: {CardID:0000}" : "N/A") + $" - {CardTitle.Replace('\u3000',' ').Trim()}";
// Search Properties
public virtual IReadOnlyList Moves { get => Array.Empty(); set { } }
public virtual IReadOnlyList Relearn { get => Array.Empty(); set { } }
public virtual int[] IVs { get => Array.Empty(); set { } }
public virtual bool IsShiny => false;
public virtual bool IsEgg { get => false; set { } }
public virtual int HeldItem { get => -1; set { } }
public virtual int AbilityType { get => -1; set { } }
public abstract int Gender { get; set; }
public abstract int Form { get; set; }
public abstract int TID { get; set; }
public abstract int SID { get; set; }
public abstract string OT_Name { get; set; }
public abstract int Location { get; set; }
public abstract int Level { get; set; }
public int LevelMin => Level;
public int LevelMax => Level;
public abstract int Ball { get; set; }
public virtual bool EggEncounter => IsEgg;
public abstract int EggLocation { get; set; }
public int TrainerID7 => (int)((uint)(TID | (SID << 16)) % 1000000);
public int TrainerSID7 => (int)((uint)(TID | (SID << 16)) / 1000000);
///
/// Checks if the has the in its current move list.
///
public bool HasMove(int move) => Moves.Contains(move);
}
}