| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | #!/usr/bin/env zsh
 | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # This is a implementation of per directory history for zsh, some | 
					
						
							|  |  |  | # implementations of which exist in bash[1,2].  It also implements | 
					
						
							|  |  |  | # a per-directory-history-toggle-history function to change from using the | 
					
						
							|  |  |  | # directory history to using the global history.  In both cases the history is | 
					
						
							|  |  |  | # always saved to both the global history and the directory history, so the | 
					
						
							|  |  |  | # toggle state will not effect the saved histories.  Being able to switch | 
					
						
							|  |  |  | # between global and directory histories on the fly is a novel feature as far | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # as I am aware. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # Configuration | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # HISTORY_BASE a global variable that defines the base directory in which the | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # directory histories are stored | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # History | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # The idea/inspiration for a per directory history is from Stewart MacArthur[1] | 
					
						
							|  |  |  | # and Dieter[2], the implementation idea is from Bart Schaefer on the the zsh | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # mailing list[3].  The implementation is by Jim Hester in September 2012. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # [1]: http://www.compbiome.com/2010/07/bash-per-directory-bash-history.html | 
					
						
							|  |  |  | # [2]: http://dieter.plaetinck.be/per_directory_bash | 
					
						
							| 
									
										
										
										
											2019-03-03 20:32:59 +01:00
										 |  |  | # [3]: http://www.zsh.org/mla/users/1997/msg00226.html | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2016-10-04 13:56:25 +02:00
										 |  |  | # Copyright (c) 2014 Jim Hester | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # This software is provided 'as-is', without any express or implied warranty.  | 
					
						
							|  |  |  | # In no event will the authors be held liable for any damages arising from the | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # use of this software. | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # Permission is granted to anyone to use this software for any purpose, | 
					
						
							|  |  |  | # including commercial applications, and to alter it and redistribute it | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # freely, subject to the following restrictions: | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # 1. The origin of this software must not be misrepresented; you must not claim | 
					
						
							|  |  |  | # that you wrote the original software. If you use this software in a product, | 
					
						
							|  |  |  | # an acknowledgment in the product documentation would be appreciated but is | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # not required. | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | # 2. Altered source versions must be plainly marked as such, and must not be | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | # misrepresented as being the original software. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # 3. This notice may not be removed or altered from any source distribution.. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # configuration, the base under which the directory histories are stored | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [[ -z $HISTORY_BASE ]] && HISTORY_BASE="$HOME/.directory_history" | 
					
						
							| 
									
										
										
										
											2016-10-04 13:56:25 +02:00
										 |  |  | [[ -z $PER_DIRECTORY_HISTORY_TOGGLE ]] && PER_DIRECTORY_HISTORY_TOGGLE='^G' | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # toggle global/directory history used for searching - ctrl-G by default | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function per-directory-history-toggle-history() { | 
					
						
							|  |  |  |   if [[ $_per_directory_history_is_global == true ]]; then | 
					
						
							|  |  |  |     _per-directory-history-set-directory-history | 
					
						
							|  |  |  |     print -n "\nusing local history" | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     _per-directory-history-set-global-history | 
					
						
							|  |  |  |     print -n "\nusing global history" | 
					
						
							|  |  |  |   fi | 
					
						
							|  |  |  |   zle .push-line | 
					
						
							|  |  |  |   zle .accept-line | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | autoload per-directory-history-toggle-history | 
					
						
							|  |  |  | zle -N per-directory-history-toggle-history | 
					
						
							| 
									
										
										
										
											2016-10-04 13:56:25 +02:00
										 |  |  | bindkey $PER_DIRECTORY_HISTORY_TOGGLE per-directory-history-toggle-history | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | # implementation details | 
					
						
							|  |  |  | #------------------------------------------------------------------------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _per_directory_history_directory="$HISTORY_BASE${PWD:A}/history" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function _per-directory-history-change-directory() { | 
					
						
							|  |  |  |   _per_directory_history_directory="$HISTORY_BASE${PWD:A}/history" | 
					
						
							|  |  |  |   mkdir -p ${_per_directory_history_directory:h} | 
					
						
							|  |  |  |   if [[ $_per_directory_history_is_global == false ]]; then | 
					
						
							|  |  |  |     #save to the global history | 
					
						
							|  |  |  |     fc -AI $HISTFILE | 
					
						
							|  |  |  |     #save history to previous file | 
					
						
							|  |  |  |     local prev="$HISTORY_BASE${OLDPWD:A}/history" | 
					
						
							|  |  |  |     mkdir -p ${prev:h} | 
					
						
							|  |  |  |     fc -AI $prev | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #discard previous directory's history | 
					
						
							|  |  |  |     local original_histsize=$HISTSIZE | 
					
						
							|  |  |  |     HISTSIZE=0 | 
					
						
							|  |  |  |     HISTSIZE=$original_histsize | 
					
						
							| 
									
										
										
										
											2013-03-13 13:07:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  |     #read history in new file | 
					
						
							|  |  |  |     if [[ -e $_per_directory_history_directory ]]; then | 
					
						
							|  |  |  |       fc -R $_per_directory_history_directory | 
					
						
							|  |  |  |     fi | 
					
						
							|  |  |  |   fi | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function _per-directory-history-addhistory() { | 
					
						
							| 
									
										
										
										
											2019-03-03 20:32:59 +01:00
										 |  |  |   # respect hist_ignore_space | 
					
						
							|  |  |  |   if [[ -o hist_ignore_space ]] && [[ "$1" == \ * ]]; then | 
					
						
							|  |  |  |       true | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |       print -Sr -- "${1%%$'\n'}" | 
					
						
							|  |  |  |       fc -p $_per_directory_history_directory | 
					
						
							|  |  |  |   fi | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function _per-directory-history-set-directory-history() { | 
					
						
							|  |  |  |   if [[ $_per_directory_history_is_global == true ]]; then | 
					
						
							|  |  |  |     fc -AI $HISTFILE | 
					
						
							|  |  |  |     local original_histsize=$HISTSIZE | 
					
						
							|  |  |  |     HISTSIZE=0 | 
					
						
							|  |  |  |     HISTSIZE=$original_histsize | 
					
						
							|  |  |  |     if [[ -e "$_per_directory_history_directory" ]]; then | 
					
						
							|  |  |  |       fc -R "$_per_directory_history_directory" | 
					
						
							|  |  |  |     fi | 
					
						
							|  |  |  |   fi | 
					
						
							|  |  |  |   _per_directory_history_is_global=false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | function _per-directory-history-set-global-history() { | 
					
						
							|  |  |  |   if [[ $_per_directory_history_is_global == false ]]; then | 
					
						
							|  |  |  |     fc -AI $_per_directory_history_directory | 
					
						
							|  |  |  |     local original_histsize=$HISTSIZE | 
					
						
							|  |  |  |     HISTSIZE=0 | 
					
						
							|  |  |  |     HISTSIZE=$original_histsize | 
					
						
							|  |  |  |     if [[ -e "$HISTFILE" ]]; then | 
					
						
							|  |  |  |       fc -R "$HISTFILE" | 
					
						
							|  |  |  |     fi | 
					
						
							|  |  |  |   fi | 
					
						
							|  |  |  |   _per_directory_history_is_global=true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #add functions to the exec list for chpwd and zshaddhistory | 
					
						
							| 
									
										
										
										
											2016-10-04 13:56:25 +02:00
										 |  |  | autoload -U add-zsh-hook | 
					
						
							|  |  |  | add-zsh-hook chpwd _per-directory-history-change-directory | 
					
						
							|  |  |  | add-zsh-hook zshaddhistory _per-directory-history-addhistory | 
					
						
							| 
									
										
										
										
											2013-02-28 10:28:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #start in directory mode | 
					
						
							|  |  |  | mkdir -p ${_per_directory_history_directory:h} | 
					
						
							|  |  |  | _per_directory_history_is_global=true | 
					
						
							|  |  |  | _per-directory-history-set-directory-history |