fix(mouse): crash with click on win-separator in statusline (#33091)

Problem: Clicking on window separator in statusline crashes Nvim due
to out of bound memory access

Solution: Check if the click location is within clicking range before
applying it.

(cherry picked from commit 18fa61049a)
This commit is contained in:
Shadman
2025-03-28 13:09:02 +06:00
committed by github-actions[bot]
parent 6514e2c7ba
commit c61e8c6e70
2 changed files with 21 additions and 2 deletions

View File

@@ -655,6 +655,7 @@ bool do_mouse(oparg_T *oap, int c, int dir, int count, bool fixindent)
bool in_winbar = (jump_flags & MOUSE_WINBAR);
bool in_statuscol = (jump_flags & MOUSE_STATUSCOL);
bool in_status_line = (jump_flags & IN_STATUS_LINE);
bool in_global_statusline = in_status_line && global_stl_height() > 0;
bool in_sep_line = (jump_flags & IN_SEP_LINE);
if ((in_winbar || in_status_line || in_statuscol) && is_click) {
@@ -671,7 +672,7 @@ bool do_mouse(oparg_T *oap, int c, int dir, int count, bool fixindent)
: in_winbar ? wp->w_winbar_click_defs
: wp->w_statuscol_click_defs;
if (in_status_line && global_stl_height() > 0) {
if (in_global_statusline) {
// global statusline is displayed for the current window,
// and spans the whole screen.
click_defs = curwin->w_status_click_defs;
@@ -681,7 +682,11 @@ bool do_mouse(oparg_T *oap, int c, int dir, int count, bool fixindent)
if (in_statuscol && wp->w_p_rl) {
click_col = wp->w_width_inner - click_col - 1;
}
if (in_statuscol && click_col >= (int)wp->w_statuscol_click_defs_size) {
if ((in_statuscol && click_col >= (int)wp->w_statuscol_click_defs_size)
|| (in_status_line
&& click_col >=
(int)(in_global_statusline ? curwin : wp)->w_status_click_defs_size)) {
return false;
}

View File

@@ -1719,6 +1719,20 @@ describe('ui/mouse/input', function()
]])
end)
it("mouse click on window separator in statusline doesn't crash", function()
api.nvim_set_option_value('winwidth', 1, {})
api.nvim_set_option_value('statusline', '%f', {})
command('vsplit')
command('redraw')
local lines = api.nvim_get_option_value('lines', {})
local columns = api.nvim_get_option_value('columns', {})
api.nvim_input_mouse('left', 'press', '', 0, lines - 1, math.floor(columns / 2))
command('redraw')
end)
it('getmousepos() works correctly', function()
local winwidth = api.nvim_get_option_value('winwidth', {})
-- Set winwidth=1 so that window sizes don't change.