X-Git-Url: http://git.madism.org/?p=apps%2Fmadmutt.git;a=blobdiff_plain;f=sidebar.c;h=92a0d2a06c4c9e046b11c0d4d890599e3a651732;hp=8adea8faf098323e79b8209cadf2f832d9bb00b7;hb=23265c6d12ecb744cb38689e9d78a448089ac351;hpb=82c978b426a6669d432c22f0773d807e2a974747 diff --git a/sidebar.c b/sidebar.c index 8adea8f..92a0d2a 100644 --- a/sidebar.c +++ b/sidebar.c @@ -1,321 +1,472 @@ /* + * Copyright notice from original mutt: * Copyright (C) ????-2004 Justin Hibbits * Copyright (C) 2004 Thomer M. Gil - * - * This program 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 of the License, or - * (at your option) any later version. - * - * This program 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 this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA. - */ - - + * + * Parts were written/modified by: + * Rocco Rutte + * Nico Golde + * + * This file is part of mutt-ng, see http://www.muttng.org/. + * It's licensed under the GNU General Public License, + * please see the file GPL in the top level source directory. + */ #include "mutt.h" #include "mutt_menu.h" #include "mutt_curses.h" #include "sidebar.h" #include "buffy.h" -#include #include "keymap.h" -#include + +#include "lib/mem.h" +#include "lib/str.h" +#include "lib/intl.h" + +#include #include -/*BUFFY *CurBuffy = 0;*/ -static BUFFY *TopBuffy = 0; -static BUFFY *BottomBuffy = 0; +static int TopBuffy = 0; +static int CurBuffy = 0; static int known_lines = 0; -static bool initialized = false; -static int prev_show_value; +static short initialized = 0; +static short prev_show_value; static short saveSidebarWidth; -static char *entry = 0; -static int quick_log10(int n) +/* computes first entry to be shown */ +void calc_boundaries (int menu) { - int len = 0; - for (; n > 9; len++, n /= 10) - ; - return (++len); -} + int lines = 0; -static int cur_is_hidden (int maxline) -{ - int l = 0, seen = 0; - BUFFY* tmp = TopBuffy; - if (!CurBuffy) - return (0); - while (tmp && l < maxline && !seen) - { - if (strcmp (tmp->path, CurBuffy->path) == 0) - seen = 1; - else - tmp = tmp->next; - l++; - } - return (seen == 0 || l == maxline); + if (list_empty(Incoming)) + return; + if (CurBuffy < 0 || CurBuffy >= Incoming->length) + CurBuffy = 0; + if (TopBuffy < 0 || TopBuffy >= Incoming->length) + TopBuffy = 0; + /* correct known_lines if it has changed because of a window resize */ + /* if (known_lines != LINES) + known_lines = LINES; */ + + lines = LINES - 2 - (menu != MENU_PAGER || option (OPTSTATUSONTOP)); + known_lines = lines; + if (option (OPTSIDEBARNEWMAILONLY)) { + int i = CurBuffy; + TopBuffy = CurBuffy - 1; + while (i >= 0) { + if (((BUFFY*) Incoming->data[i])->new > 0) + TopBuffy = i; + i--; + } + } else + TopBuffy = CurBuffy - (CurBuffy % lines); + if (TopBuffy < 0) + TopBuffy = 0; } -void calc_boundaries (int menu) +static char *shortened_hierarchy (char *box, int maxlen) { - BUFFY *tmp = Incoming; + int dots = 0; + char *last_dot = NULL; + int i, j, len = str_len (box); + char *new_box; - if ( known_lines != LINES ) { - TopBuffy = BottomBuffy = 0; - known_lines = LINES; - } - for ( ; tmp->next != 0; tmp = tmp->next ) - tmp->next->prev = tmp; - - if ( TopBuffy == 0 && BottomBuffy == 0 ) - TopBuffy = Incoming; - if ( BottomBuffy == 0 ) { - int count = LINES - 2 - (menu != MENU_PAGER || option (OPTSTATUSONTOP)); - BottomBuffy = TopBuffy; - while ( --count && BottomBuffy->next ) - BottomBuffy = BottomBuffy->next; - } - else if ( TopBuffy == CurBuffy->next ) { - int count = LINES - 2 - (menu != MENU_PAGER); - BottomBuffy = CurBuffy; - tmp = BottomBuffy; - while ( --count && tmp->prev) - tmp = tmp->prev; - TopBuffy = tmp; - } - else if ( BottomBuffy == CurBuffy->prev ) { - int count = LINES - 2 - (menu != MENU_PAGER); - TopBuffy = CurBuffy; - tmp = TopBuffy; - while ( --count && tmp->next ) - tmp = tmp->next; - BottomBuffy = tmp; - } -} + if (!SidebarBoundary || !*SidebarBoundary) + return (str_dup (box)); -static char * shortened_hierarchy(char * box) { - int dots = 0; - char * last_dot; - int i,j; - char * new_box; - for (i=0;i maxlen) { + new_box[j++] = box[i + 1]; new_box[j] = 0; } else { - strcat(&new_box[j],last_dot); + strcat (&new_box[j], last_dot); break; } } } return new_box; } - return safe_strdup(box); + return str_dup (box); } -char *make_sidebar_entry(char *box, int size, int new) -{ - char *c; - int i = 0, dlen = mutt_strlen (SidebarDelim); +static const char* sidebar_number_format (char* dest, size_t destlen, char op, + const char* src, const char* fmt, + const char* ifstr, const char* elstr, + unsigned long data, format_flag flags) { + char tmp[SHORT_STRING]; + BUFFY* b = (BUFFY*) Incoming->data[data]; + int opt = flags & M_FORMAT_OPTIONAL; + int c = Context && str_eq (Context->path, b->path); - c = realloc(entry, SidebarWidth + 1); - if ( c ) entry = c; - entry[SidebarWidth] = 0; - for (; i < SidebarWidth; entry[i++] = ' ' ); -#if USE_IMAP - if (ImapHomeNamespace && strlen(ImapHomeNamespace)>0) { - if (strncmp(box,ImapHomeNamespace,strlen(ImapHomeNamespace))==0 && strcmp(box,ImapHomeNamespace)!=0) { - box+=strlen(ImapHomeNamespace)+1; - } - } -#endif - if (option(OPTSHORTENHIERARCHY)) { - box = shortened_hierarchy(box); + switch (op) { + /* deleted */ + case 'd': + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->deleted : 0); + } else if ((c && Context->deleted == 0) || !c) + opt = 0; + break; + /* flagged */ + case 'F': + case 'f': /* for compatibility */ + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->flagged : b->msg_flagged); + } else if ((c && Context->flagged == 0) || (!c && b->msg_flagged == 0)) + opt = 0; + break; + /* total */ + case 'c': /* for compatibility */ + case 'm': + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->msgcount : b->msgcount); + } else if ((c && Context->msgcount == 0) || (!c && b->msgcount == 0)) + opt = 0; + break; + /* total shown, i.e. not hidden by limit */ + case 'M': + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->vcount : 0); + } else if ((c && Context->vcount == 0) || !c) + opt = 0; + break; + /* new */ + case 'n': + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->new : b->new); + } else if ((c && Context->new == 0) || (!c && b->new == 0)) + opt = 0; + break; + /* unread */ + case 'u': + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->unread : b->msg_unread); + } else if ((c && Context->unread == 0) || (!c && b->msg_unread == 0)) + opt = 0; + break; + /* tagged */ + case 't': + if (!opt) { + snprintf (tmp, sizeof (tmp), "%%%sd", fmt); + snprintf (dest, destlen, tmp, c ? Context->tagged : 0); + } else if ((c && Context->tagged == 0) || !c) + opt = 0; + break; } - i = strlen(box); - strncpy( entry, box, i < SidebarWidth - dlen ? i :SidebarWidth - dlen); - if ( new ) - sprintf(entry + SidebarWidth - 3 - quick_log10(size) - dlen - quick_log10(new), - "% d(%d)", size, new); - else - sprintf( entry + SidebarWidth - 1 - quick_log10(size) - dlen, "% d", size); - if (option(OPTSHORTENHIERARCHY)) { - free(box); - } - return entry; + if (opt) + mutt_FormatString (dest, destlen, ifstr, sidebar_number_format, + data, M_FORMAT_OPTIONAL); + else if (flags & M_FORMAT_OPTIONAL) + mutt_FormatString (dest, destlen, elstr, sidebar_number_format, + data, M_FORMAT_OPTIONAL); + return (src); } -void set_curbuffy(char buf[LONG_STRING]) +int sidebar_need_count (void) { + if (!option (OPTMBOXPANE) || SidebarWidth == 0 || + !SidebarNumberFormat || !*SidebarNumberFormat) + return (0); + return (1); +} + +/* print single item + * returns: + * 0 item was not printed ('cause of $sidebar_newmail_only) + * 1 item was printed + */ +int make_sidebar_entry (char* box, int idx, size_t len) { - BUFFY* tmp = CurBuffy = Incoming; + int shortened = 0, lencnt = 0; + char no[SHORT_STRING], entry[SHORT_STRING]; +#if USE_IMAP + int l = str_len (ImapHomeNamespace); +#endif + int l_m = str_len (Maildir); - if (!Incoming) - return; + if (SidebarWidth > COLS) + SidebarWidth = COLS; - while(1) { - if(!strcmp(tmp->path, buf)) { - CurBuffy = tmp; - break; + if (option (OPTSIDEBARNEWMAILONLY) && box && Context && Context->path && + !str_eq (Context->path, box) && ((BUFFY*) Incoming->data[idx])->new == 0) + /* if $sidebar_newmail_only is set, don't display the + * box only if it's not the currently opened + * (i.e. always display the currently opened) */ + return (0); + + mutt_FormatString (no, len, NONULL (SidebarNumberFormat), + sidebar_number_format, idx, M_FORMAT_OPTIONAL); + lencnt = str_len (no); + memset (&entry, ' ', sizeof (entry)); + +#if USE_IMAP + if (l > 0 && str_ncmp (box, ImapHomeNamespace, l) == 0 && + str_len (box) > l) + box += l + 1; /* we're trimming the ImapHomeNamespace, the "+ 1" is for the separator */ + else +#endif + if (l_m > 0 && str_ncmp (box, Maildir, l_m) == 0 && + str_len (box) > l_m) { + box += l_m; + if (Maildir[strlen(Maildir)-1]!='/') { + box += 1; } + } else + box = basename (box); - if(tmp->next) - tmp = tmp->next; - else - break; + if (option (OPTSHORTENHIERARCHY) && str_len (box) > len-lencnt-1) { + box = shortened_hierarchy (box, len-lencnt-1); + shortened = 1; } + + snprintf (entry, len-lencnt, "%s", box); + entry[str_len (entry)] = ' '; + strncpy (entry + (len - lencnt), no, lencnt); + + addnstr (entry, len); + + if (shortened) + mem_free(&box); + + return (1); } -void set_buffystats (CONTEXT* Context) -{ - BUFFY* tmp = Incoming; - while (tmp) - { - if (strcmp (tmp->path, Context->path) == 0) - { - tmp->msg_unread = Context->unread; - tmp->msgcount = Context->msgcount; - break; - } - tmp = tmp->next; - } +/* returns folder name of currently + * selected folder for + */ +const char* sidebar_get_current (void) { + if (list_empty(Incoming)) + return (NULL); + return ((char*) ((BUFFY*) Incoming->data[CurBuffy])->path); +} + +/* internally sets item to buf */ +void sidebar_set_current (const char* buf) { + int i = buffy_lookup (buf); + if (i >= 0) + CurBuffy = i; } -int draw_sidebar(int menu) { +/* fix counters for a context + * FIXME since ctx must not be of our business, move it elsewhere + */ +void sidebar_set_buffystats (CONTEXT* Context) { + int i = 0; + BUFFY* tmp = NULL; + if (!Context || list_empty(Incoming) || (i = buffy_lookup (Context->path)) < 0) + return; + tmp = (BUFFY*) Incoming->data[i]; + tmp->new = Context->new; + tmp->msg_unread = Context->unread; + tmp->msgcount = Context->msgcount; + tmp->msg_flagged = Context->flagged; +} + +/* actually draws something + * FIXME this needs some clue when to do it + */ +int sidebar_draw (int menu) +{ - int lines = option(OPTHELP) ? 1 : 0; + int lines = option (OPTHELP) ? 1 : 0, draw_devider = 1, i = 0; BUFFY *tmp; - short delim_len = mutt_strlen (SidebarDelim); + short delim_len = str_len (SidebarDelim); + char blank[SHORT_STRING]; /* initialize first time */ - if(!initialized) { - prev_show_value = option(OPTMBOXPANE); + if (!initialized) { + prev_show_value = option (OPTMBOXPANE); saveSidebarWidth = SidebarWidth; - if(!option(OPTMBOXPANE)) SidebarWidth = 0; - initialized = true; + if (!option (OPTMBOXPANE)){ + SidebarWidth = 0; + draw_devider = 1; + } + initialized = 1; } /* save or restore the value SidebarWidth */ - if(prev_show_value != option(OPTMBOXPANE)) { - if(prev_show_value && !option(OPTMBOXPANE)) { + if (prev_show_value != option (OPTMBOXPANE)) { + if (prev_show_value && !option (OPTMBOXPANE)) { saveSidebarWidth = SidebarWidth; SidebarWidth = 0; - } else if(!prev_show_value && option(OPTMBOXPANE)) { + } + else if (!prev_show_value && option (OPTMBOXPANE)) { SidebarWidth = saveSidebarWidth; + /* after toggle: force recounting of all mail */ + buffy_check (2); } - prev_show_value = option(OPTMBOXPANE); + prev_show_value = option (OPTMBOXPANE); } - if ( SidebarWidth == 0 ) return 0; - - /* draw the divider */ - SETCOLOR(MT_COLOR_STATUS); - for (lines = option (OPTSTATUSONTOP) ? 0 : 1; - lines < LINES-1-(menu != MENU_PAGER || option (OPTSTATUSONTOP)); lines++ ) { - move(lines, SidebarWidth - delim_len); - addstr (NONULL (SidebarDelim)); + if (SidebarWidth > 0 && option (OPTMBOXPANE) + && str_len (SidebarDelim) >= SidebarWidth) { + mutt_error (_("Value for sidebar_delim is too long. Disabling sidebar.")); + sleep (2); + unset_option (OPTMBOXPANE); + return (0); } - SETCOLOR(MT_COLOR_NORMAL); - - if ( Incoming == 0 ) return 0; - lines = option(OPTHELP) ? 1 : 0; /* go back to the top */ - - if (cur_is_hidden (LINES-1-(menu != MENU_PAGER))) - CurBuffy = TopBuffy; - - if ( known_lines != LINES || TopBuffy == 0 || BottomBuffy == 0 ) - calc_boundaries(menu); - if ( CurBuffy == 0 ) CurBuffy = Incoming; - - tmp = TopBuffy; - for ( ; tmp && lines < LINES-1 - (menu != MENU_PAGER || option (OPTSTATUSONTOP)); tmp = tmp->next ) { - if ( tmp == CurBuffy ) - SETCOLOR(MT_COLOR_INDICATOR); - else if ( tmp->msg_unread > 0 ) - SETCOLOR(MT_COLOR_NEW); - else - SETCOLOR(MT_COLOR_NORMAL); - - move( lines, 0 ); - if ( Context && !strcmp( tmp->path, Context->path ) ) { - printw( "%.*s", SidebarWidth - delim_len, - make_sidebar_entry(basename(tmp->path), - Context->msgcount, Context->unread)); - tmp->msg_unread = Context->unread; - tmp->msgcount = Context->msgcount; + if (SidebarWidth == 0 || !option (OPTMBOXPANE)) + return 0; + /* draw devider only if necessary (if the sidebar becomes visible e.g.)*/ + if (draw_devider == 1){ + /* draw the divider */ + SETCOLOR (MT_COLOR_SIDEBAR); + for (lines = 1; + lines < LINES - 1 - (menu != MENU_PAGER || option (OPTSTATUSONTOP)); + lines++) { + move (lines, SidebarWidth - delim_len); + if (option (OPTASCIICHARS)) + addstr (NONULL (SidebarDelim)); + else if (!option (OPTASCIICHARS) && !str_cmp (SidebarDelim, "|")) + addch (ACS_VLINE); + else if ((Charset_is_utf8) && !str_cmp (SidebarDelim, "|")) + addstr ("\342\224\202"); + else + addstr (NONULL (SidebarDelim)); } + } + SETCOLOR (MT_COLOR_NORMAL); + + if (list_empty(Incoming)) + return 0; + lines = option (OPTHELP) ? 1 : 0; /* go back to the top */ + calc_boundaries (menu); + + /* actually print items */ + for (i = TopBuffy; i < Incoming->length && lines < LINES - 1 - + (menu != MENU_PAGER || option (OPTSTATUSONTOP)); i++) { + tmp = (BUFFY*) Incoming->data[i]; + + if (i == CurBuffy) + SETCOLOR (MT_COLOR_INDICATOR); + else if (tmp->new > 0) + SETCOLOR (MT_COLOR_NEW); + else if (tmp->msg_flagged > 0) + SETCOLOR (MT_COLOR_FLAGGED); else - printw( "%.*s", SidebarWidth - delim_len, - make_sidebar_entry(basename(tmp->path), - tmp->msgcount,tmp->msg_unread)); - lines++; + SETCOLOR (MT_COLOR_NORMAL); + + move (lines, 0); + lines += make_sidebar_entry (tmp->path, i, SidebarWidth-delim_len); } - SETCOLOR(MT_COLOR_NORMAL); - for ( ; lines < LINES - 1 - (menu != MENU_PAGER || option (OPTSTATUSONTOP)); lines++ ) { - int i = 0; - move( lines, 0 ); - for ( ; i < SidebarWidth - delim_len; i++ ) - addch(' '); + + /* fill with blanks to bottom */ + memset (&blank, ' ', sizeof (blank)); + SETCOLOR (MT_COLOR_NORMAL); + for (; lines < LINES - 1 - (menu != MENU_PAGER || option (OPTSTATUSONTOP)); lines++) { + move (lines, 0); + addnstr (blank, SidebarWidth-delim_len); } return 0; } -void scroll_sidebar(int op, int menu) -{ - if(!SidebarWidth) return; - if(!CurBuffy) return; +/* returns index of new item with new mail or -1 */ +static int exist_next_new () { + int i = 0; + if (list_empty(Incoming)) + return (-1); + i = CurBuffy + 1; + while (i < Incoming->length) + if (((BUFFY*) Incoming->data[i++])->new > 0) + return (i-1); + return (-1); +} + +/* returns index of prev item with new mail or -1 */ +static int exist_prev_new () { + int i = 0; + if (list_empty(Incoming)) + return (-1); + i = CurBuffy - 1; + while (i >= 0) + if (((BUFFY*) Incoming->data[i--])->new > 0) + return (i+1); + return (-1); +} + +void sidebar_scroll (int op, int menu) { + int i = 0; + + if (!SidebarWidth || list_empty(Incoming)) + return; switch (op) { - case OP_SIDEBAR_NEXT: - if ( CurBuffy->next == NULL ) return; - CurBuffy = CurBuffy->next; - break; - case OP_SIDEBAR_PREV: - if ( CurBuffy == Incoming ) return; - { - BUFFY *tmp = Incoming; - while ( tmp->next && strcmp(tmp->next->path, CurBuffy->path) ) tmp = tmp->next; - CurBuffy = tmp; + case OP_SIDEBAR_NEXT: + if (!option (OPTSIDEBARNEWMAILONLY)) { + if (CurBuffy + 1 == Incoming->length) { + mutt_error (_("You are on the last mailbox.")); + return; } + CurBuffy++; break; - case OP_SIDEBAR_SCROLL_UP: - CurBuffy = TopBuffy; - if ( CurBuffy != Incoming ) { - calc_boundaries(menu); - CurBuffy = CurBuffy->prev; - } - break; - case OP_SIDEBAR_SCROLL_DOWN: - CurBuffy = BottomBuffy; - if ( CurBuffy->next ) { - calc_boundaries(menu); - CurBuffy = CurBuffy->next; + } /* the fall-through is intentional */ + case OP_SIDEBAR_NEXT_NEW: + if ((i = exist_next_new ()) < 0) { + mutt_error (_("No next mailboxes with new mail.")); + return; + } + else + CurBuffy = i; + break; + case OP_SIDEBAR_PREV: + if (!option (OPTSIDEBARNEWMAILONLY)) { + if (CurBuffy == 0) { + mutt_error (_("You are on the first mailbox.")); + return; } + CurBuffy--; break; - default: + } /* the fall-through is intentional */ + case OP_SIDEBAR_PREV_NEW: + if ((i = exist_prev_new ()) < 0) { + mutt_error (_("No previous mailbox with new mail.")); + return; + } + else + CurBuffy = i; + break; + + case OP_SIDEBAR_SCROLL_UP: + if (CurBuffy == 0) { + mutt_error (_("You are on the first mailbox.")); + return; + } + CurBuffy -= known_lines; + if (CurBuffy < 0) + CurBuffy = 0; + break; + case OP_SIDEBAR_SCROLL_DOWN: + if (CurBuffy + 1 == Incoming->length) { + mutt_error (_("You are on the last mailbox.")); return; + } + CurBuffy += known_lines; + if (CurBuffy >= Incoming->length) + CurBuffy = Incoming->length - 1; + break; + default: + return; } - calc_boundaries(menu); - draw_sidebar(menu); + calc_boundaries (menu); + sidebar_draw (menu); }