Merge pull request #937 from eressea/2737-spells-cost-and-familiars

2737 spells cost and familiars
This commit is contained in:
Enno Rehling 2021-04-08 21:47:28 +02:00 committed by GitHub
commit 14e2e96c73
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 208 additions and 99 deletions

View file

@ -7,6 +7,8 @@
- Magieresistenz: Einheiten widerstehen nicht Zaubern der eigenen Partei [2733]. - Magieresistenz: Einheiten widerstehen nicht Zaubern der eigenen Partei [2733].
- Zauberkosten steigen durch Ring der Macht nicht an. - Zauberkosten steigen durch Ring der Macht nicht an.
- Effektiv gezauberte Stufe von Zauber anhängig von Verfügbarkeit der Materialen. - Effektiv gezauberte Stufe von Zauber anhängig von Verfügbarkeit der Materialen.
- Ring der Macht und Steinkreis erhöhen nicht die Zauberkosten [2737].
- Limits für Vertrautenzauber korrekt implementiert.
# 3.27 # 3.27

View file

@ -136,11 +136,9 @@ function test_fleeing_units_can_be_transported()
u1.number = 100 u1.number = 100
u1:add_order("ATTACKIEREN " .. itoa36(u2.id)) u1:add_order("ATTACKIEREN " .. itoa36(u2.id))
u2.number = 100 u2.number = 100
u2.name = 'Passagier'
u2:add_order("FAHREN " .. itoa36(u3.id)) u2:add_order("FAHREN " .. itoa36(u3.id))
u2:add_order("KAEMPFE FLIEHE") u2:add_order("KAEMPFE FLIEHE")
u3.number = 100 u3.number = 100
u3.name = 'Transporter'
u3:add_order("KAEMPFE FLIEHE") u3:add_order("KAEMPFE FLIEHE")
u3:add_order("TRANSPORT " .. itoa36(u2.id)) u3:add_order("TRANSPORT " .. itoa36(u2.id))
u3:add_order("NACH O ") u3:add_order("NACH O ")
@ -642,7 +640,6 @@ function test_laen2()
u1:set_skill("mining", 15) u1:set_skill("mining", 15)
u1:clear_orders() u1:clear_orders()
u1:add_order("MACHEN Laen") u1:add_order("MACHEN Laen")
u1.name = "Laenmeister"
local b = building.create(r, "mine") local b = building.create(r, "mine")
b.size = 10 b.size = 10

View file

@ -115,7 +115,6 @@ function test_dwarf_bonus()
local u = unit.create(faction.create("dwarf"), r) local u = unit.create(faction.create("dwarf"), r)
assert_equal("dwarf", u.faction.race) assert_equal("dwarf", u.faction.race)
assert_equal("dwarf", u.race) assert_equal("dwarf", u.race)
u.faction.name = "Zwerge"
u.number = 10 u.number = 10
u:set_skill("mining", 1) u:set_skill("mining", 1)
u:add_order("MACHE EISEN") u:add_order("MACHE EISEN")

View file

