diff --git a/conf/e2/config.json b/conf/e2/config.json index fb231e2eb..b8ca2f178 100644 --- a/conf/e2/config.json +++ b/conf/e2/config.json @@ -35,6 +35,7 @@ "rules.guard.guard_number_stop_prob": 0.001, "rules.guard.castle_stop_prob": 0.05, "rules.guard.region_type_stop_prob": 0.05, - "rules.economy.repopulate_maximum": 500 + "rules.economy.repopulate_maximum": 500, + "rules.lighthouse.unit_capacity": true } } diff --git a/conf/e3/config.json b/conf/e3/config.json index bfd1bd0fc..3305ce221 100644 --- a/conf/e3/config.json +++ b/conf/e3/config.json @@ -92,6 +92,7 @@ "rules.grow.formula": 1, "rules.tactics.formula": 1, "rules.help.mask": "fight guard money give", + "rules.lighthouse.unit_capacity": true, "movement.shipspeed.skillbonus": 6, "alliance.auto": "fight", "alliance.restricted": "fight" diff --git a/conf/keywords.json b/conf/keywords.json index 4c9fa905d..4f8f27f5b 100644 --- a/conf/keywords.json +++ b/conf/keywords.json @@ -1,7 +1,8 @@ { "keywords": { "en" : { - "grow": [ "GROW", "BREED", "PLANT" ], + "plant": "PLANT", + "grow": [ "GROW", "BREED" ], "promote": ["PROMOTE", "PROMOTION" ], "combat": [ "COMBAT", "FIGHT" ] }, @@ -53,7 +54,8 @@ "cast": ["ZAUBERE", "ZAUBERN"], "show": ["ZEIGE", "ZEIGEN"], "destroy": ["ZERSTÖRE", "ZERSTÖREN"], - "grow": ["ZÃœCHTE", "PFLANZE", "ZÃœCHTEN", "PFLANZEN"], + "grow": ["ZÃœCHTE", "ZÃœCHTEN"], + "plant": ["PFLANZE", "PFLANZEN"], "default": "DEFAULT", "origin": "URSPRUNG", "email": "EMAIL", diff --git a/res/buildings/castle-2.xml b/res/buildings/castle-2.xml index 7e64f88af..accb92c08 100644 --- a/res/buildings/castle-2.xml +++ b/res/buildings/castle-2.xml @@ -1,5 +1,5 @@ - + diff --git a/res/buildings/castle.xml b/res/buildings/castle.xml index 78ce7066c..ce64e8030 100644 --- a/res/buildings/castle.xml +++ b/res/buildings/castle.xml @@ -1,5 +1,5 @@ - + diff --git a/res/core/common/buildings.xml b/res/core/common/buildings.xml index a799ce309..932326f1c 100644 --- a/res/core/common/buildings.xml +++ b/res/core/common/buildings.xml @@ -1,26 +1,26 @@ - + - + - + - + - + @@ -30,10 +30,10 @@ - + - + @@ -41,10 +41,10 @@ - + - + @@ -52,10 +52,10 @@ - + - + @@ -63,7 +63,7 @@ - + @@ -72,7 +72,7 @@ - + @@ -82,7 +82,7 @@ - + @@ -92,7 +92,7 @@ - + @@ -107,7 +107,7 @@ - + @@ -119,7 +119,7 @@ - + @@ -127,16 +127,16 @@ - + - + - + @@ -146,7 +146,7 @@ - + @@ -156,7 +156,7 @@ - + diff --git a/res/core/messages.xml b/res/core/messages.xml index 5bb36e345..a500a41ac 100644 --- a/res/core/messages.xml +++ b/res/core/messages.xml @@ -1062,16 +1062,6 @@ "$unit($unit) in $region($region): '$order($command)' - $race($race,0) können nichts stehelen." "$unit($unit) in $region($region): '$order($command)' - $race($race,0) cannot steal anything." - - - - - - - - "$unit($unit) in $region($region): '$order($command)' - $race($race,0) geben nichts weg." - "$unit($unit) in $region($region): '$order($command)' - $race($race,0) do not give things away." - @@ -8001,14 +7991,6 @@ "$unit($unit) tried but failed to send $unit($target) to another world." - - - - - "NUMMER PARTEI $int36($id): Die Partei kann nicht mehr als einmal ihre Nummer wecheln." - "NUMBER FACTION $int36($id): Your faction can only change its number once." - - diff --git a/res/e3a/buildings.xml b/res/e3a/buildings.xml index 8641fd89e..069a24150 100644 --- a/res/e3a/buildings.xml +++ b/res/e3a/buildings.xml @@ -3,7 +3,7 @@ - + @@ -15,7 +15,7 @@ - + diff --git a/res/e3a/races.xml b/res/e3a/races.xml index 8b0cfae6d..79d9ff9bd 100644 --- a/res/e3a/races.xml +++ b/res/e3a/races.xml @@ -700,7 +700,11 @@ - + @@ -732,7 +736,7 @@ - + @@ -750,7 +754,7 @@ - + @@ -767,7 +771,7 @@ - + @@ -783,7 +787,7 @@ - + @@ -797,7 +801,7 @@ - + @@ -812,7 +816,7 @@ - + diff --git a/res/eressea/buildings.xml b/res/eressea/buildings.xml index 670ab484e..1559fd987 100644 --- a/res/eressea/buildings.xml +++ b/res/eressea/buildings.xml @@ -2,7 +2,7 @@ - - + + diff --git a/res/eressea/races.xml b/res/eressea/races.xml index 986e2b85f..04f621526 100644 --- a/res/eressea/races.xml +++ b/res/eressea/races.xml @@ -732,7 +732,7 @@ - + @@ -983,7 +983,7 @@ - + @@ -1000,7 +1000,7 @@ - + @@ -1016,7 +1016,7 @@ - + @@ -1031,7 +1031,7 @@ - + @@ -1044,7 +1044,7 @@ - + @@ -1058,7 +1058,7 @@ - + @@ -1172,7 +1172,7 @@ - + diff --git a/res/races/orc.xml b/res/races/orc.xml index 4f4d96ed5..a0033759a 100644 --- a/res/races/orc.xml +++ b/res/races/orc.xml @@ -1,5 +1,5 @@ - + diff --git a/s/install b/s/install index 27cf185af..5115ca8ee 100755 --- a/s/install +++ b/s/install @@ -1,18 +1,25 @@ #!/bin/sh -ROOT=$(pwd) -while [ ! -d $ROOT/.git ]; do - ROOT=$(dirname $ROOT) - if [ "/" = "$ROOT" ]; then - echo "could not find root, are you in the git repository?" - exit - fi + +FORCE=0 +while getopts f o; do + case "${o}" in + f) usage ; FORCE=yes ;; + esac done +if [ "$FORCE" != "yes" ] ; then + BRANCH=$(git status -s -b | head -1 | cut -d\ -f 2 | sed 's/\..*//') + if [ "$BRANCH" != "master" ] ; then + echo "you should only install stable versions from the master branch" + exit + fi +fi [ -z "$CC" ] && [ ! -z `which clang` ] && CC="clang" [ -z "$CC" ] && [ ! -z `which gcc` ] && CC="gcc" [ -z "$CC" ] && [ ! -z `which tcc` ] && CC="tcc" [ -z "$CC" ] && [ ! -z `which cc` ] && CC="cc" BIN_DIR="Debug" +ROOT=$(git rev-parse --show-toplevel) cd $ROOT/$BIN_DIR make install diff --git a/scripts/tests/common.lua b/scripts/tests/common.lua index 936d2a79b..2f10b3e8a 100644 --- a/scripts/tests/common.lua +++ b/scripts/tests/common.lua @@ -51,7 +51,7 @@ function test_flags() local f = create_faction('halfling') local u = unit.create(f, r, 1) local no = itoa36(f.id) - local flags = 50332673 + local flags = 50332672 f.flags = flags eressea.write_game("test.dat") diff --git a/scripts/tests/e2/buildings.lua b/scripts/tests/e2/buildings.lua index 4797643e1..02298bd97 100644 --- a/scripts/tests/e2/buildings.lua +++ b/scripts/tests/e2/buildings.lua @@ -51,3 +51,27 @@ function test_build_castle_stages() process_orders() assert_equal(250, b.size) end + +function test_build_harbour() +-- try to reproduce mantis bug 2221 + local r = region.create(0, 0, "plain") + local f = faction.create("human", "harbour@eressea.de", "de") + local u = unit.create(f, r) + size = 30 + u.number = 20 + u:set_skill("building", 3) + u:add_item("money", size*250) + u:add_item("stone", size*5) + u:add_item("log", size*5) + u:clear_orders() + u:add_order("MACHE HAFEN") + process_orders() + assert_not_nil(u.building) + assert_equal("harbour", u.building.type) + assert_equal(20, u.building.size) + process_orders() + assert_equal(25, u.building.size) + process_orders() + assert_equal(25, u.building.size) +end + diff --git a/scripts/tests/e2/e2features.lua b/scripts/tests/e2/e2features.lua index 1a4047d63..7e143c0cc 100644 --- a/scripts/tests/e2/e2features.lua +++ b/scripts/tests/e2/e2features.lua @@ -62,29 +62,6 @@ function test_dwarf_bonus() assert_equal(70, r:get_resource("iron")) end -function test_build_harbour() --- try to reproduce mantis bug 2221 - local r = region.create(0, 0, "plain") - local f = faction.create("human", "harbour@eressea.de", "de") - local u = unit.create(f, r) - size = 30 - u.number = 20 - u:set_skill("building", 3) - u:add_item("money", size*250) - u:add_item("stone", size*5) - u:add_item("log", size*5) - u:clear_orders() - u:add_order("MACHE HAFEN") - process_orders() - assert_not_nil(u.building) - assert_equal("harbour", u.building.type) - assert_equal(20, u.building.size) - process_orders() - assert_equal(25, u.building.size) - process_orders() - assert_equal(25, u.building.size) -end - local function one_unit(r, f) local u = unit.create(f, r, 1) u:add_item("money", u.number * 100) @@ -408,3 +385,19 @@ end function test_calendar_season_2328() assert_equal("fall", get_season(1026)) end + +function test_give_to_other_okay() + -- can give a person to another faction + eressea.settings.set("GiveRestriction", "0") + local r = region.create(0, 0, "plain") + local f1 = faction.create("human") + local f2 = faction.create("human") + + local u1 = unit.create(f1, r, 2, "human") + local u2 = unit.create(f2, r, 1, "human") + u2:add_order("KONTAKTIERE " .. itoa36(u1.id)) + u1:add_order("GIB " .. itoa36(u2.id) .. " 1 PERSON") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end diff --git a/scripts/tests/e2/init.lua b/scripts/tests/e2/init.lua index 7fbc0eb95..fd2f0b423 100644 --- a/scripts/tests/e2/init.lua +++ b/scripts/tests/e2/init.lua @@ -21,3 +21,4 @@ require 'tests.process' require 'tests.xmas' require 'tests.production' require 'tests.spells' +require 'tests.undead' diff --git a/scripts/tests/e2/undead.lua b/scripts/tests/e2/undead.lua index 71c14ac36..e137ce3c2 100644 --- a/scripts/tests/e2/undead.lua +++ b/scripts/tests/e2/undead.lua @@ -18,11 +18,11 @@ function test_undead_give_item() assert_equal(0, u1:get_item("log")) end -function test_undead_dont_give_person() +function test_clones_dont_give_person() local r1 = region.create(0, 0, "plain") local f1 = faction.create("human", "hodor@eressea.de", "de") local u1 = unit.create(f1, r1, 2) - u1.race = "undead" + u1.race = "clone" u1:clear_orders() u1:add_item("log", 1) u1:add_order("GIB 0 1 Person") diff --git a/scripts/tests/e3/init.lua b/scripts/tests/e3/init.lua index fc0586f89..04b7da4a2 100644 --- a/scripts/tests/e3/init.lua +++ b/scripts/tests/e3/init.lua @@ -14,3 +14,4 @@ require 'tests.items' require 'tests.magicbag' require 'tests.process' require 'tests.production' +require 'tests.undead' diff --git a/scripts/tests/e3/rules.lua b/scripts/tests/e3/rules.lua index 7eb2f13a3..92e02f29b 100644 --- a/scripts/tests/e3/rules.lua +++ b/scripts/tests/e3/rules.lua @@ -991,3 +991,20 @@ function test_bug2187() set_rule("rules.food.flags", "4") end + + +function test_give_to_other_fails() + -- E3: cannot give a person to another faction + local r = region.create(0, 0, "plain") + local f1 = faction.create("human") + local f2 = faction.create("human") + + local u1 = unit.create(f1, r, 2, "human") + local u2 = unit.create(f2, r, 1, "human") + -- u2:add_order("HELFE " .. itoa36(f1.id) .. " GIB") + u2:add_order("KONTAKTIERE " .. itoa36(u1.id)) + u1:add_order("GIB " .. itoa36(u2.id) .. " 1 PERSON") + process_orders() + assert_equal(2, u1.number) + assert_equal(1, u2.number) +end diff --git a/scripts/tests/undead.lua b/scripts/tests/undead.lua new file mode 100644 index 000000000..0c9f609f1 --- /dev/null +++ b/scripts/tests/undead.lua @@ -0,0 +1,113 @@ +require "lunit" + +module("tests.undead", package.seeall, lunit.testcase) + +function setup() + eressea.free_game() + eressea.settings.set("nmr.timeout", "0") + eressea.settings.set("NewbieImmunity", "0") + eressea.settings.set("rules.food.flags", "4") + eressea.settings.set("rules.encounters", "0") + eressea.settings.set("rules.peasants.growth", "1") + eressea.settings.set("study.random_progress", "0") + eressea.settings.set("GiveRestriction", "0") +end + +function test_give_undead_to_self() + -- generic undead cannot be given + local r = region.create(0, 0, "plain") + local f = faction.create("human") + local u1 = unit.create(f, r, 2, "undead") + local u2 = unit.create(f, r, 1, "undead") + u1:add_order("GIB " .. itoa36(u2.id) .. " 1 PERSON") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end + +function test_give_self_clone_fail() + -- disallow giving basic undead units + local r = region.create(0, 0, "plain") + local f = faction.create("human") + local u1 = unit.create(f, r, 2, "clone") + local u2 = unit.create(f, r, 1, "clone") + u1:add_order("GIB " .. itoa36(u2.id) .. " 1 PERSON") + process_orders() + assert_equal(2, u1.number) + assert_equal(1, u2.number) +end + +local function setup_give_self(race) + local r = region.create(0, 0, "plain") + local f = faction.create("human") + local u1 = unit.create(f, r, 2, race) + local u2 = unit.create(f, r, 1, race) + u1:add_order("GIB " .. itoa36(u2.id) .. " 1 PERSON") + return u1, u2 +end + +local function setup_give_other(race) + local r = region.create(0, 0, "plain") + local f1 = faction.create("human") + local f2 = faction.create("human") + local u1 = unit.create(f1, r, 2, race) + local u2 = unit.create(f2, r, 1, race) + u2:add_order("KONTAKTIERE " .. itoa36(u1.id)) + u1:add_order("GIB " .. itoa36(u2.id) .. " 1 PERSON") + return u1, u2 +end + +function test_give_other_zombie_fail() + u1, u2 = setup_give_other("zombie") + process_orders() + assert_equal(2, u1.number) + assert_equal(1, u2.number) +end + +function test_give_self_zombie_okay() + -- allow giving zombie units to own units of same race + u1, u2 = setup_give_self("zombie") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end + +function test_give_self_skeleton_okay() + -- allow giving skeleton units to own units of same race + u1, u2 = setup_give_self("skeleton") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end + +function test_give_self_ghoul_okay() + -- allow giving ghoul units to own units of same race + u1, u2 = setup_give_self("ghoul") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end + +function test_give_self_ghast_okay() + -- allow giving ghast units to own units of same race + u1, u2 = setup_give_self("ghast") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end + +function test_give_self_juju_okay() + -- allow giving juju units to own units of same race + u1, u2 = setup_give_self("juju") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end + +function test_give_self_skeletonlord_okay() + -- allow giving skeletonlord units to own units of same race + u1, u2 = setup_give_self("skeletonlord") + process_orders() + assert_equal(1, u1.number) + assert_equal(2, u2.number) +end diff --git a/src/academy.c b/src/academy.c index 496c1ef8e..c25ea2476 100644 --- a/src/academy.c +++ b/src/academy.c @@ -25,9 +25,10 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "academy.h" #include "study.h" -void academy_teaching_bonus(struct unit *u, skill_t sk, int academy) { - if (academy && sk != NOSKILL) { - learn_skill(u, sk, academy / STUDYDAYS); +void academy_teaching_bonus(struct unit *u, skill_t sk, int student_days) { + if (student_days && sk != NOSKILL) { + /* actually days / STUDYDAYS * EXPERIENCEDAYS / MAX_STUDENTS */ + learn_skill(u, sk, student_days / STUDYDAYS); } } diff --git a/src/attributes/attributes.c b/src/attributes/attributes.c index f699fdb57..610bea684 100644 --- a/src/attributes/attributes.c +++ b/src/attributes/attributes.c @@ -56,9 +56,98 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include #include +#include #include +#include +typedef struct obs_data { + faction *f; + int skill; + int timer; +} obs_data; + +static void obs_init(struct attrib *a) +{ + a->data.v = malloc(sizeof(obs_data)); +} + +static void obs_done(struct attrib *a) +{ + free(a->data.v); +} + +static int obs_age(struct attrib *a, void *owner) +{ + obs_data *od = (obs_data *)a->data.v; + update_interval(od->f, (region *)owner); + return --od->timer; +} + +static void obs_write(const struct attrib *a, const void *owner, struct storage *store) +{ + obs_data *od = (obs_data *)a->data.v; + write_faction_reference(od->f, store); + WRITE_INT(store, od->skill); + WRITE_INT(store, od->timer); +} + +static int obs_read(struct attrib *a, void *owner, struct gamedata *data) +{ + obs_data *od = (obs_data *)a->data.v; + + read_reference(&od->f, data, read_faction_reference, resolve_faction); + READ_INT(data->store, &od->skill); + READ_INT(data->store, &od->timer); + return AT_READ_OK; +} + +attrib_type at_observer = { "observer", obs_init, obs_done, obs_age, obs_write, obs_read }; + +static attrib *make_observer(faction *f, int perception) +{ + attrib * a = a_new(&at_observer); + obs_data *od = (obs_data *)a->data.v; + od->f = f; + od->skill = perception; + od->timer = 2; + return a; +} + +int get_observer(region *r, faction *f) { + if (fval(r, RF_OBSERVER)) { + attrib *a = a_find(r->attribs, &at_observer); + while (a && a->type == &at_observer) { + obs_data *od = (obs_data *)a->data.v; + if (od->f == f) { + return od->skill; + } + a = a->next; + } + } + return -1; +} + +void set_observer(region *r, faction *f, int skill, int turns) +{ + update_interval(f, r); + if (fval(r, RF_OBSERVER)) { + attrib *a = a_find(r->attribs, &at_observer); + while (a && a->type == &at_observer) { + obs_data *od = (obs_data *)a->data.v; + if (od->f == f && od->skill < skill) { + od->skill = skill; + od->timer = turns; + return; + } + a = a->nexttype; + } + } + else { + fset(r, RF_OBSERVER); + } + a_add(&r->attribs, make_observer(f, skill)); +} attrib_type at_unitdissolve = { "unitdissolve", NULL, NULL, NULL, a_writechars, a_readchars @@ -123,6 +212,7 @@ void register_attributes(void) at_register(&at_stealth); at_register(&at_dict); at_register(&at_unitdissolve); + at_register(&at_observer); at_register(&at_overrideroads); at_register(&at_raceprefix); at_register(&at_iceberg); diff --git a/src/attributes/attributes.h b/src/attributes/attributes.h index 4ec3150ec..19b371dbb 100644 --- a/src/attributes/attributes.h +++ b/src/attributes/attributes.h @@ -23,8 +23,14 @@ extern "C" { #endif struct attrib_type; + struct region; + struct faction; + extern void register_attributes(void); + void set_observer(struct region *r, struct faction *f, int perception, int turns); + int get_observer(struct region *r, struct faction *f); + #ifdef __cplusplus } #endif diff --git a/src/attributes/key.c b/src/attributes/key.c index 1bd5cd9b4..6841bbd71 100644 --- a/src/attributes/key.c +++ b/src/attributes/key.c @@ -22,6 +22,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include +#include #include #include @@ -30,38 +31,107 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. static void a_writekeys(const attrib *a, const void *o, storage *store) { int i, *keys = (int *)a->data.v; - assert(keys[0] < 4096 && keys[0]>0); - WRITE_INT(store, keys[0]); - for (i = 0; i < keys[0]; ++i) { + int n = 0; + if (keys) { + assert(keys[0] < 4096 && keys[0]>0); + n = keys[0]; + } + WRITE_INT(store, n); + for (i = 0; i < n; ++i) { WRITE_INT(store, keys[i * 2 + 1]); WRITE_INT(store, keys[i * 2 + 2]); } } +static int keys_lower_bound(int *base, int k, int l, int r) { + int km = k; + int *p = base + 1; + + while (l != r) { + int m = (l + r) / 2; + km = p[m * 2]; + if (km < k) { + if (l == m) l = r; + else l = m; + } + else { + if (r == m) r = l; + else r = m; + } + } + return l; +} + +static int keys_size(int n) { + /* TODO maybe use log2 from https://graphics.stanford.edu/~seander/bithacks.html#IntegerLog */ + assert(n > 0 && n <= 4096); + if (n <= 1) return 1; + if (n <= 4) return 4; + if (n <= 16) return 16; + if (n <= 256) return 256; + return 4096; +} + static int a_readkeys(attrib * a, void *owner, gamedata *data) { - int i, *p = 0; - READ_INT(data->store, &i); - assert(i < 4096 && i>=0); - if (i == 0) { + int i, n, *keys; + + READ_INT(data->store, &n); + assert(n < 4096 && n >= 0); + if (n == 0) { return AT_READ_FAIL; } - a->data.v = p = malloc(sizeof(int)*(i*2 + 1)); - *p++ = i; - while (i--) { - READ_INT(data->store, p++); + keys = malloc(sizeof(int)*(keys_size(n) * 2 + 1)); + *keys = n; + for (i = 0; i != n; ++i) { + READ_INT(data->store, keys + i * 2 + 1); if (data->version >= KEYVAL_VERSION) { - READ_INT(data->store, p++); + READ_INT(data->store, keys + i * 2 + 2); } else { - *p++ = 1; + keys[i * 2 + 2] = 1; + } + } + if (data->version < SORTKEYS_VERSION) { + int e = 1; + for (i = 1; i != n; ++i) { + int k = keys[i * 2 + 1]; + int v = keys[i * 2 + 2]; + int l = keys_lower_bound(keys, k, 0, e); + if (l != e) { + int km = keys[l * 2 + 1]; + if (km == k) { + int vm = keys[l * 2 + 2]; + if (v != vm) { + log_error("key %d has values %d and %d", k, v, vm); + } + --e; + } + else { + if (e > l) { + memmove(keys + 2 * l + 3, keys + 2 * l + 1, (e - l) * 2 * sizeof(int)); + } + keys[2 * l + 1] = k; + keys[2 * l + 2] = v; + } + } + ++e; + } + if (e != n) { + int sz = keys_size(n); + if (e > sz) { + sz = keys_size(e); + keys = realloc(keys, sizeof(int)*(2 * sz + 1)); + keys[0] = e; + } } } + a->data.v = keys; return AT_READ_OK; } static int a_readkey(attrib *a, void *owner, struct gamedata *data) { int res = a_readint(a, owner, data); - if (data->version>=KEYVAL_VERSION) { + if (data->version >= KEYVAL_VERSION) { return AT_READ_FAIL; } return (res != AT_READ_FAIL) ? AT_READ_DEPR : res; @@ -101,9 +171,55 @@ attrib_type at_key = { a_upgradekeys }; +static int* keys_get(int *base, int i) +{ + int n = base[0]; + assert(i >= 0 && i < n); + return base + 1 + i * 2; +} + +static int *keys_update(int *base, int key, int val) +{ + int *kv; + int n = base[0]; + int l = keys_lower_bound(base, key, 0, n); + if (l < n) { + kv = keys_get(base, l); + if (kv[0] == key) { + kv[1] = val; + } + else { + int sz = keys_size(n); + assert(kv[0] > key); + if (n + 1 > sz) { + ptrdiff_t diff = kv - base; + sz = keys_size(n + 1); + base = realloc(base, (sz * 2 + 1) * sizeof(int)); + kv = base + diff; + } + base[0] = n + 1; + memmove(kv + 2, kv, 2 * sizeof(int) * (n - l)); + kv[0] = key; + kv[1] = val; + } + } + else { + int sz = keys_size(n); + if (n + 1 > sz) { + sz = keys_size(n + 1); + base = realloc(base, (sz * 2 + 1) * sizeof(int)); + } + base[0] = n + 1; + kv = keys_get(base, l); + kv[0] = key; + kv[1] = val; + } + return base; +} + void key_set(attrib ** alist, int key, int val) { - int *keys, n = 0; + int *keys; attrib *a; assert(key != 0); a = a_find(*alist, &at_keys); @@ -111,16 +227,17 @@ void key_set(attrib ** alist, int key, int val) a = a_add(alist, a_new(&at_keys)); } keys = (int *)a->data.v; - if (keys) { - n = keys[0]; + if (!keys) { + int sz = keys_size(1); + a->data.v = keys = malloc((2 * sz + 1) * sizeof(int)); + keys[0] = 1; + keys[1] = key; + keys[2] = val; + } + else { + a->data.v = keys = keys_update(keys, key, val); + assert(keys[0] < 4096 && keys[0] >= 0); } - /* TODO: too many allocations, unsorted array */ - keys = realloc(keys, sizeof(int) *(2 * n + 3)); - keys[0] = n + 1; - assert(keys[0] < 4096 && keys[0]>=0); - keys[2 * n + 1] = key; - keys[2 * n + 2] = val; - a->data.v = keys; } void key_unset(attrib ** alist, int key) @@ -129,15 +246,14 @@ void key_unset(attrib ** alist, int key) assert(key != 0); a = a_find(*alist, &at_keys); if (a) { - int i, *keys = (int *)a->data.v; + int *keys = (int *)a->data.v; if (keys) { int n = keys[0]; - assert(keys[0] < 4096 && keys[0]>0); - for (i = 0; i != n; ++i) { - if (keys[2 * i + 1] == key) { - memmove(keys + 2 * i + 1, keys + 2 * n - 1, 2 * sizeof(int)); - keys[0]--; - break; + int l = keys_lower_bound(keys, key, 0, n); + if (l < n) { + int *kv = keys_get(keys, l); + if (kv[0] == key) { + kv[1] = 0; /* do not delete, just set to 0 */ } } } @@ -149,12 +265,14 @@ int key_get(attrib *alist, int key) { assert(key != 0); a = a_find(alist, &at_keys); if (a) { - int i, *keys = (int *)a->data.v; + int *keys = (int *)a->data.v; if (keys) { - /* TODO: binary search this! */ - for (i = 0; i != keys[0]; ++i) { - if (keys[i*2+1] == key) { - return keys[i * 2 + 2]; + int n = keys[0]; + int l = keys_lower_bound(keys, key, 0, n); + if (l < n) { + int * kv = keys_get(keys, l); + if (kv[0] == key) { + return kv[1]; } } } diff --git a/src/attributes/key.test.c b/src/attributes/key.test.c index b816e1d86..79a7bde1d 100644 --- a/src/attributes/key.test.c +++ b/src/attributes/key.test.c @@ -32,8 +32,8 @@ static void test_upgrade_key(CuTest *tc) { attrib *alist = 0; key_set_orig(&alist, 40); key_set_orig(&alist, 41); - key_set_orig(&alist, 42); key_set_orig(&alist, 43); + key_set_orig(&alist, 42); key_set_orig(&alist, 44); CuAssertPtrNotNull(tc, alist->type->upgrade); alist->type->upgrade(&alist, alist); diff --git a/src/battle.c b/src/battle.c index a9130f2ef..be513f288 100644 --- a/src/battle.c +++ b/src/battle.c @@ -26,6 +26,11 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "monsters.h" #include "move.h" #include "skill.h" +#include "study.h" + +#include +#include +#include #include #include @@ -54,7 +59,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include #include -#include /* util includes */ #include @@ -138,15 +142,10 @@ static int rule_nat_armor; static int rule_cavalry_mode; static int rule_vampire; -static const curse_type *peace_ct, *slave_ct, *calm_ct; - /** initialize rules from configuration. */ static void init_rules(void) { - peace_ct = ct_find("peacezone"); - slave_ct = ct_find("slavery"); - calm_ct = ct_find("calmmonster"); rule_nat_armor = config_get_int("rules.combat.nat_armor", 0); rule_tactics_formula = config_get_int("rules.tactics.formula", 0); rule_goblin_bonus = config_get_int("rules.combat.goblinbonus", 10); @@ -440,7 +439,7 @@ static int get_row(const side * s, int row, const side * vs) return result; } -static int get_unitrow(const fighter * af, const side * vs) +int get_unitrow(const fighter * af, const side * vs) { int row = statusrow(af->status); if (vs == NULL) { @@ -956,20 +955,7 @@ void drain_exp(struct unit *u, int n) } } if (sk != NOSKILL) { - skill *sv = unit_skill(u, sk); - if (sv) { - while (n > 0) { - if (n >= 30 * u->number) { - reduce_skill(u, sv, 1); - n -= 30; - } - else { - if (rng_int() % (30 * u->number) < n) - reduce_skill(u, sv, 1); - n = 0; - } - } - } + reduce_skill_days(u, sk, n); } } @@ -1590,8 +1576,7 @@ static troop select_opponent(battle * b, troop at, int mindist, int maxdist) return dt; } -selist *fighters(battle * b, const side * vs, int minrow, int maxrow, - int mask) +selist *select_fighters(battle * b, const side * vs, int mask, select_fun cb, void *cbdata) { side *s; selist *fightervp = 0; @@ -1614,8 +1599,7 @@ selist *fighters(battle * b, const side * vs, int minrow, int maxrow, assert(mask == (FS_HELP | FS_ENEMY) || !"invalid alliance state"); } for (fig = s->fighters; fig; fig = fig->next) { - int row = get_unitrow(fig, vs); - if (row >= minrow && row <= maxrow) { + if (cb(vs, fig, cbdata)) { selist_push(&fightervp, fig); } } @@ -1624,6 +1608,26 @@ selist *fighters(battle * b, const side * vs, int minrow, int maxrow, return fightervp; } +struct selector { + int minrow; + int maxrow; +}; + +static bool select_row(const side *vs, const fighter *fig, void *cbdata) +{ + struct selector *sel = (struct selector *)cbdata; + int row = get_unitrow(fig, vs); + return (row >= sel->minrow && row <= sel->maxrow); +} + +selist *fighters(battle * b, const side * vs, int minrow, int maxrow, int mask) +{ + struct selector sel; + sel.maxrow = maxrow; + sel.minrow = minrow; + return select_fighters(b, vs, mask, select_row, &sel); +} + static void report_failed_spell(struct battle * b, struct unit * mage, const struct spell *sp) { message *m = msg_message("battle::spell_failed", "unit spell", mage, sp); @@ -1637,11 +1641,51 @@ static castorder * create_castorder_combat(castorder *co, fighter *fig, const sp return co; } +#ifdef FFL_CURSED +static void summon_igjarjuk(battle *b, spellrank spellranks[]) { + side *s; + castorder *co; + + for (s = b->sides; s != b->sides + b->nsides; ++s) { + fighter *fig = 0; + if (s->bf->attacker && fval(s->faction, FFL_CURSED)) { + spell *sp = find_spell("igjarjuk"); + if (sp) { + int si; + for (si = 0; s->enemies[si]; ++si) { + side *se = s->enemies[si]; + if (se && !fval(se->faction, FFL_NPC)) { + fighter *fi; + for (fi = se->fighters; fi; fi = fi->next) { + if (fi && (!fig || fig->unit->number > fi->unit->number)) { + fig = fi; + if (fig->unit->number == 1) { + break; + } + } + } + if (fig && fig->unit->number == 1) { + break; + } + } + } + if (fig) { + co = create_castorder_combat(0, fig, sp, 10, 10); + co->magician.fig = fig; + add_castorder(&spellranks[sp->rank], co); + break; + } + } + } + } +} +#endif + void do_combatmagic(battle * b, combatmagic_t was) { side *s; - region *r = b->region; castorder *co; + region *r = b->region; int level, rank, sl; spellrank spellranks[MAX_SPELLRANK]; @@ -1649,38 +1693,7 @@ void do_combatmagic(battle * b, combatmagic_t was) #ifdef FFL_CURSED if (was == DO_PRECOMBATSPELL) { - for (s = b->sides; s != b->sides + b->nsides; ++s) { - fighter *fig = 0; - if (s->bf->attacker && fval(s->faction, FFL_CURSED)) { - spell *sp = find_spell("igjarjuk"); - if (sp) { - int si; - for (si = 0; s->enemies[si]; ++si) { - side *se = s->enemies[si]; - if (se && !fval(se->faction, FFL_NPC)) { - fighter *fi; - for (fi = se->fighters; fi; fi = fi->next) { - if (fi && (!fig || fig->unit->number > fi->unit->number)) { - fig = fi; - if (fig->unit->number == 1) { - break; - } - } - } - if (fig && fig->unit->number == 1) { - break; - } - } - } - if (fig) { - co = create_castorder_combat(0, fig, sp, 10, 10); - co->magician.fig = fig; - add_castorder(&spellranks[sp->rank], co); - break; - } - } - } - } + summon_igjarjuk(b, spellranks); } #endif for (s = b->sides; s != b->sides + b->nsides; ++s) { @@ -1889,14 +1902,11 @@ int skilldiff(troop at, troop dt, int dist) if (df->building) { building *b = df->building; if (b->attribs) { - const curse_type *strongwall_ct = ct_find("strongwall"); - if (strongwall_ct) { - curse *c = get_curse(b->attribs, strongwall_ct); - if (curse_active(c)) { - /* wirkt auf alle Geb�ude */ - skdiff -= curse_geteffect_int(c); - is_protected = 2; - } + curse *c = get_curse(b->attribs, &ct_strongwall); + if (curse_active(c)) { + /* wirkt auf alle Geb�ude */ + skdiff -= curse_geteffect_int(c); + is_protected = 2; } } if (b->type->flags & BTF_FORTIFICATION) { @@ -3173,14 +3183,10 @@ fighter *make_fighter(battle * b, unit * u, side * s1, bool attack) /* Effekte von Spr�chen */ if (u->attribs) { - const curse_type *speed_ct; - speed_ct = ct_find("speed"); - if (speed_ct) { - curse *c = get_curse(u->attribs, speed_ct); - if (c) { - speeded = get_cursedmen(u, c); - speed = curse_geteffect_int(c); - } + curse *c = get_curse(u->attribs, &ct_speed); + if (c) { + speeded = get_cursedmen(u, c); + speed = curse_geteffect_int(c); } } @@ -3748,6 +3754,21 @@ static void flee(const troop dt) kill_troop(dt); } +static bool is_calmed(const unit *u, const faction *f) { + attrib *a = a_find(u->attribs, &at_curse); + + while (a && a->type == &at_curse) { + curse *c = (curse *)a->data.v; + if (c->type == &ct_calmmonster && curse_geteffect_int(c) == f->subscription) { + if (curse_active(c)) { + return true; + } + } + a = a->next; + } + return false; +} + static bool start_battle(region * r, battle ** bp) { battle *b = NULL; @@ -3794,12 +3815,12 @@ static bool start_battle(region * r, battle ** bp) if (fval(u, UFL_LONGACTION)) continue; - if (peace_ct && curse_active(get_curse(r->attribs, peace_ct))) { + if (curse_active(get_curse(r->attribs, &ct_peacezone))) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "peace_active", "")); continue; } - if (slave_ct && curse_active(get_curse(u->attribs, slave_ct))) { + if (curse_active(get_curse(u->attribs, &ct_slavery))) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "slave_active", "")); continue; } @@ -3847,26 +3868,11 @@ static bool start_battle(region * r, battle ** bp) NewbieImmunity())); continue; } - /* Fehler: "Die Einheit ist mit uns alliert" */ - if (calm_ct) { - attrib *a = a_find(u->attribs, &at_curse); - bool calm = false; - while (a && a->type == &at_curse) { - curse *c = (curse *)a->data.v; - if (c->type == calm_ct - && curse_geteffect_int(c) == u2->faction->subscription) { - if (curse_active(c)) { - calm = true; - break; - } - } - a = a->next; - } - if (calm) { - cmistake(u, ord, 47, MSG_BATTLE); - continue; - } + /* Fehler: "Die Einheit ist mit uns alliert" */ + if (is_calmed(u, u2->faction)) { + cmistake(u, ord, 47, MSG_BATTLE); + continue; } /* Ende Fehlerbehandlung */ if (b == NULL) { diff --git a/src/battle.h b/src/battle.h index 4e20d00d8..3c86c2c8d 100644 --- a/src/battle.h +++ b/src/battle.h @@ -229,15 +229,17 @@ extern "C" { fighter * get_fighter(battle * b, const struct unit * u); /* END battle interface */ - extern void do_battles(void); + void do_battles(void); /* for combat spells and special attacks */ enum { SELECT_ADVANCE = 0x1, SELECT_DISTANCE = 0x2, SELECT_FIND = 0x4 }; enum { ALLY_SELF, ALLY_ANY }; - extern troop select_enemy(struct fighter *af, int minrow, int maxrow, + int get_unitrow(const fighter * af, const side * vs); + + troop select_enemy(struct fighter *af, int minrow, int maxrow, int select); - extern troop select_ally(struct fighter *af, int minrow, int maxrow, + troop select_ally(struct fighter *af, int minrow, int maxrow, int allytype); int count_enemies(struct battle *b, const struct fighter *af, @@ -246,21 +248,25 @@ extern "C" { int calculate_armor(troop dt, const struct weapon_type *dwtype, const struct weapon_type *awtype, union variant *magres); bool terminate(troop dt, troop at, int type, const char *damage, bool missile); - extern void message_all(battle * b, struct message *m); - extern int hits(troop at, troop dt, weapon * awp); - extern void damage_building(struct battle *b, struct building *bldg, + void message_all(battle * b, struct message *m); + int hits(troop at, troop dt, weapon * awp); + void damage_building(struct battle *b, struct building *bldg, int damage_abs); + + typedef bool(*select_fun)(const struct side *vs, const struct fighter *fig, void *cbdata); + struct selist *select_fighters(struct battle *b, const struct side *vs, int mask, select_fun cb, void *cbdata); struct selist *fighters(struct battle *b, const struct side *vs, int minrow, int maxrow, int mask); + int count_allies(const struct side *as, int minrow, int maxrow, int select, int allytype); - extern bool helping(const struct side *as, const struct side *ds); - extern void rmfighter(fighter * df, int i); - extern struct fighter *select_corpse(struct battle *b, struct fighter *af); - extern int statusrow(int status); - extern void drain_exp(struct unit *u, int d); - extern void kill_troop(troop dt); - extern void remove_troop(troop dt); /* not the same as the badly named rmtroop */ + bool helping(const struct side *as, const struct side *ds); + void rmfighter(fighter * df, int i); + struct fighter *select_corpse(struct battle *b, struct fighter *af); + int statusrow(int status); + void drain_exp(struct unit *u, int d); + void kill_troop(troop dt); + void remove_troop(troop dt); /* not the same as the badly named rmtroop */ bool is_attacker(const fighter * fig); struct battle *make_battle(struct region * r); diff --git a/src/battle.test.c b/src/battle.test.c index 290ef6d74..3669f25d1 100644 --- a/src/battle.test.c +++ b/src/battle.test.c @@ -15,8 +15,13 @@ #include #include +#include +#include #include + +#include + #include "tests.h" static void test_make_fighter(CuTest * tc) @@ -466,11 +471,9 @@ static void test_battle_skilldiff_building(CuTest *tc) unit *ua, *ud; battle *b = NULL; building_type *btype; - const curse_type *strongwall_ct; test_setup(); btype = setup_castle(); - strongwall_ct = ct_find("strongwall"); r = test_create_region(0, 0, 0); ud = test_create_unit(test_create_faction(0), r); @@ -489,13 +492,89 @@ static void test_battle_skilldiff_building(CuTest *tc) create_curse(NULL, &ud->building->attribs, &ct_magicwalls, 1, 1, 1, 1); CuAssertIntEquals(tc, -2, skilldiff(ta, td, 0)); - create_curse(NULL, &ud->building->attribs, strongwall_ct, 1, 1, 2, 1); + create_curse(NULL, &ud->building->attribs, &ct_strongwall, 1, 1, 2, 1); CuAssertIntEquals(tc, -4, skilldiff(ta, td, 0)); free_battle(b); test_cleanup(); } +static void assert_skill(CuTest *tc, char *msg, unit *u, skill_t sk, int level, int week, int weekmax) +{ + skill *sv = unit_skill(u, sk); + char buf[256]; + if (sv) { + sprintf(buf, "%s level %d != %d", msg, sv->level, level); + CuAssertIntEquals_Msg(tc, (const char *)&buf, level, sv->level); + sprintf(buf, "%s week %d !<= %d !<= %d", msg, week, sv->weeks, weekmax); + CuAssert(tc, (const char *)&buf, sv->weeks >= week && sv->weeks <=weekmax); + } else { + CuAssertIntEquals_Msg(tc, msg, level, 0); + CuAssertIntEquals_Msg(tc, msg, week, 0); + } +} + +static void test_drain_exp(CuTest *tc) +{ + unit *u; + char *msg; + int i; + double rand; + + test_setup(); + config_set("study.random_progress", "0"); + u = test_create_unit(test_create_faction(0), test_create_region(0, 0, 0)); + set_level(u, SK_STAMINA, 3); + + CuAssertIntEquals(tc, 3, unit_skill(u, SK_STAMINA)->level); + CuAssertIntEquals(tc, 4, unit_skill(u, SK_STAMINA)->weeks); + + assert_skill(tc, msg = "base", u, SK_STAMINA, 3, 4, 4); + assert_skill(tc, msg, u, SK_STAMINA, 3, 4, 4); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + + for (i=0; i<10; ++i) { + set_level(u, SK_STAMINA, 3); + drain_exp(u, 0); + assert_skill(tc, msg = "0 change", u, SK_STAMINA, 3, 4, 4); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + + for (rand = 0.0; rand < 2.0; rand += 1) { + random_source_inject_constant(rand); + + set_level(u, SK_STAMINA, 3); + drain_exp(u, 29); + + assert_skill(tc, msg = "no change yet", u, SK_STAMINA, 3, 4, rand == 0.0?4:5); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + + set_level(u, SK_STAMINA, 3); + drain_exp(u, 1); + + assert_skill(tc, msg = "random change", u, SK_STAMINA, 3, 4, rand == 0.0?4:5); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + + set_level(u, SK_STAMINA, 3); + drain_exp(u, 30); + + assert_skill(tc, msg = "plus one", u, SK_STAMINA, 3, 5, 5); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + } + + set_level(u, SK_STAMINA, 3); + drain_exp(u, 90); + + assert_skill(tc, msg = "plus three", u, SK_STAMINA, 3, 7, 7); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + + set_level(u, SK_STAMINA, 3); + drain_exp(u, 120); + + assert_skill(tc, msg = "plus four", u, SK_STAMINA, 2, 5, 5); + assert_skill(tc, msg, u, SK_MINING, 0, 0, 0); + } +} + CuSuite *get_battle_suite(void) { CuSuite *suite = CuSuiteNew(); @@ -510,5 +589,6 @@ CuSuite *get_battle_suite(void) SUITE_ADD_TEST(suite, test_natural_armor); SUITE_ADD_TEST(suite, test_magic_resistance); SUITE_ADD_TEST(suite, test_projectile_armor); + SUITE_ADD_TEST(suite, test_drain_exp); return suite; } diff --git a/src/bind_monsters.c b/src/bind_monsters.c index 4ac15aa24..082d940ae 100644 --- a/src/bind_monsters.c +++ b/src/bind_monsters.c @@ -2,6 +2,8 @@ #include "spells/shipcurse.h" #include "monsters.h" +#include + #include #include #include @@ -9,8 +11,6 @@ #include #include -#include - #include #include diff --git a/src/chaos.c b/src/chaos.c index abd74b5d9..67cdb356d 100644 --- a/src/chaos.c +++ b/src/chaos.c @@ -125,9 +125,7 @@ static unit *random_unit(const region * r) unit *u; for (u = r->units; u; u = u->next) { - if (u_race(u) != get_race(RC_SPELL)) { - c += u->number; - } + c += u->number; } if (c == 0) { @@ -138,9 +136,7 @@ static unit *random_unit(const region * r) u = r->units; while (u && c < n) { - if (u_race(u) != get_race(RC_SPELL)) { - c += u->number; - } + c += u->number; u = u->next; } @@ -216,7 +212,7 @@ static void chaos(region * r) for (up = &r->units; *up;) { unit *u = *up; - if (u_race(u) != get_race(RC_SPELL) && u->ship == 0 && !canfly(u)) { + if (u->ship == 0 && !canfly(u)) { ADDMSG(&u->faction->msgs, msg_message("tidalwave_kill", "region unit", r, u)); remove_unit(up, u); diff --git a/src/creport.c b/src/creport.c index 10dc4c323..57ba12ee0 100644 --- a/src/creport.c +++ b/src/creport.c @@ -20,6 +20,8 @@ without prior permission by the authors of Eressea. #define BUFFERSIZE 32768 #define RESOURCECOMPAT +#include + /* modules include */ #include @@ -1361,7 +1363,7 @@ static void cr_output_region(FILE * F, report_context * ctx, region * r) if (skill_enabled(SK_ENTERTAINMENT)) { fprintf(F, "%d;Unterh\n", entertainmoney(r)); } - if (is_cursed(r->attribs, C_RIOT, 0)) { + if (is_cursed(r->attribs, &ct_riotzone)) { fputs("0;Rekruten\n", F); } else { @@ -1418,8 +1420,8 @@ static void cr_output_region(FILE * F, report_context * ctx, region * r) } cr_output_curses_compat(F, f, r, TYP_REGION); cr_borders(r, f, r->seen.mode, F); - if (r->seen.mode == seen_unit && is_astral(r) - && !is_cursed(r->attribs, C_ASTRALBLOCK, 0)) { + if (r->seen.mode >= seen_unit && is_astral(r) + && !is_cursed(r->attribs, &ct_astralblock)) { /* Sonderbehandlung Teleport-Ebene */ region_list *rl = astralregions(r, inhabitable); @@ -1441,7 +1443,7 @@ static void cr_output_region(FILE * F, report_context * ctx, region * r) } cr_output_travelthru(F, r, f); - if (r->seen.mode == seen_unit || r->seen.mode == seen_travel) { + if (r->seen.mode >= seen_travel) { message_list *mlist = r_getmessages(r, f); cr_output_messages(F, r->msgs, f); if (mlist) { diff --git a/src/economy.c b/src/economy.c index 7dd36ae62..4122fa6a1 100644 --- a/src/economy.c +++ b/src/economy.c @@ -36,6 +36,12 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "reports.h" #include "calendar.h" +#include +#include +#include +#include +#include + /* kernel includes */ #include #include @@ -67,9 +73,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -#include -#include - /* libs includes */ #include #include @@ -116,14 +119,14 @@ int entertainmoney(const region * r) { double n; - if (is_cursed(r->attribs, C_DEPRESSION, 0)) { + if (is_cursed(r->attribs, &ct_depression)) { return 0; } n = rmoney(r) / (double)ENTERTAINFRACTION; - if (is_cursed(r->attribs, C_GENEROUS, 0)) { - n *= get_curseeffect(r->attribs, C_GENEROUS, 0); + if (is_cursed(r->attribs, &ct_generous)) { + n *= get_curseeffect(r->attribs, &ct_generous); } return (int)n; @@ -500,7 +503,7 @@ static void recruit(unit * u, struct order *ord, request ** recruitorders) return; } } - if (is_cursed(r->attribs, C_RIOT, 0)) { + if (is_cursed(r->attribs, &ct_riotzone)) { /* Die Region befindet sich in Aufruhr */ cmistake(u, ord, 237, MSG_EVENT); return; @@ -650,7 +653,7 @@ static int forget_cmd(unit * u, order * ord) skill_t sk; const char *s; - if (is_cursed(u->attribs, C_SLAVE, 0)) { + if (is_cursed(u->attribs, &ct_slavery)) { /* charmed units shouldn't be losing their skills */ return 0; } @@ -728,13 +731,12 @@ static int maintain(building * b) void maintain_buildings(region * r) { - const curse_type *nocost_ct = ct_find("nocostbuilding"); building **bp = &r->buildings; while (*bp) { building *b = *bp; int flags = BLD_MAINTAINED; - if (!curse_active(get_curse(b->attribs, nocost_ct))) { + if (!curse_active(get_curse(b->attribs, &ct_nocostbuilding))) { flags = maintain(b); } fset(b, flags); @@ -2588,7 +2590,7 @@ void entertain_cmd(unit * u, struct order *ord) cmistake(u, ord, 69, MSG_INCOME); return; } - if (is_cursed(r->attribs, C_DEPRESSION, 0)) { + if (is_cursed(r->attribs, &ct_depression)) { cmistake(u, ord, 28, MSG_INCOME); return; } @@ -2653,14 +2655,10 @@ expandwork(region * r, request * work_begin, request * work_end, int maxwork) jobs = rpeasants(r); } earnings = jobs * p_wage; - if (r->attribs && rule_blessed_harvest() == HARVEST_TAXES) { + if (jobs > 0 && r->attribs && rule_blessed_harvest() == HARVEST_TAXES) { /* E3 rules */ - const curse_type *blessedharvest_ct = ct_find("blessedharvest"); - if (blessedharvest_ct) { - int happy = - (int)(jobs * curse_geteffect(get_curse(r->attribs, blessedharvest_ct))); - earnings += happy; - } + int happy = harvest_effect(r); + earnings += happy * jobs; } rsetmoney(r, money + earnings); } @@ -2982,13 +2980,12 @@ void produce(struct region *r) static int bt_cache; static const struct building_type *caravan_bt; static int rc_cache; - static const race *rc_spell, *rc_insect, *rc_aquarian; + static const race *rc_insect, *rc_aquarian; if (bt_changed(&bt_cache)) { caravan_bt = bt_find("caravan"); } if (rc_changed(&rc_cache)) { - rc_spell = get_race(RC_SPELL); rc_insect = get_race(RC_INSECT); rc_aquarian = get_race(RC_AQUARIAN); } @@ -3026,11 +3023,11 @@ void produce(struct region *r) bool trader = false; keyword_t todo; - if (u_race(u) == rc_spell || fval(u, UFL_LONGACTION)) + if (fval(u, UFL_LONGACTION)) continue; if (u_race(u) == rc_insect && r_insectstalled(r) && - !is_cursed(u->attribs, C_KAELTESCHUTZ, 0)) + !is_cursed(u->attribs, &ct_insectfur)) continue; if (fval(u, UFL_LONGACTION) && u->thisorder == NULL) { @@ -3103,7 +3100,8 @@ void produce(struct region *r) sabotage_cmd(u, u->thisorder); break; - case K_BREED: + case K_PLANT: + case K_GROW: breed_cmd(u, u->thisorder); break; diff --git a/src/give.c b/src/give.c index 060ecbbb9..b55e6687b 100644 --- a/src/give.c +++ b/src/give.c @@ -17,7 +17,12 @@ #include "economy.h" #include "laws.h" -/* kernel includes */ +#include + + /* attributes includes */ +#include + + /* kernel includes */ #include #include #include @@ -33,9 +38,6 @@ #include #include -/* attributes includes */ -#include - /* util includes */ #include #include @@ -286,7 +288,7 @@ static bool can_give_men(const unit *u, const unit *dst, order *ord, message **m /* hungry people cannot be given away */ if (msg) *msg = msg_error(u, ord, 73); } - else if (fval(u, UFL_LOCKED) || is_cursed(u->attribs, C_SLAVE, 0)) { + else if (fval(u, UFL_LOCKED) || is_cursed(u->attribs, &ct_slavery)) { if (msg) *msg = msg_error(u, ord, 74); } else { @@ -329,7 +331,7 @@ message * give_men(int n, unit * u, unit * u2, struct order *ord) else if (unit_has_cursed_item(u2)) { error = 78; } - else if (fval(u2, UFL_LOCKED) || is_cursed(u2->attribs, C_SLAVE, 0)) { + else if (fval(u2, UFL_LOCKED) || is_cursed(u2->attribs, &ct_slavery)) { error = 75; } else if (!ucontact(u2, u)) { @@ -667,12 +669,7 @@ void give_cmd(unit * u, order * ord) return; } - if (u2 && u_race(u2) == get_race(RC_SPELL)) { - ADDMSG(&u->faction->msgs, msg_feedback(u, ord, - "feedback_unit_not_found", "")); - return; - } - else if (u2 && !alliedunit(u2, u->faction, HELP_GIVE) && !ucontact(u2, u)) { + if (u2 && !alliedunit(u2, u->faction, HELP_GIVE) && !ucontact(u2, u)) { cmistake(u, ord, 40, MSG_COMMERCE); return; } @@ -683,16 +680,11 @@ void give_cmd(unit * u, order * ord) } else if (p == P_HERBS) { bool given = false; - if ((u_race(u)->ec_flags & ECF_KEEP_ITEM) && u2 != NULL) { - ADDMSG(&u->faction->msgs, - msg_feedback(u, ord, "race_nogive", "race", u_race(u))); - return; - } if (!can_give(u, u2, NULL, GIVE_HERBS)) { feedback_give_not_allowed(u, ord); return; } - if (u2 && !(u_race(u2)->ec_flags & GETITEM)) { + if (u2 && !(u_race(u2)->ec_flags & ECF_GETITEM)) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_notake", "race", u_race(u2))); return; @@ -726,7 +718,7 @@ void give_cmd(unit * u, order * ord) } else if (p == P_UNIT) { /* Einheiten uebergeben */ - if (!(u_race(u)->ec_flags & GIVEUNIT)) { + if (!(u_race(u)->ec_flags & ECF_GIVEUNIT)) { cmistake(u, ord, 167, MSG_COMMERCE); return; } @@ -746,12 +738,7 @@ void give_cmd(unit * u, order * ord) if (!s || *s == 0) { /* GIVE ALL items that you have */ /* do these checks once, not for each item we have: */ - if ((u_race(u)->ec_flags & ECF_KEEP_ITEM) && u2 != NULL) { - ADDMSG(&u->faction->msgs, - msg_feedback(u, ord, "race_nogive", "race", u_race(u))); - return; - } - if (u2 && !(u_race(u2)->ec_flags & GETITEM)) { + if (u2 && !(u_race(u2)->ec_flags & ECF_GETITEM)) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_notake", "race", u_race(u2))); return; @@ -778,7 +765,7 @@ void give_cmd(unit * u, order * ord) } else { if (isparam(s, u->faction->locale, P_PERSON)) { - if (!(u_race(u)->ec_flags & GIVEPERSON)) { + if (!(u_race(u)->ec_flags & ECF_GIVEPERSON)) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_noregroup", "race", u_race(u))); } @@ -790,11 +777,7 @@ void give_cmd(unit * u, order * ord) } } } - else if ((u_race(u)->ec_flags & ECF_KEEP_ITEM) && u2 != NULL) { - ADDMSG(&u->faction->msgs, - msg_feedback(u, ord, "race_nogive", "race", u_race(u))); - } - else if (u2 && !(u_race(u2)->ec_flags & GETITEM)) { + else if (u2 && !(u_race(u2)->ec_flags & ECF_GETITEM)) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_notake", "race", u_race(u2))); } @@ -835,7 +818,7 @@ void give_cmd(unit * u, order * ord) if (isparam(s, u->faction->locale, P_PERSON)) { message * msg; - if (!(u_race(u)->ec_flags & GIVEPERSON)) { + if (!(u_race(u)->ec_flags & ECF_GIVEPERSON)) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_noregroup", "race", u_race(u))); return; @@ -849,12 +832,7 @@ void give_cmd(unit * u, order * ord) } if (u2 != NULL) { - if ((u_race(u)->ec_flags & ECF_KEEP_ITEM)) { - ADDMSG(&u->faction->msgs, - msg_feedback(u, ord, "race_nogive", "race", u_race(u))); - return; - } - if (!(u_race(u2)->ec_flags & GETITEM)) { + if (!(u_race(u2)->ec_flags & ECF_GETITEM)) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_notake", "race", u_race(u2))); return; diff --git a/src/give.test.c b/src/give.test.c index 33a9ebfd9..f38028e10 100644 --- a/src/give.test.c +++ b/src/give.test.c @@ -36,7 +36,7 @@ static void setup_give(struct give *env) { assert(env->f1); rc = test_create_race(env->f1->race ? env->f1->race->_name : "humon"); - rc->ec_flags |= GIVEPERSON; + rc->ec_flags |= ECF_GIVEPERSON; env->r = test_create_region(0, 0, ter); env->src = test_create_unit(env->f1, env->r); diff --git a/src/gmtool.c b/src/gmtool.c index a07feab43..db556bb0a 100644 --- a/src/gmtool.c +++ b/src/gmtool.c @@ -903,6 +903,7 @@ static void seed_player(state *st, const newfaction *player) { } } } + static void handlekey(state * st, int c) { window *wnd; diff --git a/src/items.c b/src/items.c index e33aed1bc..dd4abe1ac 100644 --- a/src/items.c +++ b/src/items.c @@ -7,6 +7,12 @@ #include "move.h" #include "magic.h" +#include + +#include +#include +#include + #include #include #include @@ -21,9 +27,6 @@ #include #include -#include -#include - /* triggers includes */ #include #include @@ -167,7 +170,7 @@ struct order *ord) } if (force > 0) { - create_curse(u, &r->attribs, ct_find("antimagiczone"), force, duration, + create_curse(u, &r->attribs, &ct_antimagiczone, force, duration, effect, 0); } } @@ -185,7 +188,7 @@ int amount, struct order *ord) { int money; - if (get_curse(u->region->attribs, ct_find("depression"))) { + if (get_curse(u->region->attribs, &ct_depression)) { cmistake(u, ord, 58, MSG_MAGIC); return -1; } @@ -194,7 +197,7 @@ int amount, struct order *ord) change_money(u, money); rsetmoney(u->region, rmoney(u->region) - money); - create_curse(u, &u->region->attribs, ct_find("depression"), + create_curse(u, &u->region->attribs, &ct_depression, 20, BAGPIPEDURATION, 0.0, 0); ADDMSG(&u->faction->msgs, msg_message("bagpipeoffear_faction", @@ -354,7 +357,7 @@ use_tacticcrystal(unit * u, const struct item_type *itype, int amount, der vor den Antimagiezaubern passiert */ effect = (float)(rng_int() % 6 - 1); - c = create_curse(u, &u->attribs, ct_find("skillmod"), power, + c = create_curse(u, &u->attribs, &ct_skillmod, power, duration, effect, u->number); c->data.i = SK_TACTICS; UNUSED_ARG(ord); diff --git a/src/items/xerewards.c b/src/items/xerewards.c index a6d1ad2ea..e9a11d00d 100644 --- a/src/items/xerewards.c +++ b/src/items/xerewards.c @@ -34,6 +34,8 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. /* util includes */ #include +#include + /* libc includes */ #include #include @@ -52,6 +54,7 @@ struct order *ord) for (n = 0; n != amount; ++n) { skill *sv = u->skills; while (sv != u->skills + u->skill_size) { + /* only one person learns for 3 weeks */ learn_skill(u, (skill_t)sv->id, STUDYDAYS * 3); ++sv; } diff --git a/src/items/xerewards.test.c b/src/items/xerewards.test.c index ea9f53fde..5739141c0 100644 --- a/src/items/xerewards.test.c +++ b/src/items/xerewards.test.c @@ -31,17 +31,17 @@ static void test_skillpotion(CuTest *tc) { itype = test_create_itemtype("skillpotion"); change_resource(u, itype->rtype, 2); - learn_skill(u, SK_ENTERTAINMENT, STUDYDAYS); + learn_skill(u, SK_ENTERTAINMENT, STUDYDAYS * u->number); pSkill = unit_skill(u, SK_ENTERTAINMENT); sk_set(pSkill, 5); initialWeeks_Entertainment = pSkill->weeks = 4; - learn_skill(u, SK_STAMINA, STUDYDAYS); + learn_skill(u, SK_STAMINA, STUDYDAYS * u->number); pSkill = unit_skill(u, SK_STAMINA); sk_set(pSkill, 5); initialWeeks_Stamina = pSkill->weeks = 4; - learn_skill(u, SK_MAGIC, STUDYDAYS); + learn_skill(u, SK_MAGIC, STUDYDAYS * u->number); pSkill = unit_skill(u, SK_MAGIC); sk_set(pSkill, 5); initialWeeks_Magic = pSkill->weeks = 4; diff --git a/src/kernel/build.c b/src/kernel/build.c index 57c172905..a6e8edc77 100644 --- a/src/kernel/build.c +++ b/src/kernel/build.c @@ -717,8 +717,9 @@ build_building(unit * u, const building_type * btype, int id, int want, order * b = u->building; } - if (b) + if (b) { btype = b->type; + } if (fval(btype, BTF_UNIQUE) && buildingtype_exists(r, btype, false)) { /* only one of these per region */ diff --git a/src/kernel/building.c b/src/kernel/building.c index bcc914cca..45faccdab 100644 --- a/src/kernel/building.c +++ b/src/kernel/building.c @@ -21,6 +21,9 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include "building.h" +#include +#include + /* kernel includes */ #include "curse.h" #include "item.h" @@ -54,9 +57,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -/* attributes includes */ -#include - typedef struct building_typelist { struct building_typelist *next; building_type *type; @@ -310,6 +310,7 @@ static const int watch_bonus[3] = { 0, 1, 2 }; int building_protection(const building_type * btype, int stage) { + assert(btype->flags & BTF_FORTIFICATION); if (btype->maxsize < 0) { return castle_bonus[MIN(stage, 5)]; } @@ -679,12 +680,7 @@ default_wage(const region * r, const faction * f, const race * rc, int in_turn) building *b = largestbuilding(r, cmp_wage, false); int esize = 0; double wage; - static int ct_cache; - static const struct curse_type *drought_ct; - if (ct_changed(&ct_cache)) { - drought_ct = ct_find("drought"); - } if (b != NULL) { /* TODO: this reveals imaginary castles */ esize = buildingeffsize(b, false); @@ -709,24 +705,24 @@ default_wage(const region * r, const faction * f, const race * rc, int in_turn) } if (r->attribs && rule_blessed_harvest() == HARVEST_WORK) { /* E1 rules */ - wage += curse_geteffect(get_curse(r->attribs, ct_find("blessedharvest"))); + wage += harvest_effect(r); } } if (r->attribs) { attrib *a; - const struct curse_type *ctype; + curse *c; + /* Godcurse: Income -10 */ - ctype = ct_find("godcursezone"); - if (ctype && curse_active(get_curse(r->attribs, ctype))) { + c = get_curse(r->attribs, &ct_godcursezone); + if (c && curse_active(c)) { wage = MAX(0, wage - 10); } /* Bei einer D�rre verdient man nur noch ein Viertel */ - if (drought_ct) { - curse *c = get_curse(r->attribs, drought_ct); - if (curse_active(c)) - wage /= curse_geteffect(c); + c = get_curse(r->attribs, &ct_drought); + if (c && curse_active(c)) { + wage /= curse_geteffect(c); } a = a_find(r->attribs, &at_reduceproduction); diff --git a/src/kernel/building.test.c b/src/kernel/building.test.c index 588ec2937..e19d074da 100644 --- a/src/kernel/building.test.c +++ b/src/kernel/building.test.c @@ -570,11 +570,32 @@ static void test_buildingtype(CuTest *tc) { test_cleanup(); } +static void test_buildingcapacity(CuTest *tc) { + building *b; + building_type *btype; + test_setup(); + btype = test_create_buildingtype("lighthouse"); + btype->capacity = 1; + btype->maxcapacity = 4; + b = test_create_building(test_create_region(0, 0, NULL), btype); + + b->size = 1; + CuAssertIntEquals(tc, b->size*btype->capacity, buildingcapacity(b)); + b->size = 5; + CuAssertIntEquals(tc, btype->maxcapacity, buildingcapacity(b)); + + btype->capacity = -1; + CuAssertTrue(tc, building_finished(b)); + CuAssertIntEquals(tc, btype->maxcapacity, buildingcapacity(b)); + test_cleanup(); +} + CuSuite *get_building_suite(void) { CuSuite *suite = CuSuiteNew(); SUITE_ADD_TEST(suite, test_buildingtype); SUITE_ADD_TEST(suite, test_largestbuilding); + SUITE_ADD_TEST(suite, test_buildingcapacity); SUITE_ADD_TEST(suite, test_cmp_castle_size); SUITE_ADD_TEST(suite, test_cmp_taxes); SUITE_ADD_TEST(suite, test_cmp_wage); diff --git a/src/kernel/curse.c b/src/kernel/curse.c index 3d236474e..b0367484d 100644 --- a/src/kernel/curse.c +++ b/src/kernel/curse.c @@ -141,10 +141,11 @@ static int read_ccompat(const char *cursename, struct storage *store) struct compat { const char *name; const char *tokens; - } *seek, old_curses[] = { { - "disorientationzone", "" }, { - "shipdisorientation", "" }, { - NULL, NULL } }; + } *seek, old_curses[] = { + { "disorientationzone", "" }, + { "shipdisorientation", "" }, + { NULL, NULL } + }; for (seek = old_curses; seek->name; ++seek) { if (strcmp(seek->tokens, cursename) == 0) { const char *p; @@ -232,7 +233,7 @@ int curse_read(attrib * a, void *owner, gamedata *data) if (c->type->typ == CURSETYP_REGION) { int rr = read_reference(&c->data.v, data, read_region_reference, - RESOLVE_REGION(data->version)); + RESOLVE_REGION(data->version)); if (ur == 0 && rr == 0 && !c->data.v) { return AT_READ_FAIL; } @@ -284,27 +285,15 @@ attrib_type at_curse = { #define MAXCTHASH 128 static selist *cursetypes[MAXCTHASH]; -static int ct_changes = 1; - -bool ct_changed(int *cache) -{ - assert(cache); - if (*cache != ct_changes) { - *cache = ct_changes; - return true; - } - return false; -} void ct_register(const curse_type * ct) { unsigned int hash = tolower(ct->cname[0]) & 0xFF; selist **ctlp = cursetypes + hash; - assert(ct->age==NULL || (ct->flags&CURSE_NOAGE) == 0); + assert(ct->age == NULL || (ct->flags&CURSE_NOAGE) == 0); assert((ct->flags&CURSE_ISNEW) == 0); selist_set_insert(ctlp, (void *)ct, NULL); - ++ct_changes; } void ct_remove(const char *c) @@ -320,7 +309,6 @@ void ct_remove(const char *c) if (strcmp(c, type->cname) == 0) { selist_delete(&ctl, qi); - ++ct_changes; break; } } @@ -479,7 +467,7 @@ int curse_geteffect_int(const curse * c) /* ------------------------------------------------------------- */ static void set_curseingmagician(struct unit *magician, struct attrib *ap_target, -const curse_type * ct) + const curse_type * ct) { curse *c = get_curse(ap_target, ct); if (c) { @@ -490,7 +478,7 @@ const curse_type * ct) /* ------------------------------------------------------------- */ /* gibt bei Personenbeschränkten Verzauberungen die Anzahl der * betroffenen Personen zurück. Ansonsten wird 0 zurückgegeben. */ -int get_cursedmen(unit * u, const curse * c) +int get_cursedmen(const unit * u, const curse * c) { int cursedmen = u->number; @@ -593,28 +581,23 @@ curse *create_curse(unit * magician, attrib ** ap, const curse_type * ct, if (ct->mergeflags & M_DURATION) { c->duration = MAX(c->duration, duration); } - if (ct->mergeflags & M_SUMDURATION) { + else 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); } + else if (ct->mergeflags & M_SUMEFFECT) { + c->effect += effect; + } if (ct->mergeflags & M_VIGOUR) { c->vigour = MAX(vigour, c->vigour); } - if (ct->mergeflags & M_VIGOUR_ADD) { + else if (ct->mergeflags & M_VIGOUR_ADD) { c->vigour = vigour + c->vigour; } - if (ct->mergeflags & M_MEN) { - switch (ct->typ) { - case CURSETYP_UNIT: - { - c->data.i += men; - } - } + if (ct->mergeflags & M_MEN && ct->typ == CURSETYP_UNIT) { + c->data.i += men; } set_curseingmagician(magician, *ap, ct); } @@ -628,7 +611,7 @@ curse *create_curse(unit * magician, attrib ** ap, const curse_type * ct, /* hier müssen alle c-typen, die auf Einheiten gezaubert werden können, * berücksichtigt werden */ -static void do_transfer_curse(curse * c, unit * u, unit * u2, int n) +static void do_transfer_curse(curse * c, const unit * u, unit * u2, int n) { int cursedmen = 0; int men = get_cursedmen(u, c); @@ -675,7 +658,7 @@ static void do_transfer_curse(curse * c, unit * u, unit * u2, int n) } } -void transfer_curse(unit * u, unit * u2, int n) +void transfer_curse(const unit * u, unit * u2, int n) { attrib *a; @@ -736,51 +719,6 @@ bool is_cursed_with(const attrib * ap, const curse * c) return false; } -/* ------------------------------------------------------------- */ -/* cursedata */ -/* ------------------------------------------------------------- */ - -static const char *oldnames[MAXCURSE] = { - /* 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", - "insectfur" -}; - -const char *oldcursename(int id) -{ - return oldnames[id]; -} - /* ------------------------------------------------------------- */ message *cinfo_simple(const void *obj, objtype_t typ, const struct curse * c, int self) @@ -845,5 +783,4 @@ void curses_done(void) { selist_free(cursetypes[i]); cursetypes[i] = 0; } - ++ct_changes; } diff --git a/src/kernel/curse.h b/src/kernel/curse.h index b28427fcd..9b6113d96 100644 --- a/src/kernel/curse.h +++ b/src/kernel/curse.h @@ -267,7 +267,7 @@ extern "C" { /* gibt bei Personenbeschränkten Verzauberungen die Anzahl der * betroffenen Personen zurück. Ansonsten wird 0 zurückgegeben. */ - int get_cursedmen(struct unit *u, const struct curse *c); + int get_cursedmen(const struct unit *u, const struct curse *c); /* setzt/loescht Spezialflag einer Verzauberung (zB 'dauert ewig') */ void c_setflag(curse * c, unsigned int flag); @@ -277,7 +277,7 @@ extern "C" { * korrekt gehandhabt werden. Je nach internen Flag kann dies * unterschiedlich gewünscht sein * */ - void transfer_curse(struct unit *u, struct unit *u2, int n); + void transfer_curse(const struct unit *u, struct unit *u2, int n); /* gibt pointer auf die erste curse-struct zurück, deren Typ ctype ist, * oder einen NULL-pointer @@ -285,7 +285,6 @@ extern "C" { struct curse *get_curse(struct attrib *ap, const curse_type * ctype); const curse_type *ct_find(const char *c); - bool ct_changed(int *cache); void ct_register(const curse_type *); void ct_remove(const char *c); void ct_checknames(void); @@ -304,14 +303,13 @@ extern "C" { bool curse_active(const struct curse *c); /*** COMPATIBILITY MACROS. DO NOT USE FOR NEW CODE, REPLACE IN OLD CODE: */ - const char *oldcursename(int id); struct message *cinfo_simple(const void *obj, objtype_t typ, const struct curse *c, int self); int curse_cansee(const struct curse *c, const struct faction *viewer, objtype_t typ, const void *obj, int self); -#define is_cursed(a, id, id2) \ - (a && curse_active(get_curse(a, ct_find(oldcursename(id))))) -#define get_curseeffect(a, id, id2) \ - curse_geteffect(get_curse(a, ct_find(oldcursename(id)))) +#define is_cursed(a, ctype) \ + (a && curse_active(get_curse(a, ctype))) +#define get_curseeffect(a, ctype) \ + curse_geteffect(get_curse(a, ctype)) /* eressea-defined attribute-type flags */ #define ATF_CURSE ATF_USER_DEFINED diff --git a/src/kernel/curse.test.c b/src/kernel/curse.test.c index a9271f897..10c03a991 100644 --- a/src/kernel/curse.test.c +++ b/src/kernel/curse.test.c @@ -159,24 +159,6 @@ static void test_write_flag(CuTest *tc) { cleanup_curse(&fix); } -static void test_curse_cache(CuTest *tc) -{ - int cache = 0; - const curse_type ct_dummy = { "dummy", CURSETYP_NORM, 0, M_SUMEFFECT, NULL }; - test_setup(); - CuAssertIntEquals(tc, true, ct_changed(&cache)); - CuAssertIntEquals(tc, false, ct_changed(&cache)); - CuAssertPtrEquals(tc, NULL, (void *)ct_find(ct_dummy.cname)); - ct_register(&ct_dummy); - CuAssertIntEquals(tc, true, ct_changed(&cache)); - CuAssertPtrEquals(tc, (void *)&ct_dummy, (void *)ct_find(ct_dummy.cname)); - ct_remove(ct_dummy.cname); - CuAssertIntEquals(tc, true, ct_changed(&cache)); - CuAssertIntEquals(tc, false, ct_changed(&cache)); - CuAssertPtrEquals(tc, NULL, (void *)ct_find(ct_dummy.cname)); - test_cleanup(); -} - static void test_curse_ids(CuTest *tc) { const curse_type ct_dummy = { "dummy", CURSETYP_NORM, 0, M_SUMEFFECT, NULL }; curse *c1, *c2; @@ -218,7 +200,6 @@ CuSuite *get_curse_suite(void) { CuSuite *suite = CuSuiteNew(); SUITE_ADD_TEST(suite, test_curse); - SUITE_ADD_TEST(suite, test_curse_cache); SUITE_ADD_TEST(suite, test_magicstreet); SUITE_ADD_TEST(suite, test_magicstreet_warning); SUITE_ADD_TEST(suite, test_good_dreams); diff --git a/src/kernel/faction.c b/src/kernel/faction.c index 0b099df0f..736c9f4e6 100755 --- a/src/kernel/faction.c +++ b/src/kernel/faction.c @@ -33,6 +33,9 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "terrain.h" #include "unit.h" +#include +#include + /* util includes */ #include #include @@ -49,8 +52,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -#include - #include #include @@ -507,7 +508,6 @@ void renumber_faction(faction * f, int no) funhash(f); f->no = no; fhash(f); - fset(f, FFL_NEWID); } void update_interval(struct faction *f, struct region *r) @@ -730,7 +730,7 @@ int count_faction(const faction * f, int flags) } } else if (flags&COUNT_MIGRANTS) { - if (!is_cursed(u->attribs, C_SLAVE, 0)) { + if (!is_cursed(u->attribs, &ct_slavery)) { n += x; } } diff --git a/src/kernel/faction.h b/src/kernel/faction.h index e0cd893a1..64832053b 100644 --- a/src/kernel/faction.h +++ b/src/kernel/faction.h @@ -36,21 +36,20 @@ extern "C" { struct selist; /* faction flags */ -#define FFL_NEWID (1<<0) /* Die Partei hat bereits einmal ihre no gewechselt */ +#define FFL_NOAID (1<<0) /* Hilfsflag Kampf */ #define FFL_ISNEW (1<<1) #define FFL_PWMSG (1<<2) /* received a "new password" message */ #define FFL_QUIT (1<<3) #define FFL_CURSED (1<<4) /* you're going to have a bad time */ #define FFL_DEFENDER (1<<10) -#define FFL_NOAID (1<<21) /* Hilfsflag Kampf */ -#define FFL_SELECT (1<<22) /* ehemals f->dh, u->dh, r->dh, etc... */ -#define FFL_MARK (1<<23) /* f�r markierende algorithmen, die das - * hinterher auch wieder l�schen m�ssen! - * (FFL_SELECT muss man vorher initialisieren, - * FL_MARK hinterher l�schen) */ -#define FFL_NOIDLEOUT (1<<24) /* Partei stirbt nicht an NMRs */ -#define FFL_NPC (1<<25) /* eine Partei mit Monstern */ -#define FFL_SAVEMASK (FFL_DEFENDER|FFL_NEWID|FFL_NPC|FFL_NOIDLEOUT|FFL_CURSED) +#define FFL_SELECT (1<<22) /* ehemals f->dh, u->dh, r->dh, etc... */ +#define FFL_MARK (1<<23) /* fuer markierende algorithmen, die das + * hinterher auch wieder loeschen muessen! + * (FFL_SELECT muss man vorher initialisieren, + * FL_MARK hinterher loeschen) */ +#define FFL_NOIDLEOUT (1<<24) /* Partei stirbt nicht an NMRs */ +#define FFL_NPC (1<<25) /* eine Partei mit Monstern */ +#define FFL_SAVEMASK (FFL_DEFENDER|FFL_NPC|FFL_NOIDLEOUT|FFL_CURSED) typedef struct faction { struct faction *next; diff --git a/src/kernel/faction.test.c b/src/kernel/faction.test.c index 80faba3c1..7901cd96f 100644 --- a/src/kernel/faction.test.c +++ b/src/kernel/faction.test.c @@ -8,6 +8,7 @@ #include #include #include +#include #include #include @@ -223,6 +224,21 @@ static void test_valid_race(CuTest *tc) { test_cleanup(); } +static void test_set_email(CuTest *tc) { + char * email = NULL; + test_setup(); + CuAssertIntEquals(tc, 0, set_email(&email, "enno@eressea.de")); + CuAssertStrEquals(tc, "enno@eressea.de", email); + CuAssertIntEquals(tc, 0, set_email(&email, "bugs@eressea.de")); + CuAssertStrEquals(tc, "bugs@eressea.de", email); + CuAssertIntEquals(tc, -1, set_email(&email, "bad@@eressea.de")); + CuAssertIntEquals(tc, -1, set_email(&email, "eressea.de")); + CuAssertIntEquals(tc, -1, set_email(&email, "eressea@")); + CuAssertStrEquals(tc, "bugs@eressea.de", email); + free(email); + test_cleanup(); +} + CuSuite *get_faction_suite(void) { CuSuite *suite = CuSuiteNew(); @@ -237,5 +253,6 @@ CuSuite *get_faction_suite(void) SUITE_ADD_TEST(suite, test_set_origin_bug); SUITE_ADD_TEST(suite, test_check_passwd); SUITE_ADD_TEST(suite, test_valid_race); + SUITE_ADD_TEST(suite, test_set_email); return suite; } diff --git a/src/kernel/jsonconf.test.c b/src/kernel/jsonconf.test.c index 2f54eeb8d..cb79cafc4 100644 --- a/src/kernel/jsonconf.test.c +++ b/src/kernel/jsonconf.test.c @@ -58,10 +58,9 @@ static void test_flags(CuTest *tc) { check_flag(tc, "undead", RCF_UNDEAD); check_flag(tc, "dragon", RCF_DRAGON); check_flag(tc, "fly", RCF_FLY); - check_ec_flag(tc, "getitem", GETITEM); - check_ec_flag(tc, "keepitem", ECF_KEEP_ITEM); - check_ec_flag(tc, "giveperson", GIVEPERSON); - check_ec_flag(tc, "giveunit", GIVEUNIT); + check_ec_flag(tc, "getitem", ECF_GETITEM); + check_ec_flag(tc, "giveperson", ECF_GIVEPERSON); + check_ec_flag(tc, "giveunit", ECF_GIVEUNIT); test_cleanup(); } diff --git a/src/kernel/messages.c b/src/kernel/messages.c index d3cd4f99b..b8d33f08c 100644 --- a/src/kernel/messages.c +++ b/src/kernel/messages.c @@ -258,15 +258,14 @@ message * msg_error(const unit * u, struct order *ord, int mno) { return msg_feedback(u, ord, msgname, ""); } -message * cmistake(const unit * u, struct order *ord, int mno, int mtype) +void cmistake(const unit * u, struct order *ord, int mno, int mtype) { - message * result; + message * msg; UNUSED_ARG(mtype); - result = msg_error(u, ord, mno); - if (result) { - ADDMSG(&u->faction->msgs, result); + msg = msg_error(u, ord, mno); + if (msg) { + ADDMSG(&u->faction->msgs, msg); } - return result; } void syntax_error(const struct unit *u, struct order *ord) diff --git a/src/kernel/messages.h b/src/kernel/messages.h index a42342153..4eca65fc1 100644 --- a/src/kernel/messages.h +++ b/src/kernel/messages.h @@ -60,7 +60,7 @@ extern "C" { #define ADDMSG(msgs, mcreate) { message * mx = mcreate; if (mx) { assert(mx->refcount>=1); add_message(msgs, mx); msg_release(mx); } } void syntax_error(const struct unit *u, struct order *ord); - struct message * cmistake(const struct unit *u, struct order *ord, int mno, int mtype); + void cmistake(const struct unit *u, struct order *ord, int mno, int mtype); struct message * msg_error(const struct unit * u, struct order *ord, int mno); #ifdef __cplusplus } diff --git a/src/kernel/order.c b/src/kernel/order.c index 6fbb53040..d64da0131 100644 --- a/src/kernel/order.c +++ b/src/kernel/order.c @@ -446,7 +446,8 @@ bool is_repeated(keyword_t kwd) case K_SABOTAGE: case K_STUDY: case K_TEACH: - case K_BREED: + case K_GROW: + case K_PLANT: case K_PIRACY: case K_MAKE: case K_LOOT: @@ -485,7 +486,8 @@ bool is_exclusive(const order * ord) case K_SABOTAGE: case K_STUDY: case K_TEACH: - case K_BREED: + case K_GROW: + case K_PLANT: case K_PIRACY: case K_MAKE: case K_LOOT: @@ -525,7 +527,8 @@ bool is_long(keyword_t kwd) case K_SABOTAGE: case K_STUDY: case K_TEACH: - case K_BREED: + case K_GROW: + case K_PLANT: case K_PIRACY: case K_MAKE: case K_LOOT: diff --git a/src/kernel/pool.c b/src/kernel/pool.c index 8dfd589c3..c252be566 100644 --- a/src/kernel/pool.c +++ b/src/kernel/pool.c @@ -160,7 +160,7 @@ int count) region *r = u->region; int have = get_resource(u, rtype); - if ((u_race(u)->ec_flags & GETITEM) == 0) { + if ((u_race(u)->ec_flags & ECF_GETITEM) == 0) { mode &= (GET_SLACK | GET_RESERVE); } @@ -179,9 +179,6 @@ int count) if (u != v) { int mask; - if ((u_race(v)->ec_flags & ECF_KEEP_ITEM)) - continue; - if (v->faction == f) { mask = (mode >> 3) & (GET_SLACK | GET_RESERVE); } @@ -204,7 +201,7 @@ use_pooled(unit * u, const resource_type * rtype, unsigned int mode, int count) region *r = u->region; int n = 0, have = get_resource(u, rtype); - if ((u_race(u)->ec_flags & GETITEM) == 0) { + if ((u_race(u)->ec_flags & ECF_GETITEM) == 0) { mode &= (GET_SLACK | GET_RESERVE); } @@ -232,9 +229,6 @@ use_pooled(unit * u, const resource_type * rtype, unsigned int mode, int count) for (v = r->units; use > 0 && v != NULL; v = v->next) { if (u != v) { int mask; - if ((u_race(v)->ec_flags & ECF_KEEP_ITEM)) - continue; - if (v->faction == f) { mask = (mode >> 3) & (GET_SLACK | GET_RESERVE); } diff --git a/src/kernel/race.c b/src/kernel/race.c index a2c37d87d..14fb66331 100644 --- a/src/kernel/race.c +++ b/src/kernel/race.c @@ -69,7 +69,7 @@ const char *racenames[MAXRACES] = { "dwarf", "elf", NULL, "goblin", "human", "troll", "demon", "insect", "halfling", "cat", "aquarian", "orc", "snotling", "undead", NULL, "youngdragon", "dragon", "wyrm", "ent", "catdragon", "dracoid", - NULL, "spell", "irongolem", "stonegolem", "shadowdemon", + NULL, NULL, "irongolem", "stonegolem", "shadowdemon", "shadowmaster", "mountainguard", "alp", "toad", "braineater", "peasant", "wolf", NULL, NULL, NULL, NULL, "songdragon", NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, diff --git a/src/kernel/race.h b/src/kernel/race.h index 5c320ea05..2dec5f06b 100644 --- a/src/kernel/race.h +++ b/src/kernel/race.h @@ -72,8 +72,7 @@ extern "C" { RC_BIRTHDAYDRAGON, RC_DRACOID, - RC_SPELL = 22, - RC_IRONGOLEM, + RC_IRONGOLEM = 23, RC_STONEGOLEM, RC_SHADOW, RC_SHADOWLORD, @@ -234,10 +233,9 @@ extern "C" { #define RCF_FAMILIAR (1<<31) /* may be a familiar */ /* Economic flags */ -#define ECF_KEEP_ITEM (1<<1) /* gibt Gegenst�nde weg */ -#define GIVEPERSON (1<<2) /* �bergibt Personen */ -#define GIVEUNIT (1<<3) /* Einheiten an andere Partei �bergeben */ -#define GETITEM (1<<4) /* nimmt Gegenst�nde an */ +#define ECF_GIVEPERSON (1<<2) /* �bergibt Personen */ +#define ECF_GIVEUNIT (1<<3) /* Einheiten an andere Partei �bergeben */ +#define ECF_GETITEM (1<<4) /* nimmt Gegenst�nde an */ #define ECF_REC_ETHEREAL (1<<7) /* Rekrutiert aus dem Nichts */ #define ECF_REC_UNLIMITED (1<<8) /* Rekrutiert ohne Limit */ diff --git a/src/kernel/region.c b/src/kernel/region.c index 5cb86ff47..7e6104406 100644 --- a/src/kernel/region.c +++ b/src/kernel/region.c @@ -37,6 +37,8 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "terrainid.h" #include "unit.h" +#include + /* util includes */ #include #include @@ -157,7 +159,7 @@ void deathcounts(region * r, int fallen) if (fallen == 0) return; if (r->attribs) { - const curse_type *ctype = ct_find("holyground"); + const curse_type *ctype = &ct_holyground; if (ctype && curse_active(get_curse(r->attribs, ctype))) return; a = a_find(r->attribs, &at_deathcount); @@ -584,7 +586,7 @@ int rroad(const region * r, direction_t d) bool r_isforest(const region * r) { if (fval(r->terrain, FOREST_REGION)) { - /* needs to be covered with at leas 48% trees */ + /* needs to be covered with at least 48% trees */ int mincover = (int)(r->terrain->size * 0.48); int trees = rtrees(r, 2) + rtrees(r, 1); return (trees * TREESIZE >= mincover); @@ -917,6 +919,7 @@ void free_region(region * r) while (r->units) { unit *u = r->units; r->units = u->next; + u->region = NULL; uunhash(u); free_unit(u); free(u); @@ -1241,8 +1244,9 @@ int production(const region * r) { /* muß rterrain(r) sein, nicht rterrain() wegen rekursion */ int p = r->terrain->size; - if (curse_active(get_curse(r->attribs, ct_find("drought")))) + if (curse_active(get_curse(r->attribs, &ct_drought))) { p /= 2; + } return p; } diff --git a/src/kernel/region.h b/src/kernel/region.h index e3204d129..5117567a6 100644 --- a/src/kernel/region.h +++ b/src/kernel/region.h @@ -35,7 +35,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #define RF_MALLORN (1<<1) /* persistent */ #define RF_BLOCKED (1<<2) /* persistent */ -#define RF_UNUSED_3 (1<<3) +#define RF_OBSERVER (1<<3) /* persistent */ #define RF_UNUSED_4 (1<<4) #define RF_UNUSED_5 (1<<5) #define RF_UNUSED_6 (1<<6) diff --git a/src/kernel/save.c b/src/kernel/save.c index e53d6ea4d..a3fc7e5af 100644 --- a/src/kernel/save.c +++ b/src/kernel/save.c @@ -48,6 +48,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "lighthouse.h" /* attributes includes */ +#include #include #include @@ -131,8 +132,6 @@ static unit *unitorders(FILE * F, int enc, struct faction *f) i = getid(); u = findunitg(i, NULL); - if (u && u_race(u) == get_race(RC_SPELL)) - return NULL; if (u && u->faction == f) { order **ordp; @@ -848,7 +847,7 @@ void write_unit(struct gamedata *data, const unit * u) WRITE_SECTION(data->store); write_items(data->store, u->items); WRITE_SECTION(data->store); - if (u->hp == 0 && u_race(u)!= get_race(RC_SPELL)) { + if (u->hp == 0 && data->version < NORCSPELL_VERSION) { log_error("unit %s has 0 hitpoints, adjusting.", itoa36(u->no)); ((unit *)u)->hp = u->number; } @@ -1618,6 +1617,7 @@ int read_game(gamedata *data) { int rmax = maxregions; storage * store = data->store; const struct building_type *bt_lighthouse = bt_find("lighthouse"); + const struct race *rc_spell = rc_find("spell"); if (data->version >= SAVEGAMEID_VERSION) { int gameid; @@ -1702,19 +1702,25 @@ int read_game(gamedata *data) { while (--p >= 0) { unit *u = read_unit(data); - if (data->version < JSON_REPORT_VERSION) { - if (u->_name && fval(u->faction, FFL_NPC)) { - if (!u->_name[0] || unit_name_equals_race(u)) { - unit_setname(u, NULL); + if (data->version < NORCSPELL_VERSION && u_race(u) == rc_spell) { + set_observer(r, u->faction, get_level(u, SK_PERCEPTION), u->age); + u_setfaction(u, NULL); + free_unit(u); + } + else { + if (data->version < JSON_REPORT_VERSION) { + if (u->_name && fval(u->faction, FFL_NPC)) { + if (!u->_name[0] || unit_name_equals_race(u)) { + unit_setname(u, NULL); + } } } + assert(u->region == NULL); + u->region = r; + *up = u; + up = &u->next; + update_interval(u->faction, r); } - assert(u->region == NULL); - u->region = r; - *up = u; - up = &u->next; - - update_interval(u->faction, r); } --rmax; } diff --git a/src/kernel/ship.c b/src/kernel/ship.c index 234a91631..764500b50 100644 --- a/src/kernel/ship.c +++ b/src/kernel/ship.c @@ -20,6 +20,9 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include "ship.h" +#include +#include + /* kernel includes */ #include "build.h" #include "curse.h" @@ -41,9 +44,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -#include -#include - #include #include #include @@ -332,10 +332,10 @@ int shipspeed(const ship * sh, const unit * u) return 0; if (sh->attribs) { - if (curse_active(get_curse(sh->attribs, ct_find("stormwind")))) { + if (curse_active(get_curse(sh->attribs, &ct_stormwind))) { k *= 2; } - if (curse_active(get_curse(sh->attribs, ct_find("nodrift")))) { + if (curse_active(get_curse(sh->attribs, &ct_nodrift))) { k += 1; } } diff --git a/src/kernel/ship.test.c b/src/kernel/ship.test.c index 541ddee9a..ce55f55af 100644 --- a/src/kernel/ship.test.c +++ b/src/kernel/ship.test.c @@ -432,8 +432,11 @@ static void test_shipspeed_stormwind(CuTest *tc) { register_shipcurse(); assert(sh && cap && crew); - create_curse(0, &sh->attribs, ct_find("stormwind"), 1, 1, 1, 0); + create_curse(0, &sh->attribs, &ct_stormwind, 1, 1, 1, 0); + CuAssertPtrNotNull(tc, sh->attribs); CuAssertIntEquals_Msg(tc, "stormwind doubles ship range", sh->type->range * 2, shipspeed(sh, cap)); + a_age(&sh->attribs, sh); + CuAssertPtrEquals(tc, NULL, sh->attribs); test_cleanup(); } @@ -447,7 +450,7 @@ static void test_shipspeed_nodrift(CuTest *tc) { register_shipcurse(); assert(sh && cap && crew); - create_curse(0, &sh->attribs, ct_find("nodrift"), 1, 1, 1, 0); + create_curse(0, &sh->attribs, &ct_nodrift, 1, 1, 1, 0); CuAssertIntEquals_Msg(tc, "nodrift adds +1 to range", sh->type->range + 1, shipspeed(sh, cap)); test_cleanup(); } diff --git a/src/kernel/skills.c b/src/kernel/skills.c index b951b77ba..3bcbd0f66 100644 --- a/src/kernel/skills.c +++ b/src/kernel/skills.c @@ -153,6 +153,7 @@ int rc_skillmod(const struct race *rc, const region * r, skill_t sk) int level_days(int level) { + /* FIXME STUDYDAYS * ((level + 1) * level / 2); */ return 30 * ((level + 1) * level / 2); } @@ -202,6 +203,19 @@ int skill_weeks(int level) return level + 1; } +void increase_skill(unit * u, skill_t sk, unsigned int weeks) +{ + skill *sv = unit_skill(u, sk); + if (!sv) { + sv = add_skill(u, sk); + } + while (sv->weeks <= (int) weeks) { + weeks -= sv->weeks; + sk_set(sv, sv->level + 1); + } + sv->weeks -= weeks; +} + void reduce_skill(unit * u, skill * sv, unsigned int weeks) { sv->weeks += weeks; diff --git a/src/kernel/skills.h b/src/kernel/skills.h index 1261a7ea9..4c41109d7 100644 --- a/src/kernel/skills.h +++ b/src/kernel/skills.h @@ -52,7 +52,8 @@ extern "C" { int level(int days); #define skill_level(level) (level) - void reduce_skill(struct unit *u, skill * sv, unsigned int change); + void increase_skill(struct unit * u, skill_t sk, unsigned int weeks); + void reduce_skill(struct unit *u, skill * sv, unsigned int weeks); int skill_weeks(int level); int skill_compare(const skill * sk, const skill * sc); diff --git a/src/kernel/terrain.c b/src/kernel/terrain.c index 3d6f4c313..e9f448f97 100644 --- a/src/kernel/terrain.c +++ b/src/kernel/terrain.c @@ -18,14 +18,16 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -#include #include "terrain.h" -#include "terrainid.h" + +#include +#include /* kernel includes */ #include "curse.h" #include "region.h" #include "resources.h" +#include "terrainid.h" #include #include @@ -160,7 +162,7 @@ const char *terrain_name(const struct region *r) return r->terrain->name(r); } else if (fval(r->terrain, SEA_REGION)) { - if (curse_active(get_curse(r->attribs, ct_find("maelstrom")))) { + if (curse_active(get_curse(r->attribs, &ct_maelstrom))) { return "maelstrom"; } } diff --git a/src/kernel/types.h b/src/kernel/types.h index b9faefce7..a20e923d6 100644 --- a/src/kernel/types.h +++ b/src/kernel/types.h @@ -71,8 +71,8 @@ typedef enum { seen_neighbour, seen_lighthouse, seen_travel, - seen_far, seen_unit, + seen_spell, seen_battle } seen_mode; diff --git a/src/kernel/unit.c b/src/kernel/unit.c index ab5f72f35..9c08f9dc1 100644 --- a/src/kernel/unit.c +++ b/src/kernel/unit.c @@ -43,6 +43,9 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include +#include +#include + #include "guard.h" /* util includes */ @@ -213,7 +216,7 @@ static buddy *get_friends(const unit * u, int *numfriends) buddy *nf, **fr = &friends; /* some units won't take stuff: */ - if (u_race(u2)->ec_flags & GETITEM) { + if (u_race(u2)->ec_flags & ECF_GETITEM) { while (*fr && (*fr)->faction->no < u2->faction->no) fr = &(*fr)->next; nf = *fr; @@ -225,13 +228,6 @@ static buddy *get_friends(const unit * u, int *numfriends) nf->number = 0; *fr = nf; } - else if (nf->faction == u2->faction - && !(u_race(u2)->ec_flags & ECF_KEEP_ITEM)) { - /* we don't like to gift it to units that won't give it back */ - if ((u_race(nf->unit)->ec_flags & ECF_KEEP_ITEM)) { - nf->unit = u2; - } - } nf->number += u2->number; number += u2->number; } @@ -269,8 +265,6 @@ int gift_items(unit * u, int flags) if (u->items == NULL || fval(u_race(u), RCF_ILLUSIONARY)) return 0; - if ((u_race(u)->ec_flags & ECF_KEEP_ITEM)) - return 0; /* at first, I should try giving my crap to my own units in this region */ if (u->faction && (u->faction->flags & FFL_QUIT) == 0 && (flags & GIFT_SELF)) { @@ -278,16 +272,10 @@ int gift_items(unit * u, int flags) for (u2 = r->units; u2; u2 = u2->next) { if (u2 != u && u2->faction == u->faction && u2->number > 0) { /* some units won't take stuff: */ - if (u_race(u2)->ec_flags & GETITEM) { - /* we don't like to gift it to units that won't give it back */ - if (!(u_race(u2)->ec_flags & ECF_KEEP_ITEM)) { - i_merge(&u2->items, &u->items); - u->items = NULL; - break; - } - else { - u3 = u2; - } + if (u_race(u2)->ec_flags & ECF_GETITEM) { + i_merge(&u2->items, &u->items); + u->items = NULL; + break; } } } @@ -938,8 +926,7 @@ bool can_survive(const unit * u, const region * r) return false; if (r->attribs) { - const curse_type *ctype = ct_find("holyground"); - if (fval(u_race(u), RCF_UNDEAD) && curse_active(get_curse(r->attribs, ctype))) + if (fval(u_race(u), RCF_UNDEAD) && curse_active(get_curse(r->attribs, &ct_holyground))) return false; } return true; @@ -981,10 +968,9 @@ void move_unit(unit * u, region * r, unit ** ulist) /* ist mist, aber wegen nicht skalierender attribute notwendig: */ #include "alchemy.h" -void transfermen(unit * u, unit * dst, int n) +void clone_men(const unit * u, unit * dst, int n) { const attrib *a; - int hp = u->hp; region *r = u->region; if (n == 0) @@ -1073,12 +1059,9 @@ void transfermen(unit * u, unit * dst, int n) if (u->attribs) { transfer_curse(u, dst, n); } - } - scale_number(u, u->number - n); - if (dst) { set_number(dst, dst->number + n); - hp -= u->hp; - dst->hp += hp; + dst->hp += u->hp * dst->number / u->number; + assert(dst->hp >= dst->number); /* TODO: Das ist schnarchlahm! und gehoert nicht hierhin */ a = a_find(dst->attribs, &at_effect); while (a && a->type == &at_effect) { @@ -1100,6 +1083,12 @@ void transfermen(unit * u, unit * dst, int n) } } +void transfermen(unit * u, unit * dst, int n) +{ + clone_men(u, dst, n); + scale_number(u, u->number - n); +} + struct building *inside_building(const struct unit *u) { if (!u->building) { @@ -1268,52 +1257,45 @@ static int att_modification(const unit * u, skill_t sk) if (u->attribs) { curse *c; - static int cache; - static const curse_type *skillmod_ct, *worse_ct; - if (ct_changed(&cache)) { - skillmod_ct = ct_find("skillmod"); - worse_ct = ct_find("worse"); - } - c = get_curse(u->attribs, worse_ct); - if (c != NULL) + attrib *a; + + c = get_curse(u->attribs, &ct_worse); + if (c != NULL) { result += curse_geteffect(c); - if (skillmod_ct) { - attrib *a = a_find(u->attribs, &at_curse); - while (a && a->type == &at_curse) { - curse *c = (curse *)a->data.v; - if (c->type == skillmod_ct && c->data.i == sk) { - result += curse_geteffect(c); - break; - } - a = a->next; + } + + a = a_find(u->attribs, &at_curse); + while (a && a->type == &at_curse) { + c = (curse *)a->data.v; + if (c->type == &ct_skillmod && c->data.i == sk) { + result += curse_geteffect(c); + break; } + a = a->next; } } /* TODO hier kann nicht mit get/iscursed gearbeitet werden, da nur der * jeweils erste vom Typ C_GBDREAM zurueckgegen wird, wir aber alle * durchsuchen und aufaddieren muessen */ if (u->region && u->region->attribs) { - const curse_type *gbdream_ct = ct_find("gbdream"); - if (gbdream_ct) { - int bonus = 0, malus = 0; - attrib *a = a_find(u->region->attribs, &at_curse); - while (a && a->type == &at_curse) { - curse *c = (curse *)a->data.v; - - if (c->magician && curse_active(c) && c->type == gbdream_ct) { - int effect = curse_geteffect_int(c); - bool allied = alliedunit(c->magician, u->faction, HELP_GUARD); - if (allied) { - if (effect > bonus) bonus = effect; - } - else { - if (effect < malus) malus = effect; - } + int bonus = 0, malus = 0; + attrib *a = a_find(u->region->attribs, &at_curse); + while (a && a->type == &at_curse) { + curse *c = (curse *)a->data.v; + + if (c->magician && curse_active(c) && c->type == &ct_gbdream) { + int effect = curse_geteffect_int(c); + bool allied = alliedunit(c->magician, u->faction, HELP_GUARD); + if (allied) { + if (effect > bonus) bonus = effect; + } + else { + if (effect < malus) malus = effect; } - a = a->next; } - result = result + bonus + malus; + a = a->next; } + result = result + bonus + malus; } return (int)result; @@ -1399,6 +1381,7 @@ int invisible(const unit * target, const unit * viewer) */ void free_unit(unit * u) { + assert(!u->region); free(u->_name); free(u->display); free_order(u->thisorder); @@ -1524,7 +1507,9 @@ unit *create_unit(region * r, faction * f, int number, const struct race *urace, /* u->race muss bereits gesetzt sein, wird fuer default-hp gebraucht */ /* u->region auch */ - u->hp = unit_max_hp(u) * number; + if (number > 0) { + u->hp = unit_max_hp(u) * number; + } if (dname) { u->_name = strdup(dname); @@ -1738,16 +1723,9 @@ int unit_max_hp(const unit * u) /* der healing curse veraendert die maximalen hp */ if (u->region && u->region->attribs) { - static int cache; - static const curse_type *heal_ct; - if (ct_changed(&cache)) { - heal_ct = ct_find("healingzone"); - } - if (heal_ct) { - curse *c = get_curse(u->region->attribs, heal_ct); - if (c) { - h = (int)(h * (1.0 + (curse_geteffect(c) / 100))); - } + curse *c = get_curse(u->region->attribs, &ct_healing); + if (c) { + h = (int)(h * (1.0 + (curse_geteffect(c) / 100))); } } return h; @@ -1891,8 +1869,7 @@ void remove_empty_units_in_region(region * r) set_number(u, 0); } } - if ((u->number == 0 && u_race(u) != get_race(RC_SPELL)) || (u->age <= 0 - && u_race(u) == get_race(RC_SPELL))) { + if (u->number == 0) { remove_unit(up, u); } if (*up == u) diff --git a/src/kernel/unit.h b/src/kernel/unit.h index 8ae13c5e2..dede956b5 100644 --- a/src/kernel/unit.h +++ b/src/kernel/unit.h @@ -165,7 +165,8 @@ extern "C" { void set_level(struct unit *u, skill_t id, int level); int get_level(const struct unit *u, skill_t id); - extern void transfermen(struct unit *src, struct unit *dst, int n); + void transfermen(struct unit *src, struct unit *dst, int n); + void clone_men(const struct unit *src, struct unit *dst, int n); /* like transfer, but do not subtract from src */ int eff_skill(const struct unit *u, const struct skill *sv, const struct region *r); int effskill_study(const struct unit *u, skill_t sk, const struct region *r); diff --git a/src/kernel/unit.test.c b/src/kernel/unit.test.c index 096da9d62..6ab8ee21a 100644 --- a/src/kernel/unit.test.c +++ b/src/kernel/unit.test.c @@ -97,26 +97,6 @@ static void test_remove_units_with_dead_faction(CuTest *tc) { test_cleanup(); } -static void test_remove_units_ignores_spells(CuTest *tc) { - unit *u; - int uid; - - test_cleanup(); - test_create_world(); - - u = create_unit(findregion(0, 0), test_create_faction(test_create_race("human")), 1, test_create_race("spell"), 0, 0, 0); - uid = u->no; - u->number = 0; - u->age = 1; - remove_empty_units_in_region(u->region); - CuAssertPtrNotNull(tc, findunit(uid)); - CuAssertPtrNotNull(tc, u->region); - u->age = 0; - remove_empty_units_in_region(u->region); - CuAssertPtrEquals(tc, 0, findunit(uid)); - test_cleanup(); -} - static void test_scale_number(CuTest *tc) { unit *u; const struct potion_type *ptype; @@ -545,6 +525,29 @@ static void test_unlimited_units(CuTest *tc) { test_cleanup(); } +static void test_clone_men(CuTest *tc) { + unit *u1, *u2; + region *r; + faction *f; + test_setup(); + r = test_create_region(0, 0, NULL); + f = test_create_faction(NULL); + u1 = test_create_unit(f, r); + scale_number(u1, 10); + u2 = test_create_unit(f, r); + scale_number(u2, 0); + CuAssertIntEquals(tc, 10, u1->number); + CuAssertIntEquals(tc, 200, u1->hp); + CuAssertIntEquals(tc, 0, u2->number); + CuAssertIntEquals(tc, 0, u2->hp); + clone_men(u1, u2, 1); + CuAssertIntEquals(tc, 10, u1->number); + CuAssertIntEquals(tc, 200, u1->hp); + CuAssertIntEquals(tc, 1, u2->number); + CuAssertIntEquals(tc, 20, u2->hp); + test_cleanup(); +} + CuSuite *get_unit_suite(void) { CuSuite *suite = CuSuiteNew(); @@ -553,9 +556,9 @@ CuSuite *get_unit_suite(void) SUITE_ADD_TEST(suite, test_unit_name); SUITE_ADD_TEST(suite, test_unit_name_from_race); SUITE_ADD_TEST(suite, test_update_monster_name); + SUITE_ADD_TEST(suite, test_clone_men); SUITE_ADD_TEST(suite, test_remove_unit); SUITE_ADD_TEST(suite, test_remove_empty_units); - SUITE_ADD_TEST(suite, test_remove_units_ignores_spells); SUITE_ADD_TEST(suite, test_remove_units_without_faction); SUITE_ADD_TEST(suite, test_remove_units_with_dead_faction); SUITE_ADD_TEST(suite, test_remove_empty_units_in_region); diff --git a/src/kernel/version.c b/src/kernel/version.c index 39f4f30e6..0842b3d8e 100644 --- a/src/kernel/version.c +++ b/src/kernel/version.c @@ -5,7 +5,7 @@ #ifndef ERESSEA_VERSION /* the version number, if it was not passed to make with -D */ -#define ERESSEA_VERSION "3.12.0" +#define ERESSEA_VERSION "3.13.0" #endif const char *eressea_version(void) { diff --git a/src/kernel/xmlreader.c b/src/kernel/xmlreader.c index 9953f7335..2ef54b005 100644 --- a/src/kernel/xmlreader.c +++ b/src/kernel/xmlreader.c @@ -1320,6 +1320,12 @@ static void parse_ai(race * rc, xmlNodePtr node) rc->flags |= RCF_ATTACK_MOVED; } +static void set_study_speed(race *rc, skill_t sk, int modifier){ + if (!rc->study_speed) + rc->study_speed = calloc(1, MAXSKILLS); + rc->study_speed[sk] = (char)modifier; +} + static int parse_races(xmlDocPtr doc) { xmlXPathContextPtr xpath = xmlXPathNewContext(doc); @@ -1338,6 +1344,7 @@ static int parse_races(xmlDocPtr doc) xmlXPathObjectPtr result; int k, study_speed_base, attacks; struct att *attack; + skill_t sk; propValue = xmlGetProp(node, BAD_CAST "name"); assert(propValue != NULL); @@ -1362,6 +1369,11 @@ static int parse_races(xmlDocPtr doc) rc->hitpoints = xml_ivalue(node, "hp", rc->hitpoints); rc->armor = (char)xml_ivalue(node, "ac", rc->armor); study_speed_base = xml_ivalue(node, "studyspeed", 0); + if (study_speed_base != 0) { + for (sk = 0; sk < MAXSKILLS; ++sk) { + set_study_speed(rc, sk, study_speed_base); + } + } rc->at_default = (char)xml_ivalue(node, "unarmedattack", -2); rc->df_default = (char)xml_ivalue(node, "unarmeddefense", -2); @@ -1423,14 +1435,12 @@ static int parse_races(xmlDocPtr doc) if (xml_bvalue(node, "irongolem", false)) rc->flags |= RCF_IRONGOLEM; - if (xml_bvalue(node, "keepitem", false)) - rc->ec_flags |= ECF_KEEP_ITEM; if (xml_bvalue(node, "giveperson", false)) - rc->ec_flags |= GIVEPERSON; + rc->ec_flags |= ECF_GIVEPERSON; if (xml_bvalue(node, "giveunit", false)) - rc->ec_flags |= GIVEUNIT; + rc->ec_flags |= ECF_GIVEUNIT; if (xml_bvalue(node, "getitem", false)) - rc->ec_flags |= GETITEM; + rc->ec_flags |= ECF_GETITEM; if (xml_bvalue(node, "recruitethereal", false)) rc->ec_flags |= ECF_REC_ETHEREAL; if (xml_bvalue(node, "recruitunlimited", false)) @@ -1473,7 +1483,6 @@ static int parse_races(xmlDocPtr doc) xmlNodePtr node = result->nodesetval->nodeTab[k]; int mod = xml_ivalue(node, "modifier", 0); int speed = xml_ivalue(node, "speed", study_speed_base); - skill_t sk; propValue = xmlGetProp(node, BAD_CAST "name"); assert(propValue != NULL); @@ -1481,9 +1490,7 @@ static int parse_races(xmlDocPtr doc) if (sk != NOSKILL) { rc->bonus[sk] = (char)mod; if (speed) { - if (!rc->study_speed) - rc->study_speed = calloc(1, MAXSKILLS); - rc->study_speed[sk] = (char)speed; + set_study_speed(rc, sk, speed); } } else { diff --git a/src/keyword.c b/src/keyword.c index ba05cbe4b..cb56fe940 100644 --- a/src/keyword.c +++ b/src/keyword.c @@ -134,6 +134,7 @@ const char *keywords[MAXKEYWORDS] = { "cast", "show", "destroy", + "plant", "grow", "default", "origin", diff --git a/src/keyword.h b/src/keyword.h index 19e55e88f..9be7f9f4e 100644 --- a/src/keyword.h +++ b/src/keyword.h @@ -58,7 +58,8 @@ extern "C" K_CAST, K_RESHOW, K_DESTROY, - K_BREED, + K_PLANT, + K_GROW, K_DEFAULT, K_URSPRUNG, K_EMAIL, diff --git a/src/laws.c b/src/laws.c index 5c7d36b51..c3c7d968e 100644 --- a/src/laws.c +++ b/src/laws.c @@ -42,6 +42,15 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "calendar.h" #include "guard.h" +/* attributes includes */ +#include +#include +#include + +#include +#include +#include + /* kernel includes */ #include #include @@ -66,12 +75,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include /* for volcanoes in emigration (needs a flag) */ #include -/* attributes includes */ -#include -#include -#include -#include - /* util includes */ #include #include @@ -154,23 +157,11 @@ static bool RemoveNMRNewbie(void) static void age_unit(region * r, unit * u) { - static int rc_cache; - static const race *rc_spell; + const race *rc = u_race(u); - if (rc_changed(&rc_cache)) { - rc_spell = get_race(RC_SPELL); - } - if (u_race(u) == rc_spell) { - if (--u->age <= 0) { - remove_unit(&r->units, u); - } - } - else { - const race *rc = u_race(u); - ++u->age; - if (u->number > 0 && rc->age_unit) { - rc->age_unit(u); - } + ++u->age; + if (u->number > 0 && rc->age_unit) { + rc->age_unit(u); } if (u->region && is_astral(u->region)) { item **itemp = &u->items; @@ -427,7 +418,7 @@ static void horses(region * r) maxhorses = MAX(0, maxhorses); horses = rhorses(r); if (horses > 0) { - if (is_cursed(r->attribs, C_CURSED_BY_THE_GODS, 0)) { + if (is_cursed(r->attribs, &ct_godcursezone)) { rsethorses(r, (int)(horses * 0.9)); } else if (maxhorses) { @@ -580,7 +571,7 @@ growing_trees(region * r, const int current_season, const int last_weeks_season) a_removeall(&r->attribs, &at_germs); } - if (is_cursed(r->attribs, C_CURSED_BY_THE_GODS, 0)) { + if (is_cursed(r->attribs, &ct_godcursezone)) { rsettrees(r, 1, (int)(rtrees(r, 1) * 0.9)); rsettrees(r, 2, (int)(rtrees(r, 2) * 0.9)); return; @@ -637,7 +628,7 @@ growing_trees(region * r, const int current_season, const int last_weeks_season) } else if (current_season == SEASON_SPRING) { - if (is_cursed(r->attribs, C_CURSED_BY_THE_GODS, 0)) + if (is_cursed(r->attribs, &ct_godcursezone)) return; /* in at_germs merken uns die Zahl der Samen und Sprößlinge, die @@ -2677,8 +2668,7 @@ int guard_on_cmd(unit * u, struct order *ord) if (fval(u, UFL_MOVED)) { cmistake(u, ord, 187, MSG_EVENT); } - else if (fval(u_race(u), RCF_ILLUSIONARY) - || u_race(u) == get_race(RC_SPELL)) { + else if (fval(u_race(u), RCF_ILLUSIONARY)) { cmistake(u, ord, 95, MSG_EVENT); } else { @@ -2837,7 +2827,7 @@ static void age_stonecircle(building *b) { if (!mage && is_mage(u)) { mage = u; } - if (rtype && (u_race(u)->ec_flags & ECF_KEEP_ITEM) == 0) { + if (rtype) { int n, unicorns = 0; for (n = 0; n != u->number; ++n) { if (chance(0.02)) { @@ -2859,13 +2849,12 @@ static void age_stonecircle(building *b) { if (get_astralplane()) { region *rt = r_standard_to_astral(r); if (mage && rt && !fval(rt->terrain, FORBIDDEN_REGION)) { - const struct curse_type *ct_astralblock = ct_find("astralblock"); - curse *c = get_curse(rt->attribs, ct_astralblock); + curse *c = get_curse(rt->attribs, &ct_astralblock); if (!c) { int sk = effskill(mage, SK_MAGIC, 0); float effect = 100; /* the mage reactivates the circle */ - c = create_curse(mage, &rt->attribs, ct_astralblock, + c = create_curse(mage, &rt->attribs, &ct_astralblock, (float)MAX(1, sk), MAX(1, sk / 2), effect, 0); ADDMSG(&r->msgs, msg_message("astralshield_activate", "region unit", r, mage)); @@ -2922,11 +2911,13 @@ static void ageing(void) change_effect(u, oldpotiontype[P_BERSERK], -1 * MIN(u->number, i)); } - if (is_cursed(u->attribs, C_OLDRACE, 0)) { - curse *c = get_curse(u->attribs, ct_find("oldrace")); - if (c->duration == 1 && !(c_flags(c) & CURSE_NOAGE)) { - u_setrace(u, get_race(curse_geteffect_int(c))); - u->irace = NULL; + if (u->attribs) { + curse * c = get_curse(u->attribs, &ct_oldrace); + if (c && curse_active(c)) { + if (c->duration == 1 && !(c_flags(c) & CURSE_NOAGE)) { + u_setrace(u, get_race(curse_geteffect_int(c))); + u->irace = NULL; + } } } } @@ -3234,15 +3225,14 @@ static int use_item(unit * u, const item_type * itype, int amount, struct order void monthly_healing(void) { region *r; - const curse_type *heal_ct = ct_find("healingzone"); for (r = regions; r; r = r->next) { unit *u; double healingcurse = 0; - if (r->attribs && heal_ct) { + if (r->attribs) { /* bonus zurücksetzen */ - curse *c = get_curse(r->attribs, heal_ct); + curse *c = get_curse(r->attribs, &ct_healing); if (c != NULL) { healingcurse = curse_geteffect(c); } @@ -3547,7 +3537,7 @@ int pay_cmd(unit * u, struct order *ord) static int reserve_i(unit * u, struct order *ord, int flags) { char token[128]; - if (u->number > 0 && (u_race(u)->ec_flags & GETITEM)) { + if (u->number > 0 && (u_race(u)->ec_flags & ECF_GETITEM)) { int use, count, para; const item_type *itype; const char *s; @@ -3788,7 +3778,7 @@ void process(void) } else if (u_race(u) == get_race(RC_INSECT) && r_insectstalled(r) - && !is_cursed(u->attribs, C_KAELTESCHUTZ, 0)) { + && !is_cursed(u->attribs, &ct_insectfur)) { ord = NULL; } else if (LongHunger(u)) { diff --git a/src/lighthouse.c b/src/lighthouse.c index d29d58b71..ad0cd639a 100644 --- a/src/lighthouse.c +++ b/src/lighthouse.c @@ -63,42 +63,21 @@ void update_lighthouse(building * lh) } } -int lighthouse_range(const building * b, const faction * f) +int lighthouse_range(const building * b, const faction * f, const unit *u) { - int d = 0; if (fval(b, BLD_MAINTAINED) && b->size >= 10) { int maxd = (int)log10(b->size) + 1; - if (skill_enabled(SK_PERCEPTION)) { - region *r = b->region; - int c = 0; - int cap = buildingcapacity(b); - unit *u, *uown = building_owner(b); - - for (u = r->units; u; u = u->next) { - if (u->building == b || u == uown) { - c += u->number; - if (c > cap) { - break; - } - else if (f == NULL || u->faction == f) { - int sk = effskill(u, SK_PERCEPTION, 0) / 3; - d = MAX(d, sk); - d = MIN(maxd, d); - if (d == maxd) - break; - } - } - else if (c) - break; /* first unit that's no longer in the house ends the search */ - } - } - else { - /* E3A rule: no perception req'd */ - return maxd; + if (u && skill_enabled(SK_PERCEPTION)) { + int sk = effskill(u, SK_PERCEPTION, 0) / 3; + assert(u->building == b); + assert(u->faction == f); + maxd = MIN(maxd, sk); } + /* E3A rule: no perception req'd */ + return maxd; } - return d; + return 0; } bool check_leuchtturm(region * r, faction * f) diff --git a/src/lighthouse.h b/src/lighthouse.h index f2974ddf1..2eacfb7d8 100644 --- a/src/lighthouse.h +++ b/src/lighthouse.h @@ -29,14 +29,15 @@ extern "C" { struct faction; struct region; struct building; + struct unit; struct attrib; extern struct attrib_type at_lighthouse; /* leuchtturm */ bool check_leuchtturm(struct region *r, struct faction *f); - void update_lighthouse(struct building *lh); - int lighthouse_range(const struct building *b, - const struct faction *f); + void update_lighthouse(struct building *b); + int lighthouse_range(const struct building *b, const struct faction *f, + const struct unit *u); #ifdef __cplusplus diff --git a/src/lighthouse.test.c b/src/lighthouse.test.c index e06941bc1..6105d25b9 100644 --- a/src/lighthouse.test.c +++ b/src/lighthouse.test.c @@ -26,29 +26,29 @@ static void test_lighthouse_range(CuTest * tc) u1 = test_create_unit(test_create_faction(0), r); u2 = test_create_unit(test_create_faction(0), r); b = test_create_building(r, test_create_buildingtype("lighthouse")); - CuAssertIntEquals(tc, 0, lighthouse_range(b, NULL)); - CuAssertIntEquals(tc, 0, lighthouse_range(b, u1->faction)); + CuAssertIntEquals(tc, 0, lighthouse_range(b, NULL, NULL)); + CuAssertIntEquals(tc, 0, lighthouse_range(b, u1->faction, NULL)); b->size = 10; - CuAssertIntEquals(tc, 0, lighthouse_range(b, NULL)); + CuAssertIntEquals(tc, 0, lighthouse_range(b, NULL, NULL)); u1->building = b; u2->building = b; u1->number = 10; set_level(u1, SK_PERCEPTION, 3); set_level(u2, SK_PERCEPTION, 3); - CuAssertIntEquals(tc, 0, lighthouse_range(b, NULL)); + CuAssertIntEquals(tc, 0, lighthouse_range(b, NULL, NULL)); b->flags |= BLD_MAINTAINED; - CuAssertIntEquals(tc, 1, lighthouse_range(b, NULL)); + CuAssertIntEquals(tc, 1, lighthouse_range(b, u1->faction, u1)); set_level(u1, SK_PERCEPTION, 6); - CuAssertIntEquals(tc, 2, lighthouse_range(b, u1->faction)); - CuAssertIntEquals(tc, 0, lighthouse_range(b, u2->faction)); + CuAssertIntEquals(tc, 2, lighthouse_range(b, u1->faction, u1)); + /* lighthouse_range does not check inside_building */ + CuAssertIntEquals(tc, 1, lighthouse_range(b, u2->faction, u2)); b->size = 100; update_lighthouse(b); - CuAssertIntEquals(tc, 2, lighthouse_range(b, NULL)); + CuAssertIntEquals(tc, 3, lighthouse_range(b, NULL, NULL)); + CuAssertIntEquals(tc, 2, lighthouse_range(b, u1->faction, u1)); set_level(u1, SK_PERCEPTION, 9); - CuAssertIntEquals(tc, 3, lighthouse_range(b, NULL)); - CuAssertIntEquals(tc, 3, lighthouse_range(b, u1->faction)); - CuAssertIntEquals(tc, 1, lighthouse_range(b, u2->faction)); - CuAssertIntEquals(tc, 0, lighthouse_range(b, test_create_faction(0))); + CuAssertIntEquals(tc, 3, lighthouse_range(b, u1->faction, u1)); + CuAssertIntEquals(tc, 1, lighthouse_range(b, u2->faction, u2)); test_cleanup(); } diff --git a/src/magic.c b/src/magic.c index 883bddffa..16ee37e60 100644 --- a/src/magic.c +++ b/src/magic.c @@ -25,6 +25,18 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "study.h" #include "helpers.h" #include "laws.h" +#include "spells.h" + +#include +#include +#include +#include +#include +#include + +#include +#include +#include #include #include @@ -46,13 +58,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -#include -#include -#include -#include -#include -#include - /* util includes */ #include #include @@ -695,7 +700,7 @@ int max_spellpoints(const region * r, const unit * u) if (rtype && i_get(u->items, rtype->itype) > 0) { msp += use_item_aura(r, u); } - n = get_curseeffect(u->attribs, C_AURA, 0); + n = get_curseeffect(u->attribs, &ct_auraboost); if (n > 0) { msp = (msp * n) / 100; } @@ -1024,7 +1029,7 @@ spellpower(region * r, unit * u, const spell * sp, int cast_level, struct order curse *c; /* Antimagie in der Zielregion */ - c = get_curse(r->attribs, ct_find("antimagiczone")); + c = get_curse(r->attribs, &ct_antimagiczone); if (curse_active(c)) { unit *mage = c->magician; force -= curse_geteffect(c); @@ -1043,7 +1048,7 @@ spellpower(region * r, unit * u, const spell * sp, int cast_level, struct order } /* Patzerfluch-Effekt: */ - c = get_curse(r->attribs, ct_find("fumble")); + c = get_curse(r->attribs, &ct_fumble); if (curse_active(c)) { unit *mage = c->magician; force -= curse_geteffect(c); @@ -1100,21 +1105,22 @@ variant magic_resistance(unit * target) { attrib *a; curse *c; - const curse_type * ct_goodresist = 0, *ct_badresist = 0; const resource_type *rtype; const race *rc = u_race(target); variant v, prob = rc_magres(rc); const plane *pl = rplane(target->region); + bool good_resist = true; + bool bad_resist = true; if (rc == get_race(RC_HIRNTOETER) && !pl) { - prob = frac_mul(prob, frac_make(1, 2)); + prob = frac_mul(prob, frac_make(1, 2)); } assert(target->number > 0); /* Magier haben einen Resistenzbonus vom Magietalent * 5% */ prob = frac_add(prob, frac_make(effskill(target, SK_MAGIC, 0), 20)); /* Auswirkungen von Zaubern auf der Einheit */ - c = get_curse(target->attribs, ct_find("magicresistance")); + c = get_curse(target->attribs, &ct_magicresistance); if (c) { /* TODO: legacy. magicresistance-effect is an integer-percentage stored in a double */ int effect = curse_geteffect_int(c) * get_cursedmen(target, c); @@ -1132,27 +1138,23 @@ variant magic_resistance(unit * target) /* Auswirkungen von Zaubern auf der Region */ a = a_find(target->region->attribs, &at_curse); - if (a) { - ct_badresist = ct_find("badmagicresistancezone"); - ct_goodresist = ct_find("goodmagicresistancezone"); - } while (a && a->type == &at_curse) { curse *c = (curse *)a->data.v; unit *mage = c->magician; if (mage != NULL) { - if (ct_goodresist && c->type == ct_goodresist) { + if (good_resist && c->type == &ct_goodmagicresistancezone) { if (alliedunit(mage, target->faction, HELP_GUARD)) { /* TODO: legacy. magicresistance-effect is an integer-percentage stored in a double */ prob = frac_add(prob, frac_make(curse_geteffect_int(c), 100)); - ct_goodresist = 0; /* only one effect per region */ + good_resist = false; /* only one effect per region */ } } - else if (ct_badresist && c->type == ct_badresist) { + else if (bad_resist && c->type == &ct_badmagicresistancezone) { if (!alliedunit(mage, target->faction, HELP_GUARD)) { /* TODO: legacy. magicresistance-effect is an integer-percentage stored in a double */ prob = frac_sub(prob, frac_make(curse_geteffect_int(c), 100)); - ct_badresist = 0; /* only one effect per region */ + bad_resist = false; /* only one effect per region */ } } } @@ -1209,9 +1211,6 @@ target_resists_magic(unit * magician, void *obj, int objtyp, int t_bonus) skill *sv; unit *u = (unit *)obj; - if (u_race(u)==get_race(RC_SPELL)) { - return true; - } at = effskill(magician, SK_MAGIC, 0); for (sv = u->skills; sv != u->skills + u->skill_size; ++sv) { @@ -1250,8 +1249,7 @@ target_resists_magic(unit * magician, void *obj, int objtyp, int t_bonus) } if (a) { - const struct curse_type *ct_resist = ct_find(oldcursename(C_RESIST_MAGIC)); - curse * c = get_curse(a, ct_resist); + curse * c = get_curse(a, &ct_magicrunes); int effect = curse_geteffect_int(c); prob = frac_add(prob, frac_make(effect, 100)); } @@ -1319,10 +1317,10 @@ bool fumble(region * r, unit * u, const spell * sp, int cast_grade) if (mage->magietyp == M_DRAIG) { fumble_chance += CHAOSPATZERCHANCE; } - if (is_cursed(u->attribs, C_MBOOST, 0)) { + if (is_cursed(u->attribs, &ct_magicboost)) { fumble_chance += CHAOSPATZERCHANCE; } - if (is_cursed(u->attribs, C_FUMBLE, 0)) { + if (is_cursed(u->attribs, &ct_fumble)) { fumble_chance += CHAOSPATZERCHANCE; } @@ -1410,7 +1408,7 @@ static void do_fumble(castorder * co) /* temporary skill loss */ duration = MAX(rng_int() % level / 2, 2); effect = level / -2.0; - c = create_curse(u, &u->attribs, ct_find("skillmod"), level, + c = create_curse(u, &u->attribs, &ct_skillmod, level, duration, effect, 1); c->data.i = SK_MAGIC; ADDMSG(&u->faction->msgs, msg_message("patzer2", "unit region", u, r)); @@ -1502,7 +1500,7 @@ void regenerate_aura(void) reg_aura *= btype->auraregen; /* Bonus/Malus durch Zauber */ - mod = get_curseeffect(u->attribs, C_AURA, 0); + mod = get_curseeffect(u->attribs, &ct_auraboost); if (mod > 0) { reg_aura = (reg_aura * mod) / 100.0; } @@ -2780,7 +2778,6 @@ void magic(void) int rank; castorder *co; spellrank spellranks[MAX_SPELLRANK]; - const race *rc_spell = get_race(RC_SPELL); const race *rc_insect = get_race(RC_INSECT); memset(spellranks, 0, sizeof(spellranks)); @@ -2790,11 +2787,11 @@ void magic(void) for (u = r->units; u; u = u->next) { order *ord; - if (u->number <= 0 || u_race(u) == rc_spell) + if (u->number <= 0) continue; if (u_race(u) == rc_insect && r_insectstalled(r) && - !is_cursed(u->attribs, C_KAELTESCHUTZ, 0)) + !is_cursed(u->attribs, &ct_insectfur)) continue; if (fval(u, UFL_WERE | UFL_LONGACTION)) { diff --git a/src/monsters.c b/src/monsters.c index ffc04b52e..57b557999 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -34,6 +34,8 @@ #include #include +#include + /* kernel includes */ #include #include @@ -904,13 +906,12 @@ void spawn_undead(void) { region *r; faction *monsters = get_monsters(); - const curse_type *ctype = ct_find("holyground"); for (r = regions; r; r = r->next) { int unburied = deathcount(r); - if (r->attribs && ctype) { - if (curse_active(get_curse(r->attribs, ctype))) { + if (r->attribs) { + if (curse_active(get_curse(r->attribs, &ct_holyground))) { continue; } } diff --git a/src/morale.c b/src/morale.c index 8b0a8c6ee..a8a7cc9d9 100644 --- a/src/morale.c +++ b/src/morale.c @@ -20,6 +20,8 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include "morale.h" +#include + #include #include #include @@ -53,7 +55,7 @@ void morale_update(region *r) { if (stability > MORALE_COOLDOWN && r->land->ownership->owner && morale < MORALE_MAX) { double ch = popularity(); - if (is_cursed(r->attribs, C_GENEROUS, 0)) { + if (is_cursed(r->attribs, &ct_generous)) { ch *= 1.2; /* 20% improvement */ } if (stability >= MORALE_AVERAGE * 2 || chance(ch)) { diff --git a/src/move.c b/src/move.c index 8ad25bc5a..33ed38126 100644 --- a/src/move.c +++ b/src/move.c @@ -31,6 +31,18 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "lighthouse.h" #include "piracy.h" +#include +#include +#include +#include + +/* attributes includes */ +#include +#include +#include +#include + +/* kernel includes */ #include #include #include @@ -49,8 +61,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -#include - #include "teleport.h" #include "direction.h" #include "calendar.h" @@ -71,12 +81,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include -/* attributes includes */ -#include -#include -#include -#include - /* libc includes */ #include #include @@ -670,7 +674,7 @@ static bool is_freezing(const unit * u) { if (u_race(u) != get_race(RC_INSECT)) return false; - if (is_cursed(u->attribs, C_KAELTESCHUTZ, 0)) + if (is_cursed(u->attribs, &ct_insectfur)) return false; return true; } @@ -829,7 +833,7 @@ static void drifting_ships(region * r) } /* Schiff schon abgetrieben oder durch Zauber geschützt? */ - if (!drift || fval(sh, SF_DRIFTED) || is_cursed(sh->attribs, C_SHIP_NODRIFT, 0)) { + if (!drift || fval(sh, SF_DRIFTED) || is_cursed(sh->attribs, &ct_nodrift)) { shp = &sh->next; continue; } @@ -1001,7 +1005,7 @@ bool move_blocked(const unit * u, const region * r, const region * r2) } if (r->attribs) { - const curse_type *fogtrap_ct = ct_find("fogtrap"); + const curse_type *fogtrap_ct = &ct_fogtrap; curse *c = get_curse(r->attribs, fogtrap_ct); return curse_active(c); } @@ -1253,7 +1257,7 @@ static bool roadto(const region * r, direction_t dir) return false; } if (r->attribs || r2->attribs) { - const curse_type *roads_ct = ct_find("magicstreet"); + const curse_type *roads_ct = &ct_magicstreet; if (roads_ct != NULL) { if (get_curse(r->attribs, roads_ct) != NULL) return true; @@ -1415,13 +1419,10 @@ static int movement_speed(unit * u) } if (u->attribs) { - const curse_type *speed_ct = ct_find("speed"); - if (speed_ct) { - curse *c = get_curse(u->attribs, speed_ct); - if (c != NULL) { - int men = get_cursedmen(u, c); - dk *= 1.0 + (double)men / (double)u->number; - } + curse *c = get_curse(u->attribs, &ct_speed); + if (c != NULL) { + int men = get_cursedmen(u, c); + dk *= 1.0 + (double)men / (double)u->number; } } @@ -1580,7 +1581,7 @@ static const region_list *travel_route(unit * u, /* illusionary units disappear in antimagic zones */ if (fval(u_race(u), RCF_ILLUSIONARY)) { - curse *c = get_curse(next->attribs, ct_find("antimagiczone")); + curse *c = get_curse(next->attribs, &ct_antimagiczone); if (curse_active(c)) { curse_changevigour(&next->attribs, c, (float)-u->number); ADDMSG(&u->faction->msgs, msg_message("illusionantimagic", "unit", u)); @@ -1803,7 +1804,7 @@ static void sail(unit * u, order * ord, region_list ** routep, bool drifting) } if (rng_int() % 10000 < stormchance * sh->type->storm && fval(current_point->terrain, SEA_REGION)) { - if (!is_cursed(sh->attribs, C_SHIP_NODRIFT, 0)) { + if (!is_cursed(sh->attribs, &ct_nodrift)) { region *rnext = NULL; bool storm = true; int d_offset = rng_int() % MAXDIRECTIONS; @@ -1894,7 +1895,7 @@ static void sail(unit * u, order * ord, region_list ** routep, bool drifting) break; } - if (curse_active(get_curse(next_point->attribs, ct_find("maelstrom")))) { + if (curse_active(get_curse(next_point->attribs, &ct_maelstrom))) { if (do_maelstrom(next_point, u) == NULL) break; } @@ -1920,9 +1921,10 @@ static void sail(unit * u, order * ord, region_list ** routep, bool drifting) last_point = current_point; current_point = next_point; - if (!fval(current_point->terrain, SEA_REGION) - && !is_cursed(sh->attribs, C_SHIP_FLYING, 0)) + if (!fval(next_point->terrain, SEA_REGION) + && !is_cursed(sh->attribs, &ct_flyingship)) { break; + } token = getstrtoken(); error = movewhere(u, token, current_point, &next_point); if (error || next_point == NULL) { @@ -1958,7 +1960,7 @@ static void sail(unit * u, order * ord, region_list ** routep, bool drifting) set_order(&u->thisorder, NULL); set_coast(sh, last_point, current_point); - if (is_cursed(sh->attribs, C_SHIP_FLYING, 0)) { + if (is_cursed(sh->attribs, &ct_flyingship)) { ADDMSG(&f->msgs, msg_message("shipfly", "ship from to", sh, starting_point, current_point)); } diff --git a/src/randenc.c b/src/randenc.c index ab51d64b1..2b46499e6 100644 --- a/src/randenc.c +++ b/src/randenc.c @@ -27,6 +27,13 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "chaos.h" #include "study.h" +#include +#include + +/* attributes includes */ +#include +#include + /* kernel includes */ #include #include @@ -44,10 +51,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -/* attributes includes */ -#include -#include - /* util includes */ #include #include @@ -156,7 +159,7 @@ static bool improve_all(faction * f, skill_t sk, int by_weeks) bool result = false; for (u = f->units; u; u = u->nextF) { if (has_skill(u, sk)) { - learn_skill(u, sk, by_weeks * STUDYDAYS); + increase_skill(u, sk, by_weeks); result = true; } } @@ -242,7 +245,7 @@ void find_manual(region * r, unit * u) } if (!improve_all(u->faction, skill, 3)) { - learn_skill(u, skill, 9 * STUDYDAYS); + increase_skill(u, skill, 9); } } @@ -418,7 +421,7 @@ void drown(region * r) while (*up) { unit *u = *up; - if (!(u->ship || u_race(u) == get_race(RC_SPELL) || u->number == 0 || canswim(u) || canfly(u))) { + if (!(u->ship || u->number == 0 || canswim(u) || canfly(u))) { scale_number(u, 0); ADDMSG(&u->faction->msgs, msg_message("drown", "unit region", u, r)); } @@ -649,7 +652,7 @@ static void godcurse(void) region *r; for (r = regions; r; r = r->next) { - if (is_cursed(r->attribs, C_CURSED_BY_THE_GODS, 0)) { + if (is_cursed(r->attribs, &ct_godcursezone)) { unit *u; for (u = r->units; u; u = u->next) { skill *sv = u->skills; @@ -692,32 +695,28 @@ static void orc_growth(void) for (u = r->units; u; u = u->next) { if (u->attribs && !has_skill(u, SK_MAGIC) && !has_skill(u, SK_ALCHEMY) && !fval(u, UFL_HERO)) { - const curse_type *ct_orcish = ct_find("orcish"); - - if (ct_orcish) { - curse *c = get_curse(u->attribs, ct_orcish); - if (c) { - int n; - int increase = 0; - int num = get_cursedmen(u, c); - double prob = curse_geteffect(c); - const item_type * it_chastity = it_find("ao_chastity"); - - if (it_chastity) { - num -= i_get(u->items, it_chastity); - } - for (n = num; n > 0; n--) { - if (chance(prob)) { - ++increase; - } + curse *c = get_curse(u->attribs, &ct_orcish); + if (c) { + int n; + int increase = 0; + int num = get_cursedmen(u, c); + double prob = curse_geteffect(c); + const item_type * it_chastity = it_find("ao_chastity"); + + if (it_chastity) { + num -= i_get(u->items, it_chastity); + } + for (n = num; n > 0; n--) { + if (chance(prob)) { + ++increase; } - if (increase) { - unit *u2 = create_unit(r, u->faction, increase, u_race(u), 0, NULL, u); - transfermen(u2, u, u2->number); + } + if (increase) { + unit *u2 = create_unit(r, u->faction, increase, u_race(u), 0, NULL, u); + transfermen(u2, u, u2->number); - ADDMSG(&u->faction->msgs, msg_message("orcgrowth", - "unit amount race", u, increase, u_race(u))); - } + ADDMSG(&u->faction->msgs, msg_message("orcgrowth", + "unit amount race", u, increase, u_race(u))); } } } diff --git a/src/renumber.c b/src/renumber.c index de2b5a490..30769403f 100644 --- a/src/renumber.c +++ b/src/renumber.c @@ -40,10 +40,6 @@ void renumber_factions(void) if (!a) continue; want = a->data.i; - if (fval(f, FFL_NEWID)) { - ADDMSG(&f->msgs, msg_message("renumber_twice", "id", want)); - continue; - } if (!faction_id_is_unused(want)) { a_remove(&f->attribs, a); ADDMSG(&f->msgs, msg_message("renumber_inuse", "id", want)); diff --git a/src/report.c b/src/report.c index 188d8dfa9..2cafe3157 100644 --- a/src/report.c +++ b/src/report.c @@ -29,6 +29,9 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "monsters.h" #include "travelthru.h" +#include +#include + /* modules includes */ #include @@ -997,7 +1000,7 @@ void report_region(struct stream *out, const region * r, faction * f) } /* iron & stone */ - if (r->seen.mode == seen_unit) { + if (r->seen.mode >= seen_unit) { resource_report result[MAX_RAWMATERIALS]; int n, numresults = report_resources(r, result, MAX_RAWMATERIALS, f, true); @@ -1184,8 +1187,8 @@ void report_region(struct stream *out, const region * r, faction * f) *bufp = 0; paragraph(out, buf, 0, 0, 0); - if (r->seen.mode == seen_unit && is_astral(r) && - !is_cursed(r->attribs, C_ASTRALBLOCK, 0)) { + if (r->seen.mode >= seen_unit && is_astral(r) && + !is_cursed(r->attribs, &ct_astralblock)) { /* Sonderbehandlung Teleport-Ebene */ region_list *rl = astralregions(r, inhabitable); region_list *rl2; @@ -1390,7 +1393,6 @@ report_template(const char *filename, report_context * ctx, const char *bom) char buf[8192], *bufp; size_t size; int bytes; - const curse_type *nocost_ct = ct_find("nocostbuilding"); if (F == NULL) { perror(filename); @@ -1459,7 +1461,7 @@ report_template(const char *filename, report_context * ctx, const char *bom) WARN_STATIC_BUFFER(); if (u->building && building_owner(u->building) == u) { building *b = u->building; - if (!curse_active(get_curse(b->attribs, nocost_ct))) { + if (!curse_active(get_curse(b->attribs, &ct_nocostbuilding))) { int cost = buildingmaintenance(b, rsilver); if (cost > 0) { bytes = (int)strlcpy(bufp, ",U", size); @@ -2272,7 +2274,7 @@ report_plaintext(const char *filename, report_context * ctx, continue; /* Beschreibung */ - if (r->seen.mode == seen_unit) { + if (r->seen.mode >= seen_unit) { anyunits = 1; newline(out); report_region(out, r, f); @@ -2305,24 +2307,13 @@ report_plaintext(const char *filename, report_context * ctx, report_travelthru(out, r, f); } else { - if (r->seen.mode == seen_far) { - newline(out); - report_region(out, r, f); - newline(out); - guards(out, r, f); - newline(out); - report_travelthru(out, r, f); - } - else { - newline(out); - report_region(out, r, f); - newline(out); - report_travelthru(out, r, f); - } + report_region(out, r, f); + newline(out); + report_travelthru(out, r, f); } /* Statistik */ - if (wants_stats && r->seen.mode == seen_unit) + if (wants_stats && r->seen.mode >= seen_unit) statistics(out, r, f); /* Nachrichten an REGION in der Region */ diff --git a/src/reports.c b/src/reports.c index 6448fa58b..cc11f4006 100644 --- a/src/reports.c +++ b/src/reports.c @@ -21,10 +21,20 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "reports.h" #include "guard.h" #include "laws.h" +#include "spells.h" #include "travelthru.h" #include "lighthouse.h" #include "donations.h" +/* attributes includes */ +#include +#include +#include +#include +#include + +#include + /* kernel includes */ #include #include @@ -68,12 +78,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include -/* attributes includes */ -#include -#include -#include -#include - #include "move.h" #if defined(_MSC_VER) && _MSC_VER >= 1900 @@ -93,6 +97,7 @@ const char *visibility[] = { "travel", "far", "unit", + "spell", "battle" }; @@ -158,15 +163,15 @@ size_t report_status(const unit * u, const struct locale *lang, char *fsbuf, siz const char *lname = locale_name(lang); struct locale *wloc = get_locale(lname); log_warning("no translation for combat status %s in %s", combatstatus[u->status], lname); - locale_setstring(wloc, combatstatus[u->status], combatstatus[u->status]+7); - len = strlcpy(fsbuf, combatstatus[u->status]+7, buflen); + locale_setstring(wloc, combatstatus[u->status], combatstatus[u->status] + 7); + len = strlcpy(fsbuf, combatstatus[u->status] + 7, buflen); } else { len = strlcpy(fsbuf, status, buflen); } if (fval(u, UFL_NOAID)) { - len += strlcat(fsbuf+len, ", ", buflen-len); - len += strlcat(fsbuf+len, LOC(lang, "status_noaid"), buflen-len); + len += strlcat(fsbuf + len, ", ", buflen - len); + len += strlcat(fsbuf + len, LOC(lang, "status_noaid"), buflen - len); } return len; @@ -198,7 +203,7 @@ const char *hp_status(const unit * u) void report_item(const unit * owner, const item * i, const faction * viewer, -const char **name, const char **basename, int *number, bool singular) + const char **name, const char **basename, int *number, bool singular) { const resource_type *rsilver = get_resourcetype(R_SILVER); @@ -248,7 +253,7 @@ const char **name, const char **basename, int *number, bool singular) if (name) *name = LOC(viewer->locale, resourcename(i->type->rtype, - NMF_APPEARANCE | ((i->number != 1 && !singular) ? GR_PLURAL : 0))); + NMF_APPEARANCE | ((i->number != 1 && !singular) ? GR_PLURAL : 0))); if (basename) *basename = resourcename(i->type->rtype, NMF_APPEARANCE); if (number) { @@ -283,7 +288,7 @@ static size_t buforder(char *buffer, size_t size, const order * ord, int mode) WARN_STATIC_BUFFER(); } - return bufp-buffer; + return bufp - buffer; } /** create a report of a list of items to a non-owner. @@ -303,12 +308,9 @@ report_items(const unit *u, item * result, int size, const unit * owner, assert(size); if (u->attribs) { - const curse_type *itemcloak_ct = ct_find("itemcloak"); - if (itemcloak_ct) { - curse * cu = get_curse(u->attribs, itemcloak_ct); - if (cu && curse_active(cu)) { - return 0; - } + curse * cu = get_curse(u->attribs, &ct_itemcloak); + if (cu && curse_active(cu)) { + return 0; } } for (itm = items; itm; itm = itm->next) { @@ -346,7 +348,7 @@ report_items(const unit *u, item * result, int size, const unit * owner, return n; } -static void report_resource(resource_report * result, const resource_type *rtype, +static void report_resource(resource_report * result, const resource_type *rtype, int number, int level) { assert(rtype); @@ -378,7 +380,7 @@ void report_race(const struct unit *u, const char **name, const char **illusion) void report_building(const struct building *b, const char **name, -const char **illusion) + const char **illusion) { if (name) { *name = buildingtype(b->type, b, b->size); @@ -397,7 +399,7 @@ const char **illusion) int report_resources(const region * r, resource_report * result, int size, -const faction * viewer, bool see_unit) + const faction * viewer, bool see_unit) { int n = 0; @@ -484,7 +486,7 @@ const faction * viewer, bool see_unit) int bufunit(const faction * f, const unit * u, unsigned int indent, seen_mode mode, char *buf, -size_t size) + size_t size) { int i, dh; int getarnt = fval(u, UFL_ANON_FACTION); @@ -623,7 +625,7 @@ size_t size) skill *sv; for (sv = u->skills; sv != u->skills + u->skill_size; ++sv) { size_t bytes = spskill(bufp, size, f->locale, u, sv, &dh, 1); - assert(bytes <=INT_MAX); + assert(bytes <= INT_MAX); if (wrptr(&bufp, &size, (int)bytes) != 0) WARN_STATIC_BUFFER(); } @@ -793,7 +795,7 @@ size_t size) size_t spskill(char *buffer, size_t size, const struct locale * lang, -const struct unit * u, struct skill * sv, int *dh, int days) + const struct unit * u, struct skill * sv, int *dh, int days) { char *bufp = buffer; int i, effsk; @@ -853,7 +855,7 @@ const struct unit * u, struct skill * sv, int *dh, int days) WARN_STATIC_BUFFER(); } } - return bufp-buffer; + return bufp - buffer; } void split_paragraph(strlist ** SP, const char *s, unsigned int indent, unsigned int width, char mark) @@ -924,7 +926,7 @@ void lparagraph(struct strlist **SP, char *s, unsigned int indent, char mark) void spunit(struct strlist **SP, const struct faction *f, const unit * u, unsigned int indent, -seen_mode mode) + seen_mode mode) { char buf[DISPLAYSIZE]; int dh = bufunit(f, u, indent, mode, buf, sizeof(buf)); @@ -966,9 +968,9 @@ const struct unit *ucansee(const struct faction *f, const struct unit *u, int stealth_modifier(seen_mode mode) { switch (mode) { + case seen_spell: case seen_unit: return 0; - case seen_far: case seen_lighthouse: return -2; case seen_travel: @@ -1010,7 +1012,7 @@ static void cb_add_address(region *r, unit *ut, void *cbdata) { address_data *data = (address_data *)cbdata; faction *f = data->f; - if (ut->faction==f) { + if (ut->faction == f) { unit *u; for (u = r->units; u; u = u->next) { faction *sf = visible_faction(f, u); @@ -1137,8 +1139,8 @@ void reports_done(void) { int get_regions_distance_arr(region *rc, int radius, region *result[], int size) { int n = 0, i; - - if (size>n) { + + if (size > n) { result[n++] = rc; fset(rc, RF_MARK); } @@ -1148,7 +1150,7 @@ int get_regions_distance_arr(region *rc, int radius, region *result[], int size) r = result[i]; dist = distance(rc, r); - if (distseen.modeseen.mode < seen_neighbour) { rn->seen.mode = seen_neighbour; - if (first->index>rn->index) first = rn; - if (last->indexindex) last = rn; + if (first->index > rn->index) first = rn; + if (last->index < rn->index) last = rn; } } } @@ -1387,14 +1389,14 @@ static region *firstregion(faction * f) static void cb_add_seen(region *r, unit *u, void *cbdata) { faction *f = (faction *)cbdata; - if (u->faction==f) { + if (u->faction == f) { add_seen_nb(f, r, seen_travel); } } /** set region.seen based on visibility by one faction. * - * this function may also update ctx->last and ctx->first for potential + * this function may also update ctx->last and ctx->first for potential * lighthouses and travelthru reports */ void prepare_report(report_context *ctx, faction *f) @@ -1402,14 +1404,16 @@ void prepare_report(report_context *ctx, faction *f) region *r; static int config; static bool rule_region_owners; + static bool rule_lighthouse_units; const struct building_type *bt_lighthouse = bt_find("lighthouse"); if (bt_lighthouse && config_changed(&config)) { rule_region_owners = config_token("rules.region_owner_pay_building", bt_lighthouse->_name); + rule_lighthouse_units = config_get_int("rules.lighthouse.unit_capacity", 0) != 0; } - if (f->age<=2) { - if ((f->flags&FFL_PWMSG)==0) { + if (f->age <= 2) { + if ((f->flags&FFL_PWMSG) == 0) { /* TODO: this assumes unencrypted passwords */ f->flags |= FFL_PWMSG; ADDMSG(&f->msgs, msg_message("changepasswd", "value", f->_password)); @@ -1426,28 +1430,62 @@ void prepare_report(report_context *ctx, faction *f) ctx->last = lastregion(f); for (r = ctx->first; r != ctx->last; r = r->next) { - int range = 0; unit *u; - if (fval(r, RF_LIGHTHOUSE) && bt_lighthouse) { + building *b; + int br = 0, c = 0, range = 0; + if (fval(r, RF_OBSERVER)) { + int skill = get_observer(r, f); + if (skill >= 0) { + add_seen_nb(f, r, seen_spell); + } + } + if (fval(r, RF_LIGHTHOUSE)) { + /* region owners get the report from lighthouses */ if (rule_region_owners && f == region_get_owner(r)) { - /* region owners get the report from lighthouses */ - building *b; - - for (b = r->buildings; b; b = b->next) { - if (b->type == bt_lighthouse) { - int br = lighthouse_range(b, NULL); + for (b = rbuildings(r); b; b = b->next) { + if (b && b->type == bt_lighthouse) { + /* region owners get maximm range */ + int br = lighthouse_range(b, NULL, NULL); if (br > range) range = br; } } } } + + b = NULL; for (u = r->units; u; u = u->next) { + /* if we have any unit in this region, then we get seen_unit access */ if (u->faction == f) { add_seen_nb(f, r, seen_unit); - if (fval(r, RF_LIGHTHOUSE) && bt_lighthouse) { - if (u->building && u->building->type == bt_lighthouse && inside_building(u)) { - int br = lighthouse_range(u->building, f); - if (br > range) range = br; + /* units inside the lighthouse get range based on their perception + * or the size, if perception is not a skill + */ + if (!fval(r, RF_LIGHTHOUSE)) { + /* it's enough to add the region once, and if there are + * no lighthouses, there is no need to look at more units */ + break; + } + } + if (range == 0 && u->building && u->building->type == bt_lighthouse) { + if (u->building && b != u->building) { + b = u->building; + c = buildingcapacity(b); + br = 0; + } + if (rule_lighthouse_units) { + --c; + } + else { + c -= u->number; + } + if (u->faction == f && c >= 0) { + /* unit is one of ours, and inside the current lighthouse */ + if (br == 0) { + /* lazy-calculate the range */ + br = lighthouse_range(u->building, f, u); + } + if (br > range) { + range = br; } } } @@ -1670,7 +1708,7 @@ const char *trailinto(const region * r, const struct locale *lang) size_t sz; sz = strlcpy(ref, tname, sizeof(ref)); - sz += strlcat(ref+sz, "_trail", sizeof(ref)-sz); + sz += strlcat(ref + sz, "_trail", sizeof(ref) - sz); s = LOC(lang, ref); if (s && *s) { if (strstr(s, "%s")) diff --git a/src/reports.test.c b/src/reports.test.c index 9d7944258..891f7d4af 100644 --- a/src/reports.test.c +++ b/src/reports.test.c @@ -6,6 +6,7 @@ #include "lighthouse.h" #include "travelthru.h" #include "keyword.h" +#include "spells.h" #include #include @@ -26,6 +27,7 @@ #include #include +#include #include #include @@ -474,12 +476,11 @@ void test_prepare_lighthouse_capacity(CuTest *tc) { u1->number = 4; u1->building = b; set_level(u1, SK_PERCEPTION, 3); - CuAssertIntEquals(tc, 1, lighthouse_range(b, u1->faction)); + CuAssertIntEquals(tc, 1, lighthouse_range(b, u1->faction, u1)); CuAssertPtrEquals(tc, b, inside_building(u1)); u2 = test_create_unit(f, r1); u2->building = b; set_level(u2, SK_PERCEPTION, 3); - CuAssertIntEquals(tc, 0, lighthouse_range(b, u2->faction)); CuAssertPtrEquals(tc, NULL, inside_building(u2)); prepare_report(&ctx, u1->faction); CuAssertPtrEquals(tc, r1, ctx.first); @@ -495,6 +496,15 @@ void test_prepare_lighthouse_capacity(CuTest *tc) { CuAssertIntEquals(tc, seen_neighbour, r2->seen.mode); finish_reports(&ctx); + /* lighthouse capacity is # of units, not people: */ + config_set_int("rules.lighthouse.unit_capacity", 1); + prepare_report(&ctx, u2->faction); + CuAssertPtrEquals(tc, r1, ctx.first); + CuAssertPtrEquals(tc, 0, ctx.last); + CuAssertIntEquals(tc, seen_unit, r1->seen.mode); + CuAssertIntEquals(tc, seen_lighthouse, r2->seen.mode); + finish_reports(&ctx); + test_cleanup(); } @@ -532,7 +542,12 @@ static void test_prepare_lighthouse(CuTest *tc) { test_cleanup(); } -static void test_prepare_lighthouse_owners(CuTest *tc) { +/** + * In E3, region owners get the view range benefit of + * any lighthouse in the region. + */ +static void test_prepare_lighthouse_owners(CuTest *tc) +{ report_context ctx; faction *f; region *r1, *r2, *r3; @@ -542,6 +557,7 @@ static void test_prepare_lighthouse_owners(CuTest *tc) { const struct terrain_type *t_ocean, *t_plain; test_setup(); + enable_skill(SK_PERCEPTION, false); config_set("rules.region_owner_pay_building", "lighthouse"); config_set("rules.region_owners", "1"); t_ocean = test_create_terrain("ocean", SEA_REGION); @@ -550,6 +566,7 @@ static void test_prepare_lighthouse_owners(CuTest *tc) { r1 = test_create_region(0, 0, t_plain); r2 = test_create_region(1, 0, t_ocean); r3 = test_create_region(2, 0, t_ocean); + r3 = test_create_region(3, 0, t_ocean); btype = test_create_buildingtype("lighthouse"); b = test_create_building(r1, btype); b->flags |= BLD_MAINTAINED; @@ -558,8 +575,8 @@ static void test_prepare_lighthouse_owners(CuTest *tc) { u = test_create_unit(f, r1); u = test_create_unit(test_create_faction(0), r1); u->building = b; - set_level(u, SK_PERCEPTION, 3); region_set_owner(b->region, f, 0); + CuAssertIntEquals(tc, 2, lighthouse_range(b, NULL, NULL)); prepare_report(&ctx, f); CuAssertPtrEquals(tc, r1, ctx.first); CuAssertPtrEquals(tc, NULL, ctx.last); @@ -707,6 +724,27 @@ static void test_region_distance_ql(CuTest *tc) { test_cleanup(); } +static void test_report_far_vision(CuTest *tc) { + faction *f; + region *r1, *r2; + test_setup(); + f = test_create_faction(0); + r1 = test_create_region(0, 0, 0); + test_create_unit(f, r1); + r2 = test_create_region(10, 0, 0); + set_observer(r2, f, 10, 2); + CuAssertPtrEquals(tc, r1, f->first); + CuAssertPtrEquals(tc, r2, f->last); + report_context ctx; + prepare_report(&ctx, f); + CuAssertPtrEquals(tc, r1, ctx.first); + CuAssertPtrEquals(tc, 0, ctx.last); + CuAssertIntEquals(tc, seen_unit, r1->seen.mode); + CuAssertIntEquals(tc, seen_spell, r2->seen.mode); + finish_reports(&ctx); + test_cleanup(); +} + CuSuite *get_reports_suite(void) { CuSuite *suite = CuSuiteNew(); @@ -724,6 +762,7 @@ CuSuite *get_reports_suite(void) SUITE_ADD_TEST(suite, test_get_addresses); SUITE_ADD_TEST(suite, test_get_addresses_fstealth); SUITE_ADD_TEST(suite, test_get_addresses_travelthru); + SUITE_ADD_TEST(suite, test_report_far_vision); SUITE_ADD_TEST(suite, test_reorder_units); SUITE_ADD_TEST(suite, test_seen_faction); SUITE_ADD_TEST(suite, test_regionid); diff --git a/src/spells.c b/src/spells.c index 7931be8b8..31d75c6b3 100644 --- a/src/spells.c +++ b/src/spells.c @@ -1,4 +1,4 @@ -/* +/* * * * Eressea PB(E)M host Copyright (C) 1998-2015 @@ -25,6 +25,8 @@ #include "monsters.h" #include "teleport.h" +#include + #include #include #include @@ -437,40 +439,40 @@ report_effect(region * r, unit * mage, message * seen, message * unseen) * */ -/* ------------------------------------------------------------- */ -/* Name: Vertrauter - * Stufe: 10 - * - * Wirkung: - * Der Magier beschwoert einen Vertrauten, ein kleines Tier, welches - * dem Magier zu Diensten ist. Der Magier kann durch die Augen des - * Vertrauten sehen, und durch den Vertrauten zaubern, allerdings nur - * mit seiner halben Stufe. Je nach Vertrautem erhaelt der Magier - * evtl diverse Skillmodifikationen. Der Typ des Vertrauten ist - * zufaellig bestimmt, wird aber durch Magiegebiet und Rasse beeinflu�t. - * "Tierische" Vertraute brauchen keinen Unterhalt. - * - * Ein paar Moeglichkeiten: - * Magieg. Rasse Besonderheiten - * Eule Tybied -/- fliegt, Auraregeneration - * Rabe Ilaun -/- fliegt - * Imp Draig -/- Magieresistenz? - * Fuchs Gwyrrd -/- Wahrnehmung - * ???? Cerddor -/- ???? (Singvogel?, Papagei?) - * Adler -/- -/- fliegt, +Wahrnehmung, =^=Adlerauge-Spruch? - * Kraehe -/- -/- fliegt, +Tarnung (weil unauffaellig) - * Delphin -/- Meerm. schwimmt - * Wolf -/- Ork - * Hund -/- Mensch kann evtl BEWACHE ausfuehren - * Ratte -/- Goblin - * Albatros -/- -/- fliegt, kann auf Ozean "landen" - * Affe -/- -/- kann evtl BEKLAUE ausfuehren - * Goblin -/- !Goblin normale Einheit - * Katze -/- !Katze normale Einheit - * Daemon -/- !Daemon normale Einheit - * - * Spezielle V. fuer Katzen, Trolle, Elfen, Daemonen, Insekten, Zwerge? - */ + /* ------------------------------------------------------------- */ + /* Name: Vertrauter + * Stufe: 10 + * + * Wirkung: + * Der Magier beschwoert einen Vertrauten, ein kleines Tier, welches + * dem Magier zu Diensten ist. Der Magier kann durch die Augen des + * Vertrauten sehen, und durch den Vertrauten zaubern, allerdings nur + * mit seiner halben Stufe. Je nach Vertrautem erhaelt der Magier + * evtl diverse Skillmodifikationen. Der Typ des Vertrauten ist + * zufaellig bestimmt, wird aber durch Magiegebiet und Rasse beeinflußt. + * "Tierische" Vertraute brauchen keinen Unterhalt. + * + * Ein paar Moeglichkeiten: + * Magieg. Rasse Besonderheiten + * Eule Tybied -/- fliegt, Auraregeneration + * Rabe Ilaun -/- fliegt + * Imp Draig -/- Magieresistenz? + * Fuchs Gwyrrd -/- Wahrnehmung + * ???? Cerddor -/- ???? (Singvogel?, Papagei?) + * Adler -/- -/- fliegt, +Wahrnehmung, =^=Adlerauge-Spruch? + * Kraehe -/- -/- fliegt, +Tarnung (weil unauffaellig) + * Delphin -/- Meerm. schwimmt + * Wolf -/- Ork + * Hund -/- Mensch kann evtl BEWACHE ausfuehren + * Ratte -/- Goblin + * Albatros -/- -/- fliegt, kann auf Ozean "landen" + * Affe -/- -/- kann evtl BEKLAUE ausfuehren + * Goblin -/- !Goblin normale Einheit + * Katze -/- !Katze normale Einheit + * Daemon -/- !Daemon normale Einheit + * + * Spezielle V. fuer Katzen, Trolle, Elfen, Daemonen, Insekten, Zwerge? + */ static const race *select_familiar(const race * magerace, magic_t magiegebiet) { @@ -606,7 +608,7 @@ static int sp_summon_familiar(castorder * co) if (dh == 0) { bytes = strlcpy(bufp, (const char *)LOC(mage->faction->locale, - "list_and"), size); + "list_and"), size); } else { bytes = strlcpy(bufp, (const char *)", ", size); @@ -617,7 +619,7 @@ static int sp_summon_familiar(castorder * co) } bytes = strlcpy(bufp, (const char *)skillname((skill_t)sk, mage->faction->locale), - size); + size); assert(bytes <= INT_MAX); if (wrptr(&bufp, &size, (int)bytes) != 0) WARN_STATIC_BUFFER(); @@ -825,7 +827,7 @@ static int sp_goodwinds(castorder * co) /* keine Probleme mit C_SHIP_SPEEDUP und C_SHIP_FLYING */ /* NODRIFT bewirkt auch +1 Geschwindigkeit */ - create_curse(mage, &sh->attribs, ct_find("nodrift"), power, duration, + create_curse(mage, &sh->attribs, &ct_nodrift, power, duration, zero_effect, 0); /* melden, 1x pro Partei */ @@ -872,7 +874,7 @@ static int sp_magicstreet(castorder * co) } /* wirkt schon in der Zauberrunde! */ - create_curse(mage, &r->attribs, ct_find("magicstreet"), co->force, + create_curse(mage, &r->attribs, &ct_magicstreet, co->force, co->level + 1, zero_effect, 0); /* melden, 1x pro Partei */ @@ -1022,7 +1024,7 @@ static int sp_maelstrom(castorder * co) /* Attribut auf Region. * Existiert schon ein curse, so wird dieser verstaerkt * (Max(Dauer), Max(Staerke))*/ - c = create_curse(mage, &r->attribs, ct_find("maelstrom"), co->force, duration, co->force, 0); + c = create_curse(mage, &r->attribs, &ct_maelstrom, co->force, duration, co->force, 0); /* melden, 1x pro Partei */ if (c) { @@ -1102,7 +1104,7 @@ static int sp_blessedharvest(castorder * co) * Existiert schon ein curse, so wird dieser verstaerkt * (Max(Dauer), Max(Staerke))*/ - if (create_curse(mage, &r->attribs, ct_find("blessedharvest"), co->force, + if (create_curse(mage, &r->attribs, &ct_blessedharvest, co->force, duration, 1.0, 0)) { const char * effect = co->sp->sname[0]=='b' ? "harvest_effect" : "raindance_effect"; message *seen = msg_message(effect, "mage", mage); @@ -1249,7 +1251,7 @@ static void fumble_ents(const castorder * co) * Flag: * (FARCASTING | SPELLLEVEL | UNITSPELL | TESTCANSEE | TESTRESISTANCE) */ -/* Syntax: ZAUBER [REGION x y] [STUFE 2] "Rosthauch" 1111 2222 3333 */ + /* Syntax: ZAUBER [REGION x y] [STUFE 2] "Rosthauch" 1111 2222 3333 */ typedef struct iron_weapon { const struct item_type *type; @@ -1262,7 +1264,7 @@ static iron_weapon *ironweapons = NULL; void add_ironweapon(const struct item_type *type, const struct item_type *rusty, -float chance) + float chance) { iron_weapon *iweapon = malloc(sizeof(iron_weapon)); assert_alloc(iweapon); @@ -1369,7 +1371,7 @@ static int sp_rosthauch(castorder * co) * Flag: * (UNITSPELL | SPELLLEVEL | ONSHIPCAST | TESTCANSEE) */ -/* Syntax: ZAUBER [STUFE n] "Kaelteschutz" eh1 [eh2 [eh3 [...]]] */ + /* Syntax: ZAUBER [STUFE n] "Kaelteschutz" eh1 [eh2 [eh3 [...]]] */ static int sp_kaelteschutz(castorder * co) { @@ -1405,7 +1407,7 @@ static int sp_kaelteschutz(castorder * co) } effect = 1; - create_curse(mage, &u->attribs, ct_find("insectfur"), (float)cast_level, + create_curse(mage, &u->attribs, &ct_insectfur, (float)cast_level, duration, effect, men); force -= u->number; @@ -1413,7 +1415,7 @@ static int sp_kaelteschutz(castorder * co) u)); if (u->faction != mage->faction) ADDMSG(&u->faction->msgs, msg_message("heat_effect", "mage target", - cansee(u->faction, r, mage, 0) ? mage : NULL, u)); + cansee(u->faction, r, mage, 0) ? mage : NULL, u)); i = cast_level; } /* Erstattung? */ @@ -1433,7 +1435,7 @@ static int sp_kaelteschutz(castorder * co) * Flag: * (UNITSPELL | TESTCANSEE | SPELLLEVEL) */ -/* Syntax: ZAUBER "Funkenregen" eh1 */ + /* Syntax: ZAUBER "Funkenregen" eh1 */ static int sp_sparkle(castorder * co) { @@ -1455,7 +1457,7 @@ static int sp_sparkle(castorder * co) u = pa->param[0]->data.u; effect = (float)(rng_int() % 0xffffff); - create_curse(mage, &u->attribs, ct_find("sparkle"), (float)cast_level, + create_curse(mage, &u->attribs, &ct_sparkle, (float)cast_level, duration, effect, u->number); ADDMSG(&mage->faction->msgs, msg_message("sparkle_effect", "mage target", @@ -1506,11 +1508,11 @@ static int sp_create_irongolem(castorder * co) int number = lovar(force * 8 * RESOURCE_QUANTITY); static int cache; static const race * golem_rc; - + if (rc_changed(&cache)) { golem_rc = rc_find("irongolem"); } - + if (number < 1) { number = 1; } @@ -1532,8 +1534,8 @@ static int sp_create_irongolem(castorder * co) ADDMSG(&mage->faction->msgs, msg_message("magiccreate_effect", "region command unit amount object", - mage->region, co->order, mage, number, - LOC(mage->faction->locale, rc_name_s(golem_rc, (u2->number == 1) ? NAME_SINGULAR : NAME_PLURAL)))); + mage->region, co->order, mage, number, + LOC(mage->faction->locale, rc_name_s(golem_rc, (u2->number == 1) ? NAME_SINGULAR : NAME_PLURAL)))); return cast_level; } @@ -1575,7 +1577,7 @@ static int sp_create_stonegolem(castorder * co) int number = lovar(co->force * 5 * RESOURCE_QUANTITY); static int cache; static const race * golem_rc; - + if (rc_changed(&cache)) { golem_rc = rc_find("stonegolem"); } @@ -1599,8 +1601,8 @@ static int sp_create_stonegolem(castorder * co) ADDMSG(&mage->faction->msgs, msg_message("magiccreate_effect", "region command unit amount object", - mage->region, co->order, mage, number, - LOC(mage->faction->locale, rc_name_s(golem_rc, (u2->number == 1) ? NAME_SINGULAR : NAME_PLURAL)))); + mage->region, co->order, mage, number, + LOC(mage->faction->locale, rc_name_s(golem_rc, (u2->number == 1) ? NAME_SINGULAR : NAME_PLURAL)))); return cast_level; } @@ -1659,7 +1661,7 @@ static int sp_great_drought(castorder * co) /* Arbeitslohn = 1/4 */ effect = 4.0; /* curses: higher is stronger */ - create_curse(mage, &r->attribs, ct_find("drought"), force, duration, effect, + create_curse(mage, &r->attribs, &ct_drought, force, duration, effect, 0); /* terraforming */ @@ -1692,7 +1694,7 @@ static int sp_great_drought(castorder * co) rsetterrain(r, T_OCEAN); /* Einheiten duerfen hier auf keinen Fall geloescht werden! */ for (u = r->units; u; u = u->next) { - if (u_race(u) != get_race(RC_SPELL) && u->ship == 0) { + if (!u->ship) { set_number(u, 0); } } @@ -1785,7 +1787,7 @@ static int sp_treewalkenter(castorder * co) } rt = r_standard_to_astral(r); - if (rt == NULL || is_cursed(rt->attribs, C_ASTRALBLOCK, 0) + if (rt == NULL || is_cursed(rt->attribs, &ct_astralblock) || fval(rt->terrain, FORBIDDEN_REGION)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); @@ -1901,7 +1903,7 @@ static int sp_treewalkexit(castorder * co) "spellfail_astralonly", "")); return 0; } - if (is_cursed(r->attribs, C_ASTRALBLOCK, 0)) { + if (is_cursed(r->attribs, &ct_astralblock)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); return 0; @@ -2036,7 +2038,7 @@ static int sp_holyground(castorder * co) report_spell(mage, r, msg); msg_release(msg); - ctype = ct_find("holyground"); + ctype = &ct_holyground; create_curse(mage, &r->attribs, ctype, power * power, 1, zero_effect, 0); a_removeall(&r->attribs, &at_deathcount); @@ -2083,7 +2085,7 @@ static int sp_homestone(castorder * co) /* Magieresistenz der Burg erhoeht sich um 50% */ effect = 50.0F; c = create_curse(mage, &mage->building->attribs, - ct_find("magicresistance"), force * force, 1, effect, 0); + &ct_magicresistance, force * force, 1, effect, 0); c_setflag(c, CURSE_NOAGE); /* melden, 1x pro Partei in der Burg */ @@ -2141,7 +2143,7 @@ static int sp_drought(castorder * co) * hoch (evtl dauert dann die Duerre laenger). Ansonsten volle * Auswirkungen. */ - c = get_curse(r->attribs, ct_find("drought")); + c = get_curse(r->attribs, &ct_drought); if (c) { c->vigour = MAX(c->vigour, power); c->duration = MAX(c->duration, (int)power); @@ -2154,7 +2156,7 @@ static int sp_drought(castorder * co) rsettrees(r, 0, rtrees(r, 0) / 2); rsethorses(r, rhorses(r) / 2); - create_curse(mage, &r->attribs, ct_find("drought"), power, duration, effect, + create_curse(mage, &r->attribs, &ct_drought, power, duration, effect, 0); } return cast_level; @@ -2205,17 +2207,17 @@ static int sp_ironkeeper(castorder * co) fset(keeper, UFL_ANON_FACTION); } - { - trigger *tkill = trigger_killunit(keeper); - add_trigger(&keeper->attribs, "timer", trigger_timeout(cast_level + 2, - tkill)); - } + { + trigger *tkill = trigger_killunit(keeper); + add_trigger(&keeper->attribs, "timer", trigger_timeout(cast_level + 2, + tkill)); + } - msg = msg_message("summon_effect", "mage amount race", mage, 1, u_race(keeper)); - r_addmessage(r, NULL, msg); - msg_release(msg); + msg = msg_message("summon_effect", "mage amount race", mage, 1, u_race(keeper)); + r_addmessage(r, NULL, msg); + msg_release(msg); - return cast_level; + return cast_level; } /* ------------------------------------------------------------- */ @@ -2264,19 +2266,19 @@ static int sp_stormwinds(castorder * co) sh = pa->param[n]->data.sh; /* mit C_SHIP_NODRIFT haben wir kein Problem */ - if (is_cursed(sh->attribs, C_SHIP_FLYING, 0)) { + if (is_cursed(sh->attribs, &ct_flyingship)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "error_spell_on_flying_ship", "ship", sh)) continue; } - if (is_cursed(sh->attribs, C_SHIP_SPEEDUP, 0)) { + if (is_cursed(sh->attribs, &ct_shipspeedup)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "error_spell_on_ship_already", "ship", sh)) continue; } /* Duration = 1, nur diese Runde */ - create_curse(mage, &sh->attribs, ct_find("stormwind"), power, 1, + create_curse(mage, &sh->attribs, &ct_stormwind, power, 1, zero_effect, 0); /* Da der Spruch nur diese Runde wirkt wird er nie im Report * erscheinen */ @@ -2338,7 +2340,7 @@ static int sp_earthquake(castorder * co) while (*blist) { building *burg = *blist; - if (burg->size != 0 && !is_cursed(burg->attribs, C_MAGICWALLS, 0)) { + if (burg->size != 0 && !is_cursed(burg->attribs, &ct_magicwalls)) { /* Magieresistenz */ if (!target_resists_magic(mage, burg, TYP_BUILDING, 0)) { kaputt = MIN(10 * cast_level, burg->size / 4); @@ -2386,7 +2388,7 @@ void patzer_peasantmob(const castorder * co) u = create_unit(r, f, n, get_race(RC_PEASANT), 0, LOC(f->locale, "angry_mob"), - NULL); + NULL); fset(u, UFL_ISNEW); addlist(&u->orders, create_order(K_GUARD, lang, NULL)); set_order(&u->thisorder, default_order(lang)); @@ -2458,7 +2460,7 @@ static int sp_forest_fire(castorder * co) freset(u->faction, FFL_SELECT); msg = msg_message("forestfire_effect", "mage region amount", mage, r, - destroyed + vernichtet_schoesslinge); + destroyed + vernichtet_schoesslinge); r_addmessage(r, NULL, msg); add_message(&mage->faction->msgs, msg); msg_release(msg); @@ -2538,7 +2540,7 @@ static int sp_fumblecurse(castorder * co) duration = MAX(sx, rx) + 1; effect = force / 2; - c = create_curse(mage, &target->attribs, ct_find("fumble"), + c = create_curse(mage, &target->attribs, &ct_fumble, force, duration, effect, 0); if (c == NULL) { report_failure(mage, co->order); @@ -2561,7 +2563,7 @@ void patzer_fumblecurse(const castorder * co) curse *c; effect = force / 2; - c = create_curse(mage, &mage->attribs, ct_find("fumble"), force, + c = create_curse(mage, &mage->attribs, &ct_fumble, force, duration, effect, 0); if (c != NULL) { ADDMSG(&mage->faction->msgs, msg_message("magic_fumble", @@ -2628,10 +2630,10 @@ static int sp_summondragon(castorder * co) number = 6; break; } - { - trigger *tsummon = trigger_createunit(r, f, race, number); - add_trigger(&r->attribs, "timer", trigger_timeout(time, tsummon)); - } + { + trigger *tsummon = trigger_createunit(r, f, race, number); + add_trigger(&r->attribs, "timer", trigger_timeout(time, tsummon)); + } } } @@ -2706,15 +2708,15 @@ static int sp_firewall(castorder * co) } /* melden, 1x pro Partei */ - { - message *seen = msg_message("firewall_effect", "mage region", mage, r); - message *unseen = msg_message("firewall_effect", "mage region", NULL, r); - report_effect(r, mage, seen, unseen); - msg_release(seen); - msg_release(unseen); - } - - return cast_level; + { + message *seen = msg_message("firewall_effect", "mage region", mage, r); + message *unseen = msg_message("firewall_effect", "mage region", NULL, r); + report_effect(r, mage, seen, unseen); + msg_release(seen); + msg_release(unseen); + } + + return cast_level; } /* ------------------------------------------------------------- */ @@ -2798,7 +2800,7 @@ static int sp_unholypower(castorder * co) * korrekt abgefangen wird. Besser (aber nicht gerade einfach) * waere es, eine solche Konstruktion irgendwie zu kapseln. */ if (fval(u, UFL_LOCKED) || fval(u, UFL_HUNGER) - || is_cursed(u->attribs, C_SLAVE, 0)) { + || is_cursed(u->attribs, &ct_slavery)) { cmistake(mage, co->order, 74, MSG_MAGIC); continue; } @@ -2865,7 +2867,7 @@ static int dc_age(struct curse *c) damage /= dmg.sa[1]; hp = change_hitpoints(u, -(int)damage); - ADDMSG(&u->faction->msgs, msg_message((hp>0)?"poison_damage":"poison_death", "region unit", r, u)); + ADDMSG(&u->faction->msgs, msg_message((hp > 0) ? "poison_damage" : "poison_death", "region unit", r, u)); if (*up == u) up = &u->next; } @@ -2922,7 +2924,7 @@ static int dc_read_compat(struct attrib *a, void *target, gamedata *data) effect = strength; c = create_curse(u, &r->attribs, &ct_deathcloud, strength * 2, duration, - effect, 0); + effect, 0); c->data.v = r; if (u == NULL) { ur_add(var, &c->magician, resolve_unit); @@ -3186,7 +3188,7 @@ static int sp_chaossuction(castorder * co) cmistake(mage, co->order, 216, MSG_MAGIC); return 0; } - else if (is_cursed(rt->attribs, C_ASTRALBLOCK, 0)) { + else if (is_cursed(rt->attribs, &ct_astralblock)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); return 0; @@ -3231,28 +3233,21 @@ static int sp_magicboost(castorder * co) double power = co->force; double effect; trigger *tsummon; - const curse_type *ct_auraboost; - const curse_type *ct_magicboost; - - ct_auraboost = ct_find("auraboost"); - ct_magicboost = ct_find("magicboost"); - assert(ct_auraboost != NULL); - assert(ct_magicboost != NULL); /* fehler, wenn schon ein boost */ - if (is_cursed(mage->attribs, C_MBOOST, 0)) { + if (is_cursed(mage->attribs, &ct_magicboost)) { report_failure(mage, co->order); return 0; } effect = 6; - create_curse(mage, &mage->attribs, ct_magicboost, power, 10, effect, 1); + create_curse(mage, &mage->attribs, &ct_magicboost, power, 10, effect, 1); /* one aura boost with 200% aura now: */ effect = 200; - c = create_curse(mage, &mage->attribs, ct_auraboost, power, 4, effect, 1); + c = create_curse(mage, &mage->attribs, &ct_auraboost, power, 4, effect, 1); /* and one aura boost with 50% aura in 5 weeks: */ - tsummon = trigger_createcurse(mage, mage, ct_auraboost, power, 6, 50, 1); + tsummon = trigger_createcurse(mage, mage, &ct_auraboost, power, 6, 50, 1); add_trigger(&mage->attribs, "timer", trigger_timeout(5, tsummon)); ADDMSG(&mage->faction->msgs, msg_message("magicboost_effect", @@ -3320,7 +3315,7 @@ static int sp_bloodsacrifice(castorder * co) change_spellpoints(mage, aura); ADDMSG(&mage->faction->msgs, msg_message("sp_bloodsacrifice_effect", - "unit region command amount", mage, mage->region, co->order, aura)); + "unit region command amount", mage, mage->region, co->order, aura)); return cast_level; } @@ -3647,7 +3642,7 @@ static int sp_charmingsong(castorder * co) add_trigger(&mage->faction->attribs, "destroy", trigger_killunit(target)); } /* sperre ATTACKIERE, GIB PERSON und ueberspringe Migranten */ - create_curse(mage, &target->attribs, ct_find("slavery"), force, duration, zero_effect, 0); + create_curse(mage, &target->attribs, &ct_slavery, force, duration, zero_effect, 0); /* setze Partei um und loesche langen Befehl aus Sicherheitsgruenden */ u_setfaction(target, mage->faction); @@ -3685,7 +3680,7 @@ static int sp_song_resistmagic(castorder * co) double force = co->force; int duration = (int)force + 1; - create_curse(mage, &r->attribs, ct_find("goodmagicresistancezone"), + create_curse(mage, &r->attribs, &ct_goodmagicresistancezone, force, duration, 15, 0); /* Erfolg melden */ @@ -3714,7 +3709,7 @@ static int sp_song_susceptmagic(castorder * co) double force = co->force; int duration = (int)force + 1; - create_curse(mage, &r->attribs, ct_find("badmagicresistancezone"), + create_curse(mage, &r->attribs, &ct_badmagicresistancezone, force, duration, 15, 0); ADDMSG(&mage->faction->msgs, msg_message("regionmagic_effect", @@ -3755,7 +3750,7 @@ static int sp_rallypeasantmob(castorder * co) } } - c = get_curse(r->attribs, ct_find(oldcursename(C_RIOT))); + c = get_curse(r->attribs, &ct_riotzone); if (c != NULL) { remove_curse(&r->attribs, c); } @@ -3814,7 +3809,7 @@ static int sp_raisepeasantmob(castorder * co) u = create_unit(r, monsters, n, get_race(RC_PEASANT), 0, LOC(monsters->locale, - "furious_mob"), NULL); + "furious_mob"), NULL); fset(u, UFL_ISNEW); setguard(u, true); a = a_new(&at_unitdissolve); @@ -3822,7 +3817,7 @@ static int sp_raisepeasantmob(castorder * co) a->data.ca[1] = 15; /* 15% */ a_add(&u->attribs, a); - create_curse(mage, &r->attribs, ct_find("riotzone"), (float)cast_level, duration, + create_curse(mage, &r->attribs, &ct_riotzone, (float)cast_level, duration, (float)anteil, 0); msg = msg_message("sp_raisepeasantmob_effect", "mage region", mage, r); @@ -3921,7 +3916,7 @@ static int sp_song_of_peace(castorder * co) int duration = 2 + lovar(force / 2); message *msg[2] = { NULL, NULL }; - create_curse(mage, &r->attribs, ct_find("peacezone"), force, duration, + create_curse(mage, &r->attribs, &ct_peacezone, force, duration, zero_effect, 0); for (u = r->units; u; u = u->next) @@ -3971,14 +3966,14 @@ static int sp_generous(castorder * co) double effect; message *msg[2] = { NULL, NULL }; - if (is_cursed(r->attribs, C_DEPRESSION, 0)) { + if (is_cursed(r->attribs, &ct_depression)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_generous", "")); return 0; } effect = 2; - create_curse(mage, &r->attribs, ct_find("generous"), force, duration, effect, + create_curse(mage, &r->attribs, &ct_generous, force, duration, effect, 0); for (u = r->units; u; u = u->next) @@ -4127,7 +4122,7 @@ static int sp_bigrecruit(castorder * co) * (UNITSPELL | TESTCANSEE) */ -/* restistenz der einheit pruefen */ + /* restistenz der einheit pruefen */ static int sp_pump(castorder * co) { unit *u, *target; @@ -4170,12 +4165,7 @@ static int sp_pump(castorder * co) return cast_level / 2; } - u = - create_unit(rt, mage->faction, 1, get_race(RC_SPELL), 0, - "spell/pump", NULL); - u->age = 2; - set_level(u, SK_PERCEPTION, effskill(target, SK_PERCEPTION, 0)); - + set_observer(rt, mage->faction, effskill(target, SK_PERCEPTION, 0), 2); return cast_level; } @@ -4295,7 +4285,7 @@ static int sp_calm_monster(castorder * co) } effect = mage->faction->subscription; - c = create_curse(mage, &target->attribs, ct_find("calmmonster"), force, + c = create_curse(mage, &target->attribs, &ct_calmmonster, force, (int)force, effect, 0); if (c == NULL) { report_failure(mage, co->order); @@ -4399,7 +4389,7 @@ static int sp_raisepeasants(castorder * co) u2 = create_unit(r, mage->faction, bauern, get_race(RC_PEASANT), 0, - LOC(mage->faction->locale, "furious_mob"), mage); + LOC(mage->faction->locale, "furious_mob"), mage); fset(u2, UFL_LOCKED); if (rule_stealth_anon()) { @@ -4413,7 +4403,7 @@ static int sp_raisepeasants(castorder * co) msg = msg_message("sp_raisepeasants_effect", "mage region amount", mage, r, - u2->number); + u2->number); r_addmessage(r, NULL, msg); if (mage->region != r) { add_message(&mage->faction->msgs, msg); @@ -4443,7 +4433,7 @@ static int sp_depression(castorder * co) int duration = (int)force + 1; message *msg; - create_curse(mage, &r->attribs, ct_find("depression"), force, duration, + create_curse(mage, &r->attribs, &ct_depression, force, duration, zero_effect, 0); msg = msg_message("sp_depression_effect", "mage region", mage, r); @@ -4526,7 +4516,7 @@ int sp_icastle(castorder * co) if (type == bt_illusion) { b->size = (rng_int() % (int)((power * power) + 1) * 10); } - else if (type->maxsize >0) { + else if (type->maxsize > 0) { b->size = type->maxsize; } else { @@ -4669,7 +4659,29 @@ int sp_analysedream(castorder * co) return cast_level; } -static int sp_gbdreams(castorder * co, const char *curse_name, int effect); +static int sp_gbdreams(castorder * co, int effect) +{ + int duration; + unit *mage = co->magician.u; + int cast_level = co->level; + double power = co->force; + region *r = co_get_region(co); + + /* wirkt erst in der Folgerunde, soll mindestens eine Runde wirken, + * also duration+2 */ + duration = (int)MAX(1, power / 2); /* Stufe 1 macht sonst mist */ + duration = 2 + rng_int() % duration; + + /* Nichts machen als ein entsprechendes Attribut in die Region legen. */ + create_curse(mage, &r->attribs, &ct_gbdream, power, duration, effect, 0); + + /* Erfolg melden */ + ADDMSG(&mage->faction->msgs, msg_message("regionmagic_effect", + "unit region command", mage, mage->region, co->order)); + + return cast_level; +} + /* ------------------------------------------------------------- */ /* Name: Schlechte Traeume @@ -4686,7 +4698,7 @@ static int sp_gbdreams(castorder * co, const char *curse_name, int effect); * */ int sp_baddreams(castorder * co) { - return sp_gbdreams(co, "gbdream", -1); + return sp_gbdreams(co, -1); } /* ------------------------------------------------------------- */ @@ -4703,30 +4715,7 @@ int sp_baddreams(castorder * co) */ int sp_gooddreams(castorder * co) { - return sp_gbdreams(co, "gbdream", 1); -} - -static int sp_gbdreams(castorder * co, const char *curse_name, int effect) -{ - int duration; - unit *mage = co->magician.u; - int cast_level = co->level; - double power = co->force; - region *r = co_get_region(co); - - /* wirkt erst in der Folgerunde, soll mindestens eine Runde wirken, - * also duration+2 */ - duration = (int)MAX(1, power / 2); /* Stufe 1 macht sonst mist */ - duration = 2 + rng_int() % duration; - - /* Nichts machen als ein entsprechendes Attribut in die Region legen. */ - create_curse(mage, &r->attribs, ct_find(curse_name), power, duration, effect, 0); - - /* Erfolg melden */ - ADDMSG(&mage->faction->msgs, msg_message("regionmagic_effect", - "unit region command", mage, mage->region, co->order)); - - return cast_level; + return sp_gbdreams(co, 1); } /* ------------------------------------------------------------- */ @@ -4759,7 +4748,7 @@ int sp_clonecopy(castorder * co) "clone_of"), unitname(mage)); clone = create_unit(target_region, mage->faction, 1, get_race(RC_CLONE), 0, name, - mage); + mage); setstatus(clone, ST_FLEE); fset(clone, UFL_LOCKED); @@ -4775,7 +4764,7 @@ int sp_clonecopy(castorder * co) /* ------------------------------------------------------------- */ int sp_dreamreading(castorder * co) { - unit *u, *u2; + unit *u; region *r = co_get_region(co); unit *mage = co->magician.u; int cast_level = co->level; @@ -4808,16 +4797,11 @@ int sp_dreamreading(castorder * co) return 0; } - u2 = - create_unit(u->region, mage->faction, 1, get_race(RC_SPELL), 0, - "spell/dreamreading", NULL); - set_number(u2, 1); - u2->age = 2; /* Nur fuer diese Runde. */ - set_level(u2, SK_PERCEPTION, effskill(u, SK_PERCEPTION, u2->region)); + set_observer(u->region, mage->faction, effskill(u, SK_PERCEPTION, u->region), 2); msg = msg_message("sp_dreamreading_effect", "mage unit region", mage, u, - u->region); + u->region); r_addmessage(r, mage->faction, msg); msg_release(msg); return cast_level; @@ -4862,7 +4846,7 @@ int sp_sweetdreams(castorder * co) /* Nichts machen als ein entsprechendes Attribut an die Einheit legen. */ effect = 0.05f; - create_curse(mage, &u->attribs, ct_find("orcish"), power, duration, effect, men); + create_curse(mage, &u->attribs, &ct_orcish, power, duration, effect, men); msg = msg_message("sp_sweetdreams_effect", "mage unit region", mage, u, r); r_addmessage(r, mage->faction, msg); @@ -4885,7 +4869,7 @@ int sp_disturbingdreams(castorder * co) double effect; effect = 10; - create_curse(mage, &r->attribs, ct_find("badlearn"), power, duration, effect, 0); + create_curse(mage, &r->attribs, &ct_badlearn, power, duration, effect, 0); ADDMSG(&mage->faction->msgs, msg_message("sp_disturbingdreams_effect", "mage region", mage, r)); @@ -4979,7 +4963,7 @@ int sp_itemcloak(castorder * co) /* Zieleinheit */ target = pa->param[0]->data.u; - create_curse(mage, &target->attribs, ct_find("itemcloak"), power, duration, + create_curse(mage, &target->attribs, &ct_itemcloak, power, duration, zero_effect, 0); ADDMSG(&mage->faction->msgs, msg_message("itemcloak", "mage target", mage, target)); @@ -5031,7 +5015,7 @@ int sp_resist_magic_bonus(castorder * co) m = MIN(u->number, victims); victims -= m; - create_curse(mage, &u->attribs, ct_find("magicresistance"), + create_curse(mage, &u->attribs, &ct_magicresistance, power, duration, 20, m); msg = msg_message("magicresistance_effect", "unit", u); @@ -5082,8 +5066,8 @@ int sp_enterastral(castorder * co) return 0; } - if (is_cursed(rt->attribs, C_ASTRALBLOCK, 0) - || is_cursed(ro->attribs, C_ASTRALBLOCK, 0)) { + if (is_cursed(rt->attribs, &ct_astralblock) + || is_cursed(ro->attribs, &ct_astralblock)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); return 0; @@ -5209,8 +5193,8 @@ int sp_pullastral(castorder * co) return 0; } - if (is_cursed(rt->attribs, C_ASTRALBLOCK, 0) - || is_cursed(ro->attribs, C_ASTRALBLOCK, 0)) { + if (is_cursed(rt->attribs, &ct_astralblock) + || is_cursed(ro->attribs, &ct_astralblock)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); return 0; @@ -5353,8 +5337,8 @@ int sp_leaveastral(castorder * co) return 0; } - if (ro == NULL || is_cursed(ro->attribs, C_ASTRALBLOCK, 0) - || is_cursed(rt->attribs, C_ASTRALBLOCK, 0)) { + if (ro == NULL || is_cursed(ro->attribs, &ct_astralblock) + || is_cursed(rt->attribs, &ct_astralblock)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); return 0; @@ -5493,7 +5477,7 @@ int sp_fetchastral(castorder * co) ro = u->region; } - if (is_cursed(ro->attribs, C_ASTRALBLOCK, 0)) { + if (is_cursed(ro->attribs, &ct_astralblock)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_astralblock", "")); continue; @@ -5606,10 +5590,9 @@ int sp_showastral(castorder * co) for (rl2 = rl; rl2; rl2 = rl2->next) { region *r2 = rl2->data; - if (!is_cursed(r2->attribs, C_ASTRALBLOCK, 0)) { + if (!is_cursed(r2->attribs, &ct_astralblock)) { for (u = r2->units; u; u = u->next) { - if (u_race(u) != get_race(RC_SPECIAL) && u_race(u) != get_race(RC_SPELL)) - n++; + n++; } } } @@ -5618,7 +5601,8 @@ int sp_showastral(castorder * co) /* sprintf(buf, "%s kann niemanden im astralen Nebel entdecken.", unitname(mage)); */ cmistake(mage, co->order, 220, MSG_MAGIC); - } else { + } + else { /* Ausgeben */ @@ -5626,27 +5610,26 @@ int sp_showastral(castorder * co) "Nebel zu erkennen sind ", unitname(mage)); for (rl2 = rl; rl2; rl2 = rl2->next) { - if (!is_cursed(rl2->data->attribs, C_ASTRALBLOCK, 0)) { + if (!is_cursed(rl2->data->attribs, &ct_astralblock)) { for (u = rl2->data->units; u; u = u->next) { - if (u_race(u) != get_race(RC_SPECIAL) && u_race(u) != get_race(RC_SPELL)) { - c++; - scat(unitname(u)); - scat(" ("); - if (!fval(u, UFL_ANON_FACTION)) { - scat(factionname(u->faction)); - scat(", "); - } - icat(u->number); - scat(" "); - scat(LOC(mage->faction->locale, rc_name_s(u_race(u), (u->number==1) ? NAME_SINGULAR:NAME_PLURAL))); - scat(", Entfernung "); - icat(distance(rl2->data, rt)); - scat(")"); - if (c == n - 1) { - scat(" und "); - } else if (c < n - 1) { - scat(", "); - } + c++; + scat(unitname(u)); + scat(" ("); + if (!fval(u, UFL_ANON_FACTION)) { + scat(factionname(u->faction)); + scat(", "); + } + icat(u->number); + scat(" "); + scat(LOC(mage->faction->locale, rc_name_s(u_race(u), (u->number == 1) ? NAME_SINGULAR : NAME_PLURAL))); + scat(", Entfernung "); + icat(distance(rl2->data, rt)); + scat(")"); + if (c == n - 1) { + scat(" und "); + } + else if (c < n - 1) { + scat(", "); } } } @@ -5666,7 +5649,6 @@ int sp_showastral(castorder * co) int sp_viewreality(castorder * co) { region_list *rl, *rl2; - unit *u; region *r = co_get_region(co); unit *mage = co->magician.u; int cast_level = co->level; @@ -5684,12 +5666,8 @@ int sp_viewreality(castorder * co) /* Irgendwann mal auf Curses u/o Attribut umstellen. */ for (rl2 = rl; rl2; rl2 = rl2->next) { region *rt = rl2->data; - if (!is_cursed(rt->attribs, C_ASTRALBLOCK, 0)) { - u = - create_unit(rt, mage->faction, 1, get_race(RC_SPELL), 0, - "spell/viewreality", NULL); - set_level(u, SK_PERCEPTION, co->level / 2); - u->age = 2; + if (!is_cursed(rt->attribs, &ct_astralblock)) { + set_observer(rt, mage->faction, co->level / 2, 2); } } @@ -5742,7 +5720,7 @@ int sp_disruptastral(castorder * co) int inhab_regions = 0; region_list *trl = NULL; - if (is_cursed(r2->attribs, C_ASTRALBLOCK, 0)) + if (is_cursed(r2->attribs, &ct_astralblock)) continue; if (r2->units != NULL) { @@ -5768,24 +5746,22 @@ int sp_disruptastral(castorder * co) if (trl != NULL) { for (u = r2->units; u; u = u->next) { - if (u_race(u) != get_race(RC_SPELL)) { - region_list *trl2 = trl; - region *tr; - int c = rng_int() % inhab_regions; - - /* Zufaellige Zielregion suchen */ - while (c-- != 0) - trl2 = trl2->next; - tr = trl2->data; - - if (!is_magic_resistant(mage, u, 0) && can_survive(u, tr)) { - message *msg = msg_message("disrupt_astral", "unit region", u, tr); - add_message(&u->faction->msgs, msg); - add_message(&tr->msgs, msg); - msg_release(msg); - - move_unit(u, tr, NULL); - } + region_list *trl2 = trl; + region *tr; + int c = rng_int() % inhab_regions; + + /* Zufaellige Zielregion suchen */ + while (c-- != 0) + trl2 = trl2->next; + tr = trl2->data; + + if (!is_magic_resistant(mage, u, 0) && can_survive(u, tr)) { + message *msg = msg_message("disrupt_astral", "unit region", u, tr); + add_message(&u->faction->msgs, msg); + add_message(&tr->msgs, msg); + msg_release(msg); + + move_unit(u, tr, NULL); } } free_regionlist(trl); @@ -5793,7 +5769,7 @@ int sp_disruptastral(castorder * co) /* Kontakt unterbinden */ effect = 100; - create_curse(mage, &rl2->data->attribs, ct_find("astralblock"), + create_curse(mage, &rl2->data->attribs, &ct_astralblock, power, duration, effect, 0); } @@ -5829,7 +5805,7 @@ static int sp_eternizewall(castorder * co) return 0; b = pa->param[0]->data.b; - c = create_curse(mage, &b->attribs, ct_find("nocostbuilding"), + c = create_curse(mage, &b->attribs, &ct_nocostbuilding, power * power, 1, zero_effect, 0); if (c == NULL) { /* ist bereits verzaubert */ @@ -6107,7 +6083,7 @@ int sp_antimagiczone(castorder * co) /* Reduziert die Staerke jedes Spruchs um effect */ effect = cast_level; - create_curse(mage, &r->attribs, ct_find("antimagiczone"), power, duration, + create_curse(mage, &r->attribs, &ct_antimagiczone, power, duration, effect, 0); /* Erfolg melden */ @@ -6167,7 +6143,7 @@ static int sp_magicrunes(castorder * co) b = pa->param[0]->data.b; /* Magieresistenz der Burg erhoeht sich um 20% */ - create_curse(mage, &b->attribs, ct_find("magicrunes"), force, + create_curse(mage, &b->attribs, &ct_magicrunes, force, duration, effect, 0); /* Erfolg melden */ @@ -6181,7 +6157,7 @@ static int sp_magicrunes(castorder * co) ship *sh; sh = pa->param[0]->data.sh; /* Magieresistenz des Schiffs erhoeht sich um 20% */ - create_curse(mage, &sh->attribs, ct_find("magicrunes"), force, + create_curse(mage, &sh->attribs, &ct_magicrunes, force, duration, effect, 0); /* Erfolg melden */ @@ -6233,7 +6209,7 @@ int sp_speed2(castorder * co) men = MIN(maxmen, u->number); effect = 2; - create_curse(mage, &u->attribs, ct_find("speed"), force, dur, effect, men); + create_curse(mage, &u->attribs, &ct_speed, force, dur, effect, men); maxmen -= men; used += men; } @@ -6263,8 +6239,8 @@ int sp_speed2(castorder * co) * Flags: * (FARCASTING | SPELLLEVEL | ONSHIPCAST | TESTCANSEE) */ -/* Jeder gebrochene Zauber verbraucht c->vigour an Zauberkraft - * (force) */ + /* Jeder gebrochene Zauber verbraucht c->vigour an Zauberkraft + * (force) */ int sp_q_antimagie(castorder * co) { attrib **ap; @@ -6412,7 +6388,7 @@ int sp_break_curse(castorder * co) /* Es wurde kein Ziel gefunden */ ADDMSG(&mage->faction->msgs, msg_message("spelltargetnotfound", "unit region command", - mage, mage->region, co->order)); + mage, mage->region, co->order)); } /* curse aufloesen, wenn zauber staerker (force > vigour) */ @@ -6772,7 +6748,7 @@ void register_spells(void) #ifdef COMPAT_DEATHCLOUD at_deprecate("zauber_todeswolke", dc_read_compat); #endif - + /* init_firewall(); */ ct_register(&ct_firewall); ct_register(&ct_deathcloud); diff --git a/src/spells.h b/src/spells.h index 275f41f7a..3b6280e83 100644 --- a/src/spells.h +++ b/src/spells.h @@ -22,14 +22,17 @@ extern "C" { #endif - struct ship; - struct curse; + struct curse_type; + struct region; struct unit; + struct faction; + struct region; struct message; + extern const struct curse_type ct_magicresistance; + void register_magicresistance(void); void register_spells(void); - void set_spelldata(struct spell *sp); int sp_baddreams(castorder * co); int sp_gooddreams(castorder * co); diff --git a/src/spells.test.c b/src/spells.test.c index 07a980e03..cd72f0177 100644 --- a/src/spells.test.c +++ b/src/spells.test.c @@ -1,15 +1,20 @@ #include + +#include "spells.h" + #include #include #include #include +#include #include #include #include #include #include #include -#include "spells.h" + +#include #include #include @@ -39,7 +44,7 @@ static void test_good_dreams(CuTest *tc) { level = sp_gooddreams(&co); CuAssertIntEquals(tc, 10, level); - curse = get_curse(r->attribs, ct_find("gbdream")); + curse = get_curse(r->attribs, &ct_gbdream); CuAssertTrue(tc, curse && curse->duration > 1); CuAssertTrue(tc, curse->effect == 1); @@ -96,7 +101,7 @@ static void test_bad_dreams(CuTest *tc) { level = sp_baddreams(&co); CuAssertIntEquals(tc, 10, level); - curse = get_curse(r->attribs, ct_find("gbdream")); + curse = get_curse(r->attribs, &ct_gbdream); CuAssertTrue(tc, curse && curse->duration > 1); CuAssertTrue(tc, curse->effect == -1); @@ -108,9 +113,26 @@ static void test_bad_dreams(CuTest *tc) { test_cleanup(); } +static void test_watch_region(CuTest *tc) { + region *r; + faction *f; + test_setup(); + r = test_create_region(0, 0, 0); + f = test_create_faction(0); + CuAssertIntEquals(tc, -1, get_observer(r, f)); + set_observer(r, f, 0, 2); + CuAssertIntEquals(tc, 0, get_observer(r, f)); + set_observer(r, f, 10, 2); + CuAssertIntEquals(tc, 10, get_observer(r, f)); + CuAssertIntEquals(tc, RF_OBSERVER, fval(r, RF_OBSERVER)); + CuAssertPtrNotNull(tc, r->attribs); + test_cleanup(); +} + CuSuite *get_spells_suite(void) { CuSuite *suite = CuSuiteNew(); + SUITE_ADD_TEST(suite, test_watch_region); SUITE_ADD_TEST(suite, test_good_dreams); SUITE_ADD_TEST(suite, test_bad_dreams); SUITE_ADD_TEST(suite, test_dreams); diff --git a/src/spells/buildingcurse.c b/src/spells/buildingcurse.c index f1a3b3773..06c62e1c4 100644 --- a/src/spells/buildingcurse.c +++ b/src/spells/buildingcurse.c @@ -70,7 +70,7 @@ static message *cinfo_magicrunes(const void *obj, objtype_t typ, const curse * c return msg; } -static struct curse_type ct_magicrunes = { "magicrunes", +const struct curse_type ct_magicrunes = { "magicrunes", CURSETYP_NORM, 0, M_SUMEFFECT, cinfo_magicrunes }; @@ -80,12 +80,12 @@ CURSETYP_NORM, CURSE_ONLYONE|CURSE_NOAGE, NO_MERGE, cinfo_building }; /* Feste Mauer - Präkampfzauber, wirkt nur 1 Runde */ -static struct curse_type ct_strongwall = { "strongwall", +const struct curse_type ct_strongwall = { "strongwall", CURSETYP_NORM, 0, NO_MERGE, NULL }; /* Ewige Mauern-Zauber */ -static struct curse_type ct_nocostbuilding = { "nocostbuilding", +const struct curse_type ct_nocostbuilding = { "nocostbuilding", CURSETYP_NORM, CURSE_NOAGE | CURSE_ONLYONE, NO_MERGE, cinfo_building }; diff --git a/src/spells/buildingcurse.h b/src/spells/buildingcurse.h index 2150dc3b6..03fc46a63 100644 --- a/src/spells/buildingcurse.h +++ b/src/spells/buildingcurse.h @@ -24,6 +24,9 @@ extern "C" { struct curse_type; extern const struct curse_type ct_magicwalls; + extern const struct curse_type ct_strongwall; + extern const struct curse_type ct_magicrunes; + extern const struct curse_type ct_nocostbuilding; extern void register_buildingcurse(void); struct message *cinfo_building(const void *obj, objtype_t typ, const struct curse * c, int self); diff --git a/src/spells/combatspells.c b/src/spells/combatspells.c index 08f38c8ea..da2d52182 100644 --- a/src/spells/combatspells.c +++ b/src/spells/combatspells.c @@ -12,7 +12,9 @@ #include #include "combatspells.h" -/* kernel includes */ +#include + + /* kernel includes */ #include #include #include @@ -256,6 +258,17 @@ static void scramble_fighters(selist * ql) } } +static bool select_armed(const side *vs, const fighter *fig, void *cbdata) +{ + int row = get_unitrow(fig, vs); + + UNUSED_ARG(cbdata); + if (row >= FIGHT_ROW && row < BEHIND_ROW) { + return fig->alive > 0 && fig->weapons; + } + return false; +} + /* Rosthauch */ int sp_combatrosthauch(struct castorder * co) { @@ -275,52 +288,42 @@ int sp_combatrosthauch(struct castorder * co) return 0; } - fgs = fighters(b, fi->side, FIGHT_ROW, BEHIND_ROW - 1, FS_ENEMY); + fgs = select_fighters(b, fi->side, FS_ENEMY, select_armed, NULL); scramble_fighters(fgs); - for (qi = 0, ql = fgs; ql; selist_advance(&ql, &qi, 1)) { + for (qi = 0, ql = fgs; force>0 && ql; selist_advance(&ql, &qi, 1)) { fighter *df = (fighter *)selist_get(ql, qi); - - if (df->alive == 0) - continue; - if (force <= 0) - break; - - /* da n MIN(force, x), sollte force maximal auf 0 sinken */ - assert(force >= 0); - - if (df->weapons) { - int w; - for (w = 0; df->weapons[w].type != NULL; ++w) { - weapon *wp = df->weapons; - int n = MIN(force, wp->used); - if (n) { - requirement *mat = wp->type->itype->construction->materials; - bool iron = false; - while (mat && mat->number > 0) { - if (mat->rtype == get_resourcetype(R_IRON)) { - iron = true; - break; - } - mat++; + int w; + + for (w = 0; df->weapons[w].type != NULL; ++w) { + weapon *wp = df->weapons; + int n = MIN(force, wp->used); + if (n) { + requirement *mat = wp->type->itype->construction->materials; + bool iron = false; + while (mat && mat->number > 0) { + if (mat->rtype == get_resourcetype(R_IRON)) { + iron = true; + break; } - if (iron) { - int p; - force -= n; - wp->used -= n; - k += n; - i_change(&df->unit->items, wp->type->itype, -n); - for (p = 0; n && p != df->unit->number; ++p) { - if (df->person[p].missile == wp) { - df->person[p].missile = NULL; - --n; - } + mat++; + } + if (iron) { + int p; + force -= n; + wp->used -= n; + k += n; + i_change(&df->unit->items, wp->type->itype, -n); + for (p = 0; n && p != df->unit->number; ++p) { + if (df->person[p].missile == wp) { + df->person[p].missile = NULL; + --n; } - for (p = 0; n && p != df->unit->number; ++p) { - if (df->person[p].melee == wp) { - df->person[p].melee = NULL; - --n; - } + } + for (p = 0; n && p != df->unit->number; ++p) { + if (df->person[p].melee == wp) { + df->person[p].melee = NULL; + --n; } } } @@ -879,7 +882,7 @@ int sp_strong_wall(struct castorder * co) burg = mage->building; effect = power / 4; - create_curse(mage, &burg->attribs, ct_find("strongwall"), power, 1, effect, 0); + create_curse(mage, &burg->attribs, &ct_strongwall, power, 1, effect, 0); msg = msg_message("sp_strongwalls_effect", "mage building", mage, mage->building); @@ -889,6 +892,13 @@ int sp_strong_wall(struct castorder * co) return level; } +static bool select_alive(const side *vs, const fighter *fig, void *cbdata) +{ + UNUSED_ARG(vs); + UNUSED_ARG(cbdata); + return fig->alive > 0; +} + /** Spells: chaosrow / song of confusion. * German Title: 'Gesang der Verwirrung' */ @@ -915,7 +925,7 @@ int sp_chaosrow(struct castorder * co) power = chaosrow ? (power * 40) : get_force(power, 5); - fgs = fighters(b, fi->side, FIGHT_ROW, NUMROWS, FS_ENEMY); + fgs = select_fighters(b, fi->side, FS_ENEMY, select_alive, NULL); scramble_fighters(fgs); for (qi = 0, ql = fgs; ql; selist_advance(&ql, &qi, 1)) { @@ -977,9 +987,18 @@ int sp_chaosrow(struct castorder * co) return level; } +static bool select_afraid(const side *vs, const fighter *fig, void *cbdata) +{ + int row = get_unitrow(fig, vs); + UNUSED_ARG(cbdata); + if (row >= FIGHT_ROW && row <= AVOID_ROW) { + return fig->alive + fig->run.number < fig->unit->number; + } + return false; +} + /* Gesang der Furcht (Kampfzauber) */ /* Panik (Pr�kampfzauber) */ - int flee_spell(struct castorder * co, int strength) { fighter * fi = co->magician.fig; @@ -994,23 +1013,20 @@ int flee_spell(struct castorder * co, int strength) int force; force = (int)get_force(power, strength); - if (!count_enemies(b, fi, FIGHT_ROW, AVOID_ROW, SELECT_ADVANCE | SELECT_FIND)) { + if (force<=0 || !count_enemies(b, fi, FIGHT_ROW, AVOID_ROW, SELECT_ADVANCE | SELECT_FIND)) { msg = msg_message("sp_flee_effect_0", "mage spell", mage, sp); message_all(b, msg); msg_release(msg); return 0; } - fgs = fighters(b, fi->side, FIGHT_ROW, AVOID_ROW, FS_ENEMY); + fgs = select_fighters(b, fi->side, FS_ENEMY, select_afraid, NULL); scramble_fighters(fgs); - for (qi = 0, ql = fgs; ql; selist_advance(&ql, &qi, 1)) { + for (qi = 0, ql = fgs; force > 0 && ql; selist_advance(&ql, &qi, 1)) { fighter *df = (fighter *)selist_get(ql, qi); - for (n = 0; n != df->alive; ++n) { - if (force < 0) - break; - + for (n = 0; force > 0 && n != df->alive; ++n) { if (df->person[n].flags & FL_PANICED) { /* bei SPL_SONG_OF_FEAR m�glich */ df->person[n].attack -= 1; --force; @@ -1556,8 +1572,8 @@ int sp_healing(struct castorder * co) message *msg; bool use_item = has_ao_healing(mage); - /* bis zu 11 Personen pro Stufe (einen HP m�ssen sie ja noch - * haben, sonst w�ren sie tot) k�nnen geheilt werden */ + /* bis zu 11 Personen pro Stufe (einen HP muessen sie ja noch + * haben, sonst waeren sie tot) koennen geheilt werden */ if (use_item) { healhp *= 2; @@ -1589,6 +1605,19 @@ int sp_healing(struct castorder * co) return level; } +static bool select_hero(const side *vs, const fighter *fig, void *cbdata) +{ + UNUSED_ARG(cbdata); + + if (playerrace(u_race(fig->unit))) { + int row = get_unitrow(fig, vs); + if (row >= FIGHT_ROW && row <= AVOID_ROW) { + return fig->alive + fig->run.number < fig->unit->number; + } + } + return false; +} + int sp_undeadhero(struct castorder * co) { fighter * fi = co->magician.fig; @@ -1603,80 +1632,69 @@ int sp_undeadhero(struct castorder * co) int force = (int)get_force(power, 0); double c = 0.50 + 0.02 * power; - /* Liste aus allen K�mpfern */ - fgs = fighters(b, fi->side, FIGHT_ROW, AVOID_ROW, FS_ENEMY | FS_HELP); + /* Liste aus allen Kaempfern */ + fgs = select_fighters(b, fi->side, FS_ENEMY | FS_HELP, select_hero, NULL); scramble_fighters(fgs); - for (qi = 0, ql = fgs; ql; selist_advance(&ql, &qi, 1)) { + for (qi = 0, ql = fgs; ql && force>0; selist_advance(&ql, &qi, 1)) { fighter *df = (fighter *)selist_get(ql, qi); unit *du = df->unit; + int j = 0; - if (force <= 0) - break; - - /* keine Monster */ - if (!playerrace(u_race(du))) - continue; - - if (df->alive + df->run.number < du->number) { - int j = 0; - - /* Wieviele Untote k�nnen wir aus dieser Einheit wecken? */ - for (n = df->alive + df->run.number; n != du->number; n++) { - if (chance(c)) { - ++j; - if (--force <= 0) - break; - } + /* Wieviele Untote koennen wir aus dieser Einheit wecken? */ + for (n = df->alive + df->run.number; force>0 && n != du->number; n++) { + if (chance(c)) { + ++j; + --force; } + } - if (j > 0) { - item **ilist; - unit *u = - create_unit(r, mage->faction, 0, get_race(RC_UNDEAD), 0, unit_getname(du), - du); + if (j > 0) { + item **ilist; + unit *u = + create_unit(r, mage->faction, 0, get_race(RC_UNDEAD), 0, unit_getname(du), + du); - /* new units gets some stats from old unit */ + /* new units gets some stats from old unit */ - if (du->display) { - unit_setinfo(u, du->display); - } - else { - unit_setinfo(u, NULL); - } - setstatus(u, du->status); - setguard(u, false); - for (ilist = &du->items; *ilist;) { - item *itm = *ilist; - int loot = itm->number * j / du->number; - if (loot != itm->number) { - int split = itm->number * j % du->number; - if (split > 0 && (rng_int() % du->number) < split) { - ++loot; - } - } - i_change(&u->items, itm->type, loot); - i_change(ilist, itm->type, -loot); - if (*ilist == itm) { - ilist = &itm->next; + if (du->display) { + unit_setinfo(u, du->display); + } + else { + unit_setinfo(u, NULL); + } + setstatus(u, du->status); + setguard(u, false); + for (ilist = &du->items; *ilist;) { + item *itm = *ilist; + int loot = itm->number * j / du->number; + if (loot != itm->number) { + int split = itm->number * j % du->number; + if (split > 0 && (rng_int() % du->number) < split) { + ++loot; } } - - /* inherit stealth from magician */ - if (mage->flags & UFL_ANON_FACTION) { - u->flags |= UFL_ANON_FACTION; + i_change(&u->items, itm->type, loot); + i_change(ilist, itm->type, -loot); + if (*ilist == itm) { + ilist = &itm->next; } + } - /* transfer dead people to new unit, set hitpoints to those of old unit */ - transfermen(du, u, j); - u->hp = u->number * unit_max_hp(du); - assert(j <= df->side->casualties); - df->side->casualties -= j; - df->side->dead -= j; - - /* counting total number of undead */ - undead += j; + /* inherit stealth from magician */ + if (mage->flags & UFL_ANON_FACTION) { + u->flags |= UFL_ANON_FACTION; } + + /* transfer dead people to new unit, set hitpoints to those of old unit */ + transfermen(du, u, j); + u->hp = u->number * unit_max_hp(du); + assert(j <= df->side->casualties); + df->side->casualties -= j; + df->side->dead -= j; + + /* counting total number of undead */ + undead += j; } } selist_free(fgs); diff --git a/src/spells/flyingship.c b/src/spells/flyingship.c index eac2043dd..e23f6906b 100644 --- a/src/spells/flyingship.c +++ b/src/spells/flyingship.c @@ -1,6 +1,8 @@ #include #include "flyingship.h" +#include + #include #include #include @@ -13,8 +15,6 @@ #include -#include - #include /* libc includes */ @@ -66,11 +66,11 @@ int sp_flying_ship(castorder * co) cno = levitate_ship(sh, mage, power, 1); if (cno == 0) { - if (is_cursed(sh->attribs, C_SHIP_FLYING, 0)) { + if (is_cursed(sh->attribs, &ct_flyingship)) { /* Auf dem Schiff befindet liegt bereits so ein Zauber. */ cmistake(mage, co->order, 211, MSG_MAGIC); } - else if (is_cursed(sh->attribs, C_SHIP_SPEEDUP, 0)) { + else if (is_cursed(sh->attribs, &ct_shipspeedup)) { /* Es ist zu gefaehrlich, ein sturmgepeitschtes Schiff fliegen zu lassen. */ cmistake(mage, co->order, 210, MSG_MAGIC); } @@ -127,7 +127,7 @@ static int flyingship_age(curse * c) return 0; } -static struct curse_type ct_flyingship = { "flyingship", +const struct curse_type ct_flyingship = { "flyingship", CURSETYP_NORM, 0, NO_MERGE, cinfo_ship, NULL, flyingship_read, flyingship_write, NULL, flyingship_age }; @@ -153,7 +153,7 @@ static curse *shipcurse_flyingship(ship * sh, unit * mage, double power, int dur if (curse_active(get_curse(sh->attribs, &ct_flyingship))) { return NULL; } - if (is_cursed(sh->attribs, C_SHIP_SPEEDUP, 0)) { + if (is_cursed(sh->attribs, &ct_shipspeedup)) { return NULL; } } diff --git a/src/spells/flyingship.h b/src/spells/flyingship.h index 52013ac13..4a03e6b71 100644 --- a/src/spells/flyingship.h +++ b/src/spells/flyingship.h @@ -12,6 +12,9 @@ extern "C" { struct castorder; struct ship; struct unit; + struct curse_type; + + extern const struct curse_type ct_flyingship; int sp_flying_ship(struct castorder * co); diff --git a/src/spells/magicresistance.c b/src/spells/magicresistance.c index 7b0c1413d..6f2ec49ec 100644 --- a/src/spells/magicresistance.c +++ b/src/spells/magicresistance.c @@ -22,7 +22,7 @@ static struct message *cinfo_magicresistance(const void *obj, objtype_t typ, con return 0; } -static struct curse_type ct_magicresistance = { +const struct curse_type ct_magicresistance = { "magicresistance", CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN, cinfo_magicresistance }; diff --git a/src/spells/magicresistance.test.c b/src/spells/magicresistance.test.c index 621d2c3b4..7576911af 100644 --- a/src/spells/magicresistance.test.c +++ b/src/spells/magicresistance.test.c @@ -36,7 +36,7 @@ static void test_magicresistance_unit(CuTest *tc) { f2 = test_create_faction(NULL); u2 = test_create_unit(f2, r); - c = create_curse(u1, &u2->attribs, ct_find("magicresistance"), 10, 20, 30, u2->number); + c = create_curse(u1, &u2->attribs, &ct_magicresistance, 10, 20, 30, u2->number); CuAssertPtrNotNull(tc, u2->attribs); CuAssertPtrEquals(tc, (void *)&at_curse, (void *)u2->attribs->type); msg = c->type->curseinfo(u2, TYP_UNIT, c, 1); @@ -62,7 +62,7 @@ static void test_magicresistance_building(CuTest *tc) { b1 = test_create_building(r, NULL); - c = create_curse(u1, &b1->attribs, ct_find("magicresistance"), 10, 20, 30, 0); + c = create_curse(u1, &b1->attribs, &ct_magicresistance, 10, 20, 30, 0); CuAssertPtrNotNull(tc, b1->attribs); CuAssertPtrEquals(tc, (void *)&at_curse, (void *)b1->attribs->type); msg = c->type->curseinfo(b1, TYP_BUILDING, c, 1); diff --git a/src/spells/regioncurse.c b/src/spells/regioncurse.c index 1d69574df..8ac3eea8c 100644 --- a/src/spells/regioncurse.c +++ b/src/spells/regioncurse.c @@ -24,6 +24,7 @@ #include /* util includes */ +#include #include #include #include @@ -33,10 +34,6 @@ #include #include -/* --------------------------------------------------------------------- */ -/* CurseInfo mit Spezialabfragen - */ - /* * godcursezone */ @@ -55,7 +52,7 @@ static message *cinfo_cursed_by_the_gods(const void *obj, objtype_t typ, return msg_message("curseinfo::godcurse", "id", c->no); } -static struct curse_type ct_godcursezone = { +const struct curse_type ct_godcursezone = { "godcursezone", CURSETYP_NORM, CURSE_IMMUNE, (NO_MERGE), cinfo_cursed_by_the_gods, @@ -81,7 +78,7 @@ static message *cinfo_dreamcurse(const void *obj, objtype_t typ, const curse * c } } -static struct curse_type ct_gbdream = { +const struct curse_type ct_gbdream = { "gbdream", CURSETYP_NORM, 0, (NO_MERGE), cinfo_dreamcurse }; @@ -106,7 +103,7 @@ static message *cinfo_magicstreet(const void *obj, objtype_t typ, const curse * return msg_message("curseinfo::magicstreetwarn", "id", c->no); } -static struct curse_type ct_magicstreet = { +const struct curse_type ct_magicstreet = { "magicstreet", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_magicstreet @@ -157,7 +154,7 @@ const curse * c, int self) return self; } -static struct curse_type ct_antimagiczone = { +const struct curse_type ct_antimagiczone = { "antimagiczone", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_antimagiczone, NULL, NULL, NULL, cansee_antimagiczone @@ -188,59 +185,75 @@ static struct curse_type ct_farvision = { /* --------------------------------------------------------------------- */ -static struct curse_type ct_fogtrap = { +const struct curse_type ct_fogtrap = { "fogtrap", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_simple }; -static struct curse_type ct_maelstrom = { +const struct curse_type ct_maelstrom = { "maelstrom", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_simple }; -static struct curse_type ct_blessedharvest = { +const struct curse_type ct_blessedharvest = { "blessedharvest", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_simple }; -static struct curse_type ct_drought = { +int harvest_effect(const struct region *r) { + if (r->attribs) { + curse *c = get_curse(r->attribs, &ct_blessedharvest); + if (c) { + int happy = curse_geteffect_int(c); + if (happy != 1) { + /* https://bugs.eressea.de/view.php?id=2353 detect and fix bad harvest */ + log_error("blessedharvest curse %d has effect=%d, duration=%d", c->no, happy, c->duration); + c->effect = 1.0; + } + return happy; + } + } + return 0; +} + +const struct curse_type ct_drought = { "drought", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_simple }; -static struct curse_type ct_badlearn = { +const struct curse_type ct_badlearn = { "badlearn", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_simple }; /* Trübsal-Zauber */ -static struct curse_type ct_depression = { +const struct curse_type ct_depression = { "depression", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_simple }; /* Astralblock, auf Astralregion */ -static struct curse_type ct_astralblock = { +const struct curse_type ct_astralblock = { "astralblock", CURSETYP_NORM, 0, NO_MERGE, cinfo_simple }; /* Unterhaltungsanteil vermehren */ -static struct curse_type ct_generous = { +const struct curse_type ct_generous = { "generous", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR | M_MAXEFFECT), cinfo_simple }; /* verhindert Attackiere regional */ -static struct curse_type ct_peacezone = { +const struct curse_type ct_peacezone = { "peacezone", CURSETYP_NORM, 0, NO_MERGE, cinfo_simple @@ -248,7 +261,7 @@ static struct curse_type ct_peacezone = { /* erniedigt Magieresistenz von nicht-aliierten Einheiten, wirkt nur 1x * pro Einheit */ -static struct curse_type ct_badmagicresistancezone = { +const struct curse_type ct_badmagicresistancezone = { "badmagicresistancezone", CURSETYP_NORM, 0, NO_MERGE, cinfo_simple @@ -256,25 +269,25 @@ static struct curse_type ct_badmagicresistancezone = { /* erhöht Magieresistenz von aliierten Einheiten, wirkt nur 1x pro * Einheit */ -static struct curse_type ct_goodmagicresistancezone = { +const struct curse_type ct_goodmagicresistancezone = { "goodmagicresistancezone", CURSETYP_NORM, 0, NO_MERGE, cinfo_simple }; -static struct curse_type ct_riotzone = { +const struct curse_type ct_riotzone = { "riotzone", CURSETYP_NORM, 0, (M_DURATION), cinfo_simple }; -static struct curse_type ct_holyground = { +const struct curse_type ct_holyground = { "holyground", CURSETYP_NORM, CURSE_NOAGE, (M_VIGOUR_ADD), cinfo_simple }; -static struct curse_type ct_healing = { +const struct curse_type ct_healing = { "healingzone", CURSETYP_NORM, 0, (M_VIGOUR | M_DURATION), cinfo_simple diff --git a/src/spells/regioncurse.h b/src/spells/regioncurse.h index c261ade41..467c1fb57 100644 --- a/src/spells/regioncurse.h +++ b/src/spells/regioncurse.h @@ -17,10 +17,30 @@ extern "C" { #endif - struct curse; - struct locale; + struct region; + struct curse_type; - extern void register_regioncurse(void); + extern const struct curse_type ct_peacezone; + extern const struct curse_type ct_drought; + extern const struct curse_type ct_blessedharvest; + extern const struct curse_type ct_godcursezone; + extern const struct curse_type ct_gbdream; + extern const struct curse_type ct_healing; + extern const struct curse_type ct_antimagiczone; + extern const struct curse_type ct_depression; + extern const struct curse_type ct_astralblock; + extern const struct curse_type ct_badmagicresistancezone; + extern const struct curse_type ct_goodmagicresistancezone; + extern const struct curse_type ct_holyground; + extern const struct curse_type ct_fogtrap; + extern const struct curse_type ct_magicstreet; + extern const struct curse_type ct_maelstrom; + extern const struct curse_type ct_riotzone; + extern const struct curse_type ct_generous; + extern const struct curse_type ct_badlearn; + + int harvest_effect(const struct region *r); + void register_regioncurse(void); #ifdef __cplusplus } diff --git a/src/spells/shipcurse.c b/src/spells/shipcurse.c index 0295695a1..180c84eea 100644 --- a/src/spells/shipcurse.c +++ b/src/spells/shipcurse.c @@ -72,11 +72,11 @@ static message *cinfo_shipnodrift(const void *obj, objtype_t typ, const curse * return msg_message("curseinfo::shipnodrift_0", "ship id", sh, c->no); } -static struct curse_type ct_stormwind = { "stormwind", -CURSETYP_NORM, CURSE_NOAGE, NO_MERGE, cinfo_ship +const struct curse_type ct_stormwind = { "stormwind", +CURSETYP_NORM, 0, NO_MERGE, cinfo_ship }; -static struct curse_type ct_nodrift = { "nodrift", +const struct curse_type ct_nodrift = { "nodrift", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR), cinfo_shipnodrift }; diff --git a/src/spells/shipcurse.h b/src/spells/shipcurse.h index a41f9edee..8bb98bcbe 100644 --- a/src/spells/shipcurse.h +++ b/src/spells/shipcurse.h @@ -21,8 +21,11 @@ extern "C" { struct message; struct curse; + struct curse_type; extern const struct curse_type ct_shipspeedup; + extern const struct curse_type ct_stormwind; + extern const struct curse_type ct_nodrift; struct message *cinfo_ship(const void *obj, objtype_t typ, const struct curse *c, int self); diff --git a/src/spells/unitcurse.c b/src/spells/unitcurse.c index 0a19629f1..4f8d69f64 100644 --- a/src/spells/unitcurse.c +++ b/src/spells/unitcurse.c @@ -60,14 +60,14 @@ static message *cinfo_auraboost(const void *obj, objtype_t typ, const curse * c, return NULL; } -static struct curse_type ct_auraboost = { +const struct curse_type ct_auraboost = { "auraboost", CURSETYP_NORM, CURSE_SPREADMODULO, (NO_MERGE), cinfo_auraboost }; /* Magic Boost - Gabe des Chaos */ -static struct curse_type ct_magicboost = { +const struct curse_type ct_magicboost = { "magicboost", CURSETYP_UNIT, CURSE_SPREADMODULO | CURSE_IMMUNE, M_MEN, cinfo_simple }; @@ -92,7 +92,7 @@ static message *cinfo_slave(const void *obj, objtype_t typ, const curse * c, return NULL; } -static struct curse_type ct_slavery = { "slavery", +const struct curse_type ct_slavery = { "slavery", CURSETYP_NORM, 0, NO_MERGE, cinfo_slave }; @@ -120,7 +120,7 @@ static message *cinfo_calm(const void *obj, objtype_t typ, const curse * c, return NULL; } -static struct curse_type ct_calmmonster = { +const struct curse_type ct_calmmonster = { "calmmonster", CURSETYP_NORM, CURSE_SPREADNEVER | CURSE_ONLYONE, NO_MERGE, cinfo_calm @@ -144,7 +144,7 @@ static message *cinfo_speed(const void *obj, objtype_t typ, const curse * c, return NULL; } -static struct curse_type ct_speed = { +const struct curse_type ct_speed = { "speed", CURSETYP_UNIT, CURSE_SPREADNEVER, M_MEN, cinfo_speed @@ -168,7 +168,7 @@ message *cinfo_unit(const void *obj, objtype_t typ, const curse * c, int self) return NULL; } -static struct curse_type ct_orcish = { +const struct curse_type ct_orcish = { "orcish", CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN, cinfo_unit @@ -192,7 +192,7 @@ static message *cinfo_kaelteschutz(const void *obj, objtype_t typ, const curse * return NULL; } -static struct curse_type ct_insectfur = { +const struct curse_type ct_insectfur = { "insectfur", CURSETYP_UNIT, CURSE_SPREADMODULO, (M_MEN | M_DURATION), cinfo_kaelteschutz @@ -259,7 +259,7 @@ static message *cinfo_sparkle(const void *obj, objtype_t typ, const curse * c, } } -static struct curse_type ct_sparkle = { "sparkle", +const struct curse_type ct_sparkle = { "sparkle", CURSETYP_UNIT, CURSE_SPREADMODULO, (M_MEN | M_DURATION), cinfo_sparkle }; @@ -275,7 +275,7 @@ CURSETYP_UNIT, CURSE_SPREADMODULO, M_MEN, cinfo_simple /* * C_ALLSKILLS (Alp) */ -static struct curse_type ct_worse = { +const struct curse_type ct_worse = { "worse", CURSETYP_UNIT, CURSE_SPREADMODULO | CURSE_NOAGE, M_MEN, cinfo_unit }; @@ -284,20 +284,20 @@ static struct curse_type ct_worse = { /* * C_ITEMCLOAK */ -static struct curse_type ct_itemcloak = { +const struct curse_type ct_itemcloak = { "itemcloak", CURSETYP_UNIT, CURSE_SPREADNEVER, M_DURATION, cinfo_unit }; /* ------------------------------------------------------------- */ -static struct curse_type ct_fumble = { +const struct curse_type ct_fumble = { "fumble", CURSETYP_NORM, CURSE_SPREADNEVER | CURSE_ONLYONE, NO_MERGE, cinfo_unit }; /* ------------------------------------------------------------- */ -static struct curse_type ct_oldrace = { +const struct curse_type ct_oldrace = { "oldrace", CURSETYP_NORM, CURSE_SPREADALWAYS, NO_MERGE, NULL }; @@ -339,7 +339,7 @@ static message *cinfo_skillmod(const void *obj, objtype_t typ, const curse * c, return NULL; } -static struct curse_type ct_skillmod = { +const struct curse_type ct_skillmod = { "skillmod", CURSETYP_NORM, CURSE_SPREADMODULO, M_MEN, cinfo_skillmod, NULL, read_skill, write_skill }; diff --git a/src/spells/unitcurse.h b/src/spells/unitcurse.h index 15d535cab..681fa32de 100644 --- a/src/spells/unitcurse.h +++ b/src/spells/unitcurse.h @@ -20,8 +20,24 @@ extern "C" { #endif struct curse; + struct curse_type; struct message; - extern struct message *cinfo_unit(const void *obj, objtype_t typ, + + extern const struct curse_type ct_slavery; + extern const struct curse_type ct_calmmonster; + extern const struct curse_type ct_speed; + extern const struct curse_type ct_worse; + extern const struct curse_type ct_skillmod; + extern const struct curse_type ct_oldrace; + extern const struct curse_type ct_fumble; + extern const struct curse_type ct_orcish; + extern const struct curse_type ct_itemcloak; + extern const struct curse_type ct_insectfur; + extern const struct curse_type ct_sparkle; + extern const struct curse_type ct_magicboost; + extern const struct curse_type ct_auraboost; + + struct message *cinfo_unit(const void *obj, objtype_t typ, const struct curse *c, int self); extern void register_unitcurse(void); diff --git a/src/study.c b/src/study.c index 9ea0a860e..ea71b9376 100644 --- a/src/study.c +++ b/src/study.c @@ -28,6 +28,8 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "alchemy.h" #include "academy.h" +#include + #include #include #include @@ -52,6 +54,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include #include + #include /* libc includes */ @@ -163,8 +166,8 @@ static void init_learning(struct attrib *a) static void done_learning(struct attrib *a) { - teaching_info *teach = (teaching_info *)a->data.v; - selist_free(teach->teachers); + teaching_info *teach = (teaching_info *)a->data.v; + selist_free(teach->teachers); free(a->data.v); } @@ -174,15 +177,17 @@ const attrib_type at_learning = { ATF_UNIQUE }; +#define EXPERIENCEDAYS 10 + static int study_days(unit * student, skill_t sk) { - int speed = 30; + int speed = STUDYDAYS; if (u_race(student)->study_speed) { speed += u_race(student)->study_speed[sk]; - if (speed < 30) { + if (speed < STUDYDAYS) { skill *sv = unit_skill(student, sk); if (sv == 0) { - speed = 30; + speed = STUDYDAYS; } } } @@ -209,7 +214,7 @@ teach_unit(unit * teacher, unit * student, int nteaching, skill_t sk, return 0; } - n = 30 * student->number; + n = STUDYDAYS * student->number; a = a_find(student->attribs, &at_learning); if (a != NULL) { teach = (teaching_info *)a->data.v; @@ -231,7 +236,7 @@ teach_unit(unit * teacher, unit * student, int nteaching, skill_t sk, * Student auch in unterschiedlichen Gebaeuden stehen duerfen */ if (academy_can_teach(teacher, student, sk)) { /* Jeder Schueler zusaetzlich +10 Tage wenn in Uni. */ - teach->value += (n / 30) * 10; /* learning erhoehen */ + teach->value += (n / STUDYDAYS) * EXPERIENCEDAYS; /* learning erhoehen */ /* Lehrer zusaetzlich +1 Tag pro Schueler. */ if (academy) { *academy += n; @@ -260,56 +265,56 @@ teach_unit(unit * teacher, unit * student, int nteaching, skill_t sk, * * Ist C aber vor B dran, lehrt C 300 tage an A, und 0 tage an D, * und B lehrt auch 0 tage an A. + * (Na und? -stm) * - * Deswegen darf C D nie lehren duerfen. + * Deswegen darf C D nie lehren duerfen. (Warum? -stm) * * -> Das ist wirr. wer hat das entworfen? * Besser waere, man macht erst vorab alle zuordnungen, und dann * die Talentaenderung (enno). */ - nteaching = MAX(0, nteaching - student->number * 30); + /* FIXME: this code no effect; check if the refactoring done in 1e51d0e9e238e1e6e073cab2060777038e1acfa1 fucked this up */ + nteaching = MAX(0, nteaching - student->number * STUDYDAYS); } return n; } -int teach_cmd(unit * u, struct order *ord) +int teach_cmd(unit * teacher, struct order *ord) { plane *pl; - region *r = u->region; + region *r = teacher->region; skill_t sk_academy = NOSKILL; int teaching, i, j, count, academy = 0; - if (u->region->attribs) { - const curse_type *gbdream_ct = ct_find("gbdream"); - if (gbdream_ct) { - if (get_curse(u->region->attribs, gbdream_ct)) { - ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "gbdream_noteach", "")); - return 0; - } + if (r->attribs) { + if (get_curse(r->attribs, &ct_gbdream)) { + ADDMSG(&teacher->faction->msgs, + msg_feedback(teacher, ord, "gbdream_noteach", "")); + return 0; } } - if ((u_race(u)->flags & RCF_NOTEACH) || fval(u, UFL_WERE)) { - cmistake(u, ord, 274, MSG_EVENT); + if ((u_race(teacher)->flags & RCF_NOTEACH) || fval(teacher, UFL_WERE)) { + cmistake(teacher, ord, 274, MSG_EVENT); return 0; } pl = rplane(r); if (pl && fval(pl, PFL_NOTEACH)) { - cmistake(u, ord, 273, MSG_EVENT); + cmistake(teacher, ord, 273, MSG_EVENT); return 0; } - teaching = u->number * 30 * TEACHNUMBER; + teaching = teacher->number * STUDYDAYS * TEACHNUMBER; - if ((i = get_effect(u, oldpotiontype[P_FOOL])) > 0) { /* Trank "Dumpfbackenbrot" */ - i = MIN(i, u->number * TEACHNUMBER); + if ((i = get_effect(teacher, oldpotiontype[P_FOOL])) > 0) { /* Trank "Dumpfbackenbrot" */ + i = MIN(i, teacher->number * TEACHNUMBER); /* Trank wirkt pro Schueler, nicht pro Lehrer */ - teaching -= i * 30; - change_effect(u, oldpotiontype[P_FOOL], -i); - j = teaching / 30; - ADDMSG(&u->faction->msgs, msg_message("teachdumb", "teacher amount", u, j)); + teaching -= i * STUDYDAYS; + change_effect(teacher, oldpotiontype[P_FOOL], -i); + j = teaching / STUDYDAYS; + ADDMSG(&teacher->faction->msgs, msg_message("teachdumb", "teacher amount", teacher, j)); } if (teaching == 0) return 0; @@ -319,22 +324,22 @@ int teach_cmd(unit * u, struct order *ord) init_order(ord); #if TEACH_ALL - if (getparam(u->faction->locale) == P_ANY) { + if (getparam(teacher->faction->locale) == P_ANY) { skill_t sk; unit *student; skill_t teachskill[MAXSKILLS]; int t = 0; do { - sk = getskill(u->faction->locale); - teachskill[t] = getskill(u->faction->locale); + sk = getskill(teacher->faction->locale); + teachskill[t] = getskill(teacher->faction->locale); } while (sk != NOSKILL); for (student = r->units; teaching && student; student = student->next) { if (LongHunger(student)) { continue; } - else if (student->faction == u->faction) { + else if (student->faction == teacher->faction) { if (getkeyword(student->thisorder) == K_STUDY) { /* Input ist nun von student->thisorder !! */ init_order(student->thisorder); @@ -348,20 +353,20 @@ int teach_cmd(unit * u, struct order *ord) sk = teachskill[t]; } if (sk != NOSKILL - && effskill_study(u, sk, 0) - TEACHDIFFERENCE > effskill_study(student, sk, 0)) { - teaching -= teach_unit(u, student, teaching, sk, true, &academy); + && effskill_study(teacher, sk, 0) - TEACHDIFFERENCE > effskill_study(student, sk, 0)) { + teaching -= teach_unit(teacher, student, teaching, sk, true, &academy); } } } #ifdef TEACH_FRIENDS - else if (alliedunit(u, student->faction, HELP_GUARD)) { + else if (alliedunit(teacher, student->faction, HELP_GUARD)) { if (getkeyword(student->thisorder) == K_STUDY) { /* Input ist nun von student->thisorder !! */ init_order(student->thisorder); sk = getskill(student->faction->locale); if (sk != NOSKILL - && effskill_study(u, sk, 0) - TEACHDIFFERENCE >= effskill(student, sk, 0)) { - teaching -= teach_unit(u, student, teaching, sk, true, &academy); + && effskill_study(teacher, sk, 0) - TEACHDIFFERENCE >= effskill(student, sk, 0)) { + teaching -= teach_unit(teacher, student, teaching, sk, true, &academy); } } } @@ -380,15 +385,15 @@ int teach_cmd(unit * u, struct order *ord) while (!parser_end()) { skill_t sk; - unit *u2; + unit *student; bool feedback; - getunit(r, u->faction, &u2); + getunit(r, teacher->faction, &student); ++count; /* Falls die Unit nicht gefunden wird, Fehler melden */ - if (!u2) { + if (!student) { char tbuf[20]; const char *uid; const char *token; @@ -398,30 +403,30 @@ int teach_cmd(unit * u, struct order *ord) for (j = 0; j != count - 1; ++j) { /* skip over the first 'count' units */ - getunit(r, u->faction, NULL); + getunit(r, teacher->faction, NULL); } token = getstrtoken(); /* Beginne die Fehlermeldung */ - if (isparam(token, u->faction->locale, P_TEMP)) { + if (isparam(token, teacher->faction->locale, P_TEMP)) { token = getstrtoken(); - sprintf(tbuf, "%s %s", LOC(u->faction->locale, + sprintf(tbuf, "%s %s", LOC(teacher->faction->locale, parameters[P_TEMP]), token); uid = tbuf; } else { uid = token; } - ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "unitnotfound_id", + ADDMSG(&teacher->faction->msgs, msg_feedback(teacher, ord, "unitnotfound_id", "id", uid)); parser_popstate(); continue; } - feedback = u->faction == u2->faction - || alliedunit(u2, u->faction, HELP_GUARD); + feedback = teacher->faction == student->faction + || alliedunit(student, teacher->faction, HELP_GUARD); /* Neuen Befehl zusammenbauen. TEMP-Einheiten werden automatisch in * ihre neuen Nummern uebersetzt. */ @@ -429,58 +434,58 @@ int teach_cmd(unit * u, struct order *ord) strncat(zOrder, " ", sz - 1); --sz; } - sz -= strlcpy(zOrder + 4096 - sz, itoa36(u2->no), sz); + sz -= strlcpy(zOrder + 4096 - sz, itoa36(student->no), sz); - if (getkeyword(u2->thisorder) != K_STUDY) { - ADDMSG(&u->faction->msgs, - msg_feedback(u, ord, "teach_nolearn", "student", u2)); + if (getkeyword(student->thisorder) != K_STUDY) { + ADDMSG(&teacher->faction->msgs, + msg_feedback(teacher, ord, "teach_nolearn", "student", student)); continue; } - /* Input ist nun von u2->thisorder !! */ + /* Input ist nun von student->thisorder !! */ parser_pushstate(); - init_order(u2->thisorder); - sk = getskill(u2->faction->locale); + init_order(student->thisorder); + sk = getskill(student->faction->locale); parser_popstate(); if (sk == NOSKILL) { - ADDMSG(&u->faction->msgs, - msg_feedback(u, ord, "teach_nolearn", "student", u2)); + ADDMSG(&teacher->faction->msgs, + msg_feedback(teacher, ord, "teach_nolearn", "student", student)); continue; } - /* u is teacher, u2 is student */ - if (effskill_study(u2, sk, 0) > effskill_study(u, sk, 0) + if (effskill_study(student, sk, 0) > effskill_study(teacher, sk, 0) - TEACHDIFFERENCE) { if (feedback) { - ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "teach_asgood", - "student", u2)); + ADDMSG(&teacher->faction->msgs, msg_feedback(teacher, ord, "teach_asgood", + "student", student)); } continue; } if (sk == SK_MAGIC) { /* ist der Magier schon spezialisiert, so versteht er nur noch * Lehrer seines Gebietes */ - sc_mage *mage1 = get_mage(u); - sc_mage *mage2 = get_mage(u2); + sc_mage *mage1 = get_mage(teacher); + sc_mage *mage2 = get_mage(student); if (mage2 && mage1 && mage2->magietyp != M_GRAY && mage1->magietyp != mage2->magietyp) { if (feedback) { - ADDMSG(&u->faction->msgs, msg_feedback(u, ord, - "error_different_magic", "target", u2)); + ADDMSG(&teacher->faction->msgs, msg_feedback(teacher, ord, + "error_different_magic", "target", student)); } continue; } } sk_academy = sk; - teaching -= teach_unit(u, u2, teaching, sk, false, &academy); + teaching -= teach_unit(teacher, student, teaching, sk, false, &academy); } - new_order = create_order(K_TEACH, u->faction->locale, "%s", zOrder); - replace_order(&u->orders, ord, new_order); + new_order = create_order(K_TEACH, teacher->faction->locale, "%s", zOrder); + replace_order(&teacher->orders, ord, new_order); free_order(new_order); /* parse_order & set_order have each increased the refcount */ } if (academy && sk_academy!=NOSKILL) { - academy_teaching_bonus(u, sk_academy, academy); + /* assert(academy % STUDYDAYS == 0); bug 2355: why? */ + academy_teaching_bonus(teacher, sk_academy, academy); } return 0; } @@ -747,12 +752,12 @@ int study_cmd(unit * u, order * ord) if (get_effect(u, oldpotiontype[P_WISE])) { l = MIN(u->number, get_effect(u, oldpotiontype[P_WISE])); - teach->value += l * 10; + teach->value += l * EXPERIENCEDAYS; change_effect(u, oldpotiontype[P_WISE], -l); } if (get_effect(u, oldpotiontype[P_FOOL])) { l = MIN(u->number, get_effect(u, oldpotiontype[P_FOOL])); - teach->value -= l * 30; + teach->value -= l * STUDYDAYS; change_effect(u, oldpotiontype[P_FOOL], -l); } @@ -761,11 +766,11 @@ int study_cmd(unit * u, order * ord) /* p ist Kosten ohne Uni, studycost mit; wenn * p!=studycost, ist die Einheit zwangsweise * in einer Uni */ - teach->value += u->number * 10; + teach->value += u->number * EXPERIENCEDAYS; } - if (is_cursed(r->attribs, C_BADLEARN, 0)) { - teach->value -= u->number * 10; + if (is_cursed(r->attribs, &ct_badlearn)) { + teach->value -= u->number * EXPERIENCEDAYS; } multi *= study_speedup(u, sk, speed_rule); @@ -828,7 +833,7 @@ int study_cmd(unit * u, order * ord) static int produceexp_days(void) { static int config, rule; if (config_changed(&config)) { - rule = config_get_int("study.produceexp", 10); + rule = config_get_int("study.produceexp", EXPERIENCEDAYS); } return rule; } @@ -838,7 +843,7 @@ void produceexp_ex(struct unit *u, skill_t sk, int n, learn_fun learn) assert(u && n <= u->number); if (n > 0 && (is_monsters(u->faction) || playerrace(u_race(u)))) { int days = produceexp_days(); - learn(u, sk, days * n / u->number); + learn(u, sk, days * n); } } @@ -854,6 +859,7 @@ void inject_learn(learn_fun fun) { inject_learn_fun = fun; } #endif +/** days should be scaled by u->number; STUDYDAYS * u->number is one week worth of learning */ void learn_skill(unit *u, skill_t sk, int days) { int leveldays = STUDYDAYS * u->number; int weeks = 0; @@ -871,15 +877,24 @@ void learn_skill(unit *u, skill_t sk, int days) { ++weeks; } if (weeks > 0) { - skill *sv = unit_skill(u, sk); - if (!sv) { - sv = add_skill(u, sk); - } - while (sv->weeks <= weeks) { - weeks -= sv->weeks; - sk_set(sv, sv->level + 1); + increase_skill(u, sk, weeks); + } +} + +void reduce_skill_days(unit *u, skill_t sk, int days) { + skill *sv = unit_skill(u, sk); + if (sv) { + while (days > 0) { + if (days >= STUDYDAYS * u->number) { + reduce_skill(u, sv, 1); + days -= STUDYDAYS; + } + else { + if (chance (days / ((double) STUDYDAYS * u->number))) /* (rng_int() % (30 * u->number) < days)*/ + reduce_skill(u, sv, 1); + days = 0; + } } - sv->weeks -= weeks; } } @@ -916,7 +931,7 @@ void demon_skillchange(unit *u) } } else { - learn_skill(u, sv->id, STUDYDAYS*weeks); + learn_skill(u, sv->id, STUDYDAYS * u->number * weeks); } } ++sv; diff --git a/src/study.h b/src/study.h index 0f8ae70f5..83bd65a40 100644 --- a/src/study.h +++ b/src/study.h @@ -40,6 +40,7 @@ extern "C" { #define STUDYDAYS 30 void learn_skill(struct unit *u, skill_t sk, int days); + void reduce_skill_days(struct unit *u, skill_t sk, int days); void produceexp(struct unit *u, skill_t sk, int n); void produceexp_ex(struct unit *u, skill_t sk, int n, learn_fun learn); diff --git a/src/study.test.c b/src/study.test.c index 74104704b..c0b58b493 100644 --- a/src/study.test.c +++ b/src/study.test.c @@ -184,12 +184,12 @@ static CuTest *g_tc; static void cb_learn_one(unit *u, skill_t sk, int days) { CuAssertIntEquals(g_tc, SK_ALCHEMY, sk); - CuAssertIntEquals(g_tc, 10, days); + CuAssertIntEquals(g_tc, 20, days); } static void cb_learn_two(unit *u, skill_t sk, int days) { CuAssertIntEquals(g_tc, SK_ALCHEMY, sk); - CuAssertIntEquals(g_tc, 20, days); + CuAssertIntEquals(g_tc, 40, days); } static void test_produceexp(CuTest *tc) { @@ -247,7 +247,7 @@ static void test_academy_building(CuTest *tc) { CuAssertPtrEquals(tc, u, log_learners[0].u); CuAssertIntEquals(tc, SK_CROSSBOW, log_learners[0].sk); - CuAssertIntEquals(tc, 15, log_learners[0].days); + CuAssertIntEquals(tc, u1->number, log_learners[0].days); test_cleanup(); } @@ -377,7 +377,7 @@ static void test_study_cost(CuTest *tc) { learn_reset(); CuAssertPtrEquals(tc, u, log_learners[0].u); CuAssertIntEquals(tc, SK_ALCHEMY, log_learners[0].sk); - CuAssertIntEquals(tc, STUDYDAYS*u->number, log_learners[0].days); + CuAssertIntEquals(tc, STUDYDAYS * u->number, log_learners[0].days); CuAssertIntEquals(tc, 0, i_get(u->items, itype)); test_cleanup(); } @@ -405,7 +405,7 @@ static void test_teach_magic(CuTest *tc) { learn_reset(); CuAssertPtrEquals(tc, u, log_learners[0].u); CuAssertIntEquals(tc, SK_MAGIC, log_learners[0].sk); - CuAssertIntEquals(tc, STUDYDAYS*2, log_learners[0].days); + CuAssertIntEquals(tc, STUDYDAYS * 2, log_learners[0].days); CuAssertIntEquals(tc, 0, i_get(u->items, itype)); test_cleanup(); } @@ -426,7 +426,7 @@ static void test_teach_cmd(CuTest *tc) { learn_reset(); CuAssertPtrEquals(tc, u, log_learners[0].u); CuAssertIntEquals(tc, SK_CROSSBOW, log_learners[0].sk); - CuAssertIntEquals(tc, STUDYDAYS*2*u->number, log_learners[0].days); + CuAssertIntEquals(tc, STUDYDAYS * 2 * u->number, log_learners[0].days); test_cleanup(); } diff --git a/src/tests.c b/src/tests.c index ec2ac2a2f..cfcd6d0ca 100644 --- a/src/tests.c +++ b/src/tests.c @@ -42,7 +42,7 @@ struct race *test_create_race(const char *name) rc->maintenance = 10; rc->hitpoints = 20; rc->maxaura = 100; - rc->ec_flags |= GETITEM; + rc->ec_flags |= ECF_GETITEM; rc->battle_flags = BF_EQUIPMENT; return rc; } diff --git a/src/upkeep.c b/src/upkeep.c index 8494d7f25..d7626084d 100644 --- a/src/upkeep.c +++ b/src/upkeep.c @@ -38,11 +38,6 @@ int lifestyle(const unit * u) return need; } -static bool help_money(const unit * u) -{ - return !(u_race(u)->ec_flags & ECF_KEEP_ITEM); -} - static void help_feed(unit * donor, unit * u, int *need_p) { int need = *need_p; @@ -172,7 +167,7 @@ void get_food(region * r) unit *v; for (v = r->units; need && v; v = v->next) { - if (v->faction == u->faction && help_money(v)) { + if (v->faction == u->faction) { int give = get_money(v) - lifestyle(v); give = MIN(need, give); if (give > 0) { @@ -191,6 +186,7 @@ void get_food(region * r) int need = lifestyle(u); faction *f = u->faction; + assert(u->hp > 0); need -= MAX(0, get_money(u)); if (need > 0) { @@ -201,8 +197,7 @@ void get_food(region * r) faction *owner = region_get_owner(r); if (owner && owner != u->faction) { for (v = r->units; v; v = v->next) { - if (v->faction == owner && alliedunit(v, f, HELP_MONEY) - && help_money(v)) { + if (v->faction == owner && alliedunit(v, f, HELP_MONEY)) { help_feed(v, u, &need); break; } @@ -210,8 +205,7 @@ void get_food(region * r) } } for (v = r->units; need && v; v = v->next) { - if (v->faction != f && alliedunit(v, f, HELP_MONEY) - && help_money(v)) { + if (v->faction != f && alliedunit(v, f, HELP_MONEY)) { help_feed(v, u, &need); } } diff --git a/src/util/gamedata.h b/src/util/gamedata.h index 452a6e527..97f8416b3 100644 --- a/src/util/gamedata.h +++ b/src/util/gamedata.h @@ -34,10 +34,12 @@ #define NOWATCH_VERSION 354 /* plane->watchers is gone */ #define KEYVAL_VERSION 355 /* at_keys has values */ #define NOLANDITEM_VERSION 356 /* land_region has no items */ +#define NORCSPELL_VERSION 357 /* data contains no RC_SPELL units */ +#define SORTKEYS_VERSION 358 /* at_keys is sorted */ /* unfinished: */ #define CRYPT_VERSION 400 /* passwords are encrypted */ -#define RELEASE_VERSION NOLANDITEM_VERSION /* current datafile */ +#define RELEASE_VERSION SORTKEYS_VERSION /* current datafile */ #define MIN_VERSION INTPAK_VERSION /* minimal datafile we support */ #define MAX_VERSION RELEASE_VERSION /* change this if we can need to read the future datafile, and we can do so */ diff --git a/src/util/goodies.c b/src/util/goodies.c index c478bcf83..0794240d5 100644 --- a/src/util/goodies.c +++ b/src/util/goodies.c @@ -82,12 +82,18 @@ static int spc_email_isvalid(const char *address) if (strchr(rfc822_specials, *c)) return 0; } + if (*c!='@') { + /* no @ symbol */ + return -1; + } + domain = ++c; + if (!*c) { + return -1; + } if (c == address || *(c - 1) == '.') return 0; /* next we validate the domain portion (name@domain) */ - if (!*(domain = ++c)) - return 0; do { if (*c == '.') { if (c == domain || *(c - 1) == '.') diff --git a/src/volcano.c b/src/volcano.c index 9734bb2cd..db7166de0 100644 --- a/src/volcano.c +++ b/src/volcano.c @@ -82,7 +82,7 @@ damage_unit(unit * u, const char *dam, bool physical, bool magic) int i, dead = 0, hp_rem = 0, heiltrank; assert(u->number); - if (fval(u_race(u), RCF_ILLUSIONARY) || u_race(u) == get_race(RC_SPELL)) { + if (fval(u_race(u), RCF_ILLUSIONARY)) { return 0; } diff --git a/tests/runtests.bat b/tests/runtests.bat index b6f70768f..95ac17be4 100644 --- a/tests/runtests.bat +++ b/tests/runtests.bat @@ -3,6 +3,7 @@ IF EXIST ..\build-vs10 SET BUILD=..\build-vs10\eressea\Debug IF EXIST ..\build-vs11 SET BUILD=..\build-vs11\eressea\Debug IF EXIST ..\build-vs12 SET BUILD=..\build-vs12\eressea\Debug IF EXIST ..\build-vs14 SET BUILD=..\build-vs14\eressea\Debug +REM IF EXIST ..\build-vs15 SET BUILD=..\build-vs15\eressea\Debug SET SERVER=%BUILD%\eressea.exe %BUILD%\test_eressea.exe %SERVER% ..\scripts\run-tests.lua