fix(statusline): also allow right click when 'mousemodel' is "popup*"

Problem:    The 'statusline'-format ui elements do not receive right
            click events when "mousemodel" is "popup*"
Solution:   Do not draw popupmenu and handle click event instead.
(cherry picked from commit 6b5b6e5e07)
This commit is contained in:
Luuk van Baal
2023-04-25 03:35:42 +02:00
committed by github-actions[bot]
parent 0c8e2d128e
commit f24449d35c
4 changed files with 234 additions and 224 deletions

View File

@@ -212,24 +212,33 @@ static int get_fpos_of_mouse(pos_T *mpos)
if (wp == NULL) { if (wp == NULL) {
return IN_UNKNOWN; return IN_UNKNOWN;
} }
int winrow = row;
int wincol = col;
// compute the position in the buffer line from the posn on the screen
bool below_buffer = mouse_comp_pos(wp, &row, &col, &mpos->lnum);
if (!below_buffer && *wp->w_p_stc != NUL && mouse_col < win_col_off(wp)) {
return MOUSE_STATUSCOL;
}
// winpos and height may change in win_enter()! // winpos and height may change in win_enter()!
if (row + wp->w_winbar_height >= wp->w_height) { // In (or below) status line if (winrow + wp->w_winbar_height >= wp->w_height) { // In (or below) status line
return IN_STATUS_LINE; return IN_STATUS_LINE;
} }
if (col >= wp->w_width) { // In vertical separator line
if (winrow == -1 && wp->w_winbar_height != 0) {
return MOUSE_WINBAR;
}
if (wincol >= wp->w_width) { // In vertical separator line
return IN_SEP_LINE; return IN_SEP_LINE;
} }
if (wp != curwin) { if (wp != curwin || below_buffer) {
return IN_UNKNOWN; return IN_UNKNOWN;
} }
// compute the position in the buffer line from the posn on the screen
if (mouse_comp_pos(curwin, &row, &col, &mpos->lnum)) {
return IN_STATUS_LINE; // past bottom
}
mpos->col = vcol2col(wp, mpos->lnum, col); mpos->col = vcol2col(wp, mpos->lnum, col);
mpos->coladd = 0; mpos->coladd = 0;
@@ -530,6 +539,11 @@ bool do_mouse(oparg_T *oap, int c, int dir, long count, bool fixindent)
// shift-left button -> right button // shift-left button -> right button
// alt-left button -> alt-right button // alt-left button -> alt-right button
if (mouse_model_popup()) { if (mouse_model_popup()) {
pos_T m_pos;
int m_pos_flag = get_fpos_of_mouse(&m_pos);
if (m_pos_flag & (IN_STATUS_LINE|MOUSE_WINBAR|MOUSE_STATUSCOL)) {
goto popupexit;
}
if (which_button == MOUSE_RIGHT if (which_button == MOUSE_RIGHT
&& !(mod_mask & (MOD_MASK_SHIFT | MOD_MASK_CTRL))) { && !(mod_mask & (MOD_MASK_SHIFT | MOD_MASK_CTRL))) {
if (!is_click) { if (!is_click) {
@@ -539,17 +553,14 @@ bool do_mouse(oparg_T *oap, int c, int dir, long count, bool fixindent)
} }
jump_flags = 0; jump_flags = 0;
if (strcmp(p_mousem, "popup_setpos") == 0) { if (strcmp(p_mousem, "popup_setpos") == 0) {
// First set the cursor position before showing the popup // First set the cursor position before showing the popup menu.
// menu.
if (VIsual_active) { if (VIsual_active) {
pos_T m_pos; // set MOUSE_MAY_STOP_VIS if we are outside the selection
// set MOUSE_MAY_STOP_VIS if we are outside the // or the current window (might have false negative here)
// selection or the current window (might have false
// negative here)
if (mouse_row < curwin->w_winrow if (mouse_row < curwin->w_winrow
|| mouse_row > (curwin->w_winrow + curwin->w_height)) { || mouse_row > (curwin->w_winrow + curwin->w_height)) {
jump_flags = MOUSE_MAY_STOP_VIS; jump_flags = MOUSE_MAY_STOP_VIS;
} else if (get_fpos_of_mouse(&m_pos) != IN_BUFFER) { } else if (m_pos_flag != IN_BUFFER) {
jump_flags = MOUSE_MAY_STOP_VIS; jump_flags = MOUSE_MAY_STOP_VIS;
} else { } else {
if (VIsual_mode == 'V') { if (VIsual_mode == 'V') {
@@ -593,6 +604,7 @@ bool do_mouse(oparg_T *oap, int c, int dir, long count, bool fixindent)
mod_mask &= ~MOD_MASK_SHIFT; mod_mask &= ~MOD_MASK_SHIFT;
} }
} }
popupexit:
if ((State & (MODE_NORMAL | MODE_INSERT)) if ((State & (MODE_NORMAL | MODE_INSERT))
&& !(mod_mask & (MOD_MASK_SHIFT | MOD_MASK_CTRL))) { && !(mod_mask & (MOD_MASK_SHIFT | MOD_MASK_CTRL))) {
@@ -644,7 +656,7 @@ bool do_mouse(oparg_T *oap, int c, int dir, long count, bool fixindent)
in_sep_line = (jump_flags & IN_SEP_LINE); in_sep_line = (jump_flags & IN_SEP_LINE);
if ((in_winbar || in_status_line || in_statuscol) && is_click) { if ((in_winbar || in_status_line || in_statuscol) && is_click) {
// Handle click event on window bar or status lin // Handle click event on window bar, status line or status column
int click_grid = mouse_grid; int click_grid = mouse_grid;
int click_row = mouse_row; int click_row = mouse_row;
int click_col = mouse_col; int click_col = mouse_col;
@@ -672,7 +684,7 @@ bool do_mouse(oparg_T *oap, int c, int dir, long count, bool fixindent)
call_click_def_func(click_defs, click_col, which_button); call_click_def_func(click_defs, click_col, which_button);
break; break;
default: default:
assert(false && "winbar and statusline only support %@ for clicks"); assert(false && "winbar, statusline and statuscolumn only support %@ for clicks");
break; break;
} }
} }

View File

@@ -1840,16 +1840,6 @@ describe('ui/mouse/input', function()
eq({2, 9}, meths.win_get_cursor(0)) eq({2, 9}, meths.win_get_cursor(0))
eq('', funcs.getreg('"')) eq('', funcs.getreg('"'))
-- Try clicking on the status line
funcs.setreg('"', '')
meths.win_set_cursor(0, {1, 9})
feed('vee')
meths.input_mouse('right', 'press', '', 0, 5, 1)
meths.input_mouse('right', 'release', '', 0, 5, 1)
feed('<Down><CR>')
eq({1, 9}, meths.win_get_cursor(0))
eq('ran away', funcs.getreg('"'))
-- Try clicking outside the window -- Try clicking outside the window
funcs.setreg('"', '') funcs.setreg('"', '')
meths.win_set_cursor(0, {2, 1}) meths.win_set_cursor(0, {2, 1})

View File

@@ -9,6 +9,8 @@ local feed = helpers.feed
local meths = helpers.meths local meths = helpers.meths
local pcall_err = helpers.pcall_err local pcall_err = helpers.pcall_err
local mousemodels = { "extend", "popup", "popup_setpos" }
describe('statuscolumn', function() describe('statuscolumn', function()
local screen local screen
before_each(function() before_each(function()
@@ -420,8 +422,9 @@ describe('statuscolumn', function()
]]) ]])
end) end)
it("works with 'statuscolumn' clicks", function() for _, model in ipairs(mousemodels) do
command('set mousemodel=extend') it("works with 'statuscolumn' clicks with mousemodel=" .. model, function()
command('set mousemodel=' .. model)
command([[ command([[
function! MyClickFunc(minwid, clicks, button, mods) function! MyClickFunc(minwid, clicks, button, mods)
let g:testvar = printf("%d %d %s %d", a:minwid, a:clicks, a:button, getmousepos().line) let g:testvar = printf("%d %d %s %d", a:minwid, a:clicks, a:button, getmousepos().line)
@@ -459,6 +462,7 @@ describe('statuscolumn', function()
meths.input_mouse('right', 'press', '', 0, 13, 0) meths.input_mouse('right', 'press', '', 0, 13, 0)
eq('', eval("g:testvar")) eq('', eval("g:testvar"))
end) end)
end
it('click labels do not leak memory', function() it('click labels do not leak memory', function()
command([[ command([[

View File

@@ -12,14 +12,17 @@ local exec_lua = helpers.exec_lua
local eval = helpers.eval local eval = helpers.eval
local sleep = helpers.sleep local sleep = helpers.sleep
describe('statusline clicks', function() local mousemodels = { "extend", "popup", "popup_setpos" }
for _, model in ipairs(mousemodels) do
describe('statusline clicks with mousemodel=' .. model, function()
local screen local screen
before_each(function() before_each(function()
clear() clear()
screen = Screen.new(40, 8) screen = Screen.new(40, 8)
screen:attach() screen:attach()
command('set laststatus=2 mousemodel=extend') command('set laststatus=2 mousemodel=' .. model)
exec([=[ exec([=[
function! MyClickFunc(minwid, clicks, button, mods) function! MyClickFunc(minwid, clicks, button, mods)
let g:testvar = printf("%d %d %s", a:minwid, a:clicks, a:button) let g:testvar = printf("%d %d %s", a:minwid, a:clicks, a:button)
@@ -184,6 +187,7 @@ describe('statusline clicks', function()
eq('0 1 l', eval("g:testvar")) eq('0 1 l', eval("g:testvar"))
end) end)
end) end)
end
describe('global statusline', function() describe('global statusline', function()
local screen local screen