💾 Archived View for gemini.ctrl-c.club › ~nttp › games › tots3.py captured on 2024-12-17 at 11:53:44.

View Raw

More Information

⬅️ Previous capture (2021-12-03)

-=-=-=-=-=-=-

#!/usr/bin/env python3
# coding=utf-8

# Tomb of the Snake -- a coffeebreak roguelike for the Linux console
# 2021-10-25 Felix Pleșoianu <https://felix.plesoianu.ro/>
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
# 
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
# 
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE.

from __future__ import division

import random

import curses
from curses.textpad import rectangle

WHITE = 0
RED = 1
GREEN = 2
BLUE = 3
YELLOW = 4

dirkeys = {
	ord('h'): 'w', ord('j'): 's', ord('k'): 'n', ord('l'): 'e',
	curses.KEY_LEFT: 'w', curses.KEY_DOWN: 's',
	curses.KEY_UP: 'n', curses.KEY_RIGHT: 'e',
}

fxdescs = {"fire": "on fire", "poison": "poisoned"}

traitadjs = {"muscle": "stronger", "stamina": "sturdier",
	"agility": "nimbler", "speed": "faster", "senses": "sharper"}

blurb = [
	"You are lost in the woods.",
	"Dark clouds hang in the sky.",
	"The day has turned to night.",
	"Now you will have to brave the..."
]

title_banner = [
" ______           __          ___  __  __         ____          __      ",
"/_  __/__  __ _  / /    ___  / _/ / /_/ /  ___   / __/__  ___ _/ /_____ ",
" / / / _ \/  ' \/ _ \  / _ \/ _/ / __/ _ \/ -_) _\ \/ _ \/ _ `/  '_/ -_)",
"/_/  \___/_/_/_/_.__/  \___/_/   \__/_//_/\__/ /___/_//_/\_,_/_/\_\\__/ "
]

help_text = [
	"Dive to level 10 of the dungeon",
	"and bring the idol back above the ground.",
	"",
	"Arrow keys or h-j-k-l to move; . to wait.",
	"< and > to go up/down a level.",
	"'r' and a direction to run until stopped.",
	"",
	"? to show instructions during the game.",
	"@ to show the character sheet.",
	"",
	"Click a map square to look at it.",
	"Double-click to shoot."
]

ending_text = [
	"As fresh surface air touches the idol,",
	"its golden shine quickly becomes tarnished.",
	"Vile gasses spew out of the idol. You drop it,",
	"and it tumbles back down the stairs.",
	"The entrance to the catacomb collapses, then...",
	"...the clouds part and the sun comes out!",
	"",
	"You have won!"
]

death_banner = [
"__  __               __                            ___          ____",
"\ \/ /___  __  __   / /_  ____ __   _____     ____/ (_)__  ____/ / /",
" \  / __ \/ / / /  / __ \/ __ `/ | / / _ \   / __  / / _ \/ __  / / ",
" / / /_/ / /_/ /  / / / / /_/ /| |/ /  __/  / /_/ / /  __/ /_/ /_/  ",
"/_/\____/\__,_/  /_/ /_/\__,_/ |___/\___/   \__,_/_/\___/\__,_(_)   "
]

terrain = {
	"village": {
		ord('.'): {"name": "grass", "color": WHITE},
		ord(','): {"name": "a dirt road", "color": YELLOW},
		ord('#'): {"name": "a cabin wall", "color": RED},
		ord('='): {"name": "a wooden plank", "color": YELLOW},
		ord('|'): {"name": "a tree", "color": RED},
		ord('^'): {"name": "a tree", "color": GREEN},
		ord('~'): {"name": "water", "color": BLUE},
		ord('>'): {"name": "a way down", "color": WHITE}
	},
	
	"cave": {
		ord('.'): {"name": "the cave floor", "color": WHITE},
		ord('#'): {"name": "the cave wall", "color": WHITE},
		ord('~'): {"name": "water", "color": BLUE},
		ord('>'): {"name": "a way down", "color": WHITE},
		ord('<'): {"name": "a way up", "color": WHITE}
	},
	
	"tomb": {
		ord('.'): {"name": "the catacomb floor", "color": WHITE},
		ord('#'): {"name": "the catacomb wall", "color": RED},
		ord('|'): {"name": "a pillar", "color": YELLOW},
		ord('+'): {"name": "a stone coffin", "color": WHITE},
		ord('~'): {"name": "water", "color": BLUE},
		ord('>'): {"name": "a way down", "color": WHITE},
		ord('<'): {"name": "a way up", "color": WHITE}
	}
}

weapons = {
	"stone": {
		"char": "o",
		"color": RED,
		"name": "stone",
		"ammo": 1,
		"weight": 1,
		"damage": 2
	},
	
	"pickaxe": {
		"char": "/",
		"color": BLUE,
		"name": "pickaxe",
		"attack": 1,
		"damage": 4,
		"weight": 6
	},

	"bullet": {
		"char": "o",
		"color": BLUE,
		"name": "sling bullet",
		"ammo": 2,
		"weight": 2,
		"damage": 6
	},
	
	"sword": {
		"char": ")",
		"color": BLUE,
		"name": "short sword",
		"attack": 4,
		"damage": 4,
		"weight": 4
	},
	
	"mace": {
		"char": ")",
		"color": YELLOW,
		"name": "mace",
		"attack": 4,
		"damage": 6,
		"weight": 4
	},
	
	"staff": {
		"char": "/",
		"color": RED,
		"name": "staff",
		"attack": 8,
		"damage": 4,
		"weight": 8
	},
	
	"spear": {
		"char": "/",
		"color": YELLOW,
		"name": "spear",
		"attack": 8,
		"damage": 6,
		"weight": 8
	}
}

armor = {
	"cloth": {
		"char": "[",
		"color": WHITE,
		"name": "thick cloak",
		"armor": 4,
		"weight": 6
	},
	
	"leather": {
		"char": "[",
		"color": RED,
		"name": "leather coat",
		"armor": 6,
		"weight": 8
	},
	
	"chainmail": {
		"char": "]",
		"color": BLUE,
		"name": "chainmail",
		"armor": 8,
		"weight": 10
	},
	
	"scalemail": {
		"char": "]",
		"color": YELLOW,
		"name": "scale mail",
		"armor": 10,
		"weight": 12
	},
	
	"splintmail": {
		"char": "]",
		"color": RED,
		"name": "splint mail",
		"armor": 12,
		"weight": 20
	}
}

shields = {
	"leather": {
		"char": "(",
		"color": RED,
		"name": "leather shield",
		"defense": 4,
		"weight": 4
	},
	
	"wood": {
		"char": "(",
		"color": YELLOW,
		"name": "wooden shield",
		"defense": 6,
		"weight": 6
	},
	
	"bronze": {
		"char": "(",
		"color": GREEN,
		"name": "bronze shield",
		"defense": 10,
		"weight": 10
	}
}

items = {
	"berries": {
		"char": "%",
		"color": BLUE,
		"name": "berries",
		"article": "none",
		"food": 4,
		"weight": 1
	},
	
	"mushroom": {
		"char": "%",
		"color": WHITE,
		"name": "mushroom",
		"food": 8,
		"weight": 1
	},
	
	"ration": {
		"char": "%",
		"color": YELLOW,
		"name": "iron ration",
		"food": 12,
		"weight": 1
	},
	
	"torch": {
		"char": "\\",
		"color": RED,
		"name": "torch",
		"light": 80,
		"weight": 1,
		"attack": 1,
		"defense": 1,
		"damage": 1,
		"special": ("fire", 6, 6)
	},
	
	"idol": {
		"char": "&",
		"color": YELLOW,
		"name": "golden snake idol",
		"weight": 3
	}
}

trophies = {
	"gem": {
		"char": "$",
		"color": GREEN,
		"name": "snake eye gem",
	},

	"bones": {
		"char": "$",
		"color": WHITE,
		"name": "bone necklace",
	},

	"gold": {
		"char": "$",
		"color": YELLOW,
		"name": "gold coin",
	},

	"key": {
		"char": "?",
		"color": BLUE,
		"name": "clock key",
	},

	"ember": {
		"char": "?",
		"color": RED,
		"name": "glowing ember",
	},

	"awakening": {
		"char": "?",
		"color": YELLOW,
		"name": "awakening",
	}
}

