blob: 330de55b8106404a1de7d5d9c13c10b87f8de891 [file] [log] [blame]
/*
* Copyright (C) Research In Motion Limited 2010. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "core/svg/SVGPathByteStreamBuilder.h"
#include "core/svg/SVGPathSeg.h"
#include "wtf/OwnPtr.h"
namespace blink {
// Helper class that coalesces writes to a SVGPathByteStream to a local buffer.
class CoalescingBuffer {
public:
CoalescingBuffer(SVGPathByteStream* byteStream)
: m_currentOffset(0)
, m_byteStream(byteStream)
{
ASSERT(byteStream);
}
~CoalescingBuffer()
{
for (size_t i = 0; i < m_currentOffset; ++i)
m_byteStream->append(m_bytes[i]);
}
template<typename DataType>
void writeType(DataType value)
{
ByteType<DataType> data;
data.value = value;
size_t typeSize = sizeof(ByteType<DataType>);
ASSERT(m_currentOffset + typeSize <= sizeof(m_bytes));
memcpy(m_bytes + m_currentOffset, data.bytes, typeSize);
m_currentOffset += typeSize;
}
void writeFlag(bool value) { writeType<bool>(value); }
void writeFloat(float value) { writeType<float>(value); }
void writeFloatPoint(const FloatPoint& point)
{
writeType<float>(point.x());
writeType<float>(point.y());
}
void writeSegmentType(unsigned short value) { writeType<unsigned short>(value); }
private:
// Adjust size to fit the largest command (in serialized/byte-stream format).
// Currently a cubic segment.
size_t m_currentOffset;
unsigned char m_bytes[sizeof(unsigned short) + sizeof(FloatPoint) * 3];
SVGPathByteStream* m_byteStream;
};
SVGPathByteStreamBuilder::SVGPathByteStreamBuilder()
: m_byteStream(0)
{
}
void SVGPathByteStreamBuilder::moveTo(const FloatPoint& targetPoint, bool, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegMoveToRel : PathSegMoveToAbs);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::lineTo(const FloatPoint& targetPoint, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegLineToRel : PathSegLineToAbs);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::lineToHorizontal(float x, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegLineToHorizontalRel : PathSegLineToHorizontalAbs);
buffer.writeFloat(x);
}
void SVGPathByteStreamBuilder::lineToVertical(float y, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegLineToVerticalRel : PathSegLineToVerticalAbs);
buffer.writeFloat(y);
}
void SVGPathByteStreamBuilder::curveToCubic(const FloatPoint& point1, const FloatPoint& point2, const FloatPoint& targetPoint, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegCurveToCubicRel : PathSegCurveToCubicAbs);
buffer.writeFloatPoint(point1);
buffer.writeFloatPoint(point2);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::curveToCubicSmooth(const FloatPoint& point2, const FloatPoint& targetPoint, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegCurveToCubicSmoothRel : PathSegCurveToCubicSmoothAbs);
buffer.writeFloatPoint(point2);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::curveToQuadratic(const FloatPoint& point1, const FloatPoint& targetPoint, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegCurveToQuadraticRel : PathSegCurveToQuadraticAbs);
buffer.writeFloatPoint(point1);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::curveToQuadraticSmooth(const FloatPoint& targetPoint, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegCurveToQuadraticSmoothRel : PathSegCurveToQuadraticSmoothAbs);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::arcTo(float r1, float r2, float angle, bool largeArcFlag, bool sweepFlag, const FloatPoint& targetPoint, PathCoordinateMode mode)
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(mode == RelativeCoordinates ? PathSegArcRel : PathSegArcAbs);
buffer.writeFloat(r1);
buffer.writeFloat(r2);
buffer.writeFloat(angle);
buffer.writeFlag(largeArcFlag);
buffer.writeFlag(sweepFlag);
buffer.writeFloatPoint(targetPoint);
}
void SVGPathByteStreamBuilder::closePath()
{
CoalescingBuffer buffer(m_byteStream);
buffer.writeSegmentType(PathSegClosePath);
}
} // namespace blink