mirror of
https://git.savannah.gnu.org/git/emacs.git
synced 2024-11-22 07:09:54 +00:00
94fc8d5337
5ce02c91bc
Improve `desktop-save-mode` docstring2f98b13ed0
; Fix doc strings of splash-screen data structures77f5d4d523
Fix completing-read functional REQUIRE-MATCH behavior
1792 lines
69 KiB
EmacsLisp
1792 lines
69 KiB
EmacsLisp
;;; desktop.el --- save partial status of Emacs when killed -*- lexical-binding: t -*-
|
|
|
|
;; Copyright (C) 1993-1995, 1997, 2000-2024 Free Software Foundation,
|
|
;; Inc.
|
|
|
|
;; Author: Morten Welinder <terra@diku.dk>
|
|
;; Keywords: convenience
|
|
;; Favorite-brand-of-beer: None, I hate beer.
|
|
|
|
;; This file is part of GNU Emacs.
|
|
|
|
;; GNU Emacs is free software: you can redistribute it and/or modify
|
|
;; it under the terms of the GNU General Public License as published by
|
|
;; the Free Software Foundation, either version 3 of the License, or
|
|
;; (at your option) any later version.
|
|
|
|
;; GNU Emacs is distributed in the hope that it will be useful,
|
|
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
;; GNU General Public License for more details.
|
|
|
|
;; You should have received a copy of the GNU General Public License
|
|
;; along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
;;; Commentary:
|
|
|
|
;; 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
|
|
;; - some local variables
|
|
;; - frame and window configuration
|
|
|
|
;; To use this, use customize to turn on desktop-save-mode or add the
|
|
;; following line somewhere in your init file:
|
|
;;
|
|
;; (desktop-save-mode 1)
|
|
;;
|
|
;; For further usage information, look at the section
|
|
;; (info "(emacs)Saving Emacs Sessions") in the GNU Emacs Manual.
|
|
|
|
;; When the desktop module is loaded, the function `desktop-kill' is
|
|
;; added to the `kill-emacs-query-functions'. This function is
|
|
;; responsible for saving the desktop and deleting the desktop lock
|
|
;; file when Emacs is killed. In addition, an anonymous function is
|
|
;; added to the `after-init-hook'. This function is responsible for
|
|
;; loading the desktop when Emacs is started.
|
|
|
|
;; 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
|
|
;; to restore a desktop buffer. Elements must have the form
|
|
;;
|
|
;; (MAJOR-MODE . RESTORE-BUFFER-FUNCTION).
|
|
;;
|
|
;; Functions listed are called by `desktop-create-buffer' when `desktop-read'
|
|
;; evaluates the desktop file. Buffers with a major mode not specified here,
|
|
;; 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
|
|
;;
|
|
;; (MINOR-MODE . RESTORE-FUNCTION).
|
|
;;
|
|
;; 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))
|
|
|
|
;; in the module itself. The mode function must either be autoloaded,
|
|
;; or of the form "foobar-mode" and defined in library "foobar", so that
|
|
;; desktop can guess how to load its definition.
|
|
;; See the docstrings of `desktop-buffer-mode-handlers' and
|
|
;; `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
|
|
;; functions before writing `desktop-minor-modes'. If a minor mode has a
|
|
;; 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.
|
|
;; ---------------------------------------------------------------------------
|
|
|
|
;; 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.
|
|
|
|
;; 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.
|
|
|
|
;; 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@cs.stonybrook.edu (M. Kifer) for a bug hunt.
|
|
;; treese@lcs.mit.edu (Win Treese) for ange-ftp tips.
|
|
;; pot@cnuce.cnr.it (Francesco Potortì) for misc. tips.
|
|
;; ---------------------------------------------------------------------------
|
|
;; TODO:
|
|
;;
|
|
;; Recognize more minor modes.
|
|
;; Save mark rings.
|
|
|
|
;;; Code:
|
|
|
|
(require 'cl-lib)
|
|
(require 'frameset)
|
|
|
|
(defvar desktop-file-version "208"
|
|
"Version number of desktop file format.
|
|
Used at desktop read to provide backward compatibility.")
|
|
|
|
(defconst desktop-native-file-version 208
|
|
"Format version of the current desktop package, an integer.")
|
|
(defvar desktop-io-file-version nil
|
|
"The format version of the current desktop file (an integer) or nil.")
|
|
;; Note: Historically, the version number is embedded in the entry for
|
|
;; each buffer. It is highly inadvisable for different buffer entries
|
|
;; to have different format versions.
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;; USER OPTIONS -- settings you might want to play with.
|
|
;; ----------------------------------------------------------------------------
|
|
|
|
(defgroup desktop nil
|
|
"Save status of Emacs when you exit."
|
|
:group 'frames)
|
|
|
|
;;;###autoload
|
|
(define-minor-mode desktop-save-mode
|
|
"Toggle desktop saving (Desktop Save mode).
|
|
|
|
When Desktop Save mode is enabled, the state of Emacs is saved from one
|
|
session to another. The saved Emacs \"desktop configuration\" includes the
|
|
buffers, their file names, major modes, buffer positions, window and frame
|
|
configuration, and some important global variables.
|
|
|
|
To enable this feature for future sessions, customize `desktop-save-mode'
|
|
to t, or add this line in your init file:
|
|
|
|
(desktop-save-mode 1)
|
|
|
|
When this mode is enabled, Emacs will save the desktop when it exits
|
|
(this may prompt you, see the option `desktop-save'). The next time
|
|
Emacs starts, if this mode is active it will restore the desktop.
|
|
|
|
To manually save the desktop at any time, use the command \\[desktop-save].
|
|
To load it, use \\[desktop-read].
|
|
|
|
Once a desktop file exists, Emacs will auto-save it according to the
|
|
option `desktop-auto-save-timeout'.
|
|
|
|
To see all the options you can set, browse the `desktop' customization group.
|
|
|
|
For further details, see info node `(emacs)Saving Emacs Sessions'."
|
|
:global t
|
|
:group 'desktop
|
|
(if desktop-save-mode
|
|
(desktop-auto-save-enable)
|
|
(desktop-auto-save-disable)))
|
|
|
|
(defun desktop-save-mode-off ()
|
|
"Disable `desktop-save-mode'. Provided for use in hooks."
|
|
(desktop-save-mode 0))
|
|
|
|
(defcustom desktop-save 'ask-if-new
|
|
"Specifies whether the desktop should be saved when it is killed.
|
|
A desktop is killed when the user changes desktop or quits Emacs.
|
|
Possible values are:
|
|
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.
|
|
The desktop is never saved when `desktop-save-mode' is nil.
|
|
The variables `desktop-dirname' and `desktop-base-file-name'
|
|
determine where the desktop is saved."
|
|
:type
|
|
'(choice
|
|
(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))
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-auto-save-timeout auto-save-timeout
|
|
"Number of seconds of idle time before auto-saving the desktop.
|
|
The desktop will be auto-saved when this amount of idle time have
|
|
passed after some change in the window configuration.
|
|
This applies to an existing desktop file when `desktop-save-mode' is enabled.
|
|
Zero or nil means disable auto-saving due to idleness."
|
|
:type '(choice (const :tag "Off" nil)
|
|
(integer :tag "Seconds"))
|
|
:set (lambda (symbol value)
|
|
(set-default symbol value)
|
|
(ignore-errors
|
|
(if (and (integerp value) (> value 0))
|
|
(desktop-auto-save-enable value)
|
|
(desktop-auto-save-disable))))
|
|
:group 'desktop
|
|
:version "24.4")
|
|
|
|
(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.
|
|
check-pid -- load if locking Emacs process is missing locally.
|
|
|
|
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.
|
|
|
|
If the value is `check-pid', load the desktop if the Emacs
|
|
process that has locked it is not running on the local machine.
|
|
This should not be used in circumstances where the locking Emacs
|
|
might still be running on another machine. That could be the
|
|
case if you have remotely mounted (NFS) paths in
|
|
`desktop-dirname'."
|
|
:type
|
|
'(choice
|
|
(const :tag "Load anyway" t)
|
|
(const :tag "Don't load" nil)
|
|
(const :tag "Ask the user" ask)
|
|
(const :tag "Load if no local process" check-pid))
|
|
:group 'desktop
|
|
:version "22.2")
|
|
|
|
(defcustom desktop-base-file-name
|
|
(convert-standard-filename ".emacs.desktop")
|
|
"Name of file for Emacs desktop, excluding the directory part."
|
|
:type 'file
|
|
:group 'desktop)
|
|
|
|
(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
|
|
:version "22.2")
|
|
|
|
(defcustom desktop-path (list user-emacs-directory "~")
|
|
"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)
|
|
:group 'desktop
|
|
:version "23.2") ; user-emacs-directory added
|
|
|
|
(defcustom desktop-missing-file-warning nil
|
|
"If non-nil, offer to recreate the buffer of a deleted file.
|
|
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."
|
|
:type 'boolean
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-no-desktop-file-hook nil
|
|
"Normal hook run when `desktop-read' can't find a desktop file.
|
|
Run in the directory in which the desktop file was sought.
|
|
May be used to show a dired buffer."
|
|
:type 'hook
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-not-loaded-hook nil
|
|
"Normal hook run when the user declines to reuse 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)
|
|
:version "22.2")
|
|
|
|
(defcustom desktop-after-read-hook nil
|
|
"Normal hook run after a successful `desktop-read'.
|
|
May be used to show a buffer list."
|
|
:type 'hook
|
|
:group 'desktop
|
|
:options '(list-buffers)
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-save-hook nil
|
|
"Normal hook run before the desktop is saved in a desktop file.
|
|
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."
|
|
:type 'hook
|
|
:group 'desktop)
|
|
|
|
(defcustom desktop-globals-to-save
|
|
'(desktop-missing-file-warning
|
|
tags-file-name
|
|
tags-table-list
|
|
search-ring
|
|
regexp-search-ring
|
|
register-alist
|
|
file-name-history)
|
|
"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.
|
|
Feature: Saving `kill-ring' implies saving `kill-ring-yank-pointer'."
|
|
:type '(repeat (restricted-sexp :match-alternatives (symbolp consp)))
|
|
:group 'desktop)
|
|
|
|
(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)
|
|
"List of global variables that `desktop-clear' will clear.
|
|
An element may be variable name (a symbol) or a cons cell of the form
|
|
\(VAR . FORM). Symbols are set to nil and for cons cells VAR is set
|
|
to the value obtained by evaluating FORM."
|
|
:type '(repeat (restricted-sexp :match-alternatives (symbolp consp)))
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-clear-preserve-buffers
|
|
'("\\*scratch\\*" "\\*Messages\\*" "\\*server\\*" "\\*tramp/.+\\*"
|
|
"\\*Warnings\\*")
|
|
"List of buffers that `desktop-clear' should not delete.
|
|
Each element is a regular expression. Buffers with a name matched by any of
|
|
these won't be deleted."
|
|
:version "23.3" ; added Warnings - bug#6336
|
|
:type '(repeat regexp)
|
|
:group 'desktop)
|
|
|
|
;;;###autoload
|
|
(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
|
|
column-number-mode
|
|
size-indication-mode
|
|
buffer-file-coding-system
|
|
buffer-display-time
|
|
indent-tabs-mode
|
|
tab-width
|
|
indicate-buffer-boundaries
|
|
indicate-empty-lines
|
|
show-trailing-whitespace)
|
|
"List of local variables to save for each buffer.
|
|
The variables are saved only when they really are local. Conventional minor
|
|
modes are restored automatically; they should not be listed here."
|
|
:type '(repeat symbol)
|
|
:group 'desktop)
|
|
|
|
(defcustom desktop-buffers-not-to-save "\\` "
|
|
"Regexp identifying buffers that are to be excluded from saving.
|
|
This is in effect only for buffers that don't visit files.
|
|
To exclude buffers that visit files, use `desktop-files-not-to-save'
|
|
or `desktop-modes-not-to-save'."
|
|
:type '(choice (const :tag "None" nil)
|
|
regexp)
|
|
:version "24.4" ; skip invisible temporary buffers
|
|
:group 'desktop)
|
|
|
|
;; Skip tramp and ange-ftp files
|
|
(defcustom desktop-files-not-to-save
|
|
"\\(\\`/[^/:]*:\\|(ftp)\\'\\)"
|
|
"Regexp identifying files whose buffers are to be excluded from saving.
|
|
The default value excludes buffers visiting remote files.
|
|
|
|
If you modify this such that buffers visiting remote files are not excluded,
|
|
you may wish customizing `remote-file-name-access-timeout' to a non-nil
|
|
value, to avoid hanging the desktop restoration because some remote
|
|
host is off-line."
|
|
:type '(choice (const :tag "None" nil)
|
|
regexp)
|
|
:group 'desktop)
|
|
|
|
;; We skip TAGS files to save time (tags-file-name is saved instead).
|
|
(defcustom desktop-modes-not-to-save
|
|
'(tags-table-mode)
|
|
"List of major modes whose buffers should not be saved."
|
|
:type '(repeat symbol)
|
|
:group 'desktop)
|
|
|
|
(defcustom desktop-restore-frames t
|
|
"When non-nil, save and restore the frame and window configuration.
|
|
See related options `desktop-restore-reuses-frames',
|
|
`desktop-restore-in-current-display', and `desktop-restore-forces-onscreen'."
|
|
:type 'boolean
|
|
:group 'desktop
|
|
:version "24.4")
|
|
|
|
(defcustom desktop-restore-in-current-display t
|
|
"Controls how restoring of frames treats displays.
|
|
If t, restores frames into the current display.
|
|
If nil, restores frames into their original displays (if possible).
|
|
If `delete', deletes frames on other displays instead of restoring them."
|
|
: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-restore-forces-onscreen t
|
|
"If t, restores frames that are fully offscreen onscreen instead.
|
|
If `all', also restores frames that are partially offscreen onscreen.
|
|
|
|
Note that checking of frame boundaries is only approximate.
|
|
It can fail to reliably detect frames whose onscreen/offscreen state
|
|
depends on a few pixels, especially near the right / bottom borders
|
|
of the screen.
|
|
Text-mode frames are always considered onscreen, so this option has
|
|
no effect on restoring frames in a non-GUI session."
|
|
:type '(choice (const :tag "Only fully offscreen frames" t)
|
|
(const :tag "Also partially offscreen frames" all)
|
|
(const :tag "Do not force frames onscreen" nil))
|
|
:group 'desktop
|
|
:version "24.4")
|
|
|
|
(defcustom desktop-restore-reuses-frames t
|
|
"If t, restoring frames reuses existing frames.
|
|
If nil, deletes existing frames.
|
|
If `keep', keeps existing frames and does not reuse them."
|
|
:type '(choice (const :tag "Reuse existing frames" t)
|
|
(const :tag "Delete existing frames" nil)
|
|
(const :tag "Keep existing frames" keep))
|
|
:group 'desktop
|
|
:version "24.4")
|
|
|
|
(defcustom desktop-file-name-format 'absolute
|
|
"Format in which desktop file names should be saved.
|
|
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))
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(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."
|
|
:type '(choice (const t) integer)
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-lazy-verbose t
|
|
"Verbose reporting of lazily created buffers."
|
|
:type 'boolean
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
(defcustom desktop-lazy-idle-delay 5
|
|
"Idle delay before starting to create buffers.
|
|
See `desktop-restore-eager'."
|
|
:type 'natnum
|
|
:group 'desktop
|
|
:version "22.1")
|
|
|
|
;;;###autoload
|
|
(defvar-local desktop-save-buffer nil
|
|
"When non-nil, save buffer status in desktop file.
|
|
|
|
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
|
|
file along with the state of the buffer for which it was called.
|
|
|
|
When file names are returned, they should be formatted using the call
|
|
\"(desktop-file-name FILE-NAME DESKTOP-DIRNAME)\".
|
|
|
|
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'.")
|
|
|
|
;;;###autoload
|
|
(defvar desktop-buffer-mode-handlers nil
|
|
"Alist of major mode specific functions to restore a desktop buffer.
|
|
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).
|
|
|
|
Buffers with a major mode not specified here, are restored by the default
|
|
handler `desktop-restore-file-buffer'.
|
|
|
|
Handlers are called with argument list
|
|
|
|
(DESKTOP-BUFFER-FILE-NAME DESKTOP-BUFFER-NAME DESKTOP-BUFFER-MISC)
|
|
|
|
Furthermore, they may use the following variables:
|
|
|
|
`desktop-file-version'
|
|
`desktop-buffer-major-mode'
|
|
`desktop-buffer-minor-modes'
|
|
`desktop-buffer-point'
|
|
`desktop-buffer-mark'
|
|
`desktop-buffer-read-only'
|
|
`desktop-buffer-locals'
|
|
|
|
If a handler returns a buffer, then the saved mode settings
|
|
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
|
|
|
|
(defun foo-restore-desktop-buffer
|
|
...
|
|
(add-to-list \\='desktop-buffer-mode-handlers
|
|
\\='(foo-mode . foo-restore-desktop-buffer))
|
|
|
|
The major mode function must either be autoloaded, or of the form
|
|
\"foobar-mode\" and defined in library \"foobar\", so that desktop
|
|
can guess how to load the mode's definition.")
|
|
|
|
;;;###autoload
|
|
(put 'desktop-buffer-mode-handlers 'risky-local-variable t)
|
|
|
|
(defcustom desktop-minor-mode-table
|
|
'((defining-kbd-macro nil)
|
|
(isearch-mode nil)
|
|
(vc-mode nil)
|
|
(vc-dir-mode nil)
|
|
(erc-track-minor-mode nil)
|
|
(savehist-mode nil))
|
|
"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.
|
|
RESTORE-FUNCTION nil means don't try to restore the minor mode.
|
|
Only minor modes for which the name of the buffer-local variable
|
|
and the name of the minor mode function are different have to be added to
|
|
this table. See also `desktop-minor-mode-handlers'."
|
|
:type '(alist :key-type (symbol :tag "Minor mode")
|
|
:value-type (list :tag "Restore function"
|
|
(choice (const nil) function)))
|
|
:group 'desktop)
|
|
|
|
;;;###autoload
|
|
(defvar desktop-minor-mode-handlers nil
|
|
"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))
|
|
|
|
The minor mode function must either be autoloaded, or of the form
|
|
\"foobar-mode\" and defined in library \"foobar\", so that desktop
|
|
can guess how to load the mode's definition.
|
|
|
|
See also `desktop-minor-mode-table'.")
|
|
|
|
;;;###autoload
|
|
(put 'desktop-minor-mode-handlers 'risky-local-variable t)
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defvar desktop-dirname nil
|
|
"The directory in which the desktop file should be saved.")
|
|
|
|
(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)))
|
|
|
|
(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)))
|
|
|
|
(defconst desktop-header
|
|
";; --------------------------------------------------------------------------
|
|
;; Desktop File for Emacs
|
|
;; --------------------------------------------------------------------------
|
|
" "Header to place in Desktop file.")
|
|
|
|
(defvar desktop-delay-hook nil
|
|
"Hooks run after all buffers are loaded; intended for internal use.")
|
|
|
|
(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.")
|
|
|
|
(defvar desktop-saved-frameset nil
|
|
"Saved state of all frames.
|
|
Only valid during frame saving & restoring; intended for internal use.")
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;; 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--get-file-modtime ()
|
|
"Get desktop file modtime, in list form for desktop format version 208."
|
|
(setq desktop-file-modtime
|
|
(time-convert (file-attribute-modification-time
|
|
(file-attributes
|
|
(desktop-full-file-name)))
|
|
'list)))
|
|
|
|
(defvar desktop-var-serdes-funs
|
|
(list (list
|
|
'mark-ring
|
|
(lambda (mr)
|
|
(mapcar #'marker-position mr))
|
|
(lambda (mr)
|
|
(mapcar #'copy-marker mr))))
|
|
"Table of serialization/deserialization functions for variables.
|
|
Each record is a list of form: (var serializer deserializer).
|
|
These records can be freely reordered, deleted, or new ones added.
|
|
However, for compatibility, don't modify the functions for existing records.")
|
|
|
|
(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'."
|
|
(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)))
|
|
owner)))
|
|
|
|
(defun desktop--emacs-pid-running-p (pid)
|
|
"Return non-nil if an Emacs process whose ID is PID might still be running."
|
|
(when-let ((attr (process-attributes pid)))
|
|
(let ((proc-cmd (alist-get 'comm attr))
|
|
(my-cmd (file-name-nondirectory (car command-line-args)))
|
|
(case-fold-search t))
|
|
(or (equal proc-cmd my-cmd)
|
|
(and (eq system-type 'windows-nt)
|
|
(eq t (compare-strings proc-cmd
|
|
nil
|
|
(if (string-suffix-p ".exe" proc-cmd t)
|
|
-4)
|
|
my-cmd
|
|
nil
|
|
(if (string-suffix-p ".exe" my-cmd t)
|
|
-4))))
|
|
;; We should err on the safe side here: if any of the
|
|
;; executables is something like "emacs-nox" or "emacs-42.1"
|
|
;; or "gemacs", let's recognize them as well.
|
|
(and (string-match-p "emacs" proc-cmd)
|
|
(string-match-p "emacs" my-cmd))))))
|
|
|
|
(defun desktop--load-locked-desktop-p (owner)
|
|
"Return t if a locked desktop should be loaded.
|
|
OWNER is the pid in the lock file.
|
|
The return value of this function depends on the value of
|
|
`desktop-load-locked-desktop'."
|
|
(pcase desktop-load-locked-desktop
|
|
('ask
|
|
(unless (daemonp)
|
|
(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))))
|
|
('check-pid
|
|
(or (eq (emacs-pid) owner)
|
|
(not (desktop--emacs-pid-running-p owner))))
|
|
('nil nil)
|
|
(_ t)))
|
|
|
|
(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))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-truncate (list n)
|
|
"Truncate LIST to at most N elements destructively."
|
|
(let ((here (nthcdr (1- n) list)))
|
|
(when (consp here)
|
|
(setcdr here nil))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;;;###autoload
|
|
(defun desktop-clear ()
|
|
"Empty the Desktop.
|
|
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'.
|
|
When called interactively and `desktop-restore-frames' is non-nil, it also
|
|
deletes all frames except the selected one (and its minibuffer frame,
|
|
if different)."
|
|
(interactive)
|
|
(desktop-lazy-abort)
|
|
(setq desktop-io-file-version nil)
|
|
(dolist (var desktop-globals-to-clear)
|
|
(if (symbolp var)
|
|
(set-default var nil)
|
|
(set-default var (eval (cdr var)))))
|
|
(let ((preserve-regexp (concat "\\`\\("
|
|
(mapconcat (lambda (regexp)
|
|
(concat "\\(" regexp "\\)"))
|
|
desktop-clear-preserve-buffers
|
|
"\\|")
|
|
"\\)\\'")))
|
|
(dolist (buffer (buffer-list))
|
|
(let ((bufname (buffer-name buffer)))
|
|
(unless (or (null bufname)
|
|
(eq (aref bufname 0) ?\s) ;; Don't kill internal buffers
|
|
(string-match-p preserve-regexp bufname))
|
|
(kill-buffer buffer)))))
|
|
(delete-other-windows)
|
|
(when (and desktop-restore-frames
|
|
;; Non-interactive calls to desktop-clear happen before desktop-read
|
|
;; which already takes care of frame restoration and deletion.
|
|
(called-interactively-p 'any))
|
|
(let* ((this (selected-frame))
|
|
(mini (window-frame (minibuffer-window this)))) ; in case they differ
|
|
(dolist (frame (sort (frame-list) #'frameset-minibufferless-first-p))
|
|
(condition-case err
|
|
(unless (or (eq frame this)
|
|
(eq frame mini)
|
|
;; Don't delete daemon's initial frame, or
|
|
;; we'll never be able to close the last
|
|
;; client's frame (Bug#26912).
|
|
(if (daemonp) (not (frame-parameter frame 'client)))
|
|
(frame-parameter frame 'desktop-dont-clear))
|
|
(delete-frame frame))
|
|
(error
|
|
(delay-warning 'desktop (error-message-string err))))))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(unless noninteractive
|
|
(add-hook 'kill-emacs-query-functions #'desktop-kill)
|
|
;; Certain things should be done even if
|
|
;; `kill-emacs-query-functions' are not called.
|
|
(add-hook 'kill-emacs-hook #'desktop--on-kill))
|
|
|
|
(defun desktop-kill ()
|
|
"If `desktop-save-mode' is non-nil, do what `desktop-save' says to do.
|
|
If the desktop should be saved and `desktop-dirname'
|
|
is nil, ask the user where to save the desktop."
|
|
(when (and desktop-save-mode
|
|
(let ((exists (file-exists-p (desktop-full-file-name))))
|
|
(or (eq desktop-save t)
|
|
(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))
|
|
(and
|
|
(or (memq desktop-save '(ask ask-if-new))
|
|
(and exists (eq desktop-save 'ask-if-exists)))
|
|
(y-or-n-p "Save desktop? ")))))
|
|
(unless desktop-dirname
|
|
(setq desktop-dirname
|
|
(file-name-as-directory
|
|
(expand-file-name
|
|
(read-directory-name "Directory for desktop file: " nil nil t)))))
|
|
(condition-case err
|
|
(desktop-save desktop-dirname t)
|
|
(file-error
|
|
(unless (yes-or-no-p "Error while saving the desktop. Ignore? ")
|
|
(signal (car err) (cdr err))))))
|
|
(desktop--on-kill)
|
|
t)
|
|
|
|
(defun desktop--on-kill ()
|
|
;; If we own it, we don't anymore.
|
|
(when (eq (emacs-pid) (desktop-owner))
|
|
;; Allow exiting Emacs even if we can't delete the desktop file.
|
|
(ignore-error file-error
|
|
(desktop-release-lock))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-list* (&rest args)
|
|
(and args (apply #'cl-list* args)))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-buffer-info (buffer)
|
|
"Return information describing BUFFER.
|
|
This function is not pure, as BUFFER is made current with
|
|
`set-buffer'.
|
|
|
|
Returns a list of all the necessary information to recreate the
|
|
buffer, which is (in order):
|
|
|
|
`uniquify-buffer-base-name';
|
|
`buffer-file-name';
|
|
`buffer-name';
|
|
`major-mode';
|
|
list of minor-modes,;
|
|
`point';
|
|
`mark';
|
|
`buffer-read-only';
|
|
auxiliary information given by `desktop-save-buffer';
|
|
local variables;
|
|
auxiliary information given by `desktop-var-serdes-funs'."
|
|
(set-buffer buffer)
|
|
`(
|
|
;; base name of the buffer; replaces the buffer name if managed by uniquify
|
|
,(and (fboundp 'uniquify-buffer-base-name) (uniquify-buffer-base-name))
|
|
;; basic information
|
|
,(desktop-file-name (buffer-file-name) desktop-dirname)
|
|
,(buffer-name)
|
|
,major-mode
|
|
;; minor modes
|
|
,(seq-filter
|
|
(lambda (minor-mode)
|
|
;; Just two sanity checks.
|
|
(and (boundp minor-mode)
|
|
(symbol-value minor-mode)
|
|
(let ((special
|
|
(assq minor-mode desktop-minor-mode-table)))
|
|
(or (not special)
|
|
(cadr special)))))
|
|
local-minor-modes)
|
|
;; point and mark, and read-only status
|
|
,(point)
|
|
,(list (mark t) mark-active)
|
|
,buffer-read-only
|
|
;; auxiliary information
|
|
,(when (functionp desktop-save-buffer)
|
|
(funcall desktop-save-buffer desktop-dirname))
|
|
;; local variables
|
|
,(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)))))
|
|
ll)
|
|
,@(when (>= desktop-io-file-version 208)
|
|
(list
|
|
(mapcar (lambda (record)
|
|
(let ((var (car record)))
|
|
(list var
|
|
(funcall (cadr record) (symbol-value var)))))
|
|
desktop-var-serdes-funs)))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop--v2s (value)
|
|
"Convert VALUE to a pair (QUOTE . SEXP); (eval SEXP) gives VALUE.
|
|
SEXP is an sexp that when evaluated yields VALUE.
|
|
QUOTE may be `may' (value may be quoted),
|
|
`must' (value must be quoted), or nil (value must not be quoted)."
|
|
(cond
|
|
((or (numberp value) (null value) (eq t value) (keywordp value))
|
|
(cons 'may value))
|
|
((stringp value)
|
|
;; Get rid of unreadable text properties.
|
|
(if (condition-case nil (read (format "%S" value)) (error nil))
|
|
(cons 'may value)
|
|
(let ((copy (copy-sequence value)))
|
|
(set-text-properties 0 (length copy) nil copy)
|
|
(cons 'may copy))))
|
|
((symbolp value)
|
|
(cons 'must value))
|
|
((vectorp value)
|
|
(let* ((pass1 (mapcar #'desktop--v2s value))
|
|
(special (assq nil pass1)))
|
|
(if special
|
|
(cons nil `(vector
|
|
,@(mapcar (lambda (el)
|
|
(if (eq (car el) 'must)
|
|
`',(cdr el) (cdr el)))
|
|
pass1)))
|
|
(cons 'may `[,@(mapcar #'cdr pass1)]))))
|
|
((and (recordp value) (symbolp (aref value 0)))
|
|
(let* ((pass1 (let ((res ()))
|
|
(dotimes (i (length value))
|
|
(push (desktop--v2s (aref value i)) res))
|
|
(nreverse res)))
|
|
(special (assq nil pass1)))
|
|
(if special
|
|
(cons nil `(record
|
|
,@(mapcar (lambda (el)
|
|
(if (eq (car el) 'must)
|
|
`',(cdr el) (cdr el)))
|
|
pass1)))
|
|
(cons 'may (apply #'record (mapcar #'cdr pass1))))))
|
|
((consp value)
|
|
(let ((p value)
|
|
newlist
|
|
use-list*)
|
|
(while (consp p)
|
|
(let ((q.sexp (desktop--v2s (car p))))
|
|
(push q.sexp newlist))
|
|
(setq p (cdr p)))
|
|
(when p
|
|
(let ((last (desktop--v2s p)))
|
|
(setq use-list* t)
|
|
(push last newlist)))
|
|
(if (assq nil newlist)
|
|
(cons nil
|
|
`(,(if use-list* 'desktop-list* 'list)
|
|
,@(mapcar (lambda (el)
|
|
(if (eq (car el) 'must)
|
|
`',(cdr el) (cdr el)))
|
|
(nreverse newlist))))
|
|
(cons 'must
|
|
`(,@(mapcar #'cdr
|
|
(nreverse (if use-list* (cdr newlist) newlist)))
|
|
,@(if use-list* (cdar newlist)))))))
|
|
((subrp value)
|
|
(cons nil `(symbol-function
|
|
',(intern-soft (substring (prin1-to-string value) 7 -1)))))
|
|
((markerp value)
|
|
(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"))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-value-to-string (value)
|
|
"Convert VALUE to a string that when read evaluates to the same value.
|
|
Not all types of values are supported."
|
|
(let* ((print-escape-newlines t)
|
|
(print-length nil)
|
|
(print-level nil)
|
|
(float-output-format nil)
|
|
(quote.sexp (desktop--v2s value))
|
|
(quote (car quote.sexp))
|
|
(print-quoted t)
|
|
(txt (prin1-to-string (cdr quote.sexp))))
|
|
(if (eq quote 'must)
|
|
(concat "'" txt)
|
|
txt)))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(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),
|
|
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."
|
|
(let (var size)
|
|
(if (consp varspec)
|
|
(setq var (car varspec) size (cdr varspec))
|
|
(setq var varspec))
|
|
(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"))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defvar desktop-buffers-not-to-save-function nil
|
|
"Function identifying buffers that are to be excluded from saving.
|
|
Like `desktop-buffers-not-to-save' it can be used to check whether
|
|
a given buffer should not be saved. It takes the same arguments as
|
|
`desktop-save-buffer-p' and should return nil if buffer should not
|
|
have its state saved in the desktop file.")
|
|
|
|
(defun desktop-save-buffer-p (filename bufname mode &rest rest)
|
|
"Return t if buffer should have its state saved in the desktop file.
|
|
FILENAME is the visited file name, BUFNAME is the buffer name, and
|
|
MODE is the major mode.
|
|
\n\(fn FILENAME BUFNAME MODE)"
|
|
(let ((case-fold-search nil)
|
|
(no-regexp-to-check (not (stringp desktop-files-not-to-save)))
|
|
dired-skip)
|
|
(and (or filename
|
|
(not (stringp desktop-buffers-not-to-save))
|
|
(not (string-match-p desktop-buffers-not-to-save bufname)))
|
|
(not (memq mode desktop-modes-not-to-save))
|
|
(or (and filename
|
|
(or no-regexp-to-check
|
|
(not (string-match-p desktop-files-not-to-save filename))))
|
|
(and (memq mode '(dired-mode vc-dir-mode))
|
|
(or no-regexp-to-check
|
|
(not (setq dired-skip
|
|
(with-current-buffer bufname
|
|
(string-match-p desktop-files-not-to-save
|
|
default-directory))))))
|
|
(and (null filename)
|
|
(null dired-skip) ; bug#5755
|
|
(with-current-buffer bufname desktop-save-buffer)))
|
|
(or (null desktop-buffers-not-to-save-function)
|
|
(funcall desktop-buffers-not-to-save-function
|
|
filename bufname mode rest))
|
|
t)))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(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))))
|
|
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop--check-dont-save (frame)
|
|
(not (frame-parameter frame 'desktop-dont-save)))
|
|
|
|
(defconst desktop--app-id `(desktop . ,desktop-file-version))
|
|
|
|
(defun desktop-save-frameset ()
|
|
"Save the state of existing frames in `desktop-saved-frameset'.
|
|
Frames with a non-nil `desktop-dont-save' parameter are not saved."
|
|
(setq desktop-saved-frameset
|
|
(and desktop-restore-frames
|
|
(frameset-save nil
|
|
:app desktop--app-id
|
|
:name (concat user-login-name "@" (system-name))
|
|
:predicate #'desktop--check-dont-save))))
|
|
|
|
;;;###autoload
|
|
(defun desktop-save (dirname &optional release only-if-changed version)
|
|
"Save the state of Emacs in a desktop file in directory DIRNAME.
|
|
Optional argument RELEASE non-nil says we're done with this
|
|
desktop, in which case this function releases the lock of the
|
|
desktop file in DIRNAME.
|
|
If ONLY-IF-CHANGED is non-nil, compare the current desktop
|
|
information to that in the desktop file, and if the desktop
|
|
information has not changed since it was last saved, then do
|
|
not rewrite the file.
|
|
|
|
To restore the desktop, use `desktop-read'.
|
|
|
|
This function can save the desktop in either format version
|
|
208 (which only Emacs 25.1 and later can read) or version
|
|
206 (which is readable by any Emacs from version 22.1 onwards).
|
|
By default, it will use the same format the desktop file had when
|
|
it was last saved, or version 208 when writing a fresh desktop
|
|
file.
|
|
|
|
To upgrade a version 206 file to version 208, call this command
|
|
explicitly with a prefix argument: \\[universal-argument] \\[desktop-save].
|
|
If you are upgrading from Emacs 24 or older, we recommend to do
|
|
this once you decide you no longer need compatibility with versions
|
|
of Emacs before 25.1.
|
|
|
|
To downgrade a version 208 file to version 206, use a double prefix
|
|
argument: \\[universal-argument] \\[universal-argument] \\[desktop-save].
|
|
|
|
Emacs will ask for confirmation when you upgrade or downgrade your
|
|
desktop file.
|
|
|
|
In a non-interactive call, VERSION can be given as an integer, either
|
|
206 or 208, to specify the format version in which to save the file,
|
|
no questions asked."
|
|
(interactive (list
|
|
;; Or should we just use (car desktop-path)?
|
|
(let ((default (if (member "." desktop-path)
|
|
default-directory
|
|
user-emacs-directory)))
|
|
(read-directory-name "Directory to save desktop file in: "
|
|
default default t))
|
|
nil
|
|
nil
|
|
current-prefix-arg))
|
|
(setq desktop-dirname (file-name-as-directory (expand-file-name dirname)))
|
|
(save-excursion
|
|
(let ((eager desktop-restore-eager)
|
|
(new-modtime (file-attribute-modification-time
|
|
(file-attributes (desktop-full-file-name)))))
|
|
(when
|
|
(or (not new-modtime) ; nothing to overwrite
|
|
(time-equal-p desktop-file-modtime new-modtime)
|
|
(yes-or-no-p (if desktop-file-modtime
|
|
(if (time-less-p desktop-file-modtime
|
|
new-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)))
|
|
|
|
;; What format are we going to write the file in?
|
|
(setq desktop-io-file-version
|
|
(cond
|
|
((equal version '(4))
|
|
(if (or (eq desktop-io-file-version 208)
|
|
(yes-or-no-p "Save desktop file in format 208 \
|
|
\(Readable by Emacs 25.1 and later only)? "))
|
|
208
|
|
(or desktop-io-file-version desktop-native-file-version)))
|
|
((equal version '(16))
|
|
(if (or (eq desktop-io-file-version 206)
|
|
(yes-or-no-p "Save desktop file in format 206 \
|
|
\(Readable by all Emacs versions since 22.1)? "))
|
|
206
|
|
(or desktop-io-file-version desktop-native-file-version)))
|
|
((memq version '(206 208))
|
|
version)
|
|
((null desktop-io-file-version) ; As yet, no desktop file exists.
|
|
desktop-native-file-version)
|
|
(t
|
|
desktop-io-file-version)))
|
|
|
|
(with-temp-buffer
|
|
(insert
|
|
";; -*- mode: emacs-lisp; lexical-binding:t; coding: utf-8-emacs; -*-\n"
|
|
desktop-header
|
|
";; Created " (current-time-string) "\n"
|
|
";; Desktop file format version " (format "%d" desktop-io-file-version) "\n"
|
|
";; Emacs version " emacs-version "\n")
|
|
(save-excursion (run-hooks 'desktop-save-hook))
|
|
(goto-char (point-max))
|
|
(insert "\n;; Global section:\n")
|
|
;; Called here because we save the window/frame state as a global
|
|
;; variable for compatibility with previous Emacsen.
|
|
(desktop-save-frameset)
|
|
(unless (memq 'desktop-saved-frameset desktop-globals-to-save)
|
|
(desktop-outvar 'desktop-saved-frameset))
|
|
(mapc #'desktop-outvar desktop-globals-to-save)
|
|
(setq desktop-saved-frameset nil) ; after saving desktop-globals-to-save
|
|
(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)))
|
|
(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")
|
|
" "
|
|
(format "%d" desktop-io-file-version))
|
|
;; 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))
|
|
(dolist (e l)
|
|
(insert "\n " (desktop-value-to-string e)))
|
|
(insert ")\n\n"))))
|
|
|
|
(setq default-directory desktop-dirname)
|
|
;; When auto-saving, avoid writing if nothing has changed since the last write.
|
|
(let* ((beg (and only-if-changed
|
|
(save-excursion
|
|
(goto-char (point-min))
|
|
;; Don't check the header with changing timestamp
|
|
(and (search-forward "Global section" nil t)
|
|
;; Also skip the timestamp in desktop-saved-frameset
|
|
;; if it's saved in the first non-header line
|
|
(search-forward "desktop-saved-frameset"
|
|
(line-beginning-position 3) t)
|
|
;; This is saved after the timestamp
|
|
(search-forward (format "%S" desktop--app-id) nil t))
|
|
(point))))
|
|
(checksum (and beg (md5 (current-buffer) beg (point-max) 'utf-8-emacs))))
|
|
(unless (and checksum (equal checksum desktop-file-checksum))
|
|
(let ((coding-system-for-write 'utf-8-emacs))
|
|
(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).
|
|
(desktop--get-file-modtime))))))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;;;###autoload
|
|
(defun desktop-remove ()
|
|
"Delete desktop file in `desktop-dirname'.
|
|
This function also sets `desktop-dirname' to nil."
|
|
(interactive)
|
|
(when desktop-dirname
|
|
(let ((filename (desktop-full-file-name)))
|
|
(setq desktop-dirname nil)
|
|
(when (file-exists-p filename)
|
|
(delete-file filename)))))
|
|
|
|
(defvar desktop-buffer-args-list nil
|
|
"List of args for `desktop-create-buffer'.")
|
|
|
|
(defvar desktop-lazy-timer nil)
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-restoring-frameset-p ()
|
|
"True if calling `desktop-restore-frameset' will actually restore it."
|
|
(and desktop-restore-frames desktop-saved-frameset
|
|
;; Don't restore frames when the selected frame is the daemon's
|
|
;; initial frame.
|
|
(not (and (daemonp) (not (frame-parameter nil 'client))))
|
|
t))
|
|
|
|
(defun desktop-restore-frameset ()
|
|
"Restore the state of a set of frames.
|
|
This function depends on the value of `desktop-saved-frameset'
|
|
being set (usually, by reading it from the desktop)."
|
|
(when (desktop-restoring-frameset-p)
|
|
(frameset-restore desktop-saved-frameset
|
|
:reuse-frames (eq desktop-restore-reuses-frames t)
|
|
:cleanup-frames (not (eq desktop-restore-reuses-frames 'keep))
|
|
:force-display desktop-restore-in-current-display
|
|
:force-onscreen (and desktop-restore-forces-onscreen
|
|
(display-graphic-p)))
|
|
;; When at least one restored frame contains a tab bar,
|
|
;; enable `tab-bar-mode' that takes care about recalculating
|
|
;; the correct values of the frame parameter `tab-bar-lines'
|
|
;; (that depends on `tab-bar-show'), and also loads graphical buttons.
|
|
(when (seq-some
|
|
(lambda (frame)
|
|
(menu-bar-positive-p (frame-parameter frame 'tab-bar-lines)))
|
|
(frame-list))
|
|
(tab-bar-mode 1))))
|
|
|
|
;; Just to silence the byte compiler.
|
|
;; Dynamically bound in `desktop-read'.
|
|
(defvar desktop-first-buffer)
|
|
(defvar desktop-buffer-ok-count)
|
|
(defvar desktop-buffer-fail-count)
|
|
|
|
;;;###autoload
|
|
(defun desktop-read (&optional dirname ask)
|
|
"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
|
|
is processed and `desktop-after-read-hook' is run. If no desktop file
|
|
is found, clear the desktop and run `desktop-no-desktop-file-hook'.
|
|
Interactively, with prefix arg \\[universal-argument], ask for DIRNAME.
|
|
This function is a no-op when Emacs is running in batch mode.
|
|
It returns t if a desktop file was loaded, nil otherwise.
|
|
\n(fn DIRNAME)"
|
|
(interactive "i\nP")
|
|
(if (or noninteractive
|
|
(and (desktop-owner)
|
|
(= (desktop-owner) (emacs-pid))))
|
|
(message "Not reloading the desktop%s"
|
|
(if noninteractive
|
|
""
|
|
"; already loaded"))
|
|
(setq desktop-dirname
|
|
(file-name-as-directory
|
|
(expand-file-name
|
|
(or
|
|
;; If DIRNAME is specified, use it.
|
|
(and (< 0 (length dirname)) dirname)
|
|
;; Else, with a prefix arg, ask for a directory name.
|
|
(and ask (read-directory-name "Directory for desktop file: " nil nil t))
|
|
;; 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))
|
|
;; Default: .emacs.d.
|
|
user-emacs-directory))))
|
|
(if (file-exists-p (desktop-full-file-name))
|
|
;; 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)
|
|
(desktop-autosave-was-enabled))
|
|
(if (and owner
|
|
(not (desktop--load-locked-desktop-p owner)))
|
|
(let ((default-directory desktop-dirname))
|
|
(setq desktop-dirname nil)
|
|
(run-hooks 'desktop-not-loaded-hook)
|
|
(unless desktop-dirname
|
|
(message "Desktop file in use; not loaded.")))
|
|
(desktop-lazy-abort)
|
|
;; Temporarily disable the autosave that will leave it
|
|
;; disabled when loading the desktop fails with errors,
|
|
;; thus not overwriting the desktop with broken contents.
|
|
(setq desktop-autosave-was-enabled
|
|
(memq #'desktop-auto-save-set-timer
|
|
;; Use the global value of the hook, in case some
|
|
;; feature makes window-configuration-change-hook
|
|
;; buffer-local, and puts there stuff which
|
|
;; doesn't include our timer.
|
|
(default-value
|
|
'window-configuration-change-hook)))
|
|
(desktop-auto-save-disable)
|
|
;; Evaluate desktop buffer and remember when it was modified.
|
|
(desktop--get-file-modtime)
|
|
(load (desktop-full-file-name) t t t)
|
|
;; If it wasn't already, mark it as in-use, to bother other
|
|
;; desktop instances.
|
|
(unless (eq (emacs-pid) owner)
|
|
(condition-case nil
|
|
(desktop-claim-lock)
|
|
(file-error (message "Couldn't record use of desktop file")
|
|
(sit-for 1))))
|
|
|
|
(unless (desktop-restoring-frameset-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))))
|
|
(run-hooks 'desktop-delay-hook)
|
|
(setq desktop-delay-hook nil)
|
|
(desktop-restore-frameset)
|
|
(run-hooks 'desktop-after-read-hook)
|
|
(message "Desktop: %s%d buffer%s restored%s%s."
|
|
(if desktop-saved-frameset
|
|
(let ((fn (length (frameset-states desktop-saved-frameset))))
|
|
(format "%d frame%s, "
|
|
fn (if (= fn 1) "" "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))
|
|
""))
|
|
(unless (desktop-restoring-frameset-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))))
|
|
(setq desktop-saved-frameset nil)
|
|
(if desktop-autosave-was-enabled (desktop-auto-save-enable))
|
|
t))
|
|
;; No desktop file found.
|
|
(let ((default-directory desktop-dirname))
|
|
(run-hooks 'desktop-no-desktop-file-hook))
|
|
(message "No desktop file.")
|
|
nil)))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;;;###autoload
|
|
(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)))
|
|
(desktop-kill)
|
|
(desktop-clear)
|
|
(desktop-read dirname))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;;;###autoload
|
|
(defun desktop-save-in-desktop-dir ()
|
|
"Save the desktop in directory `desktop-dirname'."
|
|
(interactive)
|
|
(if desktop-dirname
|
|
(desktop-save desktop-dirname)
|
|
(call-interactively 'desktop-save))
|
|
(message "Desktop saved in %s" (abbreviate-file-name desktop-dirname)))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;; Auto-Saving.
|
|
(defvar desktop-auto-save-timer nil)
|
|
|
|
(defun desktop-auto-save-enable (&optional timeout)
|
|
(when (and (integerp (or timeout desktop-auto-save-timeout))
|
|
(> (or timeout desktop-auto-save-timeout) 0))
|
|
(add-hook 'window-configuration-change-hook #'desktop-auto-save-set-timer)))
|
|
|
|
(defun desktop-auto-save-disable ()
|
|
(remove-hook 'window-configuration-change-hook #'desktop-auto-save-set-timer)
|
|
(desktop-auto-save-cancel-timer))
|
|
|
|
(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)))
|
|
|
|
(defun desktop-auto-save-set-timer ()
|
|
"Set the desktop auto-save timer.
|
|
Cancel any previous timer. When `desktop-auto-save-timeout' is a positive
|
|
integer, start a new idle timer to call `desktop-auto-save' after that many
|
|
seconds of idle time.
|
|
This function is called from `window-configuration-change-hook'."
|
|
(desktop-auto-save-cancel-timer)
|
|
(when (and (integerp desktop-auto-save-timeout)
|
|
(> desktop-auto-save-timeout 0))
|
|
(setq desktop-auto-save-timer
|
|
(run-with-idle-timer desktop-auto-save-timeout nil
|
|
'desktop-auto-save))))
|
|
|
|
(defun desktop-auto-save-cancel-timer ()
|
|
(when desktop-auto-save-timer
|
|
(cancel-timer desktop-auto-save-timer)
|
|
(setq desktop-auto-save-timer nil)))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;;;###autoload
|
|
(defun desktop-revert ()
|
|
"Revert to the last loaded desktop."
|
|
(interactive)
|
|
(unless desktop-dirname
|
|
(error "Unknown desktop directory"))
|
|
(unless (file-exists-p (desktop-full-file-name))
|
|
(error "No desktop file found"))
|
|
(desktop-clear)
|
|
(desktop-read desktop-dirname))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-access-file (filename)
|
|
"Check whether FILENAME is accessible."
|
|
(ignore-errors (not (access-file filename "Restoring desktop buffer"))))
|
|
|
|
(defvar desktop-buffer-major-mode)
|
|
(defvar desktop-buffer-locals)
|
|
(defvar auto-insert) ; from autoinsert.el
|
|
;; ----------------------------------------------------------------------------
|
|
(defun desktop-restore-file-buffer (buffer-filename
|
|
_buffer-name
|
|
_buffer-misc)
|
|
"Restore a file buffer."
|
|
(when buffer-filename
|
|
(if (or (desktop-access-file buffer-filename)
|
|
(let ((msg (format "Desktop: File \"%s\" no longer accessible."
|
|
buffer-filename)))
|
|
(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))))
|
|
(buf (find-file-noselect buffer-filename :nowarn)))
|
|
(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)))
|
|
|
|
(defun desktop-load-file (function)
|
|
"Load the file where auto loaded FUNCTION is defined.
|
|
If FUNCTION is not currently defined, guess the library that defines it
|
|
and try to load that."
|
|
(if (fboundp function)
|
|
(autoload-do-load (symbol-function function) function)
|
|
;; Guess that foobar-mode is defined in foobar.
|
|
;; TODO rather than guessing or requiring an autoload, the desktop
|
|
;; file should record the name of the library.
|
|
(let ((name (symbol-name function)))
|
|
(if (string-match "\\`\\(.*\\)-mode\\'" name)
|
|
(with-demoted-errors "Require error in desktop-load-file: %S"
|
|
(require (intern (match-string 1 name)) nil t))))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;; Create a buffer, load its file, set its mode, ...;
|
|
;; called from Desktop file only.
|
|
|
|
(defun desktop-create-buffer
|
|
(file-version
|
|
buffer-filename
|
|
buffer-name
|
|
buffer-majormode
|
|
buffer-minormodes
|
|
buffer-point
|
|
buffer-mark
|
|
buffer-readonly
|
|
buffer-misc
|
|
&optional
|
|
buffer-locals
|
|
compacted-vars
|
|
&rest _unsupported)
|
|
|
|
(setq desktop-io-file-version file-version)
|
|
|
|
(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
|
|
(condition-case-unless-debug err
|
|
(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
|
|
;; the order for old desktop files (old desktop module behavior).
|
|
(unless (< desktop-file-version 206)
|
|
(dolist (buf buffer-list)
|
|
(and (buffer-live-p buf)
|
|
(bury-buffer buf)))
|
|
(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)
|
|
(progn
|
|
(move-marker (mark-marker) (car desktop-buffer-mark))
|
|
(if (car (cdr desktop-buffer-mark))
|
|
(activate-mark 'dont-touch-tmm)))
|
|
(move-marker (mark-marker) 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))
|
|
(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)))
|
|
;; adjust `buffer-display-time' for the downtime. e.g.,
|
|
;; * if `buffer-display-time' was 8:00
|
|
;; * and emacs stopped at `desktop-file-modtime' == 11:00
|
|
;; * and we are loading the desktop file at (current-time) 12:30,
|
|
;; -> then we restore `buffer-display-time' as 9:30,
|
|
;; for the sake of `clean-buffer-list': preserving the invariant
|
|
;; "how much time the user spent in Emacs without looking at this buffer".
|
|
(setq buffer-display-time
|
|
(time-since (if buffer-display-time
|
|
(time-subtract desktop-file-modtime
|
|
buffer-display-time)
|
|
0)))
|
|
(unless (< desktop-file-version 208) ; Don't misinterpret any old custom args
|
|
(dolist (record compacted-vars)
|
|
(let*
|
|
((var (car record))
|
|
(deser-fun (nth 2 (assq var desktop-var-serdes-funs))))
|
|
(if deser-fun (set var (funcall deser-fun (cadr record))))))))
|
|
result))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;; Backward compatibility -- update parameters to 205 standards.
|
|
(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
|
|
(list (cons 'truncate-lines tl)
|
|
(cons 'fill-column fc)
|
|
(cons 'case-fold-search cfs)
|
|
(cons 'case-replace cr)
|
|
(cons 'overwrite-mode (car mim)))))
|
|
|
|
(defun desktop-append-buffer-args (&rest args)
|
|
"Append ARGS at end of `desktop-buffer-args-list'.
|
|
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
|
|
(message "%s" msg))
|
|
(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)
|
|
(when (called-interactively-p 'interactive)
|
|
(message "Lazy desktop load aborted"))))
|
|
|
|
;; ----------------------------------------------------------------------------
|
|
;; When `desktop-save-mode' is non-nil and "--no-desktop" is not specified on the
|
|
;; 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
|
|
(lambda ()
|
|
(let ((key "--no-desktop"))
|
|
(when (member key command-line-args)
|
|
(setq command-line-args (delete key command-line-args))
|
|
(desktop-save-mode 0)))
|
|
(when desktop-save-mode
|
|
(desktop-read)
|
|
(setq inhibit-startup-screen t))))
|
|
|
|
(provide 'desktop)
|
|
|
|
;;; desktop.el ends here
|