Mercurial > mplayer.hg
view gui/win32/gui.c @ 36650:8b2c68d6fd89
Enable specifying a font file in the GUI preferences.
This has been broken since the introduction of Fontconfig.
Since Fontconfig is selecting fonts by font patterns and the GUI does so
by selecting a font file, it is necessary to temporarily disable
fontconfig font handling if there is a directory separator character
found in the name (or pattern) of the font to be used, i.e. assume the
font name to be a pattern if and only if it doesn't contain a directory
separator character.
Thus set option 'fontconfig' depending on font_name. Set it in guiInit()
for the font possibly given in a configuration file or on the command
line, and set it in mplayerLoadFont() whenever it is affected by GUI
preferences settings.
(Although the font selection dialog only allows files to be selected,
it is possible to simply enter a fontconfig font pattern in the
preferences' text entry field - or to enter it directly into the GUI
configuration file or to specify on the command line, both of which
always is possible.)
author | ib |
---|---|
date | Sun, 26 Jan 2014 16:40:49 +0000 |
parents | 32c93d835e3e |
children | 7d250553d451 |
line wrap: on
line source
/* * MPlayer GUI for Win32 * Copyright (C) 2003 Sascha Sommer <saschasommer@freenet.de> * Copyright (C) 2006 Erik Augustson <erik_27can@yahoo.com> * Copyright (C) 2006 Gianluigi Tiesi <sherpya@netfarm.it> * * This file is part of MPlayer. * * MPlayer is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * MPlayer 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 General Public License for more details. * * You should have received a copy of the GNU General Public License along * with MPlayer; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include <stdio.h> #include <stdlib.h> #include <ctype.h> #include <fcntl.h> #include <windows.h> #include <windowsx.h> #include <shlobj.h> #include "config.h" #include "mpcommon.h" #include "mplayer.h" #include "mp_fifo.h" #include "mp_msg.h" #include "help_mp.h" #include "input/input.h" #include "input/mouse.h" #include "osdep/keycodes.h" #include "stream/stream.h" #include "libvo/video_out.h" #include "libmpcodecs/vd.h" #include "gui/interface.h" #include "gui/ui/actions.h" #include "gui/ui/ui.h" #include "gui/util/mem.h" #include "gui.h" #include "dialogs.h" #include "version.h" /* HACK around bug in old mingw */ #undef INVALID_FILE_ATTRIBUTES #define INVALID_FILE_ATTRIBUTES ((DWORD)-1) #ifndef WM_XBUTTONDOWN # define WM_XBUTTONDOWN 0x020B # define WM_XBUTTONUP 0x020C # define WM_XBUTTONDBLCLK 0x020D #endif /* Globals / Externs */ float video_aspect; DWORD oldtime; NOTIFYICONDATA nid; int console_state = FALSE; play_tree_t *playtree = NULL; static HBRUSH colorbrush = NULL; //Handle to colorkey brush static COLORREF windowcolor = RGB(255,0,255); //Windowcolor == colorkey /** * @brief Convert an UTF-8 encoded string into ANSI codepage encoding. * * @param utf8 UTF-8 encoded string * * @return string containing ANSI codepage encoding of @a utf8 (or, in case * of error, a string containing the question mark character) */ LPSTR acp (LPCSTR utf8) { static LPSTR acp_str = NULL; int chars; LPWSTR uc_str; chars = MultiByteToWideChar(CP_UTF8, 0, utf8, -1, NULL, 0); if (chars) { uc_str = malloc(sizeof(*uc_str) * (chars + 1)); if (uc_str) { MultiByteToWideChar(CP_UTF8, 0, utf8, -1, uc_str, chars); chars = WideCharToMultiByte(CP_ACP, 0, uc_str, -1, NULL, 0, NULL, 0); if (chars) { free(acp_str); acp_str = malloc(sizeof(*acp_str) * (chars + 1)); if (acp_str) { WideCharToMultiByte(CP_ACP, 0, uc_str, -1, acp_str, chars, NULL, 0); free(uc_str); return acp_str; } } free(uc_str); } } return "?"; } static void console_toggle(gui_t *gui) { if (console_state) { FreeConsole(); console = FALSE; console_state = FALSE; } else { /* This code comes from: http://dslweb.nwnexus.com/~ast/dload/guicon.htm */ CONSOLE_SCREEN_BUFFER_INFO coninfo; FILE *fp; HWND hwnd = NULL; console = TRUE; AllocConsole(); SetConsoleTitle(mplayer_version); /* disable the close button for now */ while (!hwnd) { hwnd = FindWindow(NULL, mplayer_version); Sleep(100); } DeleteMenu(GetSystemMenu(hwnd, 0), SC_CLOSE, MF_BYCOMMAND); GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &coninfo); coninfo.dwSize.Y = 1000; SetConsoleScreenBufferSize(GetStdHandle(STD_OUTPUT_HANDLE), coninfo.dwSize); fp = freopen("con", "w", stdout); *stdout = *fp; setvbuf(stdout, NULL, _IONBF, 0); fp = freopen("con", "r", stdin); *stdin = *fp; setvbuf(stdin, NULL, _IONBF, 0); fp = freopen("con", "w", stdout); *stderr = *fp; setvbuf(stderr, NULL, _IONBF, 0); print_version(MPlayer); console_state = TRUE; } if (gui) { CheckMenuItem(gui->traymenu, ID_CONSOLE, MF_BYCOMMAND | (console_state ? MF_CHECKED : MF_UNCHECKED)); CheckMenuItem(gui->menu, ID_CONSOLE, MF_BYCOMMAND | (console_state ? MF_CHECKED : MF_UNCHECKED)); } } void capitalize(char *fname) { unsigned int i; BOOL cap = TRUE; for (i=0; i < strlen(fname); i++) { if (cap) { cap = FALSE; fname[i] = toupper(fname[i]); } else if (fname[i] == ' ') cap = TRUE; else fname[i] = tolower(fname[i]); } } static void display_about_box(HWND hWnd) { char about_msg[512]; snprintf(about_msg, sizeof(about_msg), MP_TITLE "\n" COPYRIGHT, MPlayer); MessageBox(hWnd, about_msg, acp(MSGTR_About), MB_OK); } static image *get_drawground(HWND hwnd) { gui_t * gui = (gui_t *) GetWindowLongPtr(hwnd, GWLP_USERDATA); unsigned int i; if(!gui) return NULL; for(i=0; i<gui->window_priv_count; i++) if(gui->window_priv[i]->hwnd==hwnd) return &gui->window_priv[i]->img; return NULL; } static HBITMAP get_bitmap(HWND hwnd) { gui_t *gui = (gui_t *) GetWindowLongPtr(hwnd, GWLP_USERDATA); unsigned int i; if(!gui) return NULL; for(i=0; i<gui->window_priv_count; i++) if(gui->window_priv[i]->hwnd == hwnd) return gui->window_priv[i]->bitmap; return NULL; } static int get_windowtype(HWND hwnd) { gui_t *gui = (gui_t *) GetWindowLongPtr(hwnd, GWLP_USERDATA); unsigned int i; if(!gui) return -1; for(i=0; i<gui->window_priv_count; i++) if(gui->window_priv[i]->hwnd == hwnd) return gui->window_priv[i]->type; return -1; } static void uninit(gui_t *gui) { if(gui->skin) destroy_window(gui); if(gui->playlist) gui->playlist->free_playlist(gui->playlist); gui->playlist = NULL; } /* the gui message handler tries to handle the incoming messages and passes them to the player's message handler if it can't handle them */ static void handlemsg(HWND hWnd, int msg) { gui_t *gui = (gui_t *) GetWindowLongPtr(hWnd, GWLP_USERDATA); if(msg == evNone) return; switch(msg) { case evLoadPlay: case evLoad: if(display_openfilewindow(gui, 0) && (msg == evLoadPlay)) gui->playercontrol(evLoadPlay); return; case evLoadSubtitle: display_opensubtitlewindow(gui); break; case evPreferences: display_prefswindow(gui); return; case evPlaylist: display_playlistwindow(gui); return; case evSkinBrowser: display_skinbrowser(gui); break; case evEqualizer: display_eqwindow(gui); break; case evAbout: display_about_box(hWnd); break; case evIconify: ShowWindow(hWnd, SW_MINIMIZE); break; case evIncVolume: mplayer_put_key(KEY_VOLUME_UP); break; case evDecVolume: mplayer_put_key(KEY_VOLUME_DOWN); break; default: mp_msg(MSGT_GPLAYER, MSGL_V, "[GUI] received msg %s (%i)\n", gui->skin->geteventname(msg), msg); break; } gui->playercontrol(msg); } static widget *clickedinsidewidget(gui_t *gui, int window, int x, int y) { unsigned int i; widget *item; for(i=0; i<gui->skin->widgetcount; i++) { item = gui->skin->widgets[i]; if((item->window == window) && (item->x <= x) && (item->x + item->width >= x) && (item->y <= y) && (item->y + item->height >= y)) return item; } return NULL; } /* updates sliders and the display */ static void updatedisplay(gui_t *gui, HWND hwnd) { unsigned int i; window_priv_t *priv = NULL; DWORD time = timeGetTime(); if(!hwnd) return; /* load all potmeters hpotmeters */ for(i=0; i<gui->skin->widgetcount; i++) { if(gui->skin->widgets[i]->type == tyHpotmeter || gui->skin->widgets[i]->type == tyPotmeter) { if(gui->skin->widgets[i]->msg == evSetVolume) gui->skin->widgets[i]->value = guiInfo.Volume; else if(gui->skin->widgets[i]->msg == evSetMoviePosition) gui->skin->widgets[i]->value = guiInfo.Position; else if(gui->skin->widgets[i]->msg == evSetBalance) gui->skin->widgets[i]->value = guiInfo.Balance; if(gui->skin->widgets[i]->window == get_windowtype(hwnd)) renderwidget(gui->skin, get_drawground(hwnd), gui->skin->widgets[i], gui->skin->widgets[i]->pressed ? 0 : 1); } /* update some buttons */ if(gui->skin->widgets[i]->type == tyButton && gui->skin->widgets[i]->window == get_windowtype(hwnd)) { if(gui->skin->widgets[i]->msg == evPlaySwitchToPause) { gui->skin->widgets[i]->value = guiInfo.Playing; renderwidget(gui->skin, get_drawground(hwnd), gui->skin->widgets[i], guiInfo.Playing == GUI_PLAY ? 0 : 1); } if(gui->skin->widgets[i]->msg == evMute) { gui->skin->widgets[i]->value = guiInfo.Volume; renderwidget(gui->skin, get_drawground(hwnd), gui->skin->widgets[i], guiInfo.Volume == 0.0f ? 0 : 1); } } } /* updating the display once a 100.second is enough imo */ if((time - oldtime) < 100) return; oldtime=time; for (i=0; i<gui->window_priv_count; i++) { if(gui->window_priv[i]->hwnd == hwnd) priv=gui->window_priv[i]; }// Sherpya /* display the status msgs */ renderinfobox(gui->skin, priv); RedrawWindow(hwnd, NULL, NULL, RDW_INVALIDATE); } static LRESULT CALLBACK VideoProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { float aspect; char cmd[40]; gui_t *gui = (gui_t *) GetWindowLongPtr(hWnd, GWLP_USERDATA); if (gui && (gui->videowindow != hWnd)) return FALSE; switch (message) { case WM_CLOSE: handlemsg(hWnd, evExit); return 0; case WM_DESTROY: PostQuitMessage(0); return 0; case WM_KEYDOWN: { switch(wParam) { case VK_LEFT: mplayer_put_key(KEY_LEFT); break; case VK_UP: mplayer_put_key(KEY_UP); break; case VK_RIGHT: mplayer_put_key(KEY_RIGHT); break; case VK_DOWN: mplayer_put_key(KEY_DOWN); break; case VK_TAB: mplayer_put_key(KEY_TAB); break; case VK_BACK: mplayer_put_key(KEY_BS); break; case VK_DELETE: mplayer_put_key(KEY_DELETE); break; case VK_INSERT: mplayer_put_key(KEY_INSERT); break; case VK_HOME: mplayer_put_key(KEY_HOME); break; case VK_END: mplayer_put_key(KEY_END); break; case VK_PRIOR: mplayer_put_key(KEY_PAGE_UP); break; case VK_NEXT: mplayer_put_key(KEY_PAGE_DOWN); break; case VK_ESCAPE: mplayer_put_key(KEY_ESC); break; } break; } case WM_COMMAND: { switch(LOWORD(wParam)) { case IDEXIT: PostQuitMessage(0); handlemsg(hWnd, evExit); break; case IDFILE_OPEN: handlemsg(hWnd, evLoadPlay); break; case IDURL_OPEN: display_openurlwindow(gui, 0); break; case IDDIR_OPEN: { static char path[MAX_PATH]; BROWSEINFO bi; LPITEMIDLIST pidl; memset(&bi, 0, sizeof(BROWSEINFO)); bi.lpszTitle = acp(MSGTR_DirectorySelect); pidl = SHBrowseForFolder(&bi); if (SHGetPathFromIDList(pidl, path)) { gui->playlist->clear_playlist(gui->playlist); adddirtoplaylist(gui->playlist, path, TRUE); gui->startplay(gui); } break; } case ID_PTRACK: handlemsg(hWnd, evPrev); break; case ID_SEEKB: handlemsg(hWnd, evBackward10sec); break; case ID_PLAY: handlemsg(hWnd, evPlaySwitchToPause); break; case ID_STOP: handlemsg(hWnd, evStop); break; case ID_SEEKF: handlemsg(hWnd, evForward10sec); break; case ID_NTRACK: handlemsg(hWnd, evNext); break; case ID_CHAPTERSEL: display_chapterselwindow(gui); break; case ID_FULLSCREEN: mp_input_queue_cmd(mp_input_parse_cmd("vo_fullscreen")); break; case ID_MUTE: mp_input_queue_cmd(mp_input_parse_cmd("mute")); break; case ID_ASPECT1: case ID_ASPECT2: case ID_ASPECT3: case ID_ASPECT4: switch (LOWORD(wParam)) { case ID_ASPECT1: aspect = 16.0f / 9.0f; break; case ID_ASPECT2: aspect = 4.0f / 3.0f; break; case ID_ASPECT3: aspect = 2.35f; break; default: aspect = -1; break; } snprintf(cmd, sizeof(cmd), "pausing_keep switch_ratio %f", aspect); mp_input_queue_cmd(mp_input_parse_cmd(cmd)); break; case IDSUB_TOGGLE: mp_input_queue_cmd(mp_input_parse_cmd("sub_visibility")); break; case IDSUB_CYCLE: mp_input_queue_cmd(mp_input_parse_cmd("sub_select")); break; } return 0; } case WM_CHAR: mplayer_put_key(wParam); break; case WM_DROPFILES: { if(!lParam) { char file[MAX_PATH]; int filecount = DragQueryFile((HDROP) wParam, -1, file, MAX_PATH); int i; for(i=0; i<filecount; i++) { DragQueryFile((HDROP) wParam, i, file, MAX_PATH); uiSetFile(NULL, file, STREAMTYPE_FILE); if(!parse_filename(file, playtree, mconfig, 1)) gui->playlist->add_track(gui->playlist, file, NULL, NULL, 0); } DragFinish((HDROP) wParam); gui->playercontrol(evLoadPlay); } else { gui->playlist->clear_playlist(gui->playlist); gui->playlist->add_track(gui->playlist, (const char *) wParam, NULL, NULL, 0); gui->playercontrol(evLoadPlay); } SetForegroundWindow(gui->videowindow); return 0; } case WM_LBUTTONDOWN: { if(!vo_nomouse_input) mplayer_put_key(MOUSE_BTN0); break; } case WM_MBUTTONDOWN: { if(!vo_nomouse_input) mplayer_put_key(MOUSE_BTN1); break; } case WM_RBUTTONDOWN: { POINT point; point.x = GET_X_LPARAM(lParam); point.y = GET_Y_LPARAM(lParam); ClientToScreen(hWnd, &point); if(guiInfo.StreamType == STREAMTYPE_DVD) EnableMenuItem(gui->dvdmenu, ID_CHAPTERSEL, MF_BYCOMMAND | MF_ENABLED); TrackPopupMenu(gui->videomenu, 0, point.x, point.y, 0, hWnd, NULL); return 0; } case WM_LBUTTONDBLCLK: { if(!vo_nomouse_input) mplayer_put_key(MOUSE_BTN0_DBL); break; } case WM_MBUTTONDBLCLK: { if(!vo_nomouse_input) mplayer_put_key(MOUSE_BTN1_DBL); break; } case WM_RBUTTONDBLCLK: { if(!vo_nomouse_input) mplayer_put_key(MOUSE_BTN2_DBL); break; } case WM_MOUSEWHEEL: { int x = GET_WHEEL_DELTA_WPARAM(wParam); if(vo_nomouse_input) break; if (x > 0) mplayer_put_key(MOUSE_BTN3); else mplayer_put_key(MOUSE_BTN4); break; } case WM_XBUTTONDOWN: { if(vo_nomouse_input) break; if(HIWORD(wParam) == 1) mplayer_put_key(MOUSE_BTN5); else mplayer_put_key(MOUSE_BTN6); break; } case WM_XBUTTONDBLCLK: { if(vo_nomouse_input) break; if(HIWORD(wParam) == 1) mplayer_put_key(MOUSE_BTN5_DBL); else mplayer_put_key(MOUSE_BTN6_DBL); break; } case WM_TIMER: { if(fullscreen) while(ShowCursor(FALSE) >= 0){} KillTimer(hWnd, ID_TIMER); return 0; } case WM_MOUSEMOVE: { ShowCursor(TRUE); SetTimer(hWnd, ID_TIMER, 3000, (TIMERPROC) NULL); break; } case WM_WINDOWPOSCHANGED: { uint32_t tmpheight=0; static uint32_t rect_width; static uint32_t rect_height; RECT rd; POINT pt; pt.x = 0; pt.y = 0; GetClientRect(hWnd, &rd); ClientToScreen(hWnd, &pt); rect_width = rd.right - rd.left; rect_height = rd.bottom - rd.top; /* maintain our aspect ratio */ tmpheight = ((float)rect_width/video_aspect); tmpheight += tmpheight % 2; if(tmpheight > rect_height) { rect_width = ((float)rect_height*video_aspect); rect_width += rect_width % 2; } else rect_height = tmpheight; rd.right = rd.left + rect_width; rd.bottom = rd.top + rect_height; AdjustWindowRect(&rd, WS_OVERLAPPEDWINDOW | WS_SIZEBOX, 0); SetWindowPos(hWnd, 0, fullscreen?0:pt.x+rd.left, fullscreen?0:pt.y+rd.top, fullscreen?vo_screenwidth:rd.right-rd.left, fullscreen?vo_screenheight:rd.bottom-rd.top, SWP_NOOWNERZORDER); SetForegroundWindow(hWnd); return 0; } case WM_SYSCOMMAND: { switch(wParam) { case SC_SCREENSAVE: case SC_MONITORPOWER: mp_msg(MSGT_VO, MSGL_V ,"<vo_directx><INFO>killing screensaver\n" ); return 0; } break; } case WM_PAINT: { PAINTSTRUCT ps; RECT rect; HDC hdc = BeginPaint(hWnd, &ps); HDC hMemDC = CreateCompatibleDC(hdc); HBRUSH blackbrush = (HBRUSH)GetStockObject(BLACK_BRUSH); int width, height; GetClientRect(hWnd, &rect); width = rect.right - rect.left; height = rect.bottom - rect.top; if(guiInfo.Playing == GUI_STOP) { unsigned int i; window *desc = NULL; for (i=0; i<gui->skin->windowcount; i++) if(gui->skin->windows[i]->type == wiVideo) desc = gui->skin->windows[i]; SelectObject(hMemDC, get_bitmap(hWnd)); StretchBlt(hdc, 0, 0, width, height, hMemDC, 0, 0, desc->base->bitmap[0]->width, desc->base->bitmap[0]->height, SRCCOPY); } else { FillRect(GetDC(hWnd), &rect, fullscreen?blackbrush:colorbrush); } DeleteDC(hMemDC); EndPaint(hWnd, &ps); return 0; } } return DefWindowProc(hWnd, message, wParam, lParam); } /* Window Proc for the gui Window */ static LRESULT CALLBACK EventProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { gui_t *gui = (gui_t *) GetWindowLongPtr(hWnd, GWLP_USERDATA); /* Avoid processing when then window doesn't match gui mainwindow */ if (gui && (gui->mainwindow != hWnd)) return FALSE; switch (message) { case WM_CLOSE: handlemsg(hWnd, evExit); return 0; case WM_DESTROY: PostQuitMessage(0); return 0; case WM_SYSTRAY: { switch(lParam) { POINT cursor; case WM_RBUTTONDOWN: { GetCursorPos(&cursor); SetForegroundWindow(hWnd); TrackPopupMenu(gui->traymenu, 0, cursor.x, cursor.y, 0, hWnd, NULL); break; } case WM_MBUTTONDBLCLK: case WM_LBUTTONDBLCLK: { if(IsWindowVisible(hWnd)) ShowWindow(hWnd, SW_HIDE); else { ShowWindow(hWnd, SW_SHOW); SetForegroundWindow(hWnd); } break; } } break; } case WM_KEYDOWN: { switch(wParam) { case VK_LEFT: mplayer_put_key(KEY_LEFT); break; case VK_UP: mplayer_put_key(KEY_UP); break; case VK_RIGHT: mplayer_put_key(KEY_RIGHT); break; case VK_DOWN: mplayer_put_key(KEY_DOWN); break; case VK_TAB: mplayer_put_key(KEY_TAB); break; case VK_BACK: mplayer_put_key(KEY_BS); break; case VK_DELETE: mplayer_put_key(KEY_DELETE); break; case VK_INSERT: mplayer_put_key(KEY_INSERT); break; case VK_HOME: mplayer_put_key(KEY_HOME); break; case VK_END: mplayer_put_key(KEY_END); break; case VK_PRIOR: mplayer_put_key(KEY_PAGE_UP); break; case VK_NEXT: mplayer_put_key(KEY_PAGE_DOWN); break; case VK_ESCAPE: mplayer_put_key(KEY_ESC); break; } break; } case WM_CHAR: mplayer_put_key(wParam); break; case WM_COPYDATA: { if(lParam) { PCOPYDATASTRUCT cdData; cdData = (PCOPYDATASTRUCT) lParam; uiSetFile(NULL, cdData->lpData, STREAMTYPE_FILE); if(!parse_filename(cdData->lpData, playtree, mconfig, 1)) gui->playlist->add_track(gui->playlist, cdData->lpData, NULL, NULL, 0); gui->startplay(gui); } break; } case WM_DROPFILES: { if(!lParam) { char file[MAX_PATH]; int filecount = DragQueryFile((HDROP) wParam, -1, file, MAX_PATH); int i; for(i=0; i<filecount; i++) { DragQueryFile((HDROP) wParam, i, file, MAX_PATH); uiSetFile(NULL, file, STREAMTYPE_FILE); if(!parse_filename(file, playtree, mconfig, 1)) gui->playlist->add_track(gui->playlist, file, NULL, NULL, 0); } DragFinish((HDROP) wParam); gui->playercontrol(evLoadPlay); } else { gui->playlist->clear_playlist(gui->playlist); gui->playlist->add_track(gui->playlist, (const char *) wParam, NULL, NULL, 0); gui->playercontrol(evLoadPlay); } SetForegroundWindow(gui->mainwindow); return 0; } case WM_LBUTTONDOWN: { SetCapture(hWnd); gui->mousex = GET_X_LPARAM(lParam); gui->mousey = GET_Y_LPARAM(lParam); /* inside a widget */ gui->activewidget = clickedinsidewidget(gui, get_windowtype(hWnd), gui->mousex, gui->mousey); if(gui->activewidget) { gui->activewidget->pressed = 1; gui->mousewx = gui->mousex - gui->activewidget->x; gui->mousewy = gui->mousey - gui->activewidget->y; renderwidget(gui->skin, get_drawground(hWnd), gui->activewidget, 0); RedrawWindow(hWnd, NULL, NULL, RDW_INVALIDATE); handlemsg(hWnd, gui->activewidget->msg); } break; } case WM_CAPTURECHANGED: { if(gui->activewidget) { gui->activewidget->pressed = 0; renderwidget(gui->skin, get_drawground(hWnd), gui->activewidget, 1); RedrawWindow(hWnd, NULL, NULL, RDW_INVALIDATE); gui->activewidget = NULL; } break; } case WM_LBUTTONUP: { ReleaseCapture(); if(gui->activewidget) { gui->activewidget->pressed = 0; renderwidget(gui->skin, get_drawground(hWnd), gui->activewidget, 1); RedrawWindow(hWnd, NULL, NULL, RDW_INVALIDATE); gui->activewidget = NULL; } break; } case WM_RBUTTONDOWN: { POINT point; char device[MAX_PATH]; char searchpath[MAX_PATH]; int len, pos = 0, cdromdrive = 0; UINT errmode; point.x = GET_X_LPARAM(lParam); point.y = GET_Y_LPARAM(lParam); ClientToScreen(hWnd, &point); errmode = SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOOPENFILEERRORBOX); while (GetMenuItemCount(gui->diskmenu) > 0) DeleteMenu(gui->diskmenu, 0, MF_BYPOSITION); len = GetLogicalDriveStrings(MAX_PATH, device); while(pos < len) { if(GetDriveType(device + pos) == DRIVE_CDROM) { char volname[MAX_PATH]; char menuitem[MAX_PATH]; int flags = MF_STRING, enable = FALSE; mp_msg(MSGT_GPLAYER, MSGL_V, "[GUI] checking %s for CD/VCD/SVCD/DVDs\n", device + pos); #ifdef CONFIG_DVDREAD sprintf(searchpath, "%sVIDEO_TS", device + pos); if(GetFileAttributes(searchpath) != INVALID_FILE_ATTRIBUTES) enable = TRUE; #endif sprintf(searchpath, "%sMpegav", device + pos); if(GetFileAttributes(searchpath) != INVALID_FILE_ATTRIBUTES) enable = TRUE; #ifdef CONFIG_CDDA sprintf(searchpath, "%sTrack01.cda", device + pos); if(GetFileAttributes(searchpath) != INVALID_FILE_ATTRIBUTES) enable = TRUE; #endif flags |= (enable ? MF_ENABLED : MF_GRAYED); volname[0] = 0; strcpy(menuitem, device + pos); menuitem[strlen(menuitem) - 1]=0; GetVolumeInformation(device + pos, volname, MAX_PATH, NULL, NULL, NULL, NULL, 0); if (strlen(volname)) { capitalize(volname); strcat(menuitem, " - "); strcat(menuitem, volname); } AppendMenu(gui->diskmenu, flags, IDPLAYDISK + cdromdrive, menuitem); if (++cdromdrive == IDPLAYDISK_LIMIT - IDPLAYDISK) break; } pos += strlen(device + pos) + 1; } SetErrorMode(errmode); TrackPopupMenu(gui->menu, 0, point.x, point.y, 0, hWnd, NULL); return 0; } case WM_MOUSEMOVE: { if(wParam & MK_LBUTTON) { POINT point; RECT rect; if(gui->activewidget) { widget *item = gui->activewidget; if(item->type == tyHpotmeter) { item->x = GET_X_LPARAM(lParam) - gui->mousewx; item->value = (float)((float)((item->x - item->wx) * 100.0f) / (float)(item->wwidth - item->width)); } if(item->type == tyPotmeter) { gui->mousewx = GET_X_LPARAM(lParam) - gui->activewidget->x; item->value = (float) (gui->mousewx * 100.0f) / (float) item->wwidth; } if((item->type == tyPotmeter) || (item->type == tyHpotmeter) || (item->type == tyVpotmeter)) { /* Bound checks */ if(item->value > 100.0f) item->value = 100.0f; else if(item->value < 0.0f) item->value = 0.0f; if(item->msg == evSetVolume) guiInfo.Volume = (float) item->value; else if(item->msg == evSetMoviePosition) guiInfo.Position = (float) item->value; else if(item->msg == evSetBalance) { /* make the range for 50% a bit bigger, because the sliders for balance usually suck */ if((item->value - 50.0f < 1.5f) && (item->value - 50.0f > -1.5f)) item->value = 50.0f; guiInfo.Balance = (float) item->value; } updatedisplay(gui, hWnd); handlemsg(hWnd, item->msg); } break; } point.x = GET_X_LPARAM(lParam); point.y = GET_Y_LPARAM(lParam); ClientToScreen(hWnd, &point); GetWindowRect(hWnd, &rect); MoveWindow(hWnd, point.x - gui->mousex, point.y - gui->mousey, rect.right-rect.left,rect.bottom-rect.top,TRUE); break; } break; } case WM_COMMAND: { switch(LOWORD(wParam)) { case IDEXIT: PostQuitMessage(0); handlemsg(hWnd, evExit); break; case IDFILE_OPEN: handlemsg(hWnd, evLoadPlay); break; case IDDIR_OPEN: { static char path[MAX_PATH]; BROWSEINFO bi; LPITEMIDLIST pidl; memset(&bi, 0, sizeof(BROWSEINFO)); bi.lpszTitle = acp(MSGTR_DirectorySelect); pidl = SHBrowseForFolder(&bi); if (SHGetPathFromIDList(pidl, path)) { gui->playlist->clear_playlist(gui->playlist); adddirtoplaylist(gui->playlist, path, TRUE); gui->startplay(gui); } break; } case ID_SKINBROWSER: handlemsg(hWnd, evSkinBrowser); break; case IDURL_OPEN: display_openurlwindow(gui, 0); break; case ID_MUTE: mp_input_queue_cmd(mp_input_parse_cmd("mute")); break; case IDSUBTITLE_OPEN: display_opensubtitlewindow(gui); break; case ID_PTRACK: handlemsg(hWnd, evPrev); break; case ID_SEEKB: handlemsg(hWnd, evBackward10sec); break; case ID_PLAY: handlemsg(hWnd, evPlaySwitchToPause); break; case ID_STOP: handlemsg(hWnd, evStop); break; case ID_SEEKF: handlemsg(hWnd, evForward10sec); break; case ID_NTRACK: handlemsg(hWnd, evNext); break; case ID_SHOWHIDE: { if(IsWindowVisible(hWnd)) ShowWindow(hWnd, SW_HIDE); else ShowWindow(hWnd, SW_SHOW); break; } case ID_PLAYLIST: handlemsg(hWnd, evPlaylist); break; case ID_PREFS: handlemsg(hWnd, evPreferences); break; case ID_CONSOLE: console_toggle(gui); break; case ID_ONLINEHELP: ShellExecute(NULL, "open", ONLINE_HELP_URL, NULL, NULL, SW_SHOWNORMAL); break; case IDHELP_ABOUT: handlemsg(hWnd, evAbout); break; } if(LOWORD(wParam) >= IDPLAYDISK && LOWORD(wParam) < IDPLAYDISK_LIMIT) { char device[MAX_PATH]; char searchpath[MAX_PATH]; char file[MAX_PATH]; int len, pos = 0, cdromdrive = 0; len = GetLogicalDriveStrings(MAX_PATH, device); while(pos < len) { if(GetDriveType(device + pos)==DRIVE_CDROM) { if(LOWORD(wParam) - IDPLAYDISK == cdromdrive) { sprintf(searchpath, "%sVIDEO_TS", device + pos); if(GetFileAttributes(searchpath) != INVALID_FILE_ATTRIBUTES) { free(dvd_device); dvd_device = strdup(device + pos); handlemsg(hWnd, evPlayDVD); } sprintf(searchpath, "%sTrack01.cda", device + pos); if(GetFileAttributes(searchpath) != INVALID_FILE_ATTRIBUTES) { free(cdrom_device); cdrom_device = strdup(device + pos); /* mplayer doesn't seem to like the trailing \ after the device name */ cdrom_device[2]=0; handlemsg(hWnd, evPlayCD); } else { HANDLE searchhndl; WIN32_FIND_DATA finddata; sprintf(searchpath, "%smpegav/*.dat", device + pos); if((searchhndl=FindFirstFile(searchpath, &finddata)) != INVALID_HANDLE_VALUE) { mp_msg(MSGT_GPLAYER,MSGL_V, "Opening VCD/SVCD\n"); gui->playlist->clear_playlist(gui->playlist); do { sprintf(file, "%smpegav/%s", device + pos, finddata.cFileName); gui->playlist->add_track(gui->playlist, file, NULL, NULL, 0); } while(FindNextFile(searchhndl, &finddata)); FindClose(searchhndl); } gui->startplay(gui); } } cdromdrive++; } pos += strlen(device + pos) + 1; } } break; } case WM_PAINT: { PAINTSTRUCT ps; RECT rd; HDC hdc = BeginPaint(hWnd, &ps); HDC hMemDC = CreateCompatibleDC(hdc); int width, height; GetClientRect(hWnd, &rd); width = rd.right - rd.left; height = rd.bottom - rd.top; SelectObject(hMemDC, get_bitmap(hWnd)); BitBlt(hdc, 0, 0, width, height, hMemDC, 0, 0, SRCCOPY); DeleteDC(hMemDC); EndPaint(hWnd, &ps); return 0; } return 0; } return DefWindowProc(hWnd, message, wParam, lParam); } static void startplay(gui_t *gui) { gui->playercontrol(evLoadPlay); } /* returns the bits per pixel of the desktop */ /* the format is always in BGR byte order */ static int GetDesktopBitsPerPixel(void) { HWND desktop=GetDesktopWindow(); HDC dc=GetDC(desktop); int bpp=GetDeviceCaps(dc, BITSPIXEL); ReleaseDC(desktop, dc); return bpp; } /* unloads a skin and destroys its windows */ int destroy_window(gui_t *gui) { RECT rd; unsigned int i; /* Save position: MSDN says don't pass workspace coordinates * to CreateWindow() or SetWindowPos(), as both of which expect * screen coordinates; resulting in the window appearing in the * wrong location. * -Erik */ /* main window position */ if(IsIconic(gui->mainwindow)) ShowWindow(gui->mainwindow, SW_SHOWNORMAL); GetWindowRect(gui->mainwindow, &rd); gui_main_pos_x = rd.left; gui_main_pos_y = rd.top; /* video window position */ if(IsIconic(gui->videowindow)) ShowWindow(gui->videowindow, SW_SHOWNORMAL); GetWindowRect(gui->videowindow, &rd); gui_video_pos_x = rd.left; gui_video_pos_y = rd.top; for(i=0; i<gui->window_priv_count; i++) { if(gui->window_priv[i]->bitmap) DeleteObject(gui->window_priv[i]->bitmap); free(gui->window_priv[i]); } nfree(gui->window_priv); gui->window_priv_count = 0; /* destroy the main window */ if(gui->mainwindow) DestroyWindow(gui->mainwindow); gui->mainwindow = NULL; /* destroy the video window */ if(gui->videowindow) DestroyWindow(gui->videowindow); gui->videowindow = NULL; UnregisterClass(gui->classname, 0); DestroyIcon(gui->icon); gui->skin->freeskin(gui->skin); gui->skin = NULL; return 0; } static void create_menu(gui_t *gui) { gui->diskmenu = CreatePopupMenu(); gui->menu=CreatePopupMenu(); gui->trayplaymenu = CreatePopupMenu(); AppendMenu(gui->menu, MF_STRING | MF_POPUP, (UINT_PTR) gui->trayplaymenu, acp(MSGTR_MENU_Open)); AppendMenu(gui->trayplaymenu, MF_STRING, IDFILE_OPEN, acp(MSGTR_MENU_PlayFile)); AppendMenu(gui->trayplaymenu, MF_STRING, IDURL_OPEN, acp(MSGTR_MENU_PlayURL)); AppendMenu(gui->trayplaymenu, MF_STRING, IDDIR_OPEN, acp(MSGTR_MENU_PlayDirectory)); AppendMenu(gui->menu, MF_SEPARATOR, 0, 0); AppendMenu(gui->menu, MF_STRING | MF_POPUP, (UINT_PTR) gui->diskmenu, acp(MSGTR_MENU_PlayDisc)); AppendMenu(gui->menu, MF_SEPARATOR, 0, 0); AppendMenu(gui->menu, MF_STRING, IDSUBTITLE_OPEN, acp(MSGTR_MENU_LoadSubtitle)); AppendMenu(gui->menu, MF_STRING, ID_SKINBROWSER, acp(MSGTR_MENU_SkinBrowser)); AppendMenu(gui->menu, MF_SEPARATOR, 0, 0); AppendMenu(gui->menu, MF_STRING, ID_PREFS, acp(MSGTR_MENU_Preferences)); AppendMenu(gui->menu, MF_STRING, ID_CONSOLE, acp(MSGTR_MENU_DebugConsole)); AppendMenu(gui->menu, MF_STRING, ID_ONLINEHELP, acp(MSGTR_MENU_OnlineHelp)); AppendMenu(gui->menu, MF_STRING, IDHELP_ABOUT, acp(MSGTR_MENU_AboutMPlayer)); AppendMenu(gui->menu, MF_SEPARATOR, 0, 0); AppendMenu(gui->menu, MF_STRING, IDEXIT, acp(MSGTR_MENU_Exit)); } static void create_traymenu(gui_t *gui) { gui->traymenu = CreatePopupMenu(); gui->trayplaybackmenu = CreatePopupMenu(); AppendMenu(gui->traymenu, MF_STRING | MF_POPUP, (UINT_PTR) gui->trayplaymenu, acp(MSGTR_MENU_Open)); AppendMenu(gui->traymenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->traymenu, MF_STRING | MF_POPUP, (UINT_PTR) gui->trayplaybackmenu, acp(MSGTR_MENU_Playing)); AppendMenu(gui->trayplaybackmenu, MF_STRING, ID_SEEKB, acp(MSGTR_MENU_SeekBack)); AppendMenu(gui->trayplaybackmenu, MF_STRING, ID_PTRACK, acp(MSGTR_MENU_PrevStream)); AppendMenu(gui->trayplaybackmenu, MF_STRING, ID_PLAY, acp(MSGTR_MENU_Play "/" MSGTR_MENU_Pause)); AppendMenu(gui->trayplaybackmenu, MF_STRING, ID_STOP, acp(MSGTR_MENU_Stop)); AppendMenu(gui->trayplaybackmenu, MF_STRING, ID_NTRACK, acp(MSGTR_MENU_NextStream)); AppendMenu(gui->trayplaybackmenu, MF_STRING, ID_SEEKF, acp(MSGTR_MENU_SeekForw)); AppendMenu(gui->traymenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->traymenu, MF_STRING, ID_MUTE, acp(MSGTR_MENU_Mute)); AppendMenu(gui->traymenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->traymenu, MF_STRING, IDSUBTITLE_OPEN, acp(MSGTR_MENU_LoadSubtitle)); AppendMenu(gui->traymenu, MF_STRING, ID_PLAYLIST, acp(MSGTR_MENU_PlayList)); AppendMenu(gui->traymenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->traymenu, MF_STRING, ID_SHOWHIDE, acp(MSGTR_MENU_ShowHide)); AppendMenu(gui->traymenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->traymenu, MF_STRING, ID_PREFS, acp(MSGTR_MENU_Preferences)); AppendMenu(gui->traymenu, MF_STRING, ID_CONSOLE, acp(MSGTR_MENU_DebugConsole)); AppendMenu(gui->traymenu, MF_STRING, ID_ONLINEHELP, acp(MSGTR_MENU_OnlineHelp)); AppendMenu(gui->traymenu, MF_STRING, IDHELP_ABOUT, acp(MSGTR_MENU_AboutMPlayer)); AppendMenu(gui->traymenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->traymenu, MF_STRING, IDEXIT, acp(MSGTR_MENU_Exit)); } static void create_videomenu(gui_t *gui) { gui->videomenu = CreatePopupMenu(); gui->dvdmenu = CreatePopupMenu(); gui->aspectmenu = CreatePopupMenu(); gui->subtitlemenu = CreatePopupMenu(); AppendMenu(gui->videomenu, MF_STRING | MF_POPUP, (UINT_PTR) gui->trayplaymenu, acp(MSGTR_MENU_Open)); AppendMenu(gui->videomenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->videomenu, MF_STRING, ID_SEEKB, acp(MSGTR_MENU_SeekBack)); AppendMenu(gui->videomenu, MF_STRING, ID_PTRACK, acp(MSGTR_MENU_PrevStream)); AppendMenu(gui->videomenu, MF_STRING, ID_PLAY, acp(MSGTR_MENU_Play "/" MSGTR_MENU_Pause)); AppendMenu(gui->videomenu, MF_STRING, ID_STOP, acp(MSGTR_MENU_Stop)); AppendMenu(gui->videomenu, MF_STRING, ID_NTRACK, acp(MSGTR_MENU_NextStream)); AppendMenu(gui->videomenu, MF_STRING, ID_SEEKF, acp(MSGTR_MENU_SeekForw)); AppendMenu(gui->videomenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->videomenu, MF_STRING, ID_FULLSCREEN, acp(MSGTR_MENU_FullScreen)); AppendMenu(gui->videomenu, MF_STRING, ID_MUTE, acp(MSGTR_MENU_Mute)); AppendMenu(gui->videomenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->videomenu, MF_STRING | MF_POPUP, (UINT_PTR) gui->aspectmenu, acp(MSGTR_MENU_AspectRatio)); AppendMenu(gui->videomenu, MF_STRING | MF_POPUP, (UINT_PTR) gui->subtitlemenu, acp(MSGTR_MENU_Subtitles)); #ifdef CONFIG_DVDREAD AppendMenu(gui->videomenu, MF_STRING | MF_POPUP, (UINT_PTR) gui->dvdmenu, acp(MSGTR_MENU_DVD)); AppendMenu(gui->dvdmenu, MF_STRING | MF_GRAYED, ID_CHAPTERSEL, acp(MSGTR_SelectTitleChapter)); #endif AppendMenu(gui->subtitlemenu, MF_STRING, IDSUB_TOGGLE, acp(MSGTR_MENU_SubtitlesOnOff)); AppendMenu(gui->subtitlemenu, MF_STRING, IDSUB_CYCLE, acp(MSGTR_MENU_SubtitleLanguages)); AppendMenu(gui->aspectmenu, MF_STRING, ID_ASPECT1, "16:9"); AppendMenu(gui->aspectmenu, MF_STRING, ID_ASPECT2, "4:3"); AppendMenu(gui->aspectmenu, MF_STRING, ID_ASPECT3, acp(MSGTR_MENU_235)); AppendMenu(gui->aspectmenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->aspectmenu, MF_STRING, ID_ASPECT4, acp(MSGTR_MENU_Original)); AppendMenu(gui->videomenu, MF_SEPARATOR, 0, 0); AppendMenu(gui->videomenu, MF_STRING, IDEXIT, acp(MSGTR_MENU_Exit)); } static void maketransparent(HWND hwnd, COLORREF crTransparent) { HDC mdc = GetDC(hwnd); RECT wrd, crd; HRGN crRgnres, crRgn, crRgnTmp; int iX = 0, iY = 0, iLeftX = 0; int border, title; int width, height; GetWindowRect(hwnd, &wrd); GetClientRect(hwnd, &crd); border = (wrd.right - wrd.left - crd.right) / 2; title = (wrd.bottom - wrd.top - crd.bottom) - border; width = crd.right - crd.left; height = crd.bottom - crd.top; /* create the title bar region */ crRgn = CreateRectRgn(0, 0, width + border + border, title); /* Create a region from a bitmap with transparency colour of Purple */ for (iY = -1; iY < height; iY++) { do { /* skip over transparent pixels at start of lines */ while (iX <= width && GetPixel(mdc,iX, iY) == crTransparent) iX++; /* remember this pixel */ iLeftX = iX; /* now find last non transparent pixel */ while (iX <= width && GetPixel(mdc,iX, iY) != crTransparent) ++iX; /* create a temp region on this info */ crRgnTmp = CreateRectRgn(iLeftX + border, iY + title, iX + border, iY + title + 1); /* combine into main region */ crRgnres = crRgn; CombineRgn(crRgnres, crRgn, crRgnTmp, RGN_OR); crRgn = crRgnres; /* delete the temp region for next pass (otherwise you'll get an ASSERT) */ DeleteObject(crRgnTmp); } while (iX < width); iX = 0; } /* left border region */ crRgnTmp = CreateRectRgn(0, title, border, title + height); CombineRgn(crRgn, crRgn, crRgnTmp, RGN_OR); DeleteObject(crRgnTmp); /* right border region */ crRgnTmp = CreateRectRgn(width + border, title, width + border + border, title + height); CombineRgn(crRgn, crRgn, crRgnTmp, RGN_OR); DeleteObject(crRgnTmp); /* bottom region */ crRgnTmp = CreateRectRgn(0, title + height, width + border + border, title + height + border); CombineRgn(crRgn, crRgn, crRgnTmp, RGN_OR); DeleteObject(crRgnTmp); SetWindowRgn(hwnd, crRgn, TRUE); DeleteObject(crRgn); ReleaseDC(hwnd,mdc); } static int window_render(gui_t *gui, HWND hWnd, HDC hdc, window_priv_t *priv, window *desc, BITMAPINFO binfo) { unsigned int i; SetWindowLongPtr(hWnd, GWLP_USERDATA, (LONG_PTR) gui); (gui->window_priv_count)++; gui->window_priv = realloc(gui->window_priv, sizeof(window_priv_t *) * gui->window_priv_count); priv = gui->window_priv[gui->window_priv_count - 1] = calloc(1, sizeof(window_priv_t)); priv->hwnd = hWnd; priv->type = desc->type; priv->background = desc->base->bitmap[0]; memcpy(&priv->img, desc->base->bitmap[0], sizeof(image)); hdc = GetDC(hWnd); binfo.bmiHeader.biSize = sizeof(BITMAPINFOHEADER); binfo.bmiHeader.biWidth = priv->img.width; binfo.bmiHeader.biHeight = -priv->img.height; binfo.bmiHeader.biPlanes = 1; binfo.bmiHeader.biSizeImage = priv->img.width * priv->img.height * (gui->screenbpp / 8); binfo.bmiHeader.biXPelsPerMeter = 0; binfo.bmiHeader.biYPelsPerMeter = 0; binfo.bmiHeader.biClrUsed = 0; binfo.bmiHeader.biClrImportant = 0; binfo.bmiHeader.biBitCount = gui->screenbpp; binfo.bmiHeader.biCompression = BI_RGB; priv->bitmap = CreateDIBSection(hdc, &binfo, DIB_RGB_COLORS, (void **) &priv->img.data, NULL, 0); if(!priv->bitmap) { mp_msg(MSGT_GPLAYER, MSGL_FATAL, "[GUI] unable to create bitmap for skinned window\n"); return 0; } memcpy(priv->img.data, desc->base->bitmap[0]->data, binfo.bmiHeader.biSizeImage); ReleaseDC(hWnd,hdc); for (i=0; i<gui->skin->widgetcount; i++) if(gui->skin->widgets[i]->window == desc->type) renderwidget(gui->skin, &priv->img, gui->skin->widgets[i], 1); return 0; } /* creates the video window */ int create_videowindow(gui_t *gui) { HINSTANCE instance = GetModuleHandle(NULL); WNDCLASS wc; RECT rect; DWORD style = 0; HDC hdc = NULL; BITMAPINFO binfo; window_priv_t *priv = NULL; window *desc = NULL; unsigned int i; int x = -1, y = -1; vo_colorkey = 0xff00ff; for (i=0; i<gui->skin->windowcount; i++) if(gui->skin->windows[i]->type == wiVideo) desc = gui->skin->windows[i]; if(!desc) { mp_msg(MSGT_GPLAYER, MSGL_FATAL, "[GUI] Invalid skin description\n"); return 1; } windowcolor = vo_colorkey; colorbrush = CreateSolidBrush(windowcolor); wc.style = CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS; wc.lpfnWndProc = VideoProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = instance; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hIcon = gui->icon; wc.hbrBackground = NULL; //WM_PAINT will handle background color switching; wc.lpszClassName = "MPlayer - Video"; wc.lpszMenuName = NULL; RegisterClass(&wc); /* create the video window menu */ create_videomenu(gui); rect.top = rect.left = 100; rect.bottom = rect.top+desc->base->bitmap[0]->height; rect.right = rect.left+desc->base->bitmap[0]->width; /* our window aspect */ video_aspect = (float)(rect.right-rect.left)/(rect.bottom-rect.top); style = fullscreen?WS_VISIBLE | WS_POPUP:WS_OVERLAPPEDWINDOW | WS_SYSMENU | WS_MINIMIZEBOX; AdjustWindowRect(&rect, style, 0); if (gui_video_pos_x >= 0) x = gui_video_pos_x; if (gui_video_pos_y >= 0) y = gui_video_pos_y; /* out of bounds check */ if (x <= -1 || (x+(rect.right-rect.left) > GetSystemMetrics(SM_CXSCREEN))) x = CW_USEDEFAULT; if (y <= -1 || (y+(rect.bottom-rect.top) > GetSystemMetrics(SM_CYSCREEN))) y = x; gui->videowindow = CreateWindowEx(0, "MPlayer - Video", "MPlayer - Video", style, x, y, rect.right-rect.left, rect.bottom-rect.top, NULL, NULL, instance, NULL); /* load all the window images */ window_render(gui, gui->videowindow, hdc, priv, desc, binfo); /* enable drag and drop support */ DragAcceptFiles(gui->videowindow, TRUE); if(video_window) WinID = gui->videowindow; ShowWindow(gui->videowindow, SW_SHOW); UpdateWindow(gui->videowindow); return 0; } /* loads/updates a skin and creates windows for it */ int create_window(gui_t *gui, char *skindir) { HINSTANCE instance = GetModuleHandle(NULL); WNDCLASS wc; RECT rect; DWORD style = 0; HDC hdc = NULL; BITMAPINFO binfo; window_priv_t *priv = NULL; window *desc = NULL; char dir[MAX_PATH]; unsigned int i; int x = -1, y = -1; /* destroy the current main window */ if(gui->skin) destroy_window(gui); /* get screenproperties */ gui->screenbpp = GetDesktopBitsPerPixel(); gui->screenw = GetSystemMetrics(SM_CXSCREEN); gui->screenh = GetSystemMetrics(SM_CYSCREEN); /* load the new skin */ gui->skin = loadskin(skindir, gui->screenbpp); if(!gui->skin) { mp_msg(MSGT_GPLAYER, MSGL_FATAL, "[GUI] fatal error during skinload\n"); /* Set default Skin */ free(skinName); skinName = strdup("Blue"); /* then force write conf */ cfg_write(); return 1; } /* find the description of the mainwindow */ for (i=0; i<gui->skin->windowcount; i++) if(gui->skin->windows[i]->type == wiMain) desc = gui->skin->windows[i]; if(!desc) { mp_msg(MSGT_GPLAYER, MSGL_FATAL, "[GUI] Invalid skin description\n"); return 1; } /* load the icon from the executable */ GetModuleFileName(NULL, dir, MAX_PATH); gui->icon = ExtractIcon(instance, dir, 0); /* create the window class */ wc.style = CS_HREDRAW | CS_VREDRAW; wc.lpfnWndProc = EventProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = instance; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hIcon = gui->icon; wc.hbrBackground = CreateSolidBrush(RGB(0, 0, 0)); wc.lpszClassName = gui->classname = MPlayer; wc.lpszMenuName = NULL; RegisterClass(&wc); /* create a context menu */ create_menu(gui); /* create the systray menu */ create_traymenu(gui); /* create the mainwindow */ /* TODO implement aligning as described in skin.html */ rect.top = rect.left = 100; rect.bottom = rect.top+desc->base->bitmap[0]->height; rect.right = rect.left+desc->base->bitmap[0]->width; if(desc->decoration) style = WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX; else style = WS_POPUP | WS_SYSMENU; AdjustWindowRect(&rect, style, 0); /* Check if out of screen */ if (gui_main_pos_x >= 0) x = gui_main_pos_x; if (gui_main_pos_y >= 0) y = gui_main_pos_y; if (x <= -1 || (x+(rect.right-rect.left) > GetSystemMetrics(SM_CXFULLSCREEN))) { x = (GetSystemMetrics(SM_CXSCREEN) / 2) - ((rect.right-rect.left) / 2); gui_main_pos_x = x; } if (y <= -1 || (y+(rect.bottom-rect.top) > GetSystemMetrics(SM_CYFULLSCREEN))) { y = ((GetSystemMetrics(SM_CYSCREEN)-40) - (rect.bottom-rect.top)); gui_main_pos_y = y; } gui->mainwindow = CreateWindowEx(0, gui->classname, MPlayer, style, x, y, rect.right-rect.left, rect.bottom-rect.top, NULL, NULL, instance, NULL); /* set the systray icon properties */ nid.cbSize = sizeof(NOTIFYICONDATA); nid.hWnd = gui->mainwindow; nid.uID = 1; nid.uFlags = NIF_MESSAGE | NIF_ICON | NIF_TIP; nid.uCallbackMessage = WM_SYSTRAY; nid.hIcon = gui->icon; strcpy(nid.szTip, MPlayer); /* register the systray icon */ Shell_NotifyIcon(NIM_ADD, &nid); /* load all the window images */ window_render(gui, gui->mainwindow, hdc, priv, desc, binfo); /* enable drag and drop support */ DragAcceptFiles(gui->mainwindow, TRUE); updatedisplay(gui, gui->mainwindow); /* display */ ShowWindow(gui->mainwindow, SW_SHOW); UpdateWindow(gui->mainwindow); maketransparent(gui->mainwindow, RGB(255, 0, 255)); return 0; } gui_t *create_gui(char *skindir, void (*playercontrol)(int event)) { gui_t *gui = calloc(1, sizeof(gui_t)); char temp[MAX_PATH]; HWND runningmplayer = FindWindow(MPlayer, MPlayer); if(runningmplayer) { free(gui); return NULL; } gui->startplay = startplay; gui->playercontrol = playercontrol; gui->uninit = uninit; gui->updatedisplay = updatedisplay; /* create playlist */ gui->playlist = create_playlist(); sprintf(temp, "%s/%s", skindir, skinName); if(create_window(gui, temp)) return NULL; if(create_videowindow(gui)) return NULL; if(console) console_toggle(gui); return gui; }