changeset 35160:75a8ca0bdd1d

(ansi-color-process-output): Use markers instead of positions for start and end of region. (ansi-color-apply-on-region): Rewrote code to make it more robust. Previously, occasional mistakes happend when fontifying many chunks of output (eg. ls --color=yes /dev). This happened whenever an overlay was created up to the end of the region, which coincided with the process-mark. New text would then be added within that overlay instead of after it. (ansi-color-make-extent): Overlays are created with the property `modification-hooks' set to '(ansi-color-freeze-overlay). (ansi-color-freeze-overlay): New function. When inserting text at the end of the overlay, the overlay will resize. (ansi-color-process-output): Doc change. (ansi-color-unfontify-region): Doc change. No longer installed automatically in font-lock-unfontify-region-function. (ansi-color-apply): Doc change. (ansi-color-apply-on-region): Use extents or overlays instead of text-properties. (ansi-color-make-extent): New function. (ansi-color-set-extent-face): New function. (ansi-color-process): Removed, Emacs and XEmacs both use ansi-color-process-output, now. (ansi-color-process-output): Doesn't return string anymore. It is installed in comint-output-filter-functions for both Emacs and XEmacs, now. (ansi-color-unfontify-region): Simplified code removing variables pos and start-ansi. (ansi-color-apply): Put text-property ansi-color before putting text-property face because ansi-color-unfontify-region is called immediately after the call to put-text-property. (ansi-color-context-region): Doc change. (ansi-color-filter-region): Simplified code. (ansi-color-apply-on-region): Changed start to start-marker, using a marker explicitly. Put text-property ansi-color before putting text-property face because ansi-color-unfontify-region is called immediately after the call to put-text-property. (ansi-color-faces-vector): Doc change. (ansi-color-for-comint-mode): Changed :type property to choice. (ansi-color-last-context): Removed. (ansi-color-process-output): Don't use ansi-color-last-context, as the main functions will store their context now. (ansi-color-context): Doc change. (ansi-color-filter-apply): Rewrote it based on ansi-color-apply. Uses ansi-color-context such that repeated calls will strip partial escape sequences, too. (ansi-color-apply): Simplified code. Colorize end of string if face is not null. Store context in new (FACE STRING) format, such that repeated calls will strip partial escape sequences, too. Append faces to face property using ansi-color-apply-sequence such that cumulative mode actually works. (ansi-color-context-region): New variable. (ansi-color-filter-region): Rewrote it based on ansi-color-apply-on-region. Uses ansi-color-context-region such that repeated calls will strip partial escape sequences, too. (ansi-color-apply-on-region): Simplified code. Colorize end of region if face is not null. Store context in new (FACE POS) format, such that repeated calls will strip partial escape sequences, too. Append faces to face property using ansi-color-apply-sequence such that cumulative mode actually works. (ansi-color-apply-sequence): New function. (ansi-color-get-face): When the default face is added to the list of faces, all previous settings are discarded and the list of faces is set to '(default). (ansi-color-faces-vector): Use nil for the default face, such that ansi-color-apply and ansi-color-apply-on-region will do the right thing. (ansi-color-apply): Do the right thing, ie. if ansi-color-get-face returns nil, set the list of faces back to nil instead of appending the result of ansi-color-get-face to the front of the list. (ansi-color-for-comint-mode): Doc change. (ansi-color-process): Doc change. (ansi-color-last-context): New buffer-local variable. (ansi-color-process-output): New function. It is automatically added to comint-output-filter-functions if this is XEmacs. (ansi-color-unfontify-region): New optional parameter for XEmacs compatibility. Check wether font-lock-syntactic-keywords is boundp before removing the syntax table text property, as XEmacs doesn't have it. (ansi-color-filter-region): Doc change. (ansi-color-apply-on-region): Doc change. (ansi-color-make-face): New function. Compatibility layer for XEmacs. Return temporary faces instead of cons cells for XEmacs. (ansi-color-make-color-map): Use ansi-color-make-face. (ansi-color-get-face): Avoid face text property '(nil) as results in an errow for XEmacs. (ansi-color-unfontify-region): New function. Uses text-property ansi-color in order to preserve fontification by ansi-color. When the package is loaded, a lambda expression is put onto font-lock-mode-hook. This lambda expression will check font-lock-unfontify-region-function and replace font-lock-default-unfontify-region with ansi-color-unfontify-region. (ansi-color-apply): Add text-property ansi-color in addition to text-property face. (ansi-color-apply-on-region): Add text-property ansi-color in addition to text-property face. (save-buffer-state): Copy of the macro that is also used by lazy-lock and font-lock. (ansi-color-for-comint-mode): New option. (ansi-color-for-comint-mode-on): Set ansi-color-for-comint-mode. (ansi-color-for-comint-mode-off): Ditto. (ansi-color-for-comint-mode-filter): Ditto. (ansi-color-process): New function. Uses ansi-color-for-comint-mode to decide what to do. This function is added to comint-preoutput-filter-functions when the package is loaded. (ansi-color-for-shell-mode-set): Removed. (ansi-color-for-shell-mode): Removed. (ansi-color-for-shell-mode-set): New function with the lambda expression from the ansi-color-for-shell-mode :set property. Additionally, modify shell-mode-hook to enable or disable font-lock-mode for future shell buffers. (ansi-color-for-shell-mode): The :set property calls ansi-color-for-shell-mode-set instead of a lambda expression. (ansi-color-for-shell-mode): Doc change. (ansi-color-context): New variable. (ansi-color-apply): Save context between calls.
author Gerd Moellmann <gerd@gnu.org>
date Tue, 09 Jan 2001 11:38:28 +0000
parents 9276c6d67ee4
children 0495de5487ae
files lisp/ansi-color.el
diffstat 1 files changed, 414 insertions(+), 180 deletions(-) [+]
line wrap: on
line diff
--- a/lisp/ansi-color.el	Tue Jan 09 11:38:12 2001 +0000
+++ b/lisp/ansi-color.el	Tue Jan 09 11:38:28 2001 +0000
@@ -1,10 +1,10 @@
-;;; ansi-color.el --- translate ANSI into text-properties
+;;; ansi-color.el --- translate ANSI escape sequences into faces
 
-;; Copyright (C) 1999, 2000  Free Software Foundation, Inc.
+;; Copyright (C) 1999, 2000, 2001  Free Software Foundation, Inc.
 
 ;; Author: Alex Schroeder <alex@gnu.org>
 ;; Maintainer: Alex Schroeder <alex@gnu.org>
-;; Version: 2.4.0
+;; Version: 3.4.0
 ;; Keywords: comm processes
 
 ;; This file is part of GNU Emacs.
@@ -26,15 +26,28 @@
 
 ;;; Commentary:
 
-;; This file provides a function that takes a string containing Select
-;; Graphic Rendition (SGR) control sequences (formerly known as ANSI
-;; escape sequences) and tries to replace these with text-properties.
+;; This file provides a function that takes a string or a region
+;; containing Select Graphic Rendition (SGR) control sequences (formerly
+;; known as ANSI escape sequences) and tries to translate these into
+;; faces.
+;;
+;; This allows you to run ls --color=yes in shell-mode.  In order to
+;; test this, proceed as follows:
 ;;
-;; This allows you to run ls --color=yes in shell-mode: If
-;; `ansi-color-for-shell-mode' is non-nil, the SGR control sequences are
-;; translated into text-properties, colorizing the ls output.  If
-;; `ansi-color-for-shell-mode' is nil, the SGR control sequences are
-;; stripped, making the ls output legible.
+;; 1. start a shell: M-x shell
+;; 2. load this file: M-x load-library RET ansi-color RET
+;; 3. activate ansi-color: M-x ansi-color-for-comint-mode-on
+;; 4. test ls --color=yes in the *shell* buffer
+;;
+;; Note that starting your shell from within Emacs might set the TERM
+;; environment variable.  The new setting might disable the output of
+;; SGR control sequences.  Using ls --color=yes forces ls to produce
+;; these.
+;;
+;; If you decide you like this, add the following to your .emacs file:
+;;
+;; (autoload 'ansi-color-for-comint-mode-on "ansi-color" nil t)
+;; (add-hook 'shell-mode-hook 'ansi-color-for-comint-mode-on)
 ;;
 ;; SGR control sequences are defined in section 3.8.117 of the ECMA-48
 ;; standard (identical to ISO/IEC 6429), which is freely available as a
@@ -58,10 +71,6 @@
 ;; `ansi-color-filter-region' to filter SGR control sequences from a
 ;; region.
 
