59 const unsigned int NUM_SIG_LINES = 4;
60 unsigned int count = 0;
62 while ((offset > 0) && (count <= NUM_SIG_LINES))
73 if (count > NUM_SIG_LINES)
99 const int *cnt = (
const int *) m1;
102 if (*cnt < stx->
first)
104 if (*cnt >= stx->
last)
124 static struct AttrColor last_color = { 0 };
132 last_color.
attrs = A_NORMAL;
135 if (lines[line_num].cont_line)
138 if (!cnt && c_markers)
194 matching_chunk = bsearch(&cnt, lines[m].syntax, lines[m].syntax_arr_size,
196 if (matching_chunk && (cnt >= matching_chunk->
first) &&
197 (cnt < matching_chunk->last))
206 matching_chunk = bsearch(&cnt, lines[m].
search, lines[m].search_arr_size,
208 if (matching_chunk && (cnt >= matching_chunk->
first) &&
209 (cnt < matching_chunk->last))
217 if (special & A_BOLD)
222 color.
attrs |= A_BOLD;
224 else if (special & A_UNDERLINE)
229 color.
attrs |= A_UNDERLINE;
262 lines[line_num + 1].
cid = lines[line_num].
cid;
263 (lines[line_num + 1].
syntax)[0].attr_color = (lines[line_num].syntax)[0].attr_color;
267 for (m = line_num; m >= 0; m--)
268 if (!lines[m].cont_line)
271 (lines[line_num + 1].
syntax)[0].first = m;
272 (lines[line_num + 1].
syntax)[0].last = (lines[line_num].cont_line) ?
273 cnt + (lines[line_num].
syntax)[0].last :
285 for (; (p[0] == q[0]) && (q[0] !=
'\0') && (p[0] !=
'\0') && (q[0] !=
'\a') &&
291 return (
int) (*p - *q);
325 bool is_quote =
false;
327 regmatch_t pmatch_internal[1] = { 0 };
330 pmatch = pmatch_internal;
335 regmatch_t smatch[1] = { 0 };
338 if (smatch[0].rm_so > 0)
340 char c = line[smatch[0].rm_so];
341 line[smatch[0].rm_so] = 0;
346 line[smatch[0].rm_so] = c;
369 if ((buflen > 0) && (pat[buflen - 1] ==
'\n'))
372 pat[buflen - 1] =
'\0';
378 struct RegexColorList *head = NULL;
380 bool null_rx =
false;
381 regmatch_t pmatch[1] = { 0 };
400 if (offset >= (buflen - has_nl))
413 if ((regexec(&color_line->
regex, pat + offset, 1, pmatch,
414 ((offset != 0) ? REG_NOTBOL : 0)) != 0))
423 if (pmatch[0].rm_eo == pmatch[0].rm_so)
432 if (lines[line_num].syntax_arr_size == SHRT_MAX)
437 if (++(lines[line_num].syntax_arr_size) > 1)
440 lines[line_num].syntax_arr_size,
struct TextSyntax);
444 memset(ts, 0,
sizeof(*ts));
448 pmatch[0].rm_so += offset;
449 pmatch[0].rm_eo += offset;
451 if (!found || (pmatch[0].rm_so < (lines[line_num].syntax)[i].
first) ||
452 ((pmatch[0].rm_so == (lines[line_num].syntax)[i].
first) &&
453 (pmatch[0].rm_eo > (lines[line_num].syntax)[i].
last)))
456 (lines[line_num].
syntax)[i].first = pmatch[0].rm_so;
457 (lines[line_num].
syntax)[i].
last = pmatch[0].rm_eo;
459 else if ((pmatch[0].rm_so == (lines[line_num].syntax)[i].first) &&
460 (pmatch[0].rm_eo == (lines[line_num].syntax)[i].last))
463 (lines[line_num].syntax)[i].attr_color, &color_line->
attr_color);
464 (lines[line_num].
syntax)[i].
first = pmatch[0].rm_so;
465 (lines[line_num].
syntax)[i].
last = pmatch[0].rm_eo;
476 }
while (found || null_rx);
479 pat[buflen - 1] =
'\n';
506 struct Line *lines,
int line_num,
int lines_used,
508 bool *force_redraw,
bool q_classify)
511 regmatch_t pmatch[1] = { 0 };
527 if ((line_num > 0) && ((buf[0] ==
' ') || (buf[0] ==
'\t')))
529 lines[line_num].
cid = lines[line_num - 1].
cid;
530 if (!c_header_color_partial)
533 (lines[line_num - 1].syntax)[0].attr_color;
545 if (!c_header_color_partial)
549 if (regexec(&color_line->
regex, buf, 0, NULL, 0) == 0)
557 if (lines[line_num].cont_header)
561 for (j = line_num - 1; j >= 0 && lines[j].
cont_header; --j)
563 lines[j].
cid = lines[line_num].
cid;
569 lines[j].
cid = lines[line_num].
cid;
572 *force_redraw =
true;
592 while ((i < lines_used) && (
check_sig(buf, lines, i - 1) == 0) &&
597 if (lines[i].syntax_arr_size)
605 else if (
check_sig(buf, lines, line_num - 1) == 0)
611 if (q_classify && !lines[line_num].quote)
614 pmatch[0].rm_eo - pmatch[0].rm_so,
615 force_redraw, q_level);
638 if ((nl > 0) && (buf[nl - 1] ==
'\n'))
646 bool null_rx =
false;
656 if (regexec(&color_line->
regex, buf + offset, 1, pmatch,
657 ((offset != 0) ? REG_NOTBOL : 0)) != 0)
662 if (pmatch[0].rm_eo != pmatch[0].rm_so)
666 if (++(lines[line_num].syntax_arr_size) > 1)
669 lines[line_num].syntax_arr_size,
struct TextSyntax);
673 memset(ts, 0,
sizeof(*ts));
677 pmatch[0].rm_so += offset;
678 pmatch[0].rm_eo += offset;
679 if (!found || (pmatch[0].rm_so < (lines[line_num].syntax)[i].
first) ||
680 ((pmatch[0].rm_so == (lines[line_num].syntax)[i].
first) &&
681 (pmatch[0].rm_eo > (lines[line_num].syntax)[i].
last)))
687 (lines[line_num].syntax)[i].attr_color, &color_line->
attr_color);
688 (lines[line_num].
syntax)[i].
first = pmatch[0].rm_so;
689 (lines[line_num].
syntax)[i].
last = pmatch[0].rm_eo;
691 else if ((pmatch[0].rm_so == (lines[line_num].syntax)[i].first) &&
692 (pmatch[0].rm_eo == (lines[line_num].syntax)[i].last))
695 (lines[line_num].syntax)[i].attr_color, &color_line->
attr_color);
696 (lines[line_num].
syntax)[i].
first = pmatch[0].rm_so;
697 (lines[line_num].
syntax)[i].
last = pmatch[0].rm_eo;
712 }
while (found || null_rx);
742 if ((s[0] ==
'\010') && (s > src))
748 else if (s[1] &&
buf_len(dest))
766 else if (strip_markers && (s[0] ==
'\033') && (s[1] ==
']') &&
792static int fill_buffer(FILE *fp, LOFF_T *bytes_read, LOFF_T offset,
unsigned char **buf,
793 unsigned char **fmt,
size_t *blen,
int *buf_ready)
795 static int b_read = 0;
799 if (offset != *bytes_read)
814 *bytes_read = ftello(fp);
815 b_read = (int) (*bytes_read - offset);
823 *fmt = (
unsigned char *)
buf_strdup(stripped);
849 int cnt,
int *pspace,
int *pvch,
int *pcol,
850 int *pspecial,
int width,
struct AttrColorList *ansi_list)
854 size_t col = c_markers ? (*lines)[line_num].cont_line : 0;
856 int ch, vch, last_special = -1, special = 0, t;
858 mbstate_t mbstate = { 0 };
870 for (ch = 0, vch = 0; ch < cnt; ch += k, vch += k)
873 if (buf[ch] ==
'\033')
879 while ((cnt - ch >= 2) && (buf[ch] ==
'\033') && (buf[ch + 1] ==
']') &&
883 while (buf[ch++] !=
'\a')
892 k = mbrtowc(&wc, (
char *) buf + ch, cnt - ch, &mbstate);
896 memset(&mbstate, 0,
sizeof(mbstate));
898 if ((col + 4) > wrap_cols)
912 if ((wc == 0x200B) || (wc == 0x200C) || (wc == 0xFEFF))
929 mbstate_t mbstate1 = mbstate;
930 size_t k1 = mbrtowc(&wc1, (
char *) buf + ch + k, cnt - ch - k, &mbstate1);
932 (k1 > 0) && (wc1 ==
'\b'))
934 const size_t k2 = mbrtowc(&wc1, (
char *) buf + ch + k + k1,
935 cnt - ch - k - k1, &mbstate1);
944 special |= ((wc ==
'_') && (special & A_UNDERLINE)) ? A_UNDERLINE : A_BOLD;
946 else if ((wc ==
'_') || (wc1 ==
'_'))
948 special |= A_UNDERLINE;
949 wc = (wc1 ==
'_') ? wc : wc1;
960 k1 = mbrtowc(&wc1, (
char *) buf + ch + k, cnt - ch - k, &mbstate1);
965 special || last_special || (ansi->
attrs != A_NORMAL)))
968 last_special = special;
979 if (col + t > wrap_cols)
996 for (; col < t; col++)
1001 else if ((wc < 0x20) || (wc == 0x7f))
1003 if ((col + 2) > wrap_cols)
1009 else if (wc < 0x100)
1011 if ((col + 4) > wrap_cols)
1019 if ((col + 1) > wrap_cols)
1029 *pspecial = special;
1058 unsigned char *buf = NULL, *fmt = NULL;
1060 unsigned char *buf_ptr = NULL;
1061 int ch, vch, col, cnt, b_read;
1063 bool change_last =
false;
1066 const struct AttrColor *def_color = NULL;
1070 regmatch_t pmatch[1] = { 0 };
1087 memset(&((*
lines)[ch]), 0,
sizeof(
struct Line));
1088 (*lines)[ch].cid = -1;
1089 (*lines)[ch].search_arr_size = -1;
1091 ((*lines)[ch].syntax)[0].
first = -1;
1092 ((*lines)[ch].syntax)[0].last = -1;
1096 struct Line *
const cur_line = &(*lines)[line_num];
1101 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1108 if ((cur_line->
cont_line) && (line_num > 0))
1110 struct Line *
const old_line = &(*lines)[line_num - 1];
1111 cur_line->
cid = old_line->
cid;
1119 else if (buf[11] ==
'W')
1121 else if (buf[11] ==
'E')
1131 if (cur_line->
cid == -1)
1134 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1143 resolve_types(win_pager, (
char *) fmt, (
char *) buf, *lines, line_num, *lines_used,
1152 for (m = line_num + 1;
1153 m < *lines_used && (*lines)[m].offset && (*lines)[m].cont_line; m++)
1155 (*lines)[m].cid = cur_line->
cid;
1162 (!cur_line->
quote || (cur_line->
quote->
quote_n >= c_toggle_quoted_show_levels)))
1176 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1187 pmatch[0].rm_eo - pmatch[0].rm_so,
1188 force_redraw, q_level);
1198 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1207 while (regexec(search_re, (
char *) fmt + offset, 1, pmatch,
1208 (offset ? REG_NOTBOL : 0)) == 0)
1216 memset(ts, 0,
sizeof(*ts));
1222 pmatch[0].rm_so += offset;
1223 pmatch[0].rm_eo += offset;
1227 if (pmatch[0].rm_eo == pmatch[0].rm_so)
1230 offset = pmatch[0].rm_eo;
1236 if (!(flags &
MUTT_SHOW) && ((*lines)[line_num + 1].offset > 0))
1242 if ((flags &
MUTT_SHOWCOLOR) && *force_redraw && ((*lines)[line_num + 1].offset > 0))
1249 b_read =
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready);
1258 cnt =
format_line(win_pager, lines, line_num, buf, flags, NULL, b_read, &ch,
1259 &vch, &col, &special, win_pager->
state.
cols, ansi_list);
1260 buf_ptr = buf + cnt;
1271 while (ch && ((buf[ch] ==
' ') || (buf[ch] ==
'\t') || (buf[ch] ==
'\r')))
1279 buf_ptr = buf + cnt;
1286 while ((*buf_ptr ==
' ') || (*buf_ptr ==
'\t'))
1291 if (*buf_ptr ==
'\r')
1293 if (*buf_ptr ==
'\n')
1296 if (((
int) (buf_ptr - buf) < b_read) && !(*lines)[line_num + 1].cont_line)
1297 append_line(*lines, line_num, (
int) (buf_ptr - buf));
1298 (*lines)[line_num + 1].offset = cur_line->
offset + (long) (buf_ptr - buf);
1314 format_line(win_pager, lines, line_num, buf, flags, &ansi, cnt, &ch, &vch,
1315 &col, &special, win_pager->
state.
cols, ansi_list);
1341 def_color = ((*lines)[m].syntax)[0].
attr_color;
1356 if (col < win_pager->state.cols)
int ansi_color_parse(const char *str, struct AnsiColor *ansi, struct AttrColorList *acl, bool dry_run)
Parse a string of ANSI escape sequence.
bool attr_color_match(struct AttrColor *ac1, struct AttrColor *ac2)
Do the colours match?
struct AttrColor attr_color_copy(const struct AttrColor *ac)
Copy a colour.
size_t buf_len(const struct Buffer *buf)
Calculate the length of a Buffer.
void buf_reset(struct Buffer *buf)
Reset an existing Buffer.
size_t buf_addch(struct Buffer *buf, char c)
Add a single character to a Buffer.
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
void buf_alloc(struct Buffer *buf, size_t new_size)
Make sure a buffer can store at least new_size bytes.
Color and attribute parsing.
struct RegexColorList * regex_colors_get_list(enum ColorId cid)
Return the RegexColorList for a colour id.
bool simple_color_is_set(enum ColorId cid)
Is the object coloured?
struct AttrColor * simple_color_get(enum ColorId cid)
Get the colour of an object by its ID.
ColorId
List of all coloured objects.
@ MT_COLOR_MARKERS
Pager: markers, line continuation.
@ MT_COLOR_MESSAGE
Informational message.
@ MT_COLOR_QUOTED
Pager: quoted text.
@ MT_COLOR_HEADER
Message headers (takes a pattern)
@ MT_COLOR_STRIPE_EVEN
Stripes: even lines of the Help Page.
@ MT_COLOR_ERROR
Error message.
@ MT_COLOR_BOLD
Bold text.
@ MT_COLOR_BODY
Pager: highlight body of message (takes a pattern)
@ MT_COLOR_HDRDEFAULT
Header default colour.
@ MT_COLOR_NORMAL
Plain text.
@ MT_COLOR_ATTACH_HEADERS
MIME attachment test (takes a pattern)
@ MT_COLOR_SEARCH
Pager: search matches.
@ MT_COLOR_MESSAGE_LOG
Menu showing log messages.
@ MT_COLOR_ITALIC
Italic text.
@ MT_COLOR_STRIPE_ODD
Stripes: odd lines of the Help Page.
@ MT_COLOR_ATTACHMENT
MIME attachments text (entire line)
@ MT_COLOR_WARNING
Warning messages.
@ MT_COLOR_UNDERLINE
Underlined text.
@ MT_COLOR_SIGNATURE
Pager: signature lines.
const struct Regex * cs_subset_regex(const struct ConfigSubset *sub, const char *name)
Get a regex config item by name.
short cs_subset_number(const struct ConfigSubset *sub, const char *name)
Get a number config item by name.
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Convenience wrapper for the config headers.
Convenience wrapper for the core headers.
int mutt_addwch(struct MuttWindow *win, wchar_t wc)
Addwch would be provided by an up-to-date curses library.
static int format_line(struct MuttWindow *win, struct Line **lines, int line_num, unsigned char *buf, PagerFlags flags, struct AnsiColor *ansi, int cnt, int *pspace, int *pvch, int *pcol, int *pspecial, int width, struct AttrColorList *ansi_list)
Display a line of text in the pager.
static int check_protected_header_marker(const char *p)
Check that the unique marker is present.
bool mutt_is_quote_line(char *line, regmatch_t *pmatch)
Is a line of message text a quote?
static int fill_buffer(FILE *fp, LOFF_T *bytes_read, LOFF_T offset, unsigned char **buf, unsigned char **fmt, size_t *blen, int *buf_ready)
Fill a buffer from a file.
static int check_sig(const char *s, struct Line *info, int offset)
Check for an email signature.
static void match_body_patterns(char *pat, struct Line *lines, int line_num)
Match body patterns, e.g.
void buf_strip_formatting(struct Buffer *dest, const char *src, bool strip_markers)
Removes ANSI and backspace formatting.
static int comp_syntax_t(const void *m1, const void *m2)
Search for a Syntax using bsearch(3)
bool color_is_header(enum ColorId cid)
Colour is for an Email header.
static int check_attachment_marker(const char *p)
Check that the unique marker is present.
static void resolve_color(struct MuttWindow *win, struct Line *lines, int line_num, int cnt, PagerFlags flags, int special, struct AnsiColor *ansi)
Set the colour for a line of text.
static void resolve_types(struct MuttWindow *win, char *buf, char *raw, struct Line *lines, int line_num, int lines_used, struct QuoteStyle **quote_list, int *q_level, bool *force_redraw, bool q_classify)
Determine the style for a line of text.
int display_line(FILE *fp, LOFF_T *bytes_read, struct Line **lines, int line_num, int *lines_used, int *lines_max, PagerFlags flags, struct QuoteStyle **quote_list, int *q_level, bool *force_redraw, regex_t *search_re, struct MuttWindow *win_pager, struct AttrColorList *ansi_list)
Print a line on screen.
static void append_line(struct Line *lines, int line_num, int cnt)
Add a new Line to the array.
static int check_marker(const char *q, const char *p)
Check that the unique marker is present.
char * mutt_file_read_line(char *line, size_t *size, FILE *fp, int *line_num, ReadLineFlags flags)
Read a line from a file.
bool mutt_file_seek(FILE *fp, LOFF_T offset, int whence)
Wrapper for fseeko with error handling.
#define MUTT_RL_EOL
don't strip \n / \r\n
#define mutt_debug(LEVEL,...)
Convenience wrapper for the gui headers.
@ LL_DEBUG3
Log at debug level 3.
@ LL_DEBUG2
Log at debug level 2.
@ LL_DEBUG1
Log at debug level 1.
bool mutt_mb_is_display_corrupting_utf8(wchar_t wc)
Will this character corrupt the display?
#define MUTT_MEM_CALLOC(n, type)
#define MUTT_MEM_REALLOC(pptr, n, type)
const struct AttrColor * merged_color_overlay(const struct AttrColor *base, const struct AttrColor *over)
Combine two colours.
bool CharsetIsUtf8
Is the user's current character set utf-8?
wchar_t ReplacementChar
When a Unicode character can't be displayed, use this instead.
#define ICONV_BUF_TOO_SMALL
Error value for iconv() - Buffer too small.
#define ICONV_ILLEGAL_SEQ
Error value for iconv() - Illegal sequence.
Convenience wrapper for the library headers.
bool mutt_regex_capture(const struct Regex *regex, const char *str, size_t nmatch, regmatch_t matches[])
Match a regex against a string, with provided options.
const char * state_attachment_marker(void)
Get a unique (per-run) ANSI string to mark PGP messages in an email.
const char * state_protected_header_marker(void)
Get a unique (per-run) ANSI string to mark protected headers in an email.
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
size_t mutt_str_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix.
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
const struct AttrColor * mutt_curses_set_color_by_id(enum ColorId cid)
Set the colour and attributes by the colour id.
void mutt_curses_set_color(const struct AttrColor *ac)
Set the colour and attributes for text.
int mutt_window_printf(struct MuttWindow *win, const char *fmt,...)
Write a formatted string to a Window.
void mutt_window_get_coords(struct MuttWindow *win, int *col, int *row)
Get the cursor position in the Window.
int mutt_window_wrap_cols(int width, short wrap)
Calculate the wrap column for a given screen width.
void mutt_window_clrtoeol(struct MuttWindow *win)
Clear to the end of the line.
int mutt_window_addch(struct MuttWindow *win, int ch)
Write one character to a Window.
int ansi_color_seq_length(const char *str)
Is this an ANSI escape sequence?
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.
struct QuoteStyle * qstyle_classify(struct QuoteStyle **quote_list, const char *qptr, size_t length, bool *force_redraw, int *q_level)
Find a style for a string.
#define STAILQ_FOREACH(var, head, field)
#define COLOR_QUOTED(cid)
int attrs
Text attributes, e.g. A_BOLD.
const struct AttrColor * attr_color
Curses colour of text.
A curses colour and its attributes.
int attrs
Text attributes, e.g. A_BOLD.
struct CursesColor * curses_color
Underlying Curses colour.
String manipulation buffer.
char * dptr
Current read/write position.
A line of text in the pager.
short search_arr_size
Number of items in search array.
struct TextSyntax * search
Array of search text in the line.
bool cont_line
Continuation of a previous line (wrapped by NeoMutt)
short cid
Default line colour, e.g. MT_COLOR_SIGNATURE.
struct QuoteStyle * quote
Quoting style for this line (pointer into PagerPrivateData->quote_list)
LOFF_T offset
Offset into Email file (PagerPrivateData->fp)
bool cont_header
Continuation of a header line (wrapped by MTA)
short syntax_arr_size
Number of items in syntax array.
struct TextSyntax * syntax
Array of coloured text in the line.
struct WindowState state
Current state of the Window.
void * wdata
Private data.
struct MuttWindow * parent
Parent Window.
Container for Accounts, Notifications.
struct ConfigSubset * sub
Inherited config items.
struct AttrColor * attr_color
Colour and attribute of the text.
size_t prefix_len
Length of the prefix string.
int quote_n
The quoteN colour index for this level.
A regular expression and a color to highlight a line.
regex_t regex
Compiled regex.
struct AttrColor attr_color
Colour and attributes to apply.
bool stop_matching
Used by the pager for body patterns, to prevent the color from being retried once it fails.
Cached regular expression.
Highlighting for a piece of text.
const struct AttrColor * attr_color
Curses colour of text.
int last
Last character in line to be coloured (not included)
int first
First character in line to be coloured.
short cols
Number of columns, can be MUTT_WIN_SIZE_UNLIMITED.