debug completion

This commit is contained in:
Jonas Widen 2025-03-17 07:13:07 +01:00
parent 9456cbd902
commit fa37ffe285

View File

@ -50,37 +50,52 @@ end
local function show_suggestion(suggestion, start_col) local function show_suggestion(suggestion, start_col)
clear_suggestion() clear_suggestion()
-- Only use first line of suggestion -- Split suggestion into lines
suggestion = suggestion:match("^[^\n]*") local suggestion_lines = vim.split(suggestion, "\n")
if suggestion == "" then return end if #suggestion_lines == 0 then return end
-- Get current line info
local line = vim.api.nvim_win_get_cursor(0)[1] - 1 local line = vim.api.nvim_win_get_cursor(0)[1] - 1
local line_text = vim.api.nvim_buf_get_lines(0, line, line + 1, true)[1] local line_text = vim.api.nvim_buf_get_lines(0, line, line + 1, true)[1]
-- Get text before cursor on current line -- Get text before cursor on current line
local prefix = string.sub(line_text, 1, start_col) local prefix = string.sub(line_text, 1, start_col)
-- Only show suggestion if it continues the current text -- Process first line of suggestion
if not suggestion:find("^" .. vim.pesc(prefix), 1, true) then local first_line = suggestion_lines[1]
suggestion = prefix .. suggestion if not first_line:find("^" .. vim.pesc(prefix), 1, true) then
first_line = prefix .. first_line
end end
first_line = first_line:sub(#prefix + 1)
-- Remove the prefix from the suggestion to avoid duplication if first_line == "" and #suggestion_lines == 1 then return end
suggestion = suggestion:sub(#prefix + 1)
if suggestion == "" then return end
current_suggestion.text = suggestion current_suggestion.text = suggestion
current_suggestion.start_col = start_col current_suggestion.start_col = start_col
-- Make the text virtual (doesn't affect actual buffer content) -- Show first line as virtual text
if first_line ~= "" then
vim.api.nvim_buf_set_extmark(0, current_suggestion.namespace_id, line, start_col, { vim.api.nvim_buf_set_extmark(0, current_suggestion.namespace_id, line, start_col, {
virt_text = {{suggestion, 'GeminiSuggestion'}}, virt_text = {{first_line, 'GeminiSuggestion'}},
virt_text_pos = 'inline', virt_text_pos = 'inline',
virt_text_hide = true, virt_text_hide = true,
}) })
end
debug_print("Showing suggestion: %s", suggestion) -- Show remaining lines as virtual lines
if #suggestion_lines > 1 then
local virt_lines = {}
for i = 2, #suggestion_lines do
table.insert(virt_lines, {{suggestion_lines[i], 'GeminiSuggestion'}})
end
vim.api.nvim_buf_set_extmark(0, current_suggestion.namespace_id, line, start_col, {
virt_lines = virt_lines,
virt_lines_above = false,
})
end
debug_print("Showing suggestion (%d lines)", #suggestion_lines)
end end
function M.accept_suggestion() function M.accept_suggestion()
@ -96,17 +111,35 @@ function M.accept_suggestion()
return false return false
end end
-- Split suggestion into lines
local suggestion_lines = vim.split(current_suggestion.text, "\n")
if #suggestion_lines == 0 then return false end
-- Insert first line at cursor position
vim.api.nvim_buf_set_text( vim.api.nvim_buf_set_text(
0, 0,
line, line,
col, col,
line, line,
col, col,
{current_suggestion.text} {suggestion_lines[1]}
) )
-- Insert remaining lines below
if #suggestion_lines > 1 then
vim.api.nvim_buf_set_lines(
0,
line + 1,
line + 1,
false,
vim.list_slice(suggestion_lines, 2)
)
end
-- Move cursor to end of inserted text -- Move cursor to end of inserted text
vim.api.nvim_win_set_cursor(0, {line + 1, col + #current_suggestion.text}) local final_line = line + #suggestion_lines - 1
local final_col = #suggestion_lines[#suggestion_lines]
vim.api.nvim_win_set_cursor(0, {final_line + 1, final_col})
clear_suggestion() clear_suggestion()
return true return true
@ -164,27 +197,31 @@ function M.trigger_completion()
debug_print("Getting completion for: %s", prefix) debug_print("Getting completion for: %s", prefix)
-- Construct context from buffer -- Construct context with line numbers and cursor position
local context = {} local context = {}
local cursor_line = line
local cursor_col = col
-- Add up to 10 previous lines for context -- Add up to 10 previous lines for context
local start_line = math.max(1, line - 10) local start_line = math.max(1, line - 10)
for i = start_line, line - 1 do for i = start_line, line - 1 do
if lines[i] and #lines[i] > 0 then if lines[i] and #lines[i] > 0 then
table.insert(context, lines[i]) table.insert(context, string.format("L%d: %s", i, lines[i]))
end end
end end
-- Add current line -- Add current line with cursor position marker (^)
if current_line and #current_line > 0 then if current_line and #current_line > 0 then
table.insert(context, current_line) local cursor_marker = string.rep(" ", cursor_col) .. "^"
table.insert(context, string.format("L%d (current): %s", cursor_line, current_line))
table.insert(context, cursor_marker)
end end
-- Add up to 10 lines after current line -- Add up to 10 lines after current line
local end_line = math.min(#lines, line + 10) local end_line = math.min(#lines, line + 10)
for i = line + 1, end_line do for i = line + 1, end_line do
if lines[i] and #lines[i] > 0 then if lines[i] and #lines[i] > 0 then
table.insert(context, lines[i]) table.insert(context, string.format("L%d: %s", i, lines[i]))
end end
end end
@ -195,11 +232,19 @@ function M.trigger_completion()
return return
end end
-- Get file type for better context
local file_type = vim.bo.filetype
-- Construct prompt for Gemini -- Construct prompt for Gemini
local prompt = string.format( local prompt = string.format([[
"Complete this code. Consider the context (10 lines before and after). Return ONLY the completion that would naturally follow at the cursor position, no explanation:\n%s", I'm editing a %s file. Here's the context around line %d where my cursor (^) is:
full_context
) %s
Complete the code at the cursor position. Consider the syntax and style of the surrounding code.
Return ONLY the completion that would naturally follow, no explanation.
The completion can be multiple lines if appropriate for the context.
]], file_type, cursor_line, full_context)
-- Validate prompt before sending -- Validate prompt before sending
if #prompt == 0 then if #prompt == 0 then