-;; Instead of defining lots of new faces, this package uses
-;; text-properties as described in the elisp manual
-;; *Note (elisp)Special Properties::.
-
 ;;; Thanks
 
 ;; Georges Brun-Cottan <gbruncot@emc.com> for improving ansi-color.el
@@ -69,6 +78,9 @@
 ;; of output and the filter functions.
 ;;
 ;; Markus Kuhn <Markus.Kuhn@cl.cam.ac.uk> for pointing me to ECMA-48.
+;;
+;; Stefan Monnier <foo@acm.com> explaing obscure font-lock stuff and
+;; code suggestions.
 
 
 
@@ -77,7 +89,7 @@
 ;; Customization
 
 (defgroup ansi-colors nil
-  "Translating SGR control sequences to text-properties.
+  "Translating SGR control sequences to faces.
 This translation effectively colorizes strings and regions based upon
 SGR control sequences embedded in the text.  SGR (Select Graphic
 Rendition) control sequences are defined in section 3.8.117 of the
@@ -102,6 +114,9 @@
   6        rapidly blinking   bold-italic
   7        negative image     modeline
 
+Note that the symbol `default' is special: It will not be combined
+with the current face.
+
 This vector is used by `ansi-color-make-color-map' to create a color
 map.  This color map is stored in the variable `ansi-color-map'."
   :type '(vector face face face face face face face face)
@@ -132,54 +147,6 @@
   :initialize 'custom-initialize-default
   :group 'ansi-colors)
 
-(defcustom ansi-color-for-shell-mode nil
-  "Determine wether font-lock or ansi-color get to fontify shell buffers.
-
-If non-nil and `global-font-lock-mode' is non-nil, ansi-color will be
-used.  This adds `ansi-color-apply' to
-`comint-preoutput-filter-functions' and removes
-`ansi-color-filter-apply' for all shell-mode buffers.
-
-If non-nil and global-font-lock-mode is nil, both `ansi-color-apply' and
-`ansi-color-filter-apply' will be removed from
-`comint-preoutput-filter-functions' for all shell-mode buffers.
-
-If nil, font-lock will be used (if it is enabled).  This adds
-`ansi-color-filter-apply' to `comint-preoutput-filter-functions' and
-removes `ansi-color-apply' for all shell-mode buffers."
-  :version "20.8"
-  :type 'boolean
-  :set (function (lambda (symbol value)
-		   (set-default symbol value)
-		   (save-excursion
-		     (let ((buffers (buffer-list))
-			   buffer)
-		       (while buffers
-			 (setq buffer (car buffers)
-			       buffers (cdr buffers))
-			 (set-buffer buffer)
-			 (when (eq major-mode 'shell-mode)
-			   (if value
-			       (if global-font-lock-mode
-				   (progn
-				     (font-lock-mode 0)
-				     (remove-hook 'comint-preoutput-filter-functions 
-						  'ansi-color-filter-apply)
-				     (add-hook 'comint-preoutput-filter-functions 
-					       'ansi-color-apply))
-				 (remove-hook 'comint-preoutput-filter-functions 
-					      'ansi-color-filter-apply)
-				 (remove-hook 'comint-preoutput-filter-functions 
-					      'ansi-color-apply))
-			     (if global-font-lock-mode
-				 (font-lock-mode 1))
-			     (remove-hook 'comint-preoutput-filter-functions 
-					  'ansi-color-apply)
-			     (add-hook 'comint-preoutput-filter-functions 
-				       'ansi-color-filter-apply))))))))
-  :initialize 'custom-initialize-reset
-  :group 'ansi-colors)
-
 (defconst ansi-color-regexp "\033\\[\\([0-9;]*\\)m"
   "Regexp that matches SGR control sequences.")
 
@@ -187,142 +154,404 @@
   "Regexp that matches SGR control sequence parameters.")
 
 
-;; Main functions
+;; Convenience functions for comint modes (eg. shell-mode)
 
 
-(defun ansi-color-filter-apply (s)
-  "Filter out all SGR control sequences from S.
+(defcustom ansi-color-for-comint-mode nil
+  "Determines what to do with comint output.
+If nil, do nothing.
+If the symbol `filter', then filter all SGR control sequences.
+If anything else (such as t), then translate SGR control sequences
+into text-properties.
+
+In order for this to have any effect, `ansi-color-process-output' must
+be in `comint-output-filter-functions'.
+
+This can be used to enable colorized ls --color=yes output
+in shell buffers.  You set this variable by calling one of:
+\\[ansi-color-for-comint-mode-on]
+\\[ansi-color-for-comint-mode-off]
+\\[ansi-color-for-comint-mode-filter]"
+  :version "20.8"
+  :type '(choice (const :tag "Do nothing" nil)
+		 (const :tag "Filter" filter)
+		 (const :tag "Translate" t))
+  :group 'ansi-colors)
+
+(defun ansi-color-for-comint-mode-on ()
+  "Set `ansi-color-for-comint-mode' to t."
+  (interactive)
+  (setq ansi-color-for-comint-mode t))
+
+(defun ansi-color-for-comint-mode-off ()
+  "Set `ansi-color-for-comint-mode' to nil."
+  (interactive)
+  (setq ansi-color-for-comint-mode nil))
 
