diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua index d066e80db075ad591ba24b67f9ef94440b18d74f..0ae3c4446561c1e3e93dc9df3323c54a83b3714a 100644 --- a/game/modules/tome/class/Actor.lua +++ b/game/modules/tome/class/Actor.lua @@ -360,6 +360,7 @@ function _M:useEnergy(val) end end +-- Called at the start of a turn before the actor chooses their action, energy is handled, etc function _M:actBase() -- Stupid sanity check that is actualy useful if self.life ~= self.life then self.life = self.max_life end @@ -2163,7 +2164,10 @@ function _M:onTakeHit(value, src, death_note) end if self:getPositive() >= drain then self:incPositive(- drain) - self:heal(self:combatTalentSpellDamage(self.T_SHIELD_OF_LIGHT, 5, 25), tal) + + local t = self:getTalentFromId(self.T_SHIELD_OF_LIGHT) + + self:heal(self:spellCrit(t.getHeal(self, t)), tal) end end @@ -2173,7 +2177,8 @@ function _M:onTakeHit(value, src, death_note) if tal then local sl = self:callTalent(self.T_SECOND_LIFE,"getLife") value = 0 - self.life = sl + self.life = 1 + self:heal(sl, self) game.logSeen(self, "#YELLOW#%s has been saved by a blast of positive energy!#LAST#", self.name:capitalize()) game:delayedLogDamage(tal, self, -sl, ("#LIGHT_GREEN#%d healing#LAST#"):format(sl), false) self:forceUseTalent(self.T_SECOND_LIFE, {ignore_energy=true}) @@ -2756,7 +2761,10 @@ end function _M:resetToFull() if self.dead then return end self.life = self.max_life - self.mana = self.max_mana + -- Make Disruption Shield not kill Archmages on levelup or we risk Archmages being mortal + if not (self.isTalentActive and self:isTalentActive(self.T_DISRUPTION_SHIELD)) then + self.mana = self.max_mana + end self.vim = self.max_vim self.stamina = self.max_stamina self.equilibrium = self.min_equilibrium @@ -4731,6 +4739,57 @@ function _M:removeEffectsFilter(t, nb, silent, force) return removed end +--- Randomly reduce talent cooldowns based on a filter +-- @param t the function to use as a filter on the talent definition or nil to apply to all talents on cooldown +-- @param change the amount to change the cooldown by +-- @param nb the number of times to reduce a talent cooldown +-- @param duplicate boolean representing whether the same talent can be reduced more than once +-- @return the number of times a talent cooldown was reduced +function _M:talentCooldownFilter(t, change, nb, duplicate) + nb = nb or 100000 + change = change or 1 + + local changed = 0 + local talents = {} + + -- For each talent currently on cooldown find its definition (e) and add it to another table if the filter (t) applies + for tid, cd in pairs(self.talents_cd) do + if type(t) == "function" then + local e = self:getTalentFromId(tid) + if t(e) then talents[#talents+1] = {tid, cd} end + else -- Apply to all talents on cooldown the filter isn't a function + talents[#talents+1] = {tid, cd} + end + end + + -- Pick random talents in the new table and apply the cooldown change to them + while #talents > 0 and nb > 0 do + local i = rng.range(1, #talents) + local t = talents[i] + local removed = false + + ---[[ Change the cooldown to the reduced value or mark it as off cooldown + t[2] = t[2] - change + if t[2] <= 0 then + self.talents_cd[ t[1] ] = nil + table.remove(talents, i) + removed = true + else + self.talents_cd[ t[1] ] = t[2] + end + --]] + + if not duplicate then + if not removed then table.remove(talents, i) end -- only remove if it hasn't already been removed + end + + nb = nb - 1 + changed = changed + 1 + end + + return changed +end + --- Suffocate a bit, lose air function _M:suffocate(value, src, death_message) if self:attr("no_breath") then return false, false end diff --git a/game/modules/tome/class/Game.lua b/game/modules/tome/class/Game.lua index 7202d3ac0811c91338b3bc3afb574b6669d745c0..606acc89f62987bd6d6f53a2b7601ebdfb2dcc59 100644 --- a/game/modules/tome/class/Game.lua +++ b/game/modules/tome/class/Game.lua @@ -344,8 +344,8 @@ function _M:loaded() elseif self.difficulty == self.DIFFICULTY_INSANE then zone.base_level_range = table.clone(zone.level_range, true) zone.specific_base_level.object = -10 -zone.level_range[1] - zone.level_range[1] = zone.level_range[1] * 1.7 + 5 - zone.level_range[2] = zone.level_range[2] * 1.7 + 5 + zone.level_range[1] = zone.level_range[1] * 1.5 + 5 + zone.level_range[2] = zone.level_range[2] * 1.5 + 5 elseif self.difficulty == self.DIFFICULTY_MADNESS then zone.base_level_range = table.clone(zone.level_range, true) zone.specific_base_level.object = -10 -zone.level_range[1] diff --git a/game/modules/tome/class/interface/Combat.lua b/game/modules/tome/class/interface/Combat.lua index ca69b2c2cc2ef3426b5034845b29a0a70149f2f5..d0ee9fa98a05a34725a516a1be3b50f6e91cfe61 100644 --- a/game/modules/tome/class/interface/Combat.lua +++ b/game/modules/tome/class/interface/Combat.lua @@ -576,15 +576,6 @@ function _M:attackTargetWith(target, weapon, damtype, mult, force_dam) end end end - -- Weapon of light cast - if hitted and not target.dead and self:knowTalent(self.T_WEAPON_OF_LIGHT) and self:isTalentActive(self.T_WEAPON_OF_LIGHT) then - if self:getPositive() >= 3 then - local dam = 7 + self:getTalentLevel(self.T_WEAPON_OF_LIGHT) * self:combatSpellpower(0.092) - DamageType:get(DamageType.LIGHT).projector(self, target.x, target.y, DamageType.LIGHT, dam) - self:incPositive(-3) - end - end - -- Shadow cast if hitted and not target.dead and self:knowTalent(self.T_SHADOW_COMBAT) and self:isTalentActive(self.T_SHADOW_COMBAT) and self:getMana() > 0 then local dam = 2 + self:combatTalentSpellDamage(self.T_SHADOW_COMBAT, 2, 50) diff --git a/game/modules/tome/data/birth/classes/celestial.lua b/game/modules/tome/data/birth/classes/celestial.lua index 4658924699fca4fcc500d2ec55c9c41986481cfb..f042c8f20d3f58b88ca87b433413c631738a49bf 100644 --- a/game/modules/tome/data/birth/classes/celestial.lua +++ b/game/modules/tome/data/birth/classes/celestial.lua @@ -69,6 +69,7 @@ newBirthDescriptor{ stats = { mag=4, str=5, }, talents_types = { ["technique/shield-offense"]={true, 0.1}, + ["technique/2hweapon-assault"]={true, 0.1}, ["technique/combat-techniques-active"]={false, 0.1}, ["technique/combat-techniques-passive"]={true, 0.1}, ["technique/combat-training"]={true, 0.1}, @@ -79,7 +80,7 @@ newBirthDescriptor{ ["celestial/light"]={true, 0.3}, ["celestial/guardian"]={false, 0.3}, ["celestial/radiance"]={false, 0.3}, - ["celestial/crusader"]={true, 0.3}, + ["celestial/crusader"]={false, 0.3}, }, birth_example_particles = "golden_shield", talents = { diff --git a/game/modules/tome/data/birth/classes/wilder.lua b/game/modules/tome/data/birth/classes/wilder.lua index 087f3f70d3387880f355bba81cba2788e7e7791e..ec526bbdb81f9356566c28f7f3ed9b16192ca282 100644 --- a/game/modules/tome/data/birth/classes/wilder.lua +++ b/game/modules/tome/data/birth/classes/wilder.lua @@ -137,7 +137,7 @@ newBirthDescriptor{ ["wild-gift/fungus"]={true, 0.1}, ["cunning/survival"]={false, 0}, ["technique/shield-offense"]={true, 0.1}, - ["technique/2hweapon-offense"]={true, 0.1}, + ["technique/2hweapon-assault"]={true, 0.1}, ["technique/combat-techniques-active"]={false, 0}, ["technique/combat-techniques-passive"]={true, 0}, ["technique/combat-training"]={true, 0}, diff --git a/game/modules/tome/data/birth/descriptors.lua b/game/modules/tome/data/birth/descriptors.lua index 1467261ca1ac27f772ccc51af3ba91a2f6a560e3..456c525ec3336a2ae7d218a0abc411d3299f6d88 100644 --- a/game/modules/tome/data/birth/descriptors.lua +++ b/game/modules/tome/data/birth/descriptors.lua @@ -211,10 +211,12 @@ newBirthDescriptor{ desc = { "#GOLD##{bold}#Insane mode#WHITE##{normal}#", - "Absolutely unfair game setting. You are really mentally ill to play this mode!", - "All zone levels increased by 120% + 5", - "All creature talent levels increased by 100%", - "Bosses will have randomly selected talents", + "Similar rules to Nightmare, but with more random bosses!", + "All zone levels increased by 50% + 5", + "All creature talent levels increased by 50%", + "Rare creatures are far more frequent and random bosses start to appear", + "Nonrandom bosses will have randomly selected talents", + "All enemies have 20% more life", "Player rank is normal instead of elite", "Player can earn Insane version of achievements if also playing in Roguelike or Adventure permadeath mode.", }, diff --git a/game/modules/tome/data/damage_types.lua b/game/modules/tome/data/damage_types.lua index 6b5dff5d6030db97a5761894f1547dfe063ae097..64bb9cd144113f1946161fe33472a6d8147edae5 100644 --- a/game/modules/tome/data/damage_types.lua +++ b/game/modules/tome/data/damage_types.lua @@ -327,7 +327,7 @@ setDefaultProjector(function(src, x, y, type, dam, tmp, no_martyr) end if src:attr("stunned") then - dam = dam * 0.3 + dam = dam * 0.4 print("[PROJECTOR] stunned dam", dam) end if src:attr("invisible_damage_penalty") then @@ -372,6 +372,19 @@ setDefaultProjector(function(src, x, y, type, dam, tmp, no_martyr) end end + -- Chant of Fortress, reduces damage from attackers over range 3 + if target.isTalentActive and target:isTalentActive(target.T_CHANT_OF_FORTRESS) and target:knowTalent(target.T_CHANT_OF_FORTRESS) then + if src and src.x and src.y then + -- assume instantaneous projection and check range to source + local t = target:getTalentFromId(target.T_CHANT_OF_FORTRESS) + if core.fov.distance(target.x, target.y, src.x, src.y) > 3 then + t = target:getTalentFromId(target.T_CHANT_OF_FORTRESS) + dam = dam * (100 + t.getDamageChange(target, t)) / 100 + print("[PROJECTOR] Chant of Fortress (source) dam", dam) + end + end + end + -- Psychic Projection if src.attr and src:attr("is_psychic_projection") and not game.zone.is_dream_scape then if (target.subtype and target.subtype == "ghost") or mind_linked then @@ -652,7 +665,8 @@ newDamageType{ end, death_message = {"electrocuted", "shocked", "bolted", "volted", "amped", "zapped"}, } --- Acid destroys potions + +-- Acid, few specific interactions currently aside from damage types later derived from this newDamageType{ name = "acid", type = "ACID", text_color = "#GREEN#", antimagic_resolve = true, @@ -724,6 +738,7 @@ newDamageType{ } -- Mind damage +-- Most uses of this have their damage effected by mental save and do not trigger cross tiers, ie, melee items newDamageType{ name = "mind", type = "MIND", text_color = "#YELLOW#", projector = function(src, x, y, type, dam) @@ -749,6 +764,8 @@ newDamageType{ death_message = {"psyched", "mentally tortured", "mindraped"}, } +-- Cold damage+turn energy drain, used exclusively by the Wintertide weapon +-- If you use this for something else make sure to note it has no power check or sanity check on how much turn energy is drained newDamageType{ name = "winter", type = "WINTER", projector = function(src, x, y, type, dam) @@ -827,7 +844,7 @@ newDamageType{ if target:canBe("silence") then target:setEffect(target.EFF_SILENCED, math.ceil(dam.dur), {apply_power=dam.power_check or src:combatMindpower() * 0.7}) else - game.logSeen(target, "%s resists!", target.name:capitalize()) + game.logSeen(target, "%s resists the silence!", target.name:capitalize()) end end end, @@ -862,7 +879,7 @@ newDamageType{ if target:canBe("silence") then target:setEffect(target.EFF_SILENCED, 4, {apply_power=src:combatAttack()*0.7, no_ct_effect=true}) else - game.logSeen(target, "%s resists!", target.name:capitalize()) + game.logSeen(target, "%s resists the silence!", target.name:capitalize()) end end end, @@ -1071,7 +1088,8 @@ newDamageType{ end, } --- Freezes target, checks for spellresistance +-- Freezes target, checks for spellresistance and stun resistance +-- Used on melee items but abnormally strong, not currently checking accuracy newDamageType{ name = "freeze", type = "FREEZE", projector = function(src, x, y, type, dam) @@ -1106,6 +1124,7 @@ newDamageType{ } -- Acid damage + blind chance +-- Used on melee items so check Accuracy too newDamageType{ name = "acid blind", type = "ACID_BLIND", text_color = "#GREEN#", projector = function(src, x, y, type, dam) @@ -1113,7 +1132,7 @@ newDamageType{ local target = game.level.map(x, y, Map.ACTOR) if target and rng.percent(25) then if target:canBe("blind") then - target:setEffect(target.EFF_BLINDED, 3, {src=src, apply_power=math.max(src:combatSpellpower(), src:combatMindpower())}) + target:setEffect(target.EFF_BLINDED, 3, {src=src, apply_power=math.max(src:combatAttack(), src:combatSpellpower(), src:combatMindpower())}) else game.logSeen(target, "%s resists!", target.name:capitalize()) end @@ -1415,6 +1434,7 @@ newDamageType{ } -- Spydric poison: prevents movement +-- Very special, does not have a power check newDamageType{ name = "spydric poison", type = "SPYDRIC_POISON", projector = function(src, x, y, type, dam) @@ -1819,19 +1839,50 @@ newDamageType{ end, } +-- Used by Bathe in Light, symmetric healing+shielding, damage to Undead +-- Keep an eye on this and Weapon of Light for any infinite stack shield then engage combos newDamageType{ name = "healing light", type = "HEALING_POWER", projector = function(src, x, y, type, dam) local target = game.level.map(x, y, Map.ACTOR) if target and not target:attr("undead") then - target:setEffect(target.EFF_EMPOWERED_HEALING, 1, {power=(dam/100)}) + + target:setEffect(target.EFF_EMPOWERED_HEALING, 1, {power=(dam/200)}) if dam >= 100 then target:attr("allow_on_heal", 1) end target:heal(dam, src) - if not target:hasEffect(target.EFF_DAMAGE_SHIELD) then target:setEffect(target.EFF_DAMAGE_SHIELD, 2, {power=dam * util.bound((target.healing_factor or 1), 0, 2.5)}) end + + -- If the target is shielded already then add to the shield power, else add a shield + local shield_power = dam * util.bound((target.healing_factor or 1), 0, 2.5) + if not target:hasEffect(target.EFF_DAMAGE_SHIELD) then + target:setEffect(target.EFF_DAMAGE_SHIELD, 2, {power=shield_power}) + else + -- Shields can't usually merge, so change the parameters manually + local shield = target:hasEffect(target.EFF_DAMAGE_SHIELD) + shield.power = shield.power + shield_power + target.damage_shield_absorb = target.damage_shield_absorb + shield_power + target.damage_shield_absorb_max = target.damage_shield_absorb_max + shield_power + shield.dur = math.max(2, shield.dur) + end if dam >= 100 then target:attr("allow_on_heal", -1) end elseif target then DamageType:get(DamageType.LIGHT).projector(src, x, y, DamageType.LIGHT, dam) - DamageType:get(DamageType.FIREBURN).projector(src, x, y, DamageType.FIREBURN, {dam=dam, dur=2, initial=0}) + end + end, +} + +-- Light damage+heal source, used by Radiance +newDamageType{ + name = "judgement", type = "JUDGEMENT", + projector = function(src, x, y, type, dam) + local target = game.level.map(x, y, Map.ACTOR) + if target and target ~= src then + --print("[JUDGEMENT] src ", src, "target", target, "src", src ) + DamageType:get(DamageType.LIGHT).projector(src, x, y, DamageType.LIGHT, dam) + if dam >= 100 then src:attr("allow_on_heal", 1) end + src:heal(dam / 2, src) + if dam >= 100 then src:attr("allow_on_heal", -1) end + + end end, } diff --git a/game/modules/tome/data/general/objects/special-artifacts.lua b/game/modules/tome/data/general/objects/special-artifacts.lua index 2505610f3e74d114aaf59cf5c5c600dae1cbf539..d247186b5a49f1a77b519c8d786afcf0fe8b5589 100644 --- a/game/modules/tome/data/general/objects/special-artifacts.lua +++ b/game/modules/tome/data/general/objects/special-artifacts.lua @@ -64,11 +64,11 @@ newEntity{ base = "BASE_STAFF", define_as = "TELOS_SPIRE", confusion_immune = 0.4, vim_on_crit = 6, }, - max_power = 30, power_regen = 1, + max_power = 15, power_regen = 1, use_power = { name = "turn into a corrupted losgoroth (poison, disease, cut and confusion immune; converts half damage into life drain; does not require breath", power = 30, use = function(self, who) game.logSeen(who, "%s brandishes %s, turning into a corrupted losgoroth!", who.name:capitalize(), self:getName()) - who:setEffect(who.EFF_CORRUPT_LOSGOROTH_FORM, 8, {}) + who:setEffect(who.EFF_CORRUPT_LOSGOROTH_FORM, 9, {}) return {id=true, used=true} end }, diff --git a/game/modules/tome/data/gfx/talents/weapon_of_wrath.png b/game/modules/tome/data/gfx/talents/weapon_of_wrath.png new file mode 100644 index 0000000000000000000000000000000000000000..a344eba37b40ec545d7c70da2ffe39cc69a7a963 Binary files /dev/null and b/game/modules/tome/data/gfx/talents/weapon_of_wrath.png differ diff --git a/game/modules/tome/data/talents/celestial/chants.lua b/game/modules/tome/data/talents/celestial/chants.lua index 437d9fa248f23eacb0461190c840c4647b68cfb4..ea5a78a93da14f6f49b2d419b0af54da54170a28 100644 --- a/game/modules/tome/data/talents/celestial/chants.lua +++ b/game/modules/tome/data/talents/celestial/chants.lua @@ -26,6 +26,7 @@ local function cancelChants(self) end end +-- Synergizes with melee classes (escort), Weapon of Wrath, healing mod (avoid overheal > healing efficiency), and low spellpower newTalent{ name = "Chant of Fortitude", type = {"celestial/chants", 1}, @@ -39,15 +40,18 @@ newTalent{ tactical = { BUFF = 2 }, range = 10, getResists = function(self, t) return self:combatTalentSpellDamage(t, 5, 70) end, + getLifePct = function(self, t) return math.min(0.55, self:combatTalentScale(t, 0.05, 0.25, 1)) end, getDamageOnMeleeHit = function(self, t) return self:combatTalentSpellDamage(t, 5, 25) end, activate = function(self, t) cancelChants(self) local power = t.getResists(self, t) game:playSoundNear(self, "talents/spell_generic2") + local ret = { onhit = self:addTemporaryValue("on_melee_hit", {[DamageType.LIGHT]=t.getDamageOnMeleeHit(self, t)}), phys = self:addTemporaryValue("combat_physresist", power), spell = self:addTemporaryValue("combat_spellresist", power), + life = self:addTemporaryValue("max_life", t.getLifePct(self, t)*self.max_life), particle = self:addParticles(Particles.new("golden_shield", 1)) } return ret @@ -57,19 +61,24 @@ newTalent{ self:removeTemporaryValue("on_melee_hit", p.onhit) self:removeTemporaryValue("combat_physresist", p.phys) self:removeTemporaryValue("combat_spellresist", p.spell) + self:removeTemporaryValue("max_life", p.life) return true end, info = function(self, t) local saves = t.getResists(self, t) + local life = t.getLifePct(self, t) local damageonmeleehit = t.getDamageOnMeleeHit(self, t) - return ([[You chant the glory of the Sun, granting you %d Physical Save and Spell Save. + return ([[You chant the glory of the Sun, granting you %d Physical Save and Spell Save and increasing your maximum life by %d%% (Currently: %d). In addition, this talent surrounds you with a shield of light, dealing %0.2f light damage to anything that attacks you. You may only have one Chant active at once. - The effects will increase with your Spellpower.]]): - format(saves, damDesc(self, DamageType.LIGHT, damageonmeleehit)) + The saves and light damage will increase with your Spellpower and the life with talent level.]]): + format(saves, life*100, life*self.max_life, damDesc(self, DamageType.LIGHT, damageonmeleehit)) end, } +-- Mostly the same code as Sanctuary +-- Just like Fortress we limit the interaction with spellpower a bit because this is an escort reward +-- This can be swapped to reactively with a projectile already in the air newTalent{ name = "Chant of Fortress", type = {"celestial/chants", 2}, @@ -82,14 +91,15 @@ newTalent{ dont_provide_pool = true, tactical = { BUFF = 2 }, range = 10, - getPhysicalResistance = function(self, t) return self:combatTalentSpellDamage(t, 5, 23) end, getDamageOnMeleeHit = function(self, t) return self:combatTalentSpellDamage(t, 5, 25) end, + getDamageChange = function(self, t) + return math.max(-40, -math.sqrt(self:getTalentLevel(t)) * 14) + end, activate = function(self, t) cancelChants(self) game:playSoundNear(self, "talents/spell_generic2") local ret = { onhit = self:addTemporaryValue("on_melee_hit", {[DamageType.LIGHT]=t.getDamageOnMeleeHit(self, t)}), - phys = self:addTemporaryValue("resists", {[DamageType.PHYSICAL] = t.getPhysicalResistance(self, t)}), particle = self:addParticles(Particles.new("golden_shield", 1)) } return ret @@ -97,20 +107,22 @@ newTalent{ deactivate = function(self, t, p) self:removeParticles(p.particle) self:removeTemporaryValue("on_melee_hit", p.onhit) - self:removeTemporaryValue("resists", p.phys) return true end, info = function(self, t) - local physicalresistance = t.getPhysicalResistance(self, t) + local range = -t.getDamageChange(self, t) local damageonmeleehit = t.getDamageOnMeleeHit(self, t) - return ([[You chant the glory of the Sun, granting you %d%% physical damage resistance. + return ([[You chant the glory of the Sun, reducing the damage enemies 3 or more spaces away deal by %d%%. In addition, this talent surrounds you with a shield of light, dealing %0.2f light damage to anything that attacks you. You may only have one Chant active at once. - The effects will increase with your Spellpower.]]): - format(physicalresistance, damDesc(self, DamageType.LIGHT, damageonmeleehit)) + The damage reduction will increase with talent level and light damage will increase with your Spellpower.]]): + format(range, damDesc(self, DamageType.LIGHT, damageonmeleehit)) end, } +-- Escorts can't give this one so it should have the most significant spellpower scaling +-- Ideally at high spellpower this would almost always be the best chant to use, but we can't guarantee that while still differentiating the chants in interesting ways +-- People that don't want to micromanage/math out when the other chants are better will like this and it should still outperform Fortitude most of the time newTalent{ name = "Chant of Resistance", type = {"celestial/chants",3}, @@ -123,7 +135,7 @@ newTalent{ tactical = { BUFF = 2 }, no_energy = true, range = 10, - getResists = function(self, t) return self:combatTalentSpellDamage(t, 5, 20) end, + getResists = function(self, t) return math.min(35, self:combatTalentSpellDamage(t, 5, 30)) end, getDamageOnMeleeHit = function(self, t) return self:combatTalentSpellDamage(t, 5, 25) end, activate = function(self, t) cancelChants(self) @@ -131,12 +143,7 @@ newTalent{ game:playSoundNear(self, "talents/spell_generic2") local ret = { onhit = self:addTemporaryValue("on_melee_hit", {[DamageType.LIGHT]=t.getDamageOnMeleeHit(self, t)}), - res = self:addTemporaryValue("resists", { - [DamageType.FIRE] = power, - [DamageType.LIGHTNING] = power, - [DamageType.ACID] = power, - [DamageType.COLD] = power, - }), + res = self:addTemporaryValue("resists", {all = power}), particle = self:addParticles(Particles.new("golden_shield", 1)) } return ret @@ -150,7 +157,7 @@ newTalent{ info = function(self, t) local resists = t.getResists(self, t) local damage = t.getDamageOnMeleeHit(self, t) - return ([[You chant the glory of the Sun, granting you %d%% fire, lightning, acid and cold damage resistance. + return ([[You chant the glory of the Sun, granting you %d%% resistance to all damage. In addition, this talent surrounds you with a shield of light, dealing %0.2f light damage to anything that attacks you. You may only have one Chant active at once. The effects will increase with your Spellpower.]]): @@ -158,6 +165,8 @@ newTalent{ end, } +-- Extremely niche in the name of theme +-- A defensive chant is realistically always a better choice than an offensive one but we can mitigate this by giving abnormally high value at low talent investment newTalent{ name = "Chant of Light", type = {"celestial/chants", 4}, @@ -170,7 +179,7 @@ newTalent{ dont_provide_pool = true, tactical = { BUFF = 2 }, range = 10, - getLightDamageIncrease = function(self, t) return self:combatTalentSpellDamage(t, 10, 50) end, + getLightDamageIncrease = function(self, t) return self:combatTalentSpellDamage(t, 20, 50) end, getDamageOnMeleeHit = function(self, t) return self:combatTalentSpellDamage(t, 5, 25) end, getLite = function(self, t) return math.floor(self:combatTalentScale(t, 2, 6, "log")) end, activate = function(self, t) diff --git a/game/modules/tome/data/talents/celestial/combat.lua b/game/modules/tome/data/talents/celestial/combat.lua index 8b3331d7ad02b93eadd9f6e6c9ab4d832759c876..19765dadfd582a29c640ff6dcdb7cd7754cb5b38 100644 --- a/game/modules/tome/data/talents/celestial/combat.lua +++ b/game/modules/tome/data/talents/celestial/combat.lua @@ -17,6 +17,8 @@ -- Nicolas Casalini "DarkGod" -- darkgod@te4.org +-- This concept plays well but vs. low damage levels spam bumping can make stupidly large shields +-- Leaving as is for now but will likely change somehow newTalent{ name = "Weapon of Light", type = {"celestial/combat", 1}, @@ -27,63 +29,49 @@ newTalent{ sustain_positive = 10, tactical = { BUFF = 2 }, range = 10, - getDamage = function(self, t) return 7 + self:combatSpellpower(0.092) * self:combatTalentScale(t, 1, 5) end, + getDamage = function(self, t) return 7 + self:combatSpellpower(0.092) * self:combatTalentScale(t, 1, 7) end, + getShieldFlat = function(self, t) + return t.getDamage(self, t) / 2 + end, activate = function(self, t) game:playSoundNear(self, "talents/spell_generic2") local ret = { + dam = self:addTemporaryValue("melee_project", {[DamageType.LIGHT]=t.getDamage(self, t)}), } return ret end, deactivate = function(self, t, p) + self:removeTemporaryValue("melee_project", p.dam) return true end, - info = function(self, t) - local damage = t.getDamage(self, t) - return ([[Infuse your weapon with the power of the Sun, doing %0.2f light damage at the cost of 3 positive energy for each blow dealt. - If you do not have enough positive energy, the sustain will have no effect. - The damage dealt will increase with your Spellpower.]]): - format(damDesc(self, DamageType.LIGHT, damage)) - end, -} + callbackOnMeleeAttack = function(self, t, target, hitted, crit, weapon, damtype, mult, dam) + if hitted and self:hasEffect(self.EFF_DAMAGE_SHIELD) and (self:reactionToward(target) < 0) then + -- Shields can't usually merge, so change the parameters manually + local shield = self:hasEffect(self.EFF_DAMAGE_SHIELD) + local shield_power = t.getShieldFlat(self, t) -newTalent{ - name = "Martyrdom", - type = {"celestial/combat", 2}, - require = divi_req2, - points = 5, - random_ego = "attack", - cooldown = 22, - positive = 25, - tactical = { DISABLE = 2 }, - range = 6, - reflectable = true, - requires_target = true, - getReturnDamage = function(self, t) return self:combatLimit(self:getTalentLevel(t)^.5, 100, 15, 1, 40, 2.24) end, -- Limit <100% - action = function(self, t) - local tg = {type="bolt", range=self:getTalentRange(t), talent=t} - local x, y = self:getTarget(tg) - if not x or not y then return nil end - local _ _, x, y = self:canProject(tg, x, y) - game:playSoundNear(self, "talents/spell_generic") - local target = game.level.map(x, y, Map.ACTOR) - if target then - target:setEffect(self.EFF_MARTYRDOM, 10, {src = self, power=t.getReturnDamage(self, t), apply_power=self:combatSpellpower()}) - else - return + shield.power = shield.power + shield_power + self.damage_shield_absorb = self.damage_shield_absorb + shield_power + self.damage_shield_absorb_max = self.damage_shield_absorb_max + shield_power + shield.dur = math.max(2, shield.dur) end - return true + end, info = function(self, t) - local returndamage = t.getReturnDamage(self, t) - return ([[Designate a target as a martyr for 10 turns. When the martyr deals damage, it also damages itself for %d%% of the damage dealt.]]): - format(returndamage) + local damage = t.getDamage(self, t) + local shieldflat = t.getShieldFlat(self, t) + return ([[Infuse your weapon with the power of the Sun, adding %0.2f light damage on each melee hit. + Additionally, if you have a temporary damage shield active, melee attacks will increase its power by %d. + The damage dealt and shield bonus will increase with your Spellpower.]]): + format(damDesc(self, DamageType.LIGHT, damage), shieldflat) end, } +-- Boring scaling, TL4 effect? newTalent{ name = "Wave of Power", - type = {"celestial/combat",3}, - require = divi_req3, + type = {"celestial/combat",2}, + require = divi_req2, points = 5, random_ego = "attack", cooldown = 6, @@ -91,9 +79,9 @@ newTalent{ tactical = { ATTACK = 2 }, requires_target = true, range = function(self, t) return 2 + math.max(0, self:combatStatScale("str", 0.8, 8)) end, - getDamage = function(self, t) return self:combatTalentWeaponDamage(t, 1.1, 1.9) end, + getDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.9, 2.5) end, action = function(self, t) - local tg = {type="bolt", range=self:getTalentRange(t), talent=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 _ _, x, y = self:canProject(tg, x, y) @@ -113,31 +101,87 @@ newTalent{ end, } +-- Interesting interactions with shield timing, lots of synergy and antisynergy in general newTalent{ - name = "Crusade", - type = {"celestial/combat", 4}, - require = divi_req4, - random_ego = "attack", + name = "Weapon of Wrath", + type = {"celestial/combat", 3}, + mode = "sustained", + require = divi_req3, points = 5, cooldown = 10, - positive = 10, - tactical = { ATTACK = {LIGHT = 2} }, - range = 1, - requires_target = true, - getDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.8, 1.6) end, - action = function(self, t) - 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 - if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end - self:attackTarget(target, DamageType.LIGHT, t.getDamage(self, t), true) - self:attackTarget(target, DamageType.LIGHT, t.getDamage(self, t), true) + sustain_positive = 10, + tactical = { BUFF = 2 }, + range = 10, + getMartyrDamage = function(self, t) return self:combatTalentScale(t, 5, 25) end, + getLifeDamage = function(self, t) return self:combatTalentScale(t, 0.1, 0.8) end, + activate = function(self, t) + game:playSoundNear(self, "talents/spell_generic2") + -- Is this any better than having the callback call getLifeDamage? I figure its better to calculate it once + local ret = { + martyr = self:addTemporaryValue("weapon_of_wrath_martyr", t.getMartyrDamage(self, t)), + damage = self:addTemporaryValue("weapon_of_wrath_life", t.getLifeDamage(self, t)), + } + return ret + end, + deactivate = function(self, t, p) + self:removeTemporaryValue("weapon_of_wrath_martyr", p.martyr) + self:removeTemporaryValue("weapon_of_wrath_life", p.damage) return true end, + callbackOnMeleeAttack = function(self, t, target, hitted, crit, weapon, damtype, mult, dam) + if hitted and self:attr("weapon_of_wrath_martyr") and not self.turn_procs.weapon_of_wrath and not target.dead then + target:setEffect(target.EFF_MARTYRDOM, 4, {power = self:attr("weapon_of_wrath_martyr")}) + + local damage = self:attr("weapon_of_wrath_life") * (self.max_life - math.max(0, self.life)) -- avoid problems with die_at + if damage == 0 then return end + damage = math.min(300, damage) -- No need to try to scale this in a clever way, NPC HP is too variant + + local tg = {type="hit", range=10, selffire=true, talent=t} + self:project(tg, target.x, target.y, DamageType.FIRE, damage) + + self.turn_procs.weapon_of_wrath = true + end + end, info = function(self, t) - local damage = t.getDamage(self, t) - return ([[Concentrate the power of the Sun into two blows; each blow does %d%% of your weapon damage as light damage.]]): - format(100 * damage) + local martyr = t.getMartyrDamage(self, t) + local damagepct = t.getLifeDamage(self, t) + local damage = damagepct * (self.max_life - math.max(0, self.life)) + return ([[Your weapon attacks burn with righteous fury dealing %d%% of your lost HP (Current: %d) fire damage up to 300 damage and causing your target to take %d%% of the damage they deal.]]): + format(damagepct*100, damage, martyr) + end, +} + +-- Core class defense to be compared with Bone Shield, Aegis, Indiscernable Anatomy, etc +-- !H/Shield could conceivably reactivate this in the same fight with Crusade spam if it triggers with Suncloak up, 2H never will without running +newTalent{ + name = "Second Life", + type = {"celestial/combat", 4}, + require = divi_req4, no_sustain_autoreset = true, + points = 5, + mode = "sustained", + sustain_positive = 30, + cooldown = 30, + tactical = { DEFEND = 2 }, + getLife = function(self, t) return self.max_life * self:combatTalentLimit(t, 1.5, 0.09, 0.4) end, -- Limit < 150% max life (to survive a large string of hits between turns) + activate = function(self, t) + game:playSoundNear(self, "talents/heal") + local ret = {} + if core.shader.active(4) then + ret.particle = self:addParticles(Particles.new("shader_ring_rotating", 1, {toback=true, a=0.6, rotation=0, radius=2, img="flamesgeneric"}, {type="sunaura", time_factor=6000})) + else + ret.particle = self:addParticles(Particles.new("golden_shield", 1)) + end + return ret + end, + deactivate = function(self, t, p) + self:removeParticles(p.particle) + return true + end, + info = function(self, t) + return ([[Any attack that would drop you below 1 hit point instead triggers Second Life, deactivating the talent, setting your hit points to 1, then healing you for %d.]]): + format(t.getLife(self, t)) end, } + + diff --git a/game/modules/tome/data/talents/celestial/crusader.lua b/game/modules/tome/data/talents/celestial/crusader.lua index b67d5612fd890ff93d15f05aef3d96d33800abb3..5cc1c8f42adbd94d1653c044f92ea87db7d0da85 100644 --- a/game/modules/tome/data/talents/celestial/crusader.lua +++ b/game/modules/tome/data/talents/celestial/crusader.lua @@ -17,6 +17,11 @@ -- Nicolas Casalini "DarkGod" -- darkgod@te4.org +-- NOTE: 2H may seem to have more defense than 1H/Shield at a glance but this isn't true +-- Mechanically, 2H gets bigger numbers on its defenses because they're all active, they don't do anything when you get hit at range 10 before you've taken an action unlike Retribution/Shield of Light +-- Thematically, 2H feels less defensive for the same reason--when you get hit it hurts, but you're encouraged to be up in their face fighting + +-- Part of 2H core defense to be compared with Shield of Light, Retribution, etc newTalent{ name = "Absorption Strike", type = {"celestial/crusader", 1}, @@ -27,7 +32,8 @@ newTalent{ tactical = { ATTACK = 2, DISABLE = 1 }, range = 1, requires_target = true, - getWeakness = function(self, t) return self:combatTalentScale(t, 20, 45, 0.75) end, + getWeakness = function(self, t) return self:combatTalentScale(t, 5, 20, 0.75) end, + getNumb = function(self, t) return math.min(30, self:combatTalentScale(t, 1, 15, 0.75)) end, getDamage = function(self, t) return self:combatTalentWeaponDamage(t, 1.1, 2.3) end, on_pre_use = function(self, t, silent) if not self:hasTwoHandedWeapon() then if not silent then game.logPlayer(self, "You require a two handed weapon to use this talent.") end return false end return true end, action = function(self, t) @@ -37,10 +43,12 @@ newTalent{ if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end local hit = self:attackTarget(target, nil, t.getDamage(self, t), true) if hit then + self:project({type="ball", radius=2, selffire=false}, self.x, self.y, function(px, py) local a = game.level.map(px, py, Map.ACTOR) if a then - a:setEffect(a.EFF_ABSORPTION_STRIKE, 5, {apply_power=self:combatSpellpower(), power=t.getWeakness(self, t)}) + -- No power check, this is essentially a defensive talent in debuff form. Instead of taking less damage passively 2H has to stay active, but we still want the consistency of a sustain/passive + a:setEffect(a.EFF_ABSORPTION_STRIKE, 5, {power=t.getWeakness(self, t), numb = t.getNumb(self, t)}) end end) end @@ -49,11 +57,12 @@ newTalent{ info = function(self, t) local damage = t.getDamage(self, t) return ([[You strike your foe with your two handed weapon, dealing %d%% weapon damage. - If the attack hits all foes in radius 2 will have their light resistance reduced by %d%% for 5 turns.]]): - format(100 * damage, t.getWeakness(self, t)) + If the attack hits all foes in radius 2 will have their light resistance reduced by %d%% and their damage reduced by %d%% for 5 turns.]]): + format(100 * damage, t.getWeakness(self, t), t.getNumb(self, t)) end, } +-- Part of 2H core defense to be compared with Shield of Light, Retribution, etc newTalent{ name = "Mark of Light", type = {"celestial/crusader", 2}, @@ -65,7 +74,7 @@ newTalent{ tactical = { ATTACK=0.5, DISABLE=2, HEAL=2 }, range = 1, requires_target = true, - getPower = function(self, t) return self:combatTalentScale(t, 30, 70) end, + getPower = function(self, t) return self:combatTalentScale(t, 10, 50) end, getDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.2, 0.7) end, on_pre_use = function(self, t, silent) if not self:hasTwoHandedWeapon() then if not silent then game.logPlayer(self, "You require a two handed weapon to use this talent.") end return false end return true end, action = function(self, t) @@ -96,8 +105,9 @@ newTalent{ getArmor = function(self, t) return self:combatTalentScale(t, 5, 30) end, getDamage = function(self, t) return self:combatTalentSpellDamage(t, 10, 120) end, getCrit = function(self, t) return self:combatTalentScale(t, 3, 10, 0.75) end, - getPower = function(self, t) return self:combatTalentScale(t, 5, 20) end, + getPower = function(self, t) return self:combatTalentScale(t, 5, 15) end, callbackOnCrit = function(self, t, kind, dam, chance, target) + if not self:hasTwoHandedWeapon() then return end if kind ~= "physical" or not target then return end if self.turn_procs.righteous_strength then return end self.turn_procs.righteous_strength = true @@ -116,13 +126,15 @@ newTalent{ end, } +-- Low damage, 2H Assault has plenty of AoE damage strikes, this one is fundamentally defensive or strong only if Light is scaled up +-- Part of 2H core defense to be compared with Shield of Light, Retribution, etc newTalent{ name = "Flash of the Blade", type = {"celestial/crusader", 4}, require = divi_req4, random_ego = "attack", points = 5, - cooldown = 10, + cooldown = 9, positive = 15, tactical = { ATTACKAREA = {LIGHT = 2} }, range = 0, @@ -132,8 +144,8 @@ newTalent{ return {type="ball", range=self:getTalentRange(t), selffire=false, radius=self:getTalentRadius(t)} end, on_pre_use = function(self, t, silent) if not self:hasTwoHandedWeapon() then if not silent then game.logPlayer(self, "You require a two handed weapon to use this talent.") end return false end return true end, - get1Damage = function(self, t) return self:combatTalentWeaponDamage(t, 0.8, 1.6) end, - get2Damage = function(self, t) return self:combatTalentWeaponDamage(t, 0.5, 1.2) end, + get1Damage = function(self, t) return self:combatTalentWeaponDamage(t, 0.5, 1.3) end, + get2Damage = function(self, t) return self:combatTalentWeaponDamage(t, 0.3, 1.5) end, action = function(self, t) local tg1 = self:getTalentTarget(t) tg1.radius = 1 local tg2 = self:getTalentTarget(t) @@ -143,16 +155,18 @@ newTalent{ self:attackTarget(target, nil, t.get1Damage(self, t), true) end end) + self:project(tg2, self.x, self.y, function(px, py, tg, self) local target = game.level.map(px, py, Map.ACTOR) if target and target ~= self then self:attackTarget(target, DamageType.LIGHT, t.get2Damage(self, t), true) - if self:getTalentLevel(t) >= 4 then - target:setEffect(target.EFF_DAZED, 3, {apply_power=self:combatSpellpower()}) - end end end) + if self:getTalentLevel(t) >= 4 then + self:setEffect(self.EFF_FLASH_SHIELD, 1, {}) + end + self:addParticles(Particles.new("meleestorm", 2, {radius=2, img="spinningwinds_yellow"})) self:addParticles(Particles.new("meleestorm", 1, {img="spinningwinds_yellow"})) return true @@ -161,7 +175,7 @@ newTalent{ return ([[Infuse your two handed weapon with light while spinning around. All creatures in radius one take %d%% weapon damage. In addition while spinning your weapon shines so much it deals %d%% light weapon damage to all foes in radius 2. - At level 4 creatures may also be dazed by the light.]]): + At level 4 your mystical, manly display of spinning around creates a manly shield that blocks all damage for 1 turn.]]): format(t.get1Damage(self, t) * 100, t.get2Damage(self, t) * 100) end, } diff --git a/game/modules/tome/data/talents/celestial/guardian.lua b/game/modules/tome/data/talents/celestial/guardian.lua index cc83d9702d84fd9460168b6a95089b7085d38513..d3d93874332986fc1b16dc81155b92e4e0730613 100644 --- a/game/modules/tome/data/talents/celestial/guardian.lua +++ b/game/modules/tome/data/talents/celestial/guardian.lua @@ -17,6 +17,10 @@ -- Nicolas Casalini "DarkGod" -- darkgod@te4.org +-- Core offensive scaler for 1H/S as we have no Shield Mastery +-- Core defense roughly to be compared with Absorption Strike, but in truth 1H/S gets a lot of its defense from cooldown management+Suncloak/etc +-- Its important that this can crit but its also spamming the combat log, unsure of solution +-- Flag if its a crit once for each turn then calculate damage manually? newTalent{ name = "Shield of Light", type = {"celestial/guardian", 1}, @@ -27,7 +31,8 @@ newTalent{ sustain_positive = 10, tactical = { BUFF = 2 }, range = 10, - getHeal = function(self, t) return self:combatTalentSpellDamage(t, 5, 25) end, + getHeal = function(self, t) return self:combatTalentSpellDamage(t, 5, 18) end, + getShieldDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.1, 0.8, self:getTalentLevel(self.T_SHIELD_EXPERTISE)) end, activate = function(self, t) local shield = self:hasShield() if not shield then @@ -43,26 +48,35 @@ newTalent{ deactivate = function(self, t, p) return true end, + callbackOnMeleeAttack = function(self, t, target, hitted, crit, weapon, damtype, mult, dam) + if hitted and not target.dead and weapon and not self.turn_procs.shield_of_light then + self:attackTargetWith(target, weapon.special_combat, DamageType.LIGHT, t.getShieldDamage(self, t)) + self.turn_procs.shield_of_light = true + end + end, info = function(self, t) local heal = t.getHeal(self, t) return ([[Infuse your shield with light, healing you for %0.2f each time you take damage at the expense of up to 2 positive energy. If you do not have any positive energy, the effect will not trigger. + Additionally, once per turn successful melee attacks will trigger a bonus attack with your shield dealing %d%% light damage. The healing done will increase with your Spellpower.]]): - format(heal) + format(heal, t.getShieldDamage(self, t)*100) end, } +-- Shield of Light means 1H/Shield builds actually care about positive energy, so we can give this a meaningful cost and power +-- Spamming Crusade+whatever is always more energy efficient than this newTalent{ name = "Brandish", type = {"celestial/guardian", 2}, require = divi_req_high2, points = 5, cooldown = 8, - positive = 15, + positive = 25, tactical = { ATTACK = {LIGHT = 2} }, requires_target = true, - getWeaponDamage = function(self, t) return self:combatTalentWeaponDamage(t, 1, 1.5) end, - getShieldDamage = function(self, t) return self:combatTalentWeaponDamage(t, 1, 1.5, self:getTalentLevel(self.T_SHIELD_EXPERTISE)) end, + getWeaponDamage = function(self, t) return self:combatTalentWeaponDamage(t, 1, 3) end, + getShieldDamage = function(self, t) return self:combatTalentWeaponDamage(t, 1, 2, self:getTalentLevel(self.T_SHIELD_EXPERTISE)) end, getLightDamage = function(self, t) return self:combatTalentSpellDamage(t, 20, 200) end, radius = function(self, t) return math.floor(self:combatTalentScale(t, 2.5, 4.5)) end, action = function(self, t) @@ -87,7 +101,7 @@ newTalent{ local tg = {type="ball", range=1, selffire=true, radius=self:getTalentRadius(t), talent=t} self:project(tg, x, y, DamageType.LITE, 1) tg.selffire = false - local grids = self:project(tg, x, y, DamageType.LIGHT, t.getLightDamage(self, t)) + local grids = self:project(tg, x, y, DamageType.LIGHT, self:spellCrit(t.getLightDamage(self, t))) game.level.map:particleEmitter(x, y, tg.radius, "sunburst", {radius=tg.radius, grids=grids, tx=x, ty=y, max_alpha=80}) game:playSoundNear(self, "talents/flame") end @@ -136,41 +150,70 @@ newTalent{ self.retribution_strike = nil return true end, + callbackOnRest = function(self, t) + -- Resting requires no enemies in view so we can safely clear all stored damage + self.retribution_absorb = 0 + self.retribution_strike = 0 + end, info = function(self, t) local damage = t.getDamage(self, t) + local absorb_string = "" + if self.retribution_absorb and self.retribution_strike then + absorb_string = ([[#RED#Absorb Remaining: %d]]):format(self.retribution_absorb) + end + return ([[Retribution negates half of all damage you take while it is active. Once Retribution has negated %0.2f damage, your shield will explode in a burst of light, inflicting damage equal to the amount negated in a radius of %d and deactivating the talent. - The amount absorbed will increase with your Spellpower.]]): - format(damage, self:getTalentRange(t)) + The amount absorbed will increase with your Spellpower. + %s]]): + format(damage, self:getTalentRange(t), absorb_string) end, } +-- Moderate damage but very short CD +-- Spamming this on cooldown keeps positive energy up and gives a lot of cooldown management newTalent{ - name = "Second Life", + name = "Crusade", type = {"celestial/guardian", 4}, - require = divi_req_high4, no_sustain_autoreset = true, + require = divi_req_high4, + random_ego = "attack", points = 5, - mode = "sustained", - sustain_positive = 60, - cooldown = 50, - tactical = { DEFEND = 2 }, - getLife = function(self, t) return self.max_life * self:combatTalentLimit(t, 1.5, 0.09, 0.25) end, -- Limit < 150% max life (to survive a large string of hits between turns) - activate = function(self, t) - game:playSoundNear(self, "talents/heal") - local ret = {} - if core.shader.active(4) then - ret.particle = self:addParticles(Particles.new("shader_ring_rotating", 1, {toback=true, a=0.6, rotation=0, radius=2, img="flamesgeneric"}, {type="sunaura", time_factor=6000})) - else - ret.particle = self:addParticles(Particles.new("golden_shield", 1)) + cooldown = 5, + positive = -20, + tactical = { ATTACK = {LIGHT = 2} }, + range = 1, + requires_target = true, + getWeaponDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.3, 1.2) end, + getShieldDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.3, 1.2, self:getTalentLevel(self.T_SHIELD_EXPERTISE)) end, + getCooldownReduction = function(self, t) return math.ceil(self:combatTalentScale(t, 1, 3)) end, + getDebuff = function(self, t) return 1 end, + action = function(self, t) + local shield = self:hasShield() + if not shield then + game.logPlayer(self, "You cannot use Crusade without a shield!") + return nil end - return ret - end, - deactivate = function(self, t, p) - self:removeParticles(p.particle) + 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 + if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end + + local hit = self:attackTarget(target, DamageType.LIGHT, t.getWeaponDamage(self, t), true) + if hit then self:talentCooldownFilter(nil, 1, t.getCooldownReduction(self, t), true) end + + local hit2 = self:attackTargetWith(target, shield.special_combat, DamageType.LIGHT, t.getShieldDamage(self, t)) + if hit2 then self:removeEffectsFilter({status = "detrimental"}, t.getDebuff(self, t)) end + return true end, info = function(self, t) - return ([[Any attack that would drop you below 1 hit point instead triggers Second Life, deactivating the talent and setting your hit points to %d.]]): - format(t.getLife(self, t)) + local weapon = t.getWeaponDamage(self, t)*100 + local shield = t.getShieldDamage(self, t)*100 + local cooldown = t.getCooldownReduction(self, t) + local cleanse = t.getDebuff(self, t) + return ([[You demonstrate your dedication to the light with a measured attack striking once with your weapon for %d%% damage and once with your shield for %d%% damage. + If the first strike connects %d random talent cooldowns are reduced by 1. + If the second strike connects you are cleansed of %d debuffs.]]): + format(weapon, shield, cooldown, cleanse) end, } diff --git a/game/modules/tome/data/talents/celestial/light.lua b/game/modules/tome/data/talents/celestial/light.lua index 7cd8501b0eb1dcf2f6f31185e469a841e7a61bd3..e37714927b2d8d51d412f4ca873e5a39ce2df5ff 100644 --- a/game/modules/tome/data/talents/celestial/light.lua +++ b/game/modules/tome/data/talents/celestial/light.lua @@ -53,23 +53,23 @@ newTalent{ require = spells_req2, random_ego = "defensive", points = 5, - cooldown = 10, + cooldown = 15, positive = -20, tactical = { HEAL = 3 }, range = 0, - radius = 3, + radius = 2, target = function(self, t) return {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t)} end, - getHeal = function(self, t) return self:combatTalentSpellDamage(t, 4, 40) end, - getDuration = function(self, t) return math.floor(self:combatTalentScale(t, 3, 7)) end, + getHeal = function(self, t) return self:combatTalentSpellDamage(t, 4, 80) end, + getDuration = function(self, t) return math.floor(self:combatTalentScale(t, 4, 7)) end, action = function(self, t) local tg = self:getTalentTarget(t) self:project(tg, self.x, self.y, DamageType.LITE, 1) -- Add a lasting map effect game.level.map:addEffect(self, self.x, self.y, t.getDuration(self, t), - DamageType.HEALING_POWER, t.getHeal(self, t), + DamageType.HEALING_POWER, self:spellCrit(t.getHeal(self, t)), self:getTalentRadius(t), 5, nil, {type="healing_vapour"}, @@ -85,7 +85,7 @@ newTalent{ return ([[A magical zone of Sunlight appears around you, healing and shielding all within radius %d for %0.2f per turn and increasing healing effects on everyone within by %d%%. The effect lasts for %d turns. It also lights up the affected zone. The amount healed will increase with the Magic stat]]): - format(radius, heal, heal, duration) + format(radius, heal, heal / 2, duration) end, } @@ -98,9 +98,9 @@ newTalent{ positive = -20, cooldown = 15, tactical = { DEFEND = 2 }, - getAbsorb = function(self, t) return self:combatTalentSpellDamage(t, 30, 470) end, + getAbsorb = function(self, t) return self:combatTalentSpellDamage(t, 30, 370) end, action = function(self, t) - self:setEffect(self.EFF_DAMAGE_SHIELD, 10, {color={0xe1/255, 0xcb/255, 0x3f/255}, power=t.getAbsorb(self, t)}) + self:setEffect(self.EFF_DAMAGE_SHIELD, 10, {color={0xe1/255, 0xcb/255, 0x3f/255}, power=self:spellCrit(t.getAbsorb(self, t))}) game:playSoundNear(self, "talents/heal") return true end, diff --git a/game/modules/tome/data/talents/celestial/radiance.lua b/game/modules/tome/data/talents/celestial/radiance.lua index 8e3937330a2ae4206b565980bd12bca5879ad080..c4507d6a88399ab4d2d5b9ca44e1be46317ddee5 100644 --- a/game/modules/tome/data/talents/celestial/radiance.lua +++ b/game/modules/tome/data/talents/celestial/radiance.lua @@ -18,7 +18,11 @@ -- darkgod@te4.org function radianceRadius(self) - return self:getTalentRadius(self:getTalentFromId(self.T_RADIANCE)) + if self:hasEffect(self.EFF_RADIANCE_DIM) then + return 1 + else + return self:getTalentRadius(self:getTalentFromId(self.T_RADIANCE)) + end end newTalent{ @@ -59,15 +63,11 @@ newTalent{ local ss = self:isTalentActive(self.T_SEARING_SIGHT) if ss then local dist = core.fov.distance(self.x, self.y, target.x, target.y) - 1 - local coeff = math.scale(radius - dist, 1, radius, 0.1, 1) - local realdam = DamageType:get(DamageType.LIGHT).projector(self, target.x, target.y, DamageType.LIGHT, ss.dam * coeff) + local coeff = math.max(0.1, 1 - (0.1*dist)) -- 10% less damage per distance + DamageType:get(DamageType.LIGHT).projector(self, target.x, target.y, DamageType.LIGHT, ss.dam * coeff) if ss.daze and rng.percent(ss.daze) and target:canBe("stun") then target:setEffect(target.EFF_DAZED, 3, {apply_power=self:combatSpellpower()}) end - - if realdam and realdam > 0 and self:hasEffect(self.EFF_LIGHT_BURST) then - self:setEffect(self.EFF_LIGHT_BURST_SPEED, 4, {}) - end end end end end end @@ -81,6 +81,8 @@ newTalent{ end, } +-- This doesn't work well in practice.. Its powerful but it leads to cheesy gameplay, spams combat logs, maybe even lags +-- It can stay like this for now but may be worth making better newTalent{ name = "Searing Sight", type = {"celestial/radiance",3}, @@ -90,8 +92,8 @@ newTalent{ cooldown = 15, range = function(self) return radianceRadius(self) end, tactical = { ATTACKAREA = {LIGHT=1} }, - sustain_positive = 40, - getDamage = function(self, t) return self:combatTalentSpellDamage(t, 1, 90) end, + sustain_positive = 10, + getDamage = function(self, t) return self:combatTalentSpellDamage(t, 1, 50) end, getDaze = function(self, t) return self:combatTalentLimit(t, 35, 5, 20) end, activate = function(self, t) local daze = nil @@ -115,28 +117,54 @@ newTalent{ require = divi_req4, points = 5, cooldown = 25, - positive = 15, - tactical = { DISABLE = {blind=1} }, + positive = 20, + tactical = { ATTACKAREA = {LIGHT = 2} }, + radius = function(self) return radianceRadius(self) end, range = function(self) return radianceRadius(self) end, - requires_target = true, - getDur = function(self, t) return self:combatTalentLimit(t, 9, 2, 6) end, - getMax = function(self, t) return math.floor(self:combatTalentScale(t, 2, 8)) end, + getMoveDamage = function(self, t) return self:combatTalentSpellDamage(t, 1, 40) end, + getExplosionDamage = function(self, t) return self:combatTalentSpellDamage(t, 20, 150) end, action = function(self, t) - local radius = radianceRadius(self) - local grids = core.fov.circle_grids(self.x, self.y, radius, true) - for x, yy in pairs(grids) do for y, _ in pairs(grids[x]) do local target = game.level.map(x, y, Map.ACTOR) if target and self ~= target then - if target:canBe("blind") then - target:setEffect(target.EFF_BLINDED, 4, {apply_power=self:combatSpellpower()}) - end - end end end - self:setEffect(self.EFF_LIGHT_BURST, t.getDur(self, t), {max=t.getMax(self, t)}) + local tg = {type="ball", range=self:getTalentRange(t), radius = self:getTalentRadius(t), selffire = false, talent=t} + + local movedam = self:spellCrit(t.getMoveDamage(self, t)) + local dam = self:spellCrit(t.getExplosionDamage(self, t)) + + self:project(tg, self.x, self.y, function(tx, ty) + local target = game.level.map(tx, ty, engine.Map.ACTOR) + if not target then return end + + local proj = require("mod.class.Projectile"):makeHoming( + self, + {particle="bolt_light", trail="lighttrail"}, + {speed=1, name="Judgement", dam=dam, movedam=movedam}, + target, + self:getTalentRange(t), + function(self, src) + local DT = require("engine.DamageType") + DT:get(DT.JUDGEMENT).projector(src, self.x, self.y, DT.JUDGEMENT, self.def.movedam) + end, + function(self, src, target) + local DT = require("engine.DamageType") + local grids = src:project({type="ball", radius=1, x=self.x, y=self.y}, self.x, self.y, DT.JUDGEMENT, self.def.dam) + + + game.level.map:particleEmitter(self.x, self.y, 1, "sunburst", {radius=1, grids=grids, tx=self.x, ty=self.y}) + + game:playSoundNear(self, "talents/lightning") + end + ) + game.zone:addEntity(game.level, proj, "projectile", self.x, self.y) + end) + + -- EFF_RADIANCE_DIM does nothing by itself its just used by radianceRadius + self:setEffect(self.EFF_RADIANCE_DIM, 8, {}) + return true end, info = function(self, t) - return ([[Concentrate your Radiance in a blinding flash of light. All foes caught inside will be blinded for 3 turns. - In addition for %d turns each time your Searing Sight damages a foe you gain a movement bonus of 10%%, stacking up to %d times.]]): - format(t.getDur(self, t), t.getMax(self, t)) + return ([[Fire a glowing orb of light at each enemy within your Radiance. Each orb will slowly follow its target until it connects dealing %d light damage to anything else it contacts along the way. When the target is reached the orb will explode dealing %d light damage and healing you for 50%% of the damage dealt. This powerful ability will dim your Radiance reducing its radius to 1 for 5 turns.]]): + format(t.getMoveDamage(self, t), t.getExplosionDamage(self, t)) end, } diff --git a/game/modules/tome/data/talents/celestial/sun.lua b/game/modules/tome/data/talents/celestial/sun.lua index ec9584b5a9458d831d42a68df9b3b2b1cc9e9cf1..7ab7d02c3656bfdc2d8afc5aae1fa9a33deff1d9 100644 --- a/game/modules/tome/data/talents/celestial/sun.lua +++ b/game/modules/tome/data/talents/celestial/sun.lua @@ -17,13 +17,14 @@ -- Nicolas Casalini "DarkGod" -- darkgod@te4.org +-- Baseline blind because the class has a lot of trouble with CC early game and rushing TL4 isn't reasonable newTalent{ name = "Sun Beam", type = {"celestial/sun", 1}, require = divi_req1, random_ego = "attack", points = 5, - cooldown = 6, + cooldown = 9, positive = -16, range = 7, tactical = { ATTACK = {LIGHT = 2} }, @@ -34,18 +35,21 @@ newTalent{ getDamage = function(self, t) local mult = 1 if self:attr("amplify_sun_beam") then mult = 1 + self:attr("amplify_sun_beam") / 100 end - return self:combatTalentSpellDamage(t, 6, 220) * mult + return self:combatTalentSpellDamage(t, 20, 240) * mult end, getDuration = function(self, t) return math.floor(self:combatTalentScale(t, 2, 4)) end, action = function(self, t) - local tg = {type="bolt", range=self:getTalentRange(t), talent=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 self:project(tg, x, y, DamageType.LIGHT, self:spellCrit(t.getDamage(self, t)), {type="light"}) if self:getTalentLevel(t) >= 4 then local _ _, x, y = self:canProject(tg, x, y) - self:project({type="ball", x=x, y=y, radius=1, selffire=false}, x, y, DamageType.BLIND, t.getDuration(self, t), {type="light"}) + self:project({type="ball", x=x, y=y, radius=2, selffire=false}, x, y, DamageType.BLIND, t.getDuration(self, t), {type="light"}) + else + local _ _, x, y = self:canProject(tg, x, y) + self:project({type="hit", x=x, y=y, radius=0, selffire=false}, x, y, DamageType.BLIND, t.getDuration(self, t), {type="light"}) end -- Delay removal of the effect so its still there when no_energy checks @@ -58,39 +62,43 @@ newTalent{ end, info = function(self, t) local damage = t.getDamage(self, t) - return ([[Calls the a beam of light from the Sun, doing %0.2f damage to the target. - At level 4 the beam will be so intense it blinds all foes in radius 1 for %d turns. + return ([[Calls the a beam of light from the Sun, doing %0.2f damage to the target and blinding them. + At level 4 the beam will be so intense it blinds all foes in radius 2 for %d turns. The damage dealt will increase with your Spellpower.]]): format(damDesc(self, DamageType.LIGHT, damage), t.getDuration(self, t)) end, } +-- Core class defense to be compared with Bone Shield, Aegis, Indiscernable Anatomy, etc +-- Moderate offensive scaler +-- The CD reduction effects more abilities on the class than it doesn't newTalent{ name = "Suncloak", type = {"celestial/sun", 2}, require = divi_req2, points = 5, - cooldown = 15, + cooldown = 20, -- CD reduction works on this positive = -15, tactical = { BUFF = 2 }, direct_hit = true, requires_target = true, range = 10, - getResists = function(self, t) return 10 + self:combatTalentSpellDamage(t, 4, 150) / 10 end, - getDuration = function(self, t) return 20 + self:combatTalentSpellDamage(t, 4, 400) / 10 end, + getCap = function(self, t) return math.max(50, 100 - self:getTalentLevelRaw(t) * 10) end, + getHaste = function(self, t) return math.min(1, self:combatTalentSpellDamage(t, 0.2, 0.8)) end, action = function(self, t) - self:setEffect(self.EFF_SUNCLOAK, 5, {reduce=t.getDuration(self, t), resists=t.getResists(self, t)}) + self:setEffect(self.EFF_SUNCLOAK, 6, {cap=t.getCap(self, t), haste=t.getHaste(self, t)}) game:playSoundNear(self, "talents/flame") return true end, info = function(self, t) - return ([[You wrap yourself in a cloak of sunlight for 5 turns. - While the cloak is active all new detrimental temporary effects have their duration reduced by %d%% and all your resistances are increased by %d%%. + return ([[You wrap yourself in a cloak of sunlight that empowers your magic and protects you for 6 turns. + While the cloak is active your spell casting speed is increased by %d%%, your spell cooldowns are reduced by %d%%, and you cannot take more than %d%% of your maximum life from a single blow. The effects will increase with your Spellpower.]]): - format(t.getDuration(self, t), t.getResists(self, t)) + format(t.getHaste(self, t)*100, t.getHaste(self, t)*100, t.getCap(self, t)) end, } +-- Can someone put a really obvious visual on this? newTalent{ name = "Sun's Vengeance", short_name = "SUN_VENGEANCE", type = {"celestial/sun",3}, diff --git a/game/modules/tome/data/talents/gifts/storm-drake.lua b/game/modules/tome/data/talents/gifts/storm-drake.lua index 48c1e8d257e3c285324d0a11a28e386f0335cffc..5940fe40937e28933b1e4a81c515224cc205459b 100644 --- a/game/modules/tome/data/talents/gifts/storm-drake.lua +++ b/game/modules/tome/data/talents/gifts/storm-drake.lua @@ -29,6 +29,7 @@ newTalent{ range = 10, tactical = { CLOSEIN = 2, ESCAPE = 2 }, requires_target = true, + no_energy = true, getSpeed = function(self, t) return self:combatTalentScale(t, 470, 750, 0.75) end, getDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 1.1, 2.6)) end, on_learn = function(self, t) self.resists[DamageType.LIGHTNING] = (self.resists[DamageType.LIGHTNING] or 0) + 1 end, diff --git a/game/modules/tome/data/talents/misc/npcs.lua b/game/modules/tome/data/talents/misc/npcs.lua index 370521a0fa935f97ebacacc6eb5bfe3168412f89..41ea8c596cb041a41ad0c8c530349ced9fa3f244 100644 --- a/game/modules/tome/data/talents/misc/npcs.lua +++ b/game/modules/tome/data/talents/misc/npcs.lua @@ -2245,4 +2245,84 @@ newTalent{ return ([[Each time one of your foes bites the dust, you feel a surge of power, increasing your strength by 2 up to a maximum of %d for %d turns.]]): format(math.floor(self:getTalentLevel(t) * 6), t.getDuration(self, t)) end, +} + +newTalent{ + name = "Martyrdom", + type = {"spell/other", 1}, + points = 5, + random_ego = "attack", + cooldown = 22, + positive = 25, + tactical = { DISABLE = 2 }, + range = 6, + reflectable = true, + requires_target = true, + getReturnDamage = function(self, t) return self:combatLimit(self:getTalentLevel(t)^.5, 100, 15, 1, 40, 2.24) end, -- Limit <100% + action = function(self, t) + local tg = {type="bolt", range=self:getTalentRange(t), talent=t} + local x, y = self:getTarget(tg) + if not x or not y then return nil end + local _ _, x, y = self:canProject(tg, x, y) + game:playSoundNear(self, "talents/spell_generic") + local target = game.level.map(x, y, Map.ACTOR) + if target then + target:setEffect(self.EFF_MARTYRDOM, 10, {src = self, power=t.getReturnDamage(self, t), apply_power=self:combatSpellpower()}) + else + return + end + return true + end, + info = function(self, t) + local returndamage = t.getReturnDamage(self, t) + return ([[Designate a target as a martyr for 10 turns. When the martyr deals damage, it also damages itself for %d%% of the damage dealt.]]): + format(returndamage) + end, +} + +newTalent{ + name = "Overpower", + type = {"technique/other", 1}, + points = 5, + random_ego = "attack", + cooldown = 8, + stamina = 22, + requires_target = true, + tactical = { ATTACK = 2, ESCAPE = { knockback = 1 }, DISABLE = { knockback = 1 } }, + on_pre_use = function(self, t, silent) if not self:hasShield() then if not silent then game.logPlayer(self, "You require a weapon and a shield to use this talent.") end return false end return true end, + action = function(self, t) + local shield = self:hasShield() + if not shield then + game.logPlayer(self, "You cannot use Overpower without a shield!") + return nil + end + + 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 + if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end + + -- First attack with weapon + self:attackTarget(target, nil, self:combatTalentWeaponDamage(t, 0.8, 1.3), true) + -- Second attack with shield + self:attackTargetWith(target, shield.special_combat, nil, self:combatTalentWeaponDamage(t, 0.8, 1.3, self:getTalentLevel(self.T_SHIELD_EXPERTISE))) + -- Third attack with shield + local speed, hit = self:attackTargetWith(target, shield.special_combat, nil, self:combatTalentWeaponDamage(t, 0.8, 1.3, self:getTalentLevel(self.T_SHIELD_EXPERTISE))) + + -- Try to stun ! + if hit then + if target:checkHit(self:combatAttack(shield.special_combat), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then + target:knockback(self.x, self.y, 4) + else + game.logSeen(target, "%s resists the knockback!", target.name:capitalize()) + end + end + + return true + end, + info = function(self, t) + return ([[Hits the target with your weapon doing %d%% damage and two shield strikes doing %d%% damage, trying to overpower your target. + If the last attack hits, the target is knocked back. The chance for knockback increases with your Accuracy.]]) + :format(100 * self:combatTalentWeaponDamage(t, 0.8, 1.3), 100 * self:combatTalentWeaponDamage(t, 0.8, 1.3, self:getTalentLevel(self.T_SHIELD_EXPERTISE))) + end, } \ No newline at end of file diff --git a/game/modules/tome/data/talents/techniques/2h-assault.lua b/game/modules/tome/data/talents/techniques/2h-assault.lua index ad3aa62edf02495e204b6f60cd5bf7e1baf61d64..a9a8bdc3874198c97d975b5ce81033a14af28889 100644 --- a/game/modules/tome/data/talents/techniques/2h-assault.lua +++ b/game/modules/tome/data/talents/techniques/2h-assault.lua @@ -152,6 +152,7 @@ newTalent{ end, } +-- FIX SCALING newTalent{ name = "Execution", type = {"technique/2hweapon-assault", 4}, @@ -173,10 +174,10 @@ newTalent{ if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end local perc = 1 - (target.life / target.max_life) - local power = t.getPower() - + local power = t.getPower(self, t) + game.logPlayer(self, "perc " .. perc .. " power " .. power) self.turn_procs.auto_phys_crit = true - local speed, hit = self:attackTargetWith(target, weapon.combat, nil, power * perc * 100) + local speed, hit = self:attackTargetWith(target, weapon.combat, nil, power * perc) self.turn_procs.auto_phys_crit = nil return true end, diff --git a/game/modules/tome/data/talents/techniques/weaponshield.lua b/game/modules/tome/data/talents/techniques/weaponshield.lua index 57198ec94c2c061b5ca9d2df383754dc1f130f51..e265579231b30080f49912c5d7904cff214b1e4f 100644 --- a/game/modules/tome/data/talents/techniques/weaponshield.lua +++ b/game/modules/tome/data/talents/techniques/weaponshield.lua @@ -90,20 +90,21 @@ newTalent{ } newTalent{ - name = "Overpower", + name = "Shield Slam", type = {"technique/shield-offense", 3}, require = techs_req3, points = 5, random_ego = "attack", - cooldown = 8, - stamina = 22, + cooldown = 10, + stamina = 15, requires_target = true, - tactical = { ATTACK = 2, ESCAPE = { knockback = 1 }, DISABLE = { knockback = 1 } }, + getShieldDamage = function(self, t) return self:combatTalentWeaponDamage(t, 0.1, 0.8, self:getTalentLevel(self.T_SHIELD_EXPERTISE)) end, + tactical = { ATTACK = 2}, -- is there a way to make this consider the free Block? on_pre_use = function(self, t, silent) if not self:hasShield() then if not silent then game.logPlayer(self, "You require a weapon and a shield to use this talent.") end return false end return true end, action = function(self, t) local shield = self:hasShield() if not shield then - game.logPlayer(self, "You cannot use Overpower without a shield!") + game.logPlayer(self, "You cannot use Shield Slam without a shield!") return nil end @@ -112,28 +113,19 @@ newTalent{ if not x or not y or not target then return nil end if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end - -- First attack with weapon - self:attackTarget(target, nil, self:combatTalentWeaponDamage(t, 0.8, 1.3), true) - -- Second attack with shield - self:attackTargetWith(target, shield.special_combat, nil, self:combatTalentWeaponDamage(t, 0.8, 1.3, self:getTalentLevel(self.T_SHIELD_EXPERTISE))) - -- Third attack with shield - local speed, hit = self:attackTargetWith(target, shield.special_combat, nil, self:combatTalentWeaponDamage(t, 0.8, 1.3, self:getTalentLevel(self.T_SHIELD_EXPERTISE))) - - -- Try to stun ! - if hit then - if target:checkHit(self:combatAttack(shield.special_combat), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then - target:knockback(self.x, self.y, 4) - else - game.logSeen(target, "%s resists the knockback!", target.name:capitalize()) - end - end + local damage = t.getShieldDamage(self, t) + self:forceUseTalent(self.T_BLOCK, {ignore_energy=true, ignore_cd = true, silent = true}) + + self:attackTargetWith(target, shield.special_combat, nil, damage) + self:attackTargetWith(target, shield.special_combat, nil, damage) + self:attackTargetWith(target, shield.special_combat, nil, damage) return true end, info = function(self, t) - return ([[Hits the target with your weapon doing %d%% damage and two shield strikes doing %d%% damage, trying to overpower your target. - If the last attack hits, the target is knocked back. The chance for knockback increases with your Accuracy.]]) - :format(100 * self:combatTalentWeaponDamage(t, 0.8, 1.3), 100 * self:combatTalentWeaponDamage(t, 0.8, 1.3, self:getTalentLevel(self.T_SHIELD_EXPERTISE))) + local damage = t.getShieldDamage(self, t)*100 + return ([[Hit your target with your shield 3 times for %d%% damage then quickly return to a blocking position. The bonus block will not check or trigger Block cooldown.]]) + :format(damage) end, } diff --git a/game/modules/tome/data/talents/uber/const.lua b/game/modules/tome/data/talents/uber/const.lua index 0589a718464b33607ef9a71a2c6398d3269eff77..24835422ed5af12f21dabf01b10a7298a2236705 100644 --- a/game/modules/tome/data/talents/uber/const.lua +++ b/game/modules/tome/data/talents/uber/const.lua @@ -20,7 +20,7 @@ uberTalent{ name = "Draconic Body", mode = "passive", - cooldown = 40, + cooldown = 20, require = { special={desc="Be close to the draconic world", fct=function(self) return game.state.birth.ignore_prodigies_special_reqs or (self:attr("drake_touched") and self:attr("drake_touched") >= 2) end} }, trigger = function(self, t, value) if self.life - value < self.max_life * 0.3 and not self:isTalentCoolingDown(t) then diff --git a/game/modules/tome/data/timed_effects/magical.lua b/game/modules/tome/data/timed_effects/magical.lua index 8241a0677284878aa87d4c0d6e2851615f8955a9..f1a9c6386049024f9b602648f479b7e9c648faaa 100644 --- a/game/modules/tome/data/timed_effects/magical.lua +++ b/game/modules/tome/data/timed_effects/magical.lua @@ -568,6 +568,22 @@ newEffect{ end, } +-- This only exists to mark a timer for Radiance being consumed +newEffect{ + name = "RADIANCE_DIM", image = "talents/curse_of_vulnerability.png", + desc = "Radiance Lost", + long_desc = function(self, eff) return ("You have expended the power of your Radiance temporarily reducing its radius to 1."):format() end, + type = "other", + subtype = { radiance=true }, + parameters = { }, + on_gain = function(self, err) return "#Target#'s aura dims.", "+Dim" end, + on_lose = function(self, err) return "#Target# shines with renewed light.", "-Dim" end, + activate = function(self, eff) + end, + deactivate = function(self, eff) + end, +} + newEffect{ name = "CURSE_VULNERABILITY", image = "talents/curse_of_vulnerability.png", desc = "Curse of Vulnerability", @@ -2474,16 +2490,17 @@ newEffect{ newEffect{ name = "SUNCLOAK", image = "talents/suncloak.png", desc = "Suncloak", - long_desc = function(self, eff) return ("The target is filled with the Sun's fury, next Sun Beam will be instant cast."):format() end, + long_desc = function(self, eff) return ("The target is protected by the sun, increasing their spell casting speed by 30%%, reducing spell cooldowns by 30%%, and preventing damage over %d%% of your maximum life from a single hit."):format(eff.cap) end, type = "magical", subtype = { sun=true, }, status = "beneficial", - parameters = {}, - on_gain = function(self, err) return "#Target# is filled with the Sun's fury!", "+Sun's Vengeance" end, - on_lose = function(self, err) return "#Target#'s solar fury subsides.", "-Sun's Vengeance" end, + parameters = {cap = 1, haste = 0.1}, + on_gain = function(self, err) return "#Target# is energized and protected by the Sun!", "+Suncloak" end, + on_lose = function(self, err) return "#Target#'s solar fury subsides.", "-Suncloak" end, activate = function(self, eff) - self:effectTemporaryValue(eff, "resists", {all=eff.resists}) - self:effectTemporaryValue(eff, "reduce_detrimental_status_effects_time", eff.reduce) + self:effectTemporaryValue(eff, "flat_damage_cap", {all=eff.cap}) + self:effectTemporaryValue(eff, "combat_spellspeed", eff.haste) + self:effectTemporaryValue(eff, "spell_cooldown_reduction", eff.haste) eff.particle = self:addParticles(Particles.new("suncloak", 1)) end, deactivate = function(self, eff) @@ -2491,21 +2508,6 @@ newEffect{ end, } -newEffect{ - name = "ABSORPTION_STRIKE", image = "talents/absorption_strike.png", - desc = "Absorption Strike", - long_desc = function(self, eff) return ("The target's light has been drained, reducing light resistance by %d%%."):format(eff.power) end, - type = "magical", - subtype = { sun=true, }, - status = "detrimental", - parameters = { power = 10 }, - on_gain = function(self, err) return "#Target# is drained from light!", "+Absorption Strike" end, - on_lose = function(self, err) return "#Target#'s light is back.", "-Absorption Strike" end, - activate = function(self, eff) - self:effectTemporaryValue(eff, "resists", {[DamageType.LIGHT]=-eff.power}) - end, -} - newEffect{ name = "MARK_OF_LIGHT", image = "talents/mark_of_light.png", desc = "Mark of Light", diff --git a/game/modules/tome/data/timed_effects/other.lua b/game/modules/tome/data/timed_effects/other.lua index d425d2f79ea6aaa8f59ad52fc61584c3c469016b..36a0db6bf0bb2652381d68210ec15d4b1b95cb9c 100644 --- a/game/modules/tome/data/timed_effects/other.lua +++ b/game/modules/tome/data/timed_effects/other.lua @@ -25,6 +25,40 @@ local Map = require "engine.Map" local Level = require "engine.Level" local Combat = require "mod.class.interface.Combat" +newEffect{ + name = "FLASH_SHIELD", image = "talents/flash_of_the_blade.png", + desc = "Protected by the Sun", + long_desc = function(self, eff) return "The Sun has granted a brief immunity to all damage." end, + type = "other", + subtype = { }, + status = "beneficial", + on_gain = function(self, err) return "#Target# whirls around and a radiant shield surrounds them!", "+Divine Shield" end, + parameters = {}, + activate = function(self, eff) + self:effectTemporaryValue(eff, "cancel_damage_chance", 100) + end, + deactivate = function(self, eff) + + end, +} + +-- type other because this is a core defensive mechanic in debuff form, it should not interact with saves +newEffect{ + name = "ABSORPTION_STRIKE", image = "talents/absorption_strike.png", + desc = "Absorption Strike", + long_desc = function(self, eff) return ("The target's light has been drained, reducing light resistance by %d%% and damage by %d%%."):format(eff.power, eff.numb) end, + type = "other", + subtype = { sun=true, }, + status = "detrimental", + parameters = { power = 10, numb = 1 }, + on_gain = function(self, err) return "#Target# is drained from light!", "+Absorption Strike" end, + on_lose = function(self, err) return "#Target#'s light is back.", "-Absorption Strike" end, + activate = function(self, eff) + self:effectTemporaryValue(eff, "resists", {[DamageType.LIGHT]=-eff.power}) + self:effectTemporaryValue(eff, "numbed", eff.numb) + end, +} + -- Design: Temporary immobility in exchange for a large stat buff. newEffect{ name = "TREE_OF_LIFE", image = "shockbolt/object/artifact/tree_of_life.png", diff --git a/game/modules/tome/data/timed_effects/physical.lua b/game/modules/tome/data/timed_effects/physical.lua index 2d464a3b04aaeaf5e4364d15a46569b983effd7f..ce596db746d586137879dfd0b75e5b600796792a 100644 --- a/game/modules/tome/data/timed_effects/physical.lua +++ b/game/modules/tome/data/timed_effects/physical.lua @@ -347,7 +347,7 @@ newEffect{ newEffect{ name = "STUNNED", image = "effects/stunned.png", desc = "Stunned", - long_desc = function(self, eff) return ("The target is stunned, reducing damage by 70%%, putting random talents on cooldown and reducing movement speed by 50%%. While stunned talents do not cooldown."):format() end, + long_desc = function(self, eff) return ("The target is stunned, reducing damage by 60%%, putting 3 random talents on cooldown and reducing movement speed by 50%%. While stunned talents do not cooldown."):format() end, type = "physical", subtype = { stun=true }, status = "detrimental", @@ -364,7 +364,7 @@ newEffect{ local t = self:getTalentFromId(tid) if t and not self.talents_cd[tid] and t.mode == "activated" and not t.innate and util.getval(t.no_energy, self, t) ~= true then tids[#tids+1] = t end end - for i = 1, 4 do + for i = 1, 3 do local t = rng.tableRemove(tids) if not t then break end self.talents_cd[t.id] = 1 -- Just set cooldown to 1 since cooldown does not decrease while stunned