diff options
Diffstat (limited to 'khtml/rendering/render_inline.cpp')
-rw-r--r-- | khtml/rendering/render_inline.cpp | 935 |
1 files changed, 0 insertions, 935 deletions
diff --git a/khtml/rendering/render_inline.cpp b/khtml/rendering/render_inline.cpp deleted file mode 100644 index c3eb08d4b..000000000 --- a/khtml/rendering/render_inline.cpp +++ /dev/null @@ -1,935 +0,0 @@ -/* - * This file is part of the render object implementation for KHTML. - * - * Copyright (C) 1999-2003 Lars Knoll ([email protected]) - * (C) 1999-2003 Antti Koivisto ([email protected]) - * (C) 2002-2003 Dirk Mueller ([email protected]) - * (C) 2003 Apple Computer, Inc. - * - * This library is free software; you can redistribute it and/or - * modify it under the terms of the GNU Library General Public - * License as published by the Free Software Foundation; either - * version 2 of the License, or (at your option) any later version. - * - * This library 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 - * Library General Public License for more details. - * - * You should have received a copy of the GNU Library General Public License - * along with this library; see the file COPYING.LIB. If not, write to - * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, - * Boston, MA 02110-1301, USA. - * - */ - -#include <kglobal.h> - -#include "rendering/render_arena.h" -#include "rendering/render_inline.h" -#include "rendering/render_block.h" -#include "xml/dom_docimpl.h" - -#include <tqvaluevector.h> - -using namespace khtml; - -void RenderInline::setStyle(RenderStyle* _style) -{ - RenderFlow::setStyle(_style); - setInline(true); - - // Ensure that all of the split inlines pick up the new style. We - // only do this if we're an inline, since we don't want to propagate - // a block's style to the other inlines. - // e.g., <font>foo <h4>goo</h4> moo</font>. The <font> inlines before - // and after the block share the same style, but the block doesn't - // need to pass its style on to anyone else. - RenderFlow* currCont = continuation(); - while (currCont) { - if (currCont->isInline()) { - RenderFlow* nextCont = currCont->continuation(); - currCont->setContinuation(0); - currCont->setStyle(style()); - currCont->setContinuation(nextCont); - } - currCont = currCont->continuation(); - } - - if (attached()) { - // Update replaced content - updateReplacedContent(); - // Update pseudos for ::before and ::after - updatePseudoChildren(); - } -} - -// Attach handles initial setStyle that requires parent nodes -void RenderInline::attach() -{ - RenderFlow::attach(); - - updateReplacedContent(); - updatePseudoChildren(); -} - -bool RenderInline::isInlineContinuation() const -{ - return m_isContinuation; -} - -void RenderInline::addChildToFlow(RenderObject* newChild, RenderObject* beforeChild) -{ - // Make sure we don't append things after :after-generated content if we have it. - if (!beforeChild && lastChild() && lastChild()->style()->styleType() == RenderStyle::AFTER) - beforeChild = lastChild(); - - if (!newChild->isText() && newChild->style()->position() != STATIC) - setOverhangingContents(); - - if (!newChild->isInline() && !newChild->isFloatingOrPositioned() ) - { - // We are placing a block inside an inline. We have to perform a split of this - // inline into continuations. This involves creating an anonymous block box to hold - // |newChild|. We then make that block box a continuation of this inline. We take all of - // the children after |beforeChild| and put them in a clone of this object. - - RenderBlock *newBox = createAnonymousBlock(); - RenderFlow* oldContinuation = continuation(); - setContinuation(newBox); - - splitFlow(beforeChild, newBox, newChild, oldContinuation); - return; - } - - RenderBox::addChild(newChild,beforeChild); - - newChild->setNeedsLayoutAndMinMaxRecalc(); -} - -RenderInline* RenderInline::cloneInline(RenderFlow* src) -{ - RenderInline *o = new (src->renderArena()) RenderInline(src->element()); - o->m_isContinuation = true; - o->setStyle(src->style()); - return o; -} - -void RenderInline::splitInlines(RenderBlock* fromBlock, RenderBlock* toBlock, - RenderBlock* middleBlock, - RenderObject* beforeChild, RenderFlow* oldCont) -{ - // Create a clone of this inline. - RenderInline* clone = cloneInline(this); - clone->setContinuation(oldCont); - - // Now take all of the children from beforeChild to the end and remove - // then from |this| and place them in the clone. - RenderObject* o = beforeChild; - while (o) { - RenderObject* tmp = o; - o = tmp->nextSibling(); - clone->addChildToFlow(removeChildNode(tmp), 0); - tmp->setNeedsLayoutAndMinMaxRecalc(); - } - - // Hook |clone| up as the continuation of the middle block. - middleBlock->setContinuation(clone); - - // We have been reparented and are now under the fromBlock. We need - // to walk up our inline parent chain until we hit the containing block. - // Once we hit the containing block we're done. - RenderFlow* curr = static_cast<RenderFlow*>(parent()); - RenderFlow* currChild = this; - while (curr && curr != fromBlock) { - // Create a new clone. - RenderInline* cloneChild = clone; - clone = cloneInline(curr); - - // Insert our child clone as the first child. - clone->addChildToFlow(cloneChild, 0); - - // Hook the clone up as a continuation of |curr|. - RenderFlow* oldCont = curr->continuation(); - curr->setContinuation(clone); - clone->setContinuation(oldCont); - - // Now we need to take all of the children starting from the first child - // *after* currChild and append them all to the clone. - o = currChild->nextSibling(); - while (o) { - RenderObject* tmp = o; - o = tmp->nextSibling(); - clone->appendChildNode(curr->removeChildNode(tmp)); - tmp->setNeedsLayoutAndMinMaxRecalc(); - } - - // Keep walking up the chain. - currChild = curr; - curr = static_cast<RenderFlow*>(curr->parent()); - } - - // Now we are at the block level. We need to put the clone into the toBlock. - toBlock->appendChildNode(clone); - - // Now take all the children after currChild and remove them from the fromBlock - // and put them in the toBlock. - o = currChild->nextSibling(); - while (o) { - RenderObject* tmp = o; - o = tmp->nextSibling(); - toBlock->appendChildNode(fromBlock->removeChildNode(tmp)); - } -} - -void RenderInline::splitFlow(RenderObject* beforeChild, RenderBlock* newBlockBox, - RenderObject* newChild, RenderFlow* oldCont) -{ - RenderBlock* pre = 0; - RenderBlock* block = containingBlock(); - bool madeNewBeforeBlock = false; - if (block->isAnonymousBlock()) { - // We can reuse this block and make it the preBlock of the next continuation. - pre = block; - block = block->containingBlock(); - } - else { - // No anonymous block available for use. Make one. - pre = block->createAnonymousBlock(); - madeNewBeforeBlock = true; - } - - RenderBlock* post = block->createAnonymousBlock(); - - RenderObject* boxFirst = madeNewBeforeBlock ? block->firstChild() : pre->nextSibling(); - if (madeNewBeforeBlock) - block->insertChildNode(pre, boxFirst); - block->insertChildNode(newBlockBox, boxFirst); - block->insertChildNode(post, boxFirst); - block->setChildrenInline(false); - - if (madeNewBeforeBlock) { - RenderObject* o = boxFirst; - while (o) - { - RenderObject* no = o; - o = no->nextSibling(); - pre->appendChildNode(block->removeChildNode(no)); - no->setNeedsLayoutAndMinMaxRecalc(); - } - } - - splitInlines(pre, post, newBlockBox, beforeChild, oldCont); - - // We already know the newBlockBox isn't going to contain inline kids, so avoid wasting - // time in makeChildrenNonInline by just setting this explicitly up front. - newBlockBox->setChildrenInline(false); - - // We don't just call addChild, since it would pass things off to the - // continuation, so we call addChildToFlow explicitly instead. We delayed - // adding the newChild until now so that the |newBlockBox| would be fully - // connected, thus allowing newChild access to a renderArena should it need - // to wrap itself in additional boxes (e.g., table construction). - newBlockBox->addChildToFlow(newChild, 0); - - // XXXdwh is any of this even necessary? I don't think it is. - pre->close(); - pre->setPos(0, -500000); - pre->setNeedsLayout(true); - newBlockBox->close(); - newBlockBox->setPos(0, -500000); - newBlockBox->setNeedsLayout(true); - post->close(); - post->setPos(0, -500000); - post->setNeedsLayout(true); - - updatePseudoChildren(); - - block->setNeedsLayoutAndMinMaxRecalc(); -} - -void RenderInline::paint(PaintInfo& i, int _tx, int _ty) -{ - paintLines(i, _tx, _ty); -} - -/** - * Appends the given coordinate-pair to the point-array if it is not - * equal to the last element. - * @param pointArray point-array - * @param pnt point to append - * @return \c true if \c pnt has actually been appended - */ -inline static bool appendIfNew(TQValueVector<TQPoint> &pointArray, const TQPoint &pnt) -{ -// if (!pointArray.isEmpty()) kdDebug(6040) << "appifnew: " << pointArray.back() << " == " << pnt << ": " << (pointArray.back() == pnt) << endl; -// else kdDebug(6040) << "appifnew: " << pnt << " (unconditional)" << endl; - if (!pointArray.isEmpty() && pointArray.back() == pnt) return false; - pointArray.append(pnt); - return true; -} - -/** - * Does spike-reduction on the given point-array's stack-top. - * - * Spikes are path segments of which one goes forward, and the sucessor - * goes backward on the predecessor's segment: - * - * 2 0 1 - * x------x<-----x - * (0 is stack-top in point-array) - * - * This will be reduced to - * 1 0 - * x------x - * - * Preconditions: - * - No other spikes exist in the whole point-array except at most - * one at the end - * - No two succeeding points are ever equal - * - For each two succeeding points either p1.x == p2.x or p1.y == p2.y holds - * true - * - No such spike exists where 2 is situated between 0 and 1. - * - * Postcondition: - * - No spikes exist in the whole point-array - * - * If no spike is found, the point-array is left unchanged. - * @return \c true if an actual reduction was done - */ -inline static bool reduceSpike(TQValueVector<TQPoint> &pointArray) -{ - if (pointArray.size() < 3) return false; - TQValueVector<TQPoint>::Iterator it = pointArray.end(); - TQPoint p0 = *--it; - TQPoint p1 = *--it; - TQPoint p2 = *--it; - - bool elide = false; - - if (p0.x() == p1.x() && p1.x() == p2.x() - && (p1.y() < p0.y() && p0.y() < p2.y() - || p2.y() < p0.y() && p0.y() < p1.y() - || p1.y() < p2.y() && p2.y() < p0.y() - || p0.y() < p2.y() && p2.y() < p1.y() - || (elide = p2.y() == p0.y() && p0.y() < p1.y()) - || (elide = p1.y() < p0.y() && p0.y() == p2.y())) - || p0.y() == p1.y() && p1.y() == p2.y() - && (p1.x() < p0.x() && p0.x() < p2.x() - || p2.x() < p0.x() && p0.x() < p1.x() - || p1.x() < p2.x() && p2.x() < p0.x() - || p0.x() < p2.x() && p2.x() < p1.x() - || (elide = p2.x() == p0.x() && p0.x() < p1.x()) - || (elide = p1.x() < p0.x() && p0.x() == p2.x()))) - { -// kdDebug(6040) << "spikered p2" << (elide ? " (elide)" : "") << ": " << p2 << " p1: " << p1 << " p0: " << p0 << endl; - pointArray.pop_back(); pointArray.pop_back(); - if (!elide) - pointArray.push_back(p0); - return true; - } - return false; -} - -/** - * Reduces segment separators. - * - * A segment separator separates a segment into two segments, thus causing - * two adjacent segment with the same orientation. - * - * 2 1 0 - * x-------x---->x - * (0 means stack-top) - * - * Here, 1 is a segment separator. As segment separators not only make - * the line drawing algorithm inefficient, but also make the spike-reduction - * fail, they must be eliminated: - * - * 1 0 - * x------------>x - * - * Preconditions: - * - No other segment separators exist in the whole point-array except - * at most one at the end - * - No two succeeding points are ever equal - * - For each two succeeding points either p1.x == p2.x or p1.y == p2.y holds - * true - * - No such spike exists where 2 is situated between 0 and 1. - * - * Postcondition: - * - No segment separators exist in the whole point-array - * - * If no segment separator is found at the end of the point-array, it is - * left unchanged. - * @return \c true if a segment separator was actually reduced. - */ -inline static bool reduceSegmentSeparator(TQValueVector<TQPoint> &pointArray) -{ - if (pointArray.size() < 3) return false; - TQValueVector<TQPoint>::Iterator it = pointArray.end(); - TQPoint p0 = *--it; - TQPoint p1 = *--it; - TQPoint p2 = *--it; -// kdDebug(6040) << "checking p2: " << p2 << " p1: " << p1 << " p0: " << p0 << endl; - - if (p0.x() == p1.x() && p1.x() == p2.x() - && (p2.y() < p1.y() && p1.y() < p0.y() - || p0.y() < p1.y() && p1.y() < p2.y()) - || p0.y() == p1.y() && p1.y() == p2.y() - && (p2.x() < p1.x() && p1.x() < p0.x() - || p0.x() < p1.x() && p1.x() < p2.x())) - { -// kdDebug(6040) << "segred p2: " << p2 << " p1: " << p1 << " p0: " << p0 << endl; - pointArray.pop_back(); pointArray.pop_back(); - pointArray.push_back(p0); - return true; - } - return false; -} - -/** - * Appends the given point to the point-array, doing necessary reductions to - * produce a path without spikes and segment separators. - */ -static void appendPoint(TQValueVector<TQPoint> &pointArray, TQPoint &pnt) -{ - if (!appendIfNew(pointArray, pnt)) return; -// kdDebug(6040) << "appendPoint: appended " << pnt << endl; - reduceSegmentSeparator(pointArray) - || reduceSpike(pointArray); -} - -/** - * Traverses the horizontal inline boxes and appends the point coordinates to - * the given array. - * @param box inline box - * @param pointArray array collecting coordinates - * @param bottom \c true, collect bottom coordinates, \c false, collect top - * coordinates. - * @param limit lower limit that an y-coordinate must at least reach. Note - * that limit designates the highest y-coordinate for \c bottom, and - * the lowest for !\c bottom. - */ -static void collectHorizontalBoxCoordinates(InlineBox *box, - TQValueVector<TQPoint> &pointArray, - bool bottom, int offset, int limit = -500000) -{ -// kdDebug(6000) << "collectHorizontalBoxCoordinates: " << endl; - offset = bottom ? offset:-offset; - int y = box->yPos() + bottom*box->height() + offset; - if (limit != -500000 && (bottom ? y < limit : y > limit)) - y = limit; - int x = box->xPos() + bottom*box->width() + offset; - TQPoint newPnt(x, y); - // Add intersection point if point-array not empty. - if (!pointArray.isEmpty()) { - TQPoint lastPnt = pointArray.back(); - TQPoint insPnt(newPnt.x(), lastPnt.y()); - if (offset && ((bottom && lastPnt.y() > y) || (!bottom && lastPnt.y() < y))) { - insPnt.rx() = lastPnt.x(); - insPnt.ry() = y; - } -// kdDebug(6040) << "left: " << lastPnt << " == " << insPnt << ": " << (insPnt == lastPnt) << endl; - appendPoint(pointArray, insPnt); - } - // Insert starting point of box - appendPoint(pointArray, newPnt); - - newPnt.rx() += (bottom ? -box->width() : box->width()) - 2*offset; - - if (box->isInlineFlowBox()) { - InlineFlowBox *flowBox = static_cast<InlineFlowBox *>(box); - for (InlineBox *b = bottom ? flowBox->lastChild() : flowBox->firstChild(); b; b = bottom ? b->prevOnLine() : b->nextOnLine()) { - // Don't let boxes smaller than this flow box' height influence - // the vertical position of the outline if they have a different - // x-coordinate - int l2; - if (b->xPos() != box->xPos() && b->xPos() + b->width() != box->xPos() + box->width()) - l2 = y; - else - l2 = limit; - collectHorizontalBoxCoordinates(b, pointArray, bottom, kAbs(offset), l2); - } - - // Add intersection point if flow box contained any children - if (flowBox->firstChild()) { - TQPoint lastPnt = pointArray.back(); - TQPoint insPnt(lastPnt.x(), newPnt.y()); -// kdDebug(6040) << "right: " << lastPnt << " == " << insPnt << ": " << (insPnt == lastPnt) << endl; - appendPoint(pointArray, insPnt); - } - } - - // Insert ending point of box - appendPoint(pointArray, newPnt); - -// kdDebug(6000) << "collectHorizontalBoxCoordinates: " << "ende" << endl; -} - -/** - * Checks whether the given line box' extents and the following line box' - * extents are disjount (i. e. do not share the same x-coordinate range). - * @param line line box - * @param toBegin \c true, compare with preceding line box, \c false, with - * succeeding - * @return \c true if this and the next box are disjoint - */ -inline static bool lineBoxesDisjoint(InlineRunBox *line, int offset, bool toBegin) -{ - InlineRunBox *next = toBegin ? line->prevLineBox() : line->nextLineBox(); - return !next || next->xPos() + next->width() + 2*offset < line->xPos() - || next->xPos() > line->xPos() + line->width() + 2*offset; -} - -/** - * Traverses the vertical outer borders of the given render flow's line - * boxes and appends the point coordinates to the given point array. - * @param line line box to begin traversal - * @param pointArray point array - * @param left \c true, traverse the left vertical coordinates, - * \c false, traverse the right vertical coordinates. - * @param lastline if not 0, returns the pointer to the last line box traversed - */ -static void collectVerticalBoxCoordinates(InlineRunBox *line, - TQValueVector<TQPoint> &pointArray, - bool left, int offset, InlineRunBox **lastline = 0) -{ - InlineRunBox *last = 0; - offset = left ? -offset:offset; - for (InlineRunBox* curr = line; curr && !last; curr = left ? curr->prevLineBox() : curr->nextLineBox()) { - InlineBox *root = curr; - - bool isLast = lineBoxesDisjoint(curr, kAbs(offset), left); - if (isLast) last = curr; - - if (root != line && !isLast) - while (root->parent()) root = root->parent(); - TQPoint newPnt(curr->xPos() + !left*curr->width() + offset, - (left ? root->topOverflow() : root->bottomOverflow()) + offset); - if (!pointArray.isEmpty()) { - TQPoint lastPnt = pointArray.back(); - if (newPnt.x()>lastPnt.x() && !left) - pointArray.back().setY( kMin(lastPnt.y(), root->topOverflow()-offset) ); - else if (newPnt.x()<lastPnt.x() && left) - pointArray.back().setY( kMax(lastPnt.y(), root->bottomOverflow()+offset) ); - TQPoint insPnt(newPnt.x(), pointArray.back().y()); -// kdDebug(6040) << "left: " << lastPnt << " == " << insPnt << ": " << (insPnt == lastPnt) << endl; - appendPoint(pointArray, insPnt); - } - appendPoint(pointArray, newPnt); - } - if (lastline) *lastline = last; -} - -/** - * Links up the end of the given point-array such that the starting point - * is not a segment separator. - * - * To achieve this, improper points are removed from the beginning of - * the point-array (by changing the array's starting iterator), and - * proper ones appended to the point-array's back. - * - * @param pointArray point-array - * @return actual begin of point array - */ -static TQPoint *linkEndToBegin(TQValueVector<TQPoint> &pointArray) -{ - uint index = 0; - assert(pointArray.size() >= 3); - - // if first and last points match, ignore the last one. - bool linkup = false; TQPoint linkupPnt; - if (pointArray.front() == pointArray.back()) { - linkupPnt = pointArray.back(); - pointArray.pop_back(); - linkup = true; - } - - const TQPoint *it = pointArray.begin() + index; - TQPoint pfirst = *it; - TQPoint pnext = *++it; - TQPoint plast = pointArray.back(); -// kdDebug(6040) << "linkcheck plast: " << plast << " pfirst: " << pfirst << " pnext: " << pnext << endl; - - if (plast.x() == pfirst.x() && pfirst.x() == pnext.x() - || plast.y() == pfirst.y() && pfirst.y() == pnext.y()) { - - ++index; - appendPoint(pointArray, pfirst); - appendPoint(pointArray, pnext); - } else if (linkup) - pointArray.push_back(linkupPnt); - return pointArray.begin() + index; -} - -void RenderInline::paintOutlines(TQPainter *p, int _tx, int _ty) -{ - if (style()->outlineWidth() == 0 || style()->outlineStyle() <= BHIDDEN) - return; - int offset = style()->outlineOffset(); - - // We may have to draw more than one outline path as they may be - // disjoint. - for (InlineRunBox *curr = firstLineBox(); curr; curr = curr->nextLineBox()) { - TQValueVector<TQPoint> path; - - // collect topmost outline - collectHorizontalBoxCoordinates(curr, path, false, offset); - // collect right outline - collectVerticalBoxCoordinates(curr, path, false, offset, &curr); - // collect bottommost outline - collectHorizontalBoxCoordinates(curr, path, true, offset); - // collect left outline - collectVerticalBoxCoordinates(curr, path, true, offset); - - if (path.size() < 3) continue; - - const TQPoint *begin = linkEndToBegin(path); - - // paint the outline - paintOutlinePath(p, _tx, _ty, begin, path.end(), BSLeft, -1, BSTop); - } -} - -template<class T> inline void kSwap(T &a1, T &a2) -{ - T tmp = a2; - a2 = a1; - a1 = tmp; -} - -enum BSOrientation { BSHorizontal, BSVertical }; - -/** - * Returns the orientation of the given border side. - */ -inline BSOrientation bsOrientation(RenderObject::BorderSide bs) -{ - switch (bs) { - case RenderObject::BSTop: - case RenderObject::BSBottom: - return BSHorizontal; - case RenderObject::BSLeft: - case RenderObject::BSRight: - return BSVertical; - } - return BSHorizontal; // make gcc happy (sigh) -} - -/** - * Determines the new border side by evaluating the new direction as determined - * by the given coordinates, the old border side, and the relative direction. - * - * The relative direction specifies whether the old border side meets with the - * straight given by the coordinates from below (negative), or above (positive). - */ -inline RenderObject::BorderSide newBorderSide(RenderObject::BorderSide oldBS, int direction, const TQPoint &last, const TQPoint &cur) -{ - bool below = direction < 0; - if (last.x() == cur.x()) { // new segment is vertical - bool t = oldBS == RenderObject::BSTop; - bool b = oldBS == RenderObject::BSBottom; - if ((t || b) && last.y() != cur.y()) - return (cur.y() < last.y()) ^ (t && below || b && !below) - ? RenderObject::BSLeft : RenderObject::BSRight; - } else /*if (last.y() == cur.y())*/ { // new segment is horizontal - bool l = oldBS == RenderObject::BSLeft; - bool r = oldBS == RenderObject::BSRight; - if ((l || r) && last.x() != cur.x()) - return (cur.x() < last.x()) ^ (l && below || r && !below) - ? RenderObject::BSTop : RenderObject::BSBottom; - } - return oldBS; // same direction -} - -/** - * Draws an outline segment between the given two points. - * @param o render object - * @param p painter - * @param tx absolute x-coordinate of containing block - * @param ty absolute y-coordinate of containing block - * @param p1 starting point - * @param p2 end point - * @param prevBS border side of previous segment - * @param curBS border side of this segment - * @param nextBS border side of next segment - */ -static void paintOutlineSegment(RenderObject *o, TQPainter *p, int tx, int ty, - const TQPoint &p1, const TQPoint &p2, - RenderObject::BorderSide prevBS, - RenderObject::BorderSide curBS, - RenderObject::BorderSide nextBS) -{ - int ow = o->style()->outlineWidth(); - EBorderStyle os = o->style()->outlineStyle(); - TQColor oc = o->style()->outlineColor(); - - int x1 = tx + p1.x(); - int y1 = ty + p1.y(); - int x2 = tx + p2.x(); - int y2 = ty + p2.y(); - if (x1 > x2) { - kSwap(x1, x2); - if (bsOrientation(curBS) == BSHorizontal) kSwap(prevBS, nextBS); - } - if (y1 > y2) { - kSwap(y1, y2); - if (bsOrientation(curBS) == BSVertical) kSwap(prevBS, nextBS); - } - -// kdDebug(6040) << "segment(" << x1 << "," << y1 << ") - (" << x2 << "," << y2 << ")" << endl; -/* p->setPen(Qt::gray); - p->drawLine(x1,y1,x2,y2);*/ - switch (curBS) { - case RenderObject::BSLeft: - case RenderObject::BSRight: -/* p->setPen(TQColor("#ffe4dd")); - p->drawLine( - x1 - (curBS == RenderObject::BSLeft ? ow : 0), - y1 - (prevBS == RenderObject::BSTop ? ow : 0), - x2 + (curBS == RenderObject::BSRight ? ow : 0), - y2 + (nextBS == RenderObject::BSBottom ? ow : 0) - );*/ - o->drawBorder(p, - x1 - (curBS == RenderObject::BSLeft ? ow : 0), - y1 - (prevBS == RenderObject::BSTop ? ow : 0), - x2 + (curBS == RenderObject::BSRight ? ow : 0), - y2 + (nextBS == RenderObject::BSBottom ? ow : 0), - curBS, oc, o->style()->color(), os, - prevBS == RenderObject::BSTop ? ow - : prevBS == RenderObject::BSBottom ? -ow : 0, - nextBS == RenderObject::BSTop ? -ow - : nextBS == RenderObject::BSBottom ? ow : 0, - true); - break; - case RenderObject::BSBottom: - case RenderObject::BSTop: -// kdDebug(6040) << "BSTop/BSBottom: prevBS " << prevBS << " curBS " << curBS << " nextBS " << nextBS << endl; - o->drawBorder(p, - x1 - (prevBS == RenderObject::BSLeft ? ow : 0), - y1 - (curBS == RenderObject::BSTop ? ow : 0), - x2 + (nextBS == RenderObject::BSRight ? ow : 0), - y2 + (curBS == RenderObject::BSBottom ? ow : 0), - curBS, oc, o->style()->color(), os, - prevBS == RenderObject::BSLeft ? ow - : prevBS == RenderObject::BSRight ? -ow : 0, - nextBS == RenderObject::BSLeft ? -ow - : nextBS == RenderObject::BSRight ? ow : 0, - true); - break; - } -} - -void RenderInline::paintOutlinePath(TQPainter *p, int tx, int ty, const TQPoint *begin, const TQPoint *end, BorderSide bs, int direction, BorderSide endingBS) -{ - int ow = style()->outlineWidth(); - if (ow == 0 || m_isContinuation) // Continuations get painted by the original inline. - return; - - TQPoint last = *begin; - BorderSide lastBS = bs; - Q_ASSERT(begin != end); - ++begin; - -// kdDebug(6040) << "last: " << last << endl; - - bs = newBorderSide(bs, direction, last, *begin); -// kdDebug(6040) << "newBorderSide: " << lastBS << " " << direction << "d " << last << " - " << *begin << " => " << bs << endl; - - for (const TQPoint *it = begin; it != end; ++it) { - TQPoint cur = *it; -// kdDebug(6040) << "cur: " << cur << endl; - BorderSide nextBS; - if (it + 1 != end) { - TQPoint diff = cur - last; - direction = diff.x() + diff.y(); - nextBS = newBorderSide(bs, direction, cur, *(it + 1)); -// kdDebug(6040) << "newBorderSide*: " << bs << " " << direction << "d " << cur << " - " << *(it + 1) << " => " << nextBS << endl; - } else - nextBS = endingBS; - - Q_ASSERT(bsOrientation(bs) != bsOrientation(nextBS)); - paintOutlineSegment(this, p, tx, ty, last, cur, - lastBS, bs, nextBS); - lastBS = bs; - last = cur; - bs = nextBS; - } - -} - -void RenderInline::calcMinMaxWidth() -{ - KHTMLAssert( !minMaxKnown() ); - -#ifdef DEBUG_LAYOUT - kdDebug( 6040 ) << renderName() << "(RenderInline)::calcMinMaxWidth() this=" << this << endl; -#endif - - // Irrelevant, since some enclosing block will actually measure us and our children. - m_minWidth = 0; - m_maxWidth = 0; - - setMinMaxKnown(); -} - -short RenderInline::width() const -{ - // Return the width of the minimal left side and the maximal right side. - short leftSide = 0; - short rightSide = 0; - for (InlineRunBox* curr = firstLineBox(); curr; curr = curr->nextLineBox()) { - if (curr == firstLineBox() || curr->xPos() < leftSide) - leftSide = curr->xPos(); - if (curr == firstLineBox() || curr->xPos() + curr->width() > rightSide) - rightSide = curr->xPos() + curr->width(); - } - - return rightSide - leftSide; -} - -int RenderInline::height() const -{ - int h = 0; - if (firstLineBox()) - h = lastLineBox()->yPos() + lastLineBox()->height() - firstLineBox()->yPos(); - return h; -} - -int RenderInline::offsetLeft() const -{ - int x = RenderFlow::offsetLeft(); - if (firstLineBox()) - x += firstLineBox()->xPos(); - return x; -} - -int RenderInline::offsetTop() const -{ - int y = RenderFlow::offsetTop(); - if (firstLineBox()) - y += firstLineBox()->yPos(); - return y; -} - -const char *RenderInline::renderName() const -{ - if (isRelPositioned()) - return "RenderInline (relative positioned)"; - if (isAnonymous()) - return "RenderInline (anonymous)"; - return "RenderInline"; -} - -bool RenderInline::nodeAtPoint(NodeInfo& info, int _x, int _y, int _tx, int _ty, HitTestAction hitTestAction, bool inside) -{ -/* - if ( hitTestAction != HitTestSelfOnly ) { - for (RenderObject* child = lastChild(); child; child = child->previousSibling()) - if (!child->layer() && !child->isFloating() && child->nodeAtPoint(info, _x, _y, _tx, _ty, HitTestAll)) - inside = true; - } -*/ - // Check our line boxes if we're still not inside. - if (/*hitTestAction != HitTestChildrenOnly &&*/ !inside && style()->visibility() != HIDDEN) { - // See if we're inside one of our line boxes. - inside = hitTestLines(info, _x, _y, _tx, _ty, hitTestAction); - } - - if (inside && element()) { - if (info.innerNode() && info.innerNode()->renderer() && - !info.innerNode()->renderer()->isInline()) { - // Within the same layer, inlines are ALWAYS fully above blocks. Change inner node. - info.setInnerNode(element()); - - // Clear everything else. - info.setInnerNonSharedNode(0); - info.setURLElement(0); - } - - if (!info.innerNode()) - info.setInnerNode(element()); - - if(!info.innerNonSharedNode()) - info.setInnerNonSharedNode(element()); - } - - return inside; -} - -void RenderInline::caretPos(int offset, int flags, int &_x, int &_y, int &width, int &height) -{ - _x = -1; - - RenderBlock *cb = containingBlock(); - bool rtl = cb->style()->direction() == RTL; - bool outsideEnd = flags & CFOutsideEnd; - // I need to explain that: outsideEnd contains a meaningful value if - // and only if flags & CFOutside is set. If it is not, then randomly - // either the first or the last line box is returned. - // This doesn't matter because the only case this can happen is on an - // empty inline element, whose first and last line boxes are actually - // the same. - InlineFlowBox *line = !outsideEnd ^ rtl ? firstLineBox() : lastLineBox(); - - if (!line) { // umpf, handle "gracefully" - RenderFlow::caretPos(offset, flags, _x, _y, width, height); - return; - } - - _x = line->xPos(); - width = 1; // ### regard CFOverride - - // Place caret outside the border - if (flags & CFOutside) { - RenderStyle *s = element() && element()->parent() - && element()->parent()->renderer() - ? element()->parent()->renderer()->style() - : style(); - const TQFontMetrics &fm = s->fontMetrics(); - _y = line->yPos() + line->baseline() - fm.ascent(); - height = fm.height(); - - if (!outsideEnd ^ rtl) { - _x -= line->marginBorderPaddingLeft(); - } else { - _x += line->width() + line->marginBorderPaddingRight(); - } - - } else { - const TQFontMetrics &fm = style()->fontMetrics(); - _y = line->yPos() + line->baseline() - fm.ascent(); - height = fm.height(); - } - - int absx, absy; - if (cb && cb->absolutePosition(absx,absy)) { - //kdDebug(6040) << "absx=" << absx << " absy=" << absy << endl; - _x += absx; - _y += absy; - } else { - // we don't know our absolute position, and there is no point returning - // just a relative one - _x = _y = -1; - } -} - -inline int minXPos(const RenderInline *o) -{ - int retval=6666666; - if (!o->firstLineBox()) return 0; - for (InlineRunBox* curr = o->firstLineBox(); curr; curr = curr->nextLineBox()) - retval = kMin( retval, int( curr->m_x )); - return retval; -} - -int RenderInline::inlineXPos() const -{ - return minXPos(this); -} - -int RenderInline::inlineYPos() const -{ - return firstLineBox() ? firstLineBox()->yPos() : 0; -} - |