有没有办法以独特的方式折叠zsh提示符中的当前工作目录,以便我可以将其复制并粘贴到另一个终端,点击TAB并获取原始路径?
假设我们有以下目录:
/adam/devl
/alice/devl
/alice/docs
/bob/docs
如果提示被编程为显示第一个字符,而我在/ b / d,则它是唯一的。另一方面,/ a / d不是唯一的,所以我需要/ ad / d,/ al / de和/ al / do。甚至/ ali / ...一旦用户alex出现。
是否有可能直接在zsh中破解它,或者我是否需要编写一个脚本,找到每个父目录的最短唯一开头?
感谢您的想法!
答案 0 :(得分:8)
我不知道zsh
有这种类型的内置函数,但是在不使用单个子shell或慢速管道的情况下编写脚本应该非常简单:
#!/bin/zsh
paths=(${(s:/:)PWD})
cur_path='/'
cur_short_path='/'
for directory in ${paths[@]}
do
cur_dir=''
for (( i=0; i<${#directory}; i++ )); do
cur_dir+="${directory:$i:1}"
matching=("$cur_path"/"$cur_dir"*/)
if [[ ${#matching[@]} -eq 1 ]]; then
break
fi
done
cur_short_path+="$cur_dir/"
cur_path+="$directory/"
done
printf %q "${cur_short_path: : -1}"
echo
此脚本将输出自动完成所需的最短路径。
您可以将其作为函数放入.zshrc
中,然后从任何目录运行它。
function spwd {
paths=(${(s:/:)PWD})
cur_path='/'
cur_short_path='/'
for directory in ${paths[@]}
do
cur_dir=''
for (( i=0; i<${#directory}; i++ )); do
cur_dir+="${directory:$i:1}"
matching=("$cur_path"/"$cur_dir"*/)
if [[ ${#matching[@]} -eq 1 ]]; then
break
fi
done
cur_short_path+="$cur_dir/"
cur_path+="$directory/"
done
printf %q "${cur_short_path: : -1}"
echo
}
以下是该视频的实际操作:
https://asciinema.org/a/0TyL8foqvQ8ec5ZHS3c1mn5LH
或者,如果您愿意,可以使用一些示例输出:
~/t $ ls
adam alice bob getshortcwd.zsh
~/t $ ls adam
devl
~/t $ ls alice
devl docs
~/t $ spwd
/h/v/t
~/t $ cd adam/devl
~/t/adam/devl $ spwd
/h/v/t/ad/d
~/t/adam/devl $ cd ../../alice/devl
~/t/alice/devl $ spwd
/h/v/t/al/de
~/t/alice/devl $ cd ../docs
~/t/alice/docs $ spwd
/h/v/t/al/do
~/t/alice/docs $ `spwd` [TAB]
~/t/alice/docs $ /h/v/t/al/do [TAB]
~/t/alice/docs $ /home/vsimonian/t/alice/docs
答案 1 :(得分:0)
是的,可以将目录折叠到第一个唯一字母路径,并在按[Tab]时让Z Shell扩展该路径。我只是使用compinstall(与Zsh一起安装的zsh实用程序脚本)来生成以下代码。注意扩展路径元素的重要部分是第六zstyle
命令,靠近结尾,其中用于分隔完成点的字符括号包括/
,当然是目录分隔符。这样,您建议的唯一路径将仅使用一个[Tab]按下完全填写,就像*
位于每个路径名称唯一字母的末尾一样。
# The following lines were added by compinstall
zstyle ':completion:*' add-space true
zstyle ':completion:*' completer _list _expand _complete _ignored _match _correct _approximate _prefix
zstyle ':completion:*' completions 1
zstyle ':completion:*' list-colors ${(s.:.)LS_COLORS}
zstyle ':completion:*' matcher-list 'm:{[:lower:]}={[:upper:]} r:|[._-]=* r:|=*' 'm:{[:lower:]}={[:upper:]} m:{[:lower:][:upper:]}={[:upper:][:lower:]} r:|[._-]=* r:|=*' 'r:|[._-/]=* r:|=*' 'l:|=* r:|=*'
zstyle ':completion:*' match-original both
zstyle :compinstall filename '/home/micah/.zsh/.zshrc'
autoload -Uz compinit
compinit
# End of lines added by compinstall
至于首先创建唯一路径并将其插入到提示符中,可以使用zsh脚本或函数,因此也可以用于完成者或行编辑器,但只是坚持提示,您可以向$precmd_functions
数组添加一个修改或添加到PS1
变量的函数。此特殊数组是在每个提示之前运行的函数名列表。
function precmd_unique_pwd {
local pwd_string="$(upwd)"
PS1="%B%n@%m $pwd_string => %b"
}
precmd_functions+=( precmd_unique_pwd )
为了缩短当前的PWD格式,我认为这个功能很清晰,易于理解,但不一定针对低资源使用进行优化。
#!/bin/zsh
function upwd {
emulate -LR zsh -o nullglob
local dir Path
local -a newpwd tmp stack
local -i length=1 Flag=0
newpwd=( ${(s./.)PWD} )
foreach dir ( $newpwd )
(( length=0, Flag=0 ))
repeat $#dir
do
(( length += 1 ))
tmp=( ${(j.*/.)~stack}/$dir[1,$length]*(/) )
if
(( $#tmp == 1 ))
then
Path=$Path/$dir[1,$length]
stack+=( /$dir )
(( Flag=1 ))
break
fi
done
if
(( Flag ))
then
continue
else
Path=$Path/$dir
fi
end
print -- $Path
}
upwd
请注意,由于在globbing结束时使用Zsh功能(/)
,它会找到包含目录名称的唯一路径。在最后一个目录(当前)上,这意味着如果存在具有相同名称和扩展名的文件,您可以匹配其他内容。