NeoMutt  2025-12-11-911-gd8d604
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
pattern.c File Reference

Match patterns to emails. More...

#include "config.h"
#include <stdbool.h>
#include <stddef.h>
#include "mutt/lib.h"
#include "config/lib.h"
#include "email/lib.h"
#include "core/lib.h"
#include "alias/gui.h"
#include "alias/lib.h"
#include "gui/lib.h"
#include "mutt.h"
#include "lib.h"
#include "editor/lib.h"
#include "history/lib.h"
#include "imap/lib.h"
#include "menu/lib.h"
#include "progress/lib.h"
#include "mutt_logging.h"
#include "mx.h"
#include "search_state.h"
+ Include dependency graph for pattern.c:

Go to the source code of this file.

Typedefs

typedef bool(* eat_arg_t) (struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)
 

Functions

static void quote_simple (const char *str, struct Buffer *buf)
 Apply simple quoting to a string.
 
void mutt_check_simple (struct Buffer *buf, const char *simple)
 Convert a simple search into a real request.
 
int mutt_pattern_alias_func (char *prompt, struct AliasMenuData *mdata, enum PatternAlias action, struct Menu *menu)
 Perform some Pattern matching for Alias.
 
int mutt_pattern_func (struct MailboxView *mv, int op, char *prompt)
 Perform some Pattern matching.
 
int mutt_search_command (struct MailboxView *mv, struct Menu *menu, int cur, struct SearchState *state, SearchFlags flags)
 Perform a search.
 
int mutt_search_alias_command (struct Menu *menu, int cur, struct SearchState *state, SearchFlags flags)
 Perform a search.
 

Detailed Description

Match patterns to emails.

Authors
  • Pietro Cerutti
  • R Primus
  • Romeu Vieira
  • Richard Russon
  • Dennis Schön

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, see http://www.gnu.org/licenses/.

Definition in file pattern.c.

Typedef Documentation

◆ eat_arg_t

typedef bool(* eat_arg_t) (struct Pattern *pat, PatternCompFlags flags, struct Buffer *s, struct Buffer *err)

Definition at line 65 of file pattern.c.

Function Documentation

◆ quote_simple()

static void quote_simple ( const char * str,
struct Buffer * buf )
static

Apply simple quoting to a string.

Parameters
strString to quote
bufBuffer for the result

Definition at line 73 of file pattern.c.

74{
75 buf_reset(buf);
76 buf_addch(buf, '"');
77 while (*str)
78 {
79 if ((*str == '\\') || (*str == '"'))
80 buf_addch(buf, '\\');
81 buf_addch(buf, *str++);
82 }
83 buf_addch(buf, '"');
84}
void buf_reset(struct Buffer *buf)
Reset an existing Buffer.
Definition buffer.c:76
size_t buf_addch(struct Buffer *buf, char c)
Add a single character to a Buffer.
Definition buffer.c:241
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_check_simple()

void mutt_check_simple ( struct Buffer * buf,
const char * simple )

Convert a simple search into a real request.

Parameters
bufBuffer for the result
simpleSearch string to convert

Definition at line 91 of file pattern.c.

