life.py 7.18 KB
Newer Older
1 2
#!/usr/bin/env python
# life.py -- A curses-based version of Conway's Game of Life.
3
# Contributed by AMK
4 5 6 7 8 9 10 11 12 13
#
# An empty board will be displayed, and the following commands are available:
#  E : Erase the board
#  R : Fill the board randomly
#  S : Step for a single generation
#  C : Update continuously until a key is struck
#  Q : Quit
#  Cursor keys :  Move the cursor around the board
#  Space or Enter : Toggle the contents of the cursor's position
#
14
# TODO :
15 16 17 18 19
#   Support the mouse
#   Use colour if available
#   Make board updates faster
#

20 21 22
import random, string, traceback
import curses

23 24 25 26 27 28
class LifeBoard:
    """Encapsulates a Life board

    Attributes:
    X,Y : horizontal and vertical size of the board
    state : dictionary mapping (x,y) to 0 or 1
29

30
    Methods:
31
    display(update_board) -- If update_board is true, compute the
32
                             next generation.  Then display the state
33
                             of the board and refresh the screen.
34 35 36 37 38 39 40 41
    erase() -- clear the entire board
    makeRandom() -- fill the board randomly
    set(y,x) -- set the given cell to Live; doesn't refresh the screen
    toggle(y,x) -- change the given cell from live to dead, or vice
                   versa, and refresh the screen display

    """
    def __init__(self, scr, char=ord('*')):
42 43 44 45 46
        """Create a new LifeBoard instance.

        scr -- curses screen object to use for display
        char -- character used to render live cells (default: '*')
        """
47 48
        self.state = {}
        self.scr = scr
49 50 51 52 53 54
        Y, X = self.scr.getmaxyx()
        self.X, self.Y = X-2, Y-2-1
        self.char = char
        self.scr.clear()

        # Draw a border around the board
55
        border_line = '+'+(self.X*'-')+'+'
56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
        self.scr.addstr(0, 0, border_line)
        self.scr.addstr(self.Y+1,0, border_line)
        for y in range(0, self.Y):
            self.scr.addstr(1+y, 0, '|')
            self.scr.addstr(1+y, self.X+1, '|')
        self.scr.refresh()

    def set(self, y, x):
        """Set a cell to the live state"""
        if x<0 or self.X<=x or y<0 or self.Y<=y:
            raise ValueError, "Coordinates out of range %i,%i"% (y,x)
        self.state[x,y] = 1

    def toggle(self, y, x):
        """Toggle a cell's state between live and dead"""
        if x<0 or self.X<=x or y<0 or self.Y<=y:
            raise ValueError, "Coordinates out of range %i,%i"% (y,x)
        if self.state.has_key( (x,y) ):
            del self.state[x,y]
            self.scr.addch(y+1, x+1, ' ')
        else:
77
            self.state[x,y] = 1
78 79
            self.scr.addch(y+1, x+1, self.char)
        self.scr.refresh()
80 81

    def erase(self):
82
        """Clear the entire board and update the board display"""
83 84
        self.state = {}
        self.display(update_board=False)
85

86
    def display(self, update_board=True):
87 88 89 90 91 92 93 94 95 96 97 98
        """Display the whole board, optionally computing one generation"""
        M,N = self.X, self.Y
        if not update_board:
            for i in range(0, M):
                for j in range(0, N):
                    if self.state.has_key( (i,j) ):
                        self.scr.addch(j+1, i+1, self.char)
                    else:
                        self.scr.addch(j+1, i+1, ' ')
            self.scr.refresh()
            return

99 100
        d = {}
        self.boring = 1
101
        for i in range(0, M):
102
            L = range( max(0, i-1), min(M, i+2) )
103
            for j in range(0, N):
104 105
                s = 0
                live = self.state.has_key( (i,j) )
106 107 108
                for k in range( max(0, j-1), min(N, j+2) ):
                    for l in L:
                        if self.state.has_key( (l,k) ):
109 110 111
                            s += 1
                s -= live
                if s == 3:
112
                    # Birth
113
                    d[i,j] = 1
