blob: cf9f01427e116a20fbc4dbc1c41988898f11a205 [file] [log] [blame]
/*
* Copyright (C) Research In Motion Limited 2010, 2012. All rights reserved.
* Copyright (C) 2015 Apple Inc. 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 "SVGPathUtilities.h"
#include "FloatPoint.h"
#include "Path.h"
#include "PathTraversalState.h"
#include "SVGPathBlender.h"
#include "SVGPathBuilder.h"
#include "SVGPathByteStreamBuilder.h"
#include "SVGPathByteStreamSource.h"
#include "SVGPathConsumer.h"
#include "SVGPathElement.h"
#include "SVGPathParser.h"
#include "SVGPathSegListBuilder.h"
#include "SVGPathSegListSource.h"
#include "SVGPathStringBuilder.h"
#include "SVGPathStringSource.h"
#include "SVGPathTraversalStateBuilder.h"
namespace WebCore {
Path buildPathFromString(const String& d)
{
if (d.isEmpty())
return { };
Path path;
SVGPathBuilder builder(path);
SVGPathStringSource source(d);
SVGPathParser::parse(source, builder);
return path;
}
String buildStringFromPath(const Path& path)
{
StringBuilder builder;
if (!path.isNull() && !path.isEmpty()) {
path.apply([&builder] (const PathElement& element) {
switch (element.type) {
case PathElementMoveToPoint:
builder.append('M');
builder.appendECMAScriptNumber(element.points[0].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[0].y());
break;
case PathElementAddLineToPoint:
builder.append('L');
builder.appendECMAScriptNumber(element.points[0].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[0].y());
break;
case PathElementAddQuadCurveToPoint:
builder.append('Q');
builder.appendECMAScriptNumber(element.points[0].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[0].y());
builder.append(',');
builder.appendECMAScriptNumber(element.points[1].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[1].y());
break;
case PathElementAddCurveToPoint:
builder.append('C');
builder.appendECMAScriptNumber(element.points[0].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[0].y());
builder.append(',');
builder.appendECMAScriptNumber(element.points[1].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[1].y());
builder.append(',');
builder.appendECMAScriptNumber(element.points[2].x());
builder.append(' ');
builder.appendECMAScriptNumber(element.points[2].y());
break;
case PathElementCloseSubpath:
builder.append('Z');
break;
}
});
}
return builder.toString();
}
bool buildSVGPathByteStreamFromSVGPathSegListValues(const SVGPathSegListValues& list, SVGPathByteStream& result, PathParsingMode parsingMode)
{
result.clear();
if (list.isEmpty())
return true;
SVGPathSegListSource source(list);
return SVGPathParser::parseToByteStream(source, result, parsingMode);
}
bool appendSVGPathByteStreamFromSVGPathSeg(RefPtr<SVGPathSeg>&& pathSeg, SVGPathByteStream& result, PathParsingMode parsingMode)
{
// FIXME: https://bugs.webkit.org/show_bug.cgi?id=15412 - Implement normalized path segment lists!
ASSERT(parsingMode == UnalteredParsing);
SVGPathSegListValues appendedItemList(PathSegUnalteredRole);
appendedItemList.append(WTFMove(pathSeg));
SVGPathByteStream appendedByteStream;
SVGPathSegListSource source(appendedItemList);
bool ok = SVGPathParser::parseToByteStream(source, result, parsingMode, false);
if (ok)
result.append(appendedByteStream);
return ok;
}
Path buildPathFromByteStream(const SVGPathByteStream& stream)
{
if (stream.isEmpty())
return { };
Path path;
SVGPathBuilder builder(path);
SVGPathByteStreamSource source(stream);
SVGPathParser::parse(source, builder);
return path;
}
bool buildSVGPathSegListValuesFromByteStream(const SVGPathByteStream& stream, SVGPathElement& element, SVGPathSegListValues& result, PathParsingMode parsingMode)
{
if (stream.isEmpty())
return true;
SVGPathSegListBuilder builder(element, result, parsingMode == NormalizedParsing ? PathSegNormalizedRole : PathSegUnalteredRole);
SVGPathByteStreamSource source(stream);
return SVGPathParser::parse(source, builder, parsingMode);
}
bool buildStringFromByteStream(const SVGPathByteStream& stream, String& result, PathParsingMode parsingMode)
{
if (stream.isEmpty())
return true;
SVGPathByteStreamSource source(stream);
return SVGPathParser::parseToString(source, result, parsingMode);
}
bool buildStringFromSVGPathSegListValues(const SVGPathSegListValues& list, String& result, PathParsingMode parsingMode)
{
result = String();
if (list.isEmpty())
return true;
SVGPathSegListSource source(list);
return SVGPathParser::parseToString(source, result, parsingMode);
}
bool buildSVGPathByteStreamFromString(const String& d, SVGPathByteStream& result, PathParsingMode parsingMode)
{
result.clear();
if (d.isEmpty())
return true;
SVGPathStringSource source(d);
return SVGPathParser::parseToByteStream(source, result, parsingMode);
}
bool canBlendSVGPathByteStreams(const SVGPathByteStream& fromStream, const SVGPathByteStream& toStream)
{
SVGPathByteStreamSource fromSource(fromStream);
SVGPathByteStreamSource toSource(toStream);
return SVGPathBlender::canBlendPaths(fromSource, toSource);
}
bool buildAnimatedSVGPathByteStream(const SVGPathByteStream& fromStream, const SVGPathByteStream& toStream, SVGPathByteStream& result, float progress)
{
ASSERT(&toStream != &result);
result.clear();
if (toStream.isEmpty())
return true;
SVGPathByteStreamBuilder builder(result);
SVGPathByteStreamSource fromSource(fromStream);
SVGPathByteStreamSource toSource(toStream);
return SVGPathBlender::blendAnimatedPath(fromSource, toSource, builder, progress);
}
bool addToSVGPathByteStream(SVGPathByteStream& streamToAppendTo, const SVGPathByteStream& byStream, unsigned repeatCount)
{
// The byStream will be blended with streamToAppendTo. So streamToAppendTo has to have elements.
if (streamToAppendTo.isEmpty() || byStream.isEmpty())
return true;
// builder is the destination of blending fromSource and bySource. The stream of builder
// (i.e. streamToAppendTo) has to be cleared before calling addAnimatedPath.
SVGPathByteStreamBuilder builder(streamToAppendTo);
SVGPathByteStream fromStreamCopy = WTFMove(streamToAppendTo);
SVGPathByteStreamSource fromSource(fromStreamCopy);
SVGPathByteStreamSource bySource(byStream);
return SVGPathBlender::addAnimatedPath(fromSource, bySource, builder, repeatCount);
}
bool getSVGPathSegAtLengthFromSVGPathByteStream(const SVGPathByteStream& stream, float length, unsigned& pathSeg)
{
if (stream.isEmpty())
return false;
PathTraversalState traversalState(PathTraversalState::Action::SegmentAtLength);
SVGPathTraversalStateBuilder builder(traversalState, length);
SVGPathByteStreamSource source(stream);
bool ok = SVGPathParser::parse(source, builder);
pathSeg = builder.pathSegmentIndex();
return ok;
}
bool getTotalLengthOfSVGPathByteStream(const SVGPathByteStream& stream, float& totalLength)
{
if (stream.isEmpty())
return false;
PathTraversalState traversalState(PathTraversalState::Action::TotalLength);
SVGPathTraversalStateBuilder builder(traversalState);
SVGPathByteStreamSource source(stream);
bool ok = SVGPathParser::parse(source, builder);
totalLength = builder.totalLength();
return ok;
}
bool getPointAtLengthOfSVGPathByteStream(const SVGPathByteStream& stream, float length, FloatPoint& point)
{
if (stream.isEmpty())
return false;
PathTraversalState traversalState(PathTraversalState::Action::VectorAtLength);
SVGPathTraversalStateBuilder builder(traversalState, length);
SVGPathByteStreamSource source(stream);
bool ok = SVGPathParser::parse(source, builder);
point = builder.currentPoint();
return ok;
}
}