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>
|
2015-01-30 20:37:14 +01: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 "skill.h"
|
|
|
|
|
|
|
|
#include "curse.h"
|
|
|
|
#include "race.h"
|
|
|
|
#include "region.h"
|
|
|
|
#include "terrain.h"
|
|
|
|
#include "terrainid.h"
|
|
|
|
#include "unit.h"
|
|
|
|
|
|
|
|
#include <util/attrib.h>
|
|
|
|
#include <util/goodies.h>
|
|
|
|
#include <util/language.h>
|
|
|
|
#include <util/log.h>
|
|
|
|
#include <util/rng.h>
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <assert.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
/** skillmod attribut **/
|
2018-02-09 21:20:43 +01:00
|
|
|
static void init_skillmod(variant *var)
|
2011-03-07 08:02:35 +01:00
|
|
|
{
|
2018-02-09 21:20:43 +01:00
|
|
|
var->v = calloc(sizeof(skillmod_data), 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/** temporary skill modification (NOT SAVED!). */
|
|
|
|
attrib_type at_skillmod = {
|
2015-01-30 20:37:14 +01:00
|
|
|
"skillmod",
|
|
|
|
init_skillmod,
|
2018-02-09 21:20:43 +01:00
|
|
|
a_free_voidptr,
|
2015-01-30 20:37:14 +01:00
|
|
|
NULL,
|
|
|
|
NULL, /* can't write function pointers */
|
|
|
|
NULL, /* can't read function pointers */
|
2016-02-09 06:43:19 +01:00
|
|
|
NULL,
|
2015-01-30 20:37:14 +01:00
|
|
|
ATF_PRESERVE
|
2010-08-08 10:06:34 +02:00
|
|
|
};
|
|
|
|
|
2017-04-02 20:17:39 +02:00
|
|
|
attrib *make_skillmod(skill_t sk, skillmod_fun special, double multiplier, int bonus)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
attrib *a = a_new(&at_skillmod);
|
|
|
|
skillmod_data *smd = (skillmod_data *)a->data.v;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
smd->skill = sk;
|
|
|
|
smd->special = special;
|
|
|
|
smd->bonus = bonus;
|
|
|
|
smd->multiplier = multiplier;
|
2011-03-07 08:02:35 +01:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
return a;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2017-04-02 20:17:39 +02:00
|
|
|
skillmod(const unit * u, const region * r, skill_t sk, int value)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-04-02 20:17:39 +02:00
|
|
|
const attrib * a = u->attribs;
|
2015-01-30 20:37:14 +01:00
|
|
|
for (a = a_find((attrib *)a, &at_skillmod); a && a->type == &at_skillmod;
|
|
|
|
a = a->next) {
|
|
|
|
skillmod_data *smd = (skillmod_data *)a->data.v;
|
|
|
|
if (smd->skill != NOSKILL && smd->skill != sk)
|
|
|
|
continue;
|
|
|
|
if (smd->special) {
|
|
|
|
value = smd->special(u, r, sk, value);
|
|
|
|
if (value < 0)
|
|
|
|
return value; /* pass errors back to caller */
|
|
|
|
}
|
|
|
|
if (smd->multiplier)
|
|
|
|
value = (int)(value * smd->multiplier);
|
|
|
|
value += smd->bonus;
|
2011-03-07 08:02:35 +01:00
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
return value;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int skill_mod(const race * rc, skill_t sk, const struct terrain_type *terrain)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
int result = 0;
|
2016-09-19 08:25:39 +02:00
|
|
|
static int rc_cache;
|
2016-09-20 20:27:41 +02:00
|
|
|
static const race *rc_dwarf, *rc_insect;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-09-19 08:25:39 +02:00
|
|
|
if (rc_changed(&rc_cache)) {
|
|
|
|
rc_dwarf = get_race(RC_DWARF);
|
|
|
|
rc_insect = get_race(RC_INSECT);
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
result = rc->bonus[sk];
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2016-09-19 08:25:39 +02:00
|
|
|
if (rc == rc_dwarf) {
|
2015-01-30 20:37:14 +01:00
|
|
|
if (sk == SK_TACTICS) {
|
|
|
|
if (terrain == newterrain(T_MOUNTAIN) || fval(terrain, ARCTIC_REGION))
|
|
|
|
++result;
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2016-09-19 08:25:39 +02:00
|
|
|
else if (rc == rc_insect) {
|
2015-01-30 20:37:14 +01:00
|
|
|
if (terrain == newterrain(T_MOUNTAIN) || fval(terrain, ARCTIC_REGION))
|
|
|
|
--result;
|
|
|
|
else if (terrain == newterrain(T_DESERT) || terrain == newterrain(T_SWAMP))
|
|
|
|
++result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int rc_skillmod(const struct race *rc, const region * r, skill_t sk)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
int mods = 0;
|
|
|
|
if (!skill_enabled(sk)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (r) {
|
|
|
|
mods = skill_mod(rc, sk, r->terrain);
|
|
|
|
}
|
2016-09-19 08:25:39 +02:00
|
|
|
if (r && r_isforest(r)) {
|
|
|
|
static int rc_cache;
|
2016-09-20 20:27:41 +02:00
|
|
|
static const race * rc_elf;
|
2016-09-19 08:25:39 +02:00
|
|
|
if (rc_changed(&rc_cache)) {
|
|
|
|
rc_elf = get_race(RC_ELF);
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2016-09-19 08:25:39 +02:00
|
|
|
if (rc == rc_elf) {
|
|
|
|
if (sk == SK_PERCEPTION || sk == SK_STEALTH) {
|
|
|
|
++mods;
|
|
|
|
}
|
|
|
|
else if (sk == SK_TACTICS) {
|
|
|
|
mods += 2;
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
return mods;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int level_days(int level)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-07-28 09:55:39 +02:00
|
|
|
/* FIXME STUDYDAYS * ((level + 1) * level / 2); */
|
2015-01-30 20:37:14 +01:00
|
|
|
return 30 * ((level + 1) * level / 2);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int level(int days)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
int i;
|
|
|
|
static int ldays[64];
|
|
|
|
static bool init = false;
|
|
|
|
if (!init) {
|
|
|
|
init = true;
|
|
|
|
for (i = 0; i != 64; ++i)
|
|
|
|
ldays[i] = level_days(i + 1);
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
for (i = 0; i != 64; ++i)
|
2015-01-30 20:37:14 +01:00
|
|
|
if (ldays[i] > days)
|
|
|
|
return i;
|
|
|
|
return i;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void sk_set(skill * sv, int level)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2016-03-17 21:21:23 +01:00
|
|
|
assert(sv && level != 0);
|
2015-08-15 20:25:36 +02:00
|
|
|
sv->weeks = skill_weeks(level);
|
|
|
|
sv->level = level;
|
2018-01-26 18:18:12 +01:00
|
|
|
assert(sv->weeks <= sv->level * 2 + 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2016-09-23 20:36:57 +02:00
|
|
|
static bool rule_random_progress(void)
|
2011-03-07 08:02:35 +01:00
|
|
|
{
|
2016-09-23 20:36:57 +02:00
|
|
|
static int rule, config;
|
|
|
|
if (config_changed(&config)) {
|
|
|
|
rule = config_get_int("study.random_progress", 1);
|
|
|
|
}
|
|
|
|
return rule != 0;
|
2010-10-09 08:39:16 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int skill_weeks(int level)
|
2010-08-08 10:06:34 +02:00
|
|
|
/* how many weeks must i study to get from level to level+1 */
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
if (rule_random_progress()) {
|
|
|
|
int coins = 2 * level;
|
|
|
|
int heads = 1;
|
|
|
|
while (coins--) {
|
|
|
|
heads += rng_int() % 2;
|
|
|
|
}
|
|
|
|
return heads;
|
2010-10-09 08:39:16 +02:00
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
return level + 1;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2018-01-17 19:23:10 +01:00
|
|
|
void increase_skill(unit * u, skill_t sk, int weeks)
|
2017-07-31 12:55:05 +02:00
|
|
|
{
|
|
|
|
skill *sv = unit_skill(u, sk);
|
2018-01-17 19:23:10 +01:00
|
|
|
assert(weeks >= 0);
|
2017-07-31 12:55:05 +02:00
|
|
|
if (!sv) {
|
|
|
|
sv = add_skill(u, sk);
|
|
|
|
}
|
2018-01-17 19:23:10 +01:00
|
|
|
while (sv->weeks <= weeks) {
|
2017-07-31 12:55:05 +02:00
|
|
|
weeks -= sv->weeks;
|
|
|
|
sk_set(sv, sv->level + 1);
|
|
|
|
}
|
|
|
|
sv->weeks -= weeks;
|
2018-01-26 18:18:12 +01:00
|
|
|
assert(sv->weeks <= sv->level * 2 + 1);
|
2017-07-31 12:55:05 +02:00
|
|
|
}
|
|
|
|
|
2018-01-17 19:23:10 +01:00
|
|
|
void reduce_skill(unit * u, skill * sv, int weeks)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-01-17 19:23:10 +01:00
|
|
|
int max_weeks = sv->level + 1;
|
|
|
|
|
|
|
|
assert(weeks >= 0);
|
|
|
|
if (rule_random_progress()) {
|
|
|
|
max_weeks += sv->level;
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
sv->weeks += weeks;
|
2018-01-17 19:23:10 +01:00
|
|
|
while (sv->level > 0 && sv->weeks > max_weeks) {
|
2015-01-30 20:37:14 +01:00
|
|
|
sv->weeks -= sv->level;
|
|
|
|
--sv->level;
|
2018-01-26 18:18:12 +01:00
|
|
|
max_weeks -= 2;
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
if (sv->level == 0) {
|
|
|
|
/* reroll */
|
2018-01-17 19:23:10 +01:00
|
|
|
sv->weeks = skill_weeks(sv->level);
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2018-01-26 18:18:12 +01:00
|
|
|
assert(sv->weeks <= sv->level * 2 + 1);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
int skill_compare(const skill * sk, const skill * sc)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
if (sk->level > sc->level)
|
|
|
|
return 1;
|
|
|
|
if (sk->level < sc->level)
|
|
|
|
return -1;
|
|
|
|
if (sk->weeks < sc->weeks)
|
|
|
|
return 1;
|
|
|
|
if (sk->weeks > sc->weeks)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|