forked from github/server
Merge pull request #937 from eressea/2737-spells-cost-and-familiars
2737 spells cost and familiars
This commit is contained in:
commit
14e2e96c73
14 changed files with 208 additions and 99 deletions
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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")
|
||||||
|
|
84
scripts/tests/e2/familiars.lua
Normal file
84
scripts/tests/e2/familiars.lua
Normal 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
|
|
@ -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'
|
||||||
|
|
|
@ -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")
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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))
|
||||||
|
|
||||||
|
|
|
@ -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')
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
176
src/magic.c
176
src/magic.c
|
@ -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];
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue