gecko-dev/lib/layout/laycols.c

844 lines
20 KiB
C

/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "xp.h"
#include "pa_parse.h"
#include "layout.h"
#include "laylayer.h"
#include "layers.h"
#define MULTICOL_MIN_WIDTH 10
#define MULTICOL_GUTTER_WIDTH 10
void lo_StartMultiColInit( lo_DocState *state, lo_MultiCol *multicol )
{
multicol->end_last_line = state->end_last_line;
multicol->start_ele = state->top_state->element_id;
multicol->start_line = state->line_num;
multicol->end_line = multicol->start_line;
multicol->start_x = state->x;
multicol->start_y = state->y;
multicol->end_y = multicol->start_y;
}
void lo_SetupStateForBeginMulticol( lo_DocState *state, lo_MultiCol *multicol, int32 doc_width )
{
int32 width;
/* Temporary create a tag on the stack to avoid recompile of everything! Make sure that this tag should be passed in as
a parameter before checking in.*/
PA_Tag tag;
tag.type = P_MULTICOLUMN;
tag.is_end = FALSE;
/* If absolute width specifed set width to that, else set it to full screen */
if (!multicol->isPercentWidth && multicol->width > 0)
width = multicol->width;
else
width = doc_width;
/* If percent width specified, set width to correct % of current doc_width */
if (multicol->isPercentWidth) {
int32 val = multicol->width;
if (state->allow_percent_width == FALSE) {
val = 0;
}
else {
val = doc_width * val / 100;
if (val < 1)
{
val = 1;
}
}
width = val;
}
width = width - ((multicol->cols - 1) * multicol->gutter);
multicol->col_width = width / multicol->cols;
if (multicol->col_width < MULTICOL_MIN_WIDTH)
{
multicol->col_width = MULTICOL_MIN_WIDTH;
}
multicol->orig_margin = state->right_margin;
lo_PushList(state, &tag, QUOTE_NONE);
state->right_margin = state->left_margin + multicol->col_width;
state->x = state->left_margin;
state->list_stack->old_left_margin = state->left_margin;
state->list_stack->old_right_margin = state->right_margin;
multicol->orig_min_width = state->min_width;
state->min_width = 0;
/*
* Don't display anything while processing the multicolumn text.
*/
multicol->orig_display_blocking_element_y = state->display_blocking_element_y;
state->display_blocking_element_y = -1;
multicol->orig_display_blocked = state->display_blocked;
state->display_blocked = TRUE;
multicol->next = state->current_multicol;
state->current_multicol = multicol;
}
void lo_AppendMultiColToLineList (MWContext *context, lo_DocState *state, LO_MulticolumnStruct *multicol)
{
multicol->lo_any.ele_id = NEXT_ELEMENT;
multicol->lo_any.x = state->x;
multicol->lo_any.y = state->y;
lo_AppendToLineList(context, state, (LO_Element*)multicol, 0);
}
/* Appends a zero width and height line feed to state->line_list. Flushes line list into line array */
void lo_AppendZeroWidthAndHeightLF(MWContext *context, lo_DocState *state)
{
LO_LinefeedStruct *lf;
lf = lo_NewLinefeed(state, context, LO_LINEFEED_BREAK_SOFT, LO_CLEAR_NONE);
lf->width = 0;
lf->height = 0;
lf->line_height = 0;
lo_AppendToLineList(context, state, (LO_Element *) lf, 0);
lo_AppendLineListToLineArray(context, state, (LO_Element *) lf);
}
void
lo_BeginMulticolumn(MWContext *context, lo_DocState *state, PA_Tag *tag, LO_MulticolumnStruct *multicolEle)
{
lo_MultiCol *multicol;
PA_Block buff;
char *str;
int32 val;
int32 doc_width;
/* int32 width; */
multicol = XP_NEW(lo_MultiCol);
if (multicol == NULL)
{
state->top_state->out_of_memory = TRUE;
return;
}
multicol->width = 0;
multicol->isPercentWidth = FALSE;
/* Put a pointer to lo_multicol inside the MULTICOLUMN layout element that will be placed on the
line list */
multicolEle->multicol = multicol;
/*
* If this multicol is within a layer, then we have to create an
* artificial TABLE around it (since the multicol code depends on
* the state's line array, which isn't in an updated state within
* a layer).
*/
if (state->layer_nest_level > 0)
{
PA_Tag *tmp_tag;
lo_TableRec *table;
if (state->in_paragraph != FALSE)
{
lo_CloseParagraph(context, &state, tag, 2);
}
lo_BeginTableAttributes(context, state, NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL,
NULL, NULL, NULL);
table = state->current_table;
if (table) {
lo_BeginTableRowAttributes(context, state, table,
NULL, NULL, NULL, NULL);
lo_BeginTableCellAttributes(context, state, table,
NULL, NULL, NULL, NULL, NULL,
LO_TILE_BOTH, NULL, NULL, NULL, NULL,
FALSE, TRUE);
/*
* If we've successfully created a subdoc, we need to clone
* the MULTICOL tag and save it in the subdoc, since the code
* above us doesn't realize we're now in a table.
*/
if (state->sub_state) {
state = state->sub_state;
tmp_tag = PA_CloneMDLTag(tag);
lo_SaveSubdocTags(context, state, tmp_tag);
}
multicol->close_table = TRUE;
}
}
else {
/* if (line will not be flushed by lo_SetSoftLineBreak) and (there exist some elements on the line list) */
if (state->linefeed_state >= 2 && state->line_list != NULL)
{
/* Append zero width and height line feed to the line list and flush the line list into
the line array. This forces the layout of elements contained within the MULTICOL tags
to start on a blank line_list and hence on a new line. lo_EndMultiColumn needs this to
do its line array hacking properly. */
lo_AppendZeroWidthAndHeightLF(context, state);
}
lo_SetSoftLineBreakState(context, state, FALSE, 2);
multicol->close_table = FALSE;
}
/*
* Since we are going to block layout during multicol
* processing, we need to flush the compositor of any
* pending displays.
*/
if (context->compositor)
{
CL_CompositeNow(context->compositor);
}
lo_StartMultiColInit( state, multicol );
/*
multicol->end_last_line = state->end_last_line;
multicol->start_ele = state->top_state->element_id;
multicol->start_line = state->line_num;
multicol->end_line = multicol->start_line;
multicol->start_x = state->x;
multicol->start_y = state->y;
multicol->end_y = multicol->start_y;
*/
multicol->cols = 1;
multicol->gutter = MULTICOL_GUTTER_WIDTH;
/*
* Get the cols parameter.
*/
buff = lo_FetchParamValue(context, tag, PARAM_COLS);
if (buff != NULL)
{
PA_LOCK(str, char *, buff);
multicol->cols = XP_ATOI(str);
PA_UNLOCK(buff);
PA_FREE(buff);
}
if (multicol->cols <= 1)
{
XP_DELETE(multicol);
multicolEle->multicol = NULL;
return;
}
/*
* Get the gutter parameter.
*/
buff = lo_FetchParamValue(context, tag, PARAM_GUTTER);
if (buff != NULL)
{
PA_LOCK(str, char *, buff);
multicol->gutter = XP_ATOI(str);
if (multicol->gutter < 1)
{
multicol->gutter = 1;
}
PA_UNLOCK(buff);
PA_FREE(buff);
}
multicol->gutter = FEUNITS_X(multicol->gutter, context);
doc_width = state->right_margin - state->left_margin;
/* width = doc_width; */
/*
* Get the width parameter, in absolute or percentage.
* If percentage, make it absolute.
*/
/*
buff = lo_FetchParamValue(context, tag, PARAM_WIDTH);
if (buff != NULL)
{
Bool is_percent;
PA_LOCK(str, char *, buff);
val = lo_ValueOrPercent(str, &is_percent);
if (is_percent != FALSE)
{
if (state->allow_percent_width == FALSE)
{
val = 0;
}
else
{
val = doc_width * val / 100;
if (val < 1)
{
val = 1;
}
}
}
else
{
val = FEUNITS_X(val, context);
if (val < 1)
{
val = 1;
}
}
width = val;
PA_UNLOCK(buff);
PA_FREE(buff);
}
*/
buff = lo_FetchParamValue(context, tag, PARAM_WIDTH);
if (buff != NULL)
{
Bool is_percent;
PA_LOCK(str, char *, buff);
val = lo_ValueOrPercent(str, &is_percent);
if (is_percent != FALSE)
{
multicol->width = val;
multicol->isPercentWidth = TRUE;
}
else
{
multicol->width = val;
multicol->isPercentWidth = FALSE;
val = FEUNITS_X(val, context);
if (val < 1)
{
val = 1;
}
}
PA_UNLOCK(buff);
PA_FREE(buff);
}
/*
width = width - ((multicol->cols - 1) * multicol->gutter);
multicol->col_width = width / multicol->cols;
if (multicol->col_width < MULTICOL_MIN_WIDTH)
{
multicol->col_width = MULTICOL_MIN_WIDTH;
}
multicol->orig_margin = state->right_margin;
lo_PushList(state, tag, QUOTE_NONE);
*/
lo_SetupStateForBeginMulticol( state, multicol, doc_width );
/*
state->right_margin = state->left_margin + multicol->col_width;
state->x = state->left_margin;
state->list_stack->old_left_margin = state->left_margin;
state->list_stack->old_right_margin = state->right_margin;
multicol->orig_min_width = state->min_width;
state->min_width = 0;
*/
/*
* Don't display anything while processing the multicolumn text.
*/
/*
multicol->orig_display_blocking_element_y = state->display_blocking_element_y;
state->display_blocking_element_y = -1;
multicol->orig_display_blocked = state->display_blocked;
state->display_blocked = TRUE;
multicol->next = state->current_multicol;
state->current_multicol = multicol;
*/
}
static LO_Element *
lo_capture_floating_elements(MWContext *context, lo_DocState *state,
int32 ele1, int32 ele2, int32 top_y, int32 bottom_y,
int32 *max_y)
{
LO_Element *prev_eptr;
LO_Element *eptr;
LO_Element *float_list;
*max_y = bottom_y;
float_list = NULL;
prev_eptr = NULL;
eptr = state->float_list;
while (eptr != NULL)
{
LO_Element *element;
if (((eptr->lo_any.y >= top_y)&&
(eptr->lo_any.y < bottom_y))||
((eptr->lo_any.ele_id >= ele1)&&
(eptr->lo_any.ele_id <= ele2)))
{
int32 my_y;
element = eptr;
if (prev_eptr != NULL)
{
prev_eptr->lo_any.next = eptr->lo_any.next;
}
else if (eptr == state->float_list)
{
state->float_list = eptr->lo_any.next;
}
eptr = eptr->lo_any.next;
element->lo_any.next = float_list;
float_list = element;
my_y = element->lo_any.y +
element->lo_any.y_offset +
element->lo_any.height;
if (element->type == LO_IMAGE)
{
/*
* Images need to account for border width
*/
my_y = my_y +
(2 * element->lo_image.border_width);
}
if (my_y > *max_y)
{
*max_y = my_y;
}
}
else
{
prev_eptr = eptr;
eptr = eptr->lo_any.next;
}
}
return(float_list);
}
LO_CellStruct *
lo_CaptureLines(MWContext *context, lo_DocState *state, int32 col_width,
int32 line1, int32 line2, int32 dx, int32 dy)
{
int32 height, max_y;
int32 ele1, ele2;
LO_Element *eptr;
LO_Element *last_eptr;
LO_Element *past_eptr;
LO_CellStruct *cell;
eptr = lo_FirstElementOfLine(context, state, line1);
if (eptr == NULL)
{
return(NULL);
}
last_eptr = NULL;
past_eptr = lo_FirstElementOfLine(context, state, (line2 + 1));
if (past_eptr != NULL)
{
last_eptr = past_eptr->lo_any.prev;
}
if (last_eptr == NULL)
{
last_eptr = lo_FirstElementOfLine(context, state, line2);
if (last_eptr == NULL)
{
last_eptr = eptr;
}
while (last_eptr->lo_any.next != NULL)
{
last_eptr = last_eptr->lo_any.next;
}
}
if (past_eptr != NULL)
{
height = past_eptr->lo_any.y - eptr->lo_any.y;
}
else
{
height = last_eptr->lo_any.y + last_eptr->lo_any.line_height -
eptr->lo_any.y;
}
cell = (LO_CellStruct *)lo_NewElement(context, state, LO_CELL, NULL, 0);
if (cell == NULL)
{
state->top_state->out_of_memory = TRUE;
return(NULL);
}
cell->type = LO_CELL;
cell->ele_id = NEXT_ELEMENT;
cell->x = eptr->lo_any.x + dx;
cell->x_offset = 0;
cell->y = eptr->lo_any.y + dy;
cell->y_offset = 0;
cell->width = col_width;
cell->height = height;
cell->next = NULL;
cell->prev = NULL;
cell->FE_Data = NULL;
cell->cell_float_list = NULL;
cell->backdrop.bg_color = NULL;
cell->backdrop.url = NULL;
cell->border_width = 0;
cell->border_vert_space = 0;
cell->border_horiz_space = 0;
cell->ele_attrmask = 0;
cell->sel_start = -1;
cell->sel_end = -1;
cell->cell_list = eptr;
cell->cell_list_end = last_eptr;
cell->cell_bg_layer = NULL;
cell->cell_inflow_layer = NULL;
cell->table_cell = NULL;
cell->table_row = NULL;
cell->table = NULL;
if (eptr->lo_any.prev != NULL)
{
eptr->lo_any.prev->lo_any.next = NULL;
eptr->lo_any.prev = NULL;
}
if (last_eptr->lo_any.next != NULL)
{
last_eptr->lo_any.next->lo_any.prev = NULL;
last_eptr->lo_any.next = NULL;
}
ele1 = eptr->lo_any.ele_id;
ele2 = last_eptr->lo_any.ele_id;
cell->cell_float_list = lo_capture_floating_elements(context, state,
ele1, ele2, eptr->lo_any.y, (eptr->lo_any.y + height), &max_y);
if ((max_y - eptr->lo_any.y) > cell->height)
{
cell->height = max_y - eptr->lo_any.y;
}
lo_ShiftCell(cell, dx, dy);
return(cell);
}
static int32
lo_find_breaking_line(MWContext *context, lo_DocState *state,
int32 start_line, int32 start_x, int32 end_x)
{
LO_Element *eptr;
int32 line;
/*
* Now that we overuse the BREAKABLE bitflag, this
* is easy, any line whose linefeed has that flag set
* is a breakable line.
*/
line = start_line + 1;
eptr = lo_FirstElementOfLine(context, state, line);
while (eptr != NULL)
{
LO_Element *tmp_eptr;
tmp_eptr = eptr;
while ((tmp_eptr != NULL)&&
(tmp_eptr->type != LO_LINEFEED))
{
tmp_eptr = tmp_eptr->lo_any.next;
}
if ((tmp_eptr != NULL)&&
(tmp_eptr->lo_linefeed.ele_attrmask & LO_ELE_BREAKABLE))
{
return(line - 1);
}
line++;
eptr = lo_FirstElementOfLine(context, state, line);
}
return(line - 1);
}
void
lo_EndMulticolumn(MWContext *context, lo_DocState *state, PA_Tag *tag,
lo_MultiCol *multicol, Bool relayout)
{
int32 i;
lo_ListStack *lptr;
int32 height;
int32 col_height;
int32 x, y;
int32 line1, line2;
int32 save_doc_min_width;
int32 min_multi_width;
LO_Element *cell_list;
LO_Element *cell_list_end;
LO_Element *cell_ele;
LO_TableStruct *table_ele;
cell_ele = NULL;
cell_list = NULL;
cell_list_end = NULL;
lo_SetLineBreakState (context, state, FALSE, LO_LINEFEED_BREAK_SOFT, 1, relayout);
multicol->end_line = state->line_num;
multicol->end_y = state->y;
/*
* Break to clear all left and right margins.
*/
lo_ClearToBothMargins(context, state);
/*
* Reset the margins properly in case
* we are inside a list.
*/
lo_FindLineMargins(context, state, !relayout);
state->x = state->left_margin;
height = multicol->end_y - multicol->start_y;
col_height = height / multicol->cols;
if (col_height < 1)
{
col_height = 1;
}
x = state->x;
y = multicol->start_y;
line1 = multicol->start_line - 1;
for (i=0; i<multicol->cols; i++)
{
LO_CellStruct *cell;
LO_Element *eptr;
int32 line;
int32 dx, dy;
int32 move_height;
eptr = lo_FirstElementOfLine(context, state, line1);
if (eptr == NULL)
{
break;
}
y = eptr->lo_any.y + col_height;
line = lo_PointToLine(context, state, x, y);
eptr = lo_FirstElementOfLine(context, state, line);
if (eptr->lo_any.x > multicol->start_x)
{
line = lo_find_breaking_line(context, state, line,
multicol->start_x,
(multicol->start_x + multicol->col_width));
}
line2 = line;
if (line2 > (multicol->end_line - 2))
{
line2 = (multicol->end_line - 2);
}
cell = NULL;
if (i > 0)
{
eptr = lo_FirstElementOfLine(context, state, line1);
if (eptr != NULL)
{
dx = i * (multicol->col_width +
multicol->gutter);
dy = multicol->start_y - eptr->lo_any.y;
cell = lo_CaptureLines(context, state,
multicol->col_width,
line1, line2, dx, dy);
if (cell == NULL)
{
move_height = 0;
}
else
{
move_height = cell->height - 1;
}
}
else
{
move_height = 0;
}
}
else
{
cell = lo_CaptureLines(context, state,
multicol->col_width,
line1, line2, 0, 0);
if (cell == NULL)
{
move_height = 0;
}
else
{
move_height = cell->height - 1;
}
}
line1 = line2 + 1;
if (move_height > col_height)
{
col_height = move_height;
}
if (cell != NULL)
{
if (cell_list_end == NULL)
{
cell_list = (LO_Element *)cell;
cell_list_end = cell_list;
}
else
{
cell_list_end->lo_any.next = (LO_Element *)cell;
cell_list_end = cell_list_end->lo_any.next;
}
}
}
lptr = lo_PopList(state, tag);
if (lptr != NULL)
{
XP_DELETE(lptr);
}
state->left_margin = state->list_stack->old_left_margin;
state->right_margin = state->list_stack->old_right_margin;
state->top_state->element_id = multicol->start_ele;
lo_SetLineArrayEntry(state, NULL, (multicol->start_line - 1));
state->line_num = multicol->start_line;
state->end_last_line = multicol->end_last_line;
if (state->end_last_line != NULL)
{
state->end_last_line->lo_any.next = NULL;
}
state->line_list = NULL;
state->y = multicol->start_y;
state->display_blocked = multicol->orig_display_blocked;
state->display_blocking_element_y = multicol->orig_display_blocking_element_y;
table_ele = (LO_TableStruct *)lo_NewElement(context, state, LO_TABLE,
NULL, 0);
if (table_ele == NULL)
{
state->top_state->out_of_memory = TRUE;
if (multicol->close_table)
lo_CloseTable(context, state);
return;
}
table_ele->type = LO_TABLE;
table_ele->ele_id = NEXT_ELEMENT;
table_ele->x = x;
table_ele->x_offset = 0;
table_ele->y = multicol->start_y;
table_ele->y_offset = 0;
table_ele->width = (multicol->cols * multicol->col_width) +
((multicol->cols - 1) * multicol->gutter);
table_ele->height = col_height + 1;
table_ele->line_height = col_height + 1;
table_ele->FE_Data = NULL;
table_ele->anchor_href = NULL;
table_ele->alignment = LO_ALIGN_LEFT;
table_ele->border_width = 0;
table_ele->border_vert_space = 0;
table_ele->border_horiz_space = 0;
table_ele->border_style = BORDER_NONE;
table_ele->ele_attrmask = 0;
table_ele->sel_start = -1;
table_ele->sel_end = -1;
table_ele->next = NULL;
table_ele->prev = NULL;
table_ele->table = NULL;
lo_AppendToLineList(context, state, (LO_Element *)table_ele, 0);
while (cell_list != NULL)
{
cell_ele = cell_list;
cell_list = cell_list->lo_any.next;
cell_ele->lo_any.next = NULL;
cell_ele->lo_any.ele_id = NEXT_ELEMENT;
lo_RenumberCell(state, (LO_CellStruct *)cell_ele);
lo_AppendToLineList(context, state, cell_ele, 0);
}
if (cell_ele != NULL)
{
state->x = cell_ele->lo_any.x + multicol->col_width;
}
else
{
state->x = table_ele->x + table_ele->width;
}
state->baseline = 0;
state->line_height = col_height + 1;
state->linefeed_state = 0;
state->at_begin_line = FALSE;
state->trailing_space = FALSE;
state->cur_ele_type = LO_SUBDOC;
min_multi_width = (state->min_width * multicol->cols) +
((multicol->cols - 1) * multicol->gutter);
state->min_width = multicol->orig_min_width;
save_doc_min_width = state->min_width;
lo_SetLineBreakState (context, state, FALSE, LO_LINEFEED_BREAK_SOFT, 2, relayout);
if (min_multi_width > save_doc_min_width)
{
save_doc_min_width = min_multi_width;
}
state->min_width = save_doc_min_width;
state->current_multicol = multicol->next;
if (!relayout)
{
if (multicol->close_table) {
PA_Tag *tmp_tag;
if (state->in_paragraph != FALSE)
{
lo_CloseParagraph(context, &state, tag, 2);
}
tmp_tag = PA_CloneMDLTag(tag);
lo_SaveSubdocTags(context, state, tmp_tag);
lo_CloseTable(context, state);
}
}
}