290 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			290 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*************************************************************************
 | |
|  *
 | |
|  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 | |
|  *
 | |
|  * Copyright 2000, 2010 Oracle and/or its affiliates.
 | |
|  *
 | |
|  * OpenOffice.org - a multi-platform office productivity suite
 | |
|  *
 | |
|  * This file is part of OpenOffice.org.
 | |
|  *
 | |
|  * OpenOffice.org is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Lesser General Public License version 3
 | |
|  * only, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * OpenOffice.org is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Lesser General Public License version 3 for more details
 | |
|  * (a copy is included in the LICENSE file that accompanied this code).
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public License
 | |
|  * version 3 along with OpenOffice.org.  If not, see
 | |
|  * <http://www.openoffice.org/license.html>
 | |
|  * for a copy of the LGPLv3 License.
 | |
|  *
 | |
|  ************************************************************************/
 | |
| 
 | |
| // MARKER(update_precomp.py): autogen include statement, do not remove
 | |
| #include "precompiled_drawinglayer.hxx"
 | |
| 
 | |
| #include <vclhelpergradient.hxx>
 | |
| #include <basegfx/range/b2drange.hxx>
 | |
| #include <vcl/outdev.hxx>
 | |
| #include <basegfx/polygon/b2dpolygon.hxx>
 | |
| #include <basegfx/polygon/b2dpolypolygontools.hxx>
 | |
| #include <basegfx/polygon/b2dpolygontools.hxx>
 | |
| #include <drawinglayer/texture/texture.hxx>
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////////////
 | |
| // support methods for vcl direct gradient renderering
 | |
| 
 | |
| namespace drawinglayer
 | |
