xref: /vim-8.2.3635/runtime/ftplugin/ruby.vim (revision 89bcfda6)
1071d4279SBram Moolenaar" Vim filetype plugin
2071d4279SBram Moolenaar" Language:		Ruby
3ec7944aaSBram Moolenaar" Maintainer:		Tim Pope <[email protected]>
4ec7944aaSBram Moolenaar" URL:			https://github.com/vim-ruby/vim-ruby
5c6249bb2SBram Moolenaar" Release Coordinator:  Doug Kearns <[email protected]>
660a795aaSBram Moolenaar" ----------------------------------------------------------------------------
7071d4279SBram Moolenaar
8071d4279SBram Moolenaarif (exists("b:did_ftplugin"))
9071d4279SBram Moolenaar  finish
10071d4279SBram Moolenaarendif
11071d4279SBram Moolenaarlet b:did_ftplugin = 1
12071d4279SBram Moolenaar
1360a795aaSBram Moolenaarlet s:cpo_save = &cpo
1460a795aaSBram Moolenaarset cpo&vim
1560a795aaSBram Moolenaar
16756ec0f3SBram Moolenaarif has("gui_running") && !has("gui_win32")
17ec7944aaSBram Moolenaar  setlocal keywordprg=ri\ -T\ -f\ bs
18756ec0f3SBram Moolenaarelse
19756ec0f3SBram Moolenaar  setlocal keywordprg=ri
20756ec0f3SBram Moolenaarendif
21756ec0f3SBram Moolenaar
2260a795aaSBram Moolenaar" Matchit support
2360a795aaSBram Moolenaarif exists("loaded_matchit") && !exists("b:match_words")
2460a795aaSBram Moolenaar  let b:match_ignorecase = 0
2560a795aaSBram Moolenaar
2660a795aaSBram Moolenaar  let b:match_words =
2779166c43SBram Moolenaar	\ '\<\%(if\|unless\|case\|while\|until\|for\|do\|class\|module\|def\|begin\)\>=\@!' .
2860a795aaSBram Moolenaar	\ ':' .
2979166c43SBram Moolenaar	\ '\<\%(else\|elsif\|ensure\|when\|rescue\|break\|redo\|next\|retry\)\>' .
3060a795aaSBram Moolenaar	\ ':' .
31*89bcfda6SBram Moolenaar        \ '\%(^\|[^.\:@$]\)\@<=\<end\:\@!\>' .
3291170f8aSBram Moolenaar	\ ',{:},\[:\],(:)'
3360a795aaSBram Moolenaar
3460a795aaSBram Moolenaar  let b:match_skip =
3560a795aaSBram Moolenaar	\ "synIDattr(synID(line('.'),col('.'),0),'name') =~ '" .
3679166c43SBram Moolenaar	\ "\\<ruby\\%(String\\|StringDelimiter\\|ASCIICode\\|Escape\\|" .
37*89bcfda6SBram Moolenaar        \ "Regexp\\|RegexpDelimiter\\|" .
3879166c43SBram Moolenaar	\ "Interpolation\\|NoInterpolation\\|Comment\\|Documentation\\|" .
3979166c43SBram Moolenaar	\ "ConditionalModifier\\|RepeatModifier\\|OptionalDo\\|" .
4079166c43SBram Moolenaar	\ "Function\\|BlockArgument\\|KeywordAsMethod\\|ClassVariable\\|" .
4179166c43SBram Moolenaar	\ "InstanceVariable\\|GlobalVariable\\|Symbol\\)\\>'"
4260a795aaSBram Moolenaarendif
4360a795aaSBram Moolenaar
4460a795aaSBram Moolenaarsetlocal formatoptions-=t formatoptions+=croql
4560a795aaSBram Moolenaar
46ec7944aaSBram Moolenaarsetlocal include=^\\s*\\<\\(load\\>\\\|require\\>\\\|autoload\\s*:\\=[\"']\\=\\h\\w*[\"']\\=,\\)
47*89bcfda6SBram Moolenaarsetlocal includeexpr=substitute(substitute(v:fname,'::','/','g'),'\%(\.rb\)\=$','.rb','')
4860a795aaSBram Moolenaarsetlocal suffixesadd=.rb
49c6249bb2SBram Moolenaar
50756ec0f3SBram Moolenaarif exists("&ofu") && has("ruby")
519e54a0e7SBram Moolenaar  setlocal omnifunc=rubycomplete#Complete
52c6249bb2SBram Moolenaarendif
5360a795aaSBram Moolenaar
54756ec0f3SBram Moolenaar" To activate, :set ballooneval
55756ec0f3SBram Moolenaarif has('balloon_eval') && exists('+balloonexpr')
56756ec0f3SBram Moolenaar  setlocal balloonexpr=RubyBalloonexpr()
57756ec0f3SBram Moolenaarendif
58756ec0f3SBram Moolenaar
59756ec0f3SBram Moolenaar
6060a795aaSBram Moolenaar" TODO:
6160a795aaSBram Moolenaar"setlocal define=^\\s*def
6260a795aaSBram Moolenaar
6360a795aaSBram Moolenaarsetlocal comments=:#
6460a795aaSBram Moolenaarsetlocal commentstring=#\ %s
6560a795aaSBram Moolenaar
66ec7944aaSBram Moolenaarif !exists('g:ruby_version_paths')
67ec7944aaSBram Moolenaar  let g:ruby_version_paths = {}
6860a795aaSBram Moolenaarendif
69ec7944aaSBram Moolenaar
70*89bcfda6SBram Moolenaarfunction! s:query_path(root) abort
71ec7944aaSBram Moolenaar  let code = "print $:.join %q{,}"
72*89bcfda6SBram Moolenaar  if &shell =~# 'sh'
73*89bcfda6SBram Moolenaar    let prefix = 'env PATH='.shellescape($PATH).' '
7460a795aaSBram Moolenaar  else
75ec7944aaSBram Moolenaar    let prefix = ''
76ec7944aaSBram Moolenaar  endif
77ec7944aaSBram Moolenaar  if &shellxquote == "'"
78*89bcfda6SBram Moolenaar    let path_check = prefix.'ruby --disable-gems -e "' . code . '"'
79ec7944aaSBram Moolenaar  else
80*89bcfda6SBram Moolenaar    let path_check = prefix."ruby --disable-gems -e '" . code . "'"
81ec7944aaSBram Moolenaar  endif
82ec7944aaSBram Moolenaar
83ec7944aaSBram Moolenaar  let cd = haslocaldir() ? 'lcd' : 'cd'
84*89bcfda6SBram Moolenaar  let cwd = fnameescape(getcwd())
85ec7944aaSBram Moolenaar  try
86ec7944aaSBram Moolenaar    exe cd fnameescape(a:root)
87ec7944aaSBram Moolenaar    let path = split(system(path_check),',')
88*89bcfda6SBram Moolenaar    exe cd cwd
89ec7944aaSBram Moolenaar    return path
90ec7944aaSBram Moolenaar  finally
91*89bcfda6SBram Moolenaar    exe cd cwd
92ec7944aaSBram Moolenaar  endtry
93ec7944aaSBram Moolenaarendfunction
94ec7944aaSBram Moolenaar
95*89bcfda6SBram Moolenaarfunction! s:build_path(path) abort
96ec7944aaSBram Moolenaar  let path = join(map(copy(a:path), 'v:val ==# "." ? "" : v:val'), ',')
97ec7944aaSBram Moolenaar  if &g:path !~# '\v^\.%(,/%(usr|emx)/include)=,,$'
98ec7944aaSBram Moolenaar    let path = substitute(&g:path,',,$',',','') . ',' . path
99ec7944aaSBram Moolenaar  endif
100ec7944aaSBram Moolenaar  return path
101ec7944aaSBram Moolenaarendfunction
102ec7944aaSBram Moolenaar
103ec7944aaSBram Moolenaarif !exists('b:ruby_version') && !exists('g:ruby_path') && isdirectory(expand('%:p:h'))
104ec7944aaSBram Moolenaar  let s:version_file = findfile('.ruby-version', '.;')
105*89bcfda6SBram Moolenaar  if !empty(s:version_file) && filereadable(s:version_file)
106ec7944aaSBram Moolenaar    let b:ruby_version = get(readfile(s:version_file, '', 1), '')
107ec7944aaSBram Moolenaar    if !has_key(g:ruby_version_paths, b:ruby_version)
108ec7944aaSBram Moolenaar      let g:ruby_version_paths[b:ruby_version] = s:query_path(fnamemodify(s:version_file, ':p:h'))
109ec7944aaSBram Moolenaar    endif
11060a795aaSBram Moolenaar  endif
11160a795aaSBram Moolenaarendif
11260a795aaSBram Moolenaar
113ec7944aaSBram Moolenaarif exists("g:ruby_path")
114ec7944aaSBram Moolenaar  let s:ruby_path = type(g:ruby_path) == type([]) ? join(g:ruby_path, ',') : g:ruby_path
115ec7944aaSBram Moolenaarelseif has_key(g:ruby_version_paths, get(b:, 'ruby_version', ''))
116ec7944aaSBram Moolenaar  let s:ruby_paths = g:ruby_version_paths[b:ruby_version]
117ec7944aaSBram Moolenaar  let s:ruby_path = s:build_path(s:ruby_paths)
118ec7944aaSBram Moolenaarelse
119ec7944aaSBram Moolenaar  if !exists('g:ruby_default_path')
120ec7944aaSBram Moolenaar    if has("ruby") && has("win32")
121ec7944aaSBram Moolenaar      ruby ::VIM::command( 'let g:ruby_default_path = split("%s",",")' % $:.join(%q{,}) )
122ec7944aaSBram Moolenaar    elseif executable('ruby')
123ec7944aaSBram Moolenaar      let g:ruby_default_path = s:query_path($HOME)
124ec7944aaSBram Moolenaar    else
125ec7944aaSBram Moolenaar      let g:ruby_default_path = map(split($RUBYLIB,':'), 'v:val ==# "." ? "" : v:val')
126ec7944aaSBram Moolenaar    endif
127ec7944aaSBram Moolenaar  endif
128ec7944aaSBram Moolenaar  let s:ruby_paths = g:ruby_default_path
129ec7944aaSBram Moolenaar  let s:ruby_path = s:build_path(s:ruby_paths)
130ec7944aaSBram Moolenaarendif
131ec7944aaSBram Moolenaar
132ec7944aaSBram Moolenaarif stridx(&l:path, s:ruby_path) == -1
1331d68952aSBram Moolenaar  let &l:path = s:ruby_path
134ec7944aaSBram Moolenaarendif
135ec7944aaSBram Moolenaarif exists('s:ruby_paths') && stridx(&l:tags, join(map(copy(s:ruby_paths),'v:val."/tags"'),',')) == -1
136ec7944aaSBram Moolenaar  let &l:tags = &tags . ',' . join(map(copy(s:ruby_paths),'v:val."/tags"'),',')
137ec7944aaSBram Moolenaarendif
13860a795aaSBram Moolenaar
139*89bcfda6SBram Moolenaarif (has("gui_win32") || has("gui_gtk")) && !exists("b:browsefilter")
14060a795aaSBram Moolenaar  let b:browsefilter = "Ruby Source Files (*.rb)\t*.rb\n" .
14160a795aaSBram Moolenaar                     \ "All Files (*.*)\t*.*\n"
14260a795aaSBram Moolenaarendif
14360a795aaSBram Moolenaar
144ec7944aaSBram Moolenaarlet b:undo_ftplugin = "setl fo< inc< inex< sua< def< com< cms< path< tags< kp<"
14579166c43SBram Moolenaar      \."| unlet! b:browsefilter b:match_ignorecase b:match_words b:match_skip"
14679166c43SBram Moolenaar      \."| if exists('&ofu') && has('ruby') | setl ofu< | endif"
14779166c43SBram Moolenaar      \."| if has('balloon_eval') && exists('+bexpr') | setl bexpr< | endif"
14879166c43SBram Moolenaar
149*89bcfda6SBram Moolenaarfunction! s:map(mode, flags, map) abort
150*89bcfda6SBram Moolenaar  let from = matchstr(a:map, '\S\+')
151*89bcfda6SBram Moolenaar  if empty(mapcheck(from, a:mode))
152*89bcfda6SBram Moolenaar    exe a:mode.'map' '<buffer>'.(a:0 ? a:1 : '') a:map
153*89bcfda6SBram Moolenaar    let b:undo_ftplugin .= '|sil! '.a:mode.'unmap <buffer> '.from
154*89bcfda6SBram Moolenaar  endif
155*89bcfda6SBram Moolenaarendfunction
156*89bcfda6SBram Moolenaar
157*89bcfda6SBram Moolenaarcmap <buffer><script><expr> <Plug><cword> substitute(RubyCursorIdentifier(),'^$',"\022\027",'')
158*89bcfda6SBram Moolenaarcmap <buffer><script><expr> <Plug><cfile> substitute(RubyCursorFile(),'^$',"\022\006",'')
159*89bcfda6SBram Moolenaarlet b:undo_ftplugin .= "| sil! cunmap <buffer> <Plug><cword>| sil! cunmap <buffer> <Plug><cfile>"
160*89bcfda6SBram Moolenaar
16179166c43SBram Moolenaarif !exists("g:no_plugin_maps") && !exists("g:no_ruby_maps")
162*89bcfda6SBram Moolenaar  nmap <buffer><script> <SID>:  :<C-U>
163*89bcfda6SBram Moolenaar  nmap <buffer><script> <SID>c: :<C-U><C-R>=v:count ? v:count : ''<CR>
164*89bcfda6SBram Moolenaar
1651d68952aSBram Moolenaar  nnoremap <silent> <buffer> [m :<C-U>call <SID>searchsyn('\<def\>','rubyDefine','b','n')<CR>
1661d68952aSBram Moolenaar  nnoremap <silent> <buffer> ]m :<C-U>call <SID>searchsyn('\<def\>','rubyDefine','','n')<CR>
1671d68952aSBram Moolenaar  nnoremap <silent> <buffer> [M :<C-U>call <SID>searchsyn('\<end\>','rubyDefine','b','n')<CR>
1681d68952aSBram Moolenaar  nnoremap <silent> <buffer> ]M :<C-U>call <SID>searchsyn('\<end\>','rubyDefine','','n')<CR>
1691d68952aSBram Moolenaar  xnoremap <silent> <buffer> [m :<C-U>call <SID>searchsyn('\<def\>','rubyDefine','b','v')<CR>
1701d68952aSBram Moolenaar  xnoremap <silent> <buffer> ]m :<C-U>call <SID>searchsyn('\<def\>','rubyDefine','','v')<CR>
1711d68952aSBram Moolenaar  xnoremap <silent> <buffer> [M :<C-U>call <SID>searchsyn('\<end\>','rubyDefine','b','v')<CR>
1721d68952aSBram Moolenaar  xnoremap <silent> <buffer> ]M :<C-U>call <SID>searchsyn('\<end\>','rubyDefine','','v')<CR>
17379166c43SBram Moolenaar
1741d68952aSBram Moolenaar  nnoremap <silent> <buffer> [[ :<C-U>call <SID>searchsyn('\<\%(class\<Bar>module\)\>','rubyModule\<Bar>rubyClass','b','n')<CR>
1751d68952aSBram Moolenaar  nnoremap <silent> <buffer> ]] :<C-U>call <SID>searchsyn('\<\%(class\<Bar>module\)\>','rubyModule\<Bar>rubyClass','','n')<CR>
1761d68952aSBram Moolenaar  nnoremap <silent> <buffer> [] :<C-U>call <SID>searchsyn('\<end\>','rubyModule\<Bar>rubyClass','b','n')<CR>
1771d68952aSBram Moolenaar  nnoremap <silent> <buffer> ][ :<C-U>call <SID>searchsyn('\<end\>','rubyModule\<Bar>rubyClass','','n')<CR>
1781d68952aSBram Moolenaar  xnoremap <silent> <buffer> [[ :<C-U>call <SID>searchsyn('\<\%(class\<Bar>module\)\>','rubyModule\<Bar>rubyClass','b','v')<CR>
1791d68952aSBram Moolenaar  xnoremap <silent> <buffer> ]] :<C-U>call <SID>searchsyn('\<\%(class\<Bar>module\)\>','rubyModule\<Bar>rubyClass','','v')<CR>
1801d68952aSBram Moolenaar  xnoremap <silent> <buffer> [] :<C-U>call <SID>searchsyn('\<end\>','rubyModule\<Bar>rubyClass','b','v')<CR>
1811d68952aSBram Moolenaar  xnoremap <silent> <buffer> ][ :<C-U>call <SID>searchsyn('\<end\>','rubyModule\<Bar>rubyClass','','v')<CR>
18279166c43SBram Moolenaar
18379166c43SBram Moolenaar  let b:undo_ftplugin = b:undo_ftplugin
18479166c43SBram Moolenaar        \."| sil! exe 'unmap <buffer> [[' | sil! exe 'unmap <buffer> ]]' | sil! exe 'unmap <buffer> []' | sil! exe 'unmap <buffer> ]['"
18579166c43SBram Moolenaar        \."| sil! exe 'unmap <buffer> [m' | sil! exe 'unmap <buffer> ]m' | sil! exe 'unmap <buffer> [M' | sil! exe 'unmap <buffer> ]M'"
1861d68952aSBram Moolenaar
187*89bcfda6SBram Moolenaar  if maparg('im','x') == '' && maparg('im','o') == '' && maparg('am','x') == '' && maparg('am','o') == ''
188ec7944aaSBram Moolenaar    onoremap <silent> <buffer> im :<C-U>call <SID>wrap_i('[m',']M')<CR>
189ec7944aaSBram Moolenaar    onoremap <silent> <buffer> am :<C-U>call <SID>wrap_a('[m',']M')<CR>
190ec7944aaSBram Moolenaar    xnoremap <silent> <buffer> im :<C-U>call <SID>wrap_i('[m',']M')<CR>
191ec7944aaSBram Moolenaar    xnoremap <silent> <buffer> am :<C-U>call <SID>wrap_a('[m',']M')<CR>
192ec7944aaSBram Moolenaar    let b:undo_ftplugin = b:undo_ftplugin
193ec7944aaSBram Moolenaar          \."| sil! exe 'ounmap <buffer> im' | sil! exe 'ounmap <buffer> am'"
194ec7944aaSBram Moolenaar          \."| sil! exe 'xunmap <buffer> im' | sil! exe 'xunmap <buffer> am'"
195ec7944aaSBram Moolenaar  endif
196ec7944aaSBram Moolenaar
197*89bcfda6SBram Moolenaar  if maparg('iM','x') == '' && maparg('iM','o') == '' && maparg('aM','x') == '' && maparg('aM','o') == ''
198ec7944aaSBram Moolenaar    onoremap <silent> <buffer> iM :<C-U>call <SID>wrap_i('[[','][')<CR>
199ec7944aaSBram Moolenaar    onoremap <silent> <buffer> aM :<C-U>call <SID>wrap_a('[[','][')<CR>
200ec7944aaSBram Moolenaar    xnoremap <silent> <buffer> iM :<C-U>call <SID>wrap_i('[[','][')<CR>
201ec7944aaSBram Moolenaar    xnoremap <silent> <buffer> aM :<C-U>call <SID>wrap_a('[[','][')<CR>
202ec7944aaSBram Moolenaar    let b:undo_ftplugin = b:undo_ftplugin
203ec7944aaSBram Moolenaar          \."| sil! exe 'ounmap <buffer> iM' | sil! exe 'ounmap <buffer> aM'"
204ec7944aaSBram Moolenaar          \."| sil! exe 'xunmap <buffer> iM' | sil! exe 'xunmap <buffer> aM'"
205ec7944aaSBram Moolenaar  endif
206ec7944aaSBram Moolenaar
207*89bcfda6SBram Moolenaar  call s:map('c', '', '<C-R><C-W> <Plug><cword>')
208*89bcfda6SBram Moolenaar  call s:map('c', '', '<C-R><C-F> <Plug><cfile>')
209ec7944aaSBram Moolenaar
210*89bcfda6SBram Moolenaar  cmap <buffer><script><expr> <SID>tagzv &foldopen =~# 'tag' ? '<Bar>norm! zv' : ''
211*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-]>       <SID>:exe  v:count1."tag <Plug><cword>"<SID>tagzv<CR>')
212*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', 'g<C-]>      <SID>:exe         "tjump <Plug><cword>"<SID>tagzv<CR>')
213*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', 'g]          <SID>:exe       "tselect <Plug><cword>"<SID>tagzv<CR>')
214*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>]      <SID>:exe v:count1."stag <Plug><cword>"<SID>tagzv<CR>')
215*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W><C-]>  <SID>:exe v:count1."stag <Plug><cword>"<SID>tagzv<CR>')
216*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>g<C-]> <SID>:exe        "stjump <Plug><cword>"<SID>tagzv<CR>')
217*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>g]     <SID>:exe      "stselect <Plug><cword>"<SID>tagzv<CR>')
218*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>}      <SID>:exe v:count1."ptag <Plug><cword>"<CR>')
219*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>g}     <SID>:exe        "ptjump <Plug><cword>"<CR>')
220*89bcfda6SBram Moolenaar
221*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', 'gf           <SID>c:find <Plug><cfile><CR>')
222*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>f      <SID>c:sfind <Plug><cfile><CR>')
223*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W><C-F>  <SID>c:sfind <Plug><cfile><CR>')
224*89bcfda6SBram Moolenaar  call s:map('n', '<silent>', '<C-W>gf   <SID>c:tabfind <Plug><cfile><CR>')
22579166c43SBram Moolenaarendif
22660a795aaSBram Moolenaar
22760a795aaSBram Moolenaarlet &cpo = s:cpo_save
22860a795aaSBram Moolenaarunlet s:cpo_save
22960a795aaSBram Moolenaar
230756ec0f3SBram Moolenaarif exists("g:did_ruby_ftplugin_functions")
231756ec0f3SBram Moolenaar  finish
232756ec0f3SBram Moolenaarendif
233756ec0f3SBram Moolenaarlet g:did_ruby_ftplugin_functions = 1
234756ec0f3SBram Moolenaar
235*89bcfda6SBram Moolenaarfunction! RubyBalloonexpr() abort
236756ec0f3SBram Moolenaar  if !exists('s:ri_found')
237756ec0f3SBram Moolenaar    let s:ri_found = executable('ri')
238756ec0f3SBram Moolenaar  endif
239756ec0f3SBram Moolenaar  if s:ri_found
240756ec0f3SBram Moolenaar    let line = getline(v:beval_lnum)
241756ec0f3SBram Moolenaar    let b = matchstr(strpart(line,0,v:beval_col),'\%(\w\|[:.]\)*$')
242756ec0f3SBram Moolenaar    let a = substitute(matchstr(strpart(line,v:beval_col),'^\w*\%([?!]\|\s*=\)\?'),'\s\+','','g')
243756ec0f3SBram Moolenaar    let str = b.a
244756ec0f3SBram Moolenaar    let before = strpart(line,0,v:beval_col-strlen(b))
245756ec0f3SBram Moolenaar    let after  = strpart(line,v:beval_col+strlen(a))
246756ec0f3SBram Moolenaar    if str =~ '^\.'
247756ec0f3SBram Moolenaar      let str = substitute(str,'^\.','#','g')
248756ec0f3SBram Moolenaar      if before =~ '\]\s*$'
249756ec0f3SBram Moolenaar        let str = 'Array'.str
250756ec0f3SBram Moolenaar      elseif before =~ '}\s*$'
251756ec0f3SBram Moolenaar        " False positives from blocks here
252756ec0f3SBram Moolenaar        let str = 'Hash'.str
253756ec0f3SBram Moolenaar      elseif before =~ "[\"'`]\\s*$" || before =~ '\$\d\+\s*$'
254756ec0f3SBram Moolenaar        let str = 'String'.str
255756ec0f3SBram Moolenaar      elseif before =~ '\$\d\+\.\d\+\s*$'
256756ec0f3SBram Moolenaar        let str = 'Float'.str
257756ec0f3SBram Moolenaar      elseif before =~ '\$\d\+\s*$'
258756ec0f3SBram Moolenaar        let str = 'Integer'.str
259756ec0f3SBram Moolenaar      elseif before =~ '/\s*$'
260756ec0f3SBram Moolenaar        let str = 'Regexp'.str
261756ec0f3SBram Moolenaar      else
262756ec0f3SBram Moolenaar        let str = substitute(str,'^#','.','')
263756ec0f3SBram Moolenaar      endif
264756ec0f3SBram Moolenaar    endif
265756ec0f3SBram Moolenaar    let str = substitute(str,'.*\.\s*to_f\s*\.\s*','Float#','')
266756ec0f3SBram Moolenaar    let str = substitute(str,'.*\.\s*to_i\%(nt\)\=\s*\.\s*','Integer#','')
267756ec0f3SBram Moolenaar    let str = substitute(str,'.*\.\s*to_s\%(tr\)\=\s*\.\s*','String#','')
268756ec0f3SBram Moolenaar    let str = substitute(str,'.*\.\s*to_sym\s*\.\s*','Symbol#','')
269756ec0f3SBram Moolenaar    let str = substitute(str,'.*\.\s*to_a\%(ry\)\=\s*\.\s*','Array#','')
270756ec0f3SBram Moolenaar    let str = substitute(str,'.*\.\s*to_proc\s*\.\s*','Proc#','')
271756ec0f3SBram Moolenaar    if str !~ '^\w'
272756ec0f3SBram Moolenaar      return ''
273756ec0f3SBram Moolenaar    endif
274ec7944aaSBram Moolenaar    silent! let res = substitute(system("ri -f rdoc -T \"".str.'"'),'\n$','','')
27579166c43SBram Moolenaar    if res =~ '^Nothing known about' || res =~ '^Bad argument:' || res =~ '^More than one method'
276756ec0f3SBram Moolenaar      return ''
277756ec0f3SBram Moolenaar    endif
278756ec0f3SBram Moolenaar    return res
279756ec0f3SBram Moolenaar  else
280756ec0f3SBram Moolenaar    return ""
281756ec0f3SBram Moolenaar  endif
282756ec0f3SBram Moolenaarendfunction
283756ec0f3SBram Moolenaar
284*89bcfda6SBram Moolenaarfunction! s:searchsyn(pattern, syn, flags, mode) abort
285*89bcfda6SBram Moolenaar  let cnt = v:count1
28679166c43SBram Moolenaar  norm! m'
2871d68952aSBram Moolenaar  if a:mode ==# 'v'
2881d68952aSBram Moolenaar    norm! gv
2891d68952aSBram Moolenaar  endif
29079166c43SBram Moolenaar  let i = 0
29179166c43SBram Moolenaar  while i < cnt
29279166c43SBram Moolenaar    let i = i + 1
29379166c43SBram Moolenaar    let line = line('.')
29479166c43SBram Moolenaar    let col  = col('.')
29579166c43SBram Moolenaar    let pos = search(a:pattern,'W'.a:flags)
29679166c43SBram Moolenaar    while pos != 0 && s:synname() !~# a:syn
29779166c43SBram Moolenaar      let pos = search(a:pattern,'W'.a:flags)
29879166c43SBram Moolenaar    endwhile
29979166c43SBram Moolenaar    if pos == 0
30079166c43SBram Moolenaar      call cursor(line,col)
30179166c43SBram Moolenaar      return
30279166c43SBram Moolenaar    endif
30379166c43SBram Moolenaar  endwhile
30479166c43SBram Moolenaarendfunction
30579166c43SBram Moolenaar
306*89bcfda6SBram Moolenaarfunction! s:synname() abort
30779166c43SBram Moolenaar  return synIDattr(synID(line('.'),col('.'),0),'name')
30879166c43SBram Moolenaarendfunction
309756ec0f3SBram Moolenaar
310*89bcfda6SBram Moolenaarfunction! s:wrap_i(back,forward) abort
311ec7944aaSBram Moolenaar  execute 'norm k'.a:forward
312ec7944aaSBram Moolenaar  let line = line('.')
313ec7944aaSBram Moolenaar  execute 'norm '.a:back
314ec7944aaSBram Moolenaar  if line('.') == line - 1
315ec7944aaSBram Moolenaar    return s:wrap_a(a:back,a:forward)
316ec7944aaSBram Moolenaar  endif
317ec7944aaSBram Moolenaar  execute 'norm jV'.a:forward.'k'
318ec7944aaSBram Moolenaarendfunction
319ec7944aaSBram Moolenaar
320*89bcfda6SBram Moolenaarfunction! s:wrap_a(back,forward) abort
321ec7944aaSBram Moolenaar  execute 'norm '.a:forward
322ec7944aaSBram Moolenaar  if line('.') < line('$') && getline(line('.')+1) ==# ''
323ec7944aaSBram Moolenaar    let after = 1
324ec7944aaSBram Moolenaar  endif
325ec7944aaSBram Moolenaar  execute 'norm '.a:back
326ec7944aaSBram Moolenaar  while getline(line('.')-1) =~# '^\s*#' && line('.')
327ec7944aaSBram Moolenaar    -
328ec7944aaSBram Moolenaar  endwhile
329ec7944aaSBram Moolenaar  if exists('after')
330ec7944aaSBram Moolenaar    execute 'norm V'.a:forward.'j'
331ec7944aaSBram Moolenaar  elseif line('.') > 1 && getline(line('.')-1) =~# '^\s*$'
332ec7944aaSBram Moolenaar    execute 'norm kV'.a:forward
333ec7944aaSBram Moolenaar  else
334ec7944aaSBram Moolenaar    execute 'norm V'.a:forward
335ec7944aaSBram Moolenaar  endif
336ec7944aaSBram Moolenaarendfunction
337ec7944aaSBram Moolenaar
338*89bcfda6SBram Moolenaarfunction! RubyCursorIdentifier() abort
3391d68952aSBram Moolenaar  let asciicode    = '\%(\w\|[]})\"'."'".']\)\@<!\%(?\%(\\M-\\C-\|\\C-\\M-\|\\M-\\c\|\\c\\M-\|\\c\|\\C-\|\\M-\)\=\%(\\\o\{1,3}\|\\x\x\{1,2}\|\\\=\S\)\)'
3401d68952aSBram Moolenaar  let number       = '\%(\%(\w\|[]})\"'."'".']\s*\)\@<!-\)\=\%(\<[[:digit:]_]\+\%(\.[[:digit:]_]\+\)\=\%([Ee][[:digit:]_]\+\)\=\>\|\<0[xXbBoOdD][[:xdigit:]_]\+\>\)\|'.asciicode
3411d68952aSBram Moolenaar  let operator     = '\%(\[\]\|<<\|<=>\|[!<>]=\=\|===\=\|[!=]\~\|>>\|\*\*\|\.\.\.\=\|=>\|[~^&|*/%+-]\)'
342*89bcfda6SBram Moolenaar  let method       = '\%(\.[_a-zA-Z]\w*\s*=>\@!\|\<[_a-zA-Z]\w*\>[?!]\=\)'
3431d68952aSBram Moolenaar  let global       = '$\%([!$&"'."'".'*+,./:;<=>?@\`~]\|-\=\w\+\>\)'
3441d68952aSBram Moolenaar  let symbolizable = '\%(\%(@@\=\)\w\+\>\|'.global.'\|'.method.'\|'.operator.'\)'
3451d68952aSBram Moolenaar  let pattern      = '\C\s*\%('.number.'\|\%(:\@<!:\)\='.symbolizable.'\)'
3461d68952aSBram Moolenaar  let [lnum, col]  = searchpos(pattern,'bcn',line('.'))
3471d68952aSBram Moolenaar  let raw          = matchstr(getline('.')[col-1 : ],pattern)
348*89bcfda6SBram Moolenaar  let stripped     = substitute(substitute(raw,'\s\+=$','=',''),'^\s*[:.]\=','','')
3491d68952aSBram Moolenaar  return stripped == '' ? expand("<cword>") : stripped
3501d68952aSBram Moolenaarendfunction
3511d68952aSBram Moolenaar
352*89bcfda6SBram Moolenaarfunction! RubyCursorFile() abort
353*89bcfda6SBram Moolenaar  let isfname = &isfname
354*89bcfda6SBram Moolenaar  try
355*89bcfda6SBram Moolenaar    set isfname+=:
356*89bcfda6SBram Moolenaar    let cfile = expand('<cfile>')
357*89bcfda6SBram Moolenaar  finally
358*89bcfda6SBram Moolenaar    let isfname = &isfname
359*89bcfda6SBram Moolenaar  endtry
360*89bcfda6SBram Moolenaar  let pre = matchstr(strpart(getline('.'), 0, col('.')-1), '.*\f\@<!')
361*89bcfda6SBram Moolenaar  let post = matchstr(strpart(getline('.'), col('.')), '\f\@!.*')
362*89bcfda6SBram Moolenaar  let ext = getline('.') =~# '^\s*\%(require\%(_relative\)\=\|autoload\)\>' && cfile !~# '\.rb$' ? '.rb' : ''
363*89bcfda6SBram Moolenaar  if s:synname() ==# 'rubyConstant'
364*89bcfda6SBram Moolenaar    let cfile = substitute(cfile,'\.\w\+[?!=]\=$','','')
365*89bcfda6SBram Moolenaar    let cfile = substitute(cfile,'::','/','g')
366*89bcfda6SBram Moolenaar    let cfile = substitute(cfile,'\(\u\+\)\(\u\l\)','\1_\2', 'g')
367*89bcfda6SBram Moolenaar    let cfile = substitute(cfile,'\(\l\|\d\)\(\u\)','\1_\2', 'g')
368*89bcfda6SBram Moolenaar    return tolower(cfile) . '.rb'
369*89bcfda6SBram Moolenaar  elseif getline('.') =~# '^\s*require_relative\s*\(["'']\).*\1\s*$'
370*89bcfda6SBram Moolenaar    let cfile = expand('%:p:h') . '/' . matchstr(getline('.'),'\(["'']\)\zs.\{-\}\ze\1') . ext
371*89bcfda6SBram Moolenaar  elseif getline('.') =~# '^\s*\%(require[( ]\|load[( ]\|autoload[( ]:\w\+,\)\s*\%(::\)\=File\.expand_path(\(["'']\)\.\./.*\1,\s*__FILE__)\s*$'
372*89bcfda6SBram Moolenaar    let target = matchstr(getline('.'),'\(["'']\)\.\.\zs/.\{-\}\ze\1')
373*89bcfda6SBram Moolenaar    let cfile = expand('%:p:h') . target . ext
374ec7944aaSBram Moolenaar  elseif getline('.') =~# '^\s*\%(require \|load \|autoload :\w\+,\)\s*\(["'']\).*\1\s*$'
375*89bcfda6SBram Moolenaar    let cfile = matchstr(getline('.'),'\(["'']\)\zs.\{-\}\ze\1') . ext
376*89bcfda6SBram Moolenaar  elseif pre.post =~# '\<File.expand_path[( ].*[''"]\{2\}, *__FILE__\>' && cfile =~# '^\.\.'
377*89bcfda6SBram Moolenaar    let cfile = expand('%:p:h') . strpart(cfile, 2)
378ec7944aaSBram Moolenaar  else
379*89bcfda6SBram Moolenaar    return substitute(cfile, '\C\v^(.*):(\d+)%(:in)=$', '+\2 \1', '')
380ec7944aaSBram Moolenaar  endif
381*89bcfda6SBram Moolenaar  let cwdpat = '^\M' . substitute(getcwd(), '[\/]', '\\[\\/]', 'g').'\ze\[\/]'
382*89bcfda6SBram Moolenaar  let cfile = substitute(cfile, cwdpat, '.', '')
383*89bcfda6SBram Moolenaar  if fnameescape(cfile) !=# cfile
384*89bcfda6SBram Moolenaar    return '+ '.fnameescape(cfile)
385ec7944aaSBram Moolenaar  else
386*89bcfda6SBram Moolenaar    return cfile
387ec7944aaSBram Moolenaar  endif
388ec7944aaSBram Moolenaarendfunction
389ec7944aaSBram Moolenaar
39060a795aaSBram Moolenaar"
39160a795aaSBram Moolenaar" Instructions for enabling "matchit" support:
39260a795aaSBram Moolenaar"
39360a795aaSBram Moolenaar" 1. Look for the latest "matchit" plugin at
39460a795aaSBram Moolenaar"
39560a795aaSBram Moolenaar"         http://www.vim.org/scripts/script.php?script_id=39
39660a795aaSBram Moolenaar"
39760a795aaSBram Moolenaar"    It is also packaged with Vim, in the $VIMRUNTIME/macros directory.
39860a795aaSBram Moolenaar"
39960a795aaSBram Moolenaar" 2. Copy "matchit.txt" into a "doc" directory (e.g. $HOME/.vim/doc).
40060a795aaSBram Moolenaar"
40160a795aaSBram Moolenaar" 3. Copy "matchit.vim" into a "plugin" directory (e.g. $HOME/.vim/plugin).
40260a795aaSBram Moolenaar"
40360a795aaSBram Moolenaar" 4. Ensure this file (ftplugin/ruby.vim) is installed.
40460a795aaSBram Moolenaar"
40560a795aaSBram Moolenaar" 5. Ensure you have this line in your $HOME/.vimrc:
40660a795aaSBram Moolenaar"         filetype plugin on
40760a795aaSBram Moolenaar"
40860a795aaSBram Moolenaar" 6. Restart Vim and create the matchit documentation:
40960a795aaSBram Moolenaar"
41060a795aaSBram Moolenaar"         :helptags ~/.vim/doc
41160a795aaSBram Moolenaar"
41260a795aaSBram Moolenaar"    Now you can do ":help matchit", and you should be able to use "%" on Ruby
41360a795aaSBram Moolenaar"    keywords.  Try ":echo b:match_words" to be sure.
41460a795aaSBram Moolenaar"
41560a795aaSBram Moolenaar" Thanks to Mark J. Reed for the instructions.  See ":help vimrc" for the
41660a795aaSBram Moolenaar" locations of plugin directories, etc., as there are several options, and it
41760a795aaSBram Moolenaar" differs on Windows.  Email [email protected] if you need help.
41860a795aaSBram Moolenaar"
41960a795aaSBram Moolenaar
420446cb837SBram Moolenaar" vim: nowrap sw=2 sts=2 ts=8:
421