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';
496 struct Line *lines,
int line_num,
int lines_used,
498 bool *force_redraw,
bool q_classify)
501 regmatch_t pmatch[1] = { 0 };
517 if ((line_num > 0) && ((buf[0] ==
' ') || (buf[0] ==
'\t')))
519 lines[line_num].
cid = lines[line_num - 1].
cid;
520 if (!c_header_color_partial)
523 (lines[line_num - 1].syntax)[0].attr_color;
535 if (!c_header_color_partial)
539 if (regexec(&color_line->
regex, buf, 0, NULL, 0) == 0)
547 if (lines[line_num].cont_header)
551 for (j = line_num - 1; j >= 0 && lines[j].
cont_header; --j)
553 lines[j].
cid = lines[line_num].
cid;
559 lines[j].
cid = lines[line_num].
cid;
562 *force_redraw =
true;
582 while ((i < lines_used) && (
check_sig(buf, lines, i - 1) == 0) &&
587 if (lines[i].syntax_arr_size)
595 else if (
check_sig(buf, lines, line_num - 1) == 0)
601 if (q_classify && !lines[line_num].quote)
604 pmatch[0].rm_eo - pmatch[0].rm_so,
605 force_redraw, q_level);
628 if ((nl > 0) && (buf[nl - 1] ==
'\n'))
636 bool null_rx =
false;
646 if (regexec(&color_line->
regex, buf + offset, 1, pmatch,
647 ((offset != 0) ? REG_NOTBOL : 0)) != 0)
652 if (pmatch[0].rm_eo != pmatch[0].rm_so)
656 if (++(lines[line_num].syntax_arr_size) > 1)
659 lines[line_num].syntax_arr_size,
struct TextSyntax);
663 memset(ts, 0,
sizeof(*ts));
667 pmatch[0].rm_so += offset;
668 pmatch[0].rm_eo += offset;
669 if (!found || (pmatch[0].rm_so < (lines[line_num].syntax)[i].
first) ||
670 ((pmatch[0].rm_so == (lines[line_num].syntax)[i].
first) &&
671 (pmatch[0].rm_eo > (lines[line_num].syntax)[i].
last)))
677 (lines[line_num].syntax)[i].attr_color, &color_line->
attr_color);
678 (lines[line_num].
syntax)[i].
first = pmatch[0].rm_so;
679 (lines[line_num].
syntax)[i].
last = pmatch[0].rm_eo;
681 else if ((pmatch[0].rm_so == (lines[line_num].syntax)[i].first) &&
682 (pmatch[0].rm_eo == (lines[line_num].syntax)[i].last))
685 (lines[line_num].syntax)[i].attr_color, &color_line->
attr_color);
686 (lines[line_num].
syntax)[i].
first = pmatch[0].rm_so;
687 (lines[line_num].
syntax)[i].
last = pmatch[0].rm_eo;
702 }
while (found || null_rx);
732 if ((s[0] ==
'\010') && (s > src))
738 else if (s[1] &&
buf_len(dest))
756 else if (strip_markers && (s[0] ==
'\033') && (s[1] ==
']') &&
782static int fill_buffer(FILE *fp, LOFF_T *bytes_read, LOFF_T offset,
unsigned char **buf,
783 unsigned char **fmt,
size_t *blen,
int *buf_ready)
785 static int b_read = 0;
789 if (offset != *bytes_read)
804 *bytes_read = ftello(fp);
805 b_read = (int) (*bytes_read - offset);
813 *fmt = (
unsigned char *)
buf_strdup(stripped);
839 int cnt,
int *pspace,
int *pvch,
int *pcol,
840 int *pspecial,
int width,
struct AttrColorList *ansi_list)
844 size_t col = c_markers ? (*lines)[line_num].cont_line : 0;
846 int ch, vch, last_special = -1, special = 0, t;
848 mbstate_t mbstate = { 0 };
860 for (ch = 0, vch = 0; ch < cnt; ch += k, vch += k)
863 if (buf[ch] ==
'\033')
869 while ((cnt - ch >= 2) && (buf[ch] ==
'\033') && (buf[ch + 1] ==
']') &&
873 while (buf[ch++] !=
'\a')
882 k = mbrtowc(&wc, (
char *) buf + ch, cnt - ch, &mbstate);
886 memset(&mbstate, 0,
sizeof(mbstate));
888 if ((col + 4) > wrap_cols)
902 if ((wc == 0x200B) || (wc == 0x200C) || (wc == 0xFEFF))
919 mbstate_t mbstate1 = mbstate;
920 size_t k1 = mbrtowc(&wc1, (
char *) buf + ch + k, cnt - ch - k, &mbstate1);
922 (k1 > 0) && (wc1 ==
'\b'))
924 const size_t k2 = mbrtowc(&wc1, (
char *) buf + ch + k + k1,
925 cnt - ch - k - k1, &mbstate1);
934 special |= ((wc ==
'_') && (special & A_UNDERLINE)) ? A_UNDERLINE : A_BOLD;
936 else if ((wc ==
'_') || (wc1 ==
'_'))
938 special |= A_UNDERLINE;
939 wc = (wc1 ==
'_') ? wc : wc1;
950 k1 = mbrtowc(&wc1, (
char *) buf + ch + k, cnt - ch - k, &mbstate1);
955 special || last_special || (ansi->
attrs != A_NORMAL)))
958 last_special = special;
969 if (col + t > wrap_cols)
986 for (; col < t; col++)
991 else if ((wc < 0x20) || (wc == 0x7f))
993 if ((col + 2) > wrap_cols)
1001 if ((col + 4) > wrap_cols)
1009 if ((col + 1) > wrap_cols)
1019 *pspecial = special;
1048 unsigned char *buf = NULL, *fmt = NULL;
1050 unsigned char *buf_ptr = NULL;
1051 int ch, vch, col, cnt, b_read;
1053 bool change_last =
false;
1056 const struct AttrColor *def_color = NULL;
1060 regmatch_t pmatch[1] = { 0 };
1077 memset(&((*
lines)[ch]), 0,
sizeof(
struct Line));
1078 (*lines)[ch].cid = -1;
1079 (*lines)[ch].search_arr_size = -1;
1081 ((*lines)[ch].syntax)[0].
first = -1;
1082 ((*lines)[ch].syntax)[0].last = -1;
1086 struct Line *
const cur_line = &(*lines)[line_num];
1091 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1098 if ((cur_line->
cont_line) && (line_num > 0))
1100 struct Line *
const old_line = &(*lines)[line_num - 1];
1101 cur_line->
cid = old_line->
cid;
1109 else if (buf[11] ==
'W')
1111 else if (buf[11] ==
'E')
1121 if (cur_line->
cid == -1)
1124 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1133 resolve_types(win_pager, (
char *) fmt, (
char *) buf, *lines, line_num, *lines_used,
1142 for (m = line_num + 1;
1143 m < *lines_used && (*lines)[m].offset && (*lines)[m].cont_line; m++)
1145 (*lines)[m].cid = cur_line->
cid;
1152 (!cur_line->
quote || (cur_line->
quote->
quote_n >= c_toggle_quoted_show_levels)))
1166 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1177 pmatch[0].rm_eo - pmatch[0].rm_so,
1178 force_redraw, q_level);
1188 if (
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1197 while (regexec(search_re, (
char *) fmt + offset, 1, pmatch,
1198 (offset ? REG_NOTBOL : 0)) == 0)
1206 memset(ts, 0,
sizeof(*ts));
1212 pmatch[0].rm_so += offset;
1213 pmatch[0].rm_eo += offset;
1217 if (pmatch[0].rm_eo == pmatch[0].rm_so)
1220 offset = pmatch[0].rm_eo;
1226 if (!(flags &
MUTT_SHOW) && ((*lines)[line_num + 1].offset > 0))
1232 if ((flags &
MUTT_SHOWCOLOR) && *force_redraw && ((*lines)[line_num + 1].offset > 0))
1239 b_read =
fill_buffer(fp, bytes_read, cur_line->
offset, &buf, &fmt, &buflen, &buf_ready);
1248 cnt =
format_line(win_pager, lines, line_num, buf, flags, NULL, b_read, &ch,
1249 &vch, &col, &special, win_pager->
state.
cols, ansi_list);
1250 buf_ptr = buf + cnt;
1256 if ((cnt < b_read) && (ch != -1) &&
1261 while (ch && ((buf[ch] ==
' ') || (buf[ch] ==
'\t') || (buf[ch] ==
'\r')))
1269 buf_ptr = buf + cnt;
1276 while ((*buf_ptr ==
' ') || (*buf_ptr ==
'\t'))
1281 if (*buf_ptr ==
'\r')
1283 if (*buf_ptr ==
'\n')
1286 if (((
int) (buf_ptr - buf) < b_read) && !(*lines)[line_num + 1].cont_line)
1287 append_line(*lines, line_num, (
int) (buf_ptr - buf));
1288 (*lines)[line_num + 1].offset = cur_line->
offset + (long) (buf_ptr - buf);
1304 format_line(win_pager, lines, line_num, buf, flags, &ansi, cnt, &ch, &vch,
1305 &col, &special, win_pager->
state.
cols, ansi_list);
1331 def_color = ((*lines)[m].syntax)[0].
attr_color;
1346 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_header(enum ColorId cid)
Colour is for an Email header.
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 colored 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)
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.
#define STAILQ_FOREACH(var, head, field)
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.
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_QUOTED.
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.