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