diff --git a/game/engine/Dialog.lua b/game/engine/Dialog.lua
index fedcaba11ddf5843498bbcf84be9a0b1250b74b3..c96cfbebd3be95eaa7f13ca1adff46dd4abfddee 100644
--- a/game/engine/Dialog.lua
+++ b/game/engine/Dialog.lua
@@ -45,6 +45,40 @@ function _M:simplePopup(title, text, fct, no_leave)
 	return d
 end
 
+--- Requests a simple yes-no dialog
+function _M:yesnoPopup(title, text, fct)
+	local font = core.display.newFont("/data/font/Vera.ttf", 12)
+	local w, h = font:size(text)
+	local tw, th = font:size(title)
+	local d = new(title, math.max(w, tw) + 8, h + 75, nil, nil, nil, font)
+	d.sel = 0
+	d:keyCommands({},
+	{
+		ACCEPT = function() game:unregisterDialog(d) if fct then fct(d.sel == 0) end end,
+		MOVE_LEFT = "MOVE_UP",
+		MOVE_RIGHT = "MOVE_DOWN",
+		MOVE_UP = function() d.sel = 0 d.changed = true end,
+		MOVE_DOWN = function() d.sel = 1 d.changed = true end,
+	})
+	d:mouseZones{{x=2, y=0, w=d.iw, h=d.ih, fct=function(b, _, _, _, _, x, y)
+		d.sel = (x < d.iw / 2) and 0 or 1
+		d.changed = true
+		if b ~= "none" then game:unregisterDialog(d) if fct then fct(d.sel == 0) end end
+	end}}
+	d.drawDialog = function(self, s)
+		s:drawColorStringCentered(self.font, text, 2, 2, self.iw - 2, 25 - 2)
+		if d.sel == 0 then
+			s:drawColorStringCentered(self.font, "Yes", 2, 25, self.iw / 2 - 2, 50 - 2, 0, 255, 255)
+			s:drawColorStringCentered(self.font, "No", 2 + self.iw / 2, 25, self.iw / 2 - 2, 50 - 2, 255, 255, 255)
+		else
+			s:drawColorStringCentered(self.font, "Yes", 2, 25, self.iw / 2 - 2, 50 - 2, 255, 255, 255)
+			s:drawColorStringCentered(self.font, "No", 2 + self.iw / 2, 25, self.iw / 2 - 2, 50 - 2, 0, 255, 255)
+		end
+	end
+	game:registerDialog(d)
+	return d
+end
+
 --- Create a Dialog
 function _M:init(title, w, h, x, y, alpha, font)
 	self.title = title
diff --git a/game/engine/Entity.lua b/game/engine/Entity.lua
index ef32a2976aedab8640a887bc9698ea5b17e5d753..a95846faeea0cd700ffab819405a5a8799c0d39e 100644
--- a/game/engine/Entity.lua
+++ b/game/engine/Entity.lua
@@ -241,8 +241,8 @@ function _M:loadList(file, no_default, res, mod)
 			-- Do we inherit things ?
 			if t.base then
 				-- Append array part
-				for i = 1, #t.base do
-					local b = t.base[i]
+				for i = 1, #res[t.base] do
+					local b = res[t.base][i]
 					if type(b) == "table" and not b.__CLASSNAME then b = table.clone(b, true)
 					elseif type(b) == "table" and b.__CLASSNAME then b = b:clone()
 					end
diff --git a/game/engine/Zone.lua b/game/engine/Zone.lua
index 897b286a725f93f75b2c353f84c0ef8b9b3935b0..508a3fe3d55ccc28eef3227f7da3939315853272 100644
--- a/game/engine/Zone.lua
+++ b/game/engine/Zone.lua
@@ -294,6 +294,7 @@ function _M:addEntity(level, e, typ, x, y)
 	elseif typ == "terrain" then
 		if x and y then level.map(x, y, Map.TERRAIN, e) end
 	end
+	e:check("on_added", level, x, y)
 end
 
 function _M:load(dynamic)
