From 87ad2fb104d07393e1f137ed6f020d8d96a83691 Mon Sep 17 00:00:00 2001 From: dg <dg@51575b47-30f0-44d4-a5cc-537603b46e54> Date: Thu, 27 Jan 2011 10:03:33 +0000 Subject: [PATCH] Reworked talents in Shadows to make developing shadows more fun and useful. Replaced Feed Life (passive boost to feed that steals life) with Devour Life (steal life attack used while feeding) Minor fixes and tweeks to Doomed/Cursed. git-svn-id: http://svn.net-core.org/repos/t-engine4@2510 51575b47-30f0-44d4-a5cc-537603b46e54 --- game/modules/tome/ai/shadow.lua | 47 +-- game/modules/tome/class/Actor.lua | 15 +- .../tome/data/birth/classes/afflicted.lua | 3 +- game/modules/tome/data/damage_types.lua | 17 +- .../tome/data/talents/cursed/dark-figure.lua | 1 + .../data/talents/cursed/dark-sustenance.lua | 96 ++++- .../tome/data/talents/cursed/darkness.lua | 23 +- .../data/talents/cursed/force-of-will.lua | 16 +- .../tome/data/talents/cursed/punishments.lua | 8 +- .../tome/data/talents/cursed/shadows.lua | 339 ++++++++++++------ .../tome/data/talents/cursed/slaughter.lua | 12 +- game/modules/tome/data/timed_effects.lua | 20 +- 12 files changed, 427 insertions(+), 170 deletions(-) diff --git a/game/modules/tome/ai/shadow.lua b/game/modules/tome/ai/shadow.lua index d4309addf9..2cff0cf73c 100644 --- a/game/modules/tome/ai/shadow.lua +++ b/game/modules/tome/ai/shadow.lua @@ -59,6 +59,20 @@ local function shadowChooseActorTarget(self) end local function shadowMoveToActorTarget(self) + local range = math.floor(core.fov.distance(self.x, self.y, self.ai_target.actor.x, self.ai_target.actor.y)) + + if range <= 1 and self.ai_state.close_attack_spell_chance and rng.percent(self.ai_state.close_attack_spell_chance) then + -- chance for close spell + if self:closeAttackSpell() then return true end + elseif range <= 6 and self.ai_state.far_attack_spell_chance and rng.percent(self.ai_state.far_attack_spell_chance) then + -- chance for a far spell + if self:farAttackSpell() then return true end + end + + if range <= 1 and self.ai_state.dominate_chance and rng.percent(self.ai_state.dominate_chance) then + if self:dominate() then return true end + end + -- use the target blindside chance if it was assigned; otherwise, use the normal chance local blindsideChance = self.ai_target.blindside_chance or self.ai_state.blindside_chance self.ai_target.blindside_chance = nil @@ -76,17 +90,8 @@ local function shadowMoveToActorTarget(self) end end - if self:canAttackSpell() then - -- use the attack spell chance if it was assigned; otherwise, use the normal chance - local attackSpellChance = self.ai_target.attack_spell_chance or self.ai_state.attack_spell_chance - self.ai_target.attack_spell_chance = nil - if rng.percent(attackSpellChance) and self:attackSpell() then - return true - end - end - -- chance to reset target next turn if we are attacking (unless we have been focused) - if math.floor(core.fov.distance(self.x, self.y, self.ai_target.actor.x, self.ai_target.actor.y)) <= 1 and rng.percent(20) then + if range <= 1 and rng.percent(20) then --game.logPlayer(self.summoner, "#PINK#%s is about to attack.", self.name:capitalize()) if not self.ai_state.focus_on_target then self.ai_state.target_time = self.ai_state.target_timeout @@ -198,32 +203,28 @@ newAI("shadow", function(self) clearTarget(self) end - -- apply feed - if self.summoner:knowTalent(self.summoner.T_FEED_SHADOWS) then - local t = self.summoner:getTalentFromId(self.summoner.T_FEED_SHADOWS) - self:feed(t) - end - -- shadow wall if self.ai_state.shadow_wall then + clearTarget(self) + + local defendant = self.ai_state.shadow_wall_target - if self.ai_state.shadow_wall_time <= 0 then + if self.ai_state.shadow_wall_time <= 0 or defendant.dead then self.ai_state.shadow_wall = false else self.ai_state.shadow_wall_time = self.ai_state.shadow_wall_time - 1 - local range = core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) - if range > 3 then + local range = core.fov.distance(self.x, self.y, defendant.x, defendant.y) + if range >= 3 then -- phase door into range self:useTalent(self.T_SHADOW_PHASE_DOOR) return true elseif range > 1 then - self.ai_target.x = self.summoner.x - self.ai_target.y = self.summoner.y + self.ai_target.x = defendant.x + self.ai_target.y = defendant.y if shadowMoveToLocationTarget(self) then return true end end - -- no action..look for a target to attack local newX, newY local start = rng.range(0, 8) @@ -235,7 +236,7 @@ newAI("shadow", function(self) self:attackTarget(target, nil, 1, true) return true end - if not newX and math.floor(core.fov.distance(x, y, self.summoner.x, self.summoner.y)) <= 1 and self:canMove(x, y, false) then + if not newX and math.floor(core.fov.distance(x, y, defendant.x, defendant.y)) <= 1 and self:canMove(x, y, false) then newX, newY = x, y end end diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua index 90ba745840..f489cff67f 100644 --- a/game/modules/tome/class/Actor.lua +++ b/game/modules/tome/class/Actor.lua @@ -943,8 +943,7 @@ function _M:onTakeHit(value, src) self:forceUseTalent(self.T_SHIELD_OF_LIGHT, {ignore_energy=true}) end end - - + -- Second Life if self:isTalentActive(self.T_SECOND_LIFE) and value >= self.life then local sl = self.max_life * (0.05 + self:getTalentLevelRaw(self.T_SECOND_LIFE)/25) @@ -953,6 +952,18 @@ function _M:onTakeHit(value, src) game.logSeen(self, "%s has been saved by a blast of positive energy!", self.name:capitalize()) self:forceUseTalent(self.T_SECOND_LIFE, {ignore_energy=true}) end + + if value >= self.life and self.ai_state and self.ai_state.can_reform then + local t = self:getTalentFromId(self.T_SHADOW_REFORM) + if rng.percent(t.getChance(self, t)) then + value = 0 + self.life = self.max_life + game.logSeen(self, "%s fades for a moment and then reforms whole again!", self.name:capitalize()) + game.level.map:particleEmitter(self.x, self.y, 1, "teleport_out") + game:playSoundNear(self, "talents/heal") + game.level.map:particleEmitter(self.x, self.y, 1, "teleport_in") + end + end if self:knowTalent(self.T_LEECH) and src.hasEffect and src:hasEffect(src.EFF_VIMSENSE) then self:incVim(3 + self:getTalentLevel(self.T_LEECH) * 0.7) diff --git a/game/modules/tome/data/birth/classes/afflicted.lua b/game/modules/tome/data/birth/classes/afflicted.lua index 7335b31d1d..f0db10dfe3 100644 --- a/game/modules/tome/data/birth/classes/afflicted.lua +++ b/game/modules/tome/data/birth/classes/afflicted.lua @@ -56,7 +56,6 @@ newBirthDescriptor{ ["cursed/slaughter"]={true, 0.0}, ["cursed/endless-hunt"]={true, 0.0}, ["cursed/cursed-form"]={true, 0.0}, - ["cursed/traveler"]={true, 0.0}, ["technique/combat-training"]={true, 0.3}, ["cunning/survival"]={false, 0.0}, ["cursed/rampage"]={false, 0.0}, @@ -102,7 +101,7 @@ newBirthDescriptor{ ["cursed/shadows"]={true, 0.3}, ["cursed/darkness"]={true, 0.3}, ["cursed/cursed-form"]={true, 0.0}, - ["cursed/traveler"]={true, 0.0}, + ["cunning/survival"]={false, 0.0}, ["cursed/dark-figure"]={false, 0.0}, }, talents = { diff --git a/game/modules/tome/data/damage_types.lua b/game/modules/tome/data/damage_types.lua index 7bfa4b2a55..c24f527a03 100644 --- a/game/modules/tome/data/damage_types.lua +++ b/game/modules/tome/data/damage_types.lua @@ -1301,4 +1301,19 @@ newDamageType{ target:setEffect(target.EFF_SPEED, 3, {power=0.6}) end end, -} \ No newline at end of file +} + +newDamageType{ + name = "devour life", type = "DEVOUR_LIFE", + projector = function(src, x, y, type, dam) + if _G.type(dam) == "number" then dam = {dam=dam} end + local target = game.level.map(x, y, Map.ACTOR) -- Get the target first to make sure we heal even on kill + local realdam = DamageType:get(DamageType.PHYSICAL).projector(src, x, y, DamageType.PHYSICAL, dam.dam) + if target and realdam > 0 then + local heal = realdam * (dam.healfactor or 1) + src:heal(heal) + game.logSeen(target, "%s consumes %d life from %s!", src.name:capitalize(), heal, target.name) + end + end, + hideMessage=true, +} diff --git a/game/modules/tome/data/talents/cursed/dark-figure.lua b/game/modules/tome/data/talents/cursed/dark-figure.lua index 7290078a14..b3c5c04ba7 100644 --- a/game/modules/tome/data/talents/cursed/dark-figure.lua +++ b/game/modules/tome/data/talents/cursed/dark-figure.lua @@ -31,6 +31,7 @@ newTalent{ getRadius = function(self, t) return 3 + math.floor((self:getTalentLevelRaw(t) - 1) / 2) end, getDuration = function(self, t) return 5 + math.floor(self:getTalentLevel(t) * 2) end, tactical = { DISABLE = 2 }, + requires_target = true, range = 6, action = function(self, t) local tg = {type="hit", range=self:getTalentRange(t), talent=t} diff --git a/game/modules/tome/data/talents/cursed/dark-sustenance.lua b/game/modules/tome/data/talents/cursed/dark-sustenance.lua index 2bfeeeeceb..42487b89e4 100644 --- a/game/modules/tome/data/talents/cursed/dark-sustenance.lua +++ b/game/modules/tome/data/talents/cursed/dark-sustenance.lua @@ -17,6 +17,14 @@ -- Nicolas Casalini "DarkGod" -- darkgod@te4.org +local function combatTalentDamage(self, t, min, max) + return self:combatTalentSpellDamage(t, min, max, self.level + self:getWil()) +end + +local function combatPower(self, t, multiplier) + return (self.level + self:getWil()) * (multiplier or 1) +end + newTalent{ name = "Feed", type = {"cursed/dark-sustenance", 1}, @@ -26,25 +34,25 @@ newTalent{ cooldown = 6, range = 7, tactical = { BUFF = 2, DEFEND = 1 }, - requires_target = true, + requires_target = function(self, t) return self:getTalentLevel(t) >= 5 end, + direct_hit = true, getHateGain = function(self, t) return math.sqrt(self:getTalentLevel(t)) * 0.2 + self:getWil(0.15) end, action = function(self, t) - local tg = {type="hit", range=self:getTalentRange(t)} + local range = self:getTalentRange(t) + local tg = {type="hit", range=range} local x, y, target = self:getTarget(tg) - if not x or not y or not target then return nil end + if not x or not y or not target or core.fov.distance(self.x, self.y, x, y) > range then return nil end if self:reactionToward(target) >= 0 or target.summoner == self then game.logPlayer(self, "You can only gain sustenance from your foes!"); return nil end - print("*** targeted"); - -- remove old effect - if self:hasEffect(self.EFF_FEED_HATE) then - self:removeEffect(self.EFF_FEED_HATE) + if self:hasEffect(self.EFF_FEED) then + self:removeEffect(self.EFF_FEED) end local hateGain = t.getHateGain(self, t) @@ -53,11 +61,11 @@ newTalent{ local damageGain = 0 local resistGain = 0 - local tFeedHealth = self:getTalentFromId(self.T_FEED_HEALTH) - if tFeedHealth and self:getTalentLevelRaw(tFeedHealth) > 0 then - constitutionGain = tFeedHealth.getConstitutionGain(self, tFeedHealth, target) - lifeRegenGain = tFeedHealth.getLifeRegenGain(self, tFeedHealth) - end + --local tFeedHealth = self:getTalentFromId(self.T_FEED_HEALTH) + --if tFeedHealth and self:getTalentLevelRaw(tFeedHealth) > 0 then + -- constitutionGain = tFeedHealth.getConstitutionGain(self, tFeedHealth, target) + -- lifeRegenGain = tFeedHealth.getLifeRegenGain(self, tFeedHealth) + --end local tFeedPower = self:getTalentFromId(self.T_FEED_POWER) if tFeedPower and self:getTalentLevelRaw(tFeedPower) > 0 then @@ -75,11 +83,67 @@ newTalent{ end, info = function(self, t) local hateGain = t.getHateGain(self, t) - return ([[Feed from the essence of your enemy. Draws %0.2f hate per turn from a targeted foe as long as the foe remains in your line of sight. + return ([[Feed from the essence of your enemy. Draws %0.2f hate per turn from a targeted foe as long as the they remain in your line of sight. Improves with the Willpower stat.]]):format(hateGain) end, } +newTalent{ + name = "Devour Life", + type = {"cursed/dark-sustenance", 2}, + require = cursed_wil_req2, + points = 5, + random_ego = "attack", + cooldown = 6, + range = 7, + tactical = { BUFF = 2, DEFEND = 1 }, + direct_hit = true, + requires_target = true, + getLifeSteal = function(self, t, target) + return combatTalentDamage(self, t, 20, 120) + end, + action = function(self, t) + local effect = self:hasEffect(self.EFF_FEED) + if not effect then + if self:getTalentLevel(t) >= 5 then + local tFeed = self:getTalentFromId(self.T_FEED) + if not tFeed.action(self, tFeed) then return nil end + + effect = self:hasEffect(self.EFF_FEED) + else + game.logPlayer(self, "You must begin feeding before you can Devour Life."); + return nil + end + end + local target = effect.target + + if target and not target.dead then + local lifeSteal = t.getLifeSteal(self, t) + self:project({type="hit", x=target.x,y=target.y}, target.x, target.y, DamageType.DEVOUR_LIFE, { dam=lifeSteal }) + + game.level.map:particleEmitter(self.x, self.y, math.max(math.abs(target.x-self.x), math.abs(target.y-self.y)), "dark_torrent", {tx=target.x-self.x, ty=target.y-self.y}) + --local dx, dy = target.x - self.x, target.y - self.y + --game.level.map:particleEmitter(self.x, self.y,math.max(math.abs(dx), math.abs(dy)), "feed_hate", { tx=dx, ty=dy }) + game:playSoundNear(self, "talents/fire") + + return true + end + + return nil + end, + info = function(self, t) + local lifeSteal = t.getLifeSteal(self, t) + if self:getTalentLevel(t) >= 5 then + return ([[Devours life from the target of your feeding. %d life from the victim will be added to your own. Devour Life can be used like the Feed talent to begin feeding. + Improves with the Willpower stat.]]):format(lifeSteal) + else + return ([[Devours life from the target of your feeding. %d life from the victim will be added to your own. At level 5 Devour Life can be used like the Feed talent to begin feeding. + Improves with the Willpower stat.]]):format(lifeSteal) + end + end, +} + +--[[ newTalent{ name = "Feed Health", type = {"cursed/dark-sustenance", 2}, @@ -102,11 +166,11 @@ newTalent{ info = function(self, t) local constitutionGain = t.getConstitutionGain(self, t) local lifeRegenGain = t.getLifeRegenGain(self, t) - return ([[Enhances your feeding by transferring %d constitution and %0.1f life per turn from a targeted foe to you. - Improves with the Willpower stat.]]):format(constitutionGain, lifeRegenGain) + return ([Enhances your feeding by transferring %d constitution and %0.1f life per turn from a targeted foe to you. + Improves with the Willpower stat.]):format(constitutionGain, lifeRegenGain) end, } - +]] newTalent{ name = "Feed Power", type = {"cursed/dark-sustenance", 3}, diff --git a/game/modules/tome/data/talents/cursed/darkness.lua b/game/modules/tome/data/talents/cursed/darkness.lua index 04ef6baf09..51dd4a8444 100644 --- a/game/modules/tome/data/talents/cursed/darkness.lua +++ b/game/modules/tome/data/talents/cursed/darkness.lua @@ -46,7 +46,7 @@ local function createDarkTendrils(summoner, x, y, target, damage, duration, pinD local done = false local hitTarget = false - local tCreepingDarkness = self.summoner:getTalentFromId(summoner.T_CREEPING_DARKNESS) + local tCreepingDarkness = self.summoner:getTalentFromId(self.summoner.T_CREEPING_DARKNESS) if self.finalizing then if self.duration <= 0 or self.target.dead or self.x ~= self.target.x or self.y ~= self.target.y then @@ -295,7 +295,6 @@ newTalent{ local locations = {} local grids = core.fov.circle_grids(x, y, radius, true) for darkX, yy in pairs(grids) do for darkY, _ in pairs(grids[darkX]) do - print("*** pairs", darkX, darkY) local l = line.new(x, y, darkX, darkY) local lx, ly = l() while lx and ly do @@ -307,7 +306,6 @@ newTalent{ if lx == darkX and ly == darkY and t.canCreep(darkX, darkY) then locations[#locations+1] = {darkX, darkY} - print("*** locations", darkX, darkY) end end end @@ -349,6 +347,17 @@ newTalent{ return 5 + (level - 3) * 3 end, + hasLOS = function(x1, y1, x2, y2) + local l = line.new(x1, y1, x2, y2) + local lx, ly = l() + while lx and ly do + local entity = game.level.map:checkAllEntities(lx, ly, "block_sight") + if entity and not game.level.map:checkAllEntities(lx, ly, "creepingDark") then return false end + + lx, ly = l() + end + return true + end, info = function(self, t) local damageIncrease = t.getDamageIncrease(self, t) if damageIncrease <= 0 then @@ -369,7 +378,6 @@ newTalent{ cooldown = 6, tactical = { ATTACK = 2, DISABLE = 1 }, range = 5, - direct_hit = true, reflectable = true, requires_target = true, getDamage = function(self, t) @@ -422,6 +430,8 @@ newTalent{ hate = 1.2, range = 6, tactical = { ATTACK = 2, DISABLE = 2 }, + direct_hit = true, + requires_target = true, getPinDuration = function(self, t) return 1 + math.floor(self:getTalentLevel(t) / 2) end, @@ -431,9 +441,10 @@ newTalent{ action = function(self, t) if self.dark_tendrils then return false end - local tg = {type="hit", range=self:getTalentRange(t), talent=t} + local range = self:getTalentRange(t) + local tg = {type="hit", range=range, talent=t} local x, y, target = self:getTarget(tg) - if not x or not y or not target then return nil end + if not x or not y or not target or core.fov.distance(self.x, self.y, x, y) > range then return nil end local pinDuration = t.getPinDuration(self, t) local damage = t.getDamage(self, t) diff --git a/game/modules/tome/data/talents/cursed/force-of-will.lua b/game/modules/tome/data/talents/cursed/force-of-will.lua index feaf482bb0..79ba231536 100644 --- a/game/modules/tome/data/talents/cursed/force-of-will.lua +++ b/game/modules/tome/data/talents/cursed/force-of-will.lua @@ -103,6 +103,8 @@ newTalent{ cooldown = 4, hate = 0.5, tactical = { ATTACK = 2 }, + direct_hit = true, + requires_target = true, range = function(self, t) return 4 end, @@ -117,11 +119,10 @@ newTalent{ local tg = {type="hit", range=self:getTalentRange(t)} local x, y, target = self:getTarget(tg) - if not x or not y or not target then return nil end - - local distance = math.max(1, math.floor(core.fov.distance(self.x, self.y, x, y))) + if not x or not y or not target or core.fov.distance(self.x, self.y, x, y) > range then return nil end - local power = (1 - ((distance - 1) / range)) + --local distance = math.max(1, math.floor(core.fov.distance(self.x, self.y, x, y))) + local power = 1 --(1 - ((distance - 1) / range)) local damage = t.getDamage(self, t) * power local knockback = t.getKnockback(self, t) forceHit(self, target, self.x, self.y, damage, knockback, 15, power) @@ -130,7 +131,7 @@ newTalent{ info = function(self, t) local damage = t.getDamage(self, t) local knockback = t.getKnockback(self, t) - return ([[Focusing your hate you strike your foe with unseen force for up to %d damage and %d knockback at a range of 1. Damage decreases the further you are from your target. + return ([[Focusing your hate you strike your foe with unseen force for %d damage and %d knockback. Damage increases with the Willpower stat.]]):format(damDesc(self, DamageType.PHYSICAL, damage), knockback) end, } @@ -145,7 +146,6 @@ newTalent{ cooldown = 12, tactical = { DEFEND = 2 }, no_sustain_autoreset = true, - direct_hit = true, getMaxDamage = function(self, t) return combatTalentDamage(self, t, 20, 200) end, @@ -210,6 +210,7 @@ newTalent{ random_ego = "attack", cooldown = 10, tactical = { ATTACK = 2 }, + requires_target = true, hate = 1.5, range = function(self, t) return 4 @@ -231,7 +232,7 @@ newTalent{ local tg = {type="ball", nolock=true, pass_terrain=false, friendly_fire=false, nowarning=true, range=range, radius=radius, talent=t} local blastX, blastY = self:getTarget(tg) - if not blastX or not blastY then return nil end + if not blastX or not blastY or core.fov.distance(self.x, self.y, blastX, blastY) > range then return nil end local grids = self:project(tg, blastX, blastY, function(x, y, target, self) @@ -269,7 +270,6 @@ newTalent{ hate = 2, cooldown = 50, tactical = { ATTACKAREA = 2 }, - direct_hit = true, range = function(self, t) return math.floor(4 + self:getTalentLevel(t) / 2.3) end, diff --git a/game/modules/tome/data/talents/cursed/punishments.lua b/game/modules/tome/data/talents/cursed/punishments.lua index 780c6cdf0b..7eed24bbca 100644 --- a/game/modules/tome/data/talents/cursed/punishments.lua +++ b/game/modules/tome/data/talents/cursed/punishments.lua @@ -84,6 +84,8 @@ newTalent{ hate = 0.8, range = 7, tactical = { ATTACK = 2 }, + direct_hit = true, + requires_target = true, getDuration = function(self, t) return 10 end, @@ -103,7 +105,7 @@ newTalent{ local range = self:getTalentRange(t) local tg = {type="hit", range=range} local x, y, target = self:getTarget(tg) - if not x or not y or not target or target:hasEffect(target.EFF_HATEFUL_WHISPER) then return nil end + if not x or not y or not target or core.fov.distance(self.x, self.y, x, y) > range or target:hasEffect(target.EFF_HATEFUL_WHISPER) then return nil end local duration = t.getDuration(self, t) local damage = t.getDamage(self, t) @@ -260,6 +262,8 @@ newTalent{ hate = 0.5, range = 7, tactical = { ATTACK = 2 }, + direct_hit = true, + requires_target = true, getDuration = function(self, t) return 5 end, @@ -273,7 +277,7 @@ newTalent{ local range = self:getTalentRange(t) local tg = {type="hit", range=range} local x, y, target = self:getTarget(tg) - if not x or not y or not target then return nil end + if not x or not y or not target or core.fov.distance(self.x, self.y, x, y) > range then return nil end local damage = t.getDamage(self, t) local mindpower = t.getMindpower(self, t) diff --git a/game/modules/tome/data/talents/cursed/shadows.lua b/game/modules/tome/data/talents/cursed/shadows.lua index 39ecc72c0d..b0502d8f92 100644 --- a/game/modules/tome/data/talents/cursed/shadows.lua +++ b/game/modules/tome/data/talents/cursed/shadows.lua @@ -27,7 +27,7 @@ newTalent{ action = function(self, t) local x, y, range if self.ai_state.shadow_wall then - x, y, range = self.summoner.x, self.summoner.y, 1 + x, y, range = self.ai_state.shadow_wall_target.x, self.ai_state.shadow_wall_target.y, 1 elseif self.ai_target.x and self.ai_target.y then x, y, range = self.ai_target.x, self.ai_target.y, 1 else @@ -82,6 +82,82 @@ newTalent{ end, } +newTalent{ + short_name = "SHADOW_LIGHTNING", + name = "Shadow Lightning", + type = {"spell/other", 1}, + require = { }, + points = 5, + random_ego = "attack", + range = 1, + direct_hit = true, + requires_target = true, + getDamage = function(self, t) return self:combatTalentSpellDamage(t, 20, 200) end, + action = function(self, t) + local tg = {type="hit", range=self:getTalentRange(t), talent=t} + local x, y = self:getTarget(tg) + if not x or not y then return nil end + local dam = self:spellCrit(t.getDamage(self, t)) + self:project(tg, x, y, DamageType.LIGHTNING, rng.avg(dam / 3, dam, 3)) + local _ _, x, y = self:canProject(tg, x, y) + game.level.map:particleEmitter(self.x, self.y, math.max(math.abs(x-self.x), math.abs(y-self.y)), "lightning", {tx=x-self.x, ty=y-self.y}) + game:playSoundNear(self, "talents/lightning") + return true + end, + info = function(self, t) + local damage = t.getDamage(self, t) + return ([[Strikes the target with a spark of lightning doing %0.2f to %0.2f damage. + The damage will increase with the Magic stat]]): + format(damDesc(self, DamageType.LIGHTNING, damage / 3), + damDesc(self, DamageType.LIGHTNING, damage)) + end, +} + +newTalent{ + short_name = "SHADOW_FLAMES", + name = "Shadow Flames", + type = {"spell/other", 1}, + require = { }, + points = 5, + random_ego = "attack", + range = 6, + direct_hit = true, + requires_target = true, + getDamage = function(self, t) return self:combatTalentSpellDamage(t, 20, 140) end, + action = function(self, t) + local tg = {type="hit", range=self:getTalentRange(t), talent=t} + local x, y = self:getTarget(tg) + if not x or not y then return nil end + local dam = self:spellCrit(t.getDamage(self, t)) + self:project(tg, x, y, DamageType.FIRE, dam) + game.level.map:particleEmitter(x, y, 1, "flame") + game:playSoundNear(self, "talents/fire") + return true + end, + info = function(self, t) + local damage = t.getDamage(self, t) + return ([[Bathes the target in flames doing %0.2f damage + The damage will increase with the Magic stat]]): + format(damDesc(self, DamageType.FIREBURN, damage)) + end, +} + +newTalent{ + short_name = "SHADOW_REFORM", + name = "Reform", + type = {"spell/other", 1}, + require = { }, + points = 5, + getChance = function(self, t) + return 50 --10 + self:getMag() * 0.25 + self:getTalentLevel(t) * 2 + end, + info = function(self, t) + local chance = t.getChance(self, t) + return ([[When a shadow is hit and killed, there is a %d%% chance it will reform unhurt." + The chance will increase with the Magic stat]]):format(chance) + end, +} + local function createShadow(self, level, duration, target) return require("mod.class.NPC").new{ type = "undead", subtype = "shadow", @@ -105,8 +181,8 @@ local function createShadow(self, level, duration, target) stats = { str=10 + math.floor(level * 0.2), dex=15 + math.floor(level * 0.8), - mag=15 + math.floor(level * 0.5), - wil=10 + math.floor(level * 0.4), + mag=15 + math.floor(level * 0.6), + wil=10 + math.floor(level * 0.6), cun=10 + math.floor(level * 0.2), con=5, }, @@ -119,12 +195,15 @@ local function createShadow(self, level, duration, target) }, evasion = 30, mana = 100, + summoner_hate_per_kill = 0.8, resolvers.talents{ [self.T_SHADOW_PHASE_DOOR]=math.max(5, math.floor(1 + level * 0.1)), [self.T_SHADOW_BLINDSIDE]=math.max(5, math.floor(1 + level * 0.1)), - [self.T_LIGHTNING]=math.max(5, math.floor(1 + level * 0.1)), - [self.T_SHOCK]=math.max(5, math.floor(1 + level * 0.1)), + [self.T_SHADOW_LIGHTNING]=math.max(5, math.floor(1 + level * 0.1)), + [self.T_SHADOW_FLAMES]=math.max(5, math.floor(1 + level * 0.1)), + [self.T_SHADOW_REFORM]=math.max(5, math.floor(1 + level * 0.1)), [self.T_HEAL]=math.max(5, math.floor(1 + level * 0.1)), + [self.T_DOMINATE]=math.max(5, math.floor(1 + level * 0.1)), }, undead = 1, @@ -153,7 +232,10 @@ local function createShadow(self, level, duration, target) blindside_chance = 15, phasedoor_chance = 5, - attack_spell_chance = 5, + close_attack_spell_chance = 0, + far_attack_spell_chance = 0, + can_reform = false, + dominate_chance = 0, feed_level = 0 }, @@ -166,46 +248,40 @@ local function createShadow(self, level, duration, target) healSelf = function(self) self:useTalent(self.T_HEAL) end, - canAttackSpell = function(self) - local target = self.ai_target.actor - return target and math.floor(core.fov.distance(self.x, self.y, target.x, target.y)) <= 1 + closeAttackSpell = function(self) + return self:useTalent(self.T_SHADOW_LIGHTNING) end, - attackSpell = function(self) - if self:canAttackSpell() then - local choice = rng.range(1, 2) - if choice == 1 then - return self:useTalent(self.T_LIGHTNING) - else - return self:useTalent(self.T_SHOCK) - end + farAttackSpell = function(self) + return self:useTalent(self.T_SHADOW_FLAMES) + end, + dominate = function(self) + return self:useTalent(self.T_DOMINATE) + end, + feed = function(self) + if self.summoner:knowTalent(self.summoner.T_SHADOW_MAGES) then + local tShadowMages = self.summoner:getTalentFromId(self.summoner.T_SHADOW_MAGES) + self.ai_state.close_attack_spell_chance = tShadowMages.getCloseAttackSpellChance(self.summoner, tShadowMages) + self.ai_state.far_attack_spell_chance = tShadowMages.getFarAttackSpellChance(self.summoner, tShadowMages) + self.ai_state.can_reform = self.summoner:getTalentLevel(tShadowMages) >= 5 else - return false + self.ai_state.close_attack_spell_chance = 0 + self.ai_state.far_attack_spell_chance = 0 + self.ai_state.can_reform = false end - end, - feed = function(self, t) - local level = self.summoner:getTalentLevel(t) - if self.ai_state.feed_level == level then return end - self.ai_state.feed_level = level - - -- clear old values if self.ai_state.feed_temp1 then self:removeTemporaryValue("combat_atk", self.ai_state.feed_temp1) end self.ai_state.feed_temp1 = nil if self.ai_state.feed_temp2 then self:removeTemporaryValue("inc_damage", self.ai_state.feed_temp2) end self.ai_state.feed_temp2 = nil - self.summoner_hate_per_kill = nil - - if level and level > 0 then - -- set new values - self.ai_state.feed_temp1 = self:addTemporaryValue("combat_atk", t.getCombatAtk(self.summoner, t)) - self.ai_state.feed_temp2 = self:addTemporaryValue("inc_damage", {all=t.getIncDamage(self.summoner, t)}) - self.summoner_hate_per_kill = t.getHatePerKill(self.summoner, t) + if self.summoner:knowTalent(self.summoner.T_SHADOW_WARRIORS) then + local tShadowWarriors = self.summoner:getTalentFromId(self.summoner.T_SHADOW_WARRIORS) + self.ai_state.feed_temp1 = self:addTemporaryValue("combat_atk", tShadowWarriors.getCombatAtk(self.summoner, tShadowWarriors)) + self.ai_state.feed_temp2 = self:addTemporaryValue("inc_damage", {all=tShadowWarriors.getIncDamage(self.summoner, tShadowWarriors)}) + self.ai_state.dominate_chance = tShadowWarriors.getDominateChance(self.summoner, tShadowWarriors) + else + self.ai_state.dominate_chance = 0 end end, - shadowWall = function(self, t, duration) - self.ai_state.shadow_wall = true - self.ai_state.shadow_wall_time = duration - end, } end @@ -279,6 +355,7 @@ newTalent{ shadow:resolve(nil, true) shadow:forceLevelup(level) game.zone:addEntity(game.level, shadow, "actor", x, y) + shadow:feed() game.level.map:particleEmitter(x, y, 1, "teleport_in") game:playSoundNear(self, "talents/spell_generic") @@ -294,7 +371,7 @@ newTalent{ info = function(self, t) local maxShadows = t.getMaxShadows(self, t) local level = t.getLevel(self, t) - return ([[While this ability is active you will continually call up to %d level %d shadows to aid you in battle. Each shadow costs 1 hate to summon and will be equal in level to you when it appears.]]):format(maxShadows, level) + return ([[While this ability is active you will continually call up to %d level %d shadows to aid you in battle. Shadows are weak combatants that can Heal themselves, Blindside their opponents and Phase Door from place to place. Each shadow costs 1 hate to summon and will be equal in level to you when it appears.]]):format(maxShadows, level) end, } @@ -305,71 +382,98 @@ newTalent{ points = 5, random_ego = "attack", cooldown = 10, - hate = 1, + hate = 0.5, range = 6, requires_target = true, tactical = { ATTACK = 2 }, - getDuration = function(self, t) - return self:getTalentLevel(t) + getDefenseDuration = function(self, t) + return 3 + math.floor(self:getTalentLevel(t) * 1.5) end, getBlindsideChance = function(self, t) return math.min(100, 30 + self:getTalentLevel(t) * 10) end, - getAttackSpellChance = function(self, t) - return math.min(100, 5 + self:getTalentLevel(t) * 5) - end, action = function(self, t) - local target = { type="hit", range=self:getTalentRange(t) } + local range = self:getTalentRange(t) + local target = { type="hit", range=range, nowarning=true } local x, y, target = self:getTarget(target) - if not x or not y or not target then return nil end - - local blindsideChance = t.getBlindsideChance(self, t) - local attackSpellChance = t.getAttackSpellChance(self, t) - local shadowCount = 0 - for _, e in pairs(game.level.entities) do - if e.summoner and e.summoner == self and e.subtype == "shadow" then - -- reset target and set to focus - e.ai_target.x = nil - e.ai_target.y = nil - e.ai_target.actor = target - e.ai_target.focus_on_target = true - e.ai_target.blindside_chance = blindsideChance - e.ai_target.attack_spell_chance = attackSpellChance - - shadowCount = shadowCount + 1 + if not x or not y or not target or core.fov.distance(self.x, self.y, x, y) > range then return nil end + + if self:reactionToward(target) < 0 then + -- attack the target + local blindsideChance = t.getBlindsideChance(self, t) + local shadowCount = 0 + for _, e in pairs(game.level.entities) do + if e.summoner and e.summoner == self and e.subtype == "shadow" then + -- reset target and set to focus + e.ai_target.x = nil + e.ai_target.y = nil + e.ai_target.actor = target + e.ai_target.focus_on_target = true + e.ai_target.blindside_chance = blindsideChance + + shadowCount = shadowCount + 1 + end end - end - if shadowCount > 0 then - game.logPlayer(self, "#PINK#The shadows converge on %s!", target.name) - return true + if shadowCount > 0 then + game.logPlayer(self, "#PINK#The shadows converge on %s!", target.name) + return true + else + game.logPlayer(self, "Their are no shadows to heed the call!") + return false + end else - game.logPlayer(self, "Their are no shadows to heed the call!") - return false + -- defend the target + local defenseDuration = t.getDefenseDuration(self, t) + local shadowCount = 0 + for _, e in pairs(game.level.entities) do + if e.summoner and e.summoner == self and e.subtype == "shadow" then + e.ai_state.shadow_wall = true + e.ai_state.shadow_wall_target = target + e.ai_state.shadow_wall_time = defenseDuration + + shadowCount = shadowCount + 1 + end + end + + if shadowCount > 0 then + game.logPlayer(self, "#PINK#The shadows form around %s!", target.name) + return true + else + game.logPlayer(self, "Their are no shadows to heed the call!") + return false + end end end, info = function(self, t) - local duration = t.getDuration(self, t) + local defenseDuration = t.getDefenseDuration(self, t) local blindsideChance = t.getBlindsideChance(self, t) - local attackSpellChance = t.getAttackSpellChance(self, t) - return ([[Focus your shadows on a single target. There is a %d%% chance they will blindside the target and a %d%% chance they will use an attack spell.]]):format(blindsideChance, attackSpellChance) + return ([[Focus your shadows on a single target. Friendly targets will be defended for %d turns. Hostile targets will be attacked with a %d%% chance they will blindside the target.]]):format(defenseDuration, blindsideChance) end, } newTalent{ - name = "Feed Shadows", + name = "Shadow Mages", type = {"cursed/shadows", 3}, mode = "passive", require = cursed_mag_req3, points = 5, - getIncDamage = function(self, t) - return math.floor((math.sqrt(self:getTalentLevel(t)) - 0.5) * 17) + getCloseAttackSpellChance = function(self, t) + if self:getTalentLevelRaw(t) > 0 then + return math.min(100, math.sqrt(self:getTalentLevel(t)) * 5) + else + return 0 + end end, - getCombatAtk = function(self, t) - return math.floor((math.sqrt(self:getTalentLevel(t)) - 0.5) * 17) + getFarAttackSpellChance = function(self, t) + if self:getTalentLevelRaw(t) >= 3 then + return math.min(100, math.sqrt(self:getTalentLevel(t)) * 5) + else + return 0 + end end, - getHatePerKill = function(self, t) - return (self:getTalentLevel(t) / 8) * self.hate_per_kill + canReform = function(self, t) + return self:getTalentLevelRaw(t) >= 5 end, on_learn = function(self, t) if game and game.level and game.level.entities then @@ -394,46 +498,73 @@ newTalent{ return true end, info = function(self, t) - local combatAtk = t.getCombatAtk(self, t) - local incDamage = t.getIncDamage(self, t) - local hatePerKill = t.getHatePerKill(self, t) - return ([[Your hatred of all living things begins to feed your shadows. Their new viciousness gives them %d%% extra attack and %d%% extra damage and each kill they make transfers %0.2f hatred back to you.]]):format(combatAtk, incDamage, hatePerKill) + local closeAttackSpellChance = t.getCloseAttackSpellChance(self, t) + local farAttackSpellChance = t.getFarAttackSpellChance(self, t) + + local level = self:getTalentLevelRaw(t) + if level < 3 then + return ([[Infuse magic into your shadows to give them fearsome spells. + Your shadows can strike adjacent foes with Lightning (%d%% chance at range 1). + At 3 talent points they will gain Flames and at 5 talent points they will gain Reform.]]):format(closeAttackSpellChance) + elseif level < 5 then + return ([[Infuse magic into your shadows to give them fearsome spells. + Your shadows can strike adjacent foes with Lightning (%d%% chance at range 1). + Your shadows can sear their enemies from a distance with Flames (%d%% chance at range 2 to 6). + At 5 talent points they will gain Reform.]]):format(closeAttackSpellChance, farAttackSpellChance) + else + return ([[Infuse magic into your shadows to give them fearsome spells. + Your shadows can strike adjacent foes with Lightning (%d%% chance at range 1). + Your shadows can sear their enemies from a distance with Flames (%d%% chance at range 2 to 6). + When your shadows are struck down they will attempt to Reform becoming whole again.]]):format(closeAttackSpellChance, farAttackSpellChance) + end end, } newTalent{ - name = "Shadow Wall", + name = "Shadow Warriors", type = {"cursed/shadows", 4}, + mode = "passive", require = cursed_mag_req4, points = 5, - cooldown = 10, - hate = 1, - tactical = { DEFEND = 2 }, - getDuration = function(self, t) - return 2 + self:getTalentLevel(t) * 2 + getIncDamage = function(self, t) + return math.floor((math.sqrt(self:getTalentLevel(t)) - 0.5) * 23) end, - action = function(self, t) - - local duration = t.getDuration(self, t) - local shadowCount = 0 - for _, e in pairs(game.level.entities) do - if e.summoner and e.summoner == self and e.subtype == "shadow" then - e:shadowWall(t, duration) - - shadowCount = shadowCount + 1 + getCombatAtk = function(self, t) + return math.floor((math.sqrt(self:getTalentLevel(t)) - 0.5) * 23) + end, + getDominateChance = function(self, t) + if self:getTalentLevelRaw(t) > 0 then + return math.min(100, math.sqrt(self:getTalentLevel(t)) * 5) + else + return 0 + end + end, + on_learn = function(self, t) + if game and game.level and game.level.entities then + for _, e in pairs(game.level.entities) do + if e.summoner and e.summoner == self and e.subtype == "shadow" then + e:feed(t) + end end end - - if shadowCount > 0 then - game.logPlayer(self, "#PINK#The shadows form around %s!", self.name) - return true - else - game.logPlayer(self, "Their are no shadows to heed the call!") - return false + + return { } + end, + on_unlearn = function(self, t, p) + if game and game.level and game.level.entities then + for _, e in pairs(game.level.entities) do + if e.summoner and e.summoner == self and e.subtype == "shadow" then + e:feed(t) + end + end end + + return true end, info = function(self, t) - local duration = t.getDuration(self, t) - return ([[Summon your shadows to your side to form a wall against danger. Your shadows will stay beside you for %d turns and attack anyone nearby.]]):format(duration) + local combatAtk = t.getCombatAtk(self, t) + local incDamage = t.getIncDamage(self, t) + local dominateChance = t.getDominateChance(self, t) + return ([[Instill hate in your shadows strengthening their attacks. They gain %d%% extra attack and %d%% extra damage. The fury of their attacks gives them the ability to try to Dominate their foes, increasing all damage taken by that foe for 4 turns. (%d%% chance at range 1)]]):format(combatAtk, incDamage, dominateChance) end, } diff --git a/game/modules/tome/data/talents/cursed/slaughter.lua b/game/modules/tome/data/talents/cursed/slaughter.lua index a6e338758e..88dac4a048 100644 --- a/game/modules/tome/data/talents/cursed/slaughter.lua +++ b/game/modules/tome/data/talents/cursed/slaughter.lua @@ -50,7 +50,8 @@ newTalent{ end, info = function(self, t) local multiplier = (0.17 + .23 * self:getTalentLevel(t)) - return ([[Slashes wildly at your target for 100%% (at 0 Hate) to %d%% (at 10+ Hate) damage.]]):format(multiplier * 100 + 100) + return ([[Slashes wildly at your target for 100%% (at 0 Hate) to %d%% (at 10+ Hate) damage. + Requires a one or two handed axe.]]):format(multiplier * 100 + 100) end, } @@ -96,7 +97,8 @@ newTalent{ end, info = function(self, t) local multiplier = self:combatTalentWeaponDamage(t, 0.2, 0.7) - return ([[Assault nearby foes with 4 fast attacks for %d%% (at 0 Hate) to %d%% (at 10+ Hate) damage each.]]):format(multiplier * 50, multiplier * 100) + return ([[Assault nearby foes with 4 fast attacks for %d%% (at 0 Hate) to %d%% (at 10+ Hate) damage each. + Requires a one or two handed axe.]]):format(multiplier * 50, multiplier * 100) end, } @@ -201,7 +203,8 @@ newTalent{ else size = "Small" end - return ([[Charge through your opponents, attacking anyone near your path for %d%% (at 0 Hate) to %d%% (at 10+ Hate) damage. %s opponents may be knocked from your path.]]):format(multiplier * 30, multiplier * 100, size) + return ([[Charge through your opponents, attacking anyone near your path for %d%% (at 0 Hate) to %d%% (at 10+ Hate) damage. %s opponents may be knocked from your path. + Requires a one or two handed axe.]]):format(multiplier * 30, multiplier * 100, size) end, } @@ -242,6 +245,7 @@ newTalent{ info = function(self, t) local chance = 28 + self:getTalentLevel(t) * 7 local multiplier = self:combatTalentWeaponDamage(t, 0.2, 0.7) - return ([[Every swing of your axe has a %d%% chance of striking a second target for %d%% (at 0 Hate) to %d%% (at 10+ Hate) damage.]]):format(chance, multiplier * 50, multiplier * 100) + return ([[Every swing of your axe has a %d%% chance of striking a second target for %d%% (at 0 Hate) to %d%% (at 10+ Hate) damage. + Requires a one or two handed axe.]]):format(chance, multiplier * 50, multiplier * 100) end, } diff --git a/game/modules/tome/data/timed_effects.lua b/game/modules/tome/data/timed_effects.lua index 9d5b8add06..a53c522062 100644 --- a/game/modules/tome/data/timed_effects.lua +++ b/game/modules/tome/data/timed_effects.lua @@ -2507,9 +2507,25 @@ newEffect{ if eff.extension <= 0 then self:removeEffect(self.EFF_FEED) end - elseif eff.target.dead or not self:hasLOS(eff.target.x, eff.target.y, "block_move") then + return + end + + if eff.target.dead then eff.isSevered = true - + else + local t = self:getTalentFromId(self.T_DARK_VISION) + if t then + if not t.hasLOS(self.x, self.y, eff.target.x, eff.target.y) then + eff.isSevered = true + end + else + if not self:hasLOS(eff.target.x, eff.target.y) then + eff.isSevered = true + end + end + end + + if eff.isSevered then if eff.particles then -- remove old particle emitter eff.particles.x = nil -- GitLab