213 lines
9.8 KiB
Bash
213 lines
9.8 KiB
Bash
#compdef ack ack2 ack-grep ack-standalone
|
|
# ------------------------------------------------------------------------------
|
|
# Description
|
|
# -----------
|
|
#
|
|
# Completion script for ack 1.96 and 2.14 (http://betterthangrep.com).
|
|
#
|
|
# ------------------------------------------------------------------------------
|
|
# Authors
|
|
# -------
|
|
#
|
|
# * Julien Nicoulaud <julien.nicoulaud@gmail.com> (version 1.94)
|
|
# * Zhao Cai <caizhaoff@gmail.com> (version 2.04)
|
|
#
|
|
# ------------------------------------------------------------------------------
|
|
|
|
_ack_version() {
|
|
local version
|
|
version=(${(f)"$(_call_program version $words[1] --version)"})
|
|
version=${${(z)${version[1]}}[2]}
|
|
echo $version
|
|
}
|
|
|
|
_ack() {
|
|
local context curcontext="$curcontext" state line cmds update_policy ret=1
|
|
integer NORMARG
|
|
typeset -A opt_args
|
|
|
|
# Don't complete if command doesn't exist
|
|
[[ ${+commands[${words[1]}]} -eq 0 ]] && return 0
|
|
|
|
zstyle -s ":completion:${curcontext}:" cache-policy update_policy
|
|
[[ -z "$update_policy" ]] && zstyle ":completion:${curcontext}:" cache-policy _ack_types_caching_policy
|
|
|
|
unset _ack_raw_types
|
|
if ( [[ ${+_ack_raw_types} -eq 0 ]] || _cache_invalid "ack-grep" ) && ! _retrieve_cache "ack-grep"; then
|
|
_ack_raw_types=(${(S)${(S)${(f)${${"$(_call_program types $words[1] --help=types)"}#*--\[no\]}}%; first line matches \/*\/}#*no\]})
|
|
[[ $#_ack_raw_types -gt 0 ]] && _store_cache "ack-grep" _ack_raw_types
|
|
fi
|
|
|
|
ack_20_options=(
|
|
'--ackrc[specify an ackrc file to use]:files:_files'
|
|
'(- 1 *)--bar[consult Admiral Ackbar]'
|
|
'(--nobreak --break)'{--nobreak,--break}'[print a break between results from different files, default on]'
|
|
'(- 1 *)--cathy[chocolate chocolate chocolate]'
|
|
'(- 1 *)--create-ackrc[create custom ackrc files based on the default settings loaded by ackrc]'
|
|
'(- 1 *)--dump[writes the list of options loaded and where they came from to standard output]'
|
|
'(--files-from -x)--files-from=[read the list of files to search from FILE]:files:_files'
|
|
'(--filter --nofilter)--filter[force ack to treat input as pipe]'
|
|
'(--filter --nofilter)--nofilter[force ack to treat input as tty]'
|
|
'(--noheading --heading)'{--noheading,--heading}'[print a filename heading above results, default on]'
|
|
'(- 1 *)--help-types[display all known types]'
|
|
'--ignore-ack-defaults[ignore default definitions included with ack]'
|
|
'*--ignore-file=[ignore file]:ignore file filter: _describe "ignore file filter" ignore_filter_opts'
|
|
'(-k --known-types)'{-k,--known-types}'[include only files of types that ack recognizes]'
|
|
'--lines=[only print line(s) NUM of each file]:number'
|
|
'--nopager[do not send output through a pager, overrides ackrc, ACK_PAGER & ACK_PAGER_COLOR]'
|
|
'-s[suppress error messages about nonexistent or unreadable files]'
|
|
'(- 1 *)--thpppt[bill the cat]'
|
|
'*--type-del[remove all filters associated with TYPE]' \
|
|
'(-x --files-from)-x[read the list of files to search from STDIN]'
|
|
)
|
|
|
|
ack_19_options=(
|
|
'(-a --all -u --unrestricted)'{-a,--all}'[operate on all files, regardless of type (but still skip directories like blib, CVS, etc.)]'
|
|
'-G+[only paths matching the given regex are included in the search]:regex'
|
|
'--invert-file-match[print/search handle files that do not match -g/-G]'
|
|
'--line=[only print given line of each file]:number' \
|
|
'(-u --unrestricted -a --all)'{-u,--unrestricted}'[all files and directories (including blib/, core.*, ...) are searched, nothing is skipped]'
|
|
)
|
|
|
|
if (( $(_ack_version) > 2.0 )); then
|
|
ack_version_options=(${ack_20_options})
|
|
else
|
|
ack_version_options=(${ack_19_options})
|
|
fi
|
|
|
|
_arguments -C -s -S -n \
|
|
'(- 1 *)--version[display version and copyright information]' \
|
|
'(- 1 *)--help[print a short help statement]' \
|
|
'(- 1 *)--man[print the manual page]' \
|
|
$ack_version_options \
|
|
'(-A --after-context -C --context)'{-A+,--after-context=}'[print N lines of trailing context after matching lines]:number' \
|
|
'(-B --before-context -C --context)'{-B+,--before-context=}'[print N lines of leading context before matching lines]:number' \
|
|
'(-C --context -A --after-context -B --before-context)'{-C-,--context=}'[print N lines (default 2) of context around matching lines]:number' \
|
|
'(-c --count)'{-c,--count}'[suppress normal output; instead print a count of matching lines for each input file]' \
|
|
'(--nocolor)--color[highlight the matching text]' \
|
|
'(--color --color-filename --color-match --color-lineno)--nocolor[suppress the color]' \
|
|
'(--nocolor --color)--color-filename[sets the color to be used for filenames]:color:->colors' \
|
|
'(--nocolor --color)--color-match[sets the color to be used for matches]:color:->colors' \
|
|
'(--nocolor --color)--color-lineno[sets the color to be used for line numbers]:color:->colors' \
|
|
'--column[show the column number of the first match]' \
|
|
'(--noenv)--env[enable environment processing]' \
|
|
'(--env)--noenv[disable all environment processing, no .ackrc is read and all environment variables are ignored]' \
|
|
'--flush[flush output immediately]' \
|
|
'-f[only print the files that would be searched, without actually doing any searching]' \
|
|
'(--nofollow)--follow[follow symlinks]' \
|
|
'(--follow)--nofollow[don'\''t follow symlinks]' \
|
|
'-g+[print files where the relative path + filename matches the given regex]:regex' \
|
|
'(--nogroup)--group[group matches by file name]' \
|
|
'(--group)--nogroup[do not group matches by file name]' \
|
|
'(-H --with-filename -h --no-filename)'{-H,--with-filename}'[print the filename for each match]' \
|
|
'(-h --no-filename -H --with-filename)'{-h,--no-filename}'[suppress the prefixing of filenames on output when multiple files are searched]' \
|
|
'(-i --ignore-case)'{-i,--ignore-case}'[ignore case in the search strings]' \
|
|
'*--ignore-dir=[ignore directory]:directory:_files' \
|
|
'*--noignore-dir=[ignore directory]:directory:_files' \
|
|
'(-l --files-with-matches -L --files-without-matches)'{-l,--files-with-matches}'[only print the filenames of matching files, instead of the matching text]' \
|
|
'(-L --files-without-matches -l --files-with-matches)'{-L,--files-without-matches}'[only print the filenames of files that do NOT match]' \
|
|
'--match=[specify the regular expression explicitly]:regex' \
|
|
'(-m --max-count)'{-m+,--max-count=}'[stop reading a file after N matches]:number' \
|
|
'(-r -R --recurse -n --no-recurse)'{-r,-R,--recurse}'[recurse into sub-directories]' \
|
|
'(-n --no-recurse -r -R --recurse)'{-n,--no-recurse}'[no descending into subdirectories]' \
|
|
'-o[show only the part of each line matching PATTERN (turns off text highlighting)]:pattern' \
|
|
'--output=[output the evaluation of expr for each line (turns off text highlighting)]:expression' \
|
|
'--pager=[direct ack'\''s output through program]:pager program:_command_names' \
|
|
'--passthru[prints all lines, whether or not they match the expression]' \
|
|
'--print0[the filenames are output separated with a null byte instead of the usual newline]' \
|
|
'(-Q --literal)'{-Q,--literal}'[quote all metacharacters in the pattern, it is treated as a literal]' \
|
|
'(--no-smart-case)--smart-case[ignore case in the search strings if pattern contains no uppercase characters]' \
|
|
'(--smart-case)--no-smart-case[disable --smart-case option]' \
|
|
'--sort-files[sorts the found files lexically]' \
|
|
'--show-types[outputs the filetypes that ack associates with each file]' \
|
|
'--thpppt[display the all-important Bill The Cat logo]' \
|
|
'*--type=[specify the types of files to include or exclude from a search]:type:->types' \
|
|
'*--type-add[files with the given extensions are recognized as being of the given type]:type-def:->type-defs' \
|
|
'*--type-set[files with the given extensions are recognized as being of the given type]:type-def:->type-defs' \
|
|
'(-v --invert-match)'{-v,--invert-match}'[invert match: select non-matching lines]' \
|
|
'(-w --word-regexp)'{-w,--word-regexp}'[force the given pattern to match only whole words]' \
|
|
'-1[stops after reporting first match of any kind]' \
|
|
{'--','--no'}${_ack_raw_types/ ##/\[}']' \
|
|
'*: :->args' \
|
|
&& ret=0
|
|
|
|
case $state in
|
|
args)
|
|
if [[ CURRENT -eq NORMARG && ${+opt_args[--match]} -eq 0 ]]; then
|
|
# If the current argument is the first non-option argument
|
|
# and --match isn't present then a pattern is expected
|
|
_message -e patterns 'pattern' && ret=0
|
|
else
|
|
_files
|
|
fi
|
|
;;
|
|
colors)
|
|
local colors; colors=(
|
|
'black' 'on_black'
|
|
'red' 'on_red'
|
|
'green' 'on_green'
|
|
'yellow' 'on_yellow'
|
|
'blue' 'on_blue'
|
|
'magenta' 'on_magenta'
|
|
'cyan' 'on_cyan'
|
|
'white' 'on_white'
|
|
'clear'
|
|
'reset'
|
|
'dark'
|
|
'bold'
|
|
'underline'
|
|
'underscore'
|
|
'blink'
|
|
'reverse'
|
|
'concealed'
|
|
)
|
|
_describe -t 'colors' 'color' colors && ret=0
|
|
;;
|
|
type-defs)
|
|
if compset -P '*='; then
|
|
local extensions; extensions=(*.*(:e))
|
|
_values -s ',' 'file extension' '.'$extensions && ret=0
|
|
else
|
|
_message -e type-name 'type name' && ret=0
|
|
fi
|
|
;;
|
|
types)
|
|
local types; types=({'','no'}${_ack_raw_types/ ##/:})
|
|
_describe -t 'types' 'type' types
|
|
;;
|
|
esac
|
|
|
|
return ret
|
|
}
|
|
|
|
ignore_filter_opts=(
|
|
'is\::FILENAME'
|
|
'ext\::[EXTENSION,EXTENSION2,...]'
|
|
'match\::PATTERN'
|
|
'firstlinematch\::PATTERN'
|
|
)
|
|
|
|
_ack_types_caching_policy() {
|
|
|
|
# Rebuild if ackrc more recent than cache.
|
|
[[ -f ${ACKRC:-$HOME/.ackrc} && ${ACKRC:-$HOME/.ackrc} -nt "$1" ]] && return 0
|
|
|
|
# Rebuild if cache is older than one week.
|
|
local -a oldp
|
|
oldp=( "$1"(Nmw+1) )
|
|
(( $#oldp )) && return 0
|
|
|
|
return 1
|
|
}
|
|
|
|
_ack "$@"
|
|
|
|
# Local Variables:
|
|
# mode: Shell-Script
|
|
# sh-indentation: 2
|
|
# indent-tabs-mode: nil
|
|
# sh-basic-offset: 2
|
|
# End:
|
|
# vim: ft=zsh sw=2 ts=2 et
|
|
|