From c6666417ae48fc8f3b6cc9198b7c37ab849a0d15 Mon Sep 17 00:00:00 2001
From: Shibari <ShibariTOME@Gmail.com>
Date: Sun, 2 Jul 2017 01:24:27 -0400
Subject: [PATCH] Standardize and add more information to some of the
 inscription short infos

Number changes

Redesign Biting Gale and Acid Wave some

Finalize Mirror Image code

Redesign Primal Infusion

Lower NPC max attack rune count to 1

Move Shatter Afflictions and Wild to utility instead of protect

Update AI tactical tables for inscriptions
---
 .../tome/data/general/objects/scrolls.lua     |  58 +++---
 .../data/general/objects/world-artifacts.lua  |  21 +--
 .../tome/data/talents/misc/inscriptions.lua   | 173 +++++++-----------
 .../tome/data/timed_effects/physical.lua      |  13 +-
 game/modules/tome/resolvers.lua               |   2 +-
 5 files changed, 113 insertions(+), 154 deletions(-)

diff --git a/game/modules/tome/data/general/objects/scrolls.lua b/game/modules/tome/data/general/objects/scrolls.lua
index b5757dfca8..aebc4c483f 100644
--- a/game/modules/tome/data/general/objects/scrolls.lua
+++ b/game/modules/tome/data/general/objects/scrolls.lua
@@ -138,9 +138,9 @@ newEntity{ base = "BASE_INFUSION",
 
 	inscription_kind = "heal",
 	inscription_data = {
-		cooldown = resolvers.rngrange(7, 13),
-		heal = resolvers.mbonus_level(80, 20, function(e, v) return v * 0.06 end),
-		use_stat_mod = 2.4,
+		cooldown = resolvers.rngrange(9, 13),
+		heal = resolvers.mbonus_level(80, 40, function(e, v) return v * 0.06 end),
+		use_stat_mod = 2,
 	},
 	inscription_talent = "INFUSION:_HEALING",
 }
