python tkinter实现彩球碰撞屏保


Posted in Python onJuly 30, 2019

利用Tkinter实现彩球碰撞屏保,供大家参考,具体内容如下

一、架构与思路

(1)主函数:

main():通过类启动程序;

(2)类:

ScreenSaver():用于定义屏保和主画布,调用球创建、运动等函数;
RandomBall():定义球的基本属性、球创建与运动函数;

(3)对象:单个球,需要创建、运动(包括碰撞反弹),通过循环调用实现多个球并存的效果

create_ball():单个球创建函数;
move_ball():单个球运动函数;

(4)20181215更新:此处对原有屏保程序的退出环节进行了扩展,使用messabox工具建立消息框,询问是否退出,点击“确定”会直接退出,点击“取消”仍留在程序中。

程序架构和思路如下:

python tkinter实现彩球碰撞屏保

二、代码实现

根据上述思路,利用python实现屏保程序,代码如下:

import random
import tkinter
import tkinter.messagebox


class RandomBall():
 '''
 单个球定义、运动的类
 '''
 def __init__(self, root_canvas, width, height):
  '''
  参数说明:
  canvas:从ScreenSaver类中传入的画布
  width,height:从SS类中传入的宽高,即屏幕宽高
  '''

  # 将传入变量赋为RB类的属性
  self.canvas = root_canvas
  self.width = width
  self.height = height

  # 随机生成球的中心坐标
  self.xcenter = random.randint(100, width-100)
  self.ycenter = random.randint(100, height-100)
  # 随机生成球的运动速度
  self.xvelocity = random.randint(8,16)
  self.yvelocity = random.randint(8,16)
  # 计算球的半径
  self.radius = random.randint(60, 100)
  # 利用十六进制随机数与lambda表达式生成球的颜色
  # RGB表示法:三个数字,每个数字的值是0-255之间,表示红绿蓝三个颜色的大小
  # 在某些系统中,直接用英文单词表示也可以,比如red,green
  color = lambda : random.randint(0,255)
  self.color = '#%02x%02x%02x' % (color(),color(),color())

 # 创建球的函数
 def create_ball(self):
  '''
  用构造函数定义的变量值,在canvas上画一个球
  '''
  # tkinter没有画圆形函数
  # 只有一个画椭圆函数,画椭圆需要定义两个坐标,
  # 在一个长方形内画椭圆,我们只需要定义长方形左上角和右下角就好
  # 求两个坐标的方法是,已知圆心的坐标,则圆心坐标减去半径能求出
  # 左上角坐标,加上半径能求出右下角坐标(向右x为正,向下y为正)
  xleftup = self.xcenter - self.radius
  yleftup = self.ycenter - self.radius
  xrightdown = self.xcenter + self.radius
  yrightdown = self.ycenter + self.radius
  # 创建球
  self.item = self.canvas.create_oval(xleftup,yleftup,
           xrightdown,yrightdown,
           fill=self.color,
           outline=self.color)

 # 球运动的函数
 def move_ball(self):
  # 计算球移动后的中心点坐标
  self.xcenter += self.xvelocity
  self.ycenter += self.yvelocity
  # 当球与边框发生碰撞时,需要进行回弹操作,即对应方向的速度取负
  if self.xcenter + self.radius >= self.width:
   self.xvelocity = - self.xvelocity
  if self.xcenter - self.radius <= 0:
   self.xvelocity = - self.xvelocity
  if self.ycenter + self.radius >= self.height:
   self.yvelocity = - self.yvelocity
  if self.ycenter - self.radius <= 0:
   self.yvelocity = - self.yvelocity
  # 在canvas上移动球,前提是create_ball已经调用
  self.canvas.move(self.item, self.xvelocity, self.yvelocity)



class ScreenSaver():
 '''
 屏保定义类
 程序启动
 '''
 def __init__(self):
  # 创建球存储列表
  self.balls = []
  # 随机生成球的数量
  self.num = random.randint(10,20)

  # 利用tkinter生成root窗口
  self.root = tkinter.Tk()
  # 获取屏幕宽、高尺寸
  root_w, root_h = self.root.winfo_screenwidth(), self.root.winfo_screenheight()
  # 取消边框
  self.root.overrideredirect(1)
  # 绑定退出函数与相应动作
  self.root.bind('<Motion>', self.myquit)
  self.root.bind('<Key>', self.myquit)
  self.root.bind('<Any-Button>', self.myquit)

  # 创建画布,配置尺寸与颜色属性
  self.canvas = tkinter.Canvas(self.root, width=root_w, height=root_h)
  self.canvas.pack()

  # 利用循环与RandomBall类在画布上画球,并append到列表中
  for i in range(self.num):
   ball = RandomBall(self.canvas, width=root_w, height=root_h)
   ball.create_ball()
   self.balls.append(ball)

  # 调用球运动函数
  self.run_screen_saver()
  # 启用tkinter时间消息循环mainloop
  self.root.mainloop()

 # 球运动函数
 def run_screen_saver(self):
  # 循环实例化的ball调用move_ball函数
  for ball in self.balls:
   ball.move_ball()
  # 使用after实现递归,通过不断调用各球的move_ball函数,实现位置刷新
  self.root.after(50, self.run_screen_saver)

 # 停止运行
 # 此处e只是利用了事件处理机制,际上并不关心事件的类型
 def myquit(self, e):
  # 扩展:
  # 此屏保程序扩展成,一旦捕获事件,则判断屏保不退出
  # 显示一个Button,Button上显示事件类型,点击Button后屏保才退出
  if tkinter.messagebox.askokcancel("彩球碰撞", '确定退出?'):
   self.root.destroy()
  else:
   pass


