summaryrefslogtreecommitdiff
path: root/sim/src/BaseBattle.cpp
blob: fa9113447b255cc9e392b304668a6b7bd2e9c6f2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#include "BaseBattle.h"

#include <algorithm>

#include <entt/entity/registry.hpp>
#include <entt/signal/dispatcher.hpp>

#include <kurator/engine/Context.h>
#include <kurator/campaign/Scenario.h>
#include <kurator/sim/components.h>
#include <kurator/sim/events.h>
#include <kurator/sim/FloatingMovement.h>
#include <kurator/sim/HitPoints.h>
#include <kurator/sim/TurretControl.h>
#include <kurator/stats/events.h>
#include <kurator/universe/UniqueIdentifier.h>

#include "Builder.h"
#include "RandomSpawner.h"


namespace kurator
{
namespace sim
{


BaseBattle::BaseBattle(const campaign::Scenario& scenario) :
	_registry {}
{
	RandomSpawner spawner {scenario.last_team(), scenario.radius, 0.1};
	Builder build {_registry, spawner};
	for (const auto& ship : scenario.ships) {
		const auto entity = build(ship.loadout.type, ship.team);
		_registry.emplace<universe::UniqueIdentifier>(entity, ship.identifier);
		auto& state = _registry.get<AIState>(entity);
		for (const auto& turret_type : ship.loadout.turrets) {
			build(turret_type, entity);
			state.keep_at_range = std::min(state.keep_at_range, turret_type.optimal_range);
		}
		manager.add(ship.team, entity);  // registry supports on construction events
	}
}


entt::registry&
BaseBattle::registry()
{
	return _registry;
}


entt::dispatcher&
BaseBattle::dispatcher()
{
	return _dispatcher;
}


static void keep_at_range(engine::Context& ctx);
static void kill_off_dead(engine::Context& ctx);


void
BaseBattle::update(engine::Context& ctx)
{
	pick_random_targets();
	keep_at_range(ctx);
	FloatingMovement::update(ctx);
	TurretControl::update(ctx);
	kill_off_dead(ctx);
	manager.update(ctx);
}


void
BaseBattle::pick_random_targets()
{
	auto view = _registry.view<Team, AIState>();
	for (auto&& [entity, team, ai] : view.each()) {
		if (!_registry.valid(ai.target))
			ai.target = manager.random(team.id);
	}
}


void
keep_at_range(engine::Context& ctx)
{
	auto view = ctx.registry.view<Transform, AIState>();
	for (auto&& [entity, self, ai] : view.each()) {
		if (!ctx.registry.valid(ai.target))
			continue;
		const auto target = ctx.registry.get<Transform>(ai.target);
		const auto offset = target.position - self.position;
		ai.destination = target.position - offset.normalized().scale(ai.keep_at_range);
	}
}


void
kill_off_dead(engine::Context& ctx)
{
	auto view = ctx.registry.view<HitPoints>();
	for (auto&& [entity, points] : view.each()) {
		if (points.is_alive())
			continue;
		if (ctx.registry.all_of<universe::UniqueIdentifier, Team>(entity)) {
			const auto& [identifier, team] = ctx.registry.get<universe::UniqueIdentifier, Team>(entity);
			ctx.dispatcher.trigger(stats::ShipLeft{ctx.clock.game, identifier, team.id, true});
			ctx.dispatcher.trigger(Destroyed{entity});
		}
		ctx.registry.destroy(entity);
	}
}


}  // namespace sim
}  // namespace kurator