python tkinter實(shí)現(xiàn)彩球碰撞屏保
利用Tkinter實(shí)現(xiàn)彩球碰撞屏保,供大家參考,具體內(nèi)容如下
一、架構(gòu)與思路
(1)主函數(shù):
main():通過類啟動程序;
(2)類:
ScreenSaver():用于定義屏保和主畫布,調(diào)用球創(chuàng)建、運(yùn)動等函數(shù);
RandomBall():定義球的基本屬性、球創(chuàng)建與運(yùn)動函數(shù);
(3)對象:單個(gè)球,需要?jiǎng)?chuàng)建、運(yùn)動(包括碰撞反彈),通過循環(huán)調(diào)用實(shí)現(xiàn)多個(gè)球并存的效果
create_ball():單個(gè)球創(chuàng)建函數(shù);
move_ball():單個(gè)球運(yùn)動函數(shù);
(4)20181215更新:此處對原有屏保程序的退出環(huán)節(jié)進(jìn)行了擴(kuò)展,使用messabox工具建立消息框,詢問是否退出,點(diǎn)擊“確定”會直接退出,點(diǎn)擊“取消”仍留在程序中。
程序架構(gòu)和思路如下:

二、代碼實(shí)現(xiàn)
根據(jù)上述思路,利用python實(shí)現(xiàn)屏保程序,代碼如下:
import random
import tkinter
import tkinter.messagebox
class RandomBall():
'''
單個(gè)球定義、運(yùn)動的類
'''
def __init__(self, root_canvas, width, height):
'''
參數(shù)說明:
canvas:從ScreenSaver類中傳入的畫布
width,height:從SS類中傳入的寬高,即屏幕寬高
'''
# 將傳入變量賦為RB類的屬性
self.canvas = root_canvas
self.width = width
self.height = height
# 隨機(jī)生成球的中心坐標(biāo)
self.xcenter = random.randint(100, width-100)
self.ycenter = random.randint(100, height-100)
# 隨機(jī)生成球的運(yùn)動速度
self.xvelocity = random.randint(8,16)
self.yvelocity = random.randint(8,16)
# 計(jì)算球的半徑
self.radius = random.randint(60, 100)
# 利用十六進(jìn)制隨機(jī)數(shù)與lambda表達(dá)式生成球的顏色
# RGB表示法:三個(gè)數(shù)字,每個(gè)數(shù)字的值是0-255之間,表示紅綠藍(lán)三個(gè)顏色的大小
# 在某些系統(tǒng)中,直接用英文單詞表示也可以,比如red,green
color = lambda : random.randint(0,255)
self.color = '#%02x%02x%02x' % (color(),color(),color())
# 創(chuàng)建球的函數(shù)
def create_ball(self):
'''
用構(gòu)造函數(shù)定義的變量值,在canvas上畫一個(gè)球
'''
# tkinter沒有畫圓形函數(shù)
# 只有一個(gè)畫橢圓函數(shù),畫橢圓需要定義兩個(gè)坐標(biāo),
# 在一個(gè)長方形內(nèi)畫橢圓,我們只需要定義長方形左上角和右下角就好
# 求兩個(gè)坐標(biāo)的方法是,已知圓心的坐標(biāo),則圓心坐標(biāo)減去半徑能求出
# 左上角坐標(biāo),加上半徑能求出右下角坐標(biāo)(向右x為正,向下y為正)
xleftup = self.xcenter - self.radius
yleftup = self.ycenter - self.radius
xrightdown = self.xcenter + self.radius
yrightdown = self.ycenter + self.radius
# 創(chuàng)建球
self.item = self.canvas.create_oval(xleftup,yleftup,
xrightdown,yrightdown,
fill=self.color,
outline=self.color)
# 球運(yùn)動的函數(shù)
def move_ball(self):
# 計(jì)算球移動后的中心點(diǎn)坐標(biāo)
self.xcenter += self.xvelocity
self.ycenter += self.yvelocity
# 當(dāng)球與邊框發(fā)生碰撞時(shí),需要進(jìn)行回彈操作,即對應(yīng)方向的速度取負(fù)
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已經(jīng)調(diào)用
self.canvas.move(self.item, self.xvelocity, self.yvelocity)
class ScreenSaver():
'''
屏保定義類
程序啟動
'''
def __init__(self):
# 創(chuàng)建球存儲列表
self.balls = []
# 隨機(jī)生成球的數(shù)量
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)
# 綁定退出函數(shù)與相應(yīng)動作
self.root.bind('<Motion>', self.myquit)
self.root.bind('<Key>', self.myquit)
self.root.bind('<Any-Button>', self.myquit)
# 創(chuàng)建畫布,配置尺寸與顏色屬性
self.canvas = tkinter.Canvas(self.root, width=root_w, height=root_h)
self.canvas.pack()
# 利用循環(huán)與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)
# 調(diào)用球運(yùn)動函數(shù)
self.run_screen_saver()
# 啟用tkinter時(shí)間消息循環(huán)mainloop
self.root.mainloop()
# 球運(yùn)動函數(shù)
def run_screen_saver(self):
# 循環(huán)實(shí)例化的ball調(diào)用move_ball函數(shù)
for ball in self.balls:
ball.move_ball()
# 使用after實(shí)現(xiàn)遞歸,通過不斷調(diào)用各球的move_ball函數(shù),實(shí)現(xiàn)位置刷新
self.root.after(50, self.run_screen_saver)
# 停止運(yùn)行
# 此處e只是利用了事件處理機(jī)制,際上并不關(guān)心事件的類型
def myquit(self, e):
# 擴(kuò)展:
# 此屏保程序擴(kuò)展成,一旦捕獲事件,則判斷屏保不退出
# 顯示一個(gè)Button,Button上顯示事件類型,點(diǎn)擊Button后屏保才退出
if tkinter.messagebox.askokcancel("彩球碰撞", '確定退出?'):
self.root.destroy()
else:
pass
if __name__ == '__main__':
# 啟動屏保
ScreenSaver()
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
解決pytorch下只打印tensor的數(shù)值不打印出device等信息的問題
這篇文章主要介紹了解決pytorch下只打印tensor的數(shù)值不打印出device等信息的問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-05-05
Tensorflow中的降維函數(shù)tf.reduce_*使用總結(jié)
這篇文章主要介紹了Tensorflow中的降維函數(shù)tf.reduce_*使用總結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
Python Flask基礎(chǔ)到登錄功能的實(shí)現(xiàn)代碼
這篇文章主要介紹了Python Flask基礎(chǔ)到登錄功能的實(shí)現(xiàn)代碼,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-05-05
基于python select.select模塊通信的實(shí)例講解
下面小編就為大家?guī)硪黄趐ython select.select模塊通信的實(shí)例講解。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-09-09
python?字符串模糊匹配Fuzzywuzzy的實(shí)現(xiàn)
本文主要介紹了python?字符串模糊匹配Fuzzywuzzy的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07
Python內(nèi)置數(shù)據(jù)類型list各方法的性能測試過程解析
這篇文章主要介紹了Python內(nèi)置數(shù)據(jù)類型list各方法的性能測試過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01

