Skip to content

Latest commit

 

History

History
212 lines (163 loc) · 15 KB

README.md

File metadata and controls

212 lines (163 loc) · 15 KB

OSX Vim + Tmux Awesomness

Introduction

This repo contains everything you need to get set up with a beautiful and highly productive vim + tmux environment on OSX (linux guide coming soon!) ... I use this exact setup daily as a Site Reliability Engineer at Skuid, writing Go, making quick bash scripts, troubleshooting kubernetes, etc. At home, I'm learning Rust. There are lots of similar guides out there, but none of them delivered the exact experience I was looking for.

Features

  • Everything is accomplished with sane keyboard shortcuts.
  • Full mouse support within Vim and tmux.
  • Vim/tmux/system clipboard sync! Copy stackoverflow code, paste it into vim, copy the compile errors from tmux, paste em back into stackoverflow, repeat! You're a developer now! :)
  • Distraction free workflow. Build, run, and test your code in an adjacent tmux pane without losing focus of Vim!
  • Impress your friends and coworkers with your 1337 h4x0r set up!
  • Code completion! Just like a real™ IDE!
  • Persist both Vim and tmux sessions across reboots! Have both saved automatically!

Cons

  • No debugging (yet)
  • Steep learning curve

Terminal Emulator

While iTerm2 is generally an excellent choice for a terminal emulator on OSX, I wanted something faster. This guide will work on iTerm2 (and probably the default OSX terminal), but I prefer to use kitty. It uses the GPU for rendering, supports images, and lots of other goodies. It also supports startup sessions. Though not required, I would highly recommend checking it out. We won't be using any of the tab or window features of kitty (kind of defeats the purpose of tmux lol), but some folks appreciate that sort of thing. If you intend to fully embrace tmux and thus do not need tabs or tiles, then alacritty is a great choice as well. It doesn't support tabs or panes and assumes the user will use tmux if that sort of thing is required. It is even more performant that kitty.

The main reasons to use kitty or alacritty over iTerm2 are GPU acceleration and the lightweight footprint it offers. These days, soooo many desktop applications are Electron based: slack, spotify, etc. I love the vast array of beautiful desktop applications that Electron has produced, but many end slow and memory intensive. And more powerful IDEs like JetBrains' offerings are even worse as far as resource consumption goes.

Throw in running dozens of docker containers in the background and keeping tons of firefox or chrome tabs open, my poor little MacBook just can't keep up. This is what started me on my journey to simplify my development and work environment.

Setup up Vim

MacVim

In order to take advantage of the excellent vim plugin YouCompleteMe for code completion, we need to install MacVim. If you don't want code completion, or already have a completion plugin you like, then feel free to skip this step. I've used the rest of the plugins with OSX default vim without issue.

$ echo "export PATH=\"/usr/local/sbin:$PATH\" >> ~/.bash_profile # ensure homebrew bin is in your $PATH
$ source ~/.bash_profile

Ensure homebrew vim is in your path:

/usr/local/bin/vim   # this is the default vim from OSX, not what we want

$ brew info vim
vim: stable 8.1.1100 (bottled), HEAD
Vi 'workalike' with many additional features
https://www.vim.org/
Conflicts with:
  ex-vi (because vim and ex-vi both install bin/ex and bin/view)
  macvim (because vim and macvim both install vi* binaries)
/usr/local/Cellar/vim/8.1.1100 (1,845 files, 30.9MB) *
... 

/usr/local/Cellar/vim/8.1.1100 that's the path we're looking for. Now we create a symlink to get the homebrew vim in our path. You could put this symlink anywhere in your path that's before the system default vim, but homebrew will put other symlinks at /usr/local/sbin so we may as well put it there.

$ ln -s /usr/local/Cellar/vim/8.1.1100/vim /usr/local/sbin/vim

Clone my vimrc

I don't recommned cloning this repo directly into your ~/ directory (though that would work). It's generally best practice to keep your dotfiles in separate directories, then symlink them in to place. This can be done manually as in the example below, but I recommend you checkout GNU Stow for this task later on.

$ mkdir ~/git
$ cd ~/git
$ git clone https://github.com/jamesbaker1901/vim-tmux-awesomeness.git

# make a backup of your current .vimrc
$ mv ~/.vimrc ~/.vimrc.bak
$ ln -s $HOME/git/vim-tmux-awesomeness/vimrc $HOME/.vimrc

SetUp Vim Plugins

Install Vundle

Vundle is a vim plugin manager. Let's install it first:

