Skip to content

Commit

Permalink
Fix stretched vertical video, add ffmpeg_params to VideoFileClip
Browse files Browse the repository at this point in the history
- FFMPEG_VideoReader: set size based on video rotation metadata
- use `-noautorotate` to disable autorotation (ffmpeg >=2.7 autorotates by default)
- closes Zulko#212, Zulko#577, Zulko#682
  • Loading branch information
cvn committed Dec 10, 2017
1 parent f444f8c commit a562c01
Show file tree
Hide file tree
Showing 5 changed files with 50 additions and 8 deletions.
12 changes: 10 additions & 2 deletions moviepy/video/io/VideoFileClip.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,10 @@ class VideoFileClip(VideoClip):
can be set to 'fps', which may be helpful if importing slow-motion videos
that get messed up otherwise.
ffmpeg_params:
Any additional ffmpeg parameters to pass when reading files, as a list of
terms, like ['-noautorotate'].
Attributes
-----------
Expand All @@ -60,6 +64,9 @@ class VideoFileClip(VideoClip):
fps:
Frames per second in the original file.
rotation:
Rotation metadata, in degrees.
Read docs for Clip() and VideoClip() for other, more generic, attributes.
Expand All @@ -78,7 +85,7 @@ def __init__(self, filename, has_mask=False,
audio=True, audio_buffersize = 200000,
target_resolution=None, resize_algorithm='bicubic',
audio_fps=44100, audio_nbytes=2, verbose=False,
fps_source='tbr'):
fps_source='tbr', ffmpeg_params=None):

VideoClip.__init__(self)

Expand All @@ -87,7 +94,8 @@ def __init__(self, filename, has_mask=False,
self.reader = FFMPEG_VideoReader(filename, pix_fmt=pix_fmt,
target_resolution=target_resolution,
resize_algo=resize_algorithm,
fps_source=fps_source)
fps_source=fps_source,
ffmpeg_params=ffmpeg_params)

# Make some of the reader's attributes accessible from the clip
self.duration = self.reader.duration
Expand Down
13 changes: 11 additions & 2 deletions moviepy/video/io/ffmpeg_reader.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ class FFMPEG_VideoReader:
def __init__(self, filename, print_infos=False, bufsize = None,
pix_fmt="rgb24", check_duration=True,
target_resolution=None, resize_algo='bicubic',
fps_source='tbr'):
fps_source='tbr', ffmpeg_params=None):

self.filename = filename
self.proc = None
Expand All @@ -34,6 +34,12 @@ def __init__(self, filename, print_infos=False, bufsize = None,
self.fps = infos['video_fps']
self.size = infos['video_size']
self.rotation = infos['video_rotation']
self.ffmpeg_params = ffmpeg_params or []

autorotate = not '-noautorotate' in self.ffmpeg_params
if autorotate and abs(self.rotation) in [90, 270]:
# swap dimensions to preserve aspect ratio after autorotate
self.size = self.size[::-1]

if target_resolution:
# revert the order, as ffmpeg used (width, height)
Expand Down Expand Up @@ -86,6 +92,9 @@ def initialize(self, starttime=0):
else:
i_arg = [ '-i', self.filename]

if self.ffmpeg_params:
i_arg = self.ffmpeg_params + i_arg

cmd = ([get_setting("FFMPEG_BINARY")] + i_arg +
['-loglevel', 'error',
'-f', 'image2pipe',
Expand Down Expand Up @@ -231,7 +240,7 @@ def ffmpeg_parse_infos(filename, print_infos=False, check_duration=True,
Returns a dictionnary with the fields:
"video_found", "video_fps", "duration", "video_nframes",
"video_duration", "audio_found", "audio_fps"
"video_duration", "video_rotation", "audio_found", "audio_fps"
"video_duration" is slightly smaller than "duration" to avoid
fetching the uncomplete frames at the end, which raises an error.
Expand Down
3 changes: 2 additions & 1 deletion tests/download_media.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,8 @@ def download():
'/sounds/crunching.mp3', '/images/pigs_in_a_polka.gif',
'/videos/fire2.mp4', '/videos/big_buck_bunny_0_30.webm',
'/subtitles/subtitles1.srt', '/misc/traj.txt',
'/images/vacation_2017.jpg', '/images/python_logo_upside_down.png']
'/images/vacation_2017.jpg', '/images/python_logo_upside_down.png',
'/videos/ficus_vertical.mp4']

# Loop through download url strings, build out path, and download the asset.
for url in urls:
Expand Down
11 changes: 11 additions & 0 deletions tests/test_VideoFileClip.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,17 @@ def test_ffmpeg_resizing():
frame = video.get_frame(0)
assert frame.shape[1] == target_resolution[1]

def test_ffmpeg_resizing_with_autorotate():
video_file = 'media/ficus_vertical.mp4'
target_resolution=(720, None)
with VideoFileClip(video_file, target_resolution=target_resolution) as video:
frame = video.get_frame(0)
assert frame.shape[0:2] == (720, 405)

with VideoFileClip(video_file, target_resolution=target_resolution, ffmpeg_params=['-noautorotate']) as video:
frame = video.get_frame(0)
assert frame.shape[0:2] == (720, 1280)


if __name__ == '__main__':
pytest.main()
19 changes: 16 additions & 3 deletions tests/test_ffmpeg_reader.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,12 @@
import sys

import pytest
from moviepy.video.io.ffmpeg_reader import ffmpeg_parse_infos

import download_media
from moviepy.video.io.ffmpeg_reader import ffmpeg_parse_infos, FFMPEG_VideoReader

sys.path.append("tests")

import download_media


def test_download_media(capsys):
with capsys.disabled():
Expand All @@ -22,6 +22,19 @@ def test_ffmpeg_parse_infos():
assert d['video_size'] == [314, 273]
assert d['duration'] == 3.0

def test_FFMPEG_VideoReader_autorotate():
video_file = 'media/ficus_vertical.mp4'
reader = FFMPEG_VideoReader(video_file)
assert reader.infos['video_size'] == [1920, 1080]
assert reader.infos['video_rotation'] == 90
assert reader.size == [1080, 1920]
reader.close()

reader = FFMPEG_VideoReader(video_file, ffmpeg_params=['-noautorotate'])
assert reader.size == [1920, 1080]
assert reader.rotation == 90
reader.close()


if __name__ == '__main__':
pytest.main()

0 comments on commit a562c01

Please sign in to comment.