fujirun/python/pyapple2/mazegen.py
2017-07-17 11:12:55 -07:00

1349 lines
41 KiB
Python

#!/usr/bin/env python
# Basic maze: 40x24, rightmost 7 cols are the score area
#
# 00 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX_______
# 01 X/----T----T----T----T----T----\X_______
# 02 X|XXXX|XXXX|XXXX|XXXX|XXXX|XXXX|X_______
# 03 X|XXXX|XXXX|XXXX|XXXX|XXXX|XXXX|X_______
# 04 X|XXXX|XXXX|XXXX|XXXX+----+XXXX|X_______
# 05 X|XXXX|XXXX+----+XXXX|XXXX+----+X_______
# 06 X|XXXX|XXXX|XXXX+----+XXXX|XXXX|X_______
# 07 X|XXXX+----+XXXX|XXXX|XXXX|XXXX|X_______
# 08 X+----+XXXX+----+XXXX|XXXX|XXXX|X_______
# 09 X|XXXX|XXXX|XXXX|XXXX+----+XXXX|X_______
# 10 X|XXXX+----+XXXX|XXXX|XXXX+----+X_______
# 11 X|XXXX|XXXX|XXXX+----+XXXX|XXXX|X_______
# 12 X+----+XXXX|XXXX|XXXX|XXXX|XXXX|X_______
# 13 X|XXXX+----+XXXX|XXXX|XXXX|XXXX|X_______
# 14 X|XXXX|XXXX+----+XXXX+----+XXXX|X_______
# 15 X|XXXX|XXXX|XXXX|XXXX|XXXX+----+X_______
# 16 X|XXXX|XXXX|XXXX+----+XXXX|XXXX|X_______
# 17 X|XXXX+----+XXXX|XXXX|XXXX|XXXX|X_______
# 18 X+----+XXXX+----+XXXX|XXXX+----+X_______
# 19 X|XXXX|XXXX|XXXX|XXXX+----+XXXX|X_______
# 20 X|XXXX+----+XXXX+----+XXXX|XXXX|X_______
# 21 X|XXXX|XXXX|XXXX|XXXX|XXXX|XXXX|X_______
# 22 X\----^----^----^----^----^----/X_______
# 23 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX_______
#
# Terminology:
#
# vpath - vertical path
# hpath - horizontal path
# boxes - area inside path boundaries that gets filled when dots around it are collected
# enemy - uses Amidar movement
# player - joystick control
# actor - either a player or an enemy
import time
import random
import curses
import numpy as np
import logging
logging.basicConfig(level=logging.WARNING)
init_log = logging.getLogger("init")
logic_log = logging.getLogger("logic")
draw_log = logging.getLogger("draw")
maze_log = logging.getLogger("maze")
box_log = logging.getLogger("maze")
game_log = logging.getLogger("game")
collision_log = logging.getLogger("collision")
collision_log.setLevel(logging.DEBUG)
CURSES = 1
##### Game loader
pad = None
def init():
if CURSES:
curses.wrapper(init_screen)
else:
main()
def init_screen(*args, **kwargs):
global pad, curseschars
curses.use_default_colors()
curses.start_color()
curses.init_pair(1, curses.COLOR_BLUE, curses.COLOR_WHITE)
curses.curs_set(0)
pad = curses.newpad(40, 80)
pad.timeout(0)
pad.keypad(1)
# have to define these here because initscr hasn't been called when parsing
# the python source.
curseschars = [
curses.ACS_CKBOARD, # illegal
curses.ACS_CKBOARD,
curses.ACS_CKBOARD,
curses.ACS_VLINE, # 3: up/down
curses.ACS_CKBOARD,
curses.ACS_ULCORNER, # 5: down/right
curses.ACS_LLCORNER, # 6: up/right
curses.ACS_LTEE, # 7: up/down/right
curses.ACS_CKBOARD,
curses.ACS_URCORNER, # 9: left/down
curses.ACS_LRCORNER, # 10: left/up
curses.ACS_RTEE, # 11: left/up/down
curses.ACS_HLINE, # 12: left/right
curses.ACS_TTEE, # 13: left/right/down
curses.ACS_BTEE, # 14: left/right/up
curses.ACS_CKBOARD,
# And same again, with dots
curses.ACS_CKBOARD, # illegal
curses.ACS_CKBOARD,
curses.ACS_CKBOARD,
curses.ACS_VLINE, # 3: up/down
curses.ACS_CKBOARD,
curses.ACS_ULCORNER, # 5: down/right
curses.ACS_LLCORNER, # 6: up/right
curses.ACS_LTEE, # 7: up/down/right
curses.ACS_CKBOARD,
curses.ACS_URCORNER, # 9: left/down
curses.ACS_LRCORNER, # 10: left/up
curses.ACS_RTEE, # 11: left/up/down
curses.ACS_HLINE, # 12: left/right
curses.ACS_TTEE, # 13: left/right/down
curses.ACS_BTEE, # 14: left/right/up
curses.ACS_CKBOARD,
]
main()
def main():
while True:
wipe_down()
wipe_title()
if config_quit:
return
def read_menu_input():
global config_quit, config_start
read_user_input()
if config_quit:
sys.exit()
if config_start:
play_game()
config_quit = 0
config_start = 0
def clear_line(y, c):
addr = screenrow(y)
x = 0
while x < SCREEN_COLS:
addr[x] = c
x += 1
def wipe_down():
y = 0
while y < SCREEN_ROWS:
read_menu_input()
clear_line(y, 0)
show_screen()
time.sleep(.02)
y += 1
def wipe_title():
y = 0
while y < 10:
read_menu_input()
clear_line(y, 32)
show_screen()
time.sleep(.02)
y += 1
i = 0
while i < len(title_text):
read_menu_input()
x = 0
addr = screenrow(y)
while x < SCREEN_COLS:
addr[x] = ord(title_text[i][x])
x += 1
show_screen()
time.sleep(.02)
y += 1
i += 1
while y < SCREEN_ROWS:
read_menu_input()
clear_line(y, 32)
show_screen()
time.sleep(.02)
y += 1
i = TITLE_SCREEN_TIME
while i >= 0:
read_menu_input()
show_screen()
time.sleep(.02)
i -= 1
title_text = [
" LINE 1 ",
" LINE 2 ",
" LINE 3 ",
" LINE 4 ",
]
def play_game():
init_maze()
screen[:,0:33] = maze
show_screen()
zp.level = 1
zp.num_enemies = level_enemies[zp.level]
zp.num_players = 2
init_enemies()
init_players()
init_static_background()
game_loop()
##### Memory usage
# Zero page
class ZeroPage(object):
r = 0
c = 0
round_robin_index = [0, 0] # down, up
level = 0
# enemy info
num_enemies = 0
current_enemy = 0 # index of enemy being processed
# player info
num_players = 1
current_player = 0 # index of player currently being processed
# sprite
num_sprites_drawn = 0
# box drawing workspace
next_level_box = 0
box_col_save = 0
box_row_save = 0
zp = ZeroPage()
# 2 byte addresses
maze = np.empty((24, 33), dtype=np.uint8)
screen = np.empty((24, 40), dtype=np.uint8)
TILE_DOWN = 0x1
TILE_UP = 0x2
TILE_RIGHT = 0x4
TILE_LEFT= 0x8
TILE_HORZ = TILE_LEFT | TILE_RIGHT
TILE_VERT = TILE_UP | TILE_DOWN
DIR_MASK = 0x0f
TILE_DOT = 0x10
LEFT_TILE = TILE_DOT | TILE_RIGHT
MIDDLE_TILE = TILE_DOT | TILE_LEFT | TILE_RIGHT
RIGHT_TILE = TILE_DOT | TILE_LEFT
VPATH_NUM = 6
if VPATH_NUM == 7:
BOX_WIDTH = 4
else:
VPATH_NUM = 6
BOX_WIDTH = 5
VPATH_COL_SPACING = BOX_WIDTH + 1
vpath_cols = [1 + i * VPATH_COL_SPACING for i in range(VPATH_NUM)]
vpath_top_tile = [MIDDLE_TILE | TILE_DOWN] * VPATH_NUM
vpath_top_tile[0] = LEFT_TILE | TILE_DOWN
vpath_top_tile[-1] = RIGHT_TILE | TILE_DOWN
vpath_bot_tile = [MIDDLE_TILE | TILE_UP] * VPATH_NUM
vpath_bot_tile[0] = LEFT_TILE | TILE_UP
vpath_bot_tile[-1] = RIGHT_TILE | TILE_UP
if VPATH_NUM == 7:
player_start_col = [
[255, 255, 255, 255],
[vpath_cols[3], 0, 0, 0],
[vpath_cols[2], vpath_cols[4], 0, 0],
[vpath_cols[1], vpath_cols[3], vpath_cols[5], 0],
[vpath_cols[0], vpath_cols[2], vpath_cols[4], vpath_cols[6]],
]
else:
player_start_col = [
[255, 255, 255, 255],
[(vpath_cols[2] + vpath_cols[3])//2, 0, 0, 0],
[vpath_cols[1], vpath_cols[4], 0, 0],
[vpath_cols[1], vpath_cols[3], vpath_cols[5], 0],
[vpath_cols[0], vpath_cols[2], vpath_cols[4], vpath_cols[5]],
]
# up/down/left/right would be 0xf, but this is not legal for ghost legs
tilechars = [
"x", # illegal
"x",
"x",
"|", # 3: up/down
"x",
"/", # 5: down/right
"\\", # 6: up/right
"+", # 7: up/down/right
"x",
"\\", # 9: left/down
"/", # 10: left/up
"+", # 11: left/up/down
"-", # 12: left/right
"T", # 13: left/right/down
"^", # 14: left/right/up
"x",
# And same again, with dots
"x", # illegal
"x",
"x",
"|", # 3: up/down
"x",
"/", # 5: down/right
"\\", # 6: up/right
"+", # 7: up/down/right
"x",
"\\", # 9: left/down
"/", # 10: left/up
"+", # 11: left/up/down
"-", # 12: left/right
"T", # 13: left/right/down
"^", # 14: left/right/up
"x",
"@", # 32: enemy (temporary)
"$", # 33: player
]
# up/down/left/right would be 0xf, but this is not legal for ghost legs
curseschars = None
##### Constants
# Screen has rows 0 - 23
# Maze is rows 1 - 22
MAZE_TOP_ROW = 1
MAZE_BOT_ROW = 22
SCREEN_ROWS = 24
# Screen has cols 0 - 39
# cols 0 - 32 are the maze, of which 1 - 31 are actually used
# 0 and 32 are border tiles having the value zero
# cols 33 - 39 is the score area
MAZE_LEFT_COL = 1
MAZE_RIGHT_COL = 31
MAZE_SCORE_COL = 33
SCREEN_COLS = 40
# Orbiter goes around the outside border, but not through the maze
ORBITER_START_COL = MAZE_RIGHT_COL
ORBITER_START_ROW = (MAZE_TOP_ROW + MAZE_BOT_ROW) / 2
##### Utility functions
# Returns address of tile in col 0 of row y
def mazerow(y):
return maze[y]
# Returns address of tile in col 0 of row y of the "screen" memory
def screenrow(y):
return screen[y]
# Return a random number between 3 and 5 (inclusive) to represent next row that
# contains an hpath. 3 is the minimum number so that if necessary, the last
# spacing on the bottom can be adjusted upward by 1 to guarantee no cross-
# throughs
def get_rand_spacing():
return random.randint(3, 5)
# Random number between 0 and VPATH_NUM (exclusive) used for column starting
# positions
def get_rand_col():
return random.randint(0, VPATH_NUM - 1)
def get_rand_byte():
return random.randint(0, 255)
# Get random starting columns for enemies by swapping elements in a list
# several times
def get_col_randomizer():
r = list(vpath_cols)
x = 10
while x >= 0:
i1 = get_rand_col()
i2 = get_rand_col()
old1 = r[i1]
r[i1] = r[i2]
r[i2] = old1
x -= 1
return r
###### Level creation functions
def clear_maze():
y = 0
while y < SCREEN_ROWS:
addr = mazerow(y)
x = 0
while x < MAZE_SCORE_COL:
addr[x] = 0
x += 1
y += 1
init_boxes()
# Set all elements in a row to dot + left + right; only top and bottom
def setrow(row):
addr = mazerow(row)
x = MAZE_LEFT_COL
while x <= MAZE_RIGHT_COL:
addr[x] = TILE_DOT|TILE_LEFT|TILE_RIGHT
x += 1
# Create all vpaths, using top/bot character from a list to handle both
# corners and T connections.
def setvpath(col):
x = vpath_cols[col]
y = MAZE_TOP_ROW
addr = mazerow(y)
addr[x] = vpath_top_tile[col]
y += 1
while y < MAZE_BOT_ROW:
addr = mazerow(y)
addr[x] = TILE_DOT|TILE_UP|TILE_DOWN
y += 1
addr = mazerow(y)
addr[x] = vpath_bot_tile[col]
# Create hpaths such that there are no hpaths that meet at the same row in
# adjacent columns (cross-throughs are not allowed in ghost legs). Starts at
# the rightmost vpath and moves left using the rightmost vpath as the input to
# this function and building hpaths between it and the vpath to the left. The
# first time this routine is called there won't be any existing columns to
# compare to, otherwise if a tile on the left vpath has a rightward pointing
# hpath, move up one and draw the hpath there. This works because the minimum
# hpath vertical positioning leaves 2 empty rows, so moving up by one still
# leaves 1 empty row.
def sethpath(col):
x1_save = vpath_cols[col - 1]
x2 = vpath_cols[col]
y = MAZE_TOP_ROW
start_box(y, x1_save)
y += get_rand_spacing()
while y < MAZE_BOT_ROW - 1:
addr = mazerow(y)
# If not working on the rightmost column, check to see there are
# no cross-throughs.
if col < VPATH_NUM - 1:
tile = addr[x2]
if tile & TILE_RIGHT:
maze_log.debug("at y=%d on col %d, found same hpath level at col %d" % (y, col, col + 1))
y -= 1
addr = mazerow(y)
add_box(y)
x = x1_save
addr[x] = TILE_DOT | TILE_UP | TILE_DOWN | TILE_RIGHT
x += 1
while x < x2:
addr[x] = TILE_DOT | TILE_LEFT | TILE_RIGHT
x += 1
addr[x2] = TILE_DOT | TILE_UP | TILE_DOWN | TILE_LEFT
y += get_rand_spacing()
add_box(MAZE_BOT_ROW)
def init_maze():
clear_maze()
# Draw top and bottom; no intersections anywhere. Corners and T
# intesections will be placed in setvpath
setrow(MAZE_TOP_ROW)
setrow(MAZE_BOT_ROW)
# Draw all vpaths, including corners and top/bot T intersections
counter = VPATH_NUM
counter -= 1
while counter >= 0:
setvpath(counter)
counter -= 1
# Draw connectors between vpaths, starting with the rightmost column and
# the one immediately left of it. This is performed 6 times because it
# always needs a pair of columns to work with.
counter = VPATH_NUM
counter -= 1
while counter > 0: # note >, not >=
sethpath(counter)
counter -= 1
finish_boxes()
##### Box handling/painting
# Level box storage uses the left column (we don't need to store the right side
# because they are always a fixed distance away) and a list of rows.
#
# To examine the boundary of each box to check for dots, the top row and the
# bottom row must look at BOX_WIDTH + 2 tiles, all the middle rows only have to
# check the left and right tiles
#
# The entire list of rows doesn't need to be stored, either; only the top and
# bottom because everything else is a middle row. Therefore, all we need is the
# x of the left vpath, the top row and the bottom row:
#
# x1, ytop, ybot
#
# is 3 bytes. Max number of boxes is 10 per column, 6 columns that's 10 * 6 * 3
# = 180 bytes. Less than 256, yay!
#
# n can also be used as a flag: if n == 0, the box has already been checked and
# painted. n == 0xff is the flag to end processing.
# for VPATH_NUM == 7:
# 01 X/----T----T----T----T----T----\X_______
# 02 X|XXXX|XXXX|XXXX|XXXX|XXXX|XXXX|X_______
# 03 X|XXXX|XXXX|XXXX|XXXX|XXXX|XXXX|X_______
# 04 X|XXXX|XXXX|XXXX|XXXX+----+XXXX|X_______
# for VPATH_NUM == 6:
# 01 X/-----T-----T-----T-----T-----\X_______
# 02 X|XXXXX|XXXXX|XXXXX|XXXXX|XXXXX|X_______
# 03 X|XXXXX|XXXXX|XXXXX|XXXXX|XXXXX|X_______
# 04 X|XXXXX|XXXXX|XXXXX+-----+XXXXX|X_______
NUM_LEVEL_BOX_PARAMS = 3
level_boxes = [0] * 10 * 6 * NUM_LEVEL_BOX_PARAMS
# Box painting will be in hires so this array will become a tracker for the
# hires display. It will need y address, y end address, x byte number. It's
# possible for up to 3 boxes to get triggered to start painting when collecting
# a dot, and because it will take multiple frames to paint a box there may be
# even more active at one time, so for safety use 16 as possible max.
#
# player #, xbyte, ytop, ybot
NUM_BOX_PAINTING_PARAMS = 4
box_painting = [0] * NUM_BOX_PAINTING_PARAMS * 16
def init_boxes():
zp.next_level_box = 0
def start_box(r, c):
zp.box_col_save = c
zp.box_row_save = r
def add_box(r):
i = zp.next_level_box
level_boxes[i] = zp.box_col_save
level_boxes[i + 1] = zp.box_row_save
level_boxes[i + 2] = r
zp.box_row_save = r
zp.next_level_box += NUM_LEVEL_BOX_PARAMS
def finish_boxes():
i = zp.next_level_box
level_boxes[i] = 0xff
def check_boxes():
x = 0
pad.addstr(28, 0, str(level_boxes[0:21]))
while level_boxes[x] < 0xff:
c = level_boxes[x]
if c > 0:
r1 = level_boxes[x + 1]
addr = mazerow(r1)
r1 += 1
r1_save = r1
# If there's a dot anywhere, then the box isn't painted. We don't
# care where it is so we don't need to keep track of individual
# locations.
dot = addr[c] | addr[c + 1] | addr[c + 2] | addr[c + 3] | addr[c + 4] | addr[c + 5] | addr[c + BOX_WIDTH + 1]
r2 = level_boxes[x + 2]
addr = mazerow(r2)
dot |= addr[c] | addr[c + 1] | addr[c + 2] | addr[c + 3] | addr[c + 4] | addr[c + 5] | addr[c + BOX_WIDTH + 1]
while r1 < r2:
addr = mazerow(r1)
dot |= addr[c] | addr[c + BOX_WIDTH + 1]
r1 += 1
if (dot & TILE_DOT) == 0:
# No dots anywhere! Start painting
mark_box_for_painting(r1_save, r2, c + 1)
num_rows = r2 - r1_save
player_score[zp.current_player] += num_rows * 100
level_boxes[x] = 0 # Set flag so we don't check this box again
x += 3
def mark_box_for_painting(r1, r2, c):
box_log.debug("Marking box, player $%d @ %d,%d -> %d,%d" % (zp.current_player, r1, c, r2, c + BOX_WIDTH))
x = 0
while x < NUM_BOX_PAINTING_PARAMS * 16:
if box_painting[x] == 0:
box_painting[x] = c
box_painting[x + 1] = r1
box_painting[x + 2] = r2
box_painting[x + 3] = zp.current_player
break
x += NUM_BOX_PAINTING_PARAMS
pad.addstr(27, 0, "starting box, player @ %d %d,%d -> %d,%d" % (zp.current_player, r1, c, r2, c + BOX_WIDTH))
##### Gameplay storage
config_num_players = 1
config_quit = 0
config_start = 0
level = -1
level_enemies = [255, 4, 5, 6, 7, 8] # level starts counting from 1, so dummy zeroth level info
level_speeds = [255, 0, 0, 0, 0, 0] # probably needs to be 16 bit
player_score_row = [2, 7, 12, 17]
# Hardcoded, up to 8 enemies because there are max of 7 vpaths + 1 orbiter
# Enemy #0 is the orbiter
MAX_ENEMIES = 8
enemy_col = [0, 0, 0, 0, 0, 0, 0, 0] # current tile column
enemy_row = [0, 0, 0, 0, 0, 0, 0, 0] # current tile row
enemy_updown = [0, 0, 0, 0, 0, 0, 0, 0] # preferred direction
enemy_dir = [0, 0, 0, 0, 0, 0, 0, 0] # actual direction
enemy_target_col = [0, 0, 0, 0, 0, 0, 0, 0] # target column at bot or top T
round_robin_up = [0, 0, 0, 0, 0, 0, 0]
round_robin_down = [0, 0, 0, 0, 0, 0, 0]
# Hardcoded, up to 4 players
MAX_PLAYERS = 4
STARTING_LIVES = 3
BONUS_LIFE = 10000
MAX_LIVES = 8
player_col = [0, 0, 0, 0] # current tile col
player_row = [0, 0, 0, 0] # current tile row
player_input_dir = [0, 0, 0, 0] # current joystick input direction
player_dir = [0, 0, 0, 0] # current movement direction
dot_eaten_row = [255, 255, 255, 255] # dot eaten by player
dot_eaten_col = [255, 255, 255, 255]
player_score = [0, 0, 0, 0]
player_next_target_score = [0, 0, 0, 0]
player_lives = [0, 0, 0, 0] # lives remaining
player_status = [0, 0, 0, 0] # alive, exploding, dead, regenerating, ???
player_frame_counter = [0, 0, 0, 0] # frame counter for sprite changes
PLAYER_DEAD = 0
PLAYER_ALIVE = 1
PLAYER_EXPLODING = 2
PLAYER_REGENERATING = 3
GAME_OVER = 4
# Scores
DOT_SCORE = 10
PAINT_SCORE_PER_LINE = 100
##### Gameplay initialization
def init_enemies():
enemy_col[0] = ORBITER_START_COL
enemy_row[0] = ORBITER_START_ROW
enemy_dir[0] = TILE_UP
x = 1
randcol = get_col_randomizer()
while x < zp.num_enemies:
enemy_col[x] = randcol[x]
enemy_row[x] = MAZE_TOP_ROW
enemy_updown[x] = TILE_DOWN
enemy_dir[x] = TILE_DOWN
enemy_target_col[x] = 0 # Arbitrary, just need valid default
x += 1
round_robin_up[:] = get_col_randomizer()
round_robin_down[:] = get_col_randomizer()
zp.round_robin_index[:] = [0, 0]
def init_player():
addr = player_start_col[zp.num_players]
player_col[zp.current_player] = addr[zp.current_player]
player_row[zp.current_player] = MAZE_BOT_ROW
player_input_dir[zp.current_player] = 0
player_dir[zp.current_player] = 0
player_status[zp.current_player] = PLAYER_ALIVE
player_frame_counter[zp.current_player] = 0
def init_players():
zp.current_player = 0
while zp.current_player < zp.num_players:
init_player()
player_lives[zp.current_player] = STARTING_LIVES
player_next_target_score[zp.current_player] = BONUS_LIFE
zp.current_player += 1
##### Drawing routines
# Sprites use a backing store array that captures the background before each
# sprite is drawn. Each sprite's backing store is a a rectange of bytes (always
# starting on a byte boundary) stored consecutively starting with the first row
# of bytes, then the second, etc. on down the streen for the height of the
# sprite. The backing store doesn't actually care what the x value of the
# actual sprite is, only the byte number in the row of the first pixel
# affected.
# Max 16 sprites?
last_sprite_byte = [0] * 16 # byte number in row (0 - 39)
last_sprite_y = [0] * 16 # y coord of upper left corner of sprite
last_sprite_addr = [0] * 16 # Addr of sprite? Index of sprite?
sprite_backing_store = [0] * 16 # Addr of backing store? Index into array?
sprite_bytes_per_row = [0] * 16 # backing store is a rectangle of bytes
sprite_num_rows = [0] * 16 # Addr of sprite? Index of sprite?
exploding_char = ['*', '*', '@', '#', '\\', '-', '/', '|', '\\', '-', '/', '|', '\\', '-', '/', '|', '\\', '-', '/', '|']
EXPLODING_TIME = len(exploding_char) - 1
DEAD_TIME = 40
REGENERATING_TIME = 60
END_GAME_TIME = 100
TITLE_SCREEN_TIME = 100
# Erase sprites in reverse order that they're drawn to restore the background
# properly
def erase_sprites():
while zp.num_sprites_drawn > 0:
zp.num_sprites_drawn -= 1
i = zp.num_sprites_drawn
val = sprite_backing_store[i]
r = last_sprite_y[i]
addr = screenrow(r)
c = last_sprite_byte[i]
draw_log.debug("restoring background %d @ (%d,%d)" % (i, r, c))
addr[c] = val
def save_backing_store(r, c, sprite):
addr = mazerow(r)
i = zp.num_sprites_drawn
draw_log.debug("saving background %d @ (%d,%d)" % (i, r, c))
last_sprite_byte[i] = c
last_sprite_y[i] = r
last_sprite_addr[i] = sprite
sprite_backing_store[i] = addr[c]
sprite_bytes_per_row[i] = 1
sprite_num_rows[i] = 1
zp.num_sprites_drawn += 1
def draw_sprite(r, c, sprite):
if sprite is not None:
save_backing_store(r, c, sprite)
addr = screenrow(r)
addr[c] = sprite
def draw_enemies():
zp.current_enemy = 0
while zp.current_enemy < zp.num_enemies:
r = enemy_row[zp.current_enemy]
c = enemy_col[zp.current_enemy]
sprite = get_enemy_sprite()
draw_sprite(r, c, sprite)
#enemy_history[i].append((r, c))
zp.current_enemy += 1
def draw_players():
zp.current_player = 0
while zp.current_player < zp.num_players:
r = player_row[zp.current_player]
c = player_col[zp.current_player]
sprite = get_player_sprite()
draw_sprite(r, c, sprite)
#player_history[zp.current_player].append((r, c))
zp.current_player += 1
def get_enemy_sprite():
return ord("0") + zp.current_enemy
def get_player_sprite():
a = player_status[zp.current_player]
if a == PLAYER_ALIVE:
c = ord("$") + zp.current_player
elif a == PLAYER_EXPLODING:
collision_log.debug("p%d: exploding, frame=%d" % (zp.current_player, player_frame_counter[zp.current_player]))
c = ord(exploding_char[player_frame_counter[zp.current_player]])
player_frame_counter[zp.current_player] -= 1
if player_frame_counter[zp.current_player] <= 0:
player_status[zp.current_player] = PLAYER_DEAD
player_frame_counter[zp.current_player] = DEAD_TIME
elif a == PLAYER_DEAD:
collision_log.debug("p%d: dead, waiting=%d" % (zp.current_player, player_frame_counter[zp.current_player]))
c = None
player_frame_counter[zp.current_player] -= 1
if player_frame_counter[zp.current_player] <= 0:
player_lives[zp.current_player] -= 1
if player_lives[zp.current_player] > 0:
init_player()
player_status[zp.current_player] = PLAYER_REGENERATING
player_frame_counter[zp.current_player] = REGENERATING_TIME
else:
player_status[zp.current_player] = GAME_OVER
elif a == PLAYER_REGENERATING:
collision_log.debug("p%d: regenerating, frame=%d" % (zp.current_player, player_frame_counter[zp.current_player]))
if player_frame_counter[zp.current_player] & 1:
c = ord("$") + zp.current_player
else:
c = ord(" ")
player_frame_counter[zp.current_player] -= 1
if player_frame_counter[zp.current_player] <= 0:
player_status[zp.current_player] = PLAYER_ALIVE
else:
c = None
return c
##### Game logic
# Determine which of the 4 directions is allowed at the given row, col
def get_allowed_dirs(r, c):
addr = mazerow(r)
allowed = addr[c] & DIR_MASK
return allowed
# See if current tile has a dot
def has_dot(r, c):
addr = mazerow(r)
return addr[c] & TILE_DOT
# clear a dot
def clear_dot(r, c):
addr = mazerow(r)
addr[c] &= ~TILE_DOT
# Determine the tile location given the direction of the actor's movement
def get_next_tile(r, c, dir):
if dir & TILE_UP:
r -= 1
elif dir & TILE_DOWN:
r += 1
elif dir & TILE_LEFT:
c -= 1
elif dir & TILE_RIGHT:
c += 1
else:
logic_log.error("bad direction % dir")
return r, c
# Choose a target column for the next up/down direction at a bottom or top T
def get_next_round_robin(rr_table, x):
target_col = rr_table[zp.round_robin_index[x]]
logic_log.debug("target: %d, indexes=%s, table=%s" % (target_col, str(zp.round_robin_index), rr_table))
zp.round_robin_index[x] += 1
if zp.round_robin_index[x] >= VPATH_NUM:
zp.round_robin_index[x] = 0
return target_col
# Find target column when enemy reaches top or bottom
def get_target_col(c, allowed_vert):
if allowed_vert & TILE_UP:
x = 1
rr_table = round_robin_up
else:
x = 0
rr_table = round_robin_down
target_col = get_next_round_robin(rr_table, x)
if target_col == c:
# don't go back up the same column, skip to next one
target_col = get_next_round_robin(rr_table, x)
if target_col < c:
current = TILE_LEFT
else:
current = TILE_RIGHT
enemy_target_col[zp.current_enemy] = target_col
return current
# Move orbiter
def move_orbiter():
r = enemy_row[0]
c = enemy_col[0]
current = enemy_dir[0]
r, c = get_next_tile(r, c, current)
enemy_row[0] = r
enemy_col[0] = c
allowed = get_allowed_dirs(r, c)
if allowed & current:
# Can continue the current direction, so keep on doing it
logic_log.debug("orbiter: continuing %s" % str_dirs(current))
else:
# Can't continue, and because we must be at a corner, turn 90 degrees.
# So, if we are moving vertically, go horizontally, and vice versa.
if current & TILE_VERT:
current = allowed & TILE_HORZ
else:
current = allowed & TILE_VERT
enemy_dir[0] = current
# Move enemy given the enemy index
def move_enemy():
r = enemy_row[zp.current_enemy]
c = enemy_col[zp.current_enemy]
current = enemy_dir[zp.current_enemy]
r, c = get_next_tile(r, c, current)
enemy_row[zp.current_enemy] = r
enemy_col[zp.current_enemy] = c
allowed = get_allowed_dirs(r, c)
updown = enemy_updown[zp.current_enemy]
allowed_horz = allowed & TILE_HORZ
allowed_vert = allowed & TILE_VERT
if allowed_horz:
# left or right is available, we must go that way, because that's the
# Amidar(tm) way
if allowed_horz == TILE_HORZ:
# *Both* left and right are available, which means we're either in
# the middle of an box horz segment *or* at the top or bottom (but
# not at a corner)
if allowed_vert:
# At a T junction at the top or bottom. What we do depends on
# which direction we approached from
if current & TILE_VERT:
# approaching vertically means go L or R; choose direction
# based on a round robin so the enemy doesn't go back up
# the same path. Sets the target column for this enemy to
# be used when approaching the T horizontally
current = get_target_col(c, allowed_vert)
if allowed_vert & TILE_UP:
logic_log.debug("enemy %d: at bot T, new dir %x, col=%d target=%d" % (zp.current_enemy, current, c, enemy_target_col[zp.current_enemy]))
else:
logic_log.debug("enemy %d: at top T, new dir %x, col=%d target=%d" % (zp.current_enemy, current, c, enemy_target_col[zp.current_enemy]))
else:
# approaching horizontally, so check to see if this is the
# vpath to use
if enemy_target_col[zp.current_enemy] == c:
# Going vertical! Reverse desired up/down direction
updown = allowed_vert
current = allowed_vert
if allowed_vert & TILE_UP:
logic_log.debug("enemy %d: at bot T, reached target=%d, going up" % (zp.current_enemy, c))
else:
logic_log.debug("enemy %d: at top T, reached target=%d, going down" % (zp.current_enemy, c))
else:
# skip this vertical, keep on moving
if allowed_vert & TILE_UP:
logic_log.debug("enemy %d: at bot T, col=%d target=%d; skipping" % (zp.current_enemy, c, enemy_target_col[zp.current_enemy]))
else:
logic_log.debug("enemy %d: at top T, col=%d target=%d; skipping" % (zp.current_enemy, c, enemy_target_col[zp.current_enemy]))
else:
# no up or down available, so keep marching on in the same
# direction.
logic_log.debug("enemy %d: no up/down, keep moving %s" % (zp.current_enemy, str_dirs(current)))
else:
# only one horizontal dir is available
if allowed_vert == TILE_VERT:
# At a left or right T junction...
if current & TILE_VERT:
# moving vertically. Have to take the horizontal path
current = allowed_horz
logic_log.debug("enemy %d: taking hpath, start moving %s" % (zp.current_enemy, str_dirs(current)))
else:
# moving horizontally into the T, forcing a vertical turn.
# Go back to preferred up/down direction
current = updown
logic_log.debug("enemy %d: hpath end, start moving %s" % (zp.current_enemy, str_dirs(current)))
else:
# At a corner, because this tile has exactly one vertical and
# one horizontal path.
if current & TILE_VERT:
# moving vertically, and because this is a corner, the
# target column must be set up
current = get_target_col(c, allowed_vert)
if allowed_horz & TILE_LEFT:
logic_log.debug("enemy %d: at right corner col=%d, heading left to target=%d" % (zp.current_enemy, c, enemy_target_col[zp.current_enemy]))
else:
logic_log.debug("enemy %d: at left corner col=%d, heading right to target=%d" % (zp.current_enemy, c, enemy_target_col[zp.current_enemy]))
else:
# moving horizontally along the top or bottom. If we get
# here, the target column must also be this column
current = allowed_vert
updown = allowed_vert
if allowed_vert & TILE_UP:
logic_log.debug("enemy %d: at bot corner col=%d with target %d, heading up" % (zp.current_enemy, c, enemy_target_col[zp.current_enemy]))
else:
logic_log.debug("enemy %d: at top corner col=%d with target=%d, heading down" % (zp.current_enemy, c, enemy_target_col[zp.current_enemy]))
elif allowed_vert:
# left or right is not available, so we must be in the middle of a
# vpath segment. Only thing to do is keep moving
logic_log.debug("enemy %d: keep moving %x" % (zp.current_enemy, current))
else:
# only get here when moving into an illegal space
logic_log.debug("enemy %d: illegal move to %d,%d" % (zp.current_enemy, r, c))
current = 0
enemy_updown[zp.current_enemy] = updown
enemy_dir[zp.current_enemy] = current
def move_player():
r = player_row[zp.current_player]
c = player_col[zp.current_player]
allowed = get_allowed_dirs(r, c)
current = player_dir[zp.current_player]
d = player_input_dir[zp.current_player]
pad.addstr(26, 0, "r=%d c=%d allowed=%s d=%s current=%s " % (r, c, str_dirs(allowed), str_dirs(d), str_dirs(current)))
if d:
if allowed & d:
# player wants to go in an allowed direction, so go!
player_dir[zp.current_player] = d
r, c = get_next_tile(r, c, d)
player_row[zp.current_player] = r
player_col[zp.current_player] = c
else:
# player wants to go in an illegal direction. instead, continue in
# direction that was last requested
if allowed & current:
r, c = get_next_tile(r, c, current)
player_row[zp.current_player] = r
player_col[zp.current_player] = c
##### Collision detection
# Check possible collisions between the current player and any enemies
def check_collisions():
zp.current_enemy = 0
r = player_row[zp.current_player]
c = player_col[zp.current_player]
while zp.current_enemy < zp.num_enemies:
# Will provide pac-man style bug where they could pass through each
# other because it's only checking tiles
if enemy_row[zp.current_enemy] == r and enemy_col[zp.current_enemy] == c:
start_exploding()
break
zp.current_enemy += 1
def start_exploding():
player_status[zp.current_player] = PLAYER_EXPLODING
player_frame_counter[zp.current_player] = EXPLODING_TIME
##### Scoring routines
def check_dots():
r = player_row[zp.current_player]
c = player_col[zp.current_player]
if has_dot(r, c):
dot_eaten_row[zp.current_player] = r
dot_eaten_col[zp.current_player] = c
# Update maze here so we can check which player closed off a box
addr = mazerow(r)
addr[c] &= ~TILE_DOT
player_score[zp.current_player] += DOT_SCORE
def update_background():
zp.current_player = 0
while zp.current_player < zp.num_players:
if dot_eaten_col[zp.current_player] < 128:
# Here we update the screen; note the maze has already been updated
# but we don't change the background until now so sprites can
# restore their saved backgrounds first.
r = dot_eaten_row[zp.current_player]
c = dot_eaten_col[zp.current_player]
addr = screenrow(r)
addr[c] &= ~TILE_DOT
# mark as completed
dot_eaten_col[zp.current_player] = 255
update_score()
zp.current_player += 1
paint_boxes()
def paint_boxes():
x = 0
pad.addstr(28, 0, "Checking box:")
while x < NUM_BOX_PAINTING_PARAMS * 16:
pad.addstr(29, x, "%d " % x)
if box_painting[x] > 0:
c1 = box_painting[x]
r1 = box_painting[x + 1]
r2 = box_painting[x + 2]
i = box_painting[x + 3]
box_log.debug("Painting box line, player %d at %d,%d" % (i, r1, c1))
pad.addstr(30, 0, "painting box line at %d,%d" % (r1, c1))
addr = screenrow(r1)
for c in range(BOX_WIDTH):
if i == 0:
addr[c1 + c] = ord("X")
else:
addr[c1 + c] = ord(".")
r1 += 1
print "ROW", r1
box_painting[x + 1] = r1
if r1 >= r2:
box_painting[x] = 0
x += NUM_BOX_PAINTING_PARAMS
def init_static_background():
zp.current_player = 0
while zp.current_player < zp.num_players:
row = player_score_row[zp.current_player]
pad.addstr(row - 1, MAZE_SCORE_COL, " ")
pad.addstr(row, MAZE_SCORE_COL, "Player%d" % (zp.current_player + 1))
zp.current_player += 1
def show_lives(row, num):
i = 1
col = SCREEN_COLS
while col > MAZE_SCORE_COL:
col -= 1
if i < num:
c = "*"
else:
c = " "
pad.addch(row, col, ord(c))
i += 1
def update_score():
row = player_score_row[zp.current_player]
if player_status[zp.current_player] == GAME_OVER:
pad.addstr(row - 1, MAZE_SCORE_COL, "GAME ")
pad.addstr(row, MAZE_SCORE_COL, " OVER")
else:
pad.addstr(row + 1, MAZE_SCORE_COL, " %06d" % player_score[zp.current_player])
show_lives(row + 2, player_lives[zp.current_player])
##### User input routines
def read_user_input():
if CURSES:
read_curses()
def read_curses():
global config_quit, config_start
key = pad.getch()
pad.addstr(25, 0, "key = %d " % key)
if key > 0:
print "%d " % key
if key == ord('q'):
print "QUIT!!!"
config_quit = 1
elif key == 27:
print("QUIT!!!, but need to press ESC one more time for some reason")
config_quit = 1
if key == curses.KEY_UP:
player_input_dir[0] = TILE_UP
elif key == curses.KEY_DOWN:
player_input_dir[0] = TILE_DOWN
elif key == curses.KEY_LEFT:
player_input_dir[0] = TILE_LEFT
elif key == curses.KEY_RIGHT:
player_input_dir[0] = TILE_RIGHT
else:
player_input_dir[0] = 0
if key == ord(',') or key == ord('.'):
player_input_dir[1] = TILE_UP
elif key == ord('o'):
player_input_dir[1] = TILE_DOWN
elif key == ord('a'):
player_input_dir[1] = TILE_LEFT
elif key == ord('e'):
player_input_dir[1] = TILE_RIGHT
else:
player_input_dir[1] = 0
if key == ord('1'):
config_start = 1
##### Game loop
def game_loop():
global config_quit
count = 0
zp.num_sprites_drawn = 0
countdown_time = END_GAME_TIME
while True:
game_log.debug("Turn %d" % count)
read_user_input()
if config_quit:
return
game_log.debug(chr(12))
zp.current_enemy = 0
move_orbiter() # always enemy #0
zp.current_enemy += 1
while zp.current_enemy < zp.num_enemies:
move_enemy()
zp.current_enemy += 1
zp.current_player = 0
still_alive = 0
while zp.current_player < zp.num_players:
if player_status[zp.current_player] == PLAYER_REGENERATING:
# If regenerating, change to alive if the player starts to move
if player_input_dir[zp.current_player] > 0:
player_status[zp.current_player] = PLAYER_ALIVE
if player_status[zp.current_player] == PLAYER_ALIVE:
# only move and check collisions if alive
move_player()
check_collisions()
if player_status[zp.current_player] == PLAYER_ALIVE:
# only check for points if still alive
check_dots()
check_boxes()
if player_status[zp.current_player] != GAME_OVER:
still_alive += 1
zp.current_player += 1
if still_alive == 0:
countdown_time -= 1
if countdown_time <= 0:
break
erase_sprites()
update_background()
draw_enemies()
draw_players()
show_screen()
time.sleep(.02)
count += 1
# Debugging stuff below here, things that won't get converted to 6502
def str_dirs(d):
s = ""
if d & TILE_LEFT:
s += "L"
if d & TILE_RIGHT:
s += "R"
if d & TILE_UP:
s += "U"
if d & TILE_DOWN:
s += "D"
return s
def get_text_maze(m):
lines = []
for y in range(24):
line = ""
for x in range(33):
tile = m[y][x]
if tile < 32:
line += tilechars[tile]
else:
line += chr(tile)
lines.append(line)
return lines
enemy_history = [[], [], [], [], [], [], []]
player_history = [[], [], [], []]
def print_maze(append=""):
m = maze.copy()
# Loop by time history instead of by enemy number so enemy #1 doesn't
# always overwrite enemy #0's trail
remain = True
index = 0
while remain:
remain = False
for i in range(zp.num_enemies):
if index < len(enemy_history[i]):
remain = True
r, c = enemy_history[i][index]
m[r][c] = ord("0") + i
for i in range(zp.num_players):
if index < len(player_history[i]):
remain = True
r, c = player_history[i][index]
m[r][c] = ord("$") + i
index += 1
lines = get_text_maze(m)
for i in range(24):
print "%02d %s%s" % (i, lines[i], append)
def print_screen():
print_maze("_______")
def show_screen():
for r in range(24):
for c in range(33):
tile = screen[r, c]
if tile < 16:
val = curseschars[tile]
pad.addch(r, c, val, curses.color_pair(1))
elif tile < 32:
val = curseschars[tile]
pad.addch(r, c, val, curses.A_NORMAL)
else:
val = int(tile)
pad.addch(r, c, val, curses.A_NORMAL)
pad.refresh(0, 0, 0, 0, 29, 39)
lines = get_text_maze(screen)
for i in range(24):
game_log.debug("%02d %s" % (i, lines[i]))
if __name__ == "__main__":
#random.seed(31415)
init()