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); } } }