Added completion
This commit is contained in:
parent
9eda1d231a
commit
10557a6fcf
@ -1,15 +1,21 @@
|
|||||||
local api = require("gemini.api")
|
local api = require("gemini.api")
|
||||||
local M = {}
|
local M = {}
|
||||||
|
|
||||||
-- State for managing current suggestion
|
-- State for managing current suggestion and debounce timer
|
||||||
local current_suggestion = {
|
local current_suggestion = {
|
||||||
text = nil,
|
text = nil,
|
||||||
start_col = nil,
|
start_col = nil,
|
||||||
namespace_id = vim.api.nvim_create_namespace('gemini_suggestion')
|
namespace_id = vim.api.nvim_create_namespace('gemini_suggestion'),
|
||||||
|
timer = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
-- Helper function to clear current suggestion
|
-- Helper function to clear suggestion
|
||||||
local function clear_suggestion()
|
local function clear_suggestion()
|
||||||
|
if current_suggestion.timer then
|
||||||
|
vim.fn.timer_stop(current_suggestion.timer)
|
||||||
|
current_suggestion.timer = nil
|
||||||
|
end
|
||||||
|
|
||||||
if current_suggestion.text then
|
if current_suggestion.text then
|
||||||
vim.api.nvim_buf_clear_namespace(0, current_suggestion.namespace_id, 0, -1)
|
vim.api.nvim_buf_clear_namespace(0, current_suggestion.namespace_id, 0, -1)
|
||||||
current_suggestion.text = nil
|
current_suggestion.text = nil
|
||||||
@ -39,8 +45,8 @@ local function show_suggestion(suggestion, start_col)
|
|||||||
-- Make the text virtual (doesn't affect actual buffer content)
|
-- Make the text virtual (doesn't affect actual buffer content)
|
||||||
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 = {{suggestion, 'GeminiSuggestion'}},
|
||||||
virt_text_pos = 'inline', -- Changed from 'overlay' to 'inline'
|
virt_text_pos = 'inline',
|
||||||
virt_text_hide = true, -- Hide when there's other virtual text
|
virt_text_hide = true,
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -72,47 +78,65 @@ function M.accept_suggestion()
|
|||||||
end
|
end
|
||||||
|
|
||||||
function M.trigger_completion()
|
function M.trigger_completion()
|
||||||
local cursor = vim.api.nvim_win_get_cursor(0)
|
-- Clear any existing timer
|
||||||
local line = cursor[1]
|
if current_suggestion.timer then
|
||||||
local col = cursor[2]
|
vim.fn.timer_stop(current_suggestion.timer)
|
||||||
|
|
||||||
-- Get entire buffer content
|
|
||||||
local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false)
|
|
||||||
|
|
||||||
-- Split into before and after cursor
|
|
||||||
local before_lines = {}
|
|
||||||
local after_lines = {}
|
|
||||||
|
|
||||||
-- Copy lines before cursor
|
|
||||||
for i = 1, line - 1 do
|
|
||||||
table.insert(before_lines, lines[i])
|
|
||||||
end
|
end
|
||||||
-- Add current line up to cursor
|
|
||||||
local current_line = lines[line]
|
|
||||||
table.insert(before_lines, string.sub(current_line, 1, col))
|
|
||||||
|
|
||||||
-- Combine all lines before cursor
|
-- Start a new timer for debouncing
|
||||||
local prefix = table.concat(before_lines, "\n")
|
current_suggestion.timer = vim.fn.timer_start(100, function()
|
||||||
|
local cursor = vim.api.nvim_win_get_cursor(0)
|
||||||
|
local line = cursor[1]
|
||||||
|
local col = cursor[2]
|
||||||
|
|
||||||
-- Construct prompt for Gemini
|
-- Get entire buffer content
|
||||||
local prompt = string.format(
|
local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false)
|
||||||
"Complete this code. Only provide the completion, no explanation:\n%s",
|
local current_line = lines[line]
|
||||||
prefix
|
|
||||||
)
|
|
||||||
|
|
||||||
-- Get completion from Gemini
|
-- Don't trigger completion if line is empty or cursor is at the start
|
||||||
api.get_response(prompt, nil, function(response, error)
|
if col == 0 or current_line:match("^%s*$") then
|
||||||
if error then return end
|
clear_suggestion()
|
||||||
|
return
|
||||||
if type(response) == "string" then
|
|
||||||
-- Get first non-empty line as suggestion
|
|
||||||
local suggestion = response:match("^%s*(.-)%s*$")
|
|
||||||
if suggestion and #suggestion > 0 then
|
|
||||||
vim.schedule(function()
|
|
||||||
show_suggestion(suggestion, col)
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
-- Split into before and after cursor
|
||||||
|
local before_lines = {}
|
||||||
|
|
||||||
|
-- Copy lines before current line
|
||||||
|
for i = 1, line - 1 do
|
||||||
|
table.insert(before_lines, lines[i])
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Add current line (complete line for context)
|
||||||
|
table.insert(before_lines, current_line)
|
||||||
|
|
||||||
|
-- Combine all lines
|
||||||
|
local prefix = table.concat(before_lines, "\n")
|
||||||
|
|
||||||
|
-- Construct prompt for Gemini
|
||||||
|
local prompt = string.format(
|
||||||
|
"Complete this code. Only provide the completion, no explanation:\n%s",
|
||||||
|
prefix
|
||||||
|
)
|
||||||
|
|
||||||
|
-- Get completion from Gemini
|
||||||
|
api.get_response(prompt, nil, function(response, error)
|
||||||
|
if error then return end
|
||||||
|
|
||||||
|
if type(response) == "string" then
|
||||||
|
-- Get first non-empty line as suggestion
|
||||||
|
local suggestion = response:match("^%s*(.-)%s*$")
|
||||||
|
if suggestion and #suggestion > 0 then
|
||||||
|
vim.schedule(function()
|
||||||
|
-- Check if cursor position is still the same
|
||||||
|
local new_cursor = vim.api.nvim_win_get_cursor(0)
|
||||||
|
if new_cursor[1] == line and new_cursor[2] == col then
|
||||||
|
show_suggestion(suggestion, col)
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end)
|
||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -132,6 +156,13 @@ function M.setup()
|
|||||||
end
|
end
|
||||||
end, { expr = false, noremap = true })
|
end, { expr = false, noremap = true })
|
||||||
|
|
||||||
|
-- Set up autocommand for real-time completion
|
||||||
|
vim.api.nvim_create_autocmd("TextChangedI", {
|
||||||
|
callback = function()
|
||||||
|
M.trigger_completion()
|
||||||
|
end
|
||||||
|
})
|
||||||
|
|
||||||
-- Clear suggestion on cursor move
|
-- Clear suggestion on cursor move
|
||||||
vim.api.nvim_create_autocmd({'CursorMovedI', 'CursorMoved'}, {
|
vim.api.nvim_create_autocmd({'CursorMovedI', 'CursorMoved'}, {
|
||||||
callback = function()
|
callback = function()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user