From 19a179abd640f4814064d05fde65db060cde0456 Mon Sep 17 00:00:00 2001 From: dg <dg@51575b47-30f0-44d4-a5cc-537603b46e54> Date: Thu, 14 Jun 2012 12:58:41 +0000 Subject: [PATCH] New levelup screen. Much faster, lighter, smaller and with less wall of texts! git-svn-id: http://svn.net-core.org/repos/t-engine4@5238 51575b47-30f0-44d4-a5cc-537603b46e54 --- .../tome/data/quests/anti-antimagic.lua | 2 +- .../modules/tome/data/timed_effects/other.lua | 16 + game/modules/tome/dialogs/Levelup2Dialog.lua | 36 +- game/modules/tome/dialogs/LevelupDialog.lua | 1022 ++++------------- .../tome/dialogs/elements/TalentTrees.lua | 87 +- 5 files changed, 321 insertions(+), 842 deletions(-) diff --git a/game/modules/tome/data/quests/anti-antimagic.lua b/game/modules/tome/data/quests/anti-antimagic.lua index 5f88777231..5d8c101ca1 100644 --- a/game/modules/tome/data/quests/anti-antimagic.lua +++ b/game/modules/tome/data/quests/anti-antimagic.lua @@ -113,7 +113,7 @@ myssil_dies = function(self) local corr = nil for uid, e in pairs(game.level.entities) do - if e.faction == "rhalore" then e.invulnerable = 1 end + if e.faction == "rhalore" then e:setEffect(e.EFF_VICTORY_RUSH_ZIGUR, 20, {}) end if e.define_as and e.define_as == "GRAND_CORRUPTOR" then corr = e end end if not corr then return end diff --git a/game/modules/tome/data/timed_effects/other.lua b/game/modules/tome/data/timed_effects/other.lua index 20fbb83fc3..1b4b896c20 100644 --- a/game/modules/tome/data/timed_effects/other.lua +++ b/game/modules/tome/data/timed_effects/other.lua @@ -1523,3 +1523,19 @@ newEffect{ self:removeTemporaryValue("zero_resource_cost", eff.tmpid) end, } + +newEffect{ + name = "VICTORY_RUSH_ZIGUR", image = "talents/arcane_destruction.png", + desc = "Victory Rush", + long_desc = function(self, eff) return "The thrill of victory makes this creature invulnerable." end, + type = "other", + subtype = { arcane=true }, + status = "beneficial", + parameters = { }, + activate = function(self, eff) + eff.tmpid = self:addTemporaryValue("invulnerable", 1) + end, + deactivate = function(self, eff) + self:removeTemporaryValue("invulnerable", eff.tmpid) + end, +} diff --git a/game/modules/tome/dialogs/Levelup2Dialog.lua b/game/modules/tome/dialogs/Levelup2Dialog.lua index 37608be364..3b31fe4659 100644 --- a/game/modules/tome/dialogs/Levelup2Dialog.lua +++ b/game/modules/tome/dialogs/Levelup2Dialog.lua @@ -28,13 +28,6 @@ local DamageType = require "engine.DamageType" module(..., package.seeall, class.inherit(Dialog, mod.class.interface.TooltipsData)) -local _points_text = "Stats points left: #00FF00#%d#LAST#" - -local _points_left = [[ -Category points left: #00FF00#%d#LAST# -Class talent points left: #00FF00#%d#LAST# -Generic talent points left: #00FF00#%d#LAST#]] - function _M:init(actor, on_finish, on_birth) self.on_birth = on_birth actor.no_last_learnt_talents_cap = true @@ -65,13 +58,22 @@ function _M:init(actor, on_finish, on_birth) if v.type == "subclass" and v.name == actor.descriptor.subclass then self.desc_def = v break end end - Dialog.init(self, "Levelup: "..actor.name, game.w * 0.9, game.h * 0.9, game.w * 0.05, game.h * 0.05) + Dialog.init(self, "Levelup: "..actor.name, 650, game.h * 0.9, game.w * 0.05, game.h * 0.05) self:generateList() self:loadUI(self:createDisplay()) self:setupUI() + self.key:addCommands{ + __TEXTINPUT = function(c) + if c == "+" and self.focus_ui and self.focus_ui.ui.onExpand then + self.focus_ui.ui:onExpand(self.focus_ui.ui.last_mz.item) + elseif c == "-" then + self.focus_ui.ui:onExpand(self.focus_ui.ui.last_mz.item) + end + end, + } self.key:addBinds{ EXIT = function() if self.actor.unused_stats~=self.actor_dup.unused_stats or self.actor.unused_talents_types~=self.actor_dup.unused_talents_types or @@ -491,7 +493,10 @@ function _M:generateList() -- Makes up the stats list local phys, mind = {}, {} - self.tree_stats = {{shown=true, nodes=phys, name="Physical Stats", type_stat=true}, {shown=true, nodes=mind, name="Mental Stats", type_stat=true}} + self.tree_stats = { + {shown=true, nodes=phys, name="Physical Stats", type_stat=true, desc=self.TOOLTIP_STRDEXCON}, + {shown=true, nodes=mind, name="Mental Stats", type_stat=true, desc=TOOLTIP_MAGWILCUN} + } for i, sid in ipairs{self.actor.STAT_STR, self.actor.STAT_DEX, self.actor.STAT_CON, self.actor.STAT_MAG, self.actor.STAT_WIL, self.actor.STAT_CUN } do local s = self.actor.stats_def[sid] @@ -542,7 +547,11 @@ function _M:createDisplay() tiles=game.uiset.hotkeys_display_icons, tree=self.tree, width=self.iw-200-10, height=self.ih-10, - tooltip=function(item) return self:getTalentDesc(item), self.uis[3].x - game.tooltip.max, nil end, + tooltip=function(item) + local x = self.display_x + self.uis[3].x - game.tooltip.max + if self.display_x + self.w + game.tooltip.max <= game.w then x = self.display_x + self.w end + return self:getTalentDesc(item), x, nil + end, on_use = function(item, inc) self:onUseTalent(item, inc) end, scrollbar = true, } @@ -551,7 +560,12 @@ function _M:createDisplay() tiles=game.uiset.hotkeys_display_icons, tree=self.tree_stats, no_cross = true, width=200, height=210, - tooltip=function(item) return item.desc end, + dont_select_top = true, + tooltip=function(item) + local x = self.display_x + self.uis[1].x + self.uis[1].ui.w + if self.display_x + self.w + game.tooltip.max <= game.w then x = self.display_x + self.w end + return item.desc, x, nil + end, on_use = function(item, inc) self:onUseTalent(item, inc) end, on_expand = function(item) self.actor.__hidden_talent_types[item.type] = not item.shown end, } diff --git a/game/modules/tome/dialogs/LevelupDialog.lua b/game/modules/tome/dialogs/LevelupDialog.lua index 0bea6dacd0..3b31fe4659 100644 --- a/game/modules/tome/dialogs/LevelupDialog.lua +++ b/game/modules/tome/dialogs/LevelupDialog.lua @@ -21,24 +21,13 @@ require "engine.class" require "mod.class.interface.TooltipsData" local Dialog = require "engine.ui.Dialog" -local ListColumns = require "engine.ui.ListColumns" local Textzone = require "engine.ui.Textzone" -local TextzoneList = require "engine.ui.TextzoneList" +local TalentTrees = require "mod.dialogs.elements.TalentTrees" local Separator = require "engine.ui.Separator" -local Tab = require "engine.ui.Tab" -local TreeList = require "engine.ui.TreeList" -local SurfaceZone = require "engine.ui.SurfaceZone" local DamageType = require "engine.DamageType" module(..., package.seeall, class.inherit(Dialog, mod.class.interface.TooltipsData)) -local _points_text = "Stats points left: #00FF00#%d#LAST#" - -local _points_left = [[ -Category points left: #00FF00#%d#LAST# -Class talent points left: #00FF00#%d#LAST# -Generic talent points left: #00FF00#%d#LAST#]] - function _M:init(actor, on_finish, on_birth) self.on_birth = on_birth actor.no_last_learnt_talents_cap = true @@ -69,306 +58,54 @@ function _M:init(actor, on_finish, on_birth) if v.type == "subclass" and v.name == actor.descriptor.subclass then self.desc_def = v break end end - Dialog.init(self, "Stats Levelup: "..actor.name, game.w * 0.9, game.h * 0.9, game.w * 0.05, game.h * 0.05) - - self.vs = Separator.new{dir="vertical", size=self.iw} - self.vs2 = Separator.new{dir="vertical", size=math.floor(self.iw / 2) - 20} - - self.c_t_tut = Textzone.new{width=math.floor(self.iw / 2 - 10), height=1, auto_height=true, no_color_bleed=true, text=[[ -Keyboard: #00FF00#up key/down key#FFFFFF# to select talent; #00FF00#right key#FFFFFF# to learn; #00FF00#left key#FFFFFF# to unlearn; #00FF00#+#FFFFFF# to expand a category; #00FF00#-#FFFFFF# to reduce a category. #00FF00#TAB key#FFFFFF# to switch between tabs. -Mouse: #00FF00#Left click#FFFFFF# to learn; #00FF00#right click#FFFFFF# to unlearn. -]]} - self.c_t_points = Textzone.new{width=math.floor(self.iw / 2 - 10), height=1, auto_height=true, no_color_bleed=true, text=_points_left:format(self.actor.unused_talents_types, self.actor.unused_talents, self.actor.unused_generics)} + Dialog.init(self, "Levelup: "..actor.name, 650, game.h * 0.9, game.w * 0.05, game.h * 0.05) self:generateList() - self.stats = Tab.new{title="Stats", default=true, fct=function() end, on_change=function(s) if s then self:switchTo("Stats") end end} - self.talents = Tab.new{title="Talents", default=false, fct=function() end, on_change=function(s) if s then self:switchTo("Talents") end end} - self.summary = Tab.new{title="Summary", default=false, fct=function() end, on_change=function(s) if s then self:switchTo("Summary") end end} - - self.c_t_desc = TextzoneList.new{width=math.floor(self.iw / 2 - 10), height=self.ih - self.c_t_tut.h - self.stats.h - 20 - self.vs.h, scrollbar=true, no_color_bleed=true} - - - self.hoffset = 17 + self.stats.h + self.vs.h - - self.c_t_tree = TreeList.new{width=math.floor(self.iw / 2 - 10), height=self.ih - self.stats.h - 20 - self.c_t_points.h - self.vs.h - self.vs2.h, all_clicks=true, scrollbar=true, columns={ - {width=80, display_prop="name"}, - {width=20, display_prop="status"}, - }, tree=self.tree, - fct=function(item, sel, v) self:treeSelect(item, sel, v) end, - select=function(item, sel) self:select(item) end, - on_expand=function(item) self.actor.__hidden_talent_types[item.type] = not item.shown end, - on_drawitem=function(item) if self.running then self:onDrawItem(item) end end, - } - self.c_t_tree.key:removeBind("ACCEPT") -- We want the main dialog to handle accept - - self.c_summary_desc = SurfaceZone.new{width=self.iw, height=math.min(game.h * 0.9, 400) - self.stats.h - 40 - self.vs.h,alpha=0} - - self.sel = 1 - - self.c_tut = Textzone.new{width=math.floor(self.iw / 2 - 10), auto_height=true, no_color_bleed=true, text=[[ -Keyboard: #00FF00#up key/down key#FFFFFF# to select a stat; #00FF00#right key#FFFFFF# to increase stat; #00FF00#left key#FFFFFF# to decrease a stat. #00FF00#TAB key#FFFFFF# to switch between tabs. -Mouse: #00FF00#Left click#FFFFFF# to increase a stat; #00FF00#right click#FFFFFF# to decrease a stat. -]]} - - self.c_desc = TextzoneList.new{width=math.floor(self.iw / 2 - 10), height=self.ih - self.c_tut.h - 20, no_color_bleed=true} - self.c_desc2 = TextzoneList.new{width=math.floor(self.iw / 4 - 10), height=self.ih - self.c_tut.h - 20, no_color_bleed=true} - self.c_points = Textzone.new{width=math.floor(self.iw / 2 - 10), auto_height=true, no_color_bleed=true, text=_points_text:format(self.actor.unused_stats)} - - self.c_list = ListColumns.new{width=math.floor(self.iw / 2 - 10), height=self.ih - 10 - self.vs.h - self.vs2.h, all_clicks=true, columns={ - {name="Stat", width=50, display_prop="name"}, - {name="Base", width=25, display_prop="base"}, - {name="Value", width=25, display_prop="val"}, - }, list={ - {name="Strength", base=self.actor:getStr(nil, nil, true), val=self.actor:getStr(), stat_id=self.actor.STAT_STR}, - {name="Dexterity", base=self.actor:getDex(nil, nil, true), val=self.actor:getDex(), stat_id=self.actor.STAT_DEX}, - {name="Magic", base=self.actor:getMag(nil, nil, true), val=self.actor:getMag(), stat_id=self.actor.STAT_MAG}, - {name="Willpower", base=self.actor:getWil(nil, nil, true), val=self.actor:getWil(), stat_id=self.actor.STAT_WIL}, - {name="Cunning", base=self.actor:getCun(nil, nil, true), val=self.actor:getCun(), stat_id=self.actor.STAT_CUN}, - {name="Constitution", base=self.actor:getCon(nil, nil, true), val=self.actor:getCon(), stat_id=self.actor.STAT_CON}, - }, fct=function(item, _, v) - self:incStat(v == "left" and 1 or -1) - end, select=function(item, sel) - self.sel = sel - self:onSelectStat(item) - end} - self.c_list.key:removeBind("ACCEPT") -- We want the main dialog to handle accept - - self.tab = "" - if self.actor.unused_stats > 0 then - self.stats:select() - else - self.talents:select() - end -end - -function _M:unload() - self.actor.no_last_learnt_talents_cap = nil - self.actor:capLastLearntTalents("class") - self.actor:capLastLearntTalents("generic") -end - -local prev_sel = nil -function _M:onSelectStat(item, force) - if item == prev_sel and not force then return end - prev_sel = item - local text , h = self:getStatDescription(item.stat_id) - local fh = self.font:lineSkip() - self.c_desc:createItem(item, text) - self.c_desc:switchItem(item) - local ui = self:getUIElement(self.c_desc) - if ui then - local top = ui.top - self.c_desc2.h = self.ih - fh + h + top + self.vs.h - self:moveUIElement(self.c_desc2, nil, nil, 2 * fh + h + top + self.vs.h, nil) - end - text = self:getStatNewTalents(item.stat_id) - self.c_desc2:createItem(item, text) - self.c_desc2:switchItem(item) + self:loadUI(self:createDisplay()) self:setupUI() - self:updateKeys(self.tab) -end -function _M:generateStatsList() - self.c_list.list={ - {name="Strength", color = {(self.actor_dup:getStat(self.actor.STAT_STR, nil, nil, true) == self.actor:getStat(self.actor.STAT_STR, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.actor.STAT_STR, nil, nil, true) == self.actor:getStat(self.actor.STAT_STR, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.actor.STAT_STR, nil, nil, true) - self.actor_dup:getStat(self.actor.STAT_STR, nil, nil, true) + self.actor_dup:getStat(self.actor.STAT_STR) == self.actor:getStat(self.actor.STAT_STR)) and {255, 255, 255} or {255, 215, 0}}, - base=math.floor(self.actor:getStr(nil, nil, true)), - val=math.floor(self.actor:getStr()), - stat_id=self.actor.STAT_STR}, - {name="Dexterity", color = {(self.actor_dup:getStat(self.actor.STAT_DEX, nil, nil, true) == self.actor:getStat(self.actor.STAT_DEX, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.actor.STAT_DEX, nil, nil, true) == self.actor:getStat(self.actor.STAT_DEX, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.actor.STAT_DEX, nil, nil, true) - self.actor_dup:getStat(self.actor.STAT_DEX, nil, nil, true) + self.actor_dup:getStat(self.actor.STAT_DEX) == self.actor:getStat(self.actor.STAT_DEX)) and {255, 255, 255} or {255, 215, 0}}, - base=math.floor(self.actor:getDex(nil, nil, true)), - val=math.floor(self.actor:getDex()), - stat_id=self.actor.STAT_DEX}, - {name="Magic", color = {(self.actor_dup:getStat(self.actor.STAT_MAG, nil, nil, true) == self.actor:getStat(self.actor.STAT_MAG, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.actor.STAT_MAG, nil, nil, true) == self.actor:getStat(self.actor.STAT_MAG, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.actor.STAT_MAG, nil, nil, true) - self.actor_dup:getStat(self.actor.STAT_MAG, nil, nil, true) + self.actor_dup:getStat(self.actor.STAT_MAG) == self.actor:getStat(self.actor.STAT_MAG)) and {255, 255, 255} or {255, 215, 0}}, - base=math.floor(self.actor:getMag(nil, nil, true)), - val=math.floor(self.actor:getMag()), - stat_id=self.actor.STAT_MAG}, - {name="Willpower", color = {(self.actor_dup:getStat(self.actor.STAT_WIL, nil, nil, true) == self.actor:getStat(self.actor.STAT_WIL, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.actor.STAT_WIL, nil, nil, true) == self.actor:getStat(self.actor.STAT_WIL, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.actor.STAT_WIL, nil, nil, true) - self.actor_dup:getStat(self.actor.STAT_WIL, nil, nil, true) + self.actor_dup:getStat(self.actor.STAT_WIL) == self.actor:getStat(self.actor.STAT_WIL)) and {255, 255, 255} or {255, 215, 0}}, - base=math.floor(self.actor:getWil(nil, nil, true)), - val=math.floor(self.actor:getWil()), - stat_id=self.actor.STAT_WIL}, - {name="Cunning", color = {(self.actor_dup:getStat(self.actor.STAT_CUN, nil, nil, true) == self.actor:getStat(self.actor.STAT_CUN, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.actor.STAT_CUN, nil, nil, true) == self.actor:getStat(self.actor.STAT_CUN, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.actor.STAT_CUN, nil, nil, true) - self.actor_dup:getStat(self.actor.STAT_CUN, nil, nil, true) + self.actor_dup:getStat(self.actor.STAT_CUN) == self.actor:getStat(self.actor.STAT_CUN)) and {255, 255, 255} or {255, 215, 0}}, - base=math.floor(self.actor:getCun(nil, nil, true)), - val=math.floor(self.actor:getCun()), - stat_id=self.actor.STAT_CUN}, - {name="Constitution", color = {(self.actor_dup:getStat(self.actor.STAT_CON, nil, nil, true) == self.actor:getStat(self.actor.STAT_CON, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.actor.STAT_CON, nil, nil, true) == self.actor:getStat(self.actor.STAT_CON, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.actor.STAT_CON, nil, nil, true) - self.actor_dup:getStat(self.actor.STAT_CON, nil, nil, true) + self.actor_dup:getStat(self.actor.STAT_CON) == self.actor:getStat(self.actor.STAT_CON)) and {255, 255, 255} or {255, 215, 0}}, - base=math.floor(self.actor:getCon(nil, nil, true)), - val=math.floor(self.actor:getCon()), - stat_id=self.actor.STAT_CON}, + self.key:addCommands{ + __TEXTINPUT = function(c) + if c == "+" and self.focus_ui and self.focus_ui.ui.onExpand then + self.focus_ui.ui:onExpand(self.focus_ui.ui.last_mz.item) + elseif c == "-" then + self.focus_ui.ui:onExpand(self.focus_ui.ui.last_mz.item) + end + end, } - self.c_list:onSelect(true) -end - -function _M:switchTo(kind) - if kind == "Stats" then - if self.new_talents_changed == true then - self:generateStatsList() - self.c_list:generate() - self.c_list.key:removeBind("ACCEPT") -- We want the main dialog to handle accept - self.new_talents_changed = false - end - self.talents.selected = false - self.summary.selected = false - elseif kind == "Talents" then - if self.new_stats_changed == true then - self:regenerateList() - self.new_stats_changed = false - end - self.stats.selected = false - self.summary.selected = false - - elseif kind == "Summary" then - self.talents.selected = false - self.stats.selected = false - end - self:drawDialog(kind) - self.tab = kind -end - -function _M:updateKeys(kind) - self.key:reset() - self.key:unicodeInput(true) - self.key:addBind("SCREENSHOT", function() if type(game) == "table" and game.key then game.key:triggerVirtual("SCREENSHOT") end end) - - if kind == "Stats" then - self.key:addCommands{ - _TAB = function() self:tabTabs() end, - } - self.key:addBinds{ - MOVE_LEFT = function() self:incStat(-1) end, - MOVE_RIGHT = function() self:incStat(1) end, - ACCEPT = function() self:tabTabs() end, - EXIT = function() - if self.actor.unused_stats~=self.actor_dup.unused_stats or self.actor.unused_talents_types~=self.actor_dup.unused_talents_types or - self.actor.unused_talents~=self.actor_dup.unused_talents or self.actor.unused_generics~=self.actor_dup.unused_generics then - self:yesnocancelPopup("Finish","Do you accept changes?", function(yes, cancel) - if cancel then - return nil - else - if yes then ok = self:finish() else ok = true self:cancel() end - end - if ok then - game:unregisterDialog(self) - self.actor_dup = {} - if self.on_finish then self.on_finish() end - end - end) - else - game:unregisterDialog(self) - self.actor_dup = {} - if self.on_finish then self.on_finish() end - end - end, - } - elseif kind == "Talents" then - self.key:addCommands{ - _TAB = function() self:tabTabs() end, - __TEXTINPUT = function(c) - local item = self.c_t_tree.list[self.c_t_tree.sel] - if not item or not item.type then return end - if c == "+" then - self.c_t_tree:treeExpand(true) - end - if c == "-" then - self.c_t_tree:treeExpand(false) - end - end, - } - self.key:addBinds{ - MOVE_LEFT = function() local item=self.c_t_tree.list[self.c_t_tree.sel] self:treeSelect(item, self.c_t_tree.sel, "right") end, - MOVE_RIGHT = function() local item=self.c_t_tree.list[self.c_t_tree.sel] self:treeSelect(item, self.c_t_tree.sel, "left") end, - ACCEPT = function() self:tabTabs() end, - EXIT = function() - if self.actor.unused_stats~=self.actor_dup.unused_stats or self.actor.unused_talents_types~=self.actor_dup.unused_talents_types or - self.actor.unused_talents~=self.actor_dup.unused_talents or self.actor.unused_generics~=self.actor_dup.unused_generics then - self:yesnocancelPopup("Finish","Do you accept changes?", function(yes, cancel) - if cancel then - return nil - else - if yes then ok = self:finish() else ok = true self:cancel() end - end - if ok then - game:unregisterDialog(self) - self.actor_dup = {} - if self.on_finish then self.on_finish() end - end - end) + self.key:addBinds{ + EXIT = function() + if self.actor.unused_stats~=self.actor_dup.unused_stats or self.actor.unused_talents_types~=self.actor_dup.unused_talents_types or + self.actor.unused_talents~=self.actor_dup.unused_talents or self.actor.unused_generics~=self.actor_dup.unused_generics then + self:yesnocancelPopup("Finish","Do you accept changes?", function(yes, cancel) + if cancel then + return nil else - game:unregisterDialog(self) - self.actor_dup = {} - if self.on_finish then self.on_finish() end + if yes then ok = self:finish() else ok = true self:cancel() end end - end, - } - elseif kind == "Summary" then - self.key:addCommands{ - _TAB = function() self:tabTabs() end, - } - self.key:addBinds{ - ACCEPT = "EXIT", - EXIT = function() - if self.actor.unused_stats~=self.actor_dup.unused_stats or self.actor.unused_talents_types ~= self.actor_dup.unused_talents_types or - self.actor.unused_talents ~= self.actor_dup.unused_talents or self.actor.unused_generics~=self.actor_dup.unused_generics then - self:yesnocancelPopup("Finish","Do you accept changes?", function(yes, cancel) - if cancel then - return nil - else - if yes then ok = self:finish() else ok = true self:cancel() end - end - if ok then - game:unregisterDialog(self) - self.actor_dup = {} - if self.on_finish then self.on_finish() end - end - end) - else + if ok then game:unregisterDialog(self) self.actor_dup = {} if self.on_finish then self.on_finish() end end - end, - } - end + end) + else + game:unregisterDialog(self) + self.actor_dup = {} + if self.on_finish then self.on_finish() end + end + end, + } end -function _M:cancel() ---[[ - self.actor.unused_stats = self.actor_dup.unused_stats - self.actor.unused_talents = self.actor_dup.unused_talents - self.actor.unused_generics = self.actor_dup.unused_generics - self.actor.unused_talents_types = self.actor_dup.unused_talents_types - for stat, inc in pairs(self.stats_increased) do - self.actor:incStat(stat, -inc) - end - for tt, inc in pairs(self.talent_types_learned) do - if inc[2] then - self.actor:setTalentTypeMastery(tt, self.actor:getTalentTypeMastery(tt) - 0.2) - self.actor.__increased_talent_types[tt] = self.actor_dup.__increased_talent_types[tt] - end +function _M:unload() + self.actor.no_last_learnt_talents_cap = nil + self.actor:capLastLearntTalents("class") + self.actor:capLastLearntTalents("generic") +end - if inc[1] then - self.actor:unlearnTalentType(tt) - end - end - for t_id, times in pairs(self.talents_learned) do - for i=1,times do - self.actor:unlearnTalent(t_id) - end - end - self.actor.last_learnt_talents = self.actor_dup.last_learnt_talents -]] +function _M:cancel() local ax, ay = self.actor.x, self.actor.y self.actor_dup.replacedWith = false self.actor:replaceWith(self.actor_dup) @@ -379,87 +116,11 @@ function _M:cancel() if game.level and self.actor.x then game.level.map:updateMap(self.actor.x, self.actor.y) end end -function _M:tabTabs() - if self.stats.selected == true then self.talents:select() elseif - self.talents.selected == true then self.summary:select() elseif - self.summary.selected == true then self.stats:select() end -end - -function _M:getStatDescription(stat_id) - local text = self.actor.stats_def[stat_id].description.."\n\n" - local lines = text:splitLine(self.c_desc.w, self.font) - local h = self.font:lineSkip() - local diff = self.actor:getStat(stat_id, nil, nil, true) - self.actor_dup:getStat(stat_id, nil, nil, true) - local color = diff >= 0 and "#LIGHT_GREEN#" or "#RED#" - - text = text.."#LIGHT_BLUE#Stat gives:#LAST#\n" - if stat_id == self.actor.STAT_CON then - text = text.."Max life: "..color..(diff * 4).."#LAST#\n" - text = text.."Physical save: "..color..(diff * 0.35).."#LAST#\n" - elseif stat_id == self.actor.STAT_WIL then - if self.actor:knowTalent(self.actor.T_MANA_POOL) then - text = text.."Max mana: "..color..(diff * 5).."#LAST#\n" - end - if self.actor:knowTalent(self.actor.T_STAMINA_POOL) then - text = text.."Max stamina: "..color..(diff * 2.5).."#LAST#\n" - end - if self.actor:knowTalent(self.actor.T_PSI_POOL) then - text = text.."Max PSI: "..color..(diff * 1).."#LAST#\n" - end - text = text.."Mindpower: "..color..(diff * 0.7).."#LAST#\n" - text = text.."Mental save: "..color..(diff * 0.35).."#LAST#\n" - text = text.."Spell save: "..color..(diff * 0.35).."#LAST#\n" - if self.actor.use_psi_combat then - text = text.."Accuracy: "..color..(diff * 0.35).."#LAST#\n" - end - elseif stat_id == self.actor.STAT_STR then - text = text.."Physical power: "..color..(diff).."#LAST#\n" - text = text.."Max encumberance: "..color..(diff * 1.8).."#LAST#\n" - text = text.."Physical save: "..color..(diff * 0.35).."#LAST#\n" - elseif stat_id == self.actor.STAT_CUN then - text = text.."Spell/Physical crit. chance: "..color..(diff * 0.3).."#LAST#\n" - text = text.."Mental save: "..color..(diff * 0.35).."#LAST#\n" - text = text.."Mindpower: "..color..(diff * 0.4).."#LAST#\n" - if self.actor.use_psi_combat then - text = text.."Accuracy: "..color..(diff * 0.35).."#LAST#\n" - end - elseif stat_id == self.actor.STAT_MAG then - text = text.."Spell save: "..color..(diff * 0.35).."#LAST#\n" - text = text.."Spellpower: "..color..(diff * 1).."#LAST#\n" - elseif stat_id == self.actor.STAT_DEX then - text = text.."Defense: "..color..(diff * 0.35).."#LAST#\n" - text = text.."Ranged defense: "..color..(diff * 0.35).."#LAST#\n" - text = text.."Accuracy: "..color..(diff).."#LAST#\n" - end - - if self.actor.player and self.desc_def and self.desc_def.getStatDesc and self.desc_def.getStatDesc(stat_id, self.actor) then - text = text.."#LIGHT_BLUE#Class powers:#LAST#\n" - text = text..self.desc_def.getStatDesc(stat_id, self.actor) - end - return text, h * #lines -end - function _M:getMaxTPoints(t) if t.points == 1 then return 1 end return t.points + math.max(0, math.floor((self.actor.level - 50) / 10)) end -function _M:getStatNewTalents(stat_id) - local text = "#LIGHT_BLUE#New available talents: #LAST#\n" - local stats = { "str", "dex", "mag", "wil", "cun", "con" } - local diff = self.actor:getStat(stat_id) - self.actor_dup:getStat(stat_id) - if diff ~= 0 and self.talent_stats_req[stats[stat_id]] then - for j=1,#self.talent_stats_req[stats[stat_id]] do - local t = self.actor:getTalentFromId(self.talent_stats_req[stats[stat_id]][j][1].talent) - if self.actor:canLearnTalent(t) and not self.actor_dup:canLearnTalent(t) and self:getMaxTPoints(t) > self.actor:getTalentLevelRaw(t) then - text = text.."#GOLD# - "..t.name.."#LAST#\n" - end - end - end - - return text -end - function _M:finish() local ok, dep_miss = self:checkDeps(true) if not ok then @@ -494,78 +155,34 @@ function _M:finish() if not self.actor:isTalentCoolingDown(t) and not self.actor_dup:knowTalent(t_id) then self.actor:startTalentCooldown(t) end end end - - -- Achievements checks - world:gainAchievement("ELEMENTALIST", self.actor) - world:gainAchievement("WARPER", self.actor) return true end -function _M:incStat(v) +function _M:incStat(sid, v) if v == 1 then if self.actor.unused_stats <= 0 then self:simplePopup("Not enough stat points", "You have no stat points left!") return end - if self.actor:getStat(self.sel, nil, nil, true) >= self.actor.level * 1.4 + 20 then + if self.actor:getStat(sid, nil, nil, true) >= self.actor.level * 1.4 + 20 then self:simplePopup("Stat is at the maximum for your level", "You cannot increase this stat further until next level!") return end - if self.actor:isStatMax(self.sel) or self.actor:getStat(self.sel, nil, nil, true) >= 60 + math.max(0, (self.actor.level - 50)) then + if self.actor:isStatMax(sid) or self.actor:getStat(sid, nil, nil, true) >= 60 + math.max(0, (self.actor.level - 50)) then self:simplePopup("Stat is at the maximum", "You cannot increase this stat further!") return end else - if self.actor_dup:getStat(self.sel, nil, nil, true) == self.actor:getStat(self.sel, nil, nil, true) then + if self.actor_dup:getStat(sid, nil, nil, true) == self.actor:getStat(sid, nil, nil, true) then self:simplePopup("Impossible", "You cannot take out more points!") return end end - local sel = self.sel - self.actor:incStat(sel, v) + self.actor:incStat(sid, v) self.actor.unused_stats = self.actor.unused_stats - v - self.c_list.list[sel].base = self.actor:getStat(sel, nil, nil, true) - self.c_list.list[sel].val = self.actor:getStat(sel) - self.c_list.list[sel].color = {(self.actor_dup:getStat(self.sel, nil, nil, true) == self.actor:getStat(self.sel, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor_dup:getStat(self.sel, nil, nil, true) == self.actor:getStat(self.sel, nil, nil, true)) and {255, 255, 255} or {255, 215, 0}, - (self.actor:getStat(self.sel, nil, nil, true) - self.actor_dup:getStat(self.sel, nil, nil, true) + self.actor_dup:getStat(self.sel) == self.actor:getStat(self.sel)) and {255, 255, 255} or {255, 215, 0}} - self.c_list.sel = sel - - local stats = { "str", "dex", "mag", "wil", "cun", "con" } - for i = 1,6 do - stat_sel = stats[i] - if self.talent_stats_req[stat_sel] then - to_remove = {} - for j=1,#self.talent_stats_req[stat_sel] do - local t = self.talent_stats_req[stat_sel][j][2] - if type(t.require) == "function" then - stats_req = t.require(self.actor, t).stat - for stat, _ in pairs(stats_req) do - if stat ~= stat_sel then - self.talent_stats_req[stat] = self.talent_stats_req[stat] or {} - self.talent_stats_req[stat][#self.talent_stats_req[stat] + 1] = { self.talent_stats_req[stat_sel][j][1], t } - to_remove[#to_remove+1] = j - end - end - end - end - local off = 0 - for j=1,#to_remove do - table.remove(self.talent_stats_req[stat_sel], to_remove[j] - off) - off = off + 1 - end - end - end - self.stats_increased[sel] = (self.stats_increased[sel] or 0) + v - - self.c_list:onSelect(true) - self.c_list:drawItem(self.c_list.list[self.c_list.sel]) - self.c_points.text = _points_text:format(self.actor.unused_stats) - self.c_points:generate() - self.new_stats_changed = true - self:onSelectStat(self.c_list.list[self.c_list.sel], true) + self.stats_increased[sid] = (self.stats_increased[sid] or 0) + v end function _M:computeDeps(t) @@ -599,51 +216,6 @@ function _M:computeDeps(t) end end -function _M:select(item) - if not item or not self.uis or not self.uis[5] then return end - - if not self.c_t_desc:switchItem(item) then - self:onDrawItem(item) - self.c_t_desc:switchItem(item) - end - self.cur_item = item -end - -function _M:treeSelect(item, sel, v) - if not item then return end - self:learn(v == "left" and true) - if item.nodes then - item.shown = (self.actor.__hidden_talent_types[item.type] == nil and self.actor:knowTalentType(item.type)) or (self.actor.__hidden_talent_types[item.type] ~= nil and not self.actor.__hidden_talent_types[item.type]) - self.c_t_tree:drawItem(item) - for i, n in ipairs(item.nodes) do self.c_t_tree:drawItem(n) end - elseif item.talent then - for tid, _ in pairs(self.talents_deps[item.talent] or {}) do - local it = self.c_t_tree.items_by_key[tid] - if it then self.c_t_tree:drawItem(it) end - end - local t = self.actor:getTalentFromId(item.talent) - for _, tid in ipairs(self.actor.last_learnt_talents[t.generic and "generic" or "class"]) do - local it = self.c_t_tree.items_by_key[tid] - if it then self.c_t_tree:drawItem(it) end - end - end - self.c_t_desc:switchItem(item) - self.c_t_tree:outputList() - - self.c_t_points.text = _points_left:format(self.actor.unused_talents_types, self.actor.unused_talents, self.actor.unused_generics) - self.c_t_points:generate() -end - -function _M:learn(v) - local item = self.c_t_tree.list[self.c_t_tree.sel] - if not item then return end - if item.type then - self:learnType(item.type, v) - else - self:learnTalent(item.talent, v) - end -end - function _M:checkDeps(simple) local talents = "" local stats_ok = true @@ -775,7 +347,7 @@ function _M:learnTalent(t_id, v) end end end - self:regenerateList() + self:updateTooltip() end function _M:learnType(tt, v) @@ -839,84 +411,11 @@ function _M:learnType(tt, v) end self:triggerHook{"PlayerLevelup:subTalentType", actor=self.actor, tt=tt} end -end - -function _M:onDrawItem(item) - if not item then return end - local text = tstring{} - - text:add({"color", "GOLD"}, {"font", "bold"}, util.getval(item.rawname, item), {"color", "LAST"}, {"font", "normal"}) - text:add(true, true) - - if item.type then - text:add({"color",0x00,0xFF,0xFF}, "Talent Category", true) - text:add({"color",0x00,0xFF,0xFF}, "A talent category contains talents you may learn. You gain a talent category point at level 10, 20 and 30. You may also find trainers or artifacts that allow you to learn more.\nA talent category point can be used either to learn a new category or increase the mastery of a known one.", true, true, {"color", "WHITE"}) - - if self.actor.talents_types_def[item.type].generic then - text:add({"color",0x00,0xFF,0xFF}, "Generic talent tree", true) - text:add({"color",0x00,0xFF,0xFF}, "A generic talent allows you to perform various utility actions and improve your character. It represents a skill anybody can learn (should you find a trainer for it). You gain one point every level (except every 5th level). You may also find trainers or artifacts that allow you to learn more.", true, true, {"color", "WHITE"}) - else - text:add({"color",0x00,0xFF,0xFF}, "Class talent tree", true) - text:add({"color",0x00,0xFF,0xFF}, "A class talent allows you to perform new combat moves, cast spells, and improve your character. It represents the core function of your class. You gain one point every level and two every 5th level. You may also find trainers or artifacts that allow you to learn more.", true, true, {"color", "WHITE"}) - end - - text:add(self.actor:getTalentTypeFrom(item.type).description) - - else - local t = self.actor:getTalentFromId(item.talent) - - if self:isUnlearnable(t, true) then - local max = tostring(self.actor:lastLearntTalentsMax(t.generic and "generic" or "class")) - text:add({"color","LIGHT_BLUE"}, "This talent was recently learnt, you can still unlearn it.", true, "The last ", max, t.generic and " generic" or " class", " talents you learnt are always unlearnable.", {"color","LAST"}, true, true) - elseif t.no_unlearn_last then - text:add({"color","YELLOW"}, "This talent can alter the world in a permanent way, as such you can never unlearn it once known.", {"color","LAST"}, true, true) - end - - local traw = self.actor:getTalentLevelRaw(t.id) - local diff = function(i2, i1, res) - res:add({"color", "LIGHT_GREEN"}, i1, {"color", "LAST"}, " [->", {"color", "YELLOW_GREEN"}, i2, {"color", "LAST"}, "]") - end - if traw == 0 and self:getMaxTPoints(t) >= 2 then - local req = self.actor:getTalentReqDesc(item.talent, 1):toTString():tokenize(" ()[]") - local req2 = self.actor:getTalentReqDesc(item.talent, 2):toTString():tokenize(" ()[]") - text:add{"color","WHITE"} - text:add({"font", "bold"}, "First talent level: ", tostring(traw+1), " [-> ", tostring(traw + 2), "]", {"font", "normal"}) - text:add(true) - text:merge(req2:diffWith(req, diff)) - text:merge(self.actor:getTalentFullDescription(t, 2):diffWith(self.actor:getTalentFullDescription(t, 1), diff)) - elseif traw < self:getMaxTPoints(t) then - local req = self.actor:getTalentReqDesc(item.talent):toTString():tokenize(" ()[]") - local req2 = self.actor:getTalentReqDesc(item.talent, 1):toTString():tokenize(" ()[]") - text:add{"color","WHITE"} - text:add({"font", "bold"}, traw == 0 and "Next talent level" or "Current talent level: ", tostring(traw), " [-> ", tostring(traw + 1), "]", {"font", "normal"}) - text:add(true) - text:merge(req2:diffWith(req, diff)) - text:merge(self.actor:getTalentFullDescription(t, 1):diffWith(self.actor:getTalentFullDescription(t), diff)) - else - local req = self.actor:getTalentReqDesc(item.talent) - text:add({"font", "bold"}, "Current talent level: "..traw, {"font", "normal"}) - text:add(true) - text:merge(req) - text:merge(self.actor:getTalentFullDescription(t)) - end - end - - self.c_t_desc:createItem(item, text) -end - --- Display the player tile -function _M:innerDisplay(x, y, nb_keyframes) - if self.cur_item and self.cur_item.entity and self.c_t_desc and self.ui_by_ui[self.c_t_desc] then - self.cur_item.entity:toScreen(game.uiset.hotkeys_display_icons.tiles, x + self.iw - 64, y + self.iy + self.ui_by_ui[self.c_t_desc].y - 32, 64, 64) - end + self:updateTooltip() end function _M:generateList() self.actor.__show_special_talents = self.actor.__show_special_talents or {} - self.talent_stats_req = {} - for i = 1, 6 do - self.prev_stats[i] = self.actor:getStat(i) - end -- Makes up the list local tree = {} @@ -931,12 +430,12 @@ function _M:generateList() name=function(item) return tstring{{"font", "bold"}, cat:capitalize().." / "..tt.name:capitalize() ..(" (%s)"):format((isgeneric and "generic" or "class")), {"font", "normal"}} end, rawname=function(item) return cat:capitalize().." / "..tt.name:capitalize() ..(" (%s, mastery %.2f)"):format((isgeneric and "generic" or "class"), self.actor:getTalentTypeMastery(item.type)) end, type=tt.type, - color=function(item) - return ((self.actor:knowTalentType(item.type) ~= self.actor_dup:knowTalentType(item.type)) or ((self.actor.__increased_talent_types[item.type] or 0) ~= (self.actor_dup.__increased_talent_types[item.type] or 0))) and {255, 215, 0} or self.actor:knowTalentType(item.type) and {0,200,0} or {175,175,175} - end, + color=function(item) return ((self.actor:knowTalentType(item.type) ~= self.actor_dup:knowTalentType(item.type)) or ((self.actor.__increased_talent_types[item.type] or 0) ~= (self.actor_dup.__increased_talent_types[item.type] or 0))) and {255, 215, 0} or self.actor:knowTalentType(item.type) and {0,200,0} or {175,175,175} end, shown = tshown, status = function(item) return self.actor:knowTalentType(item.type) and tstring{{"font", "bold"}, ((self.actor.__increased_talent_types[item.type] or 0) >=1) and {"color", 255, 215, 0} or {"color", 0x00, 0xFF, 0x00}, ("%.2f"):format(self.actor:getTalentTypeMastery(item.type)), {"font", "normal"}} or tstring{{"color", 0xFF, 0x00, 0x00}, "unknown"} end, nodes = {}, + isgeneric = isgeneric and 0 or 1, + order_id = i, } tree[#tree+1] = node @@ -953,7 +452,7 @@ function _M:generateList() list[#list+1] = { __id=t.id, - name=((t.display_entity and t.display_entity:getDisplayString() or "")..t.name):toTString(), + name=t.name:toTString(), rawname=t.name..(isgeneric and " (generic talent)" or " (class talent)"), entity=t.display_entity, talent=t.id, @@ -979,309 +478,196 @@ function _M:generateList() end end end - - if t.require then - local stats = {} - if type(t.require) == "table" and t.require.stat then - stats = t.require.stat - elseif type(t.require) == "function" and t.require(self.actor,t).stat then - stats = t.require(self.actor,t).stat - end - for stat, _ in pairs(stats) do - self.talent_stats_req[stat] = self.talent_stats_req[stat] or {} - self.talent_stats_req[stat][#self.talent_stats_req[stat] + 1] = { list[#list], t } - end - end end end end end + table.sort(tree, function(a, b) + if a.isgeneric == b.isgeneric then + return a.order_id < b.order_id + else + return a.isgeneric < b.isgeneric + end + end) self.tree = tree -end -function _M:regenerateList() - local stats = { "str", "dex", "mag", "wil", "cun", "con" } - local refreshed = {} - for i = 1, 6 do - local diff = self.actor:getStat(i) - self.prev_stats[i] - if diff ~= 0 and self.talent_stats_req[stats[i]] then - for j=1,#self.talent_stats_req[stats[i]] do - if not refreshed[self.talent_stats_req[stats[i]][j]] then - refreshed[self.talent_stats_req[stats[i]][j]] = true - self.c_t_tree:drawItem(self.talent_stats_req[stats[i]][j][1]) + -- Makes up the stats list + local phys, mind = {}, {} + self.tree_stats = { + {shown=true, nodes=phys, name="Physical Stats", type_stat=true, desc=self.TOOLTIP_STRDEXCON}, + {shown=true, nodes=mind, name="Mental Stats", type_stat=true, desc=TOOLTIP_MAGWILCUN} + } + + for i, sid in ipairs{self.actor.STAT_STR, self.actor.STAT_DEX, self.actor.STAT_CON, self.actor.STAT_MAG, self.actor.STAT_WIL, self.actor.STAT_CUN } do + local s = self.actor.stats_def[sid] + local e = engine.Entity.new{image="stats/"..s.name:lower()..".png", is_stat=true} + e:getMapObjects(game.uiset.hotkeys_display_icons.tiles, {}, 1) + + local stats = (i <= 3) and phys or mind + stats[#stats+1] = { + name=s.name, + rawname=s.name, + entity=e, + stat=sid, + desc=s.description, + color=function(item) + if self.actor:getStat(sid, nil, nil, true) >= self.actor.level * 1.4 + 20 or + self.actor:isStatMax(sid) or + self.actor:getStat(sid, nil, nil, true) >= 60 + math.max(0, (self.actor.level - 50)) then + return {0, 255, 0} + else + return {175,175,175} end - end - end - self.prev_stats[i] = self.actor:getStat(i) + end, + status = function(item) + if self.actor:getStat(sid, nil, nil, true) >= self.actor.level * 1.4 + 20 or + self.actor:isStatMax(sid) or + self.actor:getStat(sid, nil, nil, true) >= 60 + math.max(0, (self.actor.level - 50)) then + return tstring{{"color", 175, 175, 175}, tostring(self.actor:getStat(sid))} + else + return tstring{{"color", 0x00, 0xFF, 0x00}, tostring(self.actor:getStat(sid))} + end + end, + } end end -function _M:mouseZones(t, no_new) - -- Offset the x and y with the window position and window title - if not t.norestrict then - for i, z in ipairs(t) do - if not z.norestrict then - z.x = z.x + self.display_x + 5 - z.y = z.y + self.display_y + 20 + 3 - end - end - end - - if not no_new then self.mouse = engine.Mouse.new() end - self.mouse:registerZones(t) -end +----------------------------------------------------------------- +-- UI Stuff +----------------------------------------------------------------- -function _M:mouseTooltip(text, _, _, _, w, h, x, y) - self:mouseZones({ - { x=x, y=y + self.hoffset, w=w, h=h, fct=function(button) game.tooltip_x, game.tooltip_y = 1, 1; game:tooltipDisplayAtMap(game.w, game.h, text) end}, - }, true) -end - -function _M:drawDialog(kind) - self.mouse:reset() - if game.tooltip then - game.tooltip:erase() - end - - if kind == "Stats" then - self:resize(game.w * 0.9, math.min(game.h * 0.9, 500)) +local _points_left = [[ +Stats points left: #00FF00#%d#LAST# +Category points left: #00FF00#%d#LAST# +Class talent points left: #00FF00#%d#LAST# +Generic talent points left: #00FF00#%d#LAST#]] - self:loadUI{ - {left=0, top=self.stats.h, ui=self.vs}, +function _M:createDisplay() + self.c_tree = TalentTrees.new{ + tiles=game.uiset.hotkeys_display_icons, + tree=self.tree, + width=self.iw-200-10, height=self.ih-10, + tooltip=function(item) + local x = self.display_x + self.uis[3].x - game.tooltip.max + if self.display_x + self.w + game.tooltip.max <= game.w then x = self.display_x + self.w end + return self:getTalentDesc(item), x, nil + end, + on_use = function(item, inc) self:onUseTalent(item, inc) end, + scrollbar = true, + } - {left=15, top=0, ui=self.stats}, - {left=15+self.stats.w, top=0, ui=self.talents}, - {left=15+self.stats.w+self.talents.w, top=0, ui=self.summary}, + self.c_stat = TalentTrees.new{ + tiles=game.uiset.hotkeys_display_icons, + tree=self.tree_stats, no_cross = true, + width=200, height=210, + dont_select_top = true, + tooltip=function(item) + local x = self.display_x + self.uis[1].x + self.uis[1].ui.w + if self.display_x + self.w + game.tooltip.max <= game.w then x = self.display_x + self.w end + return item.desc, x, nil + end, + on_use = function(item, inc) self:onUseTalent(item, inc) end, + on_expand = function(item) self.actor.__hidden_talent_types[item.type] = not item.shown end, + } - {left=0, top=self.stats.h + 10 + self.vs.h, ui=self.c_points}, - {left=5, top=self.stats.h + self.c_points.h + 10 + self.vs.h, ui=self.vs2}, - {left=0, top=self.stats.h + self.c_points.h + 20 + self.vs.h + self.vs2.h, ui=self.c_list}, + self.c_points = Textzone.new{ + width=200, height=1, auto_height=true, + text=_points_left:format(self.actor.unused_stats, self.actor.unused_talents_types, self.actor.unused_talents, self.actor.unused_generics) + } - {hcenter=0, top=self.stats.h + 10 + self.vs.h, ui=Separator.new{dir="horizontal", size=self.ih - 15 - self.vs.h}}, + local vsep = Separator.new{dir="horizontal", size=self.ih - 20} + local hsep = Separator.new{dir="vertical", size=180} - {right=0, top=self.stats.h + self.c_tut.h + 25 + self.vs.h, ui=self.c_desc}, - {right=0, top=self.stats.h + self.c_tut.h + 150 + self.vs.h, ui=self.c_desc2}, + return { + {left=0, top=0, ui=self.c_stat}, + {left=self.c_stat, top=10, ui=vsep}, + {left=vsep, top=0, ui=self.c_tree}, - {right=0, top=self.stats.h + 5 + self.vs.h, ui=self.c_tut}, - } - self:setFocus(self.c_list) - self.c_list:onSelect() + {left=10, top=210, ui=hsep}, + {left=0, top=hsep, ui=self.c_points}, + } +end - self:setupUI() - self:updateTitle("Stats Levelup: "..self.actor.name) - elseif kind == "Talents" then - self:resize(game.w * 0.9, game.h * 0.9) +function _M:getTalentDesc(item) + local text = tstring{} - self:loadUI{ - {left=0, top=self.stats.h, ui=self.vs}, + text:add({"color", "GOLD"}, {"font", "bold"}, util.getval(item.rawname, item), {"color", "LAST"}, {"font", "normal"}) + text:add(true, true) - {left=15, top=0, ui=self.stats}, - {left=15+self.stats.w, top=0, ui=self.talents}, - {left=15+self.stats.w+self.talents.w, top=0, ui=self.summary}, + if item.type then + text:add({"color",0x00,0xFF,0xFF}, "Talent Category", true) + text:add({"color",0x00,0xFF,0xFF}, "A talent category contains talents you may learn. You gain a talent category point at level 10, 20 and 30. You may also find trainers or artifacts that allow you to learn more.\nA talent category point can be used either to learn a new category or increase the mastery of a known one.", true, true, {"color", "WHITE"}) - {left=0, top=self.stats.h + self.vs.h, ui=self.c_t_points}, - {left=5, top=self.stats.h + self.c_t_points.h + 5 + self.vs.h, ui=self.vs2}, - {left=0, top=self.stats.h + self.c_t_points.h + 15 + self.vs.h + self.vs2.h, ui=self.c_t_tree}, + if self.actor.talents_types_def[item.type].generic then + text:add({"color",0x00,0xFF,0xFF}, "Generic talent tree", true) + text:add({"color",0x00,0xFF,0xFF}, "A generic talent allows you to perform various utility actions and improve your character. It represents a skill anybody can learn (should you find a trainer for it). You gain one point every level (except every 5th level). You may also find trainers or artifacts that allow you to learn more.", true, true, {"color", "WHITE"}) + else + text:add({"color",0x00,0xFF,0xFF}, "Class talent tree", true) + text:add({"color",0x00,0xFF,0xFF}, "A class talent allows you to perform new combat moves, cast spells, and improve your character. It represents the core function of your class. You gain one point every level and two every 5th level. You may also find trainers or artifacts that allow you to learn more.", true, true, {"color", "WHITE"}) + end - {hcenter=0, top=self.stats.h + 5 + self.vs.h, ui=Separator.new{dir="horizontal", size=self.ih - 15 - self.vs.h}}, + text:add(self.actor:getTalentTypeFrom(item.type).description) - {right=0, top=self.stats.h + self.c_t_tut.h + 25 + self.vs.h, ui=self.c_t_desc}, - {right=0, top=self.stats.h + self.vs.h, ui=self.c_t_tut}, - } - self:setFocus(self.c_t_tree) - self:setupUI() - self:updateTitle("Talents Levelup: "..self.actor.name) else - self:resize(game.w * 0.9, math.min(game.h * 0.9, 400)) - self:loadUI{ - {left=0, top=self.stats.h, ui=self.vs}, - - {left=15, top=0, ui=self.stats}, - {left=15+self.stats.w, top=0, ui=self.talents}, - {left=15+self.stats.w+self.talents.w, top=0, ui=self.summary}, - - {left=0, top=self.stats.h + 5 + self.vs.h, ui=self.c_summary_desc}, - } + local t = self.actor:getTalentFromId(item.talent) - self:setupUI() - - local s = self.c_summary_desc.s - local h = 0 - local w = 0 - - s:erase(0,0,0,0) - - local con_diff = self.actor:getCon() - self.actor_dup:getCon() - local wil_diff = self.actor:getWil() - self.actor_dup:getWil() - local mag_diff = self.actor:getMag() - self.actor_dup:getMag() - local cun_diff = self.actor:getCun() - self.actor_dup:getCun() - local str_diff = self.actor:getStr() - self.actor_dup:getStr() - local dex_diff = self.actor:getDex() - self.actor_dup:getDex() - - s:drawColorStringBlended(self.font, ("#LIGHT_BLUE#Stats change:"):format(), w, h, 255, 255, 255, true) h = h + self.font_h - s:drawColorStringBlended(self.font, ("Strength : #00c000#%d"):format(str_diff), w, h, 255, 255, 255, true) h = h + self.font_h - s:drawColorStringBlended(self.font, ("Dexterity : #00c000#%d"):format(dex_diff), w, h, 255, 255, 255, true) h = h + self.font_h - s:drawColorStringBlended(self.font, ("Magic : #00c000#%d"):format(mag_diff), w, h, 255, 255, 255, true) h = h + self.font_h - s:drawColorStringBlended(self.font, ("Willpower : #00c000#%d"):format(wil_diff), w, h, 255, 255, 255, true) h = h + self.font_h - s:drawColorStringBlended(self.font, ("Cunning : #00c000#%d"):format(cun_diff), w, h, 255, 255, 255, true) h = h + self.font_h - s:drawColorStringBlended(self.font, ("Constitution: #00c000#%d"):format(con_diff), w, h, 255, 255, 255, true) h = h + self.font_h - h = h + self.font_h - s:drawColorStringBlended(self.font, "#LIGHT_BLUE#Saves:", w, h, 255, 255, 255, true) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_PHYS_SAVE, s:drawColorStringBlended(self.font, ("Physical : #00c000#%.2f"):format(self.actor:combatPhysicalResist(true) - self.actor_dup:combatPhysicalResist(true)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_SPELL_SAVE, s:drawColorStringBlended(self.font, ("Spell : #00c000#%.2f"):format(self.actor:combatSpellResist(true) - self.actor_dup:combatSpellResist(true)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_MENTAL_SAVE, s:drawColorStringBlended(self.font, ("Mental : #00c000#%.2f"):format(self.actor:combatMentalResist(true) - self.actor_dup:combatMentalResist(true)), w, h, 255, 255, 255, true)) h = h + self.font_h - h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_RESIST_ALL, s:drawColorStringBlended(self.font, ("All resistance: #00c000#%.2f%%"):format((self.actor.resists.all or 0) - (self.actor_dup.resists.all or 0)), w, h, 255, 255, 255, true)) h = h + self.font_h - - for i, t in ipairs(DamageType.dam_def) do - if self.actor.resists[DamageType[t.type]] and self.actor.resists[DamageType[t.type]] ~= 0 and self.actor.resists[DamageType[t.type]]~=self.actor_dup.resists[DamageType[t.type]] then - self:mouseTooltip(self.TOOLTIP_RESIST, s:drawColorStringBlended(self.font, ("%s: #00ff00#%.1f%%"):format(t.name:capitalize(), self.actor:combatGetResist(DamageType[t.type]) - (self.actor_dup:combatGetResist(DamageType[t.type]) or 0)), w, h, 255, 255, 255, true)) h = h + self.font_h - end + if self:isUnlearnable(t, true) then + local max = tostring(self.actor:lastLearntTalentsMax(t.generic and "generic" or "class")) + text:add({"color","LIGHT_BLUE"}, "This talent was recently learnt, you can still unlearn it.", true, "The last ", max, t.generic and " generic" or " class", " talents you learnt are always unlearnable.", {"color","LAST"}, true, true) + elseif t.no_unlearn_last then + text:add({"color","YELLOW"}, "This talent can alter the world in a permanent way, as such you can never unlearn it once known.", {"color","LAST"}, true, true) end - h = 0 - w = 200 - s:drawColorStringBlended(self.font, ("#LIGHT_BLUE#Stats effect:"):format(), w, h, 255, 255, 255, true) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_LIFE, s:drawColorStringBlended(self.font, ("#c00000#Max life: #00ff00#%d"):format(self.actor.max_life - self.actor_dup.max_life), w, h, 255, 255, 255, true)) h = h + self.font_h - h = h + self.font_h - if self.actor:knowTalent(self.actor.T_MANA_POOL) then - self:mouseTooltip(self.TOOLTIP_MANA,s:drawColorStringBlended(self.font, ("#7fffd4#Max mana : #00ff00#%d"):format(self.actor.max_mana - self.actor_dup.max_mana), w, h, 255, 255, 255, true)) h = h + self.font_h - end - if self.actor:knowTalent(self.actor.T_STAMINA_POOL) then - self:mouseTooltip(self.TOOLTIP_STAMINA, s:drawColorStringBlended(self.font, ("#ffcc80#Max stamina : #00ff00#%d"):format(self.actor.max_stamina - self.actor_dup.max_stamina), w, h, 255, 255, 255, true)) h = h + self.font_h - end - if self.actor:knowTalent(self.actor.T_PSI_POOL) then - self:mouseTooltip(self.TOOLTIP_PSI, s:drawColorStringBlended(self.font, ("#7fffd4#Max PSI : #00ff00#%d"):format(self.actor.max_psi - self.actor_dup.max_psi), w, h, 255, 255, 255, true)) h = h + self.font_h - end - h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_STR, s:drawColorStringBlended(self.font, ("Max encumberance: #00c000#%.1f"):format(self.actor:getMaxEncumbrance() - self.actor_dup:getMaxEncumbrance()), w, h, 255, 255, 255, true)) h = h + self.font_h - if self.actor:knowTalent(self.actor.T_MANA_POOL) or self.actor:knowTalent(self.actor.T_POSITIVE_POOL) or self.actor:knowTalent(self.actor.T_NEGATIVE_POOL) or self.actor:knowTalent(self.actor.T_VIM_POOL) or self.actor:knowTalent(self.actor.T_PARADOX_POOL) then - h = h + self.font_h - s:drawColorStringBlended(self.font, "#LIGHT_BLUE#Magical:", w, h, 255, 255, 255, true) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_SPELL_POWER, s:drawColorStringBlended(self.font, ("Spellpower: #00c000#%d"):format(self.actor:combatSpellpower() - self.actor_dup:combatSpellpower()), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_SPELL_CRIT, s:drawColorStringBlended(self.font, ("Crit. chance: #00c000#%.1f%%"):format(self.actor:combatSpellCrit() - self.actor_dup:combatSpellCrit()), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_MINDPOWER, s:drawColorStringBlended(self.font, ("Mindpower: #00c000#%.1f"):format(self.actor:combatMindpower() - self.actor_dup:combatMindpower()), w, h, 255, 255, 255, true)) h = h + self.font_h + local traw = self.actor:getTalentLevelRaw(t.id) + local diff = function(i2, i1, res) + res:add({"color", "LIGHT_GREEN"}, i1, {"color", "LAST"}, " [->", {"color", "YELLOW_GREEN"}, i2, {"color", "LAST"}, "]") end - h = h + self.font_h - local ArmorTxt = "#LIGHT_BLUE#" - if self.actor:hasHeavyArmor() then - ArmorTxt = ArmorTxt.."Heavy armor" - elseif self.actor:hasMassiveArmor() then - ArmorTxt = ArmorTxt.."Massive armor" + if traw == 0 and self:getMaxTPoints(t) >= 2 then + local req = self.actor:getTalentReqDesc(item.talent, 1):toTString():tokenize(" ()[]") + local req2 = self.actor:getTalentReqDesc(item.talent, 2):toTString():tokenize(" ()[]") + text:add{"color","WHITE"} + text:add({"font", "bold"}, "First talent level: ", tostring(traw+1), " [-> ", tostring(traw + 2), "]", {"font", "normal"}) + text:add(true) + text:merge(req2:diffWith(req, diff)) + text:merge(self.actor:getTalentFullDescription(t, 2):diffWith(self.actor:getTalentFullDescription(t, 1), diff)) + elseif traw < self:getMaxTPoints(t) then + local req = self.actor:getTalentReqDesc(item.talent):toTString():tokenize(" ()[]") + local req2 = self.actor:getTalentReqDesc(item.talent, 1):toTString():tokenize(" ()[]") + text:add{"color","WHITE"} + text:add({"font", "bold"}, traw == 0 and "Next talent level" or "Current talent level: ", tostring(traw), " [-> ", tostring(traw + 1), "]", {"font", "normal"}) + text:add(true) + text:merge(req2:diffWith(req, diff)) + text:merge(self.actor:getTalentFullDescription(t, 1):diffWith(self.actor:getTalentFullDescription(t), diff)) else - ArmorTxt = ArmorTxt.."Light armor" + local req = self.actor:getTalentReqDesc(item.talent) + text:add({"font", "bold"}, "Current talent level: "..traw, {"font", "normal"}) + text:add(true) + text:merge(req) + text:merge(self.actor:getTalentFullDescription(t)) end + end - ArmorTxt = ArmorTxt..":" - - s:drawColorStringBlended(self.font, ArmorTxt, w, h, 255, 255, 255, true) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_ARMOR_HARDINESS, s:drawColorStringBlended(self.font, ("Armor hardiness: #00c000#%d%%"):format(self.actor:combatArmorHardiness() - self.actor_dup:combatArmorHardiness()), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_ARMOR, s:drawColorStringBlended(self.font, ("Armor : #00c000#%d(%d)"):format(self.actor:combatArmor() - self.actor_dup:combatArmor(), self.actor:combatArmorHardiness() * (self.actor:combatArmor() - self.actor_dup:combatArmor()) * 0.01), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_DEFENSE, s:drawColorStringBlended(self.font, ("Defense : #00c000#%d"):format(self.actor:combatDefense(true) - self.actor_dup:combatDefense(true)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_RDEFENSE, s:drawColorStringBlended(self.font, ("Ranged defense : #00c000#%d"):format(self.actor:combatDefenseRanged(true) - self.actor_dup:combatDefenseRanged(true)), w, h, 255, 255, 255, true)) h = h + self.font_h - h = 0 - w = 400 - local mainhand = self.actor:getInven(self.actor.INVEN_MAINHAND) - if mainhand and (#mainhand > 0) then - local WeaponTxt = "#LIGHT_BLUE#Main Hand" - if self.actor:hasTwoHandedWeapon() then - WeaponTxt = WeaponTxt.."(2-handed)" - end - WeaponTxt = WeaponTxt..":" + return text +end - for i, o in ipairs(self.actor:getInven(self.actor.INVEN_MAINHAND)) do - local mean, dam = o.combat, o.combat - if o.archery and mean then - dam = (self.actor:getInven("QUIVER")[1] and self.actor:getInven("QUIVER")[1].combat) - end - if mean and dam then - s:drawColorStringBlended(self.font, WeaponTxt, w, h, 255, 255, 255, true) h = h + self.font_h - if self.actor.use_psi_combat then - self:mouseTooltip(self.TOOLTIP_COMBAT_ATTACK, s:drawColorStringBlended(self.font, ("Accuracy(PSI): #00ff00#%.1f"):format(self.actor:combatAttack(mean) - self.actor_dup:combatAttack(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - else - self:mouseTooltip(self.TOOLTIP_COMBAT_ATTACK, s:drawColorStringBlended(self.font, ("Accuracy : #00ff00#%.1f"):format(self.actor:combatAttack(mean) - self.actor_dup:combatAttack(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - end - self:mouseTooltip(self.TOOLTIP_COMBAT_DAMAGE, s:drawColorStringBlended(self.font, ("Damage : #00ff00#%.1f"):format(self.actor:combatDamage(dam) - self.actor_dup:combatDamage(dam)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_APR, s:drawColorStringBlended(self.font, ("APR : #00ff00#%.1f"):format(self.actor:combatAPR(dam) - self.actor_dup:combatAPR(dam)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_CRIT, s:drawColorStringBlended(self.font, ("Crit. chance: #00ff00#%.1f%%"):format(self.actor:combatCrit(dam) - self.actor_dup:combatCrit(dam)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_SPEED, s:drawColorStringBlended(self.font, ("Speed : #00ff00#%.2f%%"):format((self.actor:combatSpeed(mean) - self.actor_dup:combatSpeed(mean))*100), w, h, 255, 255, 255, true)) h = h + self.font_h - end - end - -- Handle bare-handed combat - else - s:drawColorStringBlended(self.font, "#LIGHT_BLUE#Unarmed:", w, h, 255, 255, 255, true) h = h + self.font_h - local mean = self.actor.combat - if mean then - if self.actor.use_psi_combat then - self:mouseTooltip(self.TOOLTIP_COMBAT_ATTACK, s:drawColorStringBlended(self.font, ("Accuracy(PSI): #00ff00#%.1f"):format(self.actor:combatAttack(mean) - self.actor_dup:combatAttack(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - else - self:mouseTooltip(self.TOOLTIP_COMBAT_ATTACK, s:drawColorStringBlended(self.font, ("Accuracy : #00ff00#%.1f"):format(self.actor:combatAttack(mean) - self.actor_dup:combatAttack(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - end - self:mouseTooltip(self.TOOLTIP_COMBAT_DAMAGE, s:drawColorStringBlended(self.font, ("Damage : #00ff00#%.1f"):format(self.actor:combatDamage(mean) - self.actor_dup:combatDamage(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_APR, s:drawColorStringBlended(self.font, ("APR : #00ff00#%.1f"):format(self.actor:combatAPR(mean) - self.actor_dup:combatAPR(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_CRIT, s:drawColorStringBlended(self.font, ("Crit. chance: #00ff00#%.1f%%"):format(self.actor:combatCrit(mean) - self.actor_dup:combatCrit(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_SPEED, s:drawColorStringBlended(self.font, ("Speed : #00ff00#%.2f%%"):format((self.actor:combatSpeed(mean) - self.actor_dup:combatSpeed(mean))*100), w, h, 255, 255, 255, true)) h = h + self.font_h - end - end - h = h + self.font_h - if self.actor:getInven(self.actor.INVEN_OFFHAND) then - for i, o in ipairs(self.actor:getInven(self.actor.INVEN_OFFHAND)) do - local act_offmult = self.actor:getOffHandMult(o.combat) - local act_dup_offmult = self.actor_dup:getOffHandMult(o.combat) - local mean, dam = o.combat, o.combat - if o.archery and mean then - dam = (self.actor:getInven("QUIVER")[1] and self.actor:getInven("QUIVER")[1].combat) - end - if mean and dam then - s:drawColorStringBlended(self.font, "#LIGHT_BLUE#Off Hand:", w, h, 255, 255, 255, true) h = h + self.font_h - if self.actor.use_psi_combat then - self:mouseTooltip(self.TOOLTIP_COMBAT_ATTACK, s:drawColorStringBlended(self.font, ("Accuracy(PSI): #00ff00#%.1f"):format(self.actor:combatAttack(mean) - self.actor_dup:combatAttack(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - else - self:mouseTooltip(self.TOOLTIP_COMBAT_ATTACK, s:drawColorStringBlended(self.font, ("Accuracy : #00ff00#%.1f"):format(self.actor:combatAttack(mean) - self.actor_dup:combatAttack(mean)), w, h, 255, 255, 255, true)) h = h + self.font_h - end - self:mouseTooltip(self.TOOLTIP_COMBAT_DAMAGE, s:drawColorStringBlended(self.font, ("Damage : #00ff00#%.1f"):format(self.actor:combatDamage(dam) * act_offmult - self.actor_dup:combatDamage(dam) * act_dup_offmult), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_APR , s:drawColorStringBlended(self.font, ("APR : #00ff00#%.1f"):format(self.actor:combatAPR(dam) - self.actor_dup:combatAPR(dam)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_CRIT , s:drawColorStringBlended(self.font, ("Crit. chance: #00ff00#%.1f%%"):format(self.actor:combatCrit(dam) - self.actor_dup:combatCrit(dam)), w, h, 255, 255, 255, true)) h = h + self.font_h - self:mouseTooltip(self.TOOLTIP_COMBAT_SPEED , s:drawColorStringBlended(self.font, ("Speed : #00ff00#%.2f%%"):format((self.actor:combatSpeed(mean) - self.actor_dup:combatSpeed(mean))*100), w, h, 255, 255, 255, true)) h = h + self.font_h - end - end - end - h = 0 - w = 600 - s:drawColorStringBlended(self.font, ("#LIGHT_BLUE#Talents learned:"):format(), w, h, 255, 255, 255, true) h = h + self.font_h - local stats = { "str", "dex", "mag", "wil", "cun", "con" } - local talents_learned = {} - - for k, _ in pairs(self.talents_changed) do - t = self.actor:getTalentFromId(k) - if ((self.actor.talents[k] or 0) ~= (self.actor_dup.talents[k] or 0)) then - talents_learned[k] = true - local desc = "#GOLD##{bold}#"..t.name.."#{normal}##WHITE#\n"..tostring(self.actor:getTalentFullDescription(t)) - self:mouseTooltip(desc, s:drawColorStringBlended(self.font, ("#GOLD#%s"):format(t.name), w, h, 255, 255, 255, true)) h = h + self.font_h - end - end - h = 0 - w = 800 - s:drawColorStringBlended(self.font, ("#LIGHT_BLUE#New available talents:"):format(), w, h, 255, 255, 255, true) h = h + self.font_h - talents_added = {} - for stat_id = 1,6 do - if self.talent_stats_req[stats[stat_id]] then - for j=1,#self.talent_stats_req[stats[stat_id]] do - local t = self.actor:getTalentFromId(self.talent_stats_req[stats[stat_id]][j][1].talent) - if self.actor:canLearnTalent(t) and not self.actor_dup:canLearnTalent(t) and self:getMaxTPoints(t) > self.actor:getTalentLevelRaw(t) and not talents_added[self.talent_stats_req[stats[stat_id]][j][1].talent] and not talents_learned[self.talent_stats_req[stats[stat_id]][j][1].talent] then - talents_added[self.talent_stats_req[stats[stat_id]][j][1].talent] = true - local desc = "#GOLD##{bold}#"..t.name.."#{normal}##WHITE#\n"..tostring(self.actor:getTalentFullDescription(t)) - self:mouseTooltip(desc, s:drawColorStringBlended(self.font, ("#GOLD#%s"):format(t.name), w, h, 255, 255, 255, true)) h = h + self.font_h - end - end - end - end +function _M:onUseTalent(item, inc) + if item.type then + self:learnType(item.type, inc) + item.shown = (self.actor.__hidden_talent_types[item.type] == nil and self.actor:knowTalentType(item.type)) or (self.actor.__hidden_talent_types[item.type] ~= nil and not self.actor.__hidden_talent_types[item.type]) + self.c_tree:redrawAllItems() + elseif item.talent then + self:learnTalent(item.talent, inc) + self.c_tree:redrawAllItems() + elseif item.stat then + self:incStat(item.stat, inc and 1 or -1) + self.c_stat:redrawAllItems() + self.c_tree:redrawAllItems() + end - self.c_summary_desc:generate() - self.c_summary_desc.can_focus = true - self:setFocus(self.c_summary_desc) + self.c_points.text = _points_left:format(self.actor.unused_stats, self.actor.unused_talents_types, self.actor.unused_talents, self.actor.unused_generics) + self.c_points:generate() +end - self:updateTitle("Levelup summary: "..self.actor.name) - end - self:updateKeys(kind) +function _M:updateTooltip() + self.c_tree:updateTooltip() end diff --git a/game/modules/tome/dialogs/elements/TalentTrees.lua b/game/modules/tome/dialogs/elements/TalentTrees.lua index d9faf57728..23997c05d2 100644 --- a/game/modules/tome/dialogs/elements/TalentTrees.lua +++ b/game/modules/tome/dialogs/elements/TalentTrees.lua @@ -35,6 +35,7 @@ function _M:init(t) self.on_expand = t.on_expand self.scrollbar = t.scrollbar self.no_cross = t.no_cross + self.dont_select_top = t.dont_select_top self.icon_size = 48 self.frame_size = 50 @@ -53,10 +54,12 @@ function _M:init(t) end function _M:onUse(item, inc) + self.last_scroll = nil self.on_use(item, inc) end function _M:onExpand(item, inc) + self.last_scroll = nil item.shown = not item.shown if self.on_expand then self.on_expand(item) end end @@ -75,6 +78,58 @@ function _M:doScroll(v) self.scroll = util.bound(self.scroll + v, 1, self.max_display) end +function _M:moveSel(i, j) + local match = nil + + if i == 0 then + local t = self.tree[self.sel_i] + if t.nodes then + self.sel_j = util.bound(self.sel_j + j, 1, #t.nodes) + match = t.nodes[self.sel_j] + end + elseif i == 1 then + local t = self.tree[self.sel_i] + if t.shown and self.sel_j == 0 and t.nodes and #t.nodes > 0 then + self.sel_j = 1 + match = t.nodes[1] + else + self.sel_i = util.bound(self.sel_i + i, 1, #self.tree) + self.sel_j = 0 + self.scroll = util.scroll(self.sel_i, self.scroll, self.max_display) + local t = self.tree[self.sel_i] + match = t + end + elseif i == -1 then + local t = self.tree[self.sel_i] + if t.shown and self.sel_j > 0 and t.nodes and #t.nodes > 0 then + self.sel_j = 0 + match = t + else + self.sel_i = util.bound(self.sel_i + i, 1, #self.tree) + self.scroll = util.scroll(self.sel_i, self.scroll, self.max_display) + local t = self.tree[self.sel_i] + if t.shown and t.nodes and #t.nodes > 0 then + self.sel_j = 1 + match = t.nodes[1] + else + self.sel_j = 0 + match = t + end + end + end + + for i = 1, #self.mousezones do + local mz = self.mousezones[i] + if mz.item == match then self.last_mz = mz break end + end + + local str, fx, fy = self.tooltip(self.last_mz.item) + self.last_mz.tx, self.last_mz.ty = fx or (self.last_display_x + self.last_mz.x2), fy or (self.last_display_y + self.last_mz.y1) + game:tooltipDisplayAtMap(self.last_mz.tx, self.last_mz.ty, str) + + self.last_scroll = nil +end + function _M:generate() self.mouse:reset() self.key:reset() @@ -86,6 +141,9 @@ function _M:generate() self.scrollbar = Slider.new{size=self.h, max=#self.tree} end + self.sel_i = 1 + self.sel_j = 1 + self.mousezones = {} self:redrawAllItems() @@ -116,6 +174,8 @@ function _M:generate() end self.last_mz = mz + self.sel_i = mz.i + self.sel_j = mz.j done = true break end @@ -123,19 +183,20 @@ function _M:generate() if not done then game.tooltip_x = nil self.last_mz = nil end end) self.key:addBinds{ - ACCEPT = function() self:onUse("left", "key") end, - MOVE_UP = function() - end, - MOVE_DOWN = function() - end, + ACCEPT = function() self:onUse(self.last_mz.item, true) end, + MOVE_UP = function() self:moveSel(-1, 0) end, + MOVE_DOWN = function() self:moveSel(1, 0) end, + MOVE_LEFT = function() self:moveSel(0, -1) end, + MOVE_RIGHT = function() self:moveSel(0, 1) end, } self.key:addCommands{ + [{"_RETURN","ctrl"}] = function() self:onUse(self.last_mz.item, false) end, [{"_UP","ctrl"}] = function() self.key:triggerVirtual("MOVE_UP") end, [{"_DOWN","ctrl"}] = function() self.key:triggerVirtual("MOVE_DOWN") end, - _HOME = function() - end, - _END = function() - end, + [{"_LEFT","ctrl"}] = function() self.key:triggerVirtual("MOVE_LEFT") end, + [{"_RIGHT","ctrl"}] = function() self.key:triggerVirtual("MOVE_RIGHT") end, + _HOME = function() self.sel_i = 1 self:moveSel(-1, 0) end, + _END = function() self.sel_i = #self.tree self:moveSel(1, 0) end, _PAGEUP = function() end, _PAGEDOWN = function() @@ -184,7 +245,7 @@ function _M:display(x, y, nb_keyframes, screen_x, screen_y) self.last_display_y = screen_y local mz = {} - self.mousezones = mz + if self.last_scroll ~= self.scroll then self.mousezones = mz end local dx, dy = 0, 0 @@ -200,7 +261,7 @@ function _M:display(x, y, nb_keyframes, screen_x, screen_y) local key = tree.text_status local cross = not tree.shown and self.plus or self.minus - mz[#mz+1] = {item=tree, x1=dx, y1=dy, x2=dx+cross.w + 3 + key.w, y2=dy+cross.h} + mz[#mz+1] = {i=i,j=1, item=tree, x1=dx, y1=dy, x2=dx+cross.w + 3 + key.w, y2=dy+cross.h} if not self.no_cross then cross.t:toScreenFull(dx+x, dy+y + (-cross.h + key.h) / 2, cross.w, cross.h, cross.tw, cross.th) @@ -228,7 +289,7 @@ function _M:display(x, y, nb_keyframes, screen_x, screen_y) addh = key.h end - mz[#mz+1] = {item=tal, x1=dx, y1=dy, x2=dx+self.frame_size, y2=dy+self.frame_size+addh} + mz[#mz+1] = {i=i, j=j, item=tal, x1=dx, y1=dy, x2=dx+self.frame_size, y2=dy+self.frame_size+addh} dx = dx + self.frame_size + self.frame_offset addh = addh + self.frame_size @@ -243,4 +304,6 @@ function _M:display(x, y, nb_keyframes, screen_x, screen_y) self.scrollbar.pos = self.scroll self.scrollbar:display(x + self.w - self.scrollbar.w, y) end + + self.last_scroll = self.scroll end -- GitLab