Использование перемотки в opencv - PullRequest
3 голосов
/ 29 января 2020

Можно ли перематывать назад, когда вы нажимаете назначенную клавишу в opencv? У меня есть ниже, но это не перематывает воспроизведение.

def run(self, filepath, fps, width, height, monochrome=False):

    video_file = self.read_file(filepath)
    previous_frame = None


    while (video_file.isOpened()):
        ret, frame = video_file.read()

        if previous_frame is not None:
            pass

        previous_frame = frame

        if ret:

            frame = self.color(frame, monochrome)

            final_frame = self.resolution(frame, width, height)

            delaytime = self.frame_per_second(fps)


            cv2.imshow('frame', final_frame)
            key = cv2.waitKey(delaytime)

            if key & 0xFF == ord("p"):
                cv2.waitKey(234320)
                if key & 0xFF == ord("r"):
                    cv2.set(cv2.CV_CAP_PROP_POS_FRAMES(2, previous_frame))


    video_file.release()
    cv2.destroyAllWindows()

Приведенный выше код переносит его в предыдущий кадр, но не воспроизводит.

Ответы [ 2 ]

3 голосов
/ 29 января 2020

Идеальным решением является установка номера кадра, который должен воспроизводиться при остановке видео для достижения эффекта перемотки. Это можно сделать с помощью свойств захвата видео :

cv2.VideoCapture.set (propId, value)

CAP_PROP_POS_FRAMES 0 на основе индекса кадра, который будет декодирован / захвачен следующим.

Полезные примеры доступны по адресу " Переход между кадрами в видеофайлах ".

Это пример исходного кода , который приостанавливает видео и предоставляет некоторый элемент управления, позволяющий перемотать назад к предыдущему кадру или перейти к нулевому кадру и перезапустить видео:

import cv2
import sys

# load input video
cap = cv2.VideoCapture('TheMandalorian.mkv')
if (cap.isOpened() == False):
    print("!!! Failed cap.isOpened()")
    sys.exit(-1)

# retrieve the total number of frames
frame_count = int(cap.get(cv2.CAP_PROP_FRAME_COUNT))

# loop to read every frame of the video
while (cap.isOpened()):

    # capture a frame
    ret, frame = cap.read()
    if ret == False:
        print("!!! Failed cap.read()")
        break

    cv2.imshow('video', frame)

    # check if 'p' was pressed and wait for a 'b' press
    key = cv2.waitKey(int(frame_count/1000))
    if (key & 0xFF == ord('p')):

        # sleep here until a valid key is pressed
        while (True):
            key = cv2.waitKey(0)

            # check if 'p' is pressed and resume playing
            if (key & 0xFF == ord('p')):
                break

            # check if 'b' is pressed and rewind video to the previous frame, but do not play
            if (key & 0xFF == ord('b')):
                cur_frame_number = cap.get(cv2.CAP_PROP_POS_FRAMES)
                print('* At frame #' + str(cur_frame_number))

                prev_frame = cur_frame_number
                if (cur_frame_number > 1):
                    prev_frame -= 1

                print('* Rewind to frame #' + str(prev_frame))
                cap.set(cv2.CAP_PROP_POS_FRAMES, prev_frame)

            # check if 'r' is pressed and rewind video to frame 0, then resume playing
            if (key & 0xFF == ord('r')):
                cap.set(cv2.CAP_PROP_POS_FRAMES, 0)
                break

    # exit when 'q' is pressed to quit
    elif (key & 0xFF == ord('q')):
        break

# release resources
cap.release()
cv2.destroyAllWindows()

Ключ нажмите опции:

  • Нажмите q для выхода из приложения.
  • Нажмите p для паузы.
  • Когда пауза, нажмите p еще раз для возобновления
  • Когда воспроизведение приостановлено, нажмите b, чтобы перемотать один кадр. Для возобновления воспроизведения необходимо нажать p.
  • Когда воспроизведение приостановлено, нажмите r, чтобы вернуться назад к кадру 0 и автоматически возобновить воспроизведение.
1 голос
/ 29 января 2020

Если вы просто хотите предыдущий кадр, его можно сохранить во временном массиве. Например:

prev_frame = curr_image.copy()
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...