Replace deprecated time-stamp-hh:mm:ss by format-time-string
[elisp/wanderlust.git] / wl / wl-util.el
1 ;;; wl-util.el --- Utility modules for Wanderlust.
2
3 ;; Copyright (C) 1998,1999,2000 Yuuichi Teranishi <teranisi@gohome.org>
4 ;; Copyright (C) 2000 A. SAGATA <sagata@nttvdt.hil.ntt.co.jp>
5 ;; Copyright (C) 2000 Katsumi Yamaoka <yamaoka@jpl.org>
6
7 ;; Author: Yuuichi Teranishi <teranisi@gohome.org>
8 ;;      A. SAGATA <sagata@nttvdt.hil.ntt.co.jp>
9 ;;      Katsumi Yamaoka <yamaoka@jpl.org>
10 ;; Keywords: mail, net news
11
12 ;; This file is part of Wanderlust (Yet Another Message Interface on Emacsen).
13
14 ;; This program is free software; you can redistribute it and/or modify
15 ;; it under the terms of the GNU General Public License as published by
16 ;; the Free Software Foundation; either version 2, or (at your option)
17 ;; any later version.
18 ;;
19 ;; This program is distributed in the hope that it will be useful,
20 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
21 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
22 ;; GNU General Public License for more details.
23 ;;
24 ;; You should have received a copy of the GNU General Public License
25 ;; along with GNU Emacs; see the file COPYING.  If not, write to the
26 ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
27 ;; Boston, MA 02111-1307, USA.
28 ;;
29
30 ;;; Commentary:
31 ;;
32
33 ;;; Code:
34 ;;
35 (require 'bytecomp)
36 (require 'elmo-util)
37 (require 'elmo-flag)
38 (require 'wl-vars)
39 (eval-when-compile (require 'elmo-pop3))
40 (eval-when-compile (require 'cl))
41 (eval-when-compile (require 'static))
42
43 (condition-case nil (require 'pp) (error nil))
44
45 (eval-when-compile
46   (require 'time-stamp)
47   (defalias-maybe 'next-command-event 'ignore)
48   (defalias-maybe 'event-to-character 'ignore)
49   (defalias-maybe 'key-press-event-p 'ignore)
50   (defalias-maybe 'button-press-event-p 'ignore)
51   (defalias-maybe 'set-process-kanji-code 'ignore)
52   (defalias-maybe 'set-process-coding-system 'ignore)
53   (defalias-maybe 'dispatch-event 'ignore))
54
55 (defalias 'wl-set-work-buf 'elmo-set-work-buf)
56 (make-obsolete 'wl-set-work-buf 'elmo-set-work-buf)
57
58 (defmacro wl-append (val func)
59   (list 'if val
60       (list 'nconc val func)
61     (list 'setq val func)))
62
63 (defalias 'wl-parse 'elmo-parse)
64 (make-obsolete 'wl-parse 'elmo-parse)
65
66 (defun wl-delete-duplicates (list &optional all hack-addresses)
67   "Delete duplicate equivalent strings from the LIST.
68 If ALL is t, then if there is more than one occurrence of a string in the LIST,
69  then all occurrences of it are removed instead of just the subsequent ones.
70 If HACK-ADDRESSES is t, then the strings are considered to be mail addresses,
71  and only the address part is compared (so that \"Name <foo>\" and \"foo\"
72  would be considered to be equivalent.)"
73   (let ((hashtable (make-vector 29 0))
74         (new-list nil)
75         sym-string sym)
76     (fillarray hashtable 0)
77     (while list
78       (setq sym-string
79             (if hack-addresses
80                 (wl-address-header-extract-address (car list))
81               (car list))
82             sym-string (or sym-string "-unparseable-garbage-")
83             sym (intern sym-string hashtable))
84       (if (boundp sym)
85           (and all (setcar (symbol-value sym) nil))
86         (setq new-list (cons (car list) new-list))
87         (set sym new-list))
88       (setq list (cdr list)))
89     (delq nil (nreverse new-list))))
90
91 ;; string utils.
92 (defalias 'wl-string-member 'elmo-string-member)
93 (defalias 'wl-string-match-member 'elmo-string-match-member)
94 (defalias 'wl-string-delete-match 'elmo-string-delete-match)
95 (defalias 'wl-string-match-assoc 'elmo-string-match-assoc)
96 (defalias 'wl-string-assoc 'elmo-string-assoc)
97 (defalias 'wl-string-rassoc 'elmo-string-rassoc)
98
99 (defalias 'wl-parse-addresses 'elmo-parse-addresses)
100
101 (defun wl-append-element (list element)
102   (if element
103       (append list (list element))
104     list))
105
106 (defmacro wl-push (v l)
107   "Insert V at the head of the list stored in L."
108   (list 'setq l (list 'cons v l)))
109
110 (defmacro wl-pop (l)
111   "Remove the head of the list stored in L."
112   (list 'car (list 'prog1 l (list 'setq l (list 'cdr l)))))
113
114 (defun wl-ask-folder (func mes-string)
115   (let* (key keve
116              (cmd (if (featurep 'xemacs)
117                       (event-to-character last-command-event)
118                     (string-to-char (format "%s" (this-command-keys))))))
119     (message "%s" mes-string)
120     (setq key (car (setq keve (wl-read-event-char))))
121     (if (or (equal key (string-to-char " "))
122             (and cmd
123                  (equal key cmd)))
124         (progn
125           (message "")
126           (funcall func))
127       (wl-push (cdr keve) unread-command-events))))
128
129 (defun wl-require-update-all-folder-p (name)
130   "Return non-nil if NAME is draft or queue folder."
131   (or (string= name wl-draft-folder)
132       (string= name wl-queue-folder)))
133
134 ;;;(defalias 'wl-make-hash 'elmo-make-hash)
135 ;;;(make-obsolete 'wl-make-hash 'elmo-make-hash)
136
137 ;;;(defalias 'wl-get-hash-val 'elmo-get-hash-val)
138 ;;;(make-obsolete 'wl-get-hash-val 'elmo-get-hash-val)
139
140 ;;;(defalias 'wl-set-hash-val 'elmo-set-hash-val)
141 ;;;(make-obsolete 'wl-set-hash-val 'elmo-set-hash-val)
142
143 (defsubst wl-set-string-width (width string &optional padding ignore-invalid)
144   "Make a new string which have specified WIDTH and content of STRING.
145 `wl-invalid-character-message' is used when invalid character is contained.
146 If WIDTH is negative number, padding chars are added to the head and
147 otherwise, padding chars are added to the tail of the string.
148 The optional 3rd arg PADDING, if non-nil, specifies a padding character
149 to add the result instead of white space.
150 If optional 4th argument is non-nil, don't use `wl-invalid-character-message'
151 even when invalid character is contained."
152   (static-cond
153    ((and (fboundp 'string-width) (fboundp 'truncate-string-to-width)
154          (not (featurep 'xemacs)))
155     (if (> (string-width string) (abs width))
156         (setq string (truncate-string-to-width string (abs width))))
157     (if (= (string-width string) (abs width))
158         string
159       (when (and (not ignore-invalid)
160                  (< (abs width) (string-width string)))
161         (setq string
162               (truncate-string-to-width wl-invalid-character-message
163                                         (abs width))))
164       (let ((paddings (make-string
165                        (max 0 (- (abs width) (string-width string)))
166                        (or padding (string-to-char " ")))))
167         (if (< width 0)
168             (concat paddings string)
169           (concat string paddings)))))
170    (t
171     (elmo-set-work-buf
172      (set-buffer-multibyte default-enable-multibyte-characters)
173      (insert string)
174      (when (> (current-column) (abs width))
175        (when (> (move-to-column (abs width)) (abs width))
176          (condition-case nil ; ignore error
177              (backward-char 1)
178            (error)))
179        (setq string (buffer-substring (point-min) (point))))
180      (if (= (current-column) (abs width))
181          string
182        (let ((paddings (make-string (- (abs width) (current-column))
183                                     (or padding (string-to-char " ")))))
184          (if (< width 0)
185              (concat paddings string)
186            (concat string paddings))))))))
187
188 (defun wl-mode-line-buffer-identification (&optional id)
189   (let ((priorities '(biff plug title)))
190     (let ((items (reverse wl-mode-line-display-priority-list))
191           item)
192       (while items
193         (setq item (car items)
194               items (cdr items))
195         (unless (memq item '(biff plug))
196           (setq item 'title))
197         (setq priorities (cons item (delq item priorities)))))
198     (let (priority result)
199       (while priorities
200         (setq priority (car priorities)
201               priorities (cdr priorities))
202         (cond
203          ((eq 'biff priority)
204           (when wl-biff-check-folder-list
205             (setq result (append result '((wl-modeline-biff-status
206                                            wl-modeline-biff-state-on
207                                            wl-modeline-biff-state-off))))))
208          ((eq 'plug priority)
209           (when wl-show-plug-status-on-modeline
210             (setq result (append result '((wl-modeline-plug-status
211                                            wl-modeline-plug-state-on
212                                            wl-modeline-plug-state-off))))))
213          (t
214           (setq result (append result (or id '("Wanderlust: %12b")))))))
215       (prog1
216           (setq mode-line-buffer-identification (if (stringp (car result))
217                                                     result
218                                                   (cons "" result)))
219         (force-mode-line-update t)))))
220
221 (defalias 'wl-display-error 'elmo-display-error)
222 (make-obsolete 'wl-display-error 'elmo-display-error)
223
224 (defun wl-get-assoc-list-value (assoc-list folder &optional match)
225   (catch 'found
226     (let ((alist assoc-list)
227           value pair)
228       (while alist
229         (setq pair (car alist))
230         (if (and (eq match 'function)
231                  (functionp (car pair)))
232             (when (funcall (car pair) folder)
233               (throw 'found (cdr pair)))
234           (if (string-match (car pair) folder)
235               (cond ((eq match 'all)
236                      (setq value (append value (list (cdr pair)))))
237                     ((eq match 'all-list)
238                      (setq value (append value (cdr pair))))
239                     ((or (not match) (eq match 'function))
240                      (throw 'found (cdr pair))))))
241         (setq alist (cdr alist)))
242       value)))
243
244 (defun wl-match-string (pos string)
245   "Substring POSth matched STRING."
246   (substring string (match-beginning pos) (match-end pos)))
247
248 (defun wl-match-buffer (pos)
249   "Substring POSth matched from the current buffer."
250   (buffer-substring-no-properties
251    (match-beginning pos) (match-end pos)))
252
253 (put 'wl-as-coding-system 'lisp-indent-function 1)
254 (put 'wl-as-mime-charset 'lisp-indent-function 1)
255
256 (eval-and-compile
257   (cond
258    (wl-on-mule3
259     (defmacro wl-as-coding-system (coding-system &rest body)
260       `(let ((coding-system-for-read ,coding-system)
261              (coding-system-for-write ,coding-system))
262          ,@body)))
263    (wl-on-mule
264     (defmacro wl-as-coding-system (coding-system &rest body)
265       `(let ((file-coding-system-for-read ,coding-system)
266              (file-coding-system ,coding-system))
267          ,@body)))
268    (t
269     (defmacro wl-as-coding-system (coding-system &rest body)
270       `(progn ,@body)))))
271
272 (defmacro wl-as-mime-charset (mime-charset &rest body)
273   `(wl-as-coding-system (mime-charset-to-coding-system ,mime-charset)
274      ,@body))
275
276 (defalias 'wl-string 'elmo-string)
277 (make-obsolete 'wl-string 'elmo-string)
278
279 (if (not (fboundp 'overlays-in))
280     (defun overlays-in (beg end)
281       "Return a list of the overlays that overlap the region BEG ... END.
282 Overlap means that at least one character is contained within the overlay
283 and also contained within the specified region.
284 Empty overlays are included in the result if they are located at BEG
285 or between BEG and END."
286       (let ((ovls (overlay-lists))
287             tmp retval)
288         (if (< end beg)
289             (setq tmp end
290                   end beg
291                   beg tmp))
292         (setq ovls (nconc (car ovls) (cdr ovls)))
293         (while ovls
294           (setq tmp (car ovls)
295                 ovls (cdr ovls))
296           (if (or (and (<= (overlay-start tmp) end)
297                        (>= (overlay-start tmp) beg))
298                   (and (<= (overlay-end tmp) end)
299                        (>= (overlay-end tmp) beg)))
300               (setq retval (cons tmp retval))))
301         retval)))
302
303 (defsubst wl-repeat-string (str times)
304   (let ((loop times)
305         ret-val)
306     (while (> loop 0)
307       (setq ret-val (concat ret-val str))
308       (setq loop (- loop 1)))
309     ret-val))
310
311 (defun wl-append-assoc-list (item value alist)
312   "make assoc list '((item1 value1-1 value1-2 ...)) (item2 value2-1 ...)))"
313   (let ((entry (assoc item alist)))
314     (if entry
315         (progn
316           (when (not (member value (cdr entry)))
317             (nconc entry (list value)))
318           alist)
319       (append alist
320               (list (list item value))))))
321
322 (defun wl-delete-alist (key alist)
323   "Delete by side effect any entries specified with KEY from ALIST.
324 Return the modified ALIST.  Key comparison is done with `assq'.
325 Write `(setq foo (wl-delete-alist key foo))' to be sure of changing
326 the value of `foo'."
327   (let (entry)
328     (while (setq entry (assq key alist))
329       (setq alist (delq entry alist)))
330     alist))
331
332 (defun wl-delete-associations (keys alist)
333   "Delete by side effect any entries specified with KEYS from ALIST.
334 Return the modified ALIST.  KEYS must be a list of keys for ALIST.
335 Deletion is done with `wl-delete-alist'.
336 Write `(setq foo (wl-delete-associations keys foo))' to be sure of
337 changing the value of `foo'."
338   (while keys
339     (setq alist (wl-delete-alist (car keys) alist))
340     (setq keys (cdr keys)))
341   alist)
342
343 (defun wl-filter-associations (keys alist)
344   (let (entry result)
345     (while keys
346       (when (setq entry (assq (car keys) alist))
347         (setq result (cons entry result)))
348       (setq keys (cdr keys)))
349     result))
350
351 (defun wl-inverse-alist (keys alist)
352   "Inverse ALIST, copying.
353 Return an association list represents the inverse mapping of ALIST,
354 from objects to KEYS.
355 The objects mapped (cdrs of elements of the ALIST) are shared."
356   (let (x y tmp result)
357     (while keys
358       (setq x (car keys))
359       (setq y (cdr (assq x alist)))
360       (if y
361           (if (setq tmp (assoc y result))
362               (setq result (cons (append tmp (list x))
363                                  (delete tmp result)))
364             (setq result (cons (list y x) result))))
365       (setq keys (cdr keys)))
366     result))
367
368 (static-unless (fboundp 'pp)
369   (defvar pp-escape-newlines t)
370   (defun pp (object &optional stream)
371     "Output the pretty-printed representation of OBJECT, any Lisp object.
372 Quoting characters are printed when needed to make output that `read'
373 can handle, whenever this is possible.
374 Output stream is STREAM, or value of `standard-output' (which see)."
375     (princ (pp-to-string object) (or stream standard-output)))
376
377   (defun pp-to-string (object)
378     "Return a string containing the pretty-printed representation of OBJECT,
379 any Lisp object.  Quoting characters are used when needed to make output
380 that `read' can handle, whenever this is possible."
381     (save-excursion
382       (set-buffer (generate-new-buffer " pp-to-string"))
383       (unwind-protect
384           (progn
385             (lisp-mode-variables t)
386             (let ((print-escape-newlines pp-escape-newlines))
387               (prin1 object (current-buffer)))
388             (goto-char (point-min))
389             (while (not (eobp))
390               (cond
391                ((looking-at "\\s(\\|#\\s(")
392                 (while (looking-at "\\s(\\|#\\s(")
393                   (forward-char 1)))
394                ((and (looking-at "\\(quote[ \t]+\\)\\([^.)]\\)")
395                      (> (match-beginning 1) 1)
396                      (= ?\( (char-after (1- (match-beginning 1))))
397                      ;; Make sure this is a two-element list.
398                      (save-excursion
399                        (goto-char (match-beginning 2))
400                        (forward-sexp)
401                        ;; Avoid mucking with match-data; does this test work?
402                        (char-equal ?\) (char-after (point)))))
403                 ;; -1 gets the paren preceding the quote as well.
404                 (delete-region (1- (match-beginning 1)) (match-end 1))
405                 (insert "'")
406                 (forward-sexp 1)
407                 (if (looking-at "[ \t]*\)")
408                     (delete-region (match-beginning 0) (match-end 0))
409                   (error "Malformed quote"))
410                 (backward-sexp 1))
411                ((condition-case err-var
412                     (prog1 t (down-list 1))
413                   (error nil))
414                 (backward-char 1)
415                 (skip-chars-backward " \t")
416                 (delete-region
417                  (point)
418                  (progn (skip-chars-forward " \t") (point)))
419                 (if (not (char-equal ?' (char-after (1- (point)))))
420                     (insert ?\n)))
421                ((condition-case err-var
422                     (prog1 t (up-list 1))
423                   (error nil))
424                 (while (looking-at "\\s)")
425                   (forward-char 1))
426                 (skip-chars-backward " \t")
427                 (delete-region
428                  (point)
429                  (progn (skip-chars-forward " \t") (point)))
430                 (if (not (char-equal ?' (char-after (1- (point)))))
431                     (insert ?\n)))
432                (t (goto-char (point-max)))))
433             (goto-char (point-min))
434             (indent-sexp)
435             (buffer-string))
436         (kill-buffer (current-buffer))))))
437
438 (defsubst wl-get-date-iso8601 (date)
439   (or (get-text-property 0 'wl-date date)
440       (let* ((d1 (timezone-fix-time date nil nil))
441              (time (format "%04d%02d%02dT%02d%02d%02d"
442                            (aref d1 0) (aref d1 1) (aref d1 2)
443                            (aref d1 3) (aref d1 4) (aref d1 5))))
444         (put-text-property 0 1 'wl-date time date)
445         time)))
446
447 (defun wl-make-date-string ()
448   (let ((s (current-time-string)))
449     (string-match "\\`\\([A-Z][a-z][a-z]\\) +[A-Z][a-z][a-z] +[0-9][0-9]? *[0-9][0-9]?:[0-9][0-9]:[0-9][0-9] *[0-9]?[0-9]?[0-9][0-9]"
450                   s)
451     (concat (wl-match-string 1 s) ", "
452             (timezone-make-date-arpa-standard s (current-time-zone)))))
453
454 (defun wl-date-iso8601 (date)
455   "Convert the DATE to YYMMDDTHHMMSS."
456   (condition-case ()
457       (wl-get-date-iso8601 date)
458     (error "")))
459
460 (defun wl-url-news (url &rest args)
461   (interactive "sURL: ")
462   (if (string-match "^news:\\(.*\\)$" url)
463       (wl-summary-goto-folder-subr
464        (concat "-" (elmo-match-string 1 url)) nil nil nil t)
465     (message "Not a news: url.")))
466
467 (defun wl-url-nntp (url &rest args)
468   (interactive "sURL: ")
469   (let (folder fld-name server port msg)
470     (if (string-match
471          "^nntp://\\([^:/]*\\):?\\([0-9]*\\)/\\([^/]*\\)/\\([0-9]*\\).*$" url)
472         (progn
473           (if (eq (length (setq fld-name
474                                 (elmo-match-string 3 url))) 0)
475               (setq fld-name nil))
476           (if (eq (length (setq port
477                                 (elmo-match-string 2 url))) 0)
478               (setq port (number-to-string elmo-nntp-default-port)))
479           (if (eq (length (setq server
480                                 (elmo-match-string 1 url))) 0)
481               (setq server elmo-nntp-default-server))
482           (setq folder (concat "-" fld-name "@" server ":" port))
483           (if (eq (length (setq msg
484                                 (elmo-match-string 4 url))) 0)
485               (wl-summary-goto-folder-subr
486                folder nil nil nil t)
487             (wl-summary-goto-folder-subr
488              folder 'update nil nil t)
489             (wl-summary-jump-to-msg (string-to-number msg))
490             (wl-summary-redisplay)))
491       (message "Not a nntp: url."))))
492
493 (defmacro wl-concat-list (list separator)
494   `(mapconcat 'identity (delete "" (delq nil ,list)) ,separator))
495
496 (defun wl-current-message-buffer ()
497   (when (buffer-live-p wl-current-summary-buffer)
498     (with-current-buffer wl-current-summary-buffer
499       (or wl-message-buffer
500           (and (wl-summary-message-number)
501                (wl-message-buffer-display
502                 wl-summary-buffer-elmo-folder
503                 (wl-summary-message-number)
504                 wl-summary-buffer-display-mime-mode
505                 nil nil))))))
506
507 (defun wl-kill-buffers (regexp)
508   (mapc
509    (lambda (x)
510      (if (and (buffer-name x)
511               (string-match regexp (buffer-name x)))
512          (and (get-buffer x)
513               (kill-buffer x))))
514    (buffer-list)))
515
516 (defun wl-collect-summary ()
517   (let (result)
518     (mapc
519      (lambda (x)
520        (if (and (string-match "^Summary"
521                               (buffer-name x))
522                 (with-current-buffer x
523                   (eq major-mode 'wl-summary-mode)))
524            (setq result (nconc result (list x)))))
525      (buffer-list))
526     result))
527
528 (defun wl-collect-draft ()
529   (let ((draft-regexp (concat "^" (regexp-quote wl-draft-folder)))
530         result)
531     (dolist (buffer (buffer-list))
532       (when (with-current-buffer buffer
533               (and (eq major-mode 'wl-draft-mode)
534                    (buffer-name)
535                    (string-match draft-regexp (buffer-name))))
536         (setq result (cons buffer result))))
537     (nreverse result)))
538
539 (defvar wl-inhibit-save-drafts nil)
540 (defvar wl-disable-auto-save nil)
541 (make-variable-buffer-local 'wl-disable-auto-save)
542
543 (defun wl-save-drafts ()
544   "Save all drafts. Return nil if there is no draft buffer."
545   (if wl-inhibit-save-drafts
546       'inhibited
547     (let ((wl-inhibit-save-drafts t)
548           (msg (current-message))
549           (buffers (wl-collect-draft)))
550       (save-excursion
551         (dolist (buffer buffers)
552           (set-buffer buffer)
553           (when (and (not wl-disable-auto-save)
554                      (buffer-modified-p))
555             (wl-draft-save))))
556       (message "%s" (or msg ""))
557       buffers)))
558
559 (static-if (fboundp 'read-directory-name)
560     (defun wl-read-directory-name (prompt dir)
561       (read-directory-name prompt dir dir))
562   (defun wl-read-directory-name (prompt dir)
563     (let ((dir (read-file-name prompt dir)))
564       (unless (file-directory-p dir)
565         (error "%s is not directory" dir))
566       dir)))
567
568 ;; local variable check.
569 (static-if (fboundp 'local-variable-p)
570     (defalias 'wl-local-variable-p 'local-variable-p)
571   (defmacro wl-local-variable-p (symbol &optional buffer)
572     `(if (assq ,symbol (buffer-local-variables ,buffer))
573          t)))
574
575 (defun wl-number-base36 (num len)
576   (if (if (< len 0)
577           (<= num 0)
578         (= len 0))
579       ""
580     (concat (wl-number-base36 (/ num 36) (1- len))
581             (char-to-string (aref "zyxwvutsrqponmlkjihgfedcba9876543210"
582                                   (% num 36))))))
583
584 (defvar wl-unique-id-char nil)
585
586 (defun wl-unique-id ()
587   ;; Don't use microseconds from (current-time), they may be unsupported.
588   ;; Instead we use this randomly inited counter.
589   (setq wl-unique-id-char
590         (% (1+ (or wl-unique-id-char (logand (random t) (1- (lsh 1 20)))))
591            ;; (current-time) returns 16-bit ints,
592            ;; and 2^16*25 just fits into 4 digits i base 36.
593            (* 25 25)))
594   (let ((tm (static-if (fboundp 'current-time)
595                 (current-time)
596               (let* ((cts (split-string (current-time-string) "[ :]"))
597                      (m (cdr (assoc (nth 1 cts)
598                                     '(("Jan" . "01") ("Feb" . "02")
599                                       ("Mar" . "03") ("Apr" . "04")
600                                       ("May" . "05") ("Jun" . "06")
601                                       ("Jul" . "07") ("Aug" . "08")
602                                       ("Sep" . "09") ("Oct" . "10")
603                                       ("Nov" . "11") ("Dec" . "12"))))))
604                 (list (string-to-number (concat (nth 6 cts) m
605                                                 (substring (nth 2 cts) 0 1)))
606                       (string-to-number (concat (substring (nth 2 cts) 1)
607                                                 (nth 4 cts) (nth 5 cts)
608                                                 (nth 6 cts))))))))
609     (concat
610      (if (memq system-type '(ms-dos emx vax-vms))
611          (let ((user (downcase (user-login-name))))
612            (while (string-match "[^a-z0-9_]" user)
613              (aset user (match-beginning 0) ?_))
614            user)
615        (wl-number-base36 (user-uid) -1))
616      (wl-number-base36 (+ (car   tm)
617                           (lsh (% wl-unique-id-char 25) 16)) 4)
618      (wl-number-base36 (+ (nth 1 tm)
619                           (lsh (/ wl-unique-id-char 25) 16)) 4)
620      ;; Append the name of the message interface, because while the
621      ;; generated ID is unique to this newsreader, other newsreaders
622      ;; might otherwise generate the same ID via another algorithm.
623      wl-unique-id-suffix)))
624
625 (defvar wl-message-id-function 'wl-draft-make-message-id-string)
626 (defun wl-draft-make-message-id-string ()
627   "Return Message-ID field value."
628   (concat "<" (wl-unique-id)
629           (let (from user domain)
630             (if (and wl-message-id-use-wl-from
631                      (progn
632                        (setq from (wl-address-header-extract-address wl-from))
633                        (and (string-match "^\\(.*\\)@\\(.*\\)$" from)
634                             (setq user   (match-string 1 from))
635                             (setq domain (match-string 2 from)))))
636                 (format "%%%s@%s>" user domain)
637               (format "@%s>"
638                       (or wl-message-id-domain
639                           (if wl-local-domain
640                               (concat (system-name) "." wl-local-domain)
641                             (system-name))))))))
642
643 ;;; Profile loading.
644 (defvar wl-load-profile-function 'wl-local-load-profile)
645 (defun wl-local-load-profile ()
646   "Load `wl-init-file'."
647   (message "Initializing...")
648   (load wl-init-file 'noerror 'nomessage))
649
650 (defun wl-load-profile ()
651   "Call `wl-load-profile-function' function."
652   (funcall wl-load-profile-function))
653
654 ;;;
655
656 (defsubst wl-count-lines ()
657   (count-lines 1 (point-at-bol)))
658
659 (defun wl-horizontal-recenter ()
660   "Recenter the current buffer horizontally."
661   (beginning-of-line)
662   (re-search-forward "[[<]" (point-at-eol) t)
663   (if (< (current-column) (/ (window-width) 2))
664       (set-window-hscroll (get-buffer-window (current-buffer) t) 0)
665     (let* ((orig (point))
666            (end (window-end (get-buffer-window (current-buffer) t)))
667            (max 0))
668       (when end
669         ;; Find the longest line currently displayed in the window.
670         (goto-char (window-start))
671         (while (and (not (eobp))
672                     (< (point) end))
673           (end-of-line)
674           (setq max (max max (current-column)))
675           (forward-line 1))
676         (goto-char orig)
677         ;; Scroll horizontally to center (sort of) the point.
678         (if (> max (window-width))
679             (set-window-hscroll
680              (get-buffer-window (current-buffer) t)
681              (min (- (current-column) (/ (window-width) 3))
682                   (+ 2 (- max (window-width)))))
683           (set-window-hscroll (get-buffer-window (current-buffer) t) 0))
684         max))))
685
686 ;; Draft auto-save
687 (defun wl-auto-save-drafts ()
688   (unless (wl-save-drafts)
689     (wl-stop-save-drafts)))
690
691 (static-cond
692  (wl-on-xemacs
693   (defvar wl-save-drafts-timer-name "wl-save-drafts")
694
695   (defun wl-start-save-drafts ()
696     (when (numberp wl-auto-save-drafts-interval)
697       (unless (get-itimer wl-save-drafts-timer-name)
698         (start-itimer wl-save-drafts-timer-name
699                       'wl-auto-save-drafts
700                       wl-auto-save-drafts-interval
701                       wl-auto-save-drafts-interval
702                       t))))
703
704   (defun wl-stop-save-drafts ()
705     (when (get-itimer wl-save-drafts-timer-name)
706       (delete-itimer wl-save-drafts-timer-name))))
707  (t
708   (defun wl-start-save-drafts ()
709     (when (numberp wl-auto-save-drafts-interval)
710       (require 'timer)
711       (if (get 'wl-save-drafts 'timer)
712           (progn
713             (timer-set-idle-time (get 'wl-save-drafts 'timer)
714                                  wl-auto-save-drafts-interval t)
715             (timer-activate-when-idle (get 'wl-save-drafts 'timer)))
716         (put 'wl-save-drafts 'timer
717              (run-with-idle-timer
718               wl-auto-save-drafts-interval t 'wl-auto-save-drafts)))))
719
720   (defun wl-stop-save-drafts ()
721     (when (get 'wl-save-drafts 'timer)
722       (cancel-timer (get 'wl-save-drafts 'timer))))))
723
724 (defun wl-set-auto-save-draft (&optional arg)
725   (interactive "P")
726   (unless (setq wl-disable-auto-save
727                 (cond
728                  ((null arg) (not wl-disable-auto-save))
729                  ((< (prefix-numeric-value arg) 0) t)
730                  (t nil)))
731     (wl-start-save-drafts))
732   (when (interactive-p)
733     (message "Auto save is %s (in this buffer)"
734              (if wl-disable-auto-save "disabled" "enabled"))))
735
736 ;; Biff
737 (static-cond
738  (wl-on-xemacs
739   (defvar wl-biff-timer-name "wl-biff")
740
741   (defun wl-biff-stop ()
742     (when (get-itimer wl-biff-timer-name)
743       (delete-itimer wl-biff-timer-name)))
744
745   (defun wl-biff-start ()
746     (wl-biff-stop)
747     (when wl-biff-check-folder-list
748       (start-itimer wl-biff-timer-name 'wl-biff-check-folders
749                     wl-biff-check-interval wl-biff-check-interval
750                     wl-biff-use-idle-timer))))
751
752  (t
753   (defun wl-biff-stop ()
754     (when (get 'wl-biff 'timer)
755       (cancel-timer (get 'wl-biff 'timer))))
756
757   (defun wl-biff-start ()
758     (require 'timer)
759     (when wl-biff-check-folder-list
760       (if wl-biff-use-idle-timer
761           (if (get 'wl-biff 'timer)
762               (progn (timer-set-idle-time (get 'wl-biff 'timer)
763                                           wl-biff-check-interval t)
764                      (timer-activate-when-idle (get 'wl-biff 'timer)))
765             (put 'wl-biff 'timer
766                  (run-with-idle-timer
767                   wl-biff-check-interval t 'wl-biff-event-handler)))
768         (if (get 'wl-biff 'timer)
769             (progn
770               (timer-set-time (get 'wl-biff 'timer)
771                               (timer-next-integral-multiple-of-time
772                                (current-time) wl-biff-check-interval)
773                               wl-biff-check-interval)
774               (timer-activate (get 'wl-biff 'timer)))
775           (put 'wl-biff 'timer
776                (run-at-time
777                 (timer-next-integral-multiple-of-time
778                  (current-time) wl-biff-check-interval)
779                 wl-biff-check-interval
780                 'wl-biff-event-handler))))))
781
782   (defun-maybe timer-next-integral-multiple-of-time (time secs)
783     "Yield the next value after TIME that is an integral multiple of SECS.
784 More precisely, the next value, after TIME, that is an integral multiple
785 of SECS seconds since the epoch.  SECS may be a fraction.
786 This function is imported from Emacs 20.7."
787     (let ((time-base (ash 1 16)))
788       (if (fboundp 'atan)
789           ;; Use floating point, taking care to not lose precision.
790           (let* ((float-time-base (float time-base))
791                  (million 1000000.0)
792                  (time-usec (+ (* million
793                                   (+ (* float-time-base (nth 0 time))
794                                      (nth 1 time)))
795                                (nth 2 time)))
796                  (secs-usec (* million secs))
797                  (mod-usec (mod time-usec secs-usec))
798                  (next-usec (+ (- time-usec mod-usec) secs-usec))
799                  (time-base-million (* float-time-base million)))
800             (list (floor next-usec time-base-million)
801                   (floor (mod next-usec time-base-million) million)
802                   (floor (mod next-usec million))))
803         ;; Floating point is not supported.
804         ;; Use integer arithmetic, avoiding overflow if possible.
805         (let* ((mod-sec (mod (+ (* (mod time-base secs)
806                                    (mod (nth 0 time) secs))
807                                 (nth 1 time))
808                              secs))
809                (next-1-sec (+ (- (nth 1 time) mod-sec) secs)))
810           (list (+ (nth 0 time) (floor next-1-sec time-base))
811                 (mod next-1-sec time-base)
812                 0)))))
813
814   (defun wl-biff-event-handler ()
815     ;; PAKURing from FSF:time.el
816     (wl-biff-check-folders)
817     ;; Do redisplay right now, if no input pending.
818     (sit-for 0)
819     (let* ((current (current-time))
820            (timer (get 'wl-biff 'timer))
821            ;; Compute the time when this timer will run again, next.
822            (next-time (timer-relative-time
823                        (list (aref timer 1) (aref timer 2) (aref timer 3))
824                        (* 5 (aref timer 4)) 0)))
825       ;; If the activation time is far in the past,
826       ;; skip executions until we reach a time in the future.
827       ;; This avoids a long pause if Emacs has been suspended for hours.
828       (or (> (nth 0 next-time) (nth 0 current))
829           (and (= (nth 0 next-time) (nth 0 current))
830                (> (nth 1 next-time) (nth 1 current)))
831           (and (= (nth 0 next-time) (nth 0 current))
832                (= (nth 1 next-time) (nth 1 current))
833                (> (nth 2 next-time) (nth 2 current)))
834           (progn
835             (timer-set-time timer (timer-next-integral-multiple-of-time
836                                    current wl-biff-check-interval)
837                             wl-biff-check-interval)
838             (timer-activate timer)))))))
839
840 (defsubst wl-biff-notify (new-mails notify-minibuf)
841   (when (and (not wl-modeline-biff-status) (> new-mails 0))
842     (run-hooks 'wl-biff-notify-hook))
843   (when (and wl-modeline-biff-status (eq new-mails 0))
844     (run-hooks 'wl-biff-unnotify-hook))
845   (setq wl-modeline-biff-status (> new-mails 0))
846   (force-mode-line-update t)
847   (when notify-minibuf
848     (cond ((zerop new-mails) (message "No mail."))
849           ((= 1 new-mails) (message "You have a new mail."))
850           (t (message "You have %d new mails." new-mails)))))
851
852 ;; Internal variable.
853 (defvar wl-biff-check-folders-running nil)
854
855 (defun wl-biff-check-folders ()
856   (interactive)
857   (if wl-biff-check-folders-running
858       (when (interactive-p)
859         (message "Biff process is running."))
860     (setq wl-biff-check-folders-running t)
861     (when (interactive-p)
862       (message "Checking new mails..."))
863     (let ((new-mails 0)
864           (flist (or wl-biff-check-folder-list (list wl-default-folder)))
865           folder)
866       (if (eq (length flist) 1)
867           (wl-biff-check-folder-async (wl-folder-get-elmo-folder
868                                        (car flist) 'biff) (interactive-p))
869         (unwind-protect
870             (while flist
871               (setq folder (wl-folder-get-elmo-folder (car flist))
872                     flist (cdr flist))
873               (elmo-folder-set-biff-internal folder t)
874               (when (and (elmo-folder-plugged-p folder)
875                          (elmo-folder-exists-p folder))
876                 (setq new-mails
877                       (+ new-mails
878                          (nth 0 (wl-biff-check-folder folder))))))
879           (setq wl-biff-check-folders-running nil)
880           (wl-biff-notify new-mails (interactive-p)))))))
881
882 (defun wl-biff-check-folder (folder)
883   (if (eq (elmo-folder-type-internal folder) 'pop3)
884       (unless (elmo-pop3-get-session folder 'any-exists)
885         (wl-folder-check-one-entity (elmo-folder-name-internal folder)
886                                     'biff))
887     (wl-folder-check-one-entity (elmo-folder-name-internal folder)
888                                 'biff)))
889
890 (defun wl-biff-check-folder-async-callback (diff data)
891   (if (nth 1 data)
892       (with-current-buffer (nth 1 data)
893         (wl-folder-entity-hashtb-set wl-folder-entity-hashtb
894                                      (nth 0 data)
895                                      (list (nth 0 diff)
896                                            (- (nth 1 diff) (nth 0 diff))
897                                            (nth 2 diff))
898                                      (current-buffer))))
899   (setq wl-folder-info-alist-modified t)
900   (setq wl-biff-check-folders-running nil)
901   (sit-for 0)
902   (wl-biff-notify (car diff) (nth 2 data)))
903
904 (defun wl-biff-check-folder-async (folder notify-minibuf)
905   (if (and (elmo-folder-plugged-p folder)
906            (wl-folder-entity-exists-p (elmo-folder-name-internal folder)))
907       (progn
908         (elmo-folder-set-biff-internal folder t)
909         (if (and (eq (elmo-folder-type-internal folder) 'imap4)
910                  (elmo-folder-use-flag-p folder))
911             ;; Check asynchronously only when IMAP4 and use server diff.
912             (progn
913               (setq elmo-folder-diff-async-callback
914                     'wl-biff-check-folder-async-callback)
915               (setq elmo-folder-diff-async-callback-data
916                     (list (elmo-folder-name-internal folder)
917                           (get-buffer wl-folder-buffer-name)
918                           notify-minibuf))
919               (elmo-folder-diff-async folder))
920           (unwind-protect
921               (wl-biff-notify (car (wl-biff-check-folder folder))
922                               notify-minibuf)
923             (setq wl-biff-check-folders-running nil))))
924     (setq wl-biff-check-folders-running nil)))
925
926 (if (and (fboundp 'regexp-opt)
927          (not (featurep 'xemacs)))
928     (defalias 'wl-regexp-opt 'regexp-opt)
929   (defun wl-regexp-opt (strings &optional paren)
930     "Return a regexp to match a string in STRINGS.
931 Each string should be unique in STRINGS and should not contain any regexps,
932 quoted or not.  If optional PAREN is non-nil, ensure that the returned regexp
933 is enclosed by at least one regexp grouping construct."
934     (let ((open-paren (if paren "\\(" "")) (close-paren (if paren "\\)" "")))
935       (concat open-paren (mapconcat 'regexp-quote strings "\\|")
936               close-paren))))
937
938 (defalias 'wl-expand-newtext 'elmo-expand-newtext)
939 (defalias 'wl-regexp-opt 'elmo-regexp-opt)
940
941 (defun wl-region-exists-p ()
942   "Return non-nil if a region exists on current buffer."
943   (static-if (featurep 'xemacs)
944       (region-active-p)
945     (and transient-mark-mode mark-active)))
946
947 (defun wl-deactivate-region ()
948   "Deactivate region on current buffer"
949   (static-if (not (featurep 'xemacs))
950       (setq mark-active nil)))
951
952 (defvar wl-line-string)
953 (defun wl-line-parse-format (format spec-alist)
954   "Make a formatter from FORMAT and SPEC-ALIST."
955   (let (f spec specs stack)
956     (setq f
957           (with-temp-buffer
958             (insert format)
959             (goto-char (point-min))
960             (while (search-forward "%" nil t)
961               (cond
962                ((looking-at "%")
963                 (goto-char (match-end 0)))
964                ((looking-at "\\(-?\\(0?\\)[0-9]*\\)\\([^0-9]\\)")
965                 (cond
966                  ((string= (match-string 3) "(")
967                   (if (zerop (length (match-string 1)))
968                       (error "No number specification for %%( line format"))
969                   (push (list
970                          (match-beginning 0) ; start
971                          (match-end 0)       ; start-content
972                          (string-to-number
973                           (match-string 1))  ; width
974                          specs) ; specs
975                         stack)
976                   (setq specs nil))
977                  ((string= (match-string 3) ")")
978                   (let ((entry (pop stack))
979                         form)
980                     (unless entry
981                       (error
982                        "No matching %%( parenthesis in summary line format"))
983                     (goto-char (car entry)) ; start
984                     (setq form (buffer-substring (nth 1 entry) ; start-content
985                                                  (- (match-beginning 0) 1)))
986                     (delete-region (car entry) (match-end 0))
987                     (insert "s")
988                     (setq specs
989                           (append
990                            (nth 3 entry)
991                            (list (list 'wl-set-string-width (nth 2 entry)
992                                        (append
993                                         (list 'format form)
994                                         specs)))))))
995                  (t
996                   (setq spec
997                         (if (setq spec (assq (string-to-char (match-string 3))
998                                              spec-alist))
999                             (nth 1 spec)
1000                           (match-string 3)))
1001                   (unless (string= "" (match-string 1))
1002                     (setq spec (list 'wl-set-string-width
1003                                      (string-to-number (match-string 1))
1004                                      spec
1005                                      (unless (string= "" (match-string 2))
1006                                        (string-to-char (match-string 2))))))
1007                   (replace-match "s" 'fixed)
1008                   (setq specs (append specs
1009                                       (list
1010                                        (list
1011                                         'setq 'wl-line-string
1012                                         spec)))))))))
1013             (buffer-string)))
1014     (append (list 'format f) specs)))
1015
1016 (defmacro wl-line-formatter-setup (formatter format alist)
1017   `(let (byte-compile-warnings)
1018      (setq ,formatter
1019            (byte-compile
1020             (list 'lambda ()
1021                   (wl-line-parse-format ,format ,alist))))
1022      (when (get-buffer "*Compile-Log*")
1023        (bury-buffer "*Compile-Log*"))
1024      (when (get-buffer "*Compile-Log-Show*")
1025        (bury-buffer "*Compile-Log-Show*"))))
1026
1027 (defsubst wl-copy-local-variables (src dst local-variables)
1028   "Copy value of LOCAL-VARIABLES from SRC buffer to DST buffer."
1029   (with-current-buffer dst
1030     (dolist (variable local-variables)
1031       (set (make-local-variable variable)
1032            (with-current-buffer src
1033              (symbol-value variable))))))
1034
1035 ;;; Search Condition
1036 (defun wl-search-condition-fields ()
1037   (let ((denial-fields
1038          (nconc (mapcar 'capitalize elmo-msgdb-extra-fields)
1039                 (mapcar 'capitalize wl-additional-search-condition-fields)
1040                 '("Flag" "Since" "Before"
1041                   "From" "Subject" "To" "Cc" "Body" "ToCc"
1042                   "Larger" "Smaller"))))
1043     (append '("Last" "First")
1044             denial-fields
1045             (mapcar (lambda (f) (concat "!" f))
1046                     denial-fields))))
1047
1048 (defun wl-read-search-condition (default)
1049   "Read search condition string interactively."
1050   (wl-read-search-condition-internal "Search by" default))
1051
1052 (defun wl-read-search-condition-internal (prompt default &optional paren)
1053   (let* ((completion-ignore-case t)
1054          (field (completing-read
1055                  (format "%s (%s): " prompt default)
1056                  (mapcar #'list
1057                          (append '("AND" "OR") (wl-search-condition-fields)))))
1058          value)
1059     (setq field (if (string= field "")
1060                     (setq field default)
1061                   field))
1062     (cond
1063      ((or (string= field "AND") (string= field "OR"))
1064       (concat (if paren "(" "")
1065               (wl-read-search-condition-internal
1066                (concat field "(1) Search by") default 'paren)
1067               (if (string= field "AND") "&" "|")
1068               (wl-read-search-condition-internal
1069                (concat field "(2) Search by") default 'paren)
1070               (if paren ")" "")))
1071      ((string-match "Since\\|Before" field)
1072       (let ((default (format-time-string "%Y-%m-%d")))
1073         (setq value (completing-read
1074                      (format "Value for '%s' [%s]: " field default)
1075                      (mapcar
1076                       (lambda (x)
1077                         (list (format "%s" (car x))))
1078                       elmo-date-descriptions)))
1079         (concat (downcase field) ":"
1080                 (if (equal value "") default value))))
1081      ((string-match "!?Flag" field)
1082       (while (null value)
1083         (setq value (downcase
1084                      (completing-read
1085                       (format "Value for '%s': " field)
1086                       (mapcar (lambda (f) (list (capitalize (symbol-name f))))
1087                               (elmo-uniq-list
1088                                (append
1089                                 '(unread answered forwarded digest any)
1090                                 (copy-sequence elmo-global-flags))
1091                                #'delq)))))
1092         (unless (elmo-flag-valid-p value)
1093           (message "Invalid char in `%s'" value)
1094           (setq value nil)
1095           (sit-for 1)))
1096       (unless (string-match (concat "^" elmo-condition-atom-regexp "$")
1097                             value)
1098         (setq value (prin1-to-string value)))
1099       (concat (downcase field) ":" value))
1100      (t
1101       (setq value (read-from-minibuffer (format "Value for '%s': " field)))
1102       (unless (string-match (concat "^" elmo-condition-atom-regexp "$")
1103                             value)
1104         (setq value (prin1-to-string value)))
1105       (concat (downcase field) ":" value)))))
1106
1107 (defun wl-y-or-n-p-with-scroll (prompt &optional scroll-by-SPC)
1108   (let ((prompt (concat prompt (if scroll-by-SPC
1109                                    "<y/n/SPC(down)/BS(up)> "
1110                                  "<y/n/j(down)/k(up)> "))))
1111     (catch 'done
1112       (while t
1113         (discard-input)
1114         (case (let ((cursor-in-echo-area t))
1115                 (cdr (wl-read-event-char prompt)))
1116           ((?y ?Y)
1117            (throw 'done t))
1118           ((string-to-char " ")
1119            (if scroll-by-SPC
1120                (ignore-errors (scroll-up))
1121              (throw 'done t)))
1122           ((?v ?j ?J next)
1123            (ignore-errors (scroll-up)))
1124           ((?^ ?k ?K prior backspace)
1125            (ignore-errors (scroll-down)))
1126           (t
1127            (throw 'done nil)))))))
1128
1129 (defun wl-find-region (beg-regexp end-regexp)
1130   (if (or (re-search-forward end-regexp nil t)
1131           (re-search-backward end-regexp nil t))
1132       (let ((end (match-end 0))
1133             (beg (re-search-backward beg-regexp nil t)))
1134         (if beg
1135             (cons beg end)))))
1136
1137 (defun wl-simple-display-progress (label action current total)
1138   (message "%s... %d%%"
1139            action
1140            (if (> total 0) (floor (* (/ current (float total)) 100)) 0)))
1141
1142 (when (fboundp 'progress-feedback-with-label)
1143   (defun wl-display-progress-with-gauge (label action current total)
1144     (progress-feedback-with-label
1145      label
1146      "%s..."
1147      (if (> total 0) (floor (* (/ current (float total)) 100)) 0)
1148      action)))
1149
1150 (defun wl-progress-callback-function (label action current total)
1151   (case current
1152     (query
1153      (let ((threshold (if (consp wl-display-progress-threshold)
1154                           (cdr (or (assq label wl-display-progress-threshold)
1155                                    (assq t wl-display-progress-threshold)))
1156                         wl-display-progress-threshold)))
1157        (and threshold
1158             (>= total threshold))))
1159     (start
1160      (message "%s..." action))
1161     (done
1162      (message "%s...done" action))
1163     (t
1164      (when wl-display-progress-function
1165        (funcall wl-display-progress-function label action current total)))))
1166
1167 ;; read multiple strings with completion
1168 (defun wl-completing-read-multiple-1 (prompt
1169                                       table
1170                                       &optional predicate
1171                                       require-match initial-input
1172                                       hist def inherit-input-method)
1173     "Read multiple strings in the minibuffer"
1174     (split-string
1175      (completing-read prompt table predicate nil
1176                       initial-input hist def inherit-input-method)
1177      ","))
1178
1179 (static-when (fboundp 'completing-read-multiple)
1180   (eval-when-compile
1181     (require 'crm))
1182   (defun wl-completing-read-multiple-2 (prompt
1183                                         table
1184                                         &optional predicate
1185                                         require-match initial-input
1186                                         hist def inherit-input-method)
1187     "Read multiple strings in the minibuffer"
1188     (let ((ret (completing-read-multiple prompt table predicate
1189                                          require-match initial-input
1190                                          hist def inherit-input-method)))
1191       (if (and def (equal ret '("")))
1192           (split-string def crm-separator)
1193         ret))))
1194
1195 (static-cond
1196  ((not (fboundp 'completing-read-multiple))
1197   (defalias 'wl-completing-read-multiple 'wl-completing-read-multiple-1))
1198  ((< emacs-major-version 22)
1199   (defun wl-completing-read-multiple (prompt
1200                                       table
1201                                       &optional predicate
1202                                       require-match initial-input
1203                                       hist def inherit-input-method)
1204     "Read multiple strings in the minibuffer"
1205     (if require-match
1206         (wl-completing-read-multiple-1 prompt table predicate
1207                                        nil initial-input
1208                                        hist def inherit-input-method)
1209       (wl-completing-read-multiple-2 prompt table predicate
1210                                      nil initial-input
1211                                      hist def inherit-input-method))))
1212  (t
1213   (defalias 'wl-completing-read-multiple 'completing-read-multiple)))
1214
1215
1216 (cond
1217  ((fboundp 'shell-command-read-minibuffer)
1218   (defun wl-read-shell-command (prompt &optional
1219                                        initial-contents keymap read hist)
1220     (shell-command-read-minibuffer prompt default-directory
1221                                    initial-contents keymap read hist)))
1222  (t
1223   (defalias 'wl-read-shell-command 'read-from-minibuffer)))
1224
1225 (require 'product)
1226 (product-provide (provide 'wl-util) (require 'wl-version))
1227
1228 ;;; wl-util.el ends here