原文: http://zetcode.com/gui/jythonswing/nibbles/

在 Jython Swing 编程教程的这一部分中,我们将创建一个贪食蛇游戏克隆。

贪食蛇是较旧的经典视频游戏。 它最初是在 70 年代后期创建的。 后来它被带到 PC 上。 在这个游戏中,玩家控制蛇。 目的是尽可能多地吃苹果。 蛇每次吃一个苹果,它的身体就会长大。 蛇必须避开墙壁和自己的身体。

开发

蛇的每个关节的大小为 10px。 蛇由光标键控制。 最初,蛇具有三个关节。 游戏立即开始。 游戏结束后,我们在窗口中心显示"Game Over"消息。

  1. import random
  2. from java.awt import Color
  3. from java.awt import Font
  4. from java.awt import Toolkit
  5. from java.awt.event import ActionListener
  6. from java.awt.event import KeyEvent
  7. from java.awt.event import KeyListener
  8. from javax.swing import ImageIcon
  9. from javax.swing import JPanel
  10. from javax.swing import Timer
  11. WIDTH = 300
  12. HEIGHT = 300
  13. DOT_SIZE = 10
  14. ALL_DOTS = WIDTH * HEIGHT / (DOT_SIZE * DOT_SIZE)
  15. RAND_POS = 29
  16. DELAY = 140
  17. x = [0] * ALL_DOTS
  18. y = [0] * ALL_DOTS
  19. class Board(JPanel, KeyListener, ActionListener):
  20. def __init__(self):
  21. super(Board, self).__init__()
  22. self.initUI()
  23. def initUI(self):
  24. self.setBackground(Color.black)
  25. iid = ImageIcon("dot.png")
  26. self.ball = iid.getImage()
  27. iia = ImageIcon("apple.png")
  28. self.apple = iia.getImage()
  29. iih = ImageIcon("head.png")
  30. self.head = iih.getImage()
  31. self.setFocusable(True)
  32. self.addKeyListener(self)
  33. self.initGame()
  34. def initGame(self):
  35. self.left = False
  36. self.right = True
  37. self.up = False
  38. self.down = False
  39. self.inGame = True
  40. self.dots = 3
  41. for i in range(self.dots):
  42. x[i] = 50 - i * 10
  43. y[i] = 50
  44. self.locateApple()
  45. self.timer = Timer(DELAY, self)
  46. self.timer.start()
  47. def paint(self, g):
  48. # due to bug, cannot call super()
  49. JPanel.paint(self, g)
  50. if self.inGame:
  51. self.drawObjects(g)
  52. else:
  53. self.gameOver(g)
  54. def drawObjects(self, g):
  55. g.drawImage(self.apple, self.apple_x, self.apple_y, self)
  56. for z in range(self.dots):
  57. if (z == 0):
  58. g.drawImage(self.head, x[z], y[z], self)
  59. else:
  60. g.drawImage(self.ball, x[z], y[z], self)
  61. Toolkit.getDefaultToolkit().sync()
  62. g.dispose()
  63. def gameOver(self, g):
  64. msg = "Game Over"
  65. small = Font("Helvetica", Font.BOLD, 14)
  66. metr = self.getFontMetrics(small)
  67. g.setColor(Color.white)
  68. g.setFont(small)
  69. g.drawString(msg, (WIDTH - metr.stringWidth(msg)) / 2,
  70. HEIGHT / 2)
  71. def checkApple(self):
  72. if x[0] == self.apple_x and y[0] == self.apple_y:
  73. self.dots = self.dots + 1
  74. self.locateApple()
  75. def move(self):
  76. z = self.dots
  77. while z > 0:
  78. x[z] = x[(z - 1)]
  79. y[z] = y[(z - 1)]
  80. z = z - 1
  81. if self.left:
  82. x[0] -= DOT_SIZE
  83. if self.right:
  84. x[0] += DOT_SIZE
  85. if self.up:
  86. y[0] -= DOT_SIZE
  87. if self.down:
  88. y[0] += DOT_SIZE
  89. def checkCollision(self):
  90. z = self.dots
  91. while z > 0:
  92. if z > 4 and x[0] == x[z] and y[0] == y[z]:
  93. self.inGame = False
  94. z = z - 1
  95. if y[0] > HEIGHT - DOT_SIZE:
  96. self.inGame = False
  97. if y[0] < 0:
  98. self.inGame = False
  99. if x[0] > WIDTH - DOT_SIZE:
  100. self.inGame = False
  101. if x[0] < 0:
  102. self.inGame = False
  103. def locateApple(self):
  104. r = random.randint(0, RAND_POS)
  105. self.apple_x = r * DOT_SIZE
  106. r = random.randint(0, RAND_POS)
  107. self.apple_y = r * DOT_SIZE
  108. # public void actionPerformed(ActionEvent e) {
  109. def actionPerformed(self, e):
  110. if self.inGame:
  111. self.checkApple()
  112. self.checkCollision()
  113. self.move()
  114. else:
  115. self.timer.stop()
  116. self.repaint()
  117. def keyPressed(self, e):
  118. key = e.getKeyCode()
  119. if key == KeyEvent.VK_LEFT and not self.right:
  120. self.left = True
  121. self.up = False
  122. self.down = False
  123. if key == KeyEvent.VK_RIGHT and not self.left:
  124. self.right = True
  125. self.up = False
  126. self.down = False
  127. if key == KeyEvent.VK_UP and not self.down:
  128. self.up = True
  129. self.right = False
  130. self.left = False
  131. if key == KeyEvent.VK_DOWN and not self.up:
  132. self.down = True
  133. self.right = False
  134. self.left = False

