mutt stable branch with some hacks
at jcs 243 lines 5.9 kB view raw
1/* 2 * Copyright (C) 1996-2000,2007 Michael R. Elkins <me@mutt.org> 3 * 4 * This program is free software; you can redistribute it and/or modify 5 * it under the terms of the GNU General Public License as published by 6 * the Free Software Foundation; either version 2 of the License, or 7 * (at your option) any later version. 8 * 9 * This program is distributed in the hope that it will be useful, 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 * GNU General Public License for more details. 13 * 14 * You should have received a copy of the GNU General Public License 15 * along with this program; if not, write to the Free Software 16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. 17 */ 18 19#if HAVE_CONFIG_H 20# include "config.h" 21#endif 22 23#include "mutt.h" 24#include "mutt_menu.h" 25#include "mapping.h" 26#include "sort.h" 27 28#include "mutt_idna.h" 29 30#include <string.h> 31#include <stdlib.h> 32#include <ctype.h> 33 34#define RSORT(x) (SortAlias & SORT_REVERSE) ? -x : x 35 36static const struct mapping_t AliasHelp[] = { 37 { N_("Exit"), OP_EXIT }, 38 { N_("Del"), OP_DELETE }, 39 { N_("Undel"), OP_UNDELETE }, 40 { N_("Select"), OP_GENERIC_SELECT_ENTRY }, 41 { N_("Help"), OP_HELP }, 42 { NULL, 0 } 43}; 44 45static const char * 46alias_format_str (char *dest, size_t destlen, size_t col, int cols, char op, const char *src, 47 const char *fmt, const char *ifstring, const char *elsestring, 48 unsigned long data, format_flag flags) 49{ 50 char tmp[SHORT_STRING], adr[SHORT_STRING]; 51 ALIAS *alias = (ALIAS *) data; 52 53 switch (op) 54 { 55 case 'f': 56 snprintf (tmp, sizeof (tmp), "%%%ss", fmt); 57 snprintf (dest, destlen, tmp, alias->del ? "D" : " "); 58 break; 59 case 'a': 60 mutt_format_s (dest, destlen, fmt, alias->name); 61 break; 62 case 'r': 63 adr[0] = 0; 64 rfc822_write_address (adr, sizeof (adr), alias->addr, 1); 65 snprintf (tmp, sizeof (tmp), "%%%ss", fmt); 66 snprintf (dest, destlen, tmp, adr); 67 break; 68 case 'n': 69 snprintf (tmp, sizeof (tmp), "%%%sd", fmt); 70 snprintf (dest, destlen, tmp, alias->num + 1); 71 break; 72 case 't': 73 dest[0] = alias->tagged ? '*' : ' '; 74 dest[1] = 0; 75 break; 76 } 77 78 return (src); 79} 80 81static void alias_entry (char *s, size_t slen, MUTTMENU *m, int num) 82{ 83 mutt_FormatString (s, slen, 0, MuttIndexWindow->cols, NONULL (AliasFmt), alias_format_str, (unsigned long) ((ALIAS **) m->data)[num], MUTT_FORMAT_ARROWCURSOR); 84} 85 86static int alias_tag (MUTTMENU *menu, int n, int m) 87{ 88 ALIAS *cur = ((ALIAS **) menu->data)[n]; 89 int ot = cur->tagged; 90 91 cur->tagged = (m >= 0 ? m : !cur->tagged); 92 93 return cur->tagged - ot; 94} 95 96static int alias_SortAlias (const void *a, const void *b) 97{ 98 ALIAS *pa = *(ALIAS **) a; 99 ALIAS *pb = *(ALIAS **) b; 100 int r = mutt_strcasecmp (pa->name, pb->name); 101 102 return (RSORT (r)); 103} 104 105static int alias_SortAddress (const void *a, const void *b) 106{ 107 ADDRESS *pa = (*(ALIAS **) a)->addr; 108 ADDRESS *pb = (*(ALIAS **) b)->addr; 109 int r; 110 111 if (pa == pb) 112 r = 0; 113 else if (pa == NULL) 114 r = -1; 115 else if (pb == NULL) 116 r = 1; 117 else if (pa->personal) 118 { 119 if (pb->personal) 120 r = mutt_strcasecmp (pa->personal, pb->personal); 121 else 122 r = 1; 123 } 124 else if (pb->personal) 125 r = -1; 126 else 127 r = ascii_strcasecmp (pa->mailbox, pb->mailbox); 128 return (RSORT (r)); 129} 130 131void mutt_alias_menu (char *buf, size_t buflen, ALIAS *aliases) 132{ 133 ALIAS *aliasp; 134 MUTTMENU *menu; 135 ALIAS **AliasTable = NULL; 136 int t = -1; 137 int i, done = 0; 138 int op; 139 char helpstr[LONG_STRING]; 140 141 int omax; 142 143 if (!aliases) 144 { 145 mutt_error _("You have no aliases!"); 146 return; 147 } 148 149 menu = mutt_new_menu (MENU_ALIAS); 150 menu->make_entry = alias_entry; 151 menu->tag = alias_tag; 152 menu->title = _("Aliases"); 153 menu->help = mutt_compile_help (helpstr, sizeof (helpstr), MENU_ALIAS, AliasHelp); 154 mutt_push_current_menu (menu); 155 156new_aliases: 157 158 omax = menu->max; 159 160 /* count the number of aliases */ 161 for (aliasp = aliases; aliasp; aliasp = aliasp->next) 162 { 163 aliasp->self->del = 0; 164 aliasp->self->tagged = 0; 165 menu->max++; 166 } 167 168 safe_realloc (&AliasTable, menu->max * sizeof (ALIAS *)); 169 menu->data = AliasTable; 170 171 for (i = omax, aliasp = aliases; aliasp; aliasp = aliasp->next, i++) 172 { 173 AliasTable[i] = aliasp->self; 174 aliases = aliasp; 175 } 176 177 if ((SortAlias & SORT_MASK) != SORT_ORDER) 178 { 179 qsort (AliasTable, i, sizeof (ALIAS *), 180 (SortAlias & SORT_MASK) == SORT_ADDRESS ? alias_SortAddress : alias_SortAlias); 181 } 182 183 for (i=0; i<menu->max; i++) AliasTable[i]->num = i; 184 185 while (!done) 186 { 187 if (aliases->next) 188 { 189 menu->redraw |= REDRAW_FULL; 190 aliases = aliases->next; 191 goto new_aliases; 192 } 193 194 switch ((op = mutt_menuLoop (menu))) 195 { 196 case OP_DELETE: 197 case OP_UNDELETE: 198 if (menu->tagprefix) 199 { 200 for (i = 0; i < menu->max; i++) 201 if (AliasTable[i]->tagged) 202 AliasTable[i]->del = (op == OP_DELETE) ? 1 : 0; 203 menu->redraw |= REDRAW_INDEX; 204 } 205 else 206 { 207 AliasTable[menu->current]->self->del = (op == OP_DELETE) ? 1 : 0; 208 menu->redraw |= REDRAW_CURRENT; 209 if (option (OPTRESOLVE) && menu->current < menu->max - 1) 210 { 211 menu->current++; 212 menu->redraw |= REDRAW_INDEX; 213 } 214 } 215 break; 216 case OP_GENERIC_SELECT_ENTRY: 217 t = menu->current; 218 /* fall through */ 219 case OP_EXIT: 220 done = 1; 221 break; 222 } 223 } 224 225 for (i = 0; i < menu->max; i++) 226 { 227 if (AliasTable[i]->tagged) 228 { 229 rfc822_write_address (buf, buflen, AliasTable[i]->addr, 1); 230 t = -1; 231 } 232 } 233 234 if (t != -1) 235 { 236 rfc822_write_address (buf, buflen, AliasTable[t]->addr, 1); 237 } 238 239 mutt_pop_current_menu (menu); 240 mutt_menuDestroy (&menu); 241 FREE (&AliasTable); 242 243}