@ -0,0 +1,84 @@
local tcname = 'tests.e2.familiars'
local lunit = require('lunit')
if _VERSION >= 'Lua 5.2' then
_ENV = module(tcname, 'seeall')
else
module(tcname, lunit.testcase, package.seeall)
end
function setup()
eressea.game.reset()
eressea.settings.set("nmr.removenewbie", "0")
eressea.settings.set("nmr.timeout", "0")
eressea.settings.set("NewbieImmunity", "0")
eressea.settings.set("rules.food.flags", "4")
eressea.settings.set("rules.peasants.growth.factor", "0")
eressea.settings.set("magic.resist.enable", "0")
eressea.settings.set("magic.fumble.enable", "0")
eressea.settings.set("magic.regeneration.enable", "0")
end
local function setup_familiars(f, r)
f.magic = 'gwyrrd'
local uf = unit.create(f, r)
uf.magic = 'gray'
local u = unit.create(f, r)
u.magic = 'gwyrrd'
u:set_skill('magic', 9)
u.familiar = uf
return u, uf
end
function test_moneyspell()
local r = region.create(0, 0, "plain")
local f = faction.create('human')
local um, uf = setup_familiars(f, r)
um.aura = 9
um:add_order('ZAUBERE STUFE 9 Viehheilung')
process_orders()
assert_equal(0, um.aura)
assert_equal(450, um:get_item('money'))
end
function test_moneyspell_through_familiar()
-- casting magician's spell with the familiar: double cost
local r = region.create(0, 0, "plain")
local f = faction.create('human')
local um, uf = setup_familiars(f, r)
um.aura = 12
uf:add_order('ZAUBERE STUFE 3 Viehheilung')
process_orders()
assert_equal(0, uf:get_skill('magic'))
assert_equal(12, um.aura) -- cannot cast, familiar needs magic skill
assert_equal(0, uf:get_item('money'))
assert_equal(0, um:get_item('money'))
uf:set_skill('magic', 2) -- can cast no higher than level 2
process_orders()
assert_equal(8, um.aura) -- double the cost
assert_equal(100, uf:get_item('money'))
assert_equal(0, um:get_item('money'))
um:set_skill('magic', 4) -- use at most half of skill
uf:set_skill('magic', 1) -- too low for level 2 spell, cast at level 1
process_orders()
assert_equal(6, um.aura) -- double cost of level 1
assert_equal(150, uf:get_item('money'))
assert_equal(0, um:get_item('money'))
end
function test_moneyspell_as_familiar()
-- familiar has the spell and has magic skills: regular spellcasting rules apply
local r = region.create(0, 0, "plain")
local f = faction.create('human')
local um, uf = setup_familiars(f, r)
um.aura = 9
uf.aura = 20
uf:set_skill('magic', 10)
uf:add_spell('earn_silver#gwyrrd')
uf:add_order('ZAUBERE STUFE 10 Viehheilung')
process_orders()
assert_equal(500, uf:get_item('money'))
assert_equal(10, uf.aura)
assert_equal(9, um.aura)
end

View file

@ -7,6 +7,7 @@ require 'tests.e2.carts'
require 'tests.e2.astral' require 'tests.e2.astral'
require 'tests.e2.spells' require 'tests.e2.spells'
require 'tests.e2.migration' require 'tests.e2.migration'
require 'tests.e2.familiars'
require 'tests.e2.e2features' require 'tests.e2.e2features'
require 'tests.e2.insects' require 'tests.e2.insects'
require 'tests.e2.production' require 'tests.e2.production'

View file

@ -22,7 +22,6 @@ function test_ship_requires_skill()
assert_not_nil(r2) assert_not_nil(r2)
local f = faction.create("human", "fake@eressea.de", "de") local f = faction.create("human", "fake@eressea.de", "de")
local u1 = unit.create(f, r1, 1) local u1 = unit.create(f, r1, 1)
u1.name = "fake"
u1.ship = ship.create(r1, "longboat") u1.ship = ship.create(r1, "longboat")
u1:clear_orders() u1:clear_orders()
u1:add_order("NACH O") u1:add_order("NACH O")

View file

