#include #include "borders.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include typedef struct wallcurse { curse *buddy; connection *wall; } wallcurse; static void cw_init(attrib * a) { curse *c; curse_init(a); c = (curse *)a->data.v; c->data.v = calloc(sizeof(wallcurse), 1); } static void cw_write(const attrib * a, const void *target, storage * store) { connection *b = ((wallcurse *)((curse *)a->data.v)->data.v)->wall; curse_write(a, target, store); WRITE_INT(store, b->id); } typedef struct bresolve { int id; curse *self; } bresolve; static int resolve_buddy(variant data, void *addr); static int cw_read(attrib * a, void *target, gamedata *data) { storage *store = data->store; bresolve *br = calloc(sizeof(bresolve), 1); curse *c = (curse *)a->data.v; wallcurse *wc = (wallcurse *)c->data.v; variant var; curse_read(a, store, target); br->self = c; READ_INT(store, &br->id); var.i = br->id; ur_add(var, &wc->wall, resolve_borderid); var.v = br; ur_add(var, &wc->buddy, resolve_buddy); return AT_READ_OK; } /* ------------------------------------------------------------- */ /* Name: Feuerwand * Stufe: * Gebiet: Draig * Kategorie: Region, negativ * Flag: * Kosten: SPC_LINEAR * Aura: * Komponenten: * * Wirkung: * eine Wand aus Feuer entsteht in der angegebenen Richtung * * Was fuer eine Wirkung hat die? */ static void wall_vigour(curse * c, double delta) { wallcurse *wc = (wallcurse *)c->data.v; assert(wc->buddy->vigour == c->vigour); wc->buddy->vigour += delta; if (wc->buddy->vigour <= 0) { erase_border(wc->wall); wc->wall = NULL; ((wallcurse *)wc->buddy->data.v)->wall = NULL; } } const curse_type ct_firewall = { "Feuerwand", CURSETYP_NORM, 0, (M_DURATION | M_VIGOUR | NO_MERGE), NULL, /* curseinfo */ wall_vigour /* change_vigour */ }; static attrib_type at_cursewall = { "cursewall", cw_init, curse_done, curse_age, cw_write, cw_read, NULL, ATF_CURSE }; static int resolve_buddy(variant data, void *addr) { curse *result = NULL; bresolve *br = (bresolve *)data.v; connection *b; assert(br->id > 0); b = find_border(br->id); if (b && b->from && b->to) { attrib *a = a_find(b->from->attribs, &at_cursewall); while (a && a->data.v != br->self) { curse *c = (curse *)a->data.v; wallcurse *wc = (wallcurse *)c->data.v; if (wc->wall->id == br->id) break; a = a->next; } if (!a || a->type != &at_cursewall) { a = a_find(b->to->attribs, &at_cursewall); while (a && a->type == &at_cursewall && a->data.v != br->self) { curse *c = (curse *)a->data.v; wallcurse *wc = (wallcurse *)c->data.v; if (wc->wall->id == br->id) break; a = a->next; } } if (a && a->type == &at_cursewall) { curse *c = (curse *)a->data.v; free(br); result = c; } } else { /* fail, object does not exist (but if you're still loading then * you may want to try again later) */ *(curse **)addr = NULL; return -1; } *(curse **)addr = result; return 0; } static void wall_init(connection * b) { wall_data *fd = (wall_data *)calloc(sizeof(wall_data), 1); fd->countdown = -1; /* infinite */ b->data.v = fd; } static void wall_destroy(connection * b) { free(b->data.v); } static void wall_read(connection * b, gamedata * data) { static wall_data dummy; wall_data *fd = b->data.v ? (wall_data *)b->data.v : &dummy; read_reference(&fd->mage, data, read_unit_reference, resolve_unit); READ_INT(data->store, &fd->force); if (data->version >= NOBORDERATTRIBS_VERSION) { READ_INT(data->store, &fd->countdown); } fd->active = true; } static void wall_write(const connection * b, storage * store) { wall_data *fd = (wall_data *)b->data.v; write_unit_reference(fd->mage, store); WRITE_INT(store, fd->force); WRITE_INT(store, fd->countdown); } static int wall_age(connection * b) { wall_data *fd = (wall_data *)b->data.v; --fd->countdown; return (fd->countdown > 0) ? AT_AGE_KEEP : AT_AGE_REMOVE; } static region *wall_move(const connection * b, struct unit *u, struct region *from, struct region *to, bool routing) { wall_data *fd = (wall_data *)b->data.v; if (!routing && fd->active) { int hp = dice(3, fd->force) * u->number; hp = MIN(u->hp, hp); u->hp -= hp; if (u->hp) { ADDMSG(&u->faction->msgs, msg_message("firewall_damage", "region unit", from, u)); } else ADDMSG(&u->faction->msgs, msg_message("firewall_death", "region unit", from, u)); if (u->number > u->hp) { scale_number(u, u->hp); u->hp = u->number; } } return to; } static const char *b_namefirewall(const connection * b, const region * r, const faction * f, int gflags) { const char *bname; UNUSED_ARG(f); UNUSED_ARG(r); UNUSED_ARG(b); if (gflags & GF_ARTICLE) bname = "a_firewall"; else bname = "firewall"; if (gflags & GF_PURE) return bname; return LOC(f->locale, mkname("border", bname)); } border_type bt_firewall = { "firewall", VAR_VOIDPTR, b_transparent, /* transparent */ wall_init, /* init */ wall_destroy, /* destroy */ wall_read, /* read */ wall_write, /* write */ b_blocknone, /* block */ b_namefirewall, /* name */ b_rvisible, /* rvisible */ b_finvisible, /* fvisible */ b_uinvisible, /* uvisible */ NULL, wall_move, wall_age }; void convert_firewall_timeouts(connection * b, attrib * a) { while (a) { if (b->type == &bt_firewall && a->type == &at_countdown) { wall_data *fd = (wall_data *)b->data.v; fd->countdown = a->data.i; } a = a->next; } } border_type bt_wisps = { /* only here for reading old data */ "wisps", VAR_VOIDPTR, b_transparent, /* transparent */ 0, /* init */ wall_destroy, /* destroy */ wall_read, /* read */ 0, /* write */ b_blocknone, /* block */ 0, /* name */ b_rvisible, /* rvisible */ b_fvisible, /* fvisible */ b_uvisible, /* uvisible */ NULL, /* visible */ 0 }; void register_borders(void) { border_convert_cb = &convert_firewall_timeouts; at_register(&at_cursewall); register_bordertype(&bt_firewall); register_bordertype(&bt_wisps); register_bordertype(&bt_chaosgate); }