| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (C) 2004, 2005, 2007, 2008 Nikolas Zimmermann <zimmermann@kde.org> | 2 * Copyright (C) 2004, 2005, 2007, 2008 Nikolas Zimmermann <zimmermann@kde.org> |
| 3 * Copyright (C) 2004, 2005, 2006, 2007, 2008 Rob Buis <buis@kde.org> | 3 * Copyright (C) 2004, 2005, 2006, 2007, 2008 Rob Buis <buis@kde.org> |
| 4 * Copyright (C) Research In Motion Limited 2009-2010. All rights reserved. | 4 * Copyright (C) Research In Motion Limited 2009-2010. All rights reserved. |
| 5 * Copyright (C) 2011 Dirk Schulze <krit@webkit.org> | 5 * Copyright (C) 2011 Dirk Schulze <krit@webkit.org> |
| 6 * | 6 * |
| 7 * This library is free software; you can redistribute it and/or | 7 * This library is free software; you can redistribute it and/or |
| 8 * modify it under the terms of the GNU Library General Public | 8 * modify it under the terms of the GNU Library General Public |
| 9 * License as published by the Free Software Foundation; either | 9 * License as published by the Free Software Foundation; either |
| 10 * version 2 of the License, or (at your option) any later version. | 10 * version 2 of the License, or (at your option) any later version. |
| (...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 82 } | 82 } |
| 83 | 83 |
| 84 bool RenderSVGResourceClipper::tryPathOnlyClipping(GraphicsContext* context, | 84 bool RenderSVGResourceClipper::tryPathOnlyClipping(GraphicsContext* context, |
| 85 const AffineTransform& animatedLocalTransform, const FloatRect& objectBoundi
ngBox) { | 85 const AffineTransform& animatedLocalTransform, const FloatRect& objectBoundi
ngBox) { |
| 86 // If the current clip-path gets clipped itself, we have to fallback to mask
ing. | 86 // If the current clip-path gets clipped itself, we have to fallback to mask
ing. |
| 87 if (!style()->svgStyle()->clipperResource().isEmpty()) | 87 if (!style()->svgStyle()->clipperResource().isEmpty()) |
| 88 return false; | 88 return false; |
| 89 WindRule clipRule = RULE_NONZERO; | 89 WindRule clipRule = RULE_NONZERO; |
| 90 Path clipPath = Path(); | 90 Path clipPath = Path(); |
| 91 | 91 |
| 92 for (Node* childNode = element()->firstChild(); childNode; childNode = child
Node->nextSibling()) { | 92 for (Element* childElement = ElementTraversal::firstWithin(*element()); chil
dElement; childElement = ElementTraversal::nextSibling(*childElement)) { |
| 93 RenderObject* renderer = childNode->renderer(); | 93 RenderObject* renderer = childElement->renderer(); |
| 94 if (!renderer) | 94 if (!renderer) |
| 95 continue; | 95 continue; |
| 96 // Only shapes or paths are supported for direct clipping. We need to fa
llback to masking for texts. | 96 // Only shapes or paths are supported for direct clipping. We need to fa
llback to masking for texts. |
| 97 if (renderer->isSVGText()) | 97 if (renderer->isSVGText()) |
| 98 return false; | 98 return false; |
| 99 if (!childNode->isSVGElement() || !toSVGElement(childNode)->isSVGGraphic
sElement()) | 99 if (!childElement->isSVGElement() || !toSVGElement(childElement)->isSVGG
raphicsElement()) |
| 100 continue; | 100 continue; |
| 101 SVGGraphicsElement* styled = toSVGGraphicsElement(childNode); | 101 SVGGraphicsElement* styled = toSVGGraphicsElement(childElement); |
| 102 RenderStyle* style = renderer->style(); | 102 RenderStyle* style = renderer->style(); |
| 103 if (!style || style->display() == NONE || style->visibility() != VISIBLE
) | 103 if (!style || style->display() == NONE || style->visibility() != VISIBLE
) |
| 104 continue; | 104 continue; |
| 105 const SVGRenderStyle* svgStyle = style->svgStyle(); | 105 const SVGRenderStyle* svgStyle = style->svgStyle(); |
| 106 // Current shape in clip-path gets clipped too. Fallback to masking. | 106 // Current shape in clip-path gets clipped too. Fallback to masking. |
| 107 if (!svgStyle->clipperResource().isEmpty()) | 107 if (!svgStyle->clipperResource().isEmpty()) |
| 108 return false; | 108 return false; |
| 109 | 109 |
| 110 if (clipPath.isEmpty()) { | 110 if (clipPath.isEmpty()) { |
| 111 // First clip shape. | 111 // First clip shape. |
| (...skipping 152 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 264 context->beginRecording(strokeBoundingBox()); | 264 context->beginRecording(strokeBoundingBox()); |
| 265 | 265 |
| 266 // Switch to a paint behavior where all children of this <clipPath> will be
rendered using special constraints: | 266 // Switch to a paint behavior where all children of this <clipPath> will be
rendered using special constraints: |
| 267 // - fill-opacity/stroke-opacity/opacity set to 1 | 267 // - fill-opacity/stroke-opacity/opacity set to 1 |
| 268 // - masker/filter not applied when rendering the children | 268 // - masker/filter not applied when rendering the children |
| 269 // - fill is set to the initial fill paint server (solid, black) | 269 // - fill is set to the initial fill paint server (solid, black) |
| 270 // - stroke is set to the initial stroke paint server (none) | 270 // - stroke is set to the initial stroke paint server (none) |
| 271 PaintBehavior oldBehavior = frame()->view()->paintBehavior(); | 271 PaintBehavior oldBehavior = frame()->view()->paintBehavior(); |
| 272 frame()->view()->setPaintBehavior(oldBehavior | PaintBehaviorRenderingSVGMas
k); | 272 frame()->view()->setPaintBehavior(oldBehavior | PaintBehaviorRenderingSVGMas
k); |
| 273 | 273 |
| 274 for (Node* childNode = element()->firstChild(); childNode; childNode = child
Node->nextSibling()) { | 274 for (Element* childElement = ElementTraversal::firstWithin(*element()); chil
dElement; childElement = ElementTraversal::nextSibling(*childElement)) { |
| 275 RenderObject* renderer = childNode->renderer(); | 275 RenderObject* renderer = childElement->renderer(); |
| 276 if (!childNode->isSVGElement() || !renderer) | 276 if (!childElement->isSVGElement() || !renderer) |
| 277 continue; | 277 continue; |
| 278 | 278 |
| 279 RenderStyle* style = renderer->style(); | 279 RenderStyle* style = renderer->style(); |
| 280 if (!style || style->display() == NONE || style->visibility() != VISIBLE
) | 280 if (!style || style->display() == NONE || style->visibility() != VISIBLE
) |
| 281 continue; | 281 continue; |
| 282 | 282 |
| 283 WindRule newClipRule = style->svgStyle()->clipRule(); | 283 WindRule newClipRule = style->svgStyle()->clipRule(); |
| 284 bool isUseElement = childNode->hasTagName(SVGNames::useTag); | 284 bool isUseElement = childElement->hasTagName(SVGNames::useTag); |
| 285 if (isUseElement) { | 285 if (isUseElement) { |
| 286 SVGUseElement* useElement = toSVGUseElement(childNode); | 286 SVGUseElement* useElement = toSVGUseElement(childElement); |
| 287 renderer = useElement->rendererClipChild(); | 287 renderer = useElement->rendererClipChild(); |
| 288 if (!renderer) | 288 if (!renderer) |
| 289 continue; | 289 continue; |
| 290 if (!useElement->hasAttribute(SVGNames::clip_ruleAttr)) | 290 if (!useElement->hasAttribute(SVGNames::clip_ruleAttr)) |
| 291 newClipRule = renderer->style()->svgStyle()->clipRule(); | 291 newClipRule = renderer->style()->svgStyle()->clipRule(); |
| 292 } | 292 } |
| 293 | 293 |
| 294 // Only shapes, paths and texts are allowed for clipping. | 294 // Only shapes, paths and texts are allowed for clipping. |
| 295 if (!renderer->isSVGShape() && !renderer->isSVGText()) | 295 if (!renderer->isSVGShape() && !renderer->isSVGText()) |
| 296 continue; | 296 continue; |
| 297 | 297 |
| 298 context->setFillRule(newClipRule); | 298 context->setFillRule(newClipRule); |
| 299 | 299 |
| 300 if (isUseElement) | 300 if (isUseElement) |
| 301 renderer = childNode->renderer(); | 301 renderer = childElement->renderer(); |
| 302 | 302 |
| 303 SVGRenderingContext::renderSubtree(context, renderer, contentTransformat
ion); | 303 SVGRenderingContext::renderSubtree(context, renderer, contentTransformat
ion); |
| 304 } | 304 } |
| 305 | 305 |
| 306 frame()->view()->setPaintBehavior(oldBehavior); | 306 frame()->view()->setPaintBehavior(oldBehavior); |
| 307 | 307 |
| 308 return context->endRecording(); | 308 return context->endRecording(); |
| 309 } | 309 } |
| 310 | 310 |
| 311 void RenderSVGResourceClipper::calculateClipContentRepaintRect() | 311 void RenderSVGResourceClipper::calculateClipContentRepaintRect() |
| 312 { | 312 { |
| 313 // This is a rough heuristic to appraise the clip size and doesn't consider
clip on clip. | 313 // This is a rough heuristic to appraise the clip size and doesn't consider
clip on clip. |
| 314 for (Node* childNode = element()->firstChild(); childNode; childNode = child
Node->nextSibling()) { | 314 for (Element* childElement = ElementTraversal::firstWithin(*element()); chil
dElement; childElement = ElementTraversal::nextSibling(*childElement)) { |
| 315 RenderObject* renderer = childNode->renderer(); | 315 RenderObject* renderer = childElement->renderer(); |
| 316 if (!childNode->isSVGElement() || !renderer) | 316 if (!childElement->isSVGElement() || !renderer) |
| 317 continue; | 317 continue; |
| 318 if (!renderer->isSVGShape() && !renderer->isSVGText() && !childNode->has
TagName(SVGNames::useTag)) | 318 if (!renderer->isSVGShape() && !renderer->isSVGText() && !childElement->
hasTagName(SVGNames::useTag)) |
| 319 continue; | 319 continue; |
| 320 RenderStyle* style = renderer->style(); | 320 RenderStyle* style = renderer->style(); |
| 321 if (!style || style->display() == NONE || style->visibility() != VISIBLE
) | 321 if (!style || style->display() == NONE || style->visibility() != VISIBLE
) |
| 322 continue; | 322 continue; |
| 323 m_clipBoundaries.unite(renderer->localToParentTransform().mapRect(render
er->repaintRectInLocalCoordinates())); | 323 m_clipBoundaries.unite(renderer->localToParentTransform().mapRect(render
er->repaintRectInLocalCoordinates())); |
| 324 } | 324 } |
| 325 m_clipBoundaries = toSVGClipPathElement(element())->animatedLocalTransform()
.mapRect(m_clipBoundaries); | 325 m_clipBoundaries = toSVGClipPathElement(element())->animatedLocalTransform()
.mapRect(m_clipBoundaries); |
| 326 } | 326 } |
| 327 | 327 |
| 328 bool RenderSVGResourceClipper::hitTestClipContent(const FloatRect& objectBoundin
gBox, const FloatPoint& nodeAtPoint) | 328 bool RenderSVGResourceClipper::hitTestClipContent(const FloatRect& objectBoundin
gBox, const FloatPoint& nodeAtPoint) |
| 329 { | 329 { |
| 330 FloatPoint point = nodeAtPoint; | 330 FloatPoint point = nodeAtPoint; |
| 331 if (!SVGRenderSupport::pointInClippingArea(this, point)) | 331 if (!SVGRenderSupport::pointInClippingArea(this, point)) |
| 332 return false; | 332 return false; |
| 333 | 333 |
| 334 SVGClipPathElement* clipPathElement = toSVGClipPathElement(element()); | 334 SVGClipPathElement* clipPathElement = toSVGClipPathElement(element()); |
| 335 if (clipPathElement->clipPathUnitsCurrentValue() == SVGUnitTypes::SVG_UNIT_T
YPE_OBJECTBOUNDINGBOX) { | 335 if (clipPathElement->clipPathUnitsCurrentValue() == SVGUnitTypes::SVG_UNIT_T
YPE_OBJECTBOUNDINGBOX) { |
| 336 AffineTransform transform; | 336 AffineTransform transform; |
| 337 transform.translate(objectBoundingBox.x(), objectBoundingBox.y()); | 337 transform.translate(objectBoundingBox.x(), objectBoundingBox.y()); |
| 338 transform.scaleNonUniform(objectBoundingBox.width(), objectBoundingBox.h
eight()); | 338 transform.scaleNonUniform(objectBoundingBox.width(), objectBoundingBox.h
eight()); |
| 339 point = transform.inverse().mapPoint(point); | 339 point = transform.inverse().mapPoint(point); |
| 340 } | 340 } |
| 341 | 341 |
| 342 point = clipPathElement->animatedLocalTransform().inverse().mapPoint(point); | 342 point = clipPathElement->animatedLocalTransform().inverse().mapPoint(point); |
| 343 | 343 |
| 344 for (Node* childNode = element()->firstChild(); childNode; childNode = child
Node->nextSibling()) { | 344 for (Element* childElement = ElementTraversal::firstWithin(*element()); chil
dElement; childElement = ElementTraversal::nextSibling(*childElement)) { |
| 345 RenderObject* renderer = childNode->renderer(); | 345 RenderObject* renderer = childElement->renderer(); |
| 346 if (!childNode->isSVGElement() || !renderer) | 346 if (!childElement->isSVGElement() || !renderer) |
| 347 continue; | 347 continue; |
| 348 if (!renderer->isSVGShape() && !renderer->isSVGText() && !childNode->has
TagName(SVGNames::useTag)) | 348 if (!renderer->isSVGShape() && !renderer->isSVGText() && !childElement->
hasTagName(SVGNames::useTag)) |
| 349 continue; | 349 continue; |
| 350 IntPoint hitPoint; | 350 IntPoint hitPoint; |
| 351 HitTestResult result(hitPoint); | 351 HitTestResult result(hitPoint); |
| 352 if (renderer->nodeAtFloatPoint(HitTestRequest(HitTestRequest::SVGClipCon
tent | HitTestRequest::ConfusingAndOftenMisusedDisallowShadowContent), result, p
oint, HitTestForeground)) | 352 if (renderer->nodeAtFloatPoint(HitTestRequest(HitTestRequest::SVGClipCon
tent | HitTestRequest::ConfusingAndOftenMisusedDisallowShadowContent), result, p
oint, HitTestForeground)) |
| 353 return true; | 353 return true; |
| 354 } | 354 } |
| 355 | 355 |
| 356 return false; | 356 return false; |
| 357 } | 357 } |
| 358 | 358 |
| (...skipping 11 matching lines...) Expand all Loading... |
| 370 AffineTransform transform; | 370 AffineTransform transform; |
| 371 transform.translate(objectBoundingBox.x(), objectBoundingBox.y()); | 371 transform.translate(objectBoundingBox.x(), objectBoundingBox.y()); |
| 372 transform.scaleNonUniform(objectBoundingBox.width(), objectBoundingBox.h
eight()); | 372 transform.scaleNonUniform(objectBoundingBox.width(), objectBoundingBox.h
eight()); |
| 373 return transform.mapRect(m_clipBoundaries); | 373 return transform.mapRect(m_clipBoundaries); |
| 374 } | 374 } |
| 375 | 375 |
| 376 return m_clipBoundaries; | 376 return m_clipBoundaries; |
| 377 } | 377 } |
| 378 | 378 |
| 379 } | 379 } |
| OLD | NEW |