2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2001-02-14 08:44:57 +01:00
|
|
|
|
* $Id: movement.c,v 1.9 2001/02/14 07:44:57 enno Exp $
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2000
|
|
|
|
|
* 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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#define SAFE_COASTS /* Schiffe an der K<>ste treiben nicht ab */
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include "eressea.h"
|
|
|
|
|
#include "movement.h"
|
|
|
|
|
|
|
|
|
|
#include "item.h"
|
|
|
|
|
#include "plane.h"
|
|
|
|
|
#include "alchemy.h"
|
|
|
|
|
#include "message.h"
|
|
|
|
|
#include "faction.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "build.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "curse.h"
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
#include "border.h"
|
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "ship.h"
|
|
|
|
|
#include "karma.h"
|
|
|
|
|
#include "building.h"
|
|
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
|
#include <goodies.h>
|
|
|
|
|
|
|
|
|
|
/* 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>
|
|
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
|
|
|
|
|
|
attrib_type at_traveldir = {
|
|
|
|
|
"traveldir",
|
|
|
|
|
DEFAULT_INIT,
|
|
|
|
|
DEFAULT_FINALIZE,
|
|
|
|
|
DEFAULT_AGE, /* Weil normales Aging an ung<6E>nstiger Stelle */
|
|
|
|
|
DEFAULT_WRITE,
|
|
|
|
|
DEFAULT_READ
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
a_traveldir_new_init(attrib *a)
|
|
|
|
|
{
|
|
|
|
|
a->data.v = calloc(1, sizeof(traveldir));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
a_traveldir_new_finalize(attrib *a)
|
|
|
|
|
{
|
|
|
|
|
free(a->data.v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
a_traveldir_new_age(attrib *a)
|
|
|
|
|
{
|
|
|
|
|
traveldir *t = (traveldir *)(a->data.v);
|
|
|
|
|
|
|
|
|
|
t->age--;
|
|
|
|
|
if(t->age == 0) return 0;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
a_traveldir_new_read(attrib *a, FILE *f)
|
|
|
|
|
{
|
|
|
|
|
traveldir *t = (traveldir *)(a->data.v);
|
|
|
|
|
int no, age, dir;
|
|
|
|
|
|
|
|
|
|
fscanf(f, "%d %d %d", &no, &dir, &age);
|
|
|
|
|
t->no = no;
|
|
|
|
|
t->dir = (direction_t)dir;
|
|
|
|
|
t->age = age;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
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
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
direction_t
|
|
|
|
|
getdirection(void)
|
|
|
|
|
{
|
|
|
|
|
return finddirection(getstrtoken());
|
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
#ifdef NEW_DRIVE
|
|
|
|
|
static attrib_type at_driveweight = {
|
|
|
|
|
"driveweight", NULL, NULL, NULL, NULL, NULL
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
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);
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(race[u->race].flags & 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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
walkingcapacity(unit * u)
|
|
|
|
|
{
|
|
|
|
|
int n, personen, pferde, pferde_fuer_wagen;
|
|
|
|
|
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);
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(race[u->race].flags & 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 (u->race == RC_TROLL) {
|
|
|
|
|
/* 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;
|
|
|
|
|
|
|
|
|
|
if (u->race == RC_TROLL) {
|
|
|
|
|
n += personen * PERSONCAPACITY(u) * 2;
|
|
|
|
|
} else {
|
|
|
|
|
n += personen * PERSONCAPACITY(u);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n += get_effect(u, oldpotiontype[P_STRONG]) * PERSONCAPACITY(u);
|
|
|
|
|
|
|
|
|
|
n += min(get_item(u, I_TROLLBELT), u->number) * STRENGTHCAPACITY;
|
|
|
|
|
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
canwalk(unit * u)
|
|
|
|
|
{
|
|
|
|
|
int wagen, maxwagen;
|
|
|
|
|
int pferde, maxpferde;
|
|
|
|
|
|
|
|
|
|
if (dragon(u)) return 0;
|
|
|
|
|
|
|
|
|
|
wagen = get_item(u, I_WAGON);
|
|
|
|
|
pferde = get_item(u, I_HORSE);
|
|
|
|
|
|
|
|
|
|
maxwagen = effskill(u, SK_RIDING) * u->number * 2;
|
|
|
|
|
if (u->race == RC_TROLL) {
|
|
|
|
|
maxwagen = max(maxwagen, u->number / 4);
|
|
|
|
|
}
|
|
|
|
|
maxpferde = effskill(u, SK_RIDING) * u->number * 4 + u->number;
|
|
|
|
|
|
|
|
|
|
if (pferde > maxpferde)
|
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
|
|
#ifdef NEW_DRIVE
|
|
|
|
|
if (walkingcapacity(u) - eff_weight(u) >= 0)
|
|
|
|
|
#else
|
|
|
|
|
if (walkingcapacity(u) - weight(u) >= 0)
|
|
|
|
|
#endif
|
|
|
|
|
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_item(u, I_HORSE)) return false;
|
|
|
|
|
|
|
|
|
|
if(get_item(u, I_PEGASUS) >= u->number && effskill(u, SK_RIDING) >= 4)
|
|
|
|
|
return true;
|
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(race[u->race].flags & RCF_FLY) return true;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
canswim(unit *u)
|
|
|
|
|
{
|
|
|
|
|
if(get_item(u, I_HORSE)) return false;
|
|
|
|
|
|
|
|
|
|
if(get_item(u, I_DOLPHIN) >= u->number && effskill(u, SK_RIDING) >= 4)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
if(fspecial(u->faction, FS_AMPHIBIAN)) return true;
|
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(race[u->race].flags & RCF_FLY) return true;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(race[u->race].flags & RCF_SWIM) return true;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
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;
|
|
|
|
|
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(!(race[u->race].flags & RCF_HORSE)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
&& ((pferde == 0 && unicorns == 0)
|
|
|
|
|
|| pferde > maxpferde || unicorns > maxunicorns)) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef NEW_DRIVE
|
|
|
|
|
if(ridingcapacity(u) - eff_weight(u) >= 0) {
|
|
|
|
|
#else
|
|
|
|
|
if(ridingcapacity(u) - weight(u) >= 0) {
|
|
|
|
|
#endif
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if(pferde == 0 && unicorns >= u->number && !(race[u->race].flags & RCF_HORSE)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return 2;
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
cansail(const region * r, ship * sh)
|
|
|
|
|
{
|
|
|
|
|
int n = 0, p = 0;
|
|
|
|
|
unit *u;
|
|
|
|
|
|
|
|
|
|
assert(sh->type->construction->improvement==NULL); /* sonst ist construction::size nicht ship_type::maxsize */
|
|
|
|
|
if (sh->size!=sh->type->construction->maxsize)
|
|
|
|
|
return false;
|
|
|
|
|
for (u = r->units; u; u = u->next)
|
|
|
|
|
if (u->ship == sh) {
|
|
|
|
|
n += weight(u);
|
|
|
|
|
p += u->number;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
if (u->ship == sh)
|
|
|
|
|
n += eff_skill(u, SK_SAILING, r) * u->number;
|
|
|
|
|
return n >= sh->type->sumskill;
|
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
do_maelstrom(region *r, unit *u)
|
|
|
|
|
{
|
|
|
|
|
int damage;
|
|
|
|
|
|
|
|
|
|
damage = rand()%150 - eff_skill(u, SK_SAILING, r)*5;
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
damage_ship(u->ship, 0.01*damage);
|
|
|
|
|
|
|
|
|
|
if (u->ship->damage >= u->ship->size * DAMAGE_SCALE) {
|
|
|
|
|
add_message(&u->faction->msgs,
|
|
|
|
|
new_message(u->faction, "entermaelstrom%r:region%h:ship%i:damage%i:sink", r, u->ship, damage, 1));
|
|
|
|
|
destroy_ship(u->ship, r);
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs,
|
|
|
|
|
new_message(u->faction, "entermaelstrom%r:region%h:ship%i:damage%i:sink", r, u->ship, damage, 0));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
travelthru(unit * u, region * r)
|
|
|
|
|
{
|
|
|
|
|
attrib *ru = a_add(&r->attribs, a_new(&at_travelunit));
|
|
|
|
|
|
|
|
|
|
ru->data.v = u;
|
|
|
|
|
fset(u, FL_TRAVELTHRU);
|
|
|
|
|
u->faction->first = 0;
|
|
|
|
|
u->faction->last = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ship *
|
|
|
|
|
move_ship(ship * sh, region * from, region * to, region ** route)
|
|
|
|
|
{
|
|
|
|
|
unit *u = from->units;
|
|
|
|
|
unit **ulist = &to->units;
|
|
|
|
|
direction_t dir;
|
|
|
|
|
attrib *a;
|
|
|
|
|
|
|
|
|
|
translist(&from->ships, &to->ships, sh);
|
|
|
|
|
|
|
|
|
|
while (u) {
|
|
|
|
|
unit *nu = u->next;
|
|
|
|
|
|
|
|
|
|
if (u->ship == sh)
|
|
|
|
|
{
|
|
|
|
|
if (route) {
|
|
|
|
|
region **ri = route;
|
|
|
|
|
region **rn;
|
|
|
|
|
|
|
|
|
|
while (*ri) {
|
|
|
|
|
rn = ri+1;
|
|
|
|
|
if(*rn) {
|
|
|
|
|
dir = reldirection(*ri, *rn);
|
|
|
|
|
/* TODO: Speichermechanismus portabel machen. */
|
|
|
|
|
a = a_find((*ri)->attribs, &at_traveldir_new);
|
|
|
|
|
while (a) {
|
|
|
|
|
if(a->data.sa[0] == sh->no) break;
|
|
|
|
|
a = a->nexttype;
|
|
|
|
|
}
|
|
|
|
|
if(!a) {
|
|
|
|
|
a = a_add(&((*ri)->attribs), a_new(&at_traveldir_new));
|
|
|
|
|
{
|
|
|
|
|
traveldir *t = (traveldir *)(a->data.v);
|
|
|
|
|
t->no = sh->no;
|
|
|
|
|
t->dir = dir;
|
|
|
|
|
t->age = 2;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
traveldir *t = (traveldir *)(a->data.v);
|
|
|
|
|
t->dir = dir;
|
|
|
|
|
t->age = 2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
travelthru(u, *ri++);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
u->ship = NULL; /* damit move_unit() kein leave() macht */
|
|
|
|
|
move_unit(u, to, ulist);
|
|
|
|
|
#if 0
|
|
|
|
|
/* das braucht man (fast) sicher nicht. (Enno) */
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
#endif
|
|
|
|
|
ulist = &u->next;
|
|
|
|
|
u->ship = sh;
|
|
|
|
|
if (route && eff_skill(u, SK_SAILING, from) >= 1) {
|
|
|
|
|
change_skill(u, SK_SAILING, PRODUCEEXP * u->number);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
u = nu;
|
|
|
|
|
}
|
|
|
|
|
return sh;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
drifting_ships(region * r)
|
|
|
|
|
{
|
|
|
|
|
direction_t d;
|
|
|
|
|
ship *sh, *sh2;
|
|
|
|
|
unit *captain;
|
|
|
|
|
region * route[2] = { 0, 0 };
|
|
|
|
|
|
|
|
|
|
if (rterrain(r) == T_OCEAN) {
|
|
|
|
|
for (sh = r->ships; sh;) {
|
|
|
|
|
sh2 = sh->next;
|
|
|
|
|
|
|
|
|
|
/* Schiff schon abgetrieben oder durch Zauber gesch<63>tzt? */
|
|
|
|
|
if (sh->drifted == 1 || is_cursed(sh->attribs, C_SHIP_NODRIFT, 0)) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 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. */
|
|
|
|
|
|
|
|
|
|
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)) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Leuchtturm: Ok. */
|
|
|
|
|
if (check_leuchtturm(r, NULL)) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Auswahl einer Richtung: Zuerst auf Land, dann
|
|
|
|
|
* zuf<EFBFBD>llig. Falls unm<EFBFBD>gliches Resultat: vergi<EFBFBD> es. */
|
|
|
|
|
|
|
|
|
|
for (d = 0; d != MAXDIRECTIONS; d++)
|
|
|
|
|
if (rconnect(r, d) && rterrain(rconnect(r, d)) != T_OCEAN)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (d == MAXDIRECTIONS)
|
|
|
|
|
d = (direction_t)(rand() % MAXDIRECTIONS);
|
|
|
|
|
|
|
|
|
|
if (!rconnect(r, d)) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
region * r2 = rconnect(r, d);
|
|
|
|
|
int c = 0;
|
|
|
|
|
while (sh->type->coast[c]!=NOTERRAIN) {
|
|
|
|
|
if (rterrain(r2) == sh->type->coast[c]) break;
|
|
|
|
|
++c;
|
|
|
|
|
}
|
|
|
|
|
if (sh->type->coast[c]==NOTERRAIN) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if(!(terrain[rterrain(rconnect(r, d))].flags & SAIL_INTO)) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Das Schiff und alle Einheiten darin werden nun von r
|
|
|
|
|
* nach rconnect(r, d) verschoben. Danach eine Meldung. */
|
|
|
|
|
#if 0
|
|
|
|
|
/* ich bin fast sicher, das man das nicht braucht. (enno) */
|
|
|
|
|
for (f = factions; f; f = f->next)
|
|
|
|
|
freset(f, FL_DH);
|
|
|
|
|
#endif
|
|
|
|
|
if (sh->drifted) {
|
|
|
|
|
/* wenn schon einmal abgetrieben, dann durchreise eintragen */
|
|
|
|
|
route[0] = r;
|
|
|
|
|
sh = move_ship(sh, r, rconnect(r, d), route);
|
|
|
|
|
} else sh = move_ship(sh, r, rconnect(r, d), NULL);
|
|
|
|
|
if (!sh) {
|
|
|
|
|
sh = sh2;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sh->drifted = 1;
|
|
|
|
|
|
|
|
|
|
if (rterrain(rconnect(r, d)) != T_OCEAN) {
|
|
|
|
|
/* Alle Attribute
|
|
|
|
|
* sicherheitshalber loeschen und
|
|
|
|
|
* gegebenenfalls neu setzen. */
|
|
|
|
|
sh->coast = back[d];
|
|
|
|
|
}
|
|
|
|
|
damage_ship(sh, 0.02);
|
|
|
|
|
|
|
|
|
|
if (sh->damage>=sh->size * DAMAGE_SCALE)
|
|
|
|
|
destroy_ship(sh, rconnect(r, d));
|
|
|
|
|
|
|
|
|
|
sh = sh2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
|
coords_or_direction(region *r, faction *f, int dir)
|
|
|
|
|
{
|
2001-02-09 14:53:53 +01:00
|
|
|
|
static char lbuf[32];
|
2001-01-25 10:37:55 +01:00
|
|
|
|
plane *pl = getplane(r);
|
|
|
|
|
|
|
|
|
|
if(fval(pl, PFL_NOCOORDS)) {
|
2001-02-09 14:53:53 +01:00
|
|
|
|
strcpy(lbuf, directions[dir]);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
} else {
|
2001-02-09 14:53:53 +01:00
|
|
|
|
sprintf(lbuf, "(%d,%d)",region_x(r,f), region_y(r,f));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-02-09 14:53:53 +01:00
|
|
|
|
return lbuf;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
ship_in_storm(unit *cap, region *next_point)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
if(is_cursed(cap->ship->attribs, C_SHIP_NODRIFT, 0)) return;
|
|
|
|
|
|
|
|
|
|
add_message(&cap->faction->msgs,
|
|
|
|
|
new_message(cap->faction, "storm%h:ship%r:region%i:sink",
|
|
|
|
|
cap->ship, next_point, cap->ship->damage>=cap->ship->size * DAMAGE_SCALE));
|
|
|
|
|
|
|
|
|
|
/* Ver<65>nderung der Drehrichtung */
|
|
|
|
|
i = rand()%2==0?1:-1;
|
|
|
|
|
#if 0 /* make a temporary attribute for this */
|
|
|
|
|
cap->wants += i;
|
|
|
|
|
#endif
|
|
|
|
|
/* Merken f<>r Sperren des Anlandens. */
|
|
|
|
|
if(i) fset(cap, FL_STORM);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define USE_CREATION 0
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
present(region * r, unit * u)
|
|
|
|
|
{
|
|
|
|
|
return (boolean) (u && u->region == r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
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));
|
|
|
|
|
} else if (!allied(target, u->faction, HELP_ALL)
|
|
|
|
|
&& 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 ! */
|
|
|
|
|
|
|
|
|
|
unit *
|
|
|
|
|
bewegung_blockiert_von(unit * reisender, region * r)
|
|
|
|
|
{
|
|
|
|
|
unit *ru;
|
|
|
|
|
int chance;
|
|
|
|
|
int hat_kontaktiert;
|
|
|
|
|
|
|
|
|
|
for (ru = r->units; ru; ru = ru->next) {
|
|
|
|
|
|
|
|
|
|
/* Ermittle ob kontaktiert */
|
|
|
|
|
|
|
|
|
|
hat_kontaktiert = 0;
|
|
|
|
|
if (ucontact(ru, reisender)) hat_kontaktiert = 1;
|
|
|
|
|
|
|
|
|
|
if (ru->faction != reisender->faction
|
|
|
|
|
&& !allied(ru, reisender->faction, HELP_GUARD)
|
|
|
|
|
&& hat_kontaktiert == 0
|
|
|
|
|
&& !illusionary(reisender)) {
|
|
|
|
|
if (getguard(ru)&GUARD_TRAVELTHRU) {
|
|
|
|
|
if (get_item(reisender, I_RING_OF_INVISIBILITY) &&
|
|
|
|
|
get_item(reisender, I_RING_OF_INVISIBILITY) >= reisender->number &&
|
|
|
|
|
!get_item(ru, I_AMULET_OF_TRUE_SEEING)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
chance = eff_skill(ru, SK_OBSERVATION, r) * ru->number;
|
|
|
|
|
chance -= eff_stealth(reisender, r) * eff_stealth(reisender, r);
|
|
|
|
|
chance += min(get_item(ru, I_AMULET_OF_TRUE_SEEING), ru->number);
|
|
|
|
|
|
|
|
|
|
if (rand() % 100 < chance) {
|
|
|
|
|
return ru;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (unit *) NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
is_guardian(unit * u2, unit *u, unsigned int mask)
|
|
|
|
|
{
|
|
|
|
|
if (u2->faction != u->faction
|
|
|
|
|
&& getguard(u2)&mask
|
|
|
|
|
&& u2->number
|
|
|
|
|
&& !ucontact(u2, u) && !besieged(u2)
|
|
|
|
|
&& allied(u2, u->faction, HELP_GUARD)
|
|
|
|
|
!= HELP_GUARD
|
|
|
|
|
#ifdef WACH_WAFF
|
|
|
|
|
&& armedmen(u2)
|
|
|
|
|
#endif
|
|
|
|
|
) 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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *shortdirections[MAXDIRECTIONS+2] =
|
|
|
|
|
{
|
|
|
|
|
"NW",
|
|
|
|
|
"NO",
|
|
|
|
|
"Ost",
|
|
|
|
|
"SO",
|
|
|
|
|
"SW",
|
|
|
|
|
"West",
|
|
|
|
|
"",
|
|
|
|
|
"Pause"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
cycle_route(unit *u, int gereist)
|
|
|
|
|
{
|
|
|
|
|
int cm = 0;
|
|
|
|
|
char tail[1024];
|
|
|
|
|
char neworder[1024];
|
|
|
|
|
char *token;
|
|
|
|
|
direction_t d = NODIRECTION;
|
|
|
|
|
boolean paused = false;
|
|
|
|
|
boolean pause;
|
|
|
|
|
|
|
|
|
|
if (igetkeyword(u->thisorder) != K_ROUTE) return;
|
|
|
|
|
tail[0] = '\0';
|
|
|
|
|
|
|
|
|
|
strcpy(neworder, keywords[K_ROUTE]);
|
|
|
|
|
|
|
|
|
|
for (cm=0;;++cm) {
|
|
|
|
|
pause = false;
|
|
|
|
|
token = getstrtoken();
|
|
|
|
|
if(findparam(token) == P_PAUSE) {
|
|
|
|
|
pause = true;
|
|
|
|
|
} else {
|
|
|
|
|
d = finddirection(token);
|
|
|
|
|
if (d==NODIRECTION) break;
|
|
|
|
|
}
|
|
|
|
|
if (cm<gereist) {
|
|
|
|
|
/* hier sollte keine PAUSE auftreten */
|
|
|
|
|
if (!pause) strcat(strcat(tail, " "), shortdirections[d]);
|
|
|
|
|
}
|
|
|
|
|
else if (cm==gereist && !paused && pause){
|
|
|
|
|
strcat(strcat(tail, " "), parameters[P_PAUSE]);
|
|
|
|
|
paused=true;
|
|
|
|
|
}
|
|
|
|
|
/* da PAUSE nicht in ein shortdirections[d] umgesetzt wird (ist
|
|
|
|
|
* hier keine normale direction), muss jede PAUSE einzeln
|
|
|
|
|
* herausgefiltert und explizit gesetzt werden */
|
|
|
|
|
else if (pause) {
|
|
|
|
|
strcat(strcat(neworder, " "), parameters[P_PAUSE]);
|
|
|
|
|
} else
|
|
|
|
|
strcat(strcat(neworder, " "), shortdirections[d]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
strcat(neworder, tail);
|
|
|
|
|
set_string(&u->lastorder, neworder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef NEW_DRIVE
|
|
|
|
|
void
|
|
|
|
|
init_drive(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u, *ut, *u2;
|
|
|
|
|
strlist *S, *St;
|
|
|
|
|
int w;
|
|
|
|
|
|
|
|
|
|
/* This function 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. */
|
|
|
|
|
|
|
|
|
|
for (r=regions; r; r=r->next) {
|
|
|
|
|
for (u=r->units; u; u=u->next) {
|
|
|
|
|
w = 0;
|
|
|
|
|
|
|
|
|
|
for (S = u->orders; S; S = S->next) if(igetkeyword(S->s) == K_TRANSPORT) {
|
|
|
|
|
ut = getunit(r, u);
|
|
|
|
|
if(!ut) continue;
|
|
|
|
|
|
|
|
|
|
for (St = ut->orders; St; St = St->next) if (igetkeyword(St->s) == K_DRIVE) {
|
|
|
|
|
u2 = getunit(r, u);
|
|
|
|
|
if(u2 == u) {
|
|
|
|
|
w += weight(ut);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(w > 0) a_add(&u->attribs, a_new(&at_driveweight))->data.i = w;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
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, FL_OWNER) && rterrain(from)==T_OCEAN) {
|
|
|
|
|
/* wir sind auf einem schiff und auf dem ozean */
|
|
|
|
|
if (rterrain(to)==T_OCEAN || (u->race != RC_AQUARIAN && !dragon(u))) {
|
|
|
|
|
/* Drache oder Meermensch sind wir auch nicht */
|
|
|
|
|
return ETRAVEL_NOSWIM;
|
|
|
|
|
} else if (get_item(u, I_HORSE) > 0) {
|
|
|
|
|
return ETRAVEL_NORIDE|ETRAVEL_NOSWIM;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
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, const region * r2)
|
|
|
|
|
{
|
|
|
|
|
/* wenn es hier genug strassen gibt, und verbunden ist, und es dort
|
|
|
|
|
* genug strassen gibt, dann existiert eine strasse in diese richtung */
|
|
|
|
|
|
|
|
|
|
direction_t i;
|
|
|
|
|
|
|
|
|
|
if (!r || !r2) return false;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i != MAXDIRECTIONS; i++)
|
|
|
|
|
if (rconnect(r, i) == r2) {
|
|
|
|
|
if (rroad(r, i) < terrain[rterrain(r)].roadreq) return false;
|
|
|
|
|
if (rroad(r2, back[i]) < terrain[rterrain(r2)].roadreq) return false;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
direction_t *
|
|
|
|
|
travel(region * first, unit * u, region * next, int flucht)
|
|
|
|
|
{
|
|
|
|
|
region *current = first;
|
|
|
|
|
int k, m, i, dh;
|
|
|
|
|
double dk;
|
|
|
|
|
unit *wache;
|
|
|
|
|
region *rv[MAXSPEED];
|
|
|
|
|
strlist *S;
|
|
|
|
|
unit *ut, *u2;
|
|
|
|
|
#ifndef NEW_DRIVE
|
|
|
|
|
int restkapazitaet;
|
|
|
|
|
#endif
|
|
|
|
|
int gereist = 0;
|
|
|
|
|
char buf2[80];
|
|
|
|
|
static direction_t route[MAXSPEED];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* 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 (!dragon(u) && rterrain(next) == T_OCEAN && (u->ship || rterrain(current)==T_OCEAN)) {
|
|
|
|
|
if (u->race != RC_AQUARIAN)
|
|
|
|
|
{
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 44, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
} else {
|
|
|
|
|
if (u->ship && get_item(u, I_HORSE) > 0) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 67, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
/* 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 NULL;
|
|
|
|
|
}
|
|
|
|
|
if (u->ship && !flucht && u->race != RC_AQUARIAN) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 143, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (rterrain(next) == T_OCEAN && u->ship && u->ship->moved == 1) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 13, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
if (!nonplayer(u)) {
|
|
|
|
|
switch (canwalk(u)) {
|
|
|
|
|
case 1:
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 57, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
case 2:
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 56, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
case 3:
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 42, MSG_MOVE);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* wir suchen so lange nach neuen Richtungen, wie es geht. Diese werden
|
|
|
|
|
* dann nacheinander ausgef<EFBFBD>hrt. */
|
|
|
|
|
|
|
|
|
|
/* im array rv[] speichern wir die Regionen ab, durch die wir wandern. */
|
|
|
|
|
|
|
|
|
|
dk = race[u->race].speed;
|
|
|
|
|
|
|
|
|
|
if(is_cursed(u->attribs, C_SPEED, 0)) {
|
|
|
|
|
int men = get_cursedmen(u->attribs, C_SPEED, 0);
|
|
|
|
|
men = max(u->number, men);
|
|
|
|
|
dk *= 1.0 + men/(double)u->number;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch(canride(u)) {
|
|
|
|
|
case 1: /* Pferd */
|
|
|
|
|
k = (int)(dk*BP_RIDING);
|
|
|
|
|
break;
|
|
|
|
|
case 2: /* Einhorn */
|
|
|
|
|
k = (int)(dk*BP_UNICORN);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
{
|
2001-02-09 14:53:53 +01:00
|
|
|
|
int mp = 1;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (get_effect(u, oldpotiontype[P_FAST]) >= u->number)
|
2001-02-09 14:53:53 +01:00
|
|
|
|
mp *= 2; /* Siebenmeilentee */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (u->number <= get_item(u, I_FEENSTIEFEL))
|
2001-02-09 14:53:53 +01:00
|
|
|
|
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! */
|
2001-02-09 14:53:53 +01:00
|
|
|
|
if ( mp == 1 && getplane(next) == astral_plane && is_mage(u)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if(get_mage(u)->magietyp == M_ASTRAL
|
|
|
|
|
|| get_mage(u)->magietyp == M_TRAUM) {
|
2001-02-09 14:53:53 +01:00
|
|
|
|
mp *= 2;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2001-02-09 14:53:53 +01:00
|
|
|
|
k = (int)(dk*mp*BP_WALKING);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch(u->race) {
|
|
|
|
|
case RC_DRAGON:
|
|
|
|
|
case RC_WYRM:
|
|
|
|
|
case RC_FIREDRAGON:
|
|
|
|
|
case RC_BIRTHDAYDRAGON:
|
|
|
|
|
case RC_PSEUDODRAGON:
|
|
|
|
|
k = BP_DRAGON;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m = 0;
|
|
|
|
|
|
|
|
|
|
/* 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) {
|
|
|
|
|
direction_t dir = reldirection(current, next);
|
|
|
|
|
border * b = get_borders(current, next);
|
|
|
|
|
while (b) {
|
|
|
|
|
if (b->type==&bt_wisps) {
|
|
|
|
|
region * rl = rconnect(current, (direction_t)((dir+MAXDIRECTIONS-1)%MAXDIRECTIONS));
|
|
|
|
|
region * rr = rconnect(current, (direction_t)((dir+1)%MAXDIRECTIONS));
|
2001-02-09 14:53:53 +01:00
|
|
|
|
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;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
b = b->next;
|
|
|
|
|
}
|
|
|
|
|
if (current!=next) { /* !pause */
|
|
|
|
|
if (roadto(current, next))
|
|
|
|
|
k -= BP_ROAD;
|
|
|
|
|
else
|
|
|
|
|
k -= BP_NORMAL;
|
|
|
|
|
if (k<0) break;
|
|
|
|
|
/* r2 -> Zielregion, r3 -> Momentane Region */
|
|
|
|
|
if (move_blocked(u, current, reldirection(current, next))
|
|
|
|
|
|| is_spell_active(current, C_FOGTRAP))
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"leavefail%u:unit%r:region", u, next));
|
|
|
|
|
if ((wache = bewegung_blockiert_von(u, current)) != (unit *) NULL
|
|
|
|
|
&& gereist != 0)
|
|
|
|
|
{
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"moveblockedbyguard%u:unit%r:region%f:faction", u, next,
|
|
|
|
|
wache->faction));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (is_spell_active(next, C_ANTIMAGICZONE) && illusionary(u)){
|
|
|
|
|
change_cursevigour(&next->attribs, C_ANTIMAGICZONE, 0, - u->number);
|
|
|
|
|
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"illusionantimagic%u:unit", u));
|
|
|
|
|
destroy_unit(u);
|
|
|
|
|
|
|
|
|
|
return route;
|
|
|
|
|
}
|
|
|
|
|
/* 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(pl && fval(pl, PFL_NOCOORDS)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"detectoceandir%u:unit%i:direction", u,
|
|
|
|
|
reldirection(current, next)));
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"detectocean%u:unit%r:region", u, next));
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (terrain[rterrain(next)].flags & FORBIDDEN_LAND) {
|
|
|
|
|
plane *pl = getplane(next);
|
|
|
|
|
if(pl && fval(pl, PFL_NOCOORDS)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"detectforbiddendir%u:unit%i:direction", u, reldirection(current, next)));
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"detectforbidden%u:unit%r:region", u, next));
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (u->race == RC_INSECT && rterrain(next) == T_GLACIER &&
|
|
|
|
|
!is_cursed(u->attribs, C_KAELTESCHUTZ,0)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"detectforbidden%u:unit%r:region", u, next));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
rv[m] = next;
|
|
|
|
|
route[m] = dir;
|
|
|
|
|
m++;
|
|
|
|
|
}
|
|
|
|
|
else break;
|
|
|
|
|
|
|
|
|
|
current = next;
|
|
|
|
|
if(!flucht) {
|
|
|
|
|
++gereist;
|
|
|
|
|
if (current==first) break; /* PAUSE beendet die Reise */
|
|
|
|
|
next = movewhere(current, u);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
route[m] = NODIRECTION;
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
change_skill(u, SK_RIDING, u->number * PRODUCEEXP);
|
|
|
|
|
} else
|
|
|
|
|
mode = 2;
|
|
|
|
|
|
|
|
|
|
/* Haben Drachen ihr Ziel erreicht? */
|
|
|
|
|
if (dragon(u)) {
|
|
|
|
|
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");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <20>ber die letzte Region braucht man keinen Bericht */
|
|
|
|
|
|
|
|
|
|
m--;
|
|
|
|
|
if (m > 0) {
|
|
|
|
|
dh = 0;
|
|
|
|
|
travelthru(u, first);
|
|
|
|
|
for (i = 0; i != m; i++) {
|
|
|
|
|
if (dh) {
|
|
|
|
|
if (i == m - 1)
|
|
|
|
|
scat(" und ");
|
|
|
|
|
else
|
|
|
|
|
scat(", ");
|
|
|
|
|
}
|
|
|
|
|
dh = 1;
|
|
|
|
|
travelthru(u, rv[i]);
|
|
|
|
|
|
|
|
|
|
sprintf(buf2, trailinto(rv[i], u->faction->locale),
|
|
|
|
|
tregionid(rv[i], u->faction));
|
|
|
|
|
scat(buf2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
add_message(&u->faction->msgs, new_message(
|
|
|
|
|
u->faction, "travel%u:unit%i:mode%r:start%r:end%s:regions",
|
|
|
|
|
u, mode, first, current, strdup(buf)));
|
|
|
|
|
|
|
|
|
|
/* und jetzt noch die transportierten Einheiten verschieben */
|
|
|
|
|
|
|
|
|
|
#ifndef NEW_DRIVE
|
|
|
|
|
if (canride(u)) {
|
|
|
|
|
restkapazitaet = ridingcapacity(u) - weight(u);
|
|
|
|
|
} else {
|
|
|
|
|
restkapazitaet = walkingcapacity(u) - weight(u);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
for (S = u->orders; S; S = S->next) {
|
|
|
|
|
if (igetkeyword(S->s) == K_TRANSPORT) {
|
|
|
|
|
|
|
|
|
|
ut = getunit(first, u);
|
|
|
|
|
|
|
|
|
|
if (ut) {
|
|
|
|
|
if (igetkeyword(ut->thisorder) == K_DRIVE) {
|
|
|
|
|
u2 = getunit(first, ut);
|
|
|
|
|
if (!u2) {
|
|
|
|
|
cmistake(ut, ut->thisorder, 64, MSG_MOVE);
|
|
|
|
|
continue;
|
|
|
|
|
} else {
|
|
|
|
|
#ifndef NEW_DRIVE
|
|
|
|
|
restkapazitaet -= weight(ut);
|
|
|
|
|
if (restkapazitaet >= 0) {
|
|
|
|
|
#endif
|
|
|
|
|
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(ut, ut->thisorder, 67, MSG_MOVE);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if(can_survive(ut, current)) {
|
|
|
|
|
for (i = 0; i != m; i++)
|
|
|
|
|
travelthru(ut, rv[i]);
|
|
|
|
|
move_unit(ut, current, NULL);
|
|
|
|
|
} else {
|
|
|
|
|
cmistake(ut, ut->thisorder, 230, MSG_MOVE);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
#ifndef NEW_DRIVE
|
|
|
|
|
} else {
|
|
|
|
|
sprintf(buf, "Keine Transportkapazit<69>ten"
|
|
|
|
|
" frei f<>r %s und deren Gep<65>ck.", unitname(ut));
|
|
|
|
|
mistake(u, u->thisorder, buf, MSG_MOVE);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
cmistake(u, u->thisorder, 90, MSG_MOVE);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (ut) {
|
|
|
|
|
sprintf(buf,
|
|
|
|
|
"Konnte %s in dieser Region nicht finden.", unitname(ut));
|
|
|
|
|
mistake(u, u->thisorder, buf, 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, "");
|
|
|
|
|
setguard(u, GUARD_NONE);
|
|
|
|
|
|
|
|
|
|
if (fval(u, FL_FOLLOWING)) caught_target(current, u);
|
|
|
|
|
return route;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
ship_ready(region * r, unit * u)
|
|
|
|
|
{
|
|
|
|
|
if (!fval(u, FL_OWNER)) {
|
|
|
|
|
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 *
|
|
|
|
|
gebaeude_vorhanden(const region * r, const building_type * bt)
|
|
|
|
|
{
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Pr<50>ft, ob Ablegen von einer K<>ste in eine der erlaubten Richtungen erfolgt. */
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
check_takeoff(ship *sh, region *from, region *to)
|
|
|
|
|
{
|
|
|
|
|
direction_t dir;
|
|
|
|
|
direction_t coast, coastl, coastr;
|
|
|
|
|
|
|
|
|
|
if (rterrain(from)!=T_OCEAN && sh->coast != NODIRECTION) {
|
|
|
|
|
coast = sh->coast;
|
|
|
|
|
dir = reldirection(from, to);
|
|
|
|
|
|
|
|
|
|
coastr = (direction_t)((coast+1) % MAXDIRECTIONS);
|
|
|
|
|
coastl = (direction_t)((coast+MAXDIRECTIONS-1) % MAXDIRECTIONS);
|
|
|
|
|
|
|
|
|
|
if(dir != coast && dir != coastl && dir != coastr
|
|
|
|
|
&& gebaeude_vorhanden(from, &bt_harbour) == NULL)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean ship_allowed(const struct ship_type * type, region * r)
|
|
|
|
|
{
|
|
|
|
|
int c = 0;
|
|
|
|
|
if (gebaeude_vorhanden(r, &bt_harbour)) return true;
|
|
|
|
|
for (c=0;type->coast[c]!=NOTERRAIN;++c) {
|
|
|
|
|
if (type->coast[c]==rterrain(r)) return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
direction_t *
|
|
|
|
|
sail(region * starting_point, unit * u, region * next_point, boolean move_on_land)
|
|
|
|
|
{
|
|
|
|
|
region *current_point, *last_point;
|
|
|
|
|
unit *u2, *hafenmeister;
|
|
|
|
|
item * trans = NULL;
|
|
|
|
|
int st, first = 1;
|
|
|
|
|
int k, m, l, i, t = 0;
|
|
|
|
|
int stormchance;
|
|
|
|
|
region *rv[MAXSPEED + 1];
|
|
|
|
|
region *tt[MAXSPEED + 1];
|
|
|
|
|
static direction_t route[MAXSPEED+1];
|
|
|
|
|
|
|
|
|
|
if (!ship_ready(starting_point, u))
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
/* Wir suchen so lange nach neuen Richtungen, wie es geht. Diese werden
|
|
|
|
|
* dann nacheinander ausgef<EFBFBD>hrt.
|
|
|
|
|
* Im array rv[] speichern wir die K<EFBFBD>stenregionen ab, durch die wir
|
|
|
|
|
* segeln (geht nur bei Halbinseln). */
|
|
|
|
|
|
|
|
|
|
k = shipspeed(u->ship, u);
|
|
|
|
|
|
|
|
|
|
l = 0;
|
|
|
|
|
m = 0;
|
|
|
|
|
|
|
|
|
|
last_point = starting_point;
|
|
|
|
|
current_point = starting_point;
|
|
|
|
|
|
|
|
|
|
/* 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!). l z<EFBFBD>hlt gesichtete
|
|
|
|
|
* K<EFBFBD>stenstreifen, rv[] speichert die gesichteten K<EFBFBD>stenstreifen, m
|
|
|
|
|
* z<EFBFBD>hlt befahrene Felder, current_point ist die letzte g<EFBFBD>ltige,
|
|
|
|
|
* befahrene Region. */
|
|
|
|
|
tt[t++] = starting_point;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
|
|
|
|
|
direction_t dir = reldirection(current_point, next_point);
|
|
|
|
|
/* Fliegende Schiffe ignorieren Terrain, Nebelwaende, Mahlstr<74>me und
|
|
|
|
|
* Blockaden */
|
|
|
|
|
|
|
|
|
|
if( !is_cursed(u->ship->attribs, C_SHIP_FLYING, 0) ) {
|
|
|
|
|
|
|
|
|
|
if (rterrain(current_point) != T_OCEAN
|
|
|
|
|
&& rterrain(next_point) != T_OCEAN) {
|
|
|
|
|
plane *pl = getplane(next_point);
|
|
|
|
|
if(pl && fval(pl, PFL_NOCOORDS)) {
|
|
|
|
|
sprintf(buf, "Die %s entdeckt, da<64> im %s Festland ist.",
|
|
|
|
|
shipname(u->ship), directions[dir]);
|
|
|
|
|
} else {
|
|
|
|
|
sprintf(buf, "Die %s entdeckt, da<64> (%d,%d) Festland ist.",
|
|
|
|
|
shipname(u->ship), region_x(next_point,u->faction),
|
|
|
|
|
region_y(next_point,u->faction));
|
|
|
|
|
}
|
|
|
|
|
addmessage(0, u->faction, buf, MSG_MOVE, ML_WARN);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (!(u->ship->type->flags & SFL_OPENSEA) && rterrain(next_point) == T_OCEAN) {
|
|
|
|
|
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(rterrain(current_point) != T_OCEAN
|
|
|
|
|
&& rterrain(next_point) == T_OCEAN) {
|
|
|
|
|
if(check_takeoff(u->ship, current_point, next_point) == false) {
|
|
|
|
|
/* Schiff kann nicht ablegen */
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 182, MSG_MOVE);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(move_on_land == false && rterrain(next_point) != T_OCEAN) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Falls Blockade, endet die Seglerei hier */
|
|
|
|
|
|
|
|
|
|
if (move_blocked(u, current_point, reldirection(current_point, next_point))
|
|
|
|
|
|| is_spell_active(current_point, C_FOGTRAP)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"sailfail%h:ship%r:region", u->ship, current_point));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (!ship_allowed(u->ship->type, next_point)) {
|
|
|
|
|
if(fval(u, FL_STORM)) {
|
|
|
|
|
if(!check_leuchtturm(current_point, NULL)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"sailnolandingstorm%h:ship", u->ship));
|
|
|
|
|
damage_ship(u->ship, 0.10);
|
|
|
|
|
if (u->ship->damage>=u->ship->size * DAMAGE_SCALE) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"shipsink%h:ship", u->ship));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"sailnolanding%h:ship%r:region", u->ship, next_point));
|
|
|
|
|
damage_ship(u->ship, 0.10);
|
|
|
|
|
if (u->ship->damage>=u->ship->size * DAMAGE_SCALE) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"shipsink%h:ship", u->ship));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if(terrain[rterrain(next_point)].flags & FORBIDDEN_LAND) {
|
|
|
|
|
plane *pl = getplane(next_point);
|
|
|
|
|
if(pl && fval(pl, PFL_NOCOORDS)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"sailforbiddendir%h:ship%i:direction",
|
|
|
|
|
u->ship, reldirection(current_point,next_point)));
|
|
|
|
|
/* sprintf(buf, "Die Mannschaft der %s weigert sich, in die Feuerwand "
|
|
|
|
|
"im %s zu fahren.",
|
|
|
|
|
|
|
|
|
|
shipname(u->ship), directions[dir]); */
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"sailforbidden%h:ship%r:region", u->ship, next_point));
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(is_cursed(next_point->attribs, C_MAELSTROM, 0)) {
|
|
|
|
|
do_maelstrom(next_point, u);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
stormchance = storms[month(0)] * 5 / shipspeed(u->ship, u);
|
|
|
|
|
if(check_leuchtturm(next_point, NULL)) stormchance /= 3;
|
|
|
|
|
#ifdef SAFE_COASTS
|
|
|
|
|
/* Sturm nur, wenn n<>chste Region Hochsee ist. */
|
|
|
|
|
if(rand()%10000 < stormchance) {
|
|
|
|
|
direction_t d;
|
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
|
region * r = rconnect(next_point, d);
|
|
|
|
|
if (rterrain(r)!=T_OCEAN) break;
|
|
|
|
|
}
|
|
|
|
|
if (d==MAXDIRECTIONS)
|
|
|
|
|
ship_in_storm(u, next_point);
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
/* Sturm nur, wenn n<>chste Region keine Landregion ist. */
|
|
|
|
|
if(rand()%10000 < stormchance && next_point->terrain == T_OCEAN) {
|
|
|
|
|
ship_in_storm(u, next_point);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
} /* endif !flying */
|
|
|
|
|
|
|
|
|
|
/* Falls kein Problem, eines weiter ziehen */
|
|
|
|
|
|
|
|
|
|
last_point = current_point;
|
|
|
|
|
current_point = next_point;
|
|
|
|
|
next_point = movewhere(current_point, u);
|
|
|
|
|
m++;
|
|
|
|
|
u->ship->moved = 1;
|
|
|
|
|
|
|
|
|
|
/* Falls eine K<>stenregion, dann in rv[] aufnehmen (die letzte
|
|
|
|
|
* K<EFBFBD>ste wird nachher nicht aufgelistet werden, wenn das
|
|
|
|
|
* Schiff dort seine Runde beendent!). */
|
|
|
|
|
route[t-1] = dir;
|
|
|
|
|
tt[t++] = current_point; /* travelthrough */
|
|
|
|
|
if (rterrain(current_point) != T_OCEAN) {
|
|
|
|
|
rv[l] = current_point;
|
|
|
|
|
l++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while ((rterrain(current_point) == T_OCEAN
|
|
|
|
|
|| is_cursed(u->ship->attribs, C_SHIP_FLYING, 0))
|
|
|
|
|
&& next_point != 0 && m < k);
|
|
|
|
|
route[t-1] = NODIRECTION;
|
|
|
|
|
|
|
|
|
|
/* nachdem alle Richtungen abgearbeitet wurden, und alle Einheiten
|
|
|
|
|
* transferiert wurden, kann der aktuelle Befehl gel<EFBFBD>scht werden. */
|
|
|
|
|
cycle_route(u, t-1);
|
|
|
|
|
set_string(&u->thisorder, "");
|
|
|
|
|
|
|
|
|
|
/* 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 (u->ship->moved) {
|
|
|
|
|
if (current_point->terrain != T_OCEAN && !is_cursed(u->ship->attribs, C_SHIP_FLYING, 0)) {
|
|
|
|
|
u->ship->coast = reldirection(current_point, last_point);
|
|
|
|
|
} else {
|
|
|
|
|
u->ship->coast = NODIRECTION;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (starting_point != current_point) {
|
|
|
|
|
ship * sh = u->ship;
|
|
|
|
|
sh->moved = 1;
|
|
|
|
|
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));
|
|
|
|
|
|
|
|
|
|
/* Falls es Kuesten gibt, denen man entlang gefahren ist,
|
|
|
|
|
* sollen sie aufgefuehrt werden. Falls wir aber unseren Zug in
|
|
|
|
|
* einer Kuestenregion beendet haben, fuehren wir sie nicht mit
|
|
|
|
|
* auf - sie wird sowieso in allen Details im Report erwaehnt
|
|
|
|
|
* werden. */
|
|
|
|
|
|
|
|
|
|
if (rterrain(current_point) != T_OCEAN)
|
|
|
|
|
l--;
|
|
|
|
|
|
|
|
|
|
if (l > 0) {
|
|
|
|
|
if( is_cursed(sh->attribs, C_SHIP_FLYING, 0) )
|
|
|
|
|
scat(". Dabei flog es <20>ber ");
|
|
|
|
|
else {
|
|
|
|
|
scat(". Dabei segelte es entlang der ");
|
|
|
|
|
if (l > 1)
|
|
|
|
|
scat("K<EFBFBD>sten");
|
|
|
|
|
else
|
|
|
|
|
scat("K<EFBFBD>ste");
|
|
|
|
|
scat(" von ");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i != l; i++) {
|
|
|
|
|
scat(regionid(rv[i]));
|
|
|
|
|
if (i < l - 2)
|
|
|
|
|
scat(", ");
|
|
|
|
|
else if (i == l - 2)
|
|
|
|
|
scat(" und ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
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 */
|
|
|
|
|
|
|
|
|
|
tt[t - 1] = 0;
|
|
|
|
|
sh = move_ship(sh, starting_point, current_point, tt);
|
|
|
|
|
/* Verfolgungen melden */
|
|
|
|
|
if (fval(u, FL_FOLLOWING)) caught_target(current_point, u);
|
|
|
|
|
|
|
|
|
|
/* Hafengeb<65>hren ? */
|
|
|
|
|
|
|
|
|
|
hafenmeister = gebaeude_vorhanden(current_point, &bt_harbour);
|
|
|
|
|
if (sh && hafenmeister != NULL) {
|
|
|
|
|
item * itm;
|
|
|
|
|
assert(trans==NULL);
|
|
|
|
|
for (u2 = current_point->units; u2; u2 = u2->next) {
|
|
|
|
|
if (u2->ship == u->ship &&
|
|
|
|
|
!allied(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) {
|
|
|
|
|
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));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (trans) {
|
|
|
|
|
sprintf(buf, "%s erhielt ", hafenmeister->name);
|
|
|
|
|
for (itm = trans; itm; itm=itm->next) {
|
|
|
|
|
if (first != 1) {
|
|
|
|
|
if (itm->next!=NULL && itm->next->next==NULL) {
|
|
|
|
|
scat(" und ");
|
|
|
|
|
} else {
|
|
|
|
|
scat(", ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
first = 0;
|
|
|
|
|
icat(trans->number);
|
|
|
|
|
scat(" ");
|
|
|
|
|
if (itm->number == 1) {
|
|
|
|
|
scat(locale_string(NULL, resourcename(itm->type->rtype, 0)));
|
|
|
|
|
} else {
|
|
|
|
|
scat(locale_string(NULL, 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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Segeln, Wandern, Reiten */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
move(region * r, unit * u, boolean move_on_land)
|
|
|
|
|
{
|
|
|
|
|
region *r2;
|
|
|
|
|
direction_t * route;
|
|
|
|
|
|
|
|
|
|
r2 = movewhere(r, u);
|
|
|
|
|
|
|
|
|
|
if (!r2) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 71, MSG_MOVE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
else if (u->ship && fval(u, FL_OWNER))
|
|
|
|
|
route = sail(r, u, r2, move_on_land);
|
|
|
|
|
else
|
|
|
|
|
route = travel(r, u, r2, 0);
|
|
|
|
|
|
|
|
|
|
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.",
|
|
|
|
|
directions[back[d]]);
|
|
|
|
|
addmessage(rc, NULL, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
set_string(&u->thisorder, "");
|
|
|
|
|
/* F<>r den Silberpool: Bewegung ist fertig, Silber darf abgezogen werden */
|
|
|
|
|
if (fval(u, FL_FOLLOWED) && route && route[0]!=NODIRECTION) {
|
|
|
|
|
unit ** up=&r->units;
|
|
|
|
|
|
|
|
|
|
while (*up) {
|
|
|
|
|
unit * uf = *up;
|
|
|
|
|
if (fval(uf, FL_FOLLOWING) && !fval(uf, FL_MOVED)) {
|
|
|
|
|
attrib * a = a_find(uf->attribs, &at_follow);
|
|
|
|
|
if (a && a->data.v==u) {
|
|
|
|
|
/* wir basteln ihm ein NACH */
|
|
|
|
|
int k, i = 0;
|
|
|
|
|
strcpy(buf, keywords[K_MOVE]);
|
|
|
|
|
while (route[i]!=NODIRECTION)
|
|
|
|
|
strcat(strcat(buf, " "), directions[route[i++]]);
|
|
|
|
|
set_string(&uf->thisorder, buf);
|
|
|
|
|
k = igetkeyword(uf->thisorder);
|
|
|
|
|
assert(k==K_MOVE);
|
|
|
|
|
move(r, uf, true);
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-02-04 10:46:48 +01:00
|
|
|
|
if (*up==uf) up = &uf->next;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrib_type at_piracy_direction = {
|
|
|
|
|
"piracy_direction",
|
|
|
|
|
DEFAULT_INIT,
|
|
|
|
|
DEFAULT_FINALIZE,
|
|
|
|
|
DEFAULT_AGE,
|
|
|
|
|
NO_WRITE,
|
|
|
|
|
NO_READ
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
piracy(unit *u)
|
|
|
|
|
{
|
|
|
|
|
region *r = u->region;
|
|
|
|
|
ship *sh = u->ship, *sh2;
|
|
|
|
|
unit *cap;
|
|
|
|
|
direction_t dir, target_dir = NODIRECTION;
|
|
|
|
|
int aff[MAXDIRECTIONS];
|
|
|
|
|
int saff = 0;
|
|
|
|
|
int *il;
|
|
|
|
|
char *s;
|
|
|
|
|
boolean all = true;
|
|
|
|
|
attrib *a;
|
|
|
|
|
|
|
|
|
|
if(!sh) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 144, MSG_MOVE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(!fval(u, FL_OWNER)) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 46, MSG_MOVE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 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) {
|
|
|
|
|
faction *f = findfaction(a->data.sa[0]);
|
|
|
|
|
|
|
|
|
|
if(allied(u, f, HELP_FIGHT)
|
|
|
|
|
&& intlist_find(il, a->data.sa[1])) {
|
|
|
|
|
target_dir = (direction_t)a->data.sa[1];
|
|
|
|
|
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) {
|
|
|
|
|
cap = shipowner(rc, sh2);
|
|
|
|
|
if(cap && !allied(u, cap->faction, HELP_FIGHT)
|
|
|
|
|
&& (intlist_find(il, cap->faction->no) || all)) {
|
|
|
|
|
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;
|
|
|
|
|
a = a_add(&r->attribs, a_new(&at_piracy_direction));
|
|
|
|
|
a->data.sa[0] = (short)u->faction->no;
|
|
|
|
|
a->data.sa[1] = (short)target_dir;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(il);
|
|
|
|
|
|
|
|
|
|
/* Wenn kein Ziel gefunden, entsprechende Meldung generieren */
|
|
|
|
|
if(target_dir == NODIRECTION) {
|
|
|
|
|
add_message(&u->faction->msgs,
|
|
|
|
|
new_message(u->faction, "piratenovictim%h:ship%r:region", sh, r));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 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", keywords[K_MOVE], directions[target_dir]);
|
|
|
|
|
set_string(&u->thisorder, buf);
|
|
|
|
|
|
|
|
|
|
/* Bewegung ausf<73>hren */
|
|
|
|
|
igetkeyword(u->thisorder); /* NACH ignorieren */
|
|
|
|
|
move(r, u, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
|
|
while(a) {
|
|
|
|
|
traveldir *t = (traveldir *)(a->data.v);
|
|
|
|
|
if(t->no == id) return t->dir;
|
|
|
|
|
a = a->nexttype;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NODIRECTION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
hunt(unit *u)
|
|
|
|
|
{
|
|
|
|
|
region *rc = u->region;
|
|
|
|
|
int moves, id;
|
|
|
|
|
char command[256];
|
|
|
|
|
direction_t dir;
|
|
|
|
|
|
|
|
|
|
if(!u->ship) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 144, MSG_MOVE);
|
|
|
|
|
return false;
|
|
|
|
|
} else if(!fval(u, FL_OWNER)) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 146, MSG_MOVE);
|
|
|
|
|
return false;
|
|
|
|
|
} else if(attacked(u)) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 52, MSG_MOVE);
|
|
|
|
|
return false;
|
2001-02-03 14:45:35 +01:00
|
|
|
|
} else if(race[u->race].flags & RCF_CANNOTMOVE) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 55, MSG_MOVE);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
id = getshipid();
|
|
|
|
|
|
|
|
|
|
if(id <= 0) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 20, MSG_MOVE);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dir = hunted_dir(rc->attribs, id);
|
|
|
|
|
|
|
|
|
|
if(dir == NODIRECTION) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 20, MSG_MOVE);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sprintf(command, "%s %s", keywords[K_MOVE], 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, directions[dir]);
|
|
|
|
|
moves++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* In command steht jetzt das NACH-Kommando. */
|
|
|
|
|
|
|
|
|
|
igetkeyword(command); /* NACH ignorieren und Parsing initialisieren. */
|
|
|
|
|
move(u->region, u, false); /* NACH ausf<73>hren */
|
|
|
|
|
return true; /* true -> Einheitenliste von vorne durchgehen */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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, r);
|
|
|
|
|
}
|
|
|
|
|
sh = shn;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
is_disorientated(unit *u)
|
|
|
|
|
{
|
|
|
|
|
if(u->ship && get_curse(u->ship->attribs, C_DISORIENTATION, 0))
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
if(is_spell_active(u->region, C_REGCONF))
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef TODO /* Wenn Feature ausgearbeitet */
|
|
|
|
|
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 */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
movement(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
|
|
|
|
|
/* Initialize the additional encumbrance by transported units */
|
|
|
|
|
#ifdef NEW_DRIVE
|
|
|
|
|
init_drive();
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
2001-02-14 08:44:57 +01:00
|
|
|
|
unit ** up = &r->units;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* Bewegungen.
|
|
|
|
|
*
|
|
|
|
|
* Zuerst m<EFBFBD>ssen sich alle Einheiten ohne u->target bewegen
|
|
|
|
|
* (NACH), dann starten die Verfolger mit u->target (FOLGE) und
|
|
|
|
|
* h<EFBFBD>ren erst auf, wenn sie ihr Ziel erreicht haben.
|
|
|
|
|
*
|
|
|
|
|
* neue NEW_FOLLOW-Variante: Verfolger folgen sofort, FL_FOLLOW
|
|
|
|
|
* ist bereits gesetzt.
|
|
|
|
|
*/
|
2001-02-14 08:44:57 +01:00
|
|
|
|
while (*up) {
|
|
|
|
|
unit *u = *up;
|
|
|
|
|
keyword_t kword;
|
|
|
|
|
|
|
|
|
|
if (fval(u, FL_FOLLOWING)) {
|
|
|
|
|
/* skip all followers */
|
|
|
|
|
do {
|
|
|
|
|
u = u->next;
|
|
|
|
|
} while (u && fval(u, FL_FOLLOWING));
|
|
|
|
|
up = &u;
|
|
|
|
|
if (u==NULL) break;
|
|
|
|
|
}
|
|
|
|
|
kword = igetkeyword(u->thisorder);
|
|
|
|
|
switch (kword) {
|
|
|
|
|
case K_ROUTE:
|
|
|
|
|
case K_MOVE:
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (attacked(u)) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 52, MSG_MOVE);
|
|
|
|
|
set_string(&u->thisorder, "");
|
2001-02-14 08:44:57 +01:00
|
|
|
|
up = &u->next;
|
2001-02-03 14:45:35 +01:00
|
|
|
|
} else if (race[u->race].flags & RCF_CANNOTMOVE) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 55, MSG_MOVE);
|
|
|
|
|
set_string(&u->thisorder, "");
|
2001-02-14 08:44:57 +01:00
|
|
|
|
up = &u->next;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
} else {
|
|
|
|
|
move(r, u, true);
|
|
|
|
|
set_string(&u->thisorder, "");
|
2001-02-14 08:44:57 +01:00
|
|
|
|
if (u==*up) up = &u->next;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2001-02-14 08:44:57 +01:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
up = &u->next;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Dann generieren die jagenden Einheiten ihre Befehle und
|
|
|
|
|
* bewegen sich. */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
2001-02-14 08:44:57 +01:00
|
|
|
|
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) == K_FOLLOW) {
|
|
|
|
|
if(attacked(u)) {
|
|
|
|
|
cmistake(u, o->s, 52, MSG_MOVE);
|
|
|
|
|
u = u2; break;
|
2001-02-03 14:45:35 +01:00
|
|
|
|
} else if(race[u->race].flags & RCF_CANNOTMOVE) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cmistake(u, o->s, 55, MSG_MOVE);
|
|
|
|
|
u = u2; break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
p = getparam();
|
|
|
|
|
if(p != P_SHIP) {
|
|
|
|
|
if(p != P_UNIT) {
|
|
|
|
|
cmistake(u, o->s, 240, MSG_MOVE);
|
|
|
|
|
}
|
|
|
|
|
u = u2; break;;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(hunt(u)) {
|
|
|
|
|
set_string(&u->thisorder, "");
|
|
|
|
|
u = r->units;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
u = u2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Regul<75>re Verfolger starten */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
2001-02-14 08:44:57 +01:00
|
|
|
|
unit * u;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for (u = r->units; u;) {
|
|
|
|
|
unit * u2 = u->next;
|
|
|
|
|
if (utarget(u)
|
|
|
|
|
&& (igetkeyword(u->thisorder) == K_MOVE
|
|
|
|
|
|| igetkeyword(u->thisorder) == K_ROUTE)) {
|
|
|
|
|
if (attacked(u)) {
|
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 52, MSG_PRODUCE);
|
2001-02-03 14:45:35 +01:00
|
|
|
|
} else if (race[u->race].flags & RCF_CANNOTMOVE) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cmistake(u, findorder(u, u->thisorder), 55, MSG_PRODUCE);
|
|
|
|
|
} else
|
|
|
|
|
move(r, u, true);
|
|
|
|
|
}
|
|
|
|
|
u = u2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Abtreiben von besch<63>digten, unterbemannten, <20>berladenen Schiffen */
|
|
|
|
|
|
|
|
|
|
drifting_ships(r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Piraten und Cleanup */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
2001-02-14 08:44:57 +01:00
|
|
|
|
unit * u;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for(u = r->units; u;) {
|
|
|
|
|
unit *un = u->next;
|
|
|
|
|
if(igetkeyword(u->thisorder) == K_PIRACY) piracy(u);
|
|
|
|
|
u = un;
|
|
|
|
|
}
|
|
|
|
|
a_removeall(&r->attribs, &at_piracy_direction);
|
|
|
|
|
age_traveldir(r);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
follow(void)
|
|
|
|
|
{
|
|
|
|
|
region * r;
|
|
|
|
|
for (r=regions;r;r=r->next) {
|
|
|
|
|
unit * u;
|
|
|
|
|
for (u=r->units;u;u=u->next) {
|
2001-02-02 09:40:49 +01:00
|
|
|
|
attrib * a;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
strlist * o;
|
2001-02-02 09:40:49 +01:00
|
|
|
|
if (fval(u, FL_HADBATTLE)) continue;
|
|
|
|
|
a = a_find(u->attribs, &at_follow);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for (o=u->orders;o;o=o->next) {
|
|
|
|
|
if (igetkeyword(o->s) == K_FOLLOW
|
|
|
|
|
&& getparam() == P_UNIT) {
|
|
|
|
|
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));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
} else if (a) {
|
|
|
|
|
a_remove(&u->attribs, a);
|
|
|
|
|
a = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (a && !fval(u, FL_MOVED)) {
|
|
|
|
|
unit * u2 = a->data.v;
|
|
|
|
|
if (!u2 || u2->region!=r || !cansee(u->faction, r, u2, 0))
|
|
|
|
|
continue;
|
|
|
|
|
for (o=u2->orders;o;o=o->next) {
|
|
|
|
|
switch (igetkeyword(o->s)) {
|
|
|
|
|
case K_MOVE:
|
|
|
|
|
case K_ROUTE:
|
|
|
|
|
case K_PIRACY:
|
|
|
|
|
fset(u, FL_FOLLOWING);
|
|
|
|
|
fset(u2, FL_FOLLOWED);
|
|
|
|
|
set_string(&u->thisorder, "");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|