Skip to content

Commit

Permalink
add trigger action to migrate legacy content IDs
Browse files Browse the repository at this point in the history
  • Loading branch information
Pathoschild committed Mar 9, 2024
1 parent 3783ca8 commit ed263a9
Show file tree
Hide file tree
Showing 5 changed files with 415 additions and 0 deletions.
24 changes: 24 additions & 0 deletions ContentPatcher/Framework/TriggerActions/MigrateIdType.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
namespace ContentPatcher.Framework.TriggerActions
{
/// <summary>A data ID type which can be migrated using <see cref="MigrateIdsAction"/>.</summary>
public enum MigrateIdType
{
/// <summary>Migrate cooking recipe IDs.</summary>
CookingRecipes,

/// <summary>Migrate crafting recipe IDs.</summary>
CraftingRecipes,

/// <summary>Migrate event IDs.</summary>
Events,

/// <summary>Migrate item local IDs.</summary>
Items,

/// <summary>Migrate mail IDs.</summary>
Mail,

/// <summary>Migrate songs-heard cue names.</summary>
Songs
}
}
346 changes: 346 additions & 0 deletions ContentPatcher/Framework/TriggerActions/MigrateIdsAction.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,346 @@
using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using StardewValley;
using StardewValley.Delegates;
using StardewValley.ItemTypeDefinitions;
using StardewValley.Triggers;

