178 lines
5.4 KiB
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 |
|
|
|
|
|
|