2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
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)
|
|
|
|
|
* 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 "race.h"
|
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
#include <races/zombies.h>
|
|
|
|
|
#include <races/dragons.h>
|
|
|
|
|
#include <races/illusion.h>
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "alchemy.h"
|
2001-02-03 14:45:35 +01:00
|
|
|
|
#include "build.h"
|
|
|
|
|
#include "building.h"
|
2005-10-23 11:05:11 +02:00
|
|
|
|
#include "equipment.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "faction.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "group.h"
|
2001-02-03 14:45:35 +01:00
|
|
|
|
#include "item.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "karma.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "names.h"
|
|
|
|
|
#include "pathfinder.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "region.h"
|
2001-02-03 14:45:35 +01:00
|
|
|
|
#include "ship.h"
|
|
|
|
|
#include "skill.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "terrain.h"
|
|
|
|
|
#include "unit.h"
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include "version.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* util includes */
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/attrib.h>
|
2007-06-27 22:51:56 +02:00
|
|
|
|
#include <util/bsdstring.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/functions.h>
|
2007-08-05 14:19:56 +02:00
|
|
|
|
#include <util/log.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/rng.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-09-05 21:40:40 +02:00
|
|
|
|
/* attrib includes */
|
|
|
|
|
#include <attributes/raceprefix.h>
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <stdio.h>
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#include <string.h>
|
2001-04-01 08:58:45 +02:00
|
|
|
|
#include <stdlib.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include <math.h>
|
2001-09-05 21:40:40 +02:00
|
|
|
|
#include <ctype.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/** external variables **/
|
|
|
|
|
race * races;
|
2006-02-10 22:15:08 +01:00
|
|
|
|
|
|
|
|
|
race_list *
|
|
|
|
|
get_familiarraces(void)
|
|
|
|
|
{
|
|
|
|
|
static int init = 0;
|
|
|
|
|
static race_list * familiarraces;
|
|
|
|
|
|
|
|
|
|
if (!init) {
|
|
|
|
|
race * rc = races;
|
|
|
|
|
for (;rc!=NULL;rc=rc->next) {
|
|
|
|
|
if (rc->init_familiar!=NULL) {
|
|
|
|
|
racelist_insert(&familiarraces, rc);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
init = false;
|
|
|
|
|
}
|
|
|
|
|
return familiarraces;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-11 21:59:02 +02:00
|
|
|
|
void
|
|
|
|
|
racelist_clear(struct race_list **rl)
|
|
|
|
|
{
|
|
|
|
|
while (*rl) {
|
|
|
|
|
race_list * rl2 = (*rl)->next;
|
|
|
|
|
free(*rl);
|
|
|
|
|
*rl = rl2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
racelist_insert(struct race_list **rl, const struct race *r)
|
|
|
|
|
{
|
|
|
|
|
race_list *rl2 = (race_list*)malloc(sizeof(race_list));
|
|
|
|
|
|
|
|
|
|
rl2->data = r;
|
|
|
|
|
rl2->next = *rl;
|
|
|
|
|
|
|
|
|
|
*rl = rl2;
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
race *
|
|
|
|
|
rc_new(const char * zName)
|
|
|
|
|
{
|
2005-08-21 11:24:14 +02:00
|
|
|
|
char zBuffer[80];
|
|
|
|
|
race * rc = calloc(sizeof(race), 1);
|
|
|
|
|
if (strchr(zName, ' ')!=NULL) {
|
|
|
|
|
log_error(("race '%s' has an invalid name. remove spaces\n", zName));
|
|
|
|
|
assert(strchr(zName, ' ')==NULL);
|
|
|
|
|
}
|
|
|
|
|
strcpy(zBuffer, zName);
|
|
|
|
|
rc->_name[0] = strdup(zBuffer);
|
|
|
|
|
sprintf(zBuffer, "%s_p", zName);
|
|
|
|
|
rc->_name[1] = strdup(zBuffer);
|
|
|
|
|
sprintf(zBuffer, "%s_d", zName);
|
|
|
|
|
rc->_name[2] = strdup(zBuffer);
|
|
|
|
|
sprintf(zBuffer, "%s_x", zName);
|
|
|
|
|
rc->_name[3] = strdup(zBuffer);
|
2005-10-08 20:27:40 +02:00
|
|
|
|
rc->precombatspell = NULL;
|
2005-10-16 22:11:34 +02:00
|
|
|
|
|
|
|
|
|
rc->attack[0].type = AT_COMBATSPELL;
|
|
|
|
|
rc->attack[1].type = AT_NONE;
|
2005-08-21 11:24:14 +02:00
|
|
|
|
return rc;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
race *
|
2002-01-21 01:31:43 +01:00
|
|
|
|
rc_add(race * rc)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-01-01 23:20:52 +01:00
|
|
|
|
rc->next = races;
|
|
|
|
|
return races = rc;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const char * racealias[2][2] = {
|
2006-01-01 23:20:52 +01:00
|
|
|
|
{ "skeletton lord", "skeleton lord" },
|
|
|
|
|
{ NULL, NULL }
|
2001-12-10 01:13:39 +01:00
|
|
|
|
};
|
2004-06-11 21:59:02 +02:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
race *
|
|
|
|
|
rc_find(const char * name)
|
|
|
|
|
{
|
|
|
|
|
const char * rname = name;
|
|
|
|
|
race * rc = races;
|
|
|
|
|
int i;
|
2004-06-11 21:59:02 +02:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
for (i=0;racealias[i][0];++i) {
|
|
|
|
|
if (strcmp(racealias[i][0], name)==0) {
|
|
|
|
|
rname = racealias[i][1];
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (rc && !strcmp(rname, rc->_name[0])==0) rc = rc->next;
|
2006-01-01 23:20:52 +01:00
|
|
|
|
return rc;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/** dragon movement **/
|
|
|
|
|
boolean
|
|
|
|
|
allowed_dragon(const region * src, const region * target)
|
|
|
|
|
{
|
2006-02-26 18:28:55 +01:00
|
|
|
|
if (fval(src->terrain, ARCTIC_REGION) && fval(target->terrain, SEA_REGION)) return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return allowed_fly(src, target);
|
|
|
|
|
}
|
|
|
|
|
|
2004-08-07 09:42:22 +02:00
|
|
|
|
char ** race_prefixes = NULL;
|
|
|
|
|
|
|
|
|
|
extern void
|
|
|
|
|
add_raceprefix(const char * prefix)
|
|
|
|
|
{
|
|
|
|
|
static size_t size = 4;
|
|
|
|
|
static unsigned int next = 0;
|
|
|
|
|
if (race_prefixes==NULL) race_prefixes = malloc(size * sizeof(char*));
|
|
|
|
|
if (next+1==size) {
|
|
|
|
|
size *= 2;
|
|
|
|
|
race_prefixes = realloc(race_prefixes, size * sizeof(char*));
|
|
|
|
|
}
|
|
|
|
|
race_prefixes[next++] = strdup(prefix);
|
|
|
|
|
race_prefixes[next] = NULL;
|
|
|
|
|
}
|
2001-09-05 21:40:40 +02:00
|
|
|
|
|
|
|
|
|
/* Die Bezeichnungen d<>rfen wegen der Art des Speicherns keine
|
|
|
|
|
* Leerzeichen enthalten! */
|
|
|
|
|
|
|
|
|
|
/* "den Zwergen", "Halblingsparteien" */
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
void
|
|
|
|
|
set_show_item(faction *f, item_t i)
|
|
|
|
|
{
|
|
|
|
|
attrib *a = a_add(&f->attribs, a_new(&at_showitem));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
a->data.v = (void*)olditemtype[i];
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
unit_max_hp(const unit * u)
|
|
|
|
|
{
|
|
|
|
|
int h;
|
|
|
|
|
double p;
|
2002-05-25 08:58:54 +02:00
|
|
|
|
static const curse_type * heal_ct = NULL;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
h = u->race->hitpoints;
|
2002-05-25 08:58:54 +02:00
|
|
|
|
if (heal_ct==NULL) heal_ct = ct_find("healingzone");
|
2002-05-24 09:12:50 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
p = pow(effskill(u, SK_AUSDAUER) / 2.0, 1.5) * 0.2;
|
|
|
|
|
h += (int) (h * p + 0.5);
|
|
|
|
|
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#ifdef KARMA_MODULE
|
|
|
|
|
if (fspecial(u->faction, FS_UNDEAD)) {
|
2001-02-28 19:25:25 +01:00
|
|
|
|
h *= 2;
|
|
|
|
|
}
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#endif /* KARMA_MODULE */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2002-05-24 09:12:50 +02:00
|
|
|
|
/* der healing curse ver<65>ndert die maximalen hp */
|
|
|
|
|
if (heal_ct) {
|
|
|
|
|
curse *c = get_curse(u->region->attribs, heal_ct);
|
|
|
|
|
if (c) {
|
|
|
|
|
h = (int) (h * (1.0+(curse_geteffect(c)/100)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return h;
|
|
|
|
|
}
|
2005-03-06 14:17:22 +01:00
|
|
|
|
|
2001-09-05 21:40:40 +02:00
|
|
|
|
boolean
|
|
|
|
|
r_insectstalled(const region * r)
|
|
|
|
|
{
|
2007-03-25 01:04:40 +01:00
|
|
|
|
return fval(r->terrain, ARCTIC_REGION);
|
2001-09-05 21:40:40 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
2001-12-10 01:13:39 +01:00
|
|
|
|
rc_name(const race * rc, int n)
|
|
|
|
|
{
|
2006-02-20 22:24:09 +01:00
|
|
|
|
return mkname("race", rc->_name[n]);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
2003-09-21 10:20:58 +02:00
|
|
|
|
raceprefix(const unit *u)
|
2001-09-05 21:40:40 +02:00
|
|
|
|
{
|
2006-02-20 22:24:09 +01:00
|
|
|
|
const attrib * asource = u->faction->attribs;
|
2001-09-05 21:40:40 +02:00
|
|
|
|
|
2005-06-12 19:57:14 +02:00
|
|
|
|
if (fval(u, UFL_GROUP)) {
|
|
|
|
|
const attrib * agroup = agroup = a_findc(u->attribs, &at_group);
|
2006-02-20 22:24:09 +01:00
|
|
|
|
if (agroup!=NULL) asource = ((const group *)(agroup->data.v))->attribs;
|
2005-06-12 19:57:14 +02:00
|
|
|
|
}
|
2006-02-20 22:24:09 +01:00
|
|
|
|
return get_prefix(asource);
|
2003-09-21 10:20:58 +02:00
|
|
|
|
}
|
2002-01-21 01:31:43 +01:00
|
|
|
|
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *
|
2003-09-21 10:20:58 +02:00
|
|
|
|
racename(const struct locale *loc, const unit *u, const race * rc)
|
|
|
|
|
{
|
2006-02-20 22:24:09 +01:00
|
|
|
|
const char * prefix = raceprefix(u);
|
|
|
|
|
|
|
|
|
|
if (prefix!=NULL) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
static char lbuf[80];
|
2007-09-02 01:40:58 +02:00
|
|
|
|
char * bufp = lbuf;
|
|
|
|
|
size_t size = sizeof(lbuf) - 1;
|
|
|
|
|
int ch, bytes;
|
|
|
|
|
|
|
|
|
|
bytes = (int)strlcpy(bufp, LOC(loc, mkname("prefix", prefix)), size);
|
|
|
|
|
if (wrptr(&bufp, &size, bytes)!=0) WARN_STATIC_BUFFER();
|
|
|
|
|
bytes = (int)strlcpy(bufp, LOC(loc, rc_name(rc, u->number != 1)), size);
|
|
|
|
|
if (wrptr(&bufp, &size, bytes)!=0) WARN_STATIC_BUFFER();
|
|
|
|
|
*bufp = 0;
|
|
|
|
|
|
|
|
|
|
assert(~lbuf[0] & 0x80|| !"unicode/not implemented");
|
|
|
|
|
ch = tolower(*(unsigned char *)lbuf);
|
|
|
|
|
lbuf[0] = (char)ch;
|
2006-02-20 22:24:09 +01:00
|
|
|
|
return lbuf;
|
|
|
|
|
}
|
|
|
|
|
return LOC(loc, rc_name(rc, u->number != 1));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-01-21 01:31:43 +01:00
|
|
|
|
int
|
2001-12-10 01:13:39 +01:00
|
|
|
|
rc_specialdamage(const race * ar, const race * dr, const struct weapon_type * wtype)
|
|
|
|
|
{
|
2004-06-11 21:59:02 +02:00
|
|
|
|
race_t art = old_race(ar);
|
|
|
|
|
int m, modifier = 0;
|
|
|
|
|
|
|
|
|
|
if (wtype!=NULL && wtype->modifiers!=NULL) for (m=0;wtype->modifiers[m].value;++m) {
|
|
|
|
|
/* weapon damage for this weapon, possibly by race */
|
|
|
|
|
if (wtype->modifiers[m].flags & WMF_DAMAGE) {
|
|
|
|
|
race_list * rlist = wtype->modifiers[m].races;
|
|
|
|
|
if (rlist!=NULL) {
|
|
|
|
|
while (rlist) {
|
|
|
|
|
if (rlist->data == ar) break;
|
|
|
|
|
rlist = rlist->next;
|
|
|
|
|
}
|
|
|
|
|
if (rlist==NULL) continue;
|
|
|
|
|
}
|
|
|
|
|
modifier += wtype->modifiers[m].value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
switch (art) {
|
|
|
|
|
case RC_HALFLING:
|
|
|
|
|
if (wtype!=NULL && dragonrace(dr)) {
|
|
|
|
|
modifier += 5;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return modifier;
|
2001-09-05 21:40:40 +02:00
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
void
|
|
|
|
|
write_race_reference(const race * rc, FILE * F)
|
|
|
|
|
{
|
|
|
|
|
fprintf(F, "%s ", rc?rc->_name[0]:"none");
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-07 02:44:01 +02:00
|
|
|
|
int
|
2001-12-10 01:13:39 +01:00
|
|
|
|
read_race_reference(const struct race ** rp, FILE * F)
|
|
|
|
|
{
|
|
|
|
|
char zName[20];
|
|
|
|
|
if (global.data_version<NEWRACE_VERSION) {
|
|
|
|
|
int i;
|
|
|
|
|
fscanf(F, "%d", &i);
|
|
|
|
|
if (i>=0) {
|
|
|
|
|
*rp = new_race[i];
|
|
|
|
|
} else {
|
|
|
|
|
*rp = NULL;
|
2002-04-07 02:44:01 +02:00
|
|
|
|
return AT_READ_FAIL;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
fscanf(F, "%s", zName);
|
|
|
|
|
if (strcmp(zName, "none")==0) {
|
|
|
|
|
*rp = NULL;
|
2002-04-07 11:58:22 +02:00
|
|
|
|
return AT_READ_OK;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2002-04-07 02:44:01 +02:00
|
|
|
|
*rp = rc_find(zName);
|
|
|
|
|
assert(*rp!=NULL);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2002-04-07 02:44:01 +02:00
|
|
|
|
return AT_READ_OK;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|