1
0
mirror of https://github.com/rkd77/elinks.git synced 2024-11-04 08:17:17 -05:00
elinks/src/document/html/tables.c
2005-12-01 11:39:39 +01:00

1329 lines
34 KiB
C

/* HTML tables renderer */
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include <stdlib.h>
#include <string.h>
#include "elinks.h"
#include "document/html/parser/parse.h"
#include "document/html/parser/table.h"
#include "document/html/parser.h"
#include "document/html/renderer.h"
#include "document/html/tables.h"
#include "document/options.h"
#include "terminal/draw.h"
#include "util/color.h"
#include "util/conv.h"
#include "util/error.h"
#include "util/memory.h"
#include "util/string.h"
/* Unsafe macros */
#include "document/html/internal.h"
struct table_frames {
unsigned int top:1;
unsigned int bottom:1;
unsigned int left:1;
unsigned int right:1;
};
static void
get_table_frames(struct table *table, struct table_frames *result)
{
assert(table && result);
if (table->border) {
result->top = !!(table->frame & TABLE_FRAME_ABOVE);
result->bottom = !!(table->frame & TABLE_FRAME_BELOW);
result->left = !!(table->frame & TABLE_FRAME_LHS);
result->right = !!(table->frame & TABLE_FRAME_RHS);
} else {
memset(result, 0, sizeof(*result));
}
}
/* Distance of the table from the left margin. */
static int
get_table_indent(struct html_context *html_context, struct table *table)
{
int width = par_format.width - table->real_width;
int indent;
switch (table->align) {
case ALIGN_CENTER:
indent = (width + par_format.leftmargin - par_format.rightmargin) / 2;
break;
case ALIGN_RIGHT:
indent = width - par_format.rightmargin;
break;
case ALIGN_LEFT:
case ALIGN_JUSTIFY:
default:
indent = par_format.leftmargin;
}
/* Don't use int_bounds(&x, 0, width) here,
* width may be < 0. --Zas */
if (indent > width) indent = width;
if (indent < 0) indent = 0;
return indent;
}
static inline struct part *
format_cell(struct html_context *html_context, struct table *table,
struct table_cell *cell, struct document *document,
int x, int y, int width)
{
if (document) {
x += table->part->box.x;
y += table->part->box.y;
}
return format_html_part(html_context, cell->start, cell->end,
cell->align, table->cellpadding, width,
document, x, y, NULL, cell->link_num);
}
static inline void
get_cell_width(struct html_context *html_context,
unsigned char *start, unsigned char *end,
int cellpadding, int width,
int a, int *min, int *max,
int link_num, int *new_link_num)
{
struct part *part;
if (min) *min = -1;
if (max) *max = -1;
if (new_link_num) *new_link_num = link_num;
part = format_html_part(html_context, start, end, ALIGN_LEFT,
cellpadding, width, NULL,
!!a, !!a, NULL, link_num);
if (!part) return;
if (min) *min = part->box.width;
if (max) *max = part->max_width;
if (new_link_num) *new_link_num = part->link_num;
if (min && max) {
assertm(*min <= *max, "get_cell_width: %d > %d", *min, *max);
}
mem_free(part);
}
static void
get_cell_widths(struct html_context *html_context, struct table *table)
{
int link_num = table->part->link_num;
if (!html_context->options->table_order) {
int col, row;
for (row = 0; row < table->rows; row++)
for (col = 0; col < table->cols; col++) {
struct table_cell *cell = CELL(table, col, row);
if (!cell->start) continue;
cell->link_num = link_num;
get_cell_width(html_context, cell->start,
cell->end, table->cellpadding,
0, 0, &cell->min_width,
&cell->max_width,
link_num, &link_num);
}
} else {
int col, row;
for (col = 0; col < table->cols; col++)
for (row = 0; row < table->rows; row++) {
struct table_cell *cell = CELL(table, col, row);
if (!cell->start) continue;
cell->link_num = link_num;
get_cell_width(html_context, cell->start,
cell->end, table->cellpadding,
0, 0, &cell->min_width,
&cell->max_width,
link_num, &link_num);
}
}
table->link_num = link_num;
}
static inline void
distribute_values(int *values, int count, int wanted, int *limits)
{
int i;
int sum = 0, d, r, t;
for (i = 0; i < count; i++) sum += values[i];
if (sum >= wanted) return;
again:
t = wanted - sum;
d = t / count;
r = t % count;
wanted = 0;
if (limits) {
for (i = 0; i < count; i++) {
int delta;
values[i] += d + (i < r);
delta = values[i] - limits[i];
if (delta > 0) {
wanted += delta;
values[i] = limits[i];
}
}
} else {
for (i = 0; i < count; i++) {
values[i] += d + (i < r);
}
}
if (wanted) {
assertm(limits, "bug in distribute_values()");
limits = NULL;
sum = 0;
goto again;
}
}
/* Returns: -1 none, 0, space, 1 line, 2 double */
static inline int
get_vline_width(struct table *table, int col)
{
int width = 0;
if (!col) return -1;
if (table->rules == TABLE_RULE_COLS || table->rules == TABLE_RULE_ALL)
width = table->cellspacing;
else if (table->rules == TABLE_RULE_GROUPS)
width = (col < table->columns_count && table->columns[col].group);
if (!width && table->cellpadding) width = -1;
return width;
}
static int
get_hline_width(struct table *table, int row)
{
if (!row) return -1;
if (table->rules == TABLE_RULE_ROWS || table->rules == TABLE_RULE_ALL) {
if (table->cellspacing || table->vcellpadding)
return table->cellspacing;
return -1;
} else if (table->rules == TABLE_RULE_GROUPS) {
int col;
for (col = 0; col < table->cols; col++)
if (CELL(table, col, row)->is_group) {
if (table->cellspacing || table->vcellpadding)
return table->cellspacing;
return -1;
}
}
return table->vcellpadding ? 0 : -1;
}
#define has_vline_width(table, col) (get_vline_width(table, col) >= 0)
#define has_hline_width(table, row) (get_hline_width(table, row) >= 0)
static int
get_column_widths(struct table *table)
{
int colspan;
if (!table->cols) return -1; /* prevents calloc(0, ...) calls */
if (!table->min_cols_widths) {
table->min_cols_widths = mem_calloc(table->cols, sizeof(*table->min_cols_widths));
if (!table->min_cols_widths) return -1;
}
if (!table->max_cols_widths) {
table->max_cols_widths = mem_calloc(table->cols, sizeof(*table->max_cols_widths));
if (!table->max_cols_widths) {
mem_free_set(&table->min_cols_widths, NULL);
return -1;
}
}
if (!table->cols_widths) {
table->cols_widths = mem_calloc(table->cols, sizeof(*table->cols_widths));
if (!table->cols_widths) {
mem_free_set(&table->min_cols_widths, NULL);
mem_free_set(&table->max_cols_widths, NULL);
return -1;
}
}
colspan = 1;
do {
int col, row;
int new_colspan = INT_MAX;
for (col = 0; col < table->cols; col++) for (row = 0; row < table->rows; row++) {
struct table_cell *cell = CELL(table, col, row);
if (cell->is_spanned || !cell->is_used) continue;
assertm(cell->colspan + col <= table->cols, "colspan out of table");
if_assert_failed return -1;
if (cell->colspan == colspan) {
int k, p = 0;
for (k = 1; k < colspan; k++)
p += has_vline_width(table, col + k);
distribute_values(&table->min_cols_widths[col],
colspan,
cell->min_width - p,
&table->max_cols_widths[col]);
distribute_values(&table->max_cols_widths[col],
colspan,
cell->max_width - p,
NULL);
for (k = 0; k < colspan; k++) {
int tmp = col + k;
int_lower_bound(&table->max_cols_widths[tmp],
table->min_cols_widths[tmp]);
}
} else if (cell->colspan > colspan
&& cell->colspan < new_colspan) {
new_colspan = cell->colspan;
}
}
colspan = new_colspan;
} while (colspan != INT_MAX);
return 0;
}
static void
get_table_width(struct table *table)
{
struct table_frames table_frames;
int min = 0;
int max = 0;
int col;
for (col = 0; col < table->cols; col++) {
int vl = has_vline_width(table, col);
min += vl + table->min_cols_widths[col];
max += vl + table->max_cols_widths[col];
if (table->cols_x[col] > table->max_cols_widths[col])
max += table->cols_x[col];
}
get_table_frames(table, &table_frames);
table->min_width = min + table_frames.left + table_frames.right;
table->max_width = max + table_frames.left + table_frames.right;
assertm(min <= max, "min(%d) > max(%d)", min, max);
/* XXX: Recovery path? --pasky */
}
/* Initialize @width and @max_width arrays depending on the @stretch_method */
static inline int
apply_stretch_method(struct table *table, int widths[], int max_widths[],
int stretch_method, int max_cols_width)
{
int col, total_width = 0;
for (col = 0; col < table->cols; col++) {
switch (stretch_method) {
case 0:
if (table->cols_widths[col] >= table->cols_x[col])
break;
widths[col] = 1;
max_widths[col] = int_min(table->cols_x[col],
table->max_cols_widths[col])
- table->cols_widths[col];
if (max_widths[col] <= 0) widths[col] = 0;
break;
case 1:
if (table->cols_x[col] > WIDTH_RELATIVE)
break;
widths[col] = WIDTH_RELATIVE - table->cols_x[col];
max_widths[col] = table->max_cols_widths[col]
- table->cols_widths[col];
if (max_widths[col] <= 0) widths[col] = 0;
break;
case 2:
if (table->cols_x[col] != WIDTH_AUTO)
break;
/* Fall-through */
case 3:
if (table->cols_widths[col] >= table->max_cols_widths[col])
break;
max_widths[col] = table->max_cols_widths[col]
- table->cols_widths[col];
if (max_cols_width) {
widths[col] = 5 + table->max_cols_widths[col] * 10 / max_cols_width;
} else {
widths[col] = 1;
}
break;
case 4:
if (table->cols_x[col] < 0)
break;
widths[col] = 1;
max_widths[col] = table->cols_x[col]
- table->cols_widths[col];
if (max_widths[col] <= 0) widths[col] = 0;
break;
case 5:
if (table->cols_x[col] >= 0)
break;
if (table->cols_x[col] <= WIDTH_RELATIVE) {
widths[col] = WIDTH_RELATIVE - table->cols_x[col];
} else {
widths[col] = 1;
}
max_widths[col] = INT_MAX;
break;
case 6:
widths[col] = 1;
max_widths[col] = INT_MAX;
break;
default:
return -1;
}
total_width += widths[col];
}
return total_width;
}
/* Stretches the table columns by distributed the @spare_width among them.
* Returns how much of @spare_width was actually distributed. */
static inline int
stretch_columns(struct table *table, int widths[], int max_widths[],
int spare_width, int total_width)
{
int total_spare_width = spare_width;
while (spare_width) {
int stretch_width = 0;
int stretch_col = -1;
int col;
for (col = 0; col < table->cols; col++) {
int col_spare_width;
if (!widths[col])
continue;
col_spare_width = total_spare_width * widths[col] / total_width;
int_bounds(&col_spare_width, 1, max_widths[col]);
if (col_spare_width > stretch_width) {
stretch_width = col_spare_width;
stretch_col = col;
}
}
/* Got stretch column? */
if (stretch_col == -1)
break;
/* Mark the column as visited */
widths[stretch_col] = 0;
if (stretch_width > spare_width)
stretch_width = spare_width;
assertm(stretch_width >= 0, "shrinking cell");
table->cols_widths[stretch_col] += stretch_width;
spare_width -= stretch_width;
}
return total_spare_width - spare_width;
}
/* This function distributes space evenly between @table columns so that it
* stretches to @width. */
static void
distribute_widths(struct table *table, int width)
{
int col;
int spare_width = width - table->min_width;
int stretch_method = 0;
int *widths, *max_widths;
int max_cols_width = 0;
int cols_array_size;
if (!table->cols)
return;
assertm(spare_width >= 0, "too small width %d, required %d",
width, table->min_width);
for (col = 0; col < table->cols; col++)
int_lower_bound(&max_cols_width, table->max_cols_widths[col]);
cols_array_size = table->cols * sizeof(*table->cols_widths);
memcpy(table->cols_widths, table->min_cols_widths, cols_array_size);
table->real_width = width;
widths = fmem_alloc(cols_array_size);
if (!widths) return;
max_widths = fmem_alloc(cols_array_size);
if (!max_widths) goto free_widths;
while (spare_width) {
int stretched, total_width;
memset(widths, 0, cols_array_size);
memset(max_widths, 0, cols_array_size);
total_width = apply_stretch_method(table, widths, max_widths, stretch_method, max_cols_width);
assertm(total_width != -1, "Could not expand table");
if_assert_failed break;
if (!total_width) {
stretch_method++;
continue;
}
stretched = stretch_columns(table, widths, max_widths, spare_width, total_width);
if (!stretched)
stretch_method++;
else
spare_width -= stretched;
}
fmem_free(max_widths);
free_widths:
fmem_free(widths);
}
static int
get_table_cellpadding(struct html_context *html_context, struct table *table)
{
struct part *part = table->part;
int cpd_pass = 0, cpd_width = 0, cpd_last = table->cellpadding;
int margins = par_format.leftmargin + par_format.rightmargin;
again:
get_cell_widths(html_context, table);
if (get_column_widths(table)) return -1;
get_table_width(table);
if (!part->document && !part->box.x) {
if (!table->full_width)
int_upper_bound(&table->max_width, table->width);
int_lower_bound(&table->max_width, table->min_width);
int_lower_bound(&part->max_width, table->max_width + margins);
int_lower_bound(&part->box.width, table->min_width + margins);
return -1;
}
if (!cpd_pass && table->min_width > table->width && table->cellpadding) {
table->cellpadding = 0;
cpd_pass = 1;
cpd_width = table->min_width;
goto again;
}
if (cpd_pass == 1 && table->min_width > cpd_width) {
table->cellpadding = cpd_last;
cpd_pass = 2;
goto again;
}
return 0;
}
#ifdef HTML_TABLE_2ND_PASS /* This is by default ON! (<setup.h>) */
static void
check_table_widths(struct html_context *html_context, struct table *table)
{
int col, row;
int colspan;
int width, new_width;
int max, max_index = 0; /* go away, warning! */
int *widths = mem_calloc(table->cols, sizeof(*widths));
if (!widths) return;
for (row = 0; row < table->rows; row++) for (col = 0; col < table->cols; col++) {
struct table_cell *cell = CELL(table, col, row);
int k, p = 0;
if (!cell->start) continue;
for (k = 0; k < cell->colspan; k++) {
p += table->cols_widths[col + k] +
(k && has_vline_width(table, col + k));
}
get_cell_width(html_context, cell->start, cell->end,
table->cellpadding, p, 1, &cell->width,
NULL, cell->link_num, NULL);
int_upper_bound(&cell->width, p);
}
colspan = 1;
do {
int new_colspan = INT_MAX;
for (col = 0; col < table->cols; col++) for (row = 0; row < table->rows; row++) {
struct table_cell *cell = CELL(table, col, row);
if (!cell->start) continue;
assertm(cell->colspan + col <= table->cols, "colspan out of table");
if_assert_failed goto end;
if (cell->colspan == colspan) {
int k, p = 0;
for (k = 1; k < colspan; k++)
p += has_vline_width(table, col + k);
distribute_values(&widths[col],
colspan,
cell->width - p,
&table->max_cols_widths[col]);
} else if (cell->colspan > colspan
&& cell->colspan < new_colspan) {
new_colspan = cell->colspan;
}
}
colspan = new_colspan;
} while (colspan != INT_MAX);
width = new_width = 0;
for (col = 0; col < table->cols; col++) {
width += table->cols_widths[col];
new_width += widths[col];
}
if (new_width > width) {
/* INTERNAL("new width(%d) is larger than previous(%d)", new_width, width); */
goto end;
}
max = -1;
for (col = 0; col < table->cols; col++)
if (table->max_cols_widths[col] > max) {
max = table->max_cols_widths[col];
max_index = col;
}
if (max != -1) {
widths[max_index] += width - new_width;
if (widths[max_index] <= table->max_cols_widths[max_index]) {
mem_free(table->cols_widths);
table->cols_widths = widths;
return;
}
}
end:
mem_free(widths);
}
#endif
static void
check_table_height(struct table *table, struct table_frames *frames, int y)
{
#ifndef CONFIG_FASTMEM
/* XXX: Cannot we simply use the @yp value we just calculated
* in draw_table_cells()? --pasky */
int old_height = table->real_height + table->part->cy;
int our_height = frames->top + y + frames->bottom + table->caption_height;
int row;
/* XXX: We cannot use get_table_real_height() because we are
* looking one row ahead - which is completely arcane to me.
* It makes a difference only when a table uses ruler="groups"
* and has non-zero cellspacing or vcellpadding. --pasky */
for (row = 0; row < table->rows; row++) {
our_height += table->rows_heights[row] +
(row < table->rows - 1 &&
has_hline_width(table, row + 1));
}
assertm(old_height == our_height, "size not matching! %d vs %d",
old_height, our_height);
#endif
}
static int
get_table_caption_height(struct html_context *html_context, struct table *table)
{
unsigned char *start = table->caption.start;
unsigned char *end = table->caption.end;
struct part *part;
if (!start || !end) return 0;
while (start < end && isspace(*start))
start++;
while (start < end && isspace(end[-1]))
end--;
if (start >= end) return 0;
part = format_html_part(html_context, start, end, table->align,
0, table->real_width, NULL, 0, 0,
NULL, table->link_num);
if (!part) {
return 0;
} else {
int height = part->box.height;
mem_free(part);
return height;
}
}
static int
get_table_real_height(struct table *table)
{
struct table_frames table_frames;
int height;
int row;
get_table_frames(table, &table_frames);
height = table_frames.top + table_frames.bottom;
height += table->caption_height;
for (row = 0; row < table->rows; row++) {
height += table->rows_heights[row];
if (row && has_hline_width(table, row))
height++;
}
return height;
}
static void
get_table_heights(struct html_context *html_context, struct table *table)
{
int rowspan;
int col, row;
table->caption_height = get_table_caption_height(html_context, table);
for (row = 0; row < table->rows; row++) {
for (col = 0; col < table->cols; col++) {
struct table_cell *cell = CELL(table, col, row);
struct part *part;
int width = 0, sp;
if (!cell->is_used || cell->is_spanned) continue;
for (sp = 0; sp < cell->colspan; sp++) {
width += table->cols_widths[col + sp] +
(sp < cell->colspan - 1 &&
has_vline_width(table, col + sp + 1));
}
part = format_cell(html_context, table, cell, NULL,
2, 2, width);
if (!part) return;
cell->height = part->box.height;
/* DBG("%d, %d.", width, cell->height); */
mem_free(part);
}
}
rowspan = 1;
do {
int new_rowspan = INT_MAX;
for (row = 0; row < table->rows; row++) {
for (col = 0; col < table->cols; col++) {
struct table_cell *cell = CELL(table, col, row);
if (!cell->is_used || cell->is_spanned) continue;
if (cell->rowspan == rowspan) {
int k, p = 0;
for (k = 1; k < rowspan; k++)
p += has_hline_width(table, row + k);
distribute_values(&table->rows_heights[row],
rowspan,
cell->height - p,
NULL);
} else if (cell->rowspan > rowspan &&
cell->rowspan < new_rowspan) {
new_rowspan = cell->rowspan;
}
}
}
rowspan = new_rowspan;
} while (rowspan != INT_MAX);
table->real_height = get_table_real_height(table);
}
static void
draw_table_cell(struct table *table, int col, int row, int x, int y,
struct html_context *html_context)
{
struct table_cell *cell = CELL(table, col, row);
struct document *document = table->part->document;
struct part *part;
int width = 0;
int height = 0;
int s, tmpy = y;
struct html_element *state;
if (!cell->start) return;
for (s = 0; s < cell->colspan; s++) {
width += table->cols_widths[col + s] +
(s < cell->colspan - 1 &&
has_vline_width(table, col + s + 1));
}
for (s = 0; s < cell->rowspan; s++) {
height += table->rows_heights[row + s] +
(s < cell->rowspan - 1 &&
has_hline_width(table, row + s + 1));
}
state = init_html_parser_state(html_context, ELEMENT_DONT_KILL,
cell->align, 0, 0);
if (cell->is_header) format.style.attr |= AT_BOLD;
format.style.bg = cell->bgcolor;
par_format.bgcolor = cell->bgcolor;
if (cell->valign == VALIGN_MIDDLE)
tmpy += (height - cell->height) / 2;
else if (cell->valign == VALIGN_BOTTOM)
tmpy += (height - cell->height);
part = format_cell(html_context, table, cell, document, x, tmpy, width);
if (part) {
/* The cell content doesn't necessarily fill out the whole cell
* height so use the calculated @height because it is an upper
* bound. */
assert(height >= cell->height);
/* The line expansion draws the _remaining_ background color of
* both untouched lines and lines that doesn't stretch the
* whole cell width. */
expand_lines(html_context, table->part, x + width - 1, y, height, cell->bgcolor);
if (cell->fragment_id)
add_fragment_identifier(html_context, part,
cell->fragment_id);
}
done_html_parser_state(html_context, state);
if (part) mem_free(part);
}
static void
draw_table_cells(struct table *table, int x, int y,
struct html_context *html_context)
{
int col, row;
int xp;
color_T bgcolor = par_format.bgcolor;
struct table_frames table_frames;
get_table_frames(table, &table_frames);
if (table->fragment_id)
add_fragment_identifier(html_context, table->part,
table->fragment_id);
/* Expand using the background color of the ``parent context'' all the
* way down the start of the left edge of the table. */
expand_lines(html_context, table->part, x - 1, y, table->real_height, bgcolor);
xp = x + table_frames.left;
for (col = 0; col < table->cols; col++) {
int yp = y + table_frames.top;
for (row = 0; row < table->rows; row++) {
int row_height = table->rows_heights[row] +
(row < table->rows - 1 && has_hline_width(table, row + 1));
draw_table_cell(table, col, row, xp, yp, html_context);
yp += row_height;
}
if (col < table->cols - 1) {
xp += table->cols_widths[col] + has_vline_width(table, col + 1);
}
}
/* Finish the table drawing by aligning the right and bottom edge of
* the table */
x += table->real_width - 1;
expand_lines(html_context, table->part, x, y, table->real_height, table->bgcolor);
/* Tables are renderer column-wise which breaks forms where the
* form items appears in a column before the actual form tag is
* parsed. Consider the folloing example:
*
* +--------+--------+ Where cell 2 has a <form>-tag
* | cell 1 | cell 2 | and cell 3 has an <input>-tag.
* +--------+--------+
* | cell 3 | cell 4 | The table is rendered by drawing
* +--------+--------+ the cells in the order: 1, 3, 2, 4.
*
* That is the <input>-tag form-item is added before the <form>-tag,
* which means a ``dummy'' form to hold it is added too.
* Calling check_html_form_hierarchy() will join the the form-item
* to the correct form from cell 2. */
check_html_form_hierarchy(table->part);
/* Do a sanity check whether the height is correct */
check_table_height(table, &table_frames, y);
}
static inline int
get_frame_pos(int a, int a_size, int b, int b_size)
{
assert(a >= -1 || a < a_size + 2 || b >= 0 || b <= b_size);
if_assert_failed return 0;
return a + 1 + (a_size + 2) * b;
}
#define H_FRAME_POSITION(table, col, row) frame[0][get_frame_pos(col, (table)->cols, row, (table)->rows)]
#define V_FRAME_POSITION(table, col, row) frame[1][get_frame_pos(row, (table)->rows, col, (table)->cols)]
static inline void
draw_frame_point(struct table *table, signed char *frame[2], int x, int y,
int col, int row, struct html_context *html_context)
{
static enum border_char const border_chars[81] = {
BORDER_NONE, BORDER_SVLINE, BORDER_DVLINE,
BORDER_SHLINE, BORDER_SDLCORNER, BORDER_DSDLCORNER,
BORDER_DHLINE, BORDER_SDDLCORNER, BORDER_DDLCORNER,
BORDER_SHLINE, BORDER_SDRCORNER, BORDER_DSDRCORNER,
BORDER_SHLINE, BORDER_SUTEE, BORDER_DSUTEE,
BORDER_DHLINE, BORDER_SDDLCORNER, BORDER_DDLCORNER,
BORDER_DHLINE, BORDER_SDDRCORNER, BORDER_DDRCORNER,
BORDER_DHLINE, BORDER_SDDRCORNER, BORDER_DDRCORNER,
BORDER_DHLINE, BORDER_SDUTEE, BORDER_DUTEE,
BORDER_SVLINE, BORDER_SVLINE, BORDER_DVLINE,
BORDER_SULCORNER, BORDER_SRTEE, BORDER_DSDLCORNER,
BORDER_SDULCORNER, BORDER_SDRTEE, BORDER_DDLCORNER,
BORDER_SURCORNER, BORDER_SLTEE, BORDER_DSDRCORNER,
BORDER_SDTEE, BORDER_SCROSS, BORDER_DSUTEE,
BORDER_SDULCORNER, BORDER_SDRTEE, BORDER_DDLCORNER,
BORDER_SDURCORNER, BORDER_SDLTEE, BORDER_DDRCORNER,
BORDER_SDURCORNER, BORDER_SDLTEE, BORDER_DDRCORNER,
BORDER_SDDTEE, BORDER_SDCROSS, BORDER_DUTEE,
BORDER_DVLINE, BORDER_DVLINE, BORDER_DVLINE,
BORDER_DSULCORNER, BORDER_DSULCORNER, BORDER_DSRTEE,
BORDER_DULCORNER, BORDER_DULCORNER, BORDER_DRTEE,
BORDER_DSURCORNER, BORDER_DSURCORNER, BORDER_DSLTEE,
BORDER_DSDTEE, BORDER_DSDTEE, BORDER_DSCROSS,
BORDER_DULCORNER, BORDER_DULCORNER, BORDER_DRTEE,
BORDER_DURCORNER, BORDER_DURCORNER, BORDER_DLTEE,
BORDER_DURCORNER, BORDER_DURCORNER, BORDER_DLTEE,
BORDER_DDTEE, BORDER_DDTEE, BORDER_DCROSS,
};
/* Note: I have no clue wether any of these names are suitable but they
* should give an idea of what is going on. --jonas */
signed char left = H_FRAME_POSITION(table, col - 1, row);
signed char right = H_FRAME_POSITION(table, col, row);
signed char top = V_FRAME_POSITION(table, col, row - 1);
signed char bottom = V_FRAME_POSITION(table, col, row);
int pos;
if (left < 0 && right < 0 && top < 0 && bottom < 0) return;
pos = int_max(top, 0)
+ 3 * int_max(right, 0)
+ 9 * int_max(left, 0)
+ 27 * int_max(bottom, 0);
draw_frame_hchars(table->part, x, y, 1, border_chars[pos],
par_format.bgcolor, table->bordercolor, html_context);
}
static inline void
draw_frame_hline(struct table *table, signed char *frame[2], int x, int y,
int col, int row, struct html_context *html_context)
{
static unsigned char const hltable[] = { ' ', BORDER_SHLINE, BORDER_DHLINE };
int pos = H_FRAME_POSITION(table, col, row);
assertm(pos < 3, "Horizontal table position out of bound [%d]", pos);
if_assert_failed return;
if (pos < 0 || table->cols_widths[col] <= 0) return;
draw_frame_hchars(table->part, x, y, table->cols_widths[col], hltable[pos],
par_format.bgcolor, table->bordercolor, html_context);
}
static inline void
draw_frame_vline(struct table *table, signed char *frame[2], int x, int y,
int col, int row, struct html_context *html_context)
{
static unsigned char const vltable[] = { ' ', BORDER_SVLINE, BORDER_DVLINE };
int pos = V_FRAME_POSITION(table, col, row);
assertm(pos < 3, "Vertical table position out of bound [%d]", pos);
if_assert_failed return;
if (pos < 0 || table->rows_heights[row] <= 0) return;
draw_frame_vchars(table->part, x, y, table->rows_heights[row], vltable[pos],
par_format.bgcolor, table->bordercolor, html_context);
}
static inline int
table_row_has_group(struct table *table, int row)
{
int col;
for (col = 0; col < table->cols; col++)
if (CELL(table, col, row)->is_group)
return 1;
return 0;
}
static void
init_table_rules(struct table *table, signed char *frame[2])
{
int col, row;
for (row = 0; row < table->rows; row++) for (col = 0; col < table->cols; col++) {
int xsp, ysp;
struct table_cell *cell = CELL(table, col, row);
if (!cell->is_used || cell->is_spanned) continue;
xsp = cell->colspan ? cell->colspan : table->cols - col;
ysp = cell->rowspan ? cell->rowspan : table->rows - row;
if (table->rules != TABLE_RULE_COLS) {
memset(&H_FRAME_POSITION(table, col, row), table->cellspacing, xsp);
memset(&H_FRAME_POSITION(table, col, row + ysp), table->cellspacing, xsp);
}
if (table->rules != TABLE_RULE_ROWS) {
memset(&V_FRAME_POSITION(table, col, row), table->cellspacing, ysp);
memset(&V_FRAME_POSITION(table, col + xsp, row), table->cellspacing, ysp);
}
}
if (table->rules == TABLE_RULE_GROUPS) {
for (col = 1; col < table->cols; col++) {
if (table->cols_x[col])
continue;
memset(&V_FRAME_POSITION(table, col, 0), 0, table->rows);
}
for (row = 1; row < table->rows; row++) {
if (table_row_has_group(table, row))
continue;
memset(&H_FRAME_POSITION(table, 0, row), 0, table->cols);
}
}
}
static void
draw_table_frames(struct table *table, int indent, int y,
struct html_context *html_context)
{
struct table_frames table_frames;
signed char *frame[2];
int col, row;
int cx, cy;
int fh_size = (table->cols + 2) * (table->rows + 1);
int fv_size = (table->cols + 1) * (table->rows + 2);
frame[0] = fmem_alloc(fh_size + fv_size);
if (!frame[0]) return;
memset(frame[0], -1, fh_size + fv_size);
frame[1] = &frame[0][fh_size];
if (table->rules != TABLE_RULE_NONE)
init_table_rules(table, frame);
get_table_frames(table, &table_frames);
memset(&H_FRAME_POSITION(table, 0, 0), table_frames.top, table->cols);
memset(&H_FRAME_POSITION(table, 0, table->rows), table_frames.bottom, table->cols);
memset(&V_FRAME_POSITION(table, 0, 0), table_frames.left, table->rows);
memset(&V_FRAME_POSITION(table, table->cols, 0), table_frames.right, table->rows);
cy = y;
for (row = 0; row <= table->rows; row++) {
cx = indent;
if ((row > 0 && row < table->rows && has_hline_width(table, row))
|| (row == 0 && table_frames.top)
|| (row == table->rows && table_frames.bottom)) {
int w = table_frames.left ? table->border : -1;
for (col = 0; col < table->cols; col++) {
if (col > 0)
w = get_vline_width(table, col);
if (w >= 0) {
draw_frame_point(table, frame, cx, cy, col, row,
html_context);
if (row < table->rows)
draw_frame_vline(table, frame, cx, cy + 1, col, row, html_context);
cx++;
}
draw_frame_hline(table, frame, cx, cy, col, row,
html_context);
cx += table->cols_widths[col];
}
if (table_frames.right) {
draw_frame_point(table, frame, cx, cy, col, row,
html_context);
if (row < table->rows)
draw_frame_vline(table, frame, cx, cy + 1, col, row, html_context);
cx++;
}
cy++;
} else if (row < table->rows) {
for (col = 0; col <= table->cols; col++) {
if ((col > 0 && col < table->cols && has_vline_width(table, col))
|| (col == 0 && table_frames.left)
|| (col == table->cols && table_frames.right)) {
draw_frame_vline(table, frame, cx, cy, col, row, html_context);
cx++;
}
if (col < table->cols) cx += table->cols_widths[col];
}
}
if (row < table->rows) cy += table->rows_heights[row];
}
fmem_free(frame[0]);
}
static void
draw_table_caption(struct html_context *html_context, struct table *table,
int x, int y)
{
unsigned char *start = table->caption.start;
unsigned char *end = table->caption.end;
struct part *part;
if (!start || !end) return;
while (start < end && isspace(*start))
start++;
while (start < end && isspace(end[-1]))
end--;
if (start >= end) return;
part = format_html_part(html_context, start, end, table->align,
0, table->real_width, table->part->document, x, y,
NULL, table->link_num);
if (!part) return;
table->part->cy += part->box.height;
table->part->cx = -1;
table->part->link_num = part->link_num;
mem_free(part);
}
/* This renders tag soup elements that the parser detected while chewing it's
* way through the table HTML. */
static void
draw_table_bad_html(struct html_context *html_context, struct table *table)
{
int i;
for (i = 0; i < table->bad_html_size; i++) {
struct html_start_end *html = &table->bad_html[i];
unsigned char *start = html->start;
unsigned char *end = html->end;
while (start < end && isspace(*start))
start++;
while (start < end && isspace(end[-1]))
end--;
if (start >= end) continue;
parse_html(start, end, table->part, NULL, html_context);
}
}
static void
distribute_table_widths(struct table *table)
{
int width = table->width;
if (table->min_width >= width)
width = table->min_width;
else if (table->max_width < width && table->full_width)
width = table->max_width;
distribute_widths(table, width);
}
void
format_table(unsigned char *attr, unsigned char *html, unsigned char *eof,
unsigned char **end, struct html_context *html_context)
{
struct part *part = html_context->part;
struct table *table;
struct node *node, *new_node;
struct html_element *state;
int indent, margins;
html_context->table_level++;
table = parse_table(html, eof, end, attr, (part->document || part->box.x),
html_context);
if (!table) goto ret0;
table->part = part;
/* XXX: This tag soup handling needs to be done outside the create
* parser state. Something to do with link numbering. */
/* It needs to be done _before_ processing the actual table, too.
* Otherwise i.e. <form> tags between <table> and <tr> are broken. */
draw_table_bad_html(html_context, table);
state = init_html_parser_state(html_context, ELEMENT_DONT_KILL,
ALIGN_LEFT, 0, 0);
margins = par_format.leftmargin + par_format.rightmargin;
if (get_table_cellpadding(html_context, table)) goto ret2;
distribute_table_widths(table);
if (!part->document && part->box.x == 1) {
int total_width = table->real_width + margins;
int_bounds(&total_width, table->real_width, par_format.width);
int_lower_bound(&part->box.width, total_width);
part->cy += table->real_height;
goto ret2;
}
#ifdef HTML_TABLE_2ND_PASS
check_table_widths(html_context, table);
#endif
get_table_heights(html_context, table);
if (!part->document) {
int_lower_bound(&part->box.width, table->real_width + margins);
part->cy += table->real_height;
goto ret2;
}
node = part->document->nodes.next;
node->box.height = part->box.y - node->box.y + part->cy;
indent = get_table_indent(html_context, table);
/* FIXME: See bug 432. It should be possible to align the caption at
* the top, bottom or the sides. */
draw_table_caption(html_context, table, indent + part->box.x, part->box.y + part->cy);
draw_table_cells(table, indent, part->cy, html_context);
draw_table_frames(table, indent, part->cy, html_context);
part->cy += table->real_height;
part->cx = -1;
new_node = mem_alloc(sizeof(*new_node));
if (new_node) {
set_box(&new_node->box, node->box.x, part->box.y + part->cy,
node->box.width, 0);
add_to_list(part->document->nodes, new_node);
}
ret2:
part->link_num = table->link_num;
int_lower_bound(&part->box.height, part->cy);
html_context->part = part; /* Might've changed in draw_table_cells(). */
done_html_parser_state(html_context, state);
free_table(table);
ret0:
html_context->table_level--;
if (!html_context->table_level) free_table_cache();
}