/* 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 "eressea.h" #include "battle.h" #include "alchemy.h" #include "alliance.h" #include "build.h" #include "building.h" #include "curse.h" #include "faction.h" #include "goodies.h" #include "group.h" #include "item.h" #include "karma.h" #include "magic.h" #include "message.h" #include "movement.h" #include "names.h" #include "order.h" #include "plane.h" #include "race.h" #include "region.h" #include "reports.h" #include "ship.h" #include "skill.h" #include "spell.h" #include "unit.h" /* attributes includes */ #include #include #include #include #include /* items includes */ #include /* util includes */ #include #include #include #include /* libc includes */ #include #include #include #include #include #include #include #define dbgprintf(a) fprintf a; static FILE *bdebug; #undef DELAYED_OFFENSE /* non-guarding factions cannot attack after moving */ #define TACTICS_RANDOM 5 /* define this as 1 to deactivate */ #define CATAPULT_INITIAL_RELOAD 4 /* erster schuss in runde 1 + rand() % INITIAL */ #define CATAPULT_STRUCTURAL_DAMAGE #define BASE_CHANCE 70 /* 70% Basis-Überlebenschance */ #ifdef NEW_COMBATSKILLS_RULE #define TDIFF_CHANGE 5 /* 5% höher pro Stufe */ #define DAMAGE_QUOTIENT 2 /* damage += skilldiff/DAMAGE_QUOTIENT */ #else #define TDIFF_CHANGE 10 # define DAMAGE_QUOTIENT 1 /* damage += skilldiff/DAMAGE_QUOTIENT */ #endif typedef enum combatmagic { DO_PRECOMBATSPELL, DO_POSTCOMBATSPELL } combatmagic_t; /* external variables */ boolean nobattledebug = false; /* globals */ static int obs_count = 0; #define TACTICS_MALUS #undef MAGIC_TURNS #define MINSPELLRANGE 1 #define MAXSPELLRANGE 7 #ifndef ROW_FACTOR # define ROW_FACTOR 10 #endif static const double EFFECT_PANIC_SPELL = 0.25; static const double TROLL_REGENERATION = 0.10; extern weapon_type * oldweapontype[]; /* Nach dem alten System: */ static int missile_range[2] = {FIGHT_ROW, BEHIND_ROW}; static int melee_range[2] = {FIGHT_ROW, FIGHT_ROW}; typedef struct armor_type { double penalty; double magres; int prot; char shield; char item; } armor_type; static armor_type armordata[AR_NONE + 1] = /* penalty; magres; prot; shield; item; */ { #ifdef COMPATIBILITY {-0.80, 5, 0, I_CLOAK_OF_INVULNERABILITY }, #endif { 0.30, 0.00, 5, 0, I_PLATE_ARMOR}, { 0.15, 0.00, 3, 0, I_CHAIN_MAIL}, { 0.30, 0.00, 3, 0, I_RUSTY_CHAIN_MAIL}, {-0.15, 0.00, 1, 1, I_SHIELD}, { 0.00, 0.00, 1, 1, I_RUSTY_SHIELD}, {-0.25, 0.30, 2, 1, I_LAENSHIELD}, { 0.00, 0.30, 6, 0, I_LAENCHAIN}, { 0.00, 0.00, 0, 0, I_SWORD}, { 0.00, 0.00, 0, 0, I_SWORD} }; #if 0 /* not in use */ static void validate_sides(battle * b) { side* s; cv_foreach(s, b->sides) { int snumber = 0; fighter *df; cv_foreach(df, s->fighters) { unit *du = df->unit; snumber += du->number; } cv_next(df); assert(snumber==s->flee+s->healed+s->alive+s->dead); } cv_next(s); } #else #define validate_sides(b) /**/ #endif const troop no_troop = {0, 0}; region * fleeregion(const unit * u) { region *r = u->region; region *neighbours[MAXDIRECTIONS]; int c = 0; direction_t i; if (u->ship && landregion(rterrain(r))) return NULL; if (u->ship && !(u->race->flags & RCF_SWIM) && !(u->race->flags & RCF_FLY)) { return NULL; } for (i = 0; i != MAXDIRECTIONS; ++i) { region * r2 = rconnect(r, i); if (r2) { if (can_survive(u,r2) && !move_blocked(u, r, r2)) neighbours[c++] = r2; } } if (!c) return NULL; return neighbours[rand() % c]; } static char * sidename(side * s, boolean truename) { #define SIDENAMEBUFLEN 256 static char sidename_buf[SIDENAMEBUFLEN]; if(s->stealthfaction && truename == false) { snprintf(sidename_buf, SIDENAMEBUFLEN, "%s", factionname(s->stealthfaction)); } else { snprintf(sidename_buf, SIDENAMEBUFLEN, "%s", factionname(s->bf->faction)); } return sidename_buf; } static const char * sideabkz(side *s, boolean truename) { static char sideabkz_buf[4]; if(s->stealthfaction && truename == false) { snprintf(sideabkz_buf, 4, "%s", abkz(s->stealthfaction->name, 3)); } else { snprintf(sideabkz_buf, 4, "%s", abkz(s->bf->faction->name, 3)); } return sideabkz_buf; } void message_faction(battle * b, faction * f, struct message * m) { region * r = b->region; if (f->battles==NULL || f->battles->r!=r) { struct bmsg * bm = calloc(1, sizeof(struct bmsg)); bm->next = f->battles; f->battles = bm; bm->r = r; } add_message(&f->battles->msgs, m); } int armedmen(const unit * u) { item * itm; int n = 0; if (!(urace(u)->flags & RCF_NOWEAPONS)) { if (urace(u)->ec_flags & CANGUARD) { /* kann ohne waffen bewachen: fuer untote und drachen */ n = u->number; } else { /* alle Waffen werden gezaehlt, und dann wird auf die Anzahl * Personen minimiert */ for (itm=u->items;itm;itm=itm->next) { const weapon_type * wtype = resource2weapon(itm->type->rtype); if (wtype==NULL) continue; if (effskill(u, wtype->skill) >= 1) n += itm->number; /* if (effskill(u, wtype->skill) >= wtype->minskill) n += itm->number; */ if (n>u->number) break; } n = min(n, u->number); } } return n; } static void battledebug(const char *s) { #if SHOW_DEBUG puts(s); putc('\n'); #endif if (bdebug) { dbgprintf((bdebug, "%s\n", s)); } } void message_all(battle * b, message * m) { bfaction * bf; plane * p = rplane(b->region); watcher * w; for (bf = b->factions;bf;bf=bf->next) { message_faction(b, bf->faction, m); } if (p) for (w=p->watchers;w;w=w->next) { for (bf = b->factions;bf;bf=bf->next) if (bf->faction==w->faction) break; if (bf==NULL) message_faction(b, w->faction, m); } } void battlerecord(battle * b, const char *s) { struct message * m = msg_message("msg_battle", "string", strdup(s)); message_all(b, m); msg_release(m); } void battlemsg(battle * b, unit * u, const char * s) { bfaction * bf; struct message * m; plane * p = rplane(b->region); watcher * w; sprintf(buf, "%s %s", unitname(u), s); m = msg_message("msg_battle", "string", strdup(buf)); for (bf=b->factions;bf;bf=bf->next) { message_faction(b, bf->faction, m); } if (p) for (w=p->watchers;w;w=w->next) { for (bf = b->factions;bf;bf=bf->next) if (bf->faction==w->faction) break; if (bf==NULL) message_faction(b, w->faction, m); } msg_release(m); } static void fbattlerecord(battle * b, faction * f, const char *s) { message * m = msg_message("msg_battle", "string", gc_add(strdup(s))); message_faction(b, f, m); msg_release(m); } #define enemy(as, ds) (as->enemy[ds->index]&E_ENEMY) static void set_enemy(side * as, side * ds, boolean attacking) { int i; for (i=0;i!=128;++i) { if (ds->enemies[i]==NULL) ds->enemies[i]=as; if (ds->enemies[i]==as) break; } for (i=0;i!=128;++i) { if (as->enemies[i]==NULL) as->enemies[i]=ds; if (as->enemies[i]==ds) break; } ds->enemy[as->index] |= E_ENEMY; as->enemy[ds->index] |= E_ENEMY; if (attacking) as->enemy[ds->index] |= E_ATTACKING; } static int allysfm(const side * s, const faction * f, int mode) { if (s->bf->faction==f) return mode; if (s->group) { return alliedgroup(s->battle->plane, s->bf->faction, f, s->group->allies, mode); } return alliedfaction(s->battle->plane, s->bf->faction, f, mode); } static int allysf(const side * s, const faction * f) { return allysfm(s, f, HELP_FIGHT); } fighter * select_corpse(battle * b, fighter * af) /* Wählt eine Leiche aus, der af hilft. casualties ist die Anzahl der * Toten auf allen Seiten (im Array). Wenn af == NULL, wird die * Parteizugehörigkeit ignoriert, und irgendeine Leiche genommen. * * Untote werden nicht ausgewählt (casualties, not dead) */ { int di, maxcasualties = 0; fighter *df; side *side; cv_foreach(side, b->sides) { if (!af || (!enemy(af->side, side) && allysf(af->side, side->bf->faction))) maxcasualties += side->casualties; } cv_next(side); di = rand() % maxcasualties; cv_foreach(df, b->fighters) { /* Geflohene haben auch 0 hp, dürfen hier aber nicht ausgewählt * werden! */ int dead = df->unit->number - (df->alive + df->run.number); if (!playerrace(df->unit->race)) continue; if (af && !helping(af->side, df->side)) continue; if (di < dead) { return df; } di -= dead; } cv_next(df); return NULL; } boolean helping(side * as, side * ds) { if (as->bf->faction==ds->bf->faction) return true; return (boolean)(!enemy(as, ds) && allysf(as, ds->bf->faction)); } /* return the number of live allies warning: this function only considers * troops that are still alive, not those that are still fighting although * dead. */ int countallies(side * as) { battle *b = as->battle; side *s; int count = 0; cv_foreach(s, b->sides) { if (!helping(as, s)) continue; count += s->size[SUM_ROW]; } cv_next(s); return count; } int statusrow(int status) { switch (status) { case ST_AGGRO: case ST_FIGHT: return FIGHT_ROW; case ST_BEHIND: case ST_CHICKEN: return BEHIND_ROW; case ST_AVOID: return AVOID_ROW; case ST_FLEE: return FLEE_ROW; default: assert(!"unknown combatrow"); } return FIGHT_ROW; } static double hpflee(int status) /* if hp drop below this percentage, run away */ { switch (status) { case ST_AGGRO: return 0.0; case ST_FIGHT: case ST_BEHIND: return 0.2; case ST_CHICKEN: case ST_AVOID: return 0.9; case ST_FLEE: return 1.0; default: assert(!"unknown combatrow"); } return 0.0; } int get_unitrow(const fighter * af) { static boolean * counted = NULL; static size_t csize = 0; battle * b = af->side->battle; int enemyfront = 0; int line, result; int retreat = 0; int size[NUMROWS]; int row = statusrow(af->status); int front = 0; size_t bsize; bsize = cv_size(&b->sides); if (csizeside->enemies[si];++si) { side *s = af->side->enemies[si]; if (s->size[line]>0) { int ai; enemyfront += s->size[line]; /* - s->nonblockers[line] (nicht, weil angreifer) */ for (ai=0;s->enemies[ai];++ai) { side * ally = s->enemies[ai]; if (!counted[ally->index] && !enemy(ally, af->side)) { int i; counted[ally->index] = true; for (i=0;i!=NUMROWS;++i) size[i] += ally->size[i] - ally->nonblockers[i]; } } } } if (enemyfront) break; } if (enemyfront) { for (line=FIRST_ROW;line!=NUMROWS;++line) { front += size[line]; if (!front || frontalive == fig->unit->number) return; #ifndef NO_RUNNING if (fig->run.region == NULL) { fig->run.region = fleeregion(fig->unit); if (fig->run.region == NULL) fig->run.region = b->region; } #endif #ifdef NO_RUNNING m = msg_message("casualties", "unit runto run alive fallen", fig->unit, NULL, fig->run.number, fig->alive, dead); #else m = msg_message("casualties", "unit runto run alive fallen", fig->unit, fig->run.region, fig->run.number, fig->alive, dead); #endif message_all(b, m); msg_release(m); } static int contest(int skilldiff, armor_t ar, armor_t sh) { int p, vw = BASE_CHANCE - TDIFF_CHANGE * skilldiff; double mod = 1.0; if (ar != AR_NONE) mod *= (1 + armordata[ar].penalty); if (sh != AR_NONE) mod *= (1 + armordata[sh].penalty); vw = (int)(100 - ((100 - vw) * mod)); do { p = rand() % 100; vw -= p; } while (vw >= 0 && p >= 90); return (vw <= 0); } static boolean riding(const troop t) { if (t.fighter->building!=NULL) return false; if (t.fighter->horses + t.fighter->elvenhorses > t.index) return true; return false; } static weapon * preferred_weapon(const troop t, boolean attacking) { weapon * missile = t.fighter->person[t.index].missile; weapon * melee = t.fighter->person[t.index].melee; if (attacking) { if (melee==NULL || (missile && missile->attackskill>melee->attackskill)) { return missile; } } else { if (melee==NULL || (missile && missile->defenseskill>melee->defenseskill)) { return missile; } } return melee; } static weapon * select_weapon(const troop t, boolean attacking, boolean ismissile) /* select the primary weapon for this trooper */ { if (attacking) { if (ismissile) { /* from the back rows, have to use your missile weapon */ return t.fighter->person[t.index].missile; } } else { if (!ismissile) { /* have to use your melee weapon if it's melee */ return t.fighter->person[t.index].melee; } } return preferred_weapon(t, attacking); } static int weapon_skill(const weapon_type * wtype, const unit * u, boolean attacking) /* the 'pure' skill when using this weapon to attack or defend. * only undiscriminate modifiers (not affected by troops or enemies) * are taken into account, e.g. no horses, magic, etc. */ { int skill; if (wtype==NULL) { skill = effskill(u, SK_WEAPONLESS); if (skill<=0) { /* wenn kein waffenloser kampf, dann den rassen-defaultwert */ if(u->race == new_race[RC_URUK]) { int sword = effskill(u, SK_SWORD); int spear = effskill(u, SK_SPEAR); skill = max(sword, spear) - 3; if (attacking) { skill = max(skill, u->race->at_default); } else { skill = max(skill, u->race->df_default); } } else { if (attacking) { skill = u->race->at_default; } else { skill = u->race->df_default; } } } else { /* der rassen-defaultwert kann höher sein als der Talentwert von * waffenloser kampf */ if (attacking) { if (skill < u->race->at_default) skill = u->race->at_default; } else { if (skill < u->race->df_default) skill = u->race->df_default; } } if (attacking) { skill += u->race->at_bonus; } else { skill += u->race->df_bonus; } } else { /* changed: if we own a weapon, we have at least a skill of 0 */ skill = effskill(u, wtype->skill); if (skill < wtype->minskill) skill = 0; if (skill > 0) { if(attacking) { skill += u->race->at_bonus; } else { skill += u->race->df_bonus; } } if (attacking) { skill += wtype->offmod; } else { skill += wtype->defmod; } } return skill; } static int weapon_effskill(troop t, troop enemy, const weapon * w, boolean attacking, boolean missile) /* effektiver Waffenskill während des Kampfes */ { /* In dieser Runde alle die Modifier berechnen, die fig durch die * Waffen bekommt. */ fighter * tf = t.fighter; unit * tu = t.fighter->unit; int skill; const weapon_type * wtype = w?w->type:NULL; if (wtype==NULL) { /* Ohne Waffe: Waffenlose Angriffe */ skill = weapon_skill(NULL, tu, attacking); } else { if (attacking) { skill = w->attackskill; } else { skill = w->defenseskill; } if (wtype->modifiers!=NULL) { /* Pferdebonus, Lanzenbonus, usw. */ int m; unsigned int flags = WMF_SKILL|(attacking?WMF_OFFENSIVE:WMF_DEFENSIVE); if (riding(t)) flags |= WMF_RIDING; else flags |= WMF_WALKING; if (riding(enemy)) flags |= WMF_AGAINST_RIDING; else flags |= WMF_AGAINST_WALKING; for (m=0;wtype->modifiers[m].value;++m) { if ((wtype->modifiers[m].flags & flags) == flags) { race_list * rlist = wtype->modifiers[m].races; if (rlist!=NULL) { while (rlist) { if (rlist->data == tu->race) break; rlist = rlist->next; } if (rlist==NULL) continue; } skill += wtype->modifiers[m].value; } } } } /* Burgenbonus, Pferdebonus */ if (riding(t) && (wtype==NULL || !fval(wtype, WTF_MISSILE))) { skill += 2; if (wtype) skill = skillmod(urace(tu)->attribs, tu, tu->region, wtype->skill, skill, SMF_RIDING); } if (t.indexelvenhorses) { /* Elfenpferde: Helfen dem Reiter, egal ob und welche Waffe. Das ist * eleganter, und vor allem einfacher, sonst muß man noch ein * WMF_ELVENHORSE einbauen. */ skill += 2; } if (skill>0 && !attacking && missile) { /* * Wenn ich verteidige, und nicht direkt meinem Feind gegenüberstehe, * halbiert sich mein Skill: (z.B. gegen Fernkämpfer. Nahkämpfer * können mich eh nicht treffen) */ skill /= 2; } return skill; } static char select_armor(troop t) { armor_t a = 0; int geschuetzt = 0; /* Drachen benutzen keine Rüstungen */ if (!(t.fighter->unit->race->battle_flags & BF_EQUIPMENT)) return AR_NONE; /* ... und Werwölfe auch nicht */ if(fval(t.fighter->unit, UFL_WERE)) { return AR_NONE; } do { if (armordata[a].shield == 0) { geschuetzt += t.fighter->armor[a]; if (geschuetzt > t.index) /* unser Kandidat wird geschuetzt */ return a; } ++a; } while (a != AR_MAX); return AR_NONE; } static char select_shield(troop t) { armor_t a = 0; int geschuetzt = 0; /* Drachen benutzen keine Rüstungen */ if (!(t.fighter->unit->race->battle_flags & BF_EQUIPMENT)) return AR_NONE; do { if (armordata[a].shield == 1) { geschuetzt += t.fighter->armor[a]; if (geschuetzt > t.index) /* unser Kandidat wird * * * geschuetzt */ return a; } ++a; } while (a != AR_MAX); return AR_NONE; } /* Hier ist zu beachten, ob und wie sich Zauber und Artefakte, die * Rüstungschutz geben, addieren. * - Artefakt I_TROLLBELT gibt Rüstung +1 * - Zauber Rindenhaut gibt Rüstung +3 */ int select_magicarmor(troop t) { unit *u = t.fighter->unit; int geschuetzt = 0; int ma = 0; geschuetzt = min(get_item(u, I_TROLLBELT), u->number); if (geschuetzt > t.index) /* unser Kandidat wird geschuetzt */ ma += 1; return ma; } /* Sind side ds und Magier des meffect verbündet, dann return 1*/ boolean meffect_protection(battle * b, meffect * s, side * ds) { if (!s->magician->alive) return false; if (s->duration <= 0) return false; if (enemy(s->magician->side, ds)) return false; if (allysf(s->magician->side, ds->bf->faction)) return true; return false; } /* Sind side as und Magier des meffect verfeindet, dann return 1*/ boolean meffect_blocked(battle *b, meffect *s, side *as) { if (!s->magician->alive) return false; if (s->duration <= 0) return false; if (enemy(s->magician->side, as)) return true; return false; } /* rmfighter wird schon im PRAECOMBAT gebraucht, da gibt es noch keine * troops */ void rmfighter(fighter *df, int i) { side *ds = df->side; /* nicht mehr personen abziehen, als in der Einheit am Leben sind */ assert(df->alive >= i); assert(df->alive <= df->unit->number); /* erst ziehen wir die Anzahl der Personen von den Kämpfern in der * Schlacht, dann von denen auf dieser Seite ab*/ df->side->alive -= i; df->side->battle->alive -= i; /* Dann die Kampfreihen aktualisieren */ ds->size[SUM_ROW] -= i; ds->size[statusrow(df->status)] -= i; /* Spezialwirkungen, z.B. Schattenritter */ if (df->unit->race->battle_flags & BF_NOBLOCK) { ds->nonblockers[SUM_ROW] -= i; ds->nonblockers[statusrow(df->status)] -= i; } /* und die Einheit selbst aktualisieren */ df->alive -= i; } void rmtroop(troop dt) { fighter *df = dt.fighter; /* troop ist immer eine einzele Person */ rmfighter(df, 1); assert(dt.index >= 0 && dt.index < df->unit->number); df->person[dt.index] = df->person[df->alive - df->removed]; if (df->removed) { df->person[df->alive - df->removed] = df->person[df->alive]; } df->person[df->alive].hp = 0; } void remove_troop(troop dt) { fighter * df = dt.fighter; unit * du = df->unit; rmtroop(dt); if (!df->alive && du->race->itemdrop) { item * drops = du->race->itemdrop(du->race, du->number-df->run.number); if (drops != NULL){ i_merge(&du->items, &drops); } } } /** reduces the target's exp by an equivalent of n points learning * 30 points = 1 week */ void drain_exp(struct unit *u, int n) { skill_t sk = (skill_t)(rand() % MAXSKILLS); skill_t ssk; ssk = sk; while (get_level(u, sk)==0) { sk++; if (sk == MAXSKILLS) sk = 0; if (sk == ssk) { sk = NOSKILL; break; } } if (sk != NOSKILL) { skill * sv = get_skill(u, sk); while (n>0) { if (n>=30*u->number) { reduce_skill(u, sv, 1); n-=30; } else { if (rand()%(30*u->number) 0.75) { return "eine klaffende Wunde"; } else if (q > 0.5) { return "eine schwere Wunde"; } else if (q > 0.25) { return "eine Wunde"; } return "eine kleine Wunde"; } boolean terminate(troop dt, troop at, int type, const char *damage, boolean missile) { #ifdef SMALL_BATTLE_MESSAGES char smallbuf[512]; #endif item ** pitm; fighter *df = dt.fighter; fighter *af = at.fighter; unit *au = af->unit; unit *du = df->unit; battle *b = df->side->battle; int heiltrank = 0; int faerie_level; char debugbuf[512]; /* Schild */ void **si; side *ds = df->side; int hp; int ar, an, am; int armor = select_armor(dt); int shield = select_shield(dt); const weapon_type *dwtype = NULL; const weapon_type *awtype = NULL; const weapon * weapon; int rda, sk = 0, sd; boolean magic = false; int da = dice_rand(damage); #ifdef SHOW_KILLS ++at.fighter->hits; #endif if (fval(au, UFL_WERE)) { int level = fspecial(du->faction, FS_LYCANTROPE); da += level; } switch (type) { case AT_STANDARD: weapon = select_weapon(at, true, missile); sk = weapon_effskill(at, dt, weapon, true, missile); if (weapon) awtype = weapon->type; if (awtype && fval(awtype, WTF_MAGICAL)) magic = true; break; case AT_NATURAL: sk = weapon_effskill(at, dt, NULL, true, missile); break; case AT_SPELL: case AT_COMBATSPELL: magic = true; break; default: break; } weapon = select_weapon(dt, false, true); /* missile=true to get the unmodified best weapon she has */ sd = weapon_effskill(dt, at, weapon, false, false); if (weapon!=NULL) dwtype=weapon->type; ar = armordata[armor].prot; ar += armordata[shield].prot; /* natürliche Rüstung */ an = du->race->armor; /* magische Rüstung durch Artefakte oder Sprüche */ /* Momentan nur Trollgürtel und Werwolf-Eigenschaft */ am = select_magicarmor(dt); if(fval(du, UFL_WERE)) { /* this counts as magical armor */ int level = fspecial(du->faction, FS_LYCANTROPE); am += level; } #if CHANGED_CROSSBOWS == 1 if(awtype && fval(awtype,WTF_ARMORPIERCING)) { /* crossbows */ ar /= 2; an /= 2; } #endif /* natürliche Rüstung ist halbkumulativ */ if (ar>0) { ar += an/2; } else { ar = an; } ar += am; if (type!=AT_COMBATSPELL && type!=AT_SPELL) /* Kein Zauber, normaler Waffenschaden */ { double kritchance = (sk * 3 - sd) / 200.0; kritchance = max(kritchance, 0.005); kritchance = min(0.9, kritchance); while (chance(kritchance)) { sprintf(debugbuf, "%s/%d landet einen kritischen Treffer", unitid(au), at.index); battledebug(debugbuf); da += dice_rand(damage); } da += rc_specialdamage(au->race, du->race, awtype); da += jihad(au->faction, du->race); faerie_level = fspecial(du->faction, FS_FAERIE); if (type == AT_STANDARD && faerie_level) { int c; for (c=0;weapon->type->itype->construction->materials[c].number; c++) { if(weapon->type->itype->construction->materials[c].type == R_IRON) { da += faerie_level; break; } } } if (awtype!=NULL && fval(awtype, WTF_MISSILE)) { /* Fernkampfschadenbonus */ da += af->person[at.index].damage_rear; } else if (awtype==NULL) { /* Waffenloser kampf, bonus von talentwert*/ da += effskill(au, SK_WEAPONLESS); } else { /* Nahkampfschadensbonus */ da += af->person[at.index].damage; } /* Skilldifferenzbonus */ da += max(0, (sk-sd)/DAMAGE_QUOTIENT); } if (magic) /* Magischer Schaden durch Spruch oder magische Waffe */ { double res = 1.0; /* magic_resistance gib x% Resistenzbonus zurück */ res -= magic_resistance(du)*3.0; if (du->race->battle_flags & BF_EQUIPMENT) { #ifdef TODO_RUNESWORD /* Runenschwert gibt im Kampf 80% Resistenzbonus */ if (dwp == WP_RUNESWORD) res -= 0.80; #endif /* der Effekt von Laen steigt nicht linear */ if (armor == AR_EOGCHAIN) res *= (1-armordata[armor].magres); if (shield == AR_EOGSHIELD) res *= (1-armordata[shield].magres); if (dwtype) res *= (1-dwtype->magres); } if (res > 0) { da = (int) (max(da * res, 0)); } /* gegen Magie wirkt nur natürliche und magische Rüstung */ ar = an+am; } rda = max(da - ar,0); if ((du->race->battle_flags & BF_INV_NONMAGIC) && !magic) rda = 0; else { unsigned int i = 0; if (du->race->battle_flags & BF_RES_PIERCE) i |= WTF_PIERCE; if (du->race->battle_flags & BF_RES_CUT) i |= WTF_CUT; if (du->race->battle_flags & BF_RES_BASH) i |= WTF_BLUNT; if (i && awtype && fval(awtype, i)) rda /= 2; /* Schilde */ for (si = b->meffects.begin; si != b->meffects.end; ++si) { meffect *meffect = *si; if (meffect_protection(b, meffect, ds) != 0) { assert(0 <= rda); /* rda sollte hier immer mindestens 0 sein */ /* jeder Schaden wird um effect% reduziert bis der Schild duration * Trefferpunkte aufgefangen hat */ if (meffect->typ == SHIELD_REDUCE) { hp = rda * (meffect->effect/100); rda -= hp; meffect->duration -= hp; } /* gibt Rüstung +effect für duration Treffer */ if (meffect->typ == SHIELD_ARMOR) { rda = max(rda - meffect->effect, 0); meffect->duration--; } } } } sprintf(debugbuf, "Verursacht %dTP, Rüstung %d: %d -> %d HP", da, ar, df->person[dt.index].hp, df->person[dt.index].hp - rda); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { if (rda > 0) { sprintf(smallbuf, "Der Treffer verursacht %s", rel_dam(rda, df->person[dt.index].hp)); } else { sprintf(smallbuf, "Der Treffer verursacht keinen Schaden"); } } #endif assert(dt.indexnumber); df->person[dt.index].hp -= rda; if (df->person[dt.index].hp > 0) { /* Hat überlebt */ battledebug(debugbuf); if (old_race(au->race) == RC_DAEMON) { #ifdef TODO_RUNESWORD if (select_weapon(dt, 0, -1) == WP_RUNESWORD) continue; #endif if (!(df->person[dt.index].flags & FL_COURAGE)) { df->person[dt.index].flags |= FL_DAZZLED; df->person[dt.index].defence--; } } df->person[dt.index].flags = (df->person[dt.index].flags & ~FL_SLEEPING); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, "."); battlerecord(b, smallbuf); } #endif return false; } #ifdef SHOW_KILLS ++at.fighter->kills; #endif #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, ", die tödlich ist"); } #endif /* Sieben Leben */ if (old_race(du->race) == RC_CAT && (chance(1.0 / 7))) { #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, ", doch die Katzengöttin ist gnädig"); battlerecord(b, smallbuf); } #endif assert(dt.index>=0 && dt.indexnumber); df->person[dt.index].hp = unit_max_hp(du); return false; } /* Heiltrank schluerfen und hoffen */ if (get_effect(du, oldpotiontype[P_HEAL]) > 0) { change_effect(du, oldpotiontype[P_HEAL], -1); heiltrank = 1; } else if (i_get(du->items, oldpotiontype[P_HEAL]->itype) > 0) { i_change(&du->items, oldpotiontype[P_HEAL]->itype, -1); change_effect(du, oldpotiontype[P_HEAL], 3); heiltrank = 1; } if (heiltrank && (chance(0.50))) { #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, ", doch ein Heiltrank bringt Rettung"); battlerecord(b, smallbuf); } else #endif { message * m = msg_message("battle::potionsave", "unit", du); message_faction(b, du->faction, m); msg_release(m); } assert(dt.index>=0 && dt.indexnumber); df->person[dt.index].hp = du->race->hitpoints; return false; } strcat(debugbuf, ", tot"); battledebug(debugbuf); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, "."); battlerecord(b, smallbuf); } #endif for (pitm=&du->items; *pitm; pitm=&(*pitm)->next) { const item_type * itype = (*pitm)->type; if (!itype->flags & ITF_CURSED && dt.index < (*pitm)->number) { /* 25% Grundchance, das ein Item kaputtgeht. */ if (rand() % 4 < 1) i_change(pitm, itype, -1); } } remove_troop(dt); return true; } static int count_side(const side * s, int minrow, int maxrow, boolean advance) { void **fi; int people = 0; int unitrow[NUMROWS]; int i; if (advance) { for (i=0;i!=NUMROWS;++i) unitrow[i] = -1; } for (fi = s->fighters.begin; fi != s->fighters.end; ++fi) { const fighter *fig = *fi; int row; if (fig->alive - fig->removed <= 0) continue; row = statusrow(fig->status); if (advance) { if (unitrow[row] == -1) { unitrow[row] = get_unitrow(fig); } row = unitrow[row]; } if (row >= minrow && row <= maxrow) { people += fig->alive - fig->removed; } } return people; } /* new implementation of count_enemies ignores mask, since it was never used */ int count_enemies(battle * b, side * as, int minrow, int maxrow, boolean advance) { int i = 0; void **si; if (maxrowsides.begin; si != b->sides.end; ++si) { side *side = *si; if (as==NULL || enemy(side, as)) { i += count_side(side, minrow, maxrow, advance); } } return i; } troop select_enemy(battle * b, fighter * af, int minrow, int maxrow, boolean advance) { side *as = af->side; int si; int enemies; if (af->unit->race->flags & RCF_FLY) { /* flying races ignore min- and maxrow and can attack anyone fighting * them */ minrow = FIGHT_ROW; maxrow = BEHIND_ROW; } minrow = max(minrow, FIGHT_ROW); enemies = count_enemies(b, as, minrow, maxrow, advance); /* Niemand ist in der angegebenen Entfernung? */ if (enemies<=0) return no_troop; enemies = rand() % enemies; for (si=0;as->enemies[si];++si) { side *ds = as->enemies[si]; void ** fi; int unitrow[NUMROWS]; if (advance) { int ui; for (ui=0;ui!=NUMROWS;++ui) unitrow[ui] = -1; } for (fi=ds->fighters.begin;fi!=ds->fighters.end;++fi) { fighter * df = *fi; int dr; dr = statusrow(df->status); if (advance) { if (unitrow[dr]<0) { unitrow[dr] = get_unitrow(df); } dr = unitrow[dr]; } if (dr < minrow || dr > maxrow) continue; if (df->alive - df->removed > enemies) { troop dt; dt.index = enemies; dt.fighter = df; return dt; } else enemies -= (df->alive - df->removed); } } assert(!enemies); return no_troop; } static troop select_opponent(battle * b, troop at, int minrow, int maxrow) { fighter * af = at.fighter; troop dt; if (af->unit->race->flags & RCF_FLY) { /* flying races ignore min- and maxrow and can attack anyone fighting * them */ minrow = FIGHT_ROW; maxrow = BEHIND_ROW; } minrow = max(minrow, FIGHT_ROW); #ifdef FIXED_OPPONENTS dt = af->person[at.index].opponent; if (dt.fighter!=NULL && dt.indexalive-dt.fighter->removed) { fighter * df = dt.fighter; int row = get_unitrow(df); if (row>=minrow && row<=maxrow) { troop ot = df->person[dt.index].opponent; if (ot.fighter==NULL || ot.index>=ot.fighter->alive-ot.fighter->removed) { df->person[dt.index].opponent = at; } return dt; } } #endif dt = select_enemy(b, at.fighter, minrow, maxrow, true); #ifdef FIXED_OPPONENTS if (dt.fighter!=NULL) { fighter * df = dt.fighter; troop ot = df->person[dt.index].opponent; if (ot.fighter==NULL || ot.index>=ot.fighter->alive-ot.fighter->removed) { df->person[dt.index].opponent = at; } } #endif return dt; } /* * Abfrage mit * * cvector *fgs=fighters(b,af,FIGHT_ROW,BEHIND_ROW, FS_HELP|FS_ENEMY); * fighter *fig; * * Optional: Verwirbeln. Vorsicht: Aufwendig! * v_scramble(fgs->begin, fgs->end); * * for (fig = fgs->begin; fig != fgs->end; ++fig) { * fighter *df = *fig; * * } * * cv_kill(fgs); Nicht vergessen */ cvector * fighters(battle *b, fighter *af, int minrow, int maxrow, int mask) { fighter *fig; cvector *fightervp; int row; fightervp = malloc(sizeof(cvector)); cv_init(fightervp); cv_foreach(fig, b->fighters) { row = get_unitrow(fig); if (row >= minrow && row <= maxrow) { switch (mask) { case FS_ENEMY: if (enemy(fig->side, af->side)) cv_pushback(fightervp, fig); break; case FS_HELP: if (!enemy(fig->side, af->side) && allysf(fig->side, af->side->bf->faction)) cv_pushback(fightervp, fig); break; case FS_HELP|FS_ENEMY: cv_pushback(fightervp, fig); break; default: assert(0 || !"Ungültiger Allianzstatus in fighters()"); } } } cv_next(fig); return fightervp; } static void report_failed_spell(battle * b, unit * mage, spell * sp) { message * m = msg_message("battle::spell_failed", "unit spell", mage, sp); message_all(b, m); msg_release(m); } void do_combatmagic(battle *b, combatmagic_t was) { void **fi; spell *sp; region *r = b->region; castorder *co; castorder *cll[MAX_SPELLRANK]; int level; int spellrank; int sl; for (spellrank = 0; spellrank < MAX_SPELLRANK; spellrank++) { cll[spellrank] = (castorder*)NULL; } for (fi = b->fighters.begin; fi != b->fighters.end; ++fi) { fighter * fig = *fi; unit * mage = fig->unit; int row = get_unitrow(fig); if (row>BEHIND_ROW) continue; if (fig->alive <= 0) continue; /* fighter kann im Kampf getötet worden sein */ level = eff_skill(mage, SK_MAGIC, r); if (level > 0) { double power; const struct locale * lang = mage->faction->locale; char cmd[128]; order * ord; switch(was) { case DO_PRECOMBATSPELL: sp = get_combatspell(mage, 0); sl = get_combatspelllevel(mage, 0); break; case DO_POSTCOMBATSPELL: sp = get_combatspell(mage, 2); sl = get_combatspelllevel(mage, 2); break; default: /* Fehler! */ return; } if (sp == NULL) continue; snprintf(cmd, 128, "%s \"%s\"", LOC(lang, keywords[K_CAST]), spell_name(sp, lang)); ord = parse_order(cmd, lang); if (cancast(mage, sp, 1, 1, ord) == false) { free_order(ord); continue; } level = eff_spelllevel(mage, sp, level, 1); if (sl > 0) level = min(sl, level); if (level < 0) { report_failed_spell(b, mage, sp); free_order(ord); continue; } power = spellpower(r, mage, sp, level, ord); free_order(ord); if (power <= 0) { /* Effekt von Antimagie */ report_failed_spell(b, mage, sp); pay_spell(mage, sp, level, 1); } else if (fumble(r, mage, sp, sp->level) == true) { report_failed_spell(b, mage, sp); pay_spell(mage, sp, level, 1); } else { co = new_castorder(fig, 0, sp, r, level, power, 0, 0, 0); add_castorder(&cll[(int)(sp->rank)], co); } } } for (spellrank = 0; spellrank < MAX_SPELLRANK; spellrank++) { for (co = cll[spellrank]; co; co = co->next) { fighter * fig = (fighter*)co->magician; spell * sp = co->sp; int level = co->level; double power = co->force; level = ((cspell_f)sp->sp_function)(fig, level, power, sp); if (level > 0) { pay_spell(fig->unit, sp, level, 1); if (was == DO_PRECOMBATSPELL) { get_mage(fig->unit)->precombataura = level; } } } } for (spellrank = 0; spellrank < MAX_SPELLRANK; spellrank++) { free_castorders(cll[spellrank]); } } static void do_combatspell(troop at, int row) { spell *sp; fighter *fi = at.fighter; unit *mage = fi->unit; battle *b = fi->side->battle; region *r = b->region; int level; double power; int fumblechance = 0; void **mg; order * ord; int sl; char cmd[128]; const struct locale * lang = mage->faction->locale; if (row>BEHIND_ROW) return; sp = get_combatspell(mage, 1); if (sp == NULL) { fi->magic = 0; /* Hat keinen Kampfzauber, kämpft nichtmagisch weiter */ return; } snprintf(cmd, 128, "%s \"%s\"", LOC(lang, keywords[K_CAST]), spell_name(sp, lang)); ord = parse_order(cmd, lang); if (cancast(mage, sp, 1, 1, ord) == false) { fi->magic = 0; /* Kann nicht mehr Zaubern, kämpft nichtmagisch weiter */ return; } level = eff_spelllevel(mage, sp, fi->magic, 1); if ((sl = get_combatspelllevel(mage, 1)) > 0) level = min(level, sl); if (fumble(r, mage, sp, sp->level) == true) { report_failed_spell(b, mage, sp); pay_spell(mage, sp, level, 1); return; } for (mg = b->meffects.begin; mg != b->meffects.end; ++mg) { meffect *mblock = *mg; if (mblock->typ == SHIELD_BLOCK) { if (meffect_blocked(b, mblock, fi->side) != 0) { fumblechance += mblock->duration; mblock->duration -= mblock->effect; } } } /* Antimagie die Fehlschlag erhöht */ if (rand()%100 < fumblechance) { report_failed_spell(b, mage, sp); pay_spell(mage, sp, level, 1); free_order(ord); return; } power = spellpower(r, mage, sp, level, ord); free_order(ord); if (power <= 0) { /* Effekt von Antimagie */ report_failed_spell(b, mage, sp); pay_spell(mage, sp, level, 1); return; } level = ((cspell_f)sp->sp_function)(fi, level, power, sp); if (level > 0) { pay_spell(mage, sp, level, 1); at.fighter->action_counter++; } } /* Sonderattacken: Monster patzern nicht und zahlen auch keine * Spruchkosten. Da die Spruchstärke direkt durch den Level bestimmt * wird, wirkt auch keine Antimagie (wird sonst in spellpower * gemacht) */ static void do_extra_spell(troop at, const att *a) { spell *sp; fighter *fi = at.fighter; unit *au = fi->unit; int power; sp = find_spellbyid((spellid_t)(a->data.iparam)); /* nur bei Monstern können mehrere 'Magier' in einer Einheit sein */ power = sp->level * au->number; ((cspell_f)sp->sp_function)(fi, sp->level, power, sp); } static int skilldiff(troop at, troop dt, int dist) { fighter *af = at.fighter, *df = dt.fighter; unit *au = af->unit, *du = df->unit; int is_protected = 0, skdiff = 0, sk; weapon * awp = select_weapon(at, true, dist>1); weapon * dwp = select_weapon(dt, false, dist>1); if (df->person[dt.index].flags & FL_SLEEPING) skdiff += 2; /* Effekte durch Rassen */ if (awp!=NULL && au->race == new_race[RC_HALFLING] && dragonrace(du->race)) { skdiff += 5; } /* Werwolf */ if(fval(au, UFL_WERE)) { skdiff += fspecial(au->faction, FS_LYCANTROPE); } if (old_race(au->race) == RC_GOBLIN && af->side->size[SUM_ROW] >= df->side->size[SUM_ROW] * 10) skdiff += 1; /* TODO this should be a skillmod */ skdiff += jihad(au->faction, du->race); skdiff += af->person[at.index].attack; skdiff -= df->person[dt.index].defence; if (df->building) { boolean init = false; static const curse_type * strongwall_ct, * magicwalls_ct; if (!init) { strongwall_ct = ct_find("strongwall"); magicwalls_ct = ct_find("magicwalls"); init=true; } if (df->building->type->flags & BTF_PROTECTION) { if(fspecial(au->faction, FS_SAPPER)) { /* Halbe Schutzwirkung, aufgerundet */ /* -1 because the tradepost has no protection value */ skdiff -= (buildingeffsize(df->building, false)-1+1)/2; } else { skdiff -= buildingeffsize(df->building, false)-1; } is_protected = 2; } if (strongwall_ct) { curse * c = get_curse(df->building->attribs, strongwall_ct); if (curse_active(c)) { /* wirkt auf alle Gebäude */ skdiff -= curse_geteffect(c); is_protected = 2; } } if (magicwalls_ct && curse_active(get_curse(df->building->attribs, magicwalls_ct))) { /* Verdoppelt Burgenbonus */ skdiff -= buildingeffsize(df->building, false); } } /* Goblin-Verteidigung * ist direkt in der Rassentabelle als df_default */ /* Effekte der Waffen */ skdiff += weapon_effskill(at, dt, awp, true, dist>1); if (awp && fval(awp->type, WTF_MISSILE)) { skdiff -= is_protected; if (awp->type->modifiers) { int w; for (w=0;awp->type->modifiers[w].value!=0;++w) { if (awp->type->modifiers[w].flags & WMF_MISSILE_TARGET) { /* skill decreases by targeting difficulty (bow -2, catapult -4) */ skdiff -= awp->type->modifiers[w].value; break; } } } } sk = weapon_effskill(dt, at, dwp, false, dist>1); skdiff -= sk; return skdiff; } static int setreload(troop at) { fighter * af = at.fighter; const weapon_type * wtype = af->person[at.index].missile->type; if (wtype->reload == 0) return 0; return af->person[at.index].reload = wtype->reload; } int getreload(troop at) { return at.fighter->person[at.index].reload; } #ifdef SMALL_BATTLE_MESSAGES static char * attack_message(const troop at, const troop dt, const weapon * wp, int dist) { static char smallbuf[512]; char a_unit[NAMESIZE+8], d_unit[NAMESIZE+8]; const char *noweap_string[4] = {"schlägt nach", "tritt nach", "beißt nach", "kratzt nach"}; if (at.fighter->unit->number > 1) sprintf(a_unit, "%s/%d", unitname(at.fighter->unit), at.index); else sprintf(a_unit, "%s", unitname(at.fighter->unit)); if (dt.fighter->unit->number > 1) sprintf(d_unit, "%s/%d", unitname(dt.fighter->unit), dt.index); else sprintf(d_unit, "%s", unitname(dt.fighter->unit)); if (wp == NULL) { sprintf(smallbuf, "%s %s %s", a_unit, noweap_string[rand()%4], d_unit); return smallbuf; } if (dist > 1 || wp->type->itype == olditemtype[I_CATAPULT]) { sprintf(smallbuf, "%s schießt mit %s auf %s", a_unit, locale_string(default_locale, resourcename(wp->type->itype->rtype, GR_INDEFINITE_ARTICLE)), d_unit); } else { sprintf(smallbuf, "%s schlägt mit %s nach %s", a_unit, locale_string(default_locale, resourcename(wp->type->itype->rtype, GR_INDEFINITE_ARTICLE)), d_unit); } return smallbuf; } #endif int hits(troop at, troop dt, weapon * awp) { #ifdef SMALL_BATTLE_MESSAGES char * smallbuf = NULL; battle * b = at.fighter->side->battle; #endif fighter *af = at.fighter, *df = dt.fighter; unit *au = af->unit, *du = df->unit; char debugbuf[512]; armor_t armor, shield; int skdiff = 0; int dist = get_unitrow(af) + get_unitrow(df) - 1; weapon * dwp = select_weapon(dt, false, dist>1); if (!df->alive) return 0; if (getreload(at)) return 0; if (dist>1 && (awp == NULL || !fval(awp->type, WTF_MISSILE))) return 0; /* mark this person as hit. */ df->person[dt.index].flags |= FL_HIT; if (af->person[at.index].flags & FL_STUNNED) { af->person[at.index].flags &= ~FL_STUNNED; return 0; } if ((af->person[at.index].flags & FL_TIRED && rand()%100 < 50) || (af->person[at.index].flags & FL_SLEEPING)) return 0; if (awp && fval(awp->type, WTF_MISSILE) && af->side->battle->reelarrow == true && rand()%100 < 50) { return 0; } skdiff = skilldiff(at, dt, dist); /* Verteidiger bekommt eine Rüstung */ armor = select_armor(dt); shield = select_shield(dt); sprintf(debugbuf, "%.4s/%d [%6s/%d] attackiert %.4s/%d [%6s/%d] mit %d dist %d", unitid(au), at.index, (awp != NULL) ? locale_string(default_locale, resourcename(awp->type->itype->rtype, 0)) : "unbewaffnet", weapon_effskill(at, dt, awp, true, dist>1), unitid(du), dt.index, (dwp != NULL) ? locale_string(default_locale, resourcename(dwp->type->itype->rtype, 0)) : "unbewaffnet", weapon_effskill(dt, at, dwp, false, dist>1), skdiff, dist); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { smallbuf = attack_message(at, dt, awp, dist); } #endif if (contest(skdiff, armor, shield)) { strcat(debugbuf, " und trifft."); battledebug(debugbuf); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, " und trifft."); battlerecord(b,smallbuf); } #endif return 1; } strcat(debugbuf, "."); battledebug(debugbuf); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcat(smallbuf, "."); battlerecord(b,smallbuf); } #endif return 0; } void dazzle(battle *b, troop *td) { #ifdef SMALL_BATTLE_MESSAGES char smallbuf[512]; #endif /* Nicht kumulativ ! */ if(td->fighter->person[td->index].flags & FL_DAZZLED) return; #ifdef TODO_RUNESWORD if (td->fighter->weapon[WP_RUNESWORD].count > td->index) { #ifdef SMALL_BATTLE_MESSAGES if (b->small) { strcpy(smallbuf, "Das Runenschwert glüht kurz auf."); battlerecord(b,smallbuf); } #endif return; } #endif if (td->fighter->person[td->index].flags & FL_COURAGE) { #ifdef SMALL_BATTLE_MESSAGES if (b->small) { sprintf(smallbuf, "Eine kurze Schwäche erfaßt %s/%d, vergeht jedoch " "schnell wieder.", unitname(td->fighter->unit), td->index); battlerecord(b,smallbuf); } #endif return; } if (td->fighter->person[td->index].flags & FL_DAZZLED) { #ifdef SMALL_BATTLE_MESSAGES if (b->small) { sprintf(smallbuf, "Eine kurze Schwäche erfaßt %s/%d, vergeht jedoch " "schnell wieder.", unitname(td->fighter->unit), td->index); battlerecord(b,smallbuf); } #endif return; } #ifdef SMALL_BATTLE_MESSAGES if (b->small) { sprintf(smallbuf, "%s/%d fühlt sich, als würde Kraft entzogen.", unitname(td->fighter->unit), td->index); battlerecord(b,smallbuf); } #endif td->fighter->person[td->index].flags |= FL_DAZZLED; td->fighter->person[td->index].defence--; } /* TODO: Gebäude/Schiffe sollten auch zerstörbar sein. Schwierig im Kampf, * besonders bei Schiffen. */ void damage_building(battle *b, building *bldg, int damage_abs) { bldg->size = max(1, bldg->size-damage_abs); /* Wenn Burg, dann gucken, ob die Leute alle noch in das Gebäude passen. */ if (bldg->type->flags & BTF_PROTECTION) { fighter *fi; bldg->sizeleft = bldg->size; cv_foreach(fi, b->fighters) { if (fi->building == bldg) { if (bldg->sizeleft >= fi->unit->number) { fi->building = bldg; bldg->sizeleft -= fi->unit->number; } else { fi->building = NULL; } } } cv_next(fi); } } static int attacks_per_round(troop t) { return t.fighter->person[t.index].speed; } #ifdef HEROES #define HERO_SPEED 10 static void make_heroes(battle * b) { side * s; cv_foreach(s, b->sides) { fighter * fig; cv_foreach(fig, s->fighters) { unit * u = fig->unit; if (fval(u, UFL_HERO)) { int i; assert(playerrace(u->race)); for (i=0;i!=u->number;++i) { fig->person[i].speed += (HERO_SPEED-1); } } } cv_next(fig); } cv_next(s); } #endif static void attack(battle *b, troop ta, const att *a, int numattack) { fighter *af = ta.fighter; troop td; unit *au = af->unit; int row = get_unitrow(af); int offset = row - FIGHT_ROW; switch(a->type) { case AT_STANDARD: /* Waffen, mag. Gegenstände, Kampfzauber */ case AT_COMBATSPELL: if (numattack==0 && af->magic > 0) { /* Magier versuchen immer erstmal zu zaubern, erst wenn das * fehlschlägt, wird af->magic == 0 und der Magier kämpft * konventionell weiter */ do_combatspell(ta, row); } else { weapon * wp = ta.fighter->person[ta.index].missile; if (row==FIGHT_ROW) wp = preferred_weapon(ta, true); /* Sonderbehandlungen */ if (getreload(ta)) { ta.fighter->person[ta.index].reload--; } else { boolean standard_attack = true; boolean reload = false; /* spezialattacken der waffe nur, wenn erste attacke in der runde. * sonst helden mit feuerschwertern zu mächtig */ if (numattack==0 && wp && wp->type->attack) { int dead = 0; standard_attack = wp->type->attack(&ta, &dead, row); if (!standard_attack) reload = true; af->catmsg += dead; if (!standard_attack && af->person[ta.index].last_action < b->turn) { af->person[ta.index].last_action = b->turn; af->action_counter++; } } if (standard_attack) { boolean missile = false; if (wp && fval(wp->type, WTF_MISSILE)) missile=true; if (missile) { if (row<=BEHIND_ROW) td = select_opponent(b, ta, missile_range[0], missile_range[1]); else return; } else { if (row<=FIGHT_ROW) td = select_opponent(b, ta, melee_range[0], melee_range[1]); else return; } if (!td.fighter) return; if (td.fighter->person[td.index].last_action < b->turn) { td.fighter->person[td.index].last_action = b->turn; td.fighter->action_counter++; } if (ta.fighter->person[ta.index].last_action < b->turn) { ta.fighter->person[ta.index].last_action = b->turn; ta.fighter->action_counter++; } reload = true; if (hits(ta, td, wp)) { const char * d; if (wp == NULL) d = au->race->def_damage; else if (riding(ta)) d = wp->type->damage[1]; else d = wp->type->damage[0]; terminate(td, ta, a->type, d, missile); } } if (reload && wp && wp->type->reload && !getreload(ta)) { int i = setreload(ta); sprintf(buf, " Nachladen gesetzt: %d", i); battledebug(buf); } } } break; case AT_SPELL: /* Extra-Sprüche. Kampfzauber in AT_COMBATSPELL! */ do_extra_spell(ta, a); break; case AT_NATURAL: td = select_opponent(b, ta, melee_range[0]-offset, melee_range[1]-offset); if (!td.fighter) return; if(td.fighter->person[td.index].last_action < b->turn) { td.fighter->person[td.index].last_action = b->turn; td.fighter->action_counter++; } if(ta.fighter->person[ta.index].last_action < b->turn) { ta.fighter->person[ta.index].last_action = b->turn; ta.fighter->action_counter++; } if (hits(ta, td, NULL)) { terminate(td, ta, a->type, a->data.dice, false); } break; case AT_DRAIN_ST: td = select_opponent(b, ta, melee_range[0]-offset, melee_range[1]-offset); if (!td.fighter) return; if(td.fighter->person[td.index].last_action < b->turn) { td.fighter->person[td.index].last_action = b->turn; td.fighter->action_counter++; } if(ta.fighter->person[ta.index].last_action < b->turn) { ta.fighter->person[ta.index].last_action = b->turn; ta.fighter->action_counter++; } if (hits(ta, td, NULL)) { int c = dice_rand(a->data.dice); while(c > 0) { if (rand()%2) { td.fighter->person[td.index].attack -= 1; } else { td.fighter->person[td.index].defence -= 1; } c--; } } break; case AT_DRAIN_EXP: td = select_opponent(b, ta, melee_range[0]-offset, melee_range[1]-offset); if (!td.fighter) return; if(td.fighter->person[td.index].last_action < b->turn) { td.fighter->person[td.index].last_action = b->turn; td.fighter->action_counter++; } if(ta.fighter->person[ta.index].last_action < b->turn) { ta.fighter->person[ta.index].last_action = b->turn; ta.fighter->action_counter++; } if (hits(ta, td, NULL)) { drain_exp(td.fighter->unit, dice_rand(a->data.dice)); } break; case AT_DAZZLE: td = select_opponent(b, ta, melee_range[0]-offset, melee_range[1]-offset); if (!td.fighter) return; if(td.fighter->person[td.index].last_action < b->turn) { td.fighter->person[td.index].last_action = b->turn; td.fighter->action_counter++; } if(ta.fighter->person[ta.index].last_action < b->turn) { ta.fighter->person[ta.index].last_action = b->turn; ta.fighter->action_counter++; } if (hits(ta, td, NULL)) { dazzle(b, &td); } break; case AT_STRUCTURAL: td = select_opponent(b, ta, melee_range[0]-offset, melee_range[1]-offset); if (!td.fighter) return; if(ta.fighter->person[ta.index].last_action < b->turn) { ta.fighter->person[ta.index].last_action = b->turn; ta.fighter->action_counter++; } if (td.fighter->unit->ship) { if(a_find(td.fighter->unit->ship->attribs, &at_nodestroy) == NULL) { td.fighter->unit->ship->damage += DAMAGE_SCALE * dice_rand(a->data.dice); } } else if (td.fighter->unit->building) { if(a_find(td.fighter->unit->building->attribs, &at_nodestroy) == NULL) { damage_building(b, td.fighter->unit->building, dice_rand(a->data.dice)); } } } /* Der letzte Katapultschütze setzt die * Ladezeit neu und generiert die Meldung. */ if (af->catmsg>=0 && ta.index==0) { sprintf(buf, "%d Opfer wurde%s getötet.", af->catmsg, af->catmsg<=1?"":"n"); battlerecord(b, buf); af->catmsg = -1; } } void do_attack(fighter * af) { troop ta; unit *au = af->unit; side *side = af->side; battle *b = side->battle; ta.fighter = af; assert(au && au->number); /* Da das Zuschlagen auf Einheiten und nicht auf den einzelnen * Kämpfern beruht, darf die Reihenfolge und Größe der Einheit keine * Rolle spielen, Das tut sie nur dann, wenn jeder, der am Anfang der * Runde lebte, auch zuschlagen darf. Ansonsten ist der, der zufällig * mit einer großen Einheit zuerst drankommt, extrem bevorteilt. */ ta.index = af->fighting; while (ta.index--) { /* Wir suchen eine beliebige Feind-Einheit aus. An der können * wir feststellen, ob noch jemand da ist. */ int enemies = count_enemies(b, af->side, FIGHT_ROW, LAST_ROW, true); int apr, attacks = attacks_per_round(ta); if (!enemies) break; for (apr=0;apr!=attacks;++apr) { int a; for (a=0; a!=10 && au->race->attack[a].type!=AT_NONE; ++a) { if (apr>0) { /* Wenn die Waffe nachladen muss, oder es sich nicht um einen * Waffen-Angriff handelt, dann gilt der Speed nicht. */ if (au->race->attack[a].type!=AT_STANDARD) continue; else { weapon * wp = preferred_weapon(ta, true); if (wp!=NULL && wp->type->reload) continue; } } attack(b, ta, &(au->race->attack[a]), apr); } } } } void do_regenerate(fighter *af) { troop ta; unit *au = af->unit; ta.fighter = af; ta.index = af->fighting; while(ta.index--) { af->person[ta.index].hp += effskill(au, SK_AUSDAUER); af->person[ta.index].hp = min(unit_max_hp(au), af->person[ta.index].hp); } } static void add_tactics(tactics * ta, fighter * fig, int value) { if (value == 0 || value < ta->value) return; if (value > ta->value) cv_kill(&ta->fighters); cv_pushback(&ta->fighters, fig); cv_pushback(&fig->side->battle->leaders, fig); ta->value = value; } double fleechance(unit * u) { double c = 0.20; /* Fluchtwahrscheinlichkeit in % */ region * r = u->region; attrib * a = a_find(u->attribs, &at_fleechance); /* Einheit u versucht, dem Getümmel zu entkommen */ c += (eff_skill(u, SK_STEALTH, r) * 0.05); if (get_item(u, I_UNICORN) >= u->number && eff_skill(u, SK_RIDING, r) >= 5) c += 0.30; else if (get_item(u, I_HORSE) >= u->number && eff_skill(u, SK_RIDING, r) >= 1) c += 0.10; if (old_race(u->race) == RC_HALFLING) { c += 0.20; c = min(c, 0.90); } else { c = min(c, 0.75); } if (a!=NULL) c += a->data.flt; return c; } int nextside = 0; side * make_side(battle * b, const faction * f, const group * g, boolean stealth, const faction *stealthfaction) { side *s1 = calloc(sizeof(struct side), 1); bfaction * bf; s1->battle = b; s1->group = g; s1->stealth = stealth; s1->stealthfaction = stealthfaction; cv_pushback(&b->sides, s1); for (bf = b->factions;bf;bf=bf->next) { faction * f2 = bf->faction; if (f2 == f) { s1->bf = bf; s1->index = nextside++; s1->nextF = bf->sides; bf->sides = s1; break; } } assert(bf); return s1; } void loot_items(fighter * corpse) { unit * u = corpse->unit; item * itm = u->items; battle * b = corpse->side->battle; u->items = NULL; while (itm) { int i; if (itm->number) { for (i = 10; i != 0; --i) { int loot = itm->number / i; /* Looten tun hier immer nur die Gegner. Das ist als Ausgleich für die * neue Loot-regel (nur ganz tote Einheiten) fair. * zusätzlich looten auch geflohene, aber nach anderen Regeln. */ if (loot>0) { int maxrow = BEHIND_ROW; int lootchance = 50 + b->keeploot; if (itm->type->flags & (ITF_CURSED|ITF_NOTLOST)) maxrow = LAST_ROW; itm->number -= loot; if (maxrow == LAST_ROW || rand() % 100 < lootchance) { fighter *fig = select_enemy(b, corpse, FIGHT_ROW, maxrow, false).fighter; if (fig) { item * l = fig->loot; while (l && l->type!=itm->type) l=l->next; if (!l) { l = calloc(sizeof(item), 1); l->next = fig->loot; fig->loot = l; l->type = itm->type; } l->number += loot; } } } } } itm = itm->next; } } #ifndef NO_RUNNING static void loot_fleeing(fighter* fig, unit* runner) { /* TODO: Vernünftig fixen */ runner->items = NULL; assert(runner->items == NULL); runner->items = fig->run.items; fig->run.items = NULL; } static void merge_fleeloot(fighter* fig, unit* u) { i_merge(&u->items, &fig->run.items); } #endif static boolean seematrix(const faction * f, const side * s) { if (f==s->bf->faction) return true; if (s->stealth) return false; return true; } static void aftermath(battle * b) { int i; region *r = b->region; ship *sh; side *s; cvector *fighters = &b->fighters; void **fi; int is = 0; bfaction * bf; int dead_peasants; boolean battle_was_relevant = (boolean)(b->turn+(b->has_tactics_turn?1:0)>2); #ifdef TROLLSAVE int *trollsave = calloc(2 * cv_size(&b->factions), sizeof(int)); #endif for (fi = fighters->begin; fi != fighters->end; ++fi) { fighter *df = *fi; unit *du = df->unit; int dead = du->number - df->alive - df->run.number; const attrib *a; int pr_mercy = 0; for (a = a_find(du->attribs, &at_prayer_effect); a; a = a->nexttype) { if (a->data.sa[0] == PR_MERCY) { pr_mercy = a->data.sa[1]; } } #ifdef TROLLSAVE /* Trolle können regenerieren */ if (df->alive > 0 && dead>0 && old_race(du->race) == RC_TROLL) { for (i = 0; i != dead; ++i) { if (chance(TROLL_REGENERATION)) { ++df->alive; ++df->side->alive; ++df->side->battle->alive; ++trollsave[df->side->index]; /* do not change dead here, or loop will not terminate! recalculate later */ } } dead = du->number - df->alive - df->run.number; } #endif /* Regeneration durch PR_MERCY */ if (dead>0 && pr_mercy) { for (i = 0; i != dead; ++i) { if (rand()%100 < pr_mercy) { ++df->alive; ++df->side->alive; ++df->side->battle->alive; /* do not change dead here, or loop will not terminate! recalculate later */ } } dead = du->number - df->alive - df->run.number; } /* tote insgesamt: */ df->side->dead += dead; /* Tote, die wiederbelebt werde können: */ if (playerrace(df->unit->race)) { df->side->casualties += dead; } #ifdef SHOW_KILLS if (df->hits + df->kills) { struct message * m = msg_message("killsandhits", "unit hits kills", du, df->hits, df->kills); message_faction(b, du->faction, m); msg_release(m); } #endif } /* Wenn die Schlacht kurz war, dann gib Aura für den Präcombatzauber * zurück. Nicht logisch, aber die einzige Lösung, den Verlust der * Aura durch Dummy-Angriffe zu verhindern. */ cv_foreach(s, b->sides) { if (s->bf->lastturn+(b->has_tactics_turn?1:0)<=1) continue; /* Prüfung, ob faction angegriffen hat. Geht nur über die Faction */ if (!s->bf->attacker) { fighter *fig; cv_foreach(fig, s->fighters) { sc_mage * mage = get_mage(fig->unit); if (mage) mage->spellpoints += mage->precombataura; } cv_next(fig); } /* Alle Fighter durchgehen, Mages suchen, Precombataura zurück */ } cv_next(s); /* validate_sides(b); */ /* POSTCOMBAT */ do_combatmagic(b, DO_POSTCOMBATSPELL); cv_foreach(s, b->sides) { int snumber = 0; fighter *df; boolean relevant = false; /* Kampf relevant für dieses Heer? */ if (s->bf->lastturn+(b->has_tactics_turn?1:0)>1) { relevant = true; } s->flee = 0; cv_foreach(df, s->fighters) { unit *du = df->unit; int dead = du->number - df->alive - df->run.number; int sum_hp = 0; int n; snumber += du->number; if (relevant && df->action_counter >= du->number) { ship * sh = du->ship?du->ship:leftship(du); if (sh) fset(sh, SF_DAMAGED); fset(du, UFL_LONGACTION); /* TODO: das sollte hier weg sobald anderswo üb * erall UFL_LONGACTION getestet wird. */ set_order(&du->thisorder, NULL); } for (n = 0; n != df->alive; ++n) { if (df->person[n].hp > 0) sum_hp += df->person[n].hp; } /* die weggerannten werden später subtrahiert! */ assert(du->number >= 0); if (df->alive == du->number) { du->hp = sum_hp; continue; /* nichts passiert */ } else if (df->run.hp) { #ifndef NO_RUNNING if (df->alive == 0) { /* Report the casualties */ reportcasualties(b, df, dead); /* Zuerst dürfen die Feinde plündern, die mitgenommenen Items * stehen in fig->run.items. Dann werden die Fliehenden auf * die leere (tote) alte Einheit gemapt */ if (fval(df,FIG_NOLOOT)){ merge_fleeloot(df, du); } else { loot_items(df); loot_fleeing(df, du); } scale_number(du, df->run.number); du->hp = df->run.hp; set_order(&du->thisorder, NULL); setguard(du, GUARD_NONE); fset(du, UFL_LONGACTION); leave(du->region, du); if (df->run.region) { run_to(du, df->run.region); df->run.region = du->region; } } else #endif { /* nur teilweise geflohene Einheiten mergen sich wieder */ df->alive += df->run.number; s->size[0] += df->run.number; s->size[statusrow(df->status)] += df->run.number; s->alive += df->run.number; sum_hp += df->run.hp; #ifndef NO_RUNNING merge_fleeloot(df, du); #endif df->run.number = 0; df->run.hp = 0; /* df->run.region = NULL;*/ reportcasualties(b, df, dead); scale_number(du, df->alive); du->hp = sum_hp; } } else { if (df->alive==0) { /* alle sind tot, niemand geflohen. Einheit auflösen */ df->run.number = 0; df->run.hp = 0; #ifndef NO_RUNNING df->run.region = NULL; #endif /* Report the casualties */ reportcasualties(b, df, dead); setguard(du, GUARD_NONE); scale_number(du, 0); /* Distribute Loot */ loot_items(df); } else { df->run.number = 0; df->run.hp = 0; reportcasualties(b, df, dead); scale_number(du, df->alive); du->hp = sum_hp; } } s->flee += df->run.number; if (playerrace(du->race)) { /* tote im kampf werden zu regionsuntoten: * for each of them, a peasant will die as well */ is += dead; } if (du->hp < du->number) { log_error(("%s has less hitpoints (%u) than people (%u)\n", itoa36(du->no), du->hp, du->number)); du->hp = du->number; } } cv_next(df); s->alive+=s->healed; assert(snumber==s->flee+s->alive+s->dead); } cv_next(s); dead_peasants = min(rpeasants(r), (is*BATTLE_KILLS_PEASANTS)/100); deathcounts(r, dead_peasants + is); chaoscounts(r, dead_peasants / 2); rsetpeasants(r, rpeasants(r) - dead_peasants); cv_foreach(s, b->sides) { message * seen = msg_message("battle::army_report", "index abbrev dead flown survived", s->index, gc_add(strdup(sideabkz(s, false))), s->dead, s->flee, s->alive); message * unseen = msg_message("battle::army_report", "index abbrev dead flown survived", s->index, "-?-", s->dead, s->flee, s->alive); for (bf=b->factions;bf;bf=bf->next) { faction * f = bf->faction; message * m = seematrix(f, s)?seen:unseen; message_faction(b, f, m); } msg_release(seen); msg_release(unseen); } cv_next(s); /* Wir benutzen drifted, um uns zu merken, ob ein Schiff * schonmal Schaden genommen hat. (moved und drifted * sollten in flags überführt werden */ for (fi = fighters->begin; fi != fighters->end; ++fi) { fighter *df = *fi; unit *du = df->unit; item * l; for (l=df->loot; l; l=l->next) { const item_type * itype = l->type; sprintf(buf, "%s erbeute%s %d %s.", unitname(du), du->number==1?"t":"n", l->number, locale_string(default_locale, resourcename(itype->rtype, l->number!=1))); fbattlerecord(b, du->faction, buf); i_change(&du->items, itype, l->number); } /* Wenn sich die Einheit auf einem Schiff befindet, wird * dieses Schiff beschädigt. Andernfalls ein Schiff, welches * evt. zuvor verlassen wurde. */ if (du->ship) sh = du->ship; else sh = leftship(du); if (sh && fval(sh, SF_DAMAGED) && b->turn+(b->has_tactics_turn?1:0)>2) { damage_ship(sh, 0.20); freset(sh, SF_DAMAGED); } } if (battle_was_relevant) { ship **sp = &r->ships; while (*sp) { ship * sh = *sp; freset(sh, SF_DAMAGED); if (sh->damage >= sh->size * DAMAGE_SCALE) { destroy_ship(sh); } if (*sp==sh) sp=&sh->next; } } #ifdef TROLLSAVE free(trollsave); #endif sprintf(buf, "The battle lasted %d turns, %s and %s.\n", b->turn, b->has_tactics_turn==true?"had a tactic turn":"had no tactic turn", battle_was_relevant==true?"was relevant":"was not relevant."); battledebug(buf); } static void battle_punit(unit * u, battle * b) { bfaction * bf; strlist *S, *x; for (bf = b->factions;bf;bf=bf->next) { faction *f = bf->faction; S = 0; spunit(&S, f, u, 4, see_battle); for (x = S; x; x = x->next) { fbattlerecord(b, f, x->s); if (u->faction == f) battledebug(x->s); } if (S) freestrlist(S); } } static void print_fighters(battle * b, cvector * fighters) { fighter *df; int lastrow = -1; cv_foreach(df, *fighters) { unit *du = df->unit; int row = get_unitrow(df); if (row != lastrow) { message * m = msg_message("battle::row_header", "row", row); message_all(b, m); msg_release(m); lastrow = row; } battle_punit(du, b); } cv_next(df); } static void print_header(battle * b) { bfaction * bf; void **fi; cvector *fighters = &b->fighters; boolean * seen = malloc(sizeof(boolean)*cv_size(&b->sides)); for (bf=b->factions;bf;bf=bf->next) { faction * f = bf->faction; const char * lastf = NULL; boolean first = false; strcpy(buf, "Der Kampf wurde ausgelöst von "); memset(seen, 0, sizeof(boolean)*cv_size(&b->sides)); for (fi = fighters->begin; fi != fighters->end; ++fi) { fighter *df = *fi; if (!fval(df, FIG_ATTACKED)) continue; if (!seen[df->side->index] ) { if (first) strcat(buf, ", "); if (lastf) { strcat(buf, lastf); first = true; } if (seematrix(f, df->side) == true) lastf = sidename(df->side, false); else lastf = "einer unbekannten Partei"; seen[df->side->index] = true; } } if (first) strcat(buf, " und "); if (lastf) strcat(buf, lastf); strcat(buf, "."); fbattlerecord(b, f, buf); } free(seen); } static void print_stats(battle * b) { side *s2; side *side; int i = 0; cv_foreach(side, b->sides) { bfaction *bf; char *k; boolean komma; ++i; for (bf=b->factions;bf;bf=bf->next) { faction * f = bf->faction; const char * loc_army = LOC(f->locale, "battle_army"); fbattlerecord(b, f, " "); sprintf(buf, "%s %d: %s", loc_army, side->index, seematrix(f, side) ? sidename(side,false) : LOC(f->locale, "unknown_faction")); fbattlerecord(b, f, buf); strcpy(buf, LOC(f->locale, "battle_opponents")); komma = false; cv_foreach(s2, b->sides) { if (enemy(s2, side)) { const char * abbrev = seematrix(f, s2)?sideabkz(s2, false):"-?-"; sprintf(buf, "%s%s %s %d(%s)", buf, komma++ ? "," : "", loc_army, s2->index, abbrev); } } cv_next(s2); fbattlerecord(b, f, buf); strcpy(buf, LOC(f->locale, "battle_attack")); komma = false; cv_foreach(s2, b->sides) { if (side->enemy[s2->index] & E_ATTACKING) { const char * abbrev = seematrix(f, s2)?sideabkz(s2, false):"-?-"; sprintf(buf, "%s%s %s %d(%s)", buf, komma++ ? "," : "", loc_army, s2->index, abbrev); } } cv_next(s2); fbattlerecord(b, f, buf); } buf[77] = (char)0; for (k = buf; *k; ++k) *k = '-'; battlerecord(b, buf); if (side->bf->faction) { if (side->bf->faction->alliance) { sprintf(buf, "##### %s (%s/%d)", side->bf->faction->name, itoa36(side->bf->faction->no), side->bf->faction->alliance?side->bf->faction->alliance->id:0); } else { sprintf(buf, "##### %s (%s)", side->bf->faction->name, itoa36(side->bf->faction->no)); } battledebug(buf); } print_fighters(b, &side->fighters); } cv_next(side); battlerecord(b, " "); /* Besten Taktiker ermitteln */ b->max_tactics = 0; cv_foreach(side, b->sides) { if (cv_size(&side->leader.fighters)) b->max_tactics = max(b->max_tactics, side->leader.value); } cv_next(side); if (b->max_tactics > 0) { cv_foreach(side, b->sides) { if (side->leader.value == b->max_tactics) { fighter *tf; cv_foreach(tf, side->leader.fighters) { unit *u = tf->unit; message * m = NULL; if (!fval(tf, FIG_ATTACKED)) { m = msg_message("battle::tactics_lost", "unit", u); } else { m = msg_message("battle::tactics_won", "unit", u); } message_all(b, m); msg_release(m); } cv_next(tf); } } cv_next(side); } } static int weapon_weight(const weapon * w, boolean missile) { if (missile == i2b(fval(w->type, WTF_MISSILE))) { return w->attackskill + w->defenseskill; } return 0; } fighter * make_fighter(battle * b, unit * u, side * s1, boolean attack) { #define WMAX 16 weapon weapons[WMAX]; int owp[WMAX]; int dwp[WMAX]; int w = 0; region *r = b->region; item * itm; fighter *fig = NULL; int i, t = eff_skill(u, SK_TACTICS, r); side *s2; int h; int berserk; int strongmen; int speeded = 0, speed = 1; boolean pr_aid = false; boolean stealth = (boolean)((fval(u, UFL_PARTEITARNUNG)!=0)?true:false); int rest; const attrib * a = a_find(u->attribs, &at_group); const group * g = a?(const group*)a->data.v:NULL; const attrib *a2 = a_find(u->attribs, &at_otherfaction); const faction *stealthfaction = a2?get_otherfaction(a2):NULL; /* Illusionen und Zauber kaempfen nicht */ if (fval(u->race, RCF_ILLUSIONARY) || idle(u->faction)) return NULL; if (s1==NULL) { cv_foreach(s2, b->sides) { if (s2->bf->faction == u->faction && s2->stealth==stealth && s2->stealthfaction == stealthfaction ) { if (s2->group==g) { s1 = s2; break; } } } cv_next(s2); /* aliances are moved out of make_fighter and will be handled later */ if (!s1) s1 = make_side(b, u->faction, g, stealth, stealthfaction); /* Zu diesem Zeitpunkt ist attacked noch 0, da die Einheit für noch * keinen Kampf ausgewählt wurde (sonst würde ein fighter existieren) */ } fig = calloc(1, sizeof(struct fighter)); cv_pushback(&s1->fighters, fig); cv_pushback(&b->fighters, fig); fig->unit = u; /* In einer Burg muß man a) nicht Angreifer sein, und b) drin sein, und * c) noch Platz finden. d) menschlich sein */ if (attack) fset(fig, FIG_ATTACKED); if ((!attack) && u->building && u->building->sizeleft >= u->number && playerrace(u->race)) { fig->building = u->building; fig->building->sizeleft -= u->number; } fig->status = u->status; fig->side = s1; fig->alive = u->number; fig->side->alive += u->number; fig->side->battle->alive += u->number; fig->catmsg = -1; /* Freigeben nicht vergessen! */ fig->person = calloc(fig->alive, sizeof(struct person)); h = u->hp / u->number; assert(h); rest = u->hp % u->number; /* Effekte von Sprüchen */ { static const curse_type * speed_ct; speed_ct = ct_find("speed"); if (speed_ct) { curse *c = get_curse(u->attribs, speed_ct); if (c) { speeded = get_cursedmen(u, c); speed = curse_geteffect(c); } } } /* Effekte von Alchemie */ berserk = get_effect(u, oldpotiontype[P_BERSERK]); /* change_effect wird in ageing gemacht */ /* Effekte von Artefakten */ strongmen = min(fig->unit->number, get_item(u, I_TROLLBELT)); for (a = a_find(u->attribs, &at_prayer_effect); a; a = a->nexttype) { if (a->data.sa[0] == PR_AID) { pr_aid = true; break; } } /* Hitpoints, Attack- und Defence-Boni für alle Personen */ for (i = 0; i < fig->alive; i++) { assert(i < fig->unit->number); fig->person[i].hp = h; if (i < rest) fig->person[i].hp++; if (i < speeded) fig->person[i].speed = speed; else fig->person[i].speed = 1; if (i < berserk) { fig->person[i].attack++; } /* Leute mit einem Aid-Prayer bekommen +1 auf fast alles. */ if (pr_aid) { fig->person[i].attack++; fig->person[i].defence++; fig->person[i].damage++; fig->person[i].damage_rear++; fig->person[i].flags |= FL_COURAGE; } /* Leute mit Kraftzauber machen +2 Schaden im Nahkampf. */ if (i < strongmen) { fig->person[i].damage += 2; } } /* Für alle Waffengattungne wird bestimmt, wie viele der Personen mit * ihr kämpfen könnten, und was ihr Wert darin ist. */ if (u->race->battle_flags & BF_EQUIPMENT) { int oi=0, di=0; for (itm=u->items;itm;itm=itm->next) { const weapon_type * wtype = resource2weapon(itm->type->rtype); if (wtype==NULL || itm->number==0) continue; weapons[w].attackskill = weapon_skill(wtype, u, true); weapons[w].defenseskill = weapon_skill(wtype, u, false); weapons[w].type = wtype; weapons[w].used = 0; weapons[w].count = itm->number; ++w; } fig->weapons = calloc(sizeof(weapon), w+1); memcpy(fig->weapons, weapons, w*sizeof(weapon)); for (i=0; i!=w; ++i) { int j, o=0, d=0; for (j=0; j!=i; ++j) { if (weapon_weight(fig->weapons+j, true)>=weapon_weight(fig->weapons+i, true)) ++d; if (weapon_weight(fig->weapons+j, false)>=weapon_weight(fig->weapons+i, false)) ++o; } for (j=i+1; j!=w; ++j) { if (weapon_weight(fig->weapons+j, true)>weapon_weight(fig->weapons+i, true)) ++d; if (weapon_weight(fig->weapons+j, false)>weapon_weight(fig->weapons+i, false)) ++o; } owp[o] = i; dwp[d] = i; } /* jetzt enthalten owp und dwp eine absteigend schlechter werdende Liste der Waffen * oi and di are the current index to the sorted owp/dwp arrays * owp, dwp contain indices to the figther::weapons array */ /* hand out melee weapons: */ for (i=0; i!=fig->alive; ++i) { int wpless = weapon_skill(NULL, u, true); while (oi!=w && (fig->weapons[owp[oi]].used==fig->weapons[owp[oi]].count || fval(fig->weapons[owp[oi]].type, WTF_MISSILE))) { ++oi; } if (oi==w) break; /* no more weapons available */ if (weapon_weight(fig->weapons+owp[oi], false)<=wpless) { continue; /* we fight better with bare hands */ } fig->person[i].melee = &fig->weapons[owp[oi]]; ++fig->weapons[owp[oi]].used; } /* hand out missile weapons (from back to front, in case of mixed troops). */ for (di=0, i=fig->alive; i--!=0;) { while (di!=w && (fig->weapons[dwp[di]].used==fig->weapons[dwp[di]].count || !fval(fig->weapons[dwp[di]].type, WTF_MISSILE))) { ++di; } if (di==w) break; /* no more weapons available */ if (weapon_weight(fig->weapons+dwp[di], true)>0) { fig->person[i].missile = &fig->weapons[dwp[di]]; ++fig->weapons[dwp[di]].used; } } } if (i_get(u->items, &it_demonseye)) { char lbuf[80]; const char * s = LOC(default_locale, rc_name(u->race, 3)); char * c = lbuf; while (*s) *c++ = (char)toupper(*s++); *c = 0; fig->person[0].hp = unit_max_hp(u) * 3; sprintf(buf, "Eine Stimme ertönt über dem Schlachtfeld. 'DIESES %sKIND IST MEIN. IHR SOLLT ES NICHT HABEN.'. Eine leuchtende Aura umgibt %s", lbuf, unitname(u)); battlerecord(b, buf); } s1->size[statusrow(fig->status)] += u->number; s1->size[SUM_ROW] += u->number; if (u->race->battle_flags & BF_NOBLOCK) { s1->nonblockers[statusrow(fig->status)] += u->number; } if (fig->unit->race->flags & RCF_HORSE) { fig->horses = fig->unit->number; fig->elvenhorses = 0; } else { fig->horses = get_item(u, I_HORSE); fig->elvenhorses = get_item(u, I_UNICORN); } for (i = 0; i != AR_MAX; ++i) if (u->race->battle_flags & BF_EQUIPMENT) fig->armor[i] = get_item(u, armordata[i].item); /* Jetzt muß noch geschaut werden, wo die Einheit die jeweils besten * Werte hat, das kommt aber erst irgendwo später. Ich entscheide * wärend des Kampfes, welche ich nehme, je nach Gegner. Deswegen auch * keine addierten boni. */ /* Zuerst mal die Spezialbehandlung gewisser Sonderfälle. */ fig->magic = eff_skill(u, SK_MAGIC, r); if (fig->horses) { if ((r->terrain != T_PLAIN && r->terrain != T_HIGHLAND && r->terrain != T_DESERT) || r_isforest(r) || eff_skill(u, SK_RIDING, r) < 2 || old_race(u->race) == RC_TROLL || fval(u, UFL_WERE)) fig->horses = 0; } if (fig->elvenhorses) { if (eff_skill(u, SK_RIDING, r) < 5 || old_race(u->race) == RC_TROLL || fval(u, UFL_WERE)) fig->elvenhorses = 0; } /* Schauen, wie gut wir in Taktik sind. */ if (t > 0 && old_race(u->race) == RC_INSECT) t -= 1 - (int) log10(fig->side->size[SUM_ROW]); if (t > 0 && get_unitrow(fig) == FIGHT_ROW) t += 1; #ifdef TACTICS_MALUS if (t > 0 && get_unitrow(fig) > BEHIND_ROW) t -= 1; #endif #ifdef TACTICS_RANDOM if (t > 0) { int bonus = 0; for (i = 0; i < fig->alive; i++) { int p_bonus = 0; int rnd; do { rnd = rand()%100; if (rnd >= 40 && rnd <= 69) p_bonus += 1; else if (rnd <= 89) p_bonus += 2; else p_bonus += 3; } while(rnd >= 97); bonus = max(p_bonus, bonus); } t += bonus; } #endif add_tactics(&fig->side->leader, fig, t); return fig; } static fighter * join_battle(battle * b, unit * u, boolean attack) { fighter *c = NULL; fighter *fig; attrib * a = a_find(u->attribs, &at_fleechance); if (a!=NULL) return NULL; cv_foreach(fig, b->fighters) { if (fig->unit == u) { c = fig; if (attack) fset(fig, FIG_ATTACKED); break; } } cv_next(fig); if (!c) c = make_fighter(b, u, NULL, attack); return c; } static const char * simplename(region * r) { int i; static char name[17]; const char * cp = rname(r, NULL); for (i=0;*cp && i!=16;++i, ++cp) { int c = *(unsigned char*)cp; while (c && !isalpha(c) && !isspace(c)) { ++cp; c = *(unsigned char*)cp; } if (isspace(c)) name[i] = '_'; else name[i] = *cp; if (c==0) break; } name[i]=0; return name; } static battle * make_battle(region * r) { battle *b = calloc(1, sizeof(struct battle)); unit *u; faction *lastf = NULL; bfaction * bf; static int max_fac_no = 0; /* need this only once */ if (!nobattledebug) { char zText[MAX_PATH]; char zFilename[MAX_PATH]; sprintf(zText, "%s/battles", basepath()); makedir(zText, 0700); sprintf(zFilename, "%s/battle-%d-%s.log", zText, obs_count, simplename(r)); bdebug = fopen(zFilename, "w"); if (!bdebug) log_error(("battles können nicht debugged werden\n")); else { dbgprintf((bdebug, "In %s findet ein Kampf statt:\n", rname(r, NULL))); } obs_count++; } nextside = 0; /* cv_init(&b->sides); */ b->region = r; b->plane = getplane(r); /* Finde alle Parteien, die den Kampf beobachten können: */ list_foreach(unit, r->units, u) if (u->number > 0) { if (lastf != u->faction) { /* Speed optimiert, aufeinanderfolgende * * gleiche nicht getestet */ lastf = u->faction; for (bf=b->factions;bf;bf=bf->next) if (bf->faction==lastf) break; if (!bf) { bf = calloc(sizeof(bfaction), 1); ++b->nfactions; bf->faction = lastf; bf->next = b->factions; b->factions = bf; } } } list_next(u); for (bf=b->factions;bf;bf=bf->next) { faction * f = bf->faction; max_fac_no = max(max_fac_no, f->no); } return b; } static void free_side(side * si) { cv_kill(&si->fighters); } static void free_fighter(fighter * fig) { free(fig->person); } static void free_battle(battle * b) { side *side; fighter *fighter; meffect *meffect; bfaction * bf; int max_fac_no = 0; if (bdebug) { fclose(bdebug); } for (bf=b->factions;bf;bf=bf->next) { faction * f = bf->faction; max_fac_no = max(max_fac_no, f->no); } cv_foreach(side, b->sides) { free_side(side); free(side); } cv_next(side); cv_kill(&b->sides); cv_foreach(fighter, b->fighters) { free_fighter(fighter); free(fighter); } cv_next(fighter); cv_kill(&b->fighters); cv_foreach(meffect, b->meffects) { free(meffect); } cv_next(meffect); cv_kill(&b->meffects); /* cv_kill(&b->factions); */ } static int * get_alive(battle * b, side * s, faction * vf, boolean see) { static int alive[NUMROWS]; fighter *f; memset(alive, 0, NUMROWS * sizeof(int)); cv_foreach(f, s->fighters) { if (f->alive && seematrix(vf, s)==see) alive[get_unitrow(f)] += f->alive; } cv_next(f); return alive; } static int battle_report(battle * b) { side *s, *s2; boolean cont = false; boolean komma; char buf2[1024]; bfaction *bf; buf[0] = 0; cv_foreach(s, b->sides) { cv_foreach(s2, b->sides) { if (s->alive-s->removed > 0 && s2->alive-s2->removed > 0 && enemy(s, s2)) { cont = true; s->bf->lastturn = b->turn; s2->bf->lastturn = b->turn; } } cv_next(s2); } cv_next(s); printf(" %d", b->turn); fflush(stdout); for (bf=b->factions;bf;bf=bf->next) { faction * fac = bf->faction; message * m; fbattlerecord(b, fac, " "); if (cont) m = msg_message("battle::lineup", "turn", b->turn); else m = msg_message("battle::after", ""); message_faction(b, fac, m); msg_release(m); buf2[0] = 0; komma = false; cv_foreach(s, b->sides) { if (s->alive) { int r, k = 0, * alive = get_alive(b, s, fac, seematrix(fac, s)); int l = FIGHT_ROW; const char * abbrev = seematrix(fac, s)?sideabkz(s, false):"-?-"; const char * loc_army = LOC(fac->locale, "battle_army"); sprintf(buf, "%s%s %2d(%s): ", (komma==true?", ":""), loc_army, s->index, abbrev); for (r=FIGHT_ROW;r!=NUMROWS;++r) { if (alive[r]) { if (l!=FIGHT_ROW) scat("+"); while(k--) scat("0+"); icat(alive[r]); k = 0; l=r+1; } else ++k; } strcat(buf2, buf); if (komma == false) { komma = true; } } } cv_next(s); fbattlerecord(b, fac, buf2); } return cont; } static void join_allies(battle * b) { region * r = b->region; unit * u; /* Die Anzahl der Teilnehmer kann sich in dieser Routine ändern. * Deshalb muß das Ende des Vektors vorher gemerkt werden, damit * neue Parteien nicht mit betrachtet werden: */ int size = cv_size(&b->sides); for (u=r->units;u;u=u->next) /* Was ist mit Schiffen? */ if (u->status != ST_FLEE && u->status != ST_AVOID && !fval(u, UFL_LONGACTION) && u->number > 0) { int si; faction * f = u->faction; fighter * c = NULL; for (si = 0; si != size; ++si) { int se; side *s = b->sides.begin[si]; /* Wenn alle attackierten noch FFL_NOAID haben, dann kämpfe nicht mit. */ if (fval(s->bf->faction, FFL_NOAID)) continue; if (s->bf->faction!=f) { /* Wenn wir attackiert haben, kommt niemand mehr hinzu: */ if (s->bf->attacker) continue; /* alliiert müssen wir schon sein, sonst ist's eh egal : */ if (!alliedunit(u, s->bf->faction, HELP_FIGHT)) continue; /* wenn die partei verborgen ist, oder gar eine andere * vorgespiegelt wird, und er sich uns gegenüber nicht zu * erkennen gibt, helfen wir ihm nicht */ if (s->stealthfaction){ if(!allysfm(s, u->faction, HELP_FSTEALTH)) { continue; } } } /* einen alliierten angreifen dürfen sie nicht, es sei denn, der * ist mit einem alliierten verfeindet, der nicht attackiert * hat: */ for (se = 0; se != size; ++se) { side * evil = b->sides.begin[se]; if (u->faction==evil->bf->faction) continue; if (alliedunit(u, evil->bf->faction, HELP_FIGHT) && !evil->bf->attacker) continue; if (enemy(s, evil)) break; } if (se==size) continue; /* Wenn die Einheit belagert ist, muß auch einer der Alliierten belagert sein: */ if (besieged(u)) { void ** fi; boolean siege = false; for (fi = s->fighters.begin; !siege && fi != s->fighters.end; ++fi) { fighter *ally = *fi; if (besieged(ally->unit)) siege = true; } if (!siege) continue; } /* keine Einwände, also soll er mitmachen: */ if (!c) c = join_battle(b, u, false); if (!c) continue; /* Die Feinde meiner Freunde sind meine Feinde: */ for (se = 0; se != size; ++se) { side * evil = b->sides.begin[se]; if (evil->bf->faction!=u->faction && enemy(s, evil)) { set_enemy(evil, c->side, false); } } } } } static void flee(const troop dt) { fighter * fig = dt.fighter; #ifndef NO_RUNNING unit * u = fig->unit; int carry = personcapacity(u) - u->race->weight; int money; item ** ip = &u->items; while (*ip) { item * itm = *ip; const item_type * itype = itm->type; int keep = 0; if (fval(itype, ITF_ANIMAL)) { /* Regeländerung: Man muß das Tier nicht reiten können, * um es vom Schlachtfeld mitzunehmen, ist ja nur * eine Region weit. * */ keep = min(1, itm->number); /* da ist das weight des tiers mit drin */ carry += itype->capacity - itype->weight; } else if (itm->type->weight <= 0) { /* if it doesn't weigh anything, it won't slow us down */ keep = itm->number; } /* jeder troop nimmt seinen eigenen Teil der Sachen mit */ if (keep>0){ if (itm->number==keep) { i_add(&fig->run.items, i_remove(ip, itm)); } else { item *run_itm = i_new(itype, keep); i_add(&fig->run.items, run_itm); i_change(ip, itype, -keep); } } if (*ip==itm) ip = &itm->next; } /* we will take money with us */ money = get_money(u); /* nur ganzgeflohene/resttote Einheiten verlassen die Region */ if (money > carry) money = carry; if (money > 0) { i_change(&u->items, i_silver, -money); i_change(&fig->run.items, i_silver, +money); } #endif fig->run.hp += fig->person[dt.index].hp; ++fig->run.number; remove_troop(dt); } #ifdef DELAYED_OFFENSE static boolean guarded_by(region * r, faction * f) { unit * u; for (u=r->units;u;u=u->next) { if (u->faction == f && getguard(u)) return true; } return false; } #endif static boolean init_battle(region * r, battle **bp) { battle * b = NULL; unit * u; boolean fighting = false; /* list_foreach geht nicht, wegen flucht */ for (u = r->units; u != NULL; u = u->next) { if (fval(u, UFL_LONGACTION)) continue; if (u->number > 0) { order * ord; for (ord=u->orders;ord;ord=ord->next) { boolean init=false; static const curse_type * peace_ct, * slave_ct, * calm_ct; if (!init) { init = true; peace_ct = ct_find("peacezone"); slave_ct = ct_find("slavery"); calm_ct = ct_find("calmmonster"); } if (get_keyword(ord) == K_ATTACK) { unit *u2; fighter *c1, *c2; if(r->planep && fval(r->planep, PFL_NOATTACK)) { cmistake(u, ord, 271, MSG_BATTLE); continue; } /** ** Fehlerbehandlung Angreifer **/ #ifdef DELAYED_OFFENSE if (get_moved(&u->attribs) && !guarded_by(r, u->faction)) { add_message(&u->faction->msgs, msg_message("no_attack_after_advance", "unit region command", u, u->region, ord)); } #endif if (LongHunger(u)) { cmistake(u, ord, 225, MSG_BATTLE); continue; } if (u->status == ST_AVOID || u->status == ST_FLEE) { cmistake(u, ord, 226, MSG_BATTLE); continue; } /* ist ein Flüchtling aus einem andern Kampf */ if (fval(u, UFL_LONGACTION)) continue; if (peace_ct && curse_active(get_curse(r->attribs, peace_ct))) { sprintf(buf, "Hier ist es so schön friedlich, %s möchte " "hier niemanden angreifen.", unitname(u)); mistake(u, ord, buf, MSG_BATTLE); continue; } if (slave_ct && curse_active(get_curse(u->attribs, slave_ct))) { sprintf(buf, "%s kämpft nicht.", unitname(u)); mistake(u, ord, buf, MSG_BATTLE); continue; } /* Fehler: "Das Schiff muß erst verlassen werden" */ if (u->ship != NULL && rterrain(r) != T_OCEAN) { cmistake(u, ord, 19, MSG_BATTLE); continue; } if (leftship(u)) { cmistake(u, ord, 234, MSG_BATTLE); continue; } /* Ende Fehlerbehandlung Angreifer */ init_tokens(ord); skip_token(); /* attackierte Einheit ermitteln */ u2 = getunit(r, u->faction); /* Beginn Fehlerbehandlung */ /* Fehler: "Die Einheit wurde nicht gefunden" */ if (!u2 || u2->number == 0 || !cansee(u->faction, u->region, u2, 0)) { cmistake(u, ord, 63, MSG_BATTLE); continue; } /* Fehler: "Die Einheit ist eine der unsrigen" */ if (u2->faction == u->faction) { cmistake(u, ord, 45, MSG_BATTLE); continue; } /* Fehler: "Die Einheit ist mit uns alliert" */ if (alliedunit(u, u2->faction, HELP_FIGHT)) { cmistake(u, ord, 47, MSG_BATTLE); continue; } /* xmas */ if (u2->no==atoi36("xmas") && old_race(u2->irace)==RC_GNOME) { a_add(&u->attribs, a_new(&at_key))->data.i = atoi36("coal"); sprintf(buf, "%s ist böse gewesen...", unitname(u)); mistake(u, ord, buf, MSG_BATTLE); continue; } if (u2->faction->age < NewbieImmunity()) { add_message(&u->faction->msgs, msg_feedback(u, u->thisorder, "newbie_immunity_error", "turns", NewbieImmunity())); continue; } /* Fehler: "Die Einheit ist mit uns alliert" */ if (calm_ct && curse_active(get_cursex(u->attribs, calm_ct, (void*)u2->faction, cmp_curseeffect))) { cmistake(u, ord, 47, MSG_BATTLE); continue; } /* Ende Fehlerbehandlung */ if (b==NULL) { unit * utmp; for (utmp=r->units; utmp!=NULL; utmp=utmp->next) { fset(utmp->faction, FFL_NOAID); } b = make_battle(r); } c1 = join_battle(b, u, true); c2 = join_battle(b, u2, false); /* Hat die attackierte Einheit keinen Noaid-Status, * wird das Flag von der Faction genommen, andere * Einheiten greifen ein. */ if (!fval(u2, UFL_NOAID)) freset(u2->faction, FFL_NOAID); if (c1!=NULL && c2!=NULL) { /* Merken, wer Angreifer ist, für die Rückzahlung der * Präcombataura bei kurzem Kampf. */ c1->side->bf->attacker = true; set_enemy(c1->side, c2->side, true); if (!enemy(c1->side, c2->side)) { sprintf(buf, "%s attackiert %s", sidename(c1->side, false), sidename(c2->side, false)); battledebug(buf); } fighting = true; } } } } } *bp = b; return fighting; } static void battle_stats(FILE * F, battle * b) { typedef struct stat_info { struct stat_info * next; const weapon_type * wtype; int level; int number; } stat_info; side * s; cv_foreach(s, b->sides) { fighter * df; stat_info * stats = NULL, * stat; cv_foreach(df, s->fighters) { unit *du = df->unit; troop dt; stat_info * slast = NULL; dt.fighter = df; for (dt.index=0;dt.index!=du->number;++dt.index) { weapon * wp = preferred_weapon(dt, true); int level = wp?wp->attackskill:0; const weapon_type * wtype = wp?wp->type:NULL; stat_info ** slist = &stats; if (slast && slast->wtype==wtype && slast->level==level) { ++slast->number; continue; } while (*slist && (*slist)->wtype!=wtype) { slist = &(*slist)->next; } while (*slist && (*slist)->wtype==wtype && (*slist)->level>level) { slist = &(*slist)->next; } stat = *slist; if (stat==NULL || stat->wtype!=wtype || stat->level!=level) { stat = calloc(1, sizeof(stat_info)); stat->wtype = wtype; stat->level = level; stat->next = *slist; *slist = stat; } slast = stat; ++slast->number; } } cv_next(df); fprintf(F, "##STATS## Heer %u - %s:\n", s->index, factionname(s->bf->faction)); for (stat=stats;stat!=NULL;stat=stat->next) { fprintf(F, "%s %u : %u\n", stat->wtype?stat->wtype->itype->rtype->_name[0]:"none", stat->level, stat->number); } freelist(stats); } cv_next(s); } void do_battle(void) { region *r; #ifdef SMALL_BATTLE_MESSAGES char smallbuf[512]; #endif for (r=regions;r!=NULL;r=r->next) { battle *b = NULL; boolean fighting = false; side *s; ship * sh; void **fi; building *bu; fighting = init_battle(r, &b); if (b==NULL) continue; /* Bevor wir die alliierten hineinziehen, sollten wir schauen, * * Ob jemand fliehen kann. Dann erübrigt sich das ganze ja * vielleicht schon. */ print_header(b); if (!fighting) { /* Niemand mehr da, Kampf kann nicht stattfinden. */ message * m = msg_message("battle::aborted", ""); message_all(b, m); msg_release(m); free_battle(b); free(b); continue; } join_allies(b); #ifdef HEROES make_heroes(b); #endif /* Alle Mann raus aus der Burg! */ for (bu=r->buildings; bu!=NULL; bu=bu->next) bu->sizeleft = bu->size; /* make sure no ships are damaged initially */ for (sh=r->ships; sh; sh=sh->next) freset(sh, SF_DAMAGED); /* Gibt es eine Taktikrunde ? */ if (cv_size(&b->leaders)) { b->turn = 0; b->has_tactics_turn = true; } else { b->turn = 1; b->has_tactics_turn = false; } if (b->region->flags & RF_COMBATDEBUG) battle_stats(bdebug, b); /* PRECOMBATSPELLS */ do_combatmagic(b, DO_PRECOMBATSPELL); /* Nun erstellen wir eine Liste von allen Kämpfern, die wir * dann scramblen. Zuerst werden sie wild gemischt, und dann wird (stabil) * nach der Kampfreihe sortiert */ v_scramble(b->fighters.begin, b->fighters.end); v_sort(b->fighters.begin, b->fighters.end, (v_sort_fun) sort_fighterrow); cv_foreach(s, b->sides) { v_sort(s->fighters.begin, s->fighters.end, (v_sort_fun) sort_fighterrow); } cv_next(side); print_stats(b); /* gibt die Kampfaufstellung aus */ printf("%s (%d, %d) : ", rname(r, NULL), r->x, r->y); b->dh = 0; /* Kämpfer zählen und festlegen, ob es ein kleiner Kampf ist */ for (fi = b->fighters.begin; fi != b->fighters.end; ++fi) { fighter *fig = *fi; b->dh += fig->unit->number; } #ifdef SMALL_BATTLE_MESSAGES if (b->dh <= 30) { b->small = true; } else { b->small = false; } #endif for (;battle_report(b) && b->turn<=COMBAT_TURNS;++b->turn) { char lbuf[256]; int flee_ops = 1; int i; sprintf(lbuf, "*** Runde: %d", b->turn); battledebug(lbuf); /* Der doppelte Versuch immer am Anfang der 1. Kampfrunde, nur * ein Versuch vor der Taktikerrunde. Die 0. Runde ist immer * die Taktikerrunde. */ if (b->turn==1) flee_ops = 2; for (i=1;i<=flee_ops;i++) { for (fi = b->fighters.begin; fi != b->fighters.end; ++fi) { fighter *fig = *fi; unit *u = fig->unit; troop dt; int runners = 0; /* Flucht nicht bei mehr als 600 HP. Damit Wyrme tötbar bleiben. */ int runhp = min(600,(int)(0.9+unit_max_hp(u)*hpflee(u->status))); side *side = fig->side; if (fval(u->race, RCF_UNDEAD) || old_race(u->race) == RC_SHADOWKNIGHT) continue; if (u->ship) continue; dt.fighter = fig; #ifndef NO_RUNNING if (!fig->run.region) fig->run.region = fleeregion(u); if (!fig->run.region) continue; #endif dt.index = fig->alive - fig->removed; while (side->size[SUM_ROW] && dt.index != 0) { double ispaniced = 0.0; --dt.index; assert(dt.index>=0 && dt.indexunit->number); assert(fig->person[dt.index].hp > 0); /* Versuche zu fliehen, wenn * - Kampfstatus fliehe * - schwer verwundet und nicht erste kampfrunde * - in panik (Zauber) * aber nicht, wenn der Zaubereffekt Held auf dir liegt! */ switch (u->status) { case ST_FLEE: break; default: if ((fig->person[dt.index].flags & FL_HIT) == 0) continue; if (b->turn<=1) continue; if (fig->person[dt.index].hp <= runhp) break; if (fig->person[dt.index].flags & FL_PANICED) { if ((fig->person[dt.index].flags & FL_COURAGE)==0) break; } continue; } if (fig->person[dt.index].flags & FL_PANICED) { ispaniced = EFFECT_PANIC_SPELL; } if (chance(min(fleechance(u)+ispaniced, 0.90))) { ++runners; flee(dt); #ifdef SMALL_BATTLE_MESSAGES if (b->small) { sprintf(smallbuf, "%s/%d gelingt es, vom Schlachtfeld zu entkommen.", unitname(fig->unit), dt.index); battlerecord(b, smallbuf); } } else if (b->small) { sprintf(smallbuf, "%s/%d versucht zu fliehen, wird jedoch aufgehalten.", unitname(fig->unit), dt.index); battlerecord(b, smallbuf); #endif } } if(runners > 0) { sprintf(lbuf, "Flucht: %d aus %s", runners, itoa36(fig->unit->no)); battledebug(lbuf); } } } /* this has to be calculated before the actual attacks take * place because otherwise the dead would not strike the * round they die. */ for (fi = b->fighters.begin; fi != b->fighters.end; ++fi) { fighter *fig = *fi; fig->fighting = fig->alive - fig->removed; } for (fi = b->fighters.begin; fi != b->fighters.end; ++fi) { fighter *fig = *fi; /* ist in dieser Einheit noch jemand handlungsfähig? */ if (fig->fighting <= 0) continue; /* Taktikrunde: */ if (b->turn == 0) { side *side; boolean yes = false; cv_foreach(side, b->sides) { if (b->max_tactics > 0 && side->leader.value == b->max_tactics && helping(side, fig->side)) { yes = true; break; } } cv_next(side); if (!yes) continue; } /* Handle the unit's attack on someone */ do_attack(fig); } /* Regeneration */ for (fi = b->fighters.begin; fi != b->fighters.end; ++fi) { fighter *fig = *fi; if(fspecial(fig->unit->faction, FS_REGENERATION)) { fig->fighting = fig->alive - fig->removed; if(fig->fighting == 0) continue; do_regenerate(fig); } } } printf("\n"); /* Auswirkungen berechnen: */ aftermath(b); /* #ifdef MALLOCDBG assert(_CrtCheckMemory()); #endif */ /* Hier ist das Gefecht beendet, und wir können die * Hilfsstrukturen * wieder löschen: */ if (b) { free_battle(b); free(b); } } } int nb_armor(const unit *u, int index) { int a, av = 0; int geschuetzt = 0; if (!(u->race->battle_flags & BF_EQUIPMENT)) return AR_NONE; /* Normale Rüstung */ a = 0; do { if (armordata[a].shield == 0) { geschuetzt += get_item(u, armordata[a].item); if (geschuetzt > index) av += armordata[a].prot; } ++a; } while (a != AR_MAX); /* Schild */ a = 0; do { if (armordata[a].shield == 1) { geschuetzt += get_item(u, armordata[a].item); if (geschuetzt > index) av += armordata[a].prot; } ++a; } while (a != AR_MAX); return av; }