Mini-project #4 - "Pong"

In this project, we will build a version of Pong,
one of the first arcade video games (1972). While Pong is not particularly exciting compared to today's video games, Pong is relatively simple to build and provides a nice opportunity to work on the skills that you will need to build a game like Asteroids.
As usual, we have provided a program
template
 that can be used to guide your development of Pong.

Mini-project development process

  1. Add code to the program template that draws a ball moving across the Pong table. We recommend that you add the positional update for the ball to the draw handler as shown in the second part of the "Motion" video.
  2. Add code to the function spawn_ball that
    spawns a ball in the middle of the table and assigns the ball a fixed velocity (for now). Ignore the parameter direction at
    this point.
  3. Add a call to spawn_ball in
    the function new_game which starts a game of Pong. Note
    that the program templates includes an initial call to new_game in
    the main body of your program to get a game going immediately.
  4. Modify your code such that the ball collides with and bounces off of the top and bottom walls. Experiment with different hard-coded initial velocities to test your code.
  5. Add randomization to the velocity in spawn_ball(direction) The
    velocity of the ball should be upwards and towards the right if direction
    == RIGHT
     and upwards and towards the left if direction
    == LEFT
    . The exact values for the horizontal and vertical components of this velocity should be generated using random.randrange().
    For the horizontal velocity, we suggest a speed of around random.randrange(120,
    240)
     pixels per second. For the vertical velocity, we suggest a speed of around random.randrange(60,
    180)
     pixels per second. (You will need to set the signs of velocities appropriately.)
  6. Add code to the draw handler that tests whether the ball touches/collides with the left and right gutters. (Remember that the gutters are offset from the left and right edges of the canvas by the width of the paddle as described
    in the "Pong" video.) When the ball touches a gutter, use either spawn_ball(LEFT) or spawn_ball(RIGHT) to
    respawn the ball in the center of the table headed towards the opposite gutter.
  7. Next, add code that draws the left and right paddles in their respective gutters. The vertical positions of these two paddles should depend on two global variables. (In the template, the variables were paddle1_pos and paddle2_pos.)
  8. Add code that modifies the values of these vertical positions via an update in the draw handler.  The update should reference two global variables that contain the vertical velocities of the paddles. (In the template, the variables
    were paddle1_vel and paddle2_vel.)
  9. Update the values of these two vertical velocities using key handlers. The "w" and "s" keys should control the vertical velocity of the left paddle while the "Up arrow" and "Down arrow" key should control the velocity of the
    right paddle. In our version of Pong, the left paddle moves up at a constant velocity if the "w" key is pressed and moves down at a constant velocity if the "s" is pressed and is motionless if neither is pressed. (The motion if both are pressed is up to you.)
    To achieve this effect, you will need to use both a keydown and a keyup handler to increase/decrease the vertical velocity in an appropriate manner.
  10. Restrict your paddles to stay entirely on the canvas by adding a check before you update the paddles' vertical positions in the draw handler. In particular, test whether the current update for a paddle's position will move part
    of the paddle off of the screen. If it does, don't allow the update.
  11. Modify your collision code for the left and right gutters in step 6 to check whether the ball is actually striking a paddle when it touches a gutter. If so, reflect the ball back into play. This collision model eliminates the
    possibility of the ball striking the edge of the paddle and greatly simplifies your collision/reflection code.
  12. To moderately increase the difficulty of your game, increase the velocity of the ball by 10% each time it strikes a paddle.
  13. Add scoring to the game as shown in the Pong video lecture. Each time the ball strikes the left or right gutter (but not a paddle), the opposite player receives a point and ball is respawned appropriately.
  14. Finally, add code to new_game which
    resets the score before calling spawn_ball. Add a "Restart"
    button that calls new_game to reset the score and relaunch
    the ball.

Your final version of Pong should be remarkably similar to the original arcade Pong game. Our full implementation of Pong took a little
more than 100 lines of code with comments.

