1
0
mirror of git://git.code.sf.net/p/zsh/code synced 2024-10-01 08:21:17 +02:00
zsh/Functions/Zle/history-pattern-search
Peter Stephenson ab899b76c7 20222: turn on max function depth
plus unposted reference to this in README
plus unposted neatening of 20219 history-pattern-search
2004-07-30 11:09:16 +00:00

84 lines
2.1 KiB
Plaintext

# Prompt for an search in the history for a pattern.
# Patterns to search are standard zsh patterns, but may include
# ^ at the start or $ at the end to anchor the pattern to the
# start or end of the history entry respectively.
#
# To search backwards, create a widget history-pattern-search-backward:
# zle -N history-pattern-search-backward history-pattern-search
# and to search forwards, create history-pattern-search-forward
# zle -N history-pattern-search-forward history-pattern-search
# Use extended globbing by default.
emulate -L zsh
setopt extendedglob
# Load required features.
autoload -U read-from-minibuffer
zmodload -i zsh/parameter
local REPLY dir new
integer i
local -a found match mbegin mend
# Decide if we are searching backwards or forwards.
if [[ $WIDGET = *forward* ]]; then
dir="forw"
else
dir="rev"
fi
# Read pattern. Prompt could be made customisable.
read-from-minibuffer "pat ($dir): " $_last_history_pattern_search
_last_history_pattern_search=$REPLY
# Abort if bad status or nothing entered
[[ $? -ne 0 || -z $REPLY ]] && return 0
# Handle start-of-line anchor.
if [[ $REPLY = \^* ]]; then
REPLY=$REPLY[2,-1]
else
REPLY="*$REPLY"
fi
# Handle end-of-line anchor.
if [[ $REPLY = *\$ ]]; then
REPLY=$REPLY[1,-2]
else
REPLY="$REPLY*"
fi
# Search history for pattern.
# As $history is an associative array we can get all matches.
found=(${(kon)history[(R)$REPLY]})
if [[ $dir = forw ]]; then
# Searching forward. Look back through matches until we
# get back to the current history number.
for (( i = ${#found}; i >= 1; i-- )); do
(( $found[$i] <= HISTNO )) && break
new=$found[$i]
done
else
# Searching backward. Look forward through matches until we
# reach the current history number.
for (( i = 1; i <= ${#found}; i++ )); do
(( $found[$i] >= HISTNO )) && break
new=$found[$i]
done
fi
if [[ -n $new ]]; then
# Match found. Move to line.
HISTNO=$new
if [[ $REPLY = *\* && $history[$new] = (#b)(${~REPLY[1,-2]})* ]]; then
# If not anchored to the end, move to the end of the pattern
# we were searching for.
CURSOR=$mend[1]
fi
return 0
else
return 1
fi