@ -181,7 +181,7 @@ function test_familiar()
local f = faction.create("human") local f = faction.create("human")
local u = unit.create(f, r) local u = unit.create(f, r)
local uid = u.id local uid = u.id
u.name = 'Hodor' u.name = 'Bonzi'
u.magic = "gwyrrd" u.magic = "gwyrrd"
u.race = "elf" u.race = "elf"
u:set_skill("magic", 10) u:set_skill("magic", 10)
@ -192,7 +192,7 @@ function test_familiar()
process_orders() process_orders()
for u in r.units do for u in r.units do
if u.id ~= uid then if u.id ~= uid then
assert_equal('Vertrauter von Hodor (' .. itoa36(uid) ..')', u.name) assert_equal('Vertrauter von Bonzi (' .. itoa36(uid) ..')', u.name)
end end
end end
end end
@ -222,6 +222,8 @@ function test_bug_2480()
end end
function test_bug_2517() function test_bug_2517()
-- Magier macht lange Befehle, wenn sein Vertrauter
-- zaubert (auch wenn es nicht eigene Zauber sind).
local r = region.create(0, 0, "plain") local r = region.create(0, 0, "plain")
local f = faction.create("elf") local f = faction.create("elf")
local um = unit.create(f, r, 1) local um = unit.create(f, r, 1)
@ -242,12 +244,25 @@ function test_bug_2517()
assert_equal('gray', uf.magic) assert_equal('gray', uf.magic)
uf:add_order('LERNE Magie') uf:add_order('LERNE Magie')
um:clear_orders() um:clear_orders()
assert_equal(1, uf:get_skill('magic'))
um:add_order('ARBEITEN') um:add_order('ARBEITEN')
assert_equal(0, um:get_item('money'))
process_orders() process_orders()
assert_equal('gray', uf.magic) assert_equal('gray', uf.magic)
uf:add_order('ZAUBERE STUFE 1 Viehheilung') uf:add_order('ZAUBERE STUFE 1 Viehheilung')
um.aura = 10
uf.aura = 10
assert_equal(10, um:get_item('money')) -- langer Befehl wurde ausgefuehrt
process_orders() process_orders()
assert_equal(50, uf:get_item('money')) assert_equal(50, uf:get_item('money'))
assert_equal(20, um:get_item('money')) -- langer Befehl wurde ausgefuehrt
assert_equal(8, um.aura) -- kein eigener Zauber, Aura des Magiers
assert_equal(10, uf.aura)
uf:add_spell('earn_silver#gwyrrd') -- ins private spellbook aufnehmen
process_orders()
assert_equal(9, uf.aura) -- einfache Kosten, aus eigener Aura
assert_equal(8, um.aura) -- keine Kosten für den Magier
assert_equal(30, um:get_item('money')) -- langer Befehl wurde ausgefuehrt
end end
function test_familiar_school() function test_familiar_school()

View file

@ -45,7 +45,6 @@ function test_stealth_faction_on()
end end
function test_stealth_faction_other() function test_stealth_faction_other()
u.name = "Enno"
u:clear_orders() u:clear_orders()
u:add_order("TARNEN PARTEI NUMMER " .. itoa36(f.id)) u:add_order("TARNEN PARTEI NUMMER " .. itoa36(f.id))

View file

@ -256,7 +256,6 @@ function test_promote_after_recruit()
local r1 = region.create(0, 0, 'plain') local r1 = region.create(0, 0, 'plain')
local r2 = region.create(1, 0, 'plain') local r2 = region.create(1, 0, 'plain')
local u1 = unit.create(f, r1, 1) local u1 = unit.create(f, r1, 1)
u1.name = 'Xolgrim'
local u2 = unit.create(f, r2, 55) local u2 = unit.create(f, r2, 55)
u2:add_order('REKRUTIERE 1') u2:add_order('REKRUTIERE 1')
u1:add_order('BEFOERDERE') u1:add_order('BEFOERDERE')

View file

@ -94,7 +94,6 @@ function test_appeasement_can_move()
r2 = region.create(1, 0, 'plain') r2 = region.create(1, 0, 'plain')
u2 = unit.create(faction.create('human'), r1, 1) u2 = unit.create(faction.create('human'), r1, 1)
u2.race = 'elf' u2.race = 'elf'
u2.name = 'Angsthase'
u2.magic = 'gwyrrd' u2.magic = 'gwyrrd'
u2:set_skill('magic', 5) u2:set_skill('magic', 5)
u2.aura = 10 u2.aura = 10
@ -118,7 +117,6 @@ function test_appeasement_break_guard()
r2 = region.create(1, 0, 'plain') r2 = region.create(1, 0, 'plain')
u2 = unit.create(faction.create('human'), r1, 1) u2 = unit.create(faction.create('human'), r1, 1)
u2.race = 'elf' u2.race = 'elf'
u2.name = 'Angsthase'
u2.magic = 'gwyrrd' u2.magic = 'gwyrrd'
u2.guard = true u2.guard = true
u2.status = 1 u2.status = 1

