forked from github/server
WIP: refactor struct sc_mage (make it private).
tests are failing and/or slow.
This commit is contained in:
parent
6f26898822
commit
4a66b558c5
17 changed files with 291 additions and 288 deletions
|
@ -276,8 +276,12 @@ static int tolua_unit_set_guard(lua_State * L)
|
|||
|
||||
static const char *unit_getmagic(const unit * u)
|
||||
{
|
||||
sc_mage *mage = get_mage_depr(u);
|
||||
return mage ? magic_school[mage->magietyp] : NULL;
|
||||
struct sc_mage *mage = get_mage(u);
|
||||
if (mage) {
|
||||
magic_t mtype = mage_get_type(mage);
|
||||
return magic_school[mtype];
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static int tolua_unit_get_magic(lua_State * L)
|
||||
|
@ -289,16 +293,15 @@ static int tolua_unit_get_magic(lua_State * L)
|
|||
|
||||
static void unit_setmagic(unit * u, const char *type)
|
||||
{
|
||||
sc_mage *mage = get_mage(u);
|
||||
int mtype;
|
||||
for (mtype = 0; mtype != MAXMAGIETYP; ++mtype) {
|
||||
if (strcmp(magic_school[mtype], type) == 0)
|
||||
break;
|
||||
}
|
||||
if (mtype == MAXMAGIETYP)
|
||||
return;
|
||||
struct sc_mage *mage = get_mage(u);
|
||||
if (mage == NULL) {
|
||||
mage = create_mage(u, (magic_t)mtype);
|
||||
int mtype;
|
||||
for (mtype = 0; mtype != MAXMAGIETYP; ++mtype) {
|
||||
if (strcmp(magic_school[mtype], type) == 0) {
|
||||
create_mage(u, (magic_t)mtype);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -521,7 +524,7 @@ static int tolua_unit_addspell(lua_State * L)
|
|||
return EINVAL;
|
||||
}
|
||||
else {
|
||||
unit_add_spell(u, 0, sp, level);
|
||||
unit_add_spell(u, sp, level);
|
||||
}
|
||||
|
||||
lua_pushinteger(L, err);
|
||||
|
@ -748,8 +751,8 @@ static int tolua_unit_get_items(lua_State * L)
|
|||
static int tolua_unit_get_spells(lua_State * L)
|
||||
{
|
||||
unit *self = (unit *) tolua_tousertype(L, 1, 0);
|
||||
sc_mage *mage = self ? get_mage_depr(self) : 0;
|
||||
spellbook *sb = mage ? mage->spellbook : 0;
|
||||
struct sc_mage *mage = self ? get_mage(self) : NULL;
|
||||
spellbook *sb = mage_get_spellbook(mage);
|
||||
selist *slist = 0;
|
||||
if (sb) {
|
||||
selist **slist_ptr = &sb->spells;
|
||||
|
|
|
@ -880,7 +880,7 @@ void cr_output_unit(stream *out, const faction * f,
|
|||
const char *xc;
|
||||
const char *c;
|
||||
int i;
|
||||
sc_mage *mage;
|
||||
struct sc_mage *mage;
|
||||
|
||||
i = ualias(u);
|
||||
if (i > 0)
|
||||
|
@ -957,19 +957,23 @@ void cr_output_unit(stream *out, const faction * f,
|
|||
}
|
||||
|
||||
/* spells that this unit can cast */
|
||||
mage = get_mage_depr(u);
|
||||
mage = get_mage(u);
|
||||
if (mage) {
|
||||
int maxlevel = effskill(u, SK_MAGIC, 0);
|
||||
cr_output_spells(out, u, maxlevel);
|
||||
|
||||
for (i = 0; i != MAXCOMBATSPELLS; ++i) {
|
||||
const spell *sp = mage->combatspells[i].sp;
|
||||
int level;
|
||||
const spell *sp = mage_get_combatspell(mage, i, &level);
|
||||
if (sp) {
|
||||
const char *name =
|
||||
translate(mkname("spell", sp->sname), spell_name(sp, lang));
|
||||
const char *name;
|
||||
if (level > maxlevel) {
|
||||
level = maxlevel;
|
||||
}
|
||||
stream_printf(out, "KAMPFZAUBER %d\n", i);
|
||||
name = translate(mkname("spell", sp->sname), spell_name(sp, lang));
|
||||
stream_printf(out, "\"%s\";name\n", name);
|
||||
stream_printf(out, "%d;level\n", mage->combatspells[i].level);
|
||||
stream_printf(out, "%d;level\n", level);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -587,14 +587,12 @@ void give_unit(unit * u, unit * u2, order * ord)
|
|||
}
|
||||
}
|
||||
if (has_skill(u, SK_MAGIC)) {
|
||||
sc_mage *mage;
|
||||
if (count_skill(u2->faction, SK_MAGIC) + u->number >
|
||||
skill_limit(u2->faction, SK_MAGIC)) {
|
||||
cmistake(u, ord, 155, MSG_COMMERCE);
|
||||
return;
|
||||
}
|
||||
mage = get_mage_depr(u);
|
||||
if (!mage || u2->faction->magiegebiet != mage->magietyp) {
|
||||
if (u2->faction->magiegebiet != unit_get_magic(u)) {
|
||||
cmistake(u, ord, 157, MSG_COMMERCE);
|
||||
return;
|
||||
}
|
||||
|
|
|
@ -1294,11 +1294,19 @@ ship *read_ship(gamedata *data)
|
|||
}
|
||||
|
||||
static void fix_fam_mage(unit *u) {
|
||||
sc_mage *m = get_mage(u);
|
||||
if (m && m->magietyp != M_GRAY) {
|
||||
struct sc_mage *mage = get_mage(u);
|
||||
magic_t mtype = mage_get_type(mage);
|
||||
if (mtype != M_GRAY) {
|
||||
int skill = get_level(u, SK_MAGIC);
|
||||
/* unit should be a familiar that has aura and a spell-list */
|
||||
if (!m->spellbook) {
|
||||
m->magietyp = M_GRAY;
|
||||
if (skill > 0) {
|
||||
struct spellbook * sb = mage_get_spellbook(mage);
|
||||
if (!sb) {
|
||||
unit_set_magic(u, M_GRAY);
|
||||
}
|
||||
}
|
||||
else {
|
||||
a_removeall(&u->attribs, &at_mage);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1507,20 +1515,18 @@ int read_game(gamedata *data)
|
|||
assert(f->units);
|
||||
for (u = f->units; u; u = u->nextF) {
|
||||
if (data->version < SPELL_LEVEL_VERSION) {
|
||||
sc_mage *mage = get_mage_depr(u);
|
||||
struct sc_mage *mage = get_mage(u);
|
||||
if (mage) {
|
||||
faction *f = u->faction;
|
||||
int skl = effskill(u, SK_MAGIC, 0);
|
||||
if (f->magiegebiet == M_GRAY) {
|
||||
log_error("faction %s had magic=gray, fixing (%s)", factionname(f), magic_school[mage->magietyp]);
|
||||
f->magiegebiet = mage->magietyp;
|
||||
f->magiegebiet = mage_get_type(mage);
|
||||
log_error("faction %s had magic=gray, fixing (%s)",
|
||||
factionname(f), magic_school[f->magiegebiet]);
|
||||
}
|
||||
if (f->max_spelllevel < skl) {
|
||||
f->max_spelllevel = skl;
|
||||
}
|
||||
if (mage->spellcount < 0) {
|
||||
mage->spellcount = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (u->number > 0) {
|
||||
|
|
|
@ -117,7 +117,7 @@ enum {
|
|||
MAXOPTIONS
|
||||
};
|
||||
|
||||
typedef enum {
|
||||
typedef enum magic_t {
|
||||
M_GRAY = 0, /* Gray */
|
||||
M_ILLAUN = 1, /* Illaun */
|
||||
M_TYBIED = 2, /* Tybied */
|
||||
|
|
|
@ -1657,34 +1657,6 @@ void u_setrace(struct unit *u, const struct race *rc)
|
|||
}
|
||||
}
|
||||
|
||||
void unit_add_spell(unit * u, sc_mage * m, struct spell * sp, int level)
|
||||
{
|
||||
sc_mage *mage = m ? m : get_mage_depr(u);
|
||||
|
||||
if (!mage) {
|
||||
log_debug("adding new spell %s to a previously non-mage unit %s\n", sp->sname, unitname(u));
|
||||
mage = create_mage(u, u->faction ? u->faction->magiegebiet : M_GRAY);
|
||||
}
|
||||
if (!mage->spellbook) {
|
||||
mage->spellbook = create_spellbook(0);
|
||||
}
|
||||
spellbook_add(mage->spellbook, sp, level);
|
||||
}
|
||||
|
||||
struct spellbook * unit_get_spellbook(const struct unit * u)
|
||||
{
|
||||
sc_mage * mage = get_mage_depr(u);
|
||||
if (mage) {
|
||||
if (mage->spellbook) {
|
||||
return mage->spellbook;
|
||||
}
|
||||
if (mage->magietyp != M_GRAY) {
|
||||
return faction_get_spellbook(u->faction);
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int effskill(const unit * u, skill_t sk, const region *r)
|
||||
{
|
||||
assert(u);
|
||||
|
|
|
@ -228,8 +228,6 @@ extern "C" {
|
|||
int unit_max_hp(const struct unit *u);
|
||||
void scale_number(struct unit *u, int n);
|
||||
|
||||
struct spellbook * unit_get_spellbook(const struct unit * u);
|
||||
void unit_add_spell(struct unit * u, struct sc_mage * m, struct spell * sp, int level);
|
||||
void remove_empty_units_in_region(struct region * r);
|
||||
void remove_empty_units(void);
|
||||
|
||||
|
|
35
src/laws.c
35
src/laws.c
|
@ -2419,7 +2419,22 @@ int combatspell_cmd(unit * u, struct order *ord)
|
|||
}
|
||||
else {
|
||||
/* KAMPFZAUBER "<Spruchname>" setzt diesen Kampfzauber */
|
||||
set_combatspell(u, sp, ord, level);
|
||||
/* knowsspell prüft auf ist_magier, ist_spruch, kennt_spruch */
|
||||
if (!knowsspell(u->region, u, sp)) {
|
||||
/* Fehler 'Spell not found' */
|
||||
cmistake(u, ord, 173, MSG_MAGIC);
|
||||
}
|
||||
else if (!u_hasspell(u, sp)) {
|
||||
/* Diesen Zauber kennt die Einheit nicht */
|
||||
cmistake(u, ord, 169, MSG_MAGIC);
|
||||
}
|
||||
else if (!(sp->sptyp & ISCOMBATSPELL)) {
|
||||
/* Diesen Kampfzauber gibt es nicht */
|
||||
cmistake(u, ord, 171, MSG_MAGIC);
|
||||
}
|
||||
else {
|
||||
set_combatspell(u, sp, ord, level);
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
@ -3169,8 +3184,8 @@ static int faction_getmages(faction * f, unit ** results, int numresults)
|
|||
|
||||
for (u = f->units; u; u = u->nextF) {
|
||||
if (u->number > 0) {
|
||||
sc_mage *mage = get_mage_depr(u);
|
||||
if (mage) {
|
||||
struct sc_mage * mage = get_mage(u);
|
||||
if (mage && mage_get_spellbook(mage)) {
|
||||
int level = effskill(u, SK_MAGIC, 0);
|
||||
if (level > maxlevel) {
|
||||
maxlevel = level;
|
||||
|
@ -3229,11 +3244,12 @@ static void update_spells(void)
|
|||
unit *mages[MAXMAGES];
|
||||
int i;
|
||||
int maxlevel = faction_getmages(f, mages, MAXMAGES);
|
||||
struct spellbook *fsb;
|
||||
|
||||
if (maxlevel && FactionSpells()) {
|
||||
spellbook * book = get_spellbook(magic_school[f->magiegebiet]);
|
||||
if (!f->spellbook) {
|
||||
f->spellbook = create_spellbook(0);
|
||||
f->spellbook = create_spellbook(NULL);
|
||||
}
|
||||
copy_spells(book, f->spellbook, maxlevel);
|
||||
if (maxlevel > f->max_spelllevel) {
|
||||
|
@ -3241,13 +3257,14 @@ static void update_spells(void)
|
|||
pick_random_spells(f, maxlevel, common_spells, COMMONSPELLS);
|
||||
}
|
||||
}
|
||||
show_new_spells(f, maxlevel, faction_get_spellbook(f));
|
||||
fsb = faction_get_spellbook(f);
|
||||
show_new_spells(f, maxlevel, fsb);
|
||||
for (i = 0; i != MAXMAGES && mages[i]; ++i) {
|
||||
unit * u = mages[i];
|
||||
sc_mage *mage = get_mage_depr(u);
|
||||
if (mage && mage->spellbook) {
|
||||
int level = effskill(u, SK_MAGIC, 0);
|
||||
show_new_spells(f, level, mage->spellbook);
|
||||
spellbook *sb = unit_get_spellbook(u);
|
||||
if (sb != fsb) {
|
||||
int level = effskill(u, SK_MAGIC, NULL);
|
||||
show_new_spells(f, level, sb);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
263
src/magic.c
263
src/magic.c
|
@ -103,6 +103,78 @@ const char *magic_school[MAXMAGIETYP] = {
|
|||
"common"
|
||||
};
|
||||
|
||||
struct combatspell {
|
||||
int level;
|
||||
const struct spell *sp;
|
||||
};
|
||||
|
||||
typedef struct sc_mage {
|
||||
magic_t magietyp;
|
||||
int spellpoints;
|
||||
int spchange;
|
||||
int spellcount;
|
||||
struct combatspell combatspells[MAXCOMBATSPELLS];
|
||||
struct spellbook *spellbook;
|
||||
} sc_mage;
|
||||
|
||||
int mage_get_spellpoints(const sc_mage *m)
|
||||
{
|
||||
return m ? m->spellpoints : 0;
|
||||
}
|
||||
|
||||
int mage_change_spellpoints(sc_mage *m, int delta)
|
||||
{
|
||||
if (m) {
|
||||
int val = m->spellpoints + delta;
|
||||
return m->spellpoints = (val > 0) ? val : m->spellpoints;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
magic_t mage_get_type(const sc_mage *m)
|
||||
{
|
||||
return m ? m->magietyp : M_GRAY;
|
||||
}
|
||||
|
||||
const spell *mage_get_combatspell(const sc_mage *mage, int nr, int *level)
|
||||
{
|
||||
assert(nr < MAXCOMBATSPELLS);
|
||||
if (mage) {
|
||||
if (level) {
|
||||
*level = mage->combatspells[nr].level;
|
||||
}
|
||||
return mage->combatspells[nr].sp;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
void unit_set_magic(struct unit *u, enum magic_t mtype)
|
||||
{
|
||||
sc_mage *mage = get_mage(u);
|
||||
if (mage) {
|
||||
mage->magietyp = mtype;
|
||||
}
|
||||
}
|
||||
|
||||
magic_t unit_get_magic(const unit *u)
|
||||
{
|
||||
return mage_get_type(get_mage(u));
|
||||
}
|
||||
|
||||
void unit_add_spell(unit * u, struct spell * sp, int level)
|
||||
{
|
||||
sc_mage *mage = get_mage(u);
|
||||
|
||||
if (!mage) {
|
||||
log_debug("adding new spell %s to a previously non-mage unit %s\n", sp->sname, unitname(u));
|
||||
mage = create_mage(u, u->faction ? u->faction->magiegebiet : M_GRAY);
|
||||
}
|
||||
if (!mage->spellbook) {
|
||||
mage->spellbook = create_spellbook(0);
|
||||
}
|
||||
spellbook_add(mage->spellbook, sp, level);
|
||||
}
|
||||
|
||||
/**
|
||||
** at_icastle
|
||||
** TODO: separate castle-appearance from illusion-effects
|
||||
|
@ -199,6 +271,23 @@ const building_type *icastle_type(const struct attrib *a) {
|
|||
|
||||
extern int dice(int count, int value);
|
||||
|
||||
bool FactionSpells(void)
|
||||
{
|
||||
static int config, rule;
|
||||
if (config_changed(&config)) {
|
||||
rule = config_get_int("rules.magic.factionlist", 0);
|
||||
}
|
||||
return rule != 0;
|
||||
}
|
||||
|
||||
int get_spell_level_mage(const spell * sp, void * cbdata)
|
||||
{
|
||||
sc_mage *mage = (sc_mage *)cbdata;
|
||||
spellbook *book = get_spellbook(magic_school[mage->magietyp]);
|
||||
spellbook_entry *sbe = spellbook_get(book, sp);
|
||||
return sbe ? sbe->level : 0;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------- */
|
||||
/* aus dem alten System übriggebliegene Funktionen, die bei der
|
||||
* Umwandlung von alt nach neu gebraucht werden */
|
||||
|
@ -219,23 +308,6 @@ static void free_mage(variant *var)
|
|||
free(mage);
|
||||
}
|
||||
|
||||
bool FactionSpells(void)
|
||||
{
|
||||
static int config, rule;
|
||||
if (config_changed(&config)) {
|
||||
rule = config_get_int("rules.magic.factionlist", 0);
|
||||
}
|
||||
return rule != 0;
|
||||
}
|
||||
|
||||
int get_spell_level_mage(const spell * sp, void * cbdata)
|
||||
{
|
||||
sc_mage *mage = (sc_mage *)cbdata;
|
||||
spellbook *book = get_spellbook(magic_school[mage->magietyp]);
|
||||
spellbook_entry *sbe = spellbook_get(book, sp);
|
||||
return sbe ? sbe->level : 0;
|
||||
}
|
||||
|
||||
static int read_mage(variant *var, void *owner, struct gamedata *data)
|
||||
{
|
||||
storage *store = data->store;
|
||||
|
@ -312,9 +384,10 @@ attrib_type at_mage = {
|
|||
ATF_UNIQUE
|
||||
};
|
||||
|
||||
bool is_mage(const unit * u)
|
||||
bool is_mage(const struct unit * u)
|
||||
{
|
||||
return get_mage_depr(u) != NULL;
|
||||
sc_mage *m = get_mage(u);
|
||||
return (m && m->magietyp != M_GRAY);
|
||||
}
|
||||
|
||||
sc_mage *get_mage(const unit * u)
|
||||
|
@ -326,12 +399,22 @@ sc_mage *get_mage(const unit * u)
|
|||
return NULL;
|
||||
}
|
||||
|
||||
sc_mage *get_mage_depr(const unit * u)
|
||||
struct spellbook * mage_get_spellbook(const struct sc_mage * mage) {
|
||||
if (mage) {
|
||||
return mage->spellbook;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
struct spellbook * unit_get_spellbook(const struct unit * u)
|
||||
{
|
||||
if (has_skill(u, SK_MAGIC)) {
|
||||
attrib *a = a_find(u->attribs, &at_mage);
|
||||
if (a) {
|
||||
return (sc_mage *)a->data.v;
|
||||
sc_mage * mage = get_mage(u);
|
||||
if (mage) {
|
||||
if (mage->spellbook) {
|
||||
return mage->spellbook;
|
||||
}
|
||||
if (mage->magietyp != M_GRAY) {
|
||||
return faction_get_spellbook(u->faction);
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
|
@ -436,17 +519,15 @@ int u_hasspell(const unit *u, const struct spell *sp)
|
|||
|
||||
int get_combatspelllevel(const unit * u, int nr)
|
||||
{
|
||||
sc_mage *m = get_mage_depr(u);
|
||||
|
||||
assert(nr < MAXCOMBATSPELLS);
|
||||
if (m) {
|
||||
int level = effskill(u, SK_MAGIC, 0);
|
||||
if (level < m->combatspells[nr].level) {
|
||||
return level;
|
||||
int level;
|
||||
if (mage_get_combatspell(get_mage(u), nr, &level) != NULL) {
|
||||
int maxlevel = effskill(u, SK_MAGIC, 0);
|
||||
if (level > maxlevel) {
|
||||
return maxlevel;
|
||||
}
|
||||
return m->combatspells[nr].level;
|
||||
return level;
|
||||
}
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------- */
|
||||
|
@ -454,40 +535,15 @@ int get_combatspelllevel(const unit * u, int nr)
|
|||
|
||||
const spell *get_combatspell(const unit * u, int nr)
|
||||
{
|
||||
sc_mage *m;
|
||||
|
||||
assert(nr < MAXCOMBATSPELLS);
|
||||
m = get_mage_depr(u);
|
||||
if (m) {
|
||||
return m->combatspells[nr].sp;
|
||||
}
|
||||
return NULL;
|
||||
return mage_get_combatspell(get_mage(u), nr, NULL);
|
||||
}
|
||||
|
||||
void set_combatspell(unit * u, spell * sp, struct order *ord, int level)
|
||||
{
|
||||
sc_mage *mage = get_mage_depr(u);
|
||||
sc_mage *mage = get_mage(u);
|
||||
int i = -1;
|
||||
|
||||
assert(mage || !"trying to set a combat spell for non-mage");
|
||||
|
||||
/* knowsspell prüft auf ist_magier, ist_spruch, kennt_spruch */
|
||||
if (!knowsspell(u->region, u, sp)) {
|
||||
/* Fehler 'Spell not found' */
|
||||
cmistake(u, ord, 173, MSG_MAGIC);
|
||||
return;
|
||||
}
|
||||
if (!u_hasspell(u, sp)) {
|
||||
/* Diesen Zauber kennt die Einheit nicht */
|
||||
cmistake(u, ord, 169, MSG_MAGIC);
|
||||
return;
|
||||
}
|
||||
if (!(sp->sptyp & ISCOMBATSPELL)) {
|
||||
/* Diesen Kampfzauber gibt es nicht */
|
||||
cmistake(u, ord, 171, MSG_MAGIC);
|
||||
return;
|
||||
}
|
||||
|
||||
if (sp->sptyp & PRECOMBATSPELL)
|
||||
i = 0;
|
||||
else if (sp->sptyp & COMBATSPELL)
|
||||
|
@ -502,13 +558,12 @@ void set_combatspell(unit * u, spell * sp, struct order *ord, int level)
|
|||
|
||||
void unset_combatspell(unit * u, spell * sp)
|
||||
{
|
||||
sc_mage *m;
|
||||
int nr = 0;
|
||||
sc_mage *m = get_mage(u);
|
||||
|
||||
m = get_mage_depr(u);
|
||||
if (!m)
|
||||
if (!m) {
|
||||
return;
|
||||
|
||||
}
|
||||
if (!sp) {
|
||||
int i;
|
||||
for (i = 0; i < MAXCOMBATSPELLS; i++) {
|
||||
|
@ -540,26 +595,15 @@ void unset_combatspell(unit * u, spell * sp)
|
|||
/* Gibt die aktuelle Anzahl der Magiepunkte der Einheit zurück */
|
||||
int get_spellpoints(const unit * u)
|
||||
{
|
||||
sc_mage *m;
|
||||
|
||||
m = get_mage_depr(u);
|
||||
if (!m)
|
||||
return 0;
|
||||
|
||||
return m->spellpoints;
|
||||
return mage_get_spellpoints(get_mage(u));
|
||||
}
|
||||
|
||||
void set_spellpoints(unit * u, int sp)
|
||||
{
|
||||
sc_mage *m;
|
||||
|
||||
m = get_mage_depr(u);
|
||||
if (!m)
|
||||
return;
|
||||
|
||||
m->spellpoints = sp;
|
||||
|
||||
return;
|
||||
sc_mage *m = get_mage(u);
|
||||
if (m) {
|
||||
m->spellpoints = sp;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -567,23 +611,7 @@ void set_spellpoints(unit * u, int sp)
|
|||
*/
|
||||
int change_spellpoints(unit * u, int mp)
|
||||
{
|
||||
sc_mage *m;
|
||||
int sp;
|
||||
|
||||
m = get_mage_depr(u);
|
||||
if (!m) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* verhindere negative Magiepunkte */
|
||||
sp = m->spellpoints + mp;
|
||||
if (sp > 0) {
|
||||
m->spellpoints = sp;
|
||||
}
|
||||
else {
|
||||
m->spellpoints = 0;
|
||||
}
|
||||
return sp;
|
||||
return mage_change_spellpoints(get_mage(u), mp);
|
||||
}
|
||||
|
||||
/* bietet die Möglichkeit, die maximale Anzahl der Magiepunkte mit
|
||||
|
@ -593,11 +621,8 @@ static int get_spchange(const unit * u)
|
|||
{
|
||||
sc_mage *m;
|
||||
|
||||
m = get_mage_depr(u);
|
||||
if (!m)
|
||||
return 0;
|
||||
|
||||
return m->spchange;
|
||||
m = get_mage(u);
|
||||
return m ? m->spchange : 0;
|
||||
}
|
||||
|
||||
/* ein Magier kann normalerweise maximal Stufe^2.1/1.2+1 Magiepunkte
|
||||
|
@ -645,9 +670,7 @@ int max_spellpoints(const region * r, const unit * u)
|
|||
|
||||
int change_maxspellpoints(unit * u, int csp)
|
||||
{
|
||||
sc_mage *m;
|
||||
|
||||
m = get_mage_depr(u);
|
||||
sc_mage *m = get_mage(u);
|
||||
if (!m) {
|
||||
return 0;
|
||||
}
|
||||
|
@ -660,30 +683,24 @@ int change_maxspellpoints(unit * u, int csp)
|
|||
*/
|
||||
int countspells(unit * u, int step)
|
||||
{
|
||||
sc_mage *m;
|
||||
int count;
|
||||
|
||||
m = get_mage_depr(u);
|
||||
if (!m) {
|
||||
return 0;
|
||||
}
|
||||
if (step == 0) {
|
||||
return m->spellcount;
|
||||
}
|
||||
count = m->spellcount + step;
|
||||
|
||||
m->spellcount = (count > 0) ? count : 0;
|
||||
return m->spellcount;
|
||||
}
|
||||
|
||||
int spellcount(const unit *u) {
|
||||
sc_mage *m = get_mage_depr(u);
|
||||
sc_mage * m = get_mage(u);
|
||||
if (m) {
|
||||
int count;
|
||||
if (step == 0) {
|
||||
return m->spellcount;
|
||||
}
|
||||
count = m->spellcount + step;
|
||||
m->spellcount = (count > 0) ? count : 0;
|
||||
return m->spellcount;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int spellcount(const unit *u) {
|
||||
sc_mage *m = get_mage(u);
|
||||
return m ? m->spellcount : 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Die Grundkosten pro Stufe werden um 2^count erhöht. countspells(u)
|
||||
* ist dabei die Anzahl der bereits gezauberten Sprüche
|
||||
|
|
40
src/magic.h
40
src/magic.h
|
@ -27,18 +27,16 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|||
extern "C" {
|
||||
#endif
|
||||
|
||||
/* ------------------------------------------------------------- */
|
||||
|
||||
#define MAXCOMBATSPELLS 3 /* PRECOMBAT COMBAT POSTCOMBAT */
|
||||
#define MAX_SPELLRANK 9 /* Standard-Rank 5 */
|
||||
#define MAXINGREDIENT 5 /* bis zu 5 Komponenten pro Zauber */
|
||||
#define CHAOSPATZERCHANCE 10 /* +10% Chance zu Patzern */
|
||||
|
||||
/* ------------------------------------------------------------- */
|
||||
|
||||
#define IRONGOLEM_CRUMBLE 15 /* monatlich Chance zu zerfallen */
|
||||
#define STONEGOLEM_CRUMBLE 10 /* monatlich Chance zu zerfallen */
|
||||
|
||||
struct sc_mage;
|
||||
struct unit;
|
||||
|
||||
extern const char *magic_school[MAXMAGIETYP];
|
||||
extern struct attrib_type at_familiar;
|
||||
extern struct attrib_type at_familiarmage;
|
||||
|
@ -99,26 +97,12 @@ extern "C" {
|
|||
* - Spruchliste
|
||||
*/
|
||||
|
||||
typedef struct combatspell {
|
||||
int level;
|
||||
const struct spell *sp;
|
||||
} combatspell;
|
||||
|
||||
typedef struct spell_names {
|
||||
struct spell_names *next;
|
||||
const struct locale *lang;
|
||||
void * tokens;
|
||||
} spell_names;
|
||||
|
||||
typedef struct sc_mage {
|
||||
magic_t magietyp;
|
||||
int spellpoints;
|
||||
int spchange;
|
||||
int spellcount;
|
||||
combatspell combatspells[MAXCOMBATSPELLS];
|
||||
struct spellbook *spellbook;
|
||||
} sc_mage;
|
||||
|
||||
/* ------------------------------------------------------------- */
|
||||
/* Zauberliste */
|
||||
|
||||
|
@ -217,12 +201,22 @@ extern "C" {
|
|||
*/
|
||||
|
||||
/* Magier */
|
||||
sc_mage *create_mage(struct unit *u, magic_t mtyp);
|
||||
struct sc_mage *create_mage(struct unit *u, magic_t mtyp);
|
||||
/* macht die struct unit zu einem neuen Magier: legt die struct u->mage an
|
||||
* und initialisiert den Magiertypus mit mtyp. */
|
||||
sc_mage *get_mage(const struct unit *u);
|
||||
sc_mage *get_mage_depr(const struct unit *u);
|
||||
/* gibt u->mage zurück, bei nicht-Magiern *NULL */
|
||||
struct sc_mage *get_mage(const struct unit *u);
|
||||
|
||||
enum magic_t mage_get_type(const struct sc_mage *mage);
|
||||
const struct spell *mage_get_combatspell(const struct sc_mage *mage, int nr, int *level);
|
||||
struct spellbook * mage_get_spellbook(const struct sc_mage * mage);
|
||||
int mage_get_spellpoints(const struct sc_mage *m);
|
||||
int mage_change_spellpoints(struct sc_mage *m, int delta);
|
||||
|
||||
enum magic_t unit_get_magic(const struct unit *u);
|
||||
void unit_set_magic(struct unit *u, enum magic_t mtype);
|
||||
struct spellbook * unit_get_spellbook(const struct unit * u);
|
||||
void unit_add_spell(struct unit * u, struct spell * sp, int level);
|
||||
|
||||
bool is_mage(const struct unit *u);
|
||||
/* gibt true, wenn u->mage gesetzt. */
|
||||
bool is_familiar(const struct unit *u);
|
||||
|
|
|
@ -117,7 +117,7 @@ void test_pay_spell(CuTest * tc)
|
|||
CuAssertPtrNotNull(tc, sp);
|
||||
|
||||
set_level(u, SK_MAGIC, 5);
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
|
||||
change_resource(u, get_resourcetype(R_SILVER), 1);
|
||||
change_resource(u, get_resourcetype(R_AURA), 3);
|
||||
|
@ -151,7 +151,7 @@ void test_pay_spell_failure(CuTest * tc)
|
|||
CuAssertPtrNotNull(tc, sp);
|
||||
|
||||
set_level(u, SK_MAGIC, 5);
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
|
||||
CuAssertIntEquals(tc, 1, change_resource(u, get_resourcetype(R_SILVER), 1));
|
||||
CuAssertIntEquals(tc, 2, change_resource(u, get_resourcetype(R_AURA), 2));
|
||||
|
@ -193,7 +193,7 @@ void test_getspell_unit(CuTest * tc)
|
|||
|
||||
CuAssertPtrEquals(tc, NULL, unit_getspell(u, "Herp-a-derp", lang));
|
||||
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
CuAssertPtrNotNull(tc, unit_getspell(u, "Herp-a-derp", lang));
|
||||
test_teardown();
|
||||
}
|
||||
|
@ -277,7 +277,7 @@ void test_set_pre_combatspell(CuTest * tc)
|
|||
sp = create_spell("testspell");
|
||||
sp->sptyp |= PRECOMBATSPELL;
|
||||
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
|
||||
set_combatspell(u, sp, 0, 2);
|
||||
CuAssertPtrEquals(tc, sp, (spell *)get_combatspell(u, index));
|
||||
|
@ -309,7 +309,7 @@ void test_set_main_combatspell(CuTest * tc)
|
|||
sp = create_spell("testspell");
|
||||
sp->sptyp |= COMBATSPELL;
|
||||
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
|
||||
set_combatspell(u, sp, 0, 2);
|
||||
CuAssertPtrEquals(tc, sp, (spell *)get_combatspell(u, index));
|
||||
|
@ -341,7 +341,7 @@ void test_set_post_combatspell(CuTest * tc)
|
|||
sp = create_spell("testspell");
|
||||
sp->sptyp |= POSTCOMBATSPELL;
|
||||
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
|
||||
set_combatspell(u, sp, 0, 2);
|
||||
CuAssertPtrEquals(tc, sp, (spell *)get_combatspell(u, index));
|
||||
|
@ -371,7 +371,7 @@ void test_hasspell(CuTest * tc)
|
|||
sp = create_spell("testspell");
|
||||
sp->sptyp |= POSTCOMBATSPELL;
|
||||
|
||||
unit_add_spell(u, 0, sp, 2);
|
||||
unit_add_spell(u, sp, 2);
|
||||
|
||||
set_level(u, SK_MAGIC, 1);
|
||||
CuAssertTrue(tc, !u_hasspell(u, sp));
|
||||
|
@ -407,7 +407,7 @@ void test_multi_cast(CuTest *tc) {
|
|||
|
||||
u = test_create_unit(test_create_faction(NULL), test_create_region(0, 0, NULL));
|
||||
set_level(u, SK_MAGIC, 10);
|
||||
unit_add_spell(u, 0, sp, 1);
|
||||
unit_add_spell(u, sp, 1);
|
||||
CuAssertPtrEquals(tc, sp, unit_getspell(u, "Feuerball", lang));
|
||||
|
||||
unit_addorder(u, create_order(K_CAST, u->faction->locale, "Feuerball"));
|
||||
|
@ -485,7 +485,7 @@ static void test_illusioncastle(CuTest *tc)
|
|||
|
||||
static void test_is_mage(CuTest *tc) {
|
||||
unit *u;
|
||||
sc_mage *mage;
|
||||
struct sc_mage *mage;
|
||||
|
||||
test_setup();
|
||||
u = test_create_unit(test_create_faction(NULL), test_create_region(0, 0, NULL));
|
||||
|
@ -502,18 +502,13 @@ static void test_is_mage(CuTest *tc) {
|
|||
|
||||
static void test_get_mage(CuTest *tc) {
|
||||
unit *u;
|
||||
sc_mage *mage;
|
||||
struct sc_mage *mage;
|
||||
|
||||
test_setup();
|
||||
u = test_create_unit(test_create_faction(NULL), test_create_region(0, 0, NULL));
|
||||
CuAssertPtrEquals(tc, NULL, get_mage(u));
|
||||
CuAssertPtrEquals(tc, NULL, get_mage_depr(u));
|
||||
CuAssertPtrNotNull(tc, mage = create_mage(u, M_CERDDOR));
|
||||
CuAssertPtrEquals(tc, mage, get_mage(u));
|
||||
CuAssertPtrEquals(tc, NULL, get_mage_depr(u));
|
||||
set_level(u, SK_MAGIC, 1);
|
||||
CuAssertPtrEquals(tc, mage, get_mage(u));
|
||||
CuAssertPtrEquals(tc, mage, get_mage_depr(u));
|
||||
test_teardown();
|
||||
}
|
||||
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
/*
|
||||
Copyright (c) 1998-2014,
|
||||
Copyright (c) 1998-2018,
|
||||
Enno Rehling <enno@eressea.de>
|
||||
Katja Zedel <katze@felidae.kn-bremen.de
|
||||
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
||||
|
@ -1430,9 +1430,10 @@ int movement_speed(const unit * u)
|
|||
/* Im Astralraum sind Tyb und Ill-Magier doppelt so schnell.
|
||||
* Nicht kumulativ mit anderen Beschleunigungen! */
|
||||
if (mp * dk <= BP_WALKING * u_race(u)->speed && is_astral(u->region)) {
|
||||
sc_mage *mage = get_mage(u);
|
||||
if (mage && (mage->magietyp == M_TYBIED || mage->magietyp == M_ILLAUN)) {
|
||||
if (has_skill(u, SK_MAGIC)) {
|
||||
struct sc_mage *mage = get_mage(u);
|
||||
if (mage) {
|
||||
magic_t mtype = mage_get_type(mage);
|
||||
if (mtype == M_TYBIED || mtype == M_ILLAUN) {
|
||||
mp *= 2;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -637,10 +637,10 @@ static size_t spskill(char *buffer, size_t size, const struct locale * lang,
|
|||
bufp = STRLCPY(bufp, " ", size);
|
||||
|
||||
if (sv->id == SK_MAGIC) {
|
||||
sc_mage *mage = get_mage(u);
|
||||
if (mage && mage->magietyp != M_GRAY) {
|
||||
bufp = STRLCPY(bufp, LOC(lang, mkname("school",
|
||||
magic_school[mage->magietyp])), size);
|
||||
magic_t mtype = unit_get_magic(u);
|
||||
if (mtype != M_GRAY) {
|
||||
bufp = STRLCPY(bufp,
|
||||
LOC(lang, mkname("school", magic_school[mtype])), size);
|
||||
bufp = STRLCPY(bufp, " ", size);
|
||||
}
|
||||
}
|
||||
|
|
63
src/spells.c
63
src/spells.c
|
@ -704,13 +704,13 @@ static int sp_destroy_magic(castorder * co)
|
|||
|
||||
static int sp_transferaura(castorder * co)
|
||||
{
|
||||
int aura, gain, multi = 2;
|
||||
int aura, used, multi = 2;
|
||||
unit *caster = co_get_caster(co);
|
||||
unit *mage = co_get_magician(co);
|
||||
int cast_level = co->level;
|
||||
spellparameter *pa = co->par;
|
||||
unit *u;
|
||||
sc_mage *scm_dst, *scm_src = get_mage(mage);
|
||||
struct sc_mage *scm_dst, *scm_src = get_mage(mage);
|
||||
|
||||
assert(scm_src);
|
||||
/* wenn kein Ziel gefunden, Zauber abbrechen */
|
||||
|
@ -725,25 +725,29 @@ static int sp_transferaura(castorder * co)
|
|||
/* Wieviel Transferieren? */
|
||||
aura = pa->param[1]->data.i;
|
||||
u = pa->param[0]->data.u;
|
||||
scm_dst = get_mage_depr(u);
|
||||
scm_dst = get_mage(u);
|
||||
|
||||
if (scm_dst == NULL) {
|
||||
/* "Zu dieser Einheit kann ich keine Aura uebertragen." */
|
||||
cmistake(caster, co->order, 207, MSG_MAGIC);
|
||||
return 0;
|
||||
}
|
||||
else if (scm_src->magietyp == M_TYBIED) {
|
||||
if (scm_src->magietyp != scm_dst->magietyp)
|
||||
multi = 3;
|
||||
}
|
||||
else if (scm_src->magietyp == M_GRAY) {
|
||||
if (scm_src->magietyp != scm_dst->magietyp)
|
||||
multi = 4;
|
||||
}
|
||||
else if (scm_dst->magietyp != scm_src->magietyp) {
|
||||
/* "Zu dieser Einheit kann ich keine Aura uebertragen." */
|
||||
cmistake(caster, co->order, 207, MSG_MAGIC);
|
||||
return 0;
|
||||
else {
|
||||
magic_t src = mage_get_type(scm_src);
|
||||
magic_t dst = mage_get_type(scm_dst);
|
||||
if (src != dst) {
|
||||
if (src == M_TYBIED) {
|
||||
multi = 3;
|
||||
}
|
||||
else if (src == M_GRAY) {
|
||||
multi = 4;
|
||||
}
|
||||
else {
|
||||
/* "Zu dieser Einheit kann ich keine Aura uebertragen." */
|
||||
cmistake(caster, co->order, 207, MSG_MAGIC);
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (aura < multi) {
|
||||
|
@ -752,16 +756,15 @@ static int sp_transferaura(castorder * co)
|
|||
return 0;
|
||||
}
|
||||
|
||||
gain = scm_src->spellpoints;
|
||||
if (gain > aura) gain = aura;
|
||||
gain = gain / multi;
|
||||
scm_src->spellpoints -= gain * multi;
|
||||
scm_dst->spellpoints += gain;
|
||||
used = mage_get_spellpoints(scm_src);
|
||||
if (used > aura) used = aura;
|
||||
mage_change_spellpoints(scm_src, -used);
|
||||
mage_change_spellpoints(scm_dst, used / multi);
|
||||
|
||||
/* sprintf(buf, "%s transferiert %d Aura auf %s", unitname(mage),
|
||||
gain, unitname(u)); */
|
||||
ADDMSG(&caster->faction->msgs, msg_message("auratransfer_success",
|
||||
"unit target aura", caster, u, gain));
|
||||
"unit target aura", caster, u, used));
|
||||
return cast_level;
|
||||
}
|
||||
|
||||
|
@ -5748,14 +5751,6 @@ static int sp_eternizewall(castorder * co)
|
|||
return cast_level;
|
||||
}
|
||||
|
||||
static magic_t get_magic_type(const struct unit *u) {
|
||||
sc_mage *mage = get_mage(u);
|
||||
if (mage) {
|
||||
return mage->magietyp;
|
||||
}
|
||||
return M_GRAY;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------- */
|
||||
/* Name: Opfere Kraft
|
||||
* Stufe: 15
|
||||
|
@ -5810,7 +5805,7 @@ int sp_permtransfer(castorder * co)
|
|||
change_maxspellpoints(mage, -aura);
|
||||
change_spellpoints(mage, -aura);
|
||||
|
||||
if (get_magic_type(tu) == get_magic_type(mage)) {
|
||||
if (unit_get_magic(tu) == unit_get_magic(mage)) {
|
||||
change_maxspellpoints(tu, aura / 2);
|
||||
}
|
||||
else {
|
||||
|
@ -5925,6 +5920,7 @@ int sp_stealaura(castorder * co)
|
|||
int cast_level = co->level;
|
||||
double power = co->force;
|
||||
spellparameter *pa = co->par;
|
||||
struct sc_mage *scm;
|
||||
|
||||
/* wenn kein Ziel gefunden, Zauber abbrechen */
|
||||
if (pa->param[0]->flag == TARGET_NOTFOUND)
|
||||
|
@ -5933,17 +5929,18 @@ int sp_stealaura(castorder * co)
|
|||
/* Zieleinheit */
|
||||
u = pa->param[0]->data.u;
|
||||
|
||||
if (!get_mage_depr(u)) {
|
||||
scm = get_mage(u);
|
||||
if (!scm) {
|
||||
ADDMSG(&caster->faction->msgs, msg_message("stealaura_fail", "unit target",
|
||||
caster, u));
|
||||
ADDMSG(&u->faction->msgs, msg_message("stealaura_fail_detect", "unit", u));
|
||||
return 0;
|
||||
}
|
||||
|
||||
taura = (get_mage_depr(u)->spellpoints * (rng_int() % (int)(3 * power) + 1)) / 100;
|
||||
taura = (mage_get_spellpoints(scm) * (rng_int() % (int)(3 * power) + 1)) / 100;
|
||||
|
||||
if (taura > 0) {
|
||||
change_spellpoints(u, -taura);
|
||||
mage_change_spellpoints(scm, -taura);
|
||||
change_spellpoints(mage, taura);
|
||||
/* sprintf(buf, "%s entzieht %s %d Aura.", unitname(mage), unitname(u),
|
||||
taura); */
|
||||
|
|
|
@ -72,11 +72,11 @@ void spy_message(int spy, const unit * u, const unit * target)
|
|||
ADDMSG(&u->faction->msgs, msg_message("spyreport", "spy target status", u,
|
||||
target, status));
|
||||
if (spy > 20) {
|
||||
sc_mage *mage = get_mage_depr(target);
|
||||
magic_t mtype = unit_get_magic(target);
|
||||
/* for mages, spells and magic school */
|
||||
if (mage) {
|
||||
if (mtype != M_GRAY) {
|
||||
ADDMSG(&u->faction->msgs, msg_message("spyreport_mage", "spy target type", u,
|
||||
target, magic_school[mage->magietyp]));
|
||||
target, magic_school[mtype]));
|
||||
}
|
||||
}
|
||||
if (spy > 6) {
|
||||
|
|
17
src/study.c
17
src/study.c
|
@ -420,15 +420,16 @@ int teach_cmd(unit * teacher, struct order *ord)
|
|||
if (sk == SK_MAGIC) {
|
||||
/* ist der Magier schon spezialisiert, so versteht er nur noch
|
||||
* Lehrer seines Gebietes */
|
||||
sc_mage *mage1 = get_mage_depr(teacher);
|
||||
sc_mage *mage2 = get_mage_depr(scholar);
|
||||
if (mage2 && mage1 && mage2->magietyp != M_GRAY
|
||||
&& mage1->magietyp != mage2->magietyp) {
|
||||
if (feedback) {
|
||||
ADDMSG(&teacher->faction->msgs, msg_feedback(teacher, ord,
|
||||
"error_different_magic", "target", scholar));
|
||||
magic_t mage2 = unit_get_magic(scholar);
|
||||
if (mage2 != M_GRAY) {
|
||||
magic_t mage1 = unit_get_magic(teacher);
|
||||
if (mage1 != mage2) {
|
||||
if (feedback) {
|
||||
ADDMSG(&teacher->faction->msgs, msg_feedback(teacher, ord,
|
||||
"error_different_magic", "target", scholar));
|
||||
}
|
||||
continue;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
}
|
||||
sk_academy = sk;
|
||||
|
|
|
@ -429,9 +429,9 @@ static void test_study_magic(CuTest *tc) {
|
|||
CuAssertIntEquals(tc, 0, study_cmd(u, u->thisorder));
|
||||
CuAssertIntEquals(tc, M_GWYRRD, f->magiegebiet);
|
||||
CuAssertIntEquals(tc, 0, i_get(u->items, itype));
|
||||
CuAssertPtrNotNull(tc, get_mage_depr(u));
|
||||
CuAssertPtrNotNull(tc, get_mage(u));
|
||||
CuAssertPtrEquals(tc, NULL, test_find_messagetype(f->msgs, "error65"));
|
||||
CuAssertIntEquals(tc, M_GWYRRD, get_mage_depr(u)->magietyp);
|
||||
CuAssertIntEquals(tc, M_GWYRRD, unit_get_magic(u));
|
||||
|
||||
test_teardown();
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue