1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
1579
1580
1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
1612
1613
1614
1615
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
1681
1682
1683
1684
1685
1686
1687
1688
1689
1690
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713
1714
1715
1716
1717
1718
1719
1720
1721
1722
1723
1724
1725
1726
1727
1728
1729
1730
1731
1732
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
1786
1787
1788
1789
1790
1791
1792
1793
1794
1795
1796
1797
1798
1799
1800
1801
1802
1803
1804
1805
1806
1807
1808
1809
1810
1811
1812
1813
1814
1815
1816
1817
1818
1819
1820
1821
1822
1823
1824
1825
1826
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
1837
1838
1839
1840
1841
1842
1843
1844
1845
1846
1847
1848
1849
1850
1851
1852
1853
1854
1855
1856
1857
1858
1859
1860
1861
1862
1863
1864
1865
1866
1867
1868
1869
1870
1871
1872
1873
1874
1875
1876
1877
1878
1879
1880
1881
1882
1883
1884
1885
1886
1887
1888
1889
1890
1891
1892
1893
1894
1895
1896
1897
1898
1899
1900
1901
1902
1903
1904
1905
1906
1907
1908
1909
1910
1911
1912
1913
1914
1915
1916
1917
1918
1919
1920
1921
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931
1932
1933
1934
1935
1936
1937
1938
1939
1940
1941
1942
1943
1944
1945
1946
1947
1948
1949
1950
1951
1952
1953
1954
1955
1956
1957
1958
1959
1960
1961
1962
1963
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
2027
2028
2029
2030
2031
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
2042
2043
2044
2045
2046
2047
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
2061
2062
2063
2064
2065
2066
2067
2068
2069
2070
2071
2072
2073
2074
2075
2076
2077
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
2091
2092
2093
2094
2095
2096
2097
2098
2099
2100
2101
2102
2103
2104
2105
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
2119
2120
2121
2122
2123
2124
2125
2126
2127
2128
2129
2130
2131
2132
2133
2134
2135
2136
2137
2138
2139
2140
2141
2142
2143
2144
2145
2146
2147
2148
2149
2150
2151
2152
2153
2154
2155
2156
2157
2158
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
2172
2173
2174
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
2190
2191
2192
2193
2194
2195
2196
2197
2198
2199
2200
2201
2202
2203
2204
2205
2206
2207
2208
2209
2210
2211
2212
2213
2214
2215
2216
2217
2218
2219
2220
2221
2222
2223
2224
2225
2226
2227
2228
2229
2230
2231
2232
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
2244
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
2255
2256
2257
2258
2259
2260
2261
2262
2263
2264
2265
2266
2267
2268
2269
2270
2271
2272
2273
2274
2275
2276
2277
2278
2279
2280
2281
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
2292
2293
2294
2295
2296
2297
2298
2299
2300
2301
2302
2303
2304
2305
2306
2307
2308
2309
2310
2311
2312
2313
2314
2315
2316
2317
2318
2319
2320
2321
2322
2323
2324
2325
2326
2327
2328
2329
2330
2331
2332
2333
2334
2335
2336
2337
2338
2339
2340
2341
2342
2343
2344
2345
2346
2347
2348
2349
2350
2351
2352
2353
2354
2355
2356
2357
2358
2359
2360
2361
2362
2363
2364
2365
2366
2367
2368
2369
2370
2371
2372
2373
2374
2375
2376
2377
2378
2379
2380
2381
2382
2383
2384
2385
2386
2387
2388
2389
2390
2391
2392
2393
2394
2395
2396
2397
2398
2399
2400
2401
2402
2403
2404
2405
2406
2407
2408
2409
2410
2411
2412
2413
2414
2415
2416
2417
2418
2419
2420
2421
2422
2423
2424
2425
2426
2427
2428
2429
2430
2431
2432
2433
2434
2435
2436
2437
2438
2439
2440
2441
2442
2443
2444
2445
2446
2447
2448
2449
2450
2451
2452
2453
2454
2455
2456
2457
2458
2459
2460
2461
2462
2463
2464
2465
2466
2467
2468
2469
2470
2471
2472
2473
2474
2475
2476
2477
2478
2479
2480
2481
2482
2483
2484
2485
2486
2487
2488
2489
2490
2491
2492
2493
2494
2495
2496
2497
2498
2499
2500
2501
2502
2503
2504
2505
2506
2507
2508
2509
2510
2511
2512
2513
2514
2515
2516
2517
2518
2519
2520
2521
2522
2523
2524
2525
2526
2527
2528
2529
2530
2531
2532
2533
2534
2535
2536
2537
2538
2539
2540
2541
2542
2543
2544
2545
2546
2547
2548
2549
2550
2551
2552
2553
2554
2555
2556
2557
2558
2559
2560
2561
2562
2563
2564
2565
2566
2567
2568
2569
2570
2571
2572
2573
2574
2575
2576
2577
2578
2579
2580
2581
2582
2583
2584
2585
2586
2587
2588
2589
2590
2591
2592
2593
2594
2595
2596
2597
2598
2599
2600
2601
2602
2603
2604
2605
2606
2607
2608
2609
2610
2611
2612
2613
2614
2615
2616
2617
2618
2619
2620
2621
2622
2623
2624
2625
2626
2627
2628
2629
2630
2631
2632
2633
2634
2635
2636
2637
2638
2639
2640
2641
2642
2643
2644
2645
2646
2647
2648
2649
2650
2651
2652
2653
2654
2655
2656
2657
|
;;; rmail.el --- main code of "RMAIL" mail reader for Emacs.
;; Copyright (C) 1985,86,87,88,93,94,95 Free Software Foundation, Inc.
;; Maintainer: FSF
;; Keywords: mail
;; 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 2, 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; see the file COPYING. If not, write to the
;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
;; Boston, MA 02111-1307, USA.
;;; Code:
;; Souped up by shane@mit-ajax based on ideas of rlk@athena.mit.edu
;; New features include attribute and keyword support, message
;; selection by dispatch table, summary by attributes and keywords,
;; expunging by dispatch table, sticky options for file commands.
;; Extended by Bob Weiner of Motorola
;; New features include: rmail and rmail-summary buffers remain
;; synchronized and key bindings basically operate the same way in both
;; buffers, summary by topic or by regular expression, rmail-reply-prefix
;; variable, and a bury rmail buffer (wipe) command.
;;
(require 'mail-utils)
;; For Emacs V18 compatibility
(and (not (fboundp 'buffer-disable-undo))
(fboundp 'buffer-flush-undo)
(defalias 'buffer-disable-undo 'buffer-flush-undo))
; These variables now declared in paths.el.
;(defvar rmail-spool-directory "/usr/spool/mail/"
; "This is the name of the directory used by the system mailer for\n\
;delivering new mail. It's name should end with a slash.")
;(defvar rmail-file-name
; (expand-file-name "~/RMAIL")
; "")
(defvar rmail-movemail-program nil
"If non-nil, name of program for fetching new mail.")
;;;###autoload
(defvar rmail-dont-reply-to-names nil "\
*A regexp specifying names to prune of reply to messages.
A value of nil means exclude your own name only.")
;;;###autoload
(defvar rmail-default-dont-reply-to-names "info-" "\
A regular expression specifying part of the value of the default value of
the variable `rmail-dont-reply-to-names', for when the user does not set
`rmail-dont-reply-to-names' explicitly. (The other part of the default
value is the user's name.)
It is useful to set this variable in the site customization file.")
;;;###autoload
(defvar rmail-ignored-headers "^via:\\|^mail-from:\\|^origin:\\|^status:\\|^received:\\|^x400-originator:\\|^x400-recipients:\\|^x400-received:\\|^x400-mts-identifier:\\|^x400-content-type:\\|^\\(resent-\\|\\)message-id:\\|^summary-line:\\|^resent-date:\\|^nntp-posting-host:"
"*Regexp to match Header fields that Rmail should normally hide.")
;;;###autoload
(defvar rmail-displayed-headers nil
"*Regexp to match Header fields that Rmail should display.
If nil, display all header fields except those matched by
`rmail-ignored-headers'.")
;;;###autoload
(defvar rmail-retry-ignored-headers nil "\
*Headers that should be stripped when retrying a failed message.")
;;;###autoload
(defvar rmail-highlighted-headers "^From:\\|^Subject:" "\
*Regexp to match Header fields that Rmail should normally highlight.
A value of nil means don't highlight.
See also `rmail-highlight-face'.")
;;;###autoload
(defvar rmail-highlight-face nil "\
*Face used by Rmail for highlighting headers.")
;;;###autoload
(defvar rmail-delete-after-output nil "\
*Non-nil means automatically delete a message that is copied to a file.")
;;;###autoload
(defvar rmail-primary-inbox-list nil "\
*List of files which are inboxes for user's primary mail file `~/RMAIL'.
`nil' means the default, which is (\"/usr/spool/mail/$USER\")
\(the name varies depending on the operating system,
and the value of the environment variable MAIL overrides it).")
;;;###autoload
(defvar rmail-mail-new-frame nil
"*Non-nil means Rmail makes a new frame for composing outgoing mail.")
;;;###autoload
(defvar rmail-secondary-file-directory "~/"
"*Directory for additional secondary Rmail files.")
;;;###autoload
(defvar rmail-secondary-file-regexp "\\.xmail$"
"*Regexp for which files are secondary Rmail files.")
;;;###autoload
(defvar rmail-mode-hook nil
"List of functions to call when Rmail is invoked.")
;;;###autoload
(defvar rmail-get-new-mail-hook nil
"List of functions to call when Rmail has retrieved new mail.")
;;;###autoload
(defvar rmail-show-message-hook nil
"List of functions to call when Rmail displays a message.")
;;;###autoload
(defvar rmail-delete-message-hook nil
"List of functions to call when Rmail deletes a message.
When the hooks are called, the message has been marked deleted but is
still the current message in the Rmail buffer.")
;; These may be altered by site-init.el to match the format of mmdf files
;; delimiting used on a given host (delim1 and delim2 from the config
;; files).
(defvar mmdf-delim1 "^\001\001\001\001\n"
"Regexp marking the start of an mmdf message")
(defvar mmdf-delim2 "^\001\001\001\001\n"
"Regexp marking the end of an mmdf message")
(defvar rmail-message-filter nil
"If non nil, a filter function for new messages in RMAIL.
Called with region narrowed to the message, including headers.")
(defvar rmail-reply-prefix "Re: "
"String to prepend to Subject line when replying to a message.")
;; Some mailers use "Re(2):" or "Re^2:" or "Re: Re:".
;; This pattern should catch all the common variants.
(defvar rmail-reply-regexp "\\`\\(Re\\(([0-9]+)\\|\\^[0-9]+\\)?: *\\)*"
"Regexp to delete from Subject line before inserting rmail-reply-prefix.")
(defvar rmail-display-summary nil
"If non nil, the summary buffer is always displayed.")
(defvar rmail-mode-map nil)
(defvar rmail-inbox-list nil)
(defvar rmail-keywords nil)
;; Message counters and markers. Deleted flags.
(defvar rmail-current-message nil)
(defvar rmail-total-messages nil)
(defvar rmail-message-vector nil)
(defvar rmail-deleted-vector nil)
(defvar rmail-overlay-list nil)
(defvar rmail-font-lock-keywords
(eval-when-compile
(let* ((cite-prefix "A-Za-z") (cite-suffix (concat cite-prefix "0-9_.@-")))
(list
'("^\\(From\\|Sender\\):" . font-lock-function-name-face)
'("^Reply-To:.*$" . font-lock-function-name-face)
'("^Subject:" . font-lock-comment-face)
'("^\\(To\\|Apparently-To\\|Cc\\):" . font-lock-keyword-face)
(cons (concat "^[ \t]*"
"\\([" cite-prefix "]+[" cite-suffix "]*\\)?"
"[>|}].*")
'font-lock-reference-face)
'("^\\(X-[A-Za-z0-9-]+\\|In-reply-to\\|Date\\):.*$"
. font-lock-string-face))))
"Additional expressions to highlight in Rmail mode.")
;; These are used by autoloaded rmail-summary.
(defvar rmail-summary-buffer nil)
(defvar rmail-summary-vector nil)
;; `Sticky' default variables.
;; Last individual label specified to a or k.
(defvar rmail-last-label nil)
;; Last set of values specified to C-M-n, C-M-p, C-M-s or C-M-l.
(defvar rmail-last-multi-labels nil)
(defvar rmail-last-regexp nil)
(defvar rmail-default-file "~/xmail"
"*Default file name for \\[rmail-output].")
(defvar rmail-default-rmail-file "~/XMAIL"
"*Default file name for \\[rmail-output-to-rmail-file].")
;;; Regexp matching the delimiter of messages in UNIX mail format
;;; (UNIX From lines), minus the initial ^. Note that if you change
;;; this expression, you must change the code in rmail-nuke-pinhead-header
;;; that knows the exact ordering of the \\( \\) subexpressions.
(defvar rmail-unix-mail-delimiter
(let ((time-zone-regexp
(concat "\\([A-Z]?[A-Z]?[A-Z][A-Z]\\( DST\\)?"
"\\|[-+]?[0-9][0-9][0-9][0-9]"
"\\|"
"\\) *")))
(concat
"From "
;; Username, perhaps with a quoted section that can contain spaces.
"\\("
"[^ \n]*"
"\\(\\|\".*\"[^ \n]*\\)"
"\\|<[^<>\n]+>"
"\\) ?"
;; The time the message was sent.
"\\([^ \n]*\\) *" ; day of the week
"\\([^ ]*\\) *" ; month
"\\([0-9]*\\) *" ; day of month
"\\([0-9:]*\\) *" ; time of day
;; Perhaps a time zone, specified by an abbreviation, or by a
;; numeric offset.
time-zone-regexp
;; The year.
" [0-9][0-9]\\([0-9]*\\) *"
;; On some systems the time zone can appear after the year, too.
time-zone-regexp
;; Old uucp cruft.
"\\(remote from .*\\)?"
"\n"))
nil)
;; Perform BODY in the summary buffer
;; in such a way that its cursor is properly updated in its own window.
(defmacro rmail-select-summary (&rest body)
(` (let ((total rmail-total-messages))
(if (rmail-summary-displayed)
(let ((window (selected-window)))
(save-excursion
(unwind-protect
(progn
(pop-to-buffer rmail-summary-buffer)
;; rmail-total-messages is a buffer-local var
;; in the rmail buffer.
;; This way we make it available for the body
;; even tho the rmail buffer is not current.
(let ((rmail-total-messages total))
(,@ body)))
(select-window window))))
(save-excursion
(set-buffer rmail-summary-buffer)
(let ((rmail-total-messages total))
(,@ body))))
(rmail-maybe-display-summary))))
;;;; *** Rmail Mode ***
;;;###autoload
(defun rmail (&optional file-name-arg)
"Read and edit incoming mail.
Moves messages into file named by `rmail-file-name' (a babyl format file)
and edits that file in RMAIL Mode.
Type \\[describe-mode] once editing that file, for a list of RMAIL commands.
May be called with file name as argument; then performs rmail editing on
that file, but does not copy any new mail into the file.
Interactively, if you supply a prefix argument, then you
have a chance to specify a file name with the minibuffer.
If `rmail-display-summary' is non-nil, make a summary for this RMAIL file."
(interactive (if current-prefix-arg
(list (read-file-name "Run rmail on RMAIL file: "))))
(let* ((file-name (expand-file-name (or file-name-arg rmail-file-name)))
(existed (get-file-buffer file-name)))
;; Like find-file, but in the case where a buffer existed
;; and the file was reverted, recompute the message-data.
(if (and existed (not (verify-visited-file-modtime existed)))
(progn
;; Don't be confused by apparent local-variables spec
;; in the last message in the RMAIL file.
(let ((enable-local-variables nil))
(find-file file-name))
(if (and (verify-visited-file-modtime existed)
(eq major-mode 'rmail-mode))
(progn (rmail-forget-messages)
(rmail-set-message-counters))))
(let ((enable-local-variables nil))
(find-file file-name)))
(if (eq major-mode 'rmail-edit-mode)
(error "Exit Rmail Edit mode before getting new mail."))
(if (and existed (> (buffer-size) 0))
;; Buffer not new and not empty; ensure in proper mode, but that's all.
(or (eq major-mode 'rmail-mode)
(rmail-mode-2))
(rmail-mode-2)
;; Convert all or part to Babyl file if possible.
(rmail-convert-file)
(goto-char (point-max))
(if (null rmail-inbox-list)
(progn
(rmail-set-message-counters)
(rmail-show-message))))
(or (and (null file-name-arg)
(rmail-get-new-mail))
(rmail-show-message (rmail-first-unseen-message)))
(if rmail-display-summary (rmail-summary))
(rmail-construct-io-menu)))
;; Given the value of MAILPATH, return a list of inbox file names.
;; This is turned off because it is not clear that the user wants
;; all these inboxes to feed into the primary rmail file.
; (defun rmail-convert-mailpath (string)
; (let (idx list)
; (while (setq idx (string-match "[%:]" string))
; (let ((this (substring string 0 idx)))
; (setq string (substring string (1+ idx)))
; (setq list (cons (if (string-match "%" this)
; (substring this 0 (string-match "%" this))
; this)
; list))))
; list))
; I have checked that adding "-*- rmail -*-" to the BABYL OPTIONS line
; will not cause emacs 18.55 problems.
(defun rmail-convert-file ()
(let (convert)
(widen)
(goto-char (point-min))
;; If file doesn't start like a Babyl file,
;; convert it to one, by adding a header and converting each message.
(cond ((looking-at "BABYL OPTIONS:"))
((looking-at "Version: 5\n")
;; Losing babyl file made by old version of Rmail.
;; Just fix the babyl file header; don't make a new one,
;; so we don't lose the Labels: file attribute, etc.
(let ((buffer-read-only nil))
(insert "BABYL OPTIONS: -*- rmail -*-\n")))
((equal (point-min) (point-max))
;; Empty RMAIL file. Just insert the header.
(rmail-insert-rmail-file-header))
(t
;; Non-empty file in non-RMAIL format. Add header and convert.
(setq convert t)
(rmail-insert-rmail-file-header)))
;; If file was not a Babyl file or if there are
;; Unix format messages added at the end,
;; convert file as necessary.
(if (or convert
(save-excursion
(goto-char (point-max))
(search-backward "\n\^_")
(forward-char 2)
(looking-at "\n*From ")))
(let ((buffer-read-only nil))
(message "Converting to Babyl format...")
;; If file needs conversion, convert it all,
;; except for the BABYL header.
;; (rmail-convert-to-babyl-format would delete the header.)
(goto-char (point-min))
(search-forward "\n\^_" nil t)
(narrow-to-region (point) (point-max))
(rmail-convert-to-babyl-format)
(message "Converting to Babyl format...done")))))
;;; I have checked that adding "-*- rmail -*-" to the BABYL OPTIONS line
;;; will not cause emacs 18.55 problems.
(defun rmail-insert-rmail-file-header ()
(let ((buffer-read-only nil))
(insert "BABYL OPTIONS: -*- rmail -*-
Version: 5
Labels:
Note: This is the header of an rmail file.
Note: If you are seeing it in rmail,
Note: it means the file has no messages in it.\n\^_")))
(if rmail-mode-map
nil
(setq rmail-mode-map (make-keymap))
(suppress-keymap rmail-mode-map)
(define-key rmail-mode-map "a" 'rmail-add-label)
(define-key rmail-mode-map "b" 'rmail-bury)
(define-key rmail-mode-map "c" 'rmail-continue)
(define-key rmail-mode-map "d" 'rmail-delete-forward)
(define-key rmail-mode-map "\C-d" 'rmail-delete-backward)
(define-key rmail-mode-map "e" 'rmail-edit-current-message)
(define-key rmail-mode-map "f" 'rmail-forward)
(define-key rmail-mode-map "g" 'rmail-get-new-mail)
(define-key rmail-mode-map "h" 'rmail-summary)
(define-key rmail-mode-map "i" 'rmail-input)
(define-key rmail-mode-map "j" 'rmail-show-message)
(define-key rmail-mode-map "k" 'rmail-kill-label)
(define-key rmail-mode-map "l" 'rmail-summary-by-labels)
(define-key rmail-mode-map "\e\C-h" 'rmail-summary)
(define-key rmail-mode-map "\e\C-l" 'rmail-summary-by-labels)
(define-key rmail-mode-map "\e\C-r" 'rmail-summary-by-recipients)
(define-key rmail-mode-map "\e\C-s" 'rmail-summary-by-regexp)
(define-key rmail-mode-map "\e\C-t" 'rmail-summary-by-topic)
(define-key rmail-mode-map "m" 'rmail-mail)
(define-key rmail-mode-map "\em" 'rmail-retry-failure)
(define-key rmail-mode-map "n" 'rmail-next-undeleted-message)
(define-key rmail-mode-map "\en" 'rmail-next-message)
(define-key rmail-mode-map "\e\C-n" 'rmail-next-labeled-message)
(define-key rmail-mode-map "o" 'rmail-output-to-rmail-file)
(define-key rmail-mode-map "\C-o" 'rmail-output)
(define-key rmail-mode-map "p" 'rmail-previous-undeleted-message)
(define-key rmail-mode-map "\ep" 'rmail-previous-message)
(define-key rmail-mode-map "\e\C-p" 'rmail-previous-labeled-message)
(define-key rmail-mode-map "q" 'rmail-quit)
(define-key rmail-mode-map "r" 'rmail-reply)
;; I find I can't live without the default M-r command -- rms.
;; (define-key rmail-mode-map "\er" 'rmail-search-backwards)
(define-key rmail-mode-map "s" 'rmail-expunge-and-save)
(define-key rmail-mode-map "\es" 'rmail-search)
(define-key rmail-mode-map "t" 'rmail-toggle-header)
(define-key rmail-mode-map "u" 'rmail-undelete-previous-message)
(define-key rmail-mode-map "w" 'rmail-edit-current-message)
(define-key rmail-mode-map "x" 'rmail-expunge)
(define-key rmail-mode-map "." 'rmail-beginning-of-message)
(define-key rmail-mode-map "<" 'rmail-first-message)
(define-key rmail-mode-map ">" 'rmail-last-message)
(define-key rmail-mode-map " " 'scroll-up)
(define-key rmail-mode-map "\177" 'scroll-down)
(define-key rmail-mode-map "?" 'describe-mode)
(define-key rmail-mode-map "\C-c\C-s\C-d" 'rmail-sort-by-date)
(define-key rmail-mode-map "\C-c\C-s\C-s" 'rmail-sort-by-subject)
(define-key rmail-mode-map "\C-c\C-s\C-a" 'rmail-sort-by-author)
(define-key rmail-mode-map "\C-c\C-s\C-r" 'rmail-sort-by-recipient)
(define-key rmail-mode-map "\C-c\C-s\C-c" 'rmail-sort-by-correspondent)
(define-key rmail-mode-map "\C-c\C-s\C-l" 'rmail-sort-by-lines)
(define-key rmail-mode-map "\C-c\C-s\C-k" 'rmail-sort-by-keywords)
(define-key rmail-mode-map "\C-c\C-n" 'rmail-next-same-subject)
(define-key rmail-mode-map "\C-c\C-p" 'rmail-previous-same-subject)
)
(define-key rmail-mode-map [menu-bar] (make-sparse-keymap))
(define-key rmail-mode-map [menu-bar classify]
(cons "Classify" (make-sparse-keymap "Classify")))
(define-key rmail-mode-map [menu-bar classify input-menu]
nil)
(define-key rmail-mode-map [menu-bar classify output-menu]
nil)
(define-key rmail-mode-map [menu-bar classify output-inbox]
'("Output (inbox)..." . rmail-output))
(define-key rmail-mode-map [menu-bar classify output]
'("Output (Rmail)..." . rmail-output-to-rmail-file))
(define-key rmail-mode-map [menu-bar classify kill-label]
'("Kill Label..." . rmail-kill-label))
(define-key rmail-mode-map [menu-bar classify add-label]
'("Add Label..." . rmail-add-label))
(define-key rmail-mode-map [menu-bar summary]
(cons "Summary" (make-sparse-keymap "Summary")))
(define-key rmail-mode-map [menu-bar summary senders]
'("By Senders..." . rmail-summary-by-senders))
(define-key rmail-mode-map [menu-bar summary labels]
'("By Labels..." . rmail-summary-by-labels))
(define-key rmail-mode-map [menu-bar summary recipients]
'("By Recipients..." . rmail-summary-by-recipients))
(define-key rmail-mode-map [menu-bar summary topic]
'("By Topic..." . rmail-summary-by-topic))
(define-key rmail-mode-map [menu-bar summary regexp]
'("By Regexp..." . rmail-summary-by-regexp))
(define-key rmail-mode-map [menu-bar summary all]
'("All" . rmail-summary))
(define-key rmail-mode-map [menu-bar mail]
(cons "Mail" (make-sparse-keymap "Mail")))
(define-key rmail-mode-map [menu-bar mail rmail-get-new-mail]
'("Get New Mail" . rmail-get-new-mail))
(define-key rmail-mode-map [menu-bar mail lambda]
'("----"))
(define-key rmail-mode-map [menu-bar mail continue]
'("Continue" . rmail-continue))
(define-key rmail-mode-map [menu-bar mail resend]
'("Re-send..." . rmail-resend))
(define-key rmail-mode-map [menu-bar mail forward]
'("Forward" . rmail-forward))
(define-key rmail-mode-map [menu-bar mail retry]
'("Retry" . rmail-retry-failure))
(define-key rmail-mode-map [menu-bar mail reply]
'("Reply" . rmail-reply))
(define-key rmail-mode-map [menu-bar mail mail]
'("Mail" . rmail-mail))
(define-key rmail-mode-map [menu-bar delete]
(cons "Delete" (make-sparse-keymap "Delete")))
(define-key rmail-mode-map [menu-bar delete expunge/save]
'("Expunge/Save" . rmail-expunge-and-save))
(define-key rmail-mode-map [menu-bar delete expunge]
'("Expunge" . rmail-expunge))
(define-key rmail-mode-map [menu-bar delete undelete]
'("Undelete" . rmail-undelete-previous-message))
(define-key rmail-mode-map [menu-bar delete delete]
'("Delete" . rmail-delete-forward))
(define-key rmail-mode-map [menu-bar move]
(cons "Move" (make-sparse-keymap "Move")))
(define-key rmail-mode-map [menu-bar move search-back]
'("Search Back..." . rmail-search-backwards))
(define-key rmail-mode-map [menu-bar move search]
'("Search..." . rmail-search))
(define-key rmail-mode-map [menu-bar move previous]
'("Previous Nondeleted" . rmail-previous-undeleted-message))
(define-key rmail-mode-map [menu-bar move next]
'("Next Nondeleted" . rmail-next-undeleted-message))
(define-key rmail-mode-map [menu-bar move last]
'("Last" . rmail-last-message))
(define-key rmail-mode-map [menu-bar move first]
'("First" . rmail-first-message))
(define-key rmail-mode-map [menu-bar move previous]
'("Previous" . rmail-previous-message))
(define-key rmail-mode-map [menu-bar move next]
'("Next" . rmail-next-message))
;; Rmail mode is suitable only for specially formatted data.
(put 'rmail-mode 'mode-class 'special)
(defun rmail-mode-kill-summary ()
(if rmail-summary-buffer (kill-buffer rmail-summary-buffer)))
;;;###autoload
(defun rmail-mode ()
"Rmail Mode is used by \\<rmail-mode-map>\\[rmail] for editing Rmail files.
All normal editing commands are turned off.
Instead, these commands are available:
\\[rmail-beginning-of-message] Move point to front of this message (same as \\[beginning-of-buffer]).
\\[scroll-up] Scroll to next screen of this message.
\\[scroll-down] Scroll to previous screen of this message.
\\[rmail-next-undeleted-message] Move to Next non-deleted message.
\\[rmail-previous-undeleted-message] Move to Previous non-deleted message.
\\[rmail-next-message] Move to Next message whether deleted or not.
\\[rmail-previous-message] Move to Previous message whether deleted or not.
\\[rmail-first-message] Move to the first message in Rmail file.
\\[rmail-last-message] Move to the last message in Rmail file.
\\[rmail-show-message] Jump to message specified by numeric position in file.
\\[rmail-search] Search for string and show message it is found in.
\\[rmail-delete-forward] Delete this message, move to next nondeleted.
\\[rmail-delete-backward] Delete this message, move to previous nondeleted.
\\[rmail-undelete-previous-message] Undelete message. Tries current message, then earlier messages
till a deleted message is found.
\\[rmail-edit-current-message] Edit the current message. \\[rmail-cease-edit] to return to Rmail.
\\[rmail-expunge] Expunge deleted messages.
\\[rmail-expunge-and-save] Expunge and save the file.
\\[rmail-quit] Quit Rmail: expunge, save, then switch to another buffer.
\\[save-buffer] Save without expunging.
\\[rmail-get-new-mail] Move new mail from system spool directory into this file.
\\[rmail-mail] Mail a message (same as \\[mail-other-window]).
\\[rmail-continue] Continue composing outgoing message started before.
\\[rmail-reply] Reply to this message. Like \\[rmail-mail] but initializes some fields.
\\[rmail-retry-failure] Send this message again. Used on a mailer failure message.
\\[rmail-forward] Forward this message to another user.
\\[rmail-output-to-rmail-file] Output this message to an Rmail file (append it).
\\[rmail-output] Output this message to a Unix-format mail file (append it).
\\[rmail-input] Input Rmail file. Run Rmail on that file.
\\[rmail-add-label] Add label to message. It will be displayed in the mode line.
\\[rmail-kill-label] Kill label. Remove a label from current message.
\\[rmail-next-labeled-message] Move to Next message with specified label
(label defaults to last one specified).
Standard labels: filed, unseen, answered, forwarded, deleted.
Any other label is present only if you add it with \\[rmail-add-label].
\\[rmail-previous-labeled-message] Move to Previous message with specified label
\\[rmail-summary] Show headers buffer, with a one line summary of each message.
\\[rmail-summary-by-labels] Summarize only messages with particular label(s).
\\[rmail-summary-by-recipients] Summarize only messages with particular recipient(s).
\\[rmail-summary-by-regexp] Summarize only messages with particular regexp(s).
\\[rmail-summary-by-topic] Summarize only messages with subject line regexp(s).
\\[rmail-toggle-header] Toggle display of complete header."
(interactive)
(rmail-mode-2)
(rmail-set-message-counters)
(rmail-show-message rmail-total-messages))
(defun rmail-mode-2 ()
(kill-all-local-variables)
(rmail-mode-1)
(rmail-variables)
(run-hooks 'rmail-mode-hook))
(defun rmail-mode-1 ()
(setq major-mode 'rmail-mode)
(setq mode-name "RMAIL")
(setq buffer-read-only t)
;; No need to auto save RMAIL files in normal circumstances
;; because they contain no info except attribute changes
;; and deletion of messages.
;; The one exception is when messages are copied into an Rmail mode buffer.
;; rmail-output-to-rmail-file enables auto save when you do that.
(setq buffer-auto-save-file-name nil)
(if (boundp 'mode-line-modified)
(setq mode-line-modified "--- ")
(setq mode-line-format
(cons "--- " (cdr (default-value 'mode-line-format)))))
(use-local-map rmail-mode-map)
(set-syntax-table text-mode-syntax-table)
(setq local-abbrev-table text-mode-abbrev-table))
(defun rmail-variables ()
(make-local-variable 'revert-buffer-function)
(setq revert-buffer-function 'rmail-revert)
(make-local-variable 'font-lock-defaults)
(setq font-lock-defaults '(rmail-font-lock-keywords t))
(make-local-variable 'rmail-last-label)
(make-local-variable 'rmail-last-regexp)
(make-local-variable 'rmail-deleted-vector)
(make-local-variable 'rmail-summary-buffer)
(make-local-variable 'rmail-summary-vector)
(make-local-variable 'rmail-current-message)
(make-local-variable 'rmail-total-messages)
(make-local-variable 'require-final-newline)
(setq require-final-newline nil)
(make-local-variable 'rmail-overlay-list)
(setq rmail-overlay-list nil)
(make-local-variable 'version-control)
(setq version-control 'never)
(make-local-variable 'kill-buffer-hook)
(add-hook 'kill-buffer-hook 'rmail-mode-kill-summary)
(make-local-variable 'file-precious-flag)
(setq file-precious-flag t)
(make-local-variable 'rmail-message-vector)
(make-local-variable 'rmail-inbox-list)
(setq rmail-inbox-list (rmail-parse-file-inboxes))
;; Provide default set of inboxes for primary mail file ~/RMAIL.
(and (null rmail-inbox-list)
(or (equal buffer-file-name (expand-file-name rmail-file-name))
(equal buffer-file-truename
(abbreviate-file-name (file-truename rmail-file-name))))
(setq rmail-inbox-list
(or rmail-primary-inbox-list
(list (or (getenv "MAIL")
(concat rmail-spool-directory
(user-login-name)))))))
(make-local-variable 'rmail-keywords)
;; this gets generated as needed
(setq rmail-keywords nil))
;; Handle M-x revert-buffer done in an rmail-mode buffer.
(defun rmail-revert (arg noconfirm)
(let ((revert-buffer-function (default-value 'revert-buffer-function)))
;; Call our caller again, but this time it does the default thing.
(if (revert-buffer arg noconfirm)
;; If the user said "yes", and we changed something,
;; reparse the messages.
(progn
(rmail-convert-file)
(goto-char (point-max))
(rmail-mode)))))
;; Return a list of files from this buffer's Mail: option.
;; Does not assume that messages have been parsed.
;; Just returns nil if buffer does not look like Babyl format.
(defun rmail-parse-file-inboxes ()
(save-excursion
(save-restriction
(widen)
(goto-char 1)
(cond ((looking-at "BABYL OPTIONS:")
(search-forward "\n\^_" nil 'move)
(narrow-to-region 1 (point))
(goto-char 1)
(if (search-forward "\nMail:" nil t)
(progn
(narrow-to-region (point) (progn (end-of-line) (point)))
(goto-char (point-min))
(mail-parse-comma-list))))))))
(defun rmail-expunge-and-save ()
"Expunge and save RMAIL file."
(interactive)
(rmail-expunge)
(save-buffer)
(if (rmail-summary-exists)
(rmail-select-summary (set-buffer-modified-p nil))))
(defun rmail-quit ()
"Quit out of RMAIL."
(interactive)
(rmail-expunge-and-save)
;; Don't switch to the summary buffer even if it was recently visible.
(if rmail-summary-buffer
(progn
(replace-buffer-in-windows rmail-summary-buffer)
(bury-buffer rmail-summary-buffer)))
(let ((obuf (current-buffer)))
(replace-buffer-in-windows obuf)
(bury-buffer obuf)))
(defun rmail-bury ()
"Bury current Rmail buffer and its summary buffer."
(interactive)
;; This let var was called rmail-buffer, but that interfered
;; with the buffer-local var used in summary buffers.
(let ((buffer-to-bury (current-buffer)))
(if (rmail-summary-exists)
(let (window)
(while (setq window (get-buffer-window rmail-summary-buffer))
(set-window-buffer window (other-buffer rmail-summary-buffer)))
(bury-buffer rmail-summary-buffer)))
(switch-to-buffer (other-buffer (current-buffer)))
(bury-buffer buffer-to-bury)))
(defun rmail-duplicate-message ()
"Create a duplicated copy of the current message.
The duplicate copy goes into the Rmail file just after the
original copy."
(interactive)
(widen)
(let ((buffer-read-only nil)
(number rmail-current-message)
(string (buffer-substring (rmail-msgbeg rmail-current-message)
(rmail-msgend rmail-current-message))))
(goto-char (rmail-msgend rmail-current-message))
(insert string)
(rmail-forget-messages)
(rmail-show-message number)
(message "Message duplicated")))
;;;###autoload
(defun rmail-input (filename)
"Run Rmail on file FILENAME."
(interactive "FRun rmail on RMAIL file: ")
(rmail filename))
;; Return a list of file names for all files in or under START
;; whose names match rmail-secondary-file-regexp.
;; This includes START itself, if that name matches.
;; But normally START is a directory.
(defun rmail-find-all-files (start)
(if (file-accessible-directory-p start)
;; Don't sort here.
(let ((files (directory-files start t
rmail-secondary-file-regexp t))
(ret nil)
file)
(while files
(setq file (car files))
(setq files (cdr files))
(setq ret (nconc
(rmail-find-all-files file)
ret)))
;; Sort here instead of in directory-files
;; because this list is usually much shorter.
(sort ret 'string<))
(let ((case-fold-search nil))
(if (string-match rmail-secondary-file-regexp start)
(list (file-name-nondirectory start))))))
(defun rmail-list-to-menu (menu-name l action &optional full-name)
(let ((menu (make-sparse-keymap menu-name)))
(mapcar
(function (lambda (item)
(let (command)
(if (consp item)
(progn
(setq command
(rmail-list-to-menu (car item) (cdr item)
action
(if full-name
(concat full-name "/"
(car item))
(car item))))
(setq name (car item)))
(progn
(setq name item)
(setq command
(list 'lambda () '(interactive)
(list action
(expand-file-name
(if full-name
(concat full-name "/" item)
item)
rmail-secondary-file-directory))))))
(define-key menu (vector (intern name))
(cons name command)))))
(reverse l))
menu))
;; This command is always "disabled" when it appears in a menu.
(put 'rmail-disable-menu 'menu-enable ''nil)
(defun rmail-construct-io-menu ()
(let ((files (rmail-find-all-files rmail-secondary-file-directory)))
(if files
(progn
(define-key rmail-mode-map [menu-bar classify input-menu]
(cons "Input Rmail File"
(rmail-list-to-menu "Input Rmail File"
files
'rmail-input)))
(define-key rmail-mode-map [menu-bar classify output-menu]
(cons "Output Rmail File"
(rmail-list-to-menu "Output Rmail File"
files
'rmail-output-to-rmail-file))))
(define-key rmail-mode-map [menu-bar classify input-menu]
'("Input Rmail File" . rmail-disable-menu))
(define-key rmail-mode-map [menu-bar classify output-menu]
'("Output Rmail File" . rmail-disable-menu)))))
;;;; *** Rmail input ***
;; RLK feature not added in this version:
;; argument specifies inbox file or files in various ways.
(defun rmail-get-new-mail (&optional file-name)
"Move any new mail from this RMAIL file's inbox files.
The inbox files can be specified with the file's Mail: option. The
variable `rmail-primary-inbox-list' specifies the inboxes for your
primary RMAIL file if it has no Mail: option. By default, this is
your /usr/spool/mail/$USER.
You can also specify the file to get new mail from. In this case, the
file of new mail is not changed or deleted. Noninteractively, you can
pass the inbox file name as an argument. Interactively, a prefix
argument causes us to read a file name and use that file as the inbox.
This function runs `rmail-get-new-mail-hook' before saving the updated file.
It returns t if it got any new messages."
(interactive
(list (if current-prefix-arg
(read-file-name "Get new mail from file: "))))
;; If the disk file has been changed from under us,
;; revert to it before we get new mail.
(or (verify-visited-file-modtime (current-buffer))
(find-file (buffer-file-name)))
(rmail-maybe-set-message-counters)
(widen)
;; Get rid of all undo records for this buffer.
(or (eq buffer-undo-list t)
(setq buffer-undo-list nil))
(unwind-protect
(let ((opoint (point))
(new-messages 0)
(delete-files ())
;; If buffer has not changed yet, and has not been saved yet,
;; don't replace the old backup file now.
(make-backup-files (and make-backup-files (buffer-modified-p)))
(buffer-read-only nil)
;; Don't make undo records for what we do in getting mail.
(buffer-undo-list t))
(goto-char (point-max))
(skip-chars-backward " \t\n") ; just in case of brain damage
(delete-region (point) (point-max)) ; caused by require-final-newline
(save-excursion
(save-restriction
(narrow-to-region (point) (point))
;; Read in the contents of the inbox files,
;; renaming them as necessary,
;; and adding to the list of files to delete eventually.
(if file-name
(rmail-insert-inbox-text (list file-name) nil)
(setq delete-files (rmail-insert-inbox-text rmail-inbox-list t)))
;; Scan the new text and convert each message to babyl format.
(goto-char (point-min))
(save-excursion
(setq new-messages (rmail-convert-to-babyl-format)))
(or (zerop new-messages)
(let (success)
(widen)
(search-backward "\n\^_" nil t)
(narrow-to-region (point) (point-max))
(goto-char (1+ (point-min)))
(rmail-count-new-messages)
(run-hooks 'rmail-get-new-mail-hook)
(save-buffer)))
;; Delete the old files, now that babyl file is saved.
(while delete-files
(condition-case ()
;; First, try deleting.
(condition-case ()
(delete-file (car delete-files))
(file-error
;; If we can't delete it, truncate it.
(write-region (point) (point) (car delete-files))))
(file-error nil))
(setq delete-files (cdr delete-files)))))
(if (= new-messages 0)
(progn (goto-char opoint)
(if (or file-name rmail-inbox-list)
(message "(No new mail has arrived)"))
nil)
(if (rmail-summary-exists)
(rmail-select-summary
(rmail-update-summary)))
(message "%d new message%s read"
new-messages (if (= 1 new-messages) "" "s"))
;; Move to the first new message
;; unless we have other unseen messages before it.
(rmail-show-message (rmail-first-unseen-message))
;; Update the displayed time, since that will clear out
;; the flag that says you have mail.
(if (eq (process-status "display-time") 'run)
(display-time-filter display-time-process ""))
t))
;; Don't leave the buffer screwed up if we get a disk-full error.
(rmail-show-message)))
(defun rmail-insert-inbox-text (files renamep)
(let (file tofile delete-files movemail popmail)
(while files
(setq file (file-truename
(expand-file-name (substitute-in-file-name (car files))))
tofile (expand-file-name
;; Generate name to move to from inbox name,
;; in case of multiple inboxes that need moving.
(concat ".newmail-" (file-name-nondirectory file))
;; Use the directory of this rmail file
;; because it's a nuisance to use the homedir
;; if that is on a full disk and this rmail
;; file isn't.
(file-name-directory
(expand-file-name buffer-file-name))))
;; Always use movemail to rename the file,
;; since there can be mailboxes in various directories.
(setq movemail t)
;;; ;; If getting from mail spool directory,
;;; ;; use movemail to move rather than just renaming,
;;; ;; so as to interlock with the mailer.
;;; (setq movemail (string= file
;;; (file-truename
;;; (concat rmail-spool-directory
;;; (file-name-nondirectory file)))))
(setq popmail (string-match "^po:" (file-name-nondirectory file)))
(if popmail (setq file (file-name-nondirectory file)
renamep t))
(if movemail
(progn
;; On some systems, /usr/spool/mail/foo is a directory
;; and the actual inbox is /usr/spool/mail/foo/foo.
(if (file-directory-p file)
(setq file (expand-file-name (user-login-name)
file)))))
(cond (popmail
(message "Getting mail from post office ..."))
((and (file-exists-p tofile)
(/= 0 (nth 7 (file-attributes tofile))))
(message "Getting mail from %s..." tofile))
((and (file-exists-p file)
(/= 0 (nth 7 (file-attributes file))))
(message "Getting mail from %s..." file)))
;; Set TOFILE if have not already done so, and
;; rename or copy the file FILE to TOFILE if and as appropriate.
(cond ((not renamep)
(setq tofile file))
((or (file-exists-p tofile) (and (not popmail)
(not (file-exists-p file))))
nil)
((and (not movemail) (not popmail))
;; Try copying. If that fails (perhaps no space),
;; rename instead.
(condition-case nil
(copy-file file tofile nil)
(error
;; Third arg is t so we can replace existing file TOFILE.
(rename-file file tofile t)))
;; Make the real inbox file empty.
;; Leaving it deleted could cause lossage
;; because mailers often won't create the file.
(condition-case ()
(write-region (point) (point) file)
(file-error nil)))
(t
(let ((errors nil))
(unwind-protect
(save-excursion
(setq errors (generate-new-buffer " *rmail loss*"))
(buffer-disable-undo errors)
(call-process
(or rmail-movemail-program
(expand-file-name "movemail" exec-directory))
nil errors nil file tofile)
(if (not (buffer-modified-p errors))
;; No output => movemail won
nil
(set-buffer errors)
(subst-char-in-region (point-min) (point-max)
?\n ?\ )
(goto-char (point-max))
(skip-chars-backward " \t")
(delete-region (point) (point-max))
(goto-char (point-min))
(if (looking-at "movemail: ")
(delete-region (point-min) (match-end 0)))
(beep t)
(message (concat "movemail: "
(buffer-substring (point-min)
(point-max))))
(sit-for 3)
nil))
(if errors (kill-buffer errors))))))
;; At this point, TOFILE contains the name to read:
;; Either the alternate name (if we renamed)
;; or the actual inbox (if not renaming).
(if (file-exists-p tofile)
(let (size)
(goto-char (point-max))
(setq size (nth 1 (insert-file-contents tofile)))
(goto-char (point-max))
(or (= (preceding-char) ?\n)
(zerop size)
(insert ?\n))
(setq delete-files (cons tofile delete-files))))
(message "")
(setq files (cdr files)))
delete-files))
;; the rmail-break-forwarded-messages feature is not implemented
(defun rmail-convert-to-babyl-format ()
(let ((count 0) start
(case-fold-search nil)
(invalid-input-resync
(function (lambda ()
(message "Invalid Babyl format in inbox!")
(sit-for 1)
;; Try to get back in sync with a real message.
(if (re-search-forward
(concat mmdf-delim1 "\\|^From") nil t)
(beginning-of-line)
(goto-char (point-max)))))))
(goto-char (point-min))
(save-restriction
(while (not (eobp))
(cond ((looking-at "BABYL OPTIONS:");Babyl header
(if (search-forward "\n\^_" nil t)
;; If we find the proper terminator, delete through there.
(delete-region (point-min) (point))
(funcall invalid-input-resync)
(delete-region (point-min) (point))))
;; Babyl format message
((looking-at "\^L")
(or (search-forward "\n\^_" nil t)
(funcall invalid-input-resync))
(setq count (1+ count))
;; Make sure there is no extra white space after the ^_
;; at the end of the message.
;; Narrowing will make sure that whatever follows the junk
;; will be treated properly.
(delete-region (point)
(save-excursion
(skip-chars-forward " \t\n")
(point)))
(narrow-to-region (point) (point-max)))
;;*** MMDF format
((let ((case-fold-search t))
(looking-at mmdf-delim1))
(let ((case-fold-search t))
(replace-match "\^L\n0, unseen,,\n*** EOOH ***\n")
(setq start (point))
(re-search-forward mmdf-delim2 nil t)
(replace-match "\^_"))
(save-excursion
(save-restriction
(narrow-to-region start (1- (point)))
(goto-char (point-min))
(while (search-forward "\n\^_" nil t); single char "\^_"
(replace-match "\n^_")))); 2 chars: "^" and "_"
(narrow-to-region (point) (point-max))
(setq count (1+ count)))
;;*** Mail format
((looking-at "^From ")
(setq start (point))
(insert "\^L\n0, unseen,,\n*** EOOH ***\n")
(rmail-nuke-pinhead-header)
;; If this message has a Content-Length field,
;; skip to the end of the contents.
(let* ((header-end (save-excursion
(and (re-search-forward "\n\n" nil t)
(1- (point)))))
(case-fold-search t)
(size
;; Get the numeric value from the Content-Length field.
(save-excursion
;; Back up to end of prev line,
;; in case the Content-Length field comes first.
(forward-char -1)
(and (search-forward "\ncontent-length: "
header-end t)
(let ((beg (point))
(eol (progn (end-of-line) (point))))
(string-to-int (buffer-substring beg eol)))))))
(and size
(if (and (natnump size)
(<= (+ header-end size) (point-max))
;; Make sure this would put us at a position
;; that we could continue from.
(save-excursion
(goto-char (+ header-end size))
(skip-chars-forward "\n")
(or (eobp)
(and (looking-at "BABYL OPTIONS:")
(search-forward "\n\^_" nil t))
(and (looking-at "\^L")
(search-forward "\n\^_" nil t))
(let ((case-fold-search t))
(looking-at mmdf-delim1))
(looking-at "From "))))
(goto-char (+ header-end size))
(message "Ignoring invalid Content-Length field")
(sit-for 1 0 t))))
(if (re-search-forward
(concat "^[\^_]?\\("
rmail-unix-mail-delimiter
"\\|"
mmdf-delim1 "\\|"
"^BABYL OPTIONS:\\|"
"\^L\n[01],\\)") nil t)
(goto-char (match-beginning 1))
(goto-char (point-max)))
(setq count (1+ count))
(save-excursion
(save-restriction
(narrow-to-region start (point))
(goto-char (point-min))
(while (search-forward "\n\^_" nil t); single char
(replace-match "\n^_")))); 2 chars: "^" and "_"
(insert ?\^_)
(narrow-to-region (point) (point-max)))
;;
;; This kludge is because some versions of sendmail.el
;; insert an extra newline at the beginning that shouldn't
;; be there. sendmail.el has been fixed, but old versions
;; may still be in use. -- rms, 7 May 1993.
((eolp) (delete-char 1))
(t (error "Cannot convert to babyl format")))))
count))
;; Delete the "From ..." line, creating various other headers with
;; information from it if they don't already exist. Now puts the
;; original line into a mail-from: header line for debugging and for
;; use by the rmail-output function.
(defun rmail-nuke-pinhead-header ()
(save-excursion
(save-restriction
(let ((start (point))
(end (progn
(condition-case ()
(search-forward "\n\n")
(error
(goto-char (point-max))
(insert "\n\n")))
(point)))
has-from has-date)
(narrow-to-region start end)
(let ((case-fold-search t))
(goto-char start)
(setq has-from (search-forward "\nFrom:" nil t))
(goto-char start)
(setq has-date (and (search-forward "\nDate:" nil t) (point)))
(goto-char start))
(let ((case-fold-search nil))
(if (re-search-forward (concat "^" rmail-unix-mail-delimiter) nil t)
(replace-match
(concat
"Mail-from: \\&"
;; Keep and reformat the date if we don't
;; have a Date: field.
(if has-date
""
(concat
"Date: \\3, \\5 \\4 \\9 \\6 "
;; The timezone could be matched by group 7 or group 10.
;; If neither of them matched, assume EST, since only
;; Easterners would be so sloppy.
;; It's a shame the substitution can't use "\\10".
(cond
((/= (match-beginning 7) (match-end 7)) "\\7")
((/= (match-beginning 10) (match-end 10))
(buffer-substring (match-beginning 10)
(match-end 10)))
(t "EST"))
"\n"))
;; Keep and reformat the sender if we don't
;; have a From: field.
(if has-from
""
"From: \\1\n"))
t)))))))
;;;; *** Rmail Message Formatting and Header Manipulation ***
(defun rmail-reformat-message (beg end)
(goto-char beg)
(forward-line 1)
(if (/= (following-char) ?0)
(error "Bad format in RMAIL file."))
(let ((buffer-read-only nil)
(delta (- (buffer-size) end)))
(delete-char 1)
(insert ?1)
(forward-line 1)
(let ((case-fold-search t))
(while (looking-at "Summary-line:\\|Mail-From:")
(forward-line 1)))
(if (looking-at "\\*\\*\\* EOOH \\*\\*\\*\n")
(delete-region (point)
(progn (forward-line 1) (point))))
(let ((str (buffer-substring (point)
(save-excursion (search-forward "\n\n" end 'move)
(point)))))
(insert str "*** EOOH ***\n")
(narrow-to-region (point) (- (buffer-size) delta)))
(goto-char (point-min))
(if rmail-message-filter (funcall rmail-message-filter))
(if (or rmail-displayed-headers rmail-ignored-headers)
(rmail-clear-headers))))
(defun rmail-clear-headers (&optional ignored-headers)
"Delete all header fields that Rmail should not show.
If the optional argument IGNORED-HEADERS is non-nil,
delete all header fields whose names match that regexp.
Otherwise, if `rmail-displayed-headers' is non-nil,
delete all header fields *except* those whose names match that regexp.
Otherwise, delete all header fields whose names match `rmail-ignored-headers'."
(if (search-forward "\n\n" nil t)
(if (and rmail-displayed-headers (null ignored-headers))
(save-restriction
(narrow-to-region (point-min) (point))
(let ((buffer-read-only nil) lim)
(goto-char (point-min))
(while (save-excursion
(re-search-forward "\n[^ \t]")
(and (not (eobp))
(setq lim (1- (point)))))
(if (save-excursion
(re-search-forward rmail-displayed-headers lim t))
(goto-char lim)
(delete-region (point) lim))))
(goto-char (point-min)))
(or ignored-headers (setq ignored-headers rmail-ignored-headers))
(save-restriction
(narrow-to-region (point-min) (point))
(let ((buffer-read-only nil))
(while (let ((case-fold-search t))
(goto-char (point-min))
(re-search-forward ignored-headers nil t))
(beginning-of-line)
(delete-region (point)
(progn (re-search-forward "\n[^ \t]")
(1- (point))))))))))
(defun rmail-toggle-header ()
"Show original message header if pruned header currently shown, or vice versa."
(interactive)
(rmail-maybe-set-message-counters)
(narrow-to-region (rmail-msgbeg rmail-current-message) (point-max))
(let ((buffer-read-only nil))
(goto-char (point-min))
(forward-line 1)
(if (= (following-char) ?1)
(progn (delete-char 1)
(insert ?0)
(forward-line 1)
(let ((case-fold-search t))
(while (looking-at "Summary-Line:\\|Mail-From:")
(forward-line 1)))
(insert "*** EOOH ***\n")
(forward-char -1)
(search-forward "\n*** EOOH ***\n")
(forward-line -1)
(let ((temp (point)))
(and (search-forward "\n\n" nil t)
(delete-region temp (point))))
(goto-char (point-min))
(search-forward "\n*** EOOH ***\n")
(narrow-to-region (point) (point-max)))
(rmail-reformat-message (point-min) (point-max))))
(rmail-highlight-headers))
;;;; *** Rmail Attributes and Keywords ***
;; Make a string describing current message's attributes and keywords
;; and set it up as the name of a minor mode
;; so it will appear in the mode line.
(defun rmail-display-labels ()
(let ((blurb "") (beg (point-min-marker)) (end (point-max-marker)))
(save-excursion
(unwind-protect
(progn
(widen)
(goto-char (rmail-msgbeg rmail-current-message))
(forward-line 1)
(if (looking-at "[01],")
(progn
(narrow-to-region (point) (progn (end-of-line) (point)))
;; Truly valid BABYL format requires a space before each
;; attribute or keyword name. Put them in if missing.
(let (buffer-read-only)
(goto-char (point-min))
(while (search-forward "," nil t)
(or (looking-at "[ ,]") (eobp)
(insert " "))))
(goto-char (point-max))
(if (search-backward ",," nil 'move)
(progn
(if (> (point) (1+ (point-min)))
(setq blurb (buffer-substring (+ 1 (point-min)) (point))))
(if (> (- (point-max) (point)) 2)
(setq blurb
(concat blurb
";"
(buffer-substring (+ (point) 3)
(1- (point-max)))))))))))
;; Note: we don't use save-restriction because that does not work right
;; if changes are made outside the saved restriction
;; before that restriction is restored.
(narrow-to-region beg end)
(set-marker beg nil)
(set-marker end nil)))
(while (string-match " +," blurb)
(setq blurb (concat (substring blurb 0 (match-beginning 0)) ","
(substring blurb (match-end 0)))))
(while (string-match ", +" blurb)
(setq blurb (concat (substring blurb 0 (match-beginning 0)) ","
(substring blurb (match-end 0)))))
(setq mode-line-process
(format " %d/%d%s"
rmail-current-message rmail-total-messages blurb))))
;; Turn an attribute of a message on or off according to STATE.
;; ATTR is the name of the attribute, as a string.
;; MSGNUM is message number to change; nil means current message.
(defun rmail-set-attribute (attr state &optional msgnum)
(let ((omax (point-max-marker))
(omin (point-min-marker))
(buffer-read-only nil))
(or msgnum (setq msgnum rmail-current-message))
(if (> msgnum 0)
(unwind-protect
(save-excursion
(widen)
(goto-char (+ 3 (rmail-msgbeg msgnum)))
(let ((curstate
(not
(null (search-backward (concat ", " attr ",")
(prog1 (point) (end-of-line)) t)))))
(or (eq curstate (not (not state)))
(if curstate
(delete-region (point) (1- (match-end 0)))
(beginning-of-line)
(forward-char 2)
(insert " " attr ","))))
(if (string= attr "deleted")
(rmail-set-message-deleted-p msgnum state)))
;; Note: we don't use save-restriction because that does not work right
;; if changes are made outside the saved restriction
;; before that restriction is restored.
(narrow-to-region omin omax)
(set-marker omin nil)
(set-marker omax nil)
(if (= msgnum rmail-current-message)
(rmail-display-labels))))))
;; Return t if the attributes/keywords line of msg number MSG
;; contains a match for the regexp LABELS.
(defun rmail-message-labels-p (msg labels)
(save-excursion
(save-restriction
(widen)
(goto-char (rmail-msgbeg msg))
(forward-char 3)
(re-search-backward labels (prog1 (point) (end-of-line)) t))))
;;;; *** Rmail Message Selection And Support ***
(defun rmail-msgend (n)
(marker-position (aref rmail-message-vector (1+ n))))
(defun rmail-msgbeg (n)
(marker-position (aref rmail-message-vector n)))
(defun rmail-widen-to-current-msgbeg (function)
"Call FUNCTION with point at start of internal data of current message.
Assumes that bounds were previously narrowed to display the message in Rmail.
The bounds are widened enough to move point where desired, then narrowed
again afterward.
FUNCTION may not change the visible text of the message, but it may
change the invisible header text."
(save-excursion
(let ((obeg (- (point-max) (point-min))))
(unwind-protect
(progn
(narrow-to-region (rmail-msgbeg rmail-current-message)
(point-max))
(goto-char (point-min))
(funcall function))
;; Note: we don't use save-restriction because that does not work right
;; if changes are made outside the saved restriction
;; before that restriction is restored.
;; Here we assume that changes made by FUNCTION
;; occur before the visible region of the message.
(narrow-to-region (- (point-max) obeg) (point-max))))))
(defun rmail-forget-messages ()
(unwind-protect
(if (vectorp rmail-message-vector)
(let* ((i 0)
(v rmail-message-vector)
(n (length v)))
(while (< i n)
(move-marker (aref v i) nil)
(setq i (1+ i)))))
(setq rmail-message-vector nil)
(setq rmail-deleted-vector nil)))
(defun rmail-maybe-set-message-counters ()
(if (not (and rmail-deleted-vector
rmail-message-vector
rmail-current-message
rmail-total-messages))
(rmail-set-message-counters)))
(defun rmail-count-new-messages (&optional nomsg)
(let* ((case-fold-search nil)
(total-messages 0)
(messages-head nil)
(deleted-head nil))
(or nomsg (message "Counting new messages..."))
(goto-char (point-max))
;; Put at the end of messages-head
;; the entry for message N+1, which marks
;; the end of message N. (N = number of messages).
(search-backward "\n\^_")
(forward-char 1)
(setq messages-head (list (point-marker)))
(rmail-set-message-counters-counter (point-min))
(setq rmail-current-message (1+ rmail-total-messages))
(setq rmail-total-messages
(+ rmail-total-messages total-messages))
(setq rmail-message-vector
(vconcat rmail-message-vector (cdr messages-head)))
(aset rmail-message-vector
rmail-current-message (car messages-head))
(setq rmail-deleted-vector
(concat rmail-deleted-vector deleted-head))
(setq rmail-summary-vector
(vconcat rmail-summary-vector (make-vector total-messages nil)))
(goto-char (point-min))
(or nomsg (message "Counting new messages...done (%d)" total-messages))))
(defun rmail-set-message-counters ()
(rmail-forget-messages)
(save-excursion
(save-restriction
(widen)
(let* ((point-save (point))
(total-messages 0)
(messages-after-point)
(case-fold-search nil)
(messages-head nil)
(deleted-head nil))
(message "Counting messages...")
(goto-char (point-max))
;; Put at the end of messages-head
;; the entry for message N+1, which marks
;; the end of message N. (N = number of messages).
(search-backward "\n\^_" nil t)
(if (/= (point) (point-max)) (forward-char 1))
(setq messages-head (list (point-marker)))
(rmail-set-message-counters-counter (min (point) point-save))
(setq messages-after-point total-messages)
(rmail-set-message-counters-counter)
(setq rmail-total-messages total-messages)
(setq rmail-current-message
(min total-messages
(max 1 (- total-messages messages-after-point))))
(setq rmail-message-vector
(apply 'vector (cons (point-min-marker) messages-head))
rmail-deleted-vector (concat "D" deleted-head)
rmail-summary-vector (make-vector rmail-total-messages nil))
(message "Counting messages...done")))))
(defun rmail-set-message-counters-counter (&optional stop)
(while (search-backward "\n\^_\^L\n" stop t)
(forward-char 1)
(setq messages-head (cons (point-marker) messages-head))
(save-excursion
(setq deleted-head
(cons (if (search-backward ", deleted,"
(prog1 (point)
(forward-line 2))
t)
?D ?\ )
deleted-head)))
(if (zerop (% (setq total-messages (1+ total-messages)) 20))
(message "Counting messages...%d" total-messages))))
(defun rmail-beginning-of-message ()
"Show current message starting from the beginning."
(interactive)
(rmail-show-message rmail-current-message))
(defun rmail-show-message (&optional n no-summary)
"Show message number N (prefix argument), counting from start of file.
If summary buffer is currently displayed, update current message there also."
(interactive "p")
(rmail-maybe-set-message-counters)
(widen)
(if (zerop rmail-total-messages)
(progn (narrow-to-region (point-min) (1- (point-max)))
(goto-char (point-min))
(setq mode-line-process nil))
(let (blurb)
(if (not n)
(setq n rmail-current-message)
(cond ((<= n 0)
(setq n 1
rmail-current-message 1
blurb "No previous message"))
((> n rmail-total-messages)
(setq n rmail-total-messages
rmail-current-message rmail-total-messages
blurb "No following message"))
(t
(setq rmail-current-message n))))
(let ((beg (rmail-msgbeg n))
(end (rmail-msgend n)))
(goto-char beg)
(forward-line 1)
(if (= (following-char) ?0)
(progn
(rmail-reformat-message beg end)
(rmail-set-attribute "unseen" nil))
(search-forward "\n*** EOOH ***\n" end t)
(narrow-to-region (point) end))
(goto-char (point-min))
(rmail-display-labels)
(rmail-highlight-headers)
(if transient-mark-mode (deactivate-mark))
(run-hooks 'rmail-show-message-hook)
;; If there is a summary buffer, try to move to this message
;; in that buffer. But don't complain if this message
;; is not mentioned in the summary.
;; Don't do this at all if we were called on behalf
;; of cursor motion in the summary buffer.
(and (rmail-summary-exists) (not no-summary)
(let ((curr-msg rmail-current-message))
(rmail-select-summary
(rmail-summary-goto-msg curr-msg t t))))
(if blurb
(message blurb))))))
;; Find all occurrences of certain fields, and highlight them.
(defun rmail-highlight-headers ()
;; Do this only if the system supports faces.
(if (and (fboundp 'internal-find-face)
rmail-highlighted-headers)
(save-excursion
(search-forward "\n\n" nil 'move)
(save-restriction
(narrow-to-region (point-min) (point))
(let ((case-fold-search t)
(inhibit-read-only t)
;; Highlight with boldface if that is available.
;; Otherwise use the `highlight' face.
(face (or rmail-highlight-face
(if (face-differs-from-default-p 'bold)
'bold 'highlight)))
;; List of overlays to reuse.
(overlays rmail-overlay-list))
(goto-char (point-min))
(while (re-search-forward rmail-highlighted-headers nil t)
(skip-chars-forward " \t")
(let ((beg (point))
overlay)
(while (progn (forward-line 1)
(looking-at "[ \t]")))
;; Back up over newline, then trailing spaces or tabs
(forward-char -1)
(while (member (preceding-char) '(? ?\t))
(forward-char -1))
(if overlays
;; Reuse an overlay we already have.
(progn
(setq overlay (car overlays)
overlays (cdr overlays))
(overlay-put overlay 'face face)
(move-overlay overlay beg (point)))
;; Make a new overlay and add it to
;; rmail-overlay-list.
(setq overlay (make-overlay beg (point)))
(overlay-put overlay 'face face)
(setq rmail-overlay-list
(cons overlay rmail-overlay-list))))))))))
(defun rmail-next-message (n)
"Show following message whether deleted or not.
With prefix arg N, moves forward N messages, or backward if N is negative."
(interactive "p")
(rmail-maybe-set-message-counters)
(rmail-show-message (+ rmail-current-message n)))
(defun rmail-previous-message (n)
"Show previous message whether deleted or not.
With prefix arg N, moves backward N messages, or forward if N is negative."
(interactive "p")
(rmail-next-message (- n)))
(defun rmail-next-undeleted-message (n)
"Show following non-deleted message.
With prefix arg N, moves forward N non-deleted messages,
or backward if N is negative.
Returns t if a new message is being shown, nil otherwise."
(interactive "p")
(rmail-maybe-set-message-counters)
(let ((lastwin rmail-current-message)
(current rmail-current-message))
(while (and (> n 0) (< current rmail-total-messages))
(setq current (1+ current))
(if (not (rmail-message-deleted-p current))
(setq lastwin current n (1- n))))
(while (and (< n 0) (> current 1))
(setq current (1- current))
(if (not (rmail-message-deleted-p current))
(setq lastwin current n (1+ n))))
(if (/= lastwin rmail-current-message)
(progn (rmail-show-message lastwin)
t)
(if (< n 0)
(message "No previous nondeleted message"))
(if (> n 0)
(message "No following nondeleted message"))
nil)))
(defun rmail-previous-undeleted-message (n)
"Show previous non-deleted message.
With prefix argument N, moves backward N non-deleted messages,
or forward if N is negative."
(interactive "p")
(rmail-next-undeleted-message (- n)))
(defun rmail-first-message ()
"Show first message in file."
(interactive)
(rmail-maybe-set-message-counters)
(rmail-show-message 1))
(defun rmail-last-message ()
"Show last message in file."
(interactive)
(rmail-maybe-set-message-counters)
(rmail-show-message rmail-total-messages))
(defun rmail-what-message ()
(let ((where (point))
(low 1)
(high rmail-total-messages)
(mid (/ rmail-total-messages 2)))
(while (> (- high low) 1)
(if (>= where (rmail-msgbeg mid))
(setq low mid)
(setq high mid))
(setq mid (+ low (/ (- high low) 2))))
(if (>= where (rmail-msgbeg high)) high low)))
(defun rmail-message-recipients-p (msg recipients &optional primary-only)
(save-restriction
(goto-char (rmail-msgbeg msg))
(search-forward "\n*** EOOH ***\n")
(narrow-to-region (point) (progn (search-forward "\n\n") (point)))
(or (string-match recipients (or (mail-fetch-field "To") ""))
(string-match recipients (or (mail-fetch-field "From") ""))
(if (not primary-only)
(string-match recipients (or (mail-fetch-field "Cc") ""))))))
(defun rmail-message-regexp-p (msg regexp)
"Return t, if for message number MSG, regexp REGEXP matches in the header."
(goto-char (rmail-msgbeg msg))
(let ((end
(save-excursion
(search-forward "*** EOOH ***" (point-max)) (point))))
(re-search-forward regexp end t)))
(defvar rmail-search-last-regexp nil)
(defun rmail-search (regexp &optional n)
"Show message containing next match for REGEXP (but not the current msg).
Prefix argument gives repeat count; negative argument means search
backwards (through earlier messages).
Interactively, empty argument means use same regexp used last time."
(interactive
(let* ((reversep (< (prefix-numeric-value current-prefix-arg) 0))
(prompt
(concat (if reversep "Reverse " "") "Rmail search (regexp): "))
regexp)
(if rmail-search-last-regexp
(setq prompt (concat prompt
"(default "
rmail-search-last-regexp
") ")))
(setq regexp (read-string prompt))
(cond ((not (equal regexp ""))
(setq rmail-search-last-regexp regexp))
((not rmail-search-last-regexp)
(error "No previous Rmail search string")))
(list rmail-search-last-regexp
(prefix-numeric-value current-prefix-arg))))
(or n (setq n 1))
(message "%sRmail search for %s..."
(if (< n 0) "Reverse " "")
regexp)
(rmail-maybe-set-message-counters)
(let ((omin (point-min))
(omax (point-max))
(opoint (point))
win
(reversep (< n 0))
(msg rmail-current-message))
(unwind-protect
(progn
(widen)
(while (/= n 0)
;; Check messages one by one, advancing message number up or down
;; but searching forward through each message.
(if reversep
(while (and (null win) (> msg 1))
(goto-char (rmail-msgbeg (setq msg (1- msg))))
(setq win (re-search-forward
regexp (rmail-msgend msg) t)))
(while (and (null win) (< msg rmail-total-messages))
(goto-char (rmail-msgbeg (setq msg (1+ msg))))
(setq win (re-search-forward regexp (rmail-msgend msg) t))))
(setq n (+ n (if reversep 1 -1)))))
(if win
(progn
;; If this is a reverse search and we found a message,
;; search backward thru this message to position point.
(if reversep
(progn
(goto-char (rmail-msgend msg))
(re-search-backward
regexp (rmail-msgbeg msg) t)))
(setq win (point))
(rmail-show-message msg)
(message "%sRmail search for %s...done"
(if reversep "Reverse " "")
regexp)
(goto-char win))
(goto-char opoint)
(narrow-to-region omin omax)
(ding)
(message "Search failed: %s" regexp)))))
(defun rmail-search-backwards (regexp &optional n)
"Show message containing previous match for REGEXP.
Prefix argument gives repeat count; negative argument means search
forward (through later messages).
Interactively, empty argument means use same regexp used last time."
(interactive
(let* ((reversep (>= (prefix-numeric-value current-prefix-arg) 0))
(prompt
(concat (if reversep "Reverse " "") "Rmail search (regexp): "))
regexp)
(if rmail-search-last-regexp
(setq prompt (concat prompt
"(default "
rmail-search-last-regexp
") ")))
(setq regexp (read-string prompt))
(cond ((not (equal regexp ""))
(setq rmail-search-last-regexp regexp))
((not rmail-search-last-regexp)
(error "No previous Rmail search string")))
(list rmail-search-last-regexp
(prefix-numeric-value current-prefix-arg))))
(rmail-search regexp (- (or n 1))))
;; Show the first message which has the `unseen' attribute.
(defun rmail-first-unseen-message ()
(rmail-maybe-set-message-counters)
(let ((current 1)
found)
(save-restriction
(widen)
(while (and (not found) (<= current rmail-total-messages))
(if (rmail-message-labels-p current ", ?\\(unseen\\),")
(setq found current))
(setq current (1+ current))))
;; Let the caller show the message.
;; (if found
;; (rmail-show-message found))
found))
(defun rmail-next-same-subject (n)
"Go to the next mail message having the same subject header.
With prefix argument N, do this N times.
If N is negative, go backwards instead."
(interactive "p")
(let* ((subject (mail-fetch-field "Subject"))
(search-regexp (concat "^Subject: *\\(Re: *\\)?"
(regexp-quote subject)
"\n"))
(forward (> n 0))
(i rmail-current-message)
found)
(if (string-match "Re:[ \t]*" subject)
(setq subject (substring subject (match-end 0))))
(save-excursion
(save-restriction
(widen)
(while (and (/= n 0)
(if forward
(< i rmail-total-messages)
(> i 1)))
(let (done)
(while (and (not done)
(if forward
(< i rmail-total-messages)
(> i 1)))
(setq i (if forward (1+ i) (1- i)))
(goto-char (rmail-msgbeg i))
(search-forward "\n*** EOOH ***\n")
(let ((beg (point)) end)
(search-forward "\n\n")
(setq end (point))
(goto-char beg)
(setq done (re-search-forward search-regexp end t))))
(if done (setq found i)))
(setq n (if forward (1- n) (1+ n))))))
(if found
(rmail-show-message found)
(error "No %s message with same subject"
(if forward "following" "previous")))))
(defun rmail-previous-same-subject (n)
"Go to the previous mail message having the same subject header.
With prefix argument N, do this N times.
If N is negative, go forwards instead."
(interactive "p")
(rmail-next-same-subject (- n)))
;;;; *** Rmail Message Deletion Commands ***
(defun rmail-message-deleted-p (n)
(= (aref rmail-deleted-vector n) ?D))
(defun rmail-set-message-deleted-p (n state)
(aset rmail-deleted-vector n (if state ?D ?\ )))
(defun rmail-delete-message ()
"Delete this message and stay on it."
(interactive)
(rmail-set-attribute "deleted" t)
(run-hooks 'rmail-delete-message-hook))
(defun rmail-undelete-previous-message ()
"Back up to deleted message, select it, and undelete it."
(interactive)
(let ((msg rmail-current-message))
(while (and (> msg 0)
(not (rmail-message-deleted-p msg)))
(setq msg (1- msg)))
(if (= msg 0)
(error "No previous deleted message")
(if (/= msg rmail-current-message)
(rmail-show-message msg))
(rmail-set-attribute "deleted" nil)
(if (rmail-summary-exists)
(save-excursion
(set-buffer rmail-summary-buffer)
(rmail-summary-mark-undeleted msg)))
(rmail-maybe-display-summary))))
(defun rmail-delete-forward (&optional backward)
"Delete this message and move to next nondeleted one.
Deleted messages stay in the file until the \\[rmail-expunge] command is given.
With prefix argument, delete and move backward.
Returns t if a new message is displayed after the delete, or nil otherwise."
(interactive "P")
(rmail-set-attribute "deleted" t)
(run-hooks 'rmail-delete-message-hook)
(let ((del-msg rmail-current-message))
(if (rmail-summary-exists)
(rmail-select-summary
(rmail-summary-mark-deleted del-msg)))
(prog1 (rmail-next-undeleted-message (if backward -1 1))
(rmail-maybe-display-summary))))
(defun rmail-delete-backward ()
"Delete this message and move to previous nondeleted one.
Deleted messages stay in the file until the \\[rmail-expunge] command is given."
(interactive)
(rmail-delete-forward t))
;; Compute the message number a given message would have after expunging.
;; The present number of the message is OLDNUM.
;; DELETEDVEC should be rmail-deleted-vector.
;; The value is nil for a message that would be deleted.
(defun rmail-msg-number-after-expunge (deletedvec oldnum)
(if (or (null oldnum) (= (aref deletedvec oldnum) ?D))
nil
(let ((i 0)
(newnum 0))
(while (< i oldnum)
(if (/= (aref deletedvec i) ?D)
(setq newnum (1+ newnum)))
(setq i (1+ i)))
newnum)))
(defun rmail-only-expunge ()
"Actually erase all deleted messages in the file."
(interactive)
(message "Expunging deleted messages...")
;; Discard all undo records for this buffer.
(or (eq buffer-undo-list t)
(setq buffer-undo-list nil))
(rmail-maybe-set-message-counters)
(let* ((omax (- (buffer-size) (point-max)))
(omin (- (buffer-size) (point-min)))
(opoint (if (and (> rmail-current-message 0)
(rmail-message-deleted-p rmail-current-message))
0
(- (point) (point-min))))
(messages-head (cons (aref rmail-message-vector 0) nil))
(messages-tail messages-head)
;; Don't make any undo records for the expunging.
(buffer-undo-list t)
(win))
(unwind-protect
(save-excursion
(widen)
(goto-char (point-min))
(let ((counter 0)
(number 1)
(total rmail-total-messages)
(new-message-number rmail-current-message)
(new-summary nil)
(rmailbuf (current-buffer))
(buffer-read-only nil)
(messages rmail-message-vector)
(deleted rmail-deleted-vector)
(summary rmail-summary-vector))
(setq rmail-total-messages nil
rmail-current-message nil
rmail-message-vector nil
rmail-deleted-vector nil
rmail-summary-vector nil)
;; Find each sendmail buffer that is set to reply
;; to a message in this buffer, and update its
;; message number.
(let ((bufs (buffer-list)))
(while bufs
(save-excursion
(set-buffer (car bufs))
(and (boundp 'rmail-send-actions-rmail-buffer)
(eq rmail-send-actions-rmail-buffer rmailbuf)
(setq rmail-send-actions-rmail-msg-number
(rmail-msg-number-after-expunge
deleted
rmail-send-actions-rmail-msg-number))))
(setq bufs (cdr bufs))))
(while (<= number total)
(if (= (aref deleted number) ?D)
(progn
(delete-region
(marker-position (aref messages number))
(marker-position (aref messages (1+ number))))
(move-marker (aref messages number) nil)
(if (> new-message-number counter)
(setq new-message-number (1- new-message-number))))
(setq counter (1+ counter))
(setq messages-tail
(setcdr messages-tail
(cons (aref messages number) nil)))
(setq new-summary
(cons (if (= counter number) (aref summary (1- number)))
new-summary)))
(if (zerop (% (setq number (1+ number)) 20))
(message "Expunging deleted messages...%d" number)))
(setq messages-tail
(setcdr messages-tail
(cons (aref messages number) nil)))
(setq rmail-current-message new-message-number
rmail-total-messages counter
rmail-message-vector (apply 'vector messages-head)
rmail-deleted-vector (make-string (1+ counter) ?\ )
rmail-summary-vector (vconcat (nreverse new-summary))
win t)))
(message "Expunging deleted messages...done")
(if (not win)
(narrow-to-region (- (buffer-size) omin) (- (buffer-size) omax)))
(rmail-show-message
(if (zerop rmail-current-message) 1 nil))
(forward-char opoint))))
(defun rmail-expunge ()
"Erase deleted messages from Rmail file and summary buffer."
(interactive)
(rmail-only-expunge)
(if (rmail-summary-exists)
(rmail-select-summary
(rmail-update-summary))))
;;;; *** Rmail Mailing Commands ***
(defun rmail-start-mail (&rest args)
(if (and window-system rmail-mail-new-frame)
(prog1
(apply 'mail-other-frame args)
(modify-frame-parameters (selected-frame)
'((dedicated . t))))
(apply 'mail-other-window args)))
(defun rmail-mail ()
"Send mail in another window.
While composing the message, use \\[mail-yank-original] to yank the
original message into it."
(interactive)
(rmail-start-mail nil nil nil nil nil (current-buffer)))
(defun rmail-continue ()
"Continue composing outgoing message previously being composed."
(interactive)
(rmail-start-mail t))
(defun rmail-reply (just-sender)
"Reply to the current message.
Normally include CC: to all other recipients of original message;
prefix argument means ignore them. While composing the reply,
use \\[mail-yank-original] to yank the original message into it."
(interactive "P")
(let (from reply-to cc subject date to message-id resent-reply-to
(msgnum rmail-current-message)
(rmail-buffer (current-buffer)))
(save-excursion
(save-restriction
(widen)
(goto-char (rmail-msgbeg rmail-current-message))
(forward-line 1)
(if (= (following-char) ?0)
(narrow-to-region
(progn (forward-line 2)
(point))
(progn (search-forward "\n\n" (rmail-msgend rmail-current-message)
'move)
(point)))
(narrow-to-region (point)
(progn (search-forward "\n*** EOOH ***\n")
(beginning-of-line) (point))))
(setq resent-reply-to (mail-fetch-field "resent-reply-to" t)
from (mail-fetch-field "from")
reply-to (or resent-reply-to
(mail-fetch-field "reply-to" nil t)
from)
cc (cond (just-sender nil)
(resent-reply-to (mail-fetch-field "resent-cc" t))
(t (mail-fetch-field "cc" nil t)))
subject (or (and resent-reply-to
(mail-fetch-field "resent-subject" t))
(mail-fetch-field "subject"))
date (or (and resent-reply-to
(mail-fetch-field "resent-date" t))
(mail-fetch-field "date"))
to (cond (resent-reply-to
(or (mail-fetch-field "resent-to" t)) "")
((mail-fetch-field "to" nil t))
;((mail-fetch-field "apparently-to")) ack gag barf
(t ""))
message-id (cond (resent-reply-to
(mail-fetch-field "resent-message-id" t))
((mail-fetch-field "message-id"))))))
(and (stringp subject)
(setq subject
(concat rmail-reply-prefix
(if (string-match rmail-reply-regexp subject)
(substring subject (match-end 0))
subject))))
(rmail-start-mail nil
(mail-strip-quoted-names reply-to)
subject
(rmail-make-in-reply-to-field from date message-id)
(if just-sender
nil
(let* ((cc-list (rmail-dont-reply-to
(mail-strip-quoted-names
(if (null cc) to (concat to ", " cc))))))
(if (string= cc-list "") nil cc-list)))
(current-buffer)
(list (list '(lambda ()
(let ((msgnum rmail-send-actions-rmail-msg-number))
(save-excursion
(set-buffer rmail-send-actions-rmail-buffer)
(if msgnum
(rmail-set-attribute "answered" t msgnum))))))))
;; We keep the rmail buffer and message number in these
;; buffer-local vars in the sendmail buffer,
;; so that rmail-only-expunge can relocate the message number.
(make-local-variable 'rmail-send-actions-rmail-buffer)
(make-local-variable 'rmail-send-actions-rmail-msg-number)
(setq rmail-send-actions-rmail-buffer rmail-buffer)
(setq rmail-send-actions-rmail-msg-number msgnum)))
(defun rmail-make-in-reply-to-field (from date message-id)
(cond ((not from)
(if message-id
message-id
nil))
(mail-use-rfc822
(require 'rfc822)
(let ((tem (car (rfc822-addresses from))))
(if message-id
(if (string-match
(regexp-quote (if (string-match "@[^@]*\\'" tem)
(substring tem 0 (match-beginning 0))
tem))
message-id)
;; Message-ID is sufficiently informative
message-id
(concat message-id " (" tem ")"))
;; Copy TEM, discarding text properties.
(setq tem (copy-sequence tem))
(set-text-properties 0 (length tem) nil tem)
(setq tem (copy-sequence tem))
;; Use prin1 to fake RFC822 quoting
(let ((field (prin1-to-string tem)))
(if date
(concat field "'s message of " date)
field)))))
((let* ((foo "[^][\000-\037\177-\377()<>@,;:\\\" ]+")
(bar "[^][\000-\037\177-\377()<>@,;:\\\"]+"))
;; Can't use format because format loses on \000 (unix *^&%*^&%$!!)
(or (string-match (concat "\\`[ \t]*\\(" bar
"\\)\\(<" foo "@" foo ">\\)?[ \t]*\\'")
;; "Unix Loser <Foo@bar.edu>" => "Unix Loser"
from)
(string-match (concat "\\`[ \t]*<" foo "@" foo ">[ \t]*(\\("
bar "\\))[ \t]*\\'")
;; "<Bugs@bar.edu>" (Losing Unix) => "Losing Unix"
from)))
(let ((start (match-beginning 1))
(end (match-end 1)))
;; Trim whitespace which above regexp match allows
(while (and (< start end)
(memq (aref from start) '(?\t ?\ )))
(setq start (1+ start)))
(while (and (< start end)
(memq (aref from (1- end)) '(?\t ?\ )))
(setq end (1- end)))
(let ((field (substring from start end)))
(if date (setq field (concat "message from " field " on " date)))
(if message-id
;; "<AA259@bar.edu> (message from Unix Loser on 1-Apr-89)"
(concat message-id " (" field ")")
field))))
(t
;; If we can't kludge it simply, do it correctly
(let ((mail-use-rfc822 t))
(rmail-make-in-reply-to-field from date message-id)))))
(defun rmail-forward (resend)
"Forward the current message to another user.
With prefix argument, \"resend\" the message instead of forwarding it;
see the documentation of `rmail-resend'."
(interactive "P")
(if resend
(call-interactively 'rmail-resend)
(let ((forward-buffer (current-buffer))
(msgnum rmail-current-message)
(subject (concat "["
(let ((from (or (mail-fetch-field "From")
(mail-fetch-field ">From"))))
(if from
(concat (mail-strip-quoted-names from) ": ")
""))
(or (mail-fetch-field "Subject") "")
"]")))
;; If only one window, use it for the mail buffer.
;; Otherwise, use another window for the mail buffer
;; so that the Rmail buffer remains visible
;; and sending the mail will get back to it.
(if (funcall (if (and (not rmail-mail-new-frame) (one-window-p t))
(function mail)
(function rmail-start-mail))
nil nil subject nil nil nil
(list (list (function
(lambda ()
(let ((msgnum
rmail-send-actions-rmail-msg-number))
(save-excursion
(set-buffer rmail-send-actions-rmail-buffer)
(if msgnum
(rmail-set-attribute
"forwarded" t msgnum)))))))))
;; The mail buffer is now current.
(save-excursion
;; We keep the rmail buffer and message number in these
;; buffer-local vars in the sendmail buffer,
;; so that rmail-only-expunge can relocate the message number.
(make-local-variable 'rmail-send-actions-rmail-buffer)
(make-local-variable 'rmail-send-actions-rmail-msg-number)
(setq rmail-send-actions-rmail-buffer forward-buffer)
(setq rmail-send-actions-rmail-msg-number msgnum)
;; Insert after header separator--before signature if any.
(goto-char (point-min))
(search-forward-regexp
(concat "^" (regexp-quote mail-header-separator) "$"))
(forward-line 1)
(insert "------- Start of forwarded message -------\n")
;; Quote lines with `- ' if they start with `-'.
(let ((beg (point)) end)
(insert-buffer-substring forward-buffer)
(goto-char beg)
(while (re-search-forward "^-" nil t)
(beginning-of-line)
(insert "- ")
(forward-line 1))
(goto-char (point-max)))
(let ((end (point)))
(skip-chars-backward "\n")
(if (< (point) end)
(forward-char 1))
(delete-region (point) end))
(insert "------- End of forwarded message -------\n")
(push-mark))))))
(defun rmail-resend (address &optional from comment mail-alias-file)
"Resend current message to ADDRESSES.
ADDRESSES should be a single address, a string consisting of several
addresses separated by commas, or a list of addresses.
Optional FROM is the address to resend the message from, and
defaults to the username of the person redistributing the message.
Optional COMMENT is a string that will be inserted as a comment in the
resent message.
Optional ALIAS-FILE is alternate aliases file to be used by sendmail,
typically for purposes of moderating a list."
(interactive "sResend to: ")
(require 'sendmail)
(require 'mailalias)
(if (not from) (setq from (user-login-name)))
(let ((tembuf (generate-new-buffer " sendmail temp"))
(mail-header-separator "")
(case-fold-search nil)
(mailbuf (current-buffer)))
(unwind-protect
(save-excursion
;;>> Copy message into temp buffer
(set-buffer tembuf)
(insert-buffer-substring mailbuf)
(goto-char (point-min))
;; Delete any Sender field, since that's not specifiable.
; Only delete Sender fields in the actual header.
(re-search-forward "^$" nil 'move)
; Using "while" here rather than "if" because some buggy mail
; software may have inserted multiple Sender fields.
(while (re-search-backward "^Sender:" nil t)
(let (beg)
(setq beg (point))
(forward-line 1)
(while (looking-at "[ \t]")
(forward-line 1))
(delete-region beg (point))))
; Go back to the beginning of the buffer so the Resent- fields
; are inserted there.
(goto-char (point-min))
;;>> Insert resent-from:
(insert "Resent-From: " from "\n")
(insert "Resent-Date: " (mail-rfc822-date) "\n")
;;>> Insert resent-to: and bcc if need be.
(let ((before (point)))
(if mail-self-blind
(insert "Resent-Bcc: " (user-login-name) "\n"))
(insert "Resent-To: " (if (stringp address)
address
(mapconcat 'identity address ",\n\t"))
"\n")
;; Expand abbrevs in the recipients.
(save-excursion
(if (featurep 'mailabbrev)
(let ((end (point-marker))
(local-abbrev-table mail-abbrevs)
(old-syntax-table (syntax-table)))
(if (and (not (vectorp mail-abbrevs))
(file-exists-p mail-personal-alias-file))
(build-mail-abbrevs))
(set-syntax-table mail-abbrev-syntax-table)
(goto-char before)
(while (and (< (point) end)
(progn (forward-word 1)
(<= (point) end)))
(expand-abbrev))
(set-syntax-table old-syntax-table))
(expand-mail-aliases before (point)))))
;;>> Set up comment, if any.
(if (and (sequencep comment) (not (zerop (length comment))))
(let ((before (point))
after)
(insert comment)
(or (eolp) (insert "\n"))
(setq after (point))
(goto-char before)
(while (< (point) after)
(insert "Resent-Comment: ")
(forward-line 1))))
;; Don't expand aliases in the destination fields
;; of the original message.
(let (mail-aliases)
(funcall send-mail-function)))
(kill-buffer tembuf))
(rmail-set-attribute "resent" t rmail-current-message)))
(defvar mail-unsent-separator
(concat "^ *---+ +Unsent message follows +---+ *$\\|"
"^ *---+ +Returned message +---+ *$\\|"
"^Start of returned message$\\|"
"^ *---+ +Original message +---+ *$\\|"
"^ *--+ +begin message +--+ *$\\|"
"^ *---+ +Original message follows +---+ *$\\|"
"^|? *---+ +Message text follows: +---+ *|?$")
"A regexp that matches the separator before the text of a failed message.")
(defun rmail-retry-failure ()
"Edit a mail message which is based on the contents of the current message.
For a message rejected by the mail system, extract the interesting headers and
the body of the original message.
The variable `mail-unsent-separator' should match the string that
delimits the returned original message.
The variable `rmail-retry-ignored-headers' is a regular expression
specifying headers which should not be copied into the new message."
(interactive)
(require 'mail-utils)
(let ((rmail-buffer (current-buffer))
(msgnum rmail-current-message)
bounce-start bounce-end bounce-indent resending)
(save-excursion
;; Narrow down to just the quoted original message
(rmail-beginning-of-message)
(let ((case-fold-search t))
(if (search-forward "This is a MIME-encapsulated message\n\n--" nil t)
(let ((codestring
(buffer-substring (progn (beginning-of-line) (point))
(progn (end-of-line) (point)))))
(re-search-forward mail-unsent-separator)
(search-forward codestring)
(or (search-forward "\n\n" nil t)
(error "Cannot find end of Mime data in failed message"))
(setq bounce-start (point))
(save-excursion
(goto-char (point-max))
(search-backward codestring)
(setq bounce-end (point)))
(or (search-forward "\n\n" nil t)
(error "Cannot find end of header in failed message")))
(or (re-search-forward mail-unsent-separator nil t)
(error "Cannot parse this as a failure message"))
(skip-chars-forward "\n")
;; Support a style of failure message in which the original
;; message is indented, and included within lines saying
;; `Start of returned message' and `End of returned message'.
(if (looking-at " +Received:")
(progn
(setq bounce-start (point))
(skip-chars-forward " ")
(setq bounce-indent (- (current-column)))
(goto-char (point-max))
(re-search-backward "^End of returned message$" nil t)
(setq bounce-end (point)))
;; One message contained a few random lines before the old
;; message header. The first line of the message started with
;; two hyphens. A blank line followed these random lines.
;; The same line beginning with two hyphens was possibly
;; marking the end of the message.
(if (looking-at "^--")
(let ((boundary (buffer-substring-no-properties
(point)
(progn (end-of-line) (point)))))
(search-forward "\n\n")
(skip-chars-forward "\n")
(setq bounce-start (point))
(goto-char (point-max))
(search-backward (concat "\n\n" boundary) bounce-start t)
(setq bounce-end (point)))
(setq bounce-start (point)
bounce-end (point-max)))
(or (search-forward "\n\n" nil t)
(error "Cannot find end of header in failed message"))
))))
;; Start sending a new message; default header fields from the original.
;; Turn off the usual actions for initializing the message body
;; because we want to get only the text from the failure message.
(let ((action
;; This function will be called when the user sends the retry.
;; It will mark the bounce message as "retried".
(function (lambda ()
(let ((msgnum rmail-send-actions-rmail-msg-number))
(save-excursion
(set-buffer rmail-send-actions-rmail-buffer)
(if msgnum
(rmail-set-attribute "retried" t msgnum)))))))
mail-signature mail-setup-hook)
(if (rmail-start-mail nil nil nil nil nil rmail-buffer
(list (list action)))
;; Insert original text as initial text of new draft message.
;; Bind inhibit-read-only since the header delimiter
;; of the previous message was probably read-only.
(let ((inhibit-read-only t))
;; We keep the rmail buffer and message number in these
;; buffer-local vars in the sendmail buffer,
;; so that the rmail-only-expunge can relocate the message number.
(make-local-variable 'rmail-send-actions-rmail-buffer)
(make-local-variable 'rmail-send-actions-rmail-msg-number)
(setq rmail-send-actions-rmail-buffer rmail-buffer)
(setq rmail-send-actions-rmail-msg-number msgnum)
(erase-buffer)
(insert-buffer-substring rmail-buffer bounce-start bounce-end)
(goto-char (point-min))
(if bounce-indent
(indent-rigidly (point-min) (point-max) bounce-indent))
(rmail-clear-headers rmail-retry-ignored-headers)
(rmail-clear-headers "^sender:\\|^from:\\|^return-path:")
(goto-char (point-min))
(save-restriction
(search-forward "\n\n")
(forward-line -1)
(narrow-to-region (point-min) (point))
(setq resending (mail-fetch-field "resent-to"))
(if mail-self-blind
(if resending
(insert "Resent-Bcc: " (user-login-name) "\n")
(insert "BCC: " (user-login-name) "\n"))))
(insert mail-header-separator)
(mail-position-on-field (if resending "Resent-To" "To") t)
(set-buffer rmail-buffer)
(rmail-beginning-of-message))))))
(defun rmail-summary-exists ()
"Non-nil iff in an RMAIL buffer and an associated summary buffer exists.
In fact, the non-nil value returned is the summary buffer itself."
(and rmail-summary-buffer (buffer-name rmail-summary-buffer)
rmail-summary-buffer))
(defun rmail-summary-displayed ()
"t iff in RMAIL buffer and an associated summary buffer is displayed."
(and rmail-summary-buffer (get-buffer-window rmail-summary-buffer)))
(defvar rmail-redisplay-summary nil
"*Non-nil means Rmail should show the summary when it changes.
This has an effect only if a summary buffer exists.")
(defvar rmail-summary-window-size nil
"*Non-nil means specify the height for an Rmail summary window.")
;; Put the summary buffer back on the screen, if user wants that.
(defun rmail-maybe-display-summary ()
(let ((selected (selected-window))
window)
;; If requested, make sure the summary is displayed.
(and rmail-summary-buffer (buffer-name rmail-summary-buffer)
rmail-redisplay-summary
(if (get-buffer-window rmail-summary-buffer 0)
;; It's already in some frame; show that one.
(let ((frame (window-frame
(get-buffer-window rmail-summary-buffer 0))))
(make-frame-visible frame)
(raise-frame frame))
(display-buffer rmail-summary-buffer)))
;; If requested, set the height of the summary window.
(and rmail-summary-buffer (buffer-name rmail-summary-buffer)
rmail-summary-window-size
(setq window (get-buffer-window rmail-summary-buffer))
;; Don't try to change the size if just one window in frame.
(not (eq window (frame-root-window (window-frame window))))
(unwind-protect
(progn
(select-window window)
(enlarge-window (- rmail-summary-window-size
(window-height))))
(select-window selected)))))
;;;; *** Rmail Specify Inbox Files ***
(autoload 'set-rmail-inbox-list "rmailmsc"
"Set the inbox list of the current RMAIL file to FILE-NAME.
This may be a list of file names separated by commas.
If FILE-NAME is empty, remove any inbox list."
t)
;;;; *** Rmail Commands for Labels ***
(autoload 'rmail-add-label "rmailkwd"
"Add LABEL to labels associated with current RMAIL message.
Completion is performed over known labels when reading."
t)
(autoload 'rmail-kill-label "rmailkwd"
"Remove LABEL from labels associated with current RMAIL message.
Completion is performed over known labels when reading."
t)
(autoload 'rmail-next-labeled-message "rmailkwd"
"Show next message with LABEL. Defaults to last label used.
With prefix argument N moves forward N messages with this label."
t)
(autoload 'rmail-previous-labeled-message "rmailkwd"
"Show previous message with LABEL. Defaults to last label used.
With prefix argument N moves backward N messages with this label."
t)
(autoload 'rmail-read-label "rmailkwd"
"PROMPT and read with completion an Rmail message label."
t)
;;;; *** Rmail Edit Mode ***
(autoload 'rmail-edit-current-message "rmailedit"
"Edit the contents of the current message"
t)
;;;; *** Rmail Sorting ***
(autoload 'rmail-sort-by-date "rmailsort"
"Sort messages of current Rmail file by date.
If prefix argument REVERSE is non-nil, sort them in reverse order." t)
(autoload 'rmail-sort-by-subject "rmailsort"
"Sort messages of current Rmail file by subject.
If prefix argument REVERSE is non-nil, sort them in reverse order." t)
(autoload 'rmail-sort-by-author "rmailsort"
"Sort messages of current Rmail file by author.
If prefix argument REVERSE is non-nil, sort them in reverse order." t)
(autoload 'rmail-sort-by-recipient "rmailsort"
"Sort messages of current Rmail file by recipient.
If prefix argument REVERSE is non-nil, sort them in reverse order." t)
(autoload 'rmail-sort-by-correspondent "rmailsort"
"Sort messages of current Rmail file by other correspondent.
If prefix argument REVERSE is non-nil, sort them in reverse order." t)
(autoload 'rmail-sort-by-lines "rmailsort"
"Sort messages of current Rmail file by number of lines.
If prefix argument REVERSE is non-nil, sort them in reverse order." t)
(autoload 'rmail-sort-by-keywords "rmailsort"
"Sort messages of current Rmail file by labels.
If prefix argument REVERSE is non-nil, sort them in reverse order.
KEYWORDS is a comma-separated list of labels." t)
;;;; *** Rmail Summary Mode ***
(autoload 'rmail-summary "rmailsum"
"Display a summary of all messages, one line per message."
t)
(autoload 'rmail-summary-by-labels "rmailsum"
"Display a summary of all messages with one or more LABELS.
LABELS should be a string containing the desired labels, separated by commas."
t)
(autoload 'rmail-summary-by-recipients "rmailsum"
"Display a summary of all messages with the given RECIPIENTS.
Normally checks the To, From and Cc fields of headers; but if PRIMARY-ONLY
is non-nil (prefix arg given), only look in the To and From fields.
RECIPIENTS is a string of regexps separated by commas."
t)
(autoload 'rmail-summary-by-regexp "rmailsum"
"Display a summary of all messages according to regexp REGEXP.
If the regular expression is found in the header of the message
\(including in the date and other lines, as well as the subject line),
Emacs will list the header line in the RMAIL-summary."
t)
(autoload 'rmail-summary-by-topic "rmailsum"
"Display a summary of all messages with the given SUBJECT.
Normally checks the Subject field of headers;
but if WHOLE-MESSAGE is non-nil (prefix arg given),
look in the whole message.
SUBJECT is a string of regexps separated by commas."
t)
(autoload 'rmail-summary-by-sender "rmailsum"
"Display a summary of all messages with the given SENDERS.
SENDERS is a string of names separated by commas."
t)
;;;; *** Rmail output messages to files ***
(autoload 'rmail-output-to-rmail-file "rmailout"
"Append the current message to an Rmail file named FILE-NAME.
If the file does not exist, ask if it should be created.
If file is being visited, the message is appended to the Emacs
buffer visiting that file."
t)
(autoload 'rmail-output "rmailout"
"Append this message to Unix mail file named FILE-NAME."
t)
;;;; *** Rmail undigestification ***
(autoload 'undigestify-rmail-message "undigest"
"Break up a digest message into its constituent messages.
Leaves original message, deleted, before the undigestified messages."
t)
(provide 'rmail)
;;; rmail.el ends here
|