2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2001-12-10 01:13:39 +01:00
|
|
|
|
*
|
2003-07-29 11:48:03 +02:00
|
|
|
|
* 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)
|
|
|
|
|
*
|
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
2001-02-10 11:40:12 +01:00
|
|
|
|
#include <eressea.h>
|
2001-02-05 17:11:59 +01:00
|
|
|
|
#include "creport.h"
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
- 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
|
|
|
|
/* tweakable features */
|
|
|
|
|
#define ENCODE_SPECIAL 1
|
|
|
|
|
#define RENDER_CRMESSAGES 1
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* modules include */
|
|
|
|
|
#include <modules/score.h>
|
|
|
|
|
|
|
|
|
|
/* attributes include */
|
|
|
|
|
#include <attributes/follow.h>
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
#include <attributes/racename.h>
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#include <attributes/orcification.h>
|
|
|
|
|
#include <attributes/otherfaction.h>
|
|
|
|
|
#include <attributes/raceprefix.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2001-02-10 11:40:12 +01:00
|
|
|
|
/* gamecode includes */
|
|
|
|
|
#include "laws.h"
|
|
|
|
|
#include "economy.h"
|
|
|
|
|
|
|
|
|
|
/* kernel includes */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include <kernel/alchemy.h>
|
2004-06-27 18:56:01 +02:00
|
|
|
|
#include <kernel/alliance.h>
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include <kernel/border.h>
|
|
|
|
|
#include <kernel/building.h>
|
|
|
|
|
#include <kernel/faction.h>
|
|
|
|
|
#include <kernel/group.h>
|
|
|
|
|
#include <kernel/item.h>
|
|
|
|
|
#include <kernel/karma.h>
|
|
|
|
|
#include <kernel/magic.h>
|
|
|
|
|
#include <kernel/message.h>
|
|
|
|
|
#include <kernel/movement.h>
|
|
|
|
|
#include <kernel/order.h>
|
|
|
|
|
#include <kernel/plane.h>
|
|
|
|
|
#include <kernel/race.h>
|
|
|
|
|
#include <kernel/region.h>
|
|
|
|
|
#include <kernel/reports.h>
|
|
|
|
|
#include <kernel/resources.h>
|
|
|
|
|
#include <kernel/ship.h>
|
|
|
|
|
#include <kernel/skill.h>
|
|
|
|
|
#include <kernel/teleport.h>
|
|
|
|
|
#include <kernel/unit.h>
|
|
|
|
|
#include <kernel/save.h>
|
2001-02-10 11:40:12 +01:00
|
|
|
|
|
|
|
|
|
/* util includes */
|
- 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
|
|
|
|
#include <util/message.h>
|
2001-02-10 11:40:12 +01:00
|
|
|
|
#include <goodies.h>
|
2001-02-24 13:50:51 +01:00
|
|
|
|
#include <crmessage.h>
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
#include <nrmessage.h>
|
2002-12-18 18:40:21 +01:00
|
|
|
|
#include <base36.h>
|
2001-04-21 13:07:54 +02:00
|
|
|
|
#include <language.h>
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
|
2001-02-10 11:40:12 +01:00
|
|
|
|
/* libc includes */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
/* imports */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
extern const char *directions[];
|
|
|
|
|
extern const char *spelldata[];
|
|
|
|
|
extern int quiet;
|
2003-12-14 11:02:29 +01:00
|
|
|
|
boolean opt_cr_absolute_coords = false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
- 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
|
|
|
|
/* globals */
|
2002-02-24 11:04:47 +01:00
|
|
|
|
#define C_REPORT_VERSION 64
|
- 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
|
|
|
|
|
2001-04-21 13:07:54 +02:00
|
|
|
|
#define TAG_LOCALE "de"
|
|
|
|
|
#ifdef TAG_LOCALE
|
2001-12-10 01:13:39 +01:00
|
|
|
|
static const char *
|
|
|
|
|
crtag(const char * key)
|
2001-04-21 13:07:54 +02:00
|
|
|
|
{
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static const struct locale * lang = NULL;
|
2001-04-21 13:07:54 +02:00
|
|
|
|
if (!lang) lang = find_locale(TAG_LOCALE);
|
|
|
|
|
return locale_string(lang, key);
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
#define crtag(x) (x)
|
|
|
|
|
#endif
|
2001-04-16 16:34:19 +02:00
|
|
|
|
/*
|
|
|
|
|
* translation table
|
|
|
|
|
*/
|
|
|
|
|
typedef struct translation {
|
|
|
|
|
struct translation * next;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
char * key;
|
2001-04-16 16:34:19 +02:00
|
|
|
|
const char * value;
|
|
|
|
|
} translation;
|
|
|
|
|
|
2004-05-26 08:42:58 +02:00
|
|
|
|
#define TRANSMAXHASH 257
|
2001-04-16 16:34:19 +02:00
|
|
|
|
static translation * translation_table[TRANSMAXHASH];
|
|
|
|
|
static translation * junkyard;
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
add_translation(const char * key, const char * value)
|
|
|
|
|
{
|
|
|
|
|
int kk = ((key[0] << 5) + key[0]) % TRANSMAXHASH;
|
|
|
|
|
translation * t = translation_table[kk];
|
|
|
|
|
while (t && strcmp(t->key, key)!=0) t=t->next;
|
|
|
|
|
if (!t) {
|
|
|
|
|
if (junkyard) {
|
|
|
|
|
t = junkyard;
|
|
|
|
|
junkyard = junkyard->next;
|
|
|
|
|
} else t = malloc(sizeof(translation));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
t->key = strdup(key);
|
2001-04-16 16:34:19 +02:00
|
|
|
|
t->value = value;
|
|
|
|
|
t->next = translation_table[kk];
|
|
|
|
|
translation_table[kk] = t;
|
|
|
|
|
}
|
2001-04-21 13:07:54 +02:00
|
|
|
|
return crtag(key);
|
2001-04-16 16:34:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
write_translations(FILE * F)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
fputs("TRANSLATION\n", F);
|
|
|
|
|
for (i=0;i!=TRANSMAXHASH;++i) {
|
|
|
|
|
translation * t = translation_table[i];
|
|
|
|
|
while (t) {
|
2001-04-22 07:36:50 +02:00
|
|
|
|
fprintf(F, "\"%s\";%s\n", t->value, crtag(t->key));
|
2001-04-16 16:34:19 +02:00
|
|
|
|
t = t->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
reset_translations(void)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
for (i=0;i!=TRANSMAXHASH;++i) {
|
|
|
|
|
translation * t = translation_table[i];
|
|
|
|
|
while (t) {
|
|
|
|
|
translation * c = t->next;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
free(t->key);
|
2001-04-16 16:34:19 +02:00
|
|
|
|
t->next = junkyard;
|
|
|
|
|
junkyard = t;
|
|
|
|
|
t = c;
|
|
|
|
|
}
|
|
|
|
|
translation_table[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
- 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
|
|
|
|
/* implementation */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cr_output_str_list(FILE * F, const char *title, const strlist * S, const faction * f)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
if (!S) return;
|
|
|
|
|
|
|
|
|
|
fprintf(F, "%s\n", title);
|
|
|
|
|
while (S) {
|
2005-01-19 21:33:13 +01:00
|
|
|
|
fprintf(F, "\"%s\"\n", S->s);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
S = S->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#include "objtypes.h"
|
|
|
|
|
|
|
|
|
|
static void
|
2002-05-10 18:49:31 +02:00
|
|
|
|
print_curses(FILE * F, const faction * viewer, const void * obj, typ_t typ)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
boolean header = false;
|
2002-05-10 18:49:31 +02:00
|
|
|
|
attrib *a = NULL;
|
|
|
|
|
int self = 0;
|
|
|
|
|
region *r;
|
|
|
|
|
|
|
|
|
|
/* Die Sichtbarkeit eines Zaubers und die Zaubermeldung sind bei
|
|
|
|
|
* Geb<EFBFBD>uden und Schiffen je nach, ob man Besitzer ist, verschieden.
|
|
|
|
|
* Bei Einheiten sieht man Wirkungen auf eigene Einheiten immer.
|
|
|
|
|
* Spezialf<EFBFBD>lle (besonderes Talent, verursachender Magier usw. werde
|
|
|
|
|
* bei jedem curse gesondert behandelt. */
|
|
|
|
|
if (typ == TYP_SHIP){
|
|
|
|
|
ship * sh = (ship*)obj;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
unit * owner = shipowner(sh);
|
2002-05-10 18:49:31 +02:00
|
|
|
|
a = sh->attribs;
|
|
|
|
|
r = sh->region;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if(owner != NULL) {
|
2002-05-10 18:49:31 +02:00
|
|
|
|
if (owner->faction == viewer){
|
|
|
|
|
self = 2;
|
|
|
|
|
} else { /* steht eine person der Partei auf dem Schiff? */
|
|
|
|
|
unit *u = NULL;
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->ship == sh) {
|
|
|
|
|
self = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (typ == TYP_BUILDING) {
|
|
|
|
|
building * b = (building*)obj;
|
|
|
|
|
unit * owner;
|
|
|
|
|
a = b->attribs;
|
|
|
|
|
r = b->region;
|
|
|
|
|
if((owner = buildingowner(r,b)) != NULL){
|
|
|
|
|
if (owner->faction == viewer){
|
|
|
|
|
self = 2;
|
|
|
|
|
} else { /* steht eine Person der Partei in der Burg? */
|
|
|
|
|
unit *u = NULL;
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->building == b) {
|
|
|
|
|
self = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (typ == TYP_UNIT) {
|
|
|
|
|
unit *u = (unit *)obj;
|
|
|
|
|
a = u->attribs;
|
|
|
|
|
r = u->region;
|
|
|
|
|
if (u->faction == viewer){
|
|
|
|
|
self = 2;
|
|
|
|
|
}
|
|
|
|
|
} else if (typ == TYP_REGION) {
|
|
|
|
|
r = (region *)obj;
|
|
|
|
|
a = r->attribs;
|
|
|
|
|
} else {
|
|
|
|
|
/* fehler */
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
while (a) {
|
|
|
|
|
int dh = 0;
|
|
|
|
|
curse *c;
|
|
|
|
|
|
|
|
|
|
if (fval(a->type, ATF_CURSE)) {
|
|
|
|
|
|
|
|
|
|
c = (curse *)a->data.v;
|
2004-07-08 00:02:42 +02:00
|
|
|
|
if (c->type->curseinfo) {
|
|
|
|
|
if (c->type->cansee) {
|
2002-05-10 19:51:20 +02:00
|
|
|
|
self = c->type->cansee(viewer, obj, typ, c, self);
|
|
|
|
|
}
|
2002-05-10 18:49:31 +02:00
|
|
|
|
dh = c->type->curseinfo(viewer->locale, obj, typ, c, self);
|
2004-07-08 00:02:42 +02:00
|
|
|
|
}
|
|
|
|
|
if (dh==0) {
|
|
|
|
|
if (c->type->info_str!=NULL) {
|
|
|
|
|
strcpy(buf, c->type->info_str);
|
|
|
|
|
} else {
|
|
|
|
|
sprintf(buf, "an unknown curse lies on the region. (%s)", itoa36(c->no));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (dh==1) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (!header) {
|
|
|
|
|
header = 1;
|
|
|
|
|
fputs("EFFECTS\n", F);
|
|
|
|
|
}
|
|
|
|
|
fprintf(F, "\"%s\"\n", buf);
|
|
|
|
|
}
|
2001-03-04 19:41:27 +01:00
|
|
|
|
} else if (a->type==&at_effect && self) {
|
|
|
|
|
effect_data * data = (effect_data *)a->data.v;
|
2001-04-16 16:34:19 +02:00
|
|
|
|
const char * key = resourcename(data->type->itype->rtype, 0);
|
2001-03-04 19:41:27 +01:00
|
|
|
|
if (!header) {
|
|
|
|
|
header = 1;
|
|
|
|
|
fputs("EFFECTS\n", F);
|
|
|
|
|
}
|
2001-05-20 10:41:11 +02:00
|
|
|
|
fprintf(F, "\"%d %s\"\n", data->value, add_translation(key, locale_string(default_locale, key)));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
a = a->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
static int
|
|
|
|
|
cr_unit(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
unit * u = (unit *)v;
|
|
|
|
|
sprintf(buffer, "%d", u?u->no:-1);
|
|
|
|
|
unused(report);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2001-02-24 13:50:51 +01:00
|
|
|
|
|
- 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
|
|
|
|
static int
|
|
|
|
|
cr_ship(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
ship * u = (ship *)v;
|
|
|
|
|
sprintf(buffer, "%d", u?u->no:-1);
|
|
|
|
|
unused(report);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2001-02-24 13:50:51 +01:00
|
|
|
|
|
- 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
|
|
|
|
static int
|
|
|
|
|
cr_building(const void * v, char * buffer, const void * userdata)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
- 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
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
building * u = (building *)v;
|
|
|
|
|
sprintf(buffer, "%d", u?u->no:-1);
|
|
|
|
|
unused(report);
|
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
- 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
|
|
|
|
static int
|
|
|
|
|
cr_faction(const void * v, char * buffer, const void * userdata)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
- 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
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
faction * f = (faction *)v;
|
|
|
|
|
sprintf(buffer, "%d", f?f->no:-1);
|
|
|
|
|
unused(report);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
cr_region(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
region * r = (region *)v;
|
|
|
|
|
if (r) {
|
|
|
|
|
plane * p = rplane(r);
|
|
|
|
|
if (!p || !(p->flags & PFL_NOCOORDS)) {
|
|
|
|
|
sprintf(buffer, "%d %d %d", region_x(r, report), region_y(r, report), p?p->id:0);
|
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
- 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
|
|
|
|
return -1;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
- 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
|
|
|
|
static int
|
|
|
|
|
cr_resource(const void * v, char * buffer, const void * userdata)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
- 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
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
const resource_type * r = (const resource_type *)v;
|
|
|
|
|
if (r) {
|
2001-04-16 16:34:19 +02:00
|
|
|
|
const char * key = resourcename(r, 0);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
sprintf(buffer, "\"%s\"",
|
2001-04-16 16:34:19 +02:00
|
|
|
|
add_translation(key, locale_string(report->locale, key)));
|
- 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
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
- 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
|
|
|
|
return -1;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-04-16 16:34:19 +02:00
|
|
|
|
static int
|
|
|
|
|
cr_race(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
const faction * report = (const faction*)userdata;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
const struct race * rc = (const race *)v;
|
|
|
|
|
const char * key = rc_name(rc, 0);
|
|
|
|
|
sprintf(buffer, "\"%s\"",
|
2001-04-16 16:34:19 +02:00
|
|
|
|
add_translation(key, locale_string(report->locale, key)));
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static int
|
|
|
|
|
cr_alliance(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
const alliance * al = (const alliance *)v;
|
|
|
|
|
if (al!=NULL) {
|
|
|
|
|
sprintf(buffer, "%d", al->id);
|
|
|
|
|
}
|
|
|
|
|
unused(userdata);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
- 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
|
|
|
|
static int
|
|
|
|
|
cr_skill(const void * v, char * buffer, const void * userdata)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-04-20 00:52:36 +02:00
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
skill_t sk = (skill_t)(int)v;
|
|
|
|
|
if (sk!=NOSKILL) sprintf(buffer, "\"%s\"",
|
|
|
|
|
add_translation(skillname(sk, NULL), skillname(sk, report->locale)));
|
|
|
|
|
else strcpy(buffer, "\"\"");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2004-06-27 11:08:22 +02:00
|
|
|
|
static int
|
|
|
|
|
cr_order(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
order * ord = (order*)v;
|
|
|
|
|
if (ord!=NULL) {
|
2004-07-02 07:41:47 +02:00
|
|
|
|
char * wp = buffer;
|
2004-06-27 11:08:22 +02:00
|
|
|
|
char * cmd = getcommand(ord);
|
2004-07-02 07:41:47 +02:00
|
|
|
|
const char * rp = cmd;
|
2004-07-03 13:36:34 +02:00
|
|
|
|
|
2004-07-02 07:41:47 +02:00
|
|
|
|
*wp++ = '\"';
|
|
|
|
|
while (*rp) {
|
|
|
|
|
switch (*rp) {
|
2004-07-03 13:36:34 +02:00
|
|
|
|
case '\"':
|
|
|
|
|
case '\\':
|
|
|
|
|
*wp++ = '\\';
|
|
|
|
|
default:
|
|
|
|
|
*wp++ = *rp++;
|
2004-07-02 07:41:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*wp++ = '\"';
|
|
|
|
|
*wp++ = 0;
|
2004-07-03 13:36:34 +02:00
|
|
|
|
/* sprintf(buffer, "\"%s\"", cmd); */
|
2004-06-27 11:08:22 +02:00
|
|
|
|
}
|
|
|
|
|
else strcpy(buffer, "\"\"");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2004-04-20 00:52:36 +02:00
|
|
|
|
static int
|
|
|
|
|
cr_spell(const void * v, char * buffer, const void * userdata)
|
|
|
|
|
{
|
|
|
|
|
const faction * report = (const faction*)userdata;
|
|
|
|
|
spell * sp = (spell*)v;
|
|
|
|
|
if (sp!=NULL) sprintf(buffer, "\"%s\"", spell_name(sp, report->locale));
|
|
|
|
|
else strcpy(buffer, "\"\"");
|
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
void
|
- 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
|
|
|
|
creport_init(void)
|
|
|
|
|
{
|
2004-06-27 11:08:22 +02:00
|
|
|
|
tsf_register("report", &cr_ignore);
|
|
|
|
|
tsf_register("string", &cr_string);
|
|
|
|
|
tsf_register("order", &cr_order);
|
|
|
|
|
tsf_register("spell", &cr_spell);
|
|
|
|
|
tsf_register("int", &cr_int);
|
|
|
|
|
tsf_register("unit", &cr_unit);
|
|
|
|
|
tsf_register("region", &cr_region);
|
|
|
|
|
tsf_register("faction", &cr_faction);
|
|
|
|
|
tsf_register("ship", &cr_ship);
|
|
|
|
|
tsf_register("building", &cr_building);
|
|
|
|
|
tsf_register("skill", &cr_skill);
|
|
|
|
|
tsf_register("resource", &cr_resource);
|
|
|
|
|
tsf_register("race", &cr_race);
|
|
|
|
|
tsf_register("direction", &cr_int);
|
|
|
|
|
tsf_register("alliance", &cr_alliance);
|
- 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
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
- 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
|
|
|
|
void
|
|
|
|
|
creport_cleanup(void)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-27 11:08:22 +02:00
|
|
|
|
while (junkyard) {
|
|
|
|
|
translation * t = junkyard;
|
|
|
|
|
junkyard = junkyard->next;
|
|
|
|
|
free(t);
|
|
|
|
|
}
|
|
|
|
|
junkyard = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2004-05-04 21:34:35 +02:00
|
|
|
|
/*static int msgno; */
|
- 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
|
|
|
|
|
2004-05-26 08:42:58 +02:00
|
|
|
|
#define MTMAXHASH 1021
|
- 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
|
|
|
|
|
|
|
|
|
static struct known_mtype {
|
2004-06-27 11:08:22 +02:00
|
|
|
|
const struct message_type * mtype;
|
|
|
|
|
struct known_mtype * nexthash;
|
- 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
|
|
|
|
} * mtypehash[MTMAXHASH];
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
static void
|
- 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
|
|
|
|
report_crtypes(FILE * F, const struct locale* lang)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
int i;
|
2001-04-22 07:36:50 +02:00
|
|
|
|
for (i=0;i!=MTMAXHASH;++i) {
|
|
|
|
|
struct known_mtype * kmt;
|
|
|
|
|
for (kmt=mtypehash[i];kmt;kmt=kmt->nexthash) {
|
|
|
|
|
const struct nrmessage_type * nrt = nrt_find(lang, kmt->mtype);
|
|
|
|
|
if (nrt) {
|
|
|
|
|
unsigned int hash = hashstring(mt_name(kmt->mtype));
|
2004-07-03 13:36:34 +02:00
|
|
|
|
fprintf(F, "MESSAGETYPE %d\n", hash);
|
|
|
|
|
fputc('\"', F);
|
2001-04-22 07:36:50 +02:00
|
|
|
|
fputs(escape_string(nrt_string(nrt), NULL, 0), F);
|
|
|
|
|
fputs("\";text\n", F);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
fprintf(F, "\"%s\";section\n", nrt_section(nrt));
|
2001-04-22 07:36:50 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (mtypehash[i]) {
|
|
|
|
|
kmt = mtypehash[i];
|
|
|
|
|
mtypehash[i] = mtypehash[i]->nexthash;
|
|
|
|
|
free(kmt);
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-02-24 13:50:51 +01:00
|
|
|
|
static void
|
|
|
|
|
render_messages(FILE * F, faction * f, message_list *msgs)
|
|
|
|
|
{
|
|
|
|
|
struct mlist* m = msgs->begin;
|
|
|
|
|
while (m) {
|
2001-05-20 12:44:37 +02:00
|
|
|
|
char crbuffer[1024*32]; /* gross, wegen spionage-messages :-( */
|
|
|
|
|
boolean printed = false;
|
|
|
|
|
const struct message_type * mtype = m->msg->type;
|
|
|
|
|
unsigned int hash = hashstring(mtype->name);
|
- 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
|
|
|
|
#if RENDER_CRMESSAGES
|
2001-05-20 12:44:37 +02:00
|
|
|
|
char nrbuffer[1024*32];
|
|
|
|
|
nrbuffer[0] = '\0';
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (nr_render(m->msg, f->locale, nrbuffer, sizeof(nrbuffer), f)==0 && nrbuffer[0]) {
|
2004-04-13 23:58:46 +02:00
|
|
|
|
fprintf(F, "MESSAGE %u\n", (unsigned int)m->msg);/*++msgno); */
|
2001-05-20 12:44:37 +02:00
|
|
|
|
fprintf(F, "%d;type\n", hash);
|
2004-07-02 21:08:53 +02:00
|
|
|
|
fwritestr(F, nrbuffer);
|
|
|
|
|
fputs(";rendered\n", F);
|
2001-05-20 12:44:37 +02:00
|
|
|
|
printed = true;
|
|
|
|
|
}
|
- 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
|
|
|
|
#endif
|
2001-05-20 12:44:37 +02:00
|
|
|
|
crbuffer[0] = '\0';
|
2002-01-20 14:00:52 +01:00
|
|
|
|
if (cr_render(m->msg, crbuffer, (const void*)f)==0) {
|
2004-04-13 23:58:46 +02:00
|
|
|
|
if (!printed) fprintf(F, "MESSAGE %u\n", (unsigned int)m->msg);/*++msgno); */
|
2002-01-20 14:00:52 +01:00
|
|
|
|
if (crbuffer[0]) fputs(crbuffer, F);
|
|
|
|
|
} else {
|
|
|
|
|
log_error(("could not render cr-message %p: %s\n", m->msg, m->msg->type->name));
|
2001-05-20 12:44:37 +02:00
|
|
|
|
}
|
|
|
|
|
if (printed) {
|
|
|
|
|
unsigned int ihash = hash % MTMAXHASH;
|
|
|
|
|
struct known_mtype * kmt = mtypehash[ihash];
|
|
|
|
|
while (kmt && kmt->mtype != mtype) kmt = kmt->nexthash;
|
|
|
|
|
if (kmt==NULL) {
|
|
|
|
|
kmt = (struct known_mtype*)malloc(sizeof(struct known_mtype));
|
|
|
|
|
kmt->nexthash = mtypehash[ihash];
|
|
|
|
|
kmt->mtype = mtype;
|
|
|
|
|
mtypehash[ihash] = kmt;
|
- 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
|
|
|
|
}
|
|
|
|
|
}
|
2001-02-24 13:50:51 +01:00
|
|
|
|
m = m->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
cr_output_messages(FILE * F, message_list *msgs, faction * f)
|
|
|
|
|
{
|
- 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
|
|
|
|
if (msgs) render_messages(F, f, msgs);
|
2001-02-24 13:50:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* prints a building */
|
|
|
|
|
static void
|
|
|
|
|
cr_output_buildings(FILE * F, building * b, unit * u, int fno, faction *f)
|
|
|
|
|
{
|
|
|
|
|
const building_type * type = b->type;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
const char * bname = buildingtype(b, b->size);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
fprintf(F, "BURG %d\n", b->no);
|
|
|
|
|
if (!u || u->faction != f) {
|
|
|
|
|
const attrib * a = a_find(b->attribs, &at_icastle);
|
|
|
|
|
if (a) type = ((icastle_data*)a->data.v)->type;
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
fprintf(F, "\"%s\";Typ\n", add_translation(bname, LOC(f->locale, bname)));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "\"%s\";Name\n", b->name);
|
2005-04-24 00:52:49 +02:00
|
|
|
|
if (b->display && strlen(b->display))
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "\"%s\";Beschr\n", b->display);
|
|
|
|
|
if (b->size)
|
|
|
|
|
fprintf(F, "%d;Groesse\n", b->size);
|
|
|
|
|
if (u)
|
|
|
|
|
fprintf(F, "%d;Besitzer\n", u ? u->no : -1);
|
|
|
|
|
if (fno >= 0)
|
|
|
|
|
fprintf(F, "%d;Partei\n", fno);
|
|
|
|
|
#ifdef TODO
|
|
|
|
|
int cost = buildingdaten[b->type].per_size * b->size + buildingdaten[b->type].unterhalt;
|
|
|
|
|
if (u && u->faction == f && cost)
|
|
|
|
|
fprintf(F, "%d;Unterhalt\n", cost);
|
|
|
|
|
#endif
|
|
|
|
|
if (b->besieged)
|
|
|
|
|
fprintf(F, "%d;Belagerer\n", b->besieged);
|
2002-05-10 18:49:31 +02:00
|
|
|
|
print_curses(F, f, b, TYP_BUILDING);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
|
|
|
|
|
|
|
|
|
|
/* prints a ship */
|
|
|
|
|
static void
|
2005-04-17 20:25:11 +02:00
|
|
|
|
cr_output_ship(FILE * F, const ship * sh, const unit * u, int fcaptain, const faction * f, const region * r)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
int w = 0;
|
2005-04-17 20:25:11 +02:00
|
|
|
|
assert(sh);
|
|
|
|
|
fprintf(F, "SCHIFF %d\n", sh->no);
|
|
|
|
|
fprintf(F, "\"%s\";Name\n", sh->name);
|
2005-04-24 00:52:49 +02:00
|
|
|
|
if (sh->display && strlen(sh->display))
|
2005-04-17 20:25:11 +02:00
|
|
|
|
fprintf(F, "\"%s\";Beschr\n", sh->display);
|
|
|
|
|
fprintf(F, "\"%s\";Typ\n", add_translation(sh->type->name[0], locale_string(f->locale, sh->type->name[0])));
|
|
|
|
|
fprintf(F, "%d;Groesse\n", sh->size);
|
|
|
|
|
if (sh->damage) {
|
|
|
|
|
int percent = sh->damage*100/(sh->size*DAMAGE_SCALE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Schaden\n", percent);
|
|
|
|
|
}
|
|
|
|
|
if (u)
|
|
|
|
|
fprintf(F, "%d;Kapitaen\n", u ? u->no : -1);
|
|
|
|
|
if (fcaptain >= 0)
|
|
|
|
|
fprintf(F, "%d;Partei\n", fcaptain);
|
|
|
|
|
|
|
|
|
|
/* calculate cargo */
|
2001-04-01 08:58:45 +02:00
|
|
|
|
if (u && (u->faction == f || omniscient(f))) {
|
2005-04-17 20:25:11 +02:00
|
|
|
|
int n = 0, p = 0;
|
|
|
|
|
getshipweight(sh, &n, &p);
|
|
|
|
|
n = (n+99) / 100; /* 1 Silber = 1 GE */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-04-17 20:25:11 +02:00
|
|
|
|
fprintf(F, "%d;Ladung\n", n);
|
|
|
|
|
fprintf(F, "%d;MaxLadung\n", shipcapacity(sh) / 100);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* shore */
|
|
|
|
|
w = NODIRECTION;
|
2005-04-17 20:25:11 +02:00
|
|
|
|
if (rterrain(r) != T_OCEAN) w = sh->coast;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (w != NODIRECTION)
|
|
|
|
|
fprintf(F, "%d;Kueste\n", w);
|
|
|
|
|
|
2005-04-17 20:25:11 +02:00
|
|
|
|
print_curses(F, f, sh, TYP_SHIP);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
|
|
|
|
|
|
|
|
|
|
/* prints all that belongs to a unit */
|
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cr_output_unit(FILE * F, const region * r,
|
|
|
|
|
const faction * f, /* observers faction */
|
|
|
|
|
const unit * u, int mode)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
/* Race attributes are always plural and item attributes always
|
|
|
|
|
* singular */
|
|
|
|
|
const item_type * lasttype;
|
|
|
|
|
int pr;
|
|
|
|
|
item *itm, *show;
|
|
|
|
|
boolean itemcloak = is_cursed(u->attribs, C_ITEMCLOAK, 0);
|
|
|
|
|
building * b;
|
- 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
|
|
|
|
const char * pzTmp;
|
2004-02-16 20:59:26 +01:00
|
|
|
|
skill * sv;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
const attrib *a_fshidden = NULL;
|
|
|
|
|
|
|
|
|
|
assert(u);
|
|
|
|
|
|
|
|
|
|
if(fspecial(u->faction, FS_HIDDEN))
|
|
|
|
|
a_fshidden = a_find(u->attribs, &at_fshidden);
|
|
|
|
|
|
|
|
|
|
fprintf(F, "EINHEIT %d\n", u->no);
|
|
|
|
|
fprintf(F, "\"%s\";Name\n", u->name);
|
2005-04-24 00:52:49 +02:00
|
|
|
|
if (u->display && strlen(u->display))
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "\"%s\";Beschr\n", u->display);
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
|
|
|
|
/* print faction information */
|
2003-09-21 10:20:58 +02:00
|
|
|
|
const faction * sf = visible_faction(f, u);
|
|
|
|
|
const char * prefix = raceprefix(u);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (u->faction == f || omniscient(f)) {
|
2005-02-05 20:05:47 +01:00
|
|
|
|
const attrib * a_otherfaction = a_find(u->attribs, &at_otherfaction);
|
2002-03-24 10:40:50 +01:00
|
|
|
|
const faction * otherfaction = a_otherfaction?get_otherfaction(a_otherfaction):NULL;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* my own faction, full info */
|
2003-09-21 10:20:58 +02:00
|
|
|
|
const attrib *a = a_find(u->attribs, &at_group);
|
|
|
|
|
if (a!=NULL) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
const group * g = (const group*)a->data.v;
|
|
|
|
|
fprintf(F, "%d;gruppe\n", g->gid);
|
2002-03-10 11:09:16 +01:00
|
|
|
|
}
|
2001-04-14 14:11:45 +02:00
|
|
|
|
fprintf(F, "%d;Partei\n", u->faction->no);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (sf!=u->faction) fprintf(F, "%d;Verkleidung\n", sf->no);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (fval(u, UFL_PARTEITARNUNG))
|
|
|
|
|
fprintf(F, "%d;Parteitarnung\n", i2b(fval(u, UFL_PARTEITARNUNG)));
|
2005-02-05 20:05:47 +01:00
|
|
|
|
if (otherfaction) {
|
|
|
|
|
if (otherfaction!=u->faction) {
|
|
|
|
|
fprintf(F, "%d;Anderepartei\n", otherfaction->no);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (fval(u, UFL_PARTEITARNUNG)) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* faction info is hidden */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
fprintf(F, "%d;Parteitarnung\n", i2b(fval(u, UFL_PARTEITARNUNG)));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
} else {
|
2005-02-05 20:05:47 +01:00
|
|
|
|
const attrib * a_otherfaction = a_find(u->attribs, &at_otherfaction);
|
2003-01-28 00:09:44 +01:00
|
|
|
|
const faction * otherfaction = a_otherfaction?get_otherfaction(a_otherfaction):NULL;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* other unit. show visible faction, not u->faction */
|
|
|
|
|
fprintf(F, "%d;Partei\n", sf->no);
|
|
|
|
|
if (sf == f) {
|
|
|
|
|
fprintf(F, "1;Verraeter\n");
|
|
|
|
|
}
|
2005-02-05 20:05:47 +01:00
|
|
|
|
if (a_otherfaction) {
|
|
|
|
|
if (otherfaction!=u->faction) {
|
|
|
|
|
if (alliedunit(u, f, HELP_FSTEALTH)) {
|
|
|
|
|
fprintf(F, "%d;Anderepartei\n", otherfaction->no);
|
|
|
|
|
}
|
|
|
|
|
}
|
2003-01-28 00:09:44 +01:00
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2002-03-10 11:09:16 +01:00
|
|
|
|
}
|
2003-09-21 10:20:58 +02:00
|
|
|
|
if (prefix) {
|
2004-08-07 09:42:22 +02:00
|
|
|
|
prefix = mkname("prefix", prefix);
|
2003-09-21 10:20:58 +02:00
|
|
|
|
fprintf(F, "\"%s\";typprefix\n", add_translation(prefix, LOC(f->locale, prefix)));
|
2001-04-14 14:11:45 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2002-03-10 11:09:16 +01:00
|
|
|
|
if (u->faction != f && a_fshidden
|
2001-03-07 16:00:18 +01:00
|
|
|
|
&& a_fshidden->data.ca[0] == 1 && effskill(u, SK_STEALTH) >= 6) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "-1;Anzahl\n");
|
|
|
|
|
} else {
|
|
|
|
|
fprintf(F, "%d;Anzahl\n", u->number);
|
|
|
|
|
}
|
|
|
|
|
|
2004-07-11 00:31:21 +02:00
|
|
|
|
pzTmp = get_racename(u->attribs);
|
|
|
|
|
if (pzTmp) {
|
|
|
|
|
fprintf(F, "\"%s\";Typ\n", pzTmp);
|
|
|
|
|
if (u->faction==f && fval(u->race, RCF_SHAPESHIFTANY)) {
|
|
|
|
|
const char * zRace = rc_name(u->race, 1);
|
|
|
|
|
fprintf(F, "\"%s\";wahrerTyp\n",
|
|
|
|
|
add_translation(zRace, locale_string(f->locale, zRace)));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
const char * zRace = rc_name(u->irace, 1);
|
|
|
|
|
fprintf(F, "\"%s\";Typ\n",
|
|
|
|
|
add_translation(zRace, locale_string(f->locale, zRace)));
|
|
|
|
|
if (u->faction==f && u->irace!=u->race) {
|
|
|
|
|
zRace = rc_name(u->race, 1);
|
|
|
|
|
fprintf(F, "\"%s\";wahrerTyp\n",
|
|
|
|
|
add_translation(zRace, locale_string(f->locale, zRace)));
|
|
|
|
|
}
|
|
|
|
|
}
|
- 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
|
|
|
|
|
2004-07-11 00:31:21 +02:00
|
|
|
|
if (u->building)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Burg\n", u->building->no);
|
|
|
|
|
if (u->ship)
|
|
|
|
|
fprintf(F, "%d;Schiff\n", u->ship->no);
|
|
|
|
|
if (getguard(u))
|
|
|
|
|
fprintf(F, "%d;bewacht\n", getguard(u)?1:0);
|
|
|
|
|
if ((b=usiege(u))!=NULL)
|
|
|
|
|
fprintf(F, "%d;belagert\n", b->no);
|
|
|
|
|
|
|
|
|
|
/* additional information for own units */
|
2004-06-26 22:51:19 +02:00
|
|
|
|
if (u->faction == f || omniscient(f)) {
|
|
|
|
|
order * ord;
|
|
|
|
|
const char *c;
|
|
|
|
|
int i;
|
|
|
|
|
const attrib * a;
|
|
|
|
|
|
|
|
|
|
a = a_find(u->attribs, &at_follow);
|
|
|
|
|
if (a) {
|
|
|
|
|
unit * u = (unit*)a->data.v;
|
|
|
|
|
if (u) fprintf(F, "%d;folgt\n", u->no);
|
|
|
|
|
}
|
|
|
|
|
i = ualias(u);
|
|
|
|
|
if (i>0)
|
|
|
|
|
fprintf(F, "%d;temp\n", i);
|
|
|
|
|
else if (i<0)
|
|
|
|
|
fprintf(F, "%d;alias\n", -i);
|
|
|
|
|
i = get_money(u);
|
|
|
|
|
fprintf(F, "%d;Kampfstatus\n", u->status);
|
|
|
|
|
if(fval(u, UFL_NOAID)) {
|
|
|
|
|
fputs("1;unaided\n", F);
|
|
|
|
|
}
|
|
|
|
|
i = u_geteffstealth(u);
|
2004-08-03 17:34:26 +02:00
|
|
|
|
if (i >= 0) {
|
2004-06-26 22:51:19 +02:00
|
|
|
|
fprintf(F, "%d;Tarnung\n", i);
|
2004-08-03 17:34:26 +02:00
|
|
|
|
}
|
2004-06-26 22:51:19 +02:00
|
|
|
|
c = uprivate(u);
|
2004-08-03 17:34:26 +02:00
|
|
|
|
if (c) {
|
2004-06-26 22:51:19 +02:00
|
|
|
|
fprintf(F, "\"%s\";privat\n", c);
|
2004-08-03 17:34:26 +02:00
|
|
|
|
}
|
2004-06-26 22:51:19 +02:00
|
|
|
|
c = hp_status(u);
|
2004-08-03 17:34:26 +02:00
|
|
|
|
if (c && *c && (u->faction == f || omniscient(f))) {
|
2004-06-26 22:51:19 +02:00
|
|
|
|
fprintf(F, "\"%s\";hp\n", add_translation(c, locale_string(u->faction->locale, c)));
|
2004-08-03 17:34:26 +02:00
|
|
|
|
}
|
|
|
|
|
#ifdef HEROES
|
|
|
|
|
if (fval(u, UFL_HERO)) {
|
|
|
|
|
fputs("1;hero\n", F);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2004-12-22 01:53:12 +01:00
|
|
|
|
|
2004-08-03 17:34:26 +02:00
|
|
|
|
if (fval(u, UFL_HUNGER) && (u->faction == f)) {
|
2004-06-26 22:51:19 +02:00
|
|
|
|
fputs("1;hunger\n", F);
|
2004-08-03 17:34:26 +02:00
|
|
|
|
}
|
2004-06-26 22:51:19 +02:00
|
|
|
|
if (is_mage(u)) {
|
|
|
|
|
fprintf(F, "%d;Aura\n", get_spellpoints(u));
|
|
|
|
|
fprintf(F, "%d;Auramax\n", max_spellpoints(u->region,u));
|
|
|
|
|
}
|
|
|
|
|
/* default commands */
|
|
|
|
|
fprintf(F, "COMMANDS\n");
|
|
|
|
|
if (u->lastorder) {
|
2004-07-02 21:08:53 +02:00
|
|
|
|
fwriteorder(F, u->lastorder, f->locale);
|
|
|
|
|
fputc('\n', F);
|
2004-06-26 22:51:19 +02:00
|
|
|
|
}
|
|
|
|
|
for (ord = u->orders; ord; ord = ord->next) {
|
|
|
|
|
if (is_persistent(ord) && ord!=u->lastorder) {
|
2004-07-02 21:08:53 +02:00
|
|
|
|
fwriteorder(F, ord, f->locale);
|
|
|
|
|
fputc('\n', F);
|
2004-06-26 22:51:19 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* talents */
|
|
|
|
|
pr = 0;
|
2004-02-16 20:59:26 +01:00
|
|
|
|
for (sv = u->skills; sv != u->skills + u->skill_size; ++sv) {
|
2004-06-26 22:51:19 +02:00
|
|
|
|
if (sv->level>0) {
|
2004-02-16 20:59:26 +01:00
|
|
|
|
skill_t sk = sv->id;
|
2004-06-26 22:51:19 +02:00
|
|
|
|
int esk = eff_skill(u, sk, r);
|
|
|
|
|
if (!pr) {
|
|
|
|
|
pr = 1;
|
|
|
|
|
fprintf(F, "TALENTE\n");
|
|
|
|
|
}
|
|
|
|
|
fprintf(F, "%d %d;%s\n", u->number*level_days(sv->level), esk,
|
|
|
|
|
add_translation(skillname(sk, NULL), skillname(sk, f->locale)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* spells */
|
|
|
|
|
if (is_mage(u)) {
|
|
|
|
|
sc_mage * mage = get_mage(u);
|
|
|
|
|
spell_ptr *spt = mage->spellptr;
|
|
|
|
|
if (spt) {
|
|
|
|
|
spell *sp;
|
|
|
|
|
int i;
|
|
|
|
|
int t = effskill(u, SK_MAGIC);
|
|
|
|
|
fprintf(F, "SPRUECHE\n");
|
|
|
|
|
for (;spt; spt = spt->next) {
|
|
|
|
|
sp = find_spellbyid(spt->spellid);
|
|
|
|
|
if (sp) {
|
|
|
|
|
const char * name = sp->sname;
|
|
|
|
|
if (sp->level > t) continue;
|
|
|
|
|
if (sp->info==NULL) {
|
|
|
|
|
name = add_translation(mkname("spell", name), spell_name(sp, f->locale));
|
|
|
|
|
}
|
|
|
|
|
fprintf(F, "\"%s\"\n", name);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for (i=0;i!=MAXCOMBATSPELLS;++i) {
|
|
|
|
|
sp = find_spellbyid(mage->combatspell[i]);
|
|
|
|
|
if (sp) {
|
|
|
|
|
const char * name = sp->sname;
|
|
|
|
|
if (sp->info==NULL) {
|
|
|
|
|
name = add_translation(mkname("spell", name), spell_name(sp, f->locale));
|
|
|
|
|
}
|
|
|
|
|
fprintf(F, "KAMPFZAUBER %d\n", i);
|
|
|
|
|
fprintf(F, "\"%s\";name\n", name);
|
|
|
|
|
fprintf(F, "%d;level\n", mage->combatspelllevel[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* items */
|
|
|
|
|
pr = 0;
|
2001-04-01 08:58:45 +02:00
|
|
|
|
if (f == u->faction || omniscient(u->faction)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
show = u->items;
|
|
|
|
|
} else if (itemcloak==false && mode>=see_unit && !(a_fshidden
|
2001-03-07 16:00:18 +01:00
|
|
|
|
&& a_fshidden->data.ca[1] == 1 && effskill(u, SK_STEALTH) >= 3)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
show = NULL;
|
|
|
|
|
for (itm=u->items;itm;itm=itm->next) {
|
|
|
|
|
item * ishow;
|
|
|
|
|
const char * ic;
|
|
|
|
|
int in;
|
2001-02-17 16:52:47 +01:00
|
|
|
|
report_item(u, itm, f, NULL, &ic, &in, true);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (in>0 && ic && *ic) {
|
|
|
|
|
for (ishow = show; ishow; ishow=ishow->next) {
|
|
|
|
|
const char * sc;
|
|
|
|
|
int sn;
|
|
|
|
|
if (ishow->type==itm->type) sc=ic;
|
2001-02-17 16:52:47 +01:00
|
|
|
|
else report_item(u, ishow, f, NULL, &sc, &sn, true);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (sc==ic || strcmp(sc, ic)==0) {
|
|
|
|
|
ishow->number+=itm->number;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (ishow==NULL) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
ishow = i_add(&show, i_new(itm->type, itm->number));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
show = NULL;
|
|
|
|
|
}
|
|
|
|
|
lasttype = NULL;
|
|
|
|
|
for (itm=show; itm; itm=itm->next) {
|
|
|
|
|
const char * ic;
|
|
|
|
|
int in;
|
|
|
|
|
assert(itm->type!=lasttype || !"error: list contains two objects of the same item");
|
2001-02-17 16:52:47 +01:00
|
|
|
|
report_item(u, itm, f, NULL, &ic, &in, true);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (in==0) continue;
|
|
|
|
|
if (!pr) {
|
|
|
|
|
pr = 1;
|
|
|
|
|
fputs("GEGENSTAENDE\n", F);
|
|
|
|
|
}
|
2001-04-22 07:36:50 +02:00
|
|
|
|
fprintf(F, "%d;%s\n", in, add_translation(ic, locale_string(f->locale, ic)));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-04-01 08:58:45 +02:00
|
|
|
|
if ((u->faction == f || omniscient(f)) && u->botschaften)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cr_output_str_list(F, "EINHEITSBOTSCHAFTEN", u->botschaften, f);
|
|
|
|
|
|
2002-05-10 18:49:31 +02:00
|
|
|
|
print_curses(F, f, u, TYP_UNIT);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
|
|
|
|
|
|
|
|
|
|
/* prints allies */
|
|
|
|
|
static void
|
2002-09-02 22:36:12 +02:00
|
|
|
|
show_allies(FILE * F, const faction * f, const ally * sf)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2002-09-02 22:36:12 +02:00
|
|
|
|
for (; sf; sf = sf->next) if (sf->faction) {
|
2003-01-12 10:33:20 +01:00
|
|
|
|
int mode = alliedgroup(NULL, f, sf->faction, sf, HELP_ALL);
|
2002-11-10 10:12:52 +01:00
|
|
|
|
if (mode!=0 && sf->status>0) {
|
2002-09-02 22:36:12 +02:00
|
|
|
|
fprintf(F, "ALLIANZ %d\n", sf->faction->no);
|
|
|
|
|
fprintf(F, "\"%s\";Parteiname\n", sf->faction->name);
|
2005-02-27 15:44:21 +01:00
|
|
|
|
fprintf(F, "%d;Status\n", sf->status & HELP_ALL);
|
2002-09-02 22:36:12 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2002-12-18 18:40:21 +01:00
|
|
|
|
|
|
|
|
|
#ifdef REGIONOWNERS
|
|
|
|
|
static void
|
|
|
|
|
show_enemies(FILE * F, const faction_list* flist)
|
|
|
|
|
{
|
|
|
|
|
for (;flist!=NULL;flist=flist->next) {
|
|
|
|
|
if (flist->data) {
|
|
|
|
|
int fno = flist->data->no;
|
2003-07-29 11:48:03 +02:00
|
|
|
|
fprintf(F, "ENEMY %u\n%u;partei\n", fno, fno);
|
2002-12-18 18:40:21 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* prints all visible spells in a region */
|
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
show_active_spells(const region * r)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
char fogwall[MAXDIRECTIONS];
|
|
|
|
|
#ifdef TODO /* alte Regionszauberanzeigen umstellen */
|
|
|
|
|
unit *u;
|
|
|
|
|
int env = 0;
|
|
|
|
|
#endif
|
|
|
|
|
memset(fogwall, 0, sizeof(char) * MAXDIRECTIONS);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
/* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
|
|
|
|
|
|
|
|
|
|
/* this is a copy of laws.c->find_address output changed. */
|
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cr_find_address(FILE * F, const faction * uf, const faction_list * addresses)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2001-12-10 01:13:39 +01:00
|
|
|
|
const faction_list * flist = addresses;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (!quiet)
|
|
|
|
|
puts(" - gebe Adressen heraus (CR)");
|
2001-12-10 01:13:39 +01:00
|
|
|
|
while (flist!=NULL) {
|
|
|
|
|
const faction * f = flist->data;
|
|
|
|
|
if (uf!=f && f->no != MONSTER_FACTION) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "PARTEI %d\n", f->no);
|
|
|
|
|
fprintf(F, "\"%s\";Parteiname\n", f->name);
|
|
|
|
|
fprintf(F, "\"%s\";email\n", f->email);
|
|
|
|
|
fprintf(F, "\"%s\";banner\n", f->banner);
|
2002-10-05 16:35:04 +02:00
|
|
|
|
if (f->alliance!=NULL && f->alliance==uf->alliance) {
|
2002-09-02 22:36:12 +02:00
|
|
|
|
fprintf(F, "%d;alliance\n", f->alliance->id);
|
|
|
|
|
fprintf(F, "\"%s\";alliancename\n", f->alliance->name);
|
|
|
|
|
}
|
2003-06-22 10:38:55 +02:00
|
|
|
|
#ifdef SHORTPWDS
|
|
|
|
|
if (f->shortpwds) {
|
|
|
|
|
shortpwd * spwd = f->shortpwds;
|
|
|
|
|
while (spwd) {
|
|
|
|
|
unsigned int vacation = 0;
|
|
|
|
|
if (spwd->used) {
|
|
|
|
|
fprintf(F, "VACATION %u\n", ++vacation);
|
|
|
|
|
fprintf(F, "\"%s\";email\n", spwd->email);
|
|
|
|
|
}
|
|
|
|
|
spwd=spwd->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-09-02 22:36:12 +02:00
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
flist = flist->next;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
|
|
|
|
|
|
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cr_reportspell(FILE * F, spellid_t id, const struct locale * lang)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-01-19 00:57:43 +01:00
|
|
|
|
int k;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
spell *sp = find_spellbyid(id);
|
2002-05-01 21:08:32 +02:00
|
|
|
|
const char * name = sp->sname;
|
|
|
|
|
if (sp->info==NULL) {
|
|
|
|
|
name = add_translation(mkname("spell", name), spell_name(sp, lang));
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2002-05-01 21:08:32 +02:00
|
|
|
|
fprintf(F, "ZAUBER %d\n", hashstring(spell_name(sp, default_locale)));
|
|
|
|
|
fprintf(F, "\"%s\";name\n", name);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;level\n", sp->level);
|
|
|
|
|
fprintf(F, "%d;rank\n", sp->rank);
|
2002-05-01 21:08:32 +02:00
|
|
|
|
fprintf(F, "\"%s\";info\n", spell_info(sp, lang));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (sp->sptyp & PRECOMBATSPELL) fputs("\"precombat\";class\n", F);
|
|
|
|
|
else if (sp->sptyp & COMBATSPELL) fputs("\"combat\";class\n", F);
|
|
|
|
|
else if (sp->sptyp & POSTCOMBATSPELL) fputs("\"postcombat\";class\n", F);
|
|
|
|
|
else fputs("\"normal\";class\n", F);
|
|
|
|
|
|
|
|
|
|
if (sp->sptyp & FARCASTING) fputs("1;far\n", F);
|
|
|
|
|
if (sp->sptyp & OCEANCASTABLE) fputs("1;ocean\n", F);
|
|
|
|
|
if (sp->sptyp & ONSHIPCAST) fputs("1;ship\n", F);
|
|
|
|
|
if (!(sp->sptyp & NOTFAMILIARCAST)) fputs("1;familiar\n", F);
|
|
|
|
|
fputs("KOMPONENTEN\n", F);
|
|
|
|
|
|
|
|
|
|
for (k = 0; k < MAXINGREDIENT; k++) {
|
2004-01-19 00:57:43 +01:00
|
|
|
|
resource_t res = sp->komponenten[k][0];
|
|
|
|
|
int itemanz = sp->komponenten[k][1];
|
|
|
|
|
int costtyp = sp->komponenten[k][2];
|
|
|
|
|
if (itemanz > 0) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
const char * name = resname(res, 0);
|
|
|
|
|
fprintf(F, "%d %d;%s\n", itemanz, costtyp == SPC_LEVEL || costtyp == SPC_LINEAR,
|
|
|
|
|
add_translation(name, LOC(lang, name)));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static unsigned int
|
2001-12-10 01:13:39 +01:00
|
|
|
|
encode_region(const faction * f, const region * r) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unsigned int id;
|
|
|
|
|
char *cp, c;
|
|
|
|
|
/* obfuscation */
|
|
|
|
|
assert(sizeof(int)==sizeof(char)*4);
|
|
|
|
|
id = (((((r->x ^ f->no) % 1024) << 20) | ((r->y ^ f->no) % 1024)));
|
|
|
|
|
cp = (char*)&id;
|
|
|
|
|
c = cp[0];
|
|
|
|
|
cp[0] = cp[2];
|
|
|
|
|
cp[2] = cp[1];
|
|
|
|
|
cp[1] = cp[3];
|
|
|
|
|
cp[3] = c;
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
static char *
|
2002-09-02 22:36:12 +02:00
|
|
|
|
report_resource(char * buf, const char * name, const struct locale * loc, int amount, int level)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
|
|
|
|
buf += sprintf(buf, "RESOURCE %u\n", hashstring(name));
|
|
|
|
|
buf += sprintf(buf, "\"%s\";type\n", add_translation(name, LOC(loc, name)));
|
|
|
|
|
if (amount>=0) {
|
|
|
|
|
if (level>=0) buf += sprintf(buf, "%d;skill\n", level);
|
|
|
|
|
buf += sprintf(buf, "%d;number\n", amount);
|
|
|
|
|
}
|
|
|
|
|
return buf;
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-07 02:44:01 +02:00
|
|
|
|
static void
|
|
|
|
|
cr_borders(const region * r, const faction * f, int seemode, FILE * F)
|
|
|
|
|
{
|
|
|
|
|
direction_t d;
|
|
|
|
|
int g = 0;
|
|
|
|
|
for (d = 0; d != MAXDIRECTIONS; d++)
|
|
|
|
|
{ /* Nachbarregionen, die gesehen werden, ermitteln */
|
|
|
|
|
const region * r2 = rconnect(r, d);
|
|
|
|
|
const border * b;
|
|
|
|
|
if (!r2) continue;
|
|
|
|
|
if (seemode==see_neighbour) {
|
|
|
|
|
seen_region * sr = find_seen(r2);
|
|
|
|
|
if (sr==NULL || sr->mode<=see_neighbour) continue;
|
|
|
|
|
}
|
|
|
|
|
b = get_borders(r, r2);
|
|
|
|
|
while (b) {
|
|
|
|
|
boolean cs = b->type->fvisible(b, f, r);
|
|
|
|
|
|
|
|
|
|
if (!cs) {
|
|
|
|
|
cs = b->type->rvisible(b, r);
|
|
|
|
|
if (!cs) {
|
|
|
|
|
unit * us = r->units;
|
|
|
|
|
while (us && !cs) {
|
|
|
|
|
if (us->faction==f) {
|
|
|
|
|
cs = b->type->uvisible(b, us);
|
|
|
|
|
if (cs) break;
|
|
|
|
|
}
|
|
|
|
|
us=us->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (cs) {
|
|
|
|
|
fprintf(F, "GRENZE %d\n", ++g);
|
|
|
|
|
fprintf(F, "\"%s\";typ\n", b->type->name(b, r, f, GF_NONE));
|
|
|
|
|
fprintf(F, "%d;richtung\n", d);
|
|
|
|
|
if (!b->type->transparent(b, f)) fputs("1;opaque\n", F);
|
|
|
|
|
/* pfusch: */
|
|
|
|
|
if (b->type==&bt_road) {
|
|
|
|
|
int p = rroad(r, d)*100/terrain[rterrain(r)].roadreq;
|
|
|
|
|
fprintf(F, "%d;prozent\n", p);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
b = b->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2004-01-25 10:58:58 +01:00
|
|
|
|
void
|
|
|
|
|
get_seen_interval(region ** first, region ** last)
|
|
|
|
|
{
|
|
|
|
|
region * r = regions;
|
|
|
|
|
while (r!=NULL) {
|
|
|
|
|
if (find_seen(r)!=NULL) {
|
2004-01-25 11:02:12 +01:00
|
|
|
|
*first = r;
|
2004-01-25 10:58:58 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
2004-01-25 11:00:31 +01:00
|
|
|
|
r = r->next;
|
2004-01-25 10:58:58 +01:00
|
|
|
|
}
|
|
|
|
|
while (r!=NULL) {
|
|
|
|
|
if (find_seen(r)!=NULL) {
|
2004-02-01 21:16:25 +01:00
|
|
|
|
*last = r->next;
|
2004-01-25 10:58:58 +01:00
|
|
|
|
}
|
2004-01-25 11:00:31 +01:00
|
|
|
|
r = r->next;
|
2004-01-25 10:58:58 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* main function of the creport. creates the header and traverses all regions */
|
2005-04-24 23:49:51 +02:00
|
|
|
|
int
|
2004-01-19 00:57:43 +01:00
|
|
|
|
report_computer(FILE * F, faction * f, const faction_list * addresses,
|
|
|
|
|
const time_t report_time)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
int i;
|
2004-12-22 01:53:12 +01:00
|
|
|
|
item * itm;
|
2004-07-09 21:14:10 +02:00
|
|
|
|
const char * prefix;
|
2004-01-19 00:57:43 +01:00
|
|
|
|
region * r;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
building *b;
|
|
|
|
|
ship *sh;
|
|
|
|
|
unit *u;
|
2005-01-03 22:28:57 +01:00
|
|
|
|
int score = 0, avgscore = 0;
|
2003-07-07 13:06:28 +02:00
|
|
|
|
const char * mailto = locale_string(f->locale, "mailto");
|
2004-01-25 11:02:12 +01:00
|
|
|
|
region * first = NULL, * last = NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
const attrib * a;
|
|
|
|
|
|
2004-01-25 10:58:58 +01:00
|
|
|
|
get_seen_interval(&first, &last);
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = */
|
|
|
|
|
/* initialisations, header and lists */
|
|
|
|
|
|
2005-04-25 18:31:36 +02:00
|
|
|
|
fprintf(stdout, "Reports f<>r %s: CR\r", factionname(f));
|
|
|
|
|
fflush(stdout);
|
- 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
|
|
|
|
|
2005-04-25 18:31:36 +02:00
|
|
|
|
fprintf(F, "VERSION %d\n", C_REPORT_VERSION);
|
|
|
|
|
fprintf(F, "\"%s\";locale\n", locale_name(f->locale));
|
2002-09-02 22:36:12 +02:00
|
|
|
|
fprintf(F, "%d;noskillpoints\n", 1);
|
2001-04-26 19:41:06 +02:00
|
|
|
|
fprintf(F, "%ld;date\n", report_time);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "\"%s\";Spiel\n", global.gamename);
|
|
|
|
|
fprintf(F, "\"%s\";Konfiguration\n", "Standard");
|
|
|
|
|
fprintf(F, "\"%s\";Koordinaten\n", "Hex");
|
|
|
|
|
fprintf(F, "%d;Basis\n", 36);
|
|
|
|
|
fprintf(F, "%d;Runde\n", turn);
|
|
|
|
|
fputs("2;Zeitalter\n", F);
|
2003-07-07 13:06:28 +02:00
|
|
|
|
if (mailto!=NULL) {
|
|
|
|
|
fprintf(F, "\"%s\";mailto\n", mailto);
|
2003-07-16 11:02:21 +02:00
|
|
|
|
fprintf(F, "\"%s\";mailcmd\n", locale_string(f->locale, "mailcmd"));
|
2003-07-07 13:06:28 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "PARTEI %d\n", f->no);
|
|
|
|
|
fprintf(F, "\"%s\";locale\n", locale_name(f->locale));
|
|
|
|
|
fprintf(F, "%d;Optionen\n", f->options);
|
|
|
|
|
if (f->options & want(O_SCORE) && f->age>DISPLAYSCORE) {
|
2005-01-03 22:28:57 +01:00
|
|
|
|
score = f->score;
|
|
|
|
|
avgscore = average_score_of_age(f->age, f->age / 24 + 1);
|
|
|
|
|
}
|
|
|
|
|
fprintf(F, "%d;Punkte\n", score);
|
|
|
|
|
fprintf(F, "%d;Punktedurchschnitt\n", avgscore);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
|
|
|
|
const char * zRace = rc_name(f->race, 1);
|
|
|
|
|
fprintf(F, "\"%s\";Typ\n", add_translation(zRace, LOC(f->locale, zRace)));
|
|
|
|
|
}
|
2004-07-09 21:14:10 +02:00
|
|
|
|
prefix = get_prefix(f->attribs);
|
|
|
|
|
if (prefix!=NULL) {
|
2004-08-07 09:42:22 +02:00
|
|
|
|
prefix = mkname("prefix", prefix);
|
|
|
|
|
fprintf(F, "\"%s\";typprefix\n",
|
|
|
|
|
add_translation(prefix, LOC(f->locale, prefix)));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
fprintf(F, "%d;Rekrutierungskosten\n", f->race->recruitcost);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Anzahl Personen\n", count_all(f));
|
|
|
|
|
fprintf(F, "\"%s\";Magiegebiet\n", neue_gebiete[f->magiegebiet]);
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (f->race == new_race[RC_HUMAN]) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Anzahl Immigranten\n", count_migrants(f));
|
|
|
|
|
fprintf(F, "%d;Max. Immigranten\n", count_maxmigrants(f));
|
|
|
|
|
}
|
2004-10-19 00:03:32 +02:00
|
|
|
|
|
2004-12-22 01:56:27 +01:00
|
|
|
|
#ifdef HEROES
|
2004-10-19 00:03:32 +02:00
|
|
|
|
i = countheroes(f);
|
|
|
|
|
if (i>0) fprintf(F, "%d;heroes\n", i);
|
|
|
|
|
i = maxheroes(f);
|
|
|
|
|
if (i>0) fprintf(F, "%d;max_heroes\n", i);
|
2004-12-20 11:09:59 +01:00
|
|
|
|
#endif
|
2004-10-19 00:03:32 +02:00
|
|
|
|
|
2004-10-19 23:29:35 +02:00
|
|
|
|
if (f->age > 1 && f->lastorders != turn) {
|
|
|
|
|
fprintf(F, "%d;nmr\n", turn-f->lastorders);
|
|
|
|
|
}
|
|
|
|
|
|
2004-10-19 00:03:32 +02:00
|
|
|
|
fprintf(F, "\"%s\";Parteiname\n", f->name);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "\"%s\";email\n", f->email);
|
|
|
|
|
fprintf(F, "\"%s\";banner\n", f->banner);
|
2004-12-22 01:53:12 +01:00
|
|
|
|
for (itm=f->items; itm; itm=itm->next) {
|
|
|
|
|
int in = itm->number;
|
|
|
|
|
const char * ic = LOC(f->locale, resourcename(itm->type->rtype, in));
|
|
|
|
|
if (itm==f->items) fputs("GEGENSTAENDE\n", F);
|
|
|
|
|
fprintf(F, "%d;%s\n", in, add_translation(ic, LOC(f->locale, ic)));
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fputs("OPTIONEN\n", F);
|
|
|
|
|
for (i=0;i!=MAXOPTIONS;++i) {
|
|
|
|
|
fprintf(F, "%d;%s\n", (f->options&want(i))?1:0, options[i]);
|
|
|
|
|
}
|
2002-12-18 18:40:21 +01:00
|
|
|
|
#ifdef REGIONOWNERS
|
|
|
|
|
show_enemies(F, f->enemies);
|
|
|
|
|
#endif
|
2002-09-02 22:36:12 +02:00
|
|
|
|
show_allies(F, f, f->allies);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
group * g;
|
|
|
|
|
for (g=f->groups;g;g=g->next) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "GRUPPE %d\n", g->gid);
|
|
|
|
|
fprintf(F, "\"%s\";name\n", g->name);
|
2004-07-09 21:14:10 +02:00
|
|
|
|
prefix = get_prefix(g->attribs);
|
|
|
|
|
if (prefix!=NULL) {
|
2004-08-07 09:42:22 +02:00
|
|
|
|
prefix = mkname("prefix", prefix);
|
|
|
|
|
fprintf(F, "\"%s\";typprefix\n",
|
|
|
|
|
add_translation(prefix, LOC(f->locale, prefix)));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2002-09-02 22:36:12 +02:00
|
|
|
|
show_allies(F, f, g->allies);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cr_output_str_list(F, "FEHLER", f->mistakes, f);
|
|
|
|
|
cr_output_messages(F, f->msgs, f);
|
|
|
|
|
{
|
|
|
|
|
struct bmsg * bm;
|
|
|
|
|
for (bm=f->battles;bm;bm=bm->next) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (!rplane(bm->r)) fprintf(F, "BATTLE %d %d\n", region_x(bm->r, f), region_y(bm->r, f));
|
|
|
|
|
else {
|
|
|
|
|
if (rplane(bm->r)->flags & PFL_NOCOORDS) fprintf(F, "BATTLESPEC %d %d\n", encode_region(f, bm->r), rplane(bm->r)->id);
|
|
|
|
|
else fprintf(F, "BATTLE %d %d %d\n", region_x(bm->r, f), region_y(bm->r, f), rplane(bm->r)->id);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cr_output_messages(F, bm->msgs, f);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cr_find_address(F, f, addresses);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
a = a_find(f->attribs, &at_reportspell);
|
|
|
|
|
while (a) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
cr_reportspell(F, (spellid_t)a->data.i, f->locale);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
a = a->nexttype;
|
|
|
|
|
}
|
|
|
|
|
for (a=a_find(f->attribs, &at_showitem);a;a=a->nexttype) {
|
|
|
|
|
const potion_type * ptype = resource2potion(((const item_type*)a->data.v)->rtype);
|
|
|
|
|
requirement * m;
|
2004-09-23 01:17:15 +02:00
|
|
|
|
const char * ch, * description;
|
|
|
|
|
|
|
|
|
|
if (ptype==NULL) continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
m = ptype->itype->construction->materials;
|
|
|
|
|
ch = resourcename(ptype->itype->rtype, 0);
|
|
|
|
|
fprintf(F, "TRANK %d\n", hashstring(ch));
|
2001-04-16 16:34:19 +02:00
|
|
|
|
fprintf(F, "\"%s\";Name\n", add_translation(ch, locale_string(f->locale, ch)));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Stufe\n", ptype->level);
|
2004-09-23 01:17:15 +02:00
|
|
|
|
|
|
|
|
|
description = ptype->text;
|
|
|
|
|
if (description==NULL || f->locale!=find_locale("de")) {
|
|
|
|
|
const char * pname = resourcename(ptype->itype->rtype, 0);
|
|
|
|
|
const char * potiontext = mkname("potion", pname);
|
|
|
|
|
description = LOC(f->locale, potiontext);
|
|
|
|
|
if (strcmp(description, potiontext)==0) {
|
|
|
|
|
/* string not found */
|
|
|
|
|
description = ptype->text;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(F, "\"%s\";Beschr\n", description);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "ZUTATEN\n");
|
|
|
|
|
|
|
|
|
|
while (m->number) {
|
2001-04-16 16:34:19 +02:00
|
|
|
|
ch = resourcename(oldresourcetype[m->type], 0);
|
|
|
|
|
fprintf(F, "\"%s\"\n", add_translation(ch, locale_string(f->locale, ch)));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
m++;
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* traverse all regions */
|
2004-01-25 10:58:58 +01:00
|
|
|
|
for (r=first;r!=last;r=r->next) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
int modifier = 0;
|
2001-01-28 09:50:46 +01:00
|
|
|
|
const char * tname;
|
2004-01-19 00:57:43 +01:00
|
|
|
|
const seen_region * sd = find_seen(r);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-01-19 00:57:43 +01:00
|
|
|
|
if (sd==NULL) continue;
|
|
|
|
|
|
2003-11-10 11:05:18 +01:00
|
|
|
|
if (!rplane(r)) {
|
2004-01-19 00:57:43 +01:00
|
|
|
|
if (opt_cr_absolute_coords) {
|
2003-11-10 11:05:18 +01:00
|
|
|
|
fprintf(F, "REGION %d %d\n", r->x, r->x);
|
|
|
|
|
} else {
|
|
|
|
|
fprintf(F, "REGION %d %d\n", region_x(r, f), region_y(r, f));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#if ENCODE_SPECIAL
|
|
|
|
|
if (rplane(r)->flags & PFL_NOCOORDS) fprintf(F, "SPEZIALREGION %d %d\n", encode_region(f, r), rplane(r)->id);
|
|
|
|
|
#else
|
|
|
|
|
if (rplane(r)->flags & PFL_NOCOORDS) continue;
|
|
|
|
|
#endif
|
|
|
|
|
else fprintf(F, "REGION %d %d %d\n", region_x(r, f), region_y(r, f), rplane(r)->id);
|
|
|
|
|
}
|
|
|
|
|
if (r->land && strlen(rname(r, f->locale))) fprintf(F, "\"%s\";Name\n", rname(r, f->locale));
|
|
|
|
|
if (is_cursed(r->attribs,C_MAELSTROM, 0))
|
|
|
|
|
tname = "maelstrom";
|
|
|
|
|
else {
|
|
|
|
|
if (r_isforest(r)) tname = "forest";
|
|
|
|
|
else tname = terrain[rterrain(r)].name;
|
|
|
|
|
}
|
|
|
|
|
|
2001-04-16 16:34:19 +02:00
|
|
|
|
fprintf(F, "\"%s\";Terrain\n", add_translation(tname, locale_string(f->locale, tname)));
|
2004-01-19 00:57:43 +01:00
|
|
|
|
switch (sd->mode) {
|
2002-01-17 20:21:20 +01:00
|
|
|
|
case see_far:
|
|
|
|
|
fputs("\"neighbourhood\";visibility\n", F);
|
|
|
|
|
break;
|
|
|
|
|
case see_lighthouse:
|
|
|
|
|
fputs("\"lighthouse\";visibility\n", F);
|
|
|
|
|
break;
|
|
|
|
|
case see_travel:
|
|
|
|
|
fputs("\"travel\";visibility\n", F);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2004-01-19 00:57:43 +01:00
|
|
|
|
{
|
|
|
|
|
unit * owner = region_owner(r);
|
|
|
|
|
if (owner) {
|
|
|
|
|
fprintf(F, "%d;owner\n", owner->faction->no);
|
|
|
|
|
}
|
2002-12-18 18:40:21 +01:00
|
|
|
|
}
|
2004-01-19 00:57:43 +01:00
|
|
|
|
if (sd->mode == see_neighbour) {
|
|
|
|
|
cr_borders(r, f, sd->mode, F);
|
2002-04-07 02:44:01 +02:00
|
|
|
|
} else {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#define RESOURCECOMPAT
|
|
|
|
|
char cbuf[8192], *pos = cbuf;
|
|
|
|
|
#ifdef RESOURCECOMPAT
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (r->display && strlen(r->display))
|
|
|
|
|
fprintf(F, "\"%s\";Beschr\n", r->display);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (landregion(rterrain(r))) {
|
2001-12-15 13:26:04 +01:00
|
|
|
|
#if GROWING_TREES
|
2001-12-10 01:13:39 +01:00
|
|
|
|
int trees = rtrees(r,2);
|
|
|
|
|
int ytrees = rtrees(r,1);
|
|
|
|
|
# ifdef RESOURCECOMPAT
|
|
|
|
|
if (trees > 0) fprintf(F, "%d;Baeume\n", trees);
|
|
|
|
|
if (ytrees > 0) fprintf(F, "%d;Schoesslinge\n", ytrees);
|
|
|
|
|
if (fval(r, RF_MALLORN) && (trees > 0 || ytrees > 0))
|
|
|
|
|
fprintf(F, "1;Mallorn\n");
|
|
|
|
|
# endif
|
|
|
|
|
if (!fval(r, RF_MALLORN)) {
|
2002-02-24 11:04:47 +01:00
|
|
|
|
if (ytrees) pos = report_resource(pos, "rm_youngtrees", f->locale, ytrees, -1);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (trees) pos = report_resource(pos, "rm_trees", f->locale, trees, -1);
|
|
|
|
|
} else {
|
2002-02-24 11:04:47 +01:00
|
|
|
|
if (ytrees) pos = report_resource(pos, "rm_youngmallorn", f->locale, ytrees, -1);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (trees) pos = report_resource(pos, "rm_mallorn", f->locale, trees, -1);
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
int trees = rtrees(r);
|
|
|
|
|
# ifdef RESOURCECOMPAT
|
|
|
|
|
fprintf(F, "%d;Baeume\n", trees);
|
|
|
|
|
if (fval(r, RF_MALLORN) && trees)
|
|
|
|
|
fprintf(F, "1;Mallorn\n");
|
|
|
|
|
# endif
|
|
|
|
|
if (!fval(r, RF_MALLORN)) {
|
|
|
|
|
if (trees) pos = report_resource(pos, "rm_trees", f->locale, trees, -1);
|
|
|
|
|
} else {
|
|
|
|
|
if (trees) pos = report_resource(pos, "rm_mallorn", f->locale, trees, -1);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Bauern\n", rpeasants(r));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if(fval(r, RF_ORCIFIED)) {
|
|
|
|
|
fprintf(F, "1;Verorkt\n");
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Pferde\n", rhorses(r));
|
|
|
|
|
|
2004-01-19 00:57:43 +01:00
|
|
|
|
if (sd->mode>=see_unit) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
struct demand * dmd = r->land->demands;
|
2001-12-15 13:26:04 +01:00
|
|
|
|
#if NEW_RESOURCEGROWTH
|
2001-12-10 01:13:39 +01:00
|
|
|
|
struct rawmaterial * res = r->resources;
|
2001-12-18 00:55:06 +01:00
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "%d;Silber\n", rmoney(r));
|
|
|
|
|
fprintf(F, "%d;Unterh\n", entertainmoney(r));
|
|
|
|
|
|
|
|
|
|
if (is_cursed(r->attribs, C_RIOT, 0)){
|
|
|
|
|
fprintf(F, "0;Rekruten\n");
|
|
|
|
|
} else {
|
|
|
|
|
fprintf(F, "%d;Rekruten\n", rpeasants(r) / RECRUITFRACTION);
|
|
|
|
|
}
|
|
|
|
|
if (production(r)) {
|
|
|
|
|
fprintf(F, "%d;Lohn\n", fwage(r, f, true));
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2001-12-18 00:55:06 +01:00
|
|
|
|
#if NEW_RESOURCEGROWTH
|
2001-12-10 01:13:39 +01:00
|
|
|
|
while (res) {
|
|
|
|
|
int maxskill = 0;
|
|
|
|
|
int level = -1;
|
|
|
|
|
int visible = -1;
|
|
|
|
|
const item_type * itype = resource2item(res->type->rtype);
|
|
|
|
|
if (res->type->visible==NULL) {
|
|
|
|
|
visible = res->amount;
|
|
|
|
|
level = res->level + itype->construction->minskill - 1;
|
|
|
|
|
} else {
|
|
|
|
|
const unit * u;
|
|
|
|
|
for (u=r->units; visible!=res->amount && u!=NULL; u=u->next) {
|
|
|
|
|
if (u->faction == f) {
|
|
|
|
|
int s = eff_skill(u, itype->construction->skill, r);
|
|
|
|
|
if (s>maxskill) {
|
|
|
|
|
if (s>=itype->construction->minskill) {
|
|
|
|
|
assert(itype->construction->minskill>0);
|
|
|
|
|
level = res->level + itype->construction->minskill - 1;
|
|
|
|
|
}
|
|
|
|
|
maxskill = s;
|
|
|
|
|
visible = res->type->visible(res, maxskill);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-04-09 17:20:58 +02:00
|
|
|
|
if (level>=0 && visible >=0) {
|
2001-12-10 01:13:39 +01:00
|
|
|
|
pos = report_resource(pos, res->type->name, f->locale, visible, level);
|
|
|
|
|
# ifdef RESOURCECOMPAT
|
|
|
|
|
if (visible>=0) fprintf(F, "%d;%s\n", visible, crtag(res->type->name));
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
res = res->next;
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
const unit * u;
|
|
|
|
|
int maxmining = 0;
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->faction == f) {
|
|
|
|
|
int s = eff_skill(u, SK_MINING, r);
|
|
|
|
|
maxmining = max(maxmining, s);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->faction == f) {
|
|
|
|
|
int s = eff_skill(u, SK_MINING, r);
|
|
|
|
|
maxmining = max(maxmining, s);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (maxmining >= 4 && riron(r) > 0)
|
|
|
|
|
fprintf(F, "%d;Eisen\n", riron(r));
|
|
|
|
|
if (maxmining >= 7 && rlaen(r) > 0)
|
|
|
|
|
fprintf(F, "%d;Laen\n", rlaen(r));
|
|
|
|
|
#endif
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* trade */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (!TradeDisabled() && rpeasants(r)/TRADE_FRACTION > 0) {
|
2001-02-03 14:45:35 +01:00
|
|
|
|
fputs("PREISE\n", F);
|
|
|
|
|
while (dmd) {
|
2001-04-16 16:34:19 +02:00
|
|
|
|
const char * ch = resourcename(dmd->type->itype->rtype, 0);
|
|
|
|
|
fprintf(F, "%d;%s\n", (dmd->value
|
|
|
|
|
? dmd->value*dmd->type->price
|
|
|
|
|
: -dmd->type->price),
|
|
|
|
|
add_translation(ch, locale_string(f->locale, ch)));
|
2001-02-03 14:45:35 +01:00
|
|
|
|
dmd=dmd->next;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (pos!=cbuf) fputs(cbuf, F);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2004-05-02 21:30:47 +02:00
|
|
|
|
print_curses(F, f, r, TYP_REGION);
|
2004-01-19 00:57:43 +01:00
|
|
|
|
cr_borders(r, f, sd->mode, F);
|
2004-05-25 10:53:59 +02:00
|
|
|
|
if (sd->mode==see_unit && rplane(r)==get_astralplane() && !is_cursed(r->attribs, C_ASTRALBLOCK, 0))
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
/* Sonderbehandlung Teleport-Ebene */
|
2004-05-25 10:53:59 +02:00
|
|
|
|
region_list *rl = astralregions(r, inhabitable);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (rl) {
|
2004-02-21 13:18:29 +01:00
|
|
|
|
region_list *rl2 = rl;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
while(rl2) {
|
2004-02-21 13:18:29 +01:00
|
|
|
|
region * r = rl2->data;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fprintf(F, "SCHEMEN %d %d\n", region_x(r, f), region_y(r, f));
|
|
|
|
|
fprintf(F, "\"%s\";Name\n", rname(r, f->locale));
|
|
|
|
|
rl2 = rl2->next;
|
|
|
|
|
if(rl2) scat(", ");
|
|
|
|
|
}
|
|
|
|
|
free_regionlist(rl);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* describe both passed and inhabited regions */
|
|
|
|
|
show_active_spells(r);
|
|
|
|
|
{
|
2002-01-05 17:14:38 +01:00
|
|
|
|
boolean seeunits = false, seeships = false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
const attrib * ru;
|
2002-01-05 17:14:38 +01:00
|
|
|
|
/* show units pulled through region */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for (ru = a_find(r->attribs, &at_travelunit); ru; ru = ru->nexttype) {
|
|
|
|
|
unit * u = (unit*)ru->data.v;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (cansee_durchgezogen(f, r, u, 0) && r!=u->region) {
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (!u->ship || !fval(u, UFL_OWNER)) continue;
|
2002-01-05 17:14:38 +01:00
|
|
|
|
if (!seeships) fprintf(F, "DURCHSCHIFFUNG\n");
|
|
|
|
|
seeships = true;
|
|
|
|
|
fprintf(F, "\"%s\"\n", shipname(u->ship));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for (ru = a_find(r->attribs, &at_travelunit); ru; ru = ru->nexttype) {
|
|
|
|
|
unit * u = (unit*)ru->data.v;
|
|
|
|
|
if (cansee_durchgezogen(f, r, u, 0) && r!=u->region) {
|
|
|
|
|
if (u->ship) continue;
|
|
|
|
|
if (!seeunits) fprintf(F, "DURCHREISE\n");
|
|
|
|
|
seeunits = true;
|
|
|
|
|
fprintf(F, "\"%s\"\n", unitname(u));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
cr_output_messages(F, r->msgs, f);
|
2001-05-20 09:46:13 +02:00
|
|
|
|
{
|
|
|
|
|
message_list * mlist = r_getmessages(r, f);
|
|
|
|
|
if (mlist) cr_output_messages(F, mlist, f);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* buildings */
|
|
|
|
|
for (b = rbuildings(r); b; b = b->next) {
|
|
|
|
|
int fno = -1;
|
|
|
|
|
u = buildingowner(r, b);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (u && !fval(u, UFL_PARTEITARNUNG)) {
|
2002-02-18 08:27:50 +01:00
|
|
|
|
const faction * sf = visible_faction(f,u);
|
|
|
|
|
fno = sf->no;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
cr_output_buildings(F, b, u, fno, f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ships */
|
|
|
|
|
for (sh = r->ships; sh; sh = sh->next) {
|
|
|
|
|
int fno = -1;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
u = shipowner(sh);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (u && !fval(u, UFL_PARTEITARNUNG)) {
|
2002-02-18 08:27:50 +01:00
|
|
|
|
const faction * sf = visible_faction(f,u);
|
|
|
|
|
fno = sf->no;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
cr_output_ship(F, sh, u, fno, f, r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* visible units */
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
boolean visible = true;
|
2004-01-19 00:57:43 +01:00
|
|
|
|
switch (sd->mode) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
case see_unit:
|
|
|
|
|
modifier=0;
|
|
|
|
|
break;
|
|
|
|
|
case see_far:
|
|
|
|
|
case see_lighthouse:
|
|
|
|
|
modifier = -2;
|
|
|
|
|
break;
|
|
|
|
|
case see_travel:
|
|
|
|
|
modifier = -1;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
visible=false;
|
|
|
|
|
}
|
|
|
|
|
if (u->building || u->ship || (visible && cansee(f, r, u, modifier)))
|
2004-01-19 00:57:43 +01:00
|
|
|
|
cr_output_unit(F, r, f, u, sd->mode);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
} /* region traversal */
|
|
|
|
|
}
|
|
|
|
|
report_crtypes(F, f->locale);
|
2001-04-16 16:34:19 +02:00
|
|
|
|
write_translations(F);
|
|
|
|
|
reset_translations();
|
2005-04-25 18:31:36 +02:00
|
|
|
|
if (errno) {
|
|
|
|
|
log_error(("%s\n", strerror(errno)));
|
|
|
|
|
errno = 0;
|
|
|
|
|
}
|
2005-04-24 23:49:51 +02:00
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|