背景描述

作为一个播放器,截图功能必不可少; 下面主要记录一下截图功能的实现;

实现流程

  • 将解码后的帧进行格式转换(目标格式为RGB24);

  • 采用独立的线程进行截图处理;

  • 截图可保存为BMP或JPG两种格式;

代码实现

#define _WIDTHBYTES(c)  ((c+31)/32*4)   // c = width * bpp
int Snapshot2File(RENDER_FORMAT renderFormat, SNAPSHOT_IMAGE_T *pSnapshot, char *pbuf)
{
PBYTE pDest = NULL, pDest16 = NULL;
INT nBpp;
DWORD dwW, dwH, dwWB; int ret = 0; int image_format = pSnapshot->imageFormat; if (renderFormat == RENDER_FORMAT_YUY2) nBpp = 16;
else if (renderFormat == RENDER_FORMAT_UYVY) nBpp = 16;
else if (renderFormat == RENDER_FORMAT_X8R8G8B8) nBpp = 32; //ok
else if (renderFormat == RENDER_FORMAT_A8R8G8B8) nBpp = 32; //ok
else if (renderFormat == RENDER_FORMAT_RGB565) nBpp = 16; //ok
else if (renderFormat == RENDER_FORMAT_RGB555) nBpp = 16; //ok
else if (renderFormat == RENDER_FORMAT_RGB24_GDI) nBpp = 24;
else return -1; //格式错误 dwW = pSnapshot->width;
dwH = pSnapshot->height;
dwWB = _WIDTHBYTES( dwW * nBpp ); if (image_format == 0x00)
{
//BMP
int iFilenameLen = (int)strlen(pSnapshot->filename); if ( (0 != memcmp(pSnapshot->filename+iFilenameLen-3, "bmp", 3)) &&
(0 != memcmp(pSnapshot->filename+iFilenameLen-3, "BMP", 3)) &&
(0 != memcmp(pSnapshot->filename+iFilenameLen-3, "Bmp", 3)) )
{
if (pSnapshot->filename[iFilenameLen-4] == '.')
{
for (int i=iFilenameLen-1; i>0; i--)
{
if (pSnapshot->filename[i] == '.')
{
pSnapshot->filename[i] = '\0';
break;
}
else
{
pSnapshot->filename[i] = '\0';
}
}
}
strcat(pSnapshot->filename, ".bmp");
} HANDLE hFile = CreateFile( pSnapshot->filename, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
if( hFile == INVALID_HANDLE_VALUE ) return E_HANDLE; // SaveFile to BMP
BITMAPFILEHEADER bfh = {0};
bfh.bfType = 0x4D42;
bfh.bfSize = 0;
bfh.bfReserved1 = 0;
bfh.bfReserved2 = 0;
bfh.bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER);
if( nBpp == 16 ) {
bfh.bfOffBits += sizeof(RGBQUAD) * 3;
}
else
if( nBpp == 24 ) {
bfh.bfOffBits += sizeof(RGBQUAD) * 1;
}
else
if( nBpp == 32 ) {
bfh.bfOffBits += sizeof(RGBQUAD) * 1;
}
DWORD dwWriteLength = sizeof(BITMAPFILEHEADER);
DWORD dwWrittenLength = 0;
WriteFile( hFile, (PVOID)&bfh, dwWriteLength, &dwWrittenLength, NULL ); BITMAPINFOHEADER bih = {0};
bih.biSize = sizeof(BITMAPINFOHEADER);
bih.biWidth = dwW;
bih.biHeight = -(INT)dwH;
bih.biPlanes = 1;
bih.biBitCount = nBpp;
bih.biCompression = (nBpp == 16) ? BI_BITFIELDS : BI_RGB;
bih.biSizeImage = dwWB * pSnapshot->height;
bih.biXPelsPerMeter = 0;
bih.biYPelsPerMeter = 0;
bih.biClrUsed = 0;
bih.biClrImportant = 0; dwWriteLength = sizeof(BITMAPINFOHEADER);
WriteFile( hFile, (PVOID)&bih, dwWriteLength, &dwWrittenLength, NULL ); if( nBpp == 16 ) { DWORD argbQuad[3] = {0};
if (renderFormat == RENDER_FORMAT_RGB565)
{
argbQuad[0] = 0x00F800; // Red mask
argbQuad[1] = 0x0007E0; // Green mask
argbQuad[2] = 0x00001F; // Blue mask
}
else
{
argbQuad[0] = 0x007C00; // Red mask
argbQuad[1] = 0x0003E0; // Green mask
argbQuad[2] = 0x00001F; // Blue mask
}
dwWriteLength = sizeof(argbQuad);
WriteFile( hFile, (PVOID)&argbQuad[0], dwWriteLength, &dwWrittenLength, NULL );
}
else
if( nBpp == 24 ) {
DWORD rgbQuad = 0;
dwWriteLength = sizeof(rgbQuad);
WriteFile( hFile, (PVOID)&rgbQuad, dwWriteLength, &dwWrittenLength, NULL );
}
else if( nBpp == 32 ) {
DWORD rgbQuad = 0;
dwWriteLength = sizeof(rgbQuad);
WriteFile( hFile, (PVOID)&rgbQuad, dwWriteLength, &dwWrittenLength, NULL );
} dwWriteLength = dwWB * pSnapshot->height;
WriteFile( hFile, (PVOID)pbuf, dwWriteLength, &dwWrittenLength, NULL );
CloseHandle( hFile ); if (dwWrittenLength < 1)
{
DeleteFile(pSnapshot->filename);
}
}
else if (image_format == 0x01)
{
//JPG
int iFilenameLen = (int)strlen(pSnapshot->filename); if ( (0 != memcmp(pSnapshot->filename+iFilenameLen-3, "jpg", 3)) &&
(0 != memcmp(pSnapshot->filename+iFilenameLen-3, "JPG", 3)) &&
(0 != memcmp(pSnapshot->filename+iFilenameLen-3, "Jpg", 3)) )
{
if (pSnapshot->filename[iFilenameLen-4] == '.')
{
for (int i=iFilenameLen-1; i>0; i--)
{
if (pSnapshot->filename[i] == '.')
{
pSnapshot->filename[i] = '\0';
break;
}
else
{
pSnapshot->filename[i] = '\0';
}
}
}
strcat(pSnapshot->filename, ".jpg");
} BOOL bres = TRUE;
IJLERR jerr;
DWORD dibPadBytes;
JPEG_CORE_PROPERTIES jcprops; __try
{
jerr = ijlInit(&jcprops); if (IJL_OK != jerr)
{
bres = FALSE;
__leave;
} dibPadBytes = IJL_DIB_PAD_BYTES(dwW, 3); jcprops.DIBWidth = dwW;
jcprops.DIBHeight = dwH;
jcprops.DIBBytes = (unsigned char*)pbuf;//reinterpret_cast<BYTE *>(&pbi->bmiHeader) + sizeof(BITMAPINFOHEADER); jcprops.DIBPadBytes = dibPadBytes;
jcprops.DIBChannels = 3;
jcprops.DIBColor = IJL_BGR; jcprops.JPGFile = const_cast<LPSTR>(pSnapshot->filename); jcprops.JPGWidth = dwW;
jcprops.JPGHeight = dwH; jcprops.JPGChannels = 3;
jcprops.JPGColor = IJL_YCBCR;
jcprops.JPGSubsampling = IJL_411;
jcprops.jquality = 95; jerr = ijlWrite(&jcprops, IJL_JFILE_WRITEWHOLEIMAGE); if (IJL_OK != jerr)
{
if (IJL_FILE_ERROR == jerr)
{
}
ret = jerr;
bres = FALSE; if (ret == -23)
{
DeleteFile(pSnapshot->filename);
}
}
} __finally
{
ijlFree(&jcprops);
} } if (NULL != pDest)
{
delete []pDest;
pDest = NULL;
} return ret;
}

