| name | hammerspoon |
| description | This skill should be used when automating macOS with Hammerspoon, configuring window management hotkeys, working with Spoons (plugins), writing Lua configuration in init.lua, or using the hs CLI for scripting and reloading. |
Hammerspoon macOS Automation
Hammerspoon bridges macOS and Lua scripting for powerful desktop automation.
Directory Structure
~/.hammerspoon/
āāā init.lua # Main entry point (always loaded on startup)
āāā Spoons/ # Plugin directory
ā āāā *.spoon/ # Individual Spoon packages
ā āāā init.lua # Spoon entry point
āāā .gitignore
Configuration Basics
init.lua - Entry Point
Hammerspoon always loads ~/.hammerspoon/init.lua on startup:
require("hs.ipc")
hs.loadSpoon("SpoonName")
spoon.SpoonName:bindHotkeys({...})
Loading Spoons
hs.loadSpoon("MySpoon")
local mySpoon = hs.loadSpoon("MySpoon", false)
When loaded, Spoons are accessible via spoon.SpoonName.
CLI Usage (hs command)
Prerequisite: Add require("hs.ipc") to init.lua, then reload manually once.
hs -c 'hs.reload()'
hs -c 'hs.alert("Hello from CLI")'
hs -c 'print(hs.host.locale.current())'
hs -c 'print(hs.window.focusedWindow():title())'
Window Management with ShiftIt
ShiftIt is a popular Spoon for window tiling.
Installation
Configuration
require("hs.ipc")
hs.loadSpoon("ShiftIt")
spoon.ShiftIt:bindHotkeys({
left = { { 'ctrl', 'cmd' }, 'left' },
right = { { 'ctrl', 'cmd' }, 'right' },
up = { { 'ctrl', 'cmd' }, 'up' },
down = { { 'ctrl', 'cmd' }, 'down' },
upleft = { { 'ctrl', 'cmd' }, '1' },
upright = { { 'ctrl', 'cmd' }, '2' },
botleft = { { 'ctrl', 'cmd' }, '3' },
botright = { { 'ctrl', 'cmd' }, '4' },
maximum = { { 'ctrl', 'cmd' }, 'm' },
toggleFullScreen = { { 'ctrl', 'cmd' }, 'f' },
center = { { 'ctrl', 'cmd' }, 'c' },
nextScreen = { { 'ctrl', 'cmd' }, 'n' },
previousScreen = { { 'ctrl', 'cmd' }, 'p' },
resizeOut = { { 'ctrl', 'cmd' }, '=' },
resizeIn = { { 'ctrl', 'cmd' }, '-' },
})
Modifier Keys
| Key | Lua Name |
|---|
| Command | 'cmd' |
| Control | 'ctrl' |
| Option/Alt | 'alt' |
| Shift | 'shift' |
Hotkey Binding (Without Spoons)
hs.hotkey.bind({'cmd', 'alt'}, 'R', function()
hs.reload()
end)
hs.hotkey.bind({'cmd', 'shift'}, 'H', function()
hs.alert.show('Hello!')
end)
Common Modules
hs.window - Window Management
local win = hs.window.focusedWindow()
win:moveToUnit('[0,0,0.5,1]')
win:maximize()
win:centerOnScreen()
local allWindows = hs.window.allWindows()
hs.application - App Control
hs.application.launchOrFocus('Safari')
local app = hs.application.get('Finder')
app:activate()
hs.alert - On-screen Messages
hs.alert.show('Message')
hs.alert.show('Message', nil, nil, 3)
hs.notify - System Notifications
hs.notify.new({title='Title', informativeText='Body'}):send()
hs.caffeinate - Sleep/Wake
hs.caffeinate.set('displayIdle', true)
hs.caffeinate.watcher.new(function(event)
if event == hs.caffeinate.watcher.systemWillSleep then
print('Going to sleep')
end
end):start()
Spoons
What is a Spoon?
Self-contained Lua plugin with standard structure:
MySpoon.spoon/
āāā init.lua # Required: exports a table with methods
Official Spoon Repository
SpoonInstall - Package Manager
hs.loadSpoon("SpoonInstall")
spoon.SpoonInstall:andUse("ReloadConfiguration", {
start = true
})
spoon.SpoonInstall.repos.Custom = {
url = "https://github.com/user/repo",
desc = "Custom spoons",
branch = "main",
}
spoon.SpoonInstall:andUse("CustomSpoon", { repo = "Custom" })
Configuration Reloading
Manual Reload
- Click menubar icon -> "Reload Config"
- Or bind a hotkey:
hs.hotkey.bind({'cmd', 'alt', 'ctrl'}, 'R', function()
hs.reload()
end)
Auto-reload on File Change
hs.loadSpoon("ReloadConfiguration")
spoon.ReloadConfiguration:start()
Or manually:
local configWatcher = hs.pathwatcher.new(os.getenv('HOME') .. '/.hammerspoon/', function(files)
for _, file in pairs(files) do
if file:sub(-4) == '.lua' then
hs.reload()
return
end
end
end):start()
CLI Reload
hs -c 'hs.reload()'
Note: Requires require("hs.ipc") in init.lua.
Troubleshooting
IPC Not Working
error: can't access Hammerspoon message port
Fix: Add require("hs.ipc") to init.lua and reload manually via menubar.
Spoon Not Loading
- Check path:
~/.hammerspoon/Spoons/Name.spoon/init.lua
- Check Lua syntax in Spoon's init.lua
- Check Hammerspoon console for errors (menubar -> Console)
Hotkey Not Working
- Check for conflicts with system shortcuts
- Verify modifier key names are lowercase strings
- Check console for binding errors
Console and Debugging
print('Debug message')
hs.inspect(someTable)
hs.openConsole()
Access console: Menubar icon -> Console (or Cmd+Alt+C if bound)
Best Practices
- Always use IPC - Add
require("hs.ipc") for CLI support
- Use Spoons - Don't reinvent window management
- Version control - Track
~/.hammerspoon/ with git
- Capture variables - Objects not stored in variables get garbage collected
- Check console - First place to look for errors
References