| {
 | |
|     namespace
 | |
|     {
 | |
|         sal_uInt32 impCalcGradientSteps(OutputDevice& rOutDev, sal_uInt32 nSteps, const basegfx::B2DRange& rRange, sal_uInt32 nMaxDist)
 | |
|         {
 | |
|             if(nSteps == 0L)
 | |
|             {
 | |
|                 const Size aSize(rOutDev.LogicToPixel(Size(basegfx::fround(rRange.getWidth()), basegfx::fround(rRange.getHeight()))));
 | |
|                 nSteps = (aSize.getWidth() + aSize.getHeight()) >> 3L;
 | |
|             }
 | |
| 
 | |
|             if(nSteps < 2L)
 | |
|             {
 | |
|                 nSteps = 2L;
 | |
|             }
 | |
| 
 | |
|             if(nSteps > nMaxDist)
 | |
|             {
 | |
|                 nSteps = nMaxDist;
 | |
|             }
 | |
| 
 | |
|             return nSteps;
 | |
|         }
 | |
| 
 | |
|         void impDrawGradientToOutDevSimple(
 | |
|             OutputDevice& rOutDev,
 | |
|             const basegfx::B2DPolyPolygon& rTargetForm,
 | |
|             const ::std::vector< basegfx::B2DHomMatrix >& rMatrices,
 | |
|             const ::std::vector< basegfx::BColor >& rColors,
 | |
|             const basegfx::B2DPolygon& rUnitPolygon)
 | |
|         {
 | |
|             rOutDev.SetLineColor();
 | |
| 
 | |
|             for(sal_uInt32 a(0L); a < rColors.size(); a++)
 | |
|             {
 | |
|                 // set correct color
 | |
|                 const basegfx::BColor aFillColor(rColors[a]);
 | |
|                 rOutDev.SetFillColor(Color(aFillColor));
 | |
| 
 | |
|                 if(a)
 | |
|                 {
 | |
|                     if(a - 1L < static_cast< sal_uInt32 >(rMatrices.size()))
 | |
|                     {
 | |
|                         basegfx::B2DPolygon aNewPoly(rUnitPolygon);
 | |
|                         aNewPoly.transform(rMatrices[a - 1L]);
 | |
|                         rOutDev.DrawPolygon(aNewPoly);
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     rOutDev.DrawPolyPolygon(rTargetForm);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void impDrawGradientToOutDevComplex(
 | |
|             OutputDevice& rOutDev,
 | |
|             const basegfx::B2DPolyPolygon& rTargetForm,
 | |
|             const ::std::vector< basegfx::B2DHomMatrix >& rMatrices,
 | |
|             const ::std::vector< basegfx::BColor >& rColors,
 | |
|             const basegfx::B2DPolygon& rUnitPolygon)
 | |
|         {
 | |
|             PolyPolygon aVclTargetForm(rTargetForm);
 | |
|             ::std::vector< Polygon > aVclPolygons;
 | |
|             sal_uInt32 a;
 | |
| 
 | |
|             // remember and set to XOR
 | |
|             rOutDev.SetLineColor();
 | |
|             rOutDev.Push(PUSH_RASTEROP);
 | |
|             rOutDev.SetRasterOp(ROP_XOR);
 | |
| 
 | |
|             // draw gradient PolyPolygons
 | |
|             for(a = 0L; a < rMatrices.size(); a++)
 | |
|             {
 | |
|                 // create polygon and remember
 | |
|                 basegfx::B2DPolygon aNewPoly(rUnitPolygon);
 | |
|                 aNewPoly.transform(rMatrices[a]);
 | |
|                 aVclPolygons.push_back(Polygon(aNewPoly));
 | |
| 
 | |
|                 // set correct color
 | |
|                 if(rColors.size() > a)
 | |
|                 {
 | |
|                     const basegfx::BColor aFillColor(rColors[a]);
 | |
|                     rOutDev.SetFillColor(Color(aFillColor));
 | |
|                 }
 | |
| 
 | |
|                 // create vcl PolyPolygon and draw it
 | |
|                 if(a)
 | |
|                 {
 | |
|                     PolyPolygon aVclPolyPoly(aVclPolygons[a - 1L]);
 | |
|                     aVclPolyPoly.Insert(aVclPolygons[a]);
 | |
|                     rOutDev.DrawPolyPolygon(aVclPolyPoly);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     PolyPolygon aVclPolyPoly(aVclTargetForm);
 | |
|                     aVclPolyPoly.Insert(aVclPolygons[0L]);
 | |
|                     rOutDev.DrawPolyPolygon(aVclPolyPoly);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // draw last poly in last color
 | |
|             if(rColors.size())
 | |
|             {
 | |
|                 const basegfx::BColor aFillColor(rColors[rColors.size() - 1L]);
 | |
|                 rOutDev.SetFillColor(Color(aFillColor));
 | |
|                 rOutDev.DrawPolygon(aVclPolygons[aVclPolygons.size() - 1L]);
 | |
|             }
 | |
| 
 | |
|             // draw object form in black and go back to XOR
 | |
|             rOutDev.SetFillColor(COL_BLACK);
 | |
|             rOutDev.SetRasterOp(ROP_0);
 | |
|             rOutDev.DrawPolyPolygon(aVclTargetForm);
 | |
|             rOutDev.SetRasterOp(ROP_XOR);
 | |
| 
 | |
|             // draw gradient PolyPolygons again
 | |
|             for(a = 0L; a < rMatrices.size(); a++)
 | |
|             {
 | |
|                 // set correct color
 | |
|                 if(rColors.size() > a)
 | |
|                 {
 | |
|                     const basegfx::BColor aFillColor(rColors[a]);
 | |
|                     rOutDev.SetFillColor(Color(aFillColor));
 | |
|                 }
 | |
| 
 | |
|                 // create vcl PolyPolygon and draw it
 | |
|                 if(a)
 | |
|                 {
 | |
|                     PolyPolygon aVclPolyPoly(aVclPolygons[a - 1L]);
 | |
|                     aVclPolyPoly.Insert(aVclPolygons[a]);
 | |
|                     rOutDev.DrawPolyPolygon(aVclPolyPoly);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     PolyPolygon aVclPolyPoly(aVclTargetForm);
 | |
|                     aVclPolyPoly.Insert(aVclPolygons[0L]);
 | |
|                     rOutDev.DrawPolyPolygon(aVclPolyPoly);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // draw last poly in last color
 | |
|             if(rColors.size())
 | |
|             {
 | |
|                 const basegfx::BColor aFillColor(rColors[rColors.size() - 1L]);
 | |
|                 rOutDev.SetFillColor(Color(aFillColor));
 | |
|                 rOutDev.DrawPolygon(aVclPolygons[aVclPolygons.size() - 1L]);
 | |
|             }
 | |
| 
 | |
|             // reset drawmode
 | |
|             rOutDev.Pop();
 | |
|         }
 | |
|     } // end of anonymous namespace
 | |
| } // end of namespace drawinglayer
 | |
| 
 | |
| namespace drawinglayer
 | |
| {
 | |
|     void impDrawGradientToOutDev(
 | |
|         OutputDevice& rOutDev,
 | |
|         const basegfx::B2DPolyPolygon& rTargetForm,
 | |
|         attribute::GradientStyle eGradientStyle,
 | |
|         sal_uInt32 nSteps,
 | |
|         const basegfx::BColor& rStart,
 | |
|         const basegfx::BColor& rEnd,
 | |
|         double fBorder, double fAngle, double fOffsetX, double fOffsetY, bool bSimple)
 | |
|     {
 | |
|         const basegfx::B2DRange aOutlineRange(basegfx::tools::getRange(rTargetForm));
 | |
|         ::std::vector< basegfx::B2DHomMatrix > aMatrices;
 | |
|         ::std::vector< basegfx::BColor > aColors;
 | |
|         basegfx::B2DPolygon aUnitPolygon;
 | |
| 
 | |
|         if(attribute::GRADIENTSTYLE_RADIAL == eGradientStyle || attribute::GRADIENTSTYLE_ELLIPTICAL == eGradientStyle)
 | |
|         {
 | |
|             const basegfx::B2DPoint aCircleCenter(0.5, 0.5);
 | |
|             aUnitPolygon = basegfx::tools::createPolygonFromEllipse(aCircleCenter, 0.5, 0.5);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             aUnitPolygon = basegfx::tools::createPolygonFromRect(basegfx::B2DRange(0.0, 0.0, 1.0, 1.0));
 | |
|         }
 | |
| 
 | |
|         // make sure steps is not too high/low
 | |
|         nSteps = impCalcGradientSteps(rOutDev, nSteps, aOutlineRange, sal_uInt32((rStart.getMaximumDistance(rEnd) * 127.5) + 0.5));
 | |
| 
 | |
|         // create geometries
 | |
|         switch(eGradientStyle)
 | |
|         {
 | |
|             case attribute::GRADIENTSTYLE_LINEAR:
 | |
|             {
 | |
|                 texture::GeoTexSvxGradientLinear aGradient(aOutlineRange, rStart, rEnd, nSteps, fBorder, fAngle);
 | |
|                 aGradient.appendTransformations(aMatrices);
 | |
|                 aGradient.appendColors(aColors);
 | |
|                 break;
 | |
|             }
 | |
|             case attribute::GRADIENTSTYLE_AXIAL:
 | |
|             {
 | |
|                 texture::GeoTexSvxGradientAxial aGradient(aOutlineRange, rStart, rEnd, nSteps, fBorder, fAngle);
 | |
|                 aGradient.appendTransformations(aMatrices);
 | |
|                 aGradient.appendColors(aColors);
 | |
|                 break;
 | |
|             }
 | |
|             case attribute::GRADIENTSTYLE_RADIAL:
 | |
|             {
 | |
|                 texture::GeoTexSvxGradientRadial aGradient(aOutlineRange, rStart, rEnd, nSteps, fBorder, fOffsetX, fOffsetY);
 | |
|                 aGradient.appendTransformations(aMatrices);
 | |
|                 aGradient.appendColors(aColors);
 | |
|                 break;
 | |
|             }
 | |
|             case attribute::GRADIENTSTYLE_ELLIPTICAL:
 | |
|             {
 | |
|                 texture::GeoTexSvxGradientElliptical aGradient(aOutlineRange, rStart, rEnd, nSteps, fBorder, fOffsetX, fOffsetX, fAngle);
 | |
|                 aGradient.appendTransformations(aMatrices);
 | |
|                 aGradient.appendColors(aColors);
 | |
|                 break;
 | |
|             }
 | |
|             case attribute::GRADIENTSTYLE_SQUARE:
 | |
|             {
 | |
|                 texture::GeoTexSvxGradientSquare aGradient(aOutlineRange, rStart, rEnd, nSteps, fBorder, fOffsetX, fOffsetX, fAngle);
 | |
|                 aGradient.appendTransformations(aMatrices);
 | |
|                 aGradient.appendColors(aColors);
 | |
|                 break;
 | |
|             }
 | |
|             case attribute::GRADIENTSTYLE_RECT:
 | |
|             {
 | |
|                 texture::GeoTexSvxGradientRect aGradient(aOutlineRange, rStart, rEnd, nSteps, fBorder, fOffsetX, fOffsetX, fAngle);
 | |
|                 aGradient.appendTransformations(aMatrices);
 | |
|                 aGradient.appendColors(aColors);
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // paint them with mask using the XOR method
 | |
|         if(aMatrices.size())
 | |
|         {
 | |
|             if(bSimple)
 | |
|             {
 | |
|                 impDrawGradientToOutDevSimple(rOutDev, rTargetForm, aMatrices, aColors, aUnitPolygon);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 impDrawGradientToOutDevComplex(rOutDev, rTargetForm, aMatrices, aColors, aUnitPolygon);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| } // end of namespace drawinglayer
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////////////
 | |
| // eof
 |