1 ;;; gnus-ems.el --- functions for making Semi-gnus work under different Emacsen
2 ;; Copyright (C) 1995, 1996, 1997, 1998, 1999, 2000, 2001
3 ;; Free Software Foundation, Inc.
5 ;; Author: Lars Magne Ingebrigtsen <larsi@gnus.org>
6 ;; Tatsuya Ichikawa <t-ichi@niagara.shiojiri.ne.jp>
9 ;; This file is part of GNU Emacs.
11 ;; GNU Emacs is free software; you can redistribute it and/or modify
12 ;; it under the terms of the GNU General Public License as published by
13 ;; the Free Software Foundation; either version 2, or (at your option)
16 ;; GNU Emacs is distributed in the hope that it will be useful,
17 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
18 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 ;; GNU General Public License for more details.
21 ;; You should have received a copy of the GNU General Public License
22 ;; along with GNU Emacs; see the file COPYING. If not, write to the
23 ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
24 ;; Boston, MA 02111-1307, USA.
34 ;;; Function aliases later to be redefined for XEmacs usage.
36 (defvar gnus-mouse-2 [mouse-2])
37 (defvar gnus-down-mouse-3 [down-mouse-3])
38 (defvar gnus-down-mouse-2 [down-mouse-2])
39 (defvar gnus-widget-button-keymap nil)
40 (defvar gnus-mode-line-modified
41 (if (or (featurep 'xemacs)
42 (< emacs-major-version 20))
47 (autoload 'gnus-xmas-define "gnus-xmas")
48 (autoload 'gnus-xmas-redefine "gnus-xmas")
49 (autoload 'appt-select-lowest-window "appt"))
51 (if (featurep 'xemacs)
52 (autoload 'gnus-smiley-display "smiley")
53 (autoload 'gnus-smiley-display "smiley-ems")) ; override XEmacs version
55 (defun gnus-kill-all-overlays ()
56 "Delete all overlays in the current buffer."
57 (let* ((overlayss (overlay-lists))
58 (buffer-read-only nil)
59 (overlays (delq nil (nconc (car overlayss) (cdr overlayss)))))
61 (delete-overlay (pop overlays)))))
66 (if (featurep 'xemacs)
68 (defvar gnus-mouse-face-prop 'mouse-face
69 "Property used for highlighting mouse regions.")))
72 (let ((case-fold-search t))
74 ((string-match "windows-nt\\|os/2\\|emx\\|cygwin32"
75 (symbol-name system-type))
76 (setq nnheader-file-name-translation-alist
77 (append nnheader-file-name-translation-alist
78 (mapcar (lambda (c) (cons c ?_))
79 '(?: ?* ?\" ?< ?> ??))
82 (defvar gnus-tmp-unread)
83 (defvar gnus-tmp-replied)
84 (defvar gnus-tmp-score-char)
85 (defvar gnus-tmp-indentation)
86 (defvar gnus-tmp-opening-bracket)
87 (defvar gnus-tmp-lines)
88 (defvar gnus-tmp-name)
89 (defvar gnus-tmp-closing-bracket)
90 (defvar gnus-tmp-subject-or-nil)
91 (defvar gnus-check-before-posting)
93 (defun gnus-ems-redefine ()
99 ;; Mule and new Emacs definitions
101 ;; [Note] Now there are three kinds of mule implementations,
102 ;; original MULE, XEmacs/mule and Emacs 20+ including
103 ;; MULE features. Unfortunately these API are different. In
104 ;; particular, Emacs (including original MULE) and XEmacs are
105 ;; quite different. However, this version of Gnus doesn't support
106 ;; anything other than XEmacs 20+ and Emacs 20.3+.
108 ;; Predicates to check are following:
109 ;; (boundp 'MULE) is t only if MULE (original; anything older than
110 ;; Mule 2.3) is running.
111 ;; (featurep 'mule) is t when every mule variants are running.
113 ;; It is possible to detect XEmacs/mule by (featurep 'mule) and
114 ;; checking `emacs-version'. In this case, the implementation for
115 ;; XEmacs/mule may be shareable between XEmacs and XEmacs/mule.
117 (defvar gnus-summary-display-table nil
118 "Display table used in summary mode buffers.")
120 (defalias 'gnus-summary-set-display-table (lambda ()))
122 (if (fboundp 'truncate-string-to-width)
123 (fset 'gnus-truncate-string 'truncate-string-to-width)
124 (fset 'gnus-truncate-string 'truncate-string))
126 (when (boundp 'gnus-check-before-posting)
127 (setq gnus-check-before-posting
129 (delq 'control-chars gnus-check-before-posting))))
131 (when (featurep 'mule)
132 (defun gnus-tilde-max-form (el max-width)
133 "Return a form that limits EL to MAX-WIDTH."
134 (let ((max (abs max-width)))
137 `(let ((width (string-width ,el)))
138 (gnus-truncate-string ,el width (- width ,max)))
139 `(gnus-truncate-string ,el ,max))
141 `(let* ((val (eval ,el))
142 (width (string-width val)))
143 (gnus-truncate-string val width (- width ,max)))
144 `(let ((val (eval ,el)))
145 (gnus-truncate-string val ,max))))))
147 (defun gnus-tilde-cut-form (el cut-width)
148 "Return a form that cuts CUT-WIDTH off of EL."
149 (let ((cut (abs cut-width)))
152 `(gnus-truncate-string ,el (- (string-width ,el) ,cut))
153 `(gnus-truncate-string ,el (string-width ,el) ,cut))
155 `(let ((val (eval ,el)))
156 (gnus-truncate-string val (- (string-width val) ,cut)))
157 `(let ((val (eval ,el)))
158 (gnus-truncate-string val (string-width val) ,cut))))))
161 (defun gnus-region-active-p ()
162 "Say whether the region is active."
163 (and (boundp 'transient-mark-mode)
165 (boundp 'mark-active)
168 (if (fboundp 'add-minor-mode)
169 (defalias 'gnus-add-minor-mode 'add-minor-mode)
170 (defun gnus-add-minor-mode (mode name map &rest rest)
171 (set (make-local-variable mode) t)
172 (unless (assq mode minor-mode-alist)
173 (push `(,mode ,name) minor-mode-alist))
174 (unless (assq mode minor-mode-map-alist)
175 (push (cons mode map)
176 minor-mode-map-alist))))
178 (defun gnus-x-splash ()
179 "Show a splash screen using a pixmap in the current buffer."
180 (let ((dir (nnheader-find-etc-directory "gnus"))
181 pixmap file height beg i)
183 (switch-to-buffer (gnus-get-buffer-create gnus-group-buffer))
184 (let ((buffer-read-only nil)
188 (file-exists-p (setq file
189 (expand-file-name "x-splash" dir))))
191 (insert-file-contents-as-binary file)
192 (goto-char (point-min))
194 (setq pixmap (read (current-buffer))))))
196 (make-face 'gnus-splash)
197 (setq height (/ (car pixmap) (frame-char-height))
198 width (/ (cadr pixmap) (frame-char-width)))
199 (set-face-foreground 'gnus-splash "Brown")
200 (set-face-stipple 'gnus-splash pixmap)
201 (insert-char ?\n (* (/ (window-height) 2 height) height))
204 (insert-char ?\ (* (/ (window-width) 2 width) width))
206 (insert-char ?\ width)
207 (set-text-properties beg (point) '(face gnus-splash))
210 (goto-char (point-min))
213 (defvar gnus-article-xface-ring-internal nil
214 "Cache for face data.")
216 ;; Worth customizing?
217 (defvar gnus-article-xface-ring-size 6
218 "Length of the ring used for `gnus-article-xface-ring-internal'.")
220 (defvar gnus-article-compface-xbm
222 (eq 0 (string-match "#define"
223 (shell-command-to-string "uncompface -X")))
225 "Non-nil means the compface program supports the -X option.
226 That produces XBM output.")
228 (defun gnus-article-display-xface (beg end &optional buffer)
229 "Display an XFace header from between BEG and END in BUFFER.
230 Requires support for images in your Emacs and the external programs
231 `uncompface', and `icontopbm'. On a GNU/Linux system these
232 might be in packages with names like `compface' or `faces-xface' and
233 `netpbm' or `libgr-progs', for instance. See also
234 `gnus-article-compface-xbm'.
236 This function is for Emacs 21+. See `gnus-xmas-article-display-xface'
238 ;; It might be worth converting uncompface's output in Lisp.
240 (when (if (fboundp 'display-graphic-p)
242 (unless gnus-article-xface-ring-internal ; Only load ring when needed.
243 (setq gnus-article-xface-ring-internal
244 (make-ring gnus-article-xface-ring-size)))
246 (let* ((cur (current-buffer))
248 (with-current-buffer buffer
249 (buffer-substring beg end))
250 (buffer-substring beg end)))
251 (image (cdr-safe (assoc data (ring-elements
252 gnus-article-xface-ring-internal))))
253 default-enable-multibyte-characters)
257 (and (eq 0 (apply #'call-process-region (point-min) (point-max)
260 (if gnus-article-compface-xbm
262 (if gnus-article-compface-xbm
264 (goto-char (point-min))
265 (progn (insert "/* Width=48, Height=48 */\n") t)
266 (eq 0 (call-process-region (point-min) (point-max)
269 ;; Miles Bader says that faces don't look right as
271 (if (eq 'dark (cdr-safe (assq 'background-mode
272 (frame-parameters))))
273 (setq image (create-image (buffer-string)
274 (if gnus-article-compface-xbm
280 :background "white"))
281 (setq image (create-image (buffer-string)
282 (if gnus-article-compface-xbm
287 (ring-insert gnus-article-xface-ring-internal (cons data image)))
289 (goto-char (point-min))
290 (re-search-forward "^From:" nil 'move)
291 (insert-image image))))))
293 (defun-maybe assoc-ignore-case (key alist)
294 "Like `assoc', but assumes KEY is a string and ignores case when comparing."
295 (setq key (downcase key))
297 (while (and alist (not element))
298 (if (equal key (downcase (car (car alist))))
299 (setq element (car alist)))
300 (setq alist (cdr alist)))
304 ;;; Language support staffs.
306 (defvar-maybe current-language-environment "English"
307 "The language environment.")
309 (defvar-maybe language-info-alist nil
310 "Alist of language environment definitions.")
312 (defun-maybe get-language-info (lang-env key)
313 "Return information listed under KEY for language environment LANG-ENV."
314 (if (symbolp lang-env)
315 (setq lang-env (symbol-name lang-env)))
316 (let ((lang-slot (assoc-ignore-case lang-env language-info-alist)))
318 (cdr (assq key (cdr lang-slot))))))
320 (defun-maybe set-language-info (lang-env key info)
321 "Modify part of the definition of language environment LANG-ENV."
322 (if (symbolp lang-env)
323 (setq lang-env (symbol-name lang-env)))
324 (let (lang-slot key-slot)
325 (setq lang-slot (assoc lang-env language-info-alist))
326 (if (null lang-slot) ; If no slot for the language, add it.
327 (setq lang-slot (list lang-env)
328 language-info-alist (cons lang-slot language-info-alist)))
329 (setq key-slot (assq key lang-slot))
330 (if (null key-slot) ; If no slot for the key, add it.
332 (setq key-slot (list key))
333 (setcdr lang-slot (cons key-slot (cdr lang-slot)))))
334 (setcdr key-slot info)))
338 ;;; gnus-ems.el ends here