💾 Archived View for gemini.ctrl-c.club › ~nttp › games › glittering-light-r2.py captured on 2024-09-29 at 03:32:40.

View Raw

More Information

⬅️ Previous capture (2020-10-31)

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

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

# Glittering Light: a casual turn-based maze game for the Linux console
# 2015-04-25 Felix Pleșoianu <https://felix.plesoianu.ro/>
# Realease 2, 2020-10-12
#
# 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

C_WHITE = 0
C_FLOOR = 1
C_WALL = 2
C_WATER = 3
C_PLANT = 4
C_ITEM = 5
C_FOUNT = 6
C_ENEMY = 7

UNIT_FEAR = 3
MAX_FEAR = 10 # So that you don't die instantly when you reach 3 fear.

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',
}

main_menu = ["[N]ew game", "[C]ontinue game",
	"[H]ow to play", "[B]est scores", "[Q]uit"]

title_banner = [
"  __  _                                          _            _        ",
" /  || | o                  o                 \_|_)  o       | |       ",
" \__|| |   _|__|_  _   ,_       _  _    __,     |        __, | |   _|_ ",
"    ||/  |  |  |  |/  /  |  |  / |/ |  /  |    _|    |  /  | |/ \   |  ",
"(__/ |__/|_/|_/|_/|__/   |_/|_/  |  |_/\_/|/  (/\___/|_/\_/|/|   |_/|_/",
"                                         /|               /|           ",
"                                         \|               \|           "
]

end_banner = [
	" _                                   _                     ",
	"(_|   |                             | |                    ",
	"  |   |  __                    __,  | |   _             _  ",
	"  |   | /  \_|   |    |  |  |_/  |  |/_) |/    |   |  |/ \_",
	"   \_/|/\__/  \_/|_/   \/ \/  \_/|_/| \_/|__/   \_/|_/|__/ ",
	"     /|                                              /|    ",
	"     \|                                              \|    ",
	"                                  _                        ",
	"  o               |              | |                       ",
	"      _  _      __|   __,   ,_   | |   _  _    _   ,   ,   ",
	"  |  / |/ |    /  |  /  |  /  |  |/_) / |/ |  |/  / \_/ \_ ",
	"  |_/  |  |_/  \_/|_/\_/|_/   |_/| \_/  |  |_/|__/ \/  \/  "
]

help_text = [
	"Get as far as you can, collecting glitter for score.",
	"Being right next to nightmares increases your fear.",
	"",
	"Arrow keys or h-j-k-l to move. '.' (period) to wait.",
	"'r' and a direction to run that way until stopped.",
	"",
	"? to show instructions during the game.",
	"",
	"Press 'x' and a digit to examine nearby items.",
	"You can also click the mouse for it, where available."
]

terrain = {
	ord('.'): {"name": "a floor tile", "color": C_FLOOR},
	ord('*'): {"name": "a fountain", "color": C_FOUNT},
	ord('+'): {"name": "a wall", "color": C_WALL},
	ord('~'): {"name": "a pool of water", "color": C_WATER},
	ord('='): {"name": "brambles", "color": C_PLANT}
}

items = {
	"light": {"char": "!", "color": C_ITEM},
	"glitter": {"char": "$", "color": C_ITEM},
	"speed": {"char": "?", "color": C_ITEM}
}

item_odds = ["light", "light", "glitter", "glitter", "speed", "speed",
	None, None, None, None]
fear_odds = ["nightmare"] * 6 + [None] * 4

world = None
log = None
highscores = []

title_screen = None
game_screen = None
score_screen = None

active_screen = None

finished = False

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

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

	curses.mousemask(curses.BUTTON1_CLICKED)

	stdscr.leaveok(0)
	stdscr.idlok(1)
	stdscr.scrollok(True)
	stdscr.setscrreg(21, 23)

	if curses.has_colors():
		curses.init_pair(
			C_FLOOR, curses.COLOR_MAGENTA, curses.COLOR_BLACK)
		curses.init_pair(
			C_WALL, curses.COLOR_WHITE, curses.COLOR_YELLOW)
		curses.init_pair(
			C_WATER, curses.COLOR_CYAN, curses.COLOR_BLUE)
		curses.init_pair(
			C_PLANT, curses.COLOR_BLACK, curses.COLOR_GREEN)
		curses.init_pair(
			C_ITEM, curses.COLOR_YELLOW, curses.COLOR_BLACK)
		curses.init_pair(
			C_FOUNT, curses.COLOR_CYAN, curses.COLOR_BLACK)
		curses.init_pair(
			C_ENEMY, curses.COLOR_RED, curses.COLOR_BLACK)

	log = GameLog()

	title_screen = TitleScreen(stdscr)
	game_screen = GameScreen(stdscr)
	score_screen = ScoreScreen(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:
			stdscr.setscrreg(21, scrheight - 1)
			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, 1, 7, 18, 66)
	window.addstr(1, 8, "[How to play]", curses.A_REVERSE)
	for i in range(0, len(help_text)):
		window.addstr(3 + i, 8, help_text[i], curses.A_REVERSE)
	window.addstr(18, 57, "[Press any key]", curses.A_REVERSE)

