348 lines
9.8 KiB
C
348 lines
9.8 KiB
C
/*
|
|
* Copyright 2000 Eric Pouech
|
|
*
|
|
*
|
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include "winbase.h"
|
|
#include "windef.h"
|
|
#include "winnls.h"
|
|
#include "wingdi.h"
|
|
#include "winuser.h"
|
|
#include "vfw.h"
|
|
#include "digitalv.h"
|
|
#include "commctrl.h"
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mci);
|
|
|
|
typedef struct {
|
|
DWORD dwStyle;
|
|
MCIDEVICEID mci;
|
|
LPSTR lpName;
|
|
HWND hWnd;
|
|
UINT uTimer;
|
|
} MCIWndInfo;
|
|
|
|
static LRESULT WINAPI MCIWndProc(HWND hWnd, UINT wMsg, WPARAM lParam1, LPARAM lParam2);
|
|
|
|
#define CTL_PLAYSTOP 0x3200
|
|
#define CTL_MENU 0x3201
|
|
#define CTL_TRACKBAR 0x3202
|
|
|
|
/***********************************************************************
|
|
* MCIWndRegisterClass [MSVFW32.@]
|
|
*/
|
|
BOOL WINAPI MCIWndRegisterClass(HINSTANCE hInst)
|
|
{
|
|
WNDCLASSA wc;
|
|
|
|
/* since window creation will also require some common controls, init them */
|
|
InitCommonControls();
|
|
|
|
wc.style = 0;
|
|
wc.lpfnWndProc = MCIWndProc;
|
|
wc.cbClsExtra = 0;
|
|
wc.cbWndExtra = sizeof(MCIWndInfo*);
|
|
wc.hInstance = hInst;
|
|
wc.hIcon = 0;
|
|
wc.hCursor = 0;
|
|
wc.hbrBackground = 0;
|
|
wc.lpszMenuName = NULL;
|
|
wc.lpszClassName = "MCIWndClass";
|
|
|
|
return RegisterClassA(&wc);
|
|
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MCIWndCreate [MSVFW32.@]
|
|
* MCIWndCreateA [MSVFW32.@]
|
|
*/
|
|
HWND VFWAPIV MCIWndCreateA(HWND hwndParent, HINSTANCE hInstance,
|
|
DWORD dwStyle, LPCSTR szFile)
|
|
{
|
|
DWORD wndStyle;
|
|
MCIWndInfo* mwi;
|
|
|
|
TRACE("%p %p %lx %s\n", hwndParent, hInstance, dwStyle, szFile);
|
|
|
|
MCIWndRegisterClass(hInstance);
|
|
|
|
mwi = HeapAlloc(GetProcessHeap(), 0, sizeof(*mwi));
|
|
if (!mwi) return 0;
|
|
|
|
mwi->dwStyle = dwStyle;
|
|
if (szFile)
|
|
mwi->lpName = strcpy(HeapAlloc(GetProcessHeap(), 0, strlen(szFile) + 1), szFile);
|
|
else
|
|
mwi->lpName = NULL;
|
|
mwi->uTimer = 0;
|
|
|
|
wndStyle = ((hwndParent) ? (WS_CHILD|WS_BORDER) : WS_OVERLAPPEDWINDOW) |
|
|
WS_VISIBLE | (dwStyle & 0xFFFF0000);
|
|
|
|
if (CreateWindowExA(0, "MCIWndClass", NULL, wndStyle,
|
|
CW_USEDEFAULT, CW_USEDEFAULT,
|
|
CW_USEDEFAULT, CW_USEDEFAULT,
|
|
hwndParent, (HMENU)0, hInstance, mwi))
|
|
return mwi->hWnd;
|
|
|
|
if(mwi->lpName) HeapFree(GetProcessHeap(), 0, mwi->lpName);
|
|
HeapFree(GetProcessHeap(), 0, mwi);
|
|
return 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MCIWndCreateW [MSVFW32.@]
|
|
*/
|
|
HWND VFWAPIV MCIWndCreateW(HWND hwndParent, HINSTANCE hInstance,
|
|
DWORD dwStyle, LPCWSTR szFile)
|
|
{
|
|
FIXME("%p %p %lx %s\n", hwndParent, hInstance, dwStyle, debugstr_w(szFile));
|
|
|
|
MCIWndRegisterClass(hInstance);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static DWORD MCIWND_GetStatus(MCIWndInfo* mwi)
|
|
{
|
|
MCI_DGV_STATUS_PARMSA mdsp;
|
|
|
|
memset(&mdsp, 0, sizeof(mdsp));
|
|
mdsp.dwItem = MCI_STATUS_MODE;
|
|
if (mciSendCommandA(mwi->mci, MCI_STATUS, MCI_WAIT|MCI_STATUS_ITEM, (DWORD)&mdsp))
|
|
return MCI_MODE_NOT_READY;
|
|
if (mdsp.dwReturn == MCI_MODE_STOP && mwi->uTimer) {
|
|
TRACE("Killing timer\n");
|
|
KillTimer(mwi->hWnd, 0);
|
|
mwi->uTimer = 0;
|
|
}
|
|
return mdsp.dwReturn;
|
|
}
|
|
|
|
static DWORD MCIWND_Get(MCIWndInfo* mwi, DWORD what)
|
|
{
|
|
MCI_DGV_STATUS_PARMSA mdsp;
|
|
|
|
memset(&mdsp, 0, sizeof(mdsp));
|
|
mdsp.dwItem = what;
|
|
if (mciSendCommandA(mwi->mci, MCI_STATUS, MCI_WAIT|MCI_STATUS_ITEM, (DWORD)&mdsp))
|
|
return 0;
|
|
return mdsp.dwReturn;
|
|
}
|
|
|
|
static void MCIWND_SetText(MCIWndInfo* mwi)
|
|
{
|
|
char buffer[1024];
|
|
|
|
if (mwi->dwStyle & MCIWNDF_SHOWNAME) {
|
|
strcpy(buffer, mwi->lpName);
|
|
} else {
|
|
*buffer = 0;
|
|
}
|
|
|
|
if (mwi->dwStyle & (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE)) {
|
|
if (*buffer) strcat(buffer, " ");
|
|
strcat(buffer, "(");
|
|
}
|
|
|
|
if (mwi->dwStyle & MCIWNDF_SHOWPOS) {
|
|
sprintf(buffer + strlen(buffer), "%ld", MCIWND_Get(mwi, MCI_STATUS_POSITION));
|
|
}
|
|
|
|
if ((mwi->dwStyle & (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE)) == (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE)) {
|
|
strcat(buffer, " - ");
|
|
}
|
|
|
|
if (mwi->dwStyle & MCIWNDF_SHOWMODE) {
|
|
switch (MCIWND_GetStatus(mwi)) {
|
|
case MCI_MODE_NOT_READY: strcat(buffer, "not ready"); break;
|
|
case MCI_MODE_PAUSE: strcat(buffer, "paused"); break;
|
|
case MCI_MODE_PLAY: strcat(buffer, "playing"); break;
|
|
case MCI_MODE_STOP: strcat(buffer, "stopped"); break;
|
|
case MCI_MODE_OPEN: strcat(buffer, "open"); break;
|
|
case MCI_MODE_RECORD: strcat(buffer, "recording"); break;
|
|
case MCI_MODE_SEEK: strcat(buffer, "seeking"); break;
|
|
default: strcat(buffer, "???"); break;
|
|
}
|
|
}
|
|
if (mwi->dwStyle & (MCIWNDF_SHOWPOS|MCIWNDF_SHOWMODE)) {
|
|
strcat(buffer, " )");
|
|
}
|
|
TRACE("=> '%s'\n", buffer);
|
|
SetWindowTextA(mwi->hWnd, buffer);
|
|
}
|
|
|
|
static void MCIWND_Create(HWND hWnd, LPCREATESTRUCTA cs)
|
|
{
|
|
MCI_DGV_OPEN_PARMSA mdopn;
|
|
MCI_DGV_RECT_PARMS mdrct;
|
|
MMRESULT mmr;
|
|
int cx, cy;
|
|
HWND hChld;
|
|
MCIWndInfo* mwi = (MCIWndInfo*)cs->lpCreateParams;
|
|
|
|
SetWindowLongA(hWnd, 0, (LPARAM)mwi);
|
|
mwi->hWnd = hWnd;
|
|
|
|
/* now open MCI player for AVI file */
|
|
memset(&mdopn, 0, sizeof(mdopn));
|
|
mdopn.lpstrElementName = mwi->lpName;
|
|
mdopn.dwStyle = WS_VISIBLE|WS_CHILD;
|
|
mdopn.hWndParent = hWnd;
|
|
|
|
mmr = mciSendCommandA(0, MCI_OPEN, MCI_OPEN_ELEMENT|MCI_DGV_OPEN_PARENT|MCI_DGV_OPEN_WS, (LPARAM)&mdopn);
|
|
if (mmr) {
|
|
MessageBoxA(GetTopWindow(hWnd), "Cannot open file", "MciWnd", MB_OK);
|
|
return;
|
|
}
|
|
mwi->mci = mdopn.wDeviceID;
|
|
|
|
/* grab AVI window size */
|
|
memset(&mdrct, 0, sizeof(mdrct));
|
|
mmr = mciSendCommandA(mwi->mci, MCI_WHERE, MCI_DGV_WHERE_DESTINATION, (LPARAM)&mdrct);
|
|
if (mmr) {
|
|
WARN("Cannot get window rect\n");
|
|
return;
|
|
}
|
|
cx = mdrct.rc.right - mdrct.rc.left;
|
|
cy = mdrct.rc.bottom - mdrct.rc.top;
|
|
|
|
AdjustWindowRect(&mdrct.rc, GetWindowLongA(hWnd, GWL_STYLE), FALSE);
|
|
SetWindowPos(hWnd, 0, 0, 0, mdrct.rc.right - mdrct.rc.left,
|
|
mdrct.rc.bottom - mdrct.rc.top + 32, SWP_NOMOVE|SWP_NOZORDER);
|
|
|
|
/* adding the other elements: play/stop button, menu button, status */
|
|
hChld = CreateWindowExA(0, "BUTTON", "Play", WS_CHILD|WS_VISIBLE, 0, cy, 32, 32,
|
|
hWnd, (HMENU)CTL_PLAYSTOP,
|
|
(HINSTANCE)GetWindowLongA(hWnd, GWL_HINSTANCE), 0L);
|
|
TRACE("Get Button1: %p\n", hChld);
|
|
hChld = CreateWindowExA(0, "BUTTON", "Menu", WS_CHILD|WS_VISIBLE, 32, cy, 32, 32,
|
|
hWnd, (HMENU)CTL_MENU,
|
|
(HINSTANCE)GetWindowLongA(hWnd, GWL_HINSTANCE), 0L);
|
|
TRACE("Get Button2: %p\n", hChld);
|
|
hChld = CreateWindowExA(0, TRACKBAR_CLASSA, "", WS_CHILD|WS_VISIBLE, 64, cy, cx - 64, 32,
|
|
hWnd, (HMENU)CTL_TRACKBAR,
|
|
(HINSTANCE)GetWindowLongA(hWnd, GWL_HINSTANCE), 0L);
|
|
TRACE("Get status: %p\n", hChld);
|
|
SendMessageA(hChld, TBM_SETRANGEMIN, 0L, 0L);
|
|
SendMessageA(hChld, TBM_SETRANGEMAX, 1L, MCIWND_Get(mwi, MCI_STATUS_LENGTH));
|
|
|
|
/* FIXME: no need to set it if child window */
|
|
MCIWND_SetText(mwi);
|
|
}
|
|
|
|
static void MCIWND_Paint(MCIWndInfo* mwi, WPARAM wParam)
|
|
{
|
|
HDC hdc;
|
|
PAINTSTRUCT ps;
|
|
|
|
hdc = (wParam) ? (HDC)wParam : BeginPaint(mwi->hWnd, &ps);
|
|
/* something to do ? */
|
|
if (!wParam) EndPaint(mwi->hWnd, &ps);
|
|
}
|
|
|
|
static void MCIWND_ToggleState(MCIWndInfo* mwi)
|
|
{
|
|
MCI_GENERIC_PARMS mgp;
|
|
MCI_DGV_PLAY_PARMS mdply;
|
|
|
|
memset(&mgp, 0, sizeof(mgp));
|
|
memset(&mdply, 0, sizeof(mdply));
|
|
|
|
switch (MCIWND_GetStatus(mwi)) {
|
|
case MCI_MODE_NOT_READY:
|
|
case MCI_MODE_RECORD:
|
|
case MCI_MODE_SEEK:
|
|
case MCI_MODE_OPEN:
|
|
TRACE("Cannot do much...\n");
|
|
break;
|
|
case MCI_MODE_PAUSE:
|
|
mciSendCommandA(mwi->mci, MCI_RESUME, MCI_WAIT, (LPARAM)&mgp);
|
|
break;
|
|
case MCI_MODE_PLAY:
|
|
mciSendCommandA(mwi->mci, MCI_PAUSE, MCI_WAIT, (LPARAM)&mgp);
|
|
break;
|
|
case MCI_MODE_STOP:
|
|
mdply.dwFrom = 0L;
|
|
mciSendCommandA(mwi->mci, MCI_PLAY, MCI_FROM, (LPARAM)&mdply);
|
|
mwi->uTimer = SetTimer(mwi->hWnd, 0, 333, 0L);
|
|
TRACE("Timer=%u\n", mwi->uTimer);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static LRESULT MCIWND_Command(MCIWndInfo* mwi, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch (LOWORD(wParam)) {
|
|
case CTL_PLAYSTOP: MCIWND_ToggleState(mwi); break;
|
|
case CTL_MENU:
|
|
case CTL_TRACKBAR:
|
|
default:
|
|
MessageBoxA(0, "ooch", "NIY", MB_OK);
|
|
}
|
|
return 0L;
|
|
}
|
|
|
|
static void MCIWND_Timer(MCIWndInfo* mwi, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
TRACE("%ld\n", MCIWND_Get(mwi, MCI_STATUS_POSITION));
|
|
SendDlgItemMessageA(mwi->hWnd, CTL_TRACKBAR, TBM_SETPOS, 1, MCIWND_Get(mwi, MCI_STATUS_POSITION));
|
|
MCIWND_SetText(mwi);
|
|
}
|
|
|
|
static void MCIWND_Close(MCIWndInfo* mwi)
|
|
{
|
|
MCI_GENERIC_PARMS mgp;
|
|
|
|
memset(&mgp, 0, sizeof(mgp));
|
|
|
|
mciSendCommandA(mwi->mci, MCI_CLOSE, 0, (LPARAM)&mgp);
|
|
}
|
|
|
|
static LRESULT WINAPI MCIWndProc(HWND hWnd, UINT wMsg, WPARAM lParam1, LPARAM lParam2)
|
|
{
|
|
MCIWndInfo* mwi = (MCIWndInfo*)GetWindowLongA(hWnd, 0);
|
|
|
|
if (mwi || wMsg == WM_CREATE) {
|
|
switch (wMsg) {
|
|
case WM_CREATE:
|
|
MCIWND_Create(hWnd, (CREATESTRUCTA*)lParam2);
|
|
return 0;
|
|
case WM_DESTROY:
|
|
MCIWND_Close(mwi);
|
|
HeapFree(GetProcessHeap(), 0, mwi->lpName);
|
|
HeapFree(GetProcessHeap(), 0, mwi);
|
|
break;
|
|
case WM_PAINT:
|
|
MCIWND_Paint(mwi, lParam1);
|
|
break;
|
|
case WM_COMMAND:
|
|
return MCIWND_Command(mwi, lParam1, lParam2);
|
|
case WM_TIMER:
|
|
MCIWND_Timer(mwi, lParam1, lParam2);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return DefWindowProcA(hWnd, wMsg, lParam1, lParam2);
|
|
}
|