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
|
#include "Government.hpp"
#include "openvic-simulation/GameManager.hpp"
using namespace OpenVic;
using namespace OpenVic::NodeTools;
GovernmentType::GovernmentType(std::string_view new_identifier, std::vector<Ideology const*>&& new_ideologies, bool new_elections, bool new_appoint_ruling_party, Timespan new_term_duration, std::string_view new_flag_type_identifier)
: HasIdentifier { new_identifier }, ideologies { std::move(new_ideologies) }, elections { new_elections }, appoint_ruling_party { new_appoint_ruling_party }, term_duration { new_term_duration }, flag_type_identifier { new_flag_type_identifier } {}
bool GovernmentType::is_ideology_compatible(Ideology const* ideology) const {
return std::find(ideologies.begin(), ideologies.end(), ideology) != ideologies.end();
}
std::vector<Ideology const*> const& GovernmentType::get_ideologies() const {
return ideologies;
}
bool GovernmentType::holds_elections() const {
return elections;
}
bool GovernmentType::can_appoint_ruling_party() const {
return appoint_ruling_party;
}
Timespan GovernmentType::get_term_duration() const {
return term_duration;
}
std::string_view GovernmentType::get_flag_type() const {
return flag_type_identifier;
}
GovernmentTypeManager::GovernmentTypeManager() : government_types { "government types" } {}
bool GovernmentTypeManager::add_government_type(std::string_view identifier, std::vector<Ideology const*>&& ideologies, bool elections, bool appoint_ruling_party, Timespan term_duration, std::string_view flag_type) {
if (identifier.empty()) {
Logger::error("Invalid government type identifier - empty!");
return false;
}
if (ideologies.empty()) {
Logger::error("No compatible ideologies defined for government type ", identifier);
return false;
}
if (elections && term_duration < 0) {
Logger::error("No or invalid term duration for government type ", identifier);
return false;
}
return government_types.add_item({ identifier, std::move(ideologies), elections, appoint_ruling_party, term_duration, flag_type });
}
/* REQUIREMENTS: FS-525, SIM-27 */
bool GovernmentTypeManager::load_government_types_file(IdeologyManager const& ideology_manager, ast::NodeCPtr root) {
bool ret = expect_dictionary_reserve_length(
government_types,
[this, &ideology_manager](std::string_view government_type_identifier, ast::NodeCPtr government_type_value) -> bool {
std::vector<Ideology const*> ideologies;
bool elections = false, appoint_ruling_party = false;
Timespan term_duration = 0;
std::string_view flag_type_identifier = "republic";
size_t total_expected_ideologies = 0;
bool ret = expect_dictionary_keys_and_default(
increment_callback(total_expected_ideologies),
"election", ONE_EXACTLY, expect_bool(assign_variable_callback(elections)),
"duration", ZERO_OR_ONE, expect_months(assign_variable_callback(term_duration)),
"appoint_ruling_party", ONE_EXACTLY, expect_bool(assign_variable_callback(appoint_ruling_party)),
"flagType", ZERO_OR_ONE, expect_identifier(assign_variable_callback(flag_type_identifier))
)(government_type_value);
ideologies.reserve(total_expected_ideologies);
ret &= expect_dictionary(
[this, &ideology_manager, &ideologies, government_type_identifier](std::string_view key, ast::NodeCPtr value) -> bool {
static const string_set_t reserved_keys = {
"election", "duration", "appoint_ruling_party", "flagType"
};
if (reserved_keys.find(key) != reserved_keys.end()) return true;
Ideology const* ideology = ideology_manager.get_ideology_by_identifier(key);
if (ideology == nullptr) {
Logger::error("When loading government type ", government_type_identifier, ", specified ideology ", key, " is invalid!");
return false;
}
return expect_bool([&ideologies, ideology, government_type_identifier](bool val) -> bool {
if (val) {
if (std::find(ideologies.begin(), ideologies.end(), ideology) == ideologies.end()) {
ideologies.push_back(ideology);
return true;
}
Logger::error("Government type ", government_type_identifier, " marked as supporting ideology ", ideology->get_identifier());
return false;
}
Logger::error("Government type ", government_type_identifier, " redundantly marked as not supporting ideology ", ideology->get_identifier(), " multiple times");
return false;
})(value);
}
)(government_type_value);
ret &= add_government_type(government_type_identifier, std::move(ideologies), elections, appoint_ruling_party, term_duration, flag_type_identifier);
return ret;
}
)(root);
lock_government_types();
return ret;
}
|