xref: /reactos/dll/win32/msvfw32/mciwnd.c (revision d5b576b2)
1 /*
2  * Copyright 2000 Eric Pouech
3  * Copyright 2003 Dmitry Timoshkov
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18  *
19  * FIXME:
20  * Add support for all remaining MCI_ commands and MCIWNDM_ messages.
21  * Add support for MCIWNDF_RECORD.
22  */
23 
24 #include <stdarg.h>
25 
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winnls.h"
29 #include "wingdi.h"
30 #include "winuser.h"
31 #include "winternl.h"
32 #include "vfw.h"
33 #include "digitalv.h"
34 #include "commctrl.h"
35 #include "wine/unicode.h"
36 #include "wine/debug.h"
37 
38 WINE_DEFAULT_DEBUG_CHANNEL(mci);
39 
40 extern HMODULE MSVFW32_hModule;
41 static const WCHAR mciWndClassW[] = {'M','C','I','W','n','d','C','l','a','s','s',0};
42 
43 typedef struct
44 {
45     DWORD       dwStyle;
46     MCIDEVICEID mci;
47     HDRVR       hdrv;
48     int         alias;
49     UINT        dev_type;
50     UINT        mode;
51     LONG        position;
52     SIZE        size; /* size of the original frame rect */
53     int         zoom;
54     LPWSTR      lpName;
55     HWND        hWnd, hwndOwner;
56     UINT        uTimer;
57     MCIERROR    lasterror;
58     WCHAR       return_string[128];
59     WORD        active_timer, inactive_timer;
60 } MCIWndInfo;
61 
62 static LRESULT WINAPI MCIWndProc(HWND hWnd, UINT wMsg, WPARAM wParam, LPARAM lParam);
63 
64 #define CTL_PLAYSTOP    0x3200
65 #define CTL_MENU        0x3201
66 #define CTL_TRACKBAR    0x3202
67 
68 /***********************************************************************
69  *                MCIWndRegisterClass                [MSVFW32.@]
70  *
71  * NOTE: Native always uses its own hInstance
72  */
73 BOOL VFWAPIV MCIWndRegisterClass(void)
74 {
75     WNDCLASSW wc;
76 
77     /* Since we are going to register a class belonging to MSVFW32
78      * and later we will create windows with a different hInstance
79      * CS_GLOBALCLASS is needed. And because the second attempt
80      * to register a global class will fail we need to test whether
81      * the class was already registered.
82      */
83     wc.style = CS_VREDRAW | CS_HREDRAW | CS_DBLCLKS | CS_OWNDC | CS_GLOBALCLASS;
84     wc.lpfnWndProc = MCIWndProc;
85     wc.cbClsExtra = 0;
86     wc.cbWndExtra = sizeof(MCIWndInfo*);
87     wc.hInstance = MSVFW32_hModule;
88     wc.hIcon = 0;
89     wc.hCursor = LoadCursorW(0, (LPWSTR)IDC_ARROW);
90     wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
91     wc.lpszMenuName = NULL;
92     wc.lpszClassName = mciWndClassW;
93 
94     if (RegisterClassW(&wc)) return TRUE;
95     if (GetLastError() == ERROR_CLASS_ALREADY_EXISTS) return TRUE;
96 
97     return FALSE;
98 }
99 
100 /***********************************************************************
101  *                MCIWndCreateW                                [MSVFW32.@]
102  */
103 HWND VFWAPIV MCIWndCreateW(HWND hwndParent, HINSTANCE hInstance,
104                            DWORD dwStyle, LPCWSTR szFile)
105 {
106     TRACE("%p %p %x %s\n", hwndParent, hInstance, dwStyle, debugstr_w(szFile));
107 
108     MCIWndRegisterClass();
109 
110     if (!hInstance) hInstance = GetModuleHandleW(0);
111 
112     if (hwndParent)
113         dwStyle |= WS_VISIBLE | WS_BORDER /*| WS_CHILD*/;
114     else
115         dwStyle |= WS_VISIBLE | WS_CAPTION | WS_SYSMENU | WS_THICKFRAME | WS_MINIMIZEBOX | WS_MAXIMIZEBOX;
116 
117     return CreateWindowExW(0, mciWndClassW, NULL,
118                            dwStyle | WS_CLIPSIBLINGS | WS_CLIPCHILDREN,
119                            0, 0, 300, 0,
120                            hwndParent, 0, hInstance, (LPVOID)szFile);
121 }
122 
123 /***********************************************************************
124  *                MCIWndCreate                [MSVFW32.@]
125  *                MCIWndCreateA                [MSVFW32.@]
126  */
127 HWND VFWAPIV MCIWndCreateA(HWND hwndParent, HINSTANCE hInstance,
128                            DWORD dwStyle, LPCSTR szFile)
129 {
130     HWND ret;
131     UNICODE_STRING fileW;
132 
133     if (szFile)
134         RtlCreateUnicodeStringFromAsciiz(&fileW, szFile);
135     else
136         fileW.Buffer = NULL;
137 
138     ret = MCIWndCreateW(hwndParent, hInstance, dwStyle, fileW.Buffer);
139 
140     RtlFreeUnicodeString(&fileW);
141     return ret;
142 }
143 
144 static inline void MCIWND_notify_mode(MCIWndInfo *mwi)
145 {
146     if (mwi->dwStyle & MCIWNDF_NOTIFYMODE)
147     {
148         UINT new_mode = SendMessageW(mwi->hWnd, MCIWNDM_GETMODEW, 0, 0);
149         if (new_mode != mwi->mode)
150         {
151             mwi->mode = new_mode;
152             SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYMODE, (WPARAM)mwi->hWnd, new_mode);
153         }
154     }
155 }
156 
157 static inline void MCIWND_notify_pos(MCIWndInfo *mwi)
158 {
159     if (mwi->dwStyle & MCIWNDF_NOTIFYPOS)
160     {
161         LONG new_pos = SendMessageW(mwi->hWnd, MCIWNDM_GETPOSITIONW, 0, 0);
162         if (new_pos != mwi->position)
163         {
164             mwi->position = new_pos;
165             SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYPOS, (WPARAM)mwi->hWnd, new_pos);
166         }
167     }
168 }
169 
170 static inline void MCIWND_notify_size(MCIWndInfo *mwi)
171 {
172     if (mwi->dwStyle & MCIWNDF_NOTIFYSIZE)
173         SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYSIZE, (WPARAM)mwi->hWnd, 0);
174 }
175 
176 static inline void MCIWND_notify_error(MCIWndInfo *mwi)
177 {
178     if (mwi->dwStyle & MCIWNDF_NOTIFYERROR)
179         SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYERROR, (WPARAM)mwi->hWnd, (LPARAM)mwi->lasterror);
180 }
181 
182 static void MCIWND_UpdateState(MCIWndInfo *mwi)
183 {
184     WCHAR buffer[1024];
185 
186     if (!mwi->mci)
187     {
188         /* FIXME: get this from resources */
189         static const WCHAR no_deviceW[] = {'N','o',' ','D','e','v','i','c','e',0};
190         SetWindowTextW(mwi->hWnd, no_deviceW);
191         return;
192     }
193 
194     MCIWND_notify_pos(mwi);
195 
196     if (!(mwi->dwStyle & MCIWNDF_NOPLAYBAR))
197         SendDlgItemMessageW(mwi->hWnd, CTL_TRACKBAR, TBM_SETPOS, TRUE, mwi->position);
198 
199     if (!(mwi->dwStyle & MCIWNDF_SHOWALL))
200         return;
201 
202     if ((mwi->dwStyle & MCIWNDF_SHOWNAME) && mwi->lpName)
203         strcpyW(buffer, mwi->lpName);
204     else
205         *buffer = 0;
206 
207     if (mwi->dwStyle & (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE))
208     {
209         static const WCHAR spaceW[] = {' ',0};
210         static const WCHAR l_braceW[] = {'(',0};
211 
212         if (*buffer) strcatW(buffer, spaceW);
213         strcatW(buffer, l_braceW);
214     }
215 
216     if (mwi->dwStyle & MCIWNDF_SHOWPOS)
217     {
218         WCHAR posW[64];
219 
220         posW[0] = 0;
221         SendMessageW(mwi->hWnd, MCIWNDM_GETPOSITIONW, 64, (LPARAM)posW);
222         strcatW(buffer, posW);
223     }
224 
225     if ((mwi->dwStyle & (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE)) == (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE))
226     {
227         static const WCHAR dashW[] = {' ','-',' ',0};
228         strcatW(buffer, dashW);
229     }
230 
231     if (mwi->dwStyle & MCIWNDF_SHOWMODE)
232     {
233         WCHAR modeW[64];
234 
235         modeW[0] = 0;
236         SendMessageW(mwi->hWnd, MCIWNDM_GETMODEW, 64, (LPARAM)modeW);
237         strcatW(buffer, modeW);
238     }
239 
240     if (mwi->dwStyle & (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE))
241     {
242         static const WCHAR r_braceW[] = {')',0};
243         strcatW(buffer, r_braceW);
244     }
245 
246     TRACE("=> %s\n", debugstr_w(buffer));
247     SetWindowTextW(mwi->hWnd, buffer);
248 }
249 
250 static LRESULT MCIWND_Create(HWND hWnd, LPCREATESTRUCTW cs)
251 {
252     HWND hChld;
253     MCIWndInfo *mwi;
254     static const WCHAR buttonW[] = {'b','u','t','t','o','n',0};
255 
256     mwi = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*mwi));
257     if (!mwi) return -1;
258 
259     SetWindowLongW(hWnd, 0, (LPARAM)mwi);
260 
261     mwi->dwStyle = cs->style;
262     /* There is no need to show stats if there is no caption */
263     if ((mwi->dwStyle & WS_CAPTION) != WS_CAPTION)
264         mwi->dwStyle &= ~MCIWNDF_SHOWALL;
265 
266     mwi->hWnd = hWnd;
267     mwi->hwndOwner = cs->hwndParent;
268     mwi->active_timer = 500;
269     mwi->inactive_timer = 2000;
270     mwi->mode = MCI_MODE_NOT_READY;
271     mwi->position = -1;
272     mwi->zoom = 100;
273 
274     if (!(mwi->dwStyle & MCIWNDF_NOMENU))
275     {
276         static const WCHAR menuW[] = {'M','e','n','u',0};
277 
278         hChld = CreateWindowExW(0, buttonW, menuW, WS_CHILD|WS_VISIBLE, 32, cs->cy, 32, 32,
279                                 hWnd, (HMENU)CTL_MENU, cs->hInstance, 0L);
280         TRACE("Get Button2: %p\n", hChld);
281     }
282 
283     if (!(mwi->dwStyle & MCIWNDF_NOPLAYBAR))
284     {
285         INITCOMMONCONTROLSEX init;
286         static const WCHAR playW[] = {'P','l','a','y',0};
287 
288         /* adding the other elements: play/stop button, menu button, status */
289         hChld = CreateWindowExW(0, buttonW, playW, WS_CHILD|WS_VISIBLE, 0, cs->cy, 32, 32,
290                                 hWnd, (HMENU)CTL_PLAYSTOP, cs->hInstance, 0L);
291         TRACE("Get Button1: %p\n", hChld);
292 
293         init.dwSize = sizeof(init);
294         init.dwICC = ICC_BAR_CLASSES;
295         InitCommonControlsEx(&init);
296 
297         hChld = CreateWindowExW(0, TRACKBAR_CLASSW, NULL, WS_CHILD|WS_VISIBLE, 64, cs->cy, cs->cx - 64, 32,
298                                 hWnd, (HMENU)CTL_TRACKBAR, cs->hInstance, 0L);
299         TRACE("Get status: %p\n", hChld);
300     }
301 
302     /* This sets the default window size */
303     SendMessageW(hWnd, MCI_CLOSE, 0, 0);
304 
305     if (cs->lpCreateParams)
306     {
307         LPARAM lParam;
308 
309         /* MCI wnd class is prepared to be embedded as an MDI child window */
310         if (cs->dwExStyle & WS_EX_MDICHILD)
311         {
312             MDICREATESTRUCTW *mdics = cs->lpCreateParams;
313             lParam = mdics->lParam;
314         }
315         else
316             lParam = (LPARAM)cs->lpCreateParams;
317 
318         /* If it's our internal class pointer, file name is a unicode string */
319         if (cs->lpszClass == mciWndClassW)
320             SendMessageW(hWnd, MCIWNDM_OPENW, 0, lParam);
321         else
322         {
323             /* Otherwise let's try to figure out what string format is used */
324             HWND parent = cs->hwndParent;
325             if (!parent) parent = GetWindow(hWnd, GW_OWNER);
326 
327             SendMessageW(hWnd, IsWindowUnicode(parent) ? MCIWNDM_OPENW : MCIWNDM_OPENA, 0, lParam);
328         }
329     }
330 
331     return 0;
332 }
333 
334 static void MCIWND_ToggleState(MCIWndInfo *mwi)
335 {
336     switch (SendMessageW(mwi->hWnd, MCIWNDM_GETMODEW, 0, 0))
337     {
338     case MCI_MODE_NOT_READY:
339     case MCI_MODE_RECORD:
340     case MCI_MODE_SEEK:
341     case MCI_MODE_OPEN:
342         TRACE("Cannot do much...\n");
343         break;
344 
345     case MCI_MODE_PAUSE:
346         SendMessageW(mwi->hWnd, MCI_RESUME, 0, 0);
347         break;
348 
349     case MCI_MODE_PLAY:
350         SendMessageW(mwi->hWnd, MCI_PAUSE, 0, 0);
351         break;
352 
353     case MCI_MODE_STOP:
354         SendMessageW(mwi->hWnd, MCI_STOP, 0, 0);
355         break;
356     }
357 }
358 
359 static LRESULT MCIWND_Command(MCIWndInfo *mwi, WPARAM wParam, LPARAM lParam)
360 {
361     switch (LOWORD(wParam))
362     {
363     case CTL_PLAYSTOP: MCIWND_ToggleState(mwi); break;
364     case CTL_MENU:
365     case CTL_TRACKBAR:
366     default:
367         FIXME("support for command %04x not implement yet\n", LOWORD(wParam));
368     }
369     return 0L;
370 }
371 
372 static void MCIWND_notify_media(MCIWndInfo *mwi)
373 {
374     if (mwi->dwStyle & (MCIWNDF_NOTIFYMEDIAA | MCIWNDF_NOTIFYMEDIAW))
375     {
376         if (!mwi->lpName)
377         {
378             static const WCHAR empty_str[1];
379             SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYMEDIA, (WPARAM)mwi->hWnd, (LPARAM)empty_str);
380         }
381         else
382         {
383             if (mwi->dwStyle & MCIWNDF_NOTIFYANSI)
384             {
385                 char *ansi_name;
386                 int len;
387 
388                 len = WideCharToMultiByte(CP_ACP, 0, mwi->lpName, -1, NULL, 0, NULL, NULL);
389                 ansi_name = HeapAlloc(GetProcessHeap(), 0, len);
390                 WideCharToMultiByte(CP_ACP, 0, mwi->lpName, -1, ansi_name, len, NULL, NULL);
391 
392                 SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYMEDIA, (WPARAM)mwi->hWnd, (LPARAM)ansi_name);
393 
394                 HeapFree(GetProcessHeap(), 0, ansi_name);
395             }
396             else
397                 SendMessageW(mwi->hwndOwner, MCIWNDM_NOTIFYMEDIA, (WPARAM)mwi->hWnd, (LPARAM)mwi->lpName);
398         }
399     }
400 }
401 
402 static MCIERROR mci_generic_command(MCIWndInfo *mwi, UINT cmd)
403 {
404     MCI_GENERIC_PARMS mci_generic;
405 
406     mci_generic.dwCallback = 0;
407     mwi->lasterror = mciSendCommandW(mwi->mci, cmd, 0, (DWORD_PTR)&mci_generic);
408 
409     if (mwi->lasterror)
410         return mwi->lasterror;
411 
412     MCIWND_notify_mode(mwi);
413     MCIWND_UpdateState(mwi);
414     return 0;
415 }
416 
417 static LRESULT mci_get_devcaps(MCIWndInfo *mwi, UINT cap)
418 {
419     MCI_GETDEVCAPS_PARMS mci_devcaps;
420 
421     mci_devcaps.dwItem = cap;
422     mwi->lasterror = mciSendCommandW(mwi->mci, MCI_GETDEVCAPS,
423                                    MCI_GETDEVCAPS_ITEM,
424                                    (DWORD_PTR)&mci_devcaps);
425     if (mwi->lasterror)
426         return 0;
427 
428     return mci_devcaps.dwReturn;
429 }
430 
431 static LRESULT MCIWND_KeyDown(MCIWndInfo *mwi, UINT key)
432 {
433     TRACE("%p, key %04x\n", mwi->hWnd, key);
434 
435     switch(key)
436     {
437     case VK_ESCAPE:
438         SendMessageW(mwi->hWnd, MCI_STOP, 0, 0);
439         return 0;
440 
441     default:
442         return 0;
443     }
444 }
445 
446 static LRESULT WINAPI MCIWndProc(HWND hWnd, UINT wMsg, WPARAM wParam, LPARAM lParam)
447 {
448     MCIWndInfo *mwi;
449 
450     TRACE("%p %04x %08lx %08lx\n", hWnd, wMsg, wParam, lParam);
451 
452     mwi = (MCIWndInfo*)GetWindowLongPtrW(hWnd, 0);
453     if (!mwi && wMsg != WM_CREATE)
454         return DefWindowProcW(hWnd, wMsg, wParam, lParam);
455 
456     switch (wMsg)
457     {
458     case WM_CREATE:
459         MCIWND_Create(hWnd, (CREATESTRUCTW *)lParam);
460         break;
461 
462     case WM_DESTROY:
463         if (mwi->uTimer)
464             KillTimer(hWnd, mwi->uTimer);
465 
466         if (mwi->mci)
467             SendMessageW(hWnd, MCI_CLOSE, 0, 0);
468 
469         HeapFree(GetProcessHeap(), 0, mwi);
470 
471         DestroyWindow(GetDlgItem(hWnd, CTL_MENU));
472         DestroyWindow(GetDlgItem(hWnd, CTL_PLAYSTOP));
473         DestroyWindow(GetDlgItem(hWnd, CTL_TRACKBAR));
474         break;
475 
476     case WM_PAINT:
477         {
478             MCI_DGV_UPDATE_PARMS mci_update;
479             PAINTSTRUCT ps;
480 
481             mci_update.hDC = (wParam) ? (HDC)wParam : BeginPaint(hWnd, &ps);
482 
483             mciSendCommandW(mwi->mci, MCI_UPDATE,
484                             MCI_DGV_UPDATE_HDC | MCI_DGV_UPDATE_PAINT,
485                             (DWORD_PTR)&mci_update);
486 
487             if (!wParam) EndPaint(hWnd, &ps);
488             return 1;
489         }
490 
491     case WM_COMMAND:
492         return MCIWND_Command(mwi, wParam, lParam);
493 
494     case WM_KEYDOWN:
495         return MCIWND_KeyDown(mwi, wParam);
496 
497     case WM_NCACTIVATE:
498         if (mwi->uTimer)
499         {
500             KillTimer(hWnd, mwi->uTimer);
501             mwi->uTimer = SetTimer(hWnd, 1, wParam ? mwi->active_timer : mwi->inactive_timer, NULL);
502         }
503         break;
504 
505     case WM_TIMER:
506         MCIWND_UpdateState(mwi);
507         return 0;
508 
509     case WM_SIZE:
510         SetWindowPos(GetDlgItem(hWnd, CTL_PLAYSTOP), 0, 0, HIWORD(lParam) - 32, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
511         SetWindowPos(GetDlgItem(hWnd, CTL_MENU), 0, 32, HIWORD(lParam) - 32, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
512         SetWindowPos(GetDlgItem(hWnd, CTL_TRACKBAR), 0, 64, HIWORD(lParam) - 32, LOWORD(lParam) - 64, 32, SWP_NOACTIVATE);
513 
514         if (!(mwi->dwStyle & MCIWNDF_NOAUTOSIZEMOVIE))
515         {
516             RECT rc;
517 
518             rc.left = rc.top = 0;
519             rc.right = LOWORD(lParam);
520             rc.bottom = HIWORD(lParam);
521             if (!(mwi->dwStyle & MCIWNDF_NOPLAYBAR))
522                 rc.bottom -= 32; /* subtract the height of the playbar */
523             SendMessageW(hWnd, MCIWNDM_PUT_DEST, 0, (LPARAM)&rc);
524         }
525         MCIWND_notify_size(mwi);
526         break;
527 
528     case MM_MCINOTIFY:
529         MCIWND_notify_mode(mwi);
530         MCIWND_UpdateState(mwi);
531         return 0;
532 
533     case MCIWNDM_OPENA:
534         {
535             UNICODE_STRING nameW;
536             TRACE("MCIWNDM_OPENA %s\n", debugstr_a((LPSTR)lParam));
537             RtlCreateUnicodeStringFromAsciiz(&nameW, (LPCSTR)lParam);
538             lParam = (LPARAM)nameW.Buffer;
539         }
540         /* fall through */
541     case MCIWNDM_OPENW:
542         {
543             RECT rc;
544             HCURSOR hCursor;
545             MCI_OPEN_PARMSW mci_open;
546             MCI_GETDEVCAPS_PARMS mci_devcaps;
547             WCHAR aliasW[64];
548             WCHAR drv_name[MAX_PATH];
549             static const WCHAR formatW[] = {'%','d',0};
550             static const WCHAR mci32W[] = {'m','c','i','3','2',0};
551             static const WCHAR system_iniW[] = {'s','y','s','t','e','m','.','i','n','i',0};
552 
553             TRACE("MCIWNDM_OPENW %s\n", debugstr_w((LPWSTR)lParam));
554 
555             if (wParam == MCIWNDOPENF_NEW)
556             {
557                 SendMessageW(hWnd, MCIWNDM_NEWW, 0, lParam);
558                 goto end_of_mci_open;
559             }
560 
561             if (mwi->uTimer)
562             {
563                 KillTimer(hWnd, mwi->uTimer);
564                 mwi->uTimer = 0;
565             }
566 
567             hCursor = LoadCursorW(0, (LPWSTR)IDC_WAIT);
568             hCursor = SetCursor(hCursor);
569 
570             mci_open.lpstrElementName = (LPWSTR)lParam;
571             wsprintfW(aliasW, formatW, HandleToLong(hWnd) + 1);
572             mci_open.lpstrAlias = aliasW;
573             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_OPEN,
574                                              MCI_OPEN_ELEMENT | MCI_OPEN_ALIAS | MCI_WAIT,
575                                              (DWORD_PTR)&mci_open);
576             SetCursor(hCursor);
577 
578             if (mwi->lasterror && !(mwi->dwStyle & MCIWNDF_NOERRORDLG))
579             {
580                 /* FIXME: get the caption from resources */
581                 static const WCHAR caption[] = {'M','C','I',' ','E','r','r','o','r',0};
582                 WCHAR error_str[MAXERRORLENGTH];
583 
584                 mciGetErrorStringW(mwi->lasterror, error_str, MAXERRORLENGTH);
585                 MessageBoxW(hWnd, error_str, caption, MB_ICONEXCLAMATION | MB_OK);
586                 MCIWND_notify_error(mwi);
587                 goto end_of_mci_open;
588             }
589 
590             mwi->mci = mci_open.wDeviceID;
591             mwi->alias = HandleToLong(hWnd) + 1;
592 
593             mwi->lpName = HeapAlloc(GetProcessHeap(), 0, (strlenW((LPWSTR)lParam) + 1) * sizeof(WCHAR));
594             strcpyW(mwi->lpName, (LPWSTR)lParam);
595 
596             MCIWND_UpdateState(mwi);
597 
598             mci_devcaps.dwItem = MCI_GETDEVCAPS_DEVICE_TYPE;
599             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_GETDEVCAPS,
600                                              MCI_GETDEVCAPS_ITEM,
601                                              (DWORD_PTR)&mci_devcaps);
602             if (mwi->lasterror)
603             {
604                 MCIWND_notify_error(mwi);
605                 goto end_of_mci_open;
606             }
607 
608             mwi->dev_type = mci_devcaps.dwReturn;
609 
610             drv_name[0] = 0;
611             SendMessageW(hWnd, MCIWNDM_GETDEVICEW, 256, (LPARAM)drv_name);
612             if (drv_name[0] && GetPrivateProfileStringW(mci32W, drv_name, NULL,
613                                             drv_name, MAX_PATH, system_iniW))
614                 mwi->hdrv = OpenDriver(drv_name, NULL, 0);
615 
616             if (mwi->dev_type == MCI_DEVTYPE_DIGITAL_VIDEO)
617             {
618                 MCI_DGV_WINDOW_PARMSW mci_window;
619 
620                 mci_window.hWnd = hWnd;
621                 mwi->lasterror = mciSendCommandW(mwi->mci, MCI_WINDOW,
622                                                  MCI_DGV_WINDOW_HWND,
623                                                  (DWORD_PTR)&mci_window);
624                 if (mwi->lasterror)
625                 {
626                     MCIWND_notify_error(mwi);
627                     goto end_of_mci_open;
628                 }
629             }
630 
631             if (SendMessageW(hWnd, MCIWNDM_GET_DEST, 0, (LPARAM)&rc) == 0)
632             {
633                 mwi->size.cx = rc.right - rc.left;
634                 mwi->size.cy = rc.bottom - rc.top;
635 
636                 rc.right = MulDiv(mwi->size.cx, mwi->zoom, 100);
637                 rc.bottom = MulDiv(mwi->size.cy, mwi->zoom, 100);
638                 SendMessageW(hWnd, MCIWNDM_PUT_DEST, 0, (LPARAM)&rc);
639             }
640             else
641             {
642                 GetClientRect(hWnd, &rc);
643                 rc.bottom = rc.top;
644             }
645 
646             if (!(mwi->dwStyle & MCIWNDF_NOPLAYBAR))
647                 rc.bottom += 32; /* add the height of the playbar */
648             AdjustWindowRect(&rc, GetWindowLongW(hWnd, GWL_STYLE), FALSE);
649             SetWindowPos(hWnd, 0, 0, 0, rc.right - rc.left,
650                          rc.bottom - rc.top, SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE);
651 
652             SendDlgItemMessageW(hWnd, CTL_TRACKBAR, TBM_SETRANGEMIN, 0L, 0L);
653             SendDlgItemMessageW(hWnd, CTL_TRACKBAR, TBM_SETRANGEMAX, 1,
654                                 SendMessageW(hWnd, MCIWNDM_GETLENGTH, 0, 0));
655             mwi->uTimer = SetTimer(hWnd, 1, mwi->active_timer, NULL);
656 
657             MCIWND_notify_media(mwi);
658 
659 end_of_mci_open:
660             if (wMsg == MCIWNDM_OPENA)
661                 HeapFree(GetProcessHeap(), 0, (void *)lParam);
662             return mwi->lasterror;
663         }
664 
665     case MCIWNDM_GETDEVICEID:
666         TRACE("MCIWNDM_GETDEVICEID\n");
667         return mwi->mci;
668 
669     case MCIWNDM_GETALIAS:
670         TRACE("MCIWNDM_GETALIAS\n");
671         return mwi->alias;
672 
673     case MCIWNDM_GET_SOURCE:
674         {
675             MCI_DGV_RECT_PARMS mci_rect;
676 
677             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_WHERE,
678                                              MCI_DGV_WHERE_SOURCE,
679                                              (DWORD_PTR)&mci_rect);
680             if (mwi->lasterror)
681             {
682                 MCIWND_notify_error(mwi);
683                 return mwi->lasterror;
684             }
685             *(RECT *)lParam = mci_rect.rc;
686             TRACE("MCIWNDM_GET_SOURCE: %s\n", wine_dbgstr_rect(&mci_rect.rc));
687             return 0;
688         }
689 
690     case MCIWNDM_GET_DEST:
691         {
692             MCI_DGV_RECT_PARMS mci_rect;
693 
694             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_WHERE,
695                                              MCI_DGV_WHERE_DESTINATION,
696                                              (DWORD_PTR)&mci_rect);
697             if (mwi->lasterror)
698             {
699                 MCIWND_notify_error(mwi);
700                 return mwi->lasterror;
701             }
702             *(RECT *)lParam = mci_rect.rc;
703             TRACE("MCIWNDM_GET_DEST: %s\n", wine_dbgstr_rect(&mci_rect.rc));
704             return 0;
705         }
706 
707     case MCIWNDM_PUT_SOURCE:
708         {
709             MCI_DGV_PUT_PARMS mci_put;
710 
711             mci_put.rc = *(RECT *)lParam;
712             TRACE("MCIWNDM_PUT_SOURCE: %s\n", wine_dbgstr_rect(&mci_put.rc));
713             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_PUT,
714                                              MCI_DGV_PUT_SOURCE,
715                                              (DWORD_PTR)&mci_put);
716             if (mwi->lasterror)
717             {
718                 MCIWND_notify_error(mwi);
719                 return mwi->lasterror;
720             }
721             return 0;
722         }
723 
724     case MCIWNDM_PUT_DEST:
725         {
726             MCI_DGV_PUT_PARMS mci_put;
727 
728             mci_put.rc = *(RECT *)lParam;
729             TRACE("MCIWNDM_PUT_DEST: %s\n", wine_dbgstr_rect(&mci_put.rc));
730 
731             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_PUT,
732                                              MCI_DGV_PUT_DESTINATION | MCI_DGV_RECT,
733                                              (DWORD_PTR)&mci_put);
734             if (mwi->lasterror)
735             {
736                 MCIWND_notify_error(mwi);
737                 return mwi->lasterror;
738             }
739             return 0;
740         }
741 
742     case MCIWNDM_GETLENGTH:
743         {
744             MCI_STATUS_PARMS mci_status;
745 
746             mci_status.dwItem = MCI_STATUS_LENGTH;
747             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_STATUS,
748                                              MCI_STATUS_ITEM,
749                                              (DWORD_PTR)&mci_status);
750             if (mwi->lasterror)
751             {
752                 MCIWND_notify_error(mwi);
753                 return 0;
754             }
755             TRACE("MCIWNDM_GETLENGTH: %ld\n", mci_status.dwReturn);
756             return mci_status.dwReturn;
757         }
758 
759     case MCIWNDM_GETSTART:
760         {
761             MCI_STATUS_PARMS mci_status;
762 
763             mci_status.dwItem = MCI_STATUS_POSITION;
764             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_STATUS,
765                                              MCI_STATUS_ITEM | MCI_STATUS_START,
766                                              (DWORD_PTR)&mci_status);
767             if (mwi->lasterror)
768             {
769                 MCIWND_notify_error(mwi);
770                 return 0;
771             }
772             TRACE("MCIWNDM_GETSTART: %ld\n", mci_status.dwReturn);
773             return mci_status.dwReturn;
774         }
775 
776     case MCIWNDM_GETEND:
777         {
778             LRESULT start, length;
779 
780             start = SendMessageW(hWnd, MCIWNDM_GETSTART, 0, 0);
781             length = SendMessageW(hWnd, MCIWNDM_GETLENGTH, 0, 0);
782             TRACE("MCIWNDM_GETEND: %ld\n", start + length);
783             return (start + length);
784         }
785 
786     case MCIWNDM_GETPOSITIONA:
787     case MCIWNDM_GETPOSITIONW:
788         {
789             MCI_STATUS_PARMS mci_status;
790 
791             TRACE("MCIWNDM_GETPOSITION\n");
792 
793             /* get position string if requested */
794             if (wParam && lParam)
795             {
796                 if (wMsg == MCIWNDM_GETPOSITIONA)
797                 {
798                     char cmd[64];
799 
800                     wsprintfA(cmd, "status %d position", mwi->alias);
801                     mwi->lasterror = mciSendStringA(cmd, (LPSTR)lParam, wParam, 0);
802                 }
803                 else
804                 {
805 
806                     WCHAR cmdW[64];
807                     static const WCHAR formatW[] = {'s','t','a','t','u','s',' ','%','d',' ','p','o','s','i','t','i','o','n',0};
808 
809                     wsprintfW(cmdW, formatW, mwi->alias);
810                     mwi->lasterror = mciSendStringW(cmdW, (LPWSTR)lParam, wParam, 0);
811                 }
812 
813                 if (mwi->lasterror)
814                     return 0;
815             }
816 
817             mci_status.dwItem = MCI_STATUS_POSITION;
818             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_STATUS,
819                                              MCI_STATUS_ITEM,
820                                              (DWORD_PTR)&mci_status);
821             if (mwi->lasterror)
822                 return 0;
823 
824             return mci_status.dwReturn;
825         }
826 
827     case MCIWNDM_GETMODEA:
828     case MCIWNDM_GETMODEW:
829         {
830             MCI_STATUS_PARMS mci_status;
831 
832             TRACE("MCIWNDM_GETMODE\n");
833 
834             if (!mwi->mci)
835                 return MCI_MODE_NOT_READY;
836 
837             /* get mode string if requested */
838             if (wParam && lParam)
839             {
840                 if (wMsg == MCIWNDM_GETMODEA)
841                 {
842                     char cmd[64];
843 
844                     wsprintfA(cmd, "status %d mode", mwi->alias);
845                     mwi->lasterror = mciSendStringA(cmd, (LPSTR)lParam, wParam, 0);
846                 }
847                 else
848                 {
849 
850                     WCHAR cmdW[64];
851                     static const WCHAR formatW[] = {'s','t','a','t','u','s',' ','%','d',' ','m','o','d','e',0};
852 
853                     wsprintfW(cmdW, formatW, mwi->alias);
854                     mwi->lasterror = mciSendStringW(cmdW, (LPWSTR)lParam, wParam, 0);
855                 }
856 
857                 if (mwi->lasterror)
858                     return MCI_MODE_NOT_READY;
859             }
860 
861             mci_status.dwItem = MCI_STATUS_MODE;
862             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_STATUS,
863                                              MCI_STATUS_ITEM,
864                                              (DWORD_PTR)&mci_status);
865             if (mwi->lasterror)
866                 return MCI_MODE_NOT_READY;
867 
868             return mci_status.dwReturn;
869         }
870 
871     case MCIWNDM_PLAYFROM:
872         {
873             MCI_PLAY_PARMS mci_play;
874 
875             TRACE("MCIWNDM_PLAYFROM %08lx\n", lParam);
876 
877             mci_play.dwCallback = (DWORD_PTR)hWnd;
878             mci_play.dwFrom = lParam;
879             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_PLAY,
880                                              MCI_FROM | MCI_NOTIFY,
881                                              (DWORD_PTR)&mci_play);
882             if (mwi->lasterror)
883             {
884                 MCIWND_notify_error(mwi);
885                 return mwi->lasterror;
886             }
887 
888             MCIWND_notify_mode(mwi);
889             MCIWND_UpdateState(mwi);
890             return 0;
891         }
892 
893     case MCIWNDM_PLAYTO:
894         {
895             MCI_PLAY_PARMS mci_play;
896 
897             TRACE("MCIWNDM_PLAYTO %08lx\n", lParam);
898 
899             mci_play.dwCallback = (DWORD_PTR)hWnd;
900             mci_play.dwTo = lParam;
901             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_PLAY,
902                                              MCI_TO | MCI_NOTIFY,
903                                              (DWORD_PTR)&mci_play);
904             if (mwi->lasterror)
905             {
906                 MCIWND_notify_error(mwi);
907                 return mwi->lasterror;
908             }
909 
910             MCIWND_notify_mode(mwi);
911             MCIWND_UpdateState(mwi);
912             return 0;
913         }
914 
915     case MCIWNDM_PLAYREVERSE:
916         {
917             MCI_PLAY_PARMS mci_play;
918             DWORD flags = MCI_NOTIFY;
919 
920             TRACE("MCIWNDM_PLAYREVERSE %08lx\n", lParam);
921 
922             mci_play.dwCallback = (DWORD_PTR)hWnd;
923             mci_play.dwFrom = lParam;
924             switch (mwi->dev_type)
925             {
926             default:
927             case MCI_DEVTYPE_ANIMATION:
928                 flags |= MCI_ANIM_PLAY_REVERSE;
929                 break;
930 
931             case MCI_DEVTYPE_DIGITAL_VIDEO:
932                 flags |= MCI_DGV_PLAY_REVERSE;
933                 break;
934 
935 #ifdef MCI_VCR_PLAY_REVERSE
936             case MCI_DEVTYPE_VCR:
937                 flags |= MCI_VCR_PLAY_REVERSE;
938                 break;
939 #endif
940 
941             case MCI_DEVTYPE_VIDEODISC:
942                 flags |= MCI_VD_PLAY_REVERSE;
943                 break;
944 
945             }
946             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_PLAY,
947                                              flags, (DWORD_PTR)&mci_play);
948             if (mwi->lasterror)
949             {
950                 MCIWND_notify_error(mwi);
951                 return mwi->lasterror;
952             }
953 
954             MCIWND_notify_mode(mwi);
955             MCIWND_UpdateState(mwi);
956             return 0;
957         }
958 
959     case MCIWNDM_GETERRORA:
960         mciGetErrorStringA(mwi->lasterror, (LPSTR)lParam, wParam);
961         TRACE("MCIWNDM_GETERRORA: %s\n", debugstr_an((LPSTR)lParam, wParam));
962         return mwi->lasterror;
963 
964     case MCIWNDM_GETERRORW:
965         mciGetErrorStringW(mwi->lasterror, (LPWSTR)lParam, wParam);
966         TRACE("MCIWNDM_GETERRORW: %s\n", debugstr_wn((LPWSTR)lParam, wParam));
967         return mwi->lasterror;
968 
969     case MCIWNDM_SETOWNER:
970         TRACE("MCIWNDM_SETOWNER %p\n", (HWND)wParam);
971         mwi->hwndOwner = (HWND)wParam;
972         return 0;
973 
974     case MCIWNDM_SENDSTRINGA:
975         {
976             UNICODE_STRING stringW;
977 
978             TRACE("MCIWNDM_SENDSTRINGA %s\n", debugstr_a((LPCSTR)lParam));
979 
980             RtlCreateUnicodeStringFromAsciiz(&stringW, (LPCSTR)lParam);
981             lParam = (LPARAM)stringW.Buffer;
982         }
983         /* fall through */
984     case MCIWNDM_SENDSTRINGW:
985         {
986             WCHAR *cmdW, *p;
987 
988             TRACE("MCIWNDM_SENDSTRINGW %s\n", debugstr_w((LPCWSTR)lParam));
989 
990             p = strchrW((LPCWSTR)lParam, ' ');
991             if (p)
992             {
993                 static const WCHAR formatW[] = {'%','d',' ',0};
994                 int len, pos;
995 
996                 pos = p - (WCHAR *)lParam + 1;
997                 len = lstrlenW((LPCWSTR)lParam) + 64;
998 
999                 cmdW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
1000 
1001                 memcpy(cmdW, (void *)lParam, pos * sizeof(WCHAR));
1002                 wsprintfW(cmdW + pos, formatW, mwi->alias);
1003                 strcatW(cmdW, (WCHAR *)lParam + pos);
1004             }
1005             else
1006                 cmdW = (LPWSTR)lParam;
1007 
1008             mwi->lasterror = mciSendStringW(cmdW, mwi->return_string,
1009                                             sizeof(mwi->return_string)/sizeof(mwi->return_string[0]),
1010                                             0);
1011             if (mwi->lasterror)
1012                 MCIWND_notify_error(mwi);
1013 
1014             if (cmdW != (LPWSTR)lParam)
1015                 HeapFree(GetProcessHeap(), 0, cmdW);
1016 
1017             if (wMsg == MCIWNDM_SENDSTRINGA)
1018                 HeapFree(GetProcessHeap(), 0, (void *)lParam);
1019 
1020             MCIWND_UpdateState(mwi);
1021             return mwi->lasterror;
1022         }
1023 
1024     case MCIWNDM_RETURNSTRINGA:
1025         WideCharToMultiByte(CP_ACP, 0, mwi->return_string, -1, (LPSTR)lParam, wParam, NULL, NULL);
1026         TRACE("MCIWNDM_RETURNTRINGA %s\n", debugstr_an((LPSTR)lParam, wParam));
1027         return mwi->lasterror;
1028 
1029     case MCIWNDM_RETURNSTRINGW:
1030         lstrcpynW((LPWSTR)lParam, mwi->return_string, wParam);
1031         TRACE("MCIWNDM_RETURNTRINGW %s\n", debugstr_wn((LPWSTR)lParam, wParam));
1032         return mwi->lasterror;
1033 
1034     case MCIWNDM_SETTIMERS:
1035         TRACE("MCIWNDM_SETTIMERS active %d ms, inactive %d ms\n", (int)wParam, (int)lParam);
1036         mwi->active_timer = (WORD)wParam;
1037         mwi->inactive_timer = (WORD)lParam;
1038         return 0;
1039 
1040     case MCIWNDM_SETACTIVETIMER:
1041         TRACE("MCIWNDM_SETACTIVETIMER %d ms\n", (int)wParam);
1042         mwi->active_timer = (WORD)wParam;
1043         return 0;
1044 
1045     case MCIWNDM_SETINACTIVETIMER:
1046         TRACE("MCIWNDM_SETINACTIVETIMER %d ms\n", (int)wParam);
1047         mwi->inactive_timer = (WORD)wParam;
1048         return 0;
1049 
1050     case MCIWNDM_GETACTIVETIMER:
1051         TRACE("MCIWNDM_GETACTIVETIMER: %d ms\n", mwi->active_timer);
1052         return mwi->active_timer;
1053 
1054     case MCIWNDM_GETINACTIVETIMER:
1055         TRACE("MCIWNDM_GETINACTIVETIMER: %d ms\n", mwi->inactive_timer);
1056         return mwi->inactive_timer;
1057 
1058     case MCIWNDM_CHANGESTYLES:
1059         TRACE("MCIWNDM_CHANGESTYLES mask %08lx, set %08lx\n", wParam, lParam);
1060         /* FIXME: update the visual window state as well:
1061          * add/remove trackbar, autosize, etc.
1062          */
1063         mwi->dwStyle &= ~wParam;
1064         mwi->dwStyle |= lParam & wParam;
1065         return 0;
1066 
1067     case MCIWNDM_GETSTYLES:
1068         TRACE("MCIWNDM_GETSTYLES: %08x\n", mwi->dwStyle & 0xffff);
1069         return mwi->dwStyle & 0xffff;
1070 
1071     case MCIWNDM_GETDEVICEA:
1072         {
1073             int len = 0;
1074             char *str = (char *)lParam;
1075             MCI_SYSINFO_PARMSA mci_sysinfo;
1076 
1077             mci_sysinfo.lpstrReturn = str;
1078             mci_sysinfo.dwRetSize = wParam;
1079             mwi->lasterror = mciSendCommandA(mwi->mci, MCI_SYSINFO,
1080                                              MCI_SYSINFO_INSTALLNAME,
1081                                              (DWORD_PTR)&mci_sysinfo);
1082             while(len < wParam && str[len]) len++;
1083             TRACE("MCIWNDM_GETDEVICEA: %s\n", debugstr_an(str, len));
1084             return 0;
1085         }
1086 
1087     case MCIWNDM_GETDEVICEW:
1088         {
1089             int len = 0;
1090             WCHAR *str = (WCHAR *)lParam;
1091             MCI_SYSINFO_PARMSW mci_sysinfo;
1092 
1093             mci_sysinfo.lpstrReturn = str;
1094             mci_sysinfo.dwRetSize = wParam;
1095             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_SYSINFO,
1096                                              MCI_SYSINFO_INSTALLNAME,
1097                                              (DWORD_PTR)&mci_sysinfo);
1098             while(len < wParam && str[len]) len++;
1099             TRACE("MCIWNDM_GETDEVICEW: %s\n", debugstr_wn(str, len));
1100             return 0;
1101         }
1102 
1103     case MCIWNDM_VALIDATEMEDIA:
1104         TRACE("MCIWNDM_VALIDATEMEDIA\n");
1105         if (mwi->mci)
1106         {
1107             SendMessageW(hWnd, MCIWNDM_GETSTART, 0, 0);
1108             SendMessageW(hWnd, MCIWNDM_GETLENGTH, 0, 0);
1109         }
1110         return 0;
1111 
1112     case MCIWNDM_GETFILENAMEA:
1113         TRACE("MCIWNDM_GETFILENAMEA: %s\n", debugstr_w(mwi->lpName));
1114         if (mwi->lpName)
1115             WideCharToMultiByte(CP_ACP, 0, mwi->lpName, -1, (LPSTR)lParam, wParam, NULL, NULL);
1116         return 0;
1117 
1118     case MCIWNDM_GETFILENAMEW:
1119         TRACE("MCIWNDM_GETFILENAMEW: %s\n", debugstr_w(mwi->lpName));
1120         if (mwi->lpName)
1121             lstrcpynW((LPWSTR)lParam, mwi->lpName, wParam);
1122         return 0;
1123 
1124     case MCIWNDM_GETTIMEFORMATA:
1125     case MCIWNDM_GETTIMEFORMATW:
1126         {
1127             MCI_STATUS_PARMS mci_status;
1128 
1129             TRACE("MCIWNDM_GETTIMEFORMAT %08lx %08lx\n", wParam, lParam);
1130 
1131             /* get format string if requested */
1132             if (wParam && lParam)
1133             {
1134                 if (wMsg == MCIWNDM_GETTIMEFORMATA)
1135                 {
1136                     char cmd[64];
1137 
1138                     wsprintfA(cmd, "status %d time format", mwi->alias);
1139                     mwi->lasterror = mciSendStringA(cmd, (LPSTR)lParam, wParam, 0);
1140                     if (mwi->lasterror)
1141                         return 0;
1142                 }
1143                 else
1144                 {
1145                     WCHAR cmdW[64];
1146                     static const WCHAR formatW[] = {'s','t','a','t','u','s',' ','%','d',' ','t','i','m','e',' ','f','o','r','m','a','t',0};
1147 
1148                     wsprintfW(cmdW, formatW, mwi->alias);
1149                     mwi->lasterror = mciSendStringW(cmdW, (LPWSTR)lParam, wParam, 0);
1150                     if (mwi->lasterror)
1151                         return 0;
1152                 }
1153             }
1154 
1155             mci_status.dwItem = MCI_STATUS_TIME_FORMAT ;
1156             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_STATUS,
1157                                              MCI_STATUS_ITEM,
1158                                              (DWORD_PTR)&mci_status);
1159             if (mwi->lasterror)
1160                 return 0;
1161 
1162             return mci_status.dwReturn;
1163         }
1164 
1165     case MCIWNDM_SETTIMEFORMATA:
1166         {
1167             UNICODE_STRING stringW;
1168 
1169             TRACE("MCIWNDM_SETTIMEFORMATA %s\n", debugstr_a((LPSTR)lParam));
1170 
1171             RtlCreateUnicodeStringFromAsciiz(&stringW, (LPCSTR)lParam);
1172             lParam = (LPARAM)stringW.Buffer;
1173         }
1174         /* fall through */
1175     case MCIWNDM_SETTIMEFORMATW:
1176         {
1177             static const WCHAR formatW[] = {'s','e','t',' ','%','d',' ','t','i','m','e',' ','f','o','r','m','a','t',' ',0};
1178             WCHAR *cmdW;
1179 
1180             TRACE("MCIWNDM_SETTIMEFORMATW %s\n", debugstr_w((LPWSTR)lParam));
1181 
1182             if (mwi->mci)
1183             {
1184                 cmdW = HeapAlloc(GetProcessHeap(), 0, (lstrlenW((LPCWSTR)lParam) + 64) * sizeof(WCHAR));
1185                 wsprintfW(cmdW, formatW, mwi->alias);
1186                 strcatW(cmdW, (WCHAR *)lParam);
1187 
1188                 mwi->lasterror = mciSendStringW(cmdW, NULL, 0, 0);
1189 
1190                 /* fix the range tracking according to the new time format */
1191                 if (!mwi->lasterror)
1192                     SendDlgItemMessageW(hWnd, CTL_TRACKBAR, TBM_SETRANGEMAX, 1,
1193                                         SendMessageW(hWnd, MCIWNDM_GETLENGTH, 0, 0));
1194 
1195                 HeapFree(GetProcessHeap(), 0, cmdW);
1196             }
1197 
1198             if (wMsg == MCIWNDM_SETTIMEFORMATA)
1199                 HeapFree(GetProcessHeap(), 0, (void *)lParam);
1200 
1201             return 0;
1202         }
1203 
1204     case MCIWNDM_CAN_PLAY:
1205         TRACE("MCIWNDM_CAN_PLAY\n");
1206         if (mwi->mci)
1207             return mci_get_devcaps(mwi, MCI_GETDEVCAPS_CAN_PLAY);
1208         return 0;
1209 
1210     case MCIWNDM_CAN_RECORD:
1211         TRACE("MCIWNDM_CAN_RECORD\n");
1212         if (mwi->mci)
1213             return mci_get_devcaps(mwi, MCI_GETDEVCAPS_CAN_RECORD);
1214         return 0;
1215 
1216     case MCIWNDM_CAN_SAVE:
1217         TRACE("MCIWNDM_CAN_SAVE\n");
1218         if (mwi->mci)
1219             return mci_get_devcaps(mwi, MCI_GETDEVCAPS_CAN_SAVE);
1220         return 0;
1221 
1222     case MCIWNDM_CAN_EJECT:
1223         TRACE("MCIWNDM_CAN_EJECT\n");
1224         if (mwi->mci)
1225             return mci_get_devcaps(mwi, MCI_GETDEVCAPS_CAN_EJECT);
1226         return 0;
1227 
1228     case MCIWNDM_CAN_WINDOW:
1229         TRACE("MCIWNDM_CAN_WINDOW\n");
1230         switch (mwi->dev_type)
1231         {
1232         case MCI_DEVTYPE_ANIMATION:
1233         case MCI_DEVTYPE_DIGITAL_VIDEO:
1234         case MCI_DEVTYPE_OVERLAY:
1235             return 1;
1236         }
1237         return 0;
1238 
1239     case MCIWNDM_CAN_CONFIG:
1240         TRACE("MCIWNDM_CAN_CONFIG\n");
1241         if (mwi->hdrv)
1242             return SendDriverMessage(mwi->hdrv, DRV_QUERYCONFIGURE, 0, 0);
1243         return 0;
1244 
1245     case MCIWNDM_SETZOOM:
1246         TRACE("MCIWNDM_SETZOOM %ld\n", lParam);
1247         mwi->zoom = lParam;
1248 
1249         if (mwi->mci && !(mwi->dwStyle & MCIWNDF_NOAUTOSIZEWINDOW))
1250         {
1251             RECT rc;
1252 
1253             rc.left = rc.top = 0;
1254             rc.right = MulDiv(mwi->size.cx, mwi->zoom, 100);
1255             rc.bottom = MulDiv(mwi->size.cy, mwi->zoom, 100);
1256 
1257             if (!(mwi->dwStyle & MCIWNDF_NOPLAYBAR))
1258                 rc.bottom += 32; /* add the height of the playbar */
1259             AdjustWindowRect(&rc, GetWindowLongW(hWnd, GWL_STYLE), FALSE);
1260             SetWindowPos(hWnd, 0, 0, 0, rc.right - rc.left, rc.bottom - rc.top,
1261                          SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE);
1262         }
1263         return 0;
1264 
1265     case MCIWNDM_GETZOOM:
1266         TRACE("MCIWNDM_GETZOOM: %d\n", mwi->zoom);
1267         return mwi->zoom;
1268 
1269     case MCIWNDM_EJECT:
1270         {
1271             MCI_SET_PARMS mci_set;
1272 
1273             TRACE("MCIWNDM_EJECT\n");
1274 
1275             mci_set.dwCallback = (DWORD_PTR)hWnd;
1276             mwi->lasterror = mciSendCommandW(mwi->mci, MCI_SET,
1277                                              MCI_SET_DOOR_OPEN | MCI_NOTIFY,
1278                                              (DWORD_PTR)&mci_set);
1279             MCIWND_notify_mode(mwi);
1280             MCIWND_UpdateState(mwi);
1281             return mwi->lasterror;
1282         }
1283 
1284     case MCIWNDM_SETVOLUME:
1285     case MCIWNDM_GETVOLUME:
1286     case MCIWNDM_SETSPEED:
1287     case MCIWNDM_GETSPEED:
1288     case MCIWNDM_SETREPEAT:
1289     case MCIWNDM_GETREPEAT:
1290     case MCIWNDM_REALIZE:
1291     case MCIWNDM_GETPALETTE:
1292     case MCIWNDM_SETPALETTE:
1293     case MCIWNDM_NEWA:
1294     case MCIWNDM_NEWW:
1295     case MCIWNDM_PALETTEKICK:
1296     case MCIWNDM_OPENINTERFACE:
1297         FIXME("support for MCIWNDM_ message WM_USER+%d not implemented\n", wMsg - WM_USER);
1298         return 0;
1299 
1300     case MCI_PLAY:
1301         {
1302             LRESULT end = SendMessageW(hWnd, MCIWNDM_GETEND, 0, 0);
1303             return SendMessageW(hWnd, MCIWNDM_PLAYTO, 0, end);
1304         }
1305 
1306     case MCI_SEEK:
1307     case MCI_STEP:
1308         {
1309             MCI_SEEK_PARMS mci_seek; /* Layout is usable as MCI_XYZ_STEP_PARMS */
1310             DWORD flags = MCI_STEP == wMsg ? 0 :
1311                           MCIWND_START == lParam ? MCI_SEEK_TO_START :
1312                           MCIWND_END   == lParam ? MCI_SEEK_TO_END : MCI_TO;
1313 
1314             mci_seek.dwTo = lParam;
1315             mwi->lasterror = mciSendCommandW(mwi->mci, wMsg,
1316                                              flags, (DWORD_PTR)&mci_seek);
1317             if (mwi->lasterror)
1318             {
1319                 MCIWND_notify_error(mwi);
1320                 return mwi->lasterror;
1321             }
1322             /* update window to reflect the state */
1323             else InvalidateRect(hWnd, NULL, TRUE);
1324             return 0;
1325         }
1326 
1327     case MCI_CLOSE:
1328         {
1329             RECT rc;
1330             MCI_GENERIC_PARMS mci_generic;
1331 
1332             if (mwi->hdrv)
1333             {
1334                 CloseDriver(mwi->hdrv, 0, 0);
1335                 mwi->hdrv = 0;
1336             }
1337 
1338             if (mwi->mci)
1339             {
1340                 mci_generic.dwCallback = 0;
1341                 mwi->lasterror = mciSendCommandW(mwi->mci, MCI_CLOSE,
1342                                                  0, (DWORD_PTR)&mci_generic);
1343                 mwi->mci = 0;
1344             }
1345 
1346             mwi->mode = MCI_MODE_NOT_READY;
1347             mwi->position = -1;
1348 
1349             HeapFree(GetProcessHeap(), 0, mwi->lpName);
1350             mwi->lpName = NULL;
1351             MCIWND_UpdateState(mwi);
1352 
1353             GetClientRect(hWnd, &rc);
1354             rc.bottom = rc.top;
1355             if (!(mwi->dwStyle & MCIWNDF_NOPLAYBAR))
1356                 rc.bottom += 32; /* add the height of the playbar */
1357             AdjustWindowRect(&rc, GetWindowLongW(hWnd, GWL_STYLE), FALSE);
1358             SetWindowPos(hWnd, 0, 0, 0, rc.right - rc.left,
1359                          rc.bottom - rc.top, SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE);
1360 
1361             MCIWND_notify_media(mwi);
1362             return 0;
1363         }
1364 
1365     case MCI_PAUSE:
1366     case MCI_STOP:
1367     case MCI_RESUME:
1368         mci_generic_command(mwi, wMsg);
1369         return mwi->lasterror;
1370 
1371     case MCI_CONFIGURE:
1372         if (mwi->hdrv)
1373             SendDriverMessage(mwi->hdrv, DRV_CONFIGURE, (LPARAM)hWnd, 0);
1374         return 0;
1375 
1376     case MCI_BREAK:
1377     case MCI_CAPTURE:
1378     case MCI_COPY:
1379     case MCI_CUE:
1380     case MCI_CUT:
1381     case MCI_DELETE:
1382     case MCI_ESCAPE:
1383     case MCI_FREEZE:
1384     case MCI_GETDEVCAPS:
1385     /*case MCI_INDEX:*/
1386     case MCI_INFO:
1387     case MCI_LIST:
1388     case MCI_LOAD:
1389     /*case MCI_MARK:*/
1390     case MCI_MONITOR:
1391     case MCI_OPEN:
1392     case MCI_PASTE:
1393     case MCI_PUT:
1394     case MCI_QUALITY:
1395     case MCI_REALIZE:
1396     case MCI_RECORD:
1397     case MCI_RESERVE:
1398     case MCI_RESTORE:
1399     case MCI_SAVE:
1400     case MCI_SET:
1401     case MCI_SETAUDIO:
1402     /*case MCI_SETTIMECODE:*/
1403     /*case MCI_SETTUNER:*/
1404     case MCI_SETVIDEO:
1405     case MCI_SIGNAL:
1406     case MCI_SPIN:
1407     case MCI_STATUS:
1408     case MCI_SYSINFO:
1409     case MCI_UNDO:
1410     case MCI_UNFREEZE:
1411     case MCI_UPDATE:
1412     case MCI_WHERE:
1413     case MCI_WINDOW:
1414         FIXME("support for MCI_ command %04x not implemented\n", wMsg);
1415         return 0;
1416     }
1417 
1418     if (wMsg >= WM_USER)
1419     {
1420         FIXME("support for MCIWNDM_ message WM_USER+%d not implemented\n", wMsg - WM_USER);
1421         return 0;
1422     }
1423 
1424     if (GetWindowLongW(hWnd, GWL_EXSTYLE) & WS_EX_MDICHILD)
1425         return DefMDIChildProcW(hWnd, wMsg, wParam, lParam);
1426 
1427     return DefWindowProcW(hWnd, wMsg, wParam, lParam);
1428 }
1429