diff --git a/game/engines/default/engine/interface/ActorAI.lua b/game/engines/default/engine/interface/ActorAI.lua
index ba2f754c4d98973526e6ab53311831484c64ff6d..8c88d41165b3736125555fa9730f378b4eb48d0f 100644
--- a/game/engines/default/engine/interface/ActorAI.lua
+++ b/game/engines/default/engine/interface/ActorAI.lua
@@ -123,7 +123,7 @@ function _M:doAI()
 	local target_pos = self.ai_target.actor and self.fov and self.fov.actors and self.fov.actors[self.ai_target.actor]
 	if target_pos then
 		local tx, ty = self:aiSeeTargetPos(self.ai_target.actor)
-		self.ai_state.target_last_seen = {x=tx, y=ty, turn=self.fov_last_turn}
+		self.ai_state.target_last_seen=table.merge(self.ai_state.target_last_seen or {}, {x=tx, y=ty, turn=self.fov_last_turn}) -- Merge to keep obfuscation data
 	end
 
 	return self:runAI(self.ai)
@@ -148,39 +148,43 @@ function _M:setTarget(target, last_seen)
 		self.ai_state.target_last_seen = last_seen
 	else
 		local target_pos = target and self.fov and self.fov.actors and self.fov.actors[self.ai_target.actor] or {x=self.x, y=self.y}
-		self.ai_state.target_last_seen = {x=target_pos.x, y=target_pos.y, turn=game.turn}
+		self.ai_state.target_last_seen=table.merge(self.ai_state.target_last_seen or {}, {x=target_pos.x, y=target_pos.y, turn=game.turn}) -- Merge to keep obfuscation data
 	end
 end
 
 --- Returns the seen coords of the target
 -- This will usually return the exact coords, but if the target is only partially visible (or not at all)
--- it will return estimates, to throw the AI a bit off
+-- it will return estimates, to throw the AI a bit off (up to 10 tiles error)
 -- @param target the target we are tracking
 -- @return x, y coords to move/cast to
 function _M:aiSeeTargetPos(target)
 	if not target then return self.x, self.y end
 	local tx, ty = target.x, target.y
+	local LSeen = self.ai_state.target_last_seen
+	if type(LSeen) ~= "table" then return tx, ty end
 	local spread = 0
