1
0
mirror of https://github.com/rkd77/elinks.git synced 2024-12-04 14:46:47 -05:00

[xml] local directory listings are functional

This commit is contained in:
Witold Filipczyk 2021-06-22 17:54:10 +02:00
parent d5b27592a1
commit 16c6926990
3 changed files with 335 additions and 7 deletions

View File

@ -162,8 +162,10 @@ put_chrs(struct html_context *html_context, char *start, int len)
break;
}
if (isspace(start[len - 1]) && !html_is_preformatted())
if (isspace(start[len - 1]) && !html_is_preformatted()) {
html_context->putsp = HTML_SPACE_SUPPRESS;
if (len == 1) return;
}
html_context->was_br = 0;
html_context->put_chars_f(html_context, start, len);

View File

@ -206,7 +206,7 @@ element:
}
if (noupdate) put_chrs(html_context, base_pos, html - base_pos);
ln_break(html_context, 1);
//ln_break(html_context, 1);
/* Restore the part in case the html_context was trashed in the last
* iteration so that when destroying the stack in the caller we still
* get the right part pointer. */
@ -228,10 +228,9 @@ dump_dom_structure(struct source_renderer *renderer, xmlpp::Node *node, int dept
std::string tag_name = node->get_name();
struct element_info2 *ei = get_tag_value(tag_name.c_str(), tag_name.size());
/* Print this node's entry */
if (ei) {
ei->open(renderer, node, NULL, NULL, NULL, NULL);
}
if (!ei) return true;
start_element_2(ei, renderer, node);
/* Get the node's first child */
auto children = node->get_children();
@ -252,10 +251,12 @@ dump_dom_structure(struct source_renderer *renderer, xmlpp::Node *node, int dept
}
}
}
end_element_2(ei, renderer, node);
/*
if (ei && ei->close) {
ei->close(renderer, node, NULL, NULL, NULL, NULL);
}
*/
return true;
}

View File

