Review and cleanup

This commit is contained in:
Jonas Widen 2025-03-18 18:07:58 +01:00
parent f34bd77732
commit 2611a0624b

View File

@ -10,6 +10,8 @@ local current_suggestion = {
last_request = 0, last_request = 0,
cache = {}, cache = {},
debounce_ms = 1000, -- Wait 1 second between requests debounce_ms = 1000, -- Wait 1 second between requests
min_chars = 3, -- Minimum characters before triggering completion
max_context_lines = 10, -- Maximum number of context lines to send
} }
-- Debug function -- Debug function
@ -168,6 +170,20 @@ function M.accept_suggestion()
return true return true
end end
-- Helper function to get relevant context around cursor
local function get_context_around_cursor(lines, current_line, max_lines)
local context = {}
local start_line = math.max(1, current_line - math.floor(max_lines/2))
local end_line = math.min(#lines, current_line + math.floor(max_lines/2))
for i = start_line, end_line do
if lines[i] and #lines[i] > 0 then
table.insert(context, string.format("L%d: %s", i, lines[i]))
end
end
return context
end
function M.trigger_completion() function M.trigger_completion()
debug_print("Triggering completion...") debug_print("Triggering completion...")
@ -176,148 +192,62 @@ function M.trigger_completion()
vim.fn.timer_stop(current_suggestion.timer) vim.fn.timer_stop(current_suggestion.timer)
end end
-- Set up debounce timer -- Set up debounce timer with shorter delay
current_suggestion.timer = vim.fn.timer_start(300, function() current_suggestion.timer = vim.fn.timer_start(150, function()
debug_print("Timer fired, checking conditions...")
-- Get current buffer and cursor info
local cursor = vim.api.nvim_win_get_cursor(0) local cursor = vim.api.nvim_win_get_cursor(0)
local line = cursor[1] local line = cursor[1]
local col = cursor[2] local col = cursor[2]
-- Get entire buffer content
local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false) local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false)
local current_line = lines[line] local current_line = lines[line]
debug_print("Current line: %s, col: %d", current_line or "nil", col) -- Get text before cursor
-- Don't trigger completion if line is empty
if not current_line or current_line:match("^%s*$") then
-- Instead of skipping, let's ask for what should come next
local context = {}
local cursor_line = line
-- Add previous lines for context
for i = 1, line - 1 do
if lines[i] and #lines[i] > 0 then
table.insert(context, string.format("L%d: %s", i, lines[i]))
end
end
-- Add empty current line with cursor position
table.insert(context, string.format("L%d (current): ", cursor_line))
table.insert(context, "^")
-- Add following lines for context
for i = line + 1, #lines do
if lines[i] and #lines[i] > 0 then
table.insert(context, string.format("L%d: %s", i, lines[i]))
end
end
local full_context = table.concat(context, "\n")
if #full_context == 0 then
debug_print("Empty context, skipping completion request")
return
end
-- Special prompt for empty line suggestions
local prompt = string.format([[I'm editing a %s file. Here's the context around line %d where my cursor (^) is on an empty line:
%s
Based on the surrounding code context, what would be the most appropriate code to add at the cursor position?
Return ONLY the raw code without any markdown formatting, language hints, or code blocks.]], vim.bo.filetype, cursor_line, full_context)
-- Get completion from Gemini
api.get_response(prompt, nil, function(response, error)
if error then
debug_print("Completion error: %s", error)
return
end
if type(response) == "string" and #response > 0 then
vim.schedule(function()
-- Check if cursor position is still valid
local new_cursor = vim.api.nvim_win_get_cursor(0)
if new_cursor[1] == line then
show_suggestion(response, 0) -- Start from column 0 since line is empty
end
end)
end
end)
return
end
-- Get text before cursor on current line
local prefix = string.sub(current_line, 1, col) local prefix = string.sub(current_line, 1, col)
debug_print("Prefix: %s", prefix) -- Don't trigger if we're in comments or strings
local syntax_group = vim.fn.synIDattr(vim.fn.synID(line, col, 1), "name")
-- Modified conditions: only skip if the line is completely empty if syntax_group:match("Comment") or syntax_group:match("String") then
-- or if we're in the middle of whitespace debug_print("Skipping: in comment or string")
if prefix:match("^%s*$") then
debug_print("Skipping: empty prefix")
clear_suggestion() clear_suggestion()
return return
end end
debug_print("Preparing context...") -- Check minimum character threshold
-- Construct context with line numbers and cursor position local trimmed_prefix = vim.trim(prefix)
local context = {} if #trimmed_prefix < current_suggestion.min_chars then
local cursor_line = line debug_print("Skipping: not enough characters")
clear_suggestion()
-- Add all previous lines for context
for i = 1, line - 1 do
if lines[i] and #lines[i] > 0 then
table.insert(context, string.format("L%d: %s", i, lines[i]))
end
end
-- Add current line with cursor position marker (^)
if current_line and #current_line > 0 then
local cursor_marker = string.rep(" ", col) .. "^"
table.insert(context, string.format("L%d (current): %s", cursor_line, current_line))
table.insert(context, cursor_marker)
end
-- Add all remaining lines
for i = line + 1, #lines do
if lines[i] and #lines[i] > 0 then
table.insert(context, string.format("L%d: %s", i, lines[i]))
end
end
-- Combine all lines and validate
local full_context = table.concat(context, "\n")
if #full_context == 0 then
debug_print("Empty context, skipping completion request")
return return
end end
-- Get file type for better context -- Get focused context around cursor
local context = get_context_around_cursor(lines, line, current_suggestion.max_context_lines)
-- Add cursor position marker
local cursor_marker = string.rep(" ", col) .. "^"
table.insert(context, string.format("L%d (current): %s", line, current_line))
table.insert(context, cursor_marker)
local full_context = table.concat(context, "\n")
local file_type = vim.bo.filetype local file_type = vim.bo.filetype
-- Construct prompt for Gemini -- Improved prompt for better completions
local prompt = string.format([[I'm editing a %s file. Here's the context around line %d where my cursor (^) is: local prompt = string.format([[In this %s file, complete the code at the cursor (^) position:
%s %s
Complete the code at the cursor position. Consider the syntax and style of the surrounding code. Return ONLY the completion text that would naturally continue from the cursor position.
Return ONLY the raw completion text without any markdown formatting, language hints, or code blocks. Focus on completing the current statement or block.
The completion can be multiple lines if appropriate for the context.]], file_type, cursor_line, full_context) Consider the syntax, style, and patterns in the surrounding code.]], file_type, full_context)
-- Check cache first -- Check cache and rate limiting
local cache_key = get_cache_key(prompt) local cache_key = get_cache_key(prompt)
if current_suggestion.cache[cache_key] then if current_suggestion.cache[cache_key] then
debug_print("Using cached completion")
show_suggestion(current_suggestion.cache[cache_key], col) show_suggestion(current_suggestion.cache[cache_key], col)
return return
end end
-- Check rate limiting
if should_rate_limit() then if should_rate_limit() then
debug_print("Rate limited, skipping completion request") debug_print("Rate limited")
return return
end end
@ -329,13 +259,14 @@ The completion can be multiple lines if appropriate for the context.]], file_typ
end end
if type(response) == "string" and #response > 0 then if type(response) == "string" and #response > 0 then
-- Cache the response -- Clean up response
response = vim.trim(response)
current_suggestion.cache[cache_key] = response current_suggestion.cache[cache_key] = response
vim.schedule(function() vim.schedule(function()
-- Check if cursor position is still valid -- Verify cursor position hasn't changed significantly
local new_cursor = vim.api.nvim_win_get_cursor(0) local new_cursor = vim.api.nvim_win_get_cursor(0)
if new_cursor[1] == line and new_cursor[2] >= col then if new_cursor[1] == line and math.abs(new_cursor[2] - col) <= 1 then
show_suggestion(response, col) show_suggestion(response, col)
end end
end) end)