2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2001-04-14 13:39:14 +02:00
|
|
|
|
*
|
2003-07-29 11:48:03 +02:00
|
|
|
|
* 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)
|
2007-09-02 20:11:17 +02:00
|
|
|
|
* Enno Rehling (enno@eressea.de)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
|
*
|
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include "eressea.h"
|
|
|
|
|
#include "border.h"
|
|
|
|
|
|
|
|
|
|
#include "region.h"
|
2006-02-10 20:48:43 +01:00
|
|
|
|
#include "save.h"
|
|
|
|
|
#include "terrain.h"
|
|
|
|
|
#include "unit.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include <util/attrib.h>
|
2007-08-05 14:19:56 +02:00
|
|
|
|
#include <util/log.h>
|
|
|
|
|
#include <util/rng.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <assert.h>
|
2005-06-10 00:10:35 +02:00
|
|
|
|
#include <limits.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include <string.h>
|
|
|
|
|
|
2005-01-03 22:28:57 +01:00
|
|
|
|
extern boolean incomplete_data;
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unsigned int nextborder = 0;
|
|
|
|
|
|
|
|
|
|
border * borders[BMAXHASH];
|
|
|
|
|
border_type * bordertypes;
|
|
|
|
|
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
void
|
|
|
|
|
free_borders(void)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
for (i=0;i!=BMAXHASH;++i) {
|
|
|
|
|
borders[i] = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
border *
|
|
|
|
|
find_border(unsigned int id)
|
|
|
|
|
{
|
2004-05-31 13:50:32 +02:00
|
|
|
|
int key;
|
|
|
|
|
for (key=0;key!=BMAXHASH;key++) {
|
|
|
|
|
border * bhash;
|
|
|
|
|
for (bhash=borders[key];bhash!=NULL;bhash=bhash->nexthash) {
|
|
|
|
|
border * b;
|
|
|
|
|
for (b=bhash;b;b=b->next) {
|
|
|
|
|
if (b->id==id) return b;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void *
|
2005-06-10 00:10:35 +02:00
|
|
|
|
resolve_borderid(variant id) {
|
|
|
|
|
return (void*)find_border(id.i);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2004-05-22 23:01:13 +02:00
|
|
|
|
static border **
|
|
|
|
|
get_borders_i(const region * r1, const region * r2)
|
|
|
|
|
{
|
|
|
|
|
border ** bp;
|
2005-06-10 00:10:35 +02:00
|
|
|
|
int key = reg_hashkey(r1);
|
|
|
|
|
int k2 = reg_hashkey(r2);
|
2004-05-22 23:01:13 +02:00
|
|
|
|
|
|
|
|
|
key = min(k2, key) % BMAXHASH;
|
|
|
|
|
bp = &borders[key];
|
|
|
|
|
while (*bp) {
|
|
|
|
|
border * b = *bp;
|
|
|
|
|
if ((b->from==r1 && b->to==r2) || (b->from==r2 && b->to==r1)) break;
|
|
|
|
|
bp = &b->nexthash;
|
|
|
|
|
}
|
|
|
|
|
return bp;
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
border *
|
|
|
|
|
get_borders(const region * r1, const region * r2)
|
|
|
|
|
{
|
2004-05-22 23:01:13 +02:00
|
|
|
|
border ** bp = get_borders_i(r1, r2);
|
|
|
|
|
return *bp;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border *
|
|
|
|
|
new_border(border_type * type, region * from, region * to)
|
|
|
|
|
{
|
2004-05-22 23:01:13 +02:00
|
|
|
|
border ** bp = get_borders_i(from, to);
|
|
|
|
|
border * b = calloc(1, sizeof(struct border));
|
|
|
|
|
|
2004-05-23 00:55:02 +02:00
|
|
|
|
while (*bp) bp = &(*bp)->next;
|
|
|
|
|
|
|
|
|
|
*bp = b;
|
2004-05-22 23:01:13 +02:00
|
|
|
|
b->type = type;
|
|
|
|
|
b->from = from;
|
|
|
|
|
b->to = to;
|
|
|
|
|
b->id = ++nextborder;
|
2004-02-07 13:10:24 +01:00
|
|
|
|
|
|
|
|
|
if (type->init) type->init(b);
|
2004-05-22 23:01:13 +02:00
|
|
|
|
return b;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
erase_border(border * b)
|
|
|
|
|
{
|
2004-05-22 23:01:13 +02:00
|
|
|
|
border ** bp = get_borders_i(b->from, b->to);
|
|
|
|
|
attrib ** ap = &b->attribs;
|
|
|
|
|
|
2004-06-12 12:16:48 +02:00
|
|
|
|
while (*ap) a_remove(&b->attribs, *ap);
|
2004-05-22 23:01:13 +02:00
|
|
|
|
|
|
|
|
|
assert(*bp!=NULL || !"error: border is not registered");
|
2004-05-23 00:55:02 +02:00
|
|
|
|
if (*bp==b) {
|
|
|
|
|
/* it is the first in the list, so it is in the nexthash list */
|
|
|
|
|
if (b->next) {
|
|
|
|
|
*bp = b->next;
|
|
|
|
|
(*bp)->nexthash = b->nexthash;
|
|
|
|
|
} else {
|
|
|
|
|
*bp = b->nexthash;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
while (*bp && *bp != b) {
|
|
|
|
|
bp = &(*bp)->next;
|
|
|
|
|
}
|
|
|
|
|
assert(*bp==b || !"error: border is not registered");
|
|
|
|
|
*bp = b->next;
|
|
|
|
|
}
|
2004-05-22 23:01:13 +02:00
|
|
|
|
if (b->type->destroy) b->type->destroy(b);
|
|
|
|
|
free(b);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
register_bordertype(border_type * type)
|
|
|
|
|
{
|
|
|
|
|
border_type ** btp = &bordertypes;
|
|
|
|
|
|
|
|
|
|
while (*btp && *btp!=type) btp = &(*btp)->next;
|
|
|
|
|
if (*btp) return;
|
|
|
|
|
*btp = type;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
border_type *
|
|
|
|
|
find_bordertype(const char * name)
|
|
|
|
|
{
|
2006-02-09 23:20:28 +01:00
|
|
|
|
border_type * bt = bordertypes;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-02-09 23:20:28 +01:00
|
|
|
|
while (bt && strcmp(bt->__name, name)) bt = bt->next;
|
|
|
|
|
return bt;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
b_read(border * b, FILE *f)
|
|
|
|
|
{
|
2005-06-10 00:10:35 +02:00
|
|
|
|
switch (b->type->datatype) {
|
|
|
|
|
case VAR_NONE:
|
|
|
|
|
case VAR_INT:
|
|
|
|
|
fscanf(f, "%x ", &b->data.i);
|
|
|
|
|
break;
|
|
|
|
|
case VAR_VOIDPTR:
|
|
|
|
|
fscanf(f, "%p ", &b->data.v);
|
|
|
|
|
break;
|
|
|
|
|
case VAR_SHORTA:
|
|
|
|
|
fscanf(f, "%hd %hd ", &b->data.sa[0], &b->data.sa[1]);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
assert(!"unhandled variant type in border");
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
b_write(const border * b, FILE *f)
|
|
|
|
|
{
|
2005-06-10 00:10:35 +02:00
|
|
|
|
switch (b->type->datatype) {
|
|
|
|
|
case VAR_NONE:
|
|
|
|
|
case VAR_INT:
|
|
|
|
|
fprintf(f, "%x ", b->data.i);
|
|
|
|
|
break;
|
|
|
|
|
case VAR_VOIDPTR:
|
|
|
|
|
fprintf(f, "%p ", b->data.v);
|
|
|
|
|
break;
|
|
|
|
|
case VAR_SHORTA:
|
|
|
|
|
fprintf(f, "%d %d ", b->data.sa[0], b->data.sa[1]);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
assert(!"unhandled variant type in border");
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean b_transparent(const border * b, const struct faction * f) { unused(b); unused(f); return true; }
|
|
|
|
|
boolean b_opaque(const border * b, const struct faction * f) { unused(b); unused(f); return false; }
|
|
|
|
|
boolean b_blockall(const border * b, const unit * u, const region * r) { unused(u); unused(r); unused(b); return true; }
|
|
|
|
|
boolean b_blocknone(const border * b, const unit * u, const region * r) { unused(u); unused(r); unused(b); return false; }
|
|
|
|
|
boolean b_rvisible(const border * b, const region * r) { return (boolean)(b->to==r || b->from==r); }
|
|
|
|
|
boolean b_fvisible(const border * b, const struct faction * f, const region * r) { unused(r); unused(f); unused(b); return true; }
|
|
|
|
|
boolean b_uvisible(const border * b, const unit * u) { unused(u); unused(b); return true; }
|
|
|
|
|
boolean b_rinvisible(const border * b, const region * r) { unused(r); unused(b); return false; }
|
|
|
|
|
boolean b_finvisible(const border * b, const struct faction * f, const region * r) { unused(r); unused(f); unused(b); return false; }
|
|
|
|
|
boolean b_uinvisible(const border * b, const unit * u) { unused(u); unused(b); return false; }
|
|
|
|
|
|
|
|
|
|
/*********************/
|
|
|
|
|
/* at_countdown */
|
|
|
|
|
/*********************/
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
a_agecountdown(attrib * a)
|
|
|
|
|
{
|
|
|
|
|
a->data.i = max(a->data.i-1, 0);
|
|
|
|
|
return a->data.i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrib_type at_countdown = {
|
|
|
|
|
"countdown",
|
|
|
|
|
DEFAULT_INIT,
|
|
|
|
|
DEFAULT_FINALIZE,
|
|
|
|
|
a_agecountdown,
|
2006-02-10 20:48:43 +01:00
|
|
|
|
a_writeint,
|
|
|
|
|
a_readint
|
2001-01-25 10:37:55 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
age_borders(void)
|
|
|
|
|
{
|
2004-05-31 13:50:32 +02:00
|
|
|
|
border_list * deleted = NULL;
|
|
|
|
|
int i;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-05-31 13:50:32 +02:00
|
|
|
|
for (i=0;i!=BMAXHASH;++i) {
|
|
|
|
|
border * bhash = borders[i];
|
|
|
|
|
for (;bhash;bhash=bhash->nexthash) {
|
|
|
|
|
border * b = bhash;
|
|
|
|
|
for (;b;b=b->next) {
|
|
|
|
|
attrib ** ap = &b->attribs;
|
|
|
|
|
while (*ap) {
|
|
|
|
|
attrib * a = *ap;
|
|
|
|
|
if (a->type->age && a->type->age(a)==0) {
|
|
|
|
|
if (a->type == &at_countdown) {
|
|
|
|
|
border_list * bnew = malloc(sizeof(border_list));
|
|
|
|
|
bnew->next = deleted;
|
|
|
|
|
bnew->data = b;
|
|
|
|
|
deleted = bnew;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2004-06-12 12:16:48 +02:00
|
|
|
|
a_remove(&b->attribs, a);
|
2004-05-31 13:50:32 +02:00
|
|
|
|
}
|
|
|
|
|
else ap=&a->next;
|
|
|
|
|
}
|
2004-02-07 13:10:24 +01:00
|
|
|
|
}
|
2004-05-31 13:50:32 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (deleted) {
|
2004-07-04 13:16:43 +02:00
|
|
|
|
border_list * blist = deleted->next;
|
|
|
|
|
border * b = deleted->data;
|
2004-05-31 13:50:32 +02:00
|
|
|
|
erase_border(b);
|
|
|
|
|
free(deleted);
|
2004-07-04 13:16:43 +02:00
|
|
|
|
deleted = blist;
|
2004-05-31 13:50:32 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/********
|
|
|
|
|
* implementation of a couple of borders. this shouldn't really be in here, so
|
|
|
|
|
* let's keep it separate from the more general stuff above
|
|
|
|
|
********/
|
|
|
|
|
|
|
|
|
|
#include "faction.h"
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
b_namewall(const border * b, const region * r, const struct faction * f, int gflags)
|
|
|
|
|
{
|
|
|
|
|
unused(f);
|
|
|
|
|
unused(r);
|
|
|
|
|
unused(b);
|
|
|
|
|
if (gflags & GF_ARTICLE) return "eine Wand";
|
|
|
|
|
return "Wand";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_wall = {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
"wall", VAR_INT,
|
2001-01-25 10:37:55 +01:00
|
|
|
|
b_opaque,
|
|
|
|
|
NULL, /* init */
|
|
|
|
|
NULL, /* destroy */
|
|
|
|
|
b_read, /* read */
|
|
|
|
|
b_write, /* write */
|
|
|
|
|
b_blockall, /* block */
|
|
|
|
|
b_namewall, /* name */
|
|
|
|
|
b_rvisible, /* rvisible */
|
|
|
|
|
b_fvisible, /* fvisible */
|
|
|
|
|
b_uvisible, /* uvisible */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
border_type bt_noway = {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
"noway", VAR_INT,
|
2001-01-25 10:37:55 +01:00
|
|
|
|
b_transparent,
|
|
|
|
|
NULL, /* init */
|
|
|
|
|
NULL, /* destroy */
|
|
|
|
|
b_read, /* read */
|
|
|
|
|
b_write, /* write */
|
|
|
|
|
b_blockall, /* block */
|
|
|
|
|
NULL, /* name */
|
|
|
|
|
b_rinvisible, /* rvisible */
|
|
|
|
|
b_finvisible, /* fvisible */
|
|
|
|
|
b_uinvisible, /* uvisible */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
b_namefogwall(const border * b, const region * r, const struct faction * f, int gflags)
|
|
|
|
|
{
|
|
|
|
|
unused(f);
|
|
|
|
|
unused(b);
|
|
|
|
|
unused(r);
|
|
|
|
|
if (gflags & GF_ARTICLE) return "eine Nebelwand";
|
|
|
|
|
return "Nebelwand";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static boolean
|
|
|
|
|
b_blockfogwall(const border * b, const unit * u, const region * r)
|
|
|
|
|
{
|
|
|
|
|
unused(b);
|
|
|
|
|
unused(r);
|
|
|
|
|
if (!u) return true;
|
|
|
|
|
return (boolean)(effskill(u, SK_OBSERVATION) > 4); /* Das ist die alte Nebelwand */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_fogwall = {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
"fogwall", VAR_INT,
|
2001-01-25 10:37:55 +01:00
|
|
|
|
b_transparent, /* transparent */
|
|
|
|
|
NULL, /* init */
|
|
|
|
|
NULL, /* destroy */
|
|
|
|
|
b_read, /* read */
|
|
|
|
|
b_write, /* write */
|
|
|
|
|
b_blockfogwall, /* block */
|
|
|
|
|
b_namefogwall, /* name */
|
|
|
|
|
b_rvisible, /* rvisible */
|
|
|
|
|
b_fvisible, /* fvisible */
|
|
|
|
|
b_uvisible, /* uvisible */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
b_nameillusionwall(const border * b, const region * r, const struct faction * f, int gflags)
|
|
|
|
|
{
|
2007-09-10 08:51:11 +02:00
|
|
|
|
/* TODO: UNICODE: f->locale bestimmt die Sprache */
|
2005-06-10 00:10:35 +02:00
|
|
|
|
int fno = b->data.i;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unused(b);
|
|
|
|
|
unused(r);
|
2005-06-10 00:10:35 +02:00
|
|
|
|
if (gflags & GF_ARTICLE) return (f && fno==f->subscription)?"eine Illusionswand":"eine Wand";
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return (f && fno==f->no)?"Illusionswand":"Wand";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_illusionwall = {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
"illusionwall", VAR_INT,
|
2001-01-25 10:37:55 +01:00
|
|
|
|
b_opaque,
|
|
|
|
|
NULL, /* init */
|
|
|
|
|
NULL, /* destroy */
|
|
|
|
|
b_read, /* read */
|
|
|
|
|
b_write, /* write */
|
|
|
|
|
b_blocknone, /* block */
|
|
|
|
|
b_nameillusionwall, /* name */
|
|
|
|
|
b_rvisible, /* rvisible */
|
|
|
|
|
b_fvisible, /* fvisible */
|
|
|
|
|
b_uvisible, /* uvisible */
|
|
|
|
|
};
|
|
|
|
|
|
2002-04-27 16:25:35 +02:00
|
|
|
|
/***
|
|
|
|
|
* special quest door
|
|
|
|
|
***/
|
|
|
|
|
|
|
|
|
|
boolean b_blockquestportal(const border * b, const unit * u, const region * r) {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
if(b->data.i > 0) return true;
|
2002-04-27 16:25:35 +02:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
b_namequestportal(const border * b, const region * r, const struct faction * f, int gflags)
|
|
|
|
|
{
|
2007-09-10 08:51:11 +02:00
|
|
|
|
/* TODO: UNICODE: f->locale bestimmt die Sprache */
|
2005-06-10 00:10:35 +02:00
|
|
|
|
int lock = b->data.i;
|
2002-04-27 16:25:35 +02:00
|
|
|
|
unused(b);
|
|
|
|
|
unused(r);
|
|
|
|
|
|
|
|
|
|
if (gflags & GF_ARTICLE) {
|
|
|
|
|
if(lock > 0) {
|
|
|
|
|
return "ein gewaltiges verschlossenes Tor";
|
|
|
|
|
} else {
|
|
|
|
|
return "ein gewaltiges offenes Tor";
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if(lock > 0) {
|
|
|
|
|
return "gewaltiges verschlossenes Tor";
|
|
|
|
|
} else {
|
|
|
|
|
return "gewaltiges offenes Tor";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_questportal = {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
"questportal", VAR_INT,
|
2002-04-27 16:25:35 +02:00
|
|
|
|
b_opaque,
|
|
|
|
|
NULL, /* init */
|
|
|
|
|
NULL, /* destroy */
|
|
|
|
|
b_read, /* read */
|
|
|
|
|
b_write, /* write */
|
|
|
|
|
b_blockquestportal, /* block */
|
|
|
|
|
b_namequestportal, /* name */
|
|
|
|
|
b_rvisible, /* rvisible */
|
|
|
|
|
b_fvisible, /* fvisible */
|
|
|
|
|
b_uvisible, /* uvisible */
|
|
|
|
|
};
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/***
|
|
|
|
|
* roads. meant to replace the old at_road or r->road attribute
|
|
|
|
|
***/
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
b_nameroad(const border * b, const region * r, const struct faction * f, int gflags)
|
|
|
|
|
{
|
|
|
|
|
region * r2 = (r==b->to)?b->from:b->to;
|
2005-06-12 10:44:59 +02:00
|
|
|
|
int local = (r==b->from)?b->data.sa[0]:b->data.sa[1];
|
2001-01-25 10:37:55 +01:00
|
|
|
|
static char buffer[64];
|
|
|
|
|
|
|
|
|
|
unused(f);
|
|
|
|
|
if (gflags & GF_ARTICLE) {
|
2007-08-28 08:19:31 +02:00
|
|
|
|
if (!(gflags & GF_DETAILED)) return LOC(f->locale, "nr_a_road");
|
2005-10-25 14:38:01 +02:00
|
|
|
|
else if (r->terrain->max_road<=local) {
|
2005-06-12 10:44:59 +02:00
|
|
|
|
int remote = (r2==b->from)?b->data.sa[0]:b->data.sa[1];
|
2005-10-25 14:38:01 +02:00
|
|
|
|
if (r2->terrain->max_road<=remote) {
|
2007-08-28 08:19:31 +02:00
|
|
|
|
return LOC(f->locale, "nr_a_road");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
} else {
|
2007-08-28 08:19:31 +02:00
|
|
|
|
return LOC(f->locale, "nr_an_incomplete_road");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2005-10-25 14:38:01 +02:00
|
|
|
|
int percent = max(1, 100*local/r->terrain->max_road);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (local) {
|
2007-08-28 08:19:31 +02:00
|
|
|
|
sprintf(buffer, LOC(f->locale, "nr_road_percent"), percent);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
} else {
|
2007-08-28 08:19:31 +02:00
|
|
|
|
return LOC(f->locale, "nr_a_road_connection");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-08-28 08:19:31 +02:00
|
|
|
|
else if (gflags & GF_PLURAL) return LOC(f->locale, "nr_roads");
|
|
|
|
|
else return LOC(f->locale, "nr_road");
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
b_readroad(border * b, FILE *f)
|
|
|
|
|
{
|
2005-06-10 00:10:35 +02:00
|
|
|
|
fscanf(f, "%hd %hd ", &b->data.sa[0], &b->data.sa[1]);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
b_writeroad(const border * b, FILE *f)
|
|
|
|
|
{
|
2005-06-10 00:10:35 +02:00
|
|
|
|
fprintf(f, "%d %d ", b->data.sa[0], b->data.sa[1]);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static boolean
|
|
|
|
|
b_validroad(const border * b)
|
|
|
|
|
{
|
2005-06-10 00:10:35 +02:00
|
|
|
|
if (b->data.sa[0]==SHRT_MAX) return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static boolean
|
|
|
|
|
b_rvisibleroad(const border * b, const region * r)
|
|
|
|
|
{
|
2007-02-18 11:32:39 +01:00
|
|
|
|
int x = b->data.i;
|
2005-06-12 10:44:59 +02:00
|
|
|
|
x = (r==b->from)?b->data.sa[0]:b->data.sa[1];
|
2007-02-18 11:32:39 +01:00
|
|
|
|
if (x==0) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if (b->to!=r && b->from!=r) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_road = {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
"road", VAR_INT,
|
2001-01-25 10:37:55 +01:00
|
|
|
|
b_transparent,
|
|
|
|
|
NULL, /* init */
|
|
|
|
|
NULL, /* destroy */
|
|
|
|
|
b_readroad, /* read */
|
|
|
|
|
b_writeroad, /* write */
|
|
|
|
|
b_blocknone, /* block */
|
|
|
|
|
b_nameroad, /* name */
|
|
|
|
|
b_rvisibleroad, /* rvisible */
|
|
|
|
|
b_finvisible, /* fvisible */
|
|
|
|
|
b_uinvisible, /* uvisible */
|
|
|
|
|
b_validroad /* valid */
|
|
|
|
|
};
|
2001-04-22 20:14:07 +02:00
|
|
|
|
|
2004-05-23 00:55:02 +02:00
|
|
|
|
void
|
|
|
|
|
write_borders(FILE * f)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
for (i=0;i!=BMAXHASH;++i) {
|
2004-05-31 13:50:32 +02:00
|
|
|
|
border * bhash;
|
|
|
|
|
for (bhash=borders[i];bhash;bhash=bhash->nexthash) {
|
|
|
|
|
border * b;
|
|
|
|
|
for (b=bhash;b!=NULL;b=b->next) {
|
|
|
|
|
if (b->type->valid && !b->type->valid(b)) continue;
|
|
|
|
|
fprintf(f, "%s %d %d %d %d %d ", b->type->__name, b->id, b->from->x, b->from->y, b->to->x, b->to->y);
|
|
|
|
|
if (b->type->write) b->type->write(b, f);
|
|
|
|
|
a_write(f, b->attribs);
|
|
|
|
|
putc('\n', f);
|
|
|
|
|
}
|
2004-05-23 00:55:02 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2004-05-31 14:19:26 +02:00
|
|
|
|
fputs("end ", f);
|
2004-05-23 00:55:02 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
read_borders(FILE * f)
|
|
|
|
|
{
|
|
|
|
|
for (;;) {
|
2005-06-10 00:10:35 +02:00
|
|
|
|
short fx, fy, tx, ty;
|
2004-05-23 00:55:02 +02:00
|
|
|
|
unsigned int bid = 0;
|
|
|
|
|
char zText[32];
|
|
|
|
|
border * b;
|
|
|
|
|
region * from, * to;
|
|
|
|
|
border_type * type;
|
|
|
|
|
|
|
|
|
|
fscanf(f, "%s", zText);
|
|
|
|
|
if (!strcmp(zText, "end")) break;
|
2005-06-10 00:10:35 +02:00
|
|
|
|
fscanf(f, "%u %hd %hd %hd %hd", &bid, &fx, &fy, &tx, &ty);
|
2005-06-11 10:18:26 +02:00
|
|
|
|
|
2004-05-23 00:55:02 +02:00
|
|
|
|
from = findregion(fx, fy);
|
2007-09-16 17:34:49 +02:00
|
|
|
|
if (!incomplete_data && from==NULL) {
|
|
|
|
|
log_error(("border for unknown region %d,%d\n", fx, fy));
|
2004-05-23 00:55:02 +02:00
|
|
|
|
}
|
|
|
|
|
to = findregion(tx, ty);
|
2007-09-16 17:34:49 +02:00
|
|
|
|
if (!incomplete_data && to==NULL) {
|
|
|
|
|
log_error(("border for unknown region %d,%d\n", tx, ty));
|
2004-05-23 00:55:02 +02:00
|
|
|
|
}
|
2005-06-10 00:10:35 +02:00
|
|
|
|
|
|
|
|
|
type = find_bordertype(zText);
|
2007-09-16 17:34:49 +02:00
|
|
|
|
if (from) {
|
|
|
|
|
if (type==NULL) {
|
|
|
|
|
log_error(("[read_borders] unknown border type %s in %s\n", zText,
|
|
|
|
|
regionname(from, NULL)));
|
|
|
|
|
assert(type || !"border type not registered");
|
|
|
|
|
}
|
2005-06-10 00:10:35 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-09-16 17:34:49 +02:00
|
|
|
|
if (to==from && type && from) {
|
2006-02-19 23:43:56 +01:00
|
|
|
|
direction_t dir = (direction_t) (rng_int() % MAXDIRECTIONS);
|
2004-05-23 00:55:02 +02:00
|
|
|
|
region * r = rconnect(from, dir);
|
2005-06-11 10:18:26 +02:00
|
|
|
|
log_error(("[read_borders] invalid %s in %s\n", type->__name,
|
2007-09-16 17:34:49 +02:00
|
|
|
|
regionname(from, NULL)));
|
2004-05-23 00:55:02 +02:00
|
|
|
|
if (r!=NULL) to = r;
|
|
|
|
|
}
|
|
|
|
|
b = new_border(type, from, to);
|
|
|
|
|
nextborder--; /* new_border erh<72>ht den Wert */
|
|
|
|
|
b->id = bid;
|
|
|
|
|
assert(bid<=nextborder);
|
|
|
|
|
if (type->read) type->read(b, f);
|
|
|
|
|
a_read(f, &b->attribs);
|
2007-09-16 17:34:49 +02:00
|
|
|
|
if (!to || !from) {
|
|
|
|
|
erase_border(b);
|
|
|
|
|
}
|
2004-05-23 00:55:02 +02:00
|
|
|
|
}
|
|
|
|
|
}
|