using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Microsoft.Xna.Framework;
using Omegasis.SaveAnywhere.Framework.Models;
using StardewModdingAPI;
using StardewValley;
using StardewValley.Characters;
using StardewValley.Monsters;
namespace Omegasis.SaveAnywhere.Framework
{
/// Provides methods for saving and loading game data.
internal class SaveManager
{
/*********
** Properties
*********/
/// Simplifies access to game code.
private readonly IReflectionHelper Reflection;
/// A callback invoked when data is loaded.
private readonly Action OnLoaded;
/// SMAPI's APIs for this mod.
private readonly IModHelper Helper;
/// The full path to the player data file.
private string SavePath => Path.Combine(this.Helper.DirectoryPath, "data", $"{Constants.SaveFolderName}.json");
/// Whether we should save at the next opportunity.
private bool WaitingToSave;
/*********
** Public methods
*********/
/// Construct an instance.
/// SMAPI's APIs for this mod.
/// Simplifies access to game code.
/// A callback invoked when data is loaded.
public SaveManager(IModHelper helper, IReflectionHelper reflection, Action onLoaded)
{
this.Helper = helper;
this.Reflection = reflection;
this.OnLoaded = onLoaded;
}
/// Perform any required update logic.
public void Update()
{
// perform passive save
if (this.WaitingToSave && Game1.activeClickableMenu == null)
{
Game1.activeClickableMenu = new NewSaveGameMenu();
this.WaitingToSave = false;
}
}
/// Clear saved data.
public void ClearData()
{
Directory.Delete(this.SavePath, recursive: true);
this.RemoveLegacyDataForThisPlayer();
}
/// Initiate a game save.
public void BeginSaveData()
{
// save game data
Farm farm = Game1.getFarm();
if (farm.shippingBin.Any())
{
Game1.activeClickableMenu = new NewShippingMenu(farm.shippingBin, this.Reflection);
farm.shippingBin.Clear();
farm.lastItemShipped = null;
this.WaitingToSave = true;
}
else
Game1.activeClickableMenu = new NewSaveGameMenu();
// get data
PlayerData data = new PlayerData
{
Time = Game1.timeOfDay,
Characters = this.GetPositions().ToArray(),
IsCharacterSwimming = Game1.player.swimming
};
// save to disk
// ReSharper disable once PossibleNullReferenceException -- not applicable
Directory.CreateDirectory(new FileInfo(this.SavePath).Directory.FullName);
this.Helper.WriteJsonFile(this.SavePath, data);
// clear any legacy data (no longer needed as backup)1
this.RemoveLegacyDataForThisPlayer();
}
/// Load all game data.
public void LoadData()
{
// get data
PlayerData data = this.Helper.ReadJsonFile(this.SavePath);
if (data == null)
return;
// apply
Game1.timeOfDay = data.Time;
this.ResumeSwimming(data);
this.SetPositions(data.Characters);
this.OnLoaded?.Invoke();
}
///
/// Checks to see if the player was swimming when the game was saved and if so, resumes the swimming animation.
///
///
public void ResumeSwimming(PlayerData data)
{
try
{
if (data.IsCharacterSwimming == true)
{
Game1.player.changeIntoSwimsuit();
Game1.player.swimming = true;
}
}
catch (Exception err)
{
//Here to allow compatability with old save files.
}
}
/*********
** Private methods
*********/
/// Get the current character positions.
private IEnumerable GetPositions()
{
// player
{
var player = Game1.player;
string name = player.name;
string map = player.currentLocation.uniqueName; //Try to get a unique name for the location and if we can't we are going to default to the actual name of the map.
if (map == ""|| map==null) map = player.currentLocation.name; //This is used to account for maps that share the same name but have a unique ID such as Coops, Barns and Sheds.
Point tile = player.getTileLocationPoint();
int facingDirection = player.facingDirection;
yield return new CharacterData(CharacterType.Player, name, map, tile, facingDirection);
}
// NPCs (including horse and pets)
foreach (NPC npc in Utility.getAllCharacters())
{
CharacterType? type = this.GetCharacterType(npc);
if (type == null)
continue;
if (npc == null || npc.currentLocation == null) continue;
string name = npc.name;
string map = npc.currentLocation.name;
Point tile = npc.getTileLocationPoint();
int facingDirection = npc.facingDirection;
yield return new CharacterData(type.Value, name, map, tile, facingDirection);
}
}
/// Reset characters to their saved state.
/// The positions to set.
/// Returns whether any NPCs changed position.
private void SetPositions(CharacterData[] positions)
{
// player
{
CharacterData data = positions.FirstOrDefault(p => p.Type == CharacterType.Player && p.Name == Game1.player.name);
if (data != null)
{
Game1.player.previousLocationName = Game1.player.currentLocation.name;
Game1.locationAfterWarp = Game1.getLocationFromName(data.Map);
Game1.xLocationAfterWarp = data.X;
Game1.yLocationAfterWarp = data.Y;
Game1.facingDirectionAfterWarp = data.FacingDirection;
Game1.fadeScreenToBlack();
Game1.warpFarmer(data.Map, data.X, data.Y, false);
Game1.player.faceDirection(data.FacingDirection);
}
}
// NPCs (including horse and pets)
foreach (NPC npc in Utility.getAllCharacters())
{
// get NPC type
CharacterType? type = this.GetCharacterType(npc);
if (type == null)
continue;
// get saved data
CharacterData data = positions.FirstOrDefault(p => p.Type == type && p.Name == npc.name);
if (data == null)
continue;
// update NPC
Game1.warpCharacter(npc, data.Map, new Point(data.X, data.Y), false, true);
npc.faceDirection(data.FacingDirection);
}
}
/// Get the character type for an NPC.
/// The NPC to check.
private CharacterType? GetCharacterType(NPC npc)
{
if (npc is Monster)
return null;
if (npc is Horse)
return CharacterType.Horse;
if (npc is Pet)
return CharacterType.Pet;
return CharacterType.Villager;
}
/// Remove legacy save data for this player.
private void RemoveLegacyDataForThisPlayer()
{
DirectoryInfo dataDir = new DirectoryInfo(Path.Combine(this.Helper.DirectoryPath, "Save_Data"));
DirectoryInfo playerDir = new DirectoryInfo(Path.Combine(dataDir.FullName, Game1.player.name));
if (playerDir.Exists)
playerDir.Delete(recursive: true);
if (dataDir.Exists && !dataDir.EnumerateDirectories().Any())
dataDir.Delete(recursive: true);
}
}
}