我正在编写一个python脚本,以在触发输入时在树莓派上显示图像,播放音乐和显示视频。当我从事这个项目时,我决定要在脚本中添加一个网络摄像头。我与不同的网络摄像头一起玩耍,直到发现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()
答案 0 :(得分:0)
好的,所以这里有一些关键问题。
一个大问题是您的try/except
声明。
您的尝试除外将总是做pass
。永远不会有您可以destroy()
然后在其上调用update()
的时候。这将始终导致错误,因此pass
的except语句将运行。
下一个root = tkinter.Toplevel()
是一个问题。因为您从未定义tkinter实例或应该root
的内容,所以您将创建一个顶级窗口,这将导致打开tkinter实例,但没有可使用的变量名。此处所说的根仅在函数的本地定义,因此,每当再次调用该函数时,就没有根的记录可供该函数尝试销毁,因为据该函数所知尚未创建根。您将需要将root定义为global
变量,诸如此类。
即使这对您有用,也应该有两个问题。一个是显示一个额外的空白窗口,另一个是当您关闭顶层窗口时该窗口没有自动关闭。
接下来,您尝试在运行tkinter实例的同时使用sleep
。这些东西在不使用线程的情况下是不兼容的,因此您需要将线程工作到此线程中,或者最好学习如何使用after()
。 After()
是tkinter如何管理定时事件。
您需要在这里使用Toplevel()
,而不是在这里使用Tk()
。不必破坏和重建GUI的每个更新,而应仅更新窗口。继续尝试的工作范围,您可能应该在尝试使用树莓派项目之前,先使用tkinter多玩一些,并了解事件管理器的工作方式。一旦掌握了Tkinter GUI及其基于事件的流程,您就可以更轻松地完成更复杂的事情。