Add server submodule
This commit is contained in:
parent
7e25579b6a
commit
897fbd8878
2 changed files with 136 additions and 0 deletions
102
server/game/game.lisp
Normal file
102
server/game/game.lisp
Normal file
|
@ -0,0 +1,102 @@
|
|||
(in-package :remote-snake-server-game)
|
||||
|
||||
(defclass game ()
|
||||
((initial-size
|
||||
:initarg :initial-size
|
||||
:initform 3)
|
||||
(initial-position
|
||||
:initarg :initial-position
|
||||
:initform (list 0 0))
|
||||
(snake
|
||||
:accessor snake)
|
||||
(dir
|
||||
:initarg :initial-direction
|
||||
:initform :right)
|
||||
(grid-size
|
||||
:initarg :grid-size
|
||||
:initform (list 30 30))
|
||||
(food
|
||||
:initform (make-list 0))
|
||||
(game-over
|
||||
:initform nil
|
||||
:accessor game-over)))
|
||||
|
||||
;;; Class constructor to initialize the snake
|
||||
(defmethod initialize-instance :after ((g game) &key)
|
||||
(with-slots (snake initial-size initial-position) g
|
||||
(setf snake (make-list initial-size :initial-element initial-position))))
|
||||
|
||||
;;; Pretty-print a game
|
||||
(defmethod print-object ((g game) stream)
|
||||
(with-slots (snake dir food) g
|
||||
(format t "Snake: ")
|
||||
(dotimes (i (length snake))
|
||||
(let ((elem (nth i snake)))
|
||||
(unless (eql i 0)
|
||||
(format t "<="))
|
||||
(format t "(~a,~a)" (first elem) (second elem))))
|
||||
(format t "~%Size: ~a" (length snake))
|
||||
(format t "~%Direction: ~a" dir)
|
||||
(format t "~%Food: ~a" food)))
|
||||
|
||||
;;; Note that there is no waranty that nb food are added (ex: if food position collide with snake position)
|
||||
(defgeneric add-food (g nb)
|
||||
(:documentation "Add food on the game grid."))
|
||||
|
||||
(defgeneric refresh (g &key)
|
||||
(:documentation "Refresh the game g (move forward or change direction)."))
|
||||
|
||||
(defmethod refresh ((g game) &key (dir nil dir-supplied-p))
|
||||
;; First, update direction
|
||||
(with-slots ((active-dir dir)) g
|
||||
(when dir-supplied-p
|
||||
(if (and
|
||||
(or (eq dir :up) (eq dir :down) (eq dir :left) (eq dir :right))
|
||||
(legal-move active-dir dir))
|
||||
(setf (slot-value g 'dir) dir)
|
||||
(error "Invalid direction supplied"))))
|
||||
;; Then, move the snake
|
||||
(with-slots (snake dir) g
|
||||
(let ((last-old-x nil) (last-old-y nil))
|
||||
(dotimes (i (length snake))
|
||||
(let ((elem (nth i snake)))
|
||||
(let ((x (first elem)) (y (second elem)))
|
||||
;; Move snake
|
||||
(if (eql i 0) ; Move head
|
||||
(progn
|
||||
;; Update last-old-x and last-old-y (to move the body when i>0)
|
||||
(setf last-old-x x)
|
||||
(setf last-old-y y)
|
||||
(cond ((eq dir :up) (incf y))
|
||||
((eq dir :down) (decf y))
|
||||
((eq dir :left) (decf x))
|
||||
((eq dir :right) (incf x))))
|
||||
(progn ; Move body
|
||||
(rotatef x last-old-x)
|
||||
(rotatef y last-old-y)))
|
||||
(setf (nth i snake) (list x y))))))) ; Apply new snake location (update snake slot)
|
||||
;; Check if snake eat something
|
||||
(with-slots (snake food) g
|
||||
(unless (eql (member (nth 0 snake) food :test #'equal-coord) nil)
|
||||
(setf food (remove-coord food (nth 0 snake))) ; Remove eated food
|
||||
(setf snake (grow-snake snake 1)))) ; Grow snake of 1 unit
|
||||
;; Update food
|
||||
(with-slots (food) g
|
||||
(when (< (length food) 1) ; For the moment, allow only one food on the map
|
||||
(add-food g 1)))
|
||||
;; Check if we loose
|
||||
(with-slots (snake game-over) g
|
||||
(when (member (nth 0 snake) (cdr snake) :test #'equal-coord) ; If head is member of body :D
|
||||
(setf game-over t))))
|
||||
|
||||
(defmethod add-food ((g game) nb)
|
||||
(with-slots (snake grid-size food) g
|
||||
(let ((size-x (first grid-size))
|
||||
(size-y (second grid-size)))
|
||||
(dotimes (i nb)
|
||||
(let ((x (random size-x))
|
||||
(y (random size-y)))
|
||||
(when (eq (member (list x y) snake :test #'equal-coord) nil) ; Add if there is no conflict between snake and food position
|
||||
(setf food (append food `(,(list x y))))))))
|
||||
(setf food (remove-duplicates food :test #'equal-coord)))) ; Just in case two food are on the same coordinate
|
||||
|
34
server/game/utils.lisp
Normal file
34
server/game/utils.lisp
Normal file
|
@ -0,0 +1,34 @@
|
|||
(in-package :remote-snake-server-game)
|
||||
|
||||
;;; Return true when doing a legal move (ex: snake can goto left when it is in the right direction)
|
||||
(defun legal-move (active-dir dir)
|
||||
(or
|
||||
(eq active-dir dir) ; Goto same direction
|
||||
(and (or (eq dir :up) (eq dir :down)) ; Got up or down only if the snake is on the left or right direction
|
||||
(or (eq active-dir :left) (eq active-dir :right)))
|
||||
(and (or (eq dir :left) (eq dir :right)) ; Goto left or right only if the snake is on the up or down direction
|
||||
(or (eq active-dir :up) (eq active-dir :down)))))
|
||||
|
||||
;;; Grow snake of grow-size amount (snake is growing by the tail)
|
||||
(defun grow-snake (snake grow-size)
|
||||
(let* ((old-size (length snake))
|
||||
(new-size (+ old-size grow-size))
|
||||
(tail (nth (- old-size 1) snake))
|
||||
(new-tail (make-list grow-size :initial-element tail)))
|
||||
(append snake new-tail)))
|
||||
|
||||
;;; Function to compare two list of two elements
|
||||
(defun equal-coord (c1 c2)
|
||||
(let ((x1 (car c1))
|
||||
(x2 (car c2))
|
||||
(y1 (car (cdr c1)))
|
||||
(y2 (car (cdr c2))))
|
||||
(and (eql x1 x2) (eql y1 y2))))
|
||||
|
||||
;; Remove a coord from a list of coord (usefull when snake is eating a food)
|
||||
(defun remove-coord (l c)
|
||||
(if (eql l '()) l ; Terminal condition
|
||||
(let ((head (first l)))
|
||||
(if (equal-coord head c)
|
||||
(remove-coord (cdr l) c)
|
||||
(append `(,head) (remove-coord (cdr l) c))))))
|
Loading…
Add table
Reference in a new issue