diff options
Diffstat (limited to 'plugins/zsh-navigation-tools')
32 files changed, 3576 insertions, 0 deletions
diff --git a/plugins/zsh-navigation-tools/.config/znt/README.txt b/plugins/zsh-navigation-tools/.config/znt/README.txt new file mode 100644 index 000000000..c3d6c821a --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/README.txt @@ -0,0 +1 @@ +These are skeletons, configuration is read from ~/.config/znt/* diff --git a/plugins/zsh-navigation-tools/.config/znt/n-aliases.conf b/plugins/zsh-navigation-tools/.config/znt/n-aliases.conf new file mode 100644 index 000000000..047b017eb --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-aliases.conf @@ -0,0 +1,33 @@ +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_aliases_active_text +active_text=${znt_aliases_active_text:-$active_text} + +# 2. znt_aliases_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_aliases_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 3. znt_aliases_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_aliases_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 4. znt_aliases_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_aliases_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 5. znt_aliases_keywords (array) +if (( ${+znt_aliases_keywords} )); then +    keywords=( "${znt_aliases_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-cd.conf b/plugins/zsh-navigation-tools/.config/znt/n-cd.conf new file mode 100644 index 000000000..8271e5d98 --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-cd.conf @@ -0,0 +1,68 @@ +# Hotlist +# Can use e.g. $ZSH_VERSION, e.g. /usr/share/zsh/$ZSH_VERSION/functions +local -a hotlist +hotlist=( +    ~/.config/znt +    /usr/share/zsh/site-functions +    /usr/share/zsh +    /usr/local/share/zsh/site-functions +    /usr/local/share/zsh +    /usr/local/bin +    /usr/lib +) + +# Suppress adding (to directory stack) directories visited by n-cd +# Value 0 is the default (directories will be added to dirstack) +local NCD_DONT_PUSHD=0 + +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + +# Colorize last segments of the paths +# (#s) is ^, (#e) is $, # is *, ## is + (comparing to regex) +local NLIST_COLORING_PATTERN="[a-zA-Z0-9 ._-]##/#(#e)" +local NLIST_COLORING_COLOR=$'\x1b[00;33m' + +# Search keywords, iterated with F2 or Ctrl-X or Ctrl-/ +local -a keywords +keywords=( "zsh" "local" "git" ) + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_cd_hotlist (array) +if (( ${+znt_cd_hotlist} )); then +    hotlist=( "${znt_cd_hotlist[@]}" ) +fi + +# 2. znt_cd_dont_pushd +NCD_DONT_PUSHD=${znt_cd_dont_pushd:-$NCD_DONT_PUSHD} + +# 3. znt_cd_active_text +active_text=${znt_cd_active_text:-$active_text} + +# 4. znt_cd_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_cd_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 5. znt_cd_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_cd_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 6. znt_cd_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_cd_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 7. znt_cd_keywords (array) +if (( ${+znt_cd_keywords} )); then +    keywords=( "${znt_cd_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-env.conf b/plugins/zsh-navigation-tools/.config/znt/n-env.conf new file mode 100644 index 000000000..85dda05ed --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-env.conf @@ -0,0 +1,38 @@ +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + +# (#s) is ^, (#e) is $, # is *, ## is + (comparing to regex) +local NLIST_COLORING_PATTERN="[a-zA-Z0-9_]##" +local NLIST_COLORING_MATCH_MULTIPLE=0 + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_env_active_text +active_text=${znt_env_active_text:-$active_text} + +# 2. znt_env_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_env_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 3. znt_env_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_env_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 4. znt_env_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_env_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 5. znt_env_keywords (array) +if (( ${+znt_env_keywords} )); then +    keywords=( "${znt_env_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-functions.conf b/plugins/zsh-navigation-tools/.config/znt/n-functions.conf new file mode 100644 index 000000000..d032e2788 --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-functions.conf @@ -0,0 +1,41 @@ +# Which editor to use, zed or vared +# vared is the default +local feditor="zed" +# local feditor="vared" + +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_functions_feditor +feditor=${znt_functions_feditor:-$feditor} + +# 2. znt_functions_active_text +active_text=${znt_functions_active_text:-$active_text} + +# 3. znt_functions_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_functions_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 4. znt_functions_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_functions_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 5. znt_functions_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_functions_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 6. znt_functions_keywords (array) +if (( ${+znt_functions_keywords} )); then +    keywords=( "${znt_functions_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-history.conf b/plugins/zsh-navigation-tools/.config/znt/n-history.conf new file mode 100644 index 000000000..d7c69e154 --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-history.conf @@ -0,0 +1,43 @@ +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=underline + +# Highlight a few keywords +local NLIST_COLORING_PATTERN="(while|for |sudo|make|(#s)git|vim(#e)|vim |emacs(#e)|emacs )" +local NLIST_COLORING_COLOR=$'\x1b[00;33m' +local NLIST_COLORING_MATCH_MULTIPLE=1 + +# Search keywords, iterated with F2 or Ctrl-X or Ctrl-/ +local -a keywords +keywords=( "git" "vim" "mplayer" ) + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_history_active_text +active_text=${znt_history_active_text:-$active_text} + +# 2. znt_history_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_history_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 3. znt_history_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_history_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 4. znt_history_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_history_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 5. znt_history_keywords (array) +if (( ${+znt_history_keywords} )); then +    keywords=( "${znt_history_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-kill.conf b/plugins/zsh-navigation-tools/.config/znt/n-kill.conf new file mode 100644 index 000000000..59807b2c3 --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-kill.conf @@ -0,0 +1,46 @@ +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + +# Colorize first number column and last path segment +# This doesn't cover scripts named "[0-9]## *", which should be very rare +# (#s) is ^, (#e) is $, # is *, ## is + (comparing to regex) +# | is alternative, but only in () +local NLIST_COLORING_PATTERN="((#s) #[0-9]## |[[][^]]#](#e)|[^ 0-9/?\\\\][^/\\\\]#(#e))" +local NLIST_COLORING_COLOR=$'\x1b[00;33m' +local NLIST_COLORING_MATCH_MULTIPLE=1 + +# Search keywords, iterated with F2 or Ctrl-X or Ctrl-/ +local -a keywords +keywords=( "vim" "git" "git commit" "firefox" "mplayer" ) + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_kill_active_text +active_text=${znt_kill_active_text:-$active_text} + +# 2. znt_kill_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_kill_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 3. znt_kill_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_kill_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 4. znt_kill_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_kill_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 5. znt_kill_keywords (array) +if (( ${+znt_kill_keywords} )); then +    keywords=( "${znt_kill_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-list.conf b/plugins/zsh-navigation-tools/.config/znt/n-list.conf new file mode 100644 index 000000000..096104df6 --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-list.conf @@ -0,0 +1,55 @@ +# Should the list (text, borders) be drawn in bold +# Value 1 is the default +local bold=0 + +# Main color pair (foreground/background) +local colorpair="white/black" + +# Should draw the border? +local border=0 + +# Combinations of colors to try out with Ctrl-T and Ctrl-G +# The last number is the bold option, 0 or 1 +local -a themes +themes=( "white/black/1" "green/black/0" "green/black/1" "white/blue/0" "white/blue/1" +         "magenta/black/0" "magenta/black/1" ) + +# Uncomment when using new (>5.2) Zsh version that supports 256 colors in zcurses +#themes=( "white/17/0" "10/17/1" "white/24/1" "white/22/0" "white/22/1" "white/25/0" "white/25/1" +#         "white/59/0" "white/59/1" "white/60/0" "white/60/1" "white/61/0" "white/61/1" "black/65/0" +#         "black/244/0"  ) + +# When 1, pressing enter when in search mode will not +# only leave the mode, but also do selection and leave +# n-history +local NLIST_INSTANT_SELECT=0 + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_list_bold +bold=${znt_list_bold:-$bold} + +# 2. znt_list_colorpair +colorpair=${znt_list_colorpair:-$colorpair} + +# 3. znt_list_border +border=${znt_list_border:-$border} + +# 4. znt_list_themes (array) +if (( ${+znt_list_themes} )); then +    themes=( "${znt_list_themes[@]}" ) +fi + +# 5. znt_list_instant_select +NLIST_INSTANT_SELECT=${znt_list_instant_select:-$NLIST_INSTANT_SELECT} + diff --git a/plugins/zsh-navigation-tools/.config/znt/n-options.conf b/plugins/zsh-navigation-tools/.config/znt/n-options.conf new file mode 100644 index 000000000..29c5e089a --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-options.conf @@ -0,0 +1,34 @@ +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_options_active_text +active_text=${znt_options_active_text:-$active_text} + +# 2. znt_options_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_options_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 3. znt_options_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_options_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 4. znt_options_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_options_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 5. znt_options_keywords (array) +if (( ${+znt_options_keywords} )); then +    keywords=( "${znt_options_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/.config/znt/n-panelize.conf b/plugins/zsh-navigation-tools/.config/znt/n-panelize.conf new file mode 100644 index 000000000..58ce1b620 --- /dev/null +++ b/plugins/zsh-navigation-tools/.config/znt/n-panelize.conf @@ -0,0 +1,34 @@ +# How should be current element of the list drawn. Possible values: reverse, +# underline. +# On Linux virtual terminal this will be enforced to reverse (because of poor +# underline support on that terminal). The same for screen/tmux. +local active_text=reverse + + + + + + + +# +# Zshrc integration. All below variables can +# be set in zshrc and will have precedence +# over config files +# + +# 1. znt_panelize_active_text +active_text=${znt_panelize_active_text:-$active_text} + +# 2. znt_panelize_nlist_coloring_pattern +NLIST_COLORING_PATTERN=${znt_panelize_nlist_coloring_pattern:-$NLIST_COLORING_PATTERN} + +# 3. znt_panelize_nlist_coloring_color +NLIST_COLORING_COLOR=${znt_panelize_nlist_coloring_color:-$NLIST_COLORING_COLOR} + +# 4. znt_panelize_nlist_coloring_match_multiple +NLIST_COLORING_MATCH_MULTIPLE=${znt_panelize_nlist_coloring_match_multiple:-$NLIST_COLORING_MATCH_MULTIPLE} + +# 5. znt_panelize_keywords (array) +if (( ${+znt_panelize_keywords} )); then +    keywords=( "${znt_panelize_keywords[@]}" ) +fi diff --git a/plugins/zsh-navigation-tools/LICENSE b/plugins/zsh-navigation-tools/LICENSE new file mode 100644 index 000000000..4ee028112 --- /dev/null +++ b/plugins/zsh-navigation-tools/LICENSE @@ -0,0 +1,700 @@ +This software is dual-licensed under MIT and GPLv3. + +MIT License +----------- + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. + +GPLv3  License +-------------- + +                    GNU GENERAL PUBLIC LICENSE +                       Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + +                            Preamble + +  The GNU General Public License is a free, copyleft license for +software and other kinds of works. + +  The licenses for most software and other practical works are designed +to take away your freedom to share and change the works.  By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users.  We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors.  You can apply it to +your programs, too. + +  When we speak of free software, we are referring to freedom, not +price.  Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + +  To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights.  Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + +  For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received.  You must make sure that they, too, receive +or can get the source code.  And you must show them these terms so they +know their rights. + +  Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + +  For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software.  For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + +  Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so.  This is fundamentally incompatible with the aim of +protecting users' freedom to change the software.  The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable.  Therefore, we +have designed this version of the GPL to prohibit the practice for those +products.  If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + +  Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary.  To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + +  The precise terms and conditions for copying, distribution and +modification follow. + +                       TERMS AND CONDITIONS + +  0. Definitions. + +  "This License" refers to version 3 of the GNU General Public License. + +  "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + +  "The Program" refers to any copyrightable work licensed under this +License.  Each licensee is addressed as "you".  "Licensees" and +"recipients" may be individuals or organizations. + +  To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy.  The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + +  A "covered work" means either the unmodified Program or a work based +on the Program. + +  To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy.  Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + +  To "convey" a work means any kind of propagation that enables other +parties to make or receive copies.  Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + +  An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License.  If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + +  1. Source Code. + +  The "source code" for a work means the preferred form of the work +for making modifications to it.  "Object code" means any non-source +form of a work. + +  A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + +  The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form.  A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + +  The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities.  However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work.  For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + +  The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + +  The Corresponding Source for a work in source code form is that +same work. + +  2. Basic Permissions. + +  All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met.  This License explicitly affirms your unlimited +permission to run the unmodified Program.  The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work.  This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + +  You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force.  You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright.  Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + +  Conveying under any other circumstances is permitted solely under +the conditions stated below.  Sublicensing is not allowed; section 10 +makes it unnecessary. + +  3. Protecting Users' Legal Rights From Anti-Circumvention Law. + +  No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + +  When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + +  4. Conveying Verbatim Copies. + +  You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + +  You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + +  5. Conveying Modified Source Versions. + +  You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + +    a) The work must carry prominent notices stating that you modified +    it, and giving a relevant date. + +    b) The work must carry prominent notices stating that it is +    released under this License and any conditions added under section +    7.  This requirement modifies the requirement in section 4 to +    "keep intact all notices". + +    c) You must license the entire work, as a whole, under this +    License to anyone who comes into possession of a copy.  This +    License will therefore apply, along with any applicable section 7 +    additional terms, to the whole of the work, and all its parts, +    regardless of how they are packaged.  This License gives no +    permission to license the work in any other way, but it does not +    invalidate such permission if you have separately received it. + +    d) If the work has interactive user interfaces, each must display +    Appropriate Legal Notices; however, if the Program has interactive +    interfaces that do not display Appropriate Legal Notices, your +    work need not make them do so. + +  A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit.  Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + +  6. Conveying Non-Source Forms. + +  You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + +    a) Convey the object code in, or embodied in, a physical product +    (including a physical distribution medium), accompanied by the +    Corresponding Source fixed on a durable physical medium +    customarily used for software interchange. + +    b) Convey the object code in, or embodied in, a physical product +    (including a physical distribution medium), accompanied by a +    written offer, valid for at least three years and valid for as +    long as you offer spare parts or customer support for that product +    model, to give anyone who possesses the object code either (1) a +    copy of the Corresponding Source for all the software in the +    product that is covered by this License, on a durable physical +    medium customarily used for software interchange, for a price no +    more than your reasonable cost of physically performing this +    conveying of source, or (2) access to copy the +    Corresponding Source from a network server at no charge. + +    c) Convey individual copies of the object code with a copy of the +    written offer to provide the Corresponding Source.  This +    alternative is allowed only occasionally and noncommercially, and +    only if you received the object code with such an offer, in accord +    with subsection 6b. + +    d) Convey the object code by offering access from a designated +    place (gratis or for a charge), and offer equivalent access to the +    Corresponding Source in the same way through the same place at no +    further charge.  You need not require recipients to copy the +    Corresponding Source along with the object code.  If the place to +    copy the object code is a network server, the Corresponding Source +    may be on a different server (operated by you or a third party) +    that supports equivalent copying facilities, provided you maintain +    clear directions next to the object code saying where to find the +    Corresponding Source.  Regardless of what server hosts the +    Corresponding Source, you remain obligated to ensure that it is +    available for as long as needed to satisfy these requirements. + +    e) Convey the object code using peer-to-peer transmission, provided +    you inform other peers where the object code and Corresponding +    Source of the work are being offered to the general public at no +    charge under subsection 6d. + +  A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + +  A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling.  In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage.  For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product.  A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + +  "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source.  The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + +  If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information.  But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + +  The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed.  Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + +  Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + +  7. Additional Terms. + +  "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law.  If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + +  When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it.  (Additional permissions may be written to require their own +removal in certain cases when you modify the work.)  You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + +  Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + +    a) Disclaiming warranty or limiting liability differently from the +    terms of sections 15 and 16 of this License; or + +    b) Requiring preservation of specified reasonable legal notices or +    author attributions in that material or in the Appropriate Legal +    Notices displayed by works containing it; or + +    c) Prohibiting misrepresentation of the origin of that material, or +    requiring that modified versions of such material be marked in +    reasonable ways as different from the original version; or + +    d) Limiting the use for publicity purposes of names of licensors or +    authors of the material; or + +    e) Declining to grant rights under trademark law for use of some +    trade names, trademarks, or service marks; or + +    f) Requiring indemnification of licensors and authors of that +    material by anyone who conveys the material (or modified versions of +    it) with contractual assumptions of liability to the recipient, for +    any liability that these contractual assumptions directly impose on +    those licensors and authors. + +  All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10.  If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term.  If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + +  If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + +  Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + +  8. Termination. + +  You may not propagate or modify a covered work except as expressly +provided under this License.  Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + +  However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + +  Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + +  Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License.  If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + +  9. Acceptance Not Required for Having Copies. + +  You are not required to accept this License in order to receive or +run a copy of the Program.  Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance.  However, +nothing other than this License grants you permission to propagate or +modify any covered work.  These actions infringe copyright if you do +not accept this License.  Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + +  10. Automatic Licensing of Downstream Recipients. + +  Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License.  You are not responsible +for enforcing compliance by third parties with this License. + +  An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations.  If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + +  You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License.  For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + +  11. Patents. + +  A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based.  The +work thus licensed is called the contributor's "contributor version". + +  A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version.  For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + +  Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + +  In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement).  To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + +  If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients.  "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + +  If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + +  A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License.  You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + +  Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + +  12. No Surrender of Others' Freedom. + +  If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License.  If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all.  For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + +  13. Use with the GNU Affero General Public License. + +  Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work.  The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + +  14. Revised Versions of this License. + +  The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time.  Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +  Each version is given a distinguishing version number.  If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation.  If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + +  If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + +  Later license versions may give you additional or different +permissions.  However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + +  15. Disclaimer of Warranty. + +  THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + +  16. Limitation of Liability. + +  IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + +  17. Interpretation of Sections 15 and 16. + +  If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + +                     END OF TERMS AND CONDITIONS + +            How to Apply These Terms to Your New Programs + +  If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + +  To do so, attach the following notices to the program.  It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + +    <one line to give the program's name and a brief idea of what it does.> +    Copyright (C) <year>  <name of author> + +    This program is free software: you can redistribute it and/or modify +    it under the terms of the GNU General Public License as published by +    the Free Software Foundation, either version 3 of the License, or +    (at your option) any later version. + +    This program is distributed in the hope that it will be useful, +    but WITHOUT ANY WARRANTY; without even the implied warranty of +    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +    GNU General Public License for more details. + +    You should have received a copy of the GNU General Public License +    along with this program.  If not, see <https://www.gnu.org/licenses/>. + +Also add information on how to contact you by electronic and paper mail. + +  If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + +    <program>  Copyright (C) <year>  <name of author> +    This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. +    This is free software, and you are welcome to redistribute it +    under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License.  Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + +  You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +<https://www.gnu.org/licenses/>. + +  The GNU General Public License does not permit incorporating your program +into proprietary programs.  If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library.  If this is what you want to do, use the GNU Lesser General +Public License instead of this License.  But first, please read +<https://www.gnu.org/philosophy/why-not-lgpl.html>. diff --git a/plugins/zsh-navigation-tools/NEWS b/plugins/zsh-navigation-tools/NEWS new file mode 100644 index 000000000..acd9f2ebc --- /dev/null +++ b/plugins/zsh-navigation-tools/NEWS @@ -0,0 +1,17 @@ +------------------------------------- +CHANGES FROM PREVIOUS VERSIONS OF ZNT +------------------------------------- + +Changes from 2.2.1 to 2.2.7 +--------------------------- + +Tmux integration has been added – bind file znt-tmux.zsh in Tmux as +described in README.md and e.g. run local history on remote hosts. Tmux +opens new window with n-history, and pastes selected history entry into +immediate previous window (e.g. a remote session). Fixed plugin.zsh file +to not use (outer scope) positional parameters. This fixes problem with +Grlm's Zsh configuration. The file now doesn't use "test" builtin (but +[[ instead), because it can be shadowed by alias or command. Private +history has been fixed to not overwrite its history file with the same +content. This improves performance when switching to private history +view. diff --git a/plugins/zsh-navigation-tools/README.md b/plugins/zsh-navigation-tools/README.md new file mode 100644 index 000000000..ed532a161 --- /dev/null +++ b/plugins/zsh-navigation-tools/README.md @@ -0,0 +1,244 @@ +# Zsh Navigation Tools + +https://raw.githubusercontent.com/psprint/zsh-navigation-tools/master/doc/img/n-history2.png + +Set of tools like n-history – multi-word history searcher, n-cd – directory +bookmark manager, n-kill – htop like kill utility, and more. Based on +n-list, a tool generating selectable curses-based list of elements that has +access to current Zsh session, i.e. has broad capabilities to work together +with it. Feature highlights include incremental multi-word searching, ANSI +coloring, unique mode, horizontal scroll, non-selectable elements, grepping and +various integrations with Zsh. + +## History Widget + +To have n-history as multi-word incremental searcher bound to Ctrl-R copy znt-* +files into the */site-functions dir (unless you use Oh My Zsh) and +add: + +    autoload znt-history-widget +    zle -N znt-history-widget +    bindkey "^R" znt-history-widget + +to .zshrc. This is done automatically when using Oh My Zsh. Two other +widgets exist, znt-cd-widget and znt-kill-widget, they can be too assigned +to key combinations (no need for autoload when using Oh My Zsh): + +    zle -N znt-cd-widget +    bindkey "^A" znt-cd-widget +    zle -N znt-kill-widget +    bindkey "^Y" znt-kill-widget + +Oh My Zsh stores history into ~/.zsh_history. When you switch to OMZ you could +want to copy your previous data (from e.g. ~/.zhistory) into the new location. + +## News + +* 06-10-2016 +  - Tmux-integration – Ctrl-b-h in Tmux to open n-history in new window. +    Then select history entry, it will be copied to the original Tmux window. +    Use this to execute local commands on remote hosts. All that is needed is +    this line added to ~/.tmux.conf: + +    bind h run-shell -b "$ZNT_REPO_DIR/znt-tmux.zsh" + +* 16-05-2016 +  - n-kill has completion. It proposes *words* from what's in `ps -A`. Giving n-kill +    arguments means grepping – it will start only with matching `ps` entries. + +* 15-05-2016 +  - Fixed problem where zsh-syntax-highlighting could render n-history slow (for +    long history entries). + +* 14-05-2016 +  - Configuration can be set from zshrc. Example: + +    znt_list_instant_select=1 +    znt_list_border=0 +    znt_list_bold=1 +    znt_list_colorpair="green/black" +    znt_functions_keywords=( "zplg" "zgen" "match" ) +    znt_cd_active_text="underline" +    znt_env_nlist_coloring_color=$'\x1b[00;33m' +    znt_cd_hotlist=( "~/.config/znt" "/usr/share/zsh/site-functions" "/usr/share/zsh" +                     "/usr/local/share/zsh/site-functions" "/usr/local/share/zsh" +                     "/usr/local/bin" ) + +* 10-05-2016 +  - Search query rotation – use Ctrl-A to rotate entered words right. +    Words `1 2 3` become `3 1 2`. + +* 09-05-2016 +  - New feature: n-help tool, available also from n-history via H key. It +    displays help screen with various information on ZNT. + +* 08-05-2016 +  - Approximate matching – pressing f or Ctrl-F will enter FIX mode, in +    which 1 or 2 errors are allowed in what is searched. This utilizes +    original Zsh approximate matching features and is intended to be used +    after entering search query, when a typo is discovered. + +* 06-05-2016 +  - Private history can be edited. Use e key or Ctrl-E for that when in +    n-history. Your $EDITOR will start. This is a way to have handy set +    of bookmarks prepared in private history's file. +  - Border can be disabled. Use following snippet in ~/.config/znt/n-list.conf +    or any other tool-targetted config file: + +    # Should draw the border? +    local border=0 + +* 30-04-2016 +  - New feature: color themes. Use Ctrl-T and Ctrl-G to browse predefined +    themes. They are listed in ~/.config/znt/n-list.conf. Use the file to +    permanently set a color scheme. Also, I sent a patch to Zsh developers +    and starting from Zsh > 5.2 (not yet released) supported will be 256 colors. +    The file ~/.config/znt/n-list.conf already has set of 256-color themes prepared :) + +* 29-04-2016 +  - New feature: private history – n-history tracks selected history entries, +    exposes them via new view (activated with F1) + +* 28-04-2016 +  - New features: +    1. New n-history view (activated with F1): Most Frequent History Words +    2. Predefined search keywords – use F2 to quickly search for chosen +       keywords (video: [https://youtu.be/DN9QqssAYB8](https://youtu.be/DN9QqssAYB8)) +    3. Configuration option for doing instant selection in search mode + +## Introduction + +The tools are: + +- n-aliases - browses aliases, relegates editing to vared +- n-cd - browses dirstack and bookmarked directories, allows to enter selected directory +- n-functions - browses functions, relegates editing to zed or vared +- n-history - browses history, allows to edit and run commands from it +- n-kill - browses processes list, allows to send signal to selected process +- n-env - browses environment, relegates editing to vared +- n-options - browses options, allows to toggle their state +- n-panelize - loads output of given command into the list for browsing + +All tools support horizontal scroll with <,>, {,}, h,l or left and right +cursors. Other keys are: + +- [,] - jump directory bookmarks in n-cd and typical signals in n-kill +- Ctrl-d, Ctrl-u - half page up or down +- Ctrl-p, Ctrl-n - previous and next (also done with vim's j,k) +- Ctrl-l - redraw of whole display +- g, G - beginning and end of the list +- Ctrl-o, o - enter uniq mode (no duplicate lines) +- / - start incremental search +- Enter - finish incremental search, retaining filter +- Esc - exit incremental search, clearing filter +- Ctrl-w (in incremental search) - delete whole word +- Ctrl-k (in incremental search) - delete whole line + +## Configuration + +`ZNT` has configuration files located in `~/.config/znt`. The files are: + +``` +n-aliases.conf +n-cd.conf +n-env.conf +n-functions.conf +n-history.conf +n-kill.conf +n-list.conf +n-options.conf +n-panelize.conf +``` + +`n-list.conf` contains main configuration variables: + +```zsh +# Should the list (text, borders) be drawn in bold +local bold=0 + +# Main color pair (foreground/background) +local colorpair="white/black" + +# Should draw the border? +local border=1 + +# Combinations of colors to try out with Ctrl-T and Ctrl-G +# The last number is the bold option, 0 or 1 +local -a themes +themes=( "white/black/1" "green/black/0" "green/black/1" "white/blue/0" "white/blue/1" +         "magenta/black/0" "magenta/black/1" ) +``` + +Read remaining configuration files to see what's in them. Nevertheless, configuration +can be also set from `zshrc`. There are `5` standard `zshrc` configuration variables: + +``` +znt_history_active_text - underline or reverse - how should be active element highlighted +znt_history_nlist_coloring_pattern - pattern that can be used to colorize elements +znt_history_nlist_coloring_color - color with which to colorize +znt_history_nlist_coloring_match_multiple - should multiple matches be colorized (0 or 1) +znt_history_keywords (array) - search keywords activated with `Ctrl-X` +``` + +Above variables will work for `n-history` tool. For other tools, change `_history_` to +e.g. `_cd_`, for the `n-cd` tool. The same works for all `8` tools. + +Common configuration of the tools uses variables with `_list_` in them: + +``` +znt_list_bold - should draw text in bold (0 or 1) +znt_list_colorpair - main pair of colors to be used, e.g "green/black" +znt_list_border - should draw borders around windows (0 or 1) +znt_list_themes (array) - list of themes to try out with Ctrl-T, e.g. ( "white/black/1" "green/black/0" ) +znt_list_instant_select - should pressing enter in search mode leave tool (0 or 1) +``` + +## Programming + +The function n-list is used as follows: + +    n-list {element1} [element2] ... [elementN] + +This is all that is needed to be done to have the features like ANSI coloring, +incremental multi-word search, unique mode, horizontal scroll, non-selectable +elements (grepping is done outside n-list, see the tools for how it can be +done). To set up non-selectable entries add their indices into array +NLIST_NONSELECTABLE_ELEMENTS: + +    typeset -a NLIST_NONSELECTABLE_ELEMENTS +    NLIST_NONSELECTABLE_ELEMENTS=( 1 ) + +Result is stored as $reply[REPLY] ($ isn't needed before REPLY because +of arithmetic context inside []). The returned array might be different from +input arguments as n-list can process them via incremental search or uniq +mode. $REPLY is the index in that possibly processed array. If $REPLY +equals -1 it means that no selection have been made (user quitted via q +key). + +To set up entries that can be jumped to with [,] keys add their indices to +NLIST_HOP_INDEXES array: + +    typeset -a NLIST_HOP_INDEXES +    NLIST_HOP_INDEXES=( 1 10 ) + +n-list can automatically colorize entries according to a Zsh pattern. +Following example will colorize all numbers with blue: + +    local NLIST_COLORING_PATTERN="[0-9]##" +    local NLIST_COLORING_COLOR=$'\x1b[00;34m' +    local NLIST_COLORING_END_COLOR=$'\x1b[0m' +    local NLIST_COLORING_MATCH_MULTIPLE=1 +    n-list "This is a number 123" "This line too has a number: 456" + +Blue is the default color, it doesn't have to be set. See zshexpn man page +for more information on Zsh patterns. Briefly, comparing to regular +expressions, (#s) is ^, (#e) is $, # is *, ## is +. Alternative +will work when in parenthesis, i.e. (a|b). BTW by using this method you can +colorize output of the tools, via their config files (check out e.g. n-cd.conf, +it uses this). + +## Performance +ZNT are fastest with Zsh before 5.0.6 and starting from 5.2 + + +vim:filetype=conf diff --git a/plugins/zsh-navigation-tools/_n-kill b/plugins/zsh-navigation-tools/_n-kill new file mode 100644 index 000000000..6f5d47971 --- /dev/null +++ b/plugins/zsh-navigation-tools/_n-kill @@ -0,0 +1,41 @@ +#compdef n-kill + +local context ret=1 + +typeset -a list linewords +typeset -aU words wordbits +local line word +integer cygwin=0 + +local IFS=" +" + +case "$OSTYPE" in +    cygwin*) list=( `command ps -Wa` ); cygwin=1 ;; +    *) list=( `command ps -o pid,uid,command -A` ) ;; +esac + +shift list + +IFS=" " +for line in "${list[@]}"; do +    linewords=( $=line ) +    if [ "$cygwin" = "1" ]; then +        linewords=( "${(@)linewords[8,-1]}" ) +    else +        linewords=( "${(@)linewords[3,-1]}" ) +    fi +    for word in "${linewords[@]}"; do +        if [ "$cygwin" = "1" ]; then +            wordbits=( "${(@s:\:)word}" ) +        else +            wordbits=( "${(@s:/:)word}" ) +        fi +        words+=( "${wordbits[@]}" ) +    done +done + +_wanted bits expl "Processes' name bits" \ +    compadd "$@" -a - words && ret=0 + +return "$ret" diff --git a/plugins/zsh-navigation-tools/n-aliases b/plugins/zsh-navigation-tools/n-aliases new file mode 100644 index 000000000..f8c1c3727 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-aliases @@ -0,0 +1,47 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-aliases` to .zshrc +# +# This function allows to choose an alias for edition with vared +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob +zmodload zsh/curses +zmodload zsh/parameter + +local IFS=" +" + +unset NLIST_COLORING_PATTERN + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-aliases.conf ] && builtin source ~/.config/znt/n-aliases.conf + +local list +local selected + +NLIST_REMEMBER_STATE=0 + +list=( "${(@k)aliases}" ) +list=( "${(@M)list:#(#i)*$1*}" ) + +local NLIST_GREP_STRING="$1" + +if [ "$#list" -eq 0 ]; then +    echo "No matching aliases" +    return 1 +fi + +list=( "${(@i)list}" ) +n-list "$list[@]" + +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" +    echo "Editing \`$selected':" +    print -rs "vared aliases\\[$selected\\]" +    vared aliases\[$selected\] +fi + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-cd b/plugins/zsh-navigation-tools/n-cd new file mode 100644 index 000000000..48c6dbf4e --- /dev/null +++ b/plugins/zsh-navigation-tools/n-cd @@ -0,0 +1,71 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-cd` to .zshrc +# +# This function allows to choose a directory from pushd stack +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob pushdignoredups + +zmodload zsh/curses +local IFS=" +" + +# Unset before configuration is read +unset NLIST_COLORING_PATTERN + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-cd.conf ] && builtin source ~/.config/znt/n-cd.conf + +local list +local selected + +NLIST_REMEMBER_STATE=0 + +list=( `dirs -p` ) +list=( "${(@M)list:#(#i)*$1*}" ) + +local NLIST_GREP_STRING="$1" + +[ "$#list" -eq 0 ] && echo "No matching directories" + +if [ "$#hotlist" -ge 1 ]; then +    typeset -a NLIST_NONSELECTABLE_ELEMENTS NLIST_HOP_INDEXES +    local tmp_list_size="$#list" +    NLIST_NONSELECTABLE_ELEMENTS=( $(( tmp_list_size+1 )) $(( tmp_list_size+2 )) ) +    list=( "$list[@]" "" $'\x1b[00;31m'"Hotlist"$'\x1b[00;00m': "$hotlist[@]" ) +    (( tmp_list_size+=3 )) +    local middle_hop=$(( (tmp_list_size+$#list) / 2 )) +    [[ "$middle_hop" -eq $tmp_list_size || "$middle_hop" -eq $#list ]] && middle_hop="" +    [ "$tmp_list_size" -eq $#list ] && tmp_list_size="" +    NLIST_HOP_INDEXES=( 1 $tmp_list_size $middle_hop $#list ) +else +    [ "$#list" -eq 0 ] && return 1 +fi + +n-list "${list[@]}" + +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" +    selected="${selected/#\~/$HOME}" + +    (( NCD_DONT_PUSHD )) && setopt NO_AUTO_PUSHD +    cd "$selected" +    local code=$? +    (( NCD_DONT_PUSHD )) && setopt AUTO_PUSHD + +    if [ "$code" -eq "0" ]; then +        # ZLE? +        if [ "${(t)CURSOR}" = "integer-local-special" ]; then +            zle -M "You have selected $selected" +        else +            echo "You have selected $selected" +        fi +    fi +else +    [ "${(t)CURSOR}" = "integer-local-special" ] && zle redisplay +fi + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-env b/plugins/zsh-navigation-tools/n-env new file mode 100644 index 000000000..c003b63a8 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-env @@ -0,0 +1,47 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-env` to .zshrc +# +# This function allows to choose an environment variable +# for edition with vared +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob +unsetopt equals +zmodload zsh/curses + +local IFS=" +" + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-env.conf ] && builtin source ~/.config/znt/n-env.conf + +local list +local selected + +NLIST_REMEMBER_STATE=0 + +list=( `env` ) +list=( "${(@M)list:#(#i)*$1*}" ) + +local NLIST_GREP_STRING="$1" + +if [ "$#list" -eq 0 ]; then +    echo "No matching variables" +    return 1 +fi + +list=( "${(@i)list}" ) +n-list "$list[@]" + +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" +    selected="${selected%%=*}" +    echo "Editing \`$selected':" +    print -rs "vared \"$selected\"" +    vared "$selected" +fi + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-functions b/plugins/zsh-navigation-tools/n-functions new file mode 100644 index 000000000..1748cbc90 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-functions @@ -0,0 +1,54 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-functions` to .zshrc +# +# This function allows to choose a function for edition with vared +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob +zmodload zsh/curses +zmodload zsh/parameter + +local IFS=" +" + +unset NLIST_COLORING_PATTERN + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-functions.conf ] && builtin source ~/.config/znt/n-functions.conf + +local list +local selected + +NLIST_REMEMBER_STATE=0 + +list=( "${(@k)functions}" ) +list=( "${(@M)list:#(#i)*$1*}" ) + +local NLIST_GREP_STRING="$1" + +if [ "$#list" -eq 0 ]; then +    echo "No matching functions" +    return 1 +fi + +list=( "${(@i)list}" ) +n-list "$list[@]" + +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" +    if [ "$feditor" = "zed" ]; then +        echo "Editing \`$selected' (ESC ZZ or Ctrl-x-w to finish):" +        autoload zed +        print -rs "zed -f -- \"$selected\"" +        zed -f -- "$selected" +    else +        echo "Editing \`$selected':" +        print -rs "vared functions\\[$selected\\]" +        vared functions\[$selected\] +    fi +fi + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-help b/plugins/zsh-navigation-tools/n-help new file mode 100644 index 000000000..c7f6328b7 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-help @@ -0,0 +1,135 @@ +autoload colors +colors + +local h1="$fg_bold[magenta]" +local h2="$fg_bold[green]" +local h3="$fg_bold[blue]" +local h4="$fg_bold[yellow]" +local h5="$fg_bold[cyan]" +local rst="$reset_color" + +LESS="-iRc" less <<<" +${h1}Key Bindings${rst} + +${h2}H${rst}, ${h2}?${rst} (from n-history) - run n-help +${h2}Ctrl-A${rst} - rotate entered words (1+2+3 -> 3+1+2) +${h2}Ctrl-F${rst} - fix mode (approximate matching) +${h2}Ctrl-L${rst} - redraw of whole display +${h2}Ctrl-T${rst} - browse themes (next theme) +${h2}Ctrl-G${rst} - browse themes (previous theme) +${h2}Ctrl-U${rst} - half page up +${h2}Ctrl-D${rst} - half page down +${h2}Ctrl-P${rst} - previous element (also done with vim's k) +${h2}Ctrl-N${rst} - next element (also done with vim's j) +${h2}[${rst}, ${h2}]${rst} - jump directory bookmarks in n-cd and typical signals in n-kill +${h2}g, ${h2}G${rst} - beginning and end of the list +${h2}/${rst} - show incremental search +${h2}F3${rst} - show/hide incremental search +${h2}Esc${rst} - exit incremental search, clearing filter +${h2}Ctrl-W${rst} (in incremental search) - delete whole word +${h2}Ctrl-K${rst} (in incremental search) - delete whole line +${h2}Ctrl-O, ${h2}o${rst} - enter uniq mode (no duplicate lines) +${h2}Ctrl-E, ${h2}e${rst} - edit private history (when in private history view) +${h2}F1${rst} - (in n-history) - switch view +${h2}F2${rst}, ${h2}Ctrl-X${rst}, ${h2}Ctrl-/${rst} - search predefined keywords (defined in config files) + +${h1}Configuration files${rst} + +Location of the files is ${h3}~/.config/znt${rst}. Skeletons are copied there +when using ${h3}zsh-navigation-tools.plugin.zsh${rst} file (sourcing it or using +a plugin manager). There's a main config file ${h3}n-list.conf${rst} and files +for each tool. + +To have a skeleton copied again into ${h3}~/.config/znt${rst}, delete it from +there and restart Zsh a few times (3-7 or so; there's a random check +that optimizes startup time). + +${h1}Predefined search keywords${rst} + +Following block of code in e.g. ${h3}~/.config/znt/n-history.conf${rst} defines +set of keywords that can be invoked (i.e. searched for) via ${h2}F2${rst}, ${h2}Ctrl-X${rst} +or ${h2}Ctrl-/${rst}: + +    ${h4}# Search keywords, iterated with F2 or Ctrl-X or Ctrl-/${rst} +    ${h2}local${rst} -a keywords +    keywords=( ${h2}\"git\" \"vim\" \"mplayer\"${rst} ) + +${h1}Search query rotation${rst} + +When searching, after pressing ${h2}Ctrl-A${rst}, words 1 2 3 will become 3 1 2, etc. +This can be used to edit some not-last word. + +${h1}Fix mode${rst} + +Approximate matching - pressing ${h2}f${rst} or ${h2}Ctrl-F${rst} will enter "FIX" mode, in which +1 or 2 errors are allowed in what's searched. This utilizes original Zsh +approximate matching features and is intended to be used after entering +search query, when a typo is discovered. + +${h1}Color themes${rst} + +Following block of code in ${h3}~/.config/znt/n-list.conf${rst} defines set of +themes that can be browsed with ${h2}Ctrl-T${rst} and ${h2}Ctrl-G${rst}: + +    ${h4}# Combinations of colors to try out with Ctrl-T and Ctrl-G +    # The last number is the bold option, 0 or 1${rst} +    ${h2}local${rst} -a themes +    themes=( ${h2}\"white/black/1\" \"green/black/0\" \"green/black/1\"${rst} +             ${h2}\"white/blue/0\" \"white/blue/1\" \"magenta/black/0\"${rst} +             ${h2}\"magenta/black/1\"${rst} ) + +It's \"foreground/background/bold\". There's support for 256-color themes +for Zsh > 5.2, defined like e.g.:  + +    themes=( ${h2}\"white/17/0\" \"10/17/1\" \"white/24/1\"${rst} ) + +i.e. with use of numbers, from 0 to 254. + +${h1}Private history${rst} + +N-history stores what's selected in its own history file. It can be +edited. Use ${h2}e${rst} or ${h2}Ctrl-E${rst} for that when in n-history. Your \$EDITOR will +start. This is a way to have handy set of bookmarks prepared in private +history's file. + +Private history is instantly shared among sessions. + +${h1}Zshrc integration${rst} + +There are 5 standard configuration variables that can be set in zshrc: + +${h4}znt_history_active_text${rst} +\"underline\" or \"reverse\" - how should be active element highlighted +${h4}znt_history_nlist_coloring_pattern${rst} +Pattern that can be used to colorize elements +${h4}znt_history_nlist_coloring_color${rst} +Color with which to colorize via the pattern +${h4}znt_history_nlist_coloring_match_multiple${rst} +Should multiple matches be colorized (${h2}\"0\"${rst} or ${h2}\"1\"${rst}) +${h4}znt_history_keywords ${h2}(array)${rst} +Search keywords activated with Ctrl-X, F2, Ctrl-/, e.g. ( ${h2}\"git\"${rst} ${h2}\"vim\"${rst} ) + +Above variables will work for n-history tool. For other tools, change +\"_history_\" to e.g. \"_cd_\", for the n-cd tool. The same works for +all 8 tools. + +Common configuration of the tools uses variables with \"_list_\" in them: + +${h4}znt_list_bold${rst} +Should draw text in bold (${h2}\"0\"${rst} or ${h2}\"1\"${rst}) +${h4}znt_list_colorpair${rst} +Main pair of colors to be used, e.g ${h2}\"green/black\"${rst} +${h4}znt_list_border${rst} +Should draw borders around windows (${h2}\"0\"${rst} or ${h2}\"1\"${rst}) +${h4}znt_list_themes ${h2}(array)${rst} +List of themes to try out with Ctrl-T, e.g. ( ${h2}\"white/black/1\"${rst} +${h2}\"green/black/0\"${rst} ) +${h4}znt_list_instant_select${rst} +Should pressing enter in search mode leave tool (${h2}\"0\"${rst} or ${h2}\"1\"${rst}) + +If you used ZNT before v2.1.12 then remove old configuration files +${h3}~/.config/znt/*.conf${rst} so that ZNT can update them to the latest versions +that support integration with Zshrc. If you used installer then run it +again (after the remove of configuration files), that is not needed when +using as plugin. +" diff --git a/plugins/zsh-navigation-tools/n-history b/plugins/zsh-navigation-tools/n-history new file mode 100644 index 000000000..b425ecd10 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-history @@ -0,0 +1,371 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-history` to .zshrc +# +# This function allows to browse Z shell's history and use the +# entries +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob +zmodload zsh/curses +zmodload zsh/parameter + +local IFS=" +" + +# Variables to save list's state when switching views +# The views are: history and "most frequent history words" +local one_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +local one_NLIST_CURRENT_IDX +local one_NLIST_IS_SEARCH_MODE +local one_NLIST_SEARCH_BUFFER +local one_NLIST_TEXT_OFFSET +local one_NLIST_IS_UNIQ_MODE +local one_NLIST_IS_F_MODE +local one_NLIST_GREP_STRING +local one_NLIST_NONSELECTABLE_ELEMENTS +local one_NLIST_REMEMBER_STATE +local one_NLIST_ENABLED_EVENTS + +local two_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +local two_NLIST_CURRENT_IDX +local two_NLIST_IS_SEARCH_MODE +local two_NLIST_SEARCH_BUFFER +local two_NLIST_TEXT_OFFSET +local two_NLIST_IS_UNIQ_MODE +local two_NLIST_IS_F_MODE +local two_NLIST_GREP_STRING +local two_NLIST_NONSELECTABLE_ELEMENTS +local two_NLIST_REMEMBER_STATE +local two_NLIST_ENABLED_EVENTS + +local three_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +local three_NLIST_CURRENT_IDX +local three_NLIST_IS_SEARCH_MODE +local three_NLIST_SEARCH_BUFFER +local three_NLIST_TEXT_OFFSET +local three_NLIST_IS_UNIQ_MODE +local three_NLIST_IS_F_MODE +local three_NLIST_GREP_STRING +local three_NLIST_NONSELECTABLE_ELEMENTS +local three_NLIST_REMEMBER_STATE +local three_NLIST_ENABLED_EVENTS + +# history view +integer active_view=0 + +# Lists are "0", "1", "2" - 1st, 2nd, 3rd +# Switching is done in cyclic manner +# i.e. 0 -> 1, 1 -> 2, 2 -> 0 +_nhistory_switch_lists_states() { +    # First argument is current, newly selected list, i.e. $active_view +    # This implies that we are switching from previous view +    +    if [ "$1" = "0" ]; then +        # Switched to 1st list, save 3rd list's state +        three_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +        three_NLIST_CURRENT_IDX=$NLIST_CURRENT_IDX +        three_NLIST_IS_SEARCH_MODE=$NLIST_IS_SEARCH_MODE +        three_NLIST_SEARCH_BUFFER=$NLIST_SEARCH_BUFFER +        three_NLIST_TEXT_OFFSET=$NLIST_TEXT_OFFSET +        three_NLIST_IS_UNIQ_MODE=$NLIST_IS_UNIQ_MODE +        three_NLIST_IS_F_MODE=$NLIST_IS_F_MODE +        three_NLIST_GREP_STRING=$NLIST_GREP_STRING +        three_NLIST_NONSELECTABLE_ELEMENTS=( ${NLIST_NONSELECTABLE_ELEMENTS[@]} ) +        three_NLIST_REMEMBER_STATE=$NLIST_REMEMBER_STATE +        three_NLIST_ENABLED_EVENTS=( ${NLIST_ENABLED_EVENTS[@]} ) + +        # ..and restore 1st list's state +        NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=$one_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +        NLIST_CURRENT_IDX=$one_NLIST_CURRENT_IDX +        NLIST_IS_SEARCH_MODE=$one_NLIST_IS_SEARCH_MODE +        NLIST_SEARCH_BUFFER=$one_NLIST_SEARCH_BUFFER +        NLIST_TEXT_OFFSET=$one_NLIST_TEXT_OFFSET +        NLIST_IS_UNIQ_MODE=$one_NLIST_IS_UNIQ_MODE +        NLIST_IS_F_MODE=$one_NLIST_IS_F_MODE +        NLIST_GREP_STRING=$one_NLIST_GREP_STRING +        NLIST_NONSELECTABLE_ELEMENTS=( ${one_NLIST_NONSELECTABLE_ELEMENTS[@]} ) +        NLIST_REMEMBER_STATE=$one_NLIST_REMEMBER_STATE +        NLIST_ENABLED_EVENTS=( ${one_NLIST_ENABLED_EVENTS[@]} ) +    elif [ "$1" = "1" ]; then +        # Switched to 2nd list, save 1st list's state +        one_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +        one_NLIST_CURRENT_IDX=$NLIST_CURRENT_IDX +        one_NLIST_IS_SEARCH_MODE=$NLIST_IS_SEARCH_MODE +        one_NLIST_SEARCH_BUFFER=$NLIST_SEARCH_BUFFER +        one_NLIST_TEXT_OFFSET=$NLIST_TEXT_OFFSET +        one_NLIST_IS_UNIQ_MODE=$NLIST_IS_UNIQ_MODE +        one_NLIST_IS_F_MODE=$NLIST_IS_F_MODE +        one_NLIST_GREP_STRING=$NLIST_GREP_STRING +        one_NLIST_NONSELECTABLE_ELEMENTS=( ${NLIST_NONSELECTABLE_ELEMENTS[@]} ) +        one_NLIST_REMEMBER_STATE=$NLIST_REMEMBER_STATE +        one_NLIST_ENABLED_EVENTS=( ${NLIST_ENABLED_EVENTS[@]} ) + +        # ..and restore 2nd list's state +        NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=$two_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +        NLIST_CURRENT_IDX=$two_NLIST_CURRENT_IDX +        NLIST_IS_SEARCH_MODE=$two_NLIST_IS_SEARCH_MODE +        NLIST_SEARCH_BUFFER=$two_NLIST_SEARCH_BUFFER +        NLIST_TEXT_OFFSET=$two_NLIST_TEXT_OFFSET +        NLIST_IS_UNIQ_MODE=$two_NLIST_IS_UNIQ_MODE +        NLIST_IS_F_MODE=$two_NLIST_IS_F_MODE +        NLIST_GREP_STRING=$two_NLIST_GREP_STRING +        NLIST_NONSELECTABLE_ELEMENTS=( ${two_NLIST_NONSELECTABLE_ELEMENTS[@]} ) +        NLIST_REMEMBER_STATE=$two_NLIST_REMEMBER_STATE +        NLIST_ENABLED_EVENTS=( ${two_NLIST_ENABLED_EVENTS[@]} ) +    elif [ "$1" = "2" ]; then +        # Switched to 3rd list, save 2nd list's state +        two_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +        two_NLIST_CURRENT_IDX=$NLIST_CURRENT_IDX +        two_NLIST_IS_SEARCH_MODE=$NLIST_IS_SEARCH_MODE +        two_NLIST_SEARCH_BUFFER=$NLIST_SEARCH_BUFFER +        two_NLIST_TEXT_OFFSET=$NLIST_TEXT_OFFSET +        two_NLIST_IS_UNIQ_MODE=$NLIST_IS_UNIQ_MODE +        two_NLIST_IS_F_MODE=$NLIST_IS_F_MODE +        two_NLIST_GREP_STRING=$NLIST_GREP_STRING +        two_NLIST_NONSELECTABLE_ELEMENTS=( ${NLIST_NONSELECTABLE_ELEMENTS[@]} ) +        two_NLIST_REMEMBER_STATE=$NLIST_REMEMBER_STATE +        two_NLIST_ENABLED_EVENTS=( ${NLIST_ENABLED_EVENTS[@]} ) + +        # ..and restore 3rd list's state +        NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=$three_NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN +        NLIST_CURRENT_IDX=$three_NLIST_CURRENT_IDX +        NLIST_IS_SEARCH_MODE=$three_NLIST_IS_SEARCH_MODE +        NLIST_SEARCH_BUFFER=$three_NLIST_SEARCH_BUFFER +        NLIST_TEXT_OFFSET=$three_NLIST_TEXT_OFFSET +        NLIST_IS_UNIQ_MODE=$three_NLIST_IS_UNIQ_MODE +        NLIST_IS_F_MODE=$three_NLIST_IS_F_MODE +        NLIST_GREP_STRING=$three_NLIST_GREP_STRING +        NLIST_NONSELECTABLE_ELEMENTS=( ${three_NLIST_NONSELECTABLE_ELEMENTS[@]} ) +        NLIST_REMEMBER_STATE=$three_NLIST_REMEMBER_STATE +        NLIST_ENABLED_EVENTS=( ${three_NLIST_ENABLED_EVENTS[@]} ) +    fi +} + +local most_frequent_db="$HOME/.config/znt/mostfrequent.db" +_nhistory_generate_most_frequent() { +    local title=$'\x1b[00;31m'"Most frequent history words:"$'\x1b[00;00m\0' + +    typeset -A uniq +    for k in "${historywords[@]}"; do +        uniq[$k]=$(( ${uniq[$k]:-0} + 1 )) +    done +    vk=() +    for k v in ${(kv)uniq}; do +        vk+="$v"$'\t'"$k" +    done + +    print -rl -- "$title" "${(On)vk[@]}" > "$most_frequent_db" +} + +# Load configuration +unset NLIST_COLORING_PATTERN +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-history.conf ] && builtin source ~/.config/znt/n-history.conf + +local list +local selected +local private_history_db="$HOME/.config/znt/privhist.db" + +local NLIST_GREP_STRING="$1" +# 2 is: init once, then remember +local NLIST_REMEMBER_STATE=2 +two_NLIST_REMEMBER_STATE=2 +three_NLIST_REMEMBER_STATE=2 + +# Only Private history has EDIT +local -a NLIST_ENABLED_EVENTS +NLIST_ENABLED_EVENTS=( "F1" "HELP" ) +two_NLIST_ENABLED_EVENTS=( "F1" "EDIT" "HELP" ) +three_NLIST_ENABLED_EVENTS=( "F1" "HELP" ) + +# All view should attempt to replace new lines with \n +local NLIST_REPLACE_NEWLINES="1" +two_NLIST_REPLACE_NEWLINES="1" +three_NLIST_REPLACE_NEWLINES="1" + +# Only second and third view has non-selectable first entry +local -a NLIST_NONSELECTABLE_ELEMENTS +NLIST_NONSELECTABLE_ELEMENTS=( ) +two_NLIST_NONSELECTABLE_ELEMENTS=( 1 ) +three_NLIST_NONSELECTABLE_ELEMENTS=( 1 ) + +while (( 1 )); do + +    # +    # View 1 - history +    # +    if [ "$active_view" = "0" ]; then +        list=( "$history[@]" ) +        list=( "${(@M)list:#(#i)*$NLIST_GREP_STRING*}" ) + +        if [ "$#list" -eq 0 ]; then +            echo "No matching history entries" +            return 1 +        fi + +        n-list "${list[@]}" + +        # Selection or quit? +        if [[ "$REPLY" = -(#c0,1)[0-9]## && ("$REPLY" -lt 0 || "$REPLY" -gt 0) ]]; then +            break +        fi + +        # View change? +        if [ "$REPLY" = "F1" ]; then +            # Target view: 2 +            active_view=1 +            _nhistory_switch_lists_states "1" +        elif [ "$REPLY" = "HELP" ]; then +            n-help +        fi + +    # +    # View 3 - most frequent words in history +    # +    elif [ "$active_view" = "2" ]; then +        local -a dbfile +        dbfile=( $most_frequent_db(Nm+1) ) + +        # Compute most frequent history words +        if [[ "${#NHISTORY_WORDS}" -eq "0" || "${#dbfile}" -ne "0" ]]; then +            # Read the list if it's there +            local -a list +            list=() +            [ -s "$most_frequent_db" ] && list=( ${(f)"$(<$most_frequent_db)"} ) + +            # Will wait for the data? +            local message=0 +            if [[ "${#list}" -eq 0 ]]; then +                message=1 +                _nlist_alternate_screen 1 +                zcurses init +                zcurses delwin info 2>/dev/null +                zcurses addwin info "$term_height" "$term_width" 0 0 +                zcurses bg info white/black +                zcurses string info "Computing most frequent history words..."$'\n' +                zcurses string info "(This is done once per day, from now on transparently)"$'\n' +                zcurses refresh info +                sleep 3 +            fi + +            # Start periodic list regeneration? +            if [[ "${#list}" -eq 0 || "${#dbfile}" -ne "0" ]]; then +                # Mark the file with current time, to prevent double +                # regeneration (on quick double change of view) +                print >> "$most_frequent_db" +                (_nhistory_generate_most_frequent &) &> /dev/null +            fi + +            # Ensure we got the list, wait for it if needed +            while [[ "${#list}" -eq 0 ]]; do +                zcurses string info "." +                zcurses refresh info +                LANG=C sleep 0.5 +                [ -s "$most_frequent_db" ] && list=( ${(f)"$(<$most_frequent_db)"} ) +            done + +            NHISTORY_WORDS=( "${list[@]}" ) + +            if [ "$message" -eq "1" ]; then +                zcurses delwin info 2>/dev/null +                zcurses refresh +                zcurses end +                _nlist_alternate_screen 0 +            fi +        else +            # Reuse most frequent history words +            local -a list +            list=( "${NHISTORY_WORDS[@]}" ) +        fi + +        n-list "${list[@]}" + +        if [ "$REPLY" = "F1" ]; then +            # Target view: 1 +            active_view=0 +            _nhistory_switch_lists_states "0" +        elif [[ "$REPLY" = -(#c0,1)[0-9]## && "$REPLY" -lt 0 ]]; then +            break +        elif [[ "$REPLY" = -(#c0,1)[0-9]## && "$REPLY" -gt 0 ]]; then +            local word="${reply[REPLY]#(#s) #[0-9]##$'\t'}" +            one_NLIST_SEARCH_BUFFER="$word" +            one_NLIST_SEARCH_BUFFER="${one_NLIST_SEARCH_BUFFER## ##}" + +            # Target view: 1 +            active_view=0 +            _nhistory_switch_lists_states "0" +        elif [ "$REPLY" = "HELP" ]; then +            n-help +        fi + +    # +    # View 2 - private history +    # +    elif [ "$active_view" = "1" ]; then +        if [ -s "$private_history_db" ]; then +            local title=$'\x1b[00;32m'"Private history:"$'\x1b[00;00m\0' +            () { fc -Rap "$private_history_db" 20000 0; list=( "$title" ${history[@]} ) } +        else +            list=( "Private history - history entries selected via this tool will be put here" ) +        fi + +        n-list "${list[@]}" + +        # Selection or quit? +        if [[ "$REPLY" = -(#c0,1)[0-9]## && ("$REPLY" -lt 0 || "$REPLY" -gt 0) ]]; then +            break +        fi + +        # View change? +        if [ "$REPLY" = "F1" ]; then +            # Target view: 3 +            active_view=2 +            _nhistory_switch_lists_states "2" +        # Edit of the history? +        elif [ "$REPLY" = "EDIT" ]; then +            "${EDITOR:-vim}" "$private_history_db" +        elif [ "$REPLY" = "HELP" ]; then +            n-help +        fi +    fi +done + +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" + +    # Append to private history +    if [[ "$active_view" = "0" ]]; then +        local newline=$'\n' +        local selected_ph="${selected//$newline/\\$newline}" +        print -r -- "$selected_ph" >> "$private_history_db" +    fi + +    # TMUX? +    if [[ "$ZNT_TMUX_MODE" = "1" ]]; then +        tmux send -t "$ZNT_TMUX_ORIGIN_SESSION:$ZNT_TMUX_ORIGIN_WINDOW.$ZNT_TMUX_ORIGIN_PANE" "$selected" +        tmux kill-window +        return 0 +    # ZLE? +    elif [ "${(t)CURSOR}" = "integer-local-special" ]; then +        zle .redisplay +        zle .kill-buffer +        LBUFFER+="$selected" +    else +        print -zr -- "$selected" +    fi +else +    # TMUX? +    if [[ "$ZNT_TMUX_MODE" = "1" ]]; then +        tmux kill-window +    # ZLE? +    elif [[ "${(t)CURSOR}" = "integer-local-special" ]]; then +        zle redisplay +    fi +fi + +return 0 + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-kill b/plugins/zsh-navigation-tools/n-kill new file mode 100644 index 000000000..76050f969 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-kill @@ -0,0 +1,96 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-kill` to .zshrc +# +# This function allows to choose a process and a signal to send to it +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob +zmodload zsh/curses + +local IFS=" +" + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-kill.conf ] && builtin source ~/.config/znt/n-kill.conf + +typeset -A signals +signals=( +     1       "1  - HUP" +     2       "2  - INT" +     3       "3  - QUIT" +     6       "6  - ABRT" +     9       "9  - KILL" +     14      "14 - ALRM" +     15      "15 - TERM" +     17      "17 - STOP" +     19      "19 - CONT" +) + +local list +local selected +local signal +local -a signal_names +local title + +NLIST_REMEMBER_STATE=0 + +typeset -a NLIST_NONSELECTABLE_ELEMENTS +NLIST_NONSELECTABLE_ELEMENTS=( 1 ) + +type ps 2>/dev/null 1>&2 || { echo >&2 "Error: \`ps' not found"; return 1 } + +case "$OSTYPE" in +    cygwin*) list=( `command ps -Wa` )  ;; +    *) list=( `command ps -o pid,uid,command -A` ) ;; +esac + +# Ask of PID +title=$'\x1b[00;31m'"${list[1]}"$'\x1b[00;00m\0' +shift list +list=( "$title" "${(@M)list:#(#i)*$1*}" ) + +local NLIST_GREP_STRING="$1" + +if [ "$#list" -eq 1 ]; then +    echo "No matching processes" +    return 1 +fi + +n-list "$list[@]" + +# Got answer? (could be Ctrl-C or 'q') +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" +    selected="${selected## #}" +    pid="${selected%% *}" + +    # Now ask of signal +    signal_names=( ${(vin)signals} ) +    typeset -a NLIST_HOP_INDEXES +    NLIST_HOP_INDEXES=( 3 6 8 ) +    unset NLIST_COLORING_PATTERN +    n-list $'\x1b[00;31mSelect signal:\x1b[00;00m' "$signal_names[@]" + +    if [ "$REPLY" -gt 0 ]; then +        selected="$reply[REPLY]" +        signal="${(k)signals[(r)$selected]}" + +        # ZLE? +        if [ "${(t)CURSOR}" = "integer-local-special" ]; then +            zle redisplay +            zle kill-whole-line +            zle -U "kill -$signal $pid" +        else +            print -zr "kill -$signal $pid" +        fi +    else +        [ "${(t)CURSOR}" = "integer-local-special" ] && zle redisplay +    fi +else +    [ "${(t)CURSOR}" = "integer-local-special" ] && zle redisplay +fi + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-list b/plugins/zsh-navigation-tools/n-list new file mode 100644 index 000000000..3fe5542a6 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-list @@ -0,0 +1,515 @@ +# $1, $2, ... - elements of the list +# $NLIST_NONSELECTABLE_ELEMENTS - array of indexes (1-based) that cannot be selected +# $REPLY is the output variable - contains index (1-based) or -1 when no selection +# $reply (array) is the second part of the output - use the index (REPLY) to get selected element +# +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-list` to .zshrc +# +# This function outputs a list of elements that can be +# navigated with keyboard. Uses curses library + +emulate -LR zsh + +setopt typesetsilent extendedglob noshortloops + +_nlist_has_terminfo=0 + +zmodload zsh/curses +zmodload zsh/terminfo 2>/dev/null && _nlist_has_terminfo=1 + +trap "REPLY=-2; reply=(); return" TERM INT QUIT +trap "_nlist_exit" EXIT + +# Drawing and input +autoload n-list-draw n-list-input + +# Cleanup before any exit +_nlist_exit() { +    setopt localoptions +    setopt extendedglob + +    [[ "$REPLY" = -(#c0,1)[0-9]## || "$REPLY" = F<-> || "$REPLY" = "EDIT" || "$REPLY" = "HELP" ]] || REPLY="-1" +    zcurses 2>/dev/null delwin inner +    zcurses 2>/dev/null delwin main +    zcurses 2>/dev/null refresh +    zcurses end +    _nlist_alternate_screen 0 +    _nlist_cursor_visibility 1 +    unset _nlist_has_terminfo +} + +# Outputs a message in the bottom of the screen +_nlist_status_msg() { +    # -1 for border, -1 for 0-based indexing +    zcurses move main $(( term_height - 1 - 1 )) 2 +    zcurses clear main eol +    zcurses string main "$1" +    #status_msg_strlen is localized in caller +    status_msg_strlen=$#1 +} + +# Prefer tput, then module terminfo +_nlist_cursor_visibility() { +    if type tput 2>/dev/null 1>&2; then +        [ "$1" = "1" ] && { tput cvvis; tput cnorm } +        [ "$1" = "0" ] && tput civis +    elif [ "$_nlist_has_terminfo" = "1" ]; then +        [ "$1" = "1" ] && { [ -n $terminfo[cvvis] ] && echo -n $terminfo[cvvis]; +                           [ -n $terminfo[cnorm] ] && echo -n $terminfo[cnorm] } +        [ "$1" = "0" ] && [ -n $terminfo[civis] ] && echo -n $terminfo[civis] +    fi  +} + +# Reason for this function is that on some systems +# smcup and rmcup are not knowing why left empty +_nlist_alternate_screen() { +    [ "$_nlist_has_terminfo" -ne "1" ] && return +    [[ "$1" = "1" && -n "$terminfo[smcup]" ]] && return +    [[ "$1" = "0" && -n "$terminfo[rmcup]" ]] && return + +    case "$TERM" in +        *rxvt*) +            [ "$1" = "1" ] && echo -n $'\x1b7\x1b[?47h' +            [ "$1" = "0" ] && echo -n $'\x1b[2J\x1b[?47l\x1b8' +            ;; +        *) +            [ "$1" = "1" ] && echo -n $'\x1b[?1049h' +            [ "$1" = "0" ] && echo -n $'\x1b[?1049l' +            # just to remember two other that work: $'\x1b7\x1b[r\x1b[?47h', $'\x1b[?47l\x1b8' +            ;; +    esac +} + +_nlist_compute_user_vars_difference() { +        if [[ "${(t)NLIST_NONSELECTABLE_ELEMENTS}" != "array" && +                "${(t)NLIST_NONSELECTABLE_ELEMENTS}" != "array-local" ]] +        then +            last_element_difference=0 +            current_difference=0 +        else +            last_element_difference=$#NLIST_NONSELECTABLE_ELEMENTS +            current_difference=0 +            local idx +            for idx in "${(n)NLIST_NONSELECTABLE_ELEMENTS[@]}"; do +                [ "$idx" -le "$NLIST_CURRENT_IDX" ] && current_difference+=1 || break +            done +        fi +} + +# List was processed, check if variables aren't off range +_nlist_verify_vars() { +    [ "$NLIST_CURRENT_IDX" -gt "$last_element" ] && NLIST_CURRENT_IDX="$last_element" +    [[ "$NLIST_CURRENT_IDX" -eq 0 && "$last_element" -ne 0 ]] && NLIST_CURRENT_IDX=1 +    (( NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=0+((NLIST_CURRENT_IDX-1)/page_height)*page_height+1 )) +} + +# Compute the variables which are shown to the user +_nlist_setup_user_vars() { +    if [ "$1" = "1" ]; then +        # Basic values when there are no non-selectables +        NLIST_USER_CURRENT_IDX="$NLIST_CURRENT_IDX" +        NLIST_USER_LAST_ELEMENT="$last_element" +    else +        _nlist_compute_user_vars_difference +        NLIST_USER_CURRENT_IDX=$(( NLIST_CURRENT_IDX - current_difference )) +        NLIST_USER_LAST_ELEMENT=$(( last_element - last_element_difference )) +    fi +} + +_nlist_colorify_disp_list() { +    local col=$'\x1b[00;34m' reset=$'\x1b[0m' +    [ -n "$NLIST_COLORING_COLOR" ] && col="$NLIST_COLORING_COLOR" +    [ -n "$NLIST_COLORING_END_COLOR" ] && reset="$NLIST_COLORING_END_COLOR" + +    if [ "$NLIST_COLORING_MATCH_MULTIPLE" -eq 1 ]; then +        disp_list=( "${(@)disp_list//(#mi)$~NLIST_COLORING_PATTERN/$col${MATCH}$reset}" ) +    else +        disp_list=( "${(@)disp_list/(#mi)$~NLIST_COLORING_PATTERN/$col${MATCH}$reset}" ) +    fi +} + +# +# Main code +# + +# Check if there is proper input +if [ "$#" -lt 1 ]; then +    echo "Usage: n-list element_1 ..." +    return 1 +fi + +REPLY="-1" +typeset -ga reply +reply=() + +integer term_height="$LINES" +integer term_width="$COLUMNS" +if [[ "$term_height" -lt 1 || "$term_width" -lt 1 ]]; then +    local stty_out=$( stty size ) +    term_height="${stty_out% *}" +    term_width="${stty_out#* }" +fi +integer inner_height=term_height-3 +integer inner_width=term_width-3 +integer page_height=inner_height +integer page_width=inner_width + +typeset -a list disp_list +integer last_element=$# +local action +local final_key +integer selection +integer last_element_difference=0 +integer current_difference=0 +local prev_search_buffer="" +integer prev_uniq_mode=0 +integer prev_start_idx=-1 +local MBEGIN MEND MATCH mbegin mend match + +# Iteration over predefined keywords +integer curkeyword nkeywords +local keywordisfresh="0" +if [[ "${(t)keywords}" != *array* ]]; then +    local -a keywords +    keywords=() +fi +curkeyword=0 +nkeywords=${#keywords} + +# Iteration over themes +integer curtheme nthemes +local themeisfresh="0" +if [[ "${(t)themes}" != *array* ]]; then +    local -a themes +    themes=() +fi +curtheme=0 +nthemes=${#themes} + +# Ability to remember the list between calls +if [[ -z "$NLIST_REMEMBER_STATE" || "$NLIST_REMEMBER_STATE" -eq 0 || "$NLIST_REMEMBER_STATE" -eq 2 ]]; then +    NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN=1 +    NLIST_CURRENT_IDX=1 +    NLIST_IS_SEARCH_MODE=0 +    NLIST_SEARCH_BUFFER="" +    NLIST_TEXT_OFFSET=0 +    NLIST_IS_UNIQ_MODE=0 +    NLIST_IS_F_MODE=0 + +    # Zero - because it isn't known, unless we +    # confirm that first element is selectable +    NLIST_USER_CURRENT_IDX=0 +    [[ ${NLIST_NONSELECTABLE_ELEMENTS[(r)1]} != 1 ]] && NLIST_USER_CURRENT_IDX=1 +    NLIST_USER_LAST_ELEMENT=$(( last_element - $#NLIST_NONSELECTABLE_ELEMENTS )) + +    # 2 is init once, then remember +    [ "$NLIST_REMEMBER_STATE" -eq 2 ] && NLIST_REMEMBER_STATE=1 +fi + +if [ "$NLIST_START_IN_SEARCH_MODE" -eq 1 ]; then +    NLIST_START_IN_SEARCH_MODE=0 +    NLIST_IS_SEARCH_MODE=1 +fi + +if [ -n "$NLIST_SET_SEARCH_TO" ]; then +    NLIST_SEARCH_BUFFER="$NLIST_SET_SEARCH_TO" +    NLIST_SET_SEARCH_TO="" +fi + +if [ "$NLIST_START_IN_UNIQ_MODE" -eq 1 ]; then +    NLIST_START_IN_UNIQ_MODE=0 +    NLIST_IS_UNIQ_MODE=1 +fi + +_nlist_alternate_screen 1 +zcurses init +zcurses delwin main 2>/dev/null +zcurses delwin inner 2>/dev/null +zcurses addwin main "$term_height" "$term_width" 0 0 +zcurses addwin inner "$inner_height" "$inner_width" 1 2 +# From n-list.conf +[ "$colorpair" = "" ] && colorpair="white/black" +[ "$border" = "0" ] || border="1" +local background="${colorpair#*/}" +local backuptheme="$colorpair/$bold" +zcurses bg main "$colorpair" +zcurses bg inner "$colorpair" +if [ "$NLIST_IS_SEARCH_MODE" -ne 1 ]; then +    _nlist_cursor_visibility 0 +fi + +zcurses refresh + +# +# Listening for input +# + +local key keypad + +# Clear input buffer +zcurses timeout main 0 +zcurses input main key keypad +zcurses timeout main -1 +key="" +keypad="" + +# This loop makes script faster on some Zsh's (e.g. 5.0.8) +repeat 1; do +    list=( "$@" ) +done + +last_element="$#list" + +while (( 1 )); do +    # Do searching (filtering with string) +    if [ -n "$NLIST_SEARCH_BUFFER" ]; then +        # Compute new list? +        if [[ "$NLIST_SEARCH_BUFFER" != "$prev_search_buffer" || "$NLIST_IS_UNIQ_MODE" -ne "$prev_uniq_mode" +                || "$NLIST_IS_F_MODE" -ne "$prev_f_mode" ]] +        then +            prev_search_buffer="$NLIST_SEARCH_BUFFER" +            prev_uniq_mode="$NLIST_IS_UNIQ_MODE" +            prev_f_mode="$NLIST_IS_F_MODE" +            # regenerating list -> regenerating disp_list +            prev_start_idx=-1 + +            # Take all elements, including duplicates and non-selectables +            typeset +U list +            repeat 1; do +                list=( "$@" ) +            done + +            # Remove non-selectable elements +            [ "$#NLIST_NONSELECTABLE_ELEMENTS" -gt 0 ] && for i in "${(nO)NLIST_NONSELECTABLE_ELEMENTS[@]}"; do +                if [[ "$i" = <-> ]]; then +                    list[$i]=() +                fi +            done + +            # Remove duplicates +            [ "$NLIST_IS_UNIQ_MODE" -eq 1 ] && typeset -U list + +            last_element="$#list" + +            # Next do the filtering +            local search_buffer="${NLIST_SEARCH_BUFFER%% ##}" +            search_buffer="${search_buffer## ##}" +            search_buffer="${search_buffer//(#m)[][*?|#~^()><\\]/\\$MATCH}" +            local search_pattern="" +            local colsearch_pattern="" +            if [ -n "$search_buffer" ]; then +                # The repeat will make the matching work on a fresh heap +                repeat 1; do +                    if [ "$NLIST_IS_F_MODE" -eq "1" ]; then +                        search_pattern="${search_buffer// ##/*~^(#a1)*}" +                        colsearch_pattern="${search_buffer// ##/|(#a1)}" +                        list=( "${(@M)list:#(#ia1)*$~search_pattern*}" ) +                    elif [ "$NLIST_IS_F_MODE" -eq "2" ]; then +                        search_pattern="${search_buffer// ##/*~^(#a2)*}" +                        colsearch_pattern="${search_buffer// ##/|(#a2)}" +                        list=( "${(@M)list:#(#ia2)*$~search_pattern*}" ) +                    else +                        # Pattern will be *foo*~^*bar* (inventor: Mikael Magnusson) +                        search_pattern="${search_buffer// ##/*~^*}" +                        # Pattern will be (foo|bar) +                        colsearch_pattern="${search_buffer// ##/|}" +                        list=( "${(@M)list:#(#i)*$~search_pattern*}" ) +                    fi +                done + +                last_element="$#list" +            fi + +            # Called after processing list +            _nlist_verify_vars +        fi + +        _nlist_setup_user_vars 1 + +        integer end_idx=$(( NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN + page_height - 1 )) +        [ "$end_idx" -gt "$last_element" ] && end_idx=last_element + +        if [ "$prev_start_idx" -ne "$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN" ]; then +            prev_start_idx="$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN" +            disp_list=( "${(@)list[NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN, end_idx]}" ) + +            if [ -n "$colsearch_pattern" ]; then +                local red=$'\x1b[00;31m' reset=$'\x1b[00;00m' +                # The repeat will make the matching work on a fresh heap +                repeat 1; do +                    if [ "$NLIST_IS_F_MODE" -eq "1" ]; then +                        disp_list=( "${(@)disp_list//(#mia1)($~colsearch_pattern)/$red${MATCH}$reset}" ) +                    elif [ "$NLIST_IS_F_MODE" -eq "2" ]; then +                        disp_list=( "${(@)disp_list//(#mia2)($~colsearch_pattern)/$red${MATCH}$reset}" ) +                    else +                        disp_list=( "${(@)disp_list//(#mi)($~colsearch_pattern)/$red${MATCH}$reset}" ) +                    fi +                done +            fi + +            # We have display list, lets replace newlines with "\n" when needed (1/2) +            [ "$NLIST_REPLACE_NEWLINES" -eq 1 ] && disp_list=( "${(@)disp_list//$'\n'/\\n}" ) +        fi + +        # Output colored list +        zcurses clear inner +        n-list-draw "$(( (NLIST_CURRENT_IDX-1) % page_height + 1 ))" \ +            "$page_height" "$page_width" 0 0 "$NLIST_TEXT_OFFSET" inner \ +            "$disp_list[@]" +    else +        # There is no search, but there was in previous loop +        # OR +        # Uniq mode was entered or left out +        # -> compute new list +        if [[ -n "$prev_search_buffer" || "$NLIST_IS_UNIQ_MODE" -ne "$prev_uniq_mode" ]]; then +            prev_search_buffer="" +            prev_uniq_mode="$NLIST_IS_UNIQ_MODE" +            # regenerating list -> regenerating disp_list +            prev_start_idx=-1 + +            # Take all elements, including duplicates and non-selectables +            typeset +U list +            repeat 1; do +                list=( "$@" ) +            done + +            # Remove non-selectable elements only when in uniq mode +            [ "$NLIST_IS_UNIQ_MODE" -eq 1 ] && [ "$#NLIST_NONSELECTABLE_ELEMENTS" -gt 0 ] && +            for i in "${(nO)NLIST_NONSELECTABLE_ELEMENTS[@]}"; do +                if [[ "$i" = <-> ]]; then +                    list[$i]=() +                fi +            done + +            # Remove duplicates when in uniq mode +            [ "$NLIST_IS_UNIQ_MODE" -eq 1 ] && typeset -U list + +            last_element="$#list" +            # Called after processing list +            _nlist_verify_vars +        fi + +        # "1" - shouldn't bother with non-selectables +        _nlist_setup_user_vars "$NLIST_IS_UNIQ_MODE" + +        integer end_idx=$(( NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN + page_height - 1 )) +        [ "$end_idx" -gt "$last_element" ] && end_idx=last_element + +        if [ "$prev_start_idx" -ne "$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN" ]; then +            prev_start_idx="$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN" +            disp_list=( "${(@)list[NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN, end_idx]}" ) + +            [ -n "$NLIST_COLORING_PATTERN" ] && _nlist_colorify_disp_list + +            # We have display list, lets replace newlines with "\n" when needed (2/2) +            [ "$NLIST_REPLACE_NEWLINES" -eq 1 ] && disp_list=( "${(@)disp_list//$'\n'/\\n}" ) +        fi + +        # Output the list +        zcurses clear inner +        n-list-draw "$(( (NLIST_CURRENT_IDX-1) % page_height + 1 ))" \ +            "$page_height" "$page_width" 0 0 "$NLIST_TEXT_OFFSET" inner \ +            "$disp_list[@]" +    fi + +    local status_msg_strlen +    local keywordmsg="" +    if [ "$keywordisfresh" = "1" ]; then +        keywordmsg="($curkeyword/$nkeywords) " +        keywordisfresh="0" +    fi + +    local thememsg="" +    if [ "$themeisfresh" = "1" ]; then +        local theme="$backuptheme" +        [ "$curtheme" -gt 0 ] && theme="${themes[curtheme]}" +        thememsg="($curtheme/$nthemes $theme) " +        themeisfresh="0" +    fi + +    local _txt2="" _txt3="" +    [ "$NLIST_IS_UNIQ_MODE" -eq 1 ] && _txt2="[-UNIQ-] " +    [ "$NLIST_IS_F_MODE" -eq 1 ] && _txt3="[-FIX-] " +    [ "$NLIST_IS_F_MODE" -eq 2 ] && _txt3="[-FIX2-] " + +    if [ "$NLIST_IS_SEARCH_MODE" = "1" ]; then +        _nlist_status_msg "${_txt2}${_txt3}${keywordmsg}${thememsg}Filtering with: ${NLIST_SEARCH_BUFFER// /+}" +    elif [[ ${NLIST_NONSELECTABLE_ELEMENTS[(r)$NLIST_CURRENT_IDX]} != $NLIST_CURRENT_IDX || +            -n "$NLIST_SEARCH_BUFFER" || "$NLIST_IS_UNIQ_MODE" -eq 1 ]]; then +        local _txt="" +        [ -n "$NLIST_GREP_STRING" ] && _txt=" [$NLIST_GREP_STRING]" +        _nlist_status_msg "${_txt2}${_txt3}${keywordmsg}${thememsg}Current #$NLIST_USER_CURRENT_IDX (of #$NLIST_USER_LAST_ELEMENT entries)$_txt" +    else +        _nlist_status_msg "${keywordmsg}${thememsg}" +    fi + +    [ "$border" = "1" ] && zcurses border main + +    local top_msg=" ${(C)ZSH_NAME} $ZSH_VERSION, shell level $SHLVL " +    if [[ "${NLIST_ENABLED_EVENTS[(r)F1]}" = "F1" ]]; then +        top_msg=" F1-change view,$top_msg" +    fi +    zcurses move main 0 $(( term_width / 2 - $#top_msg / 2 )) +    zcurses string main $top_msg + +    zcurses refresh main inner +    zcurses move main $(( term_height - 1 - 1 )) $(( status_msg_strlen + 2 )) + +    # Wait for input +    zcurses input main key keypad + +    # Get the special (i.e. "keypad") key or regular key +    if [ -n "$key" ]; then +        final_key="$key" +    elif [ -n "$keypad" ]; then +        final_key="$keypad" +    else +        _nlist_status_msg "Inproper input detected" +        zcurses refresh main inner +    fi + +    n-list-input "$NLIST_CURRENT_IDX" "$NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN" \ +                    "$page_height" "$page_width" "$last_element" "$NLIST_TEXT_OFFSET" \ +                    "$final_key" "$NLIST_IS_SEARCH_MODE" "$NLIST_SEARCH_BUFFER" \ +                    "$NLIST_IS_UNIQ_MODE" "$NLIST_IS_F_MODE" + +    selection="$reply[1]" +    action="$reply[2]" +    NLIST_CURRENT_IDX="$reply[3]" +    NLIST_FROM_WHAT_IDX_LIST_IS_SHOWN="$reply[4]" +    NLIST_TEXT_OFFSET="$reply[5]" +    NLIST_IS_SEARCH_MODE="$reply[6]" +    NLIST_SEARCH_BUFFER="$reply[7]" +    NLIST_IS_UNIQ_MODE="$reply[8]" +    NLIST_IS_F_MODE="$reply[9]" + +    if [ -z "$action" ]; then +        continue +    elif [ "$action" = "SELECT" ]; then +        REPLY="$selection" +        reply=( "$list[@]" ) +        break +    elif [ "$action" = "QUIT" ]; then +        REPLY=-1 +        reply=( "$list[@]" ) +        break +    elif [ "$action" = "REDRAW" ]; then +        zcurses clear main redraw +        zcurses clear inner redraw +    elif [[ "$action" = F<-> ]]; then +        REPLY="$action" +        reply=( "$list[@]" ) +        break +    elif [[ "$action" = "EDIT" ]]; then +        REPLY="EDIT" +        reply=( "$list[@]" ) +        break +    elif [[ "$action" = "HELP" ]]; then +        REPLY="HELP" +        reply=( "$list[@]" ) +        break +    fi +done + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-list-draw b/plugins/zsh-navigation-tools/n-list-draw new file mode 100644 index 000000000..ae1e0115f --- /dev/null +++ b/plugins/zsh-navigation-tools/n-list-draw @@ -0,0 +1,133 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-list-draw` to .zshrc +# +# This is an internal function not for direct use + +emulate -L zsh + +zmodload zsh/curses + +setopt typesetsilent extendedglob + +_nlist_print_with_ansi() { +    local win="$1" text="$2" out col chunk Xout +    integer text_offset="$3" max_text_len="$4" text_len=0 no_match=0 nochunk_text_len to_skip_from_chunk to_chop_off_from_chunk before_len + +    # 1 - non-escaped text, 2 - first number in the escaped text, with ; +    # 3 - second number, 4 - text after whole escape text + +    typeset -a c +    c=( black red green yellow blue magenta cyan white ) + +    while [[ -n "$text" && "$no_match" -eq 0 ]]; do +        if [[ "$text" = (#b)([^$'\x1b']#)$'\x1b'\[([0-9](#c0,2))(#B)(\;|)(#b)([0-9](#c0,2))m(*) ]]; then +            # Text for further processing +            text="$match[4]" +            # Text chunk to output now +            out="$match[1]" +            # Save color +            col="$match[2]" +            (( match[3] >= 30 && match[3] <= 37 )) && col="$match[3]" +        else +            out="$text" +            no_match=1 +        fi + +        if [ -n "$out" ]; then +################ Expand tabs ################ +            chunk="$out" +            before_len="$text_len" +            Xout="" + +            while [ -n "$chunk" ]; do +                [[ "$chunk" = (#b)([^$'\t']#)$'\t'(*) ]] && { +                    (( all_text_len=((before_len+${#match[1]})/8+1)*8 )) + +                    Xout+="${(r:all_text_len-before_len:: :)match[1]}" + +                    before_len+=all_text_len-before_len +                    chunk="$match[2]" +                } || { +                    Xout+="$chunk" +                    break +                } +            done +############################################# + +            # Input text length without the current chunk +            nochunk_text_len=text_len +            # Input text length up to current chunk +            text_len+="$#Xout" + +            # Should start displaying with this chunk? +            # I.e. stop skipping left part of the input text? +            if (( text_len > text_offset )); then +                to_skip_from_chunk=text_offset-nochunk_text_len + +                # LEFT - is chunk off the left skip boundary? +1 for 1-based index in string +                (( to_skip_from_chunk > 0 )) && Xout="${Xout[to_skip_from_chunk+1,-1]}" + +                # RIGHT - is text off the screen? +                if (( text_len-text_offset > max_text_len )); then +                    to_chop_off_from_chunk=0+(text_len-text_offset)-max_text_len +                    Xout="${Xout[1,-to_chop_off_from_chunk-1]}" +                fi +                 +                [ -n "$Xout" ] && zcurses string "$win" "$Xout" +            fi +        fi + +        if (( no_match == 0 )); then +            if (( col >= 30 && col <= 37 )); then +                zcurses attr "$win" $c[col-29]/"$background" +            elif [[ "$col" -eq 0 ]]; then +                zcurses attr "$win" "$colorpair" +            fi +        fi +    done +} + +integer highlight="$1" +integer page_height="$2" +integer page_width="$3" +local y_offset="$4" +local x_offset="$5" +local text_offset="$6" +local win="$7" +shift 7 +integer max_text_len=page_width-x_offset + +[[ "$bold" = "0" || "$bold" = "-bold" ]] && bold="-bold" || bold="+bold" +[[ "$active_text" = "underline" || "$active_text" = "reverse" ]] || local active_text="reverse" +# Linux has ncv 18, screen* has ncv 3 - underline won't work properly +(( ${terminfo[ncv]:-0} & 2 )) && active_text="reverse" +# FreeBSD uses TERM=xterm for newcons but doesn't actually support underline +[[ "$TERM" = "xterm" && -z "$DISPLAY" ]] && active_text="reverse" + +integer max_idx=page_height +integer end_idx=max_idx +[ "$end_idx" -gt "$#" ] && end_idx="$#" +integer y=y_offset + +zcurses attr "$win" "$bold" "$colorpair" + +integer i text_len +local text +for (( i=1; i<=end_idx; i++ )); do +    zcurses move "$win" $y "$x_offset" + +    [ "$i" = "$highlight" ] && zcurses attr "$win" +"$active_text" +    _nlist_print_with_ansi "$win" "$@[i]" "$text_offset" "$max_text_len" +    zcurses clear "$win" eol +    [ "$i" = "$highlight" ] && zcurses attr "$win" -"$active_text" + +    y+=1 +done + +if [ "$end_idx" -lt "$max_idx" ]; then +    zcurses move "$win" $y "$x_offset" +    zcurses clear "$win" eol +fi + +zcurses attr "$win" white/black +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-list-input b/plugins/zsh-navigation-tools/n-list-input new file mode 100644 index 000000000..9bafc51b6 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-list-input @@ -0,0 +1,377 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-list-input` to .zshrc +# +# This is an internal function not for direct use + +emulate -L zsh + +zmodload zsh/curses + +setopt typesetsilent + +# Compute first to show index +_nlist_compute_first_to_show_idx() { +    from_what_idx_list_is_shown=0+((current_idx-1)/page_height)*page_height+1 +} + +_nlist_update_from_keywords() { +    keywordisfresh="1" +    if [ "$nkeywords" -gt 0 ]; then +        curkeyword=$(( (curkeyword+1) % (nkeywords+1) )) +        if [ "$curkeyword" -eq "0" ]; then +            buffer="" +        else +            buffer="${keywords[curkeyword]}" +        fi +    fi +} + +_nlist_iterate_theme() { +    themeisfresh="1" +    if [ "$1" = "1" ]; then +        curtheme=$(( (curtheme+1) % (nthemes+1) )) +    else +        curtheme=curtheme-1 +        [ "$curtheme" -lt 0 ] && curtheme=nthemes +    fi + +    if [ "$nthemes" -gt 0 ]; then +        local theme=${themes[curtheme]} +        [ "$curtheme" -eq "0" ] && theme="$backuptheme" + +        colorpair="${theme%/*}" +        bold="${theme##*/}" +        background="${colorpair#*/}" +        zcurses bg main "$colorpair" +        zcurses bg inner "$colorpair" +    fi +} + +_nlist_rotate_buffer() { +    setopt localoptions noglob + +    local -a words +    words=( ${(s: :)buffer} ) +    words=( ${words[-1]} ${words[1,-2]} ) + +    local space="" +    [ "${buffer[-1]}" = " " ] && space=" " + +    buffer="${(j: :)words}$space" +} + +typeset -ga reply +reply=( -1 '' ) +integer current_idx="$1" +integer from_what_idx_list_is_shown="$2" +integer page_height="$3" +integer page_width="$4" +integer last_element="$5" +integer hscroll="$6" +local key="$7" +integer search="$8" +local buffer="$9" +integer uniq_mode="$10" +integer f_mode="$11" + +# +# Listening for input +# + +if [ "$search" = "0" ]; then + +case "$key" in +    (UP|k|$'\C-P') +        # Are there any elements before the current one? +        [ "$current_idx" -gt 1 ] && current_idx=current_idx-1; +        _nlist_compute_first_to_show_idx +        ;; +    (DOWN|j|$'\C-N') +        # Are there any elements after the current one? +        [ "$current_idx" -lt "$last_element" ] && current_idx=current_idx+1; +        _nlist_compute_first_to_show_idx +        ;; +    (PPAGE|$'\b'|$'\C-?'|BACKSPACE) +        current_idx=current_idx-page_height +        [ "$current_idx" -lt 1 ] && current_idx=1; +        _nlist_compute_first_to_show_idx +        ;; +    (NPAGE|" ") +        current_idx=current_idx+page_height +        [ "$current_idx" -gt "$last_element" ] && current_idx=last_element; +        _nlist_compute_first_to_show_idx +        ;; +    ($'\C-U') +        current_idx=current_idx-page_height/2 +        [ "$current_idx" -lt 1 ] && current_idx=1; +        _nlist_compute_first_to_show_idx +        ;; +    ($'\C-D') +        current_idx=current_idx+page_height/2 +        [ "$current_idx" -gt "$last_element" ] && current_idx=last_element; +        _nlist_compute_first_to_show_idx +        ;; +    (HOME|g) +        current_idx=1 +        _nlist_compute_first_to_show_idx +        ;; +    (END|G) +        current_idx=last_element +        _nlist_compute_first_to_show_idx +        ;; +    ($'\n'|ENTER) +        # Is that element selectable? +        # Check for this only when there is no search +        if [[ "$NLIST_SEARCH_BUFFER" != "" || "$NLIST_IS_UNIQ_MODE" -eq 1 || +            ${NLIST_NONSELECTABLE_ELEMENTS[(r)$current_idx]} != $current_idx ]] +        then +            # Save current element in the result variable +            reply=( $current_idx "SELECT" ) +        fi +        ;; +    (H|'?') +        # This event needs to be enabled +        if [[ "${NLIST_ENABLED_EVENTS[(r)HELP]}" = "HELP" ]]; then +            reply=( -1 "HELP" ) +        fi +        ;; +    (F1) +        # This event needs to be enabled +        if [[ "${NLIST_ENABLED_EVENTS[(r)F1]}" = "F1" ]]; then +            reply=( -1 "$key" ) +        fi +        ;; +    (F4|F5|F6|F7|F8|F9|F10|DC) +        # ignore; F2, F3 are used below +        ;; +    (q) +        reply=( -1 "QUIT" ) +        ;; +    (/) +        search=1 +        _nlist_cursor_visibility 1 +        ;; +    ($'\t') +        reply=( $current_idx "LEAVE" ) +        ;; +    ($'\C-L') +        reply=( -1 "REDRAW" ) +        ;; +    (\]) +        [[ "${(t)NLIST_HOP_INDEXES}" = "array" || "${(t)NLIST_HOP_INDEXES}" = "array-local" ]] && +        [ -z "$NLIST_SEARCH_BUFFER" ] && [ "$NLIST_IS_UNIQ_MODE" -eq 0 ] && +        for idx in "${(n)NLIST_HOP_INDEXES[@]}"; do +            if [ "$idx" -gt "$current_idx" ]; then +                current_idx=$idx +                _nlist_compute_first_to_show_idx +                break +            fi +        done +        ;; +    (\[) +        [[ "${(t)NLIST_HOP_INDEXES}" = "array" || "${(t)NLIST_HOP_INDEXES}" = "array-local" ]] && +        [ -z "$NLIST_SEARCH_BUFFER" ] && [ "$NLIST_IS_UNIQ_MODE" -eq 0 ] && +        for idx in "${(nO)NLIST_HOP_INDEXES[@]}"; do +            if [ "$idx" -lt "$current_idx" ]; then +                current_idx=$idx +                _nlist_compute_first_to_show_idx +                break +            fi +        done +        ;; +    ('<'|'{'|LEFT|'h') +        hscroll=hscroll-7 +        [ "$hscroll" -lt 0 ] && hscroll=0 +        ;; +    ('>'|'}'|RIGHT|'l') +        hscroll+=7 +        ;; +    ($'\E') +        buffer="" +        ;; +    (F3) +        if [ "$search" = "1" ]; then +            search=0 +            _nlist_cursor_visibility 0 +        else +            search=1 +            _nlist_cursor_visibility 1 +        fi +        ;; +    (o|$'\C-O') +        uniq_mode=1-uniq_mode +        ;; +    (f|$'\C-F') +        (( f_mode=(f_mode+1) % 3 )) +        ;; +    ($'\x1F'|F2|$'\C-X') +        search=1 +        _nlist_cursor_visibility 1 +        _nlist_update_from_keywords +        ;; +    ($'\C-T') +        _nlist_iterate_theme 1 +        ;; +    ($'\C-G') +        _nlist_iterate_theme 0 +        ;; +    ($'\C-E'|e) +        # This event needs to be enabled +        if [[ "${NLIST_ENABLED_EVENTS[(r)EDIT]}" = "EDIT" ]]; then +            reply=( -1 "EDIT" ) +        fi +        ;; +    ($'\C-A') +        _nlist_rotate_buffer +        ;; +    (*) +        ;; +esac + +else + +case "$key" in +    ($'\n'|ENTER) +        if [ "$NLIST_INSTANT_SELECT" = "1" ]; then +            if [[ "$NLIST_SEARCH_BUFFER" != "" || "$NLIST_IS_UNIQ_MODE" -eq 1 || +                ${NLIST_NONSELECTABLE_ELEMENTS[(r)$current_idx]} != $current_idx ]] +            then +                reply=( $current_idx "SELECT" ) +            fi +        else +            search=0 +            _nlist_cursor_visibility 0 +        fi +        ;; +    ($'\C-L') +        reply=( -1 "REDRAW" ) +        ;; + +    # +    # Slightly limited navigation +    # + +    (UP|$'\C-P') +        [ "$current_idx" -gt 1 ] && current_idx=current_idx-1; +        _nlist_compute_first_to_show_idx +        ;; +    (DOWN|$'\C-N') +        [ "$current_idx" -lt "$last_element" ] && current_idx=current_idx+1; +        _nlist_compute_first_to_show_idx +        ;; +    (PPAGE) +        current_idx=current_idx-page_height +        [ "$current_idx" -lt 1 ] && current_idx=1; +        _nlist_compute_first_to_show_idx +        ;; +    (NPAGE) +        current_idx=current_idx+page_height +        [ "$current_idx" -gt "$last_element" ] && current_idx=last_element; +        _nlist_compute_first_to_show_idx +        ;; +    ($'\C-U') +        current_idx=current_idx-page_height/2 +        [ "$current_idx" -lt 1 ] && current_idx=1; +        _nlist_compute_first_to_show_idx +        ;; +    ($'\C-D') +        current_idx=current_idx+page_height/2 +        [ "$current_idx" -gt "$last_element" ] && current_idx=last_element; +        _nlist_compute_first_to_show_idx +        ;; +    (HOME) +        current_idx=1 +        _nlist_compute_first_to_show_idx +        ;; +    (END) +        current_idx=last_element +        _nlist_compute_first_to_show_idx +        ;; +    (LEFT) +        hscroll=hscroll-7 +        [ "$hscroll" -lt 0 ] && hscroll=0 +        ;; +    (RIGHT) +        hscroll+=7 +        ;; +    (F1) +        # This event needs to be enabled +        if [[ "${NLIST_ENABLED_EVENTS[(r)F1]}" = "F1" ]]; then +            reply=( -1 "$key" ) +        fi +        ;; +    (F4|F5|F6|F7|F8|F9|F10|DC) +        # ignore; F2, F3 are used below +        ;; + +    # +    # The input +    # + +    ($'\b'|$'\C-?'|BACKSPACE) +        buffer="${buffer%?}" +        ;; +    ($'\C-W') +        [ "$buffer" = "${buffer% *}" ] && buffer="" || buffer="${buffer% *}" +        ;; +    ($'\C-K') +        buffer="" +        ;; +    ($'\E') +        buffer="" +        search=0 +        _nlist_cursor_visibility 0 +        ;; +    (F3) +        if [ "$search" = "1" ]; then +            search=0 +            _nlist_cursor_visibility 0 +        else +            search=1 +            _nlist_cursor_visibility 1 +        fi +        ;; +    ($'\C-O') +        uniq_mode=1-uniq_mode +        ;; +    ($'\C-F') +        (( f_mode=(f_mode+1) % 3 )) +        ;; +    ($'\x1F'|F2|$'\C-X') +        _nlist_update_from_keywords +        ;; +    ($'\C-T') +        _nlist_iterate_theme 1 +        ;; +    ($'\C-G') +        _nlist_iterate_theme 0 +        ;; +    ($'\C-E') +        # This event needs to be enabled +        if [[ "${NLIST_ENABLED_EVENTS[(r)EDIT]}" = "EDIT" ]]; then +            reply=( -1 "EDIT" ) +        fi +        ;; +    ($'\C-A') +        _nlist_rotate_buffer +        ;; +    (*) +        if [[ $#key == 1 && $((#key)) -lt 31 ]]; then +            # ignore all other control keys +        else +            buffer+="$key" +        fi +        ;; +esac + +fi + +reply[3]="$current_idx" +reply[4]="$from_what_idx_list_is_shown" +reply[5]="$hscroll" +reply[6]="$search" +reply[7]="$buffer" +reply[8]="$uniq_mode" +reply[9]="$f_mode" + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-options b/plugins/zsh-navigation-tools/n-options new file mode 100644 index 000000000..b1a934015 --- /dev/null +++ b/plugins/zsh-navigation-tools/n-options @@ -0,0 +1,84 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-options` to .zshrc +# +# This function allows to browse and toggle shell's options +# +# Uses n-list + +#emulate -L zsh + +zmodload zsh/curses + +local IFS=" +" + +unset NLIST_COLORING_PATTERN + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-options.conf ] && builtin source ~/.config/znt/n-options.conf + +# TODO restore options +unsetopt localoptions + +integer kshoptionprint=0 +[[ -o kshoptionprint ]] && kshoptionprint=1 +setopt kshoptionprint + +local list +local selected +local option +local state + +# 0 - don't remember, 1 - remember, 2 - init once, then remember +NLIST_REMEMBER_STATE=2 + +local NLIST_GREP_STRING="${1:=}" + +while (( 1 )); do +    list=( `setopt` ) +    list=( "${(M)list[@]:#*${1:=}*}" ) +    list=( "${list[@]:#kshoptionprint*}" ) + +    if [ "$#list" -eq 0 ]; then +        echo "No matching options" +        break +    fi + +    local red=$'\x1b[00;31m' green=$'\x1b[00;32m' reset=$'\x1b[00;00m' +    list=( "${list[@]/ off/${red} off$reset}" ) +    #list=( "${list[@]/ on/${green} on$reset}" ) +    list=( "${(i)list[@]}" ) + +    n-list "${list[@]}" + +    if [ "$REPLY" -gt 0 ]; then +        [[ -o ksharrays ]] && selected="${reply[$(( REPLY - 1 ))]}" || selected="${reply[$REPLY]}" +        option="${selected%% *}" +        state="${selected##* }" + +        if [[ -o globsubst ]]; then +            unsetopt globsubst +            state="${state%$reset}" +            setopt globsubst +        else +            state="${state%$reset}" +        fi + +        # Toggle the option +        if [ "$state" = "on" ]; then +            echo "Setting |$option| to off" +            unsetopt "$option" +        else +            echo "Setting |$option| to on" +            setopt "$option" +        fi +    else +        break +    fi +done + +NLIST_REMEMBER_STATE=0 + +[[ "$kshoptionprint" -eq 0 ]] && unsetopt kshoptionprint + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/n-panelize b/plugins/zsh-navigation-tools/n-panelize new file mode 100644 index 000000000..e2b567f2e --- /dev/null +++ b/plugins/zsh-navigation-tools/n-panelize @@ -0,0 +1,68 @@ +# Copy this file into /usr/share/zsh/site-functions/ +# and add 'autoload n-panelize` to .zshrc +# +# This function somewhat reminds the panelize feature from Midnight Commander +# It allows browsing output of arbitrary command. Example usage: +# v-panelize ls /usr/local/bin +# +# Uses n-list + +emulate -L zsh + +setopt extendedglob +zmodload zsh/curses + +local IFS=" +" + +unset NLIST_COLORING_PATTERN + +[ -f ~/.config/znt/n-list.conf ] && builtin source ~/.config/znt/n-list.conf +[ -f ~/.config/znt/n-panelize.conf ] && builtin source ~/.config/znt/n-panelize.conf + +local list +local selected + +NLIST_REMEMBER_STATE=0 + +if [ -t 0 ]; then +    # Check if there is proper input +    if [ "$#" -lt 1 ]; then +        echo "Usage: n-panelize {command} [option|argument] ... or command | n-panelize" +        return 1 +    fi + +    # This loop makes script faster on some Zsh's (e.g. 5.0.8) +    repeat 1; do +        list=( `"$@"` ) +    done + +    # TODO: $? doesn't reach user +    [ "$?" -eq 127 ] && return $? +else +    # Check if can reattach to terminal +    if [[ ! -c /dev/tty && ! -t 2 ]]; then +        echo "No terminal available (no /dev/tty)" +        return 1 +    fi + +    # This loop makes script faster on some Zsh's (e.g. 5.0.8) +    repeat 1; do +        list=( "${(@f)"$(<&0)"}" ) +    done + +    if [[ ! -c /dev/tty ]]; then +        exec <&2 +    else +        exec </dev/tty +    fi +fi + +n-list "${list[@]}" + +if [ "$REPLY" -gt 0 ]; then +    selected="$reply[REPLY]" +    print -zr "# $selected" +fi + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/znt-cd-widget b/plugins/zsh-navigation-tools/znt-cd-widget new file mode 100644 index 000000000..6d1be6bff --- /dev/null +++ b/plugins/zsh-navigation-tools/znt-cd-widget @@ -0,0 +1,8 @@ +autoload znt-usetty-wrapper n-cd +local NLIST_START_IN_SEARCH_MODE=0 +local NLIST_START_IN_UNIQ_MODE=0 + +znt-usetty-wrapper n-cd "$@" + +unset NLIST_START_IN_SEARCH_MODE +unset NLIST_START_IN_UNIQ_MODE diff --git a/plugins/zsh-navigation-tools/znt-history-widget b/plugins/zsh-navigation-tools/znt-history-widget new file mode 100644 index 000000000..479c2211b --- /dev/null +++ b/plugins/zsh-navigation-tools/znt-history-widget @@ -0,0 +1,22 @@ +autoload znt-usetty-wrapper n-history +local NLIST_START_IN_SEARCH_MODE=1 +local NLIST_START_IN_UNIQ_MODE=1 + +# Only if current $BUFFER doesn't come from history +if [ "$HISTCMD" = "$HISTNO" ]; then +    () { +        setopt localoptions extendedglob +        local -a match mbegin mend +        local MATCH; integer MBEGIN MEND + +        [ -n "$BUFFER" ] && BUFFER="${BUFFER%% ##} " +    } + +    local NLIST_SET_SEARCH_TO="$BUFFER" +fi + +znt-usetty-wrapper n-history "$@" + +unset NLIST_START_IN_SEARCH_MODE +unset NLIST_START_IN_UNIQ_MODE +unset NLIST_SET_SEARCH_TO diff --git a/plugins/zsh-navigation-tools/znt-kill-widget b/plugins/zsh-navigation-tools/znt-kill-widget new file mode 100644 index 000000000..1aff7bb50 --- /dev/null +++ b/plugins/zsh-navigation-tools/znt-kill-widget @@ -0,0 +1,8 @@ +autoload znt-usetty-wrapper n-kill +local NLIST_START_IN_SEARCH_MODE=0 +local NLIST_START_IN_UNIQ_MODE=0 + +znt-usetty-wrapper n-kill "$@" + +unset NLIST_START_IN_SEARCH_MODE +unset NLIST_START_IN_UNIQ_MODE diff --git a/plugins/zsh-navigation-tools/znt-tmux.zsh b/plugins/zsh-navigation-tools/znt-tmux.zsh new file mode 100755 index 000000000..6a96e97a1 --- /dev/null +++ b/plugins/zsh-navigation-tools/znt-tmux.zsh @@ -0,0 +1,50 @@ +#!/usr/bin/env zsh + +# Copyright (c) 2016, Zsolt Lengyel +# Modifications copyright (c) 2016, Sebastian Gniazdowski + +# +# This script opens a new, temporary tmux pane and runs n-history. When +# a selection is made, the result (history entry) is pasted back into +# original tmux pane, and the temporary pane is closed. This allows to +# use local history on remote machines. +# +# To use, put this line to your ~/.tmux.conf. The tool is invoked with: +# Ctrl+b h +# +# bind h run-shell -b "$ZNT_REPO_DIR/znt-tmux.zsh" +# + +# get and save the current active tmux pane id +active_pane=$(tmux display -p -F ':#{session_id}:#I:#P:#{pane_active}:#{window_active}:#{session_attached}' ) +a_active_pane=("${(@s/:/)active_pane}") + +active_session=${a_active_pane[2]//$} +active_window=$a_active_pane[3] +active_pane=$a_active_pane[4] + +# set variables for upcoming window +tmux setenv -t $active_session:$active_window.$active_pane "ZNT_TMUX_MODE" 1 +tmux setenv -t $active_session:$active_window.$active_pane "ZNT_TMUX_ORIGIN_SESSION" "$active_session" +tmux setenv -t $active_session:$active_window.$active_pane "ZNT_TMUX_ORIGIN_WINDOW" "$active_window" +tmux setenv -t $active_session:$active_window.$active_pane "ZNT_TMUX_ORIGIN_PANE" "$active_pane" + +# create a new window in the active session and call it znt-hist +tmux new-window -t $active_session: -n znt-hist + +# unset the variables, so only above single window has them +tmux setenv -u -t $active_session:$active_window.$active_pane "ZNT_TMUX_MODE" +tmux setenv -u -t $active_session:$active_window.$active_pane "ZNT_TMUX_ORIGIN_SESSION" +tmux setenv -u -t $active_session:$active_window.$active_pane "ZNT_TMUX_ORIGIN_WINDOW" +tmux setenv -u -t $active_session:$active_window.$active_pane "ZNT_TMUX_ORIGIN_PANE" + +# znt's session id +znt_active_pane=$(tmux display -p -F ':#{session_id}:#I:#P:#{pane_active}:#{window_active}:#{session_attached}' ) +znt_a_active_pane=("${(@s/:/)znt_active_pane}") + +znt_active_session=${znt_a_active_pane[2]//$} +znt_active_window=$znt_a_active_pane[3] +znt_active_pane=$znt_a_active_pane[4] + +# call znt +tmux send -t "$znt_active_session:$znt_active_window.$znt_active_pane" n-history ENTER diff --git a/plugins/zsh-navigation-tools/znt-usetty-wrapper b/plugins/zsh-navigation-tools/znt-usetty-wrapper new file mode 100644 index 000000000..19c5ac8b6 --- /dev/null +++ b/plugins/zsh-navigation-tools/znt-usetty-wrapper @@ -0,0 +1,40 @@ +emulate -L zsh + +zmodload zsh/curses + +test_fd0() { +    true <&0 +} + +local restore=0 FD + +# Reattach to terminal +if [ ! -t 0 ]; then +    # Check if can reattach to terminal in any way +    if [[ ! -c /dev/tty && ! -t 2 ]]; then +        echo "No terminal available (no /dev/tty and no terminal at stderr)" +        return 1 +    fi + +    if test_fd0 2>/dev/null; then +        exec {FD}<&0 +        restore=2 +    else +        restore=1 +    fi + +    if [[ ! -c /dev/tty ]]; then +        exec <&2 +    else +        exec </dev/tty +    fi +fi + +# Run the command +"$@" + +# Restore FD state +(( restore == 1 )) && exec <&- +(( restore == 2 )) && exec <&$FD && exec {FD}<&- + +# vim: set filetype=zsh: diff --git a/plugins/zsh-navigation-tools/zsh-navigation-tools.plugin.zsh b/plugins/zsh-navigation-tools/zsh-navigation-tools.plugin.zsh new file mode 100644 index 000000000..32b4ca064 --- /dev/null +++ b/plugins/zsh-navigation-tools/zsh-navigation-tools.plugin.zsh @@ -0,0 +1,55 @@ +#!/usr/bin/env zsh + +0="${(%):-%N}" # this gives immunity to functionargzero being unset +export ZNT_REPO_DIR="${0%/*}" +export ZNT_CONFIG_DIR="$HOME/.config/znt" + +# +# Copy configs +# + +if [[ ! -d "$HOME/.config" ]]; then +    command mkdir "$HOME/.config" +fi + +if [[ ! -d "$ZNT_CONFIG_DIR" ]]; then +    command mkdir "$ZNT_CONFIG_DIR" +fi + +# 9 files +unset __ZNT_CONFIG_FILES +typeset -ga __ZNT_CONFIG_FILES +set +A __ZNT_CONFIG_FILES n-aliases.conf n-env.conf n-history.conf n-list.conf n-panelize.conf n-cd.conf n-functions.conf n-kill.conf n-options.conf + +# Check for random 2 files if they exist +# This will shift 0 - 7 elements +shift $(( RANDOM % 8 )) __ZNT_CONFIG_FILES +if [[ ! -f "$ZNT_CONFIG_DIR/${__ZNT_CONFIG_FILES[1]}" || ! -f "$ZNT_CONFIG_DIR/${__ZNT_CONFIG_FILES[2]}" ]]; then +    # Something changed - examine every file +    set +A __ZNT_CONFIG_FILES n-aliases.conf n-env.conf n-history.conf n-list.conf n-panelize.conf n-cd.conf n-functions.conf n-kill.conf n-options.conf +    unset __ZNT_CONFIG_FILE +    typeset -g __ZNT_CONFIG_FILE +    for __ZNT_CONFIG_FILE in "${__ZNT_CONFIG_FILES[@]}"; do +        if [[ ! -f "$ZNT_CONFIG_DIR/$__ZNT_CONFIG_FILE" ]]; then +            command cp "$ZNT_REPO_DIR/.config/znt/$__ZNT_CONFIG_FILE" "$ZNT_CONFIG_DIR" +        fi +    done +    unset __ZNT_CONFIG_FILE +fi + +unset __ZNT_CONFIG_FILES + +# +# Load functions +# + +autoload n-aliases n-cd n-env n-functions n-history n-kill n-list n-list-draw n-list-input n-options n-panelize n-help +autoload znt-usetty-wrapper znt-history-widget znt-cd-widget znt-kill-widget +alias naliases=n-aliases ncd=n-cd nenv=n-env nfunctions=n-functions nhistory=n-history +alias nkill=n-kill noptions=n-options npanelize=n-panelize nhelp=n-help + +zle -N znt-history-widget +bindkey '^R' znt-history-widget +setopt AUTO_PUSHD HIST_IGNORE_DUPS PUSHD_IGNORE_DUPS +zstyle ':completion::complete:n-kill::bits' matcher 'r:|=** l:|=*' +  | 
