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实现简单温度转换的方法
Mar 13 Python
Java Web开发过程中登陆模块的验证码的实现方式总结
May 25 Python
Python制作词云的方法
Jan 03 Python
Python + selenium自动化环境搭建的完整步骤
May 19 Python
解决python3.5 正常安装 却不能直接使用Tkinter包的问题
Feb 22 Python
python将邻接矩阵输出成图的实现
Nov 21 Python
PyTorch的自适应池化Adaptive Pooling实例
Jan 03 Python
python 如何快速复制序列
Sep 07 Python
python空元组在all中返回结果详解
Dec 15 Python
python向xls写入数据(包括合并,边框,对齐,列宽)
Feb 02 Python
python程序的组织结构详解
Dec 06 Python
Python如何快速找到多个字典中的公共键(key)
Apr 29 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根据IP地址获取所在城市具体实现
2013/11/27 PHP
关于js和php对url编码的处理方法
2014/03/04 PHP
PHP实现的博客欢迎提示功能(很特别哦)
2014/06/05 PHP
PHP给源代码加密的几种方法汇总(推荐)
2018/02/06 PHP
javascript禁用Tab键脚本实例
2013/11/22 Javascript
基于jQuery实现最基本的淡入淡出效果实例
2015/02/02 Javascript
jQuery插件boxScroll实现图片轮播特效
2015/07/14 Javascript
不同js异步函数同步的实现方法
2016/05/28 Javascript
精彩的Bootstrap案例分享 重点在注释!(选项卡、栅格布局)
2016/07/01 Javascript
使用jQuery调用XML实现无刷新即时聊天
2016/08/07 Javascript
javascript 中Cookie读、写与删除操作
2017/03/29 Javascript
vue实现动态数据绑定
2017/04/28 Javascript
解决Vue2.0 watch对象属性变化监听不到的问题
2018/09/11 Javascript
详解vue 兼容IE报错解决方案
2018/12/29 Javascript
关于Layui Table隐藏列问题
2019/09/16 Javascript
解决vue-cli@3.xx安装不成功的问题及搭建ts-vue项目
2020/02/09 Javascript
[03:55]TI9战队采访——TNC Predator
2019/08/22 DOTA
python利用elaphe制作二维条形码实现代码
2012/05/25 Python
python文件比较示例分享
2014/01/10 Python
Python编写简单的HTML页面合并脚本
2016/07/11 Python
python list排序的两种方法及实例讲解
2017/03/20 Python
Python使用random.shuffle()打乱列表顺序的方法
2018/11/08 Python
Python爬虫文件下载图文教程
2018/12/23 Python
Python中的heapq模块源码详析
2019/01/08 Python
selenium处理元素定位点击无效问题
2019/06/12 Python
django使用xadmin的全局配置详解
2019/11/15 Python
python GUI库图形界面开发之PyQt5多线程中信号与槽的详细使用方法与实例
2020/03/08 Python
jupyter notebook oepncv 显示一张图像的实现
2020/04/24 Python
使用OpenCV获取图片连通域数量,并用不同颜色标记函
2020/06/04 Python
python怎么自定义捕获错误
2020/06/29 Python
这段代码难道不该打印出56吗
2013/02/27 面试题
文明美德伴我成长演讲稿
2014/05/12 职场文书
2015年销售工作总结范文
2015/03/30 职场文书
2015年秋季校长开学典礼致辞
2015/07/29 职场文书
MySQL数据库超时设置配置的方法实例
2021/10/15 MySQL
JMeter对MySQL数据库进行压力测试的实现步骤
2022/01/22 MySQL