Basic AP (Action Points) system tied to the AI actions, but there's no way to set 'has AP' for the AI?
This commit is contained in:
parent
c78b2ae75e
commit
a38bb5b691
8 changed files with 64 additions and 18 deletions
|
|
@ -62,7 +62,7 @@
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 200, "max_hp": 200, "damage": 20, "dead": false},
|
{"_type": "Combat", "hp": 200, "max_hp": 200, "ap": 0, "ap_delta": 9, "max_ap": 20, "damage": 20, "dead": false},
|
||||||
{"_type": "Sound", "attack": "Marmot_Scream_1", "death": "Creature_Death_1"},
|
{"_type": "Sound", "attack": "Marmot_Scream_1", "death": "Creature_Death_1"},
|
||||||
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"}
|
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"}
|
||||||
]
|
]
|
||||||
|
|
|
||||||
|
|
@ -6,7 +6,7 @@
|
||||||
"foreground": "enemies/fg:player",
|
"foreground": "enemies/fg:player",
|
||||||
"background": "color:transparent"
|
"background": "color:transparent"
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 200, "max_hp": 200, "damage": 10, "dead": false},
|
{"_type": "Combat", "hp": 200, "max_hp": 200, "ap": 0, "max_ap": 12, "ap_delta": 6, "damage": 10, "dead": false},
|
||||||
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
||||||
{"_type": "Collision", "has": true},
|
{"_type": "Collision", "has": true},
|
||||||
{"_type": "LightSource", "strength": 35, "radius": 2.0}
|
{"_type": "LightSource", "strength": 35, "radius": 2.0}
|
||||||
|
|
@ -18,7 +18,7 @@
|
||||||
"foreground": "enemies/fg:gold_savior",
|
"foreground": "enemies/fg:gold_savior",
|
||||||
"background": "color:transparent"
|
"background": "color:transparent"
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 1, "dead": false},
|
{"_type": "Combat", "hp": 20, "max_hp": 20, "ap": 0, "max_ap": 12, "ap_delta": 6, "damage": 1, "dead": false},
|
||||||
{"_type": "Collision", "has": true},
|
{"_type": "Collision", "has": true},
|
||||||
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
||||||
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
||||||
|
|
@ -33,7 +33,7 @@
|
||||||
"foreground": "enemies/fg:knight",
|
"foreground": "enemies/fg:knight",
|
||||||
"background": "color:transparent"
|
"background": "color:transparent"
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 1, "dead": false},
|
{"_type": "Combat", "hp": 20, "max_hp": 20, "ap": 0, "max_ap": 12, "ap_delta": 6,"damage": 1, "dead": false},
|
||||||
{"_type": "Collision", "has": true},
|
{"_type": "Collision", "has": true},
|
||||||
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
||||||
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
||||||
|
|
@ -48,7 +48,7 @@
|
||||||
"foreground": "enemies/fg:axe_ranger",
|
"foreground": "enemies/fg:axe_ranger",
|
||||||
"background": "color:transparent"
|
"background": "color:transparent"
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 40, "max_hp": 40, "damage": 10, "dead": false},
|
{"_type": "Combat", "hp": 40, "max_hp": 40, "ap": 0, "max_ap": 12, "ap_delta": 6,"damage": 10, "dead": false},
|
||||||
{"_type": "Collision", "has": true},
|
{"_type": "Collision", "has": true},
|
||||||
{"_type": "Motion", "dx": 0, "dy": 0, "random": true},
|
{"_type": "Motion", "dx": 0, "dy": 0, "random": true},
|
||||||
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
||||||
|
|
@ -63,7 +63,7 @@
|
||||||
"foreground": "enemies/fg:rat_giant",
|
"foreground": "enemies/fg:rat_giant",
|
||||||
"background": "color:transparent"
|
"background": "color:transparent"
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 50, "max_hp": 50, "damage": 2, "dead": false},
|
{"_type": "Combat", "hp": 50, "max_hp": 50, "ap": 0, "max_ap": 12, "ap_delta": 6,"damage": 2, "dead": false},
|
||||||
{"_type": "Collision", "has": true},
|
{"_type": "Collision", "has": true},
|
||||||
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
||||||
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
||||||
|
|
@ -78,7 +78,7 @@
|
||||||
"foreground": "enemies/fg:spider_giant",
|
"foreground": "enemies/fg:spider_giant",
|
||||||
"background": "color:transparent"
|
"background": "color:transparent"
|
||||||
},
|
},
|
||||||
{"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 20, "dead": false},
|
{"_type": "Combat", "hp": 20, "max_hp": 20, "ap": 0, "max_ap": 12, "ap_delta": 6,"damage": 20, "dead": false},
|
||||||
{"_type": "Collision", "has": true},
|
{"_type": "Collision", "has": true},
|
||||||
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
{"_type": "Motion", "dx": 0, "dy": 0, "random": false},
|
||||||
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
{"_type": "EnemyConfig", "ai_script": "Enemy::actions", "ai_start_name": "Enemy::initial_state", "ai_goal_name": "Enemy::final_state"},
|
||||||
|
|
|
||||||
34
battle.cpp
34
battle.cpp
|
|
@ -16,7 +16,16 @@ namespace combat {
|
||||||
|
|
||||||
int active = 0;
|
int active = 0;
|
||||||
|
|
||||||
|
fmt::println("---------- start combatants");
|
||||||
for(auto& [entity, enemy] : $combatants) {
|
for(auto& [entity, enemy] : $combatants) {
|
||||||
|
if(enemy.combat->ap < enemy.combat->max_ap) {
|
||||||
|
// only add up to the max
|
||||||
|
enemy.combat->ap = std::min(enemy.combat->max_ap, enemy.combat->ap_delta + enemy.combat->ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt::println("--- enemy {} has {} ap", entity, enemy.combat->ap);
|
||||||
|
// reset action points
|
||||||
|
|
||||||
enemy.ai->update();
|
enemy.ai->update();
|
||||||
active += enemy.ai->active();
|
active += enemy.ai->active();
|
||||||
|
|
||||||
|
|
@ -24,15 +33,38 @@ namespace combat {
|
||||||
for(auto& action : enemy.ai->plan.script) {
|
for(auto& action : enemy.ai->plan.script) {
|
||||||
BattleHostState host_state = not_host;
|
BattleHostState host_state = not_host;
|
||||||
|
|
||||||
if(enemy.is_host) {
|
if(action.cost > enemy.combat->ap) {
|
||||||
|
host_state = out_of_ap;
|
||||||
|
} else if(enemy.is_host) {
|
||||||
host_state = $player_requests.contains(action.name) ? agree : disagree;
|
host_state = $player_requests.contains(action.name) ? agree : disagree;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(host_state == out_of_ap) {
|
||||||
|
fmt::println("--- enemy CANNOT go: {}-{}={}",
|
||||||
|
enemy.combat->ap, action.cost,
|
||||||
|
enemy.combat->ap - action.cost);
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
fmt::println("--- enemy can go, {}-{}={}",
|
||||||
|
enemy.combat->ap, action.cost,
|
||||||
|
enemy.combat->ap - action.cost);
|
||||||
|
|
||||||
|
enemy.combat->ap -= action.cost;
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt::println("--- active enemy {} ap={}, host_state={}",
|
||||||
|
entity, enemy.combat->ap, int(host_state));
|
||||||
|
|
||||||
$pending_actions.emplace_back(enemy, action.name, action.cost, host_state);
|
$pending_actions.emplace_back(enemy, action.name, action.cost, host_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dbc::check(enemy.combat->ap >= 0, "enemy's AP went below 0");
|
||||||
|
dbc::check(enemy.combat->ap <= enemy.combat->max_ap, "enemy's AP went above max");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fmt::print("<---- end of enemy setup, sorting");
|
||||||
|
|
||||||
if($pending_actions.size() > 0) {
|
if($pending_actions.size() > 0) {
|
||||||
std::sort($pending_actions.begin(), $pending_actions.end(),
|
std::sort($pending_actions.begin(), $pending_actions.end(),
|
||||||
[](const auto& a, const auto& b) -> bool
|
[](const auto& a, const auto& b) -> bool
|
||||||
|
|
|
||||||
|
|
@ -11,7 +11,8 @@ namespace combat {
|
||||||
enum class BattleHostState {
|
enum class BattleHostState {
|
||||||
not_host = 0,
|
not_host = 0,
|
||||||
agree = 1,
|
agree = 1,
|
||||||
disagree = 2
|
disagree = 2,
|
||||||
|
out_of_ap = 3
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Combatant {
|
struct Combatant {
|
||||||
|
|
|
||||||
|
|
@ -89,6 +89,10 @@ namespace boss {
|
||||||
if(wants_to == "kill_enemy") {
|
if(wants_to == "kill_enemy") {
|
||||||
result.enemy_did = enemy.combat->attack(player_combat);
|
result.enemy_did = enemy.combat->attack(player_combat);
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
|
case combat::BattleHostState::out_of_ap:
|
||||||
|
fmt::println("OUT OF AP {}", wants_to);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(result.player_did > 0) {
|
if(result.player_did > 0) {
|
||||||
|
|
|
||||||
|
|
@ -102,6 +102,11 @@ namespace components {
|
||||||
int hp;
|
int hp;
|
||||||
int max_hp;
|
int max_hp;
|
||||||
int damage;
|
int damage;
|
||||||
|
int ap_delta;
|
||||||
|
int max_ap;
|
||||||
|
|
||||||
|
// everyone starts at 0 but ap_delta is added each round
|
||||||
|
int ap = 0;
|
||||||
|
|
||||||
/* NOTE: This is used to _mark_ entities as dead, to detect ones that have just died. Don't make attack automatically set it.*/
|
/* NOTE: This is used to _mark_ entities as dead, to detect ones that have just died. Don't make attack automatically set it.*/
|
||||||
bool dead = false;
|
bool dead = false;
|
||||||
|
|
@ -176,7 +181,7 @@ namespace components {
|
||||||
ENROLL_COMPONENT(EnemyConfig, ai_script, ai_start_name, ai_goal_name);
|
ENROLL_COMPONENT(EnemyConfig, ai_script, ai_start_name, ai_goal_name);
|
||||||
ENROLL_COMPONENT(Personality, hearing_distance, tough);
|
ENROLL_COMPONENT(Personality, hearing_distance, tough);
|
||||||
ENROLL_COMPONENT(Motion, dx, dy, random);
|
ENROLL_COMPONENT(Motion, dx, dy, random);
|
||||||
ENROLL_COMPONENT(Combat, hp, max_hp, damage, dead);
|
ENROLL_COMPONENT(Combat, hp, max_hp, damage, ap_delta, max_ap, dead);
|
||||||
ENROLL_COMPONENT(Device, config, events);
|
ENROLL_COMPONENT(Device, config, events);
|
||||||
ENROLL_COMPONENT(Storyboard, image, audio, layout, beats);
|
ENROLL_COMPONENT(Storyboard, image, audio, layout, beats);
|
||||||
ENROLL_COMPONENT(Animation, min_x, min_y,
|
ENROLL_COMPONENT(Animation, min_x, min_y,
|
||||||
|
|
|
||||||
|
|
@ -250,6 +250,7 @@ void System::combat(int attack_id) {
|
||||||
|
|
||||||
battle.set_all("enemy_found", true);
|
battle.set_all("enemy_found", true);
|
||||||
battle.set_all("in_combat", true);
|
battle.set_all("in_combat", true);
|
||||||
|
battle.player_request("kill_enemy");
|
||||||
battle.plan();
|
battle.plan();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -20,17 +20,17 @@ TEST_CASE("battle operations fantasy", "[combat-battle]") {
|
||||||
|
|
||||||
DinkyECS::Entity host = 0;
|
DinkyECS::Entity host = 0;
|
||||||
ai::EntityAI host_ai("Host::actions", host_start, host_goal);
|
ai::EntityAI host_ai("Host::actions", host_start, host_goal);
|
||||||
components::Combat host_combat{100, 100, 20};
|
components::Combat host_combat{100, 100, 20, 6, 12};
|
||||||
battle.add_enemy({host, &host_ai, &host_combat, true});
|
battle.add_enemy({host, &host_ai, &host_combat, true});
|
||||||
|
|
||||||
DinkyECS::Entity axe_ranger = 1;
|
DinkyECS::Entity axe_ranger = 1;
|
||||||
ai::EntityAI axe_ai("Enemy::actions", ai_start, ai_goal);
|
ai::EntityAI axe_ai("Enemy::actions", ai_start, ai_goal);
|
||||||
components::Combat axe_combat{100, 100, 20};
|
components::Combat axe_combat{100, 100, 20, 8, 12};
|
||||||
battle.add_enemy({axe_ranger, &axe_ai, &axe_combat});
|
battle.add_enemy({axe_ranger, &axe_ai, &axe_combat});
|
||||||
|
|
||||||
DinkyECS::Entity rat = 2;
|
DinkyECS::Entity rat = 2;
|
||||||
ai::EntityAI rat_ai("Enemy::actions", ai_start, ai_goal);
|
ai::EntityAI rat_ai("Enemy::actions", ai_start, ai_goal);
|
||||||
components::Combat rat_combat{10, 10, 2};
|
components::Combat rat_combat{10, 10, 2, 12, 18};
|
||||||
battle.add_enemy({rat, &rat_ai, &rat_combat});
|
battle.add_enemy({rat, &rat_ai, &rat_combat});
|
||||||
|
|
||||||
battle.set_all("enemy_found", true);
|
battle.set_all("enemy_found", true);
|
||||||
|
|
@ -42,7 +42,6 @@ TEST_CASE("battle operations fantasy", "[combat-battle]") {
|
||||||
battle.set(host, "have_healing", false);
|
battle.set(host, "have_healing", false);
|
||||||
battle.set(host, "tough_personality", false);
|
battle.set(host, "tough_personality", false);
|
||||||
|
|
||||||
|
|
||||||
while(host_combat.hp > 0) {
|
while(host_combat.hp > 0) {
|
||||||
battle.set(host, "health_good", host_combat.hp > 20);
|
battle.set(host, "health_good", host_combat.hp > 20);
|
||||||
|
|
||||||
|
|
@ -52,14 +51,14 @@ TEST_CASE("battle operations fantasy", "[combat-battle]") {
|
||||||
battle.plan();
|
battle.plan();
|
||||||
|
|
||||||
while(auto act = battle.next()) {
|
while(auto act = battle.next()) {
|
||||||
auto& [enemy, wants_to, cost, host_behavior] = *act;
|
auto& [enemy, wants_to, cost, enemy_state] = *act;
|
||||||
|
|
||||||
fmt::println(">>>>> entity: {} wants to {} cost={} and has {} HP and {} damage",
|
fmt::println(">>>>> entity: {} wants to {} cost={}; has {} HP; {} ap",
|
||||||
enemy.entity, wants_to,
|
enemy.entity, wants_to,
|
||||||
cost, enemy.combat->hp,
|
cost, enemy.combat->hp,
|
||||||
enemy.combat->damage);
|
enemy.combat->ap);
|
||||||
|
|
||||||
switch(host_behavior) {
|
switch(enemy_state) {
|
||||||
case BattleHostState::agree:
|
case BattleHostState::agree:
|
||||||
fmt::println("HOST and PLAYER requests match {}, doing it.",
|
fmt::println("HOST and PLAYER requests match {}, doing it.",
|
||||||
wants_to);
|
wants_to);
|
||||||
|
|
@ -72,6 +71,10 @@ TEST_CASE("battle operations fantasy", "[combat-battle]") {
|
||||||
if(wants_to == "kill_enemy") {
|
if(wants_to == "kill_enemy") {
|
||||||
enemy.combat->attack(host_combat);
|
enemy.combat->attack(host_combat);
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
|
case BattleHostState::out_of_ap:
|
||||||
|
fmt::println("ENEMY OUT OF AP");
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt::println("<<<<<<<<<<<<<<<<");
|
fmt::println("<<<<<<<<<<<<<<<<");
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue