标签:python
最后阶段,加入判断球拍击中小球的代码,并以小球落入窗口底部为结束。
#引入下列模块
from tkinter import * #画图模块
import random #随机数模块
import time #时间模块
#创建ball类
class Ball:
def __init__(self,canvas,paddle,color): #初始化函数,包含画布canvas和颜色color参数,并把paddle对象作为参数传给球
self.canvas = canvas #把参数canvas赋值给对象变量canvas
self.paddle = paddle #把球拍paddle参数赋值给对象变量paddle,引入球拍后就能在小球的类内定义击中球拍的函数
self.id = canvas.create_oval(10,10,25,25,fill=color) #创建椭圆,左上角和右下角xy坐标,返回代表图形的id
self.canvas.move(self.id,245,100) #把椭圆形(球)移动的画布中心,图形用id表示
starts = [-3,-2,-1,1,2,3] #给一串x分量的起始值(x和y代表横坐标和纵坐标的分量)
random.shuffle(starts) #随机混排序,赋值给对象变量x,让它起始的时候获得随机分量值,引起球每次起始角度都不同
self.x = starts[0] #对象变量x就是水平分量移动的初始值,等价于左右移动,值代表移动多少像素点
self.y = -3 #对象变量y就是垂直分量移动的初始值,等价于上下移动,值代表移动多少像素点
self.canvas_height = self.canvas.winfo_height() #获取画布的高度,画布高度是从上到下计量的
self.canvas_width = self.canvas.winfo_width() #获取画布的宽度
self.hit_bottom = False #定义一个游戏结束的因素:碰到底部,设置碰到底部变量初值为假
def hit_paddle(self,pos): #定义击中球拍函数,传入小球位置pos坐标参数,用于和球拍位置坐标值作比较判断是否击中
paddle_pos = self.canvas.coords(self.paddle.id) #由于在初始化函数中已经引入了paddle参数,就能够用球拍的id值来返回球拍的pos坐标值
if pos[2] >= paddle_pos[0] and pos[0] <= paddle_pos[2]: #如果小球的右下角x2大于等于球拍左边框x1,并且小球左上角x1小于等于球拍右边框x2,说明碰到
if pos[3] >= paddle_pos[1] and pos[3] <= paddle_pos[3]: #如果小球最底部在球拍顶边和底边之间(因为我们每次移动小球垂直分量的步长为3,可能会出现
return True #已经超过球拍顶边一点点的情况,如果只是判断是否碰到顶边,球在这种情况下就只能继续飞过球拍了)
return False #满足上述两种情况,返回真,即碰到球拍。如果不满足,返回假,即没碰到球拍
def draw(self): #定义小球的画图动作
self.canvas.move(self.id,self.x,self.y) #按照x和y定义的像素值来移动小球,比如3就是移动3个像素点位置
pos = self.canvas.coords(self.id) #coords函数通过id返回画布球的坐标列表,pos[x1,y1,x2,y2],分别代表椭圆左上右下角的xy坐标
if pos[1] <= 0: #当球左上角y1坐标小于等于0,即碰到画布顶部
self.y = 3 #重新设置对象变量y为3,开始垂直分量向下移动
if pos[3] >= self.canvas_height: #当球右下角y2坐标超过画布高度(底部)
self.hit_bottom = True #设置击中底部变量为真(刚开始定义对象变量y为-3,向上移动,但现在不让它动了,表示游戏结束)
if self.hit_paddle(pos) == True: #当击中球拍函数为真时(pos反映了小球击中球拍时的位置坐标)
self.y = -3 #重新设置对象变量y为-3,开始垂直分量向上移动
if pos[0] <= 0: #当球左上角x1坐标小于等于0,即碰到画布左边框
self.x = 3 #重新设置对象变量x为3,开始水平分量向右移动
if pos[2] >= self.canvas_width: #当球右下角x2坐标超过画布有边框
self.x = -3 #重新设置对象变量x为-3,开始水平分量向左移动
#创建Paddle类
class Paddle:
def __init__(self,canvas,color): #初始化函数,包含画布canvas和颜色color参数
self.canvas = canvas #把参数canvas赋值给对象变量canvas
self.id = canvas.create_rectangle(0,0,100,10,fill=color) #创建长方形,左上角和右下角xy坐标,返回代表图形的id
self.canvas.move(self.id,200,300) #把长方形(球拍)移动的初始位置,图形用id表示
self.x = 0 #设置对象变量x,初始值为0.也就是图形先不移动
self.canvas_width = self.canvas.winfo_width() #获取画布的宽度
self.canvas.bind_all(‘<KeyPress-Left>‘,self.turn_left) #初始化时将事件‘按下左键’和函数向左移动绑定
self.canvas.bind_all(‘<KeyPress-Right>‘,self.turn_right) #初始化时将事件‘按下右键’和函数向右移动绑定
def turn_left(self,evt): #定义使球拍向左移动的函数
self.x = -4 #每次向左移动4个像素,在画布上越来越向左,值越来越小,所以需要负数
def turn_right(self,evt): #定义使球拍向右移动的函数
self.x = 4 #每次向右移动4个像素,在画布上越来越向右,值越来越大,所以需要正数
def draw(self): #定义球拍的画图动作
self.canvas.move(self.id,self.x,0) #和小球配置大致相同,画图移动只在水平左右移动,所以y分量设置为0
pos = self.canvas.coords(self.id) #获取球拍的左上和右下角坐标值
if pos[0] <= 0: #如果左上角水平分量x,即球拍左边已经到达左边框
self.x = 0 #球拍不像小球一样需要自动回弹,所以设置水平分量x为0,即让它停止运动
elif pos[2] >= self.canvas_width: #如果右下角水平分量x,即球拍右边已经到达右边框
self.x = 0 #同样设置为0,停止运动
#创建游戏窗口和画布
tk = Tk() #用Tk()类创建一个tk对象,它就是一个基本窗口,可以在其上增加其他东西
tk.title("Game") #给Tk对象窗口加一个标题
tk.resizable(0,0) #tk窗口大小不可调整
tk.wm_attributes("-topmost",1) #告诉tkinter把窗口放到最前面
canvas = Canvas(tk,width=500,heigh=400,bd=0,highlightthickness=0) #Canvas是一个画布类
canvas.pack() #按照上面一行指定的宽度高度参数调整其自身大小
tk.update() #update强制更新屏幕,即重画
#创建球拍和小球对象
paddle = Paddle(canvas,‘blue‘) #用Paddle类创建蓝色球拍对象,球拍对象一定要优先创建,否则小球引用球拍参数时会报错
ball = Ball(canvas,paddle,‘red‘) #用Ball类创建画红色小球对象,并将球拍作为参数引入,用来在小球函数中判断是否击中球拍
#主循环,让tkinter不停地重画屏幕
while 1:
if ball.hit_bottom == False: #如果碰到底部变量为假的时候
ball.draw() #调用Ball类的作画函数
paddle.draw() #调用Paddle类的作画函数
tk.update_idletasks() #updata_idletasks和updata这两个命令
tk.update() #让tkinter快一点把画布上的东西画出来
time.sleep(0.01) #延时让动画效果慢一点标签:python
原文地址:http://142623.blog.51cto.com/132623/1881492