2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2001-12-10 01:13:39 +01:00
|
|
|
|
*
|
2006-07-28 03:31:23 +02:00
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
2007-09-02 20:11:17 +02:00
|
|
|
|
* Enno Rehling (enno@eressea.de)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
|
*
|
|
|
|
|
* 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>
|
2008-04-20 16:48:15 +02:00
|
|
|
|
#include <kernel/eressea.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "curse.h"
|
|
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
|
#include "building.h"
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include "faction.h"
|
|
|
|
|
#include "magic.h"
|
2007-02-24 00:38:44 +01:00
|
|
|
|
#include "message.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "objtypes.h"
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "ship.h"
|
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
#include "version.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* util includes */
|
2008-04-20 16:48:15 +02:00
|
|
|
|
#include <util/attrib.h>
|
2004-01-15 23:11:19 +01:00
|
|
|
|
#include <util/base36.h>
|
2007-08-05 14:19:56 +02:00
|
|
|
|
#include <util/goodies.h>
|
|
|
|
|
#include <util/log.h>
|
2007-02-24 00:38:44 +01:00
|
|
|
|
#include <util/nrmessage.h>
|
2004-02-09 23:20:40 +01:00
|
|
|
|
#include <util/rand.h>
|
2007-08-05 14:19:56 +02:00
|
|
|
|
#include <util/resolve.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/rng.h>
|
2008-04-26 16:55:22 +02:00
|
|
|
|
#include <util/storage.h>
|
2005-06-10 00:10:35 +02:00
|
|
|
|
#include <util/variant.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <math.h>
|
2004-01-15 23:29:41 +01:00
|
|
|
|
#include <ctype.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2002-05-09 12:22:12 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2004-05-26 08:42:58 +02:00
|
|
|
|
#define MAXENTITYHASH 7919
|
2003-07-29 11:48:03 +02:00
|
|
|
|
curse *cursehash[MAXENTITYHASH];
|
|
|
|
|
|
2007-04-22 02:30:12 +02:00
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
|
void
|
|
|
|
|
c_setflag(curse *c, unsigned int flags)
|
|
|
|
|
{
|
|
|
|
|
assert(c);
|
|
|
|
|
c->flags = (c->flags & ~flags) | (flags & (c->type->flags ^ flags));
|
|
|
|
|
}
|
|
|
|
|
/* -------------------------------------------------------------------------- */
|
|
|
|
|
void
|
|
|
|
|
c_clearflag(curse *c, unsigned int flags)
|
|
|
|
|
{
|
|
|
|
|
assert(c);
|
|
|
|
|
c->flags = (c->flags & ~flags) | (c->type->flags & flags);
|
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
void
|
|
|
|
|
chash(curse *c)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse *old = cursehash[c->no %MAXENTITYHASH];
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
cursehash[c->no %MAXENTITYHASH] = c;
|
|
|
|
|
c->nexthash = old;
|
2003-07-29 11:48:03 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
cunhash(curse *c)
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse **show;
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
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;
|
|
|
|
|
}
|
2003-07-29 11:48:03 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
curse *
|
|
|
|
|
cfindhash(int i)
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse *old;
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
for (old = cursehash[i % MAXENTITYHASH]; old; old = old->nexthash)
|
|
|
|
|
if (old->no == i)
|
|
|
|
|
return old;
|
|
|
|
|
return NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* at_curse */
|
|
|
|
|
void
|
|
|
|
|
curse_init(attrib * a) {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
a->data.v = calloc(1, sizeof(curse));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
curse_age(attrib * a)
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse * c = (curse*)a->data.v;
|
2006-07-28 13:41:57 +02:00
|
|
|
|
int result = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 13:41:57 +02:00
|
|
|
|
if (c->type->age) {
|
|
|
|
|
result = c->type->age(c);
|
|
|
|
|
}
|
|
|
|
|
if (result!=0) {
|
|
|
|
|
c->duration = 0;
|
2007-04-22 01:04:24 +02:00
|
|
|
|
} else if (c_flags(c) & CURSE_NOAGE) {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
c->duration = 1;
|
2006-11-05 21:14:07 +01:00
|
|
|
|
} else if (c->duration!=INT_MAX) {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
c->duration = max(0, c->duration-1);
|
|
|
|
|
}
|
2006-07-28 13:41:57 +02:00
|
|
|
|
return c->duration;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2006-07-28 03:31:23 +02:00
|
|
|
|
destroy_curse(curse * c)
|
|
|
|
|
{
|
|
|
|
|
cunhash(c);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (c->data.v && c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
free(c->data.v);
|
|
|
|
|
}
|
|
|
|
|
free(c);
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
void
|
|
|
|
|
curse_done(attrib * a) {
|
|
|
|
|
destroy_curse((curse *)a->data.v);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
/* ------------------------------------------------------------- */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
int
|
2008-04-26 16:55:22 +02:00
|
|
|
|
curse_read(attrib * a, struct storage * store)
|
2007-04-22 02:30:12 +02:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse * c = (curse*)a->data.v;
|
|
|
|
|
const curse_type * ct;
|
|
|
|
|
|
|
|
|
|
char cursename[64];
|
2007-04-22 01:04:24 +02:00
|
|
|
|
unsigned int flags;
|
2006-07-28 03:31:23 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
c->no = store->r_int(store);
|
|
|
|
|
store->r_tok_buf(store, cursename, sizeof(cursename));
|
|
|
|
|
flags = store->r_int(store);
|
|
|
|
|
c->duration = store->r_int(store);
|
|
|
|
|
if (store->version >= CURSEVIGOURISFLOAT_VERSION) {
|
|
|
|
|
c->vigour = store->r_flt(store);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
} else {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
int vigour = store->r_int(store);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
c->vigour = vigour;
|
|
|
|
|
}
|
2008-04-27 11:52:03 +02:00
|
|
|
|
if (store->version<INTPAK_VERSION) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
variant mageid;
|
|
|
|
|
mageid.i = store->r_int(store);
|
|
|
|
|
if (mageid.i <= 0) {
|
|
|
|
|
c->magician = (unit *)NULL;
|
|
|
|
|
} else {
|
|
|
|
|
c->magician = findunit(mageid.i);
|
|
|
|
|
if (!c->magician) {
|
2008-05-17 17:21:17 +02:00
|
|
|
|
ur_add(mageid, (void*)&c->magician, resolve_unit);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
read_unit_reference(&c->magician, store);
|
|
|
|
|
}
|
|
|
|
|
c->effect.i = store->r_int(store);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
ct = ct_find(cursename);
|
|
|
|
|
assert(ct!=NULL);
|
2007-04-22 01:04:24 +02:00
|
|
|
|
c->type = ct;
|
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (store->version < CURSEFLAGS_VERSION) {
|
2007-04-22 01:04:24 +02:00
|
|
|
|
c_setflag(c, flags);
|
|
|
|
|
} else {
|
|
|
|
|
c->flags = flags;
|
|
|
|
|
}
|
2007-04-22 02:30:12 +02:00
|
|
|
|
c_clearflag(c, CURSE_ISNEW);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (c->type->read) c->type->read(store, c);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
else if (c->type->typ==CURSETYP_UNIT) {
|
|
|
|
|
curse_unit * cc = calloc(1, sizeof(curse_unit));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
c->data.v = cc;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
cc->cursedmen = store->r_int(store);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
}
|
|
|
|
|
if (c->type->typ == CURSETYP_REGION) {
|
2008-05-17 17:21:17 +02:00
|
|
|
|
region * r;
|
|
|
|
|
read_region_reference(&r, store);
|
|
|
|
|
c->data.v = r;
|
2006-07-28 03:31:23 +02:00
|
|
|
|
}
|
|
|
|
|
chash(c);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return AT_READ_OK;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2008-04-26 16:55:22 +02:00
|
|
|
|
curse_write(const attrib * a, struct storage * store)
|
2007-04-22 02:30:12 +02:00
|
|
|
|
{
|
2007-04-22 01:04:24 +02:00
|
|
|
|
unsigned int flags;
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse * c = (curse*)a->data.v;
|
|
|
|
|
const curse_type * ct = c->type;
|
2008-04-27 11:52:03 +02:00
|
|
|
|
unit * mage = (c->magician && c->magician->number)?c->magician:NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-04-22 02:30:12 +02:00
|
|
|
|
/* copied from c_clearflag */
|
|
|
|
|
flags = (c->flags & ~CURSE_ISNEW) | (c->type->flags & CURSE_ISNEW);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, c->no);
|
|
|
|
|
store->w_tok(store, ct->cname);
|
|
|
|
|
store->w_int(store, flags);
|
|
|
|
|
store->w_int(store, c->duration);
|
|
|
|
|
store->w_flt(store, (float)c->vigour);
|
2008-04-27 11:52:03 +02:00
|
|
|
|
write_unit_reference(mage, store);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, c->effect.i);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2008-04-26 16:55:22 +02:00
|
|
|
|
if (c->type->write) c->type->write(store, c);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
else if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
curse_unit * cc = (curse_unit*)c->data.v;
|
2008-04-26 16:55:22 +02:00
|
|
|
|
store->w_int(store, cc->cursedmen);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
}
|
|
|
|
|
if (c->type->typ == CURSETYP_REGION) {
|
2008-04-26 16:55:22 +02:00
|
|
|
|
write_region_reference((region*)c->data.v, store);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
attrib_type at_curse =
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
"curse",
|
|
|
|
|
curse_init,
|
|
|
|
|
curse_done,
|
|
|
|
|
curse_age,
|
|
|
|
|
curse_write,
|
|
|
|
|
curse_read,
|
|
|
|
|
ATF_CURSE
|
2003-07-29 11:48:03 +02:00
|
|
|
|
};
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* Spruch identifizieren */
|
|
|
|
|
|
2008-04-20 16:48:15 +02:00
|
|
|
|
#include <util/umlaut.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
typedef struct cursetype_list {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
struct cursetype_list * next;
|
|
|
|
|
const curse_type * type;
|
2001-02-03 14:45:35 +01:00
|
|
|
|
} cursetype_list;
|
|
|
|
|
|
2004-01-15 23:45:26 +01:00
|
|
|
|
cursetype_list * cursetypes[256];
|
2004-01-15 23:29:41 +01:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
void
|
2001-02-03 14:45:35 +01:00
|
|
|
|
ct_register(const curse_type * ct)
|
|
|
|
|
{
|
2004-01-15 23:45:26 +01:00
|
|
|
|
unsigned int hash = tolower(ct->cname[0]);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
cursetype_list ** ctlp = &cursetypes[hash];
|
2007-09-02 17:42:48 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
while (*ctlp) {
|
|
|
|
|
cursetype_list * ctl = *ctlp;
|
|
|
|
|
if (ctl->type==ct) return;
|
|
|
|
|
ctlp=&ctl->next;
|
|
|
|
|
}
|
|
|
|
|
*ctlp = calloc(1, sizeof(cursetype_list));
|
|
|
|
|
(*ctlp)->type = ct;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const curse_type *
|
|
|
|
|
ct_find(const char *c)
|
|
|
|
|
{
|
2004-01-15 23:45:26 +01:00
|
|
|
|
unsigned int hash = tolower(c[0]);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
cursetype_list * ctl = cursetypes[hash];
|
|
|
|
|
while (ctl) {
|
|
|
|
|
size_t k = min(strlen(c), strlen(ctl->type->cname));
|
|
|
|
|
if (!strncasecmp(c, ctl->type->cname, k)) return ctl->type;
|
|
|
|
|
ctl = ctl->next;
|
|
|
|
|
}
|
2007-09-02 17:42:48 +02:00
|
|
|
|
/* disable this assert to be able to remove certain curses from the game
|
2006-07-28 03:31:23 +02:00
|
|
|
|
* make sure that all locations using that curse can deal with a NULL
|
|
|
|
|
* return value.
|
|
|
|
|
*/
|
|
|
|
|
assert(!"unknown cursetype");
|
|
|
|
|
return NULL;
|
2002-04-27 22:09:44 +02:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* get_curse identifiziert eine Verzauberung <20>ber die ID und gibt
|
|
|
|
|
* einen pointer auf die struct zur<EFBFBD>ck.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
boolean
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cmp_curse(const attrib * a, const void * data) {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
const curse * c = (const curse*)data;
|
|
|
|
|
if (a->type->flags & ATF_CURSE) {
|
|
|
|
|
if (!data || c == (curse*)a->data.v) return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
2002-05-01 21:08:32 +02:00
|
|
|
|
cmp_cursetype(const attrib * a, const void * data)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
const curse_type * ct = (const curse_type *)data;
|
|
|
|
|
if (a->type->flags & ATF_CURSE) {
|
|
|
|
|
if (!data || ct == ((curse*)a->data.v)->type) return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-05-01 21:08:32 +02:00
|
|
|
|
curse *
|
2005-06-10 00:10:35 +02:00
|
|
|
|
get_cursex(attrib *ap, const curse_type * ctype, variant data, boolean(*compare)(const curse *, variant))
|
2002-05-01 21:08:32 +02:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
attrib * a = a_select(ap, ctype, cmp_cursetype);
|
|
|
|
|
while (a) {
|
|
|
|
|
curse * c = (curse*)a->data.v;
|
|
|
|
|
if (compare(c, data)) return c;
|
|
|
|
|
a = a_select(a->next, ctype, cmp_cursetype);
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
curse *
|
2002-05-01 21:08:32 +02:00
|
|
|
|
get_curse(attrib *ap, const curse_type * ctype)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
attrib * a = ap;
|
2006-02-26 12:09:00 +01:00
|
|
|
|
while (a) {
|
|
|
|
|
if (a->type->flags & ATF_CURSE) {
|
|
|
|
|
const attrib_type * at = a->type;
|
|
|
|
|
while (a && a->type==at) {
|
|
|
|
|
curse* c = (curse *)a->data.v;
|
|
|
|
|
if (c->type==ctype) return c;
|
|
|
|
|
a = a->next;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
a = a->nexttype;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* findet einen curse global anhand seiner 'curse-Einheitnummer' */
|
|
|
|
|
|
|
|
|
|
curse *
|
2002-05-01 21:08:32 +02:00
|
|
|
|
findcurse(int cid)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return cfindhash(cid);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
void
|
2002-05-01 21:08:32 +02:00
|
|
|
|
remove_curse(attrib **ap, const curse *c)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
attrib *a = a_select(*ap, c, cmp_curse);
|
|
|
|
|
if (a) a_remove(ap, a);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-05-01 21:08:32 +02:00
|
|
|
|
/* gibt die allgemeine St<53>rke der Verzauberung zur<75>ck. id2 wird wie
|
|
|
|
|
* oben benutzt. Dies ist nicht die Wirkung, sondern die Kraft und
|
|
|
|
|
* damit der gegen Antimagie wirkende Widerstand einer Verzauberung */
|
2004-01-04 17:02:02 +01:00
|
|
|
|
static double
|
2002-05-01 21:08:32 +02:00
|
|
|
|
get_cursevigour(const curse *c)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (c) return c->vigour;
|
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-05-01 21:08:32 +02:00
|
|
|
|
/* setzt die St<53>rke der Verzauberung auf i */
|
|
|
|
|
static void
|
2004-01-04 17:02:02 +01:00
|
|
|
|
set_cursevigour(curse *c, double vigour)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
assert(c && vigour > 0);
|
|
|
|
|
c->vigour = vigour;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 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.
|
|
|
|
|
*/
|
2004-01-04 17:02:02 +01:00
|
|
|
|
double
|
|
|
|
|
curse_changevigour(attrib **ap, curse *c, double vigour)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
vigour += get_cursevigour(c);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (vigour <= 0) {
|
|
|
|
|
remove_curse(ap, c);
|
|
|
|
|
vigour = 0;
|
|
|
|
|
} else {
|
|
|
|
|
set_cursevigour(c, vigour);
|
|
|
|
|
}
|
|
|
|
|
return vigour;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
int
|
2002-05-01 21:08:32 +02:00
|
|
|
|
curse_geteffect(const curse *c)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2008-03-05 19:39:44 +01:00
|
|
|
|
if (c==NULL) return 0;
|
|
|
|
|
if (c_flags(c) & CURSE_ISNEW) return 0;
|
|
|
|
|
return c->effect.i;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2002-05-01 21:08:32 +02:00
|
|
|
|
set_curseingmagician(struct unit *magician, struct attrib *ap_target, const curse_type *ct)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse * c = get_curse(ap_target, ct);
|
|
|
|
|
if (c) {
|
|
|
|
|
c->magician = magician;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
2002-05-01 21:08:32 +02:00
|
|
|
|
/* gibt bei Personenbeschr<68>nkten Verzauberungen die Anzahl der
|
|
|
|
|
* betroffenen Personen zur<EFBFBD>ck. Ansonsten wird 0 zur<EFBFBD>ckgegeben. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
int
|
2002-04-27 22:09:44 +02:00
|
|
|
|
get_cursedmen(unit *u, curse *c)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
int cursedmen = u->number;
|
2002-04-27 22:09:44 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (!c) return 0;
|
2002-04-27 22:09:44 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
/* je nach curse_type andere data struct */
|
|
|
|
|
if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
curse_unit * cc = (curse_unit*)c->data.v;
|
|
|
|
|
cursedmen = cc->cursedmen;
|
|
|
|
|
}
|
2002-11-25 16:30:51 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return min(u->number, cursedmen);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-05-01 21:08:32 +02:00
|
|
|
|
/* setzt die Anzahl der betroffenen Personen auf cursedmen */
|
|
|
|
|
static void
|
|
|
|
|
set_cursedmen(curse *c, int cursedmen)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (!c) return;
|
2002-04-27 22:09:44 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
/* je nach curse_type andere data struct */
|
|
|
|
|
if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
curse_unit * cc = (curse_unit*)c->data.v;
|
|
|
|
|
cc->cursedmen = cursedmen;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* Legt eine neue Verzauberung an. Sollte es schon einen Zauber
|
|
|
|
|
* dieses Typs geben, gibt es den bestehenden zur<EFBFBD>ck.
|
|
|
|
|
*/
|
2005-06-10 00:10:35 +02:00
|
|
|
|
static curse *
|
2007-04-22 01:04:24 +02:00
|
|
|
|
make_curse(unit *mage, attrib **ap, const curse_type *ct, double vigour,
|
2006-07-28 03:31:23 +02:00
|
|
|
|
int duration, variant effect, int men)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse *c;
|
|
|
|
|
attrib * a;
|
|
|
|
|
|
|
|
|
|
a = a_new(&at_curse);
|
|
|
|
|
a_add(ap, a);
|
|
|
|
|
c = (curse*)a->data.v;
|
|
|
|
|
|
|
|
|
|
c->type = ct;
|
2007-04-22 02:30:12 +02:00
|
|
|
|
c->flags = 0;
|
2006-07-28 03:31:23 +02:00
|
|
|
|
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.v = cc;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
}
|
|
|
|
|
return c;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Mapperfunktion f<>r das Anlegen neuer curse. Automatisch wird zum
|
|
|
|
|
* passenden Typ verzweigt und die relevanten Variablen weitergegeben.
|
|
|
|
|
*/
|
|
|
|
|
curse *
|
2004-01-04 17:02:02 +01:00
|
|
|
|
create_curse(unit *magician, attrib **ap, const curse_type *ct, double vigour,
|
2006-07-28 03:31:23 +02:00
|
|
|
|
int duration, variant effect, int men)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse *c;
|
|
|
|
|
|
|
|
|
|
/* die Kraft eines Spruchs darf nicht 0 sein*/
|
|
|
|
|
assert(vigour > 0);
|
|
|
|
|
|
|
|
|
|
c = get_curse(*ap, ct);
|
|
|
|
|
|
2007-04-22 01:04:24 +02:00
|
|
|
|
if (c && (c_flags(c) & CURSE_ONLYONE)){
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
assert(c==NULL || ct==c->type);
|
|
|
|
|
|
|
|
|
|
/* es gibt schon eins diese Typs */
|
|
|
|
|
if (c && ct->mergeflags != NO_MERGE) {
|
|
|
|
|
if(ct->mergeflags & M_DURATION){
|
|
|
|
|
c->duration = max(c->duration, duration);
|
|
|
|
|
}
|
|
|
|
|
if(ct->mergeflags & M_SUMDURATION){
|
|
|
|
|
c->duration += duration;
|
|
|
|
|
}
|
|
|
|
|
if(ct->mergeflags & M_SUMEFFECT){
|
|
|
|
|
c->effect.i += effect.i;
|
|
|
|
|
}
|
|
|
|
|
if(ct->mergeflags & M_MAXEFFECT){
|
|
|
|
|
c->effect.i = max(c->effect.i, effect.i);
|
|
|
|
|
}
|
|
|
|
|
if(ct->mergeflags & M_VIGOUR){
|
|
|
|
|
c->vigour = max(vigour, c->vigour);
|
|
|
|
|
}
|
|
|
|
|
if(ct->mergeflags & M_VIGOUR_ADD){
|
|
|
|
|
c->vigour = vigour + c->vigour;
|
|
|
|
|
}
|
|
|
|
|
if(ct->mergeflags & M_MEN){
|
|
|
|
|
switch (ct->typ) {
|
|
|
|
|
case CURSETYP_UNIT:
|
|
|
|
|
{
|
|
|
|
|
curse_unit * cc = (curse_unit*)c->data.v;
|
|
|
|
|
cc->cursedmen += men;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
set_curseingmagician(magician, *ap, ct);
|
|
|
|
|
} else {
|
2007-04-22 01:04:24 +02:00
|
|
|
|
c = make_curse(magician, ap, ct, vigour, duration, effect, men);
|
2006-07-28 03:31:23 +02:00
|
|
|
|
}
|
|
|
|
|
return c;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* hier m<>ssen alle c-typen, die auf Einheiten gezaubert werden k<>nnen,
|
|
|
|
|
* ber<EFBFBD>cksichtigt werden */
|
|
|
|
|
|
2006-10-15 16:09:59 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
do_transfer_curse(curse *c, unit * u, unit * u2, int n)
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
int cursedmen = 0;
|
|
|
|
|
int men = 0;
|
|
|
|
|
boolean dogive = false;
|
|
|
|
|
const curse_type *ct = c->type;
|
|
|
|
|
|
|
|
|
|
switch (ct->typ) {
|
|
|
|
|
case CURSETYP_UNIT:
|
|
|
|
|
{
|
|
|
|
|
curse_unit * cc = (curse_unit*)c->data.v;
|
|
|
|
|
men = cc->cursedmen;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
cursedmen = u->number;
|
|
|
|
|
}
|
|
|
|
|
|
2007-04-22 01:04:24 +02:00
|
|
|
|
switch ((ct->flags | c->flags) & CURSE_SPREADMASK) {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
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 (rng_int()%u_number < cursedmen){
|
|
|
|
|
++men;
|
|
|
|
|
--cursedmen;
|
|
|
|
|
dogive = true;
|
|
|
|
|
}
|
|
|
|
|
--u_number;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case CURSE_SPREADCHANCE:
|
|
|
|
|
if (chance(u2->number/(double)(u2->number + n))) {
|
|
|
|
|
men = u2->number + n;
|
|
|
|
|
dogive = true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case CURSE_SPREADNEVER:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (dogive == true) {
|
2007-04-22 01:04:24 +02:00
|
|
|
|
curse * cnew = make_curse(c->magician, &u2->attribs, c->type, c->vigour,
|
2005-06-10 00:10:35 +02:00
|
|
|
|
c->duration, c->effect, men);
|
2007-04-22 01:04:24 +02:00
|
|
|
|
cnew->flags = c->flags;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (ct->typ == CURSETYP_UNIT) set_cursedmen(cnew, men);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
transfer_curse(unit * u, unit * u2, int n)
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
attrib * a;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
a = a_find(u->attribs, &at_curse);
|
|
|
|
|
while (a && a->type==&at_curse) {
|
|
|
|
|
curse *c = (curse*)a->data.v;
|
|
|
|
|
do_transfer_curse(c, u, u2, n);
|
|
|
|
|
a = a->next;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
boolean
|
2002-05-01 21:08:32 +02:00
|
|
|
|
curse_active(const curse *c)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (!c) return false;
|
2007-04-22 01:04:24 +02:00
|
|
|
|
if (c_flags(c) & CURSE_ISNEW) return false;
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (c->vigour <= 0) return false;
|
2002-04-06 22:45:55 +02:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return true;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
boolean
|
2002-05-01 21:08:32 +02:00
|
|
|
|
is_cursed_internal(attrib *ap, const curse_type *ct)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
curse *c = get_curse(ap, ct);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
if (!c)
|
|
|
|
|
return false;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return true;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
boolean
|
2007-06-20 02:34:02 +02:00
|
|
|
|
is_cursed_with(const attrib *ap, const curse *c)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2007-06-20 02:34:02 +02:00
|
|
|
|
const attrib *a = ap;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
while (a) {
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if ((a->type->flags & ATF_CURSE) && (c == (const curse *)a->data.v)) {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
a = a->next;
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return false;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* cursedata */
|
|
|
|
|
/* ------------------------------------------------------------- */
|
2002-11-25 16:30:51 +01:00
|
|
|
|
/*
|
|
|
|
|
* typedef struct curse_type {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
* const char *cname; (Name der Zauberwirkung, Identifizierung des curse)
|
|
|
|
|
* int typ;
|
|
|
|
|
* spread_t spread;
|
|
|
|
|
* unsigned int mergeflags;
|
|
|
|
|
* int (*curseinfo)(const struct locale*, const void*, int, curse*, int);
|
|
|
|
|
* void (*change_vigour)(curse*, double);
|
2008-04-26 16:55:22 +02:00
|
|
|
|
* int (*read)(struct storage * store, curse * c);
|
|
|
|
|
* int (*write)(struct storage * store, const curse * c);
|
2002-05-09 19:34:43 +02:00
|
|
|
|
* } curse_type;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
*/
|
2002-05-01 21:08:32 +02:00
|
|
|
|
|
2008-05-17 17:21:17 +02:00
|
|
|
|
void
|
|
|
|
|
resolve_curse(variant id, void * address)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2008-05-17 17:21:17 +02:00
|
|
|
|
curse * c = cfindhash(id.i);
|
|
|
|
|
*(curse**)address = c;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-05-09 14:06:28 +02:00
|
|
|
|
static const char * oldnames[MAXCURSE] = {
|
2006-07-28 03:31:23 +02:00
|
|
|
|
"fogtrap",
|
|
|
|
|
"antimagiczone",
|
|
|
|
|
"farvision",
|
|
|
|
|
"gbdream",
|
|
|
|
|
"auraboost",
|
|
|
|
|
"maelstrom",
|
|
|
|
|
"blessedharvest",
|
|
|
|
|
"drought",
|
|
|
|
|
"badlearn",
|
|
|
|
|
"stormwind",
|
|
|
|
|
"flyingship",
|
|
|
|
|
"nodrift",
|
|
|
|
|
"depression",
|
|
|
|
|
"magicwalls",
|
|
|
|
|
"strongwall",
|
|
|
|
|
"astralblock",
|
|
|
|
|
"generous",
|
|
|
|
|
"peacezone",
|
|
|
|
|
"magicstreet",
|
|
|
|
|
"magicrunes",
|
|
|
|
|
"badmagicresistancezone",
|
|
|
|
|
"goodmagicresistancezone",
|
|
|
|
|
"slavery",
|
|
|
|
|
"calmmonster",
|
|
|
|
|
"oldrace",
|
|
|
|
|
"fumble",
|
|
|
|
|
"riotzone",
|
|
|
|
|
"nocostbuilding",
|
|
|
|
|
"godcursezone",
|
|
|
|
|
"speed",
|
|
|
|
|
"orcish",
|
|
|
|
|
"magicboost",
|
|
|
|
|
"insectfur",
|
|
|
|
|
"strength",
|
|
|
|
|
"worse",
|
|
|
|
|
"magicresistance",
|
|
|
|
|
"itemcloak",
|
|
|
|
|
"sparkle",
|
|
|
|
|
"skillmod"
|
2002-05-09 14:06:28 +02:00
|
|
|
|
};
|
|
|
|
|
|
2002-11-25 16:30:51 +01:00
|
|
|
|
const char *
|
2002-05-09 14:06:28 +02:00
|
|
|
|
oldcursename(int id)
|
|
|
|
|
{
|
2006-07-28 03:31:23 +02:00
|
|
|
|
return oldnames[id];
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2002-05-09 19:34:43 +02:00
|
|
|
|
|
2007-02-24 00:38:44 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
2007-03-09 22:30:00 +01:00
|
|
|
|
message *
|
|
|
|
|
cinfo_simple(const void * obj, typ_t typ, const struct curse *c, int self)
|
2007-02-24 00:38:44 +01:00
|
|
|
|
{
|
|
|
|
|
struct message * msg;
|
|
|
|
|
|
|
|
|
|
unused(typ);
|
|
|
|
|
unused(self);
|
|
|
|
|
unused(obj);
|
|
|
|
|
|
|
|
|
|
msg = msg_message(mkname("curseinfo", c->type->cname), "id", c->no);
|
2007-03-09 22:30:00 +01:00
|
|
|
|
if (msg==NULL) {
|
|
|
|
|
log_error(("There is no curseinfo for %s.\n", c->type->cname));
|
2007-02-24 00:38:44 +01:00
|
|
|
|
}
|
2007-03-09 22:30:00 +01:00
|
|
|
|
return msg;
|
2007-02-24 00:38:44 +01:00
|
|
|
|
}
|