178 lines
5.4 KiB

# create a zkbd compatible hash;
# to add other keys to this hash, see: man 5 terminfo
typeset -g -A key
key[Home]="${terminfo[khome]}"
key[End]="${terminfo[kend]}"
key[Insert]="${terminfo[kich1]}"
key[Backspace]="${terminfo[kbs]}"
key[Delete]="${terminfo[kdch1]}"
key[Up]="${terminfo[kcuu1]}"
key[Down]="${terminfo[kcud1]}"
key[Left]="${terminfo[kcub1]}"
key[Right]="${terminfo[kcuf1]}"
key[PageUp]="${terminfo[kpp]}"
key[PageDown]="${terminfo[knp]}"
key[ShiftTab]="${terminfo[kcbt]}"
# setup key accordingly
[[ -n "${key[Home]}" ]] && bindkey -- "${key[Home]}" beginning-of-line
[[ -n "${key[End]}" ]] && bindkey -- "${key[End]}" end-of-line
[[ -n "${key[Insert]}" ]] && bindkey -- "${key[Insert]}" overwrite-mode
[[ -n "${key[Backspace]}" ]] && bindkey -- "${key[Backspace]}" backward-delete-char
[[ -n "${key[Delete]}" ]] && bindkey -- "${key[Delete]}" delete-char
[[ -n "${key[Up]}" ]] && bindkey -- "${key[Up]}" up-line-or-history
[[ -n "${key[Down]}" ]] && bindkey -- "${key[Down]}" down-line-or-history
[[ -n "${key[Left]}" ]] && bindkey -- "${key[Left]}" backward-char
[[ -n "${key[Right]}" ]] && bindkey -- "${key[Right]}" forward-char
[[ -n "${key[PageUp]}" ]] && bindkey -- "${key[PageUp]}" beginning-of-buffer-or-history
[[ -n "${key[PageDown]}" ]] && bindkey -- "${key[PageDown]}" end-of-buffer-or-history
[[ -n "${key[ShiftTab]}" ]] && bindkey -- "${key[ShiftTab]}" reverse-menu-complete
bindkey -M viins "^q" push-input
# Finally, make sure the terminal is in application mode, when zle is
# active. Only then are the values from $terminfo valid.
if (( ${+terminfo[smkx]} && ${+terminfo[rmkx]} )); then
autoload -Uz add-zle-hook-widget
function zle_application_mode_start { echoti smkx }
function zle_application_mode_stop { echoti rmkx }
add-zle-hook-widget -Uz zle-line-init zle_application_mode_start
add-zle-hook-widget -Uz zle-line-finish zle_application_mode_stop
fi
autoload -Uz up-line-or-beginning-search down-line-or-beginning-search
zle -N up-line-or-beginning-search
zle -N down-line-or-beginning-search
[[ -n "${key[Up]}" ]] && bindkey -- "${key[Up]}" up-line-or-beginning-search
[[ -n "${key[Down]}" ]] && bindkey -- "${key[Down]}" down-line-or-beginning-search
# By default, Ctrl+d will not close your shell if the command line is filled, this fixes it:
exit_zsh() { exit }
zle -N exit_zsh
bindkey '^D' exit_zsh
autoload -U edit-command-line
zle -N edit-command-line
bindkey '\C-x' edit-command-line
bindkey -M viins jj vi-cmd-mode
# Make CTRL-Z background things and unbackground them.
# Based off https://github.com/wincent/wincent/commit/30b502d811fbf4ca058db3a6f006aaecab68f6b7
function fg-bg() {
if [[ $#BUFFER -eq 0 ]]; then
local backgroundProgram="$(jobs | tail -n 1 | awk '{print $4}')"
case "$backgroundProgram" in
"nc"|"ncat"|"netcat"|"resize-netcat-listener"|"rnc")
# Make sure that /dev/tty is given to the stty command by doing </dev/tty
terminal-size-clip < /dev/tty
stty raw -echo < /dev/tty; fg
;;
*)
fg
;;
esac
else
zle push-input
fi
}
zle -N fg-bg
bindkey '^Z' fg-bg
rationalise-dot() {
if [[ $LBUFFER = *.. ]]; then
LBUFFER+=/..
else
LBUFFER+=.
fi
}
zle -N rationalise-dot
bindkey . rationalise-dot
make_current_word_directory(){
tokens=(${(z)LBUFFER})
local folder="${tokens[-1]}" output
if [ "${folder[1]}" = '"' ] || [ "${folder[1]}" = "'" ]; then
folder=${folder:1}
else
folder="${folder//\\ / }"
fi
folder="${~folder%/*}"
if [ -e "${~folder}" ]; then
zle -M "$folder already exists"
else
output="$(mkdir -p "${~folder}" 2>&1)"
if [ $? -eq 0 ]; then
zle -M "$folder created"
else
zle -M "$output"
fi
fi
}
zle -N make_current_word_directory
# Alt + m
bindkey '\em' make_current_word_directory
# This came about because I often find myself starting off with cd, tabbing and
# realising I would have been better off starting the command with vim
swap_command(){
# Each group should be seperated by a colon with each item in a group
# seperated by a space
local groups="cd vim ls:ping mtr:mysql mycli postgres pgcli:du:ncdu"
local tokens=(${(z)LBUFFER})
local cmd="${tokens[1]}"
local newindex=0
# If there is no command, return
[ "$cmd" = "" ] && return 0
# Find the first group group with the current command
local group="$(echo -en "$groups" | tr ':' '\n' | grep "$cmd" | head -n 1)"
# Turn the chosen group into an array, splitting on a space
local commands=(${(@s/ /)group})
# Find out where the current command is in the list
local currentindex=${commands[(ie)${cmd}]}
if [ "$currentindex" -gt "${#commands}" ]; then
# If the element isn't found in an array, zsh's search returns length + 1
return 0
elif [ "$currentindex" -eq "${#commands}" ]; then
newindex=1
else
newindex="$((currentindex + 1))"
fi
tokens[1]="${commands[$newindex]}"
LBUFFER="${tokens[@]}"
zle reset-prompt
return 0
}
zle -N swap_command
bindkey '\ec' swap_command
find_current_file(){
tokens=(${(z)LBUFFER})
local lastWord="${tokens[-1]}" filepath
# First assume I'm trying to edit a script. If it's in my path, use it
filepath="$(which "$lastWord")"
if [ "$?" -eq 0 ]; then
tokens[-1]="$filepath"
LBUFFER="${tokens[@]}"
return 0
fi
# Next try locate with an exact filename match
filepath="$(locate "*/$lastWord" | sed -n 1p)"
if [ "$?" -eq 0 ]; then
tokens[-1]="$filepath"
LBUFFER="${tokens[@]}"
return 0
fi
}
zle -N find_current_file
# ctrl + n
bindkey '^n' find_current_file