Dotfiles/nvim/init.lua

322 lines
16 KiB
Lua

-- ── Bootstrap lazy.nvim ───────────────────────────────────────────────────────
-- lazy.nvim is the plugin manager. On first launch it clones itself into the
-- Neovim data directory (~/.local/share/nvim/lazy/lazy.nvim).
-- vim.uv is the preferred libuv binding in Neovim >= 0.10; vim.loop is the
-- legacy alias kept for compatibility with older releases.
local lazypath = vim.fn.stdpath("data") .. "/lazy/lazy.nvim"
if not (vim.uv or vim.loop).fs_stat(lazypath) then
vim.fn.system({
"git", "clone",
"--filter=blob:none", -- blobless clone: skips file blobs, fetching only on checkout
"https://github.com/folke/lazy.nvim.git",
"--branch=stable", lazypath,
})
end
-- Prepend lazy's path so Neovim can find and load it before any other plugin.
vim.opt.rtp:prepend(lazypath)
-- ── Plugins ───────────────────────────────────────────────────────────────────
require("lazy").setup({
-- CyberQueer colorscheme loaded from the local dotfiles tree (not from GitHub)
-- so that apply-theme.sh changes are picked up without a plugin update.
{ dir = vim.fn.expand("~/Dotfiles/nvim/theme/cyberqueer.nvim") },
"rktjmp/lush.nvim", -- HSL-based colorscheme builder used by cyberqueer
"tpope/vim-sensible", -- Sensible Vim defaults (better backspace, history, etc.)
"junegunn/goyo.vim", -- Distraction-free writing mode (centers and pads the buffer)
"arecarn/vim-crunch", -- Evaluate math expressions inside the buffer
"preservim/nerdtree", -- File-tree sidebar navigator
"ryanoasis/vim-devicons", -- Nerd Font icons for NERDTree and airline
-- fzf binary + Vim integration; build step runs fzf#install() to compile the binary
{ "junegunn/fzf", build = function() vim.fn["fzf#install"]() end },
"junegunn/fzf.vim", -- :Files, :Rg, :Buffers etc. powered by fzf
"vim-airline/vim-airline", -- Status/tabline with Powerline-style separators
"vim-airline/vim-airline-themes", -- Theme library for vim-airline (cyberqueer theme used)
"voldikss/vim-floaterm", -- Floating terminal windows inside Neovim
"rust-lang/rust.vim", -- Rust syntax, fmt-on-save, and cargo integration
"norcalli/nvim-colorizer.lua", -- Inline hex/RGB color previews in CSS, config files, etc.
-- CoC: full LSP client with completion, hover, diagnostics; pinned to "release" branch
{ "neoclide/coc.nvim", branch = "release" },
-- vim-visual-multi: multiple-cursor editing (like Sublime Text ctrl+d)
{ "mg979/vim-visual-multi", branch = "master" },
"SirVer/ultisnips", -- Snippet engine (expand/jump triggers configured below)
"honza/vim-snippets", -- Community snippet library consumed by UltiSnips
"mfussenegger/nvim-dap", -- Debug Adapter Protocol client for step-through debugging
"elihunter173/dirbuf.nvim", -- Edit directory contents like a buffer (rename, delete, etc.)
"tpope/vim-dadbod", -- Database client (SQL queries against any DB URL)
"kristijanhusak/vim-dadbod-ui", -- TUI for vim-dadbod (tree browser + result panes)
"kristijanhusak/vim-dadbod-completion", -- CoC/native completion source for SQL via dadbod
"nvim-mini/mini.icons", -- Minimal icon provider used by dadbod-ui and others
"tadmccorkle/markdown.nvim", -- Markdown helpers: table formatting, checkboxes, etc.
-- glow.nvim: render Markdown in a floating window via the `glow` CLI
{ "ellisonleao/glow.nvim", config = true },
"itchyny/calendar.vim", -- Interactive calendar; used in the PIM overlay (key "g")
-- claude-code.nvim: Claude Code integration (opens/communicates with the `claude` CLI)
{
"greggh/claude-code.nvim",
dependencies = { "nvim-lua/plenary.nvim" },
config = function()
require("claude-code").setup()
end,
},
}, {
-- Use a bundled colorscheme during the initial lazy install so the UI is
-- readable even before cyberqueer is downloaded and compiled.
install = { colorscheme = { "habamax" } },
})
-- ── Colorscheme & UI ──────────────────────────────────────────────────────────
vim.cmd("colorscheme cyberqueer")
-- Tell airline to use Powerline-patched glyphs for segment separators.
vim.g.airline_powerline_fonts = 1
vim.g.airline_theme = "cyberqueer"
-- Embed machine identity in the airline status bar (section_x = right side).
-- Useful when SSHed into multiple machines or running nested Neovim sessions.
local ipaddr = vim.trim(vim.fn.system("hostname -i"))
local hostname = vim.trim(vim.fn.system("hostname -s"))
vim.g.airline_section_x = "IP:" .. ipaddr .. " DNS:" .. hostname
-- ── Providers ─────────────────────────────────────────────────────────────────
-- Disable unused remote plugin providers to suppress ":checkhealth" warnings
-- and avoid the startup penalty of probing for ruby/perl interpreters.
vim.g.loaded_ruby_provider = 0
vim.g.loaded_perl_provider = 0
-- ── Editor options ────────────────────────────────────────────────────────────
-- Enable filetype-specific plugins and indentation rules (e.g. Python 4-space).
vim.cmd("filetype plugin indent on")
vim.cmd("syntax enable")
vim.opt.number = true -- show absolute line numbers in the gutter
vim.opt.relativenumber = true -- show relative numbers above/below for fast j/k jumps
vim.opt.cursorline = true -- highlight the entire line the cursor is on
vim.opt.cursorcolumn = true -- highlight the entire column the cursor is in
vim.opt.showmode = false -- airline already shows mode; suppress the redundant "-- INSERT --" message
vim.opt.shiftwidth = 4 -- number of spaces for each indentation level (>> / <<)
vim.opt.scrolloff = 5 -- keep at least 5 lines visible above/below the cursor
vim.opt.wrap = false -- disable line wrapping (horizontal scroll instead)
vim.opt.incsearch = true -- show matches incrementally as you type the search pattern
vim.opt.ignorecase = true -- case-insensitive search by default
vim.opt.smartcase = true -- override ignorecase when the pattern contains uppercase
vim.opt.showcmd = true -- show the partial command being typed in the status line
vim.opt.showmatch = true -- briefly jump to the matching bracket when one is typed
vim.opt.hlsearch = true -- highlight all search matches (clear with :noh)
vim.opt.history = 1000 -- keep 1000 entries in command and search history
vim.opt.wildmenu = true -- show a completion menu for : commands
vim.opt.wildmode = "list:longest" -- first tab lists completions, second completes to longest common prefix
-- ── Keymaps ───────────────────────────────────────────────────────────────────
-- window navigation (normal mode)
-- Use Ctrl+hjkl to move between splits instead of the two-key <C-w> prefix.
-- <C-l> cycles to the next window (wraps); others jump directionally.
vim.keymap.set("n", "<C-l>", "<C-w>w")
vim.keymap.set("n", "<C-h>", "<C-w>h")
vim.keymap.set("n", "<C-j>", "<C-w>j")
vim.keymap.set("n", "<C-k>", "<C-w>k")
-- window navigation from terminal-insert mode (exit insert, then move)
-- <C-\><C-n> exits terminal-insert mode first; without it the control sequence
-- is sent to the shell process instead of being handled by Neovim.
vim.keymap.set("t", "<C-h>", "<C-\\><C-n><C-w>h", { silent = true })
vim.keymap.set("t", "<C-j>", "<C-\\><C-n><C-w>j", { silent = true })
vim.keymap.set("t", "<C-k>", "<C-\\><C-n><C-w>k", { silent = true })
vim.keymap.set("t", "<C-l>", "<C-\\><C-n><C-w>w", { silent = true })
-- auto-enter insert mode when focusing a terminal buffer (skip floaterm)
-- Without this, focusing a terminal buffer requires pressing 'i' manually.
-- Floaterm manages its own mode, so we exclude it to avoid conflicting with
-- its internal state machine.
vim.api.nvim_create_autocmd("BufEnter", {
pattern = "term://*",
callback = function()
if vim.bo.filetype ~= "floaterm" then
vim.cmd("startinsert")
end
end,
})
-- calendar.vim steals <C-hjkl> for month nav; restore window movement
-- The plugin sets buffer-local mappings on FileType calendar, so we override
-- them with { buffer = true } mappings of our own that fire after the plugin's.
vim.api.nvim_create_autocmd("FileType", {
pattern = "calendar",
callback = function()
local o = { buffer = true, silent = true }
vim.keymap.set("n", "<C-l>", "<C-w>w", o)
vim.keymap.set("n", "<C-h>", "<C-w>h", o)
vim.keymap.set("n", "<C-j>", "<C-w>j", o)
vim.keymap.set("n", "<C-k>", "<C-w>k", o)
end,
})
-- quick actions: single-key shortcuts for frequently used commands
-- t: open a new floating terminal via vim-floaterm
vim.keymap.set("n", "t", ":FloatermNew<CR>", { silent = true })
-- e: toggle the NERDTree sidebar; <C-W>l immediately moves focus back to the
-- editing window so the cursor doesn't land in the tree
vim.keymap.set("n", "e", ":NERDTreeToggle<CR><C-W>l", { silent = true })
-- s: toggle the vim-dadbod-ui database browser
vim.keymap.set("n", "s", ":DBUIToggle<CR>", { silent = true })
-- q: try to save-quit (wq); fall back to force-quit (q) for unsaved/read-only buffers
vim.keymap.set("n", "q", function()
local ok = pcall(vim.cmd, "wq")
if not ok then vim.cmd("q") end
end, { silent = true })
-- insert mode completion: Tab triggers Neovim's built-in keyword completion (<C-N>)
-- S-Tab reverts to a literal Tab character so indentation still works normally.
vim.keymap.set("i", "<TAB>", "<C-N>")
vim.keymap.set("i", "<S-TAB>", "<TAB>")
-- sudo save: :w!! pipes the buffer through sudo tee, bypassing a read-only
-- file system permission. The '%' is the current file path.
vim.cmd("ca w!! w !sudo tee '%'")
-- :Vb is a convenience alias for entering visual-block mode without pressing
-- Ctrl+v, which conflicts with paste in some terminal emulators.
vim.cmd("command! Vb normal! <C-v>")
-- ── UltiSnips ─────────────────────────────────────────────────────────────────
vim.g.UltiSnipsExpandTrigger = "<C-tab>"
vim.g.UltiSnipsJumpForwardTrigger = "<c-b>"
vim.g.UltiSnipsJumpBackwardTrigger = "<c-z>"
vim.g.UltiSnipsEditSplit = "vertical"
-- ── CoC ───────────────────────────────────────────────────────────────────────
vim.g.coc_global_extensions = {
"coc-snippets", "coc-powershell", "coc-sh", "coc-omnisharp",
"coc-clangd", "coc-json", "coc-css", "coc-git", "coc-pyright", "coc-sql",
}
vim.g.coc_snippet_next = "<c-j>"
vim.g.coc_snippet_prev = "<c-k>"
vim.keymap.set("i", "<C-l>", "<Plug>(coc-snippets-expand)", { remap = true })
vim.keymap.set("v", "<C-j>", "<Plug>(coc-snippets-select)", { remap = true })
vim.keymap.set("i", "<C-j>", "<Plug>(coc-snippets-expand-jump)", { remap = true })
vim.keymap.set("x", "<leader>x", "<Plug>(coc-convert-snippet)", { remap = true })
-- tab/s-tab navigate CoC pum, else fall through
vim.keymap.set("i", "<Tab>", function()
return vim.fn["coc#pum#visible"]() == 1 and vim.fn["coc#pum#next"](1) or "<Tab>"
end, { expr = true, silent = true })
vim.keymap.set("i", "<S-Tab>", function()
return vim.fn["coc#pum#visible"]() == 1 and vim.fn["coc#pum#prev"](1) or "<S-Tab>"
end, { expr = true, silent = true })
-- CR confirms CoC selection
vim.keymap.set("i", "<CR>", function()
return vim.fn["coc#pum#visible"]() == 1 and vim.fn["coc#pum#confirm"]() or "<CR>"
end, { expr = true, silent = true })
-- ── PIM floating windows (sideward T: left column overlay) ───────────────────
local function _pim_scratch(label, err)
vim.cmd("enew")
vim.bo.buftype = "nofile"
vim.bo.buflisted = false
vim.api.nvim_buf_set_lines(0, 0, -1, false, { "[" .. label .. " unavailable]", "", err or "" })
end
local function _pim_close_win(win)
if not vim.api.nvim_win_is_valid(win) then return end
local buf = vim.api.nvim_win_get_buf(win)
vim.api.nvim_win_close(win, true)
if vim.api.nvim_buf_is_valid(buf) and vim.bo[buf].buftype == "terminal" then
pcall(vim.api.nvim_buf_delete, buf, { force = true })
end
end
local function _pim_float(row, col, height, width, border)
local buf = vim.api.nvim_create_buf(false, true)
local win = vim.api.nvim_open_win(buf, true, {
relative = "editor",
row = row,
col = col,
height = math.max(1, height),
width = math.max(1, width),
style = "minimal",
border = border or "none",
zindex = 50,
})
vim.api.nvim_set_option_value("winhighlight", "Normal:Normal,NormalNC:Normal", { win = win })
return win
end
-- n/g/f: individual centered floating windows
local _solo = {}
local function toggle_solo(key, cmd, label)
local win = _solo[key]
if win and vim.api.nvim_win_is_valid(win) then
_pim_close_win(win)
_solo[key] = nil
return
end
local H = vim.o.lines - 2
local W = vim.o.columns
local h = math.max(1, math.floor(H * 0.85))
local w = math.max(1, math.floor(W * 0.85))
-- centre accounting for the 1-cell rounded border on each side
local r = math.max(0, math.floor((H - h - 2) / 2))
local c = math.max(0, math.floor((W - w - 2) / 2))
local win_id = _pim_float(r, c, h, w, "rounded")
local ok, err = pcall(vim.cmd, cmd)
if not ok then _pim_scratch(label, err) end
_solo[key] = win_id
end
vim.keymap.set("n", "n", function() toggle_solo("n", "terminal alot", "alot") end, { silent = true })
vim.keymap.set("n", "g", function() toggle_solo("g", "terminal khal interactive", "khal") end, { silent = true })
vim.keymap.set("n", "f", function() toggle_solo("f", "terminal abook", "abook") end, { silent = true })
-- r: sideward-T overlay — left column (bar of the T) with three stacked panes,
-- document remains visible to the right (the stem of the T)
local _pim_wins = {}
local function toggle_pim()
if #_pim_wins > 0 and vim.api.nvim_win_is_valid(_pim_wins[1]) then
for _, w in ipairs(_pim_wins) do _pim_close_win(w) end
_pim_wins = {}
return
end
-- full-screen: alot left, abook top-right, calendar bottom-right
local H = vim.o.lines - 2
local W = vim.o.columns
local right_w = math.max(80, math.floor(W * 0.45))
local left_w = math.max(1, W - right_w)
local top_h = math.max(20, math.floor(H / 2))
local bot_h = math.max(1, H - top_h)
local w1 = _pim_float(0, 0, H, left_w)
local ok, err = pcall(vim.cmd, "terminal alot")
if not ok then _pim_scratch("alot", err) end
local w2 = _pim_float(0, left_w, top_h, right_w)
ok, err = pcall(vim.cmd, "terminal abook")
if not ok then _pim_scratch("abook", err) end
local w3 = _pim_float(top_h, left_w, bot_h, right_w)
ok, err = pcall(vim.cmd, "terminal khal interactive")
if not ok then _pim_scratch("khal", err) end
_pim_wins = { w1, w2, w3 }
vim.api.nvim_set_current_win(w1)
end
vim.keymap.set("n", "x", toggle_pim, { silent = true })
-- ── CalDAV background sync on startup ────────────────────────────────────────
vim.api.nvim_create_autocmd("VimEnter", {
once = true,
callback = function()
vim.fn.jobstart(
{ "sh", "-c", "vdirsyncer sync && ics-to-calendarim" },
{ detach = true }
)
end,
})