2010-08-08 10:06:34 +02:00
|
|
|
|
/*
|
2015-01-30 22:10:29 +01:00
|
|
|
|
Copyright (c) 1998-2015, Enno Rehling <enno@eressea.de>
|
2014-10-18 18:37:44 +02:00
|
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
|
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
|
#include <kernel/config.h>
|
|
|
|
|
#include "curse.h"
|
|
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
|
#include "building.h"
|
|
|
|
|
#include "faction.h"
|
2014-06-09 18:54:48 +02:00
|
|
|
|
#include "messages.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include "objtypes.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "ship.h"
|
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
#include <util/base36.h>
|
2016-02-13 13:42:02 +01:00
|
|
|
|
#include <util/gamedata.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include <util/goodies.h>
|
|
|
|
|
#include <util/language.h>
|
|
|
|
|
#include <util/log.h>
|
|
|
|
|
#include <util/nrmessage.h>
|
|
|
|
|
#include <util/rand.h>
|
|
|
|
|
#include <util/resolve.h>
|
|
|
|
|
#include <util/rng.h>
|
|
|
|
|
#include <util/variant.h>
|
|
|
|
|
|
2013-12-31 10:06:28 +01:00
|
|
|
|
#include <storage.h>
|
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <stdio.h>
|
2014-03-15 19:29:11 +01:00
|
|
|
|
#include <stdlib.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
|
|
#define MAXENTITYHASH 7919
|
2016-02-25 23:11:44 +01:00
|
|
|
|
static curse *cursehash[MAXENTITYHASH];
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void c_setflag(curse * c, unsigned int flags)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
assert(c);
|
|
|
|
|
c->flags = (c->flags & ~flags) | (flags & (c->type->flags ^ flags));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* -------------------------------------------------------------------------- */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void c_clearflag(curse * c, unsigned int flags)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
assert(c);
|
|
|
|
|
c->flags = (c->flags & ~flags) | (c->type->flags & flags);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void chash(curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2016-02-26 17:54:25 +01:00
|
|
|
|
int i = c->no % MAXENTITYHASH;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2016-02-26 17:54:25 +01:00
|
|
|
|
c->nexthash = cursehash[i];
|
|
|
|
|
cursehash[i] = c;
|
|
|
|
|
assert(c->nexthash != c);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void cunhash(curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse **show;
|
2016-02-26 17:54:25 +01:00
|
|
|
|
int i = c->no % MAXENTITYHASH;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2016-02-26 17:54:25 +01:00
|
|
|
|
for (show = &cursehash[i]; *show;
|
2014-10-18 18:37:44 +02:00
|
|
|
|
show = &(*show)->nexthash) {
|
|
|
|
|
if ((*show)->no == c->no)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (*show) {
|
|
|
|
|
assert(*show == c);
|
|
|
|
|
*show = (*show)->nexthash;
|
|
|
|
|
c->nexthash = 0;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* at_curse */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void curse_init(attrib * a)
|
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
a->data.v = calloc(1, sizeof(curse));
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-12-16 22:18:44 +01:00
|
|
|
|
int curse_age(attrib * a, void *owner)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse *c = (curse *)a->data.v;
|
|
|
|
|
int result = 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-12-16 22:18:44 +01:00
|
|
|
|
unused_arg(owner);
|
2015-05-07 17:45:02 +02:00
|
|
|
|
c_clearflag(c, CURSE_ISNEW);
|
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (c_flags(c) & CURSE_NOAGE) {
|
|
|
|
|
c->duration = INT_MAX;
|
|
|
|
|
}
|
|
|
|
|
if (c->type->age) {
|
|
|
|
|
result = c->type->age(c);
|
|
|
|
|
}
|
|
|
|
|
if (result != 0) {
|
|
|
|
|
c->duration = 0;
|
|
|
|
|
}
|
|
|
|
|
else if (c->duration != INT_MAX) {
|
|
|
|
|
c->duration = _max(0, c->duration - 1);
|
|
|
|
|
}
|
2015-08-25 22:50:58 +02:00
|
|
|
|
return (c->duration > 0) ? AT_AGE_KEEP : AT_AGE_REMOVE;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void destroy_curse(curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
cunhash(c);
|
|
|
|
|
free(c);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void curse_done(attrib * a)
|
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
destroy_curse((curse *)a->data.v);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** reads curses that have been removed from the code */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static int read_ccompat(const char *cursename, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
struct compat {
|
|
|
|
|
const char *name;
|
|
|
|
|
const char *tokens;
|
|
|
|
|
} *seek, old_curses[] = { {
|
2015-01-30 20:37:14 +01:00
|
|
|
|
"disorientationzone", "" }, {
|
|
|
|
|
"shipdisorientation", "" }, {
|
|
|
|
|
NULL, NULL } };
|
2014-10-18 18:37:44 +02:00
|
|
|
|
for (seek = old_curses; seek->name; ++seek) {
|
|
|
|
|
if (strcmp(seek->tokens, cursename) == 0) {
|
|
|
|
|
const char *p;
|
|
|
|
|
for (p = seek->name; p; ++p) {
|
|
|
|
|
switch (*p) {
|
|
|
|
|
case 'd':
|
|
|
|
|
READ_INT(store, 0);
|
|
|
|
|
break;
|
|
|
|
|
case 's':
|
|
|
|
|
READ_STR(store, 0, 0);
|
|
|
|
|
break;
|
|
|
|
|
case 't':
|
|
|
|
|
READ_TOK(store, 0, 0);
|
|
|
|
|
break;
|
|
|
|
|
case 'i':
|
|
|
|
|
READ_INT(store, 0);
|
|
|
|
|
break;
|
|
|
|
|
case 'f':
|
|
|
|
|
READ_FLT(store, 0);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return -1;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-02-13 13:42:02 +01:00
|
|
|
|
int curse_read(attrib * a, void *owner, gamedata *data)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2016-02-13 13:42:02 +01:00
|
|
|
|
storage *store = data->store;
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse *c = (curse *)a->data.v;
|
|
|
|
|
int ur;
|
|
|
|
|
char cursename[64];
|
|
|
|
|
int n;
|
|
|
|
|
int flags;
|
|
|
|
|
float flt;
|
|
|
|
|
|
2016-02-26 17:54:25 +01:00
|
|
|
|
assert(!c->no);
|
2014-10-18 18:37:44 +02:00
|
|
|
|
READ_INT(store, &c->no);
|
|
|
|
|
chash(c);
|
|
|
|
|
READ_TOK(store, cursename, sizeof(cursename));
|
|
|
|
|
READ_INT(store, &flags);
|
|
|
|
|
READ_INT(store, &c->duration);
|
2014-12-12 23:00:30 +01:00
|
|
|
|
READ_FLT(store, &flt);
|
|
|
|
|
c->vigour = flt;
|
2016-02-13 13:42:02 +01:00
|
|
|
|
if (data->version < INTPAK_VERSION) {
|
2016-02-13 20:38:32 +01:00
|
|
|
|
ur = resolve_unit(read_int(data->store), &c->magician);
|
2014-10-18 18:37:44 +02:00
|
|
|
|
}
|
|
|
|
|
else {
|
2016-02-13 20:38:32 +01:00
|
|
|
|
ur = read_reference(&c->magician, data, read_unit_reference, resolve_unit);
|
2014-10-18 18:37:44 +02:00
|
|
|
|
}
|
2016-02-13 13:42:02 +01:00
|
|
|
|
if (data->version < CURSEFLOAT_VERSION) {
|
2014-10-18 18:37:44 +02:00
|
|
|
|
READ_INT(store, &n);
|
|
|
|
|
c->effect = (float)n;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
READ_FLT(store, &flt);
|
|
|
|
|
c->effect = flt;
|
|
|
|
|
}
|
|
|
|
|
c->type = ct_find(cursename);
|
|
|
|
|
if (c->type == NULL) {
|
|
|
|
|
int result = read_ccompat(cursename, store);
|
|
|
|
|
if (result != 0) {
|
|
|
|
|
log_error("missing curse %s, no compatibility code either.\n", cursename);
|
|
|
|
|
}
|
|
|
|
|
assert(result == 0);
|
|
|
|
|
return AT_READ_FAIL;
|
|
|
|
|
}
|
2014-12-12 22:52:23 +01:00
|
|
|
|
c->flags = flags;
|
2016-02-13 13:42:02 +01:00
|
|
|
|
if (data->version < EXPLICIT_CURSE_ISNEW_VERSION) {
|
2015-05-07 17:45:02 +02:00
|
|
|
|
c_clearflag(c, CURSE_ISNEW);
|
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
|
2016-02-13 20:38:32 +01:00
|
|
|
|
if (c->type->read) {
|
|
|
|
|
c->type->read(data, c, owner);
|
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
else if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
READ_INT(store, &c->data.i);
|
|
|
|
|
}
|
|
|
|
|
if (c->type->typ == CURSETYP_REGION) {
|
|
|
|
|
int rr =
|
2016-02-13 20:38:32 +01:00
|
|
|
|
read_reference(&c->data.v, data, read_region_reference,
|
2016-02-13 14:09:36 +01:00
|
|
|
|
RESOLVE_REGION(data->version));
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (ur == 0 && rr == 0 && !c->data.v) {
|
|
|
|
|
return AT_READ_FAIL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return AT_READ_OK;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void curse_write(const attrib * a, const void *owner, struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
unsigned int flags;
|
|
|
|
|
curse *c = (curse *)a->data.v;
|
|
|
|
|
const curse_type *ct = c->type;
|
|
|
|
|
unit *mage = (c->magician && c->magician->number) ? c->magician : NULL;
|
|
|
|
|
|
|
|
|
|
/* copied from c_clearflag */
|
2016-02-13 14:09:36 +01:00
|
|
|
|
#if RELEASE_VERSION < EXPLICIT_CURSE_ISNEW_VERSION
|
|
|
|
|
flags = (c->flags & ~CURSE_ISNEW) | (c->type->flags & CURSE_ISNEW);
|
|
|
|
|
#else
|
|
|
|
|
flags = c->flags | c->type->flags;
|
|
|
|
|
#endif
|
2014-10-18 18:37:44 +02:00
|
|
|
|
|
|
|
|
|
WRITE_INT(store, c->no);
|
|
|
|
|
WRITE_TOK(store, ct->cname);
|
|
|
|
|
WRITE_INT(store, flags);
|
|
|
|
|
WRITE_INT(store, c->duration);
|
|
|
|
|
WRITE_FLT(store, (float)c->vigour);
|
|
|
|
|
write_unit_reference(mage, store);
|
|
|
|
|
WRITE_FLT(store, (float)c->effect);
|
|
|
|
|
|
|
|
|
|
if (c->type->write)
|
|
|
|
|
c->type->write(store, c, owner);
|
|
|
|
|
else if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
WRITE_INT(store, c->data.i);
|
|
|
|
|
}
|
|
|
|
|
if (c->type->typ == CURSETYP_REGION) {
|
|
|
|
|
write_region_reference((region *)c->data.v, store);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
attrib_type at_curse = {
|
2014-10-18 18:37:44 +02:00
|
|
|
|
"curse",
|
|
|
|
|
curse_init,
|
|
|
|
|
curse_done,
|
|
|
|
|
curse_age,
|
|
|
|
|
curse_write,
|
|
|
|
|
curse_read,
|
2016-02-09 06:43:19 +01:00
|
|
|
|
NULL,
|
2014-10-18 18:37:44 +02:00
|
|
|
|
ATF_CURSE
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* Spruch identifizieren */
|
|
|
|
|
|
|
|
|
|
#include <util/umlaut.h>
|
2012-05-31 04:55:17 +02:00
|
|
|
|
#include <quicklist.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2015-10-15 08:38:43 +02:00
|
|
|
|
#define MAXCTHASH 128
|
|
|
|
|
static quicklist *cursetypes[MAXCTHASH];
|
2016-09-19 03:52:10 +02:00
|
|
|
|
static int ct_changes = 1;
|
|
|
|
|
|
|
|
|
|
bool ct_changed(int *cache)
|
|
|
|
|
{
|
|
|
|
|
assert(cache);
|
|
|
|
|
if (*cache != ct_changes) {
|
|
|
|
|
*cache = ct_changes;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void ct_register(const curse_type * ct)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-10-15 08:38:43 +02:00
|
|
|
|
unsigned int hash = tolower(ct->cname[0]) & 0xFF;
|
2014-10-18 18:37:44 +02:00
|
|
|
|
quicklist **ctlp = cursetypes + hash;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
ql_set_insert(ctlp, (void *)ct);
|
2016-09-19 03:52:10 +02:00
|
|
|
|
++ct_changes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ct_remove(const char *c)
|
|
|
|
|
{
|
|
|
|
|
unsigned int hash = tolower(c[0]);
|
|
|
|
|
quicklist *ctl = cursetypes[hash];
|
|
|
|
|
|
|
|
|
|
if (ctl) {
|
|
|
|
|
int qi;
|
|
|
|
|
|
|
|
|
|
for (qi = 0; ctl; ql_advance(&ctl, &qi, 1)) {
|
|
|
|
|
curse_type *type = (curse_type *)ql_get(ctl, qi);
|
|
|
|
|
|
|
|
|
|
if (strcmp(c, type->cname) == 0) {
|
|
|
|
|
ql_delete(&ctl, qi);
|
|
|
|
|
++ct_changes;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const curse_type *ct_find(const char *c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
unsigned int hash = tolower(c[0]);
|
|
|
|
|
quicklist *ctl = cursetypes[hash];
|
2011-02-25 16:56:00 +01:00
|
|
|
|
|
2015-07-06 19:53:09 +02:00
|
|
|
|
if (ctl) {
|
|
|
|
|
size_t c_len = strlen(c);
|
|
|
|
|
int qi;
|
2011-02-25 16:56:00 +01:00
|
|
|
|
|
2015-07-06 19:53:09 +02:00
|
|
|
|
for (qi = 0; ctl; ql_advance(&ctl, &qi, 1)) {
|
|
|
|
|
curse_type *type = (curse_type *)ql_get(ctl, qi);
|
|
|
|
|
|
|
|
|
|
if (strcmp(c, type->cname) == 0) {
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return type;
|
|
|
|
|
}
|
2015-07-06 19:53:09 +02:00
|
|
|
|
else {
|
|
|
|
|
size_t k = _min(c_len, strlen(type->cname));
|
|
|
|
|
if (!_memicmp(c, type->cname, k)) {
|
|
|
|
|
return type;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2015-05-12 09:12:58 +02:00
|
|
|
|
void ct_checknames(void) {
|
2015-01-13 22:08:12 +01:00
|
|
|
|
int i, qi;
|
|
|
|
|
quicklist *ctl;
|
|
|
|
|
|
2015-10-15 08:40:52 +02:00
|
|
|
|
for (i = 0; i < MAXCTHASH; ++i) {
|
2015-01-13 22:08:12 +01:00
|
|
|
|
ctl = cursetypes[i];
|
|
|
|
|
for (qi = 0; ctl; ql_advance(&ctl, &qi, 1)) {
|
|
|
|
|
curse_type *type = (curse_type *)ql_get(ctl, qi);
|
|
|
|
|
curse_name(type, default_locale);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* get_curse identifiziert eine Verzauberung <20>ber die ID und gibt
|
|
|
|
|
* einen pointer auf die struct zur<EFBFBD>ck.
|
|
|
|
|
*/
|
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
static bool cmp_curse(const attrib * a, const void *data)
|
2011-03-07 08:02:35 +01:00
|
|
|
|
{
|
2014-10-18 18:37:44 +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;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
curse *get_curse(attrib * ap, const curse_type * ctype)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
attrib *a = ap;
|
2016-08-21 17:19:22 +02:00
|
|
|
|
if (!ctype) return NULL;
|
2014-10-18 18:37:44 +02: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;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* findet einen curse global anhand seiner 'curse-Einheitnummer' */
|
|
|
|
|
|
2016-03-25 21:51:49 +01:00
|
|
|
|
curse *findcurse(int i)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2016-03-25 21:51:49 +01:00
|
|
|
|
curse *old;
|
|
|
|
|
for (old = cursehash[i % MAXENTITYHASH]; old; old = old->nexthash)
|
|
|
|
|
if (old->no == i)
|
|
|
|
|
return old;
|
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
2014-10-18 18:37:44 +02:00
|
|
|
|
bool remove_curse(attrib ** ap, const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
attrib *a = a_select(*ap, c, cmp_curse);
|
|
|
|
|
return a && a_remove(ap, a) == 1;
|
2010-08-08 10:06:34 +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 */
|
2015-05-15 19:08:44 +02:00
|
|
|
|
static double get_cursevigour(const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-05-15 19:08:44 +02:00
|
|
|
|
return c ? c->vigour : 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* setzt die St<53>rke der Verzauberung auf i */
|
2015-05-15 19:08:44 +02:00
|
|
|
|
static void set_cursevigour(curse * c, double vigour)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
assert(c && vigour > 0);
|
|
|
|
|
c->vigour = vigour;
|
2010-08-08 10:06:34 +02: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.
|
|
|
|
|
*/
|
2015-05-15 19:08:44 +02:00
|
|
|
|
double curse_changevigour(attrib ** ap, curse * c, double vigour)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
vigour += get_cursevigour(c);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (vigour <= 0) {
|
|
|
|
|
remove_curse(ap, c);
|
|
|
|
|
vigour = 0;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
set_cursevigour(c, vigour);
|
|
|
|
|
}
|
|
|
|
|
return vigour;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2015-05-15 19:08:44 +02:00
|
|
|
|
double curse_geteffect(const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (c == NULL)
|
|
|
|
|
return 0;
|
|
|
|
|
if (c_flags(c) & CURSE_ISNEW)
|
|
|
|
|
return 0;
|
|
|
|
|
return c->effect;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int curse_geteffect_int(const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2015-05-15 19:08:44 +02:00
|
|
|
|
double effect = curse_geteffect(c);
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (effect - (int)effect != 0) {
|
|
|
|
|
log_error("curse has an integer attribute with float value: '%s' = %lf",
|
|
|
|
|
c->type->cname, effect);
|
|
|
|
|
}
|
|
|
|
|
return (int)effect;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
static void
|
2011-03-07 08:02:35 +01:00
|
|
|
|
set_curseingmagician(struct unit *magician, struct attrib *ap_target,
|
2014-10-18 18:37:44 +02:00
|
|
|
|
const curse_type * ct)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse *c = get_curse(ap_target, ct);
|
|
|
|
|
if (c) {
|
|
|
|
|
c->magician = magician;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* gibt bei Personenbeschr<68>nkten Verzauberungen die Anzahl der
|
|
|
|
|
* betroffenen Personen zur<EFBFBD>ck. Ansonsten wird 0 zur<EFBFBD>ckgegeben. */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
int get_cursedmen(unit * u, const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
int cursedmen = u->number;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (!c)
|
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
/* je nach curse_type andere data struct */
|
|
|
|
|
if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
cursedmen = c->data.i;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return _min(u->number, cursedmen);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* setzt die Anzahl der betroffenen Personen auf cursedmen */
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void set_cursedmen(curse * c, int cursedmen)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (!c)
|
|
|
|
|
return;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
/* je nach curse_type andere data struct */
|
|
|
|
|
if (c->type->typ == CURSETYP_UNIT) {
|
|
|
|
|
c->data.i = cursedmen;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* Legt eine neue Verzauberung an. Sollte es schon einen Zauber
|
|
|
|
|
* dieses Typs geben, gibt es den bestehenden zur<EFBFBD>ck.
|
|
|
|
|
*/
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static curse *make_curse(unit * mage, attrib ** ap, const curse_type * ct,
|
2015-05-15 19:08:44 +02:00
|
|
|
|
double vigour, int duration, double effect, int men)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse *c;
|
|
|
|
|
attrib *a;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
a = a_new(&at_curse);
|
|
|
|
|
a_add(ap, a);
|
|
|
|
|
c = (curse *)a->data.v;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
c->type = ct;
|
|
|
|
|
c->flags = 0;
|
|
|
|
|
c->vigour = vigour;
|
|
|
|
|
c->duration = duration;
|
|
|
|
|
c->effect = effect;
|
|
|
|
|
c->magician = mage;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
c->no = newunitid();
|
|
|
|
|
chash(c);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
switch (c->type->typ) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case CURSETYP_NORM:
|
2014-10-18 18:37:44 +02:00
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
case CURSETYP_UNIT:
|
2014-10-18 18:37:44 +02:00
|
|
|
|
c->data.i = men;
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return c;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Mapperfunktion f<>r das Anlegen neuer curse. Automatisch wird zum
|
|
|
|
|
* passenden Typ verzweigt und die relevanten Variablen weitergegeben.
|
|
|
|
|
*/
|
2011-03-07 08:02:35 +01:00
|
|
|
|
curse *create_curse(unit * magician, attrib ** ap, const curse_type * ct,
|
2015-05-15 19:08:44 +02:00
|
|
|
|
double vigour, int duration, double effect, int men)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse *c;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
/* die Kraft eines Spruchs darf nicht 0 sein */
|
|
|
|
|
assert(vigour > 0);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
c = get_curse(*ap, ct);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (c && (c_flags(c) & CURSE_ONLYONE)) {
|
|
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
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 += effect;
|
|
|
|
|
}
|
|
|
|
|
if (ct->mergeflags & M_MAXEFFECT) {
|
|
|
|
|
c->effect = _max(c->effect, effect);
|
|
|
|
|
}
|
|
|
|
|
if (ct->mergeflags & M_VIGOUR) {
|
|
|
|
|
c->vigour = _max(vigour, c->vigour);
|
|
|
|
|
}
|
|
|
|
|
if (ct->mergeflags & M_VIGOUR_ADD) {
|
|
|
|
|
c->vigour = vigour + c->vigour;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (ct->mergeflags & M_MEN) {
|
|
|
|
|
switch (ct->typ) {
|
|
|
|
|
case CURSETYP_UNIT:
|
|
|
|
|
{
|
|
|
|
|
c->data.i += men;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
set_curseingmagician(magician, *ap, ct);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
c = make_curse(magician, ap, ct, vigour, duration, effect, men);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return c;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* hier m<>ssen alle c-typen, die auf Einheiten gezaubert werden k<>nnen,
|
|
|
|
|
* ber<EFBFBD>cksichtigt werden */
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static void do_transfer_curse(curse * c, unit * u, unit * u2, int n)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
int cursedmen = 0;
|
|
|
|
|
int men = get_cursedmen(u, c);
|
|
|
|
|
bool dogive = false;
|
|
|
|
|
const curse_type *ct = c->type;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
switch ((ct->flags | c->flags) & CURSE_SPREADMASK) {
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case CURSE_SPREADALWAYS:
|
2014-10-18 18:37:44 +02:00
|
|
|
|
dogive = true;
|
|
|
|
|
men = u2->number + n;
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
case CURSE_SPREADMODULO:
|
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
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;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2014-10-18 18:37:44 +02:00
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
case CURSE_SPREADCHANCE:
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (chance(u2->number / (double)(u2->number + n))) {
|
|
|
|
|
men = u2->number + n;
|
|
|
|
|
dogive = true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
case CURSE_SPREADNEVER:
|
2014-10-18 18:37:44 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (dogive) {
|
|
|
|
|
curse *cnew = make_curse(c->magician, &u2->attribs, c->type, c->vigour,
|
|
|
|
|
c->duration, c->effect, men);
|
|
|
|
|
cnew->flags = c->flags;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (ct->typ == CURSETYP_UNIT)
|
|
|
|
|
set_cursedmen(cnew, men);
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
void transfer_curse(unit * u, unit * u2, int n)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
attrib *a;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +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;
|
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2015-08-07 15:52:10 +02:00
|
|
|
|
int curse_cansee(const curse *c, const faction *viewer, objtype_t typ, const void *obj, int self) {
|
|
|
|
|
if (self < 3 && c->magician && c->magician->faction == viewer) {
|
|
|
|
|
// magicians can see their own curses better than anybody, no exceptions
|
|
|
|
|
self = 3;
|
|
|
|
|
}
|
|
|
|
|
else if (c->type->cansee) {
|
|
|
|
|
self = c->type->cansee(viewer, obj, typ, c, self);
|
|
|
|
|
}
|
|
|
|
|
return self;
|
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool curse_active(const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (!c)
|
|
|
|
|
return false;
|
|
|
|
|
if (c_flags(c) & CURSE_ISNEW)
|
|
|
|
|
return false;
|
|
|
|
|
if (c->vigour <= 0)
|
|
|
|
|
return false;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return true;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool is_cursed_internal(attrib * ap, const curse_type * ct)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
curse *c = get_curse(ap, ct);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (!c)
|
|
|
|
|
return false;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return true;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-24 07:41:07 +02:00
|
|
|
|
bool is_cursed_with(const attrib * ap, const curse * c)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
const attrib *a = ap;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
while (a) {
|
|
|
|
|
if ((a->type->flags & ATF_CURSE) && (c == (const curse *)a->data.v)) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
a = a->next;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return false;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
2010-08-08 10:06:34 +02:00
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* cursedata */
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
static const char *oldnames[MAXCURSE] = {
|
2014-10-18 18:37:44 +02:00
|
|
|
|
/* OBS: when removing curses, remember to update read_ccompat() */
|
|
|
|
|
"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",
|
|
|
|
|
"godcursezone",
|
|
|
|
|
"speed",
|
|
|
|
|
"orcish",
|
|
|
|
|
"magicboost",
|
2015-07-07 21:10:18 +02:00
|
|
|
|
"insectfur"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
};
|
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
|
const char *oldcursename(int id)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
return oldnames[id];
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
2012-05-17 09:13:30 +02:00
|
|
|
|
message *cinfo_simple(const void *obj, objtype_t typ, const struct curse * c,
|
2014-10-18 18:37:44 +02:00
|
|
|
|
int self)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
struct message *msg;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
unused_arg(typ);
|
|
|
|
|
unused_arg(self);
|
|
|
|
|
unused_arg(obj);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
2014-10-18 18:37:44 +02:00
|
|
|
|
msg = msg_message(mkname("curseinfo", c->type->cname), "id", c->no);
|
|
|
|
|
if (msg == NULL) {
|
|
|
|
|
log_error("There is no curseinfo for %s.\n", c->type->cname);
|
|
|
|
|
}
|
|
|
|
|
return msg;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
/* Antimagie - curse aufl<66>sen */
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
/* Wenn der Curse schw<68>cher ist als der cast_level, dann wird er
|
|
|
|
|
* aufgel<EFBFBD>st, bzw seine Kraft (vigour) auf 0 gesetzt.
|
|
|
|
|
* Ist der cast_level zu gering, hat die Antimagie nur mit einer Chance
|
|
|
|
|
* von 100-20*Stufenunterschied % eine Wirkung auf den Curse. Dann wird
|
|
|
|
|
* die Kraft des Curse um die halbe St<EFBFBD>rke der Antimagie reduziert.
|
|
|
|
|
* Zur<EFBFBD>ckgegeben wird der noch unverbrauchte Rest von force.
|
|
|
|
|
*/
|
2015-05-15 19:08:44 +02:00
|
|
|
|
double destr_curse(curse * c, int cast_level, double force)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
{
|
2014-10-18 18:37:44 +02:00
|
|
|
|
if (cast_level < c->vigour) { /* Zauber ist nicht stark genug */
|
|
|
|
|
double probability = 0.1 + (cast_level - c->vigour) * 0.2;
|
|
|
|
|
/* pro Stufe Unterschied -20% */
|
|
|
|
|
if (chance(probability)) {
|
|
|
|
|
force -= c->vigour;
|
|
|
|
|
if (c->type->change_vigour) {
|
2015-05-15 19:08:44 +02:00
|
|
|
|
c->type->change_vigour(c, -(cast_level + 1) / 2);
|
2014-10-18 18:37:44 +02:00
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
c->vigour -= cast_level + 1 / 2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else { /* Zauber ist st<73>rker als curse */
|
|
|
|
|
if (force >= c->vigour) { /* reicht die Kraft noch aus? */
|
|
|
|
|
force -= c->vigour;
|
|
|
|
|
if (c->type->change_vigour) {
|
|
|
|
|
c->type->change_vigour(c, -c->vigour);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
c->vigour = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return force;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
}
|
2015-10-15 08:38:43 +02:00
|
|
|
|
|
2016-02-26 17:54:25 +01:00
|
|
|
|
void curses_done(void) {
|
2015-10-15 08:38:43 +02:00
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i != MAXCTHASH; ++i) {
|
|
|
|
|
ql_free(cursetypes[i]);
|
2016-09-19 03:52:10 +02:00
|
|
|
|
cursetypes[i] = 0;
|
2015-10-15 08:38:43 +02:00
|
|
|
|
}
|
2016-09-19 03:52:10 +02:00
|
|
|
|
++ct_changes;
|
2016-02-25 23:11:44 +01:00
|
|
|
|
}
|