How to Properly Maximize the Active Emacs Frame on Startup on Windows

3 minute read

I’m an avid GNU Emacs user and I like to have the active frame maximized at start-up. This is easily done on Linux:

(set-frame-parameter nil 'fullscreen 'maximized)

But on MS Windows this is not so trivial… Like many of you, I have to use Windows for my daily work. I tried many different approaches to get that frame maximized on Windows, but just recently I was able to achieve that outcome in a reliable manner that works across many different versions of Windows.

What worked!

In an effort to not bury the lead, as they say… this is what I have done that works:

(defun w32-maximize-frame ()
  "Maximize the current frame (windows only)"
  (interactive)
  (w32-send-sys-command 61488))

(if (eq system-type 'windows-nt)
    (progn
      (add-hook 'window-setup-hook 'w32-maximize-frame t))
  (set-frame-parameter nil 'fullscreen 'maximized))

In the above code I address maximizing the frame in the two platforms I use most: Linux and Windows. I defined the function w32-maximize-frame, which send the Windows message to the Emacs frame for it to maximize itself. The underlying function w32-send-sys-command is well-known, but initially it did not work for me as I’ll explain below.

The strategy that did work for Windows was to wrap the call to w32-send-sys-command in a function and then add a call to it to the window-setup-hook. I tested this on the 7 and 8.1 versions of desktop Windows and on 2008 R2, 2012 versions of Windows Server and using Emacs 24.3. And if Emacs is not running on Windows, I set the frame parameter fullscreen to maximized for the current frame.

What didn’t work…

I started this journey into figuring out how to start Emacs maximized a while back. In the beginning I would let Emacs start and then just use the maximize button on the title bar, but then I started using things like initial-frame-alist and set-frame-* to get the top, left and size just right for my different screens. The problem with that approach is that I had to tweak it for every new install of Emacs on every computer I used. It got old really quick.

Then I found the set-frame-parameter trick that you can see in the code above. But that only worked for Linux variants. At this point during my research, I have already found out about the w32-send-sys-command approach, but using it on my init.el file directly was not working. The frame would think itself as being maximized, and you could see that by looking at the state of the maximize button on the top-right of the window. But visually, the window would remain small.

Next I started using the maxframe.el package, but that really didn’t last all that long. I found that there were several flaws with it: First, the package would not truly maximize the frame, but it calculates the size of the frame based on your monitor’s resolution and font face sizes and stretch your frame accordingly; Second, the package would not account for the position of my XFCE desktop panel and place the Emacs title bar under it; And third, maxframe.el would not account for a multiple monitor setup and stretch the frame across my two monitors. So I dropped it and went back to the drawing board.

Conclusion

It turns out that the proper way to achieve the maximize frame on start-up on Windows is to use the window-setup-hook. This variable is a normal hook that Emacs runs after handling the initialization files. It turns out that with the previous approaches, I was trying to send a message in Windows to an Emacs frame that didn’t yet exist. This hook is used for setting up communication with the windowing system and creating the initial window and is, therefore, the proper place, or point in time, to maximize our frame in Windows.