This source file includes following definitions.
- inheritColorFromParentStyle
- requestPaintingResource
- fillPaintingResource
- strokePaintingResource
- sharedSolidPaintingResource
- removeFromCacheAndInvalidateDependencies
- markForLayoutAndParentResourceInvalidation
#include "config.h"
#include "core/rendering/svg/RenderSVGResource.h"
#include "core/frame/FrameView.h"
#include "core/frame/LocalFrame.h"
#include "core/rendering/svg/RenderSVGResourceClipper.h"
#include "core/rendering/svg/RenderSVGResourceFilter.h"
#include "core/rendering/svg/RenderSVGResourceMasker.h"
#include "core/rendering/svg/RenderSVGResourceSolidColor.h"
#include "core/rendering/svg/SVGResources.h"
#include "core/rendering/svg/SVGResourcesCache.h"
namespace WebCore {
static inline bool inheritColorFromParentStyle(RenderObject* object, bool applyToFill, Color& color)
{
if (!object->parent() || !object->parent()->style())
return false;
const SVGRenderStyle* parentSVGStyle = object->parent()->style()->svgStyle();
SVGPaint::SVGPaintType paintType = applyToFill ? parentSVGStyle->fillPaintType() : parentSVGStyle->strokePaintType();
if (paintType != SVGPaint::SVG_PAINTTYPE_RGBCOLOR && paintType != SVGPaint::SVG_PAINTTYPE_RGBCOLOR_ICCCOLOR)
return false;
color = applyToFill ? parentSVGStyle->fillPaintColor() : parentSVGStyle->strokePaintColor();
return true;
}
static inline RenderSVGResource* requestPaintingResource(RenderSVGResourceMode mode, RenderObject* object, const RenderStyle* style, bool& hasFallback)
{
ASSERT(object);
ASSERT(style);
hasFallback = false;
const SVGRenderStyle* svgStyle = style->svgStyle();
if (!svgStyle)
return 0;
bool isRenderingMask = false;
if (object->frame() && object->frame()->view())
isRenderingMask = object->frame()->view()->paintBehavior() & PaintBehaviorRenderingSVGMask;
if (mode == ApplyToFillMode) {
if (isRenderingMask) {
RenderSVGResourceSolidColor* colorResource = RenderSVGResource::sharedSolidPaintingResource();
colorResource->setColor(SVGRenderStyle::initialFillPaintColor());
return colorResource;
}
if (!svgStyle->hasFill())
return 0;
} else {
if (!svgStyle->hasStroke() || isRenderingMask)
return 0;
}
bool applyToFill = mode == ApplyToFillMode;
SVGPaint::SVGPaintType paintType = applyToFill ? svgStyle->fillPaintType() : svgStyle->strokePaintType();
ASSERT(paintType != SVGPaint::SVG_PAINTTYPE_NONE);
Color color;
bool hasColor = false;
switch (paintType) {
case SVGPaint::SVG_PAINTTYPE_CURRENTCOLOR:
case SVGPaint::SVG_PAINTTYPE_RGBCOLOR:
case SVGPaint::SVG_PAINTTYPE_RGBCOLOR_ICCCOLOR:
case SVGPaint::SVG_PAINTTYPE_URI_CURRENTCOLOR:
case SVGPaint::SVG_PAINTTYPE_URI_RGBCOLOR:
case SVGPaint::SVG_PAINTTYPE_URI_RGBCOLOR_ICCCOLOR:
color = applyToFill ? svgStyle->fillPaintColor() : svgStyle->strokePaintColor();
hasColor = true;
default:
break;
}
if (style->insideLink() == InsideVisitedLink) {
SVGPaint::SVGPaintType visitedPaintType = applyToFill ? svgStyle->visitedLinkFillPaintType() : svgStyle->visitedLinkStrokePaintType();
if (visitedPaintType < SVGPaint::SVG_PAINTTYPE_URI_NONE && visitedPaintType != SVGPaint::SVG_PAINTTYPE_CURRENTCOLOR) {
const Color& visitedColor = applyToFill ? svgStyle->visitedLinkFillPaintColor() : svgStyle->visitedLinkStrokePaintColor();
color = Color(visitedColor.red(), visitedColor.green(), visitedColor.blue(), color.alpha());
hasColor = true;
}
}
RenderSVGResourceSolidColor* colorResource = RenderSVGResource::sharedSolidPaintingResource();
if (paintType < SVGPaint::SVG_PAINTTYPE_URI_NONE) {
if (!hasColor && !inheritColorFromParentStyle(object, applyToFill, color))
return 0;
colorResource->setColor(color);
return colorResource;
}
SVGResources* resources = SVGResourcesCache::cachedResourcesForRenderObject(object);
if (!resources) {
if (paintType == SVGPaint::SVG_PAINTTYPE_URI_NONE || (!hasColor && !inheritColorFromParentStyle(object, applyToFill, color)))
return 0;
colorResource->setColor(color);
return colorResource;
}
RenderSVGResource* uriResource = mode == ApplyToFillMode ? resources->fill() : resources->stroke();
if (!uriResource) {
if (!hasColor && !inheritColorFromParentStyle(object, applyToFill, color))
return 0;
colorResource->setColor(color);
return colorResource;
}
if (hasColor) {
colorResource->setColor(color);
hasFallback = true;
}
return uriResource;
}
RenderSVGResource* RenderSVGResource::fillPaintingResource(RenderObject* object, const RenderStyle* style, bool& hasFallback)
{
return requestPaintingResource(ApplyToFillMode, object, style, hasFallback);
}
RenderSVGResource* RenderSVGResource::strokePaintingResource(RenderObject* object, const RenderStyle* style, bool& hasFallback)
{
return requestPaintingResource(ApplyToStrokeMode, object, style, hasFallback);
}
RenderSVGResourceSolidColor* RenderSVGResource::sharedSolidPaintingResource()
{
static RenderSVGResourceSolidColor* s_sharedSolidPaintingResource = 0;
if (!s_sharedSolidPaintingResource)
s_sharedSolidPaintingResource = new RenderSVGResourceSolidColor;
return s_sharedSolidPaintingResource;
}
static inline void removeFromCacheAndInvalidateDependencies(RenderObject* object, bool needsLayout)
{
ASSERT(object);
if (SVGResources* resources = SVGResourcesCache::cachedResourcesForRenderObject(object)) {
if (RenderSVGResourceFilter* filter = resources->filter())
filter->removeClientFromCache(object);
if (RenderSVGResourceMasker* masker = resources->masker())
masker->removeClientFromCache(object);
if (RenderSVGResourceClipper* clipper = resources->clipper())
clipper->removeClientFromCache(object);
}
if (!object->node() || !object->node()->isSVGElement())
return;
HashSet<SVGElement*>* dependencies = object->document().accessSVGExtensions().setOfElementsReferencingTarget(toSVGElement(object->node()));
if (!dependencies)
return;
HashSet<SVGElement*>::iterator end = dependencies->end();
for (HashSet<SVGElement*>::iterator it = dependencies->begin(); it != end; ++it) {
if (RenderObject* renderer = (*it)->renderer())
RenderSVGResource::markForLayoutAndParentResourceInvalidation(renderer, needsLayout);
}
}
void RenderSVGResource::markForLayoutAndParentResourceInvalidation(RenderObject* object, bool needsLayout)
{
ASSERT(object);
ASSERT(object->node());
if (needsLayout && !object->documentBeingDestroyed())
object->setNeedsLayout();
removeFromCacheAndInvalidateDependencies(object, needsLayout);
RenderObject* current = object->parent();
while (current) {
removeFromCacheAndInvalidateDependencies(current, needsLayout);
if (current->isSVGResourceContainer()) {
toRenderSVGResourceContainer(current)->removeAllClientsFromCache();
break;
}
current = current->parent();
}
}
}