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