/* 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 #include #include "object.h" /* kernel includes */ #include #include #include #include #include #include /* util includes */ #include /* stdc includes */ #include 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; fprintf(F, "%s %d ", data->name, type); switch (data->type) { 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; 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; char buffer[128]; fscanf(F, "%s %d", buffer, &type); data->name = strdup(buffer); data->type = (object_type)type; switch (data->type) { case TINTEGER: fscanf(F, "%d", &data->data.i); break; case TREAL: fscanf(F, "%lf", &data->data.real); break; case TSTRING: freadstr(F, enc_gamedata, buffer, sizeof(buffer)); data->data.str = strdup(buffer); break; case TBUILDING: return read_building_reference(&data->data.b, F); case TUNIT: return read_unit_reference(&data->data.u, F); case TFACTION: return read_faction_reference(&data->data.f, F); case TREGION: return read_region_reference(&data->data.r, F); 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); free(data->name); 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; 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; if (data->type==TSTRING) free(data->data.str); 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; } } 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; } }