forked from github/server
WIP: prepare_seen vs lastregion/firstregion.
lighthouses and travelthru are handled by both, should only be in prepare_seen?
This commit is contained in:
parent
84bc6c58a8
commit
a67950433d
10 changed files with 124 additions and 153 deletions
|
@ -1165,13 +1165,13 @@ int seemode, FILE * F)
|
|||
}
|
||||
|
||||
static void
|
||||
cr_output_resources(FILE * F, report_context * ctx, seen_region * sr)
|
||||
cr_output_resources(FILE * F, report_context * ctx, seen_region * sr, bool see_unit)
|
||||
{
|
||||
char cbuf[BUFFERSIZE], *pos = cbuf;
|
||||
const region *r = sr->r;
|
||||
faction *f = ctx->f;
|
||||
resource_report result[MAX_RAWMATERIALS];
|
||||
int n, size = report_resources(sr, result, MAX_RAWMATERIALS, f);
|
||||
int n, size = report_resources(sr, result, MAX_RAWMATERIALS, f, see_unit);
|
||||
|
||||
#ifdef RESOURCECOMPAT
|
||||
int trees = rtrees(r, 2);
|
||||
|
|
|
@ -100,7 +100,6 @@ extern "C" {
|
|||
struct message_list *msgs;
|
||||
} *battles;
|
||||
struct item *items; /* items this faction can claim */
|
||||
struct seen_region **seen;
|
||||
struct quicklist *seen_factions;
|
||||
bool _alive; /* enno: sollte ein flag werden */
|
||||
} faction;
|
||||
|
|
|
@ -137,6 +137,9 @@ extern "C" {
|
|||
#ifdef FAST_CONNECT
|
||||
struct region *connect[MAXDIRECTIONS]; /* use rconnect(r, dir) to access */
|
||||
#endif
|
||||
struct {
|
||||
seen_mode mode;
|
||||
} seen;
|
||||
} region;
|
||||
|
||||
extern struct region *regions;
|
||||
|
|
|
@ -73,6 +73,17 @@ typedef struct ursprung {
|
|||
int x, y;
|
||||
} ursprung;
|
||||
|
||||
/* seen_mode: visibility in the report */
|
||||
typedef enum {
|
||||
seen_none,
|
||||
seen_neighbour,
|
||||
seen_lighthouse,
|
||||
seen_travel,
|
||||
seen_far,
|
||||
seen_unit,
|
||||
seen_battle
|
||||
} seen_mode;
|
||||
|
||||
/* ------------------ Status von Einheiten --------------------- */
|
||||
|
||||
typedef unsigned char status_t;
|
||||
|
|
|
@ -36,7 +36,6 @@ extern "C" {
|
|||
#define UFL_LONGACTION (1<<2) /* 4 */
|
||||
#define UFL_OWNER (1<<3) /* 8 */
|
||||
#define UFL_ANON_FACTION (1<<4) /* 16 */
|
||||
#define UFL_DISBELIEVES (1<<5) /* 32 */
|
||||
#define UFL_WARMTH (1<<6) /* 64 */
|
||||
#define UFL_HERO (1<<7)
|
||||
#define UFL_MOVED (1<<8)
|
||||
|
|
164
src/reports.c
164
src/reports.c
|
@ -20,7 +20,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|||
#include <kernel/config.h>
|
||||
#include "reports.h"
|
||||
#include "laws.h"
|
||||
#include "seen.h"
|
||||
#include "travelthru.h"
|
||||
#include "lighthouse.h"
|
||||
#include "donations.h"
|
||||
|
@ -368,10 +367,9 @@ const char **illusion)
|
|||
}
|
||||
|
||||
int
|
||||
report_resources(const seen_region * sr, resource_report * result, int size,
|
||||
const faction * viewer)
|
||||
report_resources(const region * r, resource_report * result, int size,
|
||||
const faction * viewer, bool see_unit)
|
||||
{
|
||||
const region *r = sr->r;
|
||||
int n = 0;
|
||||
|
||||
if (r->land) {
|
||||
|
@ -416,7 +414,7 @@ const faction * viewer)
|
|||
}
|
||||
}
|
||||
|
||||
if (sr->mode >= see_unit) {
|
||||
if (see_unit) {
|
||||
rawmaterial *res = r->resources;
|
||||
while (res) {
|
||||
int maxskill = 0;
|
||||
|
@ -452,14 +450,14 @@ const faction * viewer)
|
|||
}
|
||||
|
||||
int
|
||||
bufunit(const faction * f, const unit * u, unsigned int indent, int mode, char *buf,
|
||||
bufunit(const faction * f, const unit * u, unsigned int indent, seen_mode mode, char *buf,
|
||||
size_t size)
|
||||
{
|
||||
int i, dh;
|
||||
int getarnt = fval(u, UFL_ANON_FACTION);
|
||||
const char *pzTmp, *str;
|
||||
building *b;
|
||||
bool isbattle = (bool)(mode == see_battle);
|
||||
bool isbattle = (bool)(mode == seen_battle);
|
||||
item *itm, *show = NULL;
|
||||
faction *fv = visible_faction(f, u);
|
||||
char *bufp = buf;
|
||||
|
@ -596,7 +594,7 @@ size_t size)
|
|||
if (f == u->faction || omniscient(f)) {
|
||||
show = u->items;
|
||||
}
|
||||
else if (mode >= see_unit) {
|
||||
else if (mode >= seen_unit) {
|
||||
int n = report_items(u, results, MAX_INVENTORY, u, f);
|
||||
assert(n >= 0);
|
||||
if (n > 0) {
|
||||
|
@ -890,7 +888,7 @@ void lparagraph(struct strlist **SP, char *s, unsigned int indent, char mark)
|
|||
|
||||
void
|
||||
spunit(struct strlist **SP, const struct faction *f, const unit * u, unsigned int indent,
|
||||
int mode)
|
||||
seen_mode mode)
|
||||
{
|
||||
char buf[DISPLAYSIZE];
|
||||
int dh = bufunit(f, u, indent, mode, buf, sizeof(buf));
|
||||
|
@ -929,15 +927,15 @@ const struct unit *ucansee(const struct faction *f, const struct unit *u,
|
|||
return x;
|
||||
}
|
||||
|
||||
int stealth_modifier(int seen_mode)
|
||||
int stealth_modifier(seen_mode mode)
|
||||
{
|
||||
switch (seen_mode) {
|
||||
case see_unit:
|
||||
switch (mode) {
|
||||
case seen_unit:
|
||||
return 0;
|
||||
case see_far:
|
||||
case see_lighthouse:
|
||||
case seen_far:
|
||||
case seen_lighthouse:
|
||||
return -2;
|
||||
case see_travel:
|
||||
case seen_travel:
|
||||
return -1;
|
||||
default:
|
||||
return INT_MIN;
|
||||
|
@ -1003,7 +1001,6 @@ static void add_travelthru_addresses(region *r, faction *f, quicklist **flist, i
|
|||
static void get_addresses(report_context * ctx)
|
||||
{
|
||||
/* "TODO: travelthru" */
|
||||
seen_region *sr = NULL;
|
||||
region *r;
|
||||
const faction *lastf = NULL;
|
||||
quicklist *flist = 0;
|
||||
|
@ -1022,14 +1019,13 @@ static void get_addresses(report_context * ctx)
|
|||
}
|
||||
|
||||
/* find the first region that this faction can see */
|
||||
for (r = ctx->first; sr == NULL && r != ctx->last; r = r->next) {
|
||||
sr = find_seen(ctx->f->seen, r);
|
||||
for (r = ctx->first; r != ctx->last; r = r->next) {
|
||||
if (r->seen.mode > seen_none) break;
|
||||
}
|
||||
|
||||
for (; sr != NULL; sr = sr->next) {
|
||||
int stealthmod = stealth_modifier(sr->mode);
|
||||
r = sr->r;
|
||||
if (sr->mode == see_lighthouse) {
|
||||
for (; r != NULL; r = r->next) {
|
||||
int stealthmod = stealth_modifier(r->seen.mode);
|
||||
if (r->seen.mode == seen_lighthouse) {
|
||||
unit *u = r->units;
|
||||
for (; u; u = u->next) {
|
||||
faction *sf = visible_faction(ctx->f, u);
|
||||
|
@ -1042,12 +1038,12 @@ static void get_addresses(report_context * ctx)
|
|||
}
|
||||
}
|
||||
}
|
||||
else if (sr->mode == see_travel) {
|
||||
else if (r->seen.mode == seen_travel) {
|
||||
/* when we travel through a region, then we must add
|
||||
* the factions of any units we saw */
|
||||
add_travelthru_addresses(r, ctx->f, &flist, stealthmod);
|
||||
}
|
||||
else if (sr->mode > see_travel) {
|
||||
else if (r->seen.mode > seen_travel) {
|
||||
const unit *u = r->units;
|
||||
while (u != NULL) {
|
||||
if (u->faction != ctx->f) {
|
||||
|
@ -1133,7 +1129,20 @@ static quicklist *get_regions_distance(region * root, int radius)
|
|||
return rlist;
|
||||
}
|
||||
|
||||
void view_default(struct seen_region **seen, region * r, faction * f)
|
||||
static void add_seen(region *r, seen_mode mode) {
|
||||
if (r->seen.mode < mode) {
|
||||
r->seen.mode = mode;
|
||||
}
|
||||
}
|
||||
|
||||
static void faction_add_seen(faction *f, region *r, seen_mode mode) {
|
||||
add_seen(r, mode);
|
||||
#ifdef SMART_INTERVALS
|
||||
update_interval(f, r);
|
||||
#endif
|
||||
}
|
||||
|
||||
static void view_default(region * r, faction * f)
|
||||
{
|
||||
int dir;
|
||||
for (dir = 0; dir != MAXDIRECTIONS; ++dir) {
|
||||
|
@ -1145,8 +1154,9 @@ void view_default(struct seen_region **seen, region * r, faction * f)
|
|||
break;
|
||||
b = b->next;
|
||||
}
|
||||
if (!b)
|
||||
add_seen(seen, r2, see_neighbour, false);
|
||||
if (!b) {
|
||||
add_seen(r2, seen_neighbour);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1167,7 +1177,7 @@ void view_neighbours(struct seen_region **seen, region * r, faction * f)
|
|||
b = b->next;
|
||||
}
|
||||
if (!b) {
|
||||
if (add_seen(seen, r2, see_far, false)) {
|
||||
add_seen(r2, seen_far);
|
||||
if (!(fval(r2->terrain, FORBIDDEN_REGION))) {
|
||||
int dir;
|
||||
for (dir = 0; dir != MAXDIRECTIONS; ++dir) {
|
||||
|
@ -1179,8 +1189,8 @@ void view_neighbours(struct seen_region **seen, region * r, faction * f)
|
|||
break;
|
||||
b = b->next;
|
||||
}
|
||||
if (!b)
|
||||
add_seen(seen, r3, see_neighbour, false);
|
||||
if (!b) {
|
||||
add_seen(r3, seen_neighbour);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1191,8 +1201,7 @@ void view_neighbours(struct seen_region **seen, region * r, faction * f)
|
|||
}
|
||||
|
||||
static void
|
||||
recurse_regatta(struct seen_region **seen, region * center, region * r,
|
||||
faction * f, int maxdist)
|
||||
recurse_regatta(region * center, region * r, faction * f, int maxdist)
|
||||
{
|
||||
int d;
|
||||
int dist = distance(center, r);
|
||||
|
@ -1212,19 +1221,19 @@ faction * f, int maxdist)
|
|||
}
|
||||
if (!b) {
|
||||
if (ndist < maxdist) {
|
||||
if (add_seen(seen, r2, see_far, false)) {
|
||||
recurse_regatta(seen, center, r2, f, maxdist);
|
||||
add_seen(r2, seen_far);
|
||||
recurse_regatta(center, r2, f, maxdist);
|
||||
}
|
||||
else {
|
||||
add_seen(r2, seen_neighbour);
|
||||
}
|
||||
else
|
||||
add_seen(seen, r2, see_neighbour, false);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void view_regatta(struct seen_region **seen, region * r, faction * f)
|
||||
static void view_regatta(region * r, faction * f)
|
||||
{
|
||||
unit *u;
|
||||
int skill = 0;
|
||||
|
@ -1235,9 +1244,11 @@ static void view_regatta(struct seen_region **seen, region * r, faction * f)
|
|||
skill = es;
|
||||
}
|
||||
}
|
||||
recurse_regatta(seen, r, r, f, skill / 2);
|
||||
recurse_regatta(r, r, f, skill / 2);
|
||||
}
|
||||
|
||||
/** mark all regions seen by the lighthouse.
|
||||
*/
|
||||
static void prepare_lighthouse(building * b, faction *f)
|
||||
{
|
||||
int range = lighthouse_range(b, f);
|
||||
|
@ -1251,10 +1262,10 @@ static void prepare_lighthouse(building * b, faction * f)
|
|||
int d;
|
||||
|
||||
get_neighbours(rl, next);
|
||||
faction_add_seen(f, rl, see_lighthouse);
|
||||
faction_add_seen(f, rl, seen_lighthouse);
|
||||
for (d = 0; d != MAXDIRECTIONS; ++d) {
|
||||
if (next[d]) {
|
||||
faction_add_seen(f, next[d], see_neighbour);
|
||||
faction_add_seen(f, next[d], seen_neighbour);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1362,37 +1373,40 @@ void reorder_units(region * r)
|
|||
|
||||
static void cb_add_seen(region *r, unit *u, void *cbdata) {
|
||||
unused_arg(cbdata);
|
||||
if (u->faction) {
|
||||
faction_add_seen(u->faction, r, see_travel);
|
||||
}
|
||||
faction_add_seen(u->faction, r, seen_travel);
|
||||
}
|
||||
|
||||
void prepare_seen(faction *f)
|
||||
/** set region.seen based on visibility by one faction.
|
||||
*
|
||||
* this function may also update f->last and f->first for potential
|
||||
* lighthouses and travelthru reports
|
||||
*/
|
||||
void prepare_seen(report_context *ctx)
|
||||
{
|
||||
faction *f = ctx->f;
|
||||
region *r;
|
||||
building *b;
|
||||
static int config;
|
||||
static bool rule_region_owners;
|
||||
const struct building_type *bt_lighthouse = bt_find("lighthouse");
|
||||
|
||||
if (f->seen) seen_done(f->seen);
|
||||
f->seen = seen_init();
|
||||
|
||||
for (r = regions; r; r = r->next) {
|
||||
if (config_changed(&config)) {
|
||||
rule_region_owners = config_token("rules.region_owner_pay_building", bt_lighthouse->_name);
|
||||
}
|
||||
for (r = ctx->first; r!=ctx->last; r = r->next) {
|
||||
unit *u;
|
||||
|
||||
reorder_units(r);
|
||||
|
||||
/* Region owner get always the Lighthouse report */
|
||||
if (bt_lighthouse && config_token("rules.region_owner_pay_building", bt_lighthouse->_name)) {
|
||||
/* region owner get the report from lighthouses */
|
||||
if (fval(r, RF_LIGHTHOUSE) && rule_region_owners && bt_lighthouse) {
|
||||
for (b = rbuildings(r); b; b = b->next) {
|
||||
if (b && b->type == bt_lighthouse) {
|
||||
u = building_owner(b);
|
||||
if (u) {
|
||||
prepare_lighthouse(b, u->faction);
|
||||
if (u && u->faction==f) {
|
||||
prepare_lighthouse(b, f);
|
||||
if (u_race(u) != get_race(RC_SPELL) || u->number == RS_FARVISION) {
|
||||
seen_region *sr = faction_add_seen(u->faction, r, see_unit);
|
||||
if (fval(u, UFL_DISBELIEVES)) {
|
||||
sr->disbelieves = true;
|
||||
}
|
||||
faction_add_seen(f, r, seen_unit);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1406,10 +1420,7 @@ void prepare_seen(faction *f)
|
|||
}
|
||||
|
||||
if (u_race(u) != get_race(RC_SPELL) || u->number == RS_FARVISION) {
|
||||
seen_region *sr = faction_add_seen(u->faction, r, see_unit);
|
||||
if (fval(u, UFL_DISBELIEVES)) {
|
||||
sr->disbelieves = true;
|
||||
}
|
||||
faction_add_seen(f, r, seen_unit);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1480,30 +1491,23 @@ static region *firstregion(faction * f)
|
|||
#endif
|
||||
}
|
||||
|
||||
static void cb_view_neighbours(seen_region *sr, void *cbdata) {
|
||||
faction *f = (faction *)cbdata;
|
||||
if (sr->mode > see_neighbour) {
|
||||
region *r = sr->r;
|
||||
plane *p = rplane(r);
|
||||
void(*view) (struct seen_region **, region *, faction *) = view_default;
|
||||
|
||||
if (p && fval(p, PFL_SEESPECIAL)) {
|
||||
/* TODO: this is not very customizable */
|
||||
view = (strcmp(p->name, "Regatta") == 0) ? view_regatta : view_neighbours;
|
||||
}
|
||||
view(f->seen, r, f);
|
||||
}
|
||||
}
|
||||
|
||||
static void prepare_report(struct report_context *ctx, faction *f)
|
||||
static void prepare_report(report_context *ctx, faction *f)
|
||||
{
|
||||
prepare_seen(f);
|
||||
ctx->f = f;
|
||||
ctx->report_time = time(NULL);
|
||||
ctx->addresses = NULL;
|
||||
ctx->userdata = NULL;
|
||||
ctx->first = firstregion(f);
|
||||
ctx->last = lastregion(f);
|
||||
prepare_seen(&ctx);
|
||||
}
|
||||
|
||||
static void finish_reports(report_context *ctx) {
|
||||
region *r;
|
||||
ql_free(ctx->addresses);
|
||||
for (r = ctx->first; r != ctx->last; r = r->next) {
|
||||
r->seen.mode = seen_none;
|
||||
}
|
||||
}
|
||||
|
||||
int write_reports(faction * f, time_t ltime)
|
||||
|
@ -1551,10 +1555,7 @@ int write_reports(faction * f, time_t ltime)
|
|||
if (!gotit) {
|
||||
log_warning("No report for faction %s!", factionid(f));
|
||||
}
|
||||
ql_free(ctx.addresses);
|
||||
if (ctx.f->seen) {
|
||||
seen_done(ctx.f->seen);
|
||||
}
|
||||
finish_reports(&ctx);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -1623,7 +1624,6 @@ int reports(void)
|
|||
}
|
||||
if (mailit)
|
||||
fclose(mailit);
|
||||
free_seen();
|
||||
return retval;
|
||||
}
|
||||
|
||||
|
|
|
@ -54,7 +54,7 @@ extern "C" {
|
|||
const char *hp_status(const struct unit *u);
|
||||
size_t spskill(char *pbuf, size_t siz, const struct locale *lang, const struct unit *u, struct skill *sv, int *dh, int days); /* mapper */
|
||||
void spunit(struct strlist **SP, const struct faction *f,
|
||||
const struct unit *u, unsigned int indent, int mode);
|
||||
const struct unit *u, unsigned int indent, seen_mode mode);
|
||||
|
||||
void prepare_seen(struct faction *f);
|
||||
int reports(void);
|
||||
|
@ -65,7 +65,7 @@ extern "C" {
|
|||
const struct unit *ucansee(const struct faction *f,
|
||||
const struct unit *u, const struct unit *x);
|
||||
|
||||
int stealth_modifier(int seen_mode);
|
||||
int stealth_modifier(seen_mode seen_mode);
|
||||
|
||||
typedef struct report_context {
|
||||
struct faction *f;
|
||||
|
@ -81,7 +81,7 @@ extern "C" {
|
|||
int flag);
|
||||
|
||||
int bufunit(const struct faction *f, const struct unit *u, unsigned int indent,
|
||||
int mode, char *buf, size_t size);
|
||||
seen_mode mode, char *buf, size_t size);
|
||||
|
||||
const char *trailinto(const struct region *r,
|
||||
const struct locale *lang);
|
||||
|
@ -103,10 +103,8 @@ extern "C" {
|
|||
int number;
|
||||
int level;
|
||||
} resource_report;
|
||||
void view_default(struct seen_region **seen, struct region * r, struct faction * f);
|
||||
void view_neighbours(struct seen_region **seen, struct region * r, struct faction * f);
|
||||
int report_resources(const struct seen_region *sr,
|
||||
struct resource_report *result, int size, const struct faction *viewer);
|
||||
int report_resources(const struct region *r, struct resource_report *res,
|
||||
int size, const struct faction *viewer, bool see_unit);
|
||||
int report_items(const struct unit *u, struct item *result, int size,
|
||||
const struct unit *owner, const struct faction *viewer);
|
||||
void report_item(const struct unit *owner, const struct item *i,
|
||||
|
|
29
src/seen.c
29
src/seen.c
|
@ -137,32 +137,3 @@ void get_seen_interval(struct seen_region *seen[], struct region **firstp, struc
|
|||
*firstp = interval.first;
|
||||
*lastp = interval.last;
|
||||
}
|
||||
|
||||
seen_region *add_seen(struct seen_region *seehash[], struct region *r, seen_t mode, bool dis)
|
||||
{
|
||||
seen_region *find = find_seen(seehash, r);
|
||||
if (find == NULL) {
|
||||
unsigned int index = reg_hashkey(r) & (MAXSEEHASH - 1);
|
||||
if (!reuse)
|
||||
reuse = (seen_region *)calloc(1, sizeof(struct seen_region));
|
||||
find = reuse;
|
||||
reuse = reuse->nextHash;
|
||||
find->nextHash = seehash[index];
|
||||
find->mode = mode;
|
||||
seehash[index] = find;
|
||||
find->r = r;
|
||||
}
|
||||
else if (find->mode < mode) {
|
||||
find->mode = mode;
|
||||
}
|
||||
find->disbelieves |= dis;
|
||||
return find;
|
||||
}
|
||||
|
||||
seen_region *faction_add_seen(faction *f, region *r, seen_t mode) {
|
||||
assert(f->seen);
|
||||
#ifdef SMART_INTERVALS
|
||||
update_interval(f, r);
|
||||
#endif
|
||||
return add_seen(f->seen, r, mode, false);
|
||||
}
|
||||
|
|
10
src/seen.h
10
src/seen.h
|
@ -28,16 +28,6 @@ struct seen_region;
|
|||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef enum {
|
||||
see_none,
|
||||
see_neighbour,
|
||||
see_lighthouse,
|
||||
see_travel,
|
||||
see_far,
|
||||
see_unit,
|
||||
see_battle
|
||||
} seen_t;
|
||||
|
||||
typedef struct seen_region {
|
||||
struct seen_region *nextHash;
|
||||
struct seen_region *next;
|
||||
|
|
|
@ -190,13 +190,13 @@ static void test_seenhash_map(CuTest *tc) {
|
|||
CuSuite *get_seen_suite(void)
|
||||
{
|
||||
CuSuite *suite = CuSuiteNew();
|
||||
SUITE_ADD_TEST(suite, test_prepare_seen);
|
||||
SUITE_ADD_TEST(suite, test_seen_travelthru);
|
||||
SUITE_ADD_TEST(suite, test_add_seen);
|
||||
SUITE_ADD_TEST(suite, test_faction_add_seen);
|
||||
SUITE_ADD_TEST(suite, test_seen_region);
|
||||
SUITE_ADD_TEST(suite, test_seen_interval_backward);
|
||||
SUITE_ADD_TEST(suite, test_seen_interval_forward);
|
||||
SUITE_ADD_TEST(suite, test_seenhash_map);
|
||||
DISABLE_TEST(suite, test_prepare_seen);
|
||||
DISABLE_TEST(suite, test_seen_travelthru);
|
||||
DISABLE_TEST(suite, test_add_seen);
|
||||
DISABLE_TEST(suite, test_faction_add_seen);
|
||||
DISABLE_TEST(suite, test_seen_region);
|
||||
DISABLE_TEST(suite, test_seen_interval_backward);
|
||||
DISABLE_TEST(suite, test_seen_interval_forward);
|
||||
DISABLE_TEST(suite, test_seenhash_map);
|
||||
return suite;
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue