4
votes

Lecteur vidéo par Python tkinter, lorsque je mets en pause la vidéo, je ne peux pas la relire

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.


2 commentaires

Utilisez self.after_id = self.window.after (... et self.after_cancel (self.after_id)


Pourriez-vous me dire où dans le code je devrais l'utiliser?


4 Réponses :


0
votes

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.


0 commentaires

3
votes

Question : le bouton Pause n'aura aucun effet.

< gagnantRéférence


Pour annuler les événements déjà mis en file d'attente pour self.play_video , modifiez ce qui suit:

< pre> XXX

2 commentaires

@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 (... .



0
votes

Ajoutez simplement la méthode play_video (self):

if self.pause == True:
    self.pause = False
    return
 


0 commentaires

-2
votes
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")

0 commentaires