server/src/common/kernel/region.c

1208 lines
25 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
*
* Eressea PB(E)M host Copyright (C) 1998-2003
2001-01-25 10:37:55 +01:00
* Christian Schlittchen (corwin@amber.kn-bremen.de)
* Katja Zedel (katze@felidae.kn-bremen.de)
* Henning Peters (faroul@beyond.kn-bremen.de)
* Enno Rehling (enno@eressea-pbem.de)
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* based on:
*
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
* This program may not be sold or used commercially without prior written
* permission from the authors.
*/
#include <config.h>
#include "eressea.h"
#include "region.h"
/* kernel includes */
#include "border.h"
#include "building.h"
#include "curse.h"
#include "equipment.h"
2001-01-25 10:37:55 +01:00
#include "faction.h"
#include "item.h"
#include "message.h"
2001-01-25 10:37:55 +01:00
#include "plane.h"
#include "region.h"
#include "resources.h"
#include "save.h"
#include "terrain.h"
#include "terrainid.h"
#include "unit.h"
#include "version.h"
2001-01-25 10:37:55 +01:00
/* util includes */
2007-06-20 02:34:02 +02:00
#include <util/attrib.h>
#include <util/lists.h>
#include <util/resolve.h>
#include <util/umlaut.h>
#include <util/language.h>
#include <util/rng.h>
2001-01-25 10:37:55 +01:00
/* libc includes */
#include <assert.h>
#include <ctype.h>
#include <limits.h>
#include <rand.h>
2001-01-25 10:37:55 +01:00
#include <stdio.h>
#include <string.h>
extern int dice_rand(const char *s);
2005-11-26 01:08:47 +01:00
int
get_maxluxuries()
{
static int maxluxuries = -1;
if (maxluxuries==-1) {
const luxury_type * ltype;
maxluxuries = 0;
for (ltype = luxurytypes;ltype;ltype=ltype->next) ++maxluxuries;
}
return maxluxuries;
}
const short delta_x[MAXDIRECTIONS] =
2001-01-25 10:37:55 +01:00
{
-1, 0, 1, 1, 0, -1
2001-01-25 10:37:55 +01:00
};
const short delta_y[MAXDIRECTIONS] =
2001-01-25 10:37:55 +01:00
{
1, 1, 0, -1, -1, 0
2001-01-25 10:37:55 +01:00
};
static const direction_t back[MAXDIRECTIONS] =
2001-01-25 10:37:55 +01:00
{
D_SOUTHEAST,
D_SOUTHWEST,
D_WEST,
D_NORTHWEST,
D_NORTHEAST,
D_EAST,
2001-01-25 10:37:55 +01:00
};
direction_t
dir_invert(direction_t dir)
{
switch (dir) {
case D_PAUSE:
case D_SPECIAL:
return dir;
break;
default:
if (dir>=0 && dir<MAXDIRECTIONS) return back[dir];
}
assert(!"illegal direction");
return NODIRECTION;
}
const xmlChar *
write_regionname(const region * r, const faction * f, xmlChar * buffer, size_t size)
2001-01-25 10:37:55 +01:00
{
char * buf = (char *)buffer;
const struct locale * lang = f ? f->locale : 0;
if (r==NULL) {
strcpy(buf, "(null)");
} else {
plane *pl = r->planep;
if (pl && fval(pl, PFL_NOCOORDS)) {
strncpy(buf, (const char *)rname(r, lang), size);
} else {
snprintf(buf, size, "%s (%d,%d)", rname(r, lang),
region_x(r, f), region_y(r, f));
}
}
buf[64] = 0;
return buffer;
}
const xmlChar *
regionname(const region * r, const faction * f)
{
static xmlChar buf[65];
return write_regionname(r, f, buf, sizeof(buf));
2001-01-25 10:37:55 +01:00
}
int
2007-02-11 23:33:35 +01:00
deathcount(const region * r)
{
attrib * a = a_find(r->attribs, &at_deathcount);
if (!a) return 0;
return a->data.i;
2001-01-25 10:37:55 +01:00
}
int
2007-02-11 23:33:35 +01:00
chaoscount(const region * r)
{
attrib * a = a_find(r->attribs, &at_chaoscount);
if (!a) return 0;
return a->data.i;
2001-01-25 10:37:55 +01:00
}
void
2007-02-11 23:33:35 +01:00
deathcounts (region * r, int fallen)
{
attrib * a;
static const curse_type * ctype = NULL;
2001-01-25 10:37:55 +01:00
if (fallen==0) return;
if (!ctype) ctype = ct_find("holyground");
if (ctype && curse_active(get_curse(r->attribs, ctype))) return;
2001-01-25 10:37:55 +01:00
a = a_find(r->attribs, &at_deathcount);
if (!a) a = a_add(&r->attribs, a_new(&at_deathcount));
a->data.i += fallen;
2001-01-25 10:37:55 +01:00
if (a->data.i<=0) a_remove(&r->attribs, a);
2001-01-25 10:37:55 +01:00
}
void
2007-02-11 23:33:35 +01:00
chaoscounts(region * r, int fallen)
{
attrib * a;
2001-01-25 10:37:55 +01:00
if (fallen==0) return;
2001-01-25 10:37:55 +01:00
a = a_find(r->attribs, &at_chaoscount);
if (!a) a = a_add(&r->attribs, a_new(&at_chaoscount));
a->data.i += fallen;
2001-01-25 10:37:55 +01:00
if (a->data.i<=0) a_remove(&r->attribs, a);
2001-01-25 10:37:55 +01:00
}
2001-01-25 10:37:55 +01:00
/********************/
/* at_direction */
/********************/
2007-02-11 23:33:35 +01:00
static void
2001-01-25 10:37:55 +01:00
a_initdirection(attrib *a)
{
a->data.v = calloc(1, sizeof(spec_direction));
2001-01-25 10:37:55 +01:00
}
2007-02-11 23:33:35 +01:00
static void
a_freedirection(attrib *a)
{
free(a->data.v);
}
static int
2001-01-25 10:37:55 +01:00
a_agedirection(attrib *a)
{
spec_direction *d = (spec_direction *)(a->data.v);
2001-01-25 10:37:55 +01:00
if (d->duration > 0) d->duration--;
else d->duration = 0;
2001-01-25 10:37:55 +01:00
return d->duration;
2001-01-25 10:37:55 +01:00
}
typedef struct dir_lookup {
char * name;
const xmlChar * oldname;
struct dir_lookup * next;
} dir_lookup;
static dir_lookup * dir_name_lookup;
void
register_special_direction(const char * name)
{
struct locale * lang;
char * str = strdup(name);
for (lang=locales;lang;lang=nextlocale(lang)) {
tnode * tokens = get_translations(lang, UT_SPECDIR);
const xmlChar * token = LOC(lang, name);
if (token) {
variant var;
var.v = str;
addtoken(tokens, token, var);
if (lang==default_locale) {
dir_lookup * dl = malloc(sizeof(dir_lookup));
dl->name = str;
dl->oldname = token;
dl->next = dir_name_lookup;
dir_name_lookup = dl;
}
} else {
log_error(("no translation for spec_direction '%s' in locale '%s'\n",
name, locale_name(lang)));
}
}
}
2007-02-11 23:33:35 +01:00
static int
2001-01-25 10:37:55 +01:00
a_readdirection(attrib *a, FILE *f)
{
2007-06-20 02:34:02 +02:00
char lbuf[16];
spec_direction *d = (spec_direction *)(a->data.v);
fscanf(f, "%hd %hd %d", &d->x, &d->y, &d->duration);
2007-06-20 02:34:02 +02:00
fscanf(f, "%15s ", lbuf);
d->desc = strdup(lbuf);
2007-06-20 02:34:02 +02:00
fscanf(f, "%15s ", lbuf);
if (global.data_version<UNICODE_VERSION) {
dir_lookup * dl = dir_name_lookup;
cstring_i(d->desc);
cstring_i(lbuf);
for (;dl;dl=dl->next) {
if (strcmp(lbuf, (const char *)dl->oldname)==0) {
d->keyword=strdup(dl->name);
break;
}
}
if (dl==NULL) {
log_error(("unknown spec_direction '%s'\n", lbuf));
assert(!"not implemented");
}
}
else {
d->keyword = strdup(lbuf);
}
2007-06-20 02:34:02 +02:00
d->active = true;
return AT_READ_OK;
2001-01-25 10:37:55 +01:00
}
2007-02-11 23:33:35 +01:00
static void
2001-01-25 10:37:55 +01:00
a_writedirection(const attrib *a, FILE *f)
{
spec_direction *d = (spec_direction *)(a->data.v);
2001-01-25 10:37:55 +01:00
fprintf(f, "%d %d %d %s %s ", d->x, d->y, d->duration,
d->desc, d->keyword);
2001-01-25 10:37:55 +01:00
}
attrib_type at_direction = {
"direction",
a_initdirection,
2007-02-11 23:33:35 +01:00
a_freedirection,
a_agedirection,
a_writedirection,
a_readdirection
2001-01-25 10:37:55 +01:00
};
region *
find_special_direction(const region *r, const xmlChar *token, const struct locale * lang)
{
attrib *a;
spec_direction *d;
if (xstrlen(token)==0) return NULL;
for (a = a_find(r->attribs, &at_direction);a && a->type==&at_direction;a=a->next) {
d = (spec_direction *)(a->data.v);
if (d->active) {
tnode * tokens = get_translations(lang, UT_SPECDIR);
variant var;
if (findtoken(tokens, token, &var)==E_TOK_SUCCESS) {
if (strcmp((const char *)var.v, d->keyword)==0) {
return findregion(d->x, d->y);
}
}
}
}
return NULL;
}
2001-01-25 10:37:55 +01:00
attrib *
create_special_direction(region *r, region * rt, int duration,
const char *desc, const char *keyword)
{
attrib *a = a_add(&r->attribs, a_new(&at_direction));
spec_direction *d = (spec_direction *)(a->data.v);
d->active = false;
d->x = rt->x;
d->y = rt->y;
d->duration = duration;
d->desc = strdup(desc);
d->keyword = strdup(keyword);
return a;
}
2001-01-25 10:37:55 +01:00
/* Moveblock wird zur Zeit nicht <20>ber Attribute, sondern ein Bitfeld
r->moveblock gemacht. Sollte umgestellt werden, wenn kompliziertere
Dinge gefragt werden. */
2001-01-25 10:37:55 +01:00
/********************/
/* at_moveblock */
/********************/
void
a_initmoveblock(attrib *a)
{
a->data.v = calloc(1, sizeof(moveblock));
2001-01-25 10:37:55 +01:00
}
int
a_readmoveblock(attrib *a, FILE *f)
{
moveblock *m = (moveblock *)(a->data.v);
int i;
2001-01-25 10:37:55 +01:00
fscanf(f, "%d", &i);
m->dir = (direction_t)i;
return AT_READ_OK;
2001-01-25 10:37:55 +01:00
}
void
a_writemoveblock(const attrib *a, FILE *f)
{
moveblock *m = (moveblock *)(a->data.v);
fprintf(f, "%d ", (int)m->dir);
2001-01-25 10:37:55 +01:00
}
attrib_type at_moveblock = {
"moveblock", a_initmoveblock, NULL, NULL, a_writemoveblock, a_readmoveblock
2001-01-25 10:37:55 +01:00
};
#define RMAXHASH 65521
2001-01-25 10:37:55 +01:00
region *regionhash[RMAXHASH];
static region *
rfindhash(short x, short y)
2001-01-25 10:37:55 +01:00
{
region *old;
for (old = regionhash[coor_hashkey(x, y) % RMAXHASH]; old; old = old->nexthash)
if (old->x == x && old->y == y)
return old;
return 0;
2001-01-25 10:37:55 +01:00
}
void
rhash(region * r)
{
int key = coor_hashkey(r->x, r->y) % RMAXHASH;
region *old = regionhash[key];
regionhash[key] = r;
r->nexthash = old;
2001-01-25 10:37:55 +01:00
}
void
runhash(region * r)
{
region **show;
#ifdef FAST_CONNECT
int d, di;
for (d=0,di=MAXDIRECTIONS/2;d!=MAXDIRECTIONS;++d,++di) {
region * rc = r->connect[d];
if (rc!=NULL) {
if (di>=MAXDIRECTIONS) di-=MAXDIRECTIONS;
rc->connect[di] = NULL;
r->connect[d] = NULL;
}
}
#endif
for (show = &regionhash[coor_hashkey(r->x, r->y) % RMAXHASH]; *show; show = &(*show)->nexthash) {
if ((*show)->x == r->x && (*show)->y == r->y) {
break;
}
}
if (*show) {
assert(*show == r);
*show = (*show)->nexthash;
r->nexthash = 0;
}
2001-01-25 10:37:55 +01:00
}
region *
r_connect(const region * r, direction_t dir)
{
static int set = 0;
static region * buffer[MAXDIRECTIONS];
static const region * last = NULL;
#ifdef FAST_CONNECT
region * rmodify = (region*)r;
assert (dir>=0 && dir<MAXDIRECTIONS);
if (r->connect[dir]) return r->connect[dir];
#endif
assert(dir<MAXDIRECTIONS);
if (r != last) {
set = 0;
last = r;
}
else
if (set & (1 << dir)) return buffer[dir];
buffer[dir] = rfindhash(r->x + delta_x[dir], r->y + delta_y[dir]);
set |= (1<<dir);
#ifdef FAST_CONNECT
rmodify->connect[dir] = buffer[dir];
#endif
return buffer[dir];
2001-01-25 10:37:55 +01:00
}
region *
findregion(short x, short y)
2001-01-25 10:37:55 +01:00
{
return rfindhash(x, y);
2001-01-25 10:37:55 +01:00
}
int
koor_distance(int x1, int y1, int x2, int y2)
{
/* Contributed by Hubert Mackenberg. Thanks.
* x und y Abstand zwischen x1 und x2 berechnen
*/
int dx = x1 - x2;
int dy = y1 - y2;
/* Bei negativem dy am Ursprung spiegeln, das veraendert
* den Abstand nicht
*/
if ( dy < 0 )
{
dy = -dy;
dx = -dx;
}
/*
* dy ist jetzt >=0, fuer dx sind 3 Faelle zu untescheiden
*/
if ( dx >= 0 ) return dx + dy;
else if (-dx >= dy) return -dx;
else return dy;
2001-01-25 10:37:55 +01:00
}
int
distance(const region * r1, const region * r2)
{
return koor_distance(r1->x, r1->y, r2->x, r2->y);
2001-01-25 10:37:55 +01:00
}
static direction_t
2001-01-25 10:37:55 +01:00
koor_reldirection(int ax, int ay, int bx, int by)
{
direction_t dir;
for (dir=0;dir!=MAXDIRECTIONS;++dir) {
if (bx-ax == delta_x[dir] && by-ay == delta_y[dir]) return dir;
}
return NODIRECTION;
}
spec_direction *
special_direction(const region * from, const region * to)
{
const attrib *a = a_findc(from->attribs, &at_direction);
while (a!=NULL && a->type==&at_direction) {
spec_direction * sd = (spec_direction *)a->data.v;
if (sd->x==to->x && sd->y==to->y) return sd;
a = a->next;
}
return NULL;
2001-01-25 10:37:55 +01:00
}
direction_t
reldirection(const region * from, const region * to)
2001-01-25 10:37:55 +01:00
{
direction_t dir = koor_reldirection(from->x, from->y, to->x, to->y);
if (dir==NODIRECTION) {
spec_direction *sd = special_direction(from, to);
if (sd!=NULL && sd->active) return D_SPECIAL;
}
return dir;
2001-01-25 10:37:55 +01:00
}
void
free_regionlist(region_list *rl)
2001-01-25 10:37:55 +01:00
{
while (rl) {
region_list * rl2 = rl->next;
free(rl);
rl = rl2;
}
2001-01-25 10:37:55 +01:00
}
void
add_regionlist(region_list **rl, region *r)
2001-01-25 10:37:55 +01:00
{
region_list *rl2 = (region_list*)malloc(sizeof(region_list));
2001-01-25 10:37:55 +01:00
rl2->data = r;
rl2->next = *rl;
2001-01-25 10:37:55 +01:00
*rl = rl2;
2001-01-25 10:37:55 +01:00
}
/********************/
/* at_horseluck */
/********************/
attrib_type at_horseluck = {
"horseluck",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ,
ATF_UNIQUE
2001-01-25 10:37:55 +01:00
};
/**********************/
/* at_peasantluck */
/**********************/
attrib_type at_peasantluck = {
"peasantluck",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ,
ATF_UNIQUE
2001-01-25 10:37:55 +01:00
};
/*********************/
/* at_chaoscount */
/*********************/
attrib_type at_chaoscount = {
"chaoscount",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
a_writeint,
a_readint,
ATF_UNIQUE
2001-01-25 10:37:55 +01:00
};
/*********************/
/* at_deathcount */
/*********************/
attrib_type at_deathcount = {
"deathcount",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
a_writeint,
a_readint,
ATF_UNIQUE
2001-01-25 10:37:55 +01:00
};
/*********************/
/* at_woodcount */
/*********************/
attrib_type at_woodcount = {
"woodcount",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
a_readint,
ATF_UNIQUE
2001-01-25 10:37:55 +01:00
};
/*********************/
/* at_travelunit */
/*********************/
attrib_type at_travelunit = {
"travelunit",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ
2001-01-25 10:37:55 +01:00
};
extern int laen_read(attrib * a, FILE * F);
2001-01-25 10:37:55 +01:00
/***************/
/* at_laen */
/***************/
attrib_type at_laen = {
"laen",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
2007-06-20 02:34:02 +02:00
NULL,
laen_read,
ATF_UNIQUE
2001-01-25 10:37:55 +01:00
};
void
rsetlaen(region * r, int val)
{
attrib * a = a_find(r->attribs, &at_laen);
if (!a && val>=0) a = a_add(&r->attribs, a_new(&at_laen));
else if (a && val<0) a_remove(&r->attribs, a);
if (val>=0) a->data.i = val;
2001-01-25 10:37:55 +01:00
}
int
rlaen(const region * r)
{
attrib * a = a_find(r->attribs, &at_laen);
if (!a) return -1;
return a->data.i;
2001-01-25 10:37:55 +01:00
}
void
2005-06-12 11:48:19 +02:00
rsetroad(region * r, direction_t d, short val)
2001-01-25 10:37:55 +01:00
{
border * b;
region * r2 = rconnect(r, d);
if (!r2) return;
b = get_borders(r, r2);
while (b && b->type!=&bt_road) b = b->next;
if (!b) b = new_border(&bt_road, r, r2);
if (r==b->from) b->data.sa[0] = val;
else b->data.sa[1] = val;
2001-01-25 10:37:55 +01:00
}
2005-06-12 11:48:19 +02:00
short
2001-01-25 10:37:55 +01:00
rroad(const region * r, direction_t d)
{
int rval;
border * b;
region * r2 = rconnect(r, d);
if (!r2) return 0;
b = get_borders(r, r2);
while (b && b->type!=&bt_road) b = b->next;
if (!b) return 0;
rval = b->data.i;
2005-06-12 11:48:19 +02:00
if (r==b->from) return b->data.sa[0];
return b->data.sa[1];
2001-01-25 10:37:55 +01:00
}
boolean
r_isforest(const region * r)
{
if (fval(r->terrain, FOREST_REGION)) {
/* needs to be covered with at leas 48% trees */
int mincover = (int)(r->terrain->size * 0.48);
int trees = rtrees(r, 2) + rtrees(r, 1);
return (trees*TREESIZE >= mincover);
}
return false;
}
2001-01-25 10:37:55 +01:00
int
is_coastregion(region *r)
{
direction_t i;
int res = 0;
2001-01-25 10:37:55 +01:00
for(i=0;i<MAXDIRECTIONS;i++) {
region * rn = rconnect(r,i);
if (rn && fval(rn->terrain, SEA_REGION)) res++;
}
return res;
2001-01-25 10:37:55 +01:00
}
int
rpeasants(const region * r)
{
return ((r)->land?(r)->land->peasants:0);
2001-01-25 10:37:55 +01:00
}
void
rsetpeasants(region * r, int value)
{
((r)->land?((r)->land->peasants=(value)):(assert((value)>=0), (value)),0);
2001-01-25 10:37:55 +01:00
}
int
rmoney(const region * r)
{
return ((r)->land?(r)->land->money:0);
2001-01-25 10:37:55 +01:00
}
void
rsethorses(const region *r, int value)
{
assert(value >= 0);
if(r->land)
r->land->horses = value;
}
int
rhorses(const region *r)
{
return r->land?r->land->horses:0;
}
2001-01-25 10:37:55 +01:00
void
rsetmoney(region * r, int value)
{
((r)->land?((r)->land->money=(value)):(assert((value)>=0), (value)),0);
2001-01-25 10:37:55 +01:00
}
void
r_setdemand(region * r, const luxury_type * ltype, int value)
{
struct demand * d, ** dp = &r->land->demands;
if (ltype==NULL) return;
while (*dp && (*dp)->type != ltype) dp = &(*dp)->next;
d = *dp;
if (!d) {
d = *dp = calloc(sizeof(struct demand), 1);
d->type = ltype;
}
d->value = value;
2001-01-25 10:37:55 +01:00
}
int
r_demand(const region * r, const luxury_type * ltype)
{
struct demand * d = r->land->demands;
while (d && d->type != ltype) d = d->next;
if (!d) return -1;
return d->value;
2001-01-25 10:37:55 +01:00
}
const xmlChar *
2002-09-02 22:36:12 +02:00
rname(const region * r, const struct locale * lang) {
if (r->land) {
return r->land->name;
}
return LOC(lang, terrain_name(r));
2001-01-25 10:37:55 +01:00
}
int
rtrees(const region *r, int ageclass)
{
return ((r)->land?(r)->land->trees[ageclass]:0);
}
int
rsettrees(const region *r, int ageclass, int value)
{
if (!r->land) assert(value==0);
else {
assert(value>=0);
return r->land->trees[ageclass]=value;
}
return 0;
}
2001-01-25 10:37:55 +01:00
static region *last;
static unsigned int max_index = 0;
2001-01-25 10:37:55 +01:00
region *
new_region(short x, short y)
2001-01-25 10:37:55 +01:00
{
region *r = rfindhash(x, y);
if (r) {
fprintf(stderr, "\ndoppelte regionen entdeckt: %s(%d,%d)\n", regionname(r, NULL), x, y);
if (r->units)
fprintf(stderr, "doppelte region enth<74>lt einheiten\n");
return r;
}
r = calloc(1, sizeof(region));
r->x = x;
r->y = y;
2006-01-01 21:56:11 +01:00
r->age = 1;
r->planep = findplane(x, y);
rhash(r);
if (last)
addlist(&last, r);
else
addlist(&regions, r);
last = r;
assert(r->next==NULL);
r->index = ++max_index;
return r;
2001-01-25 10:37:55 +01:00
}
static void
freeland(land_region * lr)
{
while (lr->demands) {
struct demand * d = lr->demands;
lr->demands = d->next;
free(d);
}
if (lr->name) free(lr->name);
free(lr);
2001-01-25 10:37:55 +01:00
}
void
free_region(region * r)
{
runhash(r);
if (last == r) last = NULL;
free(r->display);
if (r->land) freeland(r->land);
if (r->msgs) {
free_messagelist(r->msgs);
r->msgs = 0;
}
while (r->individual_messages) {
struct individual_message * msg = r->individual_messages;
r->individual_messages = msg->next;
if (msg->msgs) free_messagelist(msg->msgs);
free(msg);
}
while (r->attribs) a_remove (&r->attribs, r->attribs);
while (r->resources) {
rawmaterial * res = r->resources;
r->resources = res->next;
free(res);
}
2005-06-11 10:09:55 +02:00
while (r->donations) {
donation * don = r->donations;
r->donations = don->next;
free(don);
}
free(r);
}
/** creates a name for a region
* TODO: Make this XML-configurable and allow non-ascii characters again.
*/
static xmlChar *
2001-01-25 10:37:55 +01:00
makename(void)
{
int s, v, k, e, p = 0, x = 0;
size_t nk, ne, nv, ns;
static char name[16];
const char *kons = "bcdfghklmnprstvwz",
*end = "nlrdst",
#if 0
*vowels = "aaaaaaaaa<EFBFBD><EFBFBD>eeeeeeeee<EFBFBD><EFBFBD><EFBFBD>iiiiiiiii<EFBFBD><EFBFBD>ooooooooo<EFBFBD><EFBFBD><EFBFBD>uuuuuuuuu<EFBFBD>yy",
#else
*vowels = "aaaaaaaaaaaeeeeeeeeeeeeiiiiiiiiiiioooooooooooouuuuuuuuuuyy",
#endif
*start = "bcdgtskpvfr";
nk = strlen(kons);
ne = strlen(end);
nv = strlen(vowels);
ns = strlen(start);
for (s = rng_int() % 3 + 2; s > 0; s--) {
if (x > 0) {
k = rng_int() % (int)nk;
name[p] = kons[k];
p++;
} else {
k = rng_int() % (int)ns;
name[p] = start[k];
p++;
}
v = rng_int() % (int)nv;
name[p] = vowels[v];
p++;
if (rng_int() % 3 == 2 || s == 1) {
e = rng_int() % (int)ne;
name[p] = end[e];
p++;
x = 1;
} else
x = 0;
}
name[p] = '\0';
name[0] = (char) toupper(name[0]);
return (xmlChar*)name;
2001-01-25 10:37:55 +01:00
}
void
setluxuries(region * r, const luxury_type * sale)
{
const luxury_type * ltype;
2001-02-11 09:55:48 +01:00
assert(r->land);
2001-02-11 09:55:48 +01:00
if(r->land->demands) freelist(r->land->demands);
2001-02-11 09:55:48 +01:00
for (ltype=luxurytypes; ltype; ltype=ltype->next) {
struct demand * dmd = calloc(sizeof(struct demand), 1);
dmd->type = ltype;
if (ltype!=sale) dmd->value = 1 + rng_int() % 5;
dmd->next = r->land->demands;
r->land->demands = dmd;
}
}
2001-01-25 10:37:55 +01:00
void
terraform(region * r, terrain_t t)
{
terraform_region(r, newterrain(t));
}
void
terraform_region(region * r, const terrain_type * terrain)
{
/* Resourcen, die nicht mehr vorkommen k<>nnen, l<>schen */
const terrain_type * oldterrain = r->terrain;
rawmaterial **lrm = &r->resources;
while (*lrm) {
rawmaterial *rm = *lrm;
const resource_type * rtype = NULL;
if (terrain->production!=NULL) {
int i;
for (i=0;terrain->production[i].type;++i) {
if (rm->type->rtype == terrain->production[i].type) {
rtype = rm->type->rtype;
break;
}
}
}
if (rtype==NULL) {
*lrm = rm->next;
free(rm);
} else {
lrm = &rm->next;
}
}
r->terrain = terrain;
terraform_resources(r);
if (!fval(terrain, LAND_REGION)) {
if (r->land!=NULL) {
i_freeall(&r->land->items);
freeland(r->land);
r->land = NULL;
}
rsettrees(r, 0, 0);
rsettrees(r, 1, 0);
rsettrees(r, 2, 0);
rsethorses(r, 0);
rsetpeasants(r, 0);
rsetmoney(r, 0);
freset(r, RF_ENCOUNTER);
freset(r, RF_MALLORN);
/* Beschreibung und Namen l<>schen */
return;
}
if (r->land) {
i_freeall(&r->land->items);
} else {
static struct surround {
struct surround * next;
const luxury_type * type;
int value;
} *trash =NULL, *nb = NULL;
const luxury_type * ltype = NULL;
direction_t d;
int mnr = 0;
r->land = calloc(1, sizeof(land_region));
rsetname(r, makename());
for (d=0;d!=MAXDIRECTIONS;++d) {
region * nr = rconnect(r, d);
if (nr && nr->land) {
struct demand * sale = r->land->demands;
while (sale && sale->value!=0) sale=sale->next;
if (sale) {
struct surround * sr = nb;
while (sr && sr->type!=sale->type) sr=sr->next;
if (!sr) {
if (trash) {
sr = trash;
trash = trash->next;
} else {
sr = calloc(1, sizeof(struct surround));
}
sr->next = nb;
sr->type = sale->type;
sr->value = 1;
nb = sr;
} else sr->value++;
++mnr;
}
}
}
if (!nb) {
int i = get_maxluxuries();
2005-11-26 01:08:47 +01:00
if (i>0) {
i = rng_int() % i;
2005-11-26 01:08:47 +01:00
ltype = luxurytypes;
while (i--) ltype=ltype->next;
2005-11-26 01:08:47 +01:00
}
} else {
int i = rng_int() % mnr;
struct surround * srd = nb;
while (i>srd->value) {
i-=srd->value;
srd=srd->next;
}
if (srd->type) setluxuries(r, srd->type);
while (srd->next!=NULL) srd=srd->next;
srd->next=trash;
trash = nb;
nb = NULL;
}
}
if (fval(terrain, LAND_REGION)) {
const item_type * itype = NULL;
char equip_hash[64];
/* TODO: put the equipment in struct terrain, faster */
sprintf(equip_hash, "terrain_%s", terrain->_name);
equip_items(&r->land->items, get_equipment(equip_hash));
if (r->terrain->herbs) {
int len=0;
while (r->terrain->herbs[len]) ++len;
if (len) itype = r->terrain->herbs[rng_int()%len];
}
if (itype!=NULL) {
rsetherbtype(r, itype);
rsetherbs(r, (short)(50+rng_int()%31));
}
else {
rsetherbtype(r, NULL);
}
2005-11-14 19:58:09 +01:00
if (oldterrain==NULL || !fval(oldterrain, LAND_REGION)) {
if (rng_int() % 100 < 3) fset(r, RF_MALLORN);
else freset(r, RF_MALLORN);
if (rng_int() % 100 < ENCCHANCE) {
fset(r, RF_ENCOUNTER);
}
}
}
2001-01-25 10:37:55 +01:00
2005-11-14 19:58:09 +01:00
if (oldterrain==NULL || terrain->size!=oldterrain->size) {
if (terrain==newterrain(T_PLAIN)) {
rsethorses(r, rng_int() % (terrain->size / 50));
if(rng_int()%100 < 40) {
rsettrees(r, 2, terrain->size * (30+rng_int()%40)/1000);
rsettrees(r, 1, rtrees(r, 2)/4);
rsettrees(r, 0, rtrees(r, 2)/2);
}
} else if (chance(0.2)) {
rsettrees(r, 2, terrain->size * (30 + rng_int() % 40) / 1000);
rsettrees(r, 1, rtrees(r, 2)/4);
rsettrees(r, 0, rtrees(r, 2)/2);
}
if (!fval(r, RF_CHAOTIC)) {
int peasants;
peasants = (maxworkingpeasants(r) * (20+dice_rand("6d10")))/100;
rsetpeasants(r, max(100, peasants));
rsetmoney(r, rpeasants(r) * ((wage(r, NULL, NULL)+1) + rng_int() % 5));
}
}
2001-01-25 10:37:55 +01:00
}
/** ENNO:
* ich denke, das das hier nicht sein sollte.
* statt dessen sollte ein attribut an der region sein, das das erledigt,
* egal ob durch den spell oder anderes angelegt.
**/
#include "curse.h"
int
production(const region *r)
{
/* mu<6D> rterrain(r) sein, nicht rterrain() wegen rekursion */
int p = r->terrain->size / MAXPEASANTS_PER_AREA;
if (curse_active(get_curse(r->attribs, ct_find("drought")))) p /= 2;
2001-01-25 10:37:55 +01:00
return p;
2001-01-25 10:37:55 +01:00
}
int
2001-01-25 10:37:55 +01:00
read_region_reference(region ** r, FILE * F)
{
variant coor;
fscanf(F, "%hd %hd", &coor.sa[0], &coor.sa[1]);
if (coor.sa[0]==SHRT_MAX) {
*r = NULL;
return AT_READ_FAIL;
}
*r = findregion(coor.sa[0], coor.sa[1]);
if (*r==NULL) {
ur_add(coor, (void**)r, resolve_region);
}
return AT_READ_OK;
2001-01-25 10:37:55 +01:00
}
void
write_region_reference(const region * r, FILE * F)
{
if (r) fprintf(F, "%d %d ", r->x, r->y);
else fprintf(F, "%d %d ", SHRT_MAX, SHRT_MAX);
2001-01-25 10:37:55 +01:00
}
void *
resolve_region(variant id) {
return findregion(id.sa[0], id.sa[1]);
2001-01-25 10:37:55 +01:00
}
struct message_list *
r_getmessages(const struct region * r, const struct faction * viewer)
{
struct individual_message * imsg = r->individual_messages;
while (imsg && (imsg)->viewer!=viewer) imsg = imsg->next;
if (imsg) return imsg->msgs;
return NULL;
}
struct message *
r_addmessage(struct region * r, const struct faction * viewer, struct message * msg)
{
assert(r);
if (viewer) {
struct individual_message * imsg;
imsg = r->individual_messages;
while (imsg && imsg->viewer!=viewer) imsg = imsg->next;
if (imsg==NULL) {
imsg = malloc(sizeof(struct individual_message));
imsg->next = r->individual_messages;
imsg->msgs = NULL;
r->individual_messages = imsg;
imsg->viewer = viewer;
}
return add_message(&imsg->msgs, msg);
}
return add_message(&r->msgs, msg);
2001-05-27 16:02:12 +02:00
}
struct faction *
region_owner(const struct region * r)
{
#ifdef REGIONOWNERS
return r->owner;
#else
return NULL;
#endif
}
void
region_setowner(struct region * r, struct faction * owner)
{
#ifdef REGIONOWNERS
r->owner = owner;
#else
2005-11-20 15:50:15 +01:00
unused(r);
unused(owner);
#endif
}