From 9756808797c9c48534662cf67b55189f6853c1a9 Mon Sep 17 00:00:00 2001 From: Bunny <glisa825@gmail.com> Date: Tue, 12 May 2020 23:19:46 -0400 Subject: [PATCH] Rewrite functions for scaling with diminishing returns approaching a limit --- game/modules/tome/class/interface/Combat.lua | 118 +++++++++---------- 1 file changed, 55 insertions(+), 63 deletions(-) diff --git a/game/modules/tome/class/interface/Combat.lua b/game/modules/tome/class/interface/Combat.lua index 1ee66a881c..29c6b0e65d 100644 --- a/game/modules/tome/class/interface/Combat.lua +++ b/game/modules/tome/class/interface/Combat.lua @@ -1489,27 +1489,22 @@ end -- x = a numeric value -- limit = value approached as x increases -- y_high = value to match at when x = x_high --- y_low (optional) = value to match when x = x_low --- returns (limit - add)*x/(x + halfpoint) + add (= add when x = 0 and limit when x = infinity) --- halfpoint and add are internally computed to match the desired high/low values --- note that the progression low->high->limit must be monotone, consistently increasing or decreasing +-- y_low = value to match when x = x_low +-- returns limit * (1-{a*(x)^0.75+b}) +-- (1-(e)^(-x)) ranges from 0 to 1 and can effectively be thought of as giving a percent of limit based on talent level (note that a*(x)^0.75+b will be < 0 for all x) +-- note that the progression low->high->limit must be monotone, consistently increasing or decreasing function _M:combatLimit(x, limit, y_low, x_low, y_high, x_high) --- local x_low, x_high = 1,5 -- Implied talent levels for low and high values respectively --- local tl = type(t) == "table" and (raw and self:getTalentLevelRaw(t) or self:getTalentLevel(t)) or t - if y_low and x_low then - local p = limit*(x_high-x_low) - local m = x_high*y_high - x_low*y_low --- local halfpoint = (p-m)/(y_high - y_low) --- local add = (limit*(x_high*y_low-x_low*y_high) + y_high*y_low*(x_low-x_high))/(p-m) --- return (limit-add)*x/(x + halfpoint) + add - local ah = (limit*(x_high*y_low-x_low*y_high)+ y_high*y_low*(x_low-x_high))/(y_high - y_low) -- add*halfpoint product calculated at once to avoid possible divide by zero - return (limit*x + ah)/(x + (p-m)/(y_high - y_low)) --factored version of above formula --- return (limit-add)*x/(x + halfpoint) + add, halfpoint, add - else - local add = 0 - local halfpoint = limit*x_high/(y_high-add)-x_high - return (limit-add)*x/(x + halfpoint) + add --- return (limit-add)*x/(x + halfpoint) + add, halfpoint, add + x_low = x_low^0.75 + x_high = x_high^0.75 + if y_high >= y_low then -- find a and b such that (1-exp{a*sqrt(x)+b}) * limit returns y_low at x_low and y_high at x_high + -- gaze in horror at how we find our constants + local a = math.log( (y_high-limit)/(y_low-limit) )/(x_high - x_low) + local b = -( (x_high - x_low) * math.log(1 - y_high/limit) - x_high * math.log( (y_high-limit)/(y_low-limit) ) )/(x_low - x_high) + return limit * (1 - math.exp( ((x)^.75*a+b)) ) + elseif y_low > y_high then -- find a and b such that y_low - (y_low-limit)*(1-exp{a*(x)^0.75+b}) returns y_low at x_low and y_high at x_high + local a = math.log( (y_high-limit)/(y_low-limit) ) / (x_high - x_low) + local b = -( (x_high - x_low)*math.log(1-(y_low-y_high)/(y_low-limit)) - x_high * math.log( (y_high-limit)/(y_low-limit) ) )/(x_low - x_high) + return y_low - (y_low-limit) * (1 - math.exp( ((x)^.75*a+b) )) end end @@ -1573,57 +1568,54 @@ function _M:combatStatScale(stat, low, high, power, add, shift) end end --- Compute a diminishing returns value based on talent level that cannot go beyond a limit +-- Compute a diminishing returns value based on talent level using exponentials that cannot go beyond a limit -- t = talent def table or a numeric value -- limit = value approached as talent levels increase --- high = value at talent level 5 --- low = value at talent level 1 (optional) +-- high = value at talent level 5 multiplied by mastery (default 6.5) +-- low = value at talent level 1 multiplied by mastery (default 1.3) -- raw if true specifies use of raw talent level --- returns (limit - add)*TL/(TL + halfpoint) + add == add when TL = 0 and limit when TL = infinity --- TL = talent level, halfpoint and add are internally computed to match the desired high/low values --- note that the progression low->high->limit must be monotone, consistently increasing or decreasing -function _M:combatTalentLimit(t, limit, low, high, raw) - local x_low, x_high = 1,5 -- Implied talent levels for low and high values respectively +-- mastery = value used for determining high and low +-- returns limit * (1-exp{a*sqrt(tl)+b}) +-- (1-(e)^(-x)) ranges from 0 to 1 and can effectively be thought of as giving a percent of limit based on talent level (note that a*sqrt(tl)+b will be < 0 for all tl) +-- note that the progression low->high->limit must be monotone, consistently increasing or decreasing +function _M:combatTalentLimit(t, limit, low, high, raw, mastery) + local x_low = mastery and math.sqrt(mastery) or math.sqrt(1.3) + local x_high = mastery and math.sqrt(mastery*5) or math.sqrt(6.5) local tl = type(t) == "table" and (raw and self:getTalentLevelRaw(t) or self:getTalentLevel(t)) or t - if tl <= 0 then tl = 0.1 end - if low then - local p = limit*(x_high-x_low) - local m = x_high*high - x_low*low --- local halfpoint = (p-m)/(high - low) -- point at which half progress towards the limit is reached --- local add = (limit*(x_high*low-x_low*high) + high*low*(x_low-x_high))/(p-m) - local ah = (limit*(x_high*low-x_low*high)+ high*low*(x_low-x_high))/(high - low) -- add*halfpoint product calculated at once to avoid possible divide by zero - return (limit*tl + ah)/(tl + (p-m)/(high - low)) --factored version of above formula --- return (limit-add)*tl/(tl + halfpoint) + add, halfpoint, add - else -- assume low and x_low are both 0 - local halfpoint = limit*x_high/high-x_high - return limit*tl/(tl + halfpoint) --- return (limit-add)*tl/(tl + halfpoint) + add, halfpoint, add - end -end - --- Compute a diminishing returns value based on a stat value that cannot go beyond a limit + if tl <= 0 then tl = 0.5 end + if high >= low then -- find a and b such that (1-exp{a*sqrt(tl)+b}) * limit returns low at x_low and high at x_high + -- gaze in horror at how we find our constants + local a = math.log( (high-limit)/(low-limit) )/(x_high - x_low) + local b = -( (x_high - x_low) * math.log(1 - high/limit) - x_high * math.log( (high-limit)/(low-limit) ) )/(x_low - x_high) + return limit * (1 - math.exp( (math.sqrt(tl)*a+b)) ) + elseif low > high then -- find a and b such that low - (low-limit)*(1-exp{a*sqrt(tl)+b}) returns low at x_low and high at x_high + local a = math.log( (high-limit)/(low-limit) ) / (x_high - x_low) + local b = -( (x_high - x_low)*math.log(1-(low-high)/(low-limit)) - x_high * math.log( (high-limit)/(low-limit) ) )/(x_low - x_high) + return low - (low-limit) * (1 - math.exp( (math.sqrt(tl)*a+b) )) + end +end + +-- Compute a diminishing returns value based on a stat value using exponentials that cannot go beyond a limit -- stat == "str", "con",.... or a numeric value --- limit = value approached as talent levels increase +-- limit = value approached as stats increase -- high = value to match when stat = 100 --- low = value to match when stat = 10 (optional) --- returns (limit - add)*stat/(stat + halfpoint) + add == add when STAT = 0 and limit when stat = infinity --- halfpoint and add are internally computed to match the desired high/low values --- note that the progression low->high->limit must be monotone, consistently increasing or decreasing +-- low = value to match when stat = 10 +-- returns limit * (1-exp{a*(stat)^0.75+b}) +-- (1-(e)^(-x)) ranges from 0 to 1 and can effectively be thought of as giving a percent of limit based on talent level (note that a*(stat)^0.75+b will be < 0 for all tl) +-- note that the progression low->high->limit must be monotone, consistently increasing or decreasing function _M:combatStatLimit(stat, limit, low, high) - local x_low, x_high = 10,100 -- Implied stat levels for low and high values respectively + local x_low = 5.6234 -- 10^0.75 + local x_high = 31.623 --100^0.75 stat = type(stat) == "string" and self:getStat(stat,nil,true) or stat - if low then - local p = limit*(x_high-x_low) - local m = x_high*high - x_low*low --- local halfpoint = (p-m)/(high - low) -- point at which half progress towards the limit is reached --- local add = (limit*(x_high*low-x_low*high) + high*low*(x_low-x_high))/(p-m) - local ah = (limit*(x_high*low-x_low*high)+ high*low*(x_low-x_high))/(high - low) -- add*halfpoint product calculated at once to avoid possible divide by zero - return (limit*stat + ah)/(stat + (p-m)/(high - low)) --factored version of above formula --- return (limit-add)*stat/(stat + halfpoint) + add, halfpoint, add - else -- assume low and x_low are both 0 - local halfpoint = limit*x_high/high-x_high - return limit*stat/(stat + halfpoint) --- return (limit-add)*stat/(stat + halfpoint) + add, halfpoint, add + if high >= low then -- find a and b such that (1-exp{a*(stat)^0.75+b}) * limit returns low at 10 stat and high at 100 + -- gaze in horror at how we find our constants + local a = math.log( (high-limit)/(low-limit) )/(x_high - x_low) + local b = -( (x_high - x_low) * math.log(1 - high/limit) - x_high * math.log( (high-limit)/(low-limit) ) )/(x_low - x_high) + return limit * (1 - math.exp( ((stat)^.75*a+b)) ) + elseif low > high then -- find a and b such that low - (low-limit)*(1-exp{a*(stat)^0.75+b}) returns low at 10 stat and high at 100 + local a = math.log( (high-limit)/(low-limit) ) / (x_high - x_low) + local b = -( (x_high - x_low)*math.log(1-(low-high)/(low-limit)) - x_high * math.log( (high-limit)/(low-limit) ) )/(x_low - x_high) + return low - (low-limit) * (1 - math.exp( ((stat)^.75*a+b) )) end end -- GitLab