server/src/common/kernel/curse.c

1672 lines
36 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
2001-01-31 18:40:53 +01:00
* $Id: curse.c,v 1.3 2001/01/31 17:40:49 corwin Exp $
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 "curse.h"
/* kernel includes */
#include "magic.h"
#include "unit.h"
#include "region.h"
#include "race.h"
#include "faction.h"
#include "building.h"
#include "objtypes.h"
/* util includes */
#include <resolve.h>
#include <base36.h>
/* libc includes */
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <limits.h>
#include <assert.h>
#include <math.h>
/* ------------------------------------------------------------- */
direction_t
dirmirror(direction_t dir)
{
return (direction_t)((dir+MAXDIRECTIONS/2) % MAXDIRECTIONS);
}
/* ------------------------------------------------------------- */
/* at_curse */
void
curse_init(attrib * a) {
a->data.v = calloc(1, sizeof(curse));
}
int
curse_age(attrib * a)
{
curse * c = (curse*)a->data.v;
if (c->flag & CURSE_NOAGE) {
c->duration = 1;
} else {
c->duration = max(0, c->duration-1);
}
return (c->duration);
}
void
curse_done(attrib * a) {
curse *c = (curse *)a->data.v;
if( c->data )
free(c->data);
free(c);
}
attrib_type at_curse =
{
"curse",
curse_init,
curse_done,
curse_age,
curse_write,
curse_read,
ATF_CURSE
};
/* ------------------------------------------------------------- */
#define MAXENTITYHASH 8191
curse *cursehash[MAXENTITYHASH];
void
chash(curse *c)
{
curse *old = cursehash[c->no %MAXENTITYHASH];
cursehash[c->no %MAXENTITYHASH] = c;
c->nexthash = old;
}
void
cunhash(curse *c)
{
curse **show;
for (show = &cursehash[c->no % MAXENTITYHASH]; *show; show = &(*show)->nexthash) {
if ((*show)->no == c->no)
break;
}
if (*show) {
assert(*show == c);
*show = (*show)->nexthash;
c->nexthash = 0;
}
}
curse *
cfindhash(int i)
{
curse *old;
for (old = cursehash[i % MAXENTITYHASH]; old; old = old->nexthash)
if (old->no == i)
return old;
return 0;
}
/* ------------------------------------------------------------- */
/* Spruch identifizieren */
#include "umlaut.h"
extern struct tnode cursenames;
void
ct_register(curse_type * ct) {
assert(0);
}
const curse_type *
ct_find(const char *c)
{
/* TODO: findet nur curse_types, die auch in curse_data sind.
* da fehlt noch ene registrierung wie f<EFBFBD>r attrib_type
*/
int i = (int)findtoken(&cursenames, c)-1;
if (cursedaten[i].name[0] == 0) return NULL;
return &cursedaten[i];
}
int
find_cursebyname(const char *s)
{
int i;
i = (int)findtoken(&cursenames, s)-1;
if(cursedaten[i].name[0] == 0)
return -1;
return i;
}
/* ------------------------------------------------------------- */
boolean
is_normalcurse(curse_t id)
{
if (cursedaten[id].typ == CURSETYP_NORM)
return true;
return false;
}
boolean
is_curseunit(curse_t id)
{
if (cursedaten[id].typ == CURSETYP_UNIT)
return true;
return false;
}
boolean
is_curseskill(curse_t id)
{
if (cursedaten[id].typ == CURSETYP_SKILL)
return true;
return false;
}
/* ------------------------------------------------------------- */
/* get_curse identifiziert eine Verzauberung <20>ber die ID und gibt
* einen pointer auf die struct zur<EFBFBD>ck.
*/
typedef struct twoids {
int id;
int id2;
} twoids;
boolean
cmp_curse(const attrib * a, void * data) {
if (a->type->flags & ATF_CURSE) {
if (!data || (curse*)data == (curse*)a->data.v) return true;
}
return false;
}
boolean
cmp_oldcurse(const attrib * a, void * data)
{
twoids * ti = (twoids*)data;
curse * c = (curse*)a->data.v;
if (a->type!=&at_curse || c->cspellid!=ti->id) return false;
switch(cursedaten[ti->id].typ) {
case CURSETYP_SKILL:
{
curse_skill * cc = (curse_skill*)c->data;
if (cc->skill == (skill_t) ti->id2) return true;
}
case CURSETYP_SECONDID:
{
curse_secondid * cc = (curse_secondid*)c->data;
if (cc->secondid == ti->id2) return true;
}
case CURSETYP_UNIT:
case CURSETYP_NORM:
return true;
break;
default:
assert(!"unbekannter cursetyp entdeckt");
}
return false;
}
twoids *
packids(int id, int id2) {
static twoids ti;
ti.id = id;
ti.id2 = id2;
return &ti;
}
curse *
get_curse(attrib *ap, curse_t id, int id2)
{
attrib * a = a_select(ap, packids(id, id2), cmp_oldcurse);
if (!a) return NULL;
return (curse*)a->data.v;
}
/* ------------------------------------------------------------- */
/* Normalerweise ist alles ausser der Id eines curse und dem
* verzauberten Objekt nicht bekannt. Um den Zauber eindeutig zu
* identifizieren ben<EFBFBD>tigt man je nach Typ einen weiteren Identifier.
*/
void
remove_curse(attrib **ap, curse_t id, int id2)
{
attrib *a = a_select(*ap, packids(id, id2), cmp_oldcurse);
if (a) a_remove(ap, a);
}
void
remove_allcurse(attrib **ap, void * data, boolean(*compare)(const attrib *, void *))
{
attrib * a = a_select(*ap, data, compare);
while (a) {
attrib * next = a->nexttype;
a_remove(ap, a);
a = a_select(next, data, compare);
}
}
/* ------------------------------------------------------------- */
/*
* Staerke einer Verzauberung
*/
int
get_cursevigour(attrib *ap, curse_t id, int id2)
{
int vigour = 0;
curse * c = get_curse(ap, id, id2);
if (c) vigour = c->vigour;
return vigour;
}
void
set_cursevigour(attrib *ap, curse_t id, int id2, int i)
{
curse * c = get_curse(ap, id, id2);
assert(i>0);
if (c) c->vigour = i;
}
/* ver<65>ndert die St<53>rke der Verzauberung um +i und gibt die neue
* St<EFBFBD>rke zur<EFBFBD>ck. Sollte die Zauberst<EFBFBD>rke unter Null sinken, l<EFBFBD>st er
* sich auf.
*/
int
change_cursevigour(attrib **ap, curse_t id, int id2, int i)
{
i += get_cursevigour(*ap,id,id2);
if (i <= 0) {
remove_curse(ap, id, id2);
}else{
set_cursevigour(*ap,id,id2,i);
}
return i;
}
/* ------------------------------------------------------------- */
int
get_curseeffect(attrib *ap, curse_t id, int id2)
{
int effect = 0;
curse * c = get_curse(ap, id, id2);
if (c) effect = c->effect;
return effect;
}
/* ------------------------------------------------------------- */
void
set_curseingmagician(struct unit *magician, struct attrib *ap_target, curse_t id, int id2)
{
curse * c = get_curse(ap_target, id, id2);
if (c){
c->magician = magician;
}
}
unit *
get_cursingmagician(struct attrib *ap, curse_t id, int id2)
{
curse *c = get_curse(ap, id, id2);
if( !c )
return NULL;
return c->magician;
}
/* Wichtig! Alle struct curse<typ>, die den Verweis auf den zaubenden
* Magier enthalten, m<EFBFBD>ssen sie hier mit abgepr<EFBFBD>ft werden
*/
void
remove_cursemagepointer(unit *magician, attrib *ap_target)
{
const attrib * a;
a = a_find(ap_target, &at_curse);
while (a) {
curse *c = (curse*)a->data.v;
if(c->magician == magician)
c->magician = (unit*)NULL;
a = a->nexttype;
}
}
/* ------------------------------------------------------------- */
int
get_cursedmen(attrib *ap, curse_t id, int id2)
{
switch (cursedaten[id].typ){
case CURSETYP_UNIT:
{
curse *c = get_curse(ap, id, id2);
if (c){
curse_unit * cc = (curse_unit*)c->data;
return cc->cursedmen;
}
}
case CURSETYP_SKILL:
{
curse *c = get_curse(ap, id, id2);
if (c){
curse_skill * cc = (curse_skill*)c->data;
return cc->cursedmen;
}
}
case CURSETYP_NORM:
default:
break;
}
return 0;
}
void
set_cursedmen(attrib *ap, curse_t id, int id2, int cursedmen)
{
switch (cursedaten[id].typ){
case CURSETYP_UNIT:
{
curse *c = get_curse(ap, id, id2);
if (c){
curse_unit * cc = (curse_unit*)c->data;
cc->cursedmen = cursedmen;
}
break;
}
case CURSETYP_SKILL:
{
curse *c = get_curse(ap, id, id2);
if (c){
curse_skill * cc = (curse_skill*)c->data;
cc->cursedmen = cursedmen;
}
break;
}
case CURSETYP_NORM:
default:
break;
}
}
int
change_cursedmen(attrib **ap, curse_t id, int id2, int cursedmen)
{
switch (cursedaten[id].typ){
case CURSETYP_UNIT:
case CURSETYP_SKILL:
cursedmen += get_cursedmen(*ap, id, id2);
if (cursedmen <= 0){
remove_curse(ap,id,id2);
} else {
set_cursedmen(*ap, id, id2, cursedmen);
}
break;
case CURSETYP_NORM:
default:
break;
}
return cursedmen;
}
/* ------------------------------------------------------------- */
void
set_curseflag(attrib *ap, curse_t id, int id2, int flag)
{
curse * c = get_curse(ap, id, id2);
if (c) c->flag = (c->flag | flag);
}
void
remove_curseflag(attrib *ap, curse_t id, int id2, int flag)
{
curse * c = get_curse(ap, id, id2);
if (c) c->flag = (c->flag & ~(flag));
}
/* ------------------------------------------------------------- */
/* Legt eine neue Verzauberung an. Sollte es schon einen Zauber
* dieses Typs geben, gibt es den bestehenden zur<EFBFBD>ck.
*/
curse *
set_curse(unit *mage, attrib **ap, curse_t id, int id2, int vigour,
int duration, int effect, int men)
{
curse *c;
attrib * a;
a = a_new(&at_curse);
a_add(ap, a);
c = (curse*)a->data.v;
c->type = &cursedaten[id];
c->cspellid = id;
c->flag = (0);
c->vigour = vigour;
c->duration = duration;
c->effect = effect;
c->magician = mage;
c->no = newunitid();
chash(c);
switch (c->type->typ){
case CURSETYP_NORM:
break;
case CURSETYP_UNIT:
{
curse_unit *cc = calloc(1, sizeof(curse_unit));
cc->cursedmen += men;
c->data = cc;
break;
}
case CURSETYP_SKILL:
{
curse_skill *cc = calloc(1, sizeof(curse_skill));
cc->skill = (skill_t) id2;
cc->cursedmen += men;
c->data = cc;
break;
}
case CURSETYP_SECONDID:
{
curse_secondid *cc = calloc(1, sizeof(curse_secondid));
cc->secondid = id2;
c->data = cc;
break;
}
}
return c;
}
/* Mapperfunktion f<>r das Anlegen neuer curse. Automatisch wird zum
* passenden Typ verzweigt und die relevanten Variablen weitergegeben.
*/
curse *
create_curse(unit *magician, attrib **ap, curse_t id, int id2, int vigour,
int duration, int effect, int men)
{
curse *c;
c = get_curse(*ap, id, id2);
if(c && (c->flag & CURSE_ONLYONE)){
return NULL;
}
/* es gibt schon eins diese Typs */
if (c && cursedaten[id].mergeflags != NO_MERGE) {
if(cursedaten[id].mergeflags & M_DURATION){
c->duration = max(c->duration, duration);
}
if(cursedaten[id].mergeflags & M_SUMDURATION){
c->duration += duration;
}
if(cursedaten[id].mergeflags & M_SUMEFFECT){
c->effect += effect;
}
if(cursedaten[id].mergeflags & M_MAXEFFECT){
c->effect = max(c->effect, effect);
}
if(cursedaten[id].mergeflags & M_VIGOUR){
c->vigour = max(vigour, c->vigour);
}
2001-01-31 18:40:53 +01:00
if(cursedaten[id].mergeflags & M_VIGOUR_ADD){
c->vigour = vigour + c->vigour;
}
2001-01-25 10:37:55 +01:00
if(cursedaten[id].mergeflags & M_MEN){
switch (cursedaten[id].typ) {
case CURSETYP_UNIT:
{
curse_unit * cc = (curse_unit*)c->data;
cc->cursedmen += men;
}
case CURSETYP_SKILL:
{
curse_skill * cc = (curse_skill*)c->data;
cc->cursedmen += men;
}
}
}
set_curseingmagician(magician, *ap, id, id2);
} else {
c = set_curse(magician, ap, id, id2, vigour, duration, effect, men);
}
return c;
}
/* ------------------------------------------------------------- */
/* hier m<>ssen alle c-typen, die auf Einheiten gezaubert werden k<>nnen,
* ber<EFBFBD>cksichtigt werden */
void
do_transfer_curse(curse *c, unit * u, unit * u2, int n)
{
curse_t id = c->cspellid;
int id2 = 0;
int flag = c->flag;
int duration = c->duration;
int vigour = c->vigour;
unit *magician = c->magician;
int effect = c->effect;
int cursedmen = 0;
int men = 0;
boolean dogive = false;
switch (cursedaten[id].typ) {
case CURSETYP_UNIT:
{
curse_unit * cc = (curse_unit*)c->data;
men = cc->cursedmen;
break;
}
case CURSETYP_SKILL:
{
curse_skill * cc = (curse_skill*)c->data;
id2 = (int)cc->skill;
men = cc->cursedmen;
break;
}
default:
cursedmen = u->number;
}
switch (cursedaten[id].givemenacting){
case CURSE_SPREADALWAYS:
dogive = true;
men = u2->number + n;
break;
case CURSE_SPREADMODULO:
{
int i;
int u_number = u->number;
for (i=0;i<n+1 && u_number>0;i++){
if (rand()%u_number < cursedmen){
++men;
--cursedmen;
dogive = true;
}
--u_number;
}
break;
}
case CURSE_SPREADCHANCE:
{
int chance = u2->number * 100 / (u2->number + n);
if (rand()%100 > chance ){
men = u2->number + n;
dogive = true;
}
break;
}
case CURSE_SPREADNEVER:
break;
}
if (dogive == true) {
set_curse(magician, &u2->attribs, id, id2, vigour, duration,
effect, men);
set_curseflag(u2->attribs, id, id2, flag);
switch (cursedaten[id].typ){
case CURSETYP_UNIT:
case CURSETYP_SKILL:
set_cursedmen(u2->attribs, id, id2, men);
break;
}
}
}
void
transfer_curse(unit * u, unit * u2, int n)
{
attrib * a;
a = a_find(u->attribs, &at_curse);
while (a) {
curse *c = (curse*)a->data.v;
do_transfer_curse(c, u, u2, n);
a = a->nexttype;
}
}
/* ------------------------------------------------------------- */
boolean
is_cursed(attrib *ap, curse_t id, int id2)
{
curse *c = get_curse(ap, id, id2);
if (!c)
return false;
if (c->flag & CURSE_ISNEW)
return false;
return true;
}
/* ------------------------------------------------------------- */
/* Diese Funktionen werden von reports.c:print_curses() w<>hrend der
* Generierung des Normalreports aufgerufen und ersetzen
* cursedisplay
*/
static int
cinfo_fogtrap(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Dichte Nebel bedecken diese Woche die Region. "
"Keine Einheit schafft es, diese Nebel zu durchdringen und "
"die Region zu verlassen. (%s)", curseid(c));
return 1;
}
/* C_NOCOST */
static int
cinfo_nocost(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_BUILDING);
sprintf(buf, "Der Zahn der Zeit kann diesen Mauern nichts anhaben. (%s)",
curseid(c));
return 1;
}
2001-01-31 18:40:53 +01:00
/* C_HOLYGROUND */
static int
cinfo_holyground(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_REGION);
sprintf(buf, "Untote schrecken vor dieser Region zur<75>ck. (%s)",
curseid(c));
return 1;
}
2001-01-25 10:37:55 +01:00
/* C_GBDREAM, */
static int
cinfo_dreamcurse(void * obj, typ_t typ, curse *c, int self)
{
unused(self);
unused(typ);
unused(obj);
assert(typ == TYP_REGION);
if (c->effect > 0){
sprintf(buf, "Die Leute haben sch<63>ne Tr<54>ume. (%s)", curseid(c));
}else{
sprintf(buf, "Albtr<EFBFBD>ume plagen die Leute. (%s)", curseid(c));
}
return 1;
}
/* C_AURA */
/* erh<72>ht/senkt regeneration und maxaura um effect% */
static int
cinfo_auraboost(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
if (self){
if (c->effect > 100){
sprintf(buf, "%s f<>hlt sich von starken magischen Energien "
"durchstr<EFBFBD>mt. (%s)", u->name, curseid(c));
}else{
sprintf(buf, "%s hat Schwierigkeiten seine magischen Energien "
"zu sammeln. (%s)", u->name, curseid(c));
}
return 1;
}
return 0;
}
/* C_BLESSEDHARVEST, */
static int
cinfo_blessedharvest(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_REGION);
sprintf(buf, "In dieser Gegend steht das Korn besonders gut im Feld. (%s)", curseid(c));
return 1;
}
/* C_DROUGHT, */
static int
cinfo_drought(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_REGION);
sprintf(buf, "In dieser Gegend herrscht eine D<>rre. (%s)", curseid(c));
return 1;
}
/* C_BADLEARN */
static int
cinfo_badlearn(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Alle Leute in der Region haben Schlafst<73>rungen. (%s)", curseid(c));
return 1;
}
/* C_SHIP_NODRIFT */
static int
cinfo_shipnodrift(void * obj, typ_t typ, curse *c, int self)
{
ship * sh;
unused(typ);
assert(typ == TYP_SHIP);
sh = (ship*)obj;
if (self){
sprintf(buf, "%s ist mit guten Wind gesegnet", sh->name);
if (c->duration <= 2){
scat(", doch der Zauber beginnt sich bereits aufzul<75>sen");
}
scat(".");
} else {
sprintf(buf, "Ein silberner Schimmer umgibt das Schiff.");
}
scat(" (");
scat(itoa36(c->no));
scat(")");
return 1;
}
/* C_DEPRESSION */
static int
cinfo_depression(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_REGION);
sprintf(buf, "Die Bauern sind unzufrieden. (%s)", curseid(c));
return 1;
}
/* C_MAGICSTONE*/
static int
cinfo_magicstone(void * obj, typ_t typ, curse *c, int self)
{
building * b;
unused(typ);
assert(typ == TYP_BUILDING);
b = (building*)obj;
if (self){
sprintf(buf, "Die Mauern von %s wirken, als w<>ren sie "
"aus der Erde gewachsen. (%s)", b->name, curseid(c));
return 1;
}
return 0;
}
/* C_MAGICSTONE*/
static int
cinfo_magicrunes(void * obj, typ_t typ, curse *c, int self)
{
if (typ == TYP_BUILDING){
building * b;
b = (building*)obj;
if (self){
sprintf(buf, "Auf den Mauern von %s erkennt man seltsame Runen. (%s)",
b->name, curseid(c));
return 1;
}
} else if (typ == TYP_SHIP) {
ship *sh;
sh = (ship*)obj;
if (self){
sprintf(buf, "Auf den Planken von %s erkennt man seltsame Runen. (%s)",
sh->name, curseid(c));
return 1;
}
}
return 0;
}
/* C_GENEROUS */
static int
cinfo_generous(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Es herrscht eine fr<66>hliche und ausgelassene Stimmung. (%s)",
curseid(c));
return 1;
}
/* C_ASTRALBLOCK */
static int
cinfo_astralblock(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Etwas verhindert den Kontakt zur Realit<69>t. (%s)", curseid(c));
return 1;
}
/* C_PEACE */
static int
cinfo_peace(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_REGION);
sprintf(buf, "Die ganze Region ist von einer friedlichen Stimmung "
"erfasst. (%s)", curseid(c));
return 1;
}
/* C_REGCONF */
static int
cinfo_regconf(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Ein Schleier der Verwirrung liegt <20>ber der Region. (%s)",
curseid(c));
return 1;
}
/* C_MAGICSTREET */
static int
cinfo_magicstreet(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Die Stra<72>en sind erstaunlich trocken und gut begehbar");
/* Warnung vor aufl<66>sung!*/
if (c->duration <= 2){
scat(", doch an manchen Stellen bilden sich wieder die erste Schlamml<6D>cher");
}
scat(". (");
scat(itoa36(c->no));
scat(")");
return 1;
}
/* C_SLAVE */
static int
cinfo_slave(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
if (self){
sprintf(buf, "%s wird noch %d Woche%s unter unserem Bann stehen. (%s)",
u->name, c->duration, (c->duration == 1)? "":"n", curseid(c));
return 1;
}
return 0;
}
/* C_DISORIENTATION */
static int
cinfo_disorientation(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(obj);
unused(self);
assert(typ == TYP_SHIP);
sprintf(buf, "Der Kompa<70> kaputt, die Segel zerrissen, der Himmel "
"wolkenverhangen. Wohin fahren wir? (%s)", curseid(c));
return 1;
}
/* C_CALM */
static int
cinfo_calm(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
race_t rc;
faction *f;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
if (c->magician){
rc = c->magician->irace;
f = c->magician->faction;
if (self){
sprintf(buf, "%s mag %s", u->name, factionname(f));
}else{
sprintf(buf, "%s scheint %s zu m<>gen", u->name, race[rc].name[1]);
}
scat(". (");
scat(itoa36(c->no));
scat(")");
return 1;
}
return 0;
}
/* C_SPEED */
static int
cinfo_speed(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
curse_unit * cu;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
cu = (curse_unit *)c->data;
if (self){
sprintf(buf, "%d Person%s von %s %s noch %d Woche%s beschleunigt. (%s)",
cu->cursedmen, (cu->cursedmen == 1)? "":"en", u->name,
(cu->cursedmen == 1)? "ist":"sind", c->duration,
(c->duration == 1)? "":"n",
curseid(c));
return 1;
}
return 0;
}
/* C_ORC */
static int
cinfo_orc(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
if (self){
sprintf(buf, "%s st<73>rzt sich von einem amour<75>sen Abenteuer ins "
"n<EFBFBD>chste. (%s)", u->name, curseid(c));
return 1;
}
return 0;
}
/* C_KAELTESCHUTZ */
static int
cinfo_kaelteschutz(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
curse_unit * cu;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
cu = (curse_unit *)c->data;
if (self){
sprintf(buf, "%d Person%s von %s %s sich vor K<>lte gesch<63>tzt. (%s)",
cu->cursedmen, (cu->cursedmen == 1)? "":"en", u->name,
(cu->cursedmen == 1)? "f<EFBFBD>hlt":"f<EFBFBD>hlen",
curseid(c));
return 1;
}
return 0;
}
/* C_SPARKLE */
static int
cinfo_sparkle(void * obj, typ_t typ, curse *c, int self)
{
const char * effects[] = {
NULL, /* end grau*/
"%s ist im Traum eine Fee erschienen.",
"%s wird von b<>sen Alptr<74>umen geplagt.",
NULL, /* end traum */
"%s wird von einem glitzernden Funkenregen umgeben.",
"Ein schimmernder Lichterkranz umgibt %s.",
NULL, /* end tybied */
"Eine Melodie erklingt, und %s tanzt bis sp<73>t in die Nacht hinein.",
"%s findet eine kleine Fl<46>te, die eine wundersame Melodie spielt.",
"Die Frauen des nahegelegenen Dorfes bewundern %s verstohlen.",
"Eine Gruppe vorbeiziehender Bergarbeiter rufen %s eindeutig Zweideutiges nach.",
NULL, /* end cerrdor */
"%s bekommt von einer Schlange einen Apfel angeboten.",
"Ein Einhorn ber<65>hrt %s mit seinem Horn und verschwindet kurz darauf im Unterholz.",
"Vogelzwitschern begleitet %s auf all seinen Wegen.",
"Leuchtende Blumen erbl<62>hen rund um das Lager von %s.",
NULL, /* end gwyrrd */
"<EFBFBD>ber %s zieht eine Gruppe Geier ihre Kreise.",
"Der Kopf von %s hat sich in einen grinsenden Totensch<63>del verwandelt.",
"Ratten folgen %s auf Schritt und Tritt.",
"Pestbeulen befallen den K<>rper von %s.",
"Eine dunkle Fee erscheint %s im Schlaf. Sie ist von schauriger Sch<63>nheit.",
"F<EFBFBD>ulnisgeruch dringt %s aus allen K<>rper<65>ffnungen.",
NULL, /* end draig */
};
int m, begin=0, end=0;
unit *u;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
if(!c->magician || !c->magician->faction) return 0;
for(m=0;m!=c->magician->faction->magiegebiet;++m) {
while (effects[end]!=NULL) ++end;
begin = end+1;
end = begin;
}
while (effects[end]!=NULL) ++end;
if (end==begin) return 0;
else sprintf(buf, effects[begin + c->effect % (end-begin)], u->name);
scat(" (");
scat(itoa36(c->no));
scat(")");
return 1;
}
/* C_STRENGTH */
static int
cinfo_strength(void * obj, typ_t typ, curse *c, int self)
{
unused(c);
unused(typ);
assert(typ == TYP_UNIT);
unused(obj);
if (self){
sprintf(buf, "Die Leute strotzen nur so vor Kraft. (%s)",
curseid(c));
return 1;
}
return 0;
}
/* C_ALLSKILLS */
static int
cinfo_allskills(void * obj, typ_t typ, curse *c, int self)
{
unused(obj);
unused(typ);
unused(c);
assert(typ == TYP_UNIT);
if (self){
sprintf(buf, "Wird von einem Alp geritten. (%s)", curseid(c));
return 1;
}
return 0;
}
/* C_SKILL */
static int
cinfo_skill(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
curse_skill *ck;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
ck = (curse_skill*)c->data;
if (self){
sprintf(buf, "%s ist in %s ungew<65>hnlich ungeschickt. (%s)", u->name,
skillnames[ck->skill], curseid(c));
return 1;
}
return 0;
}
/* C_ITEMCLOAK */
static int
cinfo_itemcloak(void * obj, typ_t typ, curse *c, int self)
{
unit *u;
unused(typ);
assert(typ == TYP_UNIT);
u = (unit *)obj;
if (self) {
sprintf(buf, "Die Ausr<73>stung von %s scheint unsichtbar. (%s)",
u->name, curseid(c));
return 1;
}
return 0;
}
static int
cinfo_fumble(void * obj, typ_t typ, curse *c, int self)
{
unit * u = (unit*)obj;
unused(typ);
assert(typ == TYP_UNIT);
if (self){
sprintf(buf, "%s kann sich kaum konzentrieren.i (%s)",
u->name, curseid(c));
return 1;
}
return 0;
}
/* C_RIOT */
static int
cinfo_riot(void * obj, typ_t typ, curse *c, int self)
{
unused(typ);
unused(self);
unused(obj);
assert(typ == TYP_REGION);
sprintf(buf, "Die Region befindet sich in Aufruhr.(%s)", curseid(c));
return 1;
}
/* ------------------------------------------------------------- */
/* cursedata */
/* ------------------------------------------------------------- */
/* typedef struct cursedata {
* int typ;
* int givemenacting;
* int mergeflags;
* char *name;
* char *info;
* void (*display)(void*,typ_t, curse*);
*} cursedata;
*/
/* die Beschreibung wird bei einer gelungenen Zauberanalyse ausgegeben
* und hat die Form:
* Magier (xx) gelang es folgendes herauszufinden:
* Unit (xyz) steht unter dem Einfluss des Zaubers %name,/ Auf
* Region/Schiff/Burg (xy) liegt der Zauber %name,
* der wohl noch etwa %s Wochen andauert.
* %info, "Dieser Zauber blafalsel blub"
*/
cursedata cursedaten[MAXCURSE] =
{
/* struct's vom typ curse: */
{ /* C_FOGTRAP, */
CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR),
"",
"",
(cdesc_fun)cinfo_fogtrap
},
{ /* C_ANTIMAGICZONE, */
CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR),
"Magieschw<EFBFBD>chezone",
"Dieser Zauber scheint magische Energien irgendwie abzuleiten und "
"so alle in der Region gezauberten Spr<70>che in ihrer Wirkung zu "
"schw<EFBFBD>chen oder ganz zu verhindern.",
NULL
},
{ /* C_FARVISION, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_GBDREAM, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
(cdesc_fun)cinfo_dreamcurse
},
{ /* C_AURA, Ver<65>ndert die max Aura und Regeneration um effect% */
CURSETYP_NORM, CURSE_SPREADMODULO, (NO_MERGE),
"Aurafokus",
"Dieser Zauber greift irgendwie in die Verbindung zwischen Magier "
"und Magischer Essenz ein. Mit positiver Ausrichtung kann er wohl "
"wie ein Fokus f<>r Aura wirken, jedoch genauso f<>r das Gegenteil "
"benutzt werden.",
(cdesc_fun)cinfo_auraboost
},
{ /* C_MAELSTROM, */
CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR),
"Mahlstrom",
"Dieser Zauber verursacht einen gigantischen magischen Strudel. Der "
"Mahlstrom wird alle Schiffe, die in seinen Sog geraten, schwer "
"besch<EFBFBD>digen.",
NULL
},
{ /* C_BLESSEDHARVEST, */
CURSETYP_NORM, 0, ( M_DURATION | M_VIGOUR ),
"Segen der Felder",
"Dieser Fruchtbarkeitszauber erh<72>ht die Ertr<74>ge der Felder.",
(cdesc_fun)cinfo_blessedharvest
},
{ /* C_DROUGHT, */
CURSETYP_NORM, 0, ( M_DURATION | M_VIGOUR ),
"D<EFBFBD>rre",
"Dieser Zauber strahlt starke negative Energien aus. Warscheinlich "
"ist er die Ursache der D<>rre." ,
(cdesc_fun)cinfo_drought
},
{ /* C_BADLEARN, */
CURSETYP_NORM, 0, ( M_DURATION | M_VIGOUR ),
"",
"Dieser Zauber scheint die Ursache f<>r die Schlaflosigkeit und "
"Mattigkeit zu sein, unter der die meisten Leute hier leiden und "
"die dazu f<>hrt, das Lernen weniger Erfolg bringt. ",
(cdesc_fun)cinfo_badlearn
},
{ /* C_SHIP_SPEEDUP, Sturmwind-Zauber, wirkt nur 1 Runde */
CURSETYP_NORM, 0, NO_MERGE,
"Sturmwind",
"",
NULL
},
{ /* C_SHIP_FLYING, Luftschiff-Zauber, wirkt nur 1 Runde */
CURSETYP_NORM, 0, NO_MERGE,
"Luftschiff",
"",
NULL
},
{ /* C_SHIP_NODRIFT, G<>nstigeWinde-Zauber */
CURSETYP_NORM, 0, ( M_DURATION | M_VIGOUR ),
"G<EFBFBD>nstige Winde",
"Der Zauber auf diesem Schiff ist aus den elementaren Magien der Luft "
"und des Wassers gebunden. Der dem Wasser verbundene Teil des Zaubers "
"l<EFBFBD><EFBFBD>t es leichter durch die Wellen gleiten und der der Luft verbundene "
"Teil scheint es vor widrigen Winden zu sch<63>tzen.",
(cdesc_fun)cinfo_shipnodrift
},
{ /* C_DEPRESSION, Tr<54>bsal-Zauber */
CURSETYP_NORM, 0, ( M_DURATION | M_VIGOUR ),
"Schwermut",
"Wie schon zu vermuten war, sind der ewig graue Himmel und die "
"depressive Stimmung in der Region nicht nat<61>rlich. Dieser Fluch "
"hat sich wie ein bleiernes Tuch auf die Gem<65>ter der Bev<65>lkerung "
"gelegt und eh er nicht gebrochen oder verklungen ist, wird keiner "
"sich an Gaukelleien erfreuen k<>nnen.",
(cdesc_fun)cinfo_depression
},
{ /* C_MAGICSTONE, Heimstein-Zauber */
CURSETYP_NORM, 0, NO_MERGE,
"Felsenmauern",
"Die Macht dieses Zaubers ist fast greifbar und tief in die Mauern "
"gebunden. Starke elementarmagische Kr<4B>fte sind zu sp<73>ren. "
"Vieleicht wurde gar ein Erdelementar in diese Mauern gebannt. "
"Ausser ebenso starkter Antimagie wird nichts je diese Mauern "
"gef<EFBFBD>hrden k<>nnen.",
(cdesc_fun)cinfo_magicstone
},
{ /* C_STRONGWALL, Feste Mauer - Pr<50>kampfzauber, wirkt nur 1 Runde */
CURSETYP_NORM, 0, NO_MERGE,
"Burgschutzzauber",
"",
NULL
},
{ /* C_ASTRALBLOCK, Astralblock, auf Astralregion */
CURSETYP_NORM, 0, NO_MERGE,
"",
"",
(cdesc_fun)cinfo_astralblock
},
{ /* C_GENEROUS, Unterhaltungsanteil vermehren */
CURSETYP_NORM, 0, ( M_DURATION | M_VIGOUR | M_MAXEFFECT ),
"Feierlaune",
"Dieser Zauber beeinflusst die allgemeine Stimmung in der Region positiv. "
"Die gute Laune macht die Leute freigiebiger.",
(cdesc_fun)cinfo_generous
},
{ /* C_PEACE, verhindert Attackiere regional */
CURSETYP_NORM, 0, NO_MERGE,
"Friedensbann",
"Dieser machtvoller Beeinflussungszauber erstickt jeden Streit schon im "
"Keim.",
(cdesc_fun)cinfo_peace
},
{ /* C_REGCONF, erschwert geordnete Bewegungen */
CURSETYP_NORM, 0, NO_MERGE,
"",
"",
(cdesc_fun)cinfo_regconf
},
{ /* C_MAGICSTREET, erzeugt Stra<72>ennetz */
CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR),
"magische Pfade",
"Es scheint sich um einen elementarmagischen Zauber zu handeln, der alle "
"Pfade und Wege so gut festigt, als w<>ren sie gepflastert. Wie auf einer "
"Stra<EFBFBD>e kommt man so viel besser und schneller vorw<72>rts.",
(cdesc_fun)cinfo_magicstreet
},
{ /* C_RESIST_MAGIC, */
CURSETYP_NORM, 0, M_SUMEFFECT,
"Magieschutz",
"Dieses Zauber verst<73>rkt die nat<61>rliche Widerstandskraft gegen eine "
"Verzauberung.",
(cdesc_fun)cinfo_magicrunes
},
{ /* C_SONG_BADMR,
erniedigt Magieresistenz von nicht-aliierten Einheiten, wirkt nur
1x pro Einheit */
CURSETYP_NORM, 0, NO_MERGE,
"Gesang des schwachen Geistes",
"Dieses Lied, das irgendwie in die magische Essenz der Region gewoben "
"ist, schw<68>cht die nat<61>rliche Widerstandskraft gegen eine "
"Verzauberung. Es scheint jedoch nur auf bestimmte Einheiten zu wirken.",
NULL
},
{ /* C_SONG_GOODMR,
erh<EFBFBD>ht Magieresistenz von aliierten Einheiten, wirkt nur 1x pro
Einheit */
CURSETYP_NORM, 0, NO_MERGE,
"Gesang des wachen Geistes",
"Dieser Lied, das irgendwie in die magische Essenz der Region gewoben "
"ist, verst<73>rkt die nat<61>rliche Widerstandskraft gegen eine "
"Verzauberung. Es scheint jedoch nur auf bestimmte Einheiten zu wirken.",
NULL
},
{ /* C_SLAVE,
dient fremder Partei. Z<EFBFBD>hlt nicht zu Migranten, attackiert nicht */
CURSETYP_NORM, 0, NO_MERGE,
"Sklavenketten",
"Dieser m<>chtige Bann scheint die Einheit ihres freien Willens "
"zu berauben. Solange der Zauber wirkt, wird sie nur den Befehlen "
"ihres neuen Herrn gehorchen.",
(cdesc_fun)cinfo_slave
},
{ /* C_DISORIENTATION, */
CURSETYP_NORM, 0, NO_MERGE,
"Irrfahrt",
"Dieses Schiff hat sich verfahren.",
(cdesc_fun)cinfo_disorientation
},
{ /* C_CALM, */
CURSETYP_NORM, CURSE_SPREADNEVER, NO_MERGE,
"Bes<EFBFBD>nftigen",
"Dieser Beeinflussungszauber scheint die Einheit einem ganz "
"bestimmten Volk wohlgesonnen zu machen.",
(cdesc_fun)cinfo_calm
},
{ /* C_OLDRACE,
Merkt sich die alte 'richtige' Rasse einer gestalltwandelnden
Einheit */
CURSETYP_NORM, CURSE_SPREADALWAYS, NO_MERGE,
"",
"",
NULL
},
{ /* C_FUMBLE, */
CURSETYP_NORM, CURSE_SPREADNEVER, NO_MERGE,
"Tollpatschfluch",
"Eine Wolke negativer Energie umgibt die Einheit.",
(cdesc_fun)cinfo_fumble
},
{ /* C_RIOT, */
CURSETYP_NORM, 0, (M_DURATION),
"Aufruhr",
"Eine Wolke negativer Energie liegt <20>ber der Region.",
(cdesc_fun)cinfo_riot
},
{ /* C_NOCOST, Ewige Mauern-Zauber */
CURSETYP_NORM, 0, NO_MERGE,
"Ewige Mauern",
"Die Macht dieses Zaubers ist fast greifbar und tief in die Mauern "
"gebunden. Unbeeindruck vom Zahn der Zeit wird dieses Geb<65>ude wohl "
"auf Ewig stehen.",
(cdesc_fun)cinfo_nocost
},
2001-01-31 18:40:53 +01:00
{ /* C_HOLYGROUND, */
CURSETYP_NORM, 0, (M_VIGOUR_ADD),
"Heiliger Boden",
"Verschiedene Naturgeistern sind im Boden der Region gebunden und "
"besch<EFBFBD>tzen diese vor dem der dunklen Magie des lebenden Todes.",
(cdesc_fun)cinfo_holyground
2001-01-25 10:37:55 +01:00
},
{ /* C_FREE_13, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_14, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_15, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_16, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_17, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_18, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_19, */
CURSETYP_NORM, 0, (NO_MERGE),
"",
"",
NULL
},
/* struct's vom typ curse_unit: */
{ /* C_SPEED, */
CURSETYP_UNIT, CURSE_SPREADNEVER, M_MEN,
"Beschleunigen II",
"Diese Einheit bewegt sich doppelt so schnell.",
(cdesc_fun)cinfo_speed
},
{ /* C_ORC, */
CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN,
"Orkfieber",
"Dieser Zauber scheint die Einheit zu \"orkisieren\". Wie bei Orks "
"ist eine deutliche Neigung zur Fortpflanzung zu beobachten.",
(cdesc_fun)cinfo_orc
},
{ /* C_MBOOST, */
CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN,
"",
"",
NULL
},
{ /* C_KAELTESCHUTZ, */
CURSETYP_UNIT, CURSE_SPREADMODULO, ( M_MEN | M_DURATION ),
"Winterfell",
"Dieser Zauber sch<63>tzt vor den Auswirkungen der K<>lte.",
(cdesc_fun)cinfo_kaelteschutz
},
{ /* C_STRENGTH, */
CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN,
"Trollst<EFBFBD>rke",
"Dieser Zauber vermehrt die St<53>rke der verzauberten Personen um ein "
"vielfaches.",
(cdesc_fun)cinfo_strength
},
{ /* C_ALLSKILLS, Alp */
CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN,
"",
"",
(cdesc_fun)cinfo_allskills
},
{ /* C_MAGICRESISTANCE, */
CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN,
"Magieschutz",
"Dieser Zauber verst<73>rkt die nat<61>rliche Widerstandskraft gegen eine "
"Verzauberung.",
NULL
},
{ /* C_ITEMCLOAK, */
CURSETYP_UNIT, CURSE_SPREADNEVER, M_DURATION,
"Schleieraura",
"Dieser Zauber macht die Ausr<73>stung unsichtbar.",
(cdesc_fun)cinfo_itemcloak
},
{ /* C_SPARKLE, */
CURSETYP_UNIT, CURSE_SPREADMODULO, ( M_MEN | M_DURATION ),
"Leichte Verzauberung",
"Dieser Zauber ist einer der ersten, den junge Magier in der Schule lernen.",
(cdesc_fun)cinfo_sparkle
},
{ /* C_FREE_22, */
CURSETYP_UNIT, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_23, */
CURSETYP_UNIT, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_24, */
CURSETYP_UNIT, 0, (NO_MERGE),
"",
"",
NULL
},
/* struct's vom typ curse_skill: */
{ /* C_SKILL, */
CURSETYP_SKILL, CURSE_SPREADMODULO, M_MEN,
"",
"",
(cdesc_fun)cinfo_skill
},
{ /* C_FREE_30, */
CURSETYP_UNIT, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_31, */
CURSETYP_UNIT, 0, (NO_MERGE),
"",
"",
NULL
},
{ /* C_FREE_32, */
CURSETYP_UNIT, 0, (NO_MERGE),
"",
"",
NULL
},
/* struct's vom typ curse_secondid: */
};
void *
resolve_curse(void * id)
{
return cfindhash((int)id);
}
/* ------------------------------------------------------------- */
/* <20>berbleibsel des alten source:
* Nur Regionszauber
*/
boolean
is_spell_active(const region * r, curse_t id)
{
curse *c;
c = get_curse(r->attribs, id, 0);
if (!c)
return false;
if (c->flag & CURSE_ISNEW)
return false;
return true;
}