2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
|
|
|
|
* 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)
|
|
|
|
|
*
|
|
|
|
|
* 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"
|
|
|
|
|
#include "karma.h"
|
|
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
|
#include "unit.h"
|
|
|
|
|
#include "race.h"
|
|
|
|
|
#include "region.h"
|
|
|
|
|
#include "item.h"
|
|
|
|
|
#include "group.h"
|
|
|
|
|
#include "pool.h"
|
|
|
|
|
#include "skill.h"
|
|
|
|
|
#include "faction.h"
|
|
|
|
|
#include "magic.h"
|
|
|
|
|
#include "message.h"
|
|
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
|
#include <attrib.h>
|
|
|
|
|
#include <base36.h>
|
|
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
attrib_type at_faction_special = {
|
|
|
|
|
"faction_special", NULL, NULL, NULL, a_writedefault, a_readdefault
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
age_prayer_timeout(attrib *a) {
|
|
|
|
|
return --a->data.sa[0];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrib_type at_prayer_timeout = {
|
|
|
|
|
"prayer_timeout", NULL, NULL, age_prayer_timeout, a_writedefault, a_readdefault
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
attrib_type at_prayer_effect = {
|
|
|
|
|
"prayer_effect", NULL, NULL, NULL, NULL, NULL
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
attrib_type at_wyrm = {
|
|
|
|
|
"wyrm", NULL, NULL, NULL, a_writedefault, a_readdefault
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
attrib_type at_fshidden = {
|
|
|
|
|
"fshidden", NULL, NULL, NULL, a_writedefault, a_readdefault
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
attrib_type at_jihad = {
|
|
|
|
|
"jihad", NULL, NULL, NULL, a_writedefault, a_readdefault
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
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 "
|
|
|
|
|
"Unterhalt pro Woche.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
1
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
1
|
2001-01-25 10:37:55 +01:00
|
|
|
|
},
|
|
|
|
|
{
|
2001-03-07 16:00:18 +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/+10. Die Fokussierung auf das "
|
|
|
|
|
"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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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!",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
100
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
100
|
2001-01-25 10:37:55 +01:00
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"Administrator",
|
|
|
|
|
"Das Einheitenlimit einer Partei mit dieser Eigenschaft erh<72>ht sich um "
|
2001-03-07 16:00:18 +01:00
|
|
|
|
"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 "
|
2001-03-07 16:00:18 +01:00
|
|
|
|
"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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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 "
|
2001-04-01 08:58:45 +02:00
|
|
|
|
"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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
100
|
2001-01-25 10:37:55 +01:00
|
|
|
|
},
|
|
|
|
|
/* TODO: negative Eigenschaft */
|
|
|
|
|
{
|
|
|
|
|
"Sappeur",
|
|
|
|
|
"Befestigungen wirken gegen Einheiten einer Partei mit dieser "
|
|
|
|
|
"Eigenschaft nur mit ihrer halben Schutzwirkung (aufgerundet).",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
100
|
2001-01-25 10:37:55 +01:00
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"Versteckt",
|
|
|
|
|
"Eine Partei mit dieser Eigenschaft hat die F<>higkeit zu Tarnung "
|
2001-03-07 16:00:18 +01:00
|
|
|
|
"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 "
|
2001-03-07 16:00:18 +01:00
|
|
|
|
"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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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 jedoch unm<6E>glich, in dieser "
|
|
|
|
|
"Woche ihren normalen T<>tigkeiten nachzugehen.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
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 1 auf Angriff und Schaden. Allerdings "
|
|
|
|
|
"kann es zu spontanen Pogromen gegen Angeh<65>rige der mit einem "
|
|
|
|
|
"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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
100
|
2001-02-28 19:25:25 +01:00
|
|
|
|
},
|
|
|
|
|
/* TODO: is_undead() und Sonderbehandlungen von Untoten */
|
|
|
|
|
{
|
|
|
|
|
"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.",
|
2001-04-01 08:58:45 +02:00
|
|
|
|
100
|
- 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
|
|
|
|
},
|
|
|
|
|
{ /* TODO: Namen <20>ndern */
|
|
|
|
|
"Schnell",
|
|
|
|
|
"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
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
buy_special(unit *u, strlist *S, fspecial_t special)
|
|
|
|
|
{
|
|
|
|
|
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=a->nexttype) {
|
|
|
|
|
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) {
|
|
|
|
|
cmistake(u, S->s, 250, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Alles ok, attribut geben */
|
|
|
|
|
|
|
|
|
|
if(a2) {
|
2001-04-01 08:58:45 +02:00
|
|
|
|
if(a2->data.sa[1] < fspecials[special].maxlevel) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
a2->data.sa[1]++;
|
|
|
|
|
add_message(&f->msgs, new_message(f,
|
|
|
|
|
"new_fspecial_level%S:special%d:level", special, a2->data.sa[1]));
|
|
|
|
|
} else {
|
|
|
|
|
cmistake(u, S->s, 251, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
a2 = a_add(&f->attribs, a_new(&at_faction_special));
|
|
|
|
|
a2->data.sa[0] = (short)special;
|
|
|
|
|
a2->data.sa[1] = 1;
|
|
|
|
|
add_message(&f->msgs, new_message(f, "new_fspecial%S:special", special));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
fspecial(const faction *f, fspecial_t special)
|
|
|
|
|
{
|
|
|
|
|
attrib *a;
|
|
|
|
|
|
|
|
|
|
for(a=a_find(f->attribs, &at_faction_special); a; a=a->nexttype) {
|
|
|
|
|
if(a->data.sa[0] == special) return a->data.sa[1];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
sacrificings(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u;
|
|
|
|
|
|
|
|
|
|
for(r=regions; r; r=r->next) {
|
|
|
|
|
for(u=r->units; u; u=u->next) {
|
|
|
|
|
if(igetkeyword(u->thisorder) == K_SACRIFICE) {
|
|
|
|
|
int n = 1, karma;
|
|
|
|
|
char *s = getstrtoken();
|
|
|
|
|
|
|
|
|
|
if(s && *s) n = atoi(s);
|
|
|
|
|
if(n <= 0) {
|
|
|
|
|
cmistake(u, u->thisorder, 252, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
s = getstrtoken();
|
|
|
|
|
|
|
|
|
|
switch(findparam(s)) {
|
|
|
|
|
|
|
|
|
|
case P_SILVER:
|
|
|
|
|
n = use_pooled(u, r, R_SILVER, n);
|
|
|
|
|
if(n < 10000) {
|
|
|
|
|
cmistake(u, u->thisorder, 51, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
change_resource(u, R_SILVER, n);
|
|
|
|
|
karma = n/10000;
|
|
|
|
|
u->faction->karma += karma;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case P_AURA:
|
|
|
|
|
if(!is_mage(u)) {
|
|
|
|
|
cmistake(u, u->thisorder, 214, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if(pure_skill(u, SK_MAGIC, u->region) < 10) {
|
|
|
|
|
cmistake(u, u->thisorder, 253, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
n = min(get_spellpoints(u), min(max_spellpoints(u->region, u), n));
|
|
|
|
|
if(n <= 0) {
|
|
|
|
|
cmistake(u, u->thisorder, 254, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
karma = n;
|
|
|
|
|
u->faction->karma += n;
|
|
|
|
|
change_maxspellpoints(u, -n);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
cmistake(u, u->thisorder, 255, MSG_EVENT);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
prayers(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u;
|
|
|
|
|
strlist *S;
|
|
|
|
|
|
|
|
|
|
for(r=regions; r; r=r->next) {
|
|
|
|
|
for(u=r->units; u; u=u->next) {
|
|
|
|
|
for(S = u->orders; S; S = S->next) if(igetkeyword(S->s) == K_PRAY) {
|
|
|
|
|
attrib *a, *a2;
|
|
|
|
|
unit *u2;
|
|
|
|
|
int karma_cost;
|
|
|
|
|
short mult = 1;
|
|
|
|
|
param_t p;
|
|
|
|
|
char *s = getstrtoken();
|
|
|
|
|
|
|
|
|
|
if(findparam(s) == P_FOR) s = getstrtoken();
|
|
|
|
|
|
|
|
|
|
p = findparam(s);
|
|
|
|
|
switch(p) {
|
|
|
|
|
case P_AURA:
|
|
|
|
|
if(!is_mage) {
|
|
|
|
|
cmistake(u, S->s, 214, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
case P_AID:
|
|
|
|
|
case P_MERCY:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
cmistake(u, S->s, 256, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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, S->s, 250, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u->faction->karma -= karma_cost;
|
|
|
|
|
|
|
|
|
|
add_message(&u->faction->msgs, new_message(u->faction,
|
|
|
|
|
"pray_success%u:unit",u));
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
set_jihad(void)
|
|
|
|
|
{
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u;
|
|
|
|
|
strlist *S;
|
|
|
|
|
|
|
|
|
|
for(r=regions; r; r=r->next) {
|
|
|
|
|
for(u=r->units; u; u=u->next) {
|
|
|
|
|
for(S = u->orders; S; S=S->next) if(igetkeyword(S->s) == K_SETJIHAD) {
|
|
|
|
|
faction *f = u->faction;
|
|
|
|
|
int can = fspecial(f, FS_JIHAD);
|
|
|
|
|
int has = 0;
|
|
|
|
|
race_t jrace;
|
|
|
|
|
attrib *a;
|
|
|
|
|
char *s;
|
|
|
|
|
|
|
|
|
|
for(a = a_find(f->attribs, &at_jihad); a; a = a->nexttype) {
|
|
|
|
|
has += a->data.sa[1];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(has >= can) {
|
|
|
|
|
cmistake(u, S->s, 280, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
s = getstrtoken();
|
|
|
|
|
if(!s || !*s) {
|
|
|
|
|
cmistake(u, S->s, 281, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jrace = findrace(s);
|
|
|
|
|
|
|
|
|
|
if (race[jrace].nonplayer) {
|
|
|
|
|
cmistake(u, S->s, 282, MSG_EVENT);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for(a = a_find(f->attribs, &at_jihad); a; a = a->nexttype) {
|
|
|
|
|
if(a->data.sa[0] == jrace) break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(a) {
|
|
|
|
|
a->data.sa[1]++;
|
|
|
|
|
} else {
|
|
|
|
|
a = a_add(&f->attribs, a_new(&at_jihad));
|
|
|
|
|
a->data.sa[0] = (short)jrace;
|
|
|
|
|
a->data.sa[1] = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
add_message(&f->msgs, new_message(f,
|
|
|
|
|
"setjihad%s:race", race[jrace].name[2]));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
jihad(faction *f, race_t race)
|
|
|
|
|
{
|
|
|
|
|
attrib *a;
|
|
|
|
|
|
|
|
|
|
for(a = a_find(f->attribs, &at_jihad); a; a = a->nexttype) {
|
|
|
|
|
if(a->data.sa[0] == race) return a->data.sa[1];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
jihad_attacks(void)
|
|
|
|
|
{
|
|
|
|
|
faction *f;
|
|
|
|
|
region *r;
|
|
|
|
|
unit *u, *u2;
|
|
|
|
|
strlist *S;
|
|
|
|
|
attrib *a;
|
|
|
|
|
ally *sf, **sfp;
|
|
|
|
|
|
|
|
|
|
for(f=factions; f; f=f->next) if(fspecial(f, FS_JIHAD)) {
|
|
|
|
|
for(r=f->first; r != f->last; r = r->next) if(rand()%1000 <= 1) {
|
|
|
|
|
boolean doit = false;
|
|
|
|
|
|
|
|
|
|
for(u=r->units; u; u=u->next) if(jihad(f, u->race)) {
|
|
|
|
|
doit = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(doit == false) continue;
|
|
|
|
|
|
|
|
|
|
printf("\tPogrom durch %s in %s\n", factionid(f), regionid(r));
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
a = a_find(u2->attribs, &at_group);
|
|
|
|
|
if (a) sfp = &((group*)a->data.v)->allies;
|
|
|
|
|
|
|
|
|
|
for (sf=*sfp; sf; sf = sf->next) {
|
|
|
|
|
if(sf->faction == u->faction) break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(sf) sf->status = sf->status & (HELP_ALL - HELP_FIGHT);
|
|
|
|
|
|
|
|
|
|
sprintf(buf, "%s %s", keywords[K_ATTACK], unitid(u));
|
|
|
|
|
S = makestrlist(buf);
|
|
|
|
|
addlist(&u2->orders, S);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-03-07 16:00:18 +01:00
|
|
|
|
|