diff --git a/game/modules/tome/data/chats/artifice-mastery.lua b/game/modules/tome/data/chats/artifice-mastery.lua index f77068676cc95cdc7176c748bbf0f9727abfb7fe..b2a438776fd78aa3ec32897332a38d4f338c2ed1 100644 --- a/game/modules/tome/data/chats/artifice-mastery.lua +++ b/game/modules/tome/data/chats/artifice-mastery.lua @@ -18,40 +18,41 @@ -- darkgod@te4.org local Talents = require("engine.interface.ActorTalents") - +chat_talent = player:getTalentFromId(chat_tid) +chat_level = player:getTalentLevelRaw(chat_tid) local function generate_tools() - local answers = {} - local tools = - { - } - - if player:knowTalent(player.T_HIDDEN_BLADES) then tools[Talents.T_ASSASSINATE] = 1 end - if player:knowTalent(player.T_SMOKESCREEN) then tools[Talents.T_SMOKESCREEN_MASTERY] = 1 end - if player:knowTalent(player.T_ROGUE_S_BREW) then tools[Talents.T_ROGUE_S_BREW_MASTERY] = 1 end - if player:knowTalent(player.T_DART_LAUNCHER) then tools[Talents.T_DART_LAUNCHER_MASTERY] = 1 end - - - if tools then - for tid, level in pairs(tools) do - local t = npc:getTalentFromId(tid) - level = math.min(t.points - game.player:getTalentLevelRaw(tid), level) + local answers = {{"Cancel"}} + for tid, m_tid in pairs(tool_ids) do + local t = player:getTalentFromId(tid) + local m_t = player:getTalentFromId(m_tid) + + local master_talent = function(npc, player) + local old_mastery_level = player:getTalentLevelRaw(m_tid) + if old_mastery_level == chat_level then return end + -- unlearn mastery talent(s) + for tid, m_tid in pairs(tool_ids) do + if player:knowTalent(m_tid) then player:unlearnTalentFull(m_tid) end + end - local doit = function(npc, player) - if game.player:knowTalentType(t.type[1]) == nil then player:setTalentTypeMastery(t.type[1], 1.0) end - player:learnTalent(tid, true, level, {no_unlearn=true}) - player:startTalentCooldown(tid) + player:learnTalent(m_tid, true, chat_level, {no_unlearn=true}) + player.artifice_tools_mastery = tid + + -- start talent cooldowns + if old_mastery_level == 0 then + player:startTalentCooldown(tid) player:startTalentCooldown(m_tid) + player:startTalentCooldown(chat_tid) + game.log("#LIGHT_BLUE# You enhance your preparation of %s.", t.name) end - answers[#answers+1] = {("[%s]"):format(t.name), - action=doit, - on_select=function(npc, player) - local mastery = nil - if player:knowTalentType(t.type[1]) == nil then mastery = 1.0 end - game.tooltip_x, game.tooltip_y = 1, 1 - game:tooltipDisplayAtMap(game.w, game.h, "#GOLD#"..t.name.."#LAST#\n"..tostring(player:getTalentFullDescription(t, 1, nil, mastery))) - end, - } + end - answers[#answers+1] = {"Cancel"} + answers[#answers+1] = {("%s[%s -- mastery: %s]#LAST#"):format(player.artifice_tools_mastery == tid and "#YELLOW#" or "", t.name, m_t.name), + action=master_talent, + on_select=function(npc, player) + local mastery = nil + game.tooltip_x, game.tooltip_y = 1, 1 + game:tooltipDisplayAtMap(game.w, game.h, "#GOLD#"..m_t.name.."#LAST#\n"..tostring(player:getTalentFullDescription(m_t, nil, {force_level=chat_level}, mastery))) + end, + } end return answers end diff --git a/game/modules/tome/data/chats/artifice.lua b/game/modules/tome/data/chats/artifice.lua index 5587262325070b290a359f5ea7f6a61896f476cc..2524ce70940bffb4b91d60570f999aa903e998fa 100644 --- a/game/modules/tome/data/chats/artifice.lua +++ b/game/modules/tome/data/chats/artifice.lua @@ -18,63 +18,79 @@ -- darkgod@te4.org local Talents = require("engine.interface.ActorTalents") +chat_talent = player:getTalentFromId(chat_tid) +chat_level = player:getTalentLevelRaw(chat_tid) local function generate_tools() - local answers = {} - local tools = - { - } - - --populate the tool list, also apply a temp value so talents display correctly - if not player.artifice_hidden_blades then - tools[Talents.T_HIDDEN_BLADES] = 1 - player.artifice_hidden_blades = slot - end - if not player.artifice_smokescreen then - tools[Talents.T_SMOKESCREEN] = 1 - player.artifice_smokescreen = slot - end - if not player.artifice_rogue_s_brew then - tools[Talents.T_ROGUE_S_BREW] = 1 - player.artifice_rogue_s_brew = slot - end - if not player.artifice_dart_launcher then - tools[Talents.T_DART_LAUNCHER] = 1 - player.artifice_dart_launcher = slot - end + local answers = {{"[Cancel]"}} + local tool_ids = tool_ids or player.main_env.artifice_tool_tids + player.artifice_tools = player.artifice_tools or {} - if tools then - for tid, level in pairs(tools) do - local t = npc:getTalentFromId(tid) - level = math.min(t.points - game.player:getTalentLevelRaw(tid), level) - - local doit = function(npc, player) - if game.player:knowTalentType(t.type[1]) == nil then player:setTalentTypeMastery(t.type[1], 1.0) end - player:learnTalent(tid, true, level, {no_unlearn=true}) - --remove the temp values set earlier - if not (t.name=="Hidden Blades" or player:knowTalent(player.T_HIDDEN_BLADES)) then player.artifice_hidden_blades = null end - if not (t.name=="Smokescreen" or player:knowTalent(player.T_SMOKESCREEN)) then player.artifice_smokescreen = null end - if not (t.name=="Rogue's Brew" or player:knowTalent(player.T_ROGUE_S_BREW)) then player.artifice_rogue_s_brew = null end - if not (t.name=="Dart Launcher" or player:knowTalent(player.T_DART_LAUNCHER)) then player.artifice_dart_launcher = null end - player:startTalentCooldown(tid) + for tid, m_tid in pairs(tool_ids) do + local t = player:getTalentFromId(tid) + if t then + + local tool_level = player:getTalentLevelRaw(t) + local equip_tool = function(npc, player) -- equip a tool + if tool_level == chat_level then return end -- already selected and up to date + -- unlearn the previous talent + player:unlearnTalentFull(player.artifice_tools[chat_tid]) + -- (re)learn the talent + player:unlearnTalentFull(tid) + player:learnTalent(tid, true, chat_level, {no_unlearn=true}) + -- clear other tool slots + for slot, tool_id in pairs(player.artifice_tools) do + if tool_id == tid then player.artifice_tools[slot] = nil end + end + player.artifice_tools[chat_tid] = tid + + -- start talent cooldowns and use energy + player.turn_procs._did_artifice = true -- controls energy use + player:startTalentCooldown(tid) player:startTalentCooldown(m_tid) end - answers[#answers+1] = {("[Equip %s]"):format(t.name), - action=doit, + local txt, slot + -- check for an existing slot + for slot_id, tool_id in pairs(player.artifice_tools) do + if tool_id == tid then slot = slot_id break end + end + if slot then + txt = ("[%sEquip %s%s#LAST#]"):format(slot==chat_tid and "#YELLOW#" or "", t.name, slot and (" (%s)"):format(player:getTalentFromId(slot).name) or "") + else + txt = ("[Equip %s]"):format(t.name) + end + + answers[#answers+1] = {txt, + action=equip_tool, on_select=function(npc, player) - local mastery = nil - if player:knowTalentType(t.type[1]) == nil then mastery = 1.0 end + local display_level + display_level = chat_level - tool_level game.tooltip_x, game.tooltip_y = 1, 1 - game:tooltipDisplayAtMap(game.w, game.h, "#GOLD#"..t.name.."#LAST#\n"..tostring(player:getTalentFullDescription(t, 1, nil, mastery))) + + -- set up tooltip + local text = tstring{} + if display_level ~= 0 and player:knowTalent(t) then + local diff = function(i2, i1, res) + if i2 > i1 then + res:add({"color", "LIGHT_GREEN"}, i1, {"color", "LAST"}, " [->", {"color", "YELLOW_GREEN"}, i2, {"color", "LAST"}, "]") + elseif i2 < i1 then + res:add({"color", "LIGHT_GREEN"}, i1, {"color", "LAST"}, " [->", {"color", "LIGHT_RED"}, i2, {"color", "LAST"}, "]") + end + end + text:merge(player:getTalentFullDescription(t, display_level, nil):diffWith(player:getTalentFullDescription(t, 0, nil), diff)) + else + text = player:getTalentFullDescription(t, nil, {force_level=chat_level}) + end + game:tooltipDisplayAtMap(game.w, game.h, "#GOLD#"..t.name.."#LAST#\n"..tostring(text)) end, } end - end + return answers end newChat{ id="welcome", - text = [[Equip which tools?]], + text = ([[Equip which tool for #YELLOW#%s#LAST#?]]):format(chat_talent.name), answers = generate_tools(), } diff --git a/game/modules/tome/data/damage_types.lua b/game/modules/tome/data/damage_types.lua index b8cd0bd78536c5ef0716bbb4aa8454e6c83536ef..795227a7adb551582c54318a9b233a78deffda27 100644 --- a/game/modules/tome/data/damage_types.lua +++ b/game/modules/tome/data/damage_types.lua @@ -239,7 +239,7 @@ setDefaultProjector(function(src, x, y, type, dam, state) end end - if src and dam > 0 and src.knowTalent and src:knowTalent(src.T_BACKSTAB) and src.__CLASSNAME ~= "mod.class.Grid" then + if dam > 0 and src and src.__is_actor and src:knowTalent(src.T_BACKSTAB) and src.__CLASSNAME ~= "mod.class.Grid" then local power = src:callTalent("T_BACKSTAB", "getDamageBoost") local nb = 0 for eff_id, p in pairs(target.tmp) do @@ -3878,11 +3878,13 @@ newDamageType{ newDamageType{ name = "terror", type = "TERROR", + text_color = "#YELLOW#", projector = function(src, x, y, type, dam, state) state = initState(state) useImplicitCrit(src, state) local target = game.level.map(x, y, Map.ACTOR) if target then + game:delayedLogDamage(src, target, 0, ("%s<terror chance>#LAST#"):format(DamageType:get(type).text_color or "#aaaaaa#"), false) if not src:checkHit(src:combatAttack(), target:combatMentalResist()) then return end local effect = rng.range(1, 3) if effect == 1 then @@ -3908,6 +3910,7 @@ newDamageType{ end, } +-- Random poison: 25% to be enhanced newDamageType{ name = "random poison", type = "RANDOM_POISON", text_color = "#LIGHT_GREEN#", projector = function(src, x, y, t, dam, poison, state) @@ -3915,10 +3918,10 @@ newDamageType{ useImplicitCrit(src, state) local power local target = game.level.map(x, y, Map.ACTOR) - if target and src:reactionToward(target) < 0 then + if target and src:reactionToward(target) < 0 and target:canBe("poison") then local realdam = DamageType:get(DamageType.NATURE).projector(src, x, y, DamageType.NATURE, dam.dam / 6, state) - chance = rng.range(1, 3) - if target and target:canBe("poison") and rng.percent(25) then + if rng.percent(dam.random_chance or 25) then + local chance = rng.range(1, 3) if chance == 1 then target:setEffect(target.EFF_INSIDIOUS_POISON, 5, {src=src, power=dam.dam / 6, heal_factor=dam.power*2, apply_power=dam.apply_power or (src.combatAttack and src:combatAttack()) or 0}) elseif chance == 2 then @@ -3926,7 +3929,7 @@ newDamageType{ elseif chance == 3 then target:setEffect(target.EFF_CRIPPLING_POISON, 5, {src=src, power=dam.dam / 6, fail=dam.power, apply_power=dam.apply_power or (src.combatAttack and src:combatAttack()) or 0}) end - elseif target and target:canBe("poison") then + else target:setEffect(target.EFF_POISONED, 5, {src=src, power=dam.dam / 6, apply_power=dam.apply_power or (src.combatAttack and src:combatAttack()) or 0}) end return realdam @@ -3936,11 +3939,13 @@ newDamageType{ newDamageType{ name = "blinding powder", type = "BLINDING_POWDER", + text_color = "#GREY#", projector = function(src, x, y, type, dam, state) state = initState(state) useImplicitCrit(src, state) local target = game.level.map(x, y, Map.ACTOR) if target then + game:delayedLogDamage(src, target, 0, ("%s<blinding powder>#LAST#"):format(DamageType:get(type).text_color or "#aaaaaa#"), false) if not src:checkHit(src:combatAttack(), target:combatPhysicalResist()) then return end if target:canBe("blind") then @@ -3955,12 +3960,13 @@ newDamageType{ newDamageType{ name = "smokescreen", type = "SMOKESCREEN", + text_color = "#GREY#", projector = function(src, x, y, type, dam, state) state = initState(state) useImplicitCrit(src, state) local target = game.level.map(x, y, Map.ACTOR) if target and src:reactionToward(target) < 0 then - + game:delayedLogDamage(src, target, 0, ("%s<smoke>#LAST#"):format(DamageType:get(type).text_color or "#aaaaaa#"), false) if target:canBe("blind") then target:setEffect(target.EFF_DIM_VISION, 2, {sight=dam.dam, apply_power=src:combatAttack(), no_ct_effect=true}) else diff --git a/game/modules/tome/data/talents/cunning/artifice.lua b/game/modules/tome/data/talents/cunning/artifice.lua index 5cad988f587c46c1379b5188452f242bc4afb83f..7570e20b9e48bf26e7f605eb7c917c1861f4010a 100644 --- a/game/modules/tome/data/talents/cunning/artifice.lua +++ b/game/modules/tome/data/talents/cunning/artifice.lua @@ -22,75 +22,111 @@ local Object = require "engine.Object" local Map = require "engine.Map" local Chat = require "engine.Chat" +-- equipable artifice tool talents and associated mastery talents +-- to add a new tool, define a tool talent and a mastery talent and update this table +artifice_tool_tids = {T_HIDDEN_BLADES="T_ASSASSINATE", T_SMOKESCREEN="T_SMOKESCREEN_MASTERY", T_ROGUE_S_BREW="T_ROGUE_S_BREW_MASTERY", T_DART_LAUNCHER="T_DART_LAUNCHER_MASTERY"} + +--- initialize artifice tools, update mastery level and unlearn any unselected tools talents +function artifice_tools_setup(self, t) + self.artifice_tools = self.artifice_tools or {} + self:setTalentTypeMastery("cunning/tools", self:getTalentMastery(t)) + for tid, m_tid in pairs(artifice_tool_tids) do + if self:knowTalent(tid) then + local slot + for slot_id, tool_id in pairs(self.artifice_tools) do + if tool_id == tid then slot = slot_id break end + end + if not slot then self:unlearnTalentFull(tid) end + end + if self.artifice_tools_mastery == tid then + local m_level = self:getTalentLevelRaw(self.T_MASTER_ARTIFICER) + if self:getTalentLevelRaw(m_tid) ~= m_level then + self:unlearnTalentFull(m_tid) + self:learnTalent(m_tid, true, m_level, {no_unlearn=true}) + end + elseif self:knowTalent(m_tid) then + self:unlearnTalentFull(m_tid) + end + end + return true +end + +--- generate a textual list of available artifice tools +function artifice_tools_get_descs(self, t) + if not self.artifice_tools then artifice_tools_setup(self, t) end + local tool_descs = {} + for tool_id, mt in pairs(artifice_tool_tids) do + local tool, desc = self:getTalentFromId(tool_id) + local prepped = self.artifice_tools[t.id] == tool_id + if prepped then + desc = ("#YELLOW#%s (prepared, level %s)#LAST#:\n"):format(tool.name, self:getTalentLevelRaw(tool)) + else + desc = tool.name..":\n" + end + if tool.toolInfo then + desc = desc..tool.toolInfo(self, tool, t).."\n" + else + desc = desc.."#GREY#(see talent description)#LAST#\n" + end + tool_descs[#tool_descs+1] = desc + end + return table.concatNice(tool_descs, "\n\t") +end + +--- NPC's automatically pick a tool for each tool slot if needed +-- used as the talent on_pre_use_ai function +-- this causes newly spawned NPC's to prepare their tools the first time they check for usable talents +function artifice_tools_npc_select(self, t, silent, fake) + if not self.artifice_tools[t.id] then -- slot is empty: pick a tool + local tool_ids = table.keys(artifice_tool_tids) + local tid = rng.tableRemove(tool_ids) + while tid do + if not self:knowTalent(tid) then -- select the tool + self:learnTalent(tid, true, self:getTalentLevelRaw(t), {no_unlearn=true}) + self.artifice_tools[t.id] = tid + if game.party:hasMember(self) then -- cooldowns for party members + self:startTalentCooldown(t); self:startTalentCooldown(tid) + self:useEnergy() + end + game.logSeen(self, "#GREY#You notice %s has prepared: %s.", self.name:capitalize(), self:getTalentFromId(tid).name) + break + end + tid = rng.tableRemove(tool_ids) + end + end + return false -- npc's don't need to actually use the tool slot talents +end + newTalent{ name = "Rogue's Tools", type = {"cunning/artifice", 1}, points = 5, require = cuns_req_high1, cooldown = 10, - no_npc_use = true, + stamina = 0, -- forces learning stamina pool (npcs) no_unlearn_last = true, + on_pre_use = artifice_tools_setup, on_learn = function(self, t) self:attr("show_gloves_combat", 1) end, on_unlearn = function(self, t) self:attr("show_gloves_combat", -1) end, - getHBDamage = function (self, t) return self:combatTalentWeaponDamage(t, 1.0, 1.8) end, - getRBDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 4, 9)) end, - getRBResist = function(self, t) return self:combatTalentLimit(t, 1, 0.17, 0.5) end, - getRBRawHeal = function (self, t) return self:getTalentLevel(t) * 40 end, - getRBMaxHeal = function (self, t) return self:combatTalentLimit(t, 0.4, 0.10, 0.25) end, - getRBCure = function(self, t) return math.floor(self:combatTalentScale(t, 1, 3)) end, - getSSDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 3, 5)) end, - getSSSightLoss = function(self, t) return math.floor(self:combatTalentScale(t,1, 6, "log", 0, 4)) end, -- 1@1 6@5 - getDLDamage = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 12, 150) end, - getDLSleepPower = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 15, 180) end, + tactical = {BUFF = 2}, + on_pre_use_ai = artifice_tools_npc_select, -- NPC's automatically pick a tool action = function(self, t) - if self.artifice_hidden_blades==1 then - self:unlearnTalent(self.T_HIDDEN_BLADES) - self.artifice_hidden_blades = null - if self:knowTalent(self.T_ASSASSINATE) then self:unlearnTalent(self.T_ASSASSINATE) end - end - if self.artifice_smokescreen==1 then - self:unlearnTalent(self.T_SMOKESCREEN) - self.artifice_smokescreen = null - if self:knowTalent(self.T_SMOKESCREEN_MASTERY) then self:unlearnTalent(self.T_SMOKESCREEN_MASTERY) end - end - if self.artifice_rogue_s_brew==1 then - self:unlearnTalent(self.T_ROGUE_S_BREW) - self.artifice_rogue_s_brew = null - if self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then self:unlearnTalent(self.T_ROGUE_S_BREW_MASTERY) end - end - if self.artifice_dart_launcher==1 then - self:unlearnTalent(self.T_DART_LAUNCHER) - self.artifice_dart_launcher = null - if self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then self:unlearnTalent(self.T_DART_LAUNCHER_MASTERY) end - end - - local chat = Chat.new("artifice", self, self, {player=self, slot=1}) + local chat = Chat.new("artifice", self, self, {player=self, slot=1, chat_tid=t.id, tool_ids=artifice_tool_tids}) self:talentDialog(chat:invoke()) - return true + artifice_tools_setup(self, t) + return self.turn_procs._did_artifice -- only use energy/cooldown if a tool was prepared end, info = function(self, t) - local tool = "" - if self:knowTalent(self.T_HIDDEN_BLADES) and self.artifice_hidden_blades==1 then - tool = ([[#YELLOW#Current Tool: Hidden Blades]]):format() - elseif self:knowTalent(self.T_SMOKESCREEN) and self.artifice_smokescreen==1 then - tool = ([[#YELLOW#Current Tool: Smokescreen]]):format() - elseif self:knowTalent(self.T_ROGUE_S_BREW) and self.artifice_rogue_s_brew==1 then - tool = ([[#YELLOW#Current Tool: Rogue's Brew]]):format() - elseif self:knowTalent(self.T_DART_LAUNCHER) and self.artifice_dart_launcher==1 then - tool = ([[#YELLOW#Current Tool: Dart Launcher]]):format() - end - return ([[You learn to create and equip a number of useful tools: -Hidden Blades. Melee criticals inflict %d%% bonus unarmed damage. 4 turn cooldown. -Smokescreen. Throw a vial of smoke that blocks vision in radius 2 for %d turns, and reduces the vision of enemies within by %d. 15 turn cooldown. -Rogue’s Brew. Drink a potion that restores %d life (+%d%% of maximum), %d stamina (+%d%% of maximum) and cures %d negative physical effects. 20 turn cooldown. -Dart Launcher. Fires a dart that deals %0.2f physical damage and puts the target to sleep for 4 turns. 10 turn cooldown. -You can equip a single tool at first. -%s]]): -format(t.getHBDamage(self,t)*100, t.getSSDuration(self,t), t.getSSSightLoss(self,t), t.getRBRawHeal(self,t), t.getRBMaxHeal(self,t)*100, t.getRBRawHeal(self,t)/4, t.getRBMaxHeal(self,t)*40, t.getRBCure(self,t), damDesc(self, DamageType.PHYSICAL, t.getDLDamage(self,t)), tool) + local descs = artifice_tools_get_descs(self, t) + return ([[With some advanced preparation, you learn to create and equip one of a number of useful tools (at #YELLOW#level %d#WHITE#): + +%s +Preparing a tool sets its talent level and puts it on cooldown. +]]):format(self:getTalentLevelRaw(t), descs) end, } @@ -100,128 +136,51 @@ newTalent{ points = 5, require = cuns_req_high2, cooldown = 10, - no_npc_use = true, + stamina = 0, -- forces learning stamina pool (npcs) no_unlearn_last = true, - getHBDamage = function (self, t) return self:combatTalentWeaponDamage(t, 1.0, 1.8) end, - getRBDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 4, 9)) end, - getRBResist = function(self, t) return self:combatTalentLimit(t, 1, 0.17, 0.5) end, - getRBRawHeal = function (self, t) return self:getTalentLevel(t) * 40 end, - getRBMaxHeal = function (self, t) return self:combatTalentLimit(t, 0.4, 0.10, 0.25) end, - getRBCure = function(self, t) return math.floor(self:combatTalentScale(t, 1, 3)) end, - getSSDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 3, 5)) end, - getSSSightLoss = function(self, t) return math.floor(self:combatTalentScale(t,1, 6, "log", 0, 4)) end, -- 1@1 6@5 - getDLDamage = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 12, 150) end, - getDLSleepPower = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 15, 180) end, + on_pre_use = artifice_tools_setup, + tactical = {BUFF = 2}, + on_pre_use_ai = artifice_tools_npc_select, -- NPC's automatically pick a tool action = function(self, t) - if self.artifice_hidden_blades==2 then - self:unlearnTalent(self.T_HIDDEN_BLADES) - self.artifice_hidden_blades = null - if self:knowTalent(self.T_ASSASSINATE) then self:unlearnTalent(self.T_ASSASSINATE) end - end - if self.artifice_smokescreen==2 then - self:unlearnTalent(self.T_SMOKESCREEN) - self.artifice_smokescreen = null - if self:knowTalent(self.T_SMOKESCREEN_MASTERY) then self:unlearnTalent(self.T_SMOKESCREEN_MASTERY) end - end - if self.artifice_rogue_s_brew==2 then - self:unlearnTalent(self.T_ROGUE_S_BREW) - self.artifice_rogue_s_brew = null - if self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then self:unlearnTalent(self.T_ROGUE_S_BREW_MASTERY) end - end - if self.artifice_dart_launcher==2 then - self:unlearnTalent(self.T_DART_LAUNCHER) - self.artifice_dart_launcher = null - if self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then self:unlearnTalent(self.T_DART_LAUNCHER_MASTERY) end - end - - local chat = Chat.new("artifice", self, self, {player=self, slot=2}) + local chat = Chat.new("artifice", self, self, {player=self, slot=2, chat_tid=t.id, tool_ids=artifice_tool_tids}) self:talentDialog(chat:invoke()) - return true + return self.turn_procs._did_artifice -- only use energy/cooldown if a tool was prepared end, info = function(self, t) - local tool = "" - if self:knowTalent(self.T_HIDDEN_BLADES) and self.artifice_hidden_blades==2 then - tool = ([[#YELLOW#Current Tool: Hidden Blades]]):format() - elseif self:knowTalent(self.T_SMOKESCREEN) and self.artifice_smokescreen==2 then - tool = ([[#YELLOW#Current Tool: Smokescreen]]):format() - elseif self:knowTalent(self.T_ROGUE_S_BREW) and self.artifice_rogue_s_brew==2 then - tool = ([[#YELLOW#Current Tool: Rogue's Brew]]):format() - elseif self:knowTalent(self.T_DART_LAUNCHER) and self.artifice_dart_launcher==2 then - tool = ([[#YELLOW#Current Tool: Dart Launcher]]):format() - end - return ([[You learn to equip a second tool: -Hidden Blades. Melee criticals inflict %d%% bonus unarmed damage. 4 turn cooldown. -Smokescreen. Throw a vial of smoke that blocks vision in radius 2 for %d turns, and reduces the vision of enemies within by %d. 15 turn cooldown. -Rogue’s Brew. Drink a potion that restores %d life (+%d%% of maximum), %d stamina (+%d%% of maximum) and cures %d negative physical effects. 20 turn cooldown. -Dart Launcher. Fires a dart that deals %0.2f physical damage and puts the target to sleep for 4 turns. 10 turn cooldown. -%s]]): -format(t.getHBDamage(self,t)*100, t.getSSDuration(self,t), t.getSSSightLoss(self,t), t.getRBRawHeal(self,t), t.getRBMaxHeal(self,t)*100, t.getRBRawHeal(self,t)/4, t.getRBMaxHeal(self,t)*40, t.getRBCure(self,t), damDesc(self, DamageType.PHYSICAL, t.getDLDamage(self,t)), tool) + local descs = artifice_tools_get_descs(self, t) + return ([[With some advanced preparation, you learn to create and equip a second tool (at #YELLOW#level %d#WHITE#): + +%s +Preparing a tool sets its talent level and puts it on cooldown. +Only one tool of each type can be equipped at a time. +]]):format(self:getTalentLevelRaw(t), descs) end, } - newTalent{ name = "Intricate Tools", type = {"cunning/artifice", 3}, require = cuns_req_high3, points = 5, cooldown = 10, - no_npc_use = true, + stamina = 0, -- forces learning stamina pool (npcs) no_unlearn_last = true, - getHBDamage = function (self, t) return self:combatTalentWeaponDamage(t, 1.0, 1.8) end, - getRBDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 4, 9)) end, - getRBResist = function(self, t) return self:combatTalentLimit(t, 1, 0.17, 0.5) end, - getRBRawHeal = function (self, t) return self:getTalentLevel(t) * 40 end, - getRBMaxHeal = function (self, t) return self:combatTalentLimit(t, 0.4, 0.10, 0.25) end, - getRBCure = function(self, t) return math.floor(self:combatTalentScale(t, 1, 3)) end, - getSSDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 3, 5)) end, - getSSSightLoss = function(self, t) return math.floor(self:combatTalentScale(t,1, 6, "log", 0, 4)) end, -- 1@1 6@5 - getDLDamage = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 12, 150) end, - getDLSleepPower = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 15, 180) end, + on_pre_use = artifice_tools_setup, + tactical = {BUFF = 2}, + on_pre_use_ai = artifice_tools_npc_select, -- NPC's automatically pick a tool action = function(self, t) - if self.artifice_hidden_blades==3 then - self:unlearnTalent(self.T_HIDDEN_BLADES) - self.artifice_hidden_blades = null - if self:knowTalent(self.T_ASSASSINATE) then self:unlearnTalent(self.T_ASSASSINATE) end - end - if self.artifice_smokescreen==3 then - self:unlearnTalent(self.T_SMOKESCREEN) - self.artifice_smokescreen = null - if self:knowTalent(self.T_SMOKESCREEN_MASTERY) then self:unlearnTalent(self.T_SMOKESCREEN_MASTERY) end - end - if self.artifice_rogue_s_brew==3 then - self:unlearnTalent(self.T_ROGUE_S_BREW) - self.artifice_rogue_s_brew = null - if self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then self:unlearnTalent(self.T_ROGUE_S_BREW_MASTERY) end - end - if self.artifice_dart_launcher==3 then - self:unlearnTalent(self.T_DART_LAUNCHER) - self.artifice_dart_launcher = null - if self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then self:unlearnTalent(self.T_DART_LAUNCHER_MASTERY) end - end - - local chat = Chat.new("artifice", self, self, {player=self, slot=3}) + local chat = Chat.new("artifice", self, self, {player=self, slot=3, chat_tid=t.id, tool_ids=artifice_tool_tids}) self:talentDialog(chat:invoke()) - return true + return self.turn_procs._did_artifice -- only use energy/cooldown if a tool was prepared end, info = function(self, t) - local tool = "" - if self:knowTalent(self.T_HIDDEN_BLADES) and self.artifice_hidden_blades==3 then - tool = ([[#YELLOW#Current Tool: Hidden Blades]]):format() - elseif self:knowTalent(self.T_SMOKESCREEN) and self.artifice_smokescreen==3 then - tool = ([[#YELLOW#Current Tool: Smokescreen]]):format() - elseif self:knowTalent(self.T_ROGUE_S_BREW) and self.artifice_rogue_s_brew==3 then - tool = ([[#YELLOW#Current Tool: Rogue's Brew]]):format() - elseif self:knowTalent(self.T_DART_LAUNCHER) and self.artifice_dart_launcher==3 then - tool = ([[#YELLOW#Current Tool: Dart Launcher]]):format() - end - return ([[You learn to equip a third tool: -Hidden Blades. Melee criticals inflict %d%% bonus unarmed damage. 4 turn cooldown. -Smokescreen. Throw a vial of smoke that blocks vision in radius 2 for %d turns, and reduces the vision of enemies within by %d. 15 turn cooldown. -Rogue’s Brew. Drink a potion that restores %d life (+%d%% of maximum), %d stamina (+%d%% of maximum) and cures %d negative physical effects. 20 turn cooldown. -Dart Launcher. Fires a dart that deals %0.2f physical damage and puts the target to sleep for 4 turns. 10 turn cooldown. -%s]]): -format(t.getHBDamage(self,t)*100, t.getSSDuration(self,t), t.getSSSightLoss(self,t), t.getRBRawHeal(self,t), t.getRBMaxHeal(self,t)*100, t.getRBRawHeal(self,t)/4, t.getRBMaxHeal(self,t)*40, t.getRBCure(self,t), damDesc(self, DamageType.PHYSICAL, t.getDLDamage(self,t)), tool) + local descs = artifice_tools_get_descs(self, t) + return ([[With some advanced preparation, you learn to create and equip a third tool (at #YELLOW#level %d#WHITE#): + +%s +Preparing a tool sets its talent level and puts it on cooldown. +Only one tool of each type can be equipped at a time. +]]):format(self:getTalentLevelRaw(t), descs) end, } @@ -231,57 +190,81 @@ newTalent{ require = cuns_req_high4, points = 5, cooldown = 10, - no_npc_use = true, + stamina = 0, -- forces learning stamina pool (npcs) + no_energy = true, no_unlearn_last = true, - getAssassinateDamage = function (self, t) return self:combatTalentWeaponDamage(t, 1.8, 3.0) end, - getBleed = function(self, t) return self:combatTalentScale(t, 0.2, 0.8) end, - getSSDamage = function (self, t) return 30 + self:combatTalentStatDamage(t, "cun", 10, 150) end, - getRBDieAt = function(self, t) return self:combatTalentScale(t, 100, 600) end, - getDLSlow = function(self, t) return self:combatTalentLimit(t, 50, 15, 40)/100 end, + on_pre_use = artifice_tools_setup, + tactical = {BUFF = 2}, + on_pre_use_ai = function(self, t, silent, fake) -- npc's automatically master a tool they have prepared + if self.artifice_tools and not self.artifice_tools_mastery then + game:onTickEnd(function() + local tools = table.values(self.artifice_tools) + while #tools > 0 do + local tool_id = rng.tableRemove(tools) + local m_tid = artifice_tool_tids[tool_id] + if m_tid then -- note: talent level affects AI use + local tl = self:getTalentLevelRaw(m_tid) + if self:learnTalent(m_tid, true, self:getTalentLevelRaw(t) - tl) then + self.artifice_tools_mastery = tool_id + if game.party:hasMember(self) then -- cooldowns for party members + self:startTalentCooldown(t); self:startTalentCooldown(tool_id); self:startTalentCooldown(m_tid) + end + break + end + end + end + end) + end + return false + end, action = function(self, t) - if self:knowTalent(self.T_ASSASSINATE) then self:unlearnTalent(self.T_ASSASSINATE) end - if self:knowTalent(self.T_SMOKESCREEN_MASTERY) then self:unlearnTalent(self.T_SMOKESCREEN_MASTERY) end - if self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then self:unlearnTalent(self.T_ROGUE_S_BREW_MASTERY) end - if self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then self:unlearnTalent(self.T_DART_LAUNCHER_MASTERY) end - - local chat = Chat.new("artifice-mastery", self, self, {player=self}) + local chat = Chat.new("artifice-mastery", self, self, {player=self, chat_tid=t.id, tool_ids=artifice_tool_tids}) self:talentDialog(chat:invoke()) - return true + return false -- chat handles cooldowns end, info = function(self, t) - local tool = "" - if self:knowTalent(self.T_ASSASSINATE) then - tool = ([[#YELLOW#Current Mastery: Hidden Blades]]):format() - elseif self:knowTalent(self.T_SMOKESCREEN_MASTERY) then - tool = ([[#YELLOW#Current Mastery: Smokescreen]]):format() - elseif self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then - tool = ([[#YELLOW#Current Mastery: Rogue's Brew]]):format() - elseif self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then - tool = ([[#YELLOW#Current Mastery: Dart Launcher]]):format() + local tool = "none" + if self.artifice_tools_mastery then + tool = self:getTalentFromId(self.artifice_tools_mastery).name + end + --- generate a textual list of available artifice tools enhancements + if not self.artifice_tools then artifice_tools_setup(self, t) end + local mastery_descs = {} + for tool_id, m_tid in pairs(artifice_tool_tids) do + local tool, mt = self:getTalentFromId(tool_id), self:getTalentFromId(m_tid) + local desc + local prepped = self.artifice_tools_mastery == tool_id + if prepped then + desc = ("#YELLOW#%s (%s)#LAST#\n"):format(tool.name, mt.name) + else + desc = ("%s (%s)\n"):format(tool.name, mt.name) + end + if mt.masteryInfo then + desc = desc..mt.masteryInfo(self, mt).."\n" + else + desc = desc.."#GREY#(see talent description)#LAST#\n" + end + mastery_descs[#mastery_descs+1] = desc end - return ([[You reach the height of your craft, allowing you to focus on a single tool to greatly improve its capabilities: -Hidden Blades. Grants use of the Assassinate ability, striking twice with your hidden blades for %d%% unarmed damage as a guaranteed critical strike which ignores armor and resistances. Your Hidden Blades also inflict an additional %d%% damage as bleed. -Smokescreen: Infuses your Smokescreen with chokedust, causing %0.2f nature damage each turn to enemies inside as well as silencing them. -Rogue’s Brew. The brew strengthens you for 8 turns, preventing you from dying until you reach -%d life. -Dart Launcher. The sleeping poison becomes potent enough to ignore immunity, and on waking the target will be slowed by %d%% for 4 turns. -%s]]): -format(t.getAssassinateDamage(self,t)*100, t.getBleed(self,t)*100, damDesc(self, DamageType.NATURE, t.getSSDamage(self,t)), t.getRBDieAt(self,t), t.getDLSlow(self,t)*100, tool) + mastery_descs = table.concatNice(mastery_descs, "\n\t") + return ([[You become a master of your craft, allowing you to focus on a single tool {#YELLOW#currently %s#LAST#) to greatly improve its capabilities: + +%s +The effects depend on this talent's level. +Mastering a new tool places it (and its special effects, as appropriate) on cooldown.]]):format(tool, mastery_descs) end, } +--====================-- +-- Rogue's tools and enhancements +--====================-- newTalent{ name = "Hidden Blades", type = {"cunning/tools", 1}, mode = "passive", points = 1, cooldown = 4, - getDamage = function(self, t) - if self.artifice_hidden_blades == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getHBDamage") - elseif self.artifice_hidden_blades == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getHBDamage") - elseif self.artifice_hidden_blades == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getHBDamage") - else return 0 - end - end, + getDamage = function (self, t) return self:combatTalentWeaponDamage(t, 1.0, 1.8) end, callbackOnCrit = function(self, t, kind, dam, chance, target) if not target then return end if target.turn_procs.hb then return end @@ -289,6 +272,7 @@ newTalent{ if not self:isTalentCoolingDown(t) then target.turn_procs.hb = true local oldlife = target.life + self:logCombat(target, "#Source# strikes #target# with hidden blades!") self:attackTarget(target, nil, t.getDamage(self,t), true, true) if self:knowTalent(self.T_ASSASSINATE) then @@ -302,47 +286,112 @@ newTalent{ self:startTalentCooldown(t) end end, + toolInfo = function(self, t, slot_talent) + return ([[Melee criticals trigger an extra unarmed attack, inflicting %d%% damage. 4 turn cooldown.]]):format(t.getDamage(self, slot_talent)*100) + end, info = function(self, t) local dam = t.getDamage(self, t) - return ([[You mount spring loaded blades on your wrists. On scoring a critical strike against an adjacent target, you follow up with your blades for %d%% unarmed damage. -This talent has a cooldown.]]): - format(dam*100) + local slot = "not prepared" + for slot_id, tool_id in pairs(self.artifice_tools) do + if tool_id == t.id then slot = self:getTalentFromId(slot_id).name break end + end + return ([[You conceal spring loaded blades within your equipment. On scoring a critical strike against an adjacent target, you follow up with your blades for %d%% damage (as an unarmed attack). +This talent has a cooldown. +#YELLOW#Prepared with: %s#LAST#]]):format(dam*100, slot) end, } newTalent{ - name = "Rogue's Brew", + name = "Assassinate", type = {"cunning/tools", 1}, points = 1, - cooldown = 20, - tactical = { BUFF = 2 }, + cooldown = 8, + stamina = 10, + message = false, + tactical = { ATTACK = 3 }, requires_target = true, - getRawHeal = function(self, t) - if self.artifice_rogue_s_brew == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getRBRawHeal") - elseif self.artifice_rogue_s_brew == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getRBRawHeal") - elseif self.artifice_rogue_s_brew == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getRBRawHeal") - else return 0 + is_melee = true, + target = function(self, t) return {type="hit", range=self:getTalentRange(t)} end, + range = 1, + on_pre_use = function(self, t, silent, fake) + if not self:knowTalent(self.T_HIDDEN_BLADES) then + if not silent then game.logPlayer(self, "You must have Hidden Blades prepared to use this talent.") end + return end + return true end, - getMaxHeal = function(self, t) - if self.artifice_rogue_s_brew == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getRBMaxHeal") - elseif self.artifice_rogue_s_brew == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getRBMaxHeal") - elseif self.artifice_rogue_s_brew == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getRBMaxHeal") - else return 0 + getDamage = function (self, t) return self:combatTalentWeaponDamage(self:getTalentFromId(self.T_MASTER_ARTIFICER), 1.8, 3.0) end, + getBleed = function(self, t) return self:combatTalentScale(self:getTalentFromId(self.T_MASTER_ARTIFICER), 0.3, 1) end, + action = function(self, t) + local tg = self:getTalentTarget(t) + local x, y, target = self:getTarget(tg) + if not target or not self:canSee(target) or not self:canProject(tg, x, y) then return nil end + + target.turn_procs.hb = true -- prevent a crit against this target from triggering an additional hidden blades attack + self.turn_procs.auto_melee_hit = true + -- store old values to restore later + local apr, rpen, evasion = self.combat_apr, self.resists_pen.PHYSICAL, target.evasion + self:attr("combat_apr", 10000) + self.resists_pen.PHYSICAL = 100 + target.evasion = 0 + local bleed = t.getBleed(self, t) + local oldlife = target.life + + self:logCombat(target, "#Source# strikes at a vital spot on #target#!") + local do_attack = function() self:attackTarget(target, nil, t.getDamage(self, t), true, true) end + local ok, err = pcall(do_attack) + if ok then ok, err = pcall(do_attack) end + self.combat_apr, self.resists_pen.PHYSICAL, target.evasion = apr, rpen, evasion + if not ok then error(err) end + self.turn_procs.auto_melee_hit = nil + + local life_diff = oldlife - target.life + if life_diff > 0 and target:canBe('cut') and bleed then + target:setEffect(target.EFF_CUT, 5, {power=life_diff * bleed / 5, src=self}) end + + return true + end, + masteryInfo = function(self, t) + return ([[You prime your Hidden Blades to cause bleeding and facilitate the Assassinate ability, which allows you to strike twice for %d%% unarmed damage, hitting automatically while ignoring armor and resistance.]]):format(t.getDamage(self, t)*100) + end, + info = function(self, t) + local damage = t.getDamage(self, t) * 100 + local bleed = t.getBleed(self,t) * 100 + return ([[You strike your target with your Hidden Blades twice in a vital spot for %d%% unarmed (physical) damage. You must be able to see your target to use this attack, but it always hits and ignores all armor and physical resistance. +In addition, your hidden blades now inflict a further %d%% of all damage dealt as bleeding over 5 turns.]]) + :format(damage, bleed) end, - getCure = function(self,t) - if self.artifice_rogue_s_brew == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getRBCure") - elseif self.artifice_rogue_s_brew == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getRBCure") - elseif self.artifice_rogue_s_brew == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getRBCure") - else return 0 +} + +newTalent{ + name = "Rogue's Brew", + type = {"cunning/tools", 1}, + points = 1, + cooldown = 20, + tactical = { HEAL = 1.5, STAMINA = 1.5, + CURE = function(self, t, target) + local num, max = 0, t.getCure(self, t) + for eff_id, p in pairs(self.tmp) do + local e = self.tempeffect_def[eff_id] + if e.type == "physical" and e.status == "detrimental" then + num = num + 1 + if num >= max then break end + end + end + return (2*num)^.5 end + }, + getHeal = function(self, t) + return self:combatStatScale("cun", 10, 200, 0.7) + self:combatTalentScale(t, 20, 200, 0.7) end, - getDieAt = function(self,t) return self:callTalent(self.T_MASTER_ARTIFICER, "getRBDieAt") end, + getStam = function(self, t) + return self:combatStatScale("cun", 5, 50, 0.75) + self:combatTalentScale(t, 5, 50, 0.75) + end, + getCure = function(self, t) return math.floor(self:combatTalentScale(t, 1, 3, "log")) end, action = function(self, t) - - local life = t.getRawHeal(self,t) + (t.getMaxHeal(self,t) * self.max_life) - local sta = t.getRawHeal(self,t)/4 + (t.getMaxHeal(self,t) * self.max_stamina * 0.4) + local life = t.getHeal(self, t) + local sta = t.getStam(self, t) self:incStamina(sta) self:attr("allow_on_heal", 1) self:heal(life, self) @@ -370,20 +419,41 @@ newTalent{ game.logSeen(self, "%s is cured!", self.name:capitalize()) end - if self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then self:setEffect(self.EFF_ROGUE_S_BREW, 8, {power = t.getDieAt(self,t)}) end + if self:knowTalent(self.T_ROGUE_S_BREW_MASTERY) then self:setEffect(self.EFF_ROGUE_S_BREW, 8, {power = self:callTalent(self.T_ROGUE_S_BREW_MASTERY, "getDieAt")}) end return true end, + toolInfo = function(self, t, slot_talent) + return ([[Prepare a potion that restores %d life, %d stamina, and cures %d negative physical effects. 20 turn cooldown.]]):format(t.getHeal(self, slot_talent), t.getStam(self, slot_talent), t.getCure(self, slot_talent)) + end, info = function(self, t) - local heal = t.getRawHeal(self,t) + (t.getMaxHeal(self,t) * self.max_life) - local sta = t.getRawHeal(self,t)/4 + (t.getMaxHeal(self,t) * self.max_stamina * 0.4) + local heal = t.getHeal(self, t) + local sta = t.getStam(self, t) local cure = t.getCure(self,t) - return ([[Imbibe a potent mixture of energizing and restorative substances, restoring %d life, %d stamina and curing %d negative physical effects.]]): - format(heal, sta, cure) + local slot = "not prepared" + for slot_id, tool_id in pairs(self.artifice_tools) do + if tool_id == t.id then slot = self:getTalentFromId(slot_id).name break end + end + return ([[Imbibe a potent mixture of energizing and restorative substances, restoring %d life, %d stamina and curing %d detrimental physical effects. The restorative effects improve with your Cunning. + #YELLOW#Prepared with: %s#LAST#]]):format(heal, sta, cure, slot) end, } +newTalent{ + name = "Rogue's Brew Mastery", + type = {"cunning/tools", 1}, + mode = "passive", + points = 1, + getDieAt = function(self, t) return self:combatTalentScale(self:getTalentFromId(self.T_MASTER_ARTIFICER), 100, 600) end, + masteryInfo = function(self, t) + return ([[Your Rogue's Brew fortifies you for 8 turns, preventing you from dying until you reach -%d life.]]):format(t.getDieAt(self, t)) + end, + info = function(self, t) + return ([[Adjust your Rogue's Brew formulation so that it fortifies you for 8 turns, preventing you from dying until you reach -%d life.]]):format(t.getDieAt(self,t)) + end, +} + newTalent{ name = "Smokescreen", type = {"cunning/tools", 1}, @@ -392,32 +462,22 @@ newTalent{ stamina = 10, range = 6, direct_hit = true, - tactical = { DISABLE = 2 }, + tactical = { ESCAPE = 2, DISABLE = {blind = 2} }, requires_target = true, + no_break_stealth = true, radius = 2, - getSightLoss = function(self, t) - if self.artifice_smokescreen == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getSSSightLoss") - elseif self.artifice_smokescreen == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getSSSightLoss") - elseif self.artifice_smokescreen == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getSSSightLoss") - else return 0 - end - end, getDamage = function(self,t) if self:knowTalent(self.T_SMOKESCREEN_MASTERY) then - return self:callTalent(self.T_SMOKESCREEN_MASTERY, "getSSDamage") + return self:callTalent(self.T_SMOKESCREEN_MASTERY, "getDamage") else return 0 end end, - getDuration = function(self, t) - if self.artifice_smokescreen == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getSSDuration") - elseif self.artifice_smokescreen == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getSSDuration") - elseif self.artifice_smokescreen == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getSSDuration") - else return 0 - end - end, + getDuration = function(self, t) return math.ceil(self:combatTalentScale(t, 3, 5)) end, + getSightLoss = function(self, t) return math.floor(self:combatTalentScale(t,1, 6, "log", 0, 4)) end, -- 1@1 6@5 + target = function(self, t) return {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t), talent=t} end, action = function(self, t) - local tg = {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t), talent=t} + local tg = self:getTalentTarget(t) local x, y = self:getTarget(tg) if not x or not y then return nil end @@ -460,7 +520,7 @@ newTalent{ e.particles = Particles.new("creeping_dark", 1, { }) e.particles.x = px e.particles.y = py - game.level.map:addParticleEmitter(e.particles) + game.level.map:addParticleEmitter(e.particles) end, nil, {type="dark"}) @@ -468,80 +528,19 @@ newTalent{ game.level.map:redisplay() return true end, - info = function(self, t) - return ([[Throw a vial of sticky smoke that explodes in radius %d, blocking line of sight for 5 turns. Enemies within will have their vision range reduced by %d. - Creatures affected by smokescreen can never prevent you from stealthing, even if their proximity would normally forbid it. - Use of this will not break stealth.]]): - format(self:getTalentRadius(t), t.getSightLoss(self,t)) - end, -} - -newTalent{ - name = "Assassinate", - type = {"cunning/tools", 1}, - points = 1, - cooldown = 8, - message = "@Source@ lashes out with their hidden blades!", - tactical = { ATTACK = { weapon = 2 } }, - requires_target = true, - is_melee = true, - target = function(self, t) return {type="hit", range=self:getTalentRange(t)} end, - range = 1, - getDamage = function(self, t) return self:callTalent(self.T_MASTER_ARTIFICER, "getAssassinateDamage") end, - getBleed = function(self, t) return self:combatTalentScale(t, 0.3, 1) end, - action = function(self, t) - local tg = self:getTalentTarget(t) - local x, y, target = self:getTarget(tg) - if not target or not self:canProject(tg, x, y) then return nil end - - target.turn_procs.hb = true -- we're already using our hidden blades for this attack - self.turn_procs.auto_melee_hit = true - - self:attr("combat_apr", 1000) - local penstore = self.resists_pen - local storeeva = target.evasion - target.evasion=0 - self.resists_pen = nil - self.resists_pen = {all = 100} - - local scale = nil - scale = t.getBleed(self, t) - local oldlife = target.life - - self:attackTarget(target, nil, t.getDamage(self, t), true, true) - self:attackTarget(target, nil, t.getDamage(self, t), true, true) - - local life_diff = oldlife - target.life - if life_diff > 0 and target:canBe('cut') and scale then - target:setEffect(target.EFF_CUT, 5, {power=life_diff * scale / 5, src=self}) - end - - self:attr("combat_apr", -1000) - self.turn_procs.auto_melee_hit = nil - target.evasion = storeeva - self.resists_pen = nil - self.resists_pen = penstore - - return true - end, - info = function(self, t) - local damage = t.getDamage(self, t) * 100 - local bleed = t.getBleed(self,t) * 100 - return ([[Impale the target on your hidden blades, striking twice for %d%% unarmed damage. This attack always hits and ignores all armor and resistances. -In addition, your hidden blades now inflict a further %d%% of all damage dealt as bleeding over 5 turns.]]) - :format(damage, bleed) + toolInfo = function(self, t, slot_talent) + return ([[Throw a smokebomb creating a radius 2 cloud of smoke, lasting %d turns, that blocks sight and reduces enemies' vision by %d. 15 turn cooldown.]]):format(t.getSightLoss(self, slot_talent), t.getDuration(self, slot_talent)) end, -} - -newTalent{ - name = "Rogue's Brew Mastery", - type = {"cunning/tools", 1}, - mode = "passive", - points = 1, - getDieAt = function(self,t) return self:callTalent(self.T_MASTER_ARTIFICER, "getRBDieAt") end, info = function(self, t) - return ([[The brew strengthens you for 8 turns, preventing you from dying until you reach -%d life.]]): - format(t.getDieAt(self,t)) + local slot = "not prepared" + for slot_id, tool_id in pairs(self.artifice_tools) do + if tool_id == t.id then slot = self:getTalentFromId(slot_id).name break end + end + return ([[Throw a vial of volatile liquid that explodes in a smoke cloud of radius %d, blocking line of sight for 5 turns. Enemies within will have their vision range reduced by %d. + Creatures affected by smokescreen can never prevent you from activating stealth, even if their proximity would normally forbid it. + Use of this talent will not break stealth. + #YELLOW#Prepared with: %s#LAST#]]): + format(self:getTalentRadius(t), t.getSightLoss(self,t), slot) end, } @@ -550,12 +549,13 @@ newTalent{ type = {"cunning/tools", 1}, points = 1, mode = "passive", - getSSDamage = function (self,t) return self:callTalent(self.T_MASTER_ARTIFICER, "getSSDamage") end, - getSSEvasion = function (self,t) return self:callTalent(self.T_MASTER_ARTIFICER, "getSSEvasion") end, - no_npc_use = true, + getDamage = function (self, t) return 30 + self:combatTalentStatDamage(self:getTalentFromId(self.T_MASTER_ARTIFICER), "cun", 10, 150) end, + masteryInfo = function(self, t) + return ([[Your Smokescreen is infused with chokedust. Enemies in the smoke take %0.2f nature damage and may be silenced.]]):format(t.getDamage(self, t)) + end, info = function(self, t) - return ([[Infuses your smoke bomb with chokedust, causing %0.2f nature damage each turn and silencing enemies inside.]]): - format(damDesc(self, DamageType.NATURE, t.getSSDamage(self,t)), t.getSSEvasion(self,t)) + return ([[You infuse your smoke bomb with chokedust. Each turn, enemies in the smoke take %0.2f nature damage and are 50%% likely to be silenced.]]): + format(damDesc(self, DamageType.NATURE, t.getDamage(self,t))) end, } @@ -563,31 +563,23 @@ newTalent{ name = "Dart Launcher", type = {"cunning/tools", 1}, points = 1, - tactical = { ATTACK = 2 }, + tactical = { ATTACK = {PHYSICAL = 1}, + DISABLE = function(self, t, target) + return self:knowTalent(self.T_DART_LAUNCHER_MASTERY) and 2 or {sleep = 1, poison = 1} + end + }, range = 5, no_energy = true, cooldown = 10, + stamina = 5, requires_target = true, no_break_stealth = true, - getDamage = function(self, t) - if self.artifice_dart_launcher == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getDLDamage") - elseif self.artifice_dart_launcher == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getDLDamage") - elseif self.artifice_dart_launcher == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getDLDamage") - else return 0 - end - end, - getSleepPower = function(self, t) - if self.artifice_dart_launcher == 1 then return self:callTalent(self.T_ROGUE_S_TOOLS, "getDLSleepPower") - elseif self.artifice_dart_launcher == 2 then return self:callTalent(self.T_CUNNING_TOOLS, "getDLSleepPower") - elseif self.artifice_dart_launcher == 3 then return self:callTalent(self.T_INTRICATE_TOOLS, "getDLSleepPower") - else return 0 - end - end, - getSlow = function(self, t) return self:callTalent(self.T_MASTER_ARTIFICER, "getDLSlow") end, + getDamage = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 12, 150) end, + getSleepPower = function(self, t) return 15 + self:combatTalentStatDamage(t, "cun", 15, 180) end, target = function(self, t) return {type="bolt", range=self:getTalentRange(t)} end, - action = function(self, t) + action = function(self, t) local tg = self:getTalentTarget(t) local x, y = self:getTarget(tg) if not x or not y then return nil end @@ -595,7 +587,7 @@ newTalent{ local slow = 0 - if self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then slow = t.getSlow(self,t) end + if self:knowTalent(self.T_DART_LAUNCHER_MASTERY) then slow = self:callTalent(self.T_DART_LAUNCHER_MASTERY, "getSlow") end self:project(tg, x, y, function(px, py) local target = game.level.map(px, py, engine.Map.ACTOR) @@ -612,12 +604,20 @@ newTalent{ return true end, + toolInfo = function(self, t, slot_talent) + return ([[Fire a poisoned dart dealing %0.2f physical damage that puts the target to sleep for 4 turns. 10 turn cooldown.]]):format(t.getDamage(self, slot_talent)) + end, info = function(self, t) local dam = t.getDamage(self,t) local power = t.getSleepPower(self,t) - return ([[Uses a wrist mounted launcher to fire a poisoned dart dealing %0.2f physical damage and putting the target to sleep for 4 turns, rendering them unable to act. Every %d points of damage the target take reduces the duration of the sleeping poison by 1 turn. -This can be used without breaking stealth.]]): - format(damDesc(self, DamageType.PHYSICAL, dam), power) + local slot = "not prepared" + for slot_id, tool_id in pairs(self.artifice_tools) do + if tool_id == t.id then slot = self:getTalentFromId(slot_id).name break end + end + return ([[Fire a poisoned dart from a silent, concealed launcher on your person that deals %0.2f physical damage and puts the target to sleep for 4 turns, rendering them unable to act. Every %d points of damage the target takes brings it closer to waking by 1 turn. +This can be used without breaking stealth. +#YELLOW#Prepared with: %s#LAST#]]): + format(damDesc(self, DamageType.PHYSICAL, dam), power, slot) end, } @@ -626,9 +626,12 @@ newTalent{ type = {"cunning/tools", 1}, mode = "passive", points = 1, - getSlow = function(self, t) return self:callTalent(self.T_MASTER_ARTIFICER, "getDLSlow") end, + getSlow = function(self, t) return self:combatTalentLimit(self:getTalentFromId(self.T_MASTER_ARTIFICER), 50, 15, 40)/100 end, + masteryInfo = function(self, t) + return ([[Your darts ignore poison and sleep immunity and waking targets are slowed by %d%% for 4 turns.]]):format(t.getSlow(self, t)*100) + end, info = function(self, t) return ([[The sleeping poison of your Dart Launcher becomes potent enough to ignore immunity, and upon waking the target is slowed by %d%% for 4 turns.]]): format(t.getSlow(self, t)*100) end, -} \ No newline at end of file +}