$ git clone https://github.com/VundleVim/Vundle.vim.git ~/.vim/bundle/Vundle.vim

Open vim and run :PluginInstall

$ vim
:PluginInstall
:q
:q

That's it! All of the plugins should install without issue. If anything fails or complains, just try googling the error.

Set up YouCompleteMe

This is such a great plugin. I've tinkered with most of the completion plugins for Vim and YouCompleteMe is the best. If there's one you like better, then feel free to skip the MacVim installation above (it's a requirement for YCM and the only reason I bother with it). Detailed instructions are here: github.com/Valloric/YouCompleteMe, but we since we set up homebrew vim, we'll just go straight to the installation.

$ cd ~/.vim/bundle/YouCompleteMe 
$ ./install.py --all

Passing --all will have the isntaller compile completion binaries for C#, Go, JavaScript and TypeScript, Rust, and Java. If you only want support for just one, then head here: github.com/Valloric/YouCompleteMe#mac-os-x and pick out the individual flags for just the language(s) you want and replace --all with them.

Vim Theme

Super important to pick a good theme. If you have a popular one from VS code or another editor that you like, it probably exists for Vim as well. This site, vimcolors.com has lots of themes to browse and search through with previews. Many themes can be installed with the Vundle plugin manager. Otherwise, you typically just clone the theme repo and copy/symlink the colors/theme-name.vim file to your ~/.vim/colors

More on Plugins

I use quite a few plugins, but all of them are there because I use them. Always make time to prune your plugins, lest your ~/.vimrc become unruly. If you start having problems, you'll be hard pressed to track it down to a single config line or plugin when you have dozens of them. I write Go and Rust, so a lot of these plugins are there to support that work. If you use python or another language, there are plenty of guides to help you get a more fine grained setup for your language of choice. Just google vim <language> setup and you'll get plenty of inspiration.

This repo is not meant to be an "all in one" solution like Oh My Tmux! is. It's geared specifically toward Go and Rust development, and I find it useful for general kubernetes maintenance and sysadmin duties as well. If that's your use case, then you may not need to change anything, but I highly recommend that after a few days or weeks of use with my setup, you prune out any plugins you find yourself not using (you can always install them again if you need to).

Set up tmux

$ brew install tmux

Oh My Tmux!

Oh My Tmux! is a really well done and thoughtout tmux config with lots of goodies set up for you. You could of course build your own config from scratch or clone any number of other tmux configs. If you don't care for the powerline look or some of the config decisions the Oh My Tmux! maintainer made, then definitely shop around! Or, do like I did and fork his repository and use it as the base of your own, superior config :)

Before moving forward, I recommend reading over the extensive guide that gpakosz wrote for his tmux config. I did some basic tweaks, but left his original work largely intact. I won't cover any cosmetic changes here, only utilities that we'll need later to replicate my workflow.

OSX Clipboard within TMUX sessions

To access the OSX systemc clipboard from within tmux, we'll need this helpful little utility: tmux-MacOSX-pasteboard.

Install it with homebrew:

$ brew install reattach-to-user-namespace

Powerline Symbols

If you enjoy the Powerline look, then you'll need either the standalone Powerline font, a font that already has them, or a pre-patched font. Read this for more info.

Tmux Plugins

Tmux with Oh My Tmux! is already pretty powerful, but we can go further! First, we need to install Tmux Plugin Manager:

$ git clone https://github.com/tmux-plugins/tpm ~/.tmux/plugins/tpm

Then dd this to the bottom of your ~/.tmux.conf.local file, but using the Oh My Tmux! shortcut for editing the tmux config file: <CTRL+e>. It will open a new tmux window with whatever is set in your $EDITOR env variable, though it defaults to vim.

# clipboard sync
if-shell -b 'test $(uname) = "Linux"' 'bind -T copy-mode-vi y send-keys -X copy-pipe-and-cancel "xclip -in -selection clipboard"'
if-shell -b 'test $(uname) = "Darwin"' 'bind -T copy-mode-vi y send-keys -X copy-pipe-and-cancel "pbcopy"'

# List of plugins
set -g @plugin 'tmux-plugins/tpm'
set -g @plugin 'tmux-plugins/tmux-sensible'
set -g @plugin 'tmux-plugins/tmux-resurrect'
set -g @plugin 'tmux-plugins/tmux-continuum'

# Configure plugins
set -g @continuum-restore 'on' #auto restore last saved env
set -g @resurrect-strategy-vim 'session' #auto restore vim sessions
set -g focus-events on #required for vim focus events to work

# Initialize TMUX plugin manager (keep this line at the very bottom of tmux.conf)
run -b '~/.tmux/plugins/tpm/tpm'

Workflow

Now we get to the good stuff. How do you actually get work done without an IDE? Without a mouse?! Once you've learned the keyboard shortcuts, it's very efficient, fast and easy. The learning curve will be steep at first, but eventually these shortcuts will become second nature to you and you won't even have to think about it. The same is true of any vim, tmux set up of course, not just this one.

The benefit to this particular set up, is that you won't have to leave vim (or lose focus) to compile your code, run tests, run your binaries, etc. Because we're using tmux and the lovely vim-slime plugin, we can send commands to another tmux pane without leaving vim!

Shortcuts

This set up is all about the shortcuts ... one of the main reasons to adopt a terminal only, vim+tmux development environment is the efficiency gain from being able to keep your findgers not just on the keyboard, but on the home keys as much as possible. The basic vim keybindings do a great job of this already, and this setup tries to keep true to vim conventions as much as possible, while adding additional functionality. We won't cover the basics of getting around vim, just the changes I've made in my ~/.vimrc.

Something to consider is the <leader> key. By default it's \\, but I've remapped it to , with let mapleader = "," since I find it to be more convenient to use. Feel free to keep the default or choose your own <leader> key.

Vim

Insert mode:

  • ;; | <Esc> - exit INSERT mode. This one is super useful as exiting INSERT mode happens all the time, and now you can do it with a home key! If you actually need to type two semi-colons, just do it a bit more slowly.

Normal mode:

  • ; | :w - Write the current buffer. This allows you to press ;;; to exit insert mode and save the current buffer. An extremely common vim operation, and now we don't even need to leave to home keys for it!
  • ,q :q - Close vim.
  • <CTRL>+j | :bd - Next buffer. Using j and k to navigate buffers, keeps with the vim convention for moving around.
  • <CTRL>+k | :bp - Previous buffer.
  • <CTRL>+d | :bd - Close buffer. just like <CTRL>+d closes a shell in bash.
  • <RightMouse> - Exits and enters INSERT mode. If you happen to have your hand on the mouse, right click will cycle through INSERT/NORMAL mode.

Plugin specific remaps:

  • <CTRL>+n - Open or close the NERDTree file browser.
  • <CTRL>+ww - Cycle trhough open windows. If you have NERDTree open, this will bring you back to the main file you're editing.

Vim+Tmux remaps:

  • <CTRL>+u - Uses vim-slime to send a build command to the right pane, based on the file type, i.e., while editing a *.go file, go build will be sent to the right pane. For rust (*.rs), cargo build will be sent.
  • <CTRL>+r - Uses vim-slime to send the build command for the language, and also runs ./.run.sh This way, you can just add a ./.run.sh hidden file to your project with the command to run your compiled binary, and any extra arguments it needs. These might change between projects. Of course, this is just one way to handle this. You can modify the remap to run anything you like (use make, etc.).
  • <CTRL>+t - Uses vim-slime to send the test command for the language.
  • <CTRL>+ll - Uses vim-slime to send ll to the right pane. ll is a common alias for ls -alF on linux distributions. Personally I use the beautiful exa as an ls replacement, so I have ll alias'd to exa -bghHliS.
  • <CTRL>+c - Uses tmux send-keys to send <CTRL>-c to the right pane. Generally when coding, I have vim in a left pane and a few panes on the right to run my compiled code, run tests, etc. If you're tmxu layout is different, you'll need to edit this line: nnoremap <buffer> <leader>c :!tmux send-keys -t right C-c<CR><CR>. The vim pane will flicker when you run this. That's normal. Basically, vim has to pause to run a terminal command.

The above maps really represent the meat of this workflow. It's possible of course to achieve the same basic effect from simply using tmux shortcuts to select another pane in which to compile and test your code. With mouse support on, you can even click on another pane or window to build your code, but using tmux takes you out of vim and breaks your flow and reaching for your mouse is even worse.

Now say we're editing a Go file. Our ~/.vimrc automatically adjusts the above shortcuts to map to go build etc. We make some changes and want to make sure they compile. That will look something like this:

;;; # to exit insert mode and save our file.
<CTRL+b> # to send 'go build' to the tmux pane to our right

If the compiler gives us errors we can immediately jump to the line referenced by the compiler to fix our mistake.

Tmux

I didn't really mess with the tmux bindings set up by the Oh My Tmux! author, you can view them here: github.com/gpakosz/.tmux#bindings.