creatures = {
	"human": {
		"char": "@",
		"color": WHITE,
		"name": "human",
		"dice_size": 8,
		"muscle": 3,
		"stamina": 3,
		"agility": 3,
		"speed": 3,
		"senses": 3
	},

	"snake": {
		"char": "s",
		"color": GREEN,
		"name": "snake",
		"dice_size": 4,
		"muscle": 3,
		"stamina": 3,
		"agility": 5,
		"speed": 5,
		"can_swim": True,
		"drops": [trophies["gem"], None],
		"mind": "animal",
		"special": ("poison", 4, 4)
	},

	"rat": {
		"char": "r",
		"color": RED,
		"name": "rat",
		"dice_size": 4,
		"muscle": 3,
		"stamina": 3,
		"agility": 5,
		"speed": 5,
		"can_swim": True,
		"mind": "animal"
	},
	
	"bat": {
		"char": "b",
		"color": RED,
		"name": "vampire bat",
		"dice_size": 4,
		"muscle": 4,
		"stamina": 4,
		"agility": 5,
		"speed": 5,
		"can_fly": True,
		"mind": "animal"
	},
	
	"centipede": {
		"char": "c",
		"color": YELLOW,
		"name": "centipede",
		"dice_size": 4,
		"muscle": 2,
		"stamina": 2,
		"agility": 6,
		"speed": 6,
		"mind": "animal",
		"special": ("poison", 4, 4)
	},
	
	"worms": {
		"char": "w",
		"color": YELLOW,
		"name": "mass of worms",
		"dice_size": 6,
		"muscle": 5,
		"stamina": 6,
		"agility": 5,
		"speed": 5,
		"mind": "animal"
	},

	"zombie": {
		"char": "Z",
		"color": GREEN,
		"name": "zombie",
		"dice_size": 8,
		"muscle": 5,
		"stamina": 6,
		"agility": 4,
		"speed": 3,
		"mind": "undead"
	},

	"skeleton": {
		"char": "K",
		"color": WHITE,
		"name": "skeleton",
		"dice_size": 8,
		"muscle": 5,
		"stamina": 4,
		"agility": 5,
		"speed": 5,
		"drops": [trophies["bones"], None],
		"mind": "undead"
	},

	"mummy": {
		"char": "M",
		"color": YELLOW,
		"name": "mummy",
		"dice_size": 8,
		"muscle": 5,
		"stamina": 5,
		"agility": 5,
		"speed": 6,
		"drops": [trophies["gold"], None],
		"mind": "undead"
	},

	"ghoul": {
		"char": "G",
		"color": GREEN,
		"name": "ghoul",
		"dice_size": 10,
		"muscle": 5,
		"stamina": 5,
		"agility": 4,
		"speed": 4,
		"mind": "undead"
	},

	"clockwork": {
		"char": "A",
		"color": BLUE,
		"name": "animated armor",
		"dice_size": 10,
		"muscle": 5,
		"stamina": 5,
		"agility": 5,
		"speed": 5,
		"drops": [trophies["key"], None],
		"mind": "undead"
	},
	
	"hellhound": {
		"char": "h",
		"color": RED,
		"name": "hellhound",
		"dice_size": 6,
		"muscle": 6,
		"stamina": 6,
		"agility": 6,
		"speed": 6,
		"drops": [trophies["ember"], None],
		"mind": "undead",
		"special": ("fire", 6, 6)
	},

	"nightmare": {
		"char": "N",
		"color": RED,
		"name": "nightmare",
		"dice_size": 12,
		"muscle": 4,
		"stamina": 4,
		"agility": 4,
		"speed": 4,
		"drops": [trophies["awakening"], None],
		"mind": "undead"
	}
}

levels = [
	{
		"terrain": terrain["village"],
		"name": "Abandoned village",
		"items": [items["berries"], items["berries"], items["mushroom"],
			weapons["stone"], weapons["stone"],
			items["torch"], armor["cloth"]],
		"creatures": [creatures["snake"], creatures["rat"]]
	},

	{
		"terrain": terrain["cave"],
		"name": "Shallow cave",
		"items": [items["ration"], items["mushroom"],
			items["torch"], items["torch"],
			armor["cloth"], weapons["pickaxe"], weapons["pickaxe"],
			weapons["stone"], weapons["stone"]],
		"creatures": [creatures["bat"], creatures["rat"],
			creatures["centipede"]]
	},
	{
		"terrain": terrain["cave"],
		"name": "Cave",
		"items": [items["ration"], items["mushroom"],
			items["torch"], items["torch"],
			armor["cloth"], armor["leather"],
			weapons["pickaxe"], weapons["pickaxe"],
			weapons["stone"], weapons["stone"]],
		"creatures": [creatures["worms"], creatures["centipede"]]
	},
	{
		"terrain": terrain["cave"],
		"name": "Deep cave",
		"items": [items["ration"], items["mushroom"],
			items["torch"], items["torch"], armor["leather"],
			weapons["pickaxe"], weapons["pickaxe"],
			weapons["stone"], weapons["stone"]],
		"creatures": [creatures["worms"], creatures["zombie"],
			creatures["centipede"]]
	},

	{
		"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (7, 7),
		"items": [items["torch"], items["torch"],
			weapons["stone"], weapons["bullet"],
			weapons["pickaxe"], weapons["pickaxe"], weapons["mace"],
			armor["chainmail"], shields["leather"]],
		"creatures": [creatures["skeleton"], creatures["worms"],
			creatures["zombie"]]
	},
	{
		"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (7, 7),
		"items": [items["torch"], items["torch"], weapons["pickaxe"],
			weapons["bullet"], weapons["mace"], weapons["sword"],
			armor["chainmail"], shields["leather"]],
		"creatures": [creatures["skeleton"], creatures["mummy"],
			creatures["hellhound"]]
	},
	{
		"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (7, 7),
		"items": [items["torch"], items["torch"], weapons["pickaxe"],
			weapons["bullet"], weapons["mace"], weapons["sword"],
			armor["chainmail"], shields["leather"]],
		"creatures": [creatures["skeleton"], creatures["mummy"],
			creatures["ghoul"]]
	},
	{
		"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (5, 5),
		"items": [items["torch"], items["torch"], weapons["pickaxe"],
			weapons["bullet"], weapons["sword"], weapons["staff"],
			armor["scalemail"], shields["wood"]],
		"creatures": [creatures["ghoul"], creatures["mummy"],
			creatures["hellhound"]]
	},
	{
		"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (5, 5),
		"items": [items["torch"], items["torch"], weapons["pickaxe"],
			weapons["bullet"], weapons["spear"], weapons["staff"],
			armor["scalemail"], shields["wood"]],
		"creatures": [creatures["ghoul"], creatures["mummy"],
			creatures["clockwork"]]
	},
	{
		"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (5, 5),
		"items": [items["torch"], items["torch"], weapons["pickaxe"],
			weapons["bullet"], weapons["spear"], weapons["staff"],
			armor["scalemail"], shields["wood"]],
		"creatures": [creatures["hellhound"],
			creatures["ghoul"], creatures["clockwork"]]
	},
	{	"terrain": terrain["tomb"],
		"name": "Catacomb",
		"cellsize": (3, 3),
		"items": [items["torch"], items["torch"], weapons["pickaxe"],
			weapons["bullet"], weapons["spear"],
			armor["splintmail"], shields["bronze"]],
		"creatures": [creatures["hellhound"], creatures["nightmare"],
			creatures["ghoul"], creatures["clockwork"]]
	},
]

scrheight = -1
scrwidth = -1

world = None
worldrng = None
log = None

title_screen = None
character_screen = None
inventory_screen = None
game_screen = None
active_screen = None

finished = False

