Python制作一款简单的乒乓球小游戏

开发工具

Python版本:3.6.4

相关模块:

pygame模块;

以及一些Python自带的模块。

相关文件

欢迎与我交流

环境搭建

pip安装需要的相关模块即可。

原理简介

游戏规则:

操作:

玩家1(右)通过操作↑↓键上下移动球拍;

玩家2(左)通过操作ws键上下移动球拍(仅双人模式有效)。

得分:

玩家没有接住乒乓球则失一分,即对方玩家得一分。得分先累计到11的一方即为获胜方。

逐步实现:

Step1:开始界面

开始界面其实很简单,只需要定义两个按钮,然后当检测到玩家点击按钮时,将按钮对应的值传到接下来的游戏主循环中即可。代码实现如下:

'''定义按钮'''
def Button(screen, position, text, button_size=(200, 50)):
  left, top = position
  bwidth, bheight = button_size
  pygame.draw.line(screen, (150, 150, 150), (left, top), (left+bwidth, top), 5)
  pygame.draw.line(screen, (150, 150, 150), (left, top-2), (left, top+bheight), 5)
  pygame.draw.line(screen, (50, 50, 50), (left, top+bheight), (left+bwidth, top+bheight), 5)
  pygame.draw.line(screen, (50, 50, 50), (left+bwidth, top+bheight), (left+bwidth, top), 5)
  pygame.draw.rect(screen, (100, 100, 100), (left, top, bwidth, bheight))
  font = pygame.font.Font(config.FONTPATH, 30)
  text_render = font.render(text, 1, (255, 235, 205))
  return screen.blit(text_render, (left+50, top+10))


'''
Function:
  开始界面
Input:
  --screen: 游戏界面
Return:
  --game_mode: 1(单人模式)/2(双人模式)
'''
def startInterface(screen):
  clock = pygame.time.Clock()
  while True:
    screen.fill((41, 36, 33))
    button_1 = Button(screen, (150, 175), '1 Player')
    button_2 = Button(screen, (150, 275), '2 Player')
    for event in pygame.event.get():
      if event.type == pygame.QUIT:
        pygame.quit()
        sys.exit()
      if event.type == pygame.MOUSEBUTTONDOWN:
        if button_1.collidepoint(pygame.mouse.get_pos()):
          return 1
        elif button_2.collidepoint(pygame.mouse.get_pos()):
          return 2
    clock.tick(10)
    pygame.display.update()

Step2:游戏主循环

接下来写游戏主循环。为了方便起见,先定义两个游戏精灵类,分别是球拍精灵和球精灵。其中球拍精灵应当具备被玩家手动控制而移动/根据乒乓球的位置由电脑自动控制而移动的能力,具体实现如下:

'''乒乓球拍'''
class Racket(pygame.sprite.Sprite):
  def __init__(self, imgpath, type_, **kwargs):
    pygame.sprite.Sprite.__init__(self)
    self.type_ = type_
    self.image = loadImage(imgpath, False)
    self.rect = self.image.get_rect()
    self.reset()
  '''移动'''
  def move(self, direction):
    if direction == 'UP':
      self.rect.top = max(0, self.rect.top-self.speed)
    elif direction == 'DOWN':
      self.rect.bottom = min(config.HEIGHT, self.rect.bottom+self.speed)
    else:
      raise ValueError('[direction] in Racket.move is <%s>, expect <%s> or <%s>...' % (direction, 'UP', 'DOWN'))
  '''电脑自动移动'''
  def automove(self, ball):
    if ball.rect.centery - 25 > self.rect.centery:
      self.move('DOWN')
    if ball.rect.centery + 25 < self.rect.centery:
      self.move('UP')
  '''初始化'''
  def reset(self):
    # 左/右边的拍
    self.rect.centerx = config.WIDTH-self.rect.width//2 if self.type_ == 'RIGHT' else self.rect.width//2
    self.rect.centery = config.HEIGHT // 2
    # 速度
    self.speed = 5
  '''绑定到屏幕上'''
  def draw(self, screen):
    screen.blit(self.image, self.rect)

而乒乓球则只需要根据当前的情况(包括是否撞到了墙,是否撞到了球拍等情况)自动移动即可。需要注意的一点是,为了避免游戏无限地进行下去,每次乒乓球撞到球拍/上下墙,乒乓球的运动速度都会增加。具体而言,代码实现如下:

