per-directory-history.zsh 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  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. print -n "\nusing local history"
  68. else
  69. _per-directory-history-set-global-history
  70. _per_directory_history_is_global=true
  71. print -n "\nusing global history"
  72. fi
  73. zle .push-line
  74. zle .accept-line
  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. #-------------------------------------------------------------------------------
  80. # implementation details
  81. #-------------------------------------------------------------------------------
  82. _per_directory_history_directory="$HISTORY_BASE${PWD:A}/history"
  83. function _per-directory-history-change-directory() {
  84. _per_directory_history_directory="$HISTORY_BASE${PWD:A}/history"
  85. mkdir -p ${_per_directory_history_directory:h}
  86. if [[ $_per_directory_history_is_global == false ]]; then
  87. #save to the global history
  88. fc -AI $HISTFILE
  89. #save history to previous file
  90. local prev="$HISTORY_BASE${OLDPWD:A}/history"
  91. mkdir -p ${prev:h}
  92. fc -AI $prev
  93. #discard previous directory's history
  94. local original_histsize=$HISTSIZE
  95. HISTSIZE=0
  96. HISTSIZE=$original_histsize
  97. #read history in new file
  98. if [[ -e $_per_directory_history_directory ]]; then
  99. fc -R $_per_directory_history_directory
  100. fi
  101. fi
  102. }
  103. function _per-directory-history-addhistory() {
  104. # respect hist_ignore_space
  105. if [[ -o hist_ignore_space ]] && [[ "$1" == \ * ]]; then
  106. true
  107. else
  108. print -Sr -- "${1%%$'\n'}"
  109. # instantly write history if set options require it.
  110. if [[ -o share_history ]] || \
  111. [[ -o inc_append_history ]] || \
  112. [[ -o inc_append_history_time ]]; then
  113. fc -AI $HISTFILE
  114. fc -AI $_per_directory_history_directory
  115. fi
  116. fc -p $_per_directory_history_directory
  117. fi
  118. }
  119. function _per-directory-history-precmd() {
  120. if [[ $_per_directory_history_initialized == false ]]; then
  121. _per_directory_history_initialized=true
  122. if [[ $HISTORY_START_WITH_GLOBAL == true ]]; then
  123. _per-directory-history-set-global-history
  124. _per_directory_history_is_global=true
  125. else
  126. _per-directory-history-set-directory-history
  127. _per_directory_history_is_global=false
  128. fi
  129. fi
  130. }
  131. function _per-directory-history-set-directory-history() {
  132. fc -AI $HISTFILE
  133. local original_histsize=$HISTSIZE
  134. HISTSIZE=0
  135. HISTSIZE=$original_histsize
  136. if [[ -e "$_per_directory_history_directory" ]]; then
  137. fc -R "$_per_directory_history_directory"
  138. fi
  139. }
  140. function _per-directory-history-set-global-history() {
  141. fc -AI $_per_directory_history_directory
  142. local original_histsize=$HISTSIZE
  143. HISTSIZE=0
  144. HISTSIZE=$original_histsize
  145. if [[ -e "$HISTFILE" ]]; then
  146. fc -R "$HISTFILE"
  147. fi
  148. }
  149. mkdir -p ${_per_directory_history_directory:h}
  150. #add functions to the exec list for chpwd and zshaddhistory
  151. autoload -U add-zsh-hook
  152. add-zsh-hook chpwd _per-directory-history-change-directory
  153. add-zsh-hook zshaddhistory _per-directory-history-addhistory
  154. add-zsh-hook precmd _per-directory-history-precmd
  155. # set initialized flag to false
  156. _per_directory_history_initialized=false