Skip to content

Commit 6e2f443

Browse files
committedMay 30, 2023
v1.5.8, feat: FireBall track
1 parent efc195f commit 6e2f443

12 files changed

+128
-36
lines changed
 

‎Dialog/English.txt

+1
Original file line numberDiff line numberDiff line change
@@ -121,5 +121,6 @@ TAS_HELPER_PIXEL_GRID= Pixel Grid
121121
TAS_HELPER_ENABLE_PIXEL_GRID= Pixel Grid
122122
TAS_HELPER_PIXEL_GRID_WIDTH= Pixel Grid Width
123123
TAS_HELPER_PIXEL_GRID_OPACITY= Pixel Grid Opacity
124+
TAS_HELPER_FIREBALL_TRACK= Ice/FireBall Track
124125
TAS_HELPER_CAMERA_TARGET= Camera Target
125126
TAS_HELPER_CAMERA_TARGET_VECTOR_OPACITY= Camera-Target Link Opacity

‎Dialog/Simplified Chinese.txt

+1
Original file line numberDiff line numberDiff line change
@@ -122,5 +122,6 @@ TAS_HELPER_PIXEL_GRID= 像素网格
122122
TAS_HELPER_ENABLE_PIXEL_GRID= 像素网格
123123
TAS_HELPER_PIXEL_GRID_WIDTH= 像素网格宽度
124124
TAS_HELPER_PIXEL_GRID_OPACITY= 像素网格不透明度
125+
TAS_HELPER_FIREBALL_TRACK= 冰/火球轨迹
125126
TAS_HELPER_CAMERA_TARGET= 镜头移动目标
126127
TAS_HELPER_CAMERA_TARGET_VECTOR_OPACITY= 镜头-目标 连线的不透明度

‎Entities/FireBallExt.cs

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
using Celeste.Mod.TASHelper.Utils;
2+
using Microsoft.Xna.Framework;
3+
using Monocle;
4+
using System.Reflection;
5+
6+
namespace Celeste.Mod.TASHelper.Entities;
7+
8+
public static class FireBallExt {
9+
10+
public static void Load() {
11+
On.Monocle.EntityList.DebugRender += PatchEntityListDebugRender;
12+
On.Celeste.Level.LoadLevel += OnLoadLevel;
13+
}
14+
15+
public static void Unload() {
16+
On.Monocle.EntityList.DebugRender -= PatchEntityListDebugRender;
17+
On.Celeste.Level.LoadLevel -= OnLoadLevel;
18+
}
19+
20+
public static void Initialize() {
21+
LevelExtensions.AddToTracker(typeof(FireBall));
22+
FireBallNodesGetter = typeof(FireBall).GetField("nodes",BindingFlags.Instance| BindingFlags.NonPublic);
23+
}
24+
25+
public static FieldInfo FireBallNodesGetter;
26+
27+
internal static readonly List<Vector2[]> CachedNodes = new List<Vector2[]>();
28+
29+
private static void OnLoadLevel(On.Celeste.Level.orig_LoadLevel orig, Level self, Player.IntroTypes playerIntro, bool isFromLoader) {
30+
CachedNodes.Clear();
31+
orig(self, playerIntro, isFromLoader);
32+
}
33+
private static void PatchEntityListDebugRender(On.Monocle.EntityList.orig_DebugRender orig, EntityList self, Camera camera) {
34+
orig(self, camera);
35+
if (!TasHelperSettings.UsingFireBallTrack || self.Scene is not Level level) {
36+
return;
37+
}
38+
foreach (Entity entity in level.Tracker.GetEntities<FireBall>()) {
39+
Vector2[] nodes = (Vector2[])FireBallNodesGetter.GetValue(entity);
40+
if (!CachedNodes.Contains(nodes)) {
41+
CachedNodes.Add(nodes);
42+
}
43+
}
44+
foreach (Vector2[] nodes in CachedNodes) {
45+
for (int i = 0; i < nodes.Length - 1; i++) {
46+
Monocle.Draw.Line(nodes[i], nodes[i + 1], Color.Yellow * 0.5f);
47+
}
48+
}
49+
}
50+
51+
/*
52+
* The KillBox is just those part under the bounce hitbox, unless FireBall happens to have Position.Y an integer
53+
* btw there is some OoO issue, so i decide not to render it
54+
* CelesteTAS
55+
private static void PatchFireBallDebugRender(Entity entity) {
56+
if (entity is not FireBall self || !(bool)IceModeGetter.GetValue(self)) {
57+
return;
58+
}
59+
float y = self.Y + 4f - 1f;
60+
float z = (float)Math.Ceiling(y);
61+
if (z <= y) {
62+
z += 1f;
63+
}
64+
float top = Math.Max(self.Collider.AbsoluteTop, z);
65+
Draw.Rect(self.X - 4f, top, 9f, 1f, self.Collidable ? Color.WhiteSmoke : Color.WhiteSmoke * HitboxColor.UnCollidableAlpha);
66+
}
67+
*/
68+
}