View file

@ -599,13 +599,13 @@ static int tolua_unit_get_familiar(lua_State * L)
static int tolua_unit_set_familiar(lua_State * L) static int tolua_unit_set_familiar(lua_State * L)
{ {
unit *mag = (unit *)tolua_tousertype(L, 1, NULL); unit *u = (unit *)tolua_tousertype(L, 1, NULL);
unit *fam = (unit *)tolua_tousertype(L, 2, NULL); unit *fam = (unit *)tolua_tousertype(L, 2, NULL);
if (fam) { if (fam) {
set_familiar(mag, fam); set_familiar(u, fam);
} }
else { else {
remove_familiar(mag); remove_familiar(u);
} }
return 0; return 0;
} }

View file

@ -1165,9 +1165,9 @@ faction *read_faction(gamedata * data)
} }
read_allies(data, &f->allies); read_allies(data, &f->allies);
read_groups(data, f); read_groups(data, f);
f->spellbook = 0; f->spellbook = NULL;
if (data->version >= REGIONOWNER_VERSION) { if (data->version >= REGIONOWNER_VERSION) {
read_spellbook(FactionSpells() ? &f->spellbook : 0, data, get_spell_level_faction, (void *)f); read_spellbook(FactionSpells() ? &f->spellbook : NULL, data, get_spell_level_faction, (void *)f);
} }
return f; return f;
} }

View file

