diff --git a/game/modules/tome/data/achievements/events.lua b/game/modules/tome/data/achievements/events.lua
index 433d77ec6ce72e75a0359393a4e8a411fefe4c15..3e59b3e3d353acf9e998e8f9b7e0867a74c7e6c9 100644
--- a/game/modules/tome/data/achievements/events.lua
+++ b/game/modules/tome/data/achievements/events.lua
@@ -58,3 +58,9 @@ newAchievement{
 	show = "name",
 	desc = [[Killed a Bringer of Doom.]],
 }
+
+newAchievement{
+	name = "A living one!", id = "CALDIZAR",
+	show = "name",
+	desc = [[Have been teleported into Caldizar's Fortress, far into the void between the stars.]],
+}
diff --git a/game/modules/tome/data/birth/classes/chronomancer.lua b/game/modules/tome/data/birth/classes/chronomancer.lua
index d1542da93a0bb42b87ad1e8a34ca14fdc2142c86..208054615808656c22d5fcd943ed82e22bb4b25c 100644
--- a/game/modules/tome/data/birth/classes/chronomancer.lua
+++ b/game/modules/tome/data/birth/classes/chronomancer.lua
@@ -35,6 +35,18 @@ newBirthDescriptor{
 		},
 	},
 	copy = {
+		-- Chronomancers start in Point Zero
+		class_start_check = function(self)
+			if self.descriptor.world == "Maj'Eyal" then
+				self.chronomancer_race_start_quest = self.starting_quest
+				self.default_wilderness = {"zone-pop", "angolwen-portal"}
+				self.starting_zone = "town-point-zero"
+				self.starting_quest = "start-point-zero"
+				self.starting_intro = "chronomancer"
+				self.faction = "keepers-of-reality"
+				self:learnTalent(self.T_TELEPORT_POINT_ZERO, true, nil, {no_unlearn=true})
+			end
+		end,
 	},
 }
 
diff --git a/game/modules/tome/data/chats/point-zero-zemekkys.lua b/game/modules/tome/data/chats/point-zero-zemekkys.lua
new file mode 100644
index 0000000000000000000000000000000000000000..2616b8eb9316d36e0d8b3ec89cda563969bd3ef7
--- /dev/null
+++ b/game/modules/tome/data/chats/point-zero-zemekkys.lua
@@ -0,0 +1,39 @@
+-- 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
+
+newChat{ id="welcome",
+	action = function(npc, player) npc.talked_times = (npc.talked_times or 0) + 1 end,
+	text = [[@playername@, nice to see you again! Or is it the first time you see me ?]],
+	answers = {
+		{"Farewell Grand Keeper."},
+		{"Yes it is the first time I see you.", jump="first", cond=function(npc, player) return not npc.talked_times end},
+	}
+}
+
+newChat{ id="first",
+	text = [[Ah, for you perhaps, but not for me.
+Listen, someday you will encounter me again but this will not be me as of now. A younger me if you will.
+This is very important, do not tell my previous me about me. Understood?]],
+	answers = {
+		{"I think so..."},
+		{"Yes Grand Keeper."},
+	}
+}
+
+return "welcome"
diff --git a/game/modules/tome/data/chats/shertul-fortress-caldizar.lua b/game/modules/tome/data/chats/shertul-fortress-caldizar.lua
index 68ca1d0185d09b6132004e31cc7c0271e7364706..a58b560e8fc1e11f7c9cdb8ac09305fc793c13e4 100644
--- a/game/modules/tome/data/chats/shertul-fortress-caldizar.lua
+++ b/game/modules/tome/data/chats/shertul-fortress-caldizar.lua
@@ -41,6 +41,7 @@ A wave of mental and magical power blasts into you with the might of a falling s
 			game:changeLevel(1, "shertul-fortress", {direct_switch=true})
 			local spot = game.level:pickSpot{type="spawn", subtype="farportal"} or {x=39, y=29}
 			game.player:move(spot.x, spot.y, true)
+			world:gainAchievement("CALDIZAR", game.player)
 			game.player:learnLore("shertul-fortress-caldizar")
 		end},
 	}