def prompt(window, message):
	draw_dialog_bg(window, 9, 25, 4, 32)
	window.addstr(10, 26, message[0:50].center(30), curses.A_REVERSE)
	window.addstr(11, 26, " " * 30)
	curses.echo()
	answer = window.getstr(11, 26, 30)
	curses.noecho()
	return answer

def add_high_score(window, level, glitter):
	name = prompt(window, "High score! What's your name?")
	score = (name[0:30].decode(), level, glitter)
	highscores.append(score)

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 score_value(score):
	# Over long games, you pick up 2-3 glitter piles per level.
	return score[1] * 125 + score[2]

class Screen(object):
	def __init__(self, window):
		self.window = window

	def render(self):
		self.render_viewport()
		
		self.window.addstr(
			20, 0,
			self.status_text().ljust(80),
			curses.A_REVERSE)
			
		self.render_messages()

		self.window.refresh()

	def render_messages(self):
		self.window.move(21, 0)
		self.window.clrtobot()
		height, width = self.window.getmaxyx()
		num_msg = height - 21
		tail = log.messages[-num_msg:]
		for i in range(len(tail)):
			self.window.addstr(21 + i, 0, tail[i])

class TitleScreen(Screen):
	def render_viewport(self):
		for i in range(20):
			self.window.move(i, 0)
			self.window.clrtoeol()
		
		self.window.attron(curses.A_BOLD)
		
		if curses.has_colors():
			self.window.attron(curses.color_pair(C_FLOOR))
		for i, line in enumerate(title_banner):
			self.window.addstr(i, 0, line.center(80))
		
		self.window.attroff(curses.A_BOLD)
		
		if curses.has_colors():
			self.window.attroff(curses.color_pair(C_FLOOR))
		msg = "A 2020 game by"
		self.window.addstr(i + 2, 0, msg.center(80))
		if curses.has_colors():
			self.window.attron(curses.color_pair(C_FOUNT))
		msg = "www.notimetoplay.org"
		self.window.addstr(i + 4, 0, msg.center(80))
		if curses.has_colors():
			self.window.attroff(curses.color_pair(C_FOUNT))
		
		for i, line in enumerate(main_menu):
			self.window.addstr(i + 13, 33, line)
	
	def status_text(self):
		return ""

	def handle_key(self, key):
		global finished, world, active_screen
		
		if key == ord('q') or key == 27:
			finished = True
		elif key == ord('n'):
			world = GameWorld()
			world.player.log = log

			active_screen = game_screen
		elif key == ord('c'):
			if world == None:
				world = GameWorld()
				world.player.log = log

			active_screen = game_screen
		elif key == ord('h'):
			draw_help_dialog(self.window)
			self.window.getch()
		elif key == ord('b'):
			active_screen = score_screen
		else:
			curses.flash()

class ScoreScreen(Screen):
	def render_viewport(self):
		for i in range(20):
			self.window.move(i, 0)
			self.window.clrtoeol()

		self.window.addstr(1, 2, "High Score Table")
		msg = "%-30s | %5s | %5s" % ("Player", "Level", "Score")
		self.window.addstr(3, 2, msg)
		self.window.hline(4, 2, curses.ACS_HLINE, 46)
		for i, top in enumerate(highscores):
			msg = "%-30s | %5d | %5d" % (top[0], top[1], top[2])
			self.window.addstr(5 + i, 2, msg)
	
	def status_text(self):
		return "[Esc] Menu"

	def handle_key(self, key):
		global active_screen
		
		if key == ord('q') or key == 27:
			active_screen = title_screen
		else:
			curses.flash()

