Mercurial > emacs
annotate lisp/eshell/em-dirs.el @ 108694:b27c565c6f1a
* etc/TODO: prog-mode updates.
author | Glenn Morris <rgm@gnu.org> |
---|---|
date | Tue, 18 May 2010 19:56:44 -0700 |
parents | ed16fdd2685a |
children | f57f72bb4757 376148b31b5e |
rev | line source |
---|---|
38414
67b464da13ec
Some fixes to follow coding conventions.
Pavel Janík <Pavel@Janik.cz>
parents:
37661
diff
changeset
|
1 ;;; em-dirs.el --- directory navigation commands |
29876 | 2 |
95152
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
3 ;; Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, |
106815 | 4 ;; 2008, 2009, 2010 Free Software Foundation, Inc. |
29876 | 5 |
32526 | 6 ;; Author: John Wiegley <johnw@gnu.org> |
7 | |
29876 | 8 ;; This file is part of GNU Emacs. |
9 | |
94661
b5b0801a7637
Switch to recommended form of GPLv3 permissions notice.
Glenn Morris <rgm@gnu.org>
parents:
93975
diff
changeset
|
10 ;; GNU Emacs is free software: you can redistribute it and/or modify |
29876 | 11 ;; it under the terms of the GNU General Public License as published by |
94661
b5b0801a7637
Switch to recommended form of GPLv3 permissions notice.
Glenn Morris <rgm@gnu.org>
parents:
93975
diff
changeset
|
12 ;; the Free Software Foundation, either version 3 of the License, or |
b5b0801a7637
Switch to recommended form of GPLv3 permissions notice.
Glenn Morris <rgm@gnu.org>
parents:
93975
diff
changeset
|
13 ;; (at your option) any later version. |
29876 | 14 |
15 ;; GNU Emacs is distributed in the hope that it will be useful, | |
16 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of | |
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
18 ;; GNU General Public License for more details. | |
19 | |
20 ;; You should have received a copy of the GNU General Public License | |
94661
b5b0801a7637
Switch to recommended form of GPLv3 permissions notice.
Glenn Morris <rgm@gnu.org>
parents:
93975
diff
changeset
|
21 ;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. |
29876 | 22 |
23 ;;; Commentary: | |
24 | |
25 ;; The only special feature that Eshell offers in the last-dir-ring. | |
26 ;; To view the ring, enter: | |
27 ;; | |
28 ;; cd = | |
29 ;; | |
30 ;; Changing to an index within the ring is done using: | |
31 ;; | |
32 ;; cd - ; same as cd -0 | |
33 ;; cd -4 | |
34 ;; | |
35 ;; Or, it is possible to change the first member in the ring which | |
36 ;; matches a regexp: | |
37 ;; | |
38 ;; cd =bcc ; change to the last directory visited containing "bcc" | |
39 ;; | |
40 ;; This ring is maintained automatically, and is persisted across | |
41 ;; Eshell sessions. It is a separate mechanism from `pushd' and | |
42 ;; `popd', and the two may be used at the same time. | |
43 | |
87066
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
44 ;;; Code: |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
45 |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
46 (require 'eshell) |
29876 | 47 (require 'ring) |
48 (require 'esh-opt) | |
49 | |
95152
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
50 ;;;###autoload |
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
51 (eshell-defgroup eshell-dirs nil |
87066
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
52 "Directory navigation involves changing directories, examining the |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
53 current directory, maintaining a directory stack, and also keeping |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
54 track of a history of the last directory locations the user was in. |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
55 Emacs does provide standard Lisp definitions of `pwd' and `cd', but |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
56 they lack somewhat in feel from the typical shell equivalents." |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
57 :tag "Directory navigation" |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
58 :group 'eshell-module) |
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
59 |
29876 | 60 ;;; User Variables: |
61 | |
62 (defcustom eshell-dirs-load-hook '(eshell-dirs-initialize) | |
63 "*A hook that gets run when `eshell-dirs' is loaded." | |
64 :type 'hook | |
65 :group 'eshell-dirs) | |
66 | |
67 (defcustom eshell-pwd-convert-function (if (eshell-under-windows-p) | |
68 'expand-file-name | |
69 'identity) | |
70 "*The function used to normalize the value of Eshell's `pwd'. | |
71 The value returned by `pwd' is also used when recording the | |
72 last-visited directory in the last-dir-ring, so it will affect the | |
73 form of the list used by 'cd ='." | |
74 :type '(radio (function-item file-truename) | |
75 (function-item expand-file-name) | |
76 (function-item identity) | |
77 (function :tag "Other")) | |
78 :group 'eshell-dirs) | |
79 | |
80 (defcustom eshell-ask-to-save-last-dir 'always | |
81 "*Determine if the last-dir-ring should be automatically saved. | |
82 The last-dir-ring is always preserved when exiting an Eshell buffer. | |
83 However, when Emacs is being shut down, this variable determines | |
84 whether to prompt the user, or just save the ring. | |
85 If set to nil, it means never ask whether to save the last-dir-ring. | |
86 If set to t, always ask if any Eshell buffers are open at exit time. | |
87 If set to `always', the list-dir-ring will always be saved, silently." | |
88 :type '(choice (const :tag "Never" nil) | |
89 (const :tag "Ask" t) | |
90 (const :tag "Always save" always)) | |
91 :group 'eshell-dirs) | |
92 | |
93 (defcustom eshell-cd-shows-directory nil | |
94 "*If non-nil, using `cd' will report the directory it changes to." | |
95 :type 'boolean | |
96 :group 'eshell-dirs) | |
97 | |
98 (defcustom eshell-cd-on-directory t | |
99 "*If non-nil, do a cd if a directory is in command position." | |
100 :type 'boolean | |
101 :group 'eshell-dirs) | |
102 | |
103 (defcustom eshell-directory-change-hook nil | |
104 "*A hook to run when the current directory changes." | |
105 :type 'hook | |
106 :group 'eshell-dirs) | |
107 | |
108 (defcustom eshell-list-files-after-cd nil | |
109 "*If non-nil, call \"ls\" with any remaining args after doing a cd. | |
110 This is provided for convenience, since the same effect is easily | |
111 achieved by adding a function to `eshell-directory-change-hook' that | |
112 calls \"ls\" and references `eshell-last-arguments'." | |
113 :type 'boolean | |
114 :group 'eshell-dirs) | |
115 | |
116 (defcustom eshell-pushd-tohome nil | |
117 "*If non-nil, make pushd with no arg behave as 'pushd ~' (like `cd'). | |
118 This mirrors the optional behavior of tcsh." | |
119 :type 'boolean | |
120 :group 'eshell-dirs) | |
121 | |
122 (defcustom eshell-pushd-dextract nil | |
123 "*If non-nil, make \"pushd +n\" pop the nth dir to the stack top. | |
124 This mirrors the optional behavior of tcsh." | |
125 :type 'boolean | |
126 :group 'eshell-dirs) | |
127 | |
128 (defcustom eshell-pushd-dunique nil | |
129 "*If non-nil, make pushd only add unique directories to the stack. | |
130 This mirrors the optional behavior of tcsh." | |
131 :type 'boolean | |
132 :group 'eshell-dirs) | |
133 | |
134 (defcustom eshell-dirtrack-verbose t | |
135 "*If non-nil, show the directory stack following directory change. | |
136 This is effective only if directory tracking is enabled." | |
137 :type 'boolean | |
138 :group 'eshell-dirs) | |
139 | |
140 (defcustom eshell-last-dir-ring-file-name | |
104994
48428e99bf83
* eshell/em-hist.el:
Chong Yidong <cyd@stupidchicken.com>
parents:
100908
diff
changeset
|
141 (expand-file-name "lastdir" eshell-directory-name) |
29876 | 142 "*If non-nil, name of the file to read/write the last-dir-ring. |
143 See also `eshell-read-last-dir-ring' and `eshell-write-last-dir-ring'. | |
144 If it is nil, the last-dir-ring will not be written to disk." | |
145 :type 'file | |
146 :group 'eshell-dirs) | |
147 | |
148 (defcustom eshell-last-dir-ring-size 32 | |
149 "*If non-nil, the size of the directory history ring. | |
150 This ring is added to every time `cd' or `pushd' is used. It simply | |
151 stores the most recent directory locations Eshell has been in. To | |
152 return to the most recent entry, use 'cd -' (equivalent to 'cd -0'). | |
153 To return to an older entry, use 'cd -N', where N is an integer less | |
154 than `eshell-last-dir-ring-size'. To return to the last directory | |
155 matching a particular regexp, use 'cd =REGEXP'. To display the | |
156 directory history list, use 'cd ='. | |
157 | |
158 This mechanism is very similar to that provided by `pushd', except | |
159 it's far more automatic. `pushd' allows the user to decide which | |
160 directories gets pushed, and its size is unlimited. | |
161 | |
162 `eshell-last-dir-ring' is meant for users who don't use `pushd' | |
163 explicity very much, but every once in a while would like to return to | |
164 a previously visited directory without having to type in the whole | |
165 thing again." | |
166 :type 'integer | |
167 :group 'eshell-dirs) | |
168 | |
169 (defcustom eshell-last-dir-unique t | |
170 "*If non-nil, `eshell-last-dir-ring' contains only unique entries." | |
171 :type 'boolean | |
172 :group 'eshell-dirs) | |
173 | |
46082
6303529069a9
Removed an extraneous space character from a comment.
John Wiegley <johnw@newartisans.com>
parents:
43327
diff
changeset
|
174 ;;; Internal Variables: |
29876 | 175 |
176 (defvar eshell-dirstack nil | |
177 "List of directories saved by pushd in the Eshell buffer. | |
178 Thus, this does not include the current directory.") | |
179 | |
180 (defvar eshell-last-dir-ring nil | |
107517
ed16fdd2685a
Fix typos in docstrings.
Juanma Barranquero <lekktu@gmail.com>
parents:
106815
diff
changeset
|
181 "The last directory that Eshell was in.") |
29876 | 182 |
183 ;;; Functions: | |
184 | |
185 (defun eshell-dirs-initialize () | |
186 "Initialize the builtin functions for Eshell." | |
187 (make-local-variable 'eshell-variable-aliases-list) | |
188 (setq eshell-variable-aliases-list | |
189 (append | |
190 eshell-variable-aliases-list | |
191 '(("-" (lambda (indices) | |
192 (if (not indices) | |
193 (unless (ring-empty-p eshell-last-dir-ring) | |
194 (expand-file-name | |
195 (ring-ref eshell-last-dir-ring 0))) | |
196 (expand-file-name | |
197 (eshell-apply-indices eshell-last-dir-ring indices))))) | |
198 ("+" "PWD") | |
199 ("PWD" (lambda (indices) | |
200 (expand-file-name (eshell/pwd))) t) | |
201 ("OLDPWD" (lambda (indices) | |
202 (unless (ring-empty-p eshell-last-dir-ring) | |
203 (expand-file-name | |
204 (ring-ref eshell-last-dir-ring 0)))) t)))) | |
205 | |
206 (when eshell-cd-on-directory | |
207 (make-local-variable 'eshell-interpreter-alist) | |
208 (setq eshell-interpreter-alist | |
209 (cons (cons 'eshell-lone-directory-p | |
210 'eshell-dirs-substitute-cd) | |
211 eshell-interpreter-alist))) | |
212 | |
213 (add-hook 'eshell-parse-argument-hook | |
214 'eshell-parse-user-reference nil t) | |
215 (if (eshell-under-windows-p) | |
216 (add-hook 'eshell-parse-argument-hook | |
217 'eshell-parse-drive-letter nil t)) | |
218 | |
219 (when (eshell-using-module 'eshell-cmpl) | |
220 (add-hook 'pcomplete-try-first-hook | |
221 'eshell-complete-user-reference nil t)) | |
222 | |
223 (make-local-variable 'eshell-dirstack) | |
224 (make-local-variable 'eshell-last-dir-ring) | |
225 | |
226 (if eshell-last-dir-ring-file-name | |
227 (eshell-read-last-dir-ring)) | |
228 (unless eshell-last-dir-ring | |
229 (setq eshell-last-dir-ring (make-ring eshell-last-dir-ring-size))) | |
230 | |
231 (add-hook 'eshell-exit-hook 'eshell-write-last-dir-ring nil t) | |
232 | |
233 (add-hook 'kill-emacs-hook 'eshell-save-some-last-dir)) | |
234 | |
235 (defun eshell-save-some-last-dir () | |
236 "Save the list-dir-ring for any open Eshell buffers." | |
237 (eshell-for buf (buffer-list) | |
238 (if (buffer-live-p buf) | |
239 (with-current-buffer buf | |
240 (if (and eshell-mode | |
241 eshell-ask-to-save-last-dir | |
242 (or (eq eshell-ask-to-save-last-dir 'always) | |
243 (y-or-n-p | |
244 (format "Save last dir ring for Eshell buffer `%s'? " | |
245 (buffer-name buf))))) | |
246 (eshell-write-last-dir-ring)))))) | |
247 | |
248 (defun eshell-lone-directory-p (file) | |
249 "Test whether FILE is just a directory name, and not a command name." | |
250 (and (file-directory-p file) | |
251 (or (file-name-directory file) | |
252 (not (eshell-search-path file))))) | |
253 | |
254 (defun eshell-dirs-substitute-cd (&rest args) | |
255 "Substitute the given command for a call to `cd' on that name." | |
256 (if (> (length args) 1) | |
257 (error "%s: command not found" (car args)) | |
258 (throw 'eshell-replace-command | |
31240 | 259 (eshell-parse-command "cd" (eshell-flatten-list args))))) |
29876 | 260 |
261 (defun eshell-parse-user-reference () | |
262 "An argument beginning with ~ is a filename to be expanded." | |
263 (when (and (not eshell-current-argument) | |
264 (eq (char-after) ?~)) | |
265 (add-to-list 'eshell-current-modifiers 'expand-file-name) | |
266 (forward-char) | |
267 (char-to-string (char-before)))) | |
268 | |
269 (defun eshell-parse-drive-letter () | |
107517
ed16fdd2685a
Fix typos in docstrings.
Juanma Barranquero <lekktu@gmail.com>
parents:
106815
diff
changeset
|
270 "An argument beginning with X:[^/] is a drive letter reference." |
29876 | 271 (when (and (not eshell-current-argument) |
272 (looking-at "\\([A-Za-z]:\\)\\([^/\\\\]\\|\\'\\)")) | |
273 (goto-char (match-end 1)) | |
274 (let* ((letter (match-string 1)) | |
275 (regexp (concat "\\`" letter)) | |
276 (path (eshell-find-previous-directory regexp))) | |
62915
b89e30bcd2bb
Changed all uses of `directory-sep-char' to ?/, and all uses of
John Wiegley <johnw@newartisans.com>
parents:
55195
diff
changeset
|
277 (concat (or path letter) "/")))) |
29876 | 278 |
104994
48428e99bf83
* eshell/em-hist.el:
Chong Yidong <cyd@stupidchicken.com>
parents:
100908
diff
changeset
|
279 (defvar pcomplete-stub) |
48428e99bf83
* eshell/em-hist.el:
Chong Yidong <cyd@stupidchicken.com>
parents:
100908
diff
changeset
|
280 (defvar pcomplete-last-completion-raw) |
48428e99bf83
* eshell/em-hist.el:
Chong Yidong <cyd@stupidchicken.com>
parents:
100908
diff
changeset
|
281 (declare-function pcomplete-actual-arg "pcomplete") |
48428e99bf83
* eshell/em-hist.el:
Chong Yidong <cyd@stupidchicken.com>
parents:
100908
diff
changeset
|
282 (declare-function pcomplete-uniqify-list "pcomplete") |
48428e99bf83
* eshell/em-hist.el:
Chong Yidong <cyd@stupidchicken.com>
parents:
100908
diff
changeset
|
283 |
29876 | 284 (defun eshell-complete-user-reference () |
285 "If there is a user reference, complete it." | |
286 (let ((arg (pcomplete-actual-arg))) | |
287 (when (string-match "\\`~[a-z]*\\'" arg) | |
288 (setq pcomplete-stub (substring arg 1) | |
289 pcomplete-last-completion-raw t) | |
290 (throw 'pcomplete-completions | |
291 (progn | |
292 (eshell-read-user-names) | |
293 (pcomplete-uniqify-list | |
294 (mapcar | |
295 (function | |
296 (lambda (user) | |
297 (file-name-as-directory (cdr user)))) | |
298 eshell-user-names))))))) | |
299 | |
33020 | 300 (defun eshell/pwd (&rest args) |
29876 | 301 "Change output from `pwd` to be cleaner." |
302 (let* ((path default-directory) | |
303 (len (length path))) | |
304 (if (and (> len 1) | |
62915
b89e30bcd2bb
Changed all uses of `directory-sep-char' to ?/, and all uses of
John Wiegley <johnw@newartisans.com>
parents:
55195
diff
changeset
|
305 (eq (aref path (1- len)) ?/) |
29876 | 306 (not (and (eshell-under-windows-p) |
307 (string-match "\\`[A-Za-z]:[\\\\/]\\'" path)))) | |
308 (setq path (substring path 0 (1- (length path))))) | |
309 (if eshell-pwd-convert-function | |
33020 | 310 (funcall eshell-pwd-convert-function path) |
311 path))) | |
29876 | 312 |
313 (defun eshell-expand-multiple-dots (path) | |
314 "Convert '...' to '../..', '....' to '../../..', etc.. | |
315 | |
316 With the following piece of advice, you can make this functionality | |
317 available in most of Emacs, with the exception of filename completion | |
318 in the minibuffer: | |
319 | |
320 (defadvice expand-file-name | |
321 (before translate-multiple-dots | |
322 (filename &optional directory) activate) | |
323 (setq filename (eshell-expand-multiple-dots filename)))" | |
79351
e0ca14d507a5
Change regexp to match dir like 'a...b'
Vinicius Jose Latorre <viniciusjl@ig.com.br>
parents:
78220
diff
changeset
|
324 (while (string-match "\\(?:^\\|/\\)\\.\\.\\(\\.+\\)\\(?:$\\|/\\)" path) |
29876 | 325 (let* ((extra-dots (match-string 1 path)) |
326 (len (length extra-dots)) | |
327 replace-text) | |
328 (while (> len 0) | |
62915
b89e30bcd2bb
Changed all uses of `directory-sep-char' to ?/, and all uses of
John Wiegley <johnw@newartisans.com>
parents:
55195
diff
changeset
|
329 (setq replace-text (concat replace-text "/..") |
29876 | 330 len (1- len))) |
331 (setq path | |
332 (replace-match replace-text t t path 1)))) | |
333 path) | |
334 | |
335 (defun eshell-find-previous-directory (regexp) | |
336 "Find the most recent last-dir matching REGEXP." | |
337 (let ((index 0) | |
338 (len (ring-length eshell-last-dir-ring)) | |
339 oldpath) | |
340 (if (> (length regexp) 0) | |
341 (while (< index len) | |
342 (setq oldpath (ring-ref eshell-last-dir-ring index)) | |
343 (if (string-match regexp oldpath) | |
344 (setq index len) | |
345 (setq oldpath nil | |
346 index (1+ index))))) | |
347 oldpath)) | |
348 | |
95619
45dbb3c749a6
Remove unnecessary eval-when-compiles and eval-and-compiles.
Glenn Morris <rgm@gnu.org>
parents:
95152
diff
changeset
|
349 (defvar dired-directory) |
29876 | 350 |
351 (defun eshell/cd (&rest args) ; all but first ignored | |
352 "Alias to extend the behavior of `cd'." | |
31240 | 353 (setq args (eshell-flatten-list args)) |
29876 | 354 (let ((path (car args)) |
355 (subpath (car (cdr args))) | |
46852
6eb625bead4f
Removed eshell-under-cygwin-p, and all uses of it.
John Wiegley <johnw@newartisans.com>
parents:
46820
diff
changeset
|
356 (case-fold-search (eshell-under-windows-p)) |
29876 | 357 handled) |
358 (if (numberp path) | |
359 (setq path (number-to-string path))) | |
360 (if (numberp subpath) | |
361 (setq subpath (number-to-string subpath))) | |
362 (cond | |
363 (subpath | |
364 (let ((curdir (eshell/pwd))) | |
365 (if (string-match path curdir) | |
366 (setq path (replace-match subpath nil nil curdir)) | |
367 (error "Path substring '%s' not found" path)))) | |
368 ((and path (string-match "^-\\([0-9]*\\)$" path)) | |
369 (let ((index (match-string 1 path))) | |
370 (setq path | |
371 (ring-remove eshell-last-dir-ring | |
372 (if index | |
62915
b89e30bcd2bb
Changed all uses of `directory-sep-char' to ?/, and all uses of
John Wiegley <johnw@newartisans.com>
parents:
55195
diff
changeset
|
373 (string-to-number index) |
29876 | 374 0))))) |
375 ((and path (string-match "^=\\(.*\\)$" path)) | |
376 (let ((oldpath (eshell-find-previous-directory | |
377 (match-string 1 path)))) | |
378 (if oldpath | |
379 (setq path oldpath) | |
380 (let ((len (ring-length eshell-last-dir-ring)) | |
381 (index 0)) | |
382 (if (= len 0) | |
383 (error "Directory ring empty")) | |
31241 | 384 (eshell-init-print-buffer) |
29876 | 385 (while (< index len) |
31241 | 386 (eshell-buffered-print |
29876 | 387 (concat (number-to-string index) ": " |
31241 | 388 (ring-ref eshell-last-dir-ring index) "\n")) |
29876 | 389 (setq index (1+ index))) |
31241 | 390 (eshell-flush) |
29876 | 391 (setq handled t))))) |
392 (path | |
393 (setq path (eshell-expand-multiple-dots path)))) | |
394 (unless handled | |
395 (setq dired-directory (or path "~")) | |
396 (let ((curdir (eshell/pwd))) | |
397 (unless (equal curdir dired-directory) | |
398 (eshell-add-to-dir-ring curdir)) | |
399 (let ((result (cd dired-directory))) | |
400 (and eshell-cd-shows-directory | |
401 (eshell-printn result))) | |
402 (run-hooks 'eshell-directory-change-hook) | |
403 (if eshell-list-files-after-cd | |
106264
5fa7590f242f
(eshell/cd): Don't throw to a tag outside the scope.
Glenn Morris <rgm@gnu.org>
parents:
104994
diff
changeset
|
404 ;; Let-bind eshell-last-command around this? |
5fa7590f242f
(eshell/cd): Don't throw to a tag outside the scope.
Glenn Morris <rgm@gnu.org>
parents:
104994
diff
changeset
|
405 (eshell-plain-command "ls" (cdr args))) |
29876 | 406 nil)))) |
407 | |
37661
6d7c89c79996
Set the property `eshell-no-numeric-conversions' on the following
John Wiegley <johnw@newartisans.com>
parents:
33020
diff
changeset
|
408 (put 'eshell/cd 'eshell-no-numeric-conversions t) |
6d7c89c79996
Set the property `eshell-no-numeric-conversions' on the following
John Wiegley <johnw@newartisans.com>
parents:
33020
diff
changeset
|
409 |
29876 | 410 (defun eshell-add-to-dir-ring (path) |
411 "Add PATH to the last-dir-ring, if applicable." | |
412 (unless (and (not (ring-empty-p eshell-last-dir-ring)) | |
413 (equal path (ring-ref eshell-last-dir-ring 0))) | |
414 (if eshell-last-dir-unique | |
415 (let ((index 0) | |
416 (len (ring-length eshell-last-dir-ring))) | |
417 (while (< index len) | |
418 (if (equal (ring-ref eshell-last-dir-ring index) path) | |
419 (ring-remove eshell-last-dir-ring index) | |
420 (setq index (1+ index)))))) | |
421 (ring-insert eshell-last-dir-ring path))) | |
422 | |
423 ;;; pushd [+n | dir] | |
424 (defun eshell/pushd (&rest args) ; all but first ignored | |
425 "Implementation of pushd in Lisp." | |
426 (let ((path (car args))) | |
427 (cond | |
428 ((null path) | |
429 ;; no arg -- swap pwd and car of stack unless eshell-pushd-tohome | |
430 (cond (eshell-pushd-tohome | |
431 (eshell/pushd "~")) | |
432 (eshell-dirstack | |
433 (let ((old (eshell/pwd))) | |
434 (eshell/cd (car eshell-dirstack)) | |
435 (setq eshell-dirstack (cons old (cdr eshell-dirstack))) | |
436 (eshell/dirs t))) | |
437 (t | |
438 (error "pushd: No other directory")))) | |
439 ((string-match "^\\+\\([0-9]\\)" path) | |
440 ;; pushd +n | |
441 (setq path (string-to-number (match-string 1 path))) | |
442 (cond ((> path (length eshell-dirstack)) | |
443 (error "Directory stack not that deep")) | |
444 ((= path 0) | |
445 (error "Couldn't cd")) | |
446 (eshell-pushd-dextract | |
447 (let ((dir (nth (1- path) eshell-dirstack))) | |
448 (eshell/popd path) | |
449 (eshell/pushd (eshell/pwd)) | |
450 (eshell/cd dir) | |
451 (eshell/dirs t))) | |
452 (t | |
453 (let* ((ds (cons (eshell/pwd) eshell-dirstack)) | |
454 (dslen (length ds)) | |
455 (front (nthcdr path ds)) | |
456 (back (nreverse (nthcdr (- dslen path) (reverse ds)))) | |
457 (new-ds (append front back))) | |
458 (eshell/cd (car new-ds)) | |
459 (setq eshell-dirstack (cdr new-ds)) | |
460 (eshell/dirs t))))) | |
461 (t | |
462 ;; pushd <dir> | |
463 (let ((old-wd (eshell/pwd))) | |
464 (eshell/cd path) | |
465 (if (or (null eshell-pushd-dunique) | |
466 (not (member old-wd eshell-dirstack))) | |
467 (setq eshell-dirstack (cons old-wd eshell-dirstack))) | |
468 (eshell/dirs t))))) | |
469 nil) | |
470 | |
37661
6d7c89c79996
Set the property `eshell-no-numeric-conversions' on the following
John Wiegley <johnw@newartisans.com>
parents:
33020
diff
changeset
|
471 (put 'eshell/pushd 'eshell-no-numeric-conversions t) |
6d7c89c79996
Set the property `eshell-no-numeric-conversions' on the following
John Wiegley <johnw@newartisans.com>
parents:
33020
diff
changeset
|
472 |
29876 | 473 ;;; popd [+n] |
474 (defun eshell/popd (&rest args) | |
475 "Implementation of popd in Lisp." | |
476 (let ((ref (or (car args) "+0"))) | |
477 (unless (and (stringp ref) | |
478 (string-match "\\`\\([+-][0-9]+\\)\\'" ref)) | |
479 (error "popd: bad arg `%s'" ref)) | |
480 (setq ref (string-to-number (match-string 1 ref))) | |
481 (cond ((= ref 0) | |
482 (unless eshell-dirstack | |
483 (error "popd: Directory stack empty")) | |
484 (eshell/cd (car eshell-dirstack)) | |
485 (setq eshell-dirstack (cdr eshell-dirstack)) | |
486 (eshell/dirs t)) | |
487 ((<= (abs ref) (length eshell-dirstack)) | |
488 (let* ((ds (cons nil eshell-dirstack)) | |
489 (cell (nthcdr (if (> ref 0) | |
490 (1- ref) | |
491 (+ (length eshell-dirstack) ref)) ds)) | |
492 (dir (cadr cell))) | |
493 (eshell/cd dir) | |
494 (setcdr cell (cdr (cdr cell))) | |
495 (setq eshell-dirstack (cdr ds)) | |
496 (eshell/dirs t))) | |
497 (t | |
498 (error "Couldn't popd")))) | |
499 nil) | |
500 | |
37661
6d7c89c79996
Set the property `eshell-no-numeric-conversions' on the following
John Wiegley <johnw@newartisans.com>
parents:
33020
diff
changeset
|
501 (put 'eshell/popd 'eshell-no-numeric-conversions t) |
6d7c89c79996
Set the property `eshell-no-numeric-conversions' on the following
John Wiegley <johnw@newartisans.com>
parents:
33020
diff
changeset
|
502 |
29876 | 503 (defun eshell/dirs (&optional if-verbose) |
504 "Implementation of dirs in Lisp." | |
505 (when (or (not if-verbose) eshell-dirtrack-verbose) | |
506 (let* ((msg "") | |
507 (ds (cons (eshell/pwd) eshell-dirstack)) | |
508 (home (expand-file-name "~/")) | |
509 (homelen (length home))) | |
510 (while ds | |
511 (let ((dir (car ds))) | |
512 (and (>= (length dir) homelen) | |
513 (string= home (substring dir 0 homelen)) | |
514 (setq dir (concat "~/" (substring dir homelen)))) | |
515 (setq msg (concat msg (directory-file-name dir) " ")) | |
516 (setq ds (cdr ds)))) | |
517 msg))) | |
518 | |
519 (defun eshell-read-last-dir-ring () | |
107517
ed16fdd2685a
Fix typos in docstrings.
Juanma Barranquero <lekktu@gmail.com>
parents:
106815
diff
changeset
|
520 "Set the buffer's `eshell-last-dir-ring' from a history file." |
29876 | 521 (let ((file eshell-last-dir-ring-file-name)) |
522 (cond | |
523 ((or (null file) | |
524 (equal file "") | |
525 (not (file-readable-p file))) | |
526 nil) | |
527 (t | |
528 (let* ((count 0) | |
529 (size eshell-last-dir-ring-size) | |
530 (ring (make-ring size))) | |
531 (with-temp-buffer | |
532 (insert-file-contents file) | |
533 ;; Save restriction in case file is already visited... | |
534 ;; Watch for those date stamps in history files! | |
535 (goto-char (point-max)) | |
536 (while (and (< count size) | |
537 (re-search-backward "^\\([^\n].*\\)$" nil t)) | |
538 (ring-insert-at-beginning ring (match-string 1)) | |
539 (setq count (1+ count))) | |
540 ;; never allow the top element to equal the current | |
541 ;; directory | |
542 (while (and (not (ring-empty-p ring)) | |
543 (equal (ring-ref ring 0) (eshell/pwd))) | |
544 (ring-remove ring 0))) | |
545 (setq eshell-last-dir-ring ring)))))) | |
546 | |
547 (defun eshell-write-last-dir-ring () | |
107517
ed16fdd2685a
Fix typos in docstrings.
Juanma Barranquero <lekktu@gmail.com>
parents:
106815
diff
changeset
|
548 "Write the buffer's `eshell-last-dir-ring' to a history file." |
29876 | 549 (let ((file eshell-last-dir-ring-file-name)) |
550 (cond | |
551 ((or (null file) | |
552 (equal file "") | |
553 (null eshell-last-dir-ring) | |
554 (ring-empty-p eshell-last-dir-ring)) | |
555 nil) | |
556 ((not (file-writable-p file)) | |
557 (message "Cannot write last-dir-ring file %s" file)) | |
558 (t | |
559 (let* ((ring eshell-last-dir-ring) | |
560 (index (ring-length ring))) | |
561 (with-temp-buffer | |
562 (while (> index 0) | |
563 (setq index (1- index)) | |
564 (insert (ring-ref ring index) ?\n)) | |
565 (insert (eshell/pwd) ?\n) | |
566 (eshell-with-private-file-modes | |
567 (write-region (point-min) (point-max) file nil | |
568 'no-message)))))))) | |
569 | |
87066
14174df2455d
Require individual files if needed when compiling, rather than
Glenn Morris <rgm@gnu.org>
parents:
79351
diff
changeset
|
570 (provide 'em-dirs) |
29876 | 571 |
95152
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
572 ;; Local Variables: |
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
573 ;; generated-autoload-file: "esh-groups.el" |
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
574 ;; End: |
ad5d26b1d5d1
Use eshell-defgroup rather than defgroup.
Glenn Morris <rgm@gnu.org>
parents:
94661
diff
changeset
|
575 |
93975
1e3a407766b9
Fix up comment convention on the arch-tag lines.
Stefan Monnier <monnier@iro.umontreal.ca>
parents:
87649
diff
changeset
|
576 ;; arch-tag: 1e9c5a95-f1bd-45f8-ad36-55aac706e787 |
29876 | 577 ;;; em-dirs.el ends here |