NeoMutt  2025-09-05-7-geaa2bd
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
display.h File Reference

Pager Display. More...

#include "config.h"
#include <stdbool.h>
#include <stdio.h>
#include "mutt/lib.h"
#include "lib.h"
#include "color/lib.h"
+ Include dependency graph for display.h:
+ This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Data Structures

struct  TextSyntax
 Highlighting for a piece of text. More...
 
struct  Line
 A line of text in the pager. More...
 

Functions

 ARRAY_HEAD (TextSyntaxArray, struct TextSyntax)
 
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.
 
bool color_is_header (enum ColorId cid)
 Colour is for an Email header.
 

Detailed Description

Pager Display.

Authors
  • Richard Russon

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 display.h.

Function Documentation

◆ ARRAY_HEAD()

ARRAY_HEAD ( TextSyntaxArray  ,
struct TextSyntax   
)

◆ display_line()

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.

Parameters
[in]fpFile to read from
[out]bytes_readOffset into file
[out]linesLine attributes
[in]line_numLine number
[out]lines_usedLast line
[out]lines_maxMaximum number of lines
[in]flagsFlags, see PagerFlags
[out]quote_listEmail quoting style
[out]q_levelLevel of quoting
[out]force_redrawForce a repaint
[out]search_reRegex to highlight
[in]win_pagerWindow to draw into
[in]ansi_listList of ANSI colours/attributes
Return values
-1EOF was reached
0normal exit, line was not displayed
>0normal exit, line was displayed

Definition at line 1051 of file display.c.

