From e03da4f5fe8dc81445153701feab545b6ffeb9d0 Mon Sep 17 00:00:00 2001 From: adelikat Date: Tue, 30 Jun 2015 18:54:42 -0400 Subject: [PATCH] Lua - add a userdata library, with basic api for a script to add arbitruary data to a dictionary. This dictionary is saved and loaded in savestates (if present). Scripts can use this to create counters and whatnot that need data saved in savestates --- BizHawk.Client.Common/BinarySaveStates.cs | 5 +- .../BizHawk.Client.Common.csproj | 1 + BizHawk.Client.Common/Global.cs | 3 + BizHawk.Client.Common/SavestateManager.cs | 30 +++++++++ .../lua/EmuLuaLibrary.UserData.cs | 65 +++++++++++++++++++ 5 files changed, 103 insertions(+), 1 deletion(-) create mode 100644 BizHawk.Client.Common/lua/EmuLuaLibrary.UserData.cs diff --git a/BizHawk.Client.Common/BinarySaveStates.cs b/BizHawk.Client.Common/BinarySaveStates.cs index 2824674307..6ab20315c6 100644 --- a/BizHawk.Client.Common/BinarySaveStates.cs +++ b/BizHawk.Client.Common/BinarySaveStates.cs @@ -27,7 +27,9 @@ namespace BizHawk.Client.Common StateHistorySettings, Markers, ClientSettings, - VerificationLog + VerificationLog, + + UserData } public static class BinaryStateFileNames @@ -63,6 +65,7 @@ namespace BizHawk.Client.Common AddLumpName(BinaryStateLump.Markers, "Markers.txt"); AddLumpName(BinaryStateLump.ClientSettings, "ClientSettings.json"); AddLumpName(BinaryStateLump.VerificationLog, "VerificationLog.txt"); + AddLumpName(BinaryStateLump.UserData, "UserData.txt"); } public static string GetReadName(BinaryStateLump lump) diff --git a/BizHawk.Client.Common/BizHawk.Client.Common.csproj b/BizHawk.Client.Common/BizHawk.Client.Common.csproj index f4b3331a92..b2d30486cf 100644 --- a/BizHawk.Client.Common/BizHawk.Client.Common.csproj +++ b/BizHawk.Client.Common/BizHawk.Client.Common.csproj @@ -137,6 +137,7 @@ + diff --git a/BizHawk.Client.Common/Global.cs b/BizHawk.Client.Common/Global.cs index 2a9f4081c0..7d42723553 100644 --- a/BizHawk.Client.Common/Global.cs +++ b/BizHawk.Client.Common/Global.cs @@ -2,6 +2,7 @@ using BizHawk.Emulation.Cores.Nintendo.Gameboy; using BizHawk.Emulation.Cores.Sega.MasterSystem; using BizHawk.Emulation.DiscSystem; +using System.Collections.Generic; namespace BizHawk.Client.Common { @@ -146,5 +147,7 @@ namespace BizHawk.Client.Common } } } + + public static Dictionary UserBag = new Dictionary(); } } diff --git a/BizHawk.Client.Common/SavestateManager.cs b/BizHawk.Client.Common/SavestateManager.cs index 0df15dac23..0cc11ec34a 100644 --- a/BizHawk.Client.Common/SavestateManager.cs +++ b/BizHawk.Client.Common/SavestateManager.cs @@ -1,11 +1,13 @@ using System; using System.IO; +using System.Linq; using BizHawk.Common; using BizHawk.Common.BufferExtensions; using BizHawk.Common.IOExtensions; using BizHawk.Emulation.Common; using BizHawk.Emulation.Common.IEmulatorExtensions; +using System.Collections.Generic; namespace BizHawk.Client.Common { @@ -60,6 +62,16 @@ namespace BizHawk.Client.Common Global.MovieSession.HandleMovieSaveState(tw); }); } + + if (Global.UserBag.Any()) + { + bs.PutLump(BinaryStateLump.UserData, + delegate(TextWriter tw) + { + var data = ConfigService.SaveWithType(Global.UserBag); + tw.WriteLine(data); + }); + } } } @@ -116,6 +128,24 @@ namespace BizHawk.Client.Common bl.GetCoreState(br => core.LoadStateBinary(br), tr => core.LoadStateText(tr)); bl.GetLump(BinaryStateLump.Framebuffer, false, PopulateFramebuffer); + + if (bl.HasLump(BinaryStateLump.UserData)) + { + string userData = string.Empty; + bl.GetLump(BinaryStateLump.UserData, false, delegate(TextReader tr) + { + string line; + while ((line = tr.ReadLine()) != null) + { + if (!string.IsNullOrWhiteSpace(line)) + { + userData = line; + } + } + }); + + Global.UserBag = (Dictionary)ConfigService.LoadWithType(userData); + } } catch { diff --git a/BizHawk.Client.Common/lua/EmuLuaLibrary.UserData.cs b/BizHawk.Client.Common/lua/EmuLuaLibrary.UserData.cs new file mode 100644 index 0000000000..f496134588 --- /dev/null +++ b/BizHawk.Client.Common/lua/EmuLuaLibrary.UserData.cs @@ -0,0 +1,65 @@ +using System; +using System.Collections.Generic; +using System.ComponentModel; +using System.Linq; + +using LuaInterface; +using BizHawk.Common; +using BizHawk.Emulation.Common; +using BizHawk.Client.Common; + +namespace BizHawk.Client.EmuHawk +{ + [Description("A library for setting and retrieving dynamic data that will be saved and loaded with savestates")] + public sealed class UserDataLibrary : LuaLibraryBase + { + public UserDataLibrary(Lua lua) + : base(lua) { } + + public UserDataLibrary(Lua lua, Action logOutputCallback) + : base(lua, logOutputCallback) { } + + public override string Name { get { return "userdata"; } } + + [LuaMethodAttributes( + "set", + "adds or updates the data with the given key with the given value" + )] + public void Set(string name, object value) + { + Global.UserBag[name] = value; + } + + [LuaMethodAttributes( + "get", + "gets the data with the given key, if the key does not exist it will return nil" + )] + public object Get(string key) + { + if (Global.UserBag.ContainsKey(key)) + { + return Global.UserBag[key]; + } + + return null; + } + + [LuaMethodAttributes( + "clear", + "clears all user data" + )] + public void Clear() + { + Global.UserBag.Clear(); + } + + [LuaMethodAttributes( + "remove", + "remove the data with the given key. Returns true if the element is successfully found and removed; otherwise, false." + )] + public bool Remove(string key) + { + return Global.UserBag.Remove(key); + } + } +}