3 ## 16.2.1 Changing Directories
6 COMPLETE_IN_WORD GLOB_COMPLETE REC_EXACT
7 ## 16.2.3 Expansion and Globbing
8 EXTENDED_GLOB GLOB_DOTS GLOB_STAR_SHORT MAGIC_EQUAL_SUBST MARK_DIRS
11 # NOTE: NO_HIST_SAVE_BY_COPY to allow saving histfile if updating another
12 # user's histfile. this is for compatibility with zsu.
13 HIST_FCNTL_LOCK HIST_IGNORE_DUPS HIST_IGNORE_SPACE HIST_LEX_WORDS
14 HIST_NO_STORE HIST_REDUCE_BLANKS NO_HIST_SAVE_BY_COPY SHARE_HISTORY
15 ## 16.2.6 Input/Output
16 NO_CLOBBER CLOBBER_EMPTY CORRECT_ALL INTERACTIVE_COMMENTS
19 AUTO_CONTINUE LONG_LIST_JOBS
28 export EDITOR=${$(whence -p nvim vim vi micro nano emacs)[(f)1]}
29 export PAGER=${$(whence -p less micro nano more)[(f)1]:s/micro/& -readonly true -multiopen tab/:s/nano/& --view}
31 # XXX: deprecated in GNU
32 export GREP_OPTIONS=--color=auto
34 export HISTFILE=~/.histfile
36 export SAVEHIST=$HISTSIZE
38 export PYTHONSTARTUP=${XDG_CONFIG_HOME:-~/.config}/pythonrc
40 export VIMINIT='let$MYVIMRC=($XDG_CONFIG_HOME??($HOME."/.config"))."/vim/.vimrc"|execute"source"$MYVIMRC'
44 autoload -Uz zmathfunc && zmathfunc
47 zstyle ':vcs_info:*' enable git
48 #zstyle ':vcs_info:git*' check-for-changes true #too slow
49 zstyle ':vcs_info:git*:dotfiles' check-for-changes true
50 zstyle ':vcs_info:git*' check-for-staged-changes true
54 zstyle ':completion:*' auto-description '[arg] %d'
55 zstyle ':completion:*' expand suffix
56 zstyle ':completion:*' format '# %d'
57 zstyle ':completion:*' group-name ''
58 zstyle ':completion:*' ignore-parents parent
59 zstyle ':completion:*' insert-unambiguous false
60 zstyle ':completion:*' list-colors ${(s.:.)LS_COLORS}
61 zstyle ':completion:*' list-prompt '%B%i%b'
62 zstyle ':completion:*' list-suffixes true
63 zstyle ':completion:*' matcher-list '' 'm:{[:lower:]}={[:upper:]}' 'm:{[:lower:][:upper:]}={[:upper:][:lower:]} r:|[._-]=* r:|=*' 'm:{[:lower:][:upper:]}={[:upper:][:lower:]} l:|=* r:|=*'
64 zstyle ':completion:*' menu select=1
65 zstyle ':completion:*' original false
66 zstyle ':completion:*' select-prompt '%B%l%b'
67 zstyle ':completion:*' verbose true
69 cache=${XDG_CACHE_HOME:-~/.cache}/zsh
70 [[ -d $cache ]] || mkdir -p $cache
71 compinit -d $cache/.zcompdump
75 ## set up zkbd-style key array
76 if [[ ! -v _sev_force_zkbd && ( -v terminfo || -v termcap ) ]] {
77 # use application (keypad transmit) mode if the terminal supports it
78 # NOTE: we have to do this because termcap/terminfo keys are usually
79 # defined in application mode. terminals that do not define these
80 # capabilities are assumed to always be in application mode.
81 if [[ ( -v terminfo[smkx] && -v terminfo[rmkx] ) ||
82 ( -v termcap[ks] && -v terminfo[ke] ) ]] {
83 autoload -Uz add-zle-hook-widget
84 function _enter-application-mode {
85 [[ -v terminfo[smkx] ]] && echoti smkx || echotc ks
87 add-zle-hook-widget line-init _enter-application-mode
88 function _exit-application-mode {
89 [[ -v terminfo[rmkx] ]] && echoti rmkx || echotc ke
91 add-zle-hook-widget line-finish _exit-application-mode
92 trap _exit-application-mode EXIT
94 # pull keys from terminfo/termcap
95 # TODO: Menu and more Shift- modifiers
98 if [[ -v terminfo ]] {
100 newkey=(F1 kf1 F2 kf2 F3 kf3 F4 kf4 F5 kf5 F6 kf6 F7 kf7 F8 kf8 F9 kf9
101 F10 kf10 F11 kf11 F12 kf12
115 } elif [[ -v termcap ]] {
117 newkey=(F1 k1 F2 k2 F3 k3 F4 k4 F5 k5 F6 k6 F7 k7 F8 k8 F9 k9
118 F10 k\; F11 F1 F12 F2
133 for k v (${(kv)newkey}) {
134 key[$k]=${${(P)src}[$v]}
138 # use zkbd if termcap/terminfo unavailable
139 function find_keymap {
140 for f in ${ZDOTDIR:-~}/.zkbd/$TERM{-${DISPLAY:-$VENDOR-$OSTYPE},}
141 [[ -f $f ]] && keymap=$f && break
144 if [[ -z $keymap ]] {
145 if read -q "?Can't read terminfo. Add new zkbd keymap? [y/N]"; then
147 autoload -Uz zkbd && zkbd
153 if [[ -n $keymap ]] {
156 echo -E "Failed to source keymap file $keymap" >&2
158 unfunction find_keymap; unset keymap
161 ## load history search
162 autoload -Uz up-line-or-beginning-search down-line-or-beginning-search
163 zle -N up-line-or-beginning-search
164 zle -N down-line-or-beginning-search
166 ## bind keys in both viins and vicmd modes
167 # NOTE: cursor keys are bound to normal/raw/nontransmit mode strings by
168 # default, and keys like Home and End are not bound at all, so we rebind
169 # everything to cover all scenarios
173 Backspace 'backward-delete-char vi-backward-char'
174 Insert 'overwrite-mode vi-insert'
175 Home 'beginning-of-line'
176 PageUp 'up-history -'
179 PageDown 'down-history -'
180 Up 'up-line-or-beginning-search vi-up-line-or-history'
181 Down 'down-line-or-beginning-search vi-down-line-or-history'
187 if [[ -z $k ]] { continue }
190 if [[ $v[2] == - ]] {
191 # copy viins to vicmd verbatim
192 bindkey -a -- $k $v[1]
193 } elif (( $#v != 1 )) {
194 # set vicmd to any other value
195 bindkey -a -- $k $v[2]
197 # copy viins to vicmd and prepend vi- to it
198 bindkey -a -- $k vi-$v[1]
204 # bash-style reverse-search-history (i.e. reverse-i-search)
205 if [[ -v commands[fzf] ]] {
206 function _history-incremental-pattern-search-fzf {
207 l=(${(f)"$(fc -li -1 0 | fzf -emn 1 +s --preview-window=hidden ${BUFFER:+-q $BUFFER})"})
208 l=$(for x ("${l[@]}") { echo ${${=x}:3}; })
211 zle -N _history-incremental-pattern-search-fzf
212 bindkey '^R' _history-incremental-pattern-search-fzf
214 bindkey '^R' history-incremental-pattern-search-backward
219 alias rehash='_sev_setpath; rehash'
224 if [[ "$OSTYPE" =~ '^(free|net)bsd' ]] {
232 alias syncwatch='sync & watch -d grep -Fe Dirty: -e Writeback: /proc/meminfo'
233 if [[ -v commands[grep] ]] {
234 [[ -v commands[fgrep] ]] || alias fgrep='grep -F'
235 [[ -v commands[egrep] ]] || alias fgrep='grep -E'
237 for x (cat cmp diff grep test update) {
238 [[ -v commands[zutils-z$x] ]] || alias z$x=zutils-z$x
243 # zsh zmv with noglob wildcards
244 alias zm='noglob zmv -WiM'
245 alias zc='noglob zmv -WiC'
246 alias zl='noglob zmv -WiL'
247 alias sm='noglob zmv -Wip"sudo mv"'
248 alias sc='noglob zmv -Wip"sudo cp"'
249 alias sl='noglob zmv -Wip"sudo ln"'
250 if [[ "$OSTYPE" =~ '^freebsd' ]] {
251 # don't confirm if only a few files are deleted
254 # TODO: similar behavior for non-freebsd, or impliment in zsh
257 [[ -v commands[trash-put] ]] && alias t=trash-put
259 if [[ -v commands[pstree] && $commands[pstree]:A:t != busybox ]] {
260 # use pstree, but NOT busybox pstree because it kinda sucks
262 } elif [[ "$OSTYPE" =~ '^freebsd' ]] {
263 ps='ps -aSdfxwwouser=USR -ogroup=GRP -opid,nice=NI \
264 -o%cpu,%mem,tty,stat,start=START -oetime,command'
265 } elif [[ $commands[ps]:A:t == busybox ]] {
267 ps="ps -eouser='USR ' -ogroup='GRP ' \
268 -opid=' PID' -onice=' NI' -ovsz=' MEM' \
269 -otty,stat,etime,comm"
271 # XXX: untested, posix
272 # TODO: support gnu ps
273 ps='ps -eouser=USR -ogroup=GRP -opid,nice=NI \
274 -opcpu=CPU -ovsz=MEM -otty,stat,etime,comm'
276 if [[ "$(basename "$PAGER")" = "less" ]] {
279 ps="$ps | \"${PAGER:-more}\""
285 alias va='source bin/activate'
287 alias vu="python3 -mvenv --upgrade"
288 alias svu="sudo python3 -mvenv --upgrade"
292 alias gdh='git diff HEAD'
293 alias gds='git diff --staged'
295 alias ga.='git add .'
296 alias gai='git add -i'
297 alias gap='git add -p'
298 alias gc='git commit'
299 alias gca='git commit --amend'
303 alias gt='git tree' # from gitconfig
305 alias gst='git stash'
306 alias gsp='git stash pop'
309 \cd $(printf '../%.0s' {1..${1:-1}})
312 if [[ -v commands[zoxide] ]] {
313 # https://github.com/ajeetdsouza/zoxide/issues/513
314 eval "${$(zoxide init zsh):s#_files -/#_cd#}"
319 alias dfu='function {
320 pushd -q ${$(echo -E - ~/.zshenv):P:h:h}
322 git submodule init &&
323 git submodule sync &&
328 # zsh doesnt really handle sudo very well, so ignore it
329 alias sudo='nocorrect sudo'
331 alias dcp='sudo docker compose pull'
332 alias dcu='sudo docker compose up -d'
333 alias dcr='sudo docker compose restart'
334 alias dcl='sudo docker compose logs -f'
337 autoload -Uz add-zsh-hook
338 typeset -gi _sev_exectime
339 function sev_preexec {
340 # change terminal title to show command
341 print -n "\e]2;$(print -P '%#')${SSH_CLIENT+$USER@$HOST:}$1\e\\"
342 # save last exec time for bell
343 # XXX: does not run for blank cmdline
344 _sev_exectime=$SECONDS
345 # update gpg forward, to always have unique filename and avoid clashes
346 if [[ -v _GNUPG_SOCK_DEST_EXT ]] {
347 export _GNUPG_SOCK_DEST_EXT=$(date +%s).$RANDOM
348 export _GNUPG_SOCK_DEST=$_GNUPG_SOCK_DEST_BASE.$_GNUPG_SOCK_DEST_EXT
351 add-zsh-hook preexec sev_preexec
352 function sev_precmd {
353 # change terminal title
354 # TODO: update and send BEL when job status changes
355 print -Pn "\e]2;%(1j,%j,)%#${SSH_CLIENT+$USER@$HOST:}%~\e\\"
356 # bell if exec takes 5s
357 if (( SECONDS - _sev_exectime >= 5 )) print "\a"
358 # we could update vcs_info here, but let prompt take care of it
359 # if it doesn't use vcs, it can be ignored safely
361 add-zsh-hook precmd sev_precmd
363 # echo dir on cwd change
366 add-zsh-hook chpwd sev_chpwd
368 ### system-specific configs and aliases
373 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'
376 function s { sysctl hw.snd.default_unit${1:+\=$1} }
379 ## install port dependencies from pkg (like pkgsrc `bmake bin-install')
380 # XXX: should probably use package-depends where possible, breaks when
381 # port name is different to package name
382 # (eg. graphics/sdl20 == sdl2, devel/glib20 == glib2, etc)
386 sudo pkg install -AU $(make ${1}-depends-list |
387 sed 's_/usr/ports/_ _' | tr -d '\n')
389 *) echo "Usage: \`portpkg <build|run>' in a port directory"
397 ll /dev/mixer /dev/sound /dev/audio
400 for x in mixer sound audio; do
401 ln -sf /dev/$x"$1" /dev/$x
406 for x in $(mixerctl -a | grep 'outputs\.master'); do
411 mixerctl -w outputs.master"$2"="$1"
415 # TODO: test alsa/oss/sndio/portaudio/pulse/pipewire in order of importance
421 autoload -Uz promptinit && promptinit
427 ### load site-specific
428 load-site-dotfile zshrc