2005-11-26 12:49:20 +01:00
|
|
|
/* vi: set ts=2:
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
|
|
|
* Enno Rehling (enno@eressea-pbem.de)
|
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
*
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <eressea.h>
|
|
|
|
#include "object.h"
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include <kernel/unit.h>
|
|
|
|
#include <kernel/faction.h>
|
|
|
|
#include <kernel/region.h>
|
|
|
|
#include <kernel/ship.h>
|
|
|
|
#include <kernel/building.h>
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
|
|
|
/* stdc includes */
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
typedef struct object_data {
|
|
|
|
object_type type;
|
|
|
|
char * name;
|
|
|
|
union {
|
|
|
|
int i;
|
|
|
|
char * str;
|
|
|
|
double real;
|
|
|
|
struct unit * u;
|
|
|
|
struct region * r;
|
|
|
|
struct building * b;
|
|
|
|
struct ship * sh;
|
|
|
|
struct faction * f;
|
|
|
|
} data;
|
|
|
|
} object_data;
|
|
|
|
|
|
|
|
static void
|
|
|
|
object_write(const attrib *a, FILE *F)
|
|
|
|
{
|
|
|
|
const object_data * data = (object_data *)a->data.v;
|
|
|
|
int type = (int)data->type;
|
2005-11-26 16:28:11 +01:00
|
|
|
fprintf(F, "%s %d ", data->name, type);
|
2005-11-26 12:49:20 +01:00
|
|
|
switch (data->type) {
|
2005-11-26 16:28:11 +01:00
|
|
|
case TINTEGER:
|
|
|
|
fprintf(F, "%d ", data->data.i);
|
|
|
|
break;
|
|
|
|
case TREAL:
|
|
|
|
fprintf(F, "%lf ", data->data.real);
|
|
|
|
break;
|
|
|
|
case TSTRING:
|
|
|
|
fwritestr(F, data->data.str);
|
|
|
|
break;
|
2005-11-26 12:49:20 +01:00
|
|
|
case TUNIT:
|
|
|
|
write_unit_reference(data->data.u, F);
|
|
|
|
break;
|
|
|
|
case TFACTION:
|
|
|
|
write_faction_reference(data->data.f, F);
|
|
|
|
break;
|
|
|
|
case TBUILDING:
|
|
|
|
write_building_reference(data->data.b, F);
|
|
|
|
break;
|
|
|
|
case TSHIP:
|
|
|
|
/* write_ship_reference(data->data.sh, F); */
|
|
|
|
assert(!"not implemented");
|
|
|
|
break;
|
|
|
|
case TREGION:
|
|
|
|
write_region_reference(data->data.r, F);
|
|
|
|
break;
|
|
|
|
case TNONE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(!"illegal type in object-attribute");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
object_read(attrib *a, FILE *F)
|
|
|
|
{
|
|
|
|
object_data * data = (object_data *)a->data.v;
|
|
|
|
int type;
|
2005-11-26 16:28:11 +01:00
|
|
|
char buffer[128];
|
|
|
|
fscanf(F, "%s %d", buffer, &type);
|
|
|
|
data->name = strdup(buffer);
|
2005-11-26 12:49:20 +01:00
|
|
|
data->type = (object_type)type;
|
|
|
|
switch (data->type) {
|
2005-11-26 16:28:11 +01:00
|
|
|
case TINTEGER:
|
|
|
|
fscanf(F, "%d", &data->data.i);
|
|
|
|
break;
|
|
|
|
case TREAL:
|
|
|
|
fscanf(F, "%lf", &data->data.real);
|
|
|
|
break;
|
|
|
|
case TSTRING:
|
|
|
|
freadstr(F, buffer, sizeof(buffer));
|
|
|
|
data->data.str = strdup(buffer);
|
|
|
|
break;
|
|
|
|
case TBUILDING:
|
|
|
|
return read_building_reference(&data->data.b, F);
|
2005-11-26 12:49:20 +01:00
|
|
|
case TUNIT:
|
|
|
|
return read_unit_reference(&data->data.u, F);
|
|
|
|
case TFACTION:
|
|
|
|
return read_faction_reference(&data->data.f, F);
|
2005-11-26 16:28:11 +01:00
|
|
|
case TREGION:
|
|
|
|
return read_region_reference(&data->data.r, F);
|
2005-11-26 12:49:20 +01:00
|
|
|
case TSHIP:
|
|
|
|
/* return read_ship_reference(&data->data.sh, F); */
|
|
|
|
assert(!"not implemented");
|
|
|
|
break;
|
|
|
|
case TNONE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return AT_READ_FAIL;
|
|
|
|
}
|
|
|
|
return AT_READ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
object_init(attrib * a)
|
|
|
|
{
|
|
|
|
object_data * data;
|
|
|
|
a->data.v = malloc(sizeof(object_data));
|
|
|
|
data = (object_data *)a->data.v;
|
|
|
|
data->type = TNONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
object_done(attrib * a)
|
|
|
|
{
|
|
|
|
object_data * data = (object_data *)a->data.v;
|
|
|
|
if (data->type == TSTRING) free(data->data.str);
|
2005-11-26 16:28:11 +01:00
|
|
|
free(data->name);
|
2005-11-26 12:49:20 +01:00
|
|
|
free(a->data.v);
|
|
|
|
}
|
|
|
|
|
|
|
|
attrib_type at_object = {
|
|
|
|
"object", object_init, object_done, NULL,
|
|
|
|
object_write, object_read
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *
|
|
|
|
object_name(const attrib * a)
|
|
|
|
{
|
|
|
|
object_data * data = (object_data *)a->data.v;
|
|
|
|
return data->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct attrib *
|
|
|
|
object_create(const char * name, object_type type, variant value)
|
|
|
|
{
|
|
|
|
attrib * a = a_new(&at_object);
|
|
|
|
object_data * data = (object_data *)a->data.v;
|
2005-11-26 16:28:11 +01:00
|
|
|
data->name = strdup(name);
|
|
|
|
|
|
|
|
object_set(a, type, value);
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
object_set(attrib * a, object_type type, variant value)
|
|
|
|
{
|
|
|
|
object_data * data = (object_data *)a->data.v;
|
2005-11-26 12:49:20 +01:00
|
|
|
|
2005-11-26 16:28:11 +01:00
|
|
|
if (data->type==TSTRING) free(data->data.str);
|
2005-11-26 12:49:20 +01:00
|
|
|
data->type = type;
|
|
|
|
switch (type) {
|
|
|
|
case TSTRING:
|
|
|
|
data->data.str = strdup(value.v);
|
|
|
|
break;
|
|
|
|
case TINTEGER:
|
|
|
|
data->data.i = value.i;
|
|
|
|
break;
|
|
|
|
case TREAL:
|
|
|
|
data->data.real = value.f;
|
|
|
|
break;
|
|
|
|
case TREGION:
|
|
|
|
data->data.r = (region*)value.v;
|
|
|
|
break;
|
|
|
|
case TBUILDING:
|
|
|
|
data->data.b = (building*)value.v;
|
|
|
|
break;
|
|
|
|
case TFACTION:
|
|
|
|
data->data.f = (faction*)value.v;
|
|
|
|
break;
|
|
|
|
case TUNIT:
|
|
|
|
data->data.u = (unit*)value.v;
|
|
|
|
break;
|
|
|
|
case TSHIP:
|
|
|
|
data->data.sh = (ship*)value.v;
|
|
|
|
break;
|
|
|
|
case TNONE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(!"invalid object-type");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-11-26 16:28:11 +01:00
|
|
|
void
|
|
|
|
object_get(const struct attrib * a, object_type * type, variant * value)
|
|
|
|
{
|
|
|
|
object_data * data = (object_data *)a->data.v;
|
|
|
|
*type = data->type;
|
|
|
|
switch (data->type) {
|
|
|
|
case TSTRING:
|
|
|
|
value->v = data->data.str;
|
|
|
|
break;
|
|
|
|
case TINTEGER:
|
|
|
|
value->i = data->data.i;
|
|
|
|
break;
|
|
|
|
case TREAL:
|
|
|
|
value->f = (float)data->data.real;
|
|
|
|
break;
|
|
|
|
case TREGION:
|
|
|
|
value->v = data->data.r;
|
|
|
|
break;
|
|
|
|
case TBUILDING:
|
|
|
|
value->v = data->data.b;
|
|
|
|
break;
|
|
|
|
case TFACTION:
|
|
|
|
value->v = data->data.f;
|
|
|
|
break;
|
|
|
|
case TUNIT:
|
|
|
|
value->v = data->data.u;
|
|
|
|
break;
|
|
|
|
case TSHIP:
|
|
|
|
value->v = data->data.sh;
|
|
|
|
break;
|
|
|
|
case TNONE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(!"invalid object-type");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|