def curses_main(stdscr):
	global scrheight, scrwidth, world, worldrng, log, active_screen
	global title_screen, game_screen, character_screen, inventory_screen
	
	scrheight, scrwidth = stdscr.getmaxyx()

	if (scrheight < 24 or scrwidth < 80):
		raise RuntimeError("80x24 or larger terminal required.")

	if 'mousemask' in curses.__dict__:
		curses.mousemask(
			curses.BUTTON1_CLICKED | curses.BUTTON1_DOUBLE_CLICKED)
	stdscr.leaveok(0)
	if curses.has_colors():
		curses.init_pair(
			RED, curses.COLOR_RED, curses.COLOR_BLACK)
		curses.init_pair(
			GREEN, curses.COLOR_GREEN, curses.COLOR_BLACK)
		curses.init_pair(
			BLUE, curses.COLOR_CYAN, curses.COLOR_BLACK)
		curses.init_pair(
			YELLOW, curses.COLOR_YELLOW, curses.COLOR_BLACK)
	
	worldrng = random.Random()

	title_screen = TitleScreen(stdscr)
	character_screen = CharacterScreen(stdscr)
	inventory_screen = InventoryScreen(stdscr)
	game_screen = GameScreen(stdscr)
	active_screen = title_screen
	
	while not finished:
		if (scrheight < 24 or scrwidth < 80):
			stdscr.erase()
			stdscr.addstr(
				0, 0, "80x24 or larger terminal required.")
			stdscr.refresh()
		else:
			active_screen.render()
		key = stdscr.getch()
		if key == curses.KEY_RESIZE:
			scrheight, scrwidth = stdscr.getmaxyx()
		else:
			active_screen.handle_key(key)

def draw_dialog_bg(window, top, left, height, width):
	window.attron(curses.A_REVERSE)
	for y in range(height):
		window.addstr(top + y, left, " " * width)
	rectangle(window, top, left, top + height - 1, left + width - 1)
	window.attroff(curses.A_REVERSE)
	
def draw_help_dialog(window):
	draw_dialog_bg(window, 2, 15, 20, 52)
	window.addstr(3, 16, "How to play".center(50))
	for i in range(0, len(help_text)):
		window.addstr(5 + i, 16, help_text[i], curses.A_REVERSE)
	window.addstr(20, 16, "Press any key".center(50))

def draw_help_bar(window, message):
	window.addstr(scrheight - 1, 0,
		message.ljust(scrwidth - 1), curses.A_REVERSE)

def prompt(window, message):
	draw_dialog_bg(window, 10, 15, 4, 52)
	window.addstr(11, 16, message[0:50].center(50), curses.A_REVERSE)
	window.addstr(12, 16, " " * 50)
	curses.echo()
	answer = window.getstr(12, 16, 50)
	curses.noecho()
	return answer.decode()

def menu(window, message, message2, options):
	draw_dialog_bg(window, 2, 15, 20, 52)
	window.addstr(3, 16, message.center(50), curses.A_REVERSE)
	window.addstr(4, 16, message2.center(50), curses.A_REVERSE)
	draw_help_bar(window,
		"Arrows or jk to select, space or Enter to confirm, "
		"Esc or q to cancel.")
	
	selected = 0
	
	while True:
		for i, o in enumerate(options):
			if i == selected:
				window.addstr(i + 6, 16, o.center(50))
			else:
				window.addstr(i + 6, 16,
					o.center(50),
					curses.A_REVERSE)
		key = window.getch()
		if key == curses.KEY_DOWN or key == ord('j'):
			selected += 1
			if selected > len(options) - 1:
				selected = 0
		elif key == curses.KEY_UP or key == ord('k'):
			selected -= 1
			if selected < 0:
				selected = len(options) - 1
		elif key == 10 or key == ord(' ') or key == curses.KEY_ENTER:
			if len(options) > 0:
				return options[selected]
			else:
				return None
		elif key == 27 or key == ord('q'):
			return None
		else:
			curses.flash()

def next_coords(x, y, direction):
	if direction == 'n':
		return x, y - 1
	elif direction == 's':
		return x, y + 1
	elif direction == 'e':
		return x + 1, y
	elif direction ==  'w':
		return x - 1, y
	else:
		raise ValueError("Invalid compass direction " + str(direction))

def roll_dice(count, sides):
	roll = 0
	for i in range(count):
		roll += random.randint(1, sides)
	return roll

def count_items(items):
	group = {}
	for i in items:
		if i.name in group:
			group[i.name] += 1
		else:
			group[i.name] = 1
	return group

class TitleScreen:
	def __init__(self, stdscr):
		self.stdscr = stdscr
	
	def render(self):
		self.stdscr.clear()

		self.stdscr.addstr(0, 0,
			"Tomb of the Snake".ljust(scrwidth), curses.A_REVERSE)

		for i in range(4):
			self.stdscr.addstr(
				2 + i * 2, 0, blurb[i].center(scrwidth))
		
		if curses.has_colors():
			attr = curses.color_pair(RED) | curses.A_BOLD
		else:
			attr = 0
		for i in range(4):
			self.stdscr.addstr(10 + i, 0,
				title_banner[i].center(scrwidth), attr)
		self.stdscr.addstr(17, 0,
			"No Time To Play, 2021".center(scrwidth))
		if curses.has_colors():
			attr = curses.color_pair(BLUE)
		else:
			attr = 0
		self.stdscr.addstr(19, 0,
			"http://notimetoplay.org/".center(scrwidth), attr)

		draw_help_bar(self.stdscr,
			"[N]ew game [C]ontinue game [H]ow to play [Q]uit")
		
		self.stdscr.refresh()
			
	def handle_key(self, key):
		global finished, world, log, active_screen
		
		if key == ord('q') or key == 27:
			finished = True
		elif key == ord('n'):
			log = GameLog()
			world = GameWorld()
			world.player.log = log
			
			name = prompt(self.stdscr, "What's your name, hero?")
			world.player.name = name[0:50]

			active_screen = game_screen
		elif key == ord('c'):
			if world == None:
				world = GameWorld()
				log = GameLog()
				world.player.log = log
			
				name = prompt(
					self.stdscr, "What's your name, hero?")
				world.player.name = name[0:50]
			active_screen = game_screen
		elif key == ord('h'):
			draw_help_dialog(self.stdscr)
			self.stdscr.getch()
		else:
			curses.flash()

class CharacterScreen:
	def __init__(self, stdscr):
		self.stdscr = stdscr

	def render(self):
		self.stdscr.clear()

		self.stdscr.addstr(0, 0,
			"Character sheet".ljust(scrwidth), curses.A_REVERSE)

		p = world.player
		self.stdscr.addstr(2, 2, p.name)
		self.stdscr.hline(3, 2, curses.ACS_HLINE, len(p.name))
		self.stdscr.addstr(5, 2,  "Muscle:  %2d" % (p.muscle,))
		self.stdscr.addstr(7, 2,  "Stamina: %2d" % (p.stamina,))
		self.stdscr.addstr(9, 2,  "Agility: %2d" % (p.agility,))
		self.stdscr.addstr(11, 2, "Speed:   %2d" % (p.speed,))
		self.stdscr.addstr(13, 2, "Senses:  %2d" % (p.senses,))
		
		self.stdscr.vline(5, 39, curses.ACS_VLINE, 9)

		self.stdscr.addstr(5, 41,
			"Dice size: %d" % (p.dice_size,))
		self.stdscr.addstr(7, 41,
			"Life: %d of %d" % (p.life, p.stamina * p.dice_size))
		ap = p.speed * p.dice_size
		self.stdscr.addstr(9, 41,
			"Action points: %d of %d" % (p.action_points, ap))
		self.stdscr.addstr(11, 41,
			"Experience: %d" % (p.experience,))
		effects = " / ".join(
			[fxdescs[i] for i in p.effects.keys()])		
		self.stdscr.addstr(13, 41, effects)
			
		self.stdscr.hline(15, 2, curses.ACS_HLINE, 76)
		
		self.stdscr.addstr(17, 2, "Weapon: " + str(p.weapon))
		self.stdscr.addstr(19, 2, "Ammo: " + str(p.ammo))
		self.stdscr.addstr(17, 41, "Shield: " + str(p.shield))
		self.stdscr.addstr(19, 41, "Armor: " + str(p.armor))

		draw_help_bar(self.stdscr, "[I]nventory [B]ack to game")
		
		self.stdscr.refresh()
			
	def handle_key(self, key):
		global active_screen
		
		if key == ord('b') or key == 27:
			active_screen = game_screen
		if key == ord('i'):
			active_screen = inventory_screen
		else:
			curses.flash()

class InventoryScreen:
	def __init__(self, stdscr):
		self.stdscr = stdscr

	def render(self):
		self.stdscr.clear()

		self.stdscr.addstr(0, 0,
			"Inventory".ljust(scrwidth), curses.A_REVERSE)

		group = count_items(world.player.content)
		
		row = 2
		for i in group.keys():
			self.stdscr.addstr(
				row, 1, "%2dx %-40s" % (group[i], i))
			row += 1

		draw_help_bar(self.stdscr, "[C]haracter sheet [B]ack to game")
			
	def handle_key(self, key):
		global active_screen
		
		if key == ord('b') or key == 27:
			active_screen = game_screen
		if key == ord('c'):
			active_screen = character_screen
		else:
			curses.flash()

