+++ /dev/null
-;;; find-file.el --- find a file corresponding to this one given a pattern -*- lexical-binding: t; -*-
-
-;; Author: Henry Guillaume <henri@tibco.com, henry@c032.aone.net.au>
-;; Maintainer: emacs-devel@gnu.org
-;; Keywords: c, matching, tools
-
-;; Copyright (C) 1994-1995, 2001-2025 Free Software Foundation, Inc.
-
-;; 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 of the License, 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. If not, see <https://www.gnu.org/licenses/>.
-
-;;; Commentary:
-
-;; PURPOSE:
-;; This package features a function called ff-find-other-file, which performs
-;; the following function:
-;;
-;; When in a .c file, find the first corresponding .h file in a set
-;; of directories and display it, and vice-versa from the .h file.
-;;
-;; Many people maintain their include file in a directory separate to their
-;; src directory, and very often you may be editing a file and have a need to
-;; visit the "other file". This package searches through a set of directories
-;; to find that file.
-;;
-;; THE "OTHER FILE", or "corresponding file", generally has the same basename,
-;; and just has a different extension as described by the ff-other-file-alist
-;; variable:
-;;
-;; '(("\\.cc\\'" (".hh" ".h"))
-;; ("\\.hh\\'" (".cc" ".C" ".CC" ".cxx" ".cpp")))
-;;
-;; If the current file has a .cc extension, ff-find-other-file will attempt
-;; to look for a .hh file, and then a .h file in some directory as described
-;; below. The mechanism here is to replace the matched part of the original
-;; filename with each of the corresponding extensions in turn.
-;;
-;; Alternatively, there are situations where the filename of the other file
-;; cannot be determined easily with regexps. For example, a .c file may
-;; have two corresponding .h files, for its public and private parts, or
-;; the filename for the .c file contains part of the pathname of the .h
-;; file, as between src/fooZap.cc and include/FOO/zap.hh. In that case, the
-;; format above can be changed to include a function to be called when the
-;; current file matches the regexp:
-;;
-;; '(("\\.cc\\'" cc--function)
-;; ("\\.hh\\'" hh-function))
-;;
-;; These functions must return a list consisting of the possible names of the
-;; corresponding file, with or without path. There is no real need for more
-;; than one function, and one could imagine the following value for cc-other-
-;; file-alist:
-;;
-;; (setq cc-other-file-alist
-;; '(("\\.cc\\'" ff-cc-hh-converter)
-;; ("\\.hh\\'" ff-cc-hh-converter)
-;; ("\\.c\\'" (".h"))
-;; ("\\.h\\'" (".c" ".cc" ".C" ".CC" ".cxx" ".cpp"))))
-;;
-;; ff-cc-hh-converter is included at the end of this file as a reference.
-;;
-;; SEARCHING is carried out in a set of directories specified by the
-;; ff-search-directories variable:
-;;
-;; ("." "../../src" "../include/*" "/usr/local/*/src/*" "$PROJECT/src")
-;;
-;; This means that the corresponding file will be searched for first in
-;; the current directory, then in ../../src, then in one of the directories
-;; under ../include, and so on. The star is _not_ a general wildcard
-;; character: it just indicates that the subdirectories of this directory
-;; must each be searched in turn. Environment variables will be expanded in
-;; the ff-search-directories variable.
-;;
-;; If the point is on a #include line, the file to be #included is searched
-;; for in the same manner. This can be disabled with the ff-ignore-include
-;; variable, or by calling ff-get-other-file instead of ff-find-other-file.
-;;
-;; If the file was not found, ff-find-other-file will prompt you for where
-;; to create the new "corresponding file" (defaults to the current directory),
-;; unless the variable ff-always-try-to-create is set to nil.
-;;
-;; GIVEN AN ARGUMENT (with the ^U prefix), ff-find-other-file will get the
-;; other file in another (the other?) window (see find-file-other-window and
-;; switch-to-buffer-other-window). This can be set on a more permanent basis
-;; by setting ff-always-in-other-window to t in which case the ^U prefix will
-;; do the opposite of what was described above.
-;;
-;; THERE ARE FIVE AVAILABLE HOOKS, called in this order if non-nil:
-;;
-;; - ff-pre-find-hook - called before the search for the other file starts
-;; - ff-not-found-hook - called when the other file could not be found
-;; - ff-pre-load-hook - called just before the other file is 'loaded'
-;; - ff-file-created-hook - called when the other file is created
-;; - ff-post-load-hook - called just after the other file is 'loaded'
-;;
-;; The *load-hook allow you to place point where you want it in the other
-;; file.
-
-;; CREDITS:
-;; Many thanks go to TUSC Computer Systems Pty Ltd for providing an
-;; environment that made the development of this package possible.
-;;
-;; Many thanks also go to all those who provided valuable feedback throughout
-;; the development of this package:
-;; Rolf Ebert in particular, Fritz Knabe, Heddy Boubaker, Sebastian Kremer,
-;; Vasco Lopes Paulo, Mark A. Plaksin, Robert Lang, Trevor West, Kevin
-;; Pereira, Benedict Lofstedt & Justin Vallon.
-
-;;; Code:
-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;; User definable variables:
-
-(defgroup ff nil
- "Find a file corresponding to this one given a pattern."
- :prefix "ff-"
- :link '(emacs-commentary-link "find-file")
- :group 'find-file)
-
-(defcustom ff-pre-find-hook nil
- "List of functions to be called before the search for the file starts."
- :type 'hook
- :local t)
-
-(defcustom ff-pre-load-hook nil
- "List of functions to be called before the other file is loaded."
- :type 'hook
- :local t)
-
-(defcustom ff-post-load-hook nil
- "List of functions to be called after the other file is loaded."
- :type 'hook
- :local t)
-
-(defcustom ff-not-found-hook nil
- "List of functions to be called if the other file could not be found."
- :type 'hook
- :local t)
-
-(defcustom ff-file-created-hook nil
- "List of functions to be called if the other file needs to be created."
- :type 'hook
- :local t)
-
-(defcustom ff-case-fold-search nil
- "Non-nil means ignore cases in matches (see `case-fold-search').
-If you have extensions in different cases, you will want this to be nil."
- :type 'boolean
- :local t)
-
-(defcustom ff-always-in-other-window nil
- "If non-nil, find the corresponding file in another window by default.
-To override this, give an argument to `ff-find-other-file'."
- :type 'boolean
- :local t)
-
-(defcustom ff-ignore-include nil
- "If non-nil, ignore `#include' lines."
- :type 'boolean
- :local t)
-
-(defcustom ff-always-try-to-create t
- "If non-nil, always attempt to create the other file if it was not found."
- :type 'boolean)
-
-(defcustom ff-quiet-mode nil
- "If non-nil, do not trace which directories are being searched."
- :type 'boolean
- :local t)
-
-;;;###autoload
-(defcustom ff-special-constructs
- ;; C/C++ include, for NeXTstep too
- `((,(purecopy "^#\\s *\\(include\\|import\\)\\s +[<\"]\\(.*\\)[>\"]") .
- ,(lambda () (match-string 2))))
- ;; We include `ff-treat-as-special' documentation here so that autoload
- ;; can make it available to be read prior to loading this file.
- "List of special constructs recognized by `ff-treat-as-special'.
-Each element, tried in order, has the form (REGEXP . EXTRACT).
-If REGEXP matches the current line (from the beginning of the line),
-`ff-treat-as-special' calls function EXTRACT with no args.
-If EXTRACT returns nil, keep trying. Otherwise, return the
-filename that EXTRACT returned."
- :type '(repeat (cons regexp function)))
-
-(defvaralias 'ff-related-file-alist 'ff-other-file-alist)
-(defcustom ff-other-file-alist 'cc-other-file-alist
- "Alist of extensions to find given the current file's extension.
-
-The value could be an alist or a symbol whose value is an alist.
-Each element of the alist has the form
-
- (REGEXP (EXTENSION...))
-
-where REGEXP is the regular expression matching a file's extension,
-and EXTENSIONs is the list of literal file-name extensions to search
-for. The list of extensions should contain the most used extensions
-before the others, since the search algorithm searches sequentially
-through each directory specified in `ff-search-directories'.
-
-Alist elements can also be of the form
-
- (REGEXP FUNCTION)
-
-where FUNCTION is a function of one argument, the current file's name,
-that returns the list of possible names of the corresponding files, with
-or without leading directories. Note the difference: FUNCTION returns
-the list of file names, not their extensions. This is for the case when
-REGEXP is not enough to determine the file name of the other file.
-
-If a file is not found, a new one is created with the first
-matching extension or name (e.g., `.cc' yields `.hh').
-
-This alist should be set by the major mode.
-
-Note: if an element of the alist names a FUNCTION as its cdr, that
-function must return a non-nil list of file-names. It cannot
-return nil, nor can it signal in any way a failure to find a suitable
-list of file names."
- :type '(choice (repeat (list regexp (choice (repeat string) function)))
- symbol)
- :local t)
-
-(defcustom ff-search-directories 'cc-search-directories
- "List of directories to search for a specific file.
-
-Set by default to `cc-search-directories', expanded at run-time.
-
-This list is searched through with each extension specified in
-`ff-other-file-alist' that matches this file's extension. So the
-longer the list, the longer it'll take to realize that a file
-may not exist.
-
-A typical format is
-
- (\".\" \"/usr/include\" \"$PROJECT/*/include\")
-
-Environment variables can be inserted between slashes (`/').
-They will be replaced by their definition. If a variable does
-not exist, it is replaced (silently) with an empty string.
-
-The stars are *not* wildcards: they are searched for together with
-the preceding slash. The star represents all the subdirectories except
-`..', and each of these subdirectories will be searched in turn."
- :type '(choice (repeat directory) symbol)
- :local t)
-
-(defcustom cc-search-directories
- '("." "/usr/include" "/usr/local/include/*")
- "See the description of the `ff-search-directories' variable."
- :type '(repeat directory))
-
-(defcustom cc-other-file-alist
- '(("\\.cc\\'" (".hh" ".h"))
- ("\\.hh\\'" (".cc" ".C" ".CC" ".cxx" ".cpp" ".c++"))
-
- ("\\.c\\'" (".h"))
- ("\\.m\\'" (".h"))
- ("\\.h\\'" (".c" ".cc" ".C" ".CC" ".cxx" ".cpp" ".c++" ".m"))
-
- ("\\.C\\'" (".H" ".hh" ".h"))
- ("\\.H\\'" (".C" ".CC"))
-
- ("\\.CC\\'" (".HH" ".H" ".hh" ".h"))
- ("\\.HH\\'" (".CC"))
-
- ("\\.c\\+\\+\\'" (".h++" ".hh" ".h"))
- ("\\.h\\+\\+\\'" (".c++"))
-
- ("\\.cpp\\'" (".hpp" ".hh" ".h"))
- ("\\.hpp\\'" (".cpp"))
-
- ("\\.cxx\\'" (".hxx" ".hh" ".h"))
- ("\\.hxx\\'" (".cxx")))
- "Alist of extensions to find given the current file's extension.
-
-This list should contain the most used extensions before the others,
-since the search algorithm searches sequentially through each directory
-specified in `ff-search-directories'. If a file is not found, a new one
-is created with the first matching extension (`.cc' yields `.hh')."
- :version "24.4" ; add .m
- :type '(repeat (list regexp (choice (repeat string) function))))
-
-(defcustom modula2-other-file-alist
- '(
- ("\\.mi\\'" (".md")) ;; Modula-2 module definition
- ("\\.md\\'" (".mi")) ;; and implementation.
- )
- "See the description for the `ff-search-directories' variable."
- :type '(repeat (list regexp (choice (repeat string) function))))
-
-
-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;; No user definable variables beyond this point!
-;; ==============================================
-
-
-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;; User entry points
-
-;;;###autoload
-(defun ff-get-other-file (&optional in-other-window)
- "Find the header or source file corresponding to this file.
-See also the documentation for `ff-find-other-file'.
-
-If optional IN-OTHER-WINDOW is non-nil, find the file in another window."
- (interactive "P")
- (let ((ff-ignore-include t))
- (ff-find-the-other-file in-other-window)))
-
-;;;###autoload
-(defalias 'ff-find-related-file #'ff-find-other-file)
-
-;;;###autoload
-(defun ff-find-other-file (&optional in-other-window ignore-include event)
- "Find the header or source file corresponding to this file.
-Being on a `#include' line pulls in that file.
-
-If optional IN-OTHER-WINDOW is non-nil, find the file in the other window.
-If optional IGNORE-INCLUDE is non-nil, ignore being on `#include' lines.
-
-If optional EVENT is non-nil (default `last-nonmenu-event', move
-point to the end position of that event before calling the
-various ff-* hooks.
-
-Variables of interest include:
-
- - `ff-case-fold-search'
- Non-nil means ignore cases in matches (see `case-fold-search').
- If you have extensions in different cases, you will want this to be nil.
-
- - `ff-always-in-other-window'
- If non-nil, always open the other file in another window, unless an
- argument is given to `ff-find-other-file'.
-
- - `ff-ignore-include'
- If non-nil, ignores #include lines.
-
- - `ff-always-try-to-create'
- If non-nil, always attempt to create the other file if it was not found.
-
- - `ff-quiet-mode'
- If non-nil, does not trace which directories are being searched.
-
- - `ff-special-constructs'
- A list of regular expressions specifying how to recognize special
- constructs such as include files etc, and an associated method for
- extracting the filename from that construct.
-
- - `ff-other-file-alist'
- Alist of extensions to find given the current file's extension.
-
- - `ff-search-directories'
- List of directories searched through with each extension specified in
- `ff-other-file-alist' that matches this file's extension.
-
- - `ff-pre-find-hook'
- List of functions to be called before the search for the file starts.
-
- - `ff-pre-load-hook'
- List of functions to be called before the other file is loaded.
-
- - `ff-post-load-hook'
- List of functions to be called after the other file is loaded.
-
- - `ff-not-found-hook'
- List of functions to be called if the other file could not be found.
-
- - `ff-file-created-hook'
- List of functions to be called if the other file has been created."
- (interactive (list current-prefix-arg nil last-nonmenu-event))
- ;; We want to preserve point in the current buffer. But the point of
- ;; ff-find-the-other-file is to make the other file buffer
- ;; current, so we can't use save-excursion here (see bug 48535).
- (let ((start-buffer (current-buffer))
- (start-point (point)))
- (posn-set-point (event-end event))
- (let ((ff-ignore-include ignore-include))
- (ff-find-the-other-file in-other-window))
- (with-current-buffer start-buffer
- (goto-char start-point))))
-
-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;; Support functions
-
-(defun ff-buffer-file-name (&optional buf)
- "Like `buffer-file-name' but works with indirect buffers as well.
-If BUF is nil, uses the current buffer."
- (unless buf
- (setq buf (current-buffer)))
- (or (buffer-file-name buf)
- (when (buffer-base-buffer buf)
- (buffer-file-name (buffer-base-buffer buf)))))
-
-(defun ff-find-the-other-file (&optional in-other-window)
- "Find the header or source file corresponding to the current file.
-Being on a `#include' line pulls in that file, but see the help on
-the `ff-ignore-include' variable.
-
-If optional IN-OTHER-WINDOW is non-nil, find the file in another window."
-
- (let (match ;; matching regexp for this file
- suffixes ;; set of replacing regexps for the matching regexp
- action ;; function to generate the names of the other files
- fname ;; basename of this file
- pos ;; where we start matching filenames
- stub ;; name of the file without extension
- alist ;; working copy of the list of file extensions
- pathname ;; the pathname of the file or the #include line
- default-name ;; file we should create if none found
- format ;; what we have to match
- found ;; name of the file or buffer found - nil if none
- dirs ;; local value of ff-search-directories
- no-match) ;; whether we know about this kind of file
-
- (run-hooks 'ff-pre-find-hook 'ff-pre-find-hooks)
-
- (message "Working...")
-
- (setq dirs
- (ff-list-replace-env-vars (if (symbolp ff-search-directories)
- (symbol-value ff-search-directories)
- ff-search-directories)))
-
- (setq fname (ff-treat-as-special))
-
- (cond
- ((and (not ff-ignore-include) fname)
- (setq default-name fname)
- (setq found (ff-get-file dirs fname nil in-other-window)))
-
- ;; let's just get the corresponding file
- (t
- (setq alist (if (symbolp ff-other-file-alist)
- (symbol-value ff-other-file-alist)
- ff-other-file-alist)
- pathname (or (ff-buffer-file-name) "/none.none"))
-
- (setq fname (file-name-nondirectory pathname)
- no-match nil
- match (car alist))
-
- ;; find the table entry corresponding to this file
- (setq pos (ff-string-match (car match) fname))
- (while (and match (if (and pos (>= pos 0)) nil (not pos)))
- (setq alist (cdr alist))
- (setq match (car alist))
- (setq pos (ff-string-match (car match) fname)))
-
- ;; no point going on if we haven't found anything
- (if (not match)
- (setq no-match t)
-
- ;; otherwise, suffixes contains what we need
- (setq suffixes (car (cdr match))
- action (car (cdr match))
- found nil)
-
- ;; if we have a function to generate new names,
- ;; invoke it with the name of the current file
- (if (and (atom action) (fboundp action))
- (setq suffixes (funcall action (ff-buffer-file-name))
- match (cons (car match) (list suffixes))
- stub nil
- default-name (car suffixes))
-
- ;; otherwise build our filename stub
- (cond
-
- ;; get around the problem that 0 and nil both mean false!
- ((= pos 0)
- (setq format "")
- (setq stub "")
- )
-
- (t
- (setq format (concat "\\(.+\\)" (car match)))
- (string-match format fname)
- ;; FIXME: What if `string-match' failed?
- (setq stub (match-string 1 fname))
- ))
-
- ;; if we find nothing, we should try to get a file like this one
- (setq default-name
- (concat stub (car (car (cdr match))))))
-
- ;; do the real work - find the file
- (setq found
- (ff-get-file dirs
- stub
- suffixes
- in-other-window)))))
-
- (cond
- (no-match ;; could not even determine the other file
- (message ""))
-
- (t
- (cond
-
- ((not found) ;; could not find the other file
-
- (run-hooks 'ff-not-found-hook 'ff-not-found-hooks)
-
- (cond
- (ff-always-try-to-create ;; try to create the file
- (let (name pathname)
-
- (setq name
- (expand-file-name
- (read-directory-name
- (format "Find or create %s in: " default-name)
- default-directory default-name nil)))
-
- (setq pathname
- (if (file-directory-p name)
- (concat (file-name-as-directory name) default-name)
- (setq found name)))
-
- (ff-find-file pathname in-other-window t)))
-
- (t ;; don't create the file, just whinge
- (message "No file found for %s" fname))))
-
- (t ;; matching file found
- nil))))
-
- found)) ;; return buffer-name or filename
-
-(defun ff-get-file (search-dirs filename &optional suffix-list other-window)
- "Find a file in the SEARCH-DIRS with the given FILENAME (or filename stub).
-If (optional) SUFFIX-LIST is nil, search for FILENAME, otherwise search
-for FILENAME with each of the given suffixes. Get the file or the buffer
-corresponding to the name of the first file found, or nil."
- (let ((filename (ff-get-file-name search-dirs filename suffix-list)))
-
- (cond
- ((not filename)
- nil)
-
- ((bufferp (get-file-buffer filename))
- (ff-switch-to-buffer (get-file-buffer filename) other-window)
- filename)
-
- ((file-exists-p filename)
- (ff-find-file filename other-window nil)
- filename)
-
- (t
- nil))))
-
-(defun ff-get-file-name (search-dirs fname-stub &optional suffix-list)
- "Find a file in SEARCH-DIRS with the given name (or stub) FNAME-STUB.
-If (optional) SUFFIX-LIST is nil, search for FNAME-STUB, otherwise
-search for FNAME-STUB with each of the given suffixes. Return the
-name of the first file found."
- (let (dirs ;; working copy of dirs to search
- dir ;; the current dir considered
- file ;; filename being looked for
- rest ;; pathname after first /*
- this-suffix ;; the suffix we are currently considering
- suffixes ;; working copy of suffix-list
- filename ;; built filename
- blist ;; list of live buffers
- buf ;; current buffer in blist
- found) ;; whether we have found anything
-
- (setq suffixes suffix-list)
-
- ;; suffixes is nil => fname-stub is the file we are looking for
- ;; otherwise fname-stub is a stub, and we append a suffix
- (if suffixes
- (setq this-suffix (car suffixes))
- (setq this-suffix "")
- (setq suffixes (list "")))
-
- ;; find whether the file is in a buffer first
- (while (and suffixes (not found))
- (setq filename (concat fname-stub this-suffix))
-
- (if (not ff-quiet-mode)
- (message "Finding buffer %s..." filename))
-
- (if (bufferp (get-file-buffer filename))
- (setq found (ff-buffer-file-name (get-file-buffer filename))))
-
- (setq blist (buffer-list))
- (setq buf (buffer-name (car blist)))
- (while (and blist (not found))
-
- (if (string-match-p (concat filename "<[0-9]+>") buf)
- (setq found (ff-buffer-file-name (car blist))))
-
- (setq blist (cdr blist))
- (setq buf (buffer-name (car blist))))
-
- (setq suffixes (cdr suffixes))
- (setq this-suffix (car suffixes)))
-
- ;; now look for the real file
- (setq dirs search-dirs)
- (setq dir (car dirs))
- (while (and (not found) dirs)
-
- (setq suffixes suffix-list)
-
- ;; if dir does not contain '/*', look for the file
- (if (and dir (not (string-match "\\([^*]*\\)/\\*\\(/.*\\)*" dir)))
- (progn
-
- ;; suffixes is nil => fname-stub is the file we are looking for
- ;; otherwise fname-stub is a stub, and we append a suffix
- (if suffixes
- (setq this-suffix (car suffixes))
- (setq this-suffix "")
- (setq suffixes (list "")))
-
- (while (and suffixes (not found))
-
- (setq filename (concat fname-stub this-suffix))
- (setq file (expand-file-name filename dir))
-
- (if (not ff-quiet-mode)
- (message "Finding %s..." file))
-
- (if (file-exists-p file)
- (setq found file))
-
- (setq suffixes (cdr suffixes))
- (setq this-suffix (car suffixes))))
-
- ;; otherwise dir matches the '/*', so search each dir separately
- (progn
- (setq rest (if (match-beginning 2)
- (match-string 2 dir)
- ""))
- (setq dir (match-string 1 dir))
-
- (let ((dirlist (ff-all-dirs-under dir '("..")))
- this-dir compl-dirs)
-
- (setq this-dir (car dirlist))
- (while dirlist
- (setq compl-dirs
- (append
- compl-dirs
- (list (concat this-dir rest))
- ))
- (setq dirlist (cdr dirlist))
- (setq this-dir (car dirlist)))
-
- (if compl-dirs
- (setq found (ff-get-file-name compl-dirs
- fname-stub
- suffix-list))))))
- (setq dirs (cdr dirs))
- (setq dir (car dirs)))
-
- (if found
- (message "%s found" found))
-
- found))
-
-(defun ff-string-match (regexp string &optional start)
- "Like `string-match', but set `case-fold-search' temporarily.
-The value used comes from `ff-case-fold-search'."
- (if regexp
- (let ((case-fold-search ff-case-fold-search))
- (string-match regexp string start))))
-
-(defun ff-list-replace-env-vars (search-list)
- "Replace environment variables (of the form $VARIABLE) in SEARCH-LIST."
- (let (list
- (var (car search-list)))
- (while search-list
- (if (string-match "\\(.*\\)\\$[({]*\\([[:alnum:]_]+\\)[)}]*\\(.*\\)" var)
- (setq var
- (concat
- (match-string 1 var)
- (getenv (match-string 2 var))
- (match-string 3 var))))
- (setq search-list (cdr search-list))
- (setq list (cons var list))
- (setq var (car search-list)))
- (setq search-list (reverse list))))
-
-(defun ff-treat-as-special ()
- "Return the file to look for if the construct was special, else nil.
-See variable `ff-special-constructs'."
- (save-excursion
- (beginning-of-line 1)
- (let* (fname
- (list ff-special-constructs)
- (elem (car list))
- (regexp (car elem))
- (match (cdr elem)))
- (while (and list (not fname))
- (if (and (looking-at regexp) match)
- (setq fname (funcall match)))
- (setq list (cdr list))
- (setq elem (car list))
- (setq regexp (car elem))
- (setq match (cdr elem)))
- fname)))
-
-(define-obsolete-function-alias 'ff-basename #'file-name-nondirectory "28.1")
-
-(defun ff-all-dirs-under (here &optional exclude)
- "Get all the directory files under directory HERE.
-Exclude all files in the optional EXCLUDE list."
- (if (file-directory-p here)
- (ignore-errors
- (let ((files (directory-files here t))
- (dirlist (list))
- file)
- (while files
- (setq file (car files))
- (if (and
- (file-directory-p file)
- (not (member (file-name-nondirectory file) exclude)))
- (setq dirlist (cons file dirlist)))
- (setq files (cdr files)))
- (setq dirlist (reverse dirlist))))
- nil))
-
-(defun ff-switch-file (f1 f2 file &optional in-other-window new-file)
- "Call F1 or F2 on FILE, according to IN-OTHER-WINDOW.
-In addition, this runs various hooks.
-
-Either F1 or F2 receives FILE as the sole argument.
-The decision of which one to call is based on IN-OTHER-WINDOW
-and on the global variable `ff-always-in-other-window'.
-
-F1 and F2 are typically `find-file' / `find-file-other-window'
-or `switch-to-buffer' / `switch-to-buffer-other-window' function pairs.
-
-If optional NEW-FILE is t, then a special hook (`ff-file-created-hook') is
-called before `ff-post-load-hook'."
- (run-hooks 'ff-pre-load-hook 'ff-pre-load-hooks)
- (funcall (if (or
- (and in-other-window (not ff-always-in-other-window))
- (and (not in-other-window) ff-always-in-other-window))
- f2 f1)
- file)
- (if new-file
- (run-hooks 'ff-file-created-hook 'ff-file-created-hooks))
- (run-hooks 'ff-post-load-hook 'ff-post-load-hooks))
-
-(defun ff-find-file (file &optional in-other-window new-file)
- "Like `find-file', but may show the file in another window."
- (ff-switch-file #'find-file
- #'find-file-other-window
- file in-other-window new-file))
-
-(defun ff-switch-to-buffer (buffer-or-name &optional in-other-window)
- "Like `switch-to-buffer', but may show the buffer in another window."
-
- (ff-switch-file #'switch-to-buffer
- #'switch-to-buffer-other-window
- buffer-or-name in-other-window nil))
-
-;;;###autoload
-(define-obsolete-function-alias
- 'ff-mouse-find-other-file #'ff-find-other-file "28.1")
-
-;;;###autoload
-(define-obsolete-function-alias
- 'ff-mouse-find-other-file-other-window #'ff-find-other-file-other-window "28.1")
-;;;###autoload
-(defun ff-find-other-file-other-window (event)
- "Visit the file you point at in another window."
- (interactive (list last-nonmenu-event))
- (ff-find-other-file t nil event))
-
-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;; This section offers an example of user defined function to select files
-
-(defun ff-upcase-p (string)
- "Return t if STRING is all uppercase."
- ;; FIXME: Why `ff-string-match' since `[:upper:]' only makes
- ;; sense when `case-fold-search' is nil?
- (ff-string-match "\\`[[:upper:]]*\\'" string))
-
-(defun ff-cc-hh-converter (arg)
- "Discriminate file extensions.
-Build up a new file list based possibly on part of the directory name
-and the name of the file passed in."
- (ff-string-match "\\(.*\\)/\\([^/]+\\)/\\([^.]+\\).\\([^/]+\\)$" arg)
- (let ((dire (match-string 2 arg))
- (file (match-string 3 arg))
- (extn (match-string 4 arg))
- return-list)
- (cond
- ;; fooZapJunk.cc => ZapJunk.{hh,h} or fooZapJunk.{hh,h}
- ((and (string= extn "cc")
- (ff-string-match "^\\([[:lower:]]+\\)\\([[:upper:]].+\\)$" file))
- (let ((stub (match-string 2 file)))
- (setq dire (upcase (match-string 1 file)))
- (setq return-list (list (concat stub ".hh")
- (concat stub ".h")
- (concat file ".hh")
- (concat file ".h")))
- ))
- ;; FOO/ZapJunk.hh => fooZapJunk.{cc,C} or ZapJunk.{cc,C}
- ((and (string= extn "hh") (ff-upcase-p dire) file)
- (let ((stub (concat (downcase dire) file)))
- (setq return-list (list (concat stub ".cc")
- (concat stub ".C")
- (concat file ".cc")
- (concat file ".C")))
- ))
- ;; zap.cc => zap.hh or zap.h
- ((string= extn "cc")
- (let ((stub file))
- (setq return-list (list (concat stub ".hh")
- (concat stub ".h")))
- ))
- ;; zap.hh => zap.cc or zap.C
- ((string= extn "hh")
- (let ((stub file))
- (setq return-list (list (concat stub ".cc")
- (concat stub ".C")))
- ))
- (t
- nil))
-
- return-list))
-
-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;; This section offers an example of user defined function to place point.
-;; The regexps are Ada specific.
-
-(defvar ff-function-name nil "Name of the function we are in.")
-
-;; bind with (setq ff-pre-load-hook 'ff-which-function-are-we-in)
-;;
-(defvar ada-procedure-start-regexp)
-(defvar ada-package-start-regexp)
-
-(defun ff-which-function-are-we-in ()
- "Return the name of the function whose definition/declaration point is in.
-Also remember that name in `ff-function-name'."
- (setq ff-function-name
- (save-excursion
- (if (or (re-search-backward ada-procedure-start-regexp nil t)
- (re-search-backward ada-package-start-regexp nil t))
- (match-string 0)))))
-
-;; bind with (setq ff-post-load-hook 'ff-set-point-accordingly)
-;;
-(defun ff-set-point-accordingly ()
- "Find the function specified in `ff-function-name'.
-That name was previously determined by `ff-which-function-are-we-in'."
- (if ff-function-name
- (progn
- (goto-char (point-min))
- (search-forward ff-function-name nil t))))
-
-(provide 'find-file)
-
-;;; find-file.el ends here