/* code for autogenerating rules or filters from a message */ #include #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include "Evolution.h" #include "evolution-storage.h" #include "evolution-shell-component.h" #include "folder-browser.h" #include "mail-vfolder.h" #include "camel/camel.h" #include "filter/vfolder-context.h" #include "filter/vfolder-rule.h" #include "filter/vfolder-editor.h" #include "filter/filter-context.h" #include "filter/filter-filter.h" #include "filter/filter-editor.h" #include "filter/filter-option.h" enum { AUTO_SUBJECT = 0x01, AUTO_FROM = 0x02, AUTO_TO = 0x04, }; static void rule_match_recipients(RuleContext *context, FilterRule *rule, CamelInternetAddress *iaddr) { FilterPart *part; FilterElement *element; int i; const char *real, *addr; /* address types etc should handle multiple values */ for (i=0;camel_internet_address_get(iaddr, i, &real, &addr);i++) { part = rule_context_create_part(context, "to"); filter_rule_add_part((FilterRule *)rule, part); element = filter_part_find_element(part, "recipient-type"); filter_option_set_current((FilterOption *)element, "contains"); element = filter_part_find_element(part, "recipient"); filter_input_set_value((FilterInput *)element, addr); } } /* remove 're' part of a subject */ static const char * strip_re(const char *s) { const char *p; while (*s) { while (isspace(*s)) s++; if (s[0] == 0) break; if ((s[0] == 'r' || s[0]=='R') && (s[1] == 'e' || s[1]=='E')) { p = s+2; while (isdigit(*p) || (ispunct(*p) && (*p != ':'))) p++; if (*p==':') { s = p+1; } else break; } else break; } return s; } #if 0 int reg_match(char *str, char *regstr) { regex_t reg; int error; int ret; error = regcomp(®, regstr, REG_EXTENDED|REG_ICASE|REG_NOSUB); if (error != 0) { return 0; } error = regexec(®, str, 0, NULL, 0); regfree(®); return (error == 0); } #endif static void rule_add_subject(RuleContext *context, FilterRule *rule, const char *text) { FilterPart *part; FilterElement *element; /* dont match on empty strings ever */ if (*text == 0) return; part = rule_context_create_part(context, "subject"); filter_rule_add_part((FilterRule *)rule, part); element = filter_part_find_element(part, "subject-type"); filter_option_set_current((FilterOption *)element, "contains"); element = filter_part_find_element(part, "subject"); filter_input_set_value((FilterInput *)element, text); } static void rule_add_sender(RuleContext *context, FilterRule *rule, const char *text) { FilterPart *part; FilterElement *element; /* dont match on empty strings ever */ if (*text == 0) return; part = rule_context_create_part(context, "sender"); filter_rule_add_part((FilterRule *)rule, part); element = filter_part_find_element(part, "sender-type"); filter_option_set_current((FilterOption *)element, "contains"); element = filter_part_find_element(part, "sender"); filter_input_set_value((FilterInput *)element, text); } /* do a bunch of things on the subject to try and detect mailing lists, remove unneeded stuff, etc */ static void rule_match_subject(RuleContext *context, FilterRule *rule, const char *subject) { const char *s; const char *s1, *s2; char *tmp; s = strip_re(subject); /* dont match on empty subject */ if (*s == 0) return; /* [blahblah] is probably a mailing list, match on it separately */ s1 = strchr(s, '['); s2 = strchr(s, ']'); if (s1 && s2 && s1grouping = FILTER_GROUP_ANY; if (flags & AUTO_SUBJECT) { rule_match_subject(context, rule, msg->subject); } /* should parse the from address into an internet address? */ if (flags & AUTO_FROM) { struct _header_address *haddr, *scan; haddr = header_address_decode(msg->from); scan = haddr; while (scan) { if (scan->type == HEADER_ADDRESS_NAME) { rule_add_sender(context, rule, scan->v.addr); } } header_address_unref(haddr); } if (flags & AUTO_TO) { addr = (CamelInternetAddress *)camel_mime_message_get_recipients(msg, CAMEL_RECIPIENT_TYPE_TO); rule_match_recipients(context, rule, addr); addr = (CamelInternetAddress *)camel_mime_message_get_recipients(msg, CAMEL_RECIPIENT_TYPE_CC); rule_match_recipients(context, rule, addr); } } FilterRule * vfolder_rule_from_message(RuleContext *context, FolderBrowser *fb, CamelMimeMessage *msg, int flags) { VfolderRule *rule; rule = vfolder_rule_new(); vfolder_rule_add_source(rule, fb->uri); rule_from_message((FilterRule *)rule, context, msg, flags); return (FilterRule *)rule; } FilterRule * filter_rule_from_message(RuleContext *context, FolderBrowser *fb, CamelMimeMessage *msg, int flags) { FilterFilter *rule; rule = filter_filter_new(); rule_from_message((FilterRule *)rule, context, msg, flags); /* should we define the default action? */ return (FilterRule *)rule; } #if 0 #error "Ok, this is all useless because we're not using gnome-libs afteralll ..." static void create_filter(GtkWidget *w, void *data); static void create_vfolder(GtkWidget *w, void *data); static GnomeUIInfo auto_filter_popup[] = { GNOMEUIINFO_ITEM_NONE("Subject", "", create_filter), GNOMEUIINFO_ITEM_NONE("From", "", create_filter), GNOMEUIINFO_ITEM_NONE("To", "", create_filter), GNOMEUIINFO_ITEM_NONE("Subject & To", "", create_filter), GNOMEUIINFO_ITEM_NONE("Subject & From", "", create_filter), GNOMEUIINFO_END }; static GnomeUIInfo auto_vfolder_popup[] = { GNOMEUIINFO_ITEM_NONE("Subject", "", create_vfolder), GNOMEUIINFO_ITEM_NONE("From", "", create_vfolder), GNOMEUIINFO_ITEM_NONE("To", "", create_vfolder), GNOMEUIINFO_ITEM_NONE("Subject & To", "", create_vfolder), GNOMEUIINFO_ITEM_NONE("Subject & From", "", create_vfolder), GNOMEUIINFO_END }; static GnomeUIInfo auto_popup[] = { GNOMEUIINFO_SUBTREE("New Filter", auto_filter_popup), GNOMEUIINFO_SUBTREE("New vFolder", auto_vfolder_popup), GNOMEUIINFO_END }; static int flags[] = { AUTO_SUBJECT, AUTO_FROM, AUTO_TO, AUTO_SUBJECT|AUTO_TO, AUTO_SUBJECT|AUTO_FROM }; static int widget_index(GnomeUIInfo *info, GtkWidget *w) { int i; for (i=0;info[i].widget;i++) if (info[i].widget == w) return i; return -1; } static void create_filter(GtkWidget *w, void *data) { int index; index = widget_index(auto_filter_popup, w); printf("filter index = %d\n", index); } static void create_vfolder(GtkWidget *w, void *data) { int index; index = widget_index(auto_vfolder_popup, w); printf("filter index = %d\n", index); } void message_popup(GdkEventButton *event, FolderBrowser *fb) { GtkWidget *popup_menu; int i; popup_menu = gnome_popup_menu_new(auto_popup); gnome_popup_menu_do_popup(popup_menu, NULL, NULL, event, fb); } #endif