From d8eebf6fcd82deadb86e3e89d5ba7ec65f961e41 Mon Sep 17 00:00:00 2001
From: DarkGod <darkgod@net-core.org>
Date: Sun, 28 Jun 2020 21:11:44 +0200
Subject: [PATCH] Replaced Phantasm 4th talent with Mirror Image

---
 .../default/engine/interface/ActorTalents.lua |   2 +-
 game/modules/tome/ai/summon.lua               |  13 ++
 game/modules/tome/class/Actor.lua             |   3 +
 game/modules/tome/class/interface/Combat.lua  |   2 +-
 game/modules/tome/data/damage_types.lua       |   8 +-
 .../tome/data/gfx/talents/mirror_images.png   | Bin 0 -> 7458 bytes
 .../tome/data/talents/misc/inscriptions.lua   | 116 +++++-----
 .../tome/data/talents/psionic/discharge.lua   |   2 +-
 .../tome/data/talents/spells/phantasm.lua     | 206 +++++++++---------
 .../tome/data/talents/spells/spells.lua       |   2 +-
 .../tome/data/timed_effects/magical.lua       |  32 ---
 .../modules/tome/data/timed_effects/other.lua |  46 +++-
 12 files changed, 233 insertions(+), 199 deletions(-)
 create mode 100644 game/modules/tome/data/gfx/talents/mirror_images.png

diff --git a/game/engines/default/engine/interface/ActorTalents.lua b/game/engines/default/engine/interface/ActorTalents.lua
index 362ac9ff0c..89a7a665f2 100644
--- a/game/engines/default/engine/interface/ActorTalents.lua
+++ b/game/engines/default/engine/interface/ActorTalents.lua
@@ -1054,7 +1054,7 @@ end
 function _M:getTalentTarget(t)
 	if type(t.target) == "function" then
 		typ = t.target(self, t)
-		typ.talent_mode = self:getCurrentTalentMode()
+		if typ then typ.talent_mode = self:getCurrentTalentMode() end
 		return typ
 	end
 	if t.target then t.target.talent_mode = self:getCurrentTalentMode() end -- Yes t is a global, not linked to actor, but this shouldnt matter as this will be set every time anyway
diff --git a/game/modules/tome/ai/summon.lua b/game/modules/tome/ai/summon.lua
index 7b0869f7a6..d8c71be7f6 100644
--- a/game/modules/tome/ai/summon.lua
+++ b/game/modules/tome/ai/summon.lua
@@ -23,3 +23,16 @@ newAI("summoned", function(self)
 		return self:runAI(self.ai_real)
 	end
 end)