Grading Rubric - 19 pts total (scaled to 100 pts)

  • 1 pt - The ball spawns in the middle of the canvas with either an upward left or an upward right velocity. No credit if the ball moves only horizontally left or right. Bleh, that would be boring!
  • 2 pts - The ball bounces off of the top and bottom walls correctly. (1 pt each)
  • 2 pts - The ball respawns in the middle of the screen when it strikes the left or right gutter but not the paddles. (1 pt for each side) Give credit for this item even if the ball hits the edge of the canvas instead of the gutter.
  • 1 pt - The left and right gutters (instead of the edges of the canvas) are properly used as the edges of the table.
  • 1 pt - The ball spawns moving towards the player that won the last point.
  • 2 pts - The 'w' and 's' keys correctly control the velocity of the left paddle as described above. Please test each key in isolation. (1 pt if the paddle moves, but in an incorrect manner in response to 'w' and 's' key presses.)
  • 2 pts - The up and down arrows keys correctly control the velocity of the right paddle as described above. Please test each key in isolation. (1 pt if the paddle moves, but in an incorrect manner in response to up and down arrow key presses.)
  • 2 pts - The edge of each paddle is flush with the gutter. (1 pt per paddle)
  • 2 pts - The paddles stay on the canvas at all times. (1 pt per paddle)
  • 2 pts - The ball correctly bounces off the left and right paddles. (1 pt per paddle)
  • 1 pt - The scoring text is positioned and updated appropriately. The positioning need only approximate that in the video.
  • 1 pt - The game includes a "Restart" button that resets the score and relaunches the ball.


my-work:
# Implementation of classic arcade game Pong

