用过QQ影音(或其他类似播放器)的可能都知道,QQ影音有一个功能叫“剧情连拍”,可以对一部影片在不同时段进行截图,然后把这些截图放在一张并生成单独的图片。通过剧情图,可以一目了然的看到整部影片的画面风格,也就能大致猜出这部影片的类型,再加上字幕,甚至可以大概猜出影片的故事梗概。本人前些日子针对这个功能做了一些研究,小有成果,先上两张效果图:
1.
2.
怎么样,效果还可以吧?要实现这种程序需要完成以下几种功能
- 实现从视频中截图的功能
- 分析文件格式,读取视频宽高和时长
- 把所有截图绘制在一张图像中
下面一一道来:
1. 实现从视频中截图的功能
首先第一个,也是最关键的一点,实现视频截图。对于截图这种需要解码视频的功能,我们最好借助于现有程序,著名的开源项目ffmpeg便可完成此功能,还有鼎鼎大名的开源播放器MPlayer,也提供的截图的命令接口,只需要简单的调用即可,两个程序生成的截图质量差不多,效率上却有些不同。
ffmpeg的调用命令为:
其中,-ss指定了截图的起始时间
当然,这只能截取一张图片,我们要的是剧情连拍啊!不用着急,虽然没有现成的,我们可以自己来做,只要读取到影片的时长,我们就能够计算出需要在哪些时间来截图,这样,通过均匀分布截图时间,多次调用截图程序,就能生成一连串的截图,也就是剧情了。那怎么才能读取影片时长呢,来看第二步。
2. 分析文件格式,读取视频宽高和时长
读取视频宽高和时长,需要解析视频格式。如今,媒体容器格式繁多复杂,要想准确的判断一个视频文件的格式和编码,不是件容易的事情。还好,我们有伟大的开源项目,用很多开源软件都可以读取出视频格式,这里推荐MediaInfo,因为它似乎就是为这个目的而建立的,生成的属性信息详细而准确,正是我们想要的。此外,MediaInfo还提供了一个动态链接库"MediaInfo.dll"。
很明显,MediaInfo告诉了你,这个影片的时长(Duration),宽度(Width)和高度(Height),当然还有其他你想要或不想要的信息,够了。
3. 把所有截图绘制在一张图像中
这个没什么好说的,只要前两步做出来了,这一步基本不是什么问题,根据视频的宽和高,以及最终结果图片的宽和高,计算出你需要对截图进行缩放的比例,然后把图像绘制在一起,生成新的图像即可。下面是用 Qt 实现这一步的代码:
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(" 格式:"));
19 titlename.append(QString::fromStdString(cinfo.format_name));
20 imagePainter.drawText(150, 30, titlename);
21 std::ostringstream oss;
22 oss << "文件大小:" << cinfo.size / 1000000 << "MB 视频尺寸:"
23 << cinfo.width << "x" << cinfo.height << " 视频时长:";
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开发的,当然,很简单就能转换成其他的语言和类库来实现。