将动画gif转换为linux服务器上的视频,同时保留帧速率

时间:2015-04-10 12:07:57

标签: linux video ffmpeg video-processing avconv

如何在linux服务器上以编程方式将动画gif转换为视频(例如h264 @ mp4)?

我需要这个来处理用户生成的内容,这些内容应该作为几种定义的视频格式输出;因此,用户可能想要处理动画gif文件。我已经有了一套工作的PHP脚本,使用avconv将视频文件转码为特定格式(如vpx @ webm和h264 @ mp4,缩放到特定分辨率),但我需要视频输入。

通常的方法似乎是提取gif的帧然后对其进行编码,如

convert file.gif file%03d.png 
avconv -i file%03d.png file.mp4

但这会丢弃帧速率,由gif文件中的暂停信息决定。可以使用-r定义avconv的帧速率,但

  • 这不尊重帧之间的暂停,因为它们可以不同(如第一帧100ms暂停,第二帧250ms暂停,第3帧100ms暂停,......)
  • 由于输入来自用户,它甚至可能会有所不同,因为一些GIF可能有5fps而其他的是30fps

我注意到avconv能够自己处理gif,因此可能会尊重正确的暂停,但是当我这样做时(如How to convert GIF to Mp4 is it possible?中所描述的那样)

avconv -i file.gif -r 30 file.mp4

avconv只会占用gif的第一帧,而它至少会将视频检测为视频:

Duration: 00:00:00.04, start: 0.000000, bitrate: N/A
  Stream #0.0: Video: gif, pal8, 640x480, 25 tbn

(例如gif' file.gif'有15帧,每帧有100ms暂停=> 1.5s持续时间,循环)

  • 我错过了什么?怎么回事?
  • 这个用例可能有更好的工具吗?
  • 什么是大型网站,例如9gag用于将上传的GIF转码为视频吗?

2 个答案:

答案 0 :(得分:10)

又一个Avconv Bug(YAAB)

ffmpeg有更好的GIF解复用支持(和improved GIF encoding)。我建议放弃avconv并获得ffmpeg(来自FFmpeg的真实的;不是来自Libav的旧骗子)。 static build很简单,当然也可以compile

实施例

ffmpeg -i in.gif -c:v libx264 -pix_fmt yuv420p -movflags +faststart out.mp4

有关更多示例,请参阅FFmpeg Wiki: H.264 Encoding Guide

答案 1 :(得分:1)

如果由于某种原因你需要使用avconv和imagemagick,你可能想尝试这样的事情:

ticks_per_frame = subprocess.check_output('identify -verbose -format %T_ {0}'.format(gif_path).split()).split('_')[:-1]
ticks_per_frame = [int(i) for i in ticks_per_frame]
num_frames = len(ticks_per_frame)
min_ticks = min(ticks_per_frame)

subprocess.call('convert -coalesce {0} tmp%d.png'.format(gif_path).split())

if len(set(ticks_per_frame)) > 1:
    num_dup = 0
    num_dup_total = 0
    for frame, ticks in enumerate(ticks_per_frame):
        num_dup_total += num_dup
        frame += num_dup_total
        num_dup = 0
        if ticks > min_ticks:
            num_dup = (ticks / min_ticks) - 1
            for i in range(num_frames + num_dup_total - 1, frame, -1):
                orig = 'tmp%d.png' % i
                new = 'tmp%d.png' % (i + num_dup)
                subprocess.call(['mv', orig, new])
            for i in range(1, num_dup + 1):
                curr = 'tmp%d.png' % frame
                dup = 'tmp%d.png' % (i + frame)
                subprocess.call(['cp', curr, dup])

framerate = (100 / min_ticks) if min_ticks else 10

subprocess.call('avconv -r {0} -i tmp%d.png -c:v libx264 -crf {1} -pix_fmt yuv420p \
-vf scale=trunc(iw/2)*2:trunc(ih/2)*2 -y {2}.mp4'.format(framerate, quality, STORAGE_DIR + mp4_name).split())

subprocess.call(['rm'] + glob('tmp*.png'))

因此,为gif的每一帧(通过识别)获取以厘秒为单位的刻度,转换为多个png,然后在根据刻度值进行重复时通过它们。你不用担心,png文件仍将按连续顺序保留。使用真正的FFmpeg仍然是最好的方式。