diff --git a/init.lua b/init.lua index d5ae6dc9b2a..91bc5fc7c2e 100644 --- a/init.lua +++ b/init.lua @@ -91,7 +91,7 @@ vim.g.mapleader = ' ' vim.g.maplocalleader = ' ' -- Set to true if you have a Nerd Font installed and selected in the terminal -vim.g.have_nerd_font = false +vim.g.have_nerd_font = true -- [[ Setting options ]] -- See `:help vim.o` @@ -102,7 +102,8 @@ vim.g.have_nerd_font = false vim.o.number = true -- You can also add relative line numbers, to help with jumping. -- Experiment for yourself to see if you like it! --- vim.o.relativenumber = true +vim.o.relativenumber = true +vim.o.statuscolumn = '%s %l %r ' -- Enable mouse mode, can be useful for resizing splits for example! vim.o.mouse = 'a' @@ -285,6 +286,103 @@ require('lazy').setup({ }, }, }, + { + 'L3MON4D3/LuaSnip', + version = 'v2.*', + build = 'make install_jsregexp', + event = 'InsertEnter', + config = function() + local luasnip = require 'luasnip' + + -- Optional: load VSCode-style snippets if you use them + require('luasnip.loaders.from_vscode').lazy_load() + + -- Example: no default / bindings here + -- so they stay free for tabout.nvim + end, + }, + + --------------------------------------------------------------------------- + -- COMPLETION (nvim-cmp) + --------------------------------------------------------------------------- + { + 'hrsh7th/nvim-cmp', + event = 'InsertEnter', + dependencies = { + 'L3MON4D3/LuaSnip', + 'saadparwaiz1/cmp_luasnip', + 'hrsh7th/cmp-nvim-lsp', + 'hrsh7th/cmp-buffer', + 'hrsh7th/cmp-path', + }, + config = function() + local cmp = require 'cmp' + local luasnip = require 'luasnip' + + cmp.setup { + snippet = { + expand = function(args) + luasnip.lsp_expand(args.body) + end, + }, + mapping = cmp.mapping.preset.insert { + [''] = cmp.mapping.select_next_item(), + [''] = cmp.mapping.select_prev_item(), + [''] = cmp.mapping.scroll_docs(-4), + [''] = cmp.mapping.scroll_docs(4), + [''] = cmp.mapping.complete(), + [''] = cmp.mapping.abort(), + + -- Use Ctrl-Y to confirm completion + [''] = cmp.mapping.confirm { select = true }, + + -- NOTE: no / here => free for tabout.nvim + }, + sources = cmp.config.sources({ + { name = 'nvim_lsp' }, + { name = 'luasnip' }, + }, { + { name = 'path' }, + { name = 'buffer' }, + }), + } + end, + }, + + --------------------------------------------------------------------------- + -- TABOUT (use Tab / S-Tab to jump out of brackets/quotes) + --------------------------------------------------------------------------- + { + 'abecodes/tabout.nvim', + event = 'InsertEnter', + dependencies = { + 'nvim-treesitter/nvim-treesitter', + 'L3MON4D3/LuaSnip', + 'hrsh7th/nvim-cmp', + }, + config = function() + require('tabout').setup { + tabkey = '', -- Tab to jump out + backwards_tabkey = '', -- Shift-Tab to jump back + act_as_tab = true, -- behave like normal Tab if no tabout + act_as_shift_tab = false, + default_tab = '', -- at line start: indent + default_shift_tab = '', -- at line start: dedent + enable_backwards = true, + completion = false, -- cmp uses Ctrl-Y, so no need to integrate + tabouts = { + { open = "'", close = "'" }, + { open = '"', close = '"' }, + { open = '`', close = '`' }, + { open = '(', close = ')' }, + { open = '[', close = ']' }, + { open = '{', close = '}' }, + }, + ignore_beginning = true, + exclude = {}, + } + end, + }, -- NOTE: Plugins can also be configured to run Lua code when they are loaded. -- @@ -316,7 +414,342 @@ require('lazy').setup({ }, }, }, + { + 'RRethy/vim-illuminate', + event = 'VeryLazy', + config = function() + require('illuminate').configure { + delay = 120, + providers = { 'lsp', 'treesitter', 'regex' }, + -- keep other defaults + } + + local function set_illuminate_hl() + vim.api.nvim_set_hl(0, 'IlluminatedWordText', { underline = true, bg = 'NONE' }) + vim.api.nvim_set_hl(0, 'IlluminatedWordRead', { underline = true, bg = 'NONE' }) + vim.api.nvim_set_hl(0, 'IlluminatedWordWrite', { underline = true, bg = 'NONE' }) + end + + -- set now + set_illuminate_hl() + + -- and reset whenever colorscheme changes + vim.api.nvim_create_autocmd('ColorScheme', { + callback = set_illuminate_hl, + }) + end, + }, + { + -- NOTE: Yes, you can install new plugins here! + 'mfussenegger/nvim-dap', + -- NOTE: And you can specify dependencies as well + dependencies = { + -- Creates a beautiful debugger UI + 'rcarriga/nvim-dap-ui', + + -- Required dependency for nvim-dap-ui + 'nvim-neotest/nvim-nio', + + -- Installs the debug adapters for you + 'williamboman/mason.nvim', + 'jay-babu/mason-nvim-dap.nvim', + + -- Add your own debuggers here + 'leoluz/nvim-dap-go', + 'mfussenegger/nvim-dap-python', + }, + keys = { + -- Basic debugging keymaps, feel free to change to your liking! + { + '', + function() + require('dap').continue() + end, + desc = 'Debug: Start/Continue', + }, + { + '', + function() + require('dap').step_into() + end, + desc = 'Debug: Step Into', + }, + { + '', + function() + require('dap').step_over() + end, + desc = 'Debug: Step Over', + }, + { + 'cb', + function() + require('dap').clear_breakpoints() + vim.notify('✅ Cleared all breakpoints', vim.log.levels.INFO) + end, + desc = 'Debug: Clear all Breakpoints', + }, + + { + '', + function() + require('dap').step_out() + end, + desc = 'Debug: Step Out', + }, + { + 'b', + function() + require('dap').toggle_breakpoint() + end, + desc = 'Debug: Toggle Breakpoint', + }, + { + 'B', + function() + require('dap').set_breakpoint(vim.fn.input 'Breakpoint condition: ') + end, + desc = 'Debug: Set Breakpoint', + }, + { + 'du', + function() + require('dapui').toggle() + vim.notify('DAP UI toggle', vim.log.levels.INFO) + end, + desc = 'Debug: Toggle DAP UI', + }, + + { + 'dl', + function() + require('dap').run_last() + end, + desc = 'Debug: Re-run last session', + }, -- Toggle to see last session result. Without this, you can't see session output in case of unhandled exception. + { + '', + function() + require('dapui').toggle() + end, + desc = 'Debug: See last session result.', + }, + }, + config = function() + local dap = require 'dap' + local dapui = require 'dapui' + + require('mason-nvim-dap').setup { + -- Makes a best effort to setup the various debuggers with + -- reasonable debug configurations + automatic_installation = true, + + -- You can provide additional configuration to the handlers, + -- see mason-nvim-dap README for more information + handlers = {}, + + -- You'll need to check that you have the required things installed + -- online, please don't ask me how to install them :) + ensure_installed = { + -- Update this to ensure that you have the debuggers for the langs you want + 'delve', + 'python', + 'netcoredbg', + }, + } + -- ─────────────── ADD HERE ─────────────────── + -- local dap = require 'dap' + local dapui = require 'dapui' + + local dap = require 'dap' + + dap.adapters.coreclr = { + type = 'executable', + command = vim.fn.stdpath 'data' .. '/mason/packages/netcoredbg/netcoredbg', + args = { '--interpreter=vscode' }, + } + + -- make sure this path is correct for *your* project name & TF: + local project_folder = vim.fn.getcwd() + local dll_path = project_folder .. '/bin/Debug/net8.0/MyWebApp.dll' + + dap.configurations.cs = { + { + type = 'coreclr', + name = 'Launch WebApp DLL', + request = 'launch', + program = dll_path, + cwd = project_folder, + stopAtEntry = false, + env = { + ASPNETCORE_ENVIRONMENT = 'Development', + ASPNETCORE_URLS = 'http://localhost:5064', + }, + }, + } + + dap.listeners.after.event_initialized['dapui_config'] = dapui.open + -- For more information, see |:help nvim-dap-ui| + dapui.setup { + -- Set icons to characters that are more likely to work in every terminal. + -- Feel free to remove or use ones that you like more! :) + -- Don't feel like these are good choices. + icons = { expanded = '▾', collapsed = '▸', current_frame = '*' }, + controls = { + icons = { + pause = '⏸', + play = '▶', + step_into = '⏎', + step_over = '⏭', + step_out = '⏮', + step_back = 'b', + run_last = '▶▶', + terminate = '⏹', + disconnect = '⏏', + }, + }, + } + + -- Change breakpoint icons + vim.api.nvim_set_hl(0, 'DapBreak', { fg = '#e51400' }) + vim.api.nvim_set_hl(0, 'DapStop', { fg = '#ffcc00' }) + local breakpoint_icons = vim.g.have_nerd_font + and { Breakpoint = '', BreakpointCondition = '', BreakpointRejected = '', LogPoint = '', Stopped = '' } + or { Breakpoint = '●', BreakpointCondition = '⊜', BreakpointRejected = '⊘', LogPoint = '◆', Stopped = '⭔' } + for type, icon in pairs(breakpoint_icons) do + local tp = 'Dap' .. type + local hl = (type == 'Stopped') and 'DapStop' or 'DapBreak' + vim.fn.sign_define(tp, { text = icon, texthl = hl, numhl = hl }) + end + + dap.listeners.after.event_initialized['dapui_config'] = dapui.open + dap.listeners.before.event_terminated['dapui_config'] = dapui.close + dap.listeners.before.event_exited['dapui_config'] = dapui.close + + -- Install golang specific config + require('dap-go').setup { + delve = { + -- On Windows delve must be run attached or it crashes. + -- See https://github.com/leoluz/nvim-dap-go/blob/main/README.md#configuring + detached = vim.fn.has 'win32' == 0, + }, + } + + table.insert(require('dap').configurations.python, { + type = 'python', + request = 'launch', + name = 'base', + program = '/Users/quandoan/Desktop/odoo-18.0/odoo-bin', + pythonPath = '/usr/local/bin/python3.12', + args = { + '-c', + 'debian/odoo-base.conf', + '-d', + 'base-1', + '--xmlrpc-port', + '9999', + }, + justMyCode = false, + env = { + PYTHONPATH = '/Users/quandoan/Desktop/odoo-18.0', + }, + }) + table.insert(require('dap').configurations.python, { + type = 'python', + request = 'launch', + name = 'odoo19 base', + program = '/Users/quandoan/Desktop/odoo19/odoo-bin', + pythonPath = '/Users/quandoan/Desktop/odoo19/.venv1/bin/python', + args = { + '-c', + 'debian/odoo.conf', + '-u', + 'a1_purchase_custom', + }, + justMyCode = false, + env = { + PYTHONPATH = '/Users/quandoan/Desktop/odoo19', + }, + }) + table.insert(require('dap').configurations.python, { + type = 'python', + request = 'launch', + name = 'Tayoong', + program = '/Users/quandoan/Desktop/odoo-18.0/odoo-bin', + pythonPath = '/usr/local/bin/python3.12', + args = { + '-c', + 'debian/odoo-tayoong.conf', + '-u', + 'a1_einvoice_to_gov', + -- '-d', + -- 'tayoong-test', + '--xmlrpc-port', + '8069', + }, + justMyCode = false, + env = { + PYTHONPATH = '/Users/quandoan/Desktop/odoo-18.0', + }, + }) + + table.insert(require('dap').configurations.python, { + type = 'python', + request = 'launch', + program = '/Users/quandoan/Desktop/odoo-18.0/odoo-bin', + name = 'E-invoice', + pythonPath = '/usr/local/bin/python3.12', + args = { + '-c', + 'debian/odoo-e-invoice.conf', + '-u', + 'a1_einvoice_to_gov,tayoong_issue_consolidate_invoice,issue_consolidate_invoice', + + -- '-d', + -- 'e-invoice-3', + '--xmlrpc-port', + '8099', + }, + justMyCode = false, + env = { + PYTHONPATH = '/Users/quandoan/Desktop/odoo-18.0', + }, + }) + end, + }, + { + 'kdheepak/lazygit.nvim', + cmd = 'LazyGit', + dependencies = { + 'nvim-lua/plenary.nvim', + }, + keys = { + { + 'lg', + 'LazyGit', + desc = 'Open LazyGit', + }, + }, + }, + { + 'akinsho/toggleterm.nvim', + version = '*', + config = function() + require('toggleterm').setup { + open_mapping = [[]], -- Bind Ctrl + T + direction = 'float', -- or "horizontal" | "vertical" | "tab" + size = 20, + start_in_insert = true, + shade_terminals = true, + } + -- Optional: Ensure works in both normal and terminal mode + vim.keymap.set({ 'n', 't' }, '', 'ToggleTerm', { noremap = true, silent = true }) + end, + }, + { + 'mluders/comfy-line-numbers.nvim', + }, -- NOTE: Plugins can specify dependencies. -- -- The dependencies are proper plugin specifications as well - anything @@ -477,81 +910,55 @@ require('lazy').setup({ -- Main LSP Configuration 'neovim/nvim-lspconfig', dependencies = { - -- Automatically install LSPs and related tools to stdpath for Neovim - -- Mason must be loaded before its dependents so we need to set it up here. - -- NOTE: `opts = {}` is the same as calling `require('mason').setup({})` { 'mason-org/mason.nvim', opts = {} }, 'WhoIsSethDaniel/mason-tool-installer.nvim', - -- Useful status updates for LSP. { 'j-hui/fidget.nvim', opts = {} }, - -- Allows extra capabilities provided by blink.cmp - 'saghen/blink.cmp', + -- use nvim-cmp as completion engine + 'hrsh7th/cmp-nvim-lsp', }, config = function() - -- Brief aside: **What is LSP?** - -- - -- LSP is an initialism you've probably heard, but might not understand what it is. - -- - -- LSP stands for Language Server Protocol. It's a protocol that helps editors - -- and language tooling communicate in a standardized fashion. - -- - -- In general, you have a "server" which is some tool built to understand a particular - -- language (such as `gopls`, `lua_ls`, `rust_analyzer`, etc.). These Language Servers - -- (sometimes called LSP servers, but that's kind of like ATM Machine) are standalone - -- processes that communicate with some "client" - in this case, Neovim! - -- - -- LSP provides Neovim with features like: - -- - Go to definition - -- - Find references - -- - Autocompletion - -- - Symbol Search - -- - and more! - -- - -- Thus, Language Servers are external tools that must be installed separately from - -- Neovim. This is where `mason` and related plugins come into play. - -- - -- If you're wondering about lsp vs treesitter, you can check out the wonderfully - -- and elegantly composed help section, `:help lsp-vs-treesitter` - - -- This function gets run when an LSP attaches to a particular buffer. - -- That is to say, every time a new file is opened that is associated with - -- an lsp (for example, opening `main.rs` is associated with `rust_analyzer`) this - -- function will be executed to configure the current buffer + --------------------------------------------------------------------------- + -- LspAttach: buffer-local keymaps when a server attaches + --------------------------------------------------------------------------- vim.api.nvim_create_autocmd('LspAttach', { group = vim.api.nvim_create_augroup('kickstart-lsp-attach', { clear = true }), callback = function(event) - -- NOTE: Remember that Lua is a real programming language, and as such it is possible - -- to define small helper and utility functions so you don't have to repeat yourself. - -- - -- In this case, we create a function that lets us more easily define mappings specific - -- for LSP related items. It sets the mode, buffer and description for us each time. local map = function(keys, func, desc, mode) mode = mode or 'n' vim.keymap.set(mode, keys, func, { buffer = event.buf, desc = 'LSP: ' .. desc }) end - -- Rename the variable under your cursor. - -- Most Language Servers support renaming across files, etc. map('grn', vim.lsp.buf.rename, '[R]e[n]ame') - - -- Execute a code action, usually your cursor needs to be on top of an error - -- or a suggestion from your LSP for this to activate. map('gra', vim.lsp.buf.code_action, '[G]oto Code [A]ction', { 'n', 'x' }) - -- WARN: This is not Goto Definition, this is Goto Declaration. - -- For example, in C this would take you to the header. + map('grr', require('telescope.builtin').lsp_references, '[G]oto [R]eferences') + map('gri', require('telescope.builtin').lsp_implementations, '[G]oto [I]mplementation') + map('grd', require('telescope.builtin').lsp_definitions, '[G]oto [D]efinition') map('grD', vim.lsp.buf.declaration, '[G]oto [D]eclaration') + map('gO', require('telescope.builtin').lsp_document_symbols, 'Open Document Symbols') + map('gW', require('telescope.builtin').lsp_dynamic_workspace_symbols, 'Open Workspace Symbols') + map('grt', require('telescope.builtin').lsp_type_definitions, '[G]oto [T]ype Definition') + + ---@param client vim.lsp.Client + ---@param method vim.lsp.protocol.Method + ---@param bufnr? integer + local function client_supports_method(client, method, bufnr) + if vim.fn.has 'nvim-0.11' == 1 then + return client:supports_method(method, bufnr) + else + return client.supports_method(method, { bufnr = bufnr }) + end + end - -- The following two autocommands are used to highlight references of the - -- word under your cursor when your cursor rests there for a little while. - -- See `:help CursorHold` for information about when this is executed - -- - -- When you move your cursor, the highlights will be cleared (the second autocommand). local client = vim.lsp.get_client_by_id(event.data.client_id) - if client and client:supports_method('textDocument/documentHighlight', event.buf) then + + -- Highlight under-cursor references + if client and client_supports_method(client, vim.lsp.protocol.Methods.textDocument_documentHighlight, event.buf) then + local highlight_augroup = vim.api.nvim_create_augroup('kickstart-lsp-highlight', { clear = false }) + vim.api.nvim_create_autocmd({ 'CursorHold', 'CursorHoldI' }, { buffer = event.buf, group = highlight_augroup, @@ -655,7 +1062,74 @@ require('lazy').setup({ vim.lsp.enable 'lua_ls' end, }, - + { + 'folke/flash.nvim', + event = 'VeryLazy', + opts = {}, + keys = { + { + 's', + mode = { 'n', 'x', 'o' }, + function() + require('flash').jump() + end, + desc = 'Flash Jump', + }, + }, + }, + { + 'Pocco81/auto-save.nvim', + event = { 'InsertLeave', 'TextChanged' }, + config = function() + require('auto-save').setup { + enabled = true, + execution_message = { + message = function() + return '💾 AutoSaved at ' .. vim.fn.strftime '%H:%M:%S' + end, + dim = 0.18, + cleaning_interval = 1000, + }, + trigger_events = { 'InsertLeave', 'TextChanged' }, + condition = function(buf) + local fn = vim.fn + local utils = require 'auto-save.utils.data' + if fn.getbufvar(buf, '&modifiable') == 1 and utils.not_in(fn.getbufvar(buf, '&filetype'), {}) then + return true + end + return false + end, + } + end, + }, + { + 'folke/persistence.nvim', + event = 'BufReadPre', -- load before buffers + opts = {}, + keys = { + { + 'qs', + function() + require('persistence').load() + end, + desc = 'Restore Session', + }, + { + 'ql', + function() + require('persistence').load { last = true } + end, + desc = 'Restore Last Session', + }, + { + 'qd', + function() + require('persistence').stop() + end, + desc = "Don't Save Current Session", + }, + }, + }, { -- Autoformat 'stevearc/conform.nvim', event = { 'BufWritePre' }, @@ -687,7 +1161,8 @@ require('lazy').setup({ formatters_by_ft = { lua = { 'stylua' }, -- Conform can also run multiple formatters sequentially - -- python = { "isort", "black" }, + python = { 'isort', 'black' }, + xml = { 'lemmix' }, -- -- You can use 'stop_after_first' to run the first available formatter from the list -- javascript = { "prettierd", "prettier", stop_after_first = true }, @@ -750,7 +1225,7 @@ require('lazy').setup({ -- : Toggle signature help -- -- See :h blink-cmp-config-keymap for defining your own keymap - preset = 'default', + preset = 'super-tab', -- For more advanced Luasnip keymaps (e.g. selecting choice nodes, expansion) see: -- https://github.com/L3MON4D3/LuaSnip?tab=readme-ov-file#keymaps @@ -816,26 +1291,19 @@ require('lazy').setup({ { -- Collection of various small independent plugins/modules 'nvim-mini/mini.nvim', config = function() - -- Better Around/Inside textobjects - -- - -- Examples: - -- - va) - [V]isually select [A]round [)]paren - -- - yinq - [Y]ank [I]nside [N]ext [Q]uote - -- - ci' - [C]hange [I]nside [']quote - require('mini.ai').setup { n_lines = 500 } + require('mini.ai').setup { + n_lines = 500, + custom_textobjects = { + f = require('mini.ai').gen_spec.treesitter({ + a = '@function.outer', + i = '@function.inner', + }, {}), + }, + } - -- Add/delete/replace surroundings (brackets, quotes, etc.) - -- - -- - saiw) - [S]urround [A]dd [I]nner [W]ord [)]Paren - -- - sd' - [S]urround [D]elete [']quotes - -- - sr)' - [S]urround [R]eplace [)] ['] require('mini.surround').setup() - -- Simple and easy statusline. - -- You could remove this setup call if you don't like it, - -- and try some other statusline plugin local statusline = require 'mini.statusline' - -- set use_icons to true if you have a Nerd Font statusline.setup { use_icons = vim.g.have_nerd_font } -- You can configure sections in the statusline by overriding their @@ -871,11 +1339,11 @@ require('lazy').setup({ -- Uncomment any of the lines below to enable them (you will need to restart nvim). -- -- require 'kickstart.plugins.debug', - -- require 'kickstart.plugins.indent_line', - -- require 'kickstart.plugins.lint', - -- require 'kickstart.plugins.autopairs', - -- require 'kickstart.plugins.neo-tree', - -- require 'kickstart.plugins.gitsigns', -- adds gitsigns recommend keymaps + require 'kickstart.plugins.indent_line', + require 'kickstart.plugins.lint', + require 'kickstart.plugins.autopairs', + require 'kickstart.plugins.neo-tree', + require 'kickstart.plugins.gitsigns', -- adds gitsigns recommend keymaps -- NOTE: The import below can automatically add your own plugins, configuration, etc from `lua/custom/plugins/*.lua` -- This is the easiest way to modularize your config. @@ -911,3 +1379,10 @@ require('lazy').setup({ -- The line beneath this is called `modeline`. See `:help modeline` -- vim: ts=2 sts=2 sw=2 et +vim.keymap.set('n', 'K', ':m .-2==', { desc = 'Move line up', silent = true }) +vim.keymap.set('v', 'K', ":m '<-2gv=gv", { desc = 'Move selection up', silent = true }) + +-- Move current line or selected lines down with `L` +vim.keymap.set('n', 'J', ':m .+1==', { desc = 'Move line down', silent = true }) +vim.keymap.set('v', 'J', ":m '>+1gv=gv", { desc = 'Move selection down', silent = true }) +require('comfy-line-numbers').setup() diff --git a/lua/kickstart/plugins/neo-tree.lua b/lua/kickstart/plugins/neo-tree.lua index c7067891df0..5e5f1f1616d 100644 --- a/lua/kickstart/plugins/neo-tree.lua +++ b/lua/kickstart/plugins/neo-tree.lua @@ -14,6 +14,9 @@ return { { '\\', ':Neotree reveal', desc = 'NeoTree reveal', silent = true }, }, opts = { + window = { + position = 'float', -- ← default pane on the right + }, filesystem = { window = { mappings = {