From Zero to Swift | Configuring Neovim for Swift Development
Neovim is a modern reimplementation of Vim, a popular terminal-based text editor. Neovim adds new features like asynchronous operations and powerful Lua bindings for a snappy editing experience, in addition to the improvements Vim brings to the original Vi editor.
This article walks you through configuring Neovim for Swift development, providing configurations for various plugins to build a working Swift editing experience. It is not a tutorial on how to use Neovim and assumes some familiarity with modal text editors like Neovim, Vim, or Vi. We are also assuming that you have already installed a Swift toolchain on your computer. If not, please see the Swift installation instructions.
Although the article references Ubuntu 22.04, the configuration itself works on any operating system where a recent version of Neovim and a Swift toolchain is available.
Basic setup and configuration includes:
- Installing Neovim.
- Installing
lazy.nvim
to manage our plugins. - Configuring the SourceKit-LSP server.
- Setting up Language-Server-driven autocompletion with nvim-cmp.
- Setting up snippets with LuaSnip.
The following sections are provided to help guide you through the setup:
Tip: If you already have Neovim, Swift, and a package manager installed, you can skip to setting up Language Server support.
Note: If you are bypassing the Prerequisites section, make sure your copy of Neovim is version v0.9.4 or higher, or you may experience issues with some of the Language Server Protocol (LSP) Lua APIs.
Prerequisites
To get started, you’ll need to install Neovim. The Lua APIs exposed by Neovim are under rapid development. We will want to take advantage of the recent improvements in the integrated support for Language Server Protocol (LSP), so we will need a fairly recent version of Neovim.
I’m running Ubuntu 22.04 on an x86_64
machine. Unfortunately, the
version of Neovim shipped in the Ubuntu 22.04 apt
repository is too old to
support many of the APIs that we will be using.
For this install, I used snap
to install Neovim v0.9.4.
Ubuntu 24.04 has a new enough version of Neovim, so a normal
apt install neovim
invocation will work.
For installing Neovim on other operating systems and Linux distributions,
please see the
Neovim install page.
$ sudo snap install nvim --classic
$ nvim --version
NVIM v0.9.4
Build type: RelWithDebInfo
LuaJIT 2.1.1692716794
Compilation: /usr/bin/cc -O2 -g -Og -g -Wall -Wextra -pedantic -Wno-unused-pa...
system vimrc file: "$VIM/sysinit.vim"
fall-back for $VIM: "/usr/share/nvim"
Run :checkhealth for more info
Getting Started
We have working copies of Neovim and Swift on our path. While we can start with
a vimrc
file, Neovim is transitioning from using vimscript to Lua. Lua
is easier to find documentation for since it’s an actual programming language,
tends to run faster, and pulls your configuration out of the main runloop so
your editor stays nice and snappy.
You can still use a vimrc
with vimscript, but we’ll use Lua.
The main Neovim configuration file goes in ~/.config/nvim
. The other Lua files
go in ~/.config/nvim/lua
. Go ahead and create an init.lua
now;
$ mkdir -p ~/.config/nvim/lua && cd ~/.config/nvim
$ nvim init.lua
Note: The examples below contain a GitHub link to the plugin to help you readily access the documentation. You can also explore the plugin itself.
Packaging with lazy.nvim
While it’s possible to set everything up manually, using a package manager helps keep your packages up-to-date, and ensures that everything is installed correctly when copy your configuration to a new computer. Neovim also has a built-in plugin manager, but I have found lazy.nvim to work well.
We will start with a little bootstrapping script to install lazy.nvim if it isn’t installed already, add it to our runtime path, and finally configure our packages.
At the top of your init.lua
write:
local lazypath = vim.fn.stdpath("data") .. "/lazy/lazy.nvim"
if not vim.loop.fs_stat(lazypath) then
vim.fn.system({
"git",
"clone",
"--filter=blob:none",
"https://github.com/folke/lazy.nvim.git",
"--branch=stable",
lazypath
})
end
vim.opt.rtp:prepend(lazypath)
This snippet clones lazy.nvim if it doesn’t already exist, and then adds it to the runtime path. Now we initialize lazy.nvim and tell it where to look for the plugin specs.
require("lazy").setup("plugins")
This configures lazy.nvim to look in a plugins/
directory under our lua/
directory for each plugin. We’ll also want a place to put our own non-plugin
related configurations, so we’ll stick it in config/
. Go ahead and create
those directories now.
$ mkdir lua/plugins lua/config
See lazy.nvim Configuration for details on configuring lazy.nvim.
Note that your configuration won’t look exactly like this. We have only installed lazy.nvim, so that is the only plugin that is listed on your configuration at the moment. That’s not very exciting to look at, so I’ve added a few additional plugins to make it look more appealing.
To check that it’s working:
-
Launch Neovim.
You should first see an error saying that there were no specs found for module plugins. This just means that there aren’t any plugins.
-
Press Enter and type,
:Lazy
.lazy.nvim lists the plugins installed. There should only be one right now: “lazy.nvim”. This is lazy.nvim tracking and updating itself.
-
We can manage our plugins through the lazy.nvim menu.
- Pressing
I
will install new plugins. - Pressing
U
will update installed plugins. - Pressing
X
will delete any plugins that lazy.nvim installed, but are no longer tracked in your configuration.
- Pressing
Language Server Support
Language servers respond to editor requests providing language-specific support. Neovim has support for Language Server Protocol (LSP) built-in, so you don’t need an external package for LSP, but adding a configuration for each LSP server manually is a lot of work. Neovim has a package for configuring LSP servers, nvim-lspconfig.
Go ahead and create a new file under lua/plugins/lsp.lua
. In it, we’ll start
by adding the following snippet.
return {
{
"neovim/nvim-lspconfig",
config = function()
local lspconfig = require('lspconfig')
lspconfig.sourcekit.setup {}
end,
}
}
While this gives us LSP support through sourcekit-lsp, there are no keybindings, so it’s not very practical. Lets hook those up now.
We’ll set up an auto command that fires when LSP attaches in the config
function under where we set up the sourcekit
server. The keybindings are
applied to all LSP servers so you end up with a consistent experience across
languages.
config = function()
local lspconfig = require('lspconfig')
lspconfig.sourcekit.setup {}
vim.api.nvim_create_autocmd('LspAttach', {
desc = 'LSP Actions',
callback = function(args)
vim.keymap.set('n', 'K', vim.lsp.buf.hover, {noremap = true, silent = true})
vim.keymap.set('n', 'gd', vim.lsp.buf.definition, {noremap = true, silent = true})
end,
})
end,
I’ve created a little example Swift package that computes Fibonacci
numbers asynchronously.
Pressing shift
+ k
on one of the references to the fibonacci
function
shows the documentation for that function, along with the function signature.
The LSP integration is also showing that we have an error in the code.
File Updating
SourceKit-LSP increasingly relies on the editor informing the server when certain files change. This need is communicated through dynamic registration. You don’t have to understand what that means, but Neovim doesn’t implement dynamic registration. You’ll notice this when you update your package manifest, or add new files to your compile-commands file and LSP doesn’t work without restarting Neovim.
Instead, we know that SourceKit-LSP needs this functionality, so we’ll enable it
statically. We’ll update our sourcekit
setup configuration to manually set the
didChangeWatchedFiles
capability.
lspconfig.sourcekit.setup {
capabilities = {
workspace = {
didChangeWatchedFiles = {
dynamicRegistration = true,
},
},
},
}
If you’re interested in reading more about this issue, the conversations in the following issues describe the issue in more detail:
Auto Complete
We will use nvim-cmp to act as the autocomplete mechanism. We’ll start by telling lazy.nvim to download the package and to load it lazily when we enter insert mode since you don’t need autocompletion if you’re not editing the file.
-- lua/plugins/autocomplete.lua
return {
{
"hrsh7th/nvim-cmp",
version = false,
event = "InsertEnter",
},
}
Next, we’ll configure some completion sources to provide autocompletion results. nvim-cmp doesn’t come with completion sources, those are additional plugins. For this configuration, I want results based on LSP, filepath completion, and the text in my current buffer. For more, the nvim-cmp Wiki has a list of sources.
To start, we will tell lazy.nvim about the new plugins and that nvim-cmp depends on them. This ensures that lazy.nvim will initialize each of them when nvim-cmp is loaded.
-- lua/plugins/autocomplete.lua
return {
{
"hrsh7th/nvim-cmp",
version = false,
event = "InsertEnter",
dependencies = {
"hrsh7th/cmp-nvim-lsp",
"hrsh7th/cmp-path",
"hrsh7th/cmp-buffer",
},
},
{ "hrsh7th/cmp-nvim-lsp", lazy = true },
{ "hrsh7th/cmp-path", lazy = true },
{ "hrsh7th/cmp-buffer", lazy = true },
}
Now we need to configure nvim-cmp to take advantage of the auto-completion sources. Unlike many other plugins, nvim-cmp hides many of its inner-workings, so configuring it is a little different from other plugins. Specifically, you’ll notice the differences around setting key-bindings. We start out by requiring the module from within its own configuration function and will call the setup function explicitly.
{
"hrsh7th/nvim-cmp",
version = false,
event = "InsertEnter",
dependencies = {
"hrsh7th/cmp-nvim-lsp",
"hrsh7th/cmp-path",
"hrsh7th/cmp-buffer",
},
config = function()
local cmp = require('cmp')
local opts = {
-- Where to get completion results from
sources = cmp.config.sources {
{ name = "nvim_lsp" },
{ name = "buffer"},
{ name = "path" },
},
-- Make 'enter' key select the completion
mapping = cmp.mapping.preset.insert({
["<CR>"] = cmp.mapping.confirm({ select = true })
}),
}
cmp.setup(opts)
end,
},
Using the tab
key to select completions is a fairly popular option, so we’ll
go ahead and set that up now.
mapping = cmp.mapping.preset.insert({
["<CR>"] = cmp.mapping.confirm({ select = true }),
["<tab>"] = cmp.mapping(function(original)
if cmp.visible() then
cmp.select_next_item() -- run completion selection if completing
else
original() -- run the original behavior if not completing
end
end, {"i", "s"}),
["<S-tab>"] = cmp.mapping(function(original)
if cmp.visual() then
cmp.select_prev_item()
else
original()
end
end, {"i", "s"}),
}),
Pressing tab
while the completion menu is visible will select the next
completion and shift
+ tab
will select the previous item. The tab behavior
falls back on whatever pre-defined behavior was there originally if the menu
isn’t visible.
Snippets
Snippets are a great way to improve your workflow by expanding short pieces of text into anything you like. Lets hook those up now. We’ll use LuaSnip as our snippet plugin.
Create a new file in your plugins directory for configuring the snippet plugin.
-- lua/plugins/snippets.lua
return {
{
'L3MON4D3/LuaSnip',
conifg = function(opts)
require('luasnip').setup(opts)
require('luasnip.loaders.from_snipmate').load({ paths = "./snippets" })
end,
},
}
Now we’ll wire the snippet expansions into nvim-cmp. First, we’ll add LuaSnip as a dependency of nvim-cmp to ensure that it gets loaded before nvim-cmp. Then we’ll wire it into the tab key expansion behavior.
{
"hrsh7th/nvim-cmp",
version = false,
event = "InsertEnter",
dependencies = {
"hrsh7th/cmp-nvim-lsp",
"hrsh7th/cmp-path",
"hrsh7th/cmp-buffer",
"L3MON4D3/LuaSnip",
},
config = function()
local cmp = require('cmp')
local luasnip = require('cmp')
local opts = {
-- Where to get completion results from
sources = cmp.config.sources {
{ name = "nvim_lsp" },
{ name = "buffer"},
{ name = "path" },
},
mapping = cmp.mapping.preset.insert({
-- Make 'enter' key select the completion
["<CR>"] = cmp.mapping.confirm({ select = true }),
-- Super-tab behavior
["<tab>"] = cmp.mapping(function(original)
if cmp.visible() then
cmp.select_next_item() -- run completion selection if completing
elseif luasnip.expand_or_jumpable() then
luasnip.expand_or_jump() -- expand snippets
else
original() -- run the original behavior if not completing
end
end, {"i", "s"}),
["<S-tab>"] = cmp.mapping(function(original)
if cmp.visual() then
cmp.select_prev_item()
elseif luasnip.expand_or_jumpable() then
luasnip.jump(-1)
else
original()
end
end, {"i", "s"}),
}),
snippets = {
expand = function(args)
luasnip.lsp_expand(args)
end,
},
}
cmp.setup(opts)
end,
},
Now our tab-key is thoroughly overloaded in super-tab fashion.
- If the completion window is open, pressing tab selects the next item in the list.
- If you press tab over a snippet, the snippet will expand, and continuing to press tab moves the cursor to the next selection point.
- If you’re neither auto-completing nor expanding a snippet, it will behave
like a normal
tab
key.
Now we need to write up some snippets. LuaSnip supports several snippet formats, including a subset of the popular TextMate, Visual Studio Code snippet format, its own Lua-based API, and snippets coming from an LSP server.
Here are some snippets that I’ve found to be useful:
snippet pub "public access control"
public $0
snippet priv "private access control"
private $0
snippet if "if statement"
if $1 {
$2
}$0
snippet ifl "if let"
if let $1 = ${2:$1} {
$3
}$0
snippet ifcl "if case let"
if case let $1 = ${2:$1} {
$3
}$0
snippet func "function declaration"
func $1($2) $3{
$0
}
snippet funca "async function declaration"
func $1($2) async $3{
$0
}
snippet guard
guard $1 else {
$2
}$0
snippet guardl
guard let $1 else {
$2
}$0
snippet main
@main public struct ${1:App} {
public static func main() {
$2
}
}$0
Another popular snippet plugin worth mentioning is UltiSnips which allows you to use inline Python while defining the snippet, allowing you to write some very powerful snippets.
Conclusion
Swift development with Neovim is a solid experience once everything is configured correctly. There are thousands of plugins for you to explore, this article gives you a solid foundation for building up your Swift development experience in Neovim.
Files
Here are the files for this configuration in their final form.
-- init.lua
local lazypath = vim.fn.stdpath("data") .. "/lazy/lazy.nvim"
if not vim.loop.fs_stat(lazypath) then
vim.fn.system({
"git",
"clone",
"--filter=blob:none",
"https://github.com/folke/lazy.nvim.git",
"--branch=stable",
lazypath
})
end
vim.opt.rtp:prepend(lazypath)
require("lazy").setup("plugins", {
ui = {
icons = {
cmd = "",
config = "",
event = "",
ft = "",
init = "",
keys = "",
plugin = "",
runtime = "",
require = "",
source = "",
start = "",
task = "",
lazy = "",
},
},
})
vim.opt.wildmenu = true
vim.opt.wildmode = "list:longest,list:full" -- don't insert, show options
-- line numbers
vim.opt.nu = true
vim.opt.rnu = true
-- textwrap at 80 cols
vim.opt.tw = 80
-- set solarized colorscheme.
-- NOTE: Uncomment this if you have installed solarized, otherwise you'll see
-- errors.
-- vim.cmd.background = "dark"
-- vim.cmd.colorscheme("solarized")
-- vim.api.nvim_set_hl(0, "NormalFloat", { bg = "none" })
-- lua/plugins/autocomplete.lua
return {
{
"hrsh7th/nvim-cmp",
version = false,
event = "InsertEnter",
dependencies = {
"hrsh7th/cmp-nvim-lsp",
"hrsh7th/cmp-path",
"hrsh7th/cmp-buffer",
},
config = function()
local cmp = require('cmp')
local luasnip = require('luasnip')
local opts = {
sources = cmp.config.sources {
{ name = "nvim_lsp", },
{ name = "path", },
{ name = "buffer", },
},
mapping = cmp.mapping.preset.insert({
["<CR>"] = cmp.mapping.confirm({ select = true }),
["<tab>"] = cmp.mapping(function(original)
print("tab pressed")
if cmp.visible() then
print("cmp expand")
cmp.select_next_item()
elseif luasnip.expand_or_jumpable() then
print("snippet expand")
luasnip.expand_or_jump()
else
print("fallback")
original()
end
end, {"i", "s"}),
["<S-tab>"] = cmp.mapping(function(original)
if cmp.visible() then
cmp.select_prev_item()
elseif luasnip.expand_or_jumpable() then
luasnip.jump(-1)
else
original()
end
end, {"i", "s"}),
})
}
cmp.setup(opts)
end,
},
{ "hrsh7th/cmp-nvim-lsp", lazy = true },
{ "hrsh7th/cmp-path", lazy = true },
{ "hrsh7th/cmp-buffer", lazy = true },
}
-- lua/plugins/lsp.lua
return {
{
"neovim/nvim-lspconfig",
config = function()
local lspconfig = require('lspconfig')
lspconfig.sourcekit.setup {
capabilities = {
workspace = {
didChangeWatchedFiles = {
dynamicRegistration = true,
},
},
},
}
vim.api.nvim_create_autocmd('LspAttach', {
desc = "LSP Actions",
callback = function(args)
vim.keymap.set("n", "K", vim.lsp.buf.hover, {noremap = true, silent = true})
vim.keymap.set("n", "gd", vim.lsp.buf.definition, {noremap = true, silent = true})
end,
})
end,
},
}
-- lua/plugins/snippets.lua
return {
{
'L3MON4D3/LuaSnip',
lazy = false,
config = function(opts)
local luasnip = require('luasnip')
luasnip.setup(opts)
require('luasnip.loaders.from_snipmate').load({ paths = "./snippets"})
end,
}
}
# snippets/swift.snippets
snippet pub "public access control"
public $0
snippet priv "private access control"
private $0
snippet if "if statement"
if $1 {
$2
}$0
snippet ifl "if let"
if let $1 = ${2:$1} {
$3
}$0
snippet ifcl "if case let"
if case let $1 = ${2:$1} {
$3
}$0
snippet func "function declaration"
func $1($2) $3{
$0
}
snippet funca "async function declaration"
func $1($2) async $3{
$0
}
snippet guard
guard $1 else {
$2
}$0
snippet guardl
guard let $1 else {
$2
}$0
snippet main
@main public struct ${1:App} {
public static func main() {
$2
}
}$0