UpdateLastColumn failed to find bufnr with vim-plug
Closed this issue Β· 23 comments
Hmm, thank you for trying, I appreciate you downloading vim-plug in order to try it. What's weird to me is, this never used to happen. It has only started happening with vim-plug and vim-bufkill very recently π€
@qpkorr I can give you another case to test this error, I'm using nvim and trying to make vim-plug to open in a floating window instead of a side panel. You can paste this in the init.vim
file:
function! FloatingPlug()
" Style of the floating window
let height = &lines - 3
let width = float2nr(&columns - (&columns * 2 / 10))
let col = float2nr((&columns - width) / 2)
let opts = {
\ 'relative': 'editor',
\ 'row': height * 0.3,
\ 'col': col,
\ 'width': width * 2 / 2,
\ 'height': height / 2
\ }
let buf = nvim_create_buf(v:false, v:true)
let win = nvim_open_win(buf, v:true, opts)
" Set Floating Window Highlighting
call setwinvar(win, '&winhl', 'NormalFloat:TabLine')
" this is to remove all line numbers and so on from the window
setlocal
\ buftype=nofile
\ nobuflisted
\ modifiable
\ bufhidden=hide
\ nonumber
\ norelativenumber
\ signcolumn=no
endfunction
let g:plug_window = 'call FloatingPlug()'
Every time you run :PlugInstall
a new floating window will open in the middle of the screen and if you press q
to close it, then you'll get the error described at the beginning.
I can second the floating window case. I hit that the other day and havenβt had the time to find a minimal repro yet. Thanks for posting one!
i also have the same problem when i do :Files from fzf_vim and press escape or Ctrl-C
i dont get why bufkill is even being called... i disabled all mapping i only use :BD sometimes
@dagadbm in order for :BD to work when you do use it, it has to track every time you open, close, or change buffer in a window, so it can have an up to date list of your most recent buffer usage, to know which buffer to display when you close one. So it acts upon a number of buffer events.
Thanks for the minimal error cases, this problem has been there a long time, it would be nice to finally fix it - though I've got a new baby, and am currently organizing a memorial for a close family member's recent death - so don't hold your breath. I'd always hoped that base vim would get this functionality built in so my little plugin could go the way of the dodo years ago... but seemingly that hasn't yet happened, alas.
@qpkorr don't worry, your plugin is still really useful. Take your time and hopefully, in the future you'll have more time to spent looking at this or maybe knows how to give a check and try to get a fix with all the information here.
I just wanted to say that I use this plugin A LOT, :BP
, :BB
and :BD
are precious gems for me. Well done π and thank you π
Regarding this bug, it also happens for me when using fzf in a floating window in neovim and I think I know the problem.
In the above scenario, the buffer name (bufname('%')
) is empty so we can't find the index to save in the list inside the <SID>UpdateLastColumn(event)
function.
I fixed that by adding this check to this function:
function! <SID>UpdateLastColumn(event) "{{{1
" Function to save the current column and buffer and window numbers,
let DebugF = 'UpdateColumnList'
call s:Debug(1, DebugF, 'Entering(' . a:event . '): win = ' . winnr() . ', buf = ' . bufnr('%') . ' (' . bufname('%') . ')')
call s:Debug(2, DebugF, 'w:BufKillList')
call s:Debug(2, DebugF, 'w:BufKillColumnList')
call s:Debug(2, DebugF, 'w:BufKillIndex')
if !bufname('%') " <<<<<<<<<<<<<<<<<<<< THIS CHECK HERE <<<<<<<<<<<<<<<<<<<
call s:Debug(1, DebugF, 'Buffer name is empty, nothing to do here')
return
endif
if !exists('w:BufKillList')
" Just give up for now.
return
endif
let index = index(w:BufKillList, bufnr('%'))
if index != -1
" Extend list if required, then set the value
let w:BufKillColumnList += repeat([0], index - len(w:BufKillColumnList) + 1)
let w:BufKillColumnList[index] = col('.')
else
echom 'UpdateLastColumn failed to find bufnr ' . bufnr('%') . ' in w:BufKillList'
endif
call s:Debug(2, DebugF, 'w:BufKillList')
call s:Debug(2, DebugF, 'w:BufKillColumnList')
call s:Debug(2, DebugF, 'w:BufKillIndex')
call s:Debug(1, DebugF, 'Exiting (' . a:event . ')')
" redraw " To hide call Debug messages for now!
endfunction
I'm going to open a PR to add that is that's ok π π
thanks for investigating this issue, just to add, I see a few error messages in the status bar if I spawn a floating window for fzf and terminal window or tmux pane is too small. Not sure if is related to this issue or a different thing.
Thanks for the PR, will try and confirm if it solves the problem in different scenarios. π
Awesome, thanks @Gioyik .
This bug also happens when running :tabnew
for me. I just opened the PR =)
Hi @lucasprag - thanks so much! I still feel like it's ridiculously complex code for what it's trying to achieve, and I'm super impressed when anyone else manages to understand it.
My only curiousity - without yet having had time to check out the code myself - is whether bufname being empty should be such a problem. The plugin works (should work...?) mainly with bufnums - the same bufnum can be opened in multiple windows, but can one open the same file with two different bufnums? I thought not. Hence - is the problem that bufname is empty - or that we have a new bufnum that hasn't yet been added to w:BuffKillList, but should have been? Or should these 'anonymous buffers' be rightly excluded from this buffer handling... it's too long ago for me to recall.
Either way your change is clearly an improvement :) - but do you think there's more to do?
Awesome, thanks =) Like I said, it's a great plugin and I use it everyday, great work π
Interesting, I'm trying to understand it further. It doesn't seem to happen on :tabnew like I said, but notice the different between :tabnew and opening a new floating window (in my case using fzf.vim like :Files)
Now with level 2 debugging enabled:
Not sure if this helps, but notice that for the floating window, no buffer events are triggered. BufEnter, BufAdd nor BufLeave. It should be ok since we have the same functions mapped to the window events like WinEnter and WinLeave.
Anyway, I'm still trying to understand what happens with floating windows =)
Glad you like the plugin and thanks - though it still disappoints me that it's not the standard, built in behaviour - like it is in emacs. Anyway - yay for customizability.
Thanks for your observations. That does certainly explain it I suppose (as best I understand it...) - no BufAdd etc means the lists don't get populated, so of course we can't find the buffer in the lists. In which case - the error might be expecting the lists to be populated by all buffers, and thus we should just stop issuing a warning in this scenario, just return. The question of whether we should or shouldn't include this buffer in the lists is moot if we don't know how to :)
One other minor consideration - I just noticed that the description of BufAdd includes the comment: "NOTE: When this autocommand is executed, the current buffer "%" may be different from the buffer being created """
So calling bufnr("%") as I do may not work as expected... hmmm
Oh this is really interesting, I tried to find a way to populate the list when on WinNew, but it wasn't working and I don't know why even though the event was been called, also no notes on :h WinNew
.
I think just returning when index is -1 is good due to floating windows. Is that ok if I update my PR?
wait, shouldn't the list get populated on WinEnter
?
Thanks @lucasprag. I'm hopeful this is a satisfactory approach - but in case there's more going on here (eg, bufnr('%') isn't set yet on a BufAdd event) and we get bitten in the future, rather that deleting the warning, I'd rather convert it to a debug message. Code updated - fingers crossed - and thanks everyone for your help (and patience!)
Awesome, I just updated and it works well, not seeing any messages when opening fzf on a floating window, thanks π π
can confirm is not happening to me anymore, everything looks right and clean.
maybe this can close now?
Thanks @Gioyik, @lucasprag et al.