最简单的基于DirectShow的示例:视频播放器图形界面版
=====================================================
最简单的基于DirectShow的示例文章列表:
最简单的基于DirectShow的示例:视频播放器图形界面版
最简单的基于DirectShow的示例:获取Filter信息
=====================================================
本文记录一个最简单的基于DirectShow的图形界面的视频播放器。基于DirectShow的图形界面的播放器的例子还是比较多的,但是大部分都是“层层封装”的例子。“层层封装”的例子相对来说更加稳定,但是却不是很容易理解。因为DirectShow本身的接口函数的数量就比较多,如果再加上封装DirectShow的函数,合起来的函数数量是非常大的,很容易让人搞不清哪些才是真正的DirectShow接口函数。本播放器剥去了DirectShow例子中的“层层封装”,直接调用DirectShow的接口完成视频的播放工作,更加适合DirectShow入门使用。
几个功能的实现机制
整个工程的代码比较多,不再详细记录。在这里简单记录一下代码中的几个关键点。
视频的播放/暂停/继续/停止
播放
视频“播放”的源代码如下所示。简单来说,完成了以下视频播放的初始化工作:
(1) 输入的URL转换为Unicode编码(RenderFile()函数支持的输入是Unicode字符串)。
(2) 调用RenderFile()“智能”创建Filter Graph。
(3) 调用IMediaControl的Run()方法开始播放视频。
(4) 开启定时器,用于更新视频播放的进度(后文详细记录)
void CplayerGUIDlg::OnBnClickedStart()
{
CStringA cstr_urla;
CStringW cstr_urlw;
HRESULT hr;
//Render
#ifdef _UNICODE
m_url.GetWindowText(cstr_urlw);
#else
USES_CONVERSION;
m_url.GetWindowText(cstr_urla);
cstr_urlw.Format(L"%s",A2W(cstr_urla));
#endif
if(cstr_urlw.IsEmpty()){
AfxMessageBox(_T("Input URL is NULL!"));
return;
}
hr = pGraph->RenderFile(cstr_urlw, NULL);
if(FAILED(hr)){
AfxMessageBox(_T("Can't open input file!"));
return;
}
//Set Window
HWND screen_hwnd=NULL;
RECT windowRect;
screen_hwnd = this->GetDlgItem(IDC_SCREEN)->GetSafeHwnd();
::GetClientRect(screen_hwnd, &windowRect);
pWindow->put_Visible(OAFALSE);
pWindow->put_Owner((OAHWND)screen_hwnd);
pWindow->put_Left(0);
pWindow->put_Top(0);
pWindow->put_Width(windowRect.right - windowRect.left);
pWindow->put_Height(windowRect.bottom - windowRect.top);
pWindow->put_WindowStyle(WS_CHILD|WS_CLIPCHILDREN|WS_CLIPSIBLINGS|WS_THICKFRAME);
pWindow->put_MessageDrain((OAHWND) screen_hwnd);//Receive Message
pWindow->put_Visible(OATRUE);
pEvent->SetNotifyWindow((OAHWND)screen_hwnd, WM_GRAPHNOTIFY, 0);
// Run
hr = pControl->Run();
playerstate=STATE_PLAY;
SetBtn(STATE_PLAY);
SetTimer(1,1000,NULL);
}
暂停/继续
视频“暂停/继续”的源代码如下所示。其中调用了IMediaControl的Pause()和Run()设定“暂停”或者是“继续”。
void CplayerGUIDlg::OnBnClickedPause()
{
HRESULT hr;
if(playerstate==STATE_PLAY){
hr=pControl->Pause();
playerstate=STATE_PAUSE;
GetDlgItem(ID_PAUSE)->SetWindowText(_T("Resume"));
}else if(playerstate==STATE_PAUSE){
hr=pControl->Run();
playerstate=STATE_PLAY;
GetDlgItem(ID_PAUSE)->SetWindowText(_T("Pause"));
}
}
停止
视频的“停止”的源代码如下所示。该部分代码完成了以下工作:
(1) 把播放的位置重新调整为0
(2) 调用IMediaControl的Pause()
(3) 关闭定时器
(4) 删除Filter Graph中的Filter
void CplayerGUIDlg::OnBnClickedStop()
{
long long position = 0;
HRESULT hr;
hr = pSeeking->SetPositions(&position, AM_SEEKING_AbsolutePositioning | AM_SEEKING_SeekToKeyFrame,
0, AM_SEEKING_NoPositioning);
KillTimer(1);
hr=pControl->Stop();
// Enumerate the filters And remove them
IEnumFilters *pEnum = NULL;
hr = pGraph->EnumFilters(&pEnum);
if (SUCCEEDED(hr))
{
IBaseFilter *pFilter = NULL;
while (S_OK == pEnum->Next(1, &pFilter, NULL))
{
// Remove the filter.
pGraph->RemoveFilter(pFilter);
// Reset the enumerator.
pEnum->Reset();
pFilter->Release();
}
pEnum->Release();
}
SystemClear();
}
视频播放进度在时间轴的显示
随着视频的播放,需要在视频播放进度的时间轴上更新播放进度信息。在程序中使用了一个定时器完成这个功能。
在视频开始播放的时候,调用SetTimer()开启定时器。时间间隔设置为1000ms。
SetTimer(1,1000,NULL);
在视频停止播放的时候,调用KillTimer()结束定时器。
KillTimer(1);
在定时器的消息响应函数中,调用了IMediaSeeking的GetCurrentPosition()获取视频当前播放到的时间,调用了IMediaSeeking的GetDuration ()获取视频的时长。根据以上函数得到的数值,计算后把结果设置到相应的控件上。这部分的代码如下所示。
void CplayerGUIDlg::OnTimer(UINT_PTR nIDEvent)
{
if (nIDEvent == 1){
CString curtimestr,durationstr;
long long curtime;
long long duration;
int tns, thh, tmm, tss;
int progress;
//ms
pSeeking->GetCurrentPosition(&curtime);
if(curtime!=0){
//change to second
tns = curtime/10000000;
thh = tns / 3600;
tmm = (tns % 3600) / 60;
tss = (tns % 60);
curtimestr.Format(_T("%02d:%02d:%02d"),thh,tmm,tss);
m_curtime.SetWindowText(curtimestr);
}
pSeeking->GetDuration(&duration);
if(duration!=0){
tns = duration/10000000;
thh = tns / 3600;
tmm = (tns % 3600) / 60;
tss = (tns % 60);
durationstr.Format(_T("%02d:%02d:%02d"),thh,tmm,tss);
m_duration.SetWindowText(durationstr);
progress=curtime*100/duration;
m_progress.SetPos(progress);
}
}
CDialogEx::OnTimer(nIDEvent);
}
视频播放点的调整
当鼠标拖动滑动控制条(Slider Control)控件上的滑块的时候,需要根据拖动的位置设置视频的播放进度。此时调用IMediaSeeking的SetPositions()设定视频的播放进度。消息响应函数中的代码如下所示。
void CplayerGUIDlg::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
if (pScrollBar->GetSafeHwnd() == m_progress.GetSafeHwnd()){
float pos_bar=0.0;
long long duration=0.0;
long long pos_time=0.0;
if(nSBCode==SB_THUMBPOSITION){
pos_bar=(float)nPos/100.0;
pSeeking->GetDuration(&duration);
pos_time=pos_bar*duration;
long long position = (long long)(pos_time);
HRESULT hr = pSeeking->SetPositions(&position, AM_SEEKING_AbsolutePositioning | AM_SEEKING_SeekToKeyFrame,
0, AM_SEEKING_NoPositioning);
}
}
CDialogEx::OnHScroll(nSBCode, nPos, pScrollBar);
}
“全屏播放”的问题
视频的全屏播放通过IVideoWindow的put_FullScreenMode()实现,代码如下所示。
void CplayerGUIDlg::OnBnClickedFullscreen()
{
pWindow->put_FullScreenMode(OATRUE);
}
同时,在“全屏模式”启动后,如果按“ESC”键的话,可以关闭“全屏模式”。这部分的代码在PreTranslateMessage()中实现,如下所示。
//Exit Full Screen mode when push "ESC"
BOOL CplayerGUIDlg::PreTranslateMessage(MSG* pMsg)
{
if (pMsg->message == WM_KEYDOWN){
if (pMsg->wParam == VK_RETURN || pMsg->wParam == VK_ESCAPE){
// Restore form fullscreen mode
pWindow->put_FullScreenMode(OAFALSE);
return 1;
}
}
return CDialogEx::PreTranslateMessage(pMsg);
}
在这里有一点需要注意,IVideoWindow的put_FullScreenMode()在Win7下是有问题的。只有在设置窗口样式的的时候,在样式中指定WS_THICKFRAME后才可以正常使用。例如如下代码。
pWindow->put_WindowStyle(WS_CHILD|WS_CLIPCHILDREN|WS_CLIPSIBLINGS|WS_THICKFRAME);
如果没有指定WS_THICKFRAME样式的话,在退出“全屏”模式之后,视频就显示不出来了,取而代之的是一片黑色。
但是设定WS_THICKFRAME样式之后,视频窗口的外围会有一层“白边”,会影响到视频显示的美观。因此我们如果希望正常使用全屏的话,可能需要找一种更好的方法,在这里我就没有深入研究了。
运行结果
这是使用DirectShow基于MFC开发的一个示例播放器。实现了一个播放器的基本功能:播放,暂停/继续,停止,播放时间轴的显示,以及从任一点开始播放媒体。并且支持将媒体文件拖拽至播放器进行播放。播放前将媒体文件的路径输入到“URL”栏中,然后单击“Start”即可开始播放。在软件下方包含了“start”,“Pause”,“Stop”等按钮用于控制媒体的播放。
播放时候的效果截图如下所示。
单击“Full Screen”可以全屏播放。单击“Info”可以显示正在播放媒体的信息,包括以下两种信息:
(1) 该视频的相关信息
(2) 播放该视频的 Filter Graph中的Filter。
下载
Simplest DirectShow Example
项目主页
SourceForge:https://sourceforge.net/projects/simplestdirectshowexample/
Github:https://github.com/leixiaohua1020/simplest_directshow_example
开源中国:http://git.oschina.net/leixiaohua1020/simplest_directshow_example
CDSN下载地址:http://download.csdn.net/detail/leixiaohua1020/8348163
本程序包含了DirectShow开发的示例程序。适合DirectShow初学者进行学习。
它包含了以下几个子程序:
simplest_directshow_player: 最简单的基于DirectShow的视频播放器。
simplest_directshow_player_custom: 最简单的基于DirectShow的视频播放器(Custom)。
playerGUI: 最简单的基于DirectShow的播放器-图形界面版。
simplest_directshow_info: 最简单的Directshow信息显示例子。
simplest_directshow_filter: 目前还未完成。
最简单的基于DirectShow的示例:视频播放器图形界面版的更多相关文章
- 最简单的基于DirectShow的示例:视频播放器自定义版
===================================================== 最简单的基于DirectShow的示例文章列表: 最简单的基于DirectShow的示例:视 ...
- 最简单的基于DirectShow的示例:视频播放器
===================================================== 最简单的基于DirectShow的示例文章列表: 最简单的基于DirectShow的示例:视 ...
- 最简单的基于DirectShow的示例:获取Filter信息
===================================================== 最简单的基于DirectShow的示例文章列表: 最简单的基于DirectShow的示例:视 ...
- 最简单的基于FFMPEG+SDL的视频播放器 ver2 (採用SDL2.0)
===================================================== 最简单的基于FFmpeg的视频播放器系列文章列表: 100行代码实现最简单的基于FFMPEG ...
- 最简单的基于FFMPEG+SDL的视频播放器 ver2 (采用SDL2.0)
===================================================== 最简单的基于FFmpeg的视频播放器系列文章列表: 100行代码实现最简单的基于FFMPEG ...
- 基于<最简单的基于FFMPEG+SDL的视频播放器 ver2 (采用SDL2.0)>的一些个人总结
最近因为项目接近收尾阶段,所以变的没有之前那么忙了,所以最近重新拿起了之前的一些FFMPEG和SDL的相关流媒体播放器的例子在看. 同时自己也用FFMPEG2.01,SDL2.01结合MFC以及网上罗 ...
- 用JavaCV改写“100行代码实现最简单的基于FFMPEG+SDL的视频播放器 ”
FFMPEG的文档少,JavaCV的文档就更少了.从网上找到这篇100行代码实现最简单的基于FFMPEG+SDL的视频播放器.地址是http://blog.csdn.net/leixiaohua102 ...
- 100行代码实现最简单的基于FFMPEG+SDL的视频播放器(SDL1.x)【转】
转自:http://blog.csdn.net/leixiaohua1020/article/details/8652605 版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[-] ...
- 最简单的基于FFMPEG+SDL的视频播放器:拆分-解码器和播放器
===================================================== 最简单的基于FFmpeg的视频播放器系列文章列表: 100行代码实现最简单的基于FFMPEG ...
随机推荐
- of_alias_get_id 函数与设备树中aliases节点的关系【转】
转自:https://blog.csdn.net/qq_30145093/article/details/78053823?locationNum=10&fps=1 转自http://www. ...
- jquery easyui panel title文字格式设置
$('#txtLeftPercent').panel({ title: '剩余权重:' + '<b style="color:red">' + 100 + '%< ...
- 在vue中操作DOM--this.$nextTick()
虽然 Vue.js 通常鼓励开发人员沿着"数据驱动"的方式思考,避免直接接触 DOM,但是有时我们确实要这么做.比如一个新闻滚动的列表项.如果在这里需要操作dom, 应该是等待 V ...
- 使用linux部署tomcat项目
1.下载对应的Tomcat服务器包 Apache Tomcat官网下载: http://tomcat.apache.org/download-70.cgi 比如我们使用的是 apache-tomca ...
- c++DLL编程详解
DLL(Dynamic Link Library)的概念,你可以简单的把DLL看成一种仓库,它提供给你一些可以直接拿来用的变量.函数或类.在仓库的发展史上经历了“无库-静态链接库-动态链接库”的时代. ...
- PyCharm 2018.1破解过程
一.下载 首先从官网下载 官网,如果开了酸酸乳的话无法下载,官网会自动断开连接.所以下载时请关闭酸酸乳 二.安装 选择安装路径 选择64位,创建关联.py文件 安装完后运行Pycharm 选择不导入开 ...
- 克拉默法则(Cramer's Rule)的证明
克拉默法则: 先说一下为什么要写这个,作为一个大一新生,必须要学的就包括了线性代数,而且线性代数等数学知识对计算机专业也有很大帮助.但是在学习过程中遇到一个讲解的不清楚的知识点(Cramer's Ru ...
- asp.net使用session完成: 从哪个页面进入登录页面,登录成功还回到那个页面
1.在Login.aspx页面Load中加入 if (!IsPostBack && Request.UrlReferrer != null) { Session[ " ...
- python:浅析python 中__name__ = '__main__' 的作用(转载)
每次看文章的源码时,Python的主程序中都会看到开头有这个,查了一下作用:https://www.cnblogs.com/alan-babyblog/p/5147770.html. 讲的很详细,就直 ...
- Docker标准化开发测试和生产环境
对于大部分企业来说,搭建 PaaS 既没有那个精力,也没那个必要,用 Docker 做个人的 sandbox 用处又小了点. 可以用 Docker 来标准化开发.测试.生产环境. Docker 占用资 ...