本视频监控系统默认内核采用的是ffmpeg来解析rtsp视频流,同时还支持vlc内核,ffmpeg在播放视频流的时候,可以打开文件进行视频流信息的存储,默认存储的是裸流,可以自行根据编码规则改成MP4格式的,这样存储的视频文件可以用其他播放器打开,而如果是存储的裸流的话,一般需要ffmpeg自身去打开播放,目前测试过的支持裸流直接播放的播放器是完美解码播放器potplayer,如果还有同名的aac声音文件的话,会同步播放声音。
采用ffmpeg来存储视频流和对应的声音文件,还是非常方便的,直接打开文件后写入data数据即可,在拿到视频流解码的时候,可以判断是否还带了音频流,如果带了的话,可以同步存储音频文件到aac文件,存储音频流的时候需要做个特殊处理,先写入dts头,再写入音频流数据,不然会出错。
本系统封装的ffmpeg类,提供了两种方式存储视频文件,一种是存储成单个视频文件,还有一种是按照存储间隔比如30分钟存储成多个视频文件,到了时间间隔重新生成文件存储,在视频监控领域第二种用的比较多,这样方便回放录像和拷贝录像,单个文件比较小,很容易查询和拷贝。
皮肤开源:[https://gitee.com/feiyangqingyun/QWidgetDemo](https://gitee.com/feiyangqingyun/QWidgetDemo) [https://github.com/feiyangqingyun/QWidgetDemo](https://github.com/feiyangqingyun/QWidgetDemo)文件名称:styledemo体验地址:[https://gitee.com/feiyangqingyun/QWidgetExe](https://gitee.com/feiyangqingyun/QWidgetExe) [https://github.com/feiyangqingyun/QWidgetExe](https://github.com/feiyangqingyun/QWidgetExe)文件名称:bin_video_system.zip二、功能特点1. 支持16画面切换,全屏切换等,包括1+4+6+8+9+13+16画面切换。
2. 支持alt+enter全屏,esc退出全屏。
3. 自定义信息框+错误框+询问框+右下角提示框。
4. 17套皮肤样式随意更换,所有样式全部统一,包括菜单等。
5. 云台仪表盘鼠标移上去高亮,八个方位精准识别。
6. 底部画面工具栏(画面分割切换+截图声音等设置)移上去高亮。
7. 可在配置文件更改左上角logo+中文软件名称+英文软件名称。
8. 封装了百度地图,三维切换,设备点位,鼠标按下获取经纬度等。
9. 堆栈窗体,每个窗体都是个单独的qwidget,方便编写自己的代码。
10. 顶部鼠标右键菜单,可动态控制时间CPU+左上角面板+左下角面板+右上角面板+右下角面板的显示和隐藏,支持恢复默认布局。
11. 工具栏可以放置多个小图标和关闭图标。
12. 左侧右侧可拖动拉伸,并自动记忆宽高位置,重启后恢复。
13. 双击摄像机节点自动播放视频,双击节点自动依次添加视频,会自动跳到下一个,双击父节点自动添加该节点下的所有视频。
14. 摄像机节点拖曳到对应窗体播放视频,同时支持拖曳本地文件直接播放。
15. 视频画面窗体支持拖曳交换,瞬间响应。
16. 双击节点+拖曳节点+拖曳窗体交换位置,均自动更新url.txt。
17. 支持从url.txt中加载16通道视频播放,自动记忆最后通道对应的视频,软件启动后自动打开播放。
18. 右下角音量条控件,失去焦点自动隐藏,音量条带静音图标。
19. 集成百度地图,可以添加设备对应位置,自动生成地图,支持缩放和三维地图,提供地图风格选择,共12种风格。
20. 视频拖动到通道窗体外自动删除视频。
21. 鼠标右键可删除当前+所有视频,截图当前+所有视频。
22. 录像机管理、摄像机管理,可添加删除修改导入导出打印信息,立即应用新的设备信息生成树状列表,不需重启。
23. 在pro文件中可以自由开启是否加载地图。
24. 视频播放可选四种内核自由切换,vlc+ffmpeg+easyplayer+海康sdk,均可在pro中设置。
25. 可设置1+4+9+16画面轮询,可设置轮询间隔以及轮询码流类型等,直接在主界面底部工具栏右侧单击启动轮询按钮即可,再次单击停止轮询。
26. 默认超过10秒钟未 *** 作自动隐藏鼠标指针。
27. 支持onvif搜素设备,支持任意onvif摄像机,包括但不限于海康大华宇视天地伟业华为等,支持onvif云台控制。
28. 高度可定制化,用户可以很方便的在此基础上衍生自己的功能,支持linux系统。
三、效果图四、核心代码void FFmpegThread::run(){ //计时 QTime time; while (!stopped) { //根据标志位执行初始化 *** 作 if (isPlay) { if (init()) { //启用保存文件,先关闭文件 if (saveFile) { if (fileVideo.isOpen()) { fileVideo.close(); } if (fileAudio.isOpen()) { fileAudio.close(); } //如果存储间隔大于0说明需要定时存储 if (saveInterval > 0) { fileName = QString("%1/%2.mp4").arg(savePath).arg(STRDATETIME); emit sig_startSave(); } if (videoStreamIndex >= 0) { fileVideo.setFileName(fileName); fileVideo.open(QFile::WriteOnly); } if (audioStreamIndex >= 0) { fileAudio.setFileName(fileName.replace(QFileInfo(fileName).suffix(), "aac")); fileAudio.open(QFile::WriteOnly); } } emit receivePlayOk(); } else { break; emit receivePlayError(); } isPlay = false; continue; } if (isPause) { //这里需要假设正常,暂停期间继续更新时间 lastTime = QDateTime::currentDateTime(); msleep(1); continue; } time.restart(); if (av_read_frame(avFormatContext, avPacket) >= 0) { //判断当前包是视频还是音频 int packetSize = avPacket->size; int index = avPacket->stream_index; if (index == videoStreamIndex) { //解码视频流 if (hardware == "none") { avcodec_decode_video2(videoCodec, avFrame2, &frameFinish, avPacket); } else { frameFinish = decode_packet(videoCodec, avPacket); } if (frameFinish) { //计数,只有到了设定的帧率才刷新图片 frameCount++; if (frameCount != interval) { av_packet_unref(avPacket); av_freep(avPacket); msleep(1); continue; } else { frameCount = 0; } //保存视频流数据到文件 QMutexLocker lock(&mutex); if (fileVideo.isOpen()) { //rtmp视频流需要添加pps sps#ifndef gcc45 av_bsf_filter(filter, avPacket, avFormatContext->streams[videoStreamIndex]->codecpar);#endif fileVideo.write((const char *)avPacket->data, packetSize); } //将数据转成一张图片 sws_scale(swsContext, (const uint8_t *const *)avFrame2->data, avFrame2->linesize, 0, videoHeight, avFrame3->data, avFrame3->linesize); //以下两种方法都可以 //QImage image(avFrame3->data[0], videoWidth, videoHeight, QImage::Format_RGB32); QImage image((uchar *)buffer, videoWidth, videoHeight, QImage::Format_RGB32); if (!image.isNull()) { lastTime = QDateTime::currentDateTime(); emit receiveImage(image); //计算本地视频文件等待时间 int useTime = time.elapsed(); if (!isRtsp && videoFps > 0) { //一帧解码用时+固定休眠1毫秒+其他用时1毫秒 int frameTime = useTime + 1 + 1; //等待时间=1秒钟即1000毫秒-所有帧解码完成用的毫秒数/帧数 sleepTime = (1000 - (videoFps * frameTime)) / videoFps; //有时候如果图片很大或者解码很难比如h265造成解码一张图片耗时很大可能出现负数 sleepTime = sleepTime < 0 ? 0 : sleepTime; } //qDebug() << TIMEMS << image.size() << "useTime" << time.elapsed() << "sleepTime" << sleepTime << "videoFps" << videoFps; } msleep(sleepTime); } } else if (index == audioStreamIndex) { //解码音频流,这里暂不处理,以后交给sdl播放 //保存音频流数据到文件 QMutexLocker lock(&mutex); if (fileAudio.isOpen()) { //先写入dts头,再写入音频流数据 dtsData[3] = (char)(((2 & 3) << 6) + ((7 + packetSize) >> 11)); dtsData[4] = (char)(((7 + packetSize) & 0x7FF) >> 3); dtsData[5] = (char)((((7 + packetSize) & 7) << 5) + 0x1F); fileAudio.write((const char *)dtsData, 7); fileAudio.write((const char *)avPacket->data, packetSize); } } } else if (!isRtsp) { //如果不是视频流则说明是视频文件播放完毕 break; } av_packet_unref(avPacket); av_freep(avPacket); msleep(1); } emit sig_stopSave(); //线程结束后释放资源 free(); stopped = false; isPlay = false; isPause = false; emit receivePlayFinsh(); qDebug() << TIMEMS << "stop ffmpeg thread";}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)