2007-07-27 07:13:27 -04:00
|
|
|
/** Text mode drawing functions
|
|
|
|
* @file */
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "elinks.h"
|
|
|
|
|
|
|
|
#include "bfu/dialog.h"
|
|
|
|
#include "cache/cache.h"
|
|
|
|
#include "document/document.h"
|
|
|
|
#include "document/html/frames.h"
|
|
|
|
#include "document/options.h"
|
|
|
|
#include "document/refresh.h"
|
|
|
|
#include "document/renderer.h"
|
|
|
|
#include "document/view.h"
|
|
|
|
#include "dialogs/status.h" /* print_screen_status() */
|
|
|
|
#include "intl/charsets.h"
|
|
|
|
#include "intl/gettext/libintl.h"
|
|
|
|
#include "protocol/uri.h"
|
|
|
|
#include "session/location.h"
|
|
|
|
#include "session/session.h"
|
|
|
|
#include "terminal/draw.h"
|
|
|
|
#include "terminal/tab.h"
|
|
|
|
#include "terminal/terminal.h"
|
|
|
|
#include "util/error.h"
|
|
|
|
#include "util/lists.h"
|
|
|
|
#include "util/memory.h"
|
|
|
|
#include "util/string.h"
|
|
|
|
#include "viewer/text/draw.h"
|
|
|
|
#include "viewer/text/form.h"
|
|
|
|
#include "viewer/text/link.h"
|
|
|
|
#include "viewer/text/search.h"
|
|
|
|
#include "viewer/text/view.h" /* current_frame() */
|
|
|
|
#include "viewer/text/vs.h"
|
|
|
|
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
check_document_fragment(struct session *ses, struct document_view *doc_view)
|
|
|
|
{
|
|
|
|
struct document *document = doc_view->document;
|
|
|
|
struct uri *uri = doc_view->vs->uri;
|
2006-12-05 13:04:53 -05:00
|
|
|
int vy;
|
2019-04-21 06:27:40 -04:00
|
|
|
struct string fragment;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
2006-12-05 13:04:53 -05:00
|
|
|
assert(uri->fragmentlen);
|
|
|
|
|
|
|
|
if (!init_string(&fragment)) return -2;
|
|
|
|
if (!add_uri_to_string(&fragment, uri, URI_FRAGMENT)) {
|
|
|
|
done_string(&fragment);
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
decode_uri_string(&fragment);
|
|
|
|
assert(fragment.length);
|
|
|
|
assert(*fragment.source);
|
|
|
|
|
|
|
|
/* Omit the leading '#' when calling find_tag. */
|
|
|
|
vy = find_tag(document, fragment.source + 1, fragment.length - 1);
|
2005-09-15 09:58:31 -04:00
|
|
|
if (vy == -1) {
|
2006-12-09 22:11:04 -05:00
|
|
|
struct cache_entry *cached = document->cached;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
2006-12-09 22:11:04 -05:00
|
|
|
assert(cached);
|
2008-04-20 16:25:10 -04:00
|
|
|
if (cached->incomplete || cached->cache_id != document->cache_id) {
|
2006-12-05 13:04:53 -05:00
|
|
|
done_string(&fragment);
|
2005-09-15 09:58:31 -04:00
|
|
|
return -2;
|
2006-12-05 13:04:53 -05:00
|
|
|
}
|
2005-09-15 09:58:31 -04:00
|
|
|
|
2007-08-28 12:41:18 -04:00
|
|
|
if (get_opt_bool("document.browse.links.missing_fragment",
|
2007-08-30 17:11:51 -04:00
|
|
|
ses)) {
|
2005-09-15 09:58:31 -04:00
|
|
|
info_box(ses->tab->term, MSGBOX_FREE_TEXT,
|
|
|
|
N_("Missing fragment"), ALIGN_CENTER,
|
|
|
|
msg_text(ses->tab->term, N_("The requested fragment "
|
2006-12-05 13:04:53 -05:00
|
|
|
"\"%s\" doesn't exist."),
|
|
|
|
fragment.source));
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int_bounds(&vy, 0, document->height - 1);
|
|
|
|
}
|
|
|
|
|
2006-12-05 13:04:53 -05:00
|
|
|
done_string(&fragment);
|
|
|
|
|
2005-09-15 09:58:31 -04:00
|
|
|
return vy;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
draw_frame_lines(struct terminal *term, struct frameset_desc *frameset_desc,
|
|
|
|
int xp, int yp, struct color_pair *colors)
|
|
|
|
{
|
|
|
|
int y, j;
|
|
|
|
|
|
|
|
assert(term && frameset_desc && frameset_desc->frame_desc);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
y = yp - 1;
|
|
|
|
for (j = 0; j < frameset_desc->box.height; j++) {
|
|
|
|
int x, i;
|
|
|
|
int height = frameset_desc->frame_desc[j * frameset_desc->box.width].height;
|
|
|
|
|
|
|
|
x = xp - 1;
|
|
|
|
for (i = 0; i < frameset_desc->box.width; i++) {
|
|
|
|
int width = frameset_desc->frame_desc[i].width;
|
|
|
|
|
|
|
|
if (i) {
|
2018-09-09 13:14:56 -04:00
|
|
|
struct el_box box;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
set_box(&box, x, y + 1, 1, height);
|
|
|
|
draw_box(term, &box, BORDER_SVLINE, SCREEN_ATTR_FRAME, colors);
|
|
|
|
|
|
|
|
if (j == frameset_desc->box.height - 1)
|
|
|
|
draw_border_cross(term, x, y + height + 1,
|
|
|
|
BORDER_X_UP, colors);
|
|
|
|
} else if (j) {
|
|
|
|
if (x >= 0)
|
|
|
|
draw_border_cross(term, x, y,
|
|
|
|
BORDER_X_RIGHT, colors);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (j) {
|
2018-09-09 13:14:56 -04:00
|
|
|
struct el_box box;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
set_box(&box, x + 1, y, width, 1);
|
|
|
|
draw_box(term, &box, BORDER_SHLINE, SCREEN_ATTR_FRAME, colors);
|
|
|
|
|
|
|
|
if (i == frameset_desc->box.width - 1
|
|
|
|
&& x + width + 1 < term->width)
|
|
|
|
draw_border_cross(term, x + width + 1, y,
|
|
|
|
BORDER_X_LEFT, colors);
|
|
|
|
} else if (i) {
|
|
|
|
draw_border_cross(term, x, y, BORDER_X_DOWN, colors);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i && j)
|
|
|
|
draw_border_char(term, x, y, BORDER_SCROSS, colors);
|
|
|
|
|
|
|
|
x += width + 1;
|
|
|
|
}
|
|
|
|
y += height + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
y = yp - 1;
|
|
|
|
for (j = 0; j < frameset_desc->box.height; j++) {
|
|
|
|
int x, i;
|
|
|
|
int pj = j * frameset_desc->box.width;
|
|
|
|
int height = frameset_desc->frame_desc[pj].height;
|
|
|
|
|
|
|
|
x = xp - 1;
|
|
|
|
for (i = 0; i < frameset_desc->box.width; i++) {
|
|
|
|
int width = frameset_desc->frame_desc[i].width;
|
|
|
|
int p = pj + i;
|
|
|
|
|
|
|
|
if (frameset_desc->frame_desc[p].subframe) {
|
|
|
|
draw_frame_lines(term, frameset_desc->frame_desc[p].subframe,
|
|
|
|
x + 1, y + 1, colors);
|
|
|
|
}
|
|
|
|
x += width + 1;
|
|
|
|
}
|
|
|
|
y += height + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
draw_view_status(struct session *ses, struct document_view *doc_view, int active)
|
|
|
|
{
|
|
|
|
struct terminal *term = ses->tab->term;
|
|
|
|
|
|
|
|
draw_forms(term, doc_view);
|
|
|
|
if (active) {
|
|
|
|
draw_searched(term, doc_view);
|
|
|
|
draw_current_link(ses, doc_view);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-27 07:13:27 -04:00
|
|
|
/** Checks if there is a link under the cursor so it can become the current
|
2005-09-15 09:58:31 -04:00
|
|
|
* highlighted link. */
|
|
|
|
static void
|
|
|
|
check_link_under_cursor(struct session *ses, struct document_view *doc_view)
|
|
|
|
{
|
|
|
|
int x = ses->tab->x;
|
|
|
|
int y = ses->tab->y;
|
2018-09-09 13:14:56 -04:00
|
|
|
struct el_box *box = &doc_view->box;
|
2005-09-15 09:58:31 -04:00
|
|
|
struct link *link;
|
|
|
|
|
|
|
|
link = get_link_at_coordinates(doc_view, x - box->x, y - box->y);
|
|
|
|
if (link && link != get_current_link(doc_view)) {
|
|
|
|
doc_view->vs->current_link = link - doc_view->document->links;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-27 07:13:27 -04:00
|
|
|
/** Puts the formatted document on the given terminal's screen.
|
|
|
|
* @a active indicates whether the document is focused -- i.e.,
|
2005-09-15 09:58:31 -04:00
|
|
|
* whether it is displayed in the selected frame or document. */
|
|
|
|
static void
|
|
|
|
draw_doc(struct session *ses, struct document_view *doc_view, int active)
|
|
|
|
{
|
|
|
|
struct color_pair color;
|
|
|
|
struct view_state *vs;
|
|
|
|
struct terminal *term;
|
2018-09-09 13:14:56 -04:00
|
|
|
struct el_box *box;
|
2005-09-15 09:58:31 -04:00
|
|
|
int vx, vy;
|
|
|
|
int y;
|
|
|
|
|
|
|
|
assert(ses && ses->tab && ses->tab->term && doc_view);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
box = &doc_view->box;
|
|
|
|
term = ses->tab->term;
|
|
|
|
|
|
|
|
/* The code in this function assumes that both width and height are
|
|
|
|
* bigger than 1 so we have to bail out here. */
|
|
|
|
if (box->width < 2 || box->height < 2) return;
|
|
|
|
|
|
|
|
if (active) {
|
|
|
|
/* When redrawing the document after things like link menu we
|
|
|
|
* have to reset the cursor routing state. */
|
|
|
|
if (ses->navigate_mode == NAVIGATE_CURSOR_ROUTING) {
|
|
|
|
set_cursor(term, ses->tab->x, ses->tab->y, 0);
|
|
|
|
} else {
|
|
|
|
set_cursor(term, box->x + box->width - 1, box->y + box->height - 1, 1);
|
|
|
|
set_window_ptr(ses->tab, box->x, box->y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-30 17:11:51 -04:00
|
|
|
color.foreground = get_opt_color("document.colors.text", ses);
|
2005-09-15 09:58:31 -04:00
|
|
|
color.background = doc_view->document->height
|
2007-10-12 10:50:47 -04:00
|
|
|
? doc_view->document->color.background
|
2007-08-30 17:11:51 -04:00
|
|
|
: get_opt_color("document.colors.background", ses);
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
vs = doc_view->vs;
|
|
|
|
if (!vs) {
|
|
|
|
draw_box(term, box, ' ', 0, &color);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (document_has_frames(doc_view->document)) {
|
|
|
|
draw_box(term, box, ' ', 0, &color);
|
|
|
|
draw_frame_lines(term, doc_view->document->frame_desc, box->x, box->y, &color);
|
|
|
|
if (vs->current_link == -1)
|
|
|
|
vs->current_link = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ses->navigate_mode == NAVIGATE_LINKWISE) {
|
|
|
|
check_vs(doc_view);
|
|
|
|
} else {
|
|
|
|
check_link_under_cursor(ses, doc_view);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!vs->did_fragment) {
|
|
|
|
vy = check_document_fragment(ses, doc_view);
|
|
|
|
|
|
|
|
if (vy != -2) vs->did_fragment = 1;
|
|
|
|
if (vy >= 0) {
|
|
|
|
doc_view->vs->y = vy;
|
|
|
|
set_link(doc_view);
|
|
|
|
}
|
2020-04-28 13:13:36 -04:00
|
|
|
if (vy == -1) {
|
|
|
|
struct location *loc = cur_loc(ses);
|
|
|
|
|
|
|
|
if (loc) {
|
|
|
|
struct uri *cur_uri = loc->vs.uri;
|
|
|
|
|
|
|
|
if (list_has_prev(ses->history.history, loc)) {
|
|
|
|
struct uri *prev_uri = loc->prev->vs.uri;
|
|
|
|
|
|
|
|
if (compare_uri(cur_uri, prev_uri, URI_BASE)) {
|
|
|
|
doc_view->vs->y = doc_view->prev_y;
|
|
|
|
set_link(doc_view);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
vx = vs->x;
|
|
|
|
vy = vs->y;
|
|
|
|
if (doc_view->last_x != -1
|
|
|
|
&& doc_view->last_x == vx
|
|
|
|
&& doc_view->last_y == vy
|
|
|
|
&& !has_search_word(doc_view)) {
|
|
|
|
clear_link(term, doc_view);
|
|
|
|
draw_view_status(ses, doc_view, active);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doc_view->last_x = vx;
|
|
|
|
doc_view->last_y = vy;
|
|
|
|
draw_box(term, box, ' ', 0, &color);
|
|
|
|
if (!doc_view->document->height) return;
|
|
|
|
|
|
|
|
while (vs->y >= doc_view->document->height) vs->y -= box->height;
|
|
|
|
int_lower_bound(&vs->y, 0);
|
|
|
|
if (vy != vs->y) {
|
|
|
|
vy = vs->y;
|
|
|
|
if (ses->navigate_mode == NAVIGATE_LINKWISE)
|
|
|
|
check_vs(doc_view);
|
|
|
|
}
|
|
|
|
for (y = int_max(vy, 0);
|
|
|
|
y < int_min(doc_view->document->height, box->height + vy);
|
|
|
|
y++) {
|
|
|
|
int st = int_max(vx, 0);
|
|
|
|
int en = int_min(doc_view->document->data[y].length,
|
|
|
|
box->width + vx);
|
2017-06-13 16:39:53 -04:00
|
|
|
|
2019-11-21 15:01:49 -05:00
|
|
|
if (en - st <= 0) continue;
|
|
|
|
draw_line(term, box->x + st - vx, box->y + y - vy, en - st,
|
|
|
|
&doc_view->document->data[y].chars[st]);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
draw_view_status(ses, doc_view, active);
|
|
|
|
if (has_search_word(doc_view))
|
|
|
|
doc_view->last_x = doc_view->last_y = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
draw_frames(struct session *ses)
|
|
|
|
{
|
|
|
|
struct document_view *doc_view, *current_doc_view;
|
|
|
|
int *l;
|
2005-12-13 10:59:10 -05:00
|
|
|
int n, d;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
assert(ses && ses->doc_view && ses->doc_view->document);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
if (!document_has_frames(ses->doc_view->document)) return;
|
|
|
|
|
|
|
|
n = 0;
|
|
|
|
foreach (doc_view, ses->scrn_frames) {
|
|
|
|
doc_view->last_x = doc_view->last_y = -1;
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
l = &cur_loc(ses)->vs.current_link;
|
|
|
|
*l = int_max(*l, 0) % int_max(n, 1);
|
|
|
|
|
|
|
|
current_doc_view = current_frame(ses);
|
|
|
|
d = 0;
|
2005-12-13 10:59:10 -05:00
|
|
|
while (1) {
|
|
|
|
int more = 0;
|
|
|
|
|
2005-09-15 09:58:31 -04:00
|
|
|
foreach (doc_view, ses->scrn_frames) {
|
|
|
|
if (doc_view->depth == d)
|
|
|
|
draw_doc(ses, doc_view, doc_view == current_doc_view);
|
|
|
|
else if (doc_view->depth > d)
|
|
|
|
more = 1;
|
|
|
|
}
|
2005-12-13 10:59:10 -05:00
|
|
|
|
|
|
|
if (!more) break;
|
2005-09-15 09:58:31 -04:00
|
|
|
d++;
|
2005-12-13 10:59:10 -05:00
|
|
|
};
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
2007-07-27 07:13:27 -04:00
|
|
|
/** @todo @a rerender is ridiciously wound-up. */
|
2005-09-15 09:58:31 -04:00
|
|
|
void
|
|
|
|
draw_formatted(struct session *ses, int rerender)
|
|
|
|
{
|
|
|
|
assert(ses && ses->tab);
|
|
|
|
if_assert_failed return;
|
|
|
|
|
|
|
|
if (rerender) {
|
|
|
|
rerender--; /* Mind this when analyzing @rerender. */
|
|
|
|
if (!(rerender & 2) && session_is_loading(ses))
|
|
|
|
rerender |= 2;
|
|
|
|
render_document_frames(ses, rerender);
|
|
|
|
|
|
|
|
/* Rerendering kills the document refreshing so restart it. */
|
2007-09-06 14:37:17 -04:00
|
|
|
start_document_refreshes(ses);
|
2005-09-15 09:58:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ses->tab != get_current_tab(ses->tab->term))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!ses->doc_view || !ses->doc_view->document) {
|
|
|
|
/*INTERNAL("document not formatted");*/
|
2018-09-09 13:14:56 -04:00
|
|
|
struct el_box box;
|
2005-09-15 09:58:31 -04:00
|
|
|
|
|
|
|
set_box(&box, 0, 1,
|
|
|
|
ses->tab->term->width,
|
|
|
|
ses->tab->term->height - 2);
|
|
|
|
draw_box(ses->tab->term, &box, ' ', 0, NULL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ses->doc_view->vs && have_location(ses))
|
|
|
|
ses->doc_view->vs = &cur_loc(ses)->vs;
|
|
|
|
ses->doc_view->last_x = ses->doc_view->last_y = -1;
|
|
|
|
|
|
|
|
refresh_view(ses, ses->doc_view, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
refresh_view(struct session *ses, struct document_view *doc_view, int frames)
|
|
|
|
{
|
2008-07-22 05:13:27 -04:00
|
|
|
/* If refresh_view() is being called because the value of a
|
|
|
|
* form field has changed, @ses might not be in the current
|
|
|
|
* tab: consider SELECT pop-ups behind which -remote loads
|
|
|
|
* another tab, or setTimeout in ECMAScript. */
|
|
|
|
if (ses->tab == get_current_tab(ses->tab->term)) {
|
|
|
|
draw_doc(ses, doc_view, 1);
|
|
|
|
if (frames) draw_frames(ses);
|
|
|
|
}
|
2005-09-15 09:58:31 -04:00
|
|
|
print_screen_status(ses);
|
|
|
|
}
|