X-Git-Url: https://git.sev.monster/~sev/dotfiles.git/blobdiff_plain/ff1a2414a9bf713665df692fab9cae9d2320b2f1..1ef46e010e2dc017f60cddfabf6e669a0469ef3f:/base/.zshrc diff --git a/base/.zshrc b/base/.zshrc index 2dd25ff..7890dad 100644 --- a/base/.zshrc +++ b/base/.zshrc @@ -1,122 +1,225 @@ ### options -setopt autocd extendedglob -unsetopt beep - -### history -setopt appendhistory -HISTFILE=~/.histfile -HISTSIZE=100 -SAVEHIST=100 +setopt NO_BEEP NO_CLOBBER +## cd +setopt AUTO_CD CDABLE_VARS +## completion +setopt GLOB_COMPLETE +## line +setopt EXTENDED_GLOB GLOB_DOTS MARK_DIRS NOMATCH NUMERIC_GLOB_SORT +## prompt +setopt PROMPT_SUBST +## jobs +setopt AUTO_CONTINUE +## history +setopt NO_HIST_SAVE_BY_COPY HIST_IGNORE_DUPS SHARE_HISTORY HIST_REDUCE_BLANKS ### keys +# TODO: investigate "^[[200~" bracketed-paste bindkey -v -KEYTIMEOUT=1 -## xorg -bindkey "^[[7~" beginning-of-line #Home -bindkey -a "^[[7~" beginning-of-line -bindkey "^[[8~" end-of-line #End -bindkey -a "^[[8~" end-of-line -bindkey "^[[3~" delete-char #Del -bindkey -a "^[[3~" delete-char -## history search +KEYTIMEOUT=10 +## populate key array +if (( $#terminfo == 0 )) { + # terminfo is not set or empty + function find_keymap { + for f in ${ZDOTDIR:-$HOME}/.zkbd/${TERM}{-${DISPLAY:-${VENDOR}-${OSTYPE}},} + [[ -f $f ]] && keymap=$f && break + } + find_keymap + if [[ -z $keymap ]] { + if read -q "?Can't read terminfo. Add new zkbd keymap? [y/N]"; then + echo + autoload -Uz zkbd && zkbd + unfunction zkbd + find_keymap + fi + echo + } + if [[ -n $keymap ]] { + source $keymap + } else { + echo "Failed to source keymap file $keymap" >&2 + } + unfunction find_keymap; unset keymap +} else { + # activate application mode for zle so terminfo keys work + # NOTE: don't do this for zkbd since application mode shouldn't have been + # enabled by zkbd when the keymap file was generated + if [[ -v terminfo[smkx] && -v terminfo[rmkx] ]] { + autoload -Uz add-zle-hook-widget + function _enter-application-mode { echoti smkx } + add-zle-hook-widget line-init _enter-application-mode + function _exit-application-mode { echoti rmkx } + add-zle-hook-widget line-finish _exit-application-mode + trap _exit-application-mode EXIT + } + # match zkbd hash as best we can to terminfo + typeset -gA key + key=(F1 kf1 F2 kf2 F3 kf3 F4 kf4 F5 kf5 F6 kf6 F7 kf7 F8 kf8 F9 kf9 + F10 kf10 F11 kf11 F12 kf12 + Backspace kbs + Backtab kcbt + Shift-Tab kcbt + Insert kich1 + Home khome + PageUp kpp + Delete kdch1 + End kend + PageDown knp + Up kcuu1 + Down kcud1 + Left kcub1 + Right kcuf1 + ) + for k v in ${(kv)key}; do + key[$k]=$terminfo[$v] + done; unset k v +} +## load history search 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 -### aliases -alias h="history -25" -alias j="jobs -l" -alias l="ls -AF" -alias ll="ls -lAFho" -## ps -local p="ps -aSdxwwouser=USR -ogroup=GRP -opid,nice=N \ - -o%cpu,%mem,tt,stat,start=START -oetime,command | ${PAGER:-more} -Se" -alias pa="$p" -alias spa="sudo $p" -unset p -## py venv -alias va="source bin/activate" -alias vd="deactivate" -## be paranoid +## bind keys in both viins and vicmd modes +typeset -A a +a=( + #key viins vicmd + Backspace 'backward-delete-char vi-backward-char' + Insert 'overwrite-mode vi-insert' + Home 'beginning-of-line' + PageUp 'up-history -' + Delete 'delete-char' + End 'end-of-line' + PageDown 'down-history -' + Up 'up-line-or-beginning-search vi-up-line-or-history' + Down 'down-line-or-beginning-search vi-down-line-or-history' + Left 'backward-char' + Right 'forward-char' +) +for k v in ${(kv)a}; do + k=$key[$k] + if [[ -z "$k" ]] { continue } + v=($=v) + bindkey -- $k $v[1] + if [[ $v[2] == '-' ]] { + # copy viins to vicmd verbatim + bindkey -a -- $k $v[1] + } elif (( $#v != 1 )) { + # set vicmd to any other value + bindkey -a -- $k $v[2] + } else { + # copy viins to vicmd and prepend vi- to it + bindkey -a -- $k vi-$v[1] + } +done +unset a k v + +### abbreviation aliases +alias h='history -25' +alias j='jobs -l' +alias l='ls -AF' +if [[ "$OSTYPE" =~ '^(free|net)bsd' ]] { + alias ll='ls -lAFho' +} else { + alias ll='ls -lAFh' +} +alias p="${PAGER:-more}" # TODO: make sure more is there or use alternate +alias e="${EDITOR:-vi}" # TODO: make sure vi is there or use alternate +alias se=sudoedit +# be paranoid alias cp='cp -ip' alias mv='mv -i' if [[ "$OSTYPE" =~ '^freebsd' ]] { # don't confirm if only a few files are deleted alias rm='rm -I' } else { + # TODO: similar behavior for non-freebsd, or impliment in zsh alias rm='rm -i' } - -### prompt -setopt prompt_subst -local f= -if [[ "$OSTYPE" = "netbsd" && ( "$TERM" = "wsvt25" || "$TERM" =~ '^vt' ) ]] { - # fix for wscons - f="%F{white}" +## py venv +alias va="source bin/activate" +alias vd="deactivate" +## ps +source ~/bin/.check-busybox +if which pstree >/dev/null 2>&1 && ! check-busybox pstree; then + # use pstree, but NOT busybox pstree because it kinda sucks + ps="pstree -wg3" +elif [[ "$OSTYPE" =~ '^freebsd' ]]; then + ps="ps -aSdfxwwouser=USR -ogroup=GRP -opid,nice=NI \ + -o%cpu,%mem,tty,stat,start=START -oetime,command" +elif check-busybox ps; then + # busybox compatible + ps="ps -eouser='USR ' -ogroup='GRP ' \ + -opid=' PID' -onice=' NI' -ovsz=' MEM' \ + -otty,stat,etime,comm" +else + # XXX: untested, posix + # TODO: support gnu ps + ps="ps -eouser=USR -ogroup=GRP -opid,nice=NI \ + -opcpu=CPU -ovsz=MEM -otty,stat,etime,comm" +fi +unfunction check-busybox +if [[ "$(basename "$PAGER")" = "less" ]] { + ps="$ps | less -S" } else { - f="%f" + ps="$ps | \"${PAGER:-more}\"" } -PROMPT='%F{$VICOL}%n'"$f"'@%F{$VICOL}%2m'"$f"'%(?../%F{red}$?'"$f"')%# ' -RPROMPT='%F{yellow}${vcs_info_msg_0_:-%~}'"$f"' %T' +alias pa="$ps" +alias spa="sudo $ps" +unset ps -## change color based on zle vi mode -function zle-line-init zle-keymap-select { - VICOL="${${KEYMAP:/vicmd/red}:/(main|viins)/green}" - zle reset-prompt +### specialized aliases +## go up directories +function up { + cd $(printf '../%.0s' {1..${1:-1}}) } -zle -N zle-line-init -zle -N zle-keymap-select - -## vcs -autoload -Uz vcs_info -zstyle ':vcs_info:*' enable git -zstyle ':vcs_info:git*' formats '%c%u%%F{green}%r/%b%%F{white}/%%F{yellow}%S%'"$f" -zstyle ':vcs_info:git*' actionformats '%%F{red}(%a)'"$f"' %c%u%%F{green}%r/%b'"$f"'/%%F{yellow}%S'"$f" -#zstyle ':vcs_info:git*' check-for-changes true #too slow -zstyle ':vcs_info:git*:.dotfiles' check-for-changes true -zstyle ':vcs_info:git*' check-for-staged-changes true -zstyle ':vcs_info:git*' stagedstr "%F{blue}+" -zstyle ':vcs_info:git*' unstagedstr "%F{red}*" - -unset f +## zoxide +[[ -v commands[zoxide] ]] && eval "$(zoxide init zsh)" ### hooks -local _exectime= -function precmd { +autoload -Uz add-zsh-hook +_sev_exectime= +function sev_preexec { + # change terminal title to show command + print -n "\e]2;$(print -P '%#')${SSH_CLIENT+$USER@$HOST:}$1\e\\" + # save last exec time for bell + # XXX: does not run for blank cmdline + _sev_exectime=$SECONDS + # update gpg forward, to always have unique filename and avoid clashes + if [[ -v _GNUPG_SOCK_DEST_EXT ]] { + export _GNUPG_SOCK_DEST_EXT=$(date +%s).$RANDOM + export _GNUPG_SOCK_DEST=$_GNUPG_SOCK_DEST_BASE.$_GNUPG_SOCK_DEST_EXT + } +} +add-zsh-hook preexec sev_preexec +function sev_precmd { # change terminal title + # TODO: update and send BEL when job status changes print -Pn "\e]2;%(1j,%j,)%#${SSH_CLIENT+$USER@$HOST:}%~\e\\" - # update vcs - vcs_info # bell if exec takes 5s - if ((SECONDS - _exectime >= 5)) print "\a" -} -function preexec { - # change terminal title to show command - print -Pnf "\e]2;%s\e\\" "%#${SSH_CLIENT+$USER@$HOST:}$1" - # save last exec time for bell - _exectime=$SECONDS + if (( SECONDS - _sev_exectime >= 5 )) print "\a" + # we could update vcs_info here, but let prompt take care of it + # if it doesn't use vcs, it can be ignored safely } -function chpwd { - l +add-zsh-hook precmd sev_precmd +function sev_chpwd { + # echo dir on cwd change + ls -AF } +add-zsh-hook chpwd sev_chpwd ### system-specific configs and aliases case "$OSTYPE"; in freebsd*) - ## vt binds - bindkey "^[[H" beginning-of-line #Home - bindkey -a "^[[H" beginning-of-line - bindkey "^[[F" end-of-line #End - bindkey -a "^[[F" end-of-line + # colors + export CLICOLOR= + export LS_COLORS='di=34:ln=35:so=32:pi=33:ex=31:bd=46;34:cd=43;34:su=41;30:sg=46;30:tw=42;30:ow=43;30' ## sound function s { sysctl hw.snd.default_unit${1:+\=$1} } alias vol mixer ## install port dependencies from pkg (like pkgsrc `bmake bin-install') - #XXX: should probably use package-depends where possible, breaks when + # XXX: should probably use package-depends where possible, breaks when # port name is different to package name # (eg. graphics/sdl20 == sdl2, devel/glib20 == glib2, etc) function portpkg { @@ -149,10 +252,23 @@ case "$OSTYPE"; in } mixerctl -w outputs.master"$2"="$1" };; + *) + ## sound + # TODO: test alsa/oss/sndio/portaudio/pulse in order of importance + function s {} + function vol {} esac -# The following lines were added by compinstall -zstyle ':completion:*' auto-description 'specify: %d' +### modules & styles +## vcs +zstyle ':vcs_info:*' enable git +#zstyle ':vcs_info:git*' check-for-changes true #too slow +zstyle ':vcs_info:git*:dotfiles' check-for-changes true +zstyle ':vcs_info:git*' check-for-staged-changes true +autoload -Uz vcs_info + +## compinit +zstyle ':completion:*' auto-description '[arg] %d' zstyle ':completion:*' expand suffix zstyle ':completion:*' format '# %d' zstyle ':completion:*' group-name '' @@ -166,10 +282,13 @@ zstyle ':completion:*' menu select=1 zstyle ':completion:*' original false zstyle ':completion:*' select-prompt '%B%l%b' zstyle ':completion:*' verbose true -zstyle :compinstall filename '/home/ds6/.zshrc' +autoload -Uz compinit && compinit + +### prompt +autoload -Uz promptinit && promptinit +prompt arrows -autoload -Uz compinit -compinit -# End of lines added by compinstall +### load site-specific +if [[ -f ~/.zshrc.local ]] { source ~/.zshrc.local } -# vim: ts=8:sts=4:sw=4:et:tw=79 +# vim: set et sts=4 sw=4 ts=8 tw=79 :