c8a27ad7cc2c310a38a8961e1bcfb6ad924769d2
[notmuch] / vim / plugin / notmuch.vim
1 " notmuch.vim plugin --- run notmuch within vim
2 "
3 " Copyright © Carl Worth
4 "
5 " This file is part of Notmuch.
6 "
7 " Notmuch is free software: you can redistribute it and/or modify it
8 " under the terms of the GNU General Public License as published by
9 " the Free Software Foundation, either version 3 of the License, or
10 " (at your option) any later version.
11 "
12 " Notmuch is distributed in the hope that it will be useful, but
13 " WITHOUT ANY WARRANTY; without even the implied warranty of
14 " MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15 " General Public License for more details.
16 "
17 " You should have received a copy of the GNU General Public License
18 " along with Notmuch.  If not, see <http://www.gnu.org/licenses/>.
19 "
20 " Authors: Bart Trojanowski <bart@jukie.net>
21
22 " --- configuration defaults {{{1
23
24 let s:notmuch_defaults = {
25         \ 'g:notmuch_cmd':                           'notmuch'                    ,
26         \ 'g:notmuch_debug':                         0                            ,
27         \
28         \ 'g:notmuch_search_newest_first':           1                            ,
29         \ 'g:notmuch_search_from_column_width':      20                           ,
30         \
31         \ 'g:notmuch_show_fold_signatures':          1                            ,
32         \ 'g:notmuch_show_fold_citations':           1                            ,
33         \ 'g:notmuch_show_fold_bodies':              0                            ,
34         \ 'g:notmuch_show_fold_headers':             1                            ,
35         \
36         \ 'g:notmuch_show_message_begin_regexp':     '^\fmessage{'                ,
37         \ 'g:notmuch_show_message_end_regexp':       '^\fmessage}'                ,
38         \ 'g:notmuch_show_header_begin_regexp':      '^\fheader{'                 ,
39         \ 'g:notmuch_show_header_end_regexp':        '^\fheader}'                 ,
40         \ 'g:notmuch_show_body_begin_regexp':        '^\fbody{'                   ,
41         \ 'g:notmuch_show_body_end_regexp':          '^\fbody}'                   ,
42         \ 'g:notmuch_show_attachment_begin_regexp':  '^\fattachment{'             ,
43         \ 'g:notmuch_show_attachment_end_regexp':    '^\fattachment}'             ,
44         \ 'g:notmuch_show_part_begin_regexp':        '^\fpart{'                   ,
45         \ 'g:notmuch_show_part_end_regexp':          '^\fpart}'                   ,
46         \ 'g:notmuch_show_marker_regexp':            '^\f\\(message\\|header\\|body\\|attachment\\|part\\)[{}].*$',
47         \
48         \ 'g:notmuch_show_message_parse_regexp':     '\(id:[^ ]*\) depth:\([0-9]*\) match:\([0-9]*\) filename:\(.*\)$',
49         \ 'g:notmuch_show_tags_regexp':              '(\([^)]*\))$'               ,
50         \
51         \ 'g:notmuch_show_signature_regexp':         '^\(-- \?\|_\+\)$'           ,
52         \ 'g:notmuch_show_signature_lines_max':      12                           ,
53         \
54         \ 'g:notmuch_show_citation_regexp':          '^\s*>'                      ,
55         \ }
56
57 " defaults for g:notmuch_initial_search_words
58 " override with: let g:notmuch_initial_search_words = [ ... ]
59 let s:notmuch_initial_search_words_defaults = [
60         \ 'tag:inbox and tag:unread',
61         \ ]
62
63 " defaults for g:notmuch_show_headers
64 " override with: let g:notmuch_show_headers = [ ... ]
65 let s:notmuch_show_headers_defaults = [
66         \ 'Subject',
67         \ 'To',
68         \ 'Cc',
69         \ 'Bcc',
70         \ 'Date',
71         \ ]
72
73 " defaults for g:notmuch_folders
74 " override with: let g:notmuch_folders = [ ... ]
75 let s:notmuch_folders_defaults = [
76         \ [ 'new',    'tag:inbox and tag:unread' ],
77         \ [ 'inbox',  'tag:inbox'                ],
78         \ [ 'unread', 'tag:unread'               ],
79         \ ]
80
81 " --- keyboard mapping definitions {{{1
82
83 " --- --- bindings for folders mode {{{2
84
85 let g:notmuch_folders_maps = {
86         \ 's':          ':call <SID>NM_search_prompt()<CR>',
87         \ 'q':          ':call <SID>NM_kill_this_buffer()<CR>',
88         \ '=':          ':call <SID>NM_folders_refresh_view()<CR>',
89         \ '<Enter>':    ':call <SID>NM_folders_show_search()<CR>',
90         \ }
91
92 " --- --- bindings for search screen {{{2
93 let g:notmuch_search_maps = {
94         \ '<Enter>':    ':call <SID>NM_search_show_thread()<CR>',
95         \ '<C-]>':      ':call <SID>NM_search_expand(''<cword>'')<CR>',
96         \ 'a':          ':call <SID>NM_search_archive_thread()<CR>',
97         \ 'f':          ':call <SID>NM_search_filter()<CR>',
98         \ 'm':          ':call <SID>NM_new_mail()<CR>',
99         \ 'o':          ':call <SID>NM_search_toggle_order()<CR>',
100         \ 'r':          ':call <SID>NM_search_reply_to_thread()<CR>',
101         \ 's':          ':call <SID>NM_search_prompt()<CR>',
102         \ ',s':         ':call <SID>NM_search_edit()<CR>',
103         \ 't':          ':call <SID>NM_search_filter_by_tag()<CR>',
104         \ 'q':          ':call <SID>NM_kill_this_buffer()<CR>',
105         \ '+':          ':call <SID>NM_search_add_tags([])<CR>',
106         \ '-':          ':call <SID>NM_search_remove_tags([])<CR>',
107         \ '=':          ':call <SID>NM_search_refresh_view()<CR>',
108         \ '?':          ':echo <SID>NM_search_thread_id()<CR>',
109         \ }
110
111 " --- --- bindings for show screen {{{2
112 let g:notmuch_show_maps = {
113         \ '<C-P>':      ':call <SID>NM_show_previous(1, 0)<CR>',
114         \ '<C-N>':      ':call <SID>NM_show_next(1, 0)<CR>',
115         \ '<C-]>':      ':call <SID>NM_search_expand(''<cword>'')<CR>',
116         \ 'q':          ':call <SID>NM_kill_this_buffer()<CR>',
117         \
118         \ 'b':          ':call <SID>NM_show_fold_toggle(''b'', ''bdy'', !g:notmuch_show_fold_bodies)<CR>',
119         \ 'c':          ':call <SID>NM_show_fold_toggle(''c'', ''cit'', !g:notmuch_show_fold_citations)<CR>',
120         \ 'h':          ':call <SID>NM_show_fold_toggle(''h'', ''hdr'', !g:notmuch_show_fold_headers)<CR>',
121         \ 's':          ':call <SID>NM_show_fold_toggle(''s'', ''sig'', !g:notmuch_show_fold_signatures)<CR>',
122         \
123         \ 'a':          ':call <SID>NM_show_archive_thread()<CR>',
124         \ 'A':          ':call <SID>NM_show_mark_read_then_archive_thread()<CR>',
125         \ 'N':          ':call <SID>NM_show_mark_read_then_next_open_message()<CR>',
126         \ 'v':          ':call <SID>NM_show_view_all_mime_parts()<CR>',
127         \ '+':          ':call <SID>NM_show_add_tag()<CR>',
128         \ '-':          ':call <SID>NM_show_remove_tag()<CR>',
129         \ '<Space>':    ':call <SID>NM_show_advance_marking_read_and_archiving()<CR>',
130         \ '\|':         ':call <SID>NM_show_pipe_message()<CR>',
131         \
132         \ '<S-Tab>':    ':call <SID>NM_show_previous_fold()<CR>',
133         \ '<Tab>':      ':call <SID>NM_show_next_fold()<CR>',
134         \ '<Enter>':    ':call <SID>NM_show_toggle_fold()<CR>',
135         \
136         \ 'r':          ':call <SID>NM_show_reply()<CR>',
137         \ 'm':          ':call <SID>NM_new_mail()<CR>',
138         \ '?':          ':echo <SID>NM_show_message_id() . ''  @ '' . join(<SID>NM_show_search_words())<CR>',
139         \ }
140
141
142 " --- implement folders screen {{{1
143
144 function! s:NM_cmd_folders(words)
145         if len(a:words)
146                 echoe 'Not exapecting any arguments for folders command.'
147         endif
148         let cmd = ['count']
149         let disp = []
150         let searches = []
151         for entry in g:notmuch_folders
152                 let [ name, search ] = entry
153                 let data = s:NM_run(cmd + [search])
154                 let cnt = matchlist(data, '\(\d\+\)')[1]
155                 call add(disp, printf('%9d %-20s (%s)', cnt, name, search))
156                 call add(searches, search)
157         endfor
158
159         call <SID>NM_newBuffer('folders', join(disp, "\n"))
160         let b:nm_searches = searches
161         let b:nm_timestamp = reltime()
162
163         call <SID>NM_cmd_folders_mksyntax()
164         call <SID>NM_set_map(g:notmuch_folders_maps)
165         setlocal cursorline
166         setlocal nowrap
167 endfunction
168
169 function! s:NM_cmd_folders_mksyntax()
170 endfunction
171
172 " --- --- folders screen action functions {{{2
173
174 function! s:NM_folders_refresh_view()
175         let lno = line('.')
176         setlocal bufhidden=delete
177         call s:NM_cmd_folders([])
178         exec printf('norm %dG', lno)
179 endfunction
180
181 function! s:NM_folders_show_search()
182         let line = line('.')
183         let search = b:nm_searches[line-1]
184
185         let prev_bufnr = bufnr('%')
186         setlocal bufhidden=hide
187         call <SID>NM_cmd_search([search])
188         setlocal bufhidden=delete
189         let b:nm_prev_bufnr = prev_bufnr
190 endfunction
191
192
193 " --- implement search screen {{{1
194
195 function! s:NM_cmd_search(words)
196         let cmd = ['search']
197         if g:notmuch_search_newest_first
198                 let cmd = cmd + ['--sort=newest-first']
199         else
200                 let cmd = cmd + ['--sort=oldest-first']
201         endif
202         let data = s:NM_run(cmd + a:words)
203         let lines = split(data, "\n")
204         let disp = copy(lines)
205         call map(disp, 's:NM_cmd_search_fmtline(v:val)')
206
207         call <SID>NM_newBuffer('search', join(disp, "\n"))
208         let b:nm_raw_lines = lines
209         let b:nm_search_words = a:words
210
211         call <SID>NM_cmd_search_mksyntax()
212         call <SID>NM_set_map(g:notmuch_search_maps)
213         setlocal cursorline
214         setlocal nowrap
215 endfunction
216 function! s:NM_cmd_search_fmtline(line)
217         let m = matchlist(a:line, '^\(thread:\S\+\)\s\([^]]\+\]\) \([^;]\+\); \(.*\) (\([^(]*\))$')
218         if !len(m)
219                 return 'ERROR PARSING: ' . a:line
220         endif
221         let max = g:notmuch_search_from_column_width
222         let from = m[3]
223         if strlen(from) >= max
224                 let from = substitute(m[3][0:max-4], '[^A-Za-z1-9_]*$', '', '') . '...'
225         endif
226         return printf('%-20s %-20s | %s (%s)', m[2], from, m[4], m[5])
227 endfunction
228 function! s:NM_cmd_search_mksyntax()
229         syntax clear nmSearchFrom
230         exec printf('syntax match nmSearchFrom /\(\] \)\@<=.\{%d\}/ oneline contained', g:notmuch_search_from_column_width)
231 endfunction
232
233 " --- --- search screen action functions {{{2
234
235 function! s:NM_search_show_thread()
236         let id = <SID>NM_search_thread_id()
237         if id != ''
238                 let words = [id]
239                 if exists('b:nm_search_words')
240                         let words = ['('] + b:nm_search_words + [')', 'and', id]
241                 endif
242                 if len(words)
243                         call <SID>NM_cmd_show(words)
244                 endif
245         endif
246 endfunction
247
248 function! s:NM_search_prompt()
249         " TODO: input() can support completion
250         let text = input('NotMuch Search: ')
251         if strlen(text)
252                 let tags = split(text)
253         else
254                 let tags = s:notmuch_initial_search_words_defaults
255         endif
256         let prev_bufnr = bufnr('%')
257         if b:nm_type == 'search'
258                 " TODO: we intend to replace the current buffer,
259                 "       ... maybe we could just clear it
260                 let prev_bufnr = b:nm_prev_bufnr
261                 setlocal bufhidden=delete
262         else
263                 setlocal bufhidden=hide
264         endif
265         call <SID>NM_cmd_search(tags)
266         setlocal bufhidden=delete
267         let b:nm_prev_bufnr = prev_bufnr
268 endfunction
269
270 function! s:NM_search_edit()
271         " TODO: input() can support completion
272         let text = input('NotMuch Search: ', join(b:nm_search_words, ' '))
273         if strlen(text)
274                 call <SID>NM_cmd_search(split(text))
275         endif
276 endfunction
277
278 function! s:NM_search_archive_thread()
279         call <SID>NM_add_remove_tags_on_screen('', '-', ['inbox'])
280         call <SID>NM_add_remove_tags([], '-', ['inbox'])
281         norm j
282 endfunction
283
284 function! s:NM_search_filter()
285         call <SID>NM_search_filter_helper('Filter: ', '', '')
286 endfunction
287
288 function! s:NM_search_filter_by_tag()
289         call <SID>NM_search_filter_helper('Filter Tag(s): ', 'tag:', 'and')
290 endfunction
291
292 function! s:NM_search_filter_helper(prompt, prefix, joiner)
293         " TODO: input() can support completion
294         let text = input(a:prompt)
295         if !strlen(text)
296                 return
297         endif
298
299         let tags = split(text)
300         if strlen(a:prefix)
301                 call map(tags, 'a:prefix . v:val')
302         endif
303         if strlen(a:joiner)
304                 let idx = len(tags) - 1
305                 while idx > 0
306                         call insert(tags, a:joiner, idx)
307                         let idx = idx - 1
308                 endwhile
309         endif
310         let tags = b:nm_search_words + ['and', '('] + tags + [')']
311
312         let prev_bufnr = bufnr('%')
313         setlocal bufhidden=hide
314         call <SID>NM_cmd_search(tags)
315         setlocal bufhidden=delete
316         let b:nm_prev_bufnr = prev_bufnr
317 endfunction
318
319 function! s:NM_search_toggle_order()
320         let g:notmuch_search_newest_first = !g:notmuch_search_newest_first
321         " FIXME: maybe this would be better done w/o reading re-reading the lines
322         "         reversing the b:nm_raw_lines and the buffer lines would be better
323         call <SID>NM_search_refresh_view()
324 endfunction
325
326 function! s:NM_search_reply_to_thread()
327         echo 'not implemented'
328 endfunction
329
330 function! s:NM_search_add_tags(tags)
331         call <SID>NM_search_add_remove_tags('Add Tag(s): ', '+', a:tags)
332 endfunction
333
334 function! s:NM_search_remove_tags(tags)
335         call <SID>NM_search_add_remove_tags('Remove Tag(s): ', '-', a:tags)
336 endfunction
337
338 function! s:NM_search_refresh_view()
339         let lno = line('.')
340         let prev_bufnr = b:nm_prev_bufnr
341         setlocal bufhidden=delete
342         call <SID>NM_cmd_search(b:nm_search_words)
343         let b:nm_prev_bufnr = prev_bufnr
344         " FIXME: should find the line of the thread we were on if possible
345         exec printf('norm %dG', lno)
346 endfunction
347
348 " --- --- search screen helper functions {{{2
349
350 function! s:NM_search_thread_id()
351         if !exists('b:nm_raw_lines')
352                 echoe 'no b:nm_raw_lines'
353                 return ''
354         else
355                 let line = line('.')
356                 let info = b:nm_raw_lines[line-1]
357                 let what = split(info, '\s\+')[0]
358                 return what
359         endif
360 endfunction
361
362 function! s:NM_search_add_remove_tags(prompt, prefix, intags)
363         if type(a:intags) != type([]) || len(a:intags) == 0
364                 " TODO: input() can support completion
365                 let text = input(a:prompt)
366                 if !strlen(text)
367                         return
368                 endif
369                 let tags = split(text, ' ')
370         else
371                 let tags = a:intags
372         endif
373         call <SID>NM_add_remove_tags([], a:prefix, tags)
374         call <SID>NM_add_remove_tags_on_screen('', a:prefix, tags)
375 endfunction
376
377 " --- implement show screen {{{1
378
379 function! s:NM_cmd_show(words)
380         let prev_bufnr = bufnr('%')
381         let data = s:NM_run(['show'] + a:words)
382         let lines = split(data, "\n")
383
384         let info = s:NM_cmd_show_parse(lines)
385
386         setlocal bufhidden=hide
387         call <SID>NM_newBuffer('show', join(info['disp'], "\n"))
388         setlocal bufhidden=delete
389         let b:nm_words = a:words
390         let b:nm_raw_info = info
391         let b:nm_prev_bufnr = prev_bufnr
392
393         call <SID>NM_cmd_show_mkfolds()
394         call <SID>NM_cmd_show_mksyntax()
395         call <SID>NM_set_map(g:notmuch_show_maps)
396         setlocal foldtext=NM_cmd_show_foldtext()
397         setlocal fillchars=
398         setlocal foldcolumn=6
399
400 endfunction
401
402 function! s:NM_show_previous(can_change_thread, find_matching)
403         let info = b:nm_raw_info
404         let lnum = line('.')
405         for msg in reverse(copy(info['msgs']))
406                 if a:find_matching && msg['match'] == '0'
407                         continue
408                 endif
409                 if lnum <= msg['start']
410                         continue
411                 endif
412
413                 exec printf('norm %dGzt', msg['start'])
414                 " TODO: try to fit the message on screen
415                 return
416         endfor
417         if !a:can_change_thread
418                 return
419         endif
420         call <SID>NM_kill_this_buffer()
421         if line('.') > 1
422                 norm k
423                 call <SID>NM_search_show_thread()
424                 norm G
425                 call <SID>NM_show_previous(0, a:find_matching)
426         else
427                 echo 'No more messages.'
428         endif
429 endfunction
430
431 function! s:NM_show_next(can_change_thread, find_matching)
432         let info = b:nm_raw_info
433         let lnum = line('.')
434         for msg in info['msgs']
435                 if a:find_matching && msg['match'] == '0'
436                         continue
437                 endif
438                 if lnum >= msg['start']
439                         continue
440                 endif
441
442                 exec printf('norm %dGzt', msg['start'])
443                 " TODO: try to fit the message on screen
444                 return
445         endfor
446         if a:can_change_thread
447                 call <SID>NM_show_next_thread()
448         endif
449 endfunction
450
451 function! s:NM_show_next_thread()
452         call <SID>NM_kill_this_buffer()
453         if line('.') != line('$')
454                 norm j
455                 call <SID>NM_search_show_thread()
456         else
457                 echo 'No more messages.'
458         endif
459 endfunction
460
461 function! s:NM_show_archive_thread()
462         echo 'not implemented'
463 endfunction
464
465 function! s:NM_show_mark_read_then_archive_thread()
466         echo 'not implemented'
467 endfunction
468
469 function! s:NM_show_mark_read_then_next_open_message()
470         echo 'not implemented'
471 endfunction
472
473 function! s:NM_show_previous_message()
474         echo 'not implemented'
475 endfunction
476
477 function! s:NM_show_reply()
478         echo 'not implemented'
479 endfunction
480
481 function! s:NM_show_view_all_mime_parts()
482         echo 'not implemented'
483 endfunction
484
485 function! s:NM_show_view_raw_message()
486         echo 'not implemented'
487 endfunction
488
489 function! s:NM_show_add_tag()
490         echo 'not implemented'
491 endfunction
492
493 function! s:NM_show_remove_tag()
494         echo 'not implemented'
495 endfunction
496
497 " if entire message is not visible scroll down 1/2 page or less to get to the bottom of message
498 " otherwise go to next message
499 " any message that is viewed entirely has inbox and unread tags removed
500 function! s:NM_show_advance_marking_read_and_archiving()
501         let advance_tags = ['unread', 'inbox']
502
503         let vis_top = line('w0')
504         let vis_bot = line('w$')
505
506         let msg_top = <SID>NM_show_get_message_for_line(vis_top)
507         if !has_key(msg_top,'id')
508                 echo "No top visible message."
509         endif
510
511         " if the top message is the last message, just expunge the entire thread and move on
512         if msg_top['end'] == line('$')
513                 let ids = []
514                 for msg in b:nm_raw_info['msgs']
515                         if has_key(msg,'match') && msg['match'] != '0'
516                                 if len(ids)
517                                         call add(ids, 'OR')
518                                 endif
519                                 call add(ids, msg['id'])
520                         endif
521                 endfor
522
523                 let filter = ['('] + advance_tags + [')', 'AND', '('] + ids + [')']
524 echo 'NM_add_remove_tags ALL filter=' . string(filter)
525                 call <SID>NM_add_remove_tags(filter, '-', advance_tags)
526                 call <SID>NM_show_next(1, 1)
527                 return
528         endif
529
530         let msg_bot = <SID>NM_show_get_message_for_line(vis_bot)
531         if !has_key(msg_bot,'id')
532                 echo "No bottom visible message."
533         endif
534
535         echo 'top=' . msg_top['id'] . '  bot=' . msg_top['id']
536
537         " if entire message fits on the screen, read/archive it, move to the next one
538         if msg_top['id'] != msg_bot['id'] || msg_top['end'] <= vis_bot
539                 call <SID>NM_add_remove_tags_on_screen(msg_top['start'], '-', advance_tags)
540                 exec printf('norm %dG', vis_top)
541                 call <SID>NM_show_next(0, 1)
542                 if has_key(msg_top,'match') && msg_top['match'] != '0'
543                         redraw
544                         " do this last to hide the latency
545                         let filter = ['('] + advance_tags + [')', 'AND', msg_top['id']]
546 echo 'NM_add_remove_tags 1 filter=' . string(filter)
547                         call <SID>NM_add_remove_tags(filter, '-', advance_tags)
548                 endif
549                 return
550         endif
551
552         " entire message does not fit on the screen, scroll down to bottom, max 1/2 screen
553         let jmp = winheight(winnr()) / 2
554         let max = msg_bot['end'] - vis_bot
555         if jmp > max
556                 let jmp = max
557         endif
558         exec printf('norm %dGzt', vis_top + jmp)
559         return
560 endfunction
561
562 function! s:NM_show_pipe_message()
563         echo 'not implemented'
564 endfunction
565
566 function! s:NM_show_previous_fold()
567         echo 'not implemented'
568 endfunction
569
570 function! s:NM_show_next_fold()
571         echo 'not implemented'
572 endfunction
573
574 function! s:NM_show_toggle_fold()
575         echo 'not implemented'
576 endfunction
577
578
579 " --- --- show screen helper functions {{{2
580
581 function! s:NM_show_get_message_for_line(line)
582         for msg in b:nm_raw_info['msgs']
583                 if a:line > msg['end']
584                         continue
585                 endif
586                 return msg
587         endfor
588         return {}
589 endfunction
590
591 function! s:NM_show_message_id()
592         if !exists('b:nm_raw_info')
593                 echoe 'no b:nm_raw_info'
594                 return ''
595         endif
596         let msg = <SID>NM_show_get_message_for_line(line('.'))
597         if has_key(msg,'id')
598                 return msg['id']
599         endif
600         return ''
601 endfunction
602
603 function! s:NM_show_search_words()
604         if !exists('b:nm_words')
605                 echoe 'no b:nm_words'
606                 return []
607         endif
608         return b:nm_words
609 endfunction
610
611 function! s:NM_show_fold_toggle(key, type, fold)
612         let info = b:nm_raw_info
613         let act = 'open'
614         if a:fold
615                 let act = 'close'
616         endif
617         for fld in info['folds']
618                 if fld[0] == a:type
619                         exec printf('%dfold%s', fld[1], act)
620                 endif
621         endfor
622         exec printf('nnoremap <buffer> %s :call <SID>NM_show_fold_toggle(''%s'', ''%s'', %d)<CR>', a:key, a:key, a:type, !a:fold)
623 endfunction
624
625
626 " s:NM_cmd_show_parse returns the following dictionary:
627 "    'disp':     lines to display
628 "    'msgs':     message info dicts { start, end, id, depth, filename, descr, header }
629 "    'folds':    fold info arrays [ type, start, end ]
630 "    'foldtext': fold text indexed by start line
631 function! s:NM_cmd_show_parse(inlines)
632         let info = { 'disp': [],       
633                    \ 'msgs': [],       
634                    \ 'folds': [],      
635                    \ 'foldtext': {} }  
636         let msg = {}
637         let hdr = {}
638
639         let in_message = 0
640         let in_header = 0
641         let in_body = 0
642         let in_part = ''
643
644         let body_start = -1
645         let part_start = -1
646
647         let mode_type = ''
648         let mode_start = -1
649
650         let inlnum = 0
651         for line in a:inlines
652                 let inlnum = inlnum + 1
653                 let foldinfo = []
654
655                 if strlen(in_part)
656                         let part_end = 0
657
658                         if match(line, g:notmuch_show_part_end_regexp) != -1
659                                 let part_end = len(info['disp'])
660                         else
661                                 call add(info['disp'], line)
662                         endif
663
664                         if in_part == 'text/plain'
665                                 if !part_end && mode_type == ''
666                                         if match(line, g:notmuch_show_signature_regexp) != -1
667                                                 let mode_type = 'sig'
668                                                 let mode_start = len(info['disp'])
669                                         elseif match(line, g:notmuch_show_citation_regexp) != -1
670                                                 let mode_type = 'cit'
671                                                 let mode_start = len(info['disp'])
672                                         endif
673                                 elseif mode_type == 'cit'
674                                         if part_end || match(line, g:notmuch_show_citation_regexp) == -1
675                                                 let outlnum = len(info['disp'])
676                                                 let foldinfo = [ mode_type, mode_start, outlnum-1, len(info['msgs']),
677                                                                \ printf('[ %d-line citation.  Press "c" to show. ]', outlnum - mode_start) ]
678                                                 let mode_type = ''
679                                         endif
680                                 elseif mode_type == 'sig'
681                                         let outlnum = len(info['disp'])
682                                         if (outlnum - mode_start) > g:notmuch_show_signature_lines_max
683                                                 let mode_type = ''
684                                         elseif part_end
685                                                 let foldinfo = [ mode_type, mode_start, outlnum-1, len(info['msgs']),
686                                                                \ printf('[ %d-line signature.  Press "s" to show. ]', outlnum - mode_start) ]
687                                                 let mode_type = ''
688                                         endif
689                                 endif
690                         endif
691
692                         if part_end
693                                 " FIXME: this is a hack for handling two folds being added for one line
694                                 "         we should handle addinga fold in a function
695                                 if len(foldinfo) && foldinfo[1] < foldinfo[2]
696                                         call add(info['folds'], foldinfo[0:3])
697                                         let info['foldtext'][foldinfo[1]] = foldinfo[4]
698                                 endif
699
700                                 let foldinfo = [ 'text', part_start, part_end, len(info['msgs']),
701                                                \ printf('[ %d-line %s.  Press "p" to show. ]', part_end - part_start, in_part) ]
702                                 let in_part = ''
703                                 call add(info['disp'], '')
704                         endif
705
706                 elseif in_body
707                         if !has_key(msg,'body_start')
708                                 let msg['body_start'] = len(info['disp']) + 1
709                         endif
710                         if match(line, g:notmuch_show_body_end_regexp) != -1
711                                 let body_end = len(info['disp'])
712                                 let foldinfo = [ 'bdy', body_start, body_end, len(info['msgs']),
713                                                \ printf('[ BODY %d - %d lines ]', len(info['msgs']), body_end - body_start) ]
714
715                                 let in_body = 0
716
717                         elseif match(line, g:notmuch_show_part_begin_regexp) != -1
718                                 let m = matchlist(line, 'ID: \(\d\+\), Content-type: \(\S\+\)')
719                                 let in_part = 'unknown'
720                                 if len(m)
721                                         let in_part = m[2]
722                                 endif
723                                 call add(info['disp'],
724                                          \ printf('--- %s ---', in_part))
725                                 let part_start = len(info['disp']) + 1
726                         endif
727
728                 elseif in_header
729                         if in_header == 1
730                                 let msg['descr'] = line
731                                 call add(info['disp'], line)
732                                 let in_header = 2
733                                 let msg['hdr_start'] = len(info['disp']) + 1
734
735                         else
736                                 if match(line, g:notmuch_show_header_end_regexp) != -1
737                                         let hdr_start = msg['hdr_start']+1
738                                         let hdr_end = len(info['disp'])
739                                         let foldinfo = [ 'hdr', hdr_start, hdr_end, len(info['msgs']),
740                                                \ printf('[ %d-line headers.  Press "h" to show. ]', hdr_end + 1 - hdr_start) ]
741                                         let msg['header'] = hdr
742                                         let in_header = 0
743                                         let hdr = {}
744                                 else
745                                         let m = matchlist(line, '^\(\w\+\):\s*\(.*\)$')
746                                         if len(m)
747                                                 let hdr[m[1]] = m[2]
748                                                 if match(g:notmuch_show_headers, m[1]) != -1
749                                                         call add(info['disp'], line)
750                                                 endif
751                                         endif
752                                 endif
753                         endif
754
755                 elseif in_message
756                         if match(line, g:notmuch_show_message_end_regexp) != -1
757                                 let msg['end'] = len(info['disp'])
758                                 call add(info['disp'], '')
759
760                                 let foldinfo = [ 'msg', msg['start'], msg['end'], len(info['msgs']),
761                                                \ printf('[ MSG %d - %s ]', len(info['msgs']), msg['descr']) ]
762
763                                 call add(info['msgs'], msg)
764                                 let msg = {}
765                                 let in_message = 0
766                                 let in_header = 0
767                                 let in_body = 0
768                                 let in_part = ''
769
770                         elseif match(line, g:notmuch_show_header_begin_regexp) != -1
771                                 let in_header = 1
772                                 continue
773
774                         elseif match(line, g:notmuch_show_body_begin_regexp) != -1
775                                 let body_start = len(info['disp']) + 1
776                                 let in_body = 1
777                                 continue
778                         endif
779
780                 else
781                         if match(line, g:notmuch_show_message_begin_regexp) != -1
782                                 let msg['start'] = len(info['disp']) + 1
783
784                                 let m = matchlist(line, g:notmuch_show_message_parse_regexp)
785                                 if len(m)
786                                         let msg['id'] = m[1]
787                                         let msg['depth'] = m[2]
788                                         let msg['match'] = m[3]
789                                         let msg['filename'] = m[4]
790                                 endif
791
792                                 let in_message = 1
793                         endif
794                 endif
795
796                 if len(foldinfo) && foldinfo[1] < foldinfo[2]
797                         call add(info['folds'], foldinfo[0:3])
798                         let info['foldtext'][foldinfo[1]] = foldinfo[4]
799                 endif
800         endfor
801         return info
802 endfunction
803
804 function! s:NM_cmd_show_mkfolds()
805         let info = b:nm_raw_info
806
807         for afold in info['folds']
808                 exec printf('%d,%dfold', afold[1], afold[2])
809                 let state = 'open'
810                 if (afold[0] == 'sig' && g:notmuch_show_fold_signatures)
811                  \ || (afold[0] == 'cit' && g:notmuch_show_fold_citations)
812                  \ || (afold[0] == 'bdy' && g:notmuch_show_fold_bodies)
813                  \ || (afold[0] == 'hdr' && g:notmuch_show_fold_headers)
814                         let state = 'close'
815                 elseif afold[0] == 'msg'
816                         let idx = afold[3]
817                         let msg = info['msgs'][idx]
818                         if has_key(msg,'match') && msg['match'] == '0'
819                                 let state = 'close'
820                         endif
821                 endif
822                 exec printf('%dfold%s', afold[1], state)
823         endfor
824 endfunction
825
826 function! s:NM_cmd_show_mksyntax()
827         let info = b:nm_raw_info
828         let cnt = 0
829         for msg in info['msgs']
830                 let cnt = cnt + 1
831                 let start = msg['start']
832                 let hdr_start = msg['hdr_start']
833                 let body_start = msg['body_start']
834                 let end = msg['end']
835                 exec printf('syntax region nmShowMsg%dDesc start=''\%%%dl'' end=''\%%%dl'' contains=@nmShowMsgDesc', cnt, start, start+1)
836                 exec printf('syntax region nmShowMsg%dHead start=''\%%%dl'' end=''\%%%dl'' contains=@nmShowMsgHead', cnt, hdr_start, body_start)
837                 exec printf('syntax region nmShowMsg%dBody start=''\%%%dl'' end=''\%%%dl'' contains=@nmShowMsgBody', cnt, body_start, end)
838         endfor
839 endfunction
840
841 function! NM_cmd_show_foldtext()
842         let foldtext = b:nm_raw_info['foldtext']
843         return foldtext[v:foldstart]
844 endfunction
845
846
847 " --- notmuch helper functions {{{1
848
849 function! s:NM_newBuffer(type, content)
850         enew
851         setlocal buftype=nofile readonly modifiable
852         silent put=a:content
853         keepjumps 0d
854         setlocal nomodifiable
855         set scrolloff=0
856         set sidescrolloff=0
857         execute printf('set filetype=notmuch-%s', a:type)
858         execute printf('set syntax=notmuch-%s', a:type)
859         let b:nm_type = a:type
860 endfunction
861
862 function! s:NM_shell_escape(word)
863         let word = substitute(a:word, '''', '\\''', 'g')
864         return '''' . word . ''''
865 endfunction
866
867 function! s:NM_run(args)
868         let words = a:args
869         call map(words, 's:NM_shell_escape(v:val)')
870         let cmd = g:notmuch_cmd . ' ' . join(words) . '< /dev/null'
871
872         let start = reltime()
873         let out = system(cmd)
874         let err = v:shell_error
875         let delta = reltime(start)
876
877         if exists('g:notmuch_debug') && g:notmuch_debug
878                 echo printf('[%s] {%s} %s', reltimestr(delta), string(err), string(cmd))
879         endif
880
881         if err
882                 echohl Error
883                 echo substitute(out, '\n*$', '', '')
884                 echohl None
885                 return ''
886         else
887                 return out
888         endif
889 endfunction
890
891 " --- external mail handling helpers {{{1
892
893 function! s:NM_new_mail()
894         echo 'not implemented'
895 endfunction
896
897 " --- other helpers {{{1
898
899 function! s:NM_kill_this_buffer()
900         if exists('b:nm_prev_bufnr')
901                 setlocal bufhidden=delete
902                 exec printf(":buffer %d", b:nm_prev_bufnr)
903         else
904                 echo "This is the last buffer; use :q<CR> to quit."
905         endif
906 endfunction
907
908 function! s:NM_search_expand(arg)
909         let word = expand(a:arg)
910         let prev_bufnr = bufnr('%')
911         setlocal bufhidden=hide
912         call <SID>NM_cmd_search([word])
913         setlocal bufhidden=delete
914         let b:nm_prev_bufnr = prev_bufnr
915 endfunction
916
917 function! s:NM_add_remove_tags(filter, prefix, tags)
918         let filter = len(a:filter) ? a:filter : [<SID>NM_search_thread_id()]
919         if !len(filter)
920                 echoe 'Eeek! I couldn''t find the thead id!'
921         endif
922         echo 'filter = ' . string(filter) . ' ... ' . string(type(filter))
923         call map(a:tags, 'a:prefix . v:val')
924         " TODO: handle errors
925         let args = ['tag']
926         call extend(args, a:tags)
927         call add(args, '--')
928         call extend(args, filter)
929         echo 'NUM_run( ' . string(args) . ' )'
930         call <SID>NM_run(args)
931 endfunction
932
933 function! s:NM_add_remove_tags_on_screen(online, prefix, tags)
934         setlocal modifiable
935         if a:prefix == '-'
936                 for tagname in a:tags
937                         exec printf('silent! %ss/(\([^)]*\)\<%s\>\([^)]*\))$/(\1\2)/', string(a:online), tagname)
938                 endfor
939         else
940                 for tagname in a:tags
941                         exec printf('silent! %ss/(\([^)]*\)\([^)]*\))$/(\1 %s)/', string(a:online), tagname)
942                 endfor
943         endif
944         setlocal nomodifiable
945 endfunction
946
947 " --- process and set the defaults {{{1
948
949 function! NM_set_defaults(force)
950         for [key, dflt] in items(s:notmuch_defaults)
951                 let cmd = ''
952                 if !a:force && exists(key) && type(dflt) == type(eval(key))
953                         continue
954                 elseif type(dflt) == type(0)
955                         let cmd = printf('let %s = %d', key, dflt)
956                 elseif type(dflt) == type('')
957                         let cmd = printf('let %s = ''%s''', key, dflt)
958                 " FIXME: not sure why this didn't work when dflt is an array
959                 "elseif type(dflt) == type([])
960                 "        let cmd = printf('let %s = %s', key, string(dflt))
961                 else
962                         echoe printf('E: Unknown type in NM_set_defaults(%d) using [%s,%s]',
963                                                 \ a:force, key, string(dflt))
964                         continue
965                 endif
966                 exec cmd
967         endfor
968 endfunction
969 call NM_set_defaults(0)
970
971 " for some reason NM_set_defaults() didn't work for arrays...
972 if !exists('g:notmuch_show_headers')
973         let g:notmuch_show_headers = s:notmuch_show_headers_defaults
974 endif
975 if !exists('g:notmuch_initial_search_words')
976         let g:notmuch_initial_search_words = s:notmuch_initial_search_words_defaults
977 endif
978 if !exists('g:notmuch_folders')
979         let g:notmuch_folders = s:notmuch_folders_defaults
980 endif
981
982
983 " --- assign keymaps {{{1
984
985 function! s:NM_set_map(maps)
986         nmapclear
987         for [key, code] in items(a:maps)
988                 exec printf('nnoremap <buffer> %s %s', key, code)
989         endfor
990         " --- this is a hack for development :)
991         nnoremap ,nmr :source ~/.vim/plugin/notmuch.vim<CR>:call NotMuch('')<CR>
992 endfunction
993
994 " --- command handler {{{1
995
996 function! NotMuch(args)
997         let args = a:args
998         if !strlen(args)
999                 let args = 'folders'
1000         endif
1001
1002         let words = split(args)
1003         if words[0] == 'folders'
1004                 let words = words[1:]
1005                 call <SID>NM_cmd_folders(words)
1006         elseif words[0] == 'search'
1007                 if len(words) > 1
1008                         let words = words[1:]
1009                 elseif exists('b:nm_search_words')
1010                         let words = b:nm_search_words
1011                 else
1012                         let words = g:notmuch_initial_search_words
1013                 endif
1014                 call <SID>NM_cmd_search(words)
1015
1016         elseif words[0] == 'show'
1017                 echoe 'show is not yet implemented.'
1018         endif
1019 endfunction
1020 function! CompleteNotMuch(arg_lead, cmd_line, cursor_pos)
1021         return []
1022 endfunction
1023
1024
1025 " --- glue {{{1
1026
1027 command! -nargs=* -complete=customlist,CompleteNotMuch NotMuch call NotMuch(<q-args>)
1028 cabbrev  notmuch <c-r>=(getcmdtype()==':' && getcmdpos()==1 ? 'NotMuch' : 'notmuch')<CR>
1029
1030 " vim: set ft=vim ts=8 sw=8 et foldmethod=marker :