2010-08-08 10:06:34 +02:00
|
|
|
|
/*
|
2015-01-30 22:10:29 +01:00
|
|
|
|
Copyright (c) 1998-2015, Enno Rehling <enno@eressea.de>
|
2015-01-30 20:37:14 +01:00
|
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
|
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
|
#include <kernel/config.h>
|
|
|
|
|
#include "connection.h"
|
|
|
|
|
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "save.h"
|
|
|
|
|
#include "terrain.h"
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
|
|
|
|
|
#include <util/attrib.h>
|
2012-06-04 08:36:40 +02:00
|
|
|
|
#include <util/bsdstring.h>
|
2016-02-13 13:42:02 +01:00
|
|
|
|
#include <util/gamedata.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include <util/language.h>
|
|
|
|
|
#include <util/log.h>
|
2012-05-31 04:55:17 +02:00
|
|
|
|
#include <quicklist.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include <util/rng.h>
|
2013-12-31 10:06:28 +01:00
|
|
|
|
|
|
|
|
|
#include <storage.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <string.h>
|
2014-03-15 19:29:11 +01:00
|
|
|
|
#include <stdlib.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-05-10 18:57:04 +02:00
|
|
|
|
int nextborder = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
#define BORDER_MAXHASH 8191
|
2011-03-07 08:02:35 +01:00
|
|
|
|
connection *borders[BORDER_MAXHASH];
|
|
|
|
|
border_type *bordertypes;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
void(*border_convert_cb) (struct connection * con, struct attrib * attr) = 0;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
|
|
void free_borders(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i != BORDER_MAXHASH; ++i) {
|
|
|
|
|
while (borders[i]) {
|
|
|
|
|
connection *b = borders[i];
|
|
|
|
|
borders[i] = b->nexthash;
|
|
|
|
|
while (b) {
|
|
|
|
|
connection *bf = b;
|
|
|
|
|
b = b->next;
|
|
|
|
|
assert(b == NULL || b->nexthash == NULL);
|
|
|
|
|
if (bf->type->destroy) {
|
|
|
|
|
bf->type->destroy(bf);
|
|
|
|
|
}
|
|
|
|
|
free(bf);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-14 19:39:45 +02:00
|
|
|
|
connection *find_border(int id)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int key;
|
|
|
|
|
for (key = 0; key != BORDER_MAXHASH; key++) {
|
|
|
|
|
connection *bhash;
|
|
|
|
|
for (bhash = borders[key]; bhash != NULL; bhash = bhash->nexthash) {
|
|
|
|
|
connection *b;
|
|
|
|
|
for (b = bhash; b; b = b->next) {
|
|
|
|
|
if (b->id == id)
|
|
|
|
|
return b;
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int resolve_borderid(variant id, void *addr)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-07-22 00:39:30 +02:00
|
|
|
|
int result = 0;
|
|
|
|
|
connection *b = NULL;
|
|
|
|
|
if (id.i != 0) {
|
|
|
|
|
b = find_border(id.i);
|
|
|
|
|
if (b == NULL) {
|
|
|
|
|
result = -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*(connection **)addr = b;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void walk_i(region *r, connection *b, void(*cb)(connection *, void *), void *data) {
|
|
|
|
|
for (; b; b = b->nexthash) {
|
|
|
|
|
if (b->from == r || b->to == r) {
|
|
|
|
|
connection *bn;
|
|
|
|
|
for (bn = b; bn; bn = bn->next) {
|
|
|
|
|
cb(b, data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void walk_connections(region *r, void(*cb)(connection *, void *), void *data) {
|
|
|
|
|
int key = reg_hashkey(r);
|
|
|
|
|
int d;
|
|
|
|
|
|
|
|
|
|
walk_i(r, borders[key], cb, data);
|
|
|
|
|
for (d = 0; d != MAXDIRECTIONS; ++d) {
|
|
|
|
|
region *rn = r_connect(r, d);
|
2015-10-30 12:57:07 +01:00
|
|
|
|
if (rn) {
|
|
|
|
|
int k = reg_hashkey(rn);
|
|
|
|
|
if (k < key) {
|
|
|
|
|
walk_i(r, borders[k], cb, data);
|
|
|
|
|
}
|
2014-07-22 00:39:30 +02:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static connection **get_borders_i(const region * r1, const region * r2)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
connection **bp;
|
|
|
|
|
int key = reg_hashkey(r1);
|
|
|
|
|
int k2 = reg_hashkey(r2);
|
|
|
|
|
|
|
|
|
|
key = _min(k2, key) % BORDER_MAXHASH;
|
|
|
|
|
bp = &borders[key];
|
|
|
|
|
while (*bp) {
|
|
|
|
|
connection *b = *bp;
|
|
|
|
|
if ((b->from == r1 && b->to == r2) || (b->from == r2 && b->to == r1))
|
|
|
|
|
break;
|
|
|
|
|
bp = &b->nexthash;
|
|
|
|
|
}
|
|
|
|
|
return bp;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
connection *get_borders(const region * r1, const region * r2)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
connection **bp = get_borders_i(r1, r2);
|
|
|
|
|
return *bp;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
connection *new_border(border_type * type, region * from, region * to)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-11-04 19:20:34 +01:00
|
|
|
|
connection *b, **bp = get_borders_i(from, to);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-11-04 19:20:34 +01:00
|
|
|
|
assert(from && to);
|
|
|
|
|
while (*bp) {
|
|
|
|
|
bp = &(*bp)->next;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
2015-11-04 19:20:34 +01:00
|
|
|
|
*bp = b = calloc(1, sizeof(connection));
|
2015-01-30 20:37:14 +01:00
|
|
|
|
b->type = type;
|
|
|
|
|
b->from = from;
|
|
|
|
|
b->to = to;
|
|
|
|
|
b->id = ++nextborder;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (type->init)
|
|
|
|
|
type->init(b);
|
|
|
|
|
return b;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void erase_border(connection * b)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (b->from && b->to) {
|
|
|
|
|
connection **bp = get_borders_i(b->from, b->to);
|
|
|
|
|
assert(*bp != NULL || !"error: connection is not registered");
|
|
|
|
|
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: connection is not registered");
|
|
|
|
|
*bp = b->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (b->type->destroy) {
|
|
|
|
|
b->type->destroy(b);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
free(b);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void register_bordertype(border_type * type)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
border_type **btp = &bordertypes;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
while (*btp && *btp != type)
|
|
|
|
|
btp = &(*btp)->next;
|
|
|
|
|
if (*btp)
|
|
|
|
|
return;
|
|
|
|
|
*btp = type;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
border_type *find_bordertype(const char *name)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
border_type *bt = bordertypes;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
while (bt && strcmp(bt->__name, name))
|
|
|
|
|
bt = bt->next;
|
|
|
|
|
return bt;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-02-13 14:29:44 +01:00
|
|
|
|
void b_read(connection * b, gamedata * data)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2016-02-13 14:29:44 +01:00
|
|
|
|
storage * store = data->store;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int n, result = 0;
|
|
|
|
|
switch (b->type->datatype) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case VAR_NONE:
|
|
|
|
|
case VAR_INT:
|
2015-01-30 20:37:14 +01:00
|
|
|
|
READ_INT(store, &b->data.i);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case VAR_SHORTA:
|
2015-01-30 20:37:14 +01:00
|
|
|
|
READ_INT(store, &n);
|
|
|
|
|
b->data.sa[0] = (short)n;
|
|
|
|
|
READ_INT(store, &n);
|
|
|
|
|
b->data.sa[1] = (short)n;
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case VAR_VOIDPTR:
|
|
|
|
|
default:
|
2015-01-30 20:37:14 +01:00
|
|
|
|
assert(!"invalid variant type in connection");
|
|
|
|
|
result = 0;
|
|
|
|
|
}
|
|
|
|
|
assert(result >= 0 || "EOF encountered?");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void b_write(const connection * b, storage * store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
switch (b->type->datatype) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case VAR_NONE:
|
|
|
|
|
case VAR_INT:
|
2015-01-30 20:37:14 +01:00
|
|
|
|
WRITE_INT(store, b->data.i);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case VAR_SHORTA:
|
2015-01-30 20:37:14 +01:00
|
|
|
|
WRITE_INT(store, b->data.sa[0]);
|
|
|
|
|
WRITE_INT(store, b->data.sa[1]);
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case VAR_VOIDPTR:
|
|
|
|
|
default:
|
2015-01-30 20:37:14 +01:00
|
|
|
|
assert(!"invalid variant type in connection");
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_transparent(const connection * b, const struct faction *f)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(b);
|
|
|
|
|
unused_arg(f);
|
|
|
|
|
return true;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_opaque(const connection * b, const struct faction * f)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(b);
|
|
|
|
|
unused_arg(f);
|
|
|
|
|
return false;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_blockall(const connection * b, const unit * u, const region * r)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(u);
|
|
|
|
|
unused_arg(r);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return true;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_blocknone(const connection * b, const unit * u, const region * r)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(u);
|
|
|
|
|
unused_arg(r);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return false;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_rvisible(const connection * b, const region * r)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
return (bool)(b->to == r || b->from == r);
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_fvisible(const connection * b, const struct faction * f,
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const region * r)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(r);
|
|
|
|
|
unused_arg(f);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return true;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_uvisible(const connection * b, const unit * u)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(u);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return true;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_rinvisible(const connection * b, const region * r)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(r);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return false;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_finvisible(const connection * b, const struct faction * f,
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const region * r)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(r);
|
|
|
|
|
unused_arg(f);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return false;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool b_uinvisible(const connection * b, const unit * u)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(u);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
return false;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
/**************************************/
|
|
|
|
|
/* at_countdown - legacy, do not use */
|
|
|
|
|
/**************************************/
|
|
|
|
|
|
|
|
|
|
attrib_type at_countdown = {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"countdown",
|
|
|
|
|
DEFAULT_INIT,
|
|
|
|
|
DEFAULT_FINALIZE,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
a_readint
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void age_borders(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
quicklist *deleted = NULL, *ql;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i != BORDER_MAXHASH; ++i) {
|
|
|
|
|
connection *bhash = borders[i];
|
|
|
|
|
for (; bhash; bhash = bhash->nexthash) {
|
|
|
|
|
connection *b = bhash;
|
|
|
|
|
for (; b; b = b->next) {
|
|
|
|
|
if (b->type->age) {
|
|
|
|
|
if (b->type->age(b) == AT_AGE_REMOVE) {
|
|
|
|
|
ql_push(&deleted, b);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
for (ql = deleted, i = 0; ql; ql_advance(&ql, &i, 1)) {
|
|
|
|
|
connection *b = (connection *)ql_get(ql, i);
|
|
|
|
|
erase_border(b);
|
|
|
|
|
}
|
|
|
|
|
ql_free(deleted);
|
2010-08-08 10:06:34 +02: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"
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static const char *b_namewall(const connection * b, const region * r,
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const struct faction *f, int gflags)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const char *bname = "wall";
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(f);
|
|
|
|
|
unused_arg(r);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
if (gflags & GF_ARTICLE)
|
|
|
|
|
bname = "a_wall";
|
|
|
|
|
if (gflags & GF_PURE)
|
|
|
|
|
return bname;
|
|
|
|
|
return LOC(f->locale, mkname("border", bname));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_wall = {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"wall", VAR_INT,
|
|
|
|
|
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 */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
border_type bt_noway = {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"noway", VAR_INT,
|
|
|
|
|
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 */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static const char *b_namefogwall(const connection * b, const region * r,
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const struct faction *f, int gflags)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(f);
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
unused_arg(r);
|
|
|
|
|
if (gflags & GF_PURE)
|
|
|
|
|
return "fogwall";
|
|
|
|
|
if (gflags & GF_ARTICLE)
|
|
|
|
|
return LOC(f->locale, mkname("border", "a_fogwall"));
|
|
|
|
|
return LOC(f->locale, mkname("border", "fogwall"));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
static bool
|
2010-08-08 10:06:34 +02:00
|
|
|
|
b_blockfogwall(const connection * b, const unit * u, const region * r)
|
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
unused_arg(b);
|
|
|
|
|
if (!u)
|
|
|
|
|
return true;
|
2015-08-27 16:16:55 +02:00
|
|
|
|
return (bool)(effskill(u, SK_PERCEPTION, r) > 4); /* Das ist die alte Nebelwand */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Legacy type used in old Eressea games, no longer in use. */
|
|
|
|
|
border_type bt_fogwall = {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"fogwall", VAR_INT,
|
|
|
|
|
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 */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static const char *b_nameillusionwall(const connection * b, const region * r,
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const struct faction *f, int gflags)
|
|
|
|
|
{
|
|
|
|
|
int fno = b->data.i;
|
|
|
|
|
unused_arg(b);
|
|
|
|
|
unused_arg(r);
|
|
|
|
|
if (gflags & GF_PURE)
|
|
|
|
|
return (f && fno == f->no) ? "illusionwall" : "wall";
|
|
|
|
|
if (gflags & GF_ARTICLE) {
|
|
|
|
|
return LOC(f->locale, mkname("border", (f
|
|
|
|
|
&& fno == f->subscription) ? "an_illusionwall" : "a_wall"));
|
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
return LOC(f->locale, mkname("border", (f
|
|
|
|
|
&& fno == f->no) ? "illusionwall" : "wall"));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_illusionwall = {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"illusionwall", VAR_INT,
|
|
|
|
|
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 */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/***
|
|
|
|
|
* roads. meant to replace the old at_road or r->road attribute
|
|
|
|
|
***/
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static const char *b_nameroad(const connection * b, const region * r,
|
2015-01-30 20:37:14 +01:00
|
|
|
|
const struct faction *f, int gflags)
|
|
|
|
|
{
|
|
|
|
|
region *r2 = (r == b->to) ? b->from : b->to;
|
|
|
|
|
int local = (r == b->from) ? b->data.sa[0] : b->data.sa[1];
|
|
|
|
|
static char buffer[64];
|
|
|
|
|
|
|
|
|
|
unused_arg(f);
|
|
|
|
|
if (gflags & GF_PURE)
|
|
|
|
|
return "road";
|
|
|
|
|
if (gflags & GF_ARTICLE) {
|
|
|
|
|
if (!(gflags & GF_DETAILED))
|
|
|
|
|
return LOC(f->locale, mkname("border", "a_road"));
|
|
|
|
|
else if (r->terrain->max_road <= local) {
|
|
|
|
|
int remote = (r2 == b->from) ? b->data.sa[0] : b->data.sa[1];
|
|
|
|
|
if (r2->terrain->max_road <= remote) {
|
|
|
|
|
return LOC(f->locale, mkname("border", "a_road"));
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
return LOC(f->locale, mkname("border", "an_incomplete_road"));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
int percent = _max(1, 100 * local / r->terrain->max_road);
|
|
|
|
|
if (local) {
|
|
|
|
|
slprintf(buffer, sizeof(buffer), LOC(f->locale, mkname("border",
|
|
|
|
|
"a_road_percent")), percent);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
return LOC(f->locale, mkname("border", "a_road_connection"));
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
else if (gflags & GF_PLURAL)
|
|
|
|
|
return LOC(f->locale, mkname("border", "roads"));
|
|
|
|
|
else
|
|
|
|
|
return LOC(f->locale, mkname("border", "road"));
|
|
|
|
|
return buffer;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-02-13 14:29:44 +01:00
|
|
|
|
static void b_readroad(connection * b, gamedata * data)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2016-02-13 14:29:44 +01:00
|
|
|
|
storage * store = data->store;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int n;
|
|
|
|
|
READ_INT(store, &n);
|
|
|
|
|
b->data.sa[0] = (short)n;
|
|
|
|
|
READ_INT(store, &n);
|
|
|
|
|
b->data.sa[1] = (short)n;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void b_writeroad(const connection * b, storage * store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
WRITE_INT(store, b->data.sa[0]);
|
|
|
|
|
WRITE_INT(store, b->data.sa[1]);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
static bool b_validroad(const connection * b)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
return (b->data.sa[0] != SHRT_MAX);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
static bool b_rvisibleroad(const connection * b, const region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int x = b->data.i;
|
|
|
|
|
x = (r == b->from) ? b->data.sa[0] : b->data.sa[1];
|
|
|
|
|
if (x == 0) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if (b->to != r && b->from != r) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
border_type bt_road = {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"road", VAR_INT,
|
|
|
|
|
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 */
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void write_borders(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i != BORDER_MAXHASH; ++i) {
|
|
|
|
|
connection *bhash;
|
|
|
|
|
for (bhash = borders[i]; bhash; bhash = bhash->nexthash) {
|
|
|
|
|
connection *b;
|
|
|
|
|
for (b = bhash; b != NULL; b = b->next) {
|
|
|
|
|
if (b->type->valid && !b->type->valid(b))
|
|
|
|
|
continue;
|
|
|
|
|
WRITE_TOK(store, b->type->__name);
|
|
|
|
|
WRITE_INT(store, b->id);
|
|
|
|
|
WRITE_INT(store, b->from->uid);
|
|
|
|
|
WRITE_INT(store, b->to->uid);
|
|
|
|
|
|
|
|
|
|
if (b->type->write)
|
|
|
|
|
b->type->write(b, store);
|
|
|
|
|
WRITE_SECTION(store);
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
WRITE_TOK(store, "end");
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-02-13 13:42:02 +01:00
|
|
|
|
int read_borders(gamedata *data)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2016-02-13 13:42:02 +01:00
|
|
|
|
struct storage *store = data->store;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
for (;;) {
|
|
|
|
|
int bid = 0;
|
|
|
|
|
char zText[32];
|
|
|
|
|
region *from, *to;
|
|
|
|
|
border_type *type;
|
|
|
|
|
|
|
|
|
|
READ_TOK(store, zText, sizeof(zText));
|
|
|
|
|
if (!strcmp(zText, "end"))
|
|
|
|
|
break;
|
2016-08-14 16:51:25 +02:00
|
|
|
|
type = find_bordertype(zText);
|
|
|
|
|
if (type == NULL) {
|
|
|
|
|
log_error("[read_borders] connection %d type %s is not registered", bid, zText);
|
|
|
|
|
assert(type || !"connection type not registered");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
READ_INT(store, &bid);
|
2016-02-13 13:42:02 +01:00
|
|
|
|
if (data->version < UIDHASH_VERSION) {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
int fx, fy, tx, ty;
|
|
|
|
|
READ_INT(store, &fx);
|
|
|
|
|
READ_INT(store, &fy);
|
|
|
|
|
READ_INT(store, &tx);
|
|
|
|
|
READ_INT(store, &ty);
|
|
|
|
|
from = findregion(fx, fy);
|
|
|
|
|
to = findregion(tx, ty);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
int fid, tid;
|
|
|
|
|
READ_INT(store, &fid);
|
|
|
|
|
READ_INT(store, &tid);
|
|
|
|
|
from = findregionbyid(fid);
|
|
|
|
|
to = findregionbyid(tid);
|
2016-08-14 16:51:25 +02:00
|
|
|
|
}
|
|
|
|
|
if (!to || !from) {
|
2015-11-05 10:58:08 +01:00
|
|
|
|
if (!to || !from) {
|
2016-08-14 16:51:25 +02:00
|
|
|
|
log_error("%s connection %d has missing regions", zText, bid);
|
2015-11-05 10:58:08 +01:00
|
|
|
|
}
|
2016-08-14 16:51:25 +02:00
|
|
|
|
if (type->read) {
|
|
|
|
|
// skip ahead
|
|
|
|
|
connection dummy;
|
2016-08-14 19:56:20 +02:00
|
|
|
|
type->read(&dummy, data);
|
2016-08-14 16:51:25 +02:00
|
|
|
|
}
|
|
|
|
|
continue;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
|
if (to == from && type && from) {
|
|
|
|
|
direction_t dir = (direction_t)(rng_int() % MAXDIRECTIONS);
|
|
|
|
|
region *r = rconnect(from, dir);
|
|
|
|
|
log_error("[read_borders] invalid %s in %s\n", type->__name, regionname(from, NULL));
|
|
|
|
|
if (r != NULL)
|
|
|
|
|
to = r;
|
|
|
|
|
}
|
2016-08-05 21:58:06 +02:00
|
|
|
|
if (type->read) {
|
2015-11-05 10:58:08 +01:00
|
|
|
|
connection *b = new_border(type, from, to);
|
|
|
|
|
nextborder--; /* new_border erh<72>ht den Wert */
|
|
|
|
|
b->id = bid;
|
|
|
|
|
assert(bid <= nextborder);
|
2016-08-05 21:58:06 +02:00
|
|
|
|
type->read(b, data);
|
2016-02-13 14:09:36 +01:00
|
|
|
|
if (data->version < NOBORDERATTRIBS_VERSION) {
|
2015-11-05 10:58:08 +01:00
|
|
|
|
attrib *a = NULL;
|
2016-02-13 13:42:02 +01:00
|
|
|
|
int result = read_attribs(data, &a, b);
|
2015-11-05 10:58:08 +01:00
|
|
|
|
if (border_convert_cb) {
|
|
|
|
|
border_convert_cb(b, a);
|
|
|
|
|
}
|
|
|
|
|
while (a) {
|
|
|
|
|
a_remove(&a, a);
|
|
|
|
|
}
|
|
|
|
|
if (result < 0) {
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-08-05 21:58:06 +02:00
|
|
|
|
if (!type->write) {
|
|
|
|
|
log_warning("invalid border '%s' between '%s' and '%s'\n", zText, regionname(from, 0), regionname(to, 0));
|
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2016-08-06 14:36:54 +02:00
|
|
|
|
|
|
|
|
|
const char * border_name(const connection *co, const struct region * r, const struct faction * f, int flags) {
|
|
|
|
|
return (co->type->name) ? co->type->name(co, r, f, flags) : co->type->__name;
|
|
|
|
|
}
|