92{
93 bool do_simple = true;
94
95 for (const char *p = buf_string(buf); p && (p[0] != '\0'); p++)
96 {
97 if ((p[0] == '\\') && (p[1] != '\0'))
98 {
99 p++;
100 }
101 else if ((p[0] == '~') || (p[0] == '=') || (p[0] == '%'))
102 {
103 do_simple = false;
104 break;
105 }
106 }
107
108 /* XXX - is mutt_istr_cmp() right here, or should we use locale's
109 * equivalences? */
110
111 if (do_simple) /* yup, so spoof a real request */
112 {
113 /* convert old tokens into the new format */
114 if (mutt_istr_equal("all", buf_string(buf)) || mutt_str_equal("^", buf_string(buf)) ||
115 mutt_str_equal(".", buf_string(buf))) /* ~A is more efficient */
116 {
117 buf_strcpy(buf, "~A");
118 }
119 else if (mutt_istr_equal("del", buf_string(buf)))
120 {
121 buf_strcpy(buf, "~D");
122 }
123 else if (mutt_istr_equal("flag", buf_string(buf)))
124 {
125 buf_strcpy(buf, "~F");
126 }
127 else if (mutt_istr_equal("new", buf_string(buf)))
128 {
129 buf_strcpy(buf, "~N");
130 }
131 else if (mutt_istr_equal("old", buf_string(buf)))
132 {
133 buf_strcpy(buf, "~O");
134 }
135 else if (mutt_istr_equal("repl", buf_string(buf)))
136 {
137 buf_strcpy(buf, "~Q");
138 }
139 else if (mutt_istr_equal("read", buf_string(buf)))
140 {
141 buf_strcpy(buf, "~R");
142 }
143 else if (mutt_istr_equal("tag", buf_string(buf)))
144 {
145 buf_strcpy(buf, "~T");
146 }
147 else if (mutt_istr_equal("unread", buf_string(buf)))
148 {
149 buf_strcpy(buf, "~U");
150 }
151 else
152 {
153 struct Buffer *tmp = buf_pool_get();
154 quote_simple(buf_string(buf), tmp);
155 mutt_file_expand_fmt(buf, simple, buf_string(tmp));
156 buf_pool_release(&tmp);
157 }
158 }
159}
size_t buf_strcpy(struct Buffer *buf, const char *s)
Copy a string into a Buffer.
Definition buffer.c:395
static const char * buf_string(const struct Buffer *buf)
Convert a buffer to a const char * "string".
Definition buffer.h:96
void mutt_file_expand_fmt(struct Buffer *dest, const char *fmt, const char *src)
Replace s in a string with a filename.
Definition file.c:1366
bool mutt_istr_equal(const char *a, const char *b)
Compare two strings, ignoring case.
Definition string.c:677
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
Definition string.c:665
static void quote_simple(const char *str, struct Buffer *buf)
Apply simple quoting to a string.
Definition pattern.c:73
struct Buffer * buf_pool_get(void)
Get a Buffer from the pool.
Definition pool.c:91
void buf_pool_release(struct Buffer **ptr)
Return a Buffer to the pool.
Definition pool.c:111
String manipulation buffer.
Definition buffer.h:36
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_pattern_alias_func()

int mutt_pattern_alias_func ( char * prompt,
struct AliasMenuData * mdata,
enum PatternAlias action,
struct Menu * menu )

Perform some Pattern matching for Alias.

Parameters
promptPrompt to show the user
mdataMenu data holding Aliases
actionWhat to do with the results, e.g. PAA_TAG
menuCurrent menu
Return values
0Success
-1Failure

Definition at line 170 of file pattern.c.

