2004-03-28 22:53:47 +02:00
|
|
|
/* vi: set ts=2:
|
|
|
|
+-------------------+
|
|
|
|
| | Enno Rehling <enno@eressea.de>
|
|
|
|
| Eressea PBEM host | Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
| (c) 1998 - 2004 | Katja Zedel <katze@felidae.kn-bremen.de>
|
|
|
|
| | Henning Peters <faroul@beyond.kn-bremen.de>
|
|
|
|
+-------------------+
|
|
|
|
|
|
|
|
This program may not be used, modified or distributed
|
|
|
|
without prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include "eressea.h"
|
|
|
|
#include "xmlreader.h"
|
|
|
|
|
|
|
|
#include <xml.h>
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include "building.h"
|
2005-10-02 22:28:44 +02:00
|
|
|
#include "equipment.h"
|
2004-03-28 22:53:47 +02:00
|
|
|
#include "item.h"
|
|
|
|
#include "message.h"
|
|
|
|
#include "race.h"
|
|
|
|
#include "ship.h"
|
|
|
|
#include "skill.h"
|
|
|
|
#include "spell.h"
|
2005-06-04 15:22:31 +02:00
|
|
|
#include "calendar.h"
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
#include <util/functions.h>
|
|
|
|
#include <util/message.h>
|
|
|
|
#include <util/nrmessage.h>
|
|
|
|
#include <util/crmessage.h>
|
|
|
|
|
|
|
|
/* libxml includes */
|
|
|
|
#include <libxml/tree.h>
|
|
|
|
#include <libxml/xpath.h>
|
|
|
|
#include <iconv.h>
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <assert.h>
|
|
|
|
#include <ctype.h>
|
2005-05-21 00:51:37 +02:00
|
|
|
#include <limits.h>
|
2004-03-28 22:53:47 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
2005-01-31 00:33:28 +01:00
|
|
|
static boolean gamecode_enabled = false;
|
|
|
|
|
|
|
|
void
|
|
|
|
enable_xml_gamecode(void)
|
|
|
|
{
|
|
|
|
gamecode_enabled = true;
|
|
|
|
}
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
static void
|
|
|
|
xml_readtext(xmlNodePtr node, struct locale ** lang, xmlChar **text)
|
|
|
|
{
|
|
|
|
xmlChar * property = xmlGetProp(node, BAD_CAST "locale");
|
2004-04-09 03:23:54 +02:00
|
|
|
assert(property!=NULL);
|
2004-03-28 22:53:47 +02:00
|
|
|
*lang = find_locale((const char*)property);
|
|
|
|
if (*lang==NULL) *lang = make_locale((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
*text = xmlNodeListGetString(node->doc, node->children, 1);
|
|
|
|
}
|
|
|
|
|
2005-10-08 21:22:56 +02:00
|
|
|
static const spell *
|
|
|
|
xml_spell(xmlNode * node, const char * name)
|
|
|
|
{
|
|
|
|
const spell * sp = NULL;
|
|
|
|
xmlChar * property = xmlGetProp(node, BAD_CAST name);
|
|
|
|
if (property!=NULL) {
|
|
|
|
int i = atoi((const char *)property);
|
|
|
|
if (i>0) {
|
|
|
|
sp = find_spellbyid((spellid_t)i);
|
|
|
|
}
|
|
|
|
if (sp==NULL) {
|
|
|
|
sp = find_spell(M_NONE, (const char *)property);
|
|
|
|
}
|
|
|
|
assert(sp);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
return sp;
|
|
|
|
}
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
static const char *
|
|
|
|
xml_to_locale(const xmlChar * xmlStr)
|
|
|
|
{
|
|
|
|
static iconv_t context = (iconv_t)-1;
|
|
|
|
static char zText[1024];
|
|
|
|
char * inbuf = (char*)xmlStr;
|
|
|
|
char * outbuf = zText;
|
2004-04-10 22:25:40 +02:00
|
|
|
size_t inbytes = strlen((const char*)xmlStr)+1;
|
2004-03-28 22:53:47 +02:00
|
|
|
size_t outbytes = sizeof(zText);
|
|
|
|
|
|
|
|
if (context==(iconv_t)-1) {
|
2005-06-04 15:22:31 +02:00
|
|
|
context = iconv_open("", "UTF-8");
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
assert(context!=(iconv_t)-1);
|
|
|
|
|
|
|
|
iconv(context, &inbuf, &inbytes, &outbuf, &outbytes);
|
|
|
|
if (inbytes!=0) {
|
2005-06-04 15:22:31 +02:00
|
|
|
log_error(("string is too long: %d chars remain in %s.\n", inbytes, (const char*)xmlStr));
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
return zText;
|
|
|
|
}
|
|
|
|
|
|
|
|
static xmlChar *
|
|
|
|
xml_cleanup_string(xmlChar * str)
|
|
|
|
{
|
|
|
|
xmlChar * read = str;
|
|
|
|
xmlChar * write = str;
|
|
|
|
|
|
|
|
while (*read) {
|
|
|
|
/* eat leading whitespace */
|
|
|
|
if (*read && isspace(*read)) {
|
|
|
|
while (*read && isspace(*read)) {
|
|
|
|
++read;
|
|
|
|
}
|
|
|
|
*write++ = ' ';
|
|
|
|
}
|
|
|
|
while (*read) {
|
|
|
|
if (*read== '\n') break;
|
|
|
|
if (*read== '\r') break;
|
|
|
|
*write++ = *read++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*write = 0;
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
xml_readrequirements(xmlNodePtr * nodeTab, int nodeNr, requirement ** reqArray)
|
|
|
|
{
|
|
|
|
int req;
|
|
|
|
requirement * radd = *reqArray;
|
|
|
|
|
|
|
|
assert (radd==NULL);
|
|
|
|
if (nodeNr==0) return;
|
|
|
|
|
|
|
|
radd = *reqArray = calloc(sizeof(requirement), nodeNr+1);
|
|
|
|
|
|
|
|
for (req=0;req!=nodeNr;++req) {
|
|
|
|
xmlNodePtr node = nodeTab[req];
|
|
|
|
xmlChar * property;
|
|
|
|
const resource_type * rtype;
|
|
|
|
resource_t type;
|
|
|
|
|
|
|
|
radd->number = xml_ivalue(node, "quantity", 0);
|
2004-06-11 21:59:02 +02:00
|
|
|
radd->recycle = xml_fvalue(node, "recycle", 0.0);
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "type");
|
|
|
|
rtype = rt_find((const char*)property);
|
2004-06-12 13:04:40 +02:00
|
|
|
assert(rtype!=NULL);
|
2004-03-28 22:53:47 +02:00
|
|
|
for (type=0;type!=MAX_RESOURCES;++type) {
|
|
|
|
if (oldresourcetype[type]==rtype) {
|
|
|
|
radd->type = type;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
++radd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
xml_readconstruction(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr, construction ** consPtr)
|
|
|
|
{
|
|
|
|
xmlNodePtr pushNode = xpath->node;
|
|
|
|
int k;
|
|
|
|
for (k=0;k!=nodeNr;++k) {
|
|
|
|
xmlNodePtr node = nodeTab[k];
|
|
|
|
xmlChar * property;
|
|
|
|
construction * con;
|
|
|
|
xmlXPathObjectPtr req;
|
2004-06-11 21:59:02 +02:00
|
|
|
int m;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
assert(*consPtr==NULL);
|
2005-10-03 22:59:11 +02:00
|
|
|
*consPtr = con = calloc(sizeof(construction), 1);
|
|
|
|
consPtr = &con->improvement;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "skill");
|
|
|
|
assert(property!=NULL);
|
|
|
|
con->skill = sk_find((const char*)property);
|
2004-06-12 12:38:33 +02:00
|
|
|
assert(con->skill!=NOSKILL);
|
2004-03-28 22:53:47 +02:00
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
con->maxsize = xml_ivalue(node, "maxsize", -1);
|
|
|
|
con->minskill = xml_ivalue(node, "minskill", -1);
|
|
|
|
con->reqsize = xml_ivalue(node, "reqsize", -1);
|
|
|
|
|
2004-06-11 21:59:02 +02:00
|
|
|
/* read construction/requirement */
|
2004-03-28 22:53:47 +02:00
|
|
|
xpath->node = node;
|
|
|
|
req = xmlXPathEvalExpression(BAD_CAST "requirement", xpath);
|
|
|
|
xml_readrequirements(req->nodesetval->nodeTab,
|
|
|
|
req->nodesetval->nodeNr, &con->materials);
|
|
|
|
xmlXPathFreeObject(req);
|
|
|
|
|
2004-06-11 21:59:02 +02:00
|
|
|
/* read construction/modifier */
|
|
|
|
xpath->node = node;
|
|
|
|
req = xmlXPathEvalExpression(BAD_CAST "modifier", xpath);
|
|
|
|
for (m=0;m!=req->nodesetval->nodeNr;++m) {
|
|
|
|
xmlNodePtr node = req->nodesetval->nodeTab[m];
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "function");
|
|
|
|
if (property!=NULL) {
|
|
|
|
pf_generic foo = get_function((const char*)property);
|
|
|
|
a_add(&con->attribs, make_skillmod(NOSKILL, SMF_PRODUCTION, (skillmod_fun)foo, 1.0, 0));
|
2005-05-08 02:36:11 +02:00
|
|
|
xmlFree(property);
|
2004-06-11 21:59:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(req);
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
xpath->node = pushNode;
|
|
|
|
}
|
|
|
|
|
2005-10-08 13:02:10 +02:00
|
|
|
static int
|
|
|
|
parse_function(xmlNodePtr node, pf_generic * funPtr, xmlChar ** namePtr)
|
|
|
|
{
|
|
|
|
pf_generic fun;
|
|
|
|
xmlChar * property = xmlGetProp(node, BAD_CAST "value");
|
|
|
|
assert(property!=NULL);
|
|
|
|
fun = get_function((const char*)property);
|
|
|
|
if (fun!=NULL) {
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
}
|
|
|
|
*namePtr = property;
|
|
|
|
*funPtr = fun;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
static int
|
|
|
|
parse_buildings(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr buildings;
|
|
|
|
xmlNodeSetPtr nodes;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* reading eressea/buildings/building */
|
|
|
|
buildings = xmlXPathEvalExpression(BAD_CAST "/eressea/buildings/building", xpath);
|
|
|
|
nodes = buildings->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
building_type * bt = calloc(sizeof(building_type), 1);
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
bt->_name = strdup((const char *)property);
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
bt->capacity = xml_ivalue(node, "capacity", -1);
|
|
|
|
bt->maxcapacity = xml_ivalue(node, "maxcapacity", -1);
|
|
|
|
bt->maxsize = xml_ivalue(node, "maxsize", -1);
|
|
|
|
|
|
|
|
bt->magres = xml_ivalue(node, "magres", 0);
|
|
|
|
bt->magresbonus = xml_ivalue(node, "magresbonus", 0);
|
|
|
|
bt->fumblebonus = xml_ivalue(node, "fumblebonus", 0);
|
2005-04-24 00:52:49 +02:00
|
|
|
bt->auraregen = xml_fvalue(node, "auraregen", 1.0);
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
if (xml_bvalue(node, "nodestroy", false)) bt->flags |= BTF_INDESTRUCTIBLE;
|
|
|
|
if (xml_bvalue(node, "nobuild", false)) bt->flags |= BTF_NOBUILD;
|
|
|
|
if (xml_bvalue(node, "unique", false)) bt->flags |= BTF_UNIQUE;
|
|
|
|
if (xml_bvalue(node, "decay", false)) bt->flags |= BTF_DECAY;
|
|
|
|
if (xml_bvalue(node, "magic", false)) bt->flags |= BTF_MAGIC;
|
|
|
|
if (xml_bvalue(node, "protection", false)) bt->flags |= BTF_PROTECTION;
|
|
|
|
|
|
|
|
/* reading eressea/buildings/building/construction */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
|
|
|
|
xml_readconstruction(xpath, result->nodesetval->nodeTab, result->nodesetval->nodeNr, &bt->construction);
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/buildings/building/function */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
pf_generic fun;
|
2005-10-08 13:02:10 +02:00
|
|
|
parse_function(node, &fun, &property);
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
if (fun==NULL) {
|
|
|
|
log_error(("unknown function name '%s' for building %s\n",
|
|
|
|
(const char*)property, bt->_name));
|
|
|
|
xmlFree(property);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (strcmp((const char*)property, "name")==0) {
|
|
|
|
bt->name = (const char * (*)(int size))fun;
|
|
|
|
} else if (strcmp((const char*)property, "init")==0) {
|
|
|
|
bt->init = (void (*)(struct building_type*))fun;
|
|
|
|
} else {
|
2005-10-08 13:02:10 +02:00
|
|
|
log_error(("unknown function type '%s' for building %s\n",
|
|
|
|
(const char*)property, bt->_name));
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/buildings/building/maintenance */
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "maintenance", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
maintenance * mt;
|
|
|
|
|
|
|
|
if (bt->maintenance==NULL) {
|
|
|
|
bt->maintenance = calloc(sizeof(struct maintenance), result->nodesetval->nodeNr+1);
|
|
|
|
}
|
|
|
|
mt = bt->maintenance + k;
|
|
|
|
mt->number = xml_ivalue(node, "amount", 0);
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "type");
|
|
|
|
assert(property!=NULL);
|
|
|
|
mt->rtype = rt_find((const char*)property);
|
2005-05-27 22:22:17 +02:00
|
|
|
assert(mt->rtype!=NULL);
|
2004-03-28 22:53:47 +02:00
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
if (xml_bvalue(node, "variable", false)) mt->flags |= MTF_VARIABLE;
|
|
|
|
if (xml_bvalue(node, "vital", false)) mt->flags |= MTF_VITAL;
|
|
|
|
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* finally, register the new building type */
|
|
|
|
bt_register(bt);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(buildings);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-06-04 15:22:31 +02:00
|
|
|
static int
|
|
|
|
parse_calendar(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr xpathCalendars;
|
|
|
|
xmlNodeSetPtr nsetCalendars;
|
|
|
|
int rv = 0;
|
|
|
|
|
|
|
|
/* reading eressea/buildings/building */
|
|
|
|
xpathCalendars = xmlXPathEvalExpression(BAD_CAST "/eressea/calendar", xpath);
|
|
|
|
nsetCalendars = xpathCalendars->nodesetval;
|
|
|
|
if (nsetCalendars==NULL || nsetCalendars->nodeNr!=1) {
|
|
|
|
log_error(("invalid or missing calendar data in %s\n", doc->name));
|
|
|
|
rv = -1;
|
|
|
|
} else {
|
|
|
|
xmlNodePtr calendar = nsetCalendars->nodeTab[0];
|
|
|
|
xmlXPathObjectPtr xpathWeeks, xpathMonths, xpathSeasons;
|
|
|
|
xmlNodeSetPtr nsetWeeks, nsetMonths, nsetSeasons;
|
|
|
|
xmlChar * property = xmlGetProp(calendar, BAD_CAST "name");
|
|
|
|
xmlChar * newyear = xmlGetProp(calendar, BAD_CAST "newyear");
|
|
|
|
|
|
|
|
first_turn = xml_ivalue(calendar, "start", 0);
|
|
|
|
if (property) {
|
|
|
|
agename = strdup(mkname("calendar", (const char*)property));
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
|
|
|
|
xpath->node = calendar;
|
|
|
|
xpathWeeks = xmlXPathEvalExpression(BAD_CAST "week", xpath);
|
|
|
|
nsetWeeks = xpathWeeks->nodesetval;
|
|
|
|
if (nsetWeeks!=NULL) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
weeks_per_month = nsetWeeks->nodeNr;
|
|
|
|
weeknames = malloc(sizeof(char *) * weeks_per_month);
|
|
|
|
weeknames2 = malloc(sizeof(char *) * weeks_per_month);
|
|
|
|
for (i=0;i!=nsetWeeks->nodeNr;++i) {
|
|
|
|
xmlNodePtr week = nsetWeeks->nodeTab[i];
|
|
|
|
xmlChar * property = xmlGetProp(week, BAD_CAST "name");
|
|
|
|
if (property) {
|
|
|
|
weeknames[i] = strdup(mkname("calendar", (const char*)property));
|
|
|
|
weeknames2[i] = malloc(strlen(weeknames[i])+3);
|
|
|
|
sprintf(weeknames2[i], "%s_d", weeknames[i]);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(xpathWeeks);
|
|
|
|
|
|
|
|
months_per_year = 0;
|
|
|
|
xpathSeasons = xmlXPathEvalExpression(BAD_CAST "season", xpath);
|
|
|
|
nsetSeasons = xpathSeasons->nodesetval;
|
|
|
|
if (nsetSeasons!=NULL) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
seasons = nsetSeasons->nodeNr;
|
|
|
|
seasonnames = malloc(sizeof(char *) * seasons);
|
|
|
|
|
|
|
|
for (i=0;i!=nsetSeasons->nodeNr;++i) {
|
|
|
|
xmlNodePtr season = nsetSeasons->nodeTab[i];
|
|
|
|
xmlChar * property = xmlGetProp(season, BAD_CAST "name");
|
|
|
|
if (property) {
|
|
|
|
seasonnames[i] = strdup(mkname("calendar", (const char*)property));
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
xpathMonths = xmlXPathEvalExpression(BAD_CAST "season/month", xpath);
|
|
|
|
nsetMonths = xpathMonths->nodesetval;
|
|
|
|
if (nsetMonths!=NULL) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
months_per_year = nsetMonths->nodeNr;
|
|
|
|
monthnames = malloc(sizeof(char *) * months_per_year);
|
|
|
|
month_season = malloc(sizeof(int) * months_per_year);
|
|
|
|
storms = malloc(sizeof(int) * months_per_year);
|
|
|
|
|
|
|
|
for (i=0;i!=nsetMonths->nodeNr;++i) {
|
|
|
|
xmlNodePtr month = nsetMonths->nodeTab[i];
|
|
|
|
xmlChar * property = xmlGetProp(month, BAD_CAST "name");
|
|
|
|
int j;
|
|
|
|
|
|
|
|
if (property) {
|
|
|
|
if (newyear && strcmp((const char*)newyear, (const char*)property)==0) {
|
|
|
|
first_month = i;
|
|
|
|
xmlFree(newyear);
|
|
|
|
newyear = NULL;
|
|
|
|
}
|
|
|
|
monthnames[i] = strdup(mkname("calendar", (const char*)property));
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
for (j=0;j!=seasons;++j) {
|
|
|
|
xmlNodePtr season = month->parent;
|
|
|
|
if (season==nsetSeasons->nodeTab[j]) {
|
|
|
|
month_season[i] = j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(j!=seasons);
|
|
|
|
storms[i] = xml_ivalue(nsetMonths->nodeTab[i], "storm", 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(xpathMonths);
|
|
|
|
xmlXPathFreeObject(xpathSeasons);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(xpathCalendars);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
static int
|
|
|
|
parse_ships(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr ships;
|
|
|
|
xmlNodeSetPtr nodes;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* reading eressea/ships/ship */
|
|
|
|
ships = xmlXPathEvalExpression(BAD_CAST "/eressea/ships/ship", xpath);
|
|
|
|
nodes = ships->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
ship_type * st = calloc(sizeof(ship_type), 1);
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
st->name[0] = strdup((const char *)property);
|
|
|
|
st->name[1] = strcat(strcpy(malloc(strlen(st->name[0])+3), st->name[0]),"_a");
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
st->cabins = xml_ivalue(node, "cabins", 0);
|
|
|
|
st->cargo = xml_ivalue(node, "cargo", 0);
|
|
|
|
st->combat = xml_ivalue(node, "combat", 0);
|
|
|
|
st->cptskill = xml_ivalue(node, "cptskill", 0);
|
2005-04-24 00:52:49 +02:00
|
|
|
st->damage = xml_fvalue(node, "damage", 0.0);
|
2004-03-28 22:53:47 +02:00
|
|
|
if (xml_bvalue(node, "fly", false)) st->flags |= SFL_FLY;
|
|
|
|
if (xml_bvalue(node, "opensea", false)) st->flags |= SFL_OPENSEA;
|
|
|
|
st->minskill = xml_ivalue(node, "minskill", 0);
|
|
|
|
st->range = xml_ivalue(node, "range", 0);
|
2005-04-24 00:52:49 +02:00
|
|
|
st->storm = xml_fvalue(node, "storm", 1.0);
|
2004-03-28 22:53:47 +02:00
|
|
|
st->sumskill = xml_ivalue(node, "sumskill", 0);
|
|
|
|
|
|
|
|
/* reading eressea/ships/ship/construction */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
|
|
|
|
xml_readconstruction(xpath, result->nodesetval->nodeTab, result->nodesetval->nodeNr, &st->construction);
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/ships/ship/coast */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "coast", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
terrain_t t;
|
|
|
|
|
|
|
|
if (k==0) {
|
|
|
|
assert(st->coast==NULL);
|
|
|
|
st->coast = malloc(sizeof(terrain_t) * (result->nodesetval->nodeNr+1));
|
|
|
|
st->coast[result->nodesetval->nodeNr] = NOTERRAIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "terrain");
|
|
|
|
assert(property!=NULL);
|
|
|
|
st->coast[k] = NOTERRAIN;
|
|
|
|
for (t=0;t!=MAXTERRAINS;++t) {
|
|
|
|
if (strcmp((const char*)property, terrain[t].name)==0) {
|
|
|
|
st->coast[k] = t;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(st->coast[k]!=NOTERRAIN);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* finally, register the new building type */
|
|
|
|
st_register(st);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(ships);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
race_compat(void)
|
|
|
|
{
|
|
|
|
/* required for old_race, do not change order! */
|
|
|
|
const char * oldracenames[MAXRACES] = {
|
|
|
|
"dwarf", "elf", "orc", "goblin", "human", "troll", "demon", "insect",
|
|
|
|
"halfling", "cat", "aquarian", "uruk", "snotling", "undead", "illusion",
|
2005-08-22 22:14:42 +02:00
|
|
|
"youngdragon", "dragon", "wyrm", "ent", "catdragon", "dracoid",
|
2005-08-22 11:31:28 +02:00
|
|
|
"special", "spell", "irongolem", "stonegolem", "shadowdemon",
|
2004-03-28 22:53:47 +02:00
|
|
|
"shadowmaster", "mountainguard", "alp", "toad", "braineater", "peasant",
|
|
|
|
"wolf", "lynx", "tunnelworm", "eagle", "rat", "songdragon", "nymph",
|
|
|
|
"unicorn", "direwolf", "ghost", "imp", "dreamcat", "fairy", "owl",
|
2005-08-22 11:31:28 +02:00
|
|
|
"hellcat", "tiger", "dolphin", "giantturtle", "kraken", "seaserpent",
|
2005-08-22 21:35:31 +02:00
|
|
|
"shadowknight", "centaur", "skeleton", "skeletonlord", "zombie",
|
2004-03-28 22:53:47 +02:00
|
|
|
"juju-zombie", "ghoul", "ghast", "museumghost", "gnome", "template",
|
|
|
|
"clone", "shadowdragon", "shadowbat", "nightmare", "vampunicorn",
|
2005-09-03 02:30:03 +02:00
|
|
|
"phoenix"
|
2004-03-28 22:53:47 +02:00
|
|
|
};
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i=0;i!=MAXRACES;++i) {
|
|
|
|
race * rc = rc_find(oldracenames[i]);
|
|
|
|
if (rc) {
|
|
|
|
new_race[i] = rc;
|
|
|
|
if (rc == new_race[RC_TROLL]) {
|
|
|
|
a_add(&rc->attribs, make_skillmod(NOSKILL, SMF_RIDING, NULL, 0.0, -1));
|
|
|
|
}
|
2005-08-22 22:10:03 +02:00
|
|
|
} else {
|
|
|
|
log_error(("could not find old race %s\n", oldracenames[i]));
|
|
|
|
assert(rc);
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-31 03:07:07 +02:00
|
|
|
static armor_type *
|
|
|
|
xml_readarmor(xmlXPathContextPtr xpath, item_type * itype)
|
|
|
|
{
|
|
|
|
xmlNodePtr node = xpath->node;
|
|
|
|
armor_type * atype = NULL;
|
|
|
|
unsigned int flags = ATF_NONE;
|
|
|
|
int ac = xml_ivalue(node, "ac", 0);
|
|
|
|
double penalty = xml_fvalue(node, "penalty", 0.0);
|
|
|
|
double magres = xml_fvalue(node, "magres", 0.0);
|
|
|
|
|
|
|
|
if (xml_bvalue(node, "laen", false)) flags |= ATF_LAEN;
|
|
|
|
if (xml_bvalue(node, "shield", false)) flags |= ATF_SHIELD;
|
|
|
|
|
|
|
|
atype = new_armortype(itype, penalty, magres, ac, flags);
|
|
|
|
return atype;
|
|
|
|
}
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
static weapon_type *
|
|
|
|
xml_readweapon(xmlXPathContextPtr xpath, item_type * itype)
|
|
|
|
{
|
|
|
|
xmlNodePtr node = xpath->node;
|
|
|
|
weapon_type * wtype = NULL;
|
|
|
|
unsigned int flags = WTF_NONE;
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
xmlChar * property;
|
|
|
|
int k;
|
|
|
|
skill_t sk;
|
|
|
|
int minskill = xml_ivalue(node, "minskill", 0);
|
|
|
|
int offmod = xml_ivalue(node, "offmod", 0);
|
|
|
|
int defmod = xml_ivalue(node, "defmod", 0);
|
|
|
|
int reload = xml_ivalue(node, "reload", 0);
|
|
|
|
double magres = xml_fvalue(node, "magres", 0.0);
|
|
|
|
|
2004-06-11 21:59:02 +02:00
|
|
|
if (xml_bvalue(node, "armorpiercing", false)) flags |= WTF_ARMORPIERCING;
|
|
|
|
if (xml_bvalue(node, "magical", false)) flags |= WTF_MAGICAL;
|
|
|
|
if (xml_bvalue(node, "missile", false)) flags |= WTF_MISSILE;
|
2004-03-28 22:53:47 +02:00
|
|
|
if (xml_bvalue(node, "pierce", false)) flags |= WTF_PIERCE;
|
|
|
|
if (xml_bvalue(node, "cut", false)) flags |= WTF_CUT;
|
|
|
|
if (xml_bvalue(node, "blunt", false)) flags |= WTF_BLUNT;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "skill");
|
|
|
|
assert(property!=NULL);
|
|
|
|
sk = sk_find((const char*)property);
|
2004-06-11 21:59:02 +02:00
|
|
|
assert(sk!=NOSKILL);
|
2004-03-28 22:53:47 +02:00
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
wtype = new_weapontype(itype, flags, magres, NULL, offmod, defmod, reload, sk, minskill);
|
|
|
|
|
|
|
|
/* reading weapon/damage */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "damage", xpath);
|
|
|
|
assert(result->nodesetval->nodeNr<=2);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
int pos = 0;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "type");
|
2004-06-11 21:59:02 +02:00
|
|
|
if (strcmp((const char *)property, "footman")!=0) {
|
|
|
|
pos = 1;
|
|
|
|
}
|
2004-03-28 22:53:47 +02:00
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "value");
|
|
|
|
wtype->damage[pos] = gc_add(strdup((const char*)property));
|
2004-06-11 21:59:02 +02:00
|
|
|
if (k==0) wtype->damage[1-pos] = wtype->damage[pos];
|
2004-03-28 22:53:47 +02:00
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading weapon/modifier */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "modifier", xpath);
|
|
|
|
assert(wtype->modifiers==NULL);
|
|
|
|
wtype->modifiers = calloc(sizeof(weapon_mod), result->nodesetval->nodeNr+1);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
2004-06-11 21:59:02 +02:00
|
|
|
xmlXPathObjectPtr races;
|
|
|
|
int r, flags = 0;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
if (xml_bvalue(node, "walking", false)) flags|=WMF_WALKING;
|
|
|
|
if (xml_bvalue(node, "riding", false)) flags|=WMF_RIDING;
|
|
|
|
if (xml_bvalue(node, "against_walking", false)) flags|=WMF_AGAINST_WALKING;
|
|
|
|
if (xml_bvalue(node, "against_riding", false)) flags|=WMF_AGAINST_RIDING;
|
|
|
|
if (xml_bvalue(node, "offensive", false)) flags|=WMF_OFFENSIVE;
|
|
|
|
if (xml_bvalue(node, "defensive", false)) flags|=WMF_DEFENSIVE;
|
|
|
|
|
2004-06-11 21:59:02 +02:00
|
|
|
property = xmlGetProp(node, BAD_CAST "type");
|
|
|
|
if (strcmp((const char*)property, "damage")==0) flags|=WMF_DAMAGE;
|
|
|
|
else if (strcmp((const char*)property, "skill")==0) flags|=WMF_SKILL;
|
|
|
|
else if (strcmp((const char*)property, "missile_target")==0) flags|=WMF_MISSILE_TARGET;
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
wtype->modifiers[k].flags = flags;
|
2004-03-28 22:53:47 +02:00
|
|
|
wtype->modifiers[k].value = xml_ivalue(node, "value", 0);
|
2004-06-11 21:59:02 +02:00
|
|
|
|
|
|
|
xpath->node = node;
|
|
|
|
races = xmlXPathEvalExpression(BAD_CAST "race", xpath);
|
|
|
|
for (r=0;r!=races->nodesetval->nodeNr;++r) {
|
|
|
|
xmlNodePtr node = races->nodesetval->nodeTab[r];
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
if (property!=NULL) {
|
|
|
|
const race * rc = rc_find((const char*)property);
|
2005-10-08 20:38:26 +02:00
|
|
|
if (rc==NULL) rc = rc_add(rc_new((const char*)property));
|
2004-06-11 21:59:02 +02:00
|
|
|
racelist_insert(&wtype->modifiers[k].races, rc);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
}
|
2005-05-08 02:16:32 +02:00
|
|
|
xmlXPathFreeObject(races);
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
xpath->node = node;
|
|
|
|
return wtype;
|
|
|
|
}
|
|
|
|
|
|
|
|
static item_type *
|
|
|
|
xml_readitem(xmlXPathContextPtr xpath, resource_type * rtype)
|
|
|
|
{
|
|
|
|
xmlNodePtr node = xpath->node;
|
|
|
|
item_type * itype = NULL;
|
|
|
|
unsigned int flags = ITF_NONE;
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
int weight = xml_ivalue(node, "weight", 0);
|
|
|
|
int capacity = xml_ivalue(node, "capacity", 0);
|
|
|
|
|
|
|
|
if (xml_bvalue(node, "cursed", false)) flags |= ITF_CURSED;
|
|
|
|
if (xml_bvalue(node, "notlost", false)) flags |= ITF_NOTLOST;
|
|
|
|
if (xml_bvalue(node, "big", false)) flags |= ITF_BIG;
|
|
|
|
if (xml_bvalue(node, "animal", false)) flags |= ITF_ANIMAL;
|
|
|
|
itype = new_itemtype(rtype, flags, weight, capacity);
|
2005-07-31 18:07:02 +02:00
|
|
|
#ifdef SCORE_MODULE
|
|
|
|
itype->score = xml_ivalue(node, "score", 0);
|
|
|
|
#endif
|
2004-03-28 22:53:47 +02:00
|
|
|
|
2004-06-11 21:59:02 +02:00
|
|
|
/* reading item/construction */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
|
|
|
|
xml_readconstruction(xpath, result->nodesetval->nodeTab, result->nodesetval->nodeNr, &itype->construction);
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
/* reading item/weapon */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "weapon", xpath);
|
|
|
|
assert(result->nodesetval->nodeNr<=1);
|
|
|
|
if (result->nodesetval->nodeNr!=0) {
|
|
|
|
itype->flags |= ITF_WEAPON;
|
|
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
|
|
rtype->wtype = xml_readweapon(xpath, itype);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
2005-07-31 03:07:07 +02:00
|
|
|
/* reading item/armor */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "armor", xpath);
|
|
|
|
assert(result->nodesetval->nodeNr<=1);
|
|
|
|
if (result->nodesetval->nodeNr!=0) {
|
|
|
|
itype->flags |= ITF_WEAPON;
|
|
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
|
|
rtype->atype = xml_readarmor(xpath, itype);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
2005-01-31 00:33:28 +01:00
|
|
|
if (gamecode_enabled) {
|
|
|
|
/* reading item/function */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
xmlChar * property;
|
|
|
|
pf_generic fun;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
2005-10-08 13:02:10 +02:00
|
|
|
parse_function(node, &fun, &property);
|
2005-01-31 00:33:28 +01:00
|
|
|
if (fun==NULL) {
|
|
|
|
log_error(("unknown function name '%s' for item '%s'\n",
|
|
|
|
(const char*)property, rtype->_name[0]));
|
|
|
|
xmlFree(property);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (strcmp((const char*)property, "give")==0) {
|
|
|
|
itype->give = (boolean (*)(const struct unit*, const struct unit*, const struct item_type *, int, struct order *))fun;
|
|
|
|
}
|
|
|
|
else if (strcmp((const char*)property, "use")==0) {
|
|
|
|
itype->use = (int (*)(struct unit *, const struct item_type *, int, struct order *))fun;
|
|
|
|
} else {
|
|
|
|
log_error(("unknown function type '%s' for item '%s'\n",
|
|
|
|
(const char*)property, rtype->_name[0]));
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
2005-01-31 00:33:28 +01:00
|
|
|
xmlXPathFreeObject(result);
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return itype;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_resources(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr resources;
|
|
|
|
xmlNodeSetPtr nodes;
|
|
|
|
int i;
|
|
|
|
|
2005-10-03 23:52:09 +02:00
|
|
|
/* make sure old items (used in requirements) are available */
|
|
|
|
init_resources();
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
/* reading eressea/resources/resource */
|
|
|
|
resources = xmlXPathEvalExpression(BAD_CAST "/eressea/resources/resource", xpath);
|
|
|
|
nodes = resources->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
char *names[2], *appearance[2];
|
|
|
|
resource_type * rtype;
|
|
|
|
unsigned int flags = RTF_NONE;
|
|
|
|
xmlXPathObjectPtr result;
|
2004-04-12 02:57:09 +02:00
|
|
|
int k;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
2005-03-05 21:00:42 +01:00
|
|
|
if (xml_bvalue(node, "pooled", true)) flags |= RTF_POOLED;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
names[0] = strdup((const char*)property);
|
|
|
|
names[1] = strcat(strcpy((char*)malloc(strlen(names[0])+3), names[0]), "_p");
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "appearance");
|
|
|
|
if (property!=NULL) {
|
|
|
|
assert(property!=NULL);
|
|
|
|
appearance[0] = strdup((const char*)property);
|
|
|
|
appearance[1] = strcat(strcpy((char*)malloc(strlen(appearance[0])+3), appearance[0]), "_p");
|
|
|
|
rtype = new_resourcetype((const char**)names, (const char**)appearance, flags);
|
|
|
|
xmlFree(property);
|
|
|
|
free(appearance[0]);
|
|
|
|
free(appearance[1]);
|
|
|
|
} else {
|
|
|
|
rtype = new_resourcetype((const char**)names, NULL, flags);
|
|
|
|
}
|
|
|
|
free(names[0]);
|
|
|
|
free(names[1]);
|
|
|
|
|
2004-04-12 02:57:09 +02:00
|
|
|
/* reading eressea/resources/resource/function */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
2004-04-12 03:47:03 +02:00
|
|
|
if (result->nodesetval!=NULL) for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
2004-04-12 02:57:09 +02:00
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
pf_generic fun;
|
|
|
|
|
2005-10-08 13:02:10 +02:00
|
|
|
parse_function(node, &fun, &property);
|
2004-04-12 02:57:09 +02:00
|
|
|
if (fun==NULL) {
|
|
|
|
log_error(("unknown function name '%s' for resource %s\n",
|
|
|
|
(const char*)property, rtype->_name[0]));
|
|
|
|
xmlFree(property);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (strcmp((const char*)property, "change")==0) {
|
|
|
|
rtype->uchange = (rtype_uchange)fun;
|
|
|
|
} else if (strcmp((const char*)property, "get")==0) {
|
|
|
|
rtype->uget = (rtype_uget)fun;
|
|
|
|
} else if (strcmp((const char*)property, "name")==0) {
|
|
|
|
rtype->name = (rtype_name)fun;
|
|
|
|
} else {
|
|
|
|
log_error(("unknown function type '%s' for resource %s\n",
|
|
|
|
(const char*)property, rtype->_name[0]));
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/resources/resource/resourcelimit/function */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "resourcelimit/function", xpath);
|
2004-04-12 03:47:03 +02:00
|
|
|
if (result->nodesetval!=NULL) for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
2004-04-12 02:57:09 +02:00
|
|
|
attrib * a = a_find(rtype->attribs, &at_resourcelimit);
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
pf_generic fun;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "value");
|
|
|
|
assert(property!=NULL);
|
|
|
|
fun = get_function((const char*)property);
|
|
|
|
if (fun==NULL) {
|
|
|
|
log_error(("unknown limit '%s' for resource %s\n",
|
|
|
|
(const char*)property, rtype->_name[0]));
|
|
|
|
xmlFree(property);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
if (a==NULL) a = a_add(&rtype->attribs, a_new(&at_resourcelimit));
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (strcmp((const char*)property, "use")==0) {
|
|
|
|
resource_limit * rdata = (resource_limit*)a->data.v;
|
|
|
|
rdata->use = (rlimit_use)fun;
|
|
|
|
} else if (strcmp((const char*)property, "limit")==0) {
|
|
|
|
resource_limit * rdata = (resource_limit*)a->data.v;
|
|
|
|
rdata->limit = (rlimit_limit)fun;
|
|
|
|
} else {
|
|
|
|
log_error(("unknown limit '%s' for resource %s\n",
|
|
|
|
(const char*)property, rtype->_name[0]));
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
/* reading eressea/resources/resource/item */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "item", xpath);
|
|
|
|
assert(result->nodesetval->nodeNr<=1);
|
|
|
|
if (result->nodesetval->nodeNr!=0) {
|
|
|
|
rtype->flags |= RTF_ITEM;
|
|
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
|
|
rtype->itype = xml_readitem(xpath, rtype);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(resources);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
|
2005-10-03 23:52:09 +02:00
|
|
|
/* old resources now extern (for spells */
|
|
|
|
oldresourcetype[R_SWORD] = rt_find("sword");
|
2005-10-03 23:46:57 +02:00
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-10-02 17:54:24 +02:00
|
|
|
static void
|
2005-10-02 22:28:44 +02:00
|
|
|
add_items(equipment * eq, xmlNodeSetPtr nsetItems)
|
2005-01-03 22:28:21 +01:00
|
|
|
{
|
2005-10-02 22:28:44 +02:00
|
|
|
if (nsetItems!=NULL && nsetItems->nodeNr>0) {
|
2005-10-02 17:54:24 +02:00
|
|
|
int i;
|
2005-06-04 17:18:18 +02:00
|
|
|
for (i=0;i!=nsetItems->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nsetItems->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
const struct item_type * itype;
|
2005-01-03 22:28:21 +01:00
|
|
|
|
2005-06-04 17:18:18 +02:00
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
itype = it_find((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
if (itype!=NULL) {
|
|
|
|
property = xmlGetProp(node, BAD_CAST "amount");
|
|
|
|
if (property!=NULL) {
|
2005-10-02 22:28:44 +02:00
|
|
|
equipment_setitem(eq, itype, (const char*)property);
|
2005-06-04 17:18:18 +02:00
|
|
|
xmlFree(property);
|
|
|
|
}
|
2005-01-03 22:28:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-10-02 17:54:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2005-10-02 22:28:44 +02:00
|
|
|
add_skills(equipment * eq, xmlNodeSetPtr nsetSkills)
|
2005-10-02 17:54:24 +02:00
|
|
|
{
|
2005-10-02 22:28:44 +02:00
|
|
|
if (nsetSkills!=NULL && nsetSkills->nodeNr>0) {
|
2005-10-02 17:54:24 +02:00
|
|
|
int i;
|
|
|
|
for (i=0;i!=nsetSkills->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nsetSkills->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
skill_t sk;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
sk = sk_find((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
if (sk!=NOSKILL) {
|
|
|
|
property = xmlGetProp(node, BAD_CAST "level");
|
|
|
|
if (property!=NULL) {
|
2005-10-02 22:28:44 +02:00
|
|
|
equipment_setskill(eq, sk, (const char*)property);
|
2005-10-02 17:54:24 +02:00
|
|
|
xmlFree(property);
|
2005-10-02 22:28:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_subsets(xmlDocPtr doc, equipment * eq, xmlNodeSetPtr nsetSkills)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
if (nsetSkills!=NULL && nsetSkills->nodeNr>0) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
eq->subsets = calloc(nsetSkills->nodeNr+1, sizeof(subset));
|
|
|
|
for (i=0;i!=nsetSkills->nodeNr;++i) {
|
|
|
|
xmlXPathObjectPtr xpathResult;
|
|
|
|
xmlNodePtr node = nsetSkills->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
|
|
|
|
eq->subsets[i].chance = 1.0f;
|
|
|
|
property = xmlGetProp(node, BAD_CAST "chance");
|
|
|
|
if (property!=NULL) {
|
|
|
|
eq->subsets[i].chance = (float)atof((const char *)property);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xpath->node = node;
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "set", xpath);
|
|
|
|
if (xpathResult->nodesetval) {
|
|
|
|
xmlNodeSetPtr nsetSets = xpathResult->nodesetval;
|
|
|
|
float totalChance = 0.0f;
|
|
|
|
|
|
|
|
if (nsetSets->nodeNr>0) {
|
|
|
|
int set;
|
|
|
|
eq->subsets[i].sets = calloc(nsetSets->nodeNr+1, sizeof(subsetitem));
|
|
|
|
for (set=0;set!=nsetSets->nodeNr;++set) {
|
|
|
|
xmlNodePtr nodeSet = nsetSets->nodeTab[set];
|
|
|
|
float chance = 1.0f;
|
|
|
|
|
|
|
|
property = xmlGetProp(nodeSet, BAD_CAST "chance");
|
|
|
|
if (property!=NULL) {
|
|
|
|
chance = (float)atof((const char *)property);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
totalChance += chance;
|
|
|
|
|
|
|
|
property = xmlGetProp(nodeSet, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
eq->subsets[i].sets[set].chance = chance;
|
|
|
|
eq->subsets[i].sets[set].set = create_equipment((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (totalChance>1.0f) {
|
|
|
|
log_error(("total chance exceeds 1.0: %f in equipment set %s.\n",
|
|
|
|
totalChance, eq->name));
|
2005-10-02 17:54:24 +02:00
|
|
|
}
|
|
|
|
}
|
2005-10-02 22:28:44 +02:00
|
|
|
xmlXPathFreeObject(xpathResult);
|
2005-10-02 17:54:24 +02:00
|
|
|
}
|
|
|
|
}
|
2005-10-02 22:28:44 +02:00
|
|
|
xmlXPathFreeContext(xpath);
|
2005-10-02 17:54:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_equipment(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr xpathRaces;
|
|
|
|
|
|
|
|
/* reading eressea/races/race */
|
2005-10-02 22:28:44 +02:00
|
|
|
xpathRaces = xmlXPathEvalExpression(BAD_CAST "/eressea/equipment/set", xpath);
|
2005-10-02 17:54:24 +02:00
|
|
|
if (xpathRaces->nodesetval) {
|
|
|
|
xmlNodeSetPtr nsetRaces = xpathRaces->nodesetval;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i=0;i!=nsetRaces->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nsetRaces->nodeTab[i];
|
2005-10-02 22:28:44 +02:00
|
|
|
xmlChar * property = xmlGetProp(node, BAD_CAST "name");
|
2005-10-02 17:54:24 +02:00
|
|
|
|
|
|
|
if (property!=NULL) {
|
2005-10-02 22:28:44 +02:00
|
|
|
equipment * eq = create_equipment((const char*)property);
|
|
|
|
xmlXPathObjectPtr xpathResult;
|
2005-10-02 17:54:24 +02:00
|
|
|
|
2005-10-02 22:28:44 +02:00
|
|
|
xpath->node = node;
|
2005-10-02 17:54:24 +02:00
|
|
|
|
2005-10-02 22:28:44 +02:00
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "item", xpath);
|
|
|
|
add_items(eq, xpathResult->nodesetval);
|
|
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "skill", xpath);
|
|
|
|
add_skills(eq, xpathResult->nodesetval);
|
|
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "subset", xpath);
|
|
|
|
add_subsets(doc, eq, xpathResult->nodesetval);
|
|
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
2005-10-02 17:54:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlXPathFreeObject(xpathRaces);
|
2005-01-03 22:28:21 +01:00
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-10-08 13:02:10 +02:00
|
|
|
static int
|
|
|
|
parse_spells(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr spells;
|
|
|
|
xmlNodeSetPtr nodes;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* reading eressea/spells/spell */
|
|
|
|
spells = xmlXPathEvalExpression(BAD_CAST "/eressea/spells/spell", xpath);
|
|
|
|
nodes = spells->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
int k;
|
|
|
|
spell * sp = calloc(1, sizeof(spell));
|
|
|
|
|
|
|
|
/* spellname */
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
sp->sname = strdup((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
/* magic type */
|
|
|
|
property = xmlGetProp(node, BAD_CAST "type");
|
|
|
|
assert(property!=NULL);
|
|
|
|
for (sp->magietyp=0;sp->magietyp!=MAXMAGIETYP;++sp->magietyp) {
|
|
|
|
if (strcmp(magietypen[sp->magietyp], (const char *)property)==0) break;
|
|
|
|
}
|
|
|
|
assert(sp->magietyp!=MAXMAGIETYP);
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
/* level, rank and flags */
|
|
|
|
sp->level = xml_ivalue(node, "level", -1);
|
|
|
|
sp->rank = (char)xml_ivalue(node, "rank", -1);
|
|
|
|
if (xml_bvalue(node, "ship", false)) sp->sptyp |= ONSHIPCAST;
|
|
|
|
if (xml_bvalue(node, "ocean", false)) sp->sptyp |= OCEANCASTABLE;
|
|
|
|
if (xml_bvalue(node, "far", false)) sp->sptyp |= FARCASTING;
|
|
|
|
|
|
|
|
/* reading eressea/spells/spell/function */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
pf_generic fun;
|
|
|
|
|
|
|
|
parse_function(node, &fun, &property);
|
|
|
|
if (fun==NULL) {
|
|
|
|
log_error(("unknown function name '%s' for spell '%s'\n",
|
|
|
|
(const char*)property, sp->sname));
|
|
|
|
xmlFree(property);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (strcmp((const char*)property, "cast")==0) {
|
|
|
|
sp->sp_function = (spell_f)fun;
|
|
|
|
} else if (strcmp((const char*)property, "fumble")==0) {
|
|
|
|
sp->patzer = (pspell_f)fun;
|
|
|
|
} else {
|
|
|
|
log_error(("unknown function type '%s' for spell %s\n",
|
|
|
|
(const char*)property, sp->sname));
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/spells/spell/resource */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "resource", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr && k!=MAXINGREDIENT;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
resource_t res;
|
|
|
|
sp->komponenten[k][0] = 0;
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property);
|
|
|
|
for (res=0;res!=MAX_RESOURCES;++res) {
|
|
|
|
if (strcmp(oldresourcetype[res]->_name[0], (const char *)property)==0) {
|
|
|
|
sp->komponenten[k][0] = res;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
sp->komponenten[k][1] = (resource_t)xml_ivalue(node, "amount", 1);
|
|
|
|
sp->komponenten[k][2] = SPC_FIX;
|
|
|
|
property = xmlGetProp(node, BAD_CAST "cost");
|
|
|
|
if (property!=NULL) {
|
|
|
|
if (strcmp((const char *)property, "linear")==0) {
|
|
|
|
sp->komponenten[k][2] = SPC_LINEAR;
|
|
|
|
} else if (strcmp((const char *)property, "level")==0) {
|
|
|
|
sp->komponenten[k][2] = SPC_LEVEL;
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (k<MAXINGREDIENT) sp->komponenten[k][0] = 0;
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
sp->id = 0;
|
|
|
|
register_spell(sp);
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlXPathFreeObject(spells);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
2005-10-08 20:27:40 +02:00
|
|
|
|
|
|
|
init_spells();
|
|
|
|
|
2005-10-08 13:02:10 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
static int
|
|
|
|
parse_races(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr races;
|
|
|
|
xmlNodeSetPtr nodes;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* reading eressea/races/race */
|
|
|
|
races = xmlXPathEvalExpression(BAD_CAST "/eressea/races/race", xpath);
|
|
|
|
nodes = races->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * property;
|
|
|
|
race * rc;
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
rc = rc_find((const char*)property);
|
|
|
|
if (rc==NULL) rc = rc_add(rc_new((const char*)property));
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "damage");
|
|
|
|
assert(property!=NULL);
|
|
|
|
rc->def_damage = strdup((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
|
2005-05-05 04:50:14 +02:00
|
|
|
rc->magres = (float)xml_fvalue(node, "magres", 0.0);
|
|
|
|
rc->maxaura = (float)xml_fvalue(node, "maxaura", 0.0);
|
|
|
|
rc->regaura = (float)xml_fvalue(node, "regaura", 1.0);
|
2004-03-28 22:53:47 +02:00
|
|
|
rc->recruitcost = xml_ivalue(node, "recruitcost", 0);
|
|
|
|
rc->maintenance = xml_ivalue(node, "maintenance", 0);
|
|
|
|
rc->weight = xml_ivalue(node, "weight", 0);
|
2005-09-03 02:30:03 +02:00
|
|
|
#ifdef RACE_CAPACITY
|
2004-03-28 22:53:47 +02:00
|
|
|
rc->capacity = xml_ivalue(node, "capacity", 0);
|
|
|
|
#endif
|
2005-05-05 04:50:14 +02:00
|
|
|
rc->speed = (float)xml_fvalue(node, "speed", 1.0F);
|
2004-03-28 22:53:47 +02:00
|
|
|
rc->hitpoints = xml_ivalue(node, "hp", 0);
|
|
|
|
rc->armor = (char)xml_ivalue(node, "ac", 0);
|
|
|
|
|
|
|
|
rc->at_default = (char)xml_ivalue(node, "unarmedattack", -2);
|
|
|
|
rc->df_default = (char)xml_ivalue(node, "unarmeddefense", -2);
|
|
|
|
rc->at_bonus = (char)xml_ivalue(node, "attackmodifier", 0);
|
|
|
|
rc->df_bonus = (char)xml_ivalue(node, "defensemodifier", 0);
|
|
|
|
|
|
|
|
if (xml_bvalue(node, "playerrace", false)) rc->flags |= RCF_PLAYERRACE;
|
|
|
|
if (xml_bvalue(node, "scarepeasants", false)) rc->flags |= RCF_SCAREPEASANTS;
|
|
|
|
if (xml_bvalue(node, "cannotmove", false)) rc->flags |= RCF_CANNOTMOVE;
|
|
|
|
if (xml_bvalue(node, "fly", false)) rc->flags |= RCF_FLY;
|
2005-04-16 13:24:50 +02:00
|
|
|
if (xml_bvalue(node, "coastal", false)) rc->flags |= RCF_COASTAL;
|
2004-03-28 22:53:47 +02:00
|
|
|
if (xml_bvalue(node, "swim", false)) rc->flags |= RCF_SWIM;
|
|
|
|
if (xml_bvalue(node, "walk", false)) rc->flags |= RCF_WALK;
|
|
|
|
if (xml_bvalue(node, "nolearn", false)) rc->flags |= RCF_NOLEARN;
|
|
|
|
if (xml_bvalue(node, "noteach", false)) rc->flags |= RCF_NOTEACH;
|
|
|
|
if (xml_bvalue(node, "horse", false)) rc->flags |= RCF_HORSE;
|
|
|
|
if (xml_bvalue(node, "desert", false)) rc->flags |= RCF_DESERT;
|
|
|
|
if (xml_bvalue(node, "absorbpeasants", false)) rc->flags |= RCF_ABSORBPEASANTS;
|
|
|
|
if (xml_bvalue(node, "noheal", false)) rc->flags |= RCF_NOHEAL;
|
|
|
|
if (xml_bvalue(node, "noweapons", false)) rc->flags |= RCF_NOWEAPONS;
|
|
|
|
if (xml_bvalue(node, "shapeshift", false)) rc->flags |= RCF_SHAPESHIFT;
|
|
|
|
if (xml_bvalue(node, "shapeshiftany", false)) rc->flags |= RCF_SHAPESHIFTANY;
|
|
|
|
if (xml_bvalue(node, "illusionary", false)) rc->flags |= RCF_ILLUSIONARY;
|
|
|
|
if (xml_bvalue(node, "undead", false)) rc->flags |= RCF_UNDEAD;
|
|
|
|
if (xml_bvalue(node, "dragon", false)) rc->flags |= RCF_DRAGON;
|
|
|
|
|
|
|
|
if (xml_bvalue(node, "nogive", false)) rc->ec_flags |= NOGIVE;
|
|
|
|
if (xml_bvalue(node, "giveitem", false)) rc->ec_flags |= GIVEITEM;
|
|
|
|
if (xml_bvalue(node, "giveperson", false)) rc->ec_flags |= GIVEPERSON;
|
|
|
|
if (xml_bvalue(node, "giveunit", false)) rc->ec_flags |= GIVEUNIT;
|
|
|
|
if (xml_bvalue(node, "getitem", false)) rc->ec_flags |= GETITEM;
|
|
|
|
if (xml_bvalue(node, "canguard", false)) rc->ec_flags |= CANGUARD;
|
|
|
|
if (xml_bvalue(node, "recruithorses", false)) rc->ec_flags |= ECF_REC_HORSES;
|
|
|
|
if (xml_bvalue(node, "recruitethereal", false)) rc->ec_flags |= ECF_REC_ETHEREAL;
|
|
|
|
if (xml_bvalue(node, "recruitunlimited", false)) rc->ec_flags |= ECF_REC_UNLIMITED;
|
|
|
|
|
|
|
|
if (xml_bvalue(node, "equipment", false)) rc->battle_flags |= BF_EQUIPMENT;
|
|
|
|
if (xml_bvalue(node, "noblock", false)) rc->battle_flags |= BF_NOBLOCK;
|
|
|
|
if (xml_bvalue(node, "invinciblenonmagic", false)) rc->battle_flags |= BF_INV_NONMAGIC;
|
|
|
|
if (xml_bvalue(node, "resistbash", false)) rc->battle_flags |= BF_RES_BASH;
|
|
|
|
if (xml_bvalue(node, "resistcut", false)) rc->battle_flags |= BF_RES_CUT;
|
|
|
|
if (xml_bvalue(node, "resistpierce", false)) rc->battle_flags |= BF_RES_PIERCE;
|
|
|
|
|
|
|
|
/* reading eressea/races/race/ai */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "ai", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
|
|
|
|
rc->splitsize = xml_ivalue(node, "splitsize", 0);
|
2005-05-05 04:50:14 +02:00
|
|
|
rc->aggression = (float)xml_fvalue(node, "aggression", 0.04);
|
2004-03-28 22:53:47 +02:00
|
|
|
if (xml_bvalue(node, "killpeasants", false)) rc->flags |= RCF_KILLPEASANTS;
|
|
|
|
if (xml_bvalue(node, "moverandom", false)) rc->flags |= RCF_MOVERANDOM;
|
|
|
|
if (xml_bvalue(node, "learn", false)) rc->flags |= RCF_LEARN;
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/races/race/skill */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "skill", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
int mod = xml_ivalue(node, "modifier", 0);
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (mod!=0) {
|
|
|
|
skill_t sk = sk_find((const char*)property);
|
|
|
|
if (sk!=NOSKILL) {
|
|
|
|
rc->bonus[sk] = (char)mod;
|
|
|
|
} else {
|
|
|
|
log_error(("unknown skill '%s' in race '%s'\n",
|
|
|
|
(const char*)property, rc->_name[0]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/races/race/function */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
pf_generic fun;
|
|
|
|
|
2005-10-08 13:02:10 +02:00
|
|
|
parse_function(node, &fun, &property);
|
2004-03-28 22:53:47 +02:00
|
|
|
if (fun==NULL) {
|
|
|
|
log_error(("unknown function name '%s' for race %s\n",
|
|
|
|
(const char*)property, rc->_name[0]));
|
|
|
|
xmlFree(property);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
assert(property!=NULL);
|
|
|
|
if (strcmp((const char*)property, "name")==0) {
|
|
|
|
rc->generate_name = (const char* (*)(const struct unit*))fun;
|
|
|
|
} else if (strcmp((const char*)property, "age")==0) {
|
|
|
|
rc->age = (void(*)(struct unit*))fun;
|
|
|
|
} else if (strcmp((const char*)property, "move")==0) {
|
|
|
|
rc->move_allowed = (boolean(*)(const struct region *, const struct region *))fun;
|
|
|
|
} else if (strcmp((const char*)property, "itemdrop")==0) {
|
|
|
|
rc->itemdrop = (struct item *(*)(const struct race *, int))fun;
|
|
|
|
} else if (strcmp((const char*)property, "initfamiliar")==0) {
|
|
|
|
rc->init_familiar = (void(*)(struct unit *))fun;
|
|
|
|
} else {
|
|
|
|
log_error(("unknown function type '%s' for race %s\n",
|
|
|
|
(const char*)property, rc->_name[0]));
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/races/race/familiar */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "familiar", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
race * frc;
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "race");
|
|
|
|
assert(property!=NULL);
|
|
|
|
frc = rc_find((const char *)property);
|
|
|
|
if (frc == NULL) {
|
|
|
|
log_error(("%s not registered, is familiar for %s\n",
|
|
|
|
(const char*)property, rc->_name[0]));
|
|
|
|
assert(frc!=NULL);
|
|
|
|
frc = rc_add(rc_new((const char*)property));
|
|
|
|
}
|
|
|
|
if (xml_bvalue(node, "default", false)) {
|
|
|
|
rc->familiars[k] = rc->familiars[0];
|
|
|
|
rc->familiars[0] = frc;
|
|
|
|
} else {
|
|
|
|
rc->familiars[k] = frc;
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/races/race/precombatspell */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "precombatspell", xpath);
|
2005-10-08 20:27:40 +02:00
|
|
|
assert(rc->precombatspell==NULL || !"precombatspell is already initialized");
|
2004-03-28 22:53:47 +02:00
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
2005-10-08 21:22:56 +02:00
|
|
|
rc->precombatspell = xml_spell(node, "spell");
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/races/race/attack */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "attack", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
struct att * a = &rc->attack[k];
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "damage");
|
|
|
|
if (property!=NULL) {
|
|
|
|
a->data.dice = strdup((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
} else {
|
2005-10-08 21:22:56 +02:00
|
|
|
a->data.sp = xml_spell(node, "spell");
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
a->type = xml_ivalue(node, "type", 0);
|
|
|
|
a->flags = xml_ivalue(node, "flags", 0);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(races);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
|
|
|
|
race_compat();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_messages(xmlDocPtr doc)
|
|
|
|
{
|
2005-06-13 23:40:32 +02:00
|
|
|
xmlXPathContextPtr xpath;
|
2004-03-28 22:53:47 +02:00
|
|
|
xmlXPathObjectPtr messages;
|
|
|
|
xmlNodeSetPtr nodes;
|
|
|
|
int i;
|
|
|
|
|
2005-06-13 23:40:32 +02:00
|
|
|
if (!gamecode_enabled) return 0;
|
|
|
|
|
|
|
|
xpath = xmlXPathNewContext(doc);
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
/* reading eressea/strings/string */
|
|
|
|
messages = xmlXPathEvalExpression(BAD_CAST "/eressea/messages/message", xpath);
|
|
|
|
nodes = messages->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
const char * default_section = "events";
|
|
|
|
xmlChar * section;
|
|
|
|
xmlChar * property;
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
int k;
|
|
|
|
char ** argv = NULL;
|
|
|
|
const message_type * mtype;
|
|
|
|
|
|
|
|
/* arguments */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "type/arg", xpath);
|
|
|
|
if (result->nodesetval->nodeNr>0) {
|
|
|
|
argv = malloc(sizeof(char*)*(result->nodesetval->nodeNr+1));
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
char zBuffer[128];
|
|
|
|
xmlChar * name, * type;
|
|
|
|
|
|
|
|
name = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
type = xmlGetProp(node, BAD_CAST "type");
|
|
|
|
sprintf(zBuffer, "%s:%s", (const char*)name, (const char*)type);
|
|
|
|
xmlFree(name);
|
|
|
|
xmlFree(type);
|
|
|
|
argv[k] = strdup(zBuffer);
|
|
|
|
}
|
|
|
|
argv[result->nodesetval->nodeNr] = NULL;
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* add the messagetype */
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
mtype = mt_find((const char *)property);
|
|
|
|
if (mtype==NULL) {
|
|
|
|
mtype = mt_register(mt_new((const char *)property, (const char**)argv));
|
|
|
|
} else {
|
|
|
|
assert(argv!=NULL || !"cannot redefine arguments of message now");
|
|
|
|
}
|
|
|
|
xmlFree(property);
|
|
|
|
|
|
|
|
/* register the type for the CR */
|
|
|
|
crt_register(mtype);
|
|
|
|
|
|
|
|
/* let's clean up the mess */
|
|
|
|
if (argv!=NULL) {
|
|
|
|
for (k=0;argv[k]!=NULL;++k) free(argv[k]);
|
|
|
|
free(argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
section = xmlGetProp(node, BAD_CAST "section");
|
|
|
|
if (section==NULL) section = BAD_CAST default_section;
|
|
|
|
mc_add((const char*)section);
|
|
|
|
|
|
|
|
/* strings */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "text", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
struct locale * lang;
|
|
|
|
xmlChar * text;
|
|
|
|
|
|
|
|
xml_readtext(node, &lang, &text);
|
2005-05-27 22:22:17 +02:00
|
|
|
xml_cleanup_string(text);
|
2004-03-28 22:53:47 +02:00
|
|
|
nrt_register(mtype, lang, xml_to_locale(text), 0, (const char*)section);
|
|
|
|
xmlFree(text);
|
|
|
|
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
if (section != BAD_CAST default_section) xmlFree(section);
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlXPathFreeObject(messages);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
xml_readstrings(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr, boolean names)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i=0;i!=nodeNr;++i) {
|
|
|
|
xmlNodePtr stringNode = nodeTab[i];
|
|
|
|
xmlChar * name = xmlGetProp(stringNode, BAD_CAST "name");
|
|
|
|
xmlChar * nspc = NULL;
|
|
|
|
xmlXPathObjectPtr result;
|
|
|
|
int k;
|
|
|
|
char zName[128];
|
|
|
|
|
2004-04-09 03:23:54 +02:00
|
|
|
assert(name!=NULL);
|
2004-03-28 22:53:47 +02:00
|
|
|
if (names) nspc = xmlGetProp(stringNode->parent, BAD_CAST "name");
|
|
|
|
mkname_buf((const char*)nspc, (const char*)name, zName);
|
|
|
|
if (nspc!=NULL) xmlFree(nspc);
|
|
|
|
xmlFree(name);
|
|
|
|
|
|
|
|
/* strings */
|
|
|
|
xpath->node = stringNode;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "text", xpath);
|
|
|
|
for (k=0;k!=result->nodesetval->nodeNr;++k) {
|
|
|
|
xmlNodePtr textNode = result->nodesetval->nodeTab[k];
|
|
|
|
struct locale * lang;
|
|
|
|
xmlChar * text;
|
|
|
|
|
|
|
|
xml_readtext(textNode, &lang, &text);
|
2004-04-10 22:25:40 +02:00
|
|
|
if (text!=NULL) {
|
|
|
|
assert(strcmp(zName, (const char*)xml_cleanup_string(BAD_CAST zName))==0);
|
|
|
|
xml_cleanup_string(text);
|
|
|
|
locale_setstring(lang, zName, xml_to_locale(text));
|
|
|
|
xmlFree(text);
|
|
|
|
} else {
|
|
|
|
log_warning(("string %s has no text in locale %s\n",
|
|
|
|
zName, locale_name(lang)));
|
|
|
|
}
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_strings(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr strings;
|
|
|
|
|
|
|
|
/* reading eressea/strings/string */
|
|
|
|
strings = xmlXPathEvalExpression(BAD_CAST "/eressea/strings/string", xpath);
|
|
|
|
xml_readstrings(xpath, strings->nodesetval->nodeTab, strings->nodesetval->nodeNr, false);
|
|
|
|
xmlXPathFreeObject(strings);
|
|
|
|
|
|
|
|
strings = xmlXPathEvalExpression(BAD_CAST "/eressea/strings/namespace/string", xpath);
|
|
|
|
xml_readstrings(xpath, strings->nodesetval->nodeTab, strings->nodesetval->nodeNr, true);
|
|
|
|
xmlXPathFreeObject(strings);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-08-07 09:42:22 +02:00
|
|
|
static void
|
|
|
|
xml_readprefixes(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr, boolean names)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i=0;i!=nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodeTab[i];
|
|
|
|
xmlChar * text = xmlNodeListGetString(node->doc, node->children, 1);
|
|
|
|
|
|
|
|
if (text!=NULL) {
|
|
|
|
add_raceprefix((const char*)text);
|
|
|
|
xmlFree(text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_prefixes(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr strings;
|
|
|
|
|
|
|
|
/* reading eressea/strings/string */
|
|
|
|
strings = xmlXPathEvalExpression(BAD_CAST "/eressea/prefixes/prefix", xpath);
|
|
|
|
xml_readprefixes(xpath, strings->nodesetval->nodeTab, strings->nodesetval->nodeNr, false);
|
|
|
|
xmlXPathFreeObject(strings);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
static int
|
|
|
|
parse_main(xmlDocPtr doc)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
|
|
xmlXPathObjectPtr result = xmlXPathEvalExpression(BAD_CAST "/eressea/game", xpath);
|
|
|
|
xmlNodeSetPtr nodes = result->nodesetval;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
xmlChar * property;
|
|
|
|
xmlNodePtr node = nodes->nodeTab[0];
|
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "welcome");
|
|
|
|
if (property!=NULL) {
|
|
|
|
global.welcomepath = strdup((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
2005-05-20 11:08:39 +02:00
|
|
|
|
2005-05-21 00:32:53 +02:00
|
|
|
global.unitsperalliance = xml_bvalue(node, "unitsperalliance", false);
|
2005-05-21 00:51:37 +02:00
|
|
|
global.maxunits = xml_ivalue(node, "units", INT_MAX);
|
2004-03-28 22:53:47 +02:00
|
|
|
|
|
|
|
property = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
if (property!=NULL) {
|
|
|
|
global.gamename = strdup((const char*)property);
|
|
|
|
xmlFree(property);
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/game/param */
|
|
|
|
xpath->node = node;
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "param", xpath);
|
|
|
|
nodes = result->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * name = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
xmlChar * value = xmlGetProp(node, BAD_CAST "value");
|
|
|
|
|
|
|
|
set_param(&global.parameters, (const char*)name, (const char*)value);
|
|
|
|
|
|
|
|
xmlFree(name);
|
|
|
|
xmlFree(value);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
/* reading eressea/game/order */
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "order", xpath);
|
|
|
|
nodes = result->nodesetval;
|
|
|
|
for (i=0;i!=nodes->nodeNr;++i) {
|
|
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
|
|
xmlChar * name = xmlGetProp(node, BAD_CAST "name");
|
2004-04-12 01:56:47 +02:00
|
|
|
boolean disable = xml_bvalue(node, "disable", false);
|
|
|
|
|
|
|
|
if (disable) {
|
|
|
|
int k;
|
|
|
|
for (k=0;k!=MAXKEYWORDS;++k) {
|
|
|
|
if (strcmp(keywords[k], (const char*)name)==0) {
|
|
|
|
global.disabled[k]=1;
|
|
|
|
break;
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
}
|
2004-04-12 01:56:47 +02:00
|
|
|
if (k==MAXKEYWORDS) {
|
|
|
|
log_error(("trying to disable unknown comand %s\n", (const char*)name));
|
|
|
|
}
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|
|
|
|
xmlFree(name);
|
|
|
|
}
|
|
|
|
xmlXPathFreeObject(result);
|
|
|
|
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
register_xmlreader(void)
|
|
|
|
{
|
|
|
|
xml_register_callback(parse_main);
|
|
|
|
|
|
|
|
xml_register_callback(parse_strings);
|
2004-08-07 09:42:22 +02:00
|
|
|
xml_register_callback(parse_prefixes);
|
2005-06-13 23:40:32 +02:00
|
|
|
xml_register_callback(parse_messages);
|
2004-03-28 22:53:47 +02:00
|
|
|
xml_register_callback(parse_resources);
|
2005-10-08 20:38:26 +02:00
|
|
|
|
|
|
|
xml_register_callback(parse_buildings); /* requires resources */
|
|
|
|
xml_register_callback(parse_ships); /* requires resources */
|
|
|
|
xml_register_callback(parse_equipment); /* requires resources */
|
|
|
|
xml_register_callback(parse_spells); /* requires resources */
|
|
|
|
xml_register_callback(parse_races); /* requires spells */
|
2005-06-04 15:22:31 +02:00
|
|
|
xml_register_callback(parse_calendar);
|
2004-03-28 22:53:47 +02:00
|
|
|
}
|