Fixed amidar movement down to the bottom

This commit is contained in:
Rob McMullen 2017-04-25 11:58:14 -07:00
parent bd5bed52a6
commit 6e4760fc8a

View File

@ -40,6 +40,11 @@ import random
import numpy as np import numpy as np
# Zero page
r = 0
c = 0
round_robin_index = [0, 0] # down, up
maze = np.empty((24, 33), dtype=np.uint8) maze = np.empty((24, 33), dtype=np.uint8)
level = -1 level = -1
@ -57,40 +62,40 @@ tiledot = 0x10
# up/down/left/right would be 0xf, but this is not legal for ghost legs # up/down/left/right would be 0xf, but this is not legal for ghost legs
tilechars = [ tilechars = [
"X", # illegal "x", # illegal
"X", "x",
"X", "x",
"|", # 3: up/down "|", # 3: up/down
"X", "x",
"/", # 5: down/right "/", # 5: down/right
"\\", # 6: up/right "\\", # 6: up/right
"+", # 7: up/down/right "+", # 7: up/down/right
"X", "x",
"\\", # 9: left/down "\\", # 9: left/down
"/", # 10: left/up "/", # 10: left/up
"+", # 11: left/up/down "+", # 11: left/up/down
"-", # 12: left/right "-", # 12: left/right
"T", # 13: left/right/down "T", # 13: left/right/down
"^", # 14: left/right/up "^", # 14: left/right/up
"X", "x",
# And same again, with dots # And same again, with dots
"X", # illegal "x", # illegal
"X", "x",
"X", "x",
"|", # 3: up/down "|", # 3: up/down
"X", "x",
"/", # 5: down/right "/", # 5: down/right
"\\", # 6: up/right "\\", # 6: up/right
"+", # 7: up/down/right "+", # 7: up/down/right
"X", "x",
"\\", # 9: left/down "\\", # 9: left/down
"/", # 10: left/up "/", # 10: left/up
"+", # 11: left/up/down "+", # 11: left/up/down
"-", # 12: left/right "-", # 12: left/right
"T", # 13: left/right/down "T", # 13: left/right/down
"^", # 14: left/right/up "^", # 14: left/right/up
"X", "x",
"@", # 32: enemy (temporary) "@", # 32: enemy (temporary)
"$", # 33: player "$", # 33: player
@ -224,34 +229,60 @@ def init_maze():
counter -= 1 counter -= 1
def get_text_maze(): def get_text_maze(m):
lines = [] lines = []
for y in range(24): for y in range(24):
line = "" line = ""
for x in range(33): for x in range(33):
tile = maze[y][x] tile = m[y][x]
if tile < 32:
line += tilechars[tile] line += tilechars[tile]
else:
line += chr(tile)
lines.append(line) lines.append(line)
return lines return lines
def print_maze(): enemy_history = [[], [], [], [], [], [], []]
lines = get_text_maze() 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(cur_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(cur_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): for i in range(24):
print "%02d %s" % (i, lines[i]) print "%02d %s%s" % (i, lines[i], append)
def print_screen(): def print_screen():
lines = get_text_maze() print_maze("_______")
for i in range(24):
print "%02d %s_______" % (i, lines[i])
# Hardcoded, up to 7 enemies because there are max of 7 vpaths # Hardcoded, up to 8 enemies because there are max of 7 vpaths + 1 orbiter
max_enemies = 7 max_enemies = 8
cur_enemies = -1 cur_enemies = -1
enemy_col = [0, 0, 0, 0, 0, 0, 0] # current tile column enemy_col = [0, 0, 0, 0, 0, 0, 0, 0] # current tile column
enemy_row = [0, 0, 0, 0, 0, 0, 0] # current tile row enemy_row = [0, 0, 0, 0, 0, 0, 0, 0] # current tile row
enemy_updown = [0, 0, 0, 0, 0, 0, 0] # preferred direction enemy_updown = [0, 0, 0, 0, 0, 0, 0, 0] # preferred direction
enemy_dir = [0, 0, 0, 0, 0, 0, 0] # actual direction enemy_dir = [0, 0, 0, 0, 0, 0, 0, 0] # actual direction
enemy_last_horz = [0, 0, 0, 0, 0, 0, 0] # last horizontal 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 # Hardcoded, up to 4 players
max_players = 4 max_players = 4
@ -273,10 +304,13 @@ level_start_col = [
def get_rand7(): def get_rand7():
return random.randint(0, 6) return random.randint(0, 6)
def get_rand_byte():
return random.randint(0, 255)
# Get random starting columns for enemies by swapping elements in a list # Get random starting columns for enemies by swapping elements in a list
# several times # several times
def get_col_randomizer(): def get_col_randomizer():
r = [0, 1, 2, 3, 4, 5, 6] r = list(vpath_cols)
x = 10 x = 10
while x >= 0: while x >= 0:
i1 = get_rand7() i1 = get_rand7()
@ -291,19 +325,22 @@ def init_enemies():
x = 0 x = 0
randcol = get_col_randomizer() randcol = get_col_randomizer()
while x < cur_enemies: while x < cur_enemies:
enemy_col[x] = vpath_cols[randcol[x]] enemy_col[x] = randcol[x]
enemy_row[x] = mazetoprow enemy_row[x] = mazetoprow
enemy_updown[x] = tiledown enemy_updown[x] = tiledown
enemy_dir[x] = tiledown enemy_dir[x] = tiledown
enemy_target_col[x] = 0 # Arbitrary, just need valid default
x += 1 x += 1
round_robin_up[:] = get_col_randomizer()
round_robin_down[:] = get_col_randomizer()
round_robin_index[:] = [0, 0]
def draw_enemies(): def draw_enemies():
i = 0 i = 0
while i < cur_enemies: while i < cur_enemies:
y = enemy_row[i] r = enemy_row[i]
addr = getrow(y) c = enemy_col[i]
x = enemy_col[i] enemy_history[i].append((r, c))
addr[x] = 32
i += 1 i += 1
def get_col_start(): def get_col_start():
@ -323,10 +360,9 @@ def init_players():
def draw_players(): def draw_players():
i = 0 i = 0
while i < cur_players: while i < cur_players:
y = player_row[i] r = player_row[i]
addr = getrow(y) c = player_col[i]
x = player_col[i] player_history[i].append((r, c))
addr[x] = 33
i += 1 i += 1
# Determine which of the 4 directions is allowed at the given row, col # Determine which of the 4 directions is allowed at the given row, col
@ -349,6 +385,36 @@ def get_next_tile(r, c, dir):
print("bad direction % dir") print("bad direction % dir")
return r, c 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[round_robin_index[x]]
print "target: %d, indexes=%s, table=%s" % (target_col, str(round_robin_index), rr_table)
round_robin_index[x] += 1
if round_robin_index[x] >= vpath_num:
round_robin_index[x] = 0
return target_col
# Find target column when enemy reaches top or bottom
def get_target_col(i, c, allowed_vert):
if allowed_vert & tileup:
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 = tileleft
else:
current = tileright
enemy_target_col[i] = target_col
return current
# Move enemy given the enemy index # Move enemy given the enemy index
def move_enemy(i): def move_enemy(i):
r = enemy_row[i] r = enemy_row[i]
@ -372,16 +438,41 @@ def move_enemy(i):
# not at a corner) # not at a corner)
if allowed_vert: if allowed_vert:
# at a T junction at the top or bottom. choose L or R based on # At a T junction at the top or bottom. What we do depends on
# last left or right # which direction we approached from
current = enemy_last_horz[i]
if current & tilevert:
# 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(i, c, allowed_vert)
# and reverse desired up/down direction
updown = allowed_vert
if allowed_vert & tileup: if allowed_vert & tileup:
print("enemy %d: at bot T, new dir %x" % (i, current)) print("enemy %d: at bot T, new dir %x, col=%d target=%d" % (i, current, c, enemy_target_col[i]))
else: else:
print("enemy %d: at top T, new dir %x" % (i, current)) print("enemy %d: at top T, new dir %x, col=%d target=%d" % (i, current, c, enemy_target_col[i]))
else:
# approaching horizontally, so check to see if this is the
# vpath to use
if enemy_target_col[i] == c:
# Going vertical! Reverse desired up/down direction
updown = allowed_vert
current = allowed_vert
if allowed_vert & tileup:
print("enemy %d: at bot T, reached target=%d, going up" % (i, c))
else:
print("enemy %d: at top T, reached target=%d, going down" % (i, c))
else:
# skip this vertical, keep on moving
if allowed_vert & tileup:
print("enemy %d: at bot T, col=%d target=%d; skipping" % (i, c, enemy_target_col[i]))
else:
print("enemy %d: at top T, col=%d target=%d; skipping" % (i, c, enemy_target_col[i]))
else: else:
# no up or down available, so keep marching on in the same # no up or down available, so keep marching on in the same
# direction. # direction.
@ -390,35 +481,51 @@ def move_enemy(i):
else: else:
# only one horizontal dir is available # only one horizontal dir is available
if allowed_vert & updown: if allowed_vert == tilevert:
# At a left or right T junction...
if current & tilevert: if current & tilevert:
# Moving vertially but we must take the horizontal # moving vertically. Have to take the horizontal path
# direction. Only a single direction is possible otherwise
# it would have been caught by the allowed_horz case above.
current = allowed_horz current = allowed_horz
print("enemy %d: taking hpath, start moving %x" % (i, current)) print("enemy %d: taking hpath, start moving %x" % (i, current))
else: else:
# Moving horizontally but that direction isn't available meaning we are at the end of an hpath. Start moving vertically again. # moving horizontally into the T, forcing a vertical turn.
# Go back to preferred up/down direction
current = updown current = updown
print("enemy %d: hpath end, start moving %x" % (i, current)) print("enemy %d: hpath end, start moving %x" % (i, current))
else: else:
# we must be at a corner, so we go the only available horz # At a corner, because this tile has exactly one vertical and
# direction # one horizontal path.
current = allowed_horz
# and reverse desired up/down direction if current & tilevert:
# moving vertically, and because this is a corner, the
# target column must be set up
current = get_target_col(i, c, allowed_vert)
if allowed_horz & tileleft:
print("enemy %d: at right corner col=%d, heading left to target=%d" % (i, c, enemy_target_col[i]))
else:
print("enemy %d: at left corner col=%d, heading right to target=%d" % (i, c, enemy_target_col[i]))
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 updown = allowed_vert
if allowed_vert & tileup: if allowed_vert & tileup:
print("enemy %d: at bot corner, new dir %x" % (i, current)) print("enemy %d: at bot corner col=%d with target %d, heading up" % (i, c, enemy_target_col[i]))
else: else:
print("enemy %d: at top corner, new dir %x" % (i, current)) print("enemy %d: at top corner col=%d with target=%d, heading down" % (i, c, enemy_target_col[i]))
else: elif allowed_vert:
# left or right is not available, so we must be in the middle of a # left or right is not available, so we must be in the middle of a
# vpath segment. Only thing to do is keep moving # vpath segment. Only thing to do is keep moving
print("enemy %d: keep moving %x" % (i, current)) print("enemy %d: keep moving %x" % (i, current))
else:
# only get here when moving into an illegal space
print("enemy %d: illegal move to %d,%d" % (i, r, c))
current = 0
enemy_updown[i] = updown enemy_updown[i] = updown
enemy_dir[i] = current enemy_dir[i] = current
@ -427,7 +534,7 @@ def move_player(i):
def game_loop(): def game_loop():
count = 0 count = 0
while count < 20: while count < 100:
print("Turn %d" % count) print("Turn %d" % count)
draw_enemies() draw_enemies()
draw_players() draw_players()