我正在编写一个python脚本,用于在触发输入时在树莓pi上显示图像、播放音乐和显示视频。当我在做这个项目时,我决定在脚本中添加一个网络摄像头。我玩了不同的网络摄像头拖缆,直到我找到了帧速率最快、使用资源最少的MPlayer。问题是,MPlayer gui隐藏在我用来显示图像的tkinter窗口后面。我尝试了几种不同的方法来将MPlayer窗口移到前面,并使tkinter窗口消失,但似乎都不起作用。这是我的代码:
import sys
import os
import time
import subprocess
import RPi.GPIO as GPIO
if sys.version_info[0] == 2:
import Tkinter
tkinter = Tkinter
else:
import tkinter
from PIL import Image, ImageTk
import board
import neopixel
x=1
GPIO.setmode(GPIO.BCM)
pixels = neopixel.NeoPixel(board.D10, 38)
pixels.fill((0, 0, 0))
GPIO.setwarnings(False)
GPIO.setup(17,GPIO.OUT)
GPIO.setup(18,GPIO.OUT)
GPIO.setup(27,GPIO.OUT)
GPIO.setup(22,GPIO.OUT)
GPIO.setup(24,GPIO.OUT)
GPIO.setup(25,GPIO.OUT)
GPIO.setup(23, GPIO.IN, pull_up_down=GPIO.PUD_DOWN)
GPIO.setup(4, GPIO.IN, pull_up_down=GPIO.PUD_DOWN)
GPIO.setup(5, GPIO.IN, pull_up_down=GPIO.PUD_DOWN)
GPIO.output(17,GPIO.LOW)
GPIO.output(18,GPIO.LOW)
GPIO.output(27,GPIO.LOW)
GPIO.output(22,GPIO.LOW)
GPIO.output(24,GPIO.LOW)
GPIO.output(25,GPIO.LOW)
def showPIL(pilImage, exVar = 0):
try:
root.withdraw()
root.destroy()
root.update()
except:
pass
root = tkinter.Toplevel()
if x == 1:
w, h = root.winfo_screenwidth(), root.winfo_screenheight()
else:
w, h = 100, 100
root.overrideredirect(1)
root.geometry("%dx%d+0+0" % (w, h))
#root.focus_set()
root.bind("<Escape>", lambda e: (e.widget.withdraw(), e.widget.quit()))
canvas = tkinter.Canvas(root,width=w,height=h)
canvas.pack()
canvas.configure(background='black')
imgWidth, imgHeight = pilImage.size
if imgWidth > w or imgHeight > h:
ratio = min(w/imgWidth, h/imgHeight)
imgWidth = int(imgWidth*ratio)
imgHeight = int(imgHeight*ratio)
pilImage = pilImage.resize((imgWidth,imgHeight), Image.ANTIALIAS)
image = ImageTk.PhotoImage(pilImage)
imagesprite = canvas.create_image(w/2,h/2,image=image)
root.update()
showPIL(Image.open("Data/blank.png"))
while not GPIO.input(4):
pass
music = subprocess.Popen(['cvlc', '/home/pi/Desktop/Data/music.mp3'])
showPIL(Image.open("Data/trophy.png"))
time.sleep(1)
GPIO.output(22,GPIO.HIGH)
time.sleep(0.5)
GPIO.output(27,GPIO.HIGH)
time.sleep(0.5)
GPIO.output(18,GPIO.HIGH)
time.sleep(0.5)
GPIO.output(17,GPIO.HIGH)
time.sleep(1)
showPIL(Image.open("Data/poison.png"))
pixels.fill((0, 255, 0))
os.system("pkill tk")
x=0
showPIL(Image.open("Data/blank.png"))
x=1
camera = subprocess.Popen(['mplayer', '-fs', 'tv://'])
os.system("wmctrl -a MPlayer")
time.sleep(8)
camera.kill()
os.system("omxplayer -b '/home/pi/Desktop/Data/movie.mp4'")
showPIL(Image.open("Data/gun.png"))
GPIO.output(24,GPIO.HIGH)
GPIO.output(25,GPIO.HIGH)
while not GPIO.input(23):
pass
pixels.fill((0, 0, 0))
showPIL(Image.open("Data/dumbell.png"))
time.sleep(1)
showPIL(Image.open("Data/pipe.png"))
time.sleep(1)
showPIL(Image.open("Data/noose.png"))
time.sleep(1)
music.kill()
showPIL(Image.open("Data/blank.png"))
end = subprocess.Popen(['cvlc', '/home/pi/Desktop/Data/end.wav'])
time.sleep(8)
end.kill()
好的,这里有几个关键问题。
一个巨大的问题是你的try/except
语句。
你的尝试除了总是要做pass
。永远不会有destroy()
,然后调用它的update()
的时候。这总是会导致错误,因此pass
的except语句会运行。
下一个root = tkinter.Toplevel()
是个问题。因为您从未定义tkinter实例或root
应该是什么,所以您创建了一个顶层窗口,这将导致tkinter的实例被打开,但没有可使用的变量名。这里所说的根只是在函数的本地定义的,因此每当再次调用函数时,函数都没有根的记录,因为据函数所知,它还没有创建。您需要将根定义为global
变量,以实现类似的功能。
即使这对你有效,也应该有两个问题。一个是显示一个额外的空白窗口,另一个是关闭顶层窗口时该窗口不会自行关闭。
接下来,您将尝试在运行tkinter实例的同时使用sleep
。如果不使用线程,这些东西是不兼容的,所以你需要将线程处理到这个中,或者最好学习如何使用after()
。After()
是tkinter管理定时事件的方式。
您需要使用Tk()
,而不是在此处使用Toplevel()
。与其破坏和重建你的GUI每次更新,你应该只是更新你的窗口。在尝试树莓派项目之前,根据你试图做的事情的范围,你可能应该多玩一点tkinter,并了解事件管理器是如何工作的。一旦你掌握了Tkinter GUI及其基于事件的流程,你就可以更容易地完成更复杂的事情。