172{
173 int rc = -1;
174 struct Progress *progress = NULL;
175 struct Buffer *buf = buf_pool_get();
176
177 buf_strcpy(buf, mdata->limit);
178 if (prompt)
179 {
180 if ((mw_get_field(prompt, buf, MUTT_COMP_CLEAR, HC_PATTERN, &CompletePatternOps, NULL) != 0) ||
181 buf_is_empty(buf))
182 {
183 buf_pool_release(&buf);
184 return -1;
185 }
186 }
187
188 mutt_message(_("Compiling search pattern..."));
189
190 /* Compile the pattern string; "~A" means match-all. If the simple string
191 * was expanded into a pattern, compile it for evaluation. */
192 bool match_all = false;
193 struct PatternList *pat = NULL;
194 char *simple = buf_strdup(buf);
195 if (simple)
196 {
198 const char *pbuf = buf->data;
199 while (*pbuf == ' ')
200 pbuf++;
201 match_all = mutt_str_equal(pbuf, "~A");
202
203 struct Buffer *err = buf_pool_get();
204 pat = mutt_pattern_comp(NULL, buf->data, MUTT_PC_FULL_MSG, err);
205 if (!pat)
206 {
207 mutt_error("%s", buf_string(err));
208 buf_pool_release(&err);
209 goto bail;
210 }
211 buf_pool_release(&err);
212 }
213 else
214 {
215 match_all = true;
216 }
217
218 progress = progress_new(MUTT_PROGRESS_READ, ARRAY_SIZE(&mdata->ava));
219 progress_set_message(progress, _("Executing command on matching messages..."));
220
221 /* Apply the action (tag, untag, or filter visibility) to each alias
222 * that matches the compiled pattern */
223 int vcounter = 0;
224 struct AliasView *avp = NULL;
225 ARRAY_FOREACH(avp, &mdata->ava)
226 {
227 progress_update(progress, ARRAY_FOREACH_IDX_avp, -1);
228
229 if (match_all ||
231 {
232 switch (action)
233 {
234 case PAA_TAG:
235 avp->is_tagged = true;
236 break;
237 case PAA_UNTAG:
238 avp->is_tagged = false;
239 break;
240 case PAA_VISIBLE:
241 avp->is_visible = true;
242 vcounter++;
243 break;
244 }
245 }
246 else
247 {
248 switch (action)
249 {
250 case PAA_TAG:
251 case PAA_UNTAG:
252 // Do nothing
253 break;
254 case PAA_VISIBLE:
255 avp->is_visible = false;
256 break;
257 }
258 }
259 }
260 progress_free(&progress);
261
262 FREE(&mdata->limit);
263 if (!match_all)
264 {
265 mdata->limit = simple;
266 simple = NULL;
267 }
268
269 if (menu && (action == PAA_VISIBLE))
270 {
271 menu->max = vcounter;
272 menu_set_index(menu, 0);
273 }
274
276
277 rc = 0;
278
279bail:
280 buf_pool_release(&buf);
281 FREE(&simple);
282 mutt_pattern_free(&pat);
283
284 return rc;
285}
#define ARRAY_FOREACH(elem, head)
Iterate over all elements of the array.
Definition array.h:223
#define ARRAY_SIZE(head)
The number of elements stored.
Definition array.h:87
bool buf_is_empty(const struct Buffer *buf)
Is the Buffer empty?
Definition buffer.c:291
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
Definition buffer.c:571
struct PatternList * mutt_pattern_comp(struct MailboxView *mv, const char *s, PatternCompFlags flags, struct Buffer *err)
Create a Pattern.
Definition compile.c:964
void mutt_pattern_free(struct PatternList **pat)
Free a Pattern.
Definition compile.c:836
@ MUTT_COMP_CLEAR
Clear input if printable character is pressed.
Definition wdata.h:47
int mw_get_field(const char *prompt, struct Buffer *buf, CompletionFlags complete, enum HistoryClass hclass, const struct CompleteOps *comp_api, void *cdata)
Ask the user for a string -.
Definition window.c:502
#define mutt_error(...)
Definition logging2.h:94
#define mutt_message(...)
Definition logging2.h:93
@ HC_PATTERN
Patterns.
Definition lib.h:60
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
MenuRedrawFlags menu_set_index(struct Menu *menu, int index)
Set the current selection in the Menu.
Definition menu.c:169
#define _(a)
Definition message.h:28
void mutt_clear_error(void)
Clear the message line (bottom line of screen)
const struct CompleteOps CompletePatternOps
Auto-Completion of Patterns.
Definition complete.c:98
bool mutt_pattern_alias_exec(struct Pattern *pat, PatternExecFlags flags, struct AliasView *av, struct PatternCache *cache)
Match a pattern against an alias.
Definition exec.c:1178
@ MUTT_MATCH_FULL_ADDRESS
Match the full address.
Definition lib.h:117
@ PAA_VISIBLE
Set AliasView.is_visible and hide the rest.
Definition lib.h:203
@ PAA_TAG
Set AliasView.is_tagged, but don't touch the others.
Definition lib.h:201
@ PAA_UNTAG
Unset AliasView.is_tagged, but don't touch the others.
Definition lib.h:202
#define MUTT_ALIAS_SIMPLESEARCH
Simple search pattern for aliases (from/to/cc fields)
Definition lib.h:64
@ MUTT_PC_FULL_MSG
Enable body and header matching.
Definition lib.h:74
void mutt_check_simple(struct Buffer *buf, const char *simple)
Convert a simple search into a real request.
Definition pattern.c:91
@ MUTT_PROGRESS_READ
Progress tracks elements, according to $read_inc
Definition lib.h:84
struct Progress * progress_new(enum ProgressType type, size_t size)
Create a new Progress Bar.
Definition progress.c:139
void progress_free(struct Progress **ptr)
Free a Progress Bar.
Definition progress.c:110
void progress_set_message(struct Progress *progress, const char *fmt,...) __attribute__((__format__(__printf__
bool progress_update(struct Progress *progress, size_t pos, int percent)
Update the state of the progress bar.
Definition progress.c:80
#define SLIST_FIRST(head)
Definition queue.h:227
char * limit
Limit being used.
Definition gui.h:61
struct AliasViewArray ava
All Aliases/Queries.
Definition gui.h:56
GUI data wrapping an Alias.
Definition gui.h:38
bool is_visible
Is visible?
Definition gui.h:45
bool is_tagged
Is it tagged?
Definition gui.h:43
char * data
Pointer to data.
Definition buffer.h:37
int max
Number of entries in the menu.
Definition lib.h:88
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_pattern_func()

int mutt_pattern_func ( struct MailboxView * mv,
int op,
char * prompt )

Perform some Pattern matching.

Parameters
mvMailbox View
opOperation to perform, e.g. MUTT_LIMIT
promptPrompt to show the user
Return values
0Success
-1Failure

Definition at line 295 of file pattern.c.

296{
297 if (!mv || !mv->mailbox)
298 return -1;
299
300 struct Mailbox *m = mv->mailbox;
301
302 struct Buffer *err = NULL;
303 int rc = -1;
304 struct Progress *progress = NULL;
305 struct Buffer *buf = buf_pool_get();
306 bool interrupted = false;
307
308 buf_strcpy(buf, mv->pattern);
309 if (prompt || (op != MUTT_LIMIT))
310 {
311 if ((mw_get_field(prompt, buf, MUTT_COMP_CLEAR, HC_PATTERN, &CompletePatternOps, NULL) != 0) ||
312 buf_is_empty(buf))
313 {
314 buf_pool_release(&buf);
315 return -1;
316 }
317 }
318
319 mutt_message(_("Compiling search pattern..."));
320
321 char *simple = buf_strdup(buf);
322 const char *const c_simple_search = cs_subset_string(NeoMutt->sub, "simple_search");
323 mutt_check_simple(buf, NONULL(c_simple_search));
324 const char *pbuf = buf->data;
325 while (*pbuf == ' ')
326 pbuf++;
327 const bool match_all = mutt_str_equal(pbuf, "~A");
328
329 err = buf_pool_get();
330 struct PatternList *pat = mutt_pattern_comp(mv, buf->data, MUTT_PC_FULL_MSG, err);
331 if (!pat)
332 {
333 mutt_error("%s", buf_string(err));
334 goto bail;
335 }
336
337 if ((m->type == MUTT_IMAP) && (!imap_search(m, pat)))
338 goto bail;
339
340 progress = progress_new(MUTT_PROGRESS_READ, (op == MUTT_LIMIT) ? m->msg_count : m->vcount);
341 progress_set_message(progress, _("Executing command on matching messages..."));
342
343 if (op == MUTT_LIMIT)
344 {
345 m->vcount = 0;
346 mv->vsize = 0;
347 mv->collapsed = false;
348 int padding = mx_msg_padding_size(m);
349
350 for (int i = 0; i < m->msg_count; i++)
351 {
352 struct Email *e = m->emails[i];
353 if (!e)
354 break;
355
356 if (SigInt)
357 {
358 interrupted = true;
359 SigInt = false;
360 break;
361 }
362 progress_update(progress, i, -1);
363 /* new limit pattern implicitly uncollapses all threads */
364 e->vnum = -1;
365 e->visible = false;
366 e->limit_visited = true;
367 e->collapsed = false;
368 e->num_hidden = 0;
369
370 if (match_all ||
372 {
373 e->vnum = m->vcount;
374 e->visible = true;
375 m->v2r[m->vcount] = i;
376 m->vcount++;
377 struct Body *b = e->body;
378 mv->vsize += b->length + b->offset - b->hdr_offset + padding;
379 }
380 }
381 }
382 else
383 {
384 for (int i = 0; i < m->vcount; i++)
385 {
386 struct Email *e = mutt_get_virt_email(m, i);
387 if (!e)
388 continue;
389
390 if (SigInt)
391 {
392 interrupted = true;
393 SigInt = false;
394 break;
395 }
396 progress_update(progress, i, -1);
398 {
399 switch (op)
400 {
401 case MUTT_UNDELETE:
402 mutt_set_flag(m, e, MUTT_PURGE, false, true);
404
405 case MUTT_DELETE:
406 mutt_set_flag(m, e, MUTT_DELETE, (op == MUTT_DELETE), true);
407 break;
408 case MUTT_TAG:
409 case MUTT_UNTAG:
410 mutt_set_flag(m, e, MUTT_TAG, (op == MUTT_TAG), true);
411 break;
412 }
413 }
414 }
415 }
416 progress_free(&progress);
417
419
420 if (op == MUTT_LIMIT)
421 {
422 /* drop previous limit pattern */
423 FREE(&mv->pattern);
425
426 if (m->msg_count && !m->vcount)
427 mutt_error(_("No messages matched criteria"));
428
429 /* record new limit pattern, unless match all */
430 if (!match_all)
431 {
432 mv->pattern = simple;
433 simple = NULL; /* don't clobber it */
435 }
436 }
437
438 if (interrupted)
439 mutt_error(_("Search interrupted"));
440
441 rc = 0;
442
443bail:
444 buf_pool_release(&buf);
445 buf_pool_release(&err);
446 FREE(&simple);
447 mutt_pattern_free(&pat);
448
449 return rc;
450}
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
@ MUTT_IMAP
'IMAP' Mailbox type
Definition mailbox.h:49
void mutt_set_flag(struct Mailbox *m, struct Email *e, enum MessageType flag, bool bf, bool upd_mbox)
Set a flag on an email.
Definition flags.c:54
bool imap_search(struct Mailbox *m, const struct PatternList *pat)
Find messages in mailbox matching a pattern.
Definition search.c:227
#define FALLTHROUGH
Definition lib.h:117
@ MUTT_UNDELETE
Messages to be un-deleted.
Definition mutt.h:95
@ MUTT_LIMIT
Messages in limited view.
Definition mutt.h:101
@ MUTT_UNTAG
Messages to be un-tagged.
Definition mutt.h:100
@ MUTT_PURGE
Messages to be purged (bypass trash)
Definition mutt.h:96
@ MUTT_TAG
Tagged messages.
Definition mutt.h:99
@ MUTT_DELETE
Messages to be deleted.
Definition mutt.h:94
struct Email * mutt_get_virt_email(struct Mailbox *m, int vnum)
Get a virtual Email.
Definition mview.c:376
int mx_msg_padding_size(struct Mailbox *m)
Bytes of padding between messages - Wrapper for MxOps::msg_padding_size()
Definition mx.c:1507
bool mutt_pattern_exec(struct Pattern *pat, PatternExecFlags flags, struct Mailbox *m, struct Email *e, struct PatternCache *cache)
Match a pattern against an email header.
Definition exec.c:1151
volatile sig_atomic_t SigInt
true after SIGINT is received
Definition signal.c:68
#define NONULL(x)
Definition string2.h:44
The body of an email.
Definition body.h:36
LOFF_T offset
offset where the actual data begins
Definition body.h:52
LOFF_T length
length (in bytes) of attachment
Definition body.h:53
long hdr_offset
Offset in stream where the headers begin.
Definition body.h:81
The envelope/body of an email.
Definition email.h:39
bool visible
Is this message part of the view?
Definition email.h:121
bool limit_visited
Has the limit pattern been applied to this message?
Definition email.h:122
bool collapsed
Is this message part of a collapsed thread?
Definition email.h:120
struct Body * body
List of MIME parts.
Definition email.h:69
size_t num_hidden
Number of hidden messages in this view (only valid when collapsed is set)
Definition email.h:123
int vnum
Virtual message number.
Definition email.h:114
bool collapsed
Are all threads collapsed?
Definition mview.h:49
off_t vsize
Size (in bytes) of the messages shown.
Definition mview.h:41
struct PatternList * limit_pattern
Compiled limit pattern.
Definition mview.h:43
struct Mailbox * mailbox
Current Mailbox.
Definition mview.h:51
char * pattern
Limit pattern string.
Definition mview.h:42
A mailbox.
Definition mailbox.h:81
int vcount
The number of virtual messages.
Definition mailbox.h:101
int * v2r
Mapping from virtual to real msgno.
Definition mailbox.h:100
int msg_count
Total number of messages.
Definition mailbox.h:90
enum MailboxType type
Mailbox type.
Definition mailbox.h:104
struct Email ** emails
Array of Emails.
Definition mailbox.h:98
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_search_command()

int mutt_search_command ( struct MailboxView * mv,
struct Menu * menu,
int cur,
struct SearchState * state,
SearchFlags flags )

Perform a search.

Parameters
mvMailbox view to search through
menuCurrent Menu
curIndex number of current email
stateCurrent search state
flagsSearch flags, e.g. SEARCH_PROMPT
Return values
>=0Index of matching email
-1No match, or error

Definition at line 462 of file pattern.c.

464{
465 struct Progress *progress = NULL;
466 int rc = -1;
467 struct Mailbox *m = mv ? mv->mailbox : NULL;
468 if (!m)
469 return -1;
470 bool pattern_changed = false;
471
472 if (buf_is_empty(state->string) || (flags & SEARCH_PROMPT))
473 {
474 if ((mw_get_field((state->reverse) ? _("Reverse search for: ") : _("Search for: "),
476 &CompletePatternOps, NULL) != 0) ||
477 buf_is_empty(state->string))
478 {
479 goto done;
480 }
481
482 /* compare the *expanded* version of the search pattern in case
483 * $simple_search has changed while we were searching */
484 struct Buffer *tmp = buf_pool_get();
485 buf_copy(tmp, state->string);
486 const char *const c_simple_search = cs_subset_string(NeoMutt->sub, "simple_search");
487 mutt_check_simple(tmp, NONULL(c_simple_search));
488 if (!buf_str_equal(tmp, state->string_expn))
489 {
490 mutt_pattern_free(&state->pattern);
491 buf_copy(state->string_expn, tmp);
492 buf_pool_release(&tmp);
493 }
494 }
495
496 if (!state->pattern)
497 {
498 mutt_message(_("Compiling search pattern..."));
499 mutt_pattern_free(&state->pattern);
500 struct Buffer *err = buf_pool_get();
501 state->pattern = mutt_pattern_comp(mv, state->string_expn->data, MUTT_PC_FULL_MSG, err);
502 pattern_changed = true;
503 if (!state->pattern)
504 {
505 mutt_error("%s", buf_string(err));
506 buf_free(&err);
507 buf_reset(state->string);
508 buf_reset(state->string_expn);
509 return -1;
510 }
511 buf_free(&err);
513 }
514
515 if (pattern_changed)
516 {
517 for (int i = 0; i < m->msg_count; i++)
518 m->emails[i]->searched = false;
519 if ((m->type == MUTT_IMAP) && (!imap_search(m, state->pattern)))
520 return -1;
521 }
522
523 int incr = state->reverse ? -1 : 1;
524 if (flags & SEARCH_OPPOSITE)
525 incr = -incr;
526
527 progress = progress_new(MUTT_PROGRESS_READ, m->vcount);
528 progress_set_message(progress, _("Searching..."));
529
530 const bool c_wrap_search = cs_subset_bool(NeoMutt->sub, "wrap_search");
531 for (int i = cur + incr, j = 0; j != m->vcount; j++)
532 {
533 const char *msg = NULL;
534 progress_update(progress, j, -1);
535 if (i > m->vcount - 1)
536 {
537 i = 0;
538 if (c_wrap_search)
539 {
540 msg = _("Search wrapped to top");
541 }
542 else
543 {
544 mutt_message(_("Search hit bottom without finding match"));
545 goto done;
546 }
547 }
548 else if (i < 0)
549 {
550 i = m->vcount - 1;
551 if (c_wrap_search)
552 {
553 msg = _("Search wrapped to bottom");
554 }
555 else
556 {
557 mutt_message(_("Search hit top without finding match"));
558 goto done;
559 }
560 }
561
562 struct Email *e = mutt_get_virt_email(m, i);
563 if (!e)
564 goto done;
565
566 if (e->searched)
567 {
568 /* if we've already evaluated this message, use the cached value */
569 if (e->matched)
570 {
572 if (msg && *msg)
573 mutt_message("%s", msg);
574 rc = i;
575 goto done;
576 }
577 }
578 else
579 {
580 /* remember that we've already searched this message */
581 e->searched = true;
583 MUTT_MATCH_FULL_ADDRESS, m, e, NULL);
584 if (e->matched)
585 {
587 if (msg && *msg)
588 mutt_message("%s", msg);
589 rc = i;
590 goto done;
591 }
592 }
593
594 if (SigInt)
595 {
596 mutt_error(_("Search interrupted"));
597 SigInt = false;
598 goto done;
599 }
600
601 i += incr;
602 }
603
604 mutt_error(_("Not found"));
605done:
606 progress_free(&progress);
607 return rc;
608}
void buf_free(struct Buffer **ptr)
Deallocates a buffer.
Definition buffer.c:319
bool buf_str_equal(const struct Buffer *a, const struct Buffer *b)
Return if two buffers are equal.
Definition buffer.c:683
size_t buf_copy(struct Buffer *dst, const struct Buffer *src)
Copy a Buffer's contents to another Buffer.
Definition buffer.c:601
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition helpers.c:47
@ SEARCH_PROMPT
Ask for search input.
@ SEARCH_OPPOSITE
Search in the opposite direction.
bool searched
Email has been searched.
Definition email.h:105
bool matched
Search matches this Email.
Definition email.h:102
struct Buffer * string
search string
struct Buffer * string_expn
expanded search string
bool reverse
search backwards
struct PatternList * pattern
compiled search pattern
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_search_alias_command()

int mutt_search_alias_command ( struct Menu * menu,
int cur,
struct SearchState * state,
SearchFlags flags )

Perform a search.

Parameters
menuMenu to search through
curIndex number of current email
stateCurrent search state
flagsSearch flags, e.g. SEARCH_PROMPT
Return values
>=0Index of matching alias
-1No match, or error

Definition at line 619 of file pattern.c.

621{
622 struct Progress *progress = NULL;
623 const struct AliasMenuData *mdata = menu->mdata;
624 const struct AliasViewArray *ava = &mdata->ava;
625 int rc = -1;
626 bool pattern_changed = false;
627
628 if (buf_is_empty(state->string) || flags & SEARCH_PROMPT)
629 {
630 if ((mw_get_field(state->reverse ? _("Reverse search for: ") : _("Search for: "),
632 &CompletePatternOps, NULL) != 0) ||
633 buf_is_empty(state->string))
634 {
635 goto done;
636 }
637
638 /* compare the *expanded* version of the search pattern in case
639 * $simple_search has changed while we were searching */
640 struct Buffer *tmp = buf_pool_get();
641 buf_copy(tmp, state->string);
643 if (!buf_str_equal(tmp, state->string_expn))
644 {
645 mutt_pattern_free(&state->pattern);
646 buf_copy(state->string_expn, tmp);
647 buf_pool_release(&tmp);
648 }
649 }
650
651 if (!state->pattern)
652 {
653 mutt_message(_("Compiling search pattern..."));
654 struct Buffer *err = buf_pool_get();
655 state->pattern = mutt_pattern_comp(NULL, state->string_expn->data, MUTT_PC_FULL_MSG, err);
656 pattern_changed = true;
657 if (!state->pattern)
658 {
659 mutt_error("%s", buf_string(err));
660 buf_free(&err);
661 buf_reset(state->string);
662 buf_reset(state->string_expn);
663 return -1;
664 }
665 buf_free(&err);
667 }
668
669 if (pattern_changed)
670 {
671 struct AliasView *av = NULL;
672 ARRAY_FOREACH(av, ava)
673 {
674 av->is_searched = false;
675 }
676 }
677
678 int incr = state->reverse ? -1 : 1;
679 if (flags & SEARCH_OPPOSITE)
680 incr = -incr;
681
683 progress_set_message(progress, _("Searching..."));
684
685 const bool c_wrap_search = cs_subset_bool(NeoMutt->sub, "wrap_search");
686 for (int i = cur + incr, j = 0; j != ARRAY_SIZE(ava); j++)
687 {
688 const char *msg = NULL;
689 progress_update(progress, j, -1);
690 if (i > ARRAY_SIZE(ava) - 1)
691 {
692 i = 0;
693 if (c_wrap_search)
694 {
695 msg = _("Search wrapped to top");
696 }
697 else
698 {
699 mutt_message(_("Search hit bottom without finding match"));
700 goto done;
701 }
702 }
703 else if (i < 0)
704 {
705 i = ARRAY_SIZE(ava) - 1;
706 if (c_wrap_search)
707 {
708 msg = _("Search wrapped to bottom");
709 }
710 else
711 {
712 mutt_message(_("Search hit top without finding match"));
713 goto done;
714 }
715 }
716
717 struct AliasView *av = ARRAY_GET(ava, i);
718 if (av->is_searched)
719 {
720 /* if we've already evaluated this message, use the cached value */
721 if (av->is_matched)
722 {
724 if (msg && *msg)
725 mutt_message("%s", msg);
726 rc = i;
727 goto done;
728 }
729 }
730 else
731 {
732 /* remember that we've already searched this message */
733 av->is_searched = true;
735 MUTT_MATCH_FULL_ADDRESS, av, NULL);
736 if (av->is_matched)
737 {
739 if (msg && *msg)
740 mutt_message("%s", msg);
741 rc = i;
742 goto done;
743 }
744 }
745
746 if (SigInt)
747 {
748 mutt_error(_("Search interrupted"));
749 SigInt = false;
750 goto done;
751 }
752
753 i += incr;
754 }
755
756 mutt_error(_("Not found"));
757done:
758 progress_free(&progress);
759 return rc;
760}
#define ARRAY_GET(head, idx)
Return the element at index.
Definition array.h:109
AliasView array wrapper with Pattern information -.
Definition gui.h:55
struct Menu * menu
Menu.
Definition gui.h:59
bool is_matched
Search matches this Alias.
Definition gui.h:42
bool is_searched
Alias has been searched.
Definition gui.h:41
void * mdata
Private data.
Definition lib.h:155
+ Here is the call graph for this function:
+ Here is the caller graph for this function: