mirror of
https://github.com/NotAShelf/nvf.git
synced 2025-04-15 00:58:37 +00:00
274 lines
8.4 KiB
Nix
274 lines
8.4 KiB
Nix
{
|
|
config,
|
|
pkgs,
|
|
lib,
|
|
...
|
|
}: let
|
|
inherit (builtins) attrNames;
|
|
inherit (lib.options) mkEnableOption mkOption;
|
|
inherit (lib.types) bool enum package listOf;
|
|
inherit (lib.meta) getExe;
|
|
inherit (lib.modules) mkIf mkMerge;
|
|
inherit (lib.generators) mkLuaInline;
|
|
inherit (lib.nvim.types) mkGrammarOption;
|
|
inherit (lib.nvim.attrsets) mapListToAttrs;
|
|
inherit (lib.nvim.dag) entryAfter;
|
|
|
|
cfg = config.vim.languages.clang;
|
|
|
|
defaultServer = ["clangd"];
|
|
servers = {
|
|
ccls.options = {
|
|
cmd = [(getExe pkgs.ccls)];
|
|
filetypes = ["c" "cpp" "objc" "objcpp" "cuda"];
|
|
offset_encoding = "utf-32";
|
|
root_dir = mkLuaInline ''
|
|
function(bufnr, cb)
|
|
local dir = vim.fs.root(bufnr, {"compile_commands.json", ".ccls"})
|
|
or vim.fs.root(bufnr, {".git"})
|
|
if dir then cb(dir) end
|
|
end
|
|
'';
|
|
on_attach = mkLuaInline ''
|
|
function(client, bufnr)
|
|
default_on_attach(client, bufnr)
|
|
|
|
local function switch_source_header(bufnr)
|
|
local method_name = "textDocument/switchSourceHeader"
|
|
local client = vim.lsp.get_clients({
|
|
bufnr = bufnr,
|
|
name = "ccls",
|
|
method = method_name,
|
|
})[1]
|
|
if not client then
|
|
return vim.notify(('method %s is not supported by any servers active on the current buffer'):format(method_name))
|
|
end
|
|
local params = vim.lsp.util.make_text_document_params(bufnr)
|
|
client:request(method_name, params, function(err, result)
|
|
if err then
|
|
error(tostring(err))
|
|
end
|
|
if not result then
|
|
vim.notify('corresponding file cannot be determined')
|
|
return
|
|
end
|
|
vim.cmd.edit(vim.uri_to_fname(result))
|
|
end, bufnr)
|
|
end
|
|
|
|
vim.api.nvim_buf_create_user_command(
|
|
bufnr,
|
|
"CclsSwitchSourceHeader",
|
|
function(arg)
|
|
switch_source_header(0)
|
|
end,
|
|
{desc = "Switch between source/header"}
|
|
)
|
|
end
|
|
'';
|
|
};
|
|
|
|
clangd.options = {
|
|
cmd = ["${pkgs.clang-tools}/bin/clangd"];
|
|
filetypes = ["c" "cpp" "objc" "objcpp" "cuda" "proto"];
|
|
root_dir = mkLuaInline ''
|
|
function(bufnr, cb)
|
|
local dir = vim.fs.root(bufnr, {
|
|
'.clangd',
|
|
'.clang-tidy',
|
|
'.clang-format',
|
|
'compile_commands.json',
|
|
'compile_flags.txt',
|
|
'configure.ac' -- AutoTools
|
|
}) or vim.fs.root(bufnr, {".git"})
|
|
|
|
if dir then cb(dir) end
|
|
end
|
|
'';
|
|
capabilities = {
|
|
textDocument = {
|
|
completion = {
|
|
editsNearCursor = true;
|
|
};
|
|
};
|
|
offsetEncoding = ["utf-8" "utf-16"];
|
|
};
|
|
on_attach = mkLuaInline ''
|
|
function(client, bufnr)
|
|
default_on_attach(client, bufnr)
|
|
|
|
local function switch_source_header(bufnr)
|
|
local method_name = "textDocument/switchSourceHeader"
|
|
local client = vim.lsp.get_clients({
|
|
bufnr = bufnr,
|
|
name = "clangd",
|
|
method = method_name,
|
|
})[1]
|
|
if not client then
|
|
return vim.notify(('method %s is not supported by any servers active on the current buffer'):format(method_name))
|
|
end
|
|
local params = vim.lsp.util.make_text_document_params(bufnr)
|
|
client:request(method_name, params, function(err, result)
|
|
if err then
|
|
error(tostring(err))
|
|
end
|
|
if not result then
|
|
vim.notify('corresponding file cannot be determined')
|
|
return
|
|
end
|
|
vim.cmd.edit(vim.uri_to_fname(result))
|
|
end, bufnr)
|
|
end
|
|
|
|
local function symbol_info()
|
|
local clangd_client = vim.lsp.get_clients({
|
|
bufnr = 0,
|
|
name = "clangd",
|
|
method = "textDocument/symbolInfo",
|
|
})[1]
|
|
if not clangd_client or not clangd_client.supports_method 'textDocument/symbolInfo' then
|
|
return vim.notify('Clangd client not found', vim.log.levels.ERROR)
|
|
end
|
|
local win = vim.api.nvim_get_current_win()
|
|
local params = vim.lsp.util.make_position_params(win, clangd_client.offset_encoding)
|
|
clangd_client:request('textDocument/symbolInfo', params, function(err, res)
|
|
if err or #res == 0 then
|
|
-- Clangd always returns an error, there is not reason to parse it
|
|
return
|
|
end
|
|
local container = string.format('container: %s', res[1].containerName) ---@type string
|
|
local name = string.format('name: %s', res[1].name) ---@type string
|
|
vim.lsp.util.open_floating_preview({ name, container }, "", {
|
|
height = 2,
|
|
width = math.max(string.len(name), string.len(container)),
|
|
focusable = false,
|
|
focus = false,
|
|
border = 'single',
|
|
title = 'Symbol Info',
|
|
})
|
|
end, bufnr)
|
|
end
|
|
|
|
vim.api.nvim_buf_create_user_command(
|
|
bufnr,
|
|
"ClangdSwitchSourceHeader",
|
|
function(arg)
|
|
switch_source_header(0)
|
|
end,
|
|
{desc = "Switch between source/header"}
|
|
)
|
|
|
|
vim.api.nvim_buf_create_user_command(
|
|
bufnr,
|
|
"ClangdShowSymbolInfo",
|
|
function(arg)
|
|
symbol_info()
|
|
end,
|
|
{desc = "Show symbol info"}
|
|
)
|
|
end
|
|
'';
|
|
};
|
|
};
|
|
|
|
defaultDebugger = "lldb-vscode";
|
|
debuggers = {
|
|
lldb-vscode = {
|
|
package = pkgs.lldb;
|
|
dapConfig = ''
|
|
dap.adapters.lldb = {
|
|
type = 'executable',
|
|
command = '${cfg.dap.package}/bin/lldb-dap',
|
|
name = 'lldb'
|
|
}
|
|
dap.configurations.cpp = {
|
|
{
|
|
name = 'Launch',
|
|
type = 'lldb',
|
|
request = 'launch',
|
|
program = function()
|
|
return vim.fn.input('Path to executable: ', vim.fn.getcwd() .. '/', 'file')
|
|
end,
|
|
cwd = "''${workspaceFolder}",
|
|
stopOnEntry = false,
|
|
args = {},
|
|
},
|
|
}
|
|
|
|
dap.configurations.c = dap.configurations.cpp
|
|
'';
|
|
};
|
|
};
|
|
in {
|
|
options.vim.languages.clang = {
|
|
enable = mkEnableOption "C/C++ language support";
|
|
|
|
cHeader = mkOption {
|
|
description = ''
|
|
C syntax for headers. Can fix treesitter errors, see:
|
|
https://www.reddit.com/r/neovim/comments/orfpcd/question_does_the_c_parser_from_nvimtreesitter/
|
|
'';
|
|
type = bool;
|
|
default = false;
|
|
};
|
|
|
|
treesitter = {
|
|
enable = mkEnableOption "C/C++ treesitter" // {default = config.vim.languages.enableTreesitter;};
|
|
cPackage = mkGrammarOption pkgs "c";
|
|
cppPackage = mkGrammarOption pkgs "cpp";
|
|
};
|
|
|
|
lsp = {
|
|
enable = mkEnableOption "clang LSP support" // {default = config.vim.languages.enableLSP;};
|
|
|
|
server = mkOption {
|
|
description = "The clang LSP server to use";
|
|
type = listOf (enum (attrNames servers));
|
|
default = defaultServer;
|
|
};
|
|
};
|
|
|
|
dap = {
|
|
enable = mkOption {
|
|
description = "Enable clang Debug Adapter";
|
|
type = bool;
|
|
default = config.vim.languages.enableDAP;
|
|
};
|
|
debugger = mkOption {
|
|
description = "clang debugger to use";
|
|
type = enum (attrNames debuggers);
|
|
default = defaultDebugger;
|
|
};
|
|
package = mkOption {
|
|
description = "clang debugger package.";
|
|
type = package;
|
|
default = debuggers.${cfg.dap.debugger}.package;
|
|
};
|
|
};
|
|
};
|
|
|
|
config = mkIf cfg.enable (mkMerge [
|
|
(mkIf cfg.cHeader {
|
|
vim.pluginRC.c-header = entryAfter ["basic"] "vim.g.c_syntax_for_h = 1";
|
|
})
|
|
|
|
(mkIf cfg.treesitter.enable {
|
|
vim.treesitter.enable = true;
|
|
vim.treesitter.grammars = [cfg.treesitter.cPackage cfg.treesitter.cppPackage];
|
|
})
|
|
|
|
(mkIf cfg.lsp.enable {
|
|
vim.lsp.servers =
|
|
mapListToAttrs (name: {
|
|
inherit name;
|
|
value = servers.${name}.options;
|
|
})
|
|
cfg.lsp.server;
|
|
})
|
|
|
|
(mkIf cfg.dap.enable {
|
|
vim.debugger.nvim-dap.enable = true;
|
|
vim.debugger.nvim-dap.sources.clang-debugger = debuggers.${cfg.dap.debugger}.dapConfig;
|
|
})
|
|
]);
|
|
}
|