Je crée une interface graphique pour lire des fichiers vidéo. Le problème est que lorsque je mets en pause la vidéo, le bouton Lecture ne peut pas relire cette vidéo et je dois à nouveau sélectionner le fichier vidéo.
Remarque: puisque je souhaite afficher la vidéo dans la même fenêtre tkinter, je n'utilise pas la commande imshow d'OpenCV. Au lieu de cela, j'utilise la méthode "window.after".
Voici mon code:
J'essaie d'utiliser la variable "self.pause" pour contrôler l'état de la pause. Lorsque je clique sur le bouton Pause, cette variable booléenne devient True. Cependant, je n'ai pas trouvé d'endroit approprié pour le rendre faux lorsque je clique à nouveau sur le bouton Lecture.
self.pause = False
Si j'écris le code suivant dans la fonction "play_video":
from tkinter import *
from tkinter import messagebox
from tkinter import filedialog
import PIL.Image, PIL.ImageTk
import cv2
class videoGUI:
def __init__(self, window, window_title):
self.window = window
self.window.title(window_title)
top_frame = Frame(self.window)
top_frame.pack(side=TOP, pady=5)
bottom_frame = Frame(self.window)
bottom_frame.pack(side=BOTTOM, pady=5)
self.pause = False # Parameter that controls pause button
self.canvas = Canvas(top_frame)
self.canvas.pack()
# Select Button
self.btn_select=Button(bottom_frame, text="Select video file", width=15, command=self.open_file)
self.btn_select.grid(row=0, column=0)
# Play Button
self.btn_play=Button(bottom_frame, text="Play", width=15, command=self.play_video)
self.btn_play.grid(row=0, column=1)
# Pause Button
self.btn_pause=Button(bottom_frame, text="Pause", width=15, command=self.pause_video)
self.btn_pause.grid(row=0, column=2)
self.delay = 15 # ms
self.window.mainloop()
def open_file(self):
self.pause = False
self.filename = filedialog.askopenfilename(title="Select file", filetypes=(("MP4 files", "*.mp4"),
("WMV files", "*.wmv"), ("AVI files", "*.avi")))
print(self.filename)
# Open the video file
self.cap = cv2.VideoCapture(self.filename)
self.width = self.cap.get(cv2.CAP_PROP_FRAME_WIDTH)
self.height = self.cap.get(cv2.CAP_PROP_FRAME_HEIGHT)
self.canvas.config(width = self.width, height = self.height)
def get_frame(self): # get only one frame
try:
if self.cap.isOpened():
ret, frame = self.cap.read()
return (ret, cv2.cvtColor(frame, cv2.COLOR_BGR2RGB))
except:
messagebox.showerror(title='Video file not found', message='Please select a video file.')
def play_video(self):
# Get a frame from the video source, and go to the next frame automatically
ret, frame = self.get_frame()
if ret:
self.photo = PIL.ImageTk.PhotoImage(image = PIL.Image.fromarray(frame))
self.canvas.create_image(0, 0, image = self.photo, anchor = NW)
if not self.pause:
self.window.after(self.delay, self.play_video)
def pause_video(self):
self.pause = True
# Release the video source when the object is destroyed
def __del__(self):
if self.cap.isOpened():
self.cap.release()
##### End Class #####
# Create a window and pass it to videoGUI Class
videoGUI(Tk(), "EnJapan")
Le bouton Pause ne fonctionnera pas. Parce que la méthode "window.after" appelle automatiquement la fonction "play_video" et rend le "self.pause" faux. Par conséquent, le bouton Pause n'aura aucun effet.
4 Réponses :
Je créerais une autre méthode pour le rappel du bouton de lecture. Quelque chose comme ceci:
def play_start(self):
self.pause = False
self.play_video()
Cependant, je veillerais à désactiver le bouton de lecture s'il est déjà en cours de lecture. Sinon, vous pourriez avoir plusieurs "instances" de play_video si le bouton de lecture est enfoncé plusieurs fois.
Une alternative est de combiner vos boutons de lecture et de pause, donc il bascule la valeur de self.pause . Ensuite, vous pourriez avoir un seul bouton avec une fonction de rappel.
Question : le bouton Pause n'aura aucun effet.
< gagnantRéférence
Méthode Tkinter.Widget.after -
après (delay_ms, callback = None, * args)Enregistre un rappel qui est appelé après un temps donné.
Méthode Tkinter.Widget.after_cancel -
after_cancel (id)Annule un rappel.
Pour
< pre> XXXannulerlesévénementsdéjà mis en file d'attente pourself.play_video, modifiez ce qui suit:
@stovfi que signifie «self.after_id»?
@Ekin "Que signifie self.after_id ?" : J'ai mis à jour ma réponse avec un lien de référence. after_id est l'identifiant renvoyé par .after (... .
Ajoutez simplement la méthode play_video (self):
if self.pause == True:
self.pause = False
return
from tkinter import *
from tkinter import messagebox
from tkinter import filedialog
import PIL.Image, PIL.ImageTk
import cv2
class videoGUI:
def __init__(self, window, window_title):
self.window = window
self.window.title(window_title)
top_frame = Frame(self.window)
top_frame.pack(side=TOP, pady=5)
bottom_frame = Frame(self.window)
bottom_frame.pack(side=BOTTOM, pady=5)
self.pause = False # Parameter that controls pause button
self.canvas = Canvas(top_frame)
self.canvas.pack()
# Select Button
self.btn_select=Button(bottom_frame, text="Select video file", width=15, command=self.open_file)
self.btn_select.grid(row=0, column=0)
# Play Button
self.btn_play=Button(bottom_frame, text="Play", width=15, command=self.play_video)
self.btn_play.grid(row=0, column=1)
# Pause Button
self.btn_pause=Button(bottom_frame, text="Pause", width=15, command=self.pause_video)
self.btn_pause.grid(row=0, column=2)
# Resume Button
self.btn_resume=Button(bottom_frame, text="resume", width=15, command=self.resume_video)
self.btn_resume.grid(row=0, column=3)
self.delay = 15 # ms
self.window.mainloop()
def open_file(self):
self.pause = False
self.filename = filedialog.askopenfilename(title="Select file", filetypes=(("MP4 files", "*.mp4"),
("WMV files", "*.wmv"), ("AVI files", "*.avi")))
print(self.filename)
# Open the video file
self.cap = cv2.VideoCapture(self.filename)
self.width = self.cap.get(cv2.CAP_PROP_FRAME_WIDTH)
self.height = self.cap.get(cv2.CAP_PROP_FRAME_HEIGHT)
self.canvas.config(width = self.width, height = self.height)
def get_frame(self): # get only one frame
try:
if self.cap.isOpened():
ret, frame = self.cap.read()
return (ret, cv2.cvtColor(frame, cv2.COLOR_BGR2RGB))
except:
messagebox.showerror(title='Video file not found', message='Please select a video file.')
def play_video(self):
# Get a frame from the video source, and go to the next frame automatically
ret, frame = self.get_frame()
if ret:
self.photo = PIL.ImageTk.PhotoImage(image = PIL.Image.fromarray(frame))
self.canvas.create_image(0, 0, image = self.photo, anchor = NW)
if not self.pause:
self.window.after(self.delay, self.play_video)
def pause_video(self):
self.pause = True
#Addition
def resume_video(self):
self.pause=False
self.play_video()
# Release the video source when the object is destroyed
def __del__(self):
if self.cap.isOpened():
self.cap.release()
##### End Class #####
# Create a window and pass it to videoGUI Class
videoGUI(Tk(), "EnJapan")
Utilisez
self.after_id = self.window.after (...etself.after_cancel (self.after_id)Pourriez-vous me dire où dans le code je devrais l'utiliser?