114
                    self.scr.addch(j+1, i+1, self.char)
115 116
                    if not live: self.boring = 0
                elif s == 2 and live: d[i,j] = 1       # Survival
117 118 119
                elif live:
                    # Death
                    self.scr.addch(j+1, i+1, ' ')
120 121
                    self.boring = 0
        self.state = d
122
        self.scr.refresh()
123 124

    def makeRandom(self):
125
        "Fill the board with a random pattern"
126
        self.state = {}
127
        for i in range(0, self.X):
128
            for j in range(0, self.Y):
129 130
                if random.random() > 0.5:
                    self.set(j,i)
131 132 133 134


def erase_menu(stdscr, menu_y):
    "Clear the space where the menu resides"
135 136 137 138
    stdscr.move(menu_y, 0)
    stdscr.clrtoeol()
    stdscr.move(menu_y+1, 0)
    stdscr.clrtoeol()
139 140 141 142 143 144 145 146 147

def display_menu(stdscr, menu_y):
    "Display the menu of possible keystroke commands"
    erase_menu(stdscr, menu_y)
    stdscr.addstr(menu_y, 4,
                  'Use the cursor keys to move, and space or Enter to toggle a cell.')
    stdscr.addstr(menu_y+1, 4,
                  'E)rase the board, R)andom fill, S)tep once or C)ontinuously, Q)uit')

148
def keyloop(stdscr):
149 150 151
    # Clear the screen and display the menu of keys
    stdscr.clear()
    stdscr_y, stdscr_x = stdscr.getmaxyx()
152
    menu_y = (stdscr_y-3)-1
153 154 155
    display_menu(stdscr, menu_y)

    # Allocate a subwindow for the Life board and create the board object
156 157 158
    subwin = stdscr.subwin(stdscr_y-3, stdscr_x, 0, 0)
    board = LifeBoard(subwin, char=ord('*'))
    board.display(update_board=False)
159 160

    # xpos, ypos are the cursor's position
161
    xpos, ypos = board.X//2, board.Y//2
162 163 164

    # Main loop:
    while (1):
165
        stdscr.move(1+ypos, 1+xpos)     # Move the cursor
166
        c = stdscr.getch()                # Get a keystroke
167
        if 0<c<256:
168
            c = chr(c)
169 170 171 172 173 174 175 176 177 178 179
            if c in ' \n':
                board.toggle(ypos, xpos)
            elif c in 'Cc':
                erase_menu(stdscr, menu_y)
                stdscr.addstr(menu_y, 6, ' Hit any key to stop continuously '
                              'updating the screen.')
                stdscr.refresh()
                # Activate nodelay mode; getch() will return -1
                # if no keystroke is available, instead of waiting.
                stdscr.nodelay(1)
                while (1):
180 181 182 183 184
                    c = stdscr.getch()
                    if c != -1:
                        break
                    stdscr.addstr(0,0, '/')
                    stdscr.refresh()
185
                    board.display()
186 187
                    stdscr.addstr(0,0, '+')
                    stdscr.refresh()
188 189 190 191

                stdscr.nodelay(0)       # Disable nodelay mode
                display_menu(stdscr, menu_y)

192 193 194 195
            elif c in 'Ee':
                board.erase()
            elif c in 'Qq':
                break
196 197
            elif c in 'Rr':
                board.makeRandom()
198
                board.display(update_board=False)
199 200 201
            elif c in 'Ss':
                board.display()
            else: pass                  # Ignore incorrect keys
202 203 204 205 206 207 208 209 210 211 212 213 214 215 216
        elif c == curses.KEY_UP and ypos>0:            ypos -= 1
        elif c == curses.KEY_DOWN and ypos<board.Y-1:  ypos += 1
        elif c == curses.KEY_LEFT and xpos>0:          xpos -= 1
        elif c == curses.KEY_RIGHT and xpos<board.X-1: xpos += 1
        else:
            # Ignore incorrect keys
            pass


def main(stdscr):
    keyloop(stdscr)                    # Enter the main loop


if __name__ == '__main__':
    curses.wrapper(main)