Bash's prompt control features are rather static. If you want more control, you can include variables in your prompt; make sure you haven't turned off the promptvars
option.
PS1='${PWD#"${PWD%/*/*}/"} \$ '
Note the single quotes: the variable expansions must happen at the time the prompt is displayed, not at the time the PS1
variable is defined.
If you want more control over what is displayed, you can use command substitutions. For example, the snippet above loses the ~
abbreviation for the home directory.
PS1='$(case $PWD in
$HOME) HPWD="~";;
$HOME/*/*) HPWD="${PWD#"${PWD%/*/*}/"}";;
$HOME/*) HPWD="~/${PWD##*/}";;
/*/*/*) HPWD="${PWD#"${PWD%/*/*}/"}";;
*) HPWD="$PWD";;
esac; printf %s "$HPWD") \$ '
This code is rather cumbersome, so instead of sticking it into the PS1
variable, you can use the PROMPT_COMMAND
variable to run code to set HPWD
and then use that in your prompt.
PROMPT_COMMAND='case $PWD in
$HOME) HPWD="~";;
$HOME/*/*) HPWD="${PWD#"${PWD%/*/*}/"}";;
$HOME/*) HPWD="~/${PWD##*/}";;
/*/*/*) HPWD="${PWD#"${PWD%/*/*}/"}";;
*) HPWD="$PWD";;
esac'
PS1='$HPWD \$'
Since the shortened prompt only changed on a directory change, you don't need to recalculate it each time a prompt is displayed. Bash doesn't provide a hook that runs on a current directory change, but you can simulate it by overriding cd
and its cousins.
cd () { builtin cd "$@" && chpwd; }
pushd () { builtin pushd "$@" && chpwd; }
popd () { builtin popd "$@" && chpwd; }
chpwd () {
case $PWD in
$HOME) HPWD="~";;
$HOME/*/*) HPWD="${PWD#"${PWD%/*/*}/"}";;
$HOME/*) HPWD="~/${PWD##*/}";;
/*/*/*) HPWD="${PWD#"${PWD%/*/*}/"}";;
*) HPWD="$PWD";;
esac
}
PS1='$HPWD \$'
Note that you don't need to, and should not, export PS1
, since it's a shell setting, not an environment variable. A bash PS1
setting wouldn't be understood by other shells.
P.S. If you want a nice interactive shell experience, switch to zsh, where all of these (prompt %
expansions largely encompassing tcsh's, chpwd
, etc.) are native features.
PS1='%2~ %# '
This seems to work for me to tell whether the timeout is reached:
sudo-expired() [[ $(
trap "" XFSZ
limit filesize 0
LC_ALL=C sudo -n true 2>&1) = *"password is required" ]]
That is, set the filesize limit to 0 for the update of the timestamp file to fail, but as long as we don't let that limit kill sudo
, we're still able to tell if a password was required or not.
Used like in yours (with a few simplifications):
sudo-warning()
if sudo-expired; then
echo '%F{blue}?%f'
else
echo '%F{red}?%f'
fi
TMOUT=10
TRAPALRM() zle reset-prompt
set -o promptsubst
PS1='%n$(sudo-warning) '
Best Answer
You can execute arbitrary code to display the prompt if you set the
prompt_subst
option. You don't need to look up the.git
directory every time a prompt is displayed: in practice, it's enough to update a variable on every current directory change, in thechpwd
hook, and use that variable in your prompt.