server/src/common/kernel/unit.c

1052 lines
20 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
*
2001-01-25 10:37:55 +01:00
* Eressea PB(E)M host Copyright (C) 1998-2000
* 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 "unit.h"
#include "building.h"
#include "faction.h"
#include "group.h"
#include "karma.h"
2001-01-25 10:37:55 +01:00
#include "border.h"
#include "item.h"
#include "movement.h"
#include "plane.h"
2001-01-25 10:37:55 +01:00
#include "race.h"
#include "region.h"
#include "ship.h"
#include "skill.h"
2001-01-25 10:37:55 +01:00
#include <attributes/moved.h>
2001-01-25 10:37:55 +01:00
/* util includes */
#include <base36.h>
#include <event.h>
#include <goodies.h>
#include <resolve.h>
2001-01-25 10:37:55 +01:00
#ifdef OLD_TRIGGER
# include <old/trigger.h>
2001-01-25 10:37:55 +01:00
#endif
/* libc includes */
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#define FIND_FOREIGN_TEMP
/* ------------------------------------------------------------- */
const unit u_peasants = { NULL, NULL, NULL, NULL, NULL, 2, "die Bauern" };
const unit u_unknown = { NULL, NULL, NULL, NULL, NULL, 1, "eine unbekannte Einheit" };
2001-01-25 10:37:55 +01:00
#define DMAXHASH 8191
typedef struct dead {
struct dead * nexthash;
faction * f;
int no;
} dead;
static dead* deadhash[DMAXHASH];
static void
dhash(int no, faction * f)
{
dead * hash = (dead*)calloc(1, sizeof(dead));
dead * old = deadhash[no % DMAXHASH];
hash->no = no;
hash->f = f;
deadhash[no % DMAXHASH] = hash;
hash->nexthash = old;
}
faction *
dfindhash(int no)
2001-01-25 10:37:55 +01:00
{
dead * old;
if(no < 0) return 0;
for (old = deadhash[no % DMAXHASH]; old; old = old->nexthash)
if (old->no == no)
2001-01-25 10:37:55 +01:00
return old->f;
return 0;
}
unit * udestroy = NULL;
2001-01-25 10:37:55 +01:00
#undef DESTROY
/* Einheiten werden nicht wirklich zerst<73>rt. */
void
destroy_unit(unit * u)
{
region *r = u->region;
boolean zombie = false;
unit *clone;
2001-01-25 10:37:55 +01:00
if (!ufindhash(u->no)) return;
if (!fval(u->race, RCF_ILLUSIONARY)) {
2001-01-25 10:37:55 +01:00
item ** p_item = &u->items;
unit * u3;
u->faction->no_units--;
if (r) for (u3 = r->units; u3; u3 = u3->next) {
if (u3 != u && u3->faction == u->faction && playerrace(u3->race)) {
2001-01-25 10:37:55 +01:00
i_merge(&u3->items, &u->items);
u->items = NULL;
break;
}
}
u3 = NULL;
while (*p_item) {
item * item = *p_item;
if (item->number && item->type->flags & ITF_NOTLOST) {
if (u3==NULL) {
u3 = r->units;
while (u3 && u3!=u) u3 = u3->next;
if (!u3) {
zombie = true;
break;
}
}
if (u3) {
i_add(&u3->items, i_remove(p_item, item));
}
}
if (*p_item == item) p_item=&item->next;
}
}
/* Wir machen das erst nach dem L<>schen der Items. Der Klon darf keine
* Items haben, sonst Memory-Leak. */
clone = has_clone(u);
if (clone && rand()%100 < 90) {
attrib *a;
int i;
/* TODO: Messages generieren. */
u->region = clone->region;
u->ship = clone->ship;
u->building = clone->building;
u->hp = 1;
i = u->no;
u->no = clone->no;
clone->no = i;
set_number(u, 1);
set_spellpoints(u, 0);
a = a_find(u->attribs, &at_clone);
a_remove(&u->attribs, a);
a = a_find(clone->attribs, &at_clonemage);
a_remove(&clone->attribs, a);
fset(u, FL_LONGACTION);
set_number(clone, 0);
u = clone;
zombie = false;
}
2001-01-25 10:37:55 +01:00
if (zombie) {
2001-02-10 15:18:01 +01:00
u_setfaction(u, findfaction(MONSTER_FACTION));
2001-01-25 10:37:55 +01:00
scale_number(u, 1);
u->race = u->irace = new_race[RC_ZOMBIE];
2001-01-25 10:37:55 +01:00
} else {
if (r && rterrain(r) != T_OCEAN)
rsetmoney(r, rmoney(r) + get_money(u));
dhash(u->no, u->faction);
2001-02-10 15:18:01 +01:00
u_setfaction(u, NULL);
2001-01-25 10:37:55 +01:00
if (r) leave(r, u);
uunhash(u);
if (r) choplist(&r->units, u);
u->next = udestroy;
udestroy = u;
if (u->number) set_number(u, 0);
handle_event(&u->attribs, "destroy", u);
}
}
unit *
2001-04-26 19:41:06 +02:00
findnewunit (const region * r, const faction *f, int n)
2001-01-25 10:37:55 +01:00
{
unit *u2;
if (n == 0)
return 0;
for (u2 = r->units; u2; u2 = u2->next)
if (u2->faction == f && ualias(u2) == n)
return u2;
#ifdef FIND_FOREIGN_TEMP
for (u2 = r->units; u2; u2 = u2->next)
if (ualias(u2) == n)
return u2;
#endif
return 0;
}
/* ------------------------------------------------------------- */
/*********************/
/* at_alias */
/*********************/
attrib_type at_alias = {
"alias",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ
};
int
ualias(const unit * u) {
attrib * a = a_find(u->attribs, &at_alias);
if (!a) return 0;
return a->data.i;
}
/*********************/
/* at_private */
/*********************/
attrib_type at_private = {
"private",
DEFAULT_INIT,
a_finalizestring,
DEFAULT_AGE,
a_writestring,
a_readstring
};
const char *
uprivate(const unit * u) {
attrib * a = a_find(u->attribs, &at_private);
if (!a) return NULL;
return (const char*)a->data.v;
}
void
usetprivate(unit * u, const char * str) {
attrib * a = a_find(u->attribs, &at_private);
2001-02-18 10:32:13 +01:00
if(str == NULL) {
if(a) a_remove(&u->attribs, a);
return;
}
2001-01-25 10:37:55 +01:00
if (!a) a = a_add(&u->attribs, a_new(&at_private));
if (a->data.v) free(a->data.v);
a->data.v = strdup(str);
}
/*********************/
/* at_potion */
/*********************/
/* Einheit BESITZT einen Trank */
attrib_type at_potion = {
"potion",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
#if RELEASE_VERSION<ATTRIBFIX_VERSION
DEFAULT_WRITE,
#else
NO_WRITE,
#endif
DEFAULT_READ
};
/*********************/
/* at_potionuser */
/*********************/
/* Einheit BENUTZT einen Trank */
attrib_type at_potionuser = {
"potionuser",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ
};
void
usetpotionuse(unit * u, const potion_type * ptype)
{
attrib * a = a_find(u->attribs, &at_potionuser);
if (!a) a = a_add(&u->attribs, a_new(&at_potionuser));
a->data.v = (void*)ptype;
}
const potion_type *
ugetpotionuse(const unit * u) {
attrib * a = a_find(u->attribs, &at_potionuser);
if (!a) return NULL;
return (const potion_type *)a->data.v;
}
/*********************/
/* at_target */
/*********************/
attrib_type at_target = {
"target",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ
};
unit *
utarget(const unit * u) {
attrib * a;
if (!fval(u, FL_TARGET)) return NULL;
a = a_find(u->attribs, &at_target);
assert (a || !"flag set, but no target found");
return (unit*)a->data.v;
}
void
usettarget(unit * u, const unit * t)
{
attrib * a = a_find(u->attribs, &at_target);
if (!a && t) a = a_add(&u->attribs, a_new(&at_target));
if (a) {
if (!t) {
a_remove(&u->attribs, a);
freset(u, FL_TARGET);
}
else {
a->data.v = (void*)t;
fset(u, FL_TARGET);
}
}
}
/*********************/
/* at_siege */
/*********************/
void
a_writesiege(const attrib * a, FILE * f)
{
struct building * b = (struct building*)a->data.v;
2001-05-11 22:19:22 +02:00
write_building_reference(b, f);
2001-01-25 10:37:55 +01:00
}
int
a_readsiege(attrib * a, FILE * f)
{
2001-05-11 22:19:22 +02:00
return read_building_reference((struct building**)&a->data.v, f);
2001-01-25 10:37:55 +01:00
}
attrib_type at_siege = {
"siege",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
a_writesiege,
a_readsiege
};
struct building *
usiege(const unit * u) {
attrib * a;
if (!fval(u, FL_SIEGE)) return NULL;
a = a_find(u->attribs, &at_siege);
assert (a || !"flag set, but no siege found");
return (struct building *)a->data.v;
}
void
usetsiege(unit * u, const struct building * t)
{
attrib * a = a_find(u->attribs, &at_siege);
if (!a && t) a = a_add(&u->attribs, a_new(&at_siege));
if (a) {
if (!t) {
a_remove(&u->attribs, a);
freset(u, FL_SIEGE);
}
else {
a->data.v = (void*)t;
fset(u, FL_SIEGE);
}
}
}
/*********************/
/* at_contact */
/*********************/
attrib_type at_contact = {
"contact",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
NO_WRITE,
NO_READ
};
void
usetcontact(unit * u, const unit * u2)
{
attrib * a = a_find(u->attribs, &at_contact);
while (a && a->data.v!=u2) a = a->nexttype;
if (a) return;
a_add(&u->attribs, a_new(&at_contact))->data.v = (void*)u2;
}
boolean
ucontact(const unit * u, const unit * u2)
/* Pr<50>ft, ob u den Kontaktiere-Befehl zu u2 gesetzt hat. */
{
attrib *ru;
/* Alliierte kontaktieren immer */
if (allied(u, u2->faction, HELP_GIVE) == HELP_GIVE)
return true;
/* Explizites KONTAKTIERE */
for (ru = a_find(u->attribs, &at_contact); ru; ru = ru->nexttype)
if (((unit*)ru->data.v) == u2)
return true;
return false;
}
void *
resolve_unit(void * id)
{
return ufindhash((int)id);
}
/***
** init & cleanup module
**/
void
free_units(void)
{
while (udestroy) {
unit * u = udestroy;
udestroy = udestroy->next;
stripunit(u);
free(u);
}
}
void
write_unit_reference(const unit * u, FILE * F)
{
fprintf(F, "%s ", u?itoa36(u->no):"0");
}
void
read_unit_reference(unit ** up, FILE * F)
{
char zId[10];
int i;
fscanf(F, "%s", zId);
i = atoi36(zId);
if (up) {
if (i==0) *up = NULL;
else {
*up = findunit(i);
if (*up==NULL) ur_add((void*)i, (void**)up, resolve_unit);
}
2001-01-25 10:37:55 +01:00
}
}
attrib_type at_stealth = {
"stealth", NULL, NULL, NULL, DEFAULT_WRITE, DEFAULT_READ
};
void
u_seteffstealth(unit * u, int value)
{
attrib * a = a_find(u->attribs, &at_stealth);
if (!a && value<0) return;
if (!a) a = a_add(&u->attribs, a_new(&at_stealth));
a->data.i = value;
}
int
u_geteffstealth(const struct unit * u)
{
attrib * a = a_find(u->attribs, &at_stealth);
return (a?a->data.i:-1);
}
#if SKILLPOINTS
2001-01-25 10:37:55 +01:00
int
change_skill(unit * u, skill_t id, int byvalue)
2001-01-25 10:37:55 +01:00
{
skill *i = u->skills;
2001-01-25 10:37:55 +01:00
int wounds = 0;
if (id == SK_AUSDAUER) {
wounds = unit_max_hp(u)*u->number - u->hp;
}
for (; i != u->skills + u->skill_size; ++i)
if (i->id == id) {
i->value = max(0, i->value + byvalue);
if (id == SK_AUSDAUER) {
u->hp = unit_max_hp(u)*u->number - wounds;
}
return i->value;
}
set_skill(u, id, byvalue);
if (id == SK_AUSDAUER) {
u->hp = unit_max_hp(u)*u->number - wounds;
}
return byvalue;
}
static int
change_skill_transfermen(unit * u, skill_t id, int byvalue)
2001-01-25 10:37:55 +01:00
{
skill *i = u->skills;
2001-01-25 10:37:55 +01:00
for (; i != u->skills + u->skill_size; ++i)
if (i->id == id) {
i->value = max(0, i->value + byvalue);
return i->value;
}
set_skill(u, id, byvalue);
return byvalue;
}
int
get_skill(const unit * u, skill_t id)
{
skill *i = u->skills;
for (; i != u->skills + u->skill_size; ++i)
if (i->id == id)
return i->value;
return 0;
}
2001-01-25 10:37:55 +01:00
void
set_skill(unit * u, skill_t id, int value)
2001-01-25 10:37:55 +01:00
{
skill *i = u->skills;
2001-01-25 10:37:55 +01:00
assert(value>=0);
for (; i != u->skills + u->skill_size; ++i)
if (i->id == id) {
if (value)
i->value = value;
else {
*i = *(u->skills + u->skill_size - 1);
--u->skill_size;
}
return;
}
if (!value)
return;
++u->skill_size;
u->skills = realloc(u->skills, u->skill_size * sizeof(skill));
2001-01-25 10:37:55 +01:00
(u->skills + u->skill_size - 1)->value = value;
(u->skills + u->skill_size - 1)->id = id;
}
#endif
#if SKILLPOINTS
extern int level(int days);
#endif
int
get_level(const unit * u, skill_t id)
{
skill * sv = u->skills;
while (sv != u->skills + u->skill_size) {
if (sv->id == id) {
#if SKILLPOINTS
return level(sv->value/u->number);
#else
return sv->level;
#endif
}
++sv;
}
return 0;
}
void
set_level(unit * u, skill_t id, int value)
{
#if SKILLPOINTS
set_skill(u, id, level_days(value)*u->number);
#else
skill * sv = u->skills;
while (sv != u->skills + u->skill_size) {
if (sv->id == id) {
sv->level = (unsigned char)value;
sv->learning = 0;
}
}
#endif
}
2001-01-25 10:37:55 +01:00
static attrib_type at_leftship = {
"leftship",
};
static attrib *
make_leftship(struct ship * leftship)
{
attrib * a = a_new(&at_leftship);
a->data.v = leftship;
return a;
}
void
set_leftship(unit *u, ship *sh)
{
a_add(&u->attribs, make_leftship(sh));
}
ship *
leftship(const unit *u)
{
attrib * a = a_find(u->attribs, &at_leftship);
/* Achtung: Es ist nicht garantiert, da<64> der R<>ckgabewert zu jedem
* Zeitpunkt noch auf ein existierendes Schiff zeigt! */
if (a) return (ship *)(a->data.v);
return NULL;
}
void
leave_ship(unit * u)
{
struct ship * sh = u->ship;
if (sh==NULL) return;
u->ship = NULL;
set_leftship(u, sh);
if (fval(u, FL_OWNER)) {
unit *u2, *owner = NULL;
freset(u, FL_OWNER);
for (u2 = u->region->units; u2; u2 = u2->next) {
if (u2->ship == sh) {
if (u2->faction == u->faction) {
owner = u2;
break;
}
else if (owner==NULL) owner = u2;
}
}
if (owner!=NULL) fset(owner, FL_OWNER);
}
}
void
leave_building(unit * u)
{
struct building * b = u->building;
if (!b) return;
u->building = NULL;
if (fval(u, FL_OWNER)) {
unit *u2, *owner = NULL;
freset(u, FL_OWNER);
for (u2 = u->region->units; u2; u2 = u2->next) {
if (u2->building == b) {
if (u2->faction == u->faction) {
owner = u2;
break;
}
else if (owner==NULL) owner = u2;
}
}
if (owner!=NULL) fset(owner, FL_OWNER);
}
}
void
leave(struct region * r, unit * u)
{
if (u->building) leave_building(u);
else if (u->ship) leave_ship(u);
unused(r);
}
const struct race *
urace(const struct unit * u)
{
return u->race;
}
2001-01-25 10:37:55 +01:00
boolean
can_survive(const unit *u, const region *r)
{
if (((terrain[rterrain(r)].flags & WALK_INTO)
&& (u->race->flags & RCF_WALK)) ||
2001-01-25 10:37:55 +01:00
((terrain[rterrain(r)].flags & SWIM_INTO)
&& (u->race->flags & RCF_SWIM)) ||
2001-01-25 10:37:55 +01:00
((terrain[rterrain(r)].flags & FLY_INTO)
&& (u->race->flags & RCF_FLY))) {
2001-01-25 10:37:55 +01:00
if (get_item(u, I_HORSE) && !(terrain[rterrain(r)].flags & WALK_INTO))
return false;
if (fval(u->race, RCF_UNDEAD) && is_cursed(r->attribs, C_HOLYGROUND, 0))
2001-01-31 18:40:53 +01:00
return false;
2001-01-25 10:37:55 +01:00
return true;
}
return false;
}
void
move_unit(unit * u, region * r, unit ** ulist)
{
assert(u && r);
if (u->region == r) return;
if (!ulist) ulist = (&r->units);
if (u->region) {
set_moved(&u->attribs);
2001-01-25 10:37:55 +01:00
setguard(u, GUARD_NONE);
fset(u, FL_MOVED);
if (u->ship || u->building) leave(u->region, u);
translist(&u->region->units, ulist, u);
} else
addlist(ulist, u);
u->faction->first = 0;
u->faction->last = 0;
u->region = r;
}
/* ist mist, aber wegen nicht skalierender attirbute notwendig: */
#include "alchemy.h"
void
transfermen(unit * u, unit * u2, int n)
{
#if SKILLPOINTS
2001-01-25 10:37:55 +01:00
int skills[MAXSKILLS];
#endif
const attrib * a;
int hp = u->hp;
2001-01-25 10:37:55 +01:00
region * r = u->region;
if (!n) return;
/* "hat attackiert"-status wird <20>bergeben */
if (u2) {
#if SKILLPOINTS
skill_t sk;
if (u->skills) {
for (sk = 0; sk < MAXSKILLS; ++sk) {
2001-01-25 10:37:55 +01:00
skills[sk] = get_skill(u, sk);
}
}
#else
skill * sv;
for (sv=u->skills;sv!=u->skills+u->skill_size;++sv) {
if (u2->number == 0) {
set_skill(u2, sv->id, sv->level, sv->learning);
} else {
skill * sn = get_skill(u2, sv->id);
sn->level = (unsigned char)((sv->level*n+sn->level*u2->number)/(u2->number+n));
if (sn->learning>sv->learning) sn->learning=sv->learning;
}
}
#endif
2001-01-25 10:37:55 +01:00
a = a_find(u->attribs, &at_effect);
while (a) {
effect_data * olde = (effect_data*)a->data.v;
change_effect(u2, olde->type, olde->value);
a = a->nexttype;
}
if (fval(u, FL_LONGACTION)) fset(u2, FL_LONGACTION);
2001-01-25 10:37:55 +01:00
if (u->attribs) {
transfer_curse(u, u2, n);
}
}
scale_number(u, u->number - n);
if (u2) {
#if SKILLPOINTS
skill_t sk;
#endif
2001-01-25 10:37:55 +01:00
set_number(u2, u2->number + n);
hp -= u->hp;
u2->hp += hp;
#if SKILLPOINTS
2001-01-25 10:37:55 +01:00
if (u->skills) {
for (sk = 0; sk < MAXSKILLS; ++sk) {
change_skill_transfermen(u2, sk, skills[sk] - get_skill(u, sk));
}
}
#endif
2001-01-25 10:37:55 +01:00
/* TODO: Das ist schnarchlahm! und geh<65>rt ncht hierhin */
a = a_find(u2->attribs, &at_effect);
while (a) {
attrib * an = a->nexttype;
2001-01-25 10:37:55 +01:00
effect_data * olde = (effect_data*)a->data.v;
int e = get_effect(u, olde->type);
if (e!=0) change_effect(u2, olde->type, -e);
a = an;
2001-01-25 10:37:55 +01:00
}
}
else if (r->land)
rsetpeasants(r, rpeasants(r) + n);
}
2001-05-27 10:37:56 +02:00
struct building *
inside_building(const struct unit * u)
2001-01-25 10:37:55 +01:00
{
if (u->building==NULL) return NULL;
if (!fval(u->building, BLD_WORKING)) {
/* Unterhalt nicht bezahlt */
return NULL;
} else if (u->building->size < u->building->type->maxsize) {
/* Geb<65>ude noch nicht fertig */
return NULL;
} else {
int p = 0, cap = buildingcapacity(u->building);
const unit * u2;
for (u2 = u->region->units; u2; u2 = u2->next) {
if (u2->building == u->building) {
p += u2->number;
if (u2 == u) {
if (p <= cap) return u->building;
return NULL;
}
if (p > cap) return NULL;
}
}
}
return NULL;
}
2001-02-10 15:18:01 +01:00
void
u_setfaction(unit * u, faction * f)
{
int cnt = u->number;
if (u->faction==f) return;
#ifndef NDEBUG
assert(u->debug_number == u->number);
#endif
if (u->faction) {
set_number(u, 0);
join_group(u, NULL);
}
if (u->prevF) u->prevF->nextF = u->nextF;
else if (u->faction) {
assert(u->faction->units==u);
u->faction->units = u->nextF;
}
if (u->nextF) u->nextF->prevF = u->prevF;
if (f!=NULL) {
u->nextF = f->units;
f->units = u;
}
else u->nextF = NULL;
if (u->nextF) u->nextF->prevF = u;
u->prevF = NULL;
u->faction = f;
if (cnt && f) set_number(u, cnt);
}
/* vorsicht Spr<70>che k<>nnen u->number == 0 (RS_FARVISION) haben! */
void
set_number(unit * u, int count)
2001-02-10 15:18:01 +01:00
{
assert (count >= 0);
#ifndef NDEBUG
assert (u->debug_number == u->number);
assert (u->faction != 0 || u->number > 0);
#endif
if (u->faction && u->race != u->faction->race && playerrace(u->race)
&& old_race(u->race) != RC_SPELL && old_race(u->race) != RC_SPECIAL
2001-02-10 15:18:01 +01:00
&& !(is_cursed(u->attribs, C_SLAVE, 0))){
u->faction->num_migrants += count - u->number;
}
u->faction->num_people += count - u->number;
u->number = count;
#ifndef NDEBUG
u->debug_number = count;
#endif
}
#if !SKILLPOINTS
boolean
learn_skill(const unit * u, skill_t sk, double chance)
{
/** rewrite me **/
assert(!"rewrite me!");
return 0;
}
skill *
get_skill(unit * u, skill_t sk)
{
skill * sv = u->skills;
while (sv!=u->skills+u->skill_size) {
assert(sv->level>=0 && sv->learning>=0);
if (sv->id==sk) return sv;
++sv;
}
return NULL;
}
boolean
has_skill(unit * u, skill_t sk)
{
skill * sv = u->skills;
while (sv!=u->skills+u->skill_size) {
assert(sv->level>=0 && sv->learning>=0);
if (sv->id==sk) {
return (sv->level>0 || sv->learning>0);
}
++sv;
}
return false;
}
#endif
static int
item_modification(const unit *u, skill_t sk, int val)
{
/* Presseausweis: *2 Spionage, 0 Tarnung */
if(sk == SK_SPY && get_item(u, I_PRESSCARD) >= u->number) {
val = val * 2;
}
if(sk == SK_STEALTH && get_item(u, I_PRESSCARD) >= u->number) {
val = 0;
}
return val;
}
static int
att_modification(const unit *u, skill_t sk)
{
curse *c;
unit *mage;
int result = 0;
result += get_curseeffect(u->attribs, C_ALLSKILLS, 0);
result += get_curseeffect(u->attribs, C_SKILL, (int)sk);
/* TODO hier kann nicht mit get/iscursed gearbeitet werden, da nur der
* jeweils erste vom Typ C_GBDREAM zur<EFBFBD>ckgegen wird, wir aber alle
* durchsuchen und aufaddieren m<EFBFBD>ssen */
if (is_cursed(u->region->attribs, C_GBDREAM, 0)){
attrib *a;
int bonus = 0, malus = 0;
int mod;
a = a_select(u->region->attribs, packids(C_GBDREAM, 0), cmp_oldcurse);
while(a) {
c = (curse*)a->data.v;
mage = c->magician;
mod = c->effect;
/* wir suchen jeweils den gr<67><72>ten Bonus und den gr<67><72>ten Malus */
if (mod > 0
&& (mage == NULL
|| allied(mage, u->faction, HELP_GUARD)))
{
if (mod > bonus ) bonus = mod;
} else if (mod < 0
&& (mage == NULL
|| !allied(mage, u->faction, HELP_GUARD)))
{
if (mod < malus ) malus = mod;
}
a = a_select(a->next, packids(C_GBDREAM, 0), cmp_oldcurse);
}
result = result + bonus + malus;
}
return result;
}
int
get_modifier(const unit * u, skill_t sk, int level, const region * r)
{
int bskill = level;
int skill = bskill;
if (r->planep && sk == SK_STEALTH && fval(r->planep, PFL_NOSTEALTH)) return 0;
assert(r);
skill += rc_skillmod(u->race, r, sk);
skill += att_modification(u, sk);
skill = item_modification(u, sk, skill);
skill = skillmod(u->attribs, u, r, sk, skill, SMF_ALWAYS);
if (fspecial(u->faction, FS_TELEPATHY)) {
switch(sk) {
case SK_ALCHEMY:
case SK_HERBALISM:
case SK_MAGIC:
case SK_SPY:
case SK_STEALTH:
case SK_OBSERVATION:
break;
default:
skill -= 2;
}
}
#ifdef HUNGER_REDUCES_SKILL
if (fval(u, FL_HUNGER)) {
skill = skill/2;
}
#endif
return skill - bskill;
}
int
eff_skill(const unit * u, skill_t sk, const region * r)
{
int level = get_level(u, sk);
int mlevel = level - get_modifier(u, sk, level, r);
#if SKILLPOINTS
if (mlevel<0) return 0;
#endif
return mlevel;
}