截屏时如何获得正在播放的视频部份的图像?

lxcooi 2008-02-15 08:01:21
本以为很单的功能但是实际使用时却得不到视频播放器的图像部份。
太难了,谁能帮我?
GDI的Bitblt
DirectX的GetFrontBuffer
全都得不到,谁能帮我,百分求解
另说一句,我需要在局域网内实现高实时传输,所以Windows Media API中的直接录WMV方法不适合我。

欢迎大家讨论。
另说一句DirectX的GetFrontBuffer比GDI的Bitblt慢几十倍!微软说了“就是这么设计的!”大家别像我一样,实现了才发现,痛苦
原文:“This function is very slow, by design, and should not be used in any performance-critical path.”
...全文
325 点赞 收藏 10
写回复
10 条回复
切换为时间正序
请发表友善的回复…
发表回复
lxcooi 2008-02-20
感谢大家的帮助,受教了!
前两天太忙,现在结贴!
回复
qiangv 2008-02-18
ls的全面,受教了
回复
eduyu 2008-02-18
方法1:如你所说的mirror driver,也就是一个虚拟的display driver。这样当然彻底了。
方法2:如楼上说的关掉hardware acceleration。也就是强迫所有播放器视频输出到显存而非显卡的overlay设备。
通常鼠标Cursor和视频都是在overlay上和显存合成最后输出的,这也就是为何照不到视频和鼠标的原因了。
方法3:最简单的,MPC播放器是可以自己选择视频输出方式的,让MPC选择非overlay的输出方式。不影响系统其它软硬件。如果可以让用户选播放器,这个是最合适而且最容易实现的方式。
回复
rageliu 2008-02-18
这是由于其使用了硬件加速,所以和gdi方式完全不同,禁掉加速就可以抓到了
象mirror也可以
回复
china_bai 2008-02-17
我也曾经遇到过类似的问题。
当时是对Media player中正在播放的画面进行截屏,结果出来的结果总是黑屏。

后来好像是把Media player 控件的版本设置为6.0才搞定。
回复
lxcooi 2008-02-17
播放器的截取正在播放的画面使用的是获取当前帧的方法,在这里没有任何可参考的地方。

今天第一次截到了播放器的画面,使用了WINDDK中的镜像卡驱动。
不过安装镜像卡驱动后桌面显示有点问题,正在修改中。

明天结贴。
回复
eduyu 2008-02-16
我是说MPC内建截图代码,没用过MPC?暴风影音就是MPC的汉化版本。
回复
lxcooi 2008-02-16
兄弟们,我补充一下,我要作的大约是一个截屏软件,不是播放软件,也就是说我需要在获得整个桌面区域的同时得到在其中播放的视频图像。

感谢cnzdgs兄!
回复
eduyu 2008-02-16
参考Media player classic的实现,开源的。
回复
cnzdgs 2008-02-15
我以前写过的一个类,你参考一下吧。
class CSurface
{
protected:
LPDIRECTDRAW m_pdd;
LPDDSURFACEDESC m_pddsd;
LPDIRECTDRAWSURFACE m_pdds;
public:
CSurface();
virtual ~CSurface();
bool Create(void);
bool Lock(void);
bool Unlock(void);
void Release(void);
public:
DWORD m_dwBitCount;
DWORD m_dwWidth;
DWORD m_dwHeight;
DWORD m_dwLinearSize;
DWORD m_dwSize;
PPALETTEENTRY m_ppePalette;
PBYTE m_pData;
};

CSurface::CSurface()
{
m_pdd = NULL;
m_pdds = NULL;
m_pddsd = new DDSURFACEDESC;
m_ppePalette = new PALETTEENTRY[0x100];
if ((m_pddsd == NULL) || (m_ppePalette == NULL))
return;
if (DirectDrawCreate(NULL, &m_pdd, NULL) != DD_OK)
{
m_pdd = NULL;
return;
}
if (m_pdd->SetCooperativeLevel(NULL, DDSCL_NORMAL) != DD_OK)
{
m_pdd = NULL;
return;
}
}

CSurface::~CSurface()
{
Release();
delete m_pddsd;
delete m_ppePalette;
}

bool CSurface::Create()
{
if (m_pdd == NULL)
return false;
m_pddsd->dwSize = sizeof(DDSURFACEDESC);
m_pddsd->dwFlags = DDSD_CAPS;
m_pddsd->ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
if (m_pdd->CreateSurface(m_pddsd, &m_pdds, NULL) != DD_OK)
return false;
if (m_pdds->Lock(NULL, m_pddsd, DDLOCK_WAIT | DDLOCK_READONLY, NULL) != DD_OK)
{
m_pdds->Release();
return false;
}
m_pdds->Unlock(NULL);
m_dwWidth = m_pddsd->dwWidth;
m_dwHeight = m_pddsd->dwHeight;
m_dwLinearSize = m_pddsd->dwLinearSize;
m_dwSize = m_dwLinearSize*m_dwHeight;
m_dwBitCount = m_pddsd->ddpfPixelFormat.dwRGBBitCount;
if (m_dwBitCount <= 8)
{
HDC hDC;
int nPaletteLength;
bool bResult;

if (m_pdds->GetDC(&hDC) != DD_OK)
{
m_pdds->Release();
return false;
}
if (m_dwBitCount == 8)
nPaletteLength = 256;
else
nPaletteLength = 16;
bResult = (::GetSystemPaletteEntries(hDC, 0, nPaletteLength, m_ppePalette) != 0);
m_pdds->ReleaseDC(hDC);
if (!bResult)
{
m_pdds->Release();
return false;
}
}
return true;
}

void CSurface::Release()
{
if (m_pdds != NULL)
m_pdds->Release();
m_pdds = NULL;
}

bool CSurface::Lock()
{
if ((m_pdd == NULL) || (m_pdds == NULL))
return false;
if (m_pdds->Lock(NULL, m_pddsd, DDLOCK_WAIT | DDLOCK_READONLY, NULL) != DD_OK)
return false;
m_pData = (PBYTE)m_pddsd->lpSurface;
m_pdds->Unlock(NULL);
return true;
}

回复
发动态
发帖子
图形处理/算法
创建于2007-09-28

1.9w+

社区成员

VC/MFC 图形处理/算法
申请成为版主
社区公告
暂无公告