1 /*****************************************************************************
3 *****************************************************************************
4 * Copyright (C) 2001-2009 the VideoLAN team
7 * Authors: Gildas Bazin <gbazin@videolan.org>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
25 /*****************************************************************************
26 * Preamble: This file contains the functions related to the creation of
27 * a window and the handling of its messages (events).
28 *****************************************************************************/
34 #include <vlc_common.h>
35 #include <vlc_vout_display.h>
36 #include <vlc_vout_window.h>
42 #ifdef MODULE_NAME_IS_directx
45 #ifdef MODULE_NAME_IS_direct3d
48 #ifdef MODULE_NAME_IS_glwin32
49 #include "../opengl.h"
55 #include <vlc_windows_interfaces.h>
60 //WINSHELLAPI BOOL WINAPI SHFullScreen(HWND hwndRequester, DWORD dwState);
63 static int CommonControlSetFullscreen(vout_display_t *, bool is_fullscreen);
65 static void DisableScreensaver(vout_display_t *);
66 static void RestoreScreensaver(vout_display_t *);
69 int CommonInit(vout_display_t *vd)
71 vout_display_sys_t *sys = vd->sys;
74 sys->hvideownd = NULL;
78 SetRectEmpty(&sys->rect_display);
79 SetRectEmpty(&sys->rect_parent);
80 sys->is_first_display = true;
81 sys->is_on_top = false;
83 var_Create(vd, "video-title", VLC_VAR_STRING | VLC_VAR_DOINHERIT);
84 var_Create(vd, "video-deco", VLC_VAR_BOOL | VLC_VAR_DOINHERIT);
86 /* FIXME remove mouse hide from msw */
87 var_Create(vd, "mouse-hide-timeout", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT);
90 sys->event = EventThreadCreate(vd);
95 memset(&cfg, 0, sizeof(cfg));
96 #ifdef MODULE_NAME_IS_direct3d
97 cfg.use_desktop = sys->use_desktop;
99 #ifdef MODULE_NAME_IS_directx
100 cfg.use_overlay = sys->b_using_overlay;
102 cfg.win.type = VOUT_WINDOW_TYPE_HWND;
105 cfg.win.width = vd->cfg->display.width;
106 cfg.win.height = vd->cfg->display.height;
109 if (EventThreadStart(sys->event, &hwnd, &cfg))
112 sys->parent_window = hwnd.parent_window;
113 sys->hparent = hwnd.hparent;
114 sys->hwnd = hwnd.hwnd;
115 sys->hvideownd = hwnd.hvideownd;
116 sys->hfswnd = hwnd.hfswnd;
118 if (vd->cfg->is_fullscreen) {
119 if (CommonControlSetFullscreen(vd, true))
120 vout_display_SendEventFullscreen(vd, false);
123 /* Why not with glwin32 */
124 #if !defined(UNDER_CE) && !defined(MODULE_NAME_IS_glwin32)
125 var_Create(vd, "disable-screensaver", VLC_VAR_BOOL | VLC_VAR_DOINHERIT);
126 DisableScreensaver (vd);
133 void CommonClean(vout_display_t *vd)
135 vout_display_sys_t *sys = vd->sys;
138 EventThreadStop(sys->event);
139 EventThreadDestroy(sys->event);
142 #if !defined(UNDER_CE) && !defined(MODULE_NAME_IS_glwin32)
143 RestoreScreensaver(vd);
147 void CommonManage(vout_display_t *vd)
149 vout_display_sys_t *sys = vd->sys;
151 /* We used to call the Win32 PeekMessage function here to read the window
152 * messages. But since window can stay blocked into this function for a
153 * long time (for example when you move your window on the screen), I
154 * decided to isolate PeekMessage in another thread. */
156 /* If we do not control our window, we check for geometry changes
157 * ourselves because the parent might not send us its events. */
162 GetClientRect(sys->hparent, &rect_parent);
163 point.x = point.y = 0;
164 ClientToScreen(sys->hparent, &point);
165 OffsetRect(&rect_parent, point.x, point.y);
167 if (!EqualRect(&rect_parent, &sys->rect_parent)) {
168 sys->rect_parent = rect_parent;
170 /* FIXME I find such #ifdef quite weirds. Are they really needed ? */
172 #if defined(MODULE_NAME_IS_direct3d)
173 SetWindowPos(sys->hwnd, 0, 0, 0,
174 rect_parent.right - rect_parent.left,
175 rect_parent.bottom - rect_parent.top,
177 UpdateRects(vd, NULL, NULL, true);
179 /* This one is to force the update even if only
180 * the position has changed */
181 SetWindowPos(sys->hwnd, 0, 1, 1,
182 rect_parent.right - rect_parent.left,
183 rect_parent.bottom - rect_parent.top, 0);
185 SetWindowPos(sys->hwnd, 0, 0, 0,
186 rect_parent.right - rect_parent.left,
187 rect_parent.bottom - rect_parent.top, 0);
189 #if defined(MODULE_NAME_IS_wingdi) || defined(MODULE_NAME_IS_wingapi)
190 unsigned int i_x, i_y, i_width, i_height;
191 vout_PlacePicture(vd, rect_parent.right - rect_parent.left,
192 rect_parent.bottom - rect_parent.top,
193 &i_x, &i_y, &i_width, &i_height);
195 SetWindowPos(sys->hvideownd, HWND_TOP,
196 i_x, i_y, i_width, i_height, 0);
203 if (EventThreadGetAndResetHasMoved(sys->event))
204 UpdateRects(vd, NULL, NULL, false);
207 EventThreadMouseAutoHide(sys->event);
211 * It ensures that the video window is shown after the first picture
214 void CommonDisplay(vout_display_t *vd)
216 vout_display_sys_t *sys = vd->sys;
218 if (!sys->is_first_display)
221 /* Video window is initially hidden, show it now since we got a
224 SetWindowPos(sys->hvideownd, 0, 0, 0, 0, 0,
231 sys->is_first_display = false;
234 /*****************************************************************************
235 * UpdateRects: update clipping rectangles
236 *****************************************************************************
237 * This function is called when the window position or size are changed, and
238 * its job is to update the source and destination RECTs used to display the
240 *****************************************************************************/
241 void UpdateRects(vout_display_t *vd,
242 const vout_display_cfg_t *cfg,
243 const video_format_t *source,
246 vout_display_sys_t *sys = vd->sys;
247 #define rect_src sys->rect_src
248 #define rect_src_clipped sys->rect_src_clipped
249 #define rect_dest sys->rect_dest
250 #define rect_dest_clipped sys->rect_dest_clipped
259 source = &vd->source;
261 /* Retrieve the window size */
262 GetClientRect(sys->hwnd, &rect);
264 /* Retrieve the window position */
265 point.x = point.y = 0;
266 ClientToScreen(sys->hwnd, &point);
268 /* If nothing changed, we can return */
271 EventThreadUpdateWindowPosition(sys->event, &has_moved, &is_resized,
273 rect.right, rect.bottom);
274 if (!is_forced && !has_moved && !is_resized)
277 /* Update the window position and size */
278 vout_display_cfg_t place_cfg = *cfg;
279 place_cfg.display.width = rect.right;
280 place_cfg.display.height = rect.bottom;
282 vout_display_place_t place;
283 vout_display_PlacePicture(&place, source, &place_cfg, true);
285 EventThreadUpdateSourceAndPlace(sys->event, source, &place);
288 SetWindowPos(sys->hvideownd, 0,
289 place.x, place.y, place.width, place.height,
290 SWP_NOCOPYBITS|SWP_NOZORDER|SWP_ASYNCWINDOWPOS);
292 /* Destination image position and dimensions */
293 #if defined(MODULE_NAME_IS_direct3d)
295 rect_dest.right = place.width;
297 rect_dest.bottom = place.height;
299 rect_dest.left = point.x + place.x;
300 rect_dest.right = rect_dest.left + place.width;
301 rect_dest.top = point.y + place.y;
302 rect_dest.bottom = rect_dest.top + place.height;
304 #ifdef MODULE_NAME_IS_directx
305 /* Apply overlay hardware constraints */
306 if (sys->b_using_overlay) {
307 if (sys->i_align_dest_boundary)
308 rect_dest.left = (rect_dest.left +
309 sys->i_align_dest_boundary / 2) &
310 ~sys->i_align_dest_boundary;
312 if (sys->i_align_dest_size)
313 rect_dest.right = ((rect_dest.right -
315 sys->i_align_dest_size / 2) &
316 ~sys->i_align_dest_size) + rect_dest.left;
322 #if defined(MODULE_NAME_IS_directx) || defined(MODULE_NAME_IS_direct3d)
323 /* UpdateOverlay directdraw function doesn't automatically clip to the
324 * display size so we need to do it otherwise it will fail
325 * It is also needed for d3d to avoid exceding our surface size */
327 /* Clip the destination window */
328 if (!IntersectRect(&rect_dest_clipped, &rect_dest,
329 &sys->rect_display)) {
330 SetRectEmpty(&rect_src_clipped);
335 msg_Dbg(vd, "DirectXUpdateRects image_dst_clipped coords:"
337 rect_dest_clipped.left, rect_dest_clipped.top,
338 rect_dest_clipped.right, rect_dest_clipped.bottom);
343 /* AFAIK, there are no clipping constraints in Direct3D, OpenGL and GDI */
344 rect_dest_clipped = rect_dest;
348 /* the 2 following lines are to fix a bug when clicking on the desktop */
349 if ((rect_dest_clipped.right - rect_dest_clipped.left) == 0 ||
350 (rect_dest_clipped.bottom - rect_dest_clipped.top) == 0) {
351 SetRectEmpty(&rect_src_clipped);
355 /* src image dimensions */
358 rect_src.right = source->i_width;
359 rect_src.bottom = source->i_height;
361 /* Clip the source image */
362 rect_src_clipped.left = source->i_x_offset +
363 (rect_dest_clipped.left - rect_dest.left) *
364 source->i_visible_width / (rect_dest.right - rect_dest.left);
365 rect_src_clipped.right = source->i_x_offset +
366 source->i_visible_width -
367 (rect_dest.right - rect_dest_clipped.right) *
368 source->i_visible_width / (rect_dest.right - rect_dest.left);
369 rect_src_clipped.top = source->i_y_offset +
370 (rect_dest_clipped.top - rect_dest.top) *
371 source->i_visible_height / (rect_dest.bottom - rect_dest.top);
372 rect_src_clipped.bottom = source->i_y_offset +
373 source->i_visible_height -
374 (rect_dest.bottom - rect_dest_clipped.bottom) *
375 source->i_visible_height / (rect_dest.bottom - rect_dest.top);
377 #ifdef MODULE_NAME_IS_directx
378 /* Apply overlay hardware constraints */
379 if (sys->b_using_overlay) {
380 if (sys->i_align_src_boundary)
381 rect_src_clipped.left =
382 (rect_src_clipped.left +
383 sys->i_align_src_boundary / 2) &
384 ~sys->i_align_src_boundary;
386 if (sys->i_align_src_size)
387 rect_src_clipped.right =
388 ((rect_src_clipped.right - rect_src_clipped.left +
389 sys->i_align_src_size / 2) &
390 ~sys->i_align_src_size) + rect_src_clipped.left;
392 #elif defined(MODULE_NAME_IS_direct3d)
393 /* Needed at least with YUV content */
394 rect_src_clipped.left &= ~1;
395 rect_src_clipped.right &= ~1;
396 rect_src_clipped.top &= ~1;
397 rect_src_clipped.bottom &= ~1;
401 msg_Dbg(vd, "DirectXUpdateRects image_src_clipped"
402 " coords: %li,%li,%li,%li",
403 rect_src_clipped.left, rect_src_clipped.top,
404 rect_src_clipped.right, rect_src_clipped.bottom);
407 #ifdef MODULE_NAME_IS_directx
408 /* The destination coordinates need to be relative to the current
409 * directdraw primary surface (display) */
410 rect_dest_clipped.left -= sys->rect_display.left;
411 rect_dest_clipped.right -= sys->rect_display.left;
412 rect_dest_clipped.top -= sys->rect_display.top;
413 rect_dest_clipped.bottom -= sys->rect_display.top;
415 if (sys->b_using_overlay)
416 DirectDrawUpdateOverlay(vd);
420 /* Windows 7 taskbar thumbnail code */
421 LPTASKBARLIST3 taskbl;
422 OSVERSIONINFO winVer;
423 winVer.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
424 if (GetVersionEx(&winVer) && winVer.dwMajorVersion > 5) {
427 if (S_OK == CoCreateInstance(&clsid_ITaskbarList,
428 NULL, CLSCTX_INPROC_SERVER,
431 RECT rect_video, rect_parent, rect_relative;
432 HWND hroot = GetAncestor(sys->hwnd,GA_ROOT);
434 taskbl->vt->HrInit(taskbl);
435 GetWindowRect(sys->hvideownd, &rect_video);
436 GetWindowRect(hroot, &rect_parent);
437 rect_relative.left = rect_video.left - rect_parent.left - 8;
438 rect_relative.right = rect_video.right - rect_video.left + rect_relative.left;
439 rect_relative.top = rect_video.top - rect_parent.top - 10;
440 rect_relative.bottom = rect_video.bottom - rect_video.top + rect_relative.top - 25;
442 if (S_OK != taskbl->vt->SetThumbnailClip(taskbl, hroot, &rect_relative))
443 msg_Err(vd, "SetThumbNailClip failed");
445 taskbl->vt->Release(taskbl);
450 /* Signal the change in size/position */
451 sys->changes |= DX_POSITION_CHANGE;
454 #undef rect_src_clipped
456 #undef rect_dest_clipped
459 static int CommonControlSetFullscreen(vout_display_t *vd, bool is_fullscreen)
461 vout_display_sys_t *sys = vd->sys;
463 #ifdef MODULE_NAME_IS_direct3d
464 if (sys->use_desktop && is_fullscreen)
469 if (sys->parent_window)
470 return vout_window_SetFullScreen(sys->parent_window, is_fullscreen);
473 HWND hwnd = sys->hparent && sys->hfswnd ? sys->hfswnd : sys->hwnd;
475 /* Save the current windows placement/placement to restore
476 when fullscreen is over */
477 WINDOWPLACEMENT window_placement;
478 window_placement.length = sizeof(WINDOWPLACEMENT);
479 GetWindowPlacement(hwnd, &window_placement);
482 msg_Dbg(vd, "entering fullscreen mode");
484 /* Change window style, no borders and no title bar */
485 SetWindowLong(hwnd, GWL_STYLE, WS_CLIPCHILDREN | WS_VISIBLE);
491 ClientToScreen(sys->hwnd, &point);
492 GetClientRect(sys->hwnd, &rect);
493 SetWindowPos(hwnd, 0, point.x, point.y,
494 rect.right, rect.bottom,
495 SWP_NOZORDER|SWP_FRAMECHANGED);
497 /* Retrieve current window position so fullscreen will happen
498 *on the right screen */
499 HMONITOR hmon = MonitorFromWindow(sys->hparent,
500 MONITOR_DEFAULTTONEAREST);
502 mi.cbSize = sizeof(MONITORINFO);
503 if (GetMonitorInfo(hmon, &mi))
504 SetWindowPos(hwnd, 0,
507 mi.rcMonitor.right - mi.rcMonitor.left,
508 mi.rcMonitor.bottom - mi.rcMonitor.top,
509 SWP_NOZORDER|SWP_FRAMECHANGED);
512 /* Maximize non embedded window */
513 ShowWindow(hwnd, SW_SHOWMAXIMIZED);
517 /* Hide the previous window */
519 GetClientRect(hwnd, &rect);
520 SetParent(sys->hwnd, hwnd);
521 SetWindowPos(sys->hwnd, 0, 0, 0,
522 rect.right, rect.bottom,
523 SWP_NOZORDER|SWP_FRAMECHANGED);
526 HWND topLevelParent = GetParent(sys->hparent);
528 HWND topLevelParent = GetAncestor(sys->hparent, GA_ROOT);
530 ShowWindow(topLevelParent, SW_HIDE);
532 SetForegroundWindow(hwnd);
534 msg_Dbg(vd, "leaving fullscreen mode");
536 /* Change window style, no borders and no title bar */
537 SetWindowLong(hwnd, GWL_STYLE, EventThreadGetWindowStyle(sys->event));
541 GetClientRect(sys->hparent, &rect);
542 SetParent(sys->hwnd, sys->hparent);
543 SetWindowPos(sys->hwnd, 0, 0, 0,
544 rect.right, rect.bottom,
545 SWP_NOZORDER|SWP_FRAMECHANGED);
548 HWND topLevelParent = GetParent(sys->hparent);
550 HWND topLevelParent = GetAncestor(sys->hparent, GA_ROOT);
552 ShowWindow(topLevelParent, SW_SHOW);
553 SetForegroundWindow(sys->hparent);
554 ShowWindow(hwnd, SW_HIDE);
556 /* return to normal window for non embedded vout */
557 SetWindowPlacement(hwnd, &window_placement);
558 ShowWindow(hwnd, SW_SHOWNORMAL);
561 /* Make sure the mouse cursor is displayed */
562 EventThreadMouseShow(sys->event);
567 int CommonControl(vout_display_t *vd, int query, va_list args)
569 vout_display_sys_t *sys = vd->sys;
572 case VOUT_DISPLAY_CHANGE_DISPLAY_SIZE: /* const vout_display_cfg_t *p_cfg, int is_forced */
573 case VOUT_DISPLAY_CHANGE_DISPLAY_FILLED: /* const vout_display_cfg_t *p_cfg */
574 case VOUT_DISPLAY_CHANGE_ZOOM: /* const vout_display_cfg_t *p_cfg */
575 case VOUT_DISPLAY_CHANGE_SOURCE_ASPECT: /* const video_format_t *p_source */
576 case VOUT_DISPLAY_CHANGE_SOURCE_CROP: { /* const video_format_t *p_source */
577 const vout_display_cfg_t *cfg;
578 const video_format_t *source;
579 bool is_forced = true;
580 if (query == VOUT_DISPLAY_CHANGE_SOURCE_CROP ||
581 query == VOUT_DISPLAY_CHANGE_SOURCE_ASPECT) {
583 source = va_arg(args, const video_format_t *);
585 cfg = va_arg(args, const vout_display_cfg_t *);
586 source = &vd->source;
587 if (query == VOUT_DISPLAY_CHANGE_DISPLAY_SIZE)
588 is_forced = va_arg(args, int);
590 if (query == VOUT_DISPLAY_CHANGE_DISPLAY_SIZE && is_forced) {
591 /* Update dimensions */
592 if (sys->parent_window) {
593 vout_window_SetSize(sys->parent_window, cfg->display.width, cfg->display.height);
597 rect_window.left = 0;
598 rect_window.right = cfg->display.width;
599 rect_window.bottom = cfg->display.height;
600 AdjustWindowRect(&rect_window, EventThreadGetWindowStyle(sys->event), 0);
602 SetWindowPos(sys->hwnd, 0, 0, 0,
603 rect_window.right - rect_window.left,
604 rect_window.bottom - rect_window.top, SWP_NOMOVE);
607 UpdateRects(vd, cfg, source, is_forced);
610 case VOUT_DISPLAY_CHANGE_WINDOW_STATE: { /* unsigned state */
611 const unsigned state = va_arg(args, unsigned);
612 const bool is_on_top = (state & VOUT_WINDOW_STATE_ABOVE) != 0;
613 #ifdef MODULE_NAME_IS_direct3d
614 if (sys->use_desktop && is_on_top)
617 if (sys->parent_window) {
618 if (vout_window_SetState(sys->parent_window, state))
621 HMENU hMenu = GetSystemMenu(sys->hwnd, FALSE);
623 if (is_on_top && !(GetWindowLong(sys->hwnd, GWL_EXSTYLE) & WS_EX_TOPMOST)) {
624 CheckMenuItem(hMenu, IDM_TOGGLE_ON_TOP, MF_BYCOMMAND | MFS_CHECKED);
625 SetWindowPos(sys->hwnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);
626 } else if (!is_on_top && (GetWindowLong(sys->hwnd, GWL_EXSTYLE) & WS_EX_TOPMOST)) {
627 CheckMenuItem(hMenu, IDM_TOGGLE_ON_TOP, MF_BYCOMMAND | MFS_UNCHECKED);
628 SetWindowPos(sys->hwnd, HWND_NOTOPMOST, 0, 0, 0, 0, SWP_NOSIZE|SWP_NOMOVE);
631 sys->is_on_top = is_on_top;
634 case VOUT_DISPLAY_CHANGE_FULLSCREEN: { /* const vout_display_cfg_t *p_cfg */
635 const vout_display_cfg_t *cfg = va_arg(args, const vout_display_cfg_t *);
636 return CommonControlSetFullscreen(vd, cfg->is_fullscreen);
639 case VOUT_DISPLAY_RESET_PICTURES:
640 case VOUT_DISPLAY_HIDE_MOUSE:
648 static void DisableScreensaver(vout_display_t *vd)
650 vout_display_sys_t *sys = vd->sys;
652 /* disable screensaver by temporarily changing system settings */
653 sys->i_spi_lowpowertimeout = 0;
654 sys->i_spi_powerofftimeout = 0;
655 sys->i_spi_screensavetimeout = 0;
656 if (var_GetBool(vd, "disable-screensaver")) {
657 msg_Dbg(vd, "disabling screen saver");
658 SystemParametersInfo(SPI_GETLOWPOWERTIMEOUT, 0,
659 &sys->i_spi_lowpowertimeout, 0);
660 if (0 != sys->i_spi_lowpowertimeout) {
661 SystemParametersInfo(SPI_SETLOWPOWERTIMEOUT, 0, NULL, 0);
663 SystemParametersInfo(SPI_GETPOWEROFFTIMEOUT, 0,
664 &sys->i_spi_powerofftimeout, 0);
665 if (0 != sys->i_spi_powerofftimeout) {
666 SystemParametersInfo(SPI_SETPOWEROFFTIMEOUT, 0, NULL, 0);
668 SystemParametersInfo(SPI_GETSCREENSAVETIMEOUT, 0,
669 &sys->i_spi_screensavetimeout, 0);
670 if (0 != sys->i_spi_screensavetimeout) {
671 SystemParametersInfo(SPI_SETSCREENSAVETIMEOUT, 0, NULL, 0);
676 static void RestoreScreensaver(vout_display_t *vd)
678 vout_display_sys_t *sys = vd->sys;
680 /* restore screensaver system settings */
681 if (0 != sys->i_spi_lowpowertimeout) {
682 SystemParametersInfo(SPI_SETLOWPOWERTIMEOUT,
683 sys->i_spi_lowpowertimeout, NULL, 0);
685 if (0 != sys->i_spi_powerofftimeout) {
686 SystemParametersInfo(SPI_SETPOWEROFFTIMEOUT,
687 sys->i_spi_powerofftimeout, NULL, 0);
689 if (0 != sys->i_spi_screensavetimeout) {
690 SystemParametersInfo(SPI_SETSCREENSAVETIMEOUT,
691 sys->i_spi_screensavetimeout, NULL, 0);