FIFTH ATTEMPT at getting full-screen mode right. This new attempt
authorsimon <simon@cda61777-01e9-0310-a592-d414129be87e>
Fri, 7 Dec 2001 21:21:03 +0000 (21:21 +0000)
committersimon <simon@cda61777-01e9-0310-a592-d414129be87e>
Fri, 7 Dec 2001 21:21:03 +0000 (21:21 +0000)
(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.

git-svn-id: svn://svn.tartarus.org/sgt/putty@1464 cda61777-01e9-0310-a592-d414129be87e

window.c

index 15f2387..8481d45 100644 (file)
--- a/window.c
+++ b/window.c
 #endif
 #endif
 
+#if WINVER < 0x0500
+#define COMPILE_MULTIMON_STUBS
+#include <multimon.h>
+#endif
+
 #include <stdio.h>
 #include <stdlib.h>
 #include <ctype.h>
@@ -55,6 +60,7 @@
 #define IDM_SAVED_MAX 0x2000
 
 #define WM_IGNORE_CLIP (WM_XUSER + 2)
+#define WM_FULLSCR_ON_MAX (WM_XUSER + 3)
 
 /* Needed for Chinese support and apparently not always defined. */
 #ifndef VK_PROCESSKEY
@@ -76,9 +82,13 @@ static void another_font(int);
 static void deinit_fonts(void);
 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 */
 static void reset_window(int);
-static int full_screen = 0;
 static int extra_width, extra_height;
 static int font_width, font_height, font_dualwidth;
 static int offset_width, offset_height;
@@ -90,7 +100,6 @@ static WPARAM pend_netevent_wParam = 0;
 static LPARAM pend_netevent_lParam = 0;
 static void enact_pending_netevent(void);
 static void flash_window(int mode);
-static void flip_full_screen(void);
 
 static time_t last_movement = 0;
 
@@ -1483,6 +1492,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
     HDC hdc;
     static int ignore_clip = FALSE;
     static int need_backend_resize = FALSE;
+    static int fullscr_on_max = FALSE;
 
     switch (message) {
       case WM_TIMER:
@@ -1613,9 +1623,9 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
                                   (cfg.resize_action == RESIZE_DISABLED)
                                   ? MF_GRAYED : MF_ENABLED);
                    /* Gracefully unzoom if necessary */
-                   if (full_screen &&
+                   if (IsZoomed(hwnd) &&
                        (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);
 
                    nflg = flag;
-                   if (full_screen ?
+                   if (is_full_screen() ?
                        cfg.scrollbar_in_fullscreen : cfg.scrollbar)
                        nflg |= WS_VSCROLL;
                    else
@@ -1814,8 +1824,8 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
                PostMessage(hwnd, WM_CHAR, ' ', 0);
            break;
          case IDM_FULLSCREEN:
-               flip_full_screen();     
-               break;
+           flip_full_screen();
+           break;
          default:
            if (wParam >= IDM_SAVED_MIN && wParam <= IDM_SAVED_MAX) {
                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
             * selection.
             */
-           if (full_screen && press && button == MBT_LEFT &&
+           if (is_full_screen() && press && button == MBT_LEFT &&
                X_POS(lParam) == 0 && Y_POS(lParam) == 0) {
                SendMessage(hwnd, WM_SYSCOMMAND, SC_MOUSEMENU, 0);
                return 0;
@@ -2041,7 +2051,6 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
        term_update();
        break;
       case WM_KILLFOCUS:
-       if (full_screen) flip_full_screen();
        show_mouseptr(1);
        has_focus = FALSE;
        DestroyCaret();
@@ -2158,6 +2167,9 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
            return rv;
        }
        /* break;  (never reached) */
+      case WM_FULLSCR_ON_MAX:
+       fullscr_on_max = TRUE;
+       break;
       case WM_SIZE:
 #ifdef RDB_DEBUG_PATCH
        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);
                    }
+                   if (fullscr_on_max)
+                       make_full_screen();
+                   fullscr_on_max = FALSE;
                    reset_window(0);
                } else if (wParam == SIZE_RESTORED && was_zoomed) {
                    was_zoomed = 0;
+                   clear_full_screen();
                    if (cfg.resize_action == RESIZE_TERM)
                        term_size(prev_rows, prev_cols, cfg.savelines);
                    if (cfg.resize_action != RESIZE_FONT)
@@ -3654,8 +3670,7 @@ void set_sbar(int total, int start, int page)
 {
     SCROLLINFO si;
 
-    if ((full_screen && !cfg.scrollbar_in_fullscreen) ||
-       (!full_screen && !cfg.scrollbar))
+    if (is_full_screen() ? !cfg.scrollbar_in_fullscreen : !cfg.scrollbar)
        return;
 
     si.cbSize = sizeof(si);
@@ -4096,68 +4111,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
  * request.
  */
@@ -4186,7 +4139,7 @@ void move_window(int x, int y)
  */
 void set_zorder(int top)
 {
-    if (cfg.alwaysontop || full_screen)
+    if (cfg.alwaysontop)
        return;                        /* ignore */
     SetWindowPos(hwnd, top ? HWND_TOP : HWND_BOTTOM, 0, 0, 0, 0,
                 SWP_NOMOVE | SWP_NOSIZE);
@@ -4206,13 +4159,9 @@ void refresh_window(void)
  */
 void set_zoomed(int zoomed)
 {
-    if (IsZoomed(hwnd) || full_screen) {
-       if (!zoomed) {
-           if (full_screen)
-               flip_full_screen();
-           else
-               ShowWindow(hwnd, SW_RESTORE);
-       }
+    if (IsZoomed(hwnd)) {
+        if (!zoomed)
+           ShowWindow(hwnd, SW_RESTORE);
     } else {
        if (zoomed)
            ShowWindow(hwnd, SW_MAXIMIZE);
@@ -4256,3 +4205,101 @@ char *get_window_title(int icon)
 {
     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);
+    }
+}