diff --git a/game/engine/generator/map/TileSet.lua b/game/engine/generator/map/TileSet.lua
index 840d97c9732764aebb25d8466abdbb982ac6856d..32f66e552022f913d01ce7911533feba30b0c6b8 100644
--- a/game/engine/generator/map/TileSet.lua
+++ b/game/engine/generator/map/TileSet.lua
@@ -22,6 +22,9 @@ local Map = require "engine.Map"
 require "engine.Generator"
 module(..., package.seeall, class.inherit(engine.Generator))
 
+-- Deactaivte too many prints
+local print = function() end
+
 function _M:init(zone, map, level, data)
 	engine.Generator.init(self, zone, map, level)
 	self.data = data
diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua
index 6e6163f393a7a8a106531b5d1e32c5abe99e5296..3daac2c678ca1b2d752d3c2667dbb23f8d38ffb7 100644
--- a/game/modules/tome/class/Actor.lua
+++ b/game/modules/tome/class/Actor.lua
@@ -859,6 +859,7 @@ function _M:canBe(what)
 	if what == "fear" and rng.percent(100 * (self:attr("fear_immune") or 0)) then return false end
 	if what == "knockback" and rng.percent(100 * (self:attr("knockback_immune") or 0)) then return false end
 	if what == "instakill" and rng.percent(100 * (self:attr("instakill_immune") or 0)) then return false end
+	if what == "worldport" and game.zone.no_worldport then return false end
 	return true
 end
 
diff --git a/game/modules/tome/class/Game.lua b/game/modules/tome/class/Game.lua
index de98af7ff45edc6fc05f0c86b477ff01eeb4fd3b..055a6a79cf3d2058e3edd61fbf54c27aa822c117 100644
--- a/game/modules/tome/class/Game.lua
+++ b/game/modules/tome/class/Game.lua
@@ -165,7 +165,7 @@ function _M:setupDisplayMode()
 		Map.tiles.use_images = true
 	elseif self.gfxmode == 3 then
 		print("[DISPLAY MODE] 32x32 ASCII")
-		Map:setViewPort(200, 20, self.w - 200, math.floor(self.h * 0.80) - 20, 32, 32, nil, 22, true, false)
+		Map:setViewPort(200, 20, self.w - 200, math.floor(self.h * 0.80) - 20, 32, 32, nil, 22, false, false)
 		Map:resetTiles()
 		Map.tiles.use_images = false
 	elseif self.gfxmode == 4 then
@@ -180,7 +180,7 @@ function _M:setupDisplayMode()
 		Map.tiles.use_images = false
 	elseif self.gfxmode == 6 then
 		print("[DISPLAY MODE] 16x16 ASCII/background")
-		Map:setViewPort(200, 20, self.w - 200, math.floor(self.h * 0.80) - 20, 16, 16, nil, 14, false, true)
+		Map:setViewPort(200, 20, self.w - 200, math.floor(self.h * 0.80) - 20, 16, 16, nil, 14, true, true)
 		Map:resetTiles()
 		Map.tiles.use_images = false
 	else
