`highlight_selection()` is supposed to abort if the visual mode was started on an empty line. This was done correctly for charwise-visual but failed to do so in linewise-visual.
228 lines
6.7 KiB
VimL
228 lines
6.7 KiB
VimL
" Autocommands """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""
|
|
|
|
" Bitfield for highlight_current augroup
|
|
if !exists('s:CLEAR_HIGHS_ALL')
|
|
const s:CLEAR_HIGHS_CWORD = 1
|
|
const s:CLEAR_HIGHS_VISUAL = 2
|
|
const s:CLEAR_HIGHS_ALL = 3
|
|
endif
|
|
|
|
" Terminal
|
|
if (has('nvim'))
|
|
" Disable spellcheck
|
|
augroup terminal_no_spellcheck
|
|
au!
|
|
autocmd TermOpen * setlocal nospell
|
|
" Do not highlight trailing whitespace in terminal windows
|
|
autocmd TermOpen * silent! call
|
|
\ matchdelete(filter(getmatches(), 'v:val["group"] == "TrailingWhitespace"')[0]["id"])
|
|
augroup END
|
|
endif
|
|
|
|
" change cursor shape depending on mode
|
|
augroup cursor_shape_by_mode
|
|
au!
|
|
if (has('nvim'))
|
|
" Beam when exiting
|
|
autocmd VimLeave * silent !echo -ne "\e[5 q"
|
|
else
|
|
" https://vim.fandom.com/wiki/Change_cursor_shape_in_different_modes
|
|
" https://github.com/tmux/tmux/issues/1593
|
|
if exists('$TMUX')
|
|
" Start insert mode - vertical bar/beam
|
|
let &t_SI = "\ePtmux;\e\e[5 q\e\\"
|
|
" Start replace mode - horizontal bar/underline
|
|
let &t_SR = "\ePtmux;\e\e[3 q\e\\"
|
|
" End insert or replace mode - block
|
|
let &t_EI = "\ePtmux;\e\e[1 q\e\\"
|
|
|
|
" Block when entering
|
|
autocmd VimEnter * silent !echo -ne "\ePtmux;\e\e[1 q\e\\"
|
|
" Beam when exiting
|
|
autocmd VimLeave * silent !echo -ne "\ePtmux;\e\e[5 q\e\\"
|
|
else
|
|
" Start insert mode - vertical bar/beam
|
|
let &t_SI = "\e[5 q"
|
|
" Start replace mode - horizontal bar/underline
|
|
let &t_SR = "\e[3 q"
|
|
" End insert or replace mode - block
|
|
let &t_EI = "\e[1 q"
|
|
|
|
" Block when entering
|
|
autocmd VimEnter * silent !echo -ne "\e[1 q"
|
|
" Beam when exiting
|
|
autocmd VimLeave * silent !echo -ne "\e[5 q"
|
|
endif
|
|
endif
|
|
augroup END
|
|
|
|
" Custom bindings when debugging
|
|
augroup termdebug_bindings
|
|
au!
|
|
" Go to normal mode with <Esc> like usually
|
|
autocmd SourcePost termdebug.vim tnoremap <Esc> <C-\><C-n>
|
|
augroup END
|
|
|
|
" Highlight word under cursor in other places
|
|
function! s:highlight_cword()
|
|
if exists('w:disable_highlight_cword')
|
|
return
|
|
endif
|
|
|
|
if get(w:, 'old_cword', '') == expand('<cword>')
|
|
" Nothing to do if we're still on the same word
|
|
return
|
|
endif
|
|
|
|
" Clear previous highlight and kill the possibly already running timer
|
|
call s:clear_highlights(s:CLEAR_HIGHS_CWORD)
|
|
|
|
" Delay the highlight by 100ms so that not every word is highlighted
|
|
" while moving the cursor fast. (This kind of simulates a CursorHold
|
|
" event with a custom time)
|
|
let w:cword_timer_id = timer_start(100, "s:_highlight_cword")
|
|
endfunction
|
|
|
|
function! s:_highlight_cword(timer_id)
|
|
unlet w:cword_timer_id
|
|
|
|
let l:cword = expand('<cword>')
|
|
if (l:cword != '')
|
|
let w:old_cword = l:cword
|
|
let w:cword_match_id = matchadd(
|
|
\ 'CursorColumn',
|
|
\ '\V\<' . escape(l:cword, '/\') . '\>',
|
|
\ -1)
|
|
endif
|
|
endfunction
|
|
|
|
" Highlight visual selection in other places
|
|
function! s:highlight_selection()
|
|
if exists('w:disable_highlight_visual_sel')
|
|
return
|
|
endif
|
|
|
|
" Clear previous highlight and kill the possibly already running timer
|
|
call s:clear_highlights(s:CLEAR_HIGHS_VISUAL)
|
|
|
|
" Delay the highlight by 100ms so that not every selection is highlighted
|
|
" while moving the cursor fast. (This kind of simulates a CursorHold
|
|
" event with a custom time)
|
|
let w:selection_timer_id = timer_start(100, "s:_highlight_selection")
|
|
endfunction
|
|
|
|
function! s:_highlight_selection(timer)
|
|
unlet w:selection_timer_id
|
|
|
|
let l:old_reg = getreg('"')
|
|
let l:old_regtype = getregtype('"')
|
|
" NOTE: The yank needs to be silent to mute the 'n lines yanked'
|
|
" message. But the `silent` leads to the disappearance of the selection
|
|
" size indicator (:h 'showcmd'), thus `y` and `gv` need to be executed
|
|
" in separate normal mode commands.
|
|
silent normal y
|
|
normal gv
|
|
|
|
if @" == "" || @" == "\n"
|
|
" Abort when visual mode started on an empty line
|
|
return
|
|
endif
|
|
|
|
let w:visual_match_ids = []
|
|
|
|
" Add match to all windows containing the current buffer
|
|
" NOTE: \%V\@! prevents the pattern from matching the current selection. As
|
|
" it is highlighted already this would be superfluous and inefficient.
|
|
for l:win in win_findbuf(bufnr())
|
|
let w:visual_match_ids += [[
|
|
\ matchadd(
|
|
\ 'CursorColumn',
|
|
\ '\V\%V\@!' . substitute(escape(@", '\'), '\n', '\\n', 'g'),
|
|
\ -1,
|
|
\ -1,
|
|
\ {'window': l:win}),
|
|
\ l:win
|
|
\ ]]
|
|
endfor
|
|
|
|
call setreg('"', l:old_reg, l:old_regtype)
|
|
endfunction
|
|
|
|
" Clear the highlights of <cword> and visual selection
|
|
function! s:clear_highlights(what = s:CLEAR_HIGHS_ALL)
|
|
if and(a:what, s:CLEAR_HIGHS_CWORD)
|
|
if exists('w:cword_match_id')
|
|
call matchdelete(w:cword_match_id)
|
|
unlet w:cword_match_id
|
|
unlet w:old_cword
|
|
endif
|
|
if exists('w:cword_timer_id')
|
|
call timer_stop(w:cword_timer_id)
|
|
unlet w:cword_timer_id
|
|
endif
|
|
endif
|
|
if and(a:what, s:CLEAR_HIGHS_VISUAL)
|
|
if exists('w:visual_match_ids')
|
|
for l:pairs in w:visual_match_ids
|
|
let l:id = l:pairs[0]
|
|
let l:win = l:pairs[1]
|
|
call matchdelete(l:id, l:win)
|
|
endfor
|
|
unlet w:visual_match_ids
|
|
endif
|
|
if exists('w:selection_timer_id')
|
|
call timer_stop(w:selection_timer_id)
|
|
unlet w:selection_timer_id
|
|
endif
|
|
endif
|
|
endfunction
|
|
|
|
augroup highlight_current
|
|
au!
|
|
" TODO: `viw` when on the last character of the word does not trigger
|
|
" CursorMoved, but the selection changes
|
|
au CursorMoved * if mode() == 'n' |
|
|
\ call s:highlight_cword() |
|
|
\ else |
|
|
\ call s:highlight_selection() |
|
|
\ endif
|
|
au CursorMovedI * call s:highlight_cword()
|
|
au WinLeave * call s:clear_highlights()
|
|
au ModeChanged [vV\x16]*:* call s:clear_highlights(s:CLEAR_HIGHS_VISUAL) | call s:highlight_cword()
|
|
" NOTE: s:highlight_selection is not called when entering non-linewise visual
|
|
" mode as I rarely need one-character highlighting and I can work
|
|
" around by moving back and forth once.
|
|
" TODO: Fix for other ways of entering with a longer selection
|
|
au ModeChanged *:[v\x16]* call s:clear_highlights(s:CLEAR_HIGHS_CWORD)
|
|
au ModeChanged *:V call s:clear_highlights(s:CLEAR_HIGHS_CWORD) | call s:highlight_selection()
|
|
augroup END
|
|
|
|
" When switching focus to another window, keep the cursor location underlined.
|
|
function! s:highlight_old_cursor_pos()
|
|
let w:cursor_pos_match_id = matchaddpos(
|
|
\ 'Underlined',
|
|
\ [getcurpos()[1:2]])
|
|
endfunction
|
|
function! s:clear_old_cursor_pos()
|
|
if exists('w:cursor_pos_match_id')
|
|
call matchdelete(w:cursor_pos_match_id)
|
|
unlet w:cursor_pos_match_id
|
|
endif
|
|
endfunction
|
|
augroup highlight_old_cursor_pos
|
|
au!
|
|
au WinLeave * call s:highlight_old_cursor_pos()
|
|
au WinEnter * call s:clear_old_cursor_pos()
|
|
|
|
" TODO: WinLeave is not triggered when entering command line mode and
|
|
" CmdlineEnter is triggered **after** entering
|
|
" nnoremap : :call s:highlight_old_cursor_pos()<CR>:
|
|
" au CmdlineLeave * call s:clear_old_cursor_pos()
|
|
augroup END
|
|
|
|
" Do not mark input from stdin as modified
|
|
augroup stdin_not_modified
|
|
au!
|
|
au StdinReadPost * set nomodified
|
|
augroup END
|