Add 2010 to copyright years.
[bpt/emacs.git] / lisp / play / fortune.el
CommitLineData
6e44da43 1;;; fortune.el --- use fortune to create signatures
f111f09d 2
77f03403 3;; Copyright (C) 1999, 2001, 2002, 2003, 2004, 2005, 2006, 2007,
114f9c96 4;; 2008, 2009, 2010 Free Software Foundation, Inc.
74d93533
RS
5
6;; Author: Holger Schauer <Holger.Schauer@gmx.de>
7;; Keywords: games utils mail
8
6e44da43 9;; This file is part of GNU Emacs.
74d93533 10
b1fc2b50 11;; GNU Emacs is free software: you can redistribute it and/or modify
74d93533 12;; it under the terms of the GNU General Public License as published by
b1fc2b50
GM
13;; the Free Software Foundation, either version 3 of the License, or
14;; (at your option) any later version.
74d93533
RS
15
16;; GNU Emacs is distributed in the hope that it will be useful,
17;; but WITHOUT ANY WARRANTY; without even the implied warranty of
18;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19;; GNU General Public License for more details.
20
21;; You should have received a copy of the GNU General Public License
b1fc2b50 22;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
74d93533
RS
23
24;;; Commentary:
25;; This utility allows you to automatically cut regions to a fortune
26;; file. In case that the region stems from an article buffer (mail or
27;; news), it will try to automatically determine the author of the
28;; fortune. It will also allow you to compile your fortune-database
29;; as well as providing a function to extract a fortune for use as your
30;; signature.
31;; Of course, it can simply display a fortune, too.
32;; Use prefix arguments to specify different fortune databases.
33
34;;; Installation:
35
f111f09d
GM
36;; Please check the customize settings -- you will at least have to
37;; modify the values of `fortune-dir' and `fortune-file'.
74d93533
RS
38
39;; I then use this in my .gnus:
40;;(message "Making new signature: %s" (fortune-to-signature "~/fortunes/"))
41;; This automagically creates a new signature when starting up Gnus.
42;; Note that the call to fortune-to-signature specifies a directory in which
43;; several fortune-files and their databases are stored.
44
45;; If you like to get a new signature for every message, you can also hook
46;; it into message-mode:
f111f09d 47;; (add-hook 'message-setup-hook 'fortune-to-signature)
74d93533
RS
48;; This time no fortune-file is specified, so fortune-to-signature would use
49;; the default-file as specified by fortune-file.
50
51;; I have also this in my .gnus:
52;;(add-hook 'gnus-article-mode-hook
53;; '(lambda ()
54;; (define-key gnus-article-mode-map "i" 'fortune-from-region)))
329d7667 55;; which allows marking a region and then pressing "i" so that the marked
74d93533
RS
56;; region will be automatically added to my favourite fortune-file.
57
58;;; Code:
59
60;;; **************
61;;; Customizable Settings
62(defgroup fortune nil
63 "Settings for fortune."
f111f09d 64 :link '(emacs-commentary-link "fortune.el")
329d7667 65 :version "21.1"
74d93533
RS
66 :group 'games)
67(defgroup fortune-signature nil
68 "Settings for use of fortune for signatures."
329d7667 69 :group 'fortune
74d93533
RS
70 :group 'mail)
71
72(defcustom fortune-dir "~/docs/ascii/misc/fortunes/"
77f03403 73 "The directory to look in for local fortune cookies files."
f111f09d
GM
74 :type 'directory
75 :group 'fortune)
76(defcustom fortune-file
77 (expand-file-name "usenet" fortune-dir)
77f03403 78 "The file in which local fortune cookies will be stored."
f111f09d
GM
79 :type 'file
80 :group 'fortune)
74d93533 81(defcustom fortune-database-extension ".dat"
329d7667 82 "The extension of the corresponding fortune database.
74d93533 83Normally you won't have a reason to change it."
f111f09d
GM
84 :type 'string
85 :group 'fortune)
74d93533
RS
86(defcustom fortune-program "fortune"
87 "Program to select a fortune cookie."
f111f09d
GM
88 :type 'string
89 :group 'fortune)
ca514f72 90(defcustom fortune-program-options ()
77f03403
GM
91 "List of options to pass to the fortune program."
92 :type '(choice (repeat (string :tag "Option"))
93 (string :tag "Obsolete string of options"))
94 :version "23.1"
f111f09d 95 :group 'fortune)
74d93533
RS
96(defcustom fortune-strfile "strfile"
97 "Program to compute a new fortune database."
f111f09d
GM
98 :type 'string
99 :group 'fortune)
74d93533 100(defcustom fortune-strfile-options ""
f111f09d
GM
101 "Options to pass to the strfile program (a string)."
102 :type 'string
103 :group 'fortune)
74d93533
RS
104(defcustom fortune-quiet-strfile-options "> /dev/null"
105 "Text added to the command for running `strfile'.
106By default it discards the output produced by `strfile'.
107Set this to \"\" if you would like to see the output."
f111f09d
GM
108 :type 'string
109 :group 'fortune)
74d93533
RS
110
111(defcustom fortune-always-compile t
77f03403 112 "Non-nil means automatically compile fortune files.
74d93533 113If nil, you must invoke `fortune-compile' manually to do that."
f111f09d
GM
114 :type 'boolean
115 :group 'fortune)
74d93533
RS
116(defcustom fortune-author-line-prefix " -- "
117 "Prefix to put before the author name of a fortunate."
f111f09d
GM
118 :type 'string
119 :group 'fortune-signature)
74d93533
RS
120(defcustom fortune-fill-column fill-column
121 "Fill column for fortune files."
f111f09d
GM
122 :type 'integer
123 :group 'fortune-signature)
74d93533
RS
124(defcustom fortune-from-mail "private e-mail"
125 "String to use to characterize that the fortune comes from an e-mail.
126No need to add an `in'."
127 :type 'string
128 :group 'fortune-signature)
129(defcustom fortune-sigstart ""
77f03403 130 "Some text to insert before the fortune cookie, in a mail signature."
f111f09d
GM
131 :type 'string
132 :group 'fortune-signature)
74d93533 133(defcustom fortune-sigend ""
77f03403 134 "Some text to insert after the fortune cookie, in a mail signature."
f111f09d
GM
135 :type 'string
136 :group 'fortune-signature)
74d93533
RS
137
138
139;; not customizable settings
140(defvar fortune-buffer-name "*fortune*")
329d7667 141(defconst fortune-end-sep "\n%\n")
74d93533
RS
142
143
144;;; **************
145;;; Inserting a new fortune
146(defun fortune-append (string &optional interactive file)
329d7667 147 "Appends STRING to the fortune FILE.
74d93533
RS
148
149If INTERACTIVE is non-nil, don't compile the fortune file afterwards."
329d7667 150 (setq file (expand-file-name
74d93533
RS
151 (substitute-in-file-name (or file fortune-file))))
152 (if (file-directory-p file)
329d7667 153 (error "Cannot append fortune to directory %s" file))
74d93533
RS
154 (if interactive ; switch to file and return buffer
155 (find-file-other-frame file)
156 (find-file-noselect file))
157 (let ((fortune-buffer (get-file-buffer file)))
158
159 (set-buffer fortune-buffer)
160 (goto-char (point-max))
161 (setq fill-column fortune-fill-column)
162 (setq auto-fill-inhibit-regexp "^%")
163 (turn-on-auto-fill)
164 (insert string fortune-end-sep)
165 (unless interactive
166 (save-buffer)
167 (if fortune-always-compile
168 (fortune-compile file)))))
169
170(defun fortune-ask-file ()
171 "Asks the user for a file-name."
329d7667 172 (expand-file-name
74d93533
RS
173 (read-file-name
174 "Fortune file to use: "
175 fortune-dir nil nil "")))
176
23b809c2 177;;;###autoload
74d93533
RS
178(defun fortune-add-fortune (string file)
179 "Add STRING to a fortune file FILE.
180
181Interactively, if called with a prefix argument,
182read the file name to use. Otherwise use the value of `fortune-file'."
183 (interactive
184 (list (read-string "Fortune: ")
185 (if current-prefix-arg (fortune-ask-file))))
186 (fortune-append string t file))
187
23b809c2 188;;;###autoload
74d93533 189(defun fortune-from-region (beg end file)
329d7667 190 "Append the current region to a local fortune-like data file.
74d93533
RS
191
192Interactively, if called with a prefix argument,
193read the file name to use. Otherwise use the value of `fortune-file'."
329d7667 194 (interactive
74d93533
RS
195 (list (region-beginning) (region-end)
196 (if current-prefix-arg (fortune-ask-file))))
197 (let ((string (buffer-substring beg end))
198 author newsgroup help-point)
199 ;; try to determine author ...
200 (save-excursion
201 (goto-char (point-min))
329d7667 202 (setq help-point
74d93533
RS
203 (search-forward-regexp
204 "^From: \\(.*\\)$"
205 (point-max) t))
329d7667
DL
206 (if help-point
207 (setq author (buffer-substring (match-beginning 1) help-point))
74d93533
RS
208 (setq author "An unknown author")))
209 ;; ... and newsgroup
210 (save-excursion
211 (goto-char (point-min))
212 (setq help-point
213 (search-forward-regexp
214 "^Newsgroups: \\(.*\\)$"
215 (point-max) t))
329d7667 216 (if help-point
74d93533 217 (setq newsgroup (buffer-substring (match-beginning 1) help-point))
1d14e80c
GM
218 (setq newsgroup (if (or (eq major-mode 'gnus-article-mode)
219 (eq major-mode 'vm-mode)
220 (eq major-mode 'rmail-mode))
74d93533
RS
221 fortune-from-mail
222 "unknown"))))
223
224 ;; append entry to end of fortune file, and display result
225 (setq string (concat "\"" string "\""
226 "\n"
227 fortune-author-line-prefix
228 author " in " newsgroup))
229 (fortune-append string t file)))
230
231
232;;; **************
233;;; Compile new database with strfile
23b809c2 234;;;###autoload
74d93533
RS
235(defun fortune-compile (&optional file)
236 "Compile fortune file.
237
238If called with a prefix asks for the FILE to compile, otherwise uses
239the value of `fortune-file'. This currently cannot handle directories."
329d7667 240 (interactive
74d93533
RS
241 (list
242 (if current-prefix-arg
243 (fortune-ask-file)
244 fortune-file)))
245 (let* ((fortune-file (expand-file-name (substitute-in-file-name file)))
329d7667 246 (fortune-dat (expand-file-name
74d93533
RS
247 (substitute-in-file-name
248 (concat fortune-file fortune-database-extension)))))
249 (cond ((file-exists-p fortune-file)
250 (if (file-exists-p fortune-dat)
251 (cond ((file-newer-than-file-p fortune-file fortune-dat)
252 (message "Compiling new fortune database %s" fortune-dat)
329d7667 253 (shell-command
74d93533
RS
254 (concat fortune-strfile fortune-strfile-options
255 " " fortune-file fortune-quiet-strfile-options))))))
256 (t (error "Can't compile fortune file %s" fortune-file)))))
a1506d29
JB
257
258
74d93533
RS
259;;; **************
260;;; Use fortune for signature
23b809c2 261;;;###autoload
74d93533
RS
262(defun fortune-to-signature (&optional file)
263 "Create signature from output of the fortune program.
264
265If called with a prefix asks for the FILE to choose the fortune from,
266otherwise uses the value of `fortune-file'. If you want to have fortune
267choose from a set of files in a directory, call interactively with prefix
268and choose the directory as the fortune-file."
329d7667 269 (interactive
74d93533
RS
270 (list
271 (if current-prefix-arg
272 (fortune-ask-file)
273 fortune-file)))
274 (save-excursion
1d4408c8 275 (fortune-in-buffer t file)
74d93533
RS
276 (set-buffer fortune-buffer-name)
277 (let* ((fortune (buffer-string))
278 (signature (concat fortune-sigstart fortune fortune-sigend)))
279 (setq mail-signature signature)
280 (if (boundp 'message-signature)
281 (setq message-signature signature)))))
282
283
284;;; **************
285;;; Display fortune
286(defun fortune-in-buffer (interactive &optional file)
287 "Put a fortune cookie in the *fortune* buffer.
288
1d4408c8 289INTERACTIVE is ignored. Optional argument FILE,
74d93533
RS
290when supplied, specifies the file to choose the fortune from."
291 (let ((fortune-buffer (or (get-buffer fortune-buffer-name)
292 (generate-new-buffer fortune-buffer-name)))
293 (fort-file (expand-file-name
294 (substitute-in-file-name
295 (or file fortune-file)))))
937e6a56 296 (with-current-buffer fortune-buffer
74d93533
RS
297 (toggle-read-only 0)
298 (erase-buffer)
299
300 (if fortune-always-compile
301 (fortune-compile fort-file))
302
ca514f72 303 (apply 'call-process
77f03403
GM
304 fortune-program ; program to call
305 nil fortune-buffer nil ; INFILE BUFFER DISPLAY
306 (append (if (stringp fortune-program-options)
307 (split-string fortune-program-options)
b45c803b 308 fortune-program-options) (list fort-file))))))
74d93533 309
23b809c2 310;;;###autoload
74d93533
RS
311(defun fortune (&optional file)
312 "Display a fortune cookie.
313
314If called with a prefix asks for the FILE to choose the fortune from,
315otherwise uses the value of `fortune-file'. If you want to have fortune
316choose from a set of files in a directory, call interactively with prefix
317and choose the directory as the fortune-file."
329d7667 318 (interactive
74d93533
RS
319 (list
320 (if current-prefix-arg
321 (fortune-ask-file)
322 fortune-file)))
323 (fortune-in-buffer t file)
324 (switch-to-buffer (get-buffer fortune-buffer-name))
325 (toggle-read-only 1))
326
327
328;;; Provide ourselves.
329(provide 'fortune)
330
cbee283d 331;; arch-tag: a1e4cb8a-3792-40e7-86a7-fc75ce094bcc
74d93533 332;;; fortune.el ends here