@ -4428,3 +4428,328 @@ get_tag_value(unsigned char *name, int namelen)
return ei;
}
void
start_element_2(struct element_info2 *ei, struct source_renderer *renderer, void *nod)
{
xmlpp::Element *node = nod;
struct html_context *html_context = renderer->html_context;
#define ELEMENT_RENDER_PROLOGUE \
ln_break(html_context, ei->linebreak); \
std::string id_value = node->get_attribute_value("id"); \
if (id_value != "") { \
a = memacpy(id_value.c_str(), id_value.size()); \
if (a) { \
html_context->special_f(html_context, SP_TAG, a); \
mem_free(a); \
} \
}
char *a;
struct par_attrib old_format;
int restore_format;
#ifdef CONFIG_CSS
struct css_selector *selector = NULL;
#endif
/* If the currently top element on the stack cannot contain other
* elements, pop it. */
if (html_top->type == ELEMENT_WEAK) {
pop_html_element(html_context);
}
/* If an element handler has temporarily disabled rendering by setting
* the invisible flag, skip all handling for this element.
*
* As a special case, invisible can be set to 2 or greater to indicate
* that processing has been temporarily disabled except when a <script>
* tag is encountered. This special case is necessary for 2 situations:
* 1. A <script> tag is contained by a block with the CSS "display"
* property set to "none"; or 2. one <script> tag is nested inside
* another, but the outer script is not processed, in which case ELinks
* should still run any inner script blocks. */
if (html_top->invisible
&& (ei->open != tags_html_script || html_top->invisible < 2)) {
ELEMENT_RENDER_PROLOGUE
return; // html;
}
/* Store formatting information for the currently top element on the
* stack before processing the new element. */
restore_format = html_is_preformatted();
old_format = par_elformat;
/* Check for <meta refresh="..."> and <meta> cache-control directives
* inside <body> (see bug 700). */
if (ei->open == tags_html_meta && html_context->was_body) {
/// tags_html_handle_body_meta(html_context, name - 1, eof);
html_context->was_body = 0;
}
/* If this is a style tag, parse it. */
#ifdef CONFIG_CSS
if (ei->open == tags_html_style && html_context->options->css_enable) {
std::string media_value = node->get_attribute_value("media");
char *media = NULL;
if (media_value != "") {
media = memacpy(media_value.c_str(), media_value.size());
}
int support = supports_html_media_attr(media);
mem_free_if(media);
/// if (support)
/// css_parse_stylesheet(&html_context->css_styles,
/// html_context->base_href,
/// html, eof);
}
#endif
/* If this element is inline, non-nestable, and not <li>, and the next
* stack item down is the same element, try to pop both elements.
*
* The effect is to close automatically any <a> or <tt> tag that
* encloses the current tag if it is of the same element.
*
* Otherwise, if this element is non-inline or <li> and is
* non-nestable, search down through the stack until 1. we find a
* non-killable element; 2. the element being processed is not <li> and
* we find a block; or 3. the element being processed is <li> and we
* find another <li>. Then if the element found is the same as the
* current element, try to pop all elements down to and including the
* found element.
*
* The effect is to close automatically any <hN>, <p>, or <li> tag that
* encloses the current tag if it is of the same element, ignoring any
* intervening inline elements.
*/
if (ei->type == ET_NON_NESTABLE || ei->type == ET_LI) {
struct html_element *e;
if (ei->type == ET_NON_NESTABLE) {
foreach (e, html_context->stack) {
if (e->type < ELEMENT_KILLABLE) break;
if (is_block_element(e) || is_inline_element(ei)) break;
}
} else { /* This is an <li>. */
foreach (e, html_context->stack) {
if (is_block_element(e) && is_inline_element(ei)) break;
if (e->type < ELEMENT_KILLABLE) break;
if (!c_strlcasecmp(e->name, e->namelen, ei->name, ei->namelen)) break;
}
}
if (!c_strlcasecmp(e->name, e->namelen, ei->name, ei->namelen)) {
while (e->prev != (void *) &html_context->stack)
kill_html_stack_item(html_context, e->prev);
if (e->type > ELEMENT_IMMORTAL)
kill_html_stack_item(html_context, e);
}
}
/* Create an item on the stack for the element being processed. */
html_stack_dup(html_context, ELEMENT_KILLABLE);
html_top->name = ei->name;
html_top->namelen = ei->namelen;
html_top->options = NULL;//attr;
html_top->linebreak = ei->linebreak;
/* If the element has an onClick handler for scripts, make it
* clickable. */
#ifdef CONFIG_ECMASCRIPT
std::string onclick_value = node->get_attribute_value("onclick");
if (onclick_value != "") {
/* XXX: Put something better to elformat.link. --pasky */
mem_free_set(&elformat.link, stracpy("javascript:void(0);"));
mem_free_set(&elformat.target, stracpy(html_context->base_target));
elformat.style.color.foreground = elformat.color.clink;
html_top->pseudo_class = ELEMENT_LINK;
mem_free_set(&elformat.title, stracpy("onClick placeholder"));
/* Er. I know. Well, double html_focusable()s shouldn't
* really hurt. */
tags_html_focusable(renderer, node);
}
#endif
/* Apply CSS styles. */
#ifdef CONFIG_CSS
if (html_top->options && html_context->options->css_enable) {
/* XXX: We should apply CSS otherwise as well, but that'll need
* some deeper changes in order to have options filled etc.
* Probably just applying CSS from more places, since we
* usually have type != ET_NESTABLE when we either (1)
* rescan on your own from somewhere else (2) html_stack_dup()
* in our own way. --pasky */
std::string id_value = node->get_attribute_value("id");
char *id = NULL;
if (id_value != "") {
id = memacpy(id_value.c_str(), id_value.size());
}
mem_free_set(&html_top->attr.id, id);
std::string class_value = node->get_attribute_value("class");
char *class_ = NULL;
if (class_value != "") {
class_ = memacpy(class_value.c_str(), class_value.size());
}
mem_free_set(&html_top->attr.class_, class_);
/* Call it now to gain some of the stuff which might affect
* formatting of some elements. */
/* FIXME: The caching of the CSS selector is broken, since t can
* lead to wrong styles being applied to following elements, so
* disabled for now. */
selector = get_css_selector_for_element(html_context, html_top,
&html_context->css_styles,
&html_context->stack);
if (selector) {
apply_css_selector_style(html_context, html_top, selector);
done_css_selector(selector);
}
}
#endif
/* 1. Put any linebreaks that the element calls for, and 2. register
* any fragment identifier. Step 1 cannot be done before applying CSS
* styles because the CSS "display" property may change the element's
* linebreak value.
*
* XXX: The above is wrong: ELEMENT_RENDER_PROLOGUE only looks at the
* linebreak value for the element_info structure, which CSS cannot
* modify. -- Miciah */
ELEMENT_RENDER_PROLOGUE
/* Call the element's open handler. */
if (ei->open) ei->open(renderer, node, NULL, NULL, NULL, NULL);
/* Apply CSS styles again. */
#ifdef CONFIG_CSS
if (selector && html_top->options) {
/* Call it now to override default colors of the elements. */
selector = get_css_selector_for_element(html_context, html_top,
&html_context->css_styles,
&html_context->stack);
if (selector) {
apply_css_selector_style(html_context, html_top, selector);
done_css_selector(selector);
}
}
#endif
/* If this element was not <br>, clear the was_br flag. */
if (ei->open != tags_html_br) html_context->was_br = 0;
/* If this element is not pairable, pop its stack item now. */
if (ei->type == ET_NON_PAIRABLE)
kill_html_stack_item(html_context, html_top);
/* If we are rendering preformatted text (see above), restore the
* formatting attributes that were in effect before processing this
* element. */
if (restore_format) par_elformat = old_format;
return;
#undef ELEMENT_RENDER_PROLOGUE
}
void
end_element_2(struct element_info2 *ei, struct source_renderer *renderer, void *nod)
{
xmlpp::Element *node = nod;
struct html_context *html_context = renderer->html_context;
struct html_element *e, *elt;
int lnb = 0;
int kill = 0;
html_context->was_br = 0;
/* If this was a non-pairable tag or an <li>; perform no further
* processing. */
if (ei->type == ET_NON_PAIRABLE /* || ei->type == ET_LI */)
return;
/* Call the element's close handler. */
if (ei->close) ei->close(renderer, node, NULL, NULL, NULL, NULL);
/* dump_html_stack(html_context); */
/* Search down through the stack until we find 1. a different element
* that cannot be killed or 2. the element that is currently being
* processed (NOT necessarily the same instance of that element).
*
* In the first case, we are done. In the second, if this is an inline
* element and we found a block element while searching, we kill the
* found element; else (either this is inline but no block was found or
* this is a block), output linebreaks for all of the elements down to
* and including the found element and then pop all of these elements.
*
* The effects of the procedure outlined above and implemented below
* are 1. to allow an inline element to close any elements that it
* contains iff the inline element does not contain any blocks and 2.
* to allow blocks to close any elements that they contain.
*
* Two situations in which this behaviour may not match expectations
* are demonstrated by the following HTML:
*
* <b>a<a href="file:///">b</b>c</a>d
* <a href="file:///">e<b>f<div>g</a>h</div></b>
*
* ELinks will render "a" as bold text; "b" as bold, hyperlinked text;
* both "c" and "d" as normal (non-bold, non-hyperlinked) text (note
* that "</b>" closed the link because "<b>" enclosed the "<a>" open
* tag); "e" as hyperlinked text; and "f", "g", and "h" all as bold,
* hyperlinked text (note that "</a>" did not close the link because
* "</a>" was enclosed by "<div>", a block tag, while the "<a>" open
* tag was outside of the "<div>" block).
*
* Note also that close handlers are not called, which might also lead
* to unexpected behaviour as illustrated by the following example:
*
* <b><q>I like fudge,</b> he said. <q>So do I,</q> she said.
*
* ELinks will render "I like fudge," with bold typeface but will only
* place an opening double-quotation mark before the text and no closing
* mark. "he said." will be rendered normally. "So do I," will be
* rendered using single-quotation marks (as for a quotation within a
* quotation). "she said." will be rendered normally. */
foreach (e, html_context->stack) {
if (is_block_element(e) && is_inline_element(ei)) kill = 1;
if (c_strlcasecmp(e->name, e->namelen, ei->name, ei->namelen)) {
if (e->type < ELEMENT_KILLABLE)
break;
else
continue;
}
if (kill) {
kill_html_stack_item(html_context, e);
break;
}
for (elt = e;
elt != (void *) &html_context->stack;
elt = elt->prev)
if (elt->linebreak > lnb)
lnb = elt->linebreak;
/* This hack forces a line break after a list end. It is needed
* when ending a list with the last <li> having no text the
* line_breax is 2 so the ending list's linebreak will be
* ignored when calling ln_break(). */
if (html_context->was_li)
html_context->line_breax = 0;
ln_break(html_context, lnb);
while (e->prev != (void *) &html_context->stack)
kill_html_stack_item(html_context, e->prev);
kill_html_stack_item(html_context, e);
break;
}
/* dump_html_stack(html_context); */
return;
}