if __name__ == '__main__':
 # 启动屏保
 ScreenSaver()

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python单元测试unittest实例详解
May 11 Python
使用tensorflow实现线性回归
Sep 08 Python
python从list列表中选出一个数和其对应的坐标方法
Jul 20 Python
python点击鼠标获取坐标(Graphics)
Aug 10 Python
python实现的生成word文档功能示例
Aug 23 Python
python装饰器使用实例详解
Dec 14 Python
pytorch中的transforms模块实例详解
Dec 31 Python
Tensorflow读取并输出已保存模型的权重数值方式
Jan 04 Python
Python在终端通过pip安装好包以后在Pycharm中依然无法使用的问题(三种解决方案)
Mar 10 Python
Pycharm激活码激活两种快速方式(附最新激活码和插件)
Mar 12 Python
tensorflow2.0的函数签名与图结构(推荐)
Apr 28 Python
Python的代理类实现,控制访问和修改属性的权限你都了解吗
Mar 21 Python
详解python pandas 分组统计的方法
Jul 30 #Python
python文档字符串(函数使用说明)使用详解
Jul 30 #Python
python3.6 tkinter实现屏保小程序
Jul 30 #Python
简单了解python变量的作用域
Jul 30 #Python
学习和使用python的13个理由
Jul 30 #Python
Win10系统下安装labelme及json文件批量转化方法
Jul 30 #Python
python利用tkinter实现屏保
Jul 30 #Python
You might like
缓存技术详谈―php
2006/12/14 PHP
PHP中将ip地址转成十进制数的两种实用方法
2013/08/15 PHP
PHP中的session安全吗?
2016/01/22 PHP
ThinkPHP框架表单验证操作方法
2017/07/19 PHP
php获取手机端的号码以及ip地址实例代码
2018/09/12 PHP
php写入文件不覆盖的实例讲解
2019/09/17 PHP
调用js时ie6和ie7,ff的区别
2009/08/19 Javascript
JS获取后台Cookies值的小例子
2013/03/04 Javascript
JS模板实现方法
2013/04/03 Javascript
深入探讨JavaScript、JQuery屏蔽网页鼠标右键菜单及禁止选择复制
2014/06/10 Javascript
jQuery中insertBefore()方法用法实例
2015/01/08 Javascript
JavaScript中定义函数的三种方法
2015/03/12 Javascript
js+HTML5实现视频截图的方法
2015/06/16 Javascript
JS实现样式清新的横排下拉菜单效果
2015/10/09 Javascript
理解jquery事件冒泡
2016/01/03 Javascript
jQuery动态添加及删除表单上传元素的方法(附demo源码下载)
2016/01/15 Javascript
微信小程序加载更多 点击查看更多
2016/11/29 Javascript
jQuery Chosen通用初始化
2017/03/07 Javascript
详解JavaScript函数callee、call、apply的区别
2019/03/08 Javascript
Node.js HTTP服务器中的文件、图片上传的方法
2019/09/23 Javascript
[00:58]PWL开团时刻DAY5——十人开雾0换5
2020/11/04 DOTA
JS设计模式之责任链模式实例详解
2018/02/03 Python
使用Eclipse如何开发python脚本
2018/04/11 Python
将pandas.dataframe的数据写入到文件中的方法
2018/12/07 Python
python 求10个数的平均数实例
2019/12/16 Python
Django中modelform组件实例用法总结
2020/02/10 Python
matplotlib quiver箭图绘制案例
2020/04/17 Python
Java ExcutorService优雅关闭方式解析
2020/05/30 Python
关于HTML5语义标签的实践(blog页面)
2016/07/12 HTML / CSS
俄罗斯旅游网站:Tripadvisor俄罗斯
2017/03/21 全球购物
美国一站式电动和手动工具商店:International Tool
2020/11/26 全球购物
保密工作目标责任书
2014/07/28 职场文书
海洋科学专业求职信
2014/08/10 职场文书
2014年高中教师工作总结
2014/12/19 职场文书
测量员岗位职责
2015/02/14 职场文书
圣诞晚会主持词
2015/07/01 职场文书