Skip to content

Commit eefb897

Browse files
committed
fix(ui): special handling for floats closed before VimEnter. Seems that WinClosed events dont execute before that. Fixes #1390
1 parent d37a76b commit eefb897

File tree

3 files changed

+89
-26
lines changed

3 files changed

+89
-26
lines changed

lua/lazy/util.lua

+30
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,36 @@ function M.throttle(ms, fn)
9898
end
9999
end
100100

101+
--- Creates a weak reference to an object.
102+
--- Calling the returned function will return the object if it has not been garbage collected.
103+
---@generic T: table
104+
---@param obj T
105+
---@return T|fun():T?
106+
function M.weak(obj)
107+
local weak = { _obj = obj }
108+
---@return table<any, any>
109+
local function get()
110+
local ret = rawget(weak, "_obj")
111+
return ret == nil and error("Object has been garbage collected", 2) or ret
112+
end
113+
local mt = {
114+
__mode = "v",
115+
__call = function(t)
116+
return rawget(t, "_obj")
117+
end,
118+
__index = function(_, k)
119+
return get()[k]
120+
end,
121+
__newindex = function(_, k, v)
122+
get()[k] = v
123+
end,
124+
__pairs = function()
125+
return pairs(get())
126+
end,
127+
}
128+
return setmetatable(weak, mt)
129+
end
130+
101131
---@class LazyCmdOptions: LazyFloatOptions
102132
---@field cwd? string
103133
---@field env? table<string,string>

lua/lazy/view/float.lua

+50-19
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ local ViewConfig = require("lazy.view.config")
2424
---@field win_opts LazyWinOpts
2525
---@field backdrop_buf number
2626
---@field backdrop_win number
27+
---@field id number
2728
---@overload fun(opts?:LazyFloatOptions):LazyFloat
2829
local M = {}
2930

@@ -33,6 +34,12 @@ setmetatable(M, {
3334
end,
3435
})
3536

37+
local _id = 0
38+
local function next_id()
39+
_id = _id + 1
40+
return _id
41+
end
42+
3643
---@param opts? LazyFloatOptions
3744
function M.new(opts)
3845
local self = setmetatable({}, { __index = M })
@@ -42,6 +49,7 @@ end
4249
---@param opts? LazyFloatOptions
4350
function M:init(opts)
4451
require("lazy.view.colors").setup()
52+
self.id = next_id()
4553
self.opts = vim.tbl_deep_extend("force", {
4654
size = Config.options.ui.size,
4755
style = "minimal",
@@ -65,8 +73,13 @@ function M:init(opts)
6573
title_pos = self.opts.title and self.opts.title_pos or nil,
6674
}
6775
self:mount()
68-
self:on_key(ViewConfig.keys.close, self.close)
69-
self:on({ "WinLeave", "BufDelete", "BufHidden" }, self.close, { once = false })
76+
self:on("VimEnter", function()
77+
vim.schedule(function()
78+
if not self:win_valid() then
79+
self:close()
80+
end
81+
end)
82+
end, { buffer = false })
7083
return self
7184
end
7285

@@ -138,7 +151,13 @@ function M:mount()
138151

139152
self:layout()
140153
self.win = vim.api.nvim_open_win(self.buf, true, self.win_opts)
154+
self:on("WinClosed", function()
155+
self:close()
156+
self:augroup(true)
157+
end, { win = true })
141158
self:focus()
159+
self:on_key(ViewConfig.keys.close, self.close)
160+
self:on({ "BufDelete", "BufHidden" }, self.close)
142161

143162
if vim.bo[self.buf].buftype == "" then
144163
vim.bo[self.buf].buftype = "nofile"
@@ -185,27 +204,38 @@ function M:mount()
185204
})
186205
end
187206

207+
---@param clear? boolean
208+
function M:augroup(clear)
209+
return vim.api.nvim_create_augroup("trouble.window." .. self.id, { clear = clear == true })
210+
end
211+
188212
---@param events string|string[]
189-
---@param fn fun(self?):boolean?
190-
---@param opts? table
213+
---@param fn fun(self:LazyFloat, event:{buf:number}):boolean?
214+
---@param opts? vim.api.keyset.create_autocmd | {buffer: false, win?:boolean}
191215
function M:on(events, fn, opts)
192-
if type(events) == "string" then
193-
events = { events }
216+
opts = opts or {}
217+
if opts.win then
218+
opts.pattern = self.win .. ""
219+
opts.win = nil
220+
elseif opts.buffer == nil then
221+
opts.buffer = self.buf
222+
elseif opts.buffer == false then
223+
opts.buffer = nil
194224
end
195-
for _, e in ipairs(events) do
196-
local event, pattern = e:match("(%w+) (%w+)")
197-
event = event or e
198-
vim.api.nvim_create_autocmd(
199-
event,
200-
vim.tbl_extend("force", {
201-
pattern = pattern,
202-
buffer = (not pattern) and self.buf or nil,
203-
callback = function()
204-
return fn(self)
205-
end,
206-
}, opts or {})
207-
)
225+
if opts.pattern then
226+
opts.buffer = nil
227+
end
228+
local _self = Util.weak(self)
229+
opts.callback = function(e)
230+
local this = _self()
231+
if not this then
232+
-- delete the autocmd
233+
return true
234+
end
235+
return fn(this, e)
208236
end
237+
opts.group = self:augroup()
238+
vim.api.nvim_create_autocmd(events, opts)
209239
end
210240

211241
---@param key string
@@ -223,6 +253,7 @@ end
223253

224254
---@param opts? {wipe:boolean}
225255
function M:close(opts)
256+
self:augroup(true)
226257
local buf = self.buf
227258
local win = self.win
228259
local wipe = opts and opts.wipe

lua/lazy/view/init.lua

+9-7
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ function M.create()
5353
Float.init(self, {
5454
title = Config.options.ui.title,
5555
title_pos = Config.options.ui.title_pos,
56-
noautocmd = true,
56+
noautocmd = false,
5757
})
5858

5959
if Config.options.ui.wrap then
@@ -69,12 +69,14 @@ function M.create()
6969
self.render = Render.new(self)
7070
self.update = Util.throttle(Config.options.ui.throttle, self.update)
7171

72-
self:on({ "User LazyRender", "User LazyFloatResized" }, function()
73-
if not (self.buf and vim.api.nvim_buf_is_valid(self.buf)) then
74-
return true
75-
end
76-
self:update()
77-
end)
72+
for _, pattern in ipairs({ "LazyRender", "LazyFloatResized" }) do
73+
self:on({ "User" }, function()
74+
if not (self.buf and vim.api.nvim_buf_is_valid(self.buf)) then
75+
return true
76+
end
77+
self:update()
78+
end, { pattern = pattern })
79+
end
7880

7981
vim.keymap.set("n", ViewConfig.keys.abort, function()
8082
require("lazy.manage.process").abort()

0 commit comments

Comments
 (0)