123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459 |
- #
- # Sets key bindings.
- #
- # Authors:
- # Sorin Ionescu <sorin.ionescu@gmail.com>
- #
- # Return if requirements are not found.
- if [[ "$TERM" == 'dumb' ]]; then
- return 1
- fi
- #
- # Options
- #
- setopt BEEP # Beep on error in line editor.
- #
- # Variables
- #
- # Treat these characters as part of a word.
- zstyle -s ':prezto:module:editor' wordchars 'WORDCHARS' \
- || WORDCHARS='*?_-.[]~&;!#$%^(){}<>'
- # Use human-friendly identifiers.
- zmodload zsh/terminfo
- typeset -gA key_info
- key_info=(
- 'Control' '\C-'
- 'ControlLeft' '\e[1;5D \e[5D \e\e[D \eOd'
- 'ControlRight' '\e[1;5C \e[5C \e\e[C \eOc'
- 'ControlPageUp' '\e[5;5~'
- 'ControlPageDown' '\e[6;5~'
- 'Escape' '\e'
- 'Meta' '\M-'
- 'Backspace' "^?"
- 'Delete' "^[[3~"
- 'F1' "$terminfo[kf1]"
- 'F2' "$terminfo[kf2]"
- 'F3' "$terminfo[kf3]"
- 'F4' "$terminfo[kf4]"
- 'F5' "$terminfo[kf5]"
- 'F6' "$terminfo[kf6]"
- 'F7' "$terminfo[kf7]"
- 'F8' "$terminfo[kf8]"
- 'F9' "$terminfo[kf9]"
- 'F10' "$terminfo[kf10]"
- 'F11' "$terminfo[kf11]"
- 'F12' "$terminfo[kf12]"
- 'Insert' "$terminfo[kich1]"
- 'Home' "$terminfo[khome]"
- 'PageUp' "$terminfo[kpp]"
- 'End' "$terminfo[kend]"
- 'PageDown' "$terminfo[knp]"
- 'Up' "$terminfo[kcuu1]"
- 'Left' "$terminfo[kcub1]"
- 'Down' "$terminfo[kcud1]"
- 'Right' "$terminfo[kcuf1]"
- 'BackTab' "$terminfo[kcbt]"
- )
- # Set empty $key_info values to an invalid UTF-8 sequence to induce silent
- # bindkey failure.
- for key in "${(k)key_info[@]}"; do
- if [[ -z "$key_info[$key]" ]]; then
- key_info[$key]='�'
- fi
- done
- #
- # External Editor
- #
- # Allow command line editing in an external editor.
- autoload -Uz edit-command-line
- zle -N edit-command-line
- #
- # Functions
- #
- # Runs bindkey but for all of the keymaps. Running it with no arguments will
- # print out the mappings for all of the keymaps.
- function bindkey-all {
- local keymap=''
- for keymap in $(bindkey -l); do
- [[ "$#" -eq 0 ]] && printf "#### %s\n" "${keymap}" 1>&2
- bindkey -M "${keymap}" "$@"
- done
- }
- # Exposes information about the Zsh Line Editor via the $editor_info associative
- # array.
- function editor-info {
- # Ensure that we're going to set the editor-info for prompts that
- # are prezto managed and/or compatible.
- if zstyle -t ':prezto:module:prompt' managed; then
- # Clean up previous $editor_info.
- unset editor_info
- typeset -gA editor_info
- if [[ "$KEYMAP" == 'vicmd' ]]; then
- zstyle -s ':prezto:module:editor:info:keymap:alternate' format 'REPLY'
- editor_info[keymap]="$REPLY"
- else
- zstyle -s ':prezto:module:editor:info:keymap:primary' format 'REPLY'
- editor_info[keymap]="$REPLY"
- if [[ "$ZLE_STATE" == *overwrite* ]]; then
- zstyle -s ':prezto:module:editor:info:keymap:primary:overwrite' format 'REPLY'
- editor_info[overwrite]="$REPLY"
- else
- zstyle -s ':prezto:module:editor:info:keymap:primary:insert' format 'REPLY'
- editor_info[overwrite]="$REPLY"
- fi
- fi
- unset REPLY
- zle zle-reset-prompt
- fi
- }
- zle -N editor-info
- # Reset the prompt based on the current context and whether the prompt utilizes
- # the editor:info zstyle. If the prompt does utilize the editor:info, we must
- # reset the prompt, otherwise the change in the prompt will never update. If the
- # prompt does not utilize the editor:info, we simply redisplay the command line.
- function zle-reset-prompt {
- # Explicitly check to see if there is an editor info keymap set that would
- # require a reset of the prompt
- if zstyle -L ':prezto:module:editor:info*' | grep -v 'completing' > /dev/null 2>&1; then
- # If we aren't within one of the specified contexts, then we want to reset
- # the prompt with the appropriate editor_info[keymap] if there is one.
- if [[ $CONTEXT != (select|cont) ]]; then
- zle reset-prompt
- fi
- fi
- zle -R
- }
- zle -N zle-reset-prompt
- # Updates editor information when the keymap changes.
- function zle-keymap-select {
- zle editor-info
- }
- zle -N zle-keymap-select
- # Enables terminal application mode and updates editor information.
- function zle-line-init {
- # The terminal must be in application mode when ZLE is active for $terminfo
- # values to be valid.
- if (( $+terminfo[smkx] )); then
- # Enable terminal application mode.
- echoti smkx
- fi
- # Update editor information.
- zle editor-info
- }
- zle -N zle-line-init
- # Disables terminal application mode and updates editor information.
- function zle-line-finish {
- # The terminal must be in application mode when ZLE is active for $terminfo
- # values to be valid.
- if (( $+terminfo[rmkx] )); then
- # Disable terminal application mode.
- echoti rmkx
- fi
- # Update editor information.
- zle editor-info
- }
- zle -N zle-line-finish
- # Toggles emacs overwrite mode and updates editor information.
- function overwrite-mode {
- zle .overwrite-mode
- zle editor-info
- }
- zle -N overwrite-mode
- # Enters vi insert mode and updates editor information.
- function vi-insert {
- zle .vi-insert
- zle editor-info
- }
- zle -N vi-insert
- # Moves to the first non-blank character then enters vi insert mode and updates
- # editor information.
- function vi-insert-bol {
- zle .vi-insert-bol
- zle editor-info
- }
- zle -N vi-insert-bol
- # Enters vi replace mode and updates editor information.
- function vi-replace {
- zle .vi-replace
- zle editor-info
- }
- zle -N vi-replace
- # Expands .... to ../..
- function expand-dot-to-parent-directory-path {
- if [[ $LBUFFER = *.. ]]; then
- LBUFFER+='/..'
- else
- LBUFFER+='.'
- fi
- }
- zle -N expand-dot-to-parent-directory-path
- # Displays an indicator when completing.
- function expand-or-complete-with-indicator {
- local indicator
- zstyle -s ':prezto:module:editor:info:completing' format 'indicator'
- # This is included to work around a bug in zsh which shows up when interacting
- # with multi-line prompts.
- if [[ -z "$indicator" ]]; then
- zle expand-or-complete
- return
- fi
- print -Pn "$indicator"
- zle expand-or-complete
- zle redisplay
- }
- zle -N expand-or-complete-with-indicator
- # Inserts 'sudo ' at the beginning of the line.
- function prepend-sudo {
- if [[ "$BUFFER" != su(do|)\ * ]]; then
- BUFFER="sudo $BUFFER"
- (( CURSOR += 5 ))
- fi
- }
- zle -N prepend-sudo
- # Expand aliases
- function glob-alias {
- zle _expand_alias
- zle expand-word
- zle magic-space
- }
- zle -N glob-alias
- # Toggle the comment character at the start of the line. This is meant to work
- # around a buggy implementation of pound-insert in zsh.
- #
- # This is currently only used for the emacs keys because vi-pound-insert has
- # been reported to work properly.
- function pound-toggle {
- if [[ "$BUFFER" = '#'* ]]; then
- # Because of an oddity in how zsh handles the cursor when the buffer size
- # changes, we need to make this check before we modify the buffer and let
- # zsh handle moving the cursor back if it's past the end of the line.
- if [[ $CURSOR != $#BUFFER ]]; then
- (( CURSOR -= 1 ))
- fi
- BUFFER="${BUFFER:1}"
- else
- BUFFER="#$BUFFER"
- (( CURSOR += 1 ))
- fi
- }
- zle -N pound-toggle
- # Reset to default key bindings.
- bindkey -d
- #
- # Emacs Key Bindings
- #
- for key in "$key_info[Escape]"{B,b} "${(s: :)key_info[ControlLeft]}" \
- "${key_info[Escape]}${key_info[Left]}"
- bindkey -M emacs "$key" emacs-backward-word
- for key in "$key_info[Escape]"{F,f} "${(s: :)key_info[ControlRight]}" \
- "${key_info[Escape]}${key_info[Right]}"
- bindkey -M emacs "$key" emacs-forward-word
- # Kill to the beginning of the line.
- for key in "$key_info[Escape]"{K,k}
- bindkey -M emacs "$key" backward-kill-line
- # Redo.
- bindkey -M emacs "$key_info[Escape]_" redo
- # Search previous character.
- bindkey -M emacs "$key_info[Control]X$key_info[Control]B" vi-find-prev-char
- # Match bracket.
- bindkey -M emacs "$key_info[Control]X$key_info[Control]]" vi-match-bracket
- # Edit command in an external editor.
- bindkey -M emacs "$key_info[Control]X$key_info[Control]E" edit-command-line
- if (( $+widgets[history-incremental-pattern-search-backward] )); then
- bindkey -M emacs "$key_info[Control]R" \
- history-incremental-pattern-search-backward
- bindkey -M emacs "$key_info[Control]S" \
- history-incremental-pattern-search-forward
- fi
- # Toggle comment at the start of the line. Note that we use pound-toggle which
- # is similar to pount insert, but meant to work around some issues that were
- # being seen in iTerm.
- bindkey -M emacs "$key_info[Escape];" pound-toggle
- #
- # Vi Key Bindings
- #
- # Edit command in an external editor emacs style (v is used for visual mode)
- bindkey -M vicmd "$key_info[Control]X$key_info[Control]E" edit-command-line
- # Undo/Redo
- bindkey -M vicmd "u" undo
- bindkey -M viins "$key_info[Control]_" undo
- bindkey -M vicmd "$key_info[Control]R" redo
- if (( $+widgets[history-incremental-pattern-search-backward] )); then
- bindkey -M vicmd "?" history-incremental-pattern-search-backward
- bindkey -M vicmd "/" history-incremental-pattern-search-forward
- else
- bindkey -M vicmd "?" history-incremental-search-backward
- bindkey -M vicmd "/" history-incremental-search-forward
- fi
- # Toggle comment at the start of the line.
- bindkey -M vicmd "#" vi-pound-insert
- #
- # Emacs and Vi Key Bindings
- #
- # Unbound keys in vicmd and viins mode will cause really odd things to happen
- # such as the casing of all the characters you have typed changing or other
- # undefined things. In emacs mode they just insert a tilde, but bind these keys
- # in the main keymap to a noop op so if there is no keybind in the users mode
- # it will fall back and do nothing.
- function _prezto-zle-noop { ; }
- zle -N _prezto-zle-noop
- local -a unbound_keys
- unbound_keys=(
- "${key_info[F1]}"
- "${key_info[F2]}"
- "${key_info[F3]}"
- "${key_info[F4]}"
- "${key_info[F5]}"
- "${key_info[F6]}"
- "${key_info[F7]}"
- "${key_info[F8]}"
- "${key_info[F9]}"
- "${key_info[F10]}"
- "${key_info[F11]}"
- "${key_info[F12]}"
- "${key_info[PageUp]}"
- "${key_info[PageDown]}"
- "${key_info[ControlPageUp]}"
- "${key_info[ControlPageDown]}"
- )
- for keymap in $unbound_keys; do
- bindkey -M viins "${keymap}" _prezto-zle-noop
- bindkey -M vicmd "${keymap}" _prezto-zle-noop
- done
- # Keybinds for all keymaps
- for keymap in 'emacs' 'viins' 'vicmd'; do
- bindkey -M "$keymap" "$key_info[Home]" beginning-of-line
- bindkey -M "$keymap" "$key_info[End]" end-of-line
- done
- # Keybinds for all vi keymaps
- for keymap in viins vicmd; do
- # Ctrl + Left and Ctrl + Right bindings to forward/backward word
- for key in "${(s: :)key_info[ControlLeft]}"
- bindkey -M "$keymap" "$key" vi-backward-word
- for key in "${(s: :)key_info[ControlRight]}"
- bindkey -M "$keymap" "$key" vi-forward-word
- done
- # Keybinds for emacs and vi insert mode
- for keymap in 'emacs' 'viins'; do
- bindkey -M "$keymap" "$key_info[Insert]" overwrite-mode
- bindkey -M "$keymap" "$key_info[Delete]" delete-char
- bindkey -M "$keymap" "$key_info[Backspace]" backward-delete-char
- bindkey -M "$keymap" "$key_info[Left]" backward-char
- bindkey -M "$keymap" "$key_info[Right]" forward-char
- # Expand history on space.
- bindkey -M "$keymap" ' ' magic-space
- # Clear screen.
- bindkey -M "$keymap" "$key_info[Control]L" clear-screen
- # Expand command name to full path.
- for key in "$key_info[Escape]"{E,e}
- bindkey -M "$keymap" "$key" expand-cmd-path
- # Duplicate the previous word.
- for key in "$key_info[Escape]"{M,m}
- bindkey -M "$keymap" "$key" copy-prev-shell-word
- # Use a more flexible push-line.
- for key in "$key_info[Control]Q" "$key_info[Escape]"{q,Q}
- bindkey -M "$keymap" "$key" push-line-or-edit
- # Bind Shift + Tab to go to the previous menu item.
- bindkey -M "$keymap" "$key_info[BackTab]" reverse-menu-complete
- # Complete in the middle of word.
- bindkey -M "$keymap" "$key_info[Control]I" expand-or-complete
- # Expand .... to ../..
- if zstyle -t ':prezto:module:editor' dot-expansion; then
- bindkey -M "$keymap" "." expand-dot-to-parent-directory-path
- fi
- # Display an indicator when completing.
- bindkey -M "$keymap" "$key_info[Control]I" \
- expand-or-complete-with-indicator
- # Insert 'sudo ' at the beginning of the line.
- bindkey -M "$keymap" "$key_info[Control]X$key_info[Control]S" prepend-sudo
- # control-space expands all aliases, including global
- bindkey -M "$keymap" "$key_info[Control] " glob-alias
- done
- # Delete key deletes character in vimcmd cmd mode instead of weird default functionality
- bindkey -M vicmd "$key_info[Delete]" delete-char
- # Do not expand .... to ../.. during incremental search.
- if zstyle -t ':prezto:module:editor' dot-expansion; then
- bindkey -M isearch . self-insert 2> /dev/null
- fi
- #
- # Layout
- #
- # Set the key layout.
- zstyle -s ':prezto:module:editor' key-bindings 'key_bindings'
- if [[ "$key_bindings" == (emacs|) ]]; then
- bindkey -e
- elif [[ "$key_bindings" == vi ]]; then
- bindkey -v
- else
- print "prezto: editor: invalid key bindings: $key_bindings" >&2
- fi
- unset key{,map,_bindings}
|