" EasyMotion - Vim motions on speed! " " Author: Kim Silkebækken " 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 f :call EasyMotionF(0) vnoremap f :call EasyMotionF(0, visualmode()) nnoremap F :call EasyMotionF(1) vnoremap F :call EasyMotionF(1, visualmode()) nnoremap t :call EasyMotionT(0) vnoremap t :call EasyMotionT(0, visualmode()) nnoremap T :call EasyMotionT(1) vnoremap T :call EasyMotionT(1, visualmode()) nnoremap w :call EasyMotionW() vnoremap w :call EasyMotionW(visualmode()) nnoremap e :call EasyMotionE() vnoremap e :call EasyMotionE(visualmode()) nnoremap b :call EasyMotionB() vnoremap b :call EasyMotionB(visualmode()) 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 function! EasyMotionF(direction, ...) call Prompt('Search for character') let char = getchar() let re = '\C' . escape(nr2char(char), '.$^~') call EasyMotion(re, a:direction, a:0 > 0 ? a:1 : '') endfunction " }}} " T key motions {{{ " Go to {char} to the right (before) or the left (after) function! EasyMotionT(direction, ...) call 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 call EasyMotion(re, a:direction, a:0 > 0 ? a:1 : '') endfunction " }}} " W key motions {{{ " Beginning of word forward function! EasyMotionW(...) call EasyMotion('\<.', 0, a:0 > 0 ? a:1 : '') endfunction " }}} " E key motions {{{ " End of word forward function! EasyMotionE(...) call EasyMotion('.\>', 0, a:0 > 0 ? a:1 : '') endfunction " }}} " B key motions {{{ " Beginning of word backward function! EasyMotionB(...) call EasyMotion('\<.', 1, a:0 > 0 ? a:1 : '') 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 " }}} " Prepare marker lines {{{ let lines = {} let hl_coords = [] let current_group = 0 for group in a:groups let element = 0 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) let lines[line_num] = { 'orig': current_line, 'marker': current_line } endif " 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], '') " Add highlighting coordinates call add(hl_coords, '\%' . line_num . 'l\%' . col_num . 'c') let element += 1 endfor let current_group += 1 endfor let lines_items = items(lines) " }}} " Store original buffer properties {{{ let modified = &modified let modifiable = &modifiable let readonly = &readonly " }}} let input_char = '' 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 for [line_num, line] in lines_items try undojoin catch endtry call setline(line_num, line['marker']) endfor redraw " Get target/group character if single_group call Prompt('Target character') else call Prompt('Group character') endif let input_char = nr2char(getchar()) redraw finally " Restore original lines for [line_num, line] in lines_items try undojoin catch endtry call setline(line_num, line['orig']) 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 "}}} function! s:EasyMotion(regexp, direction, ...) " {{{ let orig_pos = [line('.'), col('.')] let targets = [] let visualmode = a:0 > 0 ? a:1 : '' " 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) " Reached end of search range if pos == [0, 0] break endif " Skip folded lines if foldclosed(pos[0]) != -1 continue endif call add(targets, pos) endwhile let targets_len = len(targets) let groups_len = len(s:index_to_key) if targets_len == 0 redraw call Message('No matches') " Restore cursor position call setpos('.', [0, orig_pos[0], orig_pos[1]]) 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 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 " Backward let shade_hl_re = '\%'. line('w0') .'l\_.*' . shade_hl_pos else " Forward let shade_hl_re = shade_hl_pos . '\_.*\%'. line('w$') .'l' endif let shade_hl_id = matchadd(g:EasyMotion_shade_hl, shade_hl_re, 0) endif " Prompt user for target group/character let coords = PromptUser(groups) " Remove shading if g:EasyMotion_shade call matchdelete(shade_hl_id) endif if len(coords) != 2 " Cancelled by user call Message('Operation cancelled') " Restore cursor position/selection if ! empty(visualmode) silent exec 'normal! `<' . visualmode . '`>' else call setpos('.', [0, orig_pos[0], orig_pos[1]]) endif return else 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 call Message('Jumping to [' . coords[0] . ', ' . coords[1] . ']') endif endfunction " }}} " }}}