summaryrefslogtreecommitdiff
path: root/sim/src/BaseBattle.cpp
blob: 14a9ccd65152ac5654ce4a183f011b67f7012485 (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
#include "BaseBattle.h"

#include <algorithm>

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

#include <kurator/campaign/Scenario.h>
#include <kurator/campaign/UniqueIdentifier.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.h>

#include "Builder.h"


namespace kurator
{
namespace sim
{


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


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


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


void
BaseBattle::update(const float dt)
{
	time += dt;
	pick_random_targets();
	keep_at_range();
	update<FloatingMovement>(dt);
	update<TurretControl>(dt);
	kill_off_dead();
	manager.clear(_registry);  // registry supports on destructions events
	manager.update(_dispatcher);
}


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
BaseBattle::keep_at_range()
{
	auto view = _registry.view<Transform, AIState>();
	for (auto&& [entity, self, ai] : view.each()) {
		if (!_registry.valid(ai.target))
			continue;
		const auto target = _registry.get<Transform>(ai.target);
		const Point offset = target.position - self.position;
		ai.destination = target.position - offset.normalized().scale(ai.keep_at_range);
	}
}


void
BaseBattle::kill_off_dead()
{
	auto view = _registry.view<HitPoints>();
	for (auto&& [entity, points] : view.each()) {
		if (points.health > 0.0)
			continue;
		if (_registry.all_of<campaign::UniqueIdentifier, Team>(entity)) {
			const auto& [identifier, team] = _registry.get<campaign::UniqueIdentifier, Team>(entity);
			_dispatcher.trigger(stats::ShipLeft{time, identifier, team.id, true});
		}
		_registry.destroy(entity);
	}
}


}  // namespace sim
}  // namespace kurator