;;; x-face-mule.el --- X-Face decoder for MULE ;; Copyright (C) 1997, 1998, 1999, 2000, 2001 ;; Free Software Foundation, Inc. ;; Honorary-Author: Hiroshi Ueno ;; MORIOKA Tomohiko ;; Author: KORIYAMA Naohiro ;; Katsumi Yamaoka ;; OKUNISHI -GTO- Fujikazu ;; KOSEKI Yoshinori ;; Yuuichi Teranishi ;; Maintainer: Katsumi Yamaoka ;; Created: 1997/10/24 ;; Revised: 2001/12/19 ;; Keywords: X-Face, bitmap, Emacs, MULE, BBDB ;; This file is part of BITMAP-MULE. ;; This program 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 2, or (at your option) ;; any later version. ;; This program 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; see the file COPYING. If not, write to the ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330, ;; Boston, MA 02111-1307, USA. ;;; Commentary: ;; - How to use ;; 1. Use of this module and the BITMAP-MULE package is not recommended ;; if you're using Emacs 21 and your Emacs can display images inline. ;; For example, you can use the following file to show X-Face images: ;; ;; ftp://ftp.jpl.org/pub/elisp/x-face-e21.el.gz ;; 2. Build and install `uncompface' program which is available from: ;; ;; ftp://ftp.win.ne.jp/pub/misc/compface-1.4.tar.gz ;; ;; It is one example of many. ;; 3. Setting up ;; ;; [SEMI 1.14 and later] i.e. cmail, gnus, mh-e, VM, etc. ;; You have need to put the following lines in your ~/.emacs file: ;; ;; (if window-system ;; (progn ;; (autoload 'x-face-decode-message-header "x-face-mule") ;; (add-hook 'mime-display-header-hook ;; 'x-face-decode-message-header))) ;; ;; [SEMI 1.13 and earlier, tm-MUA] i.e. cmail, gnus, mh-e, VM, etc. ;; There is nothing to be done. ;; ;; [T-gnus 6.13.3 and later] ;; There is nothing to be done. ;; ;; [Gnus 5.7] ;; Try to put the following line in your ~/.emacs file if X-Face ;; images aren't displayed. ;; ;; (setq x-face-mule-gnus-force-decode-headers t) ;; ;; [Wanderlust] ;; Add the following code in your ~/.wl file: ;; ;; (if window-system ;; (progn ;; (autoload 'x-face-decode-message-header "x-face-mule") ;; (setq wl-highlight-x-face-func 'x-face-decode-message-header))) ;; ;; [before Mew 1.90] ;; Add the following code in your ~/.emacs: ;; ;; (if window-system ;; (progn ;; (autoload 'x-face-decode-message-header "x-face-mule") ;; (add-hook 'mew-summary-display-message-filter-hook ;; 1.90 ;; 'x-face-decode-message-header) ;; (add-hook 'mew-message-hook ;; 1.70 or older ;; 'x-face-decode-message-header))) ;; ;; [Mew 1.92] ;; Add the following code in your ~/.emacs: ;; ;; (if window-system ;; (progn ;; (autoload 'x-face-decode-message-header "x-face-mule") ;; (setq mew-opt-highlight-x-face t) ;; (setq mew-opt-highlight-x-face-function ;; 'x-face-decode-message-header))) ;; ;; [Mew 1.93 or later] ;; Add the following code in your ~/.emacs: ;; ;; (if window-system ;; (progn ;; (autoload 'x-face-decode-message-header "x-face-mule") ;; (setq mew-use-highlight-x-face t) ;; (setq mew-use-highlight-x-face-function ;; 'x-face-decode-message-header))) ;; ;; [BBDB] ;; Add the following code in your ~/.emacs, to collect `X-Face's: ;; ;; (put 'face 'field-separator "\n") ;; (setq bbdb-auto-notes-alist ;; (nconc bbdb-auto-notes-alist ;; (list (list "x-face" ;; (list (concat ;; "[ \t\n]*\\([^ \t\n]*\\)" ;; "\\([ \t\n]+\\([^ \t\n]+\\)\\)?" ;; "\\([ \t\n]+\\([^ \t\n]+\\)\\)?" ;; "\\([ \t\n]+\\([^ \t\n]+\\)\\)?" ;; "\\([ \t\n]+\\([^ \t\n]+\\)\\)?") ;; 'face ;; "\\1\\3\\5\\7\\9"))))) ;; ;; Visit for related ;; informations. ;; 4. Customization ;; ;; * If you don't want to show X-Face at "From:" field, ;; add the following code in your ~/.emacs: ;; ;; (setq x-face-mule-highlight-x-face-position 'x-face) ;; ;; and you can show X-Face at "X-Face:" field. ;; ;; * If you want to show X-Face like XEmacs style, ;; add the following code in ~/.eamcs: ;; ;; (setq x-face-mule-highlight-x-face-style 'xmas) ;; ;; But, this feature won't work well on some MUA... ;; ;; * Other features, please read this file... ;; 5. User commands ;; ;; `x-face-mule-toggle-x-face-position' ;; toggle show position. from->x-face->nil->from... ;; ;; `x-face-mule-toggle-x-face-style' ;; toggle show style. default->xmas->default->... ;; 6. Known bugs & todo ;; ;; * same x-face saved in cache file. ;; 7. Thanks to the following people have contributed many patches ;; and suggestions: ;; ;; OKUNISHI Fujikazu ;; Yuuichi Teranishi ;; TSUMURA Tomoaki ;; Shiono@FSC ;; Kazuhiro Ohta ;; Tatsuya Ichikawa ;; Shigeyuki FUKUSHIMA ;; Hideyuki SHIRAI ;; Koichiro Ohba ;;; Code: (require 'bitmap) (eval-when-compile (require 'cl)) (require 'poem) (require 'pcustom) (eval-when-compile (when (<= emacs-major-version 19) ;; The following procedure is imported from Emacs 20.2, bytecomp.el. (put 'custom-declare-variable 'byte-hunk-handler 'byte-compile-file-form-custom-declare-variable) (defun byte-compile-file-form-custom-declare-variable (form) (if (memq 'free-vars byte-compile-warnings) (setq byte-compile-bound-variables (cons (nth 1 (nth 1 form)) byte-compile-bound-variables))) form)) ;; Bind functions defined by `defun-maybe'. (put 'defun-maybe 'byte-hunk-handler 'byte-compile-file-form-defun-maybe) (defun byte-compile-file-form-defun-maybe (form) (if (and (not (fboundp (nth 1 form))) (memq 'unresolved byte-compile-warnings)) (setq byte-compile-function-environment (cons (cons (nth 1 form) (cons 'lambda (cdr (cdr form)))) byte-compile-function-environment))) form)) (defun-maybe match-string-no-properties (num &optional string) "Return string of text matched by last search, without text properties. NUM specifies which parenthesized expression in the last regexp. Value is nil if NUMth pair didn't match, or there were less than NUM pairs. Zero means the entire text matched by the whole regexp or whole string. STRING should be given if the last search was by `string-match' on STRING. \[Emacs 20.3 emulating function]" (if (match-beginning num) (if string (let ((result (substring string (match-beginning num) (match-end num)))) (set-text-properties 0 (length result) nil result) result) (buffer-substring-no-properties (match-beginning num) (match-end num))))) ;; Avoid byte compile warnings. (eval-when-compile (defvar gnus-article-buffer) (defvar vm-message-pointer) (defalias-maybe 'cmail-get-page-number-from-summary 'ignore) (defalias-maybe 'cmail-show-contents 'ignore) (defalias-maybe 'emh-show 'ignore) (defalias-maybe 'mew-summary-display 'ignore) (defalias-maybe 'mh-show-msg 'ignore) (defalias-maybe 'mime-entity-header-buffer 'ignore) (defalias-maybe 'tm-mh-e/show 'ignore)) (defgroup x-face-mule nil "Show X-Face inline for Emacs/Mule." :group 'bitmap-mule :group 'news :group 'mail) (defcustom uncompface-program "uncompface" "Program for decoding X-Face string to UNIX icon or possibly XBM data." :group 'x-face-mule :type 'string) (defcustom uncompface-program-can-generate-xbm t "Non-nil declares \"uncompface\" can generate XBM format directly with the option \"-X\"." :group 'x-face-mule :type 'boolean) (defcustom x-face-mule-highlight-x-face-position 'from "This variable says where X-Face is shown. `from' at From: field, `x-face' at X-Face: field, `off' don't show." :group 'x-face-mule :type '(radio (const :format "%v " from) (const :format "%v " x-face) (const off))) (defcustom x-face-mule-highlight-x-face-style 'default "Variable used for setting style for showing X-Face. When `default' is set, show like this: From:xxxxxx xxxxxx xxxxxx Foo Bar When `xmas' is set, show like this: xxxxxx xxxxxx From:xxxxxx Foo Bar " :set (function (lambda (symbol value) (set-default symbol value) (if (eq (symbol-value symbol) 'xmas) (set-default 'x-face-mule-highlight-x-face-position 'from)))) :group 'x-face-mule :type '(radio (const :format "%v " default) (const xmas))) (defcustom x-face-mule-highlight-x-face-position-alist nil "If non-nil, x-face-position is changed by major-mode. A format is like this: \(setq x-face-mule-highlight-x-face-position-alist '((gnus-article-mode . from) (gnus-original-article-mode . off)))" :group 'x-face-mule :type '(repeat (cons :format "%v" (symbol :tag "MAJOR-MODE") (radio :format "POSITION: %v " (const :format "%v " from) (const :format "%v " x-face) (const off))))) (defcustom x-face-mule-highlight-x-face-style-alist nil "If non-nil, x-face-style is changed by major-mode. A format is like this: \(setq x-face-mule-highlight-x-face-style-alist '((mew-message-mode . xmas) (gnus-article-mode . default)))" :group 'x-face-mule :type '(repeat (cons :format "%v" (symbol :tag "MAJOR-MODE") (radio :format "STYLE: %v" (const :format "%v " default) (const xmas))))) (defcustom x-face-mule-highlight-x-face-refresh-method-alist (list (cons 'cmail-summary-mode (function (lambda () (cmail-show-contents (cmail-get-page-number-from-summary))))) (cons 'wl-summary-mode (function wl-summary-redisplay)) (cons 'mew-summary-mode (function (lambda () (condition-case nil ;; 1.94b19 or earlier. (mew-summary-display) (wrong-number-of-arguments ;; 1.94b20 or later. (mew-summary-display t)))))) (cons 'mew-virtual-mode (function (lambda () (condition-case nil ;; 1.94b19 or earlier. (mew-summary-display) (wrong-number-of-arguments ;; 1.94b20 or later. (mew-summary-display t)))))) (cons 'mh-folder-mode (function (lambda () (cond ((fboundp (quote emh-show)) (emh-show)) ((fboundp (quote tm-mh-e/show)) (tm-mh-e/show)) (t (mh-show-msg nil))))))) "Alist for refreshing message-buffer. Format: \ '((MAJOR-MODE . (FUNCTION ARGS)))" :group 'x-face-mule :type '(repeat (cons :format "%v" (symbol :tag "MAJOR-MODE") (function :tag "FUNCTION")))) (defcustom x-face-mule-x-face-decode-message-header-hook nil "A hook called before decoding X-Face field." :group 'x-face-mule :type 'hook) (defcustom x-face-mule-delete-x-face-field 'always "A Variable says When delete X-Face field. (for NOT Gnus or Mew...) `always' delete always, `color' delete when color X-Face comes, `mono' delete when monochrome X-Face comes, `never' never delete." :group 'x-face-mule :type '(radio (const :format "%v " always) (const :format "%v " color) (const :format "%v " mono) (const never))) (defcustom x-face-mule-use-cache-file t "If non-nil, use X-Face cache file." :group 'x-face-mule :type 'boolean) (defcustom x-face-mule-cache-file "~/.x-face-cache" "Filename of saving X-Face bitmap-char cache." :group 'x-face-mule :type 'file) (defcustom x-face-mule-force-save-cache-file nil "If non-nil, save X-Face cache file without asking." :group 'x-face-mule :type 'boolean) (defcustom x-face-mule-preserve-original-from-field t "If non-nil, preserve original From: field." :group 'x-face-mule :type 'boolean) (defcustom x-face-mule-hidden-properties '(invisible t) "Property list to use for hiding text. Is is not recommended that the value includes `intangible' property because some MUAs, e.g. VM, might get hung up with it." :group 'x-face-mule :type 'sexp) (defcustom x-face-mule-highlight-x-face-face 'x-face-mule-highlight-x-face-face "Face used for highlighting X-Face in the article buffer." :group 'x-face-mule :type 'face) (defface x-face-mule-highlight-x-face-face '((((class color) (background dark)) (:foreground "Black" :background "White")) (((class color) (background light)) (:foreground "Black")) (t ())) "X-Face face." :group 'x-face-mule) ;;; Internal variables. ;; (defvar x-face-mule-cache-file-loaded-p nil) (defvar x-face-mule-cache-modified-p nil) (defvar x-face-mule-original-from-field nil) (defvar x-face-mule-original-x-face-fields nil) (defvar x-face-mule-x-face-to-rectangle-cache nil) (defconst x-face-mule-image-file-coding-system (if (> emacs-major-version 19) 'iso-2022-7bit-unix '*junet*unix)) (defconst x-face-mule-most-positive-fixnum (eval '(lsh -1 -1))) ;;; Macros. ;; (defmacro x-face-mule-overlay-put (overlay &rest plist) "Add PLIST, the property `priority' with the value `x-face-mule-most-positive-fixnum' and the property `evaporate' with the value t to OVERLAY." (if (symbolp overlay) (cond ((= 2 (length plist)) (` (progn (apply (function overlay-put) (, overlay) (list (,@ plist))) (overlay-put (, overlay) 'priority x-face-mule-most-positive-fixnum) (overlay-put (, overlay) 'evaporate t)))) ((and (> (length plist) 2) (zerop (% (length plist) 2))) (` (let ((plist (list (,@ plist)))) (while plist (overlay-put (, overlay) (car plist) (car (cdr plist))) (setq plist (cdr (cdr plist)))) (overlay-put (, overlay) 'priority x-face-mule-most-positive-fixnum) (overlay-put (, overlay) 'evaporate t))))) (cond ((= 2 (length plist)) (` (let ((overlay (, overlay))) (apply (function overlay-put) overlay (list (,@ plist))) (overlay-put overlay 'priority x-face-mule-most-positive-fixnum) (overlay-put overlay 'evaporate t)))) ((and (> (length plist) 2) (zerop (% (length plist) 2))) (` (let ((overlay (, overlay)) (plist (list (,@ plist)))) (while plist (overlay-put overlay (car plist) (car (cdr plist))) (setq plist (cdr (cdr plist)))) (overlay-put overlay 'priority x-face-mule-most-positive-fixnum) (overlay-put overlay 'evaporate t))))))) ;;; Internal functions for cache. ;; (defun x-face-mule-load-cache-file () "Load X-Face cache file. Return an alist if succeeded." (setq x-face-mule-x-face-to-rectangle-cache nil) (when (and x-face-mule-use-cache-file (file-exists-p x-face-mule-cache-file)) (with-temp-buffer (set-buffer-multibyte t) (insert-file-contents-as-coding-system x-face-mule-image-file-coding-system x-face-mule-cache-file) (when (string-match "^21\\.\\(?:0\\.\\|1\\.\\(?:[[:digit:]]+$\\|\ \\(?:[[:digit:]]\\|[0-4][[:digit:]]\\|50\\)\\.[[:digit:]]+$\\)\\)" emacs-version) ;; Just one more of the mysteries of this world that we can get ;; the correct data by re-reading a file in the same buffer ;; even if Emacs 21.1 has a bug in coding.c. (erase-buffer) (insert-file-contents-as-coding-system x-face-mule-image-file-coding-system x-face-mule-cache-file)) (when (re-search-forward "^[\t ]*(" nil t) (let ((start (point)) end x-face ;; Mule 2 would fail to search images when `mc-flag' is ;; non-nil. mc-flag) (while (and (setq end (condition-case nil (forward-list 1) (error nil))) (progn (goto-char start) (search-forward "(" end t)) (setq x-face (condition-case nil (read (current-buffer)) (error nil)))) (set-text-properties 0 (length x-face) nil x-face) (setq x-face (list x-face)) (while (re-search-forward "\"\\([^\"]+\\)\"" end t) (push (match-string 1) x-face)) (when (= 4 (length x-face)) (push (nreverse x-face) x-face-mule-x-face-to-rectangle-cache)) (goto-char (setq start end)))) (setq x-face-mule-x-face-to-rectangle-cache (nreverse x-face-mule-x-face-to-rectangle-cache)))))) (defun x-face-mule-save-cache-file () "Save X-Face cache file." (if (and x-face-mule-cache-modified-p x-face-mule-x-face-to-rectangle-cache x-face-mule-use-cache-file x-face-mule-cache-file (or x-face-mule-force-save-cache-file (y-or-n-p "Save X-Face cache now? "))) (let ((name (file-name-nondirectory x-face-mule-cache-file)) (cache x-face-mule-x-face-to-rectangle-cache) eol data) (with-temp-buffer (insert ";; This file is generated automatically by X-Face-Mule.") (setq eol (current-column)) (goto-char (point-min)) (insert ";; " name) (insert-char ?\ (max 1 (- eol 29 (current-column)))) (insert "-*- coding: iso-2022-7bit -*-\n") (goto-char (point-max)) (insert "\n\n(\n ") (while (setq data (pop cache)) (insert "(" (prin1-to-string (car data)) "\n") (while (setq data (cdr data)) (insert " \"" (car data) "\"" (if (> (length data) 1) "\n" ")\n ")))) (insert ")\n\n;; " name " ends here\n") (write-region-as-coding-system x-face-mule-image-file-coding-system (point-min) (point-max) x-face-mule-cache-file) (setq x-face-mule-cache-modified-p nil))) (message ""))) (defun x-face-mule-view-cache () "View cached X-Face images." (interactive) (if (or x-face-mule-x-face-to-rectangle-cache (x-face-mule-load-cache-file)) (let ((cache x-face-mule-x-face-to-rectangle-cache) (buffer (get-buffer-create "*cached X-Face images*")) width xmax yindex xindex) (switch-to-buffer buffer) (erase-buffer) (set-buffer-multibyte t) (delete-other-windows) (setq width (/ (window-width) 8)) (while cache (setq xmax (min width (length cache)) yindex 1) (while (< yindex 4) (setq xindex 0) (while (< xindex xmax) (put-text-property (point) (progn (insert (nth yindex (nth xindex cache))) (point)) 'face 'x-face-mule-highlight-x-face-face) (insert " ") (setq xindex (1+ xindex))) (backward-delete-char 2) (insert "\n") (setq yindex (1+ yindex))) (setq cache (nthcdr xmax cache)) (when cache (insert "\n"))) (set-buffer-modified-p nil)) (message "No images in the cache"))) ;; hooks for saving cache (for Mew, cmail, mh-e, VM, Wanderlust). (when (and window-system (>= emacs-major-version 19)) (add-hook 'cmail-quit-hook 'x-face-mule-save-cache-file) (add-hook 'mew-quit-hook 'x-face-mule-save-cache-file) (add-hook 'mh-quit-hook 'x-face-mule-save-cache-file) (add-hook 'wl-exit-hook 'x-face-mule-save-cache-file) (add-hook 'wl-folder-exit-hook 'x-face-mule-save-cache-file) (add-hook 'kill-emacs-hook 'x-face-mule-save-cache-file)) ;;; Internal functions for decoding and displaying X-Face. ;; (defun x-face-mule-convert-x-face-to-rectangle (string) "Convert x-face string to rectangle using cache." (unless x-face-mule-cache-file-loaded-p (x-face-mule-load-cache-file) (setq x-face-mule-cache-file-loaded-p t)) (let ((data (assoc string x-face-mule-x-face-to-rectangle-cache))) (unless data (setq data (cons string (if uncompface-program-can-generate-xbm ;; Attempt to generate XBM format directly. (condition-case err (x-face-mule-convert-x-face-to-bitmap string) (error ;; Oops. (prog1 (x-face-mule-convert-icon-to-rectangle (x-face-mule-convert-x-face-to-icon string)) (setq uncompface-program-can-generate-xbm nil) (message "%s" (car (cdr err)))))) (x-face-mule-convert-icon-to-rectangle (x-face-mule-convert-x-face-to-icon string))))) (setq x-face-mule-cache-modified-p t) (setq x-face-mule-x-face-to-rectangle-cache (cons data x-face-mule-x-face-to-rectangle-cache))) (cdr data))) (defun x-face-mule-convert-x-face-to-icon (string) "Decode x-face string to UNIX ICON." (with-temp-buffer (insert string) (as-binary-process (call-process-region (point-min) (point-max) uncompface-program t t nil)) (buffer-string))) (defun x-face-mule-convert-x-face-to-bitmap (string) "Decode x-face string to X11 bitmap." (with-temp-buffer (insert string) (call-process-region (point-min) (point-max) uncompface-program t t nil "-X") (goto-char (point-min)) (unless (looking-at "#define") (error "!! \"%s\" can not generate XBM format directly" uncompface-program)) (let* ((lines (bitmap-decode-xbm (bitmap-read-xbm-buffer (current-buffer)))) (len (length lines)) (i 0) rest) (while (> len i) (setq rest (nconc rest (list (bitmap-compose (aref lines i)))) i (1+ i))) rest))) (defun x-face-mule-convert-vector-to-rectangle (vector) "Make x-face rectangle from vector." (let ((i 0) ret) (while (< i 3) (let* ((line "") (k (* i 6)) (k+6 (+ k 6))) (while (< k k+6) (setq line (concat line (bitmap-compose (aref vector k)))) (setq k (1+ k))) (setq ret (nconc ret (list line)))) (setq i (1+ i))) ret)) (defun x-face-mule-convert-icon-to-rectangle (icon) "Decode UNIX ICON to rectangle." (let ((i 0) (cmp (make-vector 18 nil))) (with-temp-buffer (insert icon) (goto-char (point-min)) (search-forward "0x" nil t) (while (< i 48) (let* (temp (k (* (/ i 16) 6)) (k+6 (+ k 6))) (while (< k k+6) (setq temp (buffer-substring (point) (+ (point) 2))) (aset cmp k (concat (aref cmp k) temp)) (setq k (1+ k)) (setq temp (buffer-substring (+ (point) 2) (+ (point) 4))) (aset cmp k (concat (aref cmp k) temp)) (setq k (1+ k)) (search-forward "0x" nil t))) (setq i (1+ i)))) (x-face-mule-convert-vector-to-rectangle cmp))) (defun x-face-mule-x-face-insert-at-point (rectangles last) (let ((beg-point (point)) insertcolumn rectangle) (beginning-of-line) (setq insertcolumn (- beg-point (point))) (while (progn (setq rectangle (pop rectangles)) rectangles) (goto-char (setq beg-point (+ insertcolumn (point)))) (insert rectangle) (put-text-property beg-point (point) 'x-face-mule-bitmap-image t) (forward-line 1)) (if (and last (eq x-face-mule-highlight-x-face-position 'from) (eq x-face-mule-highlight-x-face-style 'xmas)) (goto-char (setq beg-point (+ insertcolumn (point) -5))) (goto-char (setq beg-point (+ insertcolumn (point))))) (insert rectangle) (put-text-property beg-point (point) 'x-face-mule-bitmap-image t))) (defsubst x-face-mule-insert-invisible-from () (insert "From:") (put-text-property (- (point) 5) (point) 'x-face-mule-hidden-from t) (insert " ")) (defun x-face-mule-x-face-allocate-lines (beg end height) "Allocate new lines according to x-face-mule-highlight-x-face-position. returns the begin-point of the x-face rectangle." (let ((n 1) from (mime-entity (get-text-property (point-min) 'mime-view-entity))) (cond ((or (eq x-face-mule-highlight-x-face-position 'x-face) (and (eq x-face-mule-highlight-x-face-position 'from) (progn (goto-char beg) (not (setq from (and (re-search-forward "^From:" end t) (point))))))) (goto-char beg) (insert "X-Face-Img:") (prog1 (point); return the begin-point (insert "\n \n ") (while (< n height) (insert "\n \n \n ") (setq n (1+ n))) (insert "\n") (put-text-property beg (point) 'mime-view-entity mime-entity))) ((and (eq x-face-mule-highlight-x-face-position 'from) (if from (goto-char from) (goto-char beg) (re-search-forward "^From:" end t))) (cond ((eq x-face-mule-highlight-x-face-style 'xmas) (beginning-of-line) (setq beg (point)) (x-face-mule-insert-invisible-from) (prog1 (point); return the begin-point (insert "\n") (x-face-mule-insert-invisible-from) (insert "\n") (while (< n height) (x-face-mule-insert-invisible-from) (insert "\n") (x-face-mule-insert-invisible-from) (insert "\n") (x-face-mule-insert-invisible-from) (insert "\n") (setq n (1+ n))) (put-text-property beg (point) 'mime-view-entity mime-entity))) (t (prog1 (point); return the begin-point (setq beg (- (point) 5)) (insert "\n \n ") (while (< n height) (insert "\n \n \n ") (setq n (1+ n))) (put-text-property beg (point) 'mime-view-entity mime-entity)))))))) (defun x-face-mule-change-highlight-x-face-method-by-alist () "Modify displaying position or style by the following variables: `x-face-mule-highlight-x-face-position-alist' `x-face-mule-highlight-x-face-style-alist'" (when x-face-mule-highlight-x-face-position-alist (let ((position (cdr (assq major-mode x-face-mule-highlight-x-face-position-alist)))) (when (memq position '(from x-face off)) (setq x-face-mule-highlight-x-face-position position)))) (when x-face-mule-highlight-x-face-style-alist (let ((style (cdr (assq major-mode x-face-mule-highlight-x-face-style-alist)))) (when (memq style '(default xmas)) (setq x-face-mule-highlight-x-face-style style))))) (defun x-face-mule-analyze-x-face-type (beg end) "Analyze type of X-Face." (goto-char beg) (if (re-search-forward "^X-Face-Type: \\(RGB; \\)?geometry=\\([0-9]+\\)x\\([0-9]+\\)" end t) (list (if (match-string-no-properties 1) 'color 'mono) (string-to-number (match-string-no-properties 2)) (string-to-number (match-string-no-properties 3))) (let ((i 0) (j 0)) (goto-char beg) (while (re-search-forward "^X-Face: *\\(.*\\(\n[ \t].*\\)*\\)\n" end t) (setq i (1+ i))) (setq j (if (> i 0) 1 0)) (list 'mono i j)))) (defun x-face-mule-save-original-x-face-fields (beg end) "Save original X-Face field." (let ((x-face-type nil) (x-faces nil)) (goto-char beg) (when (re-search-forward "^X-Face-Type: \\(RGB; \\)?geometry=\\([0-9]+\\)x\\([0-9]+\\)" end t) (setq x-face-type (match-string-no-properties 0))) (goto-char beg) (while (re-search-forward "^X-Face: *\\(.*\\(\n[ \t].*\\)*\\)\n" end t) (setq x-faces (nconc x-faces (list (match-string-no-properties 0))))) (make-local-variable 'x-face-mule-original-x-face-fields) (setq x-face-mule-original-x-face-fields (list x-face-type x-faces)))) (defun x-face-mule-save-original-from-field (beg end) "Save original From field." (goto-char beg) (set (make-local-variable 'x-face-mule-original-from-field) (when (re-search-forward "^From: *\\(.*\\(\n[ \t].*\\)*\\)\n" end t) (match-string-no-properties 0)))) (defun x-face-mule-insert-original-from-field () "Insert original From field and make it to be invisible." (when (and x-face-mule-original-from-field x-face-mule-preserve-original-from-field) (goto-char (point-min)) (when (re-search-forward "^From:" nil t) (goto-char (match-beginning 0))) (let ((beg-point (point))) (insert x-face-mule-original-from-field) (put-text-property beg-point ;; Don't include the last newline, if so, ;; gnus will break it after sorting headers. (1- (point)) 'x-face-mule-original-from t)))) (defsubst x-face-mule-x-face-decode-message-header-1 () "Decode and show X-Face. The buffer is expected to be narrowed to just the headers of the article." (run-hooks 'x-face-mule-x-face-decode-message-header-hook) (when (and window-system (memq x-face-mule-highlight-x-face-position '(from x-face))) (x-face-mule-save-original-x-face-fields (point-min) (point-max)) (x-face-mule-save-original-from-field (point-min) (point-max)) (let* ((inhibit-read-only t) faces faces-s (first t) (x-face-type (x-face-mule-analyze-x-face-type (point-min) (point-max))) (col-type (nth 0 x-face-type)) (width (* (nth 1 x-face-type) (if (eq col-type 'color) 3 1))) (height (nth 2 x-face-type)) w) (goto-char (point-min)) (let ((h 0)) (while (<= (setq h (1+ h)) height) (setq faces-s nil) (setq w 0) (while (and (<= (setq w (1+ w)) width) (re-search-forward "^X-Face: *\\(.*\\(\n[ \t].*\\)*\\)\n" nil t)) (setq faces-s (cons (match-string-no-properties 1) faces-s)) (when (or (eq x-face-mule-delete-x-face-field 'always) (eq x-face-mule-delete-x-face-field col-type)) (delete-region (match-beginning 0) (match-end 0)))) (setq faces (nconc faces faces-s)))) (x-face-mule-change-highlight-x-face-method-by-alist) (when faces (while faces (let ((begin-point (if first (progn (setq first nil) (x-face-mule-x-face-allocate-lines (point-min) (point-max) height)) (end-of-line 2) (point)))) (setq w 0) (while (and (<= (setq w (1+ w)) width) faces) (goto-char begin-point) (x-face-mule-x-face-insert-at-point (x-face-mule-convert-x-face-to-rectangle (pop faces)) (< (length faces) width))))) (x-face-mule-insert-original-from-field))))) (defun x-face-mule-highlight-header () "Highlight inline images and hide raw X-Face fields." (let (start (end (point-min))) (while (and (setq start (text-property-any end (point-max) 'x-face-mule-bitmap-image t)) (setq end (text-property-not-all start (point-max) 'x-face-mule-bitmap-image t))) (x-face-mule-overlay-put (make-overlay start end) 'face x-face-mule-highlight-x-face-face)) (setq end (point-min)) (let ((inhibit-read-only t)) (while (and (setq start (text-property-any end (point-max) 'x-face-mule-hidden-from t)) (setq end (text-property-not-all start (point-max) 'x-face-mule-hidden-from t))) (add-text-properties start end x-face-mule-hidden-properties)) (setq end (point-min)) (while (and (setq start (text-property-any end (point-max) 'x-face-mule-original-from t)) (setq end (text-property-not-all start (point-max) 'x-face-mule-original-from t))) (add-text-properties start (1+ end);; Include the last newline. x-face-mule-hidden-properties))))) (defun x-face-decode-message-header (&optional beg end) "Decode and show X-Face." (when window-system (save-restriction (narrow-to-region (goto-char (or beg (point-min))) (or end (if (search-forward "\n\n" nil t) (point) (point-max)))) (x-face-mule-x-face-decode-message-header-1) (unless (or (and (boundp 'gnus-article-buffer) (eq (get-buffer gnus-article-buffer) (current-buffer))) (and (boundp 'vm-message-pointer) (fboundp 'x-face-mule-original-vm-energize-headers-and-xfaces) vm-message-pointer)) (x-face-mule-highlight-header))))) ;; Redefine `x-face-decode-message-header' for MULE 1. (when (<= emacs-major-version 18) (require 'x-face-18)) ;;; Commands. ;; ;;;###autoload (defun x-face-mule-toggle-x-face-position (&optional arg) "Toggle position of showing X-Face." (interactive "P") (if window-system (let ((pos (cond ((eq x-face-mule-highlight-x-face-position 'from) 'x-face) ((eq x-face-mule-highlight-x-face-position 'x-face) 'off) (t 'from))) (table '(("From:" . from) ("X-Face:" . x-face) ("Off" . off))) (method (cdr (assq major-mode x-face-mule-highlight-x-face-refresh-method-alist))) msg) (setq x-face-mule-highlight-x-face-position (if arg (cdr (assoc (completing-read "Show X-Face at> " table nil t (concat (capitalize (symbol-name pos)) (and (memq pos '(from x-face)) ":"))) table)) pos)) (when (interactive-p) (setq msg (cond ((eq x-face-mule-highlight-x-face-position 'from) "Show X-Face at From:") ((eq x-face-mule-highlight-x-face-position 'x-face) "Show X-Face at X-Face-Img:") (t "Don't Show X-Face"))) (message "%s..." msg)) (when method (funcall method)) (when (interactive-p) (message "%s...done" msg))) (when (interactive-p) (message "You're not under window system")))) ;;;###autoload (defun x-face-mule-toggle-x-face-style (&optional arg) "Toggle style of showing X-Face." (interactive "P") (if window-system (let ((style (cond ((eq x-face-mule-highlight-x-face-style 'default) 'xmas) ((eq x-face-mule-highlight-x-face-style 'xmas) 'default))) (table '(("Xmas" . xmas) ("Default" . default))) (method (cdr (assq major-mode x-face-mule-highlight-x-face-refresh-method-alist))) msg) (setq x-face-mule-highlight-x-face-style (if arg (cdr (assoc (completing-read "Show X-Face style> " table nil t (capitalize (symbol-name style))) table)) style)) (when (interactive-p) (setq msg (cond ((eq x-face-mule-highlight-x-face-style 'xmas) "Show X-Face in XEmacs style") ((eq x-face-mule-highlight-x-face-style 'default) "Show X-Face in default style"))) (message "%s..." msg)) (when method (funcall method)) (when (interactive-p) (message "%s...done" msg))) (when (interactive-p) (message "You're not under window system")))) ;;; MUA dependencies. ;; (when (and (>= emacs-major-version 19) window-system) ;; gnus (autoload 'x-face-mule-gnus-article-display-x-face "gnus-bitmap") (unless (assq 'gnus-bitmap-redefine (cdr (assoc "gnus" after-load-alist))) (autoload 'gnus-bitmap-redefine "gnus-bitmap") (eval-after-load "gnus" '(gnus-bitmap-redefine))) ;; VM (autoload 'vm-bitmap-redefine "vm-bitmap") (eval-after-load "vm" '(vm-bitmap-redefine))) ;;; BBDB ;; (defvar x-face-mule-BBDB-display (and (locate-library "bbdb") t) "*If non-nil, display X-Faces in *BBDB* buffer.") (defvar x-face-mule-BBDB-icon (bitmap-compose "\ 0000f87c3e1f0f070205081020400000000002040810a040e0f0f87c3e1f0000") "*Bitmap image used to iconified raw X-Face data.") (defvar x-face-mule-BBDB-verbose t "*Non-nil makes x-face-mule-BBDB- functions talkative.") (eval-and-compile (autoload 'bbdb-current-record "bbdb-com") (autoload 'bbdb-record-getprop "bbdb") (autoload 'bbdb-record-name "bbdb")) ;; Byte-compiler warning. (defvar bbdb-buffer-name) (defun x-face-mule-BBDB-buffer () "Display X-Faces in *BBDB* buffer." (when (and x-face-mule-BBDB-display (get-buffer bbdb-buffer-name)) (save-excursion (set-buffer bbdb-buffer-name) (let ((inhibit-point-motion-hooks t)) (goto-char (point-min)) (while (progn (while (and (not (eobp)) (memq (following-char) '(?\t ?\n ?\ ))) (forward-line 1)) (not (eobp))) (x-face-mule-BBDB-one-record t) (forward-line 1)))))) (defun x-face-mule-BBDB-one-record (&optional beginning-of-record) "Display X-Face in *BBDB* one recode. Optional BEGINNING-OF-RECORD means that the current position is the beginning of a record." (when x-face-mule-BBDB-display (unless beginning-of-record (beginning-of-line) (if (eolp) (progn (forward-line 1) (while (and (not (eobp)) (memq (following-char) '(?\t ?\n ?\ ))) (forward-line 1))) (while (and (not (bobp)) (memq (following-char) '(?\t ?\n ?\ ))) (forward-line -1)))) (let ((record (bbdb-current-record)) (inhibit-read-only t) sfaces home start xface xfaces len (i 0) j pos) (when (and record (setq sfaces (bbdb-record-getprop record 'face))) (when x-face-mule-BBDB-verbose (message "Extracting X-Face(s) for %s..." (bbdb-record-name record))) (forward-line 1) (setq home (point)) (while (and (not (and (looking-at "[\t ]+face:[\t ]+") (setq start (match-end 0)))) (zerop (forward-line 1)))) (while (and (zerop (forward-line 1)) (not (looking-at "[\t ]+[^\t\n ]+:[\t ]\\|[\t ]*$\\|[^\t\n ]")))) (save-restriction (narrow-to-region start (1- (point))) (goto-char (point-min)) (while (re-search-forward "[^\t\n ]+[\t\n ]*" nil t) (replace-match x-face-mule-BBDB-icon))) (setq start 0) (while (string-match "\\([^\t\n\v\f\r ]+\\)[\t\n\v\f\r ]*" sfaces start) (when (setq start (match-end 0) xface (x-face-mule-convert-x-face-to-rectangle (substring sfaces (match-beginning 1) (match-end 1)))) (setq xfaces (nconc xfaces (list xface))))) (setq len (length xfaces)) (goto-char home) (while (> 3 i) (setq j 0) (while (> len j) (insert " ") (setq pos (point)) (insert (format "%s" (nth i (nth j xfaces)))) (x-face-mule-overlay-put (make-overlay pos (point)) 'face x-face-mule-highlight-x-face-face) (setq j (1+ j))) (setq i (1+ i)) (insert "\n")) (forward-line 1) (x-face-mule-overlay-put (make-overlay home (point)) 'intangible t))))) ;;; BBDB Setup. ;; (let (current-load-list) (defadvice bbdb-display-records-1 (around x-face-mule-BBDB-buffer activate) "Advised by X-Face-Mule. Display X-Faces in *BBDB* buffer." (let ((silent (or (not x-face-mule-BBDB-verbose) (and (boundp 'bbdb-gag-messages) (symbol-value 'bbdb-gag-messages)) (and (boundp 'bbdb-silent-running) (symbol-value 'bbdb-silent-running))))) (unless silent (message "Formatting...")) (let ((bbdb-silent-running t) (bbdb-list-hook bbdb-list-hook)) (remove-hook 'bbdb-list-hook 'x-face-mule-BBDB-one-record) ad-do-it) (x-face-mule-BBDB-buffer) (unless silent (message "Formatting...done")))) (defadvice bbdb-elide-all-records-internal (around dont-show-x-faces-if-records-are-made-elided activate) "Advised by X-Face-Mule for BBDB versions prior to 2.33. Don't show X-Faces if records are made elided." (let ((x-face-mule-BBDB-display (cond ((not (ad-get-arg 0)) (and (car (cdr (assq (bbdb-current-record) bbdb-records))) x-face-mule-BBDB-display)) ((eq 0 (ad-get-arg 0)) x-face-mule-BBDB-display) (t nil))) x-face-mule-BBDB-verbose) ad-do-it)) (defadvice bbdb-elide-record-internal (around dont-show-x-faces-if-record-is-made-elided activate) "Advised by X-Face-Mule for BBDB versions prior to 2.33. Don't show X-Face if a record is made elided." (let ((x-face-mule-BBDB-display (cond ((not (ad-get-arg 0)) (and (car (cdr (assq (bbdb-current-record) bbdb-records))) x-face-mule-BBDB-display)) ((eq 0 (ad-get-arg 0)) x-face-mule-BBDB-display) (t nil))) x-face-mule-BBDB-verbose) ad-do-it)) (defadvice bbdb-change-records-state-and-redisplay (around dont-show-x-faces-if-records-are-made-elided activate) "Advised by X-Face-Mule for BBDB versions 2.33 and later. Don't show X-Faces if records are made elided." (let ((x-face-mule-BBDB-display (if (eq 'multi-line (ad-get-arg 0)) x-face-mule-BBDB-display)) x-face-mule-BBDB-verbose) ad-do-it))) (add-hook 'bbdb-list-hook 'x-face-mule-BBDB-one-record) (provide 'x-face-mule) ;; x-face-mule.el ends here