189 lines
5.4 KiB
Bash
189 lines
5.4 KiB
Bash
#!/bin/zsh
|
|
# zed
|
|
#
|
|
# No other shell could do this.
|
|
# Edit small files with the command line editor.
|
|
# Use ^X^W to save (or ZZ in vicmd mode), ^C to abort.
|
|
# Option -f: edit shell functions. (Also if called as fned.)
|
|
# Option -h: edit shell history. (Also if called as histed.)
|
|
|
|
setopt localoptions noksharrays
|
|
|
|
local var opts zed_file_name
|
|
# We do not want timeout while we are editing a file
|
|
integer TMOUT=0 okargs=1 fun hist bind
|
|
local -a expand
|
|
|
|
zparseopts -D -A opts f h b x:
|
|
fun=$+opts[-f]
|
|
hist=$+opts[-h]
|
|
bind=$+opts[-b]
|
|
if [[ $opts[-x] == <-> ]]; then
|
|
expand=(-x $opts[-x])
|
|
elif (( $+opts[-x] )); then
|
|
print -r "Integer expected after -x: $opts[-x]" >&2
|
|
return 1
|
|
fi
|
|
|
|
[[ $0 = fned ]] && fun=1
|
|
[[ $0 = histed ]] && hist=1
|
|
(( hist && $# <= 2 )) && okargs=$#
|
|
(( bind )) && okargs=0
|
|
|
|
if (( $# != okargs || bind + fun + hist > 1 )); then
|
|
echo 'Usage:
|
|
zed filename
|
|
zed -f [ -x N ] function
|
|
zed -h [ filename [ size ] ]
|
|
zed -b' >&2
|
|
return 1
|
|
fi
|
|
|
|
local curcontext=zed:::
|
|
|
|
() {
|
|
# Matching used in zstyle -m: hide result from caller.
|
|
# Variables not used directly here.
|
|
local -a match mbegin mend
|
|
zstyle -m ":completion:zed:*" insert-tab '*' ||
|
|
zstyle ":completion:zed:*" insert-tab yes
|
|
}
|
|
|
|
zmodload zsh/terminfo 2>/dev/null
|
|
|
|
__zed_pg_up()
|
|
{
|
|
integer count=$(( LINES / 2 - 1 ))
|
|
while (( count -- )); do
|
|
zle up-line
|
|
done
|
|
}
|
|
|
|
__zed_pg_down()
|
|
{
|
|
integer count=$(( LINES / 2 - 1 ))
|
|
while (( count -- )); do
|
|
zle down-line
|
|
done
|
|
}
|
|
|
|
if ! zle -la __zed_pg_up __zed_pg_down; then
|
|
zle -N __zed_pg_up
|
|
zle -N __zed_pg_down
|
|
fi
|
|
|
|
if (( bind )) || ! bindkey -M zed >&/dev/null; then
|
|
# Make the zed keymap a copy of the current main.
|
|
bindkey -N zed main
|
|
# Save the current main. In zle widgets called from
|
|
# zed we may want to set this temporally.
|
|
bindkey -A main zed-normal-keymap
|
|
|
|
# Define a widget to use at startup, undo shouldn't clear initial buffer
|
|
__zed_init() {
|
|
UNDO_LIMIT_NO=$UNDO_CHANGE_NO
|
|
}
|
|
zle -N __zed_init
|
|
|
|
# Assign some default keys.
|
|
# Depending on your stty's, you may be able to use ^J as accept-line, else:
|
|
|
|
# The following isn't useful if we are copying viins, but that's
|
|
# a nicety.
|
|
bindkey -M zed '^x^w' accept-line
|
|
bindkey -M zed '^M' self-insert-unmeta
|
|
|
|
[[ ${+terminfo} = 1 ]] && {
|
|
[[ -n "$terminfo[kpp]" ]] && bindkey -M zed "$terminfo[kpp]" __zed_pg_up
|
|
[[ -n "$terminfo[knp]" ]] && bindkey -M zed "$terminfo[knp]" __zed_pg_down
|
|
[[ -n "$terminfo[khome]" ]] && bindkey -M zed "$terminfo[khome]" beginning-of-line
|
|
[[ -n "$terminfo[kend]" ]] && bindkey -M zed "$terminfo[kend]" end-of-line
|
|
|
|
# Fallback to well known code as terminfo might be wrong (often) sometimes
|
|
bindkey -M zed "^[[H" beginning-of-line
|
|
bindkey -M zed "^[[F" end-of-line
|
|
}
|
|
|
|
# Make zed-set-file-name available.
|
|
# Assume it's in fpath; there's no error at this point if it isn't
|
|
autoload -Uz zed-set-file-name
|
|
zle -N zed-set-file-name
|
|
fi
|
|
if (( bind )) || ! bindkey -M zed-vicmd >&/dev/null; then
|
|
bindkey -N zed-vicmd vicmd
|
|
|
|
bindkey -M zed-vicmd "ZZ" accept-line
|
|
[[ ${+terminfo} = 1 ]] && {
|
|
[[ -n "$terminfo[kpp]" ]] && bindkey -M zed-vicmd "$terminfo[kpp]" __zed_pg_up
|
|
[[ -n "$terminfo[knp]" ]] && bindkey -M zed-vicmd "$terminfo[knp]" __zed_pg_down
|
|
[[ -n "$terminfo[khome]" ]] && bindkey -M zed-vicmd "$terminfo[khome]" vi-beginning-of-line
|
|
[[ -n "$terminfo[kend]" ]] && bindkey -M zed-vicmd "$terminfo[kend]" vi-end-of-line
|
|
|
|
# Fallback to well known code as terminfo might be wrong (often) sometimes
|
|
bindkey -M zed-vicmd "^[[H" vi-beginning-of-line
|
|
bindkey -M zed-vicmd "^[[F" vi-end-of-line
|
|
}
|
|
fi
|
|
|
|
(( bind )) && return 0
|
|
|
|
# don't mangle !'s
|
|
setopt localoptions nobanghist
|
|
|
|
if ((fun)) then
|
|
var="$(functions $expand -- "$1")"
|
|
# If function is undefined but autoloadable, load it
|
|
if [[ $var = *\#\ undefined* ]] then
|
|
var="$(autoload +X "$1"; functions -- "$1")"
|
|
elif [[ -z $var ]] then
|
|
var="${(q-)1} () {
|
|
}"
|
|
fi
|
|
vared -M zed -m zed-vicmd -i __zed_init var && eval function "$var"
|
|
elif ((hist)) then
|
|
if [[ -n $1 ]]; then
|
|
{ fc -p -a "$1" ${2:-$({ wc -l <"$1" } 2>/dev/null)} || return }
|
|
let HISTSIZE++
|
|
print -s "" # Work around fc -p limitation
|
|
fi
|
|
# When editing the current shell history, the "zed -h" command is not
|
|
# itself included because the current event is not added to the ring
|
|
# until the next prompt is printed. This means "zed -h" is prepended
|
|
# to the result of the edit, because of the way "print -s" is defined.
|
|
var=( "${(@Oav)history}" )
|
|
IFS=$'\n' vared -M zed -m zed-vicmd -i __zed_init var
|
|
if (( ? )); then
|
|
[[ -n $1 ]] && unset HISTFILE
|
|
else
|
|
local HISTSIZE=0 savehist=$#var
|
|
fc -R /dev/null # Remove entries other than those added here
|
|
HISTSIZE=$savehist # Resets on function exit because local
|
|
[[ -n $1 ]] && SAVEHIST=$savehist # Resets via foregoing fc -a
|
|
for (( hist=1; hist <= savehist; hist++ ))
|
|
do print -rs -- "$var[hist]"
|
|
done
|
|
if [[ -n $zed_file_name ]]; then
|
|
fc -W "$zed_file_name"
|
|
[[ -n $1 ]] && unset HISTFILE
|
|
fi
|
|
# Note prepend effect when global HISTSIZE greater than $savehist.
|
|
# This does not affect file editing.
|
|
fi
|
|
else
|
|
zed_file_name="$1"
|
|
[[ -f $1 ]] && var="$(<"$1")"
|
|
while vared -M zed -m zed-vicmd -i __zed_init var
|
|
do
|
|
{
|
|
print -r -- "$var" >| "$zed_file_name"
|
|
} always {
|
|
(( TRY_BLOCK_ERROR = 0 ))
|
|
} && break
|
|
echo -n -e '\a'
|
|
done
|
|
fi
|
|
|
|
return 0
|
|
|
|
# End of zed
|