mirror of
https://github.com/reactos/reactos.git
synced 2024-11-05 06:09:58 +00:00
c424146e2c
svn path=/branches/cmake-bringup/; revision=48236
311 lines
9.9 KiB
C
311 lines
9.9 KiB
C
/*
|
|
* Digital video MCI Wine Driver
|
|
*
|
|
* Copyright 1999, 2000 Eric POUECH
|
|
* Copyright 2003 Dmitry Timoshkov
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include "private_mciavi.h"
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mciavi);
|
|
|
|
static const WCHAR mciaviW[] = {'M','C','I','A','V','I',0};
|
|
|
|
static LRESULT WINAPI MCIAVI_WindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
TRACE("hwnd=%p msg=%x wparam=%lx lparam=%lx\n", hWnd, uMsg, wParam, lParam);
|
|
|
|
switch (uMsg) {
|
|
case WM_CREATE:
|
|
SetWindowLongW(hWnd, 0, (LPARAM)((CREATESTRUCTW *)lParam)->lpCreateParams);
|
|
return DefWindowProcW(hWnd, uMsg, wParam, lParam);
|
|
|
|
case WM_DESTROY:
|
|
MCIAVI_mciClose(GetWindowLongW(hWnd, 0), MCI_WAIT, NULL);
|
|
SetWindowLongW(hWnd, 0, 0);
|
|
return DefWindowProcW(hWnd, uMsg, wParam, lParam);
|
|
|
|
case WM_ERASEBKGND:
|
|
{
|
|
RECT rect;
|
|
GetClientRect(hWnd, &rect);
|
|
FillRect((HDC)wParam, &rect, GetStockObject(BLACK_BRUSH));
|
|
}
|
|
return 1;
|
|
|
|
case WM_PAINT:
|
|
{
|
|
WINE_MCIAVI *wma = (WINE_MCIAVI *)mciGetDriverData(GetWindowLongW(hWnd, 0));
|
|
|
|
if (!wma)
|
|
return DefWindowProcW(hWnd, uMsg, wParam, lParam);
|
|
|
|
EnterCriticalSection(&wma->cs);
|
|
|
|
/* the animation isn't playing, don't paint */
|
|
if (wma->dwStatus == MCI_MODE_NOT_READY)
|
|
{
|
|
LeaveCriticalSection(&wma->cs);
|
|
/* default paint handling */
|
|
return DefWindowProcW(hWnd, uMsg, wParam, lParam);
|
|
}
|
|
|
|
if (wParam)
|
|
MCIAVI_PaintFrame(wma, (HDC)wParam);
|
|
else
|
|
{
|
|
PAINTSTRUCT ps;
|
|
BeginPaint(hWnd, &ps);
|
|
MCIAVI_PaintFrame(wma, ps.hdc);
|
|
EndPaint(hWnd, &ps);
|
|
}
|
|
|
|
LeaveCriticalSection(&wma->cs);
|
|
}
|
|
return 1;
|
|
|
|
default:
|
|
return DefWindowProcW(hWnd, uMsg, wParam, lParam);
|
|
}
|
|
}
|
|
|
|
BOOL MCIAVI_UnregisterClass(void)
|
|
{
|
|
return UnregisterClassW(mciaviW, MCIAVI_hInstance);
|
|
}
|
|
|
|
BOOL MCIAVI_RegisterClass(void)
|
|
{
|
|
WNDCLASSW wndClass;
|
|
|
|
ZeroMemory(&wndClass, sizeof(WNDCLASSW));
|
|
wndClass.style = CS_DBLCLKS;
|
|
wndClass.lpfnWndProc = MCIAVI_WindowProc;
|
|
wndClass.cbWndExtra = sizeof(MCIDEVICEID);
|
|
wndClass.hInstance = MCIAVI_hInstance;
|
|
wndClass.hCursor = LoadCursorW(0, (LPCWSTR)IDC_ARROW);
|
|
wndClass.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
|
|
wndClass.lpszClassName = mciaviW;
|
|
|
|
if (RegisterClassW(&wndClass)) return TRUE;
|
|
if (GetLastError() == ERROR_CLASS_ALREADY_EXISTS) return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL MCIAVI_CreateWindow(WINE_MCIAVI* wma, DWORD dwFlags, LPMCI_DGV_OPEN_PARMSW lpOpenParms)
|
|
{
|
|
static const WCHAR captionW[] = {'W','i','n','e',' ','M','C','I','-','A','V','I',' ','p','l','a','y','e','r',0};
|
|
HWND hParent = 0;
|
|
DWORD dwStyle = WS_OVERLAPPEDWINDOW;
|
|
RECT rc;
|
|
|
|
/* what should be done ? */
|
|
if (wma->hWnd) return TRUE;
|
|
|
|
if (dwFlags & MCI_DGV_OPEN_PARENT) hParent = lpOpenParms->hWndParent;
|
|
if (dwFlags & MCI_DGV_OPEN_WS) dwStyle = lpOpenParms->dwStyle;
|
|
|
|
rc.left = rc.top = 0;
|
|
rc.right = (wma->hic ? wma->outbih : wma->inbih)->biWidth;
|
|
rc.bottom = (wma->hic ? wma->outbih : wma->inbih)->biHeight;
|
|
AdjustWindowRect(&rc, dwStyle, FALSE);
|
|
if (!(dwStyle & (WS_CHILD|WS_POPUP))) /* overlapped window ? */
|
|
{
|
|
rc.right -= rc.left;
|
|
rc.bottom -= rc.top;
|
|
rc.left = rc.top = CW_USEDEFAULT;
|
|
}
|
|
|
|
wma->hWnd = CreateWindowW(mciaviW, captionW,
|
|
dwStyle, rc.left, rc.top,
|
|
rc.right, rc.bottom,
|
|
hParent, 0, MCIAVI_hInstance,
|
|
ULongToPtr(wma->wDevID));
|
|
wma->hWndPaint = wma->hWnd;
|
|
return wma->hWnd != 0;
|
|
}
|
|
|
|
/***************************************************************************
|
|
* MCIAVI_mciPut [internal]
|
|
*/
|
|
DWORD MCIAVI_mciPut(UINT wDevID, DWORD dwFlags, LPMCI_DGV_PUT_PARMS lpParms)
|
|
{
|
|
WINE_MCIAVI* wma = MCIAVI_mciGetOpenDev(wDevID);
|
|
RECT rc;
|
|
|
|
TRACE("(%04x, %08X, %p)\n", wDevID, dwFlags, lpParms);
|
|
|
|
if (lpParms == NULL) return MCIERR_NULL_PARAMETER_BLOCK;
|
|
if (wma == NULL) return MCIERR_INVALID_DEVICE_ID;
|
|
if (dwFlags & MCI_TEST) return 0;
|
|
|
|
EnterCriticalSection(&wma->cs);
|
|
|
|
if (dwFlags & MCI_DGV_RECT) {
|
|
/* In MCI, RECT structure is used differently: rc.right = width & rc.bottom = height
|
|
* So convert input MCI RECT into a normal RECT */
|
|
rc.left = lpParms->rc.left;
|
|
rc.top = lpParms->rc.top;
|
|
rc.right = lpParms->rc.left + lpParms->rc.right;
|
|
rc.bottom = lpParms->rc.top + lpParms->rc.bottom;
|
|
} else {
|
|
GetClientRect(wma->hWndPaint, &rc);
|
|
}
|
|
|
|
if (dwFlags & MCI_DGV_PUT_CLIENT) {
|
|
FIXME("PUT_CLIENT %s\n", wine_dbgstr_rect(&rc));
|
|
LeaveCriticalSection(&wma->cs);
|
|
return MCIERR_UNRECOGNIZED_COMMAND;
|
|
}
|
|
if (dwFlags & MCI_DGV_PUT_DESTINATION) {
|
|
TRACE("PUT_DESTINATION %s\n", wine_dbgstr_rect(&rc));
|
|
wma->dest = rc;
|
|
}
|
|
if (dwFlags & MCI_DGV_PUT_FRAME) {
|
|
FIXME("PUT_FRAME %s\n", wine_dbgstr_rect(&rc));
|
|
LeaveCriticalSection(&wma->cs);
|
|
return MCIERR_UNRECOGNIZED_COMMAND;
|
|
}
|
|
if (dwFlags & MCI_DGV_PUT_SOURCE) {
|
|
TRACE("PUT_SOURCE %s\n", wine_dbgstr_rect(&rc));
|
|
wma->source = rc;
|
|
}
|
|
if (dwFlags & MCI_DGV_PUT_VIDEO) {
|
|
FIXME("PUT_VIDEO %s\n", wine_dbgstr_rect(&rc));
|
|
LeaveCriticalSection(&wma->cs);
|
|
return MCIERR_UNRECOGNIZED_COMMAND;
|
|
}
|
|
if (dwFlags & MCI_DGV_PUT_WINDOW) {
|
|
TRACE("PUT_WINDOW %s\n", wine_dbgstr_rect(&rc));
|
|
SetWindowPos(wma->hWndPaint, NULL, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, SWP_NOZORDER);
|
|
}
|
|
LeaveCriticalSection(&wma->cs);
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* MCIAVI_mciWhere [internal]
|
|
*/
|
|
DWORD MCIAVI_mciWhere(UINT wDevID, DWORD dwFlags, LPMCI_DGV_RECT_PARMS lpParms)
|
|
{
|
|
WINE_MCIAVI* wma = MCIAVI_mciGetOpenDev(wDevID);
|
|
RECT rc;
|
|
|
|
TRACE("(%04x, %08x, %p)\n", wDevID, dwFlags, lpParms);
|
|
|
|
if (lpParms == NULL) return MCIERR_NULL_PARAMETER_BLOCK;
|
|
if (wma == NULL) return MCIERR_INVALID_DEVICE_ID;
|
|
/* Ignore MCI_TEST flag. */
|
|
|
|
EnterCriticalSection(&wma->cs);
|
|
|
|
if (dwFlags & MCI_DGV_WHERE_DESTINATION) {
|
|
if (dwFlags & MCI_DGV_WHERE_MAX) {
|
|
GetClientRect(wma->hWndPaint, &rc);
|
|
TRACE("WHERE_DESTINATION_MAX %s\n", wine_dbgstr_rect(&rc));
|
|
} else {
|
|
TRACE("WHERE_DESTINATION %s\n", wine_dbgstr_rect(&wma->dest));
|
|
rc = wma->dest;
|
|
}
|
|
}
|
|
if (dwFlags & MCI_DGV_WHERE_FRAME) {
|
|
if (dwFlags & MCI_DGV_WHERE_MAX)
|
|
FIXME("MCI_DGV_WHERE_FRAME_MAX\n");
|
|
else
|
|
FIXME("MCI_DGV_WHERE_FRAME\n");
|
|
LeaveCriticalSection(&wma->cs);
|
|
return MCIERR_UNRECOGNIZED_COMMAND;
|
|
}
|
|
if (dwFlags & MCI_DGV_WHERE_SOURCE) {
|
|
if (dwFlags & MCI_DGV_WHERE_MAX) {
|
|
rc.left = 0;
|
|
rc.top = 0;
|
|
rc.right = wma->inbih->biWidth;
|
|
rc.bottom = wma->inbih->biHeight;
|
|
TRACE("WHERE_SOURCE_MAX %s\n", wine_dbgstr_rect(&rc));
|
|
} else {
|
|
TRACE("WHERE_SOURCE %s\n", wine_dbgstr_rect(&wma->source));
|
|
rc = wma->source;
|
|
}
|
|
}
|
|
if (dwFlags & MCI_DGV_WHERE_VIDEO) {
|
|
if (dwFlags & MCI_DGV_WHERE_MAX)
|
|
FIXME("WHERE_VIDEO_MAX\n");
|
|
else
|
|
FIXME("WHERE_VIDEO\n");
|
|
LeaveCriticalSection(&wma->cs);
|
|
return MCIERR_UNRECOGNIZED_COMMAND;
|
|
}
|
|
if (dwFlags & MCI_DGV_WHERE_WINDOW) {
|
|
if (dwFlags & MCI_DGV_WHERE_MAX) {
|
|
GetWindowRect(GetDesktopWindow(), &rc);
|
|
TRACE("WHERE_WINDOW_MAX %s\n", wine_dbgstr_rect(&rc));
|
|
} else {
|
|
GetWindowRect(wma->hWndPaint, &rc);
|
|
TRACE("WHERE_WINDOW %s\n", wine_dbgstr_rect(&rc));
|
|
}
|
|
}
|
|
|
|
/* In MCI, RECT structure is used differently: rc.right = width & rc.bottom = height
|
|
* So convert the normal RECT into a MCI RECT before returning */
|
|
lpParms->rc.left = rc.left;
|
|
lpParms->rc.top = rc.top;
|
|
lpParms->rc.right = rc.right - rc.left;
|
|
lpParms->rc.bottom = rc.bottom - rc.top;
|
|
|
|
LeaveCriticalSection(&wma->cs);
|
|
return 0;
|
|
}
|
|
|
|
/***************************************************************************
|
|
* MCIAVI_mciWindow [internal]
|
|
*/
|
|
DWORD MCIAVI_mciWindow(UINT wDevID, DWORD dwFlags, LPMCI_DGV_WINDOW_PARMSW lpParms)
|
|
{
|
|
WINE_MCIAVI* wma = MCIAVI_mciGetOpenDev(wDevID);
|
|
|
|
TRACE("(%04x, %08X, %p)\n", wDevID, dwFlags, lpParms);
|
|
|
|
if (lpParms == NULL) return MCIERR_NULL_PARAMETER_BLOCK;
|
|
if (wma == NULL) return MCIERR_INVALID_DEVICE_ID;
|
|
if (dwFlags & MCI_TEST) return 0;
|
|
|
|
EnterCriticalSection(&wma->cs);
|
|
|
|
if (dwFlags & MCI_DGV_WINDOW_HWND) {
|
|
if (IsWindow(lpParms->hWnd))
|
|
{
|
|
TRACE("Setting hWnd to %p\n", lpParms->hWnd);
|
|
if (wma->hWnd) ShowWindow(wma->hWnd, SW_HIDE);
|
|
wma->hWndPaint = (lpParms->hWnd == MCI_DGV_WINDOW_DEFAULT) ? wma->hWnd : lpParms->hWnd;
|
|
}
|
|
}
|
|
if (dwFlags & MCI_DGV_WINDOW_STATE) {
|
|
TRACE("Setting nCmdShow to %d\n", lpParms->nCmdShow);
|
|
ShowWindow(wma->hWndPaint, lpParms->nCmdShow);
|
|
}
|
|
if (dwFlags & MCI_DGV_WINDOW_TEXT) {
|
|
TRACE("Setting caption to %s\n", debugstr_w(lpParms->lpstrText));
|
|
SetWindowTextW(wma->hWndPaint, lpParms->lpstrText);
|
|
}
|
|
|
|
LeaveCriticalSection(&wma->cs);
|
|
return 0;
|
|
}
|