@@ -167,7 +167,7 @@ newEntity{ base = "BASE_INFUSION",
 	rarity = 15,
 	cost = 20,
 	chance = resolvers.mbonus_level(80, -20), -- No chance of 2 cleanses until higher ilvl to discourage rerolling the earliest shops
-	inscription_kind = "protect",
+	inscription_kind = "utility",
 	inscription_data = resolvers.generic(function(e)
 		local what = {}
 		local effects = {physical=true, mental=true, magical=true}
@@ -179,10 +179,10 @@ newEntity{ base = "BASE_INFUSION",
 			what[eff2] = true
 		end
 		return {
-			cooldown = rng.range(12, 17),
-			dur = rng.mbonus(4, resolvers.current_level, resolvers.mbonus_max_level) + 4,
-			power = rng.mbonus(20, resolvers.current_level, resolvers.mbonus_max_level) + 10,
-			use_stat_mod = 0.1,
+			cooldown = rng.range(12, 16),
+			dur = rng.range(2, 4),
+			power = resolvers.mbonus_level(10, 15),  -- Low variance because duration and chance for second debuff type is enough randomness
+			use_stat_mod = 0.2,  -- +20% resist all at 100 stat
 			what=what,
 		}
 	end),
@@ -197,7 +197,7 @@ newEntity{ base = "BASE_INFUSION",
 
 	inscription_kind = "movement",
 	inscription_data = {
-		cooldown = resolvers.rngrange(13, 20),
+		cooldown = resolvers.rngrange(13, 20),  -- High variance because this is the only really important stat
 		speed = resolvers.mbonus_level(500, 300, function(e, v) return v * 0.001 end),
 		use_stat_mod = 3,
 	},
@@ -226,12 +226,12 @@ newEntity{ base = "BASE_INFUSION",
 newEntity{ base = "BASE_RUNE",
 	name = "teleportation rune",
 	level_range = {1, 50},
-	rarity = 35, -- Very rare because item quality has little impact on this 
+	rarity = 50, -- Very rare because item quality has little impact on this
 	cost = 10,
 
 	inscription_kind = "teleport",
 	inscription_data = {
-		cooldown = resolvers.rngrange(14, 19),
+		cooldown = resolvers.rngrange(10, 20),  -- High variance because this is the only really important stat
 		range = resolvers.mbonus_level(100, 20, function(e, v) return v * 0.03 end),
 		use_stat_mod = 1,
 	},
@@ -254,26 +254,22 @@ newEntity{ base = "BASE_RUNE",
 	inscription_talent = "RUNE:_SHIELDING",
 }
 
--- Very strong debuff, very low damage and scaling
--- This is mostly aimed at classes that want a decent way to land their stuns more reliably
 newEntity{ base = "BASE_RUNE",
 	name = "biting gale rune",
-	level_range = {10, 50},
+	level_range = {20, 50},
 	rarity = 20,
 	cost = 20,
 
 	inscription_kind = "attack",
 	inscription_data = {
-		cooldown = resolvers.rngrange(15, 22),
-		power = resolvers.mbonus_level(100, 30, function(e, v) return v * 0.1 end),
-		apply = resolvers.mbonus_level(5, 20, function(e, v) return v * 0.1 end),
-		radius = 6,
-		use_stat_mod = 1.2,
+		cooldown = resolvers.rngrange(15, 19),
+		power = resolvers.mbonus_level(200, 30, function(e, v) return v * 0.1 end),
+		dur = 5,
+		use_stat_mod = 2.2,
 	},
 	inscription_talent = "RUNE:_BITING_GALE",
 }
 
--- Weaker debuff, medium damage and scaling
 newEntity{ base = "BASE_RUNE",
 	name = "acid wave rune",
 	level_range = {20, 50},
@@ -282,11 +278,9 @@ newEntity{ base = "BASE_RUNE",
 
 	inscription_kind = "attack",
 	inscription_data = {
-		cooldown = resolvers.rngrange(15, 25),
-		radius = 6,
-		power = resolvers.mbonus_level(250, 40, function(e, v) return v * 0.1 end),
-		apply = resolvers.mbonus_level(5, 20, function(e, v) return v * 0.1 end),
-		use_stat_mod = 1.2,
+		cooldown = resolvers.rngrange(15, 19),
+		power = resolvers.mbonus_level(200, 30, function(e, v) return v * 0.1 end),
+		use_stat_mod = 2.2,
 		dur = 5,
 	},
 	inscription_talent = "RUNE:_ACID_WAVE",
@@ -295,7 +289,7 @@ newEntity{ base = "BASE_RUNE",
 newEntity{ base = "BASE_RUNE",
 	name = "manasurge rune",
 	level_range = {1, 50},
-	rarity = 25,
+	rarity = 50,  -- Very rare because only a limited number of classes have any use for this
 	cost = 10,
 
 	inscription_kind = "utility",
@@ -326,7 +320,7 @@ newEntity{ base = "BASE_RUNE",
 
 newEntity{ base = "BASE_RUNE",
 	name = "stormshield rune",
-	level_range = {20, 50},
+	level_range = {30, 50},
 	rarity = 25,
 	cost = 20,
 	inscription_kind = "protect",
@@ -345,10 +339,10 @@ newEntity{ base = "BASE_RUNE",
 	level_range = {1, 50},
 	rarity = 15,
 	cost = 10,
-	inscription_kind = "protect",
+	inscription_kind = "utility",
 	inscription_data = {
-		cooldown = resolvers.rngrange(14, 19),
-		shield = resolvers.mbonus_level(100, 50, function(e, v) return v * 0.06 end),
+		cooldown = resolvers.rngrange(16, 20),
+		shield = resolvers.mbonus_level(100, 20, function(e, v) return v * 0.06 end),
 		use_stat_mod = 1 -- 1x, applied up to 3 times
 	},
 	inscription_talent = "RUNE:_SHATTER_AFFLICTIONS",
@@ -362,8 +356,8 @@ newEntity{ base = "BASE_RUNE",
 	inscription_kind = "protect",
 	inscription_data = {
 		cooldown = resolvers.rngrange(16, 22),
-		dur = resolvers.mbonus_level(5, 3),
-		power = resolvers.mbonus_level(8, 7, function(e, v) return v * 1 end),
+		dur = 5,
+		power = resolvers.mbonus_level(20, 7, function(e, v) return v * 1 end),
 		resist = resolvers.mbonus_level(30, 10),
 		move = resolvers.mbonus_level(20, 10),
 		reduction = 0.5,
diff --git a/game/modules/tome/data/general/objects/world-artifacts.lua b/game/modules/tome/data/general/objects/world-artifacts.lua
index ae5b1b5df3..77a8de045a 100644
--- a/game/modules/tome/data/general/objects/world-artifacts.lua
+++ b/game/modules/tome/data/general/objects/world-artifacts.lua
@@ -65,30 +65,26 @@ newEntity{ base = "BASE_GEM",
 	},
 }
 
--- Low base values because you can stack affinity and resist
--- The 3rd type is pretty meaningless balance-wise.  Magic debuffs hardly matter.  The real advantage is the affinity.
 newEntity{ base = "BASE_INFUSION",
 	name = "Primal Infusion", unique=true, image = "object/artifact/primal_infusion.png",
 	desc = [[This wild infusion has evolved.]],
 	unided_name = "pulsing infusion",
-	level_range = {15, 40},
+	level_range = {15, 50},
 	rarity = 300,
-	cost = 300,
-	material_level = 3,
+	cost = 500,
 
 	inscription_kind = "protect",
 	inscription_data = {
 		cooldown = 18,
-		dur = 6,
+		dur = 5,
+		reduce = 1,
 		power = 10,
-		use_stat_mod = 0.1, 
-		use_stat = "con",
-		what = {physical=true, mental=true, magical=true},
+		use_stat_mod = 0.02, -- +2 duration reduction and +20% affinity at 100 stat
+		use_stat = "wil",
 	},
 	inscription_talent = "INFUSION:_PRIMAL",
 }
 
--- Unique but generated randomly without any link to tier/level
 newEntity{ base = "BASE_RUNE",
 	name = "Prismatic Rune", unique=true, define_as="RUNE_PRISMATIC",
 	level_range = {5, 50},
@@ -102,7 +98,7 @@ newEntity{ base = "BASE_RUNE",
 		num_types = resolvers.rngrange(3, 5),
 		wards = {},
 		resolvers.genericlast(function(e)
-			e.inscription_data.wards["PHYSICAL"] = resolvers.rngrange(3, 5) -- guarantee physical wards
+			e.inscription_data.wards["PHYSICAL"] = resolvers.rngrange(2, 4) -- guarantee physical wards
 			for _ = 1,e.inscription_data.num_types do
 				local pick = rng.tableRemove(e.types)
 				e.inscription_data.wards[pick] = resolvers.rngrange(3, 5)
@@ -112,7 +108,6 @@ newEntity{ base = "BASE_RUNE",
 	inscription_talent = "RUNE:_PRISMATIC",
 }
 
--- Unique but generated randomly without any link to tier/level
 newEntity{ base = "BASE_RUNE",
 	name = "Mirror Image Rune", unique=true, define_as="RUNE_MIRROR_IMAGE",
 	level_range = {5, 50},
@@ -120,7 +115,7 @@ newEntity{ base = "BASE_RUNE",
 	cost = 500,
 	inscription_kind = "protect",
 	inscription_data = {
-		cooldown = 18,
+		cooldown = 24,
 		dur = 6,
 		inheritance = 1,
 	},
diff --git a/game/modules/tome/data/talents/misc/inscriptions.lua b/game/modules/tome/data/talents/misc/inscriptions.lua
index 611fc44469..bdde843ad5 100644
--- a/game/modules/tome/data/talents/misc/inscriptions.lua
+++ b/game/modules/tome/data/talents/misc/inscriptions.lua
@@ -126,7 +126,7 @@ newInscription{
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		return ([[heal %d]]):format(data.heal + data.inc_stat)
+		return ([[heal %d; %d cd]]):format(data.heal + data.inc_stat, data.cooldown)
 	end,
 }
 
@@ -166,57 +166,34 @@ newInscription{
 		local what = table.concatNice(table.keys(data.what), ", ", " and ")
 
 		return ([[Activate the infusion to cure yourself of one random %s effect and reduce all damage taken by %d%% for %d turns.
-
 Also removes cross-tier effects of the affected types for free.]]):format(what, data.power+data.inc_stat, data.dur)
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
 		local what = table.concat(table.keys(data.what), ", ")
-		return ([[resist %d%%; cure %s]]):format(data.power + data.inc_stat, what)
+		return ([[resist %d%%; cure %s; dur %d]]):format(data.power + data.inc_stat, what, data.dur)
 	end,
 }
 
--- fixedart wild variant
--- Needs update with 1.6
 newInscription{
 	name = "Infusion: Primal", image = "talents/infusion__wild.png",
 	type = {"inscriptions/infusions", 1},
 	points = 1,
 	no_energy = true,
-	tactical = {
-		DEFEND = 2,
-		CURE = function(self, t, target)
-			local data = self:getInscriptionData(t.short_name)
-			return #self:effectsFilter({types=data.what, status="detrimental"})
-		end
-	},
+	tactical = {DEFEND = 2, CURE = 2},
 	action = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-
-		local target = self
-		local effs = {}
-		local force = {}
-		local removed = 0
-
-		removed = target:removeEffectsFilter({types=data.what, subtype={["cross tier"] = true}, status="detrimental"})
-		removed = removed + target:removeEffectsFilter({types=data.what, status="detrimental"}, 1)
-		if removed > 0 then
-			game.logSeen(self, "%s is cured!", self.name:capitalize())
-		end
-		self:setEffect(self.EFF_PRIMAL_ATTUNEMENT, data.dur, {power=data.power + data.inc_stat})
+		self:setEffect(self.EFF_PRIMAL_ATTUNEMENT, data.dur, {power=data.power + data.inc_stat*10, reduce=data.reduce + data.inc_stat})
 		return true
 	end,
 	info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		local what = table.concatNice(table.keys(data.what), ", ", " or ")
-
-		return ([[Activate the infusion to cure yourself of one random %s effect and increase affinity for all damage by %d%% (scales with Constitution) for %d turns.
-Also removes cross-tier effects of the affected types for free.]]):format(what, data.power+data.inc_stat, data.dur)
+		return ([[Activate the infusion to heal for %d%% of all damage taken (calculated before resistances) and reduce the duration of a random debuff by %d each turn for %d turns.]]):
+			format(data.power+data.inc_stat*10, data.reduce + data.inc_stat, data.dur)
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		local what = table.concat(table.keys(data.what), ", ")
-		return ([[affinity %d%%; cure %s]]):format(data.power + data.inc_stat, what)
+		return ([[affinity %d%%; reduction %d]]):format(data.power + data.inc_stat*10, data.reduce + data.inc_stat)
 	end,
 }
 
@@ -225,7 +202,7 @@ newInscription{
 	type = {"inscriptions/infusions", 1},
 	points = 1,
 	no_energy = true,
-	tactical = { DEFEND = 1, ESCAPE = 1, CLOSEIN = 1 },
+	tactical = { ESCAPE = 1, CLOSEIN = 1 },
 	on_pre_use = function(self, t) return not self:attr("never_move") end,
 	action = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
@@ -250,7 +227,7 @@ newInscription{
 	type = {"inscriptions/infusions", 1},
 	points = 1,
 	no_energy = true,
-	tactical = { BUFF = 1, DEFEND = 1 },
+	tactical = { DEFEND = 1 },
 	action = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
 		self:setEffect(self.EFF_HEROISM, data.dur, {die_at=data.die_at + data.inc_stat * 30})
@@ -264,7 +241,7 @@ newInscription{
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		return ([[%d turns, die at -%d]]):format(data.dur, data.die_at + data.inc_stat * 30)
+		return ([[%d turns; die at -%d; %d cd]]):format(data.dur, data.die_at + data.inc_stat * 30, data.cooldown)
 	end,
 }
 
@@ -317,7 +294,6 @@ local function attack_rune(self, btid)
 	end
 end
 
--- Activate a second time to return?
 newInscription{
 	name = "Rune: Teleportation",
 	type = {"inscriptions/runes", 1},
@@ -364,7 +340,7 @@ newInscription{
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		return ([[absorb %d for %d turns]]):format(data.power + data.inc_stat, data.dur)
+		return ([[absorb %d; %d turns; %d cd]]):format(data.power + data.inc_stat, data.dur, data.cooldown)
 	end,
 }
 
@@ -396,7 +372,7 @@ newInscription{
 		local data = self:getInscriptionData(t.short_name)
 		local power = 100+5*self:getMag()
 		if data.power and data.inc_stat then power = data.power + data.inc_stat end
-		return ([[absorb and reflect %d for %d turns]]):format(power, data.dur or 5)
+		return ([[absorb and reflect %d; %d turns]]):format(power, data.dur or 5)
 	end,
 }
 
@@ -407,18 +383,10 @@ newInscription{
 	is_attack_rune = true,
 	no_energy = true,
 	is_spell = true,
-	tactical = { ATTACK = { COLD = 1 }, DISABLE = { stun = 1 }, CURE = function(self, t, target)
-			local nb = 0
-			local data = self:getInscriptionData(t.short_name)
-			for eff_id, p in pairs(self.tmp) do
-				local e = self.tempeffect_def[eff_id]
-				if e.type == "mental" and e.status == "detrimental" then nb = nb + 1 end
-			end
-			return nb
-		end },
+	tactical = { ATTACK = { COLD = 1 }, DISABLE = { stun = 1 } },
 	requires_target = true,
-	range = 0,
 	radius = 6,
+	range = 0,
 	target = function(self, t)
 		return {type="cone", cone_angle=25, radius = self:getTalentRadius(t), range=self:getTalentRange(t), talent=t, display={particle="bolt_ice", trail="icetrail"}}
 	end,
@@ -428,38 +396,31 @@ newInscription{
 		local x, y = self:getTarget(tg)
 		if not x or not y then return nil end
 
-		local damage = data.power + data.inc_stat -- Cut by ~2/3rds or so
-		local apply = self:rescaleCombatStats((data.apply + data.inc_stat))
-
-	--	local apply = data.apply + data.inc_stat -- Same calculation as Sun Infusion, goes above what PCs can get on power stats pretty easily
+		local damage = data.power + data.inc_stat
 		self:project(tg, x, y, function(tx, ty)
 			local target = game.level.map(tx, ty, Map.ACTOR)
 			if not target or target == self then return end
 			
-			-- Minor damage, apply stun resist reduction, freeze
 			DamageType:get(DamageType.COLD).projector(self, tx, ty, DamageType.COLD, damage)
-			target:setEffect(target.EFF_WET, 5, {apply_power=data.inc_stat})
+			target:setEffect(target.EFF_WET, 5, {})
 			if target:canBe("stun") then
-				target:setEffect(target.EFF_FROZEN, 2, {hp=damage*1.5, apply_power=apply})
+				target:setEffect(target.EFF_FROZEN, data.dur, {hp=damage*2})
 			end
 		end, data.power + data.inc_stat, {type="freeze"})
-		self:removeEffectsFilter({status="detrimental", type="mental", ignore_crosstier=true}, 1)
 		game:playSoundNear(self, "talents/ice")
 		attack_rune(self, t.id)
 		return true
 	end,
 	info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		local apply = self:rescaleCombatStats((data.apply + data.inc_stat))
 		return ([[Activate the rune to direct a cone of chilling stormwind doing %0.2f cold damage.
-			The storm will soak enemies hit reducing their resistance to stuns by 50%% then attempt to freeze them for 3 turns (with power %d vs. physical save).
-		The deep cold also crystalizes your mind, removing one random detrimental mental effect from you.]]):
-			format(damDesc(self, DamageType.COLD, data.power + data.inc_stat), apply)
+			The storm will soak enemies hit reducing their resistance to stuns by 50%% then attempt to freeze them for %d.
+			These effects can be resisted but not saved against.]]):
+			format(damDesc(self, DamageType.COLD, data.power + data.inc_stat), data.dur)
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		local apply = self:rescaleCombatStats((data.apply + data.inc_stat))
-		return ([[%d cold damage; freeze 3 turns with power %d]]):format(damDesc(self, DamageType.COLD, data.power + data.inc_stat), apply)
+		return ([[%d damage; %d turns]]):format(damDesc(self, DamageType.COLD, data.power + data.inc_stat), data.dur)
 	end,
 }
 
@@ -472,25 +433,14 @@ newInscription{
 	is_spell = true,
 	tactical = {
 		ATTACKAREA = { ACID = 1 },
-		CURE = function(self, t, target)
-			local nb = 0
-			local data = self:getInscriptionData(t.short_name)
-			for eff_id, p in pairs(self.tmp) do
-				local e = self.tempeffect_def[eff_id]
-				if e.type == "magical" and e.status == "detrimental" then nb = nb + 1 end
-			end
-			return nb
-		end
+		DISABLE = { disarm = 1 }
 	},
 	requires_target = true,
 	direct_hit = true,
+	radius = 6,
 	range = 0,
-	radius = function(self, t)
-		local data = self:getInscriptionData(t.short_name)
-		return data.radius
-	end,
 	target = function(self, t)
-		return {type="cone", radius=self:getTalentRadius(t), range = 0, selffire=false, cone_angle=5, talent=t}
+		return {type="cone", radius=self:getTalentRadius(t), range=self:getTalentRange(t), selffire=false, cone_angle=25, talent=t}
 	end,
 	action = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
@@ -498,19 +448,14 @@ newInscription{
 		local x, y = self:getTarget(tg)
 		if not x or not y then return nil end
 
-		local apply = self:rescaleCombatStats((data.apply + data.inc_stat))
-
-		self:removeEffectsFilter({status="detrimental", type="magical", ignore_crosstier=true}, 1)
 		self:project(tg, x, y, function(tx, ty)
 			local target = game.level.map(tx, ty, Map.ACTOR)
 			if not target or target == self then return end
 
 			if target:canBe("disarm") then
-				target:setEffect(target.EFF_DISARMED, data.dur, {apply_power=apply})
+				target:setEffect(target.EFF_DISARMED, data.dur, {})
 			end
-			
 			DamageType:get(DamageType.ACID).projector(self, tx, ty, DamageType.ACID, data.power + data.inc_stat)
-
 		end)
 
 		game.level.map:particleEmitter(self.x, self.y, tg.radius, "breath_acid", {radius=tg.radius, tx=x-self.x, ty=y-self.y})
@@ -520,18 +465,15 @@ newInscription{
 	end,
 	info = function(self, t)
 		  local data = self:getInscriptionData(t.short_name)
-		  local pow = data.apply + data.inc_stat
-		  local apply = self:rescaleCombatStats((data.apply + data.inc_stat))
-		  return ([[Activate the rune to unleash a wave of acid in a cone of radius %d, doing %0.2f acid damage. The corrosive acid will also disarm enemies struck for %d turns (with power %d vs. physical save).
-	  The surge of natural acids will remove one detrimental magical effect from you.]]):
-			 format(self:getTalentRadius(t), damDesc(self, DamageType.ACID, data.power + data.inc_stat), data.dur or 3, apply)
+		  return ([[Activate the rune to unleash a cone dealing %0.2f acid damage.
+			The corrosive acid will also disarm enemies struck for %d turns.
+			This effect can be resisted but not saved against.]]):
+			format(damDesc(self, DamageType.ACID, data.power + data.inc_stat), data.dur or 3)
 	   end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
 		local pow = data.power
-		local apply = self:rescaleCombatStats((data.apply + data.inc_stat))
-
-		return ([[%d acid damage; disarm %d turns with power %d]]):format(damDesc(self, DamageType.ACID, data.power + data.inc_stat), data.dur or 3, apply)
+		return ([[%d damage; %d turns]]):format(damDesc(self, DamageType.ACID, data.power + data.inc_stat), data.dur or 3)
 	end,
 }
 
@@ -684,11 +626,17 @@ newInscription{
 	points = 1,
 	is_spell = true,
 	is_teleport = true,
-	tactical = { CLOSEIN = 2 },
+	tactical = {  ESCAPE = 1, CLOSEIN = 1 },
+	range = function(self, t)
+		local data = self:getInscriptionData(t.short_name)
+		return data.range + data.inc_stat
+	end,
+	target = function(self, t) return {type="hit", nolock=true, pass_terrain=false, nowarning=true, range=t.range(self, t), 
+		grid_params = {want_range = (not self.ai_target.actor or self.ai_state.tactic == "escape") and 6 or 1	} } end,
 	getDur = function(self, t) return 3 end,
 	action = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		local tg = {type="beam", nolock=true, pass_terrain=false, nowarning=true, range=data.range + data.inc_stat, radius=1, requires_knowledge=false}
+		local tg = self:getTalentTarget(t)
 		local x, y = self:getTarget(tg)
 		if not x then return end
 		if not self:hasLOS(x, y) then return end
@@ -728,7 +676,7 @@ newInscription{
 	points = 1,
 	is_spell = true,
 	no_energy = true,
-	--tactical = { DEFEND = 3, ESCAPE = 2 },
+	tactical = { ESCAPE = 1 },
 	getDur = function(self, t) return 5 end,
 	getResistance = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
@@ -758,7 +706,7 @@ newInscription{
 			format(t.getDur(self, t),t.getReduction(self, t) * 100, t.getResistance(self, t), t.getMove(self, t), t.getPower(self, t))
 	end,
 	short_info = function(self, t)
-		return ([[power %d, resist %d%%, move %d%%, %d turns]]):format(t.getPower(self, t), t.getResistance(self, t), t.getMove(self, t), t.getDur(self, t))
+		return ([[power %d; resist %d%%; move %d%%; %d turns]]):format(t.getPower(self, t), t.getResistance(self, t), t.getMove(self, t), t.getDur(self, t))
 	end,
 }
 
@@ -770,7 +718,8 @@ newInscription{
 	type = {"inscriptions/runes", 1},
 	points = 1,
 	is_spell = true,
-	tactical = { DEFEND = 3 },
+	no_energy = true,
+	tactical = { DEFEND = 1 },
 	getDur = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
 		return data.dur
@@ -795,7 +744,7 @@ newInscription{
 				:format(t.getDur(self, t), t.getThreshold(self, t), t.getBlocks(self, t) )
 	end,
 	short_info = function(self, t)
-		return ([[threshold %d, blocks %d, %d turns]]):format(t.getThreshold(self, t), t.getBlocks(self, t), t.getDur(self, t) )
+		return ([[threshold %d; blocks %d; %d turns]]):format(t.getThreshold(self, t), t.getBlocks(self, t), t.getDur(self, t) )
 	end,
 }
 
@@ -829,7 +778,7 @@ newInscription{
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
 		local str = table.concat(table.keys(data.wards), ", ")
-		return ([[%d turns, %s]]):format(t.getDur(self, t), str:lower() )
+		return ([[%d turns; %s]]):format(t.getDur(self, t), str:lower() )
 	end,
 }
 
@@ -840,7 +789,7 @@ newInscription{
 	image = "talents/phase_shift.png",  -- re-used icon
 	points = 1,
 	is_spell = true,
-	--tactical = { DEFEND = 3, ESCAPE = 2 },
+	no_npc_use, -- You can't taunt players
 	getDur = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
 		return data.dur
@@ -879,9 +828,12 @@ newInscription{
 						local image = NPC.new{
 							name = "Mirror Image",
 							type = "image", subtype = "image",
+							ai = "summoned", ai_real = nil, ai_state = { talent_in=1, }, ai_target = {actor=nil},
 							desc = "A blurred image.",
 							image = caster.image,
-							add_mos = caster.add_mos, -- this is horribly wrong isn't it?
+							add_mos = caster.add_mos, -- this is horribly wrong isn't it?  seems to work though
+							shader = "shadow_simulacrum", shader_args = { color = {0.0, 0.4, 0.8}, base = 0.6, time_factor = 1500 },
+							exp_worth=0,
 							max_life = caster.max_life,
 							life = caster.max_life, -- We don't want to make this only useful before you take damage
 							combat_armor_hardiness = caster:combatArmorHardiness(),
@@ -893,17 +845,18 @@ newInscription{
 							life_rating = 0,
 							cant_be_moved = 1,
 							never_move = 1,
+							never_anger = true,
 							resolvers.talents{
 								[Talents.T_TAUNT]=1, -- Add the talent so the player can see it even though we cast it manually
 							},
-							on_act = function(self) -- avoid any interaction with .. uh, anything, for now
-								self:forceUseTalent(self.T_TAUNT, {ignore_cd=true, ignore_energy=true})
+							on_act = function(self) -- avoid any interaction with .. uh, anything
+								self:forceUseTalent(self.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
 							end,
 							faction = caster.faction,
-							summoner = caster, summoner_gain_exp=true,
+							summoner = caster,
 							summon_time=t.getDur(self, t),
 							no_breath = 1,
-							remove_from_party_on_death = true
+							remove_from_party_on_death = true,
 						}
 
 						image:resolve()
@@ -912,8 +865,11 @@ newInscription{
 							control=false,
 							type="summon",
 							title="Summon",
-							orders = {target=true, leash=true, anchor=true, talents=true},
+							temporary_level = true,
+							orders = {},
 						})
+
+						image:forceUseTalent(image.T_TAUNT, {})
 					end
 				end
 			end
@@ -938,7 +894,14 @@ newInscription{
 	image = "talents/warp_mine_away.png", -- re-used icon
 	type = {"inscriptions/runes", 1},
 	points = 1,
-	tactical = { },
+	tactical = { CURE = function(self, t, target)
+		local types = 0
+		types = types + #self:effectsFilter({status="detrimental", type="physical"}, 1)
+		types = types + #self:effectsFilter({status="detrimental", type="magical"}, 1)
+		types = types + #self:effectsFilter({status="detrimental", type="mental"}, 1)
+		return types
+	end
+	},
 	is_spell = true,
 	no_energy = true,
 	getShield = function(self, t)
@@ -964,7 +927,7 @@ newInscription{
 	end,
 	short_info = function(self, t)
 		local data = self:getInscriptionData(t.short_name)
-		return ([[cooldown %d, shield %d ]]):format(data.cooldown, data.shield + data.inc_stat)
+		return ([[cooldown %d; absorb %d ]]):format(data.cooldown, data.shield + data.inc_stat)
 	end,
 }
 
@@ -974,7 +937,7 @@ newInscription{
 	type = {"inscriptions/runes", 1},
 	points = 1,
 	is_spell = true,
-	tactical = {},
+	no_npc_use = true, -- Quest reward
 	range = 10,
 	direct_hit = true,
 	action = function(self, t)
diff --git a/game/modules/tome/data/timed_effects/physical.lua b/game/modules/tome/data/timed_effects/physical.lua
index 91f0879def..9090b5ee00 100644
--- a/game/modules/tome/data/timed_effects/physical.lua
+++ b/game/modules/tome/data/timed_effects/physical.lua
@@ -854,15 +854,14 @@ newEffect{
 	end,
 }
 
--- artifact wild infusion
 newEffect{
 	name = "PRIMAL_ATTUNEMENT", image = "talents/infusion__wild.png",
 	desc = "Primal Attunement",
-	long_desc = function(self, eff) return ("The target is attuned to the wild, increasing all damage affinity by %d%%."):format(eff.power) end,
+	long_desc = function(self, eff) return ("The target is attuned to the wild, increasing all damage affinity by %d%% and reducing a random debuff duration by %d%%."):format(eff.power, eff.reduce) end,
 	type = "physical",
 	subtype = { nature=true },
 	status = "beneficial",
-	parameters = { power=20 },
+	parameters = { power=20, reduce = 3 },
 	on_gain = function(self, err) return "#Target# attunes to the wild.", "+Primal" end,
 	on_lose = function(self, err) return "#Target# is no longer one with nature.", "-Primal" end,
 	activate = function(self, eff)
@@ -871,6 +870,14 @@ newEffect{
 	deactivate = function(self, eff)
 		self:removeTemporaryValue("damage_affinity", eff.pid)
 	end,
+	on_timeout = function(self, eff)
+		local effs = self:effectsFilter({status = "detrimental", ignore_crosstier = true}, 1)
+		local eff2 = self:hasEffect(effs[1])
+		if eff2 then 
+			eff2.dur = eff2.dur - eff.reduce
+			if eff2.dur <= 0 then self:removeEffect(eff2) end
+		end
+	end,
 }
 
 newEffect{
diff --git a/game/modules/tome/resolvers.lua b/game/modules/tome/resolvers.lua
index c528c21e18..72eb40ab2f 100644
--- a/game/modules/tome/resolvers.lua
+++ b/game/modules/tome/resolvers.lua
@@ -837,7 +837,7 @@ end
 local inscriptions_max = {
 	heal = 1,
 	protect = 1,
-	attack = 4,
+	attack = 1,  -- Reduce instant cast NPC bursts
 	movement = 1,
 	utility = 6,
 	teleport = 0, -- Annoying
-- 
GitLab