using System;
using System.Collections.Generic;
using System.Linq;
namespace StardewModdingAPI.Toolkit.Framework.ModData
{
/// The parsed mod metadata from SMAPI's internal mod list.
public class ModDataRecord
{
/*********
** Accessors
*********/
/// The mod's default display name.
public string DisplayName { get; }
/// The mod's current unique ID.
public string ID { get; }
/// The former mod IDs (if any).
public string[] FormerIDs { get; }
/// Maps local versions to a semantic version for update checks.
public IDictionary MapLocalVersions { get; }
/// Maps remote versions to a semantic version for update checks.
public IDictionary MapRemoteVersions { get; }
/// The versioned field data.
public ModDataField[] Fields { get; }
/*********
** Public methods
*********/
/// Construct an instance.
/// The mod's default display name.
/// The raw data model.
internal ModDataRecord(string displayName, ModDataModel model)
{
this.DisplayName = displayName;
this.ID = model.ID;
this.FormerIDs = model.GetFormerIDs().ToArray();
this.MapLocalVersions = new Dictionary(model.MapLocalVersions, StringComparer.InvariantCultureIgnoreCase);
this.MapRemoteVersions = new Dictionary(model.MapRemoteVersions, StringComparer.InvariantCultureIgnoreCase);
this.Fields = model.GetFields().ToArray();
}
/// Get whether the mod has (or previously had) the given ID.
/// The mod ID.
public bool HasID(string id)
{
// try main ID
if (this.ID.Equals(id, StringComparison.InvariantCultureIgnoreCase))
return true;
// try former IDs
foreach (string formerID in this.FormerIDs)
{
if (formerID.Equals(id, StringComparison.InvariantCultureIgnoreCase))
return true;
}
return false;
}
/// Get a semantic local version for update checks.
/// The remote version to normalise.
public ISemanticVersion GetLocalVersionForUpdateChecks(ISemanticVersion version)
{
return this.MapLocalVersions != null && this.MapLocalVersions.TryGetValue(version.ToString(), out string newVersion)
? new SemanticVersion(newVersion)
: version;
}
/// Get a semantic remote version for update checks.
/// The remote version to normalise.
public string GetRemoteVersionForUpdateChecks(string version)
{
// normalise version if possible
if (SemanticVersion.TryParse(version, out ISemanticVersion parsed))
version = parsed.ToString();
// fetch remote version
return this.MapRemoteVersions != null && this.MapRemoteVersions.TryGetValue(version, out string newVersion)
? newVersion
: version;
}
/// Get the possible mod IDs.
public IEnumerable GetIDs()
{
return this.FormerIDs
.Concat(new[] { this.ID })
.Where(p => !string.IsNullOrWhiteSpace(p))
.Select(p => p.Trim())
.Distinct();
}
/// Get a parsed representation of the which match a given manifest.
/// The manifest to match.
public ModDataRecordVersionedFields GetVersionedFields(IManifest manifest)
{
ModDataRecordVersionedFields parsed = new ModDataRecordVersionedFields { DisplayName = this.DisplayName, DataRecord = this };
foreach (ModDataField field in this.Fields.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;
}
}
}