diff options
Diffstat (limited to 'lisp')
| -rw-r--r-- | lisp/gnus/ChangeLog | 28 | ||||
| -rw-r--r-- | lisp/gnus/gnus.el | 22 | ||||
| -rw-r--r-- | lisp/gnus/nnimap.el | 355 | ||||
| -rw-r--r-- | lisp/gnus/nnir.el | 1 | ||||
| -rw-r--r-- | lisp/gnus/nnmail.el | 2 | ||||
| -rw-r--r-- | lisp/gnus/shr.el | 14 | ||||
| -rw-r--r-- | lisp/gnus/spam.el | 3 | 
7 files changed, 284 insertions, 141 deletions
| diff --git a/lisp/gnus/ChangeLog b/lisp/gnus/ChangeLog index 7947aff31bc..3b2a61e3d3d 100644 --- a/lisp/gnus/ChangeLog +++ b/lisp/gnus/ChangeLog @@ -1,3 +1,31 @@ +2010-10-09  Lars Magne Ingebrigtsen  <larsi@gnus.org> + +	* spam.el (gnus-summary-mode-map): Bind to "$".  Suggested by Russ +	Allbery. + +	* shr.el: Rework the way things are indented by <li> slightly. + +	* gnus.el (gnus-group-set-parameter): Fix typo. + +	* nnimap.el: Start implementing QRESYNC support. + +2010-10-09  Julien Danjou  <julien@danjou.info> + +	* nnir.el (nnir-engines): Fix too many arguments. + +2010-10-09  Lars Magne Ingebrigtsen  <larsi@gnus.org> + +	* nnmail.el (nnmail-expiry-target-group): Say that every expiry target +	group is the "last", so that the backends like nnfolder actually save +	their folders. + +	* nnimap.el (nnimap-open-connection): If we have gnutls loaded, then +	try to use that for the tls stream. +	(nnimap-retrieve-group-data-early): Rework the marks code to heed +	UIDVALIDITY and find out which groups are read-only and not. +	(nnimap-get-flags): Use the same marks parsing code as the rest of +	nnimap. +  2010-10-09  Julien Danjou  <julien@danjou.info>  	* nnir.el (nnir-read-parm): Fix call to gnus-completing-read. diff --git a/lisp/gnus/gnus.el b/lisp/gnus/gnus.el index f6ffdf64ffb..d4bbfe129c4 100644 --- a/lisp/gnus/gnus.el +++ b/lisp/gnus/gnus.el @@ -3391,14 +3391,14 @@ that that variable is buffer-local to the summary buffers."  (defun gnus-news-group-p (group &optional article)    "Return non-nil if GROUP (and ARTICLE) come from a news server."    (cond ((gnus-member-of-valid 'post group) ;Ordinary news group -	 t)				;is news of course. +	 t)				    ;is news of course.  	((not (gnus-member-of-valid 'post-mail group)) ;Non-combined.  	 nil)				;must be mail then.  	((vectorp article)		;Has header info.  	 (eq (gnus-request-type group (mail-header-id article)) 'news)) -	((null article)			;Hasn't header info +	((null article)			       ;Hasn't header info  	 (eq (gnus-request-type group) 'news)) ;(unknown ==> mail) -	((< article 0)			;Virtual message +	((< article 0)			       ;Virtual message  	 nil)				;we don't know, guess mail.  	(t				;Has positive number  	 (eq (gnus-request-type group article) 'news)))) ;use it. @@ -3923,8 +3923,11 @@ If ALLOW-LIST, also allow list as a result."  			   group 'params))))  (defun gnus-group-set-parameter (group name value) -  "Set parameter NAME to VALUE in GROUP." -  (let ((info (gnus-get-info group))) +  "Set parameter NAME to VALUE in GROUP. +GROUP can also be an INFO structure." +  (let ((info (if (listp group) +		  group +		(gnus-get-info group))))      (when info        (gnus-group-remove-parameter group name)        (let ((old-params (gnus-info-params info)) @@ -3934,11 +3937,14 @@ If ALLOW-LIST, also allow list as a result."  		    (not (eq (caar old-params) name)))  	    (setq new-params (append new-params (list (car old-params)))))  	  (setq old-params (cdr old-params))) -	(gnus-group-set-info new-params group 'params))))) +	(gnus-group-set-info new-params (gnus-info-group info) 'params)))))  (defun gnus-group-remove-parameter (group name) -  "Remove parameter NAME from GROUP." -  (let ((info (gnus-get-info group))) +  "Remove parameter NAME from GROUP. +GROUP can also be an INFO structure." +  (let ((info (if (listp group) +		  group +		(gnus-get-info group))))      (when info        (let ((params (gnus-info-params info)))  	(when params diff --git a/lisp/gnus/nnimap.el b/lisp/gnus/nnimap.el index f8eb6659ad6..b30e5868669 100644 --- a/lisp/gnus/nnimap.el +++ b/lisp/gnus/nnimap.el @@ -317,8 +317,7 @@ textual parts.")  		   'starttls))  		'("imap"))  	       ((memq nnimap-stream '(ssl tls)) -		(funcall (if (and nil -				  (fboundp 'open-gnutls-stream)) +		(funcall (if (fboundp 'open-gnutls-stream)  			     'open-gnutls-stream  			   'open-tls-stream)  			 "*nnimap*" (current-buffer) nnimap-address @@ -338,7 +337,8 @@ textual parts.")  			    '(open run))))  	    (nnheader-report 'nnimap "Unable to contact %s:%s via %s"  			     nnimap-address port nnimap-stream) -	  (gnus-set-process-query-on-exit-flag (nnimap-process nnimap-object) nil) +	  (gnus-set-process-query-on-exit-flag +	   (nnimap-process nnimap-object) nil)  	  (if (not (setq connection-result (nnimap-wait-for-connection)))  	      (nnheader-report 'nnimap  			       "%s" (buffer-substring @@ -626,8 +626,10 @@ textual parts.")  	      (setq marks  		    (nnimap-flags-to-marks  		     (nnimap-parse-flags -		      (list (list group-sequence flag-sequence 1 group))))) -	      (when info +		      (list (list group-sequence flag-sequence +				  1 group "SELECT"))))) +	      (when (and info +			 marks)  		(nnimap-update-infos marks (list info)))  	      (goto-char (point-max))  	      (let ((uidnext (nth 5 (car marks)))) @@ -655,7 +657,8 @@ textual parts.")  (deffoo nnimap-request-rename-group (group new-name &optional server)    (when (nnimap-possibly-change-group nil server)      (with-current-buffer (nnimap-buffer) -      (car (nnimap-command "RENAME %S %S" (utf7-encode group t) (utf7-encode new-name t)))))) +      (car (nnimap-command "RENAME %S %S" +			   (utf7-encode group t) (utf7-encode new-name t))))))  (deffoo nnimap-request-expunge-group (group &optional server)    (when (nnimap-possibly-change-group group server) @@ -664,16 +667,19 @@ textual parts.")  (defun nnimap-get-flags (spec)    (let ((articles nil) -	elems) +	elems end)      (with-current-buffer (nnimap-buffer)        (erase-buffer)        (nnimap-wait-for-response (nnimap-send-command  				 "UID FETCH %s FLAGS" spec)) +      (setq end (point)) +      (subst-char-in-region (point-min) (point-max) +			    ?\\ ?% t)        (goto-char (point-min)) -      (while (re-search-forward "^\\* [0-9]+ FETCH (\\(.*\\))" nil t) -	(setq elems (nnimap-parse-line (match-string 1))) -	(push (cons (string-to-number (cadr (member "UID" elems))) -		    (cadr (member "FLAGS" elems))) +      (while (search-forward " FETCH " end t) +	(setq elems (read (current-buffer))) +	(push (cons (cadr (memq 'UID elems)) +		    (cadr (memq 'FLAGS elems)))  	      articles)))      (nreverse articles))) @@ -940,41 +946,45 @@ textual parts.")  (deffoo nnimap-retrieve-group-data-early (server infos)    (when (nnimap-possibly-change-group nil server)      (with-current-buffer (nnimap-buffer) +      (erase-buffer) +      (setf (nnimap-group nnimap-object) nil)        ;; QRESYNC handling isn't implemented.        (let ((qresyncp (member "notQRESYNC" (nnimap-capabilities nnimap-object))) -	    marks groups sequences) +	    params groups sequences active uidvalidity modseq group)  	;; Go through the infos and gather the data needed to know  	;; what and how to request the data.  	(dolist (info infos) -	  (setq marks (gnus-info-marks info)) -	  (push (list (gnus-group-real-name (gnus-info-group info)) -		      (cdr (assq 'active marks)) -		      (cdr (assq 'uid marks))) -		groups)) -	;; Then request the data. -	(erase-buffer) -	(setf (nnimap-group nnimap-object) nil) -	(dolist (elem groups) +	  (setq params (gnus-info-params info) +		group (gnus-group-real-name (gnus-info-group info)) +		active (cdr (assq 'active params)) +		uidvalidity (cdr (assq 'uidvalidity params)) +		modseq (cdr (assq 'modseq params)))  	  (if (and qresyncp -		   (nth 2 elem)) +		   uidvalidity +		   modseq)  	      (push -	       (list 'qresync -		     (nnimap-send-command "EXAMINE %S (QRESYNC (%s %s))" -					  (car elem) -					  (car (nth 2 elem)) -					  (cdr (nth 2 elem))) -		     nil -		     (car elem)) +	       (list (nnimap-send-command "EXAMINE %S (QRESYNC (%s %s))" +					  group uidvalidity modseq) +		     'qresync +		     nil group 'qresync)  	       sequences)  	    (let ((start -		   (if (nth 1 elem) +		   (if (and active uidvalidity)  		       ;; Fetch the last 100 flags. -		       (max 1 (- (cdr (nth 1 elem)) 100)) -		     1))) -	      (push (list (nnimap-send-command "EXAMINE %S" (car elem)) +		       (max 1 (- (cdr active) 100)) +		     1)) +		  (command +		   (if uidvalidity +		       "EXAMINE" +		     ;; If we don't have a UIDVALIDITY, then this is +		     ;; the first time we've seen the group, so we +		     ;; have to do a SELECT (which is slower than an +		     ;; examine), but will tell us whether the group +		     ;; is read-only or not. +		     "SELECT"))) +	      (push (list (nnimap-send-command "%s %S" command group)  			  (nnimap-send-command "UID FETCH %d:* FLAGS" start) -			  start -			  (car elem)) +			  start group command)  		    sequences)))  	  ;; Some servers apparently can't have many outstanding  	  ;; commands, so throttle them. @@ -988,10 +998,13 @@ textual parts.")  	     (nnimap-possibly-change-group nil server))      (with-current-buffer (nnimap-buffer)        ;; Wait for the final data to trickle in. -      (when (nnimap-wait-for-response (cadar sequences) t) -	;; Now we should have all the data we need, no matter whether -	;; we're QRESYNCING, fetching all the flags from scratch, or -	;; just fetching the last 100 flags per group. +      (when (nnimap-wait-for-response (if (eq (cadar sequences) 'qresync) +					  (caar sequences) +					(cadar sequences)) +				      t) +	;; Now we should have most of the data we need, no matter +	;; whether we're QRESYNCING, fetching all the flags from +	;; scratch, or just fetching the last 100 flags per group.  	(nnimap-update-infos (nnimap-flags-to-marks  			      (nnimap-parse-flags  			       (nreverse sequences))) @@ -1011,17 +1024,33 @@ textual parts.")  (defun nnimap-update-infos (flags infos)    (dolist (info infos) -    (let ((group (gnus-group-real-name (gnus-info-group info)))) -      (nnimap-update-info info (cdr (assoc group flags)))))) +    (let* ((group (gnus-group-real-name (gnus-info-group info))) +	   (marks (cdr (assoc group flags)))) +      (when marks +	(nnimap-update-info info marks)))))  (defun nnimap-update-info (info marks) -  (when (and marks -	     ;; Ignore groups with no UIDNEXT/marks.  This happens for -	     ;; completely empty groups. -	     (or (car marks) -		 (nth 4 marks))) -    (destructuring-bind (existing flags high low uidnext start-article -				  permanent-flags) marks +  (destructuring-bind (existing flags high low uidnext start-article +				permanent-flags uidvalidity +				vanished highestmodseq) marks +    (cond +     ;; Ignore groups with no UIDNEXT/marks.  This happens for +     ;; completely empty groups. +     ((and (not existing) +	   (not uidnext)) +      ) +     ;; We have a mismatch between the old and new UIDVALIDITY +     ;; identifiers, so we have to re-request the group info (the next +     ;; time).  This virtually never happens. +     ((let ((old-uidvalidity +	     (cdr (assq 'uidvalidity (gnus-info-params info))))) +	(and old-uidvalidity +	     (not (equal old-uidvalidity uidvalidity)) +	     (> start-article 1))) +      (gnus-group-remove-parameter info 'uidvalidity) +      (gnus-group-remove-parameter info 'modseq)) +     ;; We have the data needed to update. +     (t        (let ((group (gnus-info-group info))  	    (completep (and start-article  			    (= start-article 1)))) @@ -1046,52 +1075,89 @@ textual parts.")  	   group  	   (cons (car (gnus-active group))  		 (or high (1- uidnext))))) -	;; Then update the list of read articles. -	(let* ((unread -		(gnus-compress-sequence -		 (gnus-set-difference -		  (gnus-set-difference -		   existing -		   (cdr (assoc '%Seen flags))) -		  (cdr (assoc '%Flagged flags))))) -	       (read (gnus-range-difference -		      (cons start-article high) unread))) -	  (when (> start-article 1) -	    (setq read -		  (gnus-range-nconcat -		   (if (> start-article 1) -		       (gnus-sorted-range-intersection -			(cons 1 (1- start-article)) -			(gnus-info-read info)) -		     (gnus-info-read info)) -		   read))) -	  (gnus-info-set-read info read) -	  ;; Update the marks. -	  (setq marks (gnus-info-marks info)) -	  ;; Note the active level for the next run-through. -	  (let ((active (assq 'active marks))) -	    (if active -		(setcdr active (gnus-active group)) -	      (push (cons 'active (gnus-active group)) marks))) -	  (dolist (type (cdr nnimap-mark-alist)) -	    (let ((old-marks (assoc (car type) marks)) -		  (new-marks -		   (gnus-compress-sequence -		    (cdr (or (assoc (caddr type) flags)	    ; %Flagged -			     (assoc (intern (cadr type) obarray) flags) -			     (assoc (cadr type) flags)))))) ; "\Flagged" -	      (setq marks (delq old-marks marks)) -	      (pop old-marks) -	      (when (and old-marks -			 (> start-article 1)) -		(setq old-marks (gnus-range-difference -				 old-marks -				 (cons start-article high))) -		(setq new-marks (gnus-range-nconcat old-marks new-marks))) -	      (when new-marks -		(push (cons (car type) new-marks) marks))) -	    (gnus-info-set-marks info marks t) -	    (nnimap-store-info info (gnus-active group)))))))) +	;; See whether this is a read-only group. +	(unless (eq permanent-flags 'not-scanned) +	  (gnus-group-set-parameter +	   info 'permanent-flags +	   (if (memq '%* permanent-flags) +	       t +	     nil))) +	;; Update marks and read articles if this isn't a +	;; read-only IMAP group. +	(when (cdr (assq 'permanent-flags (gnus-info-params info))) +	  (if (and highestmodseq +		   (not start-article)) +	      ;; We've gotten the data by QRESYNCing. +	      (nnimap-update-qresync-info +	       info (nnimap-imap-ranges-to-gnus-ranges vanished) flags) +	    ;; Do normal non-QRESYNC flag updates. +	    ;; Update the list of read articles. +	    (let* ((unread +		    (gnus-compress-sequence +		     (gnus-set-difference +		      (gnus-set-difference +		       existing +		       (cdr (assoc '%Seen flags))) +		      (cdr (assoc '%Flagged flags))))) +		   (read (gnus-range-difference +			  (cons start-article high) unread))) +	      (when (> start-article 1) +		(setq read +		      (gnus-range-nconcat +		       (if (> start-article 1) +			   (gnus-sorted-range-intersection +			    (cons 1 (1- start-article)) +			    (gnus-info-read info)) +			 (gnus-info-read info)) +		       read))) +	      (gnus-info-set-read info read) +	      ;; Update the marks. +	      (setq marks (gnus-info-marks info)) +	      (dolist (type (cdr nnimap-mark-alist)) +		(let ((old-marks (assoc (car type) marks)) +		      (new-marks +		       (gnus-compress-sequence +			(cdr (or (assoc (caddr type) flags) ; %Flagged +				 (assoc (intern (cadr type) obarray) flags) +				 (assoc (cadr type) flags)))))) ; "\Flagged" +		  (setq marks (delq old-marks marks)) +		  (pop old-marks) +		  (when (and old-marks +			     (> start-article 1)) +		    (setq old-marks (gnus-range-difference +				     old-marks +				     (cons start-article high))) +		    (setq new-marks (gnus-range-nconcat old-marks new-marks))) +		  (when new-marks +		    (push (cons (car type) new-marks) marks))) +		(gnus-info-set-marks info marks t))))) +	;; Note the active level for the next run-through. +	(gnus-group-set-parameter info 'active (gnus-active group)) +	(gnus-group-set-parameter info 'uidvalidity uidvalidity) +	(gnus-group-set-parameter info 'modseq highestmodseq) +	(nnimap-store-info info (gnus-active group))))))) + +(defun nnimap-update-qresync-info (info vanished flags) +  ;; Add all the vanished articles to the list of read articles. +  (gnus-info-set-read +   info +   (gnus-range-add (gnus-info-read info) +		   vanished)) +  ) + +(defun nnimap-imap-ranges-to-gnus-ranges (irange) +  (if (zerop (length irange)) +      nil +    (let ((result nil)) +      (dolist (elem (split-string irange ",")) +	(push +	 (if (string-match ":" elem) +	     (let ((numbers (split-string elem ":"))) +	       (cons (string-to-number (car numbers)) +		     (string-to-number (cadr numbers)))) +	   (string-to-number elem)) +	 result)) +      (nreverse result))))  (defun nnimap-store-info (info active)    (let* ((group (gnus-group-real-name (gnus-info-group info))) @@ -1101,13 +1167,17 @@ textual parts.")        (push (list group info active) nnimap-current-infos))))  (defun nnimap-flags-to-marks (groups) -  (let (data group totalp uidnext articles start-article mark permanent-flags) +  (let (data group totalp uidnext articles start-article mark permanent-flags +	     uidvalidity vanished highestmodseq)      (dolist (elem groups)        (setq group (car elem)  	    uidnext (nth 1 elem)  	    start-article (nth 2 elem)  	    permanent-flags (nth 3 elem) -	    articles (nthcdr 4 elem)) +	    uidvalidity (nth 4 elem) +	    vanished (nth 5 elem) +	    highestmodseq (nth 6 elem) +	    articles (nthcdr 7 elem))        (let ((high (caar articles))  	    marks low existing)  	(dolist (article articles) @@ -1119,7 +1189,7 @@ textual parts.")  		(push (list flag (car article)) marks)  	      (setcdr mark (cons (car article) (cdr mark))))))  	(push (list group existing marks high low uidnext start-article -		    permanent-flags) +		    permanent-flags uidvalidity vanished highestmodseq)  	      data)))      data)) @@ -1128,38 +1198,69 @@ textual parts.")    ;; Change \Delete etc to %Delete, so that the reader can read it.    (subst-char-in-region (point-min) (point-max)  			?\\ ?% t) -  (let (start end articles groups uidnext elems permanent-flags) +  (let (start end articles groups uidnext elems permanent-flags +	      uidvalidity vanished highestmodseq)      (dolist (elem sequences) -      (destructuring-bind (group-sequence flag-sequence totalp group) elem +      (destructuring-bind (group-sequence flag-sequence totalp group command) +	  elem  	(setq start (point)) -	;; The EXAMINE was successful. -	(when (and (search-forward (format "\n%d OK " group-sequence) nil t) -		   (progn -		     (forward-line 1) -		     (setq end (point)) -		     (goto-char start) -		     (setq permanent-flags +	(when (and +	       ;; The EXAMINE was successful. +	       (search-forward (format "\n%d OK " group-sequence) nil t) +	       (progn +		 (forward-line 1) +		 (setq end (point)) +		 (goto-char start) +		 (setq permanent-flags +		       (if (equal command "SELECT")  			   (and (search-forward "PERMANENTFLAGS " -						 (or end (point-min)) t) -				(read (current-buffer)))) -		     (goto-char start) -		     (setq uidnext -			   (and (search-forward "UIDNEXT " -						 (or end (point-min)) t) -				(read (current-buffer)))) -		     (goto-char end) -		     (forward-line -1)) -		   ;; The UID FETCH FLAGS was successful. -		   (search-forward (format "\n%d OK " flag-sequence) nil t)) -	  (setq start (point)) -	  (goto-char end) +						(or end (point-min)) t) +				(read (current-buffer))) +			 'not-scanned)) +		 (goto-char start) +		 (setq uidnext +		       (and (search-forward "UIDNEXT " +					    (or end (point-min)) t) +			    (read (current-buffer)))) +		 (goto-char start) +		 (setq uidvalidity +		       (and (re-search-forward "UIDVALIDITY \\([0-9]+\\)" +					       (or end (point-min)) t) +			    ;; Store UIDVALIDITY as a string, as it's +			    ;; too big for 32-bit Emacsen, usually. +			    (match-string 1))) +		 (goto-char start) +		 (setq vanished +		       (and (eq flag-sequence 'qresync) +			    (re-search-forward "VANISHED.* \\([0-9:,]+\\)" +					       (or end (point-min)) t) +			    (match-string 1))) +		 (goto-char start) +		 (setq highestmodseq +		       (and (search-forward "HIGHESTMODSEQ " +					    (or end (point-min)) t) +			    (read (current-buffer)))) +		 (goto-char end) +		 (forward-line -1)) +	       ;; The UID FETCH FLAGS was successful. +	       (or (eq flag-sequence 'qresync) +		   (search-forward (format "\n%d OK " flag-sequence) nil t))) +	  (if (eq flag-sequence 'qresync) +	      (progn +		(goto-char start) +		(setq start end)) +	    (setq start (point)) +	    (goto-char end))  	  (while (search-forward " FETCH " start t)  	    (setq elems (read (current-buffer)))  	    (push (cons (cadr (memq 'UID elems))  			(cadr (memq 'FLAGS elems)))  		  articles)) -	  (push (nconc (list group uidnext totalp permanent-flags) articles) +	  (push (nconc (list group uidnext totalp permanent-flags uidvalidity +			     vanished highestmodseq) +		       articles)  		groups) +	  (goto-char end)  	  (setq articles nil))))      groups)) @@ -1293,13 +1394,15 @@ textual parts.")  	  (push  	   (cond  	    ((eql char ?\[) -	     (split-string (buffer-substring -			    (1+ (point)) -			    (1- (search-forward "]" (line-end-position) 'move))))) +	     (split-string +	      (buffer-substring +	       (1+ (point)) +	       (1- (search-forward "]" (line-end-position) 'move)))))  	    ((eql char ?\() -	     (split-string (buffer-substring -			    (1+ (point)) -			    (1- (search-forward ")" (line-end-position) 'move))))) +	     (split-string +	      (buffer-substring +	       (1+ (point)) +	       (1- (search-forward ")" (line-end-position) 'move)))))  	    ((eql char ?\")  	     (forward-char 1)  	     (buffer-substring diff --git a/lisp/gnus/nnir.el b/lisp/gnus/nnir.el index 6d64935550f..86acad16638 100644 --- a/lisp/gnus/nnir.el +++ b/lisp/gnus/nnir.el @@ -377,7 +377,6 @@ result, `gnus-retrieve-headers' will be called instead.")               ((criteria  	       "Search in: "                      ; Prompt  	       ,(mapcar 'car nnir-imap-search-arguments) ; alist for completing -	       nil                                ; no filtering  	       nil                                ; allow any user input  	       nil                                ; initial value  	       nnir-imap-search-argument-history  ; the history to use diff --git a/lisp/gnus/nnmail.el b/lisp/gnus/nnmail.el index 463f87e44fc..01cab6febe3 100644 --- a/lisp/gnus/nnmail.el +++ b/lisp/gnus/nnmail.el @@ -1914,7 +1914,7 @@ If TIME is nil, then return the cutoff time for oldness instead."      (unless (eq target 'delete)        (when (or (gnus-request-group target)  		(gnus-request-create-group target)) -	(let ((group-art (gnus-request-accept-article target nil nil t))) +	(let ((group-art (gnus-request-accept-article target nil t t)))  	  (when (consp group-art)  	    (gnus-group-mark-article-read target (cdr group-art)))))))) diff --git a/lisp/gnus/shr.el b/lisp/gnus/shr.el index 8f246a06e36..0700621a912 100644 --- a/lisp/gnus/shr.el +++ b/lisp/gnus/shr.el @@ -230,7 +230,7 @@ redirects somewhere else."  (defun shr-ensure-paragraph ()    (unless (bobp) -    (if (bolp) +    (if (<= (current-column) shr-indentation)  	(unless (save-excursion  		  (forward-line -1)  		  (looking-at " *$")) @@ -242,7 +242,8 @@ redirects somewhere else."  	(insert "\n\n")))))  (defun shr-indent () -  (insert (make-string shr-indentation ? ))) +  (when (> shr-indentation 0) +    (insert (make-string shr-indentation ? ))))  (defun shr-fontize-cont (cont &rest types)    (let (shr-start) @@ -332,6 +333,7 @@ Return a string with image data."  (defun shr-tag-p (cont)    (shr-ensure-paragraph) +  (shr-indent)    (shr-generic cont)    (shr-ensure-paragraph)) @@ -404,11 +406,13 @@ Return a string with image data."  (defun shr-tag-pre (cont)    (let ((shr-folding-mode 'none))      (shr-ensure-newline) +    (shr-indent)      (shr-generic cont)      (shr-ensure-newline)))  (defun shr-tag-blockquote (cont)    (shr-ensure-paragraph) +  (shr-indent)    (let ((shr-indentation (+ shr-indentation 4)))      (shr-generic cont))    (shr-ensure-paragraph)) @@ -426,7 +430,8 @@ Return a string with image data."    (shr-ensure-paragraph))  (defun shr-tag-li (cont) -  (shr-ensure-newline) +  (shr-ensure-paragraph) +  (shr-indent)    (let* ((bullet  	  (if (numberp shr-list-mode)  	      (prog1 @@ -439,7 +444,8 @@ Return a string with image data."  (defun shr-tag-br (cont)    (unless (bobp) -    (insert "\n")) +    (insert "\n") +    (shr-indent))    (shr-generic cont))  (defun shr-tag-h1 (cont) diff --git a/lisp/gnus/spam.el b/lisp/gnus/spam.el index b7908e5507b..9e193ccea84 100644 --- a/lisp/gnus/spam.el +++ b/lisp/gnus/spam.el @@ -689,7 +689,8 @@ order for SpamAssassin to recognize the new registered spam."    "Sx" gnus-summary-mark-as-spam    "Mst" spam-generic-score    "Msx" gnus-summary-mark-as-spam -  "\M-d" gnus-summary-mark-as-spam) +  "\M-d" gnus-summary-mark-as-spam +  "$" gnus-summary-mark-as-spam)  (defvar spam-cache-lookups t    "Whether spam.el will try to cache lookups using `spam-caches'.") | 
