以独特的方式折叠zsh提示符中的目录

时间:2017-07-23 09:59:52

标签: zsh prompt pwd

有没有办法以独特的方式折叠zsh提示符中的当前工作目录,以便我可以将其复制并粘贴到另一个终端,点击TAB并获取原始路径?

假设我们有以下目录:

/adam/devl
/alice/devl
/alice/docs
/bob/docs

如果提示被编程为显示第一个字符,而我在/ b / d,则它是唯一的。另一方面,/ a / d不是唯一的,所以我需要/ ad / d,/ al / de和/ al / do。甚至/ ali / ...一旦用户alex出现。

是否有可能直接在zsh中破解它,或者我是否需要编写一个脚本,找到每个父目录的最短唯一开头?

感谢您的想法!

2 个答案:

答案 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功能(/),它会找到包含目录名称的唯一路径。在最后一个目录(当前)上,这意味着如果存在具有相同名称和扩展名的文件,您可以匹配其他内容。