Merge pull request #942 from eressea/expel

VERTREIBE Kommando
This commit is contained in:
Enno Rehling 2021-06-11 21:11:22 +02:00 committed by GitHub
commit 317f605ff2
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
18 changed files with 338 additions and 140 deletions

View file

@ -1,11 +1,11 @@
{ {
"keywords": { "keywords": {
"en" : { "en" : {
"plant": "PLANT", "plant": "PLANT",
"grow": [ "GROW", "BREED" ], "grow": [ "GROW", "BREED" ],
"promote": ["PROMOTE", "PROMOTION" ], "promote": ["PROMOTE", "PROMOTION" ],
"locale": ["LANGUAGE", "LOCALE"], "locale": ["LANGUAGE", "LOCALE"],
"combat": [ "COMBAT", "FIGHT" ] "combat": [ "COMBAT", "FIGHT" ]
}, },
"de": { "de": {
"//" : "//", "//" : "//",
@ -36,6 +36,7 @@
"maketemp": ["MACHE TEMP", "MACHETEMP"], "maketemp": ["MACHE TEMP", "MACHETEMP"],
"move" : "NACH", "move" : "NACH",
"password" : "PASSWORT", "password" : "PASSWORT",
"expel" : "VERTREIBE",
"loot" : ["PLÜNDERE", "PLÜNDERN"], "loot" : ["PLÜNDERE", "PLÜNDERN"],
"recruit": ["REKRUTIERE", "REKRUTIEREN"], "recruit": ["REKRUTIERE", "REKRUTIEREN"],
"reserve": ["RESERVIERE", "RESERVIEREN"], "reserve": ["RESERVIERE", "RESERVIEREN"],

View file

@ -4732,6 +4732,13 @@
<arg name="command" type="order"/> <arg name="command" type="order"/>
</type> </type>
</message> </message>
<message name="feedback_not_inside" section="errors">
<type>
<arg name="unit" type="unit"/>
<arg name="region" type="region"/>
<arg name="command" type="order"/>
</type>
</message>
<message name="feedback_unit_not_found" section="errors"> <message name="feedback_unit_not_found" section="errors">
<type> <type>
<arg name="unit" type="unit"/> <arg name="unit" type="unit"/>

View file

@ -590,6 +590,9 @@ msgstr "\"$unit($unit) in $region($region): '$order($command)' - Um in Gletscher
msgid "feedback_unit_not_found" msgid "feedback_unit_not_found"
msgstr "\"$unit($unit) in $region($region): '$order($command)' - Die Einheit wurde nicht gefunden.\"" msgstr "\"$unit($unit) in $region($region): '$order($command)' - Die Einheit wurde nicht gefunden.\""
msgid "feedback_not_inside"
msgstr "\"$unit($unit) in $region($region): '$order($command)' - Die Einheit kontrolliert kein Schiff oder Gebäude.\""
msgid "error206" msgid "error206"
msgstr "\"$unit($unit) in $region($region): '$order($command)' - Auf dem Gebäude liegt bereits so ein Zauber.\"" msgstr "\"$unit($unit) in $region($region): '$order($command)' - Auf dem Gebäude liegt bereits so ein Zauber.\""

View file

@ -590,6 +590,9 @@ msgstr "\"$unit($unit) in $region($region): '$order($command)' - You must build
msgid "feedback_unit_not_found" msgid "feedback_unit_not_found"
msgstr "\"$unit($unit) in $region($region): '$order($command)' - The unit could not be found.\"" msgstr "\"$unit($unit) in $region($region): '$order($command)' - The unit could not be found.\""
msgid "feedback_unit_not_found"
msgstr "\"$unit($unit) in $region($region): '$order($command)' - The unit is not the owner of a ship or building.\""
msgid "error206" msgid "error206"
msgstr "\"$unit($unit) in $region($region): '$order($command)' - There is alrady a spell on that building.\"" msgstr "\"$unit($unit) in $region($region): '$order($command)' - There is alrady a spell on that building.\""

View file

@ -2496,6 +2496,10 @@ msgctxt "keyword"
msgid "loot" msgid "loot"
msgstr "PLÜNDERE" msgstr "PLÜNDERE"
msgctxt "keyword"
msgid "expel"
msgstr "VERTREIBE"
msgctxt "calendar" msgctxt "calendar"
msgid "month_1" msgid "month_1"
msgstr "Feldsegen" msgstr "Feldsegen"

View file

@ -2167,6 +2167,10 @@ msgctxt "keyword"
msgid "loot" msgid "loot"
msgstr "loot" msgstr "loot"
msgctxt "keyword"
msgid "expel"
msgstr "EXPEL"
msgctxt "keyword" msgctxt "keyword"
msgid "guard" msgid "guard"
msgstr "GUARD" msgstr "GUARD"

View file

@ -3925,21 +3925,25 @@ void force_leave(region *r, battle *b) {
if (u->building) { if (u->building) {
uo = building_owner(u->building); uo = building_owner(u->building);
} }
if (u->ship && r->land) { else if (u->ship && r->land) {
uo = ship_owner(u->ship); uo = ship_owner(u->ship);
} }
if (uo && is_enemy(b, uo, u)) { else {
message *msg = NULL; continue;
if (u->building) { }
msg = msg_message("force_leave_building", "unit owner building", u, uo, u->building); if (is_enemy(b, uo, u)) {
if (leave(u, true)) {
message *msg;
if (uo->building) {
msg = msg_message("force_leave_building", "unit owner building", u, uo, uo->building);
}
else {
msg = msg_message("force_leave_ship", "unit owner ship", u, uo, uo->ship);
}
add_message(&u->faction->msgs, msg);
add_message(&uo->faction->msgs, msg);
msg_release(msg);
} }
else {
msg = msg_message("force_leave_ship", "unit owner ship", u, uo, u->ship);
}
if (msg) {
ADDMSG(&u->faction->msgs, msg);
}
leave(u, false);
} }
} }
} }

View file

@ -3939,11 +3939,12 @@ void init_processor(void)
add_proc_global(p, defaultorders, "Defaults setzen"); add_proc_global(p, defaultorders, "Defaults setzen");
} }
add_proc_global(p, demographics, "Nahrung, Seuchen, Wachstum, Wanderung"); add_proc_global(p, demographics, "Nahrung, Seuchen, Wachstum, Wanderung");
p += 10;
if (!keyword_disabled(K_SORT)) { if (!keyword_disabled(K_SORT)) {
p += 10; add_proc_region(p, do_sort, "Einheiten sortieren");
add_proc_global(p, restack_units, "Einheiten sortieren");
} }
add_proc_order(p, K_EXPEL, expel_cmd, 0, "Einheiten verjagen");
if (!keyword_disabled(K_NUMBER)) { if (!keyword_disabled(K_NUMBER)) {
add_proc_order(p, K_NUMBER, renumber_cmd, 0, "Neue Nummern (Einheiten)"); add_proc_order(p, K_NUMBER, renumber_cmd, 0, "Neue Nummern (Einheiten)");
p += 10; p += 10;
@ -4187,3 +4188,66 @@ int locale_cmd(unit * u, order * ord)
} }
return 0; return 0;
} }
static void expel_building(unit *u, unit *u2, order *ord) {
building *b = u->building;
if (u != building_owner(b)) {
/* error: must be the owner */
cmistake(u, ord, 5, MSG_EVENT);
}
else {
if (leave(u2, true)) {
message *msg = msg_message("force_leave_building", "owner unit building", u, u2, u->building);
add_message(&u->faction->msgs, msg);
add_message(&u2->faction->msgs, msg);
msg_release(msg);
}
}
}
static void expel_ship(unit *u, unit *u2, order *ord) {
ship *sh = u->ship;
if (u != ship_owner(sh)) {
/* error: must be the owner */
cmistake(u, ord, 146, MSG_EVENT);
}
else if (!u->region->land) {
/* error: must not be at sea */
ADDMSG(&u->faction->msgs,
msg_feedback(u, ord, "error_onlandonly", NULL));
}
else {
if (leave(u2, true)) {
message *msg = msg_message("force_leave_ship", "owner unit ship", u, u2, u->ship);
add_message(&u->faction->msgs, msg);
add_message(&u2->faction->msgs, msg);
msg_release(msg);
}
}
}
int expel_cmd(unit *u, order *ord) {
faction *f = u->faction;
unit *u2;
init_order(ord, f->locale);
getunit(u->region, u->faction, &u2);
if (u2 == NULL) {
/* error: target unit not found */
ADDMSG(&u->faction->msgs,
msg_feedback(u, ord, "feedback_unit_not_found", NULL));
return 0;
}
if (u->building) {
expel_building(u, u2, ord);
}
else if (u->ship) {
expel_ship(u, u2, ord);
}
else {
ADDMSG(&u->faction->msgs,
msg_feedback(u, ord, "feedback_not_inside", NULL));
/* error: unit must be owner of a ship or building */
}
return 0;
}