1056{
1057 unsigned char *buf = NULL, *fmt = NULL;
1058 size_t buflen = 0;
1059 unsigned char *buf_ptr = NULL;
1060 int ch, vch, col, cnt, b_read;
1061 int buf_ready = 0;
1062 bool change_last = false;
1063 int special;
1064 int offset;
1065 const struct AttrColor *def_color = NULL;
1066 int m;
1067 int rc = -1;
1068 struct AnsiColor ansi = { { COLOR_DEFAULT, 0, 0 }, { COLOR_DEFAULT, 0, 0 }, 0, NULL };
1069 regmatch_t pmatch[1] = { 0 };
1070
1071 struct PagerPrivateData *priv = win_pager->parent->wdata;
1072 enum PagerMode mode = priv->pview->mode;
1073
1074 if (line_num == *lines_used)
1075 {
1076 (*lines_used)++;
1077 change_last = true;
1078 }
1079
1080 if (*lines_used == *lines_max)
1081 {
1082 *lines_max += LINES;
1084 for (ch = *lines_used; ch < *lines_max; ch++)
1085 {
1086 memset(&((*lines)[ch]), 0, sizeof(struct Line));
1087 (*lines)[ch].cid = -1;
1088 (*lines)[ch].search_arr_size = -1;
1089 (*lines)[ch].syntax = MUTT_MEM_CALLOC(1, struct TextSyntax);
1090 ((*lines)[ch].syntax)[0].first = -1;
1091 ((*lines)[ch].syntax)[0].last = -1;
1092 }
1093 }
1094
1095 struct Line *const cur_line = &(*lines)[line_num];
1096
1097 if (flags & MUTT_PAGER_LOGS)
1098 {
1099 /* determine the line class */
1100 if (fill_buffer(fp, bytes_read, cur_line->offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1101 {
1102 if (change_last)
1103 (*lines_used)--;
1104 goto out;
1105 }
1106
1107 if ((cur_line->cont_line) && (line_num > 0))
1108 {
1109 struct Line *const old_line = &(*lines)[line_num - 1];
1110 cur_line->cid = old_line->cid;
1111 cur_line->syntax[0].attr_color = old_line->syntax[0].attr_color;
1112 }
1113 else
1114 {
1115 cur_line->cid = MT_COLOR_NORMAL;
1116 if (buf[11] == 'M')
1118 else if (buf[11] == 'W')
1120 else if (buf[11] == 'E')
1122 else
1124 }
1125 }
1126
1127 /* only do color highlighting if we are viewing a message */
1128 if (flags & (MUTT_SHOWCOLOR | MUTT_TYPES))
1129 {
1130 if (cur_line->cid == -1)
1131 {
1132 /* determine the line class */
1133 if (fill_buffer(fp, bytes_read, cur_line->offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1134 {
1135 if (change_last)
1136 (*lines_used)--;
1137 goto out;
1138 }
1139
1140 if (mode == PAGER_MODE_EMAIL)
1141 {
1142 resolve_types(win_pager, (char *) fmt, (char *) buf, *lines, line_num, *lines_used,
1143 quote_list, q_level, force_redraw, flags & MUTT_SHOWCOLOR);
1144 }
1145 else
1146 {
1147 (*lines)[line_num].cid = MT_COLOR_NORMAL;
1148 }
1149
1150 /* avoid race condition for continuation lines when scrolling up */
1151 for (m = line_num + 1;
1152 m < *lines_used && (*lines)[m].offset && (*lines)[m].cont_line; m++)
1153 {
1154 (*lines)[m].cid = cur_line->cid;
1155 }
1156 }
1157
1158 /* this also prevents searching through the hidden lines */
1159 const short c_toggle_quoted_show_levels = cs_subset_number(NeoMutt->sub, "toggle_quoted_show_levels");
1160 if ((flags & MUTT_HIDE) && COLOR_QUOTED(cur_line->cid) &&
1161 (!cur_line->quote || (cur_line->quote->quote_n >= c_toggle_quoted_show_levels)))
1162 {
1163 flags = 0; /* MUTT_NOSHOW */
1164 }
1165 }
1166
1167 /* At this point, (*lines[line_num]).quote may still be undefined. We
1168 * don't want to compute it every time MUTT_TYPES is set, since this
1169 * would slow down the "bottom" function unacceptably. A compromise
1170 * solution is hence to call regexec() again, just to find out the
1171 * length of the quote prefix. */
1172 if ((flags & MUTT_SHOWCOLOR) && !cur_line->cont_line &&
1173 COLOR_QUOTED(cur_line->cid) && !cur_line->quote)
1174 {
1175 if (fill_buffer(fp, bytes_read, cur_line->offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1176 {
1177 if (change_last)
1178 (*lines_used)--;
1179 goto out;
1180 }
1181
1182 const struct Regex *c_quote_regex = cs_subset_regex(NeoMutt->sub, "quote_regex");
1183 if (mutt_regex_capture(c_quote_regex, (char *) fmt, 1, pmatch))
1184 {
1185 cur_line->quote = qstyle_classify(quote_list, (char *) fmt + pmatch[0].rm_so,
1186 pmatch[0].rm_eo - pmatch[0].rm_so,
1187 force_redraw, q_level);
1188 }
1189 else
1190 {
1191 goto out;
1192 }
1193 }
1194
1195 if ((flags & MUTT_SEARCH) && !cur_line->cont_line && (cur_line->search_arr_size == -1))
1196 {
1197 if (fill_buffer(fp, bytes_read, cur_line->offset, &buf, &fmt, &buflen, &buf_ready) < 0)
1198 {
1199 if (change_last)
1200 (*lines_used)--;
1201 goto out;
1202 }
1203
1204 offset = 0;
1205 cur_line->search_arr_size = 0;
1206 while (regexec(search_re, (char *) fmt + offset, 1, pmatch,
1207 (offset ? REG_NOTBOL : 0)) == 0)
1208 {
1209 if (++(cur_line->search_arr_size) > 1)
1210 {
1211 MUTT_MEM_REALLOC(&(cur_line->search), cur_line->search_arr_size, struct TextSyntax);
1212 // Zero the new entry
1213 const int index = cur_line->search_arr_size - 1;
1214 struct TextSyntax *ts = &cur_line->search[index];
1215 memset(ts, 0, sizeof(*ts));
1216 }
1217 else
1218 {
1219 cur_line->search = MUTT_MEM_CALLOC(1, struct TextSyntax);
1220 }
1221 pmatch[0].rm_so += offset;
1222 pmatch[0].rm_eo += offset;
1223 (cur_line->search)[cur_line->search_arr_size - 1].first = pmatch[0].rm_so;
1224 (cur_line->search)[cur_line->search_arr_size - 1].last = pmatch[0].rm_eo;
1225
1226 if (pmatch[0].rm_eo == pmatch[0].rm_so)
1227 offset++; /* avoid degenerate cases */
1228 else
1229 offset = pmatch[0].rm_eo;
1230 if (!fmt[offset])
1231 break;
1232 }
1233 }
1234
1235 if (!(flags & MUTT_SHOW) && ((*lines)[line_num + 1].offset > 0))
1236 {
1237 /* we've already scanned this line, so just exit */
1238 rc = 0;
1239 goto out;
1240 }
1241 if ((flags & MUTT_SHOWCOLOR) && *force_redraw && ((*lines)[line_num + 1].offset > 0))
1242 {
1243 /* no need to try to display this line... */
1244 rc = 1;
1245 goto out; /* fake display */
1246 }
1247
1248 b_read = fill_buffer(fp, bytes_read, cur_line->offset, &buf, &fmt, &buflen, &buf_ready);
1249 if (b_read < 0)
1250 {
1251 if (change_last)
1252 (*lines_used)--;
1253 goto out;
1254 }
1255
1256 /* now chose a good place to break the line */
1257 cnt = format_line(win_pager, lines, line_num, buf, flags, NULL, b_read, &ch,
1258 &vch, &col, &special, win_pager->state.cols, ansi_list);
1259 buf_ptr = buf + cnt;
1260
1261 /* move the break point only if smart_wrap is set */
1262 const bool c_smart_wrap = cs_subset_bool(NeoMutt->sub, "smart_wrap");
1263 if (c_smart_wrap)
1264 {
1265 if ((cnt < b_read) && (ch != -1) && !color_is_header(cur_line->cid) &&
1266 !mutt_isspace(buf[cnt]))
1267 {
1268 buf_ptr = buf + ch;
1269 /* skip trailing blanks */
1270 while (ch && ((buf[ch] == ' ') || (buf[ch] == '\t') || (buf[ch] == '\r')))
1271 ch--;
1272
1273 /* A folded header with a single long word shouldn't be smartwrapped.
1274 * So just disable smart_wrap if it would wrap at the beginning of the line. */
1275 if (ch == 0)
1276 buf_ptr = buf + cnt;
1277 else
1278 cnt = ch + 1;
1279 }
1280
1281 /* skip leading blanks on the next line too */
1282 while ((*buf_ptr == ' ') || (*buf_ptr == '\t'))
1283 buf_ptr++;
1284 }
1285
1286 if (*buf_ptr == '\r')
1287 buf_ptr++;
1288 if (*buf_ptr == '\n')
1289 buf_ptr++;
1290
1291 if (((int) (buf_ptr - buf) < b_read) && !(*lines)[line_num + 1].cont_line)
1292 append_line(*lines, line_num, (int) (buf_ptr - buf));
1293 (*lines)[line_num + 1].offset = cur_line->offset + (long) (buf_ptr - buf);
1294
1295 /* if we don't need to display the line we are done */
1296 if (!(flags & MUTT_SHOW))
1297 {
1298 rc = 0;
1299 goto out;
1300 }
1301
1302 if (flags & MUTT_PAGER_STRIPES)
1303 {
1304 const enum ColorId cid = ((line_num % 2) == 0) ? MT_COLOR_STRIPE_ODD : MT_COLOR_STRIPE_EVEN;
1306 }
1307
1308 /* display the line */
1309 format_line(win_pager, lines, line_num, buf, flags, &ansi, cnt, &ch, &vch,
1310 &col, &special, win_pager->state.cols, ansi_list);
1311
1312 /* avoid a bug in ncurses... */
1313 if (col == 0)
1314 {
1315 if (flags & MUTT_PAGER_STRIPES)
1316 {
1317 const enum ColorId cid = ((line_num % 2) == 0) ? MT_COLOR_STRIPE_ODD : MT_COLOR_STRIPE_EVEN;
1319 }
1320 else
1321 {
1323 }
1324
1325 mutt_window_addch(win_pager, ' ');
1326 }
1327
1328 /* Fill the blank space at the end of the line with the prevailing color.
1329 * ncurses does an implicit clrtoeol() when you do mutt_window_addch('\n') so we have
1330 * to make sure to reset the color *after* that */
1331 if (flags & MUTT_SHOWCOLOR)
1332 {
1333 m = (cur_line->cont_line) ? (cur_line->syntax)[0].first : line_num;
1334 if ((*lines)[m].cid == MT_COLOR_HEADER)
1335 {
1336 def_color = ((*lines)[m].syntax)[0].attr_color;
1337 }
1338 else
1339 {
1340 def_color = simple_color_get((*lines)[m].cid);
1341 }
1342 const struct AttrColor *ac_normal = simple_color_get(MT_COLOR_NORMAL);
1343 const struct AttrColor *ac_eol = NULL;
1344 if (def_color)
1345 ac_eol = merged_color_overlay(ac_normal, def_color);
1346 else
1347 ac_eol = ac_normal;
1348 mutt_curses_set_color(ac_eol);
1349 }
1350
1351 if (col < win_pager->state.cols)
1352 {
1353 if (flags & MUTT_PAGER_STRIPES)
1354 {
1355 const enum ColorId cid = ((line_num % 2) == 0) ? MT_COLOR_STRIPE_ODD : MT_COLOR_STRIPE_EVEN;
1356 const struct AttrColor *ac_normal = simple_color_get(MT_COLOR_NORMAL);
1357 const struct AttrColor *stripe_color = simple_color_get(cid);
1358 const struct AttrColor *ac_eol = merged_color_overlay(ac_normal, stripe_color);
1359 mutt_curses_set_color(ac_eol);
1360 }
1361 else
1362 {
1363 // Colours may be disabled, but we may still need to end the "search" colour
1364 if (!(flags & MUTT_SHOWCOLOR))
1366 }
1367 mutt_window_clrtoeol(win_pager);
1368 }
1369
1370 /* reset the color back to normal. This *must* come after the
1371 * clrtoeol, otherwise the color for this line will not be
1372 * filled to the right margin. */
1373 if (flags & MUTT_SHOWCOLOR)
1375
1376 /* build a return code */
1377 if (!(flags & MUTT_SHOW))
1378 flags = 0;
1379
1380 rc = flags;
1381
1382out:
1383 FREE(&buf);
1384 FREE(&fmt);
1385 return rc;
1386}
struct AttrColor * simple_color_get(enum ColorId cid)
Get the colour of an object by its ID.
Definition: simple.c:95
#define COLOR_DEFAULT
Definition: color.h:103
ColorId
List of all coloured objects.
Definition: color.h:36
@ MT_COLOR_MESSAGE
Informational message.
Definition: color.h:53
@ MT_COLOR_HEADER
Message headers (takes a pattern)
Definition: color.h:49
@ MT_COLOR_STRIPE_EVEN
Stripes: even lines of the Help Page.
Definition: color.h:80
@ MT_COLOR_ERROR
Error message.
Definition: color.h:47
@ MT_COLOR_NORMAL
Plain text.
Definition: color.h:54
@ MT_COLOR_STRIPE_ODD
Stripes: odd lines of the Help Page.
Definition: color.h:81
@ MT_COLOR_WARNING
Warning messages.
Definition: color.h:85
const struct Regex * cs_subset_regex(const struct ConfigSubset *sub, const char *name)
Get a regex config item by name.
Definition: helpers.c:217
short cs_subset_number(const struct ConfigSubset *sub, const char *name)
Get a number config item by name.
Definition: helpers.c:143
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition: helpers.c:47
bool mutt_isspace(int arg)
Wrapper for isspace(3)
Definition: ctype.c:95
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.
Definition: display.c:846
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.
Definition: display.c:791
bool color_is_header(enum ColorId cid)
Colour is for an Email header.
Definition: display.c:486
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.
Definition: display.c:504
static void append_line(struct Line *lines, int line_num, int cnt)
Add a new Line to the array.
Definition: display.c:257
#define FREE(x)
Definition: memory.h:62
#define MUTT_MEM_CALLOC(n, type)
Definition: memory.h:47
#define MUTT_MEM_REALLOC(pptr, n, type)
Definition: memory.h:50
const struct AttrColor * merged_color_overlay(const struct AttrColor *base, const struct AttrColor *over)
Combine two colours.
Definition: merged.c:107
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.
Definition: regex.c:597
const struct AttrColor * mutt_curses_set_normal_backed_color_by_id(enum ColorId cid)
Set the colour and attributes by the Colour ID.
Definition: mutt_curses.c:63
const struct AttrColor * mutt_curses_set_color_by_id(enum ColorId cid)
Set the colour and attributes by the Colour ID.
Definition: mutt_curses.c:79
void mutt_curses_set_color(const struct AttrColor *ac)
Set the colour and attributes for text.
Definition: mutt_curses.c:38
void mutt_window_clrtoeol(struct MuttWindow *win)
Clear to the end of the line.
Definition: mutt_window.c:247
int mutt_window_addch(struct MuttWindow *win, int ch)
Write one character to a Window.
Definition: mutt_window.c:353
#define MUTT_HIDE
Don't show quoted text.
Definition: lib.h:63
#define MUTT_TYPES
Compute line's type.
Definition: lib.h:65
#define MUTT_PAGER_STRIPES
Striped highlighting.
Definition: lib.h:74
#define MUTT_SHOWCOLOR
Show characters in color otherwise don't show characters.
Definition: lib.h:62
#define MUTT_PAGER_LOGS
Logview mode.
Definition: lib.h:72
#define MUTT_SEARCH
Resolve search patterns.
Definition: lib.h:64
PagerMode
Determine the behaviour of the Pager.
Definition: lib.h:133
@ PAGER_MODE_EMAIL
Pager is invoked via 1st path. The mime part is selected automatically.
Definition: lib.h:136
#define MUTT_SHOW
Definition: lib.h:66
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.
Definition: qstyle.c:136
#define COLOR_QUOTED(cid)
Definition: quoted.h:28
An ANSI escape sequence.
Definition: ansi.h:35
A curses colour and its attributes.
Definition: attr.h:66
A line of text in the pager.
Definition: display.h:50
short search_arr_size
Number of items in search array.
Definition: display.h:59
struct TextSyntax * search
Array of search text in the line.
Definition: display.h:60
bool cont_line
Continuation of a previous line (wrapped by NeoMutt)
Definition: display.h:53
short cid
Default line colour, e.g. MT_COLOR_SIGNATURE.
Definition: display.h:52
struct QuoteStyle * quote
Quoting style for this line (pointer into PagerPrivateData->quote_list)
Definition: display.h:62
LOFF_T offset
Offset into Email file (PagerPrivateData->fp)
Definition: display.h:51
struct TextSyntax * syntax
Array of coloured text in the line.
Definition: display.h:57
struct WindowState state
Current state of the Window.
Definition: mutt_window.h:127
void * wdata
Private data.
Definition: mutt_window.h:145
struct MuttWindow * parent
Parent Window.
Definition: mutt_window.h:135
Container for Accounts, Notifications.
Definition: neomutt.h:43
struct ConfigSubset * sub
Inherited config items.
Definition: neomutt.h:47
Private state data for the Pager.
Definition: private_data.h:41
int lines_used
Size of lines array (used entries)
Definition: private_data.h:49
int lines_max
Capacity of lines array (total entries)
Definition: private_data.h:50
struct Line * lines
Array of text lines in pager.
Definition: private_data.h:48
bool first
First time flag for toggle-new.
Definition: private_data.h:75
struct PagerView * pview
Object to view in the pager.
Definition: private_data.h:42
enum PagerMode mode
Pager mode.
Definition: lib.h:172
int quote_n
The quoteN colour index for this level.
Definition: qstyle.h:57
Cached regular expression.
Definition: regex3.h:86
Highlighting for a piece of text.
Definition: display.h:39
const struct AttrColor * attr_color
Curses colour of text.
Definition: display.h:40
short cols
Number of columns, can be MUTT_WIN_SIZE_UNLIMITED.
Definition: mutt_window.h:61
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ color_is_header()

bool color_is_header ( enum ColorId  cid)

Colour is for an Email header.

Parameters
cidColour ID, e.g. MT_COLOR_HEADER
Return values
trueColour is for an Email header

Definition at line 486 of file display.c.

487{
488 return (cid == MT_COLOR_HEADER) || (cid == MT_COLOR_HDRDEFAULT);
489}
@ MT_COLOR_HDRDEFAULT
Header default colour.
Definition: color.h:48
+ Here is the caller graph for this function: