2011-03-28 10:18:00 +02:00
|
|
|
" EasyMotion - Vim motions on speed!
|
2011-03-28 00:08:06 +02:00
|
|
|
"
|
|
|
|
" Author: Kim Silkebækken <kim.silkebaekken+github@gmail.com>
|
|
|
|
" Source: https://github.com/Lokaltog/EasyMotion
|
|
|
|
|
|
|
|
" Prevent double loading {{{
|
|
|
|
if exists('g:EasyMotion_loaded')
|
|
|
|
finish
|
|
|
|
endif
|
|
|
|
|
|
|
|
let g:EasyMotion_loaded = 1
|
|
|
|
" }}}
|
|
|
|
" Default configuration {{{
|
|
|
|
if ! exists('g:EasyMotion_keys') " {{{
|
|
|
|
let g:EasyMotion_keys = ''
|
|
|
|
let g:EasyMotion_keys .= 'abcdefghijklmnopqrstuvwxyz'
|
|
|
|
let g:EasyMotion_keys .= 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
|
|
|
|
endif " }}}
|
|
|
|
if ! exists('g:EasyMotion_target_hl') " {{{
|
|
|
|
let g:EasyMotion_target_hl = 'EasyMotionTarget'
|
|
|
|
endif " }}}
|
|
|
|
if ! exists('g:EasyMotion_shade_hl') " {{{
|
|
|
|
let g:EasyMotion_shade_hl = 'EasyMotionShade'
|
|
|
|
endif " }}}
|
|
|
|
if ! exists('g:EasyMotion_do_shade') " {{{
|
|
|
|
let g:EasyMotion_shade = 1
|
|
|
|
endif " }}}
|
|
|
|
if ! exists('g:EasyMotion_do_mapping') " {{{
|
|
|
|
let g:EasyMotion_do_mapping = 1
|
|
|
|
endif " }}}
|
|
|
|
" Create default highlighting {{{
|
|
|
|
if ! hlexists(g:EasyMotion_target_hl) " {{{
|
|
|
|
execute 'hi ' . g:EasyMotion_target_hl . ' ctermfg=red ctermbg=none cterm=bold gui=bold guibg=Red guifg=yellow'
|
|
|
|
endif " }}}
|
|
|
|
if ! hlexists(g:EasyMotion_shade_hl) " {{{
|
|
|
|
execute 'hi ' . g:EasyMotion_shade_hl . ' ctermfg=black ctermbg=none cterm=bold gui=bold guibg=none guifg=black'
|
|
|
|
endif " }}}
|
|
|
|
" }}}
|
|
|
|
" }}}
|
|
|
|
" Default key mapping {{{
|
|
|
|
if g:EasyMotion_do_mapping
|
|
|
|
nnoremap <silent> <Leader>f :call EasyMotionF(0)<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>f :<C-U>call EasyMotionF(0, visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
nnoremap <silent> <Leader>F :call EasyMotionF(1)<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>F :<C-U>call EasyMotionF(1, visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
|
|
|
|
nnoremap <silent> <Leader>t :call EasyMotionT(0)<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>t :<C-U>call EasyMotionT(0, visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
nnoremap <silent> <Leader>T :call EasyMotionT(1)<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>T :<C-U>call EasyMotionT(1, visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
|
|
|
|
nnoremap <silent> <Leader>w :call EasyMotionW()<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>w :<C-U>call EasyMotionW(visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
nnoremap <silent> <Leader>e :call EasyMotionE()<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>e :<C-U>call EasyMotionE(visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
nnoremap <silent> <Leader>b :call EasyMotionB()<CR>
|
2011-03-28 10:14:51 +02:00
|
|
|
vnoremap <silent> <Leader>b :<C-U>call EasyMotionB(visualmode())<CR>
|
2011-03-28 00:08:06 +02:00
|
|
|
endif
|
|
|
|
" }}}
|
|
|
|
" Initialize variables {{{
|
|
|
|
let s:index_to_key = split(g:EasyMotion_keys, '\zs')
|
|
|
|
let s:key_to_index = {}
|
|
|
|
|
|
|
|
let index = 0
|
|
|
|
for i in s:index_to_key
|
|
|
|
let s:key_to_index[i] = index
|
|
|
|
let index += 1
|
|
|
|
endfor
|
|
|
|
" }}}
|
|
|
|
" Motion functions {{{
|
|
|
|
" F key motions {{{
|
|
|
|
" Go to {char} to the right or the left
|
2011-03-28 10:14:51 +02:00
|
|
|
function! EasyMotionF(direction, ...)
|
2011-03-28 00:08:06 +02:00
|
|
|
call <SID>Prompt('Search for character')
|
|
|
|
|
|
|
|
let char = getchar()
|
|
|
|
|
|
|
|
let re = '\C' . escape(nr2char(char), '.$^~')
|
|
|
|
|
2011-03-28 10:14:51 +02:00
|
|
|
call <SID>EasyMotion(re, a:direction, a:0 > 0 ? a:1 : '')
|
2011-03-28 00:08:06 +02:00
|
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
" T key motions {{{
|
|
|
|
" Go to {char} to the right (before) or the left (after)
|
2011-03-28 10:14:51 +02:00
|
|
|
function! EasyMotionT(direction, ...)
|
2011-03-28 00:08:06 +02:00
|
|
|
call <SID>Prompt('Search for character')
|
|
|
|
|
|
|
|
let char = getchar()
|
|
|
|
|
|
|
|
if a:direction == 1
|
|
|
|
let re = '\C' . escape(nr2char(char), '.$^~') . '\zs.'
|
|
|
|
else
|
|
|
|
let re = '\C.' . escape(nr2char(char), '.$^~')
|
|
|
|
endif
|
|
|
|
|
2011-03-28 10:14:51 +02:00
|
|
|
call <SID>EasyMotion(re, a:direction, a:0 > 0 ? a:1 : '')
|
2011-03-28 00:08:06 +02:00
|
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
" W key motions {{{
|
|
|
|
" Beginning of word forward
|
2011-03-28 10:14:51 +02:00
|
|
|
function! EasyMotionW(...)
|
|
|
|
call <SID>EasyMotion('\<.', 0, a:0 > 0 ? a:1 : '')
|
2011-03-28 00:08:06 +02:00
|
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
" E key motions {{{
|
|
|
|
" End of word forward
|
2011-03-28 10:14:51 +02:00
|
|
|
function! EasyMotionE(...)
|
|
|
|
call <SID>EasyMotion('.\>', 0, a:0 > 0 ? a:1 : '')
|
2011-03-28 00:08:06 +02:00
|
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
" B key motions {{{
|
|
|
|
" Beginning of word backward
|
2011-03-28 10:14:51 +02:00
|
|
|
function! EasyMotionB(...)
|
|
|
|
call <SID>EasyMotion('\<.', 1, a:0 > 0 ? a:1 : '')
|
2011-03-28 00:08:06 +02:00
|
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
" }}}
|
|
|
|
" Helper functions {{{
|
|
|
|
function! s:Message(message) " {{{
|
|
|
|
echo 'EasyMotion: ' . a:message
|
|
|
|
endfunction " }}}
|
|
|
|
function! s:Prompt(message) " {{{
|
|
|
|
echohl Question
|
|
|
|
echo a:message . ': '
|
|
|
|
echohl None
|
|
|
|
endfunction " }}}
|
|
|
|
" }}}
|
|
|
|
" Core functions {{{
|
|
|
|
function! s:PromptUser(groups) "{{{
|
|
|
|
let single_group = len(a:groups) == 1
|
|
|
|
let targets_len = single_group ? len(a:groups[0]) : len(a:groups)
|
|
|
|
|
|
|
|
" Only one possible match {{{
|
|
|
|
if single_group && targets_len == 1
|
|
|
|
redraw
|
|
|
|
|
|
|
|
return a:groups[0][0]
|
|
|
|
endif
|
|
|
|
" }}}
|
2011-03-28 09:03:49 +02:00
|
|
|
" Prepare marker lines {{{
|
|
|
|
let lines = {}
|
2011-03-28 00:08:06 +02:00
|
|
|
let hl_coords = []
|
2011-03-28 09:03:49 +02:00
|
|
|
let current_group = 0
|
2011-03-28 00:08:06 +02:00
|
|
|
|
|
|
|
for group in a:groups
|
|
|
|
let element = 0
|
|
|
|
|
2011-03-28 09:03:49 +02:00
|
|
|
for [line_num, col_num] in group
|
|
|
|
" Add original line and marker line
|
|
|
|
if ! has_key(lines, line_num)
|
|
|
|
let current_line = getline(line_num)
|
|
|
|
|
2011-03-28 10:12:08 +02:00
|
|
|
let lines[line_num] = { 'orig': current_line, 'marker': current_line }
|
2011-03-28 09:03:49 +02:00
|
|
|
endif
|
|
|
|
|
2011-03-28 10:12:08 +02:00
|
|
|
" Substitute marker character
|
|
|
|
let lines[line_num]['marker'] = substitute(lines[line_num]['marker'], '\%' . col_num . 'c.', s:index_to_key[single_group ? element : current_group], '')
|
2011-03-28 00:08:06 +02:00
|
|
|
|
|
|
|
" Add highlighting coordinates
|
2011-03-28 09:03:49 +02:00
|
|
|
call add(hl_coords, '\%' . line_num . 'l\%' . col_num . 'c')
|
|
|
|
|
|
|
|
let element += 1
|
2011-03-28 00:08:06 +02:00
|
|
|
endfor
|
|
|
|
|
|
|
|
let current_group += 1
|
|
|
|
endfor
|
2011-03-28 09:04:22 +02:00
|
|
|
|
2011-03-28 09:03:49 +02:00
|
|
|
let lines_items = items(lines)
|
2011-03-28 00:08:06 +02:00
|
|
|
" }}}
|
2011-03-28 09:04:22 +02:00
|
|
|
" Store original buffer properties {{{
|
2011-03-28 00:08:06 +02:00
|
|
|
let modified = &modified
|
|
|
|
let modifiable = &modifiable
|
|
|
|
let readonly = &readonly
|
|
|
|
" }}}
|
|
|
|
|
2011-03-28 09:04:22 +02:00
|
|
|
let input_char = ''
|
|
|
|
|
2011-03-28 00:08:06 +02:00
|
|
|
try
|
|
|
|
" Highlight source
|
|
|
|
let target_hl_id = matchadd(g:EasyMotion_target_hl, join(hl_coords, '\|'), 1)
|
|
|
|
|
|
|
|
" Make sure we can change the buffer {{{
|
|
|
|
if modifiable == 0
|
|
|
|
silent setl modifiable
|
|
|
|
endif
|
|
|
|
|
|
|
|
if readonly == 1
|
|
|
|
silent setl noreadonly
|
|
|
|
endif
|
|
|
|
" }}}
|
|
|
|
|
|
|
|
" Set lines with markers
|
2011-03-28 09:03:49 +02:00
|
|
|
for [line_num, line] in lines_items
|
2011-03-28 00:08:06 +02:00
|
|
|
try
|
|
|
|
undojoin
|
|
|
|
catch
|
|
|
|
endtry
|
|
|
|
|
2011-03-28 10:12:08 +02:00
|
|
|
call setline(line_num, line['marker'])
|
2011-03-28 00:08:06 +02:00
|
|
|
endfor
|
|
|
|
|
|
|
|
redraw
|
|
|
|
|
|
|
|
" Get target/group character
|
|
|
|
if single_group
|
|
|
|
call <SID>Prompt('Target character')
|
|
|
|
else
|
|
|
|
call <SID>Prompt('Group character')
|
|
|
|
endif
|
|
|
|
|
|
|
|
let input_char = nr2char(getchar())
|
|
|
|
|
|
|
|
redraw
|
|
|
|
finally
|
|
|
|
" Restore original lines
|
2011-03-28 09:03:49 +02:00
|
|
|
for [line_num, line] in lines_items
|
2011-03-28 00:08:06 +02:00
|
|
|
try
|
|
|
|
undojoin
|
|
|
|
catch
|
|
|
|
endtry
|
|
|
|
|
2011-03-28 09:03:49 +02:00
|
|
|
call setline(line_num, line['orig'])
|
2011-03-28 00:08:06 +02:00
|
|
|
endfor
|
|
|
|
|
|
|
|
call matchdelete(target_hl_id)
|
|
|
|
|
|
|
|
redraw
|
|
|
|
|
|
|
|
" Restore original properties {{{
|
|
|
|
if modified == 0
|
|
|
|
silent setl nomodified
|
|
|
|
endif
|
|
|
|
|
|
|
|
if modifiable == 0
|
|
|
|
silent setl nomodifiable
|
|
|
|
endif
|
|
|
|
|
|
|
|
if readonly == 1
|
|
|
|
silent setl readonly
|
|
|
|
endif
|
|
|
|
" }}}
|
|
|
|
|
|
|
|
" Check if the input char is valid
|
|
|
|
if ! has_key(s:key_to_index, input_char) || s:key_to_index[input_char] >= targets_len
|
|
|
|
" Invalid input char
|
|
|
|
return []
|
|
|
|
else
|
|
|
|
if single_group
|
|
|
|
" Return target coordinates
|
|
|
|
return a:groups[0][s:key_to_index[input_char]]
|
|
|
|
else
|
|
|
|
" Prompt for target character
|
|
|
|
return s:PromptUser([a:groups[s:key_to_index[input_char]]])
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endtry
|
|
|
|
endfunction "}}}
|
2011-03-28 10:14:51 +02:00
|
|
|
function! s:EasyMotion(regexp, direction, ...) " {{{
|
2011-03-28 00:08:06 +02:00
|
|
|
let orig_pos = [line('.'), col('.')]
|
|
|
|
let targets = []
|
2011-03-28 10:14:51 +02:00
|
|
|
let visualmode = a:0 > 0 ? a:1 : ''
|
2011-03-28 00:08:06 +02:00
|
|
|
|
|
|
|
" Find motion targets
|
|
|
|
while 1
|
|
|
|
let search_direction = (a:direction == 1 ? 'b' : '')
|
|
|
|
let search_stopline = line(a:direction == 1 ? 'w0' : 'w$')
|
|
|
|
|
|
|
|
let pos = searchpos(a:regexp, search_direction, search_stopline)
|
|
|
|
|
2011-03-28 00:22:40 +02:00
|
|
|
" Reached end of search range
|
2011-03-28 00:08:06 +02:00
|
|
|
if pos == [0, 0]
|
|
|
|
break
|
|
|
|
endif
|
|
|
|
|
2011-03-28 00:22:40 +02:00
|
|
|
" Skip folded lines
|
|
|
|
if foldclosed(pos[0]) != -1
|
|
|
|
continue
|
|
|
|
endif
|
|
|
|
|
2011-03-28 00:08:06 +02:00
|
|
|
call add(targets, pos)
|
|
|
|
endwhile
|
|
|
|
|
|
|
|
let targets_len = len(targets)
|
|
|
|
let groups_len = len(s:index_to_key)
|
|
|
|
|
|
|
|
if targets_len == 0
|
|
|
|
redraw
|
|
|
|
|
|
|
|
call <SID>Message('No matches')
|
|
|
|
|
2011-03-28 10:14:51 +02:00
|
|
|
" Restore cursor position
|
|
|
|
call setpos('.', [0, orig_pos[0], orig_pos[1]])
|
|
|
|
|
2011-03-28 00:08:06 +02:00
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Restore cursor position
|
|
|
|
call setpos('.', [0, orig_pos[0], orig_pos[1]])
|
|
|
|
|
|
|
|
" Split targets into key groups {{{
|
|
|
|
let groups = []
|
|
|
|
let i = 0
|
|
|
|
|
|
|
|
while i < targets_len
|
|
|
|
call add(groups, targets[i : i + groups_len - 1])
|
|
|
|
|
|
|
|
let i += groups_len
|
|
|
|
endwhile
|
|
|
|
" }}}
|
|
|
|
" Too many groups; only display the first ones {{{
|
|
|
|
if len(groups) > groups_len
|
|
|
|
call <SID>Message('Only displaying the first matches')
|
|
|
|
|
|
|
|
let groups = groups[0 : groups_len - 1]
|
|
|
|
endif
|
|
|
|
" }}}
|
|
|
|
|
|
|
|
" Shade inactive source
|
|
|
|
if g:EasyMotion_shade
|
|
|
|
let shade_hl_pos = '\%' . orig_pos[0] . 'l\%'. orig_pos[1] .'c'
|
|
|
|
|
|
|
|
if a:direction == 1
|
2011-03-28 09:04:57 +02:00
|
|
|
" Backward
|
|
|
|
let shade_hl_re = '\%'. line('w0') .'l\_.*' . shade_hl_pos
|
2011-03-28 00:08:06 +02:00
|
|
|
else
|
2011-03-28 09:04:57 +02:00
|
|
|
" Forward
|
|
|
|
let shade_hl_re = shade_hl_pos . '\_.*\%'. line('w$') .'l'
|
2011-03-28 00:08:06 +02:00
|
|
|
endif
|
|
|
|
|
|
|
|
let shade_hl_id = matchadd(g:EasyMotion_shade_hl, shade_hl_re, 0)
|
|
|
|
endif
|
|
|
|
|
2011-03-28 09:04:22 +02:00
|
|
|
" Prompt user for target group/character
|
2011-03-28 00:08:06 +02:00
|
|
|
let coords = <SID>PromptUser(groups)
|
|
|
|
|
|
|
|
" Remove shading
|
|
|
|
if g:EasyMotion_shade
|
|
|
|
call matchdelete(shade_hl_id)
|
|
|
|
endif
|
|
|
|
|
|
|
|
if len(coords) != 2
|
|
|
|
" Cancelled by user
|
|
|
|
call <SID>Message('Operation cancelled')
|
|
|
|
|
2011-03-28 10:14:51 +02:00
|
|
|
" Restore cursor position/selection
|
|
|
|
if ! empty(visualmode)
|
|
|
|
silent exec 'normal! `<' . visualmode . '`>'
|
|
|
|
else
|
|
|
|
call setpos('.', [0, orig_pos[0], orig_pos[1]])
|
|
|
|
endif
|
|
|
|
|
2011-03-28 00:08:06 +02:00
|
|
|
return
|
|
|
|
else
|
2011-03-28 10:14:51 +02:00
|
|
|
if ! empty(visualmode)
|
|
|
|
" Store original marks
|
|
|
|
let m_a = getpos("'a")
|
|
|
|
let m_b = getpos("'b")
|
|
|
|
|
|
|
|
" Store start/end positions
|
|
|
|
call setpos("'a", [0, orig_pos[0], orig_pos[1]])
|
|
|
|
call setpos("'b", [0, coords[0], coords[1]])
|
|
|
|
|
|
|
|
" Update selection
|
|
|
|
silent exec 'normal! `a' . visualmode . '`b'
|
|
|
|
|
|
|
|
" Restore original marks
|
|
|
|
call setpos("'a", m_a)
|
|
|
|
call setpos("'b", m_b)
|
|
|
|
else
|
|
|
|
" Update cursor position
|
|
|
|
call setpos('.', [0, coords[0], coords[1]])
|
|
|
|
endif
|
2011-03-28 00:08:06 +02:00
|
|
|
|
|
|
|
call <SID>Message('Jumping to [' . coords[0] . ', ' . coords[1] . ']')
|
|
|
|
endif
|
|
|
|
endfunction " }}}
|
|
|
|
" }}}
|