Source code for xontrib.whole_word_jumping

"""Jump/delete across whole (non-whitespace) words with Ctrl+Left/Right/Delete/Backspace.

Control+left/right: Jump to previous/next whole word
Control+backspace: Delete to beginning of whole word
Control+delete: Delete to end of whole word
Shift+delete: Delete whole word

Alt+Left/Right/Delete/Backspace remain unmodified:

Alt+left/right: Jump to previous/next token
Alt+backspace: Delete to beginning of token
Alt+delete: Delete to end of token

Some terminals cannot differentiate between Backspace and Control+Backspace.
In this case, users can set `$XONSH_WHOLE_WORD_CTRL_BKSP = False` to skip
configuration of the Control+Backspace key binding.
"""

import prompt_toolkit.input.ansi_escape_sequences as ansiseq
from prompt_toolkit.filters import EmacsInsertMode, ViInsertMode
from prompt_toolkit.key_binding.bindings.named_commands import get_by_name
from prompt_toolkit.keys import Keys

from xonsh.built_ins import XSH, XonshSession
from xonsh.platform import ON_WINDOWS


[docs]def custom_keybindings(bindings, **kw): insert_mode = ViInsertMode() | EmacsInsertMode() # Key bindings for jumping over whole words (everything that's not # white space) using Ctrl+Left and Ctrl+Right; # Alt+Left and Alt+Right still jump over smaller word segments. # See https://github.com/xonsh/xonsh/issues/2403 @bindings.add(Keys.ControlLeft) def ctrl_left(event): buff = event.current_buffer pos = buff.document.find_previous_word_beginning(count=event.arg, WORD=True) if pos: buff.cursor_position += pos @bindings.add(Keys.ControlRight) def ctrl_right(event): buff = event.current_buffer pos = buff.document.find_next_word_ending(count=event.arg, WORD=True) if pos: buff.cursor_position += pos @bindings.add(Keys.ShiftDelete, filter=insert_mode) def delete_surrounding_big_word(event): buff = event.current_buffer startpos, endpos = buff.document.find_boundaries_of_current_word(WORD=True) startpos = buff.cursor_position + startpos - 1 startpos = 0 if startpos < 0 else startpos endpos = buff.cursor_position + endpos endpos = endpos + 1 if startpos == 0 else endpos buff.text = buff.text[:startpos] + buff.text[endpos:] buff.cursor_position = startpos @bindings.add(Keys.ControlDelete, filter=insert_mode) def delete_big_word(event): buff = event.current_buffer pos = buff.document.find_next_word_ending(count=event.arg, WORD=True) if pos: buff.delete(count=pos) @bindings.add(Keys.Escape, Keys.Delete, filter=insert_mode) def delete_small_word(event): get_by_name("kill-word").call(event) # PTK sets both "\x7f" (^?) and "\x08" (^H) to the same behavior. Refs: # https://github.com/prompt-toolkit/python-prompt-toolkit/blob/65c3d0607c69c19d80abb052a18569a2546280e5/src/prompt_toolkit/input/ansi_escape_sequences.py#L65 # https://github.com/prompt-toolkit/python-prompt-toolkit/issues/257#issuecomment-190328366 # We patch the ANSI sequences used by PTK. This requires a terminal # that sends different codes for <backspace> and <control-h>. # PTK sets Keys.Backspace = Keys.ControlH, so we hardcode the code. # Windows has the codes reversed, see https://github.com/xonsh/xonsh/commit/406d20f78f18af39d9bbaf9580b0a763df78a0db if XSH.env.get("XONSH_WHOLE_WORD_CTRL_BKSP", True): CONTROL_BKSP = "\x08" if ON_WINDOWS: # issue #4845: Windows only (isort competes with black :-() from prompt_toolkit.input import win32 as ptk_win32 # black:skip # On windows BKSP is "\x08" and CTRL-BKSP is "\x7f" CONTROL_BKSP = "\x7f" ptk_win32.ConsoleInputReader.mappings[b"\x7f"] = CONTROL_BKSP ansiseq.ANSI_SEQUENCES[CONTROL_BKSP] = CONTROL_BKSP ansiseq.REVERSE_ANSI_SEQUENCES[CONTROL_BKSP] = CONTROL_BKSP @bindings.add(CONTROL_BKSP, filter=insert_mode) def backward_delete_big_word(event): get_by_name("unix-word-rubout").call(event)
# backward_delete_small_word works on Alt+Backspace by default def _load_xontrib_(xsh: XonshSession, **_): xsh.builtins.events.on_ptk_create(custom_keybindings)