per-directory-history.zsh 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. #!/usr/bin/env zsh
  2. #
  3. # This is a implementation of per directory history for zsh, some
  4. # implementations of which exist in bash[1,2]. It also implements
  5. # a per-directory-history-toggle-history function to change from using the
  6. # directory history to using the global history. In both cases the history is
  7. # always saved to both the global history and the directory history, so the
  8. # toggle state will not effect the saved histories. Being able to switch
  9. # between global and directory histories on the fly is a novel feature as far
  10. # as I am aware.
  11. #
  12. #-------------------------------------------------------------------------------
  13. # Configuration
  14. #-------------------------------------------------------------------------------
  15. #
  16. # HISTORY_BASE a global variable that defines the base directory in which the
  17. # directory histories are stored
  18. #
  19. #-------------------------------------------------------------------------------
  20. # History
  21. #-------------------------------------------------------------------------------
  22. #
  23. # The idea/inspiration for a per directory history is from Stewart MacArthur[1]
  24. # and Dieter[2], the implementation idea is from Bart Schaefer on the the zsh
  25. # mailing list[3]. The implementation is by Jim Hester in September 2012.
  26. #
  27. # [1]: http://www.compbiome.com/2010/07/bash-per-directory-bash-history.html
  28. # [2]: http://dieter.plaetinck.be/per_directory_bash
  29. # [3]: http://www.zsh.org/mla/users/1997/msg00226.html
  30. #
  31. ################################################################################
  32. #
  33. # Copyright (c) 2014 Jim Hester
  34. #
  35. # This software is provided 'as-is', without any express or implied warranty.
  36. # In no event will the authors be held liable for any damages arising from the
  37. # use of this software.
  38. #
  39. # Permission is granted to anyone to use this software for any purpose,
  40. # including commercial applications, and to alter it and redistribute it
  41. # freely, subject to the following restrictions:
  42. #
  43. # 1. The origin of this software must not be misrepresented; you must not claim
  44. # that you wrote the original software. If you use this software in a product,
  45. # an acknowledgment in the product documentation would be appreciated but is
  46. # not required.
  47. #
  48. # 2. Altered source versions must be plainly marked as such, and must not be
  49. # misrepresented as being the original software.
  50. #
  51. # 3. This notice may not be removed or altered from any source distribution..
  52. #
  53. ################################################################################
  54. #-------------------------------------------------------------------------------
  55. # configuration, the base under which the directory histories are stored
  56. #-------------------------------------------------------------------------------
  57. [[ -z $HISTORY_BASE ]] && HISTORY_BASE="$HOME/.directory_history"
  58. [[ -z $HISTORY_START_WITH_GLOBAL ]] && HISTORY_START_WITH_GLOBAL=false
  59. [[ -z $PER_DIRECTORY_HISTORY_TOGGLE ]] && PER_DIRECTORY_HISTORY_TOGGLE='^G'
  60. #-------------------------------------------------------------------------------
  61. # toggle global/directory history used for searching - ctrl-G by default
  62. #-------------------------------------------------------------------------------
  63. function per-directory-history-toggle-history() {
  64. if [[ $_per_directory_history_is_global == true ]]; then
  65. _per-directory-history-set-directory-history
  66. _per_directory_history_is_global=false
  67. zle -I
  68. echo "using local history"
  69. else
  70. _per-directory-history-set-global-history
  71. _per_directory_history_is_global=true
  72. zle -I
  73. echo "using global history"
  74. fi
  75. }
  76. autoload per-directory-history-toggle-history
  77. zle -N per-directory-history-toggle-history
  78. bindkey $PER_DIRECTORY_HISTORY_TOGGLE per-directory-history-toggle-history
  79. bindkey -M vicmd $PER_DIRECTORY_HISTORY_TOGGLE per-directory-history-toggle-history
  80. #-------------------------------------------------------------------------------
  81. # implementation details
  82. #-------------------------------------------------------------------------------
  83. _per_directory_history_directory="$HISTORY_BASE${PWD:A}/history"
  84. function _per-directory-history-change-directory() {
  85. _per_directory_history_directory="$HISTORY_BASE${PWD:A}/history"
  86. mkdir -p ${_per_directory_history_directory:h}
  87. if [[ $_per_directory_history_is_global == false ]]; then
  88. #save to the global history
  89. fc -AI $HISTFILE
  90. #save history to previous file
  91. local prev="$HISTORY_BASE${OLDPWD:A}/history"
  92. mkdir -p ${prev:h}
  93. fc -AI $prev
  94. #discard previous directory's history
  95. local original_histsize=$HISTSIZE
  96. HISTSIZE=0
  97. HISTSIZE=$original_histsize
  98. #read history in new file
  99. if [[ -e $_per_directory_history_directory ]]; then
  100. fc -R $_per_directory_history_directory
  101. fi
  102. fi
  103. }
  104. function _per-directory-history-addhistory() {
  105. # respect hist_ignore_space
  106. if [[ -o hist_ignore_space ]] && [[ "$1" == \ * ]]; then
  107. true
  108. else
  109. print -Sr -- "${1%%$'\n'}"
  110. # instantly write history if set options require it.
  111. if [[ -o share_history ]] || \
  112. [[ -o inc_append_history ]] || \
  113. [[ -o inc_append_history_time ]]; then
  114. fc -AI $HISTFILE
  115. fc -AI $_per_directory_history_directory
  116. fi
  117. fc -p $_per_directory_history_directory
  118. fi
  119. }
  120. function _per-directory-history-precmd() {
  121. if [[ $_per_directory_history_initialized == false ]]; then
  122. _per_directory_history_initialized=true
  123. if [[ $HISTORY_START_WITH_GLOBAL == true ]]; then
  124. _per-directory-history-set-global-history
  125. _per_directory_history_is_global=true
  126. else
  127. _per-directory-history-set-directory-history
  128. _per_directory_history_is_global=false
  129. fi
  130. fi
  131. }
  132. function _per-directory-history-set-directory-history() {
  133. fc -AI $HISTFILE
  134. local original_histsize=$HISTSIZE
  135. HISTSIZE=0
  136. HISTSIZE=$original_histsize
  137. if [[ -e "$_per_directory_history_directory" ]]; then
  138. fc -R "$_per_directory_history_directory"
  139. fi
  140. }
  141. function _per-directory-history-set-global-history() {
  142. fc -AI $_per_directory_history_directory
  143. local original_histsize=$HISTSIZE
  144. HISTSIZE=0
  145. HISTSIZE=$original_histsize
  146. if [[ -e "$HISTFILE" ]]; then
  147. fc -R "$HISTFILE"
  148. fi
  149. }
  150. mkdir -p ${_per_directory_history_directory:h}
  151. #add functions to the exec list for chpwd and zshaddhistory
  152. autoload -U add-zsh-hook
  153. add-zsh-hook chpwd _per-directory-history-change-directory
  154. add-zsh-hook zshaddhistory _per-directory-history-addhistory
  155. add-zsh-hook precmd _per-directory-history-precmd
  156. # set initialized flag to false
  157. _per_directory_history_initialized=false