about summary refs log tree commit diff
path: root/Functions/Zle/edit-command-line
blob: 3781244b2734164221d0bbb45598974dff0a7608 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
# Edit the command line using your usual editor.
# Binding this to '!' in the vi command mode map,
#   autoload -Uz edit-command-line
#   zle -N edit-command-line
#   bindkey -M vicmd '!' edit-command-line
# will give ksh-like behaviour for that key,
# except that it will handle multi-line buffers properly.

emulate -L zsh
local left right prebuffer buffer=$BUFFER lbuffer=$LBUFFER
# set up parameters depending on which context we are called from,
# see below comment for more details
if (( REGION_ACTIVE )); then
  if (( CURSOR < MARK )); then
    left=$CURSOR right=$MARK
    lbuffer=
  else
    left=$MARK right=$CURSOR
    lbuffer[right-left,-1]=
  fi
  (( left++ ))
  buffer=$BUFFER[left,right]
elif (( ! ZLE_RECURSIVE )); then
  prebuffer=$PREBUFFER
fi

() {
  exec </dev/tty

  # Compute the cursor's position in bytes, not characters.
  setopt localoptions nomultibyte noksharrays

  (( $+zle_bracketed_paste )) && print -r -n - $zle_bracketed_paste[2]

  # Open the editor, placing the cursor at the right place if we know how.
  local -a editor
  zstyle -a :zle:$WIDGET editor editor
  if (( ! $#editor )); then
    editor=( "${(@Q)${(z)${VISUAL:-${EDITOR:-vi}}}}" )
  fi
  case $editor in
    (*vim*)
      integer byteoffset=$(( $#prebuffer + $#lbuffer + 1 ))
      "${(@)editor}" -c "normal! ${byteoffset}go" -- $1;;
    (*emacs*)
      local lines=( "${(@f):-"$prebuffer$lbuffer"}" )
      "${(@)editor}" +${#lines}:$((${#lines[-1]} + 1)) $1;;
    (*) "${(@)editor}" $1;;
  esac

  (( $+zle_bracketed_paste )) && print -r -n - $zle_bracketed_paste[1]

  # Replace the buffer with the editor output.
  # avoid drawing a new prompt when we can:
  # - in recursive-edit, the send-break will just cancel the recursive-edit
  #   rather than reload the line from print -z so in that case we want to
  #   just set $BUFFER (unfortunately, recursive-edit doesn't reset CONTEXT
  #   or PREBUFFER so we have to explicitly handle this case, which overrides
  #   the following point)
  # - when we are at PS2 (CONTEXT == cont && ! ZLE_RECURSIVE) we do want the
  #   break or otherwise the text from PREBUFFER will be inserted twice
  # - when the region is active, we only want to change the parts of BUFFER
  #   covered by the region, and any PREBUFFER stays as PREBUFFER
  # - in all other cases (that I can think of) we also just want to set
  #   $BUFFER directly.
  if (( REGION_ACTIVE )); then
    # adjust the length of the region to the length of the edited text
    local prelen=$#BUFFER
    BUFFER[left,right]="$(<$1)"
    if (( MARK > CURSOR )); then
      (( MARK += $#BUFFER - prelen ))
    else
      (( CURSOR += $#BUFFER - prelen ))
    fi
  elif [[ $CONTEXT != cont ]] || (( ZLE_RECURSIVE )); then
    BUFFER="$(<$1)" 
  else
    print -Rz - "$(<$1)"
    zle send-break
  fi

} =(<<<"$prebuffer$buffer")