I work almost exclusively from my home office and I like to use good quality products that make my working life more comfortable.
ℹ️ I change aspects of my setup fairly often so this page will serve as a living document and a place to point curious developers to when I get asked.
Hardware
Computer
14” Apple M1 MacBook Pro 💻. I love this little thing. It should last a while.
Monitor
MSI Prestige PS341WU Creator Monitor 34”. Great wide-screen monitor without being too pricey.
Keyboard
Keychron Q1 version 2 Keyboard. I bought the barebones ISO version, Gateron Oil King switches and PBTFans Resonance keycaps from Mech Mods. What’s not to love?
Mouse
I alternate between the Logitech MX Master 3 and the Logitech Vertical mouse.
Desk and chair
Autonomous Smart Desk 2. An amazing and affordable standing desk and the Herman Miller Aeron. You need to invest in a good chair if you’re going to be sitting in it for hours on end.
Software
IDE
VS Code is my daily driver, but I’ve been using Cursor at work recently. We have found that it really helps with refactoring and converting from JS to TS.
I tend to switch my colour themes fairly often, but these are the most used:
- Rosé Pine Moon
- Catppuccin Mocha
- Rigel which I made actually
I also switch the fonts I use too:
- Comic Code font (as much as I hate Comic Sans, I happen to LOVE Comic Code for coding.. who knew!)
- Recursive Mono
- Berkeley Mono.
A copy of my settings.json
can be found here and a full list of my installed extensions can be found here.
Extensions
- Surround a simple yet powerful extension to add wrapper templates around your code blocks. It has many built-in templates and you can add your own. I’ve added a few of my own templates to the settings below.
- Error Lens
- Multiple cursor case preserve
- Project Manager
- Prettier
- Rewrap
- Toggle Quotes
- Turbo Console Log
- Sublime Commands
- REST Client
Surround settings
{
"surround.custom": {
"promise": {
"description": "Promise<...>",
"disabled": false,
"label": "Promise<...>",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "Promise<$TM_SELECTED_TEXT>"
},
"console-log": {
"description": "console.log( ... )",
"disabled": false,
"label": "console.log",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "console.log(`$TM_SELECTED_TEXT`, $TM_SELECTED_TEXT)"
},
"describe": {
"description": "describe('${1:name}', () => { ... })",
"disabled": false,
"label": "describe",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact"
],
"snippet": "describe('${1:name}', () => {\n\t$TM_SELECTED_TEXT\n})$0"
},
"errorBoundary": {
"description": "<ErrorBoundary>",
"disabled": false,
"label": "errorBoundary",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact"
],
"snippet": "<ErrorBoundary>$TM_SELECTED_TEXT</ErrorBoundary>"
},
"es6StringSubstition": {
"description": "${...}",
"disabled": false,
"label": "es6StringSubstition",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "${$TM_SELECTED_TEXT}"
},
"es6StringSubstitionBackticks": {
"description": "`${...}`",
"disabled": false,
"label": "es6StringSubstitionBackticks",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "`${$TM_SELECTED_TEXT}$0`"
},
"function": {
"description": "$( ... )",
"disabled": false,
"label": "function",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "$1($TM_SELECTED_TEXT)$0"
},
"jsxConditional": {
"description": "{x && ( ... )}",
"disabled": false,
"label": "jsxConditional",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "{$0 && ($TM_SELECTED_TEXT)}"
},
"jsxConditional2": {
"description": "{x ? ( ... ) : null}",
"disabled": false,
"label": "better jsxConditional",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "{$0 ? ($TM_SELECTED_TEXT) : null}"
},
"markdown-link": {
"description": "[]( ... )",
"disabled": false,
"label": "Markdown link [url]",
"snippet": "[$0]($TM_SELECTED_TEXT)"
},
"markdown-link-alt": {
"description": "[ ... ]()",
"disabled": false,
"label": "Markdown link [text]",
"snippet": "[$TM_SELECTED_TEXT]($0)"
},
"number": {
"description": "Number( ... )",
"disabled": false,
"label": "Number",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "Number($TM_SELECTED_TEXT)"
},
"parse": {
"description": "JSON.parse( ... )",
"disabled": false,
"label": "JSON.parse",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"astro"
],
"snippet": "JSON.parse($TM_SELECTED_TEXT)$0"
},
"reactFragment": {
"description": "<Fragment>",
"disabled": false,
"label": "fragment",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "<Fragment>$TM_SELECTED_TEXT</Fragment>"
},
"reactFragment2": {
"description": "<>",
"disabled": false,
"label": "fragment (simple)",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "<>$TM_SELECTED_TEXT</>"
},
"reactUseMemo": {
"description": "useMemo",
"disabled": false,
"label": "useMemo",
"languageIds": [
"javascriptreact",
"javascript",
"typescriptreact",
"typescript"
],
"snippet": "useMemo(() => $TM_SELECTED_TEXT, [])"
},
"reactUseCallback": {
"description": "useCallback",
"disabled": false,
"label": "useCallback",
"languageIds": [
"javascriptreact",
"javascript",
"typescriptreact",
"typescript"
],
"snippet": "useCallback(() => $TM_SELECTED_TEXT, [])"
},
"reactForwardRef": {
"description": "forwardRef",
"disabled": false,
"label": "forwardRef",
"languageIds": [
"javascriptreact",
"javascript",
"typescriptreact",
"typescript"
],
"snippet": "forwardRef<$1>($TM_SELECTED_TEXT)"
},
"stringify": {
"description": "JSON.stringify( ... )",
"disabled": false,
"label": "JSON.stringify",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact",
"svelte",
"html",
"astro"
],
"snippet": "JSON.stringify($TM_SELECTED_TEXT)$0"
},
"suspense": {
"description": "<Suspense>",
"disabled": false,
"label": "suspense",
"languageIds": ["javascriptreact", "typescriptreact"],
"snippet": "<Suspense fallback={<Loader />}>$TM_SELECTED_TEXT</Suspense>"
},
"svelteIf": {
"description": "{#if}{/if}",
"disabled": false,
"label": "{#if}",
"languageIds": ["svelte"],
"snippet": "{#if $1}$TM_SELECTED_TEXT{/if}"
},
"svelteIfElse": {
"description": "{#if}{:else}{/if}",
"disabled": false,
"label": "{#if}{:else}",
"languageIds": ["svelte"],
"snippet": "{#if $1}$TM_SELECTED_TEXT{:else}$2{/if}"
},
"JSXBox": {
"description": "<Box>",
"disabled": false,
"label": "box",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "<Box>$TM_SELECTED_TEXT</Box>"
},
"JSXContainer": {
"description": "<Container>",
"disabled": false,
"label": "container",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "<Container>$TM_SELECTED_TEXT</Container>"
},
"JSXFlex": {
"description": "<Flex>",
"disabled": false,
"label": "flex",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "<Flex>$TM_SELECTED_TEXT</Flex>"
},
"JSXGrid": {
"description": "<Grid>",
"disabled": false,
"label": "grid",
"languageIds": ["javascriptreact", "typescriptreact", "astro"],
"snippet": "<Grid>$TM_SELECTED_TEXT</Grid>"
},
"TSGeneric": {
"description": "Generic<>",
"disabled": false,
"label": "TS Generic",
"languageIds": ["typescript", "typescriptreact", "astro"],
"snippet": "$1<$TM_SELECTED_TEXT$0>"
},
"rem-polished": {
"description": "${rem($)}",
"disabled": false,
"label": "rem",
"languageIds": [
"javascript",
"javascriptreact",
"typescript",
"typescriptreact"
],
"snippet": "${rem($TM_SELECTED_TEXT)}"
},
"css-var": {
"description": "var($)",
"disabled": false,
"label": "var",
"languageIds": ["css", "postcss", "javascriptreact", "typescriptreact"],
"snippet": "var($TM_SELECTED_TEXT)"
},
"consoletime": {
"description": "console.time($)",
"disabled": false,
"label": "console.time",
"languageIds": ["css", "postcss", "javascriptreact", "typescriptreact"],
"snippet": "console.time($0)\n$TM_SELECTED_TEXT\nconsole.timeEnd()"
}
}
}
Browser
Arc by The Browser Company. Arc has been my main browser for a while now. It is a fantastic Chromium-based browser so all my favourite Chrome extensions still work.
Other apps
Raycast
I use it in nearly every aspect of my work. I use it for searching the web with its custom web searches (e.g. npm, JIRA, bundlephobia, MDN and many more); searching npm with the npm extension (that I created) workflow and loads more. It truly is a massive productivity booster.
FYI I also created the GitHub Stars extension.
- Toothpick - Manage Bluetooth connections
- JIRA - Interact with JIRA
- Set Audio Device - Set the input/output audio device
1Password
I never need to worry about passwords anymore. It integrates with my browser (via their extension) and it copies one-time passwords to the clipboard automatically. It’s also great on the iPhone; since iOS 12 was released, the integration with the OS is massively improved.
DevUtils
This app is a recent addition and is an indispensible tool for my workflow.
Notion
I moved to Notion a while back and haven’t looked back. It has most of what I want for a note-taking app with a healthy free-tier and Markdown-like input method.
iTerm2 with Fish Shell
To improve my shell, I use the excellent TIDE prompt and Fig which adds IDE-style autocomplete to my terminal.
I also use a few other CLI apps to make the experience better, in particular, ni by Anthony Fu. It is great for installing npm packages and running npm scripts. I bounce from project to project a lot and they often use different package managers. This tool makes it easy to install packages and run scripts without having to remember which package manager to use.
Tower
An excellent git client for Mac.
NextDNS
A cloud-based private DNS service that gives you full control over what is allowed and what is blocked on the Internet. There are Mac and iPhone apps that essentially block ads, malicious code and more.
And the rest:
- Things: The best Mac/iPhone-based task list/productivity app I know.
- Superkey: “Keyboard superpower on macOS”. I use this mainly for its “hyperkey” functionality (where caps lock is converted to a “hyper key”, with all these four modifiers combined:
⌃⌥⌘⇧
) - Rectangle Pro: Window snapping and positioning
- Vivid - Double brightness system-wide
- Dash: Dash is an API Documentation Browser and Code Snippet Manager. Devdocs.io or the Devdocs app are free alternatives.
- NordVPN: The best personal VPN service in my opinion. Great apps for Mac and iPhone.
- Bartender: Hide some of those pesky menubar items.
- Shottr - Shottr is a small, fast, human-sized screenshot app built for those who care about pixels. It was crafted with Swift, optimized for M1, and is completely free at the moment.
- AudioSwitcher - AudioSwitcher is a menu bar that allows you to switch quickly between all available input and output devices.
This page is featured on uses.tech. If you’re reading this and want to see others like it, head on over there to find out more.
You can also see this list on Kit.co