mirror of
https://git.savannah.gnu.org/git/emacs.git
synced 2025-01-21 18:23:59 +00:00
70ad3da9ec
window-width.
3345 lines
128 KiB
EmacsLisp
3345 lines
128 KiB
EmacsLisp
;;; dired.el --- directory-browsing commands
|
||
|
||
;; Copyright (C) 1985, 1986, 1992, 1993, 1994, 1995, 1996, 1997, 2000,
|
||
;; 2001, 2002, 2003, 2004, 2005, 2006, 2007 Free Software Foundation, Inc.
|
||
|
||
;; Author: Sebastian Kremer <sk@thp.uni-koeln.de>
|
||
;; Maintainer: FSF
|
||
;; Keywords: files
|
||
|
||
;; This file is part of GNU Emacs.
|
||
|
||
;; GNU Emacs 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, or (at your option)
|
||
;; any later version.
|
||
|
||
;; GNU Emacs 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 GNU Emacs; see the file COPYING. If not, write to the
|
||
;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
||
;; Boston, MA 02110-1301, USA.
|
||
|
||
;;; Commentary:
|
||
|
||
;; This is a major mode for directory browsing and editing. It is
|
||
;; documented in the Emacs manual.
|
||
|
||
;; Rewritten in 1990/1991 to add tree features, file marking and
|
||
;; sorting by Sebastian Kremer <sk@thp.uni-koeln.de>.
|
||
;; Finished up by rms in 1992.
|
||
|
||
;;; Code:
|
||
|
||
;;; Customizable variables
|
||
|
||
(defgroup dired nil
|
||
"Directory editing."
|
||
:link '(custom-manual "(emacs)Dired")
|
||
:group 'files)
|
||
|
||
(defgroup dired-mark nil
|
||
"Handling marks in Dired."
|
||
:prefix "dired-"
|
||
:group 'dired)
|
||
|
||
|
||
;;;###autoload
|
||
(defcustom dired-listing-switches "-al"
|
||
"*Switches passed to `ls' for Dired. MUST contain the `l' option.
|
||
May contain all other options that don't contradict `-l';
|
||
may contain even `F', `b', `i' and `s'. See also the variable
|
||
`dired-ls-F-marks-symlinks' concerning the `F' switch.
|
||
On systems such as MS-DOS and MS-Windows, which use `ls' emulation in Lisp,
|
||
some of the `ls' switches are not supported; see the doc string of
|
||
`insert-directory' in `ls-lisp.el' for more details."
|
||
:type 'string
|
||
:group 'dired)
|
||
|
||
(defvar dired-subdir-switches nil
|
||
"If non-nil, switches passed to `ls' for inserting subdirectories.
|
||
If nil, `dired-listing-switches' is used.")
|
||
|
||
; Don't use absolute file names as /bin should be in any PATH and people
|
||
; may prefer /usr/local/gnu/bin or whatever. However, chown is
|
||
; usually not in PATH.
|
||
|
||
;;;###autoload
|
||
(defvar dired-chown-program
|
||
(if (memq system-type '(hpux dgux usg-unix-v irix linux gnu/linux cygwin))
|
||
"chown"
|
||
(if (file-exists-p "/usr/sbin/chown")
|
||
"/usr/sbin/chown"
|
||
"/etc/chown"))
|
||
"Name of chown command (usually `chown' or `/etc/chown').")
|
||
|
||
(defvar dired-use-ls-dired (not (not (string-match "gnu" system-configuration)))
|
||
"Non-nil means Dired should use `ls --dired'.")
|
||
|
||
(defvar dired-chmod-program "chmod"
|
||
"Name of chmod command (usually `chmod').")
|
||
|
||
(defvar dired-touch-program "touch"
|
||
"Name of touch command (usually `touch').")
|
||
|
||
;;;###autoload
|
||
(defcustom dired-ls-F-marks-symlinks nil
|
||
"*Informs Dired about how `ls -lF' marks symbolic links.
|
||
Set this to t if `ls' (or whatever program is specified by
|
||
`insert-directory-program') with `-lF' marks the symbolic link
|
||
itself with a trailing @ (usually the case under Ultrix).
|
||
|
||
Example: if `ln -s foo bar; ls -F bar' gives `bar -> foo', set it to
|
||
nil (the default), if it gives `bar@ -> foo', set it to t.
|
||
|
||
Dired checks if there is really a @ appended. Thus, if you have a
|
||
marking `ls' program on one host and a non-marking on another host, and
|
||
don't care about symbolic links which really end in a @, you can
|
||
always set this variable to t."
|
||
:type 'boolean
|
||
:group 'dired-mark)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-trivial-filenames "^\\.\\.?$\\|^#"
|
||
"*Regexp of files to skip when finding first file of a directory.
|
||
A value of nil means move to the subdir line.
|
||
A value of t means move to first file."
|
||
:type '(choice (const :tag "Move to subdir" nil)
|
||
(const :tag "Move to first" t)
|
||
regexp)
|
||
:group 'dired)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-keep-marker-rename t
|
||
;; Use t as default so that moved files "take their markers with them".
|
||
"*Controls marking of renamed files.
|
||
If t, files keep their previous marks when they are renamed.
|
||
If a character, renamed files (whether previously marked or not)
|
||
are afterward marked with that character."
|
||
:type '(choice (const :tag "Keep" t)
|
||
(character :tag "Mark"))
|
||
:group 'dired-mark)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-keep-marker-copy ?C
|
||
"*Controls marking of copied files.
|
||
If t, copied files are marked if and as the corresponding original files were.
|
||
If a character, copied files are unconditionally marked with that character."
|
||
:type '(choice (const :tag "Keep" t)
|
||
(character :tag "Mark"))
|
||
:group 'dired-mark)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-keep-marker-hardlink ?H
|
||
"*Controls marking of newly made hard links.
|
||
If t, they are marked if and as the files linked to were marked.
|
||
If a character, new links are unconditionally marked with that character."
|
||
:type '(choice (const :tag "Keep" t)
|
||
(character :tag "Mark"))
|
||
:group 'dired-mark)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-keep-marker-symlink ?Y
|
||
"*Controls marking of newly made symbolic links.
|
||
If t, they are marked if and as the files linked to were marked.
|
||
If a character, new links are unconditionally marked with that character."
|
||
:type '(choice (const :tag "Keep" t)
|
||
(character :tag "Mark"))
|
||
:group 'dired-mark)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-dwim-target nil
|
||
"*If non-nil, Dired tries to guess a default target directory.
|
||
This means: if there is a dired buffer displayed in the next window,
|
||
use its current subdir, instead of the current subdir of this dired buffer.
|
||
|
||
The target is used in the prompt for file copy, rename etc."
|
||
:type 'boolean
|
||
:group 'dired)
|
||
|
||
;;;###autoload
|
||
(defcustom dired-copy-preserve-time t
|
||
"*If non-nil, Dired preserves the last-modified time in a file copy.
|
||
\(This works on only some systems.)"
|
||
:type 'boolean
|
||
:group 'dired)
|
||
|
||
; These variables were deleted and the replacements are on files.el.
|
||
; We leave aliases behind for back-compatibility.
|
||
(defvaralias 'dired-free-space-program 'directory-free-space-program)
|
||
(defvaralias 'dired-free-space-args 'directory-free-space-args)
|
||
|
||
;;; Hook variables
|
||
|
||
(defcustom dired-load-hook nil
|
||
"Run after loading Dired.
|
||
You can customize key bindings or load extensions with this."
|
||
:group 'dired
|
||
:type 'hook)
|
||
|
||
(defcustom dired-mode-hook nil
|
||
"Run at the very end of `dired-mode'."
|
||
:group 'dired
|
||
:type 'hook)
|
||
|
||
(defcustom dired-before-readin-hook nil
|
||
"This hook is run before a dired buffer is read in (created or reverted)."
|
||
:group 'dired
|
||
:type 'hook)
|
||
|
||
(defcustom dired-after-readin-hook nil
|
||
"Hook run after each time a file or directory is read by Dired.
|
||
After each listing of a file or directory, this hook is run
|
||
with the buffer narrowed to the listing."
|
||
:group 'dired
|
||
:type 'hook)
|
||
;; Note this can't simply be run inside function `dired-ls' as the hook
|
||
;; functions probably depend on the dired-subdir-alist to be OK.
|
||
|
||
(defcustom dired-dnd-protocol-alist
|
||
'(("^file:///" . dired-dnd-handle-local-file)
|
||
("^file://" . dired-dnd-handle-file)
|
||
("^file:" . dired-dnd-handle-local-file))
|
||
"The functions to call when a drop in `dired-mode' is made.
|
||
See `dnd-protocol-alist' for more information. When nil, behave
|
||
as in other buffers. Changing this option is effective only for
|
||
new dired buffers."
|
||
:type '(choice (repeat (cons (regexp) (function)))
|
||
(const :tag "Behave as in other buffers" nil))
|
||
:version "22.1"
|
||
:group 'dired)
|
||
|
||
;; Internal variables
|
||
|
||
(defvar dired-marker-char ?* ; the answer is 42
|
||
;; so that you can write things like
|
||
;; (let ((dired-marker-char ?X))
|
||
;; ;; great code using X markers ...
|
||
;; )
|
||
;; For example, commands operating on two sets of files, A and B.
|
||
;; Or marking files with digits 0-9. This could implicate
|
||
;; concentric sets or an order for the marked files.
|
||
;; The code depends on dynamic scoping on the marker char.
|
||
"In Dired, the current mark character.
|
||
This is what the do-commands look for, and what the mark-commands store.")
|
||
|
||
(defvar dired-del-marker ?D
|
||
"Character used to flag files for deletion.")
|
||
|
||
(defvar dired-shrink-to-fit t
|
||
;; I see no reason ever to make this nil -- rms.
|
||
;; (> baud-rate search-slow-speed)
|
||
"Non-nil means Dired shrinks the display buffer to fit the marked files.")
|
||
|
||
(defvar dired-flagging-regexp nil);; Last regexp used to flag files.
|
||
|
||
(defvar dired-file-version-alist)
|
||
|
||
;;;###autoload
|
||
(defvar dired-directory nil
|
||
"The directory name or wildcard spec that this dired directory lists.
|
||
Local to each dired buffer. May be a list, in which case the car is the
|
||
directory name and the cdr is the list of files to mention.
|
||
The directory name must be absolute, but need not be fully expanded.")
|
||
|
||
(defvar dired-actual-switches nil
|
||
"The value of `dired-listing-switches' used to make this buffer's text.")
|
||
|
||
(defvar dired-re-inode-size "[0-9 \t]*"
|
||
"Regexp for optional initial inode and file size as made by `ls -i -s'.")
|
||
|
||
;; These regexps must be tested at beginning-of-line, but are also
|
||
;; used to search for next matches, so neither omitting "^" nor
|
||
;; replacing "^" by "\n" (to make it slightly faster) will work.
|
||
|
||
(defvar dired-re-mark "^[^ \n]")
|
||
;; "Regexp matching a marked line.
|
||
;; Important: the match ends just after the marker."
|
||
(defvar dired-re-maybe-mark "^. ")
|
||
;; The [^:] part after "d" and "l" is to avoid confusion with the
|
||
;; DOS/Windows-style drive letters in directory names, like in "d:/foo".
|
||
(defvar dired-re-dir (concat dired-re-maybe-mark dired-re-inode-size "d[^:]"))
|
||
(defvar dired-re-sym (concat dired-re-maybe-mark dired-re-inode-size "l[^:]"))
|
||
(defvar dired-re-exe;; match ls permission string of an executable file
|
||
(mapconcat (function
|
||
(lambda (x)
|
||
(concat dired-re-maybe-mark dired-re-inode-size x)))
|
||
'("-[-r][-w][xs][-r][-w].[-r][-w]."
|
||
"-[-r][-w].[-r][-w][xs][-r][-w]."
|
||
"-[-r][-w].[-r][-w].[-r][-w][xst]")
|
||
"\\|"))
|
||
(defvar dired-re-perms "[-bcdlps][-r][-w].[-r][-w].[-r][-w].")
|
||
(defvar dired-re-dot "^.* \\.\\.?/?$")
|
||
|
||
;; The subdirectory names in the next two lists are expanded.
|
||
(defvar dired-subdir-alist nil
|
||
"Association list of subdirectories and their buffer positions.
|
||
Each subdirectory has an element: (DIRNAME . STARTMARKER).
|
||
The order of elements is the reverse of the order in the buffer.
|
||
In simple cases, this list contains one element.")
|
||
|
||
(defvar dired-switches-alist nil
|
||
"Keeps track of which switches to use for inserted subdirectories.
|
||
This is an alist of the form (SUBDIR . SWITCHES).")
|
||
(make-variable-buffer-local 'dired-switches-alist)
|
||
|
||
(defvaralias 'dired-move-to-filename-regexp
|
||
'directory-listing-before-filename-regexp)
|
||
|
||
(defvar dired-subdir-regexp "^. \\([^\n\r]+\\)\\(:\\)[\n\r]"
|
||
"Regexp matching a maybe hidden subdirectory line in `ls -lR' output.
|
||
Subexpression 1 is the subdirectory proper, no trailing colon.
|
||
The match starts at the beginning of the line and ends after the end
|
||
of the line (\\n or \\r).
|
||
Subexpression 2 must end right before the \\n or \\r.")
|
||
|
||
(defgroup dired-faces nil
|
||
"Faces used by Dired."
|
||
:group 'dired
|
||
:group 'faces)
|
||
|
||
(defface dired-header
|
||
'((t (:inherit font-lock-type-face)))
|
||
"Face used for directory headers."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-header-face 'dired-header
|
||
"Face name used for directory headers.")
|
||
|
||
(defface dired-mark
|
||
'((t (:inherit font-lock-constant-face)))
|
||
"Face used for dired marks."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-mark-face 'dired-mark
|
||
"Face name used for dired marks.")
|
||
|
||
(defface dired-marked
|
||
'((t (:inherit font-lock-warning-face)))
|
||
"Face used for marked files."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-marked-face 'dired-marked
|
||
"Face name used for marked files.")
|
||
|
||
(defface dired-flagged
|
||
'((t (:inherit font-lock-warning-face)))
|
||
"Face used for flagged files."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-flagged-face 'dired-flagged
|
||
"Face name used for flagged files.")
|
||
|
||
(defface dired-warning
|
||
'((t (:inherit font-lock-comment-face)))
|
||
"Face used to highlight a part of a buffer that needs user attention."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-warning-face 'dired-warning
|
||
"Face name used for a part of a buffer that needs user attention.")
|
||
|
||
(defface dired-directory
|
||
'((t (:inherit font-lock-function-name-face)))
|
||
"Face used for subdirectories."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-directory-face 'dired-directory
|
||
"Face name used for subdirectories.")
|
||
|
||
(defface dired-symlink
|
||
'((t (:inherit font-lock-keyword-face)))
|
||
"Face used for symbolic links."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-symlink-face 'dired-symlink
|
||
"Face name used for symbolic links.")
|
||
|
||
(defface dired-ignored
|
||
'((t (:inherit shadow)))
|
||
"Face used for files suffixed with `completion-ignored-extensions'."
|
||
:group 'dired-faces
|
||
:version "22.1")
|
||
(defvar dired-ignored-face 'dired-ignored
|
||
"Face name used for files suffixed with `completion-ignored-extensions'.")
|
||
|
||
(defvar dired-font-lock-keywords
|
||
(list
|
||
;;
|
||
;; Directory headers.
|
||
(list dired-subdir-regexp '(1 dired-header-face))
|
||
;;
|
||
;; Dired marks.
|
||
(list dired-re-mark '(0 dired-mark-face))
|
||
;;
|
||
;; We make heavy use of MATCH-ANCHORED, since the regexps don't identify the
|
||
;; file name itself. We search for Dired defined regexps, and then use the
|
||
;; Dired defined function `dired-move-to-filename' before searching for the
|
||
;; simple regexp ".+". It is that regexp which matches the file name.
|
||
;;
|
||
;; Marked files.
|
||
(list (concat "^[" (char-to-string dired-marker-char) "]")
|
||
'(".+" (dired-move-to-filename) nil (0 dired-marked-face)))
|
||
;;
|
||
;; Flagged files.
|
||
(list (concat "^[" (char-to-string dired-del-marker) "]")
|
||
'(".+" (dired-move-to-filename) nil (0 dired-flagged-face)))
|
||
;; People who are paranoid about security would consider this more
|
||
;; important than other things such as whether it is a directory.
|
||
;; But we don't want to encourage paranoia, so our default
|
||
;; should be what's most useful for non-paranoids. -- rms.
|
||
;;; ;;
|
||
;;; ;; Files that are group or world writable.
|
||
;;; (list (concat dired-re-maybe-mark dired-re-inode-size
|
||
;;; "\\([-d]\\(....w....\\|.......w.\\)\\)")
|
||
;;; '(1 dired-warning-face)
|
||
;;; '(".+" (dired-move-to-filename) nil (0 dired-warning-face)))
|
||
;; However, we don't need to highlight the file name, only the
|
||
;; permissions, to win generally. -- fx.
|
||
;; Fixme: we could also put text properties on the permission
|
||
;; fields with keymaps to frob the permissions, somewhat a la XEmacs.
|
||
(list (concat dired-re-maybe-mark dired-re-inode-size
|
||
"[-d]....\\(w\\)....") ; group writable
|
||
'(1 dired-warning-face))
|
||
(list (concat dired-re-maybe-mark dired-re-inode-size
|
||
"[-d].......\\(w\\).") ; world writable
|
||
'(1 dired-warning-face))
|
||
;;
|
||
;; Subdirectories.
|
||
(list dired-re-dir
|
||
'(".+" (dired-move-to-filename) nil (0 dired-directory-face)))
|
||
;;
|
||
;; Symbolic links.
|
||
(list dired-re-sym
|
||
'(".+" (dired-move-to-filename) nil (0 dired-symlink-face)))
|
||
;;
|
||
;; Files suffixed with `completion-ignored-extensions'.
|
||
'(eval .
|
||
;; It is quicker to first find just an extension, then go back to the
|
||
;; start of that file name. So we do this complex MATCH-ANCHORED form.
|
||
(list (concat "\\(" (regexp-opt completion-ignored-extensions) "\\|#\\)$")
|
||
'(".+" (dired-move-to-filename) nil (0 dired-ignored-face))))
|
||
;;
|
||
;; Files suffixed with `completion-ignored-extensions'
|
||
;; plus a character put in by -F.
|
||
'(eval .
|
||
(list (concat "\\(" (regexp-opt completion-ignored-extensions)
|
||
"\\|#\\)[*=|]$")
|
||
'(".+" (progn
|
||
(end-of-line)
|
||
;; If the last character is not part of the filename,
|
||
;; move back to the start of the filename
|
||
;; so it can be fontified.
|
||
;; Otherwise, leave point at the end of the line;
|
||
;; that way, nothing is fontified.
|
||
(unless (get-text-property (1- (point)) 'mouse-face)
|
||
(dired-move-to-filename)))
|
||
nil (0 dired-ignored-face))))
|
||
)
|
||
"Additional expressions to highlight in Dired mode.")
|
||
|
||
(defvar dnd-protocol-alist)
|
||
|
||
;;; Macros must be defined before they are used, for the byte compiler.
|
||
|
||
(defmacro dired-mark-if (predicate msg)
|
||
"Mark all files for which PREDICATE evals to non-nil.
|
||
PREDICATE is evaluated on each line, with point at beginning of line.
|
||
MSG is a noun phrase for the type of files being marked.
|
||
It should end with a noun that can be pluralized by adding `s'.
|
||
Return value is the number of files marked, or nil if none were marked."
|
||
`(let (buffer-read-only count)
|
||
(save-excursion
|
||
(setq count 0)
|
||
(if ,msg (message "Marking %ss..." ,msg))
|
||
(goto-char (point-min))
|
||
(while (not (eobp))
|
||
(if ,predicate
|
||
(progn
|
||
(delete-char 1)
|
||
(insert dired-marker-char)
|
||
(setq count (1+ count))))
|
||
(forward-line 1))
|
||
(if ,msg (message "%s %s%s %s%s."
|
||
count
|
||
,msg
|
||
(dired-plural-s count)
|
||
(if (eq dired-marker-char ?\040) "un" "")
|
||
(if (eq dired-marker-char dired-del-marker)
|
||
"flagged" "marked"))))
|
||
(and (> count 0) count)))
|
||
|
||
(defmacro dired-map-over-marks (body arg &optional show-progress
|
||
distinguish-one-marked)
|
||
"Eval BODY with point on each marked line. Return a list of BODY's results.
|
||
If no marked file could be found, execute BODY on the current line.
|
||
If ARG is an integer, use the next ARG (or previous -ARG, if ARG<0)
|
||
files instead of the marked files.
|
||
In that case point is dragged along. This is so that commands on
|
||
the next ARG (instead of the marked) files can be chained easily.
|
||
If ARG is otherwise non-nil, use current file instead.
|
||
If optional third arg SHOW-PROGRESS evaluates to non-nil,
|
||
redisplay the dired buffer after each file is processed.
|
||
No guarantee is made about the position on the marked line.
|
||
BODY must ensure this itself if it depends on this.
|
||
Search starts at the beginning of the buffer, thus the car of the list
|
||
corresponds to the line nearest to the buffer's bottom. This
|
||
is also true for (positive and negative) integer values of ARG.
|
||
BODY should not be too long as it is expanded four times.
|
||
|
||
If DISTINGUISH-ONE-MARKED is non-nil, then if we find just one marked file,
|
||
return (t FILENAME) instead of (FILENAME)."
|
||
;;
|
||
;;Warning: BODY must not add new lines before point - this may cause an
|
||
;;endless loop.
|
||
;;This warning should not apply any longer, sk 2-Sep-1991 14:10.
|
||
`(prog1
|
||
(let (buffer-read-only case-fold-search found results)
|
||
(if ,arg
|
||
(if (integerp ,arg)
|
||
(progn ;; no save-excursion, want to move point.
|
||
(dired-repeat-over-lines
|
||
,arg
|
||
(function (lambda ()
|
||
(if ,show-progress (sit-for 0))
|
||
(setq results (cons ,body results)))))
|
||
(if (< ,arg 0)
|
||
(nreverse results)
|
||
results))
|
||
;; non-nil, non-integer ARG means use current file:
|
||
(list ,body))
|
||
(let ((regexp (dired-marker-regexp)) next-position)
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
;; remember position of next marked file before BODY
|
||
;; can insert lines before the just found file,
|
||
;; confusing us by finding the same marked file again
|
||
;; and again and...
|
||
(setq next-position (and (re-search-forward regexp nil t)
|
||
(point-marker))
|
||
found (not (null next-position)))
|
||
(while next-position
|
||
(goto-char next-position)
|
||
(if ,show-progress (sit-for 0))
|
||
(setq results (cons ,body results))
|
||
;; move after last match
|
||
(goto-char next-position)
|
||
(forward-line 1)
|
||
(set-marker next-position nil)
|
||
(setq next-position (and (re-search-forward regexp nil t)
|
||
(point-marker)))))
|
||
(if (and ,distinguish-one-marked (= (length results) 1))
|
||
(setq results (cons t results)))
|
||
(if found
|
||
results
|
||
(list ,body)))))
|
||
;; save-excursion loses, again
|
||
(dired-move-to-filename)))
|
||
|
||
(defun dired-get-marked-files (&optional localp arg filter distinguish-one-marked)
|
||
"Return the marked files' names as list of strings.
|
||
The list is in the same order as the buffer, that is, the car is the
|
||
first marked file.
|
||
Values returned are normally absolute file names.
|
||
Optional arg LOCALP as in `dired-get-filename'.
|
||
Optional second argument ARG specifies files near point
|
||
instead of marked files. If ARG is an integer, use the next ARG files.
|
||
If ARG is otherwise non-nil, use file. Usually ARG comes from
|
||
the command's prefix arg.
|
||
Optional third argument FILTER, if non-nil, is a function to select
|
||
some of the files--those for which (funcall FILTER FILENAME) is non-nil.
|
||
|
||
If DISTINGUISH-ONE-MARKED is non-nil, then if we find just one marked file,
|
||
return (t FILENAME) instead of (FILENAME).
|
||
Don't use that together with FILTER."
|
||
(let* ((all-of-them
|
||
(save-excursion
|
||
(dired-map-over-marks
|
||
(dired-get-filename localp)
|
||
arg nil distinguish-one-marked)))
|
||
result)
|
||
(if (not filter)
|
||
(if (and distinguish-one-marked (eq (car all-of-them) t))
|
||
all-of-them
|
||
(nreverse all-of-them))
|
||
(dolist (file all-of-them)
|
||
(if (funcall filter file)
|
||
(push file result)))
|
||
result)))
|
||
|
||
;; The dired command
|
||
|
||
(defun dired-read-dir-and-switches (str)
|
||
;; For use in interactive.
|
||
(reverse (list
|
||
(if current-prefix-arg
|
||
(read-string "Dired listing switches: "
|
||
dired-listing-switches))
|
||
;; If a dialog is about to be used, call read-directory-name so
|
||
;; the dialog code knows we want directories. Some dialogs can
|
||
;; only select directories or files when popped up, not both.
|
||
(if (next-read-file-uses-dialog-p)
|
||
(read-directory-name (format "Dired %s(directory): " str)
|
||
nil default-directory nil)
|
||
(read-file-name (format "Dired %s(directory): " str)
|
||
nil default-directory nil)))))
|
||
|
||
;;;###autoload (define-key ctl-x-map "d" 'dired)
|
||
;;;###autoload
|
||
(defun dired (dirname &optional switches)
|
||
"\"Edit\" directory DIRNAME--delete, rename, print, etc. some files in it.
|
||
Optional second argument SWITCHES specifies the `ls' options used.
|
||
\(Interactively, use a prefix argument to be able to specify SWITCHES.)
|
||
Dired displays a list of files in DIRNAME (which may also have
|
||
shell wildcards appended to select certain files). If DIRNAME is a cons,
|
||
its first element is taken as the directory name and the rest as an explicit
|
||
list of files to make directory entries for.
|
||
\\<dired-mode-map>\
|
||
You can move around in it with the usual commands.
|
||
You can flag files for deletion with \\[dired-flag-file-deletion] and then
|
||
delete them by typing \\[dired-do-flagged-delete].
|
||
Type \\[describe-mode] after entering Dired for more info.
|
||
|
||
If DIRNAME is already in a dired buffer, that buffer is used without refresh."
|
||
;; Cannot use (interactive "D") because of wildcards.
|
||
(interactive (dired-read-dir-and-switches ""))
|
||
(switch-to-buffer (dired-noselect dirname switches)))
|
||
|
||
;;;###autoload (define-key ctl-x-4-map "d" 'dired-other-window)
|
||
;;;###autoload
|
||
(defun dired-other-window (dirname &optional switches)
|
||
"\"Edit\" directory DIRNAME. Like `dired' but selects in another window."
|
||
(interactive (dired-read-dir-and-switches "in other window "))
|
||
(switch-to-buffer-other-window (dired-noselect dirname switches)))
|
||
|
||
;;;###autoload (define-key ctl-x-5-map "d" 'dired-other-frame)
|
||
;;;###autoload
|
||
(defun dired-other-frame (dirname &optional switches)
|
||
"\"Edit\" directory DIRNAME. Like `dired' but makes a new frame."
|
||
(interactive (dired-read-dir-and-switches "in other frame "))
|
||
(switch-to-buffer-other-frame (dired-noselect dirname switches)))
|
||
|
||
;;;###autoload
|
||
(defun dired-noselect (dir-or-list &optional switches)
|
||
"Like `dired' but returns the dired buffer as value, does not select it."
|
||
(or dir-or-list (setq dir-or-list default-directory))
|
||
;; This loses the distinction between "/foo/*/" and "/foo/*" that
|
||
;; some shells make:
|
||
(let (dirname initially-was-dirname)
|
||
(if (consp dir-or-list)
|
||
(setq dirname (car dir-or-list))
|
||
(setq dirname dir-or-list))
|
||
(setq initially-was-dirname
|
||
(string= (file-name-as-directory dirname) dirname))
|
||
(setq dirname (abbreviate-file-name
|
||
(expand-file-name (directory-file-name dirname))))
|
||
(if find-file-visit-truename
|
||
(setq dirname (file-truename dirname)))
|
||
;; If the argument was syntactically a directory name not a file name,
|
||
;; or if it happens to name a file that is a directory,
|
||
;; convert it syntactically to a directory name.
|
||
;; The reason for checking initially-was-dirname
|
||
;; and not just file-directory-p
|
||
;; is that file-directory-p is slow over ftp.
|
||
(if (or initially-was-dirname (file-directory-p dirname))
|
||
(setq dirname (file-name-as-directory dirname)))
|
||
(if (consp dir-or-list)
|
||
(setq dir-or-list (cons dirname (cdr dir-or-list)))
|
||
(setq dir-or-list dirname))
|
||
(dired-internal-noselect dir-or-list switches)))
|
||
|
||
;; The following is an internal dired function. It returns non-nil if
|
||
;; the directory visited by the current dired buffer has changed on
|
||
;; disk. DIRNAME should be the directory name of that directory.
|
||
(defun dired-directory-changed-p (dirname)
|
||
(not (let ((attributes (file-attributes dirname))
|
||
(modtime (visited-file-modtime)))
|
||
(or (eq modtime 0)
|
||
(not (eq (car attributes) t))
|
||
(equal (nth 5 attributes) modtime)))))
|
||
|
||
(defun dired-buffer-stale-p (&optional noconfirm)
|
||
"Return non-nil if current dired buffer needs updating.
|
||
If NOCONFIRM is non-nil, then this function always returns nil
|
||
for a remote directory. This feature is used by Auto Revert Mode."
|
||
(let ((dirname
|
||
(if (consp dired-directory) (car dired-directory) dired-directory)))
|
||
(and (stringp dirname)
|
||
(not (when noconfirm (file-remote-p dirname)))
|
||
(file-readable-p dirname)
|
||
(dired-directory-changed-p dirname))))
|
||
|
||
;; Separate function from dired-noselect for the sake of dired-vms.el.
|
||
(defun dired-internal-noselect (dir-or-list &optional switches mode)
|
||
;; If there is an existing dired buffer for DIRNAME, just leave
|
||
;; buffer as it is (don't even call dired-revert).
|
||
;; This saves time especially for deep trees or with ange-ftp.
|
||
;; The user can type `g' easily, and it is more consistent with find-file.
|
||
;; But if SWITCHES are given they are probably different from the
|
||
;; buffer's old value, so call dired-sort-other, which does
|
||
;; revert the buffer.
|
||
;; A pity we can't possibly do "Directory has changed - refresh? "
|
||
;; like find-file does.
|
||
;; Optional argument MODE is passed to dired-find-buffer-nocreate,
|
||
;; see there.
|
||
(let* (dirname
|
||
buffer
|
||
;; note that buffer already is in dired-mode, if found
|
||
new-buffer-p
|
||
(old-buf (current-buffer)))
|
||
(if (consp dir-or-list)
|
||
(setq dirname (car dir-or-list))
|
||
(setq dirname dir-or-list))
|
||
;; Look for an existing buffer.
|
||
(setq buffer (dired-find-buffer-nocreate dirname mode)
|
||
new-buffer-p (null buffer))
|
||
(or buffer
|
||
(let ((default-major-mode 'fundamental-mode))
|
||
;; We don't want default-major-mode to run hooks and set auto-fill
|
||
;; or whatever, now that dired-mode does not
|
||
;; kill-all-local-variables any longer.
|
||
(setq buffer (create-file-buffer (directory-file-name dirname)))))
|
||
(set-buffer buffer)
|
||
(if (not new-buffer-p) ; existing buffer ...
|
||
(cond (switches ; ... but new switches
|
||
;; file list may have changed
|
||
(setq dired-directory dir-or-list)
|
||
;; this calls dired-revert
|
||
(dired-sort-other switches))
|
||
;; If directory has changed on disk, offer to revert.
|
||
((when (dired-directory-changed-p dirname)
|
||
(message "%s"
|
||
(substitute-command-keys
|
||
"Directory has changed on disk; type \\[revert-buffer] to update Dired")))))
|
||
;; Else a new buffer
|
||
(setq default-directory
|
||
;; We can do this unconditionally
|
||
;; because dired-noselect ensures that the name
|
||
;; is passed in directory name syntax
|
||
;; if it was the name of a directory at all.
|
||
(file-name-directory dirname))
|
||
(or switches (setq switches dired-listing-switches))
|
||
(if mode (funcall mode)
|
||
(dired-mode dir-or-list switches))
|
||
;; default-directory and dired-actual-switches are set now
|
||
;; (buffer-local), so we can call dired-readin:
|
||
(let ((failed t))
|
||
(unwind-protect
|
||
(progn (dired-readin)
|
||
(setq failed nil))
|
||
;; dired-readin can fail if parent directories are inaccessible.
|
||
;; Don't leave an empty buffer around in that case.
|
||
(if failed (kill-buffer buffer))))
|
||
(goto-char (point-min))
|
||
(dired-initial-position dirname))
|
||
(set-buffer old-buf)
|
||
buffer))
|
||
|
||
(defvar dired-buffers nil
|
||
;; Enlarged by dired-advertise
|
||
;; Queried by function dired-buffers-for-dir. When this detects a
|
||
;; killed buffer, it is removed from this list.
|
||
"Alist of expanded directories and their associated dired buffers.")
|
||
|
||
(defun dired-find-buffer-nocreate (dirname &optional mode)
|
||
;; This differs from dired-buffers-for-dir in that it does not consider
|
||
;; subdirs of default-directory and searches for the first match only.
|
||
;; Also, the major mode must be MODE.
|
||
(setq dirname (expand-file-name dirname))
|
||
(let (found (blist dired-buffers)) ; was (buffer-list)
|
||
(or mode (setq mode 'dired-mode))
|
||
(while blist
|
||
(if (null (buffer-name (cdr (car blist))))
|
||
(setq blist (cdr blist))
|
||
(save-excursion
|
||
(set-buffer (cdr (car blist)))
|
||
(if (and (eq major-mode mode)
|
||
dired-directory ;; nil during find-alternate-file
|
||
(equal dirname
|
||
(expand-file-name
|
||
(if (consp dired-directory)
|
||
(car dired-directory)
|
||
dired-directory))))
|
||
(setq found (cdr (car blist))
|
||
blist nil)
|
||
(setq blist (cdr blist))))))
|
||
found))
|
||
|
||
|
||
;; Read in a new dired buffer
|
||
|
||
(defun dired-readin ()
|
||
"Read in a new dired buffer.
|
||
Differs from `dired-insert-subdir' in that it accepts
|
||
wildcards, erases the buffer, and builds the subdir-alist anew
|
||
\(including making it buffer-local and clearing it first)."
|
||
|
||
;; default-directory and dired-actual-switches must be buffer-local
|
||
;; and initialized by now.
|
||
(let (dirname)
|
||
(if (consp dired-directory)
|
||
(setq dirname (car dired-directory))
|
||
(setq dirname dired-directory))
|
||
(setq dirname (expand-file-name dirname))
|
||
(save-excursion
|
||
;; This hook which may want to modify dired-actual-switches
|
||
;; based on dired-directory, e.g. with ange-ftp to a SysV host
|
||
;; where ls won't understand -Al switches.
|
||
(run-hooks 'dired-before-readin-hook)
|
||
(if (consp buffer-undo-list)
|
||
(setq buffer-undo-list nil))
|
||
(make-local-variable 'file-name-coding-system)
|
||
(setq file-name-coding-system
|
||
(or coding-system-for-read file-name-coding-system))
|
||
(let (buffer-read-only
|
||
;; Don't make undo entries for readin.
|
||
(buffer-undo-list t))
|
||
(widen)
|
||
(erase-buffer)
|
||
(dired-readin-insert))
|
||
(goto-char (point-min))
|
||
;; Must first make alist buffer local and set it to nil because
|
||
;; dired-build-subdir-alist will call dired-clear-alist first
|
||
(set (make-local-variable 'dired-subdir-alist) nil)
|
||
(dired-build-subdir-alist)
|
||
(let ((attributes (file-attributes dirname)))
|
||
(if (eq (car attributes) t)
|
||
(set-visited-file-modtime (nth 5 attributes))))
|
||
(set-buffer-modified-p nil)
|
||
;; No need to narrow since the whole buffer contains just
|
||
;; dired-readin's output, nothing else. The hook can
|
||
;; successfully use dired functions (e.g. dired-get-filename)
|
||
;; as the subdir-alist has been built in dired-readin.
|
||
(run-hooks 'dired-after-readin-hook))))
|
||
|
||
;; Subroutines of dired-readin
|
||
|
||
(defun dired-readin-insert ()
|
||
;; Insert listing for the specified dir (and maybe file list)
|
||
;; already in dired-directory, assuming a clean buffer.
|
||
(let (dir file-list)
|
||
(if (consp dired-directory)
|
||
(setq dir (car dired-directory)
|
||
file-list (cdr dired-directory))
|
||
(setq dir dired-directory
|
||
file-list nil))
|
||
(setq dir (expand-file-name dir))
|
||
(if (and (equal "" (file-name-nondirectory dir))
|
||
(not file-list))
|
||
;; If we are reading a whole single directory...
|
||
(dired-insert-directory dir dired-actual-switches nil nil t)
|
||
(if (not (file-readable-p
|
||
(directory-file-name (file-name-directory dir))))
|
||
(error "Directory %s inaccessible or nonexistent" dir)
|
||
;; Else treat it as a wildcard spec
|
||
;; unless we have an explicit list of files.
|
||
(dired-insert-directory dir dired-actual-switches
|
||
file-list (not file-list) t)))))
|
||
|
||
(defun dired-align-file (beg end)
|
||
"Align the fields of a file to the ones of surrounding lines.
|
||
BEG..END is the line where the file info is located."
|
||
;; Some versions of ls try to adjust the size of each field so as to just
|
||
;; hold the largest element ("largest" in the current invocation, of
|
||
;; course). So when a single line is output, the size of each field is
|
||
;; just big enough for that one output. Thus when dired refreshes one
|
||
;; line, the alignment if this line w.r.t the rest is messed up because
|
||
;; the fields of that one line will generally be smaller.
|
||
;;
|
||
;; To work around this problem, we here add spaces to try and
|
||
;; re-align the fields as needed. Since this is purely aesthetic,
|
||
;; it is of utmost importance that it doesn't mess up anything like
|
||
;; `dired-move-to-filename'. To this end, we limit ourselves to
|
||
;; adding spaces only, and to only add them at places where there
|
||
;; was already at least one space. This way, as long as
|
||
;; `directory-listing-before-filename-regexp' always matches spaces
|
||
;; with "*" or "+", we know we haven't made anything worse. There
|
||
;; is one spot where the exact number of spaces is important, which
|
||
;; is just before the actual filename, so we refrain from adding
|
||
;; spaces there (and within the filename as well, of course).
|
||
(save-excursion
|
||
(let (file file-col other other-col)
|
||
;; Check the there is indeed a file, and that there is anoter adjacent
|
||
;; file with which to align, and that additional spaces are needed to
|
||
;; align the filenames.
|
||
(when (and (setq file (progn (goto-char beg)
|
||
(dired-move-to-filename nil end)))
|
||
(setq file-col (current-column))
|
||
(setq other
|
||
(or (and (goto-char beg)
|
||
(zerop (forward-line -1))
|
||
(dired-move-to-filename))
|
||
(and (goto-char beg)
|
||
(zerop (forward-line 1))
|
||
(dired-move-to-filename))))
|
||
(setq other-col (current-column))
|
||
(/= file other)
|
||
;; Make sure there is some work left to do.
|
||
(> other-col file-col))
|
||
;; If we've only looked at the line above, check to see if the line
|
||
;; below exists as well and if so, align with the shorter one.
|
||
(when (and (< other file)
|
||
(goto-char beg)
|
||
(zerop (forward-line 1))
|
||
(dired-move-to-filename))
|
||
(let ((alt-col (current-column)))
|
||
(when (< alt-col other-col)
|
||
(setq other-col alt-col)
|
||
(setq other (point)))))
|
||
;; Keep positions uptodate when we insert stuff.
|
||
(if (> other file) (setq other (copy-marker other)))
|
||
(setq file (copy-marker file))
|
||
;; Main loop.
|
||
(goto-char beg)
|
||
(skip-chars-forward " ") ;Skip to the first field.
|
||
(while (and (> other-col file-col)
|
||
;; Don't touch anything just before (and after) the
|
||
;; beginning of the filename.
|
||
(> file (point)))
|
||
;; We're now just in front of a field, with a space behind us.
|
||
(let* ((curcol (current-column))
|
||
;; Nums are right-aligned.
|
||
(num-align (looking-at "[0-9]"))
|
||
;; Let's look at the other line, in the same column: we
|
||
;; should be either near the end of the previous field, or
|
||
;; in the space between that field and the next.
|
||
;; [ Of course, it's also possible that we're already within
|
||
;; the next field or even past it, but that's unlikely since
|
||
;; other-col > file-col. ]
|
||
;; Let's find the distance to the alignment-point (either
|
||
;; the beginning or the end of the next field, depending on
|
||
;; whether this field is left or right aligned).
|
||
(align-pt-offset
|
||
(save-excursion
|
||
(goto-char other)
|
||
(move-to-column curcol)
|
||
(when (looking-at
|
||
(concat
|
||
(if (eq (char-before) ?\s) " *" "[^ ]* *")
|
||
(if num-align "[0-9][^ ]*")))
|
||
(- (match-end 0) (match-beginning 0)))))
|
||
;; Now, the number of spaces to insert is align-pt-offset
|
||
;; minus the distance to the equivalent point on the
|
||
;; current line.
|
||
(spaces
|
||
(if (not num-align)
|
||
align-pt-offset
|
||
(and align-pt-offset
|
||
(save-excursion
|
||
(skip-chars-forward "^ ")
|
||
(- align-pt-offset (- (current-column) curcol)))))))
|
||
(when (and spaces (> spaces 0))
|
||
(setq file-col (+ spaces file-col))
|
||
(if (> file-col other-col)
|
||
(setq spaces (- spaces (- file-col other-col))))
|
||
(insert-char ?\s spaces)
|
||
;; Let's just make really sure we did not mess up.
|
||
(unless (save-excursion
|
||
(eq (dired-move-to-filename) (marker-position file)))
|
||
;; Damn! We messed up: let's revert the change.
|
||
(delete-char (- spaces)))))
|
||
;; Now skip to next field.
|
||
(skip-chars-forward "^ ") (skip-chars-forward " "))
|
||
(set-marker file nil)))))
|
||
|
||
|
||
(defun dired-insert-directory (dir switches &optional file-list wildcard hdr)
|
||
"Insert a directory listing of DIR, Dired style.
|
||
Use SWITCHES to make the listings.
|
||
If FILE-LIST is non-nil, list only those files.
|
||
Otherwise, if WILDCARD is non-nil, expand wildcards;
|
||
in that case, DIR should be a file name that uses wildcards.
|
||
In other cases, DIR should be a directory name or a directory filename.
|
||
If HDR is non-nil, insert a header line with the directory name."
|
||
(let ((opoint (point))
|
||
(process-environment (copy-sequence process-environment))
|
||
end)
|
||
(if (or dired-use-ls-dired (file-remote-p dir))
|
||
(setq switches (concat "--dired " switches)))
|
||
;; We used to specify the C locale here, to force English month names;
|
||
;; but this should not be necessary any more,
|
||
;; with the new value of `directory-listing-before-filename-regexp'.
|
||
(if file-list
|
||
(dolist (f file-list)
|
||
(let ((beg (point)))
|
||
(insert-directory f switches nil nil)
|
||
;; Re-align fields, if necessary.
|
||
(dired-align-file beg (point))))
|
||
(insert-directory dir switches wildcard (not wildcard)))
|
||
;; Quote certain characters, unless ls quoted them for us.
|
||
(if (not (string-match "b" dired-actual-switches))
|
||
(save-excursion
|
||
(setq end (point-marker))
|
||
(goto-char opoint)
|
||
(while (search-forward "\\" end t)
|
||
(replace-match (apply #'propertize
|
||
"\\\\"
|
||
(text-properties-at (match-beginning 0)))
|
||
nil t))
|
||
(goto-char opoint)
|
||
(while (search-forward "\^m" end t)
|
||
(replace-match (apply #'propertize
|
||
"\\015"
|
||
(text-properties-at (match-beginning 0)))
|
||
nil t))
|
||
(set-marker end nil)))
|
||
(dired-insert-set-properties opoint (point))
|
||
;; If we used --dired and it worked, the lines are already indented.
|
||
;; Otherwise, indent them.
|
||
(unless (save-excursion
|
||
(goto-char opoint)
|
||
(looking-at " "))
|
||
(let ((indent-tabs-mode nil))
|
||
(indent-rigidly opoint (point) 2)))
|
||
;; Insert text at the beginning to standardize things.
|
||
(save-excursion
|
||
(goto-char opoint)
|
||
(if (and (or hdr wildcard) (not (looking-at "^ /.*:$")))
|
||
;; Note that dired-build-subdir-alist will replace the name
|
||
;; by its expansion, so it does not matter whether what we insert
|
||
;; here is fully expanded, but it should be absolute.
|
||
(insert " " (directory-file-name (file-name-directory dir)) ":\n"))
|
||
(when wildcard
|
||
;; Insert "wildcard" line where "total" line would be for a full dir.
|
||
(insert " wildcard " (file-name-nondirectory dir) "\n")))))
|
||
|
||
(defun dired-insert-set-properties (beg end)
|
||
"Make the file names highlight when the mouse is on them."
|
||
(save-excursion
|
||
(goto-char beg)
|
||
(while (< (point) end)
|
||
(condition-case nil
|
||
(if (dired-move-to-filename)
|
||
(add-text-properties
|
||
(point)
|
||
(save-excursion
|
||
(dired-move-to-end-of-filename)
|
||
(point))
|
||
'(mouse-face highlight
|
||
help-echo "mouse-2: visit this file in other window")))
|
||
(error nil))
|
||
(forward-line 1))))
|
||
|
||
;; Reverting a dired buffer
|
||
|
||
(defun dired-revert (&optional arg noconfirm)
|
||
"Reread the dired buffer.
|
||
Must also be called after `dired-actual-switches' have changed.
|
||
Should not fail even on completely garbaged buffers.
|
||
Preserves old cursor, marks/flags, hidden-p."
|
||
(widen) ; just in case user narrowed
|
||
(let ((modflag (buffer-modified-p))
|
||
(opoint (point))
|
||
(ofile (dired-get-filename nil t))
|
||
(mark-alist nil) ; save marked files
|
||
(hidden-subdirs (dired-remember-hidden))
|
||
(old-subdir-alist (cdr (reverse dired-subdir-alist))) ; except pwd
|
||
(case-fold-search nil) ; we check for upper case ls flags
|
||
buffer-read-only)
|
||
(goto-char (point-min))
|
||
(setq mark-alist;; only after dired-remember-hidden since this unhides:
|
||
(dired-remember-marks (point-min) (point-max)))
|
||
;; treat top level dir extra (it may contain wildcards)
|
||
(dired-uncache
|
||
(if (consp dired-directory) (car dired-directory) dired-directory))
|
||
;; Run dired-after-readin-hook just once, below.
|
||
(let ((dired-after-readin-hook nil))
|
||
(dired-readin)
|
||
(dired-insert-old-subdirs old-subdir-alist))
|
||
(dired-mark-remembered mark-alist) ; mark files that were marked
|
||
;; ... run the hook for the whole buffer, and only after markers
|
||
;; have been reinserted (else omitting in dired-x would omit marked files)
|
||
(run-hooks 'dired-after-readin-hook) ; no need to narrow
|
||
(or (and ofile (dired-goto-file ofile)) ; move cursor to where it
|
||
(goto-char opoint)) ; was before
|
||
(dired-move-to-filename)
|
||
(save-excursion ; hide subdirs that were hidden
|
||
(dolist (dir hidden-subdirs)
|
||
(if (dired-goto-subdir dir)
|
||
(dired-hide-subdir 1))))
|
||
(unless modflag (restore-buffer-modified-p nil)))
|
||
;; outside of the let scope
|
||
;;; Might as well not override the user if the user changed this.
|
||
;;; (setq buffer-read-only t)
|
||
)
|
||
|
||
;; Subroutines of dired-revert
|
||
;; Some of these are also used when inserting subdirs.
|
||
|
||
(defun dired-remember-marks (beg end)
|
||
"Return alist of files and their marks, from BEG to END."
|
||
(if selective-display ; must unhide to make this work.
|
||
(let (buffer-read-only)
|
||
(subst-char-in-region beg end ?\r ?\n)))
|
||
(let (fil chr alist)
|
||
(save-excursion
|
||
(goto-char beg)
|
||
(while (re-search-forward dired-re-mark end t)
|
||
(if (setq fil (dired-get-filename nil t))
|
||
(setq chr (preceding-char)
|
||
alist (cons (cons fil chr) alist)))))
|
||
alist))
|
||
|
||
(defun dired-mark-remembered (alist)
|
||
"Mark all files remembered in ALIST.
|
||
Each element of ALIST looks like (FILE . MARKERCHAR)."
|
||
(let (elt fil chr)
|
||
(while alist
|
||
(setq elt (car alist)
|
||
alist (cdr alist)
|
||
fil (car elt)
|
||
chr (cdr elt))
|
||
(if (dired-goto-file fil)
|
||
(save-excursion
|
||
(beginning-of-line)
|
||
(delete-char 1)
|
||
(insert chr))))))
|
||
|
||
(defun dired-remember-hidden ()
|
||
"Return a list of names of subdirs currently hidden."
|
||
(let ((l dired-subdir-alist) dir pos result)
|
||
(while l
|
||
(setq dir (car (car l))
|
||
pos (cdr (car l))
|
||
l (cdr l))
|
||
(goto-char pos)
|
||
(skip-chars-forward "^\r\n")
|
||
(if (eq (following-char) ?\r)
|
||
(setq result (cons dir result))))
|
||
result))
|
||
|
||
(defun dired-insert-old-subdirs (old-subdir-alist)
|
||
"Try to insert all subdirs that were displayed before.
|
||
Do so according to the former subdir alist OLD-SUBDIR-ALIST."
|
||
(or (string-match "R" dired-actual-switches)
|
||
(let (elt dir)
|
||
(while old-subdir-alist
|
||
(setq elt (car old-subdir-alist)
|
||
old-subdir-alist (cdr old-subdir-alist)
|
||
dir (car elt))
|
||
(condition-case ()
|
||
(progn
|
||
(dired-uncache dir)
|
||
(dired-insert-subdir dir))
|
||
(error nil))))))
|
||
|
||
(defun dired-uncache (dir)
|
||
"Remove directory DIR from any directory cache."
|
||
(let ((handler (find-file-name-handler dir 'dired-uncache)))
|
||
(if handler
|
||
(funcall handler 'dired-uncache dir))))
|
||
|
||
;; dired mode key bindings and initialization
|
||
|
||
(defvar dired-mode-map
|
||
;; This looks ugly when substitute-command-keys uses C-d instead d:
|
||
;; (define-key dired-mode-map "\C-d" 'dired-flag-file-deletion)
|
||
(let ((map (make-keymap)))
|
||
(suppress-keymap map)
|
||
(define-key map [mouse-2] 'dired-mouse-find-file-other-window)
|
||
(define-key map [follow-link] 'mouse-face)
|
||
;; Commands to mark or flag certain categories of files
|
||
(define-key map "#" 'dired-flag-auto-save-files)
|
||
(define-key map "." 'dired-clean-directory)
|
||
(define-key map "~" 'dired-flag-backup-files)
|
||
(define-key map "&" 'dired-flag-garbage-files)
|
||
;; Upper case keys (except !) for operating on the marked files
|
||
(define-key map "A" 'dired-do-search)
|
||
(define-key map "C" 'dired-do-copy)
|
||
(define-key map "B" 'dired-do-byte-compile)
|
||
(define-key map "D" 'dired-do-delete)
|
||
(define-key map "G" 'dired-do-chgrp)
|
||
(define-key map "H" 'dired-do-hardlink)
|
||
(define-key map "L" 'dired-do-load)
|
||
(define-key map "M" 'dired-do-chmod)
|
||
(define-key map "O" 'dired-do-chown)
|
||
(define-key map "P" 'dired-do-print)
|
||
(define-key map "Q" 'dired-do-query-replace-regexp)
|
||
(define-key map "R" 'dired-do-rename)
|
||
(define-key map "S" 'dired-do-symlink)
|
||
(define-key map "T" 'dired-do-touch)
|
||
(define-key map "X" 'dired-do-shell-command)
|
||
(define-key map "Z" 'dired-do-compress)
|
||
(define-key map "!" 'dired-do-shell-command)
|
||
;; Comparison commands
|
||
(define-key map "=" 'dired-diff)
|
||
(define-key map "\M-=" 'dired-backup-diff)
|
||
;; Tree Dired commands
|
||
(define-key map "\M-\C-?" 'dired-unmark-all-files)
|
||
(define-key map "\M-\C-d" 'dired-tree-down)
|
||
(define-key map "\M-\C-u" 'dired-tree-up)
|
||
(define-key map "\M-\C-n" 'dired-next-subdir)
|
||
(define-key map "\M-\C-p" 'dired-prev-subdir)
|
||
;; move to marked files
|
||
(define-key map "\M-{" 'dired-prev-marked-file)
|
||
(define-key map "\M-}" 'dired-next-marked-file)
|
||
;; Make all regexp commands share a `%' prefix:
|
||
;; We used to get to the submap via a symbol dired-regexp-prefix,
|
||
;; but that seems to serve little purpose, and copy-keymap
|
||
;; does a better job without it.
|
||
(define-key map "%" nil)
|
||
(define-key map "%u" 'dired-upcase)
|
||
(define-key map "%l" 'dired-downcase)
|
||
(define-key map "%d" 'dired-flag-files-regexp)
|
||
(define-key map "%g" 'dired-mark-files-containing-regexp)
|
||
(define-key map "%m" 'dired-mark-files-regexp)
|
||
(define-key map "%r" 'dired-do-rename-regexp)
|
||
(define-key map "%C" 'dired-do-copy-regexp)
|
||
(define-key map "%H" 'dired-do-hardlink-regexp)
|
||
(define-key map "%R" 'dired-do-rename-regexp)
|
||
(define-key map "%S" 'dired-do-symlink-regexp)
|
||
;; Commands for marking and unmarking.
|
||
(define-key map "*" nil)
|
||
(define-key map "**" 'dired-mark-executables)
|
||
(define-key map "*/" 'dired-mark-directories)
|
||
(define-key map "*@" 'dired-mark-symlinks)
|
||
(define-key map "*%" 'dired-mark-files-regexp)
|
||
(define-key map "*c" 'dired-change-marks)
|
||
(define-key map "*s" 'dired-mark-subdir-files)
|
||
(define-key map "*m" 'dired-mark)
|
||
(define-key map "*u" 'dired-unmark)
|
||
(define-key map "*?" 'dired-unmark-all-files)
|
||
(define-key map "*!" 'dired-unmark-all-marks)
|
||
(define-key map "U" 'dired-unmark-all-marks)
|
||
(define-key map "*\177" 'dired-unmark-backward)
|
||
(define-key map "*\C-n" 'dired-next-marked-file)
|
||
(define-key map "*\C-p" 'dired-prev-marked-file)
|
||
(define-key map "*t" 'dired-toggle-marks)
|
||
;; Lower keys for commands not operating on all the marked files
|
||
(define-key map "a" 'dired-find-alternate-file)
|
||
(define-key map "d" 'dired-flag-file-deletion)
|
||
(define-key map "e" 'dired-find-file)
|
||
(define-key map "f" 'dired-find-file)
|
||
(define-key map "\C-m" 'dired-advertised-find-file)
|
||
(define-key map "g" 'revert-buffer)
|
||
(define-key map "h" 'describe-mode)
|
||
(define-key map "i" 'dired-maybe-insert-subdir)
|
||
(define-key map "j" 'dired-goto-file)
|
||
(define-key map "k" 'dired-do-kill-lines)
|
||
(define-key map "l" 'dired-do-redisplay)
|
||
(define-key map "m" 'dired-mark)
|
||
(define-key map "n" 'dired-next-line)
|
||
(define-key map "o" 'dired-find-file-other-window)
|
||
(define-key map "\C-o" 'dired-display-file)
|
||
(define-key map "p" 'dired-previous-line)
|
||
(define-key map "q" 'quit-window)
|
||
(define-key map "s" 'dired-sort-toggle-or-edit)
|
||
(define-key map "t" 'dired-toggle-marks)
|
||
(define-key map "u" 'dired-unmark)
|
||
(define-key map "v" 'dired-view-file)
|
||
(define-key map "w" 'dired-copy-filename-as-kill)
|
||
(define-key map "x" 'dired-do-flagged-delete)
|
||
(define-key map "y" 'dired-show-file-type)
|
||
(define-key map "+" 'dired-create-directory)
|
||
;; moving
|
||
(define-key map "<" 'dired-prev-dirline)
|
||
(define-key map ">" 'dired-next-dirline)
|
||
(define-key map "^" 'dired-up-directory)
|
||
(define-key map " " 'dired-next-line)
|
||
(define-key map "\C-n" 'dired-next-line)
|
||
(define-key map "\C-p" 'dired-previous-line)
|
||
(define-key map [down] 'dired-next-line)
|
||
(define-key map [up] 'dired-previous-line)
|
||
;; hiding
|
||
(define-key map "$" 'dired-hide-subdir)
|
||
(define-key map "\M-$" 'dired-hide-all)
|
||
;; misc
|
||
(define-key map "?" 'dired-summary)
|
||
(define-key map "\177" 'dired-unmark-backward)
|
||
(define-key map [remap undo] 'dired-undo)
|
||
(define-key map [remap advertised-undo] 'dired-undo)
|
||
;; thumbnail manipulation (image-dired)
|
||
(define-key map "\C-td" 'image-dired-display-thumbs)
|
||
(define-key map "\C-tt" 'image-dired-tag-files)
|
||
(define-key map "\C-tr" 'image-dired-delete-tag)
|
||
(define-key map "\C-tj" 'image-dired-jump-thumbnail-buffer)
|
||
(define-key map "\C-ti" 'image-dired-dired-display-image)
|
||
(define-key map "\C-tx" 'image-dired-dired-display-external)
|
||
(define-key map "\C-ta" 'image-dired-display-thumbs-append)
|
||
(define-key map "\C-t." 'image-dired-display-thumb)
|
||
(define-key map "\C-tc" 'image-dired-dired-comment-files)
|
||
(define-key map "\C-tf" 'image-dired-mark-tagged-files)
|
||
(define-key map "\C-t\C-t" 'image-dired-dired-insert-marked-thumbs)
|
||
(define-key map "\C-te" 'image-dired-dired-edit-comment-and-tags)
|
||
|
||
;; Make menu bar items.
|
||
|
||
;; No need to fo this, now that top-level items are fewer.
|
||
;;;;
|
||
;; Get rid of the Edit menu bar item to save space.
|
||
;(define-key map [menu-bar edit] 'undefined)
|
||
|
||
(define-key map [menu-bar subdir]
|
||
(cons "Subdir" (make-sparse-keymap "Subdir")))
|
||
|
||
(define-key map [menu-bar subdir hide-all]
|
||
'(menu-item "Hide All" dired-hide-all
|
||
:help "Hide all subdirectories, leave only header lines"))
|
||
(define-key map [menu-bar subdir hide-subdir]
|
||
'(menu-item "Hide/UnHide Subdir" dired-hide-subdir
|
||
:help "Hide or unhide current directory listing"))
|
||
(define-key map [menu-bar subdir tree-down]
|
||
'(menu-item "Tree Down" dired-tree-down
|
||
:help "Go to first subdirectory header down the tree"))
|
||
(define-key map [menu-bar subdir tree-up]
|
||
'(menu-item "Tree Up" dired-tree-up
|
||
:help "Go to first subdirectory header up the tree"))
|
||
(define-key map [menu-bar subdir up]
|
||
'(menu-item "Up Directory" dired-up-directory
|
||
:help "Edit the parent directory"))
|
||
(define-key map [menu-bar subdir prev-subdir]
|
||
'(menu-item "Prev Subdir" dired-prev-subdir
|
||
:help "Go to previous subdirectory header line"))
|
||
(define-key map [menu-bar subdir next-subdir]
|
||
'(menu-item "Next Subdir" dired-next-subdir
|
||
:help "Go to next subdirectory header line"))
|
||
(define-key map [menu-bar subdir prev-dirline]
|
||
'(menu-item "Prev Dirline" dired-prev-dirline
|
||
:help "Move to next directory-file line"))
|
||
(define-key map [menu-bar subdir next-dirline]
|
||
'(menu-item "Next Dirline" dired-next-dirline
|
||
:help "Move to previous directory-file line"))
|
||
(define-key map [menu-bar subdir insert]
|
||
'(menu-item "Insert This Subdir" dired-maybe-insert-subdir
|
||
:help "Insert contents of subdirectory"))
|
||
|
||
(define-key map [menu-bar immediate]
|
||
(cons "Immediate" (make-sparse-keymap "Immediate")))
|
||
|
||
(define-key map
|
||
[menu-bar immediate image-dired-dired-display-external]
|
||
'(menu-item "Display Image Externally" image-dired-dired-display-external
|
||
:help "Display image in external viewer"))
|
||
(define-key map
|
||
[menu-bar immediate image-dired-dired-display-image]
|
||
'(menu-item "Display Image" image-dired-dired-display-image
|
||
:help "Display sized image in a separate window"))
|
||
|
||
(define-key map [menu-bar immediate dashes-4]
|
||
'("--"))
|
||
|
||
(define-key map [menu-bar immediate revert-buffer]
|
||
'(menu-item "Refresh" revert-buffer
|
||
:help "Update contents of shown directories"))
|
||
|
||
(define-key map [menu-bar immediate dashes]
|
||
'("--"))
|
||
|
||
(define-key map [menu-bar immediate compare-directories]
|
||
'(menu-item "Compare Directories..." dired-compare-directories
|
||
:help "Mark files with different attributes in two dired buffers"))
|
||
(define-key map [menu-bar immediate backup-diff]
|
||
'(menu-item "Compare with Backup" dired-backup-diff
|
||
:help "Diff file at cursor with its latest backup"))
|
||
(define-key map [menu-bar immediate diff]
|
||
'(menu-item "Diff..." dired-diff
|
||
:help "Compare file at cursor with another file"))
|
||
(define-key map [menu-bar immediate view]
|
||
'(menu-item "View This File" dired-view-file
|
||
:help "Examine file at cursor in read-only mode"))
|
||
(define-key map [menu-bar immediate display]
|
||
'(menu-item "Display in Other Window" dired-display-file
|
||
:help "Display file at cursor in other window"))
|
||
(define-key map [menu-bar immediate find-file-other-window]
|
||
'(menu-item "Find in Other Window" dired-find-file-other-window
|
||
:help "Edit file at cursor in other window"))
|
||
(define-key map [menu-bar immediate find-file]
|
||
'(menu-item "Find This File" dired-find-file
|
||
:help "Edit file at cursor"))
|
||
(define-key map [menu-bar immediate create-directory]
|
||
'(menu-item "Create Directory..." dired-create-directory))
|
||
(define-key map [menu-bar immediate wdired-mode]
|
||
'(menu-item "Edit File Names" wdired-change-to-wdired-mode))
|
||
|
||
(define-key map [menu-bar regexp]
|
||
(cons "Regexp" (make-sparse-keymap "Regexp")))
|
||
|
||
(define-key map
|
||
[menu-bar regexp image-dired-mark-tagged-files]
|
||
'(menu-item "Mark From Image Tag..." image-dired-mark-tagged-files
|
||
:help "Mark files whose image tags matches regexp"))
|
||
|
||
(define-key map [menu-bar regexp dashes-1]
|
||
'("--"))
|
||
|
||
(define-key map [menu-bar regexp downcase]
|
||
'(menu-item "Downcase" dired-downcase
|
||
;; When running on plain MS-DOS, there's only one
|
||
;; letter-case for file names.
|
||
:enable (or (not (fboundp 'msdos-long-file-names))
|
||
(msdos-long-file-names))
|
||
:help "Rename marked files to lower-case name"))
|
||
(define-key map [menu-bar regexp upcase]
|
||
'(menu-item "Upcase" dired-upcase
|
||
:enable (or (not (fboundp 'msdos-long-file-names))
|
||
(msdos-long-file-names))
|
||
:help "Rename marked files to upper-case name"))
|
||
(define-key map [menu-bar regexp hardlink]
|
||
'(menu-item "Hardlink..." dired-do-hardlink-regexp
|
||
:help "Make hard links for files matching regexp"))
|
||
(define-key map [menu-bar regexp symlink]
|
||
'(menu-item "Symlink..." dired-do-symlink-regexp
|
||
:visible (fboundp 'make-symbolic-link)
|
||
:help "Make symbolic links for files matching regexp"))
|
||
(define-key map [menu-bar regexp rename]
|
||
'(menu-item "Rename..." dired-do-rename-regexp
|
||
:help "Rename marked files matching regexp"))
|
||
(define-key map [menu-bar regexp copy]
|
||
'(menu-item "Copy..." dired-do-copy-regexp
|
||
:help "Copy marked files matching regexp"))
|
||
(define-key map [menu-bar regexp flag]
|
||
'(menu-item "Flag..." dired-flag-files-regexp
|
||
:help "Flag files matching regexp for deletion"))
|
||
(define-key map [menu-bar regexp mark]
|
||
'(menu-item "Mark..." dired-mark-files-regexp
|
||
:help "Mark files matching regexp for future operations"))
|
||
(define-key map [menu-bar regexp mark-cont]
|
||
'(menu-item "Mark Containing..." dired-mark-files-containing-regexp
|
||
:help "Mark files whose contents matches regexp"))
|
||
|
||
(define-key map [menu-bar mark]
|
||
(cons "Mark" (make-sparse-keymap "Mark")))
|
||
|
||
(define-key map [menu-bar mark prev]
|
||
'(menu-item "Previous Marked" dired-prev-marked-file
|
||
:help "Move to previous marked file"))
|
||
(define-key map [menu-bar mark next]
|
||
'(menu-item "Next Marked" dired-next-marked-file
|
||
:help "Move to next marked file"))
|
||
(define-key map [menu-bar mark marks]
|
||
'(menu-item "Change Marks..." dired-change-marks
|
||
:help "Replace marker with another character"))
|
||
(define-key map [menu-bar mark unmark-all]
|
||
'(menu-item "Unmark All" dired-unmark-all-marks))
|
||
(define-key map [menu-bar mark symlinks]
|
||
'(menu-item "Mark Symlinks" dired-mark-symlinks
|
||
:visible (fboundp 'make-symbolic-link)
|
||
:help "Mark all symbolic links"))
|
||
(define-key map [menu-bar mark directories]
|
||
'(menu-item "Mark Directories" dired-mark-directories
|
||
:help "Mark all directories except `.' and `..'"))
|
||
(define-key map [menu-bar mark directory]
|
||
'(menu-item "Mark Old Backups" dired-clean-directory
|
||
:help "Flag old numbered backups for deletion"))
|
||
(define-key map [menu-bar mark executables]
|
||
'(menu-item "Mark Executables" dired-mark-executables
|
||
:help "Mark all executable files"))
|
||
(define-key map [menu-bar mark garbage-files]
|
||
'(menu-item "Flag Garbage Files" dired-flag-garbage-files
|
||
:help "Flag unneeded files for deletion"))
|
||
(define-key map [menu-bar mark backup-files]
|
||
'(menu-item "Flag Backup Files" dired-flag-backup-files
|
||
:help "Flag all backup files for deletion"))
|
||
(define-key map [menu-bar mark auto-save-files]
|
||
'(menu-item "Flag Auto-save Files" dired-flag-auto-save-files
|
||
:help "Flag auto-save files for deletion"))
|
||
(define-key map [menu-bar mark deletion]
|
||
'(menu-item "Flag" dired-flag-file-deletion
|
||
:help "Flag current line's file for deletion"))
|
||
(define-key map [menu-bar mark unmark]
|
||
'(menu-item "Unmark" dired-unmark
|
||
:help "Unmark or unflag current line's file"))
|
||
(define-key map [menu-bar mark mark]
|
||
'(menu-item "Mark" dired-mark
|
||
:help "Mark current line's file for future operations"))
|
||
(define-key map [menu-bar mark toggle-marks]
|
||
'(menu-item "Toggle Marks" dired-toggle-marks
|
||
:help "Mark unmarked files, unmark marked ones"))
|
||
|
||
(define-key map [menu-bar operate]
|
||
(cons "Operate" (make-sparse-keymap "Operate")))
|
||
(define-key map
|
||
[menu-bar operate image-dired-delete-tag]
|
||
'(menu-item "Delete Image Tag..." image-dired-delete-tag
|
||
:help "Delete image tag from current or marked files"))
|
||
(define-key map
|
||
[menu-bar operate image-dired-tag-files]
|
||
'(menu-item "Add Image Tags..." image-dired-tag-files
|
||
:help "Add image tags to current or marked files"))
|
||
(define-key map
|
||
[menu-bar operate image-dired-dired-comment-files]
|
||
'(menu-item "Add Image Comment..." image-dired-dired-comment-files
|
||
:help "Add image comment to current or marked files"))
|
||
(define-key map
|
||
[menu-bar operate image-dired-display-thumbs]
|
||
'(menu-item "Display Image-Dired" image-dired-display-thumbs
|
||
:help "Display image-dired for current or marked image files"))
|
||
|
||
(define-key map [menu-bar operate dashes-3]
|
||
'("--"))
|
||
|
||
(define-key map [menu-bar operate query-replace]
|
||
'(menu-item "Query Replace in Files..." dired-do-query-replace-regexp
|
||
:help "Replace regexp in marked files"))
|
||
(define-key map [menu-bar operate search]
|
||
'(menu-item "Search Files..." dired-do-search
|
||
:help "Search marked files for regexp"))
|
||
(define-key map [menu-bar operate chown]
|
||
'(menu-item "Change Owner..." dired-do-chown
|
||
:visible (not (memq system-type '(ms-dos windows-nt)))
|
||
:help "Change the owner of marked files"))
|
||
(define-key map [menu-bar operate chgrp]
|
||
'(menu-item "Change Group..." dired-do-chgrp
|
||
:visible (not (memq system-type '(ms-dos windows-nt)))
|
||
:help "Change the group of marked files"))
|
||
(define-key map [menu-bar operate chmod]
|
||
'(menu-item "Change Mode..." dired-do-chmod
|
||
:help "Change mode (attributes) of marked files"))
|
||
(define-key map [menu-bar operate touch]
|
||
'(menu-item "Change Timestamp..." dired-do-touch
|
||
:help "Change timestamp of marked files"))
|
||
(define-key map [menu-bar operate load]
|
||
'(menu-item "Load" dired-do-load
|
||
:help "Load marked Emacs Lisp files"))
|
||
(define-key map [menu-bar operate compile]
|
||
'(menu-item "Byte-compile" dired-do-byte-compile
|
||
:help "Byte-compile marked Emacs Lisp files"))
|
||
(define-key map [menu-bar operate compress]
|
||
'(menu-item "Compress" dired-do-compress
|
||
:help "Compress/uncompress marked files"))
|
||
(define-key map [menu-bar operate print]
|
||
'(menu-item "Print..." dired-do-print
|
||
:help "Ask for print command and print marked files"))
|
||
(define-key map [menu-bar operate hardlink]
|
||
'(menu-item "Hardlink to..." dired-do-hardlink
|
||
:help "Make hard links for current or marked files"))
|
||
(define-key map [menu-bar operate symlink]
|
||
'(menu-item "Symlink to..." dired-do-symlink
|
||
:visible (fboundp 'make-symbolic-link)
|
||
:help "Make symbolic links for current or marked files"))
|
||
(define-key map [menu-bar operate command]
|
||
'(menu-item "Shell Command..." dired-do-shell-command
|
||
:help "Run a shell command on each of marked files"))
|
||
(define-key map [menu-bar operate delete]
|
||
'(menu-item "Delete" dired-do-delete
|
||
:help "Delete current file or all marked files"))
|
||
(define-key map [menu-bar operate rename]
|
||
'(menu-item "Rename to..." dired-do-rename
|
||
:help "Rename current file or move marked files"))
|
||
(define-key map [menu-bar operate copy]
|
||
'(menu-item "Copy to..." dired-do-copy
|
||
:help "Copy current file or all marked files"))
|
||
|
||
map)
|
||
"Local keymap for `dired-mode' buffers.")
|
||
|
||
;; Dired mode is suitable only for specially formatted data.
|
||
(put 'dired-mode 'mode-class 'special)
|
||
|
||
;; Autoload cookie needed by desktop.el
|
||
;;;###autoload
|
||
(defun dired-mode (&optional dirname switches)
|
||
"\
|
||
Mode for \"editing\" directory listings.
|
||
In Dired, you are \"editing\" a list of the files in a directory and
|
||
\(optionally) its subdirectories, in the format of `ls -lR'.
|
||
Each directory is a page: use \\[backward-page] and \\[forward-page] to move pagewise.
|
||
\"Editing\" means that you can run shell commands on files, visit,
|
||
compress, load or byte-compile them, change their file attributes
|
||
and insert subdirectories into the same buffer. You can \"mark\"
|
||
files for later commands or \"flag\" them for deletion, either file
|
||
by file or all files matching certain criteria.
|
||
You can move using the usual cursor motion commands.\\<dired-mode-map>
|
||
Letters no longer insert themselves. Digits are prefix arguments.
|
||
Instead, type \\[dired-flag-file-deletion] to flag a file for Deletion.
|
||
Type \\[dired-mark] to Mark a file or subdirectory for later commands.
|
||
Most commands operate on the marked files and use the current file
|
||
if no files are marked. Use a numeric prefix argument to operate on
|
||
the next ARG (or previous -ARG if ARG<0) files, or just `1'
|
||
to operate on the current file only. Prefix arguments override marks.
|
||
Mark-using commands display a list of failures afterwards. Type \\[dired-summary]
|
||
to see why something went wrong.
|
||
Type \\[dired-unmark] to Unmark a file or all files of a subdirectory.
|
||
Type \\[dired-unmark-backward] to back up one line and unflag.
|
||
Type \\[dired-do-flagged-delete] to eXecute the deletions requested.
|
||
Type \\[dired-advertised-find-file] to Find the current line's file
|
||
(or dired it in another buffer, if it is a directory).
|
||
Type \\[dired-find-file-other-window] to find file or dired directory in Other window.
|
||
Type \\[dired-maybe-insert-subdir] to Insert a subdirectory in this buffer.
|
||
Type \\[dired-do-rename] to Rename a file or move the marked files to another directory.
|
||
Type \\[dired-do-copy] to Copy files.
|
||
Type \\[dired-sort-toggle-or-edit] to toggle Sorting by name/date or change the `ls' switches.
|
||
Type \\[revert-buffer] to read all currently expanded directories aGain.
|
||
This retains all marks and hides subdirs again that were hidden before.
|
||
SPC and DEL can be used to move down and up by lines.
|
||
|
||
If Dired ever gets confused, you can either type \\[revert-buffer] \
|
||
to read the
|
||
directories again, type \\[dired-do-redisplay] \
|
||
to relist a single or the marked files or a
|
||
subdirectory, or type \\[dired-build-subdir-alist] to parse the buffer
|
||
again for the directory tree.
|
||
|
||
Customization variables (rename this buffer and type \\[describe-variable] on each line
|
||
for more info):
|
||
|
||
`dired-listing-switches'
|
||
`dired-trivial-filenames'
|
||
`dired-shrink-to-fit'
|
||
`dired-marker-char'
|
||
`dired-del-marker'
|
||
`dired-keep-marker-rename'
|
||
`dired-keep-marker-copy'
|
||
`dired-keep-marker-hardlink'
|
||
`dired-keep-marker-symlink'
|
||
|
||
Hooks (use \\[describe-variable] to see their documentation):
|
||
|
||
`dired-before-readin-hook'
|
||
`dired-after-readin-hook'
|
||
`dired-mode-hook'
|
||
`dired-load-hook'
|
||
|
||
Keybindings:
|
||
\\{dired-mode-map}"
|
||
;; Not to be called interactively (e.g. dired-directory will be set
|
||
;; to default-directory, which is wrong with wildcards).
|
||
(kill-all-local-variables)
|
||
(use-local-map dired-mode-map)
|
||
(dired-advertise) ; default-directory is already set
|
||
(setq major-mode 'dired-mode
|
||
mode-name "Dired"
|
||
;; case-fold-search nil
|
||
buffer-read-only t
|
||
selective-display t ; for subdirectory hiding
|
||
mode-line-buffer-identification
|
||
(propertized-buffer-identification "%17b"))
|
||
(set (make-local-variable 'revert-buffer-function)
|
||
(function dired-revert))
|
||
(set (make-local-variable 'buffer-stale-function)
|
||
(function dired-buffer-stale-p))
|
||
(set (make-local-variable 'page-delimiter)
|
||
"\n\n")
|
||
(set (make-local-variable 'dired-directory)
|
||
(or dirname default-directory))
|
||
;; list-buffers uses this to display the dir being edited in this buffer.
|
||
(set (make-local-variable 'list-buffers-directory)
|
||
(expand-file-name (if (listp dired-directory)
|
||
(car dired-directory)
|
||
dired-directory)))
|
||
(set (make-local-variable 'dired-actual-switches)
|
||
(or switches dired-listing-switches))
|
||
(set (make-local-variable 'font-lock-defaults)
|
||
'(dired-font-lock-keywords t nil nil beginning-of-line))
|
||
(set (make-local-variable 'desktop-save-buffer)
|
||
'dired-desktop-buffer-misc-data)
|
||
(setq dired-switches-alist nil)
|
||
(dired-sort-other dired-actual-switches t)
|
||
(when (featurep 'dnd)
|
||
(set (make-local-variable 'dnd-protocol-alist)
|
||
(append dired-dnd-protocol-alist dnd-protocol-alist)))
|
||
(run-mode-hooks 'dired-mode-hook))
|
||
|
||
;; Idiosyncratic dired commands that don't deal with marks.
|
||
|
||
(defun dired-summary ()
|
||
"Summarize basic Dired commands and show recent dired errors."
|
||
(interactive)
|
||
(dired-why)
|
||
;>> this should check the key-bindings and use substitute-command-keys if non-standard
|
||
(message
|
||
"d-elete, u-ndelete, x-punge, f-ind, o-ther window, R-ename, C-opy, h-elp"))
|
||
|
||
(defun dired-undo ()
|
||
"Undo in a dired buffer.
|
||
This doesn't recover lost files, it just undoes changes in the buffer itself.
|
||
You can use it to recover marks, killed lines or subdirs."
|
||
(interactive)
|
||
(let (buffer-read-only)
|
||
(undo))
|
||
(dired-build-subdir-alist)
|
||
(message "Change in dired buffer undone.
|
||
Actual changes in files cannot be undone by Emacs."))
|
||
|
||
(defun dired-next-line (arg)
|
||
"Move down lines then position at filename.
|
||
Optional prefix ARG says how many lines to move; default is one line."
|
||
(interactive "p")
|
||
(next-line arg)
|
||
(dired-move-to-filename))
|
||
|
||
(defun dired-previous-line (arg)
|
||
"Move up lines then position at filename.
|
||
Optional prefix ARG says how many lines to move; default is one line."
|
||
(interactive "p")
|
||
(previous-line arg)
|
||
(dired-move-to-filename))
|
||
|
||
(defun dired-next-dirline (arg &optional opoint)
|
||
"Goto ARG'th next directory file line."
|
||
(interactive "p")
|
||
(or opoint (setq opoint (point)))
|
||
(if (if (> arg 0)
|
||
(re-search-forward dired-re-dir nil t arg)
|
||
(beginning-of-line)
|
||
(re-search-backward dired-re-dir nil t (- arg)))
|
||
(dired-move-to-filename) ; user may type `i' or `f'
|
||
(goto-char opoint)
|
||
(error "No more subdirectories")))
|
||
|
||
(defun dired-prev-dirline (arg)
|
||
"Goto ARG'th previous directory file line."
|
||
(interactive "p")
|
||
(dired-next-dirline (- arg)))
|
||
|
||
(defun dired-up-directory (&optional other-window)
|
||
"Run Dired on parent directory of current directory.
|
||
Find the parent directory either in this buffer or another buffer.
|
||
Creates a buffer if necessary."
|
||
(interactive "P")
|
||
(let* ((dir (dired-current-directory))
|
||
(up (file-name-directory (directory-file-name dir))))
|
||
(or (dired-goto-file (directory-file-name dir))
|
||
;; Only try dired-goto-subdir if buffer has more than one dir.
|
||
(and (cdr dired-subdir-alist)
|
||
(dired-goto-subdir up))
|
||
(progn
|
||
(if other-window
|
||
(dired-other-window up)
|
||
(dired up))
|
||
(dired-goto-file dir)))))
|
||
|
||
(defun dired-get-file-for-visit ()
|
||
"Get the current line's file name, with an error if file does not exist."
|
||
(interactive)
|
||
;; We pass t for second arg so that we don't get error for `.' and `..'.
|
||
(let ((raw (dired-get-filename nil t))
|
||
file-name)
|
||
(if (null raw)
|
||
(error "No file on this line"))
|
||
(setq file-name (file-name-sans-versions raw t))
|
||
(if (file-exists-p file-name)
|
||
file-name
|
||
(if (file-symlink-p file-name)
|
||
(error "File is a symlink to a nonexistent target")
|
||
(error "File no longer exists; type `g' to update dired buffer")))))
|
||
|
||
;; Force `f' rather than `e' in the mode doc:
|
||
(defalias 'dired-advertised-find-file 'dired-find-file)
|
||
(defun dired-find-file ()
|
||
"In Dired, visit the file or directory named on this line."
|
||
(interactive)
|
||
;; Bind `find-file-run-dired' so that the command works on directories
|
||
;; too, independent of the user's setting.
|
||
(let ((find-file-run-dired t))
|
||
(find-file (dired-get-file-for-visit))))
|
||
|
||
(defun dired-find-alternate-file ()
|
||
"In Dired, visit this file or directory instead of the dired buffer."
|
||
(interactive)
|
||
(set-buffer-modified-p nil)
|
||
(find-alternate-file (dired-get-file-for-visit)))
|
||
;; Don't override the setting from .emacs.
|
||
;;;###autoload (put 'dired-find-alternate-file 'disabled t)
|
||
|
||
(defun dired-mouse-find-file-other-window (event)
|
||
"In Dired, visit the file or directory name you click on."
|
||
(interactive "e")
|
||
(let (window pos file)
|
||
(save-excursion
|
||
(setq window (posn-window (event-end event))
|
||
pos (posn-point (event-end event)))
|
||
(if (not (windowp window))
|
||
(error "No file chosen"))
|
||
(set-buffer (window-buffer window))
|
||
(goto-char pos)
|
||
(setq file (dired-get-file-for-visit)))
|
||
(if (file-directory-p file)
|
||
(or (and (cdr dired-subdir-alist)
|
||
(dired-goto-subdir file))
|
||
(progn
|
||
(select-window window)
|
||
(dired-other-window file)))
|
||
(select-window window)
|
||
(find-file-other-window (file-name-sans-versions file t)))))
|
||
|
||
(defun dired-view-file ()
|
||
"In Dired, examine a file in view mode, returning to Dired when done.
|
||
When file is a directory, show it in this buffer if it is inserted.
|
||
Otherwise, display it in another buffer."
|
||
(interactive)
|
||
(let ((file (dired-get-file-for-visit)))
|
||
(if (file-directory-p file)
|
||
(or (and (cdr dired-subdir-alist)
|
||
(dired-goto-subdir file))
|
||
(dired file))
|
||
(view-file file))))
|
||
|
||
(defun dired-find-file-other-window ()
|
||
"In Dired, visit this file or directory in another window."
|
||
(interactive)
|
||
(find-file-other-window (dired-get-file-for-visit)))
|
||
|
||
(defun dired-display-file ()
|
||
"In Dired, display this file or directory in another window."
|
||
(interactive)
|
||
(display-buffer (find-file-noselect (dired-get-file-for-visit))))
|
||
|
||
;;; Functions for extracting and manipulating file names in Dired buffers.
|
||
|
||
(defun dired-get-filename (&optional localp no-error-if-not-filep)
|
||
"In Dired, return name of file mentioned on this line.
|
||
Value returned normally includes the directory name.
|
||
Optional arg LOCALP with value `no-dir' means don't include directory
|
||
name in result. A value of `verbatim' means to return the name exactly as
|
||
it occurs in the buffer, and a value of t means construct name relative to
|
||
`default-directory', which still may contain slashes if in a subdirectory.
|
||
Optional arg NO-ERROR-IF-NOT-FILEP means treat `.' and `..' as
|
||
regular filenames and return nil if no filename on this line.
|
||
Otherwise, an error occurs in these cases."
|
||
(let (case-fold-search file p1 p2 already-absolute)
|
||
(save-excursion
|
||
(if (setq p1 (dired-move-to-filename (not no-error-if-not-filep)))
|
||
(setq p2 (dired-move-to-end-of-filename no-error-if-not-filep))))
|
||
;; nil if no file on this line, but no-error-if-not-filep is t:
|
||
(if (setq file (and p1 p2 (buffer-substring p1 p2)))
|
||
(progn
|
||
;; Get rid of the mouse-face property that file names have.
|
||
(set-text-properties 0 (length file) nil file)
|
||
;; Unquote names quoted by ls or by dired-insert-directory.
|
||
;; Using read to unquote is much faster than substituting
|
||
;; \007 (4 chars) -> ^G (1 char) etc. in a lisp loop.
|
||
(setq file
|
||
(read
|
||
(concat "\""
|
||
;; Some ls -b don't escape quotes, argh!
|
||
;; This is not needed for GNU ls, though.
|
||
(or (dired-string-replace-match
|
||
"\\([^\\]\\|\\`\\)\"" file "\\1\\\\\"" nil t)
|
||
file)
|
||
"\"")))
|
||
;; The above `read' will return a unibyte string if FILE
|
||
;; contains eight-bit-control/graphic characters.
|
||
(if (and enable-multibyte-characters
|
||
(not (multibyte-string-p file)))
|
||
(setq file (string-to-multibyte file)))))
|
||
(and file (file-name-absolute-p file)
|
||
;; A relative file name can start with ~.
|
||
;; Don't treat it as absolute in this context.
|
||
(not (eq (aref file 0) ?~))
|
||
(setq already-absolute t))
|
||
(cond
|
||
((null file)
|
||
nil)
|
||
((eq localp 'verbatim)
|
||
file)
|
||
((and (not no-error-if-not-filep)
|
||
(member file '("." "..")))
|
||
(error "Cannot operate on `.' or `..'"))
|
||
((and (eq localp 'no-dir) already-absolute)
|
||
(file-name-nondirectory file))
|
||
(already-absolute
|
||
(let ((handler (find-file-name-handler file nil)))
|
||
;; check for safe-magic property so that we won't
|
||
;; put /: for names that don't really need them.
|
||
;; For instance, .gz files when auto-compression-mode is on.
|
||
(if (and handler (not (get handler 'safe-magic)))
|
||
(concat "/:" file)
|
||
file)))
|
||
((eq localp 'no-dir)
|
||
file)
|
||
((equal (dired-current-directory) "/")
|
||
(setq file (concat (dired-current-directory localp) file))
|
||
(let ((handler (find-file-name-handler file nil)))
|
||
;; check for safe-magic property so that we won't
|
||
;; put /: for names that don't really need them.
|
||
;; For instance, .gz files when auto-compression-mode is on.
|
||
(if (and handler (not (get handler 'safe-magic)))
|
||
(concat "/:" file)
|
||
file)))
|
||
(t
|
||
(concat (dired-current-directory localp) file)))))
|
||
|
||
(defun dired-string-replace-match (regexp string newtext
|
||
&optional literal global)
|
||
"Replace first match of REGEXP in STRING with NEWTEXT.
|
||
If it does not match, nil is returned instead of the new string.
|
||
Optional arg LITERAL means to take NEWTEXT literally.
|
||
Optional arg GLOBAL means to replace all matches."
|
||
(if global
|
||
(let ((start 0) ret)
|
||
(while (string-match regexp string start)
|
||
(let ((from-end (- (length string) (match-end 0))))
|
||
(setq ret (setq string (replace-match newtext t literal string)))
|
||
(setq start (- (length string) from-end))))
|
||
ret)
|
||
(if (not (string-match regexp string 0))
|
||
nil
|
||
(replace-match newtext t literal string))))
|
||
|
||
(defun dired-make-absolute (file &optional dir)
|
||
;;"Convert FILE (a file name relative to DIR) to an absolute file name."
|
||
;; We can't always use expand-file-name as this would get rid of `.'
|
||
;; or expand in / instead default-directory if DIR=="".
|
||
;; This should be good enough for ange-ftp, but might easily be
|
||
;; redefined (for VMS?).
|
||
;; It should be reasonably fast, though, as it is called in
|
||
;; dired-get-filename.
|
||
(concat (or dir default-directory) file))
|
||
|
||
(defun dired-make-relative (file &optional dir ignore)
|
||
"Convert FILE (an absolute file name) to a name relative to DIR.
|
||
If this is impossible, return FILE unchanged.
|
||
DIR must be a directory name, not a file name."
|
||
(or dir (setq dir default-directory))
|
||
;; This case comes into play if default-directory is set to
|
||
;; use ~.
|
||
(if (and (> (length dir) 0) (= (aref dir 0) ?~))
|
||
(setq dir (expand-file-name dir)))
|
||
(if (string-match (concat "^" (regexp-quote dir)) file)
|
||
(substring file (match-end 0))
|
||
;;; (or no-error
|
||
;;; (error "%s: not in directory tree growing at %s" file dir))
|
||
file))
|
||
|
||
;;; Functions for finding the file name in a dired buffer line.
|
||
|
||
(defvar dired-permission-flags-regexp
|
||
"\\([^ ]\\)[-r][-w]\\([^ ]\\)[-r][-w]\\([^ ]\\)[-r][-w]\\([^ ]\\)"
|
||
"Regular expression to match the permission flags in `ls -l'.")
|
||
|
||
;; Move to first char of filename on this line.
|
||
;; Returns position (point) or nil if no filename on this line."
|
||
(defun dired-move-to-filename (&optional raise-error eol)
|
||
"Move to the beginning of the filename on the current line.
|
||
Return the position of the beginning of the filename, or nil if none found."
|
||
;; This is the UNIX version.
|
||
(or eol (setq eol (line-end-position)))
|
||
(beginning-of-line)
|
||
;; First try assuming `ls --dired' was used.
|
||
(let ((change (next-single-property-change (point) 'dired-filename nil eol)))
|
||
(cond
|
||
((and change (< change eol))
|
||
(goto-char change))
|
||
((re-search-forward directory-listing-before-filename-regexp eol t)
|
||
(goto-char (match-end 0)))
|
||
((re-search-forward dired-permission-flags-regexp eol t)
|
||
;; Ha! There *is* a file. Our regexp-from-hell just failed to find it.
|
||
(if raise-error
|
||
(error "Unrecognized line! Check directory-listing-before-filename-regexp"))
|
||
(beginning-of-line)
|
||
nil)
|
||
(raise-error
|
||
(error "No file on this line")))))
|
||
|
||
(defun dired-move-to-end-of-filename (&optional no-error)
|
||
;; Assumes point is at beginning of filename,
|
||
;; thus the rwx bit re-search-backward below will succeed in *this*
|
||
;; line if at all. So, it should be called only after
|
||
;; (dired-move-to-filename t).
|
||
;; On failure, signals an error (with non-nil NO-ERROR just returns nil).
|
||
;; This is the UNIX version.
|
||
(if (get-text-property (point) 'dired-filename)
|
||
(goto-char (next-single-property-change (point) 'dired-filename))
|
||
(let (opoint file-type executable symlink hidden case-fold-search used-F eol)
|
||
;; case-fold-search is nil now, so we can test for capital F:
|
||
(setq used-F (string-match "F" dired-actual-switches)
|
||
opoint (point)
|
||
eol (save-excursion (end-of-line) (point))
|
||
hidden (and selective-display
|
||
(save-excursion (search-forward "\r" eol t))))
|
||
(if hidden
|
||
nil
|
||
(save-excursion ;; Find out what kind of file this is:
|
||
;; Restrict perm bits to be non-blank,
|
||
;; otherwise this matches one char to early (looking backward):
|
||
;; "l---------" (some systems make symlinks that way)
|
||
;; "----------" (plain file with zero perms)
|
||
(if (re-search-backward
|
||
dired-permission-flags-regexp nil t)
|
||
(setq file-type (char-after (match-beginning 1))
|
||
symlink (eq file-type ?l)
|
||
;; Only with -F we need to know whether it's an executable
|
||
executable (and
|
||
used-F
|
||
(string-match
|
||
"[xst]" ;; execute bit set anywhere?
|
||
(concat
|
||
(match-string 2)
|
||
(match-string 3)
|
||
(match-string 4)))))
|
||
(or no-error (error "No file on this line"))))
|
||
;; Move point to end of name:
|
||
(if symlink
|
||
(if (search-forward " -> " eol t)
|
||
(progn
|
||
(forward-char -4)
|
||
(and used-F
|
||
dired-ls-F-marks-symlinks
|
||
(eq (preceding-char) ?@) ;; did ls really mark the link?
|
||
(forward-char -1))))
|
||
(goto-char eol) ;; else not a symbolic link
|
||
;; ls -lF marks dirs, sockets, fifos and executables with exactly
|
||
;; one trailing character. (Executable bits on symlinks ain't mean
|
||
;; a thing, even to ls, but we know it's not a symlink.)
|
||
(and used-F
|
||
(or (memq file-type '(?d ?s ?p))
|
||
executable)
|
||
(forward-char -1))))
|
||
(or no-error
|
||
(not (eq opoint (point)))
|
||
(error (if hidden
|
||
(substitute-command-keys
|
||
"File line is hidden, type \\[dired-hide-subdir] to unhide")
|
||
"No file on this line")))
|
||
(if (eq opoint (point))
|
||
nil
|
||
(point)))))
|
||
|
||
|
||
;;; COPY NAMES OF MARKED FILES INTO KILL-RING.
|
||
|
||
(defun dired-copy-filename-as-kill (&optional arg)
|
||
"Copy names of marked (or next ARG) files into the kill ring.
|
||
The names are separated by a space.
|
||
With a zero prefix arg, use the absolute file name of each marked file.
|
||
With \\[universal-argument], use the file name relative to the dired buffer's
|
||
`default-directory'. (This still may contain slashes if in a subdirectory.)
|
||
|
||
If on a subdir headerline, use absolute subdirname instead;
|
||
prefix arg and marked files are ignored in this case.
|
||
|
||
You can then feed the file name(s) to other commands with \\[yank]."
|
||
(interactive "P")
|
||
(let ((string
|
||
(or (dired-get-subdir)
|
||
(mapconcat (function identity)
|
||
(if arg
|
||
(cond ((zerop (prefix-numeric-value arg))
|
||
(dired-get-marked-files))
|
||
((consp arg)
|
||
(dired-get-marked-files t))
|
||
(t
|
||
(dired-get-marked-files
|
||
'no-dir (prefix-numeric-value arg))))
|
||
(dired-get-marked-files 'no-dir))
|
||
" "))))
|
||
(if (eq last-command 'kill-region)
|
||
(kill-append string nil)
|
||
(kill-new string))
|
||
(message "%s" string)))
|
||
|
||
|
||
;; Keeping Dired buffers in sync with the filesystem and with each other
|
||
|
||
(defun dired-buffers-for-dir (dir &optional file)
|
||
;; Return a list of buffers that dired DIR (top level or in-situ subdir).
|
||
;; If FILE is non-nil, include only those whose wildcard pattern (if any)
|
||
;; matches FILE.
|
||
;; The list is in reverse order of buffer creation, most recent last.
|
||
;; As a side effect, killed dired buffers for DIR are removed from
|
||
;; dired-buffers.
|
||
(setq dir (file-name-as-directory dir))
|
||
(let ((alist dired-buffers) result elt buf)
|
||
(while alist
|
||
(setq elt (car alist)
|
||
buf (cdr elt))
|
||
(if (buffer-name buf)
|
||
(if (dired-in-this-tree dir (car elt))
|
||
(with-current-buffer buf
|
||
(and (assoc dir dired-subdir-alist)
|
||
(or (null file)
|
||
(let ((wildcards
|
||
(file-name-nondirectory dired-directory)))
|
||
(or (= 0 (length wildcards))
|
||
(string-match (dired-glob-regexp wildcards)
|
||
file))))
|
||
(setq result (cons buf result)))))
|
||
;; else buffer is killed - clean up:
|
||
(setq dired-buffers (delq elt dired-buffers)))
|
||
(setq alist (cdr alist)))
|
||
result))
|
||
|
||
(defun dired-glob-regexp (pattern)
|
||
"Convert glob-pattern PATTERN to a regular expression."
|
||
(let ((matched-in-pattern 0) ;; How many chars of PATTERN we've handled.
|
||
regexp)
|
||
(while (string-match "[[?*]" pattern matched-in-pattern)
|
||
(let ((op-end (match-end 0))
|
||
(next-op (aref pattern (match-beginning 0))))
|
||
(setq regexp (concat regexp
|
||
(regexp-quote
|
||
(substring pattern matched-in-pattern
|
||
(match-beginning 0)))))
|
||
(cond ((= next-op ??)
|
||
(setq regexp (concat regexp "."))
|
||
(setq matched-in-pattern op-end))
|
||
((= next-op ?\[)
|
||
;; Fails to handle ^ yet ????
|
||
(let* ((set-start (match-beginning 0))
|
||
(set-cont
|
||
(if (= (aref pattern (1+ set-start)) ?^)
|
||
(+ 3 set-start)
|
||
(+ 2 set-start)))
|
||
(set-end (string-match "]" pattern set-cont))
|
||
(set (substring pattern set-start (1+ set-end))))
|
||
(setq regexp (concat regexp set))
|
||
(setq matched-in-pattern (1+ set-end))))
|
||
((= next-op ?*)
|
||
(setq regexp (concat regexp ".*"))
|
||
(setq matched-in-pattern op-end)))))
|
||
(concat "\\`"
|
||
regexp
|
||
(regexp-quote
|
||
(substring pattern matched-in-pattern))
|
||
"\\'")))
|
||
|
||
|
||
|
||
(defun dired-advertise ()
|
||
;;"Advertise in variable `dired-buffers' that we dired `default-directory'."
|
||
;; With wildcards we actually advertise too much.
|
||
(let ((expanded-default (expand-file-name default-directory)))
|
||
(if (memq (current-buffer) (dired-buffers-for-dir expanded-default))
|
||
t ; we have already advertised ourselves
|
||
(setq dired-buffers
|
||
(cons (cons expanded-default (current-buffer))
|
||
dired-buffers)))))
|
||
|
||
(defun dired-unadvertise (dir)
|
||
;; Remove DIR from the buffer alist in variable dired-buffers.
|
||
;; This has the effect of removing any buffer whose main directory is DIR.
|
||
;; It does not affect buffers in which DIR is a subdir.
|
||
;; Removing is also done as a side-effect in dired-buffer-for-dir.
|
||
(setq dired-buffers
|
||
(delq (assoc (expand-file-name dir) dired-buffers) dired-buffers)))
|
||
|
||
;; Tree Dired
|
||
|
||
;;; utility functions
|
||
|
||
(defun dired-in-this-tree (file dir)
|
||
;;"Is FILE part of the directory tree starting at DIR?"
|
||
(let (case-fold-search)
|
||
(string-match (concat "^" (regexp-quote dir)) file)))
|
||
|
||
(defun dired-normalize-subdir (dir)
|
||
;; Prepend default-directory to DIR if relative file name.
|
||
;; dired-get-filename must be able to make a valid file name from a
|
||
;; file and its directory DIR.
|
||
(file-name-as-directory
|
||
(if (file-name-absolute-p dir)
|
||
dir
|
||
(expand-file-name dir default-directory))))
|
||
|
||
(defun dired-get-subdir ()
|
||
;;"Return the subdir name on this line, or nil if not on a headerline."
|
||
;; Look up in the alist whether this is a headerline.
|
||
(save-excursion
|
||
(let ((cur-dir (dired-current-directory)))
|
||
(beginning-of-line) ; alist stores b-o-l positions
|
||
(and (zerop (- (point)
|
||
(dired-get-subdir-min (assoc cur-dir
|
||
dired-subdir-alist))))
|
||
cur-dir))))
|
||
|
||
;(defun dired-get-subdir-min (elt)
|
||
; (cdr elt))
|
||
;; can't use macro, must be redefinable for other alist format in dired-nstd.
|
||
(defalias 'dired-get-subdir-min 'cdr)
|
||
|
||
(defun dired-get-subdir-max (elt)
|
||
(save-excursion
|
||
(goto-char (dired-get-subdir-min elt))
|
||
(dired-subdir-max)))
|
||
|
||
(defun dired-clear-alist ()
|
||
(while dired-subdir-alist
|
||
(set-marker (dired-get-subdir-min (car dired-subdir-alist)) nil)
|
||
(setq dired-subdir-alist (cdr dired-subdir-alist))))
|
||
|
||
(defun dired-subdir-index (dir)
|
||
;; Return an index into alist for use with nth
|
||
;; for the sake of subdir moving commands.
|
||
(let (found (index 0) (alist dired-subdir-alist))
|
||
(while alist
|
||
(if (string= dir (car (car alist)))
|
||
(setq alist nil found t)
|
||
(setq alist (cdr alist) index (1+ index))))
|
||
(if found index nil)))
|
||
|
||
(defun dired-next-subdir (arg &optional no-error-if-not-found no-skip)
|
||
"Go to next subdirectory, regardless of level."
|
||
;; Use 0 arg to go to this directory's header line.
|
||
;; NO-SKIP prevents moving to end of header line, returning whatever
|
||
;; position was found in dired-subdir-alist.
|
||
(interactive "p")
|
||
(let ((this-dir (dired-current-directory))
|
||
pos index)
|
||
;; nth with negative arg does not return nil but the first element
|
||
(setq index (- (dired-subdir-index this-dir) arg))
|
||
(setq pos (if (>= index 0)
|
||
(dired-get-subdir-min (nth index dired-subdir-alist))))
|
||
(if pos
|
||
(progn
|
||
(goto-char pos)
|
||
(or no-skip (skip-chars-forward "^\n\r"))
|
||
(point))
|
||
(if no-error-if-not-found
|
||
nil ; return nil if not found
|
||
(error "%s directory" (if (> arg 0) "Last" "First"))))))
|
||
|
||
(defun dired-build-subdir-alist (&optional switches)
|
||
"Build `dired-subdir-alist' by parsing the buffer.
|
||
Returns the new value of the alist.
|
||
If optional arg SWITCHES is non-nil, use its value
|
||
instead of `dired-actual-switches'."
|
||
(interactive)
|
||
(dired-clear-alist)
|
||
(save-excursion
|
||
(let* ((count 0)
|
||
(buffer-read-only nil)
|
||
(buffer-undo-list t)
|
||
(switches (or switches dired-actual-switches))
|
||
new-dir-name
|
||
(R-ftp-base-dir-regex
|
||
;; Used to expand subdirectory names correctly in recursive
|
||
;; ange-ftp listings.
|
||
(and (string-match "R" switches)
|
||
(string-match "\\`/.*:\\(/.*\\)" default-directory)
|
||
(concat "\\`" (match-string 1 default-directory)))))
|
||
(goto-char (point-min))
|
||
(setq dired-subdir-alist nil)
|
||
(while (re-search-forward dired-subdir-regexp nil t)
|
||
;; Avoid taking a file name ending in a colon
|
||
;; as a subdir name.
|
||
(unless (save-excursion
|
||
(goto-char (match-beginning 0))
|
||
(beginning-of-line)
|
||
(forward-char 2)
|
||
(save-match-data (looking-at dired-re-perms)))
|
||
(save-excursion
|
||
(goto-char (match-beginning 1))
|
||
(setq new-dir-name
|
||
(buffer-substring-no-properties (point) (match-end 1))
|
||
new-dir-name
|
||
(save-match-data
|
||
(if (and R-ftp-base-dir-regex
|
||
(not (string= new-dir-name default-directory))
|
||
(string-match R-ftp-base-dir-regex new-dir-name))
|
||
(concat default-directory
|
||
(substring new-dir-name (match-end 0)))
|
||
(expand-file-name new-dir-name))))
|
||
(delete-region (point) (match-end 1))
|
||
(insert new-dir-name))
|
||
(setq count (1+ count))
|
||
(dired-alist-add-1 new-dir-name
|
||
;; Place a sub directory boundary between lines.
|
||
(save-excursion
|
||
(goto-char (match-beginning 0))
|
||
(beginning-of-line)
|
||
(point-marker)))))
|
||
(if (and (> count 1) (interactive-p))
|
||
(message "Buffer includes %d directories" count)))
|
||
;; We don't need to sort it because it is in buffer order per
|
||
;; constructionem. Return new alist:
|
||
dired-subdir-alist))
|
||
|
||
(defun dired-alist-add-1 (dir new-marker)
|
||
;; Add new DIR at NEW-MARKER. Don't sort.
|
||
(setq dired-subdir-alist
|
||
(cons (cons (dired-normalize-subdir dir) new-marker)
|
||
dired-subdir-alist)))
|
||
|
||
(defun dired-goto-next-nontrivial-file ()
|
||
;; Position point on first nontrivial file after point.
|
||
(dired-goto-next-file);; so there is a file to compare with
|
||
(if (stringp dired-trivial-filenames)
|
||
(while (and (not (eobp))
|
||
(string-match dired-trivial-filenames
|
||
(file-name-nondirectory
|
||
(or (dired-get-filename nil t) ""))))
|
||
(forward-line 1)
|
||
(dired-move-to-filename))))
|
||
|
||
(defun dired-goto-next-file ()
|
||
(let ((max (1- (dired-subdir-max))))
|
||
(while (and (not (dired-move-to-filename)) (< (point) max))
|
||
(forward-line 1))))
|
||
|
||
(defun dired-goto-file (file)
|
||
"Go to line describing file FILE in this dired buffer."
|
||
;; Return value of point on success, else nil.
|
||
;; FILE must be an absolute file name.
|
||
;; Loses if FILE contains control chars like "\007" for which ls
|
||
;; either inserts "?" or "\\007" into the buffer, so we won't find
|
||
;; it in the buffer.
|
||
(interactive
|
||
(prog1 ; let push-mark display its message
|
||
(list (expand-file-name
|
||
(read-file-name "Goto file: "
|
||
(dired-current-directory))))
|
||
(push-mark)))
|
||
(setq file (directory-file-name file)) ; does no harm if no directory
|
||
(let (found case-fold-search dir)
|
||
(setq dir (or (file-name-directory file)
|
||
(error "File name `%s' is not absolute" file)))
|
||
(save-excursion
|
||
;; The hair here is to get the result of dired-goto-subdir
|
||
;; without really calling it if we don't have any subdirs.
|
||
(if (if (string= dir (expand-file-name default-directory))
|
||
(goto-char (point-min))
|
||
(and (cdr dired-subdir-alist)
|
||
(dired-goto-subdir dir)))
|
||
(let ((base (file-name-nondirectory file))
|
||
search-string
|
||
(boundary (dired-subdir-max)))
|
||
(setq search-string
|
||
(replace-regexp-in-string "\^m" "\\^m" base nil t))
|
||
(setq search-string
|
||
(replace-regexp-in-string "\\\\" "\\\\" search-string nil t))
|
||
(while (and (not found)
|
||
;; filenames are preceded by SPC, this makes
|
||
;; the search faster (e.g. for the filename "-"!).
|
||
(search-forward (concat " " search-string)
|
||
boundary 'move))
|
||
;; Match could have BASE just as initial substring or
|
||
;; or in permission bits or date or
|
||
;; not be a proper filename at all:
|
||
(if (equal base (dired-get-filename 'no-dir t))
|
||
;; Must move to filename since an (actually
|
||
;; correct) match could have been elsewhere on the
|
||
;; ;; line (e.g. "-" would match somewhere in the
|
||
;; permission bits).
|
||
(setq found (dired-move-to-filename))
|
||
;; If this isn't the right line, move forward to avoid
|
||
;; trying this line again.
|
||
(forward-line 1))))))
|
||
(and found
|
||
;; return value of point (i.e., FOUND):
|
||
(goto-char found))))
|
||
|
||
(defun dired-initial-position (dirname)
|
||
;; Where point should go in a new listing of DIRNAME.
|
||
;; Point assumed at beginning of new subdir line.
|
||
;; You may redefine this function as you wish, e.g. like in dired-x.el.
|
||
(end-of-line)
|
||
(if dired-trivial-filenames (dired-goto-next-nontrivial-file)))
|
||
|
||
;; These are hooks which make tree dired work.
|
||
;; They are in this file because other parts of dired need to call them.
|
||
;; But they don't call the rest of tree dired unless there are subdirs loaded.
|
||
|
||
;; This function is called for each retrieved filename.
|
||
;; It could stand to be faster, though it's mostly function call
|
||
;; overhead. Avoiding the function call seems to save about 10% in
|
||
;; dired-get-filename. Make it a defsubst?
|
||
(defun dired-current-directory (&optional localp)
|
||
"Return the name of the subdirectory to which this line belongs.
|
||
This returns a string with trailing slash, like `default-directory'.
|
||
Optional argument means return a file name relative to `default-directory'."
|
||
(let ((here (point))
|
||
(alist (or dired-subdir-alist
|
||
;; probably because called in a non-dired buffer
|
||
(error "No subdir-alist in %s" (current-buffer))))
|
||
elt dir)
|
||
(while alist
|
||
(setq elt (car alist)
|
||
dir (car elt)
|
||
;; use `<=' (not `<') as subdir line is part of subdir
|
||
alist (if (<= (dired-get-subdir-min elt) here)
|
||
nil ; found
|
||
(cdr alist))))
|
||
(if localp
|
||
(dired-make-relative dir default-directory)
|
||
dir)))
|
||
|
||
;; Subdirs start at the beginning of their header lines and end just
|
||
;; before the beginning of the next header line (or end of buffer).
|
||
|
||
(defun dired-subdir-max ()
|
||
(save-excursion
|
||
(if (or (null (cdr dired-subdir-alist)) (not (dired-next-subdir 1 t t)))
|
||
(point-max)
|
||
(point))))
|
||
|
||
;; Deleting files
|
||
|
||
(defcustom dired-recursive-deletes 'top
|
||
"*Decide whether recursive deletes are allowed.
|
||
A value of nil means no recursive deletes.
|
||
`always' means delete recursively without asking. This is DANGEROUS!
|
||
`top' means ask for each directory at top level, but delete its subdirectories
|
||
without asking.
|
||
Anything else means ask for each directory."
|
||
:type '(choice :tag "Delete non-empty directories"
|
||
(const :tag "Yes" always)
|
||
(const :tag "No--only delete empty directories" nil)
|
||
(const :tag "Confirm for each directory" t)
|
||
(const :tag "Confirm for each top directory only" top))
|
||
:group 'dired)
|
||
|
||
;; Match anything but `.' and `..'.
|
||
(defvar dired-re-no-dot "^\\([^.]\\|\\.\\([^.]\\|\\..\\)\\).*")
|
||
|
||
;; Delete file, possibly delete a directory and all its files.
|
||
;; This function is usefull outside of dired. One could change it's name
|
||
;; to e.g. recursive-delete-file and put it somewhere else.
|
||
(defun dired-delete-file (file &optional recursive) "\
|
||
Delete FILE or directory (possibly recursively if optional RECURSIVE is true.)
|
||
RECURSIVE determines what to do with a non-empty directory. If RECURSIVE is:
|
||
nil, do not delete.
|
||
`always', delete recursively without asking.
|
||
`top', ask for each directory at top level.
|
||
Anything else, ask for each sub-directory."
|
||
(let (files)
|
||
;; This test is equivalent to
|
||
;; (and (file-directory-p fn) (not (file-symlink-p fn)))
|
||
;; but more efficient
|
||
(if (not (eq t (car (file-attributes file))))
|
||
(delete-file file)
|
||
(when (and recursive
|
||
(setq files
|
||
(directory-files file t dired-re-no-dot)) ; Not empty.
|
||
(or (eq recursive 'always)
|
||
(yes-or-no-p (format "Recursive delete of %s? "
|
||
(dired-make-relative file)))))
|
||
(if (eq recursive 'top) (setq recursive 'always)) ; Don't ask again.
|
||
(while files ; Recursively delete (possibly asking).
|
||
(dired-delete-file (car files) recursive)
|
||
(setq files (cdr files))))
|
||
(delete-directory file))))
|
||
|
||
(defun dired-do-flagged-delete (&optional nomessage)
|
||
"In Dired, delete the files flagged for deletion.
|
||
If NOMESSAGE is non-nil, we don't display any message
|
||
if there are no flagged files.
|
||
`dired-recursive-deletes' controls whether
|
||
deletion of non-empty directories is allowed."
|
||
(interactive)
|
||
(let* ((dired-marker-char dired-del-marker)
|
||
(regexp (dired-marker-regexp))
|
||
case-fold-search)
|
||
(if (save-excursion (goto-char (point-min))
|
||
(re-search-forward regexp nil t))
|
||
(dired-internal-do-deletions
|
||
;; this can't move point since ARG is nil
|
||
(dired-map-over-marks (cons (dired-get-filename) (point))
|
||
nil)
|
||
nil)
|
||
(or nomessage
|
||
(message "(No deletions requested)")))))
|
||
|
||
(defun dired-do-delete (&optional arg)
|
||
"Delete all marked (or next ARG) files.
|
||
`dired-recursive-deletes' controls whether
|
||
deletion of non-empty directories is allowed."
|
||
;; This is more consistent with the file marking feature than
|
||
;; dired-do-flagged-delete.
|
||
(interactive "P")
|
||
(dired-internal-do-deletions
|
||
;; this may move point if ARG is an integer
|
||
(dired-map-over-marks (cons (dired-get-filename) (point))
|
||
arg)
|
||
arg))
|
||
|
||
(defvar dired-deletion-confirmer 'yes-or-no-p) ; or y-or-n-p?
|
||
|
||
(defun dired-internal-do-deletions (l arg)
|
||
;; L is an alist of files to delete, with their buffer positions.
|
||
;; ARG is the prefix arg.
|
||
;; Filenames are absolute (VMS needs this for logical search paths).
|
||
;; (car L) *must* be the *last* (bottommost) file in the dired buffer.
|
||
;; That way as changes are made in the buffer they do not shift the
|
||
;; lines still to be changed, so the (point) values in L stay valid.
|
||
;; Also, for subdirs in natural order, a subdir's files are deleted
|
||
;; before the subdir itself - the other way around would not work.
|
||
(let ((files (mapcar (function car) l))
|
||
(count (length l))
|
||
(succ 0))
|
||
;; canonicalize file list for pop up
|
||
(setq files (nreverse (mapcar (function dired-make-relative) files)))
|
||
(if (dired-mark-pop-up
|
||
" *Deletions*" 'delete files dired-deletion-confirmer
|
||
(format "Delete %s " (dired-mark-prompt arg files)))
|
||
(save-excursion
|
||
(let (failures);; files better be in reverse order for this loop!
|
||
(while l
|
||
(goto-char (cdr (car l)))
|
||
(let (buffer-read-only)
|
||
(condition-case err
|
||
(let ((fn (car (car l))))
|
||
(dired-delete-file fn dired-recursive-deletes)
|
||
;; if we get here, removing worked
|
||
(setq succ (1+ succ))
|
||
(message "%s of %s deletions" succ count)
|
||
(dired-fun-in-all-buffers
|
||
(file-name-directory fn) (file-name-nondirectory fn)
|
||
(function dired-delete-entry) fn))
|
||
(error;; catch errors from failed deletions
|
||
(dired-log "%s\n" err)
|
||
(setq failures (cons (car (car l)) failures)))))
|
||
(setq l (cdr l)))
|
||
(if (not failures)
|
||
(message "%d deletion%s done" count (dired-plural-s count))
|
||
(dired-log-summary
|
||
(format "%d of %d deletion%s failed"
|
||
(length failures) count
|
||
(dired-plural-s count))
|
||
failures))))
|
||
(message "(No deletions performed)")))
|
||
(dired-move-to-filename))
|
||
|
||
(defun dired-fun-in-all-buffers (directory file fun &rest args)
|
||
;; In all buffers dired'ing DIRECTORY, run FUN with ARGS.
|
||
;; If the buffer has a wildcard pattern, check that it matches FILE.
|
||
;; (FILE does not include a directory component.)
|
||
;; FILE may be nil, in which case ignore it.
|
||
;; Return list of buffers where FUN succeeded (i.e., returned non-nil).
|
||
(let (success-list)
|
||
(dolist (buf (dired-buffers-for-dir (expand-file-name directory)
|
||
file))
|
||
(with-current-buffer buf
|
||
(if (apply fun args)
|
||
(setq success-list (cons (buffer-name buf) success-list)))))
|
||
success-list))
|
||
|
||
;; Delete the entry for FILE from
|
||
(defun dired-delete-entry (file)
|
||
(save-excursion
|
||
(and (dired-goto-file file)
|
||
(let (buffer-read-only)
|
||
(delete-region (progn (beginning-of-line) (point))
|
||
(save-excursion (forward-line 1) (point))))))
|
||
(dired-clean-up-after-deletion file))
|
||
|
||
;; This is a separate function for the sake of dired-x.el.
|
||
(defun dired-clean-up-after-deletion (fn)
|
||
;; Clean up after a deleted file or directory FN.
|
||
(save-excursion (and (cdr dired-subdir-alist)
|
||
(dired-goto-subdir fn)
|
||
(dired-kill-subdir))))
|
||
|
||
;; Confirmation
|
||
|
||
(defun dired-marker-regexp ()
|
||
(concat "^" (regexp-quote (char-to-string dired-marker-char))))
|
||
|
||
(defun dired-plural-s (count)
|
||
(if (= 1 count) "" "s"))
|
||
|
||
(defun dired-mark-prompt (arg files)
|
||
;; Return a string for use in a prompt, either the current file
|
||
;; name, or the marker and a count of marked files.
|
||
(let ((count (length files)))
|
||
(if (= count 1)
|
||
(car files)
|
||
;; more than 1 file:
|
||
(if (integerp arg)
|
||
;; abs(arg) = count
|
||
;; Perhaps this is nicer, but it also takes more screen space:
|
||
;;(format "[%s %d files]" (if (> arg 0) "next" "previous")
|
||
;; count)
|
||
(format "[next %d files]" arg)
|
||
(format "%c [%d files]" dired-marker-char count)))))
|
||
|
||
(defun dired-pop-to-buffer (buf)
|
||
;; Pop up buffer BUF.
|
||
;; If dired-shrink-to-fit is t, make its window fit its contents.
|
||
(if (not dired-shrink-to-fit)
|
||
(pop-to-buffer (get-buffer-create buf))
|
||
;; let window shrink to fit:
|
||
(let ((window (selected-window))
|
||
target-lines w2)
|
||
(cond ;; if split-height-threshold is enabled, use the largest window
|
||
((and (> (window-height (setq w2 (get-largest-window)))
|
||
split-height-threshold)
|
||
(window-full-width-p w2))
|
||
(setq window w2))
|
||
;; if the least-recently-used window is big enough, use it
|
||
((and (> (window-height (setq w2 (get-lru-window)))
|
||
(* 2 window-min-height))
|
||
(window-full-width-p w2))
|
||
(setq window w2)))
|
||
(save-excursion
|
||
(set-buffer buf)
|
||
(goto-char (point-max))
|
||
(skip-chars-backward "\n\r\t ")
|
||
(setq target-lines (count-lines (point-min) (point)))
|
||
;; Don't forget to count the last line.
|
||
(if (not (bolp))
|
||
(setq target-lines (1+ target-lines))))
|
||
(if (<= (window-height window) (* 2 window-min-height))
|
||
;; At this point, every window on the frame is too small to split.
|
||
(setq w2 (display-buffer buf))
|
||
(setq w2 (split-window window
|
||
(max window-min-height
|
||
(- (window-height window)
|
||
(1+ (max window-min-height target-lines)))))))
|
||
(set-window-buffer w2 buf)
|
||
(if (< (1- (window-height w2)) target-lines)
|
||
(progn
|
||
(select-window w2)
|
||
(enlarge-window (- target-lines (1- (window-height w2))))))
|
||
(set-window-start w2 1)
|
||
)))
|
||
|
||
(defcustom dired-no-confirm nil
|
||
"A list of symbols for commands Dired should not confirm.
|
||
Command symbols are `byte-compile', `chgrp', `chmod', `chown', `compress',
|
||
`copy', `delete', `hardlink', `load', `move', `print', `shell', `symlink',
|
||
`touch' and `uncompress'."
|
||
:group 'dired
|
||
:type '(set (const byte-compile) (const chgrp)
|
||
(const chmod) (const chown) (const compress)
|
||
(const copy) (const delete) (const hardlink)
|
||
(const load) (const move) (const print)
|
||
(const shell) (const symlink) (const touch)
|
||
(const uncompress)))
|
||
|
||
(defun dired-mark-pop-up (bufname op-symbol files function &rest args)
|
||
"Return FUNCTION's result on ARGS after showing which files are marked.
|
||
Displays the file names in a buffer named BUFNAME;
|
||
nil gives \" *Marked Files*\".
|
||
This uses function `dired-pop-to-buffer' to do that.
|
||
|
||
FUNCTION should not manipulate files, just read input
|
||
(an argument or confirmation).
|
||
The window is not shown if there is just one file or
|
||
OP-SYMBOL is a member of the list in `dired-no-confirm'.
|
||
FILES is the list of marked files. It can also be (t FILENAME)
|
||
in the case of one marked file, to distinguish that from using
|
||
just the current file."
|
||
(or bufname (setq bufname " *Marked Files*"))
|
||
(if (or (eq dired-no-confirm t)
|
||
(memq op-symbol dired-no-confirm)
|
||
;; If FILES defaulted to the current line's file.
|
||
(= (length files) 1))
|
||
(apply function args)
|
||
(with-current-buffer (get-buffer-create bufname)
|
||
(erase-buffer)
|
||
;; Handle (t FILE) just like (FILE), here.
|
||
;; That value is used (only in some cases), to mean
|
||
;; just one file that was marked, rather than the current line file.
|
||
(dired-format-columns-of-files (if (eq (car files) t) (cdr files) files))
|
||
(remove-text-properties (point-min) (point-max)
|
||
'(mouse-face nil help-echo nil)))
|
||
(save-window-excursion
|
||
(dired-pop-to-buffer bufname)
|
||
(apply function args))))
|
||
|
||
(defun dired-format-columns-of-files (files)
|
||
;; Files should be in forward order for this loop.
|
||
;; i.e., (car files) = first file in buffer.
|
||
;; Returns the number of lines used.
|
||
(let* ((maxlen (+ 2 (apply 'max (mapcar 'length files))))
|
||
(width (- (window-width (selected-window)) 2))
|
||
(columns (max 1 (/ width maxlen)))
|
||
(nfiles (length files))
|
||
(rows (+ (/ nfiles columns)
|
||
(if (zerop (% nfiles columns)) 0 1)))
|
||
(i 0)
|
||
(j 0))
|
||
(setq files (nconc (copy-sequence files) ; fill up with empty fns
|
||
(make-list (- (* columns rows) nfiles) "")))
|
||
(setcdr (nthcdr (1- (length files)) files) files) ; make circular
|
||
(while (< j rows)
|
||
(while (< i columns)
|
||
(indent-to (* i maxlen))
|
||
(insert (car files))
|
||
(setq files (nthcdr rows files)
|
||
i (1+ i)))
|
||
(insert "\n")
|
||
(setq i 0
|
||
j (1+ j)
|
||
files (cdr files)))
|
||
rows))
|
||
|
||
;; Commands to mark or flag file(s) at or near current line.
|
||
|
||
(defun dired-repeat-over-lines (arg function)
|
||
;; This version skips non-file lines.
|
||
(let ((pos (make-marker)))
|
||
(beginning-of-line)
|
||
(while (and (> arg 0) (not (eobp)))
|
||
(setq arg (1- arg))
|
||
(beginning-of-line)
|
||
(while (and (not (eobp)) (dired-between-files)) (forward-line 1))
|
||
(save-excursion
|
||
(forward-line 1)
|
||
(move-marker pos (1+ (point))))
|
||
(save-excursion (funcall function))
|
||
;; Advance to the next line--actually, to the line that *was* next.
|
||
;; (If FUNCTION inserted some new lines in between, skip them.)
|
||
(goto-char pos))
|
||
(while (and (< arg 0) (not (bobp)))
|
||
(setq arg (1+ arg))
|
||
(forward-line -1)
|
||
(while (and (not (bobp)) (dired-between-files)) (forward-line -1))
|
||
(beginning-of-line)
|
||
(save-excursion (funcall function)))
|
||
(move-marker pos nil)
|
||
(dired-move-to-filename)))
|
||
|
||
(defun dired-between-files ()
|
||
;; This used to be a regexp match of the `total ...' line output by
|
||
;; ls, which is slightly faster, but that is not very robust; notably,
|
||
;; it fails for non-english locales.
|
||
(save-excursion (not (dired-move-to-filename))))
|
||
|
||
(defun dired-next-marked-file (arg &optional wrap opoint)
|
||
"Move to the next marked file, wrapping around the end of the buffer."
|
||
(interactive "p\np")
|
||
(or opoint (setq opoint (point)));; return to where interactively started
|
||
(if (if (> arg 0)
|
||
(re-search-forward dired-re-mark nil t arg)
|
||
(beginning-of-line)
|
||
(re-search-backward dired-re-mark nil t (- arg)))
|
||
(dired-move-to-filename)
|
||
(if (null wrap)
|
||
(progn
|
||
(goto-char opoint)
|
||
(error "No next marked file"))
|
||
(message "(Wraparound for next marked file)")
|
||
(goto-char (if (> arg 0) (point-min) (point-max)))
|
||
(dired-next-marked-file arg nil opoint))))
|
||
|
||
(defun dired-prev-marked-file (arg &optional wrap)
|
||
"Move to the previous marked file, wrapping around the end of the buffer."
|
||
(interactive "p\np")
|
||
(dired-next-marked-file (- arg) wrap))
|
||
|
||
(defun dired-file-marker (file)
|
||
;; Return FILE's marker, or nil if unmarked.
|
||
(save-excursion
|
||
(and (dired-goto-file file)
|
||
(progn
|
||
(beginning-of-line)
|
||
(if (not (equal ?\040 (following-char)))
|
||
(following-char))))))
|
||
|
||
(defun dired-mark-files-in-region (start end)
|
||
(let (buffer-read-only)
|
||
(if (> start end)
|
||
(error "start > end"))
|
||
(goto-char start) ; assumed at beginning of line
|
||
(while (< (point) end)
|
||
;; Skip subdir line and following garbage like the `total' line:
|
||
(while (and (< (point) end) (dired-between-files))
|
||
(forward-line 1))
|
||
(if (and (not (looking-at dired-re-dot))
|
||
(dired-get-filename nil t))
|
||
(progn
|
||
(delete-char 1)
|
||
(insert dired-marker-char)))
|
||
(forward-line 1))))
|
||
|
||
(defun dired-mark (arg)
|
||
"Mark the current (or next ARG) files.
|
||
If on a subdir headerline, mark all its files except `.' and `..'.
|
||
|
||
Use \\[dired-unmark-all-files] to remove all marks
|
||
and \\[dired-unmark] on a subdir to remove the marks in
|
||
this subdir."
|
||
(interactive "P")
|
||
(if (dired-get-subdir)
|
||
(save-excursion (dired-mark-subdir-files))
|
||
(let (buffer-read-only)
|
||
(dired-repeat-over-lines
|
||
(prefix-numeric-value arg)
|
||
(function (lambda () (delete-char 1) (insert dired-marker-char)))))))
|
||
|
||
(defun dired-unmark (arg)
|
||
"Unmark the current (or next ARG) files.
|
||
If looking at a subdir, unmark all its files except `.' and `..'."
|
||
(interactive "P")
|
||
(let ((dired-marker-char ?\040))
|
||
(dired-mark arg)))
|
||
|
||
(defun dired-flag-file-deletion (arg)
|
||
"In Dired, flag the current line's file for deletion.
|
||
With prefix arg, repeat over several lines.
|
||
|
||
If on a subdir headerline, mark all its files except `.' and `..'."
|
||
(interactive "P")
|
||
(let ((dired-marker-char dired-del-marker))
|
||
(dired-mark arg)))
|
||
|
||
(defun dired-unmark-backward (arg)
|
||
"In Dired, move up lines and remove deletion flag there.
|
||
Optional prefix ARG says how many lines to unflag; default is one line."
|
||
(interactive "p")
|
||
(dired-unmark (- arg)))
|
||
|
||
(defun dired-toggle-marks ()
|
||
"Toggle marks: marked files become unmarked, and vice versa.
|
||
Files marked with other flags (such as `D') are not affected.
|
||
`.' and `..' are never toggled.
|
||
As always, hidden subdirs are not affected."
|
||
(interactive)
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(let (buffer-read-only)
|
||
(while (not (eobp))
|
||
(or (dired-between-files)
|
||
(looking-at dired-re-dot)
|
||
;; use subst instead of insdel because it does not move
|
||
;; the gap and thus should be faster and because
|
||
;; other characters are left alone automatically
|
||
(apply 'subst-char-in-region
|
||
(point) (1+ (point))
|
||
(if (eq ?\040 (following-char)) ; SPC
|
||
(list ?\040 dired-marker-char)
|
||
(list dired-marker-char ?\040))))
|
||
(forward-line 1)))))
|
||
|
||
;;; Commands to mark or flag files based on their characteristics or names.
|
||
|
||
(defvar dired-regexp-history nil
|
||
"History list of regular expressions used in Dired commands.")
|
||
|
||
(defun dired-read-regexp (prompt)
|
||
(read-from-minibuffer prompt nil nil nil 'dired-regexp-history))
|
||
|
||
(defun dired-mark-files-regexp (regexp &optional marker-char)
|
||
"Mark all files matching REGEXP for use in later commands.
|
||
A prefix argument means to unmark them instead.
|
||
`.' and `..' are never marked.
|
||
|
||
REGEXP is an Emacs regexp, not a shell wildcard. Thus, use `\\.o$' for
|
||
object files--just `.o' will mark more than you might think."
|
||
(interactive
|
||
(list (dired-read-regexp (concat (if current-prefix-arg "Unmark" "Mark")
|
||
" files (regexp): "))
|
||
(if current-prefix-arg ?\040)))
|
||
(let ((dired-marker-char (or marker-char dired-marker-char)))
|
||
(dired-mark-if
|
||
(and (not (looking-at dired-re-dot))
|
||
(not (eolp)) ; empty line
|
||
(let ((fn (dired-get-filename nil t)))
|
||
(and fn (string-match regexp (file-name-nondirectory fn)))))
|
||
"matching file")))
|
||
|
||
(defun dired-mark-files-containing-regexp (regexp &optional marker-char)
|
||
"Mark all files with contents containing REGEXP for use in later commands.
|
||
A prefix argument means to unmark them instead.
|
||
`.' and `..' are never marked."
|
||
(interactive
|
||
(list (dired-read-regexp (concat (if current-prefix-arg "Unmark" "Mark")
|
||
" files containing (regexp): "))
|
||
(if current-prefix-arg ?\040)))
|
||
(let ((dired-marker-char (or marker-char dired-marker-char)))
|
||
(dired-mark-if
|
||
(and (not (looking-at dired-re-dot))
|
||
(not (eolp)) ; empty line
|
||
(let ((fn (dired-get-filename nil t)))
|
||
(when (and fn (file-readable-p fn)
|
||
(not (file-directory-p fn)))
|
||
(let ((prebuf (get-file-buffer fn)))
|
||
(message "Checking %s" fn)
|
||
;; For now we do it inside emacs
|
||
;; Grep might be better if there are a lot of files
|
||
(if prebuf
|
||
(with-current-buffer prebuf
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(re-search-forward regexp nil t)))
|
||
(with-temp-buffer
|
||
(insert-file-contents fn)
|
||
(goto-char (point-min))
|
||
(re-search-forward regexp nil t))))
|
||
)))
|
||
"matching file")))
|
||
|
||
(defun dired-flag-files-regexp (regexp)
|
||
"In Dired, flag all files containing the specified REGEXP for deletion.
|
||
The match is against the non-directory part of the filename. Use `^'
|
||
and `$' to anchor matches. Exclude subdirs by hiding them.
|
||
`.' and `..' are never flagged."
|
||
(interactive (list (dired-read-regexp "Flag for deletion (regexp): ")))
|
||
(dired-mark-files-regexp regexp dired-del-marker))
|
||
|
||
(defun dired-mark-symlinks (unflag-p)
|
||
"Mark all symbolic links.
|
||
With prefix argument, unflag all those files."
|
||
(interactive "P")
|
||
(let ((dired-marker-char (if unflag-p ?\040 dired-marker-char)))
|
||
(dired-mark-if (looking-at dired-re-sym) "symbolic link")))
|
||
|
||
(defun dired-mark-directories (unflag-p)
|
||
"Mark all directory file lines except `.' and `..'.
|
||
With prefix argument, unflag all those files."
|
||
(interactive "P")
|
||
(let ((dired-marker-char (if unflag-p ?\040 dired-marker-char)))
|
||
(dired-mark-if (and (looking-at dired-re-dir)
|
||
(not (looking-at dired-re-dot)))
|
||
"directory file")))
|
||
|
||
(defun dired-mark-executables (unflag-p)
|
||
"Mark all executable files.
|
||
With prefix argument, unflag all those files."
|
||
(interactive "P")
|
||
(let ((dired-marker-char (if unflag-p ?\040 dired-marker-char)))
|
||
(dired-mark-if (looking-at dired-re-exe) "executable file")))
|
||
|
||
;; dired-x.el has a dired-mark-sexp interactive command: mark
|
||
;; files for which PREDICATE returns non-nil.
|
||
|
||
(defun dired-flag-auto-save-files (&optional unflag-p)
|
||
"Flag for deletion files whose names suggest they are auto save files.
|
||
A prefix argument says to unflag those files instead."
|
||
(interactive "P")
|
||
(let ((dired-marker-char (if unflag-p ?\040 dired-del-marker)))
|
||
(dired-mark-if
|
||
;; It is less than general to check for # here,
|
||
;; but it's the only way this runs fast enough.
|
||
(and (save-excursion (end-of-line)
|
||
(or
|
||
(eq (preceding-char) ?#)
|
||
;; Handle executables in case of -F option.
|
||
;; We need not worry about the other kinds
|
||
;; of markings that -F makes, since they won't
|
||
;; appear on real auto-save files.
|
||
(if (eq (preceding-char) ?*)
|
||
(progn
|
||
(forward-char -1)
|
||
(eq (preceding-char) ?#)))))
|
||
(not (looking-at dired-re-dir))
|
||
(let ((fn (dired-get-filename t t)))
|
||
(if fn (auto-save-file-name-p
|
||
(file-name-nondirectory fn)))))
|
||
"auto save file")))
|
||
|
||
(defcustom dired-garbage-files-regexp
|
||
;; `log' here is dubious, since it's typically used for useful log
|
||
;; files, not just TeX stuff. -- fx
|
||
(concat (regexp-opt
|
||
'(".log" ".toc" ".dvi" ".bak" ".orig" ".rej" ".aux"))
|
||
"\\'")
|
||
"Regular expression to match \"garbage\" files for `dired-flag-garbage-files'."
|
||
:type 'regexp
|
||
:group 'dired)
|
||
|
||
(defun dired-flag-garbage-files ()
|
||
"Flag for deletion all files that match `dired-garbage-files-regexp'."
|
||
(interactive)
|
||
(dired-flag-files-regexp dired-garbage-files-regexp))
|
||
|
||
(defun dired-flag-backup-files (&optional unflag-p)
|
||
"Flag all backup files (names ending with `~') for deletion.
|
||
With prefix argument, unflag these files."
|
||
(interactive "P")
|
||
(let ((dired-marker-char (if unflag-p ?\s dired-del-marker)))
|
||
(dired-mark-if
|
||
;; Don't call backup-file-name-p unless the last character looks like
|
||
;; it might be the end of a backup file name. This isn't very general,
|
||
;; but it's the only way this runs fast enough.
|
||
(and (save-excursion (end-of-line)
|
||
;; Handle executables in case of -F option.
|
||
;; We need not worry about the other kinds
|
||
;; of markings that -F makes, since they won't
|
||
;; appear on real backup files.
|
||
(if (eq (preceding-char) ?*)
|
||
(forward-char -1))
|
||
(eq (preceding-char) ?~))
|
||
(not (looking-at dired-re-dir))
|
||
(let ((fn (dired-get-filename t t)))
|
||
(if fn (backup-file-name-p fn))))
|
||
"backup file")))
|
||
|
||
(defun dired-change-marks (&optional old new)
|
||
"Change all OLD marks to NEW marks.
|
||
OLD and NEW are both characters used to mark files."
|
||
(interactive
|
||
(let* ((cursor-in-echo-area t)
|
||
(old (progn (message "Change (old mark): ") (read-char)))
|
||
(new (progn (message "Change %c marks to (new mark): " old)
|
||
(read-char))))
|
||
(list old new)))
|
||
(if (or (eq old ?\r) (eq new ?\r))
|
||
(ding)
|
||
(let ((string (format "\n%c" old))
|
||
(buffer-read-only))
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(while (search-forward string nil t)
|
||
(if (if (= old ?\s)
|
||
(save-match-data
|
||
(dired-get-filename 'no-dir t))
|
||
t)
|
||
(subst-char-in-region (match-beginning 0)
|
||
(match-end 0) old new)))))))
|
||
|
||
(defun dired-unmark-all-marks ()
|
||
"Remove all marks from all files in the dired buffer."
|
||
(interactive)
|
||
(dired-unmark-all-files ?\r))
|
||
|
||
(defun dired-unmark-all-files (mark &optional arg)
|
||
"Remove a specific mark (or any mark) from every file.
|
||
After this command, type the mark character to remove,
|
||
or type RET to remove all marks.
|
||
With prefix arg, query for each marked file.
|
||
Type \\[help-command] at that time for help."
|
||
(interactive "cRemove marks (RET means all): \nP")
|
||
(save-excursion
|
||
(let* ((count 0)
|
||
buffer-read-only case-fold-search query
|
||
(string (format "\n%c" mark))
|
||
(help-form "\
|
||
Type SPC or `y' to unmark one file, DEL or `n' to skip to next,
|
||
`!' to unmark all remaining files with no more questions."))
|
||
(goto-char (point-min))
|
||
(while (if (eq mark ?\r)
|
||
(re-search-forward dired-re-mark nil t)
|
||
(search-forward string nil t))
|
||
(if (or (not arg)
|
||
(let ((file (dired-get-filename t t)))
|
||
(and file
|
||
(dired-query 'query "Unmark file `%s'? "
|
||
file))))
|
||
(progn (subst-char-in-region (1- (point)) (point)
|
||
(preceding-char) ?\s)
|
||
(setq count (1+ count)))))
|
||
(message (if (= count 1) "1 mark removed"
|
||
"%d marks removed")
|
||
count))))
|
||
|
||
;; Logging failures operating on files, and showing the results.
|
||
|
||
(defvar dired-log-buffer "*Dired log*")
|
||
|
||
(defun dired-why ()
|
||
"Pop up a buffer with error log output from Dired.
|
||
A group of errors from a single command ends with a formfeed.
|
||
Thus, use \\[backward-page] to find the beginning of a group of errors."
|
||
(interactive)
|
||
(if (get-buffer dired-log-buffer)
|
||
(let ((owindow (selected-window))
|
||
(window (display-buffer (get-buffer dired-log-buffer))))
|
||
(unwind-protect
|
||
(progn
|
||
(select-window window)
|
||
(goto-char (point-max))
|
||
(forward-line -1)
|
||
(backward-page 1)
|
||
(recenter 0))
|
||
(select-window owindow)))))
|
||
|
||
(defun dired-log (log &rest args)
|
||
;; Log a message or the contents of a buffer.
|
||
;; If LOG is a string and there are more args, it is formatted with
|
||
;; those ARGS. Usually the LOG string ends with a \n.
|
||
;; End each bunch of errors with (dired-log t):
|
||
;; this inserts the current time and buffer at the start of the page,
|
||
;; and \f (formfeed) at the end.
|
||
(let ((obuf (current-buffer)))
|
||
(with-current-buffer (get-buffer-create dired-log-buffer)
|
||
(goto-char (point-max))
|
||
(let ((inhibit-read-only t))
|
||
(cond ((stringp log)
|
||
(insert (if args
|
||
(apply (function format) log args)
|
||
log)))
|
||
((bufferp log)
|
||
(insert-buffer-substring log))
|
||
((eq t log)
|
||
(backward-page 1)
|
||
(unless (bolp)
|
||
(insert "\n"))
|
||
(insert (current-time-string)
|
||
"\tBuffer `" (buffer-name obuf) "'\n")
|
||
(goto-char (point-max))
|
||
(insert "\f\n")))))))
|
||
|
||
(defun dired-log-summary (string failures)
|
||
"State a summary of a command's failures, in echo area and log buffer.
|
||
STRING is an overall summary of the failures.
|
||
FAILURES is a list of file names that we failed to operate on,
|
||
or nil if file names are not applicable."
|
||
(if (= (length failures) 1)
|
||
(message "%s"
|
||
(with-current-buffer dired-log-buffer
|
||
(goto-char (point-max))
|
||
(backward-page 1)
|
||
(if (eolp) (forward-line 1))
|
||
(buffer-substring (point) (point-max))))
|
||
(message (if failures "%s--type ? for details (%s)"
|
||
"%s--type ? for details")
|
||
string failures))
|
||
;; Log a summary describing a bunch of errors.
|
||
(dired-log (concat "\n" string "\n"))
|
||
(dired-log t))
|
||
|
||
;;; Sorting
|
||
|
||
;; Most ls can only sort by name or by date (with -t), nothing else.
|
||
;; GNU ls sorts on size with -S, on extension with -X, and unsorted with -U.
|
||
;; So anything that does not contain these is sort "by name".
|
||
|
||
(defvar dired-ls-sorting-switches "SXU"
|
||
"String of `ls' switches \(single letters\) except \"t\" that influence sorting.
|
||
|
||
This indicates to Dired which option switches to watch out for because they
|
||
will change the sorting order behavior of `ls'.
|
||
|
||
To change the default sorting order \(e.g. add a `-v' option\), see the
|
||
variable `dired-listing-switches'. To temporarily override the listing
|
||
format, use `\\[universal-argument] \\[dired]'.")
|
||
|
||
(defvar dired-sort-by-date-regexp
|
||
(concat "^-[^" dired-ls-sorting-switches
|
||
"]*t[^" dired-ls-sorting-switches "]*$")
|
||
"Regexp recognized by Dired to set `by date' mode.")
|
||
|
||
(defvar dired-sort-by-name-regexp
|
||
(concat "^-[^t" dired-ls-sorting-switches "]+$")
|
||
"Regexp recognized by Dired to set `by name' mode.")
|
||
|
||
(defvar dired-sort-inhibit nil
|
||
"Non-nil means the Dired sort command is disabled.
|
||
The idea is to set this buffer-locally in special dired buffers.")
|
||
|
||
(defun dired-sort-set-modeline ()
|
||
;; Set modeline display according to dired-actual-switches.
|
||
;; Modeline display of "by name" or "by date" guarantees the user a
|
||
;; match with the corresponding regexps. Non-matching switches are
|
||
;; shown literally.
|
||
(when (eq major-mode 'dired-mode)
|
||
(setq mode-name
|
||
(let (case-fold-search)
|
||
(cond ((string-match
|
||
dired-sort-by-name-regexp dired-actual-switches)
|
||
"Dired by name")
|
||
((string-match
|
||
dired-sort-by-date-regexp dired-actual-switches)
|
||
"Dired by date")
|
||
(t
|
||
(concat "Dired " dired-actual-switches)))))
|
||
(force-mode-line-update)))
|
||
|
||
(defun dired-sort-toggle-or-edit (&optional arg)
|
||
"Toggle between sort by date/name and refresh the dired buffer.
|
||
With a prefix argument you can edit the current listing switches instead."
|
||
(interactive "P")
|
||
(when dired-sort-inhibit
|
||
(error "Cannot sort this dired buffer"))
|
||
(if arg
|
||
(dired-sort-other
|
||
(read-string "ls switches (must contain -l): " dired-actual-switches))
|
||
(dired-sort-toggle)))
|
||
|
||
(defun dired-sort-toggle ()
|
||
;; Toggle between sort by date/name. Reverts the buffer.
|
||
(setq dired-actual-switches
|
||
(let (case-fold-search)
|
||
(if (string-match " " dired-actual-switches)
|
||
;; New toggle scheme: add/remove a trailing " -t"
|
||
(if (string-match " -t\\'" dired-actual-switches)
|
||
(substring dired-actual-switches 0 (match-beginning 0))
|
||
(concat dired-actual-switches " -t"))
|
||
;; old toggle scheme: look for some 't' switch and add/remove it
|
||
(concat
|
||
"-l"
|
||
(dired-replace-in-string (concat "[-lt"
|
||
dired-ls-sorting-switches "]")
|
||
""
|
||
dired-actual-switches)
|
||
(if (string-match (concat "[t" dired-ls-sorting-switches "]")
|
||
dired-actual-switches)
|
||
""
|
||
"t")))))
|
||
(dired-sort-set-modeline)
|
||
(revert-buffer))
|
||
|
||
;; Some user code loads dired especially for this.
|
||
;; Don't do that--use replace-regexp-in-string instead.
|
||
(defun dired-replace-in-string (regexp newtext string)
|
||
;; Replace REGEXP with NEWTEXT everywhere in STRING and return result.
|
||
;; NEWTEXT is taken literally---no \\DIGIT escapes will be recognized.
|
||
(let ((result "") (start 0) mb me)
|
||
(while (string-match regexp string start)
|
||
(setq mb (match-beginning 0)
|
||
me (match-end 0)
|
||
result (concat result (substring string start mb) newtext)
|
||
start me))
|
||
(concat result (substring string start))))
|
||
|
||
(defun dired-sort-other (switches &optional no-revert)
|
||
"Specify new `ls' SWITCHES for current dired buffer.
|
||
Values matching `dired-sort-by-date-regexp' or `dired-sort-by-name-regexp'
|
||
set the minor mode accordingly, others appear literally in the mode line.
|
||
With optional second arg NO-REVERT, don't refresh the listing afterwards."
|
||
(dired-sort-R-check switches)
|
||
(setq dired-actual-switches switches)
|
||
(dired-sort-set-modeline)
|
||
(or no-revert (revert-buffer)))
|
||
|
||
(defvar dired-subdir-alist-pre-R nil
|
||
"Value of `dired-subdir-alist' before -R switch added.")
|
||
(make-variable-buffer-local 'dired-subdir-alist-pre-R)
|
||
|
||
(defun dired-sort-R-check (switches)
|
||
"Additional processing of -R in ls option string SWITCHES.
|
||
Saves `dired-subdir-alist' when R is set and restores saved value
|
||
minus any directories explicitly deleted when R is cleared.
|
||
To be called first in body of `dired-sort-other', etc."
|
||
(cond
|
||
((and (string-match "R" switches)
|
||
(not (string-match "R" dired-actual-switches)))
|
||
;; Adding -R to ls switches -- save `dired-subdir-alist':
|
||
(setq dired-subdir-alist-pre-R dired-subdir-alist))
|
||
((and (string-match "R" dired-actual-switches)
|
||
(not (string-match "R" switches)))
|
||
;; Deleting -R from ls switches -- revert to pre-R subdirs
|
||
;; that are still present:
|
||
(setq dired-subdir-alist
|
||
(if dired-subdir-alist-pre-R
|
||
(let (subdirs)
|
||
(while dired-subdir-alist-pre-R
|
||
(if (assoc (caar dired-subdir-alist-pre-R)
|
||
dired-subdir-alist)
|
||
;; subdir still present...
|
||
(setq subdirs
|
||
(cons (car dired-subdir-alist-pre-R)
|
||
subdirs)))
|
||
(setq dired-subdir-alist-pre-R
|
||
(cdr dired-subdir-alist-pre-R)))
|
||
(reverse subdirs))
|
||
;; No pre-R subdir alist, so revert to main directory
|
||
;; listing:
|
||
(list (car (reverse dired-subdir-alist))))))))
|
||
|
||
|
||
;;;; Drag and drop support
|
||
|
||
(defcustom dired-recursive-copies 'top
|
||
"*Decide whether recursive copies are allowed.
|
||
A value of nil means no recursive copies.
|
||
`always' means copy recursively without asking.
|
||
`top' means ask for each directory at top level.
|
||
Anything else means ask for each directory."
|
||
:type '(choice :tag "Copy directories"
|
||
(const :tag "No recursive copies" nil)
|
||
(const :tag "Ask for each directory" t)
|
||
(const :tag "Ask for each top directory only" top)
|
||
(const :tag "Copy directories without asking" always))
|
||
:group 'dired)
|
||
|
||
(defun dired-dnd-popup-notice ()
|
||
(message-box
|
||
"Recursive copies not enabled.\nSee variable dired-recursive-copies."))
|
||
|
||
|
||
(defun dired-dnd-do-ask-action (uri)
|
||
;; No need to get actions and descriptions from the source,
|
||
;; we only have three actions anyway.
|
||
(let ((action (x-popup-menu
|
||
t
|
||
(list "What action?"
|
||
(cons ""
|
||
'(("Copy here" . copy)
|
||
("Move here" . move)
|
||
("Link here" . link)
|
||
"--"
|
||
("Cancel" . nil)))))))
|
||
(if action
|
||
(dired-dnd-handle-local-file uri action)
|
||
nil)))
|
||
|
||
(defun dired-dnd-handle-local-file (uri action)
|
||
"Copy, move or link a file to the dired directory.
|
||
URI is the file to handle, ACTION is one of copy, move, link or ask.
|
||
Ask means pop up a menu for the user to select one of copy, move or link."
|
||
(require 'dired-aux)
|
||
(let* ((from (dnd-get-local-file-name uri t))
|
||
(to (if from (concat (dired-current-directory)
|
||
(file-name-nondirectory from))
|
||
nil)))
|
||
(if from
|
||
(cond ((or (eq action 'copy)
|
||
(eq action 'private)) ; Treat private as copy.
|
||
|
||
;; If copying a directory and dired-recursive-copies is nil,
|
||
;; dired-copy-file silently fails. Pop up a notice.
|
||
(if (and (file-directory-p from)
|
||
(not dired-recursive-copies))
|
||
(dired-dnd-popup-notice)
|
||
(progn
|
||
(dired-copy-file from to 1)
|
||
(dired-relist-entry to)
|
||
action)))
|
||
|
||
((eq action 'move)
|
||
(dired-rename-file from to 1)
|
||
(dired-relist-entry to)
|
||
action)
|
||
|
||
((eq action 'link)
|
||
(make-symbolic-link from to 1)
|
||
(dired-relist-entry to)
|
||
action)
|
||
|
||
((eq action 'ask)
|
||
(dired-dnd-do-ask-action uri))
|
||
|
||
(t nil)))))
|
||
|
||
(defun dired-dnd-handle-file (uri action)
|
||
"Copy, move or link a file to the dired directory if it is a local file.
|
||
URI is the file to handle. If the hostname in the URI isn't local, do nothing.
|
||
ACTION is one of copy, move, link or ask.
|
||
Ask means pop up a menu for the user to select one of copy, move or link."
|
||
(let ((local-file (dnd-get-local-file-uri uri)))
|
||
(if local-file (dired-dnd-handle-local-file local-file action)
|
||
nil)))
|
||
|
||
|
||
;;;; Desktop support
|
||
|
||
(eval-when-compile (require 'desktop))
|
||
|
||
(defun dired-desktop-buffer-misc-data (desktop-dirname)
|
||
"Auxiliary information to be saved in desktop file."
|
||
(cons
|
||
;; Value of `dired-directory'.
|
||
(if (consp dired-directory)
|
||
;; Directory name followed by list of files.
|
||
(cons (desktop-file-name (car dired-directory) desktop-dirname)
|
||
(cdr dired-directory))
|
||
;; Directory name, optionally with shell wildcard.
|
||
(desktop-file-name dired-directory desktop-dirname))
|
||
;; Subdirectories in `dired-subdir-alist'.
|
||
(cdr
|
||
(nreverse
|
||
(mapcar
|
||
(function (lambda (f) (desktop-file-name (car f) desktop-dirname)))
|
||
dired-subdir-alist)))))
|
||
|
||
(defun dired-restore-desktop-buffer (desktop-buffer-file-name
|
||
desktop-buffer-name
|
||
desktop-buffer-misc)
|
||
"Restore a dired buffer specified in a desktop file."
|
||
;; First element of `desktop-buffer-misc' is the value of `dired-directory'.
|
||
;; This value is a directory name, optionally with shell wildcard or
|
||
;; a directory name followed by list of files.
|
||
(let* ((dired-dir (car desktop-buffer-misc))
|
||
(dir (if (consp dired-dir) (car dired-dir) dired-dir)))
|
||
(if (file-directory-p (file-name-directory dir))
|
||
(progn
|
||
(dired dired-dir)
|
||
;; The following elements of `desktop-buffer-misc' are the keys
|
||
;; from `dired-subdir-alist'.
|
||
(mapcar 'dired-maybe-insert-subdir (cdr desktop-buffer-misc))
|
||
(current-buffer))
|
||
(message "Desktop: Directory %s no longer exists." dir)
|
||
(when desktop-missing-file-warning (sit-for 1))
|
||
nil)))
|
||
|
||
(add-to-list 'desktop-buffer-mode-handlers
|
||
'(dired-mode . dired-restore-desktop-buffer))
|
||
|
||
|
||
(if (eq system-type 'vax-vms)
|
||
(load "dired-vms"))
|
||
|
||
(provide 'dired)
|
||
|
||
(run-hooks 'dired-load-hook) ; for your customizations
|
||
|
||
;; arch-tag: e1af7a8f-691c-41a0-aac1-ddd4d3c87517
|
||
;;; dired.el ends here
|