using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using Microsoft.Xna.Framework.Graphics;
using StardewModdingAPI.AssemblyRewriters;
using StardewModdingAPI.AssemblyRewriters.Finders;
using StardewModdingAPI.AssemblyRewriters.Rewriters;
using StardewModdingAPI.AssemblyRewriters.Rewriters.Wrappers;
using StardewModdingAPI.Events;
using StardewModdingAPI.Framework;
using StardewValley;
using StardewValley.BellsAndWhistles;
using StardewValley.Objects;
using StardewValley.Projectiles;
namespace StardewModdingAPI
{
/// Contains SMAPI's constants and assumptions.
public static class Constants
{
/*********
** Properties
*********/
/// The directory path containing the current save's data (if a save is loaded).
private static string RawSavePath => Context.IsSaveLoaded ? Path.Combine(Constants.SavesPath, Constants.GetSaveFolderName()) : null;
/// Whether the directory containing the current save's data exists on disk.
private static bool SavePathReady => Context.IsSaveLoaded && Directory.Exists(Constants.RawSavePath);
/*********
** Accessors
*********/
/****
** Public
****/
/// SMAPI's current semantic version.
public static ISemanticVersion ApiVersion { get; } =
#if SMAPI_1_x
new SemanticVersion(1, 15, 1); // alpha-{DateTime.UtcNow:yyyyMMddHHmm}
#else
new SemanticVersion(2, 0, 0, $"alpha-{DateTime.UtcNow:yyyyMMddHHmm}");
#endif
/// The minimum supported version of Stardew Valley.
public static ISemanticVersion MinimumGameVersion { get; } = new SemanticVersion("1.2.30");
/// The maximum supported version of Stardew Valley.
public static ISemanticVersion MaximumGameVersion { get; } = null;
/// The path to the game folder.
public static string ExecutionPath { get; } = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
/// The directory path containing Stardew Valley's app data.
public static string DataPath { get; } = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), "StardewValley");
/// The directory path in which error logs should be stored.
public static string LogDir { get; } = Path.Combine(Constants.DataPath, "ErrorLogs");
/// The directory path where all saves are stored.
public static string SavesPath { get; } = Path.Combine(Constants.DataPath, "Saves");
/// The directory name containing the current save's data (if a save is loaded and the directory exists).
public static string SaveFolderName => Context.IsSaveLoaded ? Constants.GetSaveFolderName() : "";
/// The directory path containing the current save's data (if a save is loaded and the directory exists).
public static string CurrentSavePath => Constants.SavePathReady ? Path.Combine(Constants.SavesPath, Constants.GetSaveFolderName()) : "";
/****
** Internal
****/
/// The GitHub repository to check for updates.
internal const string GitHubRepository = "Pathoschild/SMAPI";
/// The file path for the SMAPI configuration file.
internal static string ApiConfigPath => Path.Combine(Constants.ExecutionPath, $"{typeof(Program).Assembly.GetName().Name}.config.json");
/// The file path to the log where the latest output should be saved.
internal static string DefaultLogPath => Path.Combine(Constants.LogDir, "SMAPI-latest.txt");
/// A copy of the log leading up to the previous fatal crash, if any.
internal static string FatalCrashLog => Path.Combine(Constants.LogDir, "SMAPI-crash.txt");
/// The file path which stores a fatal crash message for the next run.
internal static string FatalCrashMarker => Path.Combine(Constants.ExecutionPath, "StardewModdingAPI.crash.marker");
/// The full path to the folder containing mods.
internal static string ModPath { get; } = Path.Combine(Constants.ExecutionPath, "Mods");
/// The game's current semantic version.
internal static ISemanticVersion GameVersion { get; } = new GameVersion(Constants.GetGameVersion());
/// The target game platform.
internal static Platform TargetPlatform { get; } =
#if SMAPI_FOR_WINDOWS
Platform.Windows;
#else
Platform.Mono;
#endif
/*********
** Internal methods
*********/
/// Get metadata for mapping assemblies to the current platform.
/// The target game platform.
internal static PlatformAssemblyMap GetAssemblyMap(Platform targetPlatform)
{
// get assembly changes needed for platform
string[] removeAssemblyReferences;
Assembly[] targetAssemblies;
switch (targetPlatform)
{
case Platform.Mono:
removeAssemblyReferences = new[]
{
"Stardew Valley",
"Microsoft.Xna.Framework",
"Microsoft.Xna.Framework.Game",
"Microsoft.Xna.Framework.Graphics"
};
targetAssemblies = new[]
{
typeof(StardewValley.Game1).Assembly,
typeof(Microsoft.Xna.Framework.Vector2).Assembly
};
break;
case Platform.Windows:
removeAssemblyReferences = new[]
{
"StardewValley",
"MonoGame.Framework"
};
targetAssemblies = new[]
{
typeof(StardewValley.Game1).Assembly,
typeof(Microsoft.Xna.Framework.Vector2).Assembly,
typeof(Microsoft.Xna.Framework.Game).Assembly,
typeof(Microsoft.Xna.Framework.Graphics.SpriteBatch).Assembly
};
break;
default:
throw new InvalidOperationException($"Unknown target platform '{targetPlatform}'.");
}
return new PlatformAssemblyMap(targetPlatform, removeAssemblyReferences, targetAssemblies);
}
/// Get rewriters which detect or fix incompatible CIL instructions in mod assemblies.
internal static IEnumerable GetRewriters()
{
return new IInstructionRewriter[]
{
/****
** Finders throw an exception when incompatible code is found.
****/
// changes in Stardew Valley 1.2 (with no rewriters)
new FieldFinder("StardewValley.Item", "set_Name"),
// APIs removed in SMAPI 1.9
new TypeFinder("StardewModdingAPI.Advanced.ConfigFile"),
new TypeFinder("StardewModdingAPI.Advanced.IConfigFile"),
new TypeFinder("StardewModdingAPI.Entities.SPlayer"),
new TypeFinder("StardewModdingAPI.Extensions"),
new TypeFinder("StardewModdingAPI.Inheritance.SGame"),
new TypeFinder("StardewModdingAPI.Inheritance.SObject"),
new TypeFinder("StardewModdingAPI.LogWriter"),
new TypeFinder("StardewModdingAPI.Manifest"),
new TypeFinder("StardewModdingAPI.Version"),
new EventFinder("StardewModdingAPI.Events.GraphicsEvents", "DrawDebug"),
new EventFinder("StardewModdingAPI.Events.GraphicsEvents", "DrawTick"),
new EventFinder("StardewModdingAPI.Events.GraphicsEvents", "OnPostRenderHudEventNoCheck"),
new EventFinder("StardewModdingAPI.Events.GraphicsEvents", "OnPostRenderGuiEventNoCheck"),
new EventFinder("StardewModdingAPI.Events.GraphicsEvents", "OnPreRenderHudEventNoCheck"),
new EventFinder("StardewModdingAPI.Events.GraphicsEvents", "OnPreRenderGuiEventNoCheck"),
// APIs removed in SMAPI 2.0
#if !SMAPI_1_x
new TypeFinder("StardewModdingAPI.Command"),
new TypeFinder("StardewModdingAPI.Config"),
new TypeFinder("StardewModdingAPI.Log"),
new EventFinder("StardewModdingAPI.Events.GameEvents", "Initialize"),
new EventFinder("StardewModdingAPI.Events.GameEvents", "LoadContent"),
new EventFinder("StardewModdingAPI.Events.GameEvents", "GameLoaded"),
new EventFinder("StardewModdingAPI.Events.GameEvents", "FirstUpdateTick"),
new EventFinder("StardewModdingAPI.Events.PlayerEvents", "LoadedGame"),
new EventFinder("StardewModdingAPI.Events.PlayerEvents", "FarmerChanged"),
new EventFinder("StardewModdingAPI.Events.TimeEvents", "DayOfMonthChanged"),
new EventFinder("StardewModdingAPI.Events.TimeEvents", "YearOfGameChanged"),
new EventFinder("StardewModdingAPI.Events.TimeEvents", "SeasonOfYearChanged"),
new EventFinder("StardewModdingAPI.Events.TimeEvents", "OnNewDay"),
new TypeFinder("StardewModdingAPI.Events.EventArgsCommand"),
new TypeFinder("StardewModdingAPI.Events.EventArgsFarmerChanged"),
new TypeFinder("StardewModdingAPI.Events.EventArgsLoadedGameChanged"),
new TypeFinder("StardewModdingAPI.Events.EventArgsNewDay"),
new TypeFinder("StardewModdingAPI.Events.EventArgsStringChanged"),
new PropertyFinder("StardewModdingAPI.Mod", "PathOnDisk"),
new PropertyFinder("StardewModdingAPI.Mod", "BaseConfigPath"),
new PropertyFinder("StardewModdingAPI.Mod", "PerSaveConfigFolder"),
new PropertyFinder("StardewModdingAPI.Mod", "PerSaveConfigPath"),
#endif
/****
** Rewriters change CIL as needed to fix incompatible code
****/
// crossplatform
new MethodParentRewriter(typeof(SpriteBatch), typeof(SpriteBatchWrapper), onlyIfPlatformChanged: true),
// Stardew Valley 1.2
new FieldToPropertyRewriter(typeof(Game1), nameof(Game1.activeClickableMenu)),
new FieldToPropertyRewriter(typeof(Game1), nameof(Game1.currentMinigame)),
new FieldToPropertyRewriter(typeof(Game1), nameof(Game1.gameMode)),
new FieldToPropertyRewriter(typeof(Game1), nameof(Game1.player)),
new FieldReplaceRewriter(typeof(Game1), "borderFont", nameof(Game1.smallFont)),
new FieldReplaceRewriter(typeof(Game1), "smoothFont", nameof(Game1.smallFont)),
// SMAPI 1.9
new TypeReferenceRewriter("StardewModdingAPI.Inheritance.ItemStackChange", typeof(ItemStackChange))
};
}
/// Get the game's static asset setters by (non-normalised) asset name.
/// Derived from .
internal static IDictionary> GetCoreAssetSetters()
{
return new Dictionary>
{
// from Game1.loadContent
["LooseSprites\\daybg"] = (content, key) => Game1.daybg = content.Load(key),
["LooseSprites\\daybg"] = (content, key) => Game1.daybg = content.Load(key),
["LooseSprites\\nightbg"] = (content, key) => Game1.nightbg = content.Load(key),
["Maps\\MenuTiles"] = (content, key) => Game1.menuTexture = content.Load(key),
["LooseSprites\\Lighting\\lantern"] = (content, key) => Game1.lantern = content.Load(key),
["LooseSprites\\Lighting\\windowLight"] = (content, key) => Game1.windowLight = content.Load(key),
["LooseSprites\\Lighting\\sconceLight"] = (content, key) => Game1.sconceLight = content.Load(key),
["LooseSprites\\Lighting\\greenLight"] = (content, key) => Game1.cauldronLight = content.Load(key),
["LooseSprites\\Lighting\\indoorWindowLight"] = (content, key) => Game1.indoorWindowLight = content.Load(key),
["LooseSprites\\shadow"] = (content, key) => Game1.shadowTexture = content.Load(key),
["LooseSprites\\Cursors"] = (content, key) => Game1.mouseCursors = content.Load(key),
["LooseSprites\\ControllerMaps"] = (content, key) => Game1.controllerMaps = content.Load(key),
["TileSheets\\animations"] = (content, key) => Game1.animations = content.Load(key),
["Data\\Achievements"] = (content, key) => Game1.achievements = content.Load>(key),
["Data\\NPCGiftTastes"] = (content, key) => Game1.NPCGiftTastes = content.Load>(key),
["Fonts\\SpriteFont1"] = (content, key) => Game1.dialogueFont = content.Load(key),
["Fonts\\SmallFont"] = (content, key) => Game1.smallFont = content.Load(key),
["Fonts\\tinyFont"] = (content, key) => Game1.tinyFont = content.Load(key),
["Fonts\\tinyFontBorder"] = (content, key) => Game1.tinyFontBorder = content.Load(key),
["Maps\\springobjects"] = (content, key) => Game1.objectSpriteSheet = content.Load(key),
["TileSheets\\crops"] = (content, key) => Game1.cropSpriteSheet = content.Load(key),
["TileSheets\\emotes"] = (content, key) => Game1.emoteSpriteSheet = content.Load(key),
["TileSheets\\debris"] = (content, key) => Game1.debrisSpriteSheet = content.Load(key),
["TileSheets\\Craftables"] = (content, key) => Game1.bigCraftableSpriteSheet = content.Load(key),
["TileSheets\\rain"] = (content, key) => Game1.rainTexture = content.Load(key),
["TileSheets\\BuffsIcons"] = (content, key) => Game1.buffsIcons = content.Load(key),
["Data\\ObjectInformation"] = (content, key) => Game1.objectInformation = content.Load>(key),
["Data\\BigCraftablesInformation"] = (content, key) => Game1.bigCraftablesInformation = content.Load>(key),
["Characters\\Farmer\\hairstyles"] = (content, key) => FarmerRenderer.hairStylesTexture = content.Load(key),
["Characters\\Farmer\\shirts"] = (content, key) => FarmerRenderer.shirtsTexture = content.Load(key),
["Characters\\Farmer\\hats"] = (content, key) => FarmerRenderer.hatsTexture = content.Load(key),
["Characters\\Farmer\\accessories"] = (content, key) => FarmerRenderer.accessoriesTexture = content.Load(key),
["TileSheets\\furniture"] = (content, key) => Furniture.furnitureTexture = content.Load(key),
["LooseSprites\\font_bold"] = (content, key) => SpriteText.spriteTexture = content.Load(key),
["LooseSprites\\font_colored"] = (content, key) => SpriteText.coloredTexture = content.Load(key),
["TileSheets\\weapons"] = (content, key) => Tool.weaponsTexture = content.Load(key),
["TileSheets\\Projectiles"] = (content, key) => Projectile.projectileSheet = content.Load(key),
// from Farmer constructor
["Characters\\Farmer\\farmer_base"] = (content, key) =>
{
if (Game1.player != null && Game1.player.isMale)
Game1.player.FarmerRenderer = new FarmerRenderer(content.Load(key));
},
["Characters\\Farmer\\farmer_girl_base"] = (content, key) =>
{
if (Game1.player != null && !Game1.player.isMale)
Game1.player.FarmerRenderer = new FarmerRenderer(content.Load(key));
}
};
}
/*********
** Private methods
*********/
/// Get the name of a save directory for the current player.
private static string GetSaveFolderName()
{
string prefix = new string(Game1.player.name.Where(char.IsLetterOrDigit).ToArray());
return $"{prefix}_{Game1.uniqueIDForThisGame}";
}
/// Get the game's current version string.
private static string GetGameVersion()
{
// we need reflection because it's a constant, so SMAPI's references to it are inlined at compile-time
FieldInfo field = typeof(Game1).GetField(nameof(Game1.version), BindingFlags.Public | BindingFlags.Static);
if (field == null)
throw new InvalidOperationException($"The {nameof(Game1)}.{nameof(Game1.version)} field could not be found.");
return (string)field.GetValue(null);
}
}
}