-This function can be added to `comint-preoutput-filter-functions'."
-  (while (string-match ansi-color-regexp s)
-    (setq s (replace-match "" t t s)))
-  s)
+(defun ansi-color-for-comint-mode-filter ()
+  "Set `ansi-color-for-comint-mode' to symbol `filter'."
+  (interactive)
+  (setq ansi-color-for-comint-mode 'filter))
+
+(defun ansi-color-process-output (string)
+  "Maybe translate SGR control sequences of comint output into text-properties.
+
+Depending on variable `ansi-color-for-comint-mode' the comint output is
+either not processed, SGR control sequences are filtered using
+`ansi-color-filter-region', or SGR control sequences are translated into
+text-properties using `ansi-color-apply-on-region'.
+
+The comint output is assumed to lie between the marker
+`comint-last-output-start' and the process-mark.
+
+This is a good function to put in `comint-output-filter-functions'."
+  (let ((start-marker (or comint-last-output-start
+			  (point-min-marker)))
+	(end-marker (process-mark (get-buffer-process (current-buffer)))))
+    (cond ((eq ansi-color-for-comint-mode nil))
+	  ((eq ansi-color-for-comint-mode 'filter)
+	   (ansi-color-filter-region start-marker end-marker))
+	  (t
+	   (ansi-color-apply-on-region start-marker end-marker)))))
+
+(add-hook 'comint-output-filter-functions
+	  'ansi-color-process-output)
+
+
+;; Alternative font-lock-unfontify-region-function
 
 
-(defun ansi-color-filter-region (begin end)
-  "Filter out all SGR control sequences from region START END.
-
-Returns the first point it is safe to start with.  Used to speedup
-further processing.
+(eval-when-compile
+  ;; We use this to preserve or protect things when modifying text
+  ;; properties.  Stolen from lazy-lock and font-lock.  Ugly!!!
+  ;; Probably most of this is not needed?
+  (defmacro save-buffer-state (varlist &rest body)
+    "Bind variables according to VARLIST and eval BODY restoring buffer state."
+    (` (let* ((,@ (append varlist
+		   '((modified (buffer-modified-p)) (buffer-undo-list t)
+		     (inhibit-read-only t) (inhibit-point-motion-hooks t)
+		     before-change-functions after-change-functions
+		     deactivate-mark buffer-file-name buffer-file-truename))))
+	 (,@ body)
+	 (when (and (not modified) (buffer-modified-p))
+	   (set-buffer-modified-p nil)))))
+  (put 'save-buffer-state 'lisp-indent-function 1))
 