'''乒乓球'''
class Ball(pygame.sprite.Sprite):
  def __init__(self, imgpath, **kwargs):
    pygame.sprite.Sprite.__init__(self)
    self.image = loadImage(imgpath)
    self.rect = self.image.get_rect()
    self.reset()
  '''移动'''
  def move(self, ball, racket_left, racket_right, hit_sound, goal_sound):
    self.rect.left = self.rect.left + self.speed * self.direction_x
    self.rect.top = min(max(self.rect.top+self.speed*self.direction_y, 0), config.HEIGHT-self.rect.height)
    # 撞到球拍
    if pygame.sprite.collide_rect(ball, racket_left) or pygame.sprite.collide_rect(ball, racket_right):
      self.direction_x, self.direction_y = -self.direction_x, random.choice([1, -1])
      self.speed += 1
      scores = [0, 0]
      hit_sound.play()
    # 撞到上侧的墙
    elif self.rect.top == 0:
      self.direction_y = 1
      self.speed += 1
      scores = [0, 0]
    # 撞到下侧的墙
    elif self.rect.top == config.HEIGHT - self.rect.height:
      self.direction_y = -1
      self.speed += 1
      scores = [0, 0]
    # 撞到左边的墙
    elif self.rect.left < 0:
      self.reset()
      racket_left.reset()
      racket_right.reset()
      scores = [0, 1]
      goal_sound.play()
    # 撞到右边的墙
    elif self.rect.right > config.WIDTH:
      self.reset()
      racket_left.reset()
      racket_right.reset()
      scores = [1, 0]
      goal_sound.play()
    # 普通情况
    else:
      scores = [0, 0]
    return scores
  '''初始化'''
  def reset(self):
    self.rect.centerx = config.WIDTH // 2
    self.rect.centery = random.randrange(self.rect.height//2, config.HEIGHT-self.rect.height//2)
    self.direction_x = random.choice([1, -1])
    self.direction_y = random.choice([1, -1])
    self.speed = 1
  '''绑定到屏幕上'''
  def draw(self, screen):
    screen.blit(self.image, self.rect)

定义完两个主要的游戏精灵类,我们就可以开始写游戏主循环了。逻辑其实很简单。首先,通过按键检测响应玩家的操作;然后,根据玩家操作实时更新游戏状态(乒乓球的位置,球拍等);最后统计得分,判断游戏是否已经结束,若结束,则进入结束界面,否则更新当前的游戏界面。具体而言,代码实现如下:

  while True:
    for event in pygame.event.get():
      if event.type == pygame.QUIT:
        pygame.quit()
        sys.exit(-1)
    screen.fill((41, 36, 33))
    # 玩家操作
    pressed_keys = pygame.key.get_pressed()
    if pressed_keys[pygame.K_UP]:
      racket_right.move('UP')
    elif pressed_keys[pygame.K_DOWN]:
      racket_right.move('DOWN')
    if game_mode == 2:
      if pressed_keys[pygame.K_w]:
        racket_left.move('UP')
      elif pressed_keys[pygame.K_s]:
        racket_left.move('DOWN')
    else:
      racket_left.automove(ball)
    # 球运动
    scores = ball.move(ball, racket_left, racket_right, hit_sound, goal_sound)
    score_left += scores[0]
    score_right += scores[1]
    # 显示
    # --分隔线
    pygame.draw.rect(screen, config.WHITE, (247, 0, 6, 500))
    # --球
    ball.draw(screen)
    # --拍
    racket_left.draw(screen)
    racket_right.draw(screen)
    # --得分
    screen.blit(font.render(str(score_left), False, config.WHITE), (150, 10))
    screen.blit(font.render(str(score_right), False, config.WHITE), (300, 10))
    if score_left == 11 or score_right == 11:
      return score_left, score_right
    clock.tick(100)
    pygame.display.update()

Step3:游戏结束界面

游戏结束界面和游戏开始界面的原理差不多,就不多说了,直接放代码吧:

'''结束界面'''
def endInterface(screen, score_left, score_right):
  clock = pygame.time.Clock()
  font1 = pygame.font.Font(config.FONTPATH, 30)
  font2 = pygame.font.Font(config.FONTPATH, 20)
  msg = 'Player on left won!' if score_left > score_right else 'Player on right won!'
  texts = [font1.render(msg, True, config.WHITE),
       font2.render('Press ESCAPE to quit.', True, config.WHITE),
       font2.render('Press ENTER to continue or play again.', True, config.WHITE)]
  positions = [[120, 200], [155, 270], [80, 300]]
  while True:
    screen.fill((41, 36, 33))
    for event in pygame.event.get():
      if event.type == pygame.QUIT:
        pygame.quit()
        sys.exit()
      if event.type == pygame.KEYDOWN:
        if event.key == pygame.K_RETURN:
          return
        elif event.key == pygame.K_ESCAPE:
          sys.exit()
          pygame.quit()
    for text, pos in zip(texts, positions):
      screen.blit(text, pos)
    clock.tick(10)
    pygame.display.update()

All done~完整源代码欢迎与我交流。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容