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中fileno()方法的使用
May 24 Python
解决python3 urllib中urlopen报错的问题
Mar 25 Python
Django基础之Model操作步骤(介绍)
May 27 Python
不同版本中Python matplotlib.pyplot.draw()界面绘制异常问题的解决
Sep 24 Python
python PyTorch参数初始化和Finetune
Feb 11 Python
安装2019Pycharm最新版本的教程详解
Oct 22 Python
Python安装whl文件过程图解
Feb 18 Python
Python将二维列表list的数据输出(TXT,Excel)
Apr 23 Python
2020最新pycharm汉化安装(python工程狮亲测有效)
Apr 26 Python
Python内置异常类型全面汇总
May 28 Python
Python爬虫爬取新闻资讯案例详解
Jul 14 Python
python实现无边框进度条的实例代码
Dec 30 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 HTML代码串 截取实现代码
2009/06/29 PHP
php下连接ftp实现文件的上传、下载、删除文件实例代码
2010/06/03 PHP
jQuery+php实现ajax文件即时上传的详解
2013/06/17 PHP
PHP 反射(Reflection)使用实例
2015/05/12 PHP
php文件扩展名判断及获取文件扩展名的N种方法
2015/09/12 PHP
AES加解密在php接口请求过程中的应用示例
2016/10/26 PHP
php简单中奖算法(实例)
2017/08/15 PHP
PHP7 echo和print语句实例用法
2019/02/15 PHP
js 鼠标拖动对象 可让任何div实现拖动效果
2009/11/09 Javascript
javascript中字符串拼接需注意的问题
2010/07/13 Javascript
js异步加载的三种解决方案
2013/03/04 Javascript
使用ajaxfileupload.js实现ajax上传文件php版
2014/06/26 Javascript
原生js制作日历控件实例分享
2016/04/06 Javascript
angularjs实现下拉列表的选中事件示例
2017/03/03 Javascript
JavaScript运动框架 解决速度正负取整问题(一)
2017/05/17 Javascript
详述 Sublime Text 打开 GBK 格式中文乱码的解决方法
2017/10/26 Javascript
Vue3.0结合bootstrap创建多页面应用
2019/05/28 Javascript
vue从零实现一个消息通知组件的方法详解
2020/03/16 Javascript
在Django中创建动态视图的教程
2015/07/15 Python
Golang与python线程详解及简单实例
2017/04/27 Python
Python编程实现二叉树及七种遍历方法详解
2017/06/02 Python
Python3.6日志Logging模块简单用法示例
2018/06/14 Python
numpy返回array中元素的index方法
2018/06/27 Python
python爬虫 爬取超清壁纸代码实例
2019/08/16 Python
python计算波峰波谷值的方法(极值点)
2020/02/18 Python
Python实现电视里的5毛特效实例代码详解
2020/05/15 Python
python 调整图片亮度的示例
2020/12/03 Python
加拿大最大的书店:Indigo
2017/01/01 全球购物
什么是View State?
2013/01/27 面试题
自学考试自我鉴定范文
2013/09/26 职场文书
工商治理实习生的自我评价分享
2014/02/20 职场文书
《蜗牛的奖杯》教后反思
2014/04/24 职场文书
大学自主招生推荐信
2014/05/10 职场文书
安全责任书模板
2014/07/22 职场文书
2015出纳试用期工作总结
2014/12/12 职场文书
36个正则表达式(开发效率提高80%)
2021/11/17 Javascript