/*
* Copyright (C) 2006 Nikolas Zimmermann <zimmermann@kde.org>
* 2008 Dirk Schulze <krit@webkit.org>
*
* 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 "SVGPaintServerPattern.h"
#include "GraphicsContext.h"
#include "Image.h"
#include "ImageBuffer.h"
#include "Pattern.h"
#include "RenderObject.h"
#include "SVGPatternElement.h"
#include "SVGRenderTreeAsText.h"
#include "TransformationMatrix.h"
using namespace std;
namespace WebCore {
SVGPaintServerPattern::SVGPaintServerPattern(const SVGPatternElement* owner)
: m_ownerElement(owner)
, m_pattern(0)
{
ASSERT(owner);
}
SVGPaintServerPattern::~SVGPaintServerPattern()
{
}
FloatRect SVGPaintServerPattern::patternBoundaries() const
{
return m_patternBoundaries;
}
void SVGPaintServerPattern::setPatternBoundaries(const FloatRect& rect)
{
m_patternBoundaries = rect;
}
ImageBuffer* SVGPaintServerPattern::tile() const
{
return m_tile.get();
}
void SVGPaintServerPattern::setTile(auto_ptr<ImageBuffer> tile)
{
m_tile.set(tile.release());
}
TransformationMatrix SVGPaintServerPattern::patternTransform() const
{
return m_patternTransform;
}
void SVGPaintServerPattern::setPatternTransform(const TransformationMatrix& transform)
{
m_patternTransform = transform;
}
TextStream& SVGPaintServerPattern::externalRepresentation(TextStream& ts) const
{
// Gradients/patterns aren't setup, until they are used for painting. Work around that fact.
m_ownerElement->buildPattern(FloatRect(0.0f, 0.0f, 1.0f, 1.0f));
ts << "[type=PATTERN]"
<< " [bbox=" << patternBoundaries() << "]";
if (!patternTransform().isIdentity())
ts << " [pattern transform=" << patternTransform() << "]";
return ts;
}
bool SVGPaintServerPattern::setup(GraphicsContext*& context, const RenderObject* object, SVGPaintTargetType type, bool isPaintingText) const
{
FloatRect targetRect;
if (isPaintingText) {
IntRect textBoundary = const_cast<RenderObject*>(object)->absoluteBoundingBoxRect();
targetRect = object->absoluteTransform().inverse().mapRect(textBoundary);
} else
targetRect = object->relativeBBox(false);
const SVGRenderStyle* style = object->style()->svgStyle();
bool isFilled = (type & ApplyToFillTargetType) && style->hasFill();
bool isStroked = (type & ApplyToStrokeTargetType) && style->hasStroke();
ASSERT(isFilled && !isStroked || !isFilled && isStroked);
m_ownerElement->buildPattern(targetRect);
if (!tile())
return false;
context->save();
context->translate(patternBoundaries().x(), patternBoundaries().y());
context->concatCTM(patternTransform());
ASSERT(!m_pattern);
IntRect tileRect = tile()->image()->rect();
if (tileRect.width() > patternBoundaries().width() || tileRect.height() > patternBoundaries().height()) {
// Draw the first cell of the pattern manually to support overflow="visible" on all platforms.
int tileWidth = static_cast<int>(patternBoundaries().width() + 0.5f);
int tileHeight = static_cast<int>(patternBoundaries().height() + 0.5f);
std::auto_ptr<ImageBuffer> tileImage = ImageBuffer::create(IntSize(tileWidth, tileHeight), false);
GraphicsContext* tileImageContext = tileImage->context();
int numY = static_cast<int>(ceilf(tileRect.height() / tileHeight)) + 1;
int numX = static_cast<int>(ceilf(tileRect.width() / tileWidth)) + 1;
tileImageContext->save();
tileImageContext->translate(-patternBoundaries().width() * numX, -patternBoundaries().height() * numY);
for (int i = numY; i > 0; i--) {
tileImageContext->translate(0, patternBoundaries().height());
for (int j = numX; j > 0; j--) {
tileImageContext->translate(patternBoundaries().width(), 0);
tileImageContext->drawImage(tile()->image(), tileRect, tileRect);
}
tileImageContext->translate(-patternBoundaries().width() * numX, 0);
}
tileImageContext->restore();
m_pattern = Pattern::create(tileImage->image(), true, true);
}
else
m_pattern = Pattern::create(tile()->image(), true, true);
if (isFilled) {
context->setAlpha(style->fillOpacity());
context->setFillPattern(m_pattern);
context->setFillRule(style->fillRule());
}
if (isStroked) {
context->setAlpha(style->strokeOpacity());
context->setStrokePattern(m_pattern);
applyStrokeStyleToContext(context, object->style(), object);
}
if (isPaintingText) {
context->setTextDrawingMode(isFilled ? cTextFill : cTextStroke);
#if PLATFORM(CG)
if (isFilled)
context->applyFillPattern();
else
context->applyStrokePattern();
#endif
}
return true;
}
void SVGPaintServerPattern::teardown(GraphicsContext*& context, const RenderObject*, SVGPaintTargetType, bool) const
{
m_pattern = 0;
context->restore();
}
} // namespace WebCore
#endif