class GameScreen:
	def __init__(self, stdscr):
		self.stdscr = stdscr
		self.target_mode = False
		self.look_mode = False
		self.target_squares = []
		self.terrain = None
	
	def render(self):
		self.render_status(world.player)
		self.render_map(world.player.level)
		if world.player.is_dead:
			for i in range(5):
				self.stdscr.addstr(9 + i, 0,
					death_banner[i].center(80))
		self.render_log()
		if world.player.is_dead:
			draw_help_bar(self.stdscr, "? @ [I]nventory [M]enu")
		else:
			draw_help_bar(self.stdscr,
				"? @ [I]nventory [G]et [D]rop [E]at "
				"[W]ield [F]ire e[X]amine [T]unnel [M]enu")
		
		self.stdscr.refresh()
		
	def render_status(self, player):
		l = player.level
		status = "Level: %d Life:%3d AP:%3d" % (
			l.depth,
			player.life,
			player.action_points)
		status = status.rjust(scrwidth)
		self.stdscr.addstr(0, 0, status, curses.A_REVERSE)
		self.stdscr.addstr(0, 0, l.name, curses.A_REVERSE)
		
	def render_map(self, level):
		lmap = level.level_map
		self.terrain = world.player.level.terrain
		
		x1, y1, x2, y2 = self.update_seen()
		
		self.stdscr.attron(curses.A_DIM)
		for y in range(level.height):
			for x in range(level.width):
				if level.seen[y][x]:
					self.render_tile(x, y + 1, lmap[y][x])
				else:
					self.stdscr.addch(y + 1, x, " ")
		self.stdscr.attroff(curses.A_DIM)
		
		for y in range(y1, y2 + 1):
			for x in range(x1, x2 + 1):
				self.render_tile(x, y + 1, lmap[y][x])
		
		self.target_squares = []
		self.render_things(level, Item)
		self.render_things(level, Creature)
	
	def render_tile(self, x, y, char):
		if curses.has_colors():
			self.stdscr.attron(
				curses.color_pair(
					self.terrain[char]["color"]))
		self.stdscr.addch(y, x, char)
		if curses.has_colors():
			self.stdscr.attroff(
				curses.color_pair(
					self.terrain[char]["color"]))
	
	def render_things(self, level, thing_class):
		self.stdscr.attron(curses.A_BOLD)
		for i in level.content:
			if not isinstance(i, thing_class):
				continue
			elif not world.player.can_see(i.x, i.y):
				continue
			
			char = i.char
			if i != world.player:
				square = (i.x, i.y)
				if square not in self.target_squares:
					self.target_squares.append(square)
				target_num = self.target_squares.index(square)
				if self.target_mode or self.look_mode:
					if target_num < 10:
						char = str(target_num)
				
			if curses.has_colors():
				self.stdscr.attron(
					curses.color_pair(i.color))
			self.stdscr.addch(i.y + 1, i.x, char)
			if curses.has_colors():
				self.stdscr.attroff(
					curses.color_pair(i.color))
		self.stdscr.attroff(curses.A_BOLD)
	
	def update_seen(self):
		p = world.player
		radius = p.sight_radius
		x1 = max(0, p.x - radius)
		y1 = max(0, p.y - radius)
		x2 = min(p.level.width - 1, p.x + radius)
		y2 = min(p.level.height - 1, p.y + radius)
		
		for y in range(y1, y2 + 1):
			for x in range(x1, x2 + 1):
				p.level.seen[y][x] = True
		
		return x1, y1, x2, y2
			
	def render_log(self):
		num_msg = scrheight - 22
		tail = log.messages[-num_msg:]
		for i, t in enumerate(tail):
			self.stdscr.addstr(21 + i, 0, t.ljust(scrwidth))
		
	def play_ending(self):
		self.stdscr.timeout(2400)
		for i, e in enumerate(ending_text):
			self.stdscr.addstr(3 + i * 2, 4, e, curses.A_BOLD)
			self.stdscr.getch()
		draw_help_bar(self.stdscr, "Press any key")
		self.stdscr.timeout(-1)
		self.stdscr.getch()

	def handle_key(self, key):
		global active_screen
		
		if key == ord('m') or key == 27:
			self.look_mode = False
			self.target_mode = False
			active_screen = title_screen
		elif key == ord('@'):
			self.look_mode = False
			self.target_mode = False
			active_screen = character_screen
		elif key == ord('i'):
			self.look_mode = False
			self.target_mode = False
			active_screen = inventory_screen
		elif key == ord('?'):
			self.look_mode = False
			self.target_mode = False
			draw_help_dialog(self.stdscr)
			self.stdscr.getch()
		elif world.player.is_dead:
			curses.flash()
			return
		
		if world.player.action_points <= 0:
			world.update()
		
		if key in dirkeys:
			self.look_mode = False
			self.target_mode = False
			if world.player.walk(dirkeys[key]):
				world.update()
		elif key == ord('>'):
			self.look_mode = False
			self.target_mode = False
			p = world.player
			xp = p.experience
			p.descend()
			if p.experience > xp:
				# PC just leveled up.
				self.render()
				trait = menu(self.stdscr, "Ding! New level!",
					"Select trait to upgrade:",
					["muscle", "stamina", "agility",
						"speed", "senses"])
				if trait != None:
					p.__dict__[trait] += 1
					p.log("You feel %s." % (
						traitadjs[trait],))
				else:
					p.wounds = 0
					p.log("Your wounds heal.")
		elif key == ord('<'):
			self.look_mode = False
			self.target_mode = False
			p = world.player
			p.ascend()
			if p.level == world.level0:
				if world.mcguffin in p.content:
					self.render()
					self.play_ending()
					p.level.level_map[p.y][p.x] = ord('=')
		elif key == ord('g') or key == ord(','):
			self.look_mode = False
			self.target_mode = False
			world.player.get_item_here()
			world.update()
		elif key == ord('d'):
			self.look_mode = False
			self.target_mode = False
			itlist = count_items(world.player.content).keys()
			it = menu(self.stdscr, "Drop what?", "", itlist)
			if it != None:
				world.player.drop(it)
				world.update()
			else:
				world.player.log("Canceled.")
			world.update()
		elif key == ord('e'):
			self.look_mode = False
			self.target_mode = False
			foods = [i for i in world.player.content if i.food > 0]
			itlist = list(count_items(foods).keys())
			it = menu(self.stdscr, "Eat what?", "", itlist)
			if it != None:
				world.player.eat(it)
				world.update()
			else:
				world.player.log("Canceled.")
			world.update()
		elif key == ord('w'):
			self.look_mode = False
			self.target_mode = False
			if self.wield_or_wear():
				world.update()
		elif key == ord('f'):
			self.look_mode = False
			self.target_mode = not self.target_mode
		elif key == ord('x'):
			self.target_mode = False
			self.look_mode = not self.look_mode
		elif ord('0') <= key <= ord('9'):
			tnum = key - ord('0')
			if not self.target_mode and not self.look_mode:
				world.player.log("But you're not aiming!")
			elif tnum >= len(self.target_squares):
				world.player.log("Not enough targets.")
			elif self.target_mode:
				x, y = self.target_squares[tnum]
				if world.player.fire_at(x, y):
					world.update()
				self.target_mode = False
			elif self.look_mode:
				x, y = self.target_squares[tnum]
				world.player.look(x, y)
				self.look_mode = False
		elif key == ord('t'):
			self.look_mode = False
			self.target_mode = False
			draw_help_bar(self.stdscr, "Tunnel which way?")
			direction = self.stdscr.getch()
			if direction in dirkeys:
				world.player.tunnel(dirkeys[direction])
				world.update()
			else:
				world.player.log("Canceled.")
		elif key == ord('r'):
			self.look_mode = False
			self.target_mode = False
			draw_help_bar(self.stdscr, "Run which way?")
			direction = self.stdscr.getch()
			if direction in dirkeys:
				self.stdscr.timeout(500)
				d = dirkeys[direction]
				while world.player.is_way_clear(d):
					world.player.walk(d)
					world.update()
					self.render()
					if self.stdscr.getch() > -1:
						break
				self.stdscr.timeout(-1)
			else:
				world.player.log("Canceled.")
		elif key == ord('.'):
			self.look_mode = False
			self.target_mode = False
			world.player.log("You stand still.")
			world.update()
		elif key == curses.KEY_MOUSE:
			self.look_mode = False
			self.target_mode = False
			device, x, y, z, button = curses.getmouse()
			if button == curses.BUTTON1_CLICKED:
				world.player.look(x, y - 1)
			elif button == curses.BUTTON1_DOUBLE_CLICKED:
				if world.player.fire_at(x, y - 1):
					world.update()
		else:
			curses.flash()
	
	def wield_or_wear(self):
		slot = menu(self.stdscr,
			"Wield/wear equipment", "Choose slot:",
			["weapon", "ammo", "shield", "armor"])
		if slot == None:
			world.player.log("Canceled.")
			return False
		elif slot == "weapon":
			attr = "attack"
		elif slot == "ammo":
			attr = "ammo"
		elif slot == "shield":
			attr = "defense"
		elif slot == "armor":
			attr = "armor"
		
		equipment = self.select_equipment(slot, attr)
		if equipment == None:
			world.player.log("Canceled.")
			return False
		elif equipment == "none":
			equipment = None
			
		world.player.equip(equipment, slot)
		return True
	
	def select_equipment(self, slot, attr):
		shortlist = [i for i in world.player.content
			if i.__dict__[attr] > 0]
		shortlist = ["none"] + list(count_items(shortlist).keys())
		return menu(self.stdscr, "Choose " + slot, "", shortlist)

