2014-07-03 15:16:59 +02:00
# http://ruby-lang.org
# ‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾
# Detection
# ‾‾‾‾‾‾‾‾‾
2018-09-17 16:29:05 +02:00
hook global BufCreate .*(([.](rb))|(irbrc)|(pryrc)|(Brewfile)|(Capfile|[.]cap)|(Gemfile|[.]gemspec)|(Guardfile)|(Rakefile|[.]rake)|(Thorfile|[.]thor)|(Vagrantfile)) %{
2017-11-03 08:34:41 +01:00
set-option buffer filetype ruby
2014-07-03 15:16:59 +02:00
}
2019-04-10 05:54:19 +02:00
# Initialization
# ‾‾‾‾‾‾‾‾‾‾‾‾‾‾
hook global WinSetOption filetype=ruby %{
2019-03-13 06:24:33 +01:00
require-module ruby
2019-04-10 05:54:19 +02:00
set-option window static_words %opt{ruby_static_words}
hook window InsertChar .* -group ruby-indent ruby-indent-on-char
hook window InsertChar \n -group ruby-indent ruby-indent-on-new-line
2019-06-07 23:25:33 +02:00
hook window InsertChar \n -group ruby-insert ruby-insert-on-new-line
2019-04-10 05:54:19 +02:00
alias window alt ruby-alternative-file
hook -once -always window WinSetOption filetype=.* %{
remove-hooks window ruby-.+
unalias window alt ruby-alternative-file
}
}
hook -group ruby-highlight global WinSetOption filetype=ruby %{
add-highlighter window/ruby ref ruby
hook -once -always window WinSetOption filetype=.* %{ remove-highlighter window/ruby }
2019-03-13 06:24:33 +01:00
}
provide-module ruby %[
2014-07-03 15:16:59 +02:00
# Highlighters
# ‾‾‾‾‾‾‾‾‾‾‾‾
2018-06-28 14:10:22 +02:00
add-highlighter shared/ruby regions
add-highlighter shared/ruby/code default-region group
2019-03-15 18:32:49 +01:00
add-highlighter shared/ruby/double_symbol region ':"' (?<!\\)(\\\\)*" regions
add-highlighter shared/ruby/single_symbol region ":'" (?<!\\)(\\\\)*' fill variable
add-highlighter shared/ruby/double_string region '"' (?<!\\)(\\\\)*" regions
add-highlighter shared/ruby/single_string region "'" (?<!\\)(\\\\)*' fill string
add-highlighter shared/ruby/backtick region '(?<![$:])`' (?<!\\)(\\\\)*` regions
add-highlighter shared/ruby/regex region '(?<![$:])/' (?<!\\)(\\\\)*/[imox]* regions
add-highlighter shared/ruby/ region '#' '$' fill comment
add-highlighter shared/ruby/ region ^=begin ^=end fill comment
2020-07-09 11:24:22 +02:00
add-highlighter shared/ruby/ region -recurse \( '%[iqrswxIQRSWX]?\(' \) fill meta
add-highlighter shared/ruby/ region -recurse \{ '%[iqrswxIQRSWX]?\{' \} fill meta
add-highlighter shared/ruby/ region -recurse \[ '%[iqrswxIQRSWX]?\[' \] fill meta
add-highlighter shared/ruby/ region -recurse < '%[iqrswxIQRSWX]?<' > fill meta
2018-11-21 22:44:41 +01:00
add-highlighter shared/ruby/heredoc region '<<[-~]?(?!self)(\w+)' '^\h*(\w+)$' fill string
2020-07-20 21:53:45 +02:00
add-highlighter shared/ruby/division region '[\w\)\]]\K(/|(\h+/\h+))' '\w' group # Help Kakoune to better detect /…/ literals
2014-07-03 15:16:59 +02:00
# Regular expression flags are: i → ignore case, m → multi-lines, o → only interpolate #{} blocks once, x → extended mode (ignore white spaces)
# Literals are: i → array of symbols, q → string, r → regular expression, s → symbol, w → array of words, x → capture shell result
2018-06-28 14:10:22 +02:00
add-highlighter shared/ruby/double_string/ default-region fill string
2018-07-02 12:59:12 +02:00
add-highlighter shared/ruby/double_string/interpolation region -recurse \{ \Q#{ \} fill meta
2014-07-14 22:42:19 +02:00
2019-03-15 18:32:49 +01:00
add-highlighter shared/ruby/double_symbol/ default-region fill variable
add-highlighter shared/ruby/double_symbol/interpolation region -recurse \{ \Q#{ \} fill meta
2018-06-28 14:10:22 +02:00
add-highlighter shared/ruby/backtick/ default-region fill meta
2018-07-02 12:59:12 +02:00
add-highlighter shared/ruby/backtick/interpolation region -recurse \{ \Q#{ \} fill meta
2014-07-14 22:42:19 +02:00
2018-06-28 14:10:22 +02:00
add-highlighter shared/ruby/regex/ default-region fill meta
2018-07-02 12:59:12 +02:00
add-highlighter shared/ruby/regex/interpolation region -recurse \{ \Q#{ \} fill meta
2014-07-14 22:42:19 +02:00
2018-05-06 23:29:52 +02:00
evaluate-commands %sh{
2016-03-14 13:08:51 +01:00
# Grammar
2016-03-14 17:29:11 +01:00
# Keywords are collected searching for keywords at
2016-03-14 13:08:51 +01:00
# https://github.com/ruby/ruby/blob/trunk/parse.y
2016-03-14 17:29:11 +01:00
keywords="alias|and|begin|break|case|class|def|defined|do|else|elsif|end"
2018-02-08 13:34:40 +01:00
keywords="${keywords}|ensure|false|for|if|in|module|next|nil|not|or|private|protected|public|redo"
2016-03-14 17:29:11 +01:00
keywords="${keywords}|rescue|retry|return|self|super|then|true|undef|unless|until|when|while|yield"
attributes="attr_reader|attr_writer|attr_accessor"
values="false|true|nil"
2016-12-25 20:01:56 +01:00
meta="require|include|extend"
2016-03-14 13:08:51 +01:00
# Add the language's grammar to the static completion list
2019-04-10 05:54:19 +02:00
printf %s\\n "declare-option str-list ruby_static_words ${keywords} ${attributes} ${values} ${meta}" | tr '|' ' '
2016-03-14 13:08:51 +01:00
# Highlight keywords
2016-03-17 14:30:14 +01:00
printf %s "
2020-03-23 08:20:02 +01:00
add-highlighter shared/ruby/code/ regex \b(${keywords})[^0-9A-Za-z_!?] 1:keyword
2018-06-28 14:10:22 +02:00
add-highlighter shared/ruby/code/ regex \b(${attributes})\b 0:attribute
add-highlighter shared/ruby/code/ regex \b(${values})\b 0:value
add-highlighter shared/ruby/code/ regex \b(${meta})\b 0:meta
2016-03-14 13:08:51 +01:00
"
}
2014-07-03 15:16:59 +02:00
2019-03-15 18:32:49 +01:00
add-highlighter shared/ruby/code/ regex \b(\w+:(?!:))|(:?(\$(-[0FIKWadilpvw]|["'`/~&+=!$*,:.\;<>?@\\])|(\$|@@?)\w+))|((?<!:):(![~=]|=~|>[=>]?|<((=>?)|<)?|[+\-]@?|\*\*?|===?|[/`%&!^|~]|(\w+[=?!]?)|(\[\]=?)))|([A-Z]\w*|^|\h)\K::(?=[A-Z]) 0:variable
2014-07-03 15:16:59 +02:00
# Commands
# ‾‾‾‾‾‾‾‾
2018-05-06 23:29:52 +02:00
define-command ruby-alternative-file -docstring 'Jump to the alternate file (implementation ↔ test)' %{ evaluate-commands %sh{
2016-03-16 15:05:13 +01:00
case $kak_buffile in
*spec/*_spec.rb)
altfile=$(eval echo $(echo $kak_buffile | sed s+spec/+'*'/+';'s/_spec//))
2019-10-08 18:06:54 +02:00
[ ! -f $altfile ] && echo "fail 'implementation file not found'" && exit
2016-03-16 15:05:13 +01:00
;;
2019-05-25 11:39:52 +02:00
*test/*_test.rb)
altfile=$(eval echo $(echo $kak_buffile | sed s+test/+'*'/+';'s/_test//))
2019-10-08 18:06:54 +02:00
[ ! -f $altfile ] && echo "fail 'implementation file not found'" && exit
2019-05-25 11:39:52 +02:00
;;
2016-03-16 15:05:13 +01:00
*.rb)
path=$kak_buffile
dirs=$(while [ $path ]; do echo $path; path=${path%/*}; done | tail -n +2)
for dir in $dirs; do
2019-05-25 11:39:52 +02:00
altdir=$dir/spec && suffix=spec
[ ! -d $altdir ] && altdir=$dir/test && suffix=test
2016-03-16 15:05:13 +01:00
if [ -d $altdir ]; then
2019-05-25 11:39:52 +02:00
altfile=$altdir/$(realpath $kak_buffile --relative-to $dir | sed s+[^/]'*'/++';'s/.rb$/_${suffix}.rb/)
2016-03-16 15:05:13 +01:00
break
fi
done
2019-10-08 18:06:54 +02:00
[ ! -d $altdir ] && echo "fail 'spec/ and test/ not found'" && exit
2016-03-16 15:05:13 +01:00
;;
*)
2019-10-08 18:06:54 +02:00
echo "fail 'alternative file not found'" && exit
2016-03-16 15:05:13 +01:00
;;
esac
echo "edit $altfile"
}}
2018-12-19 10:10:26 +01:00
define-command -hidden ruby-trim-indent %{
2017-11-03 09:09:45 +01:00
evaluate-commands -no-hooks -draft -itersel %{
execute-keys <a-x>
2016-03-10 14:48:12 +01:00
# remove trailing white spaces
2017-11-03 09:09:45 +01:00
try %{ execute-keys -draft s \h + $ <ret> d }
2016-03-10 14:48:12 +01:00
}
}
2017-11-03 08:34:41 +01:00
define-command -hidden ruby-indent-on-char %{
2017-11-03 09:09:45 +01:00
evaluate-commands -no-hooks -draft -itersel %{
2016-01-21 19:19:17 +01:00
# align middle and end structures to start
2020-08-01 18:52:26 +02:00
try %{ execute-keys -draft <a-x> <a-k> ^ \h * (else|elsif) $ <ret> <a-a> i <a-semicolon> <a-?> ^ \h * (if) <ret> <a-S> 1<a-&> }
try %{ execute-keys -draft <a-x> <a-k> ^ \h * (when) $ <ret> <a-a> i <a-semicolon> <a-?> ^ \h * (case) <ret> <a-S> 1<a-&> }
try %{ execute-keys -draft <a-x> <a-k> ^ \h * (rescue) $ <ret> <a-a> i <a-semicolon> <a-?> ^ \h * (begin|def) <ret> <a-S> 1<a-&> }
try %{ execute-keys -draft <a-x> <a-k> ^ \h * (end) $ <ret> <a-a> i <a-semicolon> <a-?> ^ \h * (begin|case|class|def|for|if|module|unless|until|while) <ret> <a-S> 1<a-&> }
2016-01-21 19:19:17 +01:00
}
}
2017-11-03 08:34:41 +01:00
define-command -hidden ruby-indent-on-new-line %{
2017-11-03 09:09:45 +01:00
evaluate-commands -no-hooks -draft -itersel %{
2014-07-03 15:16:59 +02:00
# preserve previous line indent
2017-11-03 09:09:45 +01:00
try %{ execute-keys -draft K <a-&> }
2016-03-10 14:48:12 +01:00
# filter previous line
2018-12-19 10:10:26 +01:00
try %{ execute-keys -draft k : ruby-trim-indent <ret> }
2016-01-21 19:19:17 +01:00
# indent after start structure
2020-03-23 08:20:02 +01:00
try %{ execute-keys -draft k <a-x> <a-k> ^ \h * (begin|case|class|def|else|elsif|ensure|for|if|module|rescue|unless|until|when|while|.+\bdo$|.+\bdo\h\|.+(?=\|)) [^0-9A-Za-z_!?] <ret> j <a-gt> }
2016-03-10 14:12:08 +01:00
}
}
2018-04-14 04:47:42 +02:00
define-command -hidden ruby-insert-on-new-line %[
evaluate-commands -no-hooks -draft -itersel %[
2016-03-10 14:12:08 +01:00
# copy _#_ comment prefix and following white spaces
2019-10-22 11:02:06 +02:00
try %{ execute-keys -draft k <a-x> s '^\h*\K#\h*' <ret> y j <a-x><semicolon> P }
2016-01-21 19:19:17 +01:00
# wisely add end structure
2018-04-14 04:47:42 +02:00
evaluate-commands -save-regs x %[
2017-11-03 09:09:45 +01:00
try %{ execute-keys -draft k <a-x> s ^ \h + <ret> \" x y } catch %{ reg x '' }
2018-04-14 04:47:42 +02:00
try %[
evaluate-commands -draft %[
2019-06-07 23:25:33 +02:00
# Check if previous line opens a block
2020-03-23 08:20:02 +01:00
execute-keys -draft k<a-x> <a-k>^<c-r>x(begin|case|class|def|for|if|module|unless|until|while|.+\bdo$|.+\bdo\h\|.+(?=\|))[^0-9A-Za-z_!?]<ret>
2019-06-07 23:25:33 +02:00
# Check that we do not already have an end for this indent level which is first set via `ruby-indent-on-new-line` hook
2020-08-01 19:54:57 +02:00
execute-keys -draft }i J <a-x> <a-K> ^<c-r>x(end|else|elsif|rescue|when)[^0-9A-Za-z_!?]<ret>
2018-04-14 04:47:42 +02:00
]
execute-keys -draft o<c-r>xend<esc> # insert a new line with containing end
]
]
]
]
2014-07-03 15:16:59 +02:00
2019-03-13 06:24:33 +01:00
]