diff --git a/game/modules/tome/data/general/encounters/arda-west.lua b/game/modules/tome/data/general/encounters/arda-west.lua
index c727c06c11c5fdfbcf908f4b946ffda3e4f95870..52c60123447bcb8c73334ef23856ce6e44bccef2 100644
--- a/game/modules/tome/data/general/encounters/arda-west.lua
+++ b/game/modules/tome/data/general/encounters/arda-west.lua
@@ -19,7 +19,7 @@
 
 newEntity{
 	name = "Novice mage",
-	type = "harmless", subtype = "quest", unique = true,
+	type = "harmless", subtype = "special", unique = true,
 	level_range = {1, 10},
 	rarity = 3,
 	coords = {{ x=10, y=23, likelymap={
@@ -55,12 +55,62 @@ newEntity{
 
 newEntity{
 	name = "Lost merchant",
-	type = "harmless", subtype = "quest", unique = true,
-	level_range = {10, 20},
-	rarity = 4,
+	type = "hostile", subtype = "special", unique = true,
+	level_range = {1, 2},
+	rarity = 1,
 	coords = {{ x=0, y=0, w=40, h=40}},
 	on_encounter = function(self, who)
+		who:runStop()
+		engine.Dialog:yesnoPopup("Encounter", "You find an hidden trap door, and hear cries for help from within... Enter ?", function(ok)
+			if not ok then
+				engine.Dialog:simplePopup("Encounter", "You carefully get away without making a sound.")
+			else
+				engine.Dialog:simplePopup("Encounter", "You carefully open the trap door and enter the underground tunnels...")
 
+				local zone = engine.Zone.new("ambush", {
+					name = "Unknown tunnels",
+					level_range = {8, 18},
+					level_scheme = "player",
+					max_level = 2,
+					actor_adjust_level = function(zone, level, e) return zone.base_level + e:getRankLevelAdjust() + level.level-1 + rng.range(-1,2) end,
+					width = 20, height = 20,
+					no_worldport = true,
+					ambiant_music = "a_lomos_del_dragon_blanco.ogg",
+					generator =  {
+						map = {
+							class = "engine.generator.map.TileSet",
+							tileset = {"3x3/base", "3x3/tunnel", "3x3/windy_tunnel"},
+							tunnel_chance = 100,
+							['.'] = "OLD_FLOOR",
+							['#'] = "OLD_WALL",
+							['+'] = "DOOR",
+							["'"] = "DOOR",
+							up = "OLD_FLOOR",
+							down = "DOWN",
+						},
+						actor = { class = "engine.generator.actor.Random",nb_npc = {5, 7}, },
+						trap = { class = "engine.generator.trap.Random", nb_trap = {3, 3}, },
+					},
+					npc_list = mod.class.NPC:loadList("/data/general/npcs/thieve.lua"),
+					grid_list = mod.class.Grid:loadList("/data/general/grids/basic.lua"),
+					object_list = mod.class.Object:loadList("/data/general/objects/objects.lua"),
+					trap_list = mod.class.Trap:loadList("/data/general/traps/alarm.lua"),
+					levels = { [2] = {
+						all_lited=true, all_remembered=true,
+						generator = {
+							map = { class = "engine.generator.map.Static", map = "quests/lost-merchant",},
+							actor = {
+								nb_npc = {1, 1},
+								post_generation = function(e) e.faction="assassin-lair" end,
+							},
+						},
+					}, },
+				})
+				game:changeLevel(1, zone)
+				game.logPlayer(who, "#LIGHT_RED#As you enter you notice the trap door has no visible handle on the inside. You are stuck here!")
+				who:grantQuest("lost-merchant")
+			end
+		end)
 		return true
 	end,
 }
diff --git a/game/modules/tome/data/general/npcs/thieve.lua b/game/modules/tome/data/general/npcs/thieve.lua
index 7fea434c414d612c32fd2e4f181db41877c61a18..ca6737f8bf212445882afe09c8b525ce8284f50a 100644
--- a/game/modules/tome/data/general/npcs/thieve.lua
+++ b/game/modules/tome/data/general/npcs/thieve.lua
@@ -27,7 +27,7 @@ newEntity{
 	display = "p", color=colors.BLUE,
 
 	body = { INVEN = 10, MAINHAND=1, OFFHAND=1, BODY=1 },
-	drops = resolvers.drops{chance=20, nb=1, {} },
+	resolvers.drops{chance=20, nb=1, {} },
 	resolvers.equip{
 		{type="weapon", subtype="dagger", autoreq=true},
 		{type="weapon", subtype="dagger", autoreq=true},
@@ -65,6 +65,7 @@ newEntity{ base = "BASE_NPC_THIEF",
 	rarity = 5,
 	combat_armor = 2, combat_def = 5,
 	resolvers.talents{ [Talents.T_STEALTH]=1,  },
+	on_added = function(self) self.energy.value = game.energy_to_act self:useTalent(self.T_STEALTH) end,
 	max_life = resolvers.rngavg(70,90),
 }
 
@@ -75,16 +76,18 @@ newEntity{ base = "BASE_NPC_THIEF",
 	rarity = 5,
 	combat_armor = 3, combat_def = 5,
 	resolvers.talents{ [Talents.T_STEALTH]=2,  },
+	on_added = function(self) self.energy.value = game.energy_to_act self:useTalent(self.T_STEALTH) end,
 	max_life = resolvers.rngavg(70,90),
 }
 
-newEntity{ base = "BASE_NPC_THIEF",
+newEntity{ base = "BASE_NPC_THIEF", define_as = "THIEF_BANDIT",
 	name = "bandit", color_r=0, color_g=0, color_b=resolvers.rngrange(175, 195),
 	desc = [[These ruffians often employ brute force over thievery but they are capable of stealing as well.]],
 	level_range = {5, 50}, exp_worth = 1,
 	rarity = 7,
 	combat_armor = 4, combat_def = 6,
 	resolvers.talents{ [Talents.T_STEALTH]=3, [Talents.T_LETHALITY]=2, },
+	on_added = function(self) self.energy.value = game.energy_to_act self:useTalent(self.T_STEALTH) end,
 	max_life = resolvers.rngavg(80,100),
 }
 
@@ -97,20 +100,22 @@ newEntity{ base = "BASE_NPC_THIEF",
 	max_life = resolvers.rngavg(90,100),
 	combat = { dam=resolvers.rngavg(6,7), atk=10, apr=4},
 	summon = {
-		{type="human", subtype="thief", name="bandit", number=1, hasxp=false},
-		{type="human", subtype="thief", name="bandit", number=1, hasxp=false},
-		{type="human", subtype="thief", name="thief", number=1, hasxp=false},
-		{type="human", subtype="thief", name="rogue", number=2, hasxp=false},
+		{type="humanoid", subtype="human", name="bandit", number=1, hasxp=false},
+		{type="humanoid", subtype="human", name="bandit", number=1, hasxp=false},
+		{type="humanoid", subtype="human", name="thief", number=1, hasxp=false},
+		{type="humanoid", subtype="human", name="rogue", number=2, hasxp=false},
 	},
 	resolvers.talents{ [Talents.T_STEALTH]=3, [Talents.T_SUMMON]=1, [Talents.T_LETHALITY]=3, },
+	on_added = function(self) self.energy.value = game.energy_to_act self:useTalent(self.T_STEALTH) end,
 }
 
-newEntity{ base = "BASE_NPC_THIEF",
+newEntity{ base = "BASE_NPC_THIEF", define_as = "THIEF_ASSASSIN",
 	name = "assassin", color_r=resolvers.rngrange(0, 10), color_g=resolvers.rngrange(0, 10), color_b=resolvers.rngrange(0, 10),
 	desc = [[Before you looms a pair of eyes... A glint of steel... death.]],
 	level_range = {12, 50}, exp_worth = 1,
 	rarity = 12,
 	combat_armor = 3, combat_def = 10,
 	resolvers.talents{ [Talents.T_STEALTH]=3, [Talents.T_PRECISION]=3, [Talents.T_DUAL_WEAPON_TRAINING]=2, [Talents.T_DUAL_WEAPON_DEFENSE]=2, [Talents.T_DUAL_STRIKE]=1, [Talents.T_SWEEP]=1, [Talents.T_SHADOWSTRIKE]=2, [Talents.T_LETHALITY]=5, },
+	on_added = function(self) self.energy.value = game.energy_to_act self:useTalent(self.T_STEALTH) end,
 	max_life = resolvers.rngavg(70,90),
 }
diff --git a/game/modules/tome/data/maps/quests/lost-merchant.lua b/game/modules/tome/data/maps/quests/lost-merchant.lua
new file mode 100644
index 0000000000000000000000000000000000000000..3ec07c950e254151da2458b5af24d5be614e0174
--- /dev/null
+++ b/game/modules/tome/data/maps/quests/lost-merchant.lua
@@ -0,0 +1,51 @@
+-- ToME - Tales of Middle-Earth
+-- Copyright (C) 2009, 2010 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
+
+defineTile('<', "UP_WILDERNESS")
+defineTile(' ', "FLOOR")
+defineTile('#', "WALL")
+defineTile('a', "FLOOR", nil, "THIEF_ASSASSIN")
+defineTile('p', "FLOOR", nil, "THIEF_BANDIT")
+defineTile('P', "FLOOR", nil, "")
+defineTile('@', "FLOOR", nil, "")
+
+startx = 1
+starty = 9
+
+return [[
+####################
+####################
+##                ##
+## ######         ##
+## #    #   #+### ##
+## #    #   #   # ##
+## ####+# p ##### ##
+##            a   ##
+##              @ ##
+#<              P ##
+##            a   ##
+##  p     #+##### ##
+## #####  #     # ##
+## #   #  ####### ##
+## #   #          ##
+## #   +          ##
+## #####          ##
+##                ##
+####################
+####################]]
diff --git a/game/modules/tome/data/quests/lost-merchant.lua b/game/modules/tome/data/quests/lost-merchant.lua
new file mode 100644
index 0000000000000000000000000000000000000000..5a62fd619e2d32864a1976a2e3caab7758e19273
--- /dev/null
+++ b/game/modules/tome/data/quests/lost-merchant.lua
@@ -0,0 +1,34 @@
+-- ToME - Tales of Middle-Earth
+-- Copyright (C) 2009, 2010 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
+
+name = "Trapped!"
+desc = function(self, who)
+	local desc = {}
+	desc[#desc+1] = "You heard a plea for help and decided to investigate..."
+	desc[#desc+1] = "Only to find yourself trapped inside an unknown tunnel complex."
+	return table.concat(desc, "\n")
+end
+
+on_status_change = function(self, who, status, sub)
+	if sub then
+		if self:isCompleted() then
+			who:setQuestStatus(self.id, engine.Quest.DONE)
+		end
+	end
+end
diff --git a/game/modules/tome/data/talents/misc/misc.lua b/game/modules/tome/data/talents/misc/misc.lua
index 1081c35eec8ff31bd70d0fb852dd6b02c6a5f69c..05d797da140be5604b50f01e6d34ec70523bd8fd 100644
--- a/game/modules/tome/data/talents/misc/misc.lua
+++ b/game/modules/tome/data/talents/misc/misc.lua
@@ -103,6 +103,11 @@ newTalent{
 	type = {"base/class", 1},
 	cooldown = 1000,
 	action = function(self, t)
+		if not self:canBe("worldport") then
+			game.logPlayer(self, "The spell fizzles...")
+			return
+		end
+
 		local seen = false
 		-- Check for visible monsters, only see LOS actors, so telepathy wont prevent it
 		core.fov.calc_circle(self.x, self.y, 20, function(_, x, y) return game.level.map:opaque(x, y) end, function(_, x, y)
diff --git a/game/modules/tome/data/zones/tol-falas-ambush/zone.lua b/game/modules/tome/data/zones/tol-falas-ambush/zone.lua
index 293ac714f45b28e65d5ef6c74ef6e42e13f6a467..cacb4f404daccb9969b68ccd141fad1ce31e47ca 100644
--- a/game/modules/tome/data/zones/tol-falas-ambush/zone.lua
+++ b/game/modules/tome/data/zones/tol-falas-ambush/zone.lua
@@ -24,6 +24,7 @@ return {
 	max_level = 1,
 	actor_adjust_level = function(zone, level, e) return zone.base_level + 20 end,
 	width = 50, height = 50,
+	no_worldport = true,
 --	all_remembered = true,
 	all_lited = true,
 --	persistant = true,
diff --git a/game/modules/tome/load.lua b/game/modules/tome/load.lua
index 9eac6c0217ef073acaeeb28403b9ee190cd50684..11171edf16c3379bc5b55b3de71d1bfb05020cca 100644
--- a/game/modules/tome/load.lua
+++ b/game/modules/tome/load.lua
@@ -91,6 +91,7 @@ ActorLevel:defineMaxLevel(75)
 
 -- Factions
 Faction:add{ name="water lair", reaction={} }
+Faction:add{ name="assassin lair", reaction={} }
 
 -- Actor autolevel schemes
 dofile("/data/autolevel_schemes.lua")
diff --git a/game/modules/tome/resolvers.lua b/game/modules/tome/resolvers.lua
index 1459c2ba41da71298fb42d3a406f6293b2966650..a5d9dce349b29977c21499df96d9543b6f8c5d4b 100644
--- a/game/modules/tome/resolvers.lua
+++ b/game/modules/tome/resolvers.lua
@@ -23,6 +23,7 @@ function resolvers.equip(t)
 end
 --- Actually resolve the equipment creation
 function resolvers.calc.equip(t, e)
+	print("Equipment resolver for", e.name)
 	-- Iterate of object requests, try to create them and equip them
 	for i, filter in ipairs(t[1]) do
 		print("Equipment resolver", e.name, filter.type, filter.subtype)