From 1789bc06e9aa50593aead336101e89092815b410 Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Mon, 16 Nov 2015 18:36:14 +0100 Subject: [PATCH 01/11] monster faction may learn from experience --- src/kernel/unit.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/kernel/unit.c b/src/kernel/unit.c index 493a3a66f..339e5693a 100644 --- a/src/kernel/unit.c +++ b/src/kernel/unit.c @@ -27,6 +27,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include "curse.h" #include "item.h" #include "move.h" +#include "monster.h" #include "order.h" #include "plane.h" #include "race.h" @@ -1945,7 +1946,7 @@ static double produceexp_chance(void) { void produceexp_ex(struct unit *u, skill_t sk, int n, bool (*learn)(unit *, skill_t, double)) { - if (n != 0 && playerrace(u_race(u))) { + if (n != 0 && (is_monsters(u->faction) || playerrace(u_race(u)))) { double chance = produceexp_chance(); if (chance > 0.0F) { learn(u, sk, (n * chance) / u->number); From 8cec4b20e0287480fbd226eaa6aaa8d1488eabdf Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Mon, 16 Nov 2015 18:41:52 +0100 Subject: [PATCH 02/11] reduce code obfuscation --- src/kernel/unit.c | 6 ++++-- src/laws.test.c | 2 +- src/monsters.c | 8 +------- 3 files changed, 6 insertions(+), 10 deletions(-) diff --git a/src/kernel/unit.c b/src/kernel/unit.c index 339e5693a..fdc7d431b 100644 --- a/src/kernel/unit.c +++ b/src/kernel/unit.c @@ -55,6 +55,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #include #include +#include #include #include #include @@ -1161,10 +1162,11 @@ void set_number(unit * u, int count) u->number = (unsigned short)count; } -bool learn_skill(unit * u, skill_t sk, double chance) +bool learn_skill(unit * u, skill_t sk, double learn_chance) { skill *sv = u->skills; - if (chance < 1.0 && rng_int() % 10000 >= chance * 10000) + if (learn_chance < 1.0 && rng_int() % 10000 >= learn_chance * 10000) + if (!chance(learn_chance)) return false; while (sv != u->skills + u->skill_size) { assert(sv->weeks > 0); diff --git a/src/laws.test.c b/src/laws.test.c index e829eff27..c070f958f 100644 --- a/src/laws.test.c +++ b/src/laws.test.c @@ -655,7 +655,7 @@ static void test_unarmed_races_can_guard(CuTest *tc) { setup_guard(&fix, false); rc = rc_get_or_create(fix.u->_race->_name); - rc->flags |= RCF_UNARMEDGUARD; + fset(rc, RCF_UNARMEDGUARD); update_guards(); CuAssertTrue(tc, fval(fix.u, UFL_GUARD)); test_cleanup(); diff --git a/src/monsters.c b/src/monsters.c index 0ab88fa61..edac0980e 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -766,8 +766,7 @@ void plan_monsters(faction * f) for (r = regions; r; r = r->next) { unit *u; - double rchance = attack_chance; - bool attacking = false; + bool attacking = chance(attack_chance); for (u = r->units; u; u = u->next) { attrib *ta; @@ -785,11 +784,6 @@ void plan_monsters(faction * f) produceexp(u, SK_PERCEPTION, u->number); } - if (rchance > 0.0) { - if (chance(rchance)) - attacking = true; - rchance = 0.0; - } if (u->status > ST_BEHIND) { setstatus(u, ST_FIGHT); /* all monsters fight */ From 069303d0ae10b7b8e1ca348e5a3587fa91548017 Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Mon, 16 Nov 2015 19:45:56 +0100 Subject: [PATCH 03/11] slightly improved monster code change normalvariate to more efficient dice based method try harder to learn a useful skill --- src/monsters.c | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/monsters.c b/src/monsters.c index edac0980e..4de62782c 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -399,10 +399,10 @@ static int dragon_affinity_value(region * r, unit * u) int m = all_money(r, u->faction); if (u_race(u) == get_race(RC_FIREDRAGON)) { - return (int)(normalvariate(m, m / 2)); + return dice(6, m / 6); } else { - return (int)(normalvariate(m, m / 4)); + return dice(4, m / 8); } } @@ -745,9 +745,10 @@ static order *plan_dragon(unit * u) } } if (long_order == NULL) { + int attempts = 0; skill_t sk = SK_PERCEPTION; /* study perception (or a random useful skill) */ - while (!skill_enabled(sk) || u_race(u)->bonus[sk] < -5) { + while ((!skill_enabled(sk) || (attempts < MAXSKILLS && u_race(u)->bonus[sk] < (++attempts < 10?1:-5 )))) { sk = (skill_t)(rng_int() % MAXSKILLS); } long_order = create_order(K_STUDY, u->faction->locale, "'%s'", From 384a2ea7d74383af23b84a6e8e4aa825b1aab173 Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Mon, 16 Nov 2015 19:28:24 +0100 Subject: [PATCH 04/11] remove duplicate monster_attacks code --- src/monsters.c | 58 ++++++++++++++++++++++---------------------------- 1 file changed, 26 insertions(+), 32 deletions(-) diff --git a/src/monsters.c b/src/monsters.c index 4de62782c..98227c787 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -157,9 +157,30 @@ static order *monster_attack(unit * u, const unit * target) return create_order(K_ATTACK, u->faction->locale, "%i", target->no); } +static int monster_attacks(unit * monster, bool respect_buildings, bool rich_only) +{ + region *r = monster->region; + unit *u2; + int money = 0; + + for (u2 = r->units; u2; u2 = u2->next) { + if (u2->faction != monster->faction && cansee(monster->faction, r, u2, 0) && !in_safe_building(u2, monster)) { + int m = get_money(u2); + if (!rich_only || m > 0) { + order *ord = monster_attack(monster, u2); + if (ord) { + addlist(&monster->orders, ord); + money += m; + } + } + } + } + return money; +} + static order *get_money_for_dragon(region * r, unit * u, int wanted) { - int n; + int money; bool attacks = monster_attack_chance() > 0.0; /* falls genug geld in der region ist, treiben wir steuern ein. */ @@ -173,26 +194,14 @@ static order *get_money_for_dragon(region * r, unit * u, int wanted) /* falls der drache launisch ist, oder das regionssilber knapp, greift er alle an * und holt sich Silber von Einheiten, vorausgesetzt er bewacht bereits */ - n = 0; + money = 0; if (attacks && is_guard(u, GUARD_TAX)) { - unit *u2; - for (u2 = r->units; u2; u2 = u2->next) { - if (u2->faction != u->faction && cansee(u->faction, r, u2, 0) && !in_safe_building(u, u2)) { - int m = get_money(u2); - if (m != 0) { - order *ord = monster_attack(u, u2); - if (ord) { - addlist(&u->orders, ord); - n += m; - } - } - } - } + money += monster_attacks(u, true, true); } /* falls die einnahmen erreicht werden, bleibt das monster noch eine */ /* runde hier. */ - if (n + rmoney(r) >= wanted) { + if (money + rmoney(r) >= wanted) { return create_order(K_LOOT, default_locale, NULL); } @@ -538,21 +547,6 @@ static order *monster_seeks_target(region * r, unit * u) } #endif -static void monster_attacks(unit * monster) -{ - region *r = monster->region; - unit *u; - - for (u = r->units; u; u = u->next) { - if (u->faction != monster->faction && cansee(monster->faction, r, u, 0) && !in_safe_building(u, monster)) { - order *ord = monster_attack(monster, u); - if (ord) { - addlist(&monster->orders, ord); - } - } - } -} - static const char *random_growl(void) { switch (rng_int() % 5) { @@ -790,7 +784,7 @@ void plan_monsters(faction * f) /* all monsters fight */ } if (attacking && is_guard(u, GUARD_TAX)) { - monster_attacks(u); + monster_attacks(u, true, false); } /* units with a plan to kill get ATTACK orders: */ ta = a_find(u->attribs, &at_hate); From a366cd4b17ae855e07258e58712c86c652ba3c3a Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Tue, 17 Nov 2015 02:07:46 +0100 Subject: [PATCH 05/11] tests for plan_monster --- src/CMakeLists.txt | 1 + src/monsters.c | 11 +- src/monsters.test.c | 247 ++++++++++++++++++++++++++++++++++++++++++++ src/test_eressea.c | 1 + src/tests.c | 4 +- 5 files changed, 259 insertions(+), 5 deletions(-) create mode 100644 src/monsters.test.c diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index ada49aad9..a9e969eda 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -200,6 +200,7 @@ set(TESTS_SRC laws.test.c magic.test.c market.test.c + monsters.test.c move.test.c piracy.test.c prefix.test.c diff --git a/src/monsters.c b/src/monsters.c index 98227c787..6af672133 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -71,7 +71,7 @@ #include #include -#define MOVECHANCE 25 /* chance fuer bewegung */ +#define MOVECHANCE .25 /* chance fuer bewegung */ #define DRAGON_RANGE 20 /* Max. Distanz zum nächsten Drachenziel */ #define MAXILLUSION_TEXTS 3 @@ -85,6 +85,10 @@ static double monster_attack_chance(void) { return get_param_flt(global.parameters, "rules.monsters.attack_chance", 0.4f); } +static double random_move_chance(void) { + return get_param_flt(global.parameters, "rules.monsters.random_move_chance", MOVECHANCE); +} + static void reduce_weight(unit * u) { int capacity, weight = 0; @@ -157,7 +161,7 @@ static order *monster_attack(unit * u, const unit * target) return create_order(K_ATTACK, u->faction->locale, "%i", target->no); } -static int monster_attacks(unit * monster, bool respect_buildings, bool rich_only) +int monster_attacks(unit * monster, bool respect_buildings, bool rich_only) { region *r = monster->region; unit *u2; @@ -786,6 +790,7 @@ void plan_monsters(faction * f) if (attacking && is_guard(u, GUARD_TAX)) { monster_attacks(u, true, false); } + /* units with a plan to kill get ATTACK orders: */ ta = a_find(u->attribs, &at_hate); if (ta && !monster_is_waiting(u)) { @@ -818,7 +823,7 @@ void plan_monsters(faction * f) } } else if (u_race(u)->flags & RCF_MOVERANDOM) { - if (rng_int() % 100 < MOVECHANCE || check_overpopulated(u)) { + if (chance(random_move_chance()) || check_overpopulated(u)) { long_order = monster_move(r, u); } } diff --git a/src/monsters.test.c b/src/monsters.test.c new file mode 100644 index 000000000..d19633277 --- /dev/null +++ b/src/monsters.test.c @@ -0,0 +1,247 @@ +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include + +#include "monster.h" +#include "guard.h" +#include "skill.h" + +#include +#include + +#include +#include +#include +#include +#include + +extern void plan_monsters(struct faction *f); +extern int monster_attacks(unit * monster, bool respect_buildings, bool rich_only); + +static void init_language(void) +{ + locale* lang; + int i; + + lang = get_or_create_locale("de"); + locale_setstring(lang, "skill::unarmed", "Waffenloser Kampf"); + locale_setstring(lang, "keyword::attack", "ATTACKIERE"); + locale_setstring(lang, "keyword::study", "LERNE"); + locale_setstring(lang, "keyword::tax", "TREIBE"); + locale_setstring(lang, "keyword::loot", "PLUENDERE"); + locale_setstring(lang, "keyword::guard", "BEWACHE"); + locale_setstring(lang, "keyword::move", "NACH"); + locale_setstring(lang, "keyword::message", "BOTSCHAFT"); + locale_setstring(lang, "REGION", "REGION"); + locale_setstring(lang, "east", "O"); + + for (i = 0; i < MAXKEYWORDS; ++i) { + if (!locale_getstring(lang, mkname("keyword", keywords[i]))) + locale_setstring(lang, mkname("keyword", keywords[i]), keywords[i]); + } + for (i = 0; i < MAXSKILLS; ++i) { + if (!locale_getstring(lang, mkname("skill", skillnames[i]))) + locale_setstring(lang, mkname("skill", skillnames[i]), skillnames[i]); + } + init_keywords(lang); + init_skills(lang); +} + +static order *find_order(const char *expected, const unit *unit) +{ + char cmd[32]; + order *order; + for (order = unit->orders; order; order = order->next) { + if (strcmp(expected, get_command(order, cmd, sizeof(cmd))) == 0) { + return order; + } + } + return NULL; +} + +static void create_monsters(faction **player, faction **monsters, region **r, unit **u, unit **m) { + race* rc; + + test_cleanup(); + + init_language(); + + test_create_world(); + *player = test_create_faction(NULL); + *monsters = get_or_create_monsters(); + assert(rc_find((*monsters)->race->_name)); + rc = rc_get_or_create((*monsters)->race->_name); + fset(rc, RCF_UNARMEDGUARD); + fset(rc, RCF_NPC); + fset(*monsters, FFL_NOIDLEOUT); + assert(fval(*monsters, FFL_NPC) && fval((*monsters)->race, RCF_UNARMEDGUARD) && fval((*monsters)->race, RCF_NPC) && fval(*monsters, FFL_NOIDLEOUT)); + (*monsters)->locale = default_locale; + + *r = findregion(0, 0); + + *u = test_create_unit(*player, *r); + unit_setid(*u, 1); + *m = test_create_unit(*monsters, *r); +} + +static void test_monsters_attack(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + + create_monsters(&f, &f2, &r, &u, &m); + + guard(m, GUARD_TAX); + + set_param(&global.parameters, "rules.monsters.attack_chance", "1"); + + plan_monsters(f2); + + CuAssertPtrNotNull(tc, find_order("ATTACKIERE 1", m)); + test_cleanup(); +} + +static void test_monsters_attack_ocean(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + + create_monsters(&f, &f2, &r, &u, &m); + r = findregion(-1, 0); + u = test_create_unit(u->faction, r); + unit_setid(u, 2); + m = test_create_unit(m->faction, r); + assert(!m->region->land); + + set_param(&global.parameters, "rules.monsters.attack_chance", "1"); + + plan_monsters(f2); + + CuAssertPtrNotNull(tc, find_order("ATTACKIERE 2", m)); + test_cleanup(); +} + +static void test_monsters_waiting(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + + create_monsters(&f, &f2, &r, &u, &m); + guard(m, GUARD_TAX); + fset(m, UFL_ISNEW); + monster_attacks(m, false, false); + CuAssertPtrEquals(tc, 0, find_order("ATTACKIERE 1", m)); + test_cleanup(); +} + +static void test_monsters_attack_not(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + + create_monsters(&f, &f2, &r, &u, &m); + + guard(m, GUARD_TAX); + guard(u, GUARD_TAX); + + set_param(&global.parameters, "rules.monsters.attack_chance", "0"); + + plan_monsters(f2); + + CuAssertPtrEquals(tc, 0, find_order("ATTACKIERE 1", m)); + test_cleanup(); +} + +static void test_dragon_attacks_the_rich(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + const item_type *i_silver; + + init_language(); + create_monsters(&f, &f2, &r, &u, &m); + + guard(m, GUARD_TAX); + set_level(m, SK_WEAPONLESS, 10); + + rsetmoney(r, 1); + rsetmoney(findregion(1, 0), 0); + i_silver = it_find("money"); + assert(i_silver); + i_change(&u->items, i_silver, 5000); + + set_param(&global.parameters, "rules.monsters.attack_chance", "0.00001"); + + plan_monsters(f2); + + CuAssertPtrNotNull(tc, find_order("ATTACKIERE 1", m)); + CuAssertPtrNotNull(tc, find_order("PLUENDERE", m)); + test_cleanup(); +} + +static void test_dragon_moves(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + + create_monsters(&f, &f2, &r, &u, &m); + rsetpeasants(r, 0); + rsetmoney(r, 0); + rsetmoney(findregion(1, 0), 1000); + + set_level(m, SK_WEAPONLESS, 10); + set_param(&global.parameters, "rules.monsters.attack_chance", ".0"); + plan_monsters(f2); + + CuAssertPtrNotNull(tc, find_order("NACH O", m)); + test_cleanup(); +} + +static void test_monsters_learn_exp(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + skill* sk; + + create_monsters(&f, &f2, &r, &u, &m); + set_param(&global.parameters, "study.from_use", "1"); + + u_setrace(u, u_race(m)); + produceexp(u, SK_MELEE, u->number); + sk = unit_skill(u, SK_MELEE); + CuAssertTrue(tc, !sk); + + produceexp(m, SK_MELEE, u->number); + sk = unit_skill(m, SK_MELEE); + CuAssertTrue(tc, sk && (sk->level > 0 || (sk->level == 0 && sk->weeks > 0))); + + test_cleanup(); +} + +CuSuite *get_monsters_suite(void) +{ + CuSuite *suite = CuSuiteNew(); + SUITE_ADD_TEST(suite, test_monsters_attack); + SUITE_ADD_TEST(suite, test_monsters_attack_ocean); + SUITE_ADD_TEST(suite, test_monsters_waiting); + SUITE_ADD_TEST(suite, test_monsters_attack_not); + SUITE_ADD_TEST(suite, test_dragon_attacks_the_rich); + SUITE_ADD_TEST(suite, test_dragon_moves); + SUITE_ADD_TEST(suite, test_monsters_learn_exp); + return suite; +} diff --git a/src/test_eressea.c b/src/test_eressea.c index 8f5c1faa0..f3d4b71ae 100644 --- a/src/test_eressea.c +++ b/src/test_eressea.c @@ -118,6 +118,7 @@ int RunAllTests(int argc, char *argv[]) ADD_SUITE(give); ADD_SUITE(laws); ADD_SUITE(market); + ADD_SUITE(monsters); ADD_SUITE(move); ADD_SUITE(piracy); ADD_SUITE(stealth); diff --git a/src/tests.c b/src/tests.c index e7518211e..d25fa72a3 100644 --- a/src/tests.c +++ b/src/tests.c @@ -238,10 +238,10 @@ void test_create_world(void) test_create_itemtype("iron"); test_create_itemtype("stone"); - t_plain = test_create_terrain("plain", LAND_REGION | FOREST_REGION | WALK_INTO | CAVALRY_REGION); + t_plain = test_create_terrain("plain", LAND_REGION | FOREST_REGION | WALK_INTO | CAVALRY_REGION | FLY_INTO); t_plain->size = 1000; t_plain->max_road = 100; - t_ocean = test_create_terrain("ocean", SEA_REGION | SAIL_INTO | SWIM_INTO); + t_ocean = test_create_terrain("ocean", SEA_REGION | SAIL_INTO | SWIM_INTO | FLY_INTO); t_ocean->size = 0; island[0] = test_create_region(0, 0, t_plain); From a144686e6675928979dd2f64344a44f924d3f3fb Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Tue, 17 Nov 2015 03:27:23 +0100 Subject: [PATCH 06/11] monsters may attack on ocean --- src/monsters.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/monsters.c b/src/monsters.c index 6af672133..3984b8f6b 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -787,7 +787,7 @@ void plan_monsters(faction * f) setstatus(u, ST_FIGHT); /* all monsters fight */ } - if (attacking && is_guard(u, GUARD_TAX)) { + if (attacking && (!r->land || is_guard(u, GUARD_TAX))) { monster_attacks(u, true, false); } From 2a569635dfe640e8f7981d2bd48723be72a94264 Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Tue, 17 Nov 2015 12:06:12 +0100 Subject: [PATCH 07/11] seaserpent test --- src/monsters.test.c | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/src/monsters.test.c b/src/monsters.test.c index d19633277..39a04959e 100644 --- a/src/monsters.test.c +++ b/src/monsters.test.c @@ -145,6 +145,31 @@ static void test_monsters_waiting(CuTest * tc) test_cleanup(); } +static void test_seaserpent_attack(CuTest * tc) +{ + faction *f, *f2; + region *r; + unit *u, *m; + race *rc; + + create_monsters(&f, &f2, &r, &u, &m); + r = findregion(-1, 0); + u = test_create_unit(u->faction, r); + unit_setid(u, 2); + m = test_create_unit(m->faction, r); + u_setrace(m, rc = test_create_race("seaserpent")); + assert(!m->region->land); + fset(m, UFL_MOVED); + fset(rc, RCF_ATTACK_MOVED); + + set_param(&global.parameters, "rules.monsters.attack_chance", "1"); + + plan_monsters(f2); + + CuAssertPtrNotNull(tc, find_order("ATTACKIERE 2", m)); + test_cleanup(); +} + static void test_monsters_attack_not(CuTest * tc) { faction *f, *f2; @@ -238,6 +263,7 @@ CuSuite *get_monsters_suite(void) CuSuite *suite = CuSuiteNew(); SUITE_ADD_TEST(suite, test_monsters_attack); SUITE_ADD_TEST(suite, test_monsters_attack_ocean); + SUITE_ADD_TEST(suite, test_seaserpent_attack); SUITE_ADD_TEST(suite, test_monsters_waiting); SUITE_ADD_TEST(suite, test_monsters_attack_not); SUITE_ADD_TEST(suite, test_dragon_attacks_the_rich); From 742a30cc3965e651cc3e374a10a1d7fbd1b0761e Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Tue, 17 Nov 2015 11:25:58 +0100 Subject: [PATCH 08/11] fix seaserpents seaserpents could not do piracy seaserpents should be able to attack after move --- res/e3a/races.xml | 2 +- res/eressea/races.xml | 2 +- src/kernel/race.h | 1 + src/kernel/xmlreader.c | 2 ++ src/monster.c | 3 ++- src/piracy.c | 3 +++ src/piracy.test.c | 16 +++++++++++++++- 7 files changed, 25 insertions(+), 4 deletions(-) diff --git a/res/e3a/races.xml b/res/e3a/races.xml index 46427ba92..164a599b7 100644 --- a/res/e3a/races.xml +++ b/res/e3a/races.xml @@ -883,7 +883,7 @@ - + diff --git a/res/eressea/races.xml b/res/eressea/races.xml index 7f6c6d873..ec119bd21 100644 --- a/res/eressea/races.xml +++ b/res/eressea/races.xml @@ -1171,7 +1171,7 @@ - + diff --git a/src/kernel/race.h b/src/kernel/race.h index 0ae925c5d..fa18cb8f2 100644 --- a/src/kernel/race.h +++ b/src/kernel/race.h @@ -214,6 +214,7 @@ extern "C" { #define RCF_SHIPSPEED (1<<26) /* race gets +1 on shipspeed */ #define RCF_STONEGOLEM (1<<27) /* race gets stonegolem properties */ #define RCF_IRONGOLEM (1<<28) /* race gets irongolem properties */ +#define RCF_ATTACK_MOVED (1<<29) /* may attack if it has moved */ /* Economic flags */ #define ECF_KEEP_ITEM (1<<1) /* gibt Gegenstände weg */ diff --git a/src/kernel/xmlreader.c b/src/kernel/xmlreader.c index 28a351a72..c5af75d04 100644 --- a/src/kernel/xmlreader.c +++ b/src/kernel/xmlreader.c @@ -1604,6 +1604,8 @@ static void parse_ai(race * rc, xmlNodePtr node) rc->flags |= RCF_MOVERANDOM; if (xml_bvalue(node, "learn", false)) rc->flags |= RCF_LEARN; + if (xml_bvalue(node, "moveattack", false)) + rc->flags |= RCF_ATTACK_MOVED; } static int parse_races(xmlDocPtr doc) diff --git a/src/monster.c b/src/monster.c index 47858fc5d..3d6fd6f82 100644 --- a/src/monster.c +++ b/src/monster.c @@ -69,7 +69,8 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. bool monster_is_waiting(const unit * u) { - if (fval(u, UFL_ISNEW | UFL_MOVED)) + int test = fval(u_race(u), RCF_ATTACK_MOVED) ? UFL_ISNEW : UFL_ISNEW | UFL_MOVED; + if (fval(u, test)) return true; return false; } diff --git a/src/piracy.c b/src/piracy.c index 8911f0882..4843880b8 100644 --- a/src/piracy.c +++ b/src/piracy.c @@ -9,6 +9,7 @@ #include #include #include +#include #include #include #include @@ -64,6 +65,8 @@ static attrib *mk_piracy(const faction * pirate, const faction * target, } static bool validate_pirate(unit *u, order *ord) { + if (fval(u_race(u), RCF_SWIM | RCF_FLY)) + return true; if (!u->ship) { cmistake(u, ord, 144, MSG_MOVE); return false; diff --git a/src/piracy.test.c b/src/piracy.test.c index 53907f4df..acc220d3a 100644 --- a/src/piracy.test.c +++ b/src/piracy.test.c @@ -8,6 +8,7 @@ #include #include #include +#include #include #include @@ -66,6 +67,7 @@ static void test_piracy_cmd(CuTest * tc) { } static void test_piracy_cmd_errors(CuTest * tc) { + race *r; faction *f; unit *u, *u2; order *ord; @@ -73,7 +75,8 @@ static void test_piracy_cmd_errors(CuTest * tc) { setup_piracy(); st_boat = st_get_or_create("boat"); - u = test_create_unit(f = test_create_faction(0), test_create_region(0, 0, get_or_create_terrain("ocean"))); + r = test_create_race("pirates"); + u = test_create_unit(f = test_create_faction(r), test_create_region(0, 0, get_or_create_terrain("ocean"))); f->locale = get_or_create_locale("de"); ord = create_order(K_PIRACY, f->locale, ""); assert(u && ord); @@ -81,6 +84,17 @@ static void test_piracy_cmd_errors(CuTest * tc) { piracy_cmd(u, ord); CuAssertPtrNotNullMsg(tc, "must be on a ship for PIRACY", test_find_messagetype(f->msgs, "error144")); + test_clear_messages(f); + fset(r, RCF_SWIM); + piracy_cmd(u, ord); + CuAssertPtrEquals_Msg(tc, "swimmers are pirates", 0, test_find_messagetype(f->msgs, "error144")); + CuAssertPtrEquals_Msg(tc, "swimmers are pirates", 0, test_find_messagetype(f->msgs, "error146")); + freset(r, RCF_SWIM); + fset(r, RCF_FLY); + CuAssertPtrEquals_Msg(tc, "flyers are pirates", 0, test_find_messagetype(f->msgs, "error144")); + freset(r, RCF_FLY); + test_clear_messages(f); + u_set_ship(u, test_create_ship(u->region, st_boat)); u2 = test_create_unit(u->faction, u->region); From 8a063c3567a1181f4f2246a14ad02aa404512bb3 Mon Sep 17 00:00:00 2001 From: Steffen Mecke Date: Tue, 17 Nov 2015 17:19:06 +0100 Subject: [PATCH 09/11] fix bug in dragon_affinity --- src/monsters.c | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/monsters.c b/src/monsters.c index 3984b8f6b..6ad98f664 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -412,10 +412,10 @@ static int dragon_affinity_value(region * r, unit * u) int m = all_money(r, u->faction); if (u_race(u) == get_race(RC_FIREDRAGON)) { - return dice(6, m / 6); + return dice(4, m / 2); } else { - return dice(4, m / 8); + return dice(6, m / 3); } } @@ -779,7 +779,6 @@ void plan_monsters(faction * f) free_orders(&u->orders); if (skill_enabled(SK_PERCEPTION)) { /* Monster bekommen jede Runde ein paar Tage Wahrnehmung dazu */ - /* TODO: this only works for playerrace */ produceexp(u, SK_PERCEPTION, u->number); } From 90b0bdd44e82008f3ad236c211b3aa0cb17ef7eb Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Wed, 25 Nov 2015 14:53:29 +0100 Subject: [PATCH 10/11] re-enable all monster tests except piracy for seaserpents, which is red. --- src/monsters.test.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/monsters.test.c b/src/monsters.test.c index 66e96153c..2ba2e2ae0 100644 --- a/src/monsters.test.c +++ b/src/monsters.test.c @@ -264,10 +264,10 @@ CuSuite *get_monsters_suite(void) SUITE_ADD_TEST(suite, test_monsters_attack); SUITE_ADD_TEST(suite, test_monsters_attack_ocean); DISABLE_TEST(suite, test_seaserpent_piracy); - DISABLE_TEST(suite, test_monsters_waiting); + SUITE_ADD_TEST(suite, test_monsters_waiting); SUITE_ADD_TEST(suite, test_monsters_attack_not); SUITE_ADD_TEST(suite, test_dragon_attacks_the_rich); - DISABLE_TEST(suite, test_dragon_moves); - DISABLE_TEST(suite, test_monsters_learn_exp); + SUITE_ADD_TEST(suite, test_dragon_moves); + SUITE_ADD_TEST(suite, test_monsters_learn_exp); return suite; } From 6c2f4be4afe8862bd49974a3ce5ee638883b0c46 Mon Sep 17 00:00:00 2001 From: Enno Rehling Date: Wed, 25 Nov 2015 15:29:26 +0100 Subject: [PATCH 11/11] fix sea serpents using piracy (by re-ordering the decision making process) --- src/monsters.c | 24 ++++++++++++------------ src/monsters.test.c | 2 +- 2 files changed, 13 insertions(+), 13 deletions(-) diff --git a/src/monsters.c b/src/monsters.c index 988045a9d..ab63cbb31 100644 --- a/src/monsters.c +++ b/src/monsters.c @@ -827,18 +827,6 @@ void plan_monsters(faction * f) } } - if (long_order == NULL && unit_can_study(u)) { - /* Einheiten, die Waffenlosen Kampf lernen könnten, lernen es um - * zu bewachen: */ - if (u_race(u)->bonus[SK_WEAPONLESS] != -99) { - if (effskill(u, SK_WEAPONLESS, 0) < 1) { - long_order = - create_order(K_STUDY, f->locale, "'%s'", - skillname(SK_WEAPONLESS, f->locale)); - } - } - } - if (long_order == NULL) { /* Ab hier noch nicht generalisierte Spezialbehandlungen. */ @@ -867,6 +855,18 @@ void plan_monsters(faction * f) break; } } + if (long_order == NULL && unit_can_study(u)) { + /* Einheiten, die Waffenlosen Kampf lernen könnten, lernen es um + * zu bewachen: */ + if (u_race(u)->bonus[SK_WEAPONLESS] != -99) { + if (effskill(u, SK_WEAPONLESS, 0) < 1) { + long_order = + create_order(K_STUDY, f->locale, "'%s'", + skillname(SK_WEAPONLESS, f->locale)); + } + } + } + if (long_order) { addlist(&u->orders, long_order); } diff --git a/src/monsters.test.c b/src/monsters.test.c index 2ba2e2ae0..9fce059f2 100644 --- a/src/monsters.test.c +++ b/src/monsters.test.c @@ -263,7 +263,7 @@ CuSuite *get_monsters_suite(void) CuSuite *suite = CuSuiteNew(); SUITE_ADD_TEST(suite, test_monsters_attack); SUITE_ADD_TEST(suite, test_monsters_attack_ocean); - DISABLE_TEST(suite, test_seaserpent_piracy); + SUITE_ADD_TEST(suite, test_seaserpent_piracy); SUITE_ADD_TEST(suite, test_monsters_waiting); SUITE_ADD_TEST(suite, test_monsters_attack_not); SUITE_ADD_TEST(suite, test_dragon_attacks_the_rich);