diff --git a/game/modules/tome/data/chats/zemekkys-start-chronomancers.lua b/game/modules/tome/data/chats/zemekkys-start-chronomancers.lua
new file mode 100644
index 0000000000000000000000000000000000000000..26df65cca581f444ce40cd64ffc358750b1ea271
--- /dev/null
+++ b/game/modules/tome/data/chats/zemekkys-start-chronomancers.lua
@@ -0,0 +1,29 @@
+-- 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
+
+newChat{ id="welcome",
+	text = [[@playername@, you are called to serve. The nearby unhallowed morass inhabitants are growing restless.
+You must go there and find the source.]],
+	answers = {
+		{"I will, Grand Keeper.", action=function() game:changeLevel(1, "unhallowed-morass") end},
+		{"I am sorry, I can not do that.", action=function(npc, player) player:setQuestStatus("start-point-zero", engine.Quest.FAILED) end},
+	}
+}
+
+return "welcome"
diff --git a/game/modules/tome/data/general/npcs/spider.lua b/game/modules/tome/data/general/npcs/spider.lua
index e06ad67b00145ee6a5a96332329718f97d60868b..6d77a589cbfb897a0b075424165130693307e15b 100644
--- a/game/modules/tome/data/general/npcs/spider.lua
+++ b/game/modules/tome/data/general/npcs/spider.lua
@@ -382,86 +382,3 @@ newEntity{ base = "BASE_NPC_SPIDER",
 		[Talents.T_LUCKY_DAY] = 1,
 	},
 }
