Home | History | Annotate | Download | only in svg
      1 /*
      2  * Copyright (C) Research In Motion Limited 2010, 2012. All rights reserved.
      3  *
      4  * This library is free software; you can redistribute it and/or
      5  * modify it under the terms of the GNU Library General Public
      6  * License as published by the Free Software Foundation; either
      7  * version 2 of the License, or (at your option) any later version.
      8  *
      9  * This library is distributed in the hope that it will be useful,
     10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
     11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
     12  * Library General Public License for more details.
     13  *
     14  * You should have received a copy of the GNU Library General Public License
     15  * along with this library; see the file COPYING.LIB.  If not, write to
     16  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
     17  * Boston, MA 02110-1301, USA.
     18  */
     19 
     20 #include "config.h"
     21 
     22 #include "core/svg/SVGPathUtilities.h"
     23 
     24 #include "core/svg/SVGPathBlender.h"
     25 #include "core/svg/SVGPathBuilder.h"
     26 #include "core/svg/SVGPathByteStreamBuilder.h"
     27 #include "core/svg/SVGPathByteStreamSource.h"
     28 #include "core/svg/SVGPathParser.h"
     29 #include "core/svg/SVGPathSegListBuilder.h"
     30 #include "core/svg/SVGPathSegListSource.h"
     31 #include "core/svg/SVGPathStringBuilder.h"
     32 #include "core/svg/SVGPathStringSource.h"
     33 #include "core/svg/SVGPathTraversalStateBuilder.h"
     34 #include "platform/graphics/PathTraversalState.h"
     35 
     36 namespace WebCore {
     37 
     38 static SVGPathBuilder* globalSVGPathBuilder(Path& result)
     39 {
     40     static SVGPathBuilder* s_builder = 0;
     41     if (!s_builder)
     42         s_builder = new SVGPathBuilder;
     43 
     44     s_builder->setCurrentPath(&result);
     45     return s_builder;
     46 }
     47 
     48 static SVGPathSegListBuilder* globalSVGPathSegListBuilder(SVGPathElement* element, SVGPathSegRole role, SVGPathSegList& result)
     49 {
     50     static SVGPathSegListBuilder* s_builder = 0;
     51     if (!s_builder)
     52         s_builder = new SVGPathSegListBuilder;
     53 
     54     s_builder->setCurrentSVGPathElement(element);
     55     s_builder->setCurrentSVGPathSegList(result);
     56     s_builder->setCurrentSVGPathSegRole(role);
     57     return s_builder;
     58 }
     59 
     60 static SVGPathByteStreamBuilder* globalSVGPathByteStreamBuilder(SVGPathByteStream* result)
     61 {
     62     static SVGPathByteStreamBuilder* s_builder = 0;
     63     if (!s_builder)
     64         s_builder = new SVGPathByteStreamBuilder;
     65 
     66     s_builder->setCurrentByteStream(result);
     67     return s_builder;
     68 }
     69 
     70 static SVGPathStringBuilder* globalSVGPathStringBuilder()
     71 {
     72     static SVGPathStringBuilder* s_builder = 0;
     73     if (!s_builder)
     74         s_builder = new SVGPathStringBuilder;
     75 
     76     return s_builder;
     77 }
     78 
     79 static SVGPathTraversalStateBuilder* globalSVGPathTraversalStateBuilder(PathTraversalState& traversalState, float length)
     80 {
     81     static SVGPathTraversalStateBuilder* s_builder = 0;
     82     if (!s_builder)
     83         s_builder = new SVGPathTraversalStateBuilder;
     84 
     85     s_builder->setCurrentTraversalState(&traversalState);
     86     s_builder->setDesiredLength(length);
     87     return s_builder;
     88 }
     89 
     90 static SVGPathParser* globalSVGPathParser(SVGPathSource* source, SVGPathConsumer* consumer)
     91 {
     92     static SVGPathParser* s_parser = 0;
     93     if (!s_parser)
     94         s_parser = new SVGPathParser;
     95 
     96     s_parser->setCurrentSource(source);
     97     s_parser->setCurrentConsumer(consumer);
     98     return s_parser;
     99 }
    100 
    101 static SVGPathBlender* globalSVGPathBlender()
    102 {
    103     static SVGPathBlender* s_blender = 0;
    104     if (!s_blender)
    105         s_blender = new SVGPathBlender;
    106 
    107     return s_blender;
    108 }
    109 
    110 bool buildPathFromString(const String& d, Path& result)
    111 {
    112     if (d.isEmpty())
    113         return false;
    114 
    115     SVGPathBuilder* builder = globalSVGPathBuilder(result);
    116 
    117     OwnPtr<SVGPathStringSource> source = SVGPathStringSource::create(d);
    118     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    119     bool ok = parser->parsePathDataFromSource(NormalizedParsing);
    120     parser->cleanup();
    121     return ok;
    122 }
    123 
    124 bool buildSVGPathByteStreamFromSVGPathSegList(const SVGPathSegList& list, SVGPathByteStream* result, PathParsingMode parsingMode)
    125 {
    126     ASSERT(result);
    127     result->clear();
    128     if (list.isEmpty())
    129         return false;
    130 
    131     SVGPathByteStreamBuilder* builder = globalSVGPathByteStreamBuilder(result);
    132 
    133     OwnPtr<SVGPathSegListSource> source = SVGPathSegListSource::create(list);
    134     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    135     bool ok = parser->parsePathDataFromSource(parsingMode);
    136     parser->cleanup();
    137     return ok;
    138 }
    139 
    140 bool appendSVGPathByteStreamFromSVGPathSeg(PassRefPtr<SVGPathSeg> pathSeg, SVGPathByteStream* result, PathParsingMode parsingMode)
    141 {
    142     ASSERT(result);
    143     // FIXME: https://bugs.webkit.org/show_bug.cgi?id=15412 - Implement normalized path segment lists!
    144     ASSERT(parsingMode == UnalteredParsing);
    145 
    146     SVGPathSegList appendedItemList(PathSegUnalteredRole);
    147     appendedItemList.append(pathSeg);
    148     OwnPtr<SVGPathByteStream> appendedByteStream = SVGPathByteStream::create();
    149 
    150     SVGPathByteStreamBuilder* builder = globalSVGPathByteStreamBuilder(appendedByteStream.get());
    151     OwnPtr<SVGPathSegListSource> source = SVGPathSegListSource::create(appendedItemList);
    152     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    153     bool ok = parser->parsePathDataFromSource(parsingMode, false);
    154     parser->cleanup();
    155 
    156     if (ok)
    157         result->append(appendedByteStream.get());
    158 
    159     return ok;
    160 }
    161 
    162 bool buildPathFromByteStream(SVGPathByteStream* stream, Path& result)
    163 {
    164     ASSERT(stream);
    165     if (stream->isEmpty())
    166         return false;
    167 
    168     SVGPathBuilder* builder = globalSVGPathBuilder(result);
    169 
    170     OwnPtr<SVGPathByteStreamSource> source = SVGPathByteStreamSource::create(stream);
    171     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    172     bool ok = parser->parsePathDataFromSource(NormalizedParsing);
    173     parser->cleanup();
    174     return ok;
    175 }
    176 
    177 bool buildSVGPathSegListFromByteStream(SVGPathByteStream* stream, SVGPathElement* element, SVGPathSegList& result, PathParsingMode parsingMode)
    178 {
    179     ASSERT(stream);
    180     if (stream->isEmpty())
    181         return false;
    182 
    183     SVGPathSegListBuilder* builder = globalSVGPathSegListBuilder(element, parsingMode == NormalizedParsing ? PathSegNormalizedRole : PathSegUnalteredRole, result);
    184 
    185     OwnPtr<SVGPathByteStreamSource> source = SVGPathByteStreamSource::create(stream);
    186     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    187     bool ok = parser->parsePathDataFromSource(parsingMode);
    188     parser->cleanup();
    189     return ok;
    190 }
    191 
    192 bool buildStringFromByteStream(SVGPathByteStream* stream, String& result, PathParsingMode parsingMode)
    193 {
    194     ASSERT(stream);
    195     if (stream->isEmpty())
    196         return false;
    197 
    198     SVGPathStringBuilder* builder = globalSVGPathStringBuilder();
    199 
    200     OwnPtr<SVGPathByteStreamSource> source = SVGPathByteStreamSource::create(stream);
    201     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    202     bool ok = parser->parsePathDataFromSource(parsingMode);
    203     result = builder->result();
    204     parser->cleanup();
    205     return ok;
    206 }
    207 
    208 bool buildStringFromSVGPathSegList(const SVGPathSegList& list, String& result, PathParsingMode parsingMode)
    209 {
    210     result = String();
    211     if (list.isEmpty())
    212         return false;
    213 
    214     SVGPathStringBuilder* builder = globalSVGPathStringBuilder();
    215 
    216     OwnPtr<SVGPathSegListSource> source = SVGPathSegListSource::create(list);
    217     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    218     bool ok = parser->parsePathDataFromSource(parsingMode);
    219     result = builder->result();
    220     parser->cleanup();
    221     return ok;
    222 }
    223 
    224 bool buildSVGPathByteStreamFromString(const String& d, SVGPathByteStream* result, PathParsingMode parsingMode)
    225 {
    226     ASSERT(result);
    227     result->clear();
    228     if (d.isEmpty())
    229         return false;
    230 
    231     SVGPathByteStreamBuilder* builder = globalSVGPathByteStreamBuilder(result);
    232 
    233     OwnPtr<SVGPathStringSource> source = SVGPathStringSource::create(d);
    234     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    235     bool ok = parser->parsePathDataFromSource(parsingMode);
    236     parser->cleanup();
    237     return ok;
    238 }
    239 
    240 bool buildAnimatedSVGPathByteStream(SVGPathByteStream* fromStream, SVGPathByteStream* toStream, SVGPathByteStream* result, float progress)
    241 {
    242     ASSERT(fromStream);
    243     ASSERT(toStream);
    244     ASSERT(result);
    245     ASSERT(toStream != result);
    246 
    247     result->clear();
    248     if (toStream->isEmpty())
    249         return false;
    250 
    251     SVGPathByteStreamBuilder* builder = globalSVGPathByteStreamBuilder(result);
    252 
    253     OwnPtr<SVGPathByteStreamSource> fromSource = SVGPathByteStreamSource::create(fromStream);
    254     OwnPtr<SVGPathByteStreamSource> toSource = SVGPathByteStreamSource::create(toStream);
    255     SVGPathBlender* blender = globalSVGPathBlender();
    256     bool ok = blender->blendAnimatedPath(progress, fromSource.get(), toSource.get(), builder);
    257     blender->cleanup();
    258     return ok;
    259 }
    260 
    261 bool addToSVGPathByteStream(SVGPathByteStream* fromStream, SVGPathByteStream* byStream, unsigned repeatCount)
    262 {
    263     ASSERT(fromStream);
    264     ASSERT(byStream);
    265     if (fromStream->isEmpty() || byStream->isEmpty())
    266         return false;
    267 
    268     SVGPathByteStreamBuilder* builder = globalSVGPathByteStreamBuilder(fromStream);
    269 
    270     OwnPtr<SVGPathByteStream> fromStreamCopy = fromStream->copy();
    271     fromStream->clear();
    272 
    273     OwnPtr<SVGPathByteStreamSource> fromSource = SVGPathByteStreamSource::create(fromStreamCopy.get());
    274     OwnPtr<SVGPathByteStreamSource> bySource = SVGPathByteStreamSource::create(byStream);
    275     SVGPathBlender* blender = globalSVGPathBlender();
    276     bool ok = blender->addAnimatedPath(fromSource.get(), bySource.get(), builder, repeatCount);
    277     blender->cleanup();
    278     return ok;
    279 }
    280 
    281 bool getSVGPathSegAtLengthFromSVGPathByteStream(SVGPathByteStream* stream, float length, unsigned& pathSeg)
    282 {
    283     ASSERT(stream);
    284     if (stream->isEmpty())
    285         return false;
    286 
    287     PathTraversalState traversalState(PathTraversalState::TraversalSegmentAtLength);
    288     SVGPathTraversalStateBuilder* builder = globalSVGPathTraversalStateBuilder(traversalState, length);
    289 
    290     OwnPtr<SVGPathByteStreamSource> source = SVGPathByteStreamSource::create(stream);
    291     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    292     bool ok = parser->parsePathDataFromSource(NormalizedParsing);
    293     pathSeg = builder->pathSegmentIndex();
    294     parser->cleanup();
    295     return ok;
    296 }
    297 
    298 bool getTotalLengthOfSVGPathByteStream(SVGPathByteStream* stream, float& totalLength)
    299 {
    300     ASSERT(stream);
    301     if (stream->isEmpty())
    302         return false;
    303 
    304     PathTraversalState traversalState(PathTraversalState::TraversalTotalLength);
    305     SVGPathTraversalStateBuilder* builder = globalSVGPathTraversalStateBuilder(traversalState, 0);
    306 
    307     OwnPtr<SVGPathByteStreamSource> source = SVGPathByteStreamSource::create(stream);
    308     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    309     bool ok = parser->parsePathDataFromSource(NormalizedParsing);
    310     totalLength = builder->totalLength();
    311     parser->cleanup();
    312     return ok;
    313 }
    314 
    315 bool getPointAtLengthOfSVGPathByteStream(SVGPathByteStream* stream, float length, SVGPoint& point)
    316 {
    317     ASSERT(stream);
    318     if (stream->isEmpty())
    319         return false;
    320 
    321     PathTraversalState traversalState(PathTraversalState::TraversalPointAtLength);
    322     SVGPathTraversalStateBuilder* builder = globalSVGPathTraversalStateBuilder(traversalState, length);
    323 
    324     OwnPtr<SVGPathByteStreamSource> source = SVGPathByteStreamSource::create(stream);
    325     SVGPathParser* parser = globalSVGPathParser(source.get(), builder);
    326     bool ok = parser->parsePathDataFromSource(NormalizedParsing);
    327     point = builder->currentPoint();
    328     parser->cleanup();
    329     return ok;
    330 }
    331 
    332 }
    333