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操作串口的方法
Jun 17 Python
Python打印斐波拉契数列实例
Jul 07 Python
Python使用lxml模块和Requests模块抓取HTML页面的教程
May 16 Python
python根据unicode判断语言类型实例代码
Jan 17 Python
Flask实现图片的上传、下载及展示示例代码
Aug 03 Python
对PyQt5中的菜单栏和工具栏实例详解
Jun 20 Python
python多线程实现TCP服务端
Sep 03 Python
vim自动补全插件YouCompleteMe(YCM)安装过程解析
Oct 21 Python
基于Python中isfile函数和isdir函数使用详解
Nov 29 Python
基于Python实现全自动下载抖音视频
Nov 06 Python
python读取pdf格式文档的实现代码
Apr 01 Python
Pyqt5将多个类组合在一个界面显示的完整示例
Sep 04 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运行环境教程
2015/02/12 PHP
php实现压缩合并js的方法【附demo源码下载】
2016/09/22 PHP
PHP的简单跳转提示的实现详解
2019/03/14 PHP
详解CSS样式中的 !important * _ 符号
2021/03/09 HTML / CSS
基于jQuery实现的Ajax 验证用户名是否存在的实现代码
2011/04/06 Javascript
JavaScript操作XML 使用百度RSS作为新闻源示例
2012/02/17 Javascript
Jquery使用css方法改变样式实例
2015/05/18 Javascript
jQuery检查元素存在性(推荐)
2016/09/17 Javascript
快速解决js开发下拉框中blur与click冲突
2016/10/10 Javascript
JavaScript利用Date实现简单的倒计时实例
2017/01/12 Javascript
Vue中的scoped实现原理及穿透方法
2018/05/15 Javascript
vue实现百度下拉列表交互操作示例
2019/03/12 Javascript
JS 设计模式之:工厂模式定义与实现方法浅析
2020/05/06 Javascript
详解Python中的正则表达式
2018/07/08 Python
Python设计模式之迭代器模式原理与用法实例分析
2019/01/10 Python
在python 不同时区之间的差值与转换方法
2019/01/14 Python
python plotly绘制直方图实例详解
2019/07/22 Python
python+django+rest框架配置创建方法
2019/08/31 Python
解决python 读取excel时 日期变成数字并加.0的问题
2019/10/08 Python
Python接口测试文件上传实例解析
2020/05/22 Python
python连接mysql有哪些方法
2020/06/24 Python
详解Python IO编程
2020/07/24 Python
Python pip使用超时问题解决方案
2020/08/03 Python
Node.js 和 Python之间该选择哪个?
2020/08/05 Python
CSS3的first-child选择器实战攻略
2016/04/28 HTML / CSS
Get The Label中文官网:英国运动时尚购物平台
2017/04/19 全球购物
KENZO官网:高田贤三在法国创立的品牌
2019/05/16 全球购物
汇科协同Java笔试题
2012/03/31 面试题
模具专业推荐信
2013/10/30 职场文书
前台接待岗位职责
2013/12/03 职场文书
园林技术个人的自我评价
2014/01/08 职场文书
餐厅筹备计划书
2014/04/25 职场文书
安全演讲稿开场白
2014/08/25 职场文书
大学生安全教育主题班会
2015/08/12 职场文书
《实心球》教学反思
2016/02/23 职场文书
python中print格式化输出的问题
2021/04/16 Python