mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-12-16 11:45:31 +00:00
850 lines
21 KiB
C
850 lines
21 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
*
|
|
* 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 "layout.h"
|
|
#include "laylayer.h"
|
|
#include "libi18n.h"
|
|
#include "xlate.h"
|
|
#include "layers.h"
|
|
|
|
#define IL_CLIENT /* XXXM12N Defined by Image Library clients */
|
|
#include "libimg.h" /* Image Library public API. */
|
|
#include "timing.h"
|
|
|
|
#ifdef PROFILE
|
|
#pragma profile on
|
|
#endif
|
|
|
|
#ifdef XP_MAC
|
|
# ifdef XP_TRACE
|
|
# undef XP_TRACE
|
|
# endif
|
|
# define XP_TRACE(X)
|
|
#else
|
|
#ifndef XP_TRACE
|
|
# define XP_TRACE(X) fprintf X
|
|
#endif
|
|
#endif /* XP_MAC */
|
|
|
|
|
|
/*
|
|
* BUGBUG LAYERS: This is a hack that is currently being used for selection
|
|
* but might be useful elsewhere. Essentially, for extending the selected
|
|
* text, we know exactly how much we want to draw (using FE_DisplaySubtext).
|
|
* Rather than ask for a synchronous refresh of the area and relying on
|
|
* LO_RefreshArea to do the correct drawing (actually, it can't do the
|
|
* correct drawing since it doesn't draw sub-strings), we temporarily replace
|
|
* the painter_func for the layer in question to one that just does the
|
|
* required drawing. We still do a synchronous refresh, but the temporary
|
|
* painter_func is called instead of the one that calls LO_RefreshArea.
|
|
*/
|
|
typedef struct LO_SelectionHackStruct {
|
|
MWContext *context;
|
|
LO_TextStruct *text;
|
|
int32 start_pos;
|
|
int32 end_pos;
|
|
Bool need_bg;
|
|
} LO_SelectionHackStruct;
|
|
|
|
static void
|
|
lo_selection_hack_painter_func(void *drawable,
|
|
CL_Layer *layer,
|
|
FE_Region update_region)
|
|
{
|
|
LO_SelectionHackStruct *hack = (LO_SelectionHackStruct *)CL_GetLayerClientData(layer);
|
|
|
|
FE_SetDrawable(hack->context, drawable);
|
|
|
|
/* For layers, we don't ever need to clear the background */
|
|
FE_DisplaySubtext(hack->context, FE_VIEW,
|
|
hack->text, hack->start_pos, hack->end_pos,
|
|
FALSE);
|
|
|
|
FE_SetDrawable(hack->context, NULL);
|
|
}
|
|
|
|
static void
|
|
lo_NormalizeStartAndEndOfSelection(LO_TextStruct *text, int32 *start,
|
|
int32 *end)
|
|
{
|
|
int32 p1, p2;
|
|
char *string;
|
|
int n;
|
|
int16 charset;
|
|
|
|
if (text->ele_attrmask & LO_ELE_SELECTED)
|
|
{
|
|
if (text->sel_start < 0)
|
|
{
|
|
p1 = 0;
|
|
}
|
|
else
|
|
{
|
|
p1 = text->sel_start;
|
|
}
|
|
|
|
if (text->sel_end < 0)
|
|
{
|
|
p2 = text->text_len - 1;
|
|
}
|
|
else
|
|
{
|
|
p2 = text->sel_end;
|
|
}
|
|
|
|
PA_LOCK(string, char *, text->text);
|
|
|
|
/*
|
|
* find beginning of first character
|
|
*/
|
|
charset = text->text_attr->charset;
|
|
switch (n = INTL_NthByteOfChar(charset, string, (int)(p1+1)))
|
|
{
|
|
case 0:
|
|
case 1:
|
|
break;
|
|
default:
|
|
p1 -= (n - 1);
|
|
if (p1 < 0)
|
|
{
|
|
p1 = 0;
|
|
}
|
|
break;
|
|
}
|
|
if (text->sel_start >= 0)
|
|
{
|
|
text->sel_start = p1;
|
|
}
|
|
|
|
/*
|
|
* find end of last character
|
|
*/
|
|
switch (n = INTL_NthByteOfChar(charset, string, (int)(p2+1)))
|
|
{
|
|
case 0:
|
|
break;
|
|
default:
|
|
p2 -= (n - 1);
|
|
if (p2 < 0)
|
|
{
|
|
p2 = 0;
|
|
}
|
|
/* fall through */
|
|
case 1:
|
|
p2 += INTL_IsLeadByte(charset, string[p2]);
|
|
if (p2 > text->text_len - 1)
|
|
{
|
|
p2 = text->text_len - 1;
|
|
}
|
|
break;
|
|
}
|
|
if (text->sel_end >= 0)
|
|
{
|
|
text->sel_end = p2;
|
|
}
|
|
|
|
*start = p1;
|
|
*end = p2;
|
|
|
|
PA_UNLOCK(text->text);
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
void
|
|
LO_DUMP_RECT(XP_Rect *rect)
|
|
{
|
|
XP_TRACE(("Compositing rectangle: [%3d,%3d] %d x %d\n", rect->left, rect->top,
|
|
rect->right - rect->left, rect->bottom - rect->top));
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* BUGBUG LAYERS: Since we know that lo_DisplaySubtext is only done
|
|
* for selection, we generate a synchronous composite. This might
|
|
* not be the right thing to do: maybe we should have an additional
|
|
* argument to the LO_Display routines
|
|
*/
|
|
void
|
|
lo_DisplaySubtext(MWContext *context, LO_TextStruct *text,
|
|
int32 start_pos, int32 end_pos, Bool need_bg,
|
|
CL_Layer *sel_layer)
|
|
{
|
|
lo_NormalizeStartAndEndOfSelection(text, &start_pos, &end_pos);
|
|
/* If we have to draw, go through the compositor */
|
|
if (context->compositor) {
|
|
XP_Rect rect;
|
|
CL_Layer *layer;
|
|
int32 x_offset, y_offset;
|
|
|
|
if (!sel_layer)
|
|
layer = CL_FindLayer(context->compositor, LO_BODY_LAYER_NAME);
|
|
else
|
|
layer = sel_layer;
|
|
|
|
FE_GetTextFrame(context, text, start_pos, end_pos, &rect);
|
|
lo_GetLayerXYShift(layer, &x_offset, &y_offset);
|
|
XP_OffsetRect(&rect, -x_offset, -y_offset);
|
|
|
|
CL_UpdateLayerRect(context->compositor, layer, &rect, PR_FALSE);
|
|
|
|
return;
|
|
}
|
|
else
|
|
/* For layers, we don't ever need to clear the background */
|
|
FE_DisplaySubtext(context, FE_VIEW, text, start_pos, end_pos,
|
|
FALSE);
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayText(MWContext *context,
|
|
LO_TextStruct *text, Bool need_bg)
|
|
{
|
|
int32 p1, p2;
|
|
LO_TextAttr tmp_attr;
|
|
LO_TextAttr *hold_attr;
|
|
|
|
TIMING_STOPCLOCK_OBJECT("layout:blank-screen", context, "displaying text");
|
|
|
|
/* Blinking text elements are placed in a separate layer */
|
|
if (context->compositor && text->text_attr->attrmask & LO_ATTR_BLINK)
|
|
{
|
|
if (! (text->ele_attrmask & LO_ELE_DRAWN))
|
|
{
|
|
/* XXX - If the BLINK is in a layer, we need to create the
|
|
blink layer as a child of that layer, not as a child of
|
|
the _BODY layer. */
|
|
CL_Layer *body_layer =
|
|
CL_FindLayer(context->compositor, LO_BODY_LAYER_NAME);
|
|
lo_CreateBlinkLayer(context, text, body_layer);
|
|
text->ele_attrmask |= LO_ELE_DRAWN;
|
|
}
|
|
|
|
/* All blink text drawing is handled by the blink layer's
|
|
painter function. Don't do any drawing here. */
|
|
return;
|
|
}
|
|
|
|
if (text->ele_attrmask & LO_ELE_SELECTED)
|
|
{
|
|
lo_NormalizeStartAndEndOfSelection(text, &p1, &p2);
|
|
|
|
if (p1 > 0)
|
|
{
|
|
/* For layers we don't ever need to clear the background */
|
|
FE_DisplaySubtext(context, FE_VIEW, text,
|
|
0, (p1 - 1), FALSE);
|
|
}
|
|
|
|
lo_CopyTextAttr(text->text_attr, &tmp_attr);
|
|
tmp_attr.fg.red = text->text_attr->bg.red;
|
|
tmp_attr.fg.green = text->text_attr->bg.green;
|
|
tmp_attr.fg.blue = text->text_attr->bg.blue;
|
|
tmp_attr.bg.red = text->text_attr->fg.red;
|
|
tmp_attr.bg.green = text->text_attr->fg.green;
|
|
tmp_attr.bg.blue = text->text_attr->fg.blue;
|
|
|
|
/* lo_CopyTextAttr doesn't copy the FE_Data. In
|
|
* this case, however, we need to copy the FE_Data
|
|
* because otherwise the front end will
|
|
* synthesize a new copy. This assumes that
|
|
* the FE_Data doesn't care about font color.
|
|
*/
|
|
tmp_attr.FE_Data = text->text_attr->FE_Data;
|
|
|
|
hold_attr = text->text_attr;
|
|
text->text_attr = &tmp_attr;
|
|
/* For layers we don't ever need to clear the background */
|
|
FE_DisplaySubtext(context, FE_VIEW, text, p1, p2,
|
|
FALSE);
|
|
text->text_attr = hold_attr;
|
|
|
|
if (p2 < (text->text_len - 1))
|
|
{
|
|
/* For layers we don't ever need to clear the background */
|
|
FE_DisplaySubtext(context, FE_VIEW, text,
|
|
(p2 + 1), (text->text_len - 1),
|
|
FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* For layers we don't ever need to clear the background */
|
|
FE_DisplayText(context, FE_VIEW, text, FALSE);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayEmbed(MWContext *context, LO_EmbedStruct *embed)
|
|
{
|
|
CL_Layer *layer;
|
|
|
|
if (! context->compositor) {
|
|
FE_DisplayEmbed(context, FE_VIEW, embed);
|
|
return;
|
|
}
|
|
|
|
/* Don't ever display hidden embeds */
|
|
if (embed->objTag.ele_attrmask & LO_ELE_HIDDEN)
|
|
return;
|
|
|
|
layer = embed->objTag.layer;
|
|
XP_ASSERT(layer);
|
|
if (! layer) /* Paranoia */
|
|
return;
|
|
|
|
if (!(embed->objTag.ele_attrmask & LO_ELE_DRAWN)) {
|
|
/* Move layer to new position */
|
|
CL_MoveLayer(layer,
|
|
embed->objTag.x + embed->objTag.x_offset,
|
|
embed->objTag.y + embed->objTag.y_offset);
|
|
CL_SetLayerHidden(layer, PR_FALSE);
|
|
embed->objTag.ele_attrmask |= LO_ELE_DRAWN;
|
|
}
|
|
}
|
|
|
|
void
|
|
lo_DisplayBuiltin(MWContext *context, LO_BuiltinStruct *builtin)
|
|
{
|
|
CL_Layer *layer;
|
|
|
|
/* need to deal with layers here XXX */
|
|
|
|
if (! context->compositor) {
|
|
FE_DisplayBuiltin(context, FE_VIEW, builtin);
|
|
return;
|
|
}
|
|
|
|
layer = builtin->layer;
|
|
XP_ASSERT(layer);
|
|
if (! layer) /* Paranoia */
|
|
return;
|
|
|
|
if (!(builtin->ele_attrmask & LO_ELE_DRAWN)) {
|
|
/* Move layer to new position */
|
|
CL_MoveLayer(layer,
|
|
builtin->x + builtin->x_offset,
|
|
builtin->y + builtin->y_offset);
|
|
CL_SetLayerHidden(layer, PR_FALSE);
|
|
builtin->ele_attrmask |= LO_ELE_DRAWN;
|
|
}
|
|
}
|
|
|
|
#ifdef JAVA
|
|
void
|
|
lo_DisplayJavaApp(MWContext *context, LO_JavaAppStruct *java_app)
|
|
{
|
|
CL_Layer *layer;
|
|
|
|
if (! context->compositor) {
|
|
FE_DisplayJavaApp(context, FE_VIEW, java_app);
|
|
return;
|
|
}
|
|
|
|
layer = java_app->objTag.layer;
|
|
XP_ASSERT(layer);
|
|
if (! layer) /* Paranoia */
|
|
return;
|
|
|
|
if (!(java_app->objTag.ele_attrmask & LO_ELE_DRAWN)) {
|
|
/* Move layer to new position */
|
|
CL_MoveLayer(layer,
|
|
java_app->objTag.x + java_app->objTag.x_offset,
|
|
java_app->objTag.y + java_app->objTag.y_offset);
|
|
|
|
/* Now that layer is layed out to its final position,
|
|
we can display it */
|
|
CL_SetLayerHidden(layer, PR_FALSE);
|
|
java_app->objTag.ele_attrmask |= LO_ELE_DRAWN;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* Make the image layer visible and draw the image border. Should only be called
|
|
if there is a compositor. */
|
|
static void
|
|
lo_DisplayImage(MWContext *context, LO_ImageStruct *image)
|
|
{
|
|
int bw = image->border_width;
|
|
|
|
XP_ASSERT(context->compositor);
|
|
|
|
/* Allow the compositor to start drawing the image layer. */
|
|
lo_ActivateImageLayer(context, image);
|
|
|
|
/* The image border, if present, draws in the parent of the image layer. */
|
|
if (bw) {
|
|
int x = image->x + image->x_offset + bw;
|
|
int y = image->y + image->y_offset + bw;
|
|
|
|
FE_DisplayBorder(context, FE_VIEW, x - bw, y - bw,
|
|
image->width + 2 * bw, image->height + 2 * bw, bw,
|
|
&image->text_attr->fg, LO_SOLID);
|
|
}
|
|
}
|
|
|
|
void
|
|
lo_DisplayImageWithoutCompositor(MWContext *context, LO_ImageStruct *image)
|
|
{
|
|
int bw = image->border_width;
|
|
int x = image->x + image->x_offset + bw;
|
|
int y = image->y + image->y_offset + bw;
|
|
int width = image->width;
|
|
int height = image->height;
|
|
|
|
/* This routine should only be called in the absence of a compositor. */
|
|
XP_ASSERT(!context->compositor);
|
|
|
|
/* Handle the TextFE. */
|
|
if (context->type == MWContextText) {
|
|
XL_DisplayTextImage(context, FE_VIEW, image);
|
|
return;
|
|
}
|
|
|
|
if (image->is_icon) {
|
|
IL_DisplayIcon(context->img_cx, image->icon_number, x, y);
|
|
}
|
|
else {
|
|
int cnv_x = context->convertPixX;
|
|
int cnv_y = context->convertPixY;
|
|
|
|
IL_DisplaySubImage(image->image_req, x/cnv_x, y/cnv_y, 0, 0,
|
|
width/cnv_x, height/cnv_y);
|
|
}
|
|
|
|
if (bw)
|
|
FE_DisplayBorder(context, FE_VIEW, x - bw, y - bw, width + 2 * bw,
|
|
height + 2 * bw, bw, &image->text_attr->fg, LO_SOLID);
|
|
}
|
|
|
|
void
|
|
lo_DisplaySubImageWithoutCompositor(MWContext *context, LO_ImageStruct *image,
|
|
int32 x, int32 y, uint32 width, uint32 height)
|
|
{
|
|
int bw = image->border_width;
|
|
int x_pos = image->x + image->x_offset + bw;
|
|
int y_pos = image->y + image->y_offset + bw;
|
|
int sub_x, sub_y, sub_w, sub_h;
|
|
|
|
/* This routine should only be called in the absence of a compositor. */
|
|
XP_ASSERT(!context->compositor);
|
|
|
|
/* Handle the TextFE. */
|
|
if (context->type == MWContextText) {
|
|
XL_DisplayTextImage(context, FE_VIEW, image);
|
|
return;
|
|
}
|
|
|
|
if (x < x_pos) {
|
|
sub_x = x_pos;
|
|
sub_w = width + x - x_pos;
|
|
}
|
|
else {
|
|
sub_x = x;
|
|
sub_w = width;
|
|
}
|
|
|
|
if (y < y_pos) {
|
|
sub_y = y_pos;
|
|
sub_h = height + y - y_pos;
|
|
}
|
|
else {
|
|
sub_y = y;
|
|
sub_h = height;
|
|
}
|
|
|
|
if (x + width > x_pos + image->width)
|
|
sub_w += (x_pos + image->width - x - width);
|
|
|
|
if (y + height > y_pos + image->height)
|
|
sub_h += (y_pos + image->height - y - height);
|
|
|
|
if (sub_w > 0 && sub_h > 0) {
|
|
if (image->is_icon) {
|
|
IL_DisplayIcon(context->img_cx, image->icon_number,
|
|
x_pos, y_pos);
|
|
}
|
|
else {
|
|
int cnv_x = context->convertPixX;
|
|
int cnv_y = context->convertPixY;
|
|
|
|
IL_DisplaySubImage(image->image_req, x_pos/cnv_x, y_pos/cnv_y,
|
|
(sub_x - x_pos)/cnv_x, (sub_y - y_pos)/cnv_y,
|
|
sub_w/cnv_x, sub_h/cnv_y);
|
|
}
|
|
}
|
|
|
|
if (bw)
|
|
FE_DisplayBorder(context, FE_VIEW, x - bw, y - bw, width + 2 * bw,
|
|
height + 2 * bw, bw, &image->text_attr->fg,
|
|
LO_SOLID);
|
|
}
|
|
|
|
/* Should only be called in the absence of a compositor. */
|
|
void
|
|
lo_ClipImage(MWContext *context, LO_ImageStruct *image,
|
|
int32 x, int32 y, uint32 width, uint32 height)
|
|
{
|
|
int32 sub_x, sub_y;
|
|
uint32 sub_w, sub_h;
|
|
|
|
/*
|
|
* If the two don't overlap, do nothing.
|
|
*/
|
|
if (((int32)(image->x + image->x_offset + image->width +
|
|
2*image->border_width) < x)||
|
|
((int32)(x + width) < (int32)(image->x + image->x_offset))||
|
|
((int32)(image->y + image->y_offset + image->height +
|
|
2*image->border_width) < y)||
|
|
((int32)(y + height) < (int32)(image->y + image->y_offset)))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ((image->x + image->x_offset) >= x)
|
|
{
|
|
sub_x = image->x + image->x_offset;
|
|
}
|
|
else
|
|
{
|
|
sub_x = x;
|
|
}
|
|
|
|
if ((int32)(image->x + image->x_offset + image->width + 2*image->border_width)
|
|
<= (int32)(x + width))
|
|
{
|
|
sub_w = image->x + image->x_offset + image->width +
|
|
2*image->border_width - sub_x;
|
|
}
|
|
else
|
|
{
|
|
sub_w = x + width - sub_x;
|
|
}
|
|
|
|
if ((image->y + image->y_offset) >= y)
|
|
{
|
|
sub_y = image->y + image->y_offset;
|
|
}
|
|
else
|
|
{
|
|
sub_y = y;
|
|
}
|
|
|
|
if ((int32)(image->y + image->y_offset + image->height + 2*image->border_width)
|
|
<= (int32)(y + height))
|
|
{
|
|
sub_h = image->y + image->y_offset + image->height +
|
|
2*image->border_width - sub_y;
|
|
}
|
|
else
|
|
{
|
|
sub_h = y + height - sub_y;
|
|
}
|
|
|
|
if (((int32)sub_w >= (int32)(image->width - 2))&&
|
|
((int32)sub_h >= (int32)(image->height - 2)))
|
|
{
|
|
lo_DisplayImageWithoutCompositor(context, (LO_ImageStruct *)image);
|
|
}
|
|
else
|
|
{
|
|
lo_DisplaySubImageWithoutCompositor(context, (LO_ImageStruct *)image,
|
|
sub_x, sub_y, sub_w, sub_h);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayEdge(MWContext *context, LO_EdgeStruct *edge)
|
|
{
|
|
if (edge->visible != FALSE)
|
|
{
|
|
FE_DisplayEdge(context, FE_VIEW, edge);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayTable(MWContext *context, LO_TableStruct *table)
|
|
{
|
|
FE_DisplayTable(context, FE_VIEW, table);
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplaySubDoc(MWContext *context, LO_SubDocStruct *subdoc)
|
|
{
|
|
FE_DisplaySubDoc(context, FE_VIEW, subdoc);
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayCell(MWContext *context, LO_CellStruct *cell)
|
|
{
|
|
/* If this cell is empty, bail */
|
|
if (cell->cell_list == NULL && cell->cell_float_list == NULL)
|
|
return;
|
|
|
|
if (context->compositor && cell->cell_bg_layer)
|
|
CL_SetLayerHidden(cell->cell_bg_layer, PR_FALSE);
|
|
|
|
FE_DisplayCell(context, FE_VIEW, cell);
|
|
}
|
|
|
|
void
|
|
lo_DisplayLineFeed(MWContext *context,
|
|
LO_LinefeedStruct *lfeed, Bool need_bg)
|
|
{
|
|
LO_TextAttr tmp_attr;
|
|
LO_TextAttr *hold_attr;
|
|
|
|
if (lfeed->ele_attrmask & LO_ELE_SELECTED)
|
|
{
|
|
lo_CopyTextAttr(lfeed->text_attr, &tmp_attr);
|
|
tmp_attr.fg.red = lfeed->text_attr->bg.red;
|
|
tmp_attr.fg.green = lfeed->text_attr->bg.green;
|
|
tmp_attr.fg.blue = lfeed->text_attr->bg.blue;
|
|
tmp_attr.bg.red = lfeed->text_attr->fg.red;
|
|
tmp_attr.bg.green = lfeed->text_attr->fg.green;
|
|
tmp_attr.bg.blue = lfeed->text_attr->fg.blue;
|
|
hold_attr = lfeed->text_attr;
|
|
lfeed->text_attr = &tmp_attr;
|
|
FE_DisplayLineFeed(context, FE_VIEW, lfeed, (XP_Bool)need_bg);
|
|
lfeed->text_attr = hold_attr;
|
|
}
|
|
else
|
|
{
|
|
FE_DisplayLineFeed(context, FE_VIEW, lfeed, (XP_Bool)need_bg);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayHR(MWContext *context, LO_HorizRuleStruct *hrule)
|
|
{
|
|
FE_DisplayHR(context, FE_VIEW, hrule);
|
|
}
|
|
|
|
|
|
void
|
|
lo_DisplayBullet(MWContext *context, LO_BulletStruct *bullet)
|
|
{
|
|
FE_DisplayBullet(context, FE_VIEW, bullet);
|
|
}
|
|
|
|
void
|
|
lo_DisplayFormElement(MWContext *context, LO_FormElementStruct *form_element)
|
|
{
|
|
CL_Layer *layer;
|
|
|
|
if (! context->compositor) {
|
|
FE_DisplayFormElement(context, FE_VIEW, form_element);
|
|
return;
|
|
}
|
|
|
|
layer = form_element->layer;
|
|
XP_ASSERT(layer);
|
|
if (! layer) /* Paranoia */
|
|
return;
|
|
|
|
if (!(form_element->ele_attrmask & LO_ELE_DRAWN)) {
|
|
/* Move layer to new position */
|
|
CL_MoveLayer(layer,
|
|
form_element->x + form_element->x_offset +
|
|
form_element->border_horiz_space,
|
|
form_element->y + form_element->y_offset +
|
|
form_element->border_vert_space);
|
|
CL_SetLayerHidden(layer, PR_FALSE);
|
|
form_element->ele_attrmask |= LO_ELE_DRAWN;
|
|
}
|
|
}
|
|
|
|
void
|
|
lo_DisplayElement(MWContext *context, LO_Element *tptr,
|
|
int32 base_x, int32 base_y,
|
|
int32 x, int32 y,
|
|
uint32 width, uint32 height)
|
|
{
|
|
LO_Any *any = &tptr->lo_any;
|
|
XP_Rect bbox;
|
|
|
|
lo_GetElementBbox(tptr, &bbox);
|
|
XP_OffsetRect(&bbox, base_x, base_y);
|
|
|
|
if (bbox.top >= (int32)(y + height))
|
|
return;
|
|
|
|
if (bbox.bottom <= y )
|
|
return;
|
|
|
|
if (bbox.left >= (int32)(x + width))
|
|
return;
|
|
|
|
if (bbox.right <= x )
|
|
return;
|
|
|
|
/* Temporarily translate to new coordinate system */
|
|
any->x += base_x;
|
|
any->y += base_y;
|
|
|
|
switch (tptr->type)
|
|
{
|
|
case LO_TEXT:
|
|
if (tptr->lo_text.text != NULL)
|
|
lo_DisplayText(context, (LO_TextStruct *)tptr, FALSE);
|
|
break;
|
|
|
|
case LO_LINEFEED:
|
|
lo_DisplayLineFeed(context, (LO_LinefeedStruct *)tptr, FALSE);
|
|
break;
|
|
|
|
case LO_HRULE:
|
|
lo_DisplayHR(context, (LO_HorizRuleStruct *)tptr);
|
|
break;
|
|
|
|
case LO_FORM_ELE:
|
|
lo_DisplayFormElement(context, (LO_FormElementStruct *)tptr);
|
|
break;
|
|
|
|
case LO_BULLET:
|
|
lo_DisplayBullet(context, (LO_BulletStruct *)tptr);
|
|
break;
|
|
|
|
case LO_IMAGE:
|
|
if (context->compositor) {
|
|
/* Allow the compositor to start drawing the image. */
|
|
lo_DisplayImage(context, (LO_ImageStruct *)tptr);
|
|
}
|
|
else {
|
|
/* There is no compositor, so draw the image directly, with the
|
|
appropriate clip. */
|
|
lo_ClipImage(context,
|
|
(LO_ImageStruct *)tptr,
|
|
(x + base_x),
|
|
(y + base_y), width, height);
|
|
}
|
|
break;
|
|
|
|
case LO_TABLE:
|
|
lo_DisplayTable(context, (LO_TableStruct *)tptr);
|
|
break;
|
|
|
|
case LO_EMBED:
|
|
lo_DisplayEmbed(context, (LO_EmbedStruct *)tptr);
|
|
break;
|
|
|
|
case LO_BUILTIN:
|
|
lo_DisplayBuiltin(context, (LO_BuiltinStruct *)tptr);
|
|
break;
|
|
|
|
#ifdef JAVA
|
|
case LO_JAVA:
|
|
lo_DisplayJavaApp(context, (LO_JavaAppStruct *)tptr);
|
|
break;
|
|
#endif
|
|
|
|
case LO_EDGE:
|
|
lo_DisplayEdge(context, (LO_EdgeStruct *)tptr);
|
|
break;
|
|
|
|
case LO_CELL:
|
|
/* If this cell is a container for an inflow layer (and
|
|
therefore not a table cell), don't descend into the cell
|
|
because the layer's painter function will handle its
|
|
display. */
|
|
if (((LO_CellStruct*)tptr)->cell_inflow_layer)
|
|
break;
|
|
|
|
/* cmanske: Order changed - display cell contents FIRST
|
|
* so selection feedback near cell border is not wiped out
|
|
* TODO: This doesn't work for images, which are display
|
|
* asynchronously. Need to add "observer" to image display
|
|
*/
|
|
lo_DisplayCellContents(context, (LO_CellStruct *)tptr,
|
|
base_x, base_y, x, y, width, height);
|
|
lo_DisplayCell(context, (LO_CellStruct *)tptr);
|
|
break;
|
|
|
|
case LO_SUBDOC:
|
|
{
|
|
LO_SubDocStruct *subdoc;
|
|
int32 new_x, new_y;
|
|
uint32 new_width, new_height;
|
|
lo_DocState *sub_state;
|
|
|
|
subdoc = (LO_SubDocStruct *)tptr;
|
|
sub_state = (lo_DocState *)subdoc->state;
|
|
|
|
if (sub_state == NULL)
|
|
{
|
|
break;
|
|
}
|
|
|
|
lo_DisplaySubDoc(context, subdoc);
|
|
|
|
new_x = subdoc->x;
|
|
new_y = subdoc->y;
|
|
new_width = subdoc->x_offset + subdoc->width;
|
|
new_height = subdoc->y_offset + subdoc->height;
|
|
|
|
new_x = new_x - subdoc->x;
|
|
new_y = new_y - subdoc->y;
|
|
sub_state->base_x = subdoc->x +
|
|
subdoc->x_offset + subdoc->border_width;
|
|
sub_state->base_y = subdoc->y +
|
|
subdoc->y_offset + subdoc->border_width;
|
|
lo_RefreshDocumentArea(context, sub_state,
|
|
new_x, new_y, new_width, new_height);
|
|
}
|
|
break;
|
|
|
|
case LO_PARAGRAPH:
|
|
case LO_CENTER:
|
|
case LO_MULTICOLUMN:
|
|
case LO_LIST:
|
|
case LO_DESCTITLE:
|
|
case LO_DESCTEXT:
|
|
case LO_BLOCKQUOTE:
|
|
case LO_HEADING:
|
|
case LO_SPAN:
|
|
/* all non-display, doc-state mutating elements. */
|
|
break;
|
|
case LO_TEXTBLOCK:
|
|
break;
|
|
case LO_FLOAT:
|
|
case LO_LAYER:
|
|
case LO_SPACER:
|
|
break;
|
|
default:
|
|
XP_TRACE(("lo_DisplayElement(%p) skip element type = %d\n", tptr, tptr->type));
|
|
break;
|
|
}
|
|
|
|
/* Restore original element's coordinates */
|
|
any->x -= base_x;
|
|
any->y -= base_y;
|
|
}
|
|
|
|
#ifdef PROFILE
|
|
#pragma profile off
|
|
#endif
|
|
|