1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 // MARKER(update_precomp.py): autogen include statement, do not remove
23 #include "precompiled_svgio.hxx"
24 
25 #include <svgio/svgreader/svgdocumenthandler.hxx>
26 #include <svgio/svgreader/svgtoken.hxx>
27 #include <svgio/svgreader/svgsvgnode.hxx>
28 #include <svgio/svgreader/svggnode.hxx>
29 #include <svgio/svgreader/svgnode.hxx>
30 #include <svgio/svgreader/svgpathnode.hxx>
31 #include <svgio/svgreader/svgrectnode.hxx>
32 #include <svgio/svgreader/svggradientnode.hxx>
33 #include <svgio/svgreader/svggradientstopnode.hxx>
34 #include <svgio/svgreader/svgsymbolnode.hxx>
35 #include <svgio/svgreader/svgusenode.hxx>
36 #include <svgio/svgreader/svgcirclenode.hxx>
37 #include <svgio/svgreader/svgellipsenode.hxx>
38 #include <svgio/svgreader/svglinenode.hxx>
39 #include <svgio/svgreader/svgpolynode.hxx>
40 #include <svgio/svgreader/svgsymbolnode.hxx>
41 #include <svgio/svgreader/svgtextnode.hxx>
42 #include <svgio/svgreader/svgcharacternode.hxx>
43 #include <svgio/svgreader/svgtspannode.hxx>
44 #include <svgio/svgreader/svgtrefnode.hxx>
45 #include <svgio/svgreader/svgtextpathnode.hxx>
46 #include <svgio/svgreader/svgstylenode.hxx>
47 #include <svgio/svgreader/svgimagenode.hxx>
48 #include <svgio/svgreader/svgclippathnode.hxx>
49 #include <svgio/svgreader/svgmasknode.hxx>
50 #include <svgio/svgreader/svgmarkernode.hxx>
51 #include <svgio/svgreader/svgpatternnode.hxx>
52 #include <svgio/svgreader/svgtitledescnode.hxx>
53 
54 //////////////////////////////////////////////////////////////////////////////
55 
56 using namespace com::sun::star;
57 
58 //////////////////////////////////////////////////////////////////////////////
59 
60 namespace
61 {
62     svgio::svgreader::SvgCharacterNode* whiteSpaceHandling(svgio::svgreader::SvgNode* pNode, svgio::svgreader::SvgCharacterNode* pLast)
63     {
64         if(pNode)
65         {
66             const svgio::svgreader::SvgNodeVector& rChilds = pNode->getChildren();
67             const sal_uInt32 nCount(rChilds.size());
68 
69             for(sal_uInt32 a(0); a < nCount; a++)
70             {
71                 svgio::svgreader::SvgNode* pCandidate = rChilds[a];
72 
73                 if(pCandidate)
74                 {
75                     switch(pCandidate->getType())
76                     {
77                         case svgio::svgreader::SVGTokenCharacter:
78                         {
79                             // clean whitespace in text span
80                             svgio::svgreader::SvgCharacterNode* pCharNode = static_cast< svgio::svgreader::SvgCharacterNode* >(pCandidate);
81                             pCharNode->whiteSpaceHandling();
82 
83                             // pCharNode may have lost all text. If that's the case, ignore
84                             // as invalid character node
85                             if(pCharNode->getText().getLength())
86                             {
87                                 if(pLast)
88                                 {
89                                     bool bAddGap(true);
90                                     static bool bNoGapsForBaselineShift(true);
91 
92                                     if(bNoGapsForBaselineShift)
93                                     {
94                                         // With this option a baseline shift between two char parts ('words')
95                                         // will not add a space 'gap' to the end of the (non-last) word. This
96                                         // seems to be the standard behaviour, see last bugdoc attached #122524#
97                                         const svgio::svgreader::SvgStyleAttributes* pStyleLast = pLast->getSvgStyleAttributes();
98                                         const svgio::svgreader::SvgStyleAttributes* pStyleCurrent = pCandidate->getSvgStyleAttributes();
99 
100                                         if(pStyleLast && pStyleCurrent && pStyleLast->getBaselineShift() != pStyleCurrent->getBaselineShift())
101                                         {
102                                             bAddGap = false;
103                                         }
104                                     }
105 
106                                     // add in-between whitespace (single space) to last
107                                     // known character node
108                                     if(bAddGap)
109                                     {
110                                         pLast->addGap();
111                                     }
112                                 }
113 
114                                 // remember new last corected character node
115                                 pLast = pCharNode;
116                             }
117                             break;
118                         }
119                         case svgio::svgreader::SVGTokenTspan:
120                         case svgio::svgreader::SVGTokenTextPath:
121                         case svgio::svgreader::SVGTokenTref:
122                         {
123                             // recursively clean whitespaces in subhierarchy
124                             pLast = whiteSpaceHandling(pCandidate, pLast);
125                             break;
126                         }
127                         default:
128                         {
129                             OSL_ENSURE(false, "Unexpected token inside SVGTokenText (!)");
130                             break;
131                         }
132                     }
133                 }
134             }
135         }
136 
137         return pLast;
138     }
139 }
140 
141 //////////////////////////////////////////////////////////////////////////////
142 
143 namespace svgio
144 {
145     namespace svgreader
146     {
147         SvgDocHdl::SvgDocHdl(const rtl::OUString& aAbsolutePath)
148         :   maDocument(aAbsolutePath),
149             mpTarget(0),
150             maCssContents()
151         {
152         }
153 
154         SvgDocHdl::~SvgDocHdl()
155         {
156 #ifdef DBG_UTIL
157             if(mpTarget)
158             {
159                 OSL_ENSURE(false, "SvgDocHdl destructed with active target (!)");
160                 delete mpTarget;
161             }
162             OSL_ENSURE(!maCssContents.size(), "SvgDocHdl destructed with active css style stack entry (!)");
163 #endif
164         }
165 
166         void SvgDocHdl::startDocument(  ) throw (xml::sax::SAXException, uno::RuntimeException)
167         {
168             OSL_ENSURE(!mpTarget, "Already a target at document start (!)");
169             OSL_ENSURE(!maCssContents.size(), "SvgDocHdl startDocument with active css style stack entry (!)");
170         }
171 
172         void SvgDocHdl::endDocument(  ) throw (xml::sax::SAXException, uno::RuntimeException)
173         {
174             OSL_ENSURE(!mpTarget, "Still a target at document end (!)");
175             OSL_ENSURE(!maCssContents.size(), "SvgDocHdl endDocument with active css style stack entry (!)");
176         }
177 
178         void SvgDocHdl::startElement( const ::rtl::OUString& aName, const uno::Reference< xml::sax::XAttributeList >& xAttribs ) throw (xml::sax::SAXException, uno::RuntimeException)
179         {
180             if(aName.getLength())
181             {
182                 const SVGToken aSVGToken(StrToSVGToken(aName, false));
183 
184                 switch(aSVGToken)
185                 {
186                     /// structural elements
187                     case SVGTokenSymbol:
188                     {
189                         /// new basic node for Symbol. Content gets scanned, but
190                         /// will not be decomposed (see SvgNode::decomposeSvgNode and bReferenced)
191                         mpTarget = new SvgSymbolNode(maDocument, mpTarget);
192                         mpTarget->parseAttributes(xAttribs);
193                         break;
194                     }
195                     case SVGTokenDefs:
196                     case SVGTokenG:
197                     {
198                         /// new node for Defs/G
199                         mpTarget = new SvgGNode(aSVGToken, maDocument, mpTarget);
200                         mpTarget->parseAttributes(xAttribs);
201                         break;
202                     }
203                     case SVGTokenSvg:
204                     {
205                         /// new node for Svg
206                         mpTarget = new SvgSvgNode(maDocument, mpTarget);
207                         mpTarget->parseAttributes(xAttribs);
208                         break;
209                     }
210                     case SVGTokenUse:
211                     {
212                         /// new node for Use
213                         mpTarget = new SvgUseNode(maDocument, mpTarget);
214                         mpTarget->parseAttributes(xAttribs);
215                         break;
216                     }
217 
218                     /// shape elements
219                     case SVGTokenCircle:
220                     {
221                         /// new node for Circle
222                         mpTarget = new SvgCircleNode(maDocument, mpTarget);
223                         mpTarget->parseAttributes(xAttribs);
224                         break;
225                     }
226                     case SVGTokenEllipse:
227                     {
228                         /// new node for Ellipse
229                         mpTarget = new SvgEllipseNode(maDocument, mpTarget);
230                         mpTarget->parseAttributes(xAttribs);
231                         break;
232                     }
233                     case SVGTokenLine:
234                     {
235                         /// new node for Line
236                         mpTarget = new SvgLineNode(maDocument, mpTarget);
237                         mpTarget->parseAttributes(xAttribs);
238                         break;
239                     }
240                     case SVGTokenPath:
241                     {
242                         /// new node for Path
243                         mpTarget = new SvgPathNode(maDocument, mpTarget);
244                         mpTarget->parseAttributes(xAttribs);
245                         break;
246                     }
247                     case SVGTokenPolygon:
248                     {
249                         /// new node for Polygon
250                         mpTarget = new SvgPolyNode(maDocument, mpTarget, false);
251                         mpTarget->parseAttributes(xAttribs);
252                         break;
253                     }
254                     case SVGTokenPolyline:
255                     {
256                         /// new node for Polyline
257                         mpTarget = new SvgPolyNode(maDocument, mpTarget, true);
258                         mpTarget->parseAttributes(xAttribs);
259                         break;
260                     }
261                     case SVGTokenRect:
262                     {
263                         /// new node for Rect
264                         mpTarget = new SvgRectNode(maDocument, mpTarget);
265                         mpTarget->parseAttributes(xAttribs);
266                         break;
267                     }
268                     case SVGTokenImage:
269                     {
270                         /// new node for Image
271                         mpTarget = new SvgImageNode(maDocument, mpTarget);
272                         mpTarget->parseAttributes(xAttribs);
273                         break;
274                     }
275 
276                     /// title and description
277                     case SVGTokenTitle:
278                     case SVGTokenDesc:
279                     {
280                         /// new node for Title and/or Desc
281                         mpTarget = new SvgTitleDescNode(aSVGToken, maDocument, mpTarget);
282                         break;
283                     }
284 
285                     /// gradients
286                     case SVGTokenLinearGradient:
287                     case SVGTokenRadialGradient:
288                     {
289                         mpTarget = new SvgGradientNode(aSVGToken, maDocument, mpTarget);
290                         mpTarget->parseAttributes(xAttribs);
291                         break;
292                     }
293 
294                     /// gradient stops
295                     case SVGTokenStop:
296                     {
297                         mpTarget = new SvgGradientStopNode(maDocument, mpTarget);
298                         mpTarget->parseAttributes(xAttribs);
299                         break;
300                     }
301 
302                     /// text
303                     case SVGTokenText:
304                     {
305                         mpTarget = new SvgTextNode(maDocument, mpTarget);
306                         mpTarget->parseAttributes(xAttribs);
307                         break;
308                     }
309                     case SVGTokenTspan:
310                     {
311                         mpTarget = new SvgTspanNode(maDocument, mpTarget);
312                         mpTarget->parseAttributes(xAttribs);
313                         break;
314                     }
315                     case SVGTokenTref:
316                     {
317                         mpTarget = new SvgTrefNode(maDocument, mpTarget);
318                         mpTarget->parseAttributes(xAttribs);
319                         break;
320                     }
321                     case SVGTokenTextPath:
322                     {
323                         mpTarget = new SvgTextPathNode(maDocument, mpTarget);
324                         mpTarget->parseAttributes(xAttribs);
325                         break;
326                     }
327 
328                     /// styles (as stylesheets)
329                     case SVGTokenStyle:
330                     {
331                         SvgStyleNode* pNew = new SvgStyleNode(maDocument, mpTarget);
332                         mpTarget = pNew;
333                         mpTarget->parseAttributes(xAttribs);
334 
335                         if(pNew->isTextCss())
336                         {
337                             maCssContents.push_back(rtl::OUString());
338                         }
339                         break;
340                     }
341 
342                     /// structural elements clip-path and mask. Content gets scanned, but
343                     /// will not be decomposed (see SvgNode::decomposeSvgNode and bReferenced)
344                     case SVGTokenClipPathNode:
345                     {
346                         /// new node for ClipPath
347                         mpTarget = new SvgClipPathNode(maDocument, mpTarget);
348                         mpTarget->parseAttributes(xAttribs);
349                         break;
350                     }
351                     case SVGTokenMask:
352                     {
353                         /// new node for Mask
354                         mpTarget = new SvgMaskNode(maDocument, mpTarget);
355                         mpTarget->parseAttributes(xAttribs);
356                         break;
357                     }
358 
359                     /// structural element marker
360                     case SVGTokenMarker:
361                     {
362                         /// new node for marker
363                         mpTarget = new SvgMarkerNode(maDocument, mpTarget);
364                         mpTarget->parseAttributes(xAttribs);
365                         break;
366                     }
367 
368                     /// structural element pattern
369                     case SVGTokenPattern:
370                     {
371                         /// new node for pattern
372                         mpTarget = new SvgPatternNode(maDocument, mpTarget);
373                         mpTarget->parseAttributes(xAttribs);
374                         break;
375                     }
376 
377                     default:
378                     {
379                         /// invalid token, ignore
380 #ifdef DBG_UTIL
381                         myAssert(
382                             rtl::OUString::createFromAscii("Unknown Base SvgToken <") +
383                             aName +
384                             rtl::OUString::createFromAscii("> (!)"));
385 #endif
386                         break;
387                     }
388                 }
389             }
390         }
391 
392         void SvgDocHdl::endElement( const ::rtl::OUString& aName ) throw (xml::sax::SAXException, uno::RuntimeException)
393         {
394             if(aName.getLength())
395             {
396                 const SVGToken aSVGToken(StrToSVGToken(aName, false));
397                 SvgNode* pWhitespaceCheck(SVGTokenText == aSVGToken ? mpTarget : 0);
398                 SvgStyleNode* pCssStyle(SVGTokenStyle == aSVGToken ? static_cast< SvgStyleNode* >(mpTarget) : 0);
399                 SvgTitleDescNode* pSvgTitleDescNode(SVGTokenTitle == aSVGToken || SVGTokenDesc == aSVGToken ? static_cast< SvgTitleDescNode* >(mpTarget) : 0);
400 
401                 switch(aSVGToken)
402                 {
403                     /// valid tokens for which a new one was created
404 
405                     /// structural elements
406                     case SVGTokenDefs:
407                     case SVGTokenG:
408                     case SVGTokenSvg:
409                     case SVGTokenSymbol:
410                     case SVGTokenUse:
411 
412                     /// shape elements
413                     case SVGTokenCircle:
414                     case SVGTokenEllipse:
415                     case SVGTokenLine:
416                     case SVGTokenPath:
417                     case SVGTokenPolygon:
418                     case SVGTokenPolyline:
419                     case SVGTokenRect:
420                     case SVGTokenImage:
421 
422                     /// title and description
423                     case SVGTokenTitle:
424                     case SVGTokenDesc:
425 
426                     /// gradients
427                     case SVGTokenLinearGradient:
428                     case SVGTokenRadialGradient:
429 
430                     /// gradient stops
431                     case SVGTokenStop:
432 
433                     /// text
434                     case SVGTokenText:
435                     case SVGTokenTspan:
436                     case SVGTokenTextPath:
437                     case SVGTokenTref:
438 
439                     /// styles (as stylesheets)
440                     case SVGTokenStyle:
441 
442                     /// structural elements clip-path and mask
443                     case SVGTokenClipPathNode:
444                     case SVGTokenMask:
445 
446                     /// structural element marker
447                     case SVGTokenMarker:
448 
449                     /// structural element pattern
450                     case SVGTokenPattern:
451 
452                     /// content handling after parsing
453                     {
454                         if(mpTarget)
455                         {
456                             if(!mpTarget->getParent())
457                             {
458                                 // last element closing, save this tree
459                                 maDocument.appendNode(mpTarget);
460                             }
461 
462                             mpTarget = const_cast< SvgNode* >(mpTarget->getParent());
463                         }
464                         else
465                         {
466                             OSL_ENSURE(false, "Closing token, but no context (!)");
467                         }
468                         break;
469                     }
470                     default:
471                     {
472                         /// invalid token, ignore
473                     }
474                 }
475 
476                 if(pSvgTitleDescNode && mpTarget)
477                 {
478                     const rtl::OUString aText(pSvgTitleDescNode->getText());
479 
480                     if(aText.getLength())
481                     {
482                         if(SVGTokenTitle == aSVGToken)
483                         {
484                             mpTarget->parseAttribute(getStrTitle(), aSVGToken, aText);
485                         }
486                         else // if(SVGTokenDesc == aSVGToken)
487                         {
488                             mpTarget->parseAttribute(getStrDesc(), aSVGToken, aText);
489                         }
490                     }
491                 }
492 
493                 if(pCssStyle && pCssStyle->isTextCss())
494                 {
495                     // css style parsing
496                     if(maCssContents.size())
497                     {
498                         // need to interpret css styles and remember them as StyleSheets
499                         pCssStyle->addCssStyleSheet(*(maCssContents.end() - 1));
500                         maCssContents.pop_back();
501                     }
502                     else
503                     {
504                         OSL_ENSURE(false, "Closing CssStyle, but no collector string on stack (!)");
505                     }
506                 }
507 
508                 if(pWhitespaceCheck)
509                 {
510                     // cleanup read strings
511                     whiteSpaceHandling(pWhitespaceCheck, 0);
512                 }
513             }
514         }
515 
516         void SvgDocHdl::characters( const ::rtl::OUString& aChars ) throw (xml::sax::SAXException, uno::RuntimeException)
517         {
518             const sal_uInt32 nLength(aChars.getLength());
519 
520             if(mpTarget && nLength)
521             {
522                 switch(mpTarget->getType())
523                 {
524                     case SVGTokenText:
525                     case SVGTokenTspan:
526                     case SVGTokenTextPath:
527                     {
528                         const SvgNodeVector& rChilds = mpTarget->getChildren();
529                         SvgCharacterNode* pTarget = 0;
530 
531                         if(rChilds.size())
532                         {
533                             pTarget = dynamic_cast< SvgCharacterNode* >(rChilds[rChilds.size() - 1]);
534                         }
535 
536                         if(pTarget)
537                         {
538                             // concatenate to current character span
539                             pTarget->concatenate(aChars);
540                         }
541                         else
542                         {
543                             // add character span as simplified tspan (no arguments)
544                             // as direct child of SvgTextNode/SvgTspanNode/SvgTextPathNode
545                             new SvgCharacterNode(maDocument, mpTarget, aChars);
546                         }
547                         break;
548                     }
549                     case SVGTokenStyle:
550                     {
551                         SvgStyleNode& rSvgStyleNode = static_cast< SvgStyleNode& >(*mpTarget);
552 
553                         if(rSvgStyleNode.isTextCss())
554                         {
555                             // collect characters for css style
556                             if(maCssContents.size())
557                             {
558                                 const ::rtl::OUString aTrimmedChars(aChars.trim());
559 
560                                 if(aTrimmedChars.getLength())
561                                 {
562                                     std::vector< rtl::OUString >::iterator aString(maCssContents.end() - 1);
563                                     (*aString) += aTrimmedChars;
564                                 }
565                             }
566                             else
567                             {
568                                 OSL_ENSURE(false, "Closing CssStyle, but no collector string on stack (!)");
569                             }
570                         }
571                         break;
572                     }
573                     case SVGTokenTitle:
574                     case SVGTokenDesc:
575                     {
576                         SvgTitleDescNode& rSvgTitleDescNode = static_cast< SvgTitleDescNode& >(*mpTarget);
577 
578                         // add text directly to SvgTitleDescNode
579                         rSvgTitleDescNode.concatenate(aChars);
580                         break;
581                     }
582                     default:
583                     {
584                         // characters not used by a known node
585                         break;
586                     }
587                 }
588             }
589         }
590 
591         void SvgDocHdl::ignorableWhitespace(const ::rtl::OUString& /*aWhitespaces*/) throw (xml::sax::SAXException, uno::RuntimeException)
592         {
593         }
594 
595         void SvgDocHdl::processingInstruction(const ::rtl::OUString& /*aTarget*/, const ::rtl::OUString& /*aData*/) throw (xml::sax::SAXException, uno::RuntimeException)
596         {
597         }
598 
599         void SvgDocHdl::setDocumentLocator(const uno::Reference< xml::sax::XLocator >& /*xLocator*/) throw (xml::sax::SAXException, uno::RuntimeException)
600         {
601         }
602     } // end of namespace svgreader
603 } // end of namespace svgio
604 
605 //////////////////////////////////////////////////////////////////////////////
606 // eof
607