-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgame-of-life.lsp
72 lines (65 loc) · 3.08 KB
/
game-of-life.lsp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
;;;;Conway's Game of Life in Lisp
;;;;
;;;;Sample patterns:
;;;; Blinker: (game-start '((0 1) (1 1) (2 1)) 3 10)
;;;; Beacon: (game-start '((1 1) (2 1) (1 2) (4 3) (3 4) (4 4)) 6 10)
;;;; Glider: (game-start '((1 0) (2 1) (0 2) (1 2) (2 2)) 20 50)
;;;; Combination: (game-start '((0 12) (1 12) (2 12) (1 6) (2 7) (0 8) (1 8) (2 8)) 20 100)
;;;; R-Pentomino: (game-start '((7 11) (6 12) (7 12) (8 12) (6 13)) 30 100)
;;;; Gosper Glider Gun: (game-start '((5 1) (5 2) (6 1) (6 2) (5 11) (6 11) (7 11) (4 12) (3 13) (3 14) (8 12) (9 13) (9 14) (6 15) (4 16) (5 17) (6 17) (7 17) (6 18) (8 16) (3 21) (4 21) (5 21) (3 22) (4 22) (5 22) (2 23) (6 23) (1 25) (2 25) (6 25) (7 25) (3 35) (4 35) (3 36) (4 36)) 40 100)
;;live-list is a list of coordinates for live cells. For example: '((0 1) (1 1) (2 1))
;;size-sq is the size of the square grid. For example, 3 to draw a 3x3 grid.
;;num-gen is the number of generations to simulate.
(defun game-start (live-list size-sq num-gen)
(setf grid (get-grid-values size-sq))
(game-recurse live-list grid size-sq num-gen))
(defun game-recurse (live-list grid size-sq num-gen)
(if (zerop num-gen)
next-gen
(progn
(setf next-gen '())
(dolist (grid-cell grid next-gen)
(let ((neighbor-list (get-neighbors (car grid-cell) (car (cdr grid-cell)))))
(setf num-alive-neighbors (get-num-alive live-list neighbor-list))
(setf currently-alive (member grid-cell live-list :test #'equal))
(if (will-live num-alive-neighbors currently-alive )
(setf next-gen (append next-gen (list grid-cell))))))
(draw next-gen grid size-sq)
(sleep 0.25)
(game-recurse next-gen grid size-sq (- num-gen 1)))))
;;Create a two-dimensional list of coordinates based on the grid size.
(defun get-grid-values (size-sq)
(setf cells '())
(do ((i 0 (+ i 1)))
((= i size-sq) cells)
(do ((j 0 (+ j 1)))
((= j size-sq) cells)
(setf cells (append cells (list (list i j))))))
cells)
;;Get the list of neighbors for a cell with coordinates (x,y).
(defun get-neighbors (x y)
(list
(list (- x 1) (- y 1)) (list (- x 1) y) (list (- x 1) (+ y 1))
(list x (- y 1)) (list x (+ y 1))
(list (+ x 1) (- y 1)) (list (+ x 1) y) (list (+ x 1) (+ y 1))))
;;Get the number of neighboring cells that are alive.
(defun get-num-alive (live-list neighbor-list)
(setf num-alive 0)
(dolist (neighbor neighbor-list num-alive)
(if (member neighbor live-list :test #'equal)
(progn
(setf num-alive (+ 1 num-alive))))))
;;Determine whether a given cell will be alive in the next generation.
(defun will-live (num-alive currently-alive)
(cond ((< num-alive 2) nil)
((and (= num-alive 2) currently-alive) t)
((= num-alive 3) t)
((> num-alive 3) nil)))
;;Print on the command line. Keep track of the x coordinate and enter new lines.
(defun draw (live-cells grid size-sq)
(dolist (grid-cell grid)
(if (member grid-cell live-cells :test #'equal)
(format t "X")
(format t "-"))
(if (zerop (mod (+ 1 (car (cdr grid-cell))) size-sq))
(format t "~%"))))