forked from CleverRaven/Cataclysm-DDA
-
Notifications
You must be signed in to change notification settings - Fork 0
/
mongroup.h
238 lines (215 loc) · 8.03 KB
/
mongroup.h
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
#pragma once
#ifndef CATA_SRC_MONGROUP_H
#define CATA_SRC_MONGROUP_H
#include <iosfwd>
#include <map>
#include <set>
#include <vector>
#include "calendar.h"
#include "coordinates.h"
#include "enums.h"
#include "io_tags.h"
#include "mapgen.h"
#include "monster.h"
#include "point.h"
#include "type_id.h"
class JsonObject;
class JsonOut;
// from overmap.h
class overmap;
struct MonsterGroupEntry;
using FreqDef = std::vector<MonsterGroupEntry>;
using FreqDef_iter = FreqDef::iterator;
struct MonsterGroupEntry {
mtype_id name;
mongroup_id group;
int frequency;
int cost_multiplier;
int pack_minimum;
int pack_maximum;
spawn_data data;
std::vector<std::string> conditions;
time_duration starts;
time_duration ends;
holiday event;
bool lasts_forever() const {
return ends <= 0_turns;
}
bool is_group() const {
return group != mongroup_id();
}
MonsterGroupEntry( const mtype_id &id, int new_freq, int new_cost, int new_pack_min,
int new_pack_max, const spawn_data &new_data, const time_duration &new_starts,
const time_duration &new_ends, holiday new_event )
: name( id )
, group( mongroup_id() )
, frequency( new_freq )
, cost_multiplier( new_cost )
, pack_minimum( new_pack_min )
, pack_maximum( new_pack_max )
, data( new_data )
, starts( new_starts )
, ends( new_ends )
, event( new_event ) {
}
MonsterGroupEntry( const mongroup_id &id, int new_freq, int new_cost, int new_pack_min,
int new_pack_max, const spawn_data &new_data, const time_duration &new_starts,
const time_duration &new_ends, holiday new_event )
: name( mtype_id() )
, group( id )
, frequency( new_freq )
, cost_multiplier( new_cost )
, pack_minimum( new_pack_min )
, pack_maximum( new_pack_max )
, data( new_data )
, starts( new_starts )
, ends( new_ends )
, event( new_event ) {
}
};
struct MonsterGroupResult {
mtype_id name;
int pack_size;
spawn_data data;
MonsterGroupResult() : name( mtype_id::NULL_ID() ), pack_size( 0 ) {
}
MonsterGroupResult( const mtype_id &id, int new_pack_size, const spawn_data &new_data )
: name( id ), pack_size( new_pack_size ), data( new_data ) {
}
};
struct MonsterGroup {
mongroup_id name;
mtype_id defaultMonster;
FreqDef monsters;
bool IsMonsterInGroup( const mtype_id &id ) const;
bool is_animal = false;
// replaces this group after a period of
// time when exploring an unexplored portion of the map
bool replace_monster_group = false;
mongroup_id new_monster_group;
time_duration monster_group_time = 0_turns;
bool is_safe = false; /// Used for @ref mongroup::is_safe()
int freq_total = 0; // max number to roll for spawns (non-event)
std::map<holiday, int> event_freq; // total freq for each event
// Get the total frequency of entries that are valid for the specified event.
// This includes entries that have an event of "none". By default, use the current holiday.
int event_adjusted_freq_total( holiday event = holiday::num_holiday ) const;
};
struct mongroup {
mongroup_id type;
// Note: position is not saved as such in the json
// Instead, a vector of positions is saved for
tripoint_om_sm pos;
tripoint_abs_sm abs_pos; // position of the mongroup in absolute submap coordinates
unsigned int radius = 1;
unsigned int population = 1;
tripoint_om_sm target; // location the horde is interested in.
tripoint_abs_sm nemesis_target; // abs target for nemesis hordes
int interest = 0; //interest to target in percents
bool dying = false;
bool horde = false;
/** This property will be ignored if the vector is empty.
* Otherwise it will keep track of the individual monsters that
* are contained in this horde, and the population property will
* be ignored instead.
*/
std::vector<monster> monsters;
/** There are two types of hordes: "city", who try to stick around cities
* and return to them whenever possible.
* And "roam", who roam around the map randomly, not taking care to return
* anywhere.
*/
std::string horde_behaviour;
bool diffuse = false; // group size ind. of dist. from center and radius invariant
mongroup( const mongroup_id &ptype, const tripoint &ppos,
unsigned int prad, unsigned int ppop )
: type( ptype )
, pos( ppos )
, radius( prad )
, population( ppop ) {
}
mongroup( const mongroup_id &ptype, const tripoint_om_sm &ppos,
unsigned int prad, unsigned int ppop ) :
// TODO: fix point types
mongroup( ptype, ppos.raw(), prad, ppop ) {}
mongroup( const std::string &ptype, tripoint ppos, unsigned int prad, unsigned int ppop,
tripoint ptarget, int pint, bool pdie, bool phorde, bool pdiff ) :
type( ptype ), pos( ppos ), radius( prad ), population( ppop ), target( ptarget ),
interest( pint ), dying( pdie ), horde( phorde ), diffuse( pdiff ) { }
mongroup() = default;
bool is_safe() const;
bool empty() const;
void clear();
void set_target( const point_om_sm &p ) {
target.x() = p.x();
target.y() = p.y();
}
void set_nemesis_target( const tripoint_abs_sm &p ) {
nemesis_target.x() = p.x();
nemesis_target.y() = p.y();
}
void wander( const overmap & );
void inc_interest( int inc ) {
interest += inc;
if( interest > 100 ) {
interest = 100;
}
}
void dec_interest( int dec ) {
interest -= dec;
if( interest < 15 ) {
interest = 15;
}
}
void set_interest( int set ) {
if( set < 15 ) {
set = 15;
}
if( set > 100 ) {
set = 100;
}
interest = set;
}
float avg_speed() const;
template<typename Archive>
void io( Archive & );
using archive_type_tag = io::object_archive_tag;
void deserialize( const JsonObject &jo );
void deserialize_legacy( JsonIn &json );
void serialize( JsonOut &json ) const;
};
class MonsterGroupManager
{
public:
static void LoadMonsterGroup( const JsonObject &jo );
static void LoadMonsterBlacklist( const JsonObject &jo );
static void LoadMonsterWhitelist( const JsonObject &jo );
static void FinalizeMonsterGroups();
static MonsterGroupResult GetResultFromGroup( const mongroup_id &group, int *quantity = nullptr,
bool *mon_found = nullptr );
static bool IsMonsterInGroup( const mongroup_id &group, const mtype_id &monster );
static bool isValidMonsterGroup( const mongroup_id &group );
static const mongroup_id &Monster2Group( const mtype_id &monster );
static std::vector<mtype_id> GetMonstersFromGroup( const mongroup_id &group, bool from_subgroups );
static const MonsterGroup &GetMonsterGroup( const mongroup_id &group );
static const MonsterGroup &GetUpgradedMonsterGroup( const mongroup_id &group );
/**
* Gets a random monster, weighted by frequency.
* Ignores cost multiplier.
*/
static const mtype_id &GetRandomMonsterFromGroup( const mongroup_id &group );
static void check_group_definitions();
static void ClearMonsterGroups();
static bool monster_is_blacklisted( const mtype_id &m );
static bool is_animal( const mongroup_id &group );
private:
static std::map<mongroup_id, MonsterGroup> monsterGroupMap;
using t_string_set = std::set<std::string>;
static t_string_set monster_blacklist;
static t_string_set monster_whitelist;
static t_string_set monster_categories_blacklist;
static t_string_set monster_categories_whitelist;
static t_string_set monster_species_blacklist;
static t_string_set monster_species_whitelist;
};
#endif // CATA_SRC_MONGROUP_H