From 022f179b5c0c6dba8d793b292983ee43f251f852 Mon Sep 17 00:00:00 2001
From: dg <dg@51575b47-30f0-44d4-a5cc-537603b46e54>
Date: Fri, 3 Aug 2012 20:03:20 +0000
Subject: [PATCH] New psionic class Solipsist - Create and destroy with your
 thoughts, manipulate and enter the dreams of others, overcome the world with
 the power of your mind

git-svn-id: http://svn.net-core.org/repos/t-engine4@5411 51575b47-30f0-44d4-a5cc-537603b46e54
---
 game/modules/tome/ai/tactical.lua             |   5 +-
 game/modules/tome/ai/target.lua               |   1 +
 game/modules/tome/class/Actor.lua             |  40 ++--
 .../tome/data/birth/classes/psionic.lua       |  10 +-
 .../modules/tome/data/general/npcs/horror.lua | 175 +++++++++++++++++-
 .../npc/horror_eldritch_dream_seed.png        | Bin 0 -> 4726 bytes
 .../npc/horror_eldritch_dreaming_horror.png   | Bin 0 -> 9482 bytes
 .../talents/chronomancy/age-manipulation.lua  |  11 +-
 .../tome/data/talents/psionic/distortion.lua  |  10 +-
 .../tome/data/talents/psionic/dreaming.lua    |   9 +-
 .../tome/data/talents/psionic/mentalism.lua   |  16 +-
 .../tome/data/talents/psionic/nightmare.lua   |   4 +-
 .../data/talents/psionic/psychic-assault.lua  | 115 ++++++------
 .../tome/data/talents/psionic/slumber.lua     |   8 +-
 .../tome/data/talents/psionic/solipsism.lua   |  37 +++-
 .../data/talents/psionic/thought-forms.lua    |  29 ++-
 .../data/texts/unlock-psionic_solipsist.lua   |  36 ++++
 .../tome/data/timed_effects/mental.lua        |  40 +++-
 .../modules/tome/data/timed_effects/other.lua |  36 +++-
 .../data/zones/dreamscape-talent/zone.lua     |   2 +-
 20 files changed, 450 insertions(+), 134 deletions(-)
 create mode 100644 game/modules/tome/data/gfx/shockbolt/npc/horror_eldritch_dream_seed.png
 create mode 100644 game/modules/tome/data/gfx/shockbolt/npc/horror_eldritch_dreaming_horror.png
 create mode 100644 game/modules/tome/data/texts/unlock-psionic_solipsist.lua

diff --git a/game/modules/tome/ai/tactical.lua b/game/modules/tome/ai/tactical.lua
index 6c7244b8c8..ef3dd160de 100644
--- a/game/modules/tome/ai/tactical.lua
+++ b/game/modules/tome/ai/tactical.lua
@@ -19,7 +19,7 @@
 
 local DamageType = require "engine.DamageType"
 
-local print = function() end
+--local print = function() end
 
 -- Internal functions
 local checkLOS = function(sx, sy, tx, ty)