class GameLog:
	def __init__(self):
		self.messages = []
	
	def say(self, message):
		message = message[0].capitalize() + message[1:]
		self.messages.append(message)
		while len(self.messages) > 100:
			del self.messages[0]

	def call_attack(self, attacker, defender, damage):
		name1 = attacker.the_name()
		name2 = defender.the_name()
		if damage <= 0:
			self.say("%s pokes %s harmlessly." % (name1, name2))
		elif damage > defender.life:
			msg = "%s strikes %s for %d damage, killing %s!"
			msg  = msg % (name1, name2, damage, defender.pronoun)
			self.say(msg)
		else:
			msg = "%s strikes %s for %d damage."
			msg  = msg % (name1, name2, damage)
			self.say(msg)

	def __call__(self, message):
		self.say(message)

class GameWorld:
	def __init__(self):
		self.animal_ai = AnimalAI()
		self.undead_ai = UndeadAI()

		self.mcguffin = Item(items["idol"])
		self.player = Creature(creatures["human"])
		self.player.pronoun = "them"
		self.player.proper_name = True
		self.level0 = DungeonLevel(0, levels[0])
		self.level0.populate()
		
		self.player.move_to(0, 10, self.level0)
	
	def update(self):
		stuff = self.player.level.content
		for i in stuff.copy():
			if not isinstance(i, Creature):
				continue
			self.update_effects(i)
			if i.is_dead:
				drop = random.choice(i.drops)
				if drop != None:
					drop = Item(drop)
					drop.move_to(i.x, i.y, i.level)
				i.move_to(-1, -1, None)
			elif i.action_points <= 0:
				i.action_points += i.speed * i.dice_size
			elif i.mind == "animal":
				self.animal_ai.take_turn(i)
			elif i.mind == "undead":
				self.undead_ai.take_turn(i)
		self.update_torch(self.player, "weapon")
		self.update_torch(self.player, "shield")
	
	def update_torch(self, player, slot):
		item = player.__dict__[slot]
		if item != None and item.light > 0:
			item.light -= 1
			if item.light == 0:
				player.__dict__[slot] = None
				player.log("Your torch burns out.")
	
	def update_effects(self, creature):
		for i in creature.effects.copy().keys():
			effect = creature.effects[i]
			if effect["turns"] < 1:
				del creature.effects[i]
			else:
				effect["turns"] -= 1
				creature.wounds += roll_dice(
					1, effect["damage"])

class DungeonLevel:
	width = 80
	height = 20
	
	def __init__(self, depth, template):
		self.depth = depth

		self.name = template["name"]
		self.terrain = template["terrain"]
		self.items = template["items"]
		self.creatures = template["creatures"]
		#self.light_level = 1

		self.level_map = [bytearray(self.width)
			for i in range(self.height)]
		self.seen = [[False] * self.width for i in range(self.height)]
		self.prev_level = None
		self.next_level = None
		
		self.content = set()
	
	def is_on_grid(self, x, y):
		return x >= 0 and y >= 0 and x < self.width and y < self.height
	
	def terrain_at(self, x, y):
		tile = self.level_map[y][x]
		if tile in self.terrain.keys():
			return self.terrain[tile]
		else:
			return None
	
	def items_at(self, x, y):
		return (i for i in self.content
			if isinstance(i, Item) and i.x == x and i.y == y)
	
	def creature_at(self, x, y):
		for i in self.content:
			if isinstance(i, Creature) and i.x == x and i.y == y:
				return i
		return None
	
	def dig_next(self):
		if self.next_level != None:
			return
		
		self.next_level = DungeonLevel(
			self.depth + 1, levels[self.depth + 1])
		self.next_level.prev_level = self
		self.next_level.populate()
	
	def populate(self):
		if self.terrain == terrain["village"]:
			gen = VillageLevelGenerator(worldrng)
		elif self.terrain == terrain["cave"]:
			gen = CaveLevelGenerator(worldrng)
		elif self.terrain == terrain["tomb"]:
			gen = TombLevelGenerator(worldrng)
		else:
			raise RuntimeError("Unknown terrain type")
		
		gen.populate(self)

class Thing:
	def __init__(self):
		self.level = None
		self.x = -1
		self.y = -1
		
		self.name = "thing"
		self.pronoun = "it"
		self.proper_name = False
		self.special = None
	
	def move_to(self, x, y, parent = None):
		l = self.level
		
		if hasattr(l, "content") and self in l.content:
			l.content.remove(self)
		if hasattr(parent, "content"):
			parent.content.add(self)
			self.level = parent
		
		if not isinstance(parent, DungeonLevel): return

		if self.level.is_on_grid(x, y):
			self.x = x
			self.y = y
		else:
			raise ValueError("Can't move outside the level map.")

	def the_name(self):
		if self.proper_name:
			return self.name
		else:
			return "the " + self.name
	
	def a_name(self):
		if self.name[0] in ('a', 'e', 'i', 'o', 'u'):
			return "an " + self.name
		else:
			return "a " + self.name

	def __str__(self):
		return self.a_name()

class Item(Thing):
	def __init__(self, template):
		Thing.__init__(self)
		self.char = "?"
		
		self.weight = 0
		self.food = 0
		self.attack = 0
		self.defense = 0
		self.damage = 0
		self.ammo = 0
		self.armor = 0
		self.light = 0
		
		for attr in template.keys():
			self.__dict__[attr] = template[attr]

