专栏首页HBStream流媒体与音视频技术借助开源项目,又好又快的实现视频文件”剧情连拍(剧情截图)”功能

借助开源项目,又好又快的实现视频文件”剧情连拍(剧情截图)”功能

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

1. 

2. 

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

  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"。

  很明显,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开发的,当然,很简单就能转换成其他的语言和类库来实现。

文章版权所有,转载请注明作者和出处,谢谢~

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • C++实现RTMP协议发送H.264编码及AAC编码的音视频,摄像头直播

      RTMP(Real Time Messaging Protocol)是专门用来传输音视频数据的流媒体协议,最初由Macromedia 公司创建,后来归Ado...

    hbstream
  • C++实现RTMP协议发送H.264编码及AAC编码的音视频,摄像头直播

    hbstream
  • 随便写了个猪八戒网的任务小助手“小八戒”,需要的拿去

    hbstream
  • leetcode-204-Count Primes

    chenjx85
  • SpringBoot下的策略模式,消灭了大量的ifelse,真香!

    项目中有这样一个场景,在公园放置了用来拍摄人像的识别杆,根据用户在不同识别杆之间采集的图象来计算用户的运动距离。由于涉及到许多公园,每个公园的布局不同,识别杆之...

    程序新视界
  • Golang Leetcode 796. Rotate String.go

    更多内容请移步我的repo:https://github.com/anakin/golang-leetcode

    anakinsun
  • 每日一题2

    属于简答题,那么一开始我自己写就很直接暴力,循环一边就好了,管他什么效率。代码如下:

    可爱见见
  • Java 并发(6)Semaphore 源码分析

    Semaphore 提供了一个许可证的概念,可以把这个许可证看作公共汽车车票,只有成功获取车票的人才能够上车,并且车票是有一定数量的,不可能毫无限制的发下去,这...

    一个优秀的废人
  • Java 反射

    黑白格
  • Golang Gin 实战(十四)| 文件托管、反向代理百度网站、自实现API网关

    Golang Gin作为一个优秀的框架,不仅为我们提供了托管文件的能力,还为我们提供了从io.Reader,这篇文章除了介绍文件托管的使用和原理外,我们还会利用...

    飞雪无情

扫码关注云+社区

领取腾讯云代金券