-
-
Notifications
You must be signed in to change notification settings - Fork 386
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add trigger action to migrate legacy content IDs
- Loading branch information
1 parent
3783ca8
commit ed263a9
Showing
5 changed files
with
415 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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
346
ContentPatcher/Framework/TriggerActions/MigrateIdsAction.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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; | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.