1ddde725dSArmin Le Grand /************************************************************** 2ddde725dSArmin Le Grand * 3ddde725dSArmin Le Grand * Licensed to the Apache Software Foundation (ASF) under one 4ddde725dSArmin Le Grand * or more contributor license agreements. See the NOTICE file 5ddde725dSArmin Le Grand * distributed with this work for additional information 6ddde725dSArmin Le Grand * regarding copyright ownership. The ASF licenses this file 7ddde725dSArmin Le Grand * to you under the Apache License, Version 2.0 (the 8ddde725dSArmin Le Grand * "License"); you may not use this file except in compliance 9ddde725dSArmin Le Grand * with the License. You may obtain a copy of the License at 10ddde725dSArmin Le Grand * 11ddde725dSArmin Le Grand * http://www.apache.org/licenses/LICENSE-2.0 12ddde725dSArmin Le Grand * 13ddde725dSArmin Le Grand * Unless required by applicable law or agreed to in writing, 14ddde725dSArmin Le Grand * software distributed under the License is distributed on an 15ddde725dSArmin Le Grand * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16ddde725dSArmin Le Grand * KIND, either express or implied. See the License for the 17ddde725dSArmin Le Grand * specific language governing permissions and limitations 18ddde725dSArmin Le Grand * under the License. 19ddde725dSArmin Le Grand * 20ddde725dSArmin Le Grand *************************************************************/ 21ddde725dSArmin Le Grand 22ddde725dSArmin Le Grand // MARKER(update_precomp.py): autogen include statement, do not remove 23ddde725dSArmin Le Grand #include "precompiled_svgio.hxx" 24ddde725dSArmin Le Grand 25ddde725dSArmin Le Grand #include <svgio/svgreader/svgmasknode.hxx> 26ddde725dSArmin Le Grand #include <drawinglayer/primitive2d/transformprimitive2d.hxx> 27ddde725dSArmin Le Grand #include <drawinglayer/primitive2d/transparenceprimitive2d.hxx> 28ddde725dSArmin Le Grand #include <basegfx/matrix/b2dhommatrixtools.hxx> 29ddde725dSArmin Le Grand #include <drawinglayer/geometry/viewinformation2d.hxx> 30ddde725dSArmin Le Grand #include <drawinglayer/primitive2d/modifiedcolorprimitive2d.hxx> 31ddde725dSArmin Le Grand #include <drawinglayer/primitive2d/maskprimitive2d.hxx> 32ddde725dSArmin Le Grand #include <basegfx/polygon/b2dpolygontools.hxx> 33ddde725dSArmin Le Grand #include <basegfx/polygon/b2dpolygon.hxx> 34ddde725dSArmin Le Grand 35ddde725dSArmin Le Grand ////////////////////////////////////////////////////////////////////////////// 36ddde725dSArmin Le Grand 37ddde725dSArmin Le Grand namespace svgio 38ddde725dSArmin Le Grand { 39ddde725dSArmin Le Grand namespace svgreader 40ddde725dSArmin Le Grand { 41ddde725dSArmin Le Grand SvgMaskNode::SvgMaskNode( 42ddde725dSArmin Le Grand SvgDocument& rDocument, 43ddde725dSArmin Le Grand SvgNode* pParent) 44ddde725dSArmin Le Grand : SvgNode(SVGTokenMask, rDocument, pParent), 45ddde725dSArmin Le Grand maSvgStyleAttributes(*this), 46ddde725dSArmin Le Grand maX(SvgNumber(-10.0, Unit_percent, true)), 47ddde725dSArmin Le Grand maY(SvgNumber(-10.0, Unit_percent, true)), 48ddde725dSArmin Le Grand maWidth(SvgNumber(120.0, Unit_percent, true)), 49ddde725dSArmin Le Grand maHeight(SvgNumber(120.0, Unit_percent, true)), 50ddde725dSArmin Le Grand mpaTransform(0), 51ddde725dSArmin Le Grand maMaskUnits(objectBoundingBox), 52ddde725dSArmin Le Grand maMaskContentUnits(userSpaceOnUse) 53ddde725dSArmin Le Grand { 54ddde725dSArmin Le Grand } 55ddde725dSArmin Le Grand 56ddde725dSArmin Le Grand SvgMaskNode::~SvgMaskNode() 57ddde725dSArmin Le Grand { 58ddde725dSArmin Le Grand if(mpaTransform) delete mpaTransform; 59ddde725dSArmin Le Grand } 60ddde725dSArmin Le Grand 61ddde725dSArmin Le Grand const SvgStyleAttributes* SvgMaskNode::getSvgStyleAttributes() const 62ddde725dSArmin Le Grand { 63ddde725dSArmin Le Grand static rtl::OUString aClassStr(rtl::OUString::createFromAscii("mask")); 64ddde725dSArmin Le Grand maSvgStyleAttributes.checkForCssStyle(aClassStr); 65ddde725dSArmin Le Grand 66ddde725dSArmin Le Grand return &maSvgStyleAttributes; 67ddde725dSArmin Le Grand } 68ddde725dSArmin Le Grand 69ddde725dSArmin Le Grand void SvgMaskNode::parseAttribute(const rtl::OUString& rTokenName, SVGToken aSVGToken, const rtl::OUString& aContent) 70ddde725dSArmin Le Grand { 71ddde725dSArmin Le Grand // call parent 72ddde725dSArmin Le Grand SvgNode::parseAttribute(rTokenName, aSVGToken, aContent); 73ddde725dSArmin Le Grand 74ddde725dSArmin Le Grand // read style attributes 75ddde725dSArmin Le Grand maSvgStyleAttributes.parseStyleAttribute(rTokenName, aSVGToken, aContent); 76ddde725dSArmin Le Grand 77ddde725dSArmin Le Grand // parse own 78ddde725dSArmin Le Grand switch(aSVGToken) 79ddde725dSArmin Le Grand { 80ddde725dSArmin Le Grand case SVGTokenStyle: 81ddde725dSArmin Le Grand { 82ddde725dSArmin Le Grand maSvgStyleAttributes.readStyle(aContent); 83ddde725dSArmin Le Grand break; 84ddde725dSArmin Le Grand } 85ddde725dSArmin Le Grand case SVGTokenX: 86ddde725dSArmin Le Grand { 87ddde725dSArmin Le Grand SvgNumber aNum; 88ddde725dSArmin Le Grand 89ddde725dSArmin Le Grand if(readSingleNumber(aContent, aNum)) 90ddde725dSArmin Le Grand { 91ddde725dSArmin Le Grand setX(aNum); 92ddde725dSArmin Le Grand } 93ddde725dSArmin Le Grand break; 94ddde725dSArmin Le Grand } 95ddde725dSArmin Le Grand case SVGTokenY: 96ddde725dSArmin Le Grand { 97ddde725dSArmin Le Grand SvgNumber aNum; 98ddde725dSArmin Le Grand 99ddde725dSArmin Le Grand if(readSingleNumber(aContent, aNum)) 100ddde725dSArmin Le Grand { 101ddde725dSArmin Le Grand setY(aNum); 102ddde725dSArmin Le Grand } 103ddde725dSArmin Le Grand break; 104ddde725dSArmin Le Grand } 105ddde725dSArmin Le Grand case SVGTokenWidth: 106ddde725dSArmin Le Grand { 107ddde725dSArmin Le Grand SvgNumber aNum; 108ddde725dSArmin Le Grand 109ddde725dSArmin Le Grand if(readSingleNumber(aContent, aNum)) 110ddde725dSArmin Le Grand { 111ddde725dSArmin Le Grand if(aNum.isPositive()) 112ddde725dSArmin Le Grand { 113ddde725dSArmin Le Grand setWidth(aNum); 114ddde725dSArmin Le Grand } 115ddde725dSArmin Le Grand } 116ddde725dSArmin Le Grand break; 117ddde725dSArmin Le Grand } 118ddde725dSArmin Le Grand case SVGTokenHeight: 119ddde725dSArmin Le Grand { 120ddde725dSArmin Le Grand SvgNumber aNum; 121ddde725dSArmin Le Grand 122ddde725dSArmin Le Grand if(readSingleNumber(aContent, aNum)) 123ddde725dSArmin Le Grand { 124ddde725dSArmin Le Grand if(aNum.isPositive()) 125ddde725dSArmin Le Grand { 126ddde725dSArmin Le Grand setHeight(aNum); 127ddde725dSArmin Le Grand } 128ddde725dSArmin Le Grand } 129ddde725dSArmin Le Grand break; 130ddde725dSArmin Le Grand } 131ddde725dSArmin Le Grand case SVGTokenTransform: 132ddde725dSArmin Le Grand { 133ddde725dSArmin Le Grand const basegfx::B2DHomMatrix aMatrix(readTransform(aContent, *this)); 134ddde725dSArmin Le Grand 135ddde725dSArmin Le Grand if(!aMatrix.isIdentity()) 136ddde725dSArmin Le Grand { 137ddde725dSArmin Le Grand setTransform(&aMatrix); 138ddde725dSArmin Le Grand } 139ddde725dSArmin Le Grand break; 140ddde725dSArmin Le Grand } 141ddde725dSArmin Le Grand case SVGTokenMaskUnits: 142ddde725dSArmin Le Grand { 143ddde725dSArmin Le Grand if(aContent.getLength()) 144ddde725dSArmin Le Grand { 145ddde725dSArmin Le Grand if(aContent.match(commonStrings::aStrUserSpaceOnUse, 0)) 146ddde725dSArmin Le Grand { 147ddde725dSArmin Le Grand setMaskUnits(userSpaceOnUse); 148ddde725dSArmin Le Grand } 149ddde725dSArmin Le Grand else if(aContent.match(commonStrings::aStrObjectBoundingBox, 0)) 150ddde725dSArmin Le Grand { 151ddde725dSArmin Le Grand setMaskUnits(objectBoundingBox); 152ddde725dSArmin Le Grand } 153ddde725dSArmin Le Grand } 154ddde725dSArmin Le Grand break; 155ddde725dSArmin Le Grand } 156ddde725dSArmin Le Grand case SVGTokenMaskContentUnits: 157ddde725dSArmin Le Grand { 158ddde725dSArmin Le Grand if(aContent.getLength()) 159ddde725dSArmin Le Grand { 160ddde725dSArmin Le Grand if(aContent.match(commonStrings::aStrUserSpaceOnUse, 0)) 161ddde725dSArmin Le Grand { 162ddde725dSArmin Le Grand setMaskContentUnits(userSpaceOnUse); 163ddde725dSArmin Le Grand } 164ddde725dSArmin Le Grand else if(aContent.match(commonStrings::aStrObjectBoundingBox, 0)) 165ddde725dSArmin Le Grand { 166ddde725dSArmin Le Grand setMaskContentUnits(objectBoundingBox); 167ddde725dSArmin Le Grand } 168ddde725dSArmin Le Grand } 169ddde725dSArmin Le Grand break; 170ddde725dSArmin Le Grand } 171*e2bf1e9dSArmin Le Grand default: 172*e2bf1e9dSArmin Le Grand { 173*e2bf1e9dSArmin Le Grand break; 174*e2bf1e9dSArmin Le Grand } 175ddde725dSArmin Le Grand } 176ddde725dSArmin Le Grand } 177ddde725dSArmin Le Grand 178ddde725dSArmin Le Grand void SvgMaskNode::decomposeSvgNode(drawinglayer::primitive2d::Primitive2DSequence& rTarget, bool bReferenced) const 179ddde725dSArmin Le Grand { 180ddde725dSArmin Le Grand drawinglayer::primitive2d::Primitive2DSequence aNewTarget; 181ddde725dSArmin Le Grand 182ddde725dSArmin Le Grand // decompose childs 183ddde725dSArmin Le Grand SvgNode::decomposeSvgNode(aNewTarget, bReferenced); 184ddde725dSArmin Le Grand 185ddde725dSArmin Le Grand if(aNewTarget.hasElements()) 186ddde725dSArmin Le Grand { 187ddde725dSArmin Le Grand if(getTransform()) 188ddde725dSArmin Le Grand { 189ddde725dSArmin Le Grand // create embedding group element with transformation 190ddde725dSArmin Le Grand const drawinglayer::primitive2d::Primitive2DReference xRef( 191ddde725dSArmin Le Grand new drawinglayer::primitive2d::TransformPrimitive2D( 192ddde725dSArmin Le Grand *getTransform(), 193ddde725dSArmin Le Grand aNewTarget)); 194ddde725dSArmin Le Grand 195ddde725dSArmin Le Grand aNewTarget = drawinglayer::primitive2d::Primitive2DSequence(&xRef, 1); 196ddde725dSArmin Le Grand } 197ddde725dSArmin Le Grand 198ddde725dSArmin Le Grand // append to current target 199ddde725dSArmin Le Grand drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(rTarget, aNewTarget); 200ddde725dSArmin Le Grand } 201ddde725dSArmin Le Grand } 202ddde725dSArmin Le Grand 203ddde725dSArmin Le Grand void SvgMaskNode::apply(drawinglayer::primitive2d::Primitive2DSequence& rTarget) const 204ddde725dSArmin Le Grand { 205ddde725dSArmin Le Grand if(rTarget.hasElements()) 206ddde725dSArmin Le Grand { 207ddde725dSArmin Le Grand drawinglayer::primitive2d::Primitive2DSequence aMaskTarget; 208ddde725dSArmin Le Grand 209ddde725dSArmin Le Grand // get mask definition as primitives 210ddde725dSArmin Le Grand decomposeSvgNode(aMaskTarget, true); 211ddde725dSArmin Le Grand 212ddde725dSArmin Le Grand if(aMaskTarget.hasElements()) 213ddde725dSArmin Le Grand { 214ddde725dSArmin Le Grand // get range of content to be masked 215ddde725dSArmin Le Grand const basegfx::B2DRange aContentRange( 216ddde725dSArmin Le Grand drawinglayer::primitive2d::getB2DRangeFromPrimitive2DSequence( 217ddde725dSArmin Le Grand rTarget, 218ddde725dSArmin Le Grand drawinglayer::geometry::ViewInformation2D())); 219ddde725dSArmin Le Grand const double fContentWidth(aContentRange.getWidth()); 220ddde725dSArmin Le Grand const double fContentHeight(aContentRange.getHeight()); 221ddde725dSArmin Le Grand 222ddde725dSArmin Le Grand if(fContentWidth > 0.0 && fContentHeight > 0.0) 223ddde725dSArmin Le Grand { 224ddde725dSArmin Le Grand // create OffscreenBufferRange 225ddde725dSArmin Le Grand basegfx::B2DRange aOffscreenBufferRange; 226ddde725dSArmin Le Grand 227ddde725dSArmin Le Grand if(objectBoundingBox == getMaskUnits()) 228ddde725dSArmin Le Grand { 229ddde725dSArmin Le Grand // fractions or percentages of the bounding box of the element to which the mask is applied 230ddde725dSArmin Le Grand const double fX(Unit_percent == getX().getUnit() ? getX().getNumber() * 0.01 : getX().getNumber()); 231ddde725dSArmin Le Grand const double fY(Unit_percent == getY().getUnit() ? getY().getNumber() * 0.01 : getY().getNumber()); 232ddde725dSArmin Le Grand const double fW(Unit_percent == getWidth().getUnit() ? getWidth().getNumber() * 0.01 : getWidth().getNumber()); 233ddde725dSArmin Le Grand const double fH(Unit_percent == getHeight().getUnit() ? getHeight().getNumber() * 0.01 : getHeight().getNumber()); 234ddde725dSArmin Le Grand 235ddde725dSArmin Le Grand aOffscreenBufferRange = basegfx::B2DRange( 236ddde725dSArmin Le Grand aContentRange.getMinX() + (fX * fContentWidth), 237ddde725dSArmin Le Grand aContentRange.getMinY() + (fY * fContentHeight), 238ddde725dSArmin Le Grand aContentRange.getMinX() + ((fX + fW) * fContentWidth), 239ddde725dSArmin Le Grand aContentRange.getMinY() + ((fY + fH) * fContentHeight)); 240ddde725dSArmin Le Grand } 241ddde725dSArmin Le Grand else 242ddde725dSArmin Le Grand { 243ddde725dSArmin Le Grand const double fX(getX().isSet() ? getX().solve(*this, xcoordinate) : 0.0); 244ddde725dSArmin Le Grand const double fY(getY().isSet() ? getY().solve(*this, ycoordinate) : 0.0); 245ddde725dSArmin Le Grand 246ddde725dSArmin Le Grand aOffscreenBufferRange = basegfx::B2DRange( 247ddde725dSArmin Le Grand fX, 248ddde725dSArmin Le Grand fY, 249ddde725dSArmin Le Grand fX + (getWidth().isSet() ? getWidth().solve(*this, xcoordinate) : 0.0), 250ddde725dSArmin Le Grand fY + (getHeight().isSet() ? getHeight().solve(*this, ycoordinate) : 0.0)); 251ddde725dSArmin Le Grand } 252ddde725dSArmin Le Grand 253ddde725dSArmin Le Grand if(objectBoundingBox == getMaskContentUnits()) 254ddde725dSArmin Le Grand { 255ddde725dSArmin Le Grand // mask is object-relative, embed in content transformation 256ddde725dSArmin Le Grand const drawinglayer::primitive2d::Primitive2DReference xTransform( 257ddde725dSArmin Le Grand new drawinglayer::primitive2d::TransformPrimitive2D( 258ddde725dSArmin Le Grand basegfx::tools::createScaleTranslateB2DHomMatrix( 259ddde725dSArmin Le Grand aContentRange.getRange(), 260ddde725dSArmin Le Grand aContentRange.getMinimum()), 261ddde725dSArmin Le Grand aMaskTarget)); 262ddde725dSArmin Le Grand 263ddde725dSArmin Le Grand aMaskTarget = drawinglayer::primitive2d::Primitive2DSequence(&xTransform, 1); 264ddde725dSArmin Le Grand } 265ddde725dSArmin Le Grand 266ddde725dSArmin Le Grand // embed content to a ModifiedColorPrimitive2D since the definitions 267ddde725dSArmin Le Grand // how content is used as alpha is special for Svg 268ddde725dSArmin Le Grand { 269ddde725dSArmin Le Grand const drawinglayer::primitive2d::Primitive2DReference xInverseMask( 270ddde725dSArmin Le Grand new drawinglayer::primitive2d::ModifiedColorPrimitive2D( 271ddde725dSArmin Le Grand aMaskTarget, 272ddde725dSArmin Le Grand basegfx::BColorModifier( 273ddde725dSArmin Le Grand basegfx::BColor(0.0, 0.0, 0.0), 274ddde725dSArmin Le Grand 0.5, 275ddde725dSArmin Le Grand basegfx::BCOLORMODIFYMODE_LUMINANCE_TO_ALPHA))); 276ddde725dSArmin Le Grand 277ddde725dSArmin Le Grand aMaskTarget = drawinglayer::primitive2d::Primitive2DSequence(&xInverseMask, 1); 278ddde725dSArmin Le Grand } 279ddde725dSArmin Le Grand 280ddde725dSArmin Le Grand // prepare new content 281ddde725dSArmin Le Grand drawinglayer::primitive2d::Primitive2DReference xNewContent( 282ddde725dSArmin Le Grand new drawinglayer::primitive2d::TransparencePrimitive2D( 283ddde725dSArmin Le Grand rTarget, 284ddde725dSArmin Le Grand aMaskTarget)); 285ddde725dSArmin Le Grand 286ddde725dSArmin Le Grand // output up to now is defined by aContentRange and mask is oriented 287ddde725dSArmin Le Grand // relative to it. It is possible that aOffscreenBufferRange defines 288ddde725dSArmin Le Grand // a smaller area. In that case, embed to a mask primitive 289ddde725dSArmin Le Grand if(!aOffscreenBufferRange.isInside(aContentRange)) 290ddde725dSArmin Le Grand { 291ddde725dSArmin Le Grand xNewContent = new drawinglayer::primitive2d::MaskPrimitive2D( 292ddde725dSArmin Le Grand basegfx::B2DPolyPolygon( 293ddde725dSArmin Le Grand basegfx::tools::createPolygonFromRect( 294ddde725dSArmin Le Grand aOffscreenBufferRange)), 295ddde725dSArmin Le Grand drawinglayer::primitive2d::Primitive2DSequence(&xNewContent, 1)); 296ddde725dSArmin Le Grand } 297ddde725dSArmin Le Grand 298ddde725dSArmin Le Grand // redefine target. Use TransparencePrimitive2D with created mask 299ddde725dSArmin Le Grand // geometry 300ddde725dSArmin Le Grand rTarget = drawinglayer::primitive2d::Primitive2DSequence(&xNewContent, 1); 301ddde725dSArmin Le Grand } 302ddde725dSArmin Le Grand else 303ddde725dSArmin Le Grand { 304ddde725dSArmin Le Grand // content is geometrically empty 305ddde725dSArmin Le Grand rTarget.realloc(0); 306ddde725dSArmin Le Grand } 307ddde725dSArmin Le Grand } 308ddde725dSArmin Le Grand else 309ddde725dSArmin Le Grand { 310ddde725dSArmin Le Grand // An empty clipping path will completely clip away the element that had 311ddde725dSArmin Le Grand // the �clip-path� property applied. (Svg spec) 312ddde725dSArmin Le Grand rTarget.realloc(0); 313ddde725dSArmin Le Grand } 314ddde725dSArmin Le Grand } 315ddde725dSArmin Le Grand } 316ddde725dSArmin Le Grand 317ddde725dSArmin Le Grand } // end of namespace svgreader 318ddde725dSArmin Le Grand } // end of namespace svgio 319ddde725dSArmin Le Grand 320ddde725dSArmin Le Grand ////////////////////////////////////////////////////////////////////////////// 321ddde725dSArmin Le Grand // eof 322