2023-01-22 04:02:33 +00:00
using System ;
2021-08-21 03:33:53 +00:00
using System.Collections.Generic ;
2022-01-03 05:35:59 +00:00
using static System . Buffers . Binary . BinaryPrimitives ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
namespace PKHeX.Core ;
/// <summary>
/// Generation 3 <see cref="SaveFile"/> object for <see cref="GameVersion.E"/>.
/// </summary>
/// <inheritdoc cref="SAV3" />
public sealed class SAV3E : SAV3 , IGen3Hoenn , IGen3Joyful , IGen3Wonder
2021-03-16 06:51:58 +00:00
{
2022-06-18 18:04:24 +00:00
// Configuration
2023-01-22 04:02:33 +00:00
protected override SAV3E CloneInternal ( ) = > new ( Write ( ) ) ;
2022-06-18 18:04:24 +00:00
public override GameVersion Version { get = > GameVersion . E ; protected set { } }
2023-01-22 04:02:33 +00:00
public override PersonalTable3 Personal = > PersonalTable . E ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public override int EventFlagCount = > 8 * 300 ;
public override int EventWorkCount = > 0x100 ;
protected override int DaycareSlotSize = > SIZE_STORED + 0x3C ; // 0x38 mail + 4 exp
public override int DaycareSeedSize = > 8 ; // 32bit
protected override int EggEventFlag = > 0x86 ;
protected override int BadgeFlagStart = > 0x867 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public SAV3E ( byte [ ] data ) : base ( data ) = > Initialize ( ) ;
public SAV3E ( bool japanese = false ) : base ( japanese ) = > Initialize ( ) ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
protected override int EventFlag = > 0x1270 ;
protected override int EventWork = > 0x139C ;
2022-04-10 01:12:57 +00:00
2022-06-18 18:04:24 +00:00
private void Initialize ( )
{
// small
PokeDex = 0x18 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
// large
DaycareOffset = 0x3030 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
// storage
Box = 0 ;
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
#region Small
public override bool NationalDex
{
get = > PokedexNationalMagicRSE = = PokedexNationalUnlockRSE ;
set
2021-03-16 06:51:58 +00:00
{
2022-06-18 18:04:24 +00:00
PokedexMode = value ? ( byte ) 1 : ( byte ) 0 ; // mode
PokedexNationalMagicRSE = value ? PokedexNationalUnlockRSE : ( byte ) 0 ; // magic
SetEventFlag ( 0x896 , value ) ;
SetWork ( 0x46 , PokedexNationalUnlockWorkRSE ) ;
2021-03-16 06:51:58 +00:00
}
2022-06-18 18:04:24 +00:00
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public override uint SecurityKey
{
get = > ReadUInt32LittleEndian ( Small . AsSpan ( 0xAC ) ) ;
set = > WriteUInt32LittleEndian ( Small . AsSpan ( 0xAC ) , value ) ;
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public RTC3 ClockInitial
{
2023-03-26 06:14:50 +00:00
get = > new ( Small . AsSpan ( 0x98 , RTC3 . Size ) . ToArray ( ) ) ;
set = > SetData ( Small . AsSpan ( 0x98 ) , value . Data ) ;
2022-06-18 18:04:24 +00:00
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public RTC3 ClockElapsed
{
2023-03-26 06:14:50 +00:00
get = > new ( Small . AsSpan ( 0xA0 , RTC3 . Size ) . ToArray ( ) ) ;
set = > SetData ( Small . AsSpan ( 0xA0 ) , value . Data ) ;
2022-06-18 18:04:24 +00:00
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public ushort JoyfulJumpInRow { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x1FC ) ) ; set = > WriteUInt16LittleEndian ( Small . AsSpan ( 0x1FC ) , Math . Min ( ( ushort ) 9999 , value ) ) ; }
// u16 field2;
public ushort JoyfulJump5InRow { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x200 ) ) ; set = > WriteUInt16LittleEndian ( Small . AsSpan ( 0x200 ) , Math . Min ( ( ushort ) 9999 , value ) ) ; }
public ushort JoyfulJumpGamesMaxPlayers { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x202 ) ) ; set = > WriteUInt16LittleEndian ( Small . AsSpan ( 0x202 ) , Math . Min ( ( ushort ) 9999 , value ) ) ; }
// u32 field8;
public uint JoyfulJumpScore { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x208 ) ) ; set = > WriteUInt32LittleEndian ( Small . AsSpan ( 0x208 ) , Math . Min ( 9999 , value ) ) ; }
2021-03-17 01:48:33 +00:00
2022-06-18 18:04:24 +00:00
public uint JoyfulBerriesScore { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x20C ) ) ; set = > WriteUInt32LittleEndian ( Small . AsSpan ( 0x20C ) , Math . Min ( 9999 , value ) ) ; }
public ushort JoyfulBerriesInRow { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x210 ) ) ; set = > WriteUInt16LittleEndian ( Small . AsSpan ( 0x210 ) , Math . Min ( ( ushort ) 9999 , value ) ) ; }
public ushort JoyfulBerries5InRow { get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0x212 ) ) ; set = > WriteUInt16LittleEndian ( Small . AsSpan ( 0x212 ) , Math . Min ( ( ushort ) 9999 , value ) ) ; }
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public uint BP
{
get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0xEB8 ) ) ;
set
2021-03-16 06:51:58 +00:00
{
2022-06-18 18:04:24 +00:00
if ( value > 9999 )
value = 9999 ;
WriteUInt16LittleEndian ( Small . AsSpan ( 0xEB8 ) , ( ushort ) value ) ;
2021-03-16 06:51:58 +00:00
}
2022-06-18 18:04:24 +00:00
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public uint BPEarned
{
get = > ReadUInt16LittleEndian ( Small . AsSpan ( 0xEBA ) ) ;
set
2021-03-16 06:51:58 +00:00
{
2022-06-18 18:04:24 +00:00
if ( value > 65535 )
value = 65535 ;
WriteUInt16LittleEndian ( Small . AsSpan ( 0xEBA ) , ( ushort ) value ) ;
2021-03-16 06:51:58 +00:00
}
2022-06-18 18:04:24 +00:00
}
#endregion
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
#region Large
public override int PartyCount { get = > Large [ 0x234 ] ; protected set = > Large [ 0x234 ] = ( byte ) value ; }
public override int GetPartyOffset ( int slot ) = > 0x238 + ( SIZE_PARTY * slot ) ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public override uint Money
{
get = > ReadUInt32LittleEndian ( Large . AsSpan ( 0x0490 ) ) ^ SecurityKey ;
set = > WriteUInt32LittleEndian ( Large . AsSpan ( 0x0490 ) , value ^ SecurityKey ) ;
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public override uint Coin
{
get = > ( ushort ) ( ReadUInt16LittleEndian ( Large . AsSpan ( 0x0494 ) ) ^ SecurityKey ) ;
set = > WriteUInt16LittleEndian ( Large . AsSpan ( 0x0494 ) , ( ushort ) ( value ^ SecurityKey ) ) ;
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
private const int OFS_PCItem = 0x0498 ;
private const int OFS_PouchHeldItem = 0x0560 ;
private const int OFS_PouchKeyItem = 0x05D8 ;
private const int OFS_PouchBalls = 0x0650 ;
private const int OFS_PouchTMHM = 0x0690 ;
private const int OFS_PouchBerry = 0x0790 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
protected override InventoryPouch3 [ ] GetItems ( )
{
const int max = 99 ;
2023-04-16 19:58:07 +00:00
var info = ItemStorage3E . Instance ;
2022-06-18 18:04:24 +00:00
return new InventoryPouch3 [ ]
2021-03-16 06:51:58 +00:00
{
2023-04-16 19:58:07 +00:00
new ( InventoryType . Items , info , max , OFS_PouchHeldItem , ( OFS_PouchKeyItem - OFS_PouchHeldItem ) / 4 ) ,
new ( InventoryType . KeyItems , info , 1 , OFS_PouchKeyItem , ( OFS_PouchBalls - OFS_PouchKeyItem ) / 4 ) ,
new ( InventoryType . Balls , info , max , OFS_PouchBalls , ( OFS_PouchTMHM - OFS_PouchBalls ) / 4 ) ,
new ( InventoryType . TMHMs , info , max , OFS_PouchTMHM , ( OFS_PouchBerry - OFS_PouchTMHM ) / 4 ) ,
new ( InventoryType . Berries , info , 999 , OFS_PouchBerry , 46 ) ,
new ( InventoryType . PCItems , info , 999 , OFS_PCItem , ( OFS_PouchHeldItem - OFS_PCItem ) / 4 ) ,
2022-06-18 18:04:24 +00:00
} ;
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public PokeBlock3Case PokeBlocks
{
get = > new ( Large , 0x848 ) ;
2023-03-26 06:14:50 +00:00
set = > SetData ( Large . AsSpan ( 0x848 ) , value . Write ( ) ) ;
2022-06-18 18:04:24 +00:00
}
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
protected override int SeenOffset2 = > 0x988 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public DecorationInventory3 Decorations = > new ( Large . AsSpan ( 0x2734 , DecorationInventory3 . SIZE ) ) ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public Swarm3 Swarm
{
2023-03-26 06:14:50 +00:00
get = > new ( Large . AsSpan ( 0x2B90 , Swarm3 . SIZE ) . ToArray ( ) ) ;
set = > SetData ( Large . AsSpan ( 0x2B90 ) , value . Data ) ;
2022-06-18 18:04:24 +00:00
}
2021-08-21 02:52:31 +00:00
2022-06-18 18:04:24 +00:00
private void ClearSwarm ( ) = > Large . AsSpan ( 0x2B90 , Swarm3 . SIZE ) . Clear ( ) ;
2022-03-26 02:47:23 +00:00
2022-06-18 18:04:24 +00:00
public IReadOnlyList < Swarm3 > DefaultSwarms = > Swarm3Details . Swarms_E ;
2021-08-21 03:33:53 +00:00
2022-06-18 18:04:24 +00:00
public int SwarmIndex
{
get = > Array . FindIndex ( Swarm3Details . Swarms_E , z = > z . MapNum = = Swarm . MapNum ) ;
set
2021-08-21 03:33:53 +00:00
{
2022-06-18 18:04:24 +00:00
var arr = DefaultSwarms ;
if ( ( uint ) value > = arr . Count )
ClearSwarm ( ) ;
else
Swarm = arr [ value ] ;
2021-08-21 03:33:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2021-08-21 03:33:53 +00:00
2022-06-18 18:04:24 +00:00
protected override int MailOffset = > 0x2BE0 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
protected override int GetDaycareEXPOffset ( int slot ) = > GetDaycareSlotOffset ( 0 , slot + 1 ) - 4 ; // @ end of each pk slot
public override string GetDaycareRNGSeed ( int loc ) = > ReadUInt32LittleEndian ( Large . AsSpan ( GetDaycareSlotOffset ( 0 , 2 ) ) ) . ToString ( "X8" ) ; // after the 2 slots, before the step counter
public override void SetDaycareRNGSeed ( int loc , string seed ) = > WriteUInt32LittleEndian ( Large . AsSpan ( GetDaycareEXPOffset ( 2 ) ) , Util . GetHexValue ( seed ) ) ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
protected override int ExternalEventData = > 0x31B3 ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
#region eBerry
private const int OFFSET_EBERRY = 0x31F8 ;
private const int SIZE_EBERRY = 0x134 ;
2021-03-16 06:51:58 +00:00
2023-03-20 23:26:23 +00:00
public override byte [ ] GetEReaderBerry ( ) = > Large . Slice ( OFFSET_EBERRY , SIZE_EBERRY ) ;
public override void SetEReaderBerry ( ReadOnlySpan < byte > data ) = > data . CopyTo ( Large . AsSpan ( OFFSET_EBERRY ) ) ;
2021-03-16 06:51:58 +00:00
2022-06-18 18:04:24 +00:00
public override string EBerryName = > GetString ( Large . AsSpan ( OFFSET_EBERRY , 7 ) ) ;
public override bool IsEBerryEngima = > Large [ OFFSET_EBERRY ] is 0 or 0xFF ;
#endregion
2021-03-16 06:51:58 +00:00
2023-03-20 23:26:23 +00:00
#region eTrainer
public override byte [ ] GetEReaderTrainer ( ) = > Small . Slice ( 0xBEC , 0xBC ) ;
public override void SetEReaderTrainer ( ReadOnlySpan < byte > data ) = > data . CopyTo ( Small . AsSpan ( 0xBEC ) ) ;
#endregion
2022-06-18 18:04:24 +00:00
public int WonderOffset = > WonderNewsOffset ;
private const int WonderNewsOffset = 0x322C ;
2023-03-20 23:26:23 +00:00
private int WonderCardOffset = > WonderNewsOffset + ( Japanese ? WonderNews3 . SIZE_JAP : WonderNews3 . SIZE ) ;
private int WonderCardExtraOffset = > WonderCardOffset + ( Japanese ? WonderCard3 . SIZE_JAP : WonderCard3 . SIZE ) ;
2021-03-28 01:58:51 +00:00
2023-03-26 06:14:50 +00:00
public WonderNews3 WonderNews { get = > new ( Large . Slice ( WonderNewsOffset , Japanese ? WonderNews3 . SIZE_JAP : WonderNews3 . SIZE ) ) ; set = > SetData ( Large . AsSpan ( WonderOffset ) , value . Data ) ; }
public WonderCard3 WonderCard { get = > new ( Large . Slice ( WonderCardOffset , Japanese ? WonderCard3 . SIZE_JAP : WonderCard3 . SIZE ) ) ; set = > SetData ( Large . AsSpan ( WonderCardOffset ) , value . Data ) ; }
public WonderCard3Extra WonderCardExtra { get = > new ( Large . Slice ( WonderCardExtraOffset , WonderCard3Extra . SIZE ) ) ; set = > SetData ( Large . AsSpan ( WonderCardExtraOffset ) , value . Data ) ; }
2022-06-18 18:04:24 +00:00
// 0x338: 4 easy chat words
// 0x340: news MENewsJisanStruct
// 0x344: uint[5], uint[5] tracking?
2021-03-28 01:58:51 +00:00
2023-03-20 23:26:23 +00:00
public override Gen3MysteryData MysteryData
2022-06-18 18:04:24 +00:00
{
2023-03-20 23:26:23 +00:00
get = > new MysteryEvent3 ( Large . Slice ( 0x3728 , MysteryEvent3 . SIZE ) ) ;
2023-03-26 06:14:50 +00:00
set = > SetData ( Large . AsSpan ( 0x3728 ) , value . Data ) ;
2022-06-18 18:04:24 +00:00
}
2021-03-28 01:58:51 +00:00
2023-03-26 06:14:50 +00:00
public RecordMixing3Gift RecordMixingGift { get = > new ( Large . Slice ( 0x3B14 , RecordMixing3Gift . SIZE ) ) ; set = > SetData ( Large . AsSpan ( 0x3B14 ) , value . Data ) ; }
2023-03-20 23:26:23 +00:00
2022-06-18 18:04:24 +00:00
protected override int SeenOffset3 = > 0x3B24 ;
2021-09-07 03:46:41 +00:00
2022-06-18 18:04:24 +00:00
private const int Walda = 0x3D70 ;
public ushort WaldaBackgroundColor { get = > ReadUInt16LittleEndian ( Large . AsSpan ( Walda + 0 ) ) ; set = > WriteUInt16LittleEndian ( Large . AsSpan ( Walda + 0 ) , value ) ; }
public ushort WaldaForegroundColor { get = > ReadUInt16LittleEndian ( Large . AsSpan ( Walda + 2 ) ) ; set = > WriteUInt16LittleEndian ( Large . AsSpan ( Walda + 2 ) , value ) ; }
public byte WaldaIconID { get = > Large [ Walda + 0x14 ] ; set = > Large [ Walda + 0x14 ] = value ; }
public byte WaldaPatternID { get = > Large [ Walda + 0x15 ] ; set = > Large [ Walda + 0x15 ] = value ; }
public bool WaldaUnlocked { get = > Large [ Walda + 0x16 ] ! = 0 ; set = > Large [ Walda + 0x16 ] = ( byte ) ( value ? 1 : 0 ) ; }
#endregion
2021-03-28 06:22:56 +00:00
2022-06-18 18:04:24 +00:00
private const uint EXTRADATA_SENTINEL = 0x0000B39D ;
private const int OFS_BV = 31 * 0x1000 ; // last sector of the save
public bool HasBattleVideo = > Data . Length > SaveUtil . SIZE_G3RAWHALF & & ReadUInt32LittleEndian ( Data . AsSpan ( OFS_BV ) ) = = EXTRADATA_SENTINEL ;
2021-03-28 06:22:56 +00:00
2022-06-18 18:04:24 +00:00
public BV3 BattleVideo
{
get = > ! HasBattleVideo ? new BV3 ( ) : new BV3 ( Data . Slice ( OFS_BV + 4 , BV3 . SIZE ) ) ;
2023-03-26 06:14:50 +00:00
set = > SetData ( Data . AsSpan ( OFS_BV + 4 , BV3 . SIZE ) , value . Data ) ;
2021-03-16 06:51:58 +00:00
}
}