mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-04-21 04:55:02 -05:00
FIFTH ATTEMPT at getting full-screen mode right. This new attempt
(which I'll comment at some stage) should combine the believable semantics of RDB's fullscreen-by-maximise mode with the multi- monitor friendliness of Wez Furlong's approach while also working on Win98. If we're lucky. [originally from svn r1464]
This commit is contained in:
parent
b0c92ec43a
commit
1f516a8bb0
209
window.c
209
window.c
@ -11,6 +11,11 @@
|
|||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if WINVER < 0x0500
|
||||||
|
#define COMPILE_MULTIMON_STUBS
|
||||||
|
#include <multimon.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
@ -55,6 +60,7 @@
|
|||||||
#define IDM_SAVED_MAX 0x2000
|
#define IDM_SAVED_MAX 0x2000
|
||||||
|
|
||||||
#define WM_IGNORE_CLIP (WM_XUSER + 2)
|
#define WM_IGNORE_CLIP (WM_XUSER + 2)
|
||||||
|
#define WM_FULLSCR_ON_MAX (WM_XUSER + 3)
|
||||||
|
|
||||||
/* Needed for Chinese support and apparently not always defined. */
|
/* Needed for Chinese support and apparently not always defined. */
|
||||||
#ifndef VK_PROCESSKEY
|
#ifndef VK_PROCESSKEY
|
||||||
@ -76,9 +82,13 @@ static void another_font(int);
|
|||||||
static void deinit_fonts(void);
|
static void deinit_fonts(void);
|
||||||
static void set_input_locale(HKL);
|
static void set_input_locale(HKL);
|
||||||
|
|
||||||
|
static int is_full_screen(void);
|
||||||
|
static void make_full_screen(void);
|
||||||
|
static void clear_full_screen(void);
|
||||||
|
static void flip_full_screen(void);
|
||||||
|
|
||||||
/* Window layout information */
|
/* Window layout information */
|
||||||
static void reset_window(int);
|
static void reset_window(int);
|
||||||
static int full_screen = 0;
|
|
||||||
static int extra_width, extra_height;
|
static int extra_width, extra_height;
|
||||||
static int font_width, font_height, font_dualwidth;
|
static int font_width, font_height, font_dualwidth;
|
||||||
static int offset_width, offset_height;
|
static int offset_width, offset_height;
|
||||||
@ -90,7 +100,6 @@ static WPARAM pend_netevent_wParam = 0;
|
|||||||
static LPARAM pend_netevent_lParam = 0;
|
static LPARAM pend_netevent_lParam = 0;
|
||||||
static void enact_pending_netevent(void);
|
static void enact_pending_netevent(void);
|
||||||
static void flash_window(int mode);
|
static void flash_window(int mode);
|
||||||
static void flip_full_screen(void);
|
|
||||||
|
|
||||||
static time_t last_movement = 0;
|
static time_t last_movement = 0;
|
||||||
|
|
||||||
@ -1483,6 +1492,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
HDC hdc;
|
HDC hdc;
|
||||||
static int ignore_clip = FALSE;
|
static int ignore_clip = FALSE;
|
||||||
static int need_backend_resize = FALSE;
|
static int need_backend_resize = FALSE;
|
||||||
|
static int fullscr_on_max = FALSE;
|
||||||
|
|
||||||
switch (message) {
|
switch (message) {
|
||||||
case WM_TIMER:
|
case WM_TIMER:
|
||||||
@ -1613,9 +1623,9 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
(cfg.resize_action == RESIZE_DISABLED)
|
(cfg.resize_action == RESIZE_DISABLED)
|
||||||
? MF_GRAYED : MF_ENABLED);
|
? MF_GRAYED : MF_ENABLED);
|
||||||
/* Gracefully unzoom if necessary */
|
/* Gracefully unzoom if necessary */
|
||||||
if (full_screen &&
|
if (IsZoomed(hwnd) &&
|
||||||
(cfg.resize_action == RESIZE_DISABLED)) {
|
(cfg.resize_action == RESIZE_DISABLED)) {
|
||||||
flip_full_screen();
|
ShowWindow(hwnd, SW_RESTORE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1670,7 +1680,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
nexflag &= ~(WS_EX_CLIENTEDGE);
|
nexflag &= ~(WS_EX_CLIENTEDGE);
|
||||||
|
|
||||||
nflg = flag;
|
nflg = flag;
|
||||||
if (full_screen ?
|
if (is_full_screen() ?
|
||||||
cfg.scrollbar_in_fullscreen : cfg.scrollbar)
|
cfg.scrollbar_in_fullscreen : cfg.scrollbar)
|
||||||
nflg |= WS_VSCROLL;
|
nflg |= WS_VSCROLL;
|
||||||
else
|
else
|
||||||
@ -1814,8 +1824,8 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
PostMessage(hwnd, WM_CHAR, ' ', 0);
|
PostMessage(hwnd, WM_CHAR, ' ', 0);
|
||||||
break;
|
break;
|
||||||
case IDM_FULLSCREEN:
|
case IDM_FULLSCREEN:
|
||||||
flip_full_screen();
|
flip_full_screen();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
if (wParam >= IDM_SAVED_MIN && wParam <= IDM_SAVED_MAX) {
|
if (wParam >= IDM_SAVED_MIN && wParam <= IDM_SAVED_MAX) {
|
||||||
SendMessage(hwnd, WM_SYSCOMMAND, IDM_SAVEDSESS, wParam);
|
SendMessage(hwnd, WM_SYSCOMMAND, IDM_SAVEDSESS, wParam);
|
||||||
@ -1911,7 +1921,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
* window, we put up the System menu instead of doing
|
* window, we put up the System menu instead of doing
|
||||||
* selection.
|
* selection.
|
||||||
*/
|
*/
|
||||||
if (full_screen && press && button == MBT_LEFT &&
|
if (is_full_screen() && press && button == MBT_LEFT &&
|
||||||
X_POS(lParam) == 0 && Y_POS(lParam) == 0) {
|
X_POS(lParam) == 0 && Y_POS(lParam) == 0) {
|
||||||
SendMessage(hwnd, WM_SYSCOMMAND, SC_MOUSEMENU, 0);
|
SendMessage(hwnd, WM_SYSCOMMAND, SC_MOUSEMENU, 0);
|
||||||
return 0;
|
return 0;
|
||||||
@ -2041,7 +2051,6 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
term_update();
|
term_update();
|
||||||
break;
|
break;
|
||||||
case WM_KILLFOCUS:
|
case WM_KILLFOCUS:
|
||||||
if (full_screen) flip_full_screen();
|
|
||||||
show_mouseptr(1);
|
show_mouseptr(1);
|
||||||
has_focus = FALSE;
|
has_focus = FALSE;
|
||||||
DestroyCaret();
|
DestroyCaret();
|
||||||
@ -2158,6 +2167,9 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
/* break; (never reached) */
|
/* break; (never reached) */
|
||||||
|
case WM_FULLSCR_ON_MAX:
|
||||||
|
fullscr_on_max = TRUE;
|
||||||
|
break;
|
||||||
case WM_SIZE:
|
case WM_SIZE:
|
||||||
#ifdef RDB_DEBUG_PATCH
|
#ifdef RDB_DEBUG_PATCH
|
||||||
debug((27, "WM_SIZE %s (%d,%d)",
|
debug((27, "WM_SIZE %s (%d,%d)",
|
||||||
@ -2197,9 +2209,13 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
|
|||||||
|
|
||||||
term_size(h, w, cfg.savelines);
|
term_size(h, w, cfg.savelines);
|
||||||
}
|
}
|
||||||
|
if (fullscr_on_max)
|
||||||
|
make_full_screen();
|
||||||
|
fullscr_on_max = FALSE;
|
||||||
reset_window(0);
|
reset_window(0);
|
||||||
} else if (wParam == SIZE_RESTORED && was_zoomed) {
|
} else if (wParam == SIZE_RESTORED && was_zoomed) {
|
||||||
was_zoomed = 0;
|
was_zoomed = 0;
|
||||||
|
clear_full_screen();
|
||||||
if (cfg.resize_action == RESIZE_TERM)
|
if (cfg.resize_action == RESIZE_TERM)
|
||||||
term_size(prev_rows, prev_cols, cfg.savelines);
|
term_size(prev_rows, prev_cols, cfg.savelines);
|
||||||
if (cfg.resize_action != RESIZE_FONT)
|
if (cfg.resize_action != RESIZE_FONT)
|
||||||
@ -3654,8 +3670,7 @@ void set_sbar(int total, int start, int page)
|
|||||||
{
|
{
|
||||||
SCROLLINFO si;
|
SCROLLINFO si;
|
||||||
|
|
||||||
if ((full_screen && !cfg.scrollbar_in_fullscreen) ||
|
if (is_full_screen() ? !cfg.scrollbar_in_fullscreen : !cfg.scrollbar)
|
||||||
(!full_screen && !cfg.scrollbar))
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
si.cbSize = sizeof(si);
|
si.cbSize = sizeof(si);
|
||||||
@ -4095,68 +4110,6 @@ void beep(int mode)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Toggle full screen mode. Thanks to cwis@nerim.fr for the
|
|
||||||
* implementation.
|
|
||||||
* Revised by <wez@thebrainroom.com>
|
|
||||||
*/
|
|
||||||
static void flip_full_screen(void)
|
|
||||||
{
|
|
||||||
WINDOWPLACEMENT wp;
|
|
||||||
LONG style;
|
|
||||||
|
|
||||||
wp.length = sizeof(wp);
|
|
||||||
GetWindowPlacement(hwnd, &wp);
|
|
||||||
|
|
||||||
full_screen = !full_screen;
|
|
||||||
|
|
||||||
if (full_screen) {
|
|
||||||
if (wp.showCmd == SW_SHOWMAXIMIZED) {
|
|
||||||
/* Ooops it was already 'zoomed' we have to unzoom it before
|
|
||||||
* everything will work right.
|
|
||||||
*/
|
|
||||||
wp.showCmd = SW_SHOWNORMAL;
|
|
||||||
SetWindowPlacement(hwnd, &wp);
|
|
||||||
}
|
|
||||||
|
|
||||||
style = GetWindowLong(hwnd, GWL_STYLE) & ~(WS_CAPTION|WS_THICKFRAME);
|
|
||||||
style &= ~WS_VSCROLL;
|
|
||||||
if (cfg.scrollbar_in_fullscreen)
|
|
||||||
style |= WS_VSCROLL;
|
|
||||||
SetWindowLong(hwnd, GWL_STYLE, style);
|
|
||||||
|
|
||||||
/* Some versions of explorer get confused and don't take
|
|
||||||
* notice of us going fullscreen, so go topmost too.
|
|
||||||
*/
|
|
||||||
SetWindowPos(hwnd, HWND_TOPMOST, 0, 0, 0, 0,
|
|
||||||
SWP_NOACTIVATE | SWP_NOCOPYBITS |
|
|
||||||
SWP_NOMOVE | SWP_NOSIZE |
|
|
||||||
SWP_FRAMECHANGED);
|
|
||||||
|
|
||||||
wp.showCmd = SW_SHOWMAXIMIZED;
|
|
||||||
SetWindowPlacement(hwnd, &wp);
|
|
||||||
} else {
|
|
||||||
style = GetWindowLong(hwnd, GWL_STYLE) | WS_CAPTION;
|
|
||||||
if (cfg.resize_action != RESIZE_DISABLED)
|
|
||||||
style |= WS_THICKFRAME;
|
|
||||||
style &= ~WS_VSCROLL;
|
|
||||||
if (cfg.scrollbar)
|
|
||||||
style |= WS_VSCROLL;
|
|
||||||
SetWindowLong(hwnd, GWL_STYLE, style);
|
|
||||||
|
|
||||||
SetWindowPos(hwnd, HWND_NOTOPMOST, 0, 0, 0, 0,
|
|
||||||
SWP_NOACTIVATE | SWP_NOCOPYBITS |
|
|
||||||
SWP_NOMOVE | SWP_NOSIZE |
|
|
||||||
SWP_FRAMECHANGED);
|
|
||||||
|
|
||||||
wp.showCmd = SW_SHOWNORMAL;
|
|
||||||
SetWindowPlacement(hwnd, &wp);
|
|
||||||
}
|
|
||||||
|
|
||||||
CheckMenuItem(GetSystemMenu(hwnd, FALSE), IDM_FULLSCREEN,
|
|
||||||
MF_BYCOMMAND| full_screen ? MF_CHECKED : MF_UNCHECKED);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Minimise or restore the window in response to a server-side
|
* Minimise or restore the window in response to a server-side
|
||||||
* request.
|
* request.
|
||||||
@ -4186,7 +4139,7 @@ void move_window(int x, int y)
|
|||||||
*/
|
*/
|
||||||
void set_zorder(int top)
|
void set_zorder(int top)
|
||||||
{
|
{
|
||||||
if (cfg.alwaysontop || full_screen)
|
if (cfg.alwaysontop)
|
||||||
return; /* ignore */
|
return; /* ignore */
|
||||||
SetWindowPos(hwnd, top ? HWND_TOP : HWND_BOTTOM, 0, 0, 0, 0,
|
SetWindowPos(hwnd, top ? HWND_TOP : HWND_BOTTOM, 0, 0, 0, 0,
|
||||||
SWP_NOMOVE | SWP_NOSIZE);
|
SWP_NOMOVE | SWP_NOSIZE);
|
||||||
@ -4206,13 +4159,9 @@ void refresh_window(void)
|
|||||||
*/
|
*/
|
||||||
void set_zoomed(int zoomed)
|
void set_zoomed(int zoomed)
|
||||||
{
|
{
|
||||||
if (IsZoomed(hwnd) || full_screen) {
|
if (IsZoomed(hwnd)) {
|
||||||
if (!zoomed) {
|
if (!zoomed)
|
||||||
if (full_screen)
|
ShowWindow(hwnd, SW_RESTORE);
|
||||||
flip_full_screen();
|
|
||||||
else
|
|
||||||
ShowWindow(hwnd, SW_RESTORE);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if (zoomed)
|
if (zoomed)
|
||||||
ShowWindow(hwnd, SW_MAXIMIZE);
|
ShowWindow(hwnd, SW_MAXIMIZE);
|
||||||
@ -4256,3 +4205,101 @@ char *get_window_title(int icon)
|
|||||||
{
|
{
|
||||||
return icon ? icon_name : window_name;
|
return icon ? icon_name : window_name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* See if we're in full-screen mode.
|
||||||
|
*/
|
||||||
|
int is_full_screen()
|
||||||
|
{
|
||||||
|
if (!IsZoomed(hwnd))
|
||||||
|
return FALSE;
|
||||||
|
if (GetWindowLong(hwnd, GWL_STYLE) & WS_CAPTION)
|
||||||
|
return FALSE;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Go full-screen. This should only be called when we are already
|
||||||
|
* maximised.
|
||||||
|
*/
|
||||||
|
void make_full_screen()
|
||||||
|
{
|
||||||
|
DWORD style;
|
||||||
|
int x, y, w, h;
|
||||||
|
|
||||||
|
assert(IsZoomed(hwnd));
|
||||||
|
|
||||||
|
/* Remove the window furniture. */
|
||||||
|
style = GetWindowLong(hwnd, GWL_STYLE);
|
||||||
|
style &= ~(WS_CAPTION | WS_BORDER | WS_THICKFRAME);
|
||||||
|
if (cfg.scrollbar_in_fullscreen)
|
||||||
|
style |= WS_VSCROLL;
|
||||||
|
else
|
||||||
|
style &= ~WS_VSCROLL;
|
||||||
|
SetWindowLong(hwnd, GWL_STYLE, style);
|
||||||
|
|
||||||
|
/* Resize ourselves to exactly cover the nearest monitor. */
|
||||||
|
#ifdef MONITOR_DEFAULTTONEAREST
|
||||||
|
{
|
||||||
|
HMONITOR mon;
|
||||||
|
MONITORINFO mi;
|
||||||
|
mon = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);
|
||||||
|
mi.cbSize = sizeof(mi);
|
||||||
|
GetMonitorInfo(mon, &mi);
|
||||||
|
x = mi.rcMonitor.left;
|
||||||
|
y = mi.rcMonitor.top;
|
||||||
|
w = mi.rcMonitor.right;
|
||||||
|
h = mi.rcMonitor.bottom;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
x = y = 0;
|
||||||
|
w = GetSystemMetrics(SM_CXSCREEN);
|
||||||
|
h = GetSystemMetrics(SM_CYSCREEN);
|
||||||
|
#endif
|
||||||
|
SetWindowPos(hwnd, HWND_TOP, x, y, w, h, SWP_FRAMECHANGED);
|
||||||
|
|
||||||
|
/* Tick the menu item in the System menu. */
|
||||||
|
CheckMenuItem(GetSystemMenu(hwnd, FALSE), IDM_FULLSCREEN,
|
||||||
|
MF_CHECKED);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Clear the full-screen attributes.
|
||||||
|
*/
|
||||||
|
void clear_full_screen()
|
||||||
|
{
|
||||||
|
DWORD oldstyle, style;
|
||||||
|
|
||||||
|
/* Reinstate the window furniture. */
|
||||||
|
style = oldstyle = GetWindowLong(hwnd, GWL_STYLE);
|
||||||
|
style |= WS_CAPTION | WS_BORDER | WS_THICKFRAME;
|
||||||
|
if (cfg.scrollbar)
|
||||||
|
style |= WS_VSCROLL;
|
||||||
|
else
|
||||||
|
style &= ~WS_VSCROLL;
|
||||||
|
if (style != oldstyle) {
|
||||||
|
SetWindowLong(hwnd, GWL_STYLE, style);
|
||||||
|
SetWindowPos(hwnd, NULL, 0, 0, 0, 0,
|
||||||
|
SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER |
|
||||||
|
SWP_FRAMECHANGED);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Untick the menu item in the System menu. */
|
||||||
|
CheckMenuItem(GetSystemMenu(hwnd, FALSE), IDM_FULLSCREEN,
|
||||||
|
MF_UNCHECKED);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Toggle full-screen mode.
|
||||||
|
*/
|
||||||
|
void flip_full_screen()
|
||||||
|
{
|
||||||
|
if (is_full_screen()) {
|
||||||
|
ShowWindow(hwnd, SW_RESTORE);
|
||||||
|
} else if (IsZoomed(hwnd)) {
|
||||||
|
make_full_screen();
|
||||||
|
} else {
|
||||||
|
SendMessage(hwnd, WM_FULLSCR_ON_MAX, 0, 0);
|
||||||
|
ShowWindow(hwnd, SW_MAXIMIZE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user