class Creature(Thing):
	def __init__(self, template):
		Thing.__init__(self)
		self.log = None

		self.char = "@"
		
		self.dice_size = 8
		self.muscle = 3
		self.agility = 3
		self.stamina = 3
		self.speed = 3
		self.senses = 3
		
		self.can_swim = False
		self.can_fly = False
		self.mind = None
		
		self.weapon = None
		self.shield = None
		self.armor = None
		self.ammo = None
		self.special = None
		
		self.drops = [None]
		
		for attr in template.keys():
			self.__dict__[attr] = template[attr]
		self.template = template
		
		self.wounds = 0
		self.experience = 0
		self.action_points = self.dice_size * self.speed
		
		self.content = set()
		self.effects = {}
		
		self.last_heading = None # For the AI system.

	def can_see(self, x, y):
		radius = self.sight_radius
		can_see_x = abs(self.x - x) <= radius
		can_see_y = abs(self.y - y) <= radius
		return can_see_x and can_see_y
		
	def can_enter(self, x, y):
		tile = chr(self.level.level_map[y][x])
		if tile in ('.', ',', '=', '>', '<'):
			return True
		elif tile == '~':
			return self.can_swim or self.can_fly
		else:
			return False

	def is_way_clear(self, direction):
		new_x, new_y = next_coords(self.x, self.y, direction)

		if not self.level.is_on_grid(new_x, new_y):
			return False
		elif self.level.creature_at(new_x, new_y) != None:
			return False
		elif not self.can_enter(new_x, new_y):
			return False
		else:
			return True

	def walk(self, direction):
		assert self.level != None, "Creature is out of play."
		new_x, new_y = next_coords(self.x, self.y, direction)

		if not self.level.is_on_grid(new_x, new_y):
			self.log("An invisible barrier stops you.")
			return False
		elif self.level.creature_at(new_x, new_y) != None:
			creature = self.level.creature_at(new_x, new_y)
			if self.is_enemy(creature):
				self.attack(creature)
			#else:
			#	self.say("You bump into " + creature.name)
			return True
		elif not self.can_enter(new_x, new_y):
			t = self.level.terrain_at(new_x, new_y)
			if t != None and t["name"] != None:
				self.log("The way is barred by "
					+ t["name"] + ".")
			else:
				self.log("The way is barred.")
			return False
		else:
			self.x = new_x
			self.y = new_y
			self.action_points -= 5
			return True

	def tunnel(self, direction):
		assert self.level != None, "Creature is out of play."
		new_x, new_y = next_coords(self.x, self.y, direction)
	
		if not self.level.is_on_grid(new_x, new_y):
			self.log("There's nothing in that direction.")
			return False
		elif self.level.level_map[new_y][new_x] != ord('#'):
			self.log("There's no wall in that direction.")
			return False
		
		for i in self.content.copy():
			if i.name == "pickaxe":
				self.level.level_map[new_y][new_x] = ord('.')
				self.content.remove(i)
				self.log("You break down the wall. "
					"The pickaxe falls apart.")
				self.action_points -= 20
				return True
		
		if self.weapon != None and self.weapon.name == "pickaxe":
				self.level.level_map[new_y][new_x] = ord('.')
				self.weapon = None
				self.log("You break down the wall. "
					"The pickaxe falls apart.")
				self.action_points -= 20
				return True

		self.log("You need a pickaxe for that.")
		return False
				
	def is_enemy(self, creature):
		return self.template != creature.template
	
	def attack(self, creature):
		if self.distance_to(creature) > 1:
			self.log(creature.the_name() +
				" is too far away for melee.")
			return False
		
		attack = roll_dice(self.agility, self.dice_size)
		if self.weapon != None:
			attack += self.weapon.attack
			if self.weapon.special != None:
				creature.apply_effect(self.weapon.special)
		elif self.special != None:
			creature.apply_effect(self.special)
		defense = roll_dice(creature.speed, creature.dice_size)
		if creature.shield != None:
			defense += creature.shield.defense
		
		if attack > defense:
			dmg_bonus = (self.muscle - 3) * self.dice_size // 2
			damage = (attack - defense) // 2 + dmg_bonus
			if self.weapon != None:
				damage += self.weapon.damage
			damage = creature.apply_armor(damage)
			if damage > 0:
				creature.wounds += damage
			self.log.call_attack(self, creature, damage)
		else:
			self.log("%s misses %s." %
				(self.the_name(), creature.the_name()))

		if self.weapon != None:
			self.action_points -= 4 + self.weapon.weight
		else:
			self.action_points -= 5
			
		return True
	
	def fire_at(self, x, y):
		if not self.can_see(x, y):
			self.log("You can't shoot what you can't see.")
			return False
		creature = self.level.creature_at(x, y)
		if creature == None:
			self.log("Nobody there for you to shoot.")
			return False
		distance = self.distance_to(creature)
		if distance < 2:
			self.log("You're too close to use your sling.")
			return False
		if self.ammo == None:
			self.log("You need ammo for your sling.")
			return False
		
		self.action_points -= 4 + self.ammo.weight
		
		attack = roll_dice(self.agility, self.dice_size)
		attack += self.ammo.ammo + self.senses - distance
		defense = roll_dice(creature.speed, creature.dice_size)
		if creature.shield != None:
			defense += creature.shield.defense

		if attack > defense:
			dmg_bonus = (self.muscle - 3) * self.dice_size // 2
			damage = (attack - defense) // 2 + dmg_bonus
			damage += self.ammo.damage
			damage = creature.apply_armor(damage)
			if damage > 0:
				creature.wounds += damage
				self.log("%s shoots %s for %d damage!" %
					(self.the_name(), creature.the_name(), damage))
			else:
				self.log("%s hits %s harmlessly." %
					(self.the_name(), creature.the_name()))
		else:
			self.ammo.move_to(x, y, self.level)
			self.log("%s's shot misses %s." %
				(self.the_name(), creature.the_name()))
		
		for item in self.content.copy():
			if item.name == self.ammo.name:
				self.ammo = item
				self.content.remove(item)
				return True
				
		self.ammo = None
		return True
	
	def distance_to(self, thing):
		if self.level != thing.level:
			return 2 * 1000 * 1000 * 1000
		else:
			return abs(self.x - thing.x) + abs(self.y - thing.y)
	
	def apply_armor(self, damage):
		if self.armor == None:
			return damage
		else:
			damage -= self.armor.armor
			if damage <= 0:
				return damage
			half = damage // 2
			self.armor.weight -= half
			if self.armor.weight < 0:
				self.log("Your armor falls apart.")
				self.armor = None
			return damage - half # In case damage was odd.
	
	def apply_effect(self, effect):
		kind, duration, damage = effect
		turns = roll_dice(1, duration)
		if kind in self.effects:
			self.effects[kind]["turns"] = max(
				turns, self.effects[kind]["turns"])
			self.effects[kind]["damage"] = max(
				damage, self.effects[kind]["damage"])
		else:
			self.effects[kind] = {"turns": turns, "damage": damage}
		self.log("%s is %s!" % (self.the_name(), fxdescs[kind]))
	
	@property
	def is_dead(self):
		return self.wounds >= self.dice_size * self.stamina
	
	@property
	def life(self):
		return int(self.dice_size * self.stamina - self.wounds)
	
	@property
	def sight_radius(self):
		w = self.weapon
		s = self.shield
		
		if self.level.depth == 0:
			modifier = 2
		else:
			modifier = 0
		if w != None and w.light > 0 and modifier < 2:
			modifier += 1
		if s != None and s.light > 0 and modifier < 2:
			modifier += 1
		radius = self.senses * modifier
		if radius < 1:
			radius = self.senses // 2
		return radius
	
	def descend(self):
		tile = chr(self.level.level_map[self.y][self.x])
		if tile == '>':
			if self.level.next_level == None:
				self.level.dig_next()
			self.move_to(self.x, self.y, self.level.next_level)
			if self.level.depth > self.experience:
				self.experience = self.level.depth
				if self.wounds > 0:
					self.wounds -= 1
			self.action_points -= 5
		else:
			self.log("There's no way down from here.")
		
	def ascend(self):
		tile = self.level.level_map[self.y][self.x]
		if tile == ord('<'):
			self.move_to(self.x, self.y, self.level.prev_level)
			self.action_points -= 5
		else:
			self.log("There's no way up from here.")
	
	def look(self, x, y):
		if not self.level.is_on_grid(x, y):
			return
		elif not self.can_see(x, y):
			self.log("You can't see that far right now.")
			return
		
		tile = self.level.level_map[y][x]
		msg = "You see: " + self.level.terrain[tile]["name"]
		for i in self.level.content:
			if i.x == x and i.y == y:
				if i == self:
					msg += ", yourself"
				else:
					msg += ", " + i.name
		self.log(msg)
	
	def get_item_here(self):
		item = next(self.level.items_at(self.x, self.y), None)
		if item == None:
			self.log("There's nothing portable here.")
			return
		
		item.move_to(-1, -1, self)
		self.log("You pick up the " + item.name)
		self.action_points -= 5
	
	def drop(self, name):
		for item in self.content.copy():
			if item.name == name:
				item.move_to(self.x, self.y, self.level)
				self.log("Dropped.")
				self.action_points -= 5
				return
		self.log("But you don't have a %s!" % (name,))
	
	def eat(self, name):
		for item in self.content.copy():
			if item.name == name:
				if item.food < 1:
					self.log("That's not edible.")
					return
				elif self.wounds < 1:
					self.log("But you're feeling full!")
					return
					
				self.content.remove(item)
				self.wounds -= item.food
				if self.wounds < 0:
					self.wounds = 0
				self.log("You eat the %s." % (item.name,))
				self.action_points -= 10
				return
		self.log("But you don't have a %s!" % (name,))
	
	def equip(self, name, slot):
		if name != None:
			for item in self.content.copy():
				if item.name == name:
					prev_gear = self.__dict__[slot]
					self.__dict__[slot] = item
					self.content.remove(item)
					if prev_gear != None:
						self.content.add(prev_gear)
					self.log(slot + " changed.")
					self.action_points -= 10
					return
			self.log("But you don't have a %s!" % (name,))
		else:
			if self.__dict__[slot] != None:
				self.content.add(self.__dict__[slot])
				self.__dict__[slot] = None
			self.action_points -= 5
	

