mirror of
https://github.com/rkd77/elinks.git
synced 2025-01-03 14:57:44 -05:00
349 lines
7.9 KiB
C
349 lines
7.9 KiB
C
/* Information about current document and current link */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <time.h>
|
|
|
|
#include "elinks.h"
|
|
|
|
#include "bfu/dialog.h"
|
|
#include "cache/cache.h"
|
|
#include "dialogs/document.h"
|
|
#include "document/document.h"
|
|
#include "document/view.h"
|
|
#include "globhist/globhist.h"
|
|
#include "intl/libintl.h"
|
|
#include "protocol/header.h"
|
|
#include "protocol/uri.h"
|
|
#include "session/location.h"
|
|
#include "session/session.h"
|
|
#include "terminal/terminal.h"
|
|
#include "terminal/window.h"
|
|
#include "util/conv.h"
|
|
#include "util/memory.h"
|
|
#include "util/string.h"
|
|
#include "viewer/text/link.h"
|
|
#include "viewer/text/view.h"
|
|
|
|
void
|
|
nowhere_box(struct terminal *term, char *title)
|
|
{
|
|
assert(term);
|
|
if_assert_failed return;
|
|
|
|
if (!title || !*title)
|
|
title = N_("Info");
|
|
|
|
info_box(term, 0, title, ALIGN_CENTER,
|
|
N_("You are nowhere!"));
|
|
}
|
|
|
|
static void
|
|
add_link_info_to_string(struct string *msg, struct session *ses)
|
|
{
|
|
struct document_view *doc_view = current_frame(ses);
|
|
struct terminal *term = ses->tab->term;
|
|
char *a;
|
|
struct link *link;
|
|
|
|
if (!doc_view) return;
|
|
|
|
add_char_to_string(msg, '\n');
|
|
|
|
a = get_current_link_info(ses, doc_view);
|
|
if (a) {
|
|
add_format_to_string(msg, "\n%s: %s",
|
|
_("Link", term), a);
|
|
mem_free(a);
|
|
}
|
|
|
|
a = get_current_link_title(doc_view);
|
|
if (a) {
|
|
add_format_to_string(msg, "\n%s: %s",
|
|
_("Link title", term), a);
|
|
mem_free(a);
|
|
}
|
|
|
|
link = get_current_link_in_view(doc_view);
|
|
if (link) {
|
|
struct string img;
|
|
#ifdef CONFIG_GLOBHIST
|
|
struct global_history_item *historyitem;
|
|
#endif
|
|
|
|
if (link->where_img && init_string(&img)) {
|
|
add_string_uri_to_string(&img, link->where_img,
|
|
URI_PUBLIC);
|
|
decode_uri_string_for_display(&img);
|
|
|
|
add_format_to_string(msg, "\n%s: %s",
|
|
_("Link image", term),
|
|
img.source);
|
|
done_string(&img);
|
|
}
|
|
|
|
#ifdef CONFIG_GLOBHIST
|
|
historyitem = get_global_history_item(link->where);
|
|
if (historyitem) {
|
|
char *last_visit;
|
|
|
|
last_visit = ctime(&historyitem->last_visit);
|
|
|
|
if (last_visit)
|
|
add_format_to_string(msg,
|
|
"\n%s: %.24s",
|
|
_("Link last visit time",
|
|
term),
|
|
last_visit);
|
|
|
|
if (*historyitem->title)
|
|
add_format_to_string(msg, "\n%s: %s",
|
|
_("Link title (from history)",
|
|
term),
|
|
historyitem->title);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/* Link info message box. */
|
|
void
|
|
link_info_dialog(struct session *ses)
|
|
{
|
|
struct terminal *term = ses->tab->term;
|
|
struct location *location = cur_loc(ses);
|
|
struct string msg;
|
|
|
|
if (!location) {
|
|
nowhere_box(term, NULL);
|
|
return;
|
|
}
|
|
|
|
if (!init_string(&msg)) return;
|
|
|
|
add_link_info_to_string(&msg, ses);
|
|
|
|
info_box(term, MSGBOX_FREE_TEXT | MSGBOX_SCROLLABLE,
|
|
N_("Info"), ALIGN_LEFT, msg.source);
|
|
}
|
|
|
|
|
|
/* Location info. message box. */
|
|
void
|
|
document_info_dialog(struct session *ses)
|
|
{
|
|
struct terminal *term = ses->tab->term;
|
|
struct location *location = cur_loc(ses);
|
|
struct document_view *doc_view;
|
|
struct string msg;
|
|
|
|
if (!location) {
|
|
nowhere_box(term, NULL);
|
|
return;
|
|
}
|
|
|
|
doc_view = current_frame(ses);
|
|
|
|
if (!init_string(&msg)) return;
|
|
|
|
add_to_string(&msg, _("URL", term));
|
|
add_to_string(&msg, ": ");
|
|
|
|
/* Add the uri with password and post info stripped */
|
|
add_uri_to_string(&msg, location->vs.uri, URI_PUBLIC);
|
|
|
|
add_char_to_string(&msg, '\n');
|
|
|
|
if (doc_view && doc_view->document->title) {
|
|
add_format_to_string(&msg, "%s: %s", _("Title", term),
|
|
doc_view->document->title);
|
|
}
|
|
|
|
add_char_to_string(&msg, '\n');
|
|
|
|
if (doc_view && doc_view->document->cached) {
|
|
struct cache_entry *cached = doc_view->document->cached;
|
|
char *a;
|
|
|
|
add_format_to_string(&msg, "\n%s: %" OFF_PRINT_FORMAT,
|
|
_("Size", term),
|
|
(off_print_T) cached->length);
|
|
|
|
if (cached->incomplete) {
|
|
add_format_to_string(&msg, " (%s)", _("incomplete", term));
|
|
}
|
|
|
|
if (doc_view) {
|
|
add_format_to_string(&msg, "\n%s: %s", _("Codepage", term),
|
|
get_cp_name(doc_view->document->cp));
|
|
|
|
if (doc_view->document->cp_status == CP_STATUS_ASSUMED) {
|
|
add_format_to_string(&msg, " (%s)", _("assumed", term));
|
|
} else if (doc_view->document->cp_status == CP_STATUS_IGNORED) {
|
|
add_format_to_string(&msg, " (%s)",
|
|
_("ignoring server setting", term));
|
|
}
|
|
}
|
|
|
|
a = parse_header(cached->head, "Server", NULL);
|
|
if (a) {
|
|
add_format_to_string(&msg, "\n%s: %s",
|
|
_("Server", term), a);
|
|
mem_free(a);
|
|
}
|
|
|
|
if (cached->ssl_info) {
|
|
add_format_to_string(&msg, "\n%s: %s",
|
|
_("SSL Cipher", term),
|
|
cached->ssl_info);
|
|
}
|
|
if (cached->encoding_info) {
|
|
add_format_to_string(&msg, "\n%s: %s",
|
|
_("Encoding", term),
|
|
cached->encoding_info);
|
|
}
|
|
|
|
a = parse_header(cached->head, "Date", NULL);
|
|
if (a) {
|
|
add_format_to_string(&msg, "\n%s: %s",
|
|
_("Date", term), a);
|
|
mem_free(a);
|
|
}
|
|
|
|
if (cached->last_modified) {
|
|
add_format_to_string(&msg, "\n%s: %s",
|
|
_("Last modified", term),
|
|
cached->last_modified);
|
|
}
|
|
|
|
if (!cached->incomplete) {
|
|
add_format_to_string(&msg, "\n%s: ",
|
|
_("Time since loading", term));
|
|
add_duration_to_string(&msg,
|
|
time(NULL) - cached->seconds);
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef CONFIG_GLOBHIST
|
|
{
|
|
char *last_visit = NULL;
|
|
struct global_history_item *historyitem;
|
|
|
|
add_format_to_string(&msg, "\n%s: ",
|
|
_("Last visit time", term));
|
|
|
|
historyitem = get_global_history_item(struri(location->vs.uri));
|
|
|
|
if (historyitem) last_visit = ctime(&historyitem->last_visit);
|
|
|
|
/* GNU's documentation says that ctime() can return NULL.
|
|
* The Open Group Base Specifications Issue 6 implies
|
|
* otherwise, but is ambiguous. Let's be safe. -- Miciah
|
|
*/
|
|
if (last_visit) {
|
|
/* The string returned by ctime() includes a newline,
|
|
* and we don't want that, so we use add_bytes_to_str.
|
|
* The string always has exactly 25 characters, so add
|
|
* 24 bytes: The length of the string, minus one for
|
|
* the newline. -- Miciah
|
|
*/
|
|
add_bytes_to_string(&msg, last_visit, 24);
|
|
} else {
|
|
add_to_string(&msg, _("Unknown", term));
|
|
}
|
|
}
|
|
#endif
|
|
|
|
add_link_info_to_string(&msg, ses);
|
|
|
|
info_box(term, MSGBOX_FREE_TEXT | MSGBOX_SCROLLABLE,
|
|
N_("Info"), ALIGN_LEFT, msg.source);
|
|
}
|
|
|
|
void
|
|
cached_header_dialog(struct session *ses, struct cache_entry *cached)
|
|
{
|
|
int msgbox_flags = 0;
|
|
char *title = N_("Header info");
|
|
char *headers = NULL;
|
|
int i = 0, j = 0;
|
|
|
|
if (!cached || !cached->head || !*cached->head)
|
|
goto display_headers;
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
/* If |cached->head| starts with a newline, it has been
|
|
* internally generated, usually to give ELinks-generated
|
|
* documents (e.g., file:// directory listings) a MIME type
|
|
* of text/html. */
|
|
if (*cached->head == '\r')
|
|
title = N_("Internal header info");
|
|
#endif
|
|
|
|
headers = (char *)mem_alloc(strlen(cached->head) + 1);
|
|
if (!headers) return;
|
|
|
|
/* Sanitize headers string. */
|
|
/* XXX: Do we need to check length and limit
|
|
* it to something reasonable? */
|
|
|
|
while (cached->head[i]) {
|
|
/* Check for control chars. */
|
|
if ((unsigned char)cached->head[i] < ' '
|
|
&& cached->head[i] != '\n') {
|
|
/* Ignore '\r' but replace
|
|
* other control chars with
|
|
* a visible char. */
|
|
if (cached->head[i] != '\r') {
|
|
headers[j] = '*';
|
|
j++;
|
|
}
|
|
} else {
|
|
headers[j] = cached->head[i];
|
|
j++;
|
|
}
|
|
i++;
|
|
}
|
|
|
|
/* Ensure null termination. */
|
|
headers[j] = '\0';
|
|
|
|
/* Remove any trailing newlines. */
|
|
while (j && headers[--j] == '\n')
|
|
headers[j] = '\0';
|
|
|
|
if (!*headers)
|
|
mem_free_set(&headers, NULL);
|
|
|
|
display_headers:
|
|
|
|
if (!headers) {
|
|
headers = N_("No header info.");
|
|
} else {
|
|
msgbox_flags = MSGBOX_FREE_TEXT | MSGBOX_SCROLLABLE;
|
|
}
|
|
|
|
/* Headers info message box. */
|
|
info_box(ses->tab->term, msgbox_flags,
|
|
title, ALIGN_LEFT, headers);
|
|
}
|
|
|
|
/* Headers info. message box. */
|
|
void
|
|
protocol_header_dialog(struct session *ses)
|
|
{
|
|
if (!have_location(ses)) {
|
|
nowhere_box(ses->tab->term, N_("Header info"));
|
|
return;
|
|
}
|
|
|
|
if (ses->doc_view && ses->doc_view->document)
|
|
cached_header_dialog(ses, ses->doc_view->document->cached);
|
|
}
|