https://bugs.webkit.org/show_bug.cgi?id=77383
[WebKit-https.git] / Source / WebCore / rendering / RenderReplaced.cpp
1 /*
2  * Copyright (C) 1999 Lars Knoll (knoll@kde.org)
3  * Copyright (C) 2000 Dirk Mueller (mueller@kde.org)
4  * Copyright (C) 2004, 2006, 2007 Apple Inc. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  *
21  */
22
23 #include "config.h"
24 #include "RenderReplaced.h"
25
26 #include "Frame.h"
27 #include "GraphicsContext.h"
28 #include "LayoutRepainter.h"
29 #include "Page.h"
30 #include "RenderBlock.h"
31 #include "RenderLayer.h"
32 #include "RenderTheme.h"
33 #include "RenderView.h"
34 #include "VisiblePosition.h"
35
36 using namespace std;
37
38 namespace WebCore {
39
40 const int cDefaultWidth = 300;
41 const int cDefaultHeight = 150;
42
43 RenderReplaced::RenderReplaced(Node* node)
44     : RenderBox(node)
45     , m_intrinsicSize(cDefaultWidth, cDefaultHeight)
46     , m_hasIntrinsicSize(false)
47 {
48     setReplaced(true);
49 }
50
51 RenderReplaced::RenderReplaced(Node* node, const IntSize& intrinsicSize)
52     : RenderBox(node)
53     , m_intrinsicSize(intrinsicSize)
54     , m_hasIntrinsicSize(true)
55 {
56     setReplaced(true);
57 }
58
59 RenderReplaced::~RenderReplaced()
60 {
61 }
62
63 void RenderReplaced::willBeDestroyed()
64 {
65     if (!documentBeingDestroyed() && parent())
66         parent()->dirtyLinesFromChangedChild(this);
67
68     RenderBox::willBeDestroyed();
69 }
70
71 void RenderReplaced::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
72 {
73     RenderBox::styleDidChange(diff, oldStyle);
74
75     bool hadStyle = (oldStyle != 0);
76     float oldZoom = hadStyle ? oldStyle->effectiveZoom() : RenderStyle::initialZoom();
77     if (style() && style()->effectiveZoom() != oldZoom)
78         intrinsicSizeChanged();
79 }
80
81 void RenderReplaced::layout()
82 {
83     ASSERT(needsLayout());
84     
85     LayoutRepainter repainter(*this, checkForRepaintDuringLayout());
86     
87     setHeight(minimumReplacedHeight());
88
89     computeLogicalWidth();
90     computeLogicalHeight();
91
92     m_overflow.clear();
93     addVisualEffectOverflow();
94     updateLayerTransform();
95     
96     repainter.repaintAfterLayout();
97     setNeedsLayout(false);
98 }
99  
100 void RenderReplaced::intrinsicSizeChanged()
101 {
102     int scaledWidth = static_cast<int>(cDefaultWidth * style()->effectiveZoom());
103     int scaledHeight = static_cast<int>(cDefaultHeight * style()->effectiveZoom());
104     m_intrinsicSize = IntSize(scaledWidth, scaledHeight);
105     setNeedsLayoutAndPrefWidthsRecalc();
106 }
107
108 void RenderReplaced::paint(PaintInfo& paintInfo, const LayoutPoint& paintOffset)
109 {
110     if (!shouldPaint(paintInfo, paintOffset))
111         return;
112     
113     LayoutPoint adjustedPaintOffset = paintOffset + location();
114     
115     if (hasBoxDecorations() && (paintInfo.phase == PaintPhaseForeground || paintInfo.phase == PaintPhaseSelection)) 
116         paintBoxDecorations(paintInfo, adjustedPaintOffset);
117     
118     if (paintInfo.phase == PaintPhaseMask) {
119         paintMask(paintInfo, adjustedPaintOffset);
120         return;
121     }
122
123     LayoutRect paintRect = LayoutRect(adjustedPaintOffset, size());
124     if ((paintInfo.phase == PaintPhaseOutline || paintInfo.phase == PaintPhaseSelfOutline) && style()->outlineWidth())
125         paintOutline(paintInfo.context, paintRect);
126     
127     if (paintInfo.phase != PaintPhaseForeground && paintInfo.phase != PaintPhaseSelection && !canHaveChildren())
128         return;
129     
130     if (!paintInfo.shouldPaintWithinRoot(this))
131         return;
132     
133     bool drawSelectionTint = selectionState() != SelectionNone && !document()->printing();
134     if (paintInfo.phase == PaintPhaseSelection) {
135         if (selectionState() == SelectionNone)
136             return;
137         drawSelectionTint = false;
138     }
139
140     if (Frame* frame = this->frame()) {
141         if (Page* page = frame->page())
142             page->addRelevantRepaintedObject(this, paintInfo.rect);
143     }
144
145     bool completelyClippedOut = false;
146     if (style()->hasBorderRadius()) {
147         LayoutRect borderRect = LayoutRect(adjustedPaintOffset, size());
148
149         if (borderRect.isEmpty())
150             completelyClippedOut = true;
151         else {
152             // Push a clip if we have a border radius, since we want to round the foreground content that gets painted.
153             paintInfo.context->save();
154             paintInfo.context->addRoundedRectClip(style()->getRoundedBorderFor(paintRect));
155         }
156     }
157
158     if (!completelyClippedOut) {
159         paintReplaced(paintInfo, adjustedPaintOffset);
160
161         if (style()->hasBorderRadius())
162             paintInfo.context->restore();
163     }
164         
165     // The selection tint never gets clipped by border-radius rounding, since we want it to run right up to the edges of
166     // surrounding content.
167     if (drawSelectionTint) {
168         LayoutRect selectionPaintingRect = localSelectionRect();
169         selectionPaintingRect.moveBy(adjustedPaintOffset);
170         paintInfo.context->fillRect(selectionPaintingRect, selectionBackgroundColor(), style()->colorSpace());
171     }
172 }
173
174 bool RenderReplaced::shouldPaint(PaintInfo& paintInfo, const LayoutPoint& paintOffset)
175 {
176     if (paintInfo.phase != PaintPhaseForeground && paintInfo.phase != PaintPhaseOutline && paintInfo.phase != PaintPhaseSelfOutline 
177             && paintInfo.phase != PaintPhaseSelection && paintInfo.phase != PaintPhaseMask)
178         return false;
179
180     if (!paintInfo.shouldPaintWithinRoot(this))
181         return false;
182         
183     // if we're invisible or haven't received a layout yet, then just bail.
184     if (style()->visibility() != VISIBLE)
185         return false;
186
187     LayoutPoint adjustedPaintOffset = paintOffset + location();
188
189     // Early exit if the element touches the edges.
190     LayoutUnit top = adjustedPaintOffset.y() + minYVisualOverflow();
191     LayoutUnit bottom = adjustedPaintOffset.y() + maxYVisualOverflow();
192     if (isSelected() && m_inlineBoxWrapper) {
193         LayoutUnit selTop = paintOffset.y() + m_inlineBoxWrapper->root()->selectionTop();
194         LayoutUnit selBottom = paintOffset.y() + selTop + m_inlineBoxWrapper->root()->selectionHeight();
195         top = min(selTop, top);
196         bottom = max(selBottom, bottom);
197     }
198     
199     LayoutUnit os = 2 * maximalOutlineSize(paintInfo.phase);
200     if (adjustedPaintOffset.x() + minXVisualOverflow() >= paintInfo.rect.maxX() + os || adjustedPaintOffset.x() + maxXVisualOverflow() <= paintInfo.rect.x() - os)
201         return false;
202     if (top >= paintInfo.rect.maxY() + os || bottom <= paintInfo.rect.y() - os)
203         return false;
204
205     return true;
206 }
207
208 int RenderReplaced::computeIntrinsicLogicalWidth(RenderBox* contentRenderer, bool includeMaxWidth) const
209 {
210     if (m_hasIntrinsicSize)
211         return computeReplacedLogicalWidthRespectingMinMaxWidth(calcAspectRatioLogicalWidth(), includeMaxWidth);
212     ASSERT(contentRenderer);
213     ASSERT(contentRenderer->style());
214     return contentRenderer->computeReplacedLogicalWidthRespectingMinMaxWidth(contentRenderer->computeReplacedLogicalWidthUsing(contentRenderer->style()->logicalWidth()), includeMaxWidth);
215 }
216
217 int RenderReplaced::computeIntrinsicLogicalHeight(RenderBox* contentRenderer) const
218 {
219     if (m_hasIntrinsicSize)
220         return computeReplacedLogicalHeightRespectingMinMaxHeight(calcAspectRatioLogicalHeight());
221     ASSERT(contentRenderer);
222     ASSERT(contentRenderer->style());
223     return contentRenderer->computeReplacedLogicalHeightRespectingMinMaxHeight(contentRenderer->computeReplacedLogicalHeightUsing(contentRenderer->style()->logicalHeight()));
224 }
225
226 static inline RenderBlock* firstContainingBlockWithLogicalWidth(const RenderReplaced* replaced)
227 {
228     // We have to lookup the containing block, which has an explicit width, which must not be equal to our direct containing block.
229     // If the embedded document appears _after_ we performed the initial layout, our intrinsic size is 300x150. If our containing
230     // block doesn't provide an explicit width, it's set to the 300 default, coming from the initial layout run.
231     RenderBlock* containingBlock = replaced->containingBlock();
232     if (!containingBlock)
233         return 0;
234
235     for (; !containingBlock->isRenderView() && !containingBlock->isBody(); containingBlock = containingBlock->containingBlock()) {
236         if (containingBlock->style()->logicalWidth().isSpecified())
237             return containingBlock;
238     }
239
240     return 0;
241 }
242
243 bool RenderReplaced::hasReplacedLogicalWidth() const
244 {
245     if (style()->logicalWidth().isSpecified())
246         return true;
247
248     if (style()->logicalWidth().isAuto())
249         return false;
250
251     return firstContainingBlockWithLogicalWidth(this);
252 }
253
254 static inline bool hasAutoHeightOrContainingBlockWithAutoHeight(const RenderReplaced* replaced)
255 {
256     Length logicalHeightLength = replaced->style()->logicalHeight();
257     if (logicalHeightLength.isAuto())
258         return true;
259
260     // For percentage heights: The percentage is calculated with respect to the height of the generated box's
261     // containing block. If the height of the containing block is not specified explicitly (i.e., it depends
262     // on content height), and this element is not absolutely positioned, the value computes to 'auto'.
263     if (!logicalHeightLength.isPercent() || replaced->isPositioned() || replaced->document()->inQuirksMode())
264         return false;
265
266     for (RenderBlock* cb = replaced->containingBlock(); !cb->isRenderView(); cb = cb->containingBlock()) {
267         if (cb->isTableCell() || (!cb->style()->logicalHeight().isAuto() || (!cb->style()->top().isAuto() && !cb->style()->bottom().isAuto())))
268             return false;
269     }
270
271     return true;
272 }
273
274 bool RenderReplaced::hasReplacedLogicalHeight() const
275 {
276     if (style()->logicalHeight().isAuto())
277         return false;
278
279     if (style()->logicalHeight().isSpecified()) {
280         if (hasAutoHeightOrContainingBlockWithAutoHeight(this))
281             return false;
282         return true;
283     }
284
285     return false;
286 }
287
288 LayoutUnit RenderReplaced::computeReplacedLogicalWidth(bool includeMaxWidth) const
289 {
290     if (style()->logicalWidth().isSpecified())
291         return computeReplacedLogicalWidthRespectingMinMaxWidth(computeReplacedLogicalWidthUsing(style()->logicalWidth()), includeMaxWidth);
292
293     RenderBox* contentRenderer = embeddedContentBox();
294
295     // 10.3.2 Inline, replaced elements: http://www.w3.org/TR/CSS21/visudet.html#inline-replaced-width
296     bool isPercentageIntrinsicSize = false;
297     double intrinsicRatio = 0;
298     FloatSize intrinsicSize;
299     if (contentRenderer)
300         contentRenderer->computeIntrinsicRatioInformation(intrinsicSize, intrinsicRatio, isPercentageIntrinsicSize);
301     else
302         computeIntrinsicRatioInformation(intrinsicSize, intrinsicRatio, isPercentageIntrinsicSize);
303
304     if (intrinsicRatio && !isHorizontalWritingMode())
305         intrinsicRatio = 1 / intrinsicRatio;
306
307     if (style()->logicalWidth().isAuto()) {
308         bool heightIsAuto = style()->logicalHeight().isAuto();
309         bool hasIntrinsicWidth = m_hasIntrinsicSize || (!isPercentageIntrinsicSize && intrinsicSize.width() > 0);
310
311         // If 'height' and 'width' both have computed values of 'auto' and the element also has an intrinsic width, then that intrinsic width is the used value of 'width'.
312         if (heightIsAuto && hasIntrinsicWidth) {
313             if (m_hasIntrinsicSize)
314                 return computeReplacedLogicalWidthRespectingMinMaxWidth(calcAspectRatioLogicalWidth(), includeMaxWidth);
315             return static_cast<LayoutUnit>(intrinsicSize.width() * style()->effectiveZoom());
316         }
317
318         bool hasIntrinsicHeight = m_hasIntrinsicSize || (!isPercentageIntrinsicSize && intrinsicSize.height() > 0);
319         if (intrinsicRatio || isPercentageIntrinsicSize) {
320             // If 'height' and 'width' both have computed values of 'auto' and the element has no intrinsic width, but does have an intrinsic height and intrinsic ratio;
321             // or if 'width' has a computed value of 'auto', 'height' has some other computed value, and the element does have an intrinsic ratio; then the used value
322             // of 'width' is: (used height) * (intrinsic ratio)
323             if (intrinsicRatio && ((heightIsAuto && !hasIntrinsicWidth && hasIntrinsicHeight) || !heightIsAuto)) {
324                 LayoutUnit logicalHeight = computeReplacedLogicalHeightUsing(style()->logicalHeight());
325                 return computeReplacedLogicalWidthRespectingMinMaxWidth(static_cast<LayoutUnit>(ceil(logicalHeight * intrinsicRatio)));
326             }
327
328             // If 'height' and 'width' both have computed values of 'auto' and the element has an intrinsic ratio but no intrinsic height or width, then the used value of
329             // 'width' is undefined in CSS 2.1. However, it is suggested that, if the containing block's width does not itself depend on the replaced element's width, then
330             // the used value of 'width' is calculated from the constraint equation used for block-level, non-replaced elements in normal flow.
331             if (heightIsAuto && !hasIntrinsicWidth && !hasIntrinsicHeight && contentRenderer) {
332                 // The aforementioned 'constraint equation' used for block-level, non-replaced elements in normal flow:
333                 // 'margin-left' + 'border-left-width' + 'padding-left' + 'width' + 'padding-right' + 'border-right-width' + 'margin-right' = width of containing block
334                 LayoutUnit logicalWidth;
335                 if (RenderBlock* blockWithWidth = firstContainingBlockWithLogicalWidth(this))
336                     logicalWidth = blockWithWidth->computeReplacedLogicalWidthRespectingMinMaxWidth(blockWithWidth->computeReplacedLogicalWidthUsing(blockWithWidth->style()->logicalWidth()), false);
337                 else
338                     logicalWidth = containingBlock()->availableLogicalWidth();
339
340                 // This solves above equation for 'width' (== logicalWidth).
341                 LayoutUnit marginStart = style()->marginStart().calcMinValue(logicalWidth);
342                 LayoutUnit marginEnd = style()->marginEnd().calcMinValue(logicalWidth);
343                 logicalWidth = max(0, logicalWidth - (marginStart + marginEnd + (width() - clientWidth())));
344                 if (isPercentageIntrinsicSize)
345                     // FIXME: Remove unnecessary rounding when layout is off ints: webkit.org/b/63656
346                     logicalWidth = static_cast<LayoutUnit>(round(logicalWidth * intrinsicSize.width() / 100));
347                 return computeReplacedLogicalWidthRespectingMinMaxWidth(logicalWidth);
348             }
349         }
350
351         // Otherwise, if 'width' has a computed value of 'auto', and the element has an intrinsic width, then that intrinsic width is the used value of 'width'.
352         if (hasIntrinsicWidth) {
353             if (isPercentageIntrinsicSize || m_hasIntrinsicSize)
354                 return computeReplacedLogicalWidthRespectingMinMaxWidth(calcAspectRatioLogicalWidth(), includeMaxWidth);
355             return static_cast<LayoutUnit>(intrinsicSize.width() * style()->effectiveZoom());
356         }
357
358         // Otherwise, if 'width' has a computed value of 'auto', but none of the conditions above are met, then the used value of 'width' becomes 300px. If 300px is too
359         // wide to fit the device, UAs should use the width of the largest rectangle that has a 2:1 ratio and fits the device instead.
360         return computeReplacedLogicalWidthRespectingMinMaxWidth(cDefaultWidth, includeMaxWidth);
361     }
362
363     return computeReplacedLogicalWidthRespectingMinMaxWidth(intrinsicLogicalWidth(), includeMaxWidth);
364 }
365
366 LayoutUnit RenderReplaced::computeReplacedLogicalHeight() const
367 {
368     // 10.5 Content height: the 'height' property: http://www.w3.org/TR/CSS21/visudet.html#propdef-height
369     if (hasReplacedLogicalHeight())
370         return computeReplacedLogicalHeightRespectingMinMaxHeight(computeReplacedLogicalHeightUsing(style()->logicalHeight()));
371
372     RenderBox* contentRenderer = embeddedContentBox();
373
374     // 10.6.2 Inline, replaced elements: http://www.w3.org/TR/CSS21/visudet.html#inline-replaced-height
375     bool isPercentageIntrinsicSize = false;
376     double intrinsicRatio = 0;
377     FloatSize intrinsicSize;
378     if (contentRenderer)
379         contentRenderer->computeIntrinsicRatioInformation(intrinsicSize, intrinsicRatio, isPercentageIntrinsicSize);
380     else
381         computeIntrinsicRatioInformation(intrinsicSize, intrinsicRatio, isPercentageIntrinsicSize);
382
383     if (intrinsicRatio && !isHorizontalWritingMode())
384         intrinsicRatio = 1 / intrinsicRatio;
385
386     bool widthIsAuto = style()->logicalWidth().isAuto();
387     bool hasIntrinsicHeight = m_hasIntrinsicSize || (!isPercentageIntrinsicSize && intrinsicSize.height() > 0);
388
389     // If 'height' and 'width' both have computed values of 'auto' and the element also has an intrinsic height, then that intrinsic height is the used value of 'height'.
390     if (widthIsAuto && hasIntrinsicHeight) {
391         if (m_hasIntrinsicSize)
392             return computeReplacedLogicalHeightRespectingMinMaxHeight(calcAspectRatioLogicalHeight());
393         return static_cast<LayoutUnit>(intrinsicSize.height() * style()->effectiveZoom());
394     }
395
396     // Otherwise, if 'height' has a computed value of 'auto', and the element has an intrinsic ratio then the used value of 'height' is:
397     // (used width) / (intrinsic ratio)
398     if (intrinsicRatio && !isPercentageIntrinsicSize) {
399         // FIXME: Remove unnecessary rounding when layout is off ints: webkit.org/b/63656
400         return computeReplacedLogicalHeightRespectingMinMaxHeight(round(availableLogicalWidth() / intrinsicRatio));
401     }
402
403     // Otherwise, if 'height' has a computed value of 'auto', and the element has an intrinsic height, then that intrinsic height is the used value of 'height'.
404     if (hasIntrinsicHeight) {
405         if (m_hasIntrinsicSize)
406             return computeReplacedLogicalHeightRespectingMinMaxHeight(calcAspectRatioLogicalHeight());
407         return static_cast<LayoutUnit>(intrinsicSize.height() * style()->effectiveZoom());
408     }
409
410     // Otherwise, if 'height' has a computed value of 'auto', but none of the conditions above are met, then the used value of 'height' must be set to the height
411     // of the largest rectangle that has a 2:1 ratio, has a height not greater than 150px, and has a width not greater than the device width.
412     return computeReplacedLogicalHeightRespectingMinMaxHeight(cDefaultHeight);
413 }
414
415 int RenderReplaced::calcAspectRatioLogicalWidth() const
416 {
417     int intrinsicWidth = intrinsicLogicalWidth();
418     int intrinsicHeight = intrinsicLogicalHeight();
419     if (!intrinsicHeight)
420         return 0;
421     return RenderBox::computeReplacedLogicalHeight() * intrinsicWidth / intrinsicHeight;
422 }
423
424 int RenderReplaced::calcAspectRatioLogicalHeight() const
425 {
426     int intrinsicWidth = intrinsicLogicalWidth();
427     int intrinsicHeight = intrinsicLogicalHeight();
428     if (!intrinsicWidth)
429         return 0;
430     return RenderBox::computeReplacedLogicalWidth() * intrinsicHeight / intrinsicWidth;
431 }
432
433 void RenderReplaced::computePreferredLogicalWidths()
434 {
435     ASSERT(preferredLogicalWidthsDirty());
436
437     LayoutUnit borderAndPadding = borderAndPaddingWidth();
438     m_maxPreferredLogicalWidth = computeReplacedLogicalWidth(false) + borderAndPadding;
439
440     if (style()->maxWidth().isFixed())
441         m_maxPreferredLogicalWidth = min(m_maxPreferredLogicalWidth, style()->maxWidth().value() + (style()->boxSizing() == CONTENT_BOX ? borderAndPadding : 0));
442
443     if (style()->width().isPercent() || style()->height().isPercent()
444         || style()->maxWidth().isPercent() || style()->maxHeight().isPercent()
445         || style()->minWidth().isPercent() || style()->minHeight().isPercent())
446         m_minPreferredLogicalWidth = 0;
447     else
448         m_minPreferredLogicalWidth = m_maxPreferredLogicalWidth;
449
450     setPreferredLogicalWidthsDirty(false);
451 }
452
453 VisiblePosition RenderReplaced::positionForPoint(const LayoutPoint& point)
454 {
455     // FIXME: This code is buggy if the replaced element is relative positioned.
456     InlineBox* box = inlineBoxWrapper();
457     RootInlineBox* rootBox = box ? box->root() : 0;
458     
459     LayoutUnit top = rootBox ? rootBox->selectionTop() : logicalTop();
460     LayoutUnit bottom = rootBox ? rootBox->selectionBottom() : logicalBottom();
461     
462     LayoutUnit blockDirectionPosition = isHorizontalWritingMode() ? point.y() + y() : point.x() + x();
463     LayoutUnit lineDirectionPosition = isHorizontalWritingMode() ? point.x() + x() : point.y() + y();
464     
465     if (blockDirectionPosition < top)
466         return createVisiblePosition(caretMinOffset(), DOWNSTREAM); // coordinates are above
467     
468     if (blockDirectionPosition >= bottom)
469         return createVisiblePosition(caretMaxOffset(), DOWNSTREAM); // coordinates are below
470     
471     if (node()) {
472         if (lineDirectionPosition <= logicalLeft() + (logicalWidth() / 2))
473             return createVisiblePosition(0, DOWNSTREAM);
474         return createVisiblePosition(1, DOWNSTREAM);
475     }
476
477     return RenderBox::positionForPoint(point);
478 }
479
480 LayoutRect RenderReplaced::selectionRectForRepaint(RenderBoxModelObject* repaintContainer, bool clipToVisibleContent)
481 {
482     ASSERT(!needsLayout());
483
484     if (!isSelected())
485         return LayoutRect();
486     
487     LayoutRect rect = localSelectionRect();
488     if (clipToVisibleContent)
489         computeRectForRepaint(repaintContainer, rect);
490     else
491         rect = localToContainerQuad(FloatRect(rect), repaintContainer).enclosingBoundingBox();
492     
493     return rect;
494 }
495
496 IntRect RenderReplaced::localSelectionRect(bool checkWhetherSelected) const
497 {
498     if (checkWhetherSelected && !isSelected())
499         return IntRect();
500
501     if (!m_inlineBoxWrapper)
502         // We're a block-level replaced element.  Just return our own dimensions.
503         return IntRect(0, 0, width(), height());
504     
505     RootInlineBox* root = m_inlineBoxWrapper->root();
506     int newLogicalTop = root->block()->style()->isFlippedBlocksWritingMode() ? m_inlineBoxWrapper->logicalBottom() - root->selectionBottom() : root->selectionTop() - m_inlineBoxWrapper->logicalTop();
507     if (root->block()->style()->isHorizontalWritingMode())
508         return IntRect(0, newLogicalTop, width(), root->selectionHeight());
509     return IntRect(newLogicalTop, 0, root->selectionHeight(), height());
510 }
511
512 void RenderReplaced::setSelectionState(SelectionState s)
513 {
514     RenderBox::setSelectionState(s); // The selection state for our containing block hierarchy is updated by the base class call.
515     if (m_inlineBoxWrapper) {
516         RootInlineBox* line = m_inlineBoxWrapper->root();
517         if (line)
518             line->setHasSelectedChildren(isSelected());
519     }
520 }
521
522 bool RenderReplaced::isSelected() const
523 {
524     SelectionState s = selectionState();
525     if (s == SelectionNone)
526         return false;
527     if (s == SelectionInside)
528         return true;
529
530     int selectionStart, selectionEnd;
531     selectionStartEnd(selectionStart, selectionEnd);
532     if (s == SelectionStart)
533         return selectionStart == 0;
534         
535     int end = node()->hasChildNodes() ? node()->childNodeCount() : 1;
536     if (s == SelectionEnd)
537         return selectionEnd == end;
538     if (s == SelectionBoth)
539         return selectionStart == 0 && selectionEnd == end;
540         
541     ASSERT(0);
542     return false;
543 }
544
545 IntSize RenderReplaced::intrinsicSize() const
546 {
547     return m_intrinsicSize;
548 }
549
550 void RenderReplaced::setIntrinsicSize(const IntSize& size)
551 {
552     ASSERT(m_hasIntrinsicSize);
553     m_intrinsicSize = size;
554 }
555
556 LayoutRect RenderReplaced::clippedOverflowRectForRepaint(RenderBoxModelObject* repaintContainer) const
557 {
558     if (style()->visibility() != VISIBLE && !enclosingLayer()->hasVisibleContent())
559         return LayoutRect();
560
561     // The selectionRect can project outside of the overflowRect, so take their union
562     // for repainting to avoid selection painting glitches.
563     LayoutRect r = unionRect(localSelectionRect(false), visualOverflowRect());
564
565     RenderView* v = view();
566     if (v) {
567         // FIXME: layoutDelta needs to be applied in parts before/after transforms and
568         // repaint containers. https://bugs.webkit.org/show_bug.cgi?id=23308
569         r.move(v->layoutDelta());
570     }
571
572     if (style()) {
573         if (v)
574             r.inflate(style()->outlineSize());
575     }
576     computeRectForRepaint(repaintContainer, r);
577     return r;
578 }
579
580 }