首先,我们将定义一些在游戏中使用的常量。

WIDTHHEIGHT常数确定电路板的大小。 DOT_SIZE是苹果的大小和蛇的点。 ALL_DOTS常数定义了板上可能的最大点数。 RAND_POS常数用于计算苹果的随机位置。 DELAY常数确定游戏的速度。

  1. x = [0] * ALL_DOTS
  2. y = [0] * ALL_DOTS

这两个数组存储蛇的所有可能关节的 x,y 坐标。

initGame()方法初始化变量,加载图像并启动超时功能。

  1. def paint(self, g):
  2. JPanel.paint(self, g)
  3. if self.inGame:
  4. self.drawObjects(g)
  5. else:
  6. self.gameOver(g)

paint()方法内部,我们检查inGame变量。 如果为真,则绘制对象。 苹果和蛇的关节。 否则,我们显示"Game Over"文本。

  1. def drawObjects(self, g):
  2. g.drawImage(self.apple, self.apple_x, self.apple_y, self)
  3. for z in range(self.dots):
  4. if (z == 0):
  5. g.drawImage(self.head, x[z], y[z], self)
  6. else:
  7. g.drawImage(self.ball, x[z], y[z], self)
  8. Toolkit.getDefaultToolkit().sync()
  9. g.dispose()

drawObjects()方法绘制苹果和蛇的关节。 蛇的第一个关节是其头部,用红色圆圈表示。 Toolkit.getDefaultToolkit().sync()方法可确保显示为最新。 这对于动画很有用。

  1. def checkApple(self):
  2. if x[0] == self.apple_x and y[0] == self.apple_y:
  3. self.dots = self.dots + 1
  4. self.locateApple()

checkApple()方法检查蛇是否击中了苹果对象。 如果是这样,我们添加另一个蛇形关节并调用locateApple()方法,该方法将随机放置一个新的Apple对象。

move()方法中,我们有游戏的关键算法。 要了解它,请看一下蛇是如何运动的。 您控制蛇的头。 您可以使用光标键更改其方向。 其余关节在链上向上移动一个位置。 第二关节移动到第一个关节的位置,第三关节移动到第二个关节的位置,依此类推。

  1. while z > 0:
  2. x[z] = x[(z - 1)]
  3. y[z] = y[(z - 1)]
  4. z = z - 1

该代码将关节向上移动。

  1. if self.left:
  2. x[0] -= DOT_SIZE

将头向左移动。

checkCollision()方法中,我们确定蛇是否击中了自己或撞墙之一。

  1. while z > 0:
  2. if z > 4 and x[0] == x[z] and y[0] == y[z]:
  3. self.inGame = False
  4. z = z - 1

如果蛇用头撞到关节之一,我们就结束游戏。

  1. if y[0] > HEIGHT - DOT_SIZE:
  2. self.inGame = False

如果蛇击中了棋盘的底部,我们就结束了游戏。

locateApple()方法在板上随机放置一个苹果。

  1. r = random.randint(0, RAND_POS)

我们得到一个从 0 到RAND_POS-1的随机数。

  1. self.apple_x = r * DOT_SIZE
  2. ...
  3. self.apple_y = r * DOT_SIZE

这些行设置了apple对象的 x,y 坐标。

  1. def actionPerformed(self, e):
  2. if self.inGame:
  3. self.checkApple()
  4. self.checkCollision()
  5. self.move()
  6. else:
  7. self.timer.stop()
  8. self.repaint()

每隔DELAY ms,将调用actionPerformed()方法。 如果我们参与了游戏,我们将调用三种构建游戏逻辑的方法。 否则,我们将停止计时器。

Board类的keyPressed()方法中,我们确定按下的键。

  1. if key == KeyEvent.VK_LEFT and not self.right:
  2. self.left = True
  3. self.up = False
  4. self.down = False

如果单击左光标键,则将left变量设置为true。 在move()方法中使用此变量来更改蛇对象的坐标。 还要注意,当蛇向右行驶时,我们不能立即向左转。

  1. #!/usr/local/bin/jython
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode Jython Swing tutorial
  5. This is a simple Nibbles game
  6. clone.
  7. author: Jan Bodnar
  8. website: zetcode.com
  9. last edited: December 2010
  10. """
  11. from java.awt import Dimension
  12. from javax.swing import JFrame
  13. from Board import Board
  14. class Nibbles(JFrame):
  15. def __init__(self):
  16. super(Nibbles, self).__init__()
  17. self.initUI()
  18. def initUI(self):
  19. self.board = Board()
  20. self.board.setPreferredSize(Dimension(300, 300))
  21. self.add(self.board)
  22. self.setTitle("Nibbles")
  23. self.pack()
  24. self.setResizable(False)
  25. self.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE)
  26. self.setLocationRelativeTo(None)
  27. self.setVisible(True)
  28. if __name__ == '__main__':
  29. Nibbles()

在这个类中,我们设置了贪食蛇游戏。

Jython Swing 中的贪食蛇 - 图1

图:贪食蛇

这是使用 Swing 库和 Jython 编程语言编写的贪食蛇电脑游戏。