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 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_drawinglayer.hxx" 26 27 #include <drawinglayer/primitive3d/sdrdecompositiontools3d.hxx> 28 #include <basegfx/polygon/b3dpolygon.hxx> 29 #include <drawinglayer/attribute/strokeattribute.hxx> 30 #include <drawinglayer/primitive3d/baseprimitive3d.hxx> 31 #include <drawinglayer/primitive3d/polygonprimitive3d.hxx> 32 #include <basegfx/polygon/b3dpolypolygon.hxx> 33 #include <drawinglayer/primitive3d/polypolygonprimitive3d.hxx> 34 #include <vcl/vclenum.hxx> 35 #include <drawinglayer/attribute/fillbitmapattribute.hxx> 36 #include <drawinglayer/attribute/sdrfillbitmapattribute.hxx> 37 #include <vcl/bmpacc.hxx> 38 #include <basegfx/polygon/b3dpolypolygontools.hxx> 39 #include <drawinglayer/primitive3d/textureprimitive3d.hxx> 40 #include <drawinglayer/primitive3d/modifiedcolorprimitive3d.hxx> 41 #include <drawinglayer/primitive3d/hatchtextureprimitive3d.hxx> 42 #include <drawinglayer/primitive3d/shadowprimitive3d.hxx> 43 #include <basegfx/range/b2drange.hxx> 44 #include <drawinglayer/attribute/sdrlineattribute.hxx> 45 #include <drawinglayer/attribute/sdrobjectattribute3d.hxx> 46 #include <drawinglayer/attribute/sdrfillattribute.hxx> 47 #include <drawinglayer/attribute/sdrshadowattribute.hxx> 48 #include <drawinglayer/primitive3d/hiddengeometryprimitive3d.hxx> 49 50 ////////////////////////////////////////////////////////////////////////////// 51 52 namespace drawinglayer 53 { 54 namespace primitive3d 55 { 56 basegfx::B3DRange getRangeFrom3DGeometry(::std::vector< basegfx::B3DPolyPolygon >& rFill) 57 { 58 basegfx::B3DRange aRetval; 59 60 for(sal_uInt32 a(0); a < rFill.size(); a++) 61 { 62 aRetval.expand(basegfx::tools::getRange(rFill[a])); 63 } 64 65 return aRetval; 66 } 67 68 void applyNormalsKindSphereTo3DGeometry(::std::vector< basegfx::B3DPolyPolygon >& rFill, const basegfx::B3DRange& rRange) 69 { 70 // create sphere normals 71 const basegfx::B3DPoint aCenter(rRange.getCenter()); 72 73 for(sal_uInt32 a(0); a < rFill.size(); a++) 74 { 75 rFill[a] = basegfx::tools::applyDefaultNormalsSphere(rFill[a], aCenter); 76 } 77 } 78 79 void applyNormalsKindFlatTo3DGeometry(::std::vector< basegfx::B3DPolyPolygon >& rFill) 80 { 81 for(sal_uInt32 a(0); a < rFill.size(); a++) 82 { 83 rFill[a].clearNormals(); 84 } 85 } 86 87 void applyNormalsInvertTo3DGeometry(::std::vector< basegfx::B3DPolyPolygon >& rFill) 88 { 89 // invert normals 90 for(sal_uInt32 a(0); a < rFill.size(); a++) 91 { 92 rFill[a] = basegfx::tools::invertNormals(rFill[a]); 93 } 94 } 95 96 void applyTextureTo3DGeometry( 97 ::com::sun::star::drawing::TextureProjectionMode eModeX, 98 ::com::sun::star::drawing::TextureProjectionMode eModeY, 99 ::std::vector< basegfx::B3DPolyPolygon >& rFill, 100 const basegfx::B3DRange& rRange, 101 const basegfx::B2DVector& rTextureSize) 102 { 103 sal_uInt32 a; 104 105 // handle texture coordinates X 106 const bool bParallelX(::com::sun::star::drawing::TextureProjectionMode_PARALLEL == eModeX); 107 const bool bSphereX(!bParallelX && (::com::sun::star::drawing::TextureProjectionMode_SPHERE == eModeX)); 108 109 // handle texture coordinates Y 110 const bool bParallelY(::com::sun::star::drawing::TextureProjectionMode_PARALLEL == eModeY); 111 const bool bSphereY(!bParallelY && (::com::sun::star::drawing::TextureProjectionMode_SPHERE == eModeY)); 112 113 if(bParallelX || bParallelY) 114 { 115 // apply parallel texture coordinates in X and/or Y 116 for(a = 0; a < rFill.size(); a++) 117 { 118 rFill[a] = basegfx::tools::applyDefaultTextureCoordinatesParallel(rFill[a], rRange, bParallelX, bParallelY); 119 } 120 } 121 122 if(bSphereX || bSphereY) 123 { 124 // apply spherical texture coordinates in X and/or Y 125 const basegfx::B3DPoint aCenter(rRange.getCenter()); 126 127 for(a = 0; a < rFill.size(); a++) 128 { 129 rFill[a] = basegfx::tools::applyDefaultTextureCoordinatesSphere(rFill[a], aCenter, bSphereX, bSphereY); 130 } 131 } 132 133 // transform texture coordinates to texture size 134 basegfx::B2DHomMatrix aTexMatrix; 135 aTexMatrix.scale(rTextureSize.getX(), rTextureSize.getY()); 136 137 for(a = 0; a < rFill.size(); a++) 138 { 139 rFill[a].transformTextureCoordiantes(aTexMatrix); 140 } 141 } 142 143 Primitive3DSequence create3DPolyPolygonLinePrimitives( 144 const basegfx::B3DPolyPolygon& rUnitPolyPolygon, 145 const basegfx::B3DHomMatrix& rObjectTransform, 146 const attribute::SdrLineAttribute& rLine) 147 { 148 // prepare fully scaled polyPolygon 149 basegfx::B3DPolyPolygon aScaledPolyPolygon(rUnitPolyPolygon); 150 aScaledPolyPolygon.transform(rObjectTransform); 151 152 // create line and stroke attribute 153 const attribute::LineAttribute aLineAttribute(rLine.getColor(), rLine.getWidth(), rLine.getJoin(), rLine.getCap()); 154 const attribute::StrokeAttribute aStrokeAttribute(rLine.getDotDashArray(), rLine.getFullDotDashLen()); 155 156 // create primitives 157 Primitive3DSequence aRetval(aScaledPolyPolygon.count()); 158 159 for(sal_uInt32 a(0L); a < aScaledPolyPolygon.count(); a++) 160 { 161 const Primitive3DReference xRef(new PolygonStrokePrimitive3D(aScaledPolyPolygon.getB3DPolygon(a), aLineAttribute, aStrokeAttribute)); 162 aRetval[a] = xRef; 163 } 164 165 if(0.0 != rLine.getTransparence()) 166 { 167 // create UnifiedTransparenceTexturePrimitive3D, add created primitives and exchange 168 const Primitive3DReference xRef(new UnifiedTransparenceTexturePrimitive3D(rLine.getTransparence(), aRetval)); 169 aRetval = Primitive3DSequence(&xRef, 1L); 170 } 171 172 return aRetval; 173 } 174 175 Primitive3DSequence create3DPolyPolygonFillPrimitives( 176 const ::std::vector< basegfx::B3DPolyPolygon >& r3DPolyPolygonVector, 177 const basegfx::B3DHomMatrix& rObjectTransform, 178 const basegfx::B2DVector& rTextureSize, 179 const attribute::Sdr3DObjectAttribute& aSdr3DObjectAttribute, 180 const attribute::SdrFillAttribute& rFill, 181 const attribute::FillGradientAttribute& rFillGradient) 182 { 183 Primitive3DSequence aRetval; 184 185 if(r3DPolyPolygonVector.size()) 186 { 187 // create list of simple fill primitives 188 aRetval.realloc(r3DPolyPolygonVector.size()); 189 190 for(sal_uInt32 a(0L); a < r3DPolyPolygonVector.size(); a++) 191 { 192 // get scaled PolyPolygon 193 basegfx::B3DPolyPolygon aScaledPolyPolygon(r3DPolyPolygonVector[a]); 194 aScaledPolyPolygon.transform(rObjectTransform); 195 196 if(aScaledPolyPolygon.areNormalsUsed()) 197 { 198 aScaledPolyPolygon.transformNormals(rObjectTransform); 199 } 200 201 const Primitive3DReference xRef(new PolyPolygonMaterialPrimitive3D( 202 aScaledPolyPolygon, 203 aSdr3DObjectAttribute.getMaterial(), 204 aSdr3DObjectAttribute.getDoubleSided())); 205 aRetval[a] = xRef; 206 } 207 208 // look for and evtl. build texture sub-group primitive 209 if(!rFill.getGradient().isDefault() 210 || !rFill.getHatch().isDefault() 211 || !rFill.getBitmap().isDefault()) 212 { 213 bool bModulate(::com::sun::star::drawing::TextureMode_MODULATE == aSdr3DObjectAttribute.getTextureMode()); 214 bool bFilter(aSdr3DObjectAttribute.getTextureFilter()); 215 BasePrimitive3D* pNewTexturePrimitive3D = 0; 216 217 if(!rFill.getGradient().isDefault()) 218 { 219 // create gradientTexture3D with sublist, add to local aRetval 220 pNewTexturePrimitive3D = new GradientTexturePrimitive3D( 221 rFill.getGradient(), 222 aRetval, 223 rTextureSize, 224 bModulate, 225 bFilter); 226 } 227 else if(!rFill.getHatch().isDefault()) 228 { 229 // create hatchTexture3D with sublist, add to local aRetval 230 pNewTexturePrimitive3D = new HatchTexturePrimitive3D( 231 rFill.getHatch(), 232 aRetval, 233 rTextureSize, 234 bModulate, 235 bFilter); 236 } 237 else // if(!rFill.getBitmap().isDefault()) 238 { 239 // create bitmapTexture3D with sublist, add to local aRetval 240 basegfx::B2DRange aTexRange(0.0, 0.0, rTextureSize.getX(), rTextureSize.getY()); 241 242 pNewTexturePrimitive3D = new BitmapTexturePrimitive3D( 243 rFill.getBitmap().getFillBitmapAttribute(aTexRange), 244 aRetval, 245 rTextureSize, 246 bModulate, 247 bFilter); 248 } 249 250 // exchange aRetval content with texture group 251 const Primitive3DReference xRef(pNewTexturePrimitive3D); 252 aRetval = Primitive3DSequence(&xRef, 1L); 253 254 if(::com::sun::star::drawing::TextureKind2_LUMINANCE == aSdr3DObjectAttribute.getTextureKind()) 255 { 256 // use modified color primitive to force textures to gray 257 const basegfx::BColorModifier aBColorModifier(basegfx::BColor(), 0.0, basegfx::BCOLORMODIFYMODE_GRAY); 258 const Primitive3DReference xRef2(new ModifiedColorPrimitive3D(aRetval, aBColorModifier)); 259 aRetval = Primitive3DSequence(&xRef2, 1L); 260 } 261 } 262 263 if(0.0 != rFill.getTransparence()) 264 { 265 // create UnifiedTransparenceTexturePrimitive3D with sublist and exchange 266 const Primitive3DReference xRef(new UnifiedTransparenceTexturePrimitive3D(rFill.getTransparence(), aRetval)); 267 aRetval = Primitive3DSequence(&xRef, 1L); 268 } 269 else if(!rFillGradient.isDefault()) 270 { 271 // create TransparenceTexturePrimitive3D with sublist and exchange 272 const Primitive3DReference xRef(new TransparenceTexturePrimitive3D(rFillGradient, aRetval, rTextureSize)); 273 aRetval = Primitive3DSequence(&xRef, 1L); 274 } 275 } 276 277 return aRetval; 278 } 279 280 Primitive3DSequence createShadowPrimitive3D( 281 const Primitive3DSequence& rSource, 282 const attribute::SdrShadowAttribute& rShadow, 283 bool bShadow3D) 284 { 285 // create Shadow primitives. Uses already created primitives 286 if(rSource.hasElements() && !basegfx::fTools::moreOrEqual(rShadow.getTransparence(), 1.0)) 287 { 288 // prepare new list for shadow geometry 289 basegfx::B2DHomMatrix aShadowOffset; 290 aShadowOffset.set(0, 2, rShadow.getOffset().getX()); 291 aShadowOffset.set(1, 2, rShadow.getOffset().getY()); 292 293 // create shadow primitive and add primitives 294 const Primitive3DReference xRef(new ShadowPrimitive3D(aShadowOffset, rShadow.getColor(), rShadow.getTransparence(), bShadow3D, rSource)); 295 return Primitive3DSequence(&xRef, 1L); 296 } 297 else 298 { 299 return Primitive3DSequence(); 300 } 301 } 302 303 Primitive3DSequence createHiddenGeometryPrimitives3D( 304 const ::std::vector< basegfx::B3DPolyPolygon >& r3DPolyPolygonVector, 305 const basegfx::B3DHomMatrix& rObjectTransform, 306 const basegfx::B2DVector& rTextureSize, 307 const attribute::Sdr3DObjectAttribute& aSdr3DObjectAttribute) 308 { 309 // create hidden sub-geometry which can be used for HitTest 310 // and BoundRect calculations, but will not be visualized 311 const attribute::SdrFillAttribute aSimplifiedFillAttribute( 312 0.0, 313 basegfx::BColor(), 314 attribute::FillGradientAttribute(), 315 attribute::FillHatchAttribute(), 316 attribute::SdrFillBitmapAttribute()); 317 318 const Primitive3DReference aHidden( 319 new HiddenGeometryPrimitive3D( 320 create3DPolyPolygonFillPrimitives( 321 r3DPolyPolygonVector, 322 rObjectTransform, 323 rTextureSize, 324 aSdr3DObjectAttribute, 325 aSimplifiedFillAttribute, 326 attribute::FillGradientAttribute()))); 327 328 return Primitive3DSequence(&aHidden, 1); 329 } 330 331 } // end of namespace primitive3d 332 } // end of namespace drawinglayer 333 334 ////////////////////////////////////////////////////////////////////////////// 335 // eof 336