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> XXXannuler
lesévénements
dé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?