2011-03-13 02:01:20 +01:00
|
|
|
#include <platform.h>
|
2014-08-27 06:40:18 +02:00
|
|
|
|
2011-03-13 02:01:20 +01:00
|
|
|
#include "battle.h"
|
2018-07-15 13:39:23 +02:00
|
|
|
|
|
|
|
#include "reports.h"
|
2011-03-16 05:04:38 +01:00
|
|
|
#include "skill.h"
|
2014-08-27 06:40:18 +02:00
|
|
|
|
2015-11-21 19:02:14 +01:00
|
|
|
#include <kernel/config.h>
|
2014-08-27 06:40:18 +02:00
|
|
|
#include <kernel/building.h>
|
|
|
|
#include <kernel/faction.h>
|
2016-08-29 09:53:09 +02:00
|
|
|
#include <kernel/curse.h>
|
2014-08-27 06:40:18 +02:00
|
|
|
#include <kernel/item.h>
|
|
|
|
#include <kernel/race.h>
|
|
|
|
#include <kernel/region.h>
|
2018-05-03 23:00:28 +02:00
|
|
|
#include <kernel/ship.h>
|
2014-08-27 06:40:18 +02:00
|
|
|
#include <kernel/unit.h>
|
2017-08-18 18:42:59 +02:00
|
|
|
|
|
|
|
#include <spells/buildingcurse.h>
|
|
|
|
|
2015-01-12 20:23:25 +01:00
|
|
|
#include <util/functions.h>
|
2018-07-15 13:39:23 +02:00
|
|
|
#include <util/language.h>
|
|
|
|
#include <util/message.h>
|
2017-07-31 12:55:05 +02:00
|
|
|
#include <util/rand.h>
|
|
|
|
#include <util/rng.h>
|
2018-05-01 15:32:06 +02:00
|
|
|
#include <util/strings.h>
|
2012-06-24 07:36:17 +02:00
|
|
|
|
2012-05-31 04:17:08 +02:00
|
|
|
#include <CuTest.h>
|
2017-07-31 12:55:05 +02:00
|
|
|
|
2018-05-01 15:32:06 +02:00
|
|
|
#include <assert.h>
|
2017-07-31 12:55:05 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2012-06-24 07:36:17 +02:00
|
|
|
#include "tests.h"
|
2011-03-13 02:01:20 +01:00
|
|
|
|
2011-03-16 05:04:38 +01:00
|
|
|
static void test_make_fighter(CuTest * tc)
|
|
|
|
{
|
2014-06-25 07:44:05 +02:00
|
|
|
unit *au;
|
|
|
|
region *r;
|
|
|
|
fighter *af;
|
|
|
|
battle *b;
|
|
|
|
side *as;
|
|
|
|
faction * f;
|
|
|
|
const resource_type *rtype;
|
2011-03-16 05:04:38 +01:00
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2015-11-07 23:34:51 +01:00
|
|
|
test_create_horse();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2015-01-12 20:23:25 +01:00
|
|
|
f = test_create_faction(NULL);
|
2014-06-25 07:44:05 +02:00
|
|
|
au = test_create_unit(f, r);
|
|
|
|
enable_skill(SK_MAGIC, true);
|
|
|
|
enable_skill(SK_RIDING, true);
|
|
|
|
set_level(au, SK_MAGIC, 3);
|
|
|
|
set_level(au, SK_RIDING, 3);
|
|
|
|
au->status = ST_BEHIND;
|
|
|
|
rtype = get_resourcetype(R_HORSE);
|
|
|
|
i_change(&au->items, rtype->itype, 1);
|
2015-01-30 20:37:14 +01:00
|
|
|
|
2014-06-25 07:44:05 +02:00
|
|
|
b = make_battle(r);
|
|
|
|
as = make_side(b, au->faction, 0, 0, 0);
|
|
|
|
af = make_fighter(b, au, as, false);
|
2015-01-30 20:37:14 +01:00
|
|
|
|
2014-06-25 07:44:05 +02:00
|
|
|
CuAssertIntEquals(tc, 1, b->nfighters);
|
|
|
|
CuAssertPtrEquals(tc, 0, af->building);
|
|
|
|
CuAssertPtrEquals(tc, as, af->side);
|
|
|
|
CuAssertIntEquals(tc, 0, af->run.hp);
|
|
|
|
CuAssertIntEquals(tc, ST_BEHIND, af->status);
|
|
|
|
CuAssertIntEquals(tc, 0, af->run.number);
|
|
|
|
CuAssertIntEquals(tc, au->hp, af->person[0].hp);
|
|
|
|
CuAssertIntEquals(tc, 1, af->person[0].speed);
|
|
|
|
CuAssertIntEquals(tc, au->number, af->alive);
|
|
|
|
CuAssertIntEquals(tc, 0, af->removed);
|
|
|
|
CuAssertIntEquals(tc, 3, af->magic);
|
|
|
|
CuAssertIntEquals(tc, 1, af->horses);
|
|
|
|
CuAssertIntEquals(tc, 0, af->elvenhorses);
|
2014-12-12 11:13:25 +01:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2011-03-16 05:04:38 +01:00
|
|
|
}
|
|
|
|
|
2018-04-29 15:24:36 +02:00
|
|
|
static void test_select_weapon_restricted(CuTest *tc) {
|
2018-04-29 15:09:07 +02:00
|
|
|
item_type *itype;
|
|
|
|
unit *au;
|
|
|
|
fighter *af;
|
|
|
|
battle *b;
|
|
|
|
race * rc;
|
|
|
|
|
|
|
|
test_setup();
|
|
|
|
au = test_create_unit(test_create_faction(NULL), test_create_plain(0, 0));
|
|
|
|
itype = test_create_itemtype("halberd");
|
2018-04-30 20:25:58 +02:00
|
|
|
new_weapontype(itype, 0, frac_zero, NULL, 0, 0, 0, SK_MELEE);
|
2018-04-29 15:09:07 +02:00
|
|
|
i_change(&au->items, itype, 1);
|
|
|
|
rc = test_create_race("smurf");
|
|
|
|
CuAssertIntEquals(tc, 0, rc->mask_item & au->_race->mask_item);
|
|
|
|
|
|
|
|
b = make_battle(au->region);
|
|
|
|
af = make_fighter(b, au, make_side(b, au->faction, 0, 0, 0), false);
|
|
|
|
CuAssertPtrNotNull(tc, af->weapons);
|
|
|
|
CuAssertIntEquals(tc, 1, af->weapons[0].count);
|
2018-04-29 15:24:36 +02:00
|
|
|
CuAssertIntEquals(tc, 0, af->weapons[1].count);
|
2018-04-29 15:09:07 +02:00
|
|
|
free_battle(b);
|
|
|
|
|
|
|
|
itype->mask_deny = rc_mask(au->_race);
|
|
|
|
b = make_battle(au->region);
|
|
|
|
af = make_fighter(b, au, make_side(b, au->faction, 0, 0, 0), false);
|
|
|
|
CuAssertPtrNotNull(tc, af->weapons);
|
|
|
|
CuAssertIntEquals(tc, 0, af->weapons[0].count);
|
|
|
|
free_battle(b);
|
|
|
|
|
|
|
|
itype->mask_deny = 0;
|
|
|
|
itype->mask_allow = rc_mask(rc);
|
|
|
|
b = make_battle(au->region);
|
|
|
|
af = make_fighter(b, au, make_side(b, au->faction, 0, 0, 0), false);
|
|
|
|
CuAssertPtrNotNull(tc, af->weapons);
|
|
|
|
CuAssertIntEquals(tc, 0, af->weapons[0].count);
|
|
|
|
free_battle(b);
|
|
|
|
|
|
|
|
itype->mask_deny = 0;
|
|
|
|
itype->mask_allow = rc_mask(au->_race);
|
|
|
|
b = make_battle(au->region);
|
|
|
|
af = make_fighter(b, au, make_side(b, au->faction, 0, 0, 0), false);
|
|
|
|
CuAssertPtrNotNull(tc, af->weapons);
|
|
|
|
CuAssertIntEquals(tc, 1, af->weapons[0].count);
|
2018-04-29 15:24:36 +02:00
|
|
|
CuAssertIntEquals(tc, 0, af->weapons[1].count);
|
|
|
|
free_battle(b);
|
|
|
|
|
|
|
|
test_teardown();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_select_armor(CuTest *tc) {
|
|
|
|
item_type *itype, *iscale;
|
|
|
|
unit *au;
|
|
|
|
fighter *af;
|
|
|
|
battle *b;
|
|
|
|
|
|
|
|
test_setup();
|
|
|
|
au = test_create_unit(test_create_faction(NULL), test_create_plain(0, 0));
|
|
|
|
itype = test_create_itemtype("plate");
|
|
|
|
new_armortype(itype, 0.0, frac_zero, 1, 0);
|
|
|
|
i_change(&au->items, itype, 2);
|
|
|
|
iscale = test_create_itemtype("scale");
|
|
|
|
new_armortype(iscale, 0.0, frac_zero, 2, 0);
|
|
|
|
i_change(&au->items, iscale, 1);
|
|
|
|
|
|
|
|
b = make_battle(au->region);
|
|
|
|
af = make_fighter(b, au, make_side(b, au->faction, 0, 0, 0), false);
|
|
|
|
CuAssertPtrNotNull(tc, af->armors);
|
|
|
|
CuAssertIntEquals(tc, 1, af->armors->count);
|
|
|
|
CuAssertPtrEquals(tc, iscale->rtype->atype, (armor_type *)af->armors->atype);
|
|
|
|
CuAssertIntEquals(tc, 2, af->armors->next->count);
|
|
|
|
CuAssertPtrEquals(tc, itype->rtype->atype, (armor_type *)af->armors->next->atype);
|
|
|
|
CuAssertPtrEquals(tc, NULL, af->armors->next->next);
|
2018-04-29 15:09:07 +02:00
|
|
|
free_battle(b);
|
|
|
|
|
|
|
|
test_teardown();
|
|
|
|
}
|
|
|
|
|
2017-02-22 19:38:46 +01:00
|
|
|
static building_type * setup_castle(void) {
|
|
|
|
building_type * btype;
|
|
|
|
construction *cons;
|
|
|
|
|
2018-05-01 15:32:06 +02:00
|
|
|
btype = test_create_buildingtype("castle");
|
|
|
|
assert(btype->stages);
|
|
|
|
assert(btype->stages->construction);
|
|
|
|
|
2017-02-22 19:38:46 +01:00
|
|
|
btype->flags |= BTF_FORTIFICATION;
|
2018-05-01 15:32:06 +02:00
|
|
|
cons = btype->stages->construction;
|
2017-02-22 19:38:46 +01:00
|
|
|
cons->maxsize = 5;
|
2018-05-01 15:32:06 +02:00
|
|
|
btype->stages->next = calloc(1, sizeof(building_stage));
|
|
|
|
cons = calloc(1, sizeof(construction));
|
2017-02-22 19:38:46 +01:00
|
|
|
cons->maxsize = -1;
|
2018-05-01 15:32:06 +02:00
|
|
|
btype->stages->next->construction = cons;
|
2017-02-22 19:38:46 +01:00
|
|
|
return btype;
|
|
|
|
}
|
|
|
|
|
2011-03-13 02:54:57 +01:00
|
|
|
static void test_defenders_get_building_bonus(CuTest * tc)
|
2011-03-13 02:01:20 +01:00
|
|
|
{
|
2015-01-12 16:57:05 +01:00
|
|
|
unit *du, *au;
|
|
|
|
region *r;
|
|
|
|
building * bld;
|
|
|
|
fighter *df, *af;
|
|
|
|
battle *b;
|
|
|
|
side *ds, *as;
|
|
|
|
troop dt, at;
|
|
|
|
building_type * btype;
|
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2017-02-22 19:38:46 +01:00
|
|
|
btype = setup_castle();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2015-01-12 16:57:05 +01:00
|
|
|
bld = test_create_building(r, btype);
|
|
|
|
|
2015-01-12 20:23:25 +01:00
|
|
|
du = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
au = test_create_unit(test_create_faction(NULL), r);
|
2015-01-12 16:57:05 +01:00
|
|
|
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);
|
|
|
|
|
|
|
|
CuAssertPtrEquals(tc, bld, df->building);
|
|
|
|
CuAssertPtrEquals(tc, 0, af->building);
|
|
|
|
|
|
|
|
dt.fighter = df;
|
|
|
|
dt.index = 0;
|
|
|
|
at.fighter = af;
|
|
|
|
at.index = 0;
|
|
|
|
|
2017-02-22 19:38:46 +01:00
|
|
|
bld->size = 10; /* stage 1 building */
|
2018-02-25 17:01:32 +01:00
|
|
|
CuAssertIntEquals(tc, 1, buildingeffsize(bld, false));
|
2017-02-22 19:38:46 +01:00
|
|
|
CuAssertIntEquals(tc, -1, skilldiff(at, dt, 0));
|
|
|
|
CuAssertIntEquals(tc, 0, skilldiff(dt, at, 0));
|
|
|
|
|
|
|
|
bld->size = 1; /* stage 0 building */
|
2018-02-25 17:01:32 +01:00
|
|
|
CuAssertIntEquals(tc, 0, buildingeffsize(bld, false));
|
2017-02-22 19:38:46 +01:00
|
|
|
CuAssertIntEquals(tc, 0, skilldiff(at, dt, 0));
|
|
|
|
CuAssertIntEquals(tc, 0, skilldiff(dt, at, 0));
|
2015-01-12 16:57:05 +01:00
|
|
|
|
2014-12-12 11:13:25 +01:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2011-03-13 02:54:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void test_attackers_get_no_building_bonus(CuTest * tc)
|
|
|
|
{
|
2015-01-12 16:57:05 +01:00
|
|
|
unit *au;
|
|
|
|
region *r;
|
|
|
|
building * bld;
|
|
|
|
fighter *af;
|
|
|
|
battle *b;
|
|
|
|
side *as;
|
|
|
|
building_type * btype;
|
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2017-02-22 19:38:46 +01:00
|
|
|
btype = setup_castle();
|
|
|
|
btype->flags |= BTF_FORTIFICATION;
|
2015-01-12 16:57:05 +01:00
|
|
|
bld = test_create_building(r, btype);
|
|
|
|
bld->size = 10;
|
|
|
|
|
2015-01-12 20:23:25 +01:00
|
|
|
au = test_create_unit(test_create_faction(NULL), r);
|
2015-01-12 16:57:05 +01:00
|
|
|
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);
|
|
|
|
|
|
|
|
CuAssertPtrEquals(tc, 0, af->building);
|
2014-12-12 11:13:25 +01:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2011-03-13 02:01:20 +01:00
|
|
|
}
|
|
|
|
|
2011-03-16 05:04:38 +01:00
|
|
|
static void test_building_bonus_respects_size(CuTest * tc)
|
|
|
|
{
|
2015-01-12 16:57:05 +01:00
|
|
|
unit *au, *du;
|
|
|
|
region *r;
|
|
|
|
building * bld;
|
|
|
|
fighter *af, *df;
|
|
|
|
battle *b;
|
|
|
|
side *as;
|
|
|
|
building_type * btype;
|
|
|
|
faction * f;
|
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2017-02-22 19:38:46 +01:00
|
|
|
btype = setup_castle();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2017-02-22 19:38:46 +01:00
|
|
|
btype->flags |= BTF_FORTIFICATION;
|
2015-01-12 16:57:05 +01:00
|
|
|
bld = test_create_building(r, btype);
|
|
|
|
bld->size = 10;
|
|
|
|
|
2015-01-12 20:23:25 +01:00
|
|
|
f = test_create_faction(NULL);
|
2015-01-12 16:57:05 +01:00
|
|
|
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);
|
|
|
|
|
|
|
|
CuAssertPtrEquals(tc, bld, af->building);
|
|
|
|
CuAssertPtrEquals(tc, 0, df->building);
|
2014-12-12 11:13:25 +01:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2011-03-16 05:04:38 +01:00
|
|
|
}
|
|
|
|
|
2015-01-12 16:57:05 +01:00
|
|
|
static void test_building_defence_bonus(CuTest * tc)
|
|
|
|
{
|
|
|
|
building_type * btype;
|
2015-10-14 12:09:35 +02:00
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2017-02-22 19:38:46 +01:00
|
|
|
btype = setup_castle();
|
|
|
|
|
|
|
|
btype->maxsize = -1; /* unlimited buildigs get the castle bonus */
|
|
|
|
CuAssertIntEquals(tc, 0, building_protection(btype, 0));
|
|
|
|
CuAssertIntEquals(tc, 1, building_protection(btype, 1));
|
|
|
|
CuAssertIntEquals(tc, 3, building_protection(btype, 2));
|
|
|
|
CuAssertIntEquals(tc, 5, building_protection(btype, 3));
|
|
|
|
CuAssertIntEquals(tc, 8, building_protection(btype, 4));
|
|
|
|
CuAssertIntEquals(tc, 12, building_protection(btype, 5));
|
|
|
|
CuAssertIntEquals(tc, 12, building_protection(btype, 6));
|
|
|
|
|
|
|
|
btype->maxsize = 10; /* limited-size buildings are treated like an E3 watchtower */
|
|
|
|
CuAssertIntEquals(tc, 0, building_protection(btype, 0));
|
|
|
|
CuAssertIntEquals(tc, 1, building_protection(btype, 1));
|
|
|
|
CuAssertIntEquals(tc, 2, building_protection(btype, 2));
|
|
|
|
CuAssertIntEquals(tc, 2, building_protection(btype, 3));
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2015-01-12 16:57:05 +01:00
|
|
|
}
|
|
|
|
|
2015-11-21 19:02:14 +01:00
|
|
|
static fighter *setup_fighter(battle **bp, unit *u) {
|
2016-08-29 09:04:43 +02:00
|
|
|
battle *b = *bp;
|
|
|
|
side *s;
|
|
|
|
|
|
|
|
if (!b) {
|
|
|
|
*bp = b = make_battle(u->region);
|
|
|
|
}
|
|
|
|
s = make_side(b, u->faction, 0, 0, 0);
|
|
|
|
return make_fighter(b, u, s, false);
|
2015-11-03 18:59:09 +01:00
|
|
|
}
|
|
|
|
|
2015-11-21 19:02:14 +01:00
|
|
|
static void test_natural_armor(CuTest * tc)
|
|
|
|
{
|
|
|
|
race *rc;
|
|
|
|
unit *u;
|
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2015-11-21 19:02:14 +01:00
|
|
|
rc = test_create_race("human");
|
2018-01-14 09:38:26 +01:00
|
|
|
u = test_create_unit(test_create_faction(rc), test_create_region(0, 0, NULL));
|
2015-11-21 19:02:14 +01:00
|
|
|
set_level(u, SK_STAMINA, 2);
|
2017-02-04 23:16:16 +01:00
|
|
|
CuAssertIntEquals(tc, 0, rc_armor_bonus(rc));
|
2015-11-21 19:02:14 +01:00
|
|
|
CuAssertIntEquals(tc, 0, natural_armor(u));
|
2017-02-05 16:55:51 +01:00
|
|
|
rc_set_param(rc, "armor.stamina", "1");
|
2017-02-04 23:16:16 +01:00
|
|
|
CuAssertIntEquals(tc, 1, rc_armor_bonus(rc));
|
2015-11-21 19:02:14 +01:00
|
|
|
CuAssertIntEquals(tc, 2, natural_armor(u));
|
2017-02-05 16:55:51 +01:00
|
|
|
rc_set_param(rc, "armor.stamina", "2");
|
2017-02-04 23:16:16 +01:00
|
|
|
CuAssertIntEquals(tc, 2, rc_armor_bonus(rc));
|
2015-11-21 19:02:14 +01:00
|
|
|
CuAssertIntEquals(tc, 1, natural_armor(u));
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2015-11-21 19:02:14 +01:00
|
|
|
}
|
|
|
|
|
2018-09-12 09:37:22 +02:00
|
|
|
static int test_armor(troop dt, weapon_type *awtype, bool magic) {
|
|
|
|
return calculate_armor(dt, 0, awtype, select_armor(dt, false), select_armor(dt, true), magic);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int test_resistance(troop dt) {
|
|
|
|
return apply_resistance(1000, dt,
|
|
|
|
select_weapon(dt, false, true) ? select_weapon(dt, false, true)->type : 0,
|
|
|
|
select_armor(dt, false), select_armor(dt, true), true);
|
|
|
|
}
|
|
|
|
|
2015-11-03 14:22:50 +01:00
|
|
|
static void test_calculate_armor(CuTest * tc)
|
|
|
|
{
|
|
|
|
troop dt;
|
2016-08-29 09:04:43 +02:00
|
|
|
battle *b = NULL;
|
2015-11-03 14:22:50 +01:00
|
|
|
region *r;
|
|
|
|
unit *du;
|
2015-11-03 18:59:09 +01:00
|
|
|
weapon_type *wtype;
|
|
|
|
armor_type *ashield, *achain;
|
|
|
|
item_type *ibelt, *ishield, *ichain;
|
|
|
|
race *rc;
|
2017-02-24 20:47:47 +01:00
|
|
|
variant v50p = frac_make(1, 2);
|
2015-11-03 14:22:50 +01:00
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2015-11-03 18:59:09 +01:00
|
|
|
ibelt = it_get_or_create(rt_get_or_create("trollbelt"));
|
|
|
|
ishield = it_get_or_create(rt_get_or_create("shield"));
|
2017-02-24 20:47:47 +01:00
|
|
|
ashield = new_armortype(ishield, 0.0, v50p, 1, ATF_SHIELD);
|
2015-11-03 18:59:09 +01:00
|
|
|
ichain = it_get_or_create(rt_get_or_create("chainmail"));
|
2017-02-24 20:47:47 +01:00
|
|
|
achain = new_armortype(ichain, 0.0, v50p, 3, ATF_NONE);
|
2017-12-17 10:16:56 +01:00
|
|
|
wtype = new_weapontype(it_get_or_create(rt_get_or_create("sword")), 0, v50p, 0, 0, 0, 0, SK_MELEE);
|
2015-11-03 18:59:09 +01:00
|
|
|
rc = test_create_race("human");
|
|
|
|
du = test_create_unit(test_create_faction(rc), r);
|
2015-11-03 19:11:03 +01:00
|
|
|
dt.index = 0;
|
2015-11-03 18:59:09 +01:00
|
|
|
|
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "default ac", 0, test_armor(dt, 0, false));
|
|
|
|
|
|
|
|
CuAssertIntEquals_Msg(tc, "magres unmodified", 1000, test_resistance(dt));
|
2015-11-03 18:59:09 +01:00
|
|
|
free_battle(b);
|
|
|
|
|
2016-08-29 09:04:43 +02:00
|
|
|
b = NULL;
|
2015-11-03 18:59:09 +01:00
|
|
|
i_change(&du->items, ibelt, 1);
|
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2018-02-25 17:01:16 +01:00
|
|
|
CuAssertIntEquals_Msg(tc, "without natural armor", 0, natural_armor(du));
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "magical armor", 1, test_armor(dt, 0, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
rc->armor = 2;
|
2018-02-25 17:01:16 +01:00
|
|
|
CuAssertIntEquals_Msg(tc, "with natural armor", 2, natural_armor(du));
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "natural armor", 3, test_armor(dt, 0, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
rc->armor = 0;
|
|
|
|
free_battle(b);
|
|
|
|
|
2016-08-29 09:04:43 +02:00
|
|
|
b = NULL;
|
2015-11-03 18:59:09 +01:00
|
|
|
i_change(&du->items, ishield, 1);
|
|
|
|
i_change(&du->items, ichain, 1);
|
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2015-12-05 17:17:21 +01:00
|
|
|
rc->battle_flags &= ~BF_EQUIPMENT;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "require BF_EQUIPMENT", 1, test_armor(dt, 0, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
free_battle(b);
|
|
|
|
|
2016-08-29 09:04:43 +02:00
|
|
|
b = NULL;
|
2015-11-03 18:59:09 +01:00
|
|
|
rc->battle_flags |= BF_EQUIPMENT;
|
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "stack equipment rc", 5, test_armor(dt, 0, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
rc->armor = 2;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "natural armor adds 50%", 6, test_armor(dt, 0, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
wtype->flags = WTF_NONE;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "regular weapon has no effect", 6, test_armor(dt, wtype, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
wtype->flags = WTF_ARMORPIERCING;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "armor piercing weapon", 3, test_armor(dt, wtype, false));
|
2015-11-03 18:59:09 +01:00
|
|
|
wtype->flags = WTF_NONE;
|
|
|
|
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "magical attack", 3, test_armor(dt, wtype, true));
|
|
|
|
CuAssertIntEquals_Msg(tc, "magres unmodified", 1000,
|
|
|
|
test_resistance(dt));
|
2015-11-03 18:59:09 +01:00
|
|
|
|
|
|
|
ashield->flags |= ATF_LAEN;
|
|
|
|
achain->flags |= ATF_LAEN;
|
2018-09-11 17:17:47 +02:00
|
|
|
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "laen armor", 3, test_armor(dt, wtype, true));
|
|
|
|
CuAssertIntEquals_Msg(tc, "laen magres bonus", 250, test_resistance(dt));
|
2015-11-03 18:59:09 +01:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2015-11-03 14:22:50 +01:00
|
|
|
}
|
|
|
|
|
2015-12-05 17:17:21 +01:00
|
|
|
static void test_magic_resistance(CuTest *tc)
|
|
|
|
{
|
|
|
|
troop dt;
|
2016-08-29 09:04:43 +02:00
|
|
|
battle *b = NULL;
|
2015-12-05 17:17:21 +01:00
|
|
|
region *r;
|
|
|
|
unit *du;
|
|
|
|
armor_type *ashield, *achain;
|
2015-12-05 17:25:47 +01:00
|
|
|
item_type *ishield, *ichain;
|
2015-12-05 17:17:21 +01:00
|
|
|
race *rc;
|
2017-02-24 20:47:47 +01:00
|
|
|
variant magres;
|
|
|
|
variant v50p = frac_make(1, 2);
|
|
|
|
variant v10p = frac_make(1, 10);
|
2015-12-05 17:17:21 +01:00
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2015-12-05 17:17:21 +01:00
|
|
|
ishield = it_get_or_create(rt_get_or_create("shield"));
|
2017-02-24 20:47:47 +01:00
|
|
|
ashield = new_armortype(ishield, 0.0, v50p, 1, ATF_SHIELD);
|
2015-12-05 17:17:21 +01:00
|
|
|
ichain = it_get_or_create(rt_get_or_create("chainmail"));
|
2017-02-24 20:47:47 +01:00
|
|
|
achain = new_armortype(ichain, 0.0, v50p, 3, ATF_NONE);
|
2015-12-05 17:17:21 +01:00
|
|
|
rc = test_create_race("human");
|
|
|
|
du = test_create_unit(test_create_faction(rc), r);
|
|
|
|
dt.index = 0;
|
|
|
|
|
2017-02-24 20:47:47 +01:00
|
|
|
i_change(&du->items, ishield, 1);
|
2015-12-05 17:17:21 +01:00
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "no magres reduction", 1000, test_resistance(dt));
|
2017-02-24 20:47:47 +01:00
|
|
|
magres = magic_resistance(du);
|
|
|
|
CuAssertIntEquals_Msg(tc, "no magres reduction", 0, magres.sa[0]);
|
2015-12-05 17:17:21 +01:00
|
|
|
|
|
|
|
ashield->flags |= ATF_LAEN;
|
2017-02-24 20:47:47 +01:00
|
|
|
ashield->magres = v10p;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "laen reduction => 10%%", 900, test_resistance(dt));
|
|
|
|
CuAssertIntEquals_Msg(tc, "no magic, no resistance", 1000,
|
|
|
|
apply_resistance(1000, dt,
|
|
|
|
select_weapon(dt, false, true) ? select_weapon(dt, false, true)->type : 0,
|
|
|
|
select_armor(dt, false), select_armor(dt, true), false));
|
2015-12-05 17:17:21 +01:00
|
|
|
free_battle(b);
|
|
|
|
|
2016-08-29 09:04:43 +02:00
|
|
|
b = NULL;
|
2015-12-05 17:17:21 +01:00
|
|
|
i_change(&du->items, ichain, 1);
|
|
|
|
achain->flags |= ATF_LAEN;
|
2017-02-24 20:47:47 +01:00
|
|
|
achain->magres = v10p;
|
2015-12-05 17:17:21 +01:00
|
|
|
ashield->flags |= ATF_LAEN;
|
2017-02-24 20:47:47 +01:00
|
|
|
ashield->magres = v10p;
|
2015-12-05 17:17:21 +01:00
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "2x laen reduction => 81%%", 810, test_resistance(dt));
|
2015-12-05 17:17:21 +01:00
|
|
|
free_battle(b);
|
|
|
|
|
2016-08-29 09:04:43 +02:00
|
|
|
b = NULL;
|
2015-12-05 17:17:21 +01:00
|
|
|
i_change(&du->items, ishield, -1);
|
|
|
|
i_change(&du->items, ichain, -1);
|
|
|
|
set_level(du, SK_MAGIC, 2);
|
|
|
|
dt.fighter = setup_fighter(&b, du);
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "skill reduction => 90%%", 900, test_resistance(dt));
|
2017-02-24 20:47:47 +01:00
|
|
|
magres = magic_resistance(du);
|
|
|
|
CuAssert(tc, "skill reduction", frac_equal(magres, v10p));
|
|
|
|
rc->magres = v50p; /* percentage, gets added to skill bonus */
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "race reduction => 40%%", 400, test_resistance(dt));
|
2017-02-24 20:47:47 +01:00
|
|
|
magres = magic_resistance(du);
|
|
|
|
CuAssert(tc, "race bonus => 60%%", frac_equal(magres, frac_make(60, 100)));
|
2015-12-06 18:34:31 +01:00
|
|
|
|
2017-02-24 20:47:47 +01:00
|
|
|
rc->magres = frac_make(15, 10); /* 150% resistance should not cause negative damage multiplier */
|
|
|
|
magres = magic_resistance(du);
|
|
|
|
CuAssert(tc, "magic resistance is never > 0.9", frac_equal(magres, frac_make(9, 10)));
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "damage reduction is never < 0.1", 100, test_resistance(dt));
|
2015-12-05 17:17:21 +01:00
|
|
|
|
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2015-12-05 17:17:21 +01:00
|
|
|
}
|
|
|
|
|
2015-11-03 19:11:03 +01:00
|
|
|
static void test_projectile_armor(CuTest * tc)
|
|
|
|
{
|
|
|
|
troop dt;
|
2016-08-29 09:04:43 +02:00
|
|
|
battle *b = NULL;
|
2015-11-03 19:11:03 +01:00
|
|
|
region *r;
|
|
|
|
unit *du;
|
|
|
|
weapon_type *wtype;
|
|
|
|
armor_type *ashield, *achain;
|
|
|
|
item_type *ishield, *ichain;
|
|
|
|
race *rc;
|
2017-02-24 20:47:47 +01:00
|
|
|
variant v50p = frac_make(1, 2);
|
2015-11-03 19:11:03 +01:00
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
2015-11-03 19:11:03 +01:00
|
|
|
ishield = it_get_or_create(rt_get_or_create("shield"));
|
2017-02-24 20:47:47 +01:00
|
|
|
ashield = new_armortype(ishield, 0.0, v50p, 1, ATF_SHIELD);
|
2015-11-03 19:11:03 +01:00
|
|
|
ichain = it_get_or_create(rt_get_or_create("chainmail"));
|
2017-02-24 20:47:47 +01:00
|
|
|
achain = new_armortype(ichain, 0.0, v50p, 3, ATF_NONE);
|
2017-12-17 10:16:56 +01:00
|
|
|
wtype = new_weapontype(it_get_or_create(rt_get_or_create("sword")), 0, v50p, 0, 0, 0, 0, SK_MELEE);
|
2015-11-03 19:11:03 +01:00
|
|
|
rc = test_create_race("human");
|
|
|
|
rc->battle_flags |= BF_EQUIPMENT;
|
|
|
|
du = test_create_unit(test_create_faction(rc), r);
|
|
|
|
dt.index = 0;
|
|
|
|
|
|
|
|
i_change(&du->items, ishield, 1);
|
|
|
|
i_change(&du->items, ichain, 1);
|
|
|
|
dt.fighter = setup_fighter(&b, du);
|
|
|
|
wtype->flags = WTF_MISSILE;
|
|
|
|
achain->projectile = 1.0;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "projectile armor", -1, test_armor(dt, wtype, false));
|
2015-11-03 19:11:03 +01:00
|
|
|
achain->projectile = 0.0;
|
|
|
|
ashield->projectile = 1.0;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "projectile shield", -1, test_armor(dt, wtype, false));
|
2015-11-03 19:11:03 +01:00
|
|
|
wtype->flags = WTF_NONE;
|
2018-09-12 09:37:22 +02:00
|
|
|
CuAssertIntEquals_Msg(tc, "no projectiles", 4, test_armor(dt, wtype, false));
|
2015-11-03 19:11:03 +01:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2015-11-03 19:11:03 +01:00
|
|
|
}
|
|
|
|
|
2016-08-29 09:53:09 +02:00
|
|
|
static void test_battle_skilldiff(CuTest *tc)
|
2016-08-29 09:04:43 +02:00
|
|
|
{
|
|
|
|
troop ta, td;
|
|
|
|
region *r;
|
|
|
|
unit *ua, *ud;
|
|
|
|
battle *b = NULL;
|
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2016-08-29 09:53:09 +02:00
|
|
|
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
|
|
|
ud = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
ua = test_create_unit(test_create_faction(NULL), r);
|
2016-08-29 09:04:43 +02:00
|
|
|
td.fighter = setup_fighter(&b, ud);
|
|
|
|
td.index = 0;
|
|
|
|
ta.fighter = setup_fighter(&b, ua);
|
|
|
|
ta.index = 0;
|
|
|
|
CuAssertIntEquals(tc, 0, skilldiff(ta, td, 0));
|
|
|
|
|
|
|
|
ta.fighter->person[0].attack = 2;
|
|
|
|
td.fighter->person[0].defence = 1;
|
|
|
|
CuAssertIntEquals(tc, 1, skilldiff(ta, td, 0));
|
|
|
|
|
2016-08-29 09:53:09 +02:00
|
|
|
td.fighter->person[0].flags |= FL_SLEEPING;
|
|
|
|
CuAssertIntEquals(tc, 3, skilldiff(ta, td, 0));
|
|
|
|
|
2017-02-18 21:15:14 +01:00
|
|
|
/* TODO: unarmed halfling vs. dragon: +5 */
|
|
|
|
/* TODO: rule_goblin_bonus */
|
|
|
|
/* TODO: weapon modifiers, missiles, skill_formula */
|
2016-08-29 09:53:09 +02:00
|
|
|
|
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2016-08-29 09:53:09 +02:00
|
|
|
}
|
|
|
|
|
2018-07-15 13:39:23 +02:00
|
|
|
static void test_battle_report_one(CuTest *tc)
|
|
|
|
{
|
|
|
|
battle * b = NULL;
|
|
|
|
region *r;
|
|
|
|
unit *u1, *u2;
|
|
|
|
message *m;
|
|
|
|
const char *expect;
|
|
|
|
fighter *fig;
|
|
|
|
|
|
|
|
test_setup();
|
|
|
|
mt_create_va(mt_new("start_battle", NULL), "factions:string", MT_NEW_END);
|
|
|
|
r = test_create_plain(0, 0);
|
|
|
|
u1 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
u2 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
b = make_battle(r);
|
|
|
|
join_battle(b, u1, true, &fig);
|
|
|
|
join_battle(b, u2, false, &fig);
|
|
|
|
faction_setname(u1->faction, "Monster");
|
|
|
|
expect = factionname(u1->faction);
|
|
|
|
|
|
|
|
report_battle_start(b);
|
|
|
|
CuAssertPtrNotNull(tc, m = test_find_messagetype(u1->faction->battles->msgs, "start_battle"));
|
|
|
|
CuAssertStrEquals(tc, expect, (const char *)m->parameters[0].v);
|
|
|
|
|
|
|
|
free_battle(b);
|
|
|
|
test_teardown();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_battle_report_two(CuTest *tc)
|
|
|
|
{
|
|
|
|
battle * b = NULL;
|
|
|
|
region *r;
|
|
|
|
unit *u1, *u2;
|
|
|
|
message *m;
|
|
|
|
char expect[64];
|
|
|
|
fighter *fig;
|
|
|
|
struct locale *lang;
|
|
|
|
|
|
|
|
test_setup();
|
|
|
|
lang = test_create_locale();
|
|
|
|
locale_setstring(lang, "and", "and");
|
|
|
|
mt_create_va(mt_new("start_battle", NULL), "factions:string", MT_NEW_END);
|
|
|
|
r = test_create_plain(0, 0);
|
|
|
|
u1 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
u1->faction->locale = lang;
|
|
|
|
u2 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
u2->faction->locale = lang;
|
|
|
|
|
|
|
|
str_slprintf(expect, sizeof(expect), "%s and %s", factionname(u1->faction), factionname(u2->faction));
|
|
|
|
b = make_battle(r);
|
|
|
|
join_battle(b, u1, true, &fig);
|
|
|
|
join_battle(b, u2, true, &fig);
|
|
|
|
report_battle_start(b);
|
|
|
|
|
|
|
|
CuAssertPtrNotNull(tc, m = test_find_messagetype(u1->faction->battles->msgs, "start_battle"));
|
|
|
|
CuAssertStrEquals(tc, expect, (const char *)m->parameters[0].v);
|
|
|
|
|
|
|
|
free_battle(b);
|
|
|
|
test_teardown();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_battle_report_three(CuTest *tc)
|
|
|
|
{
|
|
|
|
battle * b = NULL;
|
|
|
|
region *r;
|
|
|
|
unit *u1, *u2, *u3;
|
|
|
|
message *m;
|
|
|
|
char expect[64];
|
|
|
|
fighter *fig;
|
|
|
|
struct locale *lang;
|
|
|
|
|
|
|
|
test_setup();
|
|
|
|
lang = test_create_locale();
|
|
|
|
locale_setstring(lang, "and", "and");
|
|
|
|
mt_create_va(mt_new("start_battle", NULL), "factions:string", MT_NEW_END);
|
|
|
|
r = test_create_plain(0, 0);
|
|
|
|
u1 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
u1->faction->locale = lang;
|
|
|
|
u2 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
u2->faction->locale = lang;
|
|
|
|
u3 = test_create_unit(test_create_faction(NULL), r);
|
|
|
|
u3->faction->locale = lang;
|
|
|
|
|
|
|
|
str_slprintf(expect, sizeof(expect), "%s, %s and %s", factionname(u1->faction), factionname(u2->faction), factionname(u3->faction));
|
|
|
|
b = make_battle(r);
|
|
|
|
join_battle(b, u1, true, &fig);
|
|
|
|
join_battle(b, u2, true, &fig);
|
|
|
|
join_battle(b, u3, true, &fig);
|
|
|
|
report_battle_start(b);
|
|
|
|
|
|
|
|
CuAssertPtrNotNull(tc, m = test_find_messagetype(u1->faction->battles->msgs, "start_battle"));
|
|
|
|
CuAssertStrEquals(tc, expect, (const char *)m->parameters[0].v);
|
|
|
|
|
|
|
|
free_battle(b);
|
|
|
|
test_teardown();
|
|
|
|
}
|
|
|
|
|
2016-08-29 09:53:09 +02:00
|
|
|
static void test_battle_skilldiff_building(CuTest *tc)
|
|
|
|
{
|
|
|
|
troop ta, td;
|
|
|
|
region *r;
|
|
|
|
unit *ua, *ud;
|
|
|
|
battle *b = NULL;
|
|
|
|
building_type *btype;
|
|
|
|
|
2017-02-22 20:16:51 +01:00
|
|
|
test_setup();
|
2017-02-22 19:38:46 +01:00
|
|
|
btype = setup_castle();
|
2016-08-29 09:53:09 +02:00
|
|
|
|
2018-01-14 09:38:26 +01:00
|
|
|
r = test_create_region(0, 0, NULL);
|
|
|
|
ud = test_create_unit(test_create_faction(NULL), r);
|
2016-08-29 09:53:09 +02:00
|
|
|
ud->building = test_create_building(ud->region, btype);
|
2018-01-14 09:38:26 +01:00
|
|
|
ua = test_create_unit(test_create_faction(NULL), r);
|
2016-08-29 09:53:09 +02:00
|
|
|
td.fighter = setup_fighter(&b, ud);
|
|
|
|
td.index = 0;
|
|
|
|
ta.fighter = setup_fighter(&b, ua);
|
|
|
|
ta.index = 0;
|
2018-05-01 15:32:06 +02:00
|
|
|
CuAssertIntEquals(tc, 0, buildingeffsize(ud->building, false));
|
2016-08-29 09:53:09 +02:00
|
|
|
CuAssertIntEquals(tc, 0, skilldiff(ta, td, 0));
|
|
|
|
|
2017-02-22 19:38:46 +01:00
|
|
|
ud->building->size = 10;
|
2018-05-01 15:32:06 +02:00
|
|
|
CuAssertIntEquals(tc, 1, buildingeffsize(ud->building, false));
|
2017-02-22 19:38:46 +01:00
|
|
|
CuAssertIntEquals(tc, -1, skilldiff(ta, td, 0));
|
2016-08-29 09:53:09 +02:00
|
|
|
|
2017-08-18 18:42:59 +02:00
|
|
|
create_curse(NULL, &ud->building->attribs, &ct_magicwalls, 1, 1, 1, 1);
|
2017-02-22 19:38:46 +01:00
|
|
|
CuAssertIntEquals(tc, -2, skilldiff(ta, td, 0));
|
2016-08-29 09:53:09 +02:00
|
|
|
|
2017-08-21 19:43:35 +02:00
|
|
|
create_curse(NULL, &ud->building->attribs, &ct_strongwall, 1, 1, 2, 1);
|
2017-02-22 19:38:46 +01:00
|
|
|
CuAssertIntEquals(tc, -4, skilldiff(ta, td, 0));
|
2016-08-29 09:53:09 +02:00
|
|
|
|
2016-08-29 09:04:43 +02:00
|
|
|
free_battle(b);
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2016-08-29 09:04:43 +02:00
|
|
|
}
|
|
|
|
|
2018-01-17 17:37:37 +01:00
|
|
|
static void assert_skill(CuTest *tc, const char *msg, unit *u, skill_t sk, int level, int week, int weekmax)
|
2017-07-31 12:55:05 +02:00
|
|
|
{
|
2017-12-27 19:58:39 +01:00
|
|
|
skill *sv = unit_skill(u, sk);
|
|
|
|
if (sv) {
|
2018-02-25 14:18:36 +01:00
|
|
|
char buf[256];
|
2017-12-27 19:58:39 +01:00
|
|
|
sprintf(buf, "%s level %d != %d", msg, sv->level, level);
|
2018-01-17 17:37:37 +01:00
|
|
|
CuAssertIntEquals_Msg(tc, buf, level, sv->level);
|
2017-12-27 19:58:39 +01:00
|
|
|
sprintf(buf, "%s week %d !<= %d !<= %d", msg, week, sv->weeks, weekmax);
|
2018-01-17 17:37:37 +01:00
|
|
|
CuAssert(tc, buf, sv->weeks >= week && sv->weeks <= weekmax);
|
2017-12-27 19:58:39 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
CuAssertIntEquals_Msg(tc, msg, level, 0);
|
|
|
|
CuAssertIntEquals_Msg(tc, msg, week, 0);
|
|
|
|
}
|
2017-07-31 12:55:05 +02:00
|
|
|
}
|
|
|
|
|
2017-12-27 19:58:39 +01:00
|
|
|
static void test_drain_exp(CuTest *tc)
|
2017-07-31 12:55:05 +02:00
|
|
|
{
|
|
|
|
unit *u;
|
2018-01-17 17:37:37 +01:00
|
|
|
const char *msg;
|
2017-07-31 12:55:05 +02:00
|
|
|
int i;
|
|
|
|
double rand;
|
2017-12-27 19:58:39 +01:00
|
|
|
|
2017-07-31 12:55:05 +02:00
|
|
|
test_setup();
|
|
|
|
config_set("study.random_progress", "0");
|
2018-01-14 09:38:26 +01:00
|
|
|
u = test_create_unit(test_create_faction(NULL), test_create_region(0, 0, NULL));
|
2017-07-31 12:55:05 +02:00
|
|
|
set_level(u, SK_STAMINA, 3);
|
2017-12-27 19:58:39 +01:00
|
|
|
|
2017-07-31 12:55:05 +02:00
|
|
|
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);
|
|
|
|
|
2017-12-27 19:58:39 +01:00
|
|
|
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);
|
2017-07-31 12:55:05 +02:00
|
|
|
|
2017-12-27 19:58:39 +01:00
|
|
|
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);
|
2017-07-31 12:55:05 +02:00
|
|
|
}
|
2017-12-27 19:58:39 +01:00
|
|
|
test_teardown();
|
2017-07-31 12:55:05 +02:00
|
|
|
}
|
|
|
|
|
2018-05-03 23:00:28 +02:00
|
|
|
static void test_tactics_chance(CuTest *tc) {
|
|
|
|
unit *u;
|
|
|
|
ship_type *stype;
|
|
|
|
|
|
|
|
test_setup();
|
|
|
|
u = test_create_unit(test_create_faction(NULL), test_create_ocean(0, 0));
|
|
|
|
CuAssertDblEquals(tc, 0.1, tactics_chance(u, 1), 0.01);
|
|
|
|
CuAssertDblEquals(tc, 0.3, tactics_chance(u, 3), 0.01);
|
|
|
|
stype = test_create_shiptype("brot");
|
|
|
|
u->ship = test_create_ship(u->region, stype);
|
|
|
|
CuAssertDblEquals(tc, 0.2, tactics_chance(u, 2), 0.01);
|
|
|
|
stype->tac_bonus = 2.0;
|
|
|
|
CuAssertDblEquals(tc, 0.4, tactics_chance(u, 2), 0.01);
|
|
|
|
test_teardown();
|
|
|
|
}
|
|
|
|
|
2011-03-13 02:01:20 +01:00
|
|
|
CuSuite *get_battle_suite(void)
|
|
|
|
{
|
2015-01-12 16:57:05 +01:00
|
|
|
CuSuite *suite = CuSuiteNew();
|
|
|
|
SUITE_ADD_TEST(suite, test_make_fighter);
|
2018-04-29 15:24:36 +02:00
|
|
|
SUITE_ADD_TEST(suite, test_select_weapon_restricted);
|
|
|
|
SUITE_ADD_TEST(suite, test_select_armor);
|
2016-08-29 09:04:43 +02:00
|
|
|
SUITE_ADD_TEST(suite, test_battle_skilldiff);
|
2016-08-29 09:53:09 +02:00
|
|
|
SUITE_ADD_TEST(suite, test_battle_skilldiff_building);
|
2018-07-15 13:39:23 +02:00
|
|
|
SUITE_ADD_TEST(suite, test_battle_report_one);
|
|
|
|
SUITE_ADD_TEST(suite, test_battle_report_two);
|
|
|
|
SUITE_ADD_TEST(suite, test_battle_report_three);
|
2015-01-12 16:57:05 +01:00
|
|
|
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);
|
2015-11-03 14:22:50 +01:00
|
|
|
SUITE_ADD_TEST(suite, test_calculate_armor);
|
2015-11-21 19:02:14 +01:00
|
|
|
SUITE_ADD_TEST(suite, test_natural_armor);
|
2015-12-05 17:17:21 +01:00
|
|
|
SUITE_ADD_TEST(suite, test_magic_resistance);
|
2015-11-03 19:11:03 +01:00
|
|
|
SUITE_ADD_TEST(suite, test_projectile_armor);
|
2018-05-03 23:00:28 +02:00
|
|
|
SUITE_ADD_TEST(suite, test_tactics_chance);
|
2018-01-17 19:23:10 +01:00
|
|
|
DISABLE_TEST(suite, test_drain_exp);
|
2015-01-12 16:57:05 +01:00
|
|
|
return suite;
|
2011-03-13 02:01:20 +01:00
|
|
|
}
|