mirror of
https://github.com/netsurf-browser/netsurf
synced 2024-11-24 15:29:45 +03:00
e063a2a59d
Merged revisions 4195-4211,4216,4219-4220,4222-4234,4236-4250,4252-4262,4264-4266,4268-4326,4329-4335,4338-4342,4344-4411,4413-4420,4422-4436,4438-4491,4494-4506,4508-4514,4516,4518-4552,4554,4556-4564,4567-4568,4570-4574,4576-4686,4689-4692,4694,4698-4709,4715-4723,4725-4755,4757-4769,4771-4919,4921-4996,4998-5110,5112-5117 via svnmerge from svn://svn.netsurf-browser.org/branches/adamblokus/netsurf ........ r4736 | adamblokus | 2008-07-26 13:46:54 +0200 (Sat, 26 Jul 2008) | 2 lines Sorting out some problems with svn. ........ r4737 | adamblokus | 2008-07-26 13:54:36 +0200 (Sat, 26 Jul 2008) | 4 lines Added export tab to the options dialog. Added the possibility of changing some print options. ........ r4897 | adamblokus | 2008-08-04 17:59:05 +0200 (Mon, 04 Aug 2008) | 5 lines Added checking of horizontal clipping. Added better table loosening. Changed some minor bugs. Applied changes in the Export options tab according to the review from tlsa. ........ r4905 | adamblokus | 2008-08-05 01:53:34 +0200 (Tue, 05 Aug 2008) | 2 lines Fixed bug which made it impossible to export pdf's. ........ r4919 | adamblokus | 2008-08-05 16:39:33 +0200 (Tue, 05 Aug 2008) | 2 lines Fixed some memory leaks which caused Netsurf to break. ........ r4927 | adamblokus | 2008-08-06 02:26:30 +0200 (Wed, 06 Aug 2008) | 4 lines Fixed bug with filenames which crashed Netsurf. Turned anti aliasing off for printing. Fixed some scaling issues. ........ r4928 | adamblokus | 2008-08-06 17:52:44 +0200 (Wed, 06 Aug 2008) | 5 lines Added new export/print options: - suppressing images - turning off backgrounds - toggled loosening ........ r4950 | adamblokus | 2008-08-07 21:15:21 +0200 (Thu, 07 Aug 2008) | 5 lines Added new options to PDF export: - document compression - document encryption Added PDF password dialog ........ r4954 | adamblokus | 2008-08-07 22:11:31 +0200 (Thu, 07 Aug 2008) | 2 lines Added saving print settings. ........ r4956 | adamblokus | 2008-08-07 22:44:48 +0200 (Thu, 07 Aug 2008) | 2 lines Fixes to PDF encryption ........ r4970 | adamblokus | 2008-08-09 15:26:24 +0200 (Sat, 09 Aug 2008) | 3 lines Fixed bug in plotting tiled bitmaps. Fixed bug with too long text decorations. ........ r4977 | adamblokus | 2008-08-09 19:18:56 +0200 (Sat, 09 Aug 2008) | 2 lines Fixed JPG embedding bug. ........ r4988 | adamblokus | 2008-08-10 16:59:51 +0200 (Sun, 10 Aug 2008) | 3 lines Added clip checking to pdf plotters. No more "blank" clips. Made PDF compression a default setting. ........ r4995 | adamblokus | 2008-08-10 20:03:00 +0200 (Sun, 10 Aug 2008) | 2 lines Fixed Haru crash on font-size==0. ........ r4996 | adamblokus | 2008-08-10 21:04:43 +0200 (Sun, 10 Aug 2008) | 2 lines Added changing text mode only if necessary. ........ r5045 | adamblokus | 2008-08-11 21:26:26 +0200 (Mon, 11 Aug 2008) | 3 lines Removing gtk stuff from core code. Little fix in options. ........ r5048 | adamblokus | 2008-08-11 21:57:45 +0200 (Mon, 11 Aug 2008) | 2 lines Better font size checking in PDF export. ........ r5050 | adamblokus | 2008-08-11 22:19:56 +0200 (Mon, 11 Aug 2008) | 2 lines Fixed riscos text scale bug. ........ r5073 | adamblokus | 2008-08-12 17:40:57 +0200 (Tue, 12 Aug 2008) | 2 lines Added missing tooltips ........ r5092 | adamblokus | 2008-08-13 17:09:25 +0200 (Wed, 13 Aug 2008) | 2 lines Moved /pdf folder to desktop/save_pdf ........ r5110 | adamblokus | 2008-08-13 22:44:50 +0200 (Wed, 13 Aug 2008) | 2 lines Added comments. ........ r5113 | adamblokus | 2008-08-13 23:07:35 +0200 (Wed, 13 Aug 2008) | 2 lines Cosmetic changes ........ r5116 | adamblokus | 2008-08-14 16:10:18 +0200 (Thu, 14 Aug 2008) | 2 lines Fixed bug with BOX_INLINE_END in tree duplication. ........ r5117 | joty | 2008-08-14 21:47:46 +0200 (Thu, 14 Aug 2008) | 1 line Improvement for r5116: use local vars when possible; rename global last to box_duplicate_last; check on box_duplicate_main_tree failure. ........ svn path=/trunk/netsurf/; revision=5118
502 lines
13 KiB
C
502 lines
13 KiB
C
/*
|
|
* Copyright 2008 Adam Blokus <adamblokus@gmail.com>
|
|
*
|
|
* This file is part of NetSurf, http://www.netsurf-browser.org/
|
|
*
|
|
* NetSurf 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; version 2 of the License.
|
|
*
|
|
* NetSurf 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stdbool.h>
|
|
#include <assert.h>
|
|
|
|
#include "content/content.h"
|
|
|
|
#include "render/box.h"
|
|
#include "render/font.h"
|
|
|
|
#include "render/layout.h"
|
|
#include "render/loosen.h"
|
|
|
|
#include "utils/log.h"
|
|
#include "utils/talloc.h"
|
|
|
|
#define AUTO INT_MIN
|
|
#define LOOSEN_MIN_TEXT_SIZE 10
|
|
|
|
static bool loosen_text(struct box *text, int width, struct content *content);
|
|
|
|
static bool loosen_table(struct box *box, int available_width,
|
|
struct content *content);
|
|
|
|
static bool loosen_position_static(struct box *box, int width, int cx,
|
|
struct content *content);
|
|
|
|
static bool loosen_shrink_object(struct box *box, int width);
|
|
|
|
static bool loosen_all_first_pass(struct box *box, int width, int cx,
|
|
struct content *content);
|
|
static bool loosen_all_second_pass(struct box *box, int width, int cx,
|
|
struct content *content);
|
|
static bool loosen_all_margins_paddings(struct box *box, int width, int cx,
|
|
struct content *content);
|
|
|
|
static bool loosen_shrink_text(struct box *box);
|
|
|
|
/**
|
|
* Main loosing procedure
|
|
* \param content Reformated content - talloc memory pool for new boxes
|
|
* \param layout Root of the loosened box tree
|
|
* \param width Width the content is intended to fit
|
|
* \param height Height of a single page - to be taken into consideration for \
|
|
* preventing elements for being cropped at top/bottom edges of pages.
|
|
* \return true if successful, false otherwise (lack of memory)
|
|
*/
|
|
bool loosen_document_layout(struct content *content, struct box *layout,
|
|
int width, int height)
|
|
{
|
|
/* Optional try - if the current layout is not more than xx% too wide,
|
|
* maybe we scale the content to preserve the original layout?
|
|
*/
|
|
|
|
if (!loosen_all_first_pass(layout, width, 0, content))
|
|
return false;
|
|
layout->min_width = 0;
|
|
layout->max_width = UNKNOWN_MAX_WIDTH;
|
|
content_reformat(content, width, 0);
|
|
|
|
/*Check if pass 1 was enough - if re-layouting doesn't give
|
|
*us the right width, go on to pass 2. And again - if pass 2 was not
|
|
*enough - go on to pass 3
|
|
*/
|
|
|
|
if (content->width > width) {
|
|
if (!loosen_all_second_pass(layout, width, 0, content))
|
|
return false;
|
|
layout->min_width = 0;
|
|
layout->max_width = UNKNOWN_MAX_WIDTH;
|
|
content_reformat(content, width, 0);
|
|
}
|
|
|
|
if (content->width > width) {
|
|
if (!loosen_all_margins_paddings(layout, width, 0, content))
|
|
return false;
|
|
layout->min_width = 0;
|
|
layout->max_width = UNKNOWN_MAX_WIDTH;
|
|
content_reformat(content, width, 0);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/** Primarily - break too wide words into pieces.
|
|
* \param text - the box that contains text to be broken
|
|
* \param width Width the content is intended to fit
|
|
* \param content talloc memory pool for new boxes
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_text(struct box *text, int width, struct content *content)
|
|
{
|
|
size_t offset;
|
|
int actual_x;
|
|
|
|
int *breaks;
|
|
int break_count;
|
|
|
|
int position, i;
|
|
const struct font_functions *font_func;
|
|
|
|
if (content->type == CONTENT_HTML)
|
|
font_func = content->data.html.font_func;
|
|
else
|
|
return false;
|
|
|
|
if (text->width <= width) {
|
|
LOG(("loosen_text called unnecessary?"));
|
|
/*Still - not an error for this function*/
|
|
return true;
|
|
}
|
|
|
|
breaks = malloc( sizeof(int) * text->length);
|
|
if (breaks == NULL)
|
|
return false;
|
|
|
|
break_count = 0;
|
|
position = 0;
|
|
|
|
while (position < text->length) {
|
|
font_func->font_position_in_string(text->style,
|
|
text->text + position,
|
|
text->length - position,
|
|
width, &offset, &actual_x);
|
|
|
|
if (offset < text->length - position) {
|
|
/*Another break*/
|
|
LOG(("Current text broken at offset %d",
|
|
position + offset));
|
|
breaks[break_count++] = position + offset-1;
|
|
}
|
|
|
|
position += offset;
|
|
}
|
|
|
|
text->text = talloc_realloc(content, text->text, char,
|
|
text->length + break_count);
|
|
|
|
i = text->length-1;
|
|
text->length = text->length + break_count;
|
|
|
|
for (; i>=0; i--) {
|
|
text->text[i + break_count] = text->text[i];
|
|
if (i == breaks[break_count - 1]) {
|
|
break_count--;
|
|
text->text[i + break_count] = ' ';
|
|
}
|
|
}
|
|
|
|
free(breaks);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Changing table layout and structure to fit the contents width.
|
|
* Firstly the borders are collapsed and the text is shrunken.
|
|
* Secondly the text is loosened( this can be helpful for all data tables which
|
|
* contain only text)
|
|
* In the most extreme case - the table has no influence on the width
|
|
* (each row is broken into one-cell rows).
|
|
* \param table - the box that contains table to be broken
|
|
* \param width Width the content is intended to fit
|
|
* \param content talloc memory pool for new boxes
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_table(struct box *table, int width, struct content *content)
|
|
{
|
|
struct box *row_group, *row, *cell, *br, *prev, *inline_container;
|
|
|
|
struct box *text, *child;
|
|
unsigned int row_sum;
|
|
bool first_cell_in_row;
|
|
const struct font_functions *font_func;
|
|
float scale;
|
|
int new_width;
|
|
|
|
if (table->min_width <= width)
|
|
return true;
|
|
|
|
if (content->type == CONTENT_HTML)
|
|
font_func = content->data.html.font_func;
|
|
else
|
|
return false;
|
|
|
|
table->style->border_collapse = CSS_BORDER_COLLAPSE_COLLAPSE;
|
|
|
|
if (!loosen_shrink_text(table))
|
|
return false;
|
|
|
|
if (!loosen_all_margins_paddings(table, width, 0, content))
|
|
return false;
|
|
|
|
scale = width;
|
|
scale /= table->min_width;
|
|
|
|
for (row_group = table->children; row_group;
|
|
row_group = row_group->next) {
|
|
for (row = row_group->children; row; row = row->next) {
|
|
for (cell = row->children; cell; cell = cell->next) {
|
|
for (child = cell->children; child;
|
|
child = child->next) {
|
|
if (child->children)
|
|
text = child->children;
|
|
else
|
|
continue;
|
|
|
|
/*text in nested boxes won't be broken*/
|
|
if (text->type != BOX_TEXT)
|
|
continue;
|
|
|
|
|
|
/*break the words propotionally to the
|
|
current cell width*/
|
|
new_width = (float)cell->width * scale * 0.9;
|
|
loosen_text(text, new_width, content);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*check if the table is loosend enough...*/
|
|
layout_minmax_table(table, font_func);
|
|
if (table->min_width <= width)
|
|
return true;
|
|
|
|
|
|
/*...in case it's not continue with bigger changes,
|
|
table cells are changed into inline containers*/
|
|
inline_container = box_create(0, 0, 0, 0, 0, content);
|
|
inline_container->type = BOX_INLINE_CONTAINER;
|
|
inline_container->parent = table;
|
|
inline_container->style = talloc_memdup(content, table->style,
|
|
sizeof *table->style);
|
|
|
|
prev = NULL;
|
|
|
|
for (row_group = table->children; row_group;
|
|
row_group = row_group->next) {
|
|
for (row = row_group->children; row; row = row->next) {
|
|
|
|
for (cell = row->children; cell; cell = cell->next) {
|
|
cell->type = BOX_INLINE_BLOCK;
|
|
cell->prev = prev;
|
|
cell->parent = inline_container;
|
|
cell->max_width = width;
|
|
cell->min_width = 0;
|
|
|
|
if (prev!=NULL)
|
|
prev->next = cell;
|
|
else
|
|
inline_container->children = cell;
|
|
|
|
prev = cell;
|
|
}
|
|
|
|
br = box_create(0, 0, 0, 0, 0, content);
|
|
br->type = BOX_BR;
|
|
br->parent = inline_container;
|
|
br->prev = prev;
|
|
br->style = talloc_memdup(content, table->style,
|
|
sizeof *table->style);
|
|
br->style->clear = CSS_CLEAR_BOTH;
|
|
|
|
if (prev != NULL)
|
|
prev->next = br;
|
|
else
|
|
inline_container->children = br;
|
|
|
|
prev = br;
|
|
}
|
|
}
|
|
inline_container->last = prev;
|
|
|
|
table->type = BOX_BLOCK;
|
|
table->children = table->last = inline_container;
|
|
table->col = NULL;
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Recursively step through the box tree applying LOOSEN_MIN_TEXT_SIZE wherever
|
|
* text is found
|
|
* \param box the box where the shrinking should be started
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_shrink_text(struct box *box)
|
|
{
|
|
struct box *child;
|
|
|
|
box->max_width = UNKNOWN_MAX_WIDTH;
|
|
|
|
if (box->type == BOX_TEXT) {
|
|
box->style->font_size.size = CSS_FONT_SIZE_LENGTH;
|
|
box->style->font_size.value.length.unit = CSS_UNIT_PX;
|
|
box->style->font_size.value.length.value = LOOSEN_MIN_TEXT_SIZE;
|
|
}
|
|
else if (box->children)
|
|
for(child = box->children; child; child = child->next)
|
|
if (!loosen_shrink_text(child))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Change absolute and relative positioned elements into block elements
|
|
* in case they are positioned to far to the rigth
|
|
* \param box - the box that should be changed
|
|
* \param width Width the content is intended to fit
|
|
* \param cx current x - not yet in use
|
|
* \param content talloc memory pool for new boxes
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_position_static(struct box *box, int width, int cx,
|
|
struct content *content)
|
|
{
|
|
assert(box->style);
|
|
|
|
if (box->style->position == CSS_POSITION_ABSOLUTE) {
|
|
box->style->position = CSS_POSITION_NOT_SET;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Shrink an object (esp. an image) to fit the page-width
|
|
* \note Not sure wheter it won't be better for images to be cropped
|
|
* \param box - the box that should be changed
|
|
* \param width Width the content is intended to fit
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_shrink_object(struct box *box, int width)
|
|
{
|
|
assert(box->object != NULL);
|
|
|
|
box->height = AUTO;
|
|
box->width = width;
|
|
|
|
if (box->style) {
|
|
box->style->width.width = CSS_WIDTH_PERCENT;
|
|
box->style->width.value.percent = 100;
|
|
box->style->height.height= CSS_HEIGHT_AUTO;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Pass 1 of loosening - do such obvious changes as: breaking too long words,
|
|
* moving absolute positioned objects into the visibile scope of width.
|
|
* \param box - the box that should be changed
|
|
* \param width Width the content is intended to fit
|
|
* \param cx current x - not yet in use
|
|
* \param content talloc memory pool for new boxes
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_all_first_pass(struct box *box, int width, int cx,
|
|
struct content *content)
|
|
{
|
|
struct box* c;
|
|
int x;
|
|
|
|
for (c = box->children; c ; c = c->next) {
|
|
x = cx + c->x;
|
|
if (c->children != NULL)
|
|
if (!loosen_all_first_pass(c, width, x, content))
|
|
return false;
|
|
|
|
if (c->style) {
|
|
if (c->style->position == CSS_POSITION_RELATIVE ||
|
|
c->style->position == CSS_POSITION_ABSOLUTE )
|
|
if (!loosen_position_static(c, width, cx, content))
|
|
return false;
|
|
if ( c->style->width.width == CSS_WIDTH_LENGTH &&
|
|
css_len2px(&c->style->width.value.length, c->style) > width)
|
|
c->style->width.width = CSS_WIDTH_NOT_SET;
|
|
}
|
|
|
|
if (c->object && c->width > width)
|
|
if (!loosen_shrink_object(c, width))
|
|
return false;
|
|
|
|
switch (c->type) {
|
|
case BOX_TEXT:
|
|
if (!loosen_text(c, width, content))
|
|
return false;
|
|
break;
|
|
}
|
|
|
|
c->min_width = 0;
|
|
c->max_width = UNKNOWN_MAX_WIDTH;
|
|
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Pass 2 of loosening - break tables
|
|
* \param box - the box that should be changed
|
|
* \param width Width the content is intended to fit
|
|
* \param cx current x - not yet in use
|
|
* \param content talloc memory pool for new boxes
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_all_second_pass(struct box *box, int width, int cx,
|
|
struct content *content)
|
|
{
|
|
struct box *c;
|
|
int x;
|
|
|
|
for (c = box->children; c; c = c->next) {
|
|
x = cx + c->x;
|
|
if (c->children != NULL)
|
|
if (!loosen_all_second_pass(c, width, x, content))
|
|
return false;
|
|
|
|
switch (c->type) {
|
|
case BOX_TABLE:
|
|
if (!loosen_table(c, width, content))
|
|
return false;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
c->min_width = 0;
|
|
c->max_width = UNKNOWN_MAX_WIDTH;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Pass 3 of loosening -zero all margins and paddings
|
|
* \param box - the box that should be changed
|
|
* \param width Width the content is intended to fit
|
|
* \param cx current x - not yet in use
|
|
* \param content talloc memory pool for new boxes
|
|
* \return true if successful, false otherwise
|
|
*/
|
|
bool loosen_all_margins_paddings(struct box *box, int width, int cx,
|
|
struct content *content)
|
|
{
|
|
struct box *c;
|
|
int x;
|
|
|
|
for (c = box->children; c; c = c->next) {
|
|
x = cx + c->x;
|
|
if (c->children != NULL)
|
|
if (!loosen_all_margins_paddings(c, width, x, content))
|
|
return false;
|
|
|
|
c->padding[LEFT] = c->padding[RIGHT] = 0;
|
|
c->margin[LEFT] = c->margin[RIGHT] = 0;
|
|
|
|
if (c->style) {
|
|
c->style->margin[LEFT].margin = CSS_MARGIN_PERCENT;
|
|
c->style->margin[LEFT].value.percent = 0;
|
|
|
|
c->style->margin[RIGHT].margin = CSS_MARGIN_PERCENT;
|
|
c->style->margin[RIGHT].value.percent = 0;
|
|
|
|
c->style->padding[LEFT].padding = CSS_PADDING_PERCENT;
|
|
c->style->padding[LEFT].value.percent = 0;
|
|
|
|
c->style->padding[RIGHT].padding = CSS_PADDING_PERCENT;
|
|
c->style->padding[RIGHT].value.percent = 0;
|
|
|
|
}
|
|
|
|
c->min_width = 0;
|
|
c->max_width = UNKNOWN_MAX_WIDTH;
|
|
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|