简单又快速:实现视频”剧情连拍(剧情截图)”

举报
HBStream 发表于 2018/12/17 16:38:01 2018/12/17
【摘要】   用过QQ影音(或其他类似播放器)的可能都知道,QQ影音有一个功能叫“剧情连拍”,可以对一部影片在不同时段进行截图,然后把这些截图放在一张并生成单独的图片。通过剧情图,可以一目了然的看到整部影片的画面风格,也就能大致猜出这部影片的类型,再加上字幕,甚至可以大概猜出影片的故事梗概。本人前些日子针对这个功能做了一些研究,小有成果,先上两张效果图:1. 2.   怎么样,效果还可以吧?要实现这种...

  用过QQ影音(或其他类似播放器)的可能都知道,QQ影音有一个功能叫“剧情连拍”,可以对一部影片在不同时段进行截图,然后把这些截图放在一张并生成单独的图片。通过剧情图,可以一目了然的看到整部影片的画面风格,也就能大致猜出这部影片的类型,再加上字幕,甚至可以大概猜出影片的故事梗概。本人前些日子针对这个功能做了一些研究,小有成果,先上两张效果图:

1. 

2011062813180490.jpg

2. 

2011062813183144.jpg

  怎么样,效果还可以吧?要实现这种程序需要完成以下几种功能

  1. 实现从视频中截图的功能

  2. 分析文件格式,读取视频宽高和时长

  3. 把所有截图绘制在一张图像中

  下面一一道来:

1. 实现从视频中截图的功能

  首先第一个,也是最关键的一点,实现视频截图。对于截图这种需要解码视频的功能,我们最好借助于现有程序,著名的开源项目ffmpeg便可完成此功能,还有鼎鼎大名的开源播放器MPlayer,也提供的截图的命令接口,只需要简单的调用即可,两个程序生成的截图质量差不多,效率上却有些不同。

ffmpeg的调用命令为:

ffmpeg -i input.rmvb -y -f image2 -ss 08.010 -t 0.001 -s 352x240 output.jpg

  其中,-ss指定了截图的起始时间

  当然,这只能截取一张图片,我们要的是剧情连拍啊!不用着急,虽然没有现成的,我们可以自己来做,只要读取到影片的时长,我们就能够计算出需要在哪些时间来截图,这样,通过均匀分布截图时间,多次调用截图程序,就能生成一连串的截图,也就是剧情了。那怎么才能读取影片时长呢,来看第二步。

2. 分析文件格式,读取视频宽高和时长

  读取视频宽高和时长,需要解析视频格式。如今,媒体容器格式繁多复杂,要想准确的判断一个视频文件的格式和编码,不是件容易的事情。还好,我们有伟大的开源项目,用很多开源软件都可以读取出视频格式,这里推荐MediaInfo,因为它似乎就是为这个目的而建立的,生成的属性信息详细而准确,正是我们想要的。此外,MediaInfo还提供了一个动态链接库"MediaInfo.dll"。

2011062700411578.jpg

  很明显,MediaInfo告诉了你,这个影片的时长(Duration),宽度(Width)和高度(Height),当然还有其他你想要或不想要的信息,够了。

3. 把所有截图绘制在一张图像中

  这个没什么好说的,只要前两步做出来了,这一步基本不是什么问题,根据视频的宽和高,以及最终结果图片的宽和高,计算出你需要对截图进行缩放的比例,然后把图像绘制在一起,生成新的图像即可。下面是用 Qt 实现这一步的代码:

1   int headerheight = 70;  // 顶上留一部分高度用了绘制logo和文件信息
2     int interval = 4;
3   int column_count = basecolumn_;
4    int total_width = column_count * one_img_width + (column_count+1) * interval;
5    int row_count = (imglist.size() + column_count-1 ) / column_count;
6    int total_height = row_count * one_img_height + (row_count+1) * interval + headerheight;
7
8    int row_no = 0;
9    int col_no = 0;
10    QImage totalImage(total_width, total_height, QImage::Format_ARGB32_Premultiplied);
11    totalImage.fill(qRgb(255, 255, 255));
12    QPainter imagePainter(&totalImage);
13    imagePainter.setRenderHint(QPainter::Antialiasing, true);
14    imagePainter.setPen(QPen(Qt::black, 1, Qt::SolidLine/*, Qt::RoundCap, Qt::RoundJoin*/));
15    imagePainter.setFont(QFont(tr("楷体"), 13, QFont::Bold));
16    QString titlename(tr("影片名:"));
17    titlename.append(QString::fromStdString(cinfo.title.c_str()));
18    titlename.append(tr("\t格式:"));
19    titlename.append(QString::fromStdString(cinfo.format_name));
20    imagePainter.drawText(150, 30, titlename);
21    std::ostringstream oss;
22    oss << "文件大小:" << cinfo.size / 1000000 << "MB\t视频尺寸:"
23        << cinfo.width << "x" << cinfo.height << "\t视频时长:";
24    QString info = QString::fromStdString(oss.str());
25    info.append(TimetoStr(cinfo.duration));
26    imagePainter.drawText(150, 55, info);
27
28
29    for (list<QImage>::const_iterator it = imglist.begin();
30        it != imglist.end(); ++it)
31    {
32        int img_x = col_no * one_img_width + (col_no+1) * interval;
33        int img_y = row_no * one_img_height + (row_no+1) * interval + headerheight;
34
35        imagePainter.drawImage(img_x, img_y, *it);
36
37        col_no++;
38        if ( (col_no % column_count) == 0 )
39        {
40            row_no ++;
41            col_no = 0;
42        }
43    }
44
45    imagePainter.end();
46    totalImage.save(QString::fromStdString(outname));


  这样,模仿QQ影音剧情连拍的功能就实现了,然后呢,就可以在此基础上做其他的应用了。我是用Qt开发的,当然,很简单就能转换成其他的语言和类库来实现。


main-1.jpg合作请联系QQ。(转载请注明作者和出处~)


【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。