1993-06-01 20:09:25 +00:00
|
|
|
;;; desktop.el --- save partial status of Emacs when killed
|
|
|
|
|
2013-01-01 09:11:05 +00:00
|
|
|
;; Copyright (C) 1993-1995, 1997, 2000-2013 Free Software Foundation,
|
|
|
|
;; Inc.
|
1993-06-01 20:09:25 +00:00
|
|
|
|
|
|
|
;; Author: Morten Welinder <terra@diku.dk>
|
2000-02-02 14:05:36 +00:00
|
|
|
;; Keywords: convenience
|
2011-11-17 09:40:48 -08:00
|
|
|
;; Favorite-brand-of-beer: None, I hate beer.
|
1993-06-01 20:09:25 +00:00
|
|
|
|
|
|
|
;; This file is part of GNU Emacs.
|
|
|
|
|
2008-05-06 08:06:51 +00:00
|
|
|
;; GNU Emacs is free software: you can redistribute it and/or modify
|
1993-06-01 20:09:25 +00:00
|
|
|
;; it under the terms of the GNU General Public License as published by
|
2008-05-06 08:06:51 +00:00
|
|
|
;; the Free Software Foundation, either version 3 of the License, or
|
|
|
|
;; (at your option) any later version.
|
1993-06-01 20:09:25 +00:00
|
|
|
|
|
|
|
;; 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
|
2008-05-06 08:06:51 +00:00
|
|
|
;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
|
1993-06-01 20:09:25 +00:00
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; Save the Desktop, i.e.,
|
|
|
|
;; - some global variables
|
|
|
|
;; - the list of buffers with associated files. For each buffer also
|
|
|
|
;; - the major mode
|
|
|
|
;; - the default directory
|
|
|
|
;; - the point
|
|
|
|
;; - the mark & mark-active
|
|
|
|
;; - buffer-read-only
|
1994-01-06 11:34:51 +00:00
|
|
|
;; - some local variables
|
2013-07-15 02:07:51 +02:00
|
|
|
;; - frame and window configuration
|
1993-06-01 20:09:25 +00:00
|
|
|
|
2004-04-16 12:51:06 +00:00
|
|
|
;; To use this, use customize to turn on desktop-save-mode or add the
|
2012-09-17 13:41:04 +08:00
|
|
|
;; following line somewhere in your init file:
|
1993-10-07 16:50:26 +00:00
|
|
|
;;
|
2004-04-16 12:51:06 +00:00
|
|
|
;; (desktop-save-mode 1)
|
1993-10-07 16:50:26 +00:00
|
|
|
;;
|
2004-04-16 12:51:06 +00:00
|
|
|
;; For further usage information, look at the section
|
2009-03-22 20:09:34 +00:00
|
|
|
;; (info "(emacs)Saving Emacs Sessions") in the GNU Emacs Manual.
|
2004-04-16 12:51:06 +00:00
|
|
|
|
|
|
|
;; When the desktop module is loaded, the function `desktop-kill' is
|
2007-04-11 02:28:26 +00:00
|
|
|
;; added to the `kill-emacs-hook'. This function is responsible for
|
2004-04-16 12:51:06 +00:00
|
|
|
;; saving the desktop when Emacs is killed. Furthermore an anonymous
|
2007-04-11 02:28:26 +00:00
|
|
|
;; function is added to the `after-init-hook'. This function is
|
2004-04-16 12:51:06 +00:00
|
|
|
;; responsible for loading the desktop when Emacs is started.
|
1994-01-06 11:34:51 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
;; Special handling.
|
|
|
|
;; -----------------
|
|
|
|
;; Variables `desktop-buffer-mode-handlers' and `desktop-minor-mode-handlers'
|
|
|
|
;; are supplied to handle special major and minor modes respectively.
|
|
|
|
;; `desktop-buffer-mode-handlers' is an alist of major mode specific functions
|
2007-04-11 02:28:26 +00:00
|
|
|
;; to restore a desktop buffer. Elements must have the form
|
2006-02-10 11:07:50 +00:00
|
|
|
;;
|
2005-08-10 19:38:52 +00:00
|
|
|
;; (MAJOR-MODE . RESTORE-BUFFER-FUNCTION).
|
2006-02-10 11:07:50 +00:00
|
|
|
;;
|
2005-08-10 19:38:52 +00:00
|
|
|
;; Functions listed are called by `desktop-create-buffer' when `desktop-read'
|
2007-04-11 02:28:26 +00:00
|
|
|
;; evaluates the desktop file. Buffers with a major mode not specified here,
|
2005-08-10 19:38:52 +00:00
|
|
|
;; are restored by the default handler `desktop-restore-file-buffer'.
|
|
|
|
;; `desktop-minor-mode-handlers' is an alist of functions to restore
|
|
|
|
;; non-standard minor modes. Elements must have the form
|
2006-02-10 11:07:50 +00:00
|
|
|
;;
|
2005-08-10 19:38:52 +00:00
|
|
|
;; (MINOR-MODE . RESTORE-FUNCTION).
|
2006-02-10 11:07:50 +00:00
|
|
|
;;
|
2005-08-10 19:38:52 +00:00
|
|
|
;; Functions are called by `desktop-create-buffer' to restore minor modes.
|
|
|
|
;; Minor modes not specified here, are restored by the standard minor mode
|
|
|
|
;; function. If you write a module that defines a major or minor mode that
|
|
|
|
;; needs a special handler, then place code like
|
|
|
|
|
|
|
|
;; (defun foo-restore-desktop-buffer
|
|
|
|
;; ...
|
|
|
|
;; (add-to-list 'desktop-buffer-mode-handlers
|
|
|
|
;; '(foo-mode . foo-restore-desktop-buffer))
|
|
|
|
|
|
|
|
;; or
|
|
|
|
|
|
|
|
;; (defun bar-desktop-restore
|
|
|
|
;; ...
|
|
|
|
;; (add-to-list 'desktop-minor-mode-handlers
|
|
|
|
;; '(bar-mode . bar-desktop-restore))
|
|
|
|
|
2008-04-28 08:06:51 +00:00
|
|
|
;; in the module itself, and make sure that the mode function is
|
2007-04-11 02:28:26 +00:00
|
|
|
;; autoloaded. See the docstrings of `desktop-buffer-mode-handlers' and
|
2005-08-10 19:38:52 +00:00
|
|
|
;; `desktop-minor-mode-handlers' for more info.
|
|
|
|
|
|
|
|
;; Minor modes.
|
|
|
|
;; ------------
|
|
|
|
;; Conventional minor modes (see node "Minor Mode Conventions" in the elisp
|
|
|
|
;; manual) are handled in the following way:
|
|
|
|
;; When `desktop-save' saves the state of a buffer to the desktop file, it
|
|
|
|
;; saves as `desktop-minor-modes' the list of names of those variables in
|
|
|
|
;; `minor-mode-alist' that have a non-nil value.
|
|
|
|
;; When `desktop-create' restores the buffer, each of the symbols in
|
|
|
|
;; `desktop-minor-modes' is called as function with parameter 1.
|
|
|
|
;; The variables `desktop-minor-mode-table' and `desktop-minor-mode-handlers'
|
|
|
|
;; are used to handle non-conventional minor modes. `desktop-save' uses
|
|
|
|
;; `desktop-minor-mode-table' to map minor mode variables to minor mode
|
2007-04-11 02:28:26 +00:00
|
|
|
;; functions before writing `desktop-minor-modes'. If a minor mode has a
|
2005-08-10 19:38:52 +00:00
|
|
|
;; variable name that is different form its function name, an entry
|
|
|
|
|
|
|
|
;; (NAME RESTORE-FUNCTION)
|
|
|
|
|
|
|
|
;; should be added to `desktop-minor-mode-table'. If a minor mode should not
|
|
|
|
;; be restored, RESTORE-FUNCTION should be set to nil. `desktop-create' uses
|
|
|
|
;; `desktop-minor-mode-handlers' to lookup minor modes that needs a restore
|
|
|
|
;; function different from the usual minor mode function.
|
|
|
|
;; ---------------------------------------------------------------------------
|
2000-04-26 17:33:02 +00:00
|
|
|
|
1994-01-06 11:34:51 +00:00
|
|
|
;; By the way: don't use desktop.el to customize Emacs -- the file .emacs
|
|
|
|
;; in your home directory is used for that. Saving global default values
|
|
|
|
;; for buffers is an example of misuse.
|
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; PLEASE NOTE: The kill ring can be saved as specified by the variable
|
|
|
|
;; `desktop-globals-to-save' (by default it isn't). This may result in saving
|
|
|
|
;; things you did not mean to keep. Use M-x desktop-clear RET.
|
1994-01-06 11:34:51 +00:00
|
|
|
|
1994-05-01 02:07:27 +00:00
|
|
|
;; Thanks to hetrick@phys.uva.nl (Jim Hetrick) for useful ideas.
|
|
|
|
;; avk@rtsg.mot.com (Andrew V. Klein) for a dired tip.
|
|
|
|
;; chris@tecc.co.uk (Chris Boucher) for a mark tip.
|
|
|
|
;; f89-kam@nada.kth.se (Klas Mellbourn) for a mh-e tip.
|
|
|
|
;; kifer@sbkifer.cs.sunysb.edu (M. Kifer) for a bug hunt.
|
1995-04-06 20:05:39 +00:00
|
|
|
;; treese@lcs.mit.edu (Win Treese) for ange-ftp tips.
|
1996-01-14 23:24:40 +00:00
|
|
|
;; pot@cnuce.cnr.it (Francesco Potorti`) for misc. tips.
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ---------------------------------------------------------------------------
|
|
|
|
;; TODO:
|
|
|
|
;;
|
|
|
|
;; Recognize more minor modes.
|
|
|
|
;; Save mark rings.
|
1993-06-01 20:09:25 +00:00
|
|
|
|
|
|
|
;;; Code:
|
|
|
|
|
2013-07-22 03:25:47 +02:00
|
|
|
(require 'cl-lib)
|
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
(defvar desktop-file-version "206"
|
2004-04-16 12:51:06 +00:00
|
|
|
"Version number of desktop file format.
|
2003-04-09 01:37:56 +00:00
|
|
|
Written into the desktop file and used at desktop read to provide
|
|
|
|
backward compatibility.")
|
|
|
|
|
1994-01-06 11:34:51 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
1993-10-07 16:50:26 +00:00
|
|
|
;; USER OPTIONS -- settings you might want to play with.
|
|
|
|
;; ----------------------------------------------------------------------------
|
1997-04-12 03:18:33 +00:00
|
|
|
|
|
|
|
(defgroup desktop nil
|
|
|
|
"Save status of Emacs when you exit."
|
|
|
|
:group 'frames)
|
|
|
|
|
2012-05-12 23:05:06 -04:00
|
|
|
;; Maintained for backward compatibility
|
|
|
|
(define-obsolete-variable-alias 'desktop-enable 'desktop-save-mode "22.1")
|
2004-04-16 12:51:06 +00:00
|
|
|
;;;###autoload
|
|
|
|
(define-minor-mode desktop-save-mode
|
Fix minor mode docstrings for the new meaning of a nil ARG.
* abbrev.el (abbrev-mode):
* allout.el (allout-mode):
* autoinsert.el (auto-insert-mode):
* autoarg.el (autoarg-mode, autoarg-kp-mode):
* autorevert.el (auto-revert-mode, auto-revert-tail-mode)
(global-auto-revert-mode):
* battery.el (display-battery-mode):
* composite.el (global-auto-composition-mode)
(auto-composition-mode):
* delsel.el (delete-selection-mode):
* desktop.el (desktop-save-mode):
* dired-x.el (dired-omit-mode):
* dirtrack.el (dirtrack-mode):
* doc-view.el (doc-view-minor-mode):
* double.el (double-mode):
* electric.el (electric-indent-mode, electric-pair-mode):
* emacs-lock.el (emacs-lock-mode):
* epa-hook.el (auto-encryption-mode):
* follow.el (follow-mode):
* font-core.el (font-lock-mode):
* frame.el (auto-raise-mode, auto-lower-mode, blink-cursor-mode):
* help.el (temp-buffer-resize-mode):
* hilit-chg.el (highlight-changes-mode)
(highlight-changes-visible-mode):
* hi-lock.el (hi-lock-mode):
* hl-line.el (hl-line-mode, global-hl-line-mode):
* icomplete.el (icomplete-mode):
* ido.el (ido-everywhere):
* image-file.el (auto-image-file-mode):
* image-mode.el (image-minor-mode):
* iswitchb.el (iswitchb-mode):
* jka-cmpr-hook.el (auto-compression-mode):
* linum.el (linum-mode):
* longlines.el (longlines-mode):
* master.el (master-mode):
* mb-depth.el (minibuffer-depth-indicate-mode):
* menu-bar.el (menu-bar-mode):
* minibuf-eldef.el (minibuffer-electric-default-mode):
* mouse-sel.el (mouse-sel-mode):
* msb.el (msb-mode):
* mwheel.el (mouse-wheel-mode):
* outline.el (outline-minor-mode):
* paren.el (show-paren-mode):
* recentf.el (recentf-mode):
* reveal.el (reveal-mode, global-reveal-mode):
* rfn-eshadow.el (file-name-shadow-mode):
* ruler-mode.el (ruler-mode):
* savehist.el (savehist-mode):
* scroll-all.el (scroll-all-mode):
* scroll-bar.el (scroll-bar-mode):
* server.el (server-mode):
* shell.el (shell-dirtrack-mode):
* simple.el (auto-fill-mode, transient-mark-mode)
(visual-line-mode, overwrite-mode, binary-overwrite-mode)
(line-number-mode, column-number-mode, size-indication-mode)
(auto-save-mode, normal-erase-is-backspace-mode, visible-mode):
* strokes.el (strokes-mode):
* time.el (display-time-mode):
* t-mouse.el (gpm-mouse-mode):
* tool-bar.el (tool-bar-mode):
* tooltip.el (tooltip-mode):
* type-break.el (type-break-mode-line-message-mode)
(type-break-query-mode):
* view.el (view-mode):
* whitespace.el (whitespace-mode, whitespace-newline-mode)
(global-whitespace-mode, global-whitespace-newline-mode):
* xt-mouse.el (xterm-mouse-mode): Doc fix.
* emacs-lisp/easy-mmode.el (define-globalized-minor-mode): Fix
autogenerated docstring.
2011-10-19 08:54:24 -04:00
|
|
|
"Toggle desktop saving (Desktop Save mode).
|
|
|
|
With a prefix argument ARG, enable Desktop Save mode if ARG is
|
|
|
|
positive, and disable it otherwise. If called from Lisp, enable
|
|
|
|
the mode if ARG is omitted or nil.
|
|
|
|
|
|
|
|
If Desktop Save mode is enabled, the state of Emacs is saved from
|
|
|
|
one session to another. See variable `desktop-save' and function
|
|
|
|
`desktop-read' for details."
|
2004-04-16 12:51:06 +00:00
|
|
|
:global t
|
|
|
|
:group 'desktop)
|
|
|
|
|
2007-06-12 09:11:31 +00:00
|
|
|
(defun desktop-save-mode-off ()
|
|
|
|
"Disable `desktop-save-mode'. Provided for use in hooks."
|
|
|
|
(desktop-save-mode 0))
|
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
(defcustom desktop-save 'ask-if-new
|
2008-12-03 05:48:14 +00:00
|
|
|
"Specifies whether the desktop should be saved when it is killed.
|
2004-04-16 12:51:06 +00:00
|
|
|
A desktop is killed when the user changes desktop or quits Emacs.
|
|
|
|
Possible values are:
|
2003-04-09 01:37:56 +00:00
|
|
|
t -- always save.
|
|
|
|
ask -- always ask.
|
|
|
|
ask-if-new -- ask if no desktop file exists, otherwise just save.
|
|
|
|
ask-if-exists -- ask if desktop file exists, otherwise don't save.
|
|
|
|
if-exists -- save if desktop file exists, otherwise don't save.
|
|
|
|
nil -- never save.
|
2004-04-16 12:51:06 +00:00
|
|
|
The desktop is never saved when `desktop-save-mode' is nil.
|
2005-07-19 09:54:05 +00:00
|
|
|
The variables `desktop-dirname' and `desktop-base-file-name'
|
2004-04-16 12:51:06 +00:00
|
|
|
determine where the desktop is saved."
|
2006-06-23 21:42:33 +00:00
|
|
|
:type
|
|
|
|
'(choice
|
2003-04-09 01:37:56 +00:00
|
|
|
(const :tag "Always save" t)
|
|
|
|
(const :tag "Always ask" ask)
|
|
|
|
(const :tag "Ask if desktop file is new, else do save" ask-if-new)
|
|
|
|
(const :tag "Ask if desktop file exists, else don't save" ask-if-exists)
|
|
|
|
(const :tag "Save if desktop file exists, else don't" if-exists)
|
|
|
|
(const :tag "Never save" nil))
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2013-04-27 23:55:00 +03:00
|
|
|
(defcustom desktop-auto-save-timeout nil
|
|
|
|
"Number of seconds between auto-saves of the desktop.
|
|
|
|
Zero or nil means disable timer-based auto-saving."
|
|
|
|
:type '(choice (const :tag "Off" nil)
|
|
|
|
(integer :tag "Seconds"))
|
|
|
|
:set (lambda (symbol value)
|
|
|
|
(set-default symbol value)
|
2013-07-12 13:21:01 +02:00
|
|
|
(ignore-errors (desktop-auto-save-set-timer)))
|
2013-04-27 23:55:00 +03:00
|
|
|
:group 'desktop
|
|
|
|
:version "24.4")
|
|
|
|
|
2007-06-12 11:14:52 +00:00
|
|
|
(defcustom desktop-load-locked-desktop 'ask
|
|
|
|
"Specifies whether the desktop should be loaded if locked.
|
|
|
|
Possible values are:
|
|
|
|
t -- load anyway.
|
|
|
|
nil -- don't load.
|
|
|
|
ask -- ask the user.
|
|
|
|
If the value is nil, or `ask' and the user chooses not to load the desktop,
|
|
|
|
the normal hook `desktop-not-loaded-hook' is run."
|
|
|
|
:type
|
|
|
|
'(choice
|
|
|
|
(const :tag "Load anyway" t)
|
|
|
|
(const :tag "Don't load" nil)
|
|
|
|
(const :tag "Ask the user" ask))
|
|
|
|
:group 'desktop
|
2007-10-27 17:14:37 +00:00
|
|
|
:version "22.2")
|
2007-06-12 11:14:52 +00:00
|
|
|
|
2008-04-12 03:17:19 +00:00
|
|
|
(define-obsolete-variable-alias 'desktop-basefilename
|
|
|
|
'desktop-base-file-name "22.1")
|
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
(defcustom desktop-base-file-name
|
1996-09-01 19:04:28 +00:00
|
|
|
(convert-standard-filename ".emacs.desktop")
|
2004-04-16 12:51:06 +00:00
|
|
|
"Name of file for Emacs desktop, excluding the directory part."
|
1997-12-29 20:58:19 +00:00
|
|
|
:type 'file
|
|
|
|
:group 'desktop)
|
1993-06-01 20:09:25 +00:00
|
|
|
|
2007-06-12 09:11:31 +00:00
|
|
|
(defcustom desktop-base-lock-name
|
|
|
|
(convert-standard-filename ".emacs.desktop.lock")
|
|
|
|
"Name of lock file for Emacs desktop, excluding the directory part."
|
|
|
|
:type 'file
|
|
|
|
:group 'desktop
|
2007-10-27 17:14:37 +00:00
|
|
|
:version "22.2")
|
2007-06-12 09:11:31 +00:00
|
|
|
|
2012-06-21 15:15:42 +08:00
|
|
|
(defcustom desktop-path (list user-emacs-directory "~")
|
2003-04-09 01:37:56 +00:00
|
|
|
"List of directories to search for the desktop file.
|
|
|
|
The base name of the file is specified in `desktop-base-file-name'."
|
|
|
|
:type '(repeat directory)
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2010-09-01 22:38:30 -07:00
|
|
|
:version "23.2") ; user-emacs-directory added
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1997-04-12 03:18:33 +00:00
|
|
|
(defcustom desktop-missing-file-warning nil
|
2006-05-14 20:26:37 +00:00
|
|
|
"If non-nil, offer to recreate the buffer of a deleted file.
|
2004-04-26 18:58:19 +00:00
|
|
|
Also pause for a moment to display message about errors signaled in
|
|
|
|
`desktop-buffer-mode-handlers'.
|
|
|
|
|
|
|
|
If nil, just print error messages in the message buffer."
|
1997-04-12 03:18:33 +00:00
|
|
|
:type 'boolean
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
1993-06-01 20:09:25 +00:00
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
(defcustom desktop-no-desktop-file-hook nil
|
2004-04-16 12:51:06 +00:00
|
|
|
"Normal hook run when `desktop-read' can't find a desktop file.
|
2006-06-23 21:42:33 +00:00
|
|
|
Run in the directory in which the desktop file was sought.
|
2005-04-03 14:01:43 +00:00
|
|
|
May be used to show a dired buffer."
|
2003-04-09 01:37:56 +00:00
|
|
|
:type 'hook
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2007-06-12 09:11:31 +00:00
|
|
|
(defcustom desktop-not-loaded-hook nil
|
|
|
|
"Normal hook run when the user declines to re-use a desktop file.
|
|
|
|
Run in the directory in which the desktop file was found.
|
|
|
|
May be used to deal with accidental multiple Emacs jobs."
|
|
|
|
:type 'hook
|
|
|
|
:group 'desktop
|
|
|
|
:options '(desktop-save-mode-off save-buffers-kill-emacs)
|
2007-10-27 17:14:37 +00:00
|
|
|
:version "22.2")
|
2007-06-12 09:11:31 +00:00
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
(defcustom desktop-after-read-hook nil
|
2004-04-16 12:51:06 +00:00
|
|
|
"Normal hook run after a successful `desktop-read'.
|
2005-04-03 14:01:43 +00:00
|
|
|
May be used to show a buffer list."
|
2003-04-09 01:37:56 +00:00
|
|
|
:type 'hook
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2006-06-23 21:42:33 +00:00
|
|
|
:options '(list-buffers)
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2003-04-09 01:37:56 +00:00
|
|
|
|
|
|
|
(defcustom desktop-save-hook nil
|
2004-04-16 12:51:06 +00:00
|
|
|
"Normal hook run before the desktop is saved in a desktop file.
|
2006-06-23 21:42:33 +00:00
|
|
|
Run with the desktop buffer current with only the header present.
|
|
|
|
May be used to add to the desktop code or to truncate history lists,
|
|
|
|
for example."
|
2003-04-09 01:37:56 +00:00
|
|
|
:type 'hook
|
|
|
|
:group 'desktop)
|
|
|
|
|
2004-09-09 19:45:03 +00:00
|
|
|
(defcustom desktop-globals-to-save
|
|
|
|
'(desktop-missing-file-warning
|
|
|
|
tags-file-name
|
|
|
|
tags-table-list
|
|
|
|
search-ring
|
|
|
|
regexp-search-ring
|
2009-07-05 05:14:27 +00:00
|
|
|
register-alist
|
|
|
|
file-name-history)
|
2004-04-16 12:51:06 +00:00
|
|
|
"List of global variables saved by `desktop-save'.
|
|
|
|
An element may be variable name (a symbol) or a cons cell of the form
|
|
|
|
\(VAR . MAX-SIZE), which means to truncate VAR's value to at most
|
|
|
|
MAX-SIZE elements (if the value is a list) before saving the value.
|
2003-04-09 01:37:56 +00:00
|
|
|
Feature: Saving `kill-ring' implies saving `kill-ring-yank-pointer'."
|
|
|
|
:type '(repeat (restricted-sexp :match-alternatives (symbolp consp)))
|
|
|
|
:group 'desktop)
|
|
|
|
|
2004-09-09 19:45:03 +00:00
|
|
|
(defcustom desktop-globals-to-clear
|
|
|
|
'(kill-ring
|
|
|
|
kill-ring-yank-pointer
|
|
|
|
search-ring
|
|
|
|
search-ring-yank-pointer
|
|
|
|
regexp-search-ring
|
|
|
|
regexp-search-ring-yank-pointer)
|
2005-01-07 20:18:59 +00:00
|
|
|
"List of global variables that `desktop-clear' will clear.
|
2003-04-09 01:37:56 +00:00
|
|
|
An element may be variable name (a symbol) or a cons cell of the form
|
2005-07-19 09:54:05 +00:00
|
|
|
\(VAR . FORM). Symbols are set to nil and for cons cells VAR is set
|
|
|
|
to the value obtained by evaluating FORM."
|
2003-04-09 01:37:56 +00:00
|
|
|
:type '(repeat (restricted-sexp :match-alternatives (symbolp consp)))
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
(defcustom desktop-clear-preserve-buffers
|
2010-06-02 23:07:39 -07:00
|
|
|
'("\\*scratch\\*" "\\*Messages\\*" "\\*server\\*" "\\*tramp/.+\\*"
|
|
|
|
"\\*Warnings\\*")
|
2008-12-03 05:48:14 +00:00
|
|
|
"List of buffers that `desktop-clear' should not delete.
|
2005-08-10 19:38:52 +00:00
|
|
|
Each element is a regular expression. Buffers with a name matched by any of
|
|
|
|
these won't be deleted."
|
2010-06-02 23:07:39 -07:00
|
|
|
:version "23.3" ; added Warnings - bug#6336
|
2003-04-09 01:37:56 +00:00
|
|
|
:type '(repeat string)
|
|
|
|
:group 'desktop)
|
2004-09-09 19:45:03 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
;;;###autoload
|
2004-09-09 19:45:03 +00:00
|
|
|
(defcustom desktop-locals-to-save
|
|
|
|
'(desktop-locals-to-save ; Itself! Think it over.
|
|
|
|
truncate-lines
|
|
|
|
case-fold-search
|
|
|
|
case-replace
|
|
|
|
fill-column
|
|
|
|
overwrite-mode
|
|
|
|
change-log-default-name
|
|
|
|
line-number-mode
|
2005-08-10 19:38:52 +00:00
|
|
|
column-number-mode
|
|
|
|
size-indication-mode
|
|
|
|
buffer-file-coding-system
|
|
|
|
indent-tabs-mode
|
2006-06-23 21:42:33 +00:00
|
|
|
tab-width
|
2005-08-10 19:38:52 +00:00
|
|
|
indicate-buffer-boundaries
|
|
|
|
indicate-empty-lines
|
|
|
|
show-trailing-whitespace)
|
1995-01-30 06:14:33 +00:00
|
|
|
"List of local variables to save for each buffer.
|
2005-08-10 19:38:52 +00:00
|
|
|
The variables are saved only when they really are local. Conventional minor
|
|
|
|
modes are restored automatically; they should not be listed here."
|
2004-04-16 12:51:06 +00:00
|
|
|
:type '(repeat symbol)
|
|
|
|
:group 'desktop)
|
1994-01-06 11:34:51 +00:00
|
|
|
|
2009-07-18 11:59:04 +00:00
|
|
|
(defcustom desktop-buffers-not-to-save nil
|
2003-04-09 01:37:56 +00:00
|
|
|
"Regexp identifying buffers that are to be excluded from saving."
|
2009-07-18 11:59:04 +00:00
|
|
|
:type '(choice (const :tag "None" nil)
|
|
|
|
regexp)
|
2009-07-19 00:33:58 +00:00
|
|
|
:version "23.2" ; set to nil
|
2003-04-09 01:37:56 +00:00
|
|
|
:group 'desktop)
|
1993-06-01 20:09:25 +00:00
|
|
|
|
2004-04-16 12:51:06 +00:00
|
|
|
;; Skip tramp and ange-ftp files
|
1997-04-12 03:18:33 +00:00
|
|
|
(defcustom desktop-files-not-to-save
|
2009-07-18 11:59:04 +00:00
|
|
|
"\\(^/[^/:]*:\\|(ftp)$\\)"
|
1997-04-12 03:18:33 +00:00
|
|
|
"Regexp identifying files whose buffers are to be excluded from saving."
|
2009-07-18 11:59:04 +00:00
|
|
|
:type '(choice (const :tag "None" nil)
|
|
|
|
regexp)
|
1997-04-12 03:18:33 +00:00
|
|
|
:group 'desktop)
|
1994-05-01 02:07:27 +00:00
|
|
|
|
2005-10-21 08:48:53 +00:00
|
|
|
;; We skip TAGS files to save time (tags-file-name is saved instead).
|
|
|
|
(defcustom desktop-modes-not-to-save
|
|
|
|
'(tags-table-mode)
|
2000-10-10 16:42:37 +00:00
|
|
|
"List of major modes whose buffers should not be saved."
|
|
|
|
:type '(repeat symbol)
|
|
|
|
:group 'desktop)
|
|
|
|
|
2013-07-15 02:07:51 +02:00
|
|
|
(defcustom desktop-restore-frames t
|
2013-06-27 11:08:14 +02:00
|
|
|
"When non-nil, save window/frame configuration to desktop file."
|
|
|
|
:type 'boolean
|
|
|
|
:group 'desktop
|
|
|
|
:version "24.4")
|
|
|
|
|
2013-07-01 05:29:46 +02:00
|
|
|
(defcustom desktop-restore-in-current-display nil
|
2013-07-15 02:07:51 +02:00
|
|
|
"If t, frames are restored in the current display.
|
|
|
|
If nil, frames are restored, if possible, in their original displays.
|
|
|
|
If `delete', frames on other displays are deleted instead of restored."
|
|
|
|
:type '(choice (const :tag "Restore in current display" t)
|
|
|
|
(const :tag "Restore in original display" nil)
|
|
|
|
(const :tag "Delete frames in other displays" 'delete))
|
|
|
|
:group 'desktop
|
|
|
|
:version "24.4")
|
|
|
|
|
|
|
|
(defcustom desktop-restoring-reuses-frames t
|
|
|
|
"If t, restoring frames reuses existing frames.
|
|
|
|
If nil, existing frames are deleted.
|
|
|
|
If `keep', existing frames are kept and not reused."
|
|
|
|
:type '(choice (const :tag "Reuse existing frames" t)
|
|
|
|
(const :tag "Delete existing frames" nil)
|
|
|
|
(const :tag "Keep existing frames" 'keep))
|
2013-07-01 05:29:46 +02:00
|
|
|
:group 'desktop
|
|
|
|
:version "24.4")
|
|
|
|
|
2013-07-22 03:25:47 +02:00
|
|
|
(defcustom desktop-before-saving-frames-functions nil
|
|
|
|
"Abnormal hook run before saving frames.
|
|
|
|
Functions in this hook are called with one argument, a live frame."
|
|
|
|
:type 'hook
|
|
|
|
:group 'desktop
|
|
|
|
:version "24.4")
|
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
(defcustom desktop-file-name-format 'absolute
|
2008-12-03 05:48:14 +00:00
|
|
|
"Format in which desktop file names should be saved.
|
2003-04-09 01:37:56 +00:00
|
|
|
Possible values are:
|
|
|
|
absolute -- Absolute file name.
|
|
|
|
tilde -- Relative to ~.
|
|
|
|
local -- Relative to directory of desktop file."
|
|
|
|
:type '(choice (const absolute) (const tilde) (const local))
|
2004-11-11 19:02:56 +00:00
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
1996-03-05 16:37:03 +00:00
|
|
|
|
2005-01-07 20:18:59 +00:00
|
|
|
(defcustom desktop-restore-eager t
|
|
|
|
"Number of buffers to restore immediately.
|
|
|
|
Remaining buffers are restored lazily (when Emacs is idle).
|
|
|
|
If value is t, all buffers are restored immediately."
|
2005-01-09 13:31:53 +00:00
|
|
|
:type '(choice (const t) integer)
|
2005-01-07 20:18:59 +00:00
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2005-01-07 20:18:59 +00:00
|
|
|
|
|
|
|
(defcustom desktop-lazy-verbose t
|
|
|
|
"Verbose reporting of lazily created buffers."
|
|
|
|
:type 'boolean
|
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2005-01-07 20:18:59 +00:00
|
|
|
|
|
|
|
(defcustom desktop-lazy-idle-delay 5
|
|
|
|
"Idle delay before starting to create buffers.
|
|
|
|
See `desktop-restore-eager'."
|
|
|
|
:type 'integer
|
|
|
|
:group 'desktop
|
2005-02-09 15:50:47 +00:00
|
|
|
:version "22.1")
|
2005-01-07 20:18:59 +00:00
|
|
|
|
2004-04-21 20:53:35 +00:00
|
|
|
;;;###autoload
|
2013-07-12 13:21:01 +02:00
|
|
|
(defvar-local desktop-save-buffer nil
|
2004-04-26 18:58:19 +00:00
|
|
|
"When non-nil, save buffer status in desktop file.
|
|
|
|
|
2005-07-29 10:25:15 +00:00
|
|
|
If the value is a function, it is called by `desktop-save' with argument
|
|
|
|
DESKTOP-DIRNAME to obtain auxiliary information to save in the desktop
|
2004-04-26 18:58:19 +00:00
|
|
|
file along with the state of the buffer for which it was called.
|
2000-10-13 09:11:16 +00:00
|
|
|
|
2004-04-16 12:51:06 +00:00
|
|
|
When file names are returned, they should be formatted using the call
|
2004-04-21 20:53:35 +00:00
|
|
|
\"(desktop-file-name FILE-NAME DESKTOP-DIRNAME)\".
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
Later, when `desktop-read' evaluates the desktop file, auxiliary information
|
|
|
|
is passed as the argument DESKTOP-BUFFER-MISC to functions in
|
|
|
|
`desktop-buffer-mode-handlers'.")
|
2004-04-26 18:58:19 +00:00
|
|
|
(make-obsolete-variable 'desktop-buffer-modes-to-save
|
2005-07-19 09:54:05 +00:00
|
|
|
'desktop-save-buffer "22.1")
|
2004-04-21 20:53:35 +00:00
|
|
|
(make-obsolete-variable 'desktop-buffer-misc-functions
|
2005-07-19 09:54:05 +00:00
|
|
|
'desktop-save-buffer "22.1")
|
2000-10-13 09:11:16 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
;;;###autoload
|
2012-06-21 15:15:42 +08:00
|
|
|
(defvar desktop-buffer-mode-handlers nil
|
2004-04-21 20:53:35 +00:00
|
|
|
"Alist of major mode specific functions to restore a desktop buffer.
|
2005-08-10 19:38:52 +00:00
|
|
|
Functions listed are called by `desktop-create-buffer' when `desktop-read'
|
|
|
|
evaluates the desktop file. List elements must have the form
|
|
|
|
|
|
|
|
(MAJOR-MODE . RESTORE-BUFFER-FUNCTION).
|
2004-04-21 20:53:35 +00:00
|
|
|
|
|
|
|
Buffers with a major mode not specified here, are restored by the default
|
|
|
|
handler `desktop-restore-file-buffer'.
|
|
|
|
|
2004-05-07 17:58:24 +00:00
|
|
|
Handlers are called with argument list
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2004-05-07 19:41:11 +00:00
|
|
|
(DESKTOP-BUFFER-FILE-NAME DESKTOP-BUFFER-NAME DESKTOP-BUFFER-MISC)
|
2004-04-21 20:53:35 +00:00
|
|
|
|
|
|
|
Furthermore, they may use the following variables:
|
2003-04-09 01:37:56 +00:00
|
|
|
|
|
|
|
desktop-file-version
|
|
|
|
desktop-buffer-major-mode
|
|
|
|
desktop-buffer-minor-modes
|
|
|
|
desktop-buffer-point
|
|
|
|
desktop-buffer-mark
|
|
|
|
desktop-buffer-read-only
|
|
|
|
desktop-buffer-locals
|
|
|
|
|
2004-04-21 20:53:35 +00:00
|
|
|
If a handler returns a buffer, then the saved mode settings
|
2005-08-10 19:38:52 +00:00
|
|
|
and variable values for that buffer are copied into it.
|
|
|
|
|
|
|
|
Modules that define a major mode that needs a special handler should contain
|
|
|
|
code like
|
1994-01-06 11:34:51 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
(defun foo-restore-desktop-buffer
|
|
|
|
...
|
|
|
|
(add-to-list 'desktop-buffer-mode-handlers
|
|
|
|
'(foo-mode . foo-restore-desktop-buffer))
|
|
|
|
|
|
|
|
Furthermore the major mode function must be autoloaded.")
|
1994-01-06 11:34:51 +00:00
|
|
|
|
2005-08-14 23:55:03 +00:00
|
|
|
;;;###autoload
|
2004-04-21 20:53:35 +00:00
|
|
|
(put 'desktop-buffer-mode-handlers 'risky-local-variable t)
|
|
|
|
(make-obsolete-variable 'desktop-buffer-handlers
|
2005-07-19 09:54:05 +00:00
|
|
|
'desktop-buffer-mode-handlers "22.1")
|
2000-10-13 09:11:16 +00:00
|
|
|
|
2000-04-26 17:33:02 +00:00
|
|
|
(defcustom desktop-minor-mode-table
|
|
|
|
'((auto-fill-function auto-fill-mode)
|
2005-12-14 07:44:44 +00:00
|
|
|
(vc-mode nil)
|
2007-10-30 00:50:07 +00:00
|
|
|
(vc-dired-mode nil)
|
2008-04-28 08:06:51 +00:00
|
|
|
(erc-track-minor-mode nil)
|
|
|
|
(savehist-mode nil))
|
2000-04-26 17:33:02 +00:00
|
|
|
"Table mapping minor mode variables to minor mode functions.
|
|
|
|
Each entry has the form (NAME RESTORE-FUNCTION).
|
|
|
|
NAME is the name of the buffer-local variable indicating that the minor
|
|
|
|
mode is active. RESTORE-FUNCTION is the function to activate the minor mode.
|
2007-06-09 00:09:03 +00:00
|
|
|
RESTORE-FUNCTION nil means don't try to restore the minor mode.
|
2000-04-26 17:33:02 +00:00
|
|
|
Only minor modes for which the name of the buffer-local variable
|
2004-05-31 21:45:51 +00:00
|
|
|
and the name of the minor mode function are different have to be added to
|
2005-08-10 19:38:52 +00:00
|
|
|
this table. See also `desktop-minor-mode-handlers'."
|
2000-04-26 17:33:02 +00:00
|
|
|
:type 'sexp
|
|
|
|
:group 'desktop)
|
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
;;;###autoload
|
2012-06-21 15:15:42 +08:00
|
|
|
(defvar desktop-minor-mode-handlers nil
|
2005-08-10 19:38:52 +00:00
|
|
|
"Alist of functions to restore non-standard minor modes.
|
|
|
|
Functions are called by `desktop-create-buffer' to restore minor modes.
|
|
|
|
List elements must have the form
|
|
|
|
|
|
|
|
(MINOR-MODE . RESTORE-FUNCTION).
|
|
|
|
|
|
|
|
Minor modes not specified here, are restored by the standard minor mode
|
|
|
|
function.
|
|
|
|
|
|
|
|
Handlers are called with argument list
|
|
|
|
|
|
|
|
(DESKTOP-BUFFER-LOCALS)
|
|
|
|
|
|
|
|
Furthermore, they may use the following variables:
|
|
|
|
|
|
|
|
desktop-file-version
|
|
|
|
desktop-buffer-file-name
|
|
|
|
desktop-buffer-name
|
|
|
|
desktop-buffer-major-mode
|
|
|
|
desktop-buffer-minor-modes
|
|
|
|
desktop-buffer-point
|
|
|
|
desktop-buffer-mark
|
|
|
|
desktop-buffer-read-only
|
|
|
|
desktop-buffer-misc
|
|
|
|
|
|
|
|
When a handler is called, the buffer has been created and the major mode has
|
|
|
|
been set, but local variables listed in desktop-buffer-locals has not yet been
|
|
|
|
created and set.
|
|
|
|
|
|
|
|
Modules that define a minor mode that needs a special handler should contain
|
|
|
|
code like
|
|
|
|
|
|
|
|
(defun foo-desktop-restore
|
|
|
|
...
|
|
|
|
(add-to-list 'desktop-minor-mode-handlers
|
|
|
|
'(foo-mode . foo-desktop-restore))
|
|
|
|
|
|
|
|
Furthermore the minor mode function must be autoloaded.
|
|
|
|
|
|
|
|
See also `desktop-minor-mode-table'.")
|
|
|
|
|
2005-08-14 23:55:03 +00:00
|
|
|
;;;###autoload
|
2005-08-10 19:38:52 +00:00
|
|
|
(put 'desktop-minor-mode-handlers 'risky-local-variable t)
|
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
(defvar desktop-dirname nil
|
2004-04-16 12:51:06 +00:00
|
|
|
"The directory in which the desktop file should be saved.")
|
1993-06-01 20:09:25 +00:00
|
|
|
|
2006-06-23 21:42:33 +00:00
|
|
|
(defun desktop-full-file-name (&optional dirname)
|
|
|
|
"Return the full name of the desktop file in DIRNAME.
|
|
|
|
DIRNAME omitted or nil means use `desktop-dirname'."
|
|
|
|
(expand-file-name desktop-base-file-name (or dirname desktop-dirname)))
|
|
|
|
|
2007-06-12 09:11:31 +00:00
|
|
|
(defun desktop-full-lock-name (&optional dirname)
|
|
|
|
"Return the full name of the desktop lock file in DIRNAME.
|
|
|
|
DIRNAME omitted or nil means use `desktop-dirname'."
|
|
|
|
(expand-file-name desktop-base-lock-name (or dirname desktop-dirname)))
|
|
|
|
|
1993-06-01 20:09:25 +00:00
|
|
|
(defconst desktop-header
|
1993-10-07 16:50:26 +00:00
|
|
|
";; --------------------------------------------------------------------------
|
|
|
|
;; Desktop File for Emacs
|
|
|
|
;; --------------------------------------------------------------------------
|
1993-06-01 20:09:25 +00:00
|
|
|
" "*Header to place in Desktop file.")
|
1994-02-05 02:23:44 +00:00
|
|
|
|
|
|
|
(defvar desktop-delay-hook nil
|
|
|
|
"Hooks run after all buffers are loaded; intended for internal use.")
|
1997-12-29 20:58:19 +00:00
|
|
|
|
2013-04-27 23:55:00 +03:00
|
|
|
(defvar desktop-file-checksum nil
|
|
|
|
"Checksum of the last auto-saved contents of the desktop file.
|
|
|
|
Used to avoid writing contents unchanged between auto-saves.")
|
|
|
|
|
2013-07-21 19:45:12 +02:00
|
|
|
(defvar desktop-saved-frame-states nil
|
|
|
|
"Saved state of all frames.
|
|
|
|
Only valid during frame saving & restoring; intended for internal use.")
|
2013-06-27 11:08:14 +02:00
|
|
|
|
2007-06-12 09:11:31 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
;; Desktop file conflict detection
|
|
|
|
(defvar desktop-file-modtime nil
|
|
|
|
"When the desktop file was last modified to the knowledge of this Emacs.
|
|
|
|
Used to detect desktop file conflicts.")
|
|
|
|
|
|
|
|
(defun desktop-owner (&optional dirname)
|
|
|
|
"Return the PID of the Emacs process that owns the desktop file in DIRNAME.
|
|
|
|
Return nil if no desktop file found or no Emacs process is using it.
|
|
|
|
DIRNAME omitted or nil means use `desktop-dirname'."
|
2013-07-12 13:21:01 +02:00
|
|
|
(let (owner
|
|
|
|
(file (desktop-full-lock-name dirname)))
|
|
|
|
(and (file-exists-p file)
|
|
|
|
(ignore-errors
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert-file-contents-literally file)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(setq owner (read (current-buffer)))
|
|
|
|
(integerp owner)))
|
2007-06-12 09:11:31 +00:00
|
|
|
owner)))
|
|
|
|
|
|
|
|
(defun desktop-claim-lock (&optional dirname)
|
|
|
|
"Record this Emacs process as the owner of the desktop file in DIRNAME.
|
|
|
|
DIRNAME omitted or nil means use `desktop-dirname'."
|
|
|
|
(write-region (number-to-string (emacs-pid)) nil
|
|
|
|
(desktop-full-lock-name dirname)))
|
|
|
|
|
|
|
|
(defun desktop-release-lock (&optional dirname)
|
|
|
|
"Remove the lock file for the desktop in DIRNAME.
|
|
|
|
DIRNAME omitted or nil means use `desktop-dirname'."
|
|
|
|
(let ((file (desktop-full-lock-name dirname)))
|
|
|
|
(when (file-exists-p file) (delete-file file))))
|
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2004-04-16 12:51:06 +00:00
|
|
|
(defun desktop-truncate (list n)
|
1994-01-06 11:34:51 +00:00
|
|
|
"Truncate LIST to at most N elements destructively."
|
2004-04-16 12:51:06 +00:00
|
|
|
(let ((here (nthcdr (1- n) list)))
|
2007-06-12 11:14:52 +00:00
|
|
|
(when (consp here)
|
|
|
|
(setcdr here nil))))
|
1997-07-04 00:13:36 +00:00
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2006-06-23 21:42:33 +00:00
|
|
|
;;;###autoload
|
1997-07-04 00:13:36 +00:00
|
|
|
(defun desktop-clear ()
|
|
|
|
"Empty the Desktop.
|
2005-08-10 19:38:52 +00:00
|
|
|
This kills all buffers except for internal ones and those with names matched by
|
|
|
|
a regular expression in the list `desktop-clear-preserve-buffers'.
|
|
|
|
Furthermore, it clears the variables listed in `desktop-globals-to-clear'."
|
1993-06-01 20:09:25 +00:00
|
|
|
(interactive)
|
2005-01-07 20:18:59 +00:00
|
|
|
(desktop-lazy-abort)
|
2003-04-09 01:37:56 +00:00
|
|
|
(dolist (var desktop-globals-to-clear)
|
|
|
|
(if (symbolp var)
|
2007-06-12 11:14:52 +00:00
|
|
|
(eval `(setq-default ,var nil))
|
2003-04-09 01:37:56 +00:00
|
|
|
(eval `(setq-default ,(car var) ,(cdr var)))))
|
2013-07-23 03:10:54 +02:00
|
|
|
(let ((preserve-regexp (concat "^\\("
|
2005-08-10 19:38:52 +00:00
|
|
|
(mapconcat (lambda (regexp)
|
|
|
|
(concat "\\(" regexp "\\)"))
|
|
|
|
desktop-clear-preserve-buffers
|
|
|
|
"\\|")
|
|
|
|
"\\)$")))
|
2013-07-23 03:10:54 +02:00
|
|
|
(dolist (buffer (buffer-list))
|
|
|
|
(let ((bufname (buffer-name buffer)))
|
|
|
|
(unless (or (eq (aref bufname 0) ?s) ;; Don't kill internal buffers
|
|
|
|
(string-match-p preserve-regexp bufname))
|
|
|
|
(kill-buffer buffer)))))
|
1993-12-23 04:59:47 +00:00
|
|
|
(delete-other-windows))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2011-03-06 01:30:16 +01:00
|
|
|
(unless noninteractive
|
|
|
|
(add-hook 'kill-emacs-hook 'desktop-kill))
|
1994-01-06 11:34:51 +00:00
|
|
|
|
1993-06-01 20:09:25 +00:00
|
|
|
(defun desktop-kill ()
|
2004-04-16 12:51:06 +00:00
|
|
|
"If `desktop-save-mode' is non-nil, do what `desktop-save' says to do.
|
2003-04-09 01:37:56 +00:00
|
|
|
If the desktop should be saved and `desktop-dirname'
|
|
|
|
is nil, ask the user where to save the desktop."
|
2006-06-23 21:42:33 +00:00
|
|
|
(when (and desktop-save-mode
|
|
|
|
(let ((exists (file-exists-p (desktop-full-file-name))))
|
|
|
|
(or (eq desktop-save t)
|
2010-04-27 08:45:43 +02:00
|
|
|
(and exists (eq desktop-save 'if-exists))
|
|
|
|
;; If it exists, but we aren't using it, we are going
|
|
|
|
;; to ask for a new directory below.
|
|
|
|
(and exists desktop-dirname (eq desktop-save 'ask-if-new))
|
2006-06-23 21:42:33 +00:00
|
|
|
(and
|
|
|
|
(or (memq desktop-save '(ask ask-if-new))
|
|
|
|
(and exists (eq desktop-save 'ask-if-exists)))
|
|
|
|
(y-or-n-p "Save desktop? ")))))
|
2003-04-09 01:37:56 +00:00
|
|
|
(unless desktop-dirname
|
|
|
|
(setq desktop-dirname
|
2006-06-23 21:42:33 +00:00
|
|
|
(file-name-as-directory
|
|
|
|
(expand-file-name
|
2007-06-20 10:12:44 +00:00
|
|
|
(read-directory-name "Directory for desktop file: " nil nil t)))))
|
2003-04-09 01:37:56 +00:00
|
|
|
(condition-case err
|
2007-06-12 09:11:31 +00:00
|
|
|
(desktop-save desktop-dirname t)
|
2003-04-09 01:37:56 +00:00
|
|
|
(file-error
|
2006-06-23 21:42:33 +00:00
|
|
|
(unless (yes-or-no-p "Error while saving the desktop. Ignore? ")
|
2007-06-12 09:11:31 +00:00
|
|
|
(signal (car err) (cdr err))))))
|
|
|
|
;; If we own it, we don't anymore.
|
|
|
|
(when (eq (emacs-pid) (desktop-owner)) (desktop-release-lock)))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
1996-03-05 04:22:44 +00:00
|
|
|
(defun desktop-list* (&rest args)
|
2013-07-23 03:10:54 +02:00
|
|
|
(and args (cl-list* args)))
|
1996-03-05 04:22:44 +00:00
|
|
|
|
2007-06-12 09:11:31 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
(defun desktop-buffer-info (buffer)
|
|
|
|
(set-buffer buffer)
|
|
|
|
(list
|
2008-02-14 14:37:00 +00:00
|
|
|
;; base name of the buffer; replaces the buffer name if managed by uniquify
|
|
|
|
(and (fboundp 'uniquify-buffer-base-name) (uniquify-buffer-base-name))
|
2007-06-12 09:11:31 +00:00
|
|
|
;; basic information
|
2007-07-12 22:40:00 +00:00
|
|
|
(desktop-file-name (buffer-file-name) desktop-dirname)
|
2008-02-14 14:37:00 +00:00
|
|
|
(buffer-name)
|
2007-06-12 09:11:31 +00:00
|
|
|
major-mode
|
|
|
|
;; minor modes
|
|
|
|
(let (ret)
|
|
|
|
(mapc
|
|
|
|
#'(lambda (minor-mode)
|
|
|
|
(and (boundp minor-mode)
|
|
|
|
(symbol-value minor-mode)
|
|
|
|
(let* ((special (assq minor-mode desktop-minor-mode-table))
|
|
|
|
(value (cond (special (cadr special))
|
|
|
|
((functionp minor-mode) minor-mode))))
|
|
|
|
(when value (add-to-list 'ret value)))))
|
|
|
|
(mapcar #'car minor-mode-alist))
|
|
|
|
ret)
|
|
|
|
;; point and mark, and read-only status
|
|
|
|
(point)
|
|
|
|
(list (mark t) mark-active)
|
|
|
|
buffer-read-only
|
|
|
|
;; auxiliary information
|
|
|
|
(when (functionp desktop-save-buffer)
|
2007-07-12 22:40:00 +00:00
|
|
|
(funcall desktop-save-buffer desktop-dirname))
|
2007-06-12 09:11:31 +00:00
|
|
|
;; local variables
|
2013-07-23 03:10:54 +02:00
|
|
|
(let ((loclist (buffer-local-variables))
|
|
|
|
(ll nil))
|
|
|
|
(dolist (local desktop-locals-to-save)
|
|
|
|
(let ((here (assq local loclist)))
|
|
|
|
(cond (here
|
|
|
|
(push here ll))
|
|
|
|
((member local loclist)
|
|
|
|
(push local ll)))))
|
2007-06-12 09:11:31 +00:00
|
|
|
ll)))
|
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2013-03-25 23:38:18 -04:00
|
|
|
(defun desktop--v2s (value)
|
|
|
|
"Convert VALUE to a pair (QUOTE . SEXP); (eval SEXP) gives VALUE.
|
|
|
|
SEXP is an sexp that when evaluated yields VALUE.
|
1995-01-30 06:14:33 +00:00
|
|
|
QUOTE may be `may' (value may be quoted),
|
2011-12-23 01:44:27 +01:00
|
|
|
`must' (value must be quoted), or nil (value must not be quoted)."
|
1994-02-05 02:23:44 +00:00
|
|
|
(cond
|
2007-06-12 11:14:52 +00:00
|
|
|
((or (numberp value) (null value) (eq t value) (keywordp value))
|
2013-03-25 23:38:18 -04:00
|
|
|
(cons 'may value))
|
2007-06-12 11:14:52 +00:00
|
|
|
((stringp value)
|
|
|
|
(let ((copy (copy-sequence value)))
|
|
|
|
(set-text-properties 0 (length copy) nil copy)
|
2013-03-25 23:38:18 -04:00
|
|
|
;; Get rid of text properties because we cannot read them.
|
|
|
|
(cons 'may copy)))
|
2007-06-12 11:14:52 +00:00
|
|
|
((symbolp value)
|
2013-03-25 23:38:18 -04:00
|
|
|
(cons 'must value))
|
2007-06-12 11:14:52 +00:00
|
|
|
((vectorp value)
|
2013-03-25 23:38:18 -04:00
|
|
|
(let* ((pass1 (mapcar #'desktop--v2s value))
|
|
|
|
(special (assq nil pass1)))
|
2007-06-12 11:14:52 +00:00
|
|
|
(if special
|
2013-03-25 23:38:18 -04:00
|
|
|
(cons nil `(vector
|
|
|
|
,@(mapcar (lambda (el)
|
|
|
|
(if (eq (car el) 'must)
|
|
|
|
`',(cdr el) (cdr el)))
|
|
|
|
pass1)))
|
|
|
|
(cons 'may `[,@(mapcar #'cdr pass1)]))))
|
2007-06-12 11:14:52 +00:00
|
|
|
((consp value)
|
|
|
|
(let ((p value)
|
|
|
|
newlist
|
2013-07-12 13:21:01 +02:00
|
|
|
use-list*)
|
2007-06-12 11:14:52 +00:00
|
|
|
(while (consp p)
|
2013-03-25 23:38:18 -04:00
|
|
|
(let ((q.sexp (desktop--v2s (car p))))
|
|
|
|
(push q.sexp newlist))
|
2007-06-12 11:14:52 +00:00
|
|
|
(setq p (cdr p)))
|
2013-03-25 23:38:18 -04:00
|
|
|
(when p
|
|
|
|
(let ((last (desktop--v2s p)))
|
|
|
|
(setq use-list* t)
|
|
|
|
(push last newlist)))
|
|
|
|
(if (assq nil newlist)
|
2007-06-12 11:14:52 +00:00
|
|
|
(cons nil
|
2013-03-25 23:38:18 -04:00
|
|
|
`(,(if use-list* 'desktop-list* 'list)
|
|
|
|
,@(mapcar (lambda (el)
|
|
|
|
(if (eq (car el) 'must)
|
|
|
|
`',(cdr el) (cdr el)))
|
|
|
|
(nreverse newlist))))
|
2007-06-12 11:14:52 +00:00
|
|
|
(cons 'must
|
2013-03-25 23:38:18 -04:00
|
|
|
`(,@(mapcar #'cdr
|
|
|
|
(nreverse (if use-list* (cdr newlist) newlist)))
|
|
|
|
,@(if use-list* (cdar newlist)))))))
|
2007-06-12 11:14:52 +00:00
|
|
|
((subrp value)
|
2013-03-25 23:38:18 -04:00
|
|
|
(cons nil `(symbol-function
|
|
|
|
',(intern-soft (substring (prin1-to-string value) 7 -1)))))
|
2007-06-12 11:14:52 +00:00
|
|
|
((markerp value)
|
2013-03-25 23:38:18 -04:00
|
|
|
(let ((pos (marker-position value))
|
|
|
|
(buf (buffer-name (marker-buffer value))))
|
|
|
|
(cons nil
|
|
|
|
`(let ((mk (make-marker)))
|
|
|
|
(add-hook 'desktop-delay-hook
|
|
|
|
`(lambda ()
|
|
|
|
(set-marker ,mk ,,pos (get-buffer ,,buf))))
|
|
|
|
mk))))
|
|
|
|
(t ; Save as text.
|
|
|
|
(cons 'may "Unprintable entity"))))
|
1994-02-05 02:23:44 +00:00
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2004-04-16 12:51:06 +00:00
|
|
|
(defun desktop-value-to-string (value)
|
1995-01-30 06:14:33 +00:00
|
|
|
"Convert VALUE to a string that when read evaluates to the same value.
|
|
|
|
Not all types of values are supported."
|
1994-02-05 02:23:44 +00:00
|
|
|
(let* ((print-escape-newlines t)
|
|
|
|
(float-output-format nil)
|
2013-03-25 23:38:18 -04:00
|
|
|
(quote.sexp (desktop--v2s value))
|
|
|
|
(quote (car quote.sexp))
|
|
|
|
(txt
|
|
|
|
(let ((print-quoted t))
|
|
|
|
(prin1-to-string (cdr quote.sexp)))))
|
1994-02-05 02:23:44 +00:00
|
|
|
(if (eq quote 'must)
|
|
|
|
(concat "'" txt)
|
|
|
|
txt)))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1994-01-06 11:34:51 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
1995-10-08 19:37:04 +00:00
|
|
|
(defun desktop-outvar (varspec)
|
|
|
|
"Output a setq statement for variable VAR to the desktop file.
|
|
|
|
The argument VARSPEC may be the variable name VAR (a symbol),
|
2006-02-10 11:07:50 +00:00
|
|
|
or a cons cell of the form (VAR . MAX-SIZE),
|
1995-10-08 19:37:04 +00:00
|
|
|
which means to truncate VAR's value to at most MAX-SIZE elements
|
|
|
|
\(if the value is a list) before saving the value."
|
|
|
|
(let (var size)
|
|
|
|
(if (consp varspec)
|
|
|
|
(setq var (car varspec) size (cdr varspec))
|
|
|
|
(setq var varspec))
|
2007-06-12 11:14:52 +00:00
|
|
|
(when (boundp var)
|
|
|
|
(when (and (integerp size)
|
|
|
|
(> size 0)
|
|
|
|
(listp (eval var)))
|
|
|
|
(desktop-truncate (eval var) size))
|
|
|
|
(insert "(setq "
|
|
|
|
(symbol-name var)
|
|
|
|
" "
|
|
|
|
(desktop-value-to-string (symbol-value var))
|
|
|
|
")\n"))))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2011-04-19 15:44:55 +02:00
|
|
|
(defun desktop-save-buffer-p (filename bufname mode &rest _dummy)
|
2004-04-26 18:58:19 +00:00
|
|
|
"Return t if buffer should have its state saved in the desktop file.
|
1993-10-07 16:50:26 +00:00
|
|
|
FILENAME is the visited file name, BUFNAME is the buffer name, and
|
2006-02-10 11:07:50 +00:00
|
|
|
MODE is the major mode.
|
|
|
|
\n\(fn FILENAME BUFNAME MODE)"
|
2010-03-24 23:18:17 -07:00
|
|
|
(let ((case-fold-search nil)
|
|
|
|
dired-skip)
|
2009-07-18 11:59:04 +00:00
|
|
|
(and (not (and (stringp desktop-buffers-not-to-save)
|
|
|
|
(not filename)
|
2013-07-12 13:21:01 +02:00
|
|
|
(string-match-p desktop-buffers-not-to-save bufname)))
|
2004-04-26 18:58:19 +00:00
|
|
|
(not (memq mode desktop-modes-not-to-save))
|
2010-03-24 23:18:17 -07:00
|
|
|
;; FIXME this is broken if desktop-files-not-to-save is nil.
|
2004-04-26 18:58:19 +00:00
|
|
|
(or (and filename
|
2009-07-18 11:59:04 +00:00
|
|
|
(stringp desktop-files-not-to-save)
|
2013-07-12 13:21:01 +02:00
|
|
|
(not (string-match-p desktop-files-not-to-save filename)))
|
2012-11-28 20:16:51 -05:00
|
|
|
(and (memq mode '(dired-mode vc-dir-mode))
|
2004-04-26 18:58:19 +00:00
|
|
|
(with-current-buffer bufname
|
2010-03-24 23:18:17 -07:00
|
|
|
(not (setq dired-skip
|
2013-07-12 13:21:01 +02:00
|
|
|
(string-match-p desktop-files-not-to-save
|
|
|
|
default-directory)))))
|
2004-04-26 18:58:19 +00:00
|
|
|
(and (null filename)
|
2010-03-24 23:18:17 -07:00
|
|
|
(null dired-skip) ; bug#5755
|
2004-04-26 18:58:19 +00:00
|
|
|
(with-current-buffer bufname desktop-save-buffer))))))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2003-04-09 01:37:56 +00:00
|
|
|
(defun desktop-file-name (filename dirname)
|
|
|
|
"Convert FILENAME to format specified in `desktop-file-name-format'.
|
|
|
|
DIRNAME must be the directory in which the desktop file will be saved."
|
|
|
|
(cond
|
|
|
|
((not filename) nil)
|
|
|
|
((eq desktop-file-name-format 'tilde)
|
|
|
|
(let ((relative-name (file-relative-name (expand-file-name filename) "~")))
|
|
|
|
(cond
|
|
|
|
((file-name-absolute-p relative-name) relative-name)
|
|
|
|
((string= "./" relative-name) "~/")
|
|
|
|
((string= "." relative-name) "~")
|
|
|
|
(t (concat "~/" relative-name)))))
|
|
|
|
((eq desktop-file-name-format 'local) (file-relative-name filename dirname))
|
|
|
|
(t (expand-file-name filename))))
|
2002-11-04 08:21:51 +00:00
|
|
|
|
2007-06-16 22:32:13 +00:00
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2013-07-15 02:07:51 +02:00
|
|
|
(defvar desktop-filter-parameters-alist
|
|
|
|
'((background-color . desktop--filter-*-color)
|
|
|
|
(buffer-list . t)
|
|
|
|
(buffer-predicate . t)
|
|
|
|
(buried-buffer-list . t)
|
|
|
|
(desktop-font . desktop--filter-restore-desktop-parm)
|
|
|
|
(desktop-fullscreen . desktop--filter-restore-desktop-parm)
|
|
|
|
(desktop-height . desktop--filter-restore-desktop-parm)
|
|
|
|
(desktop-width . desktop--filter-restore-desktop-parm)
|
|
|
|
(font . desktop--filter-save-desktop-parm)
|
|
|
|
(font-backend . t)
|
|
|
|
(foreground-color . desktop--filter-*-color)
|
|
|
|
(fullscreen . desktop--filter-save-desktop-parm)
|
|
|
|
(height . desktop--filter-save-desktop-parm)
|
2013-07-21 04:42:11 +02:00
|
|
|
(left . desktop--filter-iconified-position)
|
2013-07-15 02:07:51 +02:00
|
|
|
(minibuffer . desktop--filter-minibuffer)
|
|
|
|
(name . t)
|
|
|
|
(outer-window-id . t)
|
|
|
|
(parent-id . t)
|
2013-07-21 04:42:11 +02:00
|
|
|
(top . desktop--filter-iconified-position)
|
2013-07-15 02:07:51 +02:00
|
|
|
(tty . desktop--filter-tty*)
|
|
|
|
(tty-type . desktop--filter-tty*)
|
|
|
|
(width . desktop--filter-save-desktop-parm)
|
|
|
|
(window-id . t)
|
|
|
|
(window-system . t))
|
|
|
|
"Alist of frame parameters and filtering functions.
|
|
|
|
|
|
|
|
Each element is a cons (PARAM . FILTER), where PARAM is a parameter
|
|
|
|
name (a symbol identifying a frame parameter), and FILTER can be t
|
|
|
|
\(meaning the parameter is removed from the parameter list on saving
|
|
|
|
and restoring), or a function that will be called with three args:
|
|
|
|
|
|
|
|
CURRENT a cons (PARAM . VALUE), where PARAM is the one being
|
|
|
|
filtered and VALUE is its current value
|
|
|
|
PARAMETERS the complete alist of parameters being filtered
|
|
|
|
SAVING non-nil if filtering before saving state, nil otherwise
|
|
|
|
|
|
|
|
The FILTER function must return:
|
|
|
|
nil CURRENT is removed from the list
|
|
|
|
t CURRENT is left as is
|
|
|
|
(PARAM' . VALUE') replace CURRENT with this
|
|
|
|
|
|
|
|
Frame parameters not on this list are passed intact.")
|
|
|
|
|
|
|
|
(defvar desktop--target-display nil
|
|
|
|
"Either (minibuffer . VALUE) or nil.
|
|
|
|
This refers to the current frame config being processed inside
|
|
|
|
`frame--restore-frames' and its auxiliary functions (like filtering).
|
|
|
|
If nil, there is no need to change the display.
|
|
|
|
If non-nil, display parameter to use when creating the frame.
|
|
|
|
Internal use only.")
|
|
|
|
|
|
|
|
(defun desktop-switch-to-gui-p (parameters)
|
|
|
|
"True when switching to a graphic display.
|
|
|
|
Return t if PARAMETERS describes a text-only terminal and
|
|
|
|
the target is a graphic display; otherwise return nil.
|
|
|
|
Only meaningful when called from a filtering function in
|
|
|
|
`desktop-filter-parameters-alist'."
|
|
|
|
(and desktop--target-display ; we're switching
|
|
|
|
(null (cdr (assq 'display parameters))) ; from a tty
|
|
|
|
(cdr desktop--target-display))) ; to a GUI display
|
|
|
|
|
|
|
|
(defun desktop-switch-to-tty-p (parameters)
|
|
|
|
"True when switching to a text-only terminal.
|
|
|
|
Return t if PARAMETERS describes a graphic display and
|
|
|
|
the target is a text-only terminal; otherwise return nil.
|
|
|
|
Only meaningful when called from a filtering function in
|
|
|
|
`desktop-filter-parameters-alist'."
|
|
|
|
(and desktop--target-display ; we're switching
|
|
|
|
(cdr (assq 'display parameters)) ; from a GUI display
|
|
|
|
(null (cdr desktop--target-display)))) ; to a tty
|
|
|
|
|
|
|
|
(defun desktop--filter-tty* (_current parameters saving)
|
|
|
|
;; Remove tty and tty-type parameters when switching
|
|
|
|
;; to a GUI frame.
|
|
|
|
(or saving
|
|
|
|
(not (desktop-switch-to-gui-p parameters))))
|
|
|
|
|
|
|
|
(defun desktop--filter-*-color (current parameters saving)
|
|
|
|
;; Remove (foreground|background)-color parameters
|
|
|
|
;; when switching to a GUI frame if they denote an
|
|
|
|
;; "unspecified" color.
|
|
|
|
(or saving
|
|
|
|
(not (desktop-switch-to-gui-p parameters))
|
|
|
|
(not (stringp (cdr current)))
|
|
|
|
(not (string-match-p "^unspecified-[fb]g$" (cdr current)))))
|
|
|
|
|
|
|
|
(defun desktop--filter-minibuffer (current _parameters saving)
|
|
|
|
;; When minibuffer is a window, save it as minibuffer . t
|
|
|
|
(or (not saving)
|
|
|
|
(if (windowp (cdr current))
|
|
|
|
'(minibuffer . t)
|
|
|
|
t)))
|
|
|
|
|
|
|
|
(defun desktop--filter-restore-desktop-parm (current parameters saving)
|
|
|
|
;; When switching to a GUI frame, convert desktop-XXX parameter to XXX
|
|
|
|
(or saving
|
|
|
|
(not (desktop-switch-to-gui-p parameters))
|
|
|
|
(let ((val (cdr current)))
|
|
|
|
(if (eq val :desktop-processed)
|
|
|
|
nil
|
|
|
|
(cons (intern (substring (symbol-name (car current))
|
|
|
|
8)) ;; (length "desktop-")
|
|
|
|
val)))))
|
|
|
|
|
|
|
|
(defun desktop--filter-save-desktop-parm (current parameters saving)
|
|
|
|
;; When switching to a tty frame, save parameter XXX as desktop-XXX so it
|
|
|
|
;; can be restored in a subsequent GUI session, unless it already exists.
|
|
|
|
(cond (saving t)
|
|
|
|
((desktop-switch-to-tty-p parameters)
|
|
|
|
(let ((sym (intern (format "desktop-%s" (car current)))))
|
|
|
|
(if (assq sym parameters)
|
|
|
|
nil
|
|
|
|
(cons sym (cdr current)))))
|
|
|
|
((desktop-switch-to-gui-p parameters)
|
|
|
|
(let* ((dtp (assq (intern (format "desktop-%s" (car current)))
|
|
|
|
parameters))
|
|
|
|
(val (cdr dtp)))
|
|
|
|
(if (eq val :desktop-processed)
|
|
|
|
nil
|
|
|
|
(setcdr dtp :desktop-processed)
|
|
|
|
(cons (car current) val))))
|
|
|
|
(t t)))
|
|
|
|
|
2013-07-21 04:42:11 +02:00
|
|
|
(defun desktop--filter-iconified-position (_current parameters saving)
|
|
|
|
;; When saving an iconified frame, top & left are meaningless,
|
|
|
|
;; so remove them to allow restoring to a default position.
|
|
|
|
(not (and saving (eq (cdr (assq 'visibility parameters)) 'icon))))
|
|
|
|
|
2013-07-15 02:07:51 +02:00
|
|
|
(defun desktop-restore-in-original-display-p ()
|
|
|
|
"True if saved frames' displays should be honored."
|
|
|
|
(cond ((daemonp) t)
|
|
|
|
((eq system-type 'windows-nt) nil)
|
|
|
|
(t (null desktop-restore-in-current-display))))
|
|
|
|
|
|
|
|
(defun desktop--filter-frame-parms (parameters saving)
|
|
|
|
"Filter frame parameters and return filtered list.
|
|
|
|
PARAMETERS is a parameter alist as returned by `frame-parameters'.
|
|
|
|
If SAVING is non-nil, filtering is happening before saving frame state;
|
|
|
|
otherwise, filtering is being done before restoring frame state.
|
|
|
|
Parameters are filtered according to the setting of
|
|
|
|
`desktop-filter-parameters-alist' (which see).
|
2013-06-27 11:08:14 +02:00
|
|
|
Internal use only."
|
2013-07-15 02:07:51 +02:00
|
|
|
(let ((filtered nil))
|
|
|
|
(dolist (param parameters)
|
|
|
|
(let ((filter (cdr (assq (car param) desktop-filter-parameters-alist)))
|
|
|
|
this)
|
|
|
|
(cond (;; no filter: pass param
|
|
|
|
(null filter)
|
|
|
|
(push param filtered))
|
|
|
|
(;; filter = t; skip param
|
|
|
|
(eq filter t))
|
|
|
|
(;; filter func returns nil: skip param
|
|
|
|
(null (setq this (funcall filter param parameters saving))))
|
|
|
|
(;; filter func returns t: pass param
|
|
|
|
(eq this t)
|
|
|
|
(push param filtered))
|
|
|
|
(;; filter func returns a new param: use it
|
|
|
|
t
|
|
|
|
(push this filtered)))))
|
|
|
|
;; Set the display parameter after filtering, so that filter functions
|
|
|
|
;; have access to its original value.
|
|
|
|
(when desktop--target-display
|
|
|
|
(let ((display (assq 'display filtered)))
|
|
|
|
(if display
|
|
|
|
(setcdr display (cdr desktop--target-display))
|
|
|
|
(push desktop--target-display filtered))))
|
|
|
|
filtered))
|
|
|
|
|
2013-07-22 03:25:47 +02:00
|
|
|
(defun desktop--process-minibuffer-frames (frames)
|
2013-07-15 02:07:51 +02:00
|
|
|
;; Adds a desktop-mini parameter to frames
|
|
|
|
;; desktop-mini is a list (MINIBUFFER NUMBER DEFAULT?) where
|
|
|
|
;; MINIBUFFER t if the frame (including minibuffer-only) owns a minibuffer
|
|
|
|
;; NUMBER if MINIBUFFER = t, an ID for the frame; if nil, the ID of
|
|
|
|
;; the frame containing the minibuffer used by this frame
|
|
|
|
;; DEFAULT? if t, this frame is the value of default-minibuffer-frame
|
2013-07-22 03:25:47 +02:00
|
|
|
(let ((count 0))
|
2013-07-15 02:07:51 +02:00
|
|
|
;; Reset desktop-mini for all frames
|
2013-07-22 03:25:47 +02:00
|
|
|
(dolist (frame (frame-list))
|
2013-07-15 02:07:51 +02:00
|
|
|
(set-frame-parameter frame 'desktop-mini nil))
|
|
|
|
;; Number all frames with its own minibuffer
|
|
|
|
(dolist (frame (minibuffer-frame-list))
|
|
|
|
(set-frame-parameter frame 'desktop-mini
|
|
|
|
(list t
|
2013-07-22 03:25:47 +02:00
|
|
|
(cl-incf count)
|
2013-07-15 02:07:51 +02:00
|
|
|
(eq frame default-minibuffer-frame))))
|
|
|
|
;; Now link minibufferless frames with their minibuffer frames
|
|
|
|
(dolist (frame frames)
|
|
|
|
(unless (frame-parameter frame 'desktop-mini)
|
2013-07-22 03:25:47 +02:00
|
|
|
(let ((mb-frame (window-frame (minibuffer-window frame))))
|
|
|
|
;; Frames whose minibuffer frame has been filtered out will have
|
|
|
|
;; desktop-mini = nil, so desktop-restore-frames will restore them
|
|
|
|
;; according to their minibuffer parameter. Set up desktop-mini
|
|
|
|
;; for the rest.
|
|
|
|
(when (memq mb-frame frames)
|
|
|
|
(set-frame-parameter frame 'desktop-mini
|
|
|
|
(list nil
|
|
|
|
(cl-second (frame-parameter mb-frame 'desktop-mini))
|
|
|
|
nil))))))))
|
2013-06-27 11:08:14 +02:00
|
|
|
|
2013-07-21 19:45:12 +02:00
|
|
|
(defun desktop-save-frames ()
|
2013-07-22 03:25:47 +02:00
|
|
|
"Save frame state in `desktop-saved-frame-states'.
|
|
|
|
Runs the hook `desktop-before-saving-frames-functions'.
|
|
|
|
Frames with a non-nil `desktop-dont-save' parameter are not saved."
|
2013-07-21 19:45:12 +02:00
|
|
|
(setq desktop-saved-frame-states
|
2013-07-01 05:29:46 +02:00
|
|
|
(and desktop-restore-frames
|
2013-07-22 03:25:47 +02:00
|
|
|
(let ((frames (cl-delete-if
|
|
|
|
(lambda (frame)
|
|
|
|
(run-hook-with-args 'desktop-before-saving-frames-functions frame)
|
|
|
|
(frame-parameter frame 'desktop-dont-save))
|
|
|
|
(frame-list))))
|
|
|
|
;; In case some frame was deleted by a hook function
|
|
|
|
(setq frames (cl-delete-if-not #'frame-live-p frames))
|
|
|
|
(desktop--process-minibuffer-frames frames)
|
2013-07-15 02:07:51 +02:00
|
|
|
(mapcar (lambda (frame)
|
|
|
|
(cons (desktop--filter-frame-parms (frame-parameters frame) t)
|
|
|
|
(window-state-get (frame-root-window frame) t)))
|
2013-07-22 03:25:47 +02:00
|
|
|
frames)))))
|
2013-06-27 11:08:14 +02:00
|
|
|
|
2006-06-23 21:42:33 +00:00
|
|
|
;;;###autoload
|
2013-04-27 23:55:00 +03:00
|
|
|
(defun desktop-save (dirname &optional release auto-save)
|
2004-04-16 12:51:06 +00:00
|
|
|
"Save the desktop in a desktop file.
|
|
|
|
Parameter DIRNAME specifies where to save the desktop file.
|
2007-06-12 09:11:31 +00:00
|
|
|
Optional parameter RELEASE says whether we're done with this desktop.
|
2013-04-27 23:55:00 +03:00
|
|
|
If AUTO-SAVE is non-nil, compare the saved contents to the one last saved,
|
|
|
|
and don't save the buffer if they are the same."
|
1993-06-01 20:09:25 +00:00
|
|
|
(interactive "DDirectory to save desktop file in: ")
|
2007-06-12 09:11:31 +00:00
|
|
|
(setq desktop-dirname (file-name-as-directory (expand-file-name dirname)))
|
1993-06-01 20:09:25 +00:00
|
|
|
(save-excursion
|
2007-06-12 09:11:31 +00:00
|
|
|
(let ((eager desktop-restore-eager)
|
|
|
|
(new-modtime (nth 5 (file-attributes (desktop-full-file-name)))))
|
|
|
|
(when
|
|
|
|
(or (not new-modtime) ; nothing to overwrite
|
|
|
|
(equal desktop-file-modtime new-modtime)
|
|
|
|
(yes-or-no-p (if desktop-file-modtime
|
|
|
|
(if (> (float-time new-modtime) (float-time desktop-file-modtime))
|
|
|
|
"Desktop file is more recent than the one loaded. Save anyway? "
|
|
|
|
"Desktop file isn't the one loaded. Overwrite it? ")
|
|
|
|
"Current desktop was not loaded from a file. Overwrite this desktop file? "))
|
|
|
|
(unless release (error "Desktop file conflict")))
|
|
|
|
|
|
|
|
;; If we're done with it, release the lock.
|
|
|
|
;; Otherwise, claim it if it's unclaimed or if we created it.
|
|
|
|
(if release
|
|
|
|
(desktop-release-lock)
|
|
|
|
(unless (and new-modtime (desktop-owner)) (desktop-claim-lock)))
|
|
|
|
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert
|
|
|
|
";; -*- mode: emacs-lisp; coding: emacs-mule; -*-\n"
|
|
|
|
desktop-header
|
|
|
|
";; Created " (current-time-string) "\n"
|
|
|
|
";; Desktop file format version " desktop-file-version "\n"
|
|
|
|
";; Emacs version " emacs-version "\n")
|
|
|
|
(save-excursion (run-hooks 'desktop-save-hook))
|
|
|
|
(goto-char (point-max))
|
|
|
|
(insert "\n;; Global section:\n")
|
2013-06-27 11:08:14 +02:00
|
|
|
;; Called here because we save the window/frame state as a global
|
|
|
|
;; variable for compatibility with previous Emacsen.
|
2013-07-21 19:45:12 +02:00
|
|
|
(desktop-save-frames)
|
|
|
|
(unless (memq 'desktop-saved-frame-states desktop-globals-to-save)
|
|
|
|
(desktop-outvar 'desktop-saved-frame-states))
|
2007-06-12 09:11:31 +00:00
|
|
|
(mapc (function desktop-outvar) desktop-globals-to-save)
|
2013-07-21 19:45:12 +02:00
|
|
|
(setq desktop-saved-frame-states nil) ; after saving desktop-globals-to-save
|
2007-06-12 09:11:31 +00:00
|
|
|
(when (memq 'kill-ring desktop-globals-to-save)
|
|
|
|
(insert
|
|
|
|
"(setq kill-ring-yank-pointer (nthcdr "
|
|
|
|
(int-to-string (- (length kill-ring) (length kill-ring-yank-pointer)))
|
|
|
|
" kill-ring))\n"))
|
|
|
|
|
|
|
|
(insert "\n;; Buffer section -- buffers listed in same order as in buffer list:\n")
|
|
|
|
(dolist (l (mapcar 'desktop-buffer-info (buffer-list)))
|
2008-02-14 14:37:00 +00:00
|
|
|
(let ((base (pop l)))
|
|
|
|
(when (apply 'desktop-save-buffer-p l)
|
|
|
|
(insert "("
|
|
|
|
(if (or (not (integerp eager))
|
|
|
|
(if (zerop eager)
|
|
|
|
nil
|
|
|
|
(setq eager (1- eager))))
|
|
|
|
"desktop-create-buffer"
|
|
|
|
"desktop-append-buffer-args")
|
|
|
|
" "
|
|
|
|
desktop-file-version)
|
2008-02-29 03:03:57 +00:00
|
|
|
;; If there's a non-empty base name, we save it instead of the buffer name
|
|
|
|
(when (and base (not (string= base "")))
|
|
|
|
(setcar (nthcdr 1 l) base))
|
2008-02-14 14:37:00 +00:00
|
|
|
(dolist (e l)
|
|
|
|
(insert "\n " (desktop-value-to-string e)))
|
|
|
|
(insert ")\n\n"))))
|
2007-06-12 09:11:31 +00:00
|
|
|
|
2007-07-12 22:40:00 +00:00
|
|
|
(setq default-directory desktop-dirname)
|
2013-04-27 23:55:00 +03:00
|
|
|
;; If auto-saving, avoid writing if nothing has changed since the last write.
|
|
|
|
;; Don't check 300 characters of the header that contains the timestamp.
|
|
|
|
(let ((checksum (and auto-save (md5 (current-buffer)
|
|
|
|
(+ (point-min) 300) (point-max)
|
|
|
|
'emacs-mule))))
|
|
|
|
(unless (and auto-save (equal checksum desktop-file-checksum))
|
|
|
|
(let ((coding-system-for-write 'emacs-mule))
|
|
|
|
(write-region (point-min) (point-max) (desktop-full-file-name) nil 'nomessage))
|
|
|
|
(setq desktop-file-checksum checksum)
|
|
|
|
;; We remember when it was modified (which is presumably just now).
|
|
|
|
(setq desktop-file-modtime (nth 5 (file-attributes (desktop-full-file-name)))))))))))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2006-06-23 21:42:33 +00:00
|
|
|
;;;###autoload
|
1993-06-01 20:09:25 +00:00
|
|
|
(defun desktop-remove ()
|
2004-04-16 12:51:06 +00:00
|
|
|
"Delete desktop file in `desktop-dirname'.
|
|
|
|
This function also sets `desktop-dirname' to nil."
|
1993-06-01 20:09:25 +00:00
|
|
|
(interactive)
|
2004-04-16 12:51:06 +00:00
|
|
|
(when desktop-dirname
|
2006-06-23 21:42:33 +00:00
|
|
|
(let ((filename (desktop-full-file-name)))
|
2004-04-16 12:51:06 +00:00
|
|
|
(setq desktop-dirname nil)
|
|
|
|
(when (file-exists-p filename)
|
2003-06-03 23:35:20 +00:00
|
|
|
(delete-file filename)))))
|
2004-04-16 12:51:06 +00:00
|
|
|
|
2005-01-07 20:18:59 +00:00
|
|
|
(defvar desktop-buffer-args-list nil
|
|
|
|
"List of args for `desktop-create-buffer'.")
|
|
|
|
|
|
|
|
(defvar desktop-lazy-timer nil)
|
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2013-07-15 02:07:51 +02:00
|
|
|
(defvar desktop--reuse-list nil
|
|
|
|
"Internal use only.")
|
|
|
|
|
|
|
|
(defun desktop--find-frame (predicate display &rest args)
|
|
|
|
"Find a suitable frame in `desktop--reuse-list'.
|
|
|
|
Look through frames whose display property matches DISPLAY and
|
|
|
|
return the first one for which (PREDICATE frame ARGS) returns t.
|
|
|
|
If PREDICATE is nil, it is always satisfied. Internal use only.
|
|
|
|
This is an auxiliary function for `desktop--select-frame'."
|
2013-07-22 03:25:47 +02:00
|
|
|
(cl-find-if (lambda (frame)
|
|
|
|
(and (equal (frame-parameter frame 'display) display)
|
|
|
|
(or (null predicate)
|
|
|
|
(apply predicate frame args))))
|
|
|
|
desktop--reuse-list))
|
2013-07-15 02:07:51 +02:00
|
|
|
|
|
|
|
(defun desktop--select-frame (display frame-cfg)
|
|
|
|
"Look for an existing frame to reuse.
|
|
|
|
DISPLAY is the display where the frame will be shown, and FRAME-CFG
|
|
|
|
is the parameter list of the frame being restored. Internal use only."
|
|
|
|
(if (eq desktop-restoring-reuses-frames t)
|
|
|
|
(let ((frame nil)
|
|
|
|
mini)
|
|
|
|
;; There are no fancy heuristics there. We could implement some
|
|
|
|
;; based on frame size and/or position, etc., but it is not clear
|
|
|
|
;; that any "gain" (in the sense of reduced flickering, etc.) is
|
|
|
|
;; worth the added complexity. In fact, the code below mainly
|
|
|
|
;; tries to work nicely when M-x desktop-read is used after a desktop
|
|
|
|
;; session has already been loaded. The other main use case, which
|
|
|
|
;; is the initial desktop-read upon starting Emacs, should usually
|
|
|
|
;; only have one, or very few, frame(s) to reuse.
|
|
|
|
(cond (;; When the target is tty, every existing frame is reusable.
|
|
|
|
(null display)
|
|
|
|
(setq frame (desktop--find-frame nil display)))
|
|
|
|
(;; If the frame has its own minibuffer, let's see whether
|
|
|
|
;; that frame has already been loaded (which can happen after
|
|
|
|
;; M-x desktop-read).
|
|
|
|
(car (setq mini (cdr (assq 'desktop-mini frame-cfg))))
|
|
|
|
(setq frame (or (desktop--find-frame
|
|
|
|
(lambda (f m)
|
|
|
|
(equal (frame-parameter f 'desktop-mini) m))
|
|
|
|
display mini))))
|
|
|
|
(;; For minibufferless frames, check whether they already exist,
|
|
|
|
;; and that they are linked to the right minibuffer frame.
|
|
|
|
mini
|
|
|
|
(setq frame (desktop--find-frame
|
|
|
|
(lambda (f n)
|
|
|
|
(let ((m (frame-parameter f 'desktop-mini)))
|
|
|
|
(and m
|
2013-07-22 03:25:47 +02:00
|
|
|
(null (cl-first m))
|
|
|
|
(= (cl-second m) n)
|
|
|
|
(equal (cl-second (frame-parameter
|
|
|
|
(window-frame (minibuffer-window f))
|
|
|
|
'desktop-mini))
|
2013-07-15 02:07:51 +02:00
|
|
|
n))))
|
2013-07-22 03:25:47 +02:00
|
|
|
display (cl-second mini))))
|
2013-07-15 02:07:51 +02:00
|
|
|
(;; Default to just finding a frame in the same display.
|
|
|
|
t
|
|
|
|
(setq frame (desktop--find-frame nil display))))
|
|
|
|
;; If found, remove from the list.
|
|
|
|
(when frame
|
|
|
|
(setq desktop--reuse-list (delq frame desktop--reuse-list)))
|
2013-06-30 07:08:23 +02:00
|
|
|
frame)
|
2013-07-15 02:07:51 +02:00
|
|
|
nil))
|
|
|
|
|
|
|
|
(defun desktop--make-frame (frame-cfg window-cfg)
|
|
|
|
"Set up a frame according to its saved state.
|
|
|
|
That means either creating a new frame or reusing an existing one.
|
|
|
|
FRAME-CFG is the parameter list of the new frame; WINDOW-CFG is
|
|
|
|
its window state. Internal use only."
|
|
|
|
(let* ((fullscreen (cdr (assq 'fullscreen frame-cfg)))
|
|
|
|
(lines (assq 'tool-bar-lines frame-cfg))
|
|
|
|
(filtered-cfg (desktop--filter-frame-parms frame-cfg nil))
|
|
|
|
(display (cdr (assq 'display filtered-cfg))) ;; post-filtering
|
|
|
|
alt-cfg frame)
|
|
|
|
|
|
|
|
;; This works around bug#14795 (or feature#14795, if not a bug :-)
|
|
|
|
(setq filtered-cfg (assq-delete-all 'tool-bar-lines filtered-cfg))
|
|
|
|
(push '(tool-bar-lines . 0) filtered-cfg)
|
|
|
|
|
|
|
|
(when fullscreen
|
|
|
|
;; Currently Emacs has the limitation that it does not record the size
|
|
|
|
;; and position of a frame before maximizing it, so we cannot save &
|
|
|
|
;; restore that info. Instead, when restoring, we resort to creating
|
|
|
|
;; invisible "fullscreen" frames of default size and then maximizing them
|
|
|
|
;; (and making them visible) which at least is somewhat user-friendly
|
|
|
|
;; when these frames are later de-maximized.
|
|
|
|
(let ((width (and (eq fullscreen 'fullheight) (cdr (assq 'width filtered-cfg))))
|
|
|
|
(height (and (eq fullscreen 'fullwidth) (cdr (assq 'height filtered-cfg))))
|
|
|
|
(visible (assq 'visibility filtered-cfg)))
|
2013-07-22 03:25:47 +02:00
|
|
|
(setq filtered-cfg (cl-delete-if (lambda (p)
|
|
|
|
(memq p '(visibility fullscreen width height)))
|
|
|
|
filtered-cfg))
|
2013-07-15 02:07:51 +02:00
|
|
|
(when width
|
|
|
|
(setq filtered-cfg (append `((user-size . t) (width . ,width))
|
|
|
|
filtered-cfg)))
|
|
|
|
(when height
|
|
|
|
(setq filtered-cfg (append `((user-size . t) (height . ,height))
|
|
|
|
filtered-cfg)))
|
|
|
|
;; These are parameters to apply after creating/setting the frame.
|
|
|
|
(push visible alt-cfg)
|
|
|
|
(push (cons 'fullscreen fullscreen) alt-cfg)))
|
|
|
|
|
|
|
|
;; Time to select or create a frame an apply the big bunch of parameters
|
|
|
|
(if (setq frame (desktop--select-frame display filtered-cfg))
|
|
|
|
(modify-frame-parameters frame filtered-cfg)
|
|
|
|
(setq frame (make-frame-on-display display filtered-cfg)))
|
|
|
|
|
|
|
|
;; Let's give the finishing touches (visibility, tool-bar, maximization).
|
|
|
|
(when lines (push lines alt-cfg))
|
|
|
|
(when alt-cfg (modify-frame-parameters frame alt-cfg))
|
|
|
|
;; Now restore window state.
|
|
|
|
(window-state-put window-cfg (frame-root-window frame) 'safe)
|
2013-06-30 07:08:23 +02:00
|
|
|
frame))
|
|
|
|
|
2013-07-15 02:07:51 +02:00
|
|
|
(defun desktop--sort-states (state1 state2)
|
|
|
|
;; Order: default minibuffer frame
|
|
|
|
;; other frames with minibuffer, ascending ID
|
|
|
|
;; minibufferless frames, ascending ID
|
|
|
|
(let ((dm1 (cdr (assq 'desktop-mini (car state1))))
|
|
|
|
(dm2 (cdr (assq 'desktop-mini (car state2)))))
|
2013-07-22 03:25:47 +02:00
|
|
|
(cond ((cl-third dm1) t)
|
|
|
|
((cl-third dm2) nil)
|
|
|
|
((eq (cl-first dm1) (cl-first dm2))
|
|
|
|
(< (cl-second dm1) (cl-second dm2)))
|
|
|
|
(t
|
|
|
|
(cl-first dm1)))))
|
2013-07-15 02:07:51 +02:00
|
|
|
|
2013-07-21 19:45:12 +02:00
|
|
|
(defun desktop-restoring-frames-p ()
|
|
|
|
"True if calling `desktop-restore-frames' will actually restore frames."
|
|
|
|
(and desktop-restore-frames desktop-saved-frame-states))
|
|
|
|
|
|
|
|
(defun desktop-restore-frames ()
|
2013-06-27 11:08:14 +02:00
|
|
|
"Restore window/frame configuration.
|
2013-07-21 19:45:12 +02:00
|
|
|
This function depends on the value of `desktop-saved-frame-states'
|
|
|
|
being set (usually, by reading it from the desktop)."
|
|
|
|
(when (desktop-restoring-frames-p)
|
2013-07-15 02:07:51 +02:00
|
|
|
(let* ((frame-mb-map nil) ;; Alist of frames with their own minibuffer
|
|
|
|
(delete-saved (eq desktop-restore-in-current-display 'delete))
|
|
|
|
(forcing (not (desktop-restore-in-original-display-p)))
|
|
|
|
(target (and forcing (cons 'display (frame-parameter nil 'display)))))
|
|
|
|
|
|
|
|
;; Sorting saved states allows us to easily restore minibuffer-owning frames
|
|
|
|
;; before minibufferless ones.
|
2013-07-21 19:45:12 +02:00
|
|
|
(setq desktop-saved-frame-states (sort desktop-saved-frame-states
|
|
|
|
#'desktop--sort-states))
|
2013-07-15 02:07:51 +02:00
|
|
|
;; Potentially all existing frames are reusable. Later we will decide which ones
|
|
|
|
;; to reuse, and how to deal with any leftover.
|
|
|
|
(setq desktop--reuse-list (frame-list))
|
|
|
|
|
2013-07-21 19:45:12 +02:00
|
|
|
(dolist (state desktop-saved-frame-states)
|
2013-06-30 07:08:23 +02:00
|
|
|
(condition-case err
|
2013-07-15 02:07:51 +02:00
|
|
|
(let* ((frame-cfg (car state))
|
|
|
|
(window-cfg (cdr state))
|
|
|
|
(d-mini (cdr (assq 'desktop-mini frame-cfg)))
|
|
|
|
num frame to-tty)
|
|
|
|
;; Only set target if forcing displays and the target display is different.
|
|
|
|
(if (or (not forcing)
|
|
|
|
(equal target (or (assq 'display frame-cfg) '(display . nil))))
|
|
|
|
(setq desktop--target-display nil)
|
|
|
|
(setq desktop--target-display target
|
|
|
|
to-tty (null (cdr target))))
|
|
|
|
;; Time to restore frames and set up their minibuffers as they were.
|
|
|
|
;; We only skip a frame (thus deleting it) if either:
|
|
|
|
;; - we're switching displays, and the user chose the option to delete, or
|
|
|
|
;; - we're switching to tty, and the frame to restore is minibuffer-only.
|
|
|
|
(unless (and desktop--target-display
|
|
|
|
(or delete-saved
|
|
|
|
(and to-tty
|
|
|
|
(eq (cdr (assq 'minibuffer frame-cfg)) 'only))))
|
|
|
|
|
|
|
|
;; Restore minibuffers. Some of this stuff could be done in a filter
|
|
|
|
;; function, but it would be messy because restoring minibuffers affects
|
|
|
|
;; global state; it's best to do it here than add a bunch of global
|
|
|
|
;; variables to pass info back-and-forth to/from the filter function.
|
|
|
|
(cond
|
|
|
|
((null d-mini)) ;; No desktop-mini. Process as normal frame.
|
|
|
|
(to-tty) ;; Ignore minibuffer stuff and process as normal frame.
|
2013-07-22 03:25:47 +02:00
|
|
|
((cl-first d-mini) ;; Frame has minibuffer (or it is minibuffer-only).
|
|
|
|
(setq num (cl-second d-mini))
|
2013-07-15 02:07:51 +02:00
|
|
|
(when (eq (cdr (assq 'minibuffer frame-cfg)) 'only)
|
|
|
|
(setq frame-cfg (append '((tool-bar-lines . 0) (menu-bar-lines . 0))
|
|
|
|
frame-cfg))))
|
2013-07-20 21:22:33 -07:00
|
|
|
(t ;; Frame depends on other frame's minibuffer window.
|
2013-07-22 03:25:47 +02:00
|
|
|
(let ((mb-frame (cdr (assq (cl-second d-mini) frame-mb-map))))
|
2013-07-15 02:07:51 +02:00
|
|
|
(unless (frame-live-p mb-frame)
|
2013-07-22 03:25:47 +02:00
|
|
|
(error "Minibuffer frame %s not found" (cl-second d-mini)))
|
2013-07-15 02:07:51 +02:00
|
|
|
(let ((mb-param (assq 'minibuffer frame-cfg))
|
|
|
|
(mb-window (minibuffer-window mb-frame)))
|
|
|
|
(unless (and (window-live-p mb-window)
|
|
|
|
(window-minibuffer-p mb-window))
|
|
|
|
(error "Not a minibuffer window %s" mb-window))
|
|
|
|
(if mb-param
|
|
|
|
(setcdr mb-param mb-window)
|
|
|
|
(push (cons 'minibuffer mb-window) frame-cfg))))))
|
|
|
|
;; OK, we're ready at last to create (or reuse) a frame and
|
|
|
|
;; restore the window config.
|
|
|
|
(setq frame (desktop--make-frame frame-cfg window-cfg))
|
|
|
|
;; Set default-minibuffer if required.
|
2013-07-22 03:25:47 +02:00
|
|
|
(when (cl-third d-mini) (setq default-minibuffer-frame frame))
|
2013-07-15 02:07:51 +02:00
|
|
|
;; Store frame/NUM to assign to minibufferless frames.
|
2013-07-22 03:25:47 +02:00
|
|
|
(when num (push (cons num frame) frame-mb-map))))
|
2013-06-30 07:08:23 +02:00
|
|
|
(error
|
2013-07-15 02:07:51 +02:00
|
|
|
(delay-warning 'desktop (error-message-string err) :error))))
|
|
|
|
|
|
|
|
;; Delete remaining frames, but do not fail if some resist being deleted.
|
|
|
|
(unless (eq desktop-restoring-reuses-frames 'keep)
|
|
|
|
(dolist (frame desktop--reuse-list)
|
|
|
|
(ignore-errors (delete-frame frame))))
|
|
|
|
(setq desktop--reuse-list nil)
|
|
|
|
;; Make sure there's at least one visible frame, and select it.
|
2013-07-22 03:25:47 +02:00
|
|
|
(unless (or (daemonp)
|
|
|
|
(cl-find-if #'frame-visible-p (frame-list)))
|
|
|
|
(let ((visible (if (frame-live-p default-minibuffer-frame)
|
|
|
|
default-minibuffer-frame
|
|
|
|
(car (frame-list)))))
|
|
|
|
(make-frame-visible visible)
|
|
|
|
(select-frame-set-input-focus visible))))))
|
2013-06-27 11:08:14 +02:00
|
|
|
|
1998-03-28 07:32:50 +00:00
|
|
|
;;;###autoload
|
2004-04-16 12:51:06 +00:00
|
|
|
(defun desktop-read (&optional dirname)
|
|
|
|
"Read and process the desktop file in directory DIRNAME.
|
|
|
|
Look for a desktop file in DIRNAME, or if DIRNAME is omitted, look in
|
|
|
|
directories listed in `desktop-path'. If a desktop file is found, it
|
2005-07-19 09:54:05 +00:00
|
|
|
is processed and `desktop-after-read-hook' is run. If no desktop file
|
2004-04-16 12:51:06 +00:00
|
|
|
is found, clear the desktop and run `desktop-no-desktop-file-hook'.
|
|
|
|
This function is a no-op when Emacs is running in batch mode.
|
|
|
|
It returns t if a desktop file was loaded, nil otherwise."
|
1993-06-01 20:09:25 +00:00
|
|
|
(interactive)
|
2003-04-09 01:37:56 +00:00
|
|
|
(unless noninteractive
|
2004-04-16 12:51:06 +00:00
|
|
|
(setq desktop-dirname
|
2006-06-23 21:42:33 +00:00
|
|
|
(file-name-as-directory
|
|
|
|
(expand-file-name
|
|
|
|
(or
|
|
|
|
;; If DIRNAME is specified, use it.
|
|
|
|
(and (< 0 (length dirname)) dirname)
|
|
|
|
;; Otherwise search desktop file in desktop-path.
|
|
|
|
(let ((dirs desktop-path))
|
|
|
|
(while (and dirs
|
|
|
|
(not (file-exists-p
|
|
|
|
(desktop-full-file-name (car dirs)))))
|
|
|
|
(setq dirs (cdr dirs)))
|
|
|
|
(and dirs (car dirs)))
|
|
|
|
;; If not found and `desktop-path' is non-nil, use its first element.
|
|
|
|
(and desktop-path (car desktop-path))
|
2012-06-21 15:25:56 +08:00
|
|
|
;; Default: .emacs.d.
|
|
|
|
user-emacs-directory))))
|
2006-06-23 21:42:33 +00:00
|
|
|
(if (file-exists-p (desktop-full-file-name))
|
2007-06-12 09:11:31 +00:00
|
|
|
;; Desktop file found, but is it already in use?
|
|
|
|
(let ((desktop-first-buffer nil)
|
|
|
|
(desktop-buffer-ok-count 0)
|
|
|
|
(desktop-buffer-fail-count 0)
|
|
|
|
(owner (desktop-owner))
|
|
|
|
;; Avoid desktop saving during evaluation of desktop buffer.
|
|
|
|
(desktop-save nil))
|
|
|
|
(if (and owner
|
2007-06-12 11:14:52 +00:00
|
|
|
(memq desktop-load-locked-desktop '(nil ask))
|
|
|
|
(or (null desktop-load-locked-desktop)
|
2012-06-21 15:15:42 +08:00
|
|
|
(daemonp)
|
2007-06-12 11:14:52 +00:00
|
|
|
(not (y-or-n-p (format "Warning: desktop file appears to be in use by PID %s.\n\
|
|
|
|
Using it may cause conflicts. Use it anyway? " owner)))))
|
2008-03-01 14:17:41 +00:00
|
|
|
(let ((default-directory desktop-dirname))
|
2007-06-20 10:12:44 +00:00
|
|
|
(setq desktop-dirname nil)
|
2008-03-01 14:17:41 +00:00
|
|
|
(run-hooks 'desktop-not-loaded-hook)
|
|
|
|
(unless desktop-dirname
|
|
|
|
(message "Desktop file in use; not loaded.")))
|
2007-06-12 09:11:31 +00:00
|
|
|
(desktop-lazy-abort)
|
|
|
|
;; Evaluate desktop buffer and remember when it was modified.
|
|
|
|
(load (desktop-full-file-name) t t t)
|
|
|
|
(setq desktop-file-modtime (nth 5 (file-attributes (desktop-full-file-name))))
|
|
|
|
;; If it wasn't already, mark it as in-use, to bother other
|
|
|
|
;; desktop instances.
|
|
|
|
(unless owner
|
|
|
|
(condition-case nil
|
|
|
|
(desktop-claim-lock)
|
|
|
|
(file-error (message "Couldn't record use of desktop file")
|
|
|
|
(sit-for 1))))
|
|
|
|
|
2013-07-21 19:45:12 +02:00
|
|
|
(unless (desktop-restoring-frames-p)
|
|
|
|
;; `desktop-create-buffer' puts buffers at end of the buffer list.
|
|
|
|
;; We want buffers existing prior to evaluating the desktop (and
|
|
|
|
;; not reused) to be placed at the end of the buffer list, so we
|
|
|
|
;; move them here.
|
|
|
|
(mapc 'bury-buffer
|
|
|
|
(nreverse (cdr (memq desktop-first-buffer (nreverse (buffer-list))))))
|
|
|
|
(switch-to-buffer (car (buffer-list))))
|
2007-06-12 09:11:31 +00:00
|
|
|
(run-hooks 'desktop-delay-hook)
|
|
|
|
(setq desktop-delay-hook nil)
|
2013-07-21 19:45:12 +02:00
|
|
|
(desktop-restore-frames)
|
2007-06-12 09:11:31 +00:00
|
|
|
(run-hooks 'desktop-after-read-hook)
|
|
|
|
(message "Desktop: %d buffer%s restored%s%s."
|
|
|
|
desktop-buffer-ok-count
|
|
|
|
(if (= 1 desktop-buffer-ok-count) "" "s")
|
|
|
|
(if (< 0 desktop-buffer-fail-count)
|
|
|
|
(format ", %d failed to restore" desktop-buffer-fail-count)
|
|
|
|
"")
|
|
|
|
(if desktop-buffer-args-list
|
|
|
|
(format ", %d to restore lazily"
|
|
|
|
(length desktop-buffer-args-list))
|
|
|
|
""))
|
2013-07-21 19:45:12 +02:00
|
|
|
(unless (desktop-restoring-frames-p)
|
|
|
|
;; Bury the *Messages* buffer to not reshow it when burying
|
|
|
|
;; the buffer we switched to above.
|
|
|
|
(when (buffer-live-p (get-buffer "*Messages*"))
|
|
|
|
(bury-buffer "*Messages*"))
|
|
|
|
;; Clear all windows' previous and next buffers, these have
|
|
|
|
;; been corrupted by the `switch-to-buffer' calls in
|
|
|
|
;; `desktop-restore-file-buffer' (bug#11556). This is a
|
|
|
|
;; brute force fix and should be replaced by a more subtle
|
|
|
|
;; strategy eventually.
|
|
|
|
(walk-window-tree (lambda (window)
|
|
|
|
(set-window-prev-buffers window nil)
|
|
|
|
(set-window-next-buffers window nil))))
|
2007-06-12 09:11:31 +00:00
|
|
|
t))
|
2004-04-16 12:51:06 +00:00
|
|
|
;; No desktop file found.
|
|
|
|
(desktop-clear)
|
|
|
|
(let ((default-directory desktop-dirname))
|
|
|
|
(run-hooks 'desktop-no-desktop-file-hook))
|
|
|
|
(message "No desktop file.")
|
|
|
|
nil)))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2004-04-16 12:51:06 +00:00
|
|
|
;; Maintained for backward compatibility
|
1998-03-28 07:32:50 +00:00
|
|
|
;;;###autoload
|
1993-06-01 20:09:25 +00:00
|
|
|
(defun desktop-load-default ()
|
1995-01-30 06:14:33 +00:00
|
|
|
"Load the `default' start-up library manually.
|
2004-04-16 12:51:06 +00:00
|
|
|
Also inhibit further loading of it."
|
Use declare forms, where possible, to mark obsolete functions.
* lisp/allout.el (allout-passphrase-hint-string): Likewise.
(allout-init): Use a declare form to mark obsolete.
* lisp/calendar/calendar.el (calendar-version):
* lisp/calendar/icalendar.el (icalendar-extract-ical-from-buffer)
(icalendar-convert-diary-to-ical):
* lisp/cus-edit.el (custom-mode):
* lisp/ansi-color.el (ansi-color-unfontify-region):
* lisp/international/latin1-disp.el (latin1-char-displayable-p):
* lisp/progmodes/cwarn.el (turn-on-cwarn-mode):
* lisp/progmodes/which-func.el (which-func-update-1): Use
define-obsolete-function-alias.
* lisp/bookmark.el (bookmark-jump-noselect): Use a declare form to mark
this function obsolete.
* lisp/calendar/cal-x.el (calendar-two-frame-setup)
(calendar-only-one-frame-setup, calendar-one-frame-setup):
* lisp/calendar/calendar.el (american-calendar, european-calendar)
(calendar-for-loop):
* lisp/comint.el (comint-dynamic-simple-complete)
(comint-dynamic-complete-as-filename, comint-unquote-filename):
* lisp/desktop.el (desktop-load-default):
* lisp/dired-x.el (dired-omit-here-always)
(dired-hack-local-variables, dired-default-directory):
* lisp/emacs-lisp/derived.el (derived-mode-class):
* lisp/emacs-lisp/timer.el (timer-set-time-with-usecs):
* lisp/emacs-lock.el (toggle-emacs-lock):
* lisp/epa.el (epa-display-verify-result):
* lisp/epg.el (epg-sign-keys, epg-start-sign-keys)
(epg-passphrase-callback-function):
* lisp/eshell/esh-util.el (eshell-for):
* lisp/eshell/eshell.el (eshell-remove-from-window-buffer-names)
(eshell-add-to-window-buffer-names):
* lisp/files.el (locate-file-completion):
* lisp/imenu.el (imenu-example--create-c-index)
(imenu-example--create-lisp-index)
(imenu-example--lisp-extract-index-name)
(imenu-example--name-and-position):
* lisp/international/mule-cmds.el (princ-list):
* lisp/international/mule-diag.el (decode-codepage-char):
* lisp/international/mule-util.el (detect-coding-with-priority):
* lisp/iswitchb.el (iswitchb-read-buffer):
* lisp/mail/mailalias.el (mail-complete):
* lisp/mail/sendmail.el (mail-sent-via):
* lisp/mouse.el (mouse-popup-menubar-stuff, mouse-popup-menubar)
(mouse-major-mode-menu):
* lisp/password-cache.el (password-read-and-add):
* lisp/pcomplete.el (pcomplete-parse-comint-arguments):
* lisp/progmodes/sh-script.el (sh-maybe-here-document):
* lisp/replace.el (query-replace-regexp-eval):
* lisp/savehist.el (savehist-load):
* lisp/simple.el (choose-completion-delete-max-match):
* lisp/term.el (term-dynamic-simple-complete):
* lisp/vc/ediff-init.el (ediff-check-version):
* lisp/vc/ediff-wind.el (ediff-choose-window-setup-function-automatically):
* lisp/vc/vc.el (vc-diff-switches-list):
* lisp/view.el (view-return-to-alist-update): Likewise.
* lisp/iswitchb.el (iswitchb-read-buffer): Move code of
iswitchb-define-mode-map here, and delete that obsolete function.
* lisp/subr.el (eval-next-after-load, makehash, insert-string)
(assoc-ignore-representation, assoc-ignore-case): Use declare to
mark obsolete.
(mode-line-inverse-video): Variable deleted.
* lisp/emacs-lisp/byte-run.el (make-obsolete): Doc fix; emphasize that
this applies to functions.
* lisp/erc/erc.el (erc-send-command): Use define-obsolete-function-alias.
* lisp/international/mule-util.el (string-to-sequence): Remove.
* lisp/net/newst-backend.el (newsticker-cache-filename):
* lisp/net/newst-treeview.el (newsticker-groups-filename): Fix
incorrect obsolescence declaration.
* lisp/net/snmp-mode.el (snmp-font-lock-keywords-3): Don't use obsolete
font-lock-reference-face.
* lisp/url/url-parse.el (url-recreate-url-attributes):
* lisp/url/url-util.el (url-generate-unique-filename): Use declare to mark
obsolete.
* src/xdisp.c (mode_line_inverse_video): Delete obsolete variable.
2012-09-25 12:13:02 +08:00
|
|
|
(declare (obsolete desktop-save-mode "22.1"))
|
2005-07-29 10:25:15 +00:00
|
|
|
(unless inhibit-default-init ; safety check
|
|
|
|
(load "default" t t)
|
|
|
|
(setq inhibit-default-init t)))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
;;;###autoload
|
2004-04-16 12:51:06 +00:00
|
|
|
(defun desktop-change-dir (dirname)
|
|
|
|
"Change to desktop saved in DIRNAME.
|
|
|
|
Kill the desktop as specified by variables `desktop-save-mode' and
|
|
|
|
`desktop-save', then clear the desktop and load the desktop file in
|
|
|
|
directory DIRNAME."
|
|
|
|
(interactive "DChange to directory: ")
|
|
|
|
(setq dirname (file-name-as-directory (expand-file-name dirname desktop-dirname)))
|
2003-04-09 01:37:56 +00:00
|
|
|
(desktop-kill)
|
|
|
|
(desktop-clear)
|
2004-04-16 12:51:06 +00:00
|
|
|
(desktop-read dirname))
|
2005-02-09 15:50:47 +00:00
|
|
|
|
2004-04-16 12:51:06 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2003-04-09 01:37:56 +00:00
|
|
|
;;;###autoload
|
2004-04-16 12:51:06 +00:00
|
|
|
(defun desktop-save-in-desktop-dir ()
|
|
|
|
"Save the desktop in directory `desktop-dirname'."
|
2003-04-09 01:37:56 +00:00
|
|
|
(interactive)
|
|
|
|
(if desktop-dirname
|
2005-07-29 10:25:15 +00:00
|
|
|
(desktop-save desktop-dirname)
|
2003-04-09 01:37:56 +00:00
|
|
|
(call-interactively 'desktop-save))
|
2009-01-05 02:49:31 +00:00
|
|
|
(message "Desktop saved in %s" (abbreviate-file-name desktop-dirname)))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2013-04-27 23:55:00 +03:00
|
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
;; Auto-Saving.
|
|
|
|
(defvar desktop-auto-save-timer nil)
|
|
|
|
|
|
|
|
(defun desktop-auto-save ()
|
|
|
|
"Save the desktop periodically.
|
|
|
|
Called by the timer created in `desktop-auto-save-set-timer'."
|
|
|
|
(when (and desktop-save-mode
|
|
|
|
(integerp desktop-auto-save-timeout)
|
|
|
|
(> desktop-auto-save-timeout 0)
|
|
|
|
;; Avoid desktop saving during lazy loading.
|
|
|
|
(not desktop-lazy-timer)
|
|
|
|
;; Save only to own desktop file.
|
|
|
|
(eq (emacs-pid) (desktop-owner))
|
|
|
|
desktop-dirname)
|
|
|
|
(desktop-save desktop-dirname nil t))
|
|
|
|
(desktop-auto-save-set-timer))
|
|
|
|
|
|
|
|
(defun desktop-auto-save-set-timer ()
|
|
|
|
"Reset the auto-save timer.
|
|
|
|
Cancel any previous timer. When `desktop-auto-save-timeout' is a positive
|
|
|
|
integer, start a new timer to call `desktop-auto-save' in that many seconds."
|
|
|
|
(when desktop-auto-save-timer
|
|
|
|
(cancel-timer desktop-auto-save-timer)
|
|
|
|
(setq desktop-auto-save-timer nil))
|
|
|
|
(when (and (integerp desktop-auto-save-timeout)
|
|
|
|
(> desktop-auto-save-timeout 0))
|
|
|
|
(setq desktop-auto-save-timer
|
|
|
|
(run-with-timer desktop-auto-save-timeout nil
|
|
|
|
'desktop-auto-save))))
|
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
;;;###autoload
|
|
|
|
(defun desktop-revert ()
|
|
|
|
"Revert to the last loaded desktop."
|
|
|
|
(interactive)
|
2004-04-16 12:51:06 +00:00
|
|
|
(unless desktop-dirname
|
|
|
|
(error "Unknown desktop directory"))
|
2006-06-23 21:42:33 +00:00
|
|
|
(unless (file-exists-p (desktop-full-file-name))
|
2004-04-16 12:51:06 +00:00
|
|
|
(error "No desktop file found"))
|
|
|
|
(desktop-clear)
|
|
|
|
(desktop-read desktop-dirname))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2007-04-11 02:28:26 +00:00
|
|
|
(defvar desktop-buffer-major-mode)
|
|
|
|
(defvar desktop-buffer-locals)
|
2011-04-19 15:44:55 +02:00
|
|
|
(defvar auto-insert) ; from autoinsert.el
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2011-04-19 15:44:55 +02:00
|
|
|
(defun desktop-restore-file-buffer (buffer-filename
|
|
|
|
_buffer-name
|
|
|
|
_buffer-misc)
|
2004-04-21 20:53:35 +00:00
|
|
|
"Restore a file buffer."
|
2011-04-19 15:44:55 +02:00
|
|
|
(when buffer-filename
|
|
|
|
(if (or (file-exists-p buffer-filename)
|
2007-06-12 11:14:52 +00:00
|
|
|
(let ((msg (format "Desktop: File \"%s\" no longer exists."
|
2011-04-19 15:44:55 +02:00
|
|
|
buffer-filename)))
|
2007-06-12 11:14:52 +00:00
|
|
|
(if desktop-missing-file-warning
|
|
|
|
(y-or-n-p (concat msg " Re-create buffer? "))
|
|
|
|
(message "%s" msg)
|
|
|
|
nil)))
|
|
|
|
(let* ((auto-insert nil) ; Disable auto insertion
|
|
|
|
(coding-system-for-read
|
|
|
|
(or coding-system-for-read
|
|
|
|
(cdr (assq 'buffer-file-coding-system
|
|
|
|
desktop-buffer-locals))))
|
2011-04-19 15:44:55 +02:00
|
|
|
(buf (find-file-noselect buffer-filename)))
|
2007-06-12 11:14:52 +00:00
|
|
|
(condition-case nil
|
|
|
|
(switch-to-buffer buf)
|
|
|
|
(error (pop-to-buffer buf)))
|
|
|
|
(and (not (eq major-mode desktop-buffer-major-mode))
|
|
|
|
(functionp desktop-buffer-major-mode)
|
|
|
|
(funcall desktop-buffer-major-mode))
|
|
|
|
buf)
|
|
|
|
nil)))
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
(defun desktop-load-file (function)
|
|
|
|
"Load the file where auto loaded FUNCTION is defined."
|
2012-07-25 21:27:33 -04:00
|
|
|
(when (fboundp function)
|
|
|
|
(autoload-do-load (symbol-function function) function)))
|
2005-08-10 19:38:52 +00:00
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2004-04-27 06:52:35 +00:00
|
|
|
;; Create a buffer, load its file, set its mode, ...;
|
|
|
|
;; called from Desktop file only.
|
2001-10-23 13:52:45 +00:00
|
|
|
|
2005-08-10 19:38:52 +00:00
|
|
|
;; Just to silence the byte compiler.
|
2007-04-11 02:28:26 +00:00
|
|
|
|
|
|
|
(defvar desktop-first-buffer) ; Dynamically bound in `desktop-read'
|
|
|
|
|
|
|
|
;; Bound locally in `desktop-read'.
|
|
|
|
(defvar desktop-buffer-ok-count)
|
|
|
|
(defvar desktop-buffer-fail-count)
|
2003-04-09 01:37:56 +00:00
|
|
|
|
2004-09-09 19:45:03 +00:00
|
|
|
(defun desktop-create-buffer
|
2011-04-19 15:44:55 +02:00
|
|
|
(file-version
|
|
|
|
buffer-filename
|
|
|
|
buffer-name
|
|
|
|
buffer-majormode
|
|
|
|
buffer-minormodes
|
|
|
|
buffer-point
|
|
|
|
buffer-mark
|
|
|
|
buffer-readonly
|
|
|
|
buffer-misc
|
|
|
|
&optional
|
|
|
|
buffer-locals)
|
|
|
|
|
|
|
|
(let ((desktop-file-version file-version)
|
|
|
|
(desktop-buffer-file-name buffer-filename)
|
|
|
|
(desktop-buffer-name buffer-name)
|
|
|
|
(desktop-buffer-major-mode buffer-majormode)
|
|
|
|
(desktop-buffer-minor-modes buffer-minormodes)
|
|
|
|
(desktop-buffer-point buffer-point)
|
|
|
|
(desktop-buffer-mark buffer-mark)
|
|
|
|
(desktop-buffer-read-only buffer-readonly)
|
|
|
|
(desktop-buffer-misc buffer-misc)
|
|
|
|
(desktop-buffer-locals buffer-locals))
|
|
|
|
;; To make desktop files with relative file names possible, we cannot
|
|
|
|
;; allow `default-directory' to change. Therefore we save current buffer.
|
|
|
|
(save-current-buffer
|
|
|
|
;; Give major mode module a chance to add a handler.
|
|
|
|
(desktop-load-file desktop-buffer-major-mode)
|
|
|
|
(let ((buffer-list (buffer-list))
|
|
|
|
(result
|
2012-02-10 23:59:29 +08:00
|
|
|
(condition-case-unless-debug err
|
2011-04-19 15:44:55 +02:00
|
|
|
(funcall (or (cdr (assq desktop-buffer-major-mode
|
|
|
|
desktop-buffer-mode-handlers))
|
|
|
|
'desktop-restore-file-buffer)
|
|
|
|
desktop-buffer-file-name
|
|
|
|
desktop-buffer-name
|
|
|
|
desktop-buffer-misc)
|
|
|
|
(error
|
|
|
|
(message "Desktop: Can't load buffer %s: %s"
|
|
|
|
desktop-buffer-name
|
|
|
|
(error-message-string err))
|
|
|
|
(when desktop-missing-file-warning (sit-for 1))
|
|
|
|
nil))))
|
|
|
|
(if (bufferp result)
|
|
|
|
(setq desktop-buffer-ok-count (1+ desktop-buffer-ok-count))
|
|
|
|
(setq desktop-buffer-fail-count (1+ desktop-buffer-fail-count))
|
|
|
|
(setq result nil))
|
|
|
|
;; Restore buffer list order with new buffer at end. Don't change
|
2011-11-14 15:59:56 -08:00
|
|
|
;; the order for old desktop files (old desktop module behavior).
|
2011-04-19 15:44:55 +02:00
|
|
|
(unless (< desktop-file-version 206)
|
|
|
|
(mapc 'bury-buffer buffer-list)
|
|
|
|
(when result (bury-buffer result)))
|
|
|
|
(when result
|
|
|
|
(unless (or desktop-first-buffer (< desktop-file-version 206))
|
|
|
|
(setq desktop-first-buffer result))
|
|
|
|
(set-buffer result)
|
|
|
|
(unless (equal (buffer-name) desktop-buffer-name)
|
|
|
|
(rename-buffer desktop-buffer-name t))
|
|
|
|
;; minor modes
|
|
|
|
(cond ((equal '(t) desktop-buffer-minor-modes) ; backwards compatible
|
|
|
|
(auto-fill-mode 1))
|
|
|
|
((equal '(nil) desktop-buffer-minor-modes) ; backwards compatible
|
|
|
|
(auto-fill-mode 0))
|
|
|
|
(t
|
|
|
|
(dolist (minor-mode desktop-buffer-minor-modes)
|
|
|
|
;; Give minor mode module a chance to add a handler.
|
|
|
|
(desktop-load-file minor-mode)
|
|
|
|
(let ((handler (cdr (assq minor-mode desktop-minor-mode-handlers))))
|
|
|
|
(if handler
|
|
|
|
(funcall handler desktop-buffer-locals)
|
|
|
|
(when (functionp minor-mode)
|
|
|
|
(funcall minor-mode 1)))))))
|
|
|
|
;; Even though point and mark are non-nil when written by
|
|
|
|
;; `desktop-save', they may be modified by handlers wanting to set
|
|
|
|
;; point or mark themselves.
|
|
|
|
(when desktop-buffer-point
|
|
|
|
(goto-char
|
|
|
|
(condition-case err
|
|
|
|
;; Evaluate point. Thus point can be something like
|
|
|
|
;; '(search-forward ...
|
|
|
|
(eval desktop-buffer-point)
|
|
|
|
(error (message "%s" (error-message-string err)) 1))))
|
|
|
|
(when desktop-buffer-mark
|
|
|
|
(if (consp desktop-buffer-mark)
|
2007-06-12 11:14:52 +00:00
|
|
|
(progn
|
2011-04-19 15:44:55 +02:00
|
|
|
(set-mark (car desktop-buffer-mark))
|
|
|
|
(setq mark-active (car (cdr desktop-buffer-mark))))
|
|
|
|
(set-mark desktop-buffer-mark)))
|
|
|
|
;; Never override file system if the file really is read-only marked.
|
|
|
|
(when desktop-buffer-read-only (setq buffer-read-only desktop-buffer-read-only))
|
2013-07-23 03:10:54 +02:00
|
|
|
(dolist (this desktop-buffer-locals)
|
|
|
|
(if (consp this)
|
|
|
|
;; an entry of this form `(symbol . value)'
|
|
|
|
(progn
|
|
|
|
(make-local-variable (car this))
|
|
|
|
(set (car this) (cdr this)))
|
|
|
|
;; an entry of the form `symbol'
|
|
|
|
(make-local-variable this)
|
|
|
|
(makunbound this))))))))
|
1994-01-06 11:34:51 +00:00
|
|
|
|
2003-04-09 01:37:56 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
1994-01-06 11:34:51 +00:00
|
|
|
;; Backward compatibility -- update parameters to 205 standards.
|
2011-04-19 15:44:55 +02:00
|
|
|
(defun desktop-buffer (buffer-filename buffer-name buffer-majormode
|
|
|
|
mim pt mk ro tl fc cfs cr buffer-misc)
|
|
|
|
(desktop-create-buffer 205 buffer-filename buffer-name
|
|
|
|
buffer-majormode (cdr mim) pt mk ro
|
|
|
|
buffer-misc
|
1994-01-06 11:34:51 +00:00
|
|
|
(list (cons 'truncate-lines tl)
|
|
|
|
(cons 'fill-column fc)
|
|
|
|
(cons 'case-fold-search cfs)
|
|
|
|
(cons 'case-replace cr)
|
|
|
|
(cons 'overwrite-mode (car mim)))))
|
2003-04-10 16:26:12 +00:00
|
|
|
|
2005-01-07 20:18:59 +00:00
|
|
|
(defun desktop-append-buffer-args (&rest args)
|
2005-07-19 09:54:05 +00:00
|
|
|
"Append ARGS at end of `desktop-buffer-args-list'.
|
2005-01-07 20:18:59 +00:00
|
|
|
ARGS must be an argument list for `desktop-create-buffer'."
|
|
|
|
(setq desktop-buffer-args-list (nconc desktop-buffer-args-list (list args)))
|
|
|
|
(unless desktop-lazy-timer
|
|
|
|
(setq desktop-lazy-timer
|
|
|
|
(run-with-idle-timer desktop-lazy-idle-delay t 'desktop-idle-create-buffers))))
|
|
|
|
|
|
|
|
(defun desktop-lazy-create-buffer ()
|
|
|
|
"Pop args from `desktop-buffer-args-list', create buffer and bury it."
|
|
|
|
(when desktop-buffer-args-list
|
|
|
|
(let* ((remaining (length desktop-buffer-args-list))
|
|
|
|
(args (pop desktop-buffer-args-list))
|
|
|
|
(buffer-name (nth 2 args))
|
|
|
|
(msg (format "Desktop lazily opening %s (%s remaining)..."
|
|
|
|
buffer-name remaining)))
|
|
|
|
(when desktop-lazy-verbose
|
2005-09-18 12:25:02 +00:00
|
|
|
(message "%s" msg))
|
2005-01-07 20:18:59 +00:00
|
|
|
(let ((desktop-first-buffer nil)
|
|
|
|
(desktop-buffer-ok-count 0)
|
|
|
|
(desktop-buffer-fail-count 0))
|
|
|
|
(apply 'desktop-create-buffer args)
|
|
|
|
(run-hooks 'desktop-delay-hook)
|
|
|
|
(setq desktop-delay-hook nil)
|
|
|
|
(bury-buffer (get-buffer buffer-name))
|
|
|
|
(when desktop-lazy-verbose
|
|
|
|
(message "%s%s" msg (if (> desktop-buffer-ok-count 0) "done" "failed")))))))
|
|
|
|
|
|
|
|
(defun desktop-idle-create-buffers ()
|
|
|
|
"Create buffers until the user does something, then stop.
|
|
|
|
If there are no buffers left to create, kill the timer."
|
|
|
|
(let ((repeat 1))
|
|
|
|
(while (and repeat desktop-buffer-args-list)
|
|
|
|
(save-window-excursion
|
|
|
|
(desktop-lazy-create-buffer))
|
|
|
|
(setq repeat (sit-for 0.2))
|
|
|
|
(unless desktop-buffer-args-list
|
|
|
|
(cancel-timer desktop-lazy-timer)
|
|
|
|
(setq desktop-lazy-timer nil)
|
|
|
|
(message "Lazy desktop load complete")
|
|
|
|
(sit-for 3)
|
|
|
|
(message "")))))
|
|
|
|
|
|
|
|
(defun desktop-lazy-complete ()
|
|
|
|
"Run the desktop load to completion."
|
|
|
|
(interactive)
|
|
|
|
(let ((desktop-lazy-verbose t))
|
|
|
|
(while desktop-buffer-args-list
|
|
|
|
(save-window-excursion
|
|
|
|
(desktop-lazy-create-buffer)))
|
|
|
|
(message "Lazy desktop load complete")))
|
|
|
|
|
|
|
|
(defun desktop-lazy-abort ()
|
|
|
|
"Abort lazy loading of the desktop."
|
|
|
|
(interactive)
|
|
|
|
(when desktop-lazy-timer
|
|
|
|
(cancel-timer desktop-lazy-timer)
|
|
|
|
(setq desktop-lazy-timer nil))
|
|
|
|
(when desktop-buffer-args-list
|
|
|
|
(setq desktop-buffer-args-list nil)
|
2009-10-02 03:48:36 +00:00
|
|
|
(when (called-interactively-p 'interactive)
|
2005-01-07 20:18:59 +00:00
|
|
|
(message "Lazy desktop load aborted"))))
|
|
|
|
|
1993-10-07 16:50:26 +00:00
|
|
|
;; ----------------------------------------------------------------------------
|
2004-04-16 12:51:06 +00:00
|
|
|
;; When `desktop-save-mode' is non-nil and "--no-desktop" is not specified on the
|
2003-04-09 01:37:56 +00:00
|
|
|
;; command line, we do the rest of what it takes to use desktop, but do it
|
|
|
|
;; after finishing loading the init file.
|
|
|
|
;; We cannot use `command-switch-alist' to process "--no-desktop" because these
|
|
|
|
;; functions are processed after `after-init-hook'.
|
|
|
|
(add-hook
|
|
|
|
'after-init-hook
|
2007-04-11 02:28:26 +00:00
|
|
|
(lambda ()
|
2003-04-09 01:37:56 +00:00
|
|
|
(let ((key "--no-desktop"))
|
2005-04-18 18:27:39 +00:00
|
|
|
(when (member key command-line-args)
|
|
|
|
(setq command-line-args (delete key command-line-args))
|
|
|
|
(setq desktop-save-mode nil)))
|
2008-02-12 23:41:19 +00:00
|
|
|
(when desktop-save-mode
|
|
|
|
(desktop-read)
|
2013-04-27 23:55:00 +03:00
|
|
|
(desktop-auto-save-set-timer)
|
2008-02-12 23:41:19 +00:00
|
|
|
(setq inhibit-startup-screen t))))
|
1997-12-29 20:58:19 +00:00
|
|
|
|
2013-05-02 13:47:39 -04:00
|
|
|
;; So we can restore vc-dir buffers.
|
|
|
|
(autoload 'vc-dir-mode "vc-dir" nil t)
|
|
|
|
|
1993-06-01 20:11:54 +00:00
|
|
|
(provide 'desktop)
|
1993-06-01 20:09:25 +00:00
|
|
|
|
2000-10-10 16:42:37 +00:00
|
|
|
;;; desktop.el ends here
|