c51292af3f5f07e8c150c56309733a3c48fdb658
[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 LayoutUnit RenderReplaced::computeReplacedLogicalWidth(bool includeMaxWidth) const
220 {
221     if (style()->logicalWidth().isSpecified())
222         return computeReplacedLogicalWidthRespectingMinMaxWidth(computeReplacedLogicalWidthUsing(style()->logicalWidth()), includeMaxWidth);
223
224     RenderBox* contentRenderer = embeddedContentBox();
225     RenderStyle* contentRenderStyle = 0;
226
227     // 10.3.2 Inline, replaced elements: http://www.w3.org/TR/CSS21/visudet.html#inline-replaced-width
228     bool isPercentageIntrinsicSize = false;
229     FloatSize intrinsicRatio;
230     if (contentRenderer) {
231         contentRenderer->computeIntrinsicRatioInformation(intrinsicRatio, isPercentageIntrinsicSize);
232         contentRenderStyle = contentRenderer->style();
233         ASSERT(contentRenderStyle);
234     } else
235         computeIntrinsicRatioInformation(intrinsicRatio, isPercentageIntrinsicSize);
236
237     if (style()->logicalWidth().isAuto()) {
238         bool heightIsAuto = style()->logicalHeight().isAuto();
239         bool hasIntrinsicWidth = m_hasIntrinsicSize || (contentRenderer && contentRenderStyle->logicalWidth().isFixed());
240
241         // 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'.
242         if (heightIsAuto && hasIntrinsicWidth)
243             return computeIntrinsicLogicalWidth(contentRenderer, includeMaxWidth);
244     
245         bool hasIntrinsicHeight = m_hasIntrinsicSize || (contentRenderer && contentRenderStyle->logicalHeight().isFixed());
246         if (!intrinsicRatio.isEmpty()) {
247             // 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;
248             // 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
249             // of 'width' is: (used height) * (intrinsic ratio)
250             if ((heightIsAuto && !hasIntrinsicWidth && hasIntrinsicHeight) || !heightIsAuto) {
251                 if (heightIsAuto) {
252                     if (contentRenderer) {
253                         LayoutUnit logicalHeight = contentRenderer->computeReplacedLogicalHeightUsing(contentRenderStyle->logicalHeight());
254                         // FIXME: Remove unnecessary rounding when layout is off ints: webkit.org/b/63656
255                         return contentRenderer->computeReplacedLogicalWidthRespectingMinMaxWidth(static_cast<int>(ceilf(logicalHeight * intrinsicRatio.width() / intrinsicRatio.height())));
256                     }
257                 } else { 
258                     LayoutUnit logicalHeight = computeReplacedLogicalHeightUsing(style()->logicalHeight());
259                     // FIXME: Remove unnecessary rounding when layout is off ints: webkit.org/b/63656
260                     return computeReplacedLogicalWidthRespectingMinMaxWidth(static_cast<int>(ceilf(logicalHeight * intrinsicRatio.width() / intrinsicRatio.height())));
261                 }
262             }
263
264             // 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
265             // '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
266             // the used value of 'width' is calculated from the constraint equation used for block-level, non-replaced elements in normal flow.
267             RenderBlock* containingBlock = this->containingBlock();
268             if (heightIsAuto && !hasIntrinsicWidth && !hasIntrinsicHeight && contentRenderer && containingBlock) {
269                 // The aforementioned 'constraint equation' used for block-level, non-replaced elements in normal flow:
270                 // 'margin-left' + 'border-left-width' + 'padding-left' + 'width' + 'padding-right' + 'border-right-width' + 'margin-right' = width of containing block
271
272                 // We have to lookup the containing block, which has an explicit width, which must not be equal to our direct containing block.
273                 // If the embedded document appears _after_ we performed the initial layout, our intrinsic size is 300x150. If our containing
274                 // block doesn't provide an explicit width, it's set to the 300 default, coming from the initial layout run.
275                 bool foundExplicitWidth = false;
276                 for (; !containingBlock->isRenderView() && !containingBlock->isBody(); containingBlock = containingBlock->containingBlock()) {
277                     if (containingBlock->style()->logicalWidth().isSpecified()) {
278                         foundExplicitWidth = true;
279                         break;
280                     }
281                 }
282                 
283                 // This solves above equation for 'width' (== logicalWidth).
284                 LayoutUnit logicalWidth = foundExplicitWidth ? computeIntrinsicLogicalWidth(containingBlock, false) : containingBlock->availableLogicalWidth();
285                 LayoutUnit marginStart = style()->marginStart().calcMinValue(logicalWidth);
286                 LayoutUnit marginEnd = style()->marginEnd().calcMinValue(logicalWidth);
287                 logicalWidth = max(0, logicalWidth - (marginStart + marginEnd + (width() - clientWidth())));
288                 if (isPercentageIntrinsicSize)
289                     // FIXME: Remove unnecessary rounding when layout is off ints: webkit.org/b/63656
290                     logicalWidth = static_cast<int>(ceilf(logicalWidth * intrinsicRatio.width() / 100));
291                 return computeReplacedLogicalWidthRespectingMinMaxWidth(logicalWidth);
292             }
293         }
294
295         // 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'.
296         if (hasIntrinsicWidth)
297             return computeIntrinsicLogicalWidth(contentRenderer, includeMaxWidth);
298
299         // 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
300         // 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.
301         return computeReplacedLogicalWidthRespectingMinMaxWidth(cDefaultWidth, includeMaxWidth);
302     }
303
304     return computeReplacedLogicalWidthRespectingMinMaxWidth(intrinsicLogicalWidth(), includeMaxWidth);
305 }
306
307 bool RenderReplaced::logicalHeightIsAuto() const
308 {
309     Length logicalHeightLength = style()->logicalHeight();
310     if (logicalHeightLength.isAuto())
311         return true;
312     
313     // For percentage heights: The percentage is calculated with respect to the height of the generated box's
314     // containing block. If the height of the containing block is not specified explicitly (i.e., it depends
315     // on content height), and this element is not absolutely positioned, the value computes to 'auto'.
316     if (!logicalHeightLength.isPercent() || isPositioned() || document()->inQuirksMode())
317         return false;
318
319     for (RenderBlock* cb = containingBlock(); !cb->isRenderView(); cb = cb->containingBlock()) {
320         if (cb->isTableCell() || (!cb->style()->logicalHeight().isAuto() || (!cb->style()->top().isAuto() && !cb->style()->bottom().isAuto())))
321             return false;
322     }
323
324     return true;
325 }
326
327 LayoutUnit RenderReplaced::computeReplacedLogicalHeight() const
328 {
329     // 10.5 Content height: the 'height' property: http://www.w3.org/TR/CSS21/visudet.html#propdef-height
330     bool heightIsAuto = logicalHeightIsAuto();
331     if (style()->logicalHeight().isSpecified() && !heightIsAuto)
332         return computeReplacedLogicalHeightRespectingMinMaxHeight(computeReplacedLogicalHeightUsing(style()->logicalHeight()));
333
334     RenderBox* contentRenderer = embeddedContentBox();
335     RenderStyle* contentRenderStyle = 0;
336
337     // 10.6.2 Inline, replaced elements: http://www.w3.org/TR/CSS21/visudet.html#inline-replaced-height
338     bool isPercentageIntrinsicSize = false;
339     FloatSize intrinsicRatio;
340     if (contentRenderer) {
341         contentRenderer->computeIntrinsicRatioInformation(intrinsicRatio, isPercentageIntrinsicSize);
342         contentRenderStyle = contentRenderer->style();
343         ASSERT(contentRenderStyle);
344     }
345
346     if (heightIsAuto) {
347         bool widthIsAuto = style()->logicalWidth().isAuto();
348         bool hasIntrinsicHeight = m_hasIntrinsicSize || (contentRenderer && contentRenderStyle->logicalHeight().isFixed());
349
350         // 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'.
351         if (widthIsAuto && hasIntrinsicHeight)
352             return computeIntrinsicLogicalHeight(contentRenderer);
353     
354         // Otherwise, if 'height' has a computed value of 'auto', and the element has an intrinsic ratio then the used value of 'height' is:
355         // (used width) / (intrinsic ratio)
356         if (!intrinsicRatio.isEmpty() && !isPercentageIntrinsicSize)
357             // FIXME: Remove unnecessary rounding when layout is off ints: webkit.org/b/63656
358             return computeReplacedLogicalHeightRespectingMinMaxHeight(static_cast<int>(ceilf(availableLogicalWidth() * intrinsicRatio.height() / intrinsicRatio.width())));
359
360         // 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'.
361         if (hasIntrinsicHeight)
362             return computeIntrinsicLogicalHeight(contentRenderer);
363
364         // 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
365         // 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.
366         return computeReplacedLogicalHeightRespectingMinMaxHeight(cDefaultHeight);
367     }
368
369     return computeReplacedLogicalHeightRespectingMinMaxHeight(intrinsicLogicalHeight());
370 }
371
372 int RenderReplaced::calcAspectRatioLogicalWidth() const
373 {
374     int intrinsicWidth = intrinsicLogicalWidth();
375     int intrinsicHeight = intrinsicLogicalHeight();
376     if (!intrinsicHeight)
377         return 0;
378     return RenderBox::computeReplacedLogicalHeight() * intrinsicWidth / intrinsicHeight;
379 }
380
381 int RenderReplaced::calcAspectRatioLogicalHeight() const
382 {
383     int intrinsicWidth = intrinsicLogicalWidth();
384     int intrinsicHeight = intrinsicLogicalHeight();
385     if (!intrinsicWidth)
386         return 0;
387     return RenderBox::computeReplacedLogicalWidth() * intrinsicHeight / intrinsicWidth;
388 }
389
390 void RenderReplaced::computePreferredLogicalWidths()
391 {
392     ASSERT(preferredLogicalWidthsDirty());
393
394     LayoutUnit borderAndPadding = borderAndPaddingWidth();
395     m_maxPreferredLogicalWidth = computeReplacedLogicalWidth(false) + borderAndPadding;
396
397     if (style()->maxWidth().isFixed())
398         m_maxPreferredLogicalWidth = min(m_maxPreferredLogicalWidth, style()->maxWidth().value() + (style()->boxSizing() == CONTENT_BOX ? borderAndPadding : 0));
399
400     if (style()->width().isPercent() || style()->height().isPercent()
401         || style()->maxWidth().isPercent() || style()->maxHeight().isPercent()
402         || style()->minWidth().isPercent() || style()->minHeight().isPercent())
403         m_minPreferredLogicalWidth = 0;
404     else
405         m_minPreferredLogicalWidth = m_maxPreferredLogicalWidth;
406
407     setPreferredLogicalWidthsDirty(false);
408 }
409
410 VisiblePosition RenderReplaced::positionForPoint(const LayoutPoint& point)
411 {
412     // FIXME: This code is buggy if the replaced element is relative positioned.
413     InlineBox* box = inlineBoxWrapper();
414     RootInlineBox* rootBox = box ? box->root() : 0;
415     
416     LayoutUnit top = rootBox ? rootBox->selectionTop() : logicalTop();
417     LayoutUnit bottom = rootBox ? rootBox->selectionBottom() : logicalBottom();
418     
419     LayoutUnit blockDirectionPosition = isHorizontalWritingMode() ? point.y() + y() : point.x() + x();
420     LayoutUnit lineDirectionPosition = isHorizontalWritingMode() ? point.x() + x() : point.y() + y();
421     
422     if (blockDirectionPosition < top)
423         return createVisiblePosition(caretMinOffset(), DOWNSTREAM); // coordinates are above
424     
425     if (blockDirectionPosition >= bottom)
426         return createVisiblePosition(caretMaxOffset(), DOWNSTREAM); // coordinates are below
427     
428     if (node()) {
429         if (lineDirectionPosition <= logicalLeft() + (logicalWidth() / 2))
430             return createVisiblePosition(0, DOWNSTREAM);
431         return createVisiblePosition(1, DOWNSTREAM);
432     }
433
434     return RenderBox::positionForPoint(point);
435 }
436
437 LayoutRect RenderReplaced::selectionRectForRepaint(RenderBoxModelObject* repaintContainer, bool clipToVisibleContent)
438 {
439     ASSERT(!needsLayout());
440
441     if (!isSelected())
442         return LayoutRect();
443     
444     LayoutRect rect = localSelectionRect();
445     if (clipToVisibleContent)
446         computeRectForRepaint(repaintContainer, rect);
447     else
448         rect = localToContainerQuad(FloatRect(rect), repaintContainer).enclosingBoundingBox();
449     
450     return rect;
451 }
452
453 IntRect RenderReplaced::localSelectionRect(bool checkWhetherSelected) const
454 {
455     if (checkWhetherSelected && !isSelected())
456         return IntRect();
457
458     if (!m_inlineBoxWrapper)
459         // We're a block-level replaced element.  Just return our own dimensions.
460         return IntRect(0, 0, width(), height());
461     
462     RootInlineBox* root = m_inlineBoxWrapper->root();
463     int newLogicalTop = root->block()->style()->isFlippedBlocksWritingMode() ? m_inlineBoxWrapper->logicalBottom() - root->selectionBottom() : root->selectionTop() - m_inlineBoxWrapper->logicalTop();
464     if (root->block()->style()->isHorizontalWritingMode())
465         return IntRect(0, newLogicalTop, width(), root->selectionHeight());
466     return IntRect(newLogicalTop, 0, root->selectionHeight(), height());
467 }
468
469 void RenderReplaced::setSelectionState(SelectionState s)
470 {
471     RenderBox::setSelectionState(s); // The selection state for our containing block hierarchy is updated by the base class call.
472     if (m_inlineBoxWrapper) {
473         RootInlineBox* line = m_inlineBoxWrapper->root();
474         if (line)
475             line->setHasSelectedChildren(isSelected());
476     }
477 }
478
479 bool RenderReplaced::isSelected() const
480 {
481     SelectionState s = selectionState();
482     if (s == SelectionNone)
483         return false;
484     if (s == SelectionInside)
485         return true;
486
487     int selectionStart, selectionEnd;
488     selectionStartEnd(selectionStart, selectionEnd);
489     if (s == SelectionStart)
490         return selectionStart == 0;
491         
492     int end = node()->hasChildNodes() ? node()->childNodeCount() : 1;
493     if (s == SelectionEnd)
494         return selectionEnd == end;
495     if (s == SelectionBoth)
496         return selectionStart == 0 && selectionEnd == end;
497         
498     ASSERT(0);
499     return false;
500 }
501
502 IntSize RenderReplaced::intrinsicSize() const
503 {
504     return m_intrinsicSize;
505 }
506
507 void RenderReplaced::setIntrinsicSize(const IntSize& size)
508 {
509     ASSERT(m_hasIntrinsicSize);
510     m_intrinsicSize = size;
511 }
512
513 LayoutRect RenderReplaced::clippedOverflowRectForRepaint(RenderBoxModelObject* repaintContainer) const
514 {
515     if (style()->visibility() != VISIBLE && !enclosingLayer()->hasVisibleContent())
516         return LayoutRect();
517
518     // The selectionRect can project outside of the overflowRect, so take their union
519     // for repainting to avoid selection painting glitches.
520     LayoutRect r = unionRect(localSelectionRect(false), visualOverflowRect());
521
522     RenderView* v = view();
523     if (v) {
524         // FIXME: layoutDelta needs to be applied in parts before/after transforms and
525         // repaint containers. https://bugs.webkit.org/show_bug.cgi?id=23308
526         r.move(v->layoutDelta());
527     }
528
529     if (style()) {
530         if (v)
531             r.inflate(style()->outlineSize());
532     }
533     computeRectForRepaint(repaintContainer, r);
534     return r;
535 }
536
537 }