123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995 |
- ;;; gnus-util.el --- utility functions for Gnus
- ;; Copyright (C) 1996-2015 Free Software Foundation, Inc.
- ;; Author: Lars Magne Ingebrigtsen <larsi@gnus.org>
- ;; Keywords: news
- ;; This file is part of GNU Emacs.
- ;; GNU Emacs 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 3 of the License, or
- ;; (at your option) any later version.
- ;; GNU Emacs 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. If not, see <http://www.gnu.org/licenses/>.
- ;;; Commentary:
- ;; Nothing in this file depends on any other parts of Gnus -- all
- ;; functions and macros in this file are utility functions that are
- ;; used by Gnus and may be used by any other package without loading
- ;; Gnus first.
- ;; [Unfortunately, it does depend on other parts of Gnus, e.g. the
- ;; autoloads and defvars below...]
- ;;; Code:
- (eval-when-compile
- (require 'cl))
- (require 'time-date)
- (defcustom gnus-completing-read-function 'gnus-emacs-completing-read
- "Function use to do completing read."
- :version "24.1"
- :group 'gnus-meta
- :type `(radio (function-item
- :doc "Use Emacs standard `completing-read' function."
- gnus-emacs-completing-read)
- ;; iswitchb.el is very old and ido.el is unavailable
- ;; in XEmacs, so we exclude those function items.
- ,@(unless (featurep 'xemacs)
- '((function-item
- :doc "Use `ido-completing-read' function."
- gnus-ido-completing-read)
- (function-item
- :doc "Use iswitchb based completing-read function."
- gnus-iswitchb-completing-read)))))
- (defcustom gnus-completion-styles
- (if (and (boundp 'completion-styles-alist)
- (boundp 'completion-styles))
- (append (when (and (assq 'substring completion-styles-alist)
- (not (memq 'substring completion-styles)))
- (list 'substring))
- completion-styles)
- nil)
- "Value of `completion-styles' to use when completing."
- :version "24.1"
- :group 'gnus-meta
- :type '(repeat symbol))
- ;; Fixme: this should be a gnus variable, not nnmail-.
- (defvar nnmail-pathname-coding-system)
- (defvar nnmail-active-file-coding-system)
- ;; Inappropriate references to other parts of Gnus.
- (defvar gnus-emphasize-whitespace-regexp)
- (defvar gnus-original-article-buffer)
- (defvar gnus-user-agent)
- (autoload 'gnus-get-buffer-window "gnus-win")
- (autoload 'nnheader-narrow-to-headers "nnheader")
- (autoload 'nnheader-replace-chars-in-string "nnheader")
- (autoload 'mail-header-remove-comments "mail-parse")
- (eval-and-compile
- (cond
- ;; Prefer `replace-regexp-in-string' (present in Emacs, XEmacs 21.5,
- ;; SXEmacs 22.1.4) over `replace-in-string'. The latter leads to inf-loops
- ;; on empty matches:
- ;; (replace-in-string "foo" "/*$" "/")
- ;; (replace-in-string "xe" "\\(x\\)?" "")
- ((fboundp 'replace-regexp-in-string)
- (defun gnus-replace-in-string (string regexp newtext &optional literal)
- "Replace all matches for REGEXP with NEWTEXT in STRING.
- If LITERAL is non-nil, insert NEWTEXT literally. Return a new
- string containing the replacements.
- This is a compatibility function for different Emacsen."
- (replace-regexp-in-string regexp newtext string nil literal)))
- ((fboundp 'replace-in-string)
- (defalias 'gnus-replace-in-string 'replace-in-string))))
- (defun gnus-boundp (variable)
- "Return non-nil if VARIABLE is bound and non-nil."
- (and (boundp variable)
- (symbol-value variable)))
- (defmacro gnus-eval-in-buffer-window (buffer &rest forms)
- "Pop to BUFFER, evaluate FORMS, and then return to the original window."
- (let ((tempvar (make-symbol "GnusStartBufferWindow"))
- (w (make-symbol "w"))
- (buf (make-symbol "buf")))
- `(let* ((,tempvar (selected-window))
- (,buf ,buffer)
- (,w (gnus-get-buffer-window ,buf 'visible)))
- (unwind-protect
- (progn
- (if ,w
- (progn
- (select-window ,w)
- (set-buffer (window-buffer ,w)))
- (pop-to-buffer ,buf))
- ,@forms)
- (select-window ,tempvar)))))
- (put 'gnus-eval-in-buffer-window 'lisp-indent-function 1)
- (put 'gnus-eval-in-buffer-window 'edebug-form-spec '(form body))
- (defmacro gnus-intern-safe (string hashtable)
- "Get hash value. Arguments are STRING and HASHTABLE."
- `(let ((symbol (intern ,string ,hashtable)))
- (or (boundp symbol)
- (set symbol nil))
- symbol))
- (defsubst gnus-goto-char (point)
- (and point (goto-char point)))
- (defmacro gnus-buffer-exists-p (buffer)
- `(let ((buffer ,buffer))
- (when buffer
- (funcall (if (stringp buffer) 'get-buffer 'buffer-name)
- buffer))))
- ;; The LOCAL arg to `add-hook' is interpreted differently in Emacs and
- ;; XEmacs. In Emacs we don't need to call `make-local-hook' first.
- ;; It's harmless, though, so the main purpose of this alias is to shut
- ;; up the byte compiler.
- (defalias 'gnus-make-local-hook (if (featurep 'xemacs)
- 'make-local-hook
- 'ignore))
- (defun gnus-delete-first (elt list)
- "Delete by side effect the first occurrence of ELT as a member of LIST."
- (if (equal (car list) elt)
- (cdr list)
- (let ((total list))
- (while (and (cdr list)
- (not (equal (cadr list) elt)))
- (setq list (cdr list)))
- (when (cdr list)
- (setcdr list (cddr list)))
- total)))
- ;; Delete the current line (and the next N lines).
- (defmacro gnus-delete-line (&optional n)
- `(delete-region (point-at-bol)
- (progn (forward-line ,(or n 1)) (point))))
- (defun gnus-extract-address-components (from)
- "Extract address components from a From header.
- Given an RFC-822 address FROM, extract full name and canonical address.
- Returns a list of the form (FULL-NAME CANONICAL-ADDRESS). Much more simple
- solution than `mail-extract-address-components', which works much better, but
- is slower."
- (let (name address)
- ;; First find the address - the thing with the @ in it. This may
- ;; not be accurate in mail addresses, but does the trick most of
- ;; the time in news messages.
- (cond (;; Check ``<foo@bar>'' first in order to handle the quite common
- ;; form ``"abc@xyz" <foo@bar>'' (i.e. ``@'' as part of a comment)
- ;; correctly.
- (string-match "<\\([^@ \t<>]+[!@][^@ \t<>]+\\)>" from)
- (setq address (substring from (match-beginning 1) (match-end 1))))
- ((string-match "\\b[^@ \t<>]+[!@][^@ \t<>]+\\b" from)
- (setq address (substring from (match-beginning 0) (match-end 0)))))
- ;; Then we check whether the "name <address>" format is used.
- (and address
- ;; Linear white space is not required.
- (string-match (concat "[ \t]*<" (regexp-quote address) ">") from)
- (and (setq name (substring from 0 (match-beginning 0)))
- ;; Strip any quotes from the name.
- (string-match "^\".*\"$" name)
- (setq name (substring name 1 (1- (match-end 0))))))
- ;; If not, then "address (name)" is used.
- (or name
- (and (string-match "(.+)" from)
- (setq name (substring from (1+ (match-beginning 0))
- (1- (match-end 0)))))
- (and (string-match "()" from)
- (setq name address))
- ;; XOVER might not support folded From headers.
- (and (string-match "(.*" from)
- (setq name (substring from (1+ (match-beginning 0))
- (match-end 0)))))
- (list (if (string= name "") nil name) (or address from))))
- (declare-function message-fetch-field "message" (header &optional not-all))
- (defun gnus-fetch-field (field)
- "Return the value of the header FIELD of current article."
- (require 'message)
- (save-excursion
- (save-restriction
- (let ((inhibit-point-motion-hooks t))
- (nnheader-narrow-to-headers)
- (message-fetch-field field)))))
- (defun gnus-fetch-original-field (field)
- "Fetch FIELD from the original version of the current article."
- (with-current-buffer gnus-original-article-buffer
- (gnus-fetch-field field)))
- (defun gnus-goto-colon ()
- (move-beginning-of-line 1)
- (let ((eol (point-at-eol)))
- (goto-char (or (text-property-any (point) eol 'gnus-position t)
- (search-forward ":" eol t)
- (point)))))
- (declare-function gnus-find-method-for-group "gnus" (group &optional info))
- (declare-function gnus-group-name-decode "gnus-group" (string charset))
- (declare-function gnus-group-name-charset "gnus-group" (method group))
- ;; gnus-group requires gnus-int which requires message.
- (declare-function message-tokenize-header "message"
- (header &optional separator))
- (defun gnus-decode-newsgroups (newsgroups group &optional method)
- (require 'gnus-group)
- (let ((method (or method (gnus-find-method-for-group group))))
- (mapconcat (lambda (group)
- (gnus-group-name-decode group (gnus-group-name-charset
- method group)))
- (message-tokenize-header newsgroups)
- ",")))
- (defun gnus-remove-text-with-property (prop)
- "Delete all text in the current buffer with text property PROP."
- (let ((start (point-min))
- end)
- (unless (get-text-property start prop)
- (setq start (next-single-property-change start prop)))
- (while start
- (setq end (text-property-any start (point-max) prop nil))
- (delete-region start (or end (point-max)))
- (setq start (when end
- (next-single-property-change start prop))))))
- (defun gnus-find-text-property-region (start end prop)
- "Return a list of text property regions that has property PROP."
- (let (regions value)
- (unless (get-text-property start prop)
- (setq start (next-single-property-change start prop)))
- (while start
- (setq value (get-text-property start prop)
- end (text-property-not-all start (point-max) prop value))
- (if (not end)
- (setq start nil)
- (when value
- (push (list (set-marker (make-marker) start)
- (set-marker (make-marker) end)
- value)
- regions))
- (setq start (next-single-property-change start prop))))
- (nreverse regions)))
- (defun gnus-newsgroup-directory-form (newsgroup)
- "Make hierarchical directory name from NEWSGROUP name."
- (let* ((newsgroup (gnus-newsgroup-savable-name newsgroup))
- (idx (string-match ":" newsgroup)))
- (concat
- (if idx (substring newsgroup 0 idx))
- (if idx "/")
- (nnheader-replace-chars-in-string
- (if idx (substring newsgroup (1+ idx)) newsgroup)
- ?. ?/))))
- (defun gnus-newsgroup-savable-name (group)
- ;; Replace any slashes in a group name (eg. an ange-ftp nndoc group)
- ;; with dots.
- (nnheader-replace-chars-in-string group ?/ ?.))
- (defun gnus-string> (s1 s2)
- (not (or (string< s1 s2)
- (string= s1 s2))))
- (defun gnus-string< (s1 s2)
- "Return t if first arg string is less than second in lexicographic order.
- Case is significant if and only if `case-fold-search' is nil.
- Symbols are also allowed; their print names are used instead."
- (if case-fold-search
- (string-lessp (downcase (if (symbolp s1) (symbol-name s1) s1))
- (downcase (if (symbolp s2) (symbol-name s2) s2)))
- (string-lessp s1 s2)))
- ;;; Time functions.
- (defun gnus-file-newer-than (file date)
- (let ((fdate (nth 5 (file-attributes file))))
- (or (> (car fdate) (car date))
- (and (= (car fdate) (car date))
- (> (nth 1 fdate) (nth 1 date))))))
- ;; Every version of Emacs Gnus supports has built-in float-time.
- ;; The featurep test silences an irritating compiler warning.
- (defalias 'gnus-float-time
- (if (or (featurep 'emacs)
- (fboundp 'float-time))
- 'float-time 'time-to-seconds))
- ;;; Keymap macros.
- (defmacro gnus-local-set-keys (&rest plist)
- "Set the keys in PLIST in the current keymap."
- `(gnus-define-keys-1 (current-local-map) ',plist))
- (defmacro gnus-define-keys (keymap &rest plist)
- "Define all keys in PLIST in KEYMAP."
- ;; Convert the key [?\S-\ ] to [(shift space)] for XEmacs.
- (when (featurep 'xemacs)
- (let ((bindings plist))
- (while bindings
- (when (equal (car bindings) [?\S-\ ])
- (setcar bindings [(shift space)]))
- (setq bindings (cddr bindings)))))
- `(gnus-define-keys-1 (quote ,keymap) (quote ,plist)))
- (defmacro gnus-define-keys-safe (keymap &rest plist)
- "Define all keys in PLIST in KEYMAP without overwriting previous definitions."
- `(gnus-define-keys-1 (quote ,keymap) (quote ,plist) t))
- (put 'gnus-define-keys 'lisp-indent-function 1)
- (put 'gnus-define-keys-safe 'lisp-indent-function 1)
- (put 'gnus-local-set-keys 'lisp-indent-function 1)
- (defmacro gnus-define-keymap (keymap &rest plist)
- "Define all keys in PLIST in KEYMAP."
- `(gnus-define-keys-1 ,keymap (quote ,plist)))
- (put 'gnus-define-keymap 'lisp-indent-function 1)
- (defun gnus-define-keys-1 (keymap plist &optional safe)
- (when (null keymap)
- (error "Can't set keys in a null keymap"))
- (cond ((symbolp keymap)
- (setq keymap (symbol-value keymap)))
- ((keymapp keymap))
- ((listp keymap)
- (set (car keymap) nil)
- (define-prefix-command (car keymap))
- (define-key (symbol-value (caddr keymap)) (cadr keymap) (car keymap))
- (setq keymap (symbol-value (car keymap)))))
- (let (key)
- (while plist
- (when (symbolp (setq key (pop plist)))
- (setq key (symbol-value key)))
- (if (or (not safe)
- (eq (lookup-key keymap key) 'undefined))
- (define-key keymap key (pop plist))
- (pop plist)))))
- (defun gnus-y-or-n-p (prompt)
- (prog1
- (y-or-n-p prompt)
- (message "")))
- (defun gnus-yes-or-no-p (prompt)
- (prog1
- (yes-or-no-p prompt)
- (message "")))
- ;; By Frank Schmitt <ich@Frank-Schmitt.net>. Allows to have
- ;; age-depending date representations. (e.g. just the time if it's
- ;; from today, the day of the week if it's within the last 7 days and
- ;; the full date if it's older)
- (defun gnus-seconds-today ()
- "Return the number of seconds passed today."
- (let ((now (decode-time)))
- (+ (car now) (* (car (cdr now)) 60) (* (car (nthcdr 2 now)) 3600))))
- (defun gnus-seconds-month ()
- "Return the number of seconds passed this month."
- (let ((now (decode-time)))
- (+ (car now) (* (car (cdr now)) 60) (* (car (nthcdr 2 now)) 3600)
- (* (- (car (nthcdr 3 now)) 1) 3600 24))))
- (defun gnus-seconds-year ()
- "Return the number of seconds passed this year."
- (let* ((current (current-time))
- (now (decode-time current))
- (days (format-time-string "%j" current)))
- (+ (car now) (* (car (cdr now)) 60) (* (car (nthcdr 2 now)) 3600)
- (* (- (string-to-number days) 1) 3600 24))))
- (defmacro gnus-date-get-time (date)
- "Convert DATE string to Emacs time.
- Cache the result as a text property stored in DATE."
- ;; Either return the cached value...
- `(let ((d ,date))
- (if (equal "" d)
- '(0 0)
- (or (get-text-property 0 'gnus-time d)
- ;; or compute the value...
- (let ((time (safe-date-to-time d)))
- ;; and store it back in the string.
- (put-text-property 0 1 'gnus-time time d)
- time)))))
- (defun gnus-dd-mmm (messy-date)
- "Return a string like DD-MMM from a big messy string."
- (condition-case ()
- (format-time-string "%d-%b" (gnus-date-get-time messy-date))
- (error " - ")))
- (defsubst gnus-time-iso8601 (time)
- "Return a string of TIME in YYYYMMDDTHHMMSS format."
- (format-time-string "%Y%m%dT%H%M%S" time))
- (defun gnus-date-iso8601 (date)
- "Convert the DATE to YYYYMMDDTHHMMSS."
- (condition-case ()
- (gnus-time-iso8601 (gnus-date-get-time date))
- (error "")))
- (defun gnus-mode-string-quote (string)
- "Quote all \"%\"'s in STRING."
- (gnus-replace-in-string string "%" "%%"))
- ;; Make a hash table (default and minimum size is 256).
- ;; Optional argument HASHSIZE specifies the table size.
- (defun gnus-make-hashtable (&optional hashsize)
- (make-vector (if hashsize (max (gnus-create-hash-size hashsize) 256) 256) 0))
- ;; Make a number that is suitable for hashing; bigger than MIN and
- ;; equal to some 2^x. Many machines (such as sparcs) do not have a
- ;; hardware modulo operation, so they implement it in software. On
- ;; many sparcs over 50% of the time to intern is spent in the modulo.
- ;; Yes, it's slower than actually computing the hash from the string!
- ;; So we use powers of 2 so people can optimize the modulo to a mask.
- (defun gnus-create-hash-size (min)
- (let ((i 1))
- (while (< i min)
- (setq i (* 2 i)))
- i))
- (defcustom gnus-verbose 6
- "*Integer that says how verbose Gnus should be.
- The higher the number, the more messages Gnus will flash to say what
- it's doing. At zero, Gnus will be totally mute; at five, Gnus will
- display most important messages; and at ten, Gnus will keep on
- jabbering all the time."
- :version "24.1"
- :group 'gnus-start
- :type 'integer)
- (defcustom gnus-add-timestamp-to-message nil
- "Non-nil means add timestamps to messages that Gnus issues.
- If it is `log', add timestamps to only the messages that go into the
- \"*Messages*\" buffer (in XEmacs, it is the \" *Message-Log*\" buffer).
- If it is neither nil nor `log', add timestamps not only to log messages
- but also to the ones displayed in the echo area."
- :version "23.1" ;; No Gnus
- :group 'gnus-various
- :type '(choice :format "%{%t%}:\n %[Value Menu%] %v"
- (const :tag "Logged messages only" log)
- (sexp :tag "All messages"
- :match (lambda (widget value) value)
- :value t)
- (const :tag "No timestamp" nil)))
- (eval-when-compile
- (defmacro gnus-message-with-timestamp-1 (format-string args)
- (let ((timestamp '(format-time-string "%Y%m%dT%H%M%S.%3N> " time)))
- (if (featurep 'xemacs)
- `(let (str time)
- (if (or (and (null ,format-string) (null ,args))
- (progn
- (setq str (apply 'format ,format-string ,args))
- (zerop (length str))))
- (prog1
- (and ,format-string str)
- (clear-message nil))
- (cond ((eq gnus-add-timestamp-to-message 'log)
- (setq time (current-time))
- (display-message 'no-log str)
- (log-message 'message (concat ,timestamp str)))
- (gnus-add-timestamp-to-message
- (setq time (current-time))
- (display-message 'message (concat ,timestamp str)))
- (t
- (display-message 'message str))))
- str)
- `(let (str time)
- (cond ((eq gnus-add-timestamp-to-message 'log)
- (setq str (let (message-log-max)
- (apply 'message ,format-string ,args)))
- (when (and message-log-max
- (> message-log-max 0)
- (/= (length str) 0))
- (setq time (current-time))
- (with-current-buffer (if (fboundp 'messages-buffer)
- (messages-buffer)
- (get-buffer-create "*Messages*"))
- (goto-char (point-max))
- (let ((inhibit-read-only t))
- (insert ,timestamp str "\n")
- (forward-line (- message-log-max))
- (delete-region (point-min) (point)))
- (goto-char (point-max))))
- str)
- (gnus-add-timestamp-to-message
- (if (or (and (null ,format-string) (null ,args))
- (progn
- (setq str (apply 'format ,format-string ,args))
- (zerop (length str))))
- (prog1
- (and ,format-string str)
- (message nil))
- (setq time (current-time))
- (message "%s" (concat ,timestamp str))
- str))
- (t
- (apply 'message ,format-string ,args))))))))
- (defvar gnus-action-message-log nil)
- (defun gnus-message-with-timestamp (format-string &rest args)
- "Display message with timestamp. Arguments are the same as `message'.
- The `gnus-add-timestamp-to-message' variable controls how to add
- timestamp to message."
- (gnus-message-with-timestamp-1 format-string args))
- (defun gnus-message (level &rest args)
- "If LEVEL is lower than `gnus-verbose' print ARGS using `message'.
- Guideline for numbers:
- 1 - error messages, 3 - non-serious error messages, 5 - messages for things
- that take a long time, 7 - not very important messages on stuff, 9 - messages
- inside loops."
- (if (<= level gnus-verbose)
- (let ((message
- (if gnus-add-timestamp-to-message
- (apply 'gnus-message-with-timestamp args)
- (apply 'message args))))
- (when (and (consp gnus-action-message-log)
- (<= level 3))
- (push message gnus-action-message-log))
- message)
- ;; We have to do this format thingy here even if the result isn't
- ;; shown - the return value has to be the same as the return value
- ;; from `message'.
- (apply 'format args)))
- (defun gnus-final-warning ()
- (when (and (consp gnus-action-message-log)
- (setq gnus-action-message-log
- (delete nil gnus-action-message-log)))
- (message "Warning: %s"
- (mapconcat #'identity gnus-action-message-log "; "))))
- (defun gnus-error (level &rest args)
- "Beep an error if LEVEL is equal to or less than `gnus-verbose'.
- ARGS are passed to `message'."
- (when (<= (floor level) gnus-verbose)
- (apply 'message args)
- (ding)
- (let (duration)
- (when (and (floatp level)
- (not (zerop (setq duration (* 10 (- level (floor level)))))))
- (sit-for duration))))
- nil)
- (defun gnus-split-references (references)
- "Return a list of Message-IDs in REFERENCES."
- (let ((beg 0)
- (references (mail-header-remove-comments (or references "")))
- ids)
- (while (string-match "<[^<]+[^< \t]" references beg)
- (push (substring references (match-beginning 0) (setq beg (match-end 0)))
- ids))
- (nreverse ids)))
- (defun gnus-extract-references (references)
- "Return a list of Message-IDs in REFERENCES (in In-Reply-To
- format), trimmed to only contain the Message-IDs."
- (let ((ids (gnus-split-references references))
- refs)
- (dolist (id ids)
- (when (string-match "<[^<>]+>" id)
- (push (match-string 0 id) refs)))
- refs))
- (defsubst gnus-parent-id (references &optional n)
- "Return the last Message-ID in REFERENCES.
- If N, return the Nth ancestor instead."
- (when (and references
- (not (zerop (length references))))
- (if n
- (let ((ids (inline (gnus-split-references references))))
- (while (nthcdr n ids)
- (setq ids (cdr ids)))
- (car ids))
- (let ((references (mail-header-remove-comments references)))
- (when (string-match "\\(<[^<]+>\\)[ \t]*\\'" references)
- (match-string 1 references))))))
- (defsubst gnus-buffer-live-p (buffer)
- "Say whether BUFFER is alive or not."
- (and buffer (buffer-live-p (get-buffer buffer))))
- (defun gnus-horizontal-recenter ()
- "Recenter the current buffer horizontally."
- (if (< (current-column) (/ (window-width) 2))
- (set-window-hscroll (gnus-get-buffer-window (current-buffer) t) 0)
- (let* ((orig (point))
- (end (window-end (gnus-get-buffer-window (current-buffer) t)))
- (max 0))
- (when end
- ;; Find the longest line currently displayed in the window.
- (goto-char (window-start))
- (while (and (not (eobp))
- (< (point) end))
- (end-of-line)
- (setq max (max max (current-column)))
- (forward-line 1))
- (goto-char orig)
- ;; Scroll horizontally to center (sort of) the point.
- (if (> max (window-width))
- (set-window-hscroll
- (gnus-get-buffer-window (current-buffer) t)
- (min (- (current-column) (/ (window-width) 3))
- (+ 2 (- max (window-width)))))
- (set-window-hscroll (gnus-get-buffer-window (current-buffer) t) 0))
- max))))
- (defun gnus-read-event-char (&optional prompt)
- "Get the next event."
- (let ((event (read-event prompt)))
- ;; should be gnus-characterp, but this can't be called in XEmacs anyway
- (cons (and (numberp event) event) event)))
- (defun gnus-copy-file (file &optional to)
- "Copy FILE to TO."
- (interactive
- (list (read-file-name "Copy file: " default-directory)
- (read-file-name "Copy file to: " default-directory)))
- (unless to
- (setq to (read-file-name "Copy file to: " default-directory)))
- (when (file-directory-p to)
- (setq to (concat (file-name-as-directory to)
- (file-name-nondirectory file))))
- (copy-file file to))
- (defvar gnus-work-buffer " *gnus work*")
- (declare-function gnus-get-buffer-create "gnus" (name))
- ;; gnus.el requires mm-util.
- (declare-function mm-enable-multibyte "mm-util")
- (defun gnus-set-work-buffer ()
- "Put point in the empty Gnus work buffer."
- (if (get-buffer gnus-work-buffer)
- (progn
- (set-buffer gnus-work-buffer)
- (erase-buffer))
- (set-buffer (gnus-get-buffer-create gnus-work-buffer))
- (kill-all-local-variables)
- (mm-enable-multibyte)))
- (defmacro gnus-group-real-name (group)
- "Find the real name of a foreign newsgroup."
- `(let ((gname ,group))
- (if (string-match "^[^:]+:" gname)
- (substring gname (match-end 0))
- gname)))
- (defmacro gnus-group-server (group)
- "Find the server name of a foreign newsgroup.
- For example, (gnus-group-server \"nnimap+yxa:INBOX.foo\") would
- yield \"nnimap:yxa\"."
- `(let ((gname ,group))
- (if (string-match "^\\([^:+]+\\)\\(?:\\+\\([^:]*\\)\\)?:" gname)
- (format "%s:%s" (match-string 1 gname) (or
- (match-string 2 gname)
- ""))
- (format "%s:%s" (car gnus-select-method) (cadr gnus-select-method)))))
- (defun gnus-make-sort-function (funs)
- "Return a composite sort condition based on the functions in FUNS."
- (cond
- ;; Just a simple function.
- ((functionp funs) funs)
- ;; No functions at all.
- ((null funs) funs)
- ;; A list of functions.
- ((or (cdr funs)
- (listp (car funs)))
- (gnus-byte-compile
- `(lambda (t1 t2)
- ,(gnus-make-sort-function-1 (reverse funs)))))
- ;; A list containing just one function.
- (t
- (car funs))))
- (defun gnus-make-sort-function-1 (funs)
- "Return a composite sort condition based on the functions in FUNS."
- (let ((function (car funs))
- (first 't1)
- (last 't2))
- (when (consp function)
- (cond
- ;; Reversed spec.
- ((eq (car function) 'not)
- (setq function (cadr function)
- first 't2
- last 't1))
- ((functionp function)
- ;; Do nothing.
- )
- (t
- (error "Invalid sort spec: %s" function))))
- (if (cdr funs)
- `(or (,function ,first ,last)
- (and (not (,function ,last ,first))
- ,(gnus-make-sort-function-1 (cdr funs))))
- `(,function ,first ,last))))
- (defun gnus-turn-off-edit-menu (type)
- "Turn off edit menu in `gnus-TYPE-mode-map'."
- (define-key (symbol-value (intern (format "gnus-%s-mode-map" type)))
- [menu-bar edit] 'undefined))
- (defmacro gnus-bind-print-variables (&rest forms)
- "Bind print-* variables and evaluate FORMS.
- This macro is used with `prin1', `pp', etc. in order to ensure printed
- Lisp objects are loadable. Bind `print-quoted' and `print-readably'
- to t, and `print-escape-multibyte', `print-escape-newlines',
- `print-escape-nonascii', `print-length', `print-level' and
- `print-string-length' to nil."
- `(let ((print-quoted t)
- (print-readably t)
- ;;print-circle
- ;;print-continuous-numbering
- print-escape-multibyte
- print-escape-newlines
- print-escape-nonascii
- ;;print-gensym
- print-length
- print-level
- print-string-length)
- ,@forms))
- (defun gnus-prin1 (form)
- "Use `prin1' on FORM in the current buffer.
- Bind `print-quoted' and `print-readably' to t, and `print-length' and
- `print-level' to nil. See also `gnus-bind-print-variables'."
- (gnus-bind-print-variables (prin1 form (current-buffer))))
- (defun gnus-prin1-to-string (form)
- "The same as `prin1'.
- Bind `print-quoted' and `print-readably' to t, and `print-length' and
- `print-level' to nil. See also `gnus-bind-print-variables'."
- (gnus-bind-print-variables (prin1-to-string form)))
- (defun gnus-pp (form &optional stream)
- "Use `pp' on FORM in the current buffer.
- Bind `print-quoted' and `print-readably' to t, and `print-length' and
- `print-level' to nil. See also `gnus-bind-print-variables'."
- (gnus-bind-print-variables (pp form (or stream (current-buffer)))))
- (defun gnus-pp-to-string (form)
- "The same as `pp-to-string'.
- Bind `print-quoted' and `print-readably' to t, and `print-length' and
- `print-level' to nil. See also `gnus-bind-print-variables'."
- (gnus-bind-print-variables (pp-to-string form)))
- (defun gnus-make-directory (directory)
- "Make DIRECTORY (and all its parents) if it doesn't exist."
- (require 'nnmail)
- (let ((file-name-coding-system nnmail-pathname-coding-system))
- (when (and directory
- (not (file-exists-p directory)))
- (make-directory directory t)))
- t)
- (defun gnus-write-buffer (file)
- "Write the current buffer's contents to FILE."
- (require 'nnmail)
- (let ((file-name-coding-system nnmail-pathname-coding-system))
- ;; Make sure the directory exists.
- (gnus-make-directory (file-name-directory file))
- ;; Write the buffer.
- (write-region (point-min) (point-max) file nil 'quietly)))
- (defun gnus-delete-file (file)
- "Delete FILE if it exists."
- (when (file-exists-p file)
- (delete-file file)))
- (defun gnus-delete-duplicates (list)
- "Remove duplicate entries from LIST."
- (let ((result nil))
- (while list
- (unless (member (car list) result)
- (push (car list) result))
- (pop list))
- (nreverse result)))
- (defun gnus-delete-directory (directory)
- "Delete files in DIRECTORY. Subdirectories remain.
- If there's no subdirectory, delete DIRECTORY as well."
- (when (file-directory-p directory)
- (let ((files (directory-files
- directory t "^\\([^.]\\|\\.\\([^.]\\|\\..\\)\\).*"))
- file dir)
- (while files
- (setq file (pop files))
- (if (eq t (car (file-attributes file)))
- ;; `file' is a subdirectory.
- (setq dir t)
- ;; `file' is a file or a symlink.
- (delete-file file)))
- (unless dir
- (delete-directory directory)))))
- (defun gnus-strip-whitespace (string)
- "Return STRING stripped of all whitespace."
- (while (string-match "[\r\n\t ]+" string)
- (setq string (replace-match "" t t string)))
- string)
- (declare-function gnus-put-text-property "gnus"
- (start end property value &optional object))
- (defsubst gnus-put-text-property-excluding-newlines (beg end prop val)
- "The same as `put-text-property', but don't put this prop on any newlines in the region."
- (save-match-data
- (save-excursion
- (save-restriction
- (goto-char beg)
- (while (re-search-forward gnus-emphasize-whitespace-regexp end 'move)
- (gnus-put-text-property beg (match-beginning 0) prop val)
- (setq beg (point)))
- (gnus-put-text-property beg (point) prop val)))))
- (defsubst gnus-put-overlay-excluding-newlines (beg end prop val)
- "The same as `put-text-property', but don't put this prop on any newlines in the region."
- (save-match-data
- (save-excursion
- (save-restriction
- (goto-char beg)
- (while (re-search-forward gnus-emphasize-whitespace-regexp end 'move)
- (overlay-put (make-overlay beg (match-beginning 0)) prop val)
- (setq beg (point)))
- (overlay-put (make-overlay beg (point)) prop val)))))
- (defun gnus-put-text-property-excluding-characters-with-faces (beg end prop val)
- "The same as `put-text-property', except where `gnus-face' is set.
- If so, and PROP is `face', set the second element of its value to VAL.
- Otherwise, do nothing."
- (while (< beg end)
- ;; Property values are compared with `eq'.
- (let ((stop (next-single-property-change beg 'face nil end)))
- (if (get-text-property beg 'gnus-face)
- (when (eq prop 'face)
- (setcar (cdr (get-text-property beg 'face)) (or val 'default)))
- (inline
- (gnus-put-text-property beg stop prop val)))
- (setq beg stop))))
- (defun gnus-get-text-property-excluding-characters-with-faces (pos prop)
- "The same as `get-text-property', except where `gnus-face' is set.
- If so, and PROP is `face', return the second element of its value.
- Otherwise, return the value."
- (let ((val (get-text-property pos prop)))
- (if (and (get-text-property pos 'gnus-face)
- (eq prop 'face))
- (cadr val)
- (get-text-property pos prop))))
- (defmacro gnus-faces-at (position)
- "Return a list of faces at POSITION."
- (if (featurep 'xemacs)
- `(let ((pos ,position))
- (mapcar-extents 'extent-face
- nil (current-buffer) pos pos nil 'face))
- `(let ((pos ,position))
- (delq nil (cons (get-text-property pos 'face)
- (mapcar
- (lambda (overlay)
- (overlay-get overlay 'face))
- (overlays-at pos)))))))
- (if (fboundp 'invisible-p)
- (defalias 'gnus-invisible-p 'invisible-p)
- ;; for Emacs < 22.2, and XEmacs.
- (defun gnus-invisible-p (pos)
- "Return non-nil if the character after POS is currently invisible."
- (let ((prop (get-char-property pos 'invisible)))
- (if (eq buffer-invisibility-spec t)
- prop
- (or (memq prop buffer-invisibility-spec)
- (assq prop buffer-invisibility-spec))))))
- ;; Note: the optional 2nd argument has a different meaning between
- ;; Emacs and XEmacs.
- ;; (next-char-property-change POSITION &optional LIMIT)
- ;; (next-extent-change POS &optional OBJECT)
- (defalias 'gnus-next-char-property-change
- (if (fboundp 'next-extent-change)
- 'next-extent-change 'next-char-property-change))
- (defalias 'gnus-previous-char-property-change
- (if (fboundp 'previous-extent-change)
- 'previous-extent-change 'previous-char-property-change))
- ;;; Protected and atomic operations. dmoore@ucsd.edu 21.11.1996
- ;; The primary idea here is to try to protect internal data structures
- ;; from becoming corrupted when the user hits C-g, or if a hook or
- ;; similar blows up. Often in Gnus multiple tables/lists need to be
- ;; updated at the same time, or information can be lost.
- (defvar gnus-atomic-be-safe t
- "If t, certain operations will be protected from interruption by C-g.")
- (defmacro gnus-atomic-progn (&rest forms)
- "Evaluate FORMS atomically, which means to protect the evaluation
- from being interrupted by the user. An error from the forms themselves
- will return without finishing the operation. Since interrupts from
- the user are disabled, it is recommended that only the most minimal
- operations are performed by FORMS. If you wish to assign many
- complicated values atomically, compute the results into temporary
- variables and then do only the assignment atomically."
- `(let ((inhibit-quit gnus-atomic-be-safe))
- ,@forms))
- (put 'gnus-atomic-progn 'lisp-indent-function 0)
- (defmacro gnus-atomic-progn-assign (protect &rest forms)
- "Evaluate FORMS, but ensure that the variables listed in PROTECT
- are not changed if anything in FORMS signals an error or otherwise
- non-locally exits. The variables listed in PROTECT are updated atomically.
- It is safe to use gnus-atomic-progn-assign with long computations.
- Note that if any of the symbols in PROTECT were unbound, they will be
- set to nil on a successful assignment. In case of an error or other
- non-local exit, it will still be unbound."
- (let* ((temp-sym-map (mapcar (lambda (x) (list (make-symbol
- (concat (symbol-name x)
- "-tmp"))
- x))
- protect))
- (sym-temp-map (mapcar (lambda (x) (list (cadr x) (car x)))
- temp-sym-map))
- (temp-sym-let (mapcar (lambda (x) (list (car x)
- `(and (boundp ',(cadr x))
- ,(cadr x))))
- temp-sym-map))
- (sym-temp-let sym-temp-map)
- (temp-sym-assign (apply 'append temp-sym-map))
- (sym-temp-assign (apply 'append sym-temp-map))
- (result (make-symbol "result-tmp")))
- `(let (,@temp-sym-let
- ,result)
- (let ,sym-temp-let
- (setq ,result (progn ,@forms))
- (setq ,@temp-sym-assign))
- (let ((inhibit-quit gnus-atomic-be-safe))
- (setq ,@sym-temp-assign))
- ,result)))
- (put 'gnus-atomic-progn-assign 'lisp-indent-function 1)
- ;(put 'gnus-atomic-progn-assign 'edebug-form-spec '(sexp body))
- (defmacro gnus-atomic-setq (&rest pairs)
- "Similar to setq, except that the real symbols are only assigned when
- there are no errors. And when the real symbols are assigned, they are
- done so atomically. If other variables might be changed via side-effect,
- see gnus-atomic-progn-assign. It is safe to use gnus-atomic-setq
- with potentially long computations."
- (let ((tpairs pairs)
- syms)
- (while tpairs
- (push (car tpairs) syms)
- (setq tpairs (cddr tpairs)))
- `(gnus-atomic-progn-assign ,syms
- (setq ,@pairs))))
- ;(put 'gnus-atomic-setq 'edebug-form-spec '(body))
- ;;; Functions for saving to babyl/mail files.
- (eval-when-compile
- (if (featurep 'xemacs)
- ;; Don't load tm and apel XEmacs packages that provide some
- ;; Emacs emulating functions and variables.
- (let ((features features))
- (provide 'tm-view)
- (unless (fboundp 'set-alist) (defalias 'set-alist 'ignore))
- (require 'rmail)) ;; It requires tm-view that loads apel.
- (require 'rmail))
- (autoload 'rmail-update-summary "rmailsum"))
- (defvar mm-text-coding-system)
- (declare-function mm-append-to-file "mm-util"
- (start end filename &optional codesys inhibit))
- (declare-function rmail-swap-buffers-maybe "rmail" ())
- (declare-function rmail-maybe-set-message-counters "rmail" ())
- (declare-function rmail-count-new-messages "rmail" (&optional nomsg))
- (declare-function rmail-summary-exists "rmail" ())
- (declare-function rmail-show-message "rmail" (&optional n no-summary))
- ;; Macroexpansion of rmail-select-summary:
- (declare-function rmail-summary-displayed "rmail" ())
- (declare-function rmail-pop-to-buffer "rmail" (&rest args))
- (declare-function rmail-maybe-display-summary "rmail" ())
- (defun gnus-output-to-rmail (filename &optional ask)
- "Append the current article to an Rmail file named FILENAME.
- In Emacs 22 this writes Babyl format; in Emacs 23 it writes mbox unless
- FILENAME exists and is Babyl format."
- (require 'rmail)
- (require 'mm-util)
- (require 'nnmail)
- ;; Some of this codes is borrowed from rmailout.el.
- (setq filename (expand-file-name filename))
- ;; FIXME should we really be messing with this defcustom?
- ;; It is not needed for the operation of this function.
- (if (boundp 'rmail-default-rmail-file)
- (setq rmail-default-rmail-file filename) ; 22
- (setq rmail-default-file filename)) ; 23
- (let ((artbuf (current-buffer))
- (tmpbuf (get-buffer-create " *Gnus-output*"))
- ;; Babyl rmail.el defines this, mbox does not.
- (babyl (fboundp 'rmail-insert-rmail-file-header)))
- (save-excursion
- ;; Note that we ignore the possibility of visiting a Babyl
- ;; format buffer in Emacs 23, since Rmail no longer supports that.
- (or (get-file-buffer filename)
- (progn
- ;; In case someone wants to write to a Babyl file from Emacs 23.
- (when (file-exists-p filename)
- (setq babyl (mail-file-babyl-p filename))
- t))
- (if (or (not ask)
- (gnus-yes-or-no-p
- (concat "\"" filename "\" does not exist, create it? ")))
- (let ((file-buffer (create-file-buffer filename)))
- (with-current-buffer file-buffer
- (if (fboundp 'rmail-insert-rmail-file-header)
- (rmail-insert-rmail-file-header))
- (let ((require-final-newline nil)
- (coding-system-for-write mm-text-coding-system))
- (gnus-write-buffer filename)))
- (kill-buffer file-buffer))
- (error "Output file does not exist")))
- (set-buffer tmpbuf)
- (erase-buffer)
- (insert-buffer-substring artbuf)
- (if babyl
- (gnus-convert-article-to-rmail)
- ;; Non-Babyl case copied from gnus-output-to-mail.
- (goto-char (point-min))
- (if (looking-at "From ")
- (forward-line 1)
- (insert "From nobody " (current-time-string) "\n"))
- (let (case-fold-search)
- (while (re-search-forward "^From " nil t)
- (beginning-of-line)
- (insert ">"))))
- ;; Decide whether to append to a file or to an Emacs buffer.
- (let ((outbuf (get-file-buffer filename)))
- (if (not outbuf)
- (progn
- (unless babyl ; from gnus-output-to-mail
- (let ((buffer-read-only nil))
- (goto-char (point-max))
- (forward-char -2)
- (unless (looking-at "\n\n")
- (goto-char (point-max))
- (unless (bolp)
- (insert "\n"))
- (insert "\n"))))
- (let ((file-name-coding-system nnmail-pathname-coding-system))
- (mm-append-to-file (point-min) (point-max) filename)))
- ;; File has been visited, in buffer OUTBUF.
- (set-buffer outbuf)
- (let ((buffer-read-only nil)
- (msg (and (boundp 'rmail-current-message)
- (symbol-value 'rmail-current-message))))
- ;; If MSG is non-nil, buffer is in RMAIL mode.
- ;; Compare this with rmail-output-to-rmail-buffer in Emacs 23.
- (when msg
- (unless babyl
- (rmail-swap-buffers-maybe)
- (rmail-maybe-set-message-counters))
- (widen)
- (narrow-to-region (point-max) (point-max)))
- (insert-buffer-substring tmpbuf)
- (when msg
- (when babyl
- (goto-char (point-min))
- (widen)
- (search-backward "\n\^_")
- (narrow-to-region (point) (point-max)))
- (rmail-count-new-messages t)
- (when (rmail-summary-exists)
- (rmail-select-summary
- (rmail-update-summary)))
- (rmail-show-message msg))
- (save-buffer)))))
- (kill-buffer tmpbuf)))
- (defun gnus-output-to-mail (filename &optional ask)
- "Append the current article to a mail file named FILENAME."
- (require 'nnmail)
- (setq filename (expand-file-name filename))
- (let ((artbuf (current-buffer))
- (tmpbuf (get-buffer-create " *Gnus-output*")))
- (save-excursion
- ;; Create the file, if it doesn't exist.
- (when (and (not (get-file-buffer filename))
- (not (file-exists-p filename)))
- (if (or (not ask)
- (gnus-y-or-n-p
- (concat "\"" filename "\" does not exist, create it? ")))
- (let ((file-buffer (create-file-buffer filename)))
- (with-current-buffer file-buffer
- (let ((require-final-newline nil)
- (coding-system-for-write mm-text-coding-system))
- (gnus-write-buffer filename)))
- (kill-buffer file-buffer))
- (error "Output file does not exist")))
- (set-buffer tmpbuf)
- (erase-buffer)
- (insert-buffer-substring artbuf)
- (goto-char (point-min))
- (if (looking-at "From ")
- (forward-line 1)
- (insert "From nobody " (current-time-string) "\n"))
- (let (case-fold-search)
- (while (re-search-forward "^From " nil t)
- (beginning-of-line)
- (insert ">")))
- ;; Decide whether to append to a file or to an Emacs buffer.
- (let ((outbuf (get-file-buffer filename)))
- (if (not outbuf)
- (let ((buffer-read-only nil))
- (save-excursion
- (goto-char (point-max))
- (forward-char -2)
- (unless (looking-at "\n\n")
- (goto-char (point-max))
- (unless (bolp)
- (insert "\n"))
- (insert "\n"))
- (goto-char (point-max))
- (let ((file-name-coding-system nnmail-pathname-coding-system))
- (mm-append-to-file (point-min) (point-max) filename))))
- ;; File has been visited, in buffer OUTBUF.
- (set-buffer outbuf)
- (let ((buffer-read-only nil))
- (goto-char (point-max))
- (unless (eobp)
- (insert "\n"))
- (insert "\n")
- (insert-buffer-substring tmpbuf)))))
- (kill-buffer tmpbuf)))
- (defun gnus-convert-article-to-rmail ()
- "Convert article in current buffer to Rmail message format."
- (let ((buffer-read-only nil))
- ;; Convert article directly into Babyl format.
- (goto-char (point-min))
- (insert "\^L\n0, unseen,,\n*** EOOH ***\n")
- (while (search-forward "\n\^_" nil t) ;single char
- (replace-match "\n^_" t t)) ;2 chars: "^" and "_"
- (goto-char (point-max))
- (insert "\^_")))
- (defun gnus-map-function (funs arg)
- "Apply the result of the first function in FUNS to the second, and so on.
- ARG is passed to the first function."
- (while funs
- (setq arg (funcall (pop funs) arg)))
- arg)
- (defun gnus-run-hooks (&rest funcs)
- "Does the same as `run-hooks', but saves the current buffer."
- (save-current-buffer
- (apply 'run-hooks funcs)))
- (defun gnus-run-hook-with-args (hook &rest args)
- "Does the same as `run-hook-with-args', but saves the current buffer."
- (save-current-buffer
- (apply 'run-hook-with-args hook args)))
- (defun gnus-run-mode-hooks (&rest funcs)
- "Run `run-mode-hooks' if it is available, otherwise `run-hooks'.
- This function saves the current buffer."
- (if (fboundp 'run-mode-hooks)
- (save-current-buffer (apply 'run-mode-hooks funcs))
- (save-current-buffer (apply 'run-hooks funcs))))
- ;;; Various
- (defvar gnus-group-buffer) ; Compiler directive
- (defun gnus-alive-p ()
- "Say whether Gnus is running or not."
- (and (boundp 'gnus-group-buffer)
- (get-buffer gnus-group-buffer)
- (with-current-buffer gnus-group-buffer
- (eq major-mode 'gnus-group-mode))))
- (defun gnus-remove-if (predicate sequence &optional hash-table-p)
- "Return a copy of SEQUENCE with all items satisfying PREDICATE removed.
- SEQUENCE should be a list, a vector, or a string. Returns always a list.
- If HASH-TABLE-P is non-nil, regards SEQUENCE as a hash table."
- (let (out)
- (if hash-table-p
- (mapatoms (lambda (symbol)
- (unless (funcall predicate symbol)
- (push symbol out)))
- sequence)
- (unless (listp sequence)
- (setq sequence (append sequence nil)))
- (while sequence
- (unless (funcall predicate (car sequence))
- (push (car sequence) out))
- (setq sequence (cdr sequence))))
- (nreverse out)))
- (defun gnus-remove-if-not (predicate sequence &optional hash-table-p)
- "Return a copy of SEQUENCE with all items not satisfying PREDICATE removed.
- SEQUENCE should be a list, a vector, or a string. Returns always a list.
- If HASH-TABLE-P is non-nil, regards SEQUENCE as a hash table."
- (let (out)
- (if hash-table-p
- (mapatoms (lambda (symbol)
- (when (funcall predicate symbol)
- (push symbol out)))
- sequence)
- (unless (listp sequence)
- (setq sequence (append sequence nil)))
- (while sequence
- (when (funcall predicate (car sequence))
- (push (car sequence) out))
- (setq sequence (cdr sequence))))
- (nreverse out)))
- (if (fboundp 'assq-delete-all)
- (defalias 'gnus-delete-alist 'assq-delete-all)
- (defun gnus-delete-alist (key alist)
- "Delete from ALIST all elements whose car is KEY.
- Return the modified alist."
- (let (entry)
- (while (setq entry (assq key alist))
- (setq alist (delq entry alist)))
- alist)))
- (defun gnus-grep-in-list (word list)
- "Find if a WORD matches any regular expression in the given LIST."
- (when (and word list)
- (catch 'found
- (dolist (r list)
- (when (string-match r word)
- (throw 'found r))))))
- (defmacro gnus-alist-pull (key alist &optional assoc-p)
- "Modify ALIST to be without KEY."
- (unless (symbolp alist)
- (error "Not a symbol: %s" alist))
- (let ((fun (if assoc-p 'assoc 'assq)))
- `(setq ,alist (delq (,fun ,key ,alist) ,alist))))
- (defun gnus-globalify-regexp (re)
- "Return a regexp that matches a whole line, if RE matches a part of it."
- (concat (unless (string-match "^\\^" re) "^.*")
- re
- (unless (string-match "\\$$" re) ".*$")))
- (defun gnus-set-window-start (&optional point)
- "Set the window start to POINT, or (point) if nil."
- (let ((win (gnus-get-buffer-window (current-buffer) t)))
- (when win
- (set-window-start win (or point (point))))))
- (defun gnus-annotation-in-region-p (b e)
- (if (= b e)
- (eq (cadr (memq 'gnus-undeletable (text-properties-at b))) t)
- (text-property-any b e 'gnus-undeletable t)))
- (defun gnus-or (&rest elems)
- "Return non-nil if any of the elements are non-nil."
- (catch 'found
- (while elems
- (when (pop elems)
- (throw 'found t)))))
- (defun gnus-and (&rest elems)
- "Return non-nil if all of the elements are non-nil."
- (catch 'found
- (while elems
- (unless (pop elems)
- (throw 'found nil)))
- t))
- ;; gnus.el requires mm-util.
- (declare-function mm-disable-multibyte "mm-util")
- (defun gnus-write-active-file (file hashtb &optional full-names)
- ;; `coding-system-for-write' should be `raw-text' or equivalent.
- (let ((coding-system-for-write nnmail-active-file-coding-system))
- (with-temp-file file
- ;; The buffer should be in the unibyte mode because group names
- ;; are ASCII text or encoded non-ASCII text (i.e., unibyte).
- (mm-disable-multibyte)
- (mapatoms
- (lambda (sym)
- (when (and sym
- (boundp sym)
- (symbol-value sym))
- (insert (format "%S %d %d y\n"
- (if full-names
- sym
- (intern (gnus-group-real-name (symbol-name sym))))
- (or (cdr (symbol-value sym))
- (car (symbol-value sym)))
- (car (symbol-value sym))))))
- hashtb)
- (goto-char (point-max))
- (while (search-backward "\\." nil t)
- (delete-char 1)))))
- ;; Fixme: Why not use `with-output-to-temp-buffer'?
- (defmacro gnus-with-output-to-file (file &rest body)
- (let ((buffer (make-symbol "output-buffer"))
- (size (make-symbol "output-buffer-size"))
- (leng (make-symbol "output-buffer-length"))
- (append (make-symbol "output-buffer-append")))
- `(let* ((,size 131072)
- (,buffer (make-string ,size 0))
- (,leng 0)
- (,append nil)
- (standard-output
- (lambda (c)
- (aset ,buffer ,leng c)
- (if (= ,size (setq ,leng (1+ ,leng)))
- (progn (write-region ,buffer nil ,file ,append 'no-msg)
- (setq ,leng 0
- ,append t))))))
- ,@body
- (when (> ,leng 0)
- (let ((coding-system-for-write 'no-conversion))
- (write-region (substring ,buffer 0 ,leng) nil ,file
- ,append 'no-msg))))))
- (put 'gnus-with-output-to-file 'lisp-indent-function 1)
- (put 'gnus-with-output-to-file 'edebug-form-spec '(form body))
- (if (fboundp 'union)
- (defalias 'gnus-union 'union)
- (defun gnus-union (l1 l2)
- "Set union of lists L1 and L2."
- (cond ((null l1) l2)
- ((null l2) l1)
- ((equal l1 l2) l1)
- (t
- (or (>= (length l1) (length l2))
- (setq l1 (prog1 l2 (setq l2 l1))))
- (while l2
- (or (member (car l2) l1)
- (push (car l2) l1))
- (pop l2))
- l1))))
- (declare-function gnus-add-text-properties "gnus"
- (start end properties &optional object))
- (defun gnus-add-text-properties-when
- (property value start end properties &optional object)
- "Like `gnus-add-text-properties', only applied on where PROPERTY is VALUE."
- (let (point)
- (while (and start
- (< start end) ;; XEmacs will loop for every when start=end.
- (setq point (text-property-not-all start end property value)))
- (gnus-add-text-properties start point properties object)
- (setq start (text-property-any point end property value)))
- (if start
- (gnus-add-text-properties start end properties object))))
- (defun gnus-remove-text-properties-when
- (property value start end properties &optional object)
- "Like `remove-text-properties', only applied on where PROPERTY is VALUE."
- (let (point)
- (while (and start
- (< start end)
- (setq point (text-property-not-all start end property value)))
- (remove-text-properties start point properties object)
- (setq start (text-property-any point end property value)))
- (if start
- (remove-text-properties start end properties object))
- t))
- (defun gnus-string-remove-all-properties (string)
- (condition-case ()
- (let ((s string))
- (set-text-properties 0 (length string) nil string)
- s)
- (error string)))
- ;; This might use `compare-strings' to reduce consing in the
- ;; case-insensitive case, but it has to cope with null args.
- ;; (`string-equal' uses symbol print names.)
- (defun gnus-string-equal (x y)
- "Like `string-equal', except it compares case-insensitively."
- (and (= (length x) (length y))
- (or (string-equal x y)
- (string-equal (downcase x) (downcase y)))))
- (defcustom gnus-use-byte-compile t
- "If non-nil, byte-compile crucial run-time code.
- Setting it to nil has no effect after the first time `gnus-byte-compile'
- is run."
- :type 'boolean
- :version "22.1"
- :group 'gnus-various)
- (defun gnus-byte-compile (form)
- "Byte-compile FORM if `gnus-use-byte-compile' is non-nil."
- (if gnus-use-byte-compile
- (progn
- (condition-case nil
- ;; Work around a bug in XEmacs 21.4
- (require 'byte-optimize)
- (error))
- (require 'bytecomp)
- (defalias 'gnus-byte-compile
- (lambda (form)
- (let ((byte-compile-warnings '(unresolved callargs redefine)))
- (byte-compile form))))
- (gnus-byte-compile form))
- form))
- (defun gnus-remassoc (key alist)
- "Delete by side effect any elements of LIST whose car is `equal' to KEY.
- The modified LIST is returned. If the first member
- of LIST has a car that is `equal' to KEY, there is no way to remove it
- by side effect; therefore, write `(setq foo (gnus-remassoc key foo))' to be
- sure of changing the value of `foo'."
- (when alist
- (if (equal key (caar alist))
- (cdr alist)
- (setcdr alist (gnus-remassoc key (cdr alist)))
- alist)))
- (defun gnus-update-alist-soft (key value alist)
- (if value
- (cons (cons key value) (gnus-remassoc key alist))
- (gnus-remassoc key alist)))
- (defun gnus-create-info-command (node)
- "Create a command that will go to info NODE."
- `(lambda ()
- (interactive)
- ,(concat "Enter the info system at node " node)
- (Info-goto-node ,node)
- (setq gnus-info-buffer (current-buffer))
- (gnus-configure-windows 'info)))
- (defun gnus-not-ignore (&rest args)
- t)
- (defvar gnus-directory-sep-char-regexp "/"
- "The regexp of directory separator character.
- If you find some problem with the directory separator character, try
- \"[/\\\\\]\" for some systems.")
- (defun gnus-url-unhex (x)
- (if (> x ?9)
- (if (>= x ?a)
- (+ 10 (- x ?a))
- (+ 10 (- x ?A)))
- (- x ?0)))
- ;; Fixme: Do it like QP.
- (defun gnus-url-unhex-string (str &optional allow-newlines)
- "Remove %XX, embedded spaces, etc in a url.
- If optional second argument ALLOW-NEWLINES is non-nil, then allow the
- decoding of carriage returns and line feeds in the string, which is normally
- forbidden in URL encoding."
- (let ((tmp "")
- (case-fold-search t))
- (while (string-match "%[0-9a-f][0-9a-f]" str)
- (let* ((start (match-beginning 0))
- (ch1 (gnus-url-unhex (elt str (+ start 1))))
- (code (+ (* 16 ch1)
- (gnus-url-unhex (elt str (+ start 2))))))
- (setq tmp (concat
- tmp (substring str 0 start)
- (cond
- (allow-newlines
- (char-to-string code))
- ((or (= code ?\n) (= code ?\r))
- " ")
- (t (char-to-string code))))
- str (substring str (match-end 0)))))
- (setq tmp (concat tmp str))
- tmp))
- (defun gnus-make-predicate (spec)
- "Transform SPEC into a function that can be called.
- SPEC is a predicate specifier that contains stuff like `or', `and',
- `not', lists and functions. The functions all take one parameter."
- `(lambda (elem) ,(gnus-make-predicate-1 spec)))
- (defun gnus-make-predicate-1 (spec)
- (cond
- ((symbolp spec)
- `(,spec elem))
- ((listp spec)
- (if (memq (car spec) '(or and not))
- `(,(car spec) ,@(mapcar 'gnus-make-predicate-1 (cdr spec)))
- (error "Invalid predicate specifier: %s" spec)))))
- (defun gnus-completing-read (prompt collection &optional require-match
- initial-input history def)
- "Call `gnus-completing-read-function'."
- (funcall gnus-completing-read-function
- (concat prompt (when def
- (concat " (default " def ")"))
- ": ")
- collection require-match initial-input history def))
- (defun gnus-emacs-completing-read (prompt collection &optional require-match
- initial-input history def)
- "Call standard `completing-read-function'."
- (let ((completion-styles gnus-completion-styles))
- (completing-read prompt
- (if (featurep 'xemacs)
- ;; Old XEmacs (at least 21.4) expect an alist,
- ;; in which the car of each element is a string,
- ;; for collection.
- (mapcar
- (lambda (elem)
- (list (format "%s" (or (car-safe elem) elem))))
- collection)
- collection)
- nil require-match initial-input history def)))
- (autoload 'ido-completing-read "ido")
- (defun gnus-ido-completing-read (prompt collection &optional require-match
- initial-input history def)
- "Call `ido-completing-read-function'."
- (ido-completing-read prompt collection nil require-match
- initial-input history def))
- (declare-function iswitchb-read-buffer "iswitchb"
- (prompt &optional default require-match
- _predicate start matches-set))
- (defvar iswitchb-temp-buflist)
- (defvar iswitchb-mode)
- (defun gnus-iswitchb-completing-read (prompt collection &optional require-match
- initial-input history def)
- "`iswitchb' based completing-read function."
- ;; Make sure iswitchb is loaded before we let-bind its variables.
- ;; If it is loaded inside the let, variables can become unbound afterwards.
- (require 'iswitchb)
- (let ((iswitchb-make-buflist-hook
- (lambda ()
- (setq iswitchb-temp-buflist
- (let ((choices (append
- (when initial-input (list initial-input))
- (symbol-value history) collection))
- filtered-choices)
- (dolist (x choices)
- (setq filtered-choices (adjoin x filtered-choices)))
- (nreverse filtered-choices))))))
- (unwind-protect
- (progn
- (or iswitchb-mode
- (add-hook 'minibuffer-setup-hook 'iswitchb-minibuffer-setup))
- (iswitchb-read-buffer prompt def require-match))
- (or iswitchb-mode
- (remove-hook 'minibuffer-setup-hook 'iswitchb-minibuffer-setup)))))
- (defun gnus-graphic-display-p ()
- (if (featurep 'xemacs)
- (device-on-window-system-p)
- (display-graphic-p)))
- (put 'gnus-parse-without-error 'lisp-indent-function 0)
- (put 'gnus-parse-without-error 'edebug-form-spec '(body))
- (defmacro gnus-parse-without-error (&rest body)
- "Allow continuing onto the next line even if an error occurs."
- `(while (not (eobp))
- (condition-case ()
- (progn
- ,@body
- (goto-char (point-max)))
- (error
- (gnus-error 4 "Invalid data on line %d"
- (count-lines (point-min) (point)))
- (forward-line 1)))))
- (defun gnus-cache-file-contents (file variable function)
- "Cache the contents of FILE in VARIABLE. The contents come from FUNCTION."
- (let ((time (nth 5 (file-attributes file)))
- contents value)
- (if (or (null (setq value (symbol-value variable)))
- (not (equal (car value) file))
- (not (equal (nth 1 value) time)))
- (progn
- (setq contents (funcall function file))
- (set variable (list file time contents))
- contents)
- (nth 2 value))))
- (defun gnus-multiple-choice (prompt choice &optional idx)
- "Ask user a multiple choice question.
- CHOICE is a list of the choice char and help message at IDX."
- (let (tchar buf)
- (save-window-excursion
- (save-excursion
- (while (not tchar)
- (message "%s (%s): "
- prompt
- (concat
- (mapconcat (lambda (s) (char-to-string (car s)))
- choice ", ") ", ?"))
- (setq tchar (read-char))
- (when (not (assq tchar choice))
- (setq tchar nil)
- (setq buf (get-buffer-create "*Gnus Help*"))
- (pop-to-buffer buf)
- (fundamental-mode) ; for Emacs 20.4+
- (buffer-disable-undo)
- (erase-buffer)
- (insert prompt ":\n\n")
- (let ((max -1)
- (list choice)
- (alist choice)
- (idx (or idx 1))
- (i 0)
- n width pad format)
- ;; find the longest string to display
- (while list
- (setq n (length (nth idx (car list))))
- (unless (> max n)
- (setq max n))
- (setq list (cdr list)))
- (setq max (+ max 4)) ; %c, `:', SPACE, a SPACE at end
- (setq n (/ (1- (window-width)) max)) ; items per line
- (setq width (/ (1- (window-width)) n)) ; width of each item
- ;; insert `n' items, each in a field of width `width'
- (while alist
- (if (< i n)
- ()
- (setq i 0)
- (delete-char -1) ; the `\n' takes a char
- (insert "\n"))
- (setq pad (- width 3))
- (setq format (concat "%c: %-" (int-to-string pad) "s"))
- (insert (format format (caar alist) (nth idx (car alist))))
- (setq alist (cdr alist))
- (setq i (1+ i))))))))
- (if (buffer-live-p buf)
- (kill-buffer buf))
- tchar))
- (if (featurep 'emacs)
- (defalias 'gnus-select-frame-set-input-focus 'select-frame-set-input-focus)
- (if (fboundp 'select-frame-set-input-focus)
- (defalias 'gnus-select-frame-set-input-focus 'select-frame-set-input-focus)
- ;; XEmacs 21.4, SXEmacs
- (defun gnus-select-frame-set-input-focus (frame)
- "Select FRAME, raise it, and set input focus, if possible."
- (raise-frame frame)
- (select-frame frame)
- (focus-frame frame))))
- (defun gnus-frame-or-window-display-name (object)
- "Given a frame or window, return the associated display name.
- Return nil otherwise."
- (if (featurep 'xemacs)
- (device-connection (dfw-device object))
- (if (or (framep object)
- (and (windowp object)
- (setq object (window-frame object))))
- (let ((display (frame-parameter object 'display)))
- (if (and (stringp display)
- ;; Exclude invalid display names.
- (string-match "\\`[^:]*:[0-9]+\\(\\.[0-9]+\\)?\\'"
- display))
- display)))))
- (defvar tool-bar-mode)
- (defun gnus-tool-bar-update (&rest ignore)
- "Update the tool bar."
- (when (and (boundp 'tool-bar-mode)
- tool-bar-mode)
- (let* ((args nil)
- (func (cond ((featurep 'xemacs)
- 'ignore)
- ((fboundp 'tool-bar-update)
- 'tool-bar-update)
- ((fboundp 'force-window-update)
- 'force-window-update)
- ((fboundp 'redraw-frame)
- (setq args (list (selected-frame)))
- 'redraw-frame)
- (t 'ignore))))
- (apply func args))))
- ;; Fixme: This has only one use (in gnus-agent), which isn't worthwhile.
- (defmacro gnus-mapcar (function seq1 &rest seqs2_n)
- "Apply FUNCTION to each element of the sequences, and make a list of the results.
- If there are several sequences, FUNCTION is called with that many arguments,
- and mapping stops as soon as the shortest sequence runs out. With just one
- sequence, this is like `mapcar'. With several, it is like the Common Lisp
- `mapcar' function extended to arbitrary sequence types."
- (if seqs2_n
- (let* ((seqs (cons seq1 seqs2_n))
- (cnt 0)
- (heads (mapcar (lambda (seq)
- (make-symbol (concat "head"
- (int-to-string
- (setq cnt (1+ cnt))))))
- seqs))
- (result (make-symbol "result"))
- (result-tail (make-symbol "result-tail")))
- `(let* ,(let* ((bindings (cons nil nil))
- (heads heads))
- (nconc bindings (list (list result '(cons nil nil))))
- (nconc bindings (list (list result-tail result)))
- (while heads
- (nconc bindings (list (list (pop heads) (pop seqs)))))
- (cdr bindings))
- (while (and ,@heads)
- (setcdr ,result-tail (cons (funcall ,function
- ,@(mapcar (lambda (h) (list 'car h))
- heads))
- nil))
- (setq ,result-tail (cdr ,result-tail)
- ,@(apply 'nconc (mapcar (lambda (h) (list h (list 'cdr h))) heads))))
- (cdr ,result)))
- `(mapcar ,function ,seq1)))
- (if (fboundp 'merge)
- (defalias 'gnus-merge 'merge)
- ;; Adapted from cl-seq.el
- (defun gnus-merge (type list1 list2 pred)
- "Destructively merge lists LIST1 and LIST2 to produce a new list.
- Argument TYPE is for compatibility and ignored.
- Ordering of the elements is preserved according to PRED, a `less-than'
- predicate on the elements."
- (let ((res nil))
- (while (and list1 list2)
- (if (funcall pred (car list2) (car list1))
- (push (pop list2) res)
- (push (pop list1) res)))
- (nconc (nreverse res) list1 list2))))
- (defvar xemacs-codename)
- (defvar sxemacs-codename)
- (defvar emacs-program-version)
- (defun gnus-emacs-version ()
- "Stringified Emacs version."
- (let* ((lst (if (listp gnus-user-agent)
- gnus-user-agent
- '(gnus emacs type)))
- (system-v (cond ((memq 'config lst)
- system-configuration)
- ((memq 'type lst)
- (symbol-name system-type))
- (t nil)))
- codename emacsname)
- (cond ((featurep 'sxemacs)
- (setq emacsname "SXEmacs"
- codename sxemacs-codename))
- ((featurep 'xemacs)
- (setq emacsname "XEmacs"
- codename xemacs-codename))
- (t
- (setq emacsname "Emacs")))
- (cond
- ((not (memq 'emacs lst))
- nil)
- ((string-match "^\\(\\([.0-9]+\\)*\\)\\.[0-9]+$" emacs-version)
- ;; Emacs:
- (concat "Emacs/" (match-string 1 emacs-version)
- (if system-v
- (concat " (" system-v ")")
- "")))
- ((or (featurep 'sxemacs) (featurep 'xemacs))
- ;; XEmacs or SXEmacs:
- (concat emacsname "/" emacs-program-version
- (let (plst)
- (when (memq 'codename lst)
- (push codename plst))
- (when system-v
- (push system-v plst))
- (unless (featurep 'mule)
- (push "no MULE" plst))
- (when (> (length plst) 0)
- (concat
- " (" (mapconcat 'identity (reverse plst) ", ") ")")))))
- (t emacs-version))))
- (defun gnus-rename-file (old-path new-path &optional trim)
- "Rename OLD-PATH as NEW-PATH. If TRIM, recursively delete
- empty directories from OLD-PATH."
- (when (file-exists-p old-path)
- (let* ((old-dir (file-name-directory old-path))
- (old-name (file-name-nondirectory old-path))
- (new-dir (file-name-directory new-path))
- (new-name (file-name-nondirectory new-path))
- temp)
- (gnus-make-directory new-dir)
- (rename-file old-path new-path t)
- (when trim
- (while (progn (setq temp (directory-files old-dir))
- (while (member (car temp) '("." ".."))
- (setq temp (cdr temp)))
- (= (length temp) 0))
- (delete-directory old-dir)
- (setq old-dir (file-name-as-directory
- (file-truename
- (concat old-dir "..")))))))))
- (defun gnus-set-file-modes (filename mode)
- "Wrapper for set-file-modes."
- (ignore-errors
- (set-file-modes filename mode)))
- (if (fboundp 'set-process-query-on-exit-flag)
- (defalias 'gnus-set-process-query-on-exit-flag
- 'set-process-query-on-exit-flag)
- (defalias 'gnus-set-process-query-on-exit-flag
- 'process-kill-without-query))
- (defalias 'gnus-read-shell-command
- (if (fboundp 'read-shell-command) 'read-shell-command 'read-string))
- (defmacro gnus-put-display-table (range value display-table)
- "Set the value for char RANGE to VALUE in DISPLAY-TABLE. "
- (if (featurep 'xemacs)
- (progn
- `(if (fboundp 'put-display-table)
- (put-display-table ,range ,value ,display-table)
- (if (sequencep ,display-table)
- (aset ,display-table ,range ,value)
- (put-char-table ,range ,value ,display-table))))
- `(aset ,display-table ,range ,value)))
- (defmacro gnus-get-display-table (character display-table)
- "Find value for CHARACTER in DISPLAY-TABLE. "
- (if (featurep 'xemacs)
- `(if (fboundp 'get-display-table)
- (get-display-table ,character ,display-table)
- (if (sequencep ,display-table)
- (aref ,display-table ,character)
- (get-char-table ,character ,display-table)))
- `(aref ,display-table ,character)))
- (declare-function image-size "image.c" (spec &optional pixels frame))
- (defun gnus-rescale-image (image size)
- "Rescale IMAGE to SIZE if possible.
- SIZE is in format (WIDTH . HEIGHT). Return a new image.
- Sizes are in pixels."
- (if (or (not (fboundp 'imagemagick-types))
- (not (get-buffer-window (current-buffer))))
- image
- (let ((new-width (car size))
- (new-height (cdr size)))
- (when (> (cdr (image-size image t)) new-height)
- (setq image (or (create-image (plist-get (cdr image) :data) 'imagemagick t
- :height new-height)
- image)))
- (when (> (car (image-size image t)) new-width)
- (setq image (or
- (create-image (plist-get (cdr image) :data) 'imagemagick t
- :width new-width)
- image)))
- image)))
- (eval-when-compile (require 'gmm-utils))
- (defun gnus-recursive-directory-files (dir)
- "Return all regular files below DIR.
- The first found will be returned if a file has hard or symbolic links."
- (let (files attr attrs)
- (gmm-labels
- ((fn (directory)
- (dolist (file (directory-files directory t))
- (setq attr (file-attributes (file-truename file)))
- (when (and (not (member attr attrs))
- (not (member (file-name-nondirectory file)
- '("." "..")))
- (file-readable-p file))
- (push attr attrs)
- (cond ((file-regular-p file)
- (push file files))
- ((file-directory-p file)
- (fn file)))))))
- (fn dir))
- files))
- (defun gnus-list-memq-of-list (elements list)
- "Return non-nil if any of the members of ELEMENTS are in LIST."
- (let ((found nil))
- (dolist (elem elements)
- (setq found (or found
- (memq elem list))))
- found))
- (eval-and-compile
- (cond
- ((fboundp 'match-substitute-replacement)
- (defalias 'gnus-match-substitute-replacement 'match-substitute-replacement))
- (t
- (defun gnus-match-substitute-replacement (replacement &optional fixedcase literal string subexp)
- "Return REPLACEMENT as it will be inserted by `replace-match'.
- In other words, all back-references in the form `\\&' and `\\N'
- are substituted with actual strings matched by the last search.
- Optional FIXEDCASE, LITERAL, STRING and SUBEXP have the same
- meaning as for `replace-match'.
- This is the definition of match-substitute-replacement in subr.el from GNU Emacs."
- (let ((match (match-string 0 string)))
- (save-match-data
- (set-match-data (mapcar (lambda (x)
- (if (numberp x)
- (- x (match-beginning 0))
- x))
- (match-data t)))
- (replace-match replacement fixedcase literal match subexp)))))))
- (if (fboundp 'string-match-p)
- (defalias 'gnus-string-match-p 'string-match-p)
- (defsubst gnus-string-match-p (regexp string &optional start)
- "\
- Same as `string-match' except this function does not change the match data."
- (save-match-data
- (string-match regexp string start))))
- (if (fboundp 'string-prefix-p)
- (defalias 'gnus-string-prefix-p 'string-prefix-p)
- (defun gnus-string-prefix-p (str1 str2 &optional ignore-case)
- "Return non-nil if STR1 is a prefix of STR2.
- If IGNORE-CASE is non-nil, the comparison is done without paying attention
- to case differences."
- (and (<= (length str1) (length str2))
- (let ((prefix (substring str2 0 (length str1))))
- (if ignore-case
- (string-equal (downcase str1) (downcase prefix))
- (string-equal str1 prefix))))))
- (if (fboundp 'format-message)
- (defalias 'gnus-format-message 'format-message)
- ;; for Emacs < 25, and XEmacs, don't worry about quote translation.
- (defalias 'gnus-format-message 'format))
- ;; Simple check: can be a macro but this way, although slow, it's really clear.
- ;; We don't use `bound-and-true-p' because it's not in XEmacs.
- (defun gnus-bound-and-true-p (sym)
- (and (boundp sym) (symbol-value sym)))
- (if (fboundp 'timer--function)
- (defalias 'gnus-timer--function 'timer--function)
- (defun gnus-timer--function (timer)
- (elt timer 5)))
- (provide 'gnus-util)
- ;;; gnus-util.el ends here
|