scriptencoding utf-8 " Author: w0rp " Author: Luan Santos " Description: Shows lint message for the current line as virtualtext, if any if !hlexists('ALEVirtualTextError') highlight link ALEVirtualTextError Comment endif if !hlexists('ALEVirtualTextStyleError') highlight link ALEVirtualTextStyleError ALEVirtualTextError endif if !hlexists('ALEVirtualTextWarning') highlight link ALEVirtualTextWarning Comment endif if !hlexists('ALEVirtualTextStyleWarning') highlight link ALEVirtualTextStyleWarning ALEVirtualTextWarning endif if !hlexists('ALEVirtualTextInfo') highlight link ALEVirtualTextInfo ALEVirtualTextWarning endif let g:ale_virtualtext_prefix = \ get(g:, 'ale_virtualtext_prefix', '%comment% %type%: ') " Controls the milliseconds delay before showing a message. let g:ale_virtualtext_delay = get(g:, 'ale_virtualtext_delay', 10) let s:cursor_timer = get(s:, 'cursor_timer', -1) let s:last_pos = get(s:, 'last_pos', [0, 0, 0]) let s:hl_list = get(s:, 'hl_list', []) let s:last_message = '' if !has_key(s:, 'has_virt_text') let s:has_virt_text = 0 let s:emulate_virt = 0 let s:last_virt = -1 if has('nvim-0.3.2') let s:ns_id = nvim_create_namespace('ale') let s:has_virt_text = 1 elseif has('textprop') && has('popupwin') let s:has_virt_text = 1 let s:emulate_virt = !has('patch-9.0.0297') if s:emulate_virt call prop_type_add('ale', {}) endif endif endif function! s:StopCursorTimer() abort if s:cursor_timer != -1 call timer_stop(s:cursor_timer) let s:cursor_timer = -1 endif endfunction function! ale#virtualtext#ResetDataForTests() abort let s:last_pos = [0, 0, 0] let s:last_message = '' endfunction function! ale#virtualtext#GetLastMessageForTests() abort return s:last_message endfunction function! ale#virtualtext#GetComment(buffer) abort let l:filetype = getbufvar(a:buffer, '&filetype') let l:split = split(getbufvar(a:buffer, '&commentstring'), '%s') return !empty(l:split) ? trim(l:split[0]) : '#' endfunction function! ale#virtualtext#Clear(buffer) abort if !s:has_virt_text || !bufexists(str2nr(a:buffer)) return endif if has('nvim') call nvim_buf_clear_namespace(a:buffer, s:ns_id, 0, -1) else if s:emulate_virt && s:last_virt != -1 call prop_remove({'type': 'ale'}) call popup_close(s:last_virt) let s:last_virt = -1 elseif !empty(s:hl_list) call prop_remove({ \ 'types': s:hl_list, \ 'all': 1, \ 'bufnr': a:buffer, \}) endif endif endfunction function! ale#virtualtext#GetGroup(item) abort let l:type = get(a:item, 'type', 'E') let l:sub_type = get(a:item, 'sub_type', '') if l:type is# 'E' if l:sub_type is# 'style' return 'ALEVirtualTextStyleError' endif return 'ALEVirtualTextError' endif if l:type is# 'W' if l:sub_type is# 'style' return 'ALEVirtualTextStyleWarning' endif return 'ALEVirtualTextWarning' endif return 'ALEVirtualTextInfo' endfunction function! ale#virtualtext#ShowMessage(buffer, item) abort if !s:has_virt_text || !bufexists(str2nr(a:buffer)) return endif let l:line = max([1, a:item.lnum]) let l:hl_group = ale#virtualtext#GetGroup(a:item) " Get a language-appropriate comment character, or default to '#'. let l:comment = ale#virtualtext#GetComment(a:buffer) let l:prefix = ale#Var(a:buffer, 'virtualtext_prefix') let l:prefix = ale#GetLocItemMessage(a:item, l:prefix) let l:prefix = substitute(l:prefix, '\V%comment%', '\=l:comment', 'g') let l:msg = l:prefix . substitute(a:item.text, '\n', ' ', 'g') " Store the last message we're going to set so we can read it in tests. let s:last_message = l:msg if has('nvim') call nvim_buf_set_virtual_text( \ a:buffer, \ s:ns_id, l:line - 1, \ [[l:msg, l:hl_group]], \ {} \) elseif s:emulate_virt let l:left_pad = col('$') call prop_add(l:line, l:left_pad, {'type': 'ale'}) let s:last_virt = popup_create(l:msg, { \ 'line': -1, \ 'padding': [0, 0, 0, 1], \ 'mask': [[1, 1, 1, 1]], \ 'textprop': 'ale', \ 'highlight': l:hl_group, \ 'fixed': 1, \ 'wrap': 0, \ 'zindex': 2 \}) else let l:type = prop_type_get(l:hl_group) if l:type == {} call prop_type_add(l:hl_group, {'highlight': l:hl_group}) endif " Add highlight groups to the list so we can clear them later. if index(s:hl_list, l:hl_group) == -1 call add(s:hl_list, l:hl_group) endif " We ignore all errors from prop_add. silent! call prop_add(l:line, 0, { \ 'type': l:hl_group, \ 'text': ' ' . l:msg, \ 'bufnr': a:buffer, \}) endif endfunction function! ale#virtualtext#ShowCursorWarning(...) abort if g:ale_virtualtext_cursor isnot# 'current' \&& g:ale_virtualtext_cursor != 1 return endif let l:buffer = bufnr('') if mode(1) isnot# 'n' return endif if ale#ShouldDoNothing(l:buffer) return endif let [l:info, l:item] = ale#util#FindItemAtCursor(l:buffer) call ale#virtualtext#Clear(l:buffer) if !empty(l:item) call ale#virtualtext#ShowMessage(l:buffer, l:item) endif endfunction function! ale#virtualtext#ShowCursorWarningWithDelay() abort let l:buffer = bufnr('') if g:ale_virtualtext_cursor isnot# 'current' \&& g:ale_virtualtext_cursor != 1 return endif if mode(1) isnot# 'n' return endif call s:StopCursorTimer() let l:pos = getpos('.')[0:2] " Check the current buffer, line, and column number against the last " recorded position. If the position has actually changed, *then* " we should show something. Otherwise we can end up doing processing " the show message far too frequently. if l:pos != s:last_pos let l:delay = ale#Var(l:buffer, 'virtualtext_delay') let s:last_pos = l:pos let s:cursor_timer = timer_start( \ l:delay, \ function('ale#virtualtext#ShowCursorWarning') \) endif endfunction function! ale#virtualtext#SetTexts(buffer, loclist) abort if !has('nvim') && s:emulate_virt return endif call ale#virtualtext#Clear(a:buffer) for l:item in a:loclist if l:item.bufnr == a:buffer call ale#virtualtext#ShowMessage(a:buffer, l:item) endif endfor endfunction