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/svgtextnode.hxx>
26 #include <svgio/svgreader/svgcharacternode.hxx>
27 #include <svgio/svgreader/svgstyleattributes.hxx>
28 #include <svgio/svgreader/svgtrefnode.hxx>
29 #include <svgio/svgreader/svgtextpathnode.hxx>
30 #include <svgio/svgreader/svgtspannode.hxx>
31 #include <drawinglayer/primitive2d/transformprimitive2d.hxx>
32 #include <drawinglayer/primitive2d/unifiedtransparenceprimitive2d.hxx>
33 
34 //////////////////////////////////////////////////////////////////////////////
35 
36 namespace svgio
37 {
38     namespace svgreader
39     {
40         SvgTextNode::SvgTextNode(
41             SvgDocument& rDocument,
42             SvgNode* pParent)
43         :   SvgNode(SVGTokenText, rDocument, pParent),
44             maSvgStyleAttributes(*this),
45             mpaTransform(0),
46             maSvgTextPositions()
47         {
48         }
49 
50         SvgTextNode::~SvgTextNode()
51         {
52             if(mpaTransform) delete mpaTransform;
53         }
54 
55         const SvgStyleAttributes* SvgTextNode::getSvgStyleAttributes() const
56         {
57             static rtl::OUString aClassStr(rtl::OUString::createFromAscii("text"));
58 
59             return checkForCssStyle(aClassStr, maSvgStyleAttributes);
60         }
61 
62         void SvgTextNode::parseAttribute(const rtl::OUString& rTokenName, SVGToken aSVGToken, const rtl::OUString& aContent)
63         {
64             // call parent
65             SvgNode::parseAttribute(rTokenName, aSVGToken, aContent);
66 
67             // read style attributes
68             maSvgStyleAttributes.parseStyleAttribute(rTokenName, aSVGToken, aContent);
69 
70             // read text position attributes
71             maSvgTextPositions.parseTextPositionAttributes(rTokenName, aSVGToken, aContent);
72 
73             // parse own
74             switch(aSVGToken)
75             {
76                 case SVGTokenStyle:
77                 {
78                     readLocalCssStyle(aContent);
79                     break;
80                 }
81                 case SVGTokenTransform:
82                 {
83                     const basegfx::B2DHomMatrix aMatrix(readTransform(aContent, *this));
84 
85                     if(!aMatrix.isIdentity())
86                     {
87                         setTransform(&aMatrix);
88                     }
89                     break;
90                 }
91                 default:
92                 {
93                     break;
94                 }
95             }
96         }
97 
98         void SvgTextNode::addTextPrimitives(
99             const SvgNode& rCandidate,
100             drawinglayer::primitive2d::Primitive2DSequence& rTarget,
101             drawinglayer::primitive2d::Primitive2DSequence& rSource) const
102         {
103             if(rSource.hasElements())
104             {
105                 const SvgStyleAttributes* pAttributes = rCandidate.getSvgStyleAttributes();
106 
107                 if(pAttributes)
108                 {
109                     // add text with taking all Fill/Stroke attributes into account
110                     pAttributes->add_text(rTarget, rSource);
111                 }
112                 else
113                 {
114                     // should not happen, every subnode from SvgTextNode will at least
115                     // return the attributes from SvgTextNode. Nonetheless, add text
116                     drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(rTarget, rSource);
117                 }
118             }
119         }
120 
121         void SvgTextNode::DecomposeChild(const SvgNode& rCandidate, drawinglayer::primitive2d::Primitive2DSequence& rTarget, SvgTextPosition& rSvgTextPosition) const
122         {
123             switch(rCandidate.getType())
124             {
125                 case SVGTokenCharacter:
126                 {
127                     // direct SvgTextPathNode derivates, decompose them
128                     const SvgCharacterNode& rSvgCharacterNode = static_cast< const SvgCharacterNode& >(rCandidate);
129                     rSvgCharacterNode.decomposeText(rTarget, rSvgTextPosition);
130                     break;
131                 }
132                 case SVGTokenTextPath:
133                 {
134                     // direct TextPath decompose
135                     const SvgTextPathNode& rSvgTextPathNode = static_cast< const SvgTextPathNode& >(rCandidate);
136                     const SvgNodeVector& rChildren = rSvgTextPathNode.getChildren();
137                     const sal_uInt32 nCount(rChildren.size());
138 
139                     if(nCount && rSvgTextPathNode.isValid())
140                     {
141                         // remember original TextStart to later detect hor/ver offsets
142                         const basegfx::B2DPoint aTextStart(rSvgTextPosition.getPosition());
143                         drawinglayer::primitive2d::Primitive2DSequence aNewTarget;
144 
145                         // decompose to regular TextPrimitives
146                         for(sal_uInt32 a(0); a < nCount; a++)
147                         {
148                             DecomposeChild(*rChildren[a], aNewTarget, rSvgTextPosition);
149                         }
150 
151                         if(aNewTarget.hasElements())
152                         {
153                             const drawinglayer::primitive2d::Primitive2DSequence aPathContent(aNewTarget);
154                             aNewTarget.realloc(0);
155 
156                             // dismantle TextPrimitives and map them on curve/path
157                             rSvgTextPathNode.decomposePathNode(aPathContent, aNewTarget, aTextStart);
158                         }
159 
160                         if(aNewTarget.hasElements())
161                         {
162                             addTextPrimitives(rCandidate, rTarget, aNewTarget);
163                         }
164                     }
165 
166                     break;
167                 }
168                 case SVGTokenTspan:
169                 {
170                     // Tspan may have children, call recursively
171                     const SvgTspanNode& rSvgTspanNode = static_cast< const SvgTspanNode& >(rCandidate);
172                     const SvgNodeVector& rChildren = rSvgTspanNode.getChildren();
173                     const sal_uInt32 nCount(rChildren.size());
174 
175                     if(nCount)
176                     {
177                         SvgTextPosition aSvgTextPosition(&rSvgTextPosition, rSvgTspanNode, rSvgTspanNode.getSvgTextPositions());
178                         drawinglayer::primitive2d::Primitive2DSequence aNewTarget;
179 
180                         for(sal_uInt32 a(0); a < nCount; a++)
181                         {
182                             DecomposeChild(*rChildren[a], aNewTarget, aSvgTextPosition);
183                         }
184 
185                         rSvgTextPosition.setPosition(aSvgTextPosition.getPosition());
186 
187                         if(aNewTarget.hasElements())
188                         {
189                             addTextPrimitives(rCandidate, rTarget, aNewTarget);
190                         }
191                     }
192                     break;
193                 }
194                 case SVGTokenTref:
195                 {
196                     const SvgTrefNode& rSvgTrefNode = static_cast< const SvgTrefNode& >(rCandidate);
197                     const SvgTextNode* pRefText = rSvgTrefNode.getReferencedSvgTextNode();
198 
199                     if(pRefText)
200                     {
201                         const SvgNodeVector& rChildren = pRefText->getChildren();
202                         const sal_uInt32 nCount(rChildren.size());
203                         drawinglayer::primitive2d::Primitive2DSequence aNewTarget;
204 
205                         if(nCount)
206                         {
207                             for(sal_uInt32 a(0); a < nCount; a++)
208                             {
209                                 const SvgNode& rChildCandidate = *rChildren[a];
210                                 const_cast< SvgNode& >(rChildCandidate).setAlternativeParent(this);
211 
212                                 DecomposeChild(rChildCandidate, aNewTarget, rSvgTextPosition);
213                                 const_cast< SvgNode& >(rChildCandidate).setAlternativeParent(0);
214                             }
215 
216                             if(aNewTarget.hasElements())
217                             {
218                                 addTextPrimitives(rCandidate, rTarget, aNewTarget);
219                             }
220                         }
221                     }
222 
223                     break;
224                 }
225                 default:
226                 {
227                     OSL_ENSURE(false, "Unexpected node in text token (!)");
228                     break;
229                 }
230             }
231         }
232 
233         void SvgTextNode::decomposeSvgNode(drawinglayer::primitive2d::Primitive2DSequence& rTarget, bool /*bReferenced`*/) const
234         {
235             // text has a group of child nodes, allowed are SVGTokenCharacter, SVGTokenTspan,
236             // SVGTokenTref and SVGTokenTextPath. These increase a given current text position
237             const SvgStyleAttributes* pStyle = getSvgStyleAttributes();
238 
239             if(pStyle && !getChildren().empty())
240             {
241                 const double fOpacity(pStyle->getOpacity().getNumber());
242 
243                 if(fOpacity > 0.0)
244                 {
245                     SvgTextPosition aSvgTextPosition(0, *this, getSvgTextPositions());
246                     drawinglayer::primitive2d::Primitive2DSequence aNewTarget;
247                     const SvgNodeVector& rChildren = getChildren();
248                     const sal_uInt32 nCount(rChildren.size());
249 
250                     for(sal_uInt32 a(0); a < nCount; a++)
251                     {
252                         const SvgNode& rCandidate = *rChildren[a];
253 
254                         DecomposeChild(rCandidate, aNewTarget, aSvgTextPosition);
255                     }
256 
257                     if(aNewTarget.hasElements())
258                     {
259                         drawinglayer::primitive2d::Primitive2DSequence aNewTarget2;
260 
261                         addTextPrimitives(*this, aNewTarget2, aNewTarget);
262                         aNewTarget = aNewTarget2;
263                     }
264 
265                     if(aNewTarget.hasElements())
266                     {
267                         pStyle->add_postProcess(rTarget, aNewTarget, getTransform());
268                     }
269                 }
270             }
271         }
272     } // end of namespace svgreader
273 } // end of namespace svgio
274 
275 //////////////////////////////////////////////////////////////////////////////
276 // eof
277