8fa9e85980
While in bash you can externally register custom completion functions using `complete` command, it was not possible to do so in zsh without changing completion.zsh as the name of the supported commands are hard-coded within the code (See #362). With this commit, fzf-completion of zsh will first look if `_fzf_COMMAND_completion` exists and calls the function, so one can externally define completion functions for specific commands. This commit also tries to make the interface of (yet undocumented) _fzf_list_completion helper function consistent across bash and zsh. So the following code works both on bash and zsh. _fzf_pass_completion() { local pwdir=${PASSWORD_STORE_DIR-~/.password-store/} local stringsize="${#pwdir}" let "stringsize+=1" _fzf_list_completion '+m' "$@" << "EOF" find "$pwdir" -name "*.gpg" -print | cut -c "$stringsize"- | sed -e 's/\(.*\)\.gpg/\1/' EOF } # Only on bash complete -F _fzf_pass_completion -o default -o bashdefault pass Note that the suggested convention and the interface are not yet final and subject to change. /cc @d4ndo
163 lines
4.6 KiB
Bash
163 lines
4.6 KiB
Bash
#!/bin/zsh
|
|
# ____ ____
|
|
# / __/___ / __/
|
|
# / /_/_ / / /_
|
|
# / __/ / /_/ __/
|
|
# /_/ /___/_/-completion.zsh
|
|
#
|
|
# - $FZF_TMUX (default: 1)
|
|
# - $FZF_TMUX_HEIGHT (default: '40%')
|
|
# - $FZF_COMPLETION_TRIGGER (default: '**')
|
|
# - $FZF_COMPLETION_OPTS (default: empty)
|
|
|
|
_fzf_path_completion() {
|
|
local base lbuf find_opts fzf_opts suffix tail fzf dir leftover matches nnm
|
|
base=${(Q)1}
|
|
lbuf=$2
|
|
find_opts=$3
|
|
fzf_opts=$4
|
|
suffix=$5
|
|
tail=$6
|
|
[ ${FZF_TMUX:-1} -eq 1 ] && fzf="fzf-tmux -d ${FZF_TMUX_HEIGHT:-40%}" || fzf="fzf"
|
|
|
|
if ! setopt | grep nonomatch > /dev/null; then
|
|
nnm=1
|
|
setopt nonomatch
|
|
fi
|
|
dir="$base"
|
|
while [ 1 ]; do
|
|
if [ -z "$dir" -o -d ${~dir} ]; then
|
|
leftover=${base/#"$dir"}
|
|
leftover=${leftover/#\/}
|
|
[ "$dir" = './' ] && dir=''
|
|
dir=${~dir}
|
|
matches=$(\find -L $dir* ${=find_opts} 2> /dev/null | ${=fzf} ${=FZF_COMPLETION_OPTS} ${=fzf_opts} -q "$leftover" | while read item; do
|
|
printf "%q$suffix " "$item"
|
|
done)
|
|
matches=${matches% }
|
|
if [ -n "$matches" ]; then
|
|
LBUFFER="$lbuf$matches$tail"
|
|
fi
|
|
zle redisplay
|
|
break
|
|
fi
|
|
dir=$(dirname "$dir")
|
|
dir=${dir%/}/
|
|
done
|
|
[ -n "$nnm" ] && unsetopt nonomatch
|
|
}
|
|
|
|
_fzf_all_completion() {
|
|
_fzf_path_completion "$1" "$2" \
|
|
"-name .git -prune -o -name .svn -prune -o -type d -print -o -type f -print -o -type l -print" \
|
|
"-m" "" " "
|
|
}
|
|
|
|
_fzf_dir_completion() {
|
|
_fzf_path_completion "$1" "$2" \
|
|
"-name .git -prune -o -name .svn -prune -o -type d -print" \
|
|
"" "/" ""
|
|
}
|
|
|
|
_fzf_list_completion() {
|
|
local fzf_opts lbuf src fzf matches
|
|
fzf_opts=$1
|
|
lbuf=$2
|
|
read -r src
|
|
[ ${FZF_TMUX:-1} -eq 1 ] && fzf="fzf-tmux -d ${FZF_TMUX_HEIGHT:-40%}" || fzf="fzf"
|
|
|
|
matches=$(eval "$src" | ${=fzf} ${=FZF_COMPLETION_OPTS} ${=fzf_opts} -q "$prefix")
|
|
if [ -n "$matches" ]; then
|
|
LBUFFER="$lbuf$matches "
|
|
fi
|
|
zle redisplay
|
|
}
|
|
|
|
_fzf_telnet_completion() {
|
|
_fzf_list_completion '+m' "$@" << "EOF"
|
|
\grep -v '^\s*\(#\|$\)' /etc/hosts | \grep -Fv '0.0.0.0' | awk '{if (length($2) > 0) {print $2}}' | sort -u
|
|
EOF
|
|
}
|
|
|
|
_fzf_ssh_completion() {
|
|
_fzf_list_completion '+m' "$@" << "EOF"
|
|
cat <(cat ~/.ssh/config /etc/ssh/ssh_config 2> /dev/null | \grep -i '^host' | \grep -v '*') <(\grep -v '^\s*\(#\|$\)' /etc/hosts | \grep -Fv '0.0.0.0') | awk '{if (length($2) > 0) {print $2}}' | sort -u
|
|
EOF
|
|
}
|
|
|
|
_fzf_export_completion() {
|
|
_fzf_list_completion '+m' "$@" << "EOF"
|
|
declare -xp | sed 's/=.*//' | sed 's/.* //'
|
|
EOF
|
|
}
|
|
|
|
_fzf_unset_completion() {
|
|
_fzf_list_completion '+m' "$@" << "EOF"
|
|
declare -xp | sed 's/=.*//' | sed 's/.* //'
|
|
EOF
|
|
}
|
|
|
|
_fzf_unalias_completion() {
|
|
_fzf_list_completion '+m' "$@" << "EOF"
|
|
alias | sed 's/=.*//'
|
|
EOF
|
|
}
|
|
|
|
fzf-completion() {
|
|
local tokens cmd prefix trigger tail fzf matches lbuf d_cmds sws
|
|
if setopt | grep shwordsplit > /dev/null; then
|
|
sws=1
|
|
unsetopt shwordsplit
|
|
fi
|
|
|
|
# http://zsh.sourceforge.net/FAQ/zshfaq03.html
|
|
# http://zsh.sourceforge.net/Doc/Release/Expansion.html#Parameter-Expansion-Flags
|
|
tokens=(${(z)LBUFFER})
|
|
if [ ${#tokens} -lt 1 ]; then
|
|
eval "zle ${fzf_default_completion:-expand-or-complete}"
|
|
return
|
|
fi
|
|
|
|
cmd=${tokens[1]}
|
|
|
|
# Explicitly allow for empty trigger.
|
|
trigger=${FZF_COMPLETION_TRIGGER-'**'}
|
|
[ -z "$trigger" -a ${LBUFFER[-1]} = ' ' ] && tokens+=("")
|
|
|
|
tail=${LBUFFER:$(( ${#LBUFFER} - ${#trigger} ))}
|
|
# Kill completion (do not require trigger sequence)
|
|
if [ $cmd = kill -a ${LBUFFER[-1]} = ' ' ]; then
|
|
[ ${FZF_TMUX:-1} -eq 1 ] && fzf="fzf-tmux -d ${FZF_TMUX_HEIGHT:-40%}" || fzf="fzf"
|
|
matches=$(ps -ef | sed 1d | ${=fzf} ${=FZF_COMPLETION_OPTS} -m | awk '{print $2}' | tr '\n' ' ')
|
|
if [ -n "$matches" ]; then
|
|
LBUFFER="$LBUFFER$matches"
|
|
fi
|
|
zle redisplay
|
|
# Trigger sequence given
|
|
elif [ ${#tokens} -gt 1 -a "$tail" = "$trigger" ]; then
|
|
d_cmds=(cd pushd rmdir)
|
|
|
|
[ -z "$trigger" ] && prefix=${tokens[-1]} || prefix=${tokens[-1]:0:-${#trigger}}
|
|
[ -z "${tokens[-1]}" ] && lbuf=$LBUFFER || lbuf=${LBUFFER:0:-${#tokens[-1]}}
|
|
|
|
if eval "type _fzf_${cmd}_completion > /dev/null"; then
|
|
eval "prefix=\"$prefix\" _fzf_${cmd}_completion \"$lbuf\""
|
|
elif [ ${d_cmds[(i)$cmd]} -le ${#d_cmds} ]; then
|
|
_fzf_dir_completion "$prefix" "$lbuf"
|
|
else
|
|
_fzf_all_completion "$prefix" "$lbuf"
|
|
fi
|
|
# Fall back to default completion
|
|
else
|
|
eval "zle ${fzf_default_completion:-expand-or-complete}"
|
|
fi
|
|
[ -n "$sws" ] && setopt shwordsplit
|
|
}
|
|
|
|
[ -z "$fzf_default_completion" ] &&
|
|
fzf_default_completion=$(bindkey '^I' | grep -v undefined-key | awk '{print $2}')
|
|
|
|
zle -N fzf-completion
|
|
bindkey '^I' fzf-completion
|
|
|