Capture to Play Object Interfacing for C++
The following code demonstrates grabbing samples from the capture object and feeding them to the play object. It is intended to demonstrate the basic function set that would be required to remote the playback of media samples.
#include "stdafx.h"
// include the LEAD Multimedia TOOLKIT header
#include "ltmm.h"
#include "resource.h"
#include <tchar.h>
#include <stdio.h>
#include <math.h>
#include <assert.h>
#define SZ_WNDCLASS_PLAY _T("PLAY WNDCLASS")
#define WM_PLAYNOTIFY (WM_USER + 1000)
HINSTANCE g_hInstance; // application instance handle
HWND g_hwndPlay; // video frame window
IltmmPlay* g_pPlay; // play object interface pointer
IltmmCapture* g_pCapture; // capture object interface pointer
IltmmSampleTarget* g_pTarget; // target object, used with capture
IltmmSampleSource* g_pSource; // source object, used with play
//
// PlayWndProc
// video frame window procedure
//
LRESULT CALLBACK PlayWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_CREATE:
g_hwndPlay = hwnd;
// window is the video window frame
g_pPlay->put_VideoWindowFrame((long) hwnd);
// assign the media source
g_pPlay->put_SourceObject(g_pSource);
#ifdef _DEBUG
{
IUnknown* punk;
g_pPlay->get_SourceObject(&punk);
assert(punk != NULL);
if(punk)
punk->Release();
}
#endif
return 0;
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
}
return DefWindowProc(hwnd, message, wParam, lParam);
}
int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
MSG msg;
HRESULT hr;
WNDCLASSEX wcex;
IltmmDevices* pDevices;
VARIANT var;
IltmmMediaTypeDisp* pMediaType;
BSTR bstr;
long index;
IltmmCompressors* pCompressors;
g_hInstance = hInstance;
// initialize COM library
hr = CoInitialize(NULL);
if(FAILED(hr))
goto error;
// register the video frame window class
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = PlayWndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = g_hInstance;
wcex.hIcon = NULL;
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH) (COLOR_APPWORKSPACE + 1);
wcex.lpszMenuName = NULL;
wcex.lpszClassName = SZ_WNDCLASS_PLAY;
wcex.hIconSm = NULL;
if(!RegisterClassEx(&wcex))
goto error;
// create the play object
hr = CoCreateInstance(CLSID_ltmmPlay, NULL, CLSCTX_INPROC_SERVER, IID_IltmmPlay, (void**) &g_pPlay);
if(FAILED(hr))
goto error;
// create the capture object
hr = CoCreateInstance(CLSID_ltmmCapture, NULL, CLSCTX_INPROC_SERVER, IID_IltmmCapture, (void**) &g_pCapture);
if(FAILED(hr))
goto error;
// create the target object
hr = CoCreateInstance(CLSID_ltmmSampleTarget, NULL, CLSCTX_INPROC_SERVER, IID_IltmmSampleTarget, (void**) &g_pTarget);
if(FAILED(hr))
goto error;
// create the source object
hr = CoCreateInstance(CLSID_ltmmSampleSource, NULL, CLSCTX_INPROC_SERVER, IID_IltmmSampleSource, (void**) &g_pSource);
if(FAILED(hr))
goto error;
// no preview
g_pCapture->put_Preview(VARIANT_FALSE);
// select the first video device
g_pCapture->get_VideoDevices(&pDevices);
pDevices->put_Selection(0);
pDevices->Release();
// select the LEAD video compressor
g_pCapture->get_VideoCompressors(&pCompressors);
bstr = SysAllocString(L"@device:sw:{33D9A760-90C8-11D0-BD43-00A0C911CE86}\\LEAD MCMP/MJPEG Codec (2.0)");
pCompressors->Find(bstr, &index);
SysFreeString(bstr);
pCompressors->put_Selection(index);
pCompressors->Release();
// set the capture output object
g_pCapture->put_TargetObject(g_pTarget);
#ifdef _DEBUG
{
IUnknown* punk;
g_pCapture->get_TargetObject(&punk);
assert(punk != NULL);
if(punk)
punk->Release();
}
#endif
// start running
hr = g_pCapture->StartCapture(ltmmCapture_Mode_Video);
if(FAILED(hr))
goto error;
// retirve the video media type
g_pTarget->GetConnectedMediaType(&pMediaType);
// could just assign the media type to the source object,
// but let's get the portable type instead
pMediaType->GetPortableType(&var);
// have the portable type, don't need this anymore
pMediaType->Release();
// create a new media type
CoCreateInstance(CLSID_ltmmMediaType, NULL, CLSCTX_INPROC_SERVER, IID_IltmmMediaTypeDisp, (void**) &pMediaType);
// initialize it with the portable type
pMediaType->SetPortableType(var);
// assign it to the source object
g_pSource->SetMediaType(pMediaType);
#ifdef _DEBUG
{
IltmmMediaTypeDisp* punk;
g_pSource->GetMediaType( &punk);
assert(punk != NULL);
if(punk)
punk->Release();
}
#endif
// release the media type
pMediaType->Release();
// done with portable type
VariantClear(&var);
// create the video frame window
if(!CreateWindow(SZ_WNDCLASS_PLAY, _T("Play"), WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, g_hInstance, NULL))
goto error;
ShowWindow(g_hwndPlay, nCmdShow);
UpdateWindow(g_hwndPlay);
// process until done
for(;;)
{
if(PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE))
{
if(!GetMessage(&msg, NULL, 0, 0))
break;
TranslateMessage(&msg);
DispatchMessage(&msg);
}
else
{
IltmmMediaSampleDisp* pSample;
// fetch a sample
hr = g_pTarget->GetSample(1000, &pSample);
if(FAILED(hr))
break;
// deliver it
hr = g_pSource->DeliverSample(1000, pSample);
pSample->Release();
if(FAILED(hr))
goto error;
}
}
g_pCapture->StopCapture();
g_pSource->DeliverEndOfStream(1000);
g_pPlay->Stop();
error:
if(g_pSource)
g_pSource->Release();
if(g_pTarget)
g_pTarget->Release();
if(g_pCapture)
g_pCapture->Release();
if(g_pPlay)
g_pPlay->Release();
CoUninitialize();
return 0;
}