namespace ContentPatcher.Framework.TriggerActions
{
/// <summary>Implements the <c>Pathoschild.ContentPatcher_MigrateIds</c> trigger action.</summary>
internal class MigrateIdsAction
{
/*********
** Public methods
*********/
/// <summary>Handle the action when it's called by the game.</summary>
/// <inheritdoc cref="TriggerActionDelegate"/>
public bool Handle(string[] args, TriggerActionContext context, [NotNullWhen(false)] out string? error)
{
// validate context
// We need to migrate IDs everywhere, including in non-synced locations and on farmhand fields that can't
// be edited remotely. That's only possible when run on the host before any other players have connected.
if (context.Data is null)
{
error = "this action must be run via Data/TriggerActions";
return false;
}
if (!context.Data.HostOnly || !string.Equals(context.Data.Trigger?.Trim(), TriggerActionManager.trigger_dayStarted))
{
error = $"this action must be run with `\"{nameof(context.Data.HostOnly)}\": true` and `\"{nameof(context.Data.Trigger)}: \"{TriggerActionManager.trigger_dayStarted}\"`";
return false;
}

// get ID type
if (!ArgUtility.TryGetEnum(args, 1, out MigrateIdType type, out error))
return false;

// get old => new IDs
var mapIds = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
for (int i = 2; i < args.Length; i += 2)
{
if (!ArgUtility.TryGet(args, i, out string oldId, out error, allowBlank: false))
return false;
if (!ArgUtility.TryGet(args, i + 1, out string newId, out error, allowBlank: false))
{
if (!ArgUtility.HasIndex(args, i + 1))
error = $"index {i} with old ID \"{oldId}\" doesn't have a corresponding new ID at index {i + 1}";
return false;
}

mapIds[oldId] = newId;
}

// apply
Farmer[] players = Game1.getAllFarmers().ToArray();
switch (type)
{
case MigrateIdType.CookingRecipes:
return this.TryMigrateCookingRecipeIds(players, mapIds, out error);

case MigrateIdType.CraftingRecipes:
return this.TryMigrateCraftingRecipeIds(players, mapIds, out error);

case MigrateIdType.Events:
return this.TryMigrateEventIds(players, mapIds, out error);

case MigrateIdType.Items:
return this.TryMigrateItemIds(mapIds, out error);

case MigrateIdType.Mail:
return this.TryMigrateMailIds(players, mapIds, out error);

case MigrateIdType.Songs:
return this.TryMigrateSongIds(players, mapIds, out error);

default:
error = $"required index 1 has unknown ID type '{type}'";
return false;
}


}


/*********
** Private methods
*********/
/// <summary>Try to migrate cooking recipe IDs.</summary>
/// <param name="players">The players to edit.</param>
/// <param name="mapIds">The old and new IDs to map.</param>
/// <param name="error">An error indicating why the migration failed.</param>
private bool TryMigrateCookingRecipeIds(IEnumerable<Farmer> players, IDictionary<string, string> mapIds, [NotNullWhen(false)] out string? error)
{
foreach (Farmer player in players)
{
// note: we iterate deliberately so keys are matched case-insensitively

foreach ((string oldKey, int oldValue) in player.cookingRecipes.Pairs.ToArray())
{
if (mapIds.TryGetValue(oldKey, out string? newKey))
{
player.cookingRecipes.Remove(oldKey);
player.cookingRecipes.TryAdd(newKey, oldValue);
}
}

foreach ((string oldKey, int oldValue) in player.craftingRecipes.Pairs.ToArray())
{
if (mapIds.TryGetValue(oldKey, out string? newKey))
{
player.craftingRecipes.Remove(oldKey);
player.craftingRecipes.TryAdd(newKey, oldValue);
}
}
}

error = null;
return true;
}

/// <summary>Try to migrate crafting recipe IDs.</summary>
/// <param name="players">The players to edit.</param>
/// <param name="mapIds">The old and new IDs to map.</param>
/// <param name="error">An error indicating why the migration failed.</param>
private bool TryMigrateCraftingRecipeIds(IEnumerable<Farmer> players, IDictionary<string, string> mapIds, [NotNullWhen(false)] out string? error)
{
foreach (Farmer player in players)
{
foreach ((string oldKey, int oldValue) in player.craftingRecipes.Pairs.ToArray())
{
if (mapIds.TryGetValue(oldKey, out string? newKey))
{
player.craftingRecipes.Remove(oldKey);
player.craftingRecipes.TryAdd(newKey, oldValue);
}
}
}

error = null;
return true;
}

/// <summary>Try to migrate event IDs.</summary>
/// <param name="players">The players to edit.</param>
/// <param name="mapIds">The old and new IDs to map.</param>
/// <param name="error">An error indicating why the migration failed.</param>
private bool TryMigrateEventIds(IEnumerable<Farmer> players, IDictionary<string, string> mapIds, [NotNullWhen(false)] out string? error)
{
foreach (Farmer player in players)
{
foreach (string oldId in player.eventsSeen.ToArray())
{
if (mapIds.TryGetValue(oldId, out string? newId))
{
player.eventsSeen.Remove(oldId);
player.eventsSeen.Add(newId);
}
}
}

error = null;
return true;
}

/// <summary>Try to migrate item IDs.</summary>
/// <param name="mapRawIds">The old and new IDs to map.</param>
/// <param name="error">An error indicating why the migration failed.</param>
private bool TryMigrateItemIds(IDictionary<string, string> mapRawIds, [NotNullWhen(false)] out string? error)
{
// validate & index item IDs
var mapQualifiedIds = new Dictionary<string, ItemMetadata>();
var mapLocalIds = new Dictionary<string, ItemMetadata>();
foreach ((string oldId, string newId) in mapRawIds)
{
if (!ItemRegistry.IsQualifiedItemId(oldId))
{
error = $"the old item ID \"{oldId}\" must be a qualified item ID (like {ItemRegistry.type_object}{oldId})";
return false;
}

ItemMetadata data = ItemRegistry.ResolveMetadata(newId);
if (data is null)
{
error = $"the new item ID \"{newId}\" doesn't match an existing item";
return false;
}

mapQualifiedIds[data.QualifiedItemId] = data;
mapLocalIds[data.LocalItemId] = data;
}

// migrate items
Utility.ForEachItem(item =>
{
if (mapQualifiedIds.TryGetValue(item.QualifiedItemId, out ItemMetadata? data))
item.ItemId = data.LocalItemId;

return true;
});

// migrate indirect references
foreach (Farmer player in Game1.getAllFarmers())
{
// artifacts (unqualified IDs)
foreach ((string oldId, int[] oldValue) in player.archaeologyFound.Pairs.ToArray())
{
if (mapLocalIds.TryGetValue(oldId, out ItemMetadata? data))
{
player.archaeologyFound.Remove(oldId);
player.archaeologyFound.TryAdd(data.LocalItemId, oldValue);
}
}

// fish caught (qualified IDs)
foreach ((string oldId, int[] oldValue) in player.fishCaught.Pairs.ToArray())
{
if (mapQualifiedIds.TryGetValue(oldId, out ItemMetadata? data))
{
player.fishCaught.Remove(oldId);
player.fishCaught.TryAdd(data.QualifiedItemId, oldValue);
}
}

// gifted items (unqualified IDs)
foreach (SerializableDictionary<string, int> giftedItems in player.giftedItems.Values)
{
foreach ((string oldId, int oldValue) in giftedItems.ToArray())
{
if (mapLocalIds.TryGetValue(oldId, out ItemMetadata? data))
{
giftedItems.Remove(oldId);
giftedItems.TryAdd(data.LocalItemId, oldValue);
}
}
}

// minerals (unqualified IDs)
foreach ((string oldId, int oldValue) in player.mineralsFound.Pairs.ToArray())
{
if (mapLocalIds.TryGetValue(oldId, out ItemMetadata? data))
{
player.mineralsFound.Remove(oldId);
player.mineralsFound.TryAdd(data.LocalItemId, oldValue);
}
}

// shipped (unqualified IDs)
foreach ((string oldId, int oldValue) in player.basicShipped.Pairs.ToArray())
{
if (mapLocalIds.TryGetValue(oldId, out ItemMetadata? data))
{
player.basicShipped.Remove(oldId);
player.basicShipped.TryAdd(data.LocalItemId, oldValue);
}
}

// tailored (IDs in legacy 'standard description' format)
foreach ((string oldTailoredId, int oldValue) in player.tailoredItems.Pairs.ToArray())
{
#pragma warning disable CS0618 // deliberately using obsolete methods used by tailoredItems

Item oldItem = Utility.getItemFromStandardTextDescription(oldTailoredId, Game1.player);

if (oldItem != null && mapQualifiedIds.TryGetValue(oldItem.QualifiedItemId, out ItemMetadata? data))
{
string newTailoredId = Utility.getStandardDescriptionFromItem(data.TypeIdentifier, data.LocalItemId, false, false, 1);

player.tailoredItems.Remove(oldTailoredId);
player.tailoredItems.TryAdd(newTailoredId, oldValue);
}
#pragma warning restore CS0618
}
}

error = null;
return true;
}

/// <summary>Try to migrate mail IDs.</summary>
/// <param name="players">The players to edit.</param>
/// <param name="mapIds">The old and new IDs to map.</param>
/// <param name="error">An error indicating why the migration failed.</param>
private bool TryMigrateMailIds(IEnumerable<Farmer> players, IDictionary<string, string> mapIds, [NotNullWhen(false)] out string? error)
{
foreach (Farmer player in players)
{
// received
foreach (string oldId in player.mailReceived.ToArray())
{
if (mapIds.TryGetValue(oldId, out string? newId))
{
player.mailReceived.Remove(oldId);
player.mailReceived.Add(newId);
}
}

// in mailbox
for (int i = 0; i < player.mailbox.Count; i++)
{
if (mapIds.TryGetValue(player.mailbox[i], out string? newId))
{
player.mailbox.RemoveAt(i);
player.mailbox.Insert(i, newId);
}
}

// queued for tomorrow
foreach (string oldId in player.mailForTomorrow.ToArray())
{
if (mapIds.TryGetValue(oldId, out string? newId))
{
player.mailForTomorrow.Remove(oldId);
player.mailForTomorrow.Add(newId);
}
}
}

error = null;
return true;
}

/// <summary>Try to migrate song IDs.</summary>
/// <param name="players">The players to edit.</param>
/// <param name="mapIds">The old and new IDs to map.</param>
/// <param name="error">An error indicating why the migration failed.</param>
private bool TryMigrateSongIds(IEnumerable<Farmer> players, IDictionary<string, string> mapIds, [NotNullWhen(false)] out string? error)
{
foreach (Farmer player in players)
{
foreach (string oldId in player.songsHeard.ToArray())
{
if (mapIds.TryGetValue(oldId, out string? newId))
{
player.songsHeard.Remove(oldId);
player.songsHeard.Add(newId);
}
}
}

error = null;
return true;
}
}
}
6 changes: 6 additions & 0 deletions ContentPatcher/ModEntry.cs
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,15 @@
using ContentPatcher.Framework.Migrations;
using ContentPatcher.Framework.Patches;
using ContentPatcher.Framework.Tokens;
using ContentPatcher.Framework.TriggerActions;
using ContentPatcher.Framework.Validators;
using Pathoschild.Stardew.Common;
using Pathoschild.Stardew.Common.Utilities;
using StardewModdingAPI;
using StardewModdingAPI.Events;
using StardewModdingAPI.Utilities;
using StardewValley;
using StardewValley.Triggers;
using TokenParser = ContentPatcher.Framework.TokenParser;

[assembly: InternalsVisibleTo("Pathoschild.Stardew.Tests.Mods")]
Expand Down Expand Up @@ -287,6 +289,10 @@ group token by token.Mod into modGroup
helper.Events.Player.Warped += this.OnWarped;
helper.Events.Specialized.LoadStageChanged += this.OnLoadStageChanged;

// set up trigger actions
// (This needs to happen before content packs are loaded below, since they may use these.)
TriggerActionManager.RegisterAction($"{this.ModManifest.UniqueID}_MigrateIds", new MigrateIdsAction().Handle);

// load screen manager
this.InitializeScreenManagerIfNeeded(this.ContentPacks);

Expand Down
Loading

0 comments on commit ed263a9

Please sign in to comment.