gemini/lua/gemini/chat.lua
2025-03-16 19:30:15 +01:00

182 lines
6.2 KiB
Lua

local config = require("gemini.config")
local M = {}
-- Store chat state
local state = {
bufnr = nil,
winnr = nil,
current_context = nil,
conversation_history = {}
}
local function setup_chat_highlighting(bufnr)
vim.api.nvim_buf_set_option(bufnr, 'syntax', '')
pcall(vim.treesitter.start, bufnr, "markdown")
vim.cmd([[
highlight default GeminiUser guifg=#EBCB8B gui=bold
highlight default GeminiSeparator guifg=#616E88 gui=bold
syntax match GeminiUser /^User:.*$/
syntax match GeminiSeparator /^━━━━━━━━━━━━━━━━━━━━━━━━━━$/
]])
end
local function setup_buffer_options(bufnr)
local options = {
buftype = 'nofile',
filetype = 'markdown',
buflisted = false,
swapfile = false,
bufhidden = 'wipe',
modifiable = false,
}
for option, value in pairs(options) do
vim.api.nvim_buf_set_option(bufnr, option, value)
end
end
local function setup_buffer_autocmds(bufnr)
local augroup = vim.api.nvim_create_augroup('GeminiChatBuffer', { clear = true })
vim.api.nvim_create_autocmd({'BufReadCmd', 'FileReadCmd', 'BufWriteCmd'}, {
group = augroup,
buffer = bufnr,
callback = function()
vim.notify('File operations are not allowed in the chat window', vim.log.levels.WARN)
return true
end
})
end
local function setup_buffer_keymaps(bufnr)
local opts = { buffer = bufnr, nowait = true }
-- Disable file operations
local operations = {'e', 'edit', 'w', 'write', 'sp', 'split', 'vs', 'vsplit',
'new', 'vnew', 'read', 'update', 'saveas'}
for _, op in ipairs(operations) do
vim.keymap.set('n', ':' .. op, function()
vim.notify('Operation not allowed in chat window', vim.log.levels.WARN)
end, opts)
end
-- Disable command mode
vim.keymap.set('n', ':', function()
vim.notify('Command mode disabled in chat window', vim.log.levels.WARN)
end, opts)
-- Set chat-specific keymaps
local mappings = config.options.mappings
vim.keymap.set('n', mappings.close, function()
vim.api.nvim_win_close(state.winnr, true)
state.winnr = nil
end, opts)
vim.keymap.set('n', mappings.return_focus, function()
vim.cmd('wincmd p')
end, opts)
vim.keymap.set('n', mappings.new_query, function()
require("gemini").prompt_query(state.current_context)
end, opts)
end
function M.create_window()
if not state.bufnr or not vim.api.nvim_buf_is_valid(state.bufnr) then
state.bufnr = vim.api.nvim_create_buf(false, true)
setup_buffer_options(state.bufnr)
setup_buffer_autocmds(state.bufnr)
setup_buffer_keymaps(state.bufnr)
end
local win_opts = config.options.window
if not state.winnr or not vim.api.nvim_win_is_valid(state.winnr) then
state.winnr = vim.api.nvim_open_win(state.bufnr, true, {
relative = "editor",
width = win_opts.width(),
height = win_opts.height(),
row = 0,
col = vim.o.columns - win_opts.width(),
border = win_opts.border,
title = win_opts.title,
title_pos = win_opts.title_pos,
style = "minimal"
})
vim.api.nvim_win_set_option(state.winnr, 'wrap', true)
vim.api.nvim_win_set_option(state.winnr, 'linebreak', true)
vim.api.nvim_win_set_option(state.winnr, 'breakindent', true)
setup_chat_highlighting(state.bufnr)
end
end
function M.update_content(content, is_new_chat, is_thinking)
vim.api.nvim_buf_set_option(state.bufnr, 'modifiable', true)
local lines = vim.split(content, "\n")
if is_thinking then
-- For thinking message, just replace/append without affecting history
vim.api.nvim_buf_set_lines(state.bufnr, 0, -1, false, lines)
else
if is_new_chat then
-- For new messages, show the complete history
local display_lines = {}
for i, msg in ipairs(state.conversation_history) do
if i > 1 then -- Add separator before messages (except first)
table.insert(display_lines, "━━━━━━━━━━━━━━━━━━━━━━━━━━")
end
local msg_lines = vim.split(msg.content, "\n")
for _, line in ipairs(msg_lines) do
table.insert(display_lines, line)
end
end
-- Add the new content
if #state.conversation_history > 0 then
table.insert(display_lines, "━━━━━━━━━━━━━━━━━━━━━━━━━━")
end
for _, line in ipairs(lines) do
table.insert(display_lines, line)
end
vim.api.nvim_buf_set_lines(state.bufnr, 0, -1, false, display_lines)
else
-- For subsequent messages in the same chat
if #state.conversation_history > 0 then
table.insert(lines, 1, "━━━━━━━━━━━━━━━━━━━━━━━━━━")
end
vim.api.nvim_buf_set_lines(state.bufnr, -1, -1, false, lines)
end
end
vim.api.nvim_buf_set_option(state.bufnr, 'modifiable', false)
-- Scroll to show new content
local line_count = vim.api.nvim_buf_line_count(state.bufnr)
local content_lines = #vim.split(content, "\n")
local start_line = line_count - content_lines + 1
vim.api.nvim_win_set_cursor(state.winnr, {start_line, 0})
vim.cmd('normal! zt')
vim.cmd('wincmd p')
end
function M.set_context(context)
state.current_context = context
end
function M.clear()
if state.bufnr and vim.api.nvim_buf_is_valid(state.bufnr) then
vim.api.nvim_buf_set_lines(state.bufnr, 0, -1, false, {})
end
state.conversation_history = {} -- Clear conversation history
end
-- Add functions to manage conversation history
function M.add_message(role, content)
table.insert(state.conversation_history, {role = role, content = content})
end
function M.get_conversation_history()
return state.conversation_history
end
return M