1
0
Fork 0
mirror of git://git.code.sf.net/p/zsh/code synced 2025-01-01 05:16:05 +01:00
zsh/Functions/Zle/select-word-match

120 lines
3.8 KiB
Text

# Select the entire word around the cursor. Intended for use as
# a vim-style text object in vi mode but with customisable
# word boundaries.
#
# For example:
# autoload -U select-word-match
# zle -N select-in-camel select-word-match
# bindkey -M viopp ic select-in-camel
# zstyle ':zle:*-camel' word-style normal-subword
emulate -L zsh
setopt extendedglob
local curcontext=:zle:$WIDGET
local -A matched_words
# Start and end of range of characters
integer pos1 pos2 num=${NUMERIC:-1}
local style word
# choose between inner word or a word style of widget
for style in $1 ${${WIDGET#*-}[1]} $KEYS[1] "i"; do
[[ $style = [ai] ]] && break
done
autoload -Uz match-words-by-style
while (( num-- )); do
if (( MARK > CURSOR )); then
# if cursor is at the start of the selection, just move back a word
match-words-by-style
if [[ $style = i && -n $matched_words[ws-before-cursor] ]]; then
word=$matched_words[ws-before-cursor]
else
word=$matched_words[word-before-cursor]$matched_words[ws-before-cursor]
fi
if [[ -n $word ]]; then
(( CURSOR -= ${#word} ))
else
return 1
fi
elif (( MARK >= 0 && MARK < CURSOR )); then
# cursor at the end, move forward a word
(( CURSOR+1 == $#BUFFER )) && return 1
(( CURSOR++ ))
match-words-by-style
if [[ -n $matched_words[ws-after-cursor] ]]; then
if [[ $style = i ]]; then
# just skip the whitespace
word=$matched_words[ws-after-cursor]
else
# skip the whitespace plus word
word=$matched_words[ws-after-cursor]$matched_words[word-after-cursor]
fi
else
if [[ $style = i ]]; then
# skip the word
word=$matched_words[word-after-cursor]
else
# skip word and following whitespace
word=$matched_words[word-after-cursor]$matched_words[ws-after-word]
fi
fi
(( CURSOR += ${#word} - 1 ))
else
match-words-by-style
if (( ${matched_words[is-word-start]} )); then
# The word we are selecting starts at the cursor position.
pos1=$CURSOR
else
# No whitespace before us, so select any wordcharacters there.
pos1="${#matched_words[start]}"
fi
if [[ -n "${matched_words[ws-after-cursor]}" ]]; then
if [[ -n "${matched_words[ws-before-cursor]}" ]] || (( CURSOR == 0 )); then
# whitespace either side, select it
(( pos1 = CURSOR - ${#matched_words[ws-before-cursor]} ))
(( pos2 = CURSOR + ${#matched_words[ws-after-cursor]} ))
else
# There's whitespace at the cursor position, so only select
# up to the cursor position.
(( pos2 = CURSOR + 1 ))
fi
else
# No whitespace at the cursor position, so select the
# current character and any following wordcharacters.
(( pos2 = CURSOR + ${#matched_words[word-after-cursor]} ))
fi
if [[ $style = a ]]; then
if [[ -n "${matched_words[ws-after-cursor]}" && ( -n "${matched_words[ws-before-cursor]}" || CURSOR -eq 0 ) ]]; then
# in the middle of whitespace so grab a word
if [[ -n "${matched_words[word-after-cursor]}" ]]; then
(( pos2 += ${#matched_words[word-after-cursor]} )) # preferably the one after
else
(( pos1 -= ${#matched_words[word-before-cursor]} )) # otherwise the one before
fi
elif [[ -n "${matched_words[ws-after-word]}" ]]; then
(( pos2 += ${#matched_words[ws-after-word]} ))
elif [[ -n "${matched_words[ws-before-cursor]}" ]]; then
# couldn't grab whitespace forwards so try backwards
(( pos1 -= ${#matched_words[ws-before-cursor]} ))
elif (( pos1 > 0 )); then
# There might have been whitespace before the word
(( CURSOR = pos1 ))
match-words-by-style
if [[ -n "${matched_words[ws-before-cursor]}" ]]; then
(( pos1 -= ${#matched_words[ws-before-cursor]} ))
fi
fi
fi
(( MARK = pos1, CURSOR = pos2-1 ))
fi
done
if [[ $KEYMAP == vicmd ]] && (( !REGION_ACTIVE )); then
(( CURSOR++ )) # Need to include cursor position for operators
fi