@ -267,16 +267,12 @@ bool FactionSpells(void)
return rule != 0; return rule != 0;
} }
int mage_get_spell_level(const sc_mage *mage, const spell *sp) {
spellbook *book = mage_get_spellbook(mage);
spellbook_entry *sbe = spellbook_get(book, sp);
return sbe ? sbe->level : 0;
}
int get_spell_level_mage(const spell * sp, void * cbdata) int get_spell_level_mage(const spell * sp, void * cbdata)
{ {
sc_mage *mage = (sc_mage *)cbdata; sc_mage *mage = (sc_mage *)cbdata;
return mage_get_spell_level(mage, sp); spellbook *book = mage_get_spellbook(mage);
spellbook_entry *sbe = spellbook_get(book, sp);
return sbe ? sbe->level : 0;
} }
/* ------------------------------------------------------------- */ /* ------------------------------------------------------------- */
@ -364,11 +360,18 @@ write_mage(const variant *var, const void *owner, struct storage *store)
write_spellbook(mage->spellbook, store); write_spellbook(mage->spellbook, store);
} }
static int reset_mage(attrib *a, void *owner) {
sc_mage *mage = (sc_mage *)a->data.v;
UNUSED_ARG(owner);
mage->spellcount = 0;
return 1;
}
attrib_type at_mage = { attrib_type at_mage = {
"mage", "mage",
init_mage, init_mage,
free_mage, free_mage,
NULL, reset_mage,
write_mage, write_mage,
read_mage, read_mage,
NULL, NULL,
@ -495,12 +498,21 @@ sc_mage *create_mage(unit * u, magic_t mtyp)
/* ------------------------------------------------------------- */ /* ------------------------------------------------------------- */
/* Funktionen fuer die Bearbeitung der List-of-known-spells */ /* Funktionen fuer die Bearbeitung der List-of-known-spells */
int unit_spell_level(const unit *u, const struct spell *sp)
{
spellbook *book = unit_get_spellbook(u);
spellbook_entry *sbe = book ? spellbook_get(book, sp) : NULL;
if (sbe) {
return sbe->level;
}
return 0;
}
bool u_hasspell(const unit *u, const struct spell *sp) bool u_hasspell(const unit *u, const struct spell *sp)
{ {
spellbook * book = unit_get_spellbook(u); int level = unit_spell_level(u, sp);
spellbook_entry * sbe = book ? spellbook_get(book, sp) : 0; if (level > 0) {
if (sbe) { return level <= effskill(u, SK_MAGIC, NULL);
return sbe->level <= effskill(u, SK_MAGIC, NULL);
} }
return false; return false;
} }
@ -819,10 +831,9 @@ void pay_spell(unit * mage, const unit *caster, const spell * sp, int cast_level
*/ */
bool knowsspell(const region * r, const unit * u, const spell * sp) bool knowsspell(const region * r, const unit * u, const spell * sp)
{ {
UNUSED_ARG(r); int level = unit_spell_level(u, sp);
assert(sp);
/* steht der Spruch in der Spruchliste? */ /* steht der Spruch in der Spruchliste? */
return u_hasspell(u, sp); return level > 0;
} }
/* Um einen Spruch zu beherrschen, muss der Magier die Stufe des /* Um einen Spruch zu beherrschen, muss der Magier die Stufe des
@ -2472,16 +2483,16 @@ static bool is_moving_ship(ship * sh)
return false; return false;
} }
static int default_spell_level(const sc_mage *mage, const spell *sp) { static int default_spell_level(const unit *u, const spell *sp) {
if (sp && sp->components) { if (sp && sp->components) {
const struct spell_component *spc; const struct spell_component *spc;
for (spc = sp->components; spc->type; ++spc) { for (spc = sp->components; spc->type; ++spc) {
if (spc->cost != SPC_FIX) { if (spc->cost != SPC_FIX) {
return -1; return 0;
} }
} }
} }
return mage_get_spell_level(mage, sp); return unit_spell_level(u, sp);
} }
#define MAX_PARAMETERS 48 #define MAX_PARAMETERS 48
@ -2496,7 +2507,7 @@ static castorder *cast_cmd(unit * u, order * ord)
spell *sp = NULL; spell *sp = NULL;
plane *pl; plane *pl;
spellparameter *args = NULL; spellparameter *args = NULL;
unit * mage = u; unit * mage = NULL;
param_t param; param_t param;
if (LongHunger(u)) { if (LongHunger(u)) {
@ -2546,40 +2557,50 @@ static castorder *cast_cmd(unit * u, order * ord)
return 0; return 0;
} }
/**
* skill = der maximale durch das Magietalent erlaubte Level.
* Entspricht dem Talent des Zaubernden, oder im Falle, dass ein
* Vertrauter einen Spruch seines Magiers zaubert, dessen halbes Talent.
*/
skill = effskill(u, SK_MAGIC, NULL);
sp = unit_getspell(u, s, u->faction->locale); sp = unit_getspell(u, s, u->faction->locale);
/* Vertraute koennen auch Zauber sprechen, die sie selbst nicht
* koennen. unit_getspell findet aber nur jene Sprueche, die /*
* die Einheit beherrscht. */ * u = Die Einheit, die den Befehl gegeben hat.
if (!sp && is_familiar(u)) { * mage = Die Einheit, deren Spruchliste und Aura benutzt wird.
*
* Vertraute koennen auch Zauber sprechen, die sie selbst nicht
* koennen. `unit_getspell` findet aber nur jene Sprueche, die
* die Einheit beherrscht. In diesem Fall ist `familiar` der Vertraute.
*/
if (sp) {
/* wir zaubern selbst */
mage = u;
}
else if (skill > 0) {
/* als Vertrauter suchen wir einen Spender-Magier mit dem Spruch */
mage = get_familiar_mage(u); mage = get_familiar_mage(u);
if (mage) { if (mage) {
familiar = u; int limit = effskill(mage, SK_MAGIC, NULL) / 2;
if (limit < skill) {
skill = limit;
}
sp = unit_getspell(mage, s, mage->faction->locale); sp = unit_getspell(mage, s, mage->faction->locale);
} if (sp->sptyp & NOTFAMILIARCAST) {
else { /* Fehler: "Diesen Spruch kann der Vertraute nicht zaubern" */
/* somehow, this familiar has no mage! */ cmistake(u, ord, 177, MSG_MAGIC);
log_error("cast_cmd: familiar %s is without a mage?\n", unitname(u)); return 0;
mage = u; }
familiar = u;
} }
} }
/* OBS: hier kein else! */
if (!sp) { if (!sp) {
/* Fehler 'Spell not found' */ /* Fehler 'Spell not found' */
cmistake(u, ord, 173, MSG_MAGIC); cmistake(u, ord, 173, MSG_MAGIC);
return 0; return 0;
} }
/* um testen auf spruchnamen zu unterbinden sollte vor allen
* fehlermeldungen die anzeigen das der magier diesen Spruch
* nur in diese Situation nicht anwenden kann, noch eine
* einfache Sicherheitspruefung kommen */
if (!knowsspell(r, u, sp)) {
/* vorsicht! u kann der familiar sein */
if (!familiar) {
cmistake(u, ord, 173, MSG_MAGIC);
return 0;
}
}
if (sp->sptyp & ISCOMBATSPELL) { if (sp->sptyp & ISCOMBATSPELL) {
/* Fehler: "Dieser Zauber ist nur im Kampf sinnvoll" */ /* Fehler: "Dieser Zauber ist nur im Kampf sinnvoll" */
cmistake(u, ord, 174, MSG_MAGIC); cmistake(u, ord, 174, MSG_MAGIC);
@ -2620,19 +2641,45 @@ static castorder *cast_cmd(unit * u, order * ord)
cmistake(u, ord, 176, MSG_MAGIC); cmistake(u, ord, 176, MSG_MAGIC);
return 0; return 0;
} }
if (range > 1024) { /* (2^10) weiter als 10 Regionen entfernt */ if (familiar) {
/* Magier zaubert durch Vertrauten: keine Fernzauber erlaubt */
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "familiar_farcast",
"mage", mage));
return 0;
}
if (range > 1024) {
/* (2^10) weiter als 10 Regionen entfernt */
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "spellfail::nocontact", ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "spellfail::nocontact",
"target", target_r)); "target", target_r));
return 0; return 0;
} }
} }
if (familiar) {
/*
* Magier zaubert durch Vertrauten: Doppelte Kosten.
*
* Das hier über range zu machen ist ein Hack, deshalb passiert es erst
* nach allen anderen Range-Checks.
*/
range *= 2;
}
skill = effskill(mage, SK_MAGIC, NULL); /**
* level = Die Stufe, auf der gezaubert werden soll.
* Kann nicht höher sein als das Talent des Zaubernden, oder im
* Falle, dass ein Vertrauter einen Spruch seines Magiers zaubert,
* nicht höher als dessen halbes Talent.
*/
if (level < 0) { if (level < 0) {
level = default_spell_level(get_mage(mage), sp); level = default_spell_level(mage, sp);
if (level < 0) { if (level <= 0) {
level = skill; level = skill;
} }
if (level > skill) {
/* die Einheit ist nicht erfahren genug fuer diesen Zauber */
cmistake(u, ord, 169, MSG_MAGIC);
return 0;
}
} }
else if (!(sp->sptyp & SPELLLEVEL)) { else if (!(sp->sptyp & SPELLLEVEL)) {
/* Stufenangabe bei nicht Stufenvariierbaren Spruechen abfangen */ /* Stufenangabe bei nicht Stufenvariierbaren Spruechen abfangen */
@ -2642,45 +2689,12 @@ static castorder *cast_cmd(unit * u, order * ord)
"mage region command", u, u->region, ord)); "mage region command", u, u->region, ord));
} }
} }
if (level > skill) { if (level > skill) {
/* die Einheit ist nicht erfahren genug fuer diesen Zauber */ /* STUFE kann nicht mehr als das erlaubte Maximum sein */
cmistake(u, ord, 169, MSG_MAGIC); level = skill;
return 0;
} }
/* Vertrautenmagie */
/* Kennt der Vertraute den Spruch, so zaubert er ganz normal.
* Ansonsten zaubert der Magier durch seinen Vertrauten, dh
* zahlt Komponenten und Aura. Dabei ist die maximale Stufe
* die des Vertrauten!
* Der Spruch wirkt dann auf die Region des Vertrauten und
* gilt nicht als Farcasting. */
if (familiar) {
if ((sp->sptyp & NOTFAMILIARCAST)) {
/* Fehler: "Diesen Spruch kann der Vertraute nicht zaubern" */
cmistake(u, ord, 177, MSG_MAGIC);
return 0;
}
if (mage != familiar) { /* Magier zaubert durch Vertrauten */
int sk;
if (range > 1) { /* Fehler! Versucht zu Farcasten */
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "familiar_farcast",
"mage", mage));
return 0;
}
sk = effskill(mage, SK_MAGIC, NULL);
if (distance(mage->region, r) > sk) {
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "familiar_toofar",
"mage", mage));
return 0;
}
/* mage auf magier setzen, level anpassen, range fuer Erhoehung
* der Spruchkosten nutzen */
range *= 2;
sk /= 2;
if (level > sk) level = sk;
}
}
/* Weitere Argumente zusammenbasteln */ /* Weitere Argumente zusammenbasteln */
if (sp->parameter) { if (sp->parameter) {
char *params[MAX_PARAMETERS]; char *params[MAX_PARAMETERS];

View file

@ -192,7 +192,6 @@ extern "C" {
enum magic_t mage_get_type(const struct sc_mage *mage); 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); 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); struct spellbook * mage_get_spellbook(const struct sc_mage * mage);
int mage_get_spell_level(const struct sc_mage *mage, const struct spell *sp);
int mage_get_spellpoints(const struct sc_mage *m); int mage_get_spellpoints(const struct sc_mage *m);
void mage_set_spellpoints(struct sc_mage *m, int aura); void mage_set_spellpoints(struct sc_mage *m, int aura);
int mage_change_spellpoints(struct sc_mage *m, int delta); int mage_change_spellpoints(struct sc_mage *m, int delta);
@ -201,6 +200,7 @@ extern "C" {
void unit_set_magic(struct unit *u, enum magic_t mtype); void unit_set_magic(struct unit *u, enum magic_t mtype);
struct spellbook * unit_get_spellbook(const struct unit * u); struct spellbook * unit_get_spellbook(const struct unit * u);
void unit_add_spell(struct unit * u, struct spell * sp, int level); void unit_add_spell(struct unit * u, struct spell * sp, int level);
int unit_spell_level(const struct unit *u, const struct spell *sp);
bool is_mage(const struct unit *u); bool is_mage(const struct unit *u);
/* gibt true, wenn u->mage gesetzt. */ /* gibt true, wenn u->mage gesetzt. */
@ -222,7 +222,9 @@ extern "C" {
bool u_hasspell(const struct unit *u, const struct spell *sp); bool u_hasspell(const struct unit *u, const struct spell *sp);
/* prueft, ob der Spruch in der Spruchliste der Einheit steht. */ /* prueft, ob der Spruch in der Spruchliste der Einheit steht. */
void pick_random_spells(struct faction *f, int level, struct spellbook * book, int num_spells); void pick_random_spells(struct faction *f, int level, struct spellbook * book, int num_spells);
bool knowsspell(const struct region *r, const struct unit *u, bool knowsspell(
const struct region *r,
const struct unit *u,
const struct spell * sp); const struct spell * sp);
/* prueft, ob die Einheit diesen Spruch gerade beherrscht, dh /* prueft, ob die Einheit diesen Spruch gerade beherrscht, dh
* mindestens die erforderliche Stufe hat. Hier koennen auch Abfragen * mindestens die erforderliche Stufe hat. Hier koennen auch Abfragen