import simplegui
import random # initialize globals - pos and vel encode vertical info for paddles
WIDTH = 600
HEIGHT = 400
BALL_RADIUS = 20
PAD_WIDTH = 8
PAD_HEIGHT = 80
HALF_PAD_WIDTH = PAD_WIDTH / 2
HALF_PAD_HEIGHT = PAD_HEIGHT / 2
LEFT = False
RIGHT = True paddle1_pos = HEIGHT/2 - HALF_PAD_HEIGHT
paddle2_pos = HEIGHT/2 - HALF_PAD_HEIGHT
paddle1_vel = 0
paddle2_vel = 0
score1 = 0
score2 = 0 # initialize ball_pos and ball_vel for new bal in middle of table
ball_pos = [WIDTH/2, HEIGHT/2]
ball_vel = [1, 1]
# if direction is RIGHT, the ball's velocity is upper right, else upper left
def spawn_ball(direction):
global ball_pos, ball_vel # these are vectors stored as lists
ball_pos = [WIDTH/2, HEIGHT/2]
ball_vel = [1, 0]
if (direction == RIGHT):
ball_vel = [-random.randrange(120, 240), -random.randrange(60, 180)]
elif (direction == LEFT):
ball_vel = [random.randrange(120, 240), -random.randrange(60, 180)] # define event handlers
def new_game():
global paddle1_pos, paddle2_pos, paddle1_vel, paddle2_vel # these are numbers
global score1, score2 # these are ints
spawn_ball(random.randrange(0, 2))
score1 = 0
score2 = 0
paddle1_pos = HEIGHT/2 - HALF_PAD_HEIGHT
paddle2_pos = HEIGHT/2 - HALF_PAD_HEIGHT
paddle1_vel = 0
paddle2_vel = 0 def draw(canvas):
global score1, score2, paddle1_pos, paddle2_pos, ball_pos, ball_vel # draw mid line and gutters
canvas.draw_line([WIDTH / 2, 0],[WIDTH / 2, HEIGHT], 1, "White")
canvas.draw_line([PAD_WIDTH, 0],[PAD_WIDTH, HEIGHT], 1, "White")
canvas.draw_line([WIDTH - PAD_WIDTH, 0],[WIDTH - PAD_WIDTH, HEIGHT], 1, "White") # draw ball
canvas.draw_circle(ball_pos, BALL_RADIUS, 2, 'white','orange') #collides with and bounces off of the top and bottom walls.
if (ball_pos[1] <= BALL_RADIUS) or (ball_pos[1]>=HEIGHT-BALL_RADIUS):
ball_vel[1] = - ball_vel[1]
# touches/collides with the left and right gutters
if ball_pos[0] <= BALL_RADIUS+PAD_WIDTH:
if (paddle1_pos - HALF_PAD_HEIGHT)<= ball_pos[1] <= (paddle1_pos + HALF_PAD_HEIGHT):
ball_vel[0] = - ball_vel[0]
ball_vel[0] = ball_vel[0] * 1.1
ball_vel[1] = ball_vel[1] * 1.1
else:
spawn_ball(LEFT)
score2 += 1
elif ball_pos[0] >= WIDTH - BALL_RADIUS:
if (paddle2_pos - HALF_PAD_HEIGHT)<= ball_pos[1] <= (paddle2_pos + HALF_PAD_HEIGHT):
ball_vel[0] = - ball_vel[0]
ball_vel[0] = ball_vel[0] * 1.1
ball_vel[1] = ball_vel[1] * 1.1
else:
spawn_ball(RIGHT)
score1 += 1
# update ball
ball_pos[0] = ball_pos[0] + ball_vel[0] / 60
ball_pos[1] = ball_pos[1] + ball_vel[1] / 60 # update paddle's vertical position, keep paddle on the screen
paddle1_pos = paddle1_pos + paddle1_vel
paddle2_pos = paddle2_pos + paddle2_vel
if paddle1_pos < 40:
paddle1_pos = 40
if paddle2_pos < 40:
paddle2_pos = 40
if paddle1_pos > 360:
paddle1_pos= 360
if paddle2_pos> 360:
paddle2_pos= 360
# draw paddles
canvas.draw_line((HALF_PAD_WIDTH,paddle1_pos - HALF_PAD_HEIGHT), (HALF_PAD_WIDTH,paddle1_pos + HALF_PAD_HEIGHT),
8, 'white') canvas.draw_line((WIDTH-HALF_PAD_WIDTH,paddle2_pos - HALF_PAD_HEIGHT), (WIDTH-HALF_PAD_WIDTH,paddle2_pos + HALF_PAD_HEIGHT),
8, 'white')
# draw scores
canvas.draw_text(str(score1),[250,50], 60, 'white')
canvas.draw_text(str(score2),[320,50], 60, 'white')
def keydown(key):
global paddle1_vel, paddle2_vel
if key == simplegui.KEY_MAP["w"]:
paddle1_vel = -10 if key == simplegui.KEY_MAP["s"]:
paddle1_vel = 10
if key == simplegui.KEY_MAP["up"]:
paddle2_vel = -10
if key == simplegui.KEY_MAP["down"]:
paddle2_vel = 10
def keyup(key):
global paddle1_vel, paddle2_vel
if key == simplegui.KEY_MAP["w"]:
paddle1_vel = 0
if key == simplegui.KEY_MAP["s"]:
paddle1_vel = 0
if key == simplegui.KEY_MAP["up"]:
paddle2_vel = 0
if key == simplegui.KEY_MAP["down"]:
paddle2_vel = 0 # create frame
frame = simplegui.create_frame("Pong", WIDTH, HEIGHT)
frame.set_draw_handler(draw)
frame.set_keydown_handler(keydown)
frame.set_keyup_handler(keyup)
button1 = frame.add_button('Restart', new_game, 50) # start frame
new_game()
frame.start()

完成链接:CodeSkulptor

