summaryrefslogtreecommitdiffstats
path: root/Completion/Unix
diff options
context:
space:
mode:
authorOliver Kiddle <opk@users.sourceforge.net>2003-02-28 13:50:44 +0000
committerOliver Kiddle <opk@users.sourceforge.net>2003-02-28 13:50:44 +0000
commit79d0de7c3ad8626d507b50176a80cf7ecb6f3996 (patch)
tree7009741bcfcf45c42ddea3be5a48807bc10f2ea9 /Completion/Unix
parent18303: fix some always-true test evaluations. (diff)
downloadzsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.tar
zsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.tar.gz
zsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.tar.bz2
zsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.tar.lz
zsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.tar.xz
zsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.tar.zst
zsh-79d0de7c3ad8626d507b50176a80cf7ecb6f3996.zip
merge changes from 4.1
Diffstat (limited to 'Completion/Unix')
-rw-r--r--Completion/Unix/Command/_tar171
-rw-r--r--Completion/Unix/Type/_tar_archive26
2 files changed, 197 insertions, 0 deletions
diff --git a/Completion/Unix/Command/_tar b/Completion/Unix/Command/_tar
new file mode 100644
index 000000000..3e3b3fc51
--- /dev/null
+++ b/Completion/Unix/Command/_tar
@@ -0,0 +1,171 @@
+#compdef tar gtar star
+
+# Tar completion. Features:
+# - Tries to collect tar commands from second position, single letter
+# option, and long options.
+# - `tar' can be called anything, will use the correct name
+# - Uses the function `_tar_archive' to complete archive files.
+# - Tries to find out if compressed archives should be used.
+# - Completes files inside archive. This is supposed to look pretty
+# much as if the files are in an ordinary directory hierarchy.
+# Handles extraction from compressed archives (GNU tar).
+# - Anywhere -- appears, gets a list of long options to complete from
+# tar itself (GNU tar)
+# - Things like --directory=... are also completed correctly.
+
+local _tar_cmd tf tmp tmpb del index
+
+# First we collect in `_tar_cmd' single letter options describing what
+# should be done with the archive and if it is compressed. This
+# collected from options arguments that start with only one hyphen,
+# from some of the possible long options, and from the second word if
+# that does not start with a hyphen.
+
+tmp=("${(@M)words:#-[^-]*}")
+_tar_cmd="${(j::)tmp#-}"
+
+(( $words[(I)--(un|)gzip] )) && _tar_cmd="z$_tar_cmd"
+(( $words[(I)--(un|)compress] )) && _tar_cmd="Z$_tar_cmd"
+(( $words[(I)--list] )) && _tar_cmd="t$_tar_cmd"
+(( $words[(I)--(extract|get)] )) && _tar_cmd="x$_tar_cmd"
+(( $words[(I)--create] )) && _tar_cmd="c$_tar_cmd"
+
+# Other ways of finding out what we're doing: first
+# look in the first argument if it's not an option
+if [[ "$words[2]" = *[txcdruA]*~-* ]]; then
+ _tar_cmd="$words[2]$_tar_cmd"
+elif [[ $_tar_cmd != *[txcdruA]* && CURRENT -gt 2 ]]; then
+ # look for more obscure long options: these aren't all handled.
+ (( $words[(I)--(diff|compare)] )) && _tar_cmd="d$_tar_cmd"
+ (( $words[(I)--append] )) && _tar_cmd="r$_tar_cmd"
+ (( $words[(I)--update] )) && _tar_cmd="u$_tar_cmd"
+ (( $words[(I)--(con|)catenate] )) && _tar_cmd="A$_tar_cmd"
+ (( $words[(I)--delete] )) && del=1
+fi
+
+# Next, we try to find the archive name and store it in `tf'. The name
+# is searched after a `--file=' long option, in the third word if the
+# second one didn't start with a hyphen but contained a `f', and after
+# an option argument starting with only one hyphen and containing a `f'.
+# unless that option argument also contains a `C'.
+
+tmp="$words[(I)--file=*]"
+tmpb="$words[(I)-*Cf*~--*]"
+
+if (( tmp )); then
+ tf=${~words[tmp][8,-1]}
+ _tar_cmd="f$_tar_cmd"
+elif [[ "$words[2]" != -* && "$words[2]" = *f* ]]; then
+ tf=${~words[3]}
+ _tar_cmd="f$_tar_cmd"
+elif (( tmpb )); then
+ tf=${~words[tmpb+2]}
+ wdir=${~words[tmpb+1]}
+ _tar_cmd="Cf$_tar_cmd"
+else
+ tmp="${words[(I)-*f*~--*]}"
+ if (( tmp )); then
+ tf=${~words[tmp+1]}
+ _tar_cmd="f$_tar_cmd"
+ fi
+fi
+
+# See if we should use a path prefix. We have to use eval as the dir can
+# be any unevaluated thing which appears on the command line, including a
+# parameter.
+
+# This isn't used right now.
+
+tmp=${words[(r)--dir[a-z]#=*]}
+
+if [[ -n $tmp ]]; then
+ eval "wdir=(${tmp#*=})"
+fi
+
+# Now we complete...
+
+if [[ "$PREFIX" = --* ]]; then
+
+ # ...long options after `--'.
+
+ _arguments -- '--owner*:user:_users' \
+ '*=(PROG|COMMAND)*:program:_command_names -e' \
+ '*=ARCHIVE*:archive: _tar_archive' \
+ '*=NAME*:file:_files' \
+ '*=DIR*:directory:_files -/' \
+ '*=CONTROL*::version control:(t numbered nil existing never simple)'
+
+elif [[ ( CURRENT -gt 2 && "$words[CURRENT-1]" = -[^C]#f* &&
+ "$words[CURRENT-1]" != --* ) ||
+ ( CURRENT -eq 3 && "$words[2]" = [^C]#f* && "$words[2]" != -* ) ||
+ ( CURRENT -gt 2 && "$words[CURRENT-2]" = -*C*f* &&
+ "$words[CURRENT-2]" != --* && "$words[CURRENT-1]" != --* ) ||
+ ( CURRENT -eq 4 && "$words[2]" = *C*f* && "$words[2]" != -* ) ]]; then
+
+ # ...archive files if we think they are wanted here.
+
+ _tar_archive
+
+elif [[ ( CURRENT -gt 2 && "$words[CURRENT-1]" = -[^f]#C*) ||
+ ( CURRENT -eq 3 && "$words[2]" = [^f]#C* ) ]]; then
+
+ # a directory for -C
+
+ _directories
+
+elif [[ ( "$_tar_cmd" = *[xt]* || -n $del ) && -n "$tf" ]]; then
+
+ # ...and files from the archive if we found an archive name and tar
+ # commands. We run `tar t...' on the file, keeping the list of
+ # filenames cached, plus the name of the tarfile so we know if it
+ # changes. We skip this test if the alleged archive is not a file.
+
+ local largs=-tf expl
+
+ if [[ $_tar_cmd = *z* ]]; then
+ largs=-tzf
+ elif [[ $_tar_cmd = *j* ]]; then
+ largs=-tjf
+ elif [[ $_tar_cmd = *y* ]]; then
+ largs=-tyf
+ elif [[ $_tar_cmd = *Z* ]]; then
+ largs=-tZf
+ elif [[ $_tar_cmd = *I* ]]; then
+ largs=-tIf
+ else
+ # Some random compression program
+ tmp="${words[(r)--use-comp*]}"
+ [[ -n $tmp ]] && largs=($tmp -tf)
+ fi
+
+ if [[ $tf != $_tar_cache_name && -f $tf ]]; then
+ _tar_cache_list=("${(@f)$($words[1] $largs $tf)}")
+ _tar_cache_name=$tf
+ fi
+
+ _wanted files expl 'file from archive' _multi_parts / _tar_cache_list
+elif (( CURRENT == 2 )); then
+ _values -s '' 'tar function' \
+ '(c t u x)A[append to an archive]' \
+ '(A t u x)c[create a new archive]' \
+ '(A c u x)t[list archive contents]' \
+ '(A c t x)u[update archive]' \
+ '(A c t u)x[extract files from an archive]' \
+ 'v[verbose output]' \
+ 'f[specify archive file or device]'
+else
+ if ! (( index=$words[(I)-*C*] )); then
+ if [[ $words[2] = [^f]#C* ]]; then
+ index=1
+ elif [[ $words[2] = *f*C* ]]; then
+ index=2
+ fi
+ fi
+ if (( index )); then
+ index="$~words[index+1]"
+ [[ $index = (.|..|)/* ]] || index=~+/$index
+ _files -W $index
+ else
+ _files
+ fi
+fi
diff --git a/Completion/Unix/Type/_tar_archive b/Completion/Unix/Type/_tar_archive
new file mode 100644
index 000000000..c5ab0e9fc
--- /dev/null
+++ b/Completion/Unix/Type/_tar_archive
@@ -0,0 +1,26 @@
+#autoload
+
+# This is used to generate filenames usable as a tar archive. This may
+# get one argument, a collection of tar option characters that may be
+# used to find out what kind of filename is needed. If no argument is
+# given but the parameter `_tar_cmd' is set, that is used.
+# If your version of `tar' supports this you may want to complete
+# things like `host:file' or `user@host:file' here.
+
+local expl
+
+[[ $# -eq 0 && $+_tar_cmd -ne 0 ]] && set "$_tar_cmd"
+
+_description files expl 'archive file'
+
+if [[ "$1" = *[urtx]* ]]; then
+ if [[ "$1" = *[zZ]* ]]; then
+ _files "$expl[@]" -g '*.((tar|TAR).(gz|GZ|Z)|tgz)'
+ elif [[ "$1" = *[Ijy]* ]]; then
+ _files "$expl[@]" -g '*.(tar|TAR).bz2'
+ else
+ _files "$expl[@]" -g '*.(tar|TAR)'
+ fi
+else
+ _files "$expl[@]"
+fi