server/src/eressea/korrektur.c

2259 lines
48 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
* Eressea PB(E)M host Copyright (C) 1998-2003
2001-01-25 10:37:55 +01:00
* Christian Schlittchen (corwin@amber.kn-bremen.de)
* Katja Zedel (katze@felidae.kn-bremen.de)
* Henning Peters (faroul@beyond.kn-bremen.de)
* Enno Rehling (enno@eressea-pbem.de)
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* based on:
*
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
* This program may not be sold or used commercially without prior written
* permission from the authors.
*/
#include <config.h>
#include <eressea.h>
2001-05-11 22:19:22 +02:00
/* misc includes */
#include <attributes/key.h>
2002-09-02 22:36:12 +02:00
#include <items/questkeys.h>
#include <items/catapultammo.h>
#include <modules/xecmd.h>
2002-09-02 22:36:12 +02:00
#ifdef ALLIANCES
#include <modules/alliance.h>
#endif
2001-01-25 10:37:55 +01:00
/* gamecode includes */
#include <economy.h>
/* kernel includes */
#include <border.h>
#include <building.h>
#include <ship.h>
2001-01-25 10:37:55 +01:00
#include <faction.h>
#include <item.h>
#include <magic.h>
#include <message.h>
#include <monster.h>
#include <movement.h>
#include <names.h>
#include <pathfinder.h>
#include <plane.h>
#include <pool.h>
#include <race.h>
#include <region.h>
#include <reports.h>
#include <resources.h>
2001-01-25 10:37:55 +01:00
#include <skill.h>
#include <teleport.h>
#include <unit.h>
#include <spell.h>
#include <alchemy.h>
#include <study.h>
2001-01-25 10:37:55 +01:00
/* util includes */
#include <attrib.h>
#include <language.h>
2001-01-25 10:37:55 +01:00
#include <base36.h>
#include <log.h>
2001-01-25 10:37:55 +01:00
#include <cvector.h>
#include <event.h>
2002-05-05 11:37:43 +02:00
#include <goodies.h>
2001-01-25 10:37:55 +01:00
#include <resolve.h>
#include <sql.h>
2001-01-25 10:37:55 +01:00
#include <vset.h>
/* libc includes */
#include <stdio.h>
#include <ctype.h>
2001-01-25 10:37:55 +01:00
#include <stdlib.h>
#include <string.h>
/* attributes includes */
#include <attributes/targetregion.h>
#include <attributes/key.h>
2001-12-30 09:24:05 +01:00
#undef XMAS1999
#undef XMAS2000
#undef XMAS2001
#undef XMAS2002
extern void reorder_owners(struct region * r);
2002-10-08 08:48:47 +02:00
#if 0
2002-09-02 22:36:12 +02:00
#define MAXALLIANCES 3
static void
init_alliances(void)
{
faction * f = factions;
alliance * aalliance[MAXALLIANCES];
if (alliances==NULL) {
aalliance[0] = makealliance(atoi36("dark"), "Ritter der dunklen Sonne");
aalliance[1] = makealliance(atoi36("deep"), "Monster aus der Tiefe");
aalliance[2] = makealliance(atoi36("pope"), "Priester des Vatikan");
}
while (f!=NULL) {
if (f->alliance==NULL && f->no!=MONSTER_FACTION) {
setalliance(f, aalliance[rand() % MAXALLIANCES]);
}
f=f->next;
}
}
#endif
static int
curse_emptiness(void)
{
const curse_type * ct = ct_find("godcursezone");
region * r;
for (r=regions;r!=NULL;r=r->next) {
unit * u = r->units;
if (r->land==NULL) continue;
if (fval(r, RF_CHAOTIC)) continue;
if (r->terrain==T_GLACIER) continue;
2002-10-08 08:48:47 +02:00
if (r->age<=200) continue;
if (get_curse(r->attribs, ct)) continue;
while (u && u->faction->no==MONSTER_FACTION) u=u->next;
if (u==NULL) fset(r, FL_MARK);
}
for (r=regions;r!=NULL;r=r->next) {
if (fval(r, FL_MARK)) {
direction_t d;
for (d=0;d!=MAXDIRECTIONS;++d) {
region * rn = rconnect(r,d);
if (rn==NULL) continue;
if (fval(rn, FL_MARK) || get_curse(rn->attribs, ct)) {
break;
}
}
if (d!=MAXDIRECTIONS) {
2002-09-02 22:36:12 +02:00
curse * c = create_curse(NULL, &r->attribs, ct,
100, 100, 0, 0);
2002-09-02 22:36:12 +02:00
curse_setflag(c, CURSE_ISNEW|CURSE_IMMUNE);
}
freset(r, FL_MARK);
2002-09-02 22:36:12 +02:00
}
}
return 0;
}
2003-05-13 17:43:18 +02:00
void
french_testers(void)
{
faction * f = factions;
const struct locale * french = find_locale("fr");
while (f!=NULL) {
if (f->locale==french) fset(f, FFL_NOTIMEOUT);
f = f->next;
}
}
static void
verify_owners(boolean bOnce)
{
region * r;
for (r=regions;r;r=r->next) {
unit * u;
boolean bFail = false;
for (u=r->units;u;u=u->next) {
if (u->building) {
unit * bo = buildingowner(r, u->building);
if (!fval(bo, UFL_OWNER)) {
log_error(("[verify_owners] %u ist Besitzer von %s, hat aber UFL_OWNER nicht.\n", unitname(bo), buildingname(u->building)));
bFail = true;
if (bOnce) break;
}
if (bo!=u && fval(u, UFL_OWNER)) {
log_error(("[verify_owners] %u ist NICHT Besitzer von %s, hat aber UFL_OWNER.\n", unitname(u), buildingname(u->building)));
bFail = true;
if (bOnce) break;
}
}
if (u->ship) {
unit * bo = shipowner(r, u->ship);
if (!fval(bo, UFL_OWNER)) {
log_error(("[verify_owners] %u ist Besitzer von %s, hat aber UFL_OWNER nicht.\n", unitname(bo), shipname(u->ship)));
bFail = true;
if (bOnce) break;
}
if (bo!=u && fval(u, UFL_OWNER)) {
log_error(("[verify_owners] %u ist NICHT Besitzer von %s, hat aber UFL_OWNER.\n", unitname(u), shipname(u->ship)));
bFail = true;
if (bOnce) break;
}
}
}
if (bFail) reorder_owners(r);
}
}
/* make sure that this is done only once! */
#define do_once(magic, fun) \
{ \
attrib * a = find_key(global.attribs, atoi36(magic)); \
if (!a) { \
log_warning(("[do_once] a unique fix %d=\"%s\" was applied.\n", atoi36(magic), magic)); \
if (fun == 0) a_add(&global.attribs, make_key(atoi36(magic))); \
2001-02-20 23:54:05 +01:00
} \
2001-01-25 10:37:55 +01:00
}
2002-03-10 11:09:16 +01:00
int
warn_items(void)
{
boolean found = 0;
region * r;
const item_type * it_money = it_find("money");
2002-03-10 11:09:16 +01:00
for (r=regions;r;r=r->next) {
unit * u;
for (u=r->units;u;u=u->next) {
item * itm;
for (itm=u->items;itm;itm=itm->next) {
if (itm->number>100000 && itm->type!=it_money) {
2002-03-10 11:09:16 +01:00
found = 1;
2002-05-02 23:19:45 +02:00
log_error(("Einheit %s hat %u %s\n",
unitid(u), itm->number,
2002-03-10 11:09:16 +01:00
resourcename(itm->type->rtype, 0)));
2001-01-25 10:37:55 +01:00
}
}
}
}
2002-03-10 11:09:16 +01:00
return found;
2001-01-25 10:37:55 +01:00
}
2001-12-30 09:24:05 +01:00
#ifdef XMAS1999
2001-01-25 10:37:55 +01:00
#include "race.h"
#include "movement.h"
static void
santa_comes_to_town(void)
{
unit * santa = ufindhash(atoi36("xmas"));
unit * tans = ufindhash(atoi36("tans"));
faction * f;
region * r;
if (!tans) r = findregion(0,0);
else r = tans->region;
if (!r) return;
while (santa && santa->race!=RC_ILLUSION) {
uunhash(santa);
santa->no = newunitid();
uhash(santa);
santa = ufindhash(atoi36("xmas"));
}
if (!santa) {
santa = createunit(r, findfaction(MONSTER_FACTION), 1, new_race[RC_ILLUSION]);
2001-01-25 10:37:55 +01:00
uunhash(santa);
santa->no = atoi36("xmas");
uhash(santa);
}
santa->age = 2;
santa->irace = new_race[RC_DAEMON];
2001-01-25 10:37:55 +01:00
set_string(&santa->name, "Ein dicker Gnom mit einem Rentierschlitten");
set_string(&santa->display, "hat: 6 Rentiere, Schlitten, Sack mit Geschenken, Kekse f<>r Khorne");
for (f=factions;f;f=f->next) {
unit * u;
runit *ru;
region * r;
item_t i = (item_t) (rand() % 4 + I_KEKS);
unit * senior = f->units;
for (u = f->units; u; u=u->nextF) {
if (senior->age < u->age) senior = u;
}
if (!senior) continue;
r = senior->region;
sprintf(buf, "%s gibt %d %s an %s.", unitname(santa), 1, itemdata[i].name[0], unitname(senior));
change_item(senior, i, 1);
addmessage(r, senior->faction, buf, MSG_COMMERCE, ML_INFO);
sprintf(buf, "von %s: 'Ho ho ho. Frohe Weihnachten, und %s f<>r %s.'", unitname(santa), itemdata[i].name[1], unitname(senior));
addmessage(r, 0, buf, MSG_MESSAGE, ML_IMPORTANT);
for (ru = r->durchgezogen; ru ; ru=ru->next)
if (ru->unit==santa) break;
if (!ru) {
ru = calloc(1, sizeof(runit));
ru->next = r->durchgezogen;
ru->unit = santa;
r->durchgezogen = ru;
}
}
}
#endif
#if 0
static void
init_hp(void)
{
region *r;
unit *u;
for (r = regions; r; r = r->next) {
for (u = r->units; u; u = u->next) {
if (u->hp == -1) {
u->hp = unit_max_hp(u) * u->number;
}
}
}
}
static void
convert(int x, int y, int *nx, int *ny)
{
*ny = -y;
*nx = x + (y + ((y>=0)?1:0))/2;
}
#endif
#if 0
static void
loesche_coasts(void)
{
region *r;
ship *sh;
for(r=regions;r;r=r->next) {
for(sh=r->ships;sh;sh=sh->next) {
sh->coast = NODIRECTION;
}
}
}
#endif
#if 0
static void
create_underworld(void)
{
int x, y;
region *r;
attrib *a;
plane * hell = create_new_plane(-1, "H<EFBFBD>lle", 100000, 100000, 100020, 100020, PFL_NONE);
2001-01-25 10:37:55 +01:00
for(x=0;x<=20;x++) {
for(y=0;y<=20;y++) {
r = new_region(x+100000,y);
r->planep = hell;
if(x==0 || x == 20 || y == 0 || y == 20) {
rsetterrain(r, T_FIREWALL);
} else {
rsetterrain(r, T_HELL);
}
}
}
/* Verbindungen schaffen */
a = create_special_direction(findregion(4,4),100005,5,-1,
"Ein dunkler Schlund fuehrt in die Tiefe der Erde",
"Schlund");
a = create_special_direction(findregion(100005,5),4,4,-1,
"Ein hell erleuchteter Tunnel fuehrt nach oben",
"Tunnel");
}
#endif
#if 0
static void
bename_dracoide(void)
{
region *r;
unit *u;
for(r=regions;r;r=r->next) {
for(u=r->units;u;u=u->next) {
if(u->race == new_race[RC_DRACOID]) name_unit(u);
2001-01-25 10:37:55 +01:00
}
}
}
#endif
static void
fix_migrants(void) {
region * r;
for (r=regions;r;r=r->next) {
unit * u;
for (u=r->units;u;u=u->next) {
if (u->race == new_race[RC_HUMAN]) u->irace = u->race;
if (u->irace!=u->race && (u->race->flags & RCF_SHAPESHIFT)==0) {
log_warning(("[fix_migrants] %s ist ein %s, als %s getarnt\n", unitname(u), LOC(default_locale, rc_name(u->race, 0)), LOC(default_locale, rc_name(u->irace, 0))));
2001-01-25 10:37:55 +01:00
}
}
}
}
static boolean
kor_teure_talente(unit *u)
{
if(effskill(u, SK_TACTICS) >= 1 ||
effskill(u, SK_MAGIC) >= 1 ||
effskill(u, SK_ALCHEMY) >= 1 ||
effskill(u, SK_HERBALISM) >= 1 ||
effskill(u, SK_SPY) >= 1) {
return true;
}
return false;
}
static void
no_teurefremde(boolean convert)
{
region *r;
unit *u;
for(r=regions;r;r=r->next) {
for(u=r->units;u;u=u->next) {
if(u->faction->no != MONSTER_FACTION
&& is_migrant(u)
2002-05-02 23:19:45 +02:00
&& kor_teure_talente(u))
{
2002-05-02 23:19:45 +02:00
log_printf("* Warnung, teurer Migrant: %s %s\n",
unitname(u), factionname(u->faction));
2001-01-25 10:37:55 +01:00
if(convert) {
u->race = u->faction->race;
u->irace = u->faction->race;
2001-01-25 10:37:55 +01:00
u->faction->num_migrants -= u->number;
sprintf(buf, "Die G<>tter segnen %s mit der richtigen Rasse",
unitname(u));
addmessage(0, u->faction, buf, MSG_MESSAGE, ML_IMPORTANT);
2001-01-25 10:37:55 +01:00
}
}
}
}
}
#if 0
static void
give_arena_gates(void)
{
faction * f;
for (f=factions;f;f=f->next) {
unit * u;
region * r;
unit * senior = f->units;
if (f->age>2) return;
for (u = f->units; u; u=u->nextF) {
if (senior->age < u->age) senior = u;
if (f->no==MONSTER_FACTION && get_item(u, I_ARENA_GATE)) return;
}
if (!senior) continue;
r = senior->region;
sprintf(buf, "Der Himmel <20>ber %s rei<65>t auf und ein Licht aus den Wolken scheint auf %s. Ein Gesang wie von tausend Walk<6C>ren ert<72>nt, dazu Schlachtenl<6E>rm. Nach einer Weile endet die Vision, und %s h<>lt ein %s in den H<>nden.", regionid(senior->region), unitname(senior), unitname(senior), itemdata[I_ARENA_GATE].name[0]);
change_item(senior, I_ARENA_GATE, 1);
addmessage(NULL, senior->faction, buf, MSG_MESSAGE, ML_IMPORTANT);
}
}
static void
remove_impossible_dragontargets(void)
{
region *r;
for (r=regions;r;r=r->next) {
unit *u;
for (u=r->units;u;u=u->next) {
attrib *a;
2001-01-25 10:37:55 +01:00
if(u->faction->no != MONSTER_FACTION) continue;
a = a_find(u->attribs, &at_targetregion);
2001-01-25 10:37:55 +01:00
if (a!=NULL) {
boolean cango = false;
region * r2 = a->data.v;
if(r2!=NULL) {
#ifdef NEW_PATH
cango = path_exists(r, r2, DRAGON_RANGE, allowed_dragon);
#else
cango = path_exists(r, r2, FLY|DRAGON_LIMIT);
#endif
}
if(!cango) {
printf("L<EFBFBD>sche Ziel von %s\n", unitname(u));
a_remove(&u->attribs, a);
}
}
}
}
}
static void
fix_score_option(void)
{
faction * f;
for (f=factions;f;f=f->next) {
f->options &= ((1<<O_SCORE)-1);
f->options &= ~(1<<O_MERIAN);
f->options |= (1<<O_SCORE);
}
}
#endif
#if 0
static void
fix_wounds(void)
{
region *r;
unit *u;
for(r=regions; r; r=r->next) {
for(u=r->units; u; u=u->next) {
int wounds = unit_max_hp(u)*u->number - u->hp;
u->hp = max(1, u->hp - wounds);
}
}
}
#endif
#if 0
static void
fix_beached(void)
{
region * r;
for (r=regions;r;r=r->next) {
if (r->land && rterrain(r)!=T_PLAIN) {
ship * s;
building *b;
for(b=r->buildings; b; b=b->next) {
if(b->typ == BT_HAFEN) break;
}
if(b == NULL) {
for (s=r->ships;s;s=s->next) {
if (s->coast!=NODIRECTION && s->type>SH_BOAT)
{
region * r2 = rconnect(r, s->coast);
assert(rterrain(r2)==T_OCEAN || fval(r2, RF_CHAOTIC));
move_ship(s, r, r2, NULL);
}
}
}
}
}
}
#endif
#if 0
static void
fix_hungrydead(void)
{
region *r;
unit *u;
for (r=regions;r;r=r->next) {
for (u=r->units;u;u=u->next) {
if(u->hp == 0) u->hp = 1;
}
}
}
static void
name_seaserpents(void)
{
region *r;
unit *u;
for (r=regions;r;r=r->next) {
for (u=r->units;u;u=u->next) {
if(u->race == new_race[RC_SEASERPENT] && strncmp(u->name, "Nummer ", 7)) {
2001-01-25 10:37:55 +01:00
set_string(&u->name, "Seeschlange");
}
}
}
}
#endif
2001-01-25 10:37:55 +01:00
attrib_type at_roadfix = {
"roadfix",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
#if RELEASE_VERSION<DISABLE_ROADFIX
DEFAULT_WRITE,
#else
NULL, /* disable writing them so they will disappear */
#endif
DEFAULT_READ,
ATF_UNIQUE
};
/* ************************************************************ */
/* GANZ WICHTIG! ALLE GE<47>NDERTEN SPR<50>CHE NEU ANZEIGEN */
/* GANZ WICHTIG! F<>GT AUCH NEUE ZAUBER IN DIE LISTE DER BEKANNTEN EIN */
/* ************************************************************ */
static void
show_newspells(void)
{
region *r;
/* Alle ge<67>nderten Zauber in das array newspellids[]. mit SPL_NOSPELL
* terminieren */
spellid_t newspellids[] = {
2003-02-28 19:15:29 +01:00
SPL_NOSPELL
};
2001-01-25 10:37:55 +01:00
/* die id's der neuen oder ver<65>nderten Spr<70>che werden in newspellids[]
* abgelegt */
for(r=regions; r; r=r->next) {
unit *u;
for(u=r->units;u;u=u->next) {
sc_mage *m = get_mage(u);
if (u->faction->no == MONSTER_FACTION) continue;
if (m != NULL) {
int i;
if (m->magietyp == M_GRAU) continue;
for (i = 0; newspellids[i] != SPL_NOSPELL; i++) {
spell *sp = find_spellbyid(newspellids[i]);
if (!sp) continue;
if (m->magietyp == sp->magietyp || getspell(u, sp->id)) {
attrib * a = a_find(u->faction->attribs, &at_reportspell);
while (a && a->data.i != sp->id) a = a->nexttype;
if (!a) {
/* spell is not being shown yet. if seen before, remove to show again */
a = a_find(u->faction->attribs, &at_seenspell);
while (a && a->data.i != sp->id) a = a->nexttype;
if (a) a_remove(&u->faction->attribs, a);
}
}
}
}
}
}
}
static int
fix_foreign(void)
{
region * r;
for (r=regions;r;r=r->next) {
struct locale * lang;
for (lang=locales;lang;lang=nextlocale(lang)) {
const char * udefault = LOC(lang, "unitdefault");
size_t udlen = strlen(udefault);
unit * u;
for (u=r->units;u;u=u->next) {
size_t unlen = strlen(u->name);
if (unlen<udlen) continue;
if (strncmp(u->name, udefault, udlen)==0) {
fset(u, FL_UNNAMED);
}
}
}
}
return 0;
}
#if 0
2001-01-25 10:37:55 +01:00
static void
fix_feuerwand_orks(void)
{
unit *u;
if((u=findunitg(atoi36("1j1L"), NULL))==NULL) {
printf("Korrektur: kann Orks nicht finden!\n");
return;
}
if(u->number < 1000) {
printf("Korrektur: Orks sind schon korrigiert!\n");
return;
}
scale_number(u,170);
u->hp=2337; /* Buggy Anzahl Personen = Normale HP */
}
static void
fix_buildings(void)
{
building *burg;
unit *u, *uf = NULL;
const requirement *req;
const construction *con;
FILE *statfile;
vset done;
int s1, s2;
char buf[256];
int a,b,c,d;
int first;
if((statfile=fopen("building.txt","r"))==NULL) {
log_warning(("fix_buildings: cannot open building.txt!\n"));
2001-01-25 10:37:55 +01:00
return;
}
vset_init(&done);
s1=s2=-1;
while(fgets(buf, 256, statfile)!=NULL) {
if(*buf=='#')
continue;
a=b=c=d=0;
first=0;
sscanf(buf,"%d;%d;%d;%d",&a,&b,&c,&d);
if((u=findunitg(b,NULL))==NULL) {
printf("fix_buildings: Einheit %d nicht gefunden.\n",b);
continue;
}
if((burg=findbuilding(a))==NULL) {
printf("fix_buildings: Burg %d nicht gefunden.\n", a);
if((burg=u->building)==NULL) {
printf("Einheit %d steht auch in keinem Geb<65>ude. (gebe auf)\n",u->no);
continue;
}
if(burg->type->construction->maxsize > 0 &&
burg->type->construction->reqsize > 0)
printf("Ersatz: Burg %d\n",burg->no);
else {
printf("Ersatzgeb<EFBFBD>ude ist nicht vom betroffenen Typ.\n");
continue;
}
}
printf("Burg %d; Einheit %d; vorher %d; + %d\n",burg->no,u->no,c,d);
s2=vset_add(&done, (void *)a);
if(s2>s1) {
printf("ERSTER !\n");
first=1;
uf=u;
}
s1=s2;
con=burg->type->construction;
if(con->maxsize != -1 || con->reqsize == 1) continue;
for(req=con->materials; req->number!=0; req++) {
int diff=burg->size-c;
if(req->type==R_SILVER)
change_resource(uf, R_SILVER, diff*req->number-d*req->number/con->reqsize);
else {
if(first)
change_resource(uf, req->type, diff*req->number);
assert(uf!=NULL);
change_resource(uf, req->type, -d*req->number/con->reqsize);
}
}
if(first)
burg->size=c+d;
else
burg->size+=d;
}
vset_destroy(&done);
fclose(statfile);
}
#endif
2001-01-25 10:37:55 +01:00
extern plane * arena;
static void
fix_age(void)
{
faction * f;
2002-03-24 13:43:12 +01:00
const race * oldorc = rc_find("orc");
const race * uruk = rc_find("uruk");
2001-01-25 10:37:55 +01:00
for (f=factions;f;f=f->next) {
if (f->no!=MONSTER_FACTION && playerrace(f->race)) continue;
2002-03-24 13:43:12 +01:00
if (f->race==oldorc) f->race= uruk;
else if (f->age!=turn) {
log_printf("Alter von Partei %s auf %d angepasst.\n", factionid(f), turn);
2001-01-25 10:37:55 +01:00
f->age = turn;
}
}
}
#if 0
static boolean
balsambug(const region *r)
{
direction_t dir;
for (dir=0; dir<MAXDIRECTIONS; dir++) {
if (landregion(rterrain(rconnect(r, dir))) && rdemand(r,0) > 0)
return false;
}
return true;
}
static void
fix_balsamfiasko(void)
{
region *r, *rc;
direction_t dir;
int i;
for (r=regions; r; r=r->next) freset(r, RF_DH);
/* Zuf<75>llig verteilen */
for (r=regions; r; r=r->next) {
if (r->x < -36 && r->y < -45 && landregion(rterrain(r))) {
if(rdemand(r,0) <= 0 && balsambug(r)) {
rsetdemand(r,0,(char)(1+rand()%5));
rsetdemand(r,rand()%7,0);
fset(r, RF_DH);
}
}
}
/* Smoothing */
for(i=0; i<3; i++) {
for (r=regions; r; r=r->next) if(fval(r, RF_DH)) {
for (dir=0; dir < MAXDIRECTIONS; dir++) {
rc = rconnect(r, dir);
if(rc && fval(rc, RF_DH) && rand()%100 < 20) {
int p1, p2;
for (p1 = 0; p1 != MAXLUXURIES; p1++)
if (rdemand(r, p1) == 0) break;
for (p2 = 0; p2 != MAXLUXURIES; p2++)
if (rdemand(rc, p2) == 0) break;
if(p1 != p2) {
rsetdemand(rc, p2, (char)(1+rand()%5));
rsetdemand(rc, p1, 0);
}
}
}
}
}
}
#endif
#if 1
2001-02-10 14:20:09 +01:00
static int
count_demand(const region *r)
{
struct demand *dmd;
int c = 0;
if (r->land) {
for (dmd=r->land->demands;dmd;dmd=dmd->next) c++;
}
return c;
}
2001-02-20 23:54:05 +01:00
#endif
static int
recurse_regions(region * r, regionlist **rlist, boolean(*fun)(const region * r))
{
if (!fun(r)) return 0;
else {
int len = 0;
direction_t d;
regionlist * rl = calloc(sizeof(regionlist), 1);
rl->next = *rlist;
rl->region = r;
(*rlist) = rl;
fset(r, FL_MARK);
for (d=0;d!=MAXDIRECTIONS;++d) {
region * nr = rconnect(r, d);
if (nr && !fval(nr, FL_MARK)) len += recurse_regions(nr, rlist, fun);
}
return len+1;
}
}
#if 1
static int maxluxuries = 0;
static boolean
f_nolux(const region * r)
{
if (r->land && count_demand(r) != maxluxuries) return true;
return false;
}
static int
fix_demand_region(region *r)
{
regionlist *rl, *rlist = NULL;
static const luxury_type **mlux = 0, ** ltypes;
const luxury_type *sale = NULL;
int maxlux = 0;
recurse_regions(r, &rlist, f_nolux);
if (mlux==0) {
int i = 0;
if (maxluxuries==0) for (sale=luxurytypes;sale;sale=sale->next) {
maxluxuries++;
}
mlux = (const luxury_type **)gc_add(calloc(maxluxuries, sizeof(const luxury_type *)));
ltypes = (const luxury_type **)gc_add(calloc(maxluxuries, sizeof(const luxury_type *)));
for (sale=luxurytypes;sale;sale=sale->next) {
ltypes[i++] = sale;
}
}
else {
int i;
for (i=0;i!=maxluxuries;++i) mlux[i] = 0;
}
for (rl=rlist;rl;rl=rl->next) {
region * r = rl->region;
direction_t d;
for (d=0;d!=MAXDIRECTIONS;++d) {
region * nr = rconnect(r, d);
if (nr && nr->land && nr->land->demands) {
struct demand * dmd;
for (dmd = nr->land->demands;dmd;dmd=dmd->next) {
if (dmd->value == 0) {
int i;
for (i=0;i!=maxluxuries;++i) {
if (mlux[i]==NULL) {
maxlux = i;
mlux[i] = dmd->type;
break;
} else if (mlux[i]==dmd->type) {
break;
}
}
break;
}
}
}
}
freset(r, FL_MARK); /* undo recursive marker */
}
if (maxlux<2) {
int i;
for (i=maxlux;i!=2;++i) {
int j;
do {
int k = rand() % maxluxuries;
mlux[i] = ltypes[k];
for (j=0;j!=i;++j) {
if (mlux[j]==mlux[i]) break;
}
} while (j!=i);
}
maxlux=2;
}
for (rl=rlist;rl;rl=rl->next) {
region * r = rl->region;
log_warning(("fixing demand in %s\n", regionname(r, NULL)));
setluxuries(r, mlux[rand() % maxlux]);
}
while (rlist) {
rl = rlist->next;
free(rlist);
rlist = rl;
}
return 0;
}
2001-02-20 23:54:05 +01:00
#endif
static void
fix_firewalls(void)
{
region * r = regions;
while (r) {
direction_t d;
for (d=0;d!=MAXDIRECTIONS;++d) {
region * r2 = rconnect(r, d);
if (r2) {
border * b = get_borders(r, r2);
while (b) {
if (b->type==&bt_firewall) {
attrib * a = a_find(b->attribs, &at_countdown);
if (a==NULL || a->data.i <= 0) {
erase_border(b);
log_warning(("firewall between regions %s and %s was bugged. removed.\n",
regionid(r), regionid(r2)));
b = get_borders(r, r2);
} else {
b = b->next;
}
} else {
b = b->next;
}
}
}
}
r = r->next;
}
}
extern attrib * make_atgmcreate(const struct item_type * itype);
extern attrib * make_atpermissions(void);
extern struct attrib_type at_permissions;
extern struct attrib_type at_gmcreate;
static void
2001-05-11 22:19:22 +02:00
update_gms(void)
{
2001-05-11 22:19:22 +02:00
faction * f;
for (f=factions;f;f=f->next) {
attrib * permissions = a_find(f->attribs, &at_permissions);
if (permissions) {
2002-02-24 10:20:38 +01:00
const char * keys[] = { "gmgate", "gmmsgr", "gmkill", "gmmsgu", NULL };
int k;
item_t i;
for (k=0;keys[k];++k) {
add_key((attrib**)&permissions->data.v, atoi36(keys[k]));
}
for (i=I_GREATSWORD;i!=I_KEKS;++i) {
attrib * a = a_find((attrib*)permissions->data.v, &at_gmcreate);
while (a && a->data.v!=(void*)olditemtype[i]) a=a->nexttype;
if (!a) a_add((attrib**)&permissions->data.v, make_atgmcreate(olditemtype[i]));
}
}
}
}
#if 1
static int
fix_demand(void)
{
region *r;
const luxury_type *sale = NULL;
if (maxluxuries==0) for (sale=luxurytypes;sale;sale=sale->next) ++maxluxuries;
for (r=regions; r; r=r->next) {
if ((r->land) && count_demand(r) != maxluxuries) {
fix_demand_region(r);
}
}
return 0;
}
2001-02-20 23:54:05 +01:00
#endif
2001-01-25 10:37:55 +01:00
#if 0
static void
read_laenrepair(boolean active)
{
FILE * f = fopen("repair.laen", "rt");
FILE * log = fopen("report.laen", "wt");
if (!f || !log) return;
while (!feof(f)) {
int n, x, y;
region * r;
fscanf(f, "%d %d %d", &x, &y, &n);
r = findregion(x, y);
if (!r) fprintf(log, "%d,%d:\tnot found\n", x, y);
else if (fval(r, RF_CHAOTIC)) continue;
else {
int laen = rlaen(r);
if (n==-1 && laen>=0) {
fprintf(log, "%d,%d:\tillegal laen (%d)\n", x, y, laen);
if (active) rsetlaen(r, n);
} else if (laen==-1 && n!=-1) {
fprintf(log, "%d,%d:\tlaen was lost (%d)\n", x, y, n);
if (active) rsetlaen(r, n);
}
}
}
fclose(f);
}
static void
write_laenrepair(void) {
FILE * f = fopen("repair.laen", "wt");
region * r;
for (r=regions;r;r=r->next) if (!fval(r, RF_CHAOTIC)) {
int laen = rlaen(r);
fprintf(f, "%d %d %d\n", r->x, r->y, laen);
}
fclose(f);
}
#endif
#include "group.h"
static void
2002-10-08 08:48:47 +02:00
fix_allies(void)
2002-09-02 22:36:12 +02:00
{
2001-01-25 10:37:55 +01:00
faction * f;
for (f=factions;f;f=f->next) {
group * g;
for (g=f->groups;g;g=g->next) {
ally ** ap=&g->allies;
while (*ap) {
ally * an, * a = *ap;
for (an = a->next;an;an=an->next) {
if (a->faction==an->faction) {
*ap = a->next;
free(a);
break;
}
}
if (an==NULL) ap = &(*ap)->next;
}
}
}
}
#ifdef FUZZY_BASE36
extern boolean enable_fuzzy;
#endif
2001-02-10 14:20:09 +01:00
static void
2001-01-25 10:37:55 +01:00
fix_icastles(void)
{
region * r;
for (r=regions; r; r=r->next) {
building * b;
for (b=r->buildings; b; b=b->next) {
attrib * a;
const building_type * btype = bt_find("castle");
2001-01-25 10:37:55 +01:00
icastle_data * data;
a = a_find(b->attribs, &at_icastle);
if (b->type!=bt_find("illusion") && !a) continue;
2001-01-25 10:37:55 +01:00
if (!a) {
/* attribut hat gefehle */
a = a_add(&b->attribs, a_new(&at_icastle));
}
if (b->type!=bt_find("illusion")) {
2001-01-25 10:37:55 +01:00
/* geb<65>udetyp war falsch */
btype = b->type;
b->type = bt_find("illusion");
2001-01-25 10:37:55 +01:00
}
data = (icastle_data*)a->data.v;
if (data->time<=0) {
/* zeit war kaputt oder abgelaufen */
data->time = 1;
}
if (!data->type) {
/* typ muss gesetzt werden, weil er nicht geladen wurde */
data->type = btype;
}
if (data->building!=b) {
/* r<>ckw<6B>rtszeiger auf das geb<65>ude reparieren */
data->building=b;
}
}
}
}
#if 0
static void
fix_ponnukis(void)
/* Es kann nur einen geben! */
{
unit * ponn = findunit(atoi36("ponn"));
if (ponn) {
region * mag = ponn->region;
unit * u;
for (u=mag->units;u;u=u->next) {
if (u!=ponn && u->faction==ponn->faction && u->race==ponn->race && !strcmp(ponn->name, u->name)) {
set_number(u, 0);
}
}
}
}
#endif
#if 0
static void
fix_traveldir(void)
{
region *r;
attrib *a, *a2;
traveldir *t;
for(r=regions; r; r=r->next) {
a = a_find(r->attribs, &at_traveldir);
while (a) {
a2 = a_add(&r->attribs, a_new(&at_traveldir_new));
t = (traveldir *)(a2->data.v);
t->no = a->data.sa[0];
t->dir = a->data.ca[2];
t->age = a->data.ca[3];
a = a->nexttype;
}
}
for(r=regions; r; r=r->next) {
a = a_find(r->attribs, &at_traveldir);
while(a) {
a_remove(&r->attribs, a);
a = a->nexttype;
}
}
}
#endif
typedef struct stats_t {
struct stats_t * next;
const struct item_type * type;
double number;
} stats_t;
static void
s_change(stats_t** s, const struct item_type * type, int count)
{
while (*s && (*s)->type!=type) s=&(*s)->next;
2003-09-21 11:45:25 +02:00
if (*s==NULL) {
*s = calloc(1, sizeof(stats_t));
(*s)->type = type;
}
(*s)->number += count;
}
static stats_t *
s_find(stats_t * s, const struct item_type * type)
{
while (s && s->type!=type) s=s->next;
return s;
}
2001-01-25 10:37:55 +01:00
static void
stats(void)
{
FILE * F;
stats_t * items = NULL;
char zText[MAX_PATH];
2001-01-25 10:37:55 +01:00
strcat(strcpy(zText, resourcepath()), "/stats");
F = fopen(zText, "wt");
2001-01-25 10:37:55 +01:00
if (F) {
region * r;
const item_type * itype;
for (r=regions;r;r=r->next) {
unit * u;
item * itm;
for (u=r->units;u;u=u->next) {
for (itm=u->items;itm;itm=itm->next) {
if (itm->number>10000000) {
log_error(("unit %s has %d %s\n", unitname(u), itm->number, resourcename(itm->type->rtype, 0)));
2003-09-28 10:05:50 +02:00
/* itm->number=1; */
}
s_change(&items, itm->type, itm->number);
2001-01-25 10:37:55 +01:00
}
}
}
for (itype=itemtypes;itype;itype=itype->next) {
stats_t * itm = s_find(items, itype);
if (itm && itm->number>0.0)
fprintf(F, "%4.0f %s\n", itm->number, locale_string(NULL, resourcename(itype->rtype, 0)));
2001-01-25 10:37:55 +01:00
else
fprintf(F, "%4.0f %s\n", 0.0, locale_string(NULL, resourcename(itype->rtype, 0)));
2001-01-25 10:37:55 +01:00
}
fclose(F);
} else {
perror(zText);
2001-01-25 10:37:55 +01:00
}
}
#if 0
2001-01-25 10:37:55 +01:00
static void
fix_prices(void)
{
region *r;
puts(" - Korrigiere Handelsg<73>terpreise");
for(r=regions; r; r=r->next) if(r->land) {
int sales = 0, buys = 0;
const luxury_type *sale = NULL, *ltype;
struct demand *dmd;
for (dmd=r->land->demands;dmd;dmd=dmd->next) {
if(dmd->value == 0) {
sales++;
sale = dmd->type;
} else {
buys++;
}
}
if(sales == 0) {
int c = 0;
for(ltype=luxurytypes; ltype; ltype=ltype->next) c++;
c = rand()%c;
for(ltype=luxurytypes; c>0; c--) ltype=ltype->next;
r_setdemand(r, ltype, 0);
sale = ltype;
}
assert(sale);
if(buys == 0) {
for(ltype=luxurytypes; ltype; ltype=ltype->next) {
if(ltype != sale) r_setdemand(r, ltype, 1 + rand() % 5);
2001-01-25 10:37:55 +01:00
}
}
}
}
static void
fix_options(void)
{
faction *f;
puts(" - Korrigiere Parteioptionen");
for(f=factions;f;f=f->next) {
f->options = f->options | Pow(O_REPORT);
f->options = f->options | Pow(O_ZUGVORLAGE);
}
}
static void
init_region_age(void)
{
region *r;
puts(" - Initialisiere r->age");
/* Initialisieren mit dem Alter der <20>ltesten Partei in 2 Feldern
* Umkreis */
for(r=regions; r; r=r->next) {
regionlist *rl = all_in_range(r,2);
regionlist *rl2;
unit *u;
int maxage = 0;
for(rl2 = rl; rl2; rl2 = rl2->next) {
for(u=rl2->region->units; u; u=u->next) {
if(u->faction->age > maxage) maxage = u->faction->age;
}
}
r->age = (unsigned short)maxage;
}
}
static void
fix_herbtypes(void)
{
region * r;
for (r=regions;r!=NULL;r=r->next) if (r->land) {
const herb_type * htype = r->land->herbtype;
if (htype==NULL) {
if (terrain[r->terrain].herbs!=NULL) {
const item_type * itype;
int i=0;
while (terrain[r->terrain].herbs[i]!=NULL) ++i;
itype = finditemtype(terrain[r->terrain].herbs[rand()%i], NULL);
r->land->herbtype = resource2herb(itype->rtype);
}
} else {
if (r->terrain==T_PLAIN) {
if (r_isforest(r)) {
int i = rand()%3+3;
const char * name = terrain[T_PLAIN].herbs[i];
const item_type * itype = finditemtype(name, NULL);
r->land->herbtype = resource2herb(itype->rtype);
}
} else {
/* Das geht so nicht, wegen der Eisberg-Terrains */
/* assert(htype->terrain==r->terrain); */
}
}
}
}
#endif
#ifdef SKILLFIX_SAVE
typedef struct skillfix_data {
unit * u;
skill_t skill;
int value;
int number;
int self;
int teach;
} skillfix_data;
static void
init_skillfix(attrib * a)
{
a->data.v = calloc(sizeof(skillfix_data), 1);
}
static void
free_skillfix(attrib * a)
{
free(a->data.v);
}
attrib_type at_skillfix = { "skillfix", init_skillfix, free_skillfix };
void
2001-01-25 10:37:55 +01:00
skillfix(struct unit * u, skill_t skill, int from, int self, int teach)
{
attrib * a = a_add(&u->attribs, a_new(&at_skillfix));
skillfix_data * data = (skillfix_data*)a->data.v;
data->u = u;
data->skill = skill;
data->number = u->number;
data->value = from;
data->self = self;
data->teach = teach;
}
void
write_skillfix(void)
{
FILE * F = fopen("skills.fix", "w+");
region * r;
fprintf(F, "S%d\n", turn);
for (r=regions;r;r=r->next) {
unit * u;
for (u=r->units;u;u=u->next) {
attrib * a = a_find(u->attribs, &at_skillfix);
while (a) {
skillfix_data * data = (skillfix_data*)a->data.v;
fprintf(F, "%s %d %d %d %d %d\n",
2001-01-25 10:37:55 +01:00
itoa36(data->u->no),
data->skill,
data->number,
data->value,
data->self,
data->teach);
a = a->nexttype;
}
}
}
fclose(F);
}
#endif
#if 0
2001-01-25 10:37:55 +01:00
static void
init_mwarden(void)
{
unit *u = findunitg(atoi36("mwar"), NULL);
attrib *a;
if(!u) return;
a = a_add(&u->attribs, a_new(&at_warden));
a->data.i = 0;
}
#endif
2001-01-25 10:37:55 +01:00
2001-02-20 23:54:05 +01:00
#if 0
#include <items/lmsreward.h>
2001-01-25 10:37:55 +01:00
static void
lms_special(unit * u)
{
if (u) i_change(&u->items, &it_lmsreward, 1);
}
2001-02-20 23:54:05 +01:00
#endif
2001-02-20 23:54:05 +01:00
#if 0
#define LIFEEXPECTANCY (27*40)
static void
undo_deadpeasants(void)
{
region * r = regions;
while (r) {
int dead = rpeasants(r) / LIFEEXPECTANCY;
deathcounts(r, -dead);
r = r->next;
}
2001-01-25 10:37:55 +01:00
}
static void
fix_undead3percent(void)
{
region * r = regions;
while (r) {
int dead = deathcount(r);
deathcounts(r, -(dead/2));
r = r->next;
}
}
static void
fix_targetregion_resolve(void)
{
region *r;
unit *u;
attrib *a;
for(r=regions; r; r=r->next) {
for(u=r->units; u; u=u->next) {
a = a_find(u->attribs, &at_targetregion);
if(a) a->data.v = findregion(a->data.sa[0], a->data.sa[1]);
}
}
}
static void
fix_herbs(void)
{
const char * plain_herbs[] = {"Flachwurz", "W<EFBFBD>rziger Wagemut", "Eulenauge", "Gr<EFBFBD>ner Spinnerich", "Blauer Baumringel", "Elfenlieb"};
const herb_type * htypes[6];
int herbs[6];
int hneed[6];
region *r;
int i, hsum = 0, left = 0;
for (i=0;i!=6;++i) {
htypes[i] = resource2herb(finditemtype(plain_herbs[i], NULL)->rtype);
herbs[i] = 0;
}
for (r=regions; r; r=r->next) if (rterrain(r) == T_PLAIN) {
const herb_type *htype = rherbtype(r);
if (htype==NULL) {
htype = htypes[i];
rsetherbtype(r, htype);
}
for (i=0;i!=6;++i) if (htypes[i]==htype) break;
assert(i!=6);
herbs[i]++;
hsum++;
}
for (i=0;i!=6;++i) {
int hwant = hsum / (6-i);
hneed[i] = hwant - herbs[i];
if (hneed[i]>0) left += hneed[i];
hsum -= hwant;
}
for (r=regions; r; r=r->next) if (rterrain(r) == T_PLAIN) {
const herb_type *htype = rherbtype(r);
assert(htype);
for (i=0;i!=6;++i) if (htypes[i]==htype) break;
assert(i!=6);
if (hneed[i]<0) {
int p;
int k = rand() % left;
for (p=0;p!=6;++p) if (hneed[p]>0) {
k-=hneed[p];
if (k<0) break;
}
assert(p!=6);
hneed[p]--;
hneed[i]++;
left--;
rsetherbtype(r, htypes[p]);
}
hsum++;
}
for (i=0;i!=6;++i) herbs[i] = 0;
for (r=regions; r; r=r->next) if (rterrain(r) == T_PLAIN) {
const herb_type *htype = rherbtype(r);
assert(htype);
for (i=0;i!=6;++i) if (htypes[i]==htype) break;
assert(i!=6);
herbs[i]++;
}
for (i=0;i!=6;++i) {
fprintf(stderr, "%s : %d\n", locale_string(NULL, resourcename(htypes[i]->itype->rtype, 0)), herbs[i]);
}
}
#endif
#include <event.h>
#include <triggers/timeout.h>
#include <triggers/changerace.h>
#include <triggers/changefaction.h>
#include <triggers/createcurse.h>
#include <triggers/createunit.h>
#include <triggers/killunit.h>
#include <triggers/giveitem.h>
typedef struct handler_info {
char * event;
trigger * triggers;
} handler_info;
typedef struct timeout_data {
trigger * triggers;
int timer;
variant trigger_data;
} timeout_data;
trigger *
get_timeout(trigger * td, trigger * tfind)
{
trigger * t = td;
while (t) {
if (t->type==&tt_timeout) {
timeout_data * tdata = (timeout_data *)t->data.v;
trigger * tr = tdata->triggers;
while (tr) {
if (tr==tfind) break;
tr=tr->next;
}
if (tr==tfind) break;
}
t=t->next;
}
return t;
}
#if 0
static void
fix_timeouts(void)
{
region * r;
for (r=regions;r;r=r->next) {
unit * u;
for (u=r->units;u;u=u->next) {
attrib * a = a_find(u->attribs, &at_eventhandler);
boolean toad = (u->race==RC_TOAD);
boolean keeper = (u->race==RC_IRONKEEPER);
while (a!=NULL) {
trigger * t;
handler_info * td = (handler_info *)a->data.v;
for (t=td->triggers;t;t=t->next) {
trigger ** tptr=&t->next;
while (*tptr) {
/* remove duplicates */
if ((*tptr)->type==t->type) {
*tptr = (*tptr)->next;
} else tptr = &(*tptr)->next;
}
if (t->type == &tt_changerace ||
t->type == &tt_changefaction ||
t->type == &tt_createcurse ||
t->type == &tt_createunit)
{
trigger * timer = get_timeout(td->triggers, t);
if (toad && t->type == &tt_changerace) {
toad = false;
}
if (timer==NULL) {
add_trigger(&u->attribs, "timer", trigger_timeout(1+(rand()%2), t));
}
}
else if (t->type == &tt_killunit) {
if (u->race==RC_IRONKEEPER) {
trigger * timer = get_timeout(td->triggers, t);
keeper = false;
if (timer==NULL) {
add_trigger(&u->attribs, "timer", trigger_timeout(1+(rand()%2), t));
}
}
}
}
a = a->nexttype;
}
if (keeper) {
int duration = 1+ (rand() % 2);
trigger * tkill = trigger_killunit(u);
add_trigger(&u->attribs, "timer", trigger_timeout(duration, tkill));
}
if (toad) {
/* repair toad-only mage */
int duration = 1+ (rand() % 2);
trigger * trestore = trigger_changerace(u, u->faction->race, u->faction->race);
if (rand()%10>2) t_add(&trestore, trigger_giveitem(u, olditemtype[I_TOADSLIME], 1));
add_trigger(&u->attribs, "timer", trigger_timeout(duration, trestore));
}
}
}
}
#endif
#include <modules/gmcmd.h>
static int
2002-03-12 23:40:23 +01:00
resize_plane(struct plane * p, int radius)
{
region * center = findregion(p->minx+(p->maxx-p->minx)/2, p->miny+(p->maxy-p->miny)/2);
2002-03-24 13:43:12 +01:00
int x, y, minx, maxx, miny, maxy;
if (!center) return 0;
minx = center->x - radius;
maxx = center->x + radius;
miny = center->y - radius;
maxy = center->y + radius;
2002-03-12 23:40:23 +01:00
for (x=minx;x<=maxx;++x) {
for (y=miny;y<=maxy;++y) {
region * r = findregion(x, y);
if (r && rplane(r)!=p) break;
}
}
if (x<=maxx) return -1;
p->maxx=maxx;
p->maxy=maxy;
p->minx=minx;
p->miny=miny;
for (x=minx;x<=maxx;++x) {
for (y=miny;y<=maxy;++y) {
region * r = findregion(x, y);
if (r==NULL) {
r = new_region(x, y);
2002-03-12 23:40:23 +01:00
freset(r, RF_ENCOUNTER);
r->planep = p;
if (distance(r, center)==radius) {
terraform(r, T_FIREWALL);
} else if (distance(r, center)<radius) {
terraform(r, T_OCEAN);
}
2002-04-14 10:22:35 +02:00
} else if (rterrain(r)==T_FIREWALL && distance(r, center)!=radius) {
terraform(r, T_OCEAN);
2002-03-12 23:40:23 +01:00
}
}
}
return 0;
}
static int
secondfaction(faction * pf)
{
unit * u = findunit(atoi36("5q9w"));
if (u!=NULL) {
plane * p = rplane(u->region);
if (p!=NULL) {
region * center = findregion((p->maxx-p->minx)/2, (p->maxy-p->miny)/2);
if (center!=NULL) {
gm_addfaction(pf->email, p, center);
return 0;
}
}
}
return -1;
}
static void
update_gmquests(void)
{
faction * f = findfaction(atoi36("gm04"));
if (f) {
unit * u = f->units;
potion_t p;
attrib * permissions = a_find(f->attribs, &at_permissions);
if (u!=NULL) {
plane * p = rplane(u->region);
/* gm04 will keine Orks */
if (p!=NULL) p->flags |= PFL_NOORCGROWTH;
/* gm04 will keine Monster */
if (p!=NULL) p->flags |= PFL_NOMONSTERS;
}
for (p=0;p!=MAX_POTIONS;++p) {
attrib * a = a_find((attrib*)permissions->data.v, &at_gmcreate);
while (a && a->data.v!=(void*)oldpotiontype[p]->itype) a=a->nexttype;
if (!a) a_add((attrib**)&permissions->data.v, make_atgmcreate(oldpotiontype[p]->itype));
}
do_once("et02", secondfaction(f));
}
do_once("renm", fix_foreign());
}
#if 0
static void
test_gmquest(void)
{
const struct faction * f;
/* enno's world */
f = gm_addquest("enno@eressea.upb.de", "GM Zone", 1, PFL_NOATTACK|PFL_NOALLIANCES|PFL_NOFEED|PFL_FRIENDLY);
log_printf("Neue Questenpartei %s\n", factionname(f));
f = gm_addquest("xandril@att.net", "Mardallas Welt", 40, 0);
log_printf("Neue Questenpartei %s\n", factionname(f));
f = gm_addquest("moritzsalinger@web.de", "Laen-Kaiser", 7, /*PFL_NORECRUITS |*/ PFL_NOMAGIC /*| PFL_NOBUILD*/);
log_printf("Neue Questenpartei %s\n", factionname(f));
f = gm_addquest("Denise.Muenstermann@home.gelsen-net.de", "Mochikas Queste", 7, PFL_NOMAGIC);
log_printf("Neue Questenpartei %s\n", factionname(f));
f = gm_addquest("feeron@aol.com", "Eternath", 11, 0);
log_printf("Neue Questenpartei %s\n", factionname(f));
f = gm_addquest("BigBear@nord-com.net", "Leonidas Verm<72>chtnis", 15, PFL_NOMAGIC|PFL_NOSTEALTH);
log_printf("Neue Questenpartei %s\n", factionname(f));
}
#endif
#define TEST_LOCALES 0
#if TEST_LOCALES
static void
setup_locales(void)
{
locale * lang = find_locale("en");
faction * f = factions;
while (f) {
f->locale = lang;
f = f->next;
}
}
#endif
#include <triggers/shock.h>
#include <triggers/killunit.h>
#if 0
#error "this is broken, it duplicates triggers"
static void
fix_unitrefs(void)
{
region * r=regions;
while (r) {
unit * u = r->units;
while (u) {
attrib * a;
a = a_find(u->attribs, &at_familiar);
if (a) {
/* magier, hat einen familiar */
unit * ufamiliar = get_familiar(u);
if (ufamiliar) {
attrib * ae;
/* killunit attribut am magier */
ae = a_find(u->attribs, &at_eventhandler);
if (ae) {
trigger * tkillunit = NULL;
trigger ** tlist = get_triggers(u->attribs, "destroy");
if (tlist!=NULL) {
tkillunit = *tlist;
while (tkillunit) {
if (strcmp(tkillunit->type->name, "killunit")==0) break;
tkillunit = tkillunit->next;
}
if (tkillunit && !tkillunit->data.v) {
log_warning(("killunit-trigger f<>r Magier %s und Vertrauten %s restauriert.\n",
itoa36(u->no), itoa36(ufamiliar->no)));
tkillunit->data.v = ufamiliar;
}
else ae = NULL;
}
}
if (ae==NULL) {
/* Wenn der Magier stirbt, dann auch der Vertraute */
add_trigger(&u->attribs, "destroy", trigger_killunit(ufamiliar));
}
/* killunit attribut am magier */
ae = a_find(ufamiliar->attribs, &at_eventhandler);
if (ae) {
trigger * tshockunit = NULL;
trigger ** tlist = get_triggers(ufamiliar->attribs, "destroy");
if (tlist!=NULL) {
tshockunit = *tlist;
while (tshockunit) {
if (strcmp(tshockunit->type->name, "shock")==0) break;
tshockunit = tshockunit->next;
}
if (tshockunit && !tshockunit->data.v) {
log_warning(("shockunit-trigger f<>r Magier %s und Vertrauten %s restauriert.\n",
itoa36(u->no), itoa36(ufamiliar->no)));
tshockunit->data.v = u;
}
else ae = NULL;
}
}
if (ae==NULL) {
/* Wenn der Vertraute stirbt, schockt er den Magier */
add_trigger(&ufamiliar->attribs, "destroy", trigger_shock(u));
}
} else {
log_error(("Magier %s hat ein at_familiar, aber keinen Vertrauten.\n",
itoa36(u->no)));
}
}
u = u->next;
}
r=r->next;
}
}
#endif
static void
update_igjarjuk_quest(void)
{
unit * u;
faction *f = findfaction(atoi36("rr"));
if (!f) return;
for (u=f->units;u;u=u->nextF) {
u->race = new_race[RC_TEMPLATE];
}
}
#if RESOURCE_CONVERSION
extern struct attrib_type at_resources;
void
init_resourcefix(void)
{
at_register(&at_resources);
}
#endif
#if 0
static void
fix_negpotion(void)
{
region *r;
unit *u;
int i;
for(r=regions; r; r=r->next) {
for(u=r->units; u; u=u->next) {
i = get_effect(u, oldpotiontype[P_FAST]);
if (i < 0){
change_effect(u, oldpotiontype[P_FAST], -i);
}
}
}
}
#endif
#if GROWING_TREES
int
growing_trees(void)
{
region *r;
for(r=regions; r; r=r->next) {
if(rtrees(r, 2)) {
rsettrees(r, 1, rtrees(r, 2)/4);
rsettrees(r, 0, rtrees(r, 2)/2);
}
}
return 0;
}
#endif
#include <triggers/gate.h>
#include <triggers/unguard.h>
typedef struct gate_data {
struct building * gate;
struct region * target;
} gate_data;
static void
fix_gates(void)
{
region * r;
for (r=regions;r;r=r->next) {
unit * u;
building * b;
for (u=r->units;u;u=u->next) {
trigger ** triggers = get_triggers(u->attribs, "timer");
if (triggers) {
trigger * t = *triggers;
while (t && t->type!= &tt_gate) t=t->next;
if (t!=NULL) {
gate_data * gd = (gate_data*)t->data.v;
struct building * b = gd->gate;
struct region * rtarget = gd->target;
if (r!=b->region) {
add_trigger(&b->attribs, "timer", trigger_gate(b, rtarget));
add_trigger(&b->attribs, "create", trigger_unguard(b));
fset(b, BLD_UNGUARDED);
}
}
remove_triggers(&u->attribs, "timer", &tt_gate);
remove_triggers(&u->attribs, "create", &tt_unguard);
}
}
for (b=r->buildings;b;b=b->next) {
trigger ** triggers = get_triggers(b->attribs, "timer");
if (triggers) {
trigger * t = *triggers;
while (t && t->type!= &tt_gate) t=t->next;
if (t!=NULL) {
gate_data * gd = (gate_data*)t->data.v;
struct building * b = gd->gate;
struct region * rtarget = gd->target;
remove_triggers(&b->attribs, "timer", &tt_gate);
remove_triggers(&b->attribs, "create", &tt_unguard);
if (r!=b->region) {
add_trigger(&b->attribs, "timer", trigger_gate(b, rtarget));
add_trigger(&b->attribs, "create", trigger_unguard(b));
fset(b, BLD_UNGUARDED);
}
}
}
}
}
}
static int
fix_astralplane(void)
{
plane * astralplane = getplanebyname("Astralraum");
region * r;
region_list * rlist = NULL;
if (astralplane) {
freset(astralplane, PFL_NOCOORDS);
freset(astralplane, PFL_NOFEED);
set_ursprung(findfaction(MONSTER_FACTION), astralplane->id, 0, 0);
}
for (r=regions;r;r=r->next) {
region * ra = r_standard_to_astral(r);
if (ra==NULL) continue;
if (r->terrain!=T_FIREWALL) continue;
if (ra->terrain==T_ASTRALB) continue;
if (ra->units!=NULL) {
region_list * rnew = malloc(sizeof(region_list));
rnew->data = ra;
rnew->next = rlist;
rlist = rnew;
}
terraform(ra, T_ASTRALB);
}
while (rlist) {
region_list * rnew = rlist;
region * r = rnew->data;
direction_t dir;
rlist = rlist->next;
for (dir=0;dir!=MAXDIRECTIONS;++dir) {
region * rnext = rconnect(r, dir);
if (rnext==NULL) continue;
if (rnext->terrain!=T_ASTRAL) continue;
while (r->units) {
unit * u = r->units;
move_unit(u, rnext, NULL);
if (u->faction->no==MONSTER_FACTION) {
set_number(u, 0);
}
}
break;
}
if (r->units!=NULL) {
unit * u;
for (u=r->units;u;u=u->next) {
if (u->faction->no!=MONSTER_FACTION) {
log_error(("Einheit %s in %u, %u steckt im Nebel fest.\n",
unitname(u), r->x, r->y));
}
}
}
free(rnew);
}
return 0;
}
static int
warn_password(void)
{
faction * f = factions;
while (f) {
boolean pwok = true;
const char * c = f->passw;
while (*c) {
if (!isalnum(*c)) pwok = false;
c++;
}
if (pwok == false) {
2002-05-02 23:19:45 +02:00
ADDMSG(&f->msgs, msg_message("msg_errors", "string",
"Dein Passwort enth<74>lt Zeichen, die bei der Nachsendung "
"von Reports Probleme bereiten k<>nnen. Bitte w<>hle ein neues "
2002-03-10 11:09:16 +01:00
"Passwort, bevorzugt nur aus Buchstaben und Zahlen bestehend."));
}
f = f->next;
}
return 0;
}
2002-04-28 17:37:48 +02:00
extern border *borders[];
static void
fix_road_borders(void)
{
border *delete[10000];
int hash;
int i = 0;
for(hash=0; hash<BMAXHASH; hash++) {
border * b;
for (b=borders[hash];b;b=b->next) {
if(b->type == &bt_road) {
int x1, x2, y1, y2;
region *r1, *r2;
x1 = b->from->x;
y1 = b->from->y;
x2 = b->to->x;
y2 = b->to->y;
r1 = findregion(x1, y1);
r2 = findregion(x2, y2);
if(r1->land == NULL || r2->land == NULL
|| terrain[r1->terrain].roadreq == 0
|| terrain[r2->terrain].roadreq == 0) {
delete[i] = b;
i++;
}
}
}
}
while(i>0) {
i--;
erase_border(delete[i]);
}
}
2003-12-12 18:17:13 +01:00
#ifdef WDW_PHOENIX
static region *
random_land_region(void)
2003-12-12 18:17:13 +01:00
{
region *r;
int c = 0;
/* count the available regions */
for(r=regions; r; r=r->next) {
if (r->land && rplane(r)==NULL) ++c;
2003-12-12 18:17:13 +01:00
}
/* choose one */
c = rand()%c;
/* this is a bit obfuscated, but should be correct */
for(r=regions; c > 0; r=r->next) {
if (r->land && rplane(r)==NULL) --c;
2003-12-12 18:17:13 +01:00
}
return(r);
}
static void
check_phoenix(void)
{
const race * phoenix_race = rc_find("phoenix");
unit * phoenix;
region * r;
faction *f;
/* check if there is a phoenix in the world */
for(f=factions; f; f=f->next) {
unit *u;
for(u=f->units; u; u=u->nextF) {
if(u->race == phoenix_race) {
return;
}
}
}
/* it is not, so we create it */
r = random_land_region();
2003-12-12 18:17:13 +01:00
phoenix = createunit(r, findfaction(MONSTER_FACTION), 1, phoenix_race);
2003-12-28 16:30:43 +01:00
phoenix->name = strdup("Der Ph<50>nix");
2003-12-12 18:17:13 +01:00
2003-12-28 16:30:43 +01:00
/* TODO: generate an appropriate region message */
2003-12-12 18:17:13 +01:00
}
#endif
2001-01-25 10:37:55 +01:00
void
korrektur(void)
{
2003-05-13 17:43:18 +02:00
french_testers();
#if TEST_LOCALES
setup_locales();
#endif
fix_astralplane();
fix_firewalls();
fix_gates();
#ifdef TEST_GM_COMMANDS
setup_gm_faction();
#endif
2001-05-11 22:19:22 +02:00
update_gms();
verify_owners(false);
2001-01-25 10:37:55 +01:00
/* fix_herbtypes(); */
#ifdef CONVERT_TRIGGER
2001-02-20 23:54:05 +01:00
convert_triggers();
2001-01-25 10:37:55 +01:00
#endif
fix_migrants();
2003-09-28 10:05:50 +02:00
/* In Vin3 k<>nnen Parteien komplett <20>bergeben werden. */
#ifdef ENHANCED_QUIT
no_teurefremde(0);
#else
no_teurefremde(1);
2003-09-28 10:05:50 +02:00
#endif
update_igjarjuk_quest();
2001-01-25 10:37:55 +01:00
fix_allies();
update_gmquests(); /* test gm quests */
/* fix_unitrefs(); */
2001-01-25 10:37:55 +01:00
stats();
warn_password();
2002-04-28 17:37:48 +02:00
fix_road_borders();
2002-10-08 08:48:47 +02:00
if (turn>1000) curse_emptiness(); /*** disabled ***/
2002-09-02 22:36:12 +02:00
#ifdef ALLIANCES
2002-09-29 23:20:04 +02:00
/* init_alliances(); */
2002-09-02 22:36:12 +02:00
#endif
2002-05-02 23:19:45 +02:00
/* seems something fishy is going on, do this just
* to be on the safe side:
*/
fix_demand();
2003-12-12 18:17:13 +01:00
/* trade_orders(); */
2001-01-25 10:37:55 +01:00
/* immer ausf<73>hren, wenn neue Spr<70>che dazugekommen sind, oder sich
* Beschreibungen ge<EFBFBD>ndert haben */
show_newspells();
fix_age();
/* Immer ausf<73>hren! Erschafft neue Teleport-Regionen, wenn n<>tig */
create_teleport_plane();
2003-12-12 18:17:13 +01:00
#ifdef WDW_PHOENIX
check_phoenix();
#endif
2001-01-25 10:37:55 +01:00
if (global.data_version<TYPES_VERSION) fix_icastles();
2001-12-30 09:24:05 +01:00
#ifdef XMAS2000
2001-01-25 10:37:55 +01:00
santa_comes_to_town();
#endif
#ifdef FUZZY_BASE36
enable_fuzzy = true;
#endif
}
2002-10-06 09:38:03 +02:00
int
astral_penger(void)
{
plane * astralplane = getplanebyname("Astralraum");
region * r;
const item_type * itype = it_find("money");
assert(itype);
for (r=regions;r;r=r->next) {
unit * u;
if (rplane(r)!=astralplane) continue;
for (u=r->units;u;u=u->next) freset(u->faction, FL_DH);
for (u=r->units;u;u=u->next) {
if (u->faction->no==MONSTER_FACTION) continue;
if (!fval(u->faction, FL_DH)) {
fset(u->faction, FL_DH);
i_change(&u->items, itype, 300);
}
}
}
return 0;
}
2001-01-25 10:37:55 +01:00
void
korrektur_end(void)
{
}
void
init_conversion(void)
{
#if defined(CONVERT_TRIGGER)
at_register(&at_relation);
at_register(&at_relbackref);
at_register(&at_trigger);
at_register(&at_action);
#endif
}