‎Entities/SimplifiedSpinner.cs

+6-26
Original file line numberDiff line numberDiff line change
@@ -85,21 +85,15 @@ public static void Initialize() {
8585
}
8686

8787
if (ModUtils.BrokemiaHelperInstalled) {
88-
typeof(Level).GetMethod("LoadLevel").IlHook((cursor, _) => {
89-
cursor.Emit(OpCodes.Ldarg_0);
90-
cursor.EmitDelegate(TrackCassetteSpinner);
91-
});
88+
TrackCassetteSpinner();
9289
typeof(Level).GetMethod("BeforeRender").IlHook((cursor, _) => {
9390
cursor.Emit(OpCodes.Ldarg_0);
9491
cursor.EmitDelegate<Action<Level>>(BrokemiaBeforeRender);
9592
});
9693
}
9794

9895
if (ModUtils.IsaGrabBagInstalled) {
99-
typeof(Level).GetMethod("LoadLevel").IlHook((cursor, _) => {
100-
cursor.Emit(OpCodes.Ldarg_0);
101-
cursor.EmitDelegate(TrackDreamSpinnerRenderer);
102-
});
96+
TrackDreamSpinnerRenderer();
10397
typeof(Level).GetMethod("BeforeRender").IlHook((cursor, _) => {
10498
cursor.Emit(OpCodes.Ldarg_0);
10599
cursor.EmitDelegate<Action<Level>>(IsaGrabBagBeforeRender);
@@ -212,15 +206,8 @@ private static void ChronoBeforeRender(Level self) {
212206
}
213207
}
214208

215-
private static void TrackCassetteSpinner(Level self) {
216-
Type t = typeof(BrokemiaHelper.CassetteSpinner);
217-
if (!Tracker.TrackedEntityTypes.ContainsKey(t)) {
218-
Tracker.TrackedEntityTypes.Add(t, new List<Type>());
219-
Tracker.TrackedEntityTypes[t].Add(t);
220-
}
221-
if (!self.Tracker.Entities.ContainsKey(t)) {
222-
self.Tracker.Entities.Add(t, new List<Entity>());
223-
}
209+
private static void TrackCassetteSpinner() {
210+
LevelExtensions.AddToTracker(typeof(BrokemiaHelper.CassetteSpinner));
224211
}
225212
private static void BrokemiaBeforeRender(Level self) {
226213
if (Updated) {
@@ -237,15 +224,8 @@ private static void BrokemiaBeforeRender(Level self) {
237224
}
238225
}
239226

240-
private static void TrackDreamSpinnerRenderer(Level self) {
241-
Type t = typeof(IsaGrabBag.DreamSpinnerRenderer);
242-
if (!Tracker.TrackedEntityTypes.ContainsKey(t)) {
243-
Tracker.TrackedEntityTypes.Add(t, new List<Type>());
244-
Tracker.TrackedEntityTypes[t].Add(t);
245-
}
246-
if (!self.Tracker.Entities.ContainsKey(t)) {
247-
self.Tracker.Entities.Add(t, new List<Entity>());
248-
}
227+
private static void TrackDreamSpinnerRenderer() {
228+
LevelExtensions.AddToTracker(typeof(IsaGrabBag.DreamSpinnerRenderer));
249229
}
250230

251231
private static void IsaGrabBagBeforeRender(Level self) {

‎Module/TASHelperMenu.cs

+1
Original file line numberDiff line numberDiff line change
@@ -97,6 +97,7 @@ private static EaseInSubMenu CreateMoreOptionsSubMenu(TextMenu menu) {
9797
subMenu.Add(new TextMenuExt.IntSlider("Pixel Grid Opacity".ToDialogText(), 1, 10, TasHelperSettings.PixelGridOpacity).Change(value => TasHelperSettings.PixelGridOpacity = value));
9898
subMenu.Add(new TextMenu.OnOff("Camera Target".ToDialogText(), TasHelperSettings.UsingCameraTarget).Change(value => TasHelperSettings.UsingCameraTarget = value));
9999
subMenu.Add(new TextMenuExt.IntSlider("Camera Target Vector Opacity".ToDialogText(), 1, 9, TasHelperSettings.CameraTargetLinkOpacity).Change(value => TasHelperSettings.CameraTargetLinkOpacity = value));
100+
subMenu.Add(new TextMenu.OnOff("FireBall Track".ToDialogText(), TasHelperSettings.UsingFireBallTrack).Change(value => TasHelperSettings.UsingFireBallTrack = value));
100101
TextMenu.Item MainSwitchStateItem;
101102
EaseInSubHeaderExtPub StateDescription = new EaseInSubHeaderExtPub("Configure At State All".ToDialogText(), false, menu) {
102103
TextColor = Color.Gray,

‎Module/TASHelperSettings.cs

+21-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ internal void OnLoadSettings() {
2222
// do nothing currently
2323

2424
// Everest will save & load settings of public fields/properties when open the game
25-
// i don't want the setters of those properties which has a do not save attribute, which will break the awakes
25+
// i don't want to call setters of those properties which has a do not save attribute, which will break the awakes
2626
// so i have to make them internal
2727
// but i also want to expose them to users, so i add TasHelperSettingsAlias class
2828
// i need to save those loadRangeMode fields, so they should be public
@@ -82,6 +82,7 @@ internal void Sleep() {
8282
Awake_PixelGrid = false;
8383
Awake_SpawnPoint = false;
8484
Awake_EntityActivatorReminder = false;
85+
Awake_FireBallTrack = false;
8586
}
8687
internal void Awake(bool awakeAll) {
8788
MainSwitch = awakeAll ? MainSwitchModes.AllowAll : MainSwitchModes.OnlyDefault;
@@ -94,6 +95,7 @@ internal void Awake(bool awakeAll) {
9495
Awake_PixelGrid = awakeAll;
9596
Awake_SpawnPoint = true;
9697
Awake_EntityActivatorReminder = true;
98+
Awake_FireBallTrack = true;
9799
}
98100

99101
#endregion
@@ -351,6 +353,19 @@ internal bool UsingSpawnPoint {
351353

352354
public int OtherSpawnPointOpacity = 2;
353355

356+
public bool Awake_FireBallTrack = true;
357+
358+
public bool usingFireBallTrack = false;
359+
360+
[SettingDoNotSave]
361+
internal bool UsingFireBallTrack {
362+
get => Enabled && Awake_FireBallTrack && usingFireBallTrack;
363+
set {
364+
usingFireBallTrack = value;
365+
Awake_FireBallTrack = true;
366+
}
367+
}
368+
354369
public bool AllowEnableModWithMainSwitch = true;
355370

356371
private bool mainSwitchStateVisualize = true;
@@ -596,5 +611,10 @@ public static bool UsingSpawnPoint {
596611
get => TasHelperSettings.UsingSpawnPoint;
597612
set => TasHelperSettings.UsingSpawnPoint = value;
598613
}
614+
615+
public static bool UsingFireBallTrack {
616+
get => TasHelperSettings.UsingFireBallTrack;
617+
set => TasHelperSettings.UsingFireBallTrack = value;
618+
}
599619
}
600620

‎README.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -38,8 +38,12 @@ A Celeste Mod designed to be a tool in TAS making.
3838

3939
# WIP:
4040

41+
- Slowdown indicator (note there's 1 frame delay between DeltaTime and TimeRate)
42+
4143
- Better custom info.
4244

45+
- Scrollable console.
46+
4347
# Known issues:
4448

4549
- Actual Collide Hitboxes are overridden -> it's actually bad to use actual collide hitboxes when doing a spinner stun, you really need the exact frame the hazard becomes collidable (opaque). So personnally i do not suggest using actual collide hitboxes in this case. Appended hitbox sounds good but current implement relies on opacity to show information. I have no good idea about it so it's set aside.
@@ -48,4 +52,4 @@ A Celeste Mod designed to be a tool in TAS making.
4852

4953
- Laggy when there are too many spinners (e.g. Strawberry Jam GrandMaster HeartSide) -> Partially solved in v1.4.7
5054

51-
- Hotkeys can't work after several savestates -> Where does this bug come from???
55+
- Hotkeys can't work after several savestates -> Seems fixed in v1.5.7

‎TASHelper.csproj

+1-5
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,7 @@
77
<Nullable>enable</Nullable>
88
<Authors>$(AssemblyName)</Authors>
99
<RootNamespace>Celeste.Mod.TASHelper</RootNamespace>
10-
</PropertyGroup>
11-
12-
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'">
13-
<WarningLevel>4</WarningLevel>
14-
<NoWarn>CS8618</NoWarn>
10+
<Configurations>Release</Configurations>
1511
</PropertyGroup>
1612

1713
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'">

‎TASHelper.sln

+1-2
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,7 @@ Global
1111
Release|Any CPU = Release|Any CPU
1212
EndGlobalSection
1313
GlobalSection(ProjectConfigurationPlatforms) = postSolution
14-
{8AA4040A-8388-48E7-9726-BC2D75BF49CB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
15-
{8AA4040A-8388-48E7-9726-BC2D75BF49CB}.Debug|Any CPU.Build.0 = Debug|Any CPU
14+
{8AA4040A-8388-48E7-9726-BC2D75BF49CB}.Debug|Any CPU.ActiveCfg = Release|Any CPU
1615
{8AA4040A-8388-48E7-9726-BC2D75BF49CB}.Release|Any CPU.ActiveCfg = Release|Any CPU
1716
{8AA4040A-8388-48E7-9726-BC2D75BF49CB}.Release|Any CPU.Build.0 = Release|Any CPU
1817
EndGlobalSection

‎Utils/Extensions.cs

+19
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
using System.Collections.Concurrent;
33
using System.Reflection;
44
using System.Reflection.Emit;
5+
using Monocle;
56

67
namespace Celeste.Mod.TASHelper.Utils;
78

@@ -427,3 +428,21 @@ public static TValue LastValueOrDefault<TKey, TValue>(this SortedDictionary<TKey
427428
return dict.Count > 0 ? dict.Last().Value : default;
428429
}
429430
}
431+
432+
internal static class LevelExtensions {
433+
public static void AddToTracker(Type t) {
434+
typeof(Level).GetMethod("LoadLevel").IlHook((cursor, _) => {
435+
cursor.Emit(Mono.Cecil.Cil.OpCodes.Ldarg_0);
436+
cursor.EmitDelegate((Level level) => {
437+
if (!Tracker.TrackedEntityTypes.ContainsKey(t)) {
438+
Tracker.TrackedEntityTypes.Add(t, new List<Type>());
439+
Tracker.TrackedEntityTypes[t].Add(t);
440+
}
441+
if (!level.Tracker.Entities.ContainsKey(t)) {
442+
level.Tracker.Entities.Add(t, new List<Entity>());
443+
}
444+
});
445+
});
446+
}
447+
448+
}

‎Utils/Loader.cs

+3
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ public static void EntityLoad() {
1111
SimplifiedSpinner.Load();
1212
Messenger.Load();
1313
SpawnPoint.Load();
14+
FireBallExt.Load();
1415
}
1516

1617
public static void EntityUnload() {
@@ -19,6 +20,7 @@ public static void EntityUnload() {
1920
SimplifiedSpinner.Unload();
2021
Messenger.Unload();
2122
SpawnPoint.Unload();
23+
FireBallExt.Unload();
2224
}
2325

2426
public static void HelperLoad() {
@@ -52,6 +54,7 @@ public static void Initialize() {
5254
Messenger.Initialize();
5355
SpawnPoint.Initialize();
5456
RestoreSettingsExt.Initialize();
57+
FireBallExt.Initialize();
5558
}
5659

5760
public static void LoadContent() {

‎everest.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
- Name: TASHelper
2-
Version: 1.5.7
2+
Version: 1.5.8
33
DLL: bin/Release/net4.5.2/TASHelper.dll
44
Dependencies:
55
- Name: Everest

0 commit comments

Comments
 (0)
Please sign in to comment.