dotfiles/config/nvim/lua/fschauen/util.lua

158 lines
4.3 KiB
Lua

local M = {}
-- Flip function arguments.
--
-- flip(f)(a, b) == f(b, a)
--
M.flip = function(f)
return function(a, b)
return f(b, a)
end
end
-- Extend lists.
--
-- extend({'a', 'b'}, {'c', 'd'}) == {'a', 'b', 'c', 'd'}
-- extend({1, 2}, {3, 4}, {5, 6}) == {1, 2, 3, 4, 5, 6}
--
M.concat = function(...)
local result = {}
for _, tbl in ipairs {...} do
for _, v in pairs(tbl) do
result[#result+1] = v
end
end
return result
end
-- Partial function application:
--
-- partial(f, x)(...) == f(x, ...)
-- partial(f, x, y)(...) == f(x, y, ...)
--
M.partial = function(f, ...)
local argv = {...}
return function(...)
return f(unpack(M.concat(argv, {...})))
end
end
--- Delayed function execution.
---@param f function: function whose evaluation will be delayed.
---@param ... any: arguments to `f`.
---@return function: a new function that calls f with provided arguments.
M.thunk = function(f, ...)
local args = {...}
return function()
return f(unpack(args))
end
end
-- Perform `func` (which can freely use register `reg`) and make sure `reg`
-- is restored afterwards.
M.with_saved_register = function(reg, func)
local saved = vim.fn.getreg(reg)
local result = func()
vim.fn.setreg(reg, saved)
return result
end
-- Get selected text, or word under cursor if not in visual mode.
M.get_selected_text = function()
if vim.fn.mode() ~= 'v' then return vim.fn.expand '<cword>' end
return M.with_saved_register('v', function()
vim.cmd [[noautocmd sil norm "vy]]
return vim.fn.getreg 'v'
end)
end
local diag_opts = {
wrap = false, -- don't wrap around the begin/end of file
float = {
border = 'rounded' -- enable border for the floating window
},
}
-- Move to the next diagnostic.
M.goto_next_diagnostic = function(opts)
vim.diagnostic.goto_next(vim.tbl_extend('keep', opts or {}, diag_opts))
end
-- Move to the previous diagnostic.
M.goto_prev_diagnostic = function(opts)
vim.diagnostic.goto_prev(vim.tbl_extend('keep', opts or {}, diag_opts))
end
--- Whether the current window is the last in a given direction.
---@param direction string: one of 'h', 'j', 'k', or 'l'
local win_is_last = function(direction)
local current = vim.api.nvim_get_current_win()
vim.cmd('wincmd ' .. direction)
local next = vim.api.nvim_get_current_win()
local is_last = current == next
if not is_last then vim.cmd('wincmd p') end
return is_last
end
--- Resize current window in a given direction.
---@param dir string: one of 'h', 'j', 'k', or 'l'
---@param size integer: how much to resize
local win_resize = function(dir, size)
if dir ~= 'h' and dir ~= 'j' and dir ~= 'k' and dir ~= 'l' then return end
size = math.abs(size)
local is_height = dir == 'j' or dir == 'k'
local is_positive = dir == 'j' or dir == 'l'
if win_is_last(is_height and 'j' or 'l') then
is_positive = not is_positive
end
local delta = string.format('%s%d', is_positive and '+' or '-', size)
local prefix = is_height and '' or 'vertical '
vim.cmd(prefix .. 'resize ' .. delta .. '<cr>')
end
--- Resize current window upwards.
---@param size integer: how much to resize
M.win_resize_up = function(size) return M.thunk(win_resize, 'k', size) end
--- Resize current window downwards.
---@param size integer: how much to resize
M.win_resize_down = function(size) return M.thunk(win_resize, 'j', size) end
--- Resize current window leftwards.
---@param size integer: how much to resize
M.win_resize_left = function(size) return M.thunk(win_resize, 'h', size) end
--- Resize current window rightwards.
---@param size integer: how much to resize
M.win_resize_right = function(size) return M.thunk(win_resize, 'l', size) end
--- Toggle quickfix (or location) list.
---@param qf string: 'c' for quickfix, 'l' for location list
local toggle_qf_list = function(qf)
local l = qf == 'l' and 1 or 0
local is_qf = function(win) return win.quickfix == 1 and win.loclist == l end
local is_open = not vim.tbl_isempty(vim.tbl_filter(is_qf, vim.fn.getwininfo()))
if is_open then
vim.cmd(qf .. 'close')
else
local ok = pcall(function(c) vim.cmd(c) end, qf .. 'open')
if not ok and qf == 'l' then
vim.notify('No location list', vim.log.levels.WARN)
end
end
end
--- Toggle quickfix list.
M.toggle_quickfix = function() toggle_qf_list('c') end
--- Toggle location list.
M.toggle_loclist = function() toggle_qf_list('l') end
return M