View file

@ -50,6 +50,7 @@ extern "C" {
bool long_order_allowed(const struct unit *u, bool flags_only); bool long_order_allowed(const struct unit *u, bool flags_only);
bool password_wellformed(const char *password); bool password_wellformed(const char *password);
int expel_cmd(struct unit *u, struct order *ord);
int locale_cmd(struct unit *u, struct order *ord); int locale_cmd(struct unit *u, struct order *ord);
int password_cmd(struct unit *u, struct order *ord); int password_cmd(struct unit *u, struct order *ord);
int banner_cmd(struct unit *u, struct order *ord); int banner_cmd(struct unit *u, struct order *ord);

View file

@ -173,6 +173,95 @@ static void test_enter_ship(CuTest * tc)
test_teardown(); test_teardown();
} }
static void test_expel_building(CuTest *tc) {
unit *u1, *u2;
order *ord;
building *b;
test_setup();
u1 = test_create_unit(test_create_faction(), test_create_plain(0, 0));
u2 = test_create_unit(test_create_faction(), u1->region);
ord = create_order(K_EXPEL, u2->faction->locale, "%s", itoa36(u1->no));
expel_cmd(u2, ord);
CuAssertPtrNotNull(tc, test_find_messagetype(u2->faction->msgs, "feedback_not_inside"));
test_clear_messages(u2->faction);
b = u2->building = u1->building = test_create_building(u1->region, NULL);
CuAssertPtrEquals(tc, u1, building_owner(b));
expel_cmd(u2, ord);
/* Nothing happened: */
CuAssertPtrEquals(tc, u1, building_owner(b));
CuAssertPtrEquals(tc, b, u1->building);
CuAssertPtrEquals(tc, b, u2->building);
CuAssertPtrNotNull(tc, test_find_messagetype(u2->faction->msgs, "error5"));
test_clear_messages(u1->faction);
free_order(ord);
ord = create_order(K_EXPEL, u1->faction->locale, "%s", itoa36(u2->no));
expel_cmd(u1, ord);
/* owner has expelled u2: */
CuAssertPtrEquals(tc, NULL, u2->building);
CuAssertPtrNotNull(tc, test_find_messagetype(u1->faction->msgs, "force_leave_building"));
CuAssertPtrNotNull(tc, test_find_messagetype(u2->faction->msgs, "force_leave_building"));
test_teardown();
}
static void test_expel_ship(CuTest *tc) {
unit *u1, *u2;
order *ord;
ship *sh;
test_setup();
u1 = test_create_unit(test_create_faction(), test_create_plain(0, 0));
u2 = test_create_unit(test_create_faction(), u1->region);
ord = create_order(K_EXPEL, u2->faction->locale, "%s", itoa36(u1->no));
expel_cmd(u2, ord);
CuAssertPtrNotNull(tc, test_find_messagetype(u2->faction->msgs, "feedback_not_inside"));
test_clear_messages(u2->faction);
sh = u2->ship = u1->ship = test_create_ship(u1->region, NULL);
CuAssertPtrEquals(tc, u1, ship_owner(sh));
expel_cmd(u2, ord);
/* Nothing happened: */
CuAssertPtrEquals(tc, u1, ship_owner(sh));
CuAssertPtrEquals(tc, sh, u1->ship);
CuAssertPtrEquals(tc, sh, u2->ship);
CuAssertPtrNotNull(tc, test_find_messagetype(u2->faction->msgs, "error146"));
test_clear_messages(u2->faction);
free_order(ord);
ord = create_order(K_EXPEL, u1->faction->locale, "%s", itoa36(u2->no));
expel_cmd(u1, ord);
/* owner has expelled u2: */
CuAssertPtrEquals(tc, NULL, u2->ship);
CuAssertPtrEquals(tc, sh, leftship(u2));
CuAssertPtrNotNull(tc, test_find_messagetype(u1->faction->msgs, "force_leave_ship"));
CuAssertPtrNotNull(tc, test_find_messagetype(u2->faction->msgs, "force_leave_ship"));
test_teardown();
}
static void test_expel_ship_at_sea(CuTest *tc) {
unit *u1, *u2;
order *ord;
ship *sh;
test_setup();
u1 = test_create_unit(test_create_faction(), test_create_ocean(0, 0));
u2 = test_create_unit(test_create_faction(), u1->region);
sh = u2->ship = u1->ship = test_create_ship(u1->region, NULL);
CuAssertPtrEquals(tc, u1, ship_owner(sh));
ord = create_order(K_EXPEL, u1->faction->locale, "%s", itoa36(u2->no));
expel_cmd(u1, ord);
/* owner has not expelled u2: */
CuAssertPtrNotNull(tc, test_find_messagetype(u1->faction->msgs, "error_onlandonly"));
CuAssertPtrEquals(tc, sh, u2->ship);
test_teardown();
}
static void test_display_cmd(CuTest *tc) { static void test_display_cmd(CuTest *tc) {
unit *u; unit *u;
faction *f; faction *f;
@ -2320,6 +2409,9 @@ CuSuite *get_laws_suite(void)
SUITE_ADD_TEST(suite, test_enter_building); SUITE_ADD_TEST(suite, test_enter_building);
SUITE_ADD_TEST(suite, test_enter_ship); SUITE_ADD_TEST(suite, test_enter_ship);
SUITE_ADD_TEST(suite, test_display_cmd); SUITE_ADD_TEST(suite, test_display_cmd);
SUITE_ADD_TEST(suite, test_expel_building);
SUITE_ADD_TEST(suite, test_expel_ship);
SUITE_ADD_TEST(suite, test_expel_ship_at_sea);
SUITE_ADD_TEST(suite, test_rule_force_leave); SUITE_ADD_TEST(suite, test_rule_force_leave);
SUITE_ADD_TEST(suite, test_force_leave_buildings); SUITE_ADD_TEST(suite, test_force_leave_buildings);
SUITE_ADD_TEST(suite, test_force_leave_ships); SUITE_ADD_TEST(suite, test_force_leave_ships);

View file

@ -95,51 +95,55 @@ static void handle_unit(void *userData, int no) {
static void handle_order(void *userData, const char *str) { static void handle_order(void *userData, const char *str) {
parser_state *state = (parser_state *)userData; parser_state *state = (parser_state *)userData;
const char * tok, *input = str; const char * tok, *input;
char buffer[64]; char buffer[64];
const struct locale *lang; const struct locale *lang;
param_t p;
faction * f = state->f; faction * f = state->f;
lang = f ? f->locale : default_locale; lang = f ? f->locale : default_locale;
ltrim(&str);
if (*str == 0) return;
input = str;
tok = parse_token(&input, buffer, sizeof(buffer)); tok = parse_token(&input, buffer, sizeof(buffer));
p = findparam(tok, lang); if (tok) {
if (p == P_FACTION || p == P_GAMENAME) { param_t p = findparam(tok, lang);
tok = parse_token(&input, buffer, sizeof(buffer)); if (p == P_FACTION || p == P_GAMENAME) {
if (tok) {
int no = atoi36(tok);
tok = parse_token(&input, buffer, sizeof(buffer)); tok = parse_token(&input, buffer, sizeof(buffer));
handle_faction(userData, no, tok); if (tok) {
int no = atoi36(tok);
tok = parse_token(&input, buffer, sizeof(buffer));
handle_faction(userData, no, tok);
}
else {
/* TODO: log_error() */
}
} }
else { else if (p == P_UNIT) {
/* TODO: log_error() */ tok = parse_token(&input, buffer, sizeof(buffer));
if (tok) {
int no = atoi36(tok);
handle_unit(userData, no);
}
} }
} else if (p == P_NEXT) {
else if (p == P_UNIT) { state->f = NULL;
tok = parse_token(&input, buffer, sizeof(buffer)); state->u = NULL;
if (tok) { state->next_order = NULL;
int no = atoi36(tok);
handle_unit(userData, no);
} }
} else if (p == P_REGION) {
else if (p == P_NEXT) { state->u = NULL;
state->f = NULL; state->next_order = NULL;
state->u = NULL;
state->next_order = NULL;
}
else if (p == P_REGION) {
state->u = NULL;
state->next_order = NULL;
}
else if (state->u) {
unit * u = state->u;
order * ord = parse_order(str, lang);
if (ord) {
*state->next_order = ord;
state->next_order = &ord->next;
} }
else { else if (state->u) {
ADDMSG(&u->faction->msgs, msg_message("parse_error", "unit command", u, str)); unit *u = state->u;
order *ord = parse_order(str, lang);
if (ord) {
*state->next_order = ord;
state->next_order = &ord->next;
}
else {
ADDMSG(&u->faction->msgs, msg_message("parse_error", "unit command", u, str));
}
} }
} }
} }

View file

@ -13,7 +13,7 @@
#include "util/param.h" #include "util/param.h"
#include "util/parser.h" #include "util/parser.h"
void sort_before(unit *v, unit **up) { static void sort_before(unit *v, unit **up) {
unit *u = *up; unit *u = *up;
region *r = u->region; region *r = u->region;
unit **vp = &r->units; unit **vp = &r->units;
@ -24,102 +24,107 @@ void sort_before(unit *v, unit **up) {
u->next = v; u->next = v;
} }
void restack_units(void) void do_sort(region *r)
{ {
region *r; unit **up = &r->units;
for (r = regions; r; r = r->next) { bool sorted = false;
unit **up = &r->units; while (*up) {
bool sorted = false; unit *u = *up;
while (*up) { if (!fval(u, UFL_MARK) && !is_paused(u->faction)) {
unit *u = *up; struct order *ord;
if (!fval(u, UFL_MARK) && !is_paused(u->faction)) { for (ord = u->orders; ord; ord = ord->next) {
struct order *ord; if (getkeyword(ord) == K_SORT) {
for (ord = u->orders; ord; ord = ord->next) { char token[128];
if (getkeyword(ord) == K_SORT) { const char *s;
char token[128]; param_t p;
const char *s; int id;
param_t p; unit *v;
int id;
unit *v;
init_order(ord, NULL); init_order(ord, NULL);
s = gettoken(token, sizeof(token)); s = gettoken(token, sizeof(token));
p = findparam(s, u->faction->locale); p = findparam(s, u->faction->locale);
id = getid(); id = getid();
v = findunit(id); v = findunit(id);
if (v == u) { if (v == u) {
syntax_error(u, ord); syntax_error(u, ord);
} }
else if (!v || v->region != r) { else if (!v || v->region != r) {
cmistake(u, ord, 258, MSG_EVENT); cmistake(u, ord, 258, MSG_EVENT);
} }
else if (v->faction != u->faction && !is_paused(v->faction)) { else if (v->faction != u->faction && !is_paused(v->faction)) {
cmistake(u, ord, 258, MSG_EVENT); cmistake(u, ord, 258, MSG_EVENT);
} }
else if (v->building != u->building || v->ship != u->ship) { else if (v->building != u->building || v->ship != u->ship) {
cmistake(u, ord, 259, MSG_EVENT); cmistake(u, ord, 259, MSG_EVENT);
} }
else if (u->building && building_owner(u->building) == u) { else if (u->building && building_owner(u->building) == u) {
cmistake(u, ord, 260, MSG_EVENT); cmistake(u, ord, 260, MSG_EVENT);
} }
else if (u->ship && ship_owner(u->ship) == u) { else if (u->ship && ship_owner(u->ship) == u) {
cmistake(u, ord, 260, MSG_EVENT); cmistake(u, ord, 260, MSG_EVENT);
} }
else { else {
switch (p) { switch (p) {
case P_AFTER: case P_AFTER:
*up = u->next; *up = u->next;
u->next = v->next; u->next = v->next;
v->next = u; v->next = u;
fset(u, UFL_MARK); fset(u, UFL_MARK);
sorted = true; sorted = true;
break; break;
case P_BEFORE: case P_BEFORE:
if (v->ship && ship_owner(v->ship) == v) { if (v->ship && ship_owner(v->ship) == v) {
if (is_paused(v->faction)) { if (is_paused(v->faction)) {
sort_before(v, up); sort_before(v, up);
ship_set_owner(u); ship_set_owner(u);
}
else {
cmistake(v, ord, 261, MSG_EVENT);
break;
}
}
else if (v->building && building_owner(v->building) == v) {
if (is_paused(v->faction)) {
sort_before(v, up);
building_set_owner(u);
}
else {
cmistake(v, ord, 261, MSG_EVENT);
break;
}
} }
else { else {
sort_before(v, up); cmistake(v, ord, 261, MSG_EVENT);
break;
} }
fset(u, UFL_MARK);
sorted = true;
break;
default:
/* TODO: syntax error message? */
break;
} }
else if (v->building && building_owner(v->building) == v) {
if (is_paused(v->faction)) {
sort_before(v, up);
building_set_owner(u);
}
else {
cmistake(v, ord, 261, MSG_EVENT);
break;
}
}
else {
sort_before(v, up);
}
fset(u, UFL_MARK);
sorted = true;
break;
default:
/* TODO: syntax error message? */
break;
} }
break;
} }
break;
} }
} }
if (u == *up)
up = &u->next;
} }
if (sorted) { if (u == *up)
unit *u; up = &u->next;
for (u = r->units; u; u = u->next) { }
freset(u, UFL_MARK); if (sorted) {
} unit *u;
for (u = r->units; u; u = u->next) {
freset(u, UFL_MARK);
} }
} }
} }
void restack_units(void)
{
region *r;
for (r = regions; r; r = r->next) {
do_sort(r);
}
}

View file

@ -1,3 +1,6 @@
#pragma once #pragma once
struct region;
void restack_units(void); void restack_units(void);
void do_sort(struct region *r);

View file

@ -119,7 +119,7 @@ int spy_cmd(unit * u, struct order *ord)
if (!target) { if (!target) {
ADDMSG(&u->faction->msgs, ADDMSG(&u->faction->msgs,
msg_feedback(u, u->thisorder, "feedback_unit_not_found", "")); msg_feedback(u, u->thisorder, "feedback_unit_not_found", NULL));
return 0; return 0;
} }
if (effskill(u, SK_SPY, NULL) < 1) { if (effskill(u, SK_SPY, NULL) < 1) {

View file

@ -156,7 +156,8 @@ const char *keywords[MAXKEYWORDS] = {
"promote", "promote",
"pay", "pay",
"loot", "loot",
"expel",
"autostudy", "autostudy",
"locale", "locale"
}; };

View file

@ -71,6 +71,7 @@ extern "C"
K_PROMOTION, K_PROMOTION,
K_PAY, K_PAY,
K_LOOT, K_LOOT,
K_EXPEL,
K_AUTOSTUDY, K_AUTOSTUDY,
K_LOCALE, K_LOCALE,
MAXKEYWORDS, MAXKEYWORDS,

View file

@ -26,7 +26,7 @@ static parse_state *states;
#define TRIMMED(wc) (iswspace(wc) || iswcntrl(wc) || (wc) == 160 || (wc) == 8199 || (wc) == 8239) #define TRIMMED(wc) (iswspace(wc) || iswcntrl(wc) || (wc) == 160 || (wc) == 8199 || (wc) == 8239)
static int ltrim(const char **str_p) int ltrim(const char **str_p)
{ {
int ret = 0; int ret = 0;
wint_t wc; wint_t wc;

View file

@ -22,6 +22,7 @@ extern "C" {
int getint(void); int getint(void);
int getid(void); int getid(void);
unsigned int atoip(const char *s); unsigned int atoip(const char *s);
int ltrim(const char **str_p);
#ifdef __cplusplus #ifdef __cplusplus
} }