blob: ea00ba563b8d2acab5071a8504e13e2d269e2465 [file] [log] [blame]
/* Output stream that produces HTML output.
Copyright (C) 2006-2009, 2019-2020 Free Software Foundation, Inc.
Written by Bruno Haible <bruno@clisp.org>, 2006.
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
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 <https://www.gnu.org/licenses/>. */
#include <config.h>
/* Specification. */
#include "html-ostream.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "gl_xlist.h"
#include "gl_array_list.h"
#include "minmax.h"
#include "unistr.h"
#include "xalloc.h"
struct html_ostream : struct ostream
{
fields:
/* The destination stream. */
ostream_t destination;
/* The current hyperlink ref. */
char *hyperlink_ref;
/* The stack of active CSS classes. */
gl_list_t /* <char *> */ class_stack;
/* Current and last size of the active portion of this stack. Always
size(class_stack) == max(curr_class_stack_size,last_class_stack_size). */
size_t curr_class_stack_size;
size_t last_class_stack_size;
/* Last few bytes that could not yet be converted. */
#define BUFSIZE 6
char buf[BUFSIZE];
size_t buflen;
};
/* Emit an HTML attribute value.
quote is either '"' or '\''. */
static void
write_attribute_value (html_ostream_t stream, const char *value, char quote)
{
/* Need to escape the '<', '>', '&', quote characters. */
ostream_t destination = stream->destination;
const char *p = value;
for (;;)
{
const char *q = p;
while (*q != '\0' && *q != '<' && *q != '>' && *q != '&' && *q != quote)
q++;
if (p < q)
ostream_write_mem (destination, p, q - p);
if (*q == '\0')
break;
switch (*q)
{
case '<':
ostream_write_str (destination, "&lt;");
break;
case '>':
ostream_write_str (destination, "&gt;");
break;
case '&':
ostream_write_str (destination, "&amp;");
break;
case '"':
ostream_write_str (destination, "&quot;");
break;
case '\'':
ostream_write_str (destination, "&apos;");
break;
default:
abort ();
}
p = q + 1;
}
}
/* Implementation of ostream_t methods. */
static void
verify_invariants (html_ostream_t stream)
{
/* Verify the invariant regarding size(class_stack). */
if (gl_list_size (stream->class_stack)
!= MAX (stream->curr_class_stack_size, stream->last_class_stack_size))
abort ();
}
/* Removes the excess elements of class_stack.
Needs to be called after max(curr_class_stack_size,last_class_stack_size)
may have been reduced. */
static void
shrink_class_stack (html_ostream_t stream)
{
size_t keep =
MAX (stream->curr_class_stack_size, stream->last_class_stack_size);
size_t i = gl_list_size (stream->class_stack);
while (i > keep)
{
i--;
free ((char *) gl_list_get_at (stream->class_stack, i));
gl_list_remove_at (stream->class_stack, i);
}
}
/* Emits <span> or </span> tags, to follow the increase / decrease of the
class_stack from last_class_stack_size to curr_class_stack_size.
When done, sets last_class_stack_size to curr_class_stack_size. */
static void
emit_pending_spans (html_ostream_t stream, bool shrink_stack)
{
if (stream->curr_class_stack_size > stream->last_class_stack_size)
{
size_t i;
for (i = stream->last_class_stack_size; i < stream->curr_class_stack_size; i++)
{
char *classname = (char *) gl_list_get_at (stream->class_stack, i);
ostream_write_str (stream->destination, "<span class=\"");
ostream_write_str (stream->destination, classname);
ostream_write_str (stream->destination, "\">");
}
stream->last_class_stack_size = stream->curr_class_stack_size;
}
else if (stream->curr_class_stack_size < stream->last_class_stack_size)
{
size_t i;
for (i = stream->last_class_stack_size; i > stream->curr_class_stack_size; i--)
ostream_write_str (stream->destination, "</span>");
stream->last_class_stack_size = stream->curr_class_stack_size;
if (shrink_stack)
shrink_class_stack (stream);
}
/* Here last_class_stack_size == curr_class_stack_size. */
if (shrink_stack)
verify_invariants (stream);
}
static void
html_ostream::write_mem (html_ostream_t stream, const void *data, size_t len)
{
if (len > 0)
{
#define BUFFERSIZE 2048
char inbuffer[BUFFERSIZE];
size_t inbufcount;
inbufcount = stream->buflen;
if (inbufcount > 0)
memcpy (inbuffer, stream->buf, inbufcount);
for (;;)
{
/* At this point, inbuffer[0..inbufcount-1] is filled. */
{
/* Combine the previous rest with a chunk of new input. */
size_t n =
(len <= BUFFERSIZE - inbufcount ? len : BUFFERSIZE - inbufcount);
if (n > 0)
{
memcpy (inbuffer + inbufcount, data, n);
data = (const char *) data + n;
inbufcount += n;
len -= n;
}
}
{
/* Handle complete UTF-8 characters. */
const char *inptr = inbuffer;
size_t insize = inbufcount;
while (insize > 0)
{
unsigned char c0;
ucs4_t uc;
int nbytes;
c0 = ((const unsigned char *) inptr)[0];
if (insize < (c0 < 0xc0 ? 1 : c0 < 0xe0 ? 2 : c0 < 0xf0 ? 3 :
c0 < 0xf8 ? 4 : c0 < 0xfc ? 5 : 6))
break;
nbytes = u8_mbtouc (&uc, (const unsigned char *) inptr, insize);
if (uc == '\n')
{
verify_invariants (stream);
/* Emit </span> tags to follow the decrease of the class stack
from last_class_stack_size to 0. Then emit the newline.
Then prepare for emitting <span> tags to go back from 0
to curr_class_stack_size. */
size_t prev_class_stack_size = stream->curr_class_stack_size;
stream->curr_class_stack_size = 0;
emit_pending_spans (stream, false);
stream->curr_class_stack_size = prev_class_stack_size;
ostream_write_str (stream->destination, "<br/>");
shrink_class_stack (stream);
verify_invariants (stream);
}
else
{
emit_pending_spans (stream, true);
switch (uc)
{
case '"':
ostream_write_str (stream->destination, "&quot;");
break;
case '&':
ostream_write_str (stream->destination, "&amp;");
break;
case '<':
ostream_write_str (stream->destination, "&lt;");
break;
case '>':
/* Needed to avoid "]]>" in the output. */
ostream_write_str (stream->destination, "&gt;");
break;
case ' ':
/* Needed because HTML viewers merge adjacent spaces
and drop spaces adjacent to <br> and similar. */
ostream_write_str (stream->destination, "&nbsp;");
break;
default:
if (uc >= 0x20 && uc < 0x7F)
{
/* Output ASCII characters as such. */
char bytes[1];
bytes[0] = uc;
ostream_write_mem (stream->destination, bytes, 1);
}
else
{
/* Output non-ASCII characters in #&nnn;
notation. */
char bytes[32];
sprintf (bytes, "&#%d;", (int) uc);
ostream_write_str (stream->destination, bytes);
}
break;
}
}
inptr += nbytes;
insize -= nbytes;
}
/* Put back the unconverted part. */
if (insize > BUFSIZE)
abort ();
if (len == 0)
{
if (insize > 0)
memcpy (stream->buf, inptr, insize);
stream->buflen = insize;
break;
}
if (insize > 0)
memmove (inbuffer, inptr, insize);
inbufcount = insize;
}
}
#undef BUFFERSIZE
}
}
static void
html_ostream::flush (html_ostream_t stream, ostream_flush_scope_t scope)
{
verify_invariants (stream);
/* stream->buf[] contains only a few bytes that don't correspond to a
character. Can't flush it. */
/* Close the open <span> tags, and prepare for reopening the same <span>
tags. */
size_t prev_class_stack_size = stream->curr_class_stack_size;
stream->curr_class_stack_size = 0;
emit_pending_spans (stream, false);
stream->curr_class_stack_size = prev_class_stack_size;
shrink_class_stack (stream);
verify_invariants (stream);
if (scope != FLUSH_THIS_STREAM)
ostream_flush (stream->destination, scope);
}
static void
html_ostream::free (html_ostream_t stream)
{
stream->curr_class_stack_size = 0;
emit_pending_spans (stream, true);
if (stream->hyperlink_ref != NULL)
{
/* Close the current <a> element. */
ostream_write_str (stream->destination, "</a>");
free (stream->hyperlink_ref);
}
verify_invariants (stream);
gl_list_free (stream->class_stack);
free (stream);
}
/* Implementation of html_ostream_t methods. */
static void
html_ostream::begin_span (html_ostream_t stream, const char *classname)
{
verify_invariants (stream);
if (stream->last_class_stack_size > stream->curr_class_stack_size
&& strcmp ((char *) gl_list_get_at (stream->class_stack,
stream->curr_class_stack_size),
classname) != 0)
emit_pending_spans (stream, true);
/* Now either
last_class_stack_size <= curr_class_stack_size
- in this case we have to append the given CLASSNAME -
or
last_class_stack_size > curr_class_stack_size
&& class_stack[curr_class_stack_size] == CLASSNAME
- in this case we only need to increment curr_class_stack_size. */
if (stream->last_class_stack_size <= stream->curr_class_stack_size)
gl_list_add_at (stream->class_stack, stream->curr_class_stack_size,
xstrdup (classname));
stream->curr_class_stack_size++;
verify_invariants (stream);
}
static void
html_ostream::end_span (html_ostream_t stream, const char *classname)
{
verify_invariants (stream);
if (stream->curr_class_stack_size > 0)
{
char *innermost_active_span =
(char *) gl_list_get_at (stream->class_stack,
stream->curr_class_stack_size - 1);
if (strcmp (innermost_active_span, classname) == 0)
{
stream->curr_class_stack_size--;
shrink_class_stack (stream);
verify_invariants (stream);
return;
}
}
/* Improperly nested begin_span/end_span calls. */
abort ();
}
static const char *
html_ostream::get_hyperlink_ref (html_ostream_t stream)
{
return stream->hyperlink_ref;
}
static void
html_ostream::set_hyperlink_ref (html_ostream_t stream, const char *ref)
{
char *ref_copy = (ref != NULL ? xstrdup (ref) : NULL);
verify_invariants (stream);
if (stream->hyperlink_ref != NULL)
{
/* Close the open <span> tags, and prepare for reopening the same <span>
tags. */
size_t prev_class_stack_size = stream->curr_class_stack_size;
stream->curr_class_stack_size = 0;
emit_pending_spans (stream, false);
stream->curr_class_stack_size = prev_class_stack_size;
/* Close the current <a> element. */
ostream_write_str (stream->destination, "</a>");
shrink_class_stack (stream);
free (stream->hyperlink_ref);
}
stream->hyperlink_ref = ref_copy;
if (stream->hyperlink_ref != NULL)
{
/* Close the open <span> tags, and prepare for reopening the same <span>
tags. */
size_t prev_class_stack_size = stream->curr_class_stack_size;
stream->curr_class_stack_size = 0;
emit_pending_spans (stream, false);
stream->curr_class_stack_size = prev_class_stack_size;
/* Open an <a> element. */
ostream_write_str (stream->destination, "<a href=\"");
write_attribute_value (stream, stream->hyperlink_ref, '"');
ostream_write_str (stream->destination, "\">");
shrink_class_stack (stream);
}
verify_invariants (stream);
}
static void
html_ostream::flush_to_current_style (html_ostream_t stream)
{
verify_invariants (stream);
/* stream->buf[] contains only a few bytes that don't correspond to a
character. Can't flush it. */
/* Open all requested <span> tags. */
emit_pending_spans (stream, true);
verify_invariants (stream);
}
/* Constructor. */
html_ostream_t
html_ostream_create (ostream_t destination)
{
html_ostream_t stream = XMALLOC (struct html_ostream_representation);
stream->base.vtable = &html_ostream_vtable;
stream->destination = destination;
stream->hyperlink_ref = NULL;
stream->class_stack =
gl_list_create_empty (GL_ARRAY_LIST, NULL, NULL, NULL, true);
stream->curr_class_stack_size = 0;
stream->last_class_stack_size = 0;
stream->buflen = 0;
return stream;
}
/* Accessors. */
static ostream_t
html_ostream::get_destination (html_ostream_t stream)
{
return stream->destination;
}
/* Instanceof test. */
bool
is_instance_of_html_ostream (ostream_t stream)
{
return IS_INSTANCE (stream, ostream, html_ostream);
}