From 6d61db045fc07311101498bd5d68842efcda9c7f Mon Sep 17 00:00:00 2001 From: adelikat Date: Mon, 7 Jul 2014 18:40:42 +0000 Subject: [PATCH] TasMovie - more reorg and restore state and lag capturing --- .../BizHawk.Client.Common.csproj | 2 + BizHawk.Client.Common/movie/bk2/Bk2Movie.cs | 9 +-- .../movie/tasproj/TasMovie.Editing.cs | 47 +++++++++++++ .../movie/tasproj/TasMovie.cs | 11 ++- .../movie/tasproj/TasStateManager.cs | 67 +++++++++++++++++++ 5 files changed, 126 insertions(+), 10 deletions(-) create mode 100644 BizHawk.Client.Common/movie/tasproj/TasMovie.Editing.cs create mode 100644 BizHawk.Client.Common/movie/tasproj/TasStateManager.cs diff --git a/BizHawk.Client.Common/BizHawk.Client.Common.csproj b/BizHawk.Client.Common/BizHawk.Client.Common.csproj index a180a24049..baa23ea0c7 100644 --- a/BizHawk.Client.Common/BizHawk.Client.Common.csproj +++ b/BizHawk.Client.Common/BizHawk.Client.Common.csproj @@ -166,8 +166,10 @@ + + diff --git a/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs b/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs index e511a7b62c..8944507942 100644 --- a/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs +++ b/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs @@ -70,7 +70,7 @@ namespace BizHawk.Client.Common Changes = true; } - public void RecordFrame(int frame, IController source) + public virtual void RecordFrame(int frame, IController source) { if (Global.Config.VBAStyleMovieLoadState) { @@ -87,7 +87,7 @@ namespace BizHawk.Client.Common Changes = true; } - public void Truncate(int frame) + public virtual void Truncate(int frame) { if (frame < _log.Count) { @@ -162,7 +162,7 @@ namespace BizHawk.Client.Common return null; } - public void PokeFrame(int frame, IController source) + public virtual void PokeFrame(int frame, IController source) { var lg = LogGeneratorInstance(); lg.SetSource(source); @@ -171,9 +171,10 @@ namespace BizHawk.Client.Common SetFrameAt(frame, lg.GenerateLogEntry()); } - public void ClearFrame(int frame) + public virtual void ClearFrame(int frame) { SetFrameAt(frame, LogGeneratorInstance().EmptyEntry); + Changes = true; } #endregion diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.Editing.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.Editing.cs new file mode 100644 index 0000000000..ee7091c47b --- /dev/null +++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.Editing.cs @@ -0,0 +1,47 @@ +using System; +using System.IO; +using System.Linq; + +using BizHawk.Common; + +namespace BizHawk.Client.Common +{ + public partial class TasMovie + { + // TODO: all these + public override void RecordFrame(int frame, Emulation.Common.IController source) + { + base.RecordFrame(frame, source); + + LagLog.RemoveRange(frame, LagLog.Count - frame); + LagLog.Add(Global.Emulator.IsLagFrame); + + StateManager.Invalidate(frame); + StateManager.Capture(); + } + + public override void Truncate(int frame) + { + base.Truncate(frame); + + LagLog.RemoveRange(frame + 2, LagLog.Count - frame - 1); + StateManager.Invalidate(frame + 1); + } + + public override void PokeFrame(int frame, Emulation.Common.IController source) + { + base.PokeFrame(frame, source); + + LagLog.RemoveRange(frame, LagLog.Count - frame); + StateManager.Invalidate(frame); + } + + public override void ClearFrame(int frame) + { + base.ClearFrame(frame); + + LagLog.RemoveRange(frame + 1, LagLog.Count - frame - 1); + StateManager.Invalidate(frame + 1); + } + } +} diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs index 22fda21b20..7d4bbaeaa0 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs @@ -12,6 +12,7 @@ namespace BizHawk.Client.Common public sealed partial class TasMovie : Bk2Movie { private readonly List LagLog = new List(); + private readonly TasStateManager StateManager = new TasStateManager(); public TasMovie(string path) : base(path) { } @@ -32,15 +33,13 @@ namespace BizHawk.Client.Common { get { - return new TasMovieRecord // TODO + return new TasMovieRecord { - State = null, - LogEntry = "", - Lagged = false + State = StateManager[index], + LogEntry = GetInput(index), + Lagged = LagLog[index] }; } } - - // _state = (byte[])Global.Emulator.SaveStateBinary().Clone(); } } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs new file mode 100644 index 0000000000..39b2f7bc1e --- /dev/null +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -0,0 +1,67 @@ +using System.Collections.Generic; +using System.Linq; + +namespace BizHawk.Client.Common +{ + /// + /// Captures savestates and manages the logic of adding, retrieving, + /// invalidating/clearing of states. Also does memory management and limiting of states + /// + public class TasStateManager + { + private readonly Dictionary States = new Dictionary(); + + /// + /// Retrieves the savestate for the given frame, + /// If this frame does not have a state currently, will return an empty array + /// + /// A savestate for the given frame or an empty array if there isn't one + public byte[] this[int frame] + { + get + { + if (States.ContainsKey(frame)) + { + return States[frame]; + } + + return new byte[0]; + } + } + + /// + /// Requests that the current emulator state be captured + /// + public void Capture() + { + var frame = Global.Emulator.Frame; + var state = (byte[])Global.Emulator.SaveStateBinary().Clone(); + + if (States.ContainsKey(frame)) + { + States[frame] = state; + } + else + { + States.Add(frame, state); + } + } + + /// + /// Clears out all savestates after the given frame number + /// + public void Invalidate(int frame) + { + // TODO be more efficient, this could get slow + var toRemove = States + .Where(x => x.Key > frame) + .Select(x => x.Key) + .ToList(); + + foreach (var f in toRemove) + { + States.Remove(f); + } + } + } +}