From 13800606275efcbb564465d13aa9921434d6f596 Mon Sep 17 00:00:00 2001
From: dg <dg@51575b47-30f0-44d4-a5cc-537603b46e54>
Date: Mon, 3 Oct 2011 15:39:42 +0000
Subject: [PATCH] Big saves overhaul: <li>Display change: Combat
 stats(Accuracy, Spellpower, Mindpower, Physical power) and Saves (Defense,
 Spell save, Mental save, Physical save) are now computed on a O to 100 scale,
 split in 5 tiers. In each tier it takes more points to upgrade. So if you
 have 22 spellpower and wear a new item granting +2 spellpower you only get up
 to 23 spellpower. *This does NOT change the damage you are doing since a
 similar scale was applied to damage done in previous betas*</li> <li>Saves
 now reduce the duration of timed effects</li> <li>When a creature is hit by a
 special effect from a creature whose combat stat is in a tier over the
 target's save tier it triggers a "cross tier effect"</li> <li>Cross tier
 effects are dependant on the nature of the effet. A powerful knockback will
 set a target offbalance, a terrible mind attack will brainlock the target, a
 overpowering spell will spellshock the target</li> <li>Tooltips show both
 combat stats & saves for all creature</li>

git-svn-id: http://svn.net-core.org/repos/t-engine4@4498 51575b47-30f0-44d4-a5cc-537603b46e54
---
 game/modules/tome/class/Actor.lua             |  62 ++++--
 game/modules/tome/class/Object.lua            |   3 +
 game/modules/tome/class/PlayerDisplay.lua     |  17 +-
 game/modules/tome/class/interface/Archery.lua |   2 +-
 game/modules/tome/class/interface/Combat.lua  | 192 +++++++++++++---
 .../tome/class/interface/TooltipsData.lua     |  37 ++--
 .../tome/data/birth/races/tutorial.lua        |   6 +-
 game/modules/tome/data/damage_types.lua       |  70 +++---
 .../data/general/objects/boss-artifacts.lua   |   2 +-
 .../tome/data/general/objects/egos/weapon.lua |   4 +-
 .../data/general/objects/quest-artifacts.lua  |   2 +-
 .../data/general/objects/world-artifacts.lua  |  14 +-
 .../tome/data/maps/vaults/trickvault.lua      |   8 +
 .../tome/data/talents/chronomancy/energy.lua  |   2 +-
 .../talents/chronomancy/spacetime-folding.lua |   1 +
 .../tome/data/talents/corruptions/blight.lua  |   1 +
 .../tome/data/talents/corruptions/scourge.lua |   4 +-
 .../tome/data/talents/cunning/ambush.lua      |   4 +-
 .../tome/data/talents/cunning/dirty.lua       |   4 +-
 .../tome/data/talents/cunning/traps.lua       |  10 +-
 .../tome/data/talents/cursed/slaughter.lua    |   3 +-
 .../tome/data/talents/gifts/fire-drake.lua    |   2 +-
 .../tome/data/talents/gifts/sand-drake.lua    |   2 +-
 .../tome/data/talents/gifts/storm-drake.lua   |   1 +
 .../data/talents/gifts/summon-utility.lua     |   2 +-
 .../tome/data/talents/misc/horrors.lua        |   5 +-
 game/modules/tome/data/talents/misc/misc.lua  |   1 +
 game/modules/tome/data/talents/misc/npcs.lua  |  20 +-
 .../tome/data/talents/misc/tutorial.lua       | 207 ++++++++++++++++++
 .../tome/data/talents/psionic/absorption.lua  |   2 +-
 .../talents/psionic/augmented-mobility.lua    |   7 +-
 .../psionic/finer-energy-manipulations.lua    |  25 +--
 .../tome/data/talents/psionic/focus.lua       |   9 +-
 .../tome/data/talents/psionic/grip.lua        |   8 +-
 .../tome/data/talents/psionic/projection.lua  |   6 +-
 .../tome/data/talents/psionic/voracity.lua    |  53 +++--
 .../data/talents/spells/arcane-shield.lua     |   4 +-
 .../tome/data/talents/spells/golem.lua        |   7 +-
 .../tome/data/talents/spells/staff-combat.lua |  10 +-
 .../tome/data/talents/techniques/2hweapon.lua |  12 +-
 .../tome/data/talents/techniques/archery.lua  |   6 +-
 .../talents/techniques/battle-tactics.lua     |   2 +-
 .../data/talents/techniques/bloodthirst.lua   |   2 +-
 .../tome/data/talents/techniques/bow.lua      |   7 +-
 .../talents/techniques/combat-training.lua    |  26 ++-
 .../data/talents/techniques/dualweapon.lua    |   2 +-
 .../data/talents/techniques/field-control.lua |   7 +-
 .../talents/techniques/finishing-moves.lua    |   6 +-
 .../data/talents/techniques/grappling.lua     |  12 +-
 .../tome/data/talents/techniques/sling.lua    |  12 +-
 .../tome/data/talents/techniques/thuggery.lua |   2 +-
 .../talents/techniques/unarmed-discipline.lua |   2 +-
 .../talents/techniques/unarmed-training.lua   |  10 +-
 .../tome/data/talents/techniques/warcries.lua |   2 +-
 .../tome/data/talents/undeads/ghoul.lua       |   2 +-
 .../texts/tutorial/combat-stats-intro.lua     |  23 ++
 .../tome/data/timed_effects/magical.lua       |  21 ++
 .../tome/data/timed_effects/mental.lua        |  48 +++-
 .../tome/data/timed_effects/physical.lua      |  18 +-
 59 files changed, 782 insertions(+), 259 deletions(-)
 create mode 100644 game/modules/tome/data/talents/misc/tutorial.lua
 create mode 100644 game/modules/tome/data/texts/tutorial/combat-stats-intro.lua

diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua
index 2218c8de7c..afb140a958 100644
--- a/game/modules/tome/class/Actor.lua
+++ b/game/modules/tome/class/Actor.lua
@@ -285,7 +285,7 @@ function _M:actBase()
 	end
 
 	-- Cooldown talents
-	if not self:attr("stunned") then self:cooldownTalents() end
+	if not self:attr("no_talents_cooldown") then self:cooldownTalents() end
 	-- Regen resources
 	self:regenLife()
 	if self:knowTalent(self.T_UNNATURAL_BODY) then
@@ -650,7 +650,7 @@ function _M:move(x, y, force)
 		if self:attr("lightning_speed") or self:attr("step_up") or self:attr("wild_speed") then blur = 3 end
 		self:setMoveAnim(ox, oy, config.settings.tome.smooth_move, blur)
 	end
-	
+
 	if moved and not force and self:knowTalent(self.T_HASTE) and self:hasEffect(self.EFF_HASTE) then
 		local t = self:getTalentFromId(self.T_HASTE)
 		t.do_haste_double(self, t, ox, oy)
@@ -871,6 +871,13 @@ function _M:getRankLifeAdjust(value)
 	end
 end
 
+
+
+
+
+
+
+
 function _M:getRankResistAdjust()
 	if self.rank == 1 then return 0.4, 0.9
 	elseif self.rank == 2 then return 0.5, 1.5
@@ -906,6 +913,26 @@ function _M:TextSizeCategory()
 	return sizecat
 end
 
+function _M:colorStats(stat)
+	local score = math.floor(self[stat](self, fake))
+
+	if score <= 9 then
+		return "#B4B4B4# "..score
+	elseif score <= 20 then
+		return "#B4B4B4#"..score
+	elseif score <= 40 then
+		return "#FFFFFF#"..score
+	elseif score <= 60 then
+		return "#00FF80#"..score
+	elseif score <= 80 then
+		return "#0080FF#"..score
+	elseif score <= 99 then
+		return "#8d55ff#"..score
+	elseif score == 100 then
+		return "#8d55ff#**"
+	end
+end
+
 function _M:tooltip(x, y, seen_by)
 	if seen_by and not seen_by:canSee(self) then return end
 	local factcolor, factstate, factlevel = "#ANTIQUE_WHITE#", "neutral", Faction:factionReaction(self.faction, game.player.faction)
@@ -944,26 +971,31 @@ function _M:tooltip(x, y, seen_by)
 	ts:add(self.type:capitalize(), " / ", self.subtype:capitalize(), true)
 	ts:add("Rank: ") ts:merge(rank_color:toTString()) ts:add(rank, {"color", "WHITE"}, true)
 	ts:add({"color", 0, 255, 255}, ("Level: %d"):format(self.level), {"color", "WHITE"}, true)
-	ts:add(("Exp: %d/%d"):format(self.exp, self:getExpChart(self.level+1) or "---"), true)
-	if self.life >= 0 then
-		ts:add({"color", 255, 0, 0}, ("HP: %d (%d%%)"):format(self.life, self.life * 100 / self.max_life), {"color", "WHITE"}, true)
-	else
-		ts:add({"color", 255, 0, 0}, "HP: ???", {"color", "WHITE"}, true)
-	end
+	ts:add({"color", 255, 0, 0}, ("HP: %d (%d%%)"):format(self.life, self.life * 100 / self.max_life), {"color", "WHITE"}, true)
+
 	if self:attr("encased_in_ice") then
 		local eff = self:hasEffect(self.EFF_FROZEN)
 		ts:add({"color", 0, 255, 128}, ("Iceblock: %d"):format(eff.hp), {"color", "WHITE"}, true)
 	end
-	ts:add(("Stats: %d / %d / %d / %d / %d / %d"):format(self:getStr(), self:getDex(), self:getCon(), self:getMag(), self:getWil(), self:getCun()), true)
-	ts:add("Resists: ", table.concat(resists, ','), true)
-	ts:add("Armour/Defense: ", tostring(math.floor(self:combatArmor())), ' / ', tostring(math.floor(self:combatDefense(true))), true)
+	--ts:add(("Stats: %d / %d / %d / %d / %d / %d"):format(self:getStr(), self:getDex(), self:getCon(), self:getMag(), self:getWil(), self:getCun()), true)
+	if #resists > 0 then ts:add("Resists: ", table.concat(resists, ','), true) end
+	ts:add("Hardiness/Armour: ", tostring(math.floor(self:combatArmorHardiness())), '% / ', tostring(math.floor(self:combatArmor())), true)
 	ts:add("Size: ", {"color", "ANTIQUE_WHITE"}, self:TextSizeCategory(), {"color", "WHITE"}, true)
+
+	ts:add("#FFD700#Accuracy#FFFFFF#: ", self:colorStats("combatAttack"), "  ")
+	ts:add("#0080FF#Defense#FFFFFF#:  ", self:colorStats("combatDefense"), true)
+	ts:add("#FFD700#P. power#FFFFFF#: ", self:colorStats("combatPhysicalpower"), "  ")
+	ts:add("#0080FF#P. save#FFFFFF#:  ", self:colorStats("combatPhysicalResist"), true)
+	ts:add("#FFD700#S. power#FFFFFF#: ", self:colorStats("combatSpellpower"), "  ")
+	ts:add("#0080FF#S. save#FFFFFF#:  ", self:colorStats("combatSpellResist"), true)
+	ts:add("#FFD700#M. power#FFFFFF#: ", self:colorStats("combatMindpower"), "  ")
+	ts:add("#0080FF#M. save#FFFFFF#:  ", self:colorStats("combatMentalResist"), true)
 	if self.summon_time then
 		ts:add("Time left: ", {"color", "ANTIQUE_WHITE"}, ("%d"):format(self.summon_time), {"color", "WHITE"}, true)
 	end
-	ts:add(self.desc, true)
+	if self.desc then ts:add(self.desc, true) end
 	if self.faction and Faction.factions[self.faction] then ts:add("Faction: ") ts:merge(factcolor:toTString()) ts:add(("%s (%s, %d)"):format(Faction.factions[self.faction].name, factstate, factlevel), {"color", "WHITE"}, true) end
-	ts:add("Personal reaction: ") ts:merge(pfactcolor:toTString()) ts:add(("%s, %d"):format(pfactstate, pfactlevel), {"color", "WHITE"}, true)
+	if game.player ~= self then ts:add("Personal reaction: ") ts:merge(pfactcolor:toTString()) ts:add(("%s, %d"):format(pfactstate, pfactlevel), {"color", "WHITE"}, true) end
 
 	for tid, act in pairs(self.sustain_talents) do
 		if act then ts:add("- ", {"color", "LIGHT_GREEN"}, self:getTalentFromId(tid).name, {"color", "WHITE"}, true) end
@@ -2939,7 +2971,8 @@ function _M:on_set_temporary_effect(eff_id, e, p)
 		p.maximum = p.dur
 		p.minimum = p.min_dur or 0 --Default minimum duration is 0. Can specify something else by putting min_dur=foo in p when calling setEffect()
 		save = self[p.apply_save or save_for_effects[e.type]](self)
-		local duration = p.maximum - math.max(0, math.floor((save - p.apply_power) / 10))
+		--local duration = p.maximum - math.max(0, math.floor((save - p.apply_power) / 5))
+		local duration = p.maximum - math.max(0, (math.floor(save/5) - math.floor(p.apply_power/5)))
 		p.dur = util.bound(duration, p.minimum or 0, p.maximum)
 		p.amount_decreased = p.maximum - p.dur
 		local save_type = nil
@@ -2951,6 +2984,7 @@ function _M:on_set_temporary_effect(eff_id, e, p)
 		elseif save_type == "combatSpellResist"
 			then p.save_string = "Spell save"
 		end
+		if not p.no_ct_effect and e.status == "detrimental" then self:crossTierEffect(eff_id, p.apply_power, p.apply_save or save_for_effects[e.type]) end
 		p.total_dur = p.dur
 		p.apply_power = nil
 	end
diff --git a/game/modules/tome/class/Object.lua b/game/modules/tome/class/Object.lua
index d15e12389c..4206e7763c 100644
--- a/game/modules/tome/class/Object.lua
+++ b/game/modules/tome/class/Object.lua
@@ -728,6 +728,9 @@ function _M:getTextualDesc(compare_with)
 
 		compare_fields(w, compare_with, field, "combat_spellpower", "%+d", "Spellpower: ")
 		compare_fields(w, compare_with, field, "combat_spellcrit", "%+d%%", "Spell crit. chance: ")
+		
+		compare_fields(w, compare_with, field, "combat_mindpower", "%+d", "Mindpower: ")
+		compare_fields(w, compare_with, field, "combat_mindcrit", "%+d%%", "Mental crit. chance: ")
 
 		compare_fields(w, compare_with, field, "lite", "%+d", "Light radius: ")
 		compare_fields(w, compare_with, field, "infravision", "%+d", "Infravision radius: ")
diff --git a/game/modules/tome/class/PlayerDisplay.lua b/game/modules/tome/class/PlayerDisplay.lua
index 04555214ee..0f76a3215f 100644
--- a/game/modules/tome/class/PlayerDisplay.lua
+++ b/game/modules/tome/class/PlayerDisplay.lua
@@ -195,7 +195,7 @@ function _M:display()
 		self:mouseTooltip(("#GOLD##{bold}#%s\n#WHITE##{normal}#Unused stats: %d\nUnused class talents: %d\nUnused generic talents: %d\nUnused categories: %d"):format(player.name, player.unused_stats, player.unused_talents, player.unused_generics, player.unused_talents_types), self.w, self.font_h, 0, h, function()
 			player:playerLevelup()
 		end)
---		h = h + self.font_h
+		h = h + self.font_h
 	end
 
 	self.font:setStyle("bold")
@@ -205,6 +205,21 @@ function _M:display()
 	self:mouseTooltip(self.TOOLTIP_LEVEL, self:makeTexture(("Level / Exp: #00ff00#%s / %2d%%"):format(player.level, 100 * cur_exp / max_exp), x, h, 255, 255, 255)) h = h + self.font_h
 	self:mouseTooltip(self.TOOLTIP_GOLD, self:makeTexture(("Gold: #00ff00#%0.2f"):format(player.money or 0), x, h, 255, 255, 255)) h = h + self.font_h
 
+	--Display attack, defense, spellpower, mindpower, and saves.
+	local attack_stats = {{"combatAttack", "TOOLTIP_COMBAT_ATTACK", "Accuracy:"}, {"combatPhysicalpower", "TOOLTIP_COMBAT_PHYSICAL_POWER", "P. power:"}, {"combatSpellpower", "TOOLTIP_SPELL_POWER", "S. power:"}, {"combatMindpower", "TOOLTIP_MINDPOWER", "M. power:"}, {"combatDefense", "TOOLTIP_DEFENSE", "Defense:"}, {"combatPhysicalResist", "TOOLTIP_PHYS_SAVE", "P. save:"}, {"combatSpellResist", "TOOLTIP_SPELL_SAVE", "S. save:"}, {"combatMentalResist", "TOOLTIP_MENTAL_SAVE", "M. save:"}}
+
+	local attack_stat_color = "#FFD700#"
+	local defense_stat_color = "#0080FF#"
+	for i = 1, 4 do
+		text = ("%s"):format(player:colorStats(attack_stats[i][1]))
+		self:mouseTooltip(self[attack_stats[i][2]], self:makeTexture((attack_stat_color.."%s"):format(attack_stats[i][3]), x, h, 255, 255, 255))
+		self:mouseTooltip(self[attack_stats[i][2]], self:makeTexture(("%s"):format(text), x+75, h, 255, 255, 255))
+		text = ("%s"):format(player:colorStats(attack_stats[i+4][1]))
+		self:mouseTooltip(self[attack_stats[i+4][2]], self:makeTexture((defense_stat_color.."%s"):format(attack_stats[i+4][3]), x+110, h, 255, 255, 255))
+		self:mouseTooltip(self[attack_stats[i+4][2]], self:makeTexture(("%s"):format(text), x+180, h, 255, 255, 255)) h = h + self.font_h
+	end
+	h = h + self.font_h
+
 	if game.level and game.level.turn_counter then
 		self:makeTexture(("Turns remaining: %d"):format(game.level.turn_counter / 10), x, h, 255, 0, 0) h = h + self.font_h
 		h = h + self.font_h
diff --git a/game/modules/tome/class/interface/Archery.lua b/game/modules/tome/class/interface/Archery.lua
index f638762896..a6fe06a62f 100644
--- a/game/modules/tome/class/interface/Archery.lua
+++ b/game/modules/tome/class/interface/Archery.lua
@@ -147,7 +147,7 @@ local function archery_projectile(tx, ty, tg, self)
 
 		local crit
 		if tg.archery.crit_chance then self.combat_physcrit = self.combat_physcrit + tg.archery.crit_chance end
-		dam, crit = self:physicalCrit(dam, ammo, target)
+		dam, crit = self:physicalCrit(dam, ammo, target, atk, def)
 		if tg.archery.crit_chance then self.combat_physcrit = self.combat_physcrit - tg.archery.crit_chance end
 		print("[ATTACK ARCHERY] after crit", dam)
 
diff --git a/game/modules/tome/class/interface/Combat.lua b/game/modules/tome/class/interface/Combat.lua
index c7a70aeb29..44625c5a41 100644
--- a/game/modules/tome/class/interface/Combat.lua
+++ b/game/modules/tome/class/interface/Combat.lua
@@ -174,7 +174,8 @@ end
 
 --- Computes a logarithmic chance to hit, opposing chance to hit to chance to miss
 -- This will be used for melee attacks, physical and spell resistance
