From: Junak Date: Wed, 11 Oct 2017 11:06:41 +0000 (+0300) Subject: Improve Zsh completion X-Git-Url: https://git.lizzy.rs/?a=commitdiff_plain;ds=sidebyside;h=469a9bf521698aa9f285e1151e67b6f5f9e05ac2;p=bspwm.git Improve Zsh completion --- diff --git a/contrib/zsh_completion b/contrib/zsh_completion index 04efa7e..079261b 100644 --- a/contrib/zsh_completion +++ b/contrib/zsh_completion @@ -1,23 +1,371 @@ #compdef bspc +_bspc_selector() { + [[ ${@[(r)--]} = '--' ]] && shift ${@[(i)--]} + local -a completions=() completions_display=() + local index=0 name id sel_type="$1" + shift 1 + case $sel_type in + (node) compset -P '*[#.:/@]' ;; + (desktop) compset -P '*[#.:]' ;; + (monitor) compset -P '*[#.]' ;; + (*) return 1 ;; + esac + case "$sel_type $IPREFIX" in + (desktop*:) + local ipfx=${${IPREFIX##*@}%:} + while do + if completions=("${(@f)$(bspc query --names -D -m ${ipfx} 2> /dev/null)}") ;then + until ((++index > $#completions)) do + completions[index]="^${index}:${completions[index]}" + done + completions+='focused' + _describe "${sel_type} selector" completions $@ -S '' -J ${sel_type} + break + else + completions=() + [ -n "$ipfx[(r)#]" ] && + ipfx="${ipfx#*#}" || + break + fi + done + ;| + (node*[^@/:.]) + bspc query -N -n .window 2> /dev/null | + while read id ;do + id=${id//:/\\:} + if which xdotool &> /dev/null ;then + name=$(xdotool getwindowname $id 2> /dev/null) + elif which xprop &> /dev/null ;then + name=$(xprop -id $id -notype WM_NAME 2> /dev/null) && + [[ "$name" = 'WM_NAME ='* ]] && + name="${${name#*\"}%\"*}" || + name="" + else + name="install xdotool or xprop to see window titles here" + fi + completions+="$id:$name" + done + ;| + ((desktop|monitor)*[^:.]) + local max_name_len=0 max_index_len i + local -a snames names ids + bspc query -${(U)sel_type[1]} 2> /dev/null | + while read id ;do + ((index++)) + name=$(bspc query --names -${(U)sel_type[1]} -${sel_type[1]} $id 2> /dev/null) + [[ "$name" == *[:.!]* ]] && + sel_name="%${name//:/\\:}" || + sel_name="$name" + ((max_name_len = $#sel_name > max_name_len ? $#sel_name : max_name_len)) + ids+="${id}" + snames+="${sel_name}" + names+="${name}" + done + max_index_len=$(($#index + 1)) + ((max_name_len >= max_name_len)) && ((max_name_len=max_name_len + 1)) + for ((i = 1 ; i <= $#ids ; i++)) ;do + (($#ids[i] <= max_name_len)) && + completions_display+="${(r($max_name_len+1))ids[i]}:${names[i]}" || + completions_display+="${ids[i]}:${names[i]}" + completions+="${ids[i]}:${names[i]}" + done + for ((i = 1 ; i <= $#ids ; i++)) ;do + completions+="${snames[i]}:${names[i]}" + completions_display+="${(r($max_name_len))snames[i]}:${names[i]}" + done + for ((i = 1 ; i <= $#ids ; i++)) ;do + completions+="^${i}:${names[i]}" + completions_display+="^${i}:${names[i]}" + done + ;| + (node*('@'(*':'|*'/'|))) + _describe 'node path' jump -S '/' -r "#. ${quote}" -J nodes + ;| + (node*'/') + ;; + (*'.') + _bspc_prefix '!' "${sel_type} modifiers" ${sel_type}_mod $@ -J ${sel_type}_mod + ;| + ((desktop|monitor)*@) + ;& + (*[^:.@/]) + if (( $#completions_display)) ;then + _describe "${sel_type} selector" ${sel_type}_desc $@ -S '.' -r ". \n:#${quote}\-" -J ${sel_type} \ + -- completions_display completions $@ -S '.' -r ". \n:#${quote}\-" -J ${sel_type} + else + _describe "${sel_type} selector" ${sel_type}_desc $@ -S '.' -r ". \n:#${quote}\-" -J ${sel_type} \ + -- completions $@ -S '.' -r ". \n:#${quote}\-" -J ${sel_type} + fi + ;| + (node*@*) + _bspc_selector desktop -S ':' -qr ".#\-\n ${quote}" + ;; + (desktop*) + _bspc_selector monitor -S ':' -r ".#\-\n ${quote}" + ;; + esac +} + +_bspc_prefix(){ + [[ ${@[(r)--]} = '--' ]] && shift ${@[(i)--]} + [[ "$PREFIX[1]" == "$1" ]] && + local a="-n" b || + local b="-n" a + _describe $@[2,-1] $a -- $@[3,-1] $b -p "$1" +} + +_bspc_query_names() { + [[ ${@[(r)--]} = '--' ]] && shift ${@[(i)--]} + local -a items=("${(@f)$(bspc query $2 --names 2> /dev/null)}") || + return + local c + for c in '\' ':' '[' '(' '*' + items=("${(@)items//$c/\\$c}") + _values -w "$1" "${items[@]}" +} + _bspc() { - local -a commands settings - commands=('node' 'desktop' 'monitor' 'query' 'rule' 'wm' 'subscribe' 'config' 'quit') - settings=('external_rules_command' 'status_prefix' 'normal_border_color' 'active_border_color' 'focused_border_color' 'presel_feedback_color' 'border_width' 'window_gap' 'top_padding' 'right_padding' 'bottom_padding' 'left_padding' 'split_ratio' 'initial_polarity' 'directional_focus_tightness' 'borderless_monocle' 'gapless_monocle' 'single_monocle' 'pointer_motion_interval' 'pointer_modifier' 'pointer_action1' 'pointer_action2' 'pointer_action3' 'click_to_focus' 'swallow_first_click' 'focus_follows_pointer' 'pointer_follows_focus' 'pointer_follows_monitor' 'ignore_ewmh_focus' 'center_pseudo_tiled' 'honor_size_hints' 'remove_disabled_monitors' 'remove_unplugged_monitors' 'merge_overlapping_monitors') - if (( CURRENT == 2 )) ; then - _values 'command' "$commands[@]" - elif (( CURRENT == 3 )) ; then - case $words[2] in - config) - _values 'setting' "$settings[@]" - ;; - *) - return 1 - ;; - esac - else - return 1 + local -a commands \ + commands=(node desktop monitor query rule wm subscribe config quit) \ + node_state=(tiled pseudo_tiled floating fullscreen) \ + flag=(hidden sticky private locked urgent) \ + layer=(below normal above) \ + dir=(north west south east) \ + cycle_dir=(next prev) + local -a jump=($dir first second brother parent 1 2) \ + node_desc=($dir $cycle_dir last older newer focused pointed biggest) \ + node_mod=($node_state $flag $layer focused automatic local \ + active leaf window same_class descendant_of ancestor_of) \ + desktop_desc=($cycle_dir last older newer focused) \ + desktop_mod=(focused occupied local urgent) \ + monitor_desc=($dir $cycle_dir primary last older newer focused pointed) \ + monitor_mod=(focused occupied) \ + presel_dir=($dir cancel) + local quote="${compstate[quote]}" context state state_descr line + typeset -A opt_args + + compset -n 2 + compset -S "${quote}" + + if ((CURRENT==1)) ;then + _describe 'command or domain' commands + return fi + + case $words[1] in + (node) + ((CURRENT==2)) && _bspc_selector node + ((CURRENT>2)) && [[ "$words[2]" != '-'* ]] && compset -n 2 + while do + _arguments -C \ + '*'{-a,--activate}'[Activate the selected or given node]::node selector:_bspc_selector -- node'\ + '*'{-B,--balance}'[Adjust the split ratios of the tree rooted at the selected node so that all windows occupy the same area]'\ + '*'{-C,--circulate}'[Circulate the windows of the tree rooted at the selected node]:direction:(forward backward)'\ + '*'{-c,--close}'[Close the windows rooted at the selected node]'\ + '*'{-d,--to-desktop}'[Send the selected node to the given desktop]:desktop selector:_bspc_selector -- desktop'\ + '*'{-E,--equalize}'[Reset the split ratios of the tree rooted at the selected node to their default value]'\ + '*'{-F,--flip}'[Flip the the tree rooted at selected node]: :(horizontal vertical)'\ + '*'{-f,--focus}'[Focus the selected or given node]::node selector:_bspc_selector -- node'\ + '*'{-g,--flag}'[Set or toggle the given flag for the selected node]: :->flag'\ + '*'{-i,--insert-receptacle}'[Insert a receptacle node at the selected node]'\ + '*'{-k,--kill}'[Kill the windows rooted at the selected node]'\ + '*'{-l,--layer}"[Set the stacking layer of the selected window]:stacking layer:($layer)"\ + '*'{-m,--to-monitor}'[Send the selected node to the given monitor]:monitor selector:_bspc_selector -- monitor'\ + '*'{-n,--to-node}'[Transplant the selected node to the given node]:node selector:_bspc_selector -- node'\ + '*'{-o,--presel-ratio}'[Set the splitting ratio of the preselection area]:preselect ratio: ( )'\ + '*'{-p,--presel-dir}'[Preselect the splitting area of the selected node or cancel the preselection]: :_bspc_prefix -- "~" preselect presel_dir'\ + '*'{-r,--ratio}'[Set the splitting ratio of the selected node (0 < ratio < 1)]: :( )'\ + '*'{-R,--rotate}'[Rotate the tree rooted at the selected node]:angle:(90 270 180)'\ + '*'{-s,--swap}'[Swap the selected node with the given node]:node selector:_bspc_selector -- node'\ + '*'{-t,--state}'[Set the state of the selected window]: :_bspc_prefix -- "~" "node state" node_state '\ + '*'{-v,--move}'[Move the selected window by dx pixels horizontally and dy pixels vertically]:dx dy:'\ + '*'{-z,--resize}'[Resize the selected window by moving the given handle by dx pixels horizontally and dy pixels vertically]:*::: :->resize' + [ "$state" = flag ] && _values 'flag' "${flag[@]:#urgent}::set flag:(on off)" + [ "$state" = resize ] || break + ((CURRENT == 1)) && + _values 'handle' resize_handle {top,bottom}{,_left,_right} left right + ((CURRENT <= 3)) && break + compset -n 3 + done + ;; + (desktop) + ((CURRENT==2)) && _bspc_selector desktop + ((CURRENT>2)) && [[ "$words[2]" != '-'* ]] && compset -n 2 + _arguments \ + '*'{-a,--activate}'[Activate the selected or given desktop]:: :_bspc_selector -- desktop'\ + '*'{-b,--bubble}"[Bubble the selected desktop in the given direction]:direction:($cycle_dir)"\ + '*'{-f,--focus}'[Focus the selected or given desktop]:: :_bspc_selector -- desktop'\ + '*'{-l,--layout}"[Set or cycle the layout of the selected desktop]:desktop layout:($cycle_dir monocle tiled)"\ + '*'{-m,--to-monitor}'[Send the selected desktop to the given monitor]: :_bspc_selector -- monitor'\ + '*'{-n,--rename}'[Rename the selected desktop]:desktop name:( )'\ + '*'{-r,--remove}'[Remove the selected desktop]'\ + '*'{-s,--swap}'[Swap the selected desktop with the given desktop]: :_bspc_selector -- desktop' + ;; + (monitor) + ((CURRENT==2)) && _bspc_selector monitor + ((CURRENT>2)) && [[ "$words[2]" != '-'* ]] && compset -n 2 + _arguments \ + '*'{-a,--add-desktops}'[Create desktops with the given names in the selected monitor]:*:add desktops:( )'\ + '*'{-d,--reset-desktops}'[Rename, add or remove desktops]:*: :->desktops'\ + '*'{-f,--focus}'[Focus the selected or given monitor]:: :_bspc_selector -- monitor'\ + '*'{-g,--rectangle}'[Set the rectangle of the selected monitor]:WxH+X+Y:( )'\ + '*'{-n,--rename}'[Rename the selected monitor]: :( )'\ + '*'{-o,--reorder-desktops}'[Reorder the desktops of the selected monitor]:*:reorder desktops:_bspc_query_names -- desktops -D'\ + '*'{-r,--remove}'[Remove the selected monitor]'\ + '*'{-s,--swap}'[Swap the selected monitor with the given monitor]: :_bspc_selector -- monitor' + ;; + (query) + local -a cmds_no_names=('-T' '--tree' '-N' '--nodes') + local -a cmds=($cmds_no_names '-D' '--desktops' '-M' '--monitors') + _arguments \ + '*'{-d,--desktop}'[Constrain matches to the selected desktop]: :_bspc_selector -- desktop'\ + '*'{-m,--monitor}'[Constrain matches to the selected monitor]: :_bspc_selector -- monitor'\ + '*'{-n,--node}'[Constrain matches to the selected node]: :_bspc_selector -- node'\ + "($cmds_no_names --names)--names[Print names instead of IDs. Can only be used with -M and -D]"\ + "($cmds --names)"{-N,--nodes}'[List the IDs of the matching nodes]'\ + "($cmds --names)"{-T,--tree}'[Print a JSON representation of the matching item]'\ + "($cmds)"{-D,--desktops}'[List the IDs (or names) of the matching desktops]'\ + "($cmds)"{-M,--monitors}'[List the IDs (or names) of the matching monitors]' + ;; + (wm) + _arguments \ + '*'{-d,--dump-state}'[Dump the current world state on standard output]'\ + '*'{-l,--load-state}'[Load a world state from the given file]:load state from file:_files'\ + '*'{-a,--add-monitor}'[Add a monitor for the given name and rectangle]:add monitor:( )'\ + '*'{-O,--reorder-monitors}'[Reorder the list of monitors to match the given order]:*: :_bspc_query_names -- monitors -M'\ + '*'{-o,--adopt-orphans}'[Manage all the unmanaged windows remaining from a previous session]'\ + '*'{-h,--record-history}'[Enable or disable the recording of node focus history]:history:(on off)'\ + '*'{-g,--get-status}'[Print the current status information]' + ;; + (subscribe) + if [[ "$words[CURRENT-1]" != (-c|--count) ]] ;then + _values -w "options" \ + '(-f --fifo)'{-f,--fifo}'[Print a path to a FIFO from which events can be read and return]'\ + '(-c --count)'{-c,--count}'[Stop the corresponding bspc process after having received specified count of events]' + _values -w -S "_" events all report pointer_action \ + "monitor:: :(add rename remove swap focus geometry)"\ + "desktop:: :(add rename remove swap transfer focus activate layout)"\ + "node:: :(add remove swap transfer focus activate presel stack geometry state flag layer)" + fi + ;; + (rule) + local -a completions by_index + local index=0 target settings class id instance json + _arguments -C \ + {-a,--add}'[Create a new rule]:*: :->add'\ + {-r,--remove}'[Remove the given rules]:*: :->remove'\ + '(-l --list)'{-l,--list}'[List the rules]' + compset -N "-([ar]|-add|-remove)" + case $state$CURRENT in + (add1) + compset -P '*:' + bspc query -N -n '.window' 2> /dev/null | + while read id ; do + json=$(bspc query -T -n $id 2>/dev/null) || continue + [[ "$json[1]" = '{' ]] || continue + class=${${json##*\"className\":\"}%%\",\"*} + instance=${${json##*\"instanceName\":\"}%%\",\"*} + [[ "$class[1]" != '{' && "$instance[1]" != '{' ]] || continue + if [ -n "$IPREFIX" ] ;then + [[ "$IPREFIX" == ("${class}:"|('\'|)'*:') ]] && + completions[(r)$instance]=$instance + else + class=${class%%:/\\:} + completions[(r)$class]="$class" + fi + done + ;; + (add*) + _values -w 'add rule' {border,focus,follow,manage,center}': :(on off)'\ + '(--one-shot)-o'\ + '(-o)--one-shot'\ + 'monitor: :_bspc_selector -- monitor'\ + 'desktop: :_bspc_selector -- desktop'\ + 'node: :_bspc_selector -- node'\ + 'rectangle: :( )'\ + 'split_ratio:split ratio:( )'\ + "split_dir:split direction:(${dir})"\ + "state:state:(${node_state})"\ + "${flag[@]:#urgent}:set flag:(on off)"\ + "layer:layer:(${layer})" + return + ;; + (remove*) + compset -P '*:' + bspc rule -l 2> /dev/null | + while IFS=" " read target settings ;do + by_index+="^$((index++)):${target} ${settings}" + if [ -n "$IPREFIX" ] ;then + completions+="${target#*:}" + else + completions+="${target%:*}" + fi + done + [[ -z "$IPREFIX" ]] && + _describe 'remove rule by position' '(head tail)' -J by_index -- by_index -J by_index + ;; + (*) + return + ;; + esac + completions[(r)\*]=* + [ -n "$IPREFIX" ] && + _describe 'match window instance' completions || + _describe 'match window class' completions -q -S ':' + ;; + (config) + local -a look behaviour input + # Boolean settings are identified by index! + look=(borderless_monocle gapless_monocle {normal,active,focused}_border_color {top,right,bottom,left}_padding presel_feedback_color border_width window_gap) + behaviour=(single_monocle ignore_ewmh_focus center_pseudo_tiled honor_size_hints remove_disabled_monitors remove_unplugged_monitors merge_overlapping_monitors status_prefix external_rules_command split_ratio initial_polarity directional_focus_tightness status_prefix) + input=({swallow_first_click,focus_follows_pointer,pointer_follows_{focus,monitor}} click_to_focus pointer_motion_interval pointer_modifier pointer_action{1,2,3} ) + if [[ "$CURRENT" == (2|3) ]];then + _arguments \ + '-d[Set settings for the selected desktop]: :_bspc_selector -- desktop'\ + '-m[Set settings for the selected monitor]: :_bspc_selector -- monitor'\ + '-n[Set settings for the selected node]: :_bspc_selector -- node' + fi + if [[ "${words[2]}" == -* ]] ;then + (( CURRENT == 3 )) && return + if (( CURRENT > 3 )) ;then + ((CURRENT-=$#words)) + words=('config' "${(@)words[4,-1]}") + ((CURRENT+=$#words)) + fi + fi + if ((CURRENT==2)) ;then + _describe 'look' look -J look + _describe 'input' input -J input + _describe 'behaviour' behaviour -J behaviour + elif ((CURRENT==3)) ;then + setting=$words[2] + case $setting in + (initial_polarity) + _values "set $setting" first_child second_child + ;; + (pointer_action(1|2|3)) + _values "set $setting" move resize_side resize_corner focus none + ;; + (pointer_modifier) + _values "set $setting" shift control lock mod1 mod2 mod3 mod4 mod5 + ;; + (directional_focus_tightness) + _values "set $setting" low high + ;; + (click_to_focus) + _values "set $setting" any button1 button2 button3 none + ;; + (*) + ((look[(i)$setting] <= 2)) || + ((behaviour[(i)$setting] <= 7)) || + ((input[(i)$setting] <= 4)) && + _values "set $setting" true false + ;; + esac + fi + ;; + esac } _bspc "$@"