Mini-project # 4 - "Pong"___An Introduction to Interactive Programming in Python"RICE"的更多相关文章

  1. Mini-project # 1 - Rock-paper-scissors-___An Introduction to Interactive Programming in Python"RICE"

    Mini-project description - Rock-paper-scissors-lizard-Spock Rock-paper-scissors is a hand game that ...

  2. An Introduction to Interactive Programming in Python (Part 1) -- Week 2_3 练习

    Mini-project description - Rock-paper-scissors-lizard-Spock Rock-paper-scissors is a hand game that ...

  3. An Introduction to Interactive Programming in Python

    这是在coursera上面的一门学习pyhton的基础课程,由RICE的四位老师主讲.生动有趣,一共是9周的课程,每一周都会有一个小游戏,经历一遍,对编程会产生很大的兴趣. 所有的程序全部在老师开发的 ...

  4. 【python】An Introduction to Interactive Programming in Python(week two)

    This is a note for https://class.coursera.org/interactivepython-005 In week two, I have learned: 1.e ...

  5. An Introduction to Interactive Programming in Python (Part 1) -- Week 2_2 练习

    #Practice Exercises for Logic and Conditionals # Solve each of the practice exercises below. # 1.Wri ...

  6. An Introduction to Interactive Programming in Python (Part 1) -- Week 2_1 练习

    # Practice Exercises for Functions # Solve each of the practice exercises below. # 1.Write a Python ...

  7. Quiz 6b Question 8————An Introduction to Interactive Programming in Python

     Question 8 We can use loops to simulate natural processes over time. Write a program that calcula ...

  8. Quiz 6b Question 7————An Introduction to Interactive Programming in Python

     Question 7 Convert the following English description into code. Initialize n to be 1000. Initiali ...

  9. Quiz 6a Question 7————An Introduction to Interactive Programming in Python

     First, complete the following class definition: class BankAccount: def __init__(self, initial_bal ...

随机推荐

  1. android-SQLite 和 Content

    SQLite 游标(Cursor)相当于指向底层数据中结果集的指针,而不是提取和返回结果值的副本,是在结果集中对位置(行)进行控制的管理方式. moveToFirst:把游标移动到查询结果的第一行 m ...

  2. ubantu root 默认密码

    安装完Ubuntu后忽然意识到没有设置root密码,不知道密码自然就无法进入根用户下.到网上搜了一下,原来是这麽回事.Ubuntu的默认root密码是随机的,即每次开机都有一个新的root密码.我们可 ...

  3. 一段关于 Unix、Linux 和 Windows 的暗黑史

    “SCO 在言语上变得越来越好斗,而且还拒绝展示有关诉讼的任何证据,一切都似乎在表明,SCO只不过是在那里拉虎皮做大旗地狂言乱语.但是,微软 决不会轻易放弃这么可以一个利用这些狂言乱语的好机会.”20 ...

  4. 浅谈Java泛型中的extends和super关键字(转)

    通配符 在本文的前面的部分里已经说过了泛型类型的子类型的不相关性.但有些时候,我们希望能够像使用普通类型那样使用泛型类型: 向上造型一个泛型对象的引用 向下造型一个泛型对象的引用 向上造型一个泛型对象 ...

  5. LINQ to SQL的CRUD操作

    创建数据对象模型 sqlmetal /code:"C:\MyProjects\VS2008\Data\LinqConsoleApp2\LinqConsoleApp2\northwnd.cs& ...

  6. nodejs服务端开发学习笔记

    正在学习中,不断改错... 学习了一段时间nodejs,对其中的很多东西还不是很理解,在网上看过很多的例子,希望通过自己的一些总结让自己了解的更全面些,同时也作为学习笔记留存备忘. 准备工作 node ...

  7. Android视频应用去广告学习实践

    注意:本文仅仅供学习研究用途 第一步 素材搜集 腾讯视频应用:http://download.csdn.net/detail/itleaks/7991795      反汇编工具:           ...

  8. loadRunner 11.0 安装及破解

    http://jingyan.baidu.com/article/20095761b31b58cb0621b463.html 破解时必须是管理员账户登录.

  9. Random类短时间大量随机重复的问题

    先声明一下,我是在那篇文章上看到的解决方法: http://dl.download.csdn.net/down10/20141103/4b173214e41ff3207305c2470524b0f3. ...

  10. 在CG/HLSL中访问着色器属性(Properties)

    在CG/HLSL中访问着色器属性 Shader在Properties块中访问材质属性.如果你想在一个着色程序中访问一些属性,你需要声明一个Cg/HLSL具有相同的名称和一个匹配的类型的变量. Prop ...