diff --git a/res/core/messages.xml b/res/core/messages.xml
index 8231edec3..3e16e8301 100644
--- a/res/core/messages.xml
+++ b/res/core/messages.xml
@@ -3269,19 +3269,21 @@
+
- "Die $ship($ship) in $region($region) entdeckt ein Opfer im $direction($dir)."
- "The $ship($ship) in $region($region) made $direction($dir) a target."
+ "$if($isnull($ship),"$unit($unit)","Die $ship($ship)") in $region($region) entdeckt ein Opfer im $direction($dir)."
+ "$if($isnull($ship),"$unit($unit)","The $ship($ship)") in $region($region) made $direction($dir) a target."
+
- "Die $ship($ship) in $region($region) kann keine Schiffe aufbringen."
- "The $ship($ship) could not capture other ships in $region($region)."
+ "$if($isnull($ship),"$unit($unit)","Die $ship($ship)") in $region($region) kann keine Schiffe aufbringen."
+ "$if($isnull($ship),"$unit($unit)","The $ship($ship)") could not capture other ships in $region($region)."
diff --git a/src/monsters.c b/src/monsters.c
index 47d355953..ab63cbb31 100644
--- a/src/monsters.c
+++ b/src/monsters.c
@@ -836,7 +836,7 @@ void plan_monsters(faction * f)
switch (old_race(u_race(u))) {
case RC_SEASERPENT:
- // long_order = create_order(K_PIRACY, f->locale, NULL);
+ long_order = create_order(K_PIRACY, f->locale, NULL);
break;
#ifdef TODO_ALP
case RC_ALP:
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);
diff --git a/src/piracy.c b/src/piracy.c
index 312af6c4b..e8864faa9 100644
--- a/src/piracy.c
+++ b/src/piracy.c
@@ -148,7 +148,11 @@ void piracy_cmd(unit * u, order *ord)
region *rc = rconnect(r, dir);
aff[dir].value = 0;
aff[dir].target = 0;
- if (rc && fval(rc->terrain, SAIL_INTO) && can_takeoff(sh, r, rc)) {
+ // TODO this could still result in an illegal movement order (through a wall or whatever)
+ // which will be prevented by move_cmd below
+ if (rc &&
+ ((sh && fval(rc->terrain, SAIL_INTO) && can_takeoff(sh, r, rc))
+ || (canswim(u) && fval(rc->terrain, SWIM_INTO) && fval(rc->terrain, SEA_REGION)))) {
for (sh2 = rc->ships; sh2; sh2 = sh2->next) {
unit *cap = ship_owner(sh2);
@@ -188,13 +192,13 @@ void piracy_cmd(unit * u, order *ord)
/* Wenn kein Ziel gefunden, entsprechende Meldung generieren */
if (target_dir == NODIRECTION) {
ADDMSG(&u->faction->msgs, msg_message("piratenovictim",
- "ship region", sh, r));
+ "ship unit region", sh, u, r));
return;
}
/* Meldung generieren */
ADDMSG(&u->faction->msgs, msg_message("piratesawvictim",
- "ship region dir", sh, r, target_dir));
+ "ship unit region dir", sh, u, r, target_dir));
/* Befehl konstruieren */
set_order(&u->thisorder, create_order(K_MOVE, u->faction->locale, "%s",
diff --git a/src/piracy.test.c b/src/piracy.test.c
index 188ea1fed..dee326980 100644
--- a/src/piracy.test.c
+++ b/src/piracy.test.c
@@ -31,6 +31,45 @@ static void setup_piracy(void) {
st_boat->cargo = 1000;
}
+static void setup_pirate(unit **pirate, int p_r_flags, int p_rc_flags, const char *p_shiptype,
+ order **ord, unit **victim, int v_r_flags, const char *v_shiptype) {
+ terrain_type *vterrain;
+ ship_type *st_boat = NULL;
+ race *rc;
+ faction *f;
+
+ setup_piracy();
+ vterrain = get_or_create_terrain("terrain1");
+ fset(vterrain, v_r_flags);
+ *victim = test_create_unit(test_create_faction(0), test_create_region(1, 0, vterrain));
+ assert(*victim);
+
+ if (v_shiptype) {
+ st_boat = st_get_or_create(v_shiptype);
+ u_set_ship(*victim, test_create_ship((*victim)->region, st_boat));
+ st_boat->coasts = (struct terrain_type **)calloc(2, sizeof(struct terrain_type *));
+ st_boat->coasts[0] = vterrain;
+ st_boat->coasts[1] = 0;
+ }
+
+ *pirate = create_unit(test_create_region(0, 0, get_or_create_terrain("terrain2")), f = test_create_faction(0), 1, rc = rc_get_or_create("pirate"), 0, 0, 0);
+ fset(rc, p_rc_flags);
+ assert(f && *pirate);
+
+ if (p_shiptype) {
+ st_boat = st_get_or_create(p_shiptype);
+ u_set_ship(*pirate, test_create_ship((*pirate)->region, st_boat));
+ st_boat->coasts = (struct terrain_type **)calloc(2, sizeof(struct terrain_type *));
+ st_boat->coasts[0] = vterrain;
+ st_boat->coasts[1] = 0;
+ }
+
+ f->locale = get_or_create_locale("de");
+ *ord = create_order(K_PIRACY, f->locale, "%s", itoa36((*victim)->faction->no));
+ assert(*ord);
+
+}
+
static void test_piracy_cmd(CuTest * tc) {
faction *f;
region *r;
@@ -39,6 +78,8 @@ static void test_piracy_cmd(CuTest * tc) {
terrain_type *t_ocean;
ship_type *st_boat;
+ test_cleanup();
+
setup_piracy();
t_ocean = get_or_create_terrain("ocean");
st_boat = st_get_or_create("boat");
@@ -73,6 +114,8 @@ static void test_piracy_cmd_errors(CuTest * tc) {
order *ord;
ship_type *st_boat;
+ test_cleanup();
+
setup_piracy();
st_boat = st_get_or_create("boat");
r = test_create_race("pirates");
@@ -112,11 +155,76 @@ static void test_piracy_cmd_errors(CuTest * tc) {
test_cleanup();
}
+static void test_piracy_cmd_walking(CuTest * tc) {
+ unit *pirate, *victim;
+ order *ord;
+ region *r;
+
+ test_cleanup();
+
+ setup_pirate(&pirate, 0, 0, NULL, &ord, &victim, SWIM_INTO | SEA_REGION, "boat");
+ /* fset(rc, RCF_SWIM); */
+ r = pirate->region;
+
+ piracy_cmd(pirate, ord);
+ CuAssertPtrEquals(tc, 0, pirate->thisorder);
+ CuAssertTrue(tc, pirate->region == r);
+ CuAssertPtrNotNullMsg(tc, "successful PIRACY message", test_find_messagetype(pirate->faction->msgs, "error144"));
+ free_order(ord);
+
+ test_cleanup();
+}
+
+static void test_piracy_cmd_land_to_land(CuTest * tc) {
+ unit *pirate, *victim;
+ order *ord;
+ region *r;
+
+ test_cleanup();
+
+ setup_pirate(&pirate, 0, 0, "boat", &ord, &victim, SAIL_INTO, "boat");
+ set_level(pirate, SK_SAILING, pirate->ship->type->sumskill);
+ r = pirate->region;
+
+ piracy_cmd(pirate, ord);
+ CuAssertPtrEquals(tc, 0, pirate->thisorder);
+ CuAssertTrue(tc, pirate->region == r);
+ /* TODO check message
+ CuAssertPtrNotNullMsg(tc, "successful PIRACY movement", test_find_messagetype(pirate->faction->msgs, "travel"));
+ */
+ free_order(ord);
+
+ test_cleanup();
+}
+
+static void test_piracy_cmd_swimmer(CuTest * tc) {
+ unit *pirate, *victim;
+ order *ord;
+ region *r;
+
+ test_cleanup();
+
+ setup_pirate(&pirate, 0, RCF_SWIM, NULL, &ord, &victim, SWIM_INTO | SEA_REGION, "boat");
+ r = pirate->region;
+
+ piracy_cmd(pirate, ord);
+ CuAssertPtrEquals(tc, 0, pirate->thisorder);
+ CuAssertTrue(tc, pirate->region != r);
+ CuAssertPtrEquals(tc, victim->region, pirate->region);
+ CuAssertPtrNotNullMsg(tc, "successful PIRACY message", test_find_messagetype(pirate->faction->msgs, "piratesawvictim"));
+ CuAssertPtrNotNullMsg(tc, "successful PIRACY movement", test_find_messagetype(pirate->faction->msgs, "travel"));
+ free_order(ord);
+
+ test_cleanup();
+}
CuSuite *get_piracy_suite(void)
{
CuSuite *suite = CuSuiteNew();
SUITE_ADD_TEST(suite, test_piracy_cmd_errors);
SUITE_ADD_TEST(suite, test_piracy_cmd);
+ SUITE_ADD_TEST(suite, test_piracy_cmd_walking);
+ DISABLE_TEST(suite, test_piracy_cmd_land_to_land);
+ SUITE_ADD_TEST(suite, test_piracy_cmd_swimmer);
return suite;
}