2022-10-31 21:04:13 +01:00
|
|
|
#!/usr/bin/env python
|
2022-10-31 22:12:02 +01:00
|
|
|
import sys, pygame, random
|
2022-10-31 21:04:13 +01:00
|
|
|
|
|
|
|
|
|
|
|
class Snake:
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Programmable Game of Snake written in PyGame
|
|
|
|
"""
|
2022-10-31 21:04:13 +01:00
|
|
|
|
2022-11-02 08:50:05 +01:00
|
|
|
def __init__(self, startat=(0,0), margin=80,length=4,grid_width=30,grid_height=30, grid_pts=30,fps=15):
|
2022-11-01 10:12:43 +01:00
|
|
|
# Init attributes
|
2022-10-31 21:04:13 +01:00
|
|
|
self.grid_width=grid_width
|
|
|
|
self.grid_height=grid_height
|
|
|
|
self.grid_pts=grid_pts
|
2022-10-31 23:19:27 +01:00
|
|
|
self.margin=margin
|
2022-11-01 09:23:24 +01:00
|
|
|
self.default_length=length
|
2022-11-01 09:14:06 +01:00
|
|
|
self.attempt=0
|
2022-11-01 09:23:24 +01:00
|
|
|
self.fps=fps
|
2022-11-01 22:12:14 +01:00
|
|
|
self.startat=startat
|
2022-11-02 08:50:05 +01:00
|
|
|
self.last_score=-1
|
2022-11-02 10:29:11 +01:00
|
|
|
self.score=0
|
2022-11-01 10:12:43 +01:00
|
|
|
# Setup pygame
|
2022-10-31 21:04:13 +01:00
|
|
|
pygame.init()
|
2022-11-01 08:21:45 +01:00
|
|
|
self.font=pygame.font.SysFont(pygame.font.get_default_font(), int(self.margin/2))
|
2022-11-01 09:14:06 +01:00
|
|
|
self.font_small=pygame.font.SysFont(pygame.font.get_default_font(), int(self.margin/2.5))
|
2022-10-31 23:19:27 +01:00
|
|
|
self.screen=pygame.display.set_mode((grid_width*grid_pts,grid_height*grid_pts+margin))
|
2022-11-02 08:50:05 +01:00
|
|
|
self.clock = pygame.time.Clock()
|
|
|
|
# Start game
|
|
|
|
self.new_game()
|
|
|
|
self.draw()
|
2022-10-31 21:04:13 +01:00
|
|
|
|
2022-11-01 09:23:24 +01:00
|
|
|
def new_game(self):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Reset game state
|
|
|
|
"""
|
2022-11-01 22:12:14 +01:00
|
|
|
self.snake=[self.startat]*self.default_length
|
2022-11-01 09:23:24 +01:00
|
|
|
self.direction=3 # Like clock (12=up, 3=right, 6=bottom, 9=left)
|
|
|
|
self.new_apple()
|
2022-11-02 10:29:11 +01:00
|
|
|
self.last_score=self.score
|
2022-11-01 09:23:24 +01:00
|
|
|
self.score=0
|
|
|
|
self.attempt+=1
|
|
|
|
|
2022-11-01 10:37:27 +01:00
|
|
|
def draw_pts(self,x,y,color=(255,255,255),scale=1):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Draw element on the snake area
|
|
|
|
"""
|
2022-11-01 10:37:27 +01:00
|
|
|
delta=int(self.grid_pts-int(self.grid_pts*scale))
|
|
|
|
rect=pygame.Rect(self.grid_pts*x+delta, self.grid_pts*y+self.margin+delta, self.grid_pts-2*delta, self.grid_pts-2*delta)
|
2022-10-31 23:19:27 +01:00
|
|
|
pygame.draw.rect(self.screen,color,rect, 0)
|
|
|
|
|
2022-11-01 10:12:43 +01:00
|
|
|
def draw_infos(self,color=(255,255,255),thickness=10):
|
|
|
|
"""
|
|
|
|
Draw the information bar
|
|
|
|
"""
|
|
|
|
# Draw separator
|
2022-10-31 23:19:27 +01:00
|
|
|
rect=pygame.Rect(0, self.margin-thickness, self.grid_width*self.grid_pts, thickness)
|
2022-11-01 10:12:43 +01:00
|
|
|
pygame.draw.rect(self.screen,(180,180,180),rect, 0)
|
|
|
|
# Draw score
|
|
|
|
text = self.font.render('Score: '+str(self.score)+" Length: "+str(len(self.snake))+' Attempt: '+str(self.attempt), True, color)
|
2022-11-01 09:14:06 +01:00
|
|
|
text_center=text.get_rect(center = (self.grid_width*self.grid_pts // 2, (self.margin-thickness) // 2))
|
2022-11-01 08:57:24 +01:00
|
|
|
self.screen.blit(text, text_center)
|
2022-11-01 10:12:43 +01:00
|
|
|
|
2022-10-31 22:12:02 +01:00
|
|
|
def new_apple(self):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Create a new apple
|
|
|
|
"""
|
2022-11-01 09:55:38 +01:00
|
|
|
self.apple=(random.randint(0,self.grid_width-1),random.randint(0,self.grid_height-1))
|
2022-11-01 20:21:34 +01:00
|
|
|
if self.apple in self.snake:
|
|
|
|
self.new_apple()
|
2022-10-31 22:12:02 +01:00
|
|
|
|
2022-10-31 21:54:49 +01:00
|
|
|
def move(self):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Move the snake
|
|
|
|
"""
|
2022-10-31 22:09:01 +01:00
|
|
|
# Update tail
|
2022-10-31 21:54:49 +01:00
|
|
|
if len(self.snake)>1:
|
2022-10-31 22:09:01 +01:00
|
|
|
tmp=self.snake[0]
|
2022-10-31 21:54:49 +01:00
|
|
|
for i in range(1,len(self.snake)):
|
|
|
|
newtmp=self.snake[i]
|
|
|
|
self.snake[i]=tmp
|
|
|
|
tmp=newtmp
|
2022-10-31 22:09:01 +01:00
|
|
|
# Update head
|
|
|
|
h=self.snake[0] # Head
|
|
|
|
if self.direction==3:
|
|
|
|
self.snake[0]=(h[0]+1,h[1])
|
|
|
|
elif self.direction==9:
|
|
|
|
self.snake[0]=(h[0]-1,h[1])
|
|
|
|
elif self.direction==12:
|
|
|
|
self.snake[0]=(h[0],h[1]-1)
|
|
|
|
else:
|
|
|
|
self.snake[0]=(h[0],h[1]+1)
|
2022-10-31 21:54:49 +01:00
|
|
|
|
|
|
|
def draw_snake(self):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Draw the snake with color effects
|
|
|
|
"""
|
2022-11-01 09:14:06 +01:00
|
|
|
for i in range(0,len(self.snake)):
|
|
|
|
color=(0,150,150) if i==0 else (0,max(255-i*10,120),0)
|
|
|
|
elt=self.snake[i]
|
|
|
|
self.draw_pts(elt[0],elt[1],color=color)
|
2022-11-01 10:37:27 +01:00
|
|
|
if i>0:
|
|
|
|
self.draw_pts(elt[0],elt[1],color=(color[0],int(color[1]/2),color[2]),scale=max(0.7,1-i/4))
|
|
|
|
else:
|
|
|
|
self.draw_pts(elt[0],elt[1],color=(color[0],int(color[1]/2),color[2]),scale=0.86)
|
2022-10-31 21:54:49 +01:00
|
|
|
|
2022-11-01 13:44:10 +01:00
|
|
|
def draw_apple(self,radius=None):
|
2022-11-01 13:45:41 +01:00
|
|
|
"""
|
|
|
|
Draw the apple on the screen
|
|
|
|
"""
|
2022-11-01 13:44:10 +01:00
|
|
|
if radius==None:
|
|
|
|
radius=self.grid_pts/2
|
|
|
|
pygame.draw.circle(self.screen, (210,0,0), (self.apple[0]*self.grid_pts+int(self.grid_pts/2),self.apple[1]*self.grid_pts+self.margin+int(self.grid_pts/2)), radius)
|
|
|
|
pygame.draw.circle(self.screen, (170,0,0), (self.apple[0]*self.grid_pts+int(self.grid_pts/2),self.apple[1]*self.grid_pts+self.margin+int(self.grid_pts/2)), radius/1.5)
|
|
|
|
|
|
|
|
|
2022-10-31 22:42:48 +01:00
|
|
|
def has_loose(self):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
|
|
|
Return true if the game is loose
|
|
|
|
"""
|
2022-10-31 22:42:48 +01:00
|
|
|
if self.snake.count(self.snake[0])>1:
|
|
|
|
return(True)
|
|
|
|
h=self.snake[0]
|
|
|
|
if h[0]<0 or h[1]<0 or h[0] >= self.grid_width or h[1] >= self.grid_height:
|
|
|
|
return(True)
|
|
|
|
return(False)
|
|
|
|
|
2022-11-02 08:50:05 +01:00
|
|
|
def draw(self):
|
|
|
|
self.screen.fill((0,0,0))
|
|
|
|
self.draw_snake()
|
|
|
|
self.draw_apple()
|
|
|
|
self.draw_infos()
|
|
|
|
pygame.display.flip()
|
2022-11-01 09:55:38 +01:00
|
|
|
|
2022-11-02 09:29:45 +01:00
|
|
|
def play(self,direction,handle_quit=True):
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
2022-11-02 08:50:05 +01:00
|
|
|
Play using wall clock directions (12=up, 3=right, 6=down and 9=left)
|
|
|
|
"""
|
|
|
|
# Play
|
2022-11-02 10:29:11 +01:00
|
|
|
if (self.direction==12 and direction!=6) or\
|
|
|
|
(self.direction==6 and direction!=12) or\
|
|
|
|
(self.direction==3 and direction!=9) or\
|
|
|
|
(self.direction==9 and direction!=3):
|
|
|
|
self.direction=direction
|
2022-11-02 08:50:05 +01:00
|
|
|
self.move()
|
|
|
|
# Return code
|
|
|
|
code=0
|
|
|
|
if self.apple==self.snake[0]:
|
|
|
|
self.snake.append(self.snake[len(self.snake)-1])
|
|
|
|
self.new_apple()
|
|
|
|
self.score+=1
|
|
|
|
code=1
|
|
|
|
elif self.has_loose():
|
|
|
|
self.new_game()
|
|
|
|
code=-1
|
|
|
|
# Refresh screen
|
|
|
|
self.draw()
|
|
|
|
self.clock.tick(self.fps)
|
2022-11-02 09:12:06 +01:00
|
|
|
# Ensure we not quit
|
2022-11-02 09:29:45 +01:00
|
|
|
if handle_quit:
|
|
|
|
for event in pygame.event.get():
|
|
|
|
if event.type == pygame.QUIT:
|
|
|
|
pygame.quit()
|
|
|
|
sys.exit()
|
2022-11-02 08:50:05 +01:00
|
|
|
return(code)
|
|
|
|
|
|
|
|
def play2(self,direction):
|
|
|
|
"""
|
|
|
|
Play using ascii directions
|
|
|
|
"""
|
|
|
|
if direction.lower()=="up":
|
|
|
|
return(self.play(self,12))
|
|
|
|
elif direction.lower()=="right":
|
|
|
|
return(self.play(self,3))
|
|
|
|
elif direction.lower()=="down":
|
|
|
|
return(self.play(self,6))
|
|
|
|
elif direction.lower()=="left":
|
|
|
|
return(self.play(self,9))
|
|
|
|
|
|
|
|
def play3(self,direction):
|
|
|
|
"""
|
|
|
|
Play using 0123 directions (0=right, 1=down, 3=left and 4=up)
|
|
|
|
"""
|
|
|
|
if direction == 0:
|
|
|
|
return(self.play(3))
|
|
|
|
elif direction == 1:
|
|
|
|
return(self.play(6))
|
|
|
|
elif direction == 2:
|
|
|
|
return(self.play(9))
|
|
|
|
elif direction == 3:
|
|
|
|
return(self.play(12))
|
|
|
|
|
|
|
|
def play_with_keyboard(self):
|
|
|
|
"""
|
|
|
|
Play a game using keyboard
|
2022-11-01 10:12:43 +01:00
|
|
|
"""
|
2022-10-31 21:04:13 +01:00
|
|
|
while True:
|
2022-10-31 22:42:48 +01:00
|
|
|
# Check inputs
|
2022-10-31 21:04:13 +01:00
|
|
|
for event in pygame.event.get():
|
2022-11-02 09:29:45 +01:00
|
|
|
if event.type == pygame.QUIT:
|
|
|
|
pygame.quit()
|
|
|
|
sys.exit()
|
|
|
|
elif event.type == pygame.KEYDOWN:
|
2022-10-31 22:09:01 +01:00
|
|
|
if event.key == pygame.K_LEFT and self.direction != 3:
|
|
|
|
self.direction=9
|
2022-10-31 22:42:48 +01:00
|
|
|
break
|
|
|
|
elif event.key == pygame.K_RIGHT and self.direction != 9:
|
2022-10-31 22:09:01 +01:00
|
|
|
self.direction=3
|
2022-10-31 22:42:48 +01:00
|
|
|
break
|
|
|
|
elif event.key == pygame.K_UP and self.direction != 6:
|
2022-10-31 22:09:01 +01:00
|
|
|
self.direction=12
|
2022-10-31 22:42:48 +01:00
|
|
|
break
|
|
|
|
elif event.key == pygame.K_DOWN and self.direction != 12:
|
2022-10-31 22:09:01 +01:00
|
|
|
self.direction=6
|
2022-10-31 22:42:48 +01:00
|
|
|
break
|
2022-11-02 08:50:05 +01:00
|
|
|
|
2022-11-02 09:29:45 +01:00
|
|
|
if self.play(self.direction,handle_quit=False) <0:
|
2022-11-02 08:50:05 +01:00
|
|
|
break
|
2022-11-02 09:29:45 +01:00
|
|
|
|