class GameScreen(Screen):
	def __init__(self, window):
		Screen.__init__(self, window)
		
		self.look_mode = False
		self.target_squares = []
	
	def render_viewport(self):
		level = world.palace
		lmap = level.level_map
		
		x1, y1, x2, y2 = self.update_seen()
		
		self.window.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, lmap[y][x])
				else:
					self.window.addch(y, x, " ")
		self.window.attroff(curses.A_DIM)
		
		for y in range(y1, y2 + 1):
			for x in range(x1, x2 + 1):
				self.render_tile(x, y, lmap[y][x])
		
		self.target_squares = []
		self.render_things(level, Thing)
		self.render_things(level, Creature)
		
		attr = curses.A_BOLD
		if curses.has_colors():
			attr = curses.color_pair(C_ENEMY) | attr
		if world.player.fear >= MAX_FEAR:
			for i, line in enumerate(end_banner):
				self.window.addstr(4 + i, 10, line, attr)
	
	def update_seen(self):
		p = world.player
		sight_radius = p.sight_radius()
		x1 = max(0, p.x - sight_radius)
		y1 = max(0, p.y - sight_radius)
		x2 = min(p.level.width - 1, p.x + sight_radius)
		y2 = min(p.level.height - 1, p.y + sight_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_tile(self, x, y, char):
		if curses.has_colors():
			self.window.attron(
				curses.color_pair(terrain[char]["color"]))
		self.window.addch(y, x, char)
		if curses.has_colors():
			self.window.attroff(
				curses.color_pair(terrain[char]["color"]))
	
	def render_things(self, level, thing_class):
		for i in level.content:
			# Strict check since Creature inherits from Thing.
			if type(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.look_mode:
					if target_num < 10:
						char = str(target_num)
				
			attr = curses.A_BOLD
			if curses.has_colors():
				attr = curses.color_pair(i.color) | attr
			self.window.addch(i.y, i.x, char, attr)

	def status_text(self):
		w = world
		p = w.player
		msg = "Level: %d Glitter: %d Light: %d Speed: %d Fear: %s"
		return msg % (w.level, w.glitter,
			p.light, p.speed, '&' * (p.fear // UNIT_FEAR))

	def handle_key(self, key):
		global world, active_screen
		
		if world.player.fear >= MAX_FEAR:
			log("You made it to level %d, with %d glitter." % (
				world.level, world.glitter))
			tmp = (None, world.level, world.glitter)
			if len(highscores) == 0:
				add_high_score(
					self.window,
					world.level,
					world.glitter)
			elif score_value(tmp) > score_value(highscores[-1]):
				add_high_score(
					self.window,
					world.level,
					world.glitter)
				highscores.sort(key=score_value, reverse=True)
				while len(highscores) > 10:
					del highscores[-1]
			self.look_mode = False
			active_screen = score_screen
			world = None
			return
		elif key == ord('q') or key == 27:
			self.look_mode = False
			active_screen = title_screen
			return
		elif key == ord('?'):
			self.look_mode = False
			draw_help_dialog(self.window)
			self.window.getch()
			return
		
		if world.player.action_points <= 0:
			world.update()

		if key in dirkeys:
			self.look_mode = False
			if world.player.walk(dirkeys[key]):
				world.update()
		elif key == ord('r'):
			self.look_mode = False
			world.player.log("Run which way?")
			self.render_messages()
			direction = self.window.getch()
			if direction in dirkeys:
				self.window.timeout(500)
				d = dirkeys[direction]
				while world.player.is_way_clear(d):
					world.player.walk(d)
					world.update()
					self.render()
					if self.window.getch() > -1:
						break
				self.window.timeout(-1)
			else:
				world.player.log("Canceled.")
		elif key == ord('.'):
			self.look_mode = False
			world.player.log("You stand still.")
			world.update()
		elif key == ord('x'):
			self.look_mode = not self.look_mode
		elif ord('0') <= key <= ord('9'):
			tnum = key - ord('0')
			if not self.look_mode:
				world.player.log("But you're not looking!")
			elif tnum >= len(self.target_squares):
				world.player.log("Too few things around.")
			elif self.look_mode:
				x, y = self.target_squares[tnum]
				world.player.look(x, y)
				self.look_mode = False
		elif key == curses.KEY_MOUSE:
			self.look_mode = False
			device, x, y, z, button = curses.getmouse()
			if button == curses.BUTTON1_CLICKED:
				world.player.look(x, y)
		else:
			curses.flash()

class GameLog(object):
	def __init__(self):
		self.messages = []
	
	def __call__(self, message):
		self.messages.append(message)
		while len(self.messages) > 100:
			del self.messages[0]

class GameWorld(object):
	def __init__(self):
		self.palace = Level()
		self.palace.populate()
		self.player = Creature(self.palace)
		self.player.color = C_WHITE
		#self.player.log = log
		#self.player.light = 200
		self.player.fear = 0
		self.player.base_speed = 100
		self.player.move_to(0, self.palace.entry_line)
		
		self.level = 1
		self.glitter = 0
	
	def update(self):
		p = self.player
		stuff = p.level.content
		for i in stuff.copy():
			if not isinstance(i, Creature):
				if i.x == p.x and i.y == p.y:
					p.get_item_here()
			else:
				if i == p:
					if i.light > 0:
						i.light -= i.light // 150 + 1
					if i.speed > 0:
						i.speed -= 1
				elif i.fear <= 0:
					stuff.remove(i)
					p.log("The nightmare plays itself out.")
				elif i.distance_to(p) < 2:
					i.fear -= 1
					p.fear += 1
					p.log("Your fear grows.")

				if i.action_points <= 0:
					i.action_points += (
						i.speed + i.base_speed) // 10
				elif i != p:
					self.run_ai(i)
	
	def run_ai(self, mob):
		p = self.player
		if mob.distance_to(p) == 1:
			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

class Level(object):
	width = 80
	height = 20
	
	def __init__(self):
		self.level_map = [bytearray(self.width)
			for i in range(self.height)]
		self.seen = [[False] * self.width for i in range(self.height)]
		
		self.entry_line = -1
		self.exit_line = 10
		
		self.content = set()
		
		self.generator = LevelGenerator()
	
	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 terrain.keys():
			return terrain[tile]
		else:
			return None
	
	def items_at(self, x, y):
		# Strict check since Creature inherits from Thing.
		return (i for i in self.content
			if type(i) == Thing and i.x == x and i.y == y)
	
	def creature_at(self, x, y):
		# Strict check since Creature inherits from Thing.
		for i in self.content:
			if type(i) == Creature and i.x == x and i.y == y:
				return i
		return None
	
	def populate(self):
		self.generator.populate(self)
		self.seen = [[False] * self.width for i in range(self.height)]

class Thing(object):
	def __init__(self, level):
		assert isinstance(level, Level)

		self.level = level
		self.level.content.add(self)
		self.x = -1
		self.y = -1
		
		self.name = "thing"
	
	def move_to(self, x, y, parent = None):
		if parent != None and parent != self.level:
			self.level.content.remove(self)
			parent.content.add(self)
			self.level = parent

		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 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 Creature(Thing):
	def __init__(self, level):
		Thing.__init__(self, level)
		self.log = None

		self.char = "@"
		
		self.base_speed = 0
		self.speed = 0
		self.light = 0
		self.fear = 0
		
		self.action_points = 0
		
		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])
		return tile == '.'

	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
		else:
			return self.can_enter(new_x, new_y)

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

		if not self.level.is_on_grid(new_x, new_y):
			if direction == 'w':
				self.log("There's no going back.")
				return False
			elif direction == 'e':
				world.level += 1
				self.level.populate()
				self.level.content.add(self)
				self.move_to(0, self.level.entry_line)
				self.fear -= UNIT_FEAR
				if self.fear < 0:
					self.fear = 0
				return True
			else:
				self.log("You can't stray from the path.")
				return False
		elif self.level.creature_at(new_x, new_y) != None:
			creature = self.level.creature_at(new_x, new_y)
			x, y = creature.x, creature.y
			creature.x, creature.y = self.x, self.y
			self.x, self.y = x, y
			self.action_points -= 5
			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 distance_to(self, thing):
		return abs(self.x - thing.x) + abs(self.y - thing.y)
	
	def sight_radius(self):
		return self.light // 50 + 2
	
	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: " + 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
		
		if item.name == "light":
			self.light += 75
		elif item.name == "speed":
			self.speed += 50
		elif item.name == "glitter":
			world.glitter += roll_dice(5, 20)
		
		item.level.content.remove(item)
		self.log("You find %s." % (item.name,))

class LevelGenerator(object):
	def __init__(self):
		self.rng = random.Random()
		self.cellw = 7
		self.cellh = 5
	
	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.level = level
		self.subdivideWide(1, 1, level.width - 2, level.height - 2)
		
		level.entry_line = level.exit_line
		level.level_map[level.entry_line][0] = ord('.')
		level.level_map[level.entry_line][1] = ord('.')
		level.exit_line = self.rng.randint(1, 18)
		level.level_map[level.exit_line][79] = ord('.')
		level.level_map[level.exit_line][78] = ord('.')

		self.level.content.clear()

		for y in range(1, level.height - 1):
			for x in range(1, level.width - 1):
				if self.level.level_map[y][x] != ord('.'):
					continue
				if self.rng.random() >= 0.03:
					continue
				kind = self.rng.choice(item_odds)
				if kind != None:
					item = Thing(level)
					item.name = kind
					item.char = items[kind]["char"]
					item.color = items[kind]["color"]
					item.move_to(x, y)

		for y in range(1, level.height - 1):
			for x in range(1, level.width - 1):
				if self.level.level_map[y][x] != ord('.'):
					continue
				if self.rng.random() >= 0.03:
					continue
				if self.rng.choice(fear_odds) != None:
					creature = Creature(level)
					creature.log = log
					creature.name = "nightmare"
					creature.char = "&"
					creature.color = C_ENEMY
					creature.fear = UNIT_FEAR
					creature.base_speed = 150
					creature.move_to(x, y)
	
	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('+')

		wall1 =	self.subdivideHigh(x1, y1, x - 1, y2)
		wall2 = self.subdivideHigh(x + 1, y1, x2, y2)
		
		if h >= 15:
			doors = [y1 + h // 3, y2 - h // 3]
		else:
			doors = [y1 + h // 2]

		for y in doors:
			self.level.level_map[y][x] = ord('.')
			# Account for walls placed deeper into recursion.
			self.level.level_map[y][x - 1] = ord('.')
			self.level.level_map[y][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('+')

		wall1 = self.subdivideWide(x1, y1, x2, y - 1)
		wall2 = self.subdivideWide(x1, y + 1, x2, y2)

		if w >= 45:
			doors = [x1 + w // 4, x1 + w // 2, x2 - w // 4]
		elif w >= 15:
			doors = [x1 + w // 3, x2 - w // 3]
		else:
			doors = [x1 + w // 2]

		for x in doors:
			self.level.level_map[y][x] = ord('.')
			# Account for walls placed deeper into recursion.
			self.level.level_map[y - 1][x] = ord('.')
			self.level.level_map[y + 1][x] = 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 and h > 2:
			for y in range(y1, y2 + 1):
				if (y + x1 + 1) % 2 == 0:
					lmap[y][x1 + 1] = ord('+')
			if y == y2 and (y + x1 + 1) % 2 == 0:
				lmap[y][x1 + 1] = ord('*')
		elif h == 3 and w > 2:
			for x in range(x1, x2 + 1):
				if (y1 + 1 + x) % 2 == 0:
					lmap[y1 + 1][x] = ord('+')
			if x == x2 and (y1 + 1 + x) % 2 == 0:
				lmap[y1 + 1][x] = ord('*')
		elif w == 4:
			if self.rng.random() < 0.5:
				char = ord('~')
			else:
				char = ord('=')
			for y in range(y1 + 1, y2):
				lmap[y][x1 + 1] = char
				lmap[y][x2 - 1] = char
		elif h == 4:
			if self.rng.random() < 0.5:
				char = ord('~')
			else:
				char = ord('=')
			for x in range(x1 + 1, x2):
				lmap[y1 + 1][x] = char
				lmap[y2 - 1][x] = char
		elif w == 5 and h == 5:
			lmap[y1 + 1][x1 + 1] = ord('*')
			lmap[y1 + 3][x1 + 3] = ord('*')
			lmap[y1 + 1][x1 + 3] = ord('*')
			lmap[y1 + 3][x1 + 1] = ord('*')
		elif w == 5 and h > 2:
			for y in range(y1, y2 + 1):
				if (y + x1 + 1) % 2 == 0:
					lmap[y][x1 + 1] = ord('+')
					lmap[y][x2 - 1] = ord('+')
			if y == y2 and (y + x1 + 1) % 2 == 0:
				lmap[y][x1 + 1] = ord('*')
				lmap[y][x2 - 1] = ord('*')
		elif h == 5 and w > 2:
			for x in range(x1, x2 + 1):
				if (y1 + 1 + x) % 2 == 0:
					lmap[y1 + 1][x] = ord('+')
					lmap[y2 - 1][x] = ord('+')
			if x == x2 and (y1 + 1 + x) % 2 == 0:
				lmap[y1 + 1][x] = ord('*')
				lmap[y2 - 1][x] = ord('*')
		elif w == 7 and h == 7:
			for y in range(y1 + 1, y1 + 3):
				for x in range(x1 + 1, x1 + 3):
					lmap[y][x] = ord('=')
			for y in range(y1 + 1, y1 + 3):
				for x in range(x1 + 4, x1 + 6):
					lmap[y][x] = ord('~')
			for y in range(y1 + 4, y1 + 6):
				for x in range(x1 + 4, x1 + 6):
					lmap[y][x] = ord('=')
			for y in range(y1 + 4, y1 + 6):
				for x in range(x1 + 1, x1 + 3):
					lmap[y][x] = 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)

if __name__ == "__main__":
	curses.wrapper(curses_main)