Zsh Mailing List Archive
Messages sorted by:
Reverse Date,
Date,
Thread,
Author
Re: [PATCH] Add customizable `vcs` prompt theme (was Re: [RFC][PATCH] `newuser` prompt theme)
On Wed, Jun 9, 2021 at 4:22 PM Roman Perepelitsa
<roman.perepelitsa@xxxxxxxxx> wrote:
>
> A few questions/comments after a cursory look.
>
> Why is prompt_vcs_chpwd using exec + read instead of $(...)?
Because I apparently forgot that $(...) runs in a subshell, too. ^_^;
> Why hook zle-line-init instead of precmd? There is a comment saying
> that it's "so you can press Enter on an empty line to update VCS info"
> but I don't understand it. When I press Enter in zle with an empty
> buffer, precmd does get called.
I could've sworn that precmd doesn't get called under those
conditions, but I guess I confused it with preexec.
> There is a race that may cause incorrect git status to be displayed.
> For example, suppose I run these commands in quick succession in a
> large git repository:
>
> git checkout -b foo
> git checkout -b bar
>
> If the two background vcs processes finish out of order, prompt will
> incorrectly display "foo" when the current branch is "bar". This error
> will persist until another command (perhaps empty) is executed.
>
> Pressing and holding Enter in a large repository may cause large (and
> unbounded) strain on the machine.
Thanks for the review! Here's a new version of the patch that
hopefully addresses the concerns you raised.
From 41ffe718d7d58411254dc35f9618ae4fb0906aa6 Mon Sep 17 00:00:00 2001
From: Marlon Richert <marlon.richert@xxxxxxxxx>
Date: Wed, 9 Jun 2021 15:53:47 +0300
Subject: [PATCH] Add customizable `vcs` prompt theme
---
Functions/Prompts/prompt_vcs_setup | 180 +++++++++++++++++++++++++++++
1 file changed, 180 insertions(+)
create mode 100644 Functions/Prompts/prompt_vcs_setup
diff --git a/Functions/Prompts/prompt_vcs_setup b/Functions/Prompts/prompt_vcs_setup
new file mode 100644
index 000000000..8adc7c04d
--- /dev/null
+++ b/Functions/Prompts/prompt_vcs_setup
@@ -0,0 +1,180 @@
+##
+# Prompt that can be customized through vcs_info
+#
+
+autoload -Uz add-zsh-hook vcs_info
+
+# Prompt segments
+readonly -gHA _prompt_vcs_ps1=(
+ chpwd $'\n%B%F{blue}%~%%b%f/'
+ left $'%%(?,%F{green},%F{red})%#%%b%f%%s '
+ right '%B%F{blue}%n%b%f%k@%F{magenta}%m%f'
+ staged '%B%F{green}+%b%f'
+ unstaged '%B%F{red}*%b%f'
+ action '%B%F{red}%a%%b%f'
+ branch '%B%F{cyan}%b%%b%f'
+ repo '|%B%F{blue}%r%%b%f'
+)
+
+prompt_vcs_help() {
+ print -r -- \
+"This prompt theme can by customized by copy-pasting any of the code below to
+your .zshrc file and editing it there:
+
+ # For each of the following three entries:
+ # 1st string is printed after changing dirs.
+ # 2nd string is the left prompt.
+ # 3rd string is the right prompt and updated asynchronously.
+
+ # Normal prompt:
+ zstyle ':vcs_info:*:prompt_vcs:*' nvcsformats \\
+ "${(q+)_prompt_vcs_ps1[chpwd]//'%%'/%}" \\
+ "${(q+)_prompt_vcs_ps1[left]//'%%'/%}" \\
+ "${(q+)_prompt_vcs_ps1[right]//'%%'/%}"
+
+ # Prompt inside VCS repo:
+ zstyle ':vcs_info:*:prompt_vcs:*' formats \\
+ ${(q+)_prompt_vcs_ps1[chpwd]} \\
+ ${(q+)_prompt_vcs_ps1[left]} \\
+ ${(q+):-%u%c$_prompt_vcs_ps1[branch]$_prompt_vcs_ps1[repo]}
+
+ # Prompt during an ongoing VCS action:
+ zstyle ':vcs_info:*:prompt_vcs:*' actionformats \\
+ ${(q+)_prompt_vcs_ps1[chpwd]} \\
+ ${(q+)_prompt_vcs_ps1[left]} \\
+ ${(q+):-%u%c$_prompt_vcs_ps1[action]$_prompt_vcs_ps1[repo]}
+
+ # These set the values of %c and %u, respectively:
+ zstyle ':vcs_info:*:prompt_vcs:*' stagedstr ${(q+)_prompt_vcs_ps1[staged]}
+ zstyle ':vcs_info:*:prompt_vcs:*' unstagedstr ${(q+)_prompt_vcs_ps1[unstaged]}
+
+For more info on the settings above, see
+http://zsh.sourceforge.net/Doc/Release/Prompt-Expansion.html and the end of
+http://zsh.sourceforge.net/Doc/Release/User-Contributions.html#vcs_005finfo-Configuration"
+}
+
+# Sets a style if it hasn't been set yet.
+_prompt_vcs_zstyle() {
+ local -a val
+ zstyle -g val "$1" "$2"
+ (( $#val )) ||
+ zstyle "$@"
+}
+
+_prompt_vcs_info() {
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' max-exports 3 # Default is 2.
+
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' nvcsformats \
+ "${_prompt_vcs_ps1[chpwd]//'%%'/%}" \
+ "${_prompt_vcs_ps1[left]//'%%'/%}" \
+ "${_prompt_vcs_ps1[right]//'%%'/%}"
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' formats \
+ "$_prompt_vcs_ps1[chpwd]" \
+ "$_prompt_vcs_ps1[left]" \
+ "%u%c$_prompt_vcs_ps1[branch]$_prompt_vcs_ps1[repo]"
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' actionformats \
+ "$_prompt_vcs_ps1[chpwd]" \
+ "$_prompt_vcs_ps1[left]" \
+ "%u%c$_prompt_vcs_ps1[action]$_prompt_vcs_ps1[repo]"
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' stagedstr \
+ "$_prompt_vcs_ps1[staged]"
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' unstagedstr \
+ "$_prompt_vcs_ps1[unstaged]"
+
+ vcs_info prompt_vcs # Call with namespace.
+
+ print -rNC1 -- \
+ "${(q+)vcs_info_msg_0_}" "${(q+)vcs_info_msg_1_}" "${(q+)vcs_info_msg_2_}"
+}
+
+prompt_vcs_chpwd() {
+ emulate -L zsh
+
+ # Don't check for changes at this point, for performance reasons.
+ typeset -ga _prompt_vcs_info_msg_=( "${(0@Q)$(
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' check-for-changes no
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' check-for-staged-changes no
+ _prompt_vcs_info
+ )}" )
+ shift -p _prompt_vcs_info_msg_
+ print -Pr -- "$_prompt_vcs_info_msg_[1]"
+ PS1="$_prompt_vcs_info_msg_[2]"
+ RPS1="$_prompt_vcs_info_msg_[3]"
+}
+
+prompt_vcs_precmd() {
+ emulate -L zsh
+
+ # Kill any unfinished process before starting a new one.
+ if [[ -n $prompt_vcs_fd ]] && { : <&$prompt_vcs_fd } 2>/dev/null; then
+ zle -F "$prompt_vcs_fd" 2> /dev/null
+ exec {prompt_vcs_fd}<&-
+ fi
+
+ # Asynchronously check for changes.
+ typeset -gH prompt_vcs_fd=
+ exec {prompt_vcs_fd}< <(
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' check-for-changes yes
+ _prompt_vcs_zstyle ':vcs_info:*:prompt_vcs:*' check-for-staged-changes yes
+ _prompt_vcs_info
+ print -r -- "$PWD"
+ )
+
+ # Add callback. Needs to be a widget, so we can refresh the prompt.
+ zle -Fw "$prompt_vcs_fd" prompt_vcs_fd-widget
+}
+
+# Callback widget function for our async fetch of VCS changes
+prompt_vcs_fd-widget() {
+ emulate -L zsh
+
+ [[ -n $1 ]] ||
+ return # Don't accidentally close fd 0.
+ {
+ local -i fd=$1
+ zle -F "$fd" # Detach ourselves, so we don't get called more than once.
+
+ [[ -z $2 ]] ||
+ return # Error occured.
+
+ local -a reply
+ IFS=$'\0' read -Aru "$fd"
+ } always {
+ exec {fd}<&-
+ }
+ [[ $reply[-1] == $PWD ]] ||
+ return # Info is not for current dir.
+
+ shift -p reply
+ typeset -ga _prompt_vcs_info_msg_=( "${(@Q)reply}" )
+ RPS1="$_prompt_vcs_info_msg_[3]"
+ zle .reset-prompt
+}
+
+prompt_vcs_setup() {
+ prompt_opts=( cr percent sp ) # Tell promptinit which options to set.
+
+ # Indent left continuation prompt for each open shell construct.
+ local -a indent=( '%('{1..36}'_, ,)' )
+ PS2="${(j::)indent}"
+
+ RPS2='%F{yellow}# %^%f'
+ PS4=$'#->%(?,%F{green},%B%F{red}%S)%?%b%f%s\t%e+%F{green}%1N%f:%I %(1_,%F{yellow}%K{black}%_%f%k ,)'
+ SPROMPT='Correct %B%F{red}%U%R%b%f%u to %B%F{green}%r%b%f? [%Sy%ses|%Sn%so|%Se%sdit|%Sa%sbort] '
+ PROMPT_EOL_MARK='%S%F{cyan}%#%s%f'
+ zle_highlight=(
+ isearch:fg=black,bg=yellow
+ special:fg=cyan,bold
+ region:fg=white,bg=blue,bold
+ suffix:fg=white,bg=blue,bold
+ paste:bold
+ )
+
+ add-zsh-hook chpwd prompt_vcs_chpwd
+ add-zsh-hook precmd prompt_vcs_precmd
+ zle -N prompt_vcs_fd-widget # Callback widget for async VCS update.
+
+ prompt_vcs_chpwd
+}
+
+prompt_vcs_setup "$@"
--
2.30.1 (Apple Git-130)
Messages sorted by:
Reverse Date,
Date,
Thread,
Author