summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--docs/release-notes.md1
-rw-r--r--src/SMAPI.Tests/Core/ModResolverTests.cs31
-rw-r--r--src/SMAPI/Framework/IModMetadata.cs2
-rw-r--r--src/SMAPI/Framework/ModData/ModDataField.cs (renamed from src/SMAPI/Framework/Models/ModDataField.cs)2
-rw-r--r--src/SMAPI/Framework/ModData/ModDataFieldKey.cs (renamed from src/SMAPI/Framework/Models/ModDataFieldKey.cs)2
-rw-r--r--src/SMAPI/Framework/ModData/ModDataRecord.cs (renamed from src/SMAPI/Framework/Models/ModDataRecord.cs)104
-rw-r--r--src/SMAPI/Framework/ModData/ModDatabase.cs168
-rw-r--r--src/SMAPI/Framework/ModData/ModStatus.cs (renamed from src/SMAPI/Framework/Models/ModStatus.cs)2
-rw-r--r--src/SMAPI/Framework/ModData/ParsedModDataRecord.cs (renamed from src/SMAPI/Framework/Models/ParsedModDataRecord.cs)5
-rw-r--r--src/SMAPI/Framework/ModLoading/ModMetadata.cs2
-rw-r--r--src/SMAPI/Framework/ModLoading/ModResolver.cs48
-rw-r--r--src/SMAPI/Framework/Models/SConfig.cs5
-rw-r--r--src/SMAPI/Program.cs8
-rw-r--r--src/SMAPI/StardewModdingAPI.config.json1130
-rw-r--r--src/SMAPI/StardewModdingAPI.csproj11
15 files changed, 803 insertions, 718 deletions
diff --git a/docs/release-notes.md b/docs/release-notes.md
index 6c4bdf94..c4c269eb 100644
--- a/docs/release-notes.md
+++ b/docs/release-notes.md
@@ -1,6 +1,7 @@
# Release notes
## 2.5
* For players:
+ * Dependency errors will now show the name of the missing mod, instead of its ID.
* Fixed mod crashes being logged under `[SMAPI]` instead of the mod name.
* Updated compatibility list and enabled update checks for more older mods.
diff --git a/src/SMAPI.Tests/Core/ModResolverTests.cs b/src/SMAPI.Tests/Core/ModResolverTests.cs
index 7c1efe53..900a6c4f 100644
--- a/src/SMAPI.Tests/Core/ModResolverTests.cs
+++ b/src/SMAPI.Tests/Core/ModResolverTests.cs
@@ -6,6 +6,7 @@ using Moq;
using Newtonsoft.Json;
using NUnit.Framework;
using StardewModdingAPI.Framework;
+using StardewModdingAPI.Framework.ModData;
using StardewModdingAPI.Framework.Models;
using StardewModdingAPI.Framework.ModLoading;
using StardewModdingAPI.Framework.Serialisation;
@@ -30,7 +31,7 @@ namespace StardewModdingAPI.Tests.Core
Directory.CreateDirectory(rootFolder);
// act
- IModMetadata[] mods = new ModResolver().ReadManifests(rootFolder, new JsonHelper(), new ModDataRecord[0]).ToArray();
+ IModMetadata[] mods = new ModResolver().ReadManifests(rootFolder, new JsonHelper(), new ModDatabase()).ToArray();
// assert
Assert.AreEqual(0, mods.Length, 0, $"Expected to find zero manifests, found {mods.Length} instead.");
@@ -45,7 +46,7 @@ namespace StardewModdingAPI.Tests.Core
Directory.CreateDirectory(modFolder);
// act
- IModMetadata[] mods = new ModResolver().ReadManifests(rootFolder, new JsonHelper(), new ModDataRecord[0]).ToArray();
+ IModMetadata[] mods = new ModResolver().ReadManifests(rootFolder, new JsonHelper(), new ModDatabase()).ToArray();
IModMetadata mod = mods.FirstOrDefault();
// assert
@@ -84,7 +85,7 @@ namespace StardewModdingAPI.Tests.Core
File.WriteAllText(filename, JsonConvert.SerializeObject(original));
// act
- IModMetadata[] mods = new ModResolver().ReadManifests(rootFolder, new JsonHelper(), new ModDataRecord[0]).ToArray();
+ IModMetadata[] mods = new ModResolver().ReadManifests(rootFolder, new JsonHelper(), new ModDatabase()).ToArray();
IModMetadata mod = mods.FirstOrDefault();
// assert
@@ -233,7 +234,7 @@ namespace StardewModdingAPI.Tests.Core
public void ProcessDependencies_NoMods_DoesNothing()
{
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new IModMetadata[0]).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new IModMetadata[0], new ModDatabase()).ToArray();
// assert
Assert.AreEqual(0, mods.Length, 0, "Expected to get an empty list of mods.");
@@ -249,7 +250,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modC = this.GetMetadata("Mod C");
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modA.Object, modB.Object, modC.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modA.Object, modB.Object, modC.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(3, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -266,7 +267,7 @@ namespace StardewModdingAPI.Tests.Core
mock.Setup(p => p.Status).Returns(ModMetadataStatus.Failed);
// act
- new ModResolver().ProcessDependencies(new[] { mock.Object });
+ new ModResolver().ProcessDependencies(new[] { mock.Object }, new ModDatabase());
// assert
mock.VerifyGet(p => p.Status, Times.Once, "The validation did not check the manifest status.");
@@ -285,7 +286,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modC = this.GetMetadata("Mod C", dependencies: new[] { "Mod A", "Mod B" });
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(3, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -305,7 +306,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modD = this.GetMetadata("Mod D", dependencies: new[] { "Mod C" });
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(4, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -331,7 +332,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modF = this.GetMetadata("Mod F", dependencies: new[] { "Mod C", "Mod E" });
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object, modF.Object, modE.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object, modF.Object, modE.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(6, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -358,7 +359,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modE = this.GetMetadata("Mod E", dependencies: new[] { "Mod C" }, allowStatusChange: true);
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object, modE.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object, modE.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(5, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -382,7 +383,7 @@ namespace StardewModdingAPI.Tests.Core
modD.Setup(p => p.Status).Returns(ModMetadataStatus.Failed);
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modC.Object, modA.Object, modB.Object, modD.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(4, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -401,7 +402,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modB = this.GetMetadata(this.GetManifest("Mod B", "1.0", new ManifestDependency("Mod A", "1.1")), allowStatusChange: true);
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modA.Object, modB.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modA.Object, modB.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(2, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -417,7 +418,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modB = this.GetMetadata(this.GetManifest("Mod B", "1.0", new ManifestDependency("Mod A", "1.0-beta")), allowStatusChange: false);
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modA.Object, modB.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modA.Object, modB.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(2, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -434,7 +435,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modB = this.GetMetadata(this.GetManifest("Mod B", "1.0", new ManifestDependency("Mod A", "1.0", required: false)), allowStatusChange: false);
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modB.Object, modA.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modB.Object, modA.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(2, mods.Length, 0, "Expected to get the same number of mods input.");
@@ -450,7 +451,7 @@ namespace StardewModdingAPI.Tests.Core
Mock<IModMetadata> modB = this.GetMetadata(this.GetManifest("Mod B", "1.0", new ManifestDependency("Mod A", "1.0", required: false)), allowStatusChange: false);
// act
- IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modB.Object }).ToArray();
+ IModMetadata[] mods = new ModResolver().ProcessDependencies(new[] { modB.Object }, new ModDatabase()).ToArray();
// assert
Assert.AreEqual(1, mods.Length, 0, "Expected to get the same number of mods input.");
diff --git a/src/SMAPI/Framework/IModMetadata.cs b/src/SMAPI/Framework/IModMetadata.cs
index 41484567..a91b0a5b 100644
--- a/src/SMAPI/Framework/IModMetadata.cs
+++ b/src/SMAPI/Framework/IModMetadata.cs
@@ -1,4 +1,4 @@
-using StardewModdingAPI.Framework.Models;
+using StardewModdingAPI.Framework.ModData;
using StardewModdingAPI.Framework.ModLoading;
namespace StardewModdingAPI.Framework
diff --git a/src/SMAPI/Framework/Models/ModDataField.cs b/src/SMAPI/Framework/ModData/ModDataField.cs
index 0812b39b..fa8dd6d0 100644
--- a/src/SMAPI/Framework/Models/ModDataField.cs
+++ b/src/SMAPI/Framework/ModData/ModDataField.cs
@@ -1,6 +1,6 @@
using System.Linq;
-namespace StardewModdingAPI.Framework.Models
+namespace StardewModdingAPI.Framework.ModData
{
/// <summary>A versioned mod metadata field.</summary>
internal class ModDataField
diff --git a/src/SMAPI/Framework/Models/ModDataFieldKey.cs b/src/SMAPI/Framework/ModData/ModDataFieldKey.cs
index 5767afc9..f68f575c 100644
--- a/src/SMAPI/Framework/Models/ModDataFieldKey.cs
+++ b/src/SMAPI/Framework/ModData/ModDataFieldKey.cs
@@ -1,4 +1,4 @@
-namespace StardewModdingAPI.Framework.Models
+namespace StardewModdingAPI.Framework.ModData
{
/// <summary>The valid field keys.</summary>
public enum ModDataFieldKey
diff --git a/src/SMAPI/Framework/Models/ModDataRecord.cs b/src/SMAPI/Framework/ModData/ModDataRecord.cs
index 2c26741c..79a954f7 100644
--- a/src/SMAPI/Framework/Models/ModDataRecord.cs
+++ b/src/SMAPI/Framework/ModData/ModDataRecord.cs
@@ -1,12 +1,11 @@
using System;
using System.Collections.Generic;
-using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Runtime.Serialization;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
-namespace StardewModdingAPI.Framework.Models
+namespace StardewModdingAPI.Framework.ModData
{
/// <summary>Raw mod metadata from SMAPI's internal mod list.</summary>
internal class ModDataRecord
@@ -63,47 +62,6 @@ namespace StardewModdingAPI.Framework.Models
/*********
** Public methods
*********/
- /// <summary>Get whether the manifest matches the <see cref="FormerIDs"/> field.</summary>
- /// <param name="manifest">The mod manifest to check.</param>
- public bool Matches(IManifest manifest)
- {
- // try main ID
- if (this.ID != null && this.ID.Equals(manifest.UniqueID, StringComparison.InvariantCultureIgnoreCase))
- return true;
-
- // try former IDs
- if (this.FormerIDs != null)
- {
- foreach (string part in this.FormerIDs.Split('|'))
- {
- // packed field snapshot
- if (part.StartsWith("{"))
- {
- FieldSnapshot snapshot = JsonConvert.DeserializeObject<FieldSnapshot>(part);
- bool isMatch =
- (snapshot.ID == null || snapshot.ID.Equals(manifest.UniqueID, StringComparison.InvariantCultureIgnoreCase))
- && (snapshot.EntryDll == null || snapshot.EntryDll.Equals(manifest.EntryDll, StringComparison.InvariantCultureIgnoreCase))
- && (
- snapshot.Author == null
- || snapshot.Author.Equals(manifest.Author, StringComparison.InvariantCultureIgnoreCase)
- || (manifest.ExtraFields.ContainsKey("Authour") && snapshot.Author.Equals(manifest.ExtraFields["Authour"].ToString(), StringComparison.InvariantCultureIgnoreCase))
- )
- && (snapshot.Name == null || snapshot.Name.Equals(manifest.Name, StringComparison.InvariantCultureIgnoreCase));
-
- if (isMatch)
- return true;
- }
-
- // plain ID
- else if (part.Equals(manifest.UniqueID, StringComparison.InvariantCultureIgnoreCase))
- return true;
- }
- }
-
- // no match
- return false;
- }
-
/// <summary>Get a parsed representation of the <see cref="Fields"/>.</summary>
public IEnumerable<ModDataField> GetFields()
{
@@ -146,41 +104,6 @@ namespace StardewModdingAPI.Framework.Models
}
}
- /// <summary>Get a parsed representation of the <see cref="Fields"/> which match a given manifest.</summary>
- /// <param name="manifest">The manifest to match.</param>
- public ParsedModDataRecord ParseFieldsFor(IManifest manifest)
- {
- ParsedModDataRecord parsed = new ParsedModDataRecord { DataRecord = this };
- foreach (ModDataField field in this.GetFields().Where(field => field.IsMatch(manifest)))
- {
- switch (field.Key)
- {
- // update key
- case ModDataFieldKey.UpdateKey:
- parsed.UpdateKey = field.Value;
- break;
-
- // alternative URL
- case ModDataFieldKey.AlternativeUrl:
- parsed.AlternativeUrl = field.Value;
- break;
-
- // status
- case ModDataFieldKey.Status:
- parsed.Status = (ModStatus)Enum.Parse(typeof(ModStatus), field.Value, ignoreCase: true);
- parsed.StatusUpperVersion = field.UpperVersion;
- break;
-
- // status reason phrase
- case ModDataFieldKey.StatusReasonPhrase:
- parsed.StatusReasonPhrase = field.Value;
- break;
- }
- }
-
- return parsed;
- }
-
/// <summary>Get a semantic local version for update checks.</summary>
/// <param name="version">The remote version to normalise.</param>
public string GetLocalVersionForUpdateChecks(string version)
@@ -214,30 +137,5 @@ namespace StardewModdingAPI.Framework.Models
this.ExtensionData = null;
}
}
-
-
- /*********
- ** Private models
- *********/
- /// <summary>A unique set of fields which identifies the mod.</summary>
- [SuppressMessage("ReSharper", "ClassNeverInstantiated.Local", Justification = "Used via JSON deserialisation.")]
- [SuppressMessage("ReSharper", "UnusedAutoPropertyAccessor.Local", Justification = "Used via JSON deserialisation.")]
- private class FieldSnapshot
- {
- /*********
- ** Accessors
- *********/
- /// <summary>The unique mod ID (or <c>null</c> to ignore it).</summary>
- public string ID { get; set; }
-
- /// <summary>The entry DLL (or <c>null</c> to ignore it).</summary>
- public string EntryDll { get; set; }
-
- /// <summary>The mod name (or <c>null</c> to ignore it).</summary>
- public string Name { get; set; }
-
- /// <summary>The author name (or <c>null</c> to ignore it).</summary>
- public string Author { get; set; }
- }
}
}
diff --git a/src/SMAPI/Framework/ModData/ModDatabase.cs b/src/SMAPI/Framework/ModData/ModDatabase.cs
new file mode 100644
index 00000000..af067f8f
--- /dev/null
+++ b/src/SMAPI/Framework/ModData/ModDatabase.cs
@@ -0,0 +1,168 @@
+using System;
+using System.Collections.Generic;
+using System.Diagnostics.CodeAnalysis;
+using System.Linq;
+using Newtonsoft.Json;
+
+namespace StardewModdingAPI.Framework.ModData
+{
+ /// <summary>Handles access to SMAPI's internal mod metadata list.</summary>
+ internal class ModDatabase
+ {
+ /*********
+ ** Properties
+ *********/
+ /// <summary>The underlying mod data records indexed by default display name.</summary>
+ private readonly IDictionary<string, ModDataRecord> Records;
+
+
+ /*********
+ ** Public methods
+ *********/
+ /// <summary>Construct an empty instance.</summary>
+ public ModDatabase()
+ : this(new Dictionary<string, ModDataRecord>()) { }
+
+ /// <summary>Construct an instance.</summary>
+ /// <param name="records">The underlying mod data records indexed by default display name.</param>
+ public ModDatabase(IDictionary<string, ModDataRecord> records)
+ {
+ this.Records = records;
+ }
+
+ /// <summary>Get a parsed representation of the <see cref="ModDataRecord.Fields"/> which match a given manifest.</summary>
+ /// <param name="manifest">The manifest to match.</param>
+ public ParsedModDataRecord GetParsed(IManifest manifest)
+ {
+ // get raw record
+ if (!this.TryGetRaw(manifest, out string displayName, out ModDataRecord record))
+ return null;
+
+ // parse fields
+ ParsedModDataRecord parsed = new ParsedModDataRecord { DisplayName = displayName, DataRecord = record };
+ foreach (ModDataField field in record.GetFields().Where(field => field.IsMatch(manifest)))
+ {
+ switch (field.Key)
+ {
+ // update key
+ case ModDataFieldKey.UpdateKey:
+ parsed.UpdateKey = field.Value;
+ break;
+
+ // alternative URL
+ case ModDataFieldKey.AlternativeUrl:
+ parsed.AlternativeUrl = field.Value;
+ break;
+
+ // status
+ case ModDataFieldKey.Status:
+ parsed.Status = (ModStatus)Enum.Parse(typeof(ModStatus), field.Value, ignoreCase: true);
+ parsed.StatusUpperVersion = field.UpperVersion;
+ break;
+
+ // status reason phrase
+ case ModDataFieldKey.StatusReasonPhrase:
+ parsed.StatusReasonPhrase = field.Value;
+ break;
+ }
+ }
+
+ return parsed;
+ }
+
+ /// <summary>Get the display name for a given mod ID (if available).</summary>
+ /// <param name="id">The unique mod ID.</param>
+ public string GetDisplayNameFor(string id)
+ {
+ foreach (var entry in this.Records)
+ {
+ if (entry.Value.ID != null && entry.Value.ID.Equals(id, StringComparison.InvariantCultureIgnoreCase))
+ return entry.Key;
+ }
+
+ return null;
+ }
+
+
+ /*********
+ ** Private models
+ *********/
+ /// <summary>Get the data record matching a given manifest.</summary>
+ /// <param name="manifest">The mod manifest.</param>
+ /// <param name="displayName">The mod's default display name.</param>
+ /// <param name="record">The raw mod record.</param>
+ private bool TryGetRaw(IManifest manifest, out string displayName, out ModDataRecord record)
+ {
+ if (manifest != null)
+ {
+ foreach (var entry in this.Records)
+ {
+ displayName = entry.Key;
+ record = entry.Value;
+
+ // try main ID
+ if (record.ID != null && record.ID.Equals(manifest.UniqueID, StringComparison.InvariantCultureIgnoreCase))
+ return true;
+
+ // try former IDs
+ if (record.FormerIDs != null)
+ {
+ foreach (string part in record.FormerIDs.Split('|'))
+ {
+ // packed field snapshot
+ if (part.StartsWith("{"))
+ {
+ FieldSnapshot snapshot = JsonConvert.DeserializeObject<FieldSnapshot>(part);
+ bool isMatch =
+ (snapshot.ID == null || snapshot.ID.Equals(manifest.UniqueID, StringComparison.InvariantCultureIgnoreCase))
+ && (snapshot.EntryDll == null || snapshot.EntryDll.Equals(manifest.EntryDll, StringComparison.InvariantCultureIgnoreCase))
+ && (
+ snapshot.Author == null
+ || snapshot.Author.Equals(manifest.Author, StringComparison.InvariantCultureIgnoreCase)
+ || (manifest.ExtraFields.ContainsKey("Authour") && snapshot.Author.Equals(manifest.ExtraFields["Authour"].ToString(), StringComparison.InvariantCultureIgnoreCase))
+ )
+ && (snapshot.Name == null || snapshot.Name.Equals(manifest.Name, StringComparison.InvariantCultureIgnoreCase));
+
+ if (isMatch)
+ return true;
+ }
+
+ // plain ID
+ else if (part.Equals(manifest.UniqueID, StringComparison.InvariantCultureIgnoreCase))
+ return true;
+ }
+ }
+ }
+ }
+
+ displayName = null;
+ record = null;
+ return false;
+ }
+
+
+ /*********
+ ** Private models
+ *********/
+ /// <summary>A unique set of fields which identifies the mod.</summary>
+ [SuppressMessage("ReSharper", "ClassNeverInstantiated.Local", Justification = "Used via JSON deserialisation.")]
+ [SuppressMessage("ReSharper", "UnusedAutoPropertyAccessor.Local", Justification = "Used via JSON deserialisation.")]
+ private class FieldSnapshot
+ {
+ /*********
+ ** Accessors
+ *********/
+ /// <summary>The unique mod ID (or <c>null</c> to ignore it).</summary>
+ public string ID { get; set; }
+
+ /// <summary>The entry DLL (or <c>null</c> to ignore it).</summary>
+ public string EntryDll { get; set; }
+
+ /// <summary>The mod name (or <c>null</c> to ignore it).</summary>
+ public string Name { get; set; }
+
+ /// <summary>The author name (or <c>null</c> to ignore it).</summary>
+ public string Author { get; set; }
+ }
+ }
+}
diff --git a/src/SMAPI/Framework/Models/ModStatus.cs b/src/SMAPI/Framework/ModData/ModStatus.cs
index 343ccb7e..0e1d94d4 100644
--- a/src/SMAPI/Framework/Models/ModStatus.cs
+++ b/src/SMAPI/Framework/ModData/ModStatus.cs
@@ -1,4 +1,4 @@
-namespace StardewModdingAPI.Framework.Models
+namespace StardewModdingAPI.Framework.ModData
{
/// <summary>Indicates how SMAPI should treat a mod.</summary>
internal enum ModStatus
diff --git a/src/SMAPI/Framework/Models/ParsedModDataRecord.cs b/src/SMAPI/Framework/ModData/ParsedModDataRecord.cs
index 0abc7b89..5a6561a7 100644
--- a/src/SMAPI/Framework/Models/ParsedModDataRecord.cs
+++ b/src/SMAPI/Framework/ModData/ParsedModDataRecord.cs
@@ -1,4 +1,4 @@
-namespace StardewModdingAPI.Framework.Models
+namespace StardewModdingAPI.Framework.ModData
{
/// <summary>A parsed representation of the fields from a <see cref="ModDataRecord"/> for a specific manifest.</summary>
internal class ParsedModDataRecord
@@ -9,6 +9,9 @@ namespace StardewModdingAPI.Framework.Models
/// <summary>The underlying data record.</summary>
public ModDataRecord DataRecord { get; set; }
+ /// <summary>The default mod name to display when the name isn't available (e.g. during dependency checks).</summary>
+ public string DisplayName { get; set; }
+
/// <summary>The update key to apply.</summary>
public string UpdateKey { get; set; }
diff --git a/src/SMAPI/Framework/ModLoading/ModMetadata.cs b/src/SMAPI/Framework/ModLoading/ModMetadata.cs
index 1a71920e..29bb6617 100644
--- a/src/SMAPI/Framework/ModLoading/ModMetadata.cs
+++ b/src/SMAPI/Framework/ModLoading/ModMetadata.cs
@@ -1,4 +1,4 @@
-using StardewModdingAPI.Framework.Models;
+using StardewModdingAPI.Framework.ModData;
namespace StardewModdingAPI.Framework.ModLoading
{
diff --git a/src/SMAPI/Framework/ModLoading/ModResolver.cs b/src/SMAPI/Framework/ModLoading/ModResolver.cs
index 6671e880..09a9299e 100644
--- a/src/SMAPI/Framework/ModLoading/ModResolver.cs
+++ b/src/SMAPI/Framework/ModLoading/ModResolver.cs
@@ -3,6 +3,7 @@ using System.Collections.Generic;
using System.IO;
using System.Linq;
using StardewModdingAPI.Framework.Exceptions;
+using StardewModdingAPI.Framework.ModData;
using StardewModdingAPI.Framework.Models;
using StardewModdingAPI.Framework.Serialisation;
@@ -17,12 +18,10 @@ namespace StardewModdingAPI.Framework.ModLoading
/// <summary>Get manifest metadata for each folder in the given root path.</summary>
/// <param name="rootPath">The root path to search for mods.</param>
/// <param name="jsonHelper">The JSON helper with which to read manifests.</param>
- /// <param name="dataRecords">Metadata about mods from SMAPI's internal data.</param>
+ /// <param name="modDatabase">Handles access to SMAPI's internal mod metadata list.</param>
/// <returns>Returns the manifests by relative folder.</returns>
- public IEnumerable<IModMetadata> ReadManifests(string rootPath, JsonHelper jsonHelper, IEnumerable<ModDataRecord> dataRecords)
+ public IEnumerable<IModMetadata> ReadManifests(string rootPath, JsonHelper jsonHelper, ModDatabase modDatabase)
{
- dataRecords = dataRecords.ToArray();
-
foreach (DirectoryInfo modDir in this.GetModFolders(rootPath))
{
// read file
@@ -54,22 +53,19 @@ namespace StardewModdingAPI.Framework.ModLoading
}
// parse internal data record (if any)
- ParsedModDataRecord dataRecord = null;
- if (manifest != null)
- {
- ModDataRecord rawDataRecord = dataRecords.FirstOrDefault(p => p.Matches(manifest));
- if (rawDataRecord != null)
- dataRecord = rawDataRecord.ParseFieldsFor(manifest);
- }
+ ParsedModDataRecord dataRecord = modDatabase.GetParsed(manifest);
+
+ // get display name
+ string displayName = manifest?.Name;
+ if (string.IsNullOrWhiteSpace(displayName))
+ displayName = dataRecord?.DisplayName;
+ if (string.IsNullOrWhiteSpace(displayName))
+ displayName = modDir.FullName.Replace(rootPath, "").Trim('/', '\\');
// build metadata
- string displayName = !string.IsNullOrWhiteSpace(manifest?.Name)
- ? manifest.Name
- : modDir.FullName.Replace(rootPath, "").Trim('/', '\\');
ModMetadataStatus status = error == null
? ModMetadataStatus.Found
: ModMetadataStatus.Failed;
-
yield return new ModMetadata(displayName, modDir.FullName, manifest, dataRecord).SetStatus(status, error);
}
}
@@ -193,7 +189,8 @@ namespace StardewModdingAPI.Framework.ModLoading
/// <summary>Sort the given mods by the order they should be loaded.</summary>
/// <param name="mods">The mods to process.</param>
- public IEnumerable<IModMetadata> ProcessDependencies(IEnumerable<IModMetadata> mods)
+ /// <param name="modDatabase">Handles access to SMAPI's internal mod metadata list.</param>
+ public IEnumerable<IModMetadata> ProcessDependencies(IEnumerable<IModMetadata> mods, ModDatabase modDatabase)
{
// initialise metadata
mods = mods.ToArray();
@@ -209,7 +206,7 @@ namespace StardewModdingAPI.Framework.ModLoading
// sort mods
foreach (IModMetadata mod in mods)
- this.ProcessDependencies(mods.ToArray(), mod, states, sortedMods, new List<IModMetadata>());
+ this.ProcessDependencies(mods.ToArray(), modDatabase, mod, states, sortedMods, new List<IModMetadata>());
return sortedMods.Reverse();
}
@@ -220,12 +217,13 @@ namespace StardewModdingAPI.Framework.ModLoading
*********/
/// <summary>Sort a mod's dependencies by the order they should be loaded, and remove any mods that can't be loaded due to missing or conflicting dependencies.</summary>
/// <param name="mods">The full list of mods being validated.</param>
+ /// <param name="modDatabase">Handles access to SMAPI's internal mod metadata list.</param>
/// <param name="mod">The mod whose dependencies to process.</param>
/// <param name="states">The dependency state for each mod.</param>
/// <param name="sortedMods">The list in which to save mods sorted by dependency order.</param>
/// <param name="currentChain">The current change of mod dependencies.</param>
/// <returns>Returns the mod dependency status.</returns>
- private ModDependencyStatus ProcessDependencies(IModMetadata[] mods, IModMetadata mod, IDictionary<IModMetadata, ModDependencyStatus> states, Stack<IModMetadata> sortedMods, ICollection<IModMetadata> currentChain)
+ private ModDependencyStatus ProcessDependencies(IModMetadata[] mods, ModDatabase modDatabase, IModMetadata mod, IDictionary<IModMetadata, ModDependencyStatus> states, Stack<IModMetadata> sortedMods, ICollection<IModMetadata> currentChain)
{
// check if already visited
switch (states[mod])
@@ -276,11 +274,17 @@ namespace StardewModdingAPI.Framework.ModLoading
// missing required dependencies, mark failed
{
- string[] failedIDs = (from entry in dependencies where entry.IsRequired && entry.Mod == null select entry.ID).ToArray();
- if (failedIDs.Any())
+ string[] failedModNames = (
+ from entry in dependencies
+ where entry.IsRequired && entry.Mod == null
+ let displayName = modDatabase.GetDisplayNameFor(entry.ID) ?? entry.ID
+ orderby displayName
+ select displayName
+ ).ToArray();
+ if (failedModNames.Any())
{
sortedMods.Push(mod);
- mod.SetStatus(ModMetadataStatus.Failed, $"it requires mods which aren't installed ({string.Join(", ", failedIDs)}).");
+ mod.SetStatus(ModMetadataStatus.Failed, $"it requires mods which aren't installed ({string.Join(", ", failedModNames)}).");
return states[mod] = ModDependencyStatus.Failed;
}
}
@@ -325,7 +329,7 @@ namespace StardewModdingAPI.Framework.ModLoading
}
// recursively process each dependency
- var substatus = this.ProcessDependencies(mods, requiredMod, states, sortedMods, subchain);
+ var substatus = this.ProcessDependencies(mods, modDatabase, requiredMod, states, sortedMods, subchain);
switch (substatus)
{
// sorted successfully
diff --git a/src/SMAPI/Framework/Models/SConfig.cs b/src/SMAPI/Framework/Models/SConfig.cs
index 401e1a3a..17169714 100644
--- a/src/SMAPI/Framework/Models/SConfig.cs
+++ b/src/SMAPI/Framework/Models/SConfig.cs
@@ -1,3 +1,6 @@
+using System.Collections.Generic;
+using StardewModdingAPI.Framework.ModData;
+
namespace StardewModdingAPI.Framework.Models
{
/// <summary>The SMAPI configuration settings.</summary>
@@ -22,6 +25,6 @@ namespace StardewModdingAPI.Framework.Models
public bool VerboseLogging { get; set; }
/// <summary>Extra metadata about mods.</summary>
- public ModDataRecord[] ModData { get; set; }
+ public IDictionary<string, ModDataRecord> ModData { get; set; }
}
}
diff --git a/src/SMAPI/Program.cs b/src/SMAPI/Program.cs
index b5ce3033..ec841f4c 100644
--- a/src/SMAPI/Program.cs
+++ b/src/SMAPI/Program.cs
@@ -19,6 +19,7 @@ using StardewModdingAPI.Events;
using StardewModdingAPI.Framework;
using StardewModdingAPI.Framework.Exceptions;
using StardewModdingAPI.Framework.Logging;
+using StardewModdingAPI.Framework.ModData;
using StardewModdingAPI.Framework.Models;
using StardewModdingAPI.Framework.ModHelpers;
using StardewModdingAPI.Framework.ModLoading;
@@ -349,17 +350,20 @@ namespace StardewModdingAPI
if (!this.ValidateContentIntegrity())
this.Monitor.Log("SMAPI found problems in your game's content files which are likely to cause errors or crashes. Consider uninstalling XNB mods or reinstalling the game.", LogLevel.Error);
+ // load mod data
+ ModDatabase modDatabase = new ModDatabase(this.Settings.ModData);
+
// load mods
{
this.Monitor.Log("Loading mod metadata...", LogLevel.Trace);
ModResolver resolver = new ModResolver();
// load manifests
- IModMetadata[] mods = resolver.ReadManifests(Constants.ModPath, new JsonHelper(), this.Settings.ModData).ToArray();
+ IModMetadata[] mods = resolver.ReadManifests(Constants.ModPath, new JsonHelper(), modDatabase).ToArray();
resolver.ValidateManifests(mods, Constants.ApiVersion, Constants.VendorModUrls);
// process dependencies
- mods = resolver.ProcessDependencies(mods).ToArray();
+ mods = resolver.ProcessDependencies(mods, modDatabase).ToArray();
// load mods
this.LoadMods(mods, new JsonHelper(), this.ContentManager);
diff --git a/src/SMAPI/StardewModdingAPI.config.json b/src/SMAPI/StardewModdingAPI.config.json
index 7518d6b3..8b92f277 100644
--- a/src/SMAPI/StardewModdingAPI.config.json
+++ b/src/SMAPI/StardewModdingAPI.config.json
@@ -44,7 +44,8 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha
*
* Standard fields
* ===============
- * The predefined fields are documented below (only 'ID' is required).
+ * The predefined fields are documented below (only 'ID' is required). Each entry's key is the
+ * default display name for the mod if one isn't available (e.g. in dependency checks).
*
* - ID: the mod's latest unique ID (if any).
*
@@ -91,1718 +92,1719 @@ This file contains advanced configuration for SMAPI. You generally shouldn't cha
* - AlternativeUrl: a URL where the player can find an unofficial update or alternative if the
* mod is no longer compatible.
*/
- "ModData": [
- {
- // AccessChestAnywhere
+ "ModData": {
+ "AccessChestAnywhere": {
"ID": "AccessChestAnywhere",
"MapLocalVersions": { "1.1-1078": "1.1" },
"Default | UpdateKey": "Nexus:257",
- "~1.1 | Status": "AssumeBroken",
+ "~1.1 | Status": "AssumeBroken",
"~1.1 | AlternativeUrl": "https://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // AdjustArtisanPrices
+
+ "AdjustArtisanPrices": {
"ID": "1e36d4ca-c7ef-4dfb-9927-d27a6c3c8bdc",
"Default | UpdateKey": "Chucklefish:3532",
- "~0.1 | Status": "AssumeBroken",
+ "~0.1 | Status": "AssumeBroken",
"~0.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Adjust Monster
+
+ "Adjust Monster": {
"ID": "mmanlapat.AdjustMonster",
"Default | UpdateKey": "Nexus:1161"
},
- {
- // Advanced Location Loader
+
+ "Advanced Location Loader": {
"ID": "Entoarox.AdvancedLocationLoader",
"~1.3.7 | UpdateKey": "Chucklefish:3619", // only enable update checks up to 1.3.7 by request (has its own update-check feature)
"~1.2.10 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Adventure Shop Inventory
+
+ "Adventure Shop Inventory": {
"ID": "HammurabiAdventureShopInventory",
"Default | UpdateKey": "Chucklefish:4608"
},
- {
- // AgingMod
+
+ "AgingMod": {
"ID": "skn.AgingMod",
"Default | UpdateKey": "Nexus:1129",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // All Crops All Seasons
+
+ "All Crops All Seasons": {
"ID": "cantorsdust.AllCropsAllSeasons",
"FormerIDs": "29ee8246-d67b-4242-a340-35a9ae0d5dd7 | community.AllCropsAllSeasons", // changed in 1.3 and 1.5
"Default | UpdateKey": "Nexus:170"
},
- {
- // All Professions
+
+ "All Professions": {
"ID": "cantorsdust.AllProfessions",
"FormerIDs": "8c37b1a7-4bfb-4916-9d8a-9533e6363ea3 | community.AllProfessions", // changed in 1.2 and 1.3.1
"Default | UpdateKey": "Nexus:174"
},
- {
- // Almighty Tool
+
+ "Almighty Tool": {
"ID": "439",
"FormerIDs": "{EntryDll: 'AlmightyTool.dll'}", // changed in 1.2.1
"MapRemoteVersions": { "1.21": "1.2.1" },
"Default | UpdateKey": "Nexus:439",
"~1.1.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Animal Mood Fix
+
+ "Animal Husbandry": {
+ "ID": "DIGUS.ANIMALHUSBANDRYMOD",
+ "FormerIDs": "DIGUS.BUTCHER", // changed in 2.0.1
+ "Default | UpdateKey": "Nexus:1538"
+ },
+
+ "Animal Mood Fix": {
"ID": "GPeters-AnimalMoodFix",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "the animal mood bugs were fixed in Stardew Valley 1.2."
},
- {
- // Animal Sitter
+
+ "Animal Sitter": {
"ID": "jwdred.AnimalSitter",
"FormerIDs": "{EntryDll: 'AnimalSitter.dll'}", // changed in 1.0.9
"Default | UpdateKey": "Nexus:581",
"~1.0.8 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // A Tapper's Dream
+
+ "A Tapper's Dream": {
"ID": "ddde5195-8f85-4061-90cc-0d4fd5459358",
"Default | UpdateKey": "Nexus:260",
"~1.4 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.4 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Auto Animal Doors
+
+ "Auto Animal Doors": {
"ID": "AaronTaggart.AutoAnimalDoors",
"MapRemoteVersions": { "1.1.1": "1.1" }, // manifest not updated
"Default | UpdateKey": "Nexus:1019"
},
- {
- // Auto-Eat
+
+ "Auto-Eat": {
"ID": "Permamiss.AutoEat",
"FormerIDs": "BALANCEMOD_AutoEat", // changed in 1.1.1
"Default | UpdateKey": "Nexus:643"
},
- {
- // AutoGate
+
+ "AutoGate": {
"ID": "AutoGate",
"Default | UpdateKey": "Nexus:820"
},
- {
- // Automate
+
+ "Automate": {
"ID": "Pathoschild.Automate",
"Default | UpdateKey": "Nexus:1063"
},
- {
- // Automated Doors
+
+ "Automated Doors": {
"ID": "azah.automated-doors",
"FormerIDs": "1abcfa07-2cf4-4dc3-a6e9-6068b642112b", // changed in 1.4.1
"MapLocalVersions": { "1.4.1-1": "1.4.1" },
"Default | UpdateKey": "GitHub:azah/AutomatedDoors"
},
- {
- // AutoSpeed
+
+ "AutoSpeed": {
"ID": "Omegasis.AutoSpeed",
"FormerIDs": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'AutoSpeed'}", // changed in 1.4; disambiguate from other Alpha_Omegasis mods
"Default | UpdateKey": "Nexus:443" // added in 1.4.1
},
- {
- // Basic Sprinkler Improved
+
+ "Basic Sprinklers Improved": {
"ID": "lrsk_sdvm_bsi.0117171308",
"MapRemoteVersions": { "1.0.2": "1.0.1-release" }, // manifest not updated
"Default | UpdateKey": "Nexus:833"
},
- {
- // Better Hay
+
+ "Better Hay": {
"ID": "cat.betterhay",
"Default | UpdateKey": "Nexus:1430"
},
- {
- // Better Quality More Seasons
+
+ "Better Quality More Seasons": {
"ID": "SB_BQMS",
"Default | UpdateKey": "Nexus:935"
},
- {
- // Better Quarry
+
+ "Better Quarry": {
"ID": "BetterQuarry",
"Default | UpdateKey": "Nexus:771"
},
- {
- // Better Ranching
+
+ "Better Ranching": {
"ID": "BetterRanching",
"Default | UpdateKey": "Nexus:859"
},
- {
- // Better Shipping Box
+
+ "Better Shipping Box": {
"ID": "Kithio:BetterShippingBox",
"MapLocalVersions": { "1.0.1": "1.0.2" },
"Default | UpdateKey": "Chucklefish:4302",
"~1.0.2 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Better Sprinklers
+
+ "Better Sprinklers": {
"ID": "Speeder.BetterSprinklers",
"FormerIDs": "SPDSprinklersMod", // changed in 2.3
"Default | UpdateKey": "Nexus:41",
"~2.3.1-pathoschild-update | Status": "AssumeBroken", // broke in SDV 1.2
"~2.3.1-pathoschild-update | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Billboard Anywhere
+
+ "Billboard Anywhere": {
"ID": "Omegasis.BillboardAnywhere",
"FormerIDs": "{ID:'7ad4f6f7-c3de-4729-a40f-7a11d2b2a358', Name:'Billboard Anywhere'}", // changed in 1.4; disambiguate from other mods by Alpha_Omegasis
"Default | UpdateKey": "Nexus:492" // added in 1.4.1
},
- {
- // Birthday Mail
+
+ "Birthday Mail": {
"ID": "KathrynHazuka.BirthdayMail",
"FormerIDs": "005e02dc-d900-425c-9c68-1ff55c5a295d", // changed in 1.2.3-pathoschild-update
"Default | UpdateKey": "Nexus:276",
"~1.2.2 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.2.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Breed Like Rabbits
+
+ "Breed Like Rabbits": {
"ID": "dycedarger.breedlikerabbits",
"Default | UpdateKey": "Nexus:948"
},
- {
- // Build Endurance
+
+ "Build Endurance": {
"ID": "Omegasis.BuildEndurance",
"FormerIDs": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'BuildEndurance'}", // changed in 1.4; disambiguate from other Alpha_Omegasis mods
"Default | UpdateKey": "Nexus:445", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Build Health
+
+ "Build Health": {
"ID": "Omegasis.BuildHealth",
"FormerIDs": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'BuildHealth'}", // changed in 1.4; disambiguate from other Alpha_Omegasis mods
"Default | UpdateKey": "Nexus:446", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Butcher Mod
- "ID": "DIGUS.BUTCHER",
- "Default | UpdateKey": "Nexus:1538"
- },
- {
- // Buy Cooking Recipes
+
+ "Buy Cooking Recipes": {
"ID": "Denifia.BuyRecipes",
"Default | UpdateKey": "Nexus:1126", // added in 1.0.1 (2017-10-04)
"~1.0 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Buy Back Collectables
+
+ "Buy Back Collectables": {
"ID": "Omegasis.BuyBackCollectables",
"FormerIDs": "BuyBackCollectables", // changed in 1.4
"Default | UpdateKey": "Nexus:507", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Carry Chest
+
+ "Carry Chest": {
"ID": "spacechase0.CarryChest",
"Default | UpdateKey": "Nexus:1333"
},
- {
- // Casks Anywhere
+
+ "Casks Anywhere": {
"ID": "CasksAnywhere",
"MapLocalVersions": { "1.1-alpha": "1.1" },
"Default | UpdateKey": "Nexus:878"
},
- {
- // Categorize Chests
+
+ "Categorize Chests": {
"ID": "CategorizeChests",
"Default | UpdateKey": "Nexus:1300"
},
- {
- // ChefsCloset
+
+ "Chefs Closet": {
"ID": "Duder.ChefsCloset",
"MapLocalVersions": { "1.3-1": "1.3" },
"Default | UpdateKey": "Nexus:1030"
},
- {
- // Chest Label System
+
+ "Chest Label System": {
"ID": "Speeder.ChestLabel",
"FormerIDs": "SPDChestLabel", // changed in 1.5.1-pathoschild-update
"Default | UpdateKey": "Nexus:242",
"~1.6 | Status": "AssumeBroken", // broke in SDV 1.1
"~1.6 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Chest Pooling
+
+ "Chest Pooling": {
"ID": "mralbobo.ChestPooling",
"FormerIDs": "{EntryDll: 'ChestPooling.dll'}", // changed in 1.3
"Default | UpdateKey": "GitHub:mralbobo/stardew-chest-pooling",
"~1.2 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Chests Anywhere
+
+ "Chests Anywhere": {
"ID": "Pathoschild.ChestsAnywhere",
"FormerIDs": "ChestsAnywhere", // changed in 1.9
"Default | UpdateKey": "Nexus:518",
"~1.9-beta | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Choose Baby Gender
+
+ "Choose Baby Gender": {
"ID": "{EntryDll: 'ChooseBabyGender.dll'}",
"Default | UpdateKey": "Nexus:590",
"~1.0.2 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // CJB Automation
+
+ "CJB Automation": {
"ID": "CJBAutomation",
"Default | UpdateKey": "Nexus:211",
"~1.4 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.4 | AlternativeUrl": "http://www.nexusmods.com/stardewvalley/mods/1063"
},
- {
- // CJB Cheats Menu
+
+ "CJB Cheats Menu": {
"ID": "CJBok.CheatsMenu",
"FormerIDs": "CJBCheatsMenu", // changed in 1.14
"Default | UpdateKey": "Nexus:4",
"~1.12 | Status": "AssumeBroken" // broke in SDV 1.1
},
- {
- // CJB Item Spawner
+
+ "CJB Item Spawner": {
"ID": "CJBok.ItemSpawner",
"FormerIDs": "CJBItemSpawner", // changed in 1.7
"Default | UpdateKey": "Nexus:93",
"~1.5 | Status": "AssumeBroken" // broke in SDV 1.1
},
- {
- // CJB Show Item Sell Price
+
+ "CJB Show Item Sell Price": {
"ID": "CJBok.ShowItemSellPrice",
"FormerIDs": "CJBShowItemSellPrice", // changed in 1.7
"Default | UpdateKey": "Nexus:5",
"~1.6 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Clean Farm
+
+ "Clean Farm": {
"ID": "tstaples.CleanFarm",
"Default | UpdateKey": "Nexus:794"
},
- {
- // Climates of Ferngill
+
+ "Climates of Ferngill": {
"ID": "KoihimeNakamura.ClimatesOfFerngill",
"Default | UpdateKey": "Nexus:604",
"~1.0 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Cold Weather Haley
+
+ "Cold Weather Haley": {
"ID": "LordXamon.ColdWeatherHaleyPRO",
"Default | UpdateKey": "Nexus:1169",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Colored Chests
+
+ "Colored Chests": {
"ID": "4befde5c-731c-4853-8e4b-c5cdf946805f",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "colored chests were added in Stardew Valley 1.1."
},
- {
- // Combat with Farm Implements
+
+ "Combat with Farm Implements": {
"ID": "SPDFarmingImplementsInCombat",
"Default | UpdateKey": "Nexus:313",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Community Bundle Item Tooltip
+
+ "Community Bundle Item Tooltip": {
"ID": "musbah.bundleTooltip",
"Default | UpdateKey": "Nexus:1329"
},
- {
- // Concentration on Farming
+
+ "Concentration on Farming": {
"ID": "punyo.ConcentrationOnFarming",
"Default | UpdateKey": "Nexus:1445"
},
- {
- // Configurable Machines
+
+ "Configurable Machines": {
"ID": "21da6619-dc03-4660-9794-8e5b498f5b97",
"MapLocalVersions": { "1.2-beta": "1.2" },
"Default | UpdateKey": "Nexus:280"
},
- {
- // Configurable Shipping Dates
+
+ "Configurable Shipping Dates": {
"ID": "ConfigurableShippingDates",
"Default | UpdateKey": "Nexus:675",
"~1.1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Cooking Skill
+
+ "Cooking Skill": {
"ID": "spacechase0.CookingSkill",
"FormerIDs": "CookingSkill", // changed in 1.0.4–6
"Default | UpdateKey": "Nexus:522",
"~1.0.6 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // CrabNet
+
+ "CrabNet": {
"ID": "jwdred.CrabNet",
"FormerIDs": "{EntryDll: 'CrabNet.dll'}", // changed in 1.0.5
"Default | UpdateKey": "Nexus:584",
"~1.0.4 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Current Location
+
+ "Current Location": {
"ID": "CurrentLocation102120161203",
"Default | UpdateKey": "Nexus:638"
},
- {
- // Custom Critters
+
+ "Custom Critters": {
"ID": "spacechase0.CustomCritters",
"Default | UpdateKey": "Nexus:1255"
},
- {
- // Custom Element Handler
+
+ "Custom Element Handler": {
"ID": "Platonymous.CustomElementHandler",
"Default | UpdateKey": "Nexus:1068" // added in 1.3.1
},
- {
- // Custom Farming
+
+ "Custom Farming": {
"ID": "Platonymous.CustomFarming",
"Default | UpdateKey": "Nexus:991" // added in 0.6.1
},
- {
- // Custom Farming Automate Bridge
+
+ "Custom Farming Automate Bridge": {
"ID": "Platonymous.CFAutomate",
"~1.0.1 | Status": "AssumeBroken", // no longer compatible with Automate
"~1.0.1 | AlternativeUrl": "https://www.nexusmods.com/stardewvalley/mods/991"
},
- {
- // Custom Farm Types
+
+ "Custom Farm Types": {
"ID": "spacechase0.CustomFarmTypes",
"Default | UpdateKey": "Nexus:1140"
},
- {
- // Custom Furniture
+
+ "Custom Furniture": {
"ID": "Platonymous.CustomFurniture",
"Default | UpdateKey": "Nexus:1254" // added in 0.4.1
},
- {
- // Customize Exterior
+
+ "Customize Exterior": {
"ID": "spacechase0.CustomizeExterior",
"FormerIDs": "CustomizeExterior", // changed in 1.0.3
"Default | UpdateKey": "Nexus:1099",
"~1.0.2 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Customizable Cart Redux
+
+
+ "Customizable Cart Redux": {
"ID": "KoihimeNakamura.CCR",
"MapLocalVersions": { "1.1-20170917": "1.1" },
"Default | UpdateKey": "Nexus:1402"
},
- {
- // Customizable Traveling Cart Days
+
+ "Customizable Traveling Cart Days": {
"ID": "TravelingCartYyeahdude",
"Default | UpdateKey": "Nexus:567",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Custom Linens
+
+ "Custom Linens": {
"ID": "Mevima.CustomLinens",
"MapRemoteVersions": { "1.1": "1.0" }, // manifest not updated
"Default | UpdateKey": "Nexus:1027"
},
- {
- // Custom Shops Redux
+
+ "Custom Shops Redux": {
"ID": "Omegasis.CustomShopReduxGui",
"Default | UpdateKey": "Nexus:1378" // added in 1.4.1
},
- {
- // Custom TV
+
+ "Custom TV": {
"ID": "Platonymous.CustomTV",
"Default | UpdateKey": "Nexus:1139" // added in 1.0.6
},
- {
- // Daily Luck Message
+
+ "Daily Luck Message": {
"ID": "Schematix.DailyLuckMessage",
"Default | UpdateKey": "Nexus:1327"
},
- {
- // Daily News
+
+ "Daily News": {
"ID": "bashNinja.DailyNews",
"Default | UpdateKey": "Nexus:1141",
"~1.1 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Daily Quest Anywhere
+
+ "Daily Quest Anywhere": {
"ID": "Omegasis.DailyQuestAnywhere",
"FormerIDs": "DailyQuest", // changed in 1.4
"Default | UpdateKey": "Nexus:513" // added in 1.4.1
},
- {
- // Debug Mode
+
+ "Debug Mode": {
"ID": "Pathoschild.DebugMode",
"FormerIDs": "Pathoschild.Stardew.DebugMode", // changed in 1.4
"Default | UpdateKey": "Nexus:679"
},
- {
- // Dynamic Checklist
+
+ "Dynamic Checklist": {
"ID": "gunnargolf.DynamicChecklist",
"Default | UpdateKey": "Nexus:1145", // added in 1.0.1-pathoschild-update
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Dynamic Horses
+
+ "Dynamic Horses": {
"ID": "Bpendragon-DynamicHorses",
"MapRemoteVersions": { "1.2": "1.1-release" }, // manifest not updated
"Default | UpdateKey": "Nexus:874"
},
- {
- // Dynamic Machines
+
+ "Dynamic Machines": {
"ID": "DynamicMachines",
"MapLocalVersions": { "1.1": "1.1.1" },
"Default | UpdateKey": "Nexus:374",
"~1.1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Dynamic NPC Sprites
+
+ "Dynamic NPC Sprites": {
"ID": "BashNinja.DynamicNPCSprites",
"Default | UpdateKey": "Nexus:1183"
},
- {
- // Easier Farming
+
+ "Easier Farming": {
"ID": "cautiouswafffle.EasierFarming",
"Default | UpdateKey": "Nexus:1426"
},
- {
- // Empty Hands
+
+ "Empty Hands": {
"ID": "QuicksilverFox.EmptyHands",
"Default | UpdateKey": "Nexus:1176", // added in 1.0.1-pathoschild-update
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Enemy Health Bars
+
+ "Enemy Health Bars": {
"ID": "Speeder.HealthBars",
"FormerIDs": "SPDHealthBar", // changed in 1.7.1-pathoschild-update
"Default | UpdateKey": "Nexus:193",
"~1.7 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.7 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Entoarox Framework
+
+ "Entoarox Framework": {
"ID": "Entoarox.EntoaroxFramework",
"FormerIDs": "eacdb74b-4080-4452-b16b-93773cda5cf9", // changed in ???
"~2.0.6 | UpdateKey": "Chucklefish:4228", // only enable update checks up to 2.0.6 by request (has its own update-check feature)
"~1.7.9 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Expanded Fridge / Dynamic Expanded Fridge
+
+ "Expanded Fridge": {
"ID": "Uwazouri.ExpandedFridge",
"Default | UpdateKey": "Nexus:1191"
},
- {
- // Experience Bars
+
+ "Experience Bars": {
"ID": "spacechase0.ExperienceBars",
"FormerIDs": "ExperienceBars", // changed in 1.0.2
"Default | UpdateKey": "Nexus:509"
},
- {
- // Extended Bus System
+
+ "Extended Bus System": {
"ID": "ExtendedBusSystem",
"Default | UpdateKey": "Chucklefish:4373"
},
- {
- // Extended Fridge
+
+ "Extended Fridge": {
"ID": "Crystalmir.ExtendedFridge",
"FormerIDs": "Mystra007ExtendedFridge", // changed in 1.0.1
"Default | UpdateKey": "Nexus:485",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Extended Greenhouse
+
+ "Extended Greenhouse": {
"ID": "ExtendedGreenhouse",
"Default | UpdateKey": "Chucklefish:4303",
"~1.0.2 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Extended Minecart
+
+ "Extended Minecart": {
"ID": "Entoarox.ExtendedMinecart",
"FormerIDs": "{ID:'EntoaroxFurnitureAnywhere', Name:'Extended Minecart'}", // changed in 1.6.1
"~1.7.1 | UpdateKey": "Chucklefish:4359" // only enable update checks up to 1.7.1 by request (has its own update-check feature)
},
- {
- // Extended Reach
+
+ "Extended Reach": {
"ID": "spacechase0.ExtendedReach",
"Default | UpdateKey": "Nexus:1493"
},
- {
- // Fall 28 Snow Day
+
+ "Fall 28 Snow Day": {
"ID": "Omegasis.Fall28SnowDay",
"FormerIDs": "{ID:'7ad4f6f7-c3de-4729-a40f-7a11d2b2a358', Name:'Fall28 Snow Day'}", // changed in 1.4; disambiguate from other mods by Alpha_Omegasis
"Default | UpdateKey": "Nexus:486", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Farm Automation: Barn Door Automation
+
+ "Farm Automation: Barn Door Automation": {
"ID": "{EntryDll: 'FarmAutomation.BarnDoorAutomation.dll'}",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Farm Automation: Item Collector
+
+ "Farm Automation: Item Collector": {
"ID": "{EntryDll: 'FarmAutomation.ItemCollector.dll'}",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Farm Automation Unofficial: Item Collector
+
+ "Farm Automation Unofficial: Item Collector": {
"ID": "Maddy99.FarmAutomation.ItemCollector",
"~0.5 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~0.5 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Farm Expansion
+
+ "Farm Expansion": {
"ID": "Advize.FarmExpansion",
"FormerIDs": "3888bdfd-73f6-4776-8bb7-8ad45aea1915 | AdvizeFarmExpansionMod-2-0 | AdvizeFarmExpansionMod-2-0-5", // changed in 2.0, 2.0.5, and 3.0
"Default | UpdateKey": "Nexus:130",
"~2.0.5 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~2.0.5 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Farm Resource Generator
+
+ "Farm Resource Generator": {
"ID": "{EntryDll: 'FarmResourceGenerator.dll'}",
"Default | UpdateKey": "Nexus:647",
"~1.0.4 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.4 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Fast Animations
+
+ "Fast Animations": {
"ID": "Pathoschild.FastAnimations",
"Default | UpdateKey": "Nexus:1089"
},
- {
- // Faster Paths
+
+ "Faster Paths": {
"ID": "Entoarox.FasterPaths",
"FormerIDs": "{ID:'821ce8f6-e629-41ad-9fde-03b54f68b0b6', Name:'Faster Paths'} | 615f85f8-5c89-44ee-aecc-c328f172e413", // changed in 1.2 and 1.3; disambiguate from Shop Expander
"~1.3.3 | UpdateKey": "Chucklefish:3641" // only enable update checks up to 1.3.3 by request (has its own update-check feature)
},
- {
- // Faster Run
+
+ "Faster Run": {
"ID": "KathrynHazuka.FasterRun",
"FormerIDs": "{EntryDll: 'FasterRun.dll'}", // changed in 1.1.1-pathoschild-update
"Default | UpdateKey": "Nexus:733", // added in 1.1.1-pathoschild-update
"~1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Fishing Adjust
+
+ "Fishing Adjust": {
"ID": "shuaiz.FishingAdjustMod",
"Default | UpdateKey": "Nexus:1350"
},
- {
- // Fishing Tuner Redux
+
+ "Fishing Tuner Redux": {
"ID": "HammurabiFishingTunerRedux",
"Default | UpdateKey": "Chucklefish:4578"
},
- {
- // FlorenceMod
+
+ "FlorenceMod": {
"ID": "{EntryDll: 'FlorenceMod.dll'}",
"MapLocalVersions": { "1.0.1": "1.1" },
"Default | UpdateKey": "Nexus:591",
"~1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Flower Color Picker
+
+ "Flower Color Picker": {
"ID": "spacechase0.FlowerColorPicker",
"Default | UpdateKey": "Nexus:1229"
},
- {
- // Forage at the Farm
+
+ "Forage at the Farm": {
"ID": "ForageAtTheFarm",
"Default | UpdateKey": "Nexus:673",
"~1.5.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.5.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Furniture Anywhere
+
+ "Furniture Anywhere": {
"ID": "Entoarox.FurnitureAnywhere",
"FormerIDs": "{ID:'EntoaroxFurnitureAnywhere', Name:'Furniture Anywhere'}", // changed in 1.1; disambiguate from Extended Minecart
"~1.1.5 | UpdateKey": "Chucklefish:4324" // only enable update checks up to 1.1.5 by request (has its own update-check feature)
},
- {
- // Game Reminder
+
+ "Game Reminder": {
"ID": "mmanlapat.GameReminder",
"Default | UpdateKey": "Nexus:1153"
},
- {
- // Gate Opener
+
+ "Gate Opener": {
"ID": "mralbobo.GateOpener",
"FormerIDs": "{EntryDll: 'GateOpener.dll'}", // changed in 1.1
"Default | UpdateKey": "GitHub:mralbobo/stardew-gate-opener",
"~1.0.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // GenericShopExtender
+
+ "GenericShopExtender": {
"ID": "GenericShopExtender",
"Default | UpdateKey": "Nexus:814", // added in 0.1.3
"~0.1.2 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Geode Info Menu
+
+ "Geode Info Menu": {
"ID": "cat.geodeinfomenu",
"Default | UpdateKey": "Nexus:1448"
},
- {
- // Get Dressed
+
+ "Get Dressed": {
"ID": "Advize.GetDressed",
"FormerIDs": "{EntryDll: 'GetDressed.dll'}", // changed in 3.3
"Default | UpdateKey": "Nexus:331",
"~3.3 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Giant Crop Ring
+
+ "Giant Crop Ring": {
"ID": "cat.giantcropring",
"Default | UpdateKey": "Nexus:1182"
},
- {
- // Gift Taste Helper
+
+ "Gift Taste Helper": {
"ID": "tstaples.GiftTasteHelper",
"FormerIDs": "8008db57-fa67-4730-978e-34b37ef191d6", // changed in 2.5
"Default | UpdateKey": "Nexus:229",
"~2.3.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Grandfather's Gift
+
+ "Grandfather's Gift": {
"ID": "ShadowDragon.GrandfathersGift",
"Default | UpdateKey": "Nexus:985"
},
- {
- // Happy Animals
+
+ "Happy Animals": {
"ID": "HappyAnimals",
"~1.0.3 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.3 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Happy Birthday (Omegasis)
+
+ "Happy Birthday (Omegasis)": {
"ID": "Omegasis.HappyBirthday",
"FormerIDs": "{ID:'HappyBirthday', Author:'Alpha_Omegasis'}", // changed in 1.4; disambiguate from Oxyligen's fork
"Default | UpdateKey": "Nexus:520", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Happy Birthday (Oxyligen fork)
+
+ "Happy Birthday (Oxyligen fork)": {
"ID": "{ID:'HappyBirthday', Author:'Alpha_Omegasis/Oxyligen'}", // disambiguate from Oxyligen's fork
"Default | UpdateKey": "Nexus:1064"
},
- {
- // Harp of Yoba Redux
+
+ "Harp of Yoba Redux": {
"ID": "Platonymous.HarpOfYobaRedux",
"Default | UpdateKey": "Nexus:914" // added in 2.0.3
},
- {
- // Harvest Moon Witch Princess
+
+ "Harvest Moon Witch Princess": {
"ID": "Sasara.WitchPrincess",
"Default | UpdateKey": "Nexus:1157"
},
- {
- // Harvest With Scythe
+
+ "Harvest With Scythe": {
"ID": "965169fd-e1ed-47d0-9f12-b104535fb4bc",
"Default | UpdateKey": "Nexus:236",
"~1.0.6 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.6 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Horse Whistle (icepuente)
+
+ "Horse Whistle (icepuente)": {
"ID": "icepuente.HorseWhistle",
"Default | UpdateKey": "Nexus:1131"
},
- {
- // Hunger (Yyeadude)
+
+ "Hunger (Yyeadude)": {
"ID": "HungerYyeadude",
"Default | UpdateKey": "Nexus:613"
},
- {
- // Hunger for Food (Tigerle)
+
+ "Hunger for Food (Tigerle)": {
"ID": "HungerForFoodByTigerle",
"Default | UpdateKey": "Nexus:810",
"~0.1.2 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~0.1.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Hunger Mod (skn)
+
+ "Hunger Mod (skn)": {
"ID": "skn.HungerMod",
"MapRemoteVersions": { "1.2.1": "1.0" }, // manifest not updated
"Default | UpdateKey": "Nexus:1127"
},
- {
- // Idle Pause
+
+ "Idle Pause": {
"ID": "Veleek.IdlePause",
"MapRemoteVersions": { "1.2": "1.1" }, // manifest not updated
"Default | UpdateKey": "Nexus:1092"
},
- {
- // Improved Quality of Life
+
+ "Improved Quality of Life": {
"ID": "Demiacle.ImprovedQualityOfLife",
"Default | UpdateKey": "Nexus:1025",
"~1.1 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Instant Geode
+
+ "Instant Geode": {
"ID": "InstantGeode",
"~1.12 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.12 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Instant Grow Trees
+
+ "Instant Grow Trees": {
"ID": "cantorsdust.InstantGrowTrees",
"FormerIDs": "dc50c58b-c7d8-4e60-86cc-e27b5d95ee59 | community.InstantGrowTrees", // changed in 1.2 and 1.3.1
"Default | UpdateKey": "Nexus:173"
},
- {
- // Interaction Helper
+
+ "Interaction Helper": {
"ID": "HammurabiInteractionHelper",
"Default | UpdateKey": "Chucklefish:4640", // added in 1.0.4-pathoschild-update
"~1.0.3 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.3 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Item Auto Stacker
+
+ "Item Auto Stacker": {
"ID": "cat.autostacker",
"MapRemoteVersions": { "1.0.1": "1.0" }, // manifest not updated
"Default | UpdateKey": "Nexus:1184"
},
- {
- // Jiggly Junimo Bundles
+
+ "Jiggly Junimo Bundles": {
"ID": "Greger.JigglyJunimoBundles",
"FormerIDs": "{EntryDll: 'JJB.dll'}", // changed in 1.1.2-pathoschild-update
"Default | UpdateKey": "GitHub:gr3ger/Stardew_JJB" // added in 1.0.4-pathoschild-update
},
- {
- // Junimo Farm
+
+ "Junimo Farm": {
"ID": "Platonymous.JunimoFarm",
"MapRemoteVersions": { "1.1.2": "1.1.1" }, // manifest not updated
"Default | UpdateKey": "Nexus:984" // added in 1.1.3
},
- {
- // Less Strict Over-Exertion (AntiExhaustion)
+
+ "Less Strict Over-Exertion (AntiExhaustion)": {
"ID": "BALANCEMOD_AntiExhaustion",
"MapLocalVersions": { "0.0": "1.1" },
"Default | UpdateKey": "Nexus:637",
"~1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Level Extender
+
+ "Level Extender": {
"ID": "Devin Lematty.Level Extender",
"MapRemoteVersions": { "1.1": "1.0" }, // manifest not updated
"Default | UpdateKey": "Nexus:1471"
},
- {
- // Level Up Notifications
+
+ "Level Up Notifications": {
"ID": "Level Up Notifications",
"Default | UpdateKey": "Nexus:855"
},
- {
- // Location and Music Logging
+
+ "Location and Music Logging": {
"ID": "Brandy Lover.LMlog",
"Default | UpdateKey": "Nexus:1366"
},
- {
- // Longevity
+
+ "Longevity": {
"ID": "RTGOAT.Longevity",
"Default | UpdateKey": "Nexus:649"
},
- {
- // Lookup Anything
+
+ "Lookup Anything": {
"ID": "Pathoschild.LookupAnything",
"FormerIDs": "LookupAnything", // changed in 1.10.1
"Default | UpdateKey": "Nexus:541",
"~1.10.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Love Bubbles
+
+ "Love Bubbles": {
"ID": "LoveBubbles",
"Default | UpdateKey": "Nexus:1318"
},
- {
- // Loved Labels
+
+ "Loved Labels": {
"ID": "Advize.LovedLabels",
"FormerIDs": "{EntryDll: 'LovedLabels.dll'}", // changed in 2.1
"Default | UpdateKey": "Nexus:279",
"~2.0 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Luck Skill
+
+ "Luck Skill": {
"ID": "spacechase0.LuckSkill",
"FormerIDs": "LuckSkill", // changed in 0.1.4
"Default | UpdateKey": "Nexus:521",
"~0.1.4 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Mail Framework
+
+ "Mail Framework": {
"ID": "DIGUS.MailFrameworkMod",
"Default | UpdateKey": "Nexus:1536"
},
- {
- // MailOrderPigs
+
+ "MailOrderPigs": {
"ID": "jwdred.MailOrderPigs",
"FormerIDs": "{EntryDll: 'MailOrderPigs.dll'}", // changed in 1.0.2
"Default | UpdateKey": "Nexus:632",
"~1.0.1 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Makeshift Multiplayer
+
+ "Makeshift Multiplayer": {
"ID": "spacechase0.StardewValleyMP",
"FormerIDs": "StardewValleyMP", // changed in 0.3
"Default | UpdateKey": "Nexus:501",
"~0.3.6 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Map Image Exporter
+
+ "Map Image Exporter": {
"ID": "spacechase0.MapImageExporter",
"FormerIDs": "MapImageExporter", // changed in 1.0.2
"Default | UpdateKey": "Nexus:1073"
},
- {
- // Message Box [API]? (ChatMod)
+
+ "Message Box [API]? (ChatMod)": {
"ID": "Kithio:ChatMod",
"Default | UpdateKey": "Chucklefish:4296",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Mining at the Farm
+
+ "Mining at the Farm": {
"ID": "MiningAtTheFarm",
"Default | UpdateKey": "Nexus:674"
},
- {
- // Mining With Explosives
+
+ "Mining With Explosives": {
"ID": "MiningWithExplosives",
"Default | UpdateKey": "Nexus:770"
},
- {
- // Modder Serialization Utility
+
+ "Modder Serialization Utility": {
"ID": "SerializerUtils-0-1",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "it's no longer maintained or used."
},
- {
- // More Animals
+
+ "More Animals": {
"ID": "Entoarox.MoreAnimals",
"FormerIDs": "821ce8f6-e629-41ad-9fde-03b54f68b0b6MOREPETS | Entoarox.MorePets", // changed in 1.3 and 2.0
"~2.0.2 | UpdateKey": "Chucklefish:4288", // only enable update checks up to 2.0.2 by request (has its own update-check feature)
"~1.3.2 | Status": "AssumeBroken" // overhauled for SMAPI 1.11+ compatibility
},
- {
- // More Artifact Spots
+
+ "More Artifact Spots": {
"ID": "451",
"Default | UpdateKey": "Nexus:451",
"~1.0.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // More Map Layers
+
+ "More Map Layers": {
"ID": "Platonymous.MoreMapLayers",
"Default | UpdateKey": "Nexus:1134" // added in 1.1.1
},
- {
- // More Rain
+
+ "More Rain": {
"ID": "Omegasis.MoreRain",
"FormerIDs": "{ID:'4108e859-333c-4fec-a1a7-d2e18c1019fe', Name:'More_Rain'}", // changed in 1.5; disambiguate from other mods by Alpha_Omegasis
"Default | UpdateKey": "Nexus:441", // added in 1.5.1
"~1.4 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // More Weapons
+
+ "More Weapons": {
"ID": "Joco80.MoreWeapons",
"Default | UpdateKey": "Nexus:1168"
},
- {
- // Move Faster
+
+ "Move Faster": {
"ID": "shuaiz.MoveFasterMod",
"Default | UpdateKey": "Nexus:1351"
},
- {
- // Multiple Sprites and Portraits On Rotation (File Loading)
+
+ "Multiple Sprites and Portraits On Rotation (File Loading)": {
"ID": "FileLoading",
"MapLocalVersions": { "1.1": "1.12" },
"Default | UpdateKey": "Nexus:1094",
"~1.12 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.12 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Museum Rearranger
+
+ "Museum Rearranger": {
"ID": "Omegasis.MuseumRearranger",
"FormerIDs": "{ID:'7ad4f6f7-c3de-4729-a40f-7a11d2b2a358', Name:'Museum Rearranger'}", // changed in 1.4; disambiguate from other mods by Alpha_Omegasis
"Default | UpdateKey": "Nexus:428", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // New Machines
+
+ "New Machines": {
"ID": "F70D4FAB-0AB2-4B78-9F1B-AF2CA2236A59",
"Default | UpdateKey": "Chucklefish:3683",
"~4.2.1343 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~4.2.1343 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Night Owl
+
+ "Night Owl": {
"ID": "Omegasis.NightOwl",
"FormerIDs": "{ID:'SaveAnywhere', Name:'Stardew_NightOwl'}", // changed in 1.4; disambiguate from Save Anywhere
"MapLocalVersions": { "2.1": "1.3" }, // 1.3 had wrong version in manifest
"Default | UpdateKey": "Nexus:433", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // No Kids Ever
+
+ "No Kids Ever": {
"ID": "Hangy.NoKidsEver",
"Default | UpdateKey": "Nexus:1464"
},
- {
- // No Debug Mode
+
+ "No Debug Mode": {
"ID": "NoDebugMode",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "debug mode was removed in SMAPI 1.0."
},
- {
- // No Fence Decay
+
+ "No Fence Decay": {
"ID": "cat.nofencedecay",
"Default | UpdateKey": "Nexus:1180"
},
- {
- // No More Pets
+
+ "No More Pets": {
"ID": "Omegasis.NoMorePets",
"FormerIDs": "NoMorePets", // changed in 1.4
"Default | UpdateKey": "Nexus:506" // added in 1.4.1
},
- {
- // NoSoilDecay
+
+ "NoSoilDecay": {
"ID": "289dee03-5f38-4d8e-8ffc-e440198e8610",
"Default | UpdateKey": "Nexus:237",
"~0.5 | Status": "AssumeBroken", // broke in SDV 1.2 and uses Assembly.GetExecutingAssembly().Location
"~0.5 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // No Soil Decay Redux
+
+ "No Soil Decay Redux": {
"ID": "Platonymous.NoSoilDecayRedux",
"Default | UpdateKey": "Nexus:1084" // added in 1.1.9
},
- {
- // NPC Map Locations
+
+ "NPC Map Locations": {
"ID": "NPCMapLocationsMod",
"Default | UpdateKey": "Nexus:239",
"1.42~1.43 | Status": "AssumeBroken",
"1.42~1.43 | StatusReasonPhrase": "this version has an update check error which crashes the game."
},
- {
- // NPC Speak
+
+ "NPC Speak": {
"ID": "{EntryDll: 'NpcEcho.dll'}",
"Default | UpdateKey": "Nexus:694",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Object Time Left
+
+ "Object Time Left": {
"ID": "spacechase0.ObjectTimeLeft",
"Default | UpdateKey": "Nexus:1315"
},
- {
- // OmniFarm
+
+ "OmniFarm": {
"ID": "PhthaloBlue.OmniFarm",
"FormerIDs": "BlueMod_OmniFarm", // changed in 2.0.2-pathoschild-update
"Default | UpdateKey": "GitHub:lambui/StardewValleyMod_OmniFarm",
"~2.0.1 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Out of Season Bonuses / Seasonal Items
+
+ "Out of Season Bonuses (Seasonal Items)": {
"ID": "midoriarmstrong.seasonalitems",
"Default | UpdateKey": "Nexus:1452"
},
- {
- // Part of the Community
+
+ "Part of the Community": {
"ID": "SB_PotC",
"Default | UpdateKey": "Nexus:923",
"~1.0.8 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // PelicanFiber
+
+ "PelicanFiber": {
"ID": "jwdred.PelicanFiber",
"FormerIDs": "{EntryDll: 'PelicanFiber.dll'}", // changed in 3.0.1
"MapRemoteVersions": { "3.0.2": "3.0.1" }, // didn't change manifest version
"Default | UpdateKey": "Nexus:631",
"~3.0 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // PelicanTTS
+
+ "PelicanTTS": {
"ID": "Platonymous.PelicanTTS",
"Default | UpdateKey": "Nexus:1079", // added in 1.6.1
"~1.6 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.6 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Persia the Mermaid - Standalone Custom NPC
+
+ "Persia the Mermaid - Standalone Custom NPC": {
"ID": "63b9f419-7449-42db-ab2e-440b4d05c073",
"Default | UpdateKey": "Nexus:1419"
},
- {
- // Persival's BundleMod
+
+ "Persival's BundleMod": {
"ID": "{EntryDll: 'BundleMod.dll'}",
"Default | UpdateKey": "Nexus:438",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.1
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Plant on Grass
+
+ "Plant on Grass": {
"ID": "Demiacle.PlantOnGrass",
"Default | UpdateKey": "Nexus:1026"
},
- {
- // Point-and-Plant
+
+ "Point-and-Plant": {
"ID": "jwdred.PointAndPlant",
"FormerIDs": "{EntryDll: 'PointAndPlant.dll'}", // changed in 1.0.3
"Default | UpdateKey": "Nexus:572",
"~1.0.2 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Pony Weight Loss Program
+
+ "Pony Weight Loss Program": {
"ID": "BadNetCode.PonyWeightLossProgram",
"Default | UpdateKey": "Nexus:1232"
},
- {
- // Portraiture
+
+ "Portraiture": {
"ID": "Platonymous.Portraiture",
"Default | UpdateKey": "Nexus:999" // added in 1.3.1
},
- {
- // Prairie King Made Easy
+
+ "Prairie King Made Easy": {
"ID": "Mucchan.PrairieKingMadeEasy",
"FormerIDs": "{EntryDll: 'PrairieKingMadeEasy.dll'}", // changed in 1.0.1
"Default | UpdateKey": "Chucklefish:3594",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Quest Delay
+
+ "Quest Delay": {
"ID": "BadNetCode.QuestDelay",
"Default | UpdateKey": "Nexus:1239"
},
- {
- // Rain Randomizer
+
+ "Rain Randomizer": {
"ID": "{EntryDll: 'RainRandomizer.dll'}",
"~1.0.3 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.3 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Recatch Legendary Fish
+
+ "Recatch Legendary Fish": {
"ID": "cantorsdust.RecatchLegendaryFish",
"FormerIDs": "b3af8c31-48f0-43cf-8343-3eb08bcfa1f9 | community.RecatchLegendaryFish", // changed in 1.3 and 1.5.1
"Default | UpdateKey": "Nexus:172"
},
- {
- // Regeneration
+
+ "Regeneration": {
"ID": "HammurabiRegeneration",
"Default | UpdateKey": "Chucklefish:4584"
},
- {
- // Relationship Bar UI
+
+ "Relationship Bar UI": {
"ID": "RelationshipBar",
"Default | UpdateKey": "Nexus:1009"
},
- {
- // RelationshipsEnhanced
+
+ "RelationshipsEnhanced": {
"ID": "relationshipsenhanced",
"Default | UpdateKey": "Chucklefish:4435",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Relationship Status
+
+ "Relationship Status": {
"ID": "relationshipstatus",
"MapRemoteVersions": { "1.0.5": "1.0.4" }, // not updated in manifest
"Default | UpdateKey": "Nexus:751",
"~1.0.5 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.5 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Rented Tools
+
+ "Rented Tools": {
"ID": "JarvieK.RentedTools",
"Default | UpdateKey": "Nexus:1307"
},
- {
- // Replanter
+
+ "Replanter": {
"ID": "jwdred.Replanter",
"FormerIDs": "{EntryDll: 'Replanter.dll'}", // changed in 1.0.5
"Default | UpdateKey": "Nexus:589",
"~1.0.4 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // ReRegeneration
+
+ "ReRegeneration": {
"ID": "lrsk_sdvm_rerg.0925160827",
"MapLocalVersions": { "1.1.2-release": "1.1.2" },
"Default | UpdateKey": "Chucklefish:4465"
},
- {
- // Reseed
+
+ "Reseed": {
"ID": "Roc.Reseed",
"Default | UpdateKey": "Nexus:887"
},
- {
- // Reusable Wallpapers and Floors (Wallpaper Retain)
+
+ "Reusable Wallpapers and Floors (Wallpaper Retain)": {
"ID": "dae1b553-2e39-43e7-8400-c7c5c836134b",
"Default | UpdateKey": "Nexus:356",
"~1.5 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.5 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Ring of Fire
+
+ "Ring of Fire": {
"ID": "Platonymous.RingOfFire",
"Default | UpdateKey": "Nexus:1166" // added in 1.0.1
},
- {
- // Rope Bridge
+
+ "Rope Bridge": {
"ID": "RopeBridge",
"Default | UpdateKey": "Nexus:824"
},
- {
- // Rotate Toolbar
+
+ "Rotate Toolbar": {
"ID": "Pathoschild.RotateToolbar",
"Default | UpdateKey": "Nexus:1100"
},
- {
- // Rush Orders
+
+ "Rush Orders": {
"ID": "spacechase0.RushOrders",
"FormerIDs": "RushOrders", // changed in 1.1
"Default | UpdateKey": "Nexus:605",
"~1.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Save Anywhere
+
+ "Save Anywhere": {
"ID": "Omegasis.SaveAnywhere",
"FormerIDs": "{ID:'SaveAnywhere', Name:'Save Anywhere'}", // changed in 2.5; disambiguate from Night Owl
"Default | UpdateKey": "Nexus:444", // added in 2.6.1
"~2.4 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Save Backup
+
+ "Save Backup": {
"ID": "Omegasis.SaveBackup",
"FormerIDs": "{ID:'4be88c18-b6f3-49b0-ba96-f94b1a5be890', Name:'Stardew_Save_Backup'}", // changed in 1.3; disambiguate from other Alpha_Omegasis mods
"Default | UpdateKey": "Nexus:435", // added in 1.3.1
"~1.2 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Scroll to Blank
+
+ "Scroll to Blank": {
"ID": "caraxian.scroll.to.blank",
"Default | UpdateKey": "Chucklefish:4405"
},
- {
- // Scythe Harvesting
+
+ "Scythe Harvesting": {
"ID": "mmanlapat.ScytheHarvesting",
"FormerIDs": "ScytheHarvesting", // changed in 1.6
"Default | UpdateKey": "Nexus:1106"
},
- {
- // Seasonal Immersion
+
+ "Seasonal Immersion": {
"ID": "Entoarox.SeasonalImmersion",
"FormerIDs": "EntoaroxSeasonalHouse | EntoaroxSeasonalBuildings | EntoaroxSeasonalImmersion", // changed in 1.1, 1.6 or earlier, and 1.7
"~1.11 | UpdateKey": "Chucklefish:4262", // only enable update checks up to 1.11 by request (has its own update-check feature)
"~1.8.2 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Seed Bag
+
+ "Seed Bag": {
"ID": "Platonymous.SeedBag",
"Default | UpdateKey": "Nexus:1133" // added in 1.1.2
},
- {
- // Self Service
+
+ "Self Service": {
"ID": "JarvieK.SelfService",
"MapRemoteVersions": { "0.2.1": "0.2" }, // manifest not updated
"Default | UpdateKey": "Nexus:1304"
},
- {
- // Send Items
+
+ "Send Items": {
"ID": "Denifia.SendItems",
"Default | UpdateKey": "Nexus:1087", // added in 1.0.3 (2017-10-04)
"~1.0.2 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Shed Notifications (BuildingsNotifications)
+
+ "Shed Notifications (BuildingsNotifications)": {
"ID": "TheCroak.BuildingsNotifications",
"Default | UpdateKey": "Nexus:620",
"~0.4.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~0.4.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Shenandoah Project
+
+ "Shenandoah Project": {
"ID": "Shenandoah Project",
"MapRemoteVersions": { "1.1.1": "1.1" }, // not updated in manifest
"Default | UpdateKey": "Nexus:756",
"~1.1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Ship Anywhere
+
+ "Ship Anywhere": {
"ID": "spacechase0.ShipAnywhere",
"Default | UpdateKey": "Nexus:1379"
},
- {
- // Shipment Tracker
+
+ "Shipment Tracker": {
"ID": "7e474181-e1a0-40f9-9c11-d08a3dcefaf3",
"Default | UpdateKey": "Nexus:321",
"~1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Shop Expander
+
+ "Shop Expander": {
"ID": "Entoarox.ShopExpander",
"FormerIDs": "{ID:'821ce8f6-e629-41ad-9fde-03b54f68b0b6', Name:'Shop Expander'} | EntoaroxShopExpander", // changed in 1.5 and 1.5.2; disambiguate from Faster Paths
"~1.5.3 | UpdateKey": "Chucklefish:4381", // only enable update checks up to 1.5.3 by request (has its own update-check feature)
"~1.5.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Showcase Mod
+
+ "Showcase Mod": {
"ID": "Igorious.Showcase",
"MapLocalVersions": { "0.9-500": "0.9" },
"Default | UpdateKey": "Chucklefish:4487",
"~0.9 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~0.9 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Shroom Spotter
+
+ "Shroom Spotter": {
"ID": "TehPers.ShroomSpotter",
"Default | UpdateKey": "Nexus:908"
},
- {
- // Simple Crop Label
+
+ "Simple Crop Label": {
"ID": "SimpleCropLabel",
"Default | UpdateKey": "Nexus:314"
},
- {
- // Simple Sound Manager
+
+ "Simple Sound Manager": {
"ID": "Omegasis.SimpleSoundManager",
"Default | UpdateKey": "Nexus:1410", // added in 1.0.1
"~1.0 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Simple Sprinklers
+
+ "Simple Sprinklers": {
"ID": "tZed.SimpleSprinkler",
"FormerIDs": "{EntryDll: 'SimpleSprinkler.dll'}", // changed in 1.5
"Default | UpdateKey": "Nexus:76",
"~1.4 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Siv's Marriage Mod
+
+ "Siv's Marriage Mod": {
"ID": "6266959802",
"MapLocalVersions": { "0.0": "1.4" },
"Default | UpdateKey": "Nexus:366",
"~1.2.2 | Status": "AssumeBroken", // broke in SMAPI 1.9 (has multiple Mod instances)
"~1.2.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Skill Prestige
+
+ "Skill Prestige": {
"ID": "alphablackwolf.skillPrestige",
"FormerIDs": "6b843e60-c8fc-4a25-a67b-4a38ac8dcf9b", // changed circa 1.2.3
"Default | UpdateKey": "Nexus:569",
"~1.0.9 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Skill Prestige: Cooking Adapter
+
+ "Skill Prestige: Cooking Adapter": {
"ID": "Alphablackwolf.CookingSkillPrestigeAdapter",
"FormerIDs": "20d6b8a3-b6e7-460b-a6e4-07c2b0cb6c63", // changed circa 1.1
"MapRemoteVersions": { "1.2.3": "1.1" }, // manifest not updated
"Default | UpdateKey": "Nexus:569",
"~1.0.9 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Skip Intro
+
+ "Skip Intro": {
"ID": "Pathoschild.SkipIntro",
"FormerIDs": "SkipIntro", // changed in 1.4
"Default | UpdateKey": "Nexus:533"
},
- {
- // Skull Cavern Elevator
+
+ "Skull Cavern Elevator": {
"ID": "SkullCavernElevator",
"Default | UpdateKey": "Nexus:963"
},
- {
- // Skull Cave Saver
+
+ "Skull Cave Saver": {
"ID": "cantorsdust.SkullCaveSaver",
"FormerIDs": "8ac06349-26f7-4394-806c-95d48fd35774 | community.SkullCaveSaver", // changed in 1.1 and 1.2.2
"Default | UpdateKey": "Nexus:175"
},
- {
- // Sleepy Eye
+
+ "Sleepy Eye": {
"ID": "spacechase0.SleepyEye",
"Default | UpdateKey": "Nexus:1152"
},
- {
- // Slower Fence Decay
+
+ "Slower Fence Decay": {
"ID": "Speeder.SlowerFenceDecay",
"FormerIDs": "SPDSlowFenceDecay", // changed in 0.5.2-pathoschild-update
"Default | UpdateKey": "Nexus:252",
"~0.5.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~0.5.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Smart Mod
+
+ "Smart Mod": {
"ID": "KuroBear.SmartMod",
"~2.2 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~2.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Solar Eclipse Event
+
+ "Solar Eclipse Event": {
"ID": "KoihimeNakamura.SolarEclipseEvent",
"Default | UpdateKey": "Nexus:897",
"MapLocalVersions": { "1.3-20170917": "1.3" },
"~1.1 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // SpaceCore
+
+ "SpaceCore": {
"ID": "spacechase0.SpaceCore",
"Default | UpdateKey": "Nexus:1348"
},
- {
- // Speedster
+
+ "Speedster": {
"ID": "Platonymous.Speedster",
"Default | UpdateKey": "Nexus:1102" // added in 1.3.1
},
- {
- // Sprinkler Range
+
+ "Sprinkler Range": {
"ID": "cat.sprinklerrange",
"MapRemoteVersions": { "1.0.1": "1.0" }, // manifest not updated
"Default | UpdateKey": "Nexus:1179"
},
- {
- // Sprinkles
+
+ "Sprinkles": {
"ID": "Platonymous.Sprinkles",
"Default | UpdateKey": "Chucklefish:4592",
"~1.1.3 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1.3 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Sprint and Dash
+
+ "Sprint and Dash": {
"ID": "SPDSprintAndDash",
"Default | UpdateKey": "Chucklefish:3531",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Sprint and Dash Redux
+
+ "Sprint and Dash Redux": {
"ID": "littleraskol.SprintAndDashRedux",
"FormerIDs": "lrsk_sdvm_sndr.0921161059", // changed in 1.3
"Default | UpdateKey": "Chucklefish:4201"
},
- {
- // Sprinting Mod
+
+ "Sprinting Mod": {
"ID": "a10d3097-b073-4185-98ba-76b586cba00c",
"MapLocalVersions": { "1.0": "2.1" }, // not updated in manifest
"Default | UpdateKey": "GitHub:oliverpl/SprintingMod",
"~2.1 | Status": "AssumeBroken", // broke in SDV 1.2
"~2.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // StackSplitX
+
+ "StackSplitX": {
"ID": "tstaples.StackSplitX",
"FormerIDs": "{EntryDll: 'StackSplitX.dll'}", // changed circa 1.3.1
"Default | UpdateKey": "Nexus:798",
"~1.2 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // StaminaRegen
+
+ "StaminaRegen": {
"ID": "{EntryDll: 'StaminaRegen.dll'}",
"~1.0.3 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.3 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Stardew Config Menu
+
+ "Stardew Config Menu": {
"ID": "Juice805.StardewConfigMenu",
"Default | UpdateKey": "Nexus:1312"
},
- {
- // Stardew Content Compatibility Layer (SCCL)
+
+ "Stardew Content Compatibility Layer (SCCL)": {
"ID": "SCCL",
"Default | UpdateKey": "Nexus:889",
"~0.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Stardew Editor Game Integration
+
+ "Stardew Editor Game Integration": {
"ID": "spacechase0.StardewEditor.GameIntegration",
"Default | UpdateKey": "Nexus:1298"
},
- {
- // Stardew Notification
+
+ "Stardew Notification": {
"ID": "stardewnotification",
"Default | UpdateKey": "GitHub:monopandora/StardewNotification",
"~1.7 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.7 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Stardew Symphony
+
+ "Stardew Symphony": {
"ID": "Omegasis.StardewSymphony",
"FormerIDs": "{ID:'4108e859-333c-4fec-a1a7-d2e18c1019fe', Name:'Stardew_Symphony'}", // changed in 1.4; disambiguate other mods by Alpha_Omegasis
"Default | UpdateKey": "Nexus:425", // added in 1.4.1
"~1.3 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // StarDustCore
+
+ "StarDustCore": {
"ID": "StarDustCore",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "it was only used by earlier versions of Save Anywhere, and is no longer used or maintained."
},
- {
- // Starting Money
+
+ "Starting Money": {
"ID": "mmanlapat.StartingMoney",
"FormerIDs": "StartingMoney", // changed in 1.1
"Default | UpdateKey": "Nexus:1138"
},
- {
- // StashItemsToChest
+
+ "StashItemsToChest": {
"ID": "BlueMod_StashItemsToChest",
"Default | UpdateKey": "GitHub:lambui/StardewValleyMod_StashItemsToChest",
"~1.0.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Stephan's Lots of Crops
+
+ "Stephan's Lots of Crops": {
"ID": "stephansstardewcrops",
"MapRemoteVersions": { "1.41": "1.1" }, // manifest not updated
"Default | UpdateKey": "Chucklefish:4314"
},
- {
- // Stone Bridge Over Pond (PondWithBridge)
+
+ "Stone Bridge Over Pond (PondWithBridge)": {
"ID": "{EntryDll: 'PondWithBridge.dll'}",
"MapLocalVersions": { "0.0": "1.0" },
"Default | UpdateKey": "Nexus:316",
"~1.0 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // Stumps to Hardwood Stumps
+
+ "Stumps to Hardwood Stumps": {
"ID": "StumpsToHardwoodStumps",
"Default | UpdateKey": "Nexus:691"
},
- {
- // Super Greenhouse Warp Modifier
+
+ "Super Greenhouse Warp Modifier": {
"ID": "SuperGreenhouse",
"Default | UpdateKey": "Chucklefish:4334",
"~1.0 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Swim Almost Anywhere / Swim Suit
+
+ "Swim Almost Anywhere / Swim Suit": {
"ID": "Platonymous.SwimSuit",
"Default | UpdateKey": "Nexus:1215" // added in 0.5.1
},
- {
- // Tainted Cellar
+
+ "Tainted Cellar": {
"ID": "{EntryDll: 'TaintedCellar.dll'}",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.1 or 1.11
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Tapper Ready
+
+ "Tapper Ready": {
"ID": "skunkkk.TapperReady",
"Default | UpdateKey": "Nexus:1219"
},
- {
- // Teh's Fishing Overhaul
+
+ "Teh's Fishing Overhaul": {
"ID": "TehPers.FishingOverhaul",
"Default | UpdateKey": "Nexus:866"
},
- {
- // Teleporter
+
+ "Teleporter": {
"ID": "Teleporter",
"Default | UpdateKey": "Chucklefish:4374",
"~1.0.2 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // The Long Night
+
+ "The Long Night": {
"ID": "Pathoschild.TheLongNight",
"Default | UpdateKey": "Nexus:1369"
},
- {
- // Three-heart Dance Partner
+
+ "Three-heart Dance Partner": {
"ID": "ThreeHeartDancePartner",
"Default | UpdateKey": "Nexus:500",
"~1.0.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // TimeFreeze
+
+ "TimeFreeze": {
"ID": "Omegasis.TimeFreeze",
"FormerIDs": "4108e859-333c-4fec-a1a7-d2e18c1019fe", // changed in 1.2
"Default | UpdateKey": "Nexus:973" // added in 1.2.1
},
- {
- // Time Reminder
+
+ "Time Reminder": {
"ID": "KoihimeNakamura.TimeReminder",
"MapLocalVersions": { "1.0-20170314": "1.0.2" },
"Default | UpdateKey": "Nexus:1000"
},
- {
- // TimeSpeed
+
+ "TimeSpeed": {
"ID": "cantorsdust.TimeSpeed",
"FormerIDs": "{EntryDll: 'TimeSpeed.dll'} | {ID:'4108e859-333c-4fec-a1a7-d2e18c1019fe', Name:'TimeSpeed'} | {ID:'4108e859-333c-4fec-a1a7-d2e18c1019fe', Name:'TimeSpeed Mod (unofficial)'} | community.TimeSpeed", // changed in 2.0.3, 2.1, and 2.3.3; disambiguate other mods by Alpha_Omegasis
"Default | UpdateKey": "Nexus:169",
"~2.2 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // TractorMod
+
+ "TractorMod": {
"ID": "Pathoschild.TractorMod",
"FormerIDs": "BlueMod_TractorMod | PhthaloBlue.TractorMod | community.TractorMod", // changed in 3.2, 4.0 beta, and 4.0
"Default | UpdateKey": "Nexus:1401"
},
- {
- // TrainerMod
+
+ "TrainerMod": {
"ID": "SMAPI.TrainerMod",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "replaced by ConsoleCommands, which is added by the SMAPI installer."
},
- {
- // Tree Transplant
+
+ "Tree Transplant": {
"ID": "TreeTransplant",
"Default | UpdateKey": "Nexus:1342"
},
- {
- // UI Info Suite
+
+ "UI Info Suite": {
"ID": "Cdaragorn.UiInfoSuite",
"Default | UpdateKey": "Nexus:1150"
},
- {
- // UiModSuite
+
+ "UiModSuite": {
"ID": "Demiacle.UiModSuite",
"MapLocalVersions": { "0.5": "1.0" }, // not updated in manifest
"Default | UpdateKey": "Nexus:1023",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Variable Grass
+
+ "Variable Grass": {
"ID": "dantheman999.VariableGrass",
"Default | UpdateKey": "GitHub:dantheman999301/StardewMods"
},
- {
- // Vertical Toolbar
+
+ "Vertical Toolbar": {
"ID": "SB_VerticalToolMenu",
"Default | UpdateKey": "Nexus:943"
},
- {
- // WakeUp
+
+ "WakeUp": {
"ID": "{EntryDll: 'WakeUp.dll'}",
"~1.0.2 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.0.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // Wallpaper Fix
+
+ "Wallpaper Fix": {
"ID": "{EntryDll: 'WallpaperFix.dll'}",
"Default | UpdateKey": "Chucklefish:4211",
"~1.1 | Status": "AssumeBroken", // broke in SMAPI 2.0
"~1.1 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // WarpAnimals
+
+ "WarpAnimals": {
"ID": "Symen.WarpAnimals",
"Default | UpdateKey": "Nexus:1400"
},
- {
- // Weather Controller
+
+ "Weather Controller": {
"ID": "{EntryDll: 'WeatherController.dll'}",
"~1.0.2 | Status": "AssumeBroken", // broke in SDV 1.2
"~1.0.2 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // What Farm Cave / WhatAMush
+
+ "What Farm Cave / WhatAMush": {
"ID": "WhatAMush",
"Default | UpdateKey": "Nexus:1097"
},
- {
- // WHats Up
+
+ "WHats Up": {
"ID": "wHatsUp",
"Default | UpdateKey": "Nexus:1082"
},
- {
- // Wonderful Farm Life
+
+ "Wonderful Farm Life": {
"ID": "{EntryDll: 'WonderfulFarmLife.dll'}",
"~1.0 | Status": "AssumeBroken", // broke in SDV 1.1 or 1.11
"~1.0 | AlternativeUrl": "http://stardewvalleywiki.com/Modding:SMAPI_2.0"
},
- {
- // XmlSerializerRetool
+
+ "XmlSerializerRetool": {
"ID": "{EntryDll: 'XmlSerializerRetool.dll'}",
"~ | Status": "Obsolete",
"~ | StatusReasonPhrase": "it's no longer maintained or used."
},
- {
- // Xnb Loader
+
+ "Xnb Loader": {
"ID": "Entoarox.XnbLoader",
"~1.1.10 | UpdateKey": "Chucklefish:4506", // only enable update checks up to 1.1.10 by request (has its own update-check feature)
"~1.0.6 | Status": "AssumeBroken" // broke in SMAPI 2.0
},
- {
- // zDailyIncrease
+
+ "zDailyIncrease": {
"ID": "zdailyincrease",
"MapRemoteVersions": { "1.3.5": "1.3.4" }, // not updated in manifest
"Default | UpdateKey": "Chucklefish:4247",
"~1.2 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoom Out Extreme
+
+ "Zoom Out Extreme": {
"ID": "RockinMods.ZoomMod",
"FormerIDs": "ZoomMod", // changed circa 1.2.1
"Default | UpdateKey": "Nexus:1326",
"~0.1 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoryn's Better RNG
+
+ "Zoryn's Better RNG": {
"ID": "Zoryn.BetterRNG",
"FormerIDs": "76b6d1e1-f7ba-4d72-8c32-5a1e6d2716f6", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods",
"~1.6 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoryn's Calendar Anywhere
+
+ "Zoryn's Calendar Anywhere": {
"ID": "Zoryn.CalendarAnywhere",
"FormerIDs": "a41c01cd-0437-43eb-944f-78cb5a53002a", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods",
"~1.6 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoryn's Durable Fences
+
+ "Zoryn's Durable Fences": {
"ID": "Zoryn.DurableFences",
"FormerIDs": "56d3439c-7b9b-497e-9496-0c4890e8a00e", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods"
},
- {
- // Zoryn's Health Bars
+
+ "Zoryn's Health Bars": {
"ID": "Zoryn.HealthBars",
"FormerIDs": "{EntryDll: 'HealthBars.dll'}", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods",
"~1.6 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoryn's Fishing Mod
+
+ "Zoryn's Fishing Mod": {
"ID": "Zoryn.FishingMod",
"FormerIDs": "fa277b1f-265e-47c3-a84f-cd320cc74949", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods"
},
- {
- // Zoryn's Junimo Deposit Anywhere
+
+ "Zoryn's Junimo Deposit Anywhere": {
"ID": "Zoryn.JunimoDepositAnywhere",
"FormerIDs": "f93a4fe8-cade-4146-9335-b5f82fbbf7bc", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods",
"~1.7 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoryn's Movement Mod
+
+ "Zoryn's Movement Mod": {
"ID": "Zoryn.MovementModifier",
"FormerIDs": "8a632929-8335-484f-87dd-c29d2ba3215d", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods",
"~1.6 | Status": "AssumeBroken" // broke in SDV 1.2
},
- {
- // Zoryn's Regen Mod
+
+ "Zoryn's Regen Mod": {
"ID": "Zoryn.RegenMod",
"FormerIDs": "dfac4383-1b6b-4f33-ae4e-37fc23e5252e", // changed in 1.6
"Default | UpdateKey": "GitHub:Zoryn4163/SMAPI-Mods",
"~1.6 | Status": "AssumeBroken" // broke in SDV 1.2
}
- ]
+ }
}
diff --git a/src/SMAPI/StardewModdingAPI.csproj b/src/SMAPI/StardewModdingAPI.csproj
index 129c88b0..e181c435 100644
--- a/src/SMAPI/StardewModdingAPI.csproj
+++ b/src/SMAPI/StardewModdingAPI.csproj
@@ -89,9 +89,10 @@
<Compile Include="Framework\Input\InputState.cs" />
<Compile Include="Framework\Input\InputStatus.cs" />
<Compile Include="Framework\LegacyManifestVersion.cs" />
- <Compile Include="Framework\Models\ModDataField.cs" />
- <Compile Include="Framework\Models\ModDataFieldKey.cs" />
- <Compile Include="Framework\Models\ParsedModDataRecord.cs" />
+ <Compile Include="Framework\ModData\ModDatabase.cs" />
+ <Compile Include="Framework\ModData\ModDataField.cs" />
+ <Compile Include="Framework\ModData\ModDataFieldKey.cs" />
+ <Compile Include="Framework\ModData\ParsedModDataRecord.cs" />
<Compile Include="Framework\ModLoading\Finders\EventFinder.cs" />
<Compile Include="Framework\ModLoading\Finders\FieldFinder.cs" />
<Compile Include="Framework\ModLoading\Finders\MethodFinder.cs" />
@@ -176,7 +177,7 @@
<Compile Include="Framework\Logging\ConsoleInterceptionManager.cs" />
<Compile Include="Framework\Logging\InterceptingTextWriter.cs" />
<Compile Include="Framework\Models\ManifestDependency.cs" />
- <Compile Include="Framework\Models\ModStatus.cs" />
+ <Compile Include="Framework\ModData\ModStatus.cs" />
<Compile Include="Framework\Models\SConfig.cs" />
<Compile Include="Framework\ModLoading\ModMetadata.cs" />
<Compile Include="Framework\Reflection\ReflectedProperty.cs" />
@@ -204,7 +205,7 @@
<Compile Include="Framework\DeprecationLevel.cs" />
<Compile Include="Framework\DeprecationManager.cs" />
<Compile Include="Framework\InternalExtensions.cs" />
- <Compile Include="Framework\Models\ModDataRecord.cs" />
+ <Compile Include="Framework\ModData\ModDataRecord.cs" />
<Compile Include="Framework\ModLoading\AssemblyLoader.cs" />
<Compile Include="Framework\Reflection\CacheEntry.cs" />
<Compile Include="Framework\Reflection\ReflectedField.cs" />