+
+newAI("mirror_image", function(self)
+	if self:runAI(self.ai_state.ai_target or "target_simple") then
+		local eff = self.summoner:hasEffect(self.summoner.EFF_MIRROR_IMAGE_REAL)
+		if self.ai_state.use_taunt and not (eff and eff.last_talent) then
+			self:forceUseTalent(self.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
+		else
+			self:useEnergy()
+		end
+		if eff then eff.last_talent = nil end
+		return true
+	end
+end)
diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua
index dea4685bd4..4a8d3714ea 100644
--- a/game/modules/tome/class/Actor.lua
+++ b/game/modules/tome/class/Actor.lua
@@ -131,6 +131,9 @@ _M.temporary_values_conf.projectile_evasion_spread = "highest"
 
 _M.temporary_values_conf.life_leech_chance = "highest"
 
+-- For archmages
+_M.temporary_values_conf.blind_inc_damage = "highest"
+
 -- Damage redirection takes last
 _M.temporary_values_conf.force_use_resist = "last"
 _M.temporary_values_conf.force_use_resist_percent = "last"
diff --git a/game/modules/tome/class/interface/Combat.lua b/game/modules/tome/class/interface/Combat.lua
index b7e990f899..504fa86f83 100644
--- a/game/modules/tome/class/interface/Combat.lua
+++ b/game/modules/tome/class/interface/Combat.lua
@@ -1766,7 +1766,7 @@ end
 
 --- Gets spellpower raw
 function _M:combatSpellpowerRaw(add)
-	if self.combat_precomputed_spellpower then return self.combat_precomputed_spellpower end
+	if self.combat_precomputed_spellpower then return self.combat_precomputed_spellpower, 1 end
 	add = add or 0
 
 	if self.combat_generic_power then
diff --git a/game/modules/tome/data/damage_types.lua b/game/modules/tome/data/damage_types.lua
index a9f6d0985d..4d2215a970 100644
--- a/game/modules/tome/data/damage_types.lua
+++ b/game/modules/tome/data/damage_types.lua
@@ -483,13 +483,13 @@ setDefaultProjector(function(src, x, y, type, dam, state)
 		end
 
 		--Dark Empathy (Reduce damage against summoner)
-		if src.necrotic_minion_be_nice and src.summoner == target then
-			dam = dam * (1 - src.necrotic_minion_be_nice)
+		if src.minion_be_nice and src.summoner == target then
+			dam = dam * (1 - src.minion_be_nice)
 		end
 
 		--Dark Empathy (Reduce damage against other minions)
-		if src.necrotic_minion_be_nice and target.summoner and src.summoner == target.summoner then
-			dam = dam * (1 - src.necrotic_minion_be_nice)
+		if src.minion_be_nice and target.summoner and src.summoner == target.summoner then
+			dam = dam * (1 - src.minion_be_nice)
 		end
 
 		-- Curse of Misfortune: Unfortunate End (chance to increase damage enough to kill)
diff --git a/game/modules/tome/data/gfx/talents/mirror_images.png b/game/modules/tome/data/gfx/talents/mirror_images.png
new file mode 100644
index 0000000000000000000000000000000000000000..bc3b8fdf174546693829a717960731edd0c6b6a2
GIT binary patch
literal 7458
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

literal 0
HcmV?d00001

diff --git a/game/modules/tome/data/talents/misc/inscriptions.lua b/game/modules/tome/data/talents/misc/inscriptions.lua
index 01e454527c..c98061b28c 100644
--- a/game/modules/tome/data/talents/misc/inscriptions.lua
+++ b/game/modules/tome/data/talents/misc/inscriptions.lua
@@ -834,61 +834,62 @@ newInscription{
 		return res
 	end,
 	action = function(self, t)
-			if not self:canBe("summon") then game.logPlayer(self, "You cannot summon; you are suppressed!") return end
-
-			-- Find all actors in radius 10 and add them to a table
-			local tg = {type="ball", radius=self.sight}
-			local grids = self:project(tg, self.x, self.y, function() end)
-			local tgts = {}
-			for x, ys in pairs(grids) do for y, _ in pairs(ys) do
-				local target = game.level.map(x, y, Map.ACTOR)
-				if target and self:reactionToward(target) < 0 then tgts[#tgts+1] = target end
-			end end
-
-			for _ = 1,3 do
-				local target = rng.tableRemove(tgts)
-				if target then
-					local tx, ty = util.findFreeGrid(target.x, target.y, 10, true, {[Map.ACTOR]=true})
-					if tx then
-						local Talents = require "engine.interface.ActorTalents"
-						local NPC = require "mod.class.NPC"
-						local caster = self
-						local image = NPC.new{
-							name = _t"Mirror Image",
-							type = "image", subtype = "image",
-							ai = "summoned", ai_real = nil, ai_state = { talent_in=1, }, ai_target = {actor=nil},
-							desc = _t"A blurred image.",
-							image = caster.image,
-							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(),
-							combat_def = caster:combatDefense(),
-							combat_armor = caster:combatArmor(),
-							size_category = caster.size_category,
-							resists = t.getInheritedResist(self, t),
-							rank = 1,
-							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
-								self:forceUseTalent(self.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
-							end,
-							faction = caster.faction,
-							summoner = caster,
-							summon_time=t.getDur(self, t),
-							no_breath = 1,
-							remove_from_party_on_death = true,
-						}
-
-						image:resolve()
-						game.zone:addEntity(game.level, image, "actor", tx, ty)
+		if not self:canBe("summon") then game.logPlayer(self, "You cannot summon; you are suppressed!") return end
+
+		-- Find all actors in radius 10 and add them to a table
+		local tg = {type="ball", radius=self.sight}
+		local grids = self:project(tg, self.x, self.y, function() end)
+		local tgts = {}
+		for x, ys in pairs(grids) do for y, _ in pairs(ys) do
+			local target = game.level.map(x, y, Map.ACTOR)
+			if target and self:reactionToward(target) < 0 then tgts[#tgts+1] = target end
+		end end
+
+		for _ = 1,3 do
+			local target = rng.tableRemove(tgts)
+			if target then
+				local tx, ty = util.findFreeGrid(target.x, target.y, 10, true, {[Map.ACTOR]=true})
+				if tx then
+					local Talents = require "engine.interface.ActorTalents"
+					local NPC = require "mod.class.NPC"
+					local caster = self
+					local image = NPC.new{
+						name = _t"Mirror Image",
+						type = "image", subtype = "image",
+						ai = "summoned", ai_real = nil, ai_state = { talent_in=1, }, ai_target = {actor=nil},
+						desc = _t"A blurred image.",
+						image = caster.image,
+						add_mos = table.clone(caster.add_mos, true),
+						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(),
+						combat_def = caster:combatDefense(),
+						combat_armor = caster:combatArmor(),
+						size_category = caster.size_category,
+						resists = t.getInheritedResist(self, t),
+						rank = 1,
+						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
+							self:forceUseTalent(self.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
+						end,
+						faction = caster.faction,
+						summoner = caster,
+						summon_time=t.getDur(self, t),
+						no_breath = 1,
+						remove_from_party_on_death = true,
+					}
+
+					image:resolve()
+					game.zone:addEntity(game.level, image, "actor", tx, ty)
+					if game.party:hasMember(self) then
 						game.party:addMember(image, {
 							control=false,
 							type="summon",
@@ -896,11 +897,12 @@ newInscription{
 							temporary_level = true,
 							orders = {},
 						})
-
-						image:forceUseTalent(image.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
 					end
+
+					image:forceUseTalent(image.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
 				end
 			end
+		end
 
 		return true
 	end,
diff --git a/game/modules/tome/data/talents/psionic/discharge.lua b/game/modules/tome/data/talents/psionic/discharge.lua
index b3cff11f6d..c26ed62867 100644
--- a/game/modules/tome/data/talents/psionic/discharge.lua
+++ b/game/modules/tome/data/talents/psionic/discharge.lua
@@ -183,7 +183,7 @@ newTalent{
 }
 
 newTalent{
-	name = "Focused Wrath",   
+	name = "Focused Wrath",
 	type = {"psionic/discharge", 4},
 	points = 5, 
 	require = psi_wil_high4,
diff --git a/game/modules/tome/data/talents/spells/phantasm.lua b/game/modules/tome/data/talents/spells/phantasm.lua
index 64abe00bae..3c42070b77 100644
--- a/game/modules/tome/data/talents/spells/phantasm.lua
+++ b/game/modules/tome/data/talents/spells/phantasm.lua
@@ -144,124 +144,128 @@ newTalent{
 }
 
 newTalent{
-	name = "Elemental Mirage", image = "talents/blur_sight.png",
+	name = "Mirror Image", short_name = "MIRROR_IMAGES",
 	type = {"spell/phantasm", 4},
-	mode = "sustained",
 	require = spells_req4,
 	points = 5,
-	sustain_mana = 30,
+	mana = 30,
 	cooldown = 20,
-	tactical = { BUFF = 2 },
-	no_npc_use = true,
-	autolearn_talent = "T_ALTER_MIRAGE",
-	getBonus = function(self, t) return self:combatTalentScale(t, 10, 30) end,
-	chooseElements = function(self, t)
-		local DT = DamageType
-		local elements = {}
-		for _, dtid in ipairs{DT.ARCANE, DT.FIRE, DT.LIGHTNING, DT.COLD, DT.PHYSICAL, DT.LIGHT, DT.TEMPORAL, DT.ACID, DT.NATURE, DT.BLIGHT, DT.DARKNESS} do
-			local dt = DT:get(dtid)
-			elements[#elements+1] = {name = dt.text_color..dt.name, id=dtid}
-		end
-
-		local e1, e2 = nil, nil
-
-		local res = self:talentDialog(Dialog:listPopup("Elemental Mirage", "Choose the first element:", elements, 400, 300, function(item) self:talentDialogReturn(item) end))
-		if not res then return nil end
-		e1 = res.id
-		table.removeFromList(elements, res)
-
-		local res = self:talentDialog(Dialog:listPopup("Elemental Mirage", "Choose the second element:", elements, 400, 300, function(item) self:talentDialogReturn(item) end))
-		if not res then return nil end
-		e2 = res.id
-
-		return e1, e2
-	end,
-	callbackOnDealDamage = function(self, t, val, target, dead, death_note)
-		self.elemental_mirage_elements = self.elemental_mirage_elements or {}
-		if not self.elemental_mirage_elements.e1 then return end
-		if not death_note or not death_note.damtype then return end
+	tactical = { DEFEND = 3,},
+	getLife = function(self, t) return math.ceil(self:combatTalentScale(t, 5, 12)) end,
+	on_pre_use = function(self, t) return self.in_combat and not self:hasEffect(self.EFF_MIRROR_IMAGE_REAL) end,
+	action = function(self, t)
+		if not self:canBe("summon") then game.logPlayer(self, "You cannot summon; you are suppressed!") return end
 
-		if death_note.damtype == self.elemental_mirage_elements.e1 then
-			if self.turn_procs.elemental_mirage1 then return end
-			self.turn_procs.elemental_mirage1 = true
+		-- Find all actors in radius 10 and add them to a table
+		local tg = {type="ball", radius=self.sight}
+		local grids = self:project(tg, self.x, self.y, function() end)
+		local tgts = {}
+		for x, ys in pairs(grids) do for y, _ in pairs(ys) do
+			local target = game.level.map(x, y, Map.ACTOR)
+			if target and self:reactionToward(target) < 0 then tgts[#tgts+1] = target end
+		end end
 
-			local pen = nil
-			if self:getTalentLevel(t) >= 5 then
-				local ep1 = self:combatGetResistPen(self.elemental_mirage_elements.e1, true)
-				local ep2 = self:combatGetResistPen(self.elemental_mirage_elements.e2, true)
-				if ep2 < ep1 then pen = ep1 - ep2 end
-			end
+		local target = rng.tableRemove(tgts)
+		if not target then return end
+		local tx, ty = util.findFreeGrid(target.x, target.y, 10, true, {[Map.ACTOR]=true})
+		if not tx then return end
 
-			self:setEffect(self.EFF_ELEMENTAL_MIRAGE2, 3, {dt=self.elemental_mirage_elements.e2, power=t.getBonus(self, t), pen=pen})
-		elseif death_note.damtype == self.elemental_mirage_elements.e2 then
-			if self.turn_procs.elemental_mirage2 then return end
-			self.turn_procs.elemental_mirage2 = true
+		local Talents = require "engine.interface.ActorTalents"
+		local NPC = require "mod.class.NPC"
+		local image = NPC.new{
+			name = ("Mirror Image (%s)"):tformat(self:getName()),
+			type = "image", subtype = "image",
+			ai = "mirror_image", ai_real = nil, ai_state = { talent_in=1, }, ai_target = {actor=nil},
+			desc = _t"A blurred image.",
+			image = self.image,
+			add_mos = table.clone(self.add_mos, true),
+			exp_worth=0,
+			level_range = {self.level, self.level},
+			level = self.level,
+			size_category = self.size_category,
+			global_speed = self.global_speed,
+			global_speed_add = self.global_speed_add,
+			global_speed_base = self.global_speed_base,
+			combat_spellspeed = self.combat_spellspeed,
+			combat_def = -1000,
+			combat_armor = 0,
+			max_mana = 10000,
+			mana = 10000,
+			rank = 1,
+			difficulty_boosted = 1,
+			life_rating = 0,
+			life_regen = 0, no_life_regen = 1,
+			cant_be_moved = 1,
+			never_move = 1,
+			never_anger = true,
+			generic_damage_penalty = 66,
+			inc_damage = table.clone(self.inc_damage or {}, true),
+			resists_pen = table.clone(self.resists_pen or {}, true),
+			combat_precomputed_spellpower = self:combatSpellpowerRaw(),
+			resolvers.talents{
+				[Talents.T_TAUNT]=1, -- Add the talent so the player can see it even though we cast it manually
+			},
+			faction = self.faction,
+			summoner = self,
+			heal = function() return 0 end, -- Cant ever heal
+			takeHit = function(self, value, src, death_note) -- Cant ever take more than one damage per turn per actor
+				if not src then return false, 0 end
+				if src ~= self then
+					if death_note.source_talent_mode ~= "active" then return false, 0 end
+					if self.turn_procs.mirror_image_dmg and self.turn_procs.mirror_image_dmg[src] then return false, 0 end
+					self.turn_procs.mirror_image_dmg = self.turn_procs.mirror_image_dmg or {}
+					self.turn_procs.mirror_image_dmg[src] = true
+				end
+				return mod.class.NPC.takeHit(self, 1, src, death_note)
+			end,
+			on_die = function(self)
+				self.summoner:removeEffect(self.summoner.EFF_MIRROR_IMAGE_REAL, true, true)
+			end,
+			spellFriendlyFire = function() return 100 end,
+			no_breath = 1,
+			remove_from_party_on_death = true,
+		}
 
-			local pen = nil
-			if self:getTalentLevel(t) >= 5 then
-				local ep1 = self:combatGetResistPen(self.elemental_mirage_elements.e1, true)
-				local ep2 = self:combatGetResistPen(self.elemental_mirage_elements.e2, true)
-				if ep1 < ep2 then pen = ep2 - ep1 end
-			end
+		image:resolve()
+		game.zone:addEntity(game.level, image, "actor", tx, ty)
+		image.max_life = t:_getLife(self)
+		image.life = t:_getLife(self)
 
-			self:setEffect(self.EFF_ELEMENTAL_MIRAGE1, 3, {dt=self.elemental_mirage_elements.e1, power=t.getBonus(self, t), pen=pen})
-		end
-	end,
-	activate = function(self, t)
-		self.elemental_mirage_elements = self.elemental_mirage_elements or {}
-		local e1, e2
-		if self.elemental_mirage_elements.e1 then e1, e2 = self.elemental_mirage_elements.e1, self.elemental_mirage_elements.e2
-		else e1, e2 = t.chooseElements(self, t) end
+		-- Clone particles
 
-		game:playSoundNear(self, "talents/heal")
-		local ret = {}
-		self.elemental_mirage_elements = {e1=e1, e2=e2}
 
-		self:addShaderAura("elemental_mirage", "crystalineaura", {time_factor=500, spikeOffset=0.123123, spikeLength=1.1, spikeWidth=3, growthSpeed=5, color={255/255, 215/255, 0/255}}, "particles_images/smoothspikes.png")
+		local dam_bonus = self:callTalent(self.T_INVISIBILITY, "getDamPower")
+		image:setEffect(image.EFF_MIRROR_IMAGE_FAKE, 1, {dam=dam_bonus})
+		self:setEffect(image.EFF_MIRROR_IMAGE_REAL, 1, {image=image, dam=dam_bonus})
 
-		return ret
-	end,
-	deactivate = function(self, t, p)
-		self:removeShaderAura("elemental_mirage")
-		return true
-	end,
-	info = function(self, t)
-		self.elemental_mirage_elements = self.elemental_mirage_elements or {}
-		local DT = DamageType
-		local e1, e2
-		if self.elemental_mirage_elements.e1 then
-			local dt1 = DT:get(self.elemental_mirage_elements.e1)
-			local dt2 = DT:get(self.elemental_mirage_elements.e2)
-			e1, e2 = dt1.text_color..dt1.name, dt2.text_color..dt2.name
+		-- Player & NPC don't work the same for this spell
+		if game.party:hasMember(self) then
+			game.party:addMember(image, {
+				control=false,
+				type="summon",
+				title=_t"Summon",
+				temporary_level = true,
+				orders = {},
+			})
+			image:forceUseTalent(image.T_TAUNT, {ignore_cd=true, no_talent_fail = true})
+			image.ai_state.use_taunt = true
 		else
-			e1, e2 = "not selected", "not selected"
+			-- Dont reveal ourself to player
+			image.tooltip = function(self, x, y, seen_by)
+				return mod.class.NPC.tooltip(self.summoner, x, y, seen_by)
+			end
+			image.ai_state.use_taunt = false
 		end
 
-		return ([[Your mastery of both illusion and elements knows no bound.
-		Upon first sustaining this spell you may select two elements. You may later change them with the Alter Mirage spell, provided automatically upon learning this one.
-
-		Any time you deal damage with one of those elements, the other gets a bonus of %d%% damage for 3 turns.
-		At level 5 if the target element has less resistance penetration, it gets increased to match the one of the source element.
-
-		Current elements selected: %s#LAST# and %s]]):
-		tformat(t.getBonus(self, t), e1, e2)
-	end,
-}
-
-newTalent{
-	name = "Alter Mirage",
-	type = {"spell/other", 1},
-	points = 5,
-	mana = 3,
-	cooldown = 3,
-	no_npc_use = true,
-	action = function(self, t)
-		local e1, e2 = self:callTalent(self.T_ELEMENTAL_MIRAGE, "chooseElements")
-		if not e1 then return end
-		self.elemental_mirage_elements = {e1=e1, e2=e2}
 		return true
 	end,
 	info = function(self, t)
-		return _t[[Change your choice of elements for Elemental Mirage.]]
+		return ([[Create a perfect lookalike of your own form made out of pure light near a creature or yourself if no creature is present.
+		This image has %d life and can never take more than 1 damage per creature per turn and is immune to any non direct damage (ground effects, damage over time, ...).
+		Whenever you cast a spell your mirror image will try to duplicate it at the same target for 66%% less damage, if possible. If it can it will loose 1 life, if not it will instead taunt a creature to focus its attention on itself.
+		While the image exists you receive the damage bonus from the Invisibility spell as if you were invisible.
+		This spell can not be cast while a Mirror Image already exists and only in combat.
+		]])
+		:tformat(t.getLife(self, t))
 	end,
 }
diff --git a/game/modules/tome/data/talents/spells/spells.lua b/game/modules/tome/data/talents/spells/spells.lua
index bc07765fd1..c7b9c44183 100644
--- a/game/modules/tome/data/talents/spells/spells.lua
+++ b/game/modules/tome/data/talents/spells/spells.lua
@@ -178,7 +178,7 @@ function necroSetupSummon(self, def, x, y, level, turns, no_control)
 	m.inc_damage = table.clone(self.inc_damage, true)
 	m.no_breath = 1
 	m.no_drops = true
-	m.necrotic_minion_be_nice = 1
+	m.minion_be_nice = 1
 
 	if self:isTalentActive(self.T_NECROTIC_AURA) then
 		local t = self:getTalentFromId(self.T_NECROTIC_AURA)
diff --git a/game/modules/tome/data/timed_effects/magical.lua b/game/modules/tome/data/timed_effects/magical.lua
index f46f204074..7e9d721221 100644
--- a/game/modules/tome/data/timed_effects/magical.lua
+++ b/game/modules/tome/data/timed_effects/magical.lua
@@ -4661,22 +4661,6 @@ for _, dt in ipairs{
 	newEffect(e)
 end
 
-newEffect{
-	name = "ELEMENTAL_MIRAGE1", image = "talents/blur_sight.png",
-	desc = _t"Elemental Mirage (First Element)",
-	long_desc = function(self, eff) return ("%s damage increased by %d%% and resistance penetration by %d%%."):tformat(DamageType:get(eff.dt).name, eff.power, eff.pen or 0) end,
-	type = "magical",
-	subtype = { phantasm=true,},
-	status = "beneficial",
-	parameters = {dt=DamageType.ARCANE, power=10},
-	on_gain = function(self, err) return nil, true end,
-	on_lose = function(self, err) return nil, true end,
-	activate = function(self, eff)
-		self:effectTemporaryValue(eff, "inc_damage", {[eff.dt] = eff.power})
-		if eff.pen then self:effectTemporaryValue(eff, "resists_pen", {[eff.dt] = eff.pen}) end
-	end,
-}
-
 newEffect{
 	name = "LICH_FEAR", image = "talents/lichform.png",
 	desc = _t"Frightening Presence",
@@ -4696,22 +4680,6 @@ newEffect{
 	end,
 }
 
-newEffect{
-	name = "ELEMENTAL_MIRAGE2", image = "talents/alter_mirage.png",
-	desc = _t"Elemental Mirage (Second Element)",
-	long_desc = function(self, eff) return ("%s damage increased by %d%% and resistance penetration by %d%%."):tformat(DamageType:get(eff.dt).name, eff.power, eff.pen or 0) end,
-	type = "magical",
-	subtype = { phantasm=true,},
-	status = "beneficial",
-	parameters = {dt=DamageType.FIRE, power=10},
-	on_gain = function(self, err) return nil, true end,
-	on_lose = function(self, err) return nil, true end,
-	activate = function(self, eff)
-		self:effectTemporaryValue(eff, "inc_damage", {[eff.dt] = eff.power})
-		if eff.pen then self:effectTemporaryValue(eff, "resists_pen", {[eff.dt] = eff.pen}) end
-	end,
-}
-
 newEffect{
 	name = "COMMANDER_OF_THE_DEAD", image = "talents/commander_of_the_dead.png",
 	desc = _t"Commander of the Dead",
diff --git a/game/modules/tome/data/timed_effects/other.lua b/game/modules/tome/data/timed_effects/other.lua
index 7dbfae454a..b7e3f2b119 100644
--- a/game/modules/tome/data/timed_effects/other.lua
+++ b/game/modules/tome/data/timed_effects/other.lua
@@ -4021,7 +4021,7 @@ newEffect{
 	name = "DOZING", image = "talents/sleep.png",
 	desc = _t"Dozing",
 	long_desc = function(self, eff) return _t"The target is completely asleep, unable to act." end,
-	type = "other",
+	type = "otber",
 	subtype = { sleep=true },
 	status = "detrimental",
 	parameters = { },
@@ -4031,3 +4031,47 @@ newEffect{
 	deactivate = function(self, eff)
 	end,
 }
+
+newEffect{
+	name = "MIRROR_IMAGE_REAL", image = "talents/mirror_images.png",
+	desc = _t"Protected by a Mirror Image",
+	long_desc = function(self, eff) return ("Target is protected by a mirror image. Increases damage dealt to blind or dazzled creatures by %d%%"):tformat(eff.dam) end,
+	type = "other",
+	subtype = { phantasm=true },
+	status = "beneficial", decrease = 0,
+	cancel_on_level_change = true,
+	parameters = { dam=10 },
+	callbackOnTalentPost = function(self, eff, ab, ret, silent)
+		if ab.id == self.T_MIRROR_IMAGES then return end
+		if not ab.is_spell or ab.is_inscription then return end
+		if ab.mode ~= "activated" then return end
+		if not ret then return end
+		local x, y, tgt = self:getTarget()
+		if not tgt or not ab.requires_target then return end
+		if eff.image:attr("dead") or not game.level:hasEntity(eff.image) then return end
+
+		pcall(function() -- Just in case
+			eff.image:forceUseTalent(ab.id, {force_level=self:getTalentLevelRaw(ab.id), ignore_cd=true, no_talent_fail=true, ignore_energy=true, force_talent_ignore_ressources=true, force_target=tgt})
+			eff.image:takeHit(1, eff.image)
+		end)
+
+		eff.last_talent = true
+	end,
+	activate = function(self, eff)
+		self:effectTemporaryValue(eff, "blind_inc_damage", eff.dam)	
+	end,
+}
+
+newEffect{
+	name = "MIRROR_IMAGE_FAKE", image = "talents/mirror_images.png",
+	desc = _t"Protected by a Mirror Image",
+	long_desc = function(self, eff) return ("Target is protected by a mirror image. Increases damage dealt to blind or dazzled creatures by %d%%"):tformat(eff.dam) end,
+	type = "other",
+	subtype = { phantasm=true },
+	status = "beneficial", decrease = 0,
+	cancel_on_level_change = true,
+	parameters = { dam=10 },
+	activate = function(self, eff)
+		self:effectTemporaryValue(eff, "blind_inc_damage", eff.dam)	
+	end,
+}
-- 
GitLab