From 8a05ea86bfb702d19d7db4d10c691d9427314ea7 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Mon, 30 Oct 2023 03:10:33 -0300 Subject: [PATCH 01/14] whitelist DynRPG Commands whitelisting easyRPG commands so they can be called even when dynRPG is off. --- src/dynrpg.cpp | 13 +++++++++++++ src/dynrpg.h | 2 ++ src/game_interpreter.cpp | 5 +++-- 3 files changed, 18 insertions(+), 2 deletions(-) diff --git a/src/dynrpg.cpp b/src/dynrpg.cpp index 8be59c4fb9..00870d9e34 100644 --- a/src/dynrpg.cpp +++ b/src/dynrpg.cpp @@ -30,6 +30,9 @@ #include "dynrpg_easyrpg.h" #include "dynrpg_textplugin.h" +#include + +std::unordered_set whiteList = { "@easyrpg_output" }; enum DynRpg_ParseMode { ParseMode_Function, @@ -534,3 +537,13 @@ void DynRpg::Reset() { dyn_rpg_functions.clear(); plugins.clear(); } + +bool DynRpg::Whitelist(std::string cmd) { + for (const std::string& whitelistItem : whiteList) { + if (cmd.find(whitelistItem) != std::string::npos) { + return true; + } + } + + return false; +} diff --git a/src/dynrpg.h b/src/dynrpg.h index adfe9e5a3b..d921f33717 100644 --- a/src/dynrpg.h +++ b/src/dynrpg.h @@ -112,6 +112,8 @@ namespace DynRpg { void Load(int slot); void Save(int slot); + bool Whitelist(std::string cmd); + template std::tuple ParseArgs(StringView func_name, dyn_arg_list args, bool* parse_okay = nullptr) { std::tuple t; diff --git a/src/game_interpreter.cpp b/src/game_interpreter.cpp index a12785e08e..62ccb1ea93 100644 --- a/src/game_interpreter.cpp +++ b/src/game_interpreter.cpp @@ -2117,7 +2117,9 @@ bool Game_Interpreter::CommandEndEventProcessing(lcf::rpg::EventCommand const& / } bool Game_Interpreter::CommandComment(const lcf::rpg::EventCommand &com) { - if (Player::IsPatchDynRpg()) { + std::string command = ToString(com.string); + + if (Player::IsPatchDynRpg() || DynRpg::Whitelist(command)) { if (com.string.empty() || com.string[0] != '@') { // Not a DynRPG command return true; @@ -2127,7 +2129,6 @@ bool Game_Interpreter::CommandComment(const lcf::rpg::EventCommand &com) { const auto& list = frame.commands; auto& index = frame.current_command; - std::string command = ToString(com.string); // Concat everything that is not another command or a new comment block for (size_t i = index + 1; i < list.size(); ++i) { const auto& cmd = list[i]; From 0270874d89817ac7c3f0ab688caa60d728456c27 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Mon, 30 Oct 2023 07:12:52 -0300 Subject: [PATCH 02/14] @easyrpg_raw DynRPG command Invoke a command using raw values, similar to how maniacs patch raw work. NOTE FIXME PLS. --- src/dynrpg.cpp | 2 +- src/dynrpg_easyrpg.cpp | 31 +++++++++++++++++++++++++++++++ src/dynrpg_easyrpg.h | 2 ++ 3 files changed, 34 insertions(+), 1 deletion(-) diff --git a/src/dynrpg.cpp b/src/dynrpg.cpp index 00870d9e34..9ecd3cd56e 100644 --- a/src/dynrpg.cpp +++ b/src/dynrpg.cpp @@ -32,7 +32,7 @@ #include "dynrpg_textplugin.h" #include -std::unordered_set whiteList = { "@easyrpg_output" }; +std::unordered_set whiteList = { "@easyrpg_output", "@easyrpg_raw" }; enum DynRpg_ParseMode { ParseMode_Function, diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index d5ba93e181..6878ae1ff8 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -48,6 +48,36 @@ static bool EasyOput(dyn_arg_list args) { return true; } +static bool EasyRaw(dyn_arg_list args) { + auto func = "raw"; + bool okay = false; + + lcf::rpg::EventCommand outputCommand; + std::vector outputParams = {}; + + for (std::size_t i = 0; i < args.size(); ++i) { + std::string currValue = DynRpg::ParseVarArg(func, args, i, okay); + Output::Warning("{}", currValue); + + if (!okay) return true; + + if (i == 0) outputCommand.code = stoi(currValue); + if (i == 1) outputCommand.string = lcf::DBString(currValue); + else outputParams.push_back(stoi(currValue)); + } + + outputCommand.parameters = lcf::DBArray(outputParams.begin(), outputParams.end()); + + //FIXME: this will crash when you two interpreters run a raw command in parallel. + // The lack to access the current interpreter frame is a lack in the dynrpg API design. + // Have to fix this. The current frame should be easy to access + std::vector cmdList = { outputCommand }; + if (Game_Battle::IsBattleRunning()) Game_Battle::GetInterpreter().Push(cmdList, 0, false); + else Game_Map::GetInterpreter().Push(cmdList, 0, false); + + return true; +} + static bool EasyCall(dyn_arg_list args) { auto token = std::get<0>(DynRpg::ParseArgs("call", args)); @@ -92,6 +122,7 @@ void DynRpg::EasyRpgPlugin::RegisterFunctions() { DynRpg::RegisterFunction("call", EasyCall); DynRpg::RegisterFunction("easyrpg_output", EasyOput); DynRpg::RegisterFunction("easyrpg_add", EasyAdd); + DynRpg::RegisterFunction("easyrpg_raw", EasyRaw); } void DynRpg::EasyRpgPlugin::Load(const std::vector& buffer) { diff --git a/src/dynrpg_easyrpg.h b/src/dynrpg_easyrpg.h index 4afd71197b..b4ba98f456 100644 --- a/src/dynrpg_easyrpg.h +++ b/src/dynrpg_easyrpg.h @@ -19,6 +19,8 @@ #define EP_DYNRPG_EASYRPG_H #include "dynrpg.h" +#include "game_battle.h" +#include "game_map.h" namespace DynRpg { /** From 8cabc64383ca4b8169e30f8a5ca01842d1d4baa4 Mon Sep 17 00:00:00 2001 From: Ghabry Date: Mon, 30 Oct 2023 12:32:05 +0100 Subject: [PATCH 03/14] DynRpg: Refactor to use a Game_DynRpg instance. This makes cleanup of state easier when switching between games. --- CMakeLists.txt | 4 +- Makefile.am | 4 +- src/dynrpg_easyrpg.cpp | 95 +++++++++++---------- src/dynrpg_easyrpg.h | 8 +- src/dynrpg_textplugin.cpp | 25 ++++-- src/dynrpg_textplugin.h | 6 +- src/{dynrpg.cpp => game_dynrpg.cpp} | 125 ++++++++++------------------ src/{dynrpg.h => game_dynrpg.h} | 69 +++++++++------ src/game_interpreter.cpp | 6 +- src/main_data.cpp | 2 + src/main_data.h | 3 +- src/player.cpp | 7 +- src/scene_map.cpp | 4 +- src/scene_save.cpp | 4 +- src/spriteset_map.cpp | 4 +- tests/dynrpg.cpp | 19 +++-- 16 files changed, 198 insertions(+), 187 deletions(-) rename src/{dynrpg.cpp => game_dynrpg.cpp} (83%) rename src/{dynrpg.h => game_dynrpg.h} (77%) diff --git a/CMakeLists.txt b/CMakeLists.txt index 89c2404311..619a901ac4 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -101,8 +101,6 @@ add_library(${PROJECT_NAME} OBJECT src/drawable_list.h src/drawable_mgr.cpp src/drawable_mgr.h - src/dynrpg.cpp - src/dynrpg.h src/dynrpg_easyrpg.cpp src/dynrpg_easyrpg.h src/dynrpg_textplugin.cpp @@ -160,6 +158,8 @@ add_library(${PROJECT_NAME} OBJECT src/game_config.h src/game_config_game.cpp src/game_config_game.h + src/game_dynrpg.cpp + src/game_dynrpg.h src/game_enemy.cpp src/game_enemy.h src/game_enemyparty.cpp diff --git a/Makefile.am b/Makefile.am index 153f235633..a2ebfe84d1 100644 --- a/Makefile.am +++ b/Makefile.am @@ -82,8 +82,6 @@ libeasyrpg_player_a_SOURCES = \ src/drawable_list.h \ src/drawable_mgr.cpp \ src/drawable_mgr.h \ - src/dynrpg.cpp \ - src/dynrpg.h \ src/dynrpg_easyrpg.cpp \ src/dynrpg_easyrpg.h \ src/dynrpg_textplugin.h \ @@ -141,6 +139,8 @@ libeasyrpg_player_a_SOURCES = \ src/game_config.h \ src/game_config_game.cpp \ src/game_config_game.h \ + src/game_dynrpg.cpp \ + src/game_dynrpg.h \ src/game_enemy.cpp \ src/game_enemy.h \ src/game_enemyparty.cpp \ diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index 6878ae1ff8..aeae66e257 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -48,53 +48,23 @@ static bool EasyOput(dyn_arg_list args) { return true; } -static bool EasyRaw(dyn_arg_list args) { - auto func = "raw"; - bool okay = false; - - lcf::rpg::EventCommand outputCommand; - std::vector outputParams = {}; - - for (std::size_t i = 0; i < args.size(); ++i) { - std::string currValue = DynRpg::ParseVarArg(func, args, i, okay); - Output::Warning("{}", currValue); - - if (!okay) return true; - - if (i == 0) outputCommand.code = stoi(currValue); - if (i == 1) outputCommand.string = lcf::DBString(currValue); - else outputParams.push_back(stoi(currValue)); - } - - outputCommand.parameters = lcf::DBArray(outputParams.begin(), outputParams.end()); - - //FIXME: this will crash when you two interpreters run a raw command in parallel. - // The lack to access the current interpreter frame is a lack in the dynrpg API design. - // Have to fix this. The current frame should be easy to access - std::vector cmdList = { outputCommand }; - if (Game_Battle::IsBattleRunning()) Game_Battle::GetInterpreter().Push(cmdList, 0, false); - else Game_Map::GetInterpreter().Push(cmdList, 0, false); +bool DynRpg::EasyRpgPlugin::EasyCall(dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { + auto func_name = std::get<0>(DynRpg::ParseArgs("call", args)); - return true; -} - -static bool EasyCall(dyn_arg_list args) { - auto token = std::get<0>(DynRpg::ParseArgs("call", args)); - - if (token.empty()) { + if (func_name.empty()) { // empty function name Output::Warning("call: Empty RPGSS function name"); return true; } - if (!DynRpg::HasFunction(token)) { - // Not a supported function - Output::Warning("Unsupported RPGSS function: {}", token); - return true; + for (auto& plugin: Main_Data::game_dynrpg->plugins) { + if (plugin->Invoke(func_name, args.subspan(1), do_yield, interpreter)) { + return true; + } } - return DynRpg::Invoke(token, args.subspan(1)); + return false; } static bool EasyAdd(dyn_arg_list args) { @@ -118,11 +88,50 @@ static bool EasyAdd(dyn_arg_list args) { return true; } -void DynRpg::EasyRpgPlugin::RegisterFunctions() { - DynRpg::RegisterFunction("call", EasyCall); - DynRpg::RegisterFunction("easyrpg_output", EasyOput); - DynRpg::RegisterFunction("easyrpg_add", EasyAdd); - DynRpg::RegisterFunction("easyrpg_raw", EasyRaw); +bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpreter) { + if (!interpreter) { + return true; + } + + auto func = "raw"; + bool okay = false; + + lcf::rpg::EventCommand outputCommand; + std::vector outputParams = {}; + + for (std::size_t i = 0; i < args.size(); ++i) { + std::string currValue = DynRpg::ParseVarArg(func, args, i, okay); + Output::Warning("{}", currValue); + + if (!okay) return true; + + if (i == 0) outputCommand.code = stoi(currValue); + if (i == 1) outputCommand.string = lcf::DBString(currValue); + else outputParams.push_back(stoi(currValue)); + } + + outputCommand.parameters = lcf::DBArray(outputParams.begin(), outputParams.end()); + + //FIXME: this will crash when you two interpreters run a raw command in parallel. + // The lack to access the current interpreter frame is a lack in the dynrpg API design. + // Have to fix this. The current frame should be easy to access + std::vector cmdList = { outputCommand }; + interpreter->Push(cmdList, 0, false); + + return true; +} + +bool DynRpg::EasyRpgPlugin::Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { + if (func == "call") { + return EasyCall(args, do_yield, interpreter); + } else if (func == "easyrpg_output") { + return EasyOput(args); + } else if (func == "easyrpg_add") { + return EasyAdd(args); + } else if (func == "easyrpg_raw") { + return EasyRaw(args, interpreter); + } + return false; } void DynRpg::EasyRpgPlugin::Load(const std::vector& buffer) { diff --git a/src/dynrpg_easyrpg.h b/src/dynrpg_easyrpg.h index b4ba98f456..aebacf858e 100644 --- a/src/dynrpg_easyrpg.h +++ b/src/dynrpg_easyrpg.h @@ -18,7 +18,7 @@ #ifndef EP_DYNRPG_EASYRPG_H #define EP_DYNRPG_EASYRPG_H -#include "dynrpg.h" +#include "game_dynrpg.h" #include "game_battle.h" #include "game_map.h" @@ -31,9 +31,13 @@ namespace DynRpg { public: EasyRpgPlugin() : DynRpgPlugin("EasyRpgPlugin") {} - void RegisterFunctions() override; + bool Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) override; void Load(const std::vector& buffer) override; std::vector Save() override; + + private: + bool EasyCall(dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter); + bool EasyRaw(dyn_arg_list args, Game_Interpreter* interpreter); }; } diff --git a/src/dynrpg_textplugin.cpp b/src/dynrpg_textplugin.cpp index 3d98ddb7f7..8c017eb98e 100644 --- a/src/dynrpg_textplugin.cpp +++ b/src/dynrpg_textplugin.cpp @@ -399,14 +399,23 @@ static bool RemoveAll(dyn_arg_list) { return true; } -void DynRpg::TextPlugin::RegisterFunctions() { - DynRpg::RegisterFunction("write_text", WriteText); - DynRpg::RegisterFunction("append_line", AppendLine); - DynRpg::RegisterFunction("append_text", AppendText); - DynRpg::RegisterFunction("change_text", ChangeText); - DynRpg::RegisterFunction("change_position", ChangePosition); - DynRpg::RegisterFunction("remove_text", RemoveText); - DynRpg::RegisterFunction("remove_all", RemoveAll); +bool DynRpg::TextPlugin::Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { + if (func == "write_text") { + return WriteText(args); + } else if (func == "append_line") { + return AppendLine(args); + } else if (func == "append_text") { + return AppendText(args); + } else if (func == "change_text") { + return ChangeText(args); + } else if (func == "change_position") { + return ChangePosition(args); + } else if (func == "remove_text") { + return RemoveText(args); + } else if (func == "remove_all") { + return RemoveAll(args); + } + return false; } void DynRpg::TextPlugin::Update() { diff --git a/src/dynrpg_textplugin.h b/src/dynrpg_textplugin.h index 14c2767218..2c21b30ad9 100644 --- a/src/dynrpg_textplugin.h +++ b/src/dynrpg_textplugin.h @@ -18,15 +18,15 @@ #ifndef EP_DYNRPG_TEXTPLUGIN_H #define EP_DYNRPG_TEXTPLUGIN_H -#include "dynrpg.h" +#include "game_dynrpg.h" namespace DynRpg { class TextPlugin : public DynRpgPlugin { public: TextPlugin() : DynRpgPlugin("DynTextPlugin") {} - ~TextPlugin(); + ~TextPlugin() override; - void RegisterFunctions() override; + bool Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) override; void Update() override; void Load(const std::vector&) override; std::vector Save() override; diff --git a/src/dynrpg.cpp b/src/game_dynrpg.cpp similarity index 83% rename from src/dynrpg.cpp rename to src/game_dynrpg.cpp index 9ecd3cd56e..5f1ac91bb1 100644 --- a/src/dynrpg.cpp +++ b/src/game_dynrpg.cpp @@ -16,7 +16,7 @@ */ // Headers -#include "dynrpg.h" +#include "game_dynrpg.h" #include "filefinder.h" #include "game_actors.h" #include "game_variables.h" @@ -26,7 +26,6 @@ #include #include -#include #include "dynrpg_easyrpg.h" #include "dynrpg_textplugin.h" @@ -42,33 +41,13 @@ enum DynRpg_ParseMode { ParseMode_Token }; -typedef std::map dyn_rpg_func; - -namespace { - bool init = false; - - // Registered DynRpg Plugins - std::vector> plugins; - - // DynRpg Function table - dyn_rpg_func dyn_rpg_functions; -} - -void DynRpg::RegisterFunction(const std::string& name, dynfunc func) { - dyn_rpg_functions[name] = func; -} - -bool DynRpg::HasFunction(const std::string& name) { - return dyn_rpg_functions.find(name) != dyn_rpg_functions.end(); -} - // Var arg referenced by $n std::string DynRpg::ParseVarArg(StringView func_name, dyn_arg_list args, int index, bool& parse_okay) { parse_okay = true; if (index >= static_cast(args.size())) { parse_okay = false; Output::Warning("{}: Vararg {} out of range", func_name, index); - return ""; + return {}; } std::string::iterator text_index, end; @@ -100,7 +79,7 @@ std::string DynRpg::ParseVarArg(StringView func_name, dyn_arg_list args, int ind // $-ref out of range parse_okay = false; Output::Warning("{}: Vararg $-ref {} out of range", func_name, i); - return ""; + return {}; } ++text_index; @@ -116,11 +95,10 @@ std::string DynRpg::ParseVarArg(StringView func_name, dyn_arg_list args, int ind } -static std::string ParseToken(const std::string& token, const std::string& function_name) { +static std::string ParseToken(std::string token, StringView function_name) { std::string::iterator text_index, end; - std::string text = token; - text_index = text.begin(); - end = text.end(); + text_index = token.begin(); + end = token.end(); char chr = *text_index; @@ -150,7 +128,7 @@ static std::string ParseToken(const std::string& token, const std::string& funct if (*it == 'N') { if (!Main_Data::game_actors->ActorExists(number)) { Output::Warning("{}: Invalid actor id {} in {}", function_name, number, token); - return ""; + return {}; } // N is last @@ -183,36 +161,36 @@ static std::string ParseToken(const std::string& token, const std::string& funct } // Normal token - return Utils::LowerCase(token); + Utils::LowerCaseInPlace(token); + return token; } -void create_all_plugins() { +void Game_DynRpg::InitPlugins() { + if (plugins_loaded) { + return; + } + plugins.emplace_back(new DynRpg::EasyRpgPlugin()); plugins.emplace_back(new DynRpg::TextPlugin()); - - for (auto& plugin : plugins) { - plugin->RegisterFunctions(); - } - init = true; + plugins_loaded = true; } -std::string DynRpg::ParseCommand(const std::string& command, std::vector& args) { +std::string DynRpg::ParseCommand(std::string command, std::vector& args) { if (command.empty()) { // Not a DynRPG function (empty comment) - return ""; + return {}; } std::string::iterator text_index, end; - std::string text = command; - text_index = text.begin(); - end = text.end(); + text_index = command.begin(); + end = command.end(); char chr = *text_index; if (chr != '@') { // Not a DynRPG function, normal comment - return ""; + return {}; } DynRpg_ParseMode mode = ParseMode_Function; @@ -244,7 +222,7 @@ std::string DynRpg::ParseCommand(const std::string& command, std::vector args; - std::string function_name = ParseCommand(command, args); + std::string function_name = DynRpg::ParseCommand(ToString(command), args); if (function_name.empty()) { return true; @@ -391,18 +367,19 @@ bool DynRpg::Invoke(const std::string& command) { return Invoke(function_name, args); } -bool DynRpg::Invoke(const std::string& func, dyn_arg_list args) { - if (!init) { - create_all_plugins(); - } +bool Game_DynRpg::Invoke(StringView func, dyn_arg_list args, Game_Interpreter* interpreter) { + InitPlugins(); - if (!DynRpg::HasFunction(func)) { - // Not a supported function - Output::Warning("Unsupported DynRPG function: {}", func); - return true; + bool yield = false; + + for (auto& plugin: plugins) { + if (plugin->Invoke(func, args, yield, interpreter)) { + return !yield; + } } - return dyn_rpg_functions[func](args); + Output::Warning("Unsupported DynRPG function: {}", func); + return true; } std::string get_filename(int slot) { @@ -419,14 +396,12 @@ std::string get_filename(int slot) { return found; } -void DynRpg::Load(int slot) { +void Game_DynRpg::Load(int slot) { if (!Player::IsPatchDynRpg()) { return; } - if (!init) { - create_all_plugins(); - } + InitPlugins(); std::string filename = get_filename(slot); @@ -460,7 +435,7 @@ void DynRpg::Load(int slot) { bool have_one = false; for (auto &plugin : plugins) { - if (strncmp((char*)in_buffer.data(), plugin->GetIdentifier().c_str(), len) == 0) { + if (strncmp((char*)in_buffer.data(), plugin->GetIdentifier().data(), len) == 0) { // Chunk length in.read((char *) &len, 4); Utils::SwapByteOrder(len); @@ -488,14 +463,12 @@ void DynRpg::Load(int slot) { } } -void DynRpg::Save(int slot) { +void Game_DynRpg::Save(int slot) { if (!Player::IsPatchDynRpg()) { return; } - if (!init) { - create_all_plugins(); - } + InitPlugins(); std::string filename = get_filename(slot); @@ -515,7 +488,7 @@ void DynRpg::Save(int slot) { Utils::SwapByteOrder(len); out.write((char*)&len, 4); - out.write(plugin->GetIdentifier().c_str(), len); + out.write(plugin->GetIdentifier().data(), len); std::vector data = plugin->Save(); len = data.size(); @@ -526,20 +499,14 @@ void DynRpg::Save(int slot) { } } -void DynRpg::Update() { +void Game_DynRpg::Update() { for (auto& plugin : plugins) { plugin->Update(); } } -void DynRpg::Reset() { - init = false; - dyn_rpg_functions.clear(); - plugins.clear(); -} - -bool DynRpg::Whitelist(std::string cmd) { - for (const std::string& whitelistItem : whiteList) { +bool Game_DynRpg::Whitelist(std::string cmd) { + for (const auto& whitelistItem : whiteList) { if (cmd.find(whitelistItem) != std::string::npos) { return true; } diff --git a/src/dynrpg.h b/src/game_dynrpg.h similarity index 77% rename from src/dynrpg.h rename to src/game_dynrpg.h index d921f33717..7dc8f2f52c 100644 --- a/src/dynrpg.h +++ b/src/game_dynrpg.h @@ -15,8 +15,8 @@ * along with EasyRPG Player. If not, see . */ -#ifndef EP_DYNRPG_H -#define EP_DYNRPG_H +#ifndef EP_GAME_DYNRPG_H +#define EP_GAME_DYNRPG_H #include #include @@ -24,23 +24,28 @@ #include #include #include +#include #include "output.h" #include "utils.h" // Headers -namespace lcf { -namespace rpg { +namespace lcf::rpg { class EventCommand; } -} + +class DynRpgPlugin; +class Game_Interpreter; using dyn_arg_list = const Span; using dynfunc = bool(*)(dyn_arg_list); -/** - * DynRPG namespace - */ +/** Contains helper functions for parsing */ namespace DynRpg { + class EasyRpgPlugin; + + std::string ParseVarArg(StringView func_name, dyn_arg_list args, int index, bool& parse_okay); + std::string ParseCommand(std::string command, std::vector& params); + namespace detail { template inline bool parse_arg(StringView, dyn_arg_list, const int, T&, bool&) { @@ -101,19 +106,6 @@ namespace DynRpg { } } - void RegisterFunction(const std::string& name, dynfunc function); - bool HasFunction(const std::string& name); - std::string ParseVarArg(StringView func_name, dyn_arg_list args, int index, bool& parse_okay); - std::string ParseCommand(const std::string& command, std::vector& params); - bool Invoke(const std::string& command); - bool Invoke(const std::string& func, dyn_arg_list args); - void Update(); - void Reset(); - void Load(int slot); - void Save(int slot); - - bool Whitelist(std::string cmd); - template std::tuple ParseArgs(StringView func_name, dyn_arg_list args, bool* parse_okay = nullptr) { std::tuple t; @@ -131,14 +123,43 @@ namespace DynRpg { } } +/** + * Implements DynRPG Patch (kinda, plugins cannot be executed directly and must be reimplemented) + */ +class Game_DynRpg { +public: + bool Invoke(StringView command, Game_Interpreter* interpreter = nullptr); + void Update(); + void Load(int slot); + void Save(int slot); + bool Whitelist(std::string cmd); + +private: + friend DynRpg::EasyRpgPlugin; + + bool Invoke(StringView func, dyn_arg_list args, Game_Interpreter* interpreter = nullptr); + void InitPlugins(); + + using dyn_rpg_func = std::unordered_map; + + bool plugins_loaded = false; + + // Registered DynRpg Plugins + std::vector> plugins; + + // DynRpg Function table + dyn_rpg_func dyn_rpg_functions; +}; + +/** Base class for implementing a DynRpg Plugins */ class DynRpgPlugin { public: explicit DynRpgPlugin(std::string identifier) : identifier(std::move(identifier)) {} DynRpgPlugin() = delete; - virtual ~DynRpgPlugin() {} + virtual ~DynRpgPlugin() = default; - const std::string& GetIdentifier() const { return identifier; } - virtual void RegisterFunctions() {} + StringView GetIdentifier() const { return identifier; } + virtual bool Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) = 0; virtual void Update() {} virtual void Load(const std::vector&) {} virtual std::vector Save() { return {}; } diff --git a/src/game_interpreter.cpp b/src/game_interpreter.cpp index 62ccb1ea93..ab1d780cde 100644 --- a/src/game_interpreter.cpp +++ b/src/game_interpreter.cpp @@ -26,8 +26,8 @@ #include #include "game_interpreter.h" #include "audio.h" -#include "dynrpg.h" #include "filefinder.h" +#include "game_dynrpg.h" #include "game_map.h" #include "game_event.h" #include "game_enemyparty.h" @@ -2119,7 +2119,7 @@ bool Game_Interpreter::CommandEndEventProcessing(lcf::rpg::EventCommand const& / bool Game_Interpreter::CommandComment(const lcf::rpg::EventCommand &com) { std::string command = ToString(com.string); - if (Player::IsPatchDynRpg() || DynRpg::Whitelist(command)) { + if (Player::IsPatchDynRpg() || Main_Data::game_dynrpg->Whitelist(command)) { if (com.string.empty() || com.string[0] != '@') { // Not a DynRPG command return true; @@ -2140,7 +2140,7 @@ bool Game_Interpreter::CommandComment(const lcf::rpg::EventCommand &com) { } } - return DynRpg::Invoke(command); + return Main_Data::game_dynrpg->Invoke(command, this); } return true; } diff --git a/src/main_data.cpp b/src/main_data.cpp index 86d4a8cc31..1ffa76aac4 100644 --- a/src/main_data.cpp +++ b/src/main_data.cpp @@ -24,6 +24,7 @@ #include "game_system.h" #include "game_actors.h" #include "game_party.h" +#include "game_dynrpg.h" #include "game_enemyparty.h" #include "game_ineluki.h" #include "game_player.h" @@ -70,6 +71,7 @@ namespace Main_Data { std::unique_ptr game_enemyparty; std::unique_ptr game_targets; std::unique_ptr game_quit; + std::unique_ptr game_dynrpg; std::unique_ptr game_ineluki; std::unique_ptr game_switches_global; std::unique_ptr game_variables_global; diff --git a/src/main_data.h b/src/main_data.h index e2d199eb96..588d99de9a 100644 --- a/src/main_data.h +++ b/src/main_data.h @@ -39,6 +39,7 @@ class Game_Variables; class Game_Strings; class Game_Targets; class Game_Quit; +class Game_DynRpg; class Game_Ineluki; class FileFinder_RTP; @@ -57,11 +58,11 @@ namespace Main_Data { extern std::unique_ptr game_enemyparty; extern std::unique_ptr game_targets; extern std::unique_ptr game_quit; + extern std::unique_ptr game_dynrpg; extern std::unique_ptr game_ineluki; extern std::unique_ptr game_switches_global; // Used by Global Save command extern std::unique_ptr game_variables_global; - extern std::unique_ptr filefinder_rtp; void Init(); diff --git a/src/player.cpp b/src/player.cpp index 2279b02d49..0eabe481ce 100644 --- a/src/player.cpp +++ b/src/player.cpp @@ -38,12 +38,12 @@ #include "cache.h" #include "rand.h" #include "cmdline_parser.h" -#include "dynrpg.h" #include "filefinder.h" #include "filefinder_rtp.h" #include "fileext_guesser.h" #include "game_actors.h" #include "game_battle.h" +#include "game_dynrpg.h" #include "game_map.h" #include "game_message.h" #include "game_enemyparty.h" @@ -404,7 +404,6 @@ void Player::Exit() { #endif Player::ResetGameObjects(); Font::Dispose(); - DynRpg::Reset(); Graphics::Quit(); Output::Quit(); FileFinder::Quit(); @@ -817,7 +816,7 @@ void Player::CreateGameObjects() { if (!FileFinder::Game().FindFile("dynloader.dll").empty()) { game_config.patch_dynrpg.Set(true); - Output::Warning("This game uses DynRPG and will not run properly."); + Output::Debug("This game uses DynRPG. Depending on the plugins used it will not run properly."); } if (!FileFinder::Game().FindFile("accord.dll").empty()) { @@ -918,8 +917,6 @@ void Player::ResetGameObjects() { Main_Data::game_variables_global = std::make_unique(min_var, max_var); Main_Data::game_ineluki = std::make_unique(); - DynRpg::Reset(); - Game_Clock::ResetFrame(Game_Clock::now()); Main_Data::game_system->ReloadSystemGraphic(); diff --git a/src/scene_map.cpp b/src/scene_map.cpp index 4267f4568d..528b967369 100644 --- a/src/scene_map.cpp +++ b/src/scene_map.cpp @@ -41,7 +41,7 @@ #include "screen.h" #include "scene_load.h" #include "output.h" -#include "dynrpg.h" +#include "game_dynrpg.h" using namespace std::chrono_literals; @@ -84,7 +84,7 @@ void Scene_Map::Start() { auto current_music = Main_Data::game_system->GetCurrentBGM(); Main_Data::game_system->BgmStop(); Main_Data::game_system->BgmPlay(current_music); - DynRpg::Load(from_save_id); + Main_Data::game_dynrpg->Load(from_save_id); } else { Game_Map::PlayBgm(); } diff --git a/src/scene_save.cpp b/src/scene_save.cpp index d2aa7dfcf4..20cc3b28f7 100644 --- a/src/scene_save.cpp +++ b/src/scene_save.cpp @@ -23,9 +23,9 @@ #endif #include -#include "dynrpg.h" #include "filefinder.h" #include "game_actor.h" +#include "game_dynrpg.h" #include "game_map.h" #include "game_party.h" #include "game_switches.h" @@ -156,7 +156,7 @@ bool Scene_Save::Save(std::ostream& os, int slot_id, bool prepare_save) { auto lcf_engine = Player::IsRPG2k3() ? lcf::EngineVersion::e2k3 : lcf::EngineVersion::e2k; bool res = lcf::LSD_Reader::Save(os, save, lcf_engine, Player::encoding); - DynRpg::Save(slot_id); + Main_Data::game_dynrpg->Save(slot_id); AsyncHandler::SaveFilesystem(); return res; diff --git a/src/spriteset_map.cpp b/src/spriteset_map.cpp index 4d19c24831..05f51d2a16 100644 --- a/src/spriteset_map.cpp +++ b/src/spriteset_map.cpp @@ -18,12 +18,12 @@ // Headers #include "spriteset_map.h" #include "cache.h" -#include "dynrpg.h" #include "game_map.h" #include "main_data.h" #include "sprite_airshipshadow.h" #include "sprite_character.h" #include "game_character.h" +#include "game_dynrpg.h" #include "game_player.h" #include "game_vehicle.h" #include "game_screen.h" @@ -114,7 +114,7 @@ void Spriteset_Map::Update() { shadow->Update(); } - DynRpg::Update(); + Main_Data::game_dynrpg->Update(); } void Spriteset_Map::ChipsetUpdated() { diff --git a/tests/dynrpg.cpp b/tests/dynrpg.cpp index ee5039f80b..6e70c2be29 100644 --- a/tests/dynrpg.cpp +++ b/tests/dynrpg.cpp @@ -1,6 +1,6 @@ #include #include "doctest.h" -#include "dynrpg.h" +#include "game_dynrpg.h" #include "game_variables.h" #include "test_mock_actor.h" @@ -148,39 +148,40 @@ TEST_CASE("Arg parse") { } TEST_CASE("easyrpg dynrpg invoke") { + Game_DynRpg dyn; const MockActor m; std::vector vars = {-1}; Main_Data::game_variables->SetData(vars); Main_Data::game_variables->SetWarning(0); - DynRpg::Invoke("@easyrpg_add 1, 2, 4"); + dyn.Invoke("@easyrpg_add 1, 2, 4"); CHECK(Main_Data::game_variables->Get(1) == 6); // Invalid, stays 6 - DynRpg::Invoke("@easyrpg_add 1, 2, a"); + dyn.Invoke("@easyrpg_add 1, 2, a"); CHECK(Main_Data::game_variables->Get(1) == 6); // Not enough args, stays 6 - DynRpg::Invoke("@easyrpg_add 1"); + dyn.Invoke("@easyrpg_add 1"); CHECK(Main_Data::game_variables->Get(1) == 6); - DynRpg::Invoke("@easyrpg_add 1, 2"); + dyn.Invoke("@easyrpg_add 1, 2"); CHECK(Main_Data::game_variables->Get(1) == 2); - DynRpg::Invoke("@call easyrpg_add, 1, 4.3, 7.7"); + dyn.Invoke("@call easyrpg_add, 1, 4.3, 7.7"); CHECK(Main_Data::game_variables->Get(1) == 11); // Extra args ignored - DynRpg::Invoke("@call easyrpg_add, 1, 4, 8, -14.0"); + dyn.Invoke("@call easyrpg_add, 1, 4, 8, -14.0"); CHECK(Main_Data::game_variables->Get(1) == -2); // Invalid func, stays -2 - DynRpg::Invoke("@call easyrpg_xxx, 1, 4, 7"); + dyn.Invoke("@call easyrpg_xxx, 1, 4, 7"); CHECK(Main_Data::game_variables->Get(1) == -2); // does not crash - DynRpg::Invoke("@unknownfunc 1, 2, 3"); + dyn.Invoke("@unknownfunc 1, 2, 3"); } TEST_CASE("Incompatible changes") { From ed02a13563bfdb289e454efc65632822bdd2cfe0 Mon Sep 17 00:00:00 2001 From: Ghabry Date: Mon, 30 Oct 2023 12:56:34 +0100 Subject: [PATCH 04/14] DynRpg: easyrpg_raw rewrite and fix memory issues --- src/dynrpg_easyrpg.cpp | 56 ++++++++++++++++++++++++--------------- src/dynrpg_easyrpg.h | 4 +-- src/dynrpg_textplugin.cpp | 2 +- src/dynrpg_textplugin.h | 2 +- src/game_dynrpg.cpp | 4 +-- src/game_dynrpg.h | 4 +-- src/game_interpreter.h | 7 +++-- src/player.cpp | 2 +- 8 files changed, 47 insertions(+), 34 deletions(-) diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index aeae66e257..3d0ec73653 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -19,6 +19,7 @@ #include #include "dynrpg_easyrpg.h" +#include "string_view.h" #include "main_data.h" #include "game_variables.h" #include "utils.h" @@ -48,7 +49,7 @@ static bool EasyOput(dyn_arg_list args) { return true; } -bool DynRpg::EasyRpgPlugin::EasyCall(dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { +bool DynRpg::EasyRpgPlugin::EasyCall(Game_DynRpg& dynrpg_instance, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { auto func_name = std::get<0>(DynRpg::ParseArgs("call", args)); if (func_name.empty()) { @@ -58,8 +59,8 @@ bool DynRpg::EasyRpgPlugin::EasyCall(dyn_arg_list args, bool& do_yield, Game_Int return true; } - for (auto& plugin: Main_Data::game_dynrpg->plugins) { - if (plugin->Invoke(func_name, args.subspan(1), do_yield, interpreter)) { + for (auto& plugin: dynrpg_instance.plugins) { + if (plugin->Invoke(dynrpg_instance, func_name, args.subspan(1), do_yield, interpreter)) { return true; } } @@ -93,37 +94,50 @@ bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpr return true; } - auto func = "raw"; + auto func = "easyrpg_raw"; bool okay = false; - lcf::rpg::EventCommand outputCommand; - std::vector outputParams = {}; + lcf::rpg::EventCommand cmd; + std::vector output_args; - for (std::size_t i = 0; i < args.size(); ++i) { - std::string currValue = DynRpg::ParseVarArg(func, args, i, okay); - Output::Warning("{}", currValue); + if (args.empty()) { + Output::Warning("easyrpg_raw: Command too short"); + return true; + } + + std::tie(cmd.code) = DynRpg::ParseArgs(func, args, &okay); + + if (!okay) { + return true; + } - if (!okay) return true; + if (args.size() >= 2) { + auto [string_arg] = DynRpg::ParseArgs(func, args.subspan(1), &okay); + cmd.string = lcf::DBString(string_arg); - if (i == 0) outputCommand.code = stoi(currValue); - if (i == 1) outputCommand.string = lcf::DBString(currValue); - else outputParams.push_back(stoi(currValue)); + if (!okay) { + return true; + } + + for (size_t i = 2; i < args.size(); ++i) { + auto [int_arg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); + + if (!okay) { + return true; + } + } } - outputCommand.parameters = lcf::DBArray(outputParams.begin(), outputParams.end()); + cmd.parameters = lcf::DBArray(output_args.begin(), output_args.end()); - //FIXME: this will crash when you two interpreters run a raw command in parallel. - // The lack to access the current interpreter frame is a lack in the dynrpg API design. - // Have to fix this. The current frame should be easy to access - std::vector cmdList = { outputCommand }; - interpreter->Push(cmdList, 0, false); + interpreter->Push({ cmd }, 0, false); return true; } -bool DynRpg::EasyRpgPlugin::Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { +bool DynRpg::EasyRpgPlugin::Invoke(Game_DynRpg& dynrpg_instance, StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { if (func == "call") { - return EasyCall(args, do_yield, interpreter); + return EasyCall(dynrpg_instance, args, do_yield, interpreter); } else if (func == "easyrpg_output") { return EasyOput(args); } else if (func == "easyrpg_add") { diff --git a/src/dynrpg_easyrpg.h b/src/dynrpg_easyrpg.h index aebacf858e..0f9e41e446 100644 --- a/src/dynrpg_easyrpg.h +++ b/src/dynrpg_easyrpg.h @@ -31,12 +31,12 @@ namespace DynRpg { public: EasyRpgPlugin() : DynRpgPlugin("EasyRpgPlugin") {} - bool Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) override; + bool Invoke(Game_DynRpg& dynrpg_instance, StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) override; void Load(const std::vector& buffer) override; std::vector Save() override; private: - bool EasyCall(dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter); + bool EasyCall(Game_DynRpg& dynrpg_instance, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter); bool EasyRaw(dyn_arg_list args, Game_Interpreter* interpreter); }; } diff --git a/src/dynrpg_textplugin.cpp b/src/dynrpg_textplugin.cpp index 8c017eb98e..397d6488e5 100644 --- a/src/dynrpg_textplugin.cpp +++ b/src/dynrpg_textplugin.cpp @@ -399,7 +399,7 @@ static bool RemoveAll(dyn_arg_list) { return true; } -bool DynRpg::TextPlugin::Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) { +bool DynRpg::TextPlugin::Invoke(Game_DynRpg&, StringView func, dyn_arg_list args, bool&, Game_Interpreter*) { if (func == "write_text") { return WriteText(args); } else if (func == "append_line") { diff --git a/src/dynrpg_textplugin.h b/src/dynrpg_textplugin.h index 2c21b30ad9..82a53438e7 100644 --- a/src/dynrpg_textplugin.h +++ b/src/dynrpg_textplugin.h @@ -26,7 +26,7 @@ namespace DynRpg { TextPlugin() : DynRpgPlugin("DynTextPlugin") {} ~TextPlugin() override; - bool Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) override; + bool Invoke(Game_DynRpg& dynrpg_instance, StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) override; void Update() override; void Load(const std::vector&) override; std::vector Save() override; diff --git a/src/game_dynrpg.cpp b/src/game_dynrpg.cpp index 5f1ac91bb1..964706fecf 100644 --- a/src/game_dynrpg.cpp +++ b/src/game_dynrpg.cpp @@ -364,7 +364,7 @@ bool Game_DynRpg::Invoke(StringView command, Game_Interpreter* interpreter) { return true; } - return Invoke(function_name, args); + return Invoke(function_name, args, interpreter); } bool Game_DynRpg::Invoke(StringView func, dyn_arg_list args, Game_Interpreter* interpreter) { @@ -373,7 +373,7 @@ bool Game_DynRpg::Invoke(StringView func, dyn_arg_list args, Game_Interpreter* i bool yield = false; for (auto& plugin: plugins) { - if (plugin->Invoke(func, args, yield, interpreter)) { + if (plugin->Invoke(*this, func, args, yield, interpreter)) { return !yield; } } diff --git a/src/game_dynrpg.h b/src/game_dynrpg.h index 7dc8f2f52c..fd789b214d 100644 --- a/src/game_dynrpg.h +++ b/src/game_dynrpg.h @@ -137,7 +137,7 @@ class Game_DynRpg { private: friend DynRpg::EasyRpgPlugin; - bool Invoke(StringView func, dyn_arg_list args, Game_Interpreter* interpreter = nullptr); + bool Invoke(StringView func, dyn_arg_list args, Game_Interpreter* interpreter); void InitPlugins(); using dyn_rpg_func = std::unordered_map; @@ -159,7 +159,7 @@ class DynRpgPlugin { virtual ~DynRpgPlugin() = default; StringView GetIdentifier() const { return identifier; } - virtual bool Invoke(StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) = 0; + virtual bool Invoke(Game_DynRpg& dynrpg_instance, StringView func, dyn_arg_list args, bool& do_yield, Game_Interpreter* interpreter) = 0; virtual void Update() {} virtual void Load(const std::vector&) {} virtual std::vector Save() { return {}; } diff --git a/src/game_interpreter.h b/src/game_interpreter.h index f349b39188..8d0e44625a 100644 --- a/src/game_interpreter.h +++ b/src/game_interpreter.h @@ -64,9 +64,9 @@ class Game_Interpreter void Update(bool reset_loop_count=true); void Push( - std::vector _list, - int _event_id, - bool started_by_decision_key = false + std::vector _list, + int _event_id, + bool started_by_decision_key = false ); void Push(Game_Event* ev); void Push(Game_Event* ev, const lcf::rpg::EventPage* page, bool triggered_by_decision_key); @@ -78,7 +78,6 @@ class Game_Interpreter bool ExecuteCommand(); virtual bool ExecuteCommand(lcf::rpg::EventCommand const& com); - /** * Returns a SaveEventExecState needed for the savefile. * diff --git a/src/player.cpp b/src/player.cpp index 0eabe481ce..e9666ba467 100644 --- a/src/player.cpp +++ b/src/player.cpp @@ -915,6 +915,7 @@ void Player::ResetGameObjects() { Main_Data::game_quit = std::make_unique(); Main_Data::game_switches_global = std::make_unique(); Main_Data::game_variables_global = std::make_unique(min_var, max_var); + Main_Data::game_dynrpg = std::make_unique(); Main_Data::game_ineluki = std::make_unique(); Game_Clock::ResetFrame(Game_Clock::now()); @@ -1541,4 +1542,3 @@ std::string Player::GetEngineVersion() { if (EngineVersion() > 0) return std::to_string(EngineVersion()); return std::string(); } - From a54d8b24ce776c8b5899a7717e67d81a56090f9b Mon Sep 17 00:00:00 2001 From: Ghabry Date: Mon, 30 Oct 2023 15:18:14 +0100 Subject: [PATCH 05/14] DynRPG: Support T (Maniac Patch string variable) in the token parser --- src/game_dynrpg.cpp | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/src/game_dynrpg.cpp b/src/game_dynrpg.cpp index 964706fecf..2b9e65e8d7 100644 --- a/src/game_dynrpg.cpp +++ b/src/game_dynrpg.cpp @@ -19,6 +19,7 @@ #include "game_dynrpg.h" #include "filefinder.h" #include "game_actors.h" +#include "game_strings.h" #include "game_variables.h" #include "main_data.h" #include "output.h" @@ -110,6 +111,9 @@ static std::string ParseToken(std::string token, StringView function_name) { std::stringstream number_part; for (;;) { + // This loop checks if the token is to be substituted + // If a token is (regex) [NT]?V+[0-9]+ it is resolved to a var or an actor + // T is an EasyRPG extension: It will return a Maniac Patch String Var if (text_index != end) { chr = *text_index; } @@ -131,8 +135,11 @@ static std::string ParseToken(std::string token, StringView function_name) { return {}; } - // N is last + // N (Actor Name) is last return ToString(Main_Data::game_actors->GetActor(number)->GetName()); + } else if (*it == 'T' && Player::IsPatchManiac()) { + // T (String Var) is last + return Main_Data::game_strings->Get(number); } else { // Variable number = Main_Data::game_variables->Get(number); @@ -152,6 +159,11 @@ static std::string ParseToken(std::string token, StringView function_name) { var_part << chr; } else if (chr == 'V') { var_part << chr; + } else if (chr == 'T' && Player::IsPatchManiac()) { + if (!first) { + break; + } + var_part << chr; } else { break; } @@ -204,7 +216,6 @@ std::string DynRpg::ParseCommand(std::string command, std::vector& // Strings are in "", a "-literal is represented by "" // Number is a valid float number // Tokens are Strings without "" and with Whitespace stripped o_O - // If a token is (regex) N?V+[0-9]+ it is resolved to a var or an actor // All arguments are passed as string to the DynRpg functions and are // converted to int or float on demand. From 163e5b6600383dedf62ef9d60c30cb3a54dfe7b5 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Tue, 31 Oct 2023 13:31:47 -0300 Subject: [PATCH 06/14] DynRPG @Raw - Calling Events by name. @easyrpg_raw @ShowMessage, "This is new" --- CMakeLists.txt | 1 + src/constants.h | 1368 ++++++++++++++++++++++++++++++++++++++++ src/dynrpg_easyrpg.cpp | 16 +- src/dynrpg_easyrpg.h | 1 + 4 files changed, 1385 insertions(+), 1 deletion(-) create mode 100644 src/constants.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 619a901ac4..f6e6fd5569 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -77,6 +77,7 @@ add_library(${PROJECT_NAME} OBJECT src/color.h src/compiler.h src/config_param.h + src/constants.h src/decoder_fluidsynth.cpp src/decoder_fluidsynth.h src/decoder_libsndfile.cpp diff --git a/src/constants.h b/src/constants.h new file mode 100644 index 0000000000..39f84a8221 --- /dev/null +++ b/src/constants.h @@ -0,0 +1,1368 @@ +/* + * This file is part of EasyRPG Player. + * + * EasyRPG Player is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * EasyRPG Player is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with EasyRPG Player. If not, see . + */ + +#include +#include +#include + +class Constants { +public: + std::unordered_map DestinyScript_str; + std::unordered_map DestinyScript; + std::unordered_map EventCode; + + Constants() { + DestinyScript_str = { + // Special Text Parts + {"CR", "Chr(0x0D)"}, + {"LF", "Chr(0x0A)"}, + {"CRLF", "CR..LF"}, + {"QUOTE", "\"\""}, + }; + + DestinyScript = { + // Activation/Deactivation + {"DISABLE", 0}, + {"ENABLE", 1}, + + // Animation Properties + {"ANIMATION_SCOPE_SINGLE", 0}, + {"ANIMATION_SCOPE_SCREEN", 1}, + + {"ANIMATION_FOCUS_TOP", 0}, + {"ANIMATION_FOCUS_CENTER", 1}, + {"ANIMATION_FOCUS_BOTTOM", 2}, + + {"ANIMATION_FLASHTYPE_NONE", 0}, + {"ANIMATION_FLASHTYPE_TARGET", 1}, + {"ANIMATION_FLASHTYPE_SCREEN", 2}, + + {"ANIMATION_SHAKETYPE_NONE", 0}, + {"ANIMATION_SHAKETYPE_TARGET", 1}, + {"ANIMATION_SHAKETYPE_SCREEN", 2}, + + // Equipment Types/Filters + {"EQUIPMENT_ALL", 0}, + {"EQUIPMENT_WEAPON", 1}, + {"EQUIPMENT_SHIELD", 2}, + {"EQUIPMENT_ARMOR", 3}, + {"EQUIPMENT_HELMET", 4}, + {"EQUIPMENT_ACCESSORY", 5}, + + // Battler Properties + {"BATTLERSPEED_SLOW", 20}, + {"BATTLERSPEED_MID", 14}, + {"BATTLERSPEED_FAST", 8}, + + {"BATTLERGRAPHIC_CHARACTER", 0}, + {"BATTLERGRAPHIC_ANIMATION", 1}, + + {"BATTLERANIM_IDLE", 0}, + {"BATTLERANIM_ATTACKRIGHT", 1}, + {"BATTLERANIM_ATTACKLEFT", 2}, + {"BATTLERANIM_SKILLUSE", 3}, + {"BATTLERANIM_KNOCKOUT", 4}, + {"BATTLERANIM_DAMAGE", 5}, + {"BATTLERANIM_CONDITION", 6}, + {"BATTLERANIM_DEFEND", 7}, + {"BATTLERANIM_WALKLEFT", 8}, + {"BATTLERANIM_WALKRIGHT", 9}, + {"BATTLERANIM_VICTORY", 10}, + {"BATTLERANIM_ITEMUSE", 11}, + + // MoveRoute Commands + {"MC_STEPUP", 0}, + {"MC_STEPRIGHT", 1}, + {"MC_STEPDOWN", 2}, + {"MC_STEPLEFT", 3}, + {"MC_STEPRIGHTUP", 4}, + {"MC_STEPRIGHTDOWN", 5}, + {"MC_STEPLEFTDOWN", 6}, + {"MC_STEPLEFTUP", 7}, + {"MC_STEPRANDOM", 8}, + {"MC_STEPTOWARDHERO", 9}, + {"MC_STEPAWAYFROMHERO", 10}, + {"MC_STEPFORWARD", 11}, + + {"MC_LOOKUP", 12}, + {"MC_LOOKRIGHT", 13}, + {"MC_LOOKDOWN", 14}, + {"MC_LOOKLEFT", 15}, + {"MC_TURNRIGHT", 16}, + {"MC_TURNLEFT", 17}, + {"MC_TURN180", 18}, + {"MC_RIGHTLEFTTURN", 19}, + {"MC_LOOKRANDOM", 20}, + {"MC_LOOKTOWARDHERO", 21}, + {"MC_LOOKAWAYFROMHERO", 22}, + + {"MC_WAIT", 23}, + {"MC_STARTJUMPING", 24}, + {"MC_STOPJUMPING", 25}, + {"MC_FIXDIRECTION", 26}, + {"MC_RELEASEDIRECTION", 27}, + {"MC_INCREASESPEED", 28}, + {"MC_DECREASESPEED", 29}, + {"MC_INCREASEFREQUENCY", 30}, + {"MC_DECREASEFREQUENCY", 31}, + {"MC_SETSWITCHON", 32}, + {"MC_SETSWITCHOFF", 33}, + {"MC_SETCHAR", 34}, + {"MC_PLAYSOUND", 35}, + {"MC_STARTPHASING", 36}, + {"MC_STOPPHASING", 37}, + {"MC_FIXANIMATION", 38}, + {"MC_RELEASEANIMATION", 39}, + {"MC_INCREASETRANSPARENCY", 40}, + {"MC_DECREASETRANSPARENCY", 41}, + + // MovePoint Types + {"MOVEPOINT_DISPLAY", 0}, + {"MOVEPOINT_HIGHRES", 1}, + + // Screen Brightness + {"SCREEN_BLACK", 0}, + {"SCREEN_DARK", 50}, + {"SCREEN_NORMAL", 100}, + {"SCREEN_BRIGHT", 150}, + {"SCREEN_WHITE", 200}, + + // Bool Values + {"FALSE", 0}, + {"TRUE", 1}, + + // File System Attributes + {"FILE_ATTRIBUTE_NORMAL", 128}, + {"FILE_ATTRIBUTE_SYSTEM", 4}, + {"FILE_ATTRIBUTE_HIDDEN", 2}, + {"FILE_ATTRIBUTE_ARCHIVE", 32}, + {"FILE_ATTRIBUTE_DIRECTORY", 16}, + {"FILE_ATTRIBUTE_READONLY", 1}, + + // File Access Types + {"FILE_READ", 1}, + {"FILE_WRITE", 2}, + {"FILE_APPEND", 6}, + + {"FILEPOS_BEGIN", 0}, + {"FILEPOS_CURRENT", 1}, + {"FILEPOS_END", 2}, + + // DataTypes + {"TYPE_VARIABLE", 1}, + {"TYPE_BOOL", 2}, + {"TYPE_SWITCH", 2}, // LEGACY + {"TYPE_DWORD", 3}, + {"TYPE_FLOAT", 4}, + {"TYPE_DOUBLE", 4}, // LEGACY + {"TYPE_STRING", 5}, + {"TYPE_BYTE", 6}, + {"TYPE_WORD", 7}, + + // Vulnerability Levels + {"EFFECT_A", 0}, + {"EFFECT_B", 1}, + {"EFFECT_C", 2}, + {"EFFECT_D", 3}, + {"EFFECT_E", 4}, + + // Element/Attribute Properties + {"ELEMENTTYPE_WEAPONATTRIBUTETYPE_WEAPON", 0}, + {"ELEMENTTYPE_MAGICATTRIBUTETYPE_MAGIC", 1}, + + // Event Facing Directions + {"DIR_UP", 0}, + {"DIR_RIGHT", 1}, + {"DIR_DOWN", 2}, + {"DIR_LEFT", 3}, + {"DIR_UP_RIGHT", 4}, + {"DIR_RIGHT_UP", 4}, // LEGACY + {"DIR_DOWN_RIGHT", 5}, + {"DIR_RIGHT_DOWN", 5}, // LEGACY + {"DIR_DOWN_LEFT", 6}, + {"DIR_LEFT_DOWN", 6}, // LEGACY + {"DIR_UP_LEFT", 7}, + {"DIR_LEFT_UP", 7}, // LEGACY + + // Event Properties + {"EVENT_MAPID", 0}, + {"EVENT_X", 1}, + {"EVENT_Y", 2}, + {"EVENT_DIRECTION", 3}, + {"EVENT_SCREENX", 4}, + {"EVENT_SCREENY", 5}, + + // Event Step Graphics + {"EVENTFRAME_LEFT", 0}, + {"EVENTFRAME_NORMAL", 1}, + {"EVENTFRAME_RIGHT", 2}, + {"EVENTFRAME_NORMAL2", 3}, + + // Eventoid Special IDs + {"PLAYER", 10001}, + {"HERO", 10001}, // LEGACY + {"BOAT", 10002}, + {"SHIP", 10003}, + {"AIRSHIP", 10004}, + {"THIS", 10005}, + + // Error Messages + {"DESTINYERROR_UNKNOWN", -1}, + {"DESTINYERROR_SYNTAX", 1}, + {"DESTINYERROR_NOVALUE", 2}, + {"DESTINYERROR_UNKNOWNNAME", 3}, + {"DESTINYERROR_CONVERT", 4}, + {"DESTINYERROR_READONLY", 5}, + {"DESTINYERROR_ARRAYBOUND", 6}, + {"DESTINYERROR_RANGE", 7}, + {"DESTINYERROR_MEMORY", 8}, + {"DESTINYERROR_VALUE", 9}, + {"DESTINYERROR_BINARYFLOAT", 10}, + {"DESTINYERROR_CALCSWITCH", 11}, + {"DESTINYERROR_CALC_STRING", 12}, + {"DESTINYERROR_FLOATERROR", 13}, + {"DESTINYERROR_FLOATLENGTH", 14}, + {"DESTINYERROR_DIVISIONBYZERO", 15}, + {"DESTINYERROR_STRINGFORMAT", 16}, + {"DESTINYERROR_STRINGRANGE", 17}, + {"DESTINYERROR_PICTURE", 18}, + {"DESTINYERROR_PIXELRANGE", 19}, + {"DESTINYERROR_SAMEPICTURE", 20}, + {"DESTINYERROR_PALETTERANGE", 21}, + {"DESTINYERROR_SOCKETSTARTUP", 22}, + {"DESTINYERROR_NOFREESOCKET", 23}, + {"DESTINYERROR_CANTCREATESOCKET", 24}, + {"DESTINYERROR_SOCKETSTILLOPEN", 25}, + {"DESTINYERROR_SOCKETNOTOPEN", 26}, + {"DESTINYERROR_CANTCONNECT", 27}, + {"DESTINYERROR_SOCKETTYPE", 28}, + {"DESTINYERROR_SOCKETERROR", 29}, + {"DESTINYERROR_OOB", 30}, + {"DESTINYERROR_STRINGTOOLONG", 31}, + {"DESTINYERROR_NOFREEFILEHANDLE", 32}, + {"DESTINYERROR_CANTRESOLVEPATH", 33}, + {"DESTINYERROR_NOPERMISSION", 34}, + {"DESTINYERROR_CANTOPENFILE", 35}, + {"DESTINYERROR_FILENOTOPEN", 36}, + {"DESTINYERROR_CANTACCESSFILE", 37}, + {"DESTINYERROR_CANTCREATEDIR", 38}, + {"DESTINYERROR_CANTREMOVEDIR", 39}, + {"DESTINYERROR_CANTRENAMEFILE", 40}, + {"DESTINYERROR_CANTCOPYFILE", 41}, + {"DESTINYERROR_CANTDELETEFILE", 42}, + {"DESTINYERROR_CANTREADATTRIBUTES", 43}, + {"DESTINYERROR_CANTWRITEATTRIBUTES", 44}, + {"DESTINYERROR_SEARCHSTILLOPEN", 45}, + {"DESTINYERROR_CANTSTARTSEARCH", 46}, + {"DESTINYERROR_NOSEARCHSTARTED", 47}, + {"DESTINYERROR_PROTECTED", 48}, + {"DESTINYERROR_CANTEXECUTEFILE", 49}, + {"DESTINYERROR_NOTENOUGHPARAMETERS", 50}, + {"DESTINYERROR_NOTCOMPARABLE", 51}, + {"DESTINYERROR_INVALIDSIGN", 52}, + {"DESTINYERROR_SETNOTALLOWEDHERE", 53}, + {"DESTINYERROR_MISSINGENDIF", 54}, + {"DESTINYERROR_MISSINGLOOP", 55}, + {"DESTINYERROR_MISSINGCONDITION", 56}, + {"DESTINYERROR_MISSINGNEXT", 57}, + {"DESTINYERROR_MISSINGCASE", 58}, + {"DESTINYERROR_MISSINGENDSWITCH", 59}, + {"DESTINYERROR_BREAKNOTALLOWEDHERE", 60}, + {"DESTINYERROR_CONTINUENOTALLOWEDHERE", 61}, + {"DESTINYERROR_INVALIDHANDLE", 62}, + {"DESTINYERROR_OPERATOREXPECTED", 63}, + {"DESTINYERROR_CANTACCESSSAVESLOT", 64}, + {"DESTINYERROR_EXCEPTION", 65}, + {"DESTINYERROR_INVALIDLOOPOPERATIONLEVEL", 66}, + {"DESTINYERROR_CONTINUEDOESNTPOINTTOLOOP", 67}, + + // GameKey Flags + {"GKF_DOWN", 1}, + {"GKF_LEFT", 2}, + {"GKF_RIGHT", 4}, + {"GKF_UP", 8}, + {"GKF_DECISION", 16}, + {"GKF_CANCEL", 32}, + {"GKF_MOUSELEFT", 64}, + {"GKF_MOUSERIGHT", 128}, + {"GKF_SHIFT", 256}, + {"GKF_CTRL", 512}, + {"GKF_NUM0", 1024}, + {"GKF_NUM1", 2048}, + {"GKF_NUM2", 4096}, + {"GKF_NUM3", 8192}, + {"GKF_NUM4", 16384}, + {"GKF_NUM5", 32768}, + {"GKF_NUM6", 65536}, + {"GKF_NUM7", 131072}, + {"GKF_NUM8", 262144}, + {"GKF_NUM9", 524288}, + {"GKF_ADD", 1048576}, + {"GKF_SUB", 2097152}, + {"GKF_MUL", 4194304}, + {"GKF_DIV", 8388608}, + {"GKF_DOT", 16777216}, + {"GKF_MOUSEMIDDLE", 33554432}, + {"GKF_WHEELUP", 67108864}, + {"GKF_WHEELDOWN", 134217728}, + {"GKF_WHEELLEFT", 268435456}, + {"GKF_WHEELRIGHT", 536870912}, + + // GameKey IDs + {"GK_DOWN", 1}, + {"GK_LEFT", 9}, + {"GK_RIGHT", 17}, + {"GK_UP", 25}, + {"GK_DECISION", 33}, + {"GK_CANCEL", 41}, + {"GK_NUM0", 81}, + {"GK_NUM1", 89}, + {"GK_NUM2", 97}, + {"GK_NUM3", 105}, + {"GK_NUM4", 113}, + {"GK_NUM5", 121}, + {"GK_NUM6", 129}, + {"GK_NUM7", 137}, + {"GK_NUM8", 145}, + {"GK_NUM9", 153}, + {"GK_ADD", 161}, + {"GK_SUB", 169}, + {"GK_MUL", 177}, + {"GK_DIV", 185}, + {"GK_DOT", 193}, + + // Gamepad Button Flags + {"JOY_BUTTON01", 1}, + {"JOY_BUTTON02", 2}, + {"JOY_BUTTON03", 4}, + {"JOY_BUTTON04", 8}, + {"JOY_BUTTON05", 16}, + {"JOY_BUTTON06", 32}, + {"JOY_BUTTON07", 64}, + {"JOY_BUTTON08", 128}, + {"JOY_BUTTON09", 256}, + {"JOY_BUTTON10", 512}, + {"JOY_BUTTON11", 1024}, + {"JOY_BUTTON12", 2048}, + {"JOY_BUTTON13", 4096}, + {"JOY_BUTTON14", 8192}, + {"JOY_BUTTON15", 16384}, + {"JOY_BUTTON16", 32768}, + {"JOY_BUTTON17", 65536}, + {"JOY_BUTTON18", 131072}, + {"JOY_BUTTON19", 262144}, + {"JOY_BUTTON20", 524288}, + {"JOY_BUTTON21", 1048576}, + {"JOY_BUTTON22", 2097152}, + {"JOY_BUTTON23", 4194304}, + {"JOY_BUTTON24", 8388608}, + {"JOY_BUTTON25", 16777216}, + {"JOY_BUTTON26", 33554432}, + {"JOY_BUTTON27", 67108864}, + {"JOY_BUTTON28", 134217728}, + {"JOY_BUTTON29", 268435456}, + {"JOY_BUTTON30", 536870912}, + {"JOY_BUTTON31", 1073741824}, + {"JOY_BUTTON32", -2147483648}, + + // Gamepad State + {"JOYPAD_UNPLUGGED", 1}, + {"JOYPAD_PLUGGED", 0}, + {"JOYPAD_NODRIVER", -1}, + {"JOYPAD_INVALIDPARAM", -2}, + {"JOYPAD_BADDEVICE", -3}, + {"JOYPAD_INVALIDJOY", -4}, + + // Gamepad DPad + {"JOYPOV_CENTER", 65536}, + {"JOYPOV_UP", 0}, + {"JOYPOV_UP_RIGHT", 4500}, + {"JOYPOV_RIGHT_UP", 4500}, // LEGACY + {"JOYPOV_RIGHT", 9000}, + {"JOYPOV_DOWN_RIGHT", 13500}, + {"JOYPOV_RIGHT_DOWN", 13500}, // LEGACY + {"JOYPOV_DOWN", 18000}, + {"JOYPOV_DOWN_LEFT", 22500}, + {"JOYPOV_LEFT_DOWN", 22500}, // LEGACY + {"JOYPOV_LEFT", 27000}, + {"JOYPOV_UP_LEFT", 31500}, + {"JOYPOV_LEFT_UP", 31500}, // LEGACY + + // Inventory and Equipment + {"ITEM_OWNED", 0}, + {"ITEM_EQUIPPED", 1}, + + // Item Properties + {"ITEMSCOPE_SINGLEALLY", 0}, + {"ITEMSCOPE_ALLALLIES", 1}, + + {"INFINITE_USES", 0}, + + {"PROJECTILE_KNIFE", 0}, + {"PROJECTILE_BOOMERANG", 1}, + + {"PROJECTILESPEED_FAST", 0}, + {"PROJECTILESPEED_MID", 1}, + {"PROJECTILESPEED_SLOW", 2}, + + {"WEAPONRANGE_MELEE", 0}, + {"WEAPONRANGE_DISTANT", 1}, + + {"WEAPONSCOPE_SINGLE", 0}, + {"WEAPONSCOPE_CENTER", 1}, + {"WEAPONSCOPE_SIMULTANEOUS", 2}, + {"WEAPONSCOPE_CONSECUTIVE", 3}, + + // Item Categories + {"ITEMTYPE_COMMON", 0}, + {"ITEMTYPE_WEAPON", 1}, + {"ITEMTYPE_SHIELD", 2}, + {"ITEMTYPE_ARMOR", 3}, + {"ITEMTYPE_HELMET", 4}, + {"ITEMTYPE_ACCESSORY", 5}, + {"ITEMTYPE_MEDICINE", 6}, + {"ITEMTYPE_BOOK", 7}, + {"ITEMTYPE_MATERIAL", 8}, + {"ITEMTYPE_UNIQUE", 9}, + {"ITEMTYPE_SWITCH", 10}, + + // Troop Order + {"TROOP_ALIGNMENT_MANUAL", 0}, + {"TROOP_ALIGNMENT_AUTOMATIC", 1}, + + // Troop Event Page Flags + {"BATTLECON_SWITCH1", 1}, + {"BATTLECON_SWITCH2", 2}, + {"BATTLECON_VARIABLE", 4}, + {"BATTLECON_TURNS", 8}, + {"BATTLECON_EXHAUSTION", 16}, + {"BATTLECON_ENEMYHP", 32}, + {"BATTLECON_ACTORHP", 64}, + {"BATTLECON_ENEMYTURN", 128}, + {"BATTLECON_ACTORTURN", 256}, + {"BATTLECON_COMMAND", 512}, + + // Enemy Pattern + {"ENEMY_PATTERNTYPE_NORMAL", 0}, + {"ENEMY_PATTERNTYPE_SWITCH", 1}, + {"ENEMY_PATTERNTYPE_ROUND", 2}, + {"ENEMY_PATTERNTYPE_TROOP", 3}, + {"ENEMY_PATTERNTYPE_HP", 4}, + {"ENEMY_PATTERNTYPE_MP", 5}, + {"ENEMY_PATTERNTYPE_PARTYLEVEL", 6}, + {"ENEMY_PATTERNTYPE_PARTYSTATE", 7}, + + {"ENEMY_PATTERNCONTENT_ACTION", 0}, + {"ENEMY_PATTERNCONTENT_SKILL", 1}, + {"ENEMY_PATTERNCONTENT_MORPH", 2}, + + {"ENEMY_ACTION_NORMALATTACK", 0}, + {"ENEMY_ACTION_DOUBLEATTACK", 1}, + {"ENEMY_ACTION_DEFEND", 2}, + {"ENEMY_ACTION_WATCH", 3}, + {"ENEMY_ACTION_POWERUP", 4}, + {"ENEMY_ACTION_SELFDESTRUCT", 5}, + {"ENEMY_ACTION_ESCAPE", 6}, + {"ENEMY_ACTION_DONOTHING", 7}, + + // Actor Properties + {"ACTOR_LEVEL", 0}, + {"ACTOR_EXP", 1}, + {"ACTOR_HP", 2}, + {"ACTOR_MP", 3}, + {"ACTOR_MAXHP", 4}, + {"ACTOR_MAXMP", 5}, + {"ACTOR_STRENGTHACTOR_ATTACK", 6}, + {"ACTOR_DEFENSE", 7}, + {"ACTOR_MIND", 8}, + {"ACTOR_SPEEDACTOR_AGILITY", 9}, + {"ACTOR_WEAPON", 10}, + {"ACTOR_SHIELD", 11}, + {"ACTOR_ARMOR", 12}, + {"ACTOR_HELMET", 13}, + {"ACTOR_ACCESSORY", 14}, + + // Battle Settings + {"BATTLELAYOUT_CLASSIC", 0}, + {"BATTLELAYOUT_MODIFIED", 1}, + {"BATTLELAYOUT_FACES", 2}, + + {"BATTLEWINDOW_LARGE", 0}, + {"BATTLEWINDOW_SMALL", 1}, + + {"WINDOWBG_NORMAL", 0}, + {"WINDOWBG_TRANSPARENT", 1}, + + {"PARTYFORMATION_MANUAL", 0}, + {"PARTYFORMATION_AUTO", 1}, + + {"DEFEAT_GAMEOVER", 0}, + {"DEFEAT_COMMONEVENT", 1}, + + {"BATTLECOMMAND_ATTACK", 0}, + {"BATTLECOMMAND_ITEM", 1}, + {"BATTLECOMMAND_DEFEND", 2}, + {"BATTLECOMMAND_ESCAPE", 3}, + {"BATTLECOMMAND_SKILL", 4}, + {"BATTLECOMMAND_SPECIAL", 5}, + {"BATTLECOMMAND_EVENT", 6}, + + // Map Configuration + {"SCROLLTYPE_NORMAL", 0}, + {"SCROLLTYPE_VERTICAL", 1}, + {"SCROLLTYPE_HORIZONTAL", 2}, + {"SCROLLTYPE_BOTH", 3}, + + {"CONFIG_TES_FROMPARENT", 0}, + {"CONFIG_TES_ALLOW", 1}, + {"CONFIG_TES_FORBID", 2}, + + {"CONFIG_BM_FROMPARENT", 0}, + {"CONFIG_BM_DONTCHANGE", 1}, + {"CONFIG_BM_SPECIFY", 2}, + + // MapTree Data + {"TREE_ENTRY_FOLDER", 0}, + {"TREE_ENTRY_MAP", 1}, + {"TREE_ENTRY_AREA", 2}, + {"TREE_NO_PARENT", -1}, + + // Messagebox Buttons (Windows) + {"MB_IDOK", 1}, + {"MB_IDCANCEL", 2}, + {"MB_IDABORT", 3}, + {"MB_IDRETRY", 4}, + {"MB_IDIGNORE", 5}, + {"MB_IDYES", 6}, + {"MB_IDNO", 7}, + {"MB_IDTRYAGAIN", 10}, + {"MB_IDCONTINUE", 11}, + + // MessageBox Flags (Windows) + {"MB_OK", 0}, + {"MB_OKCANCEL", 1}, + {"MB_ABORTRETRYIGNORE", 2}, + {"MB_YESNOCANCEL", 3}, + {"MB_YESNO", 4}, + {"MB_RETRYCANCEL", 5}, + {"MB_CANCELTRYCONTINUE", 6}, + + {"MB_ICONHAND", 16}, + {"MB_ICONERROR", 16}, // REPEAT + {"MB_ICONSTOP", 16}, // REPEAT + + {"MB_ICONQUESTION", 32}, + {"MB_ICONEXCLAMATION", 48}, + {"MB_ICONWARNING", 48}, // REPEAT + {"MB_ICONASTERISK", 64}, + {"MB_ICONFORMATION", 64}, // REPEAT + + {"MB_DEFBUTTON1", 0}, + {"MB_DEFBUTTON2", 256}, + {"MB_DEFBUTTON3", 512}, + {"MB_DEFBUTTON4", 768}, + {"MB_HELP", 16384}, + {"MB_APPLMODAL", 0}, + {"MB_SYSTEMMODAL", 4096}, + {"MB_TASKMODAL", 8192}, + {"MB_DEFAULT_DESKTOP_ONLY", 131072}, + {"MB_RIGHT", 524288}, + {"MB_RTLREADING", 1048576}, + {"MB_SETFOREGROUND", 65536}, + {"MB_TOPMOST", 262144}, + {"MB_SERVICE_NOTIFICATION", 2097152}, + + // Months + {"JANUARY", 1}, + {"FEBRUARY", 2}, + {"MARCH", 3}, + {"APRIL", 4}, + {"MAY", 5}, + {"JUNE", 6}, + {"JULY", 7}, + {"AUGUST", 8}, + {"SEPTEMBER", 9}, + {"OCTOBER", 10}, + {"NOVEMBER", 11}, + {"DECEMBER", 12}, + + // Music Backups + {"MUSICBACKUP_EVENT", 0}, + {"MUSICBACKUP_BATTLE_FIELD", 1}, + {"MUSICBACKUP_VEHICLE_WALK", 2}, + + // Message Options + {"MESSAGEFORMAT_NORMAL", 0}, + {"MESSAGEFORMAT_TRANSPARENT", 1}, + + {"MESSAGEPOSITION_TOP", 0}, + {"MESSAGEPOSITION_MIDDLE", 1}, + {"MESSAGEPOSITION_BOTTOM", 2}, + + {"FACEPOSITION_LEFT", 0}, + {"FACEPOSITION_RIGHT", 1}, + + // EnterName Pages + {"ENTRYPAGE_HIRAGANA", 0}, + {"ENTRYPAGE_KATAKANA", 1}, + + // Picture Properties + {"ACTION_NONE", 0}, + {"ACTION_ROTATION", 1}, + {"ACTION_WAVE", 2}, + {"ACTION_RIPPLE", 2}, // LEGACY + {"ACTION_SETANGLE", 3}, + + {"BLEND_MULTIPLY", 1}, + {"BLEND_ADD", 2}, + {"BLEND_OVERLAY", 3}, + + {"PICTUREDELETE_TELEPORT", 1}, + {"PICTUREDELETE_BATTLEEND", 2}, + {"PICTUREAFFECT_TINT", 16}, + {"PICTUREAFFECT_FLASH", 32}, + {"PICTUREAFFECT_SHAKE", 64}, + + {"PICTUREMIRROR_HORIZONTAL", 1}, + {"PICTUREMIRROR_VERTICAL", 2}, + {"PICTUREMIRROR_TURN180", 3}, + + {"PICTURESHEET_ANIMLOOP", -1}, + {"PICTURESHEET_ANIMDELETE", -2}, + + {"FIELDPICTURE_PANORAMA", 1}, + {"FIELDPICTURE_TILESBELOW", 2}, + {"FIELDPICTURE_EVENTSBELOW", 3}, + {"FIELDPICTURE_PASSSAME", 4}, + {"FIELDPICTURE_TILESABOVE", 5}, + {"FIELDPICTURE_EVENTSABOVE", 6}, + {"FIELDPICTURE_WEATHER", 7}, + {"FIELDPICTURE_ANIMATION", 8}, + {"FIELDPICTURE_MESSAGE", 9}, + {"FIELDPICTURE_TIMER", 10}, + + {"BATTLEPICTURE_BACKDROP", 1}, + {"BATTLEPICTURE_ANIMATION", 2}, + {"BATTLEPICTURE_WEATHER", 3}, + {"BATTLEPICTURE_WINDOW", 4}, + {"BATTLEPICTURE_TIMER", 5}, + + // Skill Properties + {"SKILLSCOPE_SINGLEENEMY", 0}, + {"SKILLSCOPE_ALLENEMIES", 1}, + {"SKILLSCOPE_USER", 2}, + {"SKILLSCOPE_SELF", 2}, // LEGACY + {"SKILLSCOPE_SINGLEALLY", 3}, + {"SKILLSCOPE_ALLALLIES", 4}, + + {"SKILLMP_NORMAL", 0}, + {"SKILLMP_PERCENT", 1}, + + {"ACTORMOVE_NONE", 0}, + {"ACTORMOVE_STEP", 1}, + {"ACTORMOVE_JUMP", 2}, + {"ACTORMOVE_DASH", 3}, + + // Skill Categories + {"SKILLTYPE_NORMAL", 0}, + {"SKILLTYPE_TELEPORT", 1}, + {"SKILLTYPE_ESCAPE", 2}, + {"SKILLTYPE_SWITCH", 3}, + + // Socket Data + {"SOCK_DESTINY", 0}, + {"SOCK_RAW", 1}, + {"NEXT_FREE_SOCKET", -1}, + {"SOCKETFLAG_NOTBLOCKING", 1}, + + {"STATE_ERROR", -1}, + {"STATE_CLOSED", 0}, + {"STATE_CONNECTED", 1}, + {"STATE_LISTENING", 2}, + {"STATE_CLOSING", 3}, + {"STATE_CONNECTING", 4}, + + // Daylight Savings + {"DAYLIGHT_UNUSED", 0}, + {"DAYLIGHT_STANDARD", 1}, + {"DAYLIGHT_SUMMER", 2}, + + // Game Engine Scenes + {"SCENE_MAP", 0}, + {"SCENE_MENU", 1}, + {"SCENE_BATTLE", 2}, + {"SCENE_SHOP", 3}, + {"SCENE_NAME", 4}, + {"SCENE_FILE", 5}, + {"SCENE_TITLE", 6}, + {"SCENE_GAMEOVER", 7}, + {"SCENE_DEBUG", 8}, + {"SCENE_FREEZE", 9}, + + // Destiny Text Languages + {"LANGUAGE_GERMAN", 0}, + {"LANGUAGE_ENGLISH", 1}, + + // Battler Parameters + {"PARAMETER_MAXHP", 0}, + {"ABILITY_MAXHP", 0}, // LEGACY + {"PARAMETER_MAXMP", 1}, + {"ABILITY_MAXMP", 1}, // LEGACY + {"PARAMETER_STRENGTH", 2}, + {"ABILITY_ATTACK", 2}, // LEGACY + {"PARAMETER_DEFENSE", 3}, + {"ABILITY_DEFENSE", 3}, // LEGACY + {"PARAMETER_MIND", 4}, + {"ABILITY_MIND", 4}, // LEGACY + {"PARAMETER_SPEED", 5}, + {"ABILITY_AGILITY", 5}, // LEGACY + + // Switch Operations + {"OPERATION_ON", 0}, + {"OPERATION_OFF", 1}, + {"OPERATION_TOGGLE", 2}, + + // SystemGrapic Properties + {"DISPLAY_STRETCH", 0}, + {"DISPLAY_REPEAT", 1}, + + {"FONT_RMG2000", 0}, + {"FONT_RM2000", 1}, + + // System Music + {"MUSIC_BATTLE", 0}, + {"MUSIC_VICTORY", 1}, + {"MUSIC_INN", 2}, + {"MUSIC_BOAT", 3}, + {"MUSIC_SHIP", 4}, + {"MUSIC_AIRSHIP", 5}, + {"MUSIC_GAMEOVER", 6}, + + // System Sounds + {"SOUND_CURSOR", 0}, + {"SOUND_DECISION", 1}, + {"SOUND_CANCEL", 2}, + {"SOUND_BUZZER", 3}, + {"SOUND_STARTBATTLE", 4}, + {"SOUND_ESCAPE", 5}, + {"SOUND_ENEMYATTACK", 6}, + {"SOUND_ENEMYDAMAGE", 7}, + {"SOUND_ACTORDAMAGE", 8}, + {"SOUND_ALLYDAMAGE", 8}, // LEGACY + {"SOUND_EVASION", 9}, + {"SOUND_KNOCKOUT", 10}, + {"SOUND_ENEMYDIE", 10}, // LEGACY + {"SOUND_ITEMUSE", 11}, + + // Keyboard Types + {"KB_UNKNOWN", 0}, + {"KB_IBM_PC_XT", 1}, + {"KB_OLIVETTI_ICO", 2}, + {"KB_IPM_PC_AT", 3}, + {"KB_IBM_ENHANCED", 4}, + {"KB_NOKIA_1050", 5}, + {"KB_NOKIA_9140", 6}, + {"KB_JAPANESE", 7}, + + // Terrain Eventoid Transparency + {"TERRAIN_EVENT_NORMAL", 0}, + {"TERRAIN_EVENT_THIRD", 1}, + {"TERRAIN_EVENT_HALF", 2}, + {"TERRAIN_EVENT_TRANSPARENT", 3}, + + // Text Colors + {"TEXTCOLOR_DEFAULT", 0}, + {"TEXTCOLOR_BLUE", 1}, + {"TEXTCOLOR_LIGHTORANGE", 2}, + {"TEXTCOLOR_BLACK", 3}, + {"TEXTCOLOR_YELLOW", 4}, + {"TEXTCOLOR_RED", 5}, + {"TEXTCOLOR_PURPLE", 6}, + {"TEXTCOLOR_MAGENTA", 7}, + {"TEXTCOLOR_ORANGE", 8}, + {"TEXTCOLOR_GREEN", 9}, + {"TEXTCOLOR_LIGHTBLUE", 10}, + {"TEXTCOLOR_LIGHTRED", 11}, + {"TEXTCOLOR_OCHER", 12}, + {"TEXTCOLOR_VIOLET", 13}, + {"TEXTCOLOR_GOLD", 14}, + {"TEXTCOLOR_SEAGREEN", 15}, + {"TEXTCOLOR_DARKPURPLE", 16}, + {"TEXTCOLOR_OCEANBLUE", 17}, + {"TEXTCOLOR_TEAL", 18}, + {"TEXTCOLOR_BROWN", 19}, + + // Tileset Properties + {"TILESET_SEQUENCE_BIDI", 0}, + {"TILESET_SEQUENCE_FORWARD", 1}, + + {"TILESET_ANIMATION_SLOW", 0}, + {"TILESET_ANIMATION_FAST", 1}, + + {"TILEPASS_BLOCKED", 0}, + {"TILEPASS_PASSABLE", 15}, + {"TILEPASS_BOTTOM", 1}, + {"TILEPASS_LEFT", 2}, + {"TILEPASS_RIGHT", 4}, + {"TILEPASS_TOP", 8}, + {"TILEPASS_ABOVE", 16}, + {"TILEPASS_TOPABOVE", 48}, + {"TILEPASS_BRIDGE", 64}, + + // Transitions + {"ERASE_FADEOUT", 0}, + {"ERASE_RANDOMBLOCKS", 1}, + {"ERASE_RANDOMBLOCKSUP", 2}, + {"ERASE_RANDOMBLOCKSDOWN", 3}, + {"ERASE_BLINDCLOSE", 4}, + {"ERASE_STRIPEUPDOWN", 5}, + {"ERASE_STRIPELEFTRIGHT", 6}, + {"ERASE_OUTSIDEINSIDE", 7}, + {"ERASE_INSIDEOUTSIDE", 8}, + {"ERASE_SCROLLUP", 9}, + {"ERASE_SCROLLDOWN", 10}, + {"ERASE_SCROLLLEFT", 11}, + {"ERASE_SCROLLRIGHT", 12}, + {"ERASE_DIVISIONUPDOWN", 13}, + {"ERASE_DIVISIONLEFTRIGHT", 14}, + {"ERASE_DIVISION", 15}, + {"ERASE_ZOOMIN", 16}, + {"ERASE_MOSAIC", 17}, + {"ERASE_ROSTERSCROLL", 18}, + {"ERASE_INSTANT", 19}, + {"ERASE_DONTERASE", 20}, + + {"SHOW_FADEIN", 0}, + {"SHOW_RANDOMBLOCKS", 1}, + {"SHOW_RANDOMBLOCKSUP", 2}, + {"SHOW_RANDOMBLOCKSDOWN", 3}, + {"SHOW_BLINDOPEN", 4}, + {"SHOW_STRIPEUPDOWN", 5}, + {"SHOW_STRIPELEFTRIGHT", 6}, + {"SHOW_OUTSIDEINSIDE", 7}, + {"SHOW_INSIDEOUTSIDE", 8}, + {"SHOW_SCROLLDOWN", 9}, + {"SHOW_SCROLLUP", 10}, + {"SHOW_SCROLLRIGHT", 11}, + {"SHOW_SCROLLLEFT", 12}, + {"SHOW_COMBINEUPDOWN", 13}, + {"SHOW_COMBINELEFTRIGHT", 14}, + {"SHOW_COMBINE", 15}, + {"SHOW_ZOOMOUT", 16}, + {"SHOW_MOSAIC", 17}, + {"SHOW_ROSTERSCROLL", 18}, + {"SHOW_INSTANT", 19}, + + // Transition Categories + {"TRANSITION_TELEPORTERASE", 0}, + {"TRANSITION_TELEPORTSHOW", 1}, + {"TRANSITION_BATTLESTARTERASE", 2}, + {"TRANSITION_BATTLESTARTSHOW", 3}, + {"TRANSITION_BATTLEENDERASE", 4}, + {"TRANSITION_BATTLEENDSHOW", 5}, + + // Variables Misc Operands + {"MISC_MONEY", 0}, + {"MISC_TIMERSECONDSLEFT", 1}, + {"MISC_PARTYSIZE", 2}, + {"MISC_NUMBEROFSAVES", 3}, + {"MISC_NUMBEROFBATTLES", 4}, + {"MISC_NUMBEROFVICTORIES", 5}, + {"MISC_NUMBEROFDEFEATS", 6}, + {"MISC_NUMBEROFESCAPES", 7}, + {"MISC_MIDITICK", 8}, + + // Variable Operations + {"OPERATION_SET", 0}, + {"OPERATION_ADD", 1}, + {"OPERATION_SUBTRACT", 2}, + {"OPERATION_MULTIPLY", 3}, + {"OPERATION_DIVIDE", 4}, + {"OPERATION_MODULO", 5}, + + // Variable Value Types + {"VALUETYPE_NUMBER", 0}, + {"VALUETYPE_VARIABLE", 1}, + {"VALUETYPE_VARIABLEINDIRECT", 2}, + {"VALUETYPE_RANDOM", 3}, + {"VALUETYPE_ITEM", 4}, + {"VALUETYPE_ACTOR", 5}, + {"VALUETYPE_EVENT", 6}, + {"VALUETYPE_MISC", 7}, + + // VirtualKey Codes + {"VK_A", 65}, + {"VK_B", 66}, + {"VK_C", 67}, + {"VK_D", 68}, + {"VK_E", 69}, + {"VK_F", 70}, + {"VK_G", 71}, + {"VK_H", 72}, + {"VK_I", 73}, + {"VK_J", 74}, + {"VK_K", 75}, + {"VK_L", 76}, + {"VK_M", 77}, + {"VK_N", 78}, + {"VK_O", 79}, + {"VK_P", 80}, + {"VK_Q", 81}, + {"VK_R", 82}, + {"VK_S", 83}, + {"VK_T", 84}, + {"VK_U", 85}, + {"VK_V", 86}, + {"VK_W", 87}, + {"VK_X", 88}, + {"VK_Y", 89}, + {"VK_Z", 90}, + + {"VK_0", 48}, + {"VK_1", 49}, + {"VK_2", 50}, + {"VK_3", 51}, + {"VK_4", 52}, + {"VK_5", 53}, + {"VK_6", 54}, + {"VK_7", 55}, + {"VK_8", 56}, + {"VK_9", 57}, + + {"VK_F1", 112}, + {"VK_F2", 113}, + {"VK_F3", 114}, + {"VK_F4", 115}, + {"VK_F5", 116}, + {"VK_F6", 117}, + {"VK_F7", 118}, + {"VK_F8", 119}, + {"VK_F9", 120}, + {"VK_F10", 121}, + {"VK_F11", 122}, + {"VK_F12", 123}, + + {"VK_LEFT", 37}, + {"VK_RIGHT", 39}, + {"VK_UP", 38}, + {"VK_DOWN", 40}, + + {"VK_TAB", 9}, + {"VK_MENU", 18}, + {"VK_SHIFT", 16}, + {"VK_SPACE", 32}, + {"VK_CONTROL", 17}, + {"VK_RETURN", 13}, + {"VK_ESCAPE", 27}, + {"VK_PAUSE", 19}, + {"VK_PRINT", 42}, + {"VK_SCROLL", 145}, + {"VK_CAPITAL", 20}, + {"VK_BACK", 8}, + + {"VK_HOME", 36}, + {"VK_END", 35}, + {"VK_PGUP", 33}, + {"VK_PGDOWN", 34}, + {"VK_INSERT", 45}, + {"VK_DELETE", 46}, + + {"VK_LBUTTON", 1}, + {"VK_MBUTTON", 4}, + {"VK_RBUTTON", 2}, + + {"VK_NUMLOCK", 144}, + {"VK_NUMPAD0", 96}, + {"VK_NUMPAD1", 97}, + {"VK_NUMPAD2", 98}, + {"VK_NUMPAD3", 99}, + {"VK_NUMPAD4", 100}, + {"VK_NUMPAD5", 101}, + {"VK_NUMPAD6", 102}, + {"VK_NUMPAD7", 103}, + {"VK_NUMPAD8", 104}, + {"VK_NUMPAD9", 105}, + {"VK_ADD", 107}, + {"VK_SUBTRACT", 109}, + {"VK_MULTIPLY", 106}, + {"VK_DIVIDE", 111}, + {"VK_DECIMAL", 110}, + + // Vocab + {"VOCAB_ENEMY_APPEARS", 1}, + {"VOCAB_FIRST_ATTACK", 2}, + {"VOCAB_RUN_SUCCESS", 3}, + {"VOCAB_RUN_FAIL", 4}, + {"VOCAB_BATTLE_WIN", 5}, + {"VOCAB_BATTLE_LOSE", 6}, + {"VOCAB_EXP_GET", 7}, + {"VOCAB_MONEY_GET1", 8}, + {"VOCAB_MONEY_GET2", 9}, + {"VOCAB_ITEM_GET", 10}, + {"VOCAB_ATTACKING", 11}, + {"VOCAB_ACTOR_CRITICAL", 12}, + {"VOCAB_ENEMY_CRITICAL", 13}, + {"VOCAB_DEFENDING", 14}, + {"VOCAB_ENEMY_WAIT", 15}, + {"VOCAB_ENEMY_POWERUP", 16}, + {"VOCAB_ENEMY_EXPLODE", 17}, + {"VOCAB_ENEMY_ESCAPE", 18}, + {"VOCAB_ENEMY_MORPH", 19}, + {"VOCAB_ENEMY_DAMAGE", 20}, + {"VOCAB_ENEMY_EVADE", 21}, + {"VOCAB_ACTOR_DAMAGE", 22}, + {"VOCAB_ACTOR_EVADE", 23}, + {"VOCAB_SKILL_FAIL1", 24}, + {"VOCAB_SKILL_FAIL2", 25}, + {"VOCAB_SKILL_FAIL3", 26}, + {"VOCAB_ATTACK_FAIL", 27}, + {"VOCAB_ITEM_USE", 28}, + {"VOCAB_PARAMETER_NORMALIZE", 29}, + {"VOCAB_PARAMETER_BUFF", 30}, + {"VOCAB_PARAMETER_DEBUFF", 31}, + {"VOCAB_ACTOR_ABSORB", 32}, + {"VOCAB_ENEMY_ABSORB", 33}, + {"VOCAB_ELEMENT_BUFF", 34}, + {"VOCAB_ELEMENT_DEBUFF", 35}, + {"VOCAB_LEVEL_UP", 36}, + {"VOCAB_SKILL_LEARN", 37}, + + {"VOCAB_SHOP1_WELCOME1", 85}, + {"VOCAB_SHOP1_WELCOME2", 86}, + {"VOCAB_SHOP1_BUY", 87}, + {"VOCAB_SHOP1_SELL", 88}, + {"VOCAB_SHOP1_LEAVE", 89}, + {"VOCAB_SHOP1_BUY_ITEM", 90}, + {"VOCAB_SHOP1_BUY_AMOUNT", 91}, + {"VOCAB_SHOP1_BUY_DONE", 92}, + {"VOCAB_SHOP1_SELL_ITEM", 93}, + {"VOCAB_SHOP1_SELL_AMOUNT", 94}, + {"VOCAB_SHOP1_SELL_DONE", 95}, + + {"VOCAB_SHOP2_WELCOME1", 98}, + {"VOCAB_SHOP2_WELCOME2", 99}, + {"VOCAB_SHOP2_BUY", 100}, + {"VOCAB_SHOP2_SELL", 101}, + {"VOCAB_SHOP2_LEAVE", 102}, + {"VOCAB_SHOP2_BUY_ITEM", 103}, + {"VOCAB_SHOP2_BUY_AMOUNT", 104}, + {"VOCAB_SHOP2_BUY_DONE", 105}, + {"VOCAB_SHOP2_SELL_ITEM", 106}, + {"VOCAB_SHOP2_SELL_AMOUNT", 107}, + {"VOCAB_SHOP2_SELL_DONE", 108}, + {"VOCAB_SHOP3_WELCOME1", 111}, + {"VOCAB_SHOP3_WELCOME2", 112}, + {"VOCAB_SHOP3_BUY", 113}, + {"VOCAB_SHOP3_SELL", 114}, + {"VOCAB_SHOP3_LEAVE", 115}, + {"VOCAB_SHOP3_BUY_ITEM", 116}, + {"VOCAB_SHOP3_BUY_AMOUNT", 117}, + {"VOCAB_SHOP3_BUY_DONE", 118}, + {"VOCAB_SHOP3_SELL_ITEM", 119}, + {"VOCAB_SHOP3_SELL_AMOUNT", 120}, + {"VOCAB_SHOP3_SELL_DONE", 121}, + {"VOCAB_INN1_COSTS1", 124}, + {"VOCAB_INN1_COSTS2", 125}, + {"VOCAB_INN1_QUESTION", 126}, + {"VOCAB_INN1_YES", 127}, + {"VOCAB_INN1_NO", 128}, + {"VOCAB_INN2_COSTS1", 131}, + {"VOCAB_INN2_COSTS2", 132}, + {"VOCAB_INN2_QUESTION", 133}, + {"VOCAB_INN2_YES", 134}, + {"VOCAB_INN2_NO", 135}, + {"VOCAB_LEVEL", 59}, + {"VOCAB_HP", 60}, + {"VOCAB_MP", 61}, + + {"VOCAB_NORMAL", 62}, + {"VOCAB_MP_COSTS", 67}, + + {"VOCAB_STRENGTH", 68}, + {"VOCAB_DEFENSE", 69}, + {"VOCAB_MIND", 70}, + {"VOCAB_SPEED", 71}, + + {"VOCAB_SHORT_LEVEL", 64}, + {"VOCAB_SHORT_HP", 65}, + {"VOCAB_SHORT_MP", 66}, + {"VOCAB_SHORT_EXP", 63}, + + {"VOCAB_WEAPON", 72}, + {"VOCAB_SHIELD", 73}, + {"VOCAB_ARMOR", 74}, + {"VOCAB_HELMET", 75}, + {"VOCAB_ACCESSORY", 76}, + + {"VOCAB_BATTLECOMMAND_START", 46}, + {"VOCAB_BATTLECOMMAND_AUTO", 47}, + {"VOCAB_BATTLECOMMAND_ESCAPE", 48}, + {"VOCAB_ACTORCOMMAND_ATTACK", 49}, + {"VOCAB_ACTORCOMMAND_DEFEND", 50}, + + {"VOCAB_PARTY_ITEMS", 51}, + {"VOCAB_PARTY_SKILLS", 52}, + {"VOCAB_PARTY_EQUIPMENT", 53}, + {"VOCAB_PARTY_SAVE", 54}, + {"VOCAB_PARTY_QUIT", 55}, + + {"VOCAB_TITLE_NEWGAME", 56}, + {"VOCAB_TITLE_CONTINUE", 57}, + {"VOCAB_TITLE_SHUTDOWN", 58}, + + {"VOCAB_FILESELECT_SAVE", 77}, + {"VOCAB_FILESELECT_LOAD", 78}, + {"VOCAB_EXIT_MESSAGE", 79}, + {"VOCAB_FILE", 80}, + {"VOCAB_EXIT_YES", 81}, + {"VOCAB_EXIT_NO", 82}, + + {"VOCAB_ITEMS_INVENTORY", 43}, + {"VOCAB_ITEMS_EQUIPPED", 44}, + {"VOCAB_CURRENCY", 45}, + + // Weather Properties + {"WEATHER_NONE", 0}, + {"WEATHER_RAIN", 1}, + {"WEATHER_SNOW", 2}, + {"WEATHER_FOG", 3}, + {"WEATHER_SANDSTORM", 4}, + + {"WEATHERSTRENGTH_WEAK", 0}, + {"WEATHERSTRENGTH_NORMAL", 1}, + {"WEATHERSTRENGTH_STRONG", 2}, + + // Angle Formats + {"DEG", 1}, + {"RAD", 2}, + {"GRAD", 3}, + {"RPG", 4}, + + // Weekdays + {"SUNDAY", 0}, + {"MONDAY", 1}, + {"TUESDAY", 2}, + {"WEDNESDAY", 3}, + {"THURSDAY", 4}, + {"FRIDAY", 5}, + {"SATURDAY", 6}, + + // State Properties + {"STATETYPE_COMBAT", 0}, + {"CONDITIONTYPE_COMBAT", 0}, // LEGACY + {"STATETYPE_MOVEMENT", 1}, + {"CONDITIONTYPE_MOVEMENT", 1}, // LEGACY + + {"STATELIMITATION_NONE", 0}, + {"CONDITIONLIMITATION_NONE", 0}, // LEGACY + {"STATELIMITATION_NOACTION", 1}, + {"CONDITIONLIMITATION_NOACTION", 1}, // LEGACY + {"STATELIMITATION_ATTACKENEMY", 2}, + {"CONDITIONLIMITATION_ATTACKENEMY", 2}, // LEGACY + {"STATELIMITATION_ATTACKALLY", 3}, + {"CONDITIONLIMITATION_ATTACKALLY", 3} // LEGACY + + }; + EventCode = { + // AJUDAS + {"END", 10}, + {"CallCommonEvent", 1005}, + {"ForceFlee", 1006}, + {"EnableCombo", 1007}, + {"ChangeClass", 1008}, + {"ChangeBattleCommands", 1009}, + {"OpenLoadMenu", 5001}, + {"ExitGame", 5002}, + {"ToggleAtbMode", 5003}, + {"ToggleFullscreen", 5004}, + {"OpenVideoOptions", 5005}, + {"ShowMessage", 10110}, + {"MessageOptions", 10120}, + {"ChangeFaceGraphic", 10130}, + {"ShowChoice", 10140}, + {"InputNumber", 10150}, + {"ControlSwitches", 10210}, + {"ControlVars", 10220}, + {"TimerOperation", 10230}, + {"ChangeGold", 10310}, + {"ChangeItems", 10320}, + {"ChangePartyMembers", 10330}, + {"ChangeExp", 10410}, + {"ChangeLevel", 10420}, + {"ChangeParameters", 10430}, + {"ChangeSkills", 10440}, + {"ChangeEquipment", 10450}, + {"ChangeHP", 10460}, + {"ChangeSP", 10470}, + {"ChangeCondition", 10480}, + {"FullHeal", 10490}, + {"SimulatedAttack", 10500}, + {"ChangeHeroName", 10610}, + {"ChangeHeroTitle", 10620}, + {"ChangeSpriteAssociation", 10630}, + {"ChangeActorFace", 10640}, + {"ChangeVehicleGraphic", 10650}, + {"ChangeSystemBGM", 10660}, + {"ChangeSystemSFX", 10670}, + {"ChangeSystemGraphics", 10680}, + {"ChangeScreenTransitions", 10690}, + {"EnemyEncounter", 10710}, + {"OpenShop", 10720}, + {"ShowInn", 10730}, + {"EnterHeroName", 10740}, + {"Teleport", 10810}, + {"MemorizeLocation", 10820}, + {"RecallToLocation", 10830}, + {"EnterExitVehicle", 10840}, + {"SetVehicleLocation", 10850}, + {"ChangeEventLocation", 10860}, + {"TradeEventLocations", 10870}, + {"StoreTerrainID", 10910}, + {"StoreEventID", 10920}, + {"EraseScreen", 11010}, + {"ShowScreen", 11020}, + {"TintScreen", 11030}, + {"FlashScreen", 11040}, + {"ShakeScreen", 11050}, + {"PanScreen", 11060}, + {"WeatherEffects", 11070}, + {"ShowPicture", 11110}, + {"MovePicture", 11120}, + {"ErasePicture", 11130}, + {"ShowBattleAnimation", 11210}, + {"PlayerVisibility", 11310}, + {"FlashSprite", 11320}, + {"MoveEvent", 11330}, + {"ProceedWithMovement", 11340}, + {"HaltAllMovement", 11350}, + {"Wait", 11410}, + {"PlayBGM", 11510}, + {"FadeOutBGM", 11520}, + {"MemorizeBGM", 11530}, + {"PlayMemorizedBGM", 11540}, + {"PlaySound", 11550}, + {"PlayMovie", 11560}, + {"KeyInputProc", 11610}, + {"ChangeMapTileset", 11710}, + {"ChangePBG", 11720}, + {"ChangeEncounterSteps", 11740}, + {"TileSubstitution", 11750}, + {"TeleportTargets", 11810}, + {"ChangeTeleportAccess", 11820}, + {"EscapeTarget", 11830}, + {"ChangeEscapeAccess", 11840}, + {"OpenSaveMenu", 11910}, + {"ChangeSaveAccess", 11930}, + {"OpenMainMenu", 11950}, + {"ChangeMainMenuAccess", 11960}, + {"ConditionalBranch", 12010}, + {"Label", 12110}, + {"JumpToLabel", 12120}, + {"Loop", 12210}, + {"BreakLoop", 12220}, + {"EndEventProcessing", 12310}, + {"EraseEvent", 12320}, + {"CallEvent", 12330}, + {"Comment", 12410}, + {"GameOver", 12420}, + {"ReturntoTitleScreen", 12510}, + {"ChangeMonsterHP", 13110}, + {"ChangeMonsterMP", 13120}, + {"ChangeMonsterCondition", 13130}, + {"ShowHiddenMonster", 13150}, + {"ChangeBattleBG", 13210}, + {"ShowBattleAnimation_B", 13260}, + {"ConditionalBranch_B", 13310}, + {"TerminateBattle", 13410}, + {"ShowMessage_2", 20110}, + {"ShowChoiceOption", 20140}, + {"ShowChoiceEnd", 20141}, + {"VictoryHandler", 20710}, + {"EscapeHandler", 20711}, + {"DefeatHandler", 20712}, + {"EndBattle", 20713}, + {"Transaction", 20720}, + {"NoTransaction", 20721}, + {"EndShop", 20722}, + {"Stay", 20730}, + {"NoStay", 20731}, + {"EndInn", 20732}, + {"ElseBranch", 22010}, + {"EndBranch", 22011}, + {"EndLoop", 22210}, + {"Comment_2", 22410}, + {"ElseBranch_B", 23310}, + {"EndBranch_B", 23311}, + {"Maniac_GetSaveInfo", 3001}, + {"Maniac_Save", 3002}, + {"Maniac_Load", 3003}, + {"Maniac_EndLoadProcess", 3004}, + {"Maniac_GetMousePosition", 3005}, + {"Maniac_SetMousePosition", 3006}, + {"Maniac_ShowStringPicture", 3007}, + {"Maniac_GetPictureInfo", 3008}, + {"Maniac_ControlBattle", 3009}, + {"Maniac_ControlAtbGauge", 3010}, + {"Maniac_ChangeBattleCommandEx", 3011}, + {"Maniac_GetBattleInfo", 3012}, + {"Maniac_ControlVarArray", 3013}, + {"Maniac_KeyInputProcEx", 3014}, + {"Maniac_RewriteMap", 3015}, + {"Maniac_ControlGlobalSave", 3016}, + {"Maniac_ChangePictureId", 3017}, + {"Maniac_SetGameOption", 3018}, + {"Maniac_CallCommand", 3019}, + {"Maniac_ControlStrings", 3020}, + {"Maniac_GetGameInfo", 3021}, + {"Maniac_EditPicture", 3025}, + {"Maniac_WritePicture", 3026}, + {"Maniac_AddMoveRoute", 3027}, + {"Maniac_EditTile", 3028}, + {"Maniac_ControlTextProcessing", 3029} + }; + }; + + std::string toLower(const std::string& str) { + std::string result = str; + std::transform(result.begin(), result.end(), result.begin(), ::tolower); + return result; + }; + + // Function to return contents from either DestinyScript or EventCode + std::unordered_map& MapCollection(const std::string& collectionName) { + std::string lowercaseCollectionName = toLower(collectionName); + + if (lowercaseCollectionName == "destinyscript") return DestinyScript; + else if (lowercaseCollectionName == "eventcode") return EventCode; + else return EventCode; + + }; + + // Function to retrieve a value from mapCollection ignoring case + std::string get(const std::string& mapName, const std::string& key) { + std::unordered_map& selectedMap = MapCollection(mapName); + std::string lowercaseKey = toLower(key); + + for (auto it = selectedMap.begin(); it != selectedMap.end(); ++it) { + std::string lowercaseMapKey = toLower(it->first); + + if (lowercaseMapKey == lowercaseKey) return std::to_string(it->second); + } + + return "0"; // Key not found + } +}; diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index 3d0ec73653..2206b2f484 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -105,9 +105,23 @@ bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpr return true; } - std::tie(cmd.code) = DynRpg::ParseArgs(func, args, &okay); + Constants Constants; + std::string keyToPrint = "ENABLE"; + + //Output::Warning("Key {}, value {}", keyToPrint, Constants.get("DestinyScript",keyToPrint)); + + auto evt = args[0]; + if (evt.find("@") == 0) { + evt = evt.substr(1); + evt = Constants.get("EventCode", evt); + + cmd.code = stoi(evt); + okay = bool(cmd.code); + } else + std::tie(cmd.code) = DynRpg::ParseArgs(func, args, &okay); if (!okay) { + Output::Warning("EasyRpgPlugin - Unknown Input: {}",args[0]); return true; } diff --git a/src/dynrpg_easyrpg.h b/src/dynrpg_easyrpg.h index 0f9e41e446..4499df9895 100644 --- a/src/dynrpg_easyrpg.h +++ b/src/dynrpg_easyrpg.h @@ -21,6 +21,7 @@ #include "game_dynrpg.h" #include "game_battle.h" #include "game_map.h" +#include "constants.h" namespace DynRpg { /** From 5ea5896b74f4738555a5b0f78cbbd88a6a942a35 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Tue, 31 Oct 2023 14:13:53 -0300 Subject: [PATCH 07/14] DynRPG Raw - fill numeric parameters. --- src/dynrpg_easyrpg.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index 2206b2f484..c772a5e6fb 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -135,7 +135,7 @@ bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpr for (size_t i = 2; i < args.size(); ++i) { auto [int_arg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); - + output_args.push_back(int_arg); if (!okay) { return true; } From a74265ebdb48c28aa5d8a7a41e66dd6a82e79fdb Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Tue, 31 Oct 2023 21:56:49 -0300 Subject: [PATCH 08/14] DynRPG - easyrpg_raw: Numeric Parameters can call destiny constants implementation is a bit ugly, need help to clear this --- src/dynrpg_easyrpg.cpp | 20 ++++++++++++++++++-- 1 file changed, 18 insertions(+), 2 deletions(-) diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index c772a5e6fb..2f7b543b4e 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -134,8 +134,24 @@ bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpr } for (size_t i = 2; i < args.size(); ++i) { - auto [int_arg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); - output_args.push_back(int_arg); + + + auto currArg = args[i]; + + if (currArg.find("@") == 0) { + currArg = currArg.substr(1); + currArg = Constants.get("DestinyScript", currArg); + + auto int_arg = stoi(currArg); + okay = true; + output_args.push_back(int_arg); + } + else { + auto [int_arg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); + output_args.push_back(int_arg); + } + + if (!okay) { return true; } From cdceab64fd8f55a08967e3447f118818ab5d434d Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Mon, 6 Nov 2023 01:23:28 -0300 Subject: [PATCH 09/14] DynRPG Easyrpg_raw - simplify commands removed repeated code, and overcomplex logic. --- src/constants.h | 24 +++-------- src/dynrpg_easyrpg.cpp | 93 ++++++++++++++++-------------------------- 2 files changed, 42 insertions(+), 75 deletions(-) diff --git a/src/constants.h b/src/constants.h index 39f84a8221..c5c96bd27d 100644 --- a/src/constants.h +++ b/src/constants.h @@ -15,6 +15,7 @@ * along with EasyRPG Player. If not, see . */ +#include "utils.h" #include #include #include @@ -1178,7 +1179,6 @@ class Constants { }; EventCode = { - // AJUDAS {"END", 10}, {"CallCommonEvent", 1005}, {"ForceFlee", 1006}, @@ -1336,33 +1336,21 @@ class Constants { }; }; - std::string toLower(const std::string& str) { - std::string result = str; - std::transform(result.begin(), result.end(), result.begin(), ::tolower); - return result; - }; - // Function to return contents from either DestinyScript or EventCode std::unordered_map& MapCollection(const std::string& collectionName) { - std::string lowercaseCollectionName = toLower(collectionName); + if ( Utils::StrICmp(collectionName, "destinyscript") == 0 ) return DestinyScript; + else if ( Utils::StrICmp(collectionName, "eventcode") == 0 ) return EventCode; - if (lowercaseCollectionName == "destinyscript") return DestinyScript; - else if (lowercaseCollectionName == "eventcode") return EventCode; else return EventCode; - }; // Function to retrieve a value from mapCollection ignoring case std::string get(const std::string& mapName, const std::string& key) { std::unordered_map& selectedMap = MapCollection(mapName); - std::string lowercaseKey = toLower(key); - - for (auto it = selectedMap.begin(); it != selectedMap.end(); ++it) { - std::string lowercaseMapKey = toLower(it->first); - if (lowercaseMapKey == lowercaseKey) return std::to_string(it->second); - } + for (auto it = selectedMap.begin(); it != selectedMap.end(); ++it) + if ( Utils::StrICmp(it->first, key) == 0 ) return std::to_string(it->second); - return "0"; // Key not found + return key; // Key not found } }; diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index 2f7b543b4e..57f24aad01 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -88,80 +88,59 @@ static bool EasyAdd(dyn_arg_list args) { return true; } - bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpreter) { - if (!interpreter) { - return true; - } - - auto func = "easyrpg_raw"; - bool okay = false; - - lcf::rpg::EventCommand cmd; - std::vector output_args; + if (!interpreter) return true; if (args.empty()) { Output::Warning("easyrpg_raw: Command too short"); return true; } - Constants Constants; - std::string keyToPrint = "ENABLE"; - - //Output::Warning("Key {}, value {}", keyToPrint, Constants.get("DestinyScript",keyToPrint)); - - auto evt = args[0]; - if (evt.find("@") == 0) { - evt = evt.substr(1); - evt = Constants.get("EventCode", evt); + Constants constList; - cmd.code = stoi(evt); - okay = bool(cmd.code); - } else - std::tie(cmd.code) = DynRpg::ParseArgs(func, args, &okay); - - if (!okay) { - Output::Warning("EasyRpgPlugin - Unknown Input: {}",args[0]); - return true; - } + const std::string func = "easyrpg_raw"; + bool okay = false; + int codeArgIndex = 0; + int stringArgIndex = 1; + bool endOfLine = false; - if (args.size() >= 2) { - auto [string_arg] = DynRpg::ParseArgs(func, args.subspan(1), &okay); - cmd.string = lcf::DBString(string_arg); + lcf::rpg::EventCommand cmd; + std::vector outputArgs; + std::vector cmdList; - if (!okay) { - return true; + for (size_t i = 0; i < args.size(); ++i) { + if (i == args.size() - 1) { + if (args[i].back() == ';') args[i] = args[i].substr(0, args[i].length() - 1); + endOfLine = true; } - for (size_t i = 2; i < args.size(); ++i) { - - - auto currArg = args[i]; - - if (currArg.find("@") == 0) { - currArg = currArg.substr(1); - currArg = Constants.get("DestinyScript", currArg); - - auto int_arg = stoi(currArg); - okay = true; - output_args.push_back(int_arg); - } - else { - auto [int_arg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); - output_args.push_back(int_arg); - } + // TODO: Implement multi-line command interpretation split by ';'. - - if (!okay) { - return true; - } + if (i == codeArgIndex) { + if (args[i].front() == '@') args[i] = constList.get("EventCode", args[i].substr(1)); + std::tie(cmd.code) = DynRpg::ParseArgs(func, args, &okay); + } + else if (i == stringArgIndex) { + auto [stringArg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); + cmd.string = lcf::DBString(stringArg); + } + else { + if (args[i].front() == '@') args[i] = constList.get("DestinyScript", args[i].substr(1)); + auto [intArg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); + outputArgs.push_back(intArg); } - } - cmd.parameters = lcf::DBArray(output_args.begin(), output_args.end()); + if (endOfLine) { + codeArgIndex = i + 1; + stringArgIndex = i + 2; + cmd.parameters = lcf::DBArray(outputArgs.begin(), outputArgs.end()); + cmdList.push_back(cmd); + } - interpreter->Push({ cmd }, 0, false); + if (!okay) return true; + } + interpreter->Push(cmdList, 0, false); return true; } From 922bf77145e2ea13caadca6f677057e3a7048a16 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Fri, 10 Nov 2023 09:18:03 -0300 Subject: [PATCH 10/14] Multiline interpreter + major refactor Managed to make it work, but the code is a bit messy and its syntax is not very flexible; --- src/dynrpg_easyrpg.cpp | 83 ++++++++++++++++++++++++++++++++---------- src/game_dynrpg.cpp | 39 +++++++++++++++++++- 2 files changed, 102 insertions(+), 20 deletions(-) diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index 57f24aad01..382bf59168 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -17,6 +17,7 @@ // Headers #include +#include #include "dynrpg_easyrpg.h" #include "string_view.h" @@ -99,42 +100,86 @@ bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpr Constants constList; const std::string func = "easyrpg_raw"; - bool okay = false; + int codeArgIndex = 0; int stringArgIndex = 1; + int indentArgIndex = -1; + bool endOfLine = false; + bool okay = false; lcf::rpg::EventCommand cmd; + lcf::rpg::EventCommand _cmd; std::vector outputArgs; std::vector cmdList; for (size_t i = 0; i < args.size(); ++i) { - if (i == args.size() - 1) { - if (args[i].back() == ';') args[i] = args[i].substr(0, args[i].length() - 1); - endOfLine = true; + + bool valid = !args[i].empty(); + + if (valid && args[i].front() == '[') args[i] = args[i].substr(1); + if (valid && args[i].back() == ']') { + args[i] = args[i].substr(0, args[i].length() - 1); + indentArgIndex = i + 1; } - // TODO: Implement multi-line command interpretation split by ';'. + valid = !args[i].empty(); - if (i == codeArgIndex) { - if (args[i].front() == '@') args[i] = constList.get("EventCode", args[i].substr(1)); - std::tie(cmd.code) = DynRpg::ParseArgs(func, args, &okay); - } - else if (i == stringArgIndex) { - auto [stringArg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); - cmd.string = lcf::DBString(stringArg); - } - else { - if (args[i].front() == '@') args[i] = constList.get("DestinyScript", args[i].substr(1)); - auto [intArg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); - outputArgs.push_back(intArg); + if (i == args.size() - 1) endOfLine = true; + + + else if (args[i] == ";") { + endOfLine = !args[i + 1].empty(); + valid = 0; } + if (valid) { + if (i == codeArgIndex) + { + size_t start_pos = args[i].find_first_not_of(" \t\r\n"); + if (start_pos != std::string::npos) + args[i] = args[i].substr(start_pos); + + // Output::Debug("code ----> {}", args[i]); + + if (args[i].front() == '$') + args[i] = constList.get("EventCode", args[i].substr(1)); + std::tie(cmd.code) = DynRpg::ParseArgs(func, args.subspan(i), &okay); + } + else if (i == stringArgIndex) + { + // Output::Debug("str ----> {}", args[i]); + auto [stringArg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); + cmd.string = lcf::DBString(stringArg); + } + else + { + if (args[i].front() == '$') + args[i] = constList.get("DestinyScript", args[i].substr(1)); + auto [intArg] = DynRpg::ParseArgs(func, args.subspan(i), &okay); + + if (indentArgIndex == i) { + // Output::Debug("idt ----> {}", args[i]); + indentArgIndex = -1; + cmd.indent = intArg; + } + else + // Output::Debug("pls ----> {}", args[i]), + outputArgs.push_back(intArg); + } + } if (endOfLine) { - codeArgIndex = i + 1; - stringArgIndex = i + 2; + // Output::Debug("com ----> {}\n\n", cmd.code); cmd.parameters = lcf::DBArray(outputArgs.begin(), outputArgs.end()); cmdList.push_back(cmd); + outputArgs.clear(); + + cmd = _cmd; + + codeArgIndex = i + 1; + stringArgIndex = i + 2; + + endOfLine = false; } if (!okay) return true; diff --git a/src/game_dynrpg.cpp b/src/game_dynrpg.cpp index 2b9e65e8d7..68ba7087a0 100644 --- a/src/game_dynrpg.cpp +++ b/src/game_dynrpg.cpp @@ -314,7 +314,44 @@ std::string DynRpg::ParseCommand(std::string command, std::vector& token.str(""); break; } - } else { + } + else if (chr == ';') { + switch (mode) { + case ParseMode_Function: + // End of function token + function_name = Utils::LowerCase(token.str()); + if (function_name.empty()) { + // empty function name + Output::Warning("Empty DynRPG function name"); + return {}; + } + token.str(""); + // Empty arg + args.emplace_back(""); + mode = ParseMode_WaitForArg; + break; + case ParseMode_WaitForComma: + mode = ParseMode_WaitForArg; + break; + case ParseMode_WaitForArg: + + // Empty arg + args.emplace_back(""); + break; + case ParseMode_String: + token << chr; + break; + case ParseMode_Token: + tmp = ParseToken(token.str(), function_name); + args.emplace_back(tmp); + if (function_name == "easyrpg_raw") args.emplace_back(";"); + // already on a comma + mode = ParseMode_WaitForArg; + token.str(""); + break; + } + } + else { // Anything else that isn't special purpose switch (mode) { case ParseMode_Function: From 715f171614840d9441a6f33f94e25c87ce2192d1 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Tue, 21 Nov 2023 00:06:34 -0300 Subject: [PATCH 11/14] easyrpg_raw: rebase and adapt old code + preparations for future features --- src/constants.h | 13 +++++++++---- src/game_dynrpg.cpp | 3 ++- 2 files changed, 11 insertions(+), 5 deletions(-) diff --git a/src/constants.h b/src/constants.h index c5c96bd27d..1bd10a5efd 100644 --- a/src/constants.h +++ b/src/constants.h @@ -1345,12 +1345,17 @@ class Constants { }; // Function to retrieve a value from mapCollection ignoring case - std::string get(const std::string& mapName, const std::string& key) { + std::string get(const std::string& mapName, const std::string& key, bool convertMode = 0 ) { std::unordered_map& selectedMap = MapCollection(mapName); - for (auto it = selectedMap.begin(); it != selectedMap.end(); ++it) - if ( Utils::StrICmp(it->first, key) == 0 ) return std::to_string(it->second); + if (convertMode == 0) // Name 2 ID + for (auto it = selectedMap.begin(); it != selectedMap.end(); ++it) + if ( Utils::StrICmp(it->first, key) == 0 ) return std::to_string(it->second); - return key; // Key not found + if (convertMode == 1) // ID 2 Name + for (auto it = selectedMap.begin(); it != selectedMap.end(); ++it) + if (Utils::StrICmp(std::to_string(it->second), key) == 0) return it->first; + + return key; // Match not found } }; diff --git a/src/game_dynrpg.cpp b/src/game_dynrpg.cpp index 68ba7087a0..c9686cc7ab 100644 --- a/src/game_dynrpg.cpp +++ b/src/game_dynrpg.cpp @@ -139,7 +139,8 @@ static std::string ParseToken(std::string token, StringView function_name) { return ToString(Main_Data::game_actors->GetActor(number)->GetName()); } else if (*it == 'T' && Player::IsPatchManiac()) { // T (String Var) is last - return Main_Data::game_strings->Get(number); + std::string output(Main_Data::game_strings->Get(number)); + return output; } else { // Variable number = Main_Data::game_variables->Get(number); From 90a98b6775d702bed223164f35d781a564370fff Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Fri, 22 Dec 2023 16:51:16 -0300 Subject: [PATCH 12/14] New Command: 2099 - StoreCommands Store commands inside a Stringvariable. //$storeCommands "preffix",[evtType_isVar, evtType, evtId_isVar, evtId, evtPage_isVar, evtPage, targetStrVar_isVar, targetStrVar] // 2099, "@easyrpg_raw", [1,4, 1,5, 1,6, 1,7], 0; --- src/dynrpg_easyrpg.h | 1 - src/game_interpreter.cpp | 72 ++++++++++++++++++++++++++++++++++++++++ src/game_interpreter.h | 3 ++ 3 files changed, 75 insertions(+), 1 deletion(-) diff --git a/src/dynrpg_easyrpg.h b/src/dynrpg_easyrpg.h index 4499df9895..0f9e41e446 100644 --- a/src/dynrpg_easyrpg.h +++ b/src/dynrpg_easyrpg.h @@ -21,7 +21,6 @@ #include "game_dynrpg.h" #include "game_battle.h" #include "game_map.h" -#include "constants.h" namespace DynRpg { /** diff --git a/src/game_interpreter.cpp b/src/game_interpreter.cpp index ab1d780cde..315eb35b9c 100644 --- a/src/game_interpreter.cpp +++ b/src/game_interpreter.cpp @@ -825,6 +825,8 @@ bool Game_Interpreter::ExecuteCommand(lcf::rpg::EventCommand const& com) { return CommandManiacControlStrings(com); case Cmd::Maniac_CallCommand: return CommandManiacCallCommand(com); + case static_cast (2099): //easyrpg_storeCommands + return CommandStoreCommands(com); default: return true; } @@ -5061,6 +5063,76 @@ bool Game_Interpreter::CommandManiacCallCommand(lcf::rpg::EventCommand const& co return true; } +bool Game_Interpreter::CommandStoreCommands(lcf::rpg::EventCommand const& com) { + //$storeCommands "preffix",[evtType_isVar, evtType, evtId_isVar, evtId, evtPage_isVar, evtPage, targetStrVar_isVar, targetStrVar] + // 2099, "@easyrpg_raw", [1,4, 1,5, 1,6, 1,7], 0; + + int evtType = ValueOrVariable(com.parameters[0], com.parameters[1]); + int evtId = ValueOrVariable(com.parameters[2], com.parameters[3]); + int evtPage = ValueOrVariable(com.parameters[4], com.parameters[5]); + int targetStrVar = ValueOrVariable(com.parameters[6], com.parameters[7]); // target string variable + + std::string textPreffix = com.string.empty() ? "" : ToString(com.string) + " "; + + Game_Event* event; + const lcf::rpg::EventPage* page; + Game_CommonEvent* common_event; + + if (evtType == 0) { // Map Event + event = static_cast(GetCharacter(evtId)); + if (!event || evtId == 10001) { + Output::Warning("StoreCommands: Can't read non-existent event {}", evtId); + return true; + } + page = event->GetPage(evtPage); + if (!page) { + Output::Warning("StoreCommands: Can't read non-existent page {} of event {}", evtPage, evtId); + return true; + } + } + else if (evtType == 1) { // Common Event + common_event = lcf::ReaderUtil::GetElement(Game_Map::GetCommonEvents(), evtId); + if (!common_event) { + Output::Warning("StoreCommands: Can't read invalid common event {}", evtId); + return true; + } + } + + std::string rawCommand = " "; + Constants constList; + + const auto& list = evtType == 1 ? common_event->GetList() : page->event_commands; + int index = 0; + + for (size_t i = index; i < list.size(); ++i) { + std::stringstream ss; + + for (size_t j = 0; j < list[i].parameters.size(); ++j) { + ss << std::to_string(list[i].parameters[j]); + if (j < list[i].parameters.size() - 1) ss << ", "; + } + + std::string inputString = ToString(list[i].string); + for (size_t j = 0; j < inputString.length(); ++j) + if (inputString[j] == '"') { + inputString.insert(j, 1, '"'); + ++j; + } + + std::string preffix(list[i].indent, ' '); + std::string suffix = i < list.size() - 1 ? "\n" : ""; + + rawCommand += fmt::format("{}${}, \"{}\", [{}], {};{}", + preffix, constList.get("EventCode", std::to_string(list[i].code), 1), inputString, ss.str(), list[i].indent, suffix); + } + + Game_Strings::Str_Params str_params = { targetStrVar,0, 0 }; + Main_Data::game_strings->Asg(str_params, textPreffix + rawCommand); + //fmt::print(rawCommand); + + return true; +} + Game_Interpreter& Game_Interpreter::GetForegroundInterpreter() { return Game_Battle::IsBattleRunning() ? Game_Battle::GetInterpreter() diff --git a/src/game_interpreter.h b/src/game_interpreter.h index 8d0e44625a..6d0fdc5893 100644 --- a/src/game_interpreter.h +++ b/src/game_interpreter.h @@ -32,6 +32,7 @@ #include #include #include "async_op.h" +#include "constants.h" class Game_Event; class Game_CommonEvent; @@ -289,6 +290,8 @@ class Game_Interpreter bool CommandManiacControlStrings(lcf::rpg::EventCommand const& com); bool CommandManiacCallCommand(lcf::rpg::EventCommand const& com); + bool CommandStoreCommands(lcf::rpg::EventCommand const& com); + int DecodeInt(lcf::DBArray::const_iterator& it); const std::string DecodeString(lcf::DBArray::const_iterator& it); lcf::rpg::MoveCommand DecodeMove(lcf::DBArray::const_iterator& it); From a97f37f9a4081c3015132677fe874ba879d4bc54 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Fri, 22 Dec 2023 18:51:18 -0300 Subject: [PATCH 13/14] StoreCommands (2099) - Use kCodeTags.tag instead of custom tag reader --- src/constants.h | 2 +- src/game_interpreter.cpp | 9 +++++++-- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/src/constants.h b/src/constants.h index 1bd10a5efd..6cf1608386 100644 --- a/src/constants.h +++ b/src/constants.h @@ -17,7 +17,7 @@ #include "utils.h" #include -#include +#include #include class Constants { diff --git a/src/game_interpreter.cpp b/src/game_interpreter.cpp index 315eb35b9c..c3c4f42db2 100644 --- a/src/game_interpreter.cpp +++ b/src/game_interpreter.cpp @@ -5099,7 +5099,6 @@ bool Game_Interpreter::CommandStoreCommands(lcf::rpg::EventCommand const& com) { } std::string rawCommand = " "; - Constants constList; const auto& list = evtType == 1 ? common_event->GetList() : page->event_commands; int index = 0; @@ -5122,8 +5121,14 @@ bool Game_Interpreter::CommandStoreCommands(lcf::rpg::EventCommand const& com) { std::string preffix(list[i].indent, ' '); std::string suffix = i < list.size() - 1 ? "\n" : ""; + auto cmd_tag = lcf::rpg::EventCommand::kCodeTags.tag(list[i].code); + std::string command_id; + + if (!cmd_tag) command_id = std::to_string(list[i].code); + else command_id = cmd_tag; + rawCommand += fmt::format("{}${}, \"{}\", [{}], {};{}", - preffix, constList.get("EventCode", std::to_string(list[i].code), 1), inputString, ss.str(), list[i].indent, suffix); + preffix, command_id, inputString, ss.str(), list[i].indent, suffix); } Game_Strings::Str_Params str_params = { targetStrVar,0, 0 }; From 04ca1bae07fc608257509fbf8e632dd52808bfe5 Mon Sep 17 00:00:00 2001 From: Mauro Junior <45118493+jetrotal@users.noreply.github.com> Date: Wed, 3 Jan 2024 20:54:29 -0300 Subject: [PATCH 14/14] @easyrpg_raw - Further use kCodeTags instead of a custom tag reader now, our raw code also use this library. --- src/constants.h | 161 +---------------------------------------- src/dynrpg_easyrpg.cpp | 11 ++- 2 files changed, 11 insertions(+), 161 deletions(-) diff --git a/src/constants.h b/src/constants.h index 6cf1608386..a0d3a5a7d7 100644 --- a/src/constants.h +++ b/src/constants.h @@ -1178,170 +1178,13 @@ class Constants { {"CONDITIONLIMITATION_ATTACKALLY", 3} // LEGACY }; - EventCode = { - {"END", 10}, - {"CallCommonEvent", 1005}, - {"ForceFlee", 1006}, - {"EnableCombo", 1007}, - {"ChangeClass", 1008}, - {"ChangeBattleCommands", 1009}, - {"OpenLoadMenu", 5001}, - {"ExitGame", 5002}, - {"ToggleAtbMode", 5003}, - {"ToggleFullscreen", 5004}, - {"OpenVideoOptions", 5005}, - {"ShowMessage", 10110}, - {"MessageOptions", 10120}, - {"ChangeFaceGraphic", 10130}, - {"ShowChoice", 10140}, - {"InputNumber", 10150}, - {"ControlSwitches", 10210}, - {"ControlVars", 10220}, - {"TimerOperation", 10230}, - {"ChangeGold", 10310}, - {"ChangeItems", 10320}, - {"ChangePartyMembers", 10330}, - {"ChangeExp", 10410}, - {"ChangeLevel", 10420}, - {"ChangeParameters", 10430}, - {"ChangeSkills", 10440}, - {"ChangeEquipment", 10450}, - {"ChangeHP", 10460}, - {"ChangeSP", 10470}, - {"ChangeCondition", 10480}, - {"FullHeal", 10490}, - {"SimulatedAttack", 10500}, - {"ChangeHeroName", 10610}, - {"ChangeHeroTitle", 10620}, - {"ChangeSpriteAssociation", 10630}, - {"ChangeActorFace", 10640}, - {"ChangeVehicleGraphic", 10650}, - {"ChangeSystemBGM", 10660}, - {"ChangeSystemSFX", 10670}, - {"ChangeSystemGraphics", 10680}, - {"ChangeScreenTransitions", 10690}, - {"EnemyEncounter", 10710}, - {"OpenShop", 10720}, - {"ShowInn", 10730}, - {"EnterHeroName", 10740}, - {"Teleport", 10810}, - {"MemorizeLocation", 10820}, - {"RecallToLocation", 10830}, - {"EnterExitVehicle", 10840}, - {"SetVehicleLocation", 10850}, - {"ChangeEventLocation", 10860}, - {"TradeEventLocations", 10870}, - {"StoreTerrainID", 10910}, - {"StoreEventID", 10920}, - {"EraseScreen", 11010}, - {"ShowScreen", 11020}, - {"TintScreen", 11030}, - {"FlashScreen", 11040}, - {"ShakeScreen", 11050}, - {"PanScreen", 11060}, - {"WeatherEffects", 11070}, - {"ShowPicture", 11110}, - {"MovePicture", 11120}, - {"ErasePicture", 11130}, - {"ShowBattleAnimation", 11210}, - {"PlayerVisibility", 11310}, - {"FlashSprite", 11320}, - {"MoveEvent", 11330}, - {"ProceedWithMovement", 11340}, - {"HaltAllMovement", 11350}, - {"Wait", 11410}, - {"PlayBGM", 11510}, - {"FadeOutBGM", 11520}, - {"MemorizeBGM", 11530}, - {"PlayMemorizedBGM", 11540}, - {"PlaySound", 11550}, - {"PlayMovie", 11560}, - {"KeyInputProc", 11610}, - {"ChangeMapTileset", 11710}, - {"ChangePBG", 11720}, - {"ChangeEncounterSteps", 11740}, - {"TileSubstitution", 11750}, - {"TeleportTargets", 11810}, - {"ChangeTeleportAccess", 11820}, - {"EscapeTarget", 11830}, - {"ChangeEscapeAccess", 11840}, - {"OpenSaveMenu", 11910}, - {"ChangeSaveAccess", 11930}, - {"OpenMainMenu", 11950}, - {"ChangeMainMenuAccess", 11960}, - {"ConditionalBranch", 12010}, - {"Label", 12110}, - {"JumpToLabel", 12120}, - {"Loop", 12210}, - {"BreakLoop", 12220}, - {"EndEventProcessing", 12310}, - {"EraseEvent", 12320}, - {"CallEvent", 12330}, - {"Comment", 12410}, - {"GameOver", 12420}, - {"ReturntoTitleScreen", 12510}, - {"ChangeMonsterHP", 13110}, - {"ChangeMonsterMP", 13120}, - {"ChangeMonsterCondition", 13130}, - {"ShowHiddenMonster", 13150}, - {"ChangeBattleBG", 13210}, - {"ShowBattleAnimation_B", 13260}, - {"ConditionalBranch_B", 13310}, - {"TerminateBattle", 13410}, - {"ShowMessage_2", 20110}, - {"ShowChoiceOption", 20140}, - {"ShowChoiceEnd", 20141}, - {"VictoryHandler", 20710}, - {"EscapeHandler", 20711}, - {"DefeatHandler", 20712}, - {"EndBattle", 20713}, - {"Transaction", 20720}, - {"NoTransaction", 20721}, - {"EndShop", 20722}, - {"Stay", 20730}, - {"NoStay", 20731}, - {"EndInn", 20732}, - {"ElseBranch", 22010}, - {"EndBranch", 22011}, - {"EndLoop", 22210}, - {"Comment_2", 22410}, - {"ElseBranch_B", 23310}, - {"EndBranch_B", 23311}, - {"Maniac_GetSaveInfo", 3001}, - {"Maniac_Save", 3002}, - {"Maniac_Load", 3003}, - {"Maniac_EndLoadProcess", 3004}, - {"Maniac_GetMousePosition", 3005}, - {"Maniac_SetMousePosition", 3006}, - {"Maniac_ShowStringPicture", 3007}, - {"Maniac_GetPictureInfo", 3008}, - {"Maniac_ControlBattle", 3009}, - {"Maniac_ControlAtbGauge", 3010}, - {"Maniac_ChangeBattleCommandEx", 3011}, - {"Maniac_GetBattleInfo", 3012}, - {"Maniac_ControlVarArray", 3013}, - {"Maniac_KeyInputProcEx", 3014}, - {"Maniac_RewriteMap", 3015}, - {"Maniac_ControlGlobalSave", 3016}, - {"Maniac_ChangePictureId", 3017}, - {"Maniac_SetGameOption", 3018}, - {"Maniac_CallCommand", 3019}, - {"Maniac_ControlStrings", 3020}, - {"Maniac_GetGameInfo", 3021}, - {"Maniac_EditPicture", 3025}, - {"Maniac_WritePicture", 3026}, - {"Maniac_AddMoveRoute", 3027}, - {"Maniac_EditTile", 3028}, - {"Maniac_ControlTextProcessing", 3029} - }; }; - // Function to return contents from either DestinyScript or EventCode + // Function to return contents from either DestinyScript or something else std::unordered_map& MapCollection(const std::string& collectionName) { if ( Utils::StrICmp(collectionName, "destinyscript") == 0 ) return DestinyScript; - else if ( Utils::StrICmp(collectionName, "eventcode") == 0 ) return EventCode; - else return EventCode; + else return DestinyScript; }; // Function to retrieve a value from mapCollection ignoring case diff --git a/src/dynrpg_easyrpg.cpp b/src/dynrpg_easyrpg.cpp index 382bf59168..a577ba200d 100644 --- a/src/dynrpg_easyrpg.cpp +++ b/src/dynrpg_easyrpg.cpp @@ -142,8 +142,15 @@ bool DynRpg::EasyRpgPlugin::EasyRaw(dyn_arg_list args, Game_Interpreter* interpr // Output::Debug("code ----> {}", args[i]); - if (args[i].front() == '$') - args[i] = constList.get("EventCode", args[i].substr(1)); + if (args[i].front() == '$') { + args[i] = args[i].substr(1); + for (const auto& pair : lcf::rpg::EventCommand::kCodeTags.tags()) { + if (Utils::StrICmp(pair.name, args[i]) == 0) { + args[i] = std::to_string(pair.value); + break; + } + } + }; std::tie(cmd.code) = DynRpg::ParseArgs(func, args.subspan(i), &okay); } else if (i == stringArgIndex)