summaryrefslogtreecommitdiff
path: root/lisp/whitespace.el
diff options
context:
space:
mode:
Diffstat (limited to 'lisp/whitespace.el')
-rw-r--r--lisp/whitespace.el325
1 files changed, 265 insertions, 60 deletions
diff --git a/lisp/whitespace.el b/lisp/whitespace.el
index 5c65e24d405..d156d47f12c 100644
--- a/lisp/whitespace.el
+++ b/lisp/whitespace.el
@@ -6,7 +6,7 @@
;; Author: Vinicius Jose Latorre <viniciusjl@ig.com.br>
;; Maintainer: Vinicius Jose Latorre <viniciusjl@ig.com.br>
;; Keywords: data, wp
-;; Version: 9.2
+;; Version: 9.3
;; X-URL: http://www.emacswiki.org/cgi-bin/wiki/ViniciusJoseLatorre
;; This file is part of GNU Emacs.
@@ -162,15 +162,18 @@
;;
;; There are also the following useful commands:
;;
+;; `whitespace-report'
+;; Report some blank problems in buffer.
+;;
+;; `whitespace-report-region'
+;; Report some blank problems in a region.
+;;
;; `whitespace-cleanup'
;; Cleanup some blank problems in all buffer or at region.
;;
;; `whitespace-cleanup-region'
;; Cleanup some blank problems at region.
;;
-;; `whitespace-buffer'
-;; Turn on `whitespace-mode' forcing some settings.
-;;
;; The problems, which are cleaned up, are:
;;
;; 1. empty lines at beginning of buffer.
@@ -188,7 +191,7 @@
;;
;; 5. SPACEs or TABs at end of line.
;; If `whitespace-chars' includes the value `trailing', remove all
-;; SPACEs or TABs at end of line."
+;; SPACEs or TABs at end of line.
;;
;; 6. 8 or more SPACEs after TAB.
;; If `whitespace-chars' includes the value `space-after-tab',
@@ -280,10 +283,16 @@
;; `whitespace-mode' is automagically
;; turned on.
;;
+;; `whitespace-action' Specify which action is taken when a
+;; buffer is visited, killed or written.
+;;
;;
;; Acknowledgements
;; ----------------
;;
+;; Thanks to Eric Cooper <ecc@cmu.edu> for the suggestion to have hook actions
+;; when buffer is written or killed as the original whitespace package had.
+;;
;; Thanks to nschum (EmacsWiki) for the idea about highlight "long"
;; lines tail. See EightyColumnRule (EmacsWiki).
;;
@@ -786,9 +795,6 @@ and `whitespace-chars' includes `lines' or `lines-tail'."
;; Hacked from `visible-whitespace-mappings' in visws.el
(defcustom whitespace-display-mappings
- ;; Due to limitations of glyph representation, the char code can not
- ;; be above ?\x1FFFF. Probably, this will be fixed after Emacs
- ;; unicode merging.
'(
(?\ [?\xB7] [?.]) ; space - centered dot
(?\xA0 [?\xA4] [?_]) ; hard space - currency
@@ -797,8 +803,8 @@ and `whitespace-chars' includes `lines' or `lines-tail'."
(?\xE20 [?\xE24] [?_]) ; hard space - currency
(?\xF20 [?\xF24] [?_]) ; hard space - currency
;; NEWLINE is displayed using the face `whitespace-newline'
- (?\n [?$ ?\n]) ; end-of-line - dollar sign
- ;; (?\n [?\u21B5 ?\n] [?$ ?\n]) ; end-of-line - downwards arrow
+ (?\n [?\u21B5 ?\n] [?$ ?\n]) ; end-of-line - downwards arrow
+ ;; (?\n [?$ ?\n]) ; end-of-line - dollar sign
;; (?\n [?\xB6 ?\n] [?$ ?\n]) ; end-of-line - pilcrow
;; (?\n [?\x8AF ?\n] [?$ ?\n]) ; end-of-line - overscore
;; (?\n [?\x8AC ?\n] [?$ ?\n]) ; end-of-line - negation
@@ -863,7 +869,8 @@ of the list is negated if it begins with `not'. For example:
means that `whitespace-mode' is turned on for buffers in C and
C++ modes only."
- :type '(choice (const :tag "None" nil)
+ :type '(choice :tag "Global Modes"
+ (const :tag "None" nil)
(const :tag "All" t)
(set :menu-tag "Mode Specific" :tag "Modes"
:value (not)
@@ -872,6 +879,41 @@ C++ modes only."
(symbol :tag "Mode"))))
:group 'whitespace)
+
+(defcustom whitespace-action nil
+ "*Specify which action is taken when a buffer is visited, killed or written.
+
+It's a list containing some or all of the following values:
+
+ nil no action is taken.
+
+ cleanup cleanup any bogus whitespace always when local
+ whitespace is turned on.
+ See `whitespace-cleanup' and
+ `whitespace-cleanup-region'.
+
+ report-on-bogus report if there is any bogus whitespace always
+ when local whitespace is turned on.
+
+ auto-cleanup cleanup any bogus whitespace when buffer is
+ written or killed.
+ See `whitespace-cleanup' and
+ `whitespace-cleanup-region'.
+
+ abort-on-bogus abort if there is any bogus whitespace and the
+ buffer is written or killed.
+
+Any other value is treated as nil."
+ :type '(choice :tag "Actions"
+ (const :tag "None" nil)
+ (repeat :tag "Action List"
+ (choice :tag "Action"
+ (const :tag "Cleanup When On" cleanup)
+ (const :tag "Report On Bogus" report-on-bogus)
+ (const :tag "Auto Cleanup" auto-cleanup)
+ (const :tag "Abort On Bogus" abort-on-bogus))))
+ :group 'whitespace)
+
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;; User commands - Local mode
@@ -893,7 +935,8 @@ Only useful with a windowing system."
(noninteractive ; running a batch job
(setq whitespace-mode nil))
(whitespace-mode ; whitespace-mode on
- (whitespace-turn-on))
+ (whitespace-turn-on)
+ (whitespace-action-when-on))
(t ; whitespace-mode off
(whitespace-turn-off))))
@@ -918,7 +961,7 @@ Only useful with a windowing system."
(setq global-whitespace-mode nil))
(global-whitespace-mode ; global-whitespace-mode on
(save-excursion
- (add-hook 'find-file-hook 'whitespace-turn-on-if-enabled t)
+ (add-hook 'find-file-hook 'whitespace-turn-on-if-enabled)
(dolist (buffer (buffer-list)) ; adjust all local mode
(set-buffer buffer)
(unless whitespace-mode
@@ -1259,14 +1302,14 @@ The problems cleaned up are:
(while (re-search-forward
whitespace-indentation-regexp rend t)
(setq tmp (current-indentation))
+ (goto-char (match-beginning 0))
(delete-horizontal-space)
(unless (eolp)
(indent-to tmp))))
;; problem 3: SPACEs or TABs at eol
;; action: remove all SPACEs or TABs at eol
(when (memq 'trailing whitespace-chars)
- (let ((regexp (concat "\\(\\(" whitespace-trailing-regexp
- "\\)+\\)$")))
+ (let ((regexp (whitespace-trailing-regexp)))
(goto-char rstart)
(while (re-search-forward regexp rend t)
(delete-region (match-beginning 1) (match-end 1)))))
@@ -1300,24 +1343,103 @@ The problems cleaned up are:
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
-;;;; User command - old whitespace compatibility
+;;;; User command - report
+
+
+(defun whitespace-trailing-regexp ()
+ "Make the `whitespace-trailing-regexp' regexp."
+ (concat "\\(\\(" whitespace-trailing-regexp "\\)+\\)$"))
+
+
+(defconst whitespace-report-list
+ (list
+ (cons 'empty whitespace-empty-at-bob-regexp)
+ (cons 'empty whitespace-empty-at-eob-regexp)
+ (cons 'indentation whitespace-indentation-regexp)
+ (cons 'space-before-tab whitespace-space-before-tab-regexp)
+ (cons 'trailing (whitespace-trailing-regexp))
+ (cons 'space-after-tab whitespace-space-after-tab-regexp)
+ )
+ "List of whitespace bogus symbol and corresponding regexp.")
+
+
+(defconst whitespace-report-text
+ "\
+ Whitespace Report
+
+ Current Setting Whitespace Problem
+
+ empty [] [] empty lines at beginning of buffer.
+ empty [] [] empty lines at end of buffer.
+ indentation [] [] 8 or more SPACEs at beginning of line.
+ space-before-tab [] [] SPACEs before TAB.
+ trailing [] [] SPACEs or TABs at end of line.
+ space-after-tab [] [] 8 or more SPACEs after TAB.\n\n"
+ "Text for whitespace bogus report.")
+
+
+(defconst whitespace-report-buffer-name "*Whitespace Report*"
+ "The buffer name for whitespace bogus report.")
;;;###autoload
-(defun whitespace-buffer ()
- "Turn on `whitespace-mode' forcing some settings.
+(defun whitespace-report (&optional force report-if-bogus)
+ "Report some whitespace problems in buffer.
-It forces `whitespace-style' to have `color'.
+Return nil if there is no whitespace problem; otherwise, return
+non-nil.
-It also forces `whitespace-chars' to have:
+If FORCE is non-nil or \\[universal-argument] was pressed just before calling
+`whitespace-report' interactively, it forces `whitespace-chars' to
+have:
- trailing
+ empty
indentation
space-before-tab
+ trailing
+ space-after-tab
+
+If REPORT-IF-BOGUS is non-nil, it reports only when there are any
+whitespace problems in buffer.
+
+Report if some of the following whitespace problems exist:
+
+ empty 1. empty lines at beginning of buffer.
+ empty 2. empty lines at end of buffer.
+ indentation 3. 8 or more SPACEs at beginning of line.
+ space-before-tab 4. SPACEs before TAB.
+ trailing 5. SPACEs or TABs at end of line.
+ space-after-tab 6. 8 or more SPACEs after TAB.
+
+See `whitespace-chars' and `whitespace-style' for documentation.
+See also `whitespace-cleanup' and `whitespace-cleanup-region' for
+cleaning up these problems."
+ (interactive (list current-prefix-arg))
+ (whitespace-report-region (point-min) (point-max)
+ force report-if-bogus))
+
+
+;;;###autoload
+(defun whitespace-report-region (start end &optional force report-if-bogus)
+ "Report some whitespace problems in a region.
+
+Return nil if there is no whitespace problem; otherwise, return
+non-nil.
+
+If FORCE is non-nil or \\[universal-argument] was pressed just before calling
+`whitespace-report-region' interactively, it forces `whitespace-chars'
+to have:
+
empty
+ indentation
+ space-before-tab
+ trailing
space-after-tab
-So, it is possible to visualize the following problems:
+If REPORT-IF-BOGUS is non-nil, it reports only when there are any
+whitespace problems in buffer.
+
+Report if some of the following whitespace problems exist:
empty 1. empty lines at beginning of buffer.
empty 2. empty lines at end of buffer.
@@ -1329,21 +1451,41 @@ So, it is possible to visualize the following problems:
See `whitespace-chars' and `whitespace-style' for documentation.
See also `whitespace-cleanup' and `whitespace-cleanup-region' for
cleaning up these problems."
- (interactive)
- (whitespace-mode 0) ; assure is off
- ;; keep original values
- (let ((whitespace-style (copy-sequence whitespace-style))
- (whitespace-chars (copy-sequence whitespace-chars)))
- ;; adjust options for whitespace bogus blanks
- (add-to-list 'whitespace-style 'color)
- (mapc #'(lambda (option)
- (add-to-list 'whitespace-chars option))
- '(trailing
- indentation
- space-before-tab
- empty
- space-after-tab))
- (whitespace-mode 1))) ; turn on
+ (interactive "r")
+ (setq force (or current-prefix-arg force))
+ (save-excursion
+ (save-match-data
+ (let* (has-bogus
+ (rstart (min start end))
+ (rend (max start end))
+ (bogus-list (mapcar
+ #'(lambda (option)
+ (when force
+ (add-to-list 'whitespace-chars (car option)))
+ (goto-char rstart)
+ (and (re-search-forward (cdr option) rend t)
+ (setq has-bogus t)))
+ whitespace-report-list)))
+ (when (if report-if-bogus has-bogus t)
+ (with-current-buffer (get-buffer-create
+ whitespace-report-buffer-name)
+ (erase-buffer)
+ (insert whitespace-report-text)
+ (goto-char (point-min))
+ (forward-line 3)
+ (dolist (option whitespace-report-list)
+ (forward-line 1)
+ (whitespace-mark-x 22 (memq (car option) whitespace-chars))
+ (whitespace-mark-x 7 (car bogus-list))
+ (setq bogus-list (cdr bogus-list)))
+ (when has-bogus
+ (goto-char (point-max))
+ (insert " Type `M-x whitespace-cleanup'"
+ " to cleanup the buffer.\n\n")
+ (insert " Type `M-x whitespace-cleanup-region'"
+ " to cleanup a region.\n\n"))
+ (whitespace-display-window (current-buffer))))
+ has-bogus))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
@@ -1392,13 +1534,18 @@ cleaning up these problems."
"The buffer name for whitespace toggle options.")
+(defun whitespace-mark-x (nchars condition)
+ "Insert the mark ('X' or ' ') after NCHARS depending on CONDITION."
+ (forward-char nchars)
+ (insert (if condition "X" " ")))
+
+
(defun whitespace-insert-option-mark (the-list the-value)
"Insert the option mark ('X' or ' ') in toggle options buffer."
(forward-line 1)
(dolist (sym the-list)
(forward-line 1)
- (forward-char 2)
- (insert (if (memq sym the-value) "X" " "))))
+ (whitespace-mark-x 2 (memq sym the-value))))
(defun whitespace-help-on (chars style)
@@ -1415,17 +1562,22 @@ cleaning up these problems."
whitespace-chars-value-list chars)
(whitespace-insert-option-mark
whitespace-style-value-list style)
- (goto-char (point-min))
- (set-buffer-modified-p nil)
- (let ((size (- (window-height)
- (max window-min-height
- (1+ (count-lines (point-min)
- (point-max)))))))
- (when (<= size 0)
- (kill-buffer buffer)
- (error "Frame height is too small; \
+ (whitespace-display-window buffer)))))
+
+
+(defun whitespace-display-window (buffer)
+ "Display BUFFER in a new window."
+ (goto-char (point-min))
+ (set-buffer-modified-p nil)
+ (let ((size (- (window-height)
+ (max window-min-height
+ (1+ (count-lines (point-min)
+ (point-max)))))))
+ (when (<= size 0)
+ (kill-buffer buffer)
+ (error "Frame height is too small; \
can't split window to display whitespace toggle options"))
- (set-window-buffer (split-window nil size) buffer))))))
+ (set-window-buffer (split-window nil size) buffer)))
(defun whitespace-help-off ()
@@ -1538,6 +1690,7 @@ options are valid."
(defun whitespace-turn-on ()
"Turn on whitespace visualization."
+ (whitespace-add-local-hook)
(setq whitespace-active-style (if (listp whitespace-style)
whitespace-style
(list whitespace-style)))
@@ -1552,6 +1705,7 @@ options are valid."
(defun whitespace-turn-off ()
"Turn off whitespace visualization."
+ (whitespace-remove-local-hook)
(when (memq 'color whitespace-active-style)
(whitespace-color-off))
(when (memq 'mark whitespace-active-style)
@@ -1590,8 +1744,7 @@ options are valid."
nil
(list
;; Show trailing blanks
- (list (concat "\\(\\(" whitespace-trailing-regexp "\\)+\\)$")
- 1 whitespace-trailing t))
+ (list (whitespace-trailing-regexp) 1 whitespace-trailing t))
t))
(when (or (memq 'lines whitespace-active-chars)
(memq 'lines-tail whitespace-active-chars))
@@ -1727,11 +1880,7 @@ options are valid."
;; faces, font-lock faces, etc.
(when (memq 'color whitespace-active-style)
(dotimes (i (length vec))
- ;; Due to limitations of glyph representation, the char
- ;; code can not be above ?\x1FFFF. Probably, this will
- ;; be fixed after Emacs unicode merging.
(or (eq (aref vec i) ?\n)
- (> (aref vec i) #x1FFFF)
(aset vec i
(make-glyph-code (aref vec i)
whitespace-newline)))))
@@ -1752,14 +1901,70 @@ options are valid."
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
+;;;; Hook
+
+
+(defun whitespace-action-when-on ()
+ "Action to be taken always when local whitespace is turned on."
+ (cond ((memq 'cleanup whitespace-action)
+ (whitespace-cleanup))
+ ((memq 'report-on-bogus whitespace-action)
+ (whitespace-report nil t))))
+
+
+(defun whitespace-add-local-hook ()
+ "Add some whitespace hooks locally."
+ (add-hook 'write-file-functions 'whitespace-write-file-hook nil t)
+ (add-hook 'kill-buffer-hook 'whitespace-kill-buffer-hook nil t))
+
+
+(defun whitespace-remove-local-hook ()
+ "Remove some whitespace hooks locally."
+ (remove-hook 'write-file-functions 'whitespace-write-file-hook t)
+ (remove-hook 'kill-buffer-hook 'whitespace-kill-buffer-hook t))
+
+
+(defun whitespace-write-file-hook ()
+ "Action to be taken when buffer is written.
+It should be added buffer-locally to `write-file-functions'."
+ (when (whitespace-action)
+ (error "Abort write due to whitespace problems in %s"
+ (buffer-name)))
+ nil) ; continue hook processing
+
+
+(defun whitespace-kill-buffer-hook ()
+ "Action to be taken when buffer is killed.
+It should be added buffer-locally to `kill-buffer-hook'."
+ (whitespace-action)
+ nil) ; continue hook processing
+
+
+(defun whitespace-action ()
+ "Action to be taken when buffer is killed or written.
+Return t when the action should be aborted."
+ (cond ((memq 'auto-cleanup whitespace-action)
+ (whitespace-cleanup)
+ nil)
+ ((memq 'abort-on-bogus whitespace-action)
+ (whitespace-report nil t))
+ (t
+ nil)))
+
+
+;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
(defun whitespace-unload-function ()
- "Unload the Whitespace library."
- (let (whitespace-mode) ;; so g-w-m thinks it is nil in all buffers
- (global-whitespace-mode -1))
- ;; continue standard unloading
- nil)
+ "Unload the whitespace library."
+ (global-whitespace-mode -1)
+ ;; be sure all local whitespace mode is turned off
+ (save-current-buffer
+ (dolist (buf (buffer-list))
+ (set-buffer buf)
+ (whitespace-mode -1)))
+ nil) ; continue standard unloading
+
(provide 'whitespace)