关于EasyPlayerPro

EasyPlayerPro是一款全功能的流媒体播放器,支持RTSP、RTMP、HTTP、HLS、UDP、RTP、File等多种流媒体协议播放、支持本地文件播放,支持本地抓拍、本地录像、播放旋转、多屏播放、倍数播放等多种功能特性,核心基于ffmpeg,稳定、高效、可靠、可控,支持Windows、Android、iOS三个平台,目前在多家教育、安防、行业型公司,都得到的应用,广受好评!

EasyPlayerPro:https://github.com/EasyDSS/EasyPlayerPro

点击链接加入群【EasyPlayer & EasyPlayerPro】:544917793

获取更多信息

邮件:support@easydarwin.org

EasyDarwin开源流媒体服务器:www.EasyDarwin.org

EasyDSS商用流媒体解决方案:www.EasyDSS.com

EasyNVR无插件直播方案:www.EasyNVR.com

Copyright © EasyDarwin Team 2012-2017

EasyPlayerPro Windows播放器本地快照抓拍截图功能实现方法的更多相关文章

  1. EasyPlayerPro windows播放器本地音频播放音量控制实现

    背景描述 作为一个播放器, 除了能播放视频和声音外,音量控制是绝对不能缺少的功能; 本文在音视频播放的基础上,增加对音量的控制: 实现流程 调用mixerGetDevCaps获取音频输出设备列表; 打 ...

  2. EasyPlayerPro windows播放器本地配置文件配置方法介绍

    需求背景 应EasyPlayerPro某客户需求,在EasyPlayerPro启动时,自动播放指定的url源, 不需要每次都去手动填写, 且实现自动播放,不需要手动的单击播放按钮: 为响应该需求,特增 ...

  3. EasyPlayerPro Windows播放器进行本地对讲喊话音频采集功能实现

    需求 在安防行业应用中,除了在本地看到摄像机的视频和进行音频监听外,还有一个重要的功能,那就是对讲. EasyPlayerPro-win为了减轻二次开发者的工作量,将本地音频采集也进行了集成: 功能特 ...

  4. EasyPlayerPro windows播放器在播放RTMP视频显示重复异常问题解决

    问题来源 2017.12.18 今日有杭州某教育领域客户反馈EasyPlayerPro在播放一个rtmp源时,画面显示异常的问题.截图如下: 问题复现 一番思考, 将显示格式改为D3D显示, 正常, ...

  5. EasyPlayerPro Windows播放器读取xml配置文件中的特殊字符问题

    问题被反馈 今日一客户反馈说播放不了带用户名密码的流, 奇怪,这个问题不存在啊-,按照客户的说法, 是将url地址保存在配置文件中,然后再打开EasyPlayerPro运行: 问题复现 在EasyPl ...

  6. EasyPlayerPro Windows播放器电子放大/局部放大播放功能实现

    背景描述 在视频监控软件中,我们看到很多的软件都有电子放大功能, 按住鼠标左键不放,框选一个区域,再松开鼠标左键,即对选中的区域进行放大显示, 且可以重复该操作,逐步放大所需显示的区域, 有没有觉得, ...

  7. EasyPlayerPro Windows播放器实时流进行本地缓冲区即时回放功能实现

    背景描述 参照国内视频监控行业监控软件,实现当前视频的即时回放功能,例如: 监控人员发现刚刚的某个视频点有可疑,就像录像回放一样,想倒回去看一下,但又不想切换到录像回放界面, 此处就体现即时回放的价值 ...

  8. EasyPlayerPro Windows播放器全屏模式下GDI显示出现黑屏问题解决

    问题来源 2017.12.21 前天有杭州某教育领域客户反馈有部分视频源在全屏模式下显示黑屏: 问题复现 EasyPlayerPro由于没有实现单个窗口完全全屏,故没有暴露该问题,晚上加班,加上单个窗 ...

  9. EasyPlayer RTSP Windows播放器D3D,GDI的几种渲染方式的选择区别

    EasyPlayer-RTSP windows播放器支持D3D和GDI两种渲染方式,其中D3D支持格式如下: DISPLAY_FORMAT_YV12 DISPLAY_FORMAT_YUY2 DISPL ...