-
--- WIP:  More Temporal Spiders, these should be cut and pasted into a Point Zero dungeon file and shouldn't normally spawn
---[=[
-newEntity{
-	define_as = "BASE_NPC_SPIDER",
-	type = "spiderkin", subtype = "spider",
-	display = "S", color=colors.WHITE,
-	desc = [[Arachnophobia...]],
-	body = { INVEN = 10 },
-
-	max_stamina = 150,
-	rank = 1,
-	size_category = 2,
-	infravision = 10,
-
-	autolevel = "spider",
-	ai = "dumb_talented_simple", ai_state = { ai_move="move_complex", talent_in=2, },
-	global_speed_base = 1.2,
-	stats = { str=10, dex=17, mag=3, con=7 },
-	combat = { dammod={dex=0.8} },
-	combat_armor = 1, combat_def = 1,
-}
-
-newEntity{ base = "BASE_NPC_SPIDER",
-	name = "orb spinner", color=colors.UMBER,
-	desc = [[A large brownish arachnid, it's fangs drip with a strange fluid.]],
-	level_range = {1, nil}, exp_worth = 1,
-	rarity = 1,
-	max_life = resolvers.rngavg(40,70),
-	combat_armor = 1, combat_def = 3,
-	combat = { dam=resolvers.levelup(5, 1, 0.7), atk=15, apr=3, damtype=DamageType.CLOCK, },
-}
-
-newEntity{ base = "BASE_NPC_SPIDER",
-	name = "orb weaver", color=colors.DARK_UMBER,
-	desc = [[A large brownish arachnid spinning it's web.  It doesn't look pleased that you've disturbed it's work.]],
-	level_range = {3, nil}, exp_worth = 1,
-	rarity = 3,
-	max_life = resolvers.rngavg(60,90),
-	combat_armor =2, combat_def = 4,
-	combat = { dam=resolvers.levelup(6, 1, 0.8), atk=15, apr=3, damtype=DamageType.TEMPORAL, },
-	resolvers.talents{
-		[Talents.T_LAY_WEB]=1,
-		[Talents.T_DIMENSIONAL_STEP]=1,
-	},
-}
-
-newEntity{ base = "BASE_NPC_SPIDER",
-	name = "fate spinner", color=colors.SLATE,
-	desc = [[Easily as big as a horse, this giant spider menaces at you with claws and fangs.]],
-	level_range = {4, nil}, exp_worth = 1,
-	rarity = 3,
-	size_category = 4,
-	max_life = resolvers.rngavg(80,110),
-	combat_armor = 3, combat_def = 5,
-	combat = { dam=resolvers.levelup(9, 1, 0.9), atk=15, apr=4, damtype=DamageType.CLOCK, },
-	resolvers.talents{
-		[Talents.T_LAY_WEB]=1,
-		[Talents.T_SPIDER_WEB]=1,
-		[Talents.T_DIMENSIONAL_STEP]=1,
-		[Talents.T_TURN_BACK_THE_CLOCK]=1,
-	},
-}
-
-newEntity{ base = "BASE_NPC_SPIDER",
-	name = "fate weaver", color=colors.WHITE,
-	desc = [[A large white spider.]],
-	level_range = {4, nil}, exp_worth = 1,
-	rarity = 3,
-	max_life = resolvers.rngavg(70,100),
-	combat_armor = 3, combat_def = 4,
-	combat = { dam=resolvers.levelup(8, 1, 0.9), atk=15, apr=3, damtype=DamageType.WASTING, },
-
-	talent_cd_reduction = {[Talents.T_RETHREAD]=-10},
-
-	resolvers.talents{
-		[Talents.T_SPIN_FATE]=2,
-		[Talents.T_BANISH]=2,
-		[Talents.T_RETHREAD]=2,
-		[Talents.T_STATIC_HISTORY]=2,
-	},
-}
-]=]
\ No newline at end of file
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/horror_temporal_temporal_defiler.png b/game/modules/tome/data/gfx/shockbolt/npc/horror_temporal_temporal_defiler.png
new file mode 100644
index 0000000000000000000000000000000000000000..d5ee1bdc8d6d89e52e2151b0fe893f4e44122ca8
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/npc/horror_temporal_temporal_defiler.png differ
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_fate_spinner.png b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_fate_spinner.png
new file mode 100644
index 0000000000000000000000000000000000000000..8f6bdfb9535bcf3ad01b34c32bd5347dab180ad6
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_fate_spinner.png differ
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_fate_weaver.png b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_fate_weaver.png
new file mode 100644
index 0000000000000000000000000000000000000000..3016fa937645e743ac45c17e69225a14c4aa4b98
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_fate_weaver.png differ
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_orb_spinner.png b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_orb_spinner.png
new file mode 100644
index 0000000000000000000000000000000000000000..6b53c881b2d7c2d5aef70aa4c2c4b789f8d6fec3
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_orb_spinner.png differ
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_orb_weaver.png b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_orb_weaver.png
new file mode 100644
index 0000000000000000000000000000000000000000..199c7bee0248b608bb82771d3702a5bda93deecb
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_orb_weaver.png differ
diff --git a/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_weaver_queen.png b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_weaver_queen.png
new file mode 100644
index 0000000000000000000000000000000000000000..ddd10e0b59b9bcf1674f075ea857044488d3db01
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/npc/spiderkin_spider_weaver_queen.png differ
diff --git a/game/modules/tome/data/gfx/shockbolt/object/artifact/time_shard.png b/game/modules/tome/data/gfx/shockbolt/object/artifact/time_shard.png
new file mode 100644
index 0000000000000000000000000000000000000000..32fe90f5a6ecf0d8ba074c85274ec17da9a8b6de
Binary files /dev/null and b/game/modules/tome/data/gfx/shockbolt/object/artifact/time_shard.png differ
diff --git a/game/modules/tome/data/gfx/talents/teleport_point_zero.png b/game/modules/tome/data/gfx/talents/teleport_point_zero.png
new file mode 100644
index 0000000000000000000000000000000000000000..0a03fde54be7b4aac7bc746c62be02a39832730d
Binary files /dev/null and b/game/modules/tome/data/gfx/talents/teleport_point_zero.png differ
diff --git a/game/modules/tome/data/quests/start-point-zero.lua b/game/modules/tome/data/quests/start-point-zero.lua
new file mode 100644
index 0000000000000000000000000000000000000000..4e47d394718d3b32f1350b4332eeb1095afcff25
--- /dev/null
+++ b/game/modules/tome/data/quests/start-point-zero.lua
@@ -0,0 +1,64 @@
+-- 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
+
+name = "Future Echoes"
+stables = 0
+desc = function(self, who)
+	local desc = {}
+	desc[#desc+1] = "The unhallowed morass is the name of the 'zone' surrounding Point Zero."
+	desc[#desc+1] = "The temporal spiders that inhabit it are growing restless and started attacking at random, you need to investigate what is going on."
+	if self:isCompleted("morass") then
+		desc[#desc+1] = "#LIGHT_GREEN#* You have explored the morass and destroyed the weaver queen, finding strange traces on it.#WHITE#"
+	else
+		desc[#desc+1] = "#SLATE#* You must explore the morass.#WHITE#"
+	end
+	if self:isCompleted("saved") then
+		desc[#desc+1] = "#LIGHT_GREEN#* You have helped defend Point Zero.#WHITE#"
+	end
+	return table.concat(desc, "\n")
+end
+
+on_status_change = function(self, who, status, sub)
+	if sub then
+		if self:isCompleted("saved") then
+			who:setQuestStatus(self.id, engine.Quest.DONE)
+			world:gainAchievement("UNHALLOWED_MORASS", game.player)
+			who:grantQuest(who.chronomancer_race_start_quest)
+		end
+	end
+	if status == self.FAILED then
+		who:grantQuest(who.chronomancer_race_start_quest)
+	end
+end
+
+on_grant = function(self, who)
+	local npc
+	for uid, e in pairs(game.level.entities) do
+		if e.define_as and e.define_as == "ZEMEKKYS" then npc = e break end
+	end
+	if not npc then return end
+	local x, y = util.findFreeGrid(npc.x, npc.y, 10, true, {[engine.Map.ACTOR]=true})
+	if not x or not y then return end
+
+	who:move(x, y, true)
+
+	local Chat = require"engine.Chat"
+	local chat = Chat.new("zemekkys-start-chronomancers", npc, who)
+	chat:invoke()
+end
diff --git a/game/modules/tome/data/talents/misc/misc.lua b/game/modules/tome/data/talents/misc/misc.lua
index 08511fd84e4fbed2de10866b4b7df2e9865fb348..718acb30785a23532de932bf15a77ff00d551922 100644
--- a/game/modules/tome/data/talents/misc/misc.lua
+++ b/game/modules/tome/data/talents/misc/misc.lua
@@ -254,6 +254,43 @@ newTalent{
 	The spell will take time to activate. You must be out of sight of any creature when you cast it and when the teleportation takes effect.]]
 }
 
+-- Chronomancer class talent, teleport to Point Zero
+newTalent{
+	short_name = "TELEPORT_POINT_ZERO",
+	name = "Timeport: Point Zero",
+	type = {"base/class", 1},
+	cooldown = 400,
+	no_npc_use = true,
+	no_unlearn_last = true,
+	no_silence=true, is_spell=true,
+	action = function(self, t)
+		if not self:canBe("worldport") or self:attr("never_move") 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, game.level.map.w, game.level.map.h, 20, function(_, x, y) return game.level.map:opaque(x, y) end, function(_, x, y)
+			local actor = game.level.map(x, y, game.level.map.ACTOR)
+			if actor and actor ~= self then seen = true end
+		end, nil)
+		if seen then
+			game.log("There are creatures that could be watching you; you cannot take the risk.")
+			return
+		end
+
+		self:setEffect(self.EFF_TELEPORT_POINT_ZERO, 40, {})
+		self:attr("temporal_touched", 1)
+		self:attr("time_travel_times", 1)
+		return true
+	end,
+	info = [[Allows a chronomancer to timeport to Point Zero.
+	You have studied the chronomancy there and have been granted a special portal spell to teleport there.
+	Nobody must learn about this spell and so it should never be used while seen by any creatures.
+	The spell will take time to activate. You must be out of sight of any creature when you cast it and when the timeportation takes effect.]]
+}
+
 newTalent{
 	name = "Relentless Pursuit",
 	type = {"base/class", 1},
diff --git a/game/modules/tome/data/texts/intro-chronomancer.lua b/game/modules/tome/data/texts/intro-chronomancer.lua
new file mode 100644
index 0000000000000000000000000000000000000000..73627b5f486f8f1e141ea1ea06619990b430cd3a
--- /dev/null
+++ b/game/modules/tome/data/texts/intro-chronomancer.lua
@@ -0,0 +1,29 @@
+-- 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 [[Welcome #LIGHT_GREEN#@name@#WHITE#.
+You are a chronomancer, a Keeper of Reality.
+Keepers of Reality are a group of chronomancers, who took upon the self-assigned task of preserving the timelines around Eyal.
+
+You have trained most of your life in Point Zero, the 'town' is the stronghold of the Keepers, placed at the very center of all time threads since the Spellblaze disrupted the temporal streams and permited chronomancy to exist.
+Point Zero keepers must constantly stay vigilant of the monstrous horrors of the unreality outside.
+
+You have heard about temporal disruptions in Maj'Eyal and want to investigate.
+However Grand Keeper of Reality Zemekkys has requested to see you.
+]]
diff --git a/game/modules/tome/data/timed_effects/magical.lua b/game/modules/tome/data/timed_effects/magical.lua
index f7b84876c23ad32c5fe5eaf7b613766796a3d289..429a95a4a8c140a64eb8fcf521e0d9959e5785aa 100644
--- a/game/modules/tome/data/timed_effects/magical.lua
+++ b/game/modules/tome/data/timed_effects/magical.lua
@@ -826,6 +826,43 @@ newEffect{
 	end,
 }
 
+newEffect{
+	name = "TELEPORT_POINT_ZERO", image = "talents/teleport_point_zero.png",
+	desc = "Timeport: Point Zero",
+	long_desc = function(self, eff) return "The target is waiting to be recalled back to Point Zero." end,
+	type = "magical",
+	subtype = { timeport=true },
+	status = "beneficial",
+	cancel_on_level_change = true,
+	parameters = { },
+	activate = function(self, eff)
+		eff.leveid = game.zone.short_name.."-"..game.level.level
+	end,
+	deactivate = function(self, eff)
+		local seen = false
+		-- Check for visible monsters, only see LOS actors, so telepathy wont prevent it
+		core.fov.calc_circle(self.x, self.y, game.level.map.w, game.level.map.h, 20, function(_, x, y) return game.level.map:opaque(x, y) end, function(_, x, y)
+			local actor = game.level.map(x, y, game.level.map.ACTOR)
+			if actor and actor ~= self then seen = true end
+		end, nil)
+		if seen then
+			game.log("There are creatures that could be watching you; you cannot take the risk of timeporting to Point Zero.")
+			return
+		end
+
+		if self:canBe("worldport") and not self:attr("never_move") and eff.dur <= 0 then
+			game:onTickEnd(function()
+				if eff.leveid == game.zone.short_name.."-"..game.level.level and game.player.can_change_zone then
+					game.logPlayer(self, "You are yanked out of this time!")
+					game:changeLevel(1, "town-point-zero")
+				end
+			end)
+		else
+			game.logPlayer(self, "Time restabilizes around you.")
+		end
+	end,
+}
+
 newEffect{
 	name = "PREMONITION_SHIELD", image = "talents/premonition.png",
 	desc = "Premonition Shield",
diff --git a/game/modules/tome/data/zones/shertul-fortress/grids.lua b/game/modules/tome/data/zones/shertul-fortress/grids.lua
index 7d1358f2063d4d6b3aa846f2fb3cabc2a59227b5..2ac64c1a3fb6a5995749f371871e8b8bf615b172 100644
--- a/game/modules/tome/data/zones/shertul-fortress/grids.lua
+++ b/game/modules/tome/data/zones/shertul-fortress/grids.lua
@@ -64,7 +64,7 @@ It should automatically create a portal back, but it might not be near your arri
 
 	checkSpecialLocation = function(self, who, q)
 		-- Caldizar space fortress
-		if rng.percent(2) and not game.state:hasSeenSpecialFarportal("caldizar-space-fortress") then
+		if rng.percent(5) and not game.state:hasSeenSpecialFarportal("caldizar-space-fortress") then
 			game:changeLevel(1, "shertul-fortress-caldizar", {direct_switch=true})
 			q:exploratory_energy()
 			game.log("#VIOLET#You enter the swirling portal and in the blink of an eye you set foot in a strangely familiar zone, right next to a farportal...")
diff --git a/game/modules/tome/data/zones/unhallowed-morass/grids.lua b/game/modules/tome/data/zones/unhallowed-morass/grids.lua
new file mode 100644
index 0000000000000000000000000000000000000000..f93eb12344f6d1a169e37afd7a7963204819aad5
--- /dev/null
+++ b/game/modules/tome/data/zones/unhallowed-morass/grids.lua
@@ -0,0 +1,57 @@
+-- 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
+
+load("/data/general/grids/basic.lua")
+load("/data/general/grids/void.lua")
+
+newEntity{
+	define_as = "RIFT",
+	name = "Temporal Rift", add_mos={{image="terrain/demon_portal2.png"}},
+	display = '&', color_r=255, color_g=0, color_b=220, back_color=colors.VIOLET,
+	notice = true,
+	always_remember = true,
+	show_tooltip = true,
+	desc = [[The rift leads to an other part of the morass.]],
+	change_level = 1,
+}
+
+newEntity{
+	define_as = "RIFT_HOME",
+	name = "Temporal Rift", add_mos={{image="terrain/demon_portal2.png"}},
+	display = '&', color_r=255, color_g=0, color_b=220, back_color=colors.VIOLET,
+	notice = true,
+	always_remember = true,
+	show_tooltip = true,
+	desc = [[The rift leads to an other part of the morass.]],
+	change_level = 1,
+	change_zone = "town-point-zero",
+}
+
+local rift_editer = { method="sandWalls_def", def="rift"}
+newEntity{
+	define_as = "SPACETIME_RIFT",
+	type = "wall", subtype = "rift",
+	name = "crack in spacetime",
+	display = '#', color=colors.YELLOW, image="terrain/rift/rift_inner_05_01.png",
+	always_remember = true,
+	block_sight = true,
+	does_block_move = true,
+	_noalpha = false,
+	nice_editer = rift_editer,
+}
diff --git a/game/modules/tome/data/zones/unhallowed-morass/npcs.lua b/game/modules/tome/data/zones/unhallowed-morass/npcs.lua
new file mode 100644
index 0000000000000000000000000000000000000000..1d775c16fd66ec6dd93f5af792cb0c6a9208fa78
--- /dev/null
+++ b/game/modules/tome/data/zones/unhallowed-morass/npcs.lua
@@ -0,0 +1,186 @@
+-- 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
+
+local Talents = require("engine.interface.ActorTalents")
+
+newEntity{
+	define_as = "BASE_NPC_SPIDER",
+	type = "spiderkin", subtype = "spider",
+	display = "S", color=colors.WHITE,
+	desc = [[Arachnophobia...]],
+	body = { INVEN = 10 },
+
+	max_stamina = 150,
+	rank = 1,
+	size_category = 2,
+	infravision = 10,
+
+	autolevel = "spider",
+	ai = "dumb_talented_simple", ai_state = { ai_move="move_complex", talent_in=2, },
+	global_speed_base = 1.2,
+	stats = { str=10, dex=17, mag=3, con=7 },
+	combat = { dammod={dex=0.8} },
+	combat_armor = 1, combat_def = 1,
+}
+
+newEntity{ base = "BASE_NPC_SPIDER",
+	name = "orb spinner", color=colors.UMBER,
+	desc = [[A large brownish arachnid, it's fangs drip with a strange fluid.]],
+	level_range = {1, nil}, exp_worth = 1,
+	rarity = 1,
+	max_life = resolvers.rngavg(20,40),
+	combat_armor = 1, combat_def = 3,
+	combat = { dam=resolvers.levelup(5, 1, 0.7), atk=15, apr=3, damtype=DamageType.CLOCK, },
+}
+
+newEntity{ base = "BASE_NPC_SPIDER",
+	name = "orb weaver", color=colors.DARK_UMBER,
+	desc = [[A large brownish arachnid spinning it's web.  It doesn't look pleased that you've disturbed it's work.]],
+	level_range = {3, nil}, exp_worth = 1,
+	rarity = 3,
+	max_life = resolvers.rngavg(40,60),
+	combat_armor =2, combat_def = 4,
+	combat = { dam=resolvers.levelup(6, 1, 0.8), atk=15, apr=3, damtype=DamageType.TEMPORAL, },
+	resolvers.talents{
+		[Talents.T_LAY_WEB]=1,
+		[Talents.T_DIMENSIONAL_STEP]=1,
+	},
+}
+
+newEntity{ base = "BASE_NPC_SPIDER",
+	name = "fate spinner", color=colors.SLATE,
+	desc = [[Easily as big as a horse, this giant spider menaces at you with claws and fangs.]],
+	level_range = {4, nil}, exp_worth = 1,
+	rarity = 3,
+	size_category = 4,
+	max_life = resolvers.rngavg(60,70),
+	combat_armor = 3, combat_def = 5,
+	combat = { dam=resolvers.levelup(9, 1, 0.9), atk=15, apr=4, damtype=DamageType.CLOCK, },
+	resolvers.talents{
+		[Talents.T_LAY_WEB]=1,
+		[Talents.T_SPIDER_WEB]=1,
+		[Talents.T_DIMENSIONAL_STEP]=1,
+		[Talents.T_TURN_BACK_THE_CLOCK]=1,
+	},
+}
+
+newEntity{ base = "BASE_NPC_SPIDER",
+	name = "fate weaver", color=colors.WHITE,
+	desc = [[A large white spider.]],
+	level_range = {4, nil}, exp_worth = 1,
+	rarity = 3,
+	max_life = resolvers.rngavg(70,100),
+	combat_armor = 3, combat_def = 4,
+	combat = { dam=resolvers.levelup(8, 1, 0.9), atk=15, apr=3, damtype=DamageType.WASTING, },
+
+	talent_cd_reduction = {[Talents.T_RETHREAD]=-10},
+
+	resolvers.talents{
+		[Talents.T_SPIN_FATE]=2,
+		[Talents.T_BANISH]=2,
+		[Talents.T_RETHREAD]=2,
+		[Talents.T_STATIC_HISTORY]=2,
+	},
+}
+
+newEntity{ base = "BASE_NPC_SPIDER",
+	name = "weaver queen", color=colors.WHITE,
+	resolvers.nice_tile{image="invis.png", add_mos = {{image="npc/spiderkin_spider_weaver_queen.png", display_h=2, display_y=-1}}},
+	desc = [[A large white spider.]],
+	level_range = {7, nil}, exp_worth = 1,
+	unique = true,
+	rarity = false,
+	max_life = 150, life_rating = 10, fixed_rating = true,
+	rank = 4,
+	tier1 = true,
+	size_category = 4,
+	instakill_immune = 1,
+
+	combat_armor = 3, combat_def = 4,
+	combat = { dam=resolvers.levelup(8, 1, 0.9), atk=15, apr=3, damtype=DamageType.CLOCK, },
+
+	body = { INVEN = 10, MAINHAND=1, OFFHAND=1, BODY=1 },
+	resolvers.drops{chance=100, nb=1, {defined="TIMESHARD"} },
+	resolvers.drops{chance=100, nb=3, {tome_drops="boss"} },
+
+	talent_cd_reduction = {[Talents.T_RETHREAD]=-10},
+
+	resolvers.talents{
+		[Talents.T_SPIN_FATE]=2,
+		[Talents.T_BANISH]=2,
+		[Talents.T_RETHREAD]=2,
+		[Talents.T_STATIC_HISTORY]=2,
+		[Talents.T_FADE_FROM_TIME]=3,
+		[Talents.T_BODY_REVERSION]=1,
+	},
+
+	autolevel = "caster",
+	ai = "tactical", ai_state = { talent_in=1, ai_move="move_astar", },
+	ai_tactic = resolvers.tactic"ranged",
+
+	on_die = function(self, who)
+		game.player:resolveSource():setQuestStatus("start-point-zero", engine.Quest.COMPLETED, "morass")
+		require("engine.ui.Dialog"):simplePopup("As you vanquish the queen you notice a temporal thread that seems to have been controlling her. It seems to go through a rift.")
+		local rift = game.zone:makeEntityByName(game.level, "terrain", "RIFT_HOME")
+		game.zone:addEntity(game.level, rift, "terrain", self.x, self.y)
+	end,
+}
+
+--[=[
+newEntity{ base="BASE_NPC_LOSGOROTH", define_as = "SPACIAL_DISTURBANCE",
+	unique = true,
+	name = "Spacial Disturbance",
+	color=colors.VIOLET,
+	resolvers.nice_tile{image="invis.png"},
+	resolvers.generic(function(e) if engine.Map.tiles.nicer_tiles then e:addParticles(engine.Particles.new("wormhole", 1, {image="shockbolt/npc/elemental_losgoroth_space_disturbance", speed=1})) end end),
+	desc = [[A hole in the fabric of space, it seems to be the source of the expanse unstability.]],
+	killer_message = "and folded out of existence",
+	level_range = {7, nil}, exp_worth = 2,
+	max_life = 150, life_rating = 10, fixed_rating = true,
+	mana_regen = 7,
+	stats = { str=10, dex=10, cun=12, mag=20, con=10 },
+	rank = 4,
+	tier1 = true,
+	size_category = 4,
+	infravision = 10,
+	instakill_immune = 1,
+	can_pass = {pass_void=0},
+
+	body = { INVEN = 10, MAINHAND=1, OFFHAND=1, BODY=1 },
+	resolvers.drops{chance=100, nb=1, {defined="VOID_STAR"} },
+	resolvers.drops{chance=100, nb=3, {tome_drops="boss"} },
+
+	resolvers.talents{
+		[Talents.T_VOID_BLAST]={base=1, every=7, max=7},
+		[Talents.T_MANATHRUST]={base=1, every=7, max=7},
+		[Talents.T_PHASE_DOOR]=2,
+	},
+	resolvers.inscriptions(1, {"manasurge rune"}),
+
+	autolevel = "caster",
+	ai = "tactical", ai_state = { talent_in=1, ai_move="move_astar", },
+	ai_tactic = resolvers.tactic"ranged",
+
+	on_die = function(self, who)
+		local q = game.player:hasQuest("start-archmage")
+		if q then q:stabilized() end
+		game.player:resolveSource():setQuestStatus("start-archmage", engine.Quest.COMPLETED, "abashed")
+	end,
+}
+]=]
diff --git a/game/modules/tome/data/zones/unhallowed-morass/objects.lua b/game/modules/tome/data/zones/unhallowed-morass/objects.lua
new file mode 100644
index 0000000000000000000000000000000000000000..d1359f8b97ce292405422d5e3f6c657e9ffe49ec
--- /dev/null
+++ b/game/modules/tome/data/zones/unhallowed-morass/objects.lua
@@ -0,0 +1,51 @@
+-- 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
+
+load("/data/general/objects/objects-maj-eyal.lua")
+
+local Talents = require "engine.interface.ActorTalents"
+
+newEntity{ base = "BASE_LITE", define_as = "VOID_STAR",
+	power_source = {arcane=true},
+	unique = true,
+	name = "Void Star", image="object/artifact/void_star.png",
+	unided_name = "tiny black star",
+	level_range = {1, 10},
+	color = colors.GREY,
+	encumber = 1,
+	rarity = false,
+	desc = [[It looks like a very tiny star - deep black - and yet it somehow shines.]],
+	cost = 120,
+	material_level = 2,
+
+	wielder = {
+		combat_spellcrit = 5,
+		inc_damage = {
+			[DamageType.ARCANE]    = 6,
+			[DamageType.FIRE]      = 6,
+			[DamageType.COLD]      = 6,
+			[DamageType.ACID]      = 6,
+			[DamageType.LIGHTNING] = 6,
+		},
+		lite = 2,
+	},
+
+	max_power = 70, power_regen = 1,
+	use_talent = { id = Talents.T_ECHOES_FROM_THE_VOID, level = 2, power = 70 },
+}
diff --git a/game/modules/tome/data/zones/unhallowed-morass/traps.lua b/game/modules/tome/data/zones/unhallowed-morass/traps.lua
new file mode 100644
index 0000000000000000000000000000000000000000..47c172b72bc64362e8ac54797c40d75079ecdc18
--- /dev/null
+++ b/game/modules/tome/data/zones/unhallowed-morass/traps.lua
@@ -0,0 +1,20 @@
+-- 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
+
+load("/data/general/traps/natural_forest.lua")
diff --git a/game/modules/tome/data/zones/unhallowed-morass/zone.lua b/game/modules/tome/data/zones/unhallowed-morass/zone.lua
new file mode 100644
index 0000000000000000000000000000000000000000..1d2fd1036076799295177b81e7e19da19a1acd15
--- /dev/null
+++ b/game/modules/tome/data/zones/unhallowed-morass/zone.lua
@@ -0,0 +1,84 @@
+-- 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 {
+	name = "Unhallowed Morass",
+	level_range = {1, 5},
+	level_scheme = "player",
+	max_level = 3,
+	decay = {300, 800},
+	actor_adjust_level = function(zone, level, e) return zone.base_level + e:getRankLevelAdjust() + level.level-1 + rng.range(-1,2) end,
+	width = 60, height = 60,
+	all_remembered = true,
+	all_lited = true,
+	tier1 = true,
+	persistent = "zone",
+	ambient_music = "Suspicion.ogg",
+	max_material_level = 2,
+	color_shown = {0.7, 0.6, 0.8, 1},
+	color_obscure = {0.7*0.6, 0.6*0.6, 0.8*0.6, 0.6},
+
+	generator =  {
+		map = {
+			class = "engine.generator.map.Cavern",
+			zoom = 25,
+			min_floor = 1100,
+			floor = "VOID",
+			wall = "SPACETIME_RIFT",
+			up = "VOID",
+			down = "RIFT",
+		},
+		actor = {
+			class = "mod.class.generator.actor.Random",
+			nb_npc = {20, 30},
+			filters = { {max_ood=2}, },
+			guardian = "WEAVER_QUEEN",
+		},
+		object = {
+			class = "engine.generator.object.Random",
+			nb_object = {0, 0},
+		},
+		trap = {
+			class = "engine.generator.trap.Random",
+			nb_trap = {0, 0},
+		},
+	},
+
+	post_process = function(level)
+		local Map = require "engine.Map"
+		level.background_particle = require("engine.Particles").new("starfield", 1, {width=Map.viewport.width, height=Map.viewport.height})
+
+		game.state:makeWeather(level, 6, {max_nb=12, chance=1, dir=120, speed={1.5, 5.9}, r=0.2, g=0.4, b=1, alpha={0.2, 0.4}, particle_name="weather/grey_cloud_%02d"})
+
+		if not config.settings.tome.weather_effects then return end
+
+		local Map = require "engine.Map"
+		level.foreground_particle = require("engine.Particles").new("snowing", 1, {width=Map.viewport.width, height=Map.viewport.height, r=0.65, g=0.25, b=1, rv=-0.001, gv=0, bv=-0.001, factor=2, dir=math.rad(110+180)})
+	end,
+
+	background = function(level, x, y, nb_keyframes)
+		local Map = require "engine.Map"
+		level.background_particle.ps:toScreen(x, y, true, 1)
+	end,
+
+	foreground = function(level, x, y, nb_keyframes)
+		if not config.settings.tome.weather_effects or not level.foreground_particle then return end
+		level.foreground_particle.ps:toScreen(x, y, true, 1)
+	end,
+}
diff --git a/game/modules/tome/dialogs/Donation.lua b/game/modules/tome/dialogs/Donation.lua
index 7fe9e0cae065b5b92df70b0dc6e21d64ae7e0466..73b17bb849e9e0c967859ffb459933828fc9f6ce 100644
--- a/game/modules/tome/dialogs/Donation.lua
+++ b/game/modules/tome/dialogs/Donation.lua
@@ -59,7 +59,7 @@ Thank you for your kindness!]]}
 
 	self.c_donate = Numberbox.new{title="Donation amount: ", number=10, max=1000, min=5, chars=5, fct=function() end}
 	local euro = Textzone.new{auto_width=true, auto_height=true, text=[[€]]}
-	self.c_recur = Checkbox.new{title="Make it a recurring montly donation", default=recur, fct=function() end}
+	self.c_recur = Checkbox.new{title="Make it a recurring monthly donation", default=recur, fct=function() end}
 	local ok = require("engine.ui.Button").new{text="Accept", fct=function() self:ok() end}
 	local cancel = require("engine.ui.Button").new{text="Cancel", fct=function() self:cancel() end}