/* * Copyright (C) 2006 Nikolas Zimmermann * 2008 Eric Seidel * 2008 Dirk Schulze * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #if ENABLE(SVG) #include "SVGPaintServerGradient.h" #include "FloatConversion.h" #include "GraphicsContext.h" #include "ImageBuffer.h" #include "RenderObject.h" #include "RenderView.h" #include "SVGGradientElement.h" #include "SVGPaintServerLinearGradient.h" #include "SVGPaintServerRadialGradient.h" #include "SVGRenderSupport.h" #include "SVGRenderTreeAsText.h" using namespace std; namespace WebCore { static TextStream& operator<<(TextStream& ts, GradientSpreadMethod m) { switch (m) { case SpreadMethodPad: ts << "PAD"; break; case SpreadMethodRepeat: ts << "REPEAT"; break; case SpreadMethodReflect: ts << "REFLECT"; break; } return ts; } static TextStream& operator<<(TextStream& ts, const Vector& l) { ts << "["; for (Vector::const_iterator it = l.begin(); it != l.end(); ++it) { ts << "(" << it->first << "," << it->second << ")"; if (it + 1 != l.end()) ts << ", "; } ts << "]"; return ts; } SVGPaintServerGradient::SVGPaintServerGradient(const SVGGradientElement* owner) : m_boundingBoxMode(true) , m_ownerElement(owner) #if PLATFORM(CG) , m_savedContext(0) , m_imageBuffer(0) #endif { ASSERT(owner); } SVGPaintServerGradient::~SVGPaintServerGradient() { } Gradient* SVGPaintServerGradient::gradient() const { return m_gradient.get(); } void SVGPaintServerGradient::setGradient(PassRefPtr gradient) { m_gradient = gradient; } bool SVGPaintServerGradient::boundingBoxMode() const { return m_boundingBoxMode; } void SVGPaintServerGradient::setBoundingBoxMode(bool mode) { m_boundingBoxMode = mode; } AffineTransform SVGPaintServerGradient::gradientTransform() const { return m_gradientTransform; } void SVGPaintServerGradient::setGradientTransform(const AffineTransform& transform) { m_gradientTransform = transform; } #if PLATFORM(CG) static inline const RenderObject* findTextRootObject(const RenderObject* start) { while (start && !start->isSVGText()) start = start->parent(); ASSERT(start); ASSERT(start->isSVGText()); return start; } static inline AffineTransform absoluteTransformForRenderer(const RenderObject* object) { AffineTransform absoluteTransform; const RenderObject* currentObject = object; while (currentObject) { absoluteTransform = currentObject->localToParentTransform() * absoluteTransform; currentObject = currentObject->parent(); } return absoluteTransform; } static inline bool createMaskAndSwapContextForTextGradient( GraphicsContext*& context, GraphicsContext*& savedContext, OwnPtr& imageBuffer, const RenderObject* object) { const RenderObject* textRootBlock = findTextRootObject(object); AffineTransform transform = absoluteTransformForRenderer(textRootBlock); FloatRect maskAbsoluteBoundingBox = transform.mapRect(textRootBlock->repaintRectInLocalCoordinates()); IntRect maskImageRect = enclosingIntRect(maskAbsoluteBoundingBox); if (maskImageRect.isEmpty()) return false; // Allocate an image buffer as big as the absolute unclipped size of the object OwnPtr maskImage = ImageBuffer::create(maskImageRect.size()); if (!maskImage) return false; GraphicsContext* maskImageContext = maskImage->context(); // Transform the mask image coordinate system to absolute screen coordinates maskImageContext->translate(-maskAbsoluteBoundingBox.x(), -maskAbsoluteBoundingBox.y()); maskImageContext->concatCTM(transform); imageBuffer.set(maskImage.release()); savedContext = context; context = maskImageContext; return true; } static inline AffineTransform clipToTextMask(GraphicsContext* context, OwnPtr& imageBuffer, const RenderObject* object, const SVGPaintServerGradient* gradientServer) { const RenderObject* textRootBlock = findTextRootObject(object); context->clipToImageBuffer(textRootBlock->repaintRectInLocalCoordinates(), imageBuffer.get()); AffineTransform matrix; if (gradientServer->boundingBoxMode()) { FloatRect maskBoundingBox = textRootBlock->objectBoundingBox(); matrix.translate(maskBoundingBox.x(), maskBoundingBox.y()); matrix.scaleNonUniform(maskBoundingBox.width(), maskBoundingBox.height()); } matrix.multiply(gradientServer->gradientTransform()); return matrix; } #endif bool SVGPaintServerGradient::setup(GraphicsContext*& context, const RenderObject* object, const RenderStyle*style, SVGPaintTargetType type, bool isPaintingText) const { m_ownerElement->buildGradient(); const SVGRenderStyle* svgStyle = style->svgStyle(); bool isFilled = (type & ApplyToFillTargetType) && svgStyle->hasFill(); bool isStroked = (type & ApplyToStrokeTargetType) && svgStyle->hasStroke(); ASSERT((isFilled && !isStroked) || (!isFilled && isStroked)); context->save(); if (isPaintingText) { #if PLATFORM(CG) if (!createMaskAndSwapContextForTextGradient(context, m_savedContext, m_imageBuffer, object)) { context->restore(); return false; } #endif context->setTextDrawingMode(isFilled ? cTextFill : cTextStroke); } if (isFilled) { context->setAlpha(svgStyle->fillOpacity()); context->setFillGradient(m_gradient); context->setFillRule(svgStyle->fillRule()); } if (isStroked) { context->setAlpha(svgStyle->strokeOpacity()); context->setStrokeGradient(m_gradient); applyStrokeStyleToContext(context, style, object); } AffineTransform matrix; // CG platforms will handle the gradient space transform for text in // teardown, so we don't apply it here. For non-CG platforms, we // want the text bounding box applied to the gradient space transform now, // so the gradient shader can use it. #if PLATFORM(CG) if (boundingBoxMode() && !isPaintingText) { #else if (boundingBoxMode()) { #endif FloatRect bbox = object->objectBoundingBox(); // Don't use gradients for 1d objects like horizontal/vertical // lines or rectangles without width or height. if (bbox.width() == 0 || bbox.height() == 0) { Color color(0, 0, 0); context->setStrokeColor(color, style->colorSpace()); return true; } matrix.translate(bbox.x(), bbox.y()); matrix.scaleNonUniform(bbox.width(), bbox.height()); } matrix.multiply(gradientTransform()); m_gradient->setGradientSpaceTransform(matrix); return true; } void SVGPaintServerGradient::teardown(GraphicsContext*& context, const RenderObject* object, SVGPaintTargetType, bool isPaintingText) const { #if PLATFORM(CG) // renderPath() is not used when painting text, so we paint the gradient during teardown() if (isPaintingText && m_savedContext) { // Restore on-screen drawing context context = m_savedContext; m_savedContext = 0; AffineTransform matrix = clipToTextMask(context, m_imageBuffer, object, this); m_gradient->setGradientSpaceTransform(matrix); context->setFillGradient(m_gradient); const RenderObject* textRootBlock = findTextRootObject(object); context->fillRect(textRootBlock->repaintRectInLocalCoordinates()); m_imageBuffer.clear(); } #endif context->restore(); } TextStream& SVGPaintServerGradient::externalRepresentation(TextStream& ts) const { // Gradients/patterns aren't setup, until they are used for painting. Work around that fact. m_ownerElement->buildGradient(); // abstract, don't stream type ts << "[stops=" << gradientStops() << "]"; if (m_gradient->spreadMethod() != SpreadMethodPad) ts << "[method=" << m_gradient->spreadMethod() << "]"; if (!boundingBoxMode()) ts << " [bounding box mode=" << boundingBoxMode() << "]"; if (!gradientTransform().isIdentity()) ts << " [transform=" << gradientTransform() << "]"; return ts; } } // namespace WebCore #endif