mirror of
https://github.com/neovim/neovim.git
synced 2025-02-25 18:55:25 -06:00
ext_cmdline: add support for highlighting
This commit is contained in:
parent
5ad591ef2d
commit
22402fb99d
@ -703,6 +703,23 @@ String cstr_to_string(const char *str)
|
||||
};
|
||||
}
|
||||
|
||||
/// Copies buffer to an allocated String.
|
||||
/// The resulting string is also NUL-terminated, to facilitate interoperating
|
||||
/// with code using C strings.
|
||||
///
|
||||
/// @param buf the buffer to copy
|
||||
/// @param size length of the buffer
|
||||
/// @return the resulting String, if the input string was NULL, an
|
||||
/// empty String is returned
|
||||
String cbuf_to_string(const char *buf, size_t size)
|
||||
FUNC_ATTR_NONNULL_ALL
|
||||
{
|
||||
return (String) {
|
||||
.data = xmemdupz(buf, size),
|
||||
.size = size
|
||||
};
|
||||
}
|
||||
|
||||
/// Creates a String using the given C string. Unlike
|
||||
/// cstr_to_string this function DOES NOT copy the C string.
|
||||
///
|
||||
|
@ -2372,6 +2372,7 @@ static bool color_cmdline(const CmdlineInfo *const colored_ccline,
|
||||
FUNC_ATTR_NONNULL_ALL FUNC_ATTR_WARN_UNUSED_RESULT
|
||||
{
|
||||
bool printed_errmsg = false;
|
||||
|
||||
#define PRINT_ERRMSG(...) \
|
||||
do { \
|
||||
msg_putchar('\n'); \
|
||||
@ -2405,7 +2406,7 @@ static bool color_cmdline(const CmdlineInfo *const colored_ccline,
|
||||
|
||||
static unsigned prev_prompt_id = UINT_MAX;
|
||||
static int prev_prompt_errors = 0;
|
||||
Callback color_cb = { .type = kCallbackNone };
|
||||
Callback color_cb = CALLBACK_NONE;
|
||||
bool can_free_cb = false;
|
||||
TryState tstate;
|
||||
Error err = ERROR_INIT;
|
||||
@ -2722,10 +2723,30 @@ draw_cmdline_no_arabicshape:
|
||||
void ui_ext_cmdline_show(void)
|
||||
{
|
||||
Array content = ARRAY_DICT_INIT;
|
||||
Array text = ARRAY_DICT_INIT;
|
||||
ADD(text, STRING_OBJ(cstr_to_string("Normal")));
|
||||
ADD(text, STRING_OBJ(cstr_to_string((char *)(ccline.cmdbuff))));
|
||||
ADD(content, ARRAY_OBJ(text));
|
||||
if (kv_size(last_ccline_colors.colors)) {
|
||||
for (size_t i = 0; i < kv_size(last_ccline_colors.colors); i++) {
|
||||
CmdlineColorChunk chunk = kv_A(last_ccline_colors.colors, i);
|
||||
Array item = ARRAY_DICT_INIT;
|
||||
|
||||
if (chunk.attr) {
|
||||
attrentry_T *aep = syn_cterm_attr2entry(chunk.attr);
|
||||
// TODO(bfredl): this desicion could be delayed by making attr_code a
|
||||
// recognized type
|
||||
HlAttrs rgb_attrs = attrentry2hlattrs(aep, true);
|
||||
ADD(item, DICTIONARY_OBJ(hlattrs2dict(rgb_attrs)));
|
||||
} else {
|
||||
ADD(item, DICTIONARY_OBJ((Dictionary)ARRAY_DICT_INIT));
|
||||
}
|
||||
ADD(item, STRING_OBJ(cbuf_to_string((char *)ccline.cmdbuff + chunk.start,
|
||||
chunk.end-chunk.start)));
|
||||
ADD(content, ARRAY_OBJ(item));
|
||||
}
|
||||
} else {
|
||||
Array item = ARRAY_DICT_INIT;
|
||||
ADD(item, DICTIONARY_OBJ((Dictionary)ARRAY_DICT_INIT));
|
||||
ADD(item, STRING_OBJ(cstr_to_string((char *)(ccline.cmdbuff))));
|
||||
ADD(content, ARRAY_OBJ(item));
|
||||
}
|
||||
ui_call_cmdline_show(content, ccline.cmdpos,
|
||||
cchar_to_string((char)ccline.cmdfirstc),
|
||||
cstr_to_string((char *)(ccline.cmdprompt)),
|
||||
|
@ -1,6 +1,7 @@
|
||||
local helpers = require('test.functional.helpers')(after_each)
|
||||
local Screen = require('test.functional.ui.screen')
|
||||
local clear, feed, eq = helpers.clear, helpers.feed, helpers.eq
|
||||
local source = helpers.source
|
||||
|
||||
if helpers.pending_win32(pending) then return end
|
||||
|
||||
@ -36,6 +37,22 @@ describe('External command line completion', function()
|
||||
screen:detach()
|
||||
end)
|
||||
|
||||
function expect_cmdline(expected)
|
||||
local attr_ids = screen._default_attr_ids
|
||||
local attr_ignore = screen._default_attr_ignore
|
||||
local actual = ''
|
||||
for _, chunk in ipairs(content or {}) do
|
||||
local attrs, text = chunk[1], chunk[2]
|
||||
if screen:_equal_attrs(attrs, {}) then
|
||||
actual = actual..text
|
||||
else
|
||||
local attr_id = screen:_get_attr_id(attr_ids, attr_ignore, attrs)
|
||||
actual = actual..'{' .. attr_id .. ':' .. text .. '}'
|
||||
end
|
||||
end
|
||||
eq(expected, actual)
|
||||
end
|
||||
|
||||
describe("'cmdline'", function()
|
||||
it(':sign', function()
|
||||
feed(':')
|
||||
@ -58,7 +75,7 @@ describe('External command line completion', function()
|
||||
~ |
|
||||
|
|
||||
]], nil, nil, function()
|
||||
eq({{'Normal', 'sign'}}, content)
|
||||
eq({{{}, 'sign'}}, content)
|
||||
eq(4, pos)
|
||||
end)
|
||||
|
||||
@ -70,7 +87,7 @@ describe('External command line completion', function()
|
||||
~ |
|
||||
|
|
||||
]], nil, nil, function()
|
||||
eq({{'Normal', 'sign'}}, content)
|
||||
eq({{{}, 'sign'}}, content)
|
||||
eq(true, shown)
|
||||
eq(3, pos)
|
||||
end)
|
||||
@ -83,7 +100,7 @@ describe('External command line completion', function()
|
||||
~ |
|
||||
|
|
||||
]], nil, nil, function()
|
||||
eq({{'Normal', 'sin'}}, content)
|
||||
eq({{{}, 'sin'}}, content)
|
||||
eq(true, shown)
|
||||
eq(2, pos)
|
||||
end)
|
||||
@ -109,7 +126,7 @@ describe('External command line completion', function()
|
||||
]], nil, nil, function()
|
||||
eq(true, shown)
|
||||
eq("input", prompt)
|
||||
eq({{'Normal', 'default'}}, content)
|
||||
eq({{{}, 'default'}}, content)
|
||||
end)
|
||||
feed('<cr>')
|
||||
|
||||
@ -132,7 +149,7 @@ describe('External command line completion', function()
|
||||
~ |
|
||||
|
|
||||
]], nil, nil, function()
|
||||
eq({{'Normal', '1+2'}}, content)
|
||||
eq({{{}, '1+2'}}, content)
|
||||
eq("\"", char)
|
||||
eq(1, shift)
|
||||
eq(2, level)
|
||||
@ -146,7 +163,7 @@ describe('External command line completion', function()
|
||||
~ |
|
||||
|
|
||||
]], nil, nil, function()
|
||||
eq({{'Normal', '3'}}, content)
|
||||
eq({{{}, '3'}}, content)
|
||||
eq(2, current_hide_level)
|
||||
eq(1, level)
|
||||
end)
|
||||
@ -210,4 +227,52 @@ describe('External command line completion', function()
|
||||
|
||||
end)
|
||||
end)
|
||||
|
||||
it('works with highlighted cmdline', function()
|
||||
source([[
|
||||
highlight RBP1 guibg=Red
|
||||
highlight RBP2 guibg=Yellow
|
||||
highlight RBP3 guibg=Green
|
||||
highlight RBP4 guibg=Blue
|
||||
let g:NUM_LVLS = 4
|
||||
function RainBowParens(cmdline)
|
||||
let ret = []
|
||||
let i = 0
|
||||
let lvl = 0
|
||||
while i < len(a:cmdline)
|
||||
if a:cmdline[i] is# '('
|
||||
call add(ret, [i, i + 1, 'RBP' . ((lvl % g:NUM_LVLS) + 1)])
|
||||
let lvl += 1
|
||||
elseif a:cmdline[i] is# ')'
|
||||
let lvl -= 1
|
||||
call add(ret, [i, i + 1, 'RBP' . ((lvl % g:NUM_LVLS) + 1)])
|
||||
endif
|
||||
let i += 1
|
||||
endwhile
|
||||
return ret
|
||||
endfunction
|
||||
map <f5> :let x = input({'prompt':'>','highlight':'RainBowParens'})<cr>
|
||||
"map <f5> :let x = input({'prompt':'>'})<cr>
|
||||
]])
|
||||
screen:set_default_attr_ids({
|
||||
RBP1={background = Screen.colors.Red},
|
||||
RBP2={background = Screen.colors.Yellow},
|
||||
RBP3={background = Screen.colors.Green},
|
||||
RBP4={background = Screen.colors.Blue},
|
||||
EOB={bold = true, foreground = Screen.colors.Blue1},
|
||||
ERR={foreground = Screen.colors.Grey100, background = Screen.colors.Red},
|
||||
SK={foreground = Screen.colors.Blue},
|
||||
PE={bold = true, foreground = Screen.colors.SeaGreen4}
|
||||
})
|
||||
feed('<f5>(a(b)a)')
|
||||
screen:expect([[
|
||||
^ |
|
||||
{EOB:~ }|
|
||||
{EOB:~ }|
|
||||
{EOB:~ }|
|
||||
|
|
||||
]], nil, nil, function()
|
||||
expect_cmdline('{RBP1:(}a{RBP2:(}b{RBP2:)}a{RBP1:)}')
|
||||
end)
|
||||
end)
|
||||
end)
|
||||
|
Loading…
Reference in New Issue
Block a user