Many people sooner or later feel restricted by the confinements of a single screen; there is no such thing as too much space. A multi-monitor arrangement is the obvious remedy; however, there is a lesser-known solution that enables the user to have more than one screen on a single display: virtual screens or, as they are called in EnhancedSpaces, mSpaces.
mSpaces have both advantages and disadvantages when compared to multi-monitor arrangements. One of the advantages is that you don't have to move your head around to look at another screen - using mSpaces is like swapping the monitors of your multi-monitor arrangement by means of a keyboard shortcut. An obvious disadvantage is that you can't look at more than one of them at a time; however, while every user prefers a different workflow, for many of us looking at two or more monitors simultaneously is unnecessary, in which case the benefits of mSpaces comfortably outweigh the downsides.
I dismissed my multi-monitor arrangement in favor of virtual screens years ago and have not looked back. I started out using macOS' built-in Spaces and also tried various alternatives; however, something was always missing, which eventually led to the decision to develop an application that manages virtual screens and windows without compromises. Enter EnhancedSpaces.
One of EnhancedSpaces' design goals has been to make managing mSpaces and windows intuitive and efficient - a screen and window manager does its job best when you hardly notice it, i.e., your windows are in the right place and have the right size with you investing the least possible time and energy to achieve that.
EnhancedSpaces' features include using sticky windows, swapping windows, showing all mSpaces on a grid while you can cycle through them, and opening windows automatically on predefined mSpaces in predefined sizes and positions. Using EnhancedSpaces' window manager you can resize and rearrange your windows with keyboard shortcuts or with your mouse or trackpad. The most common features are also available via menu and popup menu.
Focus has also been given to making EnhancedSpaces a pleasure to use. So can you customize EnhancedSpaces' menus to your needs, for example, change them to your preferred language, set individual wallpapers for each of your mSpaces, change the appearance of the window switcher, and resize windows in all directions using the whole area of the windows rather than just their borders.
Another appeal of EnhancedSpaces is that managing both virtual screens and windows results in synergies that are impossible to achieve when using separate applications, for example, you can drag a window beyond the borders of the screen and have it reemerge on another mSpace.
So who is EnhancedSpaces for? It has been made for users who want to use multiple screens on a single display and want an efficient and customizable application to manage them.
EnhancedSpaces has increased my productivity in macOS. May it do the same for you.
EnhancedSpaces requires Hammerspoon, so in case you haven't used the latter yet, go ahead with its installation.
You can simply regard Hammerspoon as a requirement for EnhancedSpaces; you install it alongside EnhancedSpaces and then forget about it.
However, Hammerspoon might be worthy of a second look in case you're interested in improving workflows on your Mac. Let me demonstrate that with an example:
-- connect to VPN
hs.hotkey.bind({ 'cmd', 'shift' }, 's', function()
os.execute('/usr/local/bin/piactl set region switzerland')
os.execute('/usr/local/bin/piactl connect')
end)
-- disconnect
hs.hotkey.bind({ 'cmd', 'shift' }, 'd', function()
os.execute('/usr/local/bin/piactl disconnect')
end)
These few lines of Hammerspoon code enable you to connect to one of PIA's VPN servers using the keyboard shortcut Command-Shift-s
, while Command-Shift-d
disconnects from the VPN server.
There is a plethora of ready-made examples available, and even without ever having written any code, it's easy to adjust the above lines to, for instance, connecting to your VPN server using c
in your hotkey instead of s
, right?
But let's move our focus back to EnhancedSpaces.
Download EnhancedSpaces and move the folder to ~/.hammerspoon/Spoons
. Make sure the name of the folder is EnhancedSpaces.spoon
.
As an alternative to manually downloading and installing, you can run the following terminal command:
mkdir -p ~/.hammerspoon/Spoons && git clone https://github.com/franzbu/EnhancedSpaces.spoon.git ~/.hammerspoon/Spoons/EnhancedSpaces.spoon
Once you've installed EnhancedSpaces, add the following lines to the file ~/.hammerspoon/init.lua
(you can edit that file by clicking the Hammerspoon icon in your menu bar and choosing 'Open Config'). You might also want to adjust the amount and names of your mSpaces and startmSpace
, which is the mSpace you are greeted with:
local EnhancedSpaces = hs.loadSpoon('EnhancedSpaces')
EnhancedSpaces:new({
mSpaces = { '1', '2', '3', 'E', 'T' }, -- default { '1', '2', '3' }
startmSpace = 'E', -- default 2
})
The following terminal command can be used as an alternative to manually editing init.lua
. In this case the default options are used, which means that the mSpaces 1
, 2
, and 3
are created, with 2
as the default mSpace:
echo -e "local EnhancedSpaces = hs.loadSpoon('EnhancedSpaces')\nEnhancedSpaces:new({\nmSpaces = { '1', '2', '3' }, -- default: { '1', '2', '3' }\nstartmSpace = '2', -- default: 2\n})" >> ~/.hammerspoon/init.lua
Reload Hammerspoon's configuration (menu bar icon - 'Reload Config') and you're ready to go. All you see for now is a new icon in your menu bar indicating your current mSpace, so let's find out how you can interact with your new mSpaces.
You can use keyboard shortcuts for handling windows and mSpaces; however, sometimes it can be convenient to get things done via menu.
EnhancedSpaces' menu lets you switch to another mSpace with or without moving a window along, swap windows, get a window from another mSpace, and create references of windows, i.e., the already mentioned 'sticky windows', which means that the same window can be shown on more than one mSpace - more about references of windows in the section Same Window on More Than One mSpace.
Clicking EnhancedSpaces' icon in the menu bar, a menu like this is shown:
mSpaces
is for switching to another mSpace:
Swap
, as its name implies, is for swapping two windows, which means that they adopt each other's positions and sizes:
The next entry in the menu shows the currently active window (in case there is no active window, for example, when you're on an empty mSpace, this entry is empty by design); here you can toggle the references of the window; more about this feature in the section Same Window on More Than One mSpace).
Selecting Send Window
you see a list of all windows on the current mSpace and the mSpaces you can send each of the windows to:
Selecting Get Windows
you see a list of all open windows that are not on your current mSpace, and by selecting one, it is moved to the current mSpace.
It is also possible to use popup menus; more about that feature in the section Additional Features - Popup Menus
For additional functionality there is the possibility for using modifier keys with your menus, for example, for creating references of a window on all mSpaces at once, or for tagging along with a window when sending it to another mSpace; more about that feature in section Advanced Menu Features.
There you can also see how to change the menu entries, for example, to your preferred language. Additionally, you can see how to include Hammerspoon's menu into EnhancedSpaces', which has the additional benefit of making Hammerspoon's menu redundant, i.e., you can then remove it from your menu bar.
You can use Control (ctrl
) - in case you're interested in an elegant alternative in the form of a hyper key, see section Notes - Hyper Key - and s
to cycle through your mSpaces. To cycle in reverse order, press ctrl
and a
. To move the active window to the mSpace on the left (right) and switch there alongside with the window, press ctrl
and q
(w
); to move the window while staying on the current mSpace press ctrl
and d
(f
).
The lines below represent the default setup, and you don't need to add them to your init.lua
unless you want to make changes:
modifierMS = { 'ctrl' }, -- default: { 'ctrl' }
modifierMSKeys = {
'a', -- cycle through mSpaces; default: 'a'
's', -- cycle through mSpaces (other direction); default: 's'
'd', -- send active window to left mSpace; default: 'd'
'f', -- send active window to right mSpace; default: 'f'
'q', -- send active window to left mSpace and switch there; default: 'q'
'w', -- send active window to right mSpace and switch there; default: 'w'
},
In case you want to disable this hotkey, change the line above as follows:
modifierMS = { '' }, -- default: { 'ctrl' }
Whenever changing modifiers, make sure to substitute them with other modifiers such as { 'cmd' }, { 'alt' }, { 'ctrl' }, { 'shift' } and combinations of modifiers, e.g., { 'alt', 'ctrl' }.
Just in case it is not entirely clear where to add these lines or future modifications to your file init.lua
:
local EnhancedSpaces = hs.loadSpoon('EnhancedSpaces')
EnhancedSpaces:new({
-- ______________ modifications to EnhancedSpaces below this line ______________
mSpaces = { '1', '2', '3', 'E', 'T' }, -- default: { '1', '2', '3' }
startmSpace = 'E', -- default: 2
modifierMS = { 'ctrl' }, -- default: { 'ctrl' }
modifierMSKeys = {
'a', -- cycle through mSpaces; default: 'a'
's', -- cycle through mSpaces (other direction); default: 's'
'd', -- send active window to left mSpace; default: 'd'
'f', -- send active window to right mSpace; default: 'f'
'q', -- send active window to left mSpace and switch there; default: 'q'
'w', -- send active window to right mSpace and switch there; default: 'w'
},
-- ______________ modifications to EnhancedSpaces above this line ______________
})
In case you add more modifications, it is entirely up to you which order you put them in, just make sure they're within the boundaries shown above.
For switching directly to any mSpace, press the Option key (alt
) and the key for your mSpace, for example, 3
.
As before, the line below represents the default setup, and you don't need to add it to your init.lua
unless you want to make changes:
modifierSwitchMS = { 'alt' }, -- default: { 'alt' }
In case you want to disable this hotkey, change the line above as follows:
modifierSwitchMS = { '' }, -- default: { 'alt' }
For moving a window to another mSpace, press alt-ctrl
and the key for the target mSpace.
As before, the line below represents the default setup, and you don't need to add it to your init.lua
unless you want to make changes:
modifierMoveWinMSpace = { 'alt', 'ctrl' }, -- default: { 'alt', 'ctrl' }
In case you want to disable this hotkey, change the line above as follows:
modifierMoveWinMSpace = { '' }, -- default: { 'alt', 'ctrl' }
This section is about showing the same window on more than one mSpace, also known as 'sticky window'.
To unlock the full potential of mSpaces, it is helpful to understand the underlying philosophy: Each mSpace is a representation of your windows rather than just a space containing them - or, in other words, an mSpace can be understood as a set of symbolic links to as many of your open windows as you want.
Due to this approach, you can, for instance, have two mSpaces with the same windows in different sizes and positions, or you can have the same Notes, Calendar, Finder or Safari window on two, three, or all your mSpaces.
To create a reference of a window via keyboard shortcut (you've already seen how to do this via menu), press the ctrl-shift
modifiers and additionally press the key corresponding to the target mSpace, for instance, 3
.
As before, the following line represents the default modifier keys, and you don't need to add it to your init.lua
unless you want to apply changes:
modifierReference = { 'ctrl', 'shift' }, -- default: { 'ctrl', 'shift' }
In case you want to disable this hotkey, change the line above as follows:
modifierReference = { '' }, -- default: { 'ctrl', 'shift' }
To delete a reference, press modifierReference
and 0
. In case you're dereferencing, i.e., delete the reference of, the last representation of a window on your mSpaces, the window gets minimized.
In case you'd like to change the key for dereferencing a window, for example, if you want to add an mSpace entitled '0', you can add the following line to your init.lua
and change it to your liking:
deReferenceKey = '0', -- default: '0'
One thing you might want to do sooner rather than later is to set up the feature to open windows in pre-arranged mSpaces as it saves you the time and hassle to move windows to their mSpaces and screen positions after EnhancedSpaces is restarted.
Apart from switching between all of your open windows, for which you can continue using macOS' integrated window switcher (Command-Tab) or the third party switcher of your choice, such as AltTab, additional possibilities have been implemented in EnhancedSpaces for window-switching, namely (1) switching between the windows on the current mSpace and (2) switching between references of windows ('sticky windows').
For switching between the windows of your current mSpace, press alt
and tab
.
As before, the lines below represent the default setup, and you don't need to add them to your init.lua
unless you prefer different shortcuts:
-- keyboard shortcuts for switching between windows on current mSpace and between references
modifierSwitchWin = { 'alt' }, -- default: { 'alt' }
modifierSwitchWinKeys = { 'a', 'q' }, -- default: { 'a', 'q' }
For cycling through the windows of your current mSpace in reverse order, additionally press shift
.
In case you want to disable this hotkey, change the line above as follows:
modifierSwitchWin = { '' }, -- default: { 'alt' }
You can also make two windows swap places using keyboard shortcuts; more about this feature in the section Swapping Windows.
For switching between the references of a window ('sticky windows'), press alt
and escape
. In case you prefer a different key, change the second element in the table modifierSwitchWinKeys
(see above).
With EnhancedSpaces you can automatically resize and position the windows on your mSpaces according to a dynamically changing grid size.
The following lines show the default keyboard shortcuts for the automatic resizing and positioning of windows.
As before, you don't need to add these lines to your init.lua
unless you want to apply changes.
modifierSnap1 = { 'cmd', 'alt' }, -- default: { 'cmd', 'alt' }
modifierSnap2 = { 'cmd', 'ctrl' }, -- default: { 'cmd', 'ctrl' }
modifierSnap3 = { 'cmd', 'shift' }, -- default: { 'cmd', 'shift' }
In case you would like to disable EnhancedSpaces' window manager because you manage your windows manually or prefer using another window manager, change the lines as follows:
modifierSnap1 = { '' }, -- default: { 'cmd', 'alt' }
modifierSnap2 = { '' }, -- default: { 'cmd', 'ctrl' }
modifierSnap3 = { '' }, -- default: { 'cmd', 'shift' }
To resize and move the active window into a 2x2 grid position, use modifierSnap1
(default the Command and Option keys) and numbers 1-8
.
To resize and move the active window into a 3x3 grid position, use modifierSnap2
and numbers 1-9
, additionally 0
, o
, and p
.
modifierSnap3
also uses a 3x3 grid with different sizes and positions, see '3x3 Grid - Additional Window Sizes' below.
Below you find the pre-assigned keyboard shortcuts.
modifierSnap1
and1
: left half of screen -> 'a1'modifierSnap1
and2
: right half of screen -> 'a2'modifierSnap1
and3
: top left quarter of screen -> 'a3'modifierSnap1
and4
: bottom left quarter of screen -> 'a4'modifierSnap1
and5
: top right quarter of screen -> 'a5'modifierSnap1
and6
: bottom right quarter of screen -> 'a6'modifierSnap1
and7
: whole screen -> 'a7'modifierSnap1
and8
: size as is, center of screen -> 'a8'
modifierSnap2
and1
: left third of screen -> 'b1'modifierSnap2
and2
: middle third of screen -> 'b2'modifierSnap2
and3
: right third of screen -> 'b3'modifierSnap2
and4
: left top ninth of screen -> 'b4'modifierSnap2
and5
: left middle ninth of screen -> 'b5'modifierSnap2
and6
: left bottom ninth of screen -> 'b6'modifierSnap2
and7
: middle top ninth of screen -> 'b7'modifierSnap2
and8
: middle middle ninth of screen -> 'b8'modifierSnap2
and9
: middle bottom ninth of screen -> 'b9'modifierSnap2
and0
: right top ninth of screen -> 'b10'modifierSnap2
ando
: right middle ninth of screen -> 'b11'modifierSnap2
andp
: right bottom ninth of screen -> 'b12'
Here, the pre-defined positions and sizes of the windows are as follows (descriptions might be tricky; so simply try them out):
modifierSnap3
and1
: left two thirds of screen: 6 cells -> 'c1'modifierSnap3
and2
: right two thirds of screen: 6 cells -> 'c2'modifierSnap3
and3
: left third, upper two cells -> 'c3'modifierSnap3
and4
: left third, lower two cells -> 'c4'modifierSnap3
and5
: middle third, upper two cells -> 'c5'modifierSnap3
and6
: middle third, lower two cells -> 'c6'modifierSnap3
and7
: right third, upper two cells -> 'c7'modifierSnap3
and8
: right third, lower two cells -> 'c8'modifierSnap3
and9
: top left and middle thirds: 4 cells -> 'c9'modifierSnap3
and0
: bottom left and middle thirds: 4 cells -> 'c10'modifierSnap3
ando
: top middle and right thirds: 4 cells -> 'c11'modifierSnap3
andp
: bottom middle and right thirds: 4 cells -> 'c12'
As has been mentioned, these keyboard shortcuts are fully customizable. Let's first have a look at the default setup:
modifierSnapKeys = {
-- modifierSnapKey1
{
{ 'a1', '1' },
{ 'a2', '2' },
{ 'a3', '3' },
{ 'a4', '4' },
{ 'a5', '5' },
{ 'a6', '6' },
{ 'a7', '7' },
{ 'a8', '8' },
},
-- modifierSnapKey2
{
{ 'b1', '1' },
{ 'b2', '2' },
{ 'b3', '3' },
{ 'b4', '4' },
{ 'b5', '5' },
{ 'b6', '6' },
{ 'b7', '7' },
{ 'b8', '8' },
{ 'b9', '9' },
{ 'b10', '0' },
{ 'b11', 'o' },
{ 'b12', 'p' },
},
-- modifierSnapKey3
{
{ 'c1', '1' },
{ 'c2', '2' },
{ 'c3', '3' },
{ 'c4', '4' },
{ 'c5', '5' },
{ 'c6', '6' },
{ 'c7', '7' },
{ 'c8', '8' },
{ 'c9', '9' },
{ 'c10', '0' },
{ 'c11', 'o' },
{ 'c12', 'p' },
},
},
In case you would like to make changes, you can combine any of the three modifiers modifierSnap1
, modifierSnap1
, and modifierSnap1
with any of the scenarios.
This is best shown by means of an example: Let's assume that you just need windows to snap into three different grid positions:
- (1) right half of screen -> 'a2'
- (2) right middle ninth of screen -> 'b11'
- (3) middle third, upper two cells -> 'c5'
Let's further assume that you would like to use modifierSnap1
with the keys j
, k
, and l
, then these would be the lines to add to your init.lua
:
modifierSnapKeys = {
-- modifierSnapKey1
{
{ 'a2', 'j' },
{ 'b11', 'k' },
{ 'c5', 'l' },
},
-- modifierSnapKey2
{
},
-- modifierSnapKey3
{
},
},
As you can see, modifierSnapKey2
and modifierSnapKey3
are not used and are therefore empty.
Now, by pressing modifierSnapKey1
and j
, for example, the active window snaps into the right half of the screen.
Similar to many operations, at times it can be simpler and faster if you use both your keyboard and pointing device at the same time. Thus EnhancedSpaces provides an alternative by enabling the use of a pointing device whenever it has the potential of being beneficial.
You can also use your pointing device to move a window to an adjacent mSpace by pressing the Option (alt
) or Control (ctrl
) key and dragging 80 percent or more of the window beyond the left or right screen border. If you release the modifier key before releasing the mouse button, the window is moved while you stay on the current mSpace, otherwise you switch to the mSpace alongside with the window.
As per default, modifier1
uses the same modifier key as modifierMS
and modifier2
the same as modifierMoveWinMSpace
. They don't interfere; however, in case you prefer to change these pointing device modifiers, you can add the following lines to your init.lua
and adjust them to your liking:
-- pointing device modifiers
modifier1 = { 'alt' }, -- default: { 'alt' }
modifier2 = { 'ctrl' }, -- default: { 'ctrl' }
To make the process of moving windows easier than the usual clicking the title bar (which you're still free to do), hold down modifier1
or modifier2
, position your cursor in any area within the window, click the left mouse button, and drag the window. If a window is dragged up to 10 percent of its width (left and right borders of screen) or its height (bottom border) outside the screen borders, it will automatically snap back within the borders of the screen. If the window is dragged beyond this 10-percent-limit, things are getting interesting because then window management with automatic resizing and positioning comes into play.
For automatic resizing and positioning of a window, you simply move between 10 and 80 percent of the window beyond the left, right, or bottom borders of your screen while pressing alt
or ctrl
.
As long as windows are resized - or moved within the borders of the screen -, it makes no difference whether you use modifier1
or modifier2
. However, once a window is moved beyond the screen borders, different positioning and resizing scenarios are called into action; they are as follows:
-
modifier1 (
alt
, unless changed):- If windows are moved beyond the left (right) borders of the screen: Imagine your screen border divided into three sections: if the cursor crosses the screen border in the middle section, the window snaps into the left (right) half of the screen. Crossing the screen border in the upper and lower sections, the window snaps into the respective quarters of the screen.
- If windows are moved beyond the bottom border of the screen: Again, imagine your bottom screen border divided into three sections: if the cursor crosses the screen border in the middle section, the window snaps into full screen. Crossing the screen border in the left or right sections, the window snaps into the respective halves of the screen.
-
modifier2 (
ctrl
, unless changed):- The difference to
modifier1
is that your screen has an underlying 3x3 grid. This means that windows snap into the left third of the 3x3 grid when dragged beyond the left screen border and into the right third when dragged beyond the right screen border. Ifctrl
is released before the left mouse button, the window will snap into the middle column.
- The difference to
-
The moment dragging of a window starts, indicators will appear around the borders of the screen to guide you. For changing the appearance of the indicators, see section Additional Features - Change Size, Color, and Opacity of Grid Indicators.
- Additional feature: If you drag a window beyond the bottom border of the screen and
modifier1
ormodifier2
is released before the left mouse button, the window will be minimized.
mSpace Control provides previews of your mSpaces and enables you to switch between them via keyboard or pointing device. mSpace Control also allows you to select windows individually.
Press alt
and tab
to open mSpace Control; in case you'd like to change these keys, add the following lines to your init.lua
and make the adjustments you like:
-- mSpace Control
mSpaceControlModifier = { 'alt' }, -- default: { 'alt' }
mSpaceControlKey = 'tab', -- default: 'tab'
To cycle through your mSpaces, keep mSpaceControlModifier
pressed - or press it again while mSpace Control is open - and use mSpaceControlKey
for cycling. For cycling in reverse order additionally press shift
.
mSpace Control can also be opened via menu.
You can click on any preview of the mSpaces to instantly switch there; selecting an individual window activates it and places the pointer at the center of it.
To change thickness, color and/or opacity of the frame highlighting the current mSpace, add the following to your init.lua
; for adjusting color and opacity you can use values between 0 and 1:
-- mSpace Control: hightlight currently active mSpace
mSpaceControlFrame = {
3, -- frame thickness; default: 3
1, -- red; default: 1
0, -- green; default: 0
0, -- blue; default: 0
1, -- opacity; default: 1
},
In case you'd like to change the padding, color and/or opacity of mSpace Control, add the following to your init.lua
, and adjust the values to your liking. For adjusting color and opacity you can use values between 0 and 1.
-- configure mSpace Control
mSpaceControlConfig = {
50, -- padding; default: 50
0, -- red; default: 0
0, -- green; default: 0
0, -- blue; default: 0
0.9, -- opacity; default: 0.9
},
You can change the opacity of the windows in mSpace Control by adding the following to your init.lua
:
-- mSpace Control: opacity of windows
mSpaceControlWinOpacity = 0.82, -- default: 1
If you want EnhancedSpaces to automatically move windows to specific mSpaces, add the following to your init.lua
:
openAppMSpace = {
{ 'Google Chrome', '2' },
{ 'Microsoft To Do', '3' },
{ 'Safari', '2' },
{ 'Email', 'E' },
}, -- default: nil
The way applications are assigned to certain mSpaces is self-explanatory. To get the names of the applications of currently open windows, you can run the following command in Hammerspoon's Console:
for _, v in pairs(hs.window.filter.default:getWindows()) do print(v:application():name()) end
You will get an output like this:
2024-10-20 07:38:13: Hammerspoon
2024-10-20 07:38:13: Terminal
2024-10-20 07:38:13: Google Chrome
2024-10-20 07:38:13: Code
2024-10-20 07:38:13: Microsoft To Do
2024-10-20 07:38:13: Email
2024-10-20 07:38:13: Finder
In case you would also like to pre-define the position of the window, you can add that information as follows:
openAppMSpace = {
{ 'Google Chrome', '2', 'a1' },
{ 'Microsoft To Do', '3', 'a2' },
{ 'Safari', '2', 'a2' },
{ 'Email', 'E' },
}, -- default: nil
'a1', for example, represents the left half of your screen, 'a2' the right half of your screen. To get the entire list of possible scenarios, see section Automatic Resizing and Positioning - Keyboard.
You have the further option to set a different than the standard padding between a window and the border and between windows; you can add that information as follows:
openAppMSpace = {
{ 'Google Chrome', '2', 'a1', 30, 20 },
{ 'Microsoft To Do', '3', 'a2' },
{ 'Safari', '2', 'a2', 30, 20 },
{ 'Email', 'E' },
}, -- default: nil
In the scenario above, Google Chrome and Safari are placed in the left and right halves of mSpace 2
with a gap of 30 to the screen borders and of 20 between the two windows.
Pressing Ctrl
and Escape
, your active window swaps places with another window of your choice. In case you'd like to change the keys, add the following lines to your init.lua
and adjust them to your liking:
swapModifier = { 'alt' }, -- default: { 'alt' }
swapKey = 's', -- default: 's'
For disabling this hotkey, add the following line to your init.lua
:
swapModifier = { '' }, -- default: { 'ctrl' }
By default, the focus stays with the same window; in case you prefer the focus to switch to the other window (and thus remain in the same position), add the following line to your init.lua
:
swapSwitchFocus = true, -- default: false
For changing the appearance of the switcher for cycling through the windows of the current mSpace when either switching or swapping windows, add the following lines to your init.lua
and adjust the values according to your preferences:
switcherConfig = {
textColor = { 0.9, 0.9, 0.9 }, -- default: { 0.9, 0.9, 0.9 }
fontName = 'Lucida Grande', -- default: 'Lucida Grande'
textSize = 16, -- in screen points; default: 16
highlightColor = { 0.8, 0.5, 0, 0.8 }, -- highlight color for the selected window; default: { 0.8, 0.5, 0, 0.8 }
backgroundColor = { 0.3, 0.3, 0.3, 0.5 }, -- default: { 0.3, 0.3, 0.3, 0.5 }
onlyActiveApplication = false, -- only show windows of the active application; default: false
showTitles = true, -- show window titles; default: true
titleBackgroundColor = { 0, 0, 0 }, -- default: { 0, 0, 0 }
showThumbnails = true, -- show window thumbnails; default: true
selectedThumbnailSize = 284, -- size of window thumbnails in screen points; default: 284
showSelectedThumbnail = true, -- show a larger thumbnail for the currently selected window; default: true
thumbnailSize = 112, -- default: 112
showSelectedTitle = false, -- show larger title for the currently selected window; default: false
},
For selecting an individual wallpaper for each mSpace, add the following to your init.lua
:
-- individual wallpaper for each mSpace
customWallpaper = true, -- default: false
Add the wallpapers you would like to use in the format jpg
to the folder ~/.hammerspoon/Spoons/EnhancedSpaces.spoon/wallpapers/
, and name each file after the corresponding mSpace, for example, 1.jpg
or e.jpg
. If you name one file default.jpg
, that wallpaper will be used whenever there is an mSpace with no corresponding wallpaper.
In case you would like to change the gap in between the windows and/or between the windows and the screen border, add the following lines with values to your liking to your init.lua
:
-- padding between window borders and screen borders
outerPadding = 5, -- default: 5
-- padding between window borders
innerPadding = 5, -- default: 5
In case you would like to change the size, color and/or opacity of the grid indicators, add the following line to your init.lua
, and alter the values according to your liking. Apart from the width, you can use values between 0 and 1::
-- change grid indicators:
gridIndicator = {
20, -- width; default: 20
1, -- red; default: 1
0, -- green; default: 0
0, -- blue; default: 0
0.33 -- opacity; default: 0.33
},
Optionally, you can open a popup menu at the position of your pointing device; this can be enabled by adding the following lines to init.lua
(as popup menus are not enabled by default in EnhancedSpaces, these lines always need to be added in case you want to use a popup menu in EnhancedSpaces, not only if you would like to make changes to the keyboard shortcuts):
-- popup menu
popupModifier = { 'cmd', 'alt', 'ctrl' }, -- default: nil
mbMainPopupKey = 'e', -- default: nil
mbSendPopupKey = 'r', -- default: nil
mbGetPopupKey = 't', -- default: nil
mbSwapPopupKey = 'y', -- default: nil
popupModifier - mbMainPopupKey
opens the main popup menu:
With popupModifier - mbSendPopupKey
the menu 'Send Windows' pops up:
Likewise, with popupModifier - mbGetPopupKey
'Get Windows' pops up:
Likewise, with popupModifier - mbSwapPopupKey
the menu for swapping two windows pops up:
If you set any of the four keys for the according popup menus to nil
(or don't include them in your init.lua
in the first place), consequently they are not available. popupModifier
needs to be set for either of the popup menus to be shown.
It is possible to use modifier keys to access additional features. Below you can see the default modifiers and their functions; you don't need to add these lines to your init.lua
unless you want to make changes:
-- menu: modifier keys to unlock additional features
menuModifier1 = { 'alt' }, -- default: { 'alt' }
menuModifier2 = { 'ctrl' }, -- default: { 'ctrl' }
menuModifier3 = { 'alt', 'ctrl' }, -- default: menuModifier1 and menuModifier1
menuModifier3
by default combines menuModifier1
and menuModifier2
, in other words, menuModifier3
means pressing menuModifier1
and menuModifier2
at the same time.
Below you can see what effect the modifier keys have; the first menu entry, 'mSpaces', reveals its whole potential without any additional modifier keys and is therefore not listed:
- no modifier: swap windows, which includes swapping sizes and positions
- menuModifier1: window chosen first snaps into position 'a1' (left half of screen), second window snaps into 'a2'
- menuModifier2: window chosen first snaps into position 'a3', second window snaps into 'a4'
- menuModifier3: window chosen first snaps into position 'a5', second window snaps into 'a6'
- no modifier: toggle reference of active window on selected mSpace; if all mSpaces end up unchecked, the window is minimized
- menuModifier1: delete all reference except for the selected mSpace
- menuModifier2: put references of window on all mSpaces
- menuModifier3: same as
menuModifier1
, additionally tagging along with the window
- no modifier: send reference of window to selected mSpace; references on other mSpaces remain unaffected
- menuModifier1: send window to selected mSpace while keeping a reference of the window on the current mSpace
- menuModifier2: put references of window on all mSpaces
- menuModifier3: same as
menuModifier1
, additionally tagging along with the window
- no modifier: move selected window to current mSpace
- menuModifier1: create reference of selected window on current mSpace; references on other mSpaces remain unaffected
- menuModifier2: create reference of selected window on all mSpaces
For changing the menu titles, for example, to have them in your preferred language, you can add the following line to your init.lua
; this is an example for changing the menu tiles to German:
-- default: { swap = 'Swap', send = 'Send Window', get = 'Get Window', help = 'Help', about = 'About', hammerspoon = 'Hammerspoon' }
menuTitles = {
swap = 'Vertauschen',
send = 'Senden',
get = 'Holen',
help = 'Hilfe',
about = 'Über',
hammerspoon = 'Hammerspoon'
},
With the entries above, you get the following menu:
As you can see, the last entry from the list above, `hammerspoon = 'Hammerspoon' isn't visible in the menu; that only happens if the integration of Hammerspoon's menu into EnhancedSpaces' is enabled; find out directly below how to do that.
For including Hammerspoon's menu in EnhancedSpaces', add the following to your init.lua
:
-- enable Hammerspoon's menu in EnhancedSpaces'
hammerspoonMenu = true, -- default: false
This results in the following addition to EnhancedSpaces' menu:
This also means that Hammerspoon's menu icon in the menu bar becomes redundant and can be disabled. To do so, uncheck 'Show menu icon' in Hammerspoon's preferences.
For changing the menu titles regarding Hammerspoon, for example, to show them in your preferred language, you can add the following to your init.lua
; this is an example for displaying the menu in German:
-- default: { reload = 'Reload Config', config = 'Open Config', console = 'Console', preferences = 'Preferences', about = 'About Hammerspoon', update = 'Check for Updates...', relaunch = 'Relaunch Hammerspoon', quit = 'Quit Hammerspoon' }
hammerspoonMenuItems = {
reload = 'Konfiguration neu laden',
config = 'Konfiguration öffnen',
console = 'Konsole',
preferences = 'Einstellungen',
about = 'Über Hammerspoon',
update = 'Nach Updates suchen',
relaunch = 'Hammerspoon neu starten',
quit = 'Hammerspoon beenden'
},
This is the resulting menu:
Features in this section have undergone some testing and are supposed to work as expected; however, as they have yet to be thoroughly tested, the occational hiccup should not be entirely unexpected.
Furthermore, as these features aren't final yet, there might still be changes to its implementation, so if one of these features stops working after an update, please return to this documentation in order to implement the necessary adjustments to your init.lua
.
Any operating system has dialogs and windows of a temporary nature, such as Spotlight or Alfred, that are dealt with differently than standard 'persistent' application windows. Hammerspoon's list of such applications by its very nature cannot be comprehensive, which is why you can extend it. Feedback of successfully applied additions is welcome, so they can be added to save other users time.
Should you encounter windows that behave erratically such as claiming focus when they shouldn't, you can add these applications to the list of windows to be left alone by adding the following lines to the configuration of EnhancedSpaces in your init.lua
:
-- window_filter.lua: windows to disregard
SKIP_APPS_TRANSIENT_WINDOWS = {
'Spotlight', 'Notification Center', 'loginwindow', 'ScreenSaverEngine', 'PressAndHold',
'PopClip','Isolator', 'CheatSheet', 'CornerClickBG', 'Moom', 'CursorSense Manager',
'Music Manager', 'Google Drive', 'Dropbox', '1Password mini', 'Colors for Hue', 'MacID',
'CrashPlan menu bar', 'Flux', 'Jettison', 'Bartender', 'SystemPal', 'BetterSnapTool', 'Grandview', 'Radium',
'MenuMetersApp', 'DemoPro', 'DockHelper', 'Maccy', 'Albert', 'Alfred',
},
As has been hinted at, the above list is Hammerspoon's, extended by 'DockHelper', 'Maccy', 'Albert', and 'Alfred'.
Adding your own applications is as easy as adding their names ('application name' followed by a comma). You are advised not to change the listed entries unless you know what you're doing.
To get the names of open applications for adding them to this list, you can use the method referred to in Additional Features - Open Windows in Pre-Arranged mSpaces
The list above represents the default setup, and you don't need to add it to your init.lua
unless you want to extend it.
In case you would like EnhancedSpaces to execute commands at startup, you can add those commands do your init.lua
:
startupCommands = {
'command 1',
'command 2',
'command 3',
...
},
Below is an example for starting JankyBorders:
startupCommands = {
'/opt/homebrew/bin/borders active_color=0xffe1e3e4 inactive_color=0xff494d64 width=5.0 &',
}
Similar to manual moving, manual resizing of windows can be initiated by positioning the cursor in any area of a window. Be aware, though, that windows of certain applications can behave in a sluggish way when being resized.
In order to enable manual resizing, add the following to your init.lua
:
-- enable resizing:
resize = true, -- default: false
To manually resize a window, hold your modifier1
or modifier2
down, then click the right mouse button in any part of the window and drag the window.
To have the additional possibility of precisely resizing windows horizontally-only or vertically-only, 30 percent of the window (15 precent left and right of the middle of each window border) is reserved for horizontal-only and vertical-only resizing. The size of this area can be adjusted; for more information see below.
At the center of the window there is an area (M) where you can also move the window by pressing the right mouse button.
You can change the size of the area of the window where the vertical-only and horizontal-only resizing applies by adjusting the option margin
. The standard value is 0.3, which corresponds to 30 percent. Changing it to 0 would result in deactivating this option, changing it to 1 would result in making resizing this way impossible.
-- adjust the size of the area with vertical-only and horizontal-only resizing:
margin = 0.2, -- default: 0.3
For Apple's Mission Control (F3) to show windows on mSpaces adequately, enable System Settings
- Desktop & Dock
- Group windows by application
.
EnhancedSpaces (and other applications for that matter) can benefit from setting the Caps Lock key up as so-called hyper key, which basically means that you get an additional modifier key, as - due to the impracticality of pressing four modifier keys at once - you would hardly be tempted to use such a combination otherwise.
Among others, you can use the application Karabiner Elements for the creation of your hyper key.
In this scenario, the original function of the Caps Lock key remains untouched. Using the aforementioned Karabiner Elements, in 'Settings - Complex Modifications' you can click 'Add predefined rule' and search for 'Caps Lock → Hyper Key (⌃⌥⇧⌘) (Caps Lock if alone)'. As the name suggests, with this modification you keep the original function of Caps Lock when it is pressed and released, while it also functions as hyper key when another key is pressed before Caps Lock is released.
As an alternative to option one, I present my setup: Caps Lock's functionality is further extended by using a single press (without any other keys) of Caps Lock as simulating pressing the hyper key and spacebar keys, which in turn can be used to open an application such as Alfred, which is what I use this for.
The original purpose of Caps Lock is still available; you can trigger that function by pressing the Shift (Command, Option, and Control work likewise) and Caps Lock keys simultaneously.
For this modification, go to 'Settings - Complex Modifications', click 'Add your own rule' and paste the following lines:
{
"description": "Hyper key implementation",
"manipulators": [
{
"from": { "key_code": "caps_lock" },
"to": [
{
"key_code": "left_option",
"modifiers": ["left_command", "left_control", "left_shift"]
}
],
"to_if_alone": [
{
"key_code": "spacebar",
"modifiers": ["left_command", "left_control", "left_option", "left_shift"]
}
],
"type": "basic"
}
]
}
Now you can assign your newly created hyper key
to any modifiers in EnhancedSpaces, for example, modifierMS
:
modifierMS = { 'cmd', 'alt', 'ctrl', 'shift' }, -- default: { 'ctrl' }
Now, pressing Caps Lock
and a
, for instance, switches to the mSpace on the left.
In case you have used the option openAppMSpace
, disable or remove that section from your init.lua
and (re)start EnhancedSpaces to move all open windows to your main mSpace, which after disabling or uninstalling EnhancedSpaces will automatically become your default space.
--[[
openAppMSpace = {
{'Google Chrome', '2', 'a1'},
{'Code', '2', 'a2'},
{'Microsoft To Do', 'T'},
{'Email', 'E'},
},
]]
As final step, stop Hammerspoon, delete the folder EnhancedSpaces.spoon
in ~/.hammerspoon/Spoons/
and remove the corresponding section from your init.lua
.