class VillageLevelGenerator:
	def __init__(self, rng):
		self.rng = rng
		self.road_coords = []
		self.last_building = None
	
	def populate(self, level):
		for y in range(level.height):
			for x in range(level.width):
				level.level_map[y][x] = ord('.')
				
		self.make_trees(level)
		self.make_road(level)
		self.make_river(level)
		self.place_buildings(level)

		for y in range(level.height):
			for x in range(level.width):
				self.maybe_place_item(level, x, y)

		for y in range(level.height):
			for x in range(level.width):
				self.maybe_place_creature(level, x, y)
				
	def make_trees(self, level):
		size = level.width * level.height
		
		for i in range(size // 5):
			position = self.rng.randint(0, size - 1)
			x = position % level.width
			y = position // level.width
			if (position % 2 == 0):
				level.level_map[y][x] = ord('|')
			else:
				level.level_map[y][x] = ord('^')
				
	def make_road(self, level):
		y = level.height // 2 - 1
		for x in range(level.width):
			level.level_map[y][x] = ord(',')
			level.level_map[y + 1][x] = ord(',')
			self.road_coords.append(y)
			if (y <= 3):
				y += self.rng.choice([0, 0, 0, 0, 0, 0, 1])
			elif (y >= level.height - 3):
				y += self.rng.choice([-1, 0, 0, 0, 0, 0, 0])
			else:
				y += self.rng.choice([-1, 0, 0, 0, 0, 0, 1])
				
	def make_river(self, level):
		x = int(level.width * 0.4)
		for y in range(level.height):
			if level.level_map[y][x] == ord(','):
				level.level_map[y][x] = ord('=')
			else:
				level.level_map[y][x] = ord('~')
			delta_x = self.rng.randint(0, 2)
			if delta_x == 2: delta_x = 1
			x += delta_x
			
	def place_buildings(self, level):
		w = level.width
		# West of the river.
		self.place_building_cluster(level, int(w * 0.1), int(w * 0.4))
		# East of the river.
		self.place_building_cluster(level, int(w * 0.6), int(w * 0.9))
		
		x, y = self.last_building
		level.level_map[y][x] = ord('>')
		level.exit_down = self.last_building
		
	def place_building_cluster(self, level, from_x, to_x):
		h = level.height
		x = from_x
		while x < to_x:
			radius = self.rng.randint(1, 2)
			x += radius
			road_y = self.road_coords[x]
			if road_y < h // 2:
				y = road_y + int(h * 0.4)
			else:
				y = road_y - int(h * 0.4)
			self.make_building(level, x, y, radius)
			x += radius + 4
	
	def make_building(self, level, x, y, radius):
		for i in range(x - radius, x + radius + 1):
			for j in range(y - radius, y + radius + 1):
				level.level_map[j][i] = ord('#')
		# Make a door and clear the space in front of it.
		if y < level.height // 2:
			level.level_map[y + radius][x] = ord('.')
			level.level_map[y + radius + 1][x] = ord('.')
			level.level_map[y + radius + 1][x - 1] = ord('.')
			level.level_map[y + radius + 1][x + 1] = ord('.')
		else:
			level.level_map[y - radius][x] = ord('.')
			level.level_map[y - radius - 1][x] = ord('.')
			level.level_map[y - radius - 1][x - 1] = ord('.')
			level.level_map[y - radius - 1][x + 1] = ord('.')
		
		self.last_building = (x, y)
		
		radius -= 1;
		for i in range(x - radius, x + radius + 1):
			for j in range(y - radius, y + radius + 1):
				level.level_map[j][i] = ord('=')
	
	def maybe_place_item(self, level, x, y):
		tile = chr(level.level_map[y][x])
		if tile != '.' and tile != '=':
			return
		if self.rng.random() > 0.01:
			return
		
		item = Item(random.choice(level.items))
		item.move_to(x, y, level)
	
	def maybe_place_creature(self, level, x, y):
		tile = chr(level.level_map[y][x])
		if tile != '.' and tile != '=':
			return
		if self.rng.random() > 0.01:
			return
		
		creature = Creature(random.choice(level.creatures))
		creature.log = log
		creature.move_to(x, y, level)

class CaveLevelGenerator:
	def __init__(self, rng):
		self.rng = rng
		self.entrances = []
		self.far_cell = None
	
	def populate(self, level):
		for x in range (level.width):
			for y in range(level.height):
				level.level_map[y][x] = ord('#')

		assert level.prev_level != None

		x, y = level.prev_level.exit_down
		self.digRoomFrom(x, y, level)
		level.level_map[y][x] = ord('<')
		if level.depth % 2 == 1:
			while self.far_cell >= (20, 10):
				x, y = self.far_cell
				self.digRoomFrom(x, y, level)
		else:
			while self.far_cell < (70, 10):
				x, y = self.far_cell
				self.digRoomFrom(x, y, level)
		x, y = self.far_cell
		level.level_map[y][x] = ord('>')
		level.exit_down = self.far_cell

		self.makeWater(level)

		for y in range(level.height):
			for x in range(level.width):
				self.maybe_place_item(level, x, y)

		for y in range(level.height):
			for x in range(level.width):
				self.maybe_place_creature(level, x, y)
	
	def digRoomFrom(self, x, y, level):
		halfw = level.width // 2
		halfh = level.height // 2
		
		if level.depth % 2 == 1:
			directions = [(-1, 0), (-1, 0), (-1, 0),
				(0, -1), (0, -1), (1, 0), (0, 1)]
		else:
			directions = [(-1, 0), (0, -1),
				(1, 0), (1, 0), (1, 0), (0, 1), (0, 1)]
		self.far_cell = (x, y)
		for i in range(halfh):
			new_x = x
			new_y = y
			for j in range(halfw):
				direction = self.rng.choice(directions)
				dx, dy = direction
				new_x += dx
				new_y += dy
				if not self.advanceCorridor(
					new_x, new_y, level):
						break
	
	def advanceCorridor(self, x, y, level):
		if not(0 < x < level.width - 1 and 0 < y < level.height - 1):
			return False
		if level.level_map[y][x] == ord('#'):
			level.level_map[y][x] = ord('.')
		if level.depth % 2 == 1:
			if (x, y) < self.far_cell:
				self.far_cell = (x, y)
		else:
			if (x, y) > self.far_cell:
				self.far_cell = (x, y)
		return True
	
	def makeWater(self, level):
		min_x = 1
		min_y = 1
		max_x = level.width - 2
		max_y = level.height - 2
		
		water_tiles = []
		for y in range(min_y, max_y + 1):
			for x in range(min_x, max_x + 1):
				if level.level_map[y][x] != ord('#'):
					continue
				cnt = self.countNeighborTiles(
					x, y, level, ord('#'))
				if (cnt < 5):
					water_tiles.append((x, y))
		for i in water_tiles:
			x, y = i
			level.level_map[y][x] = ord('~')
	
	def countNeighborTiles(self, x, y, level, kind):
		count = 0
		for i in range(y - 1, y + 2):
			for j in range(x - 1, x + 2):
				if level.level_map[i][j] == kind:
					count += 1
		return count
	
	def maybe_place_item(self, level, x, y):
		tile = chr(level.level_map[y][x])
		if tile != '.':
			return
		if self.rng.random() > 0.02:
			return
		
		item = Item(random.choice(level.items))
		item.move_to(x, y, level)
	
	def maybe_place_creature(self, level, x, y):
		tile = chr(level.level_map[y][x])
		if tile != '.':
			return
		if self.rng.random() > 0.01:
			return
		
		creature = Creature(random.choice(level.creatures))
		creature.log = log
		creature.move_to(x, y, level)

class TombLevelGenerator:
	def __init__(self, rng):
		self.rng = rng
		self.cellw = -1
		self.cellh = -1
		self.level = None
	
	def populate(self, level):
		for x in range(level.width):
			for y in range(level.height):
				level.level_map[y][x] = ord('.')
		for x in range (level.width):
			level.level_map[0][x] = ord('#')
			level.level_map[level.height - 1][x] = ord('#')
		for y in range(level.height):
			level.level_map[y][0] = ord('#')
			level.level_map[y][level.width - 1] = ord('#')
		
		self.cellw, self.cellh = levels[level.depth]["cellsize"]
		self.level = level
		self.subdivideWide(1, 1, level.width - 2, level.height - 2)

		x, y = level.prev_level.exit_down
		level.level_map[y][x] = ord('<')
		if level.depth % 2 == 1:
			if level.depth < len(levels) - 1:
				level.level_map[1][1] = ord('>')
				level.exit_down = (1, 1)
			else:
				idol = Item(items["idol"])
				idol.move_to(1, 1, level)
		elif level.depth < len(levels) - 1:
			lh2 = level.height - 2
			lw2 = level.width - 2
			level.level_map[lh2][lw2] = ord('>')
			level.exit_down = (lw2, lh2)
		else:
			world.mcguffin.move_to(
				level.width - 2, level.height - 2, level)
			
	
	def subdivideWide(self, x1, y1, x2, y2):
		w = x2 - x1 + 1
		h = y2 - y1 + 1
		# You have to check both dimensions
		# or you'll get oddly skewed levels.
		if w < self.cellw or h < self.cellh or w == h:
			self.furnishRoom(x1, y1, x2, y2)
			return 0
		
		if w == 3:
			x = x1 + 1
		else:
			x = x1 + self.rng.randint(1, w - 2)
		for y in range(y1, y2 + 1):
			self.level.level_map[y][x] = ord('#')

		self.subdivideHigh(x1, y1, x - 1, y2)
		self.subdivideHigh(x + 1, y1, x2, y2)
		
		doory = y1 + self.rng.randint(0, h - 1)

		self.level.level_map[doory][x] = ord('.')
		# Account for walls placed deeper into recursion.
		self.level.level_map[doory][x - 1] = ord('.')
		self.level.level_map[doory][x + 1] = ord('.')
			
		return x

	def subdivideHigh(self, x1, y1, x2, y2):
		w = x2 - x1 + 1
		h = y2 - y1 + 1
		# You have to check both dimensions
		# or you'll get oddly skewed levels.
		if w < self.cellw or h < self.cellh or w == h:
			self.furnishRoom(x1, y1, x2, y2)
			return 0
		
		if h == 3:
			y = y1 + 1
		else:
			y = y1 + self.rng.randint(1, h - 2)
		for x in range(x1, x2 + 1):
			self.level.level_map[y][x] = ord('#')

		self.subdivideWide(x1, y1, x2, y - 1)
		self.subdivideWide(x1, y + 1, x2, y2)
		
		doorx = x1 + self.rng.randint(0, w - 1)

		self.level.level_map[y][doorx] = ord('.')
		# Account for walls placed deeper into recursion.
		self.level.level_map[y - 1][doorx] = ord('.')
		self.level.level_map[y + 1][doorx] = ord('.')
			
		return y

	def furnishRoom(self, x1, y1, x2, y2):
		w = x2 - x1 + 1
		h = y2 - y1 + 1
		lmap = self.level.level_map
		
		if w == 3 and h == 3:
			lmap[y1 + 1][x1 + 1] = ord('+')
		elif w == 3:
			if h % 2 == 1:
				for y in range(y1 + 1, y2, 2):
					lmap[y][x1 + 1] = ord('|')
			else:
				for y in range(y1 + 1, y2):
					if self.rng.randint(0, 1) == 0:
						lmap[y][x1 + 1] = ord('~')
		elif h == 3:
			if w % 2 == 1:
				for x in range(x1 + 1, x2, 2):
					lmap[y1 + 1][x] = ord('|')
			else:
				for x in range(x1 + 1, x2):
					if self.rng.randint(0, 1) == 0:
						lmap[y1 + 1][x] = ord('~')
		elif w == 5 and h == 5:
			lmap[y1 + 1][x1 + 1] = ord('+')
			lmap[y1 + 1][x1 + 3] = ord('+')
			lmap[y1 + 3][x1 + 1] = ord('+')
			lmap[y1 + 3][x1 + 3] = ord('+')
		elif w == 5:
			if h % 2 == 1:
				for y in range(y1 + 1, y2, 2):
					lmap[y][x1 + 1] = ord('|')
					lmap[y][x2 - 1] = ord('|')
			else:
				for y in range(y1 + 1, y2):
					if self.rng.randint(0, 1) == 0:
						lmap[y][x1 + 1] = ord('~')
					if self.rng.randint(0, 1) == 0:
						lmap[y][x2 - 1] = ord('~')
		elif h == 5:
			if w % 2 == 1:
				for x in range(x1 + 1, x2, 2):
					lmap[y1 + 1][x] = ord('|')
					lmap[y2 - 1][x] = ord('|')
			else:
				for x in range(x1 + 1, x2, 2):
					if self.rng.randint(0, 1) == 0:
						lmap[y1 + 1][x] = ord('~')
					if self.rng.randint(0, 1) == 0:
						lmap[y2 - 1][x] = ord('~')
		elif w == 7 and h == 7:
			for y in range(y1 + 1, y2, 2):
				for x in range(x1 + 1, x2, 2):
					lmap[y][x] = ord('|')
			lmap[y1 + 3][x1 + 3] = ord('+')
		elif w == 9 and h == 9:
			for y in range(y1 + 1, y2, 2):
				for x in range(x1 + 1, x2, 2):
					lmap[y][x] = ord('|')
			lmap[y1 + 3][x1 + 3] = ord('+')
			lmap[y1 + 3][x1 + 5] = ord('+')
			lmap[y1 + 5][x1 + 3] = ord('+')
			lmap[y1 + 5][x1 + 5] = ord('+')
		elif w > 5 and h > 5 and (w % 2 == 1 or h % 2 == 1):
			if h % 2 == 1:
				for y in range(y1 + 1, y2, 2):
					lmap[y][x1 + 1] = ord('#')
					lmap[y][x2 - 1] = ord('#')
			if w % 2 == 1:
				for x in range(x1 + 1, x2, 2):
					lmap[y1 + 1][x] = ord('#')
					lmap[y2 - 1][x] = ord('#')
			self.furnishRoom(x1 + 2, y1 + 2, x2 - 2, y2 - 2)
		else:
			for y in range(y1, y2 + 1):
				for x in range(x1, x2 + 1):
					if self.rng.random() > 0.01:
						continue
					item = Item(
						random.choice(
							self.level.items))
					item.move_to(x, y, self.level)

			for y in range(y1, y2 + 1):
				for x in range(x1, x2 + 1):
					if self.rng.random() >= 0.01:
						continue
					creature = Creature(
						random.choice(
							self.level.creatures))
					creature.log = log
					creature.move_to(x, y, self.level)


class AnimalAI:
	def take_turn(self, mob):
		p = world.player
		if mob.distance_to(p) == 1:
			mob.attack(p)
			return

		mob.last_heading = random.choice(['n', 's', 'e', 'w'])
		x, y = next_coords(mob.x, mob.y, mob.last_heading)
		if mob.level.is_on_grid(x, y) and mob.can_enter(x, y):
			mob.walk(mob.last_heading)

class UndeadAI:
	def take_turn(self, mob):
		p = world.player
		if mob.distance_to(p) == 1:
			mob.attack(p)
			mob.last_heading = None # To reset the chase mode.
			return
		elif mob.x == p.x:
			if mob.y < p.y:
				mob.last_heading = 's'
			elif mob.y > p.y:
				mob.last_heading = 'n'
		elif mob.y == p.y:
			if mob.x < p.x:
				mob.last_heading = 'e'
			elif mob.x > p.x:
				mob.last_heading = 'w'
		elif mob.last_heading == None:
			mob.last_heading = random.choice(
				['n', 's', 'e', 'w'])

		x, y = next_coords(mob.x, mob.y, mob.last_heading)
		if mob.level.is_on_grid(x, y) and mob.can_enter(x, y):
			mob.walk(mob.last_heading)
		else:
			mob.last_heading = None

if __name__ == "__main__":
	try:
		curses.wrapper(curses_main)
	except RuntimeError as e:
		print(e)