server/src/common/kernel/movement.c

2302 lines
59 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
* Eressea PB(E)M host Copyright (C) 1998-2003
2001-01-25 10:37:55 +01:00
* Christian Schlittchen (corwin@amber.kn-bremen.de)
* Katja Zedel (katze@felidae.kn-bremen.de)
* Henning Peters (faroul@beyond.kn-bremen.de)
* Enno Rehling (enno@eressea-pbem.de)
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* based on:
*
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
* This program may not be sold or used commercially without prior written
* permission from the authors.
*/
#include <config.h>
#include "eressea.h"
#include "movement.h"
#include "alchemy.h"
#include "border.h"
#include "build.h"
#include "building.h"
#include "curse.h"
2001-01-25 10:37:55 +01:00
#include "faction.h"
#include "item.h"
#include "karma.h"
2001-01-25 10:37:55 +01:00
#include "magic.h"
#include "message.h"
#include "plane.h"
2001-01-25 10:37:55 +01:00
#include "race.h"
#include "region.h"
#include "render.h"
#include "spell.h"
2001-01-25 10:37:55 +01:00
#include "ship.h"
#include "skill.h"
#include "teleport.h"
#include "unit.h"
2001-01-25 10:37:55 +01:00
/* util includes */
#include <util/goodies.h>
#include <util/language.h>
#include <util/rand.h>
2001-01-25 10:37:55 +01:00
/* libc includes */
#include <assert.h>
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* attributes includes */
#include <attributes/follow.h>
#include <attributes/targetregion.h>
#include <attributes/at_movement.h>
2001-01-25 10:37:55 +01:00
/* TODO: boder_type::move() must be able to change target (wisps) */
extern border_type bt_wisps;
extern item_type it_demonseye;
/* ------------------------------------------------------------- */
typedef struct traveldir {
int no;
direction_t dir;
int age;
} traveldir;
static attrib_type at_traveldir = {
2001-01-25 10:37:55 +01:00
"traveldir",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE, /* Weil normales Aging an ung<6E>nstiger Stelle */
DEFAULT_WRITE,
DEFAULT_READ
};
static void
2001-01-25 10:37:55 +01:00
a_traveldir_new_init(attrib *a)
{
a->data.v = calloc(1, sizeof(traveldir));
}
static void
2001-01-25 10:37:55 +01:00
a_traveldir_new_finalize(attrib *a)
{
free(a->data.v);
}
static int
2001-01-25 10:37:55 +01:00
a_traveldir_new_age(attrib *a)
{
traveldir *t = (traveldir *)(a->data.v);
t->age--;
if(t->age == 0) return 0;
return 1;
}
static int
2001-01-25 10:37:55 +01:00
a_traveldir_new_read(attrib *a, FILE *f)
{
traveldir *t = (traveldir *)(a->data.v);
int no, age, dir;
2001-01-25 10:37:55 +01:00
fscanf(f, "%d %d %d", &no, &dir, &age);
t->no = no;
t->dir = (direction_t)dir;
t->age = age;
return AT_READ_OK;
2001-01-25 10:37:55 +01:00
}
static void
2001-01-25 10:37:55 +01:00
a_traveldir_new_write(const attrib *a, FILE *f)
{
traveldir *t = (traveldir *)(a->data.v);
fprintf(f, "%d %d %d ", t->no, (int)t->dir, t->age);
}
attrib_type at_traveldir_new = {
"traveldir_new",
a_traveldir_new_init,
a_traveldir_new_finalize,
a_traveldir_new_age,
a_traveldir_new_write,
a_traveldir_new_read
};
static int
age_speedup(attrib *a)
{
if(a->data.i > 0) {
--a->data.i;
}
return a->data.i;
}
attrib_type at_speedup = {
"speedup",
NULL, NULL,
age_speedup,
a_writedefault,
a_readdefault
};
2001-01-25 10:37:55 +01:00
/* ------------------------------------------------------------- */
direction_t
getdirection(const struct locale * lang)
2001-01-25 10:37:55 +01:00
{
return finddirection(getstrtoken(), lang);
2001-01-25 10:37:55 +01:00
}
/* ------------------------------------------------------------- */
static attrib_type at_driveweight = {
"driveweight", NULL, NULL, NULL, NULL, NULL
};
static boolean
entrance_allowed(const struct unit * u, const struct region * r)
{
#ifdef REGIONOWNERS
unit * owner = region_owner(r);
if (owner==NULL || u->faction==owner->faction) return true;
if (alliedunit(owner, u->faction, HELP_TRAVEL)) return true;
if (is_enemy(u->faction, owner->faction)) return true;
return false;
#endif
return true;
}
int
personcapacity(const unit *u)
{
#if RACE_CAPACITY
int cap = u->race->weight+u->race->capacity;
#else
int cap = u->race->weight+540;
if (old_race(u->race) == RC_TROLL)
cap += 540;
#if RACE_ADJUSTMENTS
else if(old_race(u->race) == RC_GOBLIN)
cap -= 100;
#endif
#endif
if (fspecial(u->faction, FS_QUICK))
cap -= 200;
return cap;
}
2001-01-25 10:37:55 +01:00
static int
eff_weight(const unit *u)
{
attrib *a = a_find(u->attribs, &at_driveweight);
if (a) return weight(u) + a->data.i;
return weight(u);
}
static int
2001-01-25 10:37:55 +01:00
ridingcapacity(unit * u)
{
int n;
int wagen, pferde;
n = 0;
/* Man tr<74>gt sein eigenes Gewicht plus seine Kapazit<69>t! Die Menschen
* tragen nichts (siehe walkingcapacity). Ein Wagen z<EFBFBD>hlt nur, wenn er
* von zwei Pferden gezogen wird */
pferde = min(get_item(u, I_HORSE), effskill(u, SK_RIDING) * u->number * 2);
if(fval(u->race, RCF_HORSE)) pferde += u->number;
2001-01-25 10:37:55 +01:00
/* maximal diese Pferde k<>nnen zum Ziehen benutzt werden */
wagen = min(pferde / HORSESNEEDED, get_item(u, I_WAGON));
n = wagen * WAGONCAPACITY + pferde * HORSECAPACITY;
return n;
}
static int
2001-01-25 10:37:55 +01:00
walkingcapacity(unit * u)
{
int n, tmp, personen, pferde, pferde_fuer_wagen;
2001-01-25 10:37:55 +01:00
int wagen, wagen_ohne_pferde, wagen_mit_pferden, wagen_mit_trollen;
/* Das Gewicht, welches die Pferde tragen, plus das Gewicht, welches
* die Leute tragen */
pferde = get_item(u, I_HORSE);
if (fval(u->race, RCF_HORSE)) {
2001-01-25 10:37:55 +01:00
pferde += u->number;
personen = 0;
} else {
personen = u->number;
}
wagen = get_item(u, I_WAGON);
pferde_fuer_wagen = min(pferde, effskill(u, SK_RIDING) * u->number * 4);
/* maximal diese Pferde k<>nnen zum Ziehen benutzt werden */
wagen_mit_pferden = min(wagen, pferde_fuer_wagen / HORSESNEEDED);
n = wagen_mit_pferden * WAGONCAPACITY;
if (old_race(u->race) == RC_TROLL) {
2001-01-25 10:37:55 +01:00
/* 4 Trolle ziehen einen Wagen. */
/* Unbesetzte Wagen feststellen */
wagen_ohne_pferde = wagen - wagen_mit_pferden;
/* Genug Trolle, um die Restwagen zu ziehen? */
wagen_mit_trollen = min(u->number / 4, wagen_ohne_pferde);
/* Wagenkapazit<69>t hinzuz<75>hlen */
n += wagen_mit_trollen * WAGONCAPACITY;
wagen_ohne_pferde -= wagen_mit_trollen;
}
n += pferde * HORSECAPACITY;
n += personen * personcapacity(u);
/* Goliathwasser */
tmp = get_effect(u, oldpotiontype[P_STRONG]);
n += min(u->number, tmp) * personcapacity(u);
/* change_effect wird in ageing gemacht */
tmp = get_item(u, I_TROLLBELT);
n += min(tmp, u->number) * STRENGTHCAPACITY;
2001-01-25 10:37:55 +01:00
return n;
}
int
canwalk(unit * u)
{
int wagen, maxwagen;
int pferde, maxpferde;
/* workaround: monsters are too stupid to drop items, therefore they have
* infinite carrying capacity */
if (u->faction->no == 0) return 0;
2001-01-25 10:37:55 +01:00
wagen = get_item(u, I_WAGON);
pferde = get_item(u, I_HORSE);
maxwagen = effskill(u, SK_RIDING) * u->number * 2;
if (old_race(u->race) == RC_TROLL) {
2001-01-25 10:37:55 +01:00
maxwagen = max(maxwagen, u->number / 4);
}
maxpferde = effskill(u, SK_RIDING) * u->number * 4 + u->number;
if (pferde > maxpferde)
return 2;
if (walkingcapacity(u) - eff_weight(u) >= 0)
return 0;
/* Stimmt das Gewicht, impliziert dies hier, da<64> alle Wagen ohne
* Zugpferde/-trolle als Fracht aufgeladen wurden: zu viele Pferde hat
* die Einheit nicht zum Ziehen benutzt, also nicht mehr Wagen gezogen
* als erlaubt. */
if (wagen > maxwagen)
return 3;
/* Es mu<6D> nicht zwingend an den Wagen liegen, aber egal... (man
* k<EFBFBD>nnte z.B. auch 8 Eisen abladen, damit ein weiterer Wagen als
* Fracht draufpa<EFBFBD>t) */
return 1;
}
boolean
canfly(unit *u)
{
if (get_movement(&u->attribs, MV_CANNOTMOVE)) return false;
2001-01-25 10:37:55 +01:00
if(get_item(u, I_HORSE)) return false;
if(get_item(u, I_PEGASUS) >= u->number && effskill(u, SK_RIDING) >= 4)
return true;
if (fval(u->race, RCF_FLY)) return true;
2001-01-25 10:37:55 +01:00
if (get_movement(&u->attribs, MV_FLY)) return true;
2001-01-25 10:37:55 +01:00
return false;
}
boolean
canswim(unit *u)
{
if (get_movement(&u->attribs, MV_CANNOTMOVE)) return false;
if (get_item(u, I_HORSE)) return false;
2001-01-25 10:37:55 +01:00
if (get_item(u, I_DOLPHIN) >= u->number && effskill(u, SK_RIDING) >= 4)
2001-01-25 10:37:55 +01:00
return true;
if(fspecial(u->faction, FS_AMPHIBIAN)) return true;
if (u->race->flags & RCF_FLY) return true;
2001-01-25 10:37:55 +01:00
if (u->race->flags & RCF_SWIM) return true;
2001-01-25 10:37:55 +01:00
if (get_movement(&u->attribs, MV_FLY)) return true;
if (get_movement(&u->attribs, MV_SWIM)) return true;
2001-01-25 10:37:55 +01:00
return false;
}
static int
2001-01-25 10:37:55 +01:00
canride(unit * u)
{
int pferde, maxpferde, unicorns, maxunicorns;
int skill = effskill(u, SK_RIDING);
unicorns = get_item(u, I_UNICORN);
pferde = get_item(u, I_HORSE);
maxunicorns = (skill/5) * u->number;
maxpferde = skill * u->number * 2;
if(!(u->race->flags & RCF_HORSE)
2001-01-25 10:37:55 +01:00
&& ((pferde == 0 && unicorns == 0)
|| pferde > maxpferde || unicorns > maxunicorns)) {
return 0;
}
if(ridingcapacity(u) - eff_weight(u) >= 0) {
if(pferde == 0 && unicorns >= u->number && !(u->race->flags & RCF_HORSE)) {
2001-01-25 10:37:55 +01:00
return 2;
}
return 1;
}
return 0;
}
static boolean
2001-01-25 10:37:55 +01:00
cansail(const region * r, ship * sh)
{
int n = 0, p = 0;
/* sonst ist construction:: size nicht ship_type::maxsize */
assert(!sh->type->construction || sh->type->construction->improvement==NULL);
if (sh->type->construction && sh->size!=sh->type->construction->maxsize)
2001-01-25 10:37:55 +01:00
return false;
getshipweight(sh, &n, &p);
2001-01-25 10:37:55 +01:00
if( is_cursed(sh->attribs, C_SHIP_FLYING, 0) ) {
if (sh->type->cargo>500*100)
assert(!"Ein Schiff wurde verzaubert, das zu gro<72> ist");
if (n > 10000) return false;
}
if (n > shipcapacity(sh)) return false;
if (p > sh->type->cabins) return false;
return true;
}
int
enoughsailors(region * r, ship * sh)
{
int n;
unit *u;
n = 0;
for (u = r->units; u; u = u->next) {
2001-01-25 10:37:55 +01:00
if (u->ship == sh)
n += eff_skill(u, SK_SAILING, r) * u->number;
}
2001-01-25 10:37:55 +01:00
return n >= sh->type->sumskill;
}
/* ------------------------------------------------------------- */
static ship *
2001-01-25 10:37:55 +01:00
do_maelstrom(region *r, unit *u)
{
int damage;
2001-01-25 10:37:55 +01:00
damage = rand()%150 - eff_skill(u, SK_SAILING, r)*5;
2001-01-25 10:37:55 +01:00
if(damage <= 0) {
add_message(&u->faction->msgs,
new_message(u->faction, "entermaelstrom%r:region%h:ship%i:damage%i:sink", r, u->ship, damage, 1));
return u->ship;
}
2001-01-25 10:37:55 +01:00
damage_ship(u->ship, 0.01*damage);
2001-01-25 10:37:55 +01:00
if (u->ship->damage >= u->ship->size * DAMAGE_SCALE) {
ADDMSG(&u->faction->msgs, msg_message("entermaelstrom",
"region ship damage sink", r, u->ship, damage, 1));
destroy_ship(u->ship);
return NULL;
}
ADDMSG(&u->faction->msgs, msg_message("entermaelstrom",
"region ship damage sink", r, u->ship, damage, 0));
return u->ship;
2001-01-25 10:37:55 +01:00
}
/** sets a marker in the region telling that the unit has travelled through it
* this is used for two distinctly different purposes:
* - to report that a unit has travelled through. the report function
* makes sure to only report the ships of travellers, not the travellers
* themselves
* - to report the region to the traveller
*/
2001-01-25 10:37:55 +01:00
void
travelthru(const unit * u, region * r)
2001-01-25 10:37:55 +01:00
{
attrib *ru = a_add(&r->attribs, a_new(&at_travelunit));
2001-01-25 10:37:55 +01:00
ru->data.v = (void*)u;
/* the first and last region of the faction gets reset, because travelthrough
* could be in regions that are located before the [first, last] interval,
* and recalculation is needed */
u->faction->first = 0;
u->faction->last = 0;
}
static void
leave_trail(unit * u, region * from, region_list *route)
{
ship * sh = u->ship;
region * r = from;
while (route!=NULL) {
region * rn = route->data;
direction_t dir = reldirection(r, rn);
/* TODO: we cannot leave a trail into special directions
* if we use this kind of direction-attribute */
if (dir<MAXDIRECTIONS && dir>=0) {
traveldir * td = NULL;
attrib * a = a_find(r->attribs, &at_traveldir_new);
while (a!=NULL) {
td = (traveldir *)a->data.v;
if (td->no == sh->no) break;
a = a->nexttype;
}
if (a==NULL) {
a = a_add(&(r->attribs), a_new(&at_traveldir_new));
td = (traveldir *)a->data.v;
td->no = sh->no;
}
td->dir = dir;
td->age = 2;
}
route = route->next;
r = rn;
}
}
static void
2004-04-12 20:51:42 +02:00
travel_route(const unit * u, region * r, region_list * route)
{
/* kein travelthru in der letzten region! */
2004-04-12 20:51:42 +02:00
while (route) {
travelthru(u, r);
2004-04-12 20:51:42 +02:00
r = route->data;
route = route->next;
}
2001-01-25 10:37:55 +01:00
}
ship *
move_ship(ship * sh, region * from, region * to, region_list * route)
2001-01-25 10:37:55 +01:00
{
unit **iunit = &from->units;
unit **ulist = &to->units;
boolean trail = (route==NULL);
2001-01-25 10:37:55 +01:00
if (from!=to) {
translist(&from->ships, &to->ships, sh);
sh->region = to;
}
2001-01-25 10:37:55 +01:00
while (*iunit!=NULL) {
unit *u = *iunit;
assert(u->region==from);
2001-01-25 10:37:55 +01:00
if (u->ship == sh) {
if (!trail) {
leave_trail(u, from, route);
trail = true;
}
2004-04-12 20:51:42 +02:00
if (route!=NULL) travel_route(u, from, route);
if (from!=to) {
u->ship = NULL; /* damit move_unit() kein leave() macht */
move_unit(u, to, ulist);
ulist = &u->next;
u->ship = sh;
}
if (route && eff_skill(u, SK_SAILING, from) >= 1) {
produceexp(u, SK_SAILING, u->number);
}
}
if (*iunit==u) iunit=&u->next;
}
return sh;
2001-01-25 10:37:55 +01:00
}
static void
2001-01-25 10:37:55 +01:00
drifting_ships(region * r)
{
direction_t d;
if (rterrain(r) == T_OCEAN) {
ship** shp = &r->ships;
while (*shp) {
ship * sh = *shp;
region * rnext = NULL;
unit * captain;
int d_offset;
/* Schiff schon abgetrieben oder durch Zauber gesch<63>tzt? */
if (fval(sh, SF_DRIFTED) || is_cursed(sh->attribs, C_SHIP_NODRIFT, 0)) {
shp = &sh->next;
continue;
}
2001-01-25 10:37:55 +01:00
/* Kapit<69>n bestimmen */
for (captain = r->units; captain; captain = captain->next)
if (captain->ship == sh
&& eff_skill(captain, SK_SAILING, r) >= sh->type->cptskill)
break;
/* Kapit<69>n da? Besch<63>digt? Gen<65>gend Matrosen?
* Gen<EFBFBD>gend leicht? Dann ist alles OK. */
2001-01-25 10:37:55 +01:00
assert(sh->type->construction->improvement==NULL); /* sonst ist construction::size nicht ship_type::maxsize */
if (captain && sh->size==sh->type->construction->maxsize && enoughsailors(r, sh) && cansail(r, sh)) {
shp = &sh->next;
continue;
}
2001-01-25 10:37:55 +01:00
/* Leuchtturm: Ok. */
if (check_leuchtturm(r, NULL)) {
shp = &sh->next;
continue;
}
2001-01-25 10:37:55 +01:00
/* Auswahl einer Richtung: Zuerst auf Land, dann
* zuf<EFBFBD>llig. Falls unm<EFBFBD>gliches Resultat: vergi<EFBFBD> es. */
d_offset = rand() % MAXDIRECTIONS;
for (d = 0; d != MAXDIRECTIONS; ++d) {
region * rn = rconnect(r, (direction_t)((d + d_offset) % MAXDIRECTIONS));
terrain_t t = rterrain(rn);
if (rn!=NULL && (terrain[t].flags & SAIL_INTO) && ship_allowed(sh->type, rn)) {
rnext = rn;
if (t!=T_OCEAN) break;
}
}
if (rnext==NULL) {
shp = &sh->next;
continue;
}
2001-01-25 10:37:55 +01:00
/* Das Schiff und alle Einheiten darin werden nun von r
* nach rnext verschoben. Danach eine Meldung. */
sh = move_ship(sh, r, rnext, NULL);
2001-01-25 10:37:55 +01:00
if (sh!=NULL) {
fset(sh, SF_DRIFTED);
2001-01-25 10:37:55 +01:00
if (rterrain(rnext) != T_OCEAN) {
sh->coast = dir_invert(d);
}
damage_ship(sh, 0.02);
2001-01-25 10:37:55 +01:00
if (sh->damage>=sh->size * DAMAGE_SCALE) {
destroy_ship(sh);
}
}
if (*shp != sh) shp = &sh->next;
}
}
2001-01-25 10:37:55 +01:00
}
static boolean
2001-01-25 10:37:55 +01:00
present(region * r, unit * u)
{
return (boolean) (u && u->region == r);
}
static void
2001-01-25 10:37:55 +01:00
caught_target(region * r, unit * u)
{
attrib * a = a_find(u->attribs, &at_follow);
/* Verfolgungen melden */
/* Misserfolgsmeldung, oder bei erfolgreichem Verfolgen unter
* Umstaenden eine Warnung. */
if (a) {
unit * target = (unit*)a->data.v;
if (!present(r, target)) {
add_message(&u->faction->msgs, new_message(u->faction,
"followfail%u:unit%u:follower", target, u));
2002-09-02 22:36:12 +02:00
} else if (!alliedunit(target, u->faction, HELP_ALL)
2001-01-25 10:37:55 +01:00
&& cansee(target->faction, r, u, 0))
{
add_message(&target->faction->msgs, new_message(target->faction,
"followdetect%u:unit%u:follower", target, u));
}
}
}
/* TODO: Unsichtbarkeit bedenken ! */
static unit *
2001-01-25 10:37:55 +01:00
bewegung_blockiert_von(unit * reisender, region * r)
{
unit *u;
int perception = 0;
boolean contact = false;
unit * guard = NULL;
if (fval(reisender->race, RCF_ILLUSIONARY)) return NULL;
for (u=r->units;u && !contact;u=u->next) {
if (getguard(u) & GUARD_TRAVELTHRU) {
int sk = eff_skill(u, SK_OBSERVATION, r);
if (get_item(reisender, I_RING_OF_INVISIBILITY) >= reisender->number &&
!get_item(u, I_AMULET_OF_TRUE_SEEING)) continue;
if (u->faction==reisender->faction
|| ucontact(u, reisender)
2002-09-02 22:36:12 +02:00
|| alliedunit(u, reisender->faction, HELP_GUARD))
{
contact = true;
} else if (sk>=perception) {
perception = sk;
guard = u;
2001-01-25 10:37:55 +01:00
}
}
}
if (!contact && guard) {
double prob = 0.3; /* 30% base chance */
prob += 0.1 * (perception - eff_stealth(reisender, r));
prob += 0.1 * max(guard->number, get_item(guard, I_AMULET_OF_TRUE_SEEING));
2001-01-25 10:37:55 +01:00
if (chance(prob)) {
return guard;
}
}
return NULL;
2001-01-25 10:37:55 +01:00
}
static boolean
2001-01-25 10:37:55 +01:00
is_guardian(unit * u2, unit *u, unsigned int mask)
{
if (u2->faction != u->faction
&& getguard(u2)&mask
&& u2->number
&& !ucontact(u2, u) && !besieged(u2)
2002-09-02 22:36:12 +02:00
&& alliedunit(u2, u->faction, HELP_GUARD) != HELP_GUARD
2001-01-25 10:37:55 +01:00
#ifdef WACH_WAFF
&& armedmen(u2)
2001-01-25 10:37:55 +01:00
#endif
&& cansee(u2->faction, u->region, u, 0)
2001-01-25 10:37:55 +01:00
) return true;
return false;
}
unit *
is_guarded(region * r, unit * u, unsigned int mask)
{
unit *u2;
for (u2 = r->units; u2; u2 = u2->next)
if (is_guardian(u2, u, mask)) return u2;
return NULL;
}
static const char *shortdirections[MAXDIRECTIONS] =
2001-01-25 10:37:55 +01:00
{
"dir_nw",
"dir_ne",
"dir_east",
"dir_se",
"dir_sw",
"dir_west"
2001-01-25 10:37:55 +01:00
};
static void
2001-01-25 10:37:55 +01:00
cycle_route(unit *u, int gereist)
{
int cm = 0;
char tail[1024];
char neworder[2048];
const char *token;
2001-01-25 10:37:55 +01:00
direction_t d = NODIRECTION;
boolean paused = false;
boolean pause;
if (igetkeyword(u->thisorder, u->faction->locale) != K_ROUTE) return;
2001-01-25 10:37:55 +01:00
tail[0] = '\0';
strcpy(neworder, locale_string(u->faction->locale, keywords[K_ROUTE]));
2001-01-25 10:37:55 +01:00
for (cm=0;;++cm) {
2002-09-02 22:36:12 +02:00
const struct locale * lang = u->faction->locale;
2001-01-25 10:37:55 +01:00
pause = false;
token = getstrtoken();
d = finddirection(token, lang);
2001-02-18 13:20:37 +01:00
if(d == D_PAUSE) {
2001-01-25 10:37:55 +01:00
pause = true;
2001-02-18 13:20:37 +01:00
} else if (d==NODIRECTION) {
break;
2001-01-25 10:37:55 +01:00
}
if (cm<gereist) {
/* hier sollte keine PAUSE auftreten */
assert(!pause);
2001-05-27 18:44:16 +02:00
if (!pause) strcat(strcat(tail, " "), LOC(lang, shortdirections[d]));
2001-01-25 10:37:55 +01:00
}
else if (strlen(neworder)>sizeof(neworder)/2) break;
else if (cm==gereist && !paused && pause) {
strcat(strcat(tail, " "), LOC(lang, parameters[P_PAUSE]));
2001-01-25 10:37:55 +01:00
paused=true;
}
else if (pause) {
/* da PAUSE nicht in ein shortdirections[d] umgesetzt wird (ist
* hier keine normale direction), muss jede PAUSE einzeln
* herausgefiltert und explizit gesetzt werden */
strcat(strcat(neworder, " "), LOC(lang, parameters[P_PAUSE]));
} else {
strcat(strcat(neworder, " "), LOC(lang, shortdirections[d]));
}
2001-01-25 10:37:55 +01:00
}
strcat(neworder, tail);
set_string(&u->lastorder, neworder);
}
static void
2001-01-25 10:37:55 +01:00
init_drive(void)
{
region *r;
unit *u, *ut, *u2;
strlist *S;
2001-01-25 10:37:55 +01:00
int w;
for (r=regions; r; r=r->next) {
/* This is just a simple check for non-corresponding K_TRANSPORT/
* K_DRIVE. This is time consuming for an error check, but there
* doesn't seem to be an easy way to speed this up. */
for(u=r->units; u; u=u->next) {
if(igetkeyword(u->thisorder, u->faction->locale) == K_DRIVE && !fval(u, UFL_LONGACTION) && !fval(u, UFL_HUNGER)) {
boolean found = false;
2001-04-26 19:41:06 +02:00
ut = getunit(r, u->faction);
if(!ut) {
cmistake(u, findorder(u, u->thisorder), 63, MSG_MOVE);
continue;
}
for (S = ut->orders; S; S = S->next) {
if (igetkeyword(S->s, u->faction->locale) == K_TRANSPORT) {
if (getunit(r, ut->faction) == u) {
found = true;
break;
}
}
}
if(found == false) {
if(cansee(u->faction, r, ut, 0)) {
cmistake(u, findorder(u, u->thisorder), 286, MSG_MOVE);
} else {
cmistake(u, findorder(u, u->thisorder), 63, MSG_MOVE);
}
}
}
}
/* This calculates the weights of all transported units and
* adds them to an internal counter which is used by travel() to
* calculate effective weight and movement. */
2001-01-25 10:37:55 +01:00
for (u=r->units; u; u=u->next) {
w = 0;
for (S = u->orders; S; S = S->next) if(igetkeyword(S->s, u->faction->locale) == K_TRANSPORT) {
2001-04-26 19:41:06 +02:00
ut = getunit(r, u->faction);
2001-01-25 10:37:55 +01:00
if(!ut) continue;
if (igetkeyword(ut->thisorder, ut->faction->locale) == K_DRIVE && !fval(ut, UFL_LONGACTION) && !fval(ut, UFL_HUNGER)) {
2001-04-26 19:41:06 +02:00
u2 = getunit(r, u->faction);
2001-01-25 10:37:55 +01:00
if(u2 == u) {
w += weight(ut);
break;
}
}
}
if(w > 0) a_add(&u->attribs, a_new(&at_driveweight))->data.i = w;
}
}
}
int *storms;
#if 0
/* in arbeit */
#define ETRAVEL_NOTALLOWED 1
#define ETRAVEL_NOSWIM 2
#define ETRAVEL_NORIDE 4
#define ETRAVEL_LEFTSHIP 8
int
cantravel(const unit *u, const region *from, const region * to, boolean (*allowed)(const struct region *, const struct region *))
{
/* basic allowed function */
if (allowed && !allowed(from, to)) return ETRAVEL_NOTALLOWED;
if (u->ship && fval(u, UFL_OWNER) && rterrain(from)==T_OCEAN) {
2001-01-25 10:37:55 +01:00
/* wir sind auf einem schiff und auf dem ozean */
if (rterrain(to)==T_OCEAN || (old_race(u->race) != RC_AQUARIAN && !dragon(u))) {
2001-01-25 10:37:55 +01:00
/* Drache oder Meermensch sind wir auch nicht */
return ETRAVEL_NOSWIM;
} else if (get_item(u, I_HORSE) > 0) {
return ETRAVEL_NORIDE|ETRAVEL_NOSWIM;
}
}
2001-01-25 10:37:55 +01:00
if (!u->ship && leftship(u) && is_guarded(from, u, GUARD_LANDING)) {
/* An Land kein NACH wenn in dieser Runde Schiff VERLASSEN! */
return ETRAVEL_LEFTSHIP;
}
}
#endif
static boolean
roadto(const region * r, direction_t dir)
2001-01-25 10:37:55 +01:00
{
/* wenn es hier genug strassen gibt, und verbunden ist, und es dort
* genug strassen gibt, dann existiert eine strasse in diese richtung */
region * r2;
static const curse_type * roads_ct = NULL;
if (roads_ct==NULL) roads_ct = ct_find("magicstreet");
if (roads_ct!=NULL) {
curse *c = get_curse(r->attribs, roads_ct);
if (c!=NULL) return true;
}
if (dir>=MAXDIRECTIONS || dir<0) return false;
r2 = rconnect(r, dir);
if (r==NULL || r2==NULL) return false;
if (terrain[rterrain(r)].roadreq==0) return false;
if (terrain[rterrain(r2)].roadreq==0) return false;
if (rroad(r, dir) < terrain[rterrain(r)].roadreq) return false;
if (rroad(r2, dir_invert(dir)) < terrain[rterrain(r2)].roadreq) return false;
return true;
}
2001-01-25 10:37:55 +01:00
void
travel(unit * u, region * next, int flucht, region_list ** routep)
{
region *first = u->region;
region *current = u->region;
int k, m = 0;
double dk;
strlist *S;
unit *ut, *u2;
int gereist = 0;
region_list *route = NULL;
region_list **iroute = &route;
static boolean init = false;
static const curse_type * speed_ct;
2001-01-25 10:37:55 +01:00
if (routep) *routep = NULL;
if (!init) {
init = true;
speed_ct = ct_find("speed");
}
2001-01-25 10:37:55 +01:00
/* tech:
*
* zu Fu<EFBFBD> reist man 1 Region, zu Pferd 2 Regionen. Mit Stra<EFBFBD>en reist
* man zu Fu<EFBFBD> 2, mit Pferden 3 weit.
*
* Berechnet wird das mit BPs. Zu Fu<EFBFBD> hat man 4 BPs, zu Pferd 6.
* Normalerweise verliert man 3 BP pro Region, bei Stra<EFBFBD>en nur 2 BP.
* Au<EFBFBD>erdem: Wenn Einheit transportiert, nur halbe BP */
if (rterrain(current)==T_OCEAN
&& !(u->race->flags & RCF_FLY) && rterrain(next) != T_OCEAN)
{ /* Die Einheit kann nicht fliegen, ist im Ozean, und will an Land */
if (!(u->race->flags & RCF_SWIM) && old_race(u->race) != RC_AQUARIAN) {
cmistake(u, findorder(u, u->thisorder), 44, MSG_MOVE);
return;
} else {
if (u->ship && get_item(u, I_HORSE) > 0) {
cmistake(u, findorder(u, u->thisorder), 67, MSG_MOVE);
return;
}
}
} else if (rterrain(current)!=T_OCEAN) {
/* An Land kein NACH wenn in dieser Runde Schiff VERLASSEN! */
if (leftship(u) && is_guarded(current, u, GUARD_LANDING)) {
cmistake(u, findorder(u, u->thisorder), 70, MSG_MOVE);
return;
}
if (u->ship && !flucht && u->race->flags & RCF_SWIM) {
cmistake(u, findorder(u, u->thisorder), 143, MSG_MOVE);
return;
}
} else if (rterrain(next) == T_OCEAN && u->ship && fval(u->ship, SF_MOVED)) {
cmistake(u, findorder(u, u->thisorder), 13, MSG_MOVE);
return;
}
2001-01-25 10:37:55 +01:00
switch (canwalk(u)) {
case 1:
cmistake(u, findorder(u, u->thisorder), 57, MSG_MOVE);
return;
case 2:
cmistake(u, findorder(u, u->thisorder), 56, MSG_MOVE);
return;
case 3:
cmistake(u, findorder(u, u->thisorder), 42, MSG_MOVE);
return;
}
2001-01-25 10:37:55 +01:00
/* wir suchen so lange nach neuen Richtungen, wie es geht. Diese werden
* dann nacheinander ausgef<EFBFBD>hrt. */
dk = u->race->speed;
if (speed_ct) {
curse *c = get_curse(u->attribs, speed_ct);
if(c) {
int men = get_cursedmen(u, c);
dk *= 1.0 + (double)men/(double)u->number;
}
}
2001-01-25 10:37:55 +01:00
switch(canride(u)) {
case 1: /* Pferd */
k = (int)(dk*BP_RIDING);
break;
case 2: /* Einhorn */
k = (int)(dk*BP_UNICORN);
break;
default:
{
int mp = 1;
/* faction special */
if(fspecial(u->faction, FS_QUICK))
mp = BP_RIDING;
/* Siebenmeilentee */
if (get_effect(u, oldpotiontype[P_FAST]) >= u->number) {
mp *= 2;
change_effect(u, oldpotiontype[P_FAST], -u->number);
}
2001-01-25 10:37:55 +01:00
/* unicorn in inventory */
if (u->number <= get_item(u, I_FEENSTIEFEL))
mp *= 2;
2001-01-25 10:37:55 +01:00
/* Im Astralraum sind Tyb und Ill-Magier doppelt so schnell.
* Nicht kumulativ mit anderen Beschleunigungen! */
if ( mp == 1 && getplane(next) == get_astralplane() && is_mage(u)) {
if(get_mage(u)->magietyp == M_ASTRAL
|| get_mage(u)->magietyp == M_TRAUM) {
mp *= 2;
}
}
k = (int)(dk*mp*BP_WALKING);
}
break;
}
2001-01-25 10:37:55 +01:00
switch(old_race(u->race)) {
case RC_DRAGON:
case RC_WYRM:
case RC_FIREDRAGON:
case RC_BIRTHDAYDRAGON:
case RC_PSEUDODRAGON:
k = BP_DRAGON;
}
/* die n<>chste Region, in die man wandert, wird durch movewhere() aus
* der letzten Region bestimmt.
*
* Anfangen tun wir bei r. r2 ist beim ersten Durchlauf schon gesetzt
* (Parameter!), das Ziel des Schrittes. m z<EFBFBD>hlt die Schritte, k sind
* die noch m<EFBFBD>glichen Schritte, r3 die letzte g<EFBFBD>ltige, befahrene Region. */
while (next) {
border * b = get_borders(current, next);
direction_t reldir = reldirection(current, next);
while (b!=NULL) {
if (b->type==&bt_wisps) {
wall_data * wd = (wall_data*)b->data;
assert(reldir!=D_SPECIAL);
if (wd->active) {
/* pick left and right region: */
region * rl = rconnect(current, (direction_t)((reldir+MAXDIRECTIONS-1)%MAXDIRECTIONS));
region * rr = rconnect(current, (direction_t)((reldir+1)%MAXDIRECTIONS));
int j = rand() % 3;
if (j==0) break;
else if (j==1 && rl && landregion(rterrain(rl))==landregion(rterrain(next))) next = rl;
else if (j==2 && rr && landregion(rterrain(rr))==landregion(rterrain(next))) next = rr;
break;
}
}
if (b->type->move) b->type->move(b, u, current, next);
b = b->next;
}
if (current!=next) { /* !pause */
if (roadto(current, reldir)) k-=BP_ROAD;
else k-=BP_NORMAL;
if (k<0) break;
if (reldir>=0 && move_blocked(u, current, next)) {
ADDMSG(&u->faction->msgs, msg_message("leavefail",
"unit region", u, next));
}
if (!entrance_allowed(u, next)) {
ADDMSG(&u->faction->msgs, msg_message("regionowned",
"unit region target", u, current, next));
break;
}
if (gereist) {
unit * wache = bewegung_blockiert_von(u, current);
if (wache!=NULL) {
ADDMSG(&u->faction->msgs, msg_message("moveblockedbyguard",
"unit region guard", u, current, wache));
break;
}
}
if (fval(u->race, RCF_ILLUSIONARY)) {
curse * c = get_curse(next->attribs, ct_find("antimagiczone"));
if (curse_active(c)) {
curse_changevigour(&next->attribs, c, - u->number);
add_message(&u->faction->msgs, new_message(u->faction,
"illusionantimagic%u:unit", u));
destroy_unit(u);
if (routep) *routep = route;
else free_regionlist(route);
return;
}
}
/* Ozeanfelder k<>nnen nur von Einheiten mit Schwimmen und ohne
* Pferde betreten werden. Drachen k<EFBFBD>nnen fliegen. */
if (rterrain(next) == T_OCEAN && !canswim(u)) {
plane *pl = getplane(next);
if (reldir<MAXDIRECTIONS && pl!=NULL && fval(pl, PFL_NOCOORDS)) {
ADDMSG(&u->faction->msgs, msg_message("detectoceandir",
"unit direction", u, reldir));
} else {
ADDMSG(&u->faction->msgs, msg_message("detectocean",
"unit region", u, next));
}
break;
}
2001-01-25 10:37:55 +01:00
if (terrain[rterrain(next)].flags & FORBIDDEN_LAND) {
plane *pl = getplane(next);
if (reldir<MAXDIRECTIONS && pl!=NULL && fval(pl, PFL_NOCOORDS)) {
ADDMSG(&u->faction->msgs, msg_message("detectforbiddendir",
"unit direction", u, reldir));
} else {
ADDMSG(&u->faction->msgs, msg_message("detectforbidden",
"unit region", u, next));
}
break;
}
2001-01-25 10:37:55 +01:00
if (old_race(u->race) == RC_INSECT && r_insectstalled(next) && !is_cursed(u->attribs, C_KAELTESCHUTZ,0))
{
ADDMSG(&u->faction->msgs, msg_message("detectforbidden",
"unit region", u, next));
break;
}
add_regionlist(iroute, next);
iroute = &(*iroute)->next;
m++;
} else {
break;
}
2001-01-25 10:37:55 +01:00
current = next;
if(!flucht) {
++gereist;
if (current==first) break; /* PAUSE beendet die Reise */
next = movewhere(current, u);
}
}
2001-01-25 10:37:55 +01:00
if (gereist) {
int mode;
setguard(u, GUARD_NONE);
cycle_route(u, gereist);
buf[0] = 0;
if (flucht == 1)
mode = 0;
else if (canride(u)) {
mode = 1;
produceexp(u, SK_RIDING, u->number);
} else
mode = 2;
/* Haben Drachen ihr Ziel erreicht? */
if (u->faction->no==MONSTER_FACTION) {
attrib * ta = a_find(u->attribs, &at_targetregion);
if (ta && current == (region*)ta->data.v) {
a_remove(&u->attribs, ta);
set_string(&u->lastorder, "WARTEN");
}
}
2001-01-25 10:37:55 +01:00
/* <20>ber die letzte Region braucht man keinen Bericht */
2001-01-25 10:37:55 +01:00
m--;
if (m > 0) {
2004-04-12 20:51:42 +02:00
region_list *rlist = route;
2001-01-25 10:37:55 +01:00
travelthru(u, first);
2004-04-12 20:51:42 +02:00
while (rlist!=NULL) {
region * r = rlist->data;
2004-04-12 20:51:42 +02:00
travelthru(u, r);
rlist=rlist->next;
2004-04-12 20:51:42 +02:00
if (rlist!=NULL) {
char * p;
if (r!=route->data) {
if (rlist->next==NULL) scat(" und ");
else scat(", ");
}
p = buf+strlen(buf);
if (rlist->next) {
MSG(("travelthru_trail", "region", rlist->data), p, sizeof(buf)-strlen(buf), u->faction->locale, u->faction);
}
}
}
}
ADDMSG(&u->faction->msgs, msg_message("travel",
"unit mode start end regions", u, mode, first, current, strdup(buf)));
/* und jetzt noch die transportierten Einheiten verschieben */
for (S = u->orders; S; S = S->next) {
if (igetkeyword(S->s, u->faction->locale) == K_TRANSPORT) {
ut = getunit(first, u->faction);
if (ut) {
boolean found = false;
if (igetkeyword(ut->thisorder, ut->faction->locale) == K_DRIVE
&& !fval(ut, UFL_LONGACTION) && !fval(ut, UFL_HUNGER)) {
u2 = getunit(first, ut->faction);
if(u2 == u) {
found = true;
add_message(&u->faction->msgs, new_message(
u->faction, "transport%u:unit%u:target%r:start%r:end",
u, ut, first, current));
if(!(terrain[current->terrain].flags & WALK_INTO)
&& get_item(ut, I_HORSE)) {
cmistake(u, u->thisorder, 67, MSG_MOVE);
cmistake(ut, ut->thisorder, 67, MSG_MOVE);
continue;
}
if (can_survive(ut, current)) {
2004-04-12 20:51:42 +02:00
travel_route(ut, ut->region, route);
move_unit(ut, current, NULL);
} else {
cmistake(u, u->thisorder, 287, MSG_MOVE);
cmistake(ut, ut->thisorder, 230, MSG_MOVE);
continue;
}
}
}
if (!found) {
if(cansee(u->faction, u->region, ut, 0)) {
cmistake(u, findorder(u, u->thisorder), 90, MSG_MOVE);
} else {
cmistake(u, findorder(u, u->thisorder), 63, MSG_MOVE);
}
}
} else {
if (ut) {
cmistake(u, findorder(u, u->thisorder), 63, MSG_MOVE);
} else {
cmistake(u, findorder(u, u->thisorder), 99, MSG_MOVE);
}
}
}
}
move_unit(u, current, NULL);
}
else if (flucht)
move_unit(u, current, NULL);
set_string(&u->thisorder, "");
fset(u, UFL_LONGACTION);
setguard(u, GUARD_NONE);
if (fval(u, UFL_FOLLOWING)) caught_target(current, u);
if (routep) *routep = route;
else free_regionlist(route);
return;
2001-01-25 10:37:55 +01:00
}
static boolean
2001-01-25 10:37:55 +01:00
ship_ready(region * r, unit * u)
{
if (!fval(u, UFL_OWNER)) {
2001-01-25 10:37:55 +01:00
cmistake(u, findorder(u, u->thisorder), 146, MSG_MOVE);
return false;
}
if (eff_skill(u, SK_SAILING, r) < u->ship->type->cptskill) {
sprintf(buf, "Der Kapit<69>n mu<6D> mindestens Segeln %d haben, "
"um %s zu befehligen ", u->ship->type->cptskill,
u->ship->type->name[1]);
mistake(u, u->thisorder, buf, MSG_MOVE);
return false;
}
assert(u->ship->type->construction->improvement==NULL); /* sonst ist construction::size nicht ship_type::maxsize */
if (u->ship->size!=u->ship->type->construction->maxsize) {
cmistake(u, findorder(u, u->thisorder), 15, MSG_MOVE);
return false;
}
if (!enoughsailors(r, u->ship)) {
cmistake(u, findorder(u, u->thisorder), 1, MSG_MOVE);
/* mistake(u, u->thisorder,
"Auf dem Schiff befinden sich zuwenig erfahrene Seeleute.", MSG_MOVE); */
return false;
}
if (!cansail(r, u->ship)) {
if( is_cursed(u->ship->attribs, C_SHIP_FLYING, 0) )
cmistake(u, findorder(u, u->thisorder), 17, MSG_MOVE);
else
cmistake(u, findorder(u, u->thisorder), 18, MSG_MOVE);
return false;
}
return true;
}
unit *
owner_buildingtyp(const region * r, const building_type * bt)
2001-01-25 10:37:55 +01:00
{
building *b;
unit *owner;
for (b = rbuildings(r); b; b = b->next) {
owner = buildingowner(r, b);
if (b->type == bt && owner != NULL) {
if (b->size >= bt->maxsize) {
return owner;
}
}
}
return NULL;
}
boolean
buildingtype_exists(const region * r, const building_type * bt)
{
building *b;
for (b = rbuildings(r); b; b = b->next) {
if (b->type == bt) {
if (b->size >= bt->maxsize) {
return true;
}
}
}
return false;
}
static boolean
check_working_buildingtype(const region * r, const building_type * bt)
{
building *b;
for (b = rbuildings(r); b; b = b->next) {
if (b->type == bt) {
if (b->size >= bt->maxsize && fval(b, BLD_WORKING)) {
return true;
}
}
}
return false;
}
2001-01-25 10:37:55 +01:00
/* Pr<50>ft, ob Ablegen von einer K<>ste in eine der erlaubten Richtungen erfolgt. */
static boolean
2001-01-25 10:37:55 +01:00
check_takeoff(ship *sh, region *from, region *to)
{
if (rterrain(from)!=T_OCEAN && sh->coast != NODIRECTION) {
direction_t coast = sh->coast;
direction_t dir = reldirection(from, to);
direction_t coastr = (direction_t)((coast+1) % MAXDIRECTIONS);
direction_t coastl = (direction_t)((coast+MAXDIRECTIONS-1) % MAXDIRECTIONS);
if (dir!=coast && dir!=coastl && dir!=coastr
&& !check_working_buildingtype(from, bt_find("harbour")))
{
return false;
}
}
2001-01-25 10:37:55 +01:00
return true;
2001-01-25 10:37:55 +01:00
}
static boolean
flying_ship(const ship * sh)
2001-01-25 10:37:55 +01:00
{
if (sh->type->flags & SFL_FLY) return true;
if (is_cursed(sh->attribs, C_SHIP_FLYING, 0)) return true;
return false;
}
static boolean
ship_allowed(const struct ship_type * type, const region * r)
{
int c = 0;
terrain_t t = rterrain(r);
static const building_type * bt_harbour=NULL;
if (bt_harbour==NULL) bt_harbour=bt_find("harbour");
if (check_working_buildingtype(r, bt_harbour)) return true;
for (c=0;type->coast[c]!=NOTERRAIN;++c) {
if (type->coast[c]==t) return true;
}
return false;
2001-01-25 10:37:55 +01:00
}
static region_list *
sail(unit * u, region * next_point, boolean move_on_land)
2001-01-25 10:37:55 +01:00
{
region *starting_point = u->region;
region *current_point, *last_point;
int k, step = 0;
region_list *route = NULL;
region_list **iroute = &route;
ship * sh = u->ship;
faction * f = u->faction;
2001-01-25 10:37:55 +01:00
if (!ship_ready(starting_point, u)) return NULL;
2001-01-25 10:37:55 +01:00
/* Wir suchen so lange nach neuen Richtungen, wie es geht. Diese werden
* dann nacheinander ausgef<EFBFBD>hrt. */
2001-01-25 10:37:55 +01:00
k = shipspeed(sh, u);
2001-01-25 10:37:55 +01:00
last_point = starting_point;
current_point = starting_point;
2001-01-25 10:37:55 +01:00
/* die n<>chste Region, in die man segelt, wird durch movewhere() aus der
* letzten Region bestimmt.
*
* Anfangen tun wir bei starting_point. next_point ist beim ersten
* Durchlauf schon gesetzt (Parameter!). current_point ist die letzte g<EFBFBD>ltige,
* befahrene Region. */
2001-01-25 10:37:55 +01:00
while (current_point!=next_point && step < k && next_point) {
terrain_t tthis = rterrain(current_point);
/* these values need to be updated if next_point changes (due to storms): */
terrain_t tnext = rterrain(next_point);
direction_t dir = reldirection(current_point, next_point);
assert(sh==u->ship || !"ship has sunk, but we didn't notice it");
if (terrain[tnext].flags & FORBIDDEN_LAND) {
plane *pl = getplane(next_point);
if (pl && fval(pl, PFL_NOCOORDS)) {
ADDMSG(&f->msgs, msg_message("sailforbiddendir",
"ship direction", sh, dir));
} else {
ADDMSG(&f->msgs, msg_message("sailforbidden",
"ship region", sh, next_point));
}
break;
}
if (!flying_ship(sh)) {
/* storms should be the first thing we do. */
int stormchance = storms[month(0)] * 5 / shipspeed(sh, u);
if (check_leuchtturm(next_point, NULL)) stormchance /= 3;
if (rand()%10000 < stormchance && next_point->terrain == T_OCEAN) {
if (!is_cursed(sh->attribs, C_SHIP_NODRIFT, 0)) {
region * rnext = NULL;
boolean storm = true;
int d_offset = rand() % MAXDIRECTIONS;
direction_t d;
/* Sturm nur, wenn n<>chste Region Hochsee ist. */
for (d=0;d!=MAXDIRECTIONS;++d) {
direction_t dnext = (direction_t)((d + d_offset) % MAXDIRECTIONS);
region * rn = rconnect(current_point, dnext);
if (rn!=NULL) {
terrain_t t = rterrain(rn);
if (terrain[t].flags & FORBIDDEN_LAND) continue;
if (t!=T_OCEAN) {
storm = false;
break;
}
if (rn!=next_point) rnext = rn;
}
}
if (storm && rnext!=NULL) {
ADDMSG(&f->msgs, msg_message("storm", "ship region sink",
sh, current_point, sh->damage>=sh->size * DAMAGE_SCALE));
/* damage the ship. we handle destruction in the end */
damage_ship(sh, 0.02);
if (sh->damage>=sh->size * DAMAGE_SCALE) break;
next_point = rnext;
/* these values need to be updated if next_point changes (due to storms): */
tnext = rterrain(next_point);
dir = reldirection(current_point, next_point);
}
}
}
if (tthis != T_OCEAN) {
if (tnext != T_OCEAN) {
if (!move_on_land) {
/* check that you're not traveling from one land region to another. */
plane *pl = getplane(next_point);
if (pl!=NULL && fval(pl, PFL_NOCOORDS)) {
/* we don't have that case yet, but hey... */
sprintf(buf, "Die %s entdeckt, da<64> im %s Festland ist.",
shipname(sh), locale_string(u->faction->locale, directions[dir]));
} else {
sprintf(buf, "Die %s entdeckt, da<64> (%d,%d) Festland ist.",
shipname(sh), region_x(next_point,u->faction),
region_y(next_point,u->faction));
}
addmessage(0, u->faction, buf, MSG_MOVE, ML_WARN);
break;
}
} else {
if (check_takeoff(sh, current_point, next_point) == false) {
/* Schiff kann nicht ablegen */
cmistake(u, findorder(u, u->thisorder), 182, MSG_MOVE);
break;
}
}
} else if (tnext==T_OCEAN) {
/* target region is an ocean, and we're not leaving a shore */
if (!(sh->type->flags & SFL_OPENSEA)) {
/* ship can only stay close to shore */
direction_t d;
for (d=0;d!=MAXDIRECTIONS;++d) {
region * rc = rconnect(next_point, d);
if (rterrain(rc) != T_OCEAN) break;
}
if (d==MAXDIRECTIONS) {
/* Schiff kann nicht aufs offene Meer */
cmistake(u, findorder(u, u->thisorder), 249, MSG_MOVE);
break;
}
}
}
if (!ship_allowed(sh->type, next_point)) {
/* for some reason or another, we aren't allowed in there.. */
if (check_leuchtturm(current_point, NULL)) {
ADDMSG(&f->msgs, msg_message("sailnolandingstorm", "ship", sh));
} else {
damage_ship(sh, 0.10);
/* we handle destruction at the end */
}
break;
}
2001-01-25 10:37:55 +01:00
if (is_cursed(next_point->attribs, C_MAELSTROM, 0)) {
if (do_maelstrom(next_point, u)==NULL) break;
}
2001-01-25 10:37:55 +01:00
} /* !flying_ship */
2001-01-25 10:37:55 +01:00
/* Falls Blockade, endet die Seglerei hier */
if (move_blocked(u, current_point, next_point)) {
ADDMSG(&u->faction->msgs, msg_message("sailfail", "ship region", sh, current_point));
break;
}
2001-01-25 10:37:55 +01:00
/* Falls kein Problem, eines weiter ziehen */
fset(sh, SF_MOVED);
add_regionlist(iroute, next_point);
iroute = &(*iroute)->next;
step++;
last_point = current_point;
current_point = next_point;
if (rterrain(current_point) != T_OCEAN && !is_cursed(sh->attribs, C_SHIP_FLYING, 0)) break;
next_point = movewhere(current_point, u);
}
if (sh->damage>=sh->size * DAMAGE_SCALE) {
ADDMSG(&f->msgs, msg_message("shipsink", "ship", sh));
destroy_ship(sh);
sh = NULL;
}
/* Nun enth<74>lt current_point die Region, in der das Schiff seine Runde
* beendet hat. Wir generieren hier ein Ereignis f<EFBFBD>r den Spieler, das
* ihm sagt, bis wohin er gesegelt ist, falls er <EFBFBD>berhaupt vom Fleck
* gekommen ist. Das ist nicht der Fall, wenn er von der K<EFBFBD>ste ins
* Inland zu segeln versuchte */
if (sh!=NULL && fval(sh, SF_MOVED)) {
unit * hafenmeister;
/* nachdem alle Richtungen abgearbeitet wurden, und alle Einheiten
* transferiert wurden, kann der aktuelle Befehl gel<EFBFBD>scht werden. */
cycle_route(u, step);
set_string(&u->thisorder, "");
if (current_point->terrain != T_OCEAN && !is_cursed(sh->attribs, C_SHIP_FLYING, 0)) {
sh->coast = reldirection(current_point, last_point);
} else {
sh->coast = NODIRECTION;
}
sprintf(buf, "Die %s ", shipname(sh));
if( is_cursed(sh->attribs, C_SHIP_FLYING, 0) )
scat("fliegt");
else
scat("segelt");
scat(" von ");
scat(regionid(starting_point));
scat(" nach ");
scat(regionid(current_point));
scat(".");
addmessage(0, u->faction, buf, MSG_MOVE, ML_INFO);
/* Das Schiff und alle Einheiten darin werden nun von
* starting_point nach current_point verschoben */
/* Verfolgungen melden */
if (fval(u, UFL_FOLLOWING)) caught_target(current_point, u);
sh = move_ship(sh, starting_point, current_point, route);
/* Hafengeb<65>hren ? */
hafenmeister = owner_buildingtyp(current_point, bt_find("harbour"));
if (sh && hafenmeister != NULL) {
item * itm;
unit * u2;
boolean first = true;
item * trans = NULL;
for (u2 = current_point->units; u2; u2 = u2->next) {
if (u2->ship == sh &&
!alliedunit(hafenmeister, u->faction, HELP_GUARD)) {
if (effskill(hafenmeister, SK_OBSERVATION) > effskill(u2, SK_STEALTH)) {
for (itm=u2->items; itm; itm=itm->next) {
const luxury_type * ltype = resource2luxury(itm->type->rtype);
if (ltype!=NULL && itm->number>0) {
int st = itm->number * effskill(hafenmeister, SK_TRADE) / 50;
st = min(itm->number, st);
if (st > 0) {
i_change(&u2->items, itm->type, -st);
i_change(&hafenmeister->items, itm->type, st);
i_add(&trans, i_new(itm->type, st));
}
}
}
}
}
}
if (trans) {
sprintf(buf, "%s erhielt ", hafenmeister->name);
for (itm = trans; itm; itm=itm->next) {
if (!first) {
if (itm->next!=NULL && itm->next->next==NULL) {
scat(" und ");
} else {
scat(", ");
}
}
first = false;
icat(trans->number);
scat(" ");
if (itm->number == 1) {
scat(locale_string(default_locale, resourcename(itm->type->rtype, 0)));
} else {
scat(locale_string(default_locale, resourcename(itm->type->rtype, NMF_PLURAL)));
}
}
scat(" von der ");
scat(shipname(u->ship));
scat(".");
addmessage(0, u->faction, buf, MSG_COMMERCE, ML_INFO);
addmessage(0, hafenmeister->faction, buf, MSG_INCOME, ML_INFO);
while (trans) i_remove(&trans, trans);
}
}
}
return route;
2001-01-25 10:37:55 +01:00
}
unit *
kapitaen(region * r, ship * sh)
{
unit *u;
for (u = r->units; u; u = u->next) {
if (u->ship == sh && eff_skill(u, SK_SAILING, r) >= sh->type->cptskill)
return u;
}
return NULL;
}
static const char *
direction_name(const region * from, const region * to, const struct locale * lang)
{
direction_t dir = reldirection(from, to);
if (dir<MAXDIRECTIONS && dir>=0) return locale_string(lang, directions[dir]);
if (dir==D_SPECIAL) {
spec_direction *sd = special_direction(from, to);
return sd->keyword;
}
assert(!"invalid direction");
return NULL;
}
/* Segeln, Wandern, Reiten
* when this routine returns a non-zero value, movement for the region needs
* to be done again because of followers that got new MOVE orders.
* Setting FL_LONGACTION will prevent a unit from being handled more than once
* by this routine
*/
static int
move(unit * u, boolean move_on_land)
2001-01-25 10:37:55 +01:00
{
region * r = u->region;
region_list * route = NULL;
region * r2 = movewhere(r, u);
2001-01-25 10:37:55 +01:00
if (r2==NULL) {
cmistake(u, findorder(u, u->thisorder), 71, MSG_MOVE);
return 0;
}
else if (u->ship && fval(u, UFL_OWNER)) {
route = sail(u, r2, move_on_land);
} else {
travel(u, r2, 0, &route);
}
if (i_get(u->items, &it_demonseye)) {
direction_t d;
for (d=0;d!=MAXDIRECTIONS;++d) {
region * rc = rconnect(r2,d);
if (rc) {
sprintf(buf, "Im %s ist eine ungeheure magische Pr<50>senz zu versp<73>ren.",
locale_string(u->faction->locale, directions[dir_invert(d)]));
addmessage(rc, NULL, buf, MSG_EVENT, ML_IMPORTANT);
}
}
}
2001-01-25 10:37:55 +01:00
if (u->region!=r) fset(u, UFL_LONGACTION);
set_string(&u->thisorder, "");
if (fval(u, UFL_FOLLOWED) && route!=NULL) {
int followers = 0;
unit *up;
for (up=r->units;up;up=up->next) {
if (fval(up, UFL_FOLLOWING) && !fval(up, UFL_LONGACTION)) {
const attrib * a = a_findc(up->attribs, &at_follow);
if (a && a->data.v==u) {
/* wir basteln ihm ein NACH */
int k;
region_list * rlist = route;
2004-04-12 20:51:42 +02:00
region * from = r;
strcpy(buf, locale_string(up->faction->locale, keywords[K_MOVE]));
while (rlist!=NULL) {
strcat(strcat(buf, " "), direction_name(from, rlist->data, up->faction->locale));
from = rlist->data;
rlist = rlist->next;
}
set_string(&up->thisorder, buf);
k = igetkeyword(up->thisorder, up->faction->locale);
assert(k==K_MOVE);
++followers;
}
}
}
return followers;
}
if (route!=NULL) free_regionlist(route);
return 0;
2001-01-25 10:37:55 +01:00
}
2004-01-11 10:21:06 +01:00
typedef struct piracy_data {
const struct faction * follow;
direction_t dir;
} piracy_data;
static void
piracy_init(struct attrib * a)
{
a->data.v = calloc(1, sizeof(piracy_data));
}
static void
piracy_done(struct attrib * a)
{
free(a->data.v);
}
static attrib_type at_piracy_direction = {
2001-01-25 10:37:55 +01:00
"piracy_direction",
2004-01-11 10:21:06 +01:00
piracy_init,
piracy_done,
2001-01-25 10:37:55 +01:00
DEFAULT_AGE,
NO_WRITE,
NO_READ
};
2004-01-11 10:21:06 +01:00
static attrib *
mk_piracy(const faction * f, direction_t target_dir)
{
attrib * a = a_new(&at_piracy_direction);
piracy_data * data = a->data.v;
data->follow = f;
data->dir = target_dir;
return a;
}
static int
2001-01-25 10:37:55 +01:00
piracy(unit *u)
{
region *r = u->region;
ship *sh = u->ship, *sh2;
direction_t dir, target_dir = NODIRECTION;
int aff[MAXDIRECTIONS];
int saff = 0;
int *il;
const char *s;
2001-01-25 10:37:55 +01:00
boolean all = true;
attrib *a;
if (!sh) {
2001-01-25 10:37:55 +01:00
cmistake(u, findorder(u, u->thisorder), 144, MSG_MOVE);
return 0;
2001-01-25 10:37:55 +01:00
}
if (!fval(u, UFL_OWNER)) {
2001-01-25 10:37:55 +01:00
cmistake(u, findorder(u, u->thisorder), 46, MSG_MOVE);
return 0;
2001-01-25 10:37:55 +01:00
}
/* Feststellen, ob schon ein anderer alliierter Pirat ein
* Ziel gefunden hat. */
il = intlist_init();
s = getstrtoken();
while(s && *s) {
il = intlist_add(il, atoi(s));
all = false;
s = getstrtoken();
}
for (a = a_find(r->attribs, &at_piracy_direction); a; a=a->nexttype) {
2004-01-11 10:21:06 +01:00
piracy_data * data = a->data.v;
const faction *f = data->follow;
2001-01-25 10:37:55 +01:00
2004-01-11 10:21:06 +01:00
if (alliedunit(u, f, HELP_FIGHT) && intlist_find(il, a->data.sa[1])) {
target_dir = data->dir;
2001-01-25 10:37:55 +01:00
break;
}
}
/* Wenn nicht, sehen wir, ob wir ein Ziel finden. */
if (target_dir == NODIRECTION) {
/* Einheit ist also Kapit<69>n. Jetzt gucken, in wievielen
* Nachbarregionen potentielle Opfer sind. */
for(dir = 0; dir < MAXDIRECTIONS; dir++) {
region *rc = rconnect(r, dir);
aff[dir] = 0;
if(rc && (terrain[rterrain(rc)].flags & SWIM_INTO)
&& check_takeoff(sh, r, rc) == true) {
for(sh2 = rc->ships; sh2; sh2 = sh2->next) {
2002-11-02 17:05:26 +01:00
unit * cap = shipowner(rc, sh2);
2002-10-20 12:40:42 +02:00
if (cap && (intlist_find(il, cap->faction->no) || all)) {
2001-01-25 10:37:55 +01:00
aff[dir]++;
}
}
/* Und aufaddieren. */
saff += aff[dir];
}
}
if(saff != 0) {
saff = rand()%saff;
for(dir = 0; dir < MAXDIRECTIONS; dir++) {
if(saff < aff[dir]) break;
saff -= aff[dir];
}
target_dir = dir;
2004-01-11 10:21:06 +01:00
a = a_add(&r->attribs, mk_piracy(u->faction, target_dir));
2001-01-25 10:37:55 +01:00
}
}
free(il);
/* Wenn kein Ziel gefunden, entsprechende Meldung generieren */
if (target_dir == NODIRECTION) {
2001-01-25 10:37:55 +01:00
add_message(&u->faction->msgs,
new_message(u->faction, "piratenovictim%h:ship%r:region", sh, r));
return 0;
2001-01-25 10:37:55 +01:00
}
/* Meldung generieren */
add_message(&u->faction->msgs, new_message(u->faction,
"piratesawvictim%h:ship%r:region%d:dir", sh, r, target_dir));
/* Befehl konstruieren */
sprintf(buf, "%s %s", locale_string(u->faction->locale, keywords[K_MOVE]),
locale_string(u->faction->locale, directions[target_dir]));
2001-01-25 10:37:55 +01:00
set_string(&u->thisorder, buf);
/* Bewegung ausf<73>hren */
igetkeyword(u->thisorder, u->faction->locale); /* NACH ignorieren */
return move(u, true);
2001-01-25 10:37:55 +01:00
}
static void
age_traveldir(region *r)
{
attrib *a = a_find(r->attribs, &at_traveldir);
while(a) {
a->data.ca[3]--;
if(a->data.ca[3] <= 0) {
attrib *an = a->nexttype;
a_remove(&r->attribs, a);
a = an;
} else {
a = a->nexttype;
}
}
}
static direction_t
hunted_dir(attrib *at, int id)
{
attrib *a = a_find(at, &at_traveldir_new);
2001-01-25 10:37:55 +01:00
while (a!=NULL) {
traveldir *t = (traveldir *)(a->data.v);
if (t->no == id) return t->dir;
a = a->nexttype;
}
2001-01-25 10:37:55 +01:00
return NODIRECTION;
2001-01-25 10:37:55 +01:00
}
static boolean
can_move(const unit * u)
{
if (u->race->flags & RCF_CANNOTMOVE) return false;
if (get_movement(&u->attribs, MV_CANNOTMOVE)) return false;
return true;
}
static int
2001-01-25 10:37:55 +01:00
hunt(unit *u)
{
region *rc = u->region;
int moves, id;
char command[256];
direction_t dir;
2001-01-25 10:37:55 +01:00
if(!u->ship) {
cmistake(u, findorder(u, u->thisorder), 144, MSG_MOVE);
return 0;
} else if(!fval(u, UFL_OWNER)) {
cmistake(u, findorder(u, u->thisorder), 146, MSG_MOVE);
return 0;
} else if(attacked(u)) {
cmistake(u, findorder(u, u->thisorder), 52, MSG_MOVE);
return 0;
} else if (!can_move(u)) {
cmistake(u, findorder(u, u->thisorder), 55, MSG_MOVE);
return 0;
}
2001-01-25 10:37:55 +01:00
id = getshipid();
2001-01-25 10:37:55 +01:00
if (id <= 0) {
cmistake(u, findorder(u, u->thisorder), 20, MSG_MOVE);
return 0;
}
2001-01-25 10:37:55 +01:00
dir = hunted_dir(rc->attribs, id);
2001-01-25 10:37:55 +01:00
if (dir == NODIRECTION) {
ship * sh = findship(id);
if (sh==NULL || sh->region!=rc) {
cmistake(u, findorder(u, u->thisorder), 20, MSG_MOVE);
}
return 0;
}
2001-01-25 10:37:55 +01:00
sprintf(command, "%s %s", locale_string(u->faction->locale, keywords[K_MOVE]),
locale_string(u->faction->locale, directions[dir]));
moves = 1;
rc = rconnect(rc, dir);
while (moves < shipspeed(u->ship, u) && (dir = hunted_dir(rc->attribs, id)) != NODIRECTION)
{
strcat(command, " ");
strcat(command, locale_string(u->faction->locale, directions[dir]));
moves++;
rc = rconnect(rc, dir);
}
/* In command steht jetzt das NACH-Kommando. */
2001-01-25 10:37:55 +01:00
igetkeyword(command, u->faction->locale); /* NACH ignorieren und Parsing initialisieren. */
2001-01-25 10:37:55 +01:00
/* NACH ausf<73>hren */
if (move(u, false)!=0) {
/* niemand sollte auf einen kapit<69>n direkt ein folge haben, oder? */
assert(1==0);
}
fset(u, UFL_LONGACTION); /* Von Hand setzen, um Endlosschleife zu vermeiden, wenn Verfolgung nicht erfolgreich */
return 1; /* true -> Einheitenliste von vorne durchgehen */
2001-01-25 10:37:55 +01:00
}
void
destroy_damaged_ships(void)
{
region *r;
ship *sh,*shn;
for(r=regions;r;r=r->next) {
for(sh=r->ships;sh;) {
shn = sh->next;
if (sh->damage>=sh->size * DAMAGE_SCALE) {
destroy_ship(sh);
2001-01-25 10:37:55 +01:00
}
sh = shn;
}
}
}
#ifdef TODO /* Wenn Feature ausgearbeitet */
static boolean
2001-01-25 10:37:55 +01:00
is_disorientated(unit *u)
{
static boolean init = false;
static const curse_type * shipconf_ct, * regconf_ct;
if (!init) {
init = true;
regconf_ct = ct_find("disorientationzone");
shipconf_ct = ct_find("shipdisorientation");
}
if (u->ship && curse_active(get_curse(u->ship->attribs, shipconf_ct)))
return true;
2001-01-25 10:37:55 +01:00
if (curse_active(get_curse(u->region->attribs, regconf_ct)))
return true;
2001-01-25 10:37:55 +01:00
return false;
2001-01-25 10:37:55 +01:00
}
void
regain_orientation(region * r)
{
ship *sh;
curse *c;
unit *u, *cap;
for(sh = r->ships; sh; sh = sh->next) {
c = get_curse(sh->attribs, C_DISORIENTATION, 0);
if(!c) continue;
/* Das Schiff bekommt seine Orientierung zur<75>ck, wenn es:
* a) An Land treibt.
* b) Gl<EFBFBD>ck hat.
* c) In einer Region mit einem nicht verwirrten alliierten
* Schiff steht.
*/
cap = shipowner(r, sh);
if(r->terrain != T_OCEAN || rand() % 10 >= storms[month(0)]) {
remove_curse(&sh->attribs, C_DISORIENTATION, 0);
add_message(&cap->faction->msgs,
new_message(cap->faction, "shipnoconf%h:ship", sh));
continue;
}
for(u=r->units;u;u=u->next) {
if(u != cap
&& allied(cap, u->faction, HELP_GUARD)
&& is_disorientated(u) == false) {
remove_curse(&sh->attribs, C_DISORIENTATION, 0);
add_message(&cap->faction->msgs,
new_message(cap->faction, "shipnoconf%h:ship", sh));
break;
}
}
}
}
#endif
/* Bewegung, Verfolgung, Piraterie */
/** ships that folow other ships
* Dann generieren die jagenden Einheiten ihre Befehle und
* bewegen sich.
* Following the trails of other ships.
*/
static void
move_hunters(void)
2001-01-25 10:37:55 +01:00
{
region *r;
2001-01-25 10:37:55 +01:00
for (r = regions; r; r = r->next) {
unit * u;
2001-01-25 10:37:55 +01:00
for (u = r->units; u;) {
unit *u2 = u->next;
strlist *o;
param_t p;
for (o=u->orders;o;o=o->next) {
if(igetkeyword(o->s, u->faction->locale) == K_FOLLOW) {
2001-01-25 10:37:55 +01:00
if(attacked(u)) {
cmistake(u, o->s, 52, MSG_MOVE);
break;
} else if (!can_move(u)) {
2001-01-25 10:37:55 +01:00
cmistake(u, o->s, 55, MSG_MOVE);
break;
2001-01-25 10:37:55 +01:00
}
p = getparam(u->faction->locale);
2001-01-25 10:37:55 +01:00
if(p != P_SHIP) {
if(p != P_UNIT) {
cmistake(u, o->s, 240, MSG_MOVE);
}
break;
2001-01-25 10:37:55 +01:00
}
if (!fval(u, UFL_LONGACTION) && !fval(u, UFL_HUNGER) && hunt(u)) {
u2 = r->units;
break;
2001-01-25 10:37:55 +01:00
}
}
}
u = u2;
}
}
}
2001-01-25 10:37:55 +01:00
/** Piraten and Drift
*
*/
static void
move_pirates(void)
{
region * r;
2001-01-25 10:37:55 +01:00
for (r = regions; r; r = r->next) {
unit ** up = &r->units;
2001-01-25 10:37:55 +01:00
/* Abtreiben von besch<63>digten, unterbemannten, <20>berladenen Schiffen */
drifting_ships(r);
2001-01-25 10:37:55 +01:00
while (*up) {
unit *u = *up;
if (!fval(u, UFL_LONGACTION) && igetkeyword(u->thisorder, u->faction->locale) == K_PIRACY) {
piracy(u);
fset(u, UFL_LONGACTION);
}
2001-01-25 10:37:55 +01:00
if (*up==u) {
/* not moved, use next unit */
up = &u->next;
} else if (*up && (*up)->region!=r) {
/* moved the previous unit along with u (units on same ship)
* must start from the beginning again */
up = &r->units;
}
/* else *up is already the next unit */
}
2001-01-25 10:37:55 +01:00
a_removeall(&r->attribs, &at_piracy_direction);
age_traveldir(r);
}
}
void
movement(void)
{
int ships;
/* Initialize the additional encumbrance by transported units */
init_drive();
/* Move ships in last phase, others first
* This is to make sure you can't land someplace and then get off the ship
* in the same turn.
*/
for (ships=0;ships<=1;++ships) {
region * r = regions;
while (r!=NULL) {
unit ** up = &r->units;
boolean repeat = false;
while (*up) {
unit *u = *up;
keyword_t kword = igetkeyword(u->thisorder, u->faction->locale);
switch (kword) {
case K_ROUTE:
case K_MOVE:
if (attacked(u)) {
cmistake(u, findorder(u, u->thisorder), 52, MSG_MOVE);
set_string(&u->thisorder, "");
} else if (!can_move(u)) {
cmistake(u, findorder(u, u->thisorder), 55, MSG_MOVE);
set_string(&u->thisorder, "");
} else {
if (ships) {
if (u->ship && fval(u, UFL_OWNER)) {
if (move(u, true)!=0) repeat = true;
}
} else {
if (u->ship==NULL || !fval(u, UFL_OWNER)) {
if (move(u, true)!=0) repeat = true;
}
}
}
break;
}
if (u->region==r) {
/* not moved, use next unit */
up = &u->next;
} else if (*up && (*up)->region!=r) {
/* moved the previous unit along with u (units on ships, for example)
* must start from the beginning again */
up = &r->units;
}
/* else *up is already the next unit */
}
if (!repeat) r = r->next;
}
}
move_hunters();
move_pirates();
2001-01-25 10:37:55 +01:00
}
/** Overrides long orders with a FOLLOW order if the target is moving.
* FOLLOW SHIP is a long order, and doesn't need to be treated in here.
*/
2001-01-25 10:37:55 +01:00
void
follow_unit(void)
2001-01-25 10:37:55 +01:00
{
region * r;
for (r=regions;r;r=r->next) {
2001-01-25 10:37:55 +01:00
unit * u;
for (u=r->units;u;u=u->next) {
attrib * a;
2001-01-25 10:37:55 +01:00
strlist * o;
if (fval(u, UFL_LONGACTION) || fval(u, UFL_HUNGER)) continue;
a = a_find(u->attribs, &at_follow);
2001-01-25 10:37:55 +01:00
for (o=u->orders;o;o=o->next) {
const struct locale * lang = u->faction->locale;
if (igetkeyword(o->s, lang) == K_FOLLOW && getparam(lang) == P_UNIT) {
2001-01-25 10:37:55 +01:00
int id = read_unitid(u->faction, r);
if (id>0) {
2001-02-07 21:42:31 +01:00
unit * uf = findunit(id);
2001-01-25 10:37:55 +01:00
if (!a) {
2001-02-07 21:42:31 +01:00
a = a_add(&u->attribs, make_follow(uf));
} else {
a->data.v = uf;
2001-01-25 10:37:55 +01:00
}
} else if (a) {
a_remove(&u->attribs, a);
a = NULL;
}
}
}
if (a && !fval(u, UFL_MOVED)) {
2001-01-25 10:37:55 +01:00
unit * u2 = a->data.v;
2001-01-25 10:37:55 +01:00
if (!u2 || u2->region!=r || !cansee(u->faction, r, u2, 0))
continue;
for (o=u2->orders;o;o=o->next) {
switch (igetkeyword(o->s, u2->faction->locale)) {
2001-01-25 10:37:55 +01:00
case K_MOVE:
case K_ROUTE:
case K_DRIVE:
case K_FOLLOW:
2001-01-25 10:37:55 +01:00
case K_PIRACY:
fset(u, UFL_FOLLOWING);
fset(u2, UFL_FOLLOWED);
2001-01-25 10:37:55 +01:00
set_string(&u->thisorder, "");
break;
default:
continue;
}
break;
}
}
}
}
}