@@ -185,13 +185,14 @@ newAI("use_tactical", function(self)
 
 		local need_heal = 0
 		local life = 100 * self.life / self.max_life
+		-- Subtract solipsism straight from the life value to give us higher than normal weights; helps keep clarity up and avoid solipsism
+		if self:knowTalent(self.T_SOLIPSISM) then life = life - (100 * self:getPsi() / self:getMaxPsi()) end
 		if life < 20 then need_heal = need_heal + 10 * self_compassion / 5
 		elseif life < 30 then need_heal = need_heal + 8 * self_compassion / 5
 		elseif life < 40 then need_heal = need_heal + 5 * self_compassion / 5
 		elseif life < 60 then need_heal = need_heal + 4 * self_compassion / 5
 		elseif life < 80 then need_heal = need_heal + 3 * self_compassion / 5
 		end
-
 		-- Need healing
 		if avail.heal then
 			want.heal = need_heal
diff --git a/game/modules/tome/ai/target.lua b/game/modules/tome/ai/target.lua
index 8734caedc1..3efd1ec364 100644
--- a/game/modules/tome/ai/target.lua
+++ b/game/modules/tome/ai/target.lua
@@ -59,6 +59,7 @@ newAI("target_player_radius", function(self)
 
 	if core.fov.distance(self.x, self.y, game.player.x, game.player.y) < self.ai_state.sense_radius then
 		self.ai_target.actor = game.player
+		self:check("on_acquire_target", game.player)
 		return true
 	end
 end)
diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua
index 9d362167fe..48642f4d3e 100644
--- a/game/modules/tome/class/Actor.lua
+++ b/game/modules/tome/class/Actor.lua
@@ -1513,11 +1513,6 @@ function _M:onTakeHit(value, src)
 		return 0
 	end
 
-	if self.knowTalent and self:knowTalent(self.T_DISMISSAL) then
-		local t = self:getTalentFromId(self.T_DISMISSAL)
-		value = t.doDismissalOnHit(self, value, src, t)
-	end
-
 	if self:attr("retribution") then
 	-- Absorb damage into the retribution
 		if value / 2 <= self.retribution_absorb then
@@ -1696,6 +1691,12 @@ function _M:onTakeHit(value, src)
 			eff.begone = game.turn
 		end
 	end
+	
+	-- Mind save to reduce damage to zero
+	if self:knowTalent(self.T_DISMISSAL) and value > 0 then
+		local t = self:getTalentFromId(self.T_DISMISSAL)
+		value = t.doDismissalOnHit(self, value, src, t)
+	end
 
 	-- Feedback pool: Stores damage as energy to use later
 	if self:getMaxFeedback() > 0 and src ~= self and src ~= self.summoner then
@@ -1757,20 +1758,11 @@ function _M:onTakeHit(value, src)
 		end
 	end
 
-	-- Solipsism
-	if self:knowTalent(self.T_SOLIPSISM) then
+	-- Solipsism damage; set it up here but apply it after on_take_hit
+	local damage_to_psi = 0
+	if self:knowTalent(self.T_SOLIPSISM) and value > 0 then
 		local t = self:getTalentFromId(self.T_SOLIPSISM)
-		local damage_to_psi = value * t.getConversionRatio(self, t)
-		if self:getPsi() > damage_to_psi then
-			self:incPsi(-damage_to_psi)
-		else
-			damage_to_psi = self:getPsi()
-			self:incPsi(-damage_to_psi)
-		end
-		if damage_to_psi > 0 then
-			game.logSeen(self, "%s's mind suffers #YELLOW#%d psi#LAST# damage from the attack.", self.name:capitalize(), damage_to_psi)
-		end
-		value = value - damage_to_psi
+		damage_to_psi = value * t.getConversionRatio(self, t)
 	end
 
 	-- Stoned ? SHATTER !
@@ -1858,6 +1850,18 @@ function _M:onTakeHit(value, src)
 	end
 
 	if self.on_takehit then value = self:check("on_takehit", value, src) end
+	
+	-- Apply Solipsism hit
+	if damage_to_psi > 0 then
+		if self:getPsi() > damage_to_psi then
+			self:incPsi(-damage_to_psi)
+		else
+			damage_to_psi = self:getPsi()
+			self:incPsi(-damage_to_psi)
+		end
+		game.logSeen(self, "%s's mind suffers #YELLOW#%d psi#LAST# damage from the attack.", self.name:capitalize(), damage_to_psi)
+		value = value - damage_to_psi
+	end
 
 	-- VITALITY?
 	if self:knowTalent(self.T_VITALITY) and self.life > self.max_life /2 and self.life - value <= self.max_life/2 then
diff --git a/game/modules/tome/data/birth/classes/psionic.lua b/game/modules/tome/data/birth/classes/psionic.lua
index 64dad34645..0212f663ec 100644
--- a/game/modules/tome/data/birth/classes/psionic.lua
+++ b/game/modules/tome/data/birth/classes/psionic.lua
@@ -152,11 +152,11 @@ newBirthDescriptor{
 newBirthDescriptor{
 	type = "subclass",
 	name = "Solipsist",
-	locked = function() return profile.mod.allow_build.psionic_solipsist and true or "hide"  end,
-	locked_desc = "The world as we know it is the collective dream of those that live in it.  Find and wake the sleeper and you'll unlock the potential of your dreams.",
+	locked = function() return profile.mod.allow_build.psionic_solipsist end,
+	locked_desc = "Some believe that the world is the collective dream of those that live in it.  Find and wake the sleeper and you'll unlock the potential of your dreams.",
 	desc = {
-		"The Solipsist has awakened to the truth of reality, that it's malleable, nothing more than the collective vision of those that experience it.",
-		"They wield this knowledge to both create and destory, to invade the minds of others, and to manipulate the dreams of the unenlightened.",
+		"The Solipsist believes that reality is malleable and nothing more than the collective vision of those that experience it.",
+		"They wield this knowledge to both create and destory, to invade the minds of others, and to manipulate the dreams of those around them.",
 		"This knowledge comes with a heavy price and the Solipsist must guard his thoughts, lest he come to believe that the world exists only within his own mind.",
 		"Their most important stats are: Willpower and Cunning",
 		"#GOLD#Stat modifiers:",
@@ -187,7 +187,7 @@ newBirthDescriptor{
 	talents = {
 		[ActorTalents.T_SLEEP] = 1,
 		
-		[ActorTalents.T_SUNDER_MIND] = 1,
+		[ActorTalents.T_MIND_SEAR] = 1,
 		[ActorTalents.T_SOLIPSISM] = 1,
 		[ActorTalents.T_THOUGHT_FORMS] = 1,
 	},
diff --git a/game/modules/tome/data/general/npcs/horror.lua b/game/modules/tome/data/general/npcs/horror.lua
index 940879cd25..761af21563 100644
--- a/game/modules/tome/data/general/npcs/horror.lua
+++ b/game/modules/tome/data/general/npcs/horror.lua
@@ -30,7 +30,7 @@ newEntity{
 	autolevel = "warrior",
 	ai = "dumb_talented_simple", ai_state = { ai_move="move_dmap", talent_in=3, },
 
-	stats = { str=02, dex=20, wil=20, mag=20, con=20, cun=20 },
+	stats = { str=20, dex=20, wil=20, mag=20, con=20, cun=20 },
 	combat_armor = 5, combat_def = 10,
 	combat = { dam=5, atk=10, apr=5, dammod={str=0.6} },
 	infravision = 10,
@@ -148,6 +148,7 @@ newEntity{ base = "BASE_NPC_HORROR",
 	mana_regen = 10,
 	negative_regen = 10,
 	hate_regen = 10,
+	psi_regen = 10,
 	rarity = 8,
 	rank = 3,
 	max_life = resolvers.rngavg(150,170),
@@ -175,6 +176,7 @@ newEntity{ base = "BASE_NPC_HORROR",
 		[Talents.T_DISMAY]={base=3, every=12, max=8},
 		[Talents.T_DOMINATE]={base=3, every=12, max=8},
 		[Talents.T_INVOKE_DARKNESS]={base=5, every=8, max=10},
+		[Talents.T_NIGHTMARE]={base=5, every=8, max=10},
 		[Talents.T_WAKING_NIGHTMARE]={base=3, every=8, max=10},
 		[Talents.T_ABYSSAL_SHROUD]={base=3, every=8, max=8},
 		[Talents.T_INNER_DEMONS]={base=3, every=8, max=10},
@@ -520,7 +522,7 @@ newEntity{ base = "BASE_NPC_HORROR",
 	life_regen = 0.25,
 	combat_armor = 12, combat_def = 24,
 
-		ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=2, ally_compassion=0 },
+	ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=2, ally_compassion=0 },
 
 	on_melee_hit = {[DamageType.PHYSICALBLEED]=resolvers.mbonus(14, 2)},
 	combat = { dam=resolvers.levelup(resolvers.rngavg(16,22), 1, 1.5), atk=resolvers.levelup(18, 1, 1), apr=4, dammod={wil=0.25, cun=0.1}, damtype=engine.DamageType.PHYSICALBLEED, },
@@ -583,7 +585,7 @@ newEntity{ base = "BASE_NPC_HORROR",
 		damtype=engine.DamageType.SLIME,
 	},
 
-		ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=1, ally_compassion=0 },
+	ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=1, ally_compassion=0 },
 
 	resists = {all=15, [DamageType.PHYSICAL] = -10, [DamageType.NATURE] = 100, [DamageType.ARCANE] = 40, [DamageType.BLIGHT] = 24},
 
@@ -622,7 +624,7 @@ newEntity{ base = "BASE_NPC_HORROR",
 	},
 	combat_physspeed = 2,
 
-		ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=1, ally_compassion=0 },
+	ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=1, ally_compassion=0 },
 
 	resists = {[DamageType.PHYSICAL] = -10, [DamageType.DARKNESS] = 100, [DamageType.LIGHT] = -60},
 
@@ -639,6 +641,164 @@ newEntity{ base = "BASE_NPC_HORROR",
 	},
 		resolvers.sustains_at_birth(),
 }
+
+-- Dream Horror
+newEntity{ base = "BASE_NPC_HORROR",
+	name = "dreaming horror", color=colors.ORCHID,
+	desc =[[A vaguely tentacled yet constantly changing form rests here apparently oblivious to your existence.
+With each slow breath it takes reality distorts around it.  Blue twirls into red, green twists into yellow, and the air sings softly before bursting into a myriad of pastel shapes and colors.]],
+	shader = "shadow_simulacrum",
+	shader_args = { color = {0.5, 0.5, 1.0}, base = 0.8, time_factor= 2000 },
+	level_range = {20, nil}, exp_worth = 1,
+	rarity = 30,  -- Very rare; should feel almost like uniques though they aren't
+	rank = 3,
+	max_life = 100,  -- Solipsism will take care of hit points
+	life_rating = 4, 
+	psi_rating = 6,
+	autolevel = "wildcaster",
+	combat_armor = 1, combat_def = 15,
+	combat = { dam=resolvers.levelup(20, 1, 1.1), atk=20, apr=20, dammod={wil=1}, damtype=DamageType.MIND},
+
+	ai = "tactical",  ai_tactic = resolvers.tactic"ranged",
+	ai_state = { ai_target="target_player_radius", sense_radius=20, talent_in=1 }, -- Huge radius for projections to target
+	dont_pass_target = true,
+
+	resists = { all = 35 },
+	inc_damage = { all = -50}, -- Set base damage low, in the dreamscape it will be much higher
+
+	combat_mindpower = resolvers.levelup(30, 1, 2),
+	
+	body = { INVEN = 10 },
+	resolvers.drops{chance=100, nb=5, {ego_chance=100} }, -- Gives good loot to encourage the player to wake it up
+
+	resolvers.talents{
+		[Talents.T_DISTORTION_BOLT]={base=2, every=6, max=8},
+		[Talents.T_DISTORTION_WAVE]={base=2, every=6, max=8},
+		[Talents.T_MAELSTROM]={base=2, every=6, max=8},
+		[Talents.T_RAVAGE]={base=2, every=6, max=8},
+		
+		[Talents.T_BIOFEEDBACK]={base=2, every=6, max=8},
+		[Talents.T_RESONANCE_FIELD]={base=2, every=6, max=8},
+		[Talents.T_BACKLASH]={base=2, every=6, max=8},
+		
+		[Talents.T_MENTAL_SHIELDING]={base=2, every=6, max=8},
+	
+		[Talents.T_SOLIPSISM]=7, -- Seven gives some damage to health though it's very small
+		[Talents.T_BALANCE]={base=2, every=6, max=8},
+		[Talents.T_CLARITY]={base=2, every=6, max=8},
+		[Talents.T_DISMISSAL]={base=2, every=6, max=8},
+
+		[Talents.T_LUCID_DREAMER]={base=4, every=12, max=8},
+		[Talents.T_DREAM_WALK]={base=4, every=12, max=8},
+		[Talents.T_SLUMBER]={base=2, every=6, max=8},
+		[Talents.T_RESTLESS_NIGHT]={base=2, every=6, max=8},
+		[Talents.T_DREAMSCAPE]=10,
+	},
+
+	resolvers.inscriptions(1, {"regeneration infusion"}),
+	
+	resolvers.sustains_at_birth(),
+	
+	-- Used to track if he's awake or spawning projections
+	dreamer_sleep_state = 1,
+	-- And some particles to show that we're asleep
+	resolvers.genericlast(function(e)
+		if core.shader.active() then
+			e.sleep_particle = e:addParticles(engine.Particles.new("shader_shield", 1, {img="shield2", size_factor=1}, {type="shield", time_factor=6000, aadjust=2, color={0.6, 1, 0.6}}))
+		else
+			e.sleep_particle = e:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+		end
+	end),
+	
+	-- Spawn Dream Seeds
+	on_act = function(self)
+		if self.dreamer_sleep_state and self.ai_target.actor then 
+			self.dreamer_sleep_state = math.min(self.dreamer_sleep_state + 1, 51) -- Caps at 51 so a new one doesn't spawn as soon as an old one dies
+			self:useEnergy() -- Always use energy when in the sleep state
+
+			if self.dreamer_sleep_state%10 == 0 and self.dreamer_sleep_state <= 50 then
+				-- Find Space
+				local x, y = util.findFreeGrid(self.x, self.y, 5, true, {[engine.Map.ACTOR]=true})
+				if not x then
+					return
+				end
+				
+				local seed = {type="horror", subtype="eldrtich", name="dream seed"}
+				local list = mod.class.NPC:loadList("/data/general/npcs/horror.lua")
+				local m = list.DREAM_SEED:clone()
+				if not m then return nil end
+				
+				m.exp_worth = 0
+				m.summoner = self			
+				m:resolve() m:resolve(nil, true)
+				m:forceLevelup(self.level)
+				game.zone:addEntity(game.level, m, "actor", x, y)
+				
+				game.level.map:particleEmitter(x, y, 1, "generic_teleport", {rm=225, rM=255, gm=225, gM=255, bm=225, bM=255, am=35, aM=90})
+				game.logSeen(self, "%s spawns a dream seed!", self.name:capitalize())
+			end
+		-- Script the AI to encourage opening with dream scape
+		elseif self.ai_target.actor and not game.zone.is_dream_scape then
+			if not self:isTalentCoolingDown(self.T_SLUMBER) then
+				self:forceUseTalent(self.T_SLUMBER, {})
+			elseif not self:isTalentCoolingDown(self.T_DREAMSCAPE) and self.ai_target.actor:attr("sleep") then
+				self:forceUseTalent(self.T_DREAMSCAPE, {})
+			end
+		end
+	end,
+	on_acquire_target = function(self, who)
+		self:useEnergy() -- Use energy as soon as we find a target so we don't move
+	end,
+	on_takehit = function(self, value, src)
+		if value > 0 and self.dreamer_sleep_state then
+			self.dreamer_sleep_state = nil
+			self.desc = [[A vaguely tentacled yet rapidly changing shape floats here.  With each breath you can feel reality twist, shatter, and break. 
+Blue burns into red, green bursts into yellow, and the air crackles and hisses before exploding into a thousand fragments of sharp shapes and colors.]]
+			self:removeParticles(self.sleep_particle)
+			game.logSeen(self, "#LIGHT_BLUE#The sleeper stirs...")
+		end
+		return value
+	end,
+}
+
+newEntity{ base = "BASE_NPC_HORROR", define_as = "DREAM_SEED",
+	name = "dream seed", color=colors.PINK,
+	desc ="A pinkish bubble floats here, reflecting the world not as it is, but as it would be in that surreal place that exists only in our dreams.",
+	level_range = {20, nil}, exp_worth = 1,
+	rarity = 30,  -- Very rare; but they do spawn on their own to keep the players on thier toes
+	rank = 2,
+	max_life = 1, life_rating = 4,  -- Solipsism will take care of hit points
+	autolevel = "wildcaster",
+	
+	ai = "tactical",
+	ai_state = { ai_target="target_player_radius", sense_radius=20, talent_in=3, },
+	dont_pass_target = true,
+	can_pass = {pass_wall=20},
+	levitation = 1,
+	
+	combat_armor = 1, combat_def = 5,
+	combat = { dam=resolvers.levelup(20, 1, 1.1), atk=10, apr=10, dammod={wil=1}, damtype=engine.DamageType.MIND},
+	
+	resolvers.talents{
+		[Talents.T_BACKLASH]={base=2, every=6, max=8},
+		[Talents.T_DISTORTION_BOLT]={base=2, every=6, max=8},
+	
+		[Talents.T_SOLIPSISM]=8,
+
+		[Talents.T_SLEEP]={base=2, every=6, max=8},
+		[Talents.T_LUCID_DREAMER]={base=2, every=6, max=8},
+		[Talents.T_DREAM_WALK]=5,
+	},
+
+	resolvers.sustains_at_birth(),
+	
+	-- Remove ourselves from the dream seed limit
+	on_die = function(self)
+		if self.summoner and self.summoner.dreamer_sleep_state then
+			self.summoner.dreamer_sleep_state = self.summoner.dreamer_sleep_state - 10
+		end
+	end,
+}
 ------------------------------------------------------------------------
 -- Uniques
 ------------------------------------------------------------------------
@@ -666,7 +826,7 @@ You can discern a huge round mouth covered in razor-sharp teeth.]],
 	combat = { dam=resolvers.levelup(resolvers.mbonus(100, 15), 1, 1), atk=500, apr=0, dammod={str=1.2} },
 
 	body = { INVEN = 10, MAINHAND=1, OFFHAND=1, BODY=1 },
