forked from github/server
e2808eeea9
remove default values from xml to make it smaller and simpler.
2092 lines
77 KiB
C
2092 lines
77 KiB
C
/*
|
|
+-------------------+
|
|
| | 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 <platform.h>
|
|
#include <kernel/config.h>
|
|
#include "xmlreader.h"
|
|
|
|
#include "building.h"
|
|
#include "equipment.h"
|
|
#include "item.h"
|
|
#include "keyword.h"
|
|
#include "messages.h"
|
|
#include "race.h"
|
|
#include "region.h"
|
|
#include "resources.h"
|
|
#include "ship.h"
|
|
#include "terrain.h"
|
|
#include "skills.h"
|
|
#include "spell.h"
|
|
#include "spellbook.h"
|
|
#include "calendar.h"
|
|
#include "prefix.h"
|
|
|
|
#include "vortex.h"
|
|
|
|
#include <modules/score.h>
|
|
|
|
/* util includes */
|
|
#include <util/attrib.h>
|
|
#include <util/bsdstring.h>
|
|
#include <util/crmessage.h>
|
|
#include <util/functions.h>
|
|
#include <util/language.h>
|
|
#include <util/log.h>
|
|
#include <util/message.h>
|
|
#include <util/nrmessage.h>
|
|
#include <util/xml.h>
|
|
|
|
#ifdef USE_LIBXML2
|
|
/* libxml includes */
|
|
#include <libxml/tree.h>
|
|
#include <libxml/xpath.h>
|
|
#include <libxml/encoding.h>
|
|
#endif
|
|
|
|
/* libc includes */
|
|
#include <assert.h>
|
|
#include <ctype.h>
|
|
#include <limits.h>
|
|
#include <string.h>
|
|
|
|
#ifdef USE_LIBXML2
|
|
static void xml_readtext(xmlNodePtr node, struct locale **lang, xmlChar ** text)
|
|
{
|
|
xmlChar *propValue = xmlGetProp(node, BAD_CAST "locale");
|
|
assert(propValue != NULL);
|
|
*lang = get_locale((const char *)propValue);
|
|
#ifdef MAKE_LOCALES
|
|
if (*lang == NULL)
|
|
*lang = get_or_create_locale((const char *)propValue);
|
|
#endif
|
|
xmlFree(propValue);
|
|
|
|
*text = xmlNodeListGetString(node->doc, node->children, 1);
|
|
}
|
|
|
|
static const spell *xml_spell(xmlNode * node, const char *name)
|
|
{
|
|
const spell *sp = NULL;
|
|
xmlChar *propValue = xmlGetProp(node, BAD_CAST name);
|
|
if (propValue != NULL) {
|
|
sp = find_spell((const char *)propValue);
|
|
assert(sp);
|
|
xmlFree(propValue);
|
|
}
|
|
return sp;
|
|
}
|
|
|
|
static xmlChar *xml_cleanup_string(xmlChar * str)
|
|
{
|
|
xmlChar *read = str;
|
|
xmlChar *write = str;
|
|
|
|
while (*read) {
|
|
/* eat leading whitespace */
|
|
if (*read && isxspace(*read)) {
|
|
while (*read && isxspace(*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 *propValue;
|
|
|
|
radd->number = xml_ivalue(node, "quantity", 1);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "type");
|
|
radd->rtype = rt_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
|
|
++radd;
|
|
}
|
|
}
|
|
|
|
void
|
|
xml_readconstruction(xmlXPathContextPtr xpath, xmlNodeSetPtr nodeSet,
|
|
construction ** consPtr)
|
|
{
|
|
xmlNodePtr pushNode = xpath->node;
|
|
int k;
|
|
for (k = 0; k != nodeSet->nodeNr; ++k) {
|
|
xmlNodePtr node = nodeSet->nodeTab[k];
|
|
xmlChar *propValue;
|
|
construction *con;
|
|
xmlXPathObjectPtr req;
|
|
int m;
|
|
skill_t sk = NOSKILL;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "skill");
|
|
if (propValue != NULL) {
|
|
sk = findskill((const char *)propValue);
|
|
if (sk == NOSKILL) {
|
|
log_error("construction requires skill '%s' that does not exist.\n", (const char *)propValue);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
|
|
assert(*consPtr == NULL);
|
|
|
|
*consPtr = con = (construction *)calloc(sizeof(construction), 1);
|
|
consPtr = &con->improvement;
|
|
|
|
con->skill = sk;
|
|
con->maxsize = xml_ivalue(node, "maxsize", -1);
|
|
con->minskill = xml_ivalue(node, "minskill", -1);
|
|
con->reqsize = xml_ivalue(node, "reqsize", 1);
|
|
con->defense_bonus = xml_ivalue(node, "defense_bonus", 0);
|
|
con->close_combat_bonus = xml_ivalue(node, "close_combat_bonus", 0);
|
|
con->ranged_bonus = xml_ivalue(node, "ranged_bonus", 0);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "building");
|
|
if (propValue != NULL) {
|
|
con->btype = bt_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
|
|
/* read construction/requirement */
|
|
xpath->node = node;
|
|
req = xmlXPathEvalExpression(BAD_CAST "requirement", xpath);
|
|
xml_readrequirements(req->nodesetval->nodeTab,
|
|
req->nodesetval->nodeNr, &con->materials);
|
|
xmlXPathFreeObject(req);
|
|
|
|
/* 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];
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "function");
|
|
if (propValue != NULL) {
|
|
pf_generic foo = get_function((const char *)propValue);
|
|
a_add(&con->attribs, make_skillmod(NOSKILL, SMF_PRODUCTION,
|
|
(skillmod_fun)foo, 1.0, 0));
|
|
xmlFree(propValue);
|
|
}
|
|
|
|
}
|
|
xmlXPathFreeObject(req);
|
|
}
|
|
xpath->node = pushNode;
|
|
}
|
|
|
|
static int
|
|
parse_function(xmlNodePtr node, pf_generic * funPtr, xmlChar ** namePtr)
|
|
{
|
|
pf_generic fun;
|
|
xmlChar *propValue = xmlGetProp(node, BAD_CAST "value");
|
|
assert(propValue != NULL);
|
|
fun = get_function((const char *)propValue);
|
|
if (fun != NULL) {
|
|
xmlFree(propValue);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
}
|
|
*namePtr = propValue;
|
|
*funPtr = fun;
|
|
return 0;
|
|
}
|
|
|
|
static int parse_buildings(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr buildings;
|
|
int i;
|
|
|
|
/* reading eressea/buildings/building */
|
|
buildings =
|
|
xmlXPathEvalExpression(BAD_CAST "/eressea/buildings/building", xpath);
|
|
if (buildings->nodesetval != NULL) {
|
|
xmlNodeSetPtr nodes = buildings->nodesetval;
|
|
for (i = 0; i != nodes->nodeNr; ++i) {
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
xmlChar *propValue;
|
|
building_type *btype;
|
|
xmlXPathObjectPtr result;
|
|
int k;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
btype = bt_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
|
|
btype->capacity = xml_ivalue(node, "capacity", btype->capacity);
|
|
btype->maxcapacity = xml_ivalue(node, "maxcapacity", btype->maxcapacity);
|
|
btype->maxsize = xml_ivalue(node, "maxsize", btype->maxsize);
|
|
|
|
btype->magres = xml_ivalue(node, "magres", btype->magres);
|
|
btype->magresbonus = xml_ivalue(node, "magresbonus", btype->magresbonus);
|
|
btype->fumblebonus = xml_ivalue(node, "fumblebonus", btype->fumblebonus);
|
|
btype->auraregen = xml_fvalue(node, "auraregen", btype->auraregen);
|
|
|
|
if (xml_bvalue(node, "nodestroy", false))
|
|
btype->flags |= BTF_INDESTRUCTIBLE;
|
|
if (xml_bvalue(node, "oneperturn", false))
|
|
btype->flags |= BTF_ONEPERTURN;
|
|
if (xml_bvalue(node, "nobuild", false))
|
|
btype->flags |= BTF_NOBUILD;
|
|
if (xml_bvalue(node, "namechange", true))
|
|
btype->flags |= BTF_NAMECHANGE;
|
|
if (xml_bvalue(node, "unique", false))
|
|
btype->flags |= BTF_UNIQUE;
|
|
if (xml_bvalue(node, "decay", false))
|
|
btype->flags |= BTF_DECAY;
|
|
if (xml_bvalue(node, "magic", false))
|
|
btype->flags |= BTF_MAGIC;
|
|
if (xml_bvalue(node, "fort", false))
|
|
btype->flags |= BTF_FORTIFICATION;
|
|
|
|
/* reading eressea/buildings/building/construction */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
|
|
xml_readconstruction(xpath, result->nodesetval, &btype->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;
|
|
parse_function(node, &fun, &propValue);
|
|
|
|
if (fun == NULL) {
|
|
log_error("unknown function name '%s' for building %s\n", (const char *)propValue, btype->_name);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "name") == 0) {
|
|
btype->name =
|
|
(const char *(*)(const struct building_type *,
|
|
const struct building *, int))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "init") == 0) {
|
|
btype->init = (void(*)(struct building_type *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "age") == 0) {
|
|
btype->age = (void(*)(struct building *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "protection") == 0) {
|
|
btype->protection = (int(*)(struct building *, struct unit *, building_bonus))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "taxes") == 0) {
|
|
btype->taxes = (double(*)(const struct building *, int))fun;
|
|
}
|
|
else {
|
|
log_error("unknown function type '%s' for building %s\n", (const char *)propValue, btype->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
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 (btype->maintenance == NULL) {
|
|
btype->maintenance = (struct maintenance *)
|
|
calloc(sizeof(struct maintenance), result->nodesetval->nodeNr + 1);
|
|
}
|
|
mt = btype->maintenance + k;
|
|
mt->number = xml_ivalue(node, "amount", 0);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "type");
|
|
assert(propValue != NULL);
|
|
mt->rtype = rt_find((const char *)propValue);
|
|
assert(mt->rtype != NULL);
|
|
xmlFree(propValue);
|
|
|
|
if (xml_bvalue(node, "variable", false))
|
|
mt->flags |= MTF_VARIABLE;
|
|
if (xml_bvalue(node, "vital", false))
|
|
mt->flags |= MTF_VITAL;
|
|
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* finally, initialize the new building type */
|
|
if (btype->init)
|
|
btype->init(btype);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(buildings);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
return 0;
|
|
}
|
|
|
|
static int parse_calendar(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr xpathCalendars;
|
|
xmlNodeSetPtr nsetCalendars;
|
|
int c, rv = 0;
|
|
|
|
/* reading eressea/buildings/building */
|
|
xpathCalendars = xmlXPathEvalExpression(BAD_CAST "/eressea/calendar", xpath);
|
|
nsetCalendars = xpathCalendars->nodesetval;
|
|
months_per_year = 0;
|
|
if (nsetCalendars == NULL || nsetCalendars->nodeNr == 0) {
|
|
rv = -1;
|
|
}
|
|
else
|
|
for (c = 0; c != nsetCalendars->nodeNr; ++c) {
|
|
xmlNodePtr calendar = nsetCalendars->nodeTab[c];
|
|
xmlXPathObjectPtr xpathWeeks, xpathMonths, xpathSeasons;
|
|
xmlNodeSetPtr nsetWeeks, nsetMonths, nsetSeasons;
|
|
xmlChar *propValue = xmlGetProp(calendar, BAD_CAST "name");
|
|
xmlChar *newyear = xmlGetProp(calendar, BAD_CAST "newyear");
|
|
|
|
first_turn = xml_ivalue(calendar, "start", first_turn);
|
|
if (propValue) {
|
|
free(agename);
|
|
agename = _strdup(mkname("calendar", (const char *)propValue));
|
|
xmlFree(propValue);
|
|
}
|
|
|
|
xpath->node = calendar;
|
|
xpathWeeks = xmlXPathEvalExpression(BAD_CAST "week", xpath);
|
|
nsetWeeks = xpathWeeks->nodesetval;
|
|
if (nsetWeeks != NULL && nsetWeeks->nodeNr) {
|
|
int i;
|
|
|
|
weeks_per_month = nsetWeeks->nodeNr;
|
|
free(weeknames);
|
|
free(weeknames2);
|
|
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 *propValue = xmlGetProp(week, BAD_CAST "name");
|
|
if (propValue) {
|
|
weeknames[i] = _strdup(mkname("calendar", (const char *)propValue));
|
|
weeknames2[i] = malloc(strlen(weeknames[i]) + 3);
|
|
sprintf(weeknames2[i], "%s_d", weeknames[i]);
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
}
|
|
xmlXPathFreeObject(xpathWeeks);
|
|
|
|
xpathSeasons = xmlXPathEvalExpression(BAD_CAST "season", xpath);
|
|
nsetSeasons = xpathSeasons->nodesetval;
|
|
if (nsetSeasons != NULL && nsetSeasons->nodeNr) {
|
|
int i;
|
|
|
|
seasons = nsetSeasons->nodeNr;
|
|
assert(!seasonnames);
|
|
seasonnames = malloc(sizeof(char *) * seasons);
|
|
|
|
for (i = 0; i != nsetSeasons->nodeNr; ++i) {
|
|
xmlNodePtr season = nsetSeasons->nodeTab[i];
|
|
xmlChar *propValue = xmlGetProp(season, BAD_CAST "name");
|
|
if (propValue) {
|
|
seasonnames[i] =
|
|
_strdup(mkname("calendar", (const char *)propValue));
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
xpathMonths = xmlXPathEvalExpression(BAD_CAST "season/month", xpath);
|
|
nsetMonths = xpathMonths->nodesetval;
|
|
if (nsetMonths != NULL && nsetMonths->nodeNr) {
|
|
int i;
|
|
|
|
months_per_year = nsetMonths->nodeNr;
|
|
free(monthnames);
|
|
monthnames = malloc(sizeof(char *) * months_per_year);
|
|
free(month_season);
|
|
month_season = malloc(sizeof(int) * months_per_year);
|
|
free(storms);
|
|
storms = malloc(sizeof(int) * months_per_year);
|
|
|
|
for (i = 0; i != nsetMonths->nodeNr; ++i) {
|
|
xmlNodePtr month = nsetMonths->nodeTab[i];
|
|
xmlChar *propValue = xmlGetProp(month, BAD_CAST "name");
|
|
|
|
if (propValue) {
|
|
if (newyear
|
|
&& strcmp((const char *)newyear, (const char *)propValue) == 0) {
|
|
first_month = i;
|
|
xmlFree(newyear);
|
|
newyear = NULL;
|
|
}
|
|
monthnames[i] = _strdup(mkname("calendar", (const char *)propValue));
|
|
xmlFree(propValue);
|
|
}
|
|
if (nsetSeasons) {
|
|
int j;
|
|
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;
|
|
}
|
|
|
|
static int parse_ships(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr ships;
|
|
int i;
|
|
|
|
/* reading eressea/ships/ship */
|
|
ships = xmlXPathEvalExpression(BAD_CAST "/eressea/ships/ship", xpath);
|
|
if (ships->nodesetval != NULL) {
|
|
xmlNodeSetPtr nodes = ships->nodesetval;
|
|
for (i = 0; i != nodes->nodeNr; ++i) {
|
|
xmlNodePtr child, node = nodes->nodeTab[i];
|
|
xmlChar *propValue;
|
|
ship_type *st;
|
|
xmlXPathObjectPtr result;
|
|
int k, c;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
st = st_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
|
|
st->cabins = xml_ivalue(node, "cabins", 0) * PERSON_WEIGHT;
|
|
st->cargo = xml_ivalue(node, "cargo", st->cargo);
|
|
st->combat = xml_ivalue(node, "combat", st->combat);
|
|
st->damage = xml_fvalue(node, "damage", st->damage);
|
|
if (xml_bvalue(node, "nocoast", false))
|
|
st->flags |= SFL_NOCOAST;
|
|
if (xml_bvalue(node, "fly", false))
|
|
st->flags |= SFL_FLY;
|
|
if (xml_bvalue(node, "opensea", false))
|
|
st->flags |= SFL_OPENSEA;
|
|
st->fishing = xml_ivalue(node, "fishing", st->fishing);
|
|
st->cptskill = xml_ivalue(node, "cptskill", st->cptskill);
|
|
st->minskill = xml_ivalue(node, "minskill", st->minskill);
|
|
st->sumskill = xml_ivalue(node, "sumskill", st->sumskill);
|
|
st->range = xml_ivalue(node, "range", st->range);
|
|
st->range_max = xml_ivalue(node, "maxrange", st->range_max);
|
|
st->storm = xml_fvalue(node, "storm", st->storm);
|
|
|
|
/* reading eressea/ships/ship/construction */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
|
|
xml_readconstruction(xpath, result->nodesetval, &st->construction);
|
|
xmlXPathFreeObject(result);
|
|
|
|
for (child = node->children; child; child = child->next) {
|
|
if (strcmp((const char *)child->name, "modifier") == 0) {
|
|
double value = xml_fvalue(child, "value", 0.0);
|
|
propValue = xmlGetProp(child, BAD_CAST "type");
|
|
if (strcmp((const char *)propValue, "tactics") == 0)
|
|
st->tac_bonus = (float)value;
|
|
else if (strcmp((const char *)propValue, "attack") == 0)
|
|
st->at_bonus = (int)value;
|
|
else if (strcmp((const char *)propValue, "defense") == 0)
|
|
st->df_bonus = (int)value;
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
/* reading eressea/ships/ship/coast */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "coast", xpath);
|
|
for (c = 0, k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
if (k == 0) {
|
|
assert(st->coasts == NULL);
|
|
st->coasts =
|
|
(terrain_type **)malloc(sizeof(terrain_type *) *
|
|
(result->nodesetval->nodeNr + 1));
|
|
st->coasts[result->nodesetval->nodeNr] = NULL;
|
|
}
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "terrain");
|
|
assert(propValue != NULL);
|
|
st->coasts[c] = get_or_create_terrain((const char *)propValue);
|
|
if (st->coasts[c] != NULL)
|
|
++c;
|
|
else {
|
|
log_warning("ship %s mentions a non-existing terrain %s.\n", st->_name, propValue);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(ships);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
return 0;
|
|
}
|
|
|
|
static potion_type *xml_readpotion(xmlXPathContextPtr xpath, item_type * itype)
|
|
{
|
|
int level = xml_ivalue(xpath->node, "level", 0);
|
|
|
|
assert(level > 0);
|
|
return new_potiontype(itype, level);
|
|
}
|
|
|
|
static luxury_type *xml_readluxury(xmlXPathContextPtr xpath, item_type * itype)
|
|
{
|
|
int price = xml_ivalue(xpath->node, "price", 0);
|
|
return new_luxurytype(itype, price);
|
|
}
|
|
|
|
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);
|
|
atype->projectile = (float)xml_fvalue(node, "projectile", 0.0);
|
|
return atype;
|
|
}
|
|
|
|
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 *propValue;
|
|
int k;
|
|
skill_t sk;
|
|
int minskill = xml_ivalue(node, "minskill", 1);
|
|
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);
|
|
|
|
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;
|
|
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;
|
|
if (xml_bvalue(node, "siege", false))
|
|
flags |= WTF_SIEGE;
|
|
if (xml_bvalue(node, "horse", (flags & WTF_MISSILE) == 0))
|
|
flags |= WTF_HORSEBONUS;
|
|
if (xml_bvalue(node, "useshield", true))
|
|
flags |= WTF_USESHIELD;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "skill");
|
|
assert(propValue != NULL);
|
|
sk = findskill((const char *)propValue);
|
|
assert(sk != NOSKILL);
|
|
xmlFree(propValue);
|
|
|
|
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;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "type");
|
|
if (strcmp((const char *)propValue, "footman") != 0) {
|
|
pos = 1;
|
|
}
|
|
xmlFree(propValue);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "value");
|
|
wtype->damage[pos] = _strdup((const char *)propValue); // TODO: this is a memory leak
|
|
if (k == 0)
|
|
wtype->damage[1 - pos] = wtype->damage[pos];
|
|
xmlFree(propValue);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading weapon/modifier */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "modifier", xpath);
|
|
assert(wtype->modifiers == NULL);
|
|
wtype->modifiers = calloc(result->nodesetval->nodeNr + 1, sizeof(weapon_mod));
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
xmlXPathObjectPtr races;
|
|
int r, flags = 0;
|
|
|
|
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;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "type");
|
|
if (strcmp((const char *)propValue, "damage") == 0)
|
|
flags |= WMF_DAMAGE;
|
|
else if (strcmp((const char *)propValue, "skill") == 0)
|
|
flags |= WMF_SKILL;
|
|
else if (strcmp((const char *)propValue, "missile_target") == 0)
|
|
flags |= WMF_MISSILE_TARGET;
|
|
xmlFree(propValue);
|
|
|
|
wtype->modifiers[k].flags = flags;
|
|
wtype->modifiers[k].value = xml_ivalue(node, "value", 0);
|
|
|
|
xpath->node = node;
|
|
races = xmlXPathEvalExpression(BAD_CAST "race", xpath);
|
|
for (r = 0; r != races->nodesetval->nodeNr; ++r) {
|
|
xmlNodePtr node = races->nodesetval->nodeTab[r];
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
if (propValue != NULL) {
|
|
const race *rc = rc_find((const char *)propValue);
|
|
if (rc == NULL)
|
|
rc = rc_get_or_create((const char *)propValue);
|
|
racelist_insert(&wtype->modifiers[k].races, rc);
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(races);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading weapon/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 *propValue;
|
|
pf_generic fun;
|
|
|
|
parse_function(node, &fun, &propValue);
|
|
if (fun == NULL) {
|
|
log_error("unknown function name '%s' for item '%s'\n", (const char *)propValue, itype->rtype->_name);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "attack") == 0) {
|
|
wtype->attack =
|
|
(bool(*)(const struct troop *, const struct weapon_type *,
|
|
int *))fun;
|
|
}
|
|
else {
|
|
log_error("unknown function type '%s' for item '%s'\n", (const char *)propValue, itype->rtype->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
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;
|
|
|
|
if (xml_bvalue(node, "cursed", false))
|
|
flags |= ITF_CURSED;
|
|
if (xml_bvalue(node, "notlost", false))
|
|
flags |= ITF_NOTLOST;
|
|
if (xml_bvalue(node, "herb", false))
|
|
flags |= ITF_HERB;
|
|
if (xml_bvalue(node, "big", false))
|
|
flags |= ITF_BIG;
|
|
if (xml_bvalue(node, "animal", false))
|
|
flags |= ITF_ANIMAL;
|
|
if (xml_bvalue(node, "vehicle", false))
|
|
flags |= ITF_VEHICLE;
|
|
itype = rtype->itype ? rtype->itype : it_get_or_create(rtype);
|
|
itype->weight = xml_ivalue(node, "weight", 0);
|
|
itype->capacity = xml_ivalue(node, "capacity", 0);
|
|
itype->flags |= flags;
|
|
|
|
/* reading item/construction */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "construction", xpath);
|
|
xml_readconstruction(xpath, result->nodesetval, &itype->construction);
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading item/weapon */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "weapon", xpath);
|
|
assert(result->nodesetval->nodeNr <= 1);
|
|
if (result->nodesetval->nodeNr != 0) {
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
rtype->wtype = xml_readweapon(xpath, itype);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading item/potion */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "potion", xpath);
|
|
assert(result->nodesetval->nodeNr <= 1);
|
|
if (result->nodesetval->nodeNr != 0) {
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
rtype->ptype = xml_readpotion(xpath, itype);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading item/luxury */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "luxury", xpath);
|
|
assert(result->nodesetval->nodeNr <= 1);
|
|
if (result->nodesetval->nodeNr != 0) {
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
rtype->ltype = xml_readluxury(xpath, itype);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading item/armor */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "armor", xpath);
|
|
assert(result->nodesetval->nodeNr <= 1);
|
|
if (result->nodesetval->nodeNr != 0) {
|
|
xpath->node = result->nodesetval->nodeTab[0];
|
|
rtype->atype = xml_readarmor(xpath, itype);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* 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 *propValue;
|
|
pf_generic fun;
|
|
|
|
parse_function(node, &fun, &propValue);
|
|
if (fun == NULL) {
|
|
log_error("unknown function name '%s' for item '%s'\n", (const char *)propValue, rtype->_name);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "give") == 0) {
|
|
itype->give =
|
|
(int(*)(struct unit *, struct unit *, const struct item_type *, int,
|
|
struct order *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "use") == 0) {
|
|
itype->use =
|
|
(int(*)(struct unit *, const struct item_type *, int,
|
|
struct order *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "canuse") == 0) {
|
|
itype->canuse =
|
|
(bool(*)(const struct unit *, const struct item_type *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "useonother") == 0) {
|
|
itype->useonother =
|
|
(int(*)(struct unit *, int, const struct item_type *, int,
|
|
struct order *))fun;
|
|
}
|
|
else {
|
|
log_error("unknown function type '%s' for item '%s'\n", (const char *)propValue, rtype->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
itype->score = xml_ivalue(node, "score", 0);
|
|
if (!itype->score) itype->score = default_score(itype);
|
|
xmlXPathFreeObject(result);
|
|
|
|
return itype;
|
|
}
|
|
|
|
static int parse_rules(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr functions;
|
|
xmlNodeSetPtr nodes;
|
|
int i;
|
|
|
|
/* reading eressea/resources/resource */
|
|
functions = xmlXPathEvalExpression(BAD_CAST "/eressea/rules/function", xpath);
|
|
nodes = functions->nodesetval;
|
|
for (i = 0; i != nodes->nodeNr; ++i) {
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
xmlChar *propValue;
|
|
pf_generic fun;
|
|
|
|
parse_function(node, &fun, &propValue);
|
|
|
|
if (fun == NULL) {
|
|
log_error("unknown function for rule '%s' %s\n", (const char *)propValue);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "wage") == 0) {
|
|
global.functions.wage =
|
|
(int(*)(const struct region *, const struct faction *,
|
|
const struct race *, int))fun;
|
|
}
|
|
else {
|
|
log_error("unknown function for rule '%s'\n", (const char *)propValue);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
xmlXPathFreeObject(functions);
|
|
xmlXPathFreeContext(xpath);
|
|
return 0;
|
|
}
|
|
|
|
static int parse_resources(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr resources;
|
|
xmlNodeSetPtr nodes;
|
|
int i;
|
|
|
|
/* 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 *propValue, *name, *appearance;
|
|
resource_type *rtype;
|
|
unsigned int flags = RTF_NONE;
|
|
xmlXPathObjectPtr result;
|
|
int k;
|
|
|
|
if (xml_bvalue(node, "pooled", true))
|
|
flags |= RTF_POOLED;
|
|
if (xml_bvalue(node, "limited", false))
|
|
flags |= RTF_LIMITED;
|
|
|
|
name = xmlGetProp(node, BAD_CAST "name");
|
|
if (!name) {
|
|
assert(name);
|
|
log_error("invalid resource %d has no name", i);
|
|
continue;
|
|
}
|
|
rtype = rt_get_or_create((const char *)name);
|
|
rtype->flags |= flags;
|
|
xmlFree(name);
|
|
|
|
name = xmlGetProp(node, BAD_CAST "material");
|
|
if (name) {
|
|
rmt_create(rtype, (const char *)name);
|
|
xmlFree(name);
|
|
}
|
|
|
|
/* reading eressea/resources/resource/function */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
if (result->nodesetval != NULL)
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
pf_generic fun;
|
|
|
|
parse_function(node, &fun, &propValue);
|
|
if (fun == NULL) {
|
|
log_error("unknown function name '%s' for resource %s\n", (const char *)propValue, rtype->_name);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "change") == 0) {
|
|
rtype->uchange = (rtype_uchange)fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "get") == 0) {
|
|
rtype->uget = (rtype_uget)fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "name") == 0) {
|
|
rtype->name = (rtype_name)fun;
|
|
}
|
|
else {
|
|
log_error("unknown function type '%s' for resource %s\n", (const char *)propValue, rtype->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/resources/resource/resourcelimit */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "resourcelimit", xpath);
|
|
assert(result->nodesetval->nodeNr <= 1);
|
|
if (result->nodesetval->nodeNr != 0) {
|
|
resource_limit *rdata;
|
|
attrib *a = a_find(rtype->attribs, &at_resourcelimit);
|
|
xmlNodePtr limit = result->nodesetval->nodeTab[0];
|
|
|
|
if (a == NULL)
|
|
a = a_add(&rtype->attribs, a_new(&at_resourcelimit));
|
|
rdata = (resource_limit *)a->data.v;
|
|
rtype->flags |= RTF_LIMITED;
|
|
xpath->node = limit;
|
|
xmlXPathFreeObject(result);
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "modifier", xpath);
|
|
if (result->nodesetval != NULL) {
|
|
rdata->modifiers =
|
|
calloc(result->nodesetval->nodeNr + 1, sizeof(resource_mod));
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
building_type *btype = NULL;
|
|
const race *rc = NULL;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "race");
|
|
if (propValue != NULL) {
|
|
rc = rc_find((const char *)propValue);
|
|
if (rc == NULL)
|
|
rc = rc_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
rdata->modifiers[k].race = rc;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "building");
|
|
if (propValue != NULL) {
|
|
btype = bt_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
rdata->modifiers[k].btype = btype;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "type");
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "skill") == 0) {
|
|
rdata->modifiers[k].value.i = xml_ivalue(node, "value", 0);
|
|
rdata->modifiers[k].flags = RMF_SKILL;
|
|
}
|
|
else if (strcmp((const char *)propValue, "material") == 0) {
|
|
rdata->modifiers[k].value.f = (float)xml_fvalue(node, "value", 0);
|
|
rdata->modifiers[k].flags = RMF_SAVEMATERIAL;
|
|
}
|
|
else if (strcmp((const char *)propValue, "resource") == 0) {
|
|
rdata->modifiers[k].value.f = (float)xml_fvalue(node, "value", 0);
|
|
rdata->modifiers[k].flags = RMF_SAVERESOURCE;
|
|
}
|
|
else if (strcmp((const char *)propValue, "require") == 0) {
|
|
xmlChar *propBldg = xmlGetProp(node, BAD_CAST "building");
|
|
if (propBldg != NULL) {
|
|
btype = bt_get_or_create((const char *)propBldg);
|
|
rdata->modifiers[k].btype = btype;
|
|
rdata->modifiers[k].flags = RMF_REQUIREDBUILDING;
|
|
xmlFree(propBldg);
|
|
}
|
|
}
|
|
else {
|
|
log_error("unknown type '%s' for resourcelimit-modifier '%s'\n", (const char *)propValue, rtype->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
result = xmlXPathEvalExpression(BAD_CAST "guard", xpath);
|
|
if (result->nodesetval != NULL)
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
xmlChar *propFlag = xmlGetProp(node, BAD_CAST "flag");
|
|
|
|
if (propFlag != NULL) {
|
|
if (strcmp((const char *)propFlag, "logging") == 0) {
|
|
rdata->guard |= GUARD_TREES;
|
|
}
|
|
else if (strcmp((const char *)propFlag, "mining") == 0) {
|
|
rdata->guard |= GUARD_MINING;
|
|
}
|
|
xmlFree(propFlag);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/resources/resource/resourcelimit/function */
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
if (result->nodesetval != NULL)
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
pf_generic fun;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "value");
|
|
assert(propValue != NULL);
|
|
fun = get_function((const char *)propValue);
|
|
if (fun == NULL) {
|
|
log_error("unknown limit '%s' for resource %s\n", (const char *)propValue, rtype->_name);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
xmlFree(propValue);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "produce") == 0) {
|
|
rdata->produce = (rlimit_produce)fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "limit") == 0) {
|
|
rdata->limit = (rlimit_limit)fun;
|
|
}
|
|
else {
|
|
log_error("unknown limit '%s' for resource %s\n", (const char *)propValue, rtype->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/resources/resource/resourcelimit/function */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "resourcelimit/function", xpath);
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* 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);
|
|
appearance = xmlGetProp(node, BAD_CAST "appearance");
|
|
if (appearance) {
|
|
it_set_appearance(rtype->itype, (const char *)appearance);
|
|
xmlFree(appearance);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
}
|
|
xmlXPathFreeObject(resources);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
/* make sure old items (used in requirements) are available */
|
|
init_resources();
|
|
init_itemtypes();
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void add_items(equipment * eq, xmlNodeSetPtr nsetItems)
|
|
{
|
|
if (nsetItems != NULL && nsetItems->nodeNr > 0) {
|
|
int i;
|
|
for (i = 0; i != nsetItems->nodeNr; ++i) {
|
|
xmlNodePtr node = nsetItems->nodeTab[i];
|
|
xmlChar *propValue;
|
|
const struct item_type *itype;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
itype = it_find((const char *)propValue);
|
|
xmlFree(propValue);
|
|
if (itype != NULL) {
|
|
propValue = xmlGetProp(node, BAD_CAST "amount");
|
|
if (propValue != NULL) {
|
|
equipment_setitem(eq, itype, (const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void add_callbacks(equipment * eq, xmlNodeSetPtr nsetItems)
|
|
{
|
|
if (nsetItems != NULL && nsetItems->nodeNr > 0) {
|
|
int i;
|
|
for (i = 0; i != nsetItems->nodeNr; ++i) {
|
|
xmlNodePtr node = nsetItems->nodeTab[i];
|
|
xmlChar *propValue;
|
|
pf_generic fun;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
if (propValue != NULL) {
|
|
fun = get_function((const char *)propValue);
|
|
if (fun) {
|
|
equipment_setcallback(eq, (void(*)(const struct equipment *,
|
|
struct unit *))fun);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void add_spells(equipment * eq, xmlNodeSetPtr nsetItems)
|
|
{
|
|
if (nsetItems != NULL && nsetItems->nodeNr > 0) {
|
|
int i;
|
|
for (i = 0; i != nsetItems->nodeNr; ++i) {
|
|
xmlNodePtr node = nsetItems->nodeTab[i];
|
|
xmlChar *propValue;
|
|
struct spell *sp;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
sp = find_spell((const char *)propValue);
|
|
if (!sp) {
|
|
log_error("no spell '%s' for equipment-set '%s'\n", (const char *)propValue, eq->name);
|
|
}
|
|
else {
|
|
int level = xml_ivalue(node, "level", 0);
|
|
if (level > 0) {
|
|
equipment_addspell(eq, sp, level);
|
|
}
|
|
else {
|
|
log_error("spell '%s' for equipment-set '%s' has no level\n", sp->sname, eq->name);
|
|
}
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void add_skills(equipment * eq, xmlNodeSetPtr nsetSkills)
|
|
{
|
|
if (nsetSkills != NULL && nsetSkills->nodeNr > 0) {
|
|
int i;
|
|
for (i = 0; i != nsetSkills->nodeNr; ++i) {
|
|
xmlNodePtr node = nsetSkills->nodeTab[i];
|
|
xmlChar *propValue;
|
|
skill_t sk;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
sk = findskill((const char *)propValue);
|
|
if (sk == NOSKILL) {
|
|
log_error("unknown skill '%s' in equipment-set %s\n", (const char *)propValue, eq->name);
|
|
xmlFree(propValue);
|
|
}
|
|
else {
|
|
xmlFree(propValue);
|
|
propValue = xmlGetProp(node, BAD_CAST "level");
|
|
if (propValue != NULL) {
|
|
equipment_setskill(eq, sk, (const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
add_subsets(xmlDocPtr doc, equipment * eq, xmlNodeSetPtr nsetSubsets)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
if (nsetSubsets != NULL && nsetSubsets->nodeNr > 0) {
|
|
int i;
|
|
|
|
eq->subsets = calloc(nsetSubsets->nodeNr + 1, sizeof(subset));
|
|
for (i = 0; i != nsetSubsets->nodeNr; ++i) {
|
|
xmlXPathObjectPtr xpathResult;
|
|
xmlNodePtr node = nsetSubsets->nodeTab[i];
|
|
xmlChar *propValue;
|
|
|
|
eq->subsets[i].chance = 1.0f;
|
|
propValue = xmlGetProp(node, BAD_CAST "chance");
|
|
if (propValue != NULL) {
|
|
eq->subsets[i].chance = (float)atof((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
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;
|
|
|
|
propValue = xmlGetProp(nodeSet, BAD_CAST "chance");
|
|
if (propValue != NULL) {
|
|
chance = (float)atof((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
totalChance += chance;
|
|
|
|
propValue = xmlGetProp(nodeSet, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
eq->subsets[i].sets[set].chance = chance;
|
|
eq->subsets[i].sets[set].set =
|
|
create_equipment((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
if (totalChance > 1.0f) {
|
|
log_error("total chance exceeds 1.0: %f in equipment set %s.\n", totalChance, eq->name);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(xpathResult);
|
|
}
|
|
}
|
|
xmlXPathFreeContext(xpath);
|
|
}
|
|
|
|
static int parse_equipment(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr xpathRaces;
|
|
|
|
/* reading eressea/equipment/set */
|
|
xpathRaces = xmlXPathEvalExpression(BAD_CAST "/eressea/equipment/set", xpath);
|
|
if (xpathRaces->nodesetval) {
|
|
xmlNodeSetPtr nsetRaces = xpathRaces->nodesetval;
|
|
int i;
|
|
|
|
for (i = 0; i != nsetRaces->nodeNr; ++i) {
|
|
xmlNodePtr node = nsetRaces->nodeTab[i];
|
|
xmlChar *propName = xmlGetProp(node, BAD_CAST "name");
|
|
|
|
if (propName != NULL) {
|
|
equipment *eq = create_equipment((const char *)propName);
|
|
xmlXPathObjectPtr xpathResult;
|
|
|
|
xpath->node = node;
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "callback", xpath);
|
|
assert(!eq->callback);
|
|
add_callbacks(eq, xpathResult->nodesetval);
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "item", xpath);
|
|
assert(!eq->items);
|
|
add_items(eq, xpathResult->nodesetval);
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "spell", xpath);
|
|
assert(!eq->spellbook);
|
|
add_spells(eq, xpathResult->nodesetval);
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "skill", xpath);
|
|
add_skills(eq, xpathResult->nodesetval);
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
xpathResult = xmlXPathEvalExpression(BAD_CAST "subset", xpath);
|
|
assert(!eq->subsets);
|
|
add_subsets(doc, eq, xpathResult->nodesetval);
|
|
xmlXPathFreeObject(xpathResult);
|
|
|
|
xmlFree(propName);
|
|
}
|
|
}
|
|
}
|
|
|
|
xmlXPathFreeObject(xpathRaces);
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int parse_spellbooks(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr spellbooks;
|
|
|
|
/* reading eressea/spells/spell */
|
|
spellbooks = xmlXPathEvalExpression(BAD_CAST "/eressea/spells/spellbook", xpath);
|
|
|
|
if (spellbooks->nodesetval != NULL) {
|
|
xmlNodeSetPtr nodes = spellbooks->nodesetval;
|
|
int i, k;
|
|
|
|
for (i = 0; i != nodes->nodeNr; ++i) {
|
|
xmlXPathObjectPtr result;
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
xmlChar *propValue;
|
|
spellbook * sb;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
if (propValue) {
|
|
sb = get_spellbook((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
else {
|
|
log_error("spellbook at index '%d' has n name\n", i);
|
|
continue;
|
|
}
|
|
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "entry", xpath);
|
|
|
|
if (result->nodesetval->nodeNr > 0) {
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
spell * sp = 0;
|
|
int level = xml_ivalue(node, "level", -1);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "spell");
|
|
if (propValue) {
|
|
sp = find_spell((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
if (sp && level > 0) {
|
|
spellbook_add(sb, sp, level);
|
|
}
|
|
else {
|
|
log_error("invalid entry at index '%d' in spellbook '%s'\n", k, sb->name);
|
|
}
|
|
}
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
}
|
|
}
|
|
xmlXPathFreeObject(spellbooks);
|
|
xmlXPathFreeContext(xpath);
|
|
return 0;
|
|
}
|
|
|
|
static int parse_spells(xmlDocPtr doc)
|
|
{
|
|
pf_generic cast = 0;
|
|
pf_generic fumble = 0;
|
|
xmlXPathContextPtr xpath = xmlXPathNewContext(doc);
|
|
xmlXPathObjectPtr spells;
|
|
char zText[32];
|
|
strcpy(zText, "fumble_");
|
|
|
|
/* reading eressea/spells/spell */
|
|
spells = xmlXPathEvalExpression(BAD_CAST "/eressea/spells/spell", xpath);
|
|
if (spells->nodesetval != NULL) {
|
|
xmlNodeSetPtr nodes = spells->nodesetval;
|
|
int i;
|
|
|
|
for (i = 0; i != nodes->nodeNr; ++i) {
|
|
xmlXPathObjectPtr result;
|
|
xmlNodePtr node = nodes->nodeTab[i];
|
|
xmlChar *propValue;
|
|
int k;
|
|
spell_component *component;
|
|
spell *sp;
|
|
unsigned int index;
|
|
static int modes[] = { 0, PRECOMBATSPELL, COMBATSPELL, POSTCOMBATSPELL };
|
|
|
|
/* spellname */
|
|
index = xml_ivalue(node, "index", 0);
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
sp = create_spell((const char *)propValue, index);
|
|
xmlFree(propValue);
|
|
if (!sp) {
|
|
continue;
|
|
}
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "parameters");
|
|
if (propValue) {
|
|
sp->parameter = _strdup((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "syntax");
|
|
if (propValue) {
|
|
sp->syntax = _strdup((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
#ifdef TODO /* no longer need it, spellbooks! */
|
|
/* magic type */
|
|
propValue = xmlGetProp(node, BAD_CAST "type");
|
|
assert(propValue != NULL);
|
|
for (sp->magietyp = 0; sp->magietyp != MAXMAGIETYP; ++sp->magietyp) {
|
|
if (strcmp(magic_school[sp->magietyp], (const char *)propValue) == 0)
|
|
break;
|
|
}
|
|
assert(sp->magietyp != MAXMAGIETYP);
|
|
xmlFree(propValue);
|
|
/* level, rank and flags */
|
|
#endif
|
|
sp->rank = (char)xml_ivalue(node, "rank", -1);
|
|
if (xml_bvalue(node, "los", false))
|
|
sp->sptyp |= TESTCANSEE; /* must see or have contact */
|
|
if (!xml_bvalue(node, "target_global", false))
|
|
sp->sptyp |= SEARCHLOCAL; /* must be in same region */
|
|
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;
|
|
if (xml_bvalue(node, "variable", false))
|
|
sp->sptyp |= SPELLLEVEL;
|
|
|
|
if (xml_bvalue(node, "buildingtarget", false))
|
|
sp->sptyp |= BUILDINGSPELL;
|
|
if (xml_bvalue(node, "shiptarget", false))
|
|
sp->sptyp |= SHIPSPELL;
|
|
if (xml_bvalue(node, "unittarget", false))
|
|
sp->sptyp |= UNITSPELL;
|
|
if (xml_bvalue(node, "regiontarget", false))
|
|
sp->sptyp |= REGIONSPELL;
|
|
|
|
k = xml_ivalue(node, "combat", 0);
|
|
if (k >= 0 && k <= 3)
|
|
sp->sptyp |= modes[k];
|
|
|
|
/* reading eressea/spells/spell/function */
|
|
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "function", xpath);
|
|
|
|
if (result->nodesetval->nodeNr == 0) {
|
|
cast = get_function(sp->sname);
|
|
if (!cast) {
|
|
log_error("no spell cast function registered for '%s'\n", sp->sname);
|
|
}
|
|
strlcpy(zText + 7, sp->sname, sizeof(zText) - 7);
|
|
fumble = get_function(zText);
|
|
}
|
|
else {
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
pf_generic fun;
|
|
|
|
parse_function(node, &fun, &propValue);
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "cast") == 0) {
|
|
if (fun) {
|
|
cast = fun;
|
|
}
|
|
else {
|
|
log_error("unknown function name '%s' for spell '%s'\n", (const char *)propValue, sp->sname);
|
|
}
|
|
}
|
|
else if (fun && strcmp((const char *)propValue, "fumble") == 0) {
|
|
fumble = fun;
|
|
}
|
|
else {
|
|
log_error("unknown function type '%s' for spell '%s'\n", (const char *)propValue, sp->sname);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
}
|
|
sp->cast = (spell_f)cast;
|
|
sp->fumble = (fumble_f)fumble;
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/spells/spell/resource */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "resource", xpath);
|
|
if (result->nodesetval->nodeNr) {
|
|
sp->components =
|
|
(spell_component *)malloc(sizeof(spell_component) *
|
|
(result->nodesetval->nodeNr + 1));
|
|
sp->components[result->nodesetval->nodeNr].type = 0;
|
|
}
|
|
for (component = sp->components, k = 0; k != result->nodesetval->nodeNr;
|
|
++k) {
|
|
const resource_type *rtype;
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue);
|
|
rtype = rt_find((const char *)propValue);
|
|
if (!rtype) {
|
|
log_error("spell %s uses unknown component %s.\n", sp->sname, (const char *)propValue);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
component->type = rtype;
|
|
xmlFree(propValue);
|
|
component->amount = xml_ivalue(node, "amount", 1);
|
|
component->cost = SPC_FIX;
|
|
propValue = xmlGetProp(node, BAD_CAST "cost");
|
|
if (propValue != NULL) {
|
|
if (strcmp((const char *)propValue, "linear") == 0) {
|
|
if ((sp->sptyp&SPELLLEVEL) == 0) {
|
|
log_error("spell '%s' has linear cost but fixed level\n", sp->sname);
|
|
}
|
|
component->cost = SPC_LINEAR;
|
|
}
|
|
else if (strcmp((const char *)propValue, "level") == 0) {
|
|
if ((sp->sptyp&SPELLLEVEL) == 0) {
|
|
log_error("spell '%s' has levelled cost but fixed level\n", sp->sname);
|
|
}
|
|
component->cost = SPC_LEVEL;
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
component++;
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
}
|
|
}
|
|
|
|
xmlXPathFreeObject(spells);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void parse_param(struct param **params, xmlNodePtr node)
|
|
{
|
|
xmlChar *propName = xmlGetProp(node, BAD_CAST "name");
|
|
xmlChar *propValue = xmlGetProp(node, BAD_CAST "value");
|
|
|
|
set_param(params, (const char *)propName, (const char *)propValue);
|
|
|
|
xmlFree(propName);
|
|
xmlFree(propValue);
|
|
}
|
|
|
|
static void parse_ai(race * rc, xmlNodePtr node)
|
|
{
|
|
rc->splitsize = xml_ivalue(node, "splitsize", 0);
|
|
rc->aggression = (float)xml_fvalue(node, "aggression", 0.04);
|
|
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;
|
|
if (xml_bvalue(node, "moveattack", false))
|
|
rc->flags |= RCF_ATTACK_MOVED;
|
|
}
|
|
|
|
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];
|
|
xmlNodePtr child;
|
|
xmlChar *propValue;
|
|
race *rc, *frc = 0;
|
|
xmlXPathObjectPtr result;
|
|
int k, study_speed_base, attacks;
|
|
struct att *attack;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
rc = rc_get_or_create((const char *)propValue);
|
|
xmlFree(propValue);
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "damage");
|
|
assert(propValue != NULL);
|
|
rc->def_damage = _strdup((const char *)propValue);
|
|
xmlFree(propValue);
|
|
|
|
rc->magres = (float)xml_fvalue(node, "magres", rc->magres);
|
|
rc->maxaura = (float)xml_fvalue(node, "maxaura", rc->maxaura);
|
|
rc->regaura = (float)xml_fvalue(node, "regaura", rc->regaura);
|
|
rc->recruitcost = xml_ivalue(node, "recruitcost", rc->recruitcost);
|
|
rc->maintenance = xml_ivalue(node, "maintenance", rc->maintenance);
|
|
rc->weight = xml_ivalue(node, "weight", rc->weight);
|
|
rc->capacity = xml_ivalue(node, "capacity", rc->capacity);
|
|
rc->speed = (float)xml_fvalue(node, "speed", rc->speed);
|
|
rc->hitpoints = xml_ivalue(node, "hp", rc->hitpoints);
|
|
rc->armor = (char)xml_ivalue(node, "ac", rc->armor);
|
|
study_speed_base = xml_ivalue(node, "studyspeed", 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", rc->at_bonus);
|
|
rc->df_bonus = (char)xml_ivalue(node, "defensemodifier", rc->df_bonus);
|
|
|
|
if (!xml_bvalue(node, "playerrace", false))
|
|
rc->flags |= RCF_NPC;
|
|
if (xml_bvalue(node, "scarepeasants", false))
|
|
rc->flags |= RCF_SCAREPEASANTS;
|
|
if (!xml_bvalue(node, "cansteal", true))
|
|
rc->flags |= RCF_NOSTEAL;
|
|
if (xml_bvalue(node, "cansail", true))
|
|
rc->flags |= RCF_CANSAIL;
|
|
if (xml_bvalue(node, "cannotmove", false))
|
|
rc->flags |= RCF_CANNOTMOVE;
|
|
if (xml_bvalue(node, "fly", false))
|
|
rc->flags |= RCF_FLY;
|
|
if (xml_bvalue(node, "invisible", false))
|
|
rc->flags |= RCF_INVISIBLE;
|
|
if (xml_bvalue(node, "coastal", false))
|
|
rc->flags |= RCF_COASTAL;
|
|
if (xml_bvalue(node, "unarmedguard", false))
|
|
rc->flags |= RCF_UNARMEDGUARD;
|
|
if (xml_bvalue(node, "swim", false))
|
|
rc->flags |= RCF_SWIM;
|
|
if (xml_bvalue(node, "walk", false))
|
|
rc->flags |= RCF_WALK;
|
|
if (!xml_bvalue(node, "canlearn", true))
|
|
rc->flags |= RCF_NOLEARN;
|
|
if (!xml_bvalue(node, "canteach", true))
|
|
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, "shipspeed", false))
|
|
rc->flags |= RCF_SHIPSPEED;
|
|
if (xml_bvalue(node, "stonegolem", false))
|
|
rc->flags |= RCF_STONEGOLEM;
|
|
if (xml_bvalue(node, "irongolem", false))
|
|
rc->flags |= RCF_IRONGOLEM;
|
|
|
|
if (xml_bvalue(node, "keepitem", false))
|
|
rc->ec_flags |= ECF_KEEP_ITEM;
|
|
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, "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; // TODO: invert this flag, so rc_get_or_create gets simpler
|
|
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;
|
|
if (xml_bvalue(node, "noattack", false))
|
|
rc->battle_flags |= BF_NO_ATTACK;
|
|
|
|
for (child = node->children; child; child = child->next) {
|
|
if (strcmp((const char *)child->name, "ai") == 0) {
|
|
parse_ai(rc, child);
|
|
}
|
|
else if (strcmp((const char *)child->name, "param") == 0) {
|
|
parse_param(&rc->parameters, child);
|
|
}
|
|
}
|
|
rc->recruit_multi = get_param_flt(rc->parameters, "recruit_multi", 1.0);
|
|
|
|
/* reading eressea/races/race/skill */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "skill", xpath);
|
|
memset(rc->bonus, 0, sizeof(rc->bonus));
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
int mod = xml_ivalue(node, "modifier", 0);
|
|
int speed = xml_ivalue(node, "speed", study_speed_base);
|
|
skill_t sk;
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
assert(propValue != NULL);
|
|
sk = findskill((const char *)propValue);
|
|
if (sk != NOSKILL) {
|
|
rc->bonus[sk] = (char)mod;
|
|
if (speed) {
|
|
if (!rc->study_speed)
|
|
rc->study_speed = calloc(1, MAXSKILLS);
|
|
rc->study_speed[sk] = (char)speed;
|
|
}
|
|
}
|
|
else {
|
|
log_error("unknown skill '%s' in race '%s'\n", (const char *)propValue, rc->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
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;
|
|
|
|
parse_function(node, &fun, &propValue);
|
|
if (fun == NULL) {
|
|
log_error("unknown function name '%s' for race %s\n", (const char *)propValue, rc->_name);
|
|
xmlFree(propValue);
|
|
continue;
|
|
}
|
|
assert(propValue != NULL);
|
|
if (strcmp((const char *)propValue, "name") == 0) {
|
|
rc->generate_name = (const char *(*)(const struct unit *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "describe") == 0) {
|
|
rc->describe =
|
|
(const char *(*)(const struct unit *, const struct locale *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "age") == 0) {
|
|
rc->age = (void(*)(struct unit *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "move") == 0) {
|
|
rc->move_allowed =
|
|
(bool(*)(const struct region *, const struct region *))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "itemdrop") == 0) {
|
|
rc->itemdrop = (struct item * (*)(const struct race *, int))fun;
|
|
}
|
|
else if (strcmp((const char *)propValue, "initfamiliar") == 0) {
|
|
rc->init_familiar = (void(*)(struct unit *))fun;
|
|
}
|
|
else {
|
|
log_error("unknown function type '%s' for race %s\n", (const char *)propValue, rc->_name);
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/races/race/familiar */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "familiar", xpath);
|
|
if (result->nodesetval->nodeNr > MAXMAGIETYP) {
|
|
log_error("race %s has %d potential familiars", rc->_name, result->nodesetval->nodeNr);
|
|
}
|
|
else {
|
|
for (k = 0; k != MAXMAGIETYP; ++k) {
|
|
if (k < result->nodesetval->nodeNr) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "race");
|
|
assert(propValue != NULL);
|
|
frc = rc_get_or_create((const char *)propValue);
|
|
if (xml_bvalue(node, "default", false)) {
|
|
rc->familiars[k] = rc->familiars[0];
|
|
rc->familiars[0] = frc;
|
|
}
|
|
else {
|
|
rc->familiars[k] = frc;
|
|
}
|
|
xmlFree(propValue);
|
|
}
|
|
else {
|
|
rc->familiars[k] = frc;
|
|
}
|
|
}
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/races/race/precombatspell */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "precombatspell", xpath);
|
|
assert(rc->precombatspell == NULL
|
|
|| !"precombatspell is already initialized");
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
rc->precombatspell = xml_spell(node, "spell");
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* reading eressea/races/race/attack */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "attack", xpath);
|
|
attack = rc->attack;
|
|
attacks = 0;
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
while (attack->type != AT_NONE) {
|
|
++attack;
|
|
if (attacks++ >= RACE_ATTACKS) {
|
|
log_error("too many attacks for race '%s'\n", rc->_name);
|
|
assert(!"aborting");
|
|
}
|
|
}
|
|
|
|
propValue = xmlGetProp(node, BAD_CAST "damage");
|
|
if (propValue != NULL) {
|
|
attack->data.dice = _strdup((const char *)propValue);
|
|
xmlFree(propValue);
|
|
}
|
|
else {
|
|
attack->data.sp = xml_spell(node, "spell");
|
|
if (attack->data.sp) {
|
|
attack->level = xml_ivalue(node, "level", 0);
|
|
if (attack->level <= 0) {
|
|
log_error("magical attack '%s' for race '%s' needs a level: %d\n", attack->data.sp->sname, rc->_name, attack->level);
|
|
}
|
|
}
|
|
}
|
|
attack->type = xml_ivalue(node, "type", 0);
|
|
attack->flags = xml_ivalue(node, "flags", 0);
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
}
|
|
xmlXPathFreeObject(races);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int parse_messages(xmlDocPtr doc)
|
|
{
|
|
xmlXPathContextPtr xpath;
|
|
xmlXPathObjectPtr messages;
|
|
xmlNodeSetPtr nodes;
|
|
int i;
|
|
|
|
xpath = xmlXPathNewContext(doc);
|
|
|
|
/* reading eressea/messages/message */
|
|
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 *propSection;
|
|
xmlChar *propValue;
|
|
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 && 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 *propName, *propType;
|
|
|
|
propName = xmlGetProp(node, BAD_CAST "name");
|
|
propType = xmlGetProp(node, BAD_CAST "type");
|
|
sprintf(zBuffer, "%s:%s", (const char *)propName,
|
|
(const char *)propType);
|
|
xmlFree(propName);
|
|
xmlFree(propType);
|
|
argv[k] = _strdup(zBuffer);
|
|
}
|
|
argv[result->nodesetval->nodeNr] = NULL;
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
/* add the messagetype */
|
|
propValue = xmlGetProp(node, BAD_CAST "name");
|
|
mtype = mt_find((const char *)propValue);
|
|
if (mtype == NULL) {
|
|
mtype = mt_register(mt_new((const char *)propValue, (const char **)argv));
|
|
}
|
|
else {
|
|
assert(argv != NULL || !"cannot redefine arguments of message now");
|
|
}
|
|
xmlFree(propValue);
|
|
|
|
/* 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);
|
|
}
|
|
|
|
propSection = xmlGetProp(node, BAD_CAST "section");
|
|
if (propSection == NULL)
|
|
propSection = BAD_CAST default_section;
|
|
|
|
/* strings */
|
|
xpath->node = node;
|
|
result = xmlXPathEvalExpression(BAD_CAST "text", xpath);
|
|
assert(result->nodesetval->nodeNr>0);
|
|
for (k = 0; k != result->nodesetval->nodeNr; ++k) {
|
|
xmlNodePtr node = result->nodesetval->nodeTab[k];
|
|
struct locale *lang;
|
|
xmlChar *propText;
|
|
|
|
xml_readtext(node, &lang, &propText);
|
|
if (lang) {
|
|
xml_cleanup_string(propText);
|
|
nrt_register(mtype, lang, (const char *)propText, 0,
|
|
(const char *)propSection);
|
|
}
|
|
xmlFree(propText);
|
|
|
|
}
|
|
xmlXPathFreeObject(result);
|
|
|
|
if (propSection != BAD_CAST default_section)
|
|
xmlFree(propSection);
|
|
}
|
|
|
|
xmlXPathFreeObject(messages);
|
|
|
|
xmlXPathFreeContext(xpath);
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
xml_readstrings(xmlXPathContextPtr xpath, xmlNodePtr * nodeTab, int nodeNr,
|
|
bool names)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i != nodeNr; ++i) {
|
|
xmlNodePtr stringNode = nodeTab[i];
|
|
xmlChar *propName = xmlGetProp(stringNode, BAD_CAST "name");
|
|
xmlChar *propNamespace = NULL;
|
|
xmlXPathObjectPtr result;
|
|
int k;
|
|
char zName[128];
|
|
|
|
assert(propName != NULL);
|
|
if (names)
|
|
propNamespace = xmlGetProp(stringNode->parent, BAD_CAST "name");
|
|
mkname_buf((const char *)propNamespace, (const char *)propName, zName);
|
|
if (propNamespace != NULL)
|
|
xmlFree(propNamespace);
|
|
xmlFree(propName);
|
|
|
|
/* 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 *propText;
|
|
|
|
xml_readtext(textNode, &lang, &propText);
|
|
if (propText != NULL) {
|
|
assert(strcmp(zName,
|
|
(const char *)xml_cleanup_string(BAD_CAST zName)) == 0);
|
|
if (lang) {
|
|
xml_cleanup_string(propText);
|
|
locale_setstring(lang, zName, (const char *)propText);
|
|
}
|
|
xmlFree(propText);
|
|
}
|
|
else {
|
|
log_warning("string %s has no text in locale %s\n", zName, locale_name(lang));
|
|
}
|
|
}
|
|
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;
|
|
}
|
|
|
|
void register_xmlreader(void)
|
|
{
|
|
xml_register_callback(parse_strings);
|
|
xml_register_callback(parse_messages);
|
|
xml_register_callback(parse_resources);
|
|
xml_register_callback(parse_rules);
|
|
|
|
xml_register_callback(parse_buildings); /* requires resources */
|
|
xml_register_callback(parse_ships); /* requires terrains */
|
|
xml_register_callback(parse_spells); /* requires resources */
|
|
xml_register_callback(parse_spellbooks); /* requires spells */
|
|
xml_register_callback(parse_equipment); /* requires spells */
|
|
xml_register_callback(parse_races); /* requires spells */
|
|
xml_register_callback(parse_calendar);
|
|
}
|
|
#endif
|