2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2001-04-14 14:11:45 +02:00
|
|
|
|
*
|
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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include "eressea.h"
|
|
|
|
|
#include "randenc.h"
|
|
|
|
|
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
#include "faction.h"
|
|
|
|
|
#include "alchemy.h"
|
|
|
|
|
#include "item.h"
|
|
|
|
|
#include "plane.h"
|
|
|
|
|
#include "economy.h"
|
|
|
|
|
#include "building.h"
|
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "message.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "monster.h"
|
|
|
|
|
#include "creation.h"
|
|
|
|
|
#include "names.h"
|
|
|
|
|
#include "pool.h"
|
|
|
|
|
#include "movement.h"
|
|
|
|
|
#include "curse.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "karma.h"
|
|
|
|
|
#include "ship.h"
|
|
|
|
|
#include "battle.h"
|
|
|
|
|
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
/* attributes includes */
|
|
|
|
|
#include <attributes/racename.h>
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* util includes */
|
|
|
|
|
#include <rand.h>
|
|
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
#include <attributes/iceberg.h>
|
|
|
|
|
extern attrib_type at_unitdissolve;
|
2001-01-27 20:30:07 +01:00
|
|
|
|
extern attrib_type at_orcification;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
nrand(int start, int sub)
|
|
|
|
|
{
|
|
|
|
|
int res = 0;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
if (rand() % 100 < start)
|
|
|
|
|
res++;
|
|
|
|
|
start -= sub;
|
|
|
|
|
} while (start > 0);
|
|
|
|
|
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* In a->data.ca[1] steht der Prozentsatz mit dem sich die Einheit
|
|
|
|
|
* aufl<EFBFBD>st, in a->data.ca[0] kann angegeben werden, wohin die Personen
|
|
|
|
|
* verschwinden. Passiert bereits in der ersten Runde! */
|
|
|
|
|
static void
|
|
|
|
|
dissolve_units(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u;
|
|
|
|
|
int n;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (r=regions;r;r=r->next) {
|
|
|
|
|
for (u=r->units;u;u=u->next) {
|
|
|
|
|
attrib * a = a_find(u->attribs, &at_unitdissolve);
|
|
|
|
|
if (a) {
|
|
|
|
|
if (u->age == 0 && a->data.ca[1] < 100) continue;
|
|
|
|
|
|
|
|
|
|
/* TODO: Durch einzelne Berechnung ersetzen */
|
|
|
|
|
if (a->data.ca[1] == 100) {
|
|
|
|
|
n = u->number;
|
|
|
|
|
} else {
|
|
|
|
|
n = 0;
|
|
|
|
|
for (i=0;i<u->number;i++) {
|
|
|
|
|
if (rand()%100 < a->data.ca[1]) n++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* wenn keiner verschwindet, auch keine Meldung */
|
|
|
|
|
if (n == 0) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
scale_number(u,u->number - n);
|
|
|
|
|
|
|
|
|
|
sprintf(buf, "%s in %s: %d %s ", unitname(u), regionid(r),
|
|
|
|
|
n, race[u->race].name[ n == 1 ? 0 : 1]);
|
|
|
|
|
switch(a->data.ca[0]) {
|
|
|
|
|
case 1:
|
|
|
|
|
rsetpeasants(r, rpeasants(r) + n);
|
|
|
|
|
if (n == 1) {
|
|
|
|
|
scat("kehrte auf sein Feld zur<75>ck.");
|
|
|
|
|
}else{
|
|
|
|
|
scat("kehrten auf ihre Felder zur<75>ck.");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
2001-04-01 08:58:45 +02:00
|
|
|
|
if(r->land) {
|
|
|
|
|
rsettrees(r, rtrees(r) + n);
|
|
|
|
|
if (n == 1) {
|
|
|
|
|
scat("wurde zum Baum.");
|
|
|
|
|
}else{
|
|
|
|
|
scat("wurden zu B<>umen.");
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if(n == 1) {
|
|
|
|
|
scat("verfaulte in der feuchten Seeluft.");
|
|
|
|
|
} else {
|
|
|
|
|
scat("verfaulten in der feuchten Seeluft.");
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
if (u->race == RC_STONEGOLEM || u->race == RC_IRONGOLEM) {
|
|
|
|
|
if (n == 1) {
|
|
|
|
|
scat("zerfiel zu Staub.");
|
|
|
|
|
}else{
|
|
|
|
|
scat("zerfielen zu Staub.");
|
|
|
|
|
}
|
|
|
|
|
}else{
|
|
|
|
|
if (n == 1) {
|
|
|
|
|
scat("verschwand <20>ber Nacht.");
|
|
|
|
|
}else{
|
|
|
|
|
scat("verschwanden <20>ber Nacht.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
addmessage(r, u->faction, buf, MSG_EVENT, ML_INFO);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
remove_empty_units();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
improve_all(faction * f, skill_t s, int g)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u;
|
|
|
|
|
int n = 0;
|
|
|
|
|
region *last = lastregion(f);
|
|
|
|
|
|
|
|
|
|
for (r = firstregion(f); r != last; r = r->next) {
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->faction == f && get_skill(u, s) > 0) {
|
|
|
|
|
n += u->number;
|
|
|
|
|
change_skill(u, s, u->number * g);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
find_manual(region * r, unit * u)
|
|
|
|
|
{
|
|
|
|
|
skill_t skill = NOSKILL;
|
|
|
|
|
sprintf(buf, "%s stolper%c bei der Erforschung der Region <20>ber ",
|
|
|
|
|
unitname(u), "nt"[u->number == 1]);
|
|
|
|
|
|
|
|
|
|
switch (rand() % 4) {
|
|
|
|
|
case 0:
|
|
|
|
|
scat("die Ruine eines alten Tempels");
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
scat("eine alte Burgruine");
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
scat("ein zerfallenes Bauernhaus");
|
|
|
|
|
break;
|
|
|
|
|
case 3:
|
|
|
|
|
scat("eine Leiche am Wegesrand");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
scat(". Bei der Durchsuchung ");
|
|
|
|
|
if (u->number == 1) {
|
|
|
|
|
scat("st<EFBFBD><EFBFBD>t");
|
|
|
|
|
} else {
|
|
|
|
|
scat("sto<EFBFBD>en");
|
|
|
|
|
}
|
|
|
|
|
scat(" sie auf das zerfledderte Exemplar eines alten Buches, betitelt ");
|
|
|
|
|
|
|
|
|
|
switch (rand() % 36) {
|
|
|
|
|
case 0:
|
|
|
|
|
scat("\'Magie der Elemente\'");
|
|
|
|
|
skill = SK_MAGIC;
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
case 2:
|
|
|
|
|
case 3:
|
|
|
|
|
case 4:
|
|
|
|
|
scat("\'Schwerter, Armbr<62>ste, Langb<67>gen\'");
|
|
|
|
|
skill = SK_WEAPONSMITH;
|
|
|
|
|
break;
|
|
|
|
|
case 5:
|
|
|
|
|
case 6:
|
|
|
|
|
scat("\'Gorms Almanach der Rationellen Kriegsf<73>hrung\'");
|
|
|
|
|
skill = SK_TACTICS;
|
|
|
|
|
break;
|
|
|
|
|
case 7:
|
|
|
|
|
case 8:
|
|
|
|
|
case 9:
|
|
|
|
|
case 10:
|
|
|
|
|
scat("\'Katamarane, Koggen, Karavellen\'");
|
|
|
|
|
skill = SK_SHIPBUILDING;
|
|
|
|
|
break;
|
|
|
|
|
case 11:
|
|
|
|
|
case 12:
|
|
|
|
|
case 13:
|
|
|
|
|
case 14:
|
|
|
|
|
scat("\'Wege der Sterne\'");
|
|
|
|
|
skill = SK_SAILING;
|
|
|
|
|
break;
|
|
|
|
|
case 15:
|
|
|
|
|
case 16:
|
|
|
|
|
case 17:
|
|
|
|
|
scat("\'Nadishahs Kleine Gift- und Kr<4B>uterkunde\'");
|
|
|
|
|
skill = SK_HERBALISM;
|
|
|
|
|
break;
|
|
|
|
|
case 18:
|
|
|
|
|
case 19:
|
|
|
|
|
scat("\'Mandricks Kompendium der Alchemie\'");
|
|
|
|
|
skill = SK_ALCHEMY;
|
|
|
|
|
break;
|
|
|
|
|
case 20:
|
|
|
|
|
case 21:
|
|
|
|
|
case 22:
|
|
|
|
|
case 23:
|
|
|
|
|
scat("\'Die Konstruktion der Burgen und Schl<68>sser von Zentralandune\'");
|
|
|
|
|
skill = SK_BUILDING;
|
|
|
|
|
break;
|
|
|
|
|
case 24:
|
|
|
|
|
case 25:
|
|
|
|
|
case 26:
|
|
|
|
|
case 27:
|
|
|
|
|
scat("\'Die Esse\'");
|
|
|
|
|
skill = SK_ARMORER;
|
|
|
|
|
break;
|
|
|
|
|
case 28:
|
|
|
|
|
case 29:
|
|
|
|
|
case 30:
|
|
|
|
|
case 31:
|
|
|
|
|
scat("\'<EFBFBD>ber die Gewinnung von Erzen\'");
|
|
|
|
|
skill = SK_MINING;
|
|
|
|
|
break;
|
|
|
|
|
case 32:
|
|
|
|
|
case 33:
|
|
|
|
|
case 34:
|
|
|
|
|
case 35:
|
|
|
|
|
scat("\'Barinions Lieder, eine Einf<6E>hrung f<>r Unbedarfte\'");
|
|
|
|
|
skill = SK_ENTERTAINMENT;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
scat(". Der Wissensschub ist enorm.");
|
|
|
|
|
addmessage(r, u->faction, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
|
|
|
|
|
if (improve_all(u->faction, skill, 90) == 0) {
|
|
|
|
|
change_skill(u, skill, 270);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
get_unit(region * r, unit * u)
|
|
|
|
|
{
|
|
|
|
|
unit *newunit;
|
|
|
|
|
sprintf(buf, "%s entdeck%s ein kleines Dorf. Die meisten H<>user "
|
|
|
|
|
"wurden durch einen <20>ber die Ufer getretenen Flu<6C> zerst<73>rt. Eine "
|
|
|
|
|
"Gruppe der verzweifelten Menschen schlie<69>t sich deiner Partei an.",
|
|
|
|
|
unitname(u), "en\0t" + (3 - 3 * (u->number == 1)));
|
|
|
|
|
|
|
|
|
|
addmessage(r, u->faction, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 20 + 3, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Dorfbewohner");
|
|
|
|
|
set_money(newunit, (rand() % 26 + 10) * newunit->number);
|
|
|
|
|
fset(newunit, FL_ISNEW);
|
|
|
|
|
if (fval(u, FL_PARTEITARNUNG)) fset(newunit, FL_PARTEITARNUNG);
|
|
|
|
|
switch (rand() % 4) {
|
|
|
|
|
case 0:
|
|
|
|
|
set_skill(newunit, SK_MINING, 30 * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
set_skill(newunit, SK_LUMBERJACK, 30 * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
set_skill(newunit, SK_CARTMAKER, 30 * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
case 3:
|
|
|
|
|
set_skill(newunit, SK_QUARRYING, 30 * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
set_item(newunit, I_WAGON, rand() % 2);
|
|
|
|
|
set_item(newunit, I_HORSE, min(get_item(newunit, I_WAGON) * 2, rand() % 5));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
get_allies(region * r, unit * u)
|
|
|
|
|
{
|
|
|
|
|
unit *newunit = NULL;
|
|
|
|
|
|
|
|
|
|
switch (rterrain(r)) {
|
|
|
|
|
|
|
|
|
|
case T_PLAIN:
|
|
|
|
|
if (!r_isforest(r)) {
|
|
|
|
|
if (get_money(u) / u->number < 100 + rand() % 200)
|
|
|
|
|
return;
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 8 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "S<EFBFBD>ldner");
|
|
|
|
|
set_money(newunit, (rand() % 80 + 20) * newunit->number);
|
|
|
|
|
|
|
|
|
|
switch (rand() % 4) {
|
|
|
|
|
case 0:
|
|
|
|
|
set_skill(newunit, SK_SWORD, (30 + rand() % 120) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SWORD, newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
set_skill(newunit, SK_SPEAR, (30 + rand() % 120) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SPEAR, newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
set_skill(newunit, SK_CROSSBOW, (30 + rand() % 120) * newunit->number);
|
|
|
|
|
set_item(newunit, I_CROSSBOW, newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
case 3:
|
|
|
|
|
set_skill(newunit, SK_LONGBOW, (30 + rand() % 120) * newunit->number);
|
|
|
|
|
set_item(newunit, I_LONGBOW, newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (rand() % 100 < 40) {
|
|
|
|
|
set_item(newunit, I_CHAIN_MAIL, rand() % (newunit->number + 1));
|
|
|
|
|
}
|
|
|
|
|
if (rand() % 100 < 30) {
|
|
|
|
|
set_item(newunit, I_HORSE, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_RIDING, (30 + rand() % 120) * newunit->number);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
if (eff_skill(u, SK_LONGBOW, r) < 3
|
|
|
|
|
&& eff_skill(u, SK_HERBALISM, r) < 2
|
|
|
|
|
&& eff_skill(u, SK_MAGIC, r) < 2) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 6 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Waldbewohner");
|
|
|
|
|
set_money(newunit, (rand() % 20 + 10) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_LONGBOW, (90 + rand() % 210) * newunit->number);
|
|
|
|
|
set_item(newunit, I_LONGBOW, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_OBSERVATION, (90 + rand() % 110) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_STEALTH, (30 + rand() % 80) * newunit->number);
|
|
|
|
|
if (rand() % 100 < 20) {
|
|
|
|
|
set_skill(newunit, SK_HERBALISM, (30 + rand() % 10) * newunit->number);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_SWAMP:
|
|
|
|
|
if (eff_skill(u, SK_OBSERVATION, r) <= 3) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 6 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Sumpfbewohner");
|
|
|
|
|
set_money(newunit, (rand() % 20 + 10) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_SPEAR, (90 + rand() % 210) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SPEAR, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_STEALTH, (90 + rand() % 120) * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_DESERT:
|
|
|
|
|
if (eff_skill(u, SK_RIDING, r) <= 2) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 12 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Berber");
|
|
|
|
|
set_money(newunit, (rand() % 30 + 20) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_SWORD, (30 + rand() % 60) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SWORD, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_TRADE, (30 + rand() % 60) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_RIDING, (90 + rand() % 60) * newunit->number);
|
|
|
|
|
set_item(newunit, I_HORSE, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_HORSE_TRAINING, (90 + rand() % 120) * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_HIGHLAND:
|
|
|
|
|
if (eff_skill(u, SK_SWORD, r) <= 1) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 8 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Hochlandbarbaren");
|
|
|
|
|
set_money(newunit, (rand() % 10 + 20) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_SWORD, (60 + rand() % 60) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SWORD, newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_MOUNTAIN:
|
|
|
|
|
if (eff_skill(u, SK_SWORD, r) <= 1
|
|
|
|
|
|| eff_skill(u, SK_TRADE, r) <= 2) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 6 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Bergbewohner");
|
|
|
|
|
set_money(newunit, (rand() % 40 + 60) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_SWORD, (90 + rand() % 60) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SWORD, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_ARMORER, (90 + rand() % 60) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_TRADE, (30 + rand() % 120) * newunit->number);
|
|
|
|
|
if (rand() % 100 < 60) {
|
|
|
|
|
set_item(newunit, I_PLATE_ARMOR, newunit->number);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case T_GLACIER:
|
|
|
|
|
if (eff_skill(u, SK_SWORD, r) <= 1
|
|
|
|
|
|| eff_skill(u, SK_TRADE, r) <= 1) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
newunit = createunit(r, u->faction, rand() % 4 + 2, u->faction->race);
|
|
|
|
|
set_string(&newunit->name, "Eisleute");
|
|
|
|
|
set_money(newunit, (rand() % 20 + 20) * newunit->number);
|
|
|
|
|
set_skill(newunit, SK_SWORD, (90 + rand() % 60) * newunit->number);
|
|
|
|
|
set_item(newunit, I_SWORD, newunit->number);
|
|
|
|
|
set_skill(newunit, SK_ARMORER, (90 + rand() % 60) * newunit->number);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2001-02-10 15:07:29 +01:00
|
|
|
|
u_setfaction(newunit, u->faction);
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
set_racename(&newunit->attribs, get_racename(u->attribs));
|
|
|
|
|
if(race[u->race].flags & RCF_SHAPESHIFT) {
|
|
|
|
|
newunit->irace = u->irace;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (fval(u, FL_PARTEITARNUNG)) fset(newunit, FL_PARTEITARNUNG);
|
|
|
|
|
fset(u, FL_ISNEW);
|
|
|
|
|
|
|
|
|
|
sprintf(buf, "Pl<EFBFBD>tzlich stolper%c %s <20>ber einige %s. Nach kurzem "
|
|
|
|
|
"Z<EFBFBD>gern entschlie<69>en sich die %s, sich Deiner Partei anzuschlie<69>en.",
|
|
|
|
|
u->number == 1 ? 't' : 'n', unitname(u), newunit->name, newunit->name);
|
|
|
|
|
|
|
|
|
|
addmessage(r, u->faction, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
encounter(region * r, unit * u)
|
|
|
|
|
{
|
|
|
|
|
if (!fval(r, RF_ENCOUNTER)) return;
|
|
|
|
|
freset(r, RF_ENCOUNTER);
|
|
|
|
|
if (rand() % 100>=ENCCHANCE) return;
|
|
|
|
|
switch (rand() % 3) {
|
|
|
|
|
case 0:
|
|
|
|
|
find_manual(r, u);
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
get_unit(r, u);
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
get_allies(r, u);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
encounters(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u;
|
|
|
|
|
int n;
|
|
|
|
|
int c;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
if (rterrain(r) != T_OCEAN && fval(r, RF_ENCOUNTER)) {
|
|
|
|
|
c = 0;
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
c += u->number;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c > 0) {
|
|
|
|
|
n = rand() % c;
|
|
|
|
|
u = r->units;
|
|
|
|
|
|
|
|
|
|
for (i = u->number; i < n; i += u->number) {
|
|
|
|
|
u = u->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
encounter(r, u);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
chaos(region * r)
|
|
|
|
|
{
|
|
|
|
|
unit *u = NULL, *u2;
|
|
|
|
|
building *b, *b2;
|
|
|
|
|
|
|
|
|
|
if (rand() % 100 < 8) {
|
|
|
|
|
switch (rand() % 3) {
|
|
|
|
|
case 0: /* Untote */
|
|
|
|
|
if (rterrain(r) != T_OCEAN) {
|
|
|
|
|
u = random_unit(r);
|
|
|
|
|
if (u && !nonplayer(u)) {
|
|
|
|
|
sprintf(buf, "%s scheint von einer seltsamen Krankheit befallen.",
|
|
|
|
|
unitname(u));
|
|
|
|
|
addmessage(0, u->faction, buf, MSG_EVENT, ML_IMPORTANT);
|
2001-02-10 15:18:01 +01:00
|
|
|
|
u_setfaction(u, findfaction(MONSTER_FACTION));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
u->race = RC_GHOUL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 1: /* Drachen */
|
|
|
|
|
if (random_unit(r)) {
|
|
|
|
|
int mfac = 0;
|
|
|
|
|
switch (rand() % 3) {
|
|
|
|
|
case 0:
|
|
|
|
|
mfac = 100;
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), rand() % 8 + 1, RC_FIREDRAGON);
|
|
|
|
|
if (u->number == 1) {
|
|
|
|
|
set_string(&u->name, "Feuerdrache");
|
|
|
|
|
} else {
|
|
|
|
|
set_string(&u->name, "Feuerdrachen");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
mfac = 500;
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), rand() % 4 + 1, RC_DRAGON);
|
|
|
|
|
if (u->number == 1) {
|
|
|
|
|
set_string(&u->name, "Drache");
|
|
|
|
|
} else {
|
|
|
|
|
set_string(&u->name, "Drachen");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
mfac = 1000;
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), rand() % 2 + 1, RC_WYRM);
|
|
|
|
|
if (u->number == 1) {
|
|
|
|
|
set_string(&u->name, "Wyrm");
|
|
|
|
|
} else {
|
|
|
|
|
set_string(&u->name, "Wyrme");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (mfac) set_money(u, u->number * (rand() % mfac));
|
|
|
|
|
guard(u, GUARD_ALL);
|
|
|
|
|
}
|
|
|
|
|
case 2: /* Terrainver<65>nderung */
|
|
|
|
|
if (rterrain(r) != T_FIREWALL) {
|
|
|
|
|
if (rterrain(r) != T_OCEAN) {
|
|
|
|
|
direction_t dir;
|
|
|
|
|
for (dir=0;dir!=MAXDIRECTIONS;++dir) {
|
|
|
|
|
if (rconnect(r, dir) && rterrain(rconnect(r, dir)) == T_OCEAN) break;
|
|
|
|
|
}
|
|
|
|
|
if (dir!=MAXDIRECTIONS) {
|
|
|
|
|
ship * sh = r->ships;
|
|
|
|
|
while (sh) {
|
|
|
|
|
ship * nsh = sh->next;
|
|
|
|
|
damage_ship(sh, 0.50);
|
|
|
|
|
if (sh->damage >= sh->size * DAMAGE_SCALE) destroy_ship(sh, r);
|
|
|
|
|
sh = nsh;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (u = r->units; u;) {
|
|
|
|
|
u2 = u->next;
|
|
|
|
|
if (u->race != RC_SPELL && u->ship == 0) {
|
|
|
|
|
set_number(u, 0);
|
|
|
|
|
}
|
|
|
|
|
u = u2;
|
|
|
|
|
}
|
|
|
|
|
sprintf(buf, "Ein gewaltige Flutwelle verschlingt %s und "
|
|
|
|
|
"alle Bewohner.", regionid(r));
|
|
|
|
|
addmessage(r, 0, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
|
|
|
|
|
for (b = rbuildings(r); b;) {
|
|
|
|
|
b2 = b->next;
|
|
|
|
|
destroy_building(b);
|
|
|
|
|
b = b2;
|
|
|
|
|
}
|
|
|
|
|
terraform(r, T_OCEAN);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
direction_t dir;
|
|
|
|
|
for (dir=0;dir!=MAXDIRECTIONS;++dir) {
|
|
|
|
|
if (rconnect(r, dir) && rterrain(rconnect(r, dir)) != T_OCEAN) break;
|
|
|
|
|
}
|
|
|
|
|
if (dir!=MAXDIRECTIONS) {
|
|
|
|
|
switch (rand() % 8) {
|
|
|
|
|
case 0:
|
|
|
|
|
terraform(r, T_PLAIN);
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
terraform(r, T_HIGHLAND);
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
terraform(r, T_MOUNTAIN);
|
|
|
|
|
break;
|
|
|
|
|
case 3:
|
|
|
|
|
terraform(r, T_GLACIER);
|
|
|
|
|
break;
|
|
|
|
|
case 4:
|
|
|
|
|
terraform(r, T_DESERT);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
terraform(r, T_SWAMP);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double
|
|
|
|
|
chaosfactor(region * r)
|
|
|
|
|
{
|
|
|
|
|
attrib * a = a_find(r->attribs, &at_chaoscount);
|
|
|
|
|
if (!a) return 0;
|
|
|
|
|
return ((double) a->data.i / 1000.0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
drown(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
|
|
|
|
|
for (r=regions; r; r=r->next) if (rterrain(r) == T_OCEAN) {
|
|
|
|
|
unit ** up = up=&r->units;
|
|
|
|
|
while (*up) {
|
|
|
|
|
unit *u = *up;
|
|
|
|
|
int amphibian_level = fspecial(u->faction, FS_AMPHIBIAN);
|
|
|
|
|
|
|
|
|
|
if (u->ship) {
|
|
|
|
|
up=&u->next;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (amphibian_level) {
|
|
|
|
|
int dead = damage_unit(u, "5d1", false, false);
|
|
|
|
|
if (dead) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"drown_amphibian_dead%d:amount%u:unit",dead, u));
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"drown_amphibian_nodead%u:unit",u));
|
|
|
|
|
}
|
|
|
|
|
} else if (!canswim(u)) {
|
|
|
|
|
scale_number(u, 0);
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"drown%u:unit",u));
|
|
|
|
|
}
|
|
|
|
|
if (*up==u) up=&u->next;
|
|
|
|
|
}
|
|
|
|
|
remove_empty_units_in_region(r);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
region *
|
|
|
|
|
rrandneighbour(region *r)
|
|
|
|
|
{
|
|
|
|
|
direction_t i;
|
|
|
|
|
region *rc = NULL;
|
|
|
|
|
int rr, c = 0;
|
|
|
|
|
|
|
|
|
|
/* Nachsehen, wieviele Regionen in Frage kommen */
|
|
|
|
|
|
|
|
|
|
for (i = 0; i != MAXDIRECTIONS; i++) {
|
|
|
|
|
c++;
|
|
|
|
|
}
|
|
|
|
|
/* Zuf<75>llig eine ausw<73>hlen */
|
|
|
|
|
|
|
|
|
|
rr = rand() % c;
|
|
|
|
|
|
|
|
|
|
/* Durchz<68>hlen */
|
|
|
|
|
|
|
|
|
|
c = -1;
|
|
|
|
|
for (i = 0; i != MAXDIRECTIONS; i++) {
|
|
|
|
|
rc = rconnect(r, i);
|
|
|
|
|
c++;
|
|
|
|
|
if (c == rr) break;
|
|
|
|
|
}
|
|
|
|
|
assert(i!=MAXDIRECTIONS);
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
volcano_outbreak(region *r)
|
|
|
|
|
{
|
|
|
|
|
attrib *a;
|
|
|
|
|
region *rn;
|
|
|
|
|
unit *u, **up;
|
|
|
|
|
int dead;
|
|
|
|
|
faction *f;
|
|
|
|
|
|
|
|
|
|
for (u=r->units; u; u=u->next) {
|
|
|
|
|
f = u->faction;
|
|
|
|
|
freset(f, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
rn = rrandneighbour(r);
|
|
|
|
|
|
|
|
|
|
/* Vulkan-Region verw<72>sten */
|
|
|
|
|
|
|
|
|
|
rsettrees(r, 0);
|
|
|
|
|
|
|
|
|
|
a = a_find(r->attribs, &at_reduceproduction);
|
|
|
|
|
if (!a) a = a_add(&r->attribs, a_new(&at_reduceproduction));
|
|
|
|
|
|
|
|
|
|
/* Produktion vierteln ... */
|
|
|
|
|
a->data.sa[0] = 25;
|
|
|
|
|
/* F<>r 6-17 Runden */
|
|
|
|
|
a->data.sa[1] = (short)(a->data.sa[1] + 6 + rand()%12);
|
|
|
|
|
|
|
|
|
|
/* Personen bekommen 4W10 Punkte Schaden. */
|
|
|
|
|
|
|
|
|
|
for (up=&r->units; *up;) {
|
|
|
|
|
unit * u = *up;
|
|
|
|
|
dead = damage_unit(u, "4d10", true, false);
|
|
|
|
|
if (dead) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"volcano_dead%u:unit%r:region%i:dead", u, r, dead));
|
|
|
|
|
}
|
|
|
|
|
if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
if (rn) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"volcanooutbreak%r:regionv%r:regionn", r, rn));
|
|
|
|
|
} else {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"volcanooutbreaknn%r:region", r));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (u==*up) up=&u->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
remove_empty_units_in_region(r);
|
|
|
|
|
|
|
|
|
|
/* Zuf<75>llige Nachbarregion verw<72>sten */
|
|
|
|
|
|
|
|
|
|
if (rn) {
|
|
|
|
|
|
|
|
|
|
rsettrees(r, 0);
|
|
|
|
|
|
|
|
|
|
a = a_add(&rn->attribs, a_new(&at_reduceproduction));
|
|
|
|
|
if (!a) a = a_add(&r->attribs, a_new(&at_reduceproduction));
|
|
|
|
|
|
|
|
|
|
/* Produktion vierteln ... */
|
|
|
|
|
a->data.sa[0] = 25;
|
|
|
|
|
/* F<>r 6-17 Runden */
|
|
|
|
|
a->data.sa[1] = (short)(a->data.sa[1] + 6 + rand()%12);
|
|
|
|
|
|
|
|
|
|
/* Personen bekommen 3W10 Punkte Schaden. */
|
|
|
|
|
for (up=&rn->units; *up;) {
|
|
|
|
|
unit * u = *up;
|
|
|
|
|
dead = damage_unit(u, "3d10", true, false);
|
|
|
|
|
if (dead) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"volcano_dead%u:unit%r:region%i:dead", u, rn, dead));
|
|
|
|
|
}
|
|
|
|
|
if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"volcano_dead%u:unit%r:region%i:dead", u, r, dead));
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
if (u==*up) up=&u->next;
|
|
|
|
|
}
|
|
|
|
|
remove_empty_units_in_region(rn);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
move_iceberg(region *r)
|
|
|
|
|
{
|
|
|
|
|
attrib *a;
|
|
|
|
|
direction_t dir;
|
|
|
|
|
region *rc;
|
|
|
|
|
|
|
|
|
|
a = a_find(r->attribs, &at_iceberg);
|
|
|
|
|
if (!a) {
|
|
|
|
|
dir = (direction_t)(rand()%MAXDIRECTIONS);
|
|
|
|
|
a = a_add(&r->attribs, make_iceberg(dir));
|
|
|
|
|
} else {
|
|
|
|
|
if (rand()%100 < 20) {
|
|
|
|
|
dir = (direction_t)(rand()%MAXDIRECTIONS);
|
|
|
|
|
a->data.i = dir;
|
|
|
|
|
} else {
|
|
|
|
|
dir = (direction_t)a->data.i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rc = rconnect(r, dir);
|
|
|
|
|
|
|
|
|
|
if (rc && rterrain(rc) != T_GLACIER && rterrain(rc) != T_ICEBERG
|
|
|
|
|
&& rterrain(rc) != T_ICEBERG_SLEEP) {
|
|
|
|
|
if (rterrain(rc) == T_OCEAN) { /* Eisberg treibt */
|
|
|
|
|
ship *sh, *shn;
|
|
|
|
|
unit *u;
|
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (u=r->units; u; u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u=r->units; u; u=u->next) if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"iceberg_drift%r:region%d:dir", r, dir));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
x = r->x;
|
|
|
|
|
y = r->y;
|
|
|
|
|
|
|
|
|
|
runhash(r);
|
|
|
|
|
runhash(rc);
|
|
|
|
|
r->x = rc->x;
|
|
|
|
|
r->y = rc->y;
|
|
|
|
|
rc->x = x;
|
|
|
|
|
rc->y = y;
|
|
|
|
|
rhash(rc);
|
|
|
|
|
rhash(r);
|
|
|
|
|
|
|
|
|
|
/* rc ist der Ozean (Ex-Eisberg), r der Eisberg (Ex-Ozean) */
|
|
|
|
|
|
|
|
|
|
/* Schiffe aus dem Zielozean werden in den Eisberg transferiert
|
|
|
|
|
* und nehmen Schaden. */
|
|
|
|
|
|
|
|
|
|
for (sh = r->ships; sh; sh=sh->next) freset(sh, FL_DH);
|
|
|
|
|
|
|
|
|
|
for (sh = r->ships; sh; sh = sh->next) {
|
|
|
|
|
/* Meldung an Kapit<69>n */
|
|
|
|
|
damage_ship(sh, 0.10);
|
|
|
|
|
fset(sh, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Personen, Schiffe und Geb<65>ude verschieben */
|
|
|
|
|
while (rc->buildings) {
|
|
|
|
|
translist(&rc->buildings, &r->buildings, rc->buildings);
|
|
|
|
|
}
|
|
|
|
|
while (rc->ships) {
|
|
|
|
|
fset(rc->ships, FL_DH);
|
|
|
|
|
damage_ship(rc->ships, 0.10);
|
|
|
|
|
move_ship(rc->ships, rc, r, NULL);
|
|
|
|
|
}
|
|
|
|
|
while (rc->units) {
|
|
|
|
|
building * b = rc->units->building;
|
|
|
|
|
u = rc->units;
|
|
|
|
|
move_unit(rc->units, r, NULL);
|
|
|
|
|
u->building = b; /* move_unit macht ein leave() */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Besch<63>digte Schiffe k<>nnen sinken */
|
|
|
|
|
|
|
|
|
|
for (sh = r->ships; sh;) {
|
|
|
|
|
shn = sh->next;
|
|
|
|
|
if (fval(sh, FL_DH)) {
|
|
|
|
|
u = captain(sh, r);
|
|
|
|
|
if (sh->damage>=sh->size * DAMAGE_SCALE) {
|
|
|
|
|
if (u) add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"overrun_by_iceberg_des%h:ship", sh));
|
|
|
|
|
destroy_ship(sh, r);
|
|
|
|
|
} else {
|
|
|
|
|
if (u) add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"overrun_by_iceberg%h:ship", sh));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
sh = shn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else if (rand()%100 < 20) { /* Eisberg bleibt als Gletscher liegen */
|
|
|
|
|
unit *u;
|
|
|
|
|
|
|
|
|
|
rsetterrain(r, T_GLACIER);
|
|
|
|
|
a_remove(&r->attribs, a);
|
|
|
|
|
|
|
|
|
|
for (u=r->units; u; u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u=r->units; u; u=u->next) if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"iceberg_land%r:region", r));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
move_icebergs(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
|
|
|
|
|
for (r=regions; r; r=r->next) if (rterrain(r) == T_ICEBERG && !fval(r, RF_DH) && rand()%100 < 60) {
|
|
|
|
|
fset(r, RF_DH);
|
|
|
|
|
move_iceberg(r);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
create_icebergs(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
|
|
|
|
|
for (r=regions; r; r=r->next) if (rterrain(r) == T_ICEBERG_SLEEP && rand()%100 < 5) {
|
|
|
|
|
boolean has_ocean_neighbour = false;
|
|
|
|
|
direction_t dir;
|
|
|
|
|
region *rc;
|
|
|
|
|
unit *u;
|
|
|
|
|
|
|
|
|
|
for (dir=0; dir < MAXDIRECTIONS; dir++) {
|
|
|
|
|
rc = rconnect(r, dir);
|
|
|
|
|
if (rc && rterrain(rc) == T_OCEAN) {
|
|
|
|
|
has_ocean_neighbour = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!has_ocean_neighbour) continue;
|
|
|
|
|
|
|
|
|
|
rsetterrain(r, T_ICEBERG);
|
|
|
|
|
|
|
|
|
|
fset(r, RF_DH);
|
|
|
|
|
move_iceberg(r);
|
|
|
|
|
|
|
|
|
|
for (u=r->units; u; u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u=r->units; u; u=u->next) if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"iceberg_create%r:region", r));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
randomevents(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
building *b, *b2;
|
|
|
|
|
unit *u, *u2;
|
|
|
|
|
int n;
|
|
|
|
|
int unfed;
|
|
|
|
|
|
|
|
|
|
/* Eiseberge */
|
|
|
|
|
for (r=regions; r; r=r->next) freset(r, RF_DH);
|
|
|
|
|
create_icebergs();
|
|
|
|
|
move_icebergs();
|
|
|
|
|
drown();
|
|
|
|
|
|
|
|
|
|
/* Orks vermehren sich */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
plane * p = rplane(r);
|
|
|
|
|
/* there is a flag for planes without orc growth: */
|
|
|
|
|
if (p && (p->flags & PFL_NOORCGROWTH)) continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if ( (u->race == RC_ORC || is_cursed(u->attribs, C_ORC, 0))
|
|
|
|
|
&& get_skill(u, SK_MAGIC) == 0
|
|
|
|
|
&& get_skill(u, SK_ALCHEMY) == 0) {
|
|
|
|
|
int increase = 0;
|
|
|
|
|
int num, prob;
|
|
|
|
|
|
|
|
|
|
if (u->race == RC_ORC) {
|
|
|
|
|
num = u->number;
|
|
|
|
|
prob = 5;
|
|
|
|
|
} else {
|
|
|
|
|
num = min(get_cursedmen(u->attribs, C_ORC, 0), u->number);
|
|
|
|
|
prob = get_curseeffect(u->attribs, C_ORC, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (n = (num - get_item(u, I_CHASTITY_BELT)); n > 0; n--) {
|
|
|
|
|
if (rand() % 100 < prob) {
|
|
|
|
|
++increase;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (increase) {
|
|
|
|
|
if (u->race == RC_ORC) {
|
|
|
|
|
int s;
|
|
|
|
|
s = get_skill(u, SK_SWORD) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_SWORD,
|
|
|
|
|
increase * max(30, s));
|
|
|
|
|
s = get_skill(u, SK_SPEAR) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_SPEAR,
|
|
|
|
|
increase * max(30, s));
|
|
|
|
|
s = get_skill(u, SK_TACTICS) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_TACTICS,
|
|
|
|
|
increase * s);
|
|
|
|
|
s = get_skill(u, SK_LONGBOW) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_LONGBOW,
|
|
|
|
|
increase * s);
|
|
|
|
|
s = get_skill(u, SK_CROSSBOW) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_CROSSBOW,
|
|
|
|
|
increase * s);
|
|
|
|
|
s = get_skill(u, SK_CATAPULT) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_CATAPULT,
|
|
|
|
|
increase * s);
|
|
|
|
|
s = get_skill(u, SK_AUSDAUER) / (u->number * 2);
|
|
|
|
|
change_skill(u, SK_AUSDAUER,
|
|
|
|
|
increase * s);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set_number(u, u->number + increase);
|
|
|
|
|
|
|
|
|
|
u->hp += unit_max_hp(u) * increase;
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"orcgrowth%u:unit%i:amount%s:race",
|
|
|
|
|
u, increase, increase == 1 ? race[u->race].name[0] : race[u->race].name[1]));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Talente von D<>monen verschieben sich und D<>monen fressen Bauern */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
int bauernblut = 0;
|
|
|
|
|
boolean bfind = false;
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->race == RC_DAEMON) {
|
|
|
|
|
/* Alles Bauernblut der Region z<>hlen.
|
|
|
|
|
* warnung: bauernblut einer partei hilft im moment der anderen
|
|
|
|
|
* so selten wie das benutzt wird, ist das erstmal wursht,
|
|
|
|
|
* aber ein TODO f<EFBFBD>rs BUGS File.
|
|
|
|
|
* Es ist auch deshalb fast egal, weil es ja im Grunde nicht dem D<EFBFBD>mon,
|
|
|
|
|
* sondern der Region zu gute kommt - und da ist der anwender schnuppe
|
|
|
|
|
*/
|
|
|
|
|
skill_t sk;
|
|
|
|
|
int dc;
|
|
|
|
|
if (!bfind) {
|
|
|
|
|
unit * ud = u;
|
|
|
|
|
while (ud) {
|
|
|
|
|
attrib * a = a_find(ud->attribs, &at_bauernblut);
|
|
|
|
|
if (a) bauernblut += a->data.i;
|
|
|
|
|
do { ud=ud->next; } while (ud && ud->race!=RC_DAEMON);
|
|
|
|
|
}
|
|
|
|
|
bfind = true;
|
|
|
|
|
}
|
|
|
|
|
if (r->planep && !fval(r->planep, PFL_NOFEED)) {
|
|
|
|
|
unfed = (u->number - bauernblut) - rpeasants(r);
|
|
|
|
|
if (unfed > 0) {
|
|
|
|
|
#ifdef DAEMON_HUNGER
|
|
|
|
|
hunger(u, unfed); /* nicht gef<65>tterte d<>monen hungern */
|
|
|
|
|
#else
|
|
|
|
|
c = 0;
|
|
|
|
|
for (n = 0; n < unfed; n++) {
|
|
|
|
|
if (rand() % 100 < 10) {
|
|
|
|
|
c++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (c) {
|
|
|
|
|
scale_number(u, u->number - c);
|
|
|
|
|
sprintf(buf, "%d D<>monen von %s sind hungrig in "
|
|
|
|
|
"ihre Sph<70>re zur<75>ckgekehrt.", c, unitname(u));
|
|
|
|
|
addmessage(0, u->faction, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
dc = max(u->number - bauernblut, 0);
|
|
|
|
|
dc = min(dc, rpeasants(r));
|
|
|
|
|
dc = rpeasants(r) - max(u->number - bauernblut, 0);
|
|
|
|
|
dc = max(0, dc);
|
|
|
|
|
rsetpeasants(r, dc);
|
|
|
|
|
for (sk = 0; sk != MAXSKILLS; sk++) {
|
|
|
|
|
if (get_skill(u, sk) > 0 && rand() % 100 < 25) {
|
|
|
|
|
if (rand() % 100 < 60) {
|
|
|
|
|
change_skill(u, sk, u->number * (rand() % 90 + 1));
|
|
|
|
|
} else {
|
|
|
|
|
int s = get_skill(u, sk);
|
|
|
|
|
int r = u->number * (rand() % 90 + 1);
|
|
|
|
|
r = min(s, r);
|
|
|
|
|
change_skill(u, sk, -r);
|
|
|
|
|
#if 0
|
|
|
|
|
/* ist kaputt. wei<65> nicht, wieso. */
|
|
|
|
|
if (r==s) {
|
|
|
|
|
sprintf(buf, "Durch Konzentrationsmangel vergi<67>t %s das Talent %s vollst<73>ndig", unitname(u), skillnames[sk]);
|
|
|
|
|
addmessage(0, u->faction, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Elfen generieren Wald, Personen in Steinkreisen k<>nnen Einh<6E>rner
|
|
|
|
|
* bekommen. */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
if (r->land && !fval(r, RF_MALLORN)) {
|
|
|
|
|
int trees = rtrees(r);
|
|
|
|
|
int maxgen = (production(r) * MAXPEASANTS_PER_AREA)/8;
|
|
|
|
|
for (u = r->units; u && maxgen > 0; u = u->next) {
|
|
|
|
|
if (u->race == RC_ELF) {
|
|
|
|
|
for (n = u->number; n && maxgen > 0; n--) {
|
|
|
|
|
if (rand() % 1000 < 15) { /* 1.5% Chance */
|
|
|
|
|
trees++;
|
|
|
|
|
}
|
|
|
|
|
maxgen--;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
rsettrees(r, trees);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (u=r->units; u; u=u->next) {
|
|
|
|
|
if (!(race[u->race].ec_flags & NOGIVE)) {
|
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
|
|
|
|
if (btype == &bt_blessedstonecircle) {
|
|
|
|
|
int n, c = 0;
|
|
|
|
|
for (n=0; n<u->number; n++) if (rand()%100 < 2) {
|
|
|
|
|
change_item(u, I_UNICORN, 1);
|
|
|
|
|
c++;
|
|
|
|
|
}
|
|
|
|
|
if (c) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
2001-04-13 16:39:57 +02:00
|
|
|
|
"scunicorn%u:unit%i:amount%X:type",u,c,
|
|
|
|
|
olditemtype[I_UNICORN]->rtype));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Orkifizierte Regionen mutieren und mutieren zur<75>ck */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
if (fval(r, RF_ORCIFIED)) {
|
|
|
|
|
direction_t dir;
|
|
|
|
|
int chance = 0;
|
|
|
|
|
for (dir = 0; dir < MAXDIRECTIONS; dir++) {
|
|
|
|
|
region *rc = rconnect(r, dir);
|
|
|
|
|
if (rc && rpeasants(rc) > 0 && !fval(rc, RF_ORCIFIED)) chance += 2;
|
|
|
|
|
}
|
|
|
|
|
if (rand()%100 < chance) {
|
|
|
|
|
freset(r, RF_ORCIFIED);
|
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next ) freset(u2->faction, FL_DH);
|
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next ) {
|
|
|
|
|
if (!fval(u2->faction, FL_DH)) {
|
|
|
|
|
add_message(&r->msgs, new_message(u2->faction,
|
|
|
|
|
"deorcified%r:region", r));
|
|
|
|
|
fset(u2->faction, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
attrib *a = a_find(r->attribs, &at_orcification);
|
|
|
|
|
if (a!=NULL) {
|
|
|
|
|
int chance = 0;
|
|
|
|
|
if (rpeasants(r) <= 0) continue;
|
|
|
|
|
chance = (a->data.i*100)/rpeasants(r);
|
|
|
|
|
if (rand()%100 < chance) {
|
|
|
|
|
fset(r, RF_ORCIFIED);
|
|
|
|
|
a_remove(&r->attribs, a);
|
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next) freset(u2->faction, FL_DH);
|
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next) {
|
|
|
|
|
if (!fval(u2->faction, FL_DH)) {
|
|
|
|
|
add_message(&r->msgs, new_message(u2->faction,
|
|
|
|
|
"orcified%r:region", r));
|
|
|
|
|
fset(u2->faction, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
a->data.i -= max(10,a->data.i/10);
|
|
|
|
|
if (a->data.i <= 0) a_remove(&r->attribs, a);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Vulkane qualmen, brechen aus ... */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
switch(rterrain(r)) {
|
|
|
|
|
case T_VOLCANO:
|
|
|
|
|
if (rand()%100 < 5) {
|
|
|
|
|
rsetterrain(r, T_VOLCANO_SMOKING);
|
|
|
|
|
/* Meldungen generieren */
|
|
|
|
|
for (u = r->units; u; u = u->next ) freset(u->faction, FL_DH);
|
|
|
|
|
for (u = r->units; u; u = u->next ) {
|
|
|
|
|
if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
add_message(&r->msgs, new_message(u->faction,
|
|
|
|
|
"volcanostartsmoke%r:region", r));
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case T_VOLCANO_SMOKING:
|
|
|
|
|
if (rand()%100 < 10) {
|
|
|
|
|
rsetterrain(r, T_VOLCANO);
|
|
|
|
|
/* Meldungen generieren */
|
|
|
|
|
for (u = r->units; u; u = u->next ) freset(u->faction, FL_DH);
|
|
|
|
|
for (u = r->units; u; u = u->next ) {
|
|
|
|
|
if (!fval(u->faction, FL_DH)) {
|
|
|
|
|
add_message(&r->msgs, new_message(u->faction,
|
|
|
|
|
"volcanostopsmoke%r:region", r));
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (rand()%100 < 8) {
|
|
|
|
|
volcano_outbreak(r);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Monumente zerfallen, Schiffe verfaulen */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
for (b = rbuildings(r); b; b = b2) {
|
|
|
|
|
b2 = b->next;
|
|
|
|
|
if (fval(b->type, BTF_DECAY) && !buildingowner(r, b)) {
|
|
|
|
|
b->size -= max(1, (b->size * 20) / 100);
|
|
|
|
|
if (b->size == 0) {
|
|
|
|
|
destroy_building(b);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Drachen und Seeschlangen k<>nnen entstehen */
|
|
|
|
|
|
|
|
|
|
printf("\n");
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
unit * u;
|
|
|
|
|
if (rterrain(r) == T_OCEAN && rand()%10000 < 1) {
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), 1, RC_SEASERPENT);
|
|
|
|
|
set_skill(u, SK_MAGIC, u->number * 300);
|
|
|
|
|
set_skill(u, SK_OBSERVATION, u->number * 180);
|
|
|
|
|
set_skill(u, SK_STEALTH, u->number * 90);
|
|
|
|
|
set_skill(u, SK_AUSDAUER, u->number * 30);
|
|
|
|
|
set_string(&u->name, "Seeschlange");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((rterrain(r) == T_GLACIER
|
|
|
|
|
|| rterrain(r) == T_SWAMP || rterrain(r) == T_DESERT)
|
|
|
|
|
&& rand() % 10000 < (5 + 100 * chaosfactor(r))) {
|
|
|
|
|
|
|
|
|
|
switch (rand() % 10) {
|
|
|
|
|
case 0:
|
|
|
|
|
case 1:
|
|
|
|
|
case 2:
|
|
|
|
|
case 3:
|
|
|
|
|
case 4:
|
|
|
|
|
case 5:
|
|
|
|
|
case 6:
|
|
|
|
|
case 7:
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), nrand(60, 20) + 1, RC_FIREDRAGON);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), nrand(30, 20) + 1, RC_DRAGON);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set_money(u, u->number * (rand() % 500 + 100));
|
|
|
|
|
set_skill(u, SK_MAGIC, u->number * 300);
|
|
|
|
|
set_skill(u, SK_OBSERVATION, u->number * (rand() % 180));
|
|
|
|
|
set_skill(u, SK_AUSDAUER, u->number * 30);
|
|
|
|
|
set_skill(u, SK_STEALTH, u->number * 30);
|
2001-04-14 14:11:45 +02:00
|
|
|
|
log_printf("%d %s in %s.\n", u->number,
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
race[u->race].name[1], regionname(r, NULL));
|
2001-04-14 14:11:45 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
name_unit(u);
|
|
|
|
|
set_string(&u->lastorder, "WARTEN");
|
|
|
|
|
|
|
|
|
|
if (u->number == 1) {
|
|
|
|
|
sprintf(buf, "Es wurde ein %s gesichtet.",
|
|
|
|
|
race[u->race].name[0]);
|
|
|
|
|
} else {
|
|
|
|
|
sprintf(buf, "Es wurden %d %s gesichtet.",
|
|
|
|
|
u->number, race[u->race].name[1]);
|
|
|
|
|
}
|
|
|
|
|
addmessage(r, 0, buf, MSG_COMMENT, ML_IMPORTANT);
|
|
|
|
|
if (u->number == 1) {
|
|
|
|
|
sprintf(buf, "In %s wurde ein %s gesichtet.", regionid(r),
|
|
|
|
|
race[u->race].name[0]);
|
|
|
|
|
} else {
|
|
|
|
|
sprintf(buf, "In %s wurden %d %s gesichtet.", regionid(r),
|
|
|
|
|
u->number, race[u->race].name[1]);
|
|
|
|
|
}
|
|
|
|
|
for (u=r->units;u;u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u=r->units;u;u=u->next) {
|
|
|
|
|
faction * f = u->faction;
|
|
|
|
|
if (!fval(f, FL_DH)) {
|
|
|
|
|
addmessage(0, f, buf, MSG_EVENT, ML_IMPORTANT);
|
|
|
|
|
fset(f, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Untote k<>nnen entstehen */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
2001-02-02 09:40:49 +01:00
|
|
|
|
int unburied = deathcount(r);
|
2001-01-31 18:40:53 +01:00
|
|
|
|
|
|
|
|
|
if(is_cursed(r->attribs, C_HOLYGROUND, 0)) continue;
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* Chance 0.1% * chaosfactor */
|
2001-02-10 11:40:12 +01:00
|
|
|
|
if (r->land && unburied > r->land->peasants / 20 && rand() % 10000 < (100 + 100 * chaosfactor(r))) {
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
/* es ist sinnfrei, wenn irgendwo im Wald 3er-Einheiten Untote entstehen.
|
2001-02-10 11:40:12 +01:00
|
|
|
|
* Lieber sammeln lassen, bis sie mindestens 5% der Bev<EFBFBD>lkerung sind, und
|
|
|
|
|
* dann erst auferstehen. */
|
2001-02-02 09:40:49 +01:00
|
|
|
|
int undead = unburied / (rand() % 2 + 1);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
race_t rc;
|
|
|
|
|
int i;
|
|
|
|
|
|
2001-02-02 09:40:49 +01:00
|
|
|
|
if (!undead || r->age < 20) continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
switch(rand()%3) {
|
|
|
|
|
case 0:
|
|
|
|
|
rc = RC_SKELETON; break;
|
|
|
|
|
case 1:
|
|
|
|
|
rc = RC_ZOMBIE; break;
|
|
|
|
|
default:
|
|
|
|
|
rc = RC_GHOUL; break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION), undead, rc);
|
|
|
|
|
if ((rc == RC_SKELETON || rc == RC_ZOMBIE) && rand()%10 < 4) {
|
|
|
|
|
set_item(u, I_RUSTY_SWORD, undead);
|
|
|
|
|
if (rand()%10 < 3) {
|
|
|
|
|
set_item(u, I_RUSTY_SHIELD, undead);
|
|
|
|
|
}
|
|
|
|
|
if (rand()%10 < 2) {
|
|
|
|
|
set_item(u, I_RUSTY_CHAIN_MAIL, undead);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i=0;i < MAXSKILLS;i++) {
|
|
|
|
|
if (race[rc].bonus[i] >= 1) {
|
|
|
|
|
set_skill(u, i, 30 * u->number);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
u->hp = unit_max_hp(u) * u->number;
|
|
|
|
|
|
|
|
|
|
deathcounts(r, -undead);
|
|
|
|
|
set_string(&u->lastorder, "WARTEN");
|
|
|
|
|
name_unit(u);
|
|
|
|
|
|
2001-04-14 14:11:45 +02:00
|
|
|
|
log_printf("%d %s in %s.\n", u->number,
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
race[u->race].name[1], regionname(r, NULL));
|
2001-04-14 14:11:45 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
add_message(&r->msgs, new_message(NULL,
|
|
|
|
|
"undeadrise%r:region", r));
|
|
|
|
|
for (u=r->units;u;u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u=r->units;u;u=u->next) {
|
|
|
|
|
if (fval(u->faction, FL_DH)) continue;
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
add_message(&u->faction->msgs, new_message(NULL,
|
|
|
|
|
"undeadrise%r:region", r));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2001-02-02 09:40:49 +01:00
|
|
|
|
int i = deathcount(r);
|
|
|
|
|
if (i) {
|
|
|
|
|
/* Gr<47>ber verwittern, 3% der Untoten finden die ewige Ruhe */
|
|
|
|
|
deathcounts(r, (int)(1+i*0.03));
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r=r->next) {
|
|
|
|
|
for (u=r->units; u; u=u->next) {
|
2001-02-03 14:45:35 +01:00
|
|
|
|
if (u->faction->no != MONSTER_FACTION && (race[u->race].flags & RCF_DESERT)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (rand()%100 < 5) {
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"desertion%u:unit%r:region", u, r));
|
2001-02-10 15:07:29 +01:00
|
|
|
|
u_setfaction(u, findfaction(MONSTER_FACTION));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Fr<46>hling, die B<>ume schlagen aus. */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
if (fval(r, RF_CHAOTIC) ||(r->x >= -13 && r->x <= -6 && r->y >= 50 && r->y <= 57)) {
|
|
|
|
|
if (woodcount(r) >= 40 && rand()%100 < 33) {
|
|
|
|
|
int trees = rtrees(r);
|
|
|
|
|
int treemen = rand()%(max(50,trees)/3);
|
|
|
|
|
treemen = max(25, treemen);
|
|
|
|
|
woodcounts(r, -40);
|
|
|
|
|
trees = max(0, trees-treemen);
|
|
|
|
|
rsettrees(r, trees);
|
|
|
|
|
u = createunit(r, findfaction(MONSTER_FACTION),treemen, RC_TREEMAN);
|
|
|
|
|
set_string(&u->lastorder, "WARTEN");
|
|
|
|
|
/* guard(u, GUARD_ALL); kein auto-guard! erst in monster.c! */
|
|
|
|
|
change_skill(u,SK_OBSERVATION,180*u->number);
|
|
|
|
|
if (u->number == 1)
|
|
|
|
|
set_string(&u->name, "Ein w<>tender Ent");
|
|
|
|
|
else
|
|
|
|
|
set_string(&u->name, "W<EFBFBD>tende Ents");
|
|
|
|
|
|
2001-04-14 14:11:45 +02:00
|
|
|
|
log_printf("%d Ents in %s.\n", u->number, regionname(r, NULL));
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
add_message(&r->msgs, new_message(NULL,
|
|
|
|
|
"entrise%r:region", r));
|
|
|
|
|
for (u=r->units;u;u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u=r->units;u;u=u->next) {
|
|
|
|
|
if (fval(u->faction, FL_DH)) continue;
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
add_message(&u->faction->msgs, new_message(NULL,
|
|
|
|
|
"entrise%r:region", r));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Chaos */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
int i;
|
|
|
|
|
if (fval(r, RF_CHAOTIC)) {
|
|
|
|
|
chaos(r);
|
|
|
|
|
}
|
|
|
|
|
i = chaoscount(r);
|
|
|
|
|
if (!i) continue;
|
|
|
|
|
chaoscounts(r, -(int) (i * ((double) (rand() % 10)) / 100.0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef HERBS_ROT
|
|
|
|
|
/* Kr<4B>uter verrotten */
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
for (u = r->units; u; u=u->next) {
|
|
|
|
|
item **itmp = &u->items, *hbag = *i_find(&u->items, olditemtype[I_SACK_OF_CONSERVATION]);
|
|
|
|
|
int rot_chance = HERBROTCHANCE;
|
|
|
|
|
|
|
|
|
|
if (hbag) rot_chance = (HERBROTCHANCE*2)/5;
|
|
|
|
|
|
|
|
|
|
while (*itmp) {
|
|
|
|
|
item * itm = *itmp;
|
|
|
|
|
const herb_type * htype = resource2herb(itm->type->rtype);
|
|
|
|
|
int n = itm->number;
|
|
|
|
|
double k = n*rot_chance/100.0;
|
|
|
|
|
if (htype!=NULL) {
|
|
|
|
|
n = (int)(min(n, normalvariate(k, k/4)));
|
|
|
|
|
i_change(itmp, itm->type, -n);
|
|
|
|
|
}
|
|
|
|
|
if (itm==*itmp) itmp=&itm->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
dissolve_units();
|
|
|
|
|
check_split();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void growlaen(void) {
|
|
|
|
|
region *r;
|
|
|
|
|
regionlist *Berge=NULL, *rl;
|
|
|
|
|
unit *u;
|
|
|
|
|
int b=0, Laen, z, add;
|
|
|
|
|
short *add_laen;
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
|
|
|
if (!fval(r, RF_CHAOTIC) &&
|
|
|
|
|
rterrain(r) == T_MOUNTAIN)
|
|
|
|
|
{
|
|
|
|
|
Berge=add_regionlist(Berge,r);
|
|
|
|
|
b++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
add_laen=(short *)malloc(b*sizeof(short));
|
|
|
|
|
memset(add_laen, 0, b*sizeof(short));
|
|
|
|
|
|
|
|
|
|
Laen=b*MAXLAENPERTURN/14; /* Anzahl Berge * MAXLAENPERTURN/2 * Chance pro Berg */
|
|
|
|
|
|
|
|
|
|
while(Laen) { /* Laenverteilung <20>ber alle Berge */
|
|
|
|
|
z = rand()%b;
|
|
|
|
|
add = min(10 + rand() % 30, Laen);
|
|
|
|
|
add_laen[z] = (short)(add_laen[z]+add);
|
|
|
|
|
Laen -= add;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
z=0;
|
|
|
|
|
for (rl=Berge; rl; rl=rl->next, z++) {
|
|
|
|
|
region * r = rl->region;
|
|
|
|
|
if (add_laen[z]>0) {
|
|
|
|
|
if (a_find(rl->region->attribs, &at_laen))
|
|
|
|
|
rsetlaen(rl->region, add_laen[z]+rlaen(rl->region));
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
attrib *a=a_new(&at_laen);
|
|
|
|
|
a_add(&rl->region->attribs, a);
|
|
|
|
|
rsetlaen(rl->region, add_laen[z]);
|
|
|
|
|
|
|
|
|
|
/* Meldungen generieren */
|
|
|
|
|
for (u=r->units;u;u=u->next) freset(u->faction, FL_DH);
|
|
|
|
|
for (u = r->units; u; u = u->next ) {
|
|
|
|
|
if (!fval(u->faction, FL_DH) && eff_skill(u, SK_MINING, rl->region) >= olditemtype[I_EOG]->minskill)
|
|
|
|
|
{
|
|
|
|
|
add_message(&rl->region->msgs, new_message(u->faction,
|
|
|
|
|
"unveileog%u:unit%r:region", u, rl->region));
|
|
|
|
|
fset(u->faction, FL_DH);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free(add_laen);
|
|
|
|
|
free_regionlist(Berge);
|
|
|
|
|
}
|