From 42bc500a77f4b29d952058aede6abfaf91bd74c8 Mon Sep 17 00:00:00 2001 From: Star Rauchenberger Date: Sun, 23 Feb 2025 16:42:22 -0500 Subject: Add support for lasers The three lasers in Blue are item locked, and the laser in Akshardham requires you to have planted all other god cubes in order to activate. --- ArchipelagoManager.cs | 1 + GameData.cs | 51 +++++++++++++++++++++++++++++++++++++++++++++++++++ GameState.cs | 46 ++++++++++++++++++++++++++++++++++++++++++++++ GameplayPatches.cs | 48 ++++++++++++++++++++++++++++++++++++++++++++++++ game_data.yaml | 31 +++++++++++++++++++++++++++++-- 5 files changed, 175 insertions(+), 2 deletions(-) diff --git a/ArchipelagoManager.cs b/ArchipelagoManager.cs index 998e2d9..5e1d867 100644 --- a/ArchipelagoManager.cs +++ b/ArchipelagoManager.cs @@ -29,6 +29,7 @@ namespace ManifoldGardenArchipelago } catch (Exception ex) { + Plugin.Logger.LogError(ex.GetBaseException().Message); return new LoginFailure(ex.GetBaseException().Message); } diff --git a/GameData.cs b/GameData.cs index 6b5eddc..f8adb17 100644 --- a/GameData.cs +++ b/GameData.cs @@ -27,6 +27,7 @@ namespace ManifoldGardenArchipelago public readonly Dictionary doors = []; public readonly Dictionary smokeWalls = []; public readonly Dictionary worldGrows = []; + public readonly Dictionary lasers = []; } public class GameStateListeners @@ -35,6 +36,7 @@ namespace ManifoldGardenArchipelago public readonly Dictionary doors = []; public readonly Dictionary smokeWalls = []; public readonly Dictionary worldGrows = []; + public readonly Dictionary lasers = []; } public static class GameData @@ -419,6 +421,8 @@ namespace ManifoldGardenArchipelago Requirement req = ParseRequirement(scenePair.Key, (Dictionary)worldPair.Value); sceneDescription.worldGrows[sir.index] = req; + Plugin.Logger.LogInfo($"World {sir} requirements: {req}"); + foreach (var reqScene in req.references.scenes) { GetOrAddListeners(listenersByScene, reqScene).worldGrows[sir] = req; @@ -456,6 +460,53 @@ namespace ManifoldGardenArchipelago } } + if (sceneDetails.ContainsKey("lasers")) + { + foreach (var laserPair in (Dictionary)sceneDetails["lasers"]) + { + SceneItemReference sir = new(scenePair.Key, int.Parse((string)laserPair.Key)); + Requirement req = ParseRequirement(scenePair.Key, (Dictionary)laserPair.Value); + sceneDescription.lasers[sir.index] = req; + + Plugin.Logger.LogInfo($"Laser {sir} requirements: {req}"); + + foreach (var reqScene in req.references.scenes) + { + GetOrAddListeners(listenersByScene, reqScene).lasers[sir] = req; + } + + foreach (var button in req.references.buttons) + { + GetOrAddListeners(listenersByButton, button).lasers[sir] = req; + } + + foreach (var socket in req.references.sockets) + { + GetOrAddListeners(listenersBySocket, socket).lasers[sir] = req; + } + + foreach (var pad in req.references.pads) + { + GetOrAddListeners(listenersByPad, pad).lasers[sir] = req; + } + + foreach (var waterwheel in req.references.waterwheels) + { + GetOrAddListeners(listenersByWaterwheel, waterwheel).lasers[sir] = req; + } + + foreach (var sphere in req.references.spheres) + { + GetOrAddListeners(listenersBySphere, sphere).lasers[sir] = req; + } + + foreach (var item in req.references.items) + { + GetOrAddListeners(listenersByItem, item).lasers[sir] = req; + } + } + } + scenes[scenePair.Key] = sceneDescription; } } diff --git a/GameState.cs b/GameState.cs index bdf5760..9280eb3 100644 --- a/GameState.cs +++ b/GameState.cs @@ -33,6 +33,21 @@ namespace ManifoldGardenArchipelago throw new Exception("Shouldn't happen"); } + public static SceneItemReference GetGameplayComponentSceneReference(Component component) + { + LevelSystems levelSystem = GetLevelSystems(component); + + for (int i = 0; i < levelSystem.gameplayComponentsInLevel.Count(); i++) + { + if (levelSystem.gameplayComponentsInLevel[i] == component) + { + return new(component.gameObject.scene.name, i); + } + } + + throw new Exception("Shouldn't happen"); + } + public static SceneItemReference GetButtonSceneReference(ButtonLineActivator arg) { LevelSystems levelSystem = GetLevelSystems(arg); @@ -191,6 +206,37 @@ namespace ManifoldGardenArchipelago } } } + + foreach (var laser in listeners.lasers) + { + Requirement.Decision decision = laser.Value.Check(); + if (decision == Requirement.Decision.Maybe) + { + continue; + } + + bool shouldOpen = (decision == Requirement.Decision.Yes); + + if (SceneManager.GetSceneByName(laser.Key.scene) is Scene laserScene && laserScene.isLoaded) + { + LevelSystems levelSystems = GetLevelSystems(laserScene); + if (levelSystems.isActiveAndEnabled) + { + DarkModeCageController ldc = levelSystems.gameplayComponentsInLevel[laser.Key.index].GetComponent(); + + if (shouldOpen) + { + MethodInfo methodInfo = typeof(DarkModeCageController).GetMethod("OpenLaserSource", BindingFlags.Instance | BindingFlags.NonPublic); + methodInfo.Invoke(ldc, [false]); + } + else + { + MethodInfo methodInfo = typeof(DarkModeCageController).GetMethod("CloseLaserSource", BindingFlags.Instance | BindingFlags.NonPublic); + methodInfo.Invoke(ldc, [false]); + } + } + } + } } } } diff --git a/GameplayPatches.cs b/GameplayPatches.cs index bdcbcae..37a671d 100644 --- a/GameplayPatches.cs +++ b/GameplayPatches.cs @@ -389,4 +389,52 @@ namespace ManifoldGardenArchipelago return false; } } + + [HarmonyPatch(typeof(DarkModeCageController), "StartFilling")] + static class DarkModeCageControllerStartFillingPatch + { + static bool Prefix(DarkModeCageController __instance) + { + SceneItemReference sir = GameState.GetGameplayComponentSceneReference(__instance); + + if (GameData.scenes.TryGetValue(sir.scene, out var sceneDescription)) + { + if (sceneDescription.lasers.TryGetValue(sir.index, out var requirement)) + { + Requirement.Decision decision = requirement.Check(); + + if (decision == Requirement.Decision.No) + { + return false; + } + } + } + + return true; + } + } + + [HarmonyPatch(typeof(DarkModeCageController), "StartUnfilling")] + static class DarkModeCageControllerStartUnfillingPatch + { + static bool Prefix(DarkModeCageController __instance) + { + SceneItemReference sir = GameState.GetGameplayComponentSceneReference(__instance); + + if (GameData.scenes.TryGetValue(sir.scene, out var sceneDescription)) + { + if (sceneDescription.lasers.TryGetValue(sir.index, out var requirement)) + { + Requirement.Decision decision = requirement.Check(); + + if (decision == Requirement.Decision.Yes) + { + return false; + } + } + } + + return true; + } + } } diff --git a/game_data.yaml b/game_data.yaml index 891b169..fe5d99a 100644 --- a/game_data.yaml +++ b/game_data.yaml @@ -75,7 +75,6 @@ Hallway_W062_W063_Optimized: scene: World_063_TowerX_Optimized index: 0 World_001_Optimized: - # lasers are called DarkModeCageControllers; will handle them separately bc most worlds don't item-lock them locations: Blue - Red Laser Activated: socket: 5 @@ -85,6 +84,17 @@ World_001_Optimized: socket: 3 Blue - Tree Purified: entry: AudioVisual_001_Optimized + lasers: + # These are gameplayComponentsInLevel indicies + 127: + item: Blue - Green Laser + socket: 3 + 133: + item: Blue - Yellow Laser + socket: 4 + 149: + item: Blue - Red Laser + socket: 5 World_044_CubicSpaceDivision_Optimized: locations: Blue God Cube Planted: @@ -437,7 +447,7 @@ Hallway_W073_W018_Optimized: Hallway_W018_W063_Optimized: doors: 0: - - button: [0, 1] + button: [0, 1] 1: item: Orange Secret Path or: @@ -758,3 +768,20 @@ World_806_Optimized: locations: Rainbow - Second Water Tetromino Puzzle Solved: waterwheel: 0 +World_071_AkshardhamTemple_Optimized: + lasers: + 49: + waterwheel: 0 + socket: + - scene: World_044_CubicSpaceDivision_Optimized + index: 0 + - scene: World_044_CubicSpaceDivision_Optimized + index: 1 + - scene: World_044_CubicSpaceDivision_Optimized + index: 2 + - scene: World_044_CubicSpaceDivision_Optimized + index: 3 + - scene: World_044_CubicSpaceDivision_Optimized + index: 4 + - scene: World_044_CubicSpaceDivision_Optimized + index: 5 -- cgit 1.4.1