* Functions necessary to display a deck of slides in different color modes
* using ncurses. Only white, red, and blue are supported, as they can be
* faded in 256 color mode.
- * Copyright (C) 2016 Michael Goehler
+ * Copyright (C) 2018 Michael Goehler
*
* This file is part of mdp.
*
#include <unistd.h> // usleep
#include <stdlib.h> // getenv
#include "viewer.h"
+#include "config.h"
// color ramp for fading from black to color
static short white_ramp[24] = { 16, 232, 233, 234, 235, 236,
206, 207, 201, 200, 199, 199,
198, 198, 197, 197, 196, 196};
-// unordered list characters
-//
-// override via env vars:
-// export MDP_LIST_OPEN1=" " MDP_LIST_OPEN2=" " MDP_LIST_OPEN3=" "
-// export MDP_LIST_HEAD1=" ■ " MDP_LIST_HEAD2=" ● " MDP_LIST_HEAD3=" ▫ "
-static const char *list_open1 = " | ";
-static const char *list_open2 = " | ";
-static const char *list_open3 = " | ";
-static const char *list_head1 = " +- ";
-static const char *list_head2 = " +- ";
-static const char *list_head3 = " +- ";
-
-int ncurses_display(deck_t *deck, int notrans, int nofade, int invert, int reload, int noreload, int slidenum) {
+int ncurses_display(deck_t *deck, int notrans, int nofade, int invert, int reload, int noreload, int slidenum, int nocodebg) {
int c = 0; // char
int i = 0; // iterate
max_lines_slide = sc;
}
+ slide->lines_consumed = lc;
slide = slide->next;
++sc;
}
if(notrans) {
if(invert) {
- trans = 7; // white in 8 color mode
+ trans = FG_COLOR; // white in 8 color mode
} else {
- trans = 0; // black in 8 color mode
+ trans = BG_COLOR; // black in 8 color mode
}
}
if(invert) {
- init_pair(CP_WHITE, 0, trans);
- init_pair(CP_BLACK, 7, 0);
+ init_pair(CP_WHITE, BG_COLOR, trans);
+ init_pair(CP_BLACK, FG_COLOR, BG_COLOR);
} else {
- init_pair(CP_WHITE, 7, trans);
- init_pair(CP_BLACK, 0, 7);
+ init_pair(CP_WHITE, FG_COLOR, trans);
+ init_pair(CP_BLACK, BG_COLOR, FG_COLOR);
}
- init_pair(CP_BLUE, 4, trans);
- init_pair(CP_RED, 1, trans);
- init_pair(CP_YELLOW, 3, trans);
+ init_pair(CP_BLUE, HEADER_COLOR, trans);
+ init_pair(CP_RED, BOLD_COLOR, trans);
+ init_pair(CP_YELLOW, TITLE_COLOR, trans);
}
colors = 1;
// print lines
while(line) {
- add_line(content, l, (COLS - max_cols) / 2, line, max_cols, colors);
+ add_line(content, l + ((LINES - slide->lines_consumed - bar_top - bar_bottom) / 2),
+ (COLS - max_cols) / 2, line, max_cols, colors, nocodebg);
// raise stop counter if we pass a line having a stop bit
if(CHECK_BIT(line->bits, IS_STOP))
// evaluate user input
i = 0;
- switch(c) {
+ if (evaluate_binding(prev_slide_binding, c)) {
// show previous slide or stop bit
- case KEY_UP:
- case KEY_LEFT:
- case KEY_PPAGE:
- case 8: // BACKSPACE (ascii)
- case 127: // BACKSPACE (xterm)
- case 263: // BACKSPACE (getty)
- case 'h':
- case 'k':
- if(stop > 1 || (stop == 1 && !line)) {
- // show current slide again
- // but stop one stop bit earlier
- slide->stop--;
- fade = false;
+ if(stop > 1 || (stop == 1 && !line)) {
+ // show current slide again
+ // but stop one stop bit earlier
+ slide->stop--;
+ fade = false;
+ } else {
+ if(slide->prev) {
+ // show previous slide
+ slide = slide->prev;
+ sc--;
+ //stop on first bullet point always
+ if(slide->stop > 0)
+ slide->stop = 0;
} else {
- if(slide->prev) {
- // show previous slide
- slide = slide->prev;
- sc--;
- //stop on first bullet point always
- if(slide->stop > 0)
- slide->stop = 0;
- } else {
- // do nothing
- fade = false;
- }
+ // do nothing
+ fade = false;
}
- break;
-
+ }
+ } else if (evaluate_binding(next_slide_binding, c)) {
// show next slide or stop bit
- case KEY_DOWN:
- case KEY_RIGHT:
- case KEY_NPAGE:
- case '\n': // ENTER
- case ' ': // SPACE
- case 'j':
- case 'l':
- if(stop && line) {
- // show current slide again
- // but stop one stop bit later (or at end of slide)
- slide->stop++;
- fade = false;
+ if(stop && line) {
+ // show current slide again
+ // but stop one stop bit later (or at end of slide)
+ slide->stop++;
+ fade = false;
+ } else {
+ if(slide->next) {
+ // show next slide
+ slide = slide->next;
+ sc++;
} else {
- if(slide->next) {
- // show next slide
- slide = slide->next;
- sc++;
- } else {
- // do nothing
- fade = false;
- }
+ // do nothing
+ fade = false;
}
- break;
-
+ }
+ } else if (isdigit(c) && c != '0') {
// show slide n
- case '9':
- case '8':
- case '7':
- case '6':
- case '5':
- case '4':
- case '3':
- case '2':
- case '1':
- i = get_slide_number(c);
- if(i > 0 && i <= deck->slides) {
- while(sc != i) {
- // search forward
- if(sc < i) {
- if(slide->next) {
- slide = slide->next;
- sc++;
- }
- // search backward
- } else {
- if(slide->prev) {
- slide = slide->prev;
- sc--;
- }
+ i = get_slide_number(c);
+ if(i > 0 && i <= deck->slides) {
+ while(sc != i) {
+ // search forward
+ if(sc < i) {
+ if(slide->next) {
+ slide = slide->next;
+ sc++;
+ }
+ // search backward
+ } else {
+ if(slide->prev) {
+ slide = slide->prev;
+ sc--;
}
}
- } else {
- // disable fading if slide n doesn't exist
- fade = false;
}
- break;
-
+ }
+ } else if (evaluate_binding(first_slide_binding, c)) {
// show first slide
- case 'g':
- case KEY_HOME:
- slide = deck->slide;
- sc = 1;
- break;
-
+ slide = deck->slide;
+ sc = 1;
+ } else if (evaluate_binding(last_slide_binding, c)) {
// show last slide
- case 'G':
- case KEY_END:
- for(i = sc; i <= deck->slides; i++) {
- if(slide->next) {
- slide = slide->next;
- sc++;
- }
+ for(i = sc; i <= deck->slides; i++) {
+ if(slide->next) {
+ slide = slide->next;
+ sc++;
}
- break;
-
+ }
+ } else if (evaluate_binding(reload_binding, c)) {
// reload
- case 'r':
- if(noreload == 0) {
- // reload slide N
- reload = sc;
- slide = NULL;
- } else {
- // disable fading if reload is not possible
- fade = false;
- }
- break;
-
- // quit
- case 'q':
- // do not fade out on exit
- fade = false;
- // do not reload
- reload = 0;
+ if(noreload == 0) {
+ // reload slide N
+ reload = sc;
slide = NULL;
- break;
-
- default:
- // disable fading on undefined key press
+ } else {
+ // disable fading if reload is not possible
fade = false;
- break;
+ }
+ } else if (evaluate_binding(quit_binding, c)) {
+ // quit
+ // do not fade out on exit
+ fade = false;
+ // do not reload
+ reload = 0;
+ slide = NULL;
+ } else {
+ // disable fading on undefined key press
+ fade = false;
}
// fade out
}
}
-void add_line(WINDOW *window, int y, int x, line_t *line, int max_cols, int colors) {
+void add_line(WINDOW *window, int y, int x, line_t *line, int max_cols, int colors, int nocodebg) {
int i; // increment
int offset = 0; // text offset
// fill rest off line with spaces if we are in a code block
if(CHECK_BIT(line->bits, IS_CODE) && colors) {
- if(colors)
+ if(colors && !nocodebg)
wattron(window, COLOR_PAIR(CP_BLACK));
for(i = getcurx(window) - x; i < max_cols; i++)
wprintw(window, "%s", " ");
"%s", prompt);
if(!CHECK_BIT(line->bits, IS_CODE))
- inline_display(window, &line->text->value[offset], colors);
+ inline_display(window, &line->text->value[offset], colors, nocodebg);
// IS_UNORDERED_LIST_2
} else if(CHECK_BIT(line->bits, IS_UNORDERED_LIST_2)) {
"%s", prompt);
if(!CHECK_BIT(line->bits, IS_CODE))
- inline_display(window, &line->text->value[offset], colors);
+ inline_display(window, &line->text->value[offset], colors, nocodebg);
// IS_UNORDERED_LIST_1
} else if(CHECK_BIT(line->bits, IS_UNORDERED_LIST_1)) {
"%s", prompt);
if(!CHECK_BIT(line->bits, IS_CODE))
- inline_display(window, &line->text->value[offset], colors);
+ inline_display(window, &line->text->value[offset], colors, nocodebg);
}
// IS_CODE
}
// reverse color for code blocks
- if(colors)
+ if(colors && !nocodebg)
wattron(window, COLOR_PAIR(CP_BLACK));
// print whole lines
offset = next_word(line->text, offset);
}
- inline_display(window, &line->text->value[offset], colors);
+ inline_display(window, &line->text->value[offset], colors, nocodebg);
} else {
// IS_CENTER
// no line-wide markdown
} else {
- inline_display(window, &line->text->value[offset], colors);
+ inline_display(window, &line->text->value[offset], colors, nocodebg);
}
}
}
wattroff(window, A_UNDERLINE);
}
-void inline_display(WINDOW *window, const wchar_t *c, const int colors) {
+void inline_display(WINDOW *window, const wchar_t *c, const int colors, int nocodebg) {
const static wchar_t *special = L"\\*_`!["; // list of interpreted chars
const wchar_t *i = c; // iterator
const wchar_t *start_link_name, *start_url;
break;
// enable inline code
case L'`':
- if(colors)
+ if(colors && !nocodebg)
wattron(window, COLOR_PAIR(CP_BLACK));
break;
// do nothing for backslashes
int get_slide_number(char init) {
int retval = init - '0';
- char c;
+ int c;
// block for tenths of a second when using getch, ERR if no input
halfdelay(GOTO_SLIDE_DELAY);
while((c = getch()) != ERR) {
cbreak(); // go back to cbreak
return retval;
}
+
+bool evaluate_binding(const int bindings[], char c) {
+ int binding;
+ int ind = 0;
+ while((binding = bindings[ind]) != 0) {
+ if (c == binding) return true;
+ ind++;
+ }
+ return false;
+}
+