+	LSeen.GCache_turn = LSeen.GCache_turn or game.turn -- Guess Cache turn to update position guess (so it's consistent during a turn)
+	LSeen.GCknown_turn = LSeen.GCknown_turn or game.turn -- Guess Cache known turn for spread calculation (self.ai_state.target_last_seen.turn can't be used because it's needed by FOV code)
 
-	-- Adding some type-safety checks, but this isn't fixing the source of the errors
-	if target == self.ai_target.actor and self.ai_state.target_last_seen and type(self.ai_state.target_last_seen) == "table" and self.ai_state.target_last_seen.x and not self:hasLOS(self.ai_state.target_last_seen.x, self.ai_state.target_last_seen.y) then
-		tx, ty = self.ai_state.target_last_seen.x, self.ai_state.target_last_seen.y
-		spread = spread + math.floor((game.turn - (self.ai_state.target_last_seen.turn or game.turn)) / (game.energy_to_act / game.energy_per_tick))
-	end
-	
+	-- Check if target is currently seen
 	local see, chance = self:canSee(target)
-
-	-- Compute the maximum spread if we need to obfuscate 
-	local spread = see and 0 or math.floor((100 - chance) / 10)
-	
-
-	-- We don't know the exact position, so we obfuscate
-	if spread > 0 then
-		tx = tx + rng.range(0, spread * 2) - spread
-		ty = ty + rng.range(0, spread * 2) - spread
-		return util.bound(tx, 0, game.level.map.w - 1), util.bound(ty, 0, game.level.map.h - 1)
-	-- Directly seeing it, no spread at all
+	if see and self:hasLOS(target.x, target.y) then -- canSee doesn't check LOS
+		LSeen.GCache_x, LSeen.GCache_y = nil, nil
+		LSeen.GCknown_turn = game.turn
+		LSeen.GCache_turn = game.turn
 	else
-		return util.bound(tx, 0, game.level.map.w - 1), util.bound(ty, 0, game.level.map.h - 1)
+		if target == self.ai_target.actor and (LSeen.GCache_turn or 0) + 10 <= game.turn and LSeen.x then
+			spread = spread + math.min(10, math.floor((game.turn - (LSeen.GCknown_turn or game.turn)) / (game.energy_to_act / game.energy_per_tick))) -- Limit spread to 10 tiles
+			tx, ty = util.bound(tx + rng.range(-spread, spread), 0, game.level.map.w - 1), util.bound(ty + rng.range(-spread, spread), 0, game.level.map.h - 1)
+			-- Inertial average with last guess: can specify another method here to make the targeting position less random
+			if LSeen.GCache_x then -- update guess with new random position. Could use util.findFreeGrid here at cost of speed
+				tx = math.floor(LSeen.GCache_x + (tx-LSeen.GCache_x)/2)
+				ty = math.floor(LSeen.GCache_y + (ty-LSeen.GCache_y)/2)
+			end
+			LSeen.GCache_x, LSeen.GCache_y = tx, ty
+			LSeen.GCache_turn = game.turn
+		end
+		if LSeen.GCache_x then return LSeen.GCache_x, LSeen.GCache_y end
 	end
-end
+	return tx, ty -- Fall through to correct coords
+end
\ No newline at end of file
diff --git a/game/modules/tome/class/NPC.lua b/game/modules/tome/class/NPC.lua
index be6ebd9468f1601c24d7b77ba0b7fea644cc724a..f3e6edbe59f52f773517ebf0d5dc079f97a48fe2 100644
--- a/game/modules/tome/class/NPC.lua
+++ b/game/modules/tome/class/NPC.lua
@@ -409,14 +409,28 @@ function _M:tooltip(x, y, seen_by)
 	local str = mod.class.Actor.tooltip(self, x, y, seen_by)
 	if not str then return end
 	local killed = game:getPlayer(true).all_kills and (game:getPlayer(true).all_kills[self.name] or 0) or 0
-
+	local target = self.ai_target.actor
+	
 	str:add(
 		true,
 		("Killed by you: %s"):format(killed), true,
-		"Target: ", self.ai_target.actor and self.ai_target.actor.name or "none"
+		"Target: ", target and target.name or "none"
 	)
-	if config.settings.cheat then str:add(true, "UID: "..self.uid, true, self.image) end
-
+	-- Give hints to stealthed/invisible players about where the NPC is looking (if they have LOS)
+	if target == game.player and (game.player:attr("stealth") or game.player:attr("invisible")) and game.player:hasLOS(self.x, self.y) then
+		local tx, ty = self:aiSeeTargetPos(self.ai_target.actor)
+		local dx, dy = tx - self.ai_target.actor.x, ty - self.ai_target.actor.y
+		local offset = engine.Map:compassDirection(dx, dy)
+		if offset then
+			str:add(" looking " ..offset)
+			if config.settings.cheat then str:add((" (%+d, %+d)"):format(dx, dy)) end
+		else
+			str:add(" looking at you.")
+		end
+	end
+	if config.settings.cheat then
+		str:add(true, "UID: "..self.uid, true, self.image)
+	end
 	return str
 end
 
diff --git a/game/modules/tome/class/Player.lua b/game/modules/tome/class/Player.lua
index f6224d44c7e9fec1cde6c834fc0498a8c088c30a..216ffe6468de26a3efe307850edfe43e1d62ec61 100644
--- a/game/modules/tome/class/Player.lua
+++ b/game/modules/tome/class/Player.lua
@@ -154,6 +154,14 @@ function _M:onEnterLevel(zone, level)
 		end
 	end
 	for i, eff_id in ipairs(effs) do self:removeEffect(eff_id) end
+
+	-- Clear existing player created effects on the map
+	for i, eff in ipairs(level.map.effects) do
+		if eff.src and eff.src == game.player then
+			eff.duration = 0
+			eff.grids = {}
+		end
+	end
 end
 
 function _M:onEnterLevelEnd(zone, level)