-Design to cope with arbitrary chunk of output such as the ones get by
-comint-output-filter-functions, e.g.:
+(defun ansi-color-unfontify-region (beg end &rest xemacs-stuff)
+  "Replacement function for `font-lock-default-unfontify-region'.  
+When font-lock is active in a buffer, you cannot simply add face
+text-properties to the buffer.  Font-lock will remove the face
+text-property using `font-lock-unfontify-region-function'.  If you want
+to insert the strings returned by `ansi-color-apply' into such buffers,
+you must set `font-lock-unfontify-region-function' to
+`ansi-color-unfontify-region'.  This function will not remove all face
+text-properties unconditionally.  It will keep the face text-properties
+if the property `ansi-color' is set.
 
-\(defvar last-context nil)
-\(make-variable-buffer-local 'last-context)
+The region from BEG to END is unfontified.  XEMACS-STUFF is ignored.
+
+A possible way to install this would be:
 
-\(defun filter-out-color-in-buffer (s)
-  \(setq last-context
-        \(ansi-color-filter-region
-         \(if last-context
-             last-context
-           \(if (marker-position comint-last-output-start)
-               \(marker-position comint-last-output-start)
-             1))
-         \(marker-position (process-mark (get-buffer-process (current-buffer)))) ))
-  s)
+\(add-hook 'font-lock-mode-hook
+	  \(function (lambda ()
+		      \(setq font-lock-unfontify-region-function
+			    'ansi-color-unfontify-region))))"
+  ;; save-buffer-state is a macro in font-lock.el!
+  (save-buffer-state nil
+    (when (boundp 'font-lock-syntactic-keywords)
+      (remove-text-properties beg end '(syntax-table nil)))
+    ;; instead of just using (remove-text-properties beg end '(face
+    ;; nil)), we find regions with a non-nil face test-property, skip
+    ;; positions with the ansi-color property set, and remove the
+    ;; remaining face test-properties.
+    (while (setq beg (text-property-not-all beg end 'face nil))
+      (setq beg (or (text-property-not-all beg end 'ansi-color t) end))
+      (when (get-text-property beg 'face)
+	(let ((end-face (or (text-property-any beg end 'face nil)
+			    end)))
+	  (remove-text-properties beg end-face '(face nil))
+	  (setq beg end-face))))))
+
+;; Working with strings
+
+(defvar ansi-color-context nil
+  "Context saved between two calls to `ansi-color-apply'.
+This is a list of the form (FACES FRAGMENT) or nil.  FACES is a list of
+faces the last call to `ansi-color-apply' ended with, and FRAGMENT is a
+string starting with an escape sequence, possibly the start of a new
+escape sequence.")
+(make-variable-buffer-local 'ansi-color-context)
 
-\(add-hook 'comint-output-filter-functions 'filter-out-color-in-buffer)
-"
-  (let ((endm (copy-marker end)))
-    (save-excursion
-      (goto-char begin)
-      (while (re-search-forward ansi-color-regexp endm t)
-        (replace-match ""))
-      (if (re-search-forward "\033" endm t)
-          (match-beginning 0)
-        (marker-position endm)))))
+(defun ansi-color-filter-apply (string)
+  "Filter out all SGR control sequences from STRING.
+
+Every call to this function will set and use the buffer-local variable
+`ansi-color-context' to save partial escape sequences.  This information
+will be used for the next call to `ansi-color-apply'.  Set
+`ansi-color-context' to nil if you don't want this.
 
+This function can be added to `comint-preoutput-filter-functions'."
+  (let ((start 0) end result)
+    ;; if context was saved and is a string, prepend it
+    (if (cadr ansi-color-context)
+        (setq string (concat (cadr ansi-color-context) string)
+              ansi-color-context nil))
+    ;; find the next escape sequence
+    (while (setq end (string-match ansi-color-regexp string start))
+      (setq result (concat result (substring string start end))
+	    start (match-end 0)))
+    ;; save context, add the remainder of the string to the result
+    (let (fragment)
+      (if (string-match "\033" string start)
+	  (let ((pos (match-beginning 0)))
+	    (setq fragment (substring string pos)
+		  result (concat result (substring string start pos))))
+	(setq result (concat result (substring string start))))
+      (if fragment
+	  (setq ansi-color-context (list nil fragment))
+	(setq ansi-color-context nil)))
+    result))
 
 (defun ansi-color-apply (string)
   "Translates SGR control sequences into text-properties.
 
 Applies SGR control sequences setting foreground and background colors
-to STRING and returns the result.  The colors used are given in
-`ansi-color-faces-vector' and `ansi-color-names-vector'.
+to STRING using text-properties and returns the result.  The colors used
+are given in `ansi-color-faces-vector' and `ansi-color-names-vector'.
+See function `ansi-color-apply-sequence' for details.
+
+Every call to this function will set and use the buffer-local variable
+`ansi-color-context' to save partial escape sequences and current face.
+This information will be used for the next call to `ansi-color-apply'.
+Set `ansi-color-context' to nil if you don't want this.
 
-This function can be added to `comint-preoutput-filter-functions'."
-  (let (face (start 0) end escape-sequence null-sequence result)
+This function can be added to `comint-preoutput-filter-functions'.
+
+You cannot insert the strings returned into buffers using font-lock.
+See `ansi-color-unfontify-region' for a way around this."
+  (let ((face (car ansi-color-context))
+	(start 0) end escape-sequence result)
+    ;; if context was saved and is a string, prepend it
+    (if (cadr ansi-color-context)
+        (setq string (concat (cadr ansi-color-context) string)
+              ansi-color-context nil))
     ;; find the next escape sequence
     (while (setq end (string-match ansi-color-regexp string start))
       ;; store escape sequence
-      (setq escape-sequence (match-string 1 string)
-	    null-sequence (string-equal escape-sequence ""))
+      (setq escape-sequence (match-string 1 string))
       ;; colorize the old block from start to end using old face
-      (if face
-	  (put-text-property start end 'face face string))
+      (when face
+	(put-text-property start end 'ansi-color t string)
+	(put-text-property start end 'face face string))
       (setq result (concat result (substring string start end))
 	    start (match-end 0))
-      ;; create new face by applying all the parameters in the escape sequence
-      (if null-sequence
-	  (setq face nil)
-	(setq face (ansi-color-get-face escape-sequence))))
-    (concat result (substring string start))))
+      ;; create new face by applying all the parameters in the escape
+      ;; sequence
+      (setq face (ansi-color-apply-sequence escape-sequence face)))
+    ;; if the rest of the string should have a face, put it there
+    (when face
+      (put-text-property start (length string) 'ansi-color t string)
+      (put-text-property start (length string) 'face face string))
+    ;; save context, add the remainder of the string to the result
+    (let (fragment)
+      (if (string-match "\033" string start)
+	  (let ((pos (match-beginning 0)))
+	    (setq fragment (substring string pos)
+		  result (concat result (substring string start pos))))
+	(setq result (concat result (substring string start))))
+      (if (or face fragment)
+	  (setq ansi-color-context (list face fragment))
+	(setq ansi-color-context nil)))
+    result))
+
+;; Working with regions
 
+(defvar ansi-color-context-region nil
+  "Context saved between two calls to `ansi-color-apply-on-region'.
+This is a list of the form (FACES MARKER) or nil.  FACES is a list of
+faces the last call to `ansi-color-apply-on-region' ended with, and
+MARKER is a buffer position within an escape sequence or the last
+position processed.")
+(make-variable-buffer-local 'ansi-color-context-region)
 
-(defun ansi-color-apply-on-region (begin end &optional context)
-  "Translates SGR control sequences into text-properties.
+(defun ansi-color-filter-region (begin end)
+  "Filter out all SGR control sequences from region BEGIN to END.
+
+Every call to this function will set and use the buffer-local variable
+`ansi-color-context-region' to save position.  This information will be
+used for the next call to `ansi-color-apply-on-region'.  Specifically,
+it will override BEGIN, the start of the region.  Set
+`ansi-color-context-region' to nil if you don't want this."
+  (let ((end-marker (copy-marker end))
+	(start (or (cadr ansi-color-context-region) begin)))
+    (save-excursion
+      (goto-char start)
+      ;; find the next escape sequence
+      (while (re-search-forward ansi-color-regexp end-marker t)
+	;; delete the escape sequence
+        (replace-match ""))
+    ;; save context, add the remainder of the string to the result
+    (if (re-search-forward "\033" end-marker t)
+	(setq ansi-color-context-region (list nil (match-beginning 0)))
+      (setq ansi-color-context-region nil)))))
+
+(defun ansi-color-apply-on-region (begin end)
+  "Translates SGR control sequences into overlays or extents.
 
 Applies SGR control sequences setting foreground and background colors
-to text in region. The colors used are given in
-`ansi-color-faces-vector' and `ansi-color-names-vector'.
-Returns a context than can be used to speedup further processing.
-Context is a (begin (start . face)) list.
-
-Design to cope with arbitrary chunk of output such as the ones get by
-comint-output-filter-functions, e.g.:
-
-\(defvar last-context nil)
-\(make-variable-buffer-local 'last-context)
+to text in region between BEGIN and END using extents or overlays.
+Emacs will use overlays, XEmacs will use extents.  The colors used are
+given in `ansi-color-faces-vector' and `ansi-color-names-vector'.  See
+function `ansi-color-apply-sequence' for details.
 
-\(defun ansi-output-filter (s)
-  \(setq last-context
-        \(ansi-color-apply-on-region
-         \(if last-context
-             \(car last-context)
-           \(if (marker-position comint-last-output-start)
-               \(marker-position comint-last-output-start)
-             1))
-         \(process-mark (get-buffer-process (current-buffer)))
-         last-context ))
-  s)
+Every call to this function will set and use the buffer-local variable
+`ansi-color-context-region' to save position and current face.  This
+information will be used for the next call to
+`ansi-color-apply-on-region'.  Specifically, it will override BEGIN, the
+start of the region and set the face with which to start.  Set
+`ansi-color-context-region' to nil if you don't want this."
+  (let ((face (car ansi-color-context-region))
+	(start-marker (or (cadr ansi-color-context-region) 
+			  (copy-marker begin)))
+	(end-marker (copy-marker end))
+	escape-sequence)
+    (save-excursion
+      (goto-char start-marker)
+      ;; find the next escape sequence
+      (while (re-search-forward ansi-color-regexp end-marker t)
+	;; colorize the old block from start to end using old face
+	(when face
+	  (ansi-color-set-extent-face
+	   (ansi-color-make-extent start-marker (match-beginning 0))
+	   face))
+        ;; store escape sequence and new start position
+        (setq escape-sequence (match-string 1)
+	      start-marker (copy-marker (match-end 0)))
+	;; delete the escape sequence
+	(replace-match "")
+	;; create new face by applying all the parameters in the escape
+	;; sequence
+	(setq face (ansi-color-apply-sequence escape-sequence face)))
+      ;; search for the possible start of a new escape sequence
+      (if (re-search-forward "\033" end-marker t)
+	  (progn
+	    ;; if the rest of the region should have a face, put it there
+	    (when face
+	      (ansi-color-set-extent-face
+	       (ansi-color-make-extent start-marker (point))
+	       face))
+	    ;; save face and point
+	    (setq ansi-color-context-region
+		  (list face (copy-marker (match-beginning 0)))))
+	;; if the rest of the region should have a face, put it there
+	(if face
+	    (progn
+	      (ansi-color-set-extent-face
+	       (ansi-color-make-extent start-marker end-marker)
+	       face)
+	      (setq ansi-color-context-region (list face)))
+	  ;; reset context
+	  (setq ansi-color-context-region nil))))))
 
-\(add-hook 'comint-output-filter-functions 'ansi-output-filter)
-"
-  (let ((endm (copy-marker end))
-        (face (if (and context (cdr context))
-                  (cdr (cdr context))))
-	(face-start (if (and context (cdr context))
-                        (car (cdr context))))
-        (next-safe-start begin)
-        escape-sequence
-        null-sequence
-        stop )
-    (save-excursion
-      (goto-char begin)
-      ;; find the next escape sequence
-      (while (setq stop (re-search-forward ansi-color-regexp endm t))
-        ;; store escape sequence
-        (setq escape-sequence (match-string 1))
-        (setq null-sequence (string-equal (match-string 1) ""))
-        (setq next-safe-start (match-beginning 0))
-        (if face
-            (put-text-property face-start next-safe-start 'face face)) ; colorize
-        (replace-match "") ; delete the ANSI sequence
-        (if null-sequence
-            (setq face nil)
-          (setq face-start next-safe-start)
-          (setq face (ansi-color-get-face escape-sequence))))
-      (setq next-safe-start
-            (if (re-search-forward "\033" endm t)
-                (match-beginning 0)
-              (marker-position endm))))
-    (cons next-safe-start
-          (if face
-              (cons face-start face))) ))
+;; This function helps you look for overlapping overlays.  This is
+;; usefull in comint-buffers.  Overlapping overlays should not happen!
+;; A possible cause for bugs are the markers.  If you create an overlay
+;; up to the end of the region, then that end might coincide with the
+;; process-mark.  As text is added BEFORE the process-mark, the overlay
+;; will keep growing.  Therefore, as more overlays are created later on,
+;; there will be TWO OR MORE overlays covering the buffer at that point.
+;; This function helps you check your buffer for these situations.
+; (defun ansi-color-debug-overlays ()
+;   (interactive)
+;   (let ((pos (point-min)))
+;     (while (< pos (point-max))
+;       (if (<= 2 (length (overlays-at pos)))
+; 	  (progn
+; 	    (goto-char pos)
+; 	    (error "%d overlays at %d" (length (overlays-at pos)) pos))
+; 	(let (message-log-max)
+; 	  (message  "Reached %d." pos)))
+;       (setq pos (next-overlay-change pos)))))
+
+;; Emacs/XEmacs compatibility layer
+
+(defun ansi-color-make-face (property color)
+  "Return a face with PROPERTY set to COLOR.
+PROPERTY can be either symbol `foreground' or symbol `background'.  
+
+For Emacs, we just return the cons cell \(PROPERTY . COLOR).
+For XEmacs, we create a temporary face and return it."
+  (if (featurep 'xemacs)
+      (let ((face (make-face (intern (concat color "-" (symbol-name property)))
+			     "Temporary face created by ansi-color."
+			     t)))
+	(set-face-property face property color)
+	face)
+    (cond ((eq property 'foreground)
+	   (cons 'foreground-color color))
+	  ((eq property 'background)
+	   (cons 'background-color color))
+	  (t
+	   (cons property color)))))
+
+(defun ansi-color-make-extent (from to &optional object)
+  "Make an extent for the range [FROM, TO) in OBJECT.
+
+OBJECT defaults to the current buffer.  XEmacs uses `make-extent', Emacs
+uses `make-overlay'.  XEmacs can use a buffer or a string for OBJECT,
+Emacs requires OBJECT to be a buffer."
+  (if (functionp 'make-extent)
+      (make-extent from to object)
+    ;; In Emacs, the overlay might end at the process-mark in comint
+    ;; buffers.  In that case, new text will be inserted before the
+    ;; process-mark, ie. inside the overlay (using insert-before-marks).
+    ;; In order to avoid this, we use the `insert-behind-hooks' overlay
+    ;; property to make sure it works.
+    (let ((overlay (make-overlay from to object)))
+      (overlay-put overlay 'modification-hooks '(ansi-color-freeze-overlay))
+      overlay)))
+
+(defun ansi-color-freeze-overlay (overlay is-after begin end &optional len)
+  "Prevent OVERLAY from being extended.
+This function can be used for the `modification-hooks' overlay
+property."
+  ;; if stuff was inserted at the end of the overlay
+  (when (and is-after
+	     (= 0 len)
+	     (= end (overlay-end overlay)))
+    ;; reset the end of the overlay
+    (move-overlay overlay (overlay-start overlay) begin)))
+
+(defun ansi-color-set-extent-face (extent face)
+  "Set the `face' property of EXTENT to FACE.
+XEmacs uses `set-extent-face', Emacs  uses `overlay-put'."
+  (if (functionp 'set-extent-face)
+      (set-extent-face extent face)
+    (overlay-put extent 'face face)))
 
 ;; Helper functions
 
+(defun ansi-color-apply-sequence (escape-sequence faces)
+  "Apply ESCAPE-SEQ to FACES and return the new list of faces.
+
+ESCAPE-SEQ is an escape sequences parsed by `ansi-color-get-face'.
+
+If the new faces start with the symbol `default', then the new
+faces are returned.  If the faces start with something else,
+they are appended to the front of the FACES list, and the new
+list of faces is returned.
+
+If `ansi-color-get-face' returns nil, then we either got a
+null-sequence, or we stumbled upon some garbage.  In either
+case we return nil."
+  (let ((new-faces (ansi-color-get-face escape-sequence)))
+    (cond ((null new-faces)
+	   nil)
+	  ((eq (car new-faces) 'default)
+	   (cdr new-faces))
+	  (t
+	   (append new-faces face)))))
+
 (defun ansi-color-make-color-map ()
   "Creates a vector of face definitions and returns it.
 
@@ -339,28 +568,26 @@
                  (aset ansi-color-map index e)
                  (setq index (1+ index)) ))
      ansi-color-faces-vector)
-
     ;; foreground attributes
     (setq index 30)
     (mapcar
      (function (lambda (e)
                  (aset ansi-color-map index
-                       (cons 'foreground-color e))
+		       (ansi-color-make-face 'foreground e))
                  (setq index (1+ index)) ))
      ansi-color-names-vector)
-
     ;; background attributes
     (setq index 40)
     (mapcar
      (function (lambda (e)
                  (aset ansi-color-map index
-                       (cons 'background-color e))
+		       (ansi-color-make-face 'background e))
                  (setq index (1+ index)) ))
      ansi-color-names-vector)
     ansi-color-map))
 
 (defvar ansi-color-map (ansi-color-make-color-map)
-  "A brand new color map suitable for ansi-color-get-face.
+  "A brand new color map suitable for `ansi-color-get-face'.
 
 The value of this variable is usually constructed by
 `ansi-color-make-color-map'.  The values in the array are such that the
@@ -390,16 +617,23 @@
 (defun ansi-color-get-face (escape-seq)
   "Create a new face by applying all the parameters in ESCAPE-SEQ.
 
-ESCAPE-SEQ is a SGR control sequences such as \033[34m.  The parameter
+Should any of the parameters result in the default face (usually this is
+the parameter 0), then the effect of all previous parameters is cancelled.
+
+ESCAPE-SEQ is a SGR control sequences such as \\033[34m.  The parameter
 34 is used by `ansi-color-get-face-1' to return a face definition."
   (let ((ansi-color-r "[0-9][0-9]?")
         (i 0)
-        f)
+        f val)
     (while (string-match ansi-color-r escape-seq i)
-      (setq i (match-end 0))
-      (add-to-list 'f
-                   (ansi-color-get-face-1
-                    (string-to-int (match-string 0 escape-seq) 10))))
+      (setq i (match-end 0)
+	    val (ansi-color-get-face-1
+		 (string-to-int (match-string 0 escape-seq) 10)))
+      (cond ((not val))
+	    ((eq val 'default)
+	     (setq f (list val)))
+	    (t
+	     (add-to-list 'f val))))
     f))
 
 (provide 'ansi-color)