956{
957
958
959 struct PatternList *curlist = NULL;
960 bool pat_not = false;
961 bool all_addr = false;
962 bool pat_or = false;
963 bool implicit = true;
964 bool is_alias = false;
966 char *p = NULL;
967 char *buf = NULL;
969
970 if (!s || (s[0] == '\0'))
971 {
973 return NULL;
974 }
975
979
982 {
984 {
985 case '^':
987 all_addr = !all_addr;
988 break;
989 case '!':
991 pat_not = !pat_not;
992 break;
993 case '@':
995 is_alias = !is_alias;
996 break;
997 case '|':
998 if (!pat_or)
999 {
1000 if (!curlist)
1001 {
1004 return NULL;
1005 }
1006
1009 {
1010
1013 }
1014
1015 pat_or = true;
1016 }
1018 implicit = false;
1022 break;
1023 case '%':
1024 case '=':
1025 case '~':
1026 {
1027 if (ps->
dptr[1] ==
'\0')
1028 {
1030 goto cleanup;
1031 }
1032 short thread_op = 0;
1033 if (ps->
dptr[1] ==
'(')
1035 else if ((ps->
dptr[1] ==
'<') && (ps->
dptr[2] ==
'('))
1037 else if ((ps->
dptr[1] ==
'>') && (ps->
dptr[2] ==
'('))
1039 if (thread_op != 0)
1040 {
1046 {
1048 goto cleanup;
1049 }
1051 leaf->
op = thread_op;
1058
1062 {
1064 goto cleanup;
1065 }
1068 break;
1069 }
1070 if (implicit && pat_or)
1071 {
1072
1075 pat_or = false;
1076 }
1077
1078 char prefix = ps->
dptr[0];
1080 if (!entry)
1081 {
1083 goto cleanup;
1084 }
1085 if (entry->
flags && ((flags & entry->
flags) == 0))
1086 {
1087 buf_printf(err,
_(
"%c%c: not supported in this mode"), prefix, ps->
dptr[1]);
1088 goto cleanup;
1089 }
1090
1096 leaf->
op = entry->
op;
1100
1101
1102
1103
1105 if ((entry->
prefix ==
'~') && (prefix ==
'=') && (eat_arg ==
EAT_REGEX))
1107 else if ((entry->
prefix ==
'~') && (prefix ==
'%') && (eat_arg ==
EAT_REGEX))
1109
1113 if (eat_arg)
1114 {
1115 if (ps->
dptr[0] ==
'\0')
1116 {
1118 goto cleanup;
1119 }
1120 switch (eat_arg)
1121 {
1124 goto cleanup;
1125 break;
1128 goto cleanup;
1129 break;
1132 goto cleanup;
1133 break;
1135 if (!
eat_date(leaf, flags, ps, err))
1136 goto cleanup;
1137 break;
1140 goto cleanup;
1141 break;
1144 goto cleanup;
1145 break;
1147 if (!
eat_query(leaf, flags, ps, err, m))
1148 goto cleanup;
1149 break;
1150 default:
1151 break;
1152 }
1153 }
1154 implicit = true;
1155 break;
1156 }
1157
1158 case '(':
1159 {
1162 {
1164 goto cleanup;
1165 }
1166
1170 if (!sub)
1171 goto cleanup;
1173 if (curlist)
1174 {
1177 }
1178 else
1179 {
1180 curlist = sub;
1181 }
1189 break;
1190 }
1191
1192 default:
1194 goto cleanup;
1195 }
1197 }
1199
1200 if (!curlist)
1201 {
1203 return NULL;
1204 }
1205
1207 {
1210 }
1211
1212 return curlist;
1213
1214cleanup:
1217 return NULL;
1218}
void buf_seek(struct Buffer *buf, size_t offset)
Set current read/write position to offset from beginning.
size_t buf_addstr(struct Buffer *buf, const char *s)
Add a string to a Buffer.
size_t buf_strcpy(struct Buffer *buf, const char *s)
Copy a string into a Buffer.
static struct Pattern * attach_new_leaf(struct PatternList **curlist)
Attach a new Pattern to a List.
struct PatternList * mutt_pattern_comp(struct MailboxView *mv, const char *s, PatternCompFlags flags, struct Buffer *err)
Create a Pattern.
static char * find_matching_paren(char *s)
Find the matching parenthesis.
static bool eat_group(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)
Parse a group name - Implements eat_arg_t -.
static bool eat_string(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)
Parse a plain string - Implements eat_arg_t -.
static bool eat_query(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err, struct Mailbox *m)
Parse a query for an external search program - Implements eat_arg_t -.
bool eat_message_range(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err, struct MailboxView *mv)
Parse a range of message numbers - Implements eat_arg_t -.
static bool eat_range(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)
Parse a number range - Implements eat_arg_t -.
static bool eat_date(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)
Parse a date pattern - Implements eat_arg_t -.
static bool eat_regex(struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)
Parse a regex - Implements eat_arg_t -.
char * mutt_strn_dup(const char *begin, size_t len)
Duplicate a sub-string.
const struct PatternFlags * lookup_tag(char prefix, char tag)
Lookup a pattern modifier.
#define MUTT_PC_SEND_MODE_SEARCH
Allow send-mode body searching.
@ MUTT_PAT_OR
Either pattern can match.
@ MUTT_PAT_CHILDREN
Pattern matches a child email.
@ MUTT_PAT_PARENT
Pattern matches parent.
@ MUTT_PAT_AND
Both patterns must match.
@ MUTT_PAT_THREAD
Pattern matches email thread.
PatternEat
Function to process pattern arguments.
@ EAT_STRING
Process a plain string.
@ EAT_RANGE
Process a number (range)
@ EAT_GROUP
Process a group name.
@ EAT_MESSAGE_RANGE
Process a message number (range)
@ EAT_DATE
Process a date (range)
@ EAT_QUERY
Process a query string.
@ EAT_REGEX
Process a regex.
struct Buffer * buf_pool_get(void)
Get a Buffer from the pool.
void buf_pool_release(struct Buffer **ptr)
Return a Buffer to the pool.
String manipulation buffer.
char * dptr
Current read/write position.
struct Mailbox * mailbox
Current Mailbox.
Mapping between user character and internal constant.
enum PatternEat eat_arg
Type of function needed to parse flag, e.g. EAT_DATE.
PatternCompFlags flags
Pattern flags, e.g. MUTT_PC_FULL_MSG.
int op
Operation to perform, e.g. MUTT_PAT_SCORE.
char prefix
Prefix for this pattern, e.g. '~', '', or '='.
bool all_addr
All Addresses in the list must match.
bool is_alias
Is there an alias for this Address?
short op
Operation, e.g. MUTT_PAT_SCORE.
bool sendmode
Evaluate searches in send-mode.
bool pat_not
Pattern should be inverted (not)