| /* | 
 |  * 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; | 
 | } |