-function _M:checkHit(atk, def, min, max, factor)
+
+function _M:checkHitOld(atk, def, min, max, factor)
 	print("checkHit", atk, def)
 	if atk == 0 then atk = 1 end
 	local hit = nil
@@ -190,6 +191,57 @@ function _M:checkHit(atk, def, min, max, factor)
 	return rng.percent(hit), hit
 end
 
+--Tells the tier difference between two values
+function _M:crossTierEffect(eff_id, apply_power, apply_save, use_given_e)
+	local q = game.player:hasQuest("tutorial-combat-stats")
+	if q and not q:isCompleted("final-lesson")then
+		return
+	end
+	local ct_effect
+	local save_for_effects = {
+		physical = "combatPhysicalResist",
+		magical = "combatSpellResist",
+		mental = "combatMentalResist",
+	}
+	local cross_tier_effects = {
+		combatPhysicalResist = self.EFF_OFFBALANCE,
+		combatSpellResist = self.EFF_SPELLSHOCKED,
+		combatMentalResist = self.EFF_BRAINLOCKED,
+	}
+	local e = self.tempeffect_def[eff_id]
+	if not apply_power or not save_for_effects[e.type] then return end
+	local save = self[apply_save or save_for_effects[e.type]](self)
+
+	if use_given_e then
+		ct_effect = self["EFF_"..e.name]
+	else
+		ct_effect = cross_tier_effects[save_for_effects[e.type]]
+	end
+	local dur = self:getTierDiff(apply_power, save)
+	self:setEffect(ct_effect, dur, {})
+end
+
+function _M:getTierDiff(atk, def)
+	atk = math.floor(atk)
+	def = math.floor(def)
+	return math.max(0, math.max(math.ceil(atk/20), 1) - math.max(math.ceil(def/20), 1))
+end
+
+--New, simpler checkHit that relies on rescaleCombatStats() being used elsewhere
+function _M:checkHit(atk, def, min, max, factor, p)
+	local min = min or 0
+	local max = max or 100
+	if game.player:hasQuest("tutorial-combat-stats") then
+		min = 0
+		max = 100
+	end --ensures predictable combat for the tutorial
+	print("checkHit", atk, def)
+	local hit = 50 + 5 * (atk - def)
+	hit = util.bound(hit, min, max)
+	print("=> chance to hit", hit)
+	return rng.percent(hit), hit
+end
+
 --- Try to totally evade an attack
 function _M:checkEvasion(target)
 	if not target:attr("evasion") then return end
@@ -251,7 +303,7 @@ function _M:attackTargetWith(target, weapon, damtype, mult, force_dam)
 		local damrange = self:combatDamageRange(weapon)
 		dam = rng.range(dam, dam * damrange)
 		print("[ATTACK] after range", dam)
-		dam, crit = self:physicalCrit(dam, weapon, target)
+		dam, crit = self:physicalCrit(dam, weapon, target, atk, def)
 		print("[ATTACK] after crit", dam)
 		dam = dam * mult
 		print("[ATTACK] after mult", dam)
@@ -375,12 +427,15 @@ function _M:attackTargetWith(target, weapon, damtype, mult, force_dam)
 
 		if target:checkHit(self:combatAttack(weapon), target:combatPhysicalResist(), 0, 95, 10) and target:canBe("knockback") then
 			target:knockback(self.x, self.y, self:attr("onslaught"))
+			target:crossTierEffect(target.EFF_OFFBALANCE, src:combatAttack())
 		end
 		if lt and lt:checkHit(self:combatAttack(weapon), lt:combatPhysicalResist(), 0, 95, 10) and lt:canBe("knockback") then
 			lt:knockback(self.x, self.y, self:attr("onslaught"))
+			target:crossTierEffect(target.EFF_OFFBALANCE, src:combatAttack())
 		end
 		if rt and rt:checkHit(self:combatAttack(weapon), rt:combatPhysicalResist(), 0, 95, 10) and rt:canBe("knockback") then
 			rt:knockback(self.x, self.y, self:attr("onslaught"))
+			target:crossTierEffect(target.EFF_OFFBALANCE, src:combatAttack())
 		end
 	end
 
@@ -530,12 +585,12 @@ end
 
 --- Gets the defense
 --- Fake denotes a check not actually being made, used by character sheets etc.
-function _M:combatDefense(fake)
+function _M:combatDefenseBase(fake)
 	local add = 0
 	if self:hasDualWeapon() and self:knowTalent(self.T_DUAL_WEAPON_DEFENSE) then
 		add = add + 4 + (self:getTalentLevel(self.T_DUAL_WEAPON_DEFENSE) * self:getDex()) / 12
 	end
-	if not fake then 
+	if not fake then
 		add = add + (self:checkOnDefenseCall("defense") or 0)
 	end
 	if self:knowTalent(self.T_TACTICAL_EXPERT) then
@@ -560,11 +615,15 @@ function _M:combatDefense(fake)
 end
 
 --- Gets the defense ranged
---- Fake denotes a check not actually being made, used by character sheets etc.
-function _M:combatDefenseRanged(fake)
-	local base_defense = self:combatDefense(true)
-	if not fake then base_defense = self:combatDefense() end
-	return math.max(0, base_defense + (self.combat_def_ranged or 0))
+function _M:combatDefense()
+	local d = self:combatDefenseBase()
+	return self:rescaleCombatStats(d)
+end
+
+--- Gets the defense ranged
+function _M:combatDefenseRanged()
+	local d = math.max(0, self:combatDefenseBase() + (self.combat_def_ranged or 0))
+	return self:rescaleCombatStats(d)
 end
 
 --- Gets the armor
@@ -603,10 +662,10 @@ function _M:combatAttackBase(weapon, ammo)
 end
 function _M:combatAttack(weapon, ammo)
 	local stats
-	if self.use_psi_combat then stats = (self:getWil(50, true) - 5) + (self:getCun(50, true) - 5)
-	else stats = (self:getStr(50, true) - 5) + (self:getDex(50, true) - 5)
+	if self.use_psi_combat then stats = self:getCun(100, true) - 10
+	else stats = self:getDex(100, true) - 10
 	end
-	return self:combatAttackBase(weapon, ammo) + stats
+	return self:rescaleCombatStats(self:combatAttackBase(weapon, ammo) + stats)
 end
 
 --- Gets the attack using only strength
@@ -659,10 +718,23 @@ end
 
 
 --- Scale damage values
--- This makes low damage values equal to what they should be and puts disminishing returns to super high values
+-- This currently beefs up high-end damage values to make up for the combat stat rescale nerf.
 function _M:rescaleDamage(dam)
 	if dam <= 0 then return dam end
-	return dam * (1 - math.log10(dam * 2) / 7)
+--	return dam * (1 - math.log10(dam * 2) / 7) --this is the old version, pre-combat-stat-rescale
+	return dam ^ 1.04
+end
+--Diminishing-returns method of scaling combat stats, observing this rule: the first twenty ranks cost 1 point each, the second twenty cost two each, and so on. This is much, much better for players than some logarithmic mess, since they always know exactly what's going on, and there are nice breakpoints to strive for.
+function _M:rescaleCombatStats(raw_combat_stat_value)
+	local x = raw_combat_stat_value
+	local tiers = 5 -- Just increase this if you want to add high-level content that allows for combat stat scores over 100.
+	--return math.floor(math.min(x, 20) + math.min(math.max((x-20), 0)/2, 20) + math.min(math.max((x-60), 0)/3, 20) + math.min(math.max((x-120), 0)/4, 20) + math.min(math.max((x-200), 0)/5, 20)) --Five terms of the summation below.
+	local total = 0
+	for i = 1, tiers do
+		local sub = 20*(i*(i-1)/2)
+		total = total + math.min(math.max(x-sub, 0)/i, 20)
+	end
+	return total
 end
 
 --- Gets the damage
@@ -689,6 +761,16 @@ function _M:combatDamage(weapon)
 		end
 	end
 
+	local talented_mod = math.sqrt(self:combatCheckTraining(weapon) / 10) / 2 + 1
+
+	local power = math.max((weapon.dam or 1), 1)
+	power = (math.sqrt(power / 10) - 1) * 0.5 + 1
+	--print(("[COMBAT DAMAGE] power(%f) totstat(%f) talent_mod(%f)"):format(power, totstat, talented_mod))
+	return self:rescaleDamage(0.3*(self:combatPhysicalpower() + totstat) * power * talented_mod)
+end
+
+function _M:combatPhysicalpower(mod)
+	mod = mod or 1
 	local add = 0
 	if self:knowTalent(Talents.T_ARCANE_DESTRUCTION) then
 		add = add + self:combatSpellpower() * self:getTalentLevel(Talents.T_ARCANE_DESTRUCTION) / 9
@@ -696,17 +778,33 @@ function _M:combatDamage(weapon)
 	if self:isTalentActive(Talents.T_BLOOD_FRENZY) then
 		add = add + self.blood_frenzy
 	end
-	if self:knowTalent(self.T_EMPTY_HAND) and weapon == self.combat then
+	if self:knowTalent(self.T_EMPTY_HAND) and self:isUnarmed() then
 		local t = self:getTalentFromId(self.T_EMPTY_HAND)
 		add = add + t.getDamage(self, t)
 	end
+	if self:knowTalent(Talents.T_WEAPONS_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_WEAPONS_MASTERY)
+	end
+	if self:knowTalent(Talents.T_KNIFE_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_WEAPONS_MASTERY)
+	end
+	if self:knowTalent(Talents.T_EXOTIC_WEAPONS_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_WEAPONS_MASTERY)
+	end
+	if self:knowTalent(Talents.T_UNARMED_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_UNARMED_MASTERY)
+	end
+	if self:knowTalent(Talents.T_STAFF_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_STAFF_MASTERY)
+	end
+	if self:knowTalent(Talents.T_BOW_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_BOW_MASTERY)
+	end
+	if self:knowTalent(Talents.T_SLING_MASTERY) then
+		add = add + 5 * self:getTalentLevel(Talents.T_SLING_MASTERY)
+	end
 
-	local talented_mod = math.sqrt(self:combatCheckTraining(weapon) / 10) + 1
-
-	local power = math.max(self.combat_dam + (weapon.dam or 1) + add, 1)
-	power = (math.sqrt(power / 10) - 1) * 0.8 + 1
-	print(("[COMBAT DAMAGE] power(%f) totstat(%f) talent_mod(%f)"):format(power, totstat, talented_mod))
-	return self:rescaleDamage(totstat / 1.5 * power * talented_mod)
+	return self:rescaleCombatStats((self.combat_dam > 0 and self.combat_dam or 0) + add + self:getStr()) * mod
 end
 
 --- Gets spellpower
@@ -723,7 +821,7 @@ function _M:combatSpellpower(mod)
 		add = add + self:hasEffect(self.EFF_BLOODLUST).dur
 	end
 
-	return ((self.combat_spellpower > 0 and self.combat_spellpower or 0) + add + self:getMag()) * mod
+	return self:rescaleCombatStats((self.combat_spellpower > 0 and self.combat_spellpower or 0) + add + self:getMag()) * mod
 end
 
 --- Gets damage based on talent
@@ -777,12 +875,14 @@ function _M:combatSummonSpeed()
 end
 
 --- Computes physical crit for a damage
-function _M:physicalCrit(dam, weapon, target)
+function _M:physicalCrit(dam, weapon, target, atk, def)
+	local tier_diff = self:getTierDiff(atk, def)
 	if self:isTalentActive(self.T_STEALTH) and self:knowTalent(self.T_SHADOWSTRIKE) then
 		return dam * (1.5 + self:getTalentLevel(self.T_SHADOWSTRIKE) / 7), true
 	end
 
 	local chance = self:combatCrit(weapon)
+	local crit_power_add = 0
 	local crit = false
 	if self:knowTalent(self.T_BACKSTAB) and target:attr("stunned") then chance = chance + self:getTalentLevel(self.T_BACKSTAB) * 10 end
 
@@ -796,6 +896,7 @@ function _M:physicalCrit(dam, weapon, target)
 		end
 	end
 
+
 	if target:hasHeavyArmor() and target:knowTalent(target.T_ARMOUR_TRAINING) then
 		chance = chance - target:getTalentLevel(target.T_ARMOUR_TRAINING) * 1.9
 	end
@@ -804,7 +905,13 @@ function _M:physicalCrit(dam, weapon, target)
 
 	print("[PHYS CRIT %]", chance)
 	if rng.percent(chance) then
-		dam = dam * (1.5 + (self.combat_critical_power or 0) / 100)
+		if tier_diff > 0 then
+			target:crossTierEffect(target.EFF_OFFBALANCE, atk, "combatDefense")
+		end
+		if target:hasEffect(target.EFF_OFFBALANCE) then
+			crit_power_add = 0.25
+		end
+		dam = dam * (1.5 + crit_power_add + (self.combat_critical_power or 0) / 100)
 		crit = true
 
 	end
@@ -854,7 +961,7 @@ end
 function _M:combatMindpower(mod)
 	mod = mod or 1
 	local add = 0
-	return ((self.combat_mindpower > 0 and self.combat_mindpower or 0) + add + self:getWil() * 0.7 + self:getCun() * 0.4) * mod
+	return self:rescaleCombatStats((self.combat_mindpower > 0 and self.combat_mindpower or 0) + add + self:getWil() * 0.7 + self:getCun() * 0.4) * mod
 end
 
 --- Gets damage based on talent
@@ -870,47 +977,59 @@ function _M:combatTalentStatDamage(t, stat, base, max)
 	-- Compute at "max"
 	local mod = max / ((base + 100) * ((math.sqrt(5) - 1) * 0.8 + 1))
 	-- Compute real
-	return self:rescaleDamage((base + (self:getStat(stat))) * ((math.sqrt(self:getTalentLevel(t)) - 1) * 0.8 + 1) * mod)
+	local dam = (base + (self:getStat(stat))) * ((math.sqrt(self:getTalentLevel(t)) - 1) * 0.8 + 1) * mod
+	dam =  dam * (1 - math.log10(dam * 2) / 7)
+	dam = dam ^ (1 / 1.04)
+	return self:rescaleDamage(dam)
 end
 
---- Gets damage based on talent, stat, and interval
+--- Gets damage based on talent, basic stat, and interval
 function _M:combatTalentIntervalDamage(t, stat, min, max, stat_weight)
 	local stat_weight = stat_weight or 0.5
-	--return self:rescaleDamage(min + (1 + (self:getStat(stat) / 100) * (max / 6.5 - 1)) * self:getTalentLevel(t))
-	return self:rescaleDamage(min + (max - min)*((stat_weight * self:getStat(stat)/100) + (1 - stat_weight) * self:getTalentLevel(t)/6.5))
+	local dam = min + (max - min)*((stat_weight * self:getStat(stat)/100) + (1 - stat_weight) * self:getTalentLevel(t)/6.5)
+	dam =  dam * (1 - math.log10(dam * 2) / 7)
+	dam = dam ^ (1 / 1.04)
+	return self:rescaleDamage(dam)
+end
+
+--- Gets damage based on talent, stat, and interval
+function _M:combatStatTalentIntervalDamage(t, stat, min, max, stat_weight)
+	local stat_weight = stat_weight or 0.5
+	scaled_stat = self[stat](self)
+	return self:rescaleDamage(min + (max - min)*((stat_weight * self[stat](self)/100) + (1 - stat_weight) * self:getTalentLevel(t)/6.5))
 end
 
 --- Computes physical resistance
 --- Fake denotes a check not actually being made, used by character sheets etc.
 function _M:combatPhysicalResist(fake)
 	local add = 0
-	if not fake then 
+	if not fake then
 		add = add + (self:checkOnDefenseCall("physical") or 0)
 	end
 	if self:knowTalent(self.T_POWER_IS_MONEY) then
 		add = add + util.bound(self.money / (80 - self:getTalentLevelRaw(self.T_POWER_IS_MONEY) * 5), 0, self:getTalentLevelRaw(self.T_POWER_IS_MONEY) * 10)
 	end
-	return self.combat_physresist + (self:getCon() + self:getStr() + (self:getLck() - 50) * 0.5) * 0.35 + add
+	return self:rescaleCombatStats(self.combat_physresist + (self:getCon() + self:getStr() + (self:getLck() - 50) * 0.5) * 0.35 + add)
 end
 
 --- Computes spell resistance
 --- Fake denotes a check not actually being made, used by character sheets etc.
 function _M:combatSpellResist(fake)
 	local add = 0
-	if not fake then 
+	if not fake then
 		add = add + (self:checkOnDefenseCall("spell") or 0)
 	end
 	if self:knowTalent(self.T_POWER_IS_MONEY) then
 		add = add + util.bound(self.money / (80 - self:getTalentLevelRaw(self.T_POWER_IS_MONEY) * 5), 0, self:getTalentLevelRaw(self.T_POWER_IS_MONEY) * 10)
 	end
-	return self.combat_spellresist + (self:getMag() + self:getWil() + (self:getLck() - 50) * 0.5) * 0.35 + add
+	return self:rescaleCombatStats(self.combat_spellresist + (self:getMag() + self:getWil() + (self:getLck() - 50) * 0.5) * 0.35 + add)
 end
 
 --- Computes mental resistance
 --- Fake denotes a check not actually being made, used by character sheets etc.
 function _M:combatMentalResist(fake)
 	local add = 0
-	if not fake then 
+	if not fake then
 		add = add + (self:checkOnDefenseCall("mental") or 0)
 	end
 	if self:knowTalent(self.T_STEADY_MIND) then
@@ -920,7 +1039,7 @@ function _M:combatMentalResist(fake)
 	if self:knowTalent(self.T_POWER_IS_MONEY) then
 		add = add + util.bound(self.money / (80 - self:getTalentLevelRaw(self.T_POWER_IS_MONEY) * 5), 0, self:getTalentLevelRaw(self.T_POWER_IS_MONEY) * 10)
 	end
-	return self.combat_mentalresist + (self:getCun() + self:getWil() + (self:getLck() - 50) * 0.5) * 0.35 + add
+	return self:rescaleCombatStats(self.combat_mentalresist + (self:getCun() + self:getWil() + (self:getLck() - 50) * 0.5) * 0.35 + add)
 end
 
 -- Called when a Save or Defense is checked
@@ -1191,7 +1310,8 @@ function _M:startGrapple(target)
 		self:setEffect(self.EFF_GRAPPLING, duration, {trgt=target}, true)
 		return true
 	elseif target:canBe("pin") then
-		target:setEffect(target.EFF_GRAPPLED, duration, {src=self, power=power, apply_power=self:combatAttackStr()})
+		target:setEffect(target.EFF_GRAPPLED, duration, {src=self, power=power, apply_power=self:combatPhysicalpower()})
+		target:crossTierEffect(target.EFF_GRAPPLED, self:combatPhysicalpower())
 		self:setEffect(self.EFF_GRAPPLING, duration, {trgt=target})
 		return true
 	else
