/* vi: set ts=2: * * $Id: building.c,v 1.1 2001/01/25 09:37:56 enno Exp $ * Eressea PB(E)M host Copyright (C) 1998-2000 * 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 "eressea.h" #include "building.h" /* kernel includes */ #include "item.h" #include "curse.h" /* für C_NOCOST */ #include "unit.h" #include "region.h" #include "language.h" #include "skill.h" #include "save.h" /* util includes */ #include #include #include /* libc includes */ #include #include #include /* attributes includes */ #include building_typelist *buildingtypes; const building_type * bt_find(const char* name) { const struct building_typelist * btl = buildingtypes; while (btl && strcasecmp(btl->type->_name, name)) btl = btl->next; if (!btl) { btl = buildingtypes; while (btl && strncasecmp(btl->type->_name, name, strlen(name))) btl = btl->next; } return btl?btl->type:NULL; } void bt_register(const building_type * type) { struct building_typelist * btl = malloc(sizeof(building_type)); btl->type = type; btl->next = buildingtypes; buildingtypes = btl; } int buildingcapacity(const building * b) { if (b->type->capacity>=0) { if (b->type->maxcapacity>=0) return min(b->type->maxcapacity, b->size * b->type->capacity); return b->size * b->type->capacity; } if (b->size>=b->type->maxsize) return b->type->maxcapacity; return 0; } attrib_type at_building_generic_type = { "building_generic_type", NULL, NULL, NULL, a_writestring, a_readstring, ATF_UNIQUE }; const char * buildingtype(const building * b, int bsize, const locale * lang) { const char * s = NULL; const building_type * btype = b->type; if (btype == &bt_generic) { const attrib *a = a_find(b->attribs, &at_building_generic_type); if (a) s = (const char*)a->data.v; } if (btype->name) s = btype->name(bsize, lang); if (s==NULL) s = locale_string(lang, btype->_name); if (s==NULL) s = btype->_name; return s; } int buildingmaintenance(const building * b, resource_t rtype) { const building_type * bt = b->type; int c, cost=0; if (is_cursed(b->attribs, C_NOCOST, 0)) { return 0; } for (c=0;bt->maintenance && bt->maintenance[c].number;++c) { const maintenance * m = bt->maintenance + c; if (m->type==rtype) { if (fval(m, MTF_VARIABLE)) cost += (b->size * m->number); else cost += m->number; } } return cost; } #define BMAXHASH 8191 static building *buildhash[BMAXHASH]; void bhash(building * b) { building *old = buildhash[b->no % BMAXHASH]; buildhash[b->no % BMAXHASH] = b; b->nexthash = old; } void bunhash(building * b) { building **show; for (show = &buildhash[b->no % BMAXHASH]; *show; show = &(*show)->nexthash) { if ((*show)->no == b->no) break; } if (*show) { assert(*show == b); *show = (*show)->nexthash; b->nexthash = 0; } } static building * bfindhash(int i) { building *old; for (old = buildhash[i % BMAXHASH]; old; old = old->nexthash) if (old->no == i) return old; return 0; } building * findbuilding(int i) { return bfindhash(i); } /* ** old building types ** */ /** Building: Fortification */ #ifdef NEW_BUILDINGS enum { B_SITE, B_FORTIFICATION, B_TOWER, B_CASTLE, B_FORTRESS, B_CITADEL, MAXBUILDINGS }; #endif static const char * castle_name(int bsize, const locale * lang) { const char * fname[MAXBUILDINGS] = { "site", "fortification", "tower", "castle", "fortress", "citadel" }; const construction * ctype = bt_castle.construction; int i = 0; while (ctype && ctype->maxsize != -1 && ctype->maxsize<=bsize) { bsize-=ctype->maxsize; ctype=ctype->improvement; ++i; } return locale_string(lang, fname[i]); } static requirement castle_req[] = { { R_STONE, 1, 0.5 }, { NORESOURCE, 0, 0.0 }, }; static const construction castle_bld[MAXBUILDINGS] = { { SK_BUILDING, 1, 2, 1, castle_req, &castle_bld[1] }, { SK_BUILDING, 2, 8, 1, castle_req, &castle_bld[2] }, { SK_BUILDING, 3, 40, 1, castle_req, &castle_bld[3] }, { SK_BUILDING, 4, 200, 1, castle_req, &castle_bld[4] }, { SK_BUILDING, 5, 1000, 1, castle_req, &castle_bld[5] }, { SK_BUILDING, 6, -1, 1, castle_req, NULL } }; building_type bt_castle = { "castle", BFL_NONE, 1, 4, -1, NULL, &castle_bld[0], castle_name }; /** Building: Lighthouse */ static const maintenance lighthouse_keep[] = { { R_SILVER, 100, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement lighthouse_req[] = { { R_IRON, 1, 0.5 }, { R_WOOD, 1, 0.5 }, { R_STONE, 2, 0.5 }, { R_SILVER, 100, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction lighthouse_bld = { SK_BUILDING, 3, -1, 1, lighthouse_req, NULL }; building_type bt_lighthouse = { "lighthouse", BFL_NONE, 1, 4, -1, lighthouse_keep, &lighthouse_bld }; /** Building: Mine */ static const maintenance mine_keep[] = { /* resource, number, flags */ { R_SILVER, 500, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement mine_req[] = { /* resource, number, recycle */ { R_IRON, 1, 0.5 }, { R_WOOD, 10, 0.5 }, { R_STONE, 5, 0.5 }, { R_SILVER, 250, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction mine_bld = { SK_BUILDING, 4, /* skill, minskill */ -1, 1, mine_req, /* maxsize, reqsize, required */ NULL /* improvement */ }; building_type bt_mine = { "mine", /* _name */ BFL_NONE, /* flags */ 1, -1, -1, /* capac, maxcap, maxsize */ mine_keep, /* maintenance */ &mine_bld, /* construction */ NULL /* name() */ }; /** Building: Quarry */ static const maintenance quarry_keep[] = { /* resource, number, flags */ { R_SILVER, 250, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement quarry_req[] = { /* resource, number, recycle */ { R_IRON, 1, 0.5 }, { R_WOOD, 5, 0.5 }, { R_STONE, 1, 0.5 }, { R_SILVER, 250, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction quarry_bld = { SK_BUILDING, 2, /* skill, minskill */ -1, 1, quarry_req, /* maxsize, reqsize, required */ NULL /* improvement */ }; building_type bt_quarry = { "quarry", /* _name */ BFL_NONE, /* flags */ 1, -1, -1, /* capac, maxcap, maxsize */ quarry_keep, /* maintenance */ &quarry_bld, /* construction */ NULL /* name() */ }; /** Building: harbour */ static const maintenance harbour_keep[] = { /* resource, number, flags */ { R_SILVER, 250, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement harbour_req[] = { /* resource, number, recycle */ { R_WOOD, 125, 0.5 }, { R_STONE, 125, 0.5 }, { R_SILVER, 6250, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction harbour_bld = { SK_BUILDING, 3, /* skill, minskill */ 25, 25, harbour_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_harbour = { "harbour", /* _name */ BFL_NONE, /* flags */ 1, 25, 25, /* capac/size, maxcapac, maxsize */ harbour_keep, /* maintenance */ &harbour_bld, /* construction */ NULL /* name() */ }; /** Building: academy */ static const maintenance academy_keep[] = { /* resource, number, flags */ { R_SILVER, 1000, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement academy_req[] = { /* resource, number, recycle */ { R_WOOD, 125, 0.5 }, { R_STONE, 125, 0.5 }, { R_IRON, 25, 0.0 }, { R_SILVER, 12500, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction academy_bld = { SK_BUILDING, 3, /* skill, minskill */ 25, 25, academy_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_academy = { "academy", /* _name */ BFL_NONE, /* flags */ -1, 25, 25, /* capac/size, maxcapac, maxsize */ academy_keep, /* maintenance */ &academy_bld, /* construction */ NULL /* name() */ }; /** Building: magictower */ static const maintenance magictower_keep[] = { /* resource, number, flags */ { R_SILVER, 1000, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement magictower_req[] = { /* resource, number, recycle */ { R_WOOD, 150, 0.5 }, { R_STONE, 250, 0.5 }, { R_MALLORN, 100, 0.5 }, { R_IRON, 150, 0.5 }, { R_EOG, 100, 0.5 }, { R_SILVER, 25000, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction magictower_bld = { SK_BUILDING, 5, /* skill, minskill */ 50, 50, magictower_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_magictower = { "magictower", /* _name */ BFL_NONE, /* flags */ -1, 2, 50, /* capac/size, maxcapac, maxsize */ magictower_keep, /* maintenance */ &magictower_bld, /* construction */ NULL /* name() */ }; /** Building: smithy */ static const maintenance smithy_keep[] = { /* resource, number, flags */ { R_SILVER, 300, MTF_VITAL }, { R_WOOD, 1, MTF_NONE }, { NORESOURCE, 0, MTF_NONE }, }; static requirement smithy_req[] = { /* resource, number, recycle */ { R_WOOD, 5, 0.5 }, { R_STONE, 5, 0.5 }, { R_IRON, 2, 0.5 }, { R_SILVER, 200, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction smithy_bld = { SK_BUILDING, 3, /* skill, minskill */ -1, 1, smithy_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_smithy = { "smithy", /* _name */ BFL_NONE, /* flags */ 1, -1, -1, /* capac/size, maxcapac, maxsize */ smithy_keep, /* maintenance */ &smithy_bld, /* construction */ NULL /* name() */ }; /** Building: sawmill */ static const maintenance sawmill_keep[] = { /* resource, number, flags */ { R_SILVER, 250, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement sawmill_req[] = { /* resource, number, recycle */ { R_WOOD, 5, 0.5 }, { R_STONE, 5, 0.5 }, { R_IRON, 3, 0.5 }, { R_SILVER, 200, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction sawmill_bld = { SK_BUILDING, 3, /* skill, minskill */ -1, 1, sawmill_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_sawmill = { "sawmill", /* _name */ BFL_NONE, /* flags */ 1, -1, -1, /* capac/size, maxcapac, maxsize */ sawmill_keep, /* maintenance */ &sawmill_bld, /* construction */ NULL /* name() */ }; /** Building: stables */ static const maintenance stables_keep[] = { /* resource, number, flags */ { R_SILVER, 150, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement stables_req[] = { /* resource, number, recycle */ { R_WOOD, 4, 0.5 }, { R_STONE, 2, 0.5 }, { R_IRON, 1, 0.5 }, { R_SILVER, 100, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction stables_bld = { SK_BUILDING, 2, /* skill, minskill */ -1, 1, stables_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_stables = { "stables", /* _name */ BFL_NONE, /* flags */ 1, -1, -1, /* capac/size, maxcapac, maxsize */ stables_keep, /* maintenance */ &stables_bld, /* construction */ NULL /* name() */ }; static requirement monument_req[] = { /* resource, number, recycle */ { R_WOOD, 1, 0.5 }, { R_STONE, 1, 0.5 }, { R_IRON, 1, 0.5 }, { R_SILVER, 400, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction monument_bld = { SK_BUILDING, 4, /* skill, minskill */ -1, 1, monument_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_monument = { "monument", /* _name */ BFL_NONE, /* flags */ 1, -1, -1, /* capac/size, maxcapac, maxsize */ NULL, /* maintenance */ &monument_bld, /* construction */ NULL /* name() */ }; /** Building: dam */ static const maintenance dam_keep[] = { /* resource, number, flags */ { R_WOOD, 3, MTF_NONE }, { R_SILVER, 1000, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement dam_req[] = { /* resource, number, recycle */ { R_IRON, 50, 0.5 }, { R_WOOD, 500, 0.5 }, { R_STONE, 250, 0.5 }, { R_SILVER, 25000, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction dam_bld = { SK_BUILDING, 4, /* skill, minskill */ 50, 50, dam_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_dam = { "dam", /* _name */ BFL_NONE, /* flags */ 1, -1, 50, /* capac/size, maxcapac, maxsize */ dam_keep, /* maintenance */ &dam_bld, /* construction */ NULL /* name() */ }; /** Building: caravan */ static const maintenance caravan_keep[] = { /* resource, number, flags */ { R_HORSE, 2, MTF_NONE }, { R_SILVER, 3000, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement caravan_req[] = { /* resource, number, recycle */ { R_IRON, 10, 0.5 }, { R_WOOD, 50, 0.5 }, { R_STONE, 10, 0.5 }, { R_SILVER, 5000, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction caravan_bld = { SK_BUILDING, 2, /* skill, minskill */ 10, 10, caravan_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_caravan = { "caravan", /* _name */ BFL_NONE, /* flags */ 1, -1, 10, /* capac/size, maxcapac, maxsize */ caravan_keep, /* maintenance */ &caravan_bld, /* construction */ NULL /* name() */ }; /** Building: tunnel */ static const maintenance tunnel_keep[] = { /* resource, number, flags */ { R_HORSE, 2, MTF_NONE }, { R_SILVER, 100, MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement tunnel_req[] = { /* resource, number, recycle */ { R_IRON, 100, 0.5 }, { R_WOOD, 500, 0.5 }, { R_STONE, 1000, 0.5 }, { R_SILVER,30000, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction tunnel_bld = { SK_BUILDING, 6, /* skill, minskill */ 100, 100, tunnel_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_tunnel = { "tunnel", /* _name */ BFL_NONE, /* flags */ 1, -1, 100, /* capac/size, maxcapac, maxsize */ tunnel_keep, /* maintenance */ &tunnel_bld, /* construction */ NULL /* name() */ }; /** Building: inn */ static const maintenance inn_keep[] = { /* resource, number, flags */ { R_SILVER, 5, MTF_VARIABLE|MTF_VITAL }, { NORESOURCE, 0, MTF_NONE }, }; static requirement inn_req[] = { /* resource, number, recycle */ { R_IRON, 10, 0.5 }, { R_WOOD, 30, 0.5 }, { R_STONE, 40, 0.5 }, { R_SILVER, 2000, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction inn_bld = { SK_BUILDING, 2, /* skill, minskill */ 10, 10, inn_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_inn = { "inn", /* _name */ BFL_NONE, /* flags */ 1, 10, 10, /* capac/size, maxcapac, maxsize */ inn_keep, /* maintenance */ &inn_bld, /* construction */ NULL /* name() */ }; /** Building: stonecircle */ static requirement stonecircle_req[] = { /* resource, number, recycle */ { R_WOOD, 500, 0.5 }, { R_STONE, 500, 0.5 }, { R_SILVER, 0, 0.0 }, { NORESOURCE, 0, 0.0 }, }; static const construction stonecircle_bld = { SK_BUILDING, 2, /* skill, minskill */ 100, 100, stonecircle_req, /* maxsize, reqsize, required for size */ NULL /* improvement */ }; building_type bt_stonecircle = { "stonecircle", /* _name */ BFL_NONE, /* flags */ -1, -1, 100, /* capac/size, maxcapac, maxsize */ NULL, /* maintenance */ &stonecircle_bld, /* construction */ NULL /* name() */ }; /** Building: blessedstonecircle */ building_type bt_blessedstonecircle = { "blessedstonecircle", /* _name */ BTF_NOBUILD, /* flags */ -1, -1, 100, /* capac/size, maxcapac, maxsize */ NULL, /* maintenance */ &stonecircle_bld, /* construction */ NULL /* name() */ }; /** Building: illusion */ building_type bt_illusion = { "illusion", /* _name */ BTF_NOBUILD, /* flags */ 0, 0, 0, /* capac/size, maxcapac, maxsize */ NULL, /* maintenance */ NULL, /* construction */ NULL /* name() */ }; /** Building: Generisches Gebäude */ building_type bt_generic = { "generic", /* _name */ BTF_NOBUILD, /* flags */ -1, -1, 1, /* capac/size, maxcapac, maxsize */ NULL, /* maintenance */ NULL, /* construction */ NULL /* name() */ }; /* name, maxsize, minskill, kapazitaet, M_EISEN, M_HOLZ, M_STEIN, M_SILBER, M_EOG, M_MALLORN, M_MAX_MAT, unterhalt, per_size, spezial, unterhalt_spezial, flags buildingt buildingdaten[MAXBUILDINGTYPES] = { {"Burg", -1, 1, -1, {0, 0, 1, 0, 0, 0}, 0, 0, 0, 0, 0}, {"Leuchtturm", -1, 3, 4, {1, 1, 2, 100, 0, 0}, 100, 0, 0, 0, 0}, {"Bergwerk", -1, 4, -1, {1, 10, 5, 250, 0, 0}, 500, 0, 0, 0, 0}, {"Steinbruch", -1, 2, -1, {1, 5, 1, 250, 0, 0}, 250, 0, 0, 0, 0}, {"Hafen", 25, 3, -1, {0, 5, 5, 250, 0, 0}, 250, 0, 0, 0, 0}, {"Akademie", 25, 3, 25, {1, 5, 5, 500, 0, 0}, 1000, 0, 0, 0, 0}, {"Magierturm", 50, 5, 2, {3, 3, 5, 500, 2, 2}, 1000, 0, 0, 0, 0}, {"Schmiede", -1, 3, -1, {2, 5, 5, 200, 0, 0}, 300, 0, I_WOOD, 1, 0}, {"Sägewerk", -1, 3, -1, {3, 5, 5, 200, 0, 0}, 250, 0, 0, 0, 0}, {"Pferdezucht", -1, 2, -1, {1, 4, 2, 100, 0, 0}, 150, 0, 0, 0, 0}, {"Monument", -1, 4, -1, {1, 1, 1, 400, 0, 0}, 0, 0, 0, 0, 0}, {"Damm", 50, 4, -1, {1, 10, 5, 500, 0, 0}, 1000, 0, I_WOOD, 3, 0}, {"Karawanserei", 10, 2, -1, {1, 5, 1, 500, 0, 0}, 3000, 0, I_HORSE, 2, 0}, {"Tunnel", 100, 6, -1, {1, 5, 10, 300, 0, 0}, 100, 0, I_STONE, 2, 0}, {"Taverne", -1, 2, 10, {1, 3, 4, 200, 0, 0}, 0, 5, 0, 0, 0}, {"Steinkreis", 100, 5, -1, {0, 5, 5, 0, 0, 0}, 0, 0, 0, 0, 0}, {"Gesegneter Steinkreis", 100, 5, -1, {0, 5, 5, 0, 0, 0}, 0, 0, 0, 0, NO_BUILD}, {"Traumschlößchen", 0, 0, 0, {0, 0, 0, 0, 0, 0}, 0, 0, 0, 0, NO_BUILD} }; */ const building_type * oldbuildings[MAXBUILDINGTYPES] = { &bt_castle, &bt_lighthouse, &bt_mine, &bt_quarry, &bt_harbour, &bt_academy, &bt_magictower, &bt_smithy, &bt_sawmill, &bt_stables, &bt_monument, &bt_dam, &bt_caravan, &bt_tunnel, &bt_inn, &bt_stonecircle, &bt_blessedstonecircle, &bt_illusion, }; /* for finding out what was meant by a particular building string */ static local_names * bnames; const building_type * findbuildingtype(const char * name, const locale * lang) { local_names * bn = bnames; void * i; while (bn) { if (bn->lang==lang) break; bn=bn->next; } if (!bn) { struct building_typelist * btl = buildingtypes; bn = calloc(sizeof(local_names), 1); bn->next = bnames; bn->lang = lang; while (btl) { const char * n = locale_string(lang, btl->type->_name); addtoken(&bn->names, n, (void*)btl->type); btl=btl->next; } bnames = bn; } i = findtoken(&bn->names, name); if (i==E_TOK_NOMATCH) return NULL; return (const building_type*)i; } static int sm_smithy(const unit * u, const region * r, skill_t sk, int value) /* skillmod */ { if (u->region == r && (sk==SK_ARMORER || sk==SK_WEAPONSMITH)) return value + 1; return value; } static int mm_smithy(const unit * u, const resource_type * rtype, int value) /* material-mod */ { if (rtype == oldresourcetype[R_IRON]) return value * 2; return value; } void init_buildings(void) { skillmod_data * smd; attrib * a; a = a_add(&bt_smithy.attribs, a_new(&at_skillmod)); smd = (skillmod_data *)a->data.v; smd->special = sm_smithy; smd->flags = SMF_PRODUCTION; a = a_add(&bt_smithy.attribs, make_matmod(mm_smithy)); } void bt_write(FILE * F, const building_type * bt) { fprintf(F, "BUILDINGTYPE %s\n", bt->_name); a_write(F, bt->attribs); /* scheisse, weil nicht CR. */ fprintf(F, "\"%s\";name\n", bt->_name); fprintf(F, "%d;flags\n", bt->flags); fprintf(F, "%d;capacity\n", bt->capacity); fprintf(F, "%d;maxcapacity\n", bt->maxcapacity); fprintf(F, "%d;maxsize\n", bt->maxsize); if (bt->maintenance!=NULL) assert(!"not implemented"); if (bt->construction!=NULL) assert(!"not implemented"); if (bt->construction!=NULL) assert(!"not implemented"); if (bt->name!=NULL) assert(!"not implemented"); fputs("END BUILDINGTYPE\n", F); } building_type * bt_read(FILE * F) /* this function is pretty picky */ { building_type * bt = calloc(sizeof(building_type), 1); int i = fscanf(F, "%s\n", buf); if (i==0 || i==EOF) { free(bt); return NULL; } bt->_name = strdup(buf); a_read(F, &bt->attribs); /* scheisse, weil nicht CR. */ for (;;) { char * semi = buf; fgets(buf, 1024, F); buf[strlen(buf)-1]=0; for(;;) { char * s = strchr(semi, ';'); if (s==NULL) break; semi = s + 1; } if (semi==buf) { assert(!strcmp(buf, "END BUILDINGTYPE")); break; } *(semi-1)=0; if (buf[0]=='\"') { char * s = buf+1; assert(*(semi-2)=='\"'); *(semi-2)=0; if (!strcmp(semi, "name")) bt->_name = strdup(s); } else { int i = atoi(buf); switch (semi[0]) { case 'c': if (!strcmp(semi, "capacity")) bt->capacity=i; break; case 'f': if (!strcmp(semi, "flags")) bt->flags=i; break; case 'm': if (!strcmp(semi, "maxcapacity")) bt->maxcapacity=i; else if (!strcmp(semi, "maxsize")) bt->maxsize=i; break; } } } bt_register(bt); return bt; } building_type * bt_make(const char * name, int flags, int capacity, int maxcapacity, int maxsize) { building_type * btype = calloc(sizeof(building_type), 1); btype->_name = name; btype->flags = flags | BTF_DYNAMIC; btype->capacity = capacity; btype->maxcapacity = maxcapacity; btype->maxsize = maxsize; bt_register(btype); return btype; } void * resolve_building(void * id) { return findbuilding((int)id); } void building_write(const struct building * b, FILE * F) { fprintf(F, "%s ", b?itoa36(b->no):"0"); } int building_read(struct building ** b, FILE * F) { int id; char zText[10]; fscanf(F, "%s ", zText); id = atoi36(zText); if (id==0) { *b = NULL; return 0; } else { *b = findbuilding(id); if (*b==NULL) ur_add((void*)id, (void**)b, resolve_building); return 1; } } building * new_building(const struct building_type * btype, region * r, const struct locale * lang) { building *b = (building *) calloc(1, sizeof(building)); b->no = newcontainerid(); bhash(b); b->type = btype; set_string(&b->display, ""); fset(b, FL_UNNAMED); b->region = r; addlist(&r->buildings, b); { static char buffer[IDSIZE + 1 + NAMESIZE + 1]; if(b->type==&bt_castle) sprintf(buffer, "%s %s", locale_string(lang, btype->_name), buildingid(b)); else sprintf(buffer, "%s %s", buildingtype(b, 0, lang), buildingid(b)); set_string(&b->name, buffer); } return b; } void destroy_building(building * b) { unit *u; if(!bfindhash(b->no)) return; #ifdef OLD_TRIGGER do_trigger(b, TYP_BUILDING, TR_DESTRUCT); #endif for(u=b->region->units; u; u=u->next) { if(u->building == b) leave(b->region, u); } b->size = 0; if (b->type == &bt_lighthouse) update_lighthouse(b); bunhash(b); #ifdef OLD_TRIGGER change_all_pointers(b, TYP_BUILDING, NULL); #endif #if 0 /* Memoryleak. Aber ohne klappt das Rendern nicht! */ removelist(&b->region->buildings, b); #endif /* Stattdessen nur aus Liste entfernen, aber im Speicher halten. */ choplist(&b->region->buildings, b); handle_event(&b->attribs, "destroy", b); } extern attrib_type at_icastle; int buildingeffsize(const building * b, boolean img) { int i = b->size, n = 0; const building_type * btype = &bt_castle; const construction * cons = btype->construction; if (b==NULL) return 0; if (b->type!=btype) { if (img) { const attrib * a = a_find(b->attribs, &at_icastle); if (!a || a->data.v != btype) return 0; } else return 0; } assert(cons); while (cons && cons->maxsize != -1 && i>=cons->maxsize) { i-=cons->maxsize; cons = cons->improvement; ++n; } if (n>0) return n; return 0; } unit * buildingowner(const region * r, const building * b) { unit *u = NULL; unit *first = NULL; assert(r == b->region); /* Prüfen ob Eigentümer am leben. */ for (u = r->units; u; u = u->next) { if (u->building == b) { if (!first && u->number > 0) first = u; if (fval(u, FL_OWNER) && u->number > 0) return u; if (u->number == 0) freset(u, FL_OWNER); } } /* Eigentümer tot oder kein Eigentümer vorhanden. Erste lebende Einheit * nehmen. */ if (first) fset(first, FL_OWNER); return first; }