随机推荐

  1. NOIP2016模拟赛三 Problem C: 不虚就是要AK

    题目大意 给定一棵带有边权的树, 问你在树上随机选两个点, 它们最短路径上的边权之和为\(4\)的倍数的概率为多少. Solution 树分治. 没什么好讲的. #include <cstdio ...

  2. 我眼中的AI

    !!!!本文禁止转载,引用,仅供观看 最初了解人工智能是在我上大二的时候,在看头条的时候经常看到有关人工智能的事情,当时的我和大多数的人一样,感觉人工智能很神奇.当时就整晚整晚的想人工智能会取代人类呀 ...

  3. 数据库访问的弹性化---WebLogic和Oracle RAC的整合:Active GridLink

        1.  什么是Active GridLink Data Source 从Oracle WebLogic Server 10.3.4版本开始引进了一种单数据源实现来支持Oracle RAC集群. ...

  4. struts2.16中文乱码问题解决

    方法1.在struts.xml文件中添加<constant name="struts.i18n.encoding" value="GBK" /> 方 ...

  5. JVM技术部分总结

    1.JVM内存模型 1.1 JVM内存模型图解 Java虚拟机在执行Java程序的过程中,会把它所管理的内存划分为若干个不同的数据区.这些区域有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程 ...

  6. JAVA_Could not find property [struts.actionMapping]怎么办

    你的项目中不包含log4j.jar这个文件,包含进去即可

  7. GEM演唱会

    周六去魔都看邓紫棋演唱会,各位看官可能要问.杭州不是也有嚒.为嘛去魔都-..由于po主是逗比哈哈(- ̄▽ ̄-) 早上睡到自然醒,然后開始做午饭.吃完躺沙发上看电视,看到一点多认为应该要出发了(演唱会7 ...

  8. 阿里云数据库RDS迁移,DTS 迁移过程中,是否会锁表,对源数据库是否有影响?

    阿里云数据库RDS迁移,DTS 迁移过程中,是否会锁表,对源数据库是否有影响? DTS 在进行全量数据迁移和增量数据迁移的过程中,均不会对源端数据库进行锁表,因此在全量数据迁移和增量数据迁移的过程中, ...

  9. 实用国际(XX)计量单位表

    很多实用附录简表:http://www.zdic.net/appendix/f1.htm 计量单位简表 时间的单位换算 : 1秒=1000毫秒(ms) 1毫秒=1/1,000秒(s)  1秒=1,00 ...

  10. json解析:[1]gson解析json

    客户端与服务器进行数据交互时,常常需要将数据在服务器端将数据转化成字符串并在客户端对json数据进行解析生成对象.但是用jsonObject和jsonArray解析相对麻烦.利用Gson和阿里的fas ...