diff --git a/game/modules/tome/class/interface/TooltipsData.lua b/game/modules/tome/class/interface/TooltipsData.lua
index a510c278fe..657b5f1adf 100644
--- a/game/modules/tome/class/interface/TooltipsData.lua
+++ b/game/modules/tome/class/interface/TooltipsData.lua
@@ -164,22 +164,22 @@ I.E: at 100% you will be able to do 100% more attacks (aka twice as many) in the
 -- Stats
 -------------------------------------------------------------
 TOOLTIP_STR = [[#GOLD#Strength#LAST#
-Strength defines your character's ability to apply physical force. It increases your melee damage, damage done with heavy weapons, your chance to hit, your chance to save against physical effects, and carrying capacity.
+Strength defines your character's ability to apply physical force. It increases Physical Power, damage done with heavy weapons, Physical Save, and carrying capacity.
 ]]
 TOOLTIP_DEX = [[#GOLD#Dexterity#LAST#
-Dexterity defines your character's ability to be agile and alert. It increases your chance to hit, your ability to avoid attacks, and your damage with light weapons.
+Dexterity defines your character's ability to be agile and alert. It increases Accuracy, Defense, and your damage with light weapons.
 ]]
 TOOLTIP_CON = [[#GOLD#Constitution#LAST#
-Constitution defines your character's ability to withstand and resist damage. It increases your maximum life and your chance to save against physical effects.
+Constitution defines your character's ability to withstand and resist damage. It increases your maximum life and Physical Save.
 ]]
 TOOLTIP_MAG = [[#GOLD#Magic#LAST#
-Magic defines your character's ability to manipulate the magical energy of the world. It increases your spell power, your chance to save against magical effects, and the effect of spells and other magic items.
+Magic defines your character's ability to manipulate the magical energy of the world. It increases your Spellpower, Spell Save, and the effect of spells and other magic items.
 ]]
 TOOLTIP_WIL = [[#GOLD#Willpower#LAST#
-Willpower defines your character's ability to concentrate. It increases your mana, stamina, psi capacity, and your chance to save against magical and mental effects.
+Willpower defines your character's ability to concentrate. It increases your mana, stamina, psi capacity, Mindpower, Spell Save, and Mental Save.
 ]]
 TOOLTIP_CUN = [[#GOLD#Cunning#LAST#
-Cunning defines your character's ability to learn, think, and react. It allows you to learn many worldly abilities, and increases your chance to save against mental effects and critical chance.
+Cunning defines your character's ability to learn, think, and react. It allows you to learn many worldly abilities, and increases your Mindpower, Mental Save, and critical chance.
 ]]
 TOOLTIP_STRDEXCON = "#AQUAMARINE#Physical stats#LAST#\n---\n"..TOOLTIP_STR.."\n---\n"..TOOLTIP_DEX.."\n---\n"..TOOLTIP_CON
 TOOLTIP_MAGWILCUN = "#AQUAMARINE#Mental stats#LAST#\n---\n"..TOOLTIP_MAG.."\n---\n"..TOOLTIP_WIL.."\n---\n"..TOOLTIP_CUN
@@ -188,9 +188,12 @@ TOOLTIP_MAGWILCUN = "#AQUAMARINE#Mental stats#LAST#\n---\n"..TOOLTIP_MAG.."\n---
 -- Melee
 -------------------------------------------------------------
 TOOLTIP_COMBAT_ATTACK = [[#GOLD#Accuracy#LAST#
-Your accuracy value represents your chance to hit your opponents, it is measured directly against the target's defense rating.
-When you inflict temporary physical effects on an enemy, every ten points of accuracy counteracts a single turn of duration reduction granted by the enemy's saving throws.
-It is improved by both Strength and Dexterity.
+Determines your chance to hit your target when measured against the target's Defense.
+When you use Accuracy to inflict temporary physical effects on an enemy, every five points of Accuracy counteracts a single turn of duration reduction granted by the enemy's saving throws.
+]]
+TOOLTIP_COMBAT_PHYSICAL_POWER = [[#GOLD#Physical Power#LAST#
+Measures your ability to deal physical damage in combat.
+When you use Physical Power to inflict temporary physical effects on an enemy, every five points of Physical Power counteracts a single turn of duration reduction granted by the enemy's saving throws.
 ]]
 TOOLTIP_COMBAT_DAMAGE = [[#GOLD#Damage#LAST#
 This is the damage you inflict on your foes when you hit them.
@@ -237,19 +240,19 @@ Armour hardiness value represents how much of every incoming blows the armour wi
 Absorbs (hardiness)% of incoming physical damage, up to a maximum of (armour) damage absorbed.
 ]]
 TOOLTIP_DEFENSE = [[#GOLD#Defense#LAST#
-Defense represents your chance to avoid being hit at all by a melee attack, it is measured against the attacker's accuracy chance.
+Defense represents your chance to avoid physical melee attacks. It is measured against the attacker's Accuracy.
 ]]
 TOOLTIP_RDEFENSE = [[#GOLD#Ranged Defense#LAST#
-Ranged defense represents your chance to avoid being hit at all by a ranged attack, it is measured against the attacker's accuracy chance.
+Defense represents your chance to avoid physical ranged attacks. It is measured against the attacker's Accuracy.
 ]]
 TOOLTIP_PHYS_SAVE = [[#GOLD#Physical saving throw#LAST#
-Increases chance to shrug off physically-induced effects. Also reduces duration of detrimental physical effects by up to one turn for every ten points, depending on the opponent.
+Increases chance to shrug off physically-induced effects. Also reduces duration of detrimental physical effects by up to one turn for every five points, depending on the opponent.
 ]]
 TOOLTIP_SPELL_SAVE = [[#GOLD#Spell saving throw#LAST#
-Increases chance to shrug off magically-induced effects. Also reduces duration of detrimental magical effects by up to one turn for every ten points, depending on the opponent.
+Increases chance to shrug off magically-induced effects. Also reduces duration of detrimental magical effects by up to one turn for every five points, depending on the opponent.
 ]]
 TOOLTIP_MENTAL_SAVE = [[#GOLD#Mental saving throw#LAST#
-Increases chance to shrug off mentally-induced effects. Also reduces duration of detrimental mental effects by up to one turn for every ten points, depending on the opponent.
+Increases chance to shrug off mentally-induced effects. Also reduces duration of detrimental mental effects by up to one turn for every five points, depending on the opponent.
 ]]
 
 -------------------------------------------------------------
@@ -257,8 +260,7 @@ Increases chance to shrug off mentally-induced effects. Also reduces duration of
 -------------------------------------------------------------
 TOOLTIP_SPELL_POWER = [[#GOLD#Spellpower#LAST#
 Your spellpower value represents how powerful your spells are.
-In addition, when your spells inflict temporary detrimental effects, every ten points of spellpower counteracts a single turn of duration reduction granted by the enemy's saving throws.
-It is improved by Magic, equipment, and certain talents.
+In addition, when your spells inflict temporary detrimental effects, every five points of spellpower counteracts a single turn of duration reduction granted by the enemy's saving throws.
 ]]
 TOOLTIP_SPELL_CRIT = [[#GOLD#Spell critical chance#LAST#
 Each time you deal damage with a spell you have a chance to make a critical hit that deals 150% of the normal damage.
@@ -271,8 +273,7 @@ The lower it is the faster it is.
 ]]
 TOOLTIP_MINDPOWER = [[#GOLD#Mindpower#LAST#
 Your mindpower value represents how powerful your mental abilities are.
-In addition, when you use mental abilities inflict temporary detrimental effects, every ten points of mindpower counteracts a single turn of duration reduction granted by the enemy's saving throws.
-It is improved by both Willpower and Cunning, equipment, and certain talents.
+In addition, when you use mental abilities inflict temporary detrimental effects, every five points of mindpower counteracts a single turn of duration reduction granted by the enemy's saving throws.
 ]]
 
 -------------------------------------------------------------
diff --git a/game/modules/tome/data/birth/races/tutorial.lua b/game/modules/tome/data/birth/races/tutorial.lua
index ca5cd3f32f..7b26b53faf 100644
--- a/game/modules/tome/data/birth/races/tutorial.lua
+++ b/game/modules/tome/data/birth/races/tutorial.lua
@@ -51,9 +51,9 @@ newBirthDescriptor
 	},
 	copy = {
 		default_wilderness = {1, 1, "wilderness"},
-		starting_zone = "tutorial",
-		starting_quest = "tutorial",
-		starting_intro = "tutorial",
+		starting_zone = "tutorial-combat-stats",
+		starting_quest = "tutorial-combat-stats",
+		starting_intro = "tutorial-combat-stats",
 		moddable_tile = "human_#sex#",
 		moddable_tile_base = "base_cornac_01.png",
 	},
