/* 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.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 #include /* util includes */ #include #include #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, struct storage * store) { const object_data * data = (object_data *)a->data.v; int type = (int)data->type; store->w_tok(store, data->name); store->w_int(store, type); switch (data->type) { case TINTEGER: store->w_int(store, data->data.i); break; case TREAL: store->w_flt(store, (float)data->data.real); break; case TSTRING: store->w_str(store, data->data.str); break; case TUNIT: write_unit_reference(data->data.u, store ); break; case TFACTION: write_faction_reference(data->data.f, store); break; case TBUILDING: write_building_reference(data->data.b, store); break; case TSHIP: /* write_ship_reference(data->data.sh, store); */ assert(!"not implemented"); break; case TREGION: write_region_reference(data->data.r, store); break; case TNONE: break; default: assert(!"illegal type in object-attribute"); } } static int object_read(attrib *a, struct storage * store) { object_data * data = (object_data *)a->data.v; int result; data->name = store->r_str(store); data->type = (object_type)store->r_int(store); switch (data->type) { case TINTEGER: data->data.i = store->r_int(store); break; case TREAL: data->data.real = store->r_flt(store); break; case TSTRING: data->data.str = store->r_str(store); break; case TBUILDING: result = read_reference(&data->data.b, store, read_building_reference, resolve_building); if (result==0 && !data->data.b) { return AT_READ_FAIL; } break; case TUNIT: result = read_reference(&data->data.u, store, read_unit_reference, resolve_unit); if (result==0 && !data->data.u) { return AT_READ_FAIL; } break; case TFACTION: result = read_reference(&data->data.f, store, read_faction_reference, resolve_faction); if (result==0 && !data->data.f) { return AT_READ_FAIL; } break; case TREGION: result = read_reference(&data->data.r, store, read_region_reference, RESOLVE_REGION(store->version)); if (result==0 && !data->data.r) { return AT_READ_FAIL; } break; case TSHIP: /* return read_ship_reference(&data->data.sh, store); */ 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; } }