2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2001-04-14 13:39:14 +02:00
|
|
|
|
*
|
2003-07-29 11:48:03 +02:00
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
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 "skill.h"
|
|
|
|
|
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "curse.h"
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include "karma.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include "item.h"
|
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "region.h"
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include "terrain.h"
|
|
|
|
|
#include "terrainid.h"
|
|
|
|
|
#include "unit.h"
|
2005-06-10 00:10:35 +02:00
|
|
|
|
|
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
#include <util/goodies.h>
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include <util/log.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/rng.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* libc includes */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include <assert.h>
|
2002-02-23 17:18:26 +01:00
|
|
|
|
#include <math.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include <stdlib.h>
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#include <string.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
const char *skillnames[MAXSKILLS] =
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2005-10-25 16:27:24 +02:00
|
|
|
|
"alchemy",
|
|
|
|
|
"crossbow",
|
|
|
|
|
"mining",
|
|
|
|
|
"bow",
|
|
|
|
|
"building",
|
|
|
|
|
"trade",
|
|
|
|
|
"forestry",
|
|
|
|
|
"catapult",
|
|
|
|
|
"herbalism",
|
|
|
|
|
"magic",
|
|
|
|
|
"training",
|
|
|
|
|
"riding",
|
|
|
|
|
"armorer",
|
|
|
|
|
"shipcraft",
|
|
|
|
|
"melee",
|
|
|
|
|
"sailing",
|
|
|
|
|
"polearm",
|
|
|
|
|
"espionage",
|
|
|
|
|
"quarrying",
|
|
|
|
|
"roadwork",
|
|
|
|
|
"tactics",
|
|
|
|
|
"stealth",
|
|
|
|
|
"entertainment",
|
|
|
|
|
"weaponsmithing",
|
|
|
|
|
"cartmaking",
|
|
|
|
|
"perception",
|
|
|
|
|
"taxation",
|
|
|
|
|
"stamina",
|
|
|
|
|
"unarmed"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
};
|
|
|
|
|
|
2005-11-25 23:09:59 +01:00
|
|
|
|
static boolean skill_enabled[MAXSKILLS];
|
|
|
|
|
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *
|
2001-04-16 16:34:19 +02:00
|
|
|
|
skillname(skill_t sk, const struct locale * lang)
|
|
|
|
|
{
|
2005-11-25 23:09:59 +01:00
|
|
|
|
if (skill_enabled[sk]) {
|
|
|
|
|
return locale_string(lang, mkname("skill", skillnames[sk]));
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
enable_skill(const char * skname, boolean value)
|
|
|
|
|
{
|
|
|
|
|
skill_t sk;
|
2005-11-25 23:17:41 +01:00
|
|
|
|
for (sk=0;sk!=MAXSKILLS;++sk) {
|
2005-11-25 23:09:59 +01:00
|
|
|
|
if (strcmp(skillnames[sk], skname)==0) {
|
|
|
|
|
skill_enabled[sk] = value;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
log_error(("Trying to set unknown skill %s to %u", skname, value));
|
2001-04-16 16:34:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
skill_t
|
|
|
|
|
sk_find(const char * name)
|
|
|
|
|
{
|
2007-03-11 21:52:09 +01:00
|
|
|
|
skill_t i;
|
|
|
|
|
if (name==NULL) return NOSKILL;
|
2005-10-25 16:27:24 +02:00
|
|
|
|
if (strncmp(name, "sk_", 3)==0) name+=3;
|
2007-03-11 21:52:09 +01:00
|
|
|
|
for (i=0;i!=MAXSKILLS;++i) {
|
2005-11-25 23:09:59 +01:00
|
|
|
|
if (skill_enabled[i]) {
|
|
|
|
|
if (strcmp(name, skillnames[i])==0) return i;
|
|
|
|
|
}
|
2007-03-11 21:52:09 +01:00
|
|
|
|
}
|
|
|
|
|
return NOSKILL;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/** skillmod attribut **/
|
|
|
|
|
static void
|
|
|
|
|
init_skillmod(attrib * a) {
|
|
|
|
|
a->data.v = calloc(sizeof(skillmod_data), 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
finalize_skillmod(attrib * a)
|
|
|
|
|
{
|
|
|
|
|
free(a->data.v);
|
|
|
|
|
}
|
|
|
|
|
|
2008-03-29 23:06:20 +01:00
|
|
|
|
/** temporary skill modification (NOT SAVED!). */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
attrib_type at_skillmod = {
|
|
|
|
|
"skillmod",
|
|
|
|
|
init_skillmod,
|
|
|
|
|
finalize_skillmod,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL, /* can't write function pointers */
|
|
|
|
|
NULL, /* can't read function pointers */
|
|
|
|
|
ATF_PRESERVE
|
|
|
|
|
};
|
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
attrib *
|
2004-06-11 21:59:02 +02:00
|
|
|
|
make_skillmod(skill_t sk, unsigned int flags, skillmod_fun special, double multiplier, int bonus)
|
2001-02-03 14:45:35 +01:00
|
|
|
|
{
|
|
|
|
|
attrib * a = a_new(&at_skillmod);
|
|
|
|
|
skillmod_data * smd = (skillmod_data*)a->data.v;
|
|
|
|
|
|
2002-02-15 17:13:30 +01:00
|
|
|
|
smd->skill=sk;
|
2001-02-03 14:45:35 +01:00
|
|
|
|
smd->special=special;
|
|
|
|
|
smd->bonus=bonus;
|
|
|
|
|
smd->multiplier=multiplier;
|
|
|
|
|
smd->flags=flags;
|
|
|
|
|
|
|
|
|
|
return a;
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
int
|
|
|
|
|
skillmod(const attrib * a, const unit * u, const region * r, skill_t sk, int value, int flags)
|
|
|
|
|
{
|
2006-02-25 01:12:48 +01:00
|
|
|
|
for (a = a_find((attrib*)a, &at_skillmod); a && a->type==&at_skillmod; a=a->next) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
skillmod_data * smd = (skillmod_data *)a->data.v;
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if (smd->skill!=NOSKILL && smd->skill!=sk) continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (flags!=SMF_ALWAYS && (smd->flags & flags) == 0) continue;
|
|
|
|
|
if (smd->special) {
|
|
|
|
|
value = smd->special(u, r, sk, value);
|
|
|
|
|
if (value<0) return value; /* pass errors back to caller */
|
|
|
|
|
}
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if (smd->multiplier) value = (int)(value*smd->multiplier);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
value += smd->bonus;
|
|
|
|
|
}
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
int
|
2005-10-25 14:38:01 +02:00
|
|
|
|
skill_mod(const race * rc, skill_t sk, const struct terrain_type * terrain)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2001-12-10 01:13:39 +01:00
|
|
|
|
int result = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
result = rc->bonus[sk];
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-10-25 14:38:01 +02:00
|
|
|
|
if (rc == new_race[RC_DWARF]) {
|
|
|
|
|
if (sk==SK_TACTICS) {
|
2005-11-26 00:52:53 +01:00
|
|
|
|
if (terrain == newterrain(T_MOUNTAIN) || fval(terrain, ARCTIC_REGION))
|
|
|
|
|
++result;
|
2005-10-25 14:38:01 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (rc == new_race[RC_INSECT]) {
|
2005-11-26 00:52:53 +01:00
|
|
|
|
if (terrain == newterrain(T_MOUNTAIN) || fval(terrain, ARCTIC_REGION))
|
2001-01-25 10:37:55 +01:00
|
|
|
|
--result;
|
2005-11-26 00:52:53 +01:00
|
|
|
|
else if (terrain == newterrain(T_DESERT) || terrain == newterrain(T_SWAMP))
|
2001-01-25 10:37:55 +01:00
|
|
|
|
++result;
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#define RCMODMAXHASH 31
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#ifdef FASTER_SKILLMOD
|
2001-12-10 01:13:39 +01:00
|
|
|
|
static struct skillmods {
|
|
|
|
|
struct skillmods * next;
|
|
|
|
|
const struct race * race;
|
|
|
|
|
struct modifiers {
|
|
|
|
|
int value[MAXSKILLS];
|
|
|
|
|
} mod[MAXTERRAINS];
|
|
|
|
|
} * modhash[RCMODMAXHASH];
|
|
|
|
|
|
|
|
|
|
static struct skillmods *
|
|
|
|
|
init_skills(const race * rc)
|
|
|
|
|
{
|
|
|
|
|
terrain_t t;
|
|
|
|
|
struct skillmods *mods = (struct skillmods*)calloc(1, sizeof(struct skillmods));
|
|
|
|
|
mods->race = rc;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
for (t=0;t!=MAXTERRAINS;++t) {
|
|
|
|
|
skill_t sk;
|
|
|
|
|
for (sk=0;sk!=MAXSKILLS;++sk) {
|
2005-10-25 14:38:01 +02:00
|
|
|
|
mods->mod[t].value[sk] = skill_mod(rc, sk, newterrain(t));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return mods;
|
|
|
|
|
}
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#endif
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
|
|
|
|
int
|
2002-02-15 17:13:30 +01:00
|
|
|
|
rc_skillmod(const struct race * rc, const region *r, skill_t sk)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2001-12-10 01:13:39 +01:00
|
|
|
|
int mods;
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#ifdef FASTER_SKILLMOD
|
|
|
|
|
unsigned int index = hashstring(rc->_name[0]) % RCMODMAXHASH;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
struct skillmods **imods = &modhash[index];
|
|
|
|
|
while (*imods && (*imods)->race!=rc) imods = &(*imods)->next;
|
|
|
|
|
if (*imods==NULL) {
|
|
|
|
|
*imods = init_skills(rc);
|
|
|
|
|
}
|
2002-02-15 17:13:30 +01:00
|
|
|
|
mods = (*imods)->mod[rterrain(r)].value[sk];
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#else
|
|
|
|
|
mods = skill_mod(rc, sk, r->terrain);
|
|
|
|
|
#endif
|
2002-02-15 17:13:30 +01:00
|
|
|
|
if (rc == new_race[RC_ELF] && r_isforest(r)) switch (sk) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
case SK_OBSERVATION:
|
|
|
|
|
++mods;
|
|
|
|
|
break;
|
|
|
|
|
case SK_STEALTH:
|
|
|
|
|
if (r_isforest(r)) ++mods;
|
|
|
|
|
break;
|
|
|
|
|
case SK_TACTICS:
|
|
|
|
|
if (r_isforest(r)) mods += 2;
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
|
|
|
|
return mods;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
skill_init(void)
|
|
|
|
|
{
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
void
|
|
|
|
|
skill_done(void)
|
|
|
|
|
{
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#ifdef FASTER_SKILLMOD
|
2001-12-10 01:13:39 +01:00
|
|
|
|
int i;
|
|
|
|
|
for (i = 0;i!=RCMODMAXHASH;++i) {
|
|
|
|
|
while (modhash[i]) {
|
|
|
|
|
struct skillmods * mods = modhash[i];
|
|
|
|
|
modhash[i] = mods->next;
|
|
|
|
|
free(mods);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
level_days(int level)
|
|
|
|
|
{
|
|
|
|
|
return 30 * ((level+1) * level / 2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
level(int days)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
static int ldays[64];
|
2001-01-25 10:37:55 +01:00
|
|
|
|
static boolean init = false;
|
|
|
|
|
if (!init) {
|
|
|
|
|
init = true;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
for (i=0;i!=64;++i) ldays[i] = level_days(i+1);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
for (i=0;i!=64;++i) if (ldays[i]>days) return i;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-23 17:18:26 +01:00
|
|
|
|
void
|
|
|
|
|
sk_set(skill * sv, int level)
|
|
|
|
|
{
|
2002-03-11 01:06:14 +01:00
|
|
|
|
assert(level!=0);
|
2002-02-23 17:18:26 +01:00
|
|
|
|
sv->weeks = (unsigned char)skill_weeks(level);
|
|
|
|
|
sv->level = (unsigned char)level;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
skill_weeks(int level)
|
|
|
|
|
/* how many weeks must i study to get from level to level+1 */
|
|
|
|
|
{
|
2002-02-24 12:04:23 +01:00
|
|
|
|
int coins = 2*level;
|
|
|
|
|
int heads = 1;
|
|
|
|
|
while (coins--) {
|
2006-02-19 23:43:56 +01:00
|
|
|
|
heads += rng_int() % 2;
|
2002-02-24 12:04:23 +01:00
|
|
|
|
}
|
|
|
|
|
return heads;
|
2002-02-23 17:18:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-10-08 08:46:03 +02:00
|
|
|
|
reduce_skill(unit * u, skill * sv, unsigned int weeks)
|
2002-02-23 17:18:26 +01:00
|
|
|
|
{
|
2003-07-29 11:48:03 +02:00
|
|
|
|
sv->weeks+=weeks;
|
|
|
|
|
while (sv->level>0 && sv->level*2+1<sv->weeks) {
|
|
|
|
|
sv->weeks -= sv->level;
|
2002-02-23 17:18:26 +01:00
|
|
|
|
--sv->level;
|
|
|
|
|
}
|
2002-10-08 08:46:03 +02:00
|
|
|
|
if (sv->level==0) {
|
|
|
|
|
/* reroll */
|
|
|
|
|
sv->weeks = (unsigned char)skill_weeks(sv->level);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-10-08 08:46:03 +02:00
|
|
|
|
|
2002-02-15 17:13:30 +01:00
|
|
|
|
int
|
|
|
|
|
skill_compare(const skill * sk, const skill * sc)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2002-02-15 17:13:30 +01:00
|
|
|
|
if (sk->level > sc->level) return 1;
|
|
|
|
|
if (sk->level < sc->level) return -1;
|
2002-02-23 17:18:26 +01:00
|
|
|
|
if (sk->weeks < sc->weeks) return 1;
|
|
|
|
|
if (sk->weeks > sc->weeks) return -1;
|
2002-02-15 17:13:30 +01:00
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|