server/src/common/kernel/karma.c

607 lines
18 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.de)
2001-01-25 10:37:55 +01:00
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
*/
/* TODO: enum auf fst_ umstellen. Pointer auf Display-Routine */
#include <config.h>
#include "eressea.h"
2001-01-25 10:37:55 +01:00
#include "karma.h"
/* kernel includes */
#include "unit.h"
#include "save.h"
2001-01-25 10:37:55 +01:00
#include "race.h"
#include "region.h"
#include "item.h"
#include "group.h"
2004-06-21 18:45:27 +02:00
#include "order.h"
2001-01-25 10:37:55 +01:00
#include "pool.h"
#include "skill.h"
#include "faction.h"
#include "magic.h"
#include "message.h"
/* util includes */
#include <util/attrib.h>
#include <util/base36.h>
#include <util/rng.h>
2001-01-25 10:37:55 +01:00
/* libc includes */
#include <math.h>
#include <stdlib.h>
attrib_type at_faction_special = {
"faction_special", NULL, NULL, NULL, a_writeshorts, a_readshorts
2001-01-25 10:37:55 +01:00
};
int
age_prayer_timeout(attrib *a) {
return --a->data.sa[0];
}
attrib_type at_prayer_timeout = {
"prayer_timeout", NULL, NULL, age_prayer_timeout, a_writeshorts, a_readshorts
2001-01-25 10:37:55 +01:00
};
attrib_type at_prayer_effect = {
"prayer_effect", NULL, NULL, NULL, NULL, NULL
};
attrib_type at_wyrm = {
"wyrm", NULL, NULL, NULL, a_writeint, a_readint
2001-01-25 10:37:55 +01:00
};
attrib_type at_fshidden = {
"fshidden", NULL, NULL, NULL, a_writeint, a_readint
2001-01-25 10:37:55 +01:00
};
attrib_type at_jihad = {
"jihad", NULL, NULL, NULL, a_writeshorts, a_readshorts
2001-01-25 10:37:55 +01:00
};
#ifdef KARMA_MODULE
2001-01-25 10:37:55 +01:00
struct fspecialdata fspecials[MAXFACTIONSPECIALS] = {
{
"Regeneration",
"Personen in einer Partei mit dieser Eigenschaft heilen jeden "
"Schaden innerhalb einer Woche und zus<75>tzlich in jeder Kampfrunde "
"HP entsprechend ihres Ausdauer-Talents. Sie ben<65>tigen jedoch 11 "
"Silber Unterhalt pro Woche.",
1
2001-01-25 10:37:55 +01:00
},
{ /* TODO: F<>r alte Parteien zu stark */
2001-01-25 10:37:55 +01:00
"St<EFBFBD>dter",
"Personen einer Partei mit dieser Eigenschaft lieben die St<53>dte und "
"verabscheuen das Leben in der freien Natur. Ihr Arbeitlohn ver<65>ndert "
"sich in Abh<62>ngigkeit vom gr<67><72>ten Geb<65>ude in der Region. Ist es eine "
"Zitadelle so erhalten sie 2 Silber mehr pro Runde, bei einer Festung "
"1 Silber mehr. Bei einer Burg bekommen sie den normalen Arbeitslohn, "
"bei einem Turm 1 Silber, bei einer Befestigung 3 Silber weniger. Gibt "
"es kein entsprechendes Geb<65>ude in der Region, verringert sich ihr "
"Arbeitslohn um 5 Silber.",
1
2001-01-25 10:37:55 +01:00
},
{
"Barbar",
2001-01-25 10:37:55 +01:00
"Einheiten dieser Partei erhalten durch Lernen von Waffentalenten "
"(Taktik und Reiten z<>hlen nicht dazu!) 40 statt 30 Lerntage. Weitere "
"Stufen erh<72>hen den Bonus um 5 Lerntage. Die Fokussierung auf das "
2001-01-25 10:37:55 +01:00
"Kriegerdasein f<>hrt jedoch zu Problemen, andere Talente zu erlernen. "
"In allen nichtkriegerischen Talenten einschlie<69>lich Magie und Taktik "
"erhalten sie die entsprechende Anzahl von Lerntagen weniger pro "
"Lernwoche.",
100
2001-01-25 10:37:55 +01:00
},
/* TODO: Noch nicht so implementiert. */
{
"Wyrm",
"Eine Partei mit dieser Eigenschaft kann einen ihrer Magier mit Hilfe "
"eines speziellen Zaubers permanent in einen Drachen verwandeln. Der "
"Drache ist zun<75>chst jung, kann sich jedoch durch Lernen des speziellen "
"Talents 'Drachenwesen' in einen gr<67><72>eren Drachen verwandeln. Um zu "
"einem ausgewachsenen Drachen zu werden ben<65>tigt der verwandelte Magier "
"die Talentstufe 6, um zu einem Wyrm zu werden die Talentstufe 12. "
"Ein junger Drache ben<65>tigt 1000 Silber Unterhalt pro Woche, ein "
"ausgewachsener Drache 5000 und ein Wyrm 10000 Silber. Bekommt er "
"dieses Silber nicht, besteht eine Wahrscheinlichkeit, das er desertiert!",
100
2001-01-25 10:37:55 +01:00
},
/* TODO: Relativ sinnlose Eigenschaft. */
{
"Miliz",
"Alle Personen dieser Partei beginnen mit 30 Talenttagen in allen "
"Waffentalenten, in denen ihre Rasse keinen Malus hat. Zus<75>tzliche "
"Stufen bringen jeweils einen zus<75>tzlichen Talentpunkt.",
100
2001-01-25 10:37:55 +01:00
},
{ /* Ohne Schiffsunterhaltskosten schwache Eigenschaft. */
/* Aufpassen: Tragkraft? */
2001-01-25 10:37:55 +01:00
"Feenreich",
"Alle Personen dieser Partei wiegen nur noch die H<>lfte ihres normalen "
"Gewichtes. Die N<>he zum Feenreich macht sie jedoch besonders "
"anf<EFBFBD>llig gegen Waffen aus Eisen, bei einem Treffer durch eine "
"solche Waffe nehmen sie einen zus<75>tzlichen Schadenspunkt. Zus<75>tzliche "
"Stufen dieser Eigenschaft verringern das Gewicht auf 1/3, 1/4, ... und "
"erh<EFBFBD>hen den Schaden durch Eisenwaffen um einen weiteren Punkt.",
100
2001-01-25 10:37:55 +01:00
},
{
"Administrator",
"Das Einheitenlimit einer Partei mit dieser Eigenschaft erh<72>ht sich um "
"400 Einheiten. Leider verschlingt der Verwaltungsaufwand viel Silber, "
2001-01-25 10:37:55 +01:00
"so dass sich der Unterhalt pro Person um 1 Silberst<73>ck erh<72>ht. Weitere "
"Stufen der Eigenschaft erh<72>hen das Limit um weitere 400 Einheiten und "
2001-01-25 10:37:55 +01:00
"den Unterhalt um ein weiteres Silberst<73>ck.",
100
2001-01-25 10:37:55 +01:00
},
/* TODO: Noch nicht so implementiert */
{
"Telepath",
"Der Geist fremder Personen bleibt den Magiern einer Partei mit dieser "
"Eigenschaft nicht verschlossen. Fremde Einheiten erscheinen im Report "
"wie eigene Einheiten, mit allen Talentwerten und Gegenst<73>nden. Leider "
"f<EFBFBD>hrt eine so intensive Besch<63>ftigung mit dem Geistigen zur k<>rperlichen "
"Verk<EFBFBD>mmerung, und die Partei erh<72>lt -1 auf alle Talente au<61>er "
"Alchemie, Kr<4B>uterkunde, Magie, Spionage, Tarnung und Wahrnehmung. Wird "
"diese Eigenschaft ein zweites Mal erworben, so bleibt die Wirkung nicht "
"auf Magier beschr<68>nkt, sondern alle Einheiten einer Partei k<>nnen "
"die Talente und Gegenst<73>nde aller fremden Einheiten sehen. Allerdings "
"gibt es in beiden F<>llen eine Einschr<68>nkung: Die Einheit mu<6D> sich einen "
"Monat lang auf die psychischen Str<74>me einer Region einstellen, bevor "
"sie in der Lage ist, die Gedanken der anderen zu lesen.",
2
2001-01-25 10:37:55 +01:00
},
{
"Amphibium",
"Einheiten dieser Partei k<>nnen durch Ozeanfelder laufen. Allerdings "
"nehmen sie 10 Schadenspunkte, wenn sie am Ende einer Woche in einem "
"Ozeanfeld stehen. Pferde weigern sich, in ein Ozeanfeld zu laufen. "
"Zus<EFBFBD>tzliche Stufen dieser Eigenschaft reduzieren den Schaden um jeweils "
"5 Punkte. Achtung: Auf dem Ozean wird kein Schaden regeneriert.",
3
2001-01-25 10:37:55 +01:00
},
/* TODO: negative Eigenschaft */
{
"Magokrat",
"Eine Partei mit dieser Eigenschaft hat eine so hohe magische "
"Affinit<EFBFBD>t, dass sie pro Stufe der Eigenschaft zwei zus<75>tzlich "
"Magier ausbilden kann.",
100
2001-01-25 10:37:55 +01:00
},
/* TODO: negative Eigenschaft, vergleichsweise schwach */
2001-01-25 10:37:55 +01:00
{
"Sappeur",
"Befestigungen wirken gegen Einheiten einer Partei mit dieser "
"Eigenschaft nur mit ihrer halben Schutzwirkung (aufgerundet).",
1
2001-01-25 10:37:55 +01:00
},
/* TODO: Noch nicht implementiert */
{
"Springer",
"Einheiten einer Partei mit dieser Eigenschaft k<>nnen sich mit "
"Hilfe eines speziellen Befehls <20>ber eine Entfernung von zwei "
"Regionen teleportieren. Einheiten, die sich so teleportieren, "
"bleiben jedoch f<>r einige Wochen instabil und k<>nnen sich in "
"seltenen F<>llen selbst<73>ndig in eine zuf<75>llige Nachbarregion "
"versetzen. Zus<75>tzliche Stufen erh<72>hen die Reichweite um jeweils "
"eine Region, erh<72>hen jedoch die Wahrscheinlichkeit eines zuf<75>lligen "
"Versetzens geringf<67>gig.",
100
2001-01-25 10:37:55 +01:00
},
{
/* Evt. zwei Stufen draus machen */
2001-01-25 10:37:55 +01:00
"Versteckt",
"Eine Partei mit dieser Eigenschaft hat die F<>higkeit zu Tarnung "
"zur Perfektion getrieben. Jede Einheit mit mindestens Tarnung 3 "
2001-01-25 10:37:55 +01:00
"versteckt alle ihre Gegenst<73>nde so, da<64> sie von anderen Parteien "
"nicht mehr gesehen werden k<>nnen. Jede Einheit mit mindestens Tarnung 6 "
2001-01-25 10:37:55 +01:00
"schafft es auch, die Zahl der in ihr befindlichen Personen zu verbergen. "
"Um diese Eigenschaft steuern zu k<>nnen, stehen diesen Parteien die "
"Befehle TARNE ANZAHL [NICHT] und TARNE GEGENST<53>NDE [NICHT] zur "
"Verf<EFBFBD>gung.",
1
2001-01-25 10:37:55 +01:00
},
/* TODO: Noch nicht implementiert */
{
"Erdelementarist",
"Alle Geb<65>ude dieser Partei sind von Erdelementaren beseelt und k<>nnen "
"sich mit Hilfe eines speziellen Befehls jede Woche um eine Region "
"bewegen. Dies macht es den Bewohnern - welche alle einer Partei mit "
"dieser Eigenschaft angeh<65>ren m<>ssen - jedoch unm<6E>glich, in dieser "
2001-01-25 10:37:55 +01:00
"Woche ihren normalen T<>tigkeiten nachzugehen.",
1
2001-01-25 10:37:55 +01:00
},
{
"Magische Immunit<69>t",
"Eine Partei mit dieser Eigenschaft ist v<>llig immun gegen alle Arten "
"von Magie. Allerdings verlieren die Magier einer solchen Partei ihre "
"F<EFBFBD>higkeit, Aura zu regenerieren, v<>llig.",
1
2001-01-25 10:37:55 +01:00
},
/* TODO: Noch nicht implementiert */
{
"Vielseitige Magie",
"Eine Partei mit dieser Eigenschaft kann einen ihrer Magier in einem "
"anderen als dem Parteimagiegebiet ausbilden. Weitere Stufen erm<72>glichen "
"jeweils einem weiteren Magier das Lernen eines anderen Gebiets.",
100
2001-01-25 10:37:55 +01:00
},
{
"Jihad",
"Eine Partei mit dieser Eigenschaft kann eine (Spieler)-Rasse "
"mit dem speziellen kurzen Befehl JIHAD <RASSE> zum Feind erkl<6B>ren. "
"Bei einem Kampf gegen einen Angeh<65>rigen dieser Rasse bekommen ihre "
"K<EFBFBD>mpfer grunds<64>tzlich einen Bonus von +1 auf Angriff und Schaden. "
"Allerdings kann es zu spontanen Pogromen gegen Angeh<65>rige der mit einem "
2001-01-25 10:37:55 +01:00
"Jihad belegten Rasse kommen. Wird die Eigenschaft mehrmals erworben "
"k<EFBFBD>nnen entweder mehrere Rassen mit einem Jihad belegt werden, "
"oder eine Rasse mehrfach, in diesem Fall addiert sich die Wirkung. "
"Ein einmal erkl<6B>rter Jihad kann nicht wieder r<>ckg<6B>ngig gemacht "
"werden.",
100
},
/* TODO: is_undead() und Sonderbehandlungen von Untoten */
2001-05-20 09:23:54 +02:00
/* in dieser Form ultimative Eigenschaft f<>r Orks */
{
"Untot",
"Personen einer Partei mit dieser Eigenschaft bekommen automatisch doppelt "
"soviele Trefferpunkte wie normale Angeh<65>rige der entsprechenden Rasse, "
"verlieren jedoch ihre F<>higkeit zur Regeneration erlittenen Schadens "
"komplett.",
1
},
{
"Windvolk",
"Ein solches Volk ist sehr athletisch und die Kinder <20>ben besonders "
"den Langstreckenlauf von kleinauf. Nach jahrelangem Training sind "
"sie dann in der Lage, sich zu Fu<46> so schnell zu bewegen als w<>rden "
"sie reiten. Allerdings hat das jahrelange Konditionstraining ihre "
"Kr<EFBFBD>fte schwinden lassen, und ihre Tragkraft ist um 2 Gewichtseinheiten "
"verringert.",
1
},
{
"Gl<EFBFBD>ck",
"Diese Eigenschaft bewirkt, das der Partei gelegentlich positive "
"Ereignisse zusto<74>en. Dies k<>nnen spontane Verbesserungen des "
"Wissensstandes, ein Fund wertvoller Gegenst<73>nde oder <20>hnliches "
"sein. Je h<>ufiger diese Eigenschaft erworben wird, desto gr<67><72>er die "
"Wahrscheinlichkeit f<>r solche Ereignisse, und desto postiver ihre "
"Auswirkungen.",
100
},
{
"Lykanthrop",
"Angeh<EFBFBD>rige einer Partei mit dieser Eigenschaft sind Werwesen. Einheiten "
2002-04-12 14:46:16 +02:00
"einer solchen Partei k<>nnen sich mit Hilfe der langen Befehle 'WERWESEN' und "
"'WERWESEN NICHT' in eine andere Form verwandeln. Beide Befehle haben "
2002-04-12 14:46:16 +02:00
"nur eine gewisse Erfolgswahrscheinlichkeit und funktionieren nicht immer: "
"Je h<>her die Stufe der Eigenschaft, desto gr<67><72>er die Wahrscheinlichkeit, "
"dass die Verwandlung gelingt, und je geringer die Chance, sich "
"zur<EFBFBD>ckzuverwandeln. Zudem besteht eine zunehmende kleine Chance der "
"spontanen Verwandlung. In Werform erhalten die Einheiten einen Bonus auf "
"Angriff, Schaden und nat<61>rliche R<>stung in H<>he der Eigenschaft. Sie "
"benutzen jedoch keine Pferde im Kampf, und verwenden als R<>stungen nur "
"Schilde. Sie sind eingeschr<68>nkt und k<>nnen in Werform kein Geld verdienen, "
"nicht Zaubern und nicht Lernen oder Lehren.",
4
},
/* TODO: Noch nicht implementiert */
{
"Elite",
"F<EFBFBD>r eine Partei mit dieser Eigenschaft verdoppeln sich alle Boni- und "
"Mali ihrer Rasse. Ihre Unterhaltskosten erh<72>hen sich auf 12 Silber pro "
2002-04-12 14:46:16 +02:00
"Runde, ihr Einheitenlimit reduziert sich auf 25%%. Diese Eigenschaft "
"kann nicht erworben werden, wenn die Partei nach dem Erwerb zuviele "
"Einheiten h<>tte.",
1
2001-01-25 10:37:55 +01:00
}
};
void
2004-06-21 18:45:27 +02:00
buy_special(unit *u, struct order * ord, fspecial_t special)
2001-01-25 10:37:55 +01:00
{
int count = 0;
int cost;
attrib *a, *a2 = NULL;
faction *f = u->faction;
/* Kosten berechnen */
for(a=a_find(f->attribs, &at_faction_special); a && a->type==&at_faction_special; a=a->next) {
2001-01-25 10:37:55 +01:00
count += a->data.sa[1];
if(a->data.sa[0] == special) a2 = a;
}
cost = (int)(100 * pow(3, count));
/* Pr<50>fen, ob genug Karma vorhanden. */
if(f->karma < cost) {
2004-06-21 18:45:27 +02:00
cmistake(u, ord, 250, MSG_EVENT);
2001-01-25 10:37:55 +01:00
return;
}
/* Alles ok, attribut geben */
if (a2) {
if(a2->data.sa[1] < fspecials[special].maxlevel) {
2001-01-25 10:37:55 +01:00
a2->data.sa[1]++;
ADDMSG(&f->msgs, msg_message("new_fspecial_level",
"special level", special, a2->data.sa[1]));
2001-01-25 10:37:55 +01:00
} else {
2004-06-21 18:45:27 +02:00
cmistake(u, ord, 251, MSG_EVENT);
2001-01-25 10:37:55 +01:00
return;
}
} else {
a2 = a_add(&f->attribs, a_new(&at_faction_special));
a2->data.sa[0] = (short)special;
a2->data.sa[1] = 1;
ADDMSG(&f->msgs, msg_message("new_fspecial", "special", special));
2001-01-25 10:37:55 +01:00
}
}
int
fspecial(const faction *f, fspecial_t special)
{
attrib *a;
2001-01-25 10:37:55 +01:00
for (a=a_find(f->attribs, &at_faction_special); a && a->type==&at_faction_special; a=a->next) {
if(a->data.sa[0] == special) return a->data.sa[1];
}
return 0;
2001-01-25 10:37:55 +01:00
}
2004-06-21 18:45:27 +02:00
static int
sacrifice_cmd(unit * u, struct order * ord)
2001-01-25 10:37:55 +01:00
{
2004-06-21 18:45:27 +02:00
int n = 1, karma;
const char *s;
init_tokens(ord);
skip_token();
s = getstrtoken();
if (s && *s) n = atoi(s);
if (n <= 0) {
cmistake(u, ord, 252, MSG_EVENT);
return 0;
}
s = getstrtoken();
switch(findparam(s, u->faction->locale)) {
case P_MONEY:
n = use_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, n);
2004-06-21 18:45:27 +02:00
if(n < 10000) {
cmistake(u, ord, 51, MSG_EVENT);
return 0;
}
change_resource(u, oldresourcetype[R_SILVER], n);
2004-06-21 18:45:27 +02:00
karma = n/10000;
u->faction->karma += karma;
break;
case P_AURA:
if(!is_mage(u)) {
cmistake(u, ord, 214, MSG_EVENT);
return 0;
}
if (get_level(u, SK_MAGIC) < 10) {
cmistake(u, ord, 253, MSG_EVENT);
return 0;
}
n = min(get_spellpoints(u), min(max_spellpoints(u->region, u), n));
if(n <= 0) {
cmistake(u, ord, 254, MSG_EVENT);
return 0;
}
karma = n;
u->faction->karma += n;
change_maxspellpoints(u, -n);
break;
default:
cmistake(u, u->thisorder, 255, MSG_EVENT);
}
return 0;
2001-01-25 10:37:55 +01:00
}
2004-06-21 18:45:27 +02:00
static int
prayer_cmd(unit * u, struct order * ord)
2001-01-25 10:37:55 +01:00
{
2004-06-21 18:45:27 +02:00
region *r = u->region;
attrib *a, *a2;
unit *u2;
int karma_cost;
short mult = 1;
param_t p;
const char *s;
init_tokens(ord);
skip_token();
s = getstrtoken();
if (findparam(s, u->faction->locale) == P_FOR) s = getstrtoken();
p = findparam(s, u->faction->locale);
switch(p) {
case P_AURA:
if (!is_mage(u)) {
cmistake(u, ord, 214, MSG_EVENT);
return 0;
}
case P_AID:
case P_MERCY:
break;
default:
cmistake(u, ord, 256, MSG_EVENT);
return 0;
}
a = a_find(u->faction->attribs, &at_prayer_timeout);
if (a) mult = (short)(2 * a->data.sa[1]);
karma_cost = 10 * mult;
if (u->faction->karma < karma_cost) {
cmistake(u, ord, 250, MSG_EVENT);
return 0;
}
u->faction->karma -= karma_cost;
ADDMSG(&u->faction->msgs, msg_message("pray_success", "unit", u));
2004-06-21 18:45:27 +02:00
switch (p) {
case P_AURA:
set_spellpoints(u, max_spellpoints(u->region, u));
break;
case P_AID:
for(u2 = r->units; u2; u2=u2->next) if(u2->faction == u->faction) {
a2 = a_add(&u->attribs, a_new(&at_prayer_effect));
a2->data.sa[0] = PR_AID;
a2->data.sa[1] = 1;
}
break;
case P_MERCY:
for(u2 = r->units; u2; u2=u2->next) if(u2->faction == u->faction) {
a2 = a_add(&u->attribs, a_new(&at_prayer_effect));
a2->data.sa[0] = PR_MERCY;
a2->data.sa[1] = 80;
}
break;
}
if(!a) a = a_add(&u->faction->attribs, a_new(&at_prayer_timeout));
a->data.sa[0] = (short)(mult * 14);
a->data.sa[1] = (short)mult;
return 0;
2001-01-25 10:37:55 +01:00
}
2004-06-21 18:45:27 +02:00
static int
jihad_cmd(unit * u, struct order * ord)
2001-01-25 10:37:55 +01:00
{
2004-06-21 18:45:27 +02:00
faction *f = u->faction;
int can = fspecial(f, FS_JIHAD);
int has = 0;
const race * jrace;
race_t jrt;
attrib *a;
const char *s;
for (a = a_find(f->attribs, &at_jihad); a && a->type==&at_jihad; a = a->next) {
2004-06-21 18:45:27 +02:00
has += a->data.sa[1];
}
if (has >= can) {
cmistake(u, ord, 280, MSG_EVENT);
return 0;
}
init_tokens(ord);
skip_token();
s = getstrtoken();
if (!s || !*s) {
cmistake(u, ord, 281, MSG_EVENT);
return 0;
}
jrace = rc_find(s);
jrt = old_race(jrace);
if (!playerrace(jrace)) {
cmistake(u, ord, 282, MSG_EVENT);
return 0;
}
for (a = a_find(f->attribs, &at_jihad); a && a->type==&at_jihad; a = a->next) {
2004-06-21 18:45:27 +02:00
if (a->data.sa[0] == jrt) break;
}
if (a) {
a->data.sa[1]++;
} else {
a = a_add(&f->attribs, a_new(&at_jihad));
a->data.sa[0] = (short)jrt;
a->data.sa[1] = 1;
}
add_message(&f->msgs, msg_message("setjihad", "race", jrace));
return 0;
2001-01-25 10:37:55 +01:00
}
int
jihad(faction *f, const race * rc)
2001-01-25 10:37:55 +01:00
{
attrib *a;
race_t jrt = old_race(rc);
2001-01-25 10:37:55 +01:00
for(a = a_find(f->attribs, &at_jihad); a && a->type==&at_jihad; a = a->next) {
if(a->data.sa[0] == jrt) return a->data.sa[1];
2001-01-25 10:37:55 +01:00
}
return 0;
}
void
jihad_attacks(void)
{
faction *f;
region *r;
unit *u, *u2;
ally *sf, **sfp;
2001-01-25 10:37:55 +01:00
for(f=factions; f; f=f->next) if(fspecial(f, FS_JIHAD)) {
region * last = lastregion(f);
for (r=firstregion(f); r != last; r = r->next) if (rng_int()%1000 <= 1) {
2001-01-25 10:37:55 +01:00
boolean doit = false;
for(u=r->units; u; u=u->next) if(jihad(f, u->race)) {
doit = true;
break;
}
if(doit == false) continue;
log_printf("-->> Pogrom durch %s in %s\n", factionid(f), regionname(r, NULL));
2001-01-25 10:37:55 +01:00
for(u2 = r->units; u; u=u->next) if(u2->faction == f) {
for(u=r->units; u; u=u->next) if(jihad(f, u->race)) {
/* Allianz aufl<66>sen */
sfp = &u2->faction->allies;
if (fval(u, UFL_GROUP)) {
attrib * a = a_find(u2->attribs, &at_group);
if (a) sfp = &((group*)a->data.v)->allies;
}
2001-01-25 10:37:55 +01:00
for (sf=*sfp; sf; sf = sf->next) {
if(sf->faction == u->faction) break;
}
2002-09-02 22:36:12 +02:00
if (sf) sf->status = sf->status & (HELP_ALL - HELP_FIGHT);
2001-01-25 10:37:55 +01:00
addlist(&u2->orders, create_order(K_ATTACK, u->faction->locale, "%i", u->no));
2001-01-25 10:37:55 +01:00
}
}
}
}
}
2004-06-21 18:45:27 +02:00
void
karma(void)
{
parse(K_PRAY, prayer_cmd, false);
parse(K_SETJIHAD, jihad_cmd, false);
parse(K_SACRIFICE, sacrifice_cmd, true);
}
#endif