diff --git a/game/modules/tome/data/damage_types.lua b/game/modules/tome/data/damage_types.lua
index 695aa94d92..686c4027d5 100644
--- a/game/modules/tome/data/damage_types.lua
+++ b/game/modules/tome/data/damage_types.lua
@@ -119,7 +119,7 @@ setDefaultProjector(function(src, x, y, type, dam, tmp, no_martyr)
 		-- Damage Smearing
 		if type ~= DamageType.TEMPORAL and target:hasEffect(target.EFF_DAMAGE_SMEARING) then
 			local smear = dam
-			target:setEffect(target.EFF_SMEARED, 6, {src=src, power=smear/6})
+			target:setEffect(target.EFF_SMEARED, 6, {src=src, power=smear/6, no_ct_effect=true})
 			dam = 0
 		end
 
@@ -196,7 +196,7 @@ setDefaultProjector(function(src, x, y, type, dam, tmp, no_martyr)
 
 		if not target.dead and dam > 0 and target:attr("elemental_harmony") and not target:hasEffect(target.EFF_ELEMENTAL_HARMONY) then
 			if type == DamageType.FIRE or type == DamageType.COLD or type == DamageType.LIGHTNING or type == DamageType.ACID or type == DamageType.NATURE then
-				target:setEffect(target.EFF_ELEMENTAL_HARMONY, 5 + math.ceil(target:attr("elemental_harmony")), {power=target:attr("elemental_harmony"), type=type})
+				target:setEffect(target.EFF_ELEMENTAL_HARMONY, 5 + math.ceil(target:attr("elemental_harmony")), {power=target:attr("elemental_harmony"), type=type, no_ct_effect=true})
 			end
 		end
 
@@ -435,7 +435,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and rng.percent(dam) then
 			if target:canBe("silence") then
-				target:setEffect(target.EFF_SILENCED, 4, {apply_power=src:combatAttackDex()*0.7})
+				target:setEffect(target.EFF_SILENCED, 4, {apply_power=src:combatAttack()*0.7, no_ct_effect=true})
 			else
 				game.logSeen(target, "%s resists!", target.name:capitalize())
 			end
@@ -476,7 +476,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target:canBe("blind") then
-				target:setEffect(target.EFF_BLINDED, math.ceil(dam), {apply_power=src:combatAttackStr(), apply_save="combatPhysicalResist"})
+				target:setEffect(target.EFF_BLINDED, math.ceil(dam), {apply_power=src:combatPhysicalpower(), apply_save="combatPhysicalResist"})
 			else
 				game.logSeen(target, "%s avoids the blinding ink!", target.name:capitalize())
 			end
@@ -489,7 +489,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target:canBe("blind") then
-				target:setEffect(target.EFF_BLINDED, math.ceil(dam.turns), {apply_power=dam.power, apply_save="combatMentalResist"})
+				target:setEffect(target.EFF_BLINDED, math.ceil(dam.turns), {apply_power=dam.power, apply_save="combatMentalResist", no_ct_effect=true})
 			else
 				game.logSeen(target, "%s resists the blinding light!", target.name:capitalize())
 			end
@@ -510,7 +510,7 @@ newDamageType{
 		if target then
 			-- Set on fire!
 			dam = dam - init_dam
-			target:setEffect(target.EFF_BURNING, dur, {src=src, power=dam / dur})
+			target:setEffect(target.EFF_BURNING, dur, {src=src, power=dam / dur, no_ct_effect=true})
 		end
 		return init_dam
 	end,
@@ -643,7 +643,6 @@ newDamageType{
 			local sx, sy = game.level.map:getTileToScreen(x, y)
 			if target:canBe("stun") then
 				target:setEffect(target.EFF_FROZEN, dam.dur, {hp=dam.hp * 1.5, apply_power=src:combatSpellpower(), min_dur=1})
-
 				game.flyers:add(sx, sy, 30, (rng.range(0,2)-1) * 0.5, -3, "Frozen!", {0,255,155})
 			else
 				game.flyers:add(sx, sy, 30, (rng.range(0,2)-1) * 0.5, -3, "Resist!", {0,255,155})
@@ -660,7 +659,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target:canBe("blind") then
-				target:setEffect(target.EFF_DIM_VISION, 7, {sight=dam, apply_power=src:combatAttackDex()})
+				target:setEffect(target.EFF_DIM_VISION, 7, {sight=dam, apply_power=src:combatAttack()})
 			else
 				game.logSeen(target, "%s resists!", target.name:capitalize())
 			end
@@ -734,6 +733,7 @@ newDamageType{
 		if target then
 			if target:checkHit(src:combatSpellpower(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, 1)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatSpellpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the wave!", target.name:capitalize())
@@ -754,6 +754,7 @@ newDamageType{
 			DamageType:get(DamageType.FIREBURN).projector(src, x, y, DamageType.FIREBURN, dam.dam)
 			if target:checkHit(src:combatSpellpower(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, dam.dist)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatSpellpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the punch!", target.name:capitalize())
@@ -774,6 +775,7 @@ newDamageType{
 			DamageType:get(DamageType.DARKNESS).projector(src, x, y, DamageType.DARKNESS, dam.dam)
 			if target:checkHit(src:combatSpellpower(), target:combatMentalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, dam.dist)
+				target:crossTierEffect(target.EFF_BRAINLOCKED, src:combatSpellpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the darkness!", target.name:capitalize())
@@ -794,6 +796,7 @@ newDamageType{
 			realdam = DamageType:get(DamageType.PHYSICAL).projector(src, x, y, DamageType.PHYSICAL, dam)
 			if target:checkHit(src:combatSpellpower(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, 3)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatSpellpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the punch!", target.name:capitalize())
@@ -814,6 +817,7 @@ newDamageType{
 			DamageType:get(DamageType.PHYSICAL).projector(src, x, y, DamageType.PHYSICAL, dam)
 			if target:checkHit(src:combatMindpower() * 0.8, target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, 3)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatMindpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the punch!", target.name:capitalize())
@@ -831,8 +835,9 @@ newDamageType{
 		if target and not tmp[target] then
 			tmp[target] = true
 			DamageType:get(DamageType.PHYSICAL).projector(src, x, y, DamageType.PHYSICAL, dam.dam)
-			if target:checkHit(src:combatAttackStr(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
+			if target:checkHit(src:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, dam.dist)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatPhysicalpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
@@ -885,7 +890,7 @@ newDamageType{
 		DamageType:get(DamageType.NATURE).projector(src, x, y, DamageType.NATURE, dam.dam / dam.dur)
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and target:canBe("poison") then
-			target:setEffect(target.EFF_SPYDRIC_POISON, dam.dur, {src=src, power=dam.dam / dam.dur})
+			target:setEffect(target.EFF_SPYDRIC_POISON, dam.dur, {src=src, power=dam.dam / dam.dur, no_ct_effect=true})
 		end
 	end,
 }
@@ -897,7 +902,7 @@ newDamageType{
 		DamageType:get(DamageType.NATURE).projector(src, x, y, DamageType.NATURE, dam.dam / dam.dur)
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and target:canBe("poison") then
-			target:setEffect(target.EFF_INSIDIOUS_POISON, dam.dur, {src=src, power=dam.dam / dam.dur, heal_factor=dam.heal_factor})
+			target:setEffect(target.EFF_INSIDIOUS_POISON, dam.dur, {src=src, power=dam.dam / dam.dur, heal_factor=dam.heal_factor, no_ct_effect=true})
 		end
 	end,
 }
@@ -911,7 +916,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and target:canBe("cut") then
 			-- Set on fire!
-			target:setEffect(target.EFF_CUT, 5, {src=src, power=dam / 5})
+			target:setEffect(target.EFF_CUT, 5, {src=src, power=dam / 5, no_ct_effect=true})
 		end
 	end,
 }
@@ -923,7 +928,7 @@ newDamageType{
 		local realdam = DamageType:get(DamageType.PHYSICAL).projector(src, x, y, DamageType.PHYSICAL, dam)
 		local target = game.level.map(x, y, Map.ACTOR)
 		if realdam > 0 and target and target:canBe("cut") then
-			target:setEffect(target.EFF_CUT, 5, {src=src, power=dam * 0.1})
+			target:setEffect(target.EFF_CUT, 5, {src=src, power=dam * 0.1, no_ct_effect=true})
 		end
 	end,
 }
@@ -935,7 +940,7 @@ newDamageType{
 		DamageType:get(DamageType.NATURE).projector(src, x, y, DamageType.NATURE, dam)
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
-			target:setEffect(target.EFF_SLOW, 3, {power=0.3})
+			target:setEffect(target.EFF_SLOW, 3, {power=0.3, no_ct_effect=true})
 		end
 	end,
 }
@@ -981,11 +986,11 @@ newDamageType{
 			-- Freeze it, if we pass the test
 			local sx, sy = game.level.map:getTileToScreen(x, y)
 			if src == target then
-				target:setEffect(target.EFF_TIME_PRISON, dam, {})
-				target:setEffect(target.EFF_CONTINUUM_DESTABILIZATION, 100, {power=src:combatSpellpower(0.3)})
+				target:setEffect(target.EFF_TIME_PRISON, dam, {no_ct_effect=true})
+				target:setEffect(target.EFF_CONTINUUM_DESTABILIZATION, 100, {power=src:combatSpellpower(0.3), no_ct_effect=true})
 			elseif target:checkHit(src:combatSpellpower() - (target:attr("continuum_destabilization") or 0), target:combatSpellResist(), 0, 95, 15) then
-				target:setEffect(target.EFF_TIME_PRISON, dam, {apply_power=src:combatSpellpower() - (target:attr("continuum_destabilization") or 0), apply_save="combatSpellResist"})
-				target:setEffect(target.EFF_CONTINUUM_DESTABILIZATION, 100, {power=src:combatSpellpower(0.3)})
+				target:setEffect(target.EFF_TIME_PRISON, dam, {apply_power=src:combatSpellpower() - (target:attr("continuum_destabilization") or 0), apply_save="combatSpellResist", no_ct_effect=true})
+				target:setEffect(target.EFF_CONTINUUM_DESTABILIZATION, 100, {power=src:combatSpellpower(0.3), no_ct_effect=true})
 			else
 				game.logSeen(target, "%s resists the time prison.", target.name:capitalize())
 			end
@@ -1016,7 +1021,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and rng.percent(dam.dam) then
 			if target:canBe("confusion") then
-				target:setEffect(target.EFF_CONFUSED, dam.dam, {power=75, apply_power=(dam.power_check or src.combatSpellpower)(src)})
+				target:setEffect(target.EFF_CONFUSED, dam.dam, {power=75, apply_power=(dam.power_check or src.combatSpellpower)(src), no_ct_effect=true})
 			else
 				game.logSeen(target, "%s resists!", target.name:capitalize())
 			end
@@ -1032,7 +1037,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and rng.percent(dam.dam) then
 			if target:canBe("blind") then
-				target:setEffect(target.EFF_BLINDED, dam.dam, {apply_power=(dam.power_check or src.combatSpellpower)(src)})
+				target:setEffect(target.EFF_BLINDED, dam.dam, {apply_power=(dam.power_check or src.combatSpellpower)(src), no_ct_effect=true})
 			else
 				game.logSeen(target, "%s resists!", target.name:capitalize())
 			end
@@ -1048,7 +1053,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target:canBe("blind") then
-				target:setEffect(target.EFF_BLINDED, dam.dur, {apply_power=src:combatAttackStr(), apply_save="combatPhysicalResist"})
+				target:setEffect(target.EFF_BLINDED, dam.dur, {apply_power=src:combatPhysicalpower(), apply_save="combatPhysicalResist"})
 			else
 				game.logSeen(target, "%s resists the sandstorm!", target.name:capitalize())
 			end
@@ -1064,7 +1069,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target:canBe("pin") then
-				target:setEffect(target.EFF_PINNED, dam.dur, {apply_power=src:combatAttackStr()})
+				target:setEffect(target.EFF_PINNED, dam.dur, {apply_power=src:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists!", target.name:capitalize())
 			end
@@ -1203,7 +1208,7 @@ newDamageType{
 		DamageType:get(DamageType.BLIGHT).projector(src, x, y, DamageType.BLIGHT, dam)
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and not target.undead and not target.construct then
-			target:setEffect(target.EFF_SLOW, 4, {power=0.2})
+			target:setEffect(target.EFF_SLOW, 4, {power=0.2, no_ct_effect=true})
 		end
 	end,
 }
@@ -1243,7 +1248,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target and target:canBe("cut") then
 			-- Set on fire!
-			target:setEffect(target.EFF_CUT, 5, {src=src, power=dam / 11})
+			target:setEffect(target.EFF_CUT, 5, {src=src, power=dam / 11, no_ct_effect=true})
 		end
 	end,
 }
@@ -1308,6 +1313,7 @@ newDamageType{
 			tmp[target] = true
 			if target:checkHit(src:combatSpellpower(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 				target:knockback(src.x, src.y, 2)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatSpellpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
@@ -1340,7 +1346,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target == src then
-				target:setEffect(target.EFF_SANCTITY, 1, {power=dam})
+				target:setEffect(target.EFF_SANCTITY, 1, {power=dam, no_ct_effect=true})
 			elseif target:canBe("silence") then
 				target:setEffect(target.EFF_SILENCED, 2, {apply_power=src:combatSpellpower(), min_dur=1}, true)
 			else
@@ -1356,7 +1362,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target == src then
-				target:setEffect(target.EFF_SHIFTING_SHADOWS, 1, {power= dam})
+				target:setEffect(target.EFF_SHIFTING_SHADOWS, 1, {power= dam, no_ct_effect=true})
 			else
 				DamageType:get(DamageType.DARKNESS).projector(src, x, y, DamageType.DARKNESS, dam)
 			end
@@ -1370,7 +1376,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target == src then
-				target:setEffect(target.EFF_BLAZING_LIGHT, 1, {power= 1 + (dam / 4)})
+				target:setEffect(target.EFF_BLAZING_LIGHT, 1, {power= 1 + (dam / 4), no_ct_effect=true})
 			else
 				DamageType:get(DamageType.FIRE).projector(src, x, y, DamageType.FIRE, dam)
 				DamageType:get(DamageType.LIGHT).projector(src, x, y, DamageType.LIGHT, dam)
@@ -1385,12 +1391,13 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target == src then
-				target:setEffect(target.EFF_WARDING, 1, {power=dam})
+				target:setEffect(target.EFF_WARDING, 1, {power=dam, no_ct_effect=true})
 			elseif target ~= src then
 				DamageType:get(DamageType.LIGHT).projector(src, x, y, DamageType.LIGHT, dam )
 				DamageType:get(DamageType.DARKNESS).projector(src, x, y, DamageType.DARKNESS, dam)
 				if target:checkHit(src:combatSpellpower(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 					target:knockback(src.x, src.y, 1)
+					target:crossTierEffect(target.EFF_OFFBALANCE, src:combatSpellpower())
 					game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 				else
 					game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
@@ -1408,6 +1415,7 @@ newDamageType{
 		if target then
 			if target:canBe("knockback") then
 				target:knockback(src.x, src.y, 3)
+				target:crossTierEffect(target.EFF_OFFBALANCE, src:combatMindpower())
 				game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 			else
 				game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
@@ -1486,7 +1494,7 @@ newDamageType{
 		if target then
 			-- Set on fire!
 			dam = dam - init_dam
-			target:setEffect(target.EFF_WASTING, dur, {src=src, power=dam / dur})
+			target:setEffect(target.EFF_WASTING, dur, {src=src, power=dam / dur, no_ct_effect=true})
 		end
 		return init_dam
 	end,
@@ -1610,7 +1618,7 @@ newDamageType{
 		local target = game.level.map(x, y, Map.ACTOR)
 		if target then
 			if target:canBe("pin") then
-				target:setEffect(target.EFF_PINNED, 5, {})
+				target:setEffect(target.EFF_PINNED, 5, {no_ct_effect=true})
 			else
 				game.logSeen(target, "%s resists!", target.name:capitalize())
 			end
@@ -1627,7 +1635,7 @@ newDamageType{
 			if game.zone.void_blast_hits and game.party:hasMember(target) then game.zone.void_blast_hits = game.zone.void_blast_hits + 1 end
 
 			if target:knowTalent(target.T_MANA_POOL) then
-				target:setEffect(target.EFF_MANAWORM, 5, {power=dam * 5, src=src})
+				target:setEffect(target.EFF_MANAWORM, 5, {power=dam * 5, src=src, no_ct_effect=true})
 				src:disappear(src)
 			else
 				game.logSeen(target, "%s is unaffected.", target.name:capitalize())
diff --git a/game/modules/tome/data/general/objects/boss-artifacts.lua b/game/modules/tome/data/general/objects/boss-artifacts.lua
index bb5171bba5..f834824773 100644
--- a/game/modules/tome/data/general/objects/boss-artifacts.lua
+++ b/game/modules/tome/data/general/objects/boss-artifacts.lua
@@ -201,7 +201,6 @@ Tridents require the exotic weapons mastery talent to use correctly.]],
 	material_level = 4,
 	combat = {
 		dam = 60,
-		atk = 10,
 		apr = 4,
 		physcrit = 15,
 		dammod = {str=1.3},
@@ -213,6 +212,7 @@ Tridents require the exotic weapons mastery talent to use correctly.]],
 	},
 
 	wielder = {
+		combat_atk = 10,
 		combat_spellresist = 18,
 		see_invisible = 2,
 		resists={[DamageType.COLD] = 25},
diff --git a/game/modules/tome/data/general/objects/egos/weapon.lua b/game/modules/tome/data/general/objects/egos/weapon.lua
index 4e18aba67f..e75ad8e24d 100644
--- a/game/modules/tome/data/general/objects/egos/weapon.lua
+++ b/game/modules/tome/data/general/objects/egos/weapon.lua
@@ -85,7 +85,7 @@ newEntity{
 	level_range = {1, 50},
 	rarity = 3,
 	cost = 4,
-	combat={atk = resolvers.mbonus_material(20, 5)},
+	wielder={combat_atk = resolvers.mbonus_material(20, 5)},
 }
 
 newEntity{
@@ -157,6 +157,7 @@ newEntity{
 	rarity = 25,
 	cost = 35,
 	wielder = {
+		combat_atk = resolvers.mbonus_material(10, 2),
 		inc_damage = {
 			[DamageType.PHYSICAL] = resolvers.mbonus_material(15, 4),
 		},
@@ -168,7 +169,6 @@ newEntity{
 	},
 	combat = {
 		apr = resolvers.mbonus_material(8, 1),
-		atk = resolvers.mbonus_material(10, 2),
 	},
 }
 
diff --git a/game/modules/tome/data/general/objects/quest-artifacts.lua b/game/modules/tome/data/general/objects/quest-artifacts.lua
index 2b41b0ebb6..fe8e8808ad 100644
--- a/game/modules/tome/data/general/objects/quest-artifacts.lua
+++ b/game/modules/tome/data/general/objects/quest-artifacts.lua
@@ -38,12 +38,12 @@ Light around it seems to dim and you can feel its tremendous power simply by tou
 	combat = {
 		dam = 30,
 		apr = 4,
-		atk = 20,
 		dammod = {mag=1},
 		damtype = DamageType.ARCANE,
 		talented = "staff",
 	},
 	wielder = {
+		combat_atk = 20,
 		combat_spellpower = 20,
 		combat_spellcrit = 10,
 	},
diff --git a/game/modules/tome/data/general/objects/world-artifacts.lua b/game/modules/tome/data/general/objects/world-artifacts.lua
index 6a54374117..6ca1028cd8 100644
--- a/game/modules/tome/data/general/objects/world-artifacts.lua
+++ b/game/modules/tome/data/general/objects/world-artifacts.lua
@@ -163,11 +163,11 @@ newEntity{ base = "BASE_STAFF",
 	combat = {
 		dam = 64,
 		apr = 10,
-		atk = 7,
 		dammod = {mag=1.4},
 		damtype = DamageType.PHYSICAL,
 	},
 	wielder = {
+		combat_atk = 7,
 		combat_spellpower = 12,
 		combat_spellcrit = 18,
 		inc_damage={
@@ -651,10 +651,10 @@ newEntity{ base = "BASE_KNIFE",
 	combat = {
 		dam = 15,
 		apr = 10,
-		atk = 40,
 		physcrit = 0,
 		dammod = {dex=0.55, str=0.45},
 	},
+	wielder = {combat_atk=20},
 }
 
 newEntity{ base = "BASE_LEATHER_BOOT",
@@ -886,12 +886,12 @@ newEntity{ base = "BASE_KNIFE",
 	combat = {
 		dam = 25,
 		apr = 10,
-		atk = 15,
 		physcrit = 8,
 		dammod = {dex=0.55,str=0.35},
 		no_stealth_break = true,
 		melee_project={[DamageType.RANDOM_SILENCE] = 10},
 	},
+	wielder = {combat_atk = 10},
 }
 
 newEntity{ base = "BASE_KNIFE", define_as = "ART_PAIR_MOON",
@@ -1037,12 +1037,12 @@ newEntity{ base = "BASE_GREATMAUL",
 	combat = {
 		dam = 48,
 		apr = 15,
-		atk = 20,
 		physcrit = 3,
 		dammod = {str=1.2},
 		talent_on_hit = { [Talents.T_SUNDER_ARMOUR] = {level=3, chance=15} },
 	},
 	wielder = {
+		combat_atk = 20,
 		pin_immune = 1,
 		knockback_immune = 1,
 	},
@@ -1062,11 +1062,11 @@ newEntity{ base = "BASE_MACE",
 	combat = {
 		dam = 25,
 		apr = 4,
-		atk = 12,
 		physcrit = 10,
 		dammod = {str=1},
 		melee_project={[DamageType.RANDOM_CONFUSION] = 14},
 	},
+	wielder = {combat_atk=12,},
 }
 
 newEntity{ base = "BASE_MACE",
@@ -1083,11 +1083,11 @@ newEntity{ base = "BASE_MACE",
 	combat = {
 		dam = 40,
 		apr = 4,
-		atk = 6,
 		physcrit = 9,
 		dammod = {str=1},
 		melee_project={[DamageType.RANDOM_SILENCE] = 10, [DamageType.NATURE] = 18},
 	},
+	wielder = {combat_atk=6},
 
 	max_power = 25, power_regen = 1,
 	use_talent = { id = Talents.T_RUSH, level = 3, power = 15 },
@@ -1618,12 +1618,12 @@ newEntity{ base = "BASE_WHIP",
 	combat = {
 		dam = 28,
 		apr = 8,
-		atk = 10,
 		physcrit = 5,
 		dammod = {dex=1},
 		melee_project={[DamageType.POISON] = 22},
 	},
 	wielder = {
+		combat_atk = 10,
 		see_invisible = 9,
 	},
 }
diff --git a/game/modules/tome/data/maps/vaults/trickvault.lua b/game/modules/tome/data/maps/vaults/trickvault.lua
index 2d4d08d698..3f73ad40cf 100644
--- a/game/modules/tome/data/maps/vaults/trickvault.lua
+++ b/game/modules/tome/data/maps/vaults/trickvault.lua
@@ -20,6 +20,14 @@
 startx = 7
 starty = 12
 
+setStatusAll{no_teleport=true}
+
+specialList("terrain", {
+	"/data/general/grids/basic.lua",
+	"/data/general/grids/lava.lua",
+	"/data/general/grids/.lua",
+})
+
 defineTile('#', "HARDWALL")
 defineTile('O', "GLASSWALL")
 defineTile('!', "DOOR_VAULT", nil, nil, nil, {block = true})
diff --git a/game/modules/tome/data/talents/chronomancy/energy.lua b/game/modules/tome/data/talents/chronomancy/energy.lua
index dde3068f03..42920c25a5 100644
--- a/game/modules/tome/data/talents/chronomancy/energy.lua
+++ b/game/modules/tome/data/talents/chronomancy/energy.lua
@@ -141,7 +141,7 @@ newTalent{
 				self.talents_cd[tid] = self.talents_cd[tid] - cdr
 			end
 		end
-
+		target:crossTierEffect(target.EFF_SPELLSHOCKED, self:combatSpellpower())
 		game.level.map:particleEmitter(tx, ty, 1, "charge_matter")
 		game.level.map:particleEmitter(self.x, self.y, 1, "charge")
 		game:playSoundNear(self, "talents/spell_generic")
diff --git a/game/modules/tome/data/talents/chronomancy/spacetime-folding.lua b/game/modules/tome/data/talents/chronomancy/spacetime-folding.lua
index 560ab943ed..fd21e2a644 100644
--- a/game/modules/tome/data/talents/chronomancy/spacetime-folding.lua
+++ b/game/modules/tome/data/talents/chronomancy/spacetime-folding.lua
@@ -77,6 +77,7 @@ newTalent{
 		end
 		
 		if target:canBe("teleport") and self:checkHit(power, target:combatSpellResist() + (target:attr("continuum_destabilization") or 0)) then
+			target:crossTierEffect(target.EFF_SPELLSHOCKED, self:combatSpellpower())
 			-- first remove the target so the destination tile is empty
 			game.level.map:remove(target.x, target.y, Map.ACTOR)
 			local px, py 
diff --git a/game/modules/tome/data/talents/corruptions/blight.lua b/game/modules/tome/data/talents/corruptions/blight.lua
index d2eefbe025..601585457b 100644
--- a/game/modules/tome/data/talents/corruptions/blight.lua
+++ b/game/modules/tome/data/talents/corruptions/blight.lua
@@ -92,6 +92,7 @@ newTalent{
 				local eff = rng.tableRemove(effs)
 
 				if self:checkHit(self:combatSpellpower(), target:combatSpellResist(), 0, 95, 5) then
+					target:crossTierEffect(target.EFF_SPELLSHOCKED, self:combatSpellpower())
 					if eff[1] == "effect" then
 						target:removeEffect(eff[2])
 					else
diff --git a/game/modules/tome/data/talents/corruptions/scourge.lua b/game/modules/tome/data/talents/corruptions/scourge.lua
index 71470033b7..0d7ec63be5 100644
--- a/game/modules/tome/data/talents/corruptions/scourge.lua
+++ b/game/modules/tome/data/talents/corruptions/scourge.lua
@@ -45,14 +45,14 @@ newTalent{
 		-- Try to bleed !
 		if hit1 then
 			if target:canBe("cut") then
-				target:setEffect(target.EFF_CUT, 5, {power=self:combatTalentSpellDamage(t, 5, 40), src=self, apply_power=self:combatAttackStr(weapon.combat)})
+				target:setEffect(target.EFF_CUT, 5, {power=self:combatTalentSpellDamage(t, 5, 40), src=self, apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the cut!", target.name:capitalize())
 			end
 		end
 		if hit2 then
 			if target:canBe("cut") then
-				target:setEffect(target.EFF_CUT, 5, {power=self:combatTalentSpellDamage(t, 5, 40), src=self, apply_power=self:combatAttackStr(offweapon.combat)})
+				target:setEffect(target.EFF_CUT, 5, {power=self:combatTalentSpellDamage(t, 5, 40), src=self, apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the cut!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/cunning/ambush.lua b/game/modules/tome/data/talents/cunning/ambush.lua
index 359151e3ce..76070cfe92 100644
--- a/game/modules/tome/data/talents/cunning/ambush.lua
+++ b/game/modules/tome/data/talents/cunning/ambush.lua
@@ -38,7 +38,7 @@ newTalent{
 		if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end
 
 		if target:canBe("disarm") then
-			target:setEffect(target.EFF_DISARMED, t.getDuration(self, t), {apply_power=self:combatAttackDex()})
+			target:setEffect(target.EFF_DISARMED, t.getDuration(self, t), {apply_power=self:combatAttack()})
 		else
 			game.logSeen(target, "%s resists the shadow!", target.name:capitalize())
 		end
@@ -80,7 +80,7 @@ newTalent{
 
 		if core.fov.distance(self.x, self.y, sx, sy) <= 1 then
 			if target:canBe("silence") then
-				target:setEffect(target.EFF_SILENCED, t.getDuration(self, t), {apply_power=self:combatAttackDex()})
+				target:setEffect(target.EFF_SILENCED, t.getDuration(self, t), {apply_power=self:combatAttack()})
 			else
 				game.logSeen(target, "%s resists the shadow!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/cunning/dirty.lua b/game/modules/tome/data/talents/cunning/dirty.lua
index b92d5f5829..b4618c636a 100644
--- a/game/modules/tome/data/talents/cunning/dirty.lua
+++ b/game/modules/tome/data/talents/cunning/dirty.lua
@@ -40,7 +40,7 @@ newTalent{
 
 		if hitted then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, t.getDuration(self, t), {apply_power=self:combatAttackDex()})
+				target:setEffect(target.EFF_STUNNED, t.getDuration(self, t), {apply_power=self:combatAttack()})
 			else
 				game.logSeen(target, "%s resists the stun!", target.name:capitalize())
 			end
@@ -140,7 +140,7 @@ newTalent{
 			tw = tw or target.combat
 			local atk = target:combatAttack(tw) * (t.getAttackPenalty(self, t)) / 100
 			local dam = target:combatDamage(tw) * (t.getDamagePenalty(self, t)) / 100
-			target:setEffect(target.EFF_CRIPPLE, t.getDuration(self, t), {atk=atk, dam=dam, apply_power=self:combatAttackDex()})
+			target:setEffect(target.EFF_CRIPPLE, t.getDuration(self, t), {atk=atk, dam=dam, apply_power=self:combatAttack()})
 		end
 
 		return true
diff --git a/game/modules/tome/data/talents/cunning/traps.lua b/game/modules/tome/data/talents/cunning/traps.lua
index b2b3a8c7a3..94fdca2662 100644
--- a/game/modules/tome/data/talents/cunning/traps.lua
+++ b/game/modules/tome/data/talents/cunning/traps.lua
@@ -281,7 +281,7 @@ newTalent{
 		local t = basetrap(self, t, x, y, 8 + self:getTalentLevel(self.T_TRAP_MASTERY), {
 			type = "physical", name = "bear trap", color=colors.UMBER, image = "trap/beartrap01.png",
 			dam = dam,
-			check_hit = self:combatAttackDex(),
+			check_hit = self:combatAttack(),
 			triggered = function(self, x, y, who)
 				if who and who:canBe("cut") then who:setEffect(who.EFF_CUT, 5, {src=self.summoner, power=self.dam}) end
 				if who:canBe("pin") then
@@ -329,7 +329,7 @@ newTalent{
 		local t = basetrap(self, t, x, y, 8 + self:getTalentLevel(self.T_TRAP_MASTERY), {
 			type = "physical", name = "catapult trap", color=colors.LIGHT_UMBER, image = "trap/trap_catapult_01_64.png",
 			dist = 2 + math.ceil(self:getTalentLevel(self.T_TRAP_MASTERY)),
-			check_hit = self:combatAttackDex(),
+			check_hit = self:combatAttack(),
 			triggered = function(self, x, y, who)
 				-- Try to knockback !
 				local can = function(target)
@@ -383,7 +383,7 @@ newTalent{
 		local t = basetrap(self, t, x, y, 8 + self:getTalentLevel(self.T_TRAP_MASTERY), {
 			type = "physical", name = "disarming trap", color=colors.DARK_GREY, image = "trap/trap_magical_disarm_01_64.png",
 			dur = 2 + math.ceil(self:getTalentLevel(self.T_TRAP_MASTERY) / 2),
-			check_hit = self:combatAttackDex(),
+			check_hit = self:combatAttack(),
 			triggered = function(self, x, y, who)
 				if who:canBe("disarm") then
 					who:setEffect(who.EFF_DISARMED, self.dur, {apply_power=self.check_hit})
@@ -431,7 +431,7 @@ newTalent{
 		local t = basetrap(self, t, x, y, 5 + self:getTalentLevel(self.T_TRAP_MASTERY), {
 			type = "nature", name = "nightshade trap", color=colors.LIGHT_BLUE, image = "trap/poison_vines01.png",
 			dam = dam,
-			check_hit = self:combatAttackDex(),
+			check_hit = self:combatAttack(),
 			triggered = function(self, x, y, who)
 				self:project({type="hit", x=x,y=y}, x, y, engine.DamageType.NATURE, self.dam, {type="slime"})
 				if who:canBe("stun") then
@@ -476,7 +476,7 @@ newTalent{
 		local t = basetrap(self, t, x, y, 5 + self:getTalentLevel(self.T_TRAP_MASTERY), {
 			type = "elemental", name = "flash bang trap", color=colors.YELLOW, image = "trap/blast_acid01.png",
 			dur = math.floor(self:getTalentLevel(self.T_TRAP_MASTERY) + 4),
-			check_hit = self:combatAttackDex(),
+			check_hit = self:combatAttack(),
 			lure_trigger = true,
 			triggered = function(self, x, y, who)
 				self:project({type="ball", x=x,y=y, radius=2}, x, y, function(px, py)
diff --git a/game/modules/tome/data/talents/cursed/slaughter.lua b/game/modules/tome/data/talents/cursed/slaughter.lua
index a4405f893d..8678b3dfca 100644
--- a/game/modules/tome/data/talents/cursed/slaughter.lua
+++ b/game/modules/tome/data/talents/cursed/slaughter.lua
@@ -199,7 +199,8 @@ newTalent{
 
 				local blocked = true
 				if blockingTarget.size_category <= maxSize then
-					if blockingTarget:checkHit(self:combatAttackStr(), blockingTarget:combatPhysicalResist(), 0, 95, 15) and blockingTarget:canBe("knockback") then
+					if blockingTarget:checkHit(self:combatPhysicalpower(), blockingTarget:combatPhysicalResist(), 0, 95, 15) and blockingTarget:canBe("knockback") then
+						blockingTarget:crossTierEffect(blockingTarget.EFF_OFFBALANCE, self:combatPhysicalpower())
 						-- determine where to move the target (any adjacent square that isn't next to the attacker)
 						local start = rng.range(0, 8)
 						for i = start, start + 8 do
diff --git a/game/modules/tome/data/talents/gifts/fire-drake.lua b/game/modules/tome/data/talents/gifts/fire-drake.lua
index 5123e775fb..0add2d8a97 100644
--- a/game/modules/tome/data/talents/gifts/fire-drake.lua
+++ b/game/modules/tome/data/talents/gifts/fire-drake.lua
@@ -39,7 +39,7 @@ newTalent{
 		self:project(tg, self.x, self.y, DamageType.CONFUSION, {
 			dur=3,
 			dam=40 + 6 * self:getTalentLevel(t),
-			power_check=function() return self:combatAttackStr() end,
+			power_check=function() return self:combatPhysicalpower() end,
 			resist_check=self.combatPhysicalResist,
 		}, {type="flame"})
 		return true
diff --git a/game/modules/tome/data/talents/gifts/sand-drake.lua b/game/modules/tome/data/talents/gifts/sand-drake.lua
index d45db8bf6b..7e4cd350a9 100644
--- a/game/modules/tome/data/talents/gifts/sand-drake.lua
+++ b/game/modules/tome/data/talents/gifts/sand-drake.lua
@@ -42,7 +42,7 @@ newTalent{
 			return true
 		end
 
-		if (target:checkHit(self:combatAttackStr(), target:combatPhysicalResist(), 0, 95, 15) or target.dead) and (target:canBe("instakill") or target.life * 100 / target.max_life <= 5) then
+		if (target:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 15) or target.dead) and (target:canBe("instakill") or target.life * 100 / target.max_life <= 5) then
 			if not target.dead then target:die(self) end
 			world:gainAchievement("EAT_BOSSES", self, target)
 			self:incEquilibrium(-target.level - 5)
diff --git a/game/modules/tome/data/talents/gifts/storm-drake.lua b/game/modules/tome/data/talents/gifts/storm-drake.lua
index 2e10d76bff..20b5b6fe8e 100644
--- a/game/modules/tome/data/talents/gifts/storm-drake.lua
+++ b/game/modules/tome/data/talents/gifts/storm-drake.lua
@@ -67,6 +67,7 @@ newTalent{
 				game.logSeen(target, "%s resists the static field!", target.name:capitalize())
 				return
 			end
+			target:crossTierEffect(target.EFF_OFFBALANCE, self:combatMindpower())
 			game.logSeen(target, "%s is caught in the static field!", target.name:capitalize())
 			local dam = target.life * t.getPercent(self, t) / 100
 			if target.life - dam < 0 then dam = target.life end
diff --git a/game/modules/tome/data/talents/gifts/summon-utility.lua b/game/modules/tome/data/talents/gifts/summon-utility.lua
index 8e2240a9c2..8d29a638e3 100644
--- a/game/modules/tome/data/talents/gifts/summon-utility.lua
+++ b/game/modules/tome/data/talents/gifts/summon-utility.lua
@@ -75,7 +75,7 @@ newTalent{ short_name="SPIDER_WEB",
 		self:project(tg, x, y, function(tx, ty)
 			local target = game.level.map(tx, ty, Map.ACTOR)
 			if target and target:canBe("pin") then
-				target:setEffect(target.EFF_PINNED, 3 + self:getTalentLevel(t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_PINNED, 3 + self:getTalentLevel(t), {apply_power=self:combatPhysicalpower()})
 			end
 		end)
 		return true
diff --git a/game/modules/tome/data/talents/misc/horrors.lua b/game/modules/tome/data/talents/misc/horrors.lua
index 04c5d77c48..1c609aa059 100644
--- a/game/modules/tome/data/talents/misc/horrors.lua
+++ b/game/modules/tome/data/talents/misc/horrors.lua
@@ -49,7 +49,7 @@ newTalent{
 		local hit = self:attackTarget(target, nil, t.getDamage(self, t), true)
 		if hit then
 			if target:canBe("cut") then
-				target:setEffect(target.EFF_DEEP_WOUND, 5, {src=self, heal_factor=t.getHealingPenalty(self, t), power=t.getBleedDamage(self, t)/5, apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_DEEP_WOUND, 5, {src=self, heal_factor=t.getHealingPenalty(self, t), power=t.getBleedDamage(self, t)/5, apply_power=self:combatPhysicalpower()})
 			end
 		end
 		return true
@@ -151,7 +151,7 @@ newTalent{
 		local hit = self:attackTarget(target, nil, t.getDamage(self, t), true)
 
 		if target:canBe("cut") then
-			target:setEffect(target.EFF_CUT, 5, {power=t.getBleedDamage(self, t), src=self, apply_power=self:combatAttackStr()})
+			target:setEffect(target.EFF_CUT, 5, {power=t.getBleedDamage(self, t), src=self, apply_power=self:combatPhysicalpower()})
 			t.do_devourer_frenzy(self, target, t)
 		else
 			game.logSeen(target, "%s resists the cut!", target.name:capitalize())
@@ -392,6 +392,7 @@ newTalent{
 
 		if target:canBe("fear") then
 			target:setEffect(target.EFF_VOID_ECHOES, 6, {src=self, power=t.getDamage(self, t), apply_power=self:combatMindpower()})
+			target:crossTierEffect(target.EFF_VOID_ECHOES, self:combatMindpower())
 		else
 			game.logSeen(target, "%s resists the void!", target.name:capitalize())
 		end
diff --git a/game/modules/tome/data/talents/misc/misc.lua b/game/modules/tome/data/talents/misc/misc.lua
index f751948f18..aa02567bd1 100644
--- a/game/modules/tome/data/talents/misc/misc.lua
+++ b/game/modules/tome/data/talents/misc/misc.lua
@@ -29,6 +29,7 @@ load("/data/talents/misc/inscriptions.lua")
 load("/data/talents/misc/npcs.lua")
 load("/data/talents/misc/horrors.lua")
 load("/data/talents/misc/races.lua")
+load("/data/talents/misc/tutorial.lua")
 
 -- Default melee attack
 newTalent{
diff --git a/game/modules/tome/data/talents/misc/npcs.lua b/game/modules/tome/data/talents/misc/npcs.lua
index 22812349de..c4202a8e11 100644
--- a/game/modules/tome/data/talents/misc/npcs.lua
+++ b/game/modules/tome/data/talents/misc/npcs.lua
@@ -187,7 +187,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevel(t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevel(t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the stunning blow!", target.name:capitalize())
 			end
@@ -217,7 +217,8 @@ newTalent{
 		local hit = self:attackTarget(target, nil, self:combatTalentWeaponDamage(t, 0.5, 1), true)
 
 		if hit and target:canBe("disarm") then
-			target:setEffect(target.EFF_DISARMED, 2 + self:getTalentLevel(t), {apply_power=self:combatAttackStr()})
+			target:setEffect(target.EFF_DISARMED, 2 + self:getTalentLevel(t), {apply_power=self:combatPhysicalpower()})
+			target:crossTierEffect(target.EFF_DISARMED, self:combatPhysicalpower())
 		else
 			game.logSeen(target, "%s resists the blow!", target.name:capitalize())
 		end
@@ -248,7 +249,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_CONSTRICTED, (2 + self:getTalentLevel(t)) * 10, {src=self, power=1.5 * self:getTalentLevel(t), apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_CONSTRICTED, (2 + self:getTalentLevel(t)) * 10, {src=self, power=1.5 * self:getTalentLevel(t), apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the constriction!", target.name:capitalize())
 			end
@@ -279,8 +280,9 @@ newTalent{
 
 		-- Try to knockback !
 		if hit then
-			if target:checkHit(self:combatAttackStr(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then
+			if target:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then
 				target:knockback(self.x, self.y, 4)
+				target:crossTierEffect(target.EFF_OFFBALANCE, self:combatPhysicalpower())
 			else
 				game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
 			end
@@ -395,7 +397,7 @@ newTalent{
 		-- Try to rot !
 		if hit then
 			if target:canBe("disease") then
-				target:setEffect(target.EFF_ROTTING_DISEASE, 10 + self:getTalentLevel(t) * 3, {src=self, dam=self:getStr() / 3 + self:getTalentLevel(t) * 2, con=math.floor(4 + target:getCon() * 0.1), apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_ROTTING_DISEASE, 10 + self:getTalentLevel(t) * 3, {src=self, dam=self:getStr() / 3 + self:getTalentLevel(t) * 2, con=math.floor(4 + target:getCon() * 0.1), apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the disease!", target.name:capitalize())
 			end
@@ -426,7 +428,7 @@ newTalent{
 		-- Try to rot !
 		if hit then
 			if target:canBe("disease") then
-				target:setEffect(target.EFF_DECREPITUDE_DISEASE, 10 + self:getTalentLevel(t) * 3, {src=self, dam=self:getStr() / 3 + self:getTalentLevel(t) * 2, dex=math.floor(4 + target:getDex() * 0.1), apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_DECREPITUDE_DISEASE, 10 + self:getTalentLevel(t) * 3, {src=self, dam=self:getStr() / 3 + self:getTalentLevel(t) * 2, dex=math.floor(4 + target:getDex() * 0.1), apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the disease!", target.name:capitalize())
 			end
@@ -457,7 +459,7 @@ newTalent{
 		-- Try to rot !
 		if hit then
 			if target:canBe("disease") then
-				target:setEffect(target.EFF_WEAKNESS_DISEASE, 10 + self:getTalentLevel(t) * 3, {src=self, dam=self:getStr() / 3 + self:getTalentLevel(t) * 2, str=math.floor(4 + target:getStr() * 0.1), apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_WEAKNESS_DISEASE, 10 + self:getTalentLevel(t) * 3, {src=self, dam=self:getStr() / 3 + self:getTalentLevel(t) * 2, str=math.floor(4 + target:getStr() * 0.1), apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the disease!", target.name:capitalize())
 			end
@@ -653,7 +655,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_PINNED, 1 + self:getTalentLevel(t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_PINNED, 1 + self:getTalentLevel(t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the grab!", target.name:capitalize())
 			end
@@ -1044,7 +1046,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_PINNED, 2 + self:getTalentLevel(t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_PINNED, 2 + self:getTalentLevel(t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the crushing!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/misc/tutorial.lua b/game/modules/tome/data/talents/misc/tutorial.lua
new file mode 100644
index 0000000000..1377e2ed03
--- /dev/null
+++ b/game/modules/tome/data/talents/misc/tutorial.lua
@@ -0,0 +1,207 @@
+-- ToME - Tales of Maj'Eyal
+-- Copyright (C) 2009, 2010, 2011 Nicolas Casalini
+--
+-- This program is free software: you can redistribute it and/or modify
+-- it under the terms of the GNU General Public License as published by
+-- the Free Software Foundation, either version 3 of the License, or
+-- (at your option) any later version.
+--
+-- This program is distributed in the hope that it will be useful,
+-- but WITHOUT ANY WARRANTY; without even the implied warranty of
+-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+-- GNU General Public License for more details.
+--
+-- You should have received a copy of the GNU General Public License
+-- along with this program.  If not, see <http://www.gnu.org/licenses/>.
+--
+-- Nicolas Casalini "DarkGod"
+-- darkgod@te4.org
+
+-- race & classes
+newTalentType{ type="tutorial", name = "tutorial", hide = true, description = "Tutorial-specific talents." }
+
+newTalent{
+	name = "Shove", short_name = "TUTORIAL_PHYS_KB",
+	type = {"tutorial", 1},
+	points = 5,
+	random_ego = "attack",
+	cooldown = 0,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > 1 then return nil end
+		if self:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist()) then
+			target:knockback(self.x, self.y, 1)
+		else
+			game.logSeen(target, "%s resists the shove!", target.name:capitalize())
+		end
+		return true
+	end,
+	info = function(self, t)
+		return ([[Give the target a good old-fashioned shove, knocking it back a square.]])
+	end,
+}
+
+newTalent{
+	name = "Mana Gale", short_name = "TUTORIAL_SPELL_KB",
+	type = {"tutorial", 1},
+	points = 5,
+	range = 3,
+	random_ego = "attack",
+	cooldown = 0,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > self:getTalentRange(t) then return nil end
+		if self:checkHit(self:combatSpellpower(), target:combatPhysicalResist()) then
+			target:knockback(self.x, self.y, self:getTalentLevel(t))
+			game.logSeen(target, "%s is knocked back by the gale!", target.name:capitalize())
+			target:crossTierEffect(target.EFF_OFFBALANCE, self:combatSpellpower())
+		else
+			game.logSeen(target, "%s remains firmly planted in the face of the gale!", target.name:capitalize())
+		end
+		return true
+	end,
+	info = function(self, t)
+		local dist = self:getTalentLevel(t)
+		return ([[Conjure up a powerful magical wind, pushing the target back a distance of %d.]]):format(dist)
+	end,
+}
+
+newTalent{
+	name = "Telekinetic Punt", short_name = "TUTORIAL_MIND_KB",
+	type = {"tutorial", 1},
+	points = 5,
+	range = 3,
+	random_ego = "attack",
+	cooldown = 0,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > self:getTalentRange(t) then return nil end
+		if self:checkHit(self:combatMindpower(), target:combatPhysicalResist()) then
+			target:knockback(self.x, self.y, 1)
+			game.logSeen(target, "%s is knocked back by the telekinetic blow!", target.name:capitalize())
+			target:crossTierEffect(target.EFF_OFFBALANCE, self:combatMindpower())
+		else
+			game.logSeen(target, "%s holds its ground!", target.name:capitalize())
+		end
+		return true
+	end,
+	info = function(self, t)
+		return ([[Knock the target backwards with a powerful telekinetic blow.]])
+	end,
+}
+
+newTalent{
+	name = "Blink", short_name = "TUTORIAL_SPELL_BLINK",
+	type = {"tutorial", 1},
+	points = 5,
+	range = 3,
+	random_ego = "attack",
+	cooldown = 0,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > self:getTalentRange(t) then return nil end
+		if self:checkHit(self:combatSpellpower(), target:combatSpellResist()) then
+			target:knockback(self.x, self.y, 1)
+			game.logSeen(target, "%s is teleported a short distance!", target.name:capitalize())
+			target:crossTierEffect(target.EFF_SPELLSHOCKED, self:combatSpellpower())
+		else
+			game.logSeen(target, "%s resists the teleportation!", target.name:capitalize())
+		end
+		return true
+	end,
+	info = function(self, t)
+		return ([[Attempts to magically teleport a target slightly farther from you.]])
+	end,
+}
+
+newTalent{
+	name = "Fear", short_name = "TUTORIAL_MIND_FEAR",
+	type = {"tutorial", 1},
+	points = 5,
+	range = 3,
+	random_ego = "attack",
+	cooldown = 0,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > self:getTalentRange(t) then return nil end
+		if self:checkHit(self:combatMindpower(), target:combatMentalResist()) then
+			target:knockback(self.x, self.y, 1)
+			game.logSeen(target, "%s retreats in terror!", target.name:capitalize())
+			target:crossTierEffect(target.EFF_BRAINLOCKED, self:combatMindpower())
+		else
+			game.logSeen(target, "%s shakes off the fear!", target.name:capitalize())
+		end
+		return true
+	end,
+	info = function(self, t)
+		return ([[Attempts to briefly terrify a target into retreating.]])
+	end,
+}
+
+newTalent{
+	name = "Bleed", short_name = "TUTORIAL_SPELL_BLEED",
+	type = {"tutorial", 1},
+	points = 5,
+	range = 5,
+	random_ego = "attack",
+	cooldown = 0,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > self:getTalentRange(t) then return nil end
+		if target then
+			target:setEffect(self.EFF_CUT, 10, {power=1, apply_power=self:combatSpellpower()})
+		end
+		return true
+	end,
+	info = function(self, t)
+		return ([[Inflicts a 10-turn bleed effect.]])
+	end,
+}
+
+newTalent{
+	name = "Confusion", short_name = "TUTORIAL_MIND_CONFUSION",
+	type = {"tutorial", 1},
+	points = 5,
+	range = 3,
+	random_ego = "attack",
+	cooldown = 6,
+	requires_target = true,
+	tactical = { ATTACK = 2},
+	action = function(self, t)
+		local tg = {type="hit", range=self:getTalentRange(t)}
+		local x, y, target = self:getTarget(tg)
+		if not x or not y or not target then return nil end
+		if core.fov.distance(self.x, self.y, x, y) > self:getTalentRange(t) then return nil end
+		if target then
+			target:setEffect(self.EFF_CONFUSED, 5, {power=100, apply_power=self:combatMindpower()})
+		end
+		return true
+	end,
+	info = function(self, t)
+		return ([[Use your mental powers to confuse the target for five turns.]])
+	end,
+}
diff --git a/game/modules/tome/data/talents/psionic/absorption.lua b/game/modules/tome/data/talents/psionic/absorption.lua
index 8875028e53..30ad5080f0 100644
--- a/game/modules/tome/data/talents/psionic/absorption.lua
+++ b/game/modules/tome/data/talents/psionic/absorption.lua
@@ -23,7 +23,7 @@ local function getShieldStrength(self, t)
 		add = getGemLevel(self)*(1 + 0.1*(self:getTalentLevel(self.T_FOCUSED_CHANNELING) or 0))
 	end
 	--return 2 + (1+ self:getWil(8))*self:getTalentLevel(t) + add
-	return self:combatTalentIntervalDamage(t, "wil", 3, 50) + add
+	return self:combatStatTalentIntervalDamage(t, "combatMindpower", 3, 40) + add
 end
 
 local function getSpikeStrength(self, t)
diff --git a/game/modules/tome/data/talents/psionic/augmented-mobility.lua b/game/modules/tome/data/talents/psionic/augmented-mobility.lua
index 2f3f5cf758..8e1f5b54e7 100644
--- a/game/modules/tome/data/talents/psionic/augmented-mobility.lua
+++ b/game/modules/tome/data/talents/psionic/augmented-mobility.lua
@@ -69,8 +69,11 @@ newTalent{
 	psi = 30,
 	no_energy = true,
 	require = psi_wil_20_2,
+	getDuration = function(self, t) 
+		return 10 + self:combatMindpower(0.1)
+	end,
 	action = function(self, t)
-		self:setEffect(self.EFF_QUICKNESS, 10+self:getWil(10), {power=self:getTalentLevel(t) * 0.2})
+		self:setEffect(self.EFF_QUICKNESS, t.getDuration(self, t), {power=self:getTalentLevel(t) * 0.2})
 		return true
 	end,
 	info = function(self, t)
@@ -78,7 +81,7 @@ newTalent{
 		local percentinc = 100 * inc
 		--local percentinc = ((1/(1-inc))-1)*100
 		return ([[You encase your legs in precise sheathes of force, increasing your movement speed by %d%% for %d turns.]]):
-		format(percentinc, 10+self:getWil(10))
+		format(percentinc, t.getDuration(self, t))
 	end,
 }
 
diff --git a/game/modules/tome/data/talents/psionic/finer-energy-manipulations.lua b/game/modules/tome/data/talents/psionic/finer-energy-manipulations.lua
index 24cd16f7a1..fa0c0b08be 100644
--- a/game/modules/tome/data/talents/psionic/finer-energy-manipulations.lua
+++ b/game/modules/tome/data/talents/psionic/finer-energy-manipulations.lua
@@ -21,19 +21,22 @@ newTalent{
 	name = "Perfect Control",
 	type = {"psionic/finer-energy-manipulations", 1},
 	require = psi_cun_high1,
-	cooldown = 100,
+	cooldown = 50,
 	psi = 15,
 	points = 5,
 	tactical = { BUFF = 2 },
+	getBoost = function(self, t)
+		return 15 + math.ceil(self:getTalentLevel(t)*self:combatStatTalentIntervalDamage(t, "combatMindpower", 1, 9))
+	end,
 	action = function(self, t)
-		self:setEffect(self.EFF_CONTROL, 5 + self:getTalentLevelRaw(t), {power=15 + math.ceil(self:getTalentLevel(t)*(1 + self:getCun(8, true)))})
+		self:setEffect(self.EFF_CONTROL, 5 + self:getTalentLevelRaw(t), {power= t.getBoost(self, t)})
 		return true
 	end,
 	info = function(self, t)
-		local boost = 15 + math.ceil(self:getTalentLevel(t)*(1 + self:getCun(8, true)))
+		local boost = t.getBoost(self, t)
 		local dur = 5 + self:getTalentLevelRaw(t)
 		return ([[Encase your body in a sheath of thought-quick forces, allowing you to control your body's movements directly without the inefficiency of dealing with crude mechanisms like nerves and muscles.
-		Increases attack by %d and critical strike chance by %0.2f%% for %d turns. The effect scales with Cunning.]]):
+		Increases attack by %d and critical strike chance by %0.2f%% for %d turns.]]):
 		format(boost, 0.5*boost, dur)
 	end,
 }
@@ -48,7 +51,7 @@ newTalent{
 	no_npc_use = true,
 	no_unlearn_last = true,
 	boost = function(self, t)
-		return math.floor(self:combatTalentIntervalDamage(t, "wil", 3, 20))
+		return math.floor(self:combatStatTalentIntervalDamage(t, "combatMindpower", 3, 20))
 	end,
 	action = function(self, t)
 		local d d = self:showInventory("Reshape which weapon?", self:getInven("INVEN"), function(o) return not o.quest and o.type == "weapon" and not o.fully_reshaped end, function(o, item)
@@ -77,8 +80,7 @@ newTalent{
 	end,
 	info = function(self, t)
 		local weapon_boost = t.boost(self, t)
-		return ([[Manipulate forces on the molecular level to realign, rebalance, and hone your weapon. Permanently increases the accuracy and damage of any weapon by %d.
-		This value scales with Willpower.]]):
+		return ([[Manipulate forces on the molecular level to realign, rebalance, and hone your weapon. Permanently increases the accuracy and damage of any weapon by %d.]]):
 		format(weapon_boost)
 	end,
 }
@@ -93,8 +95,6 @@ newTalent{
 	no_npc_use = true,
 	no_unlearn_last = true,
 	arm_boost = function(self, t)
-		--return math.floor(0.1*self:combatTalentIntervalDamage(t, "wil", 10, 30))
-		--return math.floor(0.25*t.fat_red(self, t))
 		local arm_values = {
 		0 + self:getWil(2),
 		1 + self:getWil(2),
@@ -106,7 +106,6 @@ newTalent{
 		return arm_values[index] * (self:getTalentLevel(t) / self:getTalentLevelRaw(t))
 	end,
 	fat_red = function(self, t)
-		--return math.floor(0.1*self:combatTalentIntervalDamage(t, "wil", 50, 100))
 		local fat_values = {
 		1 + self:getWil(3),
 		1 + self:getWil(3),
@@ -168,13 +167,11 @@ newTalent{
 	points = 5,
 	no_npc_use = true,
 	energy_per_turn = function(self, t)
-		--return 5 + 2 * math.ceil(self:getTalentLevel(t)) + self:getCun(5)
-		return self:combatTalentIntervalDamage(t, "cun", 10, 40, 0.25)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 10, 40, 0.25)
 	end,
 	action = function(self, t)
 		local d d = self:showInventory("Use which gem?", self:getInven("INVEN"), function(gem) return gem.type == "gem" and gem.material_level and not gem.unique end, function(gem, gem_item)
 			self:removeObject(self:getInven("INVEN"), gem_item)
-			--local amt = 0.1*self:combatTalentIntervalDamage(t, "cun", 100, 400)
 			local amt = t.energy_per_turn(self, t)
 			local dur = 3 + 2*(gem.material_level or 0)
 			self:setEffect(self.EFF_PSI_REGEN, dur, {power=amt})
@@ -189,7 +186,7 @@ newTalent{
 	info = function(self, t)
 		local amt = t.energy_per_turn(self, t)
 		return ([[Matter is energy, as any good Mindslayer knows. Unfortunately, the various bonds and particles involved are just too numerous and complex to make the conversion feasible in most cases. Fortunately, the organized, crystalline structure of gems makes it possible to transform a small percentage of its matter into usable energy.
-		Grants %d energy per turn for between five and thirteen turns, depending on the quality of the gem used. The amount of energy granted per turn scales with Cunning.]]):
+		Grants %d energy per turn for between five and thirteen turns, depending on the quality of the gem used.]]):
 		format(amt)
 	end,
 }
diff --git a/game/modules/tome/data/talents/psionic/focus.lua b/game/modules/tome/data/talents/psionic/focus.lua
index b7aca2203c..5e84fd9b8c 100644
--- a/game/modules/tome/data/talents/psionic/focus.lua
+++ b/game/modules/tome/data/talents/psionic/focus.lua
@@ -44,13 +44,12 @@ newTalent{
 	end,
 	getDamage = function (self, t)
 		local gem_level = getGemLevel(self)
-		return self:combatTalentIntervalDamage(t, "wil", 6, 265)*(1 + 0.3*gem_level)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 6, 170)*(1 + 0.3*gem_level)
 	end,
 	requires_target = true,
 	target = function(self, t) return {type="ball", range=self:getTalentRange(t), radius=0, selffire=false, talent=t} end,
 	action = function(self, t)
 		local gem_level = getGemLevel(self)
-		--local dam = (5 + self:getTalentLevel(t) * self:getWil(40))*(1 + 0.3*gem_level)
 		local dam = t.getDamage(self, t)
 		local tg = self:getTalentTarget(t)
 		local x, y = self:getTarget(tg)
@@ -81,8 +80,6 @@ newTalent{
 		return true
 	end,
 	info = function(self, t)
-		--local gem_level = getGemLevel(self)
-		--local dam = (5 + self:getTalentLevel(t) * self:getWil(40))*(1 + 0.3*gem_level)
 		local dam = t.getDamage(self, t)
 		return ([[Focus energies on a distant target to lash it with physical force, doing %d damage.
 		Mindslayers do not do this sort of ranged attack naturally. The use of a telekinetically-wielded gem as a focus will improve the effects considerably.]]):
@@ -113,7 +110,7 @@ newTalent{
 	end,
 	getDamage = function (self, t)
 		local gem_level = getGemLevel(self)
-		return self:combatTalentIntervalDamage(t, "wil", 21, 281)*(1 + 0.3*gem_level)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 21, 200)*(1 + 0.3*gem_level)
 	end,
 	target = function(self, t)
 		return {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t), friendlyfire=false}
@@ -126,8 +123,6 @@ newTalent{
 	end,
 	info = function(self, t)
 		local radius = self:getTalentRadius(t)
-		--local gem_level = getGemLevel(self)
-		--local dam = (20 + self:getTalentLevel(t) * self:getWil(40))*(1 + 0.3*gem_level)
 		local dam = t.getDamage(self, t)
 		return ([[Focus energies on all foes within %d squares, setting them ablaze. Does %d damage over ten turns.
 		Mindslayers do not do this sort of ranged attack naturally. The use of a telekinetically-wielded gem as a focus will improve the effects considerably.]]):
diff --git a/game/modules/tome/data/talents/psionic/grip.lua b/game/modules/tome/data/talents/psionic/grip.lua
index b2764f5696..f0f2dd7f29 100644
--- a/game/modules/tome/data/talents/psionic/grip.lua
+++ b/game/modules/tome/data/talents/psionic/grip.lua
@@ -70,8 +70,7 @@ newTalent{
 	end,
 	getDuration = function (self, t)
 		local gem_level = getGemLevel(self)
-		--return 5 + self:getWil(5) + self:getTalentLevel(t) + gem_level
-		return self:combatTalentIntervalDamage(t, "wil", 3, 12, 0.2) + gem_level
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 3, 10, 0.2) + gem_level
 	end,
 	requires_target = true,
 	target = function(self, t) return {type="ball", range=self:getTalentRange(t), radius=0, selffire=false, talent=t} end,
@@ -121,11 +120,11 @@ newTalent{
 	end,
 	getDuration = function (self, t)
 		local gem_level = getGemLevel(self)
-		return self:combatTalentIntervalDamage(t, "wil", 2, 6, 0.2)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 2, 6, 0.2)
 	end,
 	getDamage = function (self, t)
 		local gem_level = getGemLevel(self)
-		return self:combatTalentIntervalDamage(t, "wil", 100, 200, 0.25)*(1 + 0.1*gem_level)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 100, 150, 0.25)*(1 + 0.1*gem_level)
 	end,
 	requires_target = true,
 	target = function(self, t) return {type="ball", range=self:getTalentRange(t), radius=0, selffire=false, talent=t} end,
@@ -136,6 +135,7 @@ newTalent{
 		local x, y = self:getTarget(tg)
 		if not x or not y then return nil end
 		self:project(tg, x, y, DamageType.IMPLOSION, {dur=dur, dam=dam})
+		target:crossTierEffect(target.EFF_OFFBALANCE, self:combatMindpower())
 
 		return true
 	end,
diff --git a/game/modules/tome/data/talents/psionic/projection.lua b/game/modules/tome/data/talents/psionic/projection.lua
index 3fbc875ad8..d5d460acab 100644
--- a/game/modules/tome/data/talents/psionic/projection.lua
+++ b/game/modules/tome/data/talents/psionic/projection.lua
@@ -21,8 +21,7 @@ local function aura_strength(self, t)
 	if self:knowTalent(self.T_FOCUSED_CHANNELING) then
 		add = getGemLevel(self)*(1 + 0.1*(self:getTalentLevel(self.T_FOCUSED_CHANNELING) or 0))
 	end
-	--return 5 + (1+ self:getWil(5))*self:getTalentLevel(t) + add
-	return self:combatTalentIntervalDamage(t, "wil", 10, 50) + add
+	return self:combatStatTalentIntervalDamage(t, "combatMindpower", 10, 40) + add
 end
 
 local function aura_spike_strength(self, t)
@@ -183,7 +182,6 @@ newTalent{
 		local spikecost = t.getSpikeCost(self, t)
 		return ([[Fills the air around you with reactive currents of force that do %d physical damage to all who approach. All damage done by the aura will drain one point of energy per %0.2f points of damage dealt.
 		When deactivated, if you have at least %d energy, a massive spike of kinetic energy is released as a beam, smashing targets for %d physical damage and sending them flying. Telekinetically wielding a gem instead of a weapon will result in improved spike efficiency.
-		The damage will increase with the Willpower stat.
 		To turn off an aura without spiking it, deactivate it and target yourself.]]):format(dam, mast, spikecost, spikedam)
 	end,
 }
@@ -300,7 +298,6 @@ newTalent{
 		local spikecost = t.getSpikeCost(self, t)
 		return ([[Fills the air around you with reactive currents of furnace-like heat that do %d fire damage to all who approach. All damage done by the aura will drain one point of energy per %0.2f points of damage dealt.
 		When deactivated, if you have at least %d energy, a massive spike of thermal energy is released as a conical blast (radius %d) of superheated air. Anybody caught in it will suffer %d fire damage over several turns. Telekinetically wielding a gem instead of a weapon will result in improved spike efficiency.
-		The damage will increase with the Willpower stat.
 		To turn off an aura without spiking it, deactivate it and target yourself.]]):format(dam, mast, spikecost, rad, spikedam)
 	end,
 }
@@ -462,7 +459,6 @@ newTalent{
 		local nb = t.getNumSpikeTargets(self, t)
 		return ([[Fills the air around you with crackling energy, doing %d lightning damage to all who stand nearby. All damage done by the aura will drain one point of energy per %0.2f points of damage dealt.
 		When deactivated, if you have at least %d energy, a massive spike of electrical energy jumps between up to %d nearby targets, doing %d lightning damage to each. Telekinetically wielding a gem instead of a weapon will result in improved spike efficiency.
-		The damage will increase with the Willpower stat.
 		To turn off an aura without spiking it, deactivate it and target yourself.]]):format(dam, mast, spikecost, nb, spikedam)
 	end,
 }
diff --git a/game/modules/tome/data/talents/psionic/voracity.lua b/game/modules/tome/data/talents/psionic/voracity.lua
index 7fbe499001..e0a721d7ce 100644
--- a/game/modules/tome/data/talents/psionic/voracity.lua
+++ b/game/modules/tome/data/talents/psionic/voracity.lua
@@ -38,9 +38,15 @@ newTalent{
 	target = function(self, t)
 		return {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t), selffire=false, talent=t}
 	end,
+	getLeech = function(self, t)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 4, 20)
+	end,
+	getSlow = function(self, t)
+		return math.min(5 * self:getTalentLevel(t) + 15, 50)
+	end,
 	action = function(self, t)
-		local en = ( 3 + self:getTalentLevel(t)) * (100 + self:getWil())/100
-		local dam = math.min(0.1 + 0.03*self:getTalentLevel(t), 0.4)
+		local en = t.getLeech(self, t)
+		local dam = t.getSlow(self, t)/100
 		local tg = self:getTalentTarget(t)
 		self:project(tg, self.x, self.y, function(tx, ty)
 			local act = game.level.map(tx, ty, engine.Map.ACTOR)
@@ -53,11 +59,10 @@ newTalent{
 	end,
 	info = function(self, t)
 		local range = self:getTalentRadius(t)
-		local slow = 3 * self:getTalentLevel(t) + 10
-		local en = ( 3 + self:getTalentLevel(t)) * (100 + self:getWil())/100
+		local slow = t.getSlow(self, t)
+		local en = t.getLeech(self, t)
 		return ([[You suck the kinetic energy out of your surroundings, slowing all targets in a radius of %d by %d%% for four turns.
-		For each target drained, you gain %d energy.
-		The energy gained scales with Willpower.]]):format(range, slow, en)
+		For each target drained, you gain %d energy.]]):format(range, slow, en)
 	end,
 }
 
@@ -81,9 +86,15 @@ newTalent{
 	target = function(self, t)
 		return {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t), selffire=false, talent=t}
 	end,
+	getLeech = function(self, t)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 5, 25)
+	end,
+	getDam = function(self, t)
+		return math.ceil(1 + 0.5*self:getTalentLevel(t))
+	end,
 	action = function(self, t)
-		local en = ( 4 + self:getTalentLevel(t)) * (100 + self:getWil())/85
-		local dam = math.ceil(1 + 0.5*self:getTalentLevel(t))
+		local en = t.getLeech(self, t)
+		local dam = t.getDam(self, t)
 		local tg = self:getTalentTarget(t)
 		self:project(tg, self.x, self.y, function(tx, ty)
 			local act = game.level.map(tx, ty, engine.Map.ACTOR)
@@ -96,11 +107,10 @@ newTalent{
 	end,
 	info = function(self, t)
 		local range = self:getTalentRadius(t)
-		local dam = math.ceil(1 + 0.5*self:getTalentLevel(t))
-		local en = ( 4 + self:getTalentLevel(t)) * (100 + self:getWil())/85
+		local dam = t.getDam(self, t)
+		local en = t.getLeech(self, t)
 		--local duration = self:getTalentLevel(t) + 2
-		return ([[You leech the heat out of all targets in a radius of %d, freezing them for up to %d turns and gaining %d energy for each target frozen.
-		The energy gained scales with Willpower.]]):
+		return ([[You leech the heat out of all targets in a radius of %d, freezing them for up to %d turns and gaining %d energy for each target frozen.]]):
 		format(range, dam, en)
 	end,
 }
@@ -126,16 +136,22 @@ newTalent{
 	target = function(self, t)
 		return {type="ball", range=self:getTalentRange(t), radius=self:getTalentRadius(t), selffire=false, talent=t}
 	end,
+	getLeech = function(self, t)
+		return self:combatStatTalentIntervalDamage(t, "combatMindpower", 6, 30)
+	end,
+	getDam = function(self, t)
+		return self:spellCrit(self:combatTalentMindDamage(t, 28, 270))
+	end,
 	action = function(self, t)
-		local en = ( 5 + self:getTalentLevel(t)) * (100 + self:getWil())/75
-		local dam = self:spellCrit(self:combatTalentMindDamage(t, 28, 270))
+		local en = t.getLeech(self, t)
+		local dam = t.getDam(self, t)
 		local tg = self:getTalentTarget(t)
 		self:project(tg, self.x, self.y, function(tx, ty)
 			local act = game.level.map(tx, ty, engine.Map.ACTOR)
 			if act then
 				self:incPsi(en)
 			end
-			DamageType:get(DamageType.LIGHTNING_DAZE).projector(self, tx, ty, DamageType.LIGHTNING_DAZE, dam)
+			DamageType:get(DamageType.LIGHTNING_DAZE).projector(self, tx, ty, DamageType.LIGHTNING_DAZE, rng.avg(dam/3, dam, 3))
 		end)
 		-- Lightning ball gets a special treatment to make it look neat
 		local sradius = (tg.radius + 0.5) * (engine.Map.tile_w + engine.Map.tile_h) / 2
@@ -153,10 +169,9 @@ newTalent{
 	end,
 	info = function(self, t)
 		local range = self:getTalentRadius(t)
-		local en = ( 5 + self:getTalentLevel(t)) * (100 + self:getWil())/75
-		local dam = damDesc(self, DamageType.LIGHTNING, self:combatTalentMindDamage(t, 28, 270))
-		return ([[You pull electric potential from all targets around you in a radius of %d, gaining %d energy for each one affected and giving them a nasty shock in the process. Deals between %d and %d damage.
-		The energy gained scales with Willpower.]]):format(range, en, dam / 3, dam)
+		local en = t.getLeech(self, t)
+		local dam = damDesc(self, DamageType.LIGHTNING, t.getDam(self, t))
+		return ([[You pull electric potential from all targets around you in a radius of %d, gaining %d energy for each one affected and giving them a nasty shock in the process. Deals between %d and %d damage.]]):format(range, en, dam / 3, dam)
 	end,
 }
 newTalent{
diff --git a/game/modules/tome/data/talents/spells/arcane-shield.lua b/game/modules/tome/data/talents/spells/arcane-shield.lua
index 2f04669913..1ffa1feeec 100644
--- a/game/modules/tome/data/talents/spells/arcane-shield.lua
+++ b/game/modules/tome/data/talents/spells/arcane-shield.lua
@@ -43,7 +43,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, 2 + math.floor(self:getTalentLevel(t) / 2), {apply_power=self:combatAttackStr(shield.special_combat)})
+				target:setEffect(target.EFF_STUNNED, 2 + math.floor(self:getTalentLevel(t) / 2), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the stun!", target.name:capitalize())
 			end
@@ -113,7 +113,7 @@ newTalent{
 		-- Try to stun !
 		if hit1 or hit2 or hit3 then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_DAZED, 3 + math.floor(self:getTalentLevel(t)), {apply_power=self:combatAttackStr(shield.special_combat), apply_save="combatSpellResist"})
+				target:setEffect(target.EFF_DAZED, 3 + math.floor(self:getTalentLevel(t)), {apply_power=self:combatPhysicalpower(), apply_save="combatSpellResist"})
 			else
 				game.logSeen(target, "%s resists the dazing blows!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/spells/golem.lua b/game/modules/tome/data/talents/spells/golem.lua
index 9036fcc199..86e41a9d2a 100644
--- a/game/modules/tome/data/talents/spells/golem.lua
+++ b/game/modules/tome/data/talents/spells/golem.lua
@@ -73,8 +73,9 @@ newTalent{
 
 		-- Try to knockback !
 		if hit then
-			if target:checkHit(self:combatAttackStr(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then
+			if target:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then
 				target:knockback(self.x, self.y, 3)
+				target:crossTierEffect(target.EFF_OFFBALANCE, self:combatPhysicalpower())
 			else
 				game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
 			end
@@ -183,7 +184,7 @@ newTalent{
 		-- Try to pin
 		if hit then
 			if target:canBe("pin") then
-				target:setEffect(target.EFF_PINNED, t.getPinDuration(self, t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_PINNED, t.getPinDuration(self, t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the crushing!", target.name:capitalize())
 			end
@@ -257,7 +258,7 @@ newTalent{
 			local target = game.level.map(xx, yy, Map.ACTOR)
 			if target and self:attackTarget(target, nil, t.getGolemDamage(self, t), true) then
 				if target:canBe("stun") then
-					target:setEffect(target.EFF_DAZED, t.getDazeDuration(self, t), {apply_power=self:combatAttackStr()})
+					target:setEffect(target.EFF_DAZED, t.getDazeDuration(self, t), {apply_power=self:combatPhysicalpower()})
 				else
 					game.logSeen(target, "%s resists the dazing blow!", target.name:capitalize())
 				end
diff --git a/game/modules/tome/data/talents/spells/staff-combat.lua b/game/modules/tome/data/talents/spells/staff-combat.lua
index bfbe467ad4..29e0b42e6d 100644
--- a/game/modules/tome/data/talents/spells/staff-combat.lua
+++ b/game/modules/tome/data/talents/spells/staff-combat.lua
@@ -106,11 +106,13 @@ newTalent{
 	mode = "passive",
 	require = spells_req2,
 	points = 5,
-	getDamage = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) end,
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		return ([[Increases damage done with staves by %d%%.]]):
-		format(100 * damage)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. Also increases damage done with staves by %d%%.]]):
+		format(damage, 100 * inc)
 	end,
 }
 
@@ -181,7 +183,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_DAZED, t.getDazeDuration(self, t), {apply_power=self:combatAttackStr(weapon.combat)})
+				target:setEffect(target.EFF_DAZED, t.getDazeDuration(self, t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the dazing blow!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/techniques/2hweapon.lua b/game/modules/tome/data/talents/techniques/2hweapon.lua
index 464d821fe3..df9ab66541 100644
--- a/game/modules/tome/data/talents/techniques/2hweapon.lua
+++ b/game/modules/tome/data/talents/techniques/2hweapon.lua
@@ -135,7 +135,7 @@ newTalent{
 		self:project(tg, x, y, DamageType.CONFUSION, {
 			dur=3+self:getTalentLevelRaw(t),
 			dam=50+self:getTalentLevelRaw(t)*10,
-			power_check=function() return self:combatAttackStr(weapon) end,
+			power_check=function() return self:combatPhysicalpower() end,
 			resist_check=self.combatPhysicalResist,
 		}, {type="flame"})
 		return true
@@ -185,7 +185,7 @@ newTalent{
 
 		-- Try to insta-kill
 		if hit then
-			if target:checkHit(self:combatAttackStr(weapon.combat), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("instakill") and target.life > 0 and target.life < target.max_life * 0.2 then
+			if target:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("instakill") and target.life > 0 and target.life < target.max_life * 0.2 then
 				-- KILL IT !
 				game.logSeen(target, "%s feels the pain of the death blow!", target.name:capitalize())
 				target:die(self)
@@ -232,7 +232,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevel(t), {apply_power=self:combatAttackStr(weapon.combat)})
+				target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevel(t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the stunning blow!", target.name:capitalize())
 			end
@@ -244,7 +244,7 @@ newTalent{
 		return ([[Hits the target with your weapon doing %d%% damage. If the attack hits, the target is stunned for %d turns.
 		Stun chance increase with your Strength stat.]])
 		:format(100 * self:combatTalentWeaponDamage(t, 1, 1.5),
-		2 + self:getTalentLevel(t))
+		2 + math.floor(self:getTalentLevel(t)))
 	end,
 }
 
@@ -275,7 +275,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_SUNDER_ARMOUR, 4 + self:getTalentLevel(t), {power=5*self:getTalentLevel(t), apply_power=self:combatAttackStr(weapon.combat)})
+				target:setEffect(target.EFF_SUNDER_ARMOUR, 4 + self:getTalentLevel(t), {power=5*self:getTalentLevel(t), apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the sundering!", target.name:capitalize())
 			end
@@ -321,7 +321,7 @@ newTalent{
 		-- Try to stun !
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_SUNDER_ARMS, 4 + self:getTalentLevel(t), {power=3*self:getTalentLevel(t), apply_power=self:combatAttackStr(weapon.combat)})
+				target:setEffect(target.EFF_SUNDER_ARMS, 4 + self:getTalentLevel(t), {power=3*self:getTalentLevel(t), apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the sundering!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/techniques/archery.lua b/game/modules/tome/data/talents/techniques/archery.lua
index e4730d1366..7a62e69482 100644
--- a/game/modules/tome/data/talents/techniques/archery.lua
+++ b/game/modules/tome/data/talents/techniques/archery.lua
@@ -250,7 +250,7 @@ newTalent{
 	requires_target = true,
 	on_pre_use = function(self, t, silent) if not self:hasArcheryWeapon() then if not silent then game.logPlayer(self, "You require a bow or sling for this talent.") end return false end return true end,
 	archery_onhit = function(self, t, target, x, y)
-		target:setEffect(target.EFF_SLOW, 7, {power=util.bound((self:combatAttack() * 0.15 * self:getTalentLevel(t)) / 100, 0.1, 0.4), apply_power=self:combatAttackDex()})
+		target:setEffect(target.EFF_SLOW, 7, {power=util.bound((self:combatAttack() * 0.15 * self:getTalentLevel(t)) / 100, 0.1, 0.4), apply_power=self:combatAttack()})
 	end,
 	action = function(self, t)
 		local targets = self:archeryAcquireTargets(nil, {one_shot=true})
@@ -278,7 +278,7 @@ newTalent{
 	on_pre_use = function(self, t, silent) if not self:hasArcheryWeapon() then if not silent then game.logPlayer(self, "You require a bow or sling for this talent.") end return false end return true end,
 	archery_onhit = function(self, t, target, x, y)
 		if target:canBe("pin") then
-			target:setEffect(target.EFF_PINNED, 2 + self:getTalentLevelRaw(t), {apply_power=self:combatAttackDex()})
+			target:setEffect(target.EFF_PINNED, 2 + self:getTalentLevelRaw(t), {apply_power=self:combatAttack()})
 		else
 			game.logSeen(target, "%s resists!", target.name:capitalize())
 		end
@@ -318,7 +318,7 @@ newTalent{
 	on_pre_use = function(self, t, silent) if not self:hasArcheryWeapon() then if not silent then game.logPlayer(self, "You require a bow or sling for this talent.") end return false end return true end,
 	archery_onhit = function(self, t, target, x, y)
 		if target:canBe("stun") then
-			target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevelRaw(t), {apply_power=self:combatAttackDex()})
+			target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevelRaw(t), {apply_power=self:combatAttack()})
 		else
 			game.logSeen(target, "%s resists!", target.name:capitalize())
 		end
diff --git a/game/modules/tome/data/talents/techniques/battle-tactics.lua b/game/modules/tome/data/talents/techniques/battle-tactics.lua
index c772756348..abb242821f 100644
--- a/game/modules/tome/data/talents/techniques/battle-tactics.lua
+++ b/game/modules/tome/data/talents/techniques/battle-tactics.lua
@@ -79,7 +79,7 @@ newTalent{
 				dam = rng.range(dam, dam * damrange)
 				dam = dam * self:combatTalentWeaponDamage(t, 2, 3.2)
 
-				target:setEffect(target.EFF_DEEP_WOUND, 7, {src=self, heal_factor=self:getTalentLevel(t) * 10, power=dam / 7, apply_power=self:combatAttackDex()})
+				target:setEffect(target.EFF_DEEP_WOUND, 7, {src=self, heal_factor=self:getTalentLevel(t) * 10, power=dam / 7, apply_power=self:combatAttack()})
 			end
 		end
 		return true
diff --git a/game/modules/tome/data/talents/techniques/bloodthirst.lua b/game/modules/tome/data/talents/techniques/bloodthirst.lua
index 35771e493a..e5e3668a1b 100644
--- a/game/modules/tome/data/talents/techniques/bloodthirst.lua
+++ b/game/modules/tome/data/talents/techniques/bloodthirst.lua
@@ -49,7 +49,7 @@ newTalent{
 		weapon = weapon or target.combat
 
 		if target:canBe("stun") then
-			target:setEffect(target.EFF_DAZED, 5, {apply_power=self:combatAttackStr(weapon.combat)})
+			target:setEffect(target.EFF_DAZED, 5, {apply_power=self:combatPhysicalpower()})
 		else
 			game.logSeen(target, "%s resists the terror!", target.name:capitalize())
 		end
diff --git a/game/modules/tome/data/talents/techniques/bow.lua b/game/modules/tome/data/talents/techniques/bow.lua
index 79d773d37c..8a64949202 100644
--- a/game/modules/tome/data/talents/techniques/bow.lua
+++ b/game/modules/tome/data/talents/techniques/bow.lua
@@ -23,8 +23,13 @@ newTalent{
 	points = 10,
 	require = { stat = { dex=function(level) return 12 + level * 3 end }, },
 	mode = "passive",
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
-		return ([[Increases damage done with bows by %d%%.]]):format(100 * (math.sqrt(self:getTalentLevel(t) / 10)))
+		local damage = t.getDamage(self, t)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. Also increases damage done with bows by %d%%.]]):
+		format(damage, inc * 100)
 	end,
 }
 
diff --git a/game/modules/tome/data/talents/techniques/combat-training.lua b/game/modules/tome/data/talents/techniques/combat-training.lua
index fa741a62e3..60ad2e8771 100644
--- a/game/modules/tome/data/talents/techniques/combat-training.lua
+++ b/game/modules/tome/data/talents/techniques/combat-training.lua
@@ -102,11 +102,13 @@ newTalent{
 	points = 10,
 	require = { stat = { str=function(level) return 12 + level * 3 end }, },
 	mode = "passive",
-	getDamage = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) end,
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		return ([[Increases damage done with swords, maces and axes by %d%%.]]):
-		format(100 * damage)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. Also increases damage done with swords, axes, maces, knives, and exotic weapons by %d%%]]):
+		format(damage, 100*inc)
 	end,
 }
 
@@ -117,11 +119,13 @@ newTalent{
 	points = 10,
 	require = { stat = { dex=function(level) return 10 + level * 3 end }, },
 	mode = "passive",
-	getDamage = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) end,
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		return ([[Increases damage done with knives by %d%%.]]):
-		format(100 * damage)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. Also increases damage done with knives by %d%%]]):
+		format(damage, 100*inc)
 	end,
 }
 
@@ -132,10 +136,12 @@ newTalent{
 	points = 10,
 	require = { stat = { str=function(level) return 10 + level * 3 end, dex=function(level) return 10 + level * 3 end }, },
 	mode = "passive",
-	getDamage = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) end,
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		return ([[Increases damage done with exotic weapons (whips, tridents, ...) by %d%%.]]):
-		format(100 * damage)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. Also increases damage done with exotic weapons by %d%%]]):
+		format(damage, 100*inc)
 	end,
-}
+}
\ No newline at end of file
diff --git a/game/modules/tome/data/talents/techniques/dualweapon.lua b/game/modules/tome/data/talents/techniques/dualweapon.lua
index 415e97b296..c54ab12380 100644
--- a/game/modules/tome/data/talents/techniques/dualweapon.lua
+++ b/game/modules/tome/data/talents/techniques/dualweapon.lua
@@ -136,7 +136,7 @@ newTalent{
 		-- Second attack with mainhand
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevel(t), {apply_power=self:combatAttackDex(weapon.combat)})
+				target:setEffect(target.EFF_STUNNED, 2 + self:getTalentLevel(t), {apply_power=self:combatAttack()})
 			else
 				game.logSeen(target, "%s resists the stunning strike!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/techniques/field-control.lua b/game/modules/tome/data/talents/techniques/field-control.lua
index 171b1550f7..0f4e47c7b1 100644
--- a/game/modules/tome/data/talents/techniques/field-control.lua
+++ b/game/modules/tome/data/talents/techniques/field-control.lua
@@ -85,14 +85,17 @@ newTalent{
 
 		-- Try to knockback !
 		local can = function(target)
-			if target:checkHit(self:combatAttackDex(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then
+			if target:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("knockback") then
 				return true
 			else
 				game.logSeen(target, "%s resists the knockback!", target.name:capitalize())
 			end
 		end
 
-		if can(target) then target:knockback(self.x, self.y, math.floor(2 + self:getTalentLevel(t)), can) end
+		if can(target) then 
+			target:knockback(self.x, self.y, math.floor(2 + self:getTalentLevel(t)), can) 
+			target:crossTierEffect(target.EFF_OFFBALANCE, self:combatPhysicalpower())
+		end
 
 		return true
 	end,
diff --git a/game/modules/tome/data/talents/techniques/finishing-moves.lua b/game/modules/tome/data/talents/techniques/finishing-moves.lua
index 2577ad106f..2a68e66425 100644
--- a/game/modules/tome/data/talents/techniques/finishing-moves.lua
+++ b/game/modules/tome/data/talents/techniques/finishing-moves.lua
@@ -48,7 +48,7 @@ newTalent{
 
 		if hit then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, t.getDuration(self, t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_STUNNED, t.getDuration(self, t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the stun!", target.name:capitalize())
 			end
@@ -159,7 +159,7 @@ newTalent{
 		if hit then
 			-- try to daze
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_DAZED, t.getDuration(self, t), {apply_power=self:combatAttackStr()})
+				target:setEffect(target.EFF_DAZED, t.getDuration(self, t), {apply_power=self:combatPhysicalpower()})
 			else
 				game.logSeen(target, "%s resists the body shot!", target.name:capitalize())
 			end
@@ -218,7 +218,7 @@ newTalent{
 
 		-- Try to insta-kill
 		if hit then
-			if target:checkHit(self:combatAttackStr(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("instakill") and target.life > 0 and target.life < target.max_life * 0.2 then
+			if target:checkHit(self:combatPhysicalpower(), target:combatPhysicalResist(), 0, 95, 5 - self:getTalentLevel(t) / 2) and target:canBe("instakill") and target.life > 0 and target.life < target.max_life * 0.2 then
 				-- KILL IT !
 				game.logSeen(target, "%s feels the pain of the death blow!", target.name:capitalize())
 				target:die(self)
diff --git a/game/modules/tome/data/talents/techniques/grappling.lua b/game/modules/tome/data/talents/techniques/grappling.lua
index e7676b91ed..382313a1d0 100644
--- a/game/modules/tome/data/talents/techniques/grappling.lua
+++ b/game/modules/tome/data/talents/techniques/grappling.lua
@@ -168,10 +168,10 @@ newTalent{
 		-- deal damage and maim if appropriate
 		if hit then
 			if grappled then
-				self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getDamage(self, t), nil, target))
+				self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getDamage(self, t), nil, target, self:combatAttack(), target:combatDefense()))
 				target:setEffect(target.EFF_MAIMED, t.getDuration(self, t), {power=t.getMaim(self, t)})
 			else
-				self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getDamage(self, t), nil, target))
+				self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getDamage(self, t), nil, target, self:combatAttack(), target:combatDefense()))
 			end
 		end
 
@@ -264,16 +264,16 @@ newTalent{
 			-- takedown or slam as appropriate
 			if hit then
 				if grappled then
-					self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getSlam(self, t), nil, target))
+					self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getSlam(self, t), nil, target, self:combatAttack(), target:combatDefense()))
 					if target:canBe("stun") then
-						target:setEffect(target.EFF_STUNNED, t.getDuration(self, t), {apply_power=self:combatAttackStr()})
+						target:setEffect(target.EFF_STUNNED, t.getDuration(self, t), {apply_power=self:combatPhysicalpower()})
 					else
 						game.logSeen(target, "%s resists the stun!", target.name:capitalize())
 					end
 				else
-					self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getTakeDown(self, t), nil, target))
+					self:project(target, x, y, DamageType.PHYSICAL, self:physicalCrit(t.getTakeDown(self, t), nil, target, self:combatAttack(), target:combatDefense()))
 					if target:canBe("stun") then
-						target:setEffect(target.EFF_DAZED, t.getDuration(self, t), {apply_power=self:combatAttackStr()})
+						target:setEffect(target.EFF_DAZED, t.getDuration(self, t), {apply_power=self:combatPhysicalpower()})
 					else
 						game.logSeen(target, "%s resists the daze!", target.name:capitalize())
 					end
diff --git a/game/modules/tome/data/talents/techniques/sling.lua b/game/modules/tome/data/talents/techniques/sling.lua
index b43770857e..77f0ee9261 100644
--- a/game/modules/tome/data/talents/techniques/sling.lua
+++ b/game/modules/tome/data/talents/techniques/sling.lua
@@ -23,8 +23,13 @@ newTalent{
 	points = 10,
 	require = { stat = { dex=function(level) return 12 + level * 3 end }, },
 	mode = "passive",
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
-		return ([[Increases damage done with slings by %d%%.]]):format(100 * (math.sqrt(self:getTalentLevel(t) / 10)))
+		local damage = t.getDamage(self, t)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. Also increases damage done with slings by %d%%.]]):
+		format(damage, inc * 100)
 	end,
 }
 
@@ -42,7 +47,7 @@ newTalent{
 	on_pre_use = function(self, t, silent) if not self:hasArcheryWeapon("sling") then if not silent then game.logPlayer(self, "You require a sling for this talent.") end return false end return true end,
 	archery_onhit = function(self, t, target, x, y)
 		if target:canBe("blind") then
-			target:setEffect(target.EFF_BLINDED, 2 + self:getTalentLevelRaw(t), {apply_power=self:combatAttackDex()})
+			target:setEffect(target.EFF_BLINDED, 2 + self:getTalentLevelRaw(t), {apply_power=self:combatAttack()})
 		else
 			game.logSeen(target, "%s resists!", target.name:capitalize())
 		end
@@ -76,8 +81,9 @@ newTalent{
 	tactical = { ATTACK = 2, DISABLE = 2, ESCAPE = 1 },
 	on_pre_use = function(self, t, silent) if not self:hasArcheryWeapon("sling") then if not silent then game.logPlayer(self, "You require a sling for this talent.") end return false end return true end,
 	archery_onhit = function(self, t, target, x, y)
-		if target:checkHit(self:combatAttackDex(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
+		if target:checkHit(self:combatAttack(), target:combatPhysicalResist(), 0, 95, 15) and target:canBe("knockback") then
 			target:knockback(self.x, self.y, 4)
+			target:crossTierEffect(target.EFF_OFFBALANCE, self:combatAttack())
 			game.logSeen(target, "%s is knocked back!", target.name:capitalize())
 		else
 			game.logSeen(target, "%s stands firm!", target.name:capitalize())
diff --git a/game/modules/tome/data/talents/techniques/thuggery.lua b/game/modules/tome/data/talents/techniques/thuggery.lua
index d380e0cfe1..f68c6afad1 100644
--- a/game/modules/tome/data/talents/techniques/thuggery.lua
+++ b/game/modules/tome/data/talents/techniques/thuggery.lua
@@ -56,7 +56,7 @@ newTalent{
 
 		if hitted then
 			if target:canBe("confusion") then
-				target:setEffect(target.EFF_CONFUSED, t.getDuration(self, t), {power=30 + self:getDex(70), apply_power=self:combatAttackDex()})
+				target:setEffect(target.EFF_CONFUSED, t.getDuration(self, t), {power=30 + self:getDex(70), apply_power=self:combatAttack()})
 			else
 				game.logSeen(target, "%s resists the headblow!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/talents/techniques/unarmed-discipline.lua b/game/modules/tome/data/talents/techniques/unarmed-discipline.lua
index 01c0216750..616e1bda1e 100644
--- a/game/modules/tome/data/talents/techniques/unarmed-discipline.lua
+++ b/game/modules/tome/data/talents/techniques/unarmed-discipline.lua
@@ -91,7 +91,7 @@ newTalent{
 	do_throw = function(self, target, t)
 		local hit = self:checkHit(self:combatAttack(), target:combatDefense(), 0, 95, 5 - self:getTalentLevel(t) / 2)
 		if hit then
-			self:project(target, target.x, target.y, DamageType.PHYSICAL, self:physicalCrit(t.getDamageTwo(self, t), nil, target))
+			self:project(target, target.x, target.y, DamageType.PHYSICAL, self:physicalCrit(t.getDamageTwo(self, t), nil, target, self:combatAttack(), target:combatDefense()))
 			-- if grappled stun
 			if target:isGrappled(self) and target:canBe("stun") then
 				target:setEffect(target.EFF_STUNNED, 2, {apply_power=self:combatAttack(), min_dur=1})
diff --git a/game/modules/tome/data/talents/techniques/unarmed-training.lua b/game/modules/tome/data/talents/techniques/unarmed-training.lua
index e75dc040f9..ac37f2ff70 100644
--- a/game/modules/tome/data/talents/techniques/unarmed-training.lua
+++ b/game/modules/tome/data/talents/techniques/unarmed-training.lua
@@ -49,12 +49,14 @@ newTalent{
 	points = 10,
 	require = { stat = { cun=function(level) return 12 + level * 3 end }, },
 	mode = "passive",
-	getDamage = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) end,
+	getDamage = function(self, t) return self:getTalentLevel(t) * 5 end,
+	getPercentInc = function(self, t) return math.sqrt(self:getTalentLevel(t) / 10) / 2 end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		return ([[Increases damage done with unarmed attacks by %d%% 
-		Note this only applies to your base unarmed damage and things that work off of it (pugilism and finishing moves for instance) and will not increase grappling or kick damage.]]):
-		format(100 * damage)
+		local inc = t.getPercentInc(self, t)
+		return ([[Increases Physical Power by %d. 
+		Also increases damage done with unarmed attacks by %d%%. Note this only applies to your base unarmed damage and things that work off of it (pugilism and finishing moves for instance) and will not increase grappling or kick damage.]]):
+		format(damage, 100*inc)
 	end,
 }
 
diff --git a/game/modules/tome/data/talents/techniques/warcries.lua b/game/modules/tome/data/talents/techniques/warcries.lua
index 8f68003ae6..1f767e769c 100644
--- a/game/modules/tome/data/talents/techniques/warcries.lua
+++ b/game/modules/tome/data/talents/techniques/warcries.lua
@@ -108,7 +108,7 @@ newTalent{
 		self:project(tg, x, y, function(px, py)
 			local target = game.level.map(px, py, Map.ACTOR)
 			if not target then return end
-			target:setEffect(target.EFF_BATTLE_CRY, 7, {power=7 * self:getTalentLevel(t), apply_power=self:combatAttackStr()})
+			target:setEffect(target.EFF_BATTLE_CRY, 7, {power=7 * self:getTalentLevel(t), apply_power=self:combatPhysicalpower()})
 		end, {type="flame"})
 		return true
 	end,
diff --git a/game/modules/tome/data/talents/undeads/ghoul.lua b/game/modules/tome/data/talents/undeads/ghoul.lua
index 0d8ab7e5b5..2f0196a596 100644
--- a/game/modules/tome/data/talents/undeads/ghoul.lua
+++ b/game/modules/tome/data/talents/undeads/ghoul.lua
@@ -132,7 +132,7 @@ newTalent{
 
 		if hitted then
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_STUNNED, 3 + math.ceil(self:getTalentLevel(t)), {apply_power=self:combatAttackDex()})
+				target:setEffect(target.EFF_STUNNED, 3 + math.ceil(self:getTalentLevel(t)), {apply_power=self:combatAttack()})
 			else
 				game.logSeen(target, "%s resists the stun!", target.name:capitalize())
 			end
diff --git a/game/modules/tome/data/texts/tutorial/combat-stats-intro.lua b/game/modules/tome/data/texts/tutorial/combat-stats-intro.lua
new file mode 100644
index 0000000000..1061d3e727
--- /dev/null
+++ b/game/modules/tome/data/texts/tutorial/combat-stats-intro.lua
@@ -0,0 +1,23 @@
+-- ToME - Tales of Maj'Eyal
+-- Copyright (C) 2009, 2010, 2011 Nicolas Casalini
+--
+-- This program is free software: you can redistribute it and/or modify
+-- it under the terms of the GNU General Public License as published by
+-- the Free Software Foundation, either version 3 of the License, or
+-- (at your option) any later version.
+--
+-- This program is distributed in the hope that it will be useful,
+-- but WITHOUT ANY WARRANTY; without even the implied warranty of
+-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+-- GNU General Public License for more details.
+--
+-- You should have received a copy of the GNU General Public License
+-- along with this program.  If not, see <http://www.gnu.org/licenses/>.
+--
+-- Nicolas Casalini "DarkGod"
+-- darkgod@te4.org
+
+return [[
+Onward! The #GOLD#Dungeon of Adventurer Enlightenment#WHITE# awaits you!
+]]
+
diff --git a/game/modules/tome/data/timed_effects/magical.lua b/game/modules/tome/data/timed_effects/magical.lua
index d12119288c..c4923aecac 100644
--- a/game/modules/tome/data/timed_effects/magical.lua
+++ b/game/modules/tome/data/timed_effects/magical.lua
@@ -1406,3 +1406,24 @@ newEffect{
 		self:removeParticles(eff.particle)
 	end,
 }
+
+
+newEffect{
+	name = "SPELLSHOCKED",
+	desc = "Spellshocked",
+	long_desc = function(self, eff) return ("Overwhelming magic has temporarily interfered with all damage resistances, lowering them by 25%.") end,
+	type = "magical",
+	subtype = { cross_tier=true },
+	status = "detrimental",
+	parameters = { power=10 },
+	on_gain = function(self, err) return nil, "+Spellshocked" end,
+	on_lose = function(self, err) return nil, "-Spellshocked" end,
+	activate = function(self, eff)
+		eff.tmpid = self:addTemporaryValue("resists", {
+			all = -25,
+		})
+	end,
+	deactivate = function(self, eff)
+		self:removeTemporaryValue("resists", eff.tmpid)
+	end,
+}
\ No newline at end of file
diff --git a/game/modules/tome/data/timed_effects/mental.lua b/game/modules/tome/data/timed_effects/mental.lua
index bcbe57d428..5edf2dfb43 100644
--- a/game/modules/tome/data/timed_effects/mental.lua
+++ b/game/modules/tome/data/timed_effects/mental.lua
@@ -382,27 +382,27 @@ newEffect{
 	on_lose = function(self, err) return "#Target# is no longer beckoned.", "-Beckoned" end,
 	activate = function(self, eff)
 		eff.particle = self:addParticles(Particles.new("beckoned", 1))
-		
+
 		eff.spellpowerChangeId = self:addTemporaryValue("combat_spellpower", eff.spellpowerChange)
 		eff.mindpowerChangeId = self:addTemporaryValue("combat_mindpower", eff.mindpowerChange)
 	end,
 	deactivate = function(self, eff)
 		if eff.particle then self:removeParticles(eff.particle) end
-		
+
 		if eff.spellpowerChangeId then self:removeTemporaryValue("combat_spellpower", eff.spellpowerChangeId) end
 		if eff.mindpowerChangeId then self:removeTemporaryValue("combat_mindpower", eff.spellpowerChangeId) end
 	end,
 	on_timeout = function(self, eff)
 		if eff.source.dead then return nil end
-		
+
 		local distance = core.fov.distance(self.x, self.y, eff.source.x, eff.source.y)
 		if math.floor(distance) > 1 and distance <= eff.range then
 			-- in range but not adjacent
-			
+
 			-- add debuffs
 			if not eff.spellpowerChangeId then eff.spellpowerChangeId = self:addTemporaryValue("combat_spellpower", eff.spellpowerChange) end
 			if not eff.mindpowerChangeId then eff.mindpowerChangeId = self:addTemporaryValue("combat_mindpower", eff.mindpowerChange) end
-			
+
 			-- custom pull logic (adapted from move_dmap; forces movement, pushes others aside, custom particles)
 			if not self:attr("never_move") and rng.percent(eff.chance) then
 				local source = eff.source
@@ -421,7 +421,7 @@ newEffect{
 					--		moveX, moveY = util.coordAddDir(self.x, self.y, dir)
 					--	end
 					--end
-					
+
 					-- move a-star (far more useful than dmap)
 					local a = Astar.new(game.level.map, self)
 					local path = a:calc(self.x, self.y, source.x, source.y)
@@ -429,7 +429,7 @@ newEffect{
 						moveX, moveY = path[1].x, path[1].y
 					end
 				end
-				
+
 				if moveX and moveY then
 					local old_move_others, old_x, old_y = self.move_others, self.x, self.y
 					self.move_others = true
@@ -505,14 +505,14 @@ newEffect{
 		eff.neverMoveId = self:addTemporaryValue("never_move", 1)
 		eff.armorId = self:addTemporaryValue("combat_armor", eff.armorChange)
 		eff.defenseId = self:addTemporaryValue("combat_def", eff.armorChange)
-		
+
 		eff.particle = self:addParticles(Particles.new("dominated", 1))
 	end,
 	deactivate = function(self, eff)
 		self:removeTemporaryValue("never_move", eff.neverMoveId)
 		self:removeTemporaryValue("combat_armor", eff.armorId)
 		self:removeTemporaryValue("combat_def", eff.defenseId)
-		
+
 		self:removeParticles(eff.particle)
 	end,
 }
@@ -1461,4 +1461,32 @@ newEffect{
 		self:removeTemporaryValue("combat_spellpower", eff.spellid)
 		self:removeTemporaryValue("combat_mindpower", eff.mindid)
 	end,
-}
\ No newline at end of file
+}
+
+newEffect{
+	name = "BRAINLOCKED",
+	desc = "Brainlocked",
+	long_desc = function(self, eff) return ("Renders a random talent unavailable. No talents will cool down until the effect has worn off."):format() end,
+	type = "mental",
+	subtype = { cross_tier=true },
+	status = "detrimental",
+	parameters = {},
+	on_gain = function(self, err) return nil, "+Brainlocked" end,
+	on_lose = function(self, err) return nil, "-Brainlocked" end,
+	activate = function(self, eff)
+		eff.tcdid = self:addTemporaryValue("no_talents_cooldown", 1)
+		local tids = {}
+		for tid, lev in pairs(self.talents) do
+			local t = self:getTalentFromId(tid)
+			if t and not self.talents_cd[tid] and t.mode == "activated" and not t.innate then tids[#tids+1] = t end
+		end
+		for i = 1, 1 do
+			local t = rng.tableRemove(tids)
+			if not t then break end
+			self.talents_cd[t.id] = 1
+		end
+	end,
+	deactivate = function(self, eff)
+		self:removeTemporaryValue("no_talents_cooldown", eff.tmpid)
+	end,
+}
diff --git a/game/modules/tome/data/timed_effects/physical.lua b/game/modules/tome/data/timed_effects/physical.lua
index e8872a04fc..330f939c49 100644
--- a/game/modules/tome/data/timed_effects/physical.lua
+++ b/game/modules/tome/data/timed_effects/physical.lua
@@ -267,6 +267,7 @@ newEffect{
 	on_lose = function(self, err) return "#Target# is not stunned anymore.", "-Burning Shock" end,
 	activate = function(self, eff)
 		eff.tmpid = self:addTemporaryValue("stunned", 1)
+		eff.tcdid = self:addTemporaryValue("no_talents_cooldown", 1)
 		eff.speedid = self:addTemporaryValue("movement_speed", -0.5)
 
 		local tids = {}
@@ -285,6 +286,7 @@ newEffect{
 	end,
 	deactivate = function(self, eff)
 		self:removeTemporaryValue("stunned", eff.tmpid)
+		self:removeTemporaryValue("no_talents_cooldown", eff.tcdid)
 		self:removeTemporaryValue("movement_speed", eff.speedid)
 	end,
 }
@@ -301,6 +303,7 @@ newEffect{
 	on_lose = function(self, err) return "#Target# is not stunned anymore.", "-Stunned" end,
 	activate = function(self, eff)
 		eff.tmpid = self:addTemporaryValue("stunned", 1)
+		eff.tcdid = self:addTemporaryValue("no_talents_cooldown", 1)
 		eff.speedid = self:addTemporaryValue("movement_speed", -0.5)
 
 		local tids = {}
@@ -316,6 +319,7 @@ newEffect{
 	end,
 	deactivate = function(self, eff)
 		self:removeTemporaryValue("stunned", eff.tmpid)
+		self:removeTemporaryValue("no_talents_cooldown", eff.tcdid)
 		self:removeTemporaryValue("movement_speed", eff.speedid)
 	end,
 }
@@ -1642,4 +1646,16 @@ newEffect{
 	deactivate = function(self, eff)
 		self:removeTemporaryValue("combat_def", eff.defenseChangeId)
 	end,
-}
\ No newline at end of file
+}
+
+newEffect{
+	name = "OFFBALANCE",
+	desc = "Off-balance",
+	long_desc = function(self, eff) return ("Badly off balance. Attackers gain a 25% bonus to critical strike power.") end,
+	type = "physical",
+	subtype = { cross_tier=true },
+	status = "detrimental",
+	parameters = {power = 1},
+	on_gain = function(self, err) return nil, "+Off-balance" end,
+	on_lose = function(self, err) return nil, "-Off-balance" end,
+}
-- 
GitLab