From da273cbee620610e36615511257c3da18160e3ae Mon Sep 17 00:00:00 2001 From: "Zed A. Shaw" Date: Mon, 24 Mar 2025 12:30:58 -0400 Subject: [PATCH] Refactored rituals so they can be used in different situations. --- Makefile | 4 +-- ai.hpp | 5 ++-- meson.build | 1 + rituals.cpp | 70 +++++++++++++++++++++++++++++++++++++---------- rituals.hpp | 22 +++++++++++++-- tests/ai.cpp | 21 -------------- tests/combat.cpp | 9 ++++++ tests/rituals.cpp | 55 ++++++++++++++++++++++--------------- 8 files changed, 122 insertions(+), 65 deletions(-) create mode 100644 tests/combat.cpp diff --git a/Makefile b/Makefile index 562ac6d..da5e861 100644 --- a/Makefile +++ b/Makefile @@ -22,7 +22,7 @@ tracy_build: meson compile -j 10 -C builddir test: build - ./builddir/runtests + ./builddir/runtests "[combat]" run: build test powershell "cp ./builddir/zedcaster.exe ." @@ -41,7 +41,7 @@ clean: meson compile --clean -C builddir debug_test: build - gdb --nx -x .gdbinit --ex run --args builddir/runtests.exe -e + gdb --nx -x .gdbinit --ex run --args builddir/runtests.exe -e "[combat]" win_installer: powershell 'start "C:\Program Files (x86)\solicus\InstallForge\bin\ifbuilderenvx86.exe" win_installer.ifp' diff --git a/ai.hpp b/ai.hpp index 76a1777..23a5e1e 100644 --- a/ai.hpp +++ b/ai.hpp @@ -33,7 +33,6 @@ namespace ai { struct AIManager { AIProfile profile; - std::unordered_map actions; std::unordered_map states; std::unordered_map> scripts; @@ -43,8 +42,8 @@ namespace ai { void reset(); void init(std::string config_path); - Action config_action(nlohmann::json& profile, nlohmann::json& config); - State config_state(nlohmann::json& profile, nlohmann::json& config); + Action config_action(AIProfile& profile, nlohmann::json& config); + State config_state(AIProfile& profile, nlohmann::json& config); int state_id(std::string name); State load_state(std::string state_name); diff --git a/meson.build b/meson.build index 945449b..debab18 100644 --- a/meson.build +++ b/meson.build @@ -141,6 +141,7 @@ executable('runtests', sources + [ 'tests/matrix.cpp', 'tests/pathing.cpp', 'tests/rituals.cpp', + 'tests/combat.cpp', 'tests/sound.cpp', 'tests/spatialmap.cpp', 'tests/animation.cpp', diff --git a/rituals.cpp b/rituals.cpp index 98d8de9..48f7596 100644 --- a/rituals.cpp +++ b/rituals.cpp @@ -1,33 +1,75 @@ #include "rituals.hpp" #include "ai_debug.hpp" +#include "ai.hpp" namespace combat { - void RitualAI::reset() { - start = original; + + RitualEngine::RitualEngine(std::string config_path) : + $config(config_path) + { + $profile = $config["profile"]; + + auto& actions = $config["actions"]; + + for(auto& ac : actions) { + auto action = ai::config_action($profile, ac); + $actions.insert_or_assign(action.name, action); + } + + for(auto& [name, sc] : $config["states"].items()) { + auto state = ai::config_state($profile, sc); + $states.insert_or_assign(name, state); + } + + auto& scripts = $config["scripts"]; + for(auto& [script_name, action_names] : scripts.items()) { + std::vector the_script; + for(auto name : action_names) { + the_script.push_back($actions.at(name)); + } + + $scripts.insert_or_assign(script_name, the_script); + } } - bool RitualAI::will_do(std::string name) { - ai::check_valid_action(name, "RitualAI::is_able_to"); - return plan.script[0].name == name; + ai::State RitualEngine::load_state(std::string name) { + return $states.at(name); + } + + ai::Action RitualEngine::load_action(std::string name) { + return $actions.at(name); + } + + RitualAI RitualEngine::start() { + auto start = load_state("initial"); + auto goal = load_state("final"); + return {"actions", start, goal}; } - void RitualAI::set_state(std::string name, bool setting) { - ai::set(start, name, setting); + void RitualEngine::set_state(RitualAI& ritual, std::string name, bool setting) { + ritual.start.set($profile.at(name), setting); + } + + void RitualEngine::reset(RitualAI& ritual) { + ritual.start = ritual.original; + } + + void RitualEngine::plan(RitualAI& ritual) { + ritual.plan = ai::plan_actions($scripts.at(ritual.script), ritual.start, ritual.goal); + } + + bool RitualAI::will_do(std::string name) { + if(plan.script.size() == 0) return false; + + return plan.script[0].name == name; } - /* - * BUG: I don't like this, maybe an iterator is better? - */ ai::Action RitualAI::pop() { auto result = plan.script.front(); plan.script.pop_front(); return result; } - void RitualAI::update() { - plan = ai::plan(script, start, goal); - } - void RitualAI::dump() { ai::dump_script(script, start, plan.script); } diff --git a/rituals.hpp b/rituals.hpp index 0188829..a6f4b73 100644 --- a/rituals.hpp +++ b/rituals.hpp @@ -1,5 +1,7 @@ #pragma once +#include "goap.hpp" #include "ai.hpp" +#include "config.hpp" namespace combat { struct RitualAI { @@ -16,11 +18,25 @@ namespace combat { RitualAI() {}; - void reset(); bool will_do(std::string name); - void set_state(std::string name, bool setting); - void update(); void dump(); ai::Action pop(); }; + + struct RitualEngine { + Config $config; + ai::AIProfile $profile; + std::unordered_map $actions; + std::unordered_map $states; + std::unordered_map> $scripts; + + RitualEngine(std::string config_path); + + ai::State load_state(std::string name); + ai::Action load_action(std::string name); + RitualAI start(); + void reset(RitualAI& ritual); + void set_state(RitualAI& ritual, std::string name, bool setting); + void plan(RitualAI& ritual); + }; } diff --git a/tests/ai.cpp b/tests/ai.cpp index 52ab15e..5ad2367 100644 --- a/tests/ai.cpp +++ b/tests/ai.cpp @@ -209,24 +209,3 @@ TEST_CASE("Confirm EntityAI behaves as expected", "[ai]") { enemy.update(); REQUIRE(enemy.wants_to("run_away")); } - -TEST_CASE("confirm that cycles are avoided/detected", "[ai]") { - ai::reset(); - ai::init("tests/cyclic_rituals.json"); - - auto start = ai::load_state("initial"); - auto goal = ai::load_state("final"); - - combat::RitualAI ritual("actions", start, goal); - ritual.reset(); - ritual.set_state("has_magick", true); - ritual.set_state("cursed_item", true); - ritual.set_state("shiny_bauble", true); - - bool it_throws = false; - try { ritual.update(); } catch(...) { it_throws = true; } - REQUIRE(it_throws); - - fmt::println("\n\n------------ CYCLES AVOIDED"); - ritual.dump(); -} diff --git a/tests/combat.cpp b/tests/combat.cpp new file mode 100644 index 0000000..4fcb887 --- /dev/null +++ b/tests/combat.cpp @@ -0,0 +1,9 @@ +#include +#include +#include "rituals.hpp" + +using namespace combat; + +TEST_CASE("turn based combat engine sorted", "[combat]") { + dbc::log("does nothing."); +} diff --git a/tests/rituals.cpp b/tests/rituals.cpp index a8f2eb7..a2172e2 100644 --- a/tests/rituals.cpp +++ b/tests/rituals.cpp @@ -5,24 +5,24 @@ using namespace combat; TEST_CASE("prototype combat system ideas", "[combat]") { - ai::reset(); - ai::init("assets/rituals.json"); + RitualEngine re("assets/rituals.json"); + auto ritual = re.start(); - auto start = ai::load_state("initial"); - auto goal = ai::load_state("final"); + re.set_state(ritual, "has_spikes", true); + re.plan(ritual); - RitualAI ritual("actions", start, goal); - - ritual.set_state("has_spikes", true); - ritual.update(); fmt::println("\n\n------------ TEST WILL DO PIERCE"); ritual.dump(); REQUIRE(ritual.will_do("pierce_type")); - ritual.reset(); - ritual.set_state("has_magick", true); - ritual.set_state("has_spikes", true); - ritual.update(); + REQUIRE(ritual.start != ritual.original); + re.reset(ritual); + REQUIRE(ritual.start == ritual.original); + + re.set_state(ritual, "has_magick", true); + re.set_state(ritual, "has_spikes", true); + re.plan(ritual); + fmt::println("\n\n------------ TEST WILL DO MAGICK TOO"); ritual.dump(); REQUIRE(ritual.will_do("magick_type")); @@ -30,20 +30,31 @@ TEST_CASE("prototype combat system ideas", "[combat]") { ritual.pop(); REQUIRE(ritual.will_do("pierce_type")); - ritual.reset(); - ritual.set_state("has_magick", true); - ritual.set_state("has_spikes", true); - ritual.set_state("shiny_bauble", true); - ritual.update(); + re.reset(ritual); + re.set_state(ritual, "has_magick", true); + re.set_state(ritual, "has_spikes", true); + re.set_state(ritual, "shiny_bauble", true); + re.plan(ritual); fmt::println("\n\n------------ TEST WILL DO DAMAGE BOOST"); ritual.dump(); - ritual.reset(); - ritual.set_state("has_magick", true); - ritual.set_state("cursed_item", true); - ritual.set_state("shiny_bauble", true); - ritual.update(); + re.reset(ritual); + re.set_state(ritual, "has_magick", true); + re.set_state(ritual, "cursed_item", true); + re.set_state(ritual, "shiny_bauble", true); + re.plan(ritual); fmt::println("\n\n------------ TEST WILL DO LARGE DAMAGE BOOST"); ritual.dump(); +} +TEST_CASE("confirm that cycles are avoided/detected", "[combat]") { + RitualEngine re("assets/rituals.json"); + auto ritual = re.start(); + + re.set_state(ritual, "has_magick", true); + re.set_state(ritual, "cursed_item", true); + re.set_state(ritual, "shiny_bauble", true); + + fmt::println("\n\n------------ CYCLES AVOIDED"); + ritual.dump(); }