;; proof-compat.el Operating system and Emacs version compatibility ;; ;; Copyright (C) 2000-2002 LFCS Edinburgh. ;; Author: David Aspinall and others ;; License: GPL (GNU GENERAL PUBLIC LICENSE) ;; ;; $Id$ ;; ;; This file collects together compatibility hacks for different ;; operating systems and Emacs versions. This is to help keep ;; track of them. ;; ;; The development policy for Proof General (since v3.7) is for the ;; main codebase to be written for the latest stable version of GNU ;; Emacs, following GNU Emacs advice on obsolete function calls. ;; (eval-when-compile (require 'cl) (require 'easymenu)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; Architecture flags ;;; ;; can use eval-and-compile to allow optimisation, but that would ;; require recompilation for Windows (defvar proof-running-on-win32 (memq system-type '(win32 windows-nt cygwin)) "Non-nil if Proof General is running on a windows variant system.") ;; Workaround a small bug in Carbon Emacs Winter 2008 (at least) ;; Menu presses query this variable, but it's not bound unless ;; mode engaged. Not noticeable in normal use, but it is as soon ;; as debug-on-error is engaged. (if (and (boundp 'carbon-emacs-package-version) (not (boundp 'mac-key-mode))) (setq mac-key-mode nil)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; Emacs and XEmacs modifications and adjustments ;;; ;; Remove a custom setting. Needed to support dynamic reconfiguration. ;; (We'd prefer that repeated defcustom calls acted like repeated ;; "defvar treated as defconst" in XEmacs) (defun pg-custom-undeclare-variable (symbol) "Remove a custom setting SYMBOL. Done by `makunbound' and removing all properties mentioned by custom library." (mapcar (lambda (prop) (remprop symbol prop)) '(default standard-value force-value variable-comment saved-variable-comment variable-documentation group-documentation custom-set custom-get custom-options custom-requests custom-group custom-prefix custom-tag custom-links custom-version saved-value theme-value theme-face)) (makunbound symbol)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; XEmacs compatibility ;;; ;; browse-url function isn't autoloaded in XEmacs 20.4 (or (fboundp 'browse-url) (autoload 'browse-url "browse-url" "Ask a WWW browser to load URL." t)) ;; executable-find isn't autoloaded in XEmacs 21.4.6 (or (fboundp 'executable-find) (autoload 'executable-find "executable" "\ Search for COMMAND in exec-path and return the absolute file name. Return nil if COMMAND is not found anywhere in `exec-path'." nil nil)) ;; Compatibility with XEmacs 20.3 (or (fboundp 'split-path) (defun split-path (path) "Explode a search path into a list of strings. The path components are separated with the characters specified with `path-separator'." (split-string path (regexp-quote path-separator)))) ;; Compatibility with XEmacs 21.4, API change in add-hook (when (and (featurep 'xemacs) (eq emacs-major-version 21) (<= emacs-minor-version 4)) (fset 'old-add-hook (symbol-function 'add-hook)) (defun add-hook (hook function &optional append local) "Add to the value of HOOK the function FUNCTION. FUNCTION is not added if already present. FUNCTION is added (if necessary) at the beginning of the hook list unless the optional argument APPEND is non-nil, in which case FUNCTION is added at the end. The optional fourth argument, LOCAL, if non-nil, says to modify the hook's buffer-local value rather than its default value. This makes the hook buffer-local if needed. To make a hook variable buffer-local, always use `make-local-hook', not `make-local-variable'. HOOK should be a symbol, and FUNCTION may be any valid function. If HOOK is void, it is first set to nil. If HOOK's value is a single function, it is changed to a list of functions. You can remove this hook yourself using `remove-hook'. See also `add-one-shot-hook'." (if local (make-local-hook hook)) (old-add-hook hook function append local))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; XEmacs compatibility with GNU Emacs ;;; (or (fboundp 'subst-char-in-string) ;; Code is taken from Emacs 21.2.1/subr.el (defun subst-char-in-string (fromchar tochar string &optional inplace) "Replace FROMCHAR with TOCHAR in STRING each time it occurs. Unless optional argument INPLACE is non-nil, return a new string." (let ((i (length string)) (newstr (if inplace string (copy-sequence string)))) (while (> i 0) (setq i (1- i)) (if (eq (aref newstr i) fromchar) (aset newstr i tochar))) newstr))) ;; Required by xmltok.el [not used at present], proof-shell.el (or (fboundp 'replace-regexp-in-string) ;; Code is taken from Emacs 21.1.1/subr.el. Now in XEmacs (21.5b28, at least) (defun replace-regexp-in-string (regexp rep string &optional fixedcase literal subexp start) "Replace all matches for REGEXP with REP in STRING. Return a new string containing the replacements. Optional arguments FIXEDCASE, LITERAL and SUBEXP are like the arguments with the same names of function `replace-match'. If START is non-nil, start replacements at that index in STRING. REP is either a string used as the NEWTEXT arg of `replace-match' or a function. If it is a function it is applied to each match to generate the replacement passed to `replace-match'; the match-data at this point are such that match 0 is the function's argument. To replace only the first match (if any), make REGEXP match up to \\' and replace a sub-expression, e.g. (replace-regexp-in-string \"\\(foo\\).*\\'\" \"bar\" \" foo foo\" nil nil 1) => \" bar foo\" " ;; To avoid excessive consing from multiple matches in long strings, ;; don't just call `replace-match' continually. Walk down the ;; string looking for matches of REGEXP and building up a (reversed) ;; list MATCHES. This comprises segments of STRING which weren't ;; matched interspersed with replacements for segments that were. ;; [For a `large' number of replacments it's more efficient to ;; operate in a temporary buffer; we can't tell from the function's ;; args whether to choose the buffer-based implementation, though it ;; might be reasonable to do so for long enough STRING.] (let ((l (length string)) (start (or start 0)) matches str mb me) (save-match-data (while (and (< start l) (string-match regexp string start)) (setq mb (match-beginning 0) me (match-end 0)) ;; If we matched the empty string, make sure we advance by one char (when (= me mb) (setq me (min l (1+ mb)))) ;; Generate a replacement for the matched substring. ;; Operate only on the substring to minimize string consing. ;; Set up match data for the substring for replacement; ;; presumably this is likely to be faster than munging the ;; match data directly in Lisp. (string-match regexp (setq str (substring string mb me))) (setq matches (cons (replace-match (if (stringp rep) rep (funcall rep (match-string 0 str))) fixedcase literal str subexp) (cons (substring string start mb) ; unmatched prefix matches))) (setq start me)) ;; Reconstruct a string from the pieces. (setq matches (cons (substring string start l) matches)) ; leftover (apply #'concat (nreverse matches)))))) ;; The GNU Emacs implementation of easy-menu-define has a very handy ;; :visible keyword. To use that when it's available, we set a ;; constant to be :visible or :active (defconst menuvisiblep (if (featurep 'xemacs) :active :visible) ":visible (on GNU Emacs) or :active (otherwise). The GNU Emacs implementation of easy-menu-define has a very handy :visible keyword. To use that when it's available, we use this constant.") (or (fboundp 'frame-parameter) (defalias 'frame-parameter 'frame-property)) (or (boundp 'window-size-fixed) (defvar window-size-fixed nil "Fudged version of GNU Emacs' setting. Completely ignored.")) (or (fboundp 'window-text-height) (defalias 'window-text-height 'window-text-area-height)) (or (fboundp 'set-window-text-height) (defun set-window-text-height (window height) "Sets the height in lines of the text display area of WINDOW to HEIGHT. This doesn't include the mode-line (or header-line if any) or any partial-height lines in the text display area. If WINDOW is nil, the selected window is used. Note that the current implementation of this function cannot always set the height exactly, but attempts to be conservative, by allocating more lines than are actually needed in the case where some error may be present." (let ((delta (- height (window-text-height window)))) (unless (zerop delta) (let ((window-min-height 1)) (if (and window (not (eq window (selected-window)))) (save-selected-window (select-window window) (enlarge-window delta)) (enlarge-window delta))))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; GNU Emacs compatibility with XEmacs ;;; (unless (fboundp 'save-selected-frame) (defmacro save-selected-frame (&rest body) "Execute forms in BODY, then restore the selected frame. The value returned is the value of the last form in BODY." (let ((old-frame (gensym "ssf"))) `(let ((,old-frame (selected-frame))) (unwind-protect (progn ,@body) (select-frame ,old-frame)))))) ;; Chars (borrowed from x-symbol-emacs.el compatability file) (unless (fboundp 'characterp) (defalias 'characterp 'integerp)) (unless (fboundp 'int-to-char) (defalias 'int-to-char 'identity)) (unless (fboundp 'char-to-int) (defalias 'char-to-int 'identity)) ;; Missing function, but anyway Emacs has no datatype for events... (unless (fboundp 'events-to-keys) (defalias 'events-to-keys 'identity)) (unless (fboundp 'region-exists-p) (defun region-exists-p () mark-active)) ;; completion not autoloaded in GNU 20.6.1; we must call ;; dynamic-completion-mode after loading it. (or (fboundp 'complete) (autoload 'complete "completion")) (unless (featurep 'xemacs) (eval-after-load "completion" '(dynamic-completion-mode))) ;; These days cl is dumped with XEmacs (20.4,21.1) but not GNU Emacs ;; 20.2. Would rather it was autoloaded but the autoloads are broken ;; in GNU so we load it now. (require 'cl) ;; Give a warning, (or (fboundp 'warn) (defun warn (str &rest args) "Issue a warning STR. Defined by PG for GNU compatibility." (apply 'message str args) (sit-for 2))) ;; Modeline redrawing (actually force-mode-line-update is alias on XEmacs) (or (fboundp 'redraw-modeline) (defun redraw-modeline (&rest args) "Dummy function for Proof General on GNU Emacs." (force-mode-line-update))) ;; Replace in string: XEmacs original now in GNU Emacs as replace-regexp-in-string (or (fboundp 'replace-in-string) (defun replace-in-string (str regexp newtext &optional literal) (replace-regexp-in-string regexp newtext str 'fixedcase literal))) ;; An implemenation of buffer-syntactic-context for GNU Emacs (defun proof-buffer-syntactic-context-emulate (&optional buffer) "Return the syntactic context of BUFFER at point. If BUFFER is nil or omitted, the current buffer is assumed. The returned value is one of the following symbols: nil ; meaning no special interpretation string ; meaning point is within a string comment ; meaning point is within a line comment" (save-excursion (if buffer (set-buffer buffer)) (let ((pp (parse-partial-sexp (point-min) (point)))) (cond ((nth 3 pp) 'string) ;; ((nth 7 pp) 'block-comment) ;; "Stefan Monnier" suggests ;; distinguishing between block comments and ordinary comments ;; is problematic: not what XEmacs claims and different to what ;; (nth 7 pp) tells us in GNU Emacs. ((nth 4 pp) 'comment))))) ;; In case Emacs is not aware of the function read-shell-command, ;; we duplicate some code adjusted from minibuf.el distributed ;; with XEmacs 21.1.9 ;; ;; This code is still required as of GNU Emacs 20.6.1 ;; ;; da: I think bothering with this just to give completion for ;; when proof-prog-name-ask=t is rather a big overkill! ;; Still, now it's here we'll leave it in as a pleasant surprise ;; for GNU Emacs users. ;; (or (fboundp 'read-shell-command) (defvar read-shell-command-map (let ((map (make-sparse-keymap 'read-shell-command-map))) (if (fboundp 'set-keymap-parents) ;; XEmacs versions without read-shell-command? (set-keymap-parents map minibuffer-local-map) (if (fboundp 'set-keymap-parent) ;; GNU Emacs 20.2 (set-keymap-parent map minibuffer-local-map) ;; Earlier GNU Emacs (setq map (append minibuffer-local-map map)))) (define-key map "\t" 'comint-dynamic-complete) (define-key map "\M-\t" 'comint-dynamic-complete) (define-key map "\M-?" 'comint-dynamic-list-completions) map) "Minibuffer keymap used by `shell-command' and related commands.")) (or (fboundp 'read-shell-command) (defun read-shell-command (prompt &optional initial-input history) "Just like read-string, but uses read-shell-command-map: \\{read-shell-command-map}" (let ((minibuffer-completion-table nil)) (read-from-minibuffer prompt initial-input read-shell-command-map nil (or history 'shell-command-history))))) ;; Emulate a useful builtin from XEmacs. (or (fboundp 'remassq) ;; NB: Emacs has assoc package with assq-delete-all function (defun remassq (key alist) "Delete any elements of ALIST whose car is `eq' to KEY. The modified ALIST is returned." ;; The builtin version deletes by side-effect, but don't bother here. (let (newalist) (while alist (unless (eq key (caar alist)) (setq newalist (cons (car alist) newalist))) (setq alist (cdr alist))) (nreverse newalist)))) (or (fboundp 'remassoc) (defun remassoc (key alist) "Delete any elements of ALIST whose car is `equal' to KEY. The modified ALIST is returned." ;; The builtin version deletes by side-effect, but don't bother here. (let (newalist) (while alist (unless (equal key (caar alist)) (setq newalist (cons (car alist) newalist))) (setq alist (cdr alist))) (nreverse newalist)))) (or (fboundp 'frames-of-buffer) ;; From XEmacs 21.4.12, aliases expanded (defun frames-of-buffer (&optional buffer visible-only) "Return list of frames that BUFFER is currently being displayed on. If the buffer is being displayed on the currently selected frame, that frame is first in the list. VISIBLE-ONLY will only list non-iconified frames." (let ((list (get-buffer-window-list buffer)) (cur-frame (selected-frame)) next-frame frames save-frame) (while list (if (memq (setq next-frame (window-frame (car list))) frames) nil (if (eq cur-frame next-frame) (setq save-frame next-frame) (and (or (not visible-only) (frame-visible-p next-frame)) (setq frames (append frames (list next-frame)))))) (setq list (cdr list))) (if save-frame (append (list save-frame) frames) frames)))) ;; These functions are used in the intricate logic around ;; shrink-to-fit. ;; window-leftmost-p, window-rightmost-p: my implementations (or (fboundp 'window-leftmost-p) (defun window-leftmost-p (window) (zerop (car (window-edges window))))) (or (fboundp 'window-rightmost-p) (defun window-rightmost-p (window) (>= (nth 2 (window-edges window)) (frame-width (window-frame window))))) ;; with-selected-window from XEmacs 21.4.12 (or (fboundp 'with-selected-window) (defmacro with-selected-window (window &rest body) "Execute forms in BODY with WINDOW as the selected window. The value returned is the value of the last form in BODY." `(save-selected-window (select-window ,window) ,@body))) ;; find-coding-system emulation for GNU Emacs (unless (fboundp 'find-coding-system) (defun find-coding-system (name) "Retrieve the coding system of the given name, or nil if non-such." (condition-case nil (check-coding-system name) (error nil)))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; A naughty hack to completion.el ;;; ;;; At the moment IMO completion too eagerly adds stuff to ;;; its database: the completion-before-command function ;;; makes every suffix be added as a completion! (eval-after-load "completion" '(defun completion-before-command () (if (and (symbolp this-command) (get this-command 'completion-function)) (funcall (get this-command 'completion-function))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; Attempt to harmonise pop-to-buffer behaviour ;;; (or (featurep 'xemacs) ;; NB: GNU Emacs version has fewer args (defalias 'pg-pop-to-buffer 'pop-to-buffer)) (if (featurep 'xemacs) ;; Version from XEmacs 21.4.12, with args to match GNU Emacs ;; NB: GNU Emacs version has fewer args, we don't use ON-FRAME (defun pg-pop-to-buffer (bufname &optional not-this-window-p no-record on-frame) "Select buffer BUFNAME in some window, preferably a different one. If BUFNAME is nil, then some other buffer is chosen. If `pop-up-windows' is non-nil, windows can be split to do this. If optional second arg NOT-THIS-WINDOW-P is non-nil, insist on finding another window even if BUFNAME is already visible in the selected window. If optional fourth arg is non-nil, it is the frame to pop to this buffer on. If optional third arg is non-nil, do not record this in switching history. (addition for PG). If `focus-follows-mouse' is non-nil, keyboard focus is left unchanged." (let ((oldbuf (current-buffer)) buf window frame) (if (null bufname) (setq buf (other-buffer (current-buffer))) (setq buf (get-buffer bufname)) (if (null buf) (progn (setq buf (get-buffer-create bufname)) (set-buffer-major-mode buf)))) (push-window-configuration) (set-buffer buf) (setq window (display-buffer buf not-this-window-p on-frame)) (setq frame (window-frame window)) ;; if the display-buffer hook decided to show this buffer in another ;; frame, then select that frame, (unless obeying focus-follows-mouse -sb). (if (and (not focus-follows-mouse) (not (eq frame (selected-frame)))) (select-frame frame)) (unless no-record (record-buffer buf)) (if (and focus-follows-mouse on-frame (not (eq on-frame (selected-frame)))) (set-buffer oldbuf) ;; select-window will modify the internal keyboard focus of XEmacs (select-window window)) buf)) ) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; Old Emacs version compatibility (to be gradually removed...) ;;; ;; Create a menu from a customize group, for older/non-existent customize (or (fboundp 'process-live-p) (defun process-live-p (obj) "Return t if OBJECT is a process that is alive" (and (processp obj) (memq (process-status obj) '(open run stop))))) (or (fboundp 'buffer-substring-no-properties) (defalias 'buffer-substring-no-properties 'buffer-substring)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; General Emacs version compatibility ;;; ;; Handle buggy buffer-syntactic-context workaround in XEmacs, ;; and GNU Emacs non-implementation. ;; Latest: block comment is unreliable still in XEmacs 21.5, ;; doesn't seem worth attempting to use the native function at all. (defalias 'proof-buffer-syntactic-context 'proof-buffer-syntactic-context-emulate) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; Nasty: Emacs bug/problem fix section ;;; ;; NB: customize-menu-create is buggy in some versions of GNU Emacs ;; (bad in 21.1.0, good in 21.1.1, bad in 21.2.1, ...). Comment ;; these next lines out if you must use one of these versions. ;; PG 3.5.1: add hack in proof-compat.el to deal with this (if (and (not (featurep 'xemacs)) (or (string-equal emacs-version "21.2.1") (string-equal emacs-version "21.1.0"))) (defun customize-menu-create (symbol &optional name) (cons (or name "Customize") (list ["Your version of Emacs is buggy; update to get this menu" '(w3-goto-url "http://www.gnu.org/software/emacs/") t])))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;; ;;; Tweak to XEmacs buffer tabs (not really compatibility) ;;; ;; We remove PG auxiliary buffers from tabs. ;; TODO: complain to XEmacs developers about overly generous matching ;; in this function. In XEmacs post 21.5 one can set names of buffers ;; to omit just from tabs list. (if (featurep 'xemacs) (progn (fset 'select-buffers-tab-buffers-by-mode-old (symbol-function 'select-buffers-tab-buffers-by-mode)) (defun select-buffers-tab-buffers-by-mode (buf1 buf2) (let* ((mode1 (symbol-value-in-buffer 'major-mode buf1)) ;; candidate buf (mode2 (symbol-value-in-buffer 'major-mode buf2)) ;; displayed buf (auxes '(proof-goals-mode proof-shell-mode proof-response-mode)) (mode1aux (memq (get mode1 'derived-mode-parent) auxes)) (mode2aux (memq (get mode2 'derived-mode-parent) auxes))) (cond (mode1aux mode2aux) (mode2aux nil) (t (select-buffers-tab-buffers-by-mode-old buf1 buf2))))) )) ;; end XEmacs featurep ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; Workaround GNU Emacs problems in easymenu-add ;; (if (not (featurep 'xemacs)) ;; This has a nasty side effect of removing accelerators ;; from existing menus when easy-menu-add is called. ;; Problem confirmed in versions: 21.4.1, OK: 22.1.1 (or (< emacs-major-version 22) (setq easy-menu-precalculate-equivalent-keybindings nil))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; ;; XEmacs/x-symbol compatibility for XEmacs 21.5 ;; ;; See http://thread.gmane.org/gmane.emacs.xemacs.beta/20171/focus=20172 (if (and (fboundp 'valid-specifier-tag-p) (not (valid-specifier-tag-p 'mule-fonts))) (if (fboundp 'define-specifier-tag) (define-specifier-tag 'mule-fonts))) ;; End of proof-compat.el (provide 'proof-compat)