diff --git a/res/buildings/castle-2.xml b/res/buildings/castle-2.xml
index 5f48e73ee..76c908572 100644
--- a/res/buildings/castle-2.xml
+++ b/res/buildings/castle-2.xml
@@ -1,24 +1,24 @@
-
+
-
+
-
+
-
+
-
+
-
+
-
+
diff --git a/res/buildings/castle.xml b/res/buildings/castle.xml
index 296d1f716..49d1ca5a8 100644
--- a/res/buildings/castle.xml
+++ b/res/buildings/castle.xml
@@ -1,26 +1,26 @@
-
-
+
+
-
+
-
+
-
+
-
+
-
+
-
+
diff --git a/res/e3a/buildings.xml b/res/e3a/buildings.xml
index b3e59f0a4..227a2e47a 100644
--- a/res/e3a/buildings.xml
+++ b/res/e3a/buildings.xml
@@ -5,15 +5,15 @@
-
+
-
+
-
+
-
+
diff --git a/scripts/eressea/e3/rules.lua b/scripts/eressea/e3/rules.lua
index 0c5e9be85..c0ac8978a 100644
--- a/scripts/eressea/e3/rules.lua
+++ b/scripts/eressea/e3/rules.lua
@@ -26,10 +26,6 @@ function item_canuse(u, iname)
return true
end
-function building_protection(b, u)
- return 1
-end
-
function building_taxes(b, blevel)
btype = b.type
if btype=="castle" then
diff --git a/src/battle.c b/src/battle.c
index 7065ad11b..ed4354049 100644
--- a/src/battle.c
+++ b/src/battle.c
@@ -1901,7 +1901,7 @@ int skilldiff(troop at, troop dt, int dist)
init = true;
}
if (df->building->type->protection) {
- int beff = df->building->type->protection(df->building, du);
+ int beff = df->building->type->protection(df->building, du, DEFENSE_BONUS);
if (beff) {
skdiff -= beff;
is_protected = 2;
@@ -1918,7 +1918,7 @@ int skilldiff(troop at, troop dt, int dist)
if (magicwalls_ct
&& curse_active(get_curse(df->building->attribs, magicwalls_ct))) {
/* Verdoppelt Burgenbonus */
- skdiff -= buildingeffsize(df->building, false);
+ skdiff -= df->building->type->protection(df->building, du, DEFENSE_BONUS);
}
}
/* Goblin-Verteidigung
diff --git a/src/battle.test.c b/src/battle.test.c
index e35864c12..1a329252c 100644
--- a/src/battle.test.c
+++ b/src/battle.test.c
@@ -9,6 +9,7 @@
#include
#include
#include
+#include
#include
#include "tests.h"
@@ -26,7 +27,7 @@ static void test_make_fighter(CuTest * tc)
test_cleanup();
test_create_world();
r = findregion(0, 0);
- f = test_create_faction(rc_find("human"));
+ f = test_create_faction(NULL);
au = test_create_unit(f, r);
enable_skill(SK_MAGIC, true);
enable_skill(SK_RIDING, true);
@@ -57,131 +58,160 @@ static void test_make_fighter(CuTest * tc)
test_cleanup();
}
-static int add_two(building * b, unit * u) {
+static int add_two(building * b, unit * u, building_bonus bonus) {
return 2;
}
static void test_defenders_get_building_bonus(CuTest * tc)
{
- unit *du, *au;
- region *r;
- building * bld;
- fighter *df, *af;
- battle *b;
- side *ds, *as;
- int diff;
- troop dt, at;
- building_type * btype;
+ unit *du, *au;
+ region *r;
+ building * bld;
+ fighter *df, *af;
+ battle *b;
+ side *ds, *as;
+ int diff;
+ troop dt, at;
+ building_type * btype;
- test_cleanup();
- test_create_world();
- r = findregion(0, 0);
- btype = bt_get_or_create("castle");
- btype->protection = &add_two;
- bld = test_create_building(r, btype);
- bld->size = 10;
+ test_cleanup();
+ test_create_world();
+ r = findregion(0, 0);
+ btype = bt_get_or_create("castle");
+ btype->protection = &add_two;
+ bld = test_create_building(r, btype);
+ bld->size = 10;
- du = test_create_unit(test_create_faction(rc_find("human")), r);
- au = test_create_unit(test_create_faction(rc_find("human")), r);
- u_set_building(du, bld);
+ du = test_create_unit(test_create_faction(NULL), r);
+ au = test_create_unit(test_create_faction(NULL), r);
+ u_set_building(du, bld);
- b = make_battle(r);
- ds = make_side(b, du->faction, 0, 0, 0);
- df = make_fighter(b, du, ds, false);
- as = make_side(b, au->faction, 0, 0, 0);
- af = make_fighter(b, au, as, true);
+ b = make_battle(r);
+ ds = make_side(b, du->faction, 0, 0, 0);
+ df = make_fighter(b, du, ds, false);
+ as = make_side(b, au->faction, 0, 0, 0);
+ af = make_fighter(b, au, as, true);
- CuAssertPtrEquals(tc, bld, df->building);
- CuAssertPtrEquals(tc, 0, af->building);
+ CuAssertPtrEquals(tc, bld, df->building);
+ CuAssertPtrEquals(tc, 0, af->building);
- dt.fighter = df;
- dt.index = 0;
- at.fighter = af;
- at.index = 0;
+ dt.fighter = df;
+ dt.index = 0;
+ at.fighter = af;
+ at.index = 0;
- diff = skilldiff(at, dt, 0);
- CuAssertIntEquals(tc, -2, diff);
+ diff = skilldiff(at, dt, 0);
+ CuAssertIntEquals(tc, -2, diff);
- diff = skilldiff(dt, at, 0);
- CuAssertIntEquals(tc, 0, diff);
+ diff = skilldiff(dt, at, 0);
+ CuAssertIntEquals(tc, 0, diff);
free_battle(b);
test_cleanup();
}
static void test_attackers_get_no_building_bonus(CuTest * tc)
{
- unit *au;
- region *r;
- building * bld;
- fighter *af;
- battle *b;
- side *as;
- building_type * btype;
+ unit *au;
+ region *r;
+ building * bld;
+ fighter *af;
+ battle *b;
+ side *as;
+ building_type * btype;
- test_cleanup();
- test_create_world();
- r = findregion(0, 0);
- btype = bt_get_or_create("castle");
- btype->protection = &add_two;
- bld = test_create_building(r, btype);
- bld->size = 10;
+ test_cleanup();
+ test_create_world();
+ r = findregion(0, 0);
+ btype = bt_get_or_create("castle");
+ btype->protection = &add_two;
+ bld = test_create_building(r, btype);
+ bld->size = 10;
- au = test_create_unit(test_create_faction(rc_find("human")), r);
- u_set_building(au, bld);
+ au = test_create_unit(test_create_faction(NULL), r);
+ u_set_building(au, bld);
- b = make_battle(r);
- as = make_side(b, au->faction, 0, 0, 0);
- af = make_fighter(b, au, as, true);
+ b = make_battle(r);
+ as = make_side(b, au->faction, 0, 0, 0);
+ af = make_fighter(b, au, as, true);
- CuAssertPtrEquals(tc, 0, af->building);
+ CuAssertPtrEquals(tc, 0, af->building);
free_battle(b);
test_cleanup();
}
static void test_building_bonus_respects_size(CuTest * tc)
{
- unit *au, *du;
- region *r;
- building * bld;
- fighter *af, *df;
- battle *b;
- side *as;
- building_type * btype;
- faction * f;
+ unit *au, *du;
+ region *r;
+ building * bld;
+ fighter *af, *df;
+ battle *b;
+ side *as;
+ building_type * btype;
+ faction * f;
- test_cleanup();
- test_create_world();
- r = findregion(0, 0);
- btype = bt_get_or_create("castle");
- btype->protection = &add_two;
- bld = test_create_building(r, btype);
- bld->size = 10;
+ test_cleanup();
+ test_create_world();
+ r = findregion(0, 0);
+ btype = bt_get_or_create("castle");
+ btype->protection = &add_two;
+ bld = test_create_building(r, btype);
+ bld->size = 10;
- f = test_create_faction(rc_find("human"));
- au = test_create_unit(f, r);
- scale_number(au, 9);
- u_set_building(au, bld);
- du = test_create_unit(f, r);
- u_set_building(du, bld);
- scale_number(du, 2);
+ f = test_create_faction(NULL);
+ au = test_create_unit(f, r);
+ scale_number(au, 9);
+ u_set_building(au, bld);
+ du = test_create_unit(f, r);
+ u_set_building(du, bld);
+ scale_number(du, 2);
- b = make_battle(r);
- as = make_side(b, au->faction, 0, 0, 0);
- af = make_fighter(b, au, as, false);
- df = make_fighter(b, du, as, false);
+ b = make_battle(r);
+ as = make_side(b, au->faction, 0, 0, 0);
+ af = make_fighter(b, au, as, false);
+ df = make_fighter(b, du, as, false);
- CuAssertPtrEquals(tc, bld, af->building);
- CuAssertPtrEquals(tc, 0, df->building);
+ CuAssertPtrEquals(tc, bld, af->building);
+ CuAssertPtrEquals(tc, 0, df->building);
free_battle(b);
test_cleanup();
}
+static void test_building_defence_bonus(CuTest * tc)
+{
+ unit *au;
+ region *r;
+ building * bld;
+ building_type * btype;
+ faction * f;
+ int def;
+ test_cleanup();
+ test_create_world();
+ r = findregion(0, 0);
+ register_buildings();
+ btype = bt_get_or_create("castle");
+ btype->protection = (int (*)(struct building *, struct unit *, building_bonus))get_function("building_protection");
+ btype->construction->defense_bonus = 3;
+ bld = test_create_building(r, btype);
+ bld->size = 1;
+
+ f = test_create_faction(NULL);
+ au = test_create_unit(f, r);
+ scale_number(au, 1);
+ u_set_building(au, bld);
+
+ def = btype->protection(bld, au, DEFENSE_BONUS);
+ CuAssertIntEquals(tc, 3, def);
+ test_cleanup();
+}
+
CuSuite *get_battle_suite(void)
{
- CuSuite *suite = CuSuiteNew();
- SUITE_ADD_TEST(suite, test_make_fighter);
- SUITE_ADD_TEST(suite, test_defenders_get_building_bonus);
- SUITE_ADD_TEST(suite, test_attackers_get_no_building_bonus);
- SUITE_ADD_TEST(suite, test_building_bonus_respects_size);
- return suite;
+ CuSuite *suite = CuSuiteNew();
+ SUITE_ADD_TEST(suite, test_make_fighter);
+ SUITE_ADD_TEST(suite, test_defenders_get_building_bonus);
+ SUITE_ADD_TEST(suite, test_attackers_get_no_building_bonus);
+ SUITE_ADD_TEST(suite, test_building_bonus_respects_size);
+ SUITE_ADD_TEST(suite, test_building_defence_bonus);
+ return suite;
}
diff --git a/src/helpers.c b/src/helpers.c
index 5f0b3a12d..e37865be5 100644
--- a/src/helpers.c
+++ b/src/helpers.c
@@ -388,32 +388,6 @@ static void lua_agebuilding(building * b)
}
}
-static int lua_building_protection(building * b, unit * u)
-{
- lua_State *L = (lua_State *) global.vm_state;
- const char *fname = "building_protection";
- int result = 0;
-
- lua_getglobal(L, fname);
- if (lua_isfunction(L, -1)) {
- tolua_pushusertype(L, (void *)b, TOLUA_CAST "building");
- tolua_pushusertype(L, (void *)u, TOLUA_CAST "unit");
-
- if (lua_pcall(L, 2, 1, 0) != 0) {
- const char *error = lua_tostring(L, -1);
- log_error("building_protection(%s, %s) calling '%s': %s.\n", buildingname(b), unitname(u), fname, error);
- lua_pop(L, 1);
- } else {
- result = (int)lua_tonumber(L, -1);
- lua_pop(L, 1);
- }
- } else {
- log_error("building_protection(%s, %s) calling '%s': not a function.\n", buildingname(b), unitname(u), fname);
- lua_pop(L, 1);
- }
- return result;
-}
-
static double lua_building_taxes(building * b, int level)
{
lua_State *L = (lua_State *) global.vm_state;
@@ -545,8 +519,6 @@ void register_tolua_helpers(void)
at_register(&at_direction);
at_register(&at_building_action);
- register_function((pf_generic) & lua_building_protection,
- TOLUA_CAST "lua_building_protection");
register_function((pf_generic) & lua_building_taxes,
TOLUA_CAST "lua_building_taxes");
register_function((pf_generic) & lua_agebuilding,
diff --git a/src/kernel/build.h b/src/kernel/build.h
index 4e3bbedaa..8bd41e409 100644
--- a/src/kernel/build.h
+++ b/src/kernel/build.h
@@ -45,6 +45,9 @@ extern "C" {
int maxsize; /* maximum size of this type */
int reqsize; /* size of object using up 1 set of requirement. */
+ int defense_bonus; /* protection bonus (defense) during combat */
+ int close_combat_bonus; /* close combat attack bonus*/
+ int ranged_bonus; /* ranged attack bonus */
requirement *materials; /* material req'd to build one object */
const struct building_type *btype;
/* building type required to make this thing */
diff --git a/src/kernel/building.c b/src/kernel/building.c
index a382a63bf..f74976661 100644
--- a/src/kernel/building.c
+++ b/src/kernel/building.c
@@ -316,12 +316,32 @@ const building_type *findbuildingtype(const char *name,
return (const building_type *)type.v;
}
-static int eressea_building_protection(building * b, unit * u)
+static int building_protection(building * b, unit * u, building_bonus bonus)
{
- int beff = buildingeffsize(b, false) - 1;
- /* -1 because the tradepost has no protection value */
- return beff;
+ int i = 0;
+ int bsize = buildingeffsize(b, false);
+ const construction *cons = b->type->construction;
+ if (!cons) {
+ return 0;
+ }
+
+ for (i = 0; i < bsize; i++)
+ {
+ cons = cons->improvement;
+ }
+
+ switch (bonus)
+ {
+ case DEFENSE_BONUS:
+ return cons->defense_bonus;
+ case CLOSE_COMBAT_ATTACK_BONUS:
+ return cons->close_combat_bonus;
+ case RANGED_ATTACK_BONUS:
+ return cons->ranged_bonus;
+ default:
+ return 0;
+ }
}
static int meropis_building_protection(building * b, unit * u)
@@ -331,8 +351,8 @@ static int meropis_building_protection(building * b, unit * u)
void register_buildings(void)
{
- register_function((pf_generic)& eressea_building_protection,
- "eressea_building_protection");
+ register_function((pf_generic)& building_protection,
+ "building_protection");
register_function((pf_generic)& meropis_building_protection,
"meropis_building_protection");
register_function((pf_generic)& init_smithy, "init_smithy");
diff --git a/src/kernel/building.h b/src/kernel/building.h
index cf80de84f..98db1bd2f 100644
--- a/src/kernel/building.h
+++ b/src/kernel/building.h
@@ -48,6 +48,12 @@ extern "C" {
#define BTF_ONEPERTURN 0x80 /* one one sizepoint can be added per turn */
#define BTF_NAMECHANGE 0x100 /* name and description can be changed more than once */
+ typedef enum {
+ DEFENSE_BONUS,
+ CLOSE_COMBAT_ATTACK_BONUS,
+ RANGED_ATTACK_BONUS,
+ } building_bonus;
+
typedef struct building_type {
char *_name;
@@ -66,7 +72,7 @@ extern "C" {
const struct building * b, int size);
void (*init) (struct building_type *);
void (*age) (struct building *);
- int (*protection) (struct building *, struct unit *);
+ int (*protection) (struct building *, struct unit *, building_bonus);
double (*taxes) (const struct building *, int size);
struct attrib *attribs;
} building_type;
diff --git a/src/kernel/xmlreader.c b/src/kernel/xmlreader.c
index 77d2270e1..1b1c6b356 100644
--- a/src/kernel/xmlreader.c
+++ b/src/kernel/xmlreader.c
@@ -167,7 +167,10 @@ construction ** consPtr)
con->maxsize = xml_ivalue(node, "maxsize", -1);
con->minskill = xml_ivalue(node, "minskill", -1);
con->reqsize = xml_ivalue(node, "reqsize", -1);
-
+ con->defense_bonus = xml_ivalue(node, "defense_bonus", 0);
+ con->close_combat_bonus = xml_ivalue(node, "close_combat_bonus", 0);
+ con->ranged_bonus = xml_ivalue(node, "ranged_bonus", 0);
+
propValue = xmlGetProp(node, BAD_CAST "building");
if (propValue != NULL) {
con->btype = bt_get_or_create((const char *)propValue);
@@ -297,7 +300,7 @@ static int parse_buildings(xmlDocPtr doc)
btype->age = (void(*)(struct building *))fun;
}
else if (strcmp((const char *)propValue, "protection") == 0) {
- btype->protection = (int(*)(struct building *, struct unit *))fun;
+ btype->protection = (int(*)(struct building *, struct unit *, building_bonus))fun;
}
else if (strcmp((const char *)propValue, "taxes") == 0) {
btype->taxes = (double(*)(const struct building *, int))fun;