fix(sidebar): fix handling of multiple windows when toggling full view (#2758)

This commit is contained in:
Dmitry Torokhov
2025-10-11 09:58:21 -07:00
committed by GitHub
parent 27f650d295
commit 5aafbdf3f9

View File

@@ -63,7 +63,6 @@ Sidebar.__index = Sidebar
---@field id integer ---@field id integer
---@field augroup integer ---@field augroup integer
---@field code avante.CodeState ---@field code avante.CodeState
---@field win_width_store table
---@field containers { result?: NuiSplit, todos?: NuiSplit, selected_code?: NuiSplit, selected_files?: NuiSplit, input?: NuiSplit } ---@field containers { result?: NuiSplit, todos?: NuiSplit, selected_code?: NuiSplit, selected_files?: NuiSplit, input?: NuiSplit }
---@field file_selector FileSelector ---@field file_selector FileSelector
---@field chat_history avante.ChatHistory | nil ---@field chat_history avante.ChatHistory | nil
@@ -85,13 +84,14 @@ Sidebar.__index = Sidebar
---@field tool_message_positions table<string, [integer, integer]> ---@field tool_message_positions table<string, [integer, integer]>
---@field skip_line_count integer | nil ---@field skip_line_count integer | nil
---@field current_tool_use_extmark_id integer | nil ---@field current_tool_use_extmark_id integer | nil
---@field private win_size_store table<integer, {width: integer, height: integer}>
---@field private is_in_full_view boolean
---@param id integer the tabpage id retrieved from api.nvim_get_current_tabpage() ---@param id integer the tabpage id retrieved from api.nvim_get_current_tabpage()
function Sidebar:new(id) function Sidebar:new(id)
return setmetatable({ return setmetatable({
id = id, id = id,
code = { bufnr = 0, winid = 0, selection = nil, old_winhl = nil }, code = { bufnr = 0, winid = 0, selection = nil, old_winhl = nil },
win_width_store = {},
winids = { winids = {
result_container = 0, result_container = 0,
todos_container = 0, todos_container = 0,
@@ -120,6 +120,8 @@ function Sidebar:new(id)
tool_message_positions = {}, tool_message_positions = {},
expanded_message_ids = {}, expanded_message_ids = {},
current_tool_use_extmark_id = nil, current_tool_use_extmark_id = nil,
win_width_store = {},
is_in_full_view = false,
}, Sidebar) }, Sidebar)
end end
@@ -151,7 +153,6 @@ function Sidebar:reset()
self:delete_containers() self:delete_containers()
self.win_width_store = {}
self.code = { bufnr = 0, winid = 0, selection = nil } self.code = { bufnr = 0, winid = 0, selection = nil }
self.scroll = true self.scroll = true
self.old_result_lines = {} self.old_result_lines = {}
@@ -159,6 +160,8 @@ function Sidebar:reset()
self.tool_message_positions = {} self.tool_message_positions = {}
self.expanded_message_uuids = {} self.expanded_message_uuids = {}
self.current_tool_use_extmark_id = nil self.current_tool_use_extmark_id = nil
self.win_size_store = {}
self.is_in_full_view = false
end end
---@class SidebarOpenOptions: AskOptions ---@class SidebarOpenOptions: AskOptions
@@ -258,6 +261,10 @@ end
function Sidebar:close(opts) function Sidebar:close(opts)
opts = vim.tbl_extend("force", { goto_code_win = true }, opts or {}) opts = vim.tbl_extend("force", { goto_code_win = true }, opts or {})
-- If sidebar was maximized make it normal size so that other windows
-- will not be left minimized.
if self.is_in_full_view then self:toggle_code_window() end
self:delete_autocmds() self:delete_autocmds()
self:delete_containers() self:delete_containers()
@@ -1612,31 +1619,44 @@ function Sidebar:render_logo()
end end
function Sidebar:toggle_code_window() function Sidebar:toggle_code_window()
local winids = api.nvim_tabpage_list_wins(self.id) -- Collect all windows that do not belong to the sidebar
local container_winids = vim.tbl_map(function(container) return container.winid end, self.containers) local winids = vim
local win_width = api.nvim_win_get_width(self.code.winid) .iter(api.nvim_tabpage_list_wins(self.id))
if win_width == 0 then :filter(function(winid) return not self:is_sidebar_winid(winid) end)
self.is_in_full_view = false :totable()
if self.is_in_full_view then
-- Transitioning to normal view: restore sizes of all non-sidebar windows
for _, winid in ipairs(winids) do for _, winid in ipairs(winids) do
if api.nvim_win_is_valid(winid) and not vim.tbl_contains(container_winids, winid) then local old_size = self.win_size_store[winid]
local old_width = self.win_width_store[winid] if old_size then
if old_width ~= nil then api.nvim_win_set_width(winid, old_width) end api.nvim_win_set_width(winid, old_size.width)
api.nvim_win_set_height(winid, old_size.height)
end end
end end
else else
self.is_in_full_view = true -- Transitioning to full view: hide all non-sidebar windows
-- We need do this in 2 phases: first phase is to collect window sizes
-- and 2nd phase is to actually maximize the sidebar. If we attempt to do
-- everything is one pass sizes of windows may change in the process and
-- we'll end up with a mess.
self.win_size_store = {}
for _, winid in ipairs(winids) do for _, winid in ipairs(winids) do
if api.nvim_win_is_valid(winid) and not vim.tbl_contains(container_winids, winid) then if Utils.is_floating_window(winid) then
if Utils.is_floating_window(winid) then api.nvim_win_close(winid, true)
api.nvim_win_close(winid, true) else
else self.win_size_store[winid] = { width = api.nvim_win_get_width(winid), height = api.nvim_win_get_height(winid) }
local width = api.nvim_win_get_width(winid)
self.win_width_store[winid] = width
api.nvim_win_set_width(winid, 0)
end
end end
end end
if self:get_layout() == "vertical" then
api.nvim_win_set_width(self.containers.result.winid, vim.o.columns)
else
api.nvim_win_set_height(self.containers.result.winid, vim.o.lines)
end
end end
self.is_in_full_view = not self.is_in_full_view
end end
--- Initialize the sidebar instance. --- Initialize the sidebar instance.