I wanted speed. IĀ neededĀ speed.
All this clicking around, all this searching, all the repetitive thinking and mental overhead was cutting into actually getting the job done.
I needed the robots to do the work for me.
Iād recently moved over to macOS after many years on Windows where Iād tinkered around with AutoHotkey.
As a relative Mac noob I started hacking around with Alfred,Ā KarabinerĀ and quickly found solace in the form of Hammerspoon.
Ah,Ā Hammerspoon. Yes, I had consumed the code and modules ā āSpoonsā ā from the super-smart people coding and extending Hammerspoon and that gained me some great mileage, yet I neededĀ more.
Then, there was Lua. I wanted to upskill in Lua.
The best way to do so?Code something.
EnterĀ Hellfred,Ā a collection of 3 mini-apps built on top ofĀ
Itās a way to map repetitive, time-consuming tasks to key sequences, commands or searchable texts.
TL;DR
So you want quick-fire? Skip to the installation andĀ tutorial section belowĀ orĀ checkout
Ā theĀ basics
Ā branch.
git checkout basics
The Apps
Hellfire
A quick-fire, mode-based, hotkey-to-action mapping utility.
- Supports single key triggers as well as key chord sequences as triggers.
- Exposes virtually every key on the keyboard including modifier keys to use as triggers.
- Modes allow the same trigger to have different behaviours under different contexts.
English, please?
When I type a character or a sequence of characters, then execute a specific function, but only if I am in a particular mode.
Ok. An example maybe?
- When I typeĀ c then run functionĀ
launchGoogleChromeApp
Ā (but only if I am in Default Mode) - When I typeĀ wĀ followed byĀ m then run functionĀ
changeToWindowManagerMode
Ā (again, whilst in Default Mode) - When I type c then run functionĀ
centerWindowOnScreen
Ā (whilst in WindowManager mode)
Hellfuzz
A fuzzy-search chooser utility with choice-to-action mapping.
- Supports multi-level (nested) choice sets.
English, please?
When I search through a list of choices and select one, then execute a specific function.Ā If my choice has subchoices (think: parent => children
), then show me those so I can search through them.
Ok. An example maybe?
Suppose you have this structure:
- When I type āgoogā, then highlight the choice āOpen Googleā. Selecting this option will execute the functionĀ
openGoogleInBrowser
. - Alternatively, if I type āappā, then highlight the choice āLaunch Appsā. Selecting this option will replace the current choices withĀ Terminal,Ā Notes, andĀ CalendarĀ (the subset of choices forĀ Launch Apps)
- When I fuzzy search throughĀ thoseĀ and select one, Hellfuzz will executeĀ
launchOrOpenApp
Ā with the selected app.
Hellprompt
A commandline-like utility with basic string matching support.
English, please?
When I type out a command and submit it, then inspect my command for any matching string patterns and execute functions related to that command.
Ok. An example maybe?
-
When I type the command āopen notesā and then hitĀ
enter
, then execute any function with aĀfilter
Ā (e.g. command must start with the word āopenā) and behaviour (e.g. open app associated with ānotesā) suitable to open the Notes app. -
When I type ābrowse githubā and then hitĀ
enter
, then execute any function with a filter (e.g. command starts with the word ābrowseā) and behaviour (open url associated with āgithubā) suitable to open the link.
Installation: Firestarter
- Download and install Hammerspoon
- Install Hellfred: Clone the repository to your `
~/.hammerspoon
` directory:
git clone https://github.com/braddevelop/hellfred.git ~/.hammerspoon
Bootstrap: Light it up
There is aĀ bootstrapĀ fileĀ for Hellfred with a pre-configured setup. Letās reference it in HammerspoonāsĀ init.lua
Ā file.
https://gist.github.com/braddevelop/bac92b6797c087ca42f9363aab4972e0
Save the file and reload the configuration (or save yourself some time and useĀ
Whatās in the box? Try out the pre-configuration
Out-the-box the 3 Hellfred apps are ready to use and are pre-configured with a quick-start example. Letās test it out to make sure everything is wiring and firing.
Try Hellfire
- OpenĀ HellfireĀ with the hotkeyĀ
shift
+ā
+h
- Type the characterĀ
c
- The repo for Hellfred will open in a browser.
Try Hellfuzz
- OpenĀ HellfuzzĀ with the hotkeyĀ
shift
+ā„
+h
- Type in the word āwikiā
- This highlights the option āOpen Hellfred wikiā
- HitĀ
enter
Ā and the wiki for Hellfred will open in a browser
Try Hellprompt
- OpenĀ HellpromptĀ with the hotkeyĀ
shift
+^
+h
- Type āopen codeā
- HitĀ
enter
Ā and the code repo for Hellfred will open in a browser
WhatĀ Hellfire,Ā HellfuzzĀ andĀ HellpromptĀ achieve is map aĀ triggerĀ orĀ inputĀ to anĀ actionĀ orĀ behaviour,Ā if-this-then-that
, and whilst we have just demonstrated using each app to achieve the same outcome, you will find each app more suited to certain use cases than others.
Tutorial: A basic setup
What we will be programming
Now letās turn up the heat and configure something a little more useful. We are going to program each app to solve the following scenarios so that you can get the hang of things:
- A simple app launcher
- A url launcher for commonly visited links
You can find the final files for this walkthrough in the hellfred/extend/basics directory on the repositoryās basics branch.
Patterns to note
Each app follows a similar set of steps.
- Initialise the app with a hotkey binding
- Configure `
Subscriber
` objects (This could be done in hellfred-bootstrap.lua but we will be creating separate files to keep things squeaky clean. Weāll leverage factory methods to make object creation a breeze) - Register the subscribers with the app
- Hotkey to run the app
- Destroy time-sucking tasks
Setup for Hellfire
Hellfire works a little something like this:
Application Launcher
Initialise the app
This is already done with the pre-configuration in hellfred-bootstrap.lua
. Feel free to change the hotkey to something else.
https://gist.github.com/braddevelop/49c7e7c7c62350d93e7798ff2bf40541
ConfigureĀ Subscriber
Ā objects
triggers
and callbacks
are user-defined and wrapped inside simple configuration objects. These objects, act as subscribers
when registered with the respective app and notified whenever something important happens inside the app.
Subscriber objects for Hellfire follow this structure:
https://gist.github.com/braddevelop/20c9c5b97a85a9a5c4cb924b511e494a
Note: IfĀ fireIfModeIs
Ā is not defined, Hellfire will set the ANY
Ā mode by default, meaning the callback will fire in any mode when triggered.
Create the following directory structure if it does not exist:Ā hellfred/extend/basics
.
Then create a new Lua file inside theĀ basics
Ā directory calledĀ hellfirepack-applications.lua
ā the file naming convention has no importance.
Add this code:
https://gist.github.com/braddevelop/c12d37d64ce75b83a195666317a10aa2
Register the subscribers with the app
Back inĀ hellfred-bootstrap.lua
, we need to register the pack of subscribers we have just configured.
https://gist.github.com/braddevelop/2298469b805978062421f5a09d01d11a
Run the app
Enter Hellfire (shift
+ ā
+ h
) and type any of the new triggers:f
Ā to open Finder app,Ā t
Ā to open Terminal orĀ n
Ā to open Notes app.
Link Launcher (using Hellfire Modes)
ConfigureĀ Subscriber
Ā objects
Alright now letās configure the subscribers for our Common Links url launcher. Create a new Lua file inĀ hellfred/extend/basics
Ā calledĀ hellfirepack-common-links.lua
Add this code:
https://gist.github.com/braddevelop/7d637464f45d2d9a4641e365ad2f20a1
Register the subscribers with the app
Back inĀ hellfred-bootstrap.lua
, we need to register the pack of subscribers we have just configured.
https://gist.github.com/braddevelop/f0dafb518797620747044afbc20da060
Run the app
Enter Hellfire (shift
+ ā
+ h
) and type any of the new triggers:Ā t
,Ā g
,Ā h
Ā orĀ s
.
Hang on!Ā Did you notice that typing the triggerĀ t
Ā launchedĀ TerminalĀ as well asĀ opened theĀ TechCrunchĀ website? Thatās probably not what we want to happen. Letās take advantage ofĀ Hellfireās ModeĀ feature.
Modes offer a way to have the same trigger behave differently under different contexts.
By default, Hellfire initialises in a mode calledā¦you guessed it⦠āDefaultā mode.
We can configure some custom modes to use with Hellfire so that triggers can behave differently under different modes ā or ānamespacesā if you like.
Consider this flow:
Create a new Lua file inĀ hellfred/extend/basics
calledĀ hellfire-modes-extended.lua
.
We will create a separate mode for the common links triggersĀ to fire in.
Add this code:
https://gist.github.com/braddevelop/1e3f345076a8cd5c4cdf843cad785c9e
We are going to require this file inĀ hellfred-bootstrap.lua
Ā so it isĀ
Add the following code inĀ hellfred-bootstrap.lua
Ā (under the metadata section, towards the top of the file)
https://gist.github.com/braddevelop/04a4ec21aab06f50760d1082b216426f
Now we need a way to change the mode to our newĀ Common Links mode.
Weāll use the key sequence ofc
followed byĀ l
.
We also need to be able to get back toĀ Default mode
Weāll use the semi-colon;
Ā as a trigger.
Thatās nextā¦
ConfigureĀ Subscriber
Ā objects that trigger mode changes
Create a new Lua file inĀ hellfred/extend/basics
Ā calledĀ hellfire-mode-triggers.lua
Add this code:
https://gist.github.com/braddevelop/283acacb9634590208c3b29b6039dc12
Register the subscribers with the app
InĀ hellfred-bootstrap.lua
, register the subscribers for the new mode triggers:
https://gist.github.com/braddevelop/8afc4a576d550415f22c3c6fbf4152f1
Test switching between modes
Now enter Hellfire (shift
+ ā
+ h
) and toggle between the two modes.Ā Modes FTW!
Update subscribers to work in modes
Now we need to update our subscribers inĀ hellfirepack-common-links.lua
Ā so that they only fire whenĀ Common Links modeĀ is active.
We will update the factory method and assign _G.HELLFIRE_MODES_EXTENDED.COMMON_LINKS
toĀ fireIfModeIs
Ā instead ofĀ nil
.
The updated method should look like this:
https://gist.github.com/braddevelop/152a16f87efe55f08e7ed7139225f757
We also need to update our subscribers in hellfirepack-applications.lua
so that they only fire whenĀ Hellfireās Default modeĀ is active.
The updated method should look like this:
https://gist.github.com/braddevelop/8ed8dea04397b0471299c8c89f8a9ae0
Run the app
Enter Hellfire (shift
+ ā
+ h
) and toggle between the modes. The triggerĀ t
now behaves differently depending on the mode that Hellfire is in.Ā Hell yeah!
Setup for Hellprompt
This is how Hellprompt functions:
Application and URL Launcher
Initialise the app
This is already done with the pre-configuration inĀ hellfred-bootstrap.lua
. Feel free to change the hotkey to something else.
https://gist.github.com/braddevelop/11e9b4871182f90c57313ab6ffa939a4
ConfigureĀ Subscriber
Ā objects
Subscribers for Hellprompt take a different structure to those for Hellfire.
Consider this structure:
https://gist.github.com/braddevelop/c247d1d1429234c96f3e8c7d8a0b48df
Note: IfĀ filter
Ā is not defined then the callback will always be executed.
Create a new Lua file inĀ hellfred/extend/basics
Ā calledĀ hellpromptpack-commands.lua
Add this code:
https://gist.github.com/braddevelop/6b18219f83fcf69826083d5dfce5d06f
Register the subscribers with the app
Back inĀ hellfred-bootstrap.lua
, we need to register the pack of subscribers we have just configured.
https://gist.github.com/braddevelop/f1b75646b7330fcb1c993a5737fd7d61
Run the app
Enter Hellprompt (shift
+ ^
+ h
) and test out those commands.Ā Inferno!
Try:
browse news
and:
open terminal
Setup for Hellfuzz
This is how Hellfuzz works:
Application Launcher
Initialise the app
This is already done with the pre-configuration inĀ hellfred-bootstrap.lua
. Feel free to change the hotkey to something else.
https://gist.github.com/braddevelop/b18f575ec347503628e7457217b6e187
ConfigureĀ Subscriber
Ā objects
Subscribers for Hellfuzz take a different structure to the other apps.
Consider this structure:
https://gist.github.com/braddevelop/56e9c22c6f921575637c73d60920d208
Note: IfĀ nextChoicesFn
Ā is defined thenĀ callback
Ā is ignored.
To make things easier weāll use a helper method to configure subscribers for Hellfuzz
Create a new Lua file inĀ hellfred/extend/basics
Ā calledĀ hellfuzzpack-apps-and-links.lua
Add this code:
https://gist.github.com/braddevelop/3564ad1ec286a30eca1abb94cfd6a69a
Register the subscribers with the app
In hellfred-bootstrap.lua
, register the pack of subscribers.
https://gist.github.com/braddevelop/a90f9d5635377d74b145a892845c8273
Run the app
Enter Hellfuzz (shift
+ ā„
+ h
) and type in a command. For example start typing the wordĀ āTerminalā, youāll see the option to open Terminal is highlighted. PressĀ enter
Ā and Terminal opens.Ā Smoking hot!
Link Launcher (using nested choice sets)
A handy feature of Hellfuzz is the ability to nest sets of choices.
Consider this updated flow:
Letās try this out on our Link LauncherĀ task, weāll create the following hierarchical choice structure:
Update the code inĀ hellfuzzpack-apps-and-links.lua
Ā to the following:
https://gist.github.com/braddevelop/384b0ae9629faca95d5d4cb519cd8e61
Now enter Hellfuzz (shift
+ ā„
+ h
) and start searching for āCommon linksā. You can select the āCommon linksā choice, and the sub-set of choices fromĀ commonLinkNextChoices
Ā will be displayed and can be fuzzy searched. Selecting any of the link options will open the respective url.
Extensions: Add fuel to the fire.
Look out for upcoming Hellfred experiments and extensions on the repo by checking out theĀ extend
Ā branch.
git checkout extend
Now go raise hellĀ \m/