diff options
author | Oliver Kiddle <opk@zsh.org> | 2015-05-13 23:05:20 +0200 |
---|---|---|
committer | Oliver Kiddle <opk@zsh.org> | 2015-05-13 23:05:20 +0200 |
commit | d257f0143e69c3724466c4c92f59538d2f3fffd1 (patch) | |
tree | a9c51888931eb9329ad0918f82b1563ca1ad7588 /Functions/Zle/select-bracketed | |
parent | f454ee26a8f644a2d37874ea8274054203892f91 (diff) | |
download | zsh-d257f0143e69c3724466c4c92f59538d2f3fffd1.tar.gz zsh-d257f0143e69c3724466c4c92f59538d2f3fffd1.tar.xz zsh-d257f0143e69c3724466c4c92f59538d2f3fffd1.zip |
35093: new zle widgets for Vim-style text objects
Diffstat (limited to 'Functions/Zle/select-bracketed')
-rw-r--r-- | Functions/Zle/select-bracketed | 56 |
1 files changed, 56 insertions, 0 deletions
diff --git a/Functions/Zle/select-bracketed b/Functions/Zle/select-bracketed new file mode 100644 index 000000000..00f51be2c --- /dev/null +++ b/Functions/Zle/select-bracketed @@ -0,0 +1,56 @@ +# Text object for matching characters between matching pairs of brackets +# +# So for example, given (( i+1 )), the vi command ci( will change +# all the text between matching colons. +# +# The following is an example of how to enable this: +# autoload -U select-bracketed +# zle -N select-bracketed +# for m in visual viopp; do +# for c in {a,i}${(s..)^:-'()[]{}<>bB'}; do +# bindkey -M $m $c select-bracketed +# done +# done + +local style=${${1:-$KEYS}[1]} matching="(){}[]<>bbBB" +local -i find=${NUMERIC:-1} idx=${matching[(I)[${${1:-$KEYS}[2]}]]}%9 +(( idx )) || return 1 # no corresponding closing bracket +local lmatch=${matching[1 + (idx-1) & ~1]} +local rmatch=${matching[1 + (idx-1) | 1]} +local -i start=CURSOR+1 end=CURSOR+1 rfind=find + +[[ $BUFFER[start] = "$rmatch" ]] && (( start--, end-- )) +if (( REGION_ACTIVE && MARK != CURSOR)); then + (( MARK < CURSOR && (start=end=MARK+1) )) + local -i origstart=start-1 + [[ $style = i ]] && (( origstart-- )) +fi + +while (( find )); do + for (( ; find && start; --start )); do + case $BUFFER[start] in + "$lmatch") (( find-- )) ;; + "$rmatch") (( find++ )) ;; + esac + done + + (( find )) && return 1 # opening bracket not found + + while (( rfind && end++ < $#BUFFER )); do + case $BUFFER[end] in + "$lmatch") (( rfind++ )) ;; + "$rmatch") (( rfind-- )) ;; + esac + done + + (( rfind )) && return 1 # closing bracket not found + + (( REGION_ACTIVE && MARK != CURSOR && start >= origstart && + ( find=rfind=${NUMERIC:-1} ) )) +done + +[[ $style = i ]] && (( start++, end-- )) +(( REGION_ACTIVE = !!REGION_ACTIVE )) +[[ $KEYMAP = vicmd ]] && (( REGION_ACTIVE && end-- )) +MARK=$start +CURSOR=$end |