2009-02-08 03:27:30 +00:00
|
|
|
/*
|
2015-04-13 07:36:54 +00:00
|
|
|
* This file is part of mpv.
|
2009-02-08 03:27:30 +00:00
|
|
|
*
|
2017-06-24 10:54:45 +00:00
|
|
|
* mpv 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.
|
2009-02-08 03:27:30 +00:00
|
|
|
*
|
2015-04-13 07:36:54 +00:00
|
|
|
* mpv is distributed in the hope that it will be useful,
|
2009-02-08 03:27:30 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2017-06-24 10:54:45 +00:00
|
|
|
* GNU Lesser General Public License for more details.
|
2009-02-08 03:27:30 +00:00
|
|
|
*
|
2017-06-24 10:54:45 +00:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with mpv. If not, see <http://www.gnu.org/licenses/>.
|
2009-02-08 03:27:30 +00:00
|
|
|
*/
|
|
|
|
|
2007-03-04 13:20:51 +00:00
|
|
|
#include <stdio.h>
|
2003-09-19 14:33:51 +00:00
|
|
|
#include <limits.h>
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
#include <pthread.h>
|
win32: fix window creation and size handling
This commit fixes various issues with the way the window position and
size is setup. Most importantly, it fixes some bugs with restoring from
fullscreen state.
Rename create_rendering_context() to reinit_window_state(). This function
doesn't create anything, it just sets the window bounds and styles.
Do not use vo_dx/dy for the window position, as video_out.c overwrites it
with each vo_config() call. Use private variables window_x/y instead.
A big cause for issues was that reinit_window_state() accidentally cleared
the WS_VISIBLE style. I suspect that the API call to temporarily hide the
window was a hack to deal with this. Another bug was that the window style
was changed without calling SetWindowPos with SWP_FRAMECHANGED (as the
MSDN documentation says).
Properly initialize window position and size on vo_config following the
same rules as the x11 backend:
- Never change the window position. The window position should be kept, as
the user might move the window, and resetting the window position e.g.
during ordered chapter playback is not desired.
- Never change the window size, unless the size of the video changes.
These rules don't apply to fullscreen. When switching from fullscreen to
windowed mode, the backend should restore the previous windowed size and
position. When the VO was reconfigured during playback (vo_config() etc.),
the saved window position and size should be changed according to the
rules above, even if the window was in fullscreen mode during
reconfiguring.
Note that these rules might be perceived as awkward by some users: if you
play multiple files with different resolutions, the window won't be
centered when playing the files after the first. This is not a bug.
2011-10-22 00:11:14 +00:00
|
|
|
#include <assert.h>
|
2003-09-19 14:33:51 +00:00
|
|
|
#include <windows.h>
|
2007-03-04 13:20:51 +00:00
|
|
|
#include <windowsx.h>
|
2016-12-09 18:22:33 +00:00
|
|
|
#include <dwmapi.h>
|
2014-01-06 04:21:54 +00:00
|
|
|
#include <ole2.h>
|
2014-12-08 06:06:14 +00:00
|
|
|
#include <shobjidl.h>
|
2015-11-08 18:31:09 +00:00
|
|
|
#include <avrt.h>
|
2003-09-19 14:33:51 +00:00
|
|
|
|
2019-12-11 18:40:33 +00:00
|
|
|
#include "options/m_config.h"
|
2013-12-17 01:02:25 +00:00
|
|
|
#include "options/options.h"
|
2013-12-17 00:23:09 +00:00
|
|
|
#include "input/keycodes.h"
|
|
|
|
#include "input/input.h"
|
2014-01-06 04:21:54 +00:00
|
|
|
#include "input/event.h"
|
2015-11-26 11:11:45 +00:00
|
|
|
#include "stream/stream.h"
|
2013-12-17 01:39:45 +00:00
|
|
|
#include "common/msg.h"
|
|
|
|
#include "common/common.h"
|
2012-11-09 00:06:43 +00:00
|
|
|
#include "vo.h"
|
2014-05-06 21:01:19 +00:00
|
|
|
#include "win_state.h"
|
2003-09-19 14:33:51 +00:00
|
|
|
#include "w32_common.h"
|
2015-02-03 04:25:50 +00:00
|
|
|
#include "win32/displayconfig.h"
|
2017-03-25 12:30:54 +00:00
|
|
|
#include "win32/droptarget.h"
|
2011-12-07 00:36:19 +00:00
|
|
|
#include "osdep/io.h"
|
2014-10-19 21:32:34 +00:00
|
|
|
#include "osdep/threads.h"
|
2014-01-19 13:20:02 +00:00
|
|
|
#include "osdep/w32_keyboard.h"
|
2016-09-07 09:26:25 +00:00
|
|
|
#include "osdep/atomic.h"
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
#include "misc/dispatch.h"
|
|
|
|
#include "misc/rendezvous.h"
|
2016-01-11 18:03:40 +00:00
|
|
|
#include "mpv_talloc.h"
|
2003-09-19 14:33:51 +00:00
|
|
|
|
2016-06-10 12:37:29 +00:00
|
|
|
EXTERN_C IMAGE_DOS_HEADER __ImageBase;
|
|
|
|
#define HINST_THISCOMPONENT ((HINSTANCE)&__ImageBase)
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
#ifndef WM_DPICHANGED
|
|
|
|
#define WM_DPICHANGED (0x02E0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef DPI_ENUMS_DECLARED
|
|
|
|
typedef enum MONITOR_DPI_TYPE {
|
|
|
|
MDT_EFFECTIVE_DPI = 0,
|
|
|
|
MDT_ANGULAR_DPI = 1,
|
|
|
|
MDT_RAW_DPI = 2,
|
|
|
|
MDT_DEFAULT = MDT_EFFECTIVE_DPI
|
|
|
|
} MONITOR_DPI_TYPE;
|
|
|
|
#endif
|
|
|
|
|
2017-12-17 00:19:36 +00:00
|
|
|
#define rect_w(r) ((r).right - (r).left)
|
|
|
|
#define rect_h(r) ((r).bottom - (r).top)
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
struct w32_api {
|
|
|
|
HRESULT (WINAPI *pGetDpiForMonitor)(HMONITOR, MONITOR_DPI_TYPE, UINT*, UINT*);
|
2017-11-15 01:50:00 +00:00
|
|
|
BOOL (WINAPI *pImmDisableIME)(DWORD);
|
2018-11-17 20:56:56 +00:00
|
|
|
BOOL (WINAPI *pAdjustWindowRectExForDpi)(LPRECT lpRect, DWORD dwStyle, BOOL bMenu, DWORD dwExStyle, UINT dpi);
|
2016-12-09 18:22:33 +00:00
|
|
|
};
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
struct vo_w32_state {
|
|
|
|
struct mp_log *log;
|
|
|
|
struct vo *vo;
|
|
|
|
struct mp_vo_opts *opts;
|
2019-12-11 18:40:33 +00:00
|
|
|
struct m_config_cache *opts_cache;
|
2014-07-26 18:27:03 +00:00
|
|
|
struct input_ctx *input_ctx;
|
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
pthread_t thread;
|
|
|
|
bool terminate;
|
|
|
|
struct mp_dispatch_queue *dispatch; // used to run stuff on the GUI thread
|
2018-04-17 16:29:32 +00:00
|
|
|
bool in_dispatch;
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
struct w32_api api; // stores functions from dynamically loaded DLLs
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
HWND window;
|
2014-12-09 20:28:35 +00:00
|
|
|
HWND parent; // 0 normally, set in embedding mode
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
HHOOK parent_win_hook;
|
|
|
|
HWINEVENTHOOK parent_evt_hook;
|
2014-07-26 18:27:03 +00:00
|
|
|
|
2015-02-03 04:25:50 +00:00
|
|
|
HMONITOR monitor; // Handle of the current screen
|
2015-11-26 11:11:45 +00:00
|
|
|
char *color_profile; // Path of the current screen's color profile
|
2014-07-26 18:27:03 +00:00
|
|
|
|
2014-09-30 02:21:35 +00:00
|
|
|
// Has the window seen a WM_DESTROY? If so, don't call DestroyWindow again.
|
|
|
|
bool destroyed;
|
|
|
|
|
2021-05-27 09:07:28 +00:00
|
|
|
bool focused;
|
|
|
|
|
2022-04-25 11:27:18 +00:00
|
|
|
// whether the window position and size were initialized
|
2014-07-26 18:27:03 +00:00
|
|
|
bool window_bounds_initialized;
|
|
|
|
|
|
|
|
bool current_fs;
|
2016-12-13 03:29:50 +00:00
|
|
|
bool toggle_fs; // whether the current fullscreen state needs to be switched
|
2014-07-26 18:27:03 +00:00
|
|
|
|
2021-04-22 12:52:07 +00:00
|
|
|
// Note: maximized state doesn't involve nor modify windowrc
|
|
|
|
RECT windowrc; // currently known normal/fullscreen window client rect
|
|
|
|
RECT prev_windowrc; // saved normal window client rect while in fullscreen
|
2014-07-26 18:27:03 +00:00
|
|
|
|
|
|
|
// video size
|
|
|
|
uint32_t o_dwidth;
|
|
|
|
uint32_t o_dheight;
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
int dpi;
|
2018-11-17 20:56:56 +00:00
|
|
|
double dpi_scale;
|
2016-12-09 18:22:33 +00:00
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
bool disable_screensaver;
|
|
|
|
bool cursor_visible;
|
2016-08-05 14:01:26 +00:00
|
|
|
atomic_uint event_flags;
|
2014-07-26 18:27:03 +00:00
|
|
|
|
|
|
|
BOOL tracking;
|
|
|
|
TRACKMOUSEEVENT trackEvent;
|
|
|
|
|
|
|
|
int mouse_x;
|
|
|
|
int mouse_y;
|
|
|
|
|
2015-03-13 08:46:54 +00:00
|
|
|
// Should SetCursor be called when handling VOCTRL_SET_CURSOR_VISIBILITY?
|
|
|
|
bool can_set_cursor;
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
// UTF-16 decoding state for WM_CHAR and VK_PACKET
|
|
|
|
int high_surrogate;
|
2014-12-08 06:06:14 +00:00
|
|
|
|
2021-04-22 12:52:07 +00:00
|
|
|
// Fit the window to one monitor working area next time it's not fullscreen
|
|
|
|
// and not maximized. Used once after every new "untrusted" size comes from
|
|
|
|
// mpv, else we assume that the last known size is valid and don't fit.
|
|
|
|
// FIXME: on a multi-monitor setup one bit is not enough, because the first
|
|
|
|
// fit (autofit etc) should be to one monitor, but later size changes from
|
|
|
|
// mpv like window-scale (VOCTRL_SET_UNFS_WINDOW_SIZE) should allow the
|
|
|
|
// entire virtual desktop area - but we still limit to one monitor size.
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
bool fit_on_screen;
|
|
|
|
|
2014-12-08 06:06:14 +00:00
|
|
|
ITaskbarList2 *taskbar_list;
|
2015-11-15 22:03:48 +00:00
|
|
|
ITaskbarList3 *taskbar_list3;
|
|
|
|
UINT tbtnCreatedMsg;
|
|
|
|
bool tbtnCreated;
|
2015-03-10 18:25:30 +00:00
|
|
|
|
2016-09-02 16:08:49 +00:00
|
|
|
struct voctrl_playback_state current_pstate;
|
|
|
|
|
2015-03-10 18:25:30 +00:00
|
|
|
// updates on move/resize/displaychange
|
|
|
|
double display_fps;
|
2015-11-08 18:31:09 +00:00
|
|
|
|
2017-12-04 23:21:34 +00:00
|
|
|
bool moving;
|
2016-12-09 18:22:33 +00:00
|
|
|
bool snapped;
|
|
|
|
int snap_dx;
|
|
|
|
int snap_dy;
|
|
|
|
|
2015-11-08 18:31:09 +00:00
|
|
|
HANDLE avrt_handle;
|
2014-07-26 18:27:03 +00:00
|
|
|
};
|
|
|
|
|
2018-11-17 20:56:56 +00:00
|
|
|
static void add_window_borders(struct vo_w32_state *w32, HWND hwnd, RECT *rc)
|
2011-10-28 13:19:51 +00:00
|
|
|
{
|
2018-11-17 20:56:56 +00:00
|
|
|
if (w32->api.pAdjustWindowRectExForDpi) {
|
|
|
|
w32->api.pAdjustWindowRectExForDpi(rc,
|
|
|
|
GetWindowLongPtrW(hwnd, GWL_STYLE), 0,
|
|
|
|
GetWindowLongPtrW(hwnd, GWL_EXSTYLE), w32->dpi);
|
|
|
|
} else {
|
|
|
|
AdjustWindowRect(rc, GetWindowLongPtrW(hwnd, GWL_STYLE), 0);
|
|
|
|
}
|
2011-10-28 13:19:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// basically a reverse AdjustWindowRect (win32 doesn't appear to have this)
|
2018-11-17 20:56:56 +00:00
|
|
|
static void subtract_window_borders(struct vo_w32_state *w32, HWND hwnd, RECT *rc)
|
2011-10-28 13:19:51 +00:00
|
|
|
{
|
|
|
|
RECT b = { 0, 0, 0, 0 };
|
2018-11-17 20:56:56 +00:00
|
|
|
add_window_borders(w32, hwnd, &b);
|
2011-10-28 13:19:51 +00:00
|
|
|
rc->left -= b.left;
|
|
|
|
rc->top -= b.top;
|
|
|
|
rc->right -= b.right;
|
|
|
|
rc->bottom -= b.bottom;
|
|
|
|
}
|
|
|
|
|
2014-12-29 13:52:34 +00:00
|
|
|
static LRESULT borderless_nchittest(struct vo_w32_state *w32, int x, int y)
|
|
|
|
{
|
2015-03-09 11:15:01 +00:00
|
|
|
if (IsMaximized(w32->window))
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTCLIENT;
|
|
|
|
|
|
|
|
POINT mouse = { x, y };
|
|
|
|
ScreenToClient(w32->window, &mouse);
|
|
|
|
|
2016-07-03 06:56:52 +00:00
|
|
|
// The horizontal frame should be the same size as the vertical frame,
|
|
|
|
// since the NONCLIENTMETRICS structure does not distinguish between them
|
|
|
|
int frame_size = GetSystemMetrics(SM_CXFRAME) +
|
|
|
|
GetSystemMetrics(SM_CXPADDEDBORDER);
|
2014-12-29 13:52:34 +00:00
|
|
|
// The diagonal size handles are slightly wider than the side borders
|
2016-07-03 06:56:52 +00:00
|
|
|
int diagonal_width = frame_size * 2 + GetSystemMetrics(SM_CXBORDER);
|
2014-12-29 13:52:34 +00:00
|
|
|
|
|
|
|
// Hit-test top border
|
2016-07-03 06:56:52 +00:00
|
|
|
if (mouse.y < frame_size) {
|
|
|
|
if (mouse.x < diagonal_width)
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTTOPLEFT;
|
2017-12-17 00:19:36 +00:00
|
|
|
if (mouse.x >= rect_w(w32->windowrc) - diagonal_width)
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTTOPRIGHT;
|
|
|
|
return HTTOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hit-test bottom border
|
2017-12-17 00:19:36 +00:00
|
|
|
if (mouse.y >= rect_h(w32->windowrc) - frame_size) {
|
2016-07-03 06:56:52 +00:00
|
|
|
if (mouse.x < diagonal_width)
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTBOTTOMLEFT;
|
2017-12-17 00:19:36 +00:00
|
|
|
if (mouse.x >= rect_w(w32->windowrc) - diagonal_width)
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTBOTTOMRIGHT;
|
|
|
|
return HTBOTTOM;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hit-test side borders
|
2016-07-03 06:56:52 +00:00
|
|
|
if (mouse.x < frame_size)
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTLEFT;
|
2017-12-17 00:19:36 +00:00
|
|
|
if (mouse.x >= rect_w(w32->windowrc) - frame_size)
|
2014-12-29 13:52:34 +00:00
|
|
|
return HTRIGHT;
|
|
|
|
return HTCLIENT;
|
|
|
|
}
|
|
|
|
|
2011-10-28 13:19:51 +00:00
|
|
|
// turn a WMSZ_* input value in v into the border that should be resized
|
|
|
|
// returns: 0=left, 1=top, 2=right, 3=bottom, -1=undefined
|
2014-01-06 11:26:42 +00:00
|
|
|
static int get_resize_border(int v)
|
|
|
|
{
|
2011-10-28 13:19:51 +00:00
|
|
|
switch (v) {
|
|
|
|
case WMSZ_LEFT: return 3;
|
|
|
|
case WMSZ_TOP: return 2;
|
|
|
|
case WMSZ_RIGHT: return 3;
|
|
|
|
case WMSZ_BOTTOM: return 2;
|
|
|
|
case WMSZ_TOPLEFT: return 1;
|
|
|
|
case WMSZ_TOPRIGHT: return 1;
|
|
|
|
case WMSZ_BOTTOMLEFT: return 3;
|
|
|
|
case WMSZ_BOTTOMRIGHT: return 3;
|
|
|
|
default: return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
static bool key_state(int vk)
|
2012-04-14 11:27:34 +00:00
|
|
|
{
|
|
|
|
return GetKeyState(vk) & 0x8000;
|
|
|
|
}
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
static int mod_state(struct vo_w32_state *w32)
|
2012-04-06 21:14:21 +00:00
|
|
|
{
|
|
|
|
int res = 0;
|
2014-04-17 15:25:45 +00:00
|
|
|
|
|
|
|
// AltGr is represented as LCONTROL+RMENU on Windows
|
2014-07-26 18:27:03 +00:00
|
|
|
bool alt_gr = mp_input_use_alt_gr(w32->input_ctx) &&
|
|
|
|
key_state(VK_RMENU) && key_state(VK_LCONTROL);
|
2014-04-17 15:25:45 +00:00
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
if (key_state(VK_RCONTROL) || (key_state(VK_LCONTROL) && !alt_gr))
|
2013-02-12 00:34:04 +00:00
|
|
|
res |= MP_KEY_MODIFIER_CTRL;
|
2014-07-26 18:27:03 +00:00
|
|
|
if (key_state(VK_SHIFT))
|
2013-02-12 00:34:04 +00:00
|
|
|
res |= MP_KEY_MODIFIER_SHIFT;
|
2014-07-26 18:27:03 +00:00
|
|
|
if (key_state(VK_LMENU) || (key_state(VK_RMENU) && !alt_gr))
|
2013-02-12 00:34:04 +00:00
|
|
|
res |= MP_KEY_MODIFIER_ALT;
|
2012-04-06 21:14:21 +00:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2015-08-01 18:51:52 +00:00
|
|
|
static int decode_surrogate_pair(wchar_t lead, wchar_t trail)
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
2015-03-04 10:43:02 +00:00
|
|
|
return 0x10000 + (((lead & 0x3ff) << 10) | (trail & 0x3ff));
|
2014-04-17 15:25:45 +00:00
|
|
|
}
|
|
|
|
|
2015-08-01 18:51:52 +00:00
|
|
|
static int decode_utf16(struct vo_w32_state *w32, wchar_t c)
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
|
|
|
// Decode UTF-16, keeping state in w32->high_surrogate
|
|
|
|
if (IS_HIGH_SURROGATE(c)) {
|
|
|
|
w32->high_surrogate = c;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (IS_LOW_SURROGATE(c)) {
|
|
|
|
if (!w32->high_surrogate) {
|
2014-07-26 18:27:03 +00:00
|
|
|
MP_ERR(w32, "Invalid UTF-16 input\n");
|
2014-04-17 15:25:45 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int codepoint = decode_surrogate_pair(w32->high_surrogate, c);
|
|
|
|
w32->high_surrogate = 0;
|
|
|
|
return codepoint;
|
|
|
|
}
|
|
|
|
if (w32->high_surrogate != 0) {
|
|
|
|
w32->high_surrogate = 0;
|
2014-07-26 18:27:03 +00:00
|
|
|
MP_ERR(w32, "Invalid UTF-16 input\n");
|
2014-04-17 15:25:45 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void clear_keyboard_buffer(void)
|
|
|
|
{
|
|
|
|
static const UINT vkey = VK_DECIMAL;
|
|
|
|
static const BYTE keys[256] = { 0 };
|
|
|
|
UINT scancode = MapVirtualKey(vkey, MAPVK_VK_TO_VSC);
|
2015-08-01 18:51:52 +00:00
|
|
|
wchar_t buf[10];
|
2014-04-17 15:25:45 +00:00
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
// Use the method suggested by Michael Kaplan to clear any pending dead
|
|
|
|
// keys from the current keyboard layout. See:
|
|
|
|
// https://web.archive.org/web/20101004154432/http://blogs.msdn.com/b/michkap/archive/2006/04/06/569632.aspx
|
|
|
|
// https://web.archive.org/web/20100820152419/http://blogs.msdn.com/b/michkap/archive/2007/10/27/5717859.aspx
|
|
|
|
do {
|
|
|
|
ret = ToUnicode(vkey, scancode, keys, buf, MP_ARRAY_SIZE(buf), 0);
|
|
|
|
} while (ret < 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int to_unicode(UINT vkey, UINT scancode, const BYTE keys[256])
|
|
|
|
{
|
|
|
|
// This wraps ToUnicode to be stateless and to return only one character
|
|
|
|
|
|
|
|
// Make the buffer 10 code units long to be safe, same as here:
|
|
|
|
// https://web.archive.org/web/20101013215215/http://blogs.msdn.com/b/michkap/archive/2006/03/24/559169.aspx
|
2015-08-01 18:51:52 +00:00
|
|
|
wchar_t buf[10] = { 0 };
|
2014-04-17 15:25:45 +00:00
|
|
|
|
|
|
|
// Dead keys aren't useful for key shortcuts, so clear the keyboard state
|
|
|
|
clear_keyboard_buffer();
|
|
|
|
|
|
|
|
int len = ToUnicode(vkey, scancode, keys, buf, MP_ARRAY_SIZE(buf), 0);
|
|
|
|
|
|
|
|
// Return the last complete UTF-16 code point. A negative return value
|
|
|
|
// indicates a dead key, however there should still be a non-combining
|
|
|
|
// version of the key in the buffer.
|
|
|
|
if (len < 0)
|
|
|
|
len = -len;
|
|
|
|
if (len >= 2 && IS_SURROGATE_PAIR(buf[len - 2], buf[len - 1]))
|
|
|
|
return decode_surrogate_pair(buf[len - 2], buf[len - 1]);
|
|
|
|
if (len >= 1)
|
|
|
|
return buf[len - 1];
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
static int decode_key(struct vo_w32_state *w32, UINT vkey, UINT scancode)
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
|
|
|
BYTE keys[256];
|
|
|
|
GetKeyboardState(keys);
|
|
|
|
|
|
|
|
// If mp_input_use_alt_gr is false, detect and remove AltGr so normal
|
|
|
|
// characters are generated. Note that AltGr is represented as
|
|
|
|
// LCONTROL+RMENU on Windows.
|
|
|
|
if ((keys[VK_RMENU] & 0x80) && (keys[VK_LCONTROL] & 0x80) &&
|
2014-07-26 18:27:03 +00:00
|
|
|
!mp_input_use_alt_gr(w32->input_ctx))
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
|
|
|
keys[VK_RMENU] = keys[VK_LCONTROL] = 0;
|
|
|
|
keys[VK_MENU] = keys[VK_LMENU];
|
|
|
|
keys[VK_CONTROL] = keys[VK_RCONTROL];
|
|
|
|
}
|
|
|
|
|
|
|
|
int c = to_unicode(vkey, scancode, keys);
|
|
|
|
|
|
|
|
// Some shift states prevent ToUnicode from working or cause it to produce
|
|
|
|
// control characters. If this is detected, remove modifiers until it
|
|
|
|
// starts producing normal characters.
|
|
|
|
if (c < 0x20 && (keys[VK_MENU] & 0x80)) {
|
|
|
|
keys[VK_LMENU] = keys[VK_RMENU] = keys[VK_MENU] = 0;
|
|
|
|
c = to_unicode(vkey, scancode, keys);
|
|
|
|
}
|
|
|
|
if (c < 0x20 && (keys[VK_CONTROL] & 0x80)) {
|
|
|
|
keys[VK_LCONTROL] = keys[VK_RCONTROL] = keys[VK_CONTROL] = 0;
|
|
|
|
c = to_unicode(vkey, scancode, keys);
|
|
|
|
}
|
|
|
|
if (c < 0x20)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Decode lone UTF-16 surrogates (VK_PACKET can generate these)
|
|
|
|
if (c < 0x10000)
|
2014-07-26 18:27:03 +00:00
|
|
|
return decode_utf16(w32, c);
|
2014-04-17 15:25:45 +00:00
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2017-07-25 13:51:40 +00:00
|
|
|
static bool handle_appcommand(struct vo_w32_state *w32, UINT cmd)
|
|
|
|
{
|
|
|
|
if (!mp_input_use_media_keys(w32->input_ctx))
|
|
|
|
return false;
|
|
|
|
int mpkey = mp_w32_appcmd_to_mpkey(cmd);
|
|
|
|
if (!mpkey)
|
|
|
|
return false;
|
|
|
|
mp_input_put_key(w32->input_ctx, mpkey | mod_state(w32));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-11-08 11:14:29 +00:00
|
|
|
static void handle_key_down(struct vo_w32_state *w32, UINT vkey, UINT scancode)
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
|
|
|
// Ignore key repeat
|
|
|
|
if (scancode & KF_REPEAT)
|
2014-11-08 11:14:29 +00:00
|
|
|
return;
|
2014-04-17 15:25:45 +00:00
|
|
|
|
|
|
|
int mpkey = mp_w32_vkey_to_mpkey(vkey, scancode & KF_EXTENDED);
|
|
|
|
if (!mpkey) {
|
2014-07-26 18:27:03 +00:00
|
|
|
mpkey = decode_key(w32, vkey, scancode & (0xff | KF_EXTENDED));
|
2014-04-17 15:25:45 +00:00
|
|
|
if (!mpkey)
|
2014-11-08 11:14:29 +00:00
|
|
|
return;
|
2014-04-17 15:25:45 +00:00
|
|
|
}
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, mpkey | mod_state(w32) | MP_KEY_STATE_DOWN);
|
2014-04-17 15:25:45 +00:00
|
|
|
}
|
|
|
|
|
2014-11-08 11:14:29 +00:00
|
|
|
static void handle_key_up(struct vo_w32_state *w32, UINT vkey, UINT scancode)
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
|
|
|
switch (vkey) {
|
|
|
|
case VK_MENU:
|
|
|
|
case VK_CONTROL:
|
|
|
|
case VK_SHIFT:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Releasing all keys on key-up is simpler and ensures no keys can be
|
|
|
|
// get "stuck." This matches the behaviour of other VOs.
|
2014-07-26 18:27:03 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, MP_INPUT_RELEASE_ALL);
|
2014-04-17 15:25:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-01 18:51:52 +00:00
|
|
|
static bool handle_char(struct vo_w32_state *w32, wchar_t wc)
|
2014-04-17 15:25:45 +00:00
|
|
|
{
|
2014-07-26 18:27:03 +00:00
|
|
|
int c = decode_utf16(w32, wc);
|
2014-04-17 15:25:45 +00:00
|
|
|
|
|
|
|
if (c == 0)
|
|
|
|
return true;
|
|
|
|
if (c < 0x20)
|
|
|
|
return false;
|
|
|
|
|
2014-07-26 18:27:03 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, c | mod_state(w32));
|
2014-04-17 15:25:45 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-02 02:59:35 +00:00
|
|
|
static bool handle_mouse_down(struct vo_w32_state *w32, int btn, int x, int y)
|
|
|
|
{
|
|
|
|
btn |= mod_state(w32);
|
|
|
|
mp_input_put_key(w32->input_ctx, btn | MP_KEY_STATE_DOWN);
|
|
|
|
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
if (btn == MP_MBTN_LEFT && !w32->current_fs &&
|
2017-04-02 02:59:35 +00:00
|
|
|
!mp_input_test_dragging(w32->input_ctx, x, y))
|
|
|
|
{
|
|
|
|
// Window dragging hack
|
|
|
|
ReleaseCapture();
|
|
|
|
SendMessage(w32->window, WM_NCLBUTTONDOWN, HTCAPTION, 0);
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, MP_MBTN_LEFT | MP_KEY_STATE_UP);
|
2017-04-02 02:59:35 +00:00
|
|
|
|
|
|
|
// Indicate the message was handled, so DefWindowProc won't be called
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
SetCapture(w32->window);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_mouse_up(struct vo_w32_state *w32, int btn)
|
|
|
|
{
|
|
|
|
btn |= mod_state(w32);
|
|
|
|
mp_input_put_key(w32->input_ctx, btn | MP_KEY_STATE_UP);
|
|
|
|
|
|
|
|
ReleaseCapture();
|
|
|
|
}
|
|
|
|
|
2017-04-25 12:58:18 +00:00
|
|
|
static void handle_mouse_wheel(struct vo_w32_state *w32, bool horiz, int val)
|
|
|
|
{
|
|
|
|
int code;
|
|
|
|
if (horiz)
|
2017-09-02 14:00:52 +00:00
|
|
|
code = val > 0 ? MP_WHEEL_RIGHT : MP_WHEEL_LEFT;
|
2017-04-25 12:58:18 +00:00
|
|
|
else
|
2017-09-02 14:00:52 +00:00
|
|
|
code = val > 0 ? MP_WHEEL_UP : MP_WHEEL_DOWN;
|
|
|
|
mp_input_put_wheel(w32->input_ctx, code | mod_state(w32), abs(val) / 120.);
|
2017-04-25 12:58:18 +00:00
|
|
|
}
|
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
static void signal_events(struct vo_w32_state *w32, int events)
|
|
|
|
{
|
2016-08-05 14:01:26 +00:00
|
|
|
atomic_fetch_or(&w32->event_flags, events);
|
video: move display and timing to a separate thread
The VO is run inside its own thread. It also does most of video timing.
The playloop hands the image data and a realtime timestamp to the VO,
and the VO does the rest.
In particular, this allows the playloop to do other things, instead of
blocking for video redraw. But if anything accesses the VO during video
timing, it will block.
This also fixes vo_sdl.c event handling; but that is only a side-effect,
since reimplementing the broken way would require more effort.
Also drop --softsleep. In theory, this option helps if the kernel's
sleeping mechanism is too inaccurate for video timing. In practice, I
haven't ever encountered a situation where it helps, and it just burns
CPU cycles. On the other hand it's probably actively harmful, because
it prevents the libavcodec decoder threads from doing real work.
Side note:
Originally, I intended that multiple frames can be queued to the VO. But
this is not done, due to problems with OSD and other certain features.
OSD in particular is simply designed in a way that it can be neither
timed nor copied, so you do have to render it into the video frame
before you can draw the next frame. (Subtitles have no such restriction.
sd_lavc was even updated to fix this.) It seems the right solution to
queuing multiple VO frames is rendering on VO-backed framebuffers, like
vo_vdpau.c does. This requires VO driver support, and is out of scope
of this commit.
As consequence, the VO has a queue size of 1. The existing video queue
is just needed to compute frame duration, and will be moved out in the
next commit.
2014-08-12 21:02:08 +00:00
|
|
|
vo_wakeup(w32->vo);
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void wakeup_gui_thread(void *ctx)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = ctx;
|
2017-07-03 13:29:14 +00:00
|
|
|
// Wake up the window procedure (which processes the dispatch queue)
|
|
|
|
if (GetWindowThreadProcessId(w32->window, NULL) == GetCurrentThreadId()) {
|
|
|
|
PostMessageW(w32->window, WM_NULL, 0, 0);
|
|
|
|
} else {
|
|
|
|
// Use a sent message when cross-thread, since the queue of sent
|
|
|
|
// messages is processed in some cases when posted messages are blocked
|
|
|
|
SendNotifyMessageW(w32->window, WM_NULL, 0, 0);
|
|
|
|
}
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
}
|
|
|
|
|
2015-02-03 04:25:50 +00:00
|
|
|
static double get_refresh_rate_from_gdi(const wchar_t *device)
|
2015-03-10 18:25:30 +00:00
|
|
|
{
|
2015-02-03 04:25:50 +00:00
|
|
|
DEVMODEW dm = { .dmSize = sizeof dm };
|
|
|
|
if (!EnumDisplaySettingsW(device, ENUM_CURRENT_SETTINGS, &dm))
|
|
|
|
return 0.0;
|
2015-03-10 18:25:30 +00:00
|
|
|
|
|
|
|
// May return 0 or 1 which "represent the display hardware's default refresh rate"
|
|
|
|
// https://msdn.microsoft.com/en-us/library/windows/desktop/dd183565%28v=vs.85%29.aspx
|
|
|
|
// mpv validates this value with a threshold of 1, so don't return exactly 1
|
|
|
|
if (dm.dmDisplayFrequency == 1)
|
2015-02-03 04:25:50 +00:00
|
|
|
return 0.0;
|
2015-03-10 18:25:30 +00:00
|
|
|
|
|
|
|
// dm.dmDisplayFrequency is an integer which is rounded down, so it's
|
|
|
|
// highly likely that 23 represents 24/1.001, 59 represents 60/1.001, etc.
|
|
|
|
// A caller can always reproduce the original value by using floor.
|
|
|
|
double rv = dm.dmDisplayFrequency;
|
|
|
|
switch (dm.dmDisplayFrequency) {
|
|
|
|
case 23:
|
|
|
|
case 29:
|
2015-03-21 05:47:59 +00:00
|
|
|
case 47:
|
2015-03-10 18:25:30 +00:00
|
|
|
case 59:
|
|
|
|
case 71:
|
2015-03-21 05:47:59 +00:00
|
|
|
case 89:
|
|
|
|
case 95:
|
2015-03-10 18:25:30 +00:00
|
|
|
case 119:
|
2015-03-21 05:47:59 +00:00
|
|
|
case 143:
|
2015-03-10 18:25:30 +00:00
|
|
|
rv = (rv + 1) / 1.001;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2015-11-26 11:11:45 +00:00
|
|
|
static char *get_color_profile(void *ctx, const wchar_t *device)
|
|
|
|
{
|
|
|
|
char *name = NULL;
|
|
|
|
|
|
|
|
HDC ic = CreateICW(device, NULL, NULL, NULL);
|
|
|
|
if (!ic)
|
|
|
|
goto done;
|
|
|
|
wchar_t wname[MAX_PATH + 1];
|
|
|
|
if (!GetICMProfileW(ic, &(DWORD){ MAX_PATH }, wname))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
name = mp_to_utf8(ctx, wname);
|
|
|
|
done:
|
|
|
|
if (ic)
|
|
|
|
DeleteDC(ic);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
static void update_dpi(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
UINT dpiX, dpiY;
|
2021-08-09 19:04:29 +00:00
|
|
|
HDC hdc = NULL;
|
|
|
|
int dpi = 0;
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
if (w32->api.pGetDpiForMonitor && w32->api.pGetDpiForMonitor(w32->monitor,
|
|
|
|
MDT_EFFECTIVE_DPI, &dpiX, &dpiY) == S_OK) {
|
2021-08-09 19:04:29 +00:00
|
|
|
dpi = (int)dpiX;
|
|
|
|
MP_VERBOSE(w32, "DPI detected from the new API: %d\n", dpi);
|
|
|
|
} else if ((hdc = GetDC(NULL))) {
|
|
|
|
dpi = GetDeviceCaps(hdc, LOGPIXELSX);
|
2016-12-09 18:22:33 +00:00
|
|
|
ReleaseDC(NULL, hdc);
|
2021-08-09 19:04:29 +00:00
|
|
|
MP_VERBOSE(w32, "DPI detected from the old API: %d\n", dpi);
|
2016-12-09 18:22:33 +00:00
|
|
|
}
|
2022-05-12 15:12:31 +00:00
|
|
|
|
2021-08-09 19:04:29 +00:00
|
|
|
if (dpi <= 0) {
|
|
|
|
dpi = 96;
|
|
|
|
MP_VERBOSE(w32, "Couldn't determine DPI, falling back to %d\n", dpi);
|
|
|
|
}
|
|
|
|
|
|
|
|
w32->dpi = dpi;
|
|
|
|
w32->dpi_scale = w32->opts->hidpi_window_scale ? w32->dpi / 96.0 : 1.0;
|
2016-12-09 18:22:33 +00:00
|
|
|
}
|
|
|
|
|
2015-11-26 11:11:45 +00:00
|
|
|
static void update_display_info(struct vo_w32_state *w32)
|
2015-03-10 18:25:30 +00:00
|
|
|
{
|
2015-02-03 04:25:50 +00:00
|
|
|
HMONITOR monitor = MonitorFromWindow(w32->window, MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
if (w32->monitor == monitor)
|
|
|
|
return;
|
|
|
|
w32->monitor = monitor;
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
update_dpi(w32);
|
|
|
|
|
2015-02-03 04:25:50 +00:00
|
|
|
MONITORINFOEXW mi = { .cbSize = sizeof mi };
|
|
|
|
GetMonitorInfoW(monitor, (MONITORINFO*)&mi);
|
|
|
|
|
|
|
|
// Try to get the monitor refresh rate.
|
|
|
|
double freq = 0.0;
|
|
|
|
|
|
|
|
if (freq == 0.0)
|
|
|
|
freq = mp_w32_displayconfig_get_refresh_rate(mi.szDevice);
|
|
|
|
if (freq == 0.0)
|
|
|
|
freq = get_refresh_rate_from_gdi(mi.szDevice);
|
|
|
|
|
|
|
|
if (freq != w32->display_fps) {
|
|
|
|
MP_VERBOSE(w32, "display-fps: %f\n", freq);
|
|
|
|
if (freq == 0.0)
|
|
|
|
MP_WARN(w32, "Couldn't determine monitor refresh rate\n");
|
|
|
|
w32->display_fps = freq;
|
2015-03-10 18:25:30 +00:00
|
|
|
signal_events(w32, VO_EVENT_WIN_STATE);
|
|
|
|
}
|
2015-11-26 11:11:45 +00:00
|
|
|
|
|
|
|
char *color_profile = get_color_profile(w32, mi.szDevice);
|
|
|
|
if ((color_profile == NULL) != (w32->color_profile == NULL) ||
|
|
|
|
(color_profile && strcmp(color_profile, w32->color_profile)))
|
|
|
|
{
|
|
|
|
if (color_profile)
|
|
|
|
MP_VERBOSE(w32, "color-profile: %s\n", color_profile);
|
|
|
|
talloc_free(w32->color_profile);
|
|
|
|
w32->color_profile = color_profile;
|
|
|
|
color_profile = NULL;
|
|
|
|
signal_events(w32, VO_EVENT_ICC_PROFILE_CHANGED);
|
|
|
|
}
|
|
|
|
|
|
|
|
talloc_free(color_profile);
|
2015-03-10 18:25:30 +00:00
|
|
|
}
|
|
|
|
|
2015-11-26 11:11:45 +00:00
|
|
|
static void force_update_display_info(struct vo_w32_state *w32)
|
2015-02-03 04:25:50 +00:00
|
|
|
{
|
|
|
|
w32->monitor = 0;
|
2015-11-26 11:11:45 +00:00
|
|
|
update_display_info(w32);
|
2015-02-03 04:25:50 +00:00
|
|
|
}
|
|
|
|
|
2016-09-02 16:08:49 +00:00
|
|
|
static void update_playback_state(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
struct voctrl_playback_state *pstate = &w32->current_pstate;
|
|
|
|
|
|
|
|
if (!w32->taskbar_list3 || !w32->tbtnCreated)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!pstate->playing || !pstate->taskbar_progress) {
|
|
|
|
ITaskbarList3_SetProgressState(w32->taskbar_list3, w32->window,
|
|
|
|
TBPF_NOPROGRESS);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ITaskbarList3_SetProgressValue(w32->taskbar_list3, w32->window,
|
|
|
|
pstate->percent_pos, 100);
|
|
|
|
ITaskbarList3_SetProgressState(w32->taskbar_list3, w32->window,
|
|
|
|
pstate->paused ? TBPF_PAUSED :
|
|
|
|
TBPF_NORMAL);
|
|
|
|
}
|
|
|
|
|
2017-12-22 16:19:28 +00:00
|
|
|
struct get_monitor_data {
|
|
|
|
int i;
|
|
|
|
int target;
|
|
|
|
HMONITOR mon;
|
|
|
|
};
|
|
|
|
|
|
|
|
static BOOL CALLBACK get_monitor_proc(HMONITOR mon, HDC dc, LPRECT r, LPARAM p)
|
|
|
|
{
|
|
|
|
struct get_monitor_data *data = (struct get_monitor_data*)p;
|
|
|
|
|
|
|
|
if (data->i == data->target) {
|
|
|
|
data->mon = mon;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
data->i++;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HMONITOR get_monitor(int id)
|
|
|
|
{
|
|
|
|
struct get_monitor_data data = { .target = id };
|
|
|
|
EnumDisplayMonitors(NULL, NULL, get_monitor_proc, (LPARAM)&data);
|
|
|
|
return data.mon;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HMONITOR get_default_monitor(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
const int id = w32->current_fs ? w32->opts->fsscreen_id :
|
|
|
|
w32->opts->screen_id;
|
|
|
|
|
|
|
|
// Handle --fs-screen=<all|default> and --screen=default
|
|
|
|
if (id < 0)
|
|
|
|
return MonitorFromWindow(w32->window, MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
|
|
|
|
HMONITOR mon = get_monitor(id);
|
|
|
|
if (mon)
|
|
|
|
return mon;
|
|
|
|
MP_VERBOSE(w32, "Screen %d does not exist, falling back to primary\n", id);
|
|
|
|
return MonitorFromPoint((POINT){0, 0}, MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MONITORINFO get_monitor_info(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
HMONITOR mon;
|
|
|
|
if (IsWindowVisible(w32->window) && !w32->current_fs) {
|
|
|
|
mon = MonitorFromWindow(w32->window, MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
} else {
|
|
|
|
// The window is not visible during initialization, so get the
|
|
|
|
// monitor by --screen or --fs-screen id, or fallback to primary.
|
|
|
|
mon = get_default_monitor(w32);
|
|
|
|
}
|
|
|
|
MONITORINFO mi = { .cbSize = sizeof(mi) };
|
|
|
|
GetMonitorInfoW(mon, &mi);
|
|
|
|
return mi;
|
|
|
|
}
|
|
|
|
|
|
|
|
static RECT get_screen_area(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
// Handle --fs-screen=all
|
|
|
|
if (w32->current_fs && w32->opts->fsscreen_id == -2) {
|
|
|
|
const int x = GetSystemMetrics(SM_XVIRTUALSCREEN);
|
|
|
|
const int y = GetSystemMetrics(SM_YVIRTUALSCREEN);
|
|
|
|
return (RECT) { x, y, x + GetSystemMetrics(SM_CXVIRTUALSCREEN),
|
|
|
|
y + GetSystemMetrics(SM_CYVIRTUALSCREEN) };
|
|
|
|
}
|
|
|
|
return get_monitor_info(w32).rcMonitor;
|
|
|
|
}
|
|
|
|
|
|
|
|
static RECT get_working_area(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
return w32->current_fs ? get_screen_area(w32) :
|
|
|
|
get_monitor_info(w32).rcWork;
|
|
|
|
}
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
static bool snap_to_screen_edges(struct vo_w32_state *w32, RECT *rc)
|
|
|
|
{
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
if (w32->parent || w32->current_fs || IsMaximized(w32->window))
|
|
|
|
return false;
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
if (!w32->opts->snap_window) {
|
|
|
|
w32->snapped = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
RECT rect;
|
|
|
|
POINT cursor;
|
|
|
|
if (!GetWindowRect(w32->window, &rect) || !GetCursorPos(&cursor))
|
|
|
|
return false;
|
2017-12-04 23:12:49 +00:00
|
|
|
// Check if window is going to be aero-snapped
|
2017-12-17 00:19:36 +00:00
|
|
|
if (rect_w(*rc) != rect_w(rect) || rect_h(*rc) != rect_h(rect))
|
2016-12-09 18:22:33 +00:00
|
|
|
return false;
|
|
|
|
|
2017-12-04 23:12:49 +00:00
|
|
|
// Check if window has already been aero-snapped
|
|
|
|
WINDOWPLACEMENT wp = {0};
|
|
|
|
wp.length = sizeof(wp);
|
|
|
|
if (!GetWindowPlacement(w32->window, &wp))
|
|
|
|
return false;
|
|
|
|
RECT wr = wp.rcNormalPosition;
|
2017-12-17 00:19:36 +00:00
|
|
|
if (rect_w(*rc) != rect_w(wr) || rect_h(*rc) != rect_h(wr))
|
2017-12-04 23:12:49 +00:00
|
|
|
return false;
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
// Get the work area to let the window snap to taskbar
|
2017-12-22 16:19:28 +00:00
|
|
|
wr = get_working_area(w32);
|
2016-12-09 18:22:33 +00:00
|
|
|
|
|
|
|
// Check for invisible borders and adjust the work area size
|
|
|
|
RECT frame = {0};
|
|
|
|
if (DwmGetWindowAttribute(w32->window, DWMWA_EXTENDED_FRAME_BOUNDS,
|
|
|
|
&frame, sizeof(RECT)) == S_OK) {
|
|
|
|
wr.left -= frame.left - rect.left;
|
|
|
|
wr.top -= frame.top - rect.top;
|
|
|
|
wr.right += rect.right - frame.right;
|
|
|
|
wr.bottom += rect.bottom - frame.bottom;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Let the window to unsnap by changing its position,
|
|
|
|
// otherwise it will stick to the screen edges forever
|
|
|
|
rect = *rc;
|
|
|
|
if (w32->snapped) {
|
|
|
|
OffsetRect(&rect, cursor.x - rect.left - w32->snap_dx,
|
|
|
|
cursor.y - rect.top - w32->snap_dy);
|
|
|
|
}
|
|
|
|
|
|
|
|
int threshold = (w32->dpi * 16) / 96;
|
|
|
|
bool snapped = false;
|
|
|
|
// Adjust X position
|
|
|
|
if (abs(rect.left - wr.left) < threshold) {
|
|
|
|
snapped = true;
|
|
|
|
OffsetRect(&rect, wr.left - rect.left, 0);
|
|
|
|
} else if (abs(rect.right - wr.right) < threshold) {
|
|
|
|
snapped = true;
|
|
|
|
OffsetRect(&rect, wr.right - rect.right, 0);
|
|
|
|
}
|
|
|
|
// Adjust Y position
|
|
|
|
if (abs(rect.top - wr.top) < threshold) {
|
|
|
|
snapped = true;
|
|
|
|
OffsetRect(&rect, 0, wr.top - rect.top);
|
|
|
|
} else if (abs(rect.bottom - wr.bottom) < threshold) {
|
|
|
|
snapped = true;
|
|
|
|
OffsetRect(&rect, 0, wr.bottom - rect.bottom);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!w32->snapped && snapped) {
|
|
|
|
w32->snap_dx = cursor.x - rc->left;
|
|
|
|
w32->snap_dy = cursor.y - rc->top;
|
|
|
|
}
|
|
|
|
|
|
|
|
w32->snapped = snapped;
|
|
|
|
*rc = rect;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-01 13:13:18 +00:00
|
|
|
static DWORD update_style(struct vo_w32_state *w32, DWORD style)
|
|
|
|
{
|
|
|
|
const DWORD NO_FRAME = WS_OVERLAPPED | WS_MINIMIZEBOX;
|
|
|
|
const DWORD FRAME = WS_OVERLAPPEDWINDOW;
|
|
|
|
const DWORD FULLSCREEN = NO_FRAME | WS_SYSMENU;
|
|
|
|
style &= ~(NO_FRAME | FRAME | FULLSCREEN);
|
|
|
|
if (w32->current_fs) {
|
|
|
|
style |= FULLSCREEN;
|
|
|
|
} else {
|
|
|
|
style |= w32->opts->border ? FRAME : NO_FRAME;
|
|
|
|
}
|
|
|
|
return style;
|
|
|
|
}
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
static void update_window_style(struct vo_w32_state *w32)
|
2017-04-01 13:13:18 +00:00
|
|
|
{
|
|
|
|
if (w32->parent)
|
|
|
|
return;
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// SetWindowLongPtr can trigger a WM_SIZE event, so window rect
|
|
|
|
// has to be saved now and restored after setting the new style.
|
|
|
|
const RECT wr = w32->windowrc;
|
|
|
|
const DWORD style = GetWindowLongPtrW(w32->window, GWL_STYLE);
|
|
|
|
SetWindowLongPtrW(w32->window, GWL_STYLE, update_style(w32, style));
|
|
|
|
w32->windowrc = wr;
|
|
|
|
}
|
2017-04-01 13:13:18 +00:00
|
|
|
|
2021-04-17 12:18:27 +00:00
|
|
|
// If rc is wider/taller than n_w/n_h, shrink rc size while keeping the center.
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// returns true if the rectangle was modified.
|
2021-04-17 12:18:27 +00:00
|
|
|
static bool fit_rect_size(RECT *rc, long n_w, long n_h)
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
{
|
2021-04-17 12:18:27 +00:00
|
|
|
// nothing to do if we already fit.
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
int o_w = rect_w(*rc), o_h = rect_h(*rc);
|
|
|
|
if (o_w <= n_w && o_h <= n_h)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Apply letterboxing
|
|
|
|
const float o_asp = o_w / (float)MPMAX(o_h, 1);
|
|
|
|
const float n_asp = n_w / (float)MPMAX(n_h, 1);
|
|
|
|
if (o_asp > n_asp) {
|
|
|
|
n_h = n_w / o_asp;
|
|
|
|
} else {
|
|
|
|
n_w = n_h * o_asp;
|
2017-04-01 13:13:18 +00:00
|
|
|
}
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// Calculate new position and save the rect
|
|
|
|
const int x = rc->left + o_w / 2 - n_w / 2;
|
|
|
|
const int y = rc->top + o_h / 2 - n_h / 2;
|
|
|
|
SetRect(rc, x, y, x + n_w, y + n_h);
|
|
|
|
return true;
|
|
|
|
}
|
2017-04-01 13:13:18 +00:00
|
|
|
|
2021-04-17 12:18:27 +00:00
|
|
|
// If the window is bigger than the desktop, shrink to fit with same center.
|
|
|
|
// Also, if the top edge is above the working area, move down to align.
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
static void fit_window_on_screen(struct vo_w32_state *w32)
|
|
|
|
{
|
2017-12-22 16:19:28 +00:00
|
|
|
RECT screen = get_working_area(w32);
|
options: win32: ignore and deprecate --fit-border
The accurate description of this option was:
- fit-border is enabled by default. When disabled, it adds a bug where
if the window has borders and mpv shrinks it to fit the desktop, then
the calculation ignores the borders and adds incorrect video crop.
The option was added at commits 70f64f3c and 949247d6, in order to
solve an issue (#2935) where if mpv wanted to display a video with
size WxH, then w32_common.c incorrectly set the window to WxH, while
down-scaling the video slightly to fit (even with small sizes).
It was addressed with a new option which is enabled by default, but
does the right thing (sets the client area to WxH) only when disabled,
so that everyone who prefers their video slightly downscaled could
keep their default behavior.
(#2935 also addressed an off-by-one issue, fixed before fit-border)
While disabling the option did avoid unnecessary downscaling, it also
added a bug when disabled: the borders are no longer taken into
account when the size is too big for the desktop. Most users don't
notice and are unaffected as it's enabled by default.
Shortly later (981048e0) the core issue is fixed, and now the client
area is correctly set to WxH instead of the window (and together with
the three following commits which center the video, adds a new bug
where the window title can be outside the display - addressed next).
However, fit-border remained, now without any effect, except that it
still has the same bug when disabled and the window is too big.
Later code changes and refactoring preserved this issue with great
attention to details, and it remained in identical form until now.
Simply rip out fit-border.
2021-04-17 10:18:43 +00:00
|
|
|
if (w32->opts->border)
|
2018-11-17 20:56:56 +00:00
|
|
|
subtract_window_borders(w32, w32->window, &screen);
|
2017-04-01 13:13:18 +00:00
|
|
|
|
2021-04-17 12:18:27 +00:00
|
|
|
bool adjusted = fit_rect_size(&w32->windowrc, rect_w(screen), rect_h(screen));
|
|
|
|
|
|
|
|
if (w32->windowrc.top < screen.top) {
|
|
|
|
// if the top-edge of client area is above the target area (mainly
|
|
|
|
// because the client-area is centered but the title bar is taller
|
|
|
|
// than the bottom border), then move it down to align the edges.
|
|
|
|
// Windows itself applies the same constraint during manual move.
|
|
|
|
w32->windowrc.bottom += screen.top - w32->windowrc.top;
|
|
|
|
w32->windowrc.top = screen.top;
|
|
|
|
adjusted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (adjusted) {
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
MP_VERBOSE(w32, "adjusted window bounds: %d:%d:%d:%d\n",
|
|
|
|
(int)w32->windowrc.left, (int)w32->windowrc.top,
|
|
|
|
(int)rect_w(w32->windowrc), (int)rect_h(w32->windowrc));
|
2017-04-01 13:13:18 +00:00
|
|
|
}
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
}
|
2017-04-01 13:13:18 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// Calculate new fullscreen state and change window size and position.
|
2021-04-22 12:52:07 +00:00
|
|
|
static void update_fullscreen_state(struct vo_w32_state *w32)
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
{
|
|
|
|
if (w32->parent)
|
2021-04-22 12:52:07 +00:00
|
|
|
return;
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
|
|
|
|
bool new_fs = w32->opts->fullscreen;
|
|
|
|
if (w32->toggle_fs) {
|
|
|
|
new_fs = !w32->current_fs;
|
|
|
|
w32->toggle_fs = false;
|
2017-04-01 13:13:18 +00:00
|
|
|
}
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
bool toggle_fs = w32->current_fs != new_fs;
|
2019-12-11 18:40:33 +00:00
|
|
|
w32->opts->fullscreen = w32->current_fs = new_fs;
|
|
|
|
m_config_cache_write_opt(w32->opts_cache,
|
|
|
|
&w32->opts->fullscreen);
|
2017-12-17 00:19:36 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
if (toggle_fs) {
|
|
|
|
if (w32->current_fs) {
|
|
|
|
// Save window rect when switching to fullscreen.
|
2019-05-10 10:47:05 +00:00
|
|
|
w32->prev_windowrc = w32->windowrc;
|
|
|
|
MP_VERBOSE(w32, "save window bounds: %d:%d:%d:%d\n",
|
|
|
|
(int)w32->windowrc.left, (int)w32->windowrc.top,
|
|
|
|
(int)rect_w(w32->windowrc), (int)rect_h(w32->windowrc));
|
2017-04-01 13:13:18 +00:00
|
|
|
} else {
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// Restore window rect when switching from fullscreen.
|
2019-05-10 10:47:05 +00:00
|
|
|
w32->windowrc = w32->prev_windowrc;
|
2017-04-01 13:13:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
if (w32->current_fs)
|
2017-12-22 16:19:28 +00:00
|
|
|
w32->windowrc = get_screen_area(w32);
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
|
2017-04-01 13:13:18 +00:00
|
|
|
MP_VERBOSE(w32, "reset window bounds: %d:%d:%d:%d\n",
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
(int)w32->windowrc.left, (int)w32->windowrc.top,
|
|
|
|
(int)rect_w(w32->windowrc), (int)rect_h(w32->windowrc));
|
|
|
|
}
|
|
|
|
|
w32_common: support minimized and maximized properties
Add support for setting window-minimized and window-maximized in
Windows. The minimized and maximized state can be set independently.
When the window is minimized, the value of window-maximized will
determine whether the window is restored to the maximized state or not.
Changing state is done with ShowWindow(), which has commands that change
the window state and activate it (eg. SW_RESTORE) and commands that
change the window state without activating it (eg. SW_SHOWNOACTIVATE.)
It would be nice if we could use commands that don't activate the
window, so scripts could change the window state in the backrgound
without bringing it to the foreground, but there are some problems with
that. There is no command to maximize a window without activating it, so
SW_MAXIMIZE is used instead. Also, restoring a window from minimize
without activating it seems buggy. On my Windows 10 1909 PC, it always
moves the window to the back of the z-order. SW_RESTORE is used instead
of SW_SHOWNOACTIVATE because of this.
This also changes the way the window is initially shown. Previously, the
window was made visible as a consequence of the SWP_SHOWWINDOW flag in
the first call to SetWindowPos. In order to set the initial minimized or
maximized state of the window, the window is shown with the ShowWindow
function instead, where the ShowWindow command is determined by whether
the window should be initially maximized or minimized.
Even when showing the window normally, we should still call ShowWindow
with the SW_SHOW command instead of using SetWindowPos, since the first
call a process makes to ShowWindow(SW_SHOW) has special behaviour
where it uses the show command in the process' STARTUPINFO instead of
the command passed to the function, which should fix #5724.
Note: While changes to window-minimized while in fullscreen mode should
work as expected, changing window-maximized while in fullscreen does not
work and won't result in the window changing state, even after leaving
fullscreen. For this to work correctly, the fullscreen logic needs to be
changed to apply the new maximized state on leaving fullscreen.
Fixes: #5724
Fixes: #7351
2019-12-27 15:12:44 +00:00
|
|
|
static void update_minimized_state(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
if (w32->parent)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!!IsMinimized(w32->window) != w32->opts->window_minimized) {
|
|
|
|
if (w32->opts->window_minimized) {
|
|
|
|
ShowWindow(w32->window, SW_SHOWMINNOACTIVE);
|
|
|
|
} else {
|
|
|
|
ShowWindow(w32->window, SW_RESTORE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void update_maximized_state(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
if (w32->parent)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Don't change the maximized state in fullscreen for now. In future, this
|
|
|
|
// should be made to apply the maximized state on leaving fullscreen.
|
|
|
|
if (w32->current_fs)
|
|
|
|
return;
|
|
|
|
|
|
|
|
WINDOWPLACEMENT wp = { .length = sizeof wp };
|
|
|
|
GetWindowPlacement(w32->window, &wp);
|
|
|
|
|
|
|
|
if (wp.showCmd == SW_SHOWMINIMIZED) {
|
|
|
|
// When the window is minimized, setting this property just changes
|
|
|
|
// whether it will be maximized when it's restored
|
|
|
|
if (w32->opts->window_maximized) {
|
|
|
|
wp.flags |= WPF_RESTORETOMAXIMIZED;
|
|
|
|
} else {
|
|
|
|
wp.flags &= ~WPF_RESTORETOMAXIMIZED;
|
|
|
|
}
|
|
|
|
SetWindowPlacement(w32->window, &wp);
|
|
|
|
} else if ((wp.showCmd == SW_SHOWMAXIMIZED) != w32->opts->window_maximized) {
|
|
|
|
if (w32->opts->window_maximized) {
|
|
|
|
ShowWindow(w32->window, SW_SHOWMAXIMIZED);
|
|
|
|
} else {
|
|
|
|
ShowWindow(w32->window, SW_SHOWNOACTIVATE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool is_visible(HWND window)
|
|
|
|
{
|
|
|
|
// Unlike IsWindowVisible, this doesn't check the window's parents
|
|
|
|
return GetWindowLongPtrW(window, GWL_STYLE) & WS_VISIBLE;
|
|
|
|
}
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
static void update_window_state(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
if (w32->parent)
|
|
|
|
return;
|
2017-04-01 13:13:18 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
RECT wr = w32->windowrc;
|
2018-11-17 20:56:56 +00:00
|
|
|
add_window_borders(w32, w32->window, &wr);
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
|
|
|
|
SetWindowPos(w32->window, w32->opts->ontop ? HWND_TOPMOST : HWND_NOTOPMOST,
|
|
|
|
wr.left, wr.top, rect_w(wr), rect_h(wr),
|
w32_common: support minimized and maximized properties
Add support for setting window-minimized and window-maximized in
Windows. The minimized and maximized state can be set independently.
When the window is minimized, the value of window-maximized will
determine whether the window is restored to the maximized state or not.
Changing state is done with ShowWindow(), which has commands that change
the window state and activate it (eg. SW_RESTORE) and commands that
change the window state without activating it (eg. SW_SHOWNOACTIVATE.)
It would be nice if we could use commands that don't activate the
window, so scripts could change the window state in the backrgound
without bringing it to the foreground, but there are some problems with
that. There is no command to maximize a window without activating it, so
SW_MAXIMIZE is used instead. Also, restoring a window from minimize
without activating it seems buggy. On my Windows 10 1909 PC, it always
moves the window to the back of the z-order. SW_RESTORE is used instead
of SW_SHOWNOACTIVATE because of this.
This also changes the way the window is initially shown. Previously, the
window was made visible as a consequence of the SWP_SHOWWINDOW flag in
the first call to SetWindowPos. In order to set the initial minimized or
maximized state of the window, the window is shown with the ShowWindow
function instead, where the ShowWindow command is determined by whether
the window should be initially maximized or minimized.
Even when showing the window normally, we should still call ShowWindow
with the SW_SHOW command instead of using SetWindowPos, since the first
call a process makes to ShowWindow(SW_SHOW) has special behaviour
where it uses the show command in the process' STARTUPINFO instead of
the command passed to the function, which should fix #5724.
Note: While changes to window-minimized while in fullscreen mode should
work as expected, changing window-maximized while in fullscreen does not
work and won't result in the window changing state, even after leaving
fullscreen. For this to work correctly, the fullscreen logic needs to be
changed to apply the new maximized state on leaving fullscreen.
Fixes: #5724
Fixes: #7351
2019-12-27 15:12:44 +00:00
|
|
|
SWP_FRAMECHANGED | SWP_NOACTIVATE | SWP_NOOWNERZORDER);
|
|
|
|
|
|
|
|
// Show the window if it's not yet visible
|
|
|
|
if (!is_visible(w32->window)) {
|
|
|
|
if (w32->opts->window_minimized) {
|
|
|
|
ShowWindow(w32->window, SW_SHOWMINIMIZED);
|
|
|
|
update_maximized_state(w32); // Set the WPF_RESTORETOMAXIMIZED flag
|
|
|
|
} else if (w32->opts->window_maximized) {
|
|
|
|
ShowWindow(w32->window, SW_SHOWMAXIMIZED);
|
|
|
|
} else {
|
|
|
|
ShowWindow(w32->window, SW_SHOW);
|
|
|
|
}
|
|
|
|
}
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
|
|
|
|
// Notify the taskbar about the fullscreen state only after the window
|
|
|
|
// is visible, to make sure the taskbar item has already been created
|
|
|
|
if (w32->taskbar_list) {
|
|
|
|
ITaskbarList2_MarkFullscreenWindow(w32->taskbar_list,
|
|
|
|
w32->window, w32->current_fs);
|
|
|
|
}
|
2017-04-01 13:13:18 +00:00
|
|
|
|
|
|
|
signal_events(w32, VO_EVENT_RESIZE);
|
2016-12-13 03:29:50 +00:00
|
|
|
}
|
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
static void reinit_window_state(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
if (w32->parent)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// The order matters: fs state should be updated prior to changing styles
|
2021-04-22 12:52:07 +00:00
|
|
|
update_fullscreen_state(w32);
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
update_window_style(w32);
|
|
|
|
|
2021-04-22 12:52:07 +00:00
|
|
|
// fit_on_screen is applied at most once when/if applicable (normal win).
|
|
|
|
if (w32->fit_on_screen && !w32->current_fs && !IsMaximized(w32->window)) {
|
|
|
|
fit_window_on_screen(w32);
|
|
|
|
w32->fit_on_screen = false;
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Show and activate the window after all window state parameters were set
|
|
|
|
update_window_state(w32);
|
|
|
|
}
|
|
|
|
|
2012-04-14 11:39:53 +00:00
|
|
|
static LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam,
|
|
|
|
LPARAM lParam)
|
|
|
|
{
|
2017-05-15 14:27:26 +00:00
|
|
|
struct vo_w32_state *w32 = (void*)GetWindowLongPtrW(hWnd, GWLP_USERDATA);
|
|
|
|
if (!w32) {
|
|
|
|
// WM_NCCREATE is supposed to be the first message that a window
|
|
|
|
// receives. It allows struct vo_w32_state to be passed from
|
|
|
|
// CreateWindow's lpParam to the window procedure. However, as a
|
|
|
|
// longstanding Windows bug, overlapped top-level windows will get a
|
|
|
|
// WM_GETMINMAXINFO before WM_NCCREATE. This can be ignored.
|
|
|
|
if (message != WM_NCCREATE)
|
|
|
|
return DefWindowProcW(hWnd, message, wParam, lParam);
|
|
|
|
|
|
|
|
CREATESTRUCTW *cs = (CREATESTRUCTW *)lParam;
|
|
|
|
w32 = cs->lpCreateParams;
|
|
|
|
w32->window = hWnd;
|
|
|
|
SetWindowLongPtrW(hWnd, GWLP_USERDATA, (LONG_PTR)w32);
|
|
|
|
}
|
2012-04-14 11:39:53 +00:00
|
|
|
|
2017-07-03 13:29:14 +00:00
|
|
|
// The dispatch queue should be processed as soon as possible to prevent
|
|
|
|
// playback glitches, since it is likely blocking the VO thread
|
2018-04-17 16:29:32 +00:00
|
|
|
if (!w32->in_dispatch) {
|
|
|
|
w32->in_dispatch = true;
|
|
|
|
mp_dispatch_queue_process(w32->dispatch, 0);
|
|
|
|
w32->in_dispatch = false;
|
|
|
|
}
|
2017-07-03 13:29:14 +00:00
|
|
|
|
2003-09-19 14:33:51 +00:00
|
|
|
switch (message) {
|
2014-08-06 18:00:26 +00:00
|
|
|
case WM_ERASEBKGND: // no need to erase background separately
|
2014-01-06 14:35:27 +00:00
|
|
|
return 1;
|
|
|
|
case WM_PAINT:
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
signal_events(w32, VO_EVENT_EXPOSE);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_MOVE: {
|
2017-12-17 00:19:36 +00:00
|
|
|
const int x = GET_X_LPARAM(lParam), y = GET_Y_LPARAM(lParam);
|
|
|
|
OffsetRect(&w32->windowrc, x - w32->windowrc.left,
|
|
|
|
y - w32->windowrc.top);
|
2016-09-29 13:21:23 +00:00
|
|
|
|
|
|
|
// Window may intersect with new monitors (see VOCTRL_GET_DISPLAY_NAMES)
|
|
|
|
signal_events(w32, VO_EVENT_WIN_STATE);
|
|
|
|
|
2015-11-26 11:11:45 +00:00
|
|
|
update_display_info(w32); // if we moved between monitors
|
2017-12-17 00:19:36 +00:00
|
|
|
MP_DBG(w32, "move window: %d:%d\n", x, y);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-12-09 18:22:33 +00:00
|
|
|
case WM_MOVING: {
|
2017-12-04 23:21:34 +00:00
|
|
|
w32->moving = true;
|
2016-12-09 18:22:33 +00:00
|
|
|
RECT *rc = (RECT*)lParam;
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
if (snap_to_screen_edges(w32, rc))
|
2016-12-09 18:22:33 +00:00
|
|
|
return TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case WM_ENTERSIZEMOVE:
|
2017-12-04 23:21:34 +00:00
|
|
|
w32->moving = true;
|
2016-12-09 18:22:33 +00:00
|
|
|
if (w32->snapped) {
|
|
|
|
// Save the cursor offset from the window borders,
|
|
|
|
// so the player window can be unsnapped later
|
|
|
|
RECT rc;
|
|
|
|
POINT cursor;
|
|
|
|
if (GetWindowRect(w32->window, &rc) && GetCursorPos(&cursor)) {
|
|
|
|
w32->snap_dx = cursor.x - rc.left;
|
|
|
|
w32->snap_dy = cursor.y - rc.top;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2017-12-04 23:21:34 +00:00
|
|
|
case WM_EXITSIZEMOVE:
|
|
|
|
w32->moving = false;
|
|
|
|
break;
|
2014-01-06 14:35:27 +00:00
|
|
|
case WM_SIZE: {
|
2017-12-04 23:21:34 +00:00
|
|
|
if (w32->moving)
|
|
|
|
w32->snapped = false;
|
|
|
|
|
2017-12-17 00:19:36 +00:00
|
|
|
const int w = LOWORD(lParam), h = HIWORD(lParam);
|
|
|
|
if (w > 0 && h > 0) {
|
|
|
|
w32->windowrc.right = w32->windowrc.left + w;
|
|
|
|
w32->windowrc.bottom = w32->windowrc.top + h;
|
2015-02-02 21:52:13 +00:00
|
|
|
signal_events(w32, VO_EVENT_RESIZE);
|
2017-12-17 00:19:36 +00:00
|
|
|
MP_VERBOSE(w32, "resize window: %d:%d\n", w, h);
|
2015-02-02 21:52:13 +00:00
|
|
|
}
|
2015-03-09 09:30:33 +00:00
|
|
|
|
w32_common: support minimized and maximized properties
Add support for setting window-minimized and window-maximized in
Windows. The minimized and maximized state can be set independently.
When the window is minimized, the value of window-maximized will
determine whether the window is restored to the maximized state or not.
Changing state is done with ShowWindow(), which has commands that change
the window state and activate it (eg. SW_RESTORE) and commands that
change the window state without activating it (eg. SW_SHOWNOACTIVATE.)
It would be nice if we could use commands that don't activate the
window, so scripts could change the window state in the backrgound
without bringing it to the foreground, but there are some problems with
that. There is no command to maximize a window without activating it, so
SW_MAXIMIZE is used instead. Also, restoring a window from minimize
without activating it seems buggy. On my Windows 10 1909 PC, it always
moves the window to the back of the z-order. SW_RESTORE is used instead
of SW_SHOWNOACTIVATE because of this.
This also changes the way the window is initially shown. Previously, the
window was made visible as a consequence of the SWP_SHOWWINDOW flag in
the first call to SetWindowPos. In order to set the initial minimized or
maximized state of the window, the window is shown with the ShowWindow
function instead, where the ShowWindow command is determined by whether
the window should be initially maximized or minimized.
Even when showing the window normally, we should still call ShowWindow
with the SW_SHOW command instead of using SetWindowPos, since the first
call a process makes to ShowWindow(SW_SHOW) has special behaviour
where it uses the show command in the process' STARTUPINFO instead of
the command passed to the function, which should fix #5724.
Note: While changes to window-minimized while in fullscreen mode should
work as expected, changing window-maximized while in fullscreen does not
work and won't result in the window changing state, even after leaving
fullscreen. For this to work correctly, the fullscreen logic needs to be
changed to apply the new maximized state on leaving fullscreen.
Fixes: #5724
Fixes: #7351
2019-12-27 15:12:44 +00:00
|
|
|
// Window may have been minimized, maximized or restored
|
|
|
|
if (is_visible(w32->window)) {
|
|
|
|
WINDOWPLACEMENT wp = { .length = sizeof wp };
|
|
|
|
GetWindowPlacement(w32->window, &wp);
|
|
|
|
|
|
|
|
bool is_minimized = wp.showCmd == SW_SHOWMINIMIZED;
|
|
|
|
if (w32->opts->window_minimized != is_minimized) {
|
|
|
|
w32->opts->window_minimized = is_minimized;
|
|
|
|
m_config_cache_write_opt(w32->opts_cache,
|
|
|
|
&w32->opts->window_minimized);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_maximized = wp.showCmd == SW_SHOWMAXIMIZED ||
|
|
|
|
(wp.showCmd == SW_SHOWMINIMIZED &&
|
|
|
|
(wp.flags & WPF_RESTORETOMAXIMIZED));
|
|
|
|
if (w32->opts->window_maximized != is_maximized) {
|
|
|
|
w32->opts->window_maximized = is_maximized;
|
|
|
|
m_config_cache_write_opt(w32->opts_cache,
|
|
|
|
&w32->opts->window_maximized);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-09 09:30:33 +00:00
|
|
|
signal_events(w32, VO_EVENT_WIN_STATE);
|
2015-02-03 04:25:50 +00:00
|
|
|
|
2015-11-26 11:11:45 +00:00
|
|
|
update_display_info(w32);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case WM_SIZING:
|
2014-10-04 20:17:36 +00:00
|
|
|
if (w32->opts->keepaspect && w32->opts->keepaspect_window &&
|
2015-06-20 11:33:23 +00:00
|
|
|
!w32->current_fs && !w32->parent)
|
2014-01-06 14:35:27 +00:00
|
|
|
{
|
|
|
|
RECT *rc = (RECT*)lParam;
|
|
|
|
// get client area of the windows if it had the rect rc
|
|
|
|
// (subtracting the window borders)
|
|
|
|
RECT r = *rc;
|
2018-11-17 20:56:56 +00:00
|
|
|
subtract_window_borders(w32, w32->window, &r);
|
2017-12-17 00:19:36 +00:00
|
|
|
int c_w = rect_w(r), c_h = rect_h(r);
|
2014-01-06 14:35:27 +00:00
|
|
|
float aspect = w32->o_dwidth / (float) MPMAX(w32->o_dheight, 1);
|
|
|
|
int d_w = c_h * aspect - c_w;
|
|
|
|
int d_h = c_w / aspect - c_h;
|
|
|
|
int d_corners[4] = { d_w, d_h, -d_w, -d_h };
|
|
|
|
int corners[4] = { rc->left, rc->top, rc->right, rc->bottom };
|
|
|
|
int corner = get_resize_border(wParam);
|
|
|
|
if (corner >= 0)
|
|
|
|
corners[corner] -= d_corners[corner];
|
|
|
|
*rc = (RECT) { corners[0], corners[1], corners[2], corners[3] };
|
|
|
|
return TRUE;
|
2012-04-06 21:14:21 +00:00
|
|
|
}
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
2016-12-09 18:22:33 +00:00
|
|
|
case WM_DPICHANGED:
|
|
|
|
update_display_info(w32);
|
2020-05-07 19:13:51 +00:00
|
|
|
|
|
|
|
RECT *rc = (RECT*)lParam;
|
|
|
|
w32->windowrc = *rc;
|
|
|
|
subtract_window_borders(w32, w32->window, &w32->windowrc);
|
|
|
|
update_window_state(w32);
|
2016-12-09 18:22:33 +00:00
|
|
|
break;
|
2014-01-06 14:35:27 +00:00
|
|
|
case WM_CLOSE:
|
2017-12-11 18:48:45 +00:00
|
|
|
// Don't destroy the window yet to not lose wakeup events.
|
2014-07-26 18:27:03 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, MP_KEY_CLOSE_WIN);
|
2014-09-30 02:21:35 +00:00
|
|
|
return 0;
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
case WM_NCDESTROY: // Sometimes only WM_NCDESTROY is received in --wid mode
|
2014-09-30 02:21:35 +00:00
|
|
|
case WM_DESTROY:
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
if (w32->destroyed)
|
|
|
|
break;
|
|
|
|
// If terminate is not set, something else destroyed the window. This
|
|
|
|
// can also happen in --wid mode when the parent window is destroyed.
|
|
|
|
if (!w32->terminate)
|
|
|
|
mp_input_put_key(w32->input_ctx, MP_KEY_CLOSE_WIN);
|
|
|
|
RevokeDragDrop(w32->window);
|
2014-09-30 02:21:35 +00:00
|
|
|
w32->destroyed = true;
|
2014-10-17 20:22:10 +00:00
|
|
|
w32->window = NULL;
|
2014-09-30 02:21:35 +00:00
|
|
|
PostQuitMessage(0);
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
break;
|
2014-01-06 14:35:27 +00:00
|
|
|
case WM_SYSCOMMAND:
|
2016-12-16 15:53:09 +00:00
|
|
|
switch (wParam & 0xFFF0) {
|
2014-01-06 14:35:27 +00:00
|
|
|
case SC_SCREENSAVE:
|
|
|
|
case SC_MONITORPOWER:
|
|
|
|
if (w32->disable_screensaver) {
|
2014-07-26 18:27:03 +00:00
|
|
|
MP_VERBOSE(w32, "killing screensaver\n");
|
2012-04-06 21:14:21 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2012-04-07 12:47:06 +00:00
|
|
|
break;
|
2016-12-13 03:29:50 +00:00
|
|
|
case SC_RESTORE:
|
|
|
|
if (IsMaximized(w32->window) && w32->current_fs) {
|
2017-04-01 13:13:18 +00:00
|
|
|
w32->toggle_fs = true;
|
|
|
|
reinit_window_state(w32);
|
2019-12-11 18:40:33 +00:00
|
|
|
|
2016-12-13 03:29:50 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
2012-04-06 21:14:21 +00:00
|
|
|
}
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
2014-12-29 13:52:34 +00:00
|
|
|
case WM_NCHITTEST:
|
|
|
|
// Provide sizing handles for borderless windows
|
2015-06-20 11:33:23 +00:00
|
|
|
if (!w32->opts->border && !w32->current_fs) {
|
2014-12-29 13:52:34 +00:00
|
|
|
return borderless_nchittest(w32, GET_X_LPARAM(lParam),
|
|
|
|
GET_Y_LPARAM(lParam));
|
|
|
|
}
|
|
|
|
break;
|
2017-07-25 13:51:40 +00:00
|
|
|
case WM_APPCOMMAND:
|
|
|
|
if (handle_appcommand(w32, GET_APPCOMMAND_LPARAM(lParam)))
|
|
|
|
return TRUE;
|
|
|
|
break;
|
2014-01-14 11:27:42 +00:00
|
|
|
case WM_SYSKEYDOWN:
|
2014-11-08 10:43:14 +00:00
|
|
|
// Open the window menu on Alt+Space. Normally DefWindowProc opens the
|
|
|
|
// window menu in response to WM_SYSCHAR, but since mpv translates its
|
|
|
|
// own keyboard input, WM_SYSCHAR isn't generated, so the window menu
|
|
|
|
// must be opened manually.
|
|
|
|
if (wParam == VK_SPACE) {
|
|
|
|
SendMessage(w32->window, WM_SYSCOMMAND, SC_KEYMENU, ' ');
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-11-08 11:14:29 +00:00
|
|
|
handle_key_down(w32, wParam, HIWORD(lParam));
|
|
|
|
if (wParam == VK_F10)
|
2014-04-17 15:25:45 +00:00
|
|
|
return 0;
|
|
|
|
break;
|
2019-12-28 15:30:31 +00:00
|
|
|
case WM_KEYDOWN:
|
|
|
|
handle_key_down(w32, wParam, HIWORD(lParam));
|
|
|
|
break;
|
2014-04-17 15:25:45 +00:00
|
|
|
case WM_SYSKEYUP:
|
|
|
|
case WM_KEYUP:
|
2014-11-08 11:14:29 +00:00
|
|
|
handle_key_up(w32, wParam, HIWORD(lParam));
|
|
|
|
if (wParam == VK_F10)
|
2014-01-06 14:35:27 +00:00
|
|
|
return 0;
|
|
|
|
break;
|
|
|
|
case WM_CHAR:
|
2014-04-17 15:25:45 +00:00
|
|
|
case WM_SYSCHAR:
|
2014-07-26 18:27:03 +00:00
|
|
|
if (handle_char(w32, wParam))
|
2014-01-06 14:35:27 +00:00
|
|
|
return 0;
|
|
|
|
break;
|
2014-04-17 15:25:45 +00:00
|
|
|
case WM_KILLFOCUS:
|
2014-07-26 18:27:03 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, MP_INPUT_RELEASE_ALL);
|
2021-05-27 09:07:28 +00:00
|
|
|
w32->focused = false;
|
|
|
|
signal_events(w32, VO_EVENT_FOCUS);
|
|
|
|
return 0;
|
|
|
|
case WM_SETFOCUS:
|
|
|
|
w32->focused = true;
|
|
|
|
signal_events(w32, VO_EVENT_FOCUS);
|
|
|
|
return 0;
|
2014-01-06 14:35:27 +00:00
|
|
|
case WM_SETCURSOR:
|
2015-03-13 08:46:54 +00:00
|
|
|
// The cursor should only be hidden if the mouse is in the client area
|
|
|
|
// and if the window isn't in menu mode (HIWORD(lParam) is non-zero)
|
|
|
|
w32->can_set_cursor = LOWORD(lParam) == HTCLIENT && HIWORD(lParam);
|
|
|
|
if (w32->can_set_cursor && !w32->cursor_visible) {
|
2014-01-06 14:35:27 +00:00
|
|
|
SetCursor(NULL);
|
|
|
|
return TRUE;
|
2013-06-20 21:39:53 +00:00
|
|
|
}
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_MOUSELEAVE:
|
|
|
|
w32->tracking = FALSE;
|
2014-07-26 18:27:03 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, MP_KEY_MOUSE_LEAVE);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_MOUSEMOVE: {
|
2015-02-17 05:50:57 +00:00
|
|
|
if (!w32->tracking) {
|
2014-01-06 14:35:27 +00:00
|
|
|
w32->tracking = TrackMouseEvent(&w32->trackEvent);
|
2015-02-17 05:50:57 +00:00
|
|
|
mp_input_put_key(w32->input_ctx, MP_KEY_MOUSE_ENTER);
|
|
|
|
}
|
2014-01-06 14:35:27 +00:00
|
|
|
// Windows can send spurious mouse events, which would make the mpv
|
|
|
|
// core unhide the mouse cursor on completely unrelated events. See:
|
|
|
|
// https://blogs.msdn.com/b/oldnewthing/archive/2003/10/01/55108.aspx
|
|
|
|
int x = GET_X_LPARAM(lParam);
|
|
|
|
int y = GET_Y_LPARAM(lParam);
|
|
|
|
if (x != w32->mouse_x || y != w32->mouse_y) {
|
|
|
|
w32->mouse_x = x;
|
|
|
|
w32->mouse_y = y;
|
2014-07-27 19:53:29 +00:00
|
|
|
mp_input_set_mouse_pos(w32->input_ctx, x, y);
|
2014-01-06 14:35:27 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case WM_LBUTTONDOWN:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
if (handle_mouse_down(w32, MP_MBTN_LEFT, GET_X_LPARAM(lParam),
|
|
|
|
GET_Y_LPARAM(lParam)))
|
2017-04-02 02:59:35 +00:00
|
|
|
return 0;
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_LBUTTONUP:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
handle_mouse_up(w32, MP_MBTN_LEFT);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_MBUTTONDOWN:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
handle_mouse_down(w32, MP_MBTN_MID, GET_X_LPARAM(lParam),
|
|
|
|
GET_Y_LPARAM(lParam));
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_MBUTTONUP:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
handle_mouse_up(w32, MP_MBTN_MID);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_RBUTTONDOWN:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
handle_mouse_down(w32, MP_MBTN_RIGHT, GET_X_LPARAM(lParam),
|
2017-04-02 02:59:35 +00:00
|
|
|
GET_Y_LPARAM(lParam));
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_RBUTTONUP:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
handle_mouse_up(w32, MP_MBTN_RIGHT);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
2017-04-25 12:58:18 +00:00
|
|
|
case WM_MOUSEWHEEL:
|
|
|
|
handle_mouse_wheel(w32, false, GET_WHEEL_DELTA_WPARAM(wParam));
|
|
|
|
return 0;
|
|
|
|
case WM_MOUSEHWHEEL:
|
|
|
|
handle_mouse_wheel(w32, true, GET_WHEEL_DELTA_WPARAM(wParam));
|
|
|
|
// Some buggy mouse drivers (SetPoint) stop delivering WM_MOUSEHWHEEL
|
|
|
|
// events when the message loop doesn't return TRUE (even on Windows 7)
|
|
|
|
return TRUE;
|
2014-01-06 14:35:27 +00:00
|
|
|
case WM_XBUTTONDOWN:
|
2017-04-02 02:59:35 +00:00
|
|
|
handle_mouse_down(w32,
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
HIWORD(wParam) == 1 ? MP_MBTN_BACK : MP_MBTN_FORWARD,
|
|
|
|
GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam));
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
|
|
|
case WM_XBUTTONUP:
|
input: use mnemonic names for mouse buttons
mpv's mouse button numbering is based on X11 button numbering, which
allows for an arbitrary number of buttons and includes mouse wheel input
as buttons 3-6. This button numbering was used throughout the codebase
and exposed in input.conf, and it was difficult to remember which
physical button each number actually referred to and which referred to
the scroll wheel.
In practice, PC mice only have between two and five buttons and one or
two scroll wheel axes, which are more or less in the same location and
have more or less the same function. This allows us to use names to
refer to the buttons instead of numbers, which makes input.conf syntax a
lot easier to remember. It also makes the syntax robust to changes in
mpv's underlying numbering. The old MOUSE_BTNx names are still
understood as deprecated aliases of the named buttons.
This changes both the input.conf syntax and the MP_MOUSE_BTNx symbols in
the codebase, since I think both would benefit from using names over
numbers, especially since some platforms don't use X11 button numbering
and handle different mouse buttons in different windowing system events.
This also makes the names shorter, since otherwise they would be pretty
long, and it removes the high-numbered MOUSE_BTNx_DBL names, since they
weren't used.
Names are the same as used in Qt:
https://doc.qt.io/qt-5/qt.html#MouseButton-enum
2017-08-08 11:34:38 +00:00
|
|
|
handle_mouse_up(w32,
|
|
|
|
HIWORD(wParam) == 1 ? MP_MBTN_BACK : MP_MBTN_FORWARD);
|
2014-01-06 14:35:27 +00:00
|
|
|
break;
|
2015-03-10 18:25:30 +00:00
|
|
|
case WM_DISPLAYCHANGE:
|
2015-11-26 11:11:45 +00:00
|
|
|
force_update_display_info(w32);
|
2015-03-10 18:25:30 +00:00
|
|
|
break;
|
2013-06-20 21:39:53 +00:00
|
|
|
}
|
|
|
|
|
2015-11-15 22:03:48 +00:00
|
|
|
if (message == w32->tbtnCreatedMsg) {
|
|
|
|
w32->tbtnCreated = true;
|
2016-09-02 16:08:49 +00:00
|
|
|
update_playback_state(w32);
|
2015-11-15 22:03:48 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-04-06 21:42:02 +00:00
|
|
|
return DefWindowProcW(hWnd, message, wParam, lParam);
|
2003-09-19 14:33:51 +00:00
|
|
|
}
|
|
|
|
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
static pthread_once_t window_class_init_once = PTHREAD_ONCE_INIT;
|
|
|
|
static ATOM window_class;
|
|
|
|
static void register_window_class(void)
|
|
|
|
{
|
|
|
|
window_class = RegisterClassExW(&(WNDCLASSEXW) {
|
|
|
|
.cbSize = sizeof(WNDCLASSEXW),
|
|
|
|
.style = CS_HREDRAW | CS_VREDRAW,
|
|
|
|
.lpfnWndProc = WndProc,
|
|
|
|
.hInstance = HINST_THISCOMPONENT,
|
|
|
|
.hIcon = LoadIconW(HINST_THISCOMPONENT, L"IDI_ICON1"),
|
|
|
|
.hCursor = LoadCursor(NULL, IDC_ARROW),
|
|
|
|
.lpszClassName = L"mpv",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static ATOM get_window_class(void)
|
|
|
|
{
|
|
|
|
pthread_once(&window_class_init_once, register_window_class);
|
|
|
|
return window_class;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void resize_child_win(HWND parent)
|
|
|
|
{
|
|
|
|
// Check if an mpv window is a child of this window. This will not
|
|
|
|
// necessarily be the case because the hook functions will run for all
|
|
|
|
// windows on the parent window's thread.
|
|
|
|
ATOM cls = get_window_class();
|
|
|
|
HWND child = FindWindowExW(parent, NULL, (LPWSTR)MAKEINTATOM(cls), NULL);
|
|
|
|
if (!child)
|
|
|
|
return;
|
|
|
|
// Make sure the window was created by this instance
|
|
|
|
if (GetWindowLongPtrW(child, GWLP_HINSTANCE) != (LONG_PTR)HINST_THISCOMPONENT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Resize the mpv window to match its parent window's size
|
|
|
|
RECT rm, rp;
|
|
|
|
if (!GetClientRect(child, &rm))
|
|
|
|
return;
|
|
|
|
if (!GetClientRect(parent, &rp))
|
|
|
|
return;
|
|
|
|
if (EqualRect(&rm, &rp))
|
|
|
|
return;
|
|
|
|
SetWindowPos(child, NULL, 0, 0, rp.right, rp.bottom, SWP_ASYNCWINDOWPOS |
|
2017-07-03 11:29:17 +00:00
|
|
|
SWP_NOACTIVATE | SWP_NOZORDER | SWP_NOOWNERZORDER | SWP_NOSENDCHANGING);
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static LRESULT CALLBACK parent_win_hook(int nCode, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
if (nCode != HC_ACTION)
|
|
|
|
goto done;
|
|
|
|
CWPSTRUCT *cwp = (CWPSTRUCT*)lParam;
|
|
|
|
if (cwp->message != WM_WINDOWPOSCHANGED)
|
|
|
|
goto done;
|
|
|
|
resize_child_win(cwp->hwnd);
|
|
|
|
done:
|
|
|
|
return CallNextHookEx(NULL, nCode, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void CALLBACK parent_evt_hook(HWINEVENTHOOK hWinEventHook, DWORD event,
|
|
|
|
HWND hwnd, LONG idObject, LONG idChild, DWORD dwEventThread,
|
|
|
|
DWORD dwmsEventTime)
|
|
|
|
{
|
|
|
|
if (event != EVENT_OBJECT_LOCATIONCHANGE)
|
|
|
|
return;
|
|
|
|
if (!hwnd || idObject != OBJID_WINDOW || idChild != CHILDID_SELF)
|
|
|
|
return;
|
|
|
|
resize_child_win(hwnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void install_parent_hook(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
DWORD pid;
|
|
|
|
DWORD tid = GetWindowThreadProcessId(w32->parent, &pid);
|
|
|
|
|
|
|
|
// If the parent lives inside the current process, install a Windows hook
|
|
|
|
if (pid == GetCurrentProcessId()) {
|
|
|
|
w32->parent_win_hook = SetWindowsHookExW(WH_CALLWNDPROC,
|
|
|
|
parent_win_hook, NULL, tid);
|
|
|
|
} else {
|
|
|
|
// Otherwise, use a WinEvent hook. These don't seem to be as smooth as
|
|
|
|
// Windows hooks, but they can be delivered across process boundaries.
|
|
|
|
w32->parent_evt_hook = SetWinEventHook(
|
|
|
|
EVENT_OBJECT_LOCATIONCHANGE, EVENT_OBJECT_LOCATIONCHANGE,
|
|
|
|
NULL, parent_evt_hook, pid, tid, WINEVENT_OUTOFCONTEXT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void remove_parent_hook(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
if (w32->parent_win_hook)
|
|
|
|
UnhookWindowsHookEx(w32->parent_win_hook);
|
|
|
|
if (w32->parent_evt_hook)
|
|
|
|
UnhookWinEvent(w32->parent_evt_hook);
|
|
|
|
}
|
|
|
|
|
2014-07-26 18:27:57 +00:00
|
|
|
// Dispatch incoming window events and handle them.
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
// This returns only when the thread is asked to terminate.
|
|
|
|
static void run_message_loop(struct vo_w32_state *w32)
|
2012-04-14 11:39:53 +00:00
|
|
|
{
|
2003-09-19 14:33:51 +00:00
|
|
|
MSG msg;
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
while (GetMessageW(&msg, 0, 0, 0) > 0)
|
2012-04-06 21:42:02 +00:00
|
|
|
DispatchMessageW(&msg);
|
2014-01-06 11:26:42 +00:00
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
// Even if the message loop somehow exits, we still have to respond to
|
|
|
|
// external requests until termination is requested.
|
|
|
|
while (!w32->terminate)
|
|
|
|
mp_dispatch_queue_process(w32->dispatch, 1000);
|
2003-09-19 14:33:51 +00:00
|
|
|
}
|
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
static void gui_thread_reconfig(void *ptr)
|
2012-04-14 11:39:53 +00:00
|
|
|
{
|
2015-10-02 16:13:24 +00:00
|
|
|
struct vo_w32_state *w32 = ptr;
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
struct vo *vo = w32->vo;
|
2010-11-10 22:10:30 +00:00
|
|
|
|
2017-12-22 16:19:28 +00:00
|
|
|
RECT r = get_working_area(w32);
|
2022-02-01 06:22:35 +00:00
|
|
|
// for normal window which is auto-positioned (centered), center the window
|
|
|
|
// rather than the content (by subtracting the borders from the work area)
|
|
|
|
if (!w32->current_fs && !IsMaximized(w32->window) && w32->opts->border &&
|
|
|
|
!w32->opts->geometry.xy_valid /* specific position not requested */)
|
|
|
|
{
|
win32: initial position: center with borders
Previously, the initial positioning and fit ignored the borders, and
centered the content (the video itself) at the working area.
Now, the initial positioning centers the window, by subtracting the
borders (if needed) from the target area for the initial fit/position.
While this does mean that the initial maximum content area is now
smaller than before, ultimately this has no impact on the window size,
because fit_on_screen is called later and, if needed, further shrinks
the window to fit the borders too - but without centering the window.
So the net impact of this commit is only the initial positioning (same
size as before), which now centers the window instead of the content.
Note that on Windows 10 the borders include invisible areas at the
sides and bottom of the window (for mouse edge-drag), so visibly the
window is nearer to the top than to the bottom, but these are the
metrics we have (fit_on_screen uses the same border size values).
On Windows 7 it looks perfectly centered.
2021-08-12 21:49:09 +00:00
|
|
|
subtract_window_borders(w32, w32->window, &r);
|
2022-02-01 06:22:35 +00:00
|
|
|
}
|
2017-12-22 16:19:28 +00:00
|
|
|
struct mp_rect screen = { r.left, r.top, r.right, r.bottom };
|
2014-05-06 21:01:19 +00:00
|
|
|
struct vo_win_geometry geo;
|
2017-12-22 16:19:28 +00:00
|
|
|
|
2021-08-12 18:14:37 +00:00
|
|
|
RECT monrc = get_monitor_info(w32).rcMonitor;
|
|
|
|
struct mp_rect mon = { monrc.left, monrc.top, monrc.right, monrc.bottom };
|
|
|
|
|
2021-08-09 19:04:29 +00:00
|
|
|
if (w32->dpi_scale == 0)
|
|
|
|
force_update_display_info(w32);
|
|
|
|
|
2021-08-12 18:14:37 +00:00
|
|
|
vo_calc_window_geometry3(vo, &screen, &mon, w32->dpi_scale, &geo);
|
2014-05-06 21:01:19 +00:00
|
|
|
vo_apply_window_geometry(vo, &geo);
|
|
|
|
|
2017-12-17 00:19:36 +00:00
|
|
|
bool reset_size = w32->o_dwidth != vo->dwidth ||
|
|
|
|
w32->o_dheight != vo->dheight;
|
win32: fix window creation and size handling
This commit fixes various issues with the way the window position and
size is setup. Most importantly, it fixes some bugs with restoring from
fullscreen state.
Rename create_rendering_context() to reinit_window_state(). This function
doesn't create anything, it just sets the window bounds and styles.
Do not use vo_dx/dy for the window position, as video_out.c overwrites it
with each vo_config() call. Use private variables window_x/y instead.
A big cause for issues was that reinit_window_state() accidentally cleared
the WS_VISIBLE style. I suspect that the API call to temporarily hide the
window was a hack to deal with this. Another bug was that the window style
was changed without calling SetWindowPos with SWP_FRAMECHANGED (as the
MSDN documentation says).
Properly initialize window position and size on vo_config following the
same rules as the x11 backend:
- Never change the window position. The window position should be kept, as
the user might move the window, and resetting the window position e.g.
during ordered chapter playback is not desired.
- Never change the window size, unless the size of the video changes.
These rules don't apply to fullscreen. When switching from fullscreen to
windowed mode, the backend should restore the previous windowed size and
position. When the VO was reconfigured during playback (vo_config() etc.),
the saved window position and size should be changed according to the
rules above, even if the window was in fullscreen mode during
reconfiguring.
Note that these rules might be perceived as awkward by some users: if you
play multiple files with different resolutions, the window won't be
centered when playing the files after the first. This is not a bug.
2011-10-22 00:11:14 +00:00
|
|
|
|
2014-05-06 21:01:19 +00:00
|
|
|
w32->o_dwidth = vo->dwidth;
|
|
|
|
w32->o_dheight = vo->dheight;
|
2006-04-12 14:11:26 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
if (!w32->parent && !w32->window_bounds_initialized) {
|
|
|
|
SetRect(&w32->windowrc, geo.win.x0, geo.win.y0,
|
|
|
|
geo.win.x0 + vo->dwidth, geo.win.y0 + vo->dheight);
|
|
|
|
w32->prev_windowrc = w32->windowrc;
|
|
|
|
w32->window_bounds_initialized = true;
|
|
|
|
w32->fit_on_screen = true;
|
|
|
|
goto finish;
|
|
|
|
}
|
2014-01-06 11:26:42 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// The rect which size is going to be modified.
|
|
|
|
RECT *rc = &w32->windowrc;
|
|
|
|
|
|
|
|
// The desired size always matches the window size in wid mode.
|
|
|
|
if (!reset_size || w32->parent) {
|
2013-03-19 22:36:43 +00:00
|
|
|
GetClientRect(w32->window, &r);
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// Restore vo_dwidth and vo_dheight, which were reset in vo_config()
|
2013-03-19 22:36:43 +00:00
|
|
|
vo->dwidth = r.right;
|
|
|
|
vo->dheight = r.bottom;
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
} else {
|
|
|
|
if (w32->current_fs)
|
|
|
|
rc = &w32->prev_windowrc;
|
|
|
|
w32->fit_on_screen = true;
|
2008-12-03 10:38:50 +00:00
|
|
|
}
|
2006-04-12 14:11:26 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
// Save new window size and position.
|
|
|
|
const int x = rc->left + rect_w(*rc) / 2 - vo->dwidth / 2;
|
|
|
|
const int y = rc->top + rect_h(*rc) / 2 - vo->dheight / 2;
|
|
|
|
SetRect(rc, x, y, x + vo->dwidth, y + vo->dheight);
|
2015-05-31 15:54:14 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
finish:
|
2015-10-02 16:13:24 +00:00
|
|
|
reinit_window_state(w32);
|
2006-04-12 14:11:26 +00:00
|
|
|
}
|
|
|
|
|
2015-10-02 16:09:13 +00:00
|
|
|
// Resize the window. On the first call, it's also made visible.
|
2015-10-31 11:52:02 +00:00
|
|
|
void vo_w32_config(struct vo *vo)
|
2012-04-14 11:39:53 +00:00
|
|
|
{
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
struct vo_w32_state *w32 = vo->w32;
|
2015-10-02 16:13:24 +00:00
|
|
|
mp_dispatch_run(w32->dispatch, gui_thread_reconfig, w32);
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
}
|
2003-09-19 14:33:51 +00:00
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
static void w32_api_load(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
HMODULE shcore_dll = LoadLibraryW(L"shcore.dll");
|
2017-11-15 01:50:00 +00:00
|
|
|
// Available since Win8.1
|
2016-12-09 18:22:33 +00:00
|
|
|
w32->api.pGetDpiForMonitor = !shcore_dll ? NULL :
|
|
|
|
(void *)GetProcAddress(shcore_dll, "GetDpiForMonitor");
|
2017-11-15 01:50:00 +00:00
|
|
|
|
2018-11-17 20:56:56 +00:00
|
|
|
HMODULE user32_dll = LoadLibraryW(L"user32.dll");
|
|
|
|
// Available since Win10
|
|
|
|
w32->api.pAdjustWindowRectExForDpi = !user32_dll ? NULL :
|
|
|
|
(void *)GetProcAddress(user32_dll, "AdjustWindowRectExForDpi");
|
|
|
|
|
2017-11-15 01:50:00 +00:00
|
|
|
// imm32.dll must be loaded dynamically
|
|
|
|
// to account for machines without East Asian language support
|
|
|
|
HMODULE imm32_dll = LoadLibraryW(L"imm32.dll");
|
|
|
|
w32->api.pImmDisableIME = !imm32_dll ? NULL :
|
|
|
|
(void *)GetProcAddress(imm32_dll, "ImmDisableIME");
|
2016-12-09 18:22:33 +00:00
|
|
|
}
|
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
static void *gui_thread(void *ptr)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = ptr;
|
|
|
|
bool ole_ok = false;
|
|
|
|
int res = 0;
|
2009-07-06 23:26:13 +00:00
|
|
|
|
2014-10-19 21:32:34 +00:00
|
|
|
mpthread_set_name("win32 window");
|
|
|
|
|
2016-12-09 18:22:33 +00:00
|
|
|
w32_api_load(w32);
|
2017-11-15 01:50:00 +00:00
|
|
|
|
|
|
|
// Disables the IME for windows on this thread
|
|
|
|
if (w32->api.pImmDisableIME)
|
|
|
|
w32->api.pImmDisableIME(0);
|
2015-10-22 08:36:26 +00:00
|
|
|
|
2014-12-09 20:28:35 +00:00
|
|
|
if (w32->opts->WinID >= 0)
|
|
|
|
w32->parent = (HWND)(intptr_t)(w32->opts->WinID);
|
|
|
|
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
ATOM cls = get_window_class();
|
2014-12-09 20:28:35 +00:00
|
|
|
if (w32->parent) {
|
2008-02-09 14:14:35 +00:00
|
|
|
RECT r;
|
2014-12-09 20:28:35 +00:00
|
|
|
GetClientRect(w32->parent, &r);
|
2017-05-15 14:27:26 +00:00
|
|
|
CreateWindowExW(WS_EX_NOPARENTNOTIFY, (LPWSTR)MAKEINTATOM(cls), L"mpv",
|
|
|
|
WS_CHILD | WS_VISIBLE, 0, 0, r.right, r.bottom,
|
|
|
|
w32->parent, 0, HINST_THISCOMPONENT, w32);
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
|
|
|
|
// Install a hook to get notifications when the parent changes size
|
|
|
|
if (w32->window)
|
|
|
|
install_parent_hook(w32);
|
2012-04-14 11:39:53 +00:00
|
|
|
} else {
|
2017-05-15 14:27:26 +00:00
|
|
|
CreateWindowExW(0, (LPWSTR)MAKEINTATOM(cls), L"mpv",
|
|
|
|
update_style(w32, 0), CW_USEDEFAULT, SW_HIDE, 100, 100,
|
|
|
|
0, 0, HINST_THISCOMPONENT, w32);
|
2012-04-14 11:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!w32->window) {
|
2014-07-26 18:27:03 +00:00
|
|
|
MP_ERR(w32, "unable to create window!\n");
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
goto done;
|
2003-09-19 14:33:51 +00:00
|
|
|
}
|
|
|
|
|
2014-12-08 06:06:14 +00:00
|
|
|
if (SUCCEEDED(OleInitialize(NULL))) {
|
|
|
|
ole_ok = true;
|
|
|
|
|
2017-03-25 12:30:54 +00:00
|
|
|
IDropTarget *dt = mp_w32_droptarget_create(w32->log, w32->input_ctx);
|
|
|
|
RegisterDragDrop(w32->window, dt);
|
2014-12-08 06:06:14 +00:00
|
|
|
|
|
|
|
// ITaskbarList2 has the MarkFullscreenWindow method, which is used to
|
|
|
|
// make sure the taskbar is hidden when mpv goes fullscreen
|
|
|
|
if (SUCCEEDED(CoCreateInstance(&CLSID_TaskbarList, NULL,
|
2014-12-16 10:19:33 +00:00
|
|
|
CLSCTX_INPROC_SERVER, &IID_ITaskbarList2,
|
2014-12-08 06:06:14 +00:00
|
|
|
(void**)&w32->taskbar_list)))
|
|
|
|
{
|
|
|
|
if (FAILED(ITaskbarList2_HrInit(w32->taskbar_list))) {
|
|
|
|
ITaskbarList2_Release(w32->taskbar_list);
|
|
|
|
w32->taskbar_list = NULL;
|
|
|
|
}
|
|
|
|
}
|
2015-11-15 22:03:48 +00:00
|
|
|
|
|
|
|
// ITaskbarList3 has methods for status indication on taskbar buttons,
|
|
|
|
// however that interface is only available on Win7/2008 R2 or newer
|
|
|
|
if (SUCCEEDED(CoCreateInstance(&CLSID_TaskbarList, NULL,
|
|
|
|
CLSCTX_INPROC_SERVER, &IID_ITaskbarList3,
|
|
|
|
(void**)&w32->taskbar_list3)))
|
|
|
|
{
|
|
|
|
if (FAILED(ITaskbarList3_HrInit(w32->taskbar_list3))) {
|
|
|
|
ITaskbarList3_Release(w32->taskbar_list3);
|
|
|
|
w32->taskbar_list3 = NULL;
|
|
|
|
} else {
|
|
|
|
w32->tbtnCreatedMsg = RegisterWindowMessage(L"TaskbarButtonCreated");
|
|
|
|
}
|
|
|
|
}
|
2014-12-08 06:06:14 +00:00
|
|
|
} else {
|
|
|
|
MP_ERR(w32, "Failed to initialize OLE/COM\n");
|
2014-01-06 04:21:54 +00:00
|
|
|
}
|
2014-01-06 11:26:42 +00:00
|
|
|
|
2013-07-25 17:47:49 +00:00
|
|
|
w32->tracking = FALSE;
|
|
|
|
w32->trackEvent = (TRACKMOUSEEVENT){
|
|
|
|
.cbSize = sizeof(TRACKMOUSEEVENT),
|
|
|
|
.dwFlags = TME_LEAVE,
|
|
|
|
.hwndTrack = w32->window,
|
|
|
|
};
|
2013-07-25 17:19:12 +00:00
|
|
|
|
2014-12-09 20:28:35 +00:00
|
|
|
if (w32->parent)
|
2012-04-14 11:39:53 +00:00
|
|
|
EnableWindow(w32->window, 0);
|
2014-01-06 11:26:42 +00:00
|
|
|
|
2013-07-14 12:57:39 +00:00
|
|
|
w32->cursor_visible = true;
|
2017-12-04 23:21:34 +00:00
|
|
|
w32->moving = false;
|
|
|
|
w32->snapped = false;
|
|
|
|
w32->snap_dx = w32->snap_dy = 0;
|
2012-04-14 11:39:53 +00:00
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
mp_dispatch_set_wakeup_fn(w32->dispatch, wakeup_gui_thread, w32);
|
|
|
|
|
|
|
|
res = 1;
|
|
|
|
done:
|
|
|
|
|
|
|
|
mp_rendezvous(w32, res); // init barrier
|
|
|
|
|
|
|
|
// This blocks until the GUI thread is to be exited.
|
|
|
|
if (res)
|
|
|
|
run_message_loop(w32);
|
|
|
|
|
|
|
|
MP_VERBOSE(w32, "uninit\n");
|
|
|
|
|
w32_common: use hooks to detect parent window resize
Because VOCTRL_CHECK_EVENTS is processed asynchronously (as of 088a007,)
the GUI thread no longer gets regular wakeups, so the old check that
made sure the video window matched the parent window's size in --wid
embedding mode did not run very often. This made --wid embedding not
very usable.
Instead of polling for window size changes, use Windows hooks to react
to them when they happen. When the parent window is owned by the same
process as the video window, use a WH_CALLWNDPROC hook. When the parent
window is not owned by the same process, WinEvents must be used, which
are not as smooth, but still work for this purpose.
Since neither SetWindowsHookEx nor SetWinEventHook take a context
parameter to send data to the hook function, the hook functions must
find the child window by its class instead, so there are a few changes
to ensure this is fast and the class is unique.
This also fixes up the logic to handle window destruction. When a parent
window is destroyed, its children are also destroyed, so this gives us a
way to react to parent window destruction without polling.
2016-08-25 13:07:42 +00:00
|
|
|
remove_parent_hook(w32);
|
|
|
|
if (w32->window && !w32->destroyed)
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
DestroyWindow(w32->window);
|
2014-12-08 06:06:14 +00:00
|
|
|
if (w32->taskbar_list)
|
|
|
|
ITaskbarList2_Release(w32->taskbar_list);
|
2015-11-15 22:03:48 +00:00
|
|
|
if (w32->taskbar_list3)
|
|
|
|
ITaskbarList3_Release(w32->taskbar_list3);
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
if (ole_ok)
|
|
|
|
OleUninitialize();
|
|
|
|
SetThreadExecutionState(ES_CONTINUOUS);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns: 1 = Success, 0 = Failure
|
|
|
|
int vo_w32_init(struct vo *vo)
|
|
|
|
{
|
|
|
|
assert(!vo->w32);
|
|
|
|
|
|
|
|
struct vo_w32_state *w32 = talloc_ptrtype(vo, w32);
|
|
|
|
*w32 = (struct vo_w32_state){
|
|
|
|
.log = mp_log_new(w32, vo->log, "win32"),
|
|
|
|
.vo = vo,
|
2019-12-11 18:40:33 +00:00
|
|
|
.opts_cache = m_config_cache_alloc(w32, vo->global, &vo_sub_opts),
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
.input_ctx = vo->input_ctx,
|
|
|
|
.dispatch = mp_dispatch_create(w32),
|
|
|
|
};
|
2019-12-11 18:40:33 +00:00
|
|
|
w32->opts = w32->opts_cache->opts;
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
vo->w32 = w32;
|
|
|
|
|
|
|
|
if (pthread_create(&w32->thread, NULL, gui_thread, w32))
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
if (!mp_rendezvous(w32, 0)) { // init barrier
|
|
|
|
pthread_join(w32->thread, NULL);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2015-11-08 18:31:09 +00:00
|
|
|
// While the UI runs in its own thread, the thread in which this function
|
|
|
|
// runs in will be the renderer thread. Apply magic MMCSS cargo-cult,
|
|
|
|
// which might stop Windows from throttling clock rate and so on.
|
2015-12-06 18:20:23 +00:00
|
|
|
if (vo->opts->mmcss_profile[0]) {
|
|
|
|
wchar_t *profile = mp_from_utf8(NULL, vo->opts->mmcss_profile);
|
|
|
|
w32->avrt_handle = AvSetMmThreadCharacteristicsW(profile, &(DWORD){0});
|
|
|
|
talloc_free(profile);
|
|
|
|
}
|
2015-11-08 18:31:09 +00:00
|
|
|
|
2003-09-19 14:33:51 +00:00
|
|
|
return 1;
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
fail:
|
|
|
|
talloc_free(w32);
|
|
|
|
vo->w32 = NULL;
|
|
|
|
return 0;
|
2003-09-19 14:33:51 +00:00
|
|
|
}
|
|
|
|
|
2016-09-29 13:21:23 +00:00
|
|
|
struct disp_names_data {
|
|
|
|
HMONITOR assoc;
|
|
|
|
int count;
|
|
|
|
char **names;
|
|
|
|
};
|
|
|
|
|
|
|
|
static BOOL CALLBACK disp_names_proc(HMONITOR mon, HDC dc, LPRECT r, LPARAM p)
|
|
|
|
{
|
|
|
|
struct disp_names_data *data = (struct disp_names_data*)p;
|
|
|
|
|
|
|
|
// get_disp_names() adds data->assoc to the list, so skip it here
|
|
|
|
if (mon == data->assoc)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
MONITORINFOEXW mi = { .cbSize = sizeof mi };
|
|
|
|
if (GetMonitorInfoW(mon, (MONITORINFO*)&mi)) {
|
|
|
|
MP_TARRAY_APPEND(NULL, data->names, data->count,
|
|
|
|
mp_to_utf8(NULL, mi.szDevice));
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char **get_disp_names(struct vo_w32_state *w32)
|
|
|
|
{
|
|
|
|
// Get the client area of the window in screen space
|
|
|
|
RECT rect = { 0 };
|
|
|
|
GetClientRect(w32->window, &rect);
|
|
|
|
MapWindowPoints(w32->window, NULL, (POINT*)&rect, 2);
|
|
|
|
|
|
|
|
struct disp_names_data data = { .assoc = w32->monitor };
|
|
|
|
|
|
|
|
// Make sure the monitor that Windows considers to be associated with the
|
|
|
|
// window is first in the list
|
|
|
|
MONITORINFOEXW mi = { .cbSize = sizeof mi };
|
|
|
|
if (GetMonitorInfoW(data.assoc, (MONITORINFO*)&mi)) {
|
|
|
|
MP_TARRAY_APPEND(NULL, data.names, data.count,
|
|
|
|
mp_to_utf8(NULL, mi.szDevice));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the names of the other monitors that intersect the client rect
|
|
|
|
EnumDisplayMonitors(NULL, &rect, disp_names_proc, (LPARAM)&data);
|
|
|
|
MP_TARRAY_APPEND(NULL, data.names, data.count, NULL);
|
|
|
|
return data.names;
|
|
|
|
}
|
|
|
|
|
2015-01-08 17:32:23 +00:00
|
|
|
static int gui_thread_control(struct vo_w32_state *w32, int request, void *arg)
|
2013-05-15 16:17:18 +00:00
|
|
|
{
|
|
|
|
switch (request) {
|
2019-12-11 18:40:33 +00:00
|
|
|
case VOCTRL_VO_OPTS_CHANGED: {
|
|
|
|
void *changed_option;
|
|
|
|
|
|
|
|
while (m_config_cache_get_next_changed(w32->opts_cache,
|
|
|
|
&changed_option))
|
|
|
|
{
|
|
|
|
struct mp_vo_opts *vo_opts = w32->opts_cache->opts;
|
|
|
|
|
|
|
|
if (changed_option == &vo_opts->fullscreen) {
|
|
|
|
reinit_window_state(w32);
|
2019-12-17 20:46:57 +00:00
|
|
|
} else if (changed_option == &vo_opts->ontop) {
|
|
|
|
update_window_state(w32);
|
2019-12-17 20:49:58 +00:00
|
|
|
} else if (changed_option == &vo_opts->border) {
|
|
|
|
update_window_style(w32);
|
|
|
|
update_window_state(w32);
|
w32_common: support minimized and maximized properties
Add support for setting window-minimized and window-maximized in
Windows. The minimized and maximized state can be set independently.
When the window is minimized, the value of window-maximized will
determine whether the window is restored to the maximized state or not.
Changing state is done with ShowWindow(), which has commands that change
the window state and activate it (eg. SW_RESTORE) and commands that
change the window state without activating it (eg. SW_SHOWNOACTIVATE.)
It would be nice if we could use commands that don't activate the
window, so scripts could change the window state in the backrgound
without bringing it to the foreground, but there are some problems with
that. There is no command to maximize a window without activating it, so
SW_MAXIMIZE is used instead. Also, restoring a window from minimize
without activating it seems buggy. On my Windows 10 1909 PC, it always
moves the window to the back of the z-order. SW_RESTORE is used instead
of SW_SHOWNOACTIVATE because of this.
This also changes the way the window is initially shown. Previously, the
window was made visible as a consequence of the SWP_SHOWWINDOW flag in
the first call to SetWindowPos. In order to set the initial minimized or
maximized state of the window, the window is shown with the ShowWindow
function instead, where the ShowWindow command is determined by whether
the window should be initially maximized or minimized.
Even when showing the window normally, we should still call ShowWindow
with the SW_SHOW command instead of using SetWindowPos, since the first
call a process makes to ShowWindow(SW_SHOW) has special behaviour
where it uses the show command in the process' STARTUPINFO instead of
the command passed to the function, which should fix #5724.
Note: While changes to window-minimized while in fullscreen mode should
work as expected, changing window-maximized while in fullscreen does not
work and won't result in the window changing state, even after leaving
fullscreen. For this to work correctly, the fullscreen logic needs to be
changed to apply the new maximized state on leaving fullscreen.
Fixes: #5724
Fixes: #7351
2019-12-27 15:12:44 +00:00
|
|
|
} else if (changed_option == &vo_opts->window_minimized) {
|
|
|
|
update_minimized_state(w32);
|
|
|
|
} else if (changed_option == &vo_opts->window_maximized) {
|
|
|
|
update_maximized_state(w32);
|
2019-12-11 18:40:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-06 14:35:27 +00:00
|
|
|
return VO_TRUE;
|
2019-12-11 18:40:33 +00:00
|
|
|
}
|
2021-08-09 17:40:53 +00:00
|
|
|
case VOCTRL_GET_HIDPI_SCALE: {
|
|
|
|
*(double *)arg = w32->dpi_scale;
|
|
|
|
return VO_TRUE;
|
|
|
|
}
|
2014-09-04 20:53:50 +00:00
|
|
|
case VOCTRL_GET_UNFS_WINDOW_SIZE: {
|
2014-01-06 14:35:27 +00:00
|
|
|
int *s = arg;
|
|
|
|
|
|
|
|
if (!w32->window_bounds_initialized)
|
|
|
|
return VO_FALSE;
|
|
|
|
|
2017-12-17 00:19:36 +00:00
|
|
|
RECT *rc = w32->current_fs ? &w32->prev_windowrc : &w32->windowrc;
|
2021-08-09 17:12:25 +00:00
|
|
|
s[0] = rect_w(*rc) / w32->dpi_scale;
|
|
|
|
s[1] = rect_h(*rc) / w32->dpi_scale;
|
2014-01-06 14:35:27 +00:00
|
|
|
return VO_TRUE;
|
|
|
|
}
|
2014-09-04 20:53:50 +00:00
|
|
|
case VOCTRL_SET_UNFS_WINDOW_SIZE: {
|
2014-01-06 14:35:27 +00:00
|
|
|
int *s = arg;
|
|
|
|
|
|
|
|
if (!w32->window_bounds_initialized)
|
|
|
|
return VO_FALSE;
|
2017-12-17 00:19:36 +00:00
|
|
|
|
2021-08-09 17:12:25 +00:00
|
|
|
s[0] *= w32->dpi_scale;
|
|
|
|
s[1] *= w32->dpi_scale;
|
|
|
|
|
2017-12-17 00:19:36 +00:00
|
|
|
RECT *rc = w32->current_fs ? &w32->prev_windowrc : &w32->windowrc;
|
|
|
|
const int x = rc->left + rect_w(*rc) / 2 - s[0] / 2;
|
|
|
|
const int y = rc->top + rect_h(*rc) / 2 - s[1] / 2;
|
|
|
|
SetRect(rc, x, y, x + s[0], y + s[1]);
|
2013-07-14 12:57:39 +00:00
|
|
|
|
w32_common: refactor and improve window state handling
Refactored and split the `reinit_window_state` code into four
separate functions:
- `update_window_style` used to update window styles without
modifying the window rect.
- `fit_window_on_screen` used to adjust the window size when it is
larger than the screen size. Added a helper function `fit_rect` to
fit one rect on another without using any data from w32 struct.
- `update_fullscreen_state` used to calculate the new fullscreen
state and adjust the window rect accordingly.
- `update_window_state` used to display the window on screen with
new size, position and ontop state.
This commit fixes three issues:
- fixed #4753 by skipping `fit_window_on_screen` for a maximized
window, since maximized window should already fit on the screen.
It should be noted that this bug was only reproducible with
`--fit-border` option which is enabled by default. The cause of the
bug is that after calling the `add_window_borders` for a maximized
window, the rect in result is slightly larger than the screen rect,
which is okay, `SetWindowPos` will interpret it as a maximized state
later, so no auto-fitting to screen size is needed here.
- fixed #5215 by skipping `fit_window_on_screen` when leaving fullscreen.
On a multi-monitor system if the mpv window was stretched to cover
multiple monitors, its size was reset after switching back from
fullscreen to fit the size of the active monitor. Also, when changing
`--ontop` and `--border` options, now only the
`update_window_style` and `update_window_state` functions are used,
so `fit_window_on_screen` is not used for them too.
- fixed #2451 by moving the `ITaskbarList2_MarkFullscreenWindow`
below the `SetWindowPos`. If the taskbar is notified about fullscreen
state before the window is shown on screen, the taskbar button could
be missing until Alt-TAB is pressed, usually it was reproducible on
Windows 8.
Other changes:
- In `update_fullscreen_state` the `reset window bounds` debug
message now reports client area size and position, instead of window area
size and position. This is done for consistency with debug messages
in handling fullscreen state above in this function, since they also print
window bounds of the client area.
- Refactored `gui_thread_reconfig`. Added a new window flag `fit_on_screen`
to fit the window on screen even when leaving fullscreen. This is needed
for the case when the new video opened while the window is still in the
fullscreen state.
- Moved parent and fullscreen state checks out from the WM_MOVING to
`snap_to_screen_edges` function for consistency with other functions.
There's no point in keeping these checks out of the function body.
2017-12-17 00:20:53 +00:00
|
|
|
w32->fit_on_screen = true;
|
2014-07-26 18:27:03 +00:00
|
|
|
reinit_window_state(w32);
|
2014-01-06 14:35:27 +00:00
|
|
|
return VO_TRUE;
|
|
|
|
}
|
|
|
|
case VOCTRL_SET_CURSOR_VISIBILITY:
|
|
|
|
w32->cursor_visible = *(bool *)arg;
|
|
|
|
|
2015-03-13 08:46:54 +00:00
|
|
|
if (w32->can_set_cursor && w32->tracking) {
|
2014-01-06 14:35:27 +00:00
|
|
|
if (w32->cursor_visible)
|
|
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
|
|
else
|
|
|
|
SetCursor(NULL);
|
2013-05-16 22:10:46 +00:00
|
|
|
}
|
2014-01-06 14:35:27 +00:00
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_KILL_SCREENSAVER:
|
|
|
|
w32->disable_screensaver = true;
|
2015-04-20 10:24:19 +00:00
|
|
|
SetThreadExecutionState(ES_CONTINUOUS | ES_SYSTEM_REQUIRED |
|
|
|
|
ES_DISPLAY_REQUIRED);
|
2014-01-06 14:35:27 +00:00
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_RESTORE_SCREENSAVER:
|
|
|
|
w32->disable_screensaver = false;
|
|
|
|
SetThreadExecutionState(ES_CONTINUOUS);
|
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_UPDATE_WINDOW_TITLE: {
|
2015-08-01 18:51:52 +00:00
|
|
|
wchar_t *title = mp_from_utf8(NULL, (char *)arg);
|
2014-01-06 14:35:27 +00:00
|
|
|
SetWindowTextW(w32->window, title);
|
|
|
|
talloc_free(title);
|
|
|
|
return VO_TRUE;
|
|
|
|
}
|
2015-11-15 22:03:48 +00:00
|
|
|
case VOCTRL_UPDATE_PLAYBACK_STATE: {
|
2016-09-02 16:08:49 +00:00
|
|
|
w32->current_pstate = *(struct voctrl_playback_state *)arg;
|
2015-11-22 10:40:20 +00:00
|
|
|
|
2016-09-02 16:08:49 +00:00
|
|
|
update_playback_state(w32);
|
2015-11-15 22:03:48 +00:00
|
|
|
return VO_TRUE;
|
|
|
|
}
|
2015-03-10 18:25:30 +00:00
|
|
|
case VOCTRL_GET_DISPLAY_FPS:
|
2015-11-26 11:11:45 +00:00
|
|
|
update_display_info(w32);
|
2015-03-10 18:25:30 +00:00
|
|
|
*(double*) arg = w32->display_fps;
|
|
|
|
return VO_TRUE;
|
2021-04-12 17:51:41 +00:00
|
|
|
case VOCTRL_GET_DISPLAY_RES: ;
|
|
|
|
RECT r = get_screen_area(w32);
|
|
|
|
((int *)arg)[0] = r.right;
|
|
|
|
((int *)arg)[1] = r.bottom;
|
|
|
|
return VO_TRUE;
|
2016-09-29 13:21:23 +00:00
|
|
|
case VOCTRL_GET_DISPLAY_NAMES:
|
|
|
|
*(char ***)arg = get_disp_names(w32);
|
|
|
|
return VO_TRUE;
|
2015-11-26 11:11:45 +00:00
|
|
|
case VOCTRL_GET_ICC_PROFILE:
|
|
|
|
update_display_info(w32);
|
|
|
|
if (w32->color_profile) {
|
|
|
|
bstr *p = arg;
|
|
|
|
*p = stream_read_file(w32->color_profile, NULL,
|
|
|
|
w32->vo->global, 100000000); // 100 MB
|
|
|
|
return p->len ? VO_TRUE : VO_FALSE;
|
|
|
|
}
|
|
|
|
return VO_FALSE;
|
2021-05-27 09:07:28 +00:00
|
|
|
case VOCTRL_GET_FOCUSED:
|
|
|
|
*(bool *)arg = w32->focused;
|
|
|
|
return VO_TRUE;
|
2013-05-15 16:17:18 +00:00
|
|
|
}
|
|
|
|
return VO_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
static void do_control(void *ptr)
|
|
|
|
{
|
|
|
|
void **p = ptr;
|
|
|
|
struct vo_w32_state *w32 = p[0];
|
|
|
|
int *events = p[1];
|
|
|
|
int request = *(int *)p[2];
|
|
|
|
void *arg = p[3];
|
|
|
|
int *ret = p[4];
|
2015-01-08 17:32:23 +00:00
|
|
|
*ret = gui_thread_control(w32, request, arg);
|
2016-08-05 14:01:26 +00:00
|
|
|
*events |= atomic_fetch_and(&w32->event_flags, 0);
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
// Safe access, since caller (owner of vo) is blocked.
|
|
|
|
if (*events & VO_EVENT_RESIZE) {
|
2017-12-17 00:19:36 +00:00
|
|
|
w32->vo->dwidth = rect_w(w32->windowrc);
|
|
|
|
w32->vo->dheight = rect_h(w32->windowrc);
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int vo_w32_control(struct vo *vo, int *events, int request, void *arg)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = vo->w32;
|
2016-08-05 14:01:26 +00:00
|
|
|
if (request == VOCTRL_CHECK_EVENTS) {
|
|
|
|
*events |= atomic_fetch_and(&w32->event_flags, 0);
|
2016-08-07 14:33:57 +00:00
|
|
|
if (*events & VO_EVENT_RESIZE) {
|
|
|
|
mp_dispatch_lock(w32->dispatch);
|
2017-12-17 00:19:36 +00:00
|
|
|
vo->dwidth = rect_w(w32->windowrc);
|
|
|
|
vo->dheight = rect_h(w32->windowrc);
|
2016-08-07 14:33:57 +00:00
|
|
|
mp_dispatch_unlock(w32->dispatch);
|
|
|
|
}
|
2016-08-05 14:01:26 +00:00
|
|
|
return VO_TRUE;
|
|
|
|
} else {
|
|
|
|
int r;
|
|
|
|
void *p[] = {w32, events, &request, arg, &r};
|
|
|
|
mp_dispatch_run(w32->dispatch, do_control, p);
|
|
|
|
return r;
|
|
|
|
}
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void do_terminate(void *ptr)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = ptr;
|
|
|
|
w32->terminate = true;
|
2014-09-30 02:21:35 +00:00
|
|
|
|
|
|
|
if (!w32->destroyed)
|
|
|
|
DestroyWindow(w32->window);
|
2016-09-16 12:25:50 +00:00
|
|
|
|
|
|
|
mp_dispatch_interrupt(w32->dispatch);
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
}
|
|
|
|
|
2012-04-14 11:39:53 +00:00
|
|
|
void vo_w32_uninit(struct vo *vo)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = vo->w32;
|
2012-04-23 22:41:12 +00:00
|
|
|
if (!w32)
|
|
|
|
return;
|
2014-01-06 11:26:42 +00:00
|
|
|
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
mp_dispatch_run(w32->dispatch, do_terminate, w32);
|
|
|
|
pthread_join(w32->thread, NULL);
|
2014-07-26 18:27:03 +00:00
|
|
|
|
2015-11-08 18:31:09 +00:00
|
|
|
AvRevertMmThreadCharacteristics(w32->avrt_handle);
|
|
|
|
|
2012-04-14 11:39:53 +00:00
|
|
|
talloc_free(w32);
|
|
|
|
vo->w32 = NULL;
|
2003-09-19 14:33:51 +00:00
|
|
|
}
|
2014-07-26 18:27:03 +00:00
|
|
|
|
|
|
|
HWND vo_w32_hwnd(struct vo *vo)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = vo->w32;
|
win32: move window handling to a separate thread
The windows message loop now runs in a separate thread. Rendering,
such as with Direct3D or OpenGL, still happens in the main thread.
In particular, this should prevent the video from freezing if the
window is dragged. (The reason was that the message dispatcher won't
return while the dragging is active, so mpv couldn't update the
video at all.)
This is pretty "rough" and just hacked in, and there's no API yet to
make this easier for other backends. It will be cleaned up later
once we're sure that it works, or when we know how exactly it should
work. One oddity is that OpenGL is actually completely managed in the
renderer thread, while e.g. Cocoa (which has its own threading code)
creates the context in the GUI thread, and then lets the renderer
thread access it.
One strange issue is that we now have to stop WM_CLOSE from actually
closing the window. Instead, we wait until the playloop handles the
close command, and requests the VO to shutdown. This is done mainly
because closing the window apparently destroys it, and then WM_USER
can't be handled anymore - which means the playloop has no way to
wakeup the GUI thread. It seems you can't really win here... maybe
there's a better way to have a thread receive messages with and
without a window, but I didn't find one yet.
Dragging the window (by clicking into the middle of it) behaves
strangely in wine, but didn't before the change. Reason unknown.
2014-07-26 18:31:31 +00:00
|
|
|
return w32->window; // immutable, so no synchronization needed
|
2014-07-26 18:27:03 +00:00
|
|
|
}
|
2014-08-05 17:40:57 +00:00
|
|
|
|
|
|
|
void vo_w32_run_on_thread(struct vo *vo, void (*cb)(void *ctx), void *ctx)
|
|
|
|
{
|
|
|
|
struct vo_w32_state *w32 = vo->w32;
|
|
|
|
mp_dispatch_run(w32->dispatch, cb, ctx);
|
|
|
|
}
|