-		  resolvers.drops{chance=100, nb=1, {unique=true} },
+	resolvers.drops{chance=100, nb=1, {unique=true} },
 	resolvers.drops{chance=100, nb=5, {ego_chance=100} },
 
 	resists = { all=500 },
@@ -757,7 +917,7 @@ newEntity{ base = "BASE_NPC_HORROR",
 	
 	resolvers.drops{chance=100, nb=1, {defined="BLADE_RIFT"} },
 	
-		ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=2, ally_compassion=0 },
+	ai = "tactical", ai_state = { ai_move="move_dmap", talent_in=2, ally_compassion=0 },
 		
 	on_melee_hit = {[DamageType.PHYSICALBLEED]=resolvers.mbonus(30, 4)},
 	combat = { dam=resolvers.levelup(resolvers.rngavg(20,28), 1, 1.5), physspeed = 0.25,atk=resolvers.levelup(24, 1.2, 1.2), apr=4, dammod={wil=0.3, cun=0.15}, damtype=engine.DamageType.PHYSICALBLEED, },
@@ -908,5 +1068,4 @@ newEntity{ base="BASE_NPC_HORROR", define_as = "DISTORTED_BLADE",
 			game.logSeen(self, "#AQUAMARINE#With the horror's death the chaotic blade clatters to the ground!")
 		end
 	end,
-}
-
+}
\ No newline at end of file
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/horror_eldritch_dream_seed.png b/game/modules/tome/data/gfx/shockbolt/npc/horror_eldritch_dream_seed.png
new file mode 100644
index 0000000000000000000000000000000000000000..b32c83d97aadb6bd783e97afba34df480ef03cc9
GIT binary patch
literal 4726
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%0DX^N!Er0cEr8c)Cg2vp9zp+B%m%Cq
zUwQztlF_mlEB6dc69Rx2w&fQjKlP*kFJeu;17m-m6N8qFN5~ZzcY80?3x3cTZv8IB
z^J8Bz0>H$IO-!qpxZ!oU3qMn@hXDUs%J;96)!(G`Hif{w9QVm+YZO?|4N8`y?&WOY
zMVb}30&v9|-v#+Gf~GV$agmhJ($KMC9G0rMMU9RXTkCl3qmCMNuTc+=MegJ9f~3HP
z7i@T;gMJHx`9Xh#w1?wC5%eB1!Dl7hCM-B)9AGR+(8-|W+Ni7*+vQC$C^gRu?O5Xr
zkbkLB2O4hDF7giK6<d6a2>~^lG&vRrFANnXjH!#`3Gk|n(J9f|kfz6ST6;j;_dkvK
zxb8CHk3|Yxyr7&FuFtf@^6tqT4lG7oCQbM)O?v<vIh?Ty>#f7iHuibA^`E1cL>$<r
zJ-V<|flO^HrEK^Zr?I#hPttLux3l%{GHc(Z^^gfa6oRh`zj|Jv@Pcy7Jl}TOEH0mo
z(E!-6#tV?2BzjsW&xu(iPLNR{&3Bxh5>duOMgXJj?E4COql12uxqiOxGM7KD%nPr&
z{<GuqpZxemNGC<C<j$EF<kWe}h|^`9*wX8yoDFRuQ#E(|YXGjjewUTEmvW2<y3Oj5
zIrUuucrC{tvq{kV?0#{18vaI(Ujz7crq~ZVFp<4bMJqWJe8JB!#|efi)#Q5sTlwQb
z3g7!rQYP6iy?Vy}gB-83&+7s7deB#yg=eAn8A)4;H#vTl!0_Zy)pJK?-gRoMJji4#
zP1Gl+J2c@oxqW_!q5)8A!mBLH%rgOhC_2$&QO*tpxaGi@H$1+2ZZP_T4^0(k2)J_i
zX&$Nj77bSkncGb0eC5Y%Y-Kmf-$M!KIz3a-)YZEz|8<Vdp*S0u@5|2uE1Ss3&kxP_
z`D}l6=D^gN5^`b3Vn3i{dhad#2OXR&%4S=6@S)fW848Qf_vKjb8m>$$n={ihN9p{z
zLlNKD1I~nlo$mCXjG&x5FKC7JxiZDQc1|#?0I#1rCilM9{~!gdC;?{%52`{~t{l6w
zGhmxraVh|~baOBF`aVSPucd%zaG`W-tPNJ2nMvTvVTL@kROED(b^X^7{2MFaOjzUt
z2<2hvwJQo7SWCZHDcJ=8pZ?Q7NdNfv_$EL6b3Z=+|JkSVj0zEfhX4Qo07*qoM6N<$
Ef?-g-MgRZ+

literal 0
HcmV?d00001

diff --git a/game/modules/tome/data/gfx/shockbolt/npc/horror_eldritch_dreaming_horror.png b/game/modules/tome/data/gfx/shockbolt/npc/horror_eldritch_dreaming_horror.png
new file mode 100644
index 0000000000000000000000000000000000000000..be39923cad14ba4edf2b2ab93fe6dc2bc19c88b8
GIT binary patch
literal 9482
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

literal 0
HcmV?d00001

diff --git a/game/modules/tome/data/talents/chronomancy/age-manipulation.lua b/game/modules/tome/data/talents/chronomancy/age-manipulation.lua
index 20ac8ff034..f38fc65041 100644
--- a/game/modules/tome/data/talents/chronomancy/age-manipulation.lua
+++ b/game/modules/tome/data/talents/chronomancy/age-manipulation.lua
@@ -157,7 +157,16 @@ newTalent{
 	points = 5,
 	paradox = 10,
 	cooldown = 10,
-	tactical = { HEAL = 2, CURE = 2 },
+	tactical = { HEAL = 2, CURE = function(self, t, target)
+		local nb = 0
+		for eff_id, p in pairs(self.tmp) do
+			local e = self.tempeffect_def[eff_id]
+			if e.status == "detrimental" and e.type == "physical" then
+				nb = nb + 1
+			end
+		end
+		return nb
+	end },
 	is_heal = true,
 	getHeal = function(self, t) return self:combatTalentSpellDamage(t, 40, 440)*getParadoxModifier(self, pm) end,
 	getRemoveCount = function(self, t) return math.floor(self:getTalentLevel(t)) end,
diff --git a/game/modules/tome/data/talents/psionic/distortion.lua b/game/modules/tome/data/talents/psionic/distortion.lua
index 8f342f4218..2483c75c7a 100644
--- a/game/modules/tome/data/talents/psionic/distortion.lua
+++ b/game/modules/tome/data/talents/psionic/distortion.lua
@@ -31,7 +31,7 @@ newTalent{
 	radius = function(self, t) return 1 + math.floor(self:getTalentLevel(t)/3) end,
 	requires_target = true,
 	getDamage = function(self, t) return self:combatTalentMindDamage(t, 10, 150) end,
-	getDetonateDamage = function(self, t) return self:combatTalentMindDamage(t, 20, 200) end,
+	getDetonateDamage = function(self, t) return self:combatTalentMindDamage(t, 20, 300) end,
 	target = function(self, t)
 		return {type="bolt", range=self:getTalentRange(t), talent=t, display={trail="distortion_trail"}}
 	end,
@@ -60,7 +60,9 @@ newTalent{
 	require = psi_wil_req2,
 	cooldown = 6,
 	psi = 10,
-	tactical = { ATTACK = { PHYSICAL = 2}, DISABLE = 2},
+	tactical = { ATTACK = { PHYSICAL = 2},
+		DISABLE = function(self, t, target) if target:hasEffect(target.EFF_DISTORION) then return 4 else return 2 end end,
+	},
 	range = 0,
 	radius = function(self, t) return 3 + math.ceil(self:getTalentLevel(t)/2) end,
 	requires_target = true,
@@ -97,7 +99,9 @@ newTalent{
 	require = psi_wil_req3,
 	cooldown = 12,
 	psi = 20,
-	tactical = { ATTACK = { PHYSICAL = 2}, DISABLE = 2},
+	tactical = { ATTACK = { PHYSICAL = 2},
+		DISABLE = function(self, t, target) if target:hasEffect(target.EFF_DISTORION) then return 4 else return 0 end end,
+	},
 	range = 10,
 	requires_target = true,
 	direct_hit = true,
diff --git a/game/modules/tome/data/talents/psionic/dreaming.lua b/game/modules/tome/data/talents/psionic/dreaming.lua
index f72626a2e5..31e72dfca9 100644
--- a/game/modules/tome/data/talents/psionic/dreaming.lua
+++ b/game/modules/tome/data/talents/psionic/dreaming.lua
@@ -142,7 +142,10 @@ newTalent{
 	require = psi_wil_req3,
 	psi= 10,
 	cooldown = 10,
-	tactical = { CLOSEIN = 2, ESCAPE = 2 },
+	tactical = { 
+		CLOSEIN = function(self, t, target) if target:attr("sleep") or (self:getTalentLevel(t) >= 5 and self:attr("lucid_dreamer")) then return 2 else return 0 end end,
+		ESCAPE = function(self, t, target) if target:attr("sleep") or (self:getTalentLevel(t) >= 5 and self:attr("lucid_dreamer")) then return 2 else return 0 end end,
+	},
 	range = function(self, t) return 5 + math.min(5, self:getTalentLevelRaw(t)) end,
 	requires_target = true,
 	target = function(self, t)
@@ -196,7 +199,7 @@ newTalent{
 	mode = "sustained",
 	sustain_psi = 40,
 	cooldown = 50,
-	tactical = { DISABLE=2 },
+	tactical = { DISABLE = function(self, t, target) if target:attr("sleep") then return 4 else return 0 end end},
 	range = function(self, t) return 5 + math.min(5, self:getTalentLevelRaw(t)) end,
 	requires_target = true,
 	target = function(self, t)
@@ -209,13 +212,11 @@ newTalent{
 		game:playSoundNear(self, "talents/spell_generic")
 		local ret = {
 			drain = self:addTemporaryValue("psi_regen", -t.getDrain(self, t)),
-			particles = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.8, density=60, life=14, instop=20, static=80}))
 		}
 		return ret
 	end,
 	deactivate = function(self, t, p)
 		self:removeTemporaryValue("psi_regen", p.drain)
-		self:removeParticles(p.particles)
 		return true
 	end,
 	info = function(self, t)
diff --git a/game/modules/tome/data/talents/psionic/mentalism.lua b/game/modules/tome/data/talents/psionic/mentalism.lua
index 0013ba7a31..b3239d44c2 100644
--- a/game/modules/tome/data/talents/psionic/mentalism.lua
+++ b/game/modules/tome/data/talents/psionic/mentalism.lua
@@ -64,7 +64,16 @@ newTalent{
 	require = psi_wil_req2,
 	psi = 15,
 	cooldown = 24,
-	tactical = { CURE=2},
+	tactical = { BUFF = 1, CURE = function(self, t, target)
+		local nb = 0
+		for eff_id, p in pairs(self.tmp) do
+			local e = self.tempeffect_def[eff_id]
+			if e.status == "detrimental" and e.type == "mental" then
+				nb = nb + 1
+			end
+		end
+		return nb
+	end,},
 	no_energy = true,
 	getRemoveCount = function(self, t) return math.ceil(self:getTalentLevel(t)) end,
 	action = function(self, t)
@@ -252,6 +261,11 @@ newTalent{
 			esp = self:addTemporaryValue("esp", {[target.type] = 1}),
 		}
 		
+		-- Update for ESP
+		game:onTickEnd(function() 
+			self:resetCanSeeCache()
+		end)
+		
 		return ret
 	end,
 	deactivate = function(self, t, p)
diff --git a/game/modules/tome/data/talents/psionic/nightmare.lua b/game/modules/tome/data/talents/psionic/nightmare.lua
index 986e04de99..b1ef267bed 100644
--- a/game/modules/tome/data/talents/psionic/nightmare.lua
+++ b/game/modules/tome/data/talents/psionic/nightmare.lua
@@ -95,7 +95,7 @@ newTalent{
 	range = 10,
 	direct_hit = true,
 	requires_target = true,
-	tactical = { ATTACK = 3 },
+	tactical = { ATTACK = function(self, t, target) if target:attr("sleep") then return 4 else return 2 end end },
 	getChance = function(self, t) return self:combatTalentMindDamage(t, 15, 50) end,
 	getDuration = function(self, t) return 2 + math.ceil(self:getTalentLevel(t) * 2) end,
 	summon_inner_demons = function(self, target, t)
@@ -220,7 +220,7 @@ newTalent{
 	range = 10,
 	direct_hit = true,
 	requires_target = true,
-	tactical = { ATTACK = { DARKNESS = 2 }, DISABLE = { confusion = 1, stun = 1, blind = 1 } },
+	tactical = { ATTACK = { DARKNESS = 2 }, DISABLE = function(self, t, target) if target:attr("sleep") then return 4 else return 2 end end },
 	getChance = function(self, t) return self:combatTalentMindDamage(t, 15, 50) end,
 	getDamage = function(self, t) return self:combatTalentMindDamage(t, 5, 50) end,
 	getDuration = function(self, t) return 4 + math.ceil(self:getTalentLevel(t)) end,
diff --git a/game/modules/tome/data/talents/psionic/psychic-assault.lua b/game/modules/tome/data/talents/psionic/psychic-assault.lua
index 5f73e9a4da..43efd7a9fa 100644
--- a/game/modules/tome/data/talents/psionic/psychic-assault.lua
+++ b/game/modules/tome/data/talents/psionic/psychic-assault.lua
@@ -18,77 +18,40 @@
 -- darkgod@te4.org
 
 newTalent{
-	name = "Sunder Mind",
+	name = "Mind Sear",
 	type = {"psionic/psychic-assault", 1},
 	require = psi_wil_req1,
 	points = 5,
 	cooldown = 2,
 	psi = 5,
-	tactical = { ATTACK = { MIND = 2}, DISABLE = 1},
 	range = 7,
+	direct_hit = true,
 	requires_target = true,
-	getDamage = function(self, t) return self:combatTalentMindDamage(t, 10, 150) end,
 	target = function(self, t)
-		return {type="hit", range=self:getTalentRange(t), talent=t}
+		return {type="beam", range=self:getTalentRange(t), talent=t}
 	end,
+	tactical = { ATTACKAREA = { MIND = 3 } },
+	getDamage = function(self, t) return self:combatTalentMindDamage(t, 10, 300) end,
 	action = function(self, t)
 		local tg = self:getTalentTarget(t)
 		local x, y = self:getTarget(tg)
 		if not x or not y then return nil end
-		local _ _, x, y = self:canProject(tg, x, y)
-		local target = game.level.map(x, y, Map.ACTOR)
-		if not target then return end
-		
-		local dam =self:mindCrit(t.getDamage(self, t))
-		self:project(tg, x, y, DamageType.MIND, {dam=dam, alwaysHit=true}, {type="mind"})
-		target:setEffect(target.EFF_SUNDER_MIND, 2, {power=dam/10})
-		
-		game:playSoundNear(self, "talents/warp")
+		self:project(tg, x, y, DamageType.MIND, self:mindCrit(t.getDamage(self, t)), {type="mind"})
+		game:playSoundNear(self, "talents/spell_generic")
 		return true
 	end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		local power = t.getDamage(self, t) / 10
-		return ([[Cripples the target's mind, inflicting %0.2f mind damage and reducing it's mental save by %d.  This attack always hits and the mental save reduction stacks.
-		The damage and save reduction will scale with your mindpower.]]):
-		format(damDesc(self, DamageType.MIND, (damage)), power)
+		return ([[Sends a telepathic attack, trying to destroy the brains of any target in the beam, doing %0.2f mind damage.
+		The damage will increase with your mindpower.]]):format(damDesc(self, DamageType.MIND, damage))
 	end,
 }
 
 newTalent{
-	name = "Synaptic Static",
+	name = "Psychic Lobotomy",
 	type = {"psionic/psychic-assault", 2},
 	require = psi_wil_req2,
 	points = 5,
-	cooldown = 10,
-	psi = 10,
-	range = 0,
-	direct_hit = true,
-	requires_target = true,
-	radius = function(self, t) return math.min(10, 3 + math.ceil(self:getTalentLevel(t)/2)) end,
-	target = function(self, t) return {type="ball", radius=self:getTalentRadius(t), range=self:getTalentRange(t), talent=t, selffire=false} end,
-	tactical = { ATTACKAREA = { MIND = 3 }, DISABLE=1 },
-	getDamage = function(self, t) return self:combatTalentMindDamage(t, 20, 200) end,
-	action = function(self, t)
-		local tg = self:getTalentTarget(t)
-		self:project(tg, self.x, self.y, DamageType.MIND, {dam=self:mindCrit(self:combatTalentMindDamage(t, 20, 200)), crossTierChance=100} )
-		game.level.map:particleEmitter(self.x, self.y, self:getTalentRadius(t), "generic_ball", {radius=self:getTalentRadius(t), rm=100, rM=125, gm=100, gM=125, bm=100, bM=125, am=200, aM=255})
-		game:playSoundNear(self, "talents/echo")
-		return true
-	end,
-	info = function(self, t)
-		local damage = t.getDamage(self, t)
-		local radius = self:getTalentRadius(t)
-		return ([[Sends out a blast of telepathic static in a %d radius, inflicting %0.2f mind damage.  This attack can brain-lock affected targets.
-		The damage will increase with your mindpower.]]):format(radius, damDesc(self, DamageType.MIND, damage))
-	end,
-}
-
-newTalent{
-	name = "Psychic Lobotomy",
-	type = {"psionic/psychic-assault", 3},
-	require = psi_wil_req3,
-	points = 5,
 	cooldown = 8,
 	range = 7,
 	psi = 10,
@@ -109,8 +72,9 @@ newTalent{
 		if not target then return nil end
 		local ai = target.ai or nil		
 		
+		local dam = self:mindCrit(t.getDamage(self, t))
 		if target:canBe("confused") then
-			target:setEffect(target.EFF_LOBOTOMIZED, t.getDuration(self, t), {src=self, dam=t.getDamage(self, t), power=t.getPower(self, t), apply_power=self:combatMindpower()})
+			target:setEffect(target.EFF_LOBOTOMIZED, t.getDuration(self, t), {src=self, dam=dam, power=t.getPower(self, t), apply_power=self:combatMindpower()})
 		else
 			game.logSeen(target, "%s resists the lobotomy!", target.name:capitalize())
 		end
@@ -130,31 +94,68 @@ newTalent{
 }
 
 newTalent{
-	name = "Mind Sear",
+	name = "Synaptic Static",
+	type = {"psionic/psychic-assault", 3},
+	require = psi_wil_req3,
+	points = 5,
+	cooldown = 10,
+	psi = 10,
+	range = 0,
+	direct_hit = true,
+	requires_target = true,
+	radius = function(self, t) return math.min(10, 3 + math.ceil(self:getTalentLevel(t)/2)) end,
+	target = function(self, t) return {type="ball", radius=self:getTalentRadius(t), range=self:getTalentRange(t), talent=t, selffire=false} end,
+	tactical = { ATTACKAREA = { MIND = 3 }, DISABLE=1 },
+	getDamage = function(self, t) return self:combatTalentMindDamage(t, 20, 200) end,
+	action = function(self, t)
+		local tg = self:getTalentTarget(t)
+		self:project(tg, self.x, self.y, DamageType.MIND, {dam=self:mindCrit(self:combatTalentMindDamage(t, 20, 200)), crossTierChance=100} )
+		game.level.map:particleEmitter(self.x, self.y, self:getTalentRadius(t), "generic_ball", {radius=self:getTalentRadius(t), rm=100, rM=125, gm=100, gM=125, bm=100, bM=125, am=200, aM=255})
+		game:playSoundNear(self, "talents/echo")
+		return true
+	end,
+	info = function(self, t)
+		local damage = t.getDamage(self, t)
+		local radius = self:getTalentRadius(t)
+		return ([[Sends out a blast of telepathic static in a %d radius, inflicting %0.2f mind damage.  This attack can brain-lock affected targets.
+		The damage will increase with your mindpower.]]):format(radius, damDesc(self, DamageType.MIND, damage))
+	end,
+}
+
+newTalent{
+	name = "Sunder Mind",
 	type = {"psionic/psychic-assault", 4},
 	require = psi_wil_req4,
 	points = 5,
 	cooldown = 2,
 	psi = 5,
+	tactical = { ATTACK = { MIND = 2}, DISABLE = 1},
 	range = 7,
-	direct_hit = true,
 	requires_target = true,
+	getDamage = function(self, t) return self:combatTalentMindDamage(t, 10, 300) end,
 	target = function(self, t)
-		return {type="beam", range=self:getTalentRange(t), talent=t}
+		return {type="hit", range=self:getTalentRange(t), talent=t}
 	end,
-	tactical = { ATTACKAREA = { MIND = 3 } },
-	getDamage = function(self, t) return self:combatTalentMindDamage(t, 10, 300) end,
 	action = function(self, t)
 		local tg = self:getTalentTarget(t)
 		local x, y = self:getTarget(tg)
 		if not x or not y then return nil end
-		self:project(tg, x, y, DamageType.MIND, self:mindCrit(t.getDamage(self, t)), {type="mind"})
-		game:playSoundNear(self, "talents/spell_generic")
+		local _ _, x, y = self:canProject(tg, x, y)
+		local target = game.level.map(x, y, Map.ACTOR)
+		if not target then return end
+		
+		local dam =self:mindCrit(t.getDamage(self, t))
+		self:project(tg, x, y, DamageType.MIND, {dam=dam, alwaysHit=true}, {type="mind"})
+		target:setEffect(target.EFF_SUNDER_MIND, 2, {power=dam/10})
+		
+		game:playSoundNear(self, "talents/warp")
 		return true
 	end,
 	info = function(self, t)
 		local damage = t.getDamage(self, t)
-		return ([[Sends a telepathic attack, trying to destroy the brains of any target in the beam, doing %0.2f mind damage.
-		The damage will increase with your mindpower.]]):format(damDesc(self, DamageType.MIND, damage))
+		local power = t.getDamage(self, t) / 10
+		return ([[Cripples the target's mind, inflicting %0.2f mind damage and reducing it's mental save by %d for 2 turns.  This attack always hits and the mental save reduction stacks.
+		The damage and save reduction will scale with your mindpower.]]):
+		format(damDesc(self, DamageType.MIND, (damage)), power)
 	end,
 }
\ No newline at end of file
diff --git a/game/modules/tome/data/talents/psionic/slumber.lua b/game/modules/tome/data/talents/psionic/slumber.lua
index 3b320948d4..5dc7a56fdb 100644
--- a/game/modules/tome/data/talents/psionic/slumber.lua
+++ b/game/modules/tome/data/talents/psionic/slumber.lua
@@ -116,7 +116,7 @@ newTalent{
 	require = psi_wil_req4,
 	cooldown = 24,
 	psi = 40,
-	tactical = { DISABLE = {sleep = 2} },
+	tactical = { DISABLE = function(self, t, target) if target:attr("sleep") then return 4 else return 0 end end},
 	direct_hit = true,
 	requires_target = true,
 	range = function(self, t) return 5 + math.min(5, self:getTalentLevelRaw(t)) end,
@@ -212,6 +212,12 @@ newTalent{
 			end
 
 			game.logPlayer(game.player, "#LIGHT_BLUE#You are taken to the Dreamscape!")
+			
+			-- Learn about solipsists
+			if target == game.player then
+				game:setAllowedBuild("psionic_solipsist", true)
+			end
+			
 		end)
 		
 		local power = self:mindCrit(t.getPower(self, t))
diff --git a/game/modules/tome/data/talents/psionic/solipsism.lua b/game/modules/tome/data/talents/psionic/solipsism.lua
index b8b808de5d..3965eceff2 100644
--- a/game/modules/tome/data/talents/psionic/solipsism.lua
+++ b/game/modules/tome/data/talents/psionic/solipsism.lua
@@ -27,13 +27,19 @@ newTalent{
 	getConversionRatio = function(self, t) return math.min(0.25 + self:getTalentLevel(t) * 0.1, 1) end,
 	on_learn = function(self, t)
 		if self:getTalentLevelRaw(t) == 1 then
-			self:incMaxPsi((self:getWil()-10) * 1)
-			self.max_life = self.max_life - (self:getCon()-10) * 0.5
 			self.inc_resource_multi.psi = (self.inc_resource_multi.psi or 0) + 1
 			self.inc_resource_multi.life = (self.inc_resource_multi.life or 0) - 0.5
 			self.life_rating = math.ceil(self.life_rating/2)
 			self.psi_rating =  self.psi_rating + 10
 			self.solipsism_threshold = (self.solipsism_threshold or 0) + 0.2
+			
+			-- Adjust the values onTickEnd for NPCs to make sure these table values are resolved
+			-- If we're not the player, we resetToFull to ensure correct values
+			game:onTickEnd(function()
+				self:incMaxPsi((self:getWil()-10) * 1)
+				self.max_life = self.max_life - (self:getCon()-10) * 0.5
+				if self ~= game.player then self:resetToFull() end
+			end)
 		end
 	end,
 	on_unlearn = function(self, t)
@@ -63,11 +69,16 @@ newTalent{
 	getBalanceRatio = function(self, t) return math.min(0.25 + self:getTalentLevel(t) * 0.1, 1) end,
 	on_learn = function(self, t)
 		if self:getTalentLevelRaw(t) == 1 then
-			self:incMaxPsi((self:getWil()-10) * 1)
-			self.max_life = self.max_life - (self:getCon()-10) * 0.5
 			self.inc_resource_multi.psi = (self.inc_resource_multi.psi or 0) + 1
 			self.inc_resource_multi.life = (self.inc_resource_multi.life or 0) - 0.5
 			self.solipsism_threshold = (self.solipsism_threshold or 0) + 0.1
+			-- Adjust the values onTickEnd for NPCs to make sure these table values are filled out
+			-- If we're not the player, we resetToFull to ensure correct values
+			game:onTickEnd(function()
+				self:incMaxPsi((self:getWil()-10) * 1)
+				self.max_life = self.max_life - (self:getCon()-10) * 0.5
+				if self ~= game.player then self:resetToFull() end
+			end)
 		end
 	end,
 	on_unlearn = function(self, t)
@@ -97,11 +108,16 @@ newTalent{
 	on_learn = function(self, t)
 		self.clarity_threshold = t.getClarityThreshold(self, t)
 		if self:getTalentLevelRaw(t) == 1 then
-			self:incMaxPsi((self:getWil()-10) * 1)
-			self.max_life = self.max_life - (self:getCon()-10) * 0.5
 			self.inc_resource_multi.psi = (self.inc_resource_multi.psi or 0) + 1
 			self.inc_resource_multi.life = (self.inc_resource_multi.life or 0) - 0.5
 			self.solipsism_threshold = (self.solipsism_threshold or 0) + 0.1
+			-- Adjust the values onTickEnd for NPCs to make sure these table values are resolved
+			-- If we're not the player, we resetToFull to ensure correct values
+			game:onTickEnd(function()
+				self:incMaxPsi((self:getWil()-10) * 1)
+				self.max_life = self.max_life - (self:getCon()-10) * 0.5
+				if self ~= game.player then self:resetToFull() end
+			end)
 		end
 	end,
 	on_unlearn = function(self, t)
@@ -133,11 +149,16 @@ newTalent{
 	getSavePercentage = function(self, t) return math.min(0.4 + self:getTalentLevel(t) * 0.16, 1.5) end,
 	on_learn = function(self, t)
 		if self:getTalentLevelRaw(t) == 1 then
-			self:incMaxPsi((self:getWil()-10) * 1)
-			self.max_life = self.max_life - (self:getCon()-10) * 0.5
 			self.inc_resource_multi.psi = (self.inc_resource_multi.psi or 0) + 1
 			self.inc_resource_multi.life = (self.inc_resource_multi.life or 0) - 0.5
 			self.solipsism_threshold = (self.solipsism_threshold or 0) + 0.1
+			-- Adjust the values onTickEnd for NPCs to make sure these table values are resolved
+			-- If we're not the player, we resetToFull to ensure correct values
+			game:onTickEnd(function()
+				self:incMaxPsi((self:getWil()-10) * 1)
+				self.max_life = self.max_life - (self:getCon()-10) * 0.5
+				if self ~= game.player then self:resetToFull() end
+			end)
 		end
 	end,
 	on_unlearn = function(self, t)
diff --git a/game/modules/tome/data/talents/psionic/thought-forms.lua b/game/modules/tome/data/talents/psionic/thought-forms.lua
index 5af9d16220..28cd2e8544 100644
--- a/game/modules/tome/data/talents/psionic/thought-forms.lua
+++ b/game/modules/tome/data/talents/psionic/thought-forms.lua
@@ -80,7 +80,10 @@ newTalent{
 			-- Keep them on a leash
 			on_act = function(self)
 				local t = self.summoner:getTalentFromId(self.summoner.T_TF_BOWMAN)
-				if not game.level:hasEntity(self.summoner) or core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) > self.summoner:getTalentRange(t) then
+				if not game.level:hasEntity(self.summoner) or self.summoner.dead or not self.summoner:isTalentActive(self.summoner.T_TF_BOWMAN) then
+					self:die(self)
+				end
+				if game.level:hasEntity(self.summoner) and core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) > self.summoner:getTalentRange(t) then
 					local Map = require "engine.Map"
 					local x, y = util.findFreeGrid(self.summoner.x, self.summoner.y, 5, true, {[Map.ACTOR]=true})
 					if not x then
@@ -143,7 +146,9 @@ newTalent{
 		return ret
 	end,
 	deactivate = function(self, t, p)
-		p.summon:die(p.summon)
+		if p.summon and p.summon.summoner == self then
+			p.summon:die(p.summon)
+		end
 		if p.speed then self:removeTemporaryValue("combat_mindspeed", p.speed) end
 		return true
 	end,
@@ -216,7 +221,10 @@ newTalent{
 			-- Keep them on a leash
 			on_act = function(self)
 				local t = self.summoner:getTalentFromId(self.summoner.T_TF_WARRIOR)
-				if not game.level:hasEntity(self.summoner) or core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) > self.summoner:getTalentRange(t) then
+				if not game.level:hasEntity(self.summoner) or self.summoner.dead or not self.summoner:isTalentActive(self.summoner.T_TF_WARRIOR) then
+					self:die(self)
+				end
+				if game.level:hasEntity(self.summoner) and core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) > self.summoner:getTalentRange(t) then
 					local Map = require "engine.Map"
 					local x, y = util.findFreeGrid(self.summoner.x, self.summoner.y, 5, true, {[Map.ACTOR]=true})
 					if not x then
@@ -278,7 +286,9 @@ newTalent{
 		return ret
 	end,
 	deactivate = function(self, t, p)
-		p.summon:die(p.summon)
+		if p.summon and p.summon.summoner == self then
+			p.summon:die(p.summon)
+		end
 		if p.power then self:removeTemporaryValue("combat_mindpower", p.power) end
 		return true
 	end,
@@ -351,7 +361,10 @@ newTalent{
 			-- Keep them on a leash
 			on_act = function(self)
 				local t = self.summoner:getTalentFromId(self.summoner.T_TF_DEFENDER)
-				if not game.level:hasEntity(self.summoner) or core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) > self.summoner:getTalentRange(t) then
+				if not game.level:hasEntity(self.summoner) or self.summoner.dead or not self.summoner:isTalentActive(self.summoner.T_TF_DEFENDER) then
+					self:die(self)
+				end
+				if game.level:hasEntity(self.summoner) and core.fov.distance(self.x, self.y, self.summoner.x, self.summoner.y) > self.summoner:getTalentRange(t) then
 					local Map = require "engine.Map"
 					local x, y = util.findFreeGrid(self.summoner.x, self.summoner.y, 5, true, {[Map.ACTOR]=true})
 					if not x then
@@ -415,7 +428,9 @@ newTalent{
 		return ret
 	end,
 	deactivate = function(self, t, p)
-		p.summon:die(p.summon)
+		if p.summon and p.summon.summoner == self then
+			p.summon:die(p.summon)
+		end
 		if p.resist then self:removeTemporaryValue("resists", p.resist) end
 		return true
 	end,
@@ -464,7 +479,7 @@ newTalent{
 		local bonus = t.getStatBonus(self, t)
 		local range = self:getTalentRange(t)
 		return([[Forge a guardian from your thoughts alone.  Your guardian's primary stat will be improved by %d and it's two secondary stats by %d.
-		At talent level one you may forge a mighty bowman clad in leather armor, at level three a powerful warrior wielding a two-handed weapon, and at level five a strong defender using a sword and shield, and at talent level 5 
+		At talent level one you may forge a mighty bowman clad in leather armor, at level three a powerful warrior wielding a two-handed weapon, and at level five a strong defender using a sword and shield.
 		Thought forms can only be maintained up to a range of %d and will rematerialize next to you if this range is exceeded.
 		Only one thought-form may be active at a time and the stat bonuses will improve with your mindpower.]]):format(bonus, bonus/2, range)
 	end,
diff --git a/game/modules/tome/data/texts/unlock-psionic_solipsist.lua b/game/modules/tome/data/texts/unlock-psionic_solipsist.lua
new file mode 100644
index 0000000000..900bda75b9
--- /dev/null
+++ b/game/modules/tome/data/texts/unlock-psionic_solipsist.lua
@@ -0,0 +1,36 @@
+-- ToME - Tales of Maj'Eyal
+-- Copyright (C) 2009, 2010, 2011, 2012 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 "New Class: #LIGHT_GREEN#Solipsist (Psionic)",
+[[Solipsists are powerful psionicists that believe that the world is made up of nothing more than the thoughts and dreams of those that live in it.
+This power does not come without a price however, and the Solipsist must constantly fight with their own ego in order to keep a clear view of reality, lest they fall into a state of solipsism, the belief that the world and those that live in it is nothing more than a figment of thier mind.
+
+You've experienced the power of dreams first hand and may now create characters with the #LIGHT_GREEN#Solipsist class#WHITE#.
+
+Solipsists use the power of thought and dreams to manipulate the world around them.
+Class features:#YELLOW#
+- Distort the fabric of reality
+- Store and discharge psionic feedback
+- Summon powerful warriors birthed from your own consciousness
+- Convert damage you take into Psi damage and keep yourself alive with your mental reserves
+- Put your foes to sleep, enter their dreams, and become their worst nightmare#WHITE#
+
+Solipsists use their mind to manipulate the world around them.
+They require energy to do so, which they recover naturally over time and through methods others use to heal the body.
+]]
diff --git a/game/modules/tome/data/timed_effects/mental.lua b/game/modules/tome/data/timed_effects/mental.lua
index e7edf0c872..1f5f7d9c15 100644
--- a/game/modules/tome/data/timed_effects/mental.lua
+++ b/game/modules/tome/data/timed_effects/mental.lua
@@ -2273,7 +2273,7 @@ newEffect{
 	on_lose = function(self, err) return "#Target#'s regains it's senses.", "-Lobotomized" end,
 	parameters = { power=1, dam=1 },
 	activate = function(self, eff)
-		DamageType:get(DamageType.MIND).projector(eff.src or self, self.x, self.y, DamageType.MIND, eff.src:mindCrit(eff.dam))
+		DamageType:get(DamageType.MIND).projector(eff.src or self, self.x, self.y, DamageType.MIND, {dam=eff.dam, alwaysHit=true})
 		eff.power = math.max(eff.power - (self:attr("confusion_immune") or 0) * 100, 10)
 		eff.tmpid = self:addTemporaryValue("confused", eff.power)
 		eff.cid = self:addTemporaryValue("inc_stats", {[Stats.STAT_CUN]=-eff.power/2})
@@ -2345,10 +2345,22 @@ newEffect{
 	parameters = { power=100 },
 	on_gain = function(self, err) return "A psychic field forms around #target#.", "+Resonance Shield" end,
 	on_lose = function(self, err) return "The psychic field around #target# crumbles.", "-Resonance Shield" end,
+	damage_feedback = function(self, eff, src, value)
+		if eff.particle and eff.particle._shader and eff.particle._shader.shad and src and src.x and src.y then
+			local r = -rng.float(0.2, 0.4)
+			local a = math.atan2(src.y - self.y, src.x - self.x)
+			eff.particle._shader:setUniform("impact", {math.cos(a) * r, math.sin(a) * r})
+			eff.particle._shader:setUniform("impact_tick", core.game.getTime())
+		end
+	end,
 	activate = function(self, eff)
 		self.resonance_field_absorb = eff.power
-		eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=255, rM=255, gm=180, gM=255, bm=0, bM=0, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
 		eff.sid = self:addTemporaryValue("resonance_field", eff.power)
+		if core.shader.active() then
+			eff.particle = self:addParticles(Particles.new("shader_shield", 1, {img="shield2", size_factor=1.25}, {type="shield", time_factor=6000, color={1, 1, 0}}))
+		else
+			eff.particle = self:addParticles(Particles.new("time_shield_bubble", 1))
+		end
 	end,
 	deactivate = function(self, eff)
 		self.resonance_field_absorb = nil
@@ -2433,7 +2445,11 @@ newEffect{
 		if dream_prison then
 			eff.dur = eff.dur + 1
 			if not eff.particle then
-				eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+				if core.shader.active() then
+					eff.particle = self:addParticles(Particles.new("shader_shield", 1, {img="shield2", size_factor=1.25}, {type="shield", time_factor=6000, aadjust=5, color={0, 1, 1}}))
+				else
+					eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+				end
 			end
 		elseif eff.contagious > 0 and eff.dur > 1 then
 			local t = eff.src:getTalentFromId(eff.src.T_SLEEP)
@@ -2443,7 +2459,7 @@ newEffect{
 			self:removeParticles(eff.particle)
 		end
 		-- Incriment Insomnia Duration
-		eff.insomnia_duration = eff.insomnia_duration + 1
+		eff.insomnia_duration = math.min(eff.insomnia_duration + 1, 10)
 	end,
 	activate = function(self, eff)
 		eff.insomnia_duration = 0
@@ -2485,13 +2501,17 @@ newEffect{
 		if dream_prison then
 			eff.dur = eff.dur + 1
 			if not eff.particle then
-				eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+				if core.shader.active() then
+					eff.particle = self:addParticles(Particles.new("shader_shield", 1, {img="shield2", size_factor=1.25}, {type="shield", time_factor=6000, aadjust=5, color={0, 1, 1}}))
+				else
+					eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+				end
 			end
 		elseif eff.particle and not dream_prison then
 			self:removeParticles(eff.particle)
 		end
 		-- Incriment Insomnia Duration
-		eff.insomnia_duration = eff.insomnia_duration + 1
+		eff.insomnia_duration = math.min(eff.insomnia_duration + 1, 10)
 	end,
 	activate = function(self, eff)
 		eff.insomnia_duration = 0
@@ -2533,7 +2553,11 @@ newEffect{
 		if dream_prison then
 			eff.dur = eff.dur + 1
 			if not eff.particle then
-				eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+				if core.shader.active() then
+					eff.particle = self:addParticles(Particles.new("shader_shield", 1, {img="shield2", size_factor=1.25}, {type="shield", aadjust=5, color={0, 1, 1}}))
+				else
+					eff.particle = self:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
+				end
 			end
 		else
 			-- Store the power for later
@@ -2548,7 +2572,7 @@ newEffect{
 			self:removeParticles(eff.particle)
 		end
 		-- Incriment Insomnia Duration
-		eff.insomnia_duration = eff.insomnia_duration + 1
+		eff.insomnia_duration = math.min(eff.insomnia_duration + 1, 10)
 	end,
 	activate = function(self, eff)
 		eff.insomnia_duration = 0
diff --git a/game/modules/tome/data/timed_effects/other.lua b/game/modules/tome/data/timed_effects/other.lua
index 6bf3208081..8364315403 100644
--- a/game/modules/tome/data/timed_effects/other.lua
+++ b/game/modules/tome/data/timed_effects/other.lua
@@ -1557,7 +1557,7 @@ newEffect{
 newEffect{
 	name = "SOLIPSISM", image = "talents/solipsism.png",
 	desc = "Solipsism",
-	long_desc = function(self, eff) return ("This creature has fallen into a solipsistic state and is caught up in its own thoughts (-%d%% global speed)."):format(eff.power * 100) end,
+	long_desc = function(self, eff) return ("This creature has fallen into a solipsistic state and is caught up in its own egoic thoughts (-%d%% global speed)."):format(eff.power * 100) end,
 	type = "other",
 	subtype = { psionic=true },
 	status = "detrimental",
@@ -1575,7 +1575,7 @@ newEffect{
 newEffect{
 	name = "CLARITY", image = "talents/clarity.png",
 	desc = "Clarity",
-	long_desc = function(self, eff) return ("The creature has found a state of clarity (+%d%% global speed)."):format(eff.power * 100) end,
+	long_desc = function(self, eff) return ("The creature has found a state of clarity and sees the world for what it is (+%d%% global speed)."):format(eff.power * 100) end,
 	type = "other",
 	subtype = { psionic=true },
 	status = "beneficial",
@@ -1616,14 +1616,16 @@ newEffect{
 			-- Create a clone for later spawning
 			local m = require("mod.class.NPC").new(eff.target:clone{
 				shader = "shadow_simulacrum",
-				shader_args = { color = {0.0, 0.4, 0.8}, base = 0.6 },
+				shader_args = { color = {0.0, 1, 1}, base = 0.6 },
 				no_drops = true,
 				faction = eff.target.faction,
 				summoner = eff.target, summoner_gain_exp=true,
 				ai_target = {actor=nil},
 				ai = "summoned", ai_real = "tactical",
+				ai_state = eff.target.ai_state or { ai_move="move_dmap", talent_in=1 },
 				name = eff.target.name.."'s dream projection",
 			})
+			m.ai_state.ally_compassion = 10
 			m:removeAllMOs()
 			m.make_escort = nil
 			m.on_added_to_level = nil
@@ -1659,8 +1661,11 @@ newEffect{
 			-- track number killed
 			m.on_die = function(self, who)
 				if who then
-					local p = who:hasEffect(who.EFF_DREAMSCAPE) or who.summoner:hasEffect(who.summoner.EFF_DREAMSCAPE)
-					p.projections_killed = p.projections_killed + 1
+					local p = who:hasEffect(who.EFF_DREAMSCAPE)
+					if p then -- For the rare instance we die after the effect ends but before the dreamscape instance closes
+						p.projections_killed = p.projections_killed + 1
+						game.logSeen(p.target, "#LIGHT_RED#%s writhes in agony as a fragment of it's mind is destroyed!", p.target.name:capitalize())
+					end
 				end
 			end
 
@@ -1679,6 +1684,11 @@ newEffect{
 					m:resetCanSeeCache()
 				end
 			end
+			
+			-- Try to insure the AI isn't attacking the invulnerable actor
+			if self.ai_target and self.ai_target.actor and self.ai_target.actor:attr("invulnerable") then
+				self.ai_target.actor = nil
+			end
 		end
 	end,
 	activate = function(self, eff)
@@ -1687,8 +1697,13 @@ newEffect{
 		eff.sid = eff.target:addTemporaryValue("time_prison", 1)
 		eff.tid = eff.target:addTemporaryValue("no_timeflow", 1)
 		eff.imid = eff.target:addTemporaryValue("status_effect_immune", 1)
-		eff.particle = eff.target:addParticles(engine.Particles.new("ultrashield", 1, {rm=0, rM=0, gm=180, gM=255, bm=180, bM=255, am=70, aM=180, radius=0.4, density=60, life=14, instop=1, static=100}))
 		eff.target.energy.value = 0
+		if core.shader.active() then
+			eff.particle = eff.target:addParticles(Particles.new("shader_shield", 1, {img="shield2", size_factor=1.25}, {type="shield", time_factor=6000, aadjust=5, color={0, 1, 1}}))
+		else
+			eff.particle = eff.target:addParticles(Particles.new("damage_shield", 1))
+		end
+		
 		-- Make the invader deadly
 		eff.pid = self:addTemporaryValue("inc_damage", {all=eff.power})
 		eff.did = self:addTemporaryValue("lucid_dreamer", 1)
@@ -1772,8 +1787,13 @@ newEffect{
 
 			-- Apply Dreamscape hit
 			if eff.projections_killed > 0 then
-				eff.target:takeHit(eff.target.max_life/10 * eff.projections_killed, self)
-				eff.target:setEffect(eff.target.EFF_BRAINLOCKED, eff.projections_killed, {})
+				local kills = eff.projections_killed
+				eff.target:takeHit(eff.target.max_life/10 * kills, self)
+				eff.target:setEffect(eff.target.EFF_BRAINLOCKED, kills, {})
+				
+				local loss = "loss"
+				if kills >= 10 then loss = "potentially fatal loss" elseif kills >=8 then loss = "devastating loss" elseif kills >=6 then loss = "tremendous loss" elseif kills >=4 then loss = "terrible loss" end
+				game.logSeen(eff.target, "#LIGHT_RED#%s suffered a %s of self in the Dreamscape!", eff.target.name:capitalize(), loss)
 			end
 		end)
 	end,
diff --git a/game/modules/tome/data/zones/dreamscape-talent/zone.lua b/game/modules/tome/data/zones/dreamscape-talent/zone.lua
index 77d915b063..d9596b8510 100644
--- a/game/modules/tome/data/zones/dreamscape-talent/zone.lua
+++ b/game/modules/tome/data/zones/dreamscape-talent/zone.lua
@@ -46,7 +46,7 @@ return {
 		},
 	},
 	post_process = function(level)
-		game.state:makeWeather(level, 6, {max_nb=2, chance=1, dir=120, speed={0.1, 0.9}, alpha={0.2, 0.4}, particle_name="weather/grey_cloud_%02d"})
+		game.state:makeWeather(level, 6, {max_nb=2, chance=1, dir=120, r=0.8, g=0.4, b=0.8, speed={0.1, 0.9}, alpha={0.2, 0.4}, particle_name="weather/grey_cloud_%02d"})
 	end,
 	foreground = function(level, dx, dx, nb_keyframes)
 		local tick = core.game.getTime()
-- 
GitLab