diff options
39 files changed, 3904 insertions, 3080 deletions
diff --git a/canvas/source/vcl/canvashelper.cxx b/canvas/source/vcl/canvashelper.cxx index 0f3718feed64..10a6f5328f01 100644 --- a/canvas/source/vcl/canvashelper.cxx +++ b/canvas/source/vcl/canvashelper.cxx @@ -1231,7 +1231,7 @@ namespace vclcanvas // accumulate non-empty clips into one region // ========================================== - Region aClipRegion( REGION_NULL ); + Region aClipRegion(true); if( viewState.Clip.is() ) { diff --git a/cppcanvas/source/mtfrenderer/implrenderer.cxx b/cppcanvas/source/mtfrenderer/implrenderer.cxx index 2fcff5ad4ba1..a85239140b0e 100644 --- a/cppcanvas/source/mtfrenderer/implrenderer.cxx +++ b/cppcanvas/source/mtfrenderer/implrenderer.cxx @@ -1368,7 +1368,7 @@ namespace cppcanvas } else { - if( !pClipAction->GetRegion().HasPolyPolygon() ) + if( !pClipAction->GetRegion().HasPolyPolygonOrB2DPolyPolygon() ) { VERBOSE_TRACE( "ImplRenderer::createActions(): non-polygonal clip " "region encountered, falling back to bounding box!" ); @@ -1390,9 +1390,11 @@ namespace cppcanvas // with old one, just set it) // #121806# explicitly kept integer + basegfx::B2DPolyPolygon aPolyPolygon(pClipAction->GetRegion().GetAsB2DPolyPolygon()); + + aPolyPolygon.transform(rVDev.GetViewTransformation()); updateClipping( - rVDev.LogicToPixel( - pClipAction->GetRegion().GetPolyPolygon() ).getB2DPolyPolygon(), + aPolyPolygon, rFactoryParms, false ); } @@ -1422,7 +1424,7 @@ namespace cppcanvas { MetaISectRegionClipRegionAction* pClipAction = static_cast<MetaISectRegionClipRegionAction*>(pCurrAct); - if( !pClipAction->GetRegion().HasPolyPolygon() ) + if( !pClipAction->GetRegion().HasPolyPolygonOrB2DPolyPolygon() ) { VERBOSE_TRACE( "ImplRenderer::createActions(): non-polygonal clip " "region encountered, falling back to bounding box!" ); @@ -1442,9 +1444,11 @@ namespace cppcanvas // intersect current clip with given clip polygon // #121806# explicitly kept integer + basegfx::B2DPolyPolygon aPolyPolygon(pClipAction->GetRegion().GetAsB2DPolyPolygon()); + + aPolyPolygon.transform(rVDev.GetViewTransformation()); updateClipping( - rVDev.LogicToPixel( - pClipAction->GetRegion().GetPolyPolygon() ).getB2DPolyPolygon(), + aPolyPolygon, rFactoryParms, true ); } diff --git a/drawinglayer/source/primitive2d/metafileprimitive2d.cxx b/drawinglayer/source/primitive2d/metafileprimitive2d.cxx index 4a0719ea0ffb..d08a8ed89ffa 100644 --- a/drawinglayer/source/primitive2d/metafileprimitive2d.cxx +++ b/drawinglayer/source/primitive2d/metafileprimitive2d.cxx @@ -380,26 +380,8 @@ namespace if(!rRegion.IsEmpty()) { Region aRegion(rRegion); - aRetval = aRegion.GetB2DPolyPolygon(); - if(!aRetval.count()) - { - RegionHandle aRegionHandle(aRegion.BeginEnumRects()); - Rectangle aRegionRectangle; - - while(aRegion.GetEnumRects(aRegionHandle, aRegionRectangle)) - { - if(!aRegionRectangle.IsEmpty()) - { - const basegfx::B2DRange aRegionRange( - aRegionRectangle.Left(), aRegionRectangle.Top(), - aRegionRectangle.Right(), aRegionRectangle.Bottom()); - aRetval.append(basegfx::tools::createPolygonFromRect(aRegionRange)); - } - } - - aRegion.EndEnumRects(aRegionHandle); - } + aRetval = aRegion.GetAsB2DPolyPolygon(); } return aRetval; diff --git a/filter/source/graphicfilter/eps/eps.cxx b/filter/source/graphicfilter/eps/eps.cxx index 454d683393ed..12a18e69de14 100644 --- a/filter/source/graphicfilter/eps/eps.cxx +++ b/filter/source/graphicfilter/eps/eps.cxx @@ -1649,15 +1649,16 @@ void PSWriter::ImplSetClipRegion( Region& rClipRegion ) { if ( !rClipRegion.IsEmpty() ) { - Rectangle aRect; - RegionHandle hRegionHandle = rClipRegion.BeginEnumRects(); + RectangleVector aRectangles; + rClipRegion.GetRegionRectangles(aRectangles); - while ( rClipRegion.GetNextEnumRect( hRegionHandle, aRect ) ) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - double nX1 = aRect.Left(); - double nY1 = aRect.Top(); - double nX2 = aRect.Right(); - double nY2 = aRect.Bottom(); + double nX1(aRectIter->Left()); + double nY1(aRectIter->Top()); + double nX2(aRectIter->Right()); + double nY2(aRectIter->Bottom()); + ImplWriteDouble( nX1 ); ImplWriteDouble( nY1 ); ImplWriteByte( 'm' ); @@ -1673,8 +1674,34 @@ void PSWriter::ImplSetClipRegion( Region& rClipRegion ) ImplWriteDouble( nX1 ); ImplWriteDouble( nY1 ); ImplWriteByte( 'l', PS_SPACE | PS_WRAP ); - }; - rClipRegion.EndEnumRects( hRegionHandle ); + } + + //Rectangle aRect; + //RegionHandle hRegionHandle = rClipRegion.BeginEnumRects(); + // + //while ( rClipRegion.GetEnumRects( hRegionHandle, aRect ) ) + //{ + // double nX1 = aRect.Left(); + // double nY1 = aRect.Top(); + // double nX2 = aRect.Right(); + // double nY2 = aRect.Bottom(); + // ImplWriteDouble( nX1 ); + // ImplWriteDouble( nY1 ); + // ImplWriteByte( 'm' ); + // ImplWriteDouble( nX2 ); + // ImplWriteDouble( nY1 ); + // ImplWriteByte( 'l' ); + // ImplWriteDouble( nX2 ); + // ImplWriteDouble( nY2 ); + // ImplWriteByte( 'l' ); + // ImplWriteDouble( nX1 ); + // ImplWriteDouble( nY2 ); + // ImplWriteByte( 'l' ); + // ImplWriteDouble( nX1 ); + // ImplWriteDouble( nY1 ); + // ImplWriteByte( 'l', PS_SPACE | PS_WRAP ); + //}; + //rClipRegion.EndEnumRects( hRegionHandle ); ImplWriteLine( "eoclip newpath" ); } } @@ -1723,12 +1750,18 @@ void PSWriter::ImplBmp( Bitmap* pBitmap, Bitmap* pMaskBitmap, const Point & rPoi aRect = Rectangle( Point( 0, nHeightOrg - nHeightLeft ), Size( (long)nWidth, (long)nHeight ) ); aRegion = Region( pMaskBitmap->CreateRegion( COL_BLACK, aRect ) ); - if ( ( mnLevel == 1 ) && ( aRegion.GetRectCount() * 5 > 1000 ) ) + if( mnLevel == 1 ) { - nHeight >>= 1; - if ( nHeight < 2 ) - return; - continue; + RectangleVector aRectangleVector; + aRegion.GetRegionRectangles(aRectangleVector); + + if ( aRectangleVector.size() * 5 > 1000 ) + { + nHeight >>= 1; + if ( nHeight < 2 ) + return; + continue; + } } break; } @@ -1743,26 +1776,50 @@ void PSWriter::ImplBmp( Bitmap* pBitmap, Bitmap* pMaskBitmap, const Point & rPoi ImplWriteLine( "gs\npum" ); ImplTranslate( aSourcePos.X(), aSourcePos.Y() ); ImplScale( nXWidth / nWidth, nYHeight / nHeight ); - RegionHandle hRegionHandle = aRegion.BeginEnumRects(); - while ( aRegion.GetNextEnumRect( hRegionHandle, aRect ) ) + RectangleVector aRectangles; + aRegion.GetRegionRectangles(aRectangles); + const long nMoveVertical(nHeightLeft - nHeightOrg); + + for(RectangleVector::iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - aRect.Move( 0, - ( nHeightOrg - nHeightLeft ) ); - ImplWriteLong( aRect.Left() ); - ImplWriteLong( aRect.Top() ); + aRectIter->Move(0, nMoveVertical); + + ImplWriteLong( aRectIter->Left() ); + ImplWriteLong( aRectIter->Top() ); ImplWriteByte( 'm' ); - ImplWriteLong( aRect.Right() + 1 ); - ImplWriteLong( aRect.Top() ); + ImplWriteLong( aRectIter->Right() + 1 ); + ImplWriteLong( aRectIter->Top() ); ImplWriteByte( 'l' ); - ImplWriteLong( aRect.Right() + 1 ); - ImplWriteLong( aRect.Bottom() + 1 ); + ImplWriteLong( aRectIter->Right() + 1 ); + ImplWriteLong( aRectIter->Bottom() + 1 ); ImplWriteByte( 'l' ); - ImplWriteLong( aRect.Left() ); - ImplWriteLong( aRect.Bottom() + 1 ); + ImplWriteLong( aRectIter->Left() ); + ImplWriteLong( aRectIter->Bottom() + 1 ); ImplWriteByte( 'l' ); ImplWriteByte( 'p', PS_SPACE | PS_WRAP ); - }; - aRegion.EndEnumRects( hRegionHandle ); + } + + //RegionHandle hRegionHandle = aRegion.BeginEnumRects(); + // + //while ( aRegion.GetEnumRects( hRegionHandle, aRect ) ) + //{ + // aRect.Move( 0, - ( nHeightOrg - nHeightLeft ) ); + // ImplWriteLong( aRect.Left() ); + // ImplWriteLong( aRect.Top() ); + // ImplWriteByte( 'm' ); + // ImplWriteLong( aRect.Right() + 1 ); + // ImplWriteLong( aRect.Top() ); + // ImplWriteByte( 'l' ); + // ImplWriteLong( aRect.Right() + 1 ); + // ImplWriteLong( aRect.Bottom() + 1 ); + // ImplWriteByte( 'l' ); + // ImplWriteLong( aRect.Left() ); + // ImplWriteLong( aRect.Bottom() + 1 ); + // ImplWriteByte( 'l' ); + // ImplWriteByte( 'p', PS_SPACE | PS_WRAP ); + //}; + //aRegion.EndEnumRects( hRegionHandle ); ImplWriteLine( "eoclip newpath" ); ImplWriteLine( "pom" ); } diff --git a/include/vcl/bmpacc.hxx b/include/vcl/bmpacc.hxx index 306f28a5b0b2..f3f5301c2567 100644 --- a/include/vcl/bmpacc.hxx +++ b/include/vcl/bmpacc.hxx @@ -195,6 +195,12 @@ public: void FillRect( const Rectangle& rRect ); void DrawRect( const Rectangle& rRect ); + void FillPolygon( const Polygon& rPoly ); + void DrawPolygon( const Polygon& rPoly ); + + void FillPolyPolygon( const PolyPolygon& rPoly ); + void DrawPolyPolygon( const PolyPolygon& rPolyPoly ); + private: BitmapColor* mpLineColor; diff --git a/include/vcl/outdev.hxx b/include/vcl/outdev.hxx index b3a2f9ef87e1..b76598f08452 100644 --- a/include/vcl/outdev.hxx +++ b/include/vcl/outdev.hxx @@ -965,6 +965,7 @@ public: Rectangle LogicToPixel( const Rectangle& rLogicRect ) const; Polygon LogicToPixel( const Polygon& rLogicPoly ) const; PolyPolygon LogicToPixel( const PolyPolygon& rLogicPolyPoly ) const; + basegfx::B2DPolyPolygon LogicToPixel( const basegfx::B2DPolyPolygon& rLogicPolyPoly ) const; Region LogicToPixel( const Region& rLogicRegion )const; Point LogicToPixel( const Point& rLogicPt, const MapMode& rMapMode ) const; @@ -974,12 +975,19 @@ public: const MapMode& rMapMode ) const; Polygon LogicToPixel( const Polygon& rLogicPoly, const MapMode& rMapMode ) const; + basegfx::B2DPolygon LogicToPixel( const basegfx::B2DPolygon& rLogicPoly, + const MapMode& rMapMode ) const; + PolyPolygon LogicToPixel( const PolyPolygon& rLogicPolyPoly, + const MapMode& rMapMode ) const; basegfx::B2DPolyPolygon LogicToPixel( const basegfx::B2DPolyPolygon& rLogicPolyPoly, const MapMode& rMapMode ) const; + Region LogicToPixel( const Region& rLogicRegion, + const MapMode& rMapMode ) const; Point PixelToLogic( const Point& rDevicePt ) const; Size PixelToLogic( const Size& rDeviceSize ) const; Rectangle PixelToLogic( const Rectangle& rDeviceRect ) const; Polygon PixelToLogic( const Polygon& rDevicePoly ) const; + basegfx::B2DPolygon LogicToPixel( const basegfx::B2DPolygon& rLogicPoly ) const; PolyPolygon PixelToLogic( const PolyPolygon& rDevicePolyPoly ) const; basegfx::B2DPolyPolygon PixelToLogic( const basegfx::B2DPolyPolygon& rDevicePolyPoly ) const; Region PixelToLogic( const Region& rDeviceRegion ) const; @@ -993,8 +1001,12 @@ public: const MapMode& rMapMode ) const; basegfx::B2DPolygon PixelToLogic( const basegfx::B2DPolygon& rDevicePoly, const MapMode& rMapMode ) const; + PolyPolygon PixelToLogic( const PolyPolygon& rDevicePolyPoly, + const MapMode& rMapMode ) const; basegfx::B2DPolyPolygon PixelToLogic( const basegfx::B2DPolyPolygon& rDevicePolyPoly, const MapMode& rMapMode ) const; + Region PixelToLogic( const Region& rDeviceRegion, + const MapMode& rMapMode ) const; Point LogicToLogic( const Point& rPtSource, const MapMode* pMapModeSource, const MapMode* pMapModeDest ) const; diff --git a/include/vcl/regband.hxx b/include/vcl/regband.hxx index 0b60a801d83e..24fe7584a7b7 100644 --- a/include/vcl/regband.hxx +++ b/include/vcl/regband.hxx @@ -49,7 +49,7 @@ struct ImplRegionBandSep ImplRegionBandSep* mpNextSep; long mnXLeft; long mnXRight; - sal_Bool mbRemoved; + bool mbRemoved; }; enum LineType { LINE_ASCENDING, LINE_DESCENDING, LINE_HORIZONTAL }; @@ -60,7 +60,7 @@ struct ImplRegionBandPoint ImplRegionBandPoint* mpNextBandPoint; long mnX; long mnLineId; - sal_Bool mbEndPoint; + bool mbEndPoint; LineType meLineType; }; @@ -77,7 +77,9 @@ public: ImplRegionBandPoint* mpFirstBandPoint; // root of the list with lines long mnYTop; // actual boundary of the band long mnYBottom; - sal_Bool mbTouched; + + // bitfield + bool mbTouched : 1; // create y-band with boundaries ImplRegionBand( long nYTop, long nYBottom ); @@ -102,14 +104,14 @@ public: long GetXRightBoundary() const; // combine overlapping bands - sal_Bool OptimizeBand(); + bool OptimizeBand(); // generate separations from lines and process // union with existing separations void ProcessPoints(); // insert point in the list for later processing - sal_Bool InsertPoint( long nX, long nLineID, - sal_Bool bEndPoint, LineType eLineType ); + bool InsertPoint( long nX, long nLineID, + bool bEndPoint, LineType eLineType ); void Union( long nXLeft, long nXRight ); void Intersect( long nXLeft, long nXRight ); @@ -119,11 +121,14 @@ public: void MoveX( long nHorzMove ); void ScaleX( double fHorzScale ); - sal_Bool IsInside( long nX ); + bool IsInside( long nX ); + bool IsInside( long nLeft, long nRight ); + bool IsOver( long nLeft, long nRight ); + - sal_Bool IsEmpty() const { return ((!mpFirstSep) && (!mpFirstBandPoint)); } + bool IsEmpty() const { return ((!mpFirstSep) && (!mpFirstBandPoint)); } - sal_Bool operator==( const ImplRegionBand& rRegionBand ) const; + bool operator==( const ImplRegionBand& rRegionBand ) const; /** Split the called band at the given vertical coordinate. After the split the called band will cover the upper part not including nY. diff --git a/include/vcl/region.hxx b/include/vcl/region.hxx index 88c55ea5b5fc..f2298e7a5d2a 100644 --- a/include/vcl/region.hxx +++ b/include/vcl/region.hxx @@ -25,115 +25,96 @@ #include <vcl/dllapi.h> #include <basegfx/polygon/b2dpolypolygon.hxx> +#include <boost/shared_ptr.hpp> -class ImplRegion; class ImplRegionBand; +class RegionBand; class Polygon; class PolyPolygon; -struct ImplRegionInfo; -// -------------- -// - RegionType - -// -------------- +////////////////////////////////////////////////////////////////////////////// -enum RegionType { REGION_NULL, REGION_EMPTY, REGION_RECTANGLE, REGION_COMPLEX }; -enum RegionOverlapType { REGION_INSIDE, REGION_OVER, REGION_OUTSIDE }; +typedef boost::shared_ptr< RegionBand > RegionBandPtr; +typedef boost::shared_ptr< PolyPolygon > PolyPolygonPtr; +typedef boost::shared_ptr< basegfx::B2DPolyPolygon > B2DPolyPolygonPtr; +typedef std::vector< Rectangle > RectangleVector; -typedef sal_IntPtr RegionHandle; - -// ---------- -// - Region - -// ---------- +////////////////////////////////////////////////////////////////////////////// class VCL_DLLPUBLIC Region { +private: friend class OutputDevice; friend class Window; friend class Bitmap; -private: - ImplRegion* mpImplRegion; - - SAL_DLLPRIVATE void ImplCopyData(); - SAL_DLLPRIVATE void ImplCreateRectRegion( const Rectangle& rRect ); - SAL_DLLPRIVATE void ImplCreatePolyPolyRegion( const PolyPolygon& rPolyPoly ); - SAL_DLLPRIVATE void ImplCreatePolyPolyRegion( const basegfx::B2DPolyPolygon& rPolyPoly ); - SAL_DLLPRIVATE void ImplPolyPolyRegionToBandRegionFunc(); - SAL_DLLPRIVATE inline void ImplPolyPolyRegionToBandRegion(); - SAL_DLLPRIVATE const ImplRegion* ImplGetImplRegion() const { return mpImplRegion; } - SAL_DLLPRIVATE ImplRegion* ImplGetImplRegion() { return mpImplRegion; } - SAL_DLLPRIVATE void ImplBeginAddRect( ); - SAL_DLLPRIVATE sal_Bool ImplAddRect( const Rectangle& rRect ); - SAL_DLLPRIVATE void ImplEndAddRect( ); - SAL_DLLPRIVATE void ImplIntersectWithPolyPolygon( const Region& ); - SAL_DLLPRIVATE void ImplExcludePolyPolygon( const Region& ); - SAL_DLLPRIVATE void ImplUnionPolyPolygon( const Region& ); - SAL_DLLPRIVATE void ImplXOrPolyPolygon( const Region& ); - -public: // public within vcl - VCL_PLUGIN_PUBLIC bool ImplGetFirstRect( ImplRegionInfo& rImplRegionInfo, - long& nX, long& nY, long& nWidth, long& nHeight ) const; - VCL_PLUGIN_PUBLIC bool ImplGetNextRect( ImplRegionInfo& rImplRegionInfo, - long& nX, long& nY, long& nWidth, long& nHeight ) const; -#ifdef DBG_UTIL - friend const char* ImplDbgTestRegion( const void* pObj ); -#endif + // possible contents + B2DPolyPolygonPtr mpB2DPolyPolygon; + PolyPolygonPtr mpPolyPolygon; + RegionBandPtr mpRegionBand; + + /// bitfield + bool mbIsNull : 1; + + // helpers + SAL_DLLPRIVATE void ImplCreatePolyPolyRegion( const PolyPolygon& rPolyPoly ); + SAL_DLLPRIVATE void ImplCreatePolyPolyRegion( const basegfx::B2DPolyPolygon& rPolyPoly ); + + SAL_DLLPRIVATE PolyPolygon ImplCreatePolyPolygonFromRegionBand() const; + SAL_DLLPRIVATE basegfx::B2DPolyPolygon ImplCreateB2DPolyPolygonFromRegionBand() const; public: - explicit Region(); - explicit Region( RegionType eType ); - explicit Region( const Rectangle& rRect ); - explicit Region( const Polygon& rPolygon ); - explicit Region( const PolyPolygon& rPolyPoly ); - explicit Region( const basegfx::B2DPolyPolygon& ); - Region( const Region& rRegion ); - ~Region(); - - void Move( long nHorzMove, long nVertMove ); - void Scale( double fScaleX, double fScaleY ); - void Union( const Rectangle& rRegion ); - void Intersect( const Rectangle& rRegion ); - void Exclude( const Rectangle& rRegion ); - void XOr( const Rectangle& rRegion ); - void Union( const Region& rRegion ); - void Intersect( const Region& rRegion ); - void Exclude( const Region& rRegion ); - void XOr( const Region& rRegion ); - - RegionType GetType() const; - sal_Bool IsEmpty() const { return GetType() == REGION_EMPTY; }; - sal_Bool IsNull() const { return GetType() == REGION_NULL; }; - - void SetEmpty(); - void SetNull(); - - Rectangle GetBoundRect() const; - - sal_Bool HasPolyPolygon() const; - PolyPolygon GetPolyPolygon() const; - // returns an empty polypolygon in case HasPolyPolygon is sal_False - const basegfx::B2DPolyPolygon GetB2DPolyPolygon() const; - // returns a PolyPolygon either copied from the set PolyPolygon region - // or created from the constituent rectangles - basegfx::B2DPolyPolygon ConvertToB2DPolyPolygon(); - - sal_uLong GetRectCount() const; - RegionHandle BeginEnumRects(); - sal_Bool GetEnumRects( RegionHandle hRegionHandle, Rectangle& rRect ); - sal_Bool GetNextEnumRect( RegionHandle hRegionHandle, Rectangle& rRect ) - { return GetEnumRects( hRegionHandle, rRect ); } - void EndEnumRects( RegionHandle hRegionHandle ); - - sal_Bool IsInside( const Point& rPoint ) const; - sal_Bool IsInside( const Rectangle& rRect ) const; - sal_Bool IsOver( const Rectangle& rRect ) const; - - Region& operator=( const Region& rRegion ); - Region& operator=( const Rectangle& rRect ); - - sal_Bool operator==( const Region& rRegion ) const; - sal_Bool operator!=( const Region& rRegion ) const - { return !(Region::operator==( rRegion )); } + + explicit Region(bool bIsNull = false); // default creates empty region, with true a null region is created + explicit Region(const Rectangle& rRect); + explicit Region(const Polygon& rPolygon); + explicit Region(const PolyPolygon& rPolyPoly); + explicit Region(const basegfx::B2DPolyPolygon&); + Region(const Region& rRegion); + ~Region(); + + // direct access to contents + const basegfx::B2DPolyPolygon* getB2DPolyPolygon() const { return mpB2DPolyPolygon.get(); } + const PolyPolygon* getPolyPolygon() const { return mpPolyPolygon.get(); } + const RegionBand* getRegionBand() const { return mpRegionBand.get(); } + + // access with converters, the asked data will be created from the most + // valuable data, buffered and returned + const PolyPolygon GetAsPolyPolygon() const; + const basegfx::B2DPolyPolygon GetAsB2DPolyPolygon() const; + const RegionBand* GetAsRegionBand() const; + + // manipulators + void Move( long nHorzMove, long nVertMove ); + void Scale( double fScaleX, double fScaleY ); + bool Union( const Rectangle& rRegion ); + bool Intersect( const Rectangle& rRegion ); + bool Exclude( const Rectangle& rRegion ); + bool XOr( const Rectangle& rRegion ); + bool Union( const Region& rRegion ); + bool Intersect( const Region& rRegion ); + bool Exclude( const Region& rRegion ); + bool XOr( const Region& rRegion ); + + bool IsEmpty() const; + bool IsNull() const; + + void SetEmpty(); + void SetNull(); + + Rectangle GetBoundRect() const; + bool HasPolyPolygonOrB2DPolyPolygon() const { return (getB2DPolyPolygon() || getPolyPolygon()); } + void GetRegionRectangles(RectangleVector& rTarget) const; + + bool IsInside( const Point& rPoint ) const; + bool IsInside( const Rectangle& rRect ) const; + bool IsOver( const Rectangle& rRect ) const; + + Region& operator=( const Region& rRegion ); + Region& operator=( const Rectangle& rRect ); + + bool operator==( const Region& rRegion ) const; + bool operator!=( const Region& rRegion ) const { return !(Region::operator==( rRegion )); } friend VCL_DLLPUBLIC SvStream& operator>>( SvStream& rIStm, Region& rRegion ); friend VCL_DLLPUBLIC SvStream& operator<<( SvStream& rOStm, const Region& rRegion ); diff --git a/sd/source/ui/slidesorter/view/SlsLayeredDevice.cxx b/sd/source/ui/slidesorter/view/SlsLayeredDevice.cxx index 601525951231..3575c24e5656 100644 --- a/sd/source/ui/slidesorter/view/SlsLayeredDevice.cxx +++ b/sd/source/ui/slidesorter/view/SlsLayeredDevice.cxx @@ -79,19 +79,26 @@ void DeviceCopy ( void ForAllRectangles (const Region& rRegion, ::boost::function<void(const Rectangle&)> aFunction) { OSL_ASSERT(aFunction); + RectangleVector aRectangles; + rRegion.GetRegionRectangles(aRectangles); - if (rRegion.GetRectCount() <= 1) + if(0 == aRectangles.size()) { - aFunction(rRegion.GetBoundRect()); + aFunction(Rectangle()); } else { - Region aMutableRegionCopy (rRegion); - RegionHandle aHandle(aMutableRegionCopy.BeginEnumRects()); - Rectangle aBox; - while (aMutableRegionCopy.GetNextEnumRect(aHandle, aBox)) - aFunction(aBox); - aMutableRegionCopy.EndEnumRects(aHandle); + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) + { + aFunction(*aRectIter); + } + + //Region aMutableRegionCopy (rRegion); + //RegionHandle aHandle(aMutableRegionCopy.BeginEnumRects()); + //Rectangle aBox; + //while (aMutableRegionCopy.GetEnumRects(aHandle, aBox)) + // aFunction(aBox); + //aMutableRegionCopy.EndEnumRects(aHandle); } } diff --git a/svx/source/sdr/overlay/overlaymanagerbuffered.cxx b/svx/source/sdr/overlay/overlaymanagerbuffered.cxx index 81b53b27bb0e..47d76f835d00 100644 --- a/svx/source/sdr/overlay/overlaymanagerbuffered.cxx +++ b/svx/source/sdr/overlay/overlaymanagerbuffered.cxx @@ -109,33 +109,33 @@ namespace sdr void OverlayManagerBuffered::ImpRestoreBackground(const Region& rRegionPixel) const { - // local region - Region aRegionPixel(rRegionPixel); - RegionHandle aRegionHandle(aRegionPixel.BeginEnumRects()); - Rectangle aRegionRectanglePixel; - // MapModes off const bool bMapModeWasEnabledDest(getOutputDevice().IsMapModeEnabled()); const bool bMapModeWasEnabledSource(maBufferDevice.IsMapModeEnabled()); getOutputDevice().EnableMapMode(false); ((OverlayManagerBuffered*)this)->maBufferDevice.EnableMapMode(false); - while(aRegionPixel.GetEnumRects(aRegionHandle, aRegionRectanglePixel)) + // local region + RectangleVector aRectangles; + rRegionPixel.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { #ifdef DBG_UTIL // #i72754# possible graphical region test only with non-pro static bool bDoPaintForVisualControl(false); + if(bDoPaintForVisualControl) { getOutputDevice().SetLineColor(COL_LIGHTGREEN); getOutputDevice().SetFillColor(); - getOutputDevice().DrawRect(aRegionRectanglePixel); + getOutputDevice().DrawRect(*aRectIter); } #endif // restore the area - const Point aTopLeft(aRegionRectanglePixel.TopLeft()); - const Size aSize(aRegionRectanglePixel.GetSize()); + const Point aTopLeft(aRectIter->TopLeft()); + const Size aSize(aRectIter->GetSize()); getOutputDevice().DrawOutDev( aTopLeft, aSize, // destination @@ -143,7 +143,33 @@ namespace sdr maBufferDevice); } - aRegionPixel.EndEnumRects(aRegionHandle); + //Region aRegionPixel(rRegionPixel); + //RegionHandle aRegionHandle(aRegionPixel.BeginEnumRects()); + //Rectangle aRegionRectanglePixel; + // + //while(aRegionPixel.GetEnumRects(aRegionHandle, aRegionRectanglePixel)) + //{ +#ifdef DBG_U//TIL + // // #i72754# possible graphical region test only with non-pro + // static bool bDoPaintForVisualControl(false); + // if(bDoPaintForVisualControl) + // { + // getOutputDevice().SetLineColor(COL_LIGHTGREEN); + // getOutputDevice().SetFillColor(); + // getOutputDevice().DrawRect(aRegionRectanglePixel); + // } +#endif // + // // restore the area + // const Point aTopLeft(aRegionRectanglePixel.TopLeft()); + // const Size aSize(aRegionRectanglePixel.GetSize()); + // + // getOutputDevice().DrawOutDev( + // aTopLeft, aSize, // destination + // aTopLeft, aSize, // source + // maBufferDevice); + //} + // + //aRegionPixel.EndEnumRects(aRegionHandle); // restore MapModes getOutputDevice().EnableMapMode(bMapModeWasEnabledDest); @@ -175,24 +201,24 @@ namespace sdr } // also limit to buffer size - const Rectangle aBufferDeviceRectanglePixel = Rectangle(Point(), maBufferDevice.GetOutputSizePixel()); + const Rectangle aBufferDeviceRectanglePixel(Point(), maBufferDevice.GetOutputSizePixel()); aRegion.Intersect(aBufferDeviceRectanglePixel); - // prepare to iterate over the rectangles from the region in pixels - RegionHandle aRegionHandle(aRegion.BeginEnumRects()); - Rectangle aRegionRectanglePixel; - // MapModes off const bool bMapModeWasEnabledDest(rSource.IsMapModeEnabled()); const bool bMapModeWasEnabledSource(maBufferDevice.IsMapModeEnabled()); rSource.EnableMapMode(false); maBufferDevice.EnableMapMode(false); - while(aRegion.GetEnumRects(aRegionHandle, aRegionRectanglePixel)) + // prepare to iterate over the rectangles from the region in pixels + RectangleVector aRectangles; + aRegion.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { // for each rectangle, save the area - Point aTopLeft(aRegionRectanglePixel.TopLeft()); - Size aSize(aRegionRectanglePixel.GetSize()); + const Point aTopLeft(aRectIter->TopLeft()); + const Size aSize(aRectIter->GetSize()); maBufferDevice.DrawOutDev( aTopLeft, aSize, // destination @@ -200,7 +226,23 @@ namespace sdr rSource); } - aRegion.EndEnumRects(aRegionHandle); + //RegionHandle aRegionHandle(aRegion.BeginEnumRects()); + //Rectangle aRegionRectanglePixel; + // + //while(aRegion.GetEnumRects(aRegionHandle, aRegionRectanglePixel)) + //{ + // // for each rectangle, save the area + // Point aTopLeft(aRegionRectanglePixel.TopLeft()); + // Size aSize(aRegionRectanglePixel.GetSize()); + // + // maBufferDevice.DrawOutDev( + // aTopLeft, aSize, // destination + // aTopLeft, aSize, // source + // rSource); + // + //} + // + //aRegion.EndEnumRects(aRegionHandle); // restore MapModes rSource.EnableMapMode(bMapModeWasEnabledDest); diff --git a/svx/source/sdr/properties/itemsettools.cxx b/svx/source/sdr/properties/itemsettools.cxx index b5f763ab6245..cea89d0e6248 100644 --- a/svx/source/sdr/properties/itemsettools.cxx +++ b/svx/source/sdr/properties/itemsettools.cxx @@ -21,10 +21,9 @@ #include <tools/debug.hxx> #include <svl/itemset.hxx> #include <svl/whiter.hxx> - -#include <vector> #include <svx/svdogrp.hxx> #include <svx/svditer.hxx> +#include <vcl/region.hxx> ////////////////////////////////////////////////////////////////////////////// // class to remember broadcast start positions @@ -33,9 +32,6 @@ namespace sdr { namespace properties { - // helper vector to remember rectangles - typedef ::std::vector< Rectangle > RectangleVector; - ItemChangeBroadcaster::ItemChangeBroadcaster(const SdrObject& rObj) { if(rObj.ISA(SdrObjGroup)) diff --git a/svx/source/svdraw/sdrpaintwindow.cxx b/svx/source/svdraw/sdrpaintwindow.cxx index 3df44a1bac2b..9651ae710cae 100644 --- a/svx/source/svdraw/sdrpaintwindow.cxx +++ b/svx/source/svdraw/sdrpaintwindow.cxx @@ -56,9 +56,9 @@ void SdrPreRenderDevice::PreparePreRenderDevice() void SdrPreRenderDevice::OutputPreRenderDevice(const Region& rExpandedRegion) { // region to pixels - Region aRegionPixel(mrOutputDevice.LogicToPixel(rExpandedRegion)); - RegionHandle aRegionHandle(aRegionPixel.BeginEnumRects()); - Rectangle aRegionRectanglePixel; + const Region aRegionPixel(mrOutputDevice.LogicToPixel(rExpandedRegion)); + //RegionHandle aRegionHandle(aRegionPixel.BeginEnumRects()); + //Rectangle aRegionRectanglePixel; // MapModes off sal_Bool bMapModeWasEnabledDest(mrOutputDevice.IsMapModeEnabled()); @@ -66,11 +66,14 @@ void SdrPreRenderDevice::OutputPreRenderDevice(const Region& rExpandedRegion) mrOutputDevice.EnableMapMode(sal_False); maPreRenderDevice.EnableMapMode(sal_False); - while(aRegionPixel.GetEnumRects(aRegionHandle, aRegionRectanglePixel)) + RectangleVector aRectangles; + aRegionPixel.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { // for each rectangle, copy the area - const Point aTopLeft(aRegionRectanglePixel.TopLeft()); - const Size aSize(aRegionRectanglePixel.GetSize()); + const Point aTopLeft(aRectIter->TopLeft()); + const Size aSize(aRectIter->GetSize()); mrOutputDevice.DrawOutDev( aTopLeft, aSize, @@ -80,17 +83,43 @@ void SdrPreRenderDevice::OutputPreRenderDevice(const Region& rExpandedRegion) #ifdef DBG_UTIL // #i74769# static bool bDoPaintForVisualControlRegion(false); + if(bDoPaintForVisualControlRegion) { - Color aColor((((((rand()&0x7f)|0x80)<<8L)|((rand()&0x7f)|0x80))<<8L)|((rand()&0x7f)|0x80)); + const Color aColor((((((rand()&0x7f)|0x80)<<8L)|((rand()&0x7f)|0x80))<<8L)|((rand()&0x7f)|0x80)); + mrOutputDevice.SetLineColor(aColor); mrOutputDevice.SetFillColor(); - mrOutputDevice.DrawRect(aRegionRectanglePixel); + mrOutputDevice.DrawRect(*aRectIter); } #endif } - aRegionPixel.EndEnumRects(aRegionHandle); +// while(aRegionPixel.GetEnumRects(aRegionHandle, aRegionRectanglePixel)) +// { +// // for each rectangle, copy the area +// const Point aTopLeft(aRegionRectanglePixel.TopLeft()); +// const Size aSize(aRegionRectanglePixel.GetSize()); +// +// mrOutputDevice.DrawOutDev( +// aTopLeft, aSize, +// aTopLeft, aSize, +// maPreRenderDevice); +// +//#ifdef DBG_UTIL +// // #i74769# +// static bool bDoPaintForVisualControlRegion(false); +// if(bDoPaintForVisualControlRegion) +// { +// Color aColor((((((rand()&0x7f)|0x80)<<8L)|((rand()&0x7f)|0x80))<<8L)|((rand()&0x7f)|0x80)); +// mrOutputDevice.SetLineColor(aColor); +// mrOutputDevice.SetFillColor(); +// mrOutputDevice.DrawRect(aRegionRectanglePixel); +// } +//#endif +// } +// +// aRegionPixel.EndEnumRects(aRegionHandle); mrOutputDevice.EnableMapMode(bMapModeWasEnabledDest); maPreRenderDevice.EnableMapMode(bMapModeWasEnabledSource); diff --git a/svx/source/svdraw/svdfmtf.cxx b/svx/source/svdraw/svdfmtf.cxx index 0789a39f3af5..f6c485859713 100644 --- a/svx/source/svdraw/svdfmtf.cxx +++ b/svx/source/svdraw/svdfmtf.cxx @@ -800,9 +800,7 @@ void ImpSdrGDIMetaFileImport::checkClip() { if(maVD.IsClipRegion()) { - Region aRegion(maVD.GetClipRegion()); - - maClip = aRegion.ConvertToB2DPolyPolygon(); + maClip = maVD.GetClipRegion().GetAsB2DPolyPolygon(); if(isClip()) { diff --git a/svx/source/svdraw/svdpntv.cxx b/svx/source/svdraw/svdpntv.cxx index db7feab72467..222b05c314c5 100644 --- a/svx/source/svdraw/svdpntv.cxx +++ b/svx/source/svdraw/svdpntv.cxx @@ -617,19 +617,31 @@ void SdrPaintView::CompleteRedraw(OutputDevice* pOut, const Region& rReg, sdr::c #ifdef DBG_UTIL // #i74769# test-paint repaint region static bool bDoPaintForVisualControl(false); + if(bDoPaintForVisualControl) { - RegionHandle aRegionHandle(aOptimizedRepaintRegion.BeginEnumRects()); - Rectangle aRegionRectangle; + RectangleVector aRectangles; + aOptimizedRepaintRegion.GetRegionRectangles(aRectangles); + + pWindow->SetLineColor(COL_LIGHTGREEN); + pWindow->SetFillColor(); - while(aOptimizedRepaintRegion.GetEnumRects(aRegionHandle, aRegionRectangle)) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - pWindow->SetLineColor(COL_LIGHTGREEN); - pWindow->SetFillColor(); - pWindow->DrawRect(aRegionRectangle); + pWindow->DrawRect(*aRectIter); } - aOptimizedRepaintRegion.EndEnumRects(aRegionHandle); + //RegionHandle aRegionHandle(aOptimizedRepaintRegion.BeginEnumRects()); + //Rectangle aRegionRectangle; + // + //while(aOptimizedRepaintRegion.GetEnumRects(aRegionHandle, aRegionRectangle)) + //{ + // pWindow->SetLineColor(COL_LIGHTGREEN); + // pWindow->SetFillColor(); + // pWindow->DrawRect(aRegionRectangle); + //} + // + //aOptimizedRepaintRegion.EndEnumRects(aRegionHandle); } #endif } @@ -840,25 +852,6 @@ Region SdrPaintView::OptimizeDrawLayersRegion(OutputDevice* pOut, const Region& if(!pWindow->GetPaintRegion().IsEmpty()) { aOptimizedRepaintRegion.Intersect(pWindow->GetPaintRegion()); - -#ifdef DBG_UTIL - // #i74769# test-paint repaint region - static bool bDoPaintForVisualControl(false); - if(bDoPaintForVisualControl) - { - RegionHandle aRegionHandle(aOptimizedRepaintRegion.BeginEnumRects()); - Rectangle aRegionRectangle; - - while(aOptimizedRepaintRegion.GetEnumRects(aRegionHandle, aRegionRectangle)) - { - pWindow->SetLineColor(COL_LIGHTGREEN); - pWindow->SetFillColor(); - pWindow->DrawRect(aRegionRectangle); - } - - aOptimizedRepaintRegion.EndEnumRects(aRegionHandle); - } -#endif } } } diff --git a/sw/source/core/view/viewsh.cxx b/sw/source/core/view/viewsh.cxx index b1ac5c30314c..6dabfbcd3319 100644 --- a/sw/source/core/view/viewsh.cxx +++ b/sw/source/core/view/viewsh.cxx @@ -1729,12 +1729,20 @@ void ViewShell::Paint(const Rectangle &rRect) if ( mbInEndAction && GetWin() ) { - Region aRegion( GetWin()->GetPaintRegion() ); - RegionHandle hHdl( aRegion.BeginEnumRects() ); - Rectangle aRect; - while ( aRegion.GetNextEnumRect( hHdl, aRect ) ) - Imp()->AddPaintRect( aRect ); - aRegion.EndEnumRects( hHdl ); + const Region aRegion(GetWin()->GetPaintRegion()); + RectangleVector aRectangles; + aRegion.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) + { + Imp()->AddPaintRect(*aRectIter); + } + + //RegionHandle hHdl( aRegion.BeginEnumRects() ); + //Rectangle aRect; + //while ( aRegion.GetEnumRects( hHdl, aRect ) ) + // Imp()->AddPaintRect( aRect ); + //aRegion.EndEnumRects( hHdl ); } else if ( SfxProgress::GetActiveProgress( GetDoc()->GetDocShell() ) && GetOut() == GetWin() ) diff --git a/toolkit/source/awt/vclxregion.cxx b/toolkit/source/awt/vclxregion.cxx index 7ed03bfc08a2..2375befe403a 100644 --- a/toolkit/source/awt/vclxregion.cxx +++ b/toolkit/source/awt/vclxregion.cxx @@ -141,15 +141,24 @@ void VCLXRegion::xOrRegion( const ::com::sun::star::uno::Reference< ::com::sun:: { ::osl::Guard< ::osl::Mutex > aGuard( GetMutex() ); - sal_uLong nRects = maRegion.GetRectCount(); - ::com::sun::star::uno::Sequence< ::com::sun::star::awt::Rectangle > aRects( nRects ); - - Rectangle aRect; - sal_uInt32 nR = 0; - RegionHandle h = maRegion.BeginEnumRects(); - while ( maRegion.GetNextEnumRect( h, aRect ) ) - aRects.getArray()[nR++] = AWTRectangle( aRect ); - maRegion.EndEnumRects( h ); + RectangleVector aRectangles; + maRegion.GetRegionRectangles(aRectangles); + +// sal_uLong nRects = maRegion.GetRectCount(); + ::com::sun::star::uno::Sequence< ::com::sun::star::awt::Rectangle > aRects(aRectangles.size()); + sal_uInt32 a(0); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) + { + aRects.getArray()[a++] = AWTRectangle(*aRectIter); + } + + //Rectangle aRect; + //sal_uInt32 nR = 0; + //RegionHandle h = maRegion.BeginEnumRects(); + //while ( maRegion.GetEnumRects( h, aRect ) ) + // aRects.getArray()[nR++] = AWTRectangle( aRect ); + //maRegion.EndEnumRects( h ); return aRects; } diff --git a/vcl/Library_vcl.mk b/vcl/Library_vcl.mk index 8daa5bb87333..d9b19fa332c0 100644 --- a/vcl/Library_vcl.mk +++ b/vcl/Library_vcl.mk @@ -246,6 +246,7 @@ $(eval $(call gb_Library_add_exception_objects,vcl,\ vcl/source/gdi/print \ vcl/source/gdi/regband \ vcl/source/gdi/region \ + vcl/source/gdi/regionband \ vcl/source/gdi/salgdilayout \ vcl/source/gdi/sallayout \ vcl/source/gdi/salmisc \ diff --git a/vcl/aqua/source/gdi/salgdicommon.cxx b/vcl/aqua/source/gdi/salgdicommon.cxx index afedd3b9a339..ad484e28d84f 100644 --- a/vcl/aqua/source/gdi/salgdicommon.cxx +++ b/vcl/aqua/source/gdi/salgdicommon.cxx @@ -27,7 +27,6 @@ #include "aqua/salgdi.h" #include "fontsubset.hxx" -#include "region.h" #include "sft.hxx" using namespace vcl; @@ -1444,24 +1443,31 @@ bool AquaSalGraphics::setClipRegion( const Region& i_rClip ) mxClipPath = CGPathCreateMutable(); // set current path, either as polypolgon or sequence of rectangles - if( i_rClip.HasPolyPolygon() ) + if(i_rClip.HasPolyPolygonOrB2DPolyPolygon()) { - basegfx::B2DPolyPolygon aClip( const_cast<Region&>(i_rClip).ConvertToB2DPolyPolygon() ); + const basegfx::B2DPolyPolygon aClip(i_rClip.GetAsB2DPolyPolygon()); + AddPolyPolygonToPath( mxClipPath, aClip, !getAntiAliasB2DDraw(), false ); } else { - long nX, nY, nW, nH; - ImplRegionInfo aInfo; - bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); - while( bRegionRect ) + RectangleVector aRectangles; + i_rClip.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - if( nW && nH ) + const long nW(aRectIter->Right() - aRectIter->Left() + 1); // uses +1 logic in original + + if(nW) { - CGRect aRect = { { static_cast<CGFloat>(nX), static_cast<CGFloat>(nY) }, { static_cast<CGFloat>(nW), static_cast<CGFloat>(nH) } }; - CGPathAddRect( mxClipPath, NULL, aRect ); + const long nH(aRectIter->Bottom() - aRectIter->Top() + 1); // uses +1 logic in original + + if(nH) + { + CGRect aRect = {{ aRectIter->Left(), aRectIter->Top() }, { nW, nH }}; + CGPathAddRect( mxClipPath, NULL, aRect ); + } } - bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); } } // set the current path as clip region diff --git a/vcl/generic/fontmanager/fontsubst.cxx b/vcl/generic/fontmanager/fontsubst.cxx index 22b3d837863b..35f8294900be 100644 --- a/vcl/generic/fontmanager/fontsubst.cxx +++ b/vcl/generic/fontmanager/fontsubst.cxx @@ -30,7 +30,6 @@ #include "outdev.h" #include "fontsubset.hxx" #include "salprn.hxx" -#include "region.h" #include <unotools/fontdefs.hxx> #include <list> diff --git a/vcl/generic/print/genpspgraphics.cxx b/vcl/generic/print/genpspgraphics.cxx index 10af66051ddb..b9a1164b6aa8 100644 --- a/vcl/generic/print/genpspgraphics.cxx +++ b/vcl/generic/print/genpspgraphics.cxx @@ -43,7 +43,6 @@ #include "outfont.hxx" #include "fontsubset.hxx" #include "salprn.hxx" -#include "region.h" #include "langboost.hxx" #include <config_graphite.h> @@ -322,20 +321,45 @@ void GenPspGraphics::ResetClipRegion() bool GenPspGraphics::setClipRegion( const Region& i_rClip ) { // TODO: support polygonal clipregions here - m_pPrinterGfx->BeginSetClipRegion( i_rClip.GetRectCount() ); + RectangleVector aRectangles; + i_rClip.GetRegionRectangles(aRectangles); + m_pPrinterGfx->BeginSetClipRegion(aRectangles.size()); - ImplRegionInfo aInfo; - long nX, nY, nW, nH; - bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); - while( bRegionRect ) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - if ( nW && nH ) + const long nW(aRectIter->GetWidth()); + + if(nW) { - m_pPrinterGfx->UnionClipRegion( nX, nY, nW, nH ); + const long nH(aRectIter->GetHeight()); + + if(nH) + { + m_pPrinterGfx->UnionClipRegion( + aRectIter->Left(), + aRectIter->Top(), + nW, + nH); + } } - bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); } + m_pPrinterGfx->EndSetClipRegion(); + + //m_pPrinterGfx->BeginSetClipRegion( i_rClip.GetRectCount() ); + // + //ImplRegionInfo aInfo; + //long nX, nY, nW, nH; + //bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); + //while( bRegionRect ) + //{ + // if ( nW && nH ) + // { + // m_pPrinterGfx->UnionClipRegion( nX, nY, nW, nH ); + // } + // bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); + //} + //m_pPrinterGfx->EndSetClipRegion(); return true; } diff --git a/vcl/headless/svpgdi.cxx b/vcl/headless/svpgdi.cxx index dbd233c3692f..baab42e5d25d 100644 --- a/vcl/headless/svpgdi.cxx +++ b/vcl/headless/svpgdi.cxx @@ -37,7 +37,6 @@ #include <sys/stat.h> #endif -#include <region.h> #include <stdio.h> inline void dbgOut( const basebmp::BitmapDeviceSharedPtr& @@ -165,21 +164,30 @@ void SvpSalGraphics::ensureClip() m_aClipMap = basebmp::createBitmapDevice( aSize, false, basebmp::Format::ONE_BIT_MSB_GREY ); m_aClipMap->clear( basebmp::Color(0xFFFFFFFF) ); - // fprintf( stderr, "non rect clip region set with %d rects:\n", - // (int)m_aClipRegion.GetRectCount() ); - ImplRegionInfo aInfo; - long nX, nY, nW, nH; - bool bRegionRect = m_aClipRegion.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); - while( bRegionRect ) + RectangleVector aRectangles; + m_aClipRegion.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - if ( nW && nH ) + const long nW(aRectIter->GetWidth()); + if(nW) { - basegfx::B2DPolyPolygon aFull; - aFull.append( basegfx::tools::createPolygonFromRect( basegfx::B2DRectangle( nX, nY, nX+nW, nY+nH ) ) ); - m_aClipMap->fillPolyPolygon( aFull, basebmp::Color(0), basebmp::DrawMode_PAINT ); + const long nH(aRectIter->GetHeight()); + + if(nH) + { + basegfx::B2DPolyPolygon aFull; + + aFull.append( + basegfx::tools::createPolygonFromRect( + basegfx::B2DRectangle( + aRectIter->Left(), + aRectIter->Top(), + aRectIter->Left() + nW, + aRectIter->Top() + nH))); + m_aClipMap->fillPolyPolygon(aFull, basebmp::Color(0), basebmp::DrawMode_PAINT); + } } - // fprintf( stderr, "\t %ld,%ld %ldx%ld\n", nX, nY, nW, nH ); - bRegionRect = m_aClipRegion.ImplGetNextRect( aInfo, nX, nY, nW, nH ); } m_bClipSetup = true; } @@ -213,17 +221,17 @@ bool SvpSalGraphics::isClippedSetup( const basegfx::B2IBox &aRange, SvpSalGraphi // then see if we are overlapping with just one int nHit = 0; - Rectangle aIterRect, aHitRect; - RegionHandle aHnd = m_aClipRegion.BeginEnumRects(); - while( m_aClipRegion.GetNextEnumRect( aHnd, aIterRect ) ) + Rectangle aHitRect; + RectangleVector aRectangles; + m_aClipRegion.GetRegionRectangles(aRectangles); + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - if( aIterRect.IsOver( aRect ) ) + if( aRectIter->IsOver( aRect ) ) { - aHitRect = aIterRect; + aHitRect = *aRectIter; nHit++; } } - m_aClipRegion.EndEnumRects (aHnd); if( nHit == 0 ) // rendering outside any clipping region { @@ -270,14 +278,23 @@ bool SvpSalGraphics::setClipRegion( const Region& i_rClip ) m_aClipRegion = i_rClip; m_aClipMap.reset(); if( i_rClip.IsEmpty() ) + { m_bClipSetup = true; + return true; + } + + RectangleVector aRectangles; + i_rClip.GetRegionRectangles(aRectangles); - else if( i_rClip.GetRectCount() == 1 ) + if(1 == aRectangles.size()) { m_aClipMap.reset(); - Rectangle aBoundRect( i_rClip.GetBoundRect() ); - m_aDevice = basebmp::subsetBitmapDevice( m_aOrigDevice, - basegfx::B2IBox(aBoundRect.Left(),aBoundRect.Top(),aBoundRect.Right(),aBoundRect.Bottom()) ); + + const Rectangle& aBoundRect = aRectangles[0]; + m_aDevice = basebmp::subsetBitmapDevice( + m_aOrigDevice, + basegfx::B2IBox(aBoundRect.Left(),aBoundRect.Top(),aBoundRect.Right(),aBoundRect.Bottom()) ); + m_bClipSetup = true; } else diff --git a/vcl/inc/region.h b/vcl/inc/region.h deleted file mode 100644 index 457bb1e6fa79..000000000000 --- a/vcl/inc/region.h +++ /dev/null @@ -1,128 +0,0 @@ -/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ -/* - * This file is part of the LibreOffice project. - * - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at http://mozilla.org/MPL/2.0/. - * - * This file incorporates work covered by the following license notice: - * - * Licensed to the Apache Software Foundation (ASF) under one or more - * contributor license agreements. See the NOTICE file distributed - * with this work for additional information regarding copyright - * ownership. The ASF licenses this file to you under the Apache - * License, Version 2.0 (the "License"); you may not use this file - * except in compliance with the License. You may obtain a copy of - * the License at http://www.apache.org/licenses/LICENSE-2.0 . - */ - -#ifndef _SV_REGION_H -#define _SV_REGION_H - -#include <vcl/regband.hxx> -#include <tools/poly.hxx> -#include <vcl/region.hxx> - -// ----------------- -// - Hilfsmethoden - -// ----------------- - -#ifdef DBG_UTIL -const char* ImplDbgTestRegion( const void* pObj ); -#endif - -// -------------------- -// - ImplRegionHandle - -// -------------------- - -struct ImplRegionHandle -{ - Region* mpRegion; - ImplRegionBand* mpCurrRectBand; - ImplRegionBandSep* mpCurrRectBandSep; - sal_Bool mbFirst; -}; - -// ------------------ -// - ImplRegionInfo - -// ------------------ - -struct ImplRegionInfo -{ - void* mpVoidCurrRectBand; - void* mpVoidCurrRectBandSep; -}; - -// -------------- -// - ImplRegion - -// -------------- - -struct ImplRegionBase -{ -public: - ImplRegionBase( int nCount = 1 ); // TODO: replace manual refcounting - virtual ~ImplRegionBase(); -public: - sal_uIntPtr mnRefCount; - sal_uIntPtr mnRectCount; - PolyPolygon* mpPolyPoly; - basegfx::B2DPolyPolygon* mpB2DPolyPoly; -}; - -class ImplRegion : public ImplRegionBase -{ - friend class Region; - -private: - ImplRegionBand* mpFirstBand; // root of the list with y-bands - ImplRegionBand* mpLastCheckedBand; - -public: - ImplRegion(); - ImplRegion( const PolyPolygon& rPolyPoly ); - ImplRegion( const basegfx::B2DPolyPolygon& ); - ImplRegion( const ImplRegion& rImplRegion ); - ~ImplRegion(); - - ImplRegionBand* ImplGetFirstRegionBand() const { return mpFirstBand; } - PolyPolygon* ImplGetPolyPoly() const { return mpPolyPoly; } - - void CreateBandRange( long nYTop, long nYBottom ); - void InsertBands( long nYTop, long nYBottom ); - sal_Bool InsertSingleBand( ImplRegionBand* mpImplRegionBand, - long nYBandPosition ); - sal_Bool InsertLine( const Point & rFirstPoint, - const Point & rSecondPoint, - long nLineID ); - sal_Bool InsertPoint( const Point &rPoint, - long nLineID, - sal_Bool bEndPoint, LineType eLineType ); - - /** Insert one band either after another band or as the first or only - band. Both the forward as well as the backward links are updated. - @param pPreviousBand - When <NULL/> then pBandToInsert is inserted as first band or as - only band when there are no other bands. - When not <NULL/> then pBandToInsert is inserted directly after - pPreviousBand. - @param pBandToInsert - The band to insert. - */ - void InsertBand (ImplRegionBand* pPreviousBand, - ImplRegionBand* pBandToInsert); - - void Union( long nLeft, long nTop, long nRight, long nBottom ); - void Exclude( long nLeft, long nTop, long nRight, long nBottom ); - void XOr( long nLeft, long nTop, long nRight, long nBottom ); - - // remove emtpy rects - sal_Bool OptimizeBandList(); - - friend SvStream& operator>>( SvStream& rIStm, Region& rRegion ); - friend SvStream& operator<<( SvStream& rOStm, const Region& rRegion ); -}; - -#endif // _SV_REGION_H - -/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ diff --git a/vcl/inc/regionband.hxx b/vcl/inc/regionband.hxx new file mode 100644 index 000000000000..89657036adff --- /dev/null +++ b/vcl/inc/regionband.hxx @@ -0,0 +1,83 @@ +/* + * This file is part of the LibreOffice project. + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * This file incorporates work covered by the following license notice: + * + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed + * with this work for additional information regarding copyright + * ownership. The ASF licenses this file to you under the Apache + * License, Version 2.0 (the "License"); you may not use this file + * except in compliance with the License. You may obtain a copy of + * the License at http://www.apache.org/licenses/LICENSE-2.0 . + */ +#ifndef _SV_REGIONBAND_HXX +#define _SV_REGIONBAND_HXX + +#include <vcl/regband.hxx> +#include <vcl/region.hxx> + +#ifdef DBG_UTIL +const char* ImplDbgTestRegionBand(const void*); +#endif + +////////////////////////////////////////////////////////////////////////////// + +class RegionBand +{ +private: + friend const char* ImplDbgTestRegionBand(const void*); + + ImplRegionBand* mpFirstBand; // root of the list with y-bands + ImplRegionBand* mpLastCheckedBand; + + void implReset(); + +public: + RegionBand(); + RegionBand(const RegionBand&); + RegionBand& operator=(const RegionBand&); + RegionBand(const Rectangle&); + ~RegionBand(); + + bool operator==( const RegionBand& rRegionBand ) const; + bool operator!=( const RegionBand& rRegionBand ) const { return !(RegionBand::operator==( rRegionBand )); } + + void load(SvStream& rIStrm); + void save(SvStream& rIStrm) const; + + bool isSingleRectangle() const; + ImplRegionBand* ImplGetFirstRegionBand() const { return mpFirstBand; } + void ImplAddMissingBands(const long nTop, const long nBottom); + void InsertBand(ImplRegionBand* pPreviousBand, ImplRegionBand* pBandToInsert); + void processPoints(); + void CreateBandRange(long nYTop, long nYBottom); + bool InsertLine(const Point& rStartPt, const Point& rEndPt, long nLineId); + bool InsertPoint(const Point &rPoint, long nLineID, bool bEndPoint, LineType eLineType); + bool OptimizeBandList(); + void Move(long nHorzMove, long nVertMove); + void Scale(double fScaleX, double fScaleY); + void InsertBands(long nTop, long nBottom); + bool InsertSingleBand(ImplRegionBand* pBand, long nYBandPosition); + void Union(long nLeft, long nTop, long nRight, long nBottom); + void Intersect(long nLeft, long nTop, long nRight, long nBottom); + void Union(const RegionBand& rSource); + void Exclude(long nLeft, long nTop, long nRight, long nBottom); + void XOr(long nLeft, long nTop, long nRight, long nBottom); + void Intersect(const RegionBand& rSource); + bool Exclude(const RegionBand& rSource); + void XOr(const RegionBand& rSource); + Rectangle GetBoundRect() const; + bool IsInside(const Point& rPoint) const; + sal_uInt32 getRectangleCount() const; // only users are Region::Intersect and PSWriter::ImplBmp + void GetRegionRectangles(RectangleVector& rTarget) const; +}; + +#endif // _SV_REGIONBAND_HXX + +////////////////////////////////////////////////////////////////////////////// +//eof diff --git a/vcl/source/gdi/bitmap.cxx b/vcl/source/gdi/bitmap.cxx index 2bb1a9d55889..7ba90c7dbc53 100644 --- a/vcl/source/gdi/bitmap.cxx +++ b/vcl/source/gdi/bitmap.cxx @@ -1281,38 +1281,92 @@ Region Bitmap::CreateRegion( const Color& rColor, const Rectangle& rRect ) const if( pReadAcc ) { - Rectangle aSubRect; + //Rectangle aSubRect; const long nLeft = aRect.Left(); const long nTop = aRect.Top(); const long nRight = aRect.Right(); const long nBottom = aRect.Bottom(); const BitmapColor aMatch( pReadAcc->GetBestMatchingColor( rColor ) ); - aRegion.ImplBeginAddRect(); + //RectangleVector aRectangles; + //aRegion.ImplBeginAddRect(); + std::vector< long > aLine; + long nYStart(nTop); + long nY(nTop); - for( long nY = nTop; nY <= nBottom; nY++ ) + for( ; nY <= nBottom; nY++ ) { - aSubRect.Top() = aSubRect.Bottom() = nY; + //aSubRect.Top() = aSubRect.Bottom() = nY; + std::vector< long > aNewLine; + long nX(nLeft); - for( long nX = nLeft; nX <= nRight; ) + for( ; nX <= nRight; ) { while( ( nX <= nRight ) && ( aMatch != pReadAcc->GetPixel( nY, nX ) ) ) nX++; if( nX <= nRight ) { - aSubRect.Left() = nX; + aNewLine.push_back(nX); + //aSubRect.Left() = nX; while( ( nX <= nRight ) && ( aMatch == pReadAcc->GetPixel( nY, nX ) ) ) nX++; - aSubRect.Right() = nX - 1L; - aRegion.ImplAddRect( aSubRect ); + //aSubRect.Right() = nX - 1L; + aNewLine.push_back(nX - 1); + + //aRegion.ImplAddRect( aSubRect ); + //aRectangles.push_back(aSubRect); + //aRegion.Union(aSubRect); } } + + if(aNewLine != aLine) + { + // need to write aLine, it's different from the next line + if(aLine.size()) + { + Rectangle aSubRect; + + // enter y values and proceed ystart + aSubRect.Top() = nYStart; + aSubRect.Bottom() = nY ? nY - 1 : 0; + + for(sal_uInt32 a(0); a < aLine.size();) + { + aSubRect.Left() = aLine[a++]; + aSubRect.Right() = aLine[a++]; + aRegion.Union(aSubRect); + } + } + + // copy line as new line + aLine = aNewLine; + nYStart = nY; + } } - aRegion.ImplEndAddRect(); + // write last line if used + if(aLine.size()) + { + Rectangle aSubRect; + + // enter y values + aSubRect.Top() = nYStart; + aSubRect.Bottom() = nY ? nY - 1 : 0; + + for(sal_uInt32 a(0); a < aLine.size();) + { + aSubRect.Left() = aLine[a++]; + aSubRect.Right() = aLine[a++]; + aRegion.Union(aSubRect); + } + } + + //aRegion.ImplEndAddRect(); + //aRegion.SetRegionRectangles(aRectangles); + ( (Bitmap*) this )->ReleaseAccess( pReadAcc ); } else diff --git a/vcl/source/gdi/bmpacc3.cxx b/vcl/source/gdi/bmpacc3.cxx index 791a75d9329d..c35f531c5433 100644 --- a/vcl/source/gdi/bmpacc3.cxx +++ b/vcl/source/gdi/bmpacc3.cxx @@ -243,4 +243,133 @@ void BitmapWriteAccess::DrawRect( const Rectangle& rRect ) } } +// ------------------------------------------------------------------ + +void BitmapWriteAccess::FillPolygon( const Polygon& rPoly ) +{ + const sal_uInt16 nSize = rPoly.GetSize(); + + if( nSize && mpFillColor ) + { + const BitmapColor& rFillColor = *mpFillColor; + Region aRegion( rPoly ); +// Rectangle aRect; + + aRegion.Intersect( Rectangle( Point(), Size( Width(), Height() ) ) ); + + if( !aRegion.IsEmpty() ) + { + RectangleVector aRectangles; + aRegion.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) + { + for(long nY = aRectIter->Top(), nEndY = aRectIter->Bottom(); nY <= nEndY; nY++) + { + for(long nX = aRectIter->Left(), nEndX = aRectIter->Right(); nX <= nEndX; nX++) + { + SetPixel(nY, nX, rFillColor); + } + } + } + + //RegionHandle aRegHandle( aRegion.BeginEnumRects() ); + // + //while( aRegion.GetEnumRects( aRegHandle, aRect ) ) + // for( long nY = aRect.Top(), nEndY = aRect.Bottom(); nY <= nEndY; nY++ ) + // for( long nX = aRect.Left(), nEndX = aRect.Right(); nX <= nEndX; nX++ ) + // SetPixel( nY, nX, rFillColor ); + // + //aRegion.EndEnumRects( aRegHandle ); + } + } +} + +// ------------------------------------------------------------------ + +void BitmapWriteAccess::DrawPolygon( const Polygon& rPoly ) +{ + if( mpFillColor ) + FillPolygon( rPoly ); + + if( mpLineColor && ( !mpFillColor || ( *mpFillColor != *mpLineColor ) ) ) + { + const sal_uInt16 nSize = rPoly.GetSize(); + + for( sal_uInt16 i = 0, nSize1 = nSize - 1; i < nSize1; i++ ) + DrawLine( rPoly[ i ], rPoly[ i + 1 ] ); + + if( rPoly[ nSize - 1 ] != rPoly[ 0 ] ) + DrawLine( rPoly[ nSize - 1 ], rPoly[ 0 ] ); + } +} + +// ------------------------------------------------------------------ + +void BitmapWriteAccess::FillPolyPolygon( const PolyPolygon& rPolyPoly ) +{ + const sal_uInt16 nCount = rPolyPoly.Count(); + + if( nCount && mpFillColor ) + { + const BitmapColor& rFillColor = *mpFillColor; + Region aRegion( rPolyPoly ); + //Rectangle aRect; + + aRegion.Intersect( Rectangle( Point(), Size( Width(), Height() ) ) ); + + if( !aRegion.IsEmpty() ) + { + RectangleVector aRectangles; + aRegion.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) + { + for(long nY = aRectIter->Top(), nEndY = aRectIter->Bottom(); nY <= nEndY; nY++) + { + for(long nX = aRectIter->Left(), nEndX = aRectIter->Right(); nX <= nEndX; nX++) + { + SetPixel(nY, nX, rFillColor); + } + } + } + + //RegionHandle aRegHandle( aRegion.BeginEnumRects() ); + // + //while( aRegion.GetEnumRects( aRegHandle, aRect ) ) + // for( long nY = aRect.Top(), nEndY = aRect.Bottom(); nY <= nEndY; nY++ ) + // for( long nX = aRect.Left(), nEndX = aRect.Right(); nX <= nEndX; nX++ ) + // SetPixel( nY, nX, rFillColor ); + // + //aRegion.EndEnumRects( aRegHandle ); + } + } +} + +// ------------------------------------------------------------------ + +void BitmapWriteAccess::DrawPolyPolygon( const PolyPolygon& rPolyPoly ) +{ + if( mpFillColor ) + FillPolyPolygon( rPolyPoly ); + + if( mpLineColor && ( !mpFillColor || ( *mpFillColor != *mpLineColor ) ) ) + { + for( sal_uInt16 n = 0, nCount = rPolyPoly.Count(); n < nCount; ) + { + const Polygon& rPoly = rPolyPoly[ n++ ]; + const sal_uInt16 nSize = rPoly.GetSize(); + + if( nSize ) + { + for( sal_uInt16 i = 0, nSize1 = nSize - 1; i < nSize1; i++ ) + DrawLine( rPoly[ i ], rPoly[ i + 1 ] ); + + if( rPoly[ nSize - 1 ] != rPoly[ 0 ] ) + DrawLine( rPoly[ nSize - 1 ], rPoly[ 0 ] ); + } + } + } +} + /* vim:set shiftwidth=4 softtabstop=4 expandtab: */ diff --git a/vcl/source/gdi/gdimtf.cxx b/vcl/source/gdi/gdimtf.cxx index fce3d091ab9c..77a426f2da7e 100644 --- a/vcl/source/gdi/gdimtf.cxx +++ b/vcl/source/gdi/gdimtf.cxx @@ -1282,8 +1282,8 @@ void GDIMetaFile::Rotate( long nAngle10 ) { MetaClipRegionAction* pAct = (MetaClipRegionAction*) pAction; - if( pAct->IsClipping() && pAct->GetRegion().HasPolyPolygon() ) - aMtf.AddAction( new MetaClipRegionAction( Region( ImplGetRotatedPolyPolygon( pAct->GetRegion().GetPolyPolygon(), aRotAnchor, aRotOffset, fSin, fCos ) ), sal_True ) ); + if( pAct->IsClipping() && pAct->GetRegion().HasPolyPolygonOrB2DPolyPolygon() ) + aMtf.AddAction( new MetaClipRegionAction( Region( ImplGetRotatedPolyPolygon( pAct->GetRegion().GetAsPolyPolygon(), aRotAnchor, aRotOffset, fSin, fCos ) ), sal_True ) ); else { pAction->Duplicate(); @@ -1306,8 +1306,8 @@ void GDIMetaFile::Rotate( long nAngle10 ) MetaISectRegionClipRegionAction* pAct = (MetaISectRegionClipRegionAction*) pAction; const Region& rRegion = pAct->GetRegion(); - if( rRegion.HasPolyPolygon() ) - aMtf.AddAction( new MetaISectRegionClipRegionAction( Region( ImplGetRotatedPolyPolygon( rRegion.GetPolyPolygon(), aRotAnchor, aRotOffset, fSin, fCos ) ) ) ); + if( rRegion.HasPolyPolygonOrB2DPolyPolygon() ) + aMtf.AddAction( new MetaISectRegionClipRegionAction( Region( ImplGetRotatedPolyPolygon( rRegion.GetAsPolyPolygon(), aRotAnchor, aRotOffset, fSin, fCos ) ) ) ); else { pAction->Duplicate(); @@ -2600,12 +2600,12 @@ sal_uLong GDIMetaFile::GetChecksum() const MetaClipRegionAction* pAct = dynamic_cast< MetaClipRegionAction* >(pAction); const Region& rRegion = pAct->GetRegion(); - if(rRegion.HasPolyPolygon()) + if(rRegion.HasPolyPolygonOrB2DPolyPolygon()) { // It has shown that this is a possible bottleneck for checksum calculation. // In worst case a very expensive RegionHandle representation gets created. // In this case it's cheaper to use the PolyPolygon - const basegfx::B2DPolyPolygon aPolyPolygon(rRegion.GetB2DPolyPolygon()); + const basegfx::B2DPolyPolygon aPolyPolygon(rRegion.GetAsB2DPolyPolygon()); const sal_uInt32 nPolyCount(aPolyPolygon.count()); SVBT64 aSVBT64; diff --git a/vcl/source/gdi/outdev.cxx b/vcl/source/gdi/outdev.cxx index 65119b6141f9..090f22990a80 100644 --- a/vcl/source/gdi/outdev.cxx +++ b/vcl/source/gdi/outdev.cxx @@ -41,7 +41,6 @@ #include <svdata.hxx> #include <window.h> #include <outdev.h> -#include <region.h> #include <outdata.hxx> #include <basegfx/point/b2dpoint.hxx> @@ -322,7 +321,7 @@ void OutputDevice::ImplDrawPolyPolygon( sal_uInt16 nPoly, const PolyPolygon& rPo // ======================================================================= OutputDevice::OutputDevice() : - maRegion( REGION_NULL ), + maRegion(true), maFillColor( COL_WHITE ), maTextLineColor( COL_TRANSPARENT ), maSettings( Application::GetSettings() ) @@ -530,23 +529,35 @@ void OutputDevice::ImplReMirror( Rectangle &rRect ) const } void OutputDevice::ImplReMirror( Region &rRegion ) const { - long nX; - long nY; - long nWidth; - long nHeight; - ImplRegionInfo aInfo; - sal_Bool bRegionRect; - Region aMirroredRegion; + RectangleVector aRectangles; + rRegion.GetRegionRectangles(aRectangles); + Region aMirroredRegion; - bRegionRect = rRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); - while ( bRegionRect ) + for(RectangleVector::iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - Rectangle aRect( Point(nX, nY), Size(nWidth, nHeight) ); - ImplReMirror( aRect ); - aMirroredRegion.Union( aRect ); - bRegionRect = rRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + ImplReMirror(*aRectIter); + aMirroredRegion.Union(*aRectIter); } + rRegion = aMirroredRegion; + +// long nX; +// long nY; +// long nWidth; +// long nHeight; +// ImplRegionInfo aInfo; +// sal_Bool bRegionRect; +// Region aMirroredRegion; +// +// bRegionRect = rRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); +// while ( bRegionRect ) +// { +// Rectangle aRect( Point(nX, nY), Size(nWidth, nHeight) ); +// ImplReMirror( aRect ); +// aMirroredRegion.Union( aRect ); +// bRegionRect = rRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); +// } +// rRegion = aMirroredRegion; } SalGraphics* OutputDevice::ImplGetGraphics() const @@ -1004,7 +1015,7 @@ void OutputDevice::ImplSetClipRegion( const Region* pRegion ) { if ( mbClipRegion ) { - maRegion = Region( REGION_NULL ); + maRegion = Region(true); mbClipRegion = sal_False; mbInitClipRegion = sal_True; } @@ -1033,13 +1044,14 @@ void OutputDevice::SetClipRegion() void OutputDevice::SetClipRegion( const Region& rRegion ) { DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); if ( mpMetaFile ) mpMetaFile->AddAction( new MetaClipRegionAction( rRegion, sal_True ) ); - if ( rRegion.GetType() == REGION_NULL ) + if ( rRegion.IsNull() ) + { ImplSetClipRegion( NULL ); + } else { Region aRegion = LogicToPixel( rRegion ); @@ -1063,7 +1075,7 @@ Region OutputDevice::GetActiveClipRegion() const if ( GetOutDevType() == OUTDEV_WINDOW ) { - Region aRegion( REGION_NULL ); + Region aRegion(true); Window* pWindow = (Window*)this; if ( pWindow->mpWindowImpl->mbInPaint ) { @@ -1115,11 +1127,8 @@ void OutputDevice::IntersectClipRegion( const Rectangle& rRect ) void OutputDevice::IntersectClipRegion( const Region& rRegion ) { DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - - RegionType eType = rRegion.GetType(); - if ( eType != REGION_NULL ) + if(!rRegion.IsNull()) { if ( mpMetaFile ) mpMetaFile->AddAction( new MetaISectRegionClipRegionAction( rRegion ) ); diff --git a/vcl/source/gdi/outdev2.cxx b/vcl/source/gdi/outdev2.cxx index e17842609cf7..915fb89aa122 100644 --- a/vcl/source/gdi/outdev2.cxx +++ b/vcl/source/gdi/outdev2.cxx @@ -37,7 +37,6 @@ #include <image.h> #include <outdev.h> #include <window.h> -#include <region.h> #include <outdata.hxx> DBG_NAMEEX( OutputDevice ) @@ -1945,7 +1944,7 @@ void OutputDevice::ImplPrintTransparent( const Bitmap& rBmp, const Bitmap& rMask // do painting const long nSrcWidth = aSrcRect.GetWidth(), nSrcHeight = aSrcRect.GetHeight(); - long nX, nY, nWorkX, nWorkY, nWorkWidth, nWorkHeight; + long nX, nY; // , nWorkX, nWorkY, nWorkWidth, nWorkHeight; long* pMapX = new long[ nSrcWidth + 1 ]; long* pMapY = new long[ nSrcHeight + 1 ]; const bool bOldMap = mbMap; @@ -1960,22 +1959,38 @@ void OutputDevice::ImplPrintTransparent( const Bitmap& rBmp, const Bitmap& rMask pMapY[ nY ] = aDestPt.Y() + FRound( (double) aDestSz.Height() * nY / nSrcHeight ); // walk through all rectangles of mask - Region aWorkRgn( aMask.CreateRegion( COL_BLACK, Rectangle( Point(), aMask.GetSizePixel() ) ) ); - ImplRegionInfo aInfo; - bool bRgnRect = aWorkRgn.ImplGetFirstRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + const Region aWorkRgn(aMask.CreateRegion(COL_BLACK, Rectangle(Point(), aMask.GetSizePixel()))); + RectangleVector aRectangles; + aWorkRgn.GetRegionRectangles(aRectangles); - while( bRgnRect ) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - Bitmap aBandBmp( aPaint ); - const Rectangle aBandRect( Point( nWorkX, nWorkY ), Size( nWorkWidth, nWorkHeight ) ); - const Point aMapPt( pMapX[ nWorkX ], pMapY[ nWorkY ] ); - const Size aMapSz( pMapX[ nWorkX + nWorkWidth ] - aMapPt.X(), pMapY[ nWorkY + nWorkHeight ] - aMapPt.Y() ); - - aBandBmp.Crop( aBandRect ); - ImplDrawBitmap( aMapPt, aMapSz, Point(), aBandBmp.GetSizePixel(), aBandBmp, META_BMPSCALEPART_ACTION ); - bRgnRect = aWorkRgn.ImplGetNextRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + const Point aMapPt(pMapX[aRectIter->Left()], pMapY[aRectIter->Top()]); + const Size aMapSz( + pMapX[aRectIter->Right() + 1] - aMapPt.X(), // pMapX[L + W] -> L + ((R - L) + 1) -> R + 1 + pMapY[aRectIter->Bottom() + 1] - aMapPt.Y()); // same for Y + Bitmap aBandBmp(aPaint); + + aBandBmp.Crop(*aRectIter); + ImplDrawBitmap(aMapPt, aMapSz, Point(), aBandBmp.GetSizePixel(), aBandBmp, META_BMPSCALEPART_ACTION); } + //Region aWorkRgn( aMask.CreateRegion( COL_BLACK, Rectangle( Point(), aMask.GetSizePixel() ) ) ); + //ImplRegionInfo aInfo; + //sal_Bool bRgnRect = aWorkRgn.ImplGetFirstRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + // + //while( bRgnRect ) + //{ + // Bitmap aBandBmp( aPaint ); + // const Rectangle aBandRect( Point( nWorkX, nWorkY ), Size( nWorkWidth, nWorkHeight ) ); + // const Point aMapPt( pMapX[ nWorkX ], pMapY[ nWorkY ] ); + // const Size aMapSz( pMapX[ nWorkX + nWorkWidth ] - aMapPt.X(), pMapY[ nWorkY + nWorkHeight ] - aMapPt.Y() ); + // + // aBandBmp.Crop( aBandRect ); + // ImplDrawBitmap( aMapPt, aMapSz, Point(), aBandBmp.GetSizePixel(), aBandBmp, META_BMPSCALEPART_ACTION ); + // bRgnRect = aWorkRgn.ImplGetNextRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + //} + mbMap = bOldMap; delete[] pMapX; @@ -2028,7 +2043,7 @@ void OutputDevice::ImplPrintMask( const Bitmap& rMask, const Color& rMaskColor, // do painting const long nSrcWidth = aSrcRect.GetWidth(), nSrcHeight = aSrcRect.GetHeight(); - long nX, nY, nWorkX, nWorkY, nWorkWidth, nWorkHeight; + long nX, nY; //, nWorkX, nWorkY, nWorkWidth, nWorkHeight; long* pMapX = new long[ nSrcWidth + 1 ]; long* pMapY = new long[ nSrcHeight + 1 ]; GDIMetaFile* pOldMetaFile = mpMetaFile; @@ -2050,19 +2065,33 @@ void OutputDevice::ImplPrintMask( const Bitmap& rMask, const Color& rMaskColor, pMapY[ nY ] = aDestPt.Y() + FRound( (double) aDestSz.Height() * nY / nSrcHeight ); // walk through all rectangles of mask - Region aWorkRgn( aMask.CreateRegion( COL_BLACK, Rectangle( Point(), aMask.GetSizePixel() ) ) ); - ImplRegionInfo aInfo; - bool bRgnRect = aWorkRgn.ImplGetFirstRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + const Region aWorkRgn(aMask.CreateRegion(COL_BLACK, Rectangle(Point(), aMask.GetSizePixel()))); + RectangleVector aRectangles; + aWorkRgn.GetRegionRectangles(aRectangles); - while( bRgnRect ) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - const Point aMapPt( pMapX[ nWorkX ], pMapY[ nWorkY ] ); - const Size aMapSz( pMapX[ nWorkX + nWorkWidth ] - aMapPt.X(), pMapY[ nWorkY + nWorkHeight ] - aMapPt.Y() ); + const Point aMapPt(pMapX[aRectIter->Left()], pMapY[aRectIter->Top()]); + const Size aMapSz( + pMapX[aRectIter->Right() + 1] - aMapPt.X(), // pMapX[L + W] -> L + ((R - L) + 1) -> R + 1 + pMapY[aRectIter->Bottom() + 1] - aMapPt.Y()); // same for Y - DrawRect( Rectangle( aMapPt, aMapSz ) ); - bRgnRect = aWorkRgn.ImplGetNextRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + DrawRect(Rectangle(aMapPt, aMapSz)); } + //Region aWorkRgn( aMask.CreateRegion( COL_BLACK, Rectangle( Point(), aMask.GetSizePixel() ) ) ); + //ImplRegionInfo aInfo; + //sal_Bool bRgnRect = aWorkRgn.ImplGetFirstRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + // + //while( bRgnRect ) + //{ + // const Point aMapPt( pMapX[ nWorkX ], pMapY[ nWorkY ] ); + // const Size aMapSz( pMapX[ nWorkX + nWorkWidth ] - aMapPt.X(), pMapY[ nWorkY + nWorkHeight ] - aMapPt.Y() ); + // + // DrawRect( Rectangle( aMapPt, aMapSz ) ); + // bRgnRect = aWorkRgn.ImplGetNextRect( aInfo, nWorkX, nWorkY, nWorkWidth, nWorkHeight ); + //} + Pop(); delete[] pMapX; delete[] pMapY; diff --git a/vcl/source/gdi/outmap.cxx b/vcl/source/gdi/outmap.cxx index 25cbb53460bf..7ef70f8b9411 100644 --- a/vcl/source/gdi/outmap.cxx +++ b/vcl/source/gdi/outmap.cxx @@ -34,7 +34,6 @@ #include <vcl/outdev.hxx> #include <svdata.hxx> -#include <region.h> #include <window.h> #include <outdev.h> #include <salgdi.hxx> @@ -679,8 +678,6 @@ Rectangle OutputDevice::ImplDevicePixelToLogic( const Rectangle& rPixelRect ) co Region OutputDevice::ImplPixelToDevicePixel( const Region& rRegion ) const { - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - if ( !mnOutOffX && !mnOutOffY ) return rRegion; @@ -1132,48 +1129,73 @@ PolyPolygon OutputDevice::LogicToPixel( const PolyPolygon& rLogicPolyPoly ) cons // ----------------------------------------------------------------------- +basegfx::B2DPolygon OutputDevice::LogicToPixel( const basegfx::B2DPolygon& rLogicPoly ) const +{ + basegfx::B2DPolygon aTransformedPoly = rLogicPoly; + const ::basegfx::B2DHomMatrix& rTransformationMatrix = GetViewTransformation(); + aTransformedPoly.transform( rTransformationMatrix ); + return aTransformedPoly; +} + +// ----------------------------------------------------------------------- + +basegfx::B2DPolyPolygon OutputDevice::LogicToPixel( const basegfx::B2DPolyPolygon& rLogicPolyPoly ) const +{ + basegfx::B2DPolyPolygon aTransformedPoly = rLogicPolyPoly; + const ::basegfx::B2DHomMatrix& rTransformationMatrix = GetViewTransformation(); + aTransformedPoly.transform( rTransformationMatrix ); + return aTransformedPoly; +} + +// ----------------------------------------------------------------------- + Region OutputDevice::LogicToPixel( const Region& rLogicRegion ) const { DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); - DBG_CHKOBJ( &rLogicRegion, Region, ImplDbgTestRegion ); - - RegionType eType = rLogicRegion.GetType(); - if ( !mbMap || (eType == REGION_EMPTY) || (eType == REGION_NULL) ) + if(!mbMap || rLogicRegion.IsNull() || rLogicRegion.IsEmpty()) + { return rLogicRegion; + } - Region aRegion; - const ImplRegion& rImplRegion = *rLogicRegion.ImplGetImplRegion(); - const PolyPolygon* pPolyPoly = rImplRegion.mpPolyPoly; - const basegfx::B2DPolyPolygon* pB2DPolyPoly = rImplRegion.mpB2DPolyPoly; + Region aRegion; - if ( pPolyPoly ) - aRegion = Region( LogicToPixel( *pPolyPoly ) ); - else if( pB2DPolyPoly ) + if(rLogicRegion.getB2DPolyPolygon()) { - basegfx::B2DPolyPolygon aTransformedPoly = *pB2DPolyPoly; - const ::basegfx::B2DHomMatrix& rTransformationMatrix = GetViewTransformation(); - aTransformedPoly.transform( rTransformationMatrix ); - aRegion = Region( aTransformedPoly ); + aRegion = Region(LogicToPixel(*rLogicRegion.getB2DPolyPolygon())); } - else + else if(rLogicRegion.getPolyPolygon()) + { + aRegion = Region(LogicToPixel(*rLogicRegion.getPolyPolygon())); + } + else if(rLogicRegion.getRegionBand()) { - long nX; - long nY; - long nWidth; - long nHeight; - ImplRegionInfo aInfo; - sal_Bool bRegionRect; + RectangleVector aRectangles; + rLogicRegion.GetRegionRectangles(aRectangles); + const RectangleVector& rRectangles(aRectangles); // needed to make the '!=' work - aRegion.ImplBeginAddRect(); - bRegionRect = rLogicRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); - while ( bRegionRect ) + // make reverse run to fill new region bottom-up, this will speed it up due to the used data structuring + for(RectangleVector::const_reverse_iterator aRectIter(rRectangles.rbegin()); aRectIter != rRectangles.rend(); aRectIter++) { - Rectangle aRect( Point( nX, nY ), Size( nWidth, nHeight ) ); - aRegion.ImplAddRect( LogicToPixel( aRect ) ); - bRegionRect = rLogicRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + aRegion.Union(LogicToPixel(*aRectIter)); } - aRegion.ImplEndAddRect(); + + //long nX(0); + //long nY(0); + //long nWidth(0); + //long nHeight(0); + //ImplRegionInfo aInfo; + //aRegion.ImplBeginAddRect(); + //bool bRegionRect(rLogicRegion.ImplGetFirstRect(aInfo, nX, nY, nWidth, nHeight)); + // + //while(bRegionRect) + //{ + // const Rectangle aRect(Point(nX, nY), Size(nWidth, nHeight)); + // aRegion.ImplAddRect(LogicToPixel(aRect)); + // bRegionRect = rLogicRegion.ImplGetNextRect(aInfo, nX, nY, nWidth, nHeight); + //} + // + //aRegion.ImplEndAddRect(); } return aRegion; @@ -1295,6 +1317,27 @@ Polygon OutputDevice::LogicToPixel( const Polygon& rLogicPoly, // ----------------------------------------------------------------------- +PolyPolygon OutputDevice::LogicToPixel( const PolyPolygon& rLogicPolyPoly, + const MapMode& rMapMode ) const +{ + DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); + DBG_CHKOBJ( &rLogicPolyPoly, PolyPolygon, NULL ); + + if ( rMapMode.IsDefault() ) + return rLogicPolyPoly; + + PolyPolygon aPolyPoly( rLogicPolyPoly ); + sal_uInt16 nPoly = aPolyPoly.Count(); + for( sal_uInt16 i = 0; i < nPoly; i++ ) + { + Polygon& rPoly = aPolyPoly[i]; + rPoly = LogicToPixel( rPoly, rMapMode ); + } + return aPolyPoly; +} + +// ----------------------------------------------------------------------- + basegfx::B2DPolyPolygon OutputDevice::LogicToPixel( const basegfx::B2DPolyPolygon& rLogicPolyPoly, const MapMode& rMapMode ) const { @@ -1306,6 +1349,71 @@ basegfx::B2DPolyPolygon OutputDevice::LogicToPixel( const basegfx::B2DPolyPolygo // ----------------------------------------------------------------------- +basegfx::B2DPolygon OutputDevice::LogicToPixel( const basegfx::B2DPolygon& rLogicPoly, + const MapMode& rMapMode ) const +{ + basegfx::B2DPolygon aTransformedPoly = rLogicPoly; + const ::basegfx::B2DHomMatrix& rTransformationMatrix = GetViewTransformation( rMapMode ); + aTransformedPoly.transform( rTransformationMatrix ); + return aTransformedPoly; +} + +// ----------------------------------------------------------------------- + +Region OutputDevice::LogicToPixel( const Region& rLogicRegion, const MapMode& rMapMode ) const +{ + DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); + + if(rMapMode.IsDefault() || rLogicRegion.IsNull() || rLogicRegion.IsEmpty()) + { + return rLogicRegion; + } + + Region aRegion; + + if(rLogicRegion.getB2DPolyPolygon()) + { + aRegion = Region(LogicToPixel(*rLogicRegion.getB2DPolyPolygon(), rMapMode)); + } + else if(rLogicRegion.getPolyPolygon()) + { + aRegion = Region(LogicToPixel(*rLogicRegion.getPolyPolygon(), rMapMode)); + } + else if(rLogicRegion.getRegionBand()) + { + RectangleVector aRectangles; + rLogicRegion.GetRegionRectangles(aRectangles); + const RectangleVector& rRectangles(aRectangles); // needed to make the '!=' work + + // make reverse run to fill new region bottom-up, this will speed it up due to the used data structuring + for(RectangleVector::const_reverse_iterator aRectIter(rRectangles.rbegin()); aRectIter != rRectangles.rend(); aRectIter++) + { + aRegion.Union(LogicToPixel(*aRectIter, rMapMode)); + } + + //long nX(0); + //long nY(0); + //long nWidth(0); + //long nHeight(0); + //ImplRegionInfo aInfo; + //aRegion.ImplBeginAddRect(); + //bool bRegionRect(rLogicRegion.ImplGetFirstRect(aInfo, nX, nY, nWidth, nHeight)); + // + //while(bRegionRect) + //{ + // const Rectangle aRect(Point(nX, nY), Size(nWidth, nHeight)); + // aRegion.ImplAddRect(LogicToPixel(aRect, rMapMode)); + // bRegionRect = rLogicRegion.ImplGetNextRect(aInfo, nX, nY, nWidth, nHeight); + //} + // + //aRegion.ImplEndAddRect(); + } + + return aRegion; +} + +// ----------------------------------------------------------------------- + Point OutputDevice::PixelToLogic( const Point& rDevicePt ) const { DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); @@ -1427,43 +1535,50 @@ basegfx::B2DPolyPolygon OutputDevice::PixelToLogic( const basegfx::B2DPolyPolygo Region OutputDevice::PixelToLogic( const Region& rDeviceRegion ) const { DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); - DBG_CHKOBJ( &rDeviceRegion, Region, ImplDbgTestRegion ); - - RegionType eType = rDeviceRegion.GetType(); - if ( !mbMap || (eType == REGION_EMPTY) || (eType == REGION_NULL) ) + if(!mbMap || rDeviceRegion.IsNull() || rDeviceRegion.IsEmpty()) + { return rDeviceRegion; + } - Region aRegion; - basegfx::B2DPolyPolygon* pB2DPolyPoly = rDeviceRegion.ImplGetImplRegion()->mpB2DPolyPoly; - PolyPolygon* pPolyPoly = pB2DPolyPoly ? 0 : rDeviceRegion.ImplGetImplRegion()->mpPolyPoly; + Region aRegion; - if ( pB2DPolyPoly ) // conversion with B2DPolyPolygon lost polygon-based ClipRegion + if(rDeviceRegion.getB2DPolyPolygon()) { - aRegion = Region( PixelToLogic( *pB2DPolyPoly ) ); + aRegion = Region(PixelToLogic(*rDeviceRegion.getB2DPolyPolygon())); } - else if ( pPolyPoly ) + else if(rDeviceRegion.getPolyPolygon()) { - aRegion = Region( PixelToLogic( *pPolyPoly ) ); + aRegion = Region(PixelToLogic(*rDeviceRegion.getPolyPolygon())); } - else + else if(rDeviceRegion.getRegionBand()) { - long nX; - long nY; - long nWidth; - long nHeight; - ImplRegionInfo aInfo; - sal_Bool bRegionRect; + RectangleVector aRectangles; + rDeviceRegion.GetRegionRectangles(aRectangles); + const RectangleVector& rRectangles(aRectangles); // needed to make the '!=' work - aRegion.ImplBeginAddRect(); - bRegionRect = rDeviceRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); - while ( bRegionRect ) + // make reverse run to fill new region bottom-up, this will speed it up due to the used data structuring + for(RectangleVector::const_reverse_iterator aRectIter(rRectangles.rbegin()); aRectIter != rRectangles.rend(); aRectIter++) { - Rectangle aRect( Point( nX, nY ), Size( nWidth, nHeight ) ); - aRegion.ImplAddRect( PixelToLogic( aRect ) ); - bRegionRect = rDeviceRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + aRegion.Union(PixelToLogic(*aRectIter)); } - aRegion.ImplEndAddRect(); + + //long nX(0); + //long nY(0); + //long nWidth(0); + //long nHeight(0); + //ImplRegionInfo aInfo; + //aRegion.ImplBeginAddRect(); + //bool bRegionRect(rDeviceRegion.ImplGetFirstRect(aInfo, nX, nY, nWidth, nHeight)); + // + //while(bRegionRect) + //{ + // const Rectangle aRect(Point(nX, nY), Size(nWidth, nHeight)); + // aRegion.ImplAddRect(PixelToLogic(aRect)); + // bRegionRect = rDeviceRegion.ImplGetNextRect(aInfo, nX, nY, nWidth, nHeight); + //} + // + //aRegion.ImplEndAddRect(); } return aRegion; @@ -1589,6 +1704,27 @@ Polygon OutputDevice::PixelToLogic( const Polygon& rDevicePoly, // ----------------------------------------------------------------------- +PolyPolygon OutputDevice::PixelToLogic( const PolyPolygon& rDevicePolyPoly, + const MapMode& rMapMode ) const +{ + DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); + DBG_CHKOBJ( &rDevicePolyPoly, PolyPolygon, NULL ); + + if ( rMapMode.IsDefault() ) + return rDevicePolyPoly; + + PolyPolygon aPolyPoly( rDevicePolyPoly ); + sal_uInt16 nPoly = aPolyPoly.Count(); + for( sal_uInt16 i = 0; i < nPoly; i++ ) + { + Polygon& rPoly = aPolyPoly[i]; + rPoly = PixelToLogic( rPoly, rMapMode ); + } + return aPolyPoly; +} + +// ----------------------------------------------------------------------- + basegfx::B2DPolygon OutputDevice::PixelToLogic( const basegfx::B2DPolygon& rPixelPoly, const MapMode& rMapMode ) const { @@ -1611,6 +1747,60 @@ basegfx::B2DPolyPolygon OutputDevice::PixelToLogic( const basegfx::B2DPolyPolygo // ----------------------------------------------------------------------- +Region OutputDevice::PixelToLogic( const Region& rDeviceRegion, const MapMode& rMapMode ) const +{ + DBG_CHKTHIS( OutputDevice, ImplDbgCheckOutputDevice ); + + if(rMapMode.IsDefault() || rDeviceRegion.IsNull() || rDeviceRegion.IsEmpty()) + { + return rDeviceRegion; + } + + Region aRegion; + + if(rDeviceRegion.getB2DPolyPolygon()) + { + aRegion = Region(PixelToLogic(*rDeviceRegion.getB2DPolyPolygon(), rMapMode)); + } + else if(rDeviceRegion.getPolyPolygon()) + { + aRegion = Region(PixelToLogic(*rDeviceRegion.getPolyPolygon(), rMapMode)); + } + else if(rDeviceRegion.getRegionBand()) + { + RectangleVector aRectangles; + rDeviceRegion.GetRegionRectangles(aRectangles); + const RectangleVector& rRectangles(aRectangles); // needed to make the '!=' work + + // make reverse run to fill new region bottom-up, this will speed it up due to the used data structuring + for(RectangleVector::const_reverse_iterator aRectIter(rRectangles.rbegin()); aRectIter != rRectangles.rend(); aRectIter++) + { + aRegion.Union(PixelToLogic(*aRectIter, rMapMode)); + } + + //long nX(0); + //long nY(0); + //long nWidth(0); + //long nHeight(0); + //ImplRegionInfo aInfo; + //aRegion.ImplBeginAddRect(); + //bool bRegionRect(rDeviceRegion.ImplGetFirstRect(aInfo, nX, nY, nWidth, nHeight)); + // + //while(bRegionRect) + //{ + // const Rectangle aRect(Point(nX, nY), Size(nWidth, nHeight)); + // aRegion.ImplAddRect(PixelToLogic(aRect, rMapMode)); + // bRegionRect = rDeviceRegion.ImplGetNextRect(aInfo, nX, nY, nWidth, nHeight); + //} + // + //aRegion.ImplEndAddRect(); + } + + return aRegion; +} + +// ----------------------------------------------------------------------- + #define ENTER0( rSource, pMapModeSource, pMapModeDest ) \ if ( !pMapModeSource ) \ pMapModeSource = &maMapMode; \ diff --git a/vcl/source/gdi/pdfwriter_impl2.cxx b/vcl/source/gdi/pdfwriter_impl2.cxx index 4cae63095920..56765434b943 100644 --- a/vcl/source/gdi/pdfwriter_impl2.cxx +++ b/vcl/source/gdi/pdfwriter_impl2.cxx @@ -879,7 +879,7 @@ void PDFWriterImpl::playMetafile( const GDIMetaFile& i_rMtf, vcl::PDFExtOutDevDa else { Region aReg( pA->GetRegion() ); - m_rOuterFace.SetClipRegion( aReg.ConvertToB2DPolyPolygon() ); + m_rOuterFace.SetClipRegion( aReg.GetAsB2DPolyPolygon() ); } } else @@ -898,7 +898,7 @@ void PDFWriterImpl::playMetafile( const GDIMetaFile& i_rMtf, vcl::PDFExtOutDevDa { const MetaISectRegionClipRegionAction* pA = (const MetaISectRegionClipRegionAction*) pAction; Region aReg( pA->GetRegion() ); - m_rOuterFace.IntersectClipRegion( aReg.ConvertToB2DPolyPolygon() ); + m_rOuterFace.IntersectClipRegion( aReg.GetAsB2DPolyPolygon() ); } break; diff --git a/vcl/source/gdi/regband.cxx b/vcl/source/gdi/regband.cxx index 5bd481b6f3de..a5a28fac6caa 100644 --- a/vcl/source/gdi/regband.cxx +++ b/vcl/source/gdi/regband.cxx @@ -54,7 +54,7 @@ ImplRegionBand::ImplRegionBand( long nTop, long nBottom ) mpPrevBand = NULL; mpFirstSep = NULL; mpFirstBandPoint = NULL; - mbTouched = sal_False; + mbTouched = false; } // ----------------------------------------------------------------------- @@ -202,8 +202,8 @@ void ImplRegionBand::ProcessPoints() // generate separations from lines and process union with existing // separations -sal_Bool ImplRegionBand::InsertPoint( long nX, long nLineId, - sal_Bool bEndPoint, LineType eLineType ) +bool ImplRegionBand::InsertPoint( long nX, long nLineId, + bool bEndPoint, LineType eLineType ) { if ( !mpFirstBandPoint ) { @@ -213,7 +213,7 @@ sal_Bool ImplRegionBand::InsertPoint( long nX, long nLineId, mpFirstBandPoint->mbEndPoint = bEndPoint; mpFirstBandPoint->meLineType = eLineType; mpFirstBandPoint->mpNextBandPoint = NULL; - return sal_True; + return true; } // look if line already touched the band @@ -232,8 +232,8 @@ sal_Bool ImplRegionBand::InsertPoint( long nX, long nLineId, { // if we've only got one point => replace first point pRegionBandPoint->mnX = nX; - pRegionBandPoint->mbEndPoint = sal_True; - return sal_True; + pRegionBandPoint->mbEndPoint = true; + return true; } else { @@ -257,7 +257,7 @@ sal_Bool ImplRegionBand::InsertPoint( long nX, long nLineId, } } else - return sal_False; + return false; } // use next element @@ -288,7 +288,7 @@ sal_Bool ImplRegionBand::InsertPoint( long nX, long nLineId, else pLastTestedRegionBandPoint->mpNextBandPoint = pNewRegionBandPoint; - return sal_True; + return true; } // use next element @@ -307,7 +307,7 @@ sal_Bool ImplRegionBand::InsertPoint( long nX, long nLineId, // connections to the new point pLastTestedRegionBandPoint->mpNextBandPoint = pNewRegionBandPoint; - return sal_True; + return true; } // ----------------------------------------------------------------------- @@ -341,7 +341,7 @@ void ImplRegionBand::ScaleX( double fHorzScale ) // // combine overlaping sparations -sal_Bool ImplRegionBand::OptimizeBand() +bool ImplRegionBand::OptimizeBand() { ImplRegionBandSep* pPrevSep = 0; ImplRegionBandSep* pSep = mpFirstSep; @@ -379,7 +379,7 @@ sal_Bool ImplRegionBand::OptimizeBand() pSep = pSep->mpNextSep; } - return sal_True; + return true; } // ----------------------------------------------------------------------- @@ -394,7 +394,7 @@ void ImplRegionBand::Union( long nXLeft, long nXRight ) mpFirstSep = new ImplRegionBandSep; mpFirstSep->mnXLeft = nXLeft; mpFirstSep->mnXRight = nXRight; - mpFirstSep->mbRemoved = sal_False; + mpFirstSep->mbRemoved = false; mpFirstSep->mpNextSep = NULL; return; } @@ -415,7 +415,7 @@ void ImplRegionBand::Union( long nXLeft, long nXRight ) pNewSep = new ImplRegionBandSep; pNewSep->mnXLeft = nXLeft; pNewSep->mnXRight = nXRight; - pNewSep->mbRemoved = sal_False; + pNewSep->mbRemoved = false; pNewSep->mpNextSep = pSep; if ( pSep == mpFirstSep ) @@ -442,7 +442,7 @@ void ImplRegionBand::Union( long nXLeft, long nXRight ) pNewSep = new ImplRegionBandSep; pNewSep->mnXLeft = nXLeft; pNewSep->mnXRight = nXRight; - pNewSep->mbRemoved = sal_False; + pNewSep->mbRemoved = false; pSep->mpNextSep = pNewSep; pNewSep->mpNextSep = NULL; @@ -463,7 +463,7 @@ void ImplRegionBand::Intersect( long nXLeft, long nXRight ) DBG_ASSERT( nXLeft <= nXRight, "ImplRegionBand::Intersect(): nxLeft > nXRight" ); // band has been touched - mbTouched = sal_True; + mbTouched = true; // band empty? -> nothing to do if ( !mpFirstSep ) @@ -476,7 +476,7 @@ void ImplRegionBand::Intersect( long nXLeft, long nXRight ) // new separation completely outside? -> remove separation if ( (nXRight < pSep->mnXLeft) || (nXLeft > pSep->mnXRight) ) // will be removed from the optimizer - pSep->mbRemoved = sal_True; + pSep->mbRemoved = true; // new separation overlaping from left? -> reduce right boundary if ( (nXLeft <= pSep->mnXLeft) && @@ -510,7 +510,7 @@ void ImplRegionBand::Exclude( long nXLeft, long nXRight ) DBG_ASSERT( nXLeft <= nXRight, "ImplRegionBand::Exclude(): nxLeft > nXRight" ); // band has been touched - mbTouched = sal_True; + mbTouched = true; // band empty? -> nothing to do if ( !mpFirstSep ) @@ -522,14 +522,14 @@ void ImplRegionBand::Exclude( long nXLeft, long nXRight ) ImplRegionBandSep* pSep = mpFirstSep; while ( pSep ) { - sal_Bool bSepProcessed = sal_False; + bool bSepProcessed = false; // new separation completely overlapping? -> remove separation if ( (nXLeft <= pSep->mnXLeft) && (nXRight >= pSep->mnXRight) ) { // will be removed from the optimizer - pSep->mbRemoved = sal_True; - bSepProcessed = sal_True; + pSep->mbRemoved = true; + bSepProcessed = true; } // new separation overlaping from left? -> reduce boundary @@ -538,7 +538,7 @@ void ImplRegionBand::Exclude( long nXLeft, long nXRight ) if ( (nXRight >= pSep->mnXLeft) && (nXLeft <= pSep->mnXLeft) ) { pSep->mnXLeft = nXRight+1; - bSepProcessed = sal_True; + bSepProcessed = true; } } @@ -548,7 +548,7 @@ void ImplRegionBand::Exclude( long nXLeft, long nXRight ) if ( (nXLeft <= pSep->mnXRight) && (nXRight > pSep->mnXRight) ) { pSep->mnXRight = nXLeft-1; - bSepProcessed = sal_True; + bSepProcessed = true; } } @@ -561,7 +561,7 @@ void ImplRegionBand::Exclude( long nXLeft, long nXRight ) pNewSep = new ImplRegionBandSep; pNewSep->mnXLeft = pSep->mnXLeft; pNewSep->mnXRight = nXLeft-1; - pNewSep->mbRemoved = sal_False; + pNewSep->mbRemoved = false; pSep->mnXLeft = nXRight+1; @@ -641,7 +641,7 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) mpFirstSep = new ImplRegionBandSep; mpFirstSep->mnXLeft = nXLeft; mpFirstSep->mnXRight = nXRight; - mpFirstSep->mbRemoved = sal_False; + mpFirstSep->mbRemoved = false; mpFirstSep->mpNextSep = NULL; return; } @@ -670,7 +670,7 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) pNewSep->mnXLeft = nXLeft; pNewSep->mnXRight = nXRight; pNewSep->mpNextSep = pSep; - pNewSep->mbRemoved = sal_False; + pNewSep->mbRemoved = false; // connections from the new separation pNewSep->mpNextSep = pSep; @@ -686,7 +686,7 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) else if( nXLeft == nOldLeft && nXRight == nOldRight ) { // #3 - pSep->mbRemoved = sal_True; + pSep->mbRemoved = true; pPrevSep = NULL; // do not run accidentally into the "right" case when breaking the loop break; } @@ -719,7 +719,7 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) // cannot break here, simply mark segment as removed, // and go on with adapted nXLeft/nXRight - pSep->mbRemoved = sal_True; + pSep->mbRemoved = true; } else { @@ -799,7 +799,7 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) pNewSep->mnXLeft = nXLeft; pNewSep->mnXRight = nXRight; pNewSep->mpNextSep = pSep->mpNextSep; - pNewSep->mbRemoved = sal_False; + pNewSep->mbRemoved = false; // connections from the new separation pSep->mpNextSep = pNewSep; @@ -821,7 +821,7 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) pNewSep->mnXLeft = nXLeft; pNewSep->mnXRight = nXRight; pNewSep->mpNextSep = NULL; - pNewSep->mbRemoved = sal_False; + pNewSep->mbRemoved = false; // connections from the new separation pPrevSep->mpNextSep = pNewSep; @@ -832,18 +832,50 @@ void ImplRegionBand::XOr( long nXLeft, long nXRight ) // ----------------------------------------------------------------------- -sal_Bool ImplRegionBand::IsInside( long nX ) +bool ImplRegionBand::IsInside( long nX ) { ImplRegionBandSep* pSep = mpFirstSep; while ( pSep ) { if ( (pSep->mnXLeft <= nX) && (pSep->mnXRight >= nX) ) - return sal_True; + return true; pSep = pSep->mpNextSep; } - return sal_False; + return false; +} + +// ----------------------------------------------------------------------- + +bool ImplRegionBand::IsOver( long nLeft, long nRight ) +{ + ImplRegionBandSep* pSep = mpFirstSep; + while ( pSep ) + { + if ( (pSep->mnXLeft < nRight) && (pSep->mnXRight > nLeft) ) + return true; + + pSep = pSep->mpNextSep; + } + + return false; +} + +// ----------------------------------------------------------------------- + +bool ImplRegionBand::IsInside( long nLeft, long nRight ) +{ + ImplRegionBandSep* pSep = mpFirstSep; + while ( pSep ) + { + if ( (pSep->mnXLeft >= nLeft) && (nRight <= pSep->mnXRight) ) + return true; + + pSep = pSep->mpNextSep; + } + + return false; } // ----------------------------------------------------------------------- @@ -870,7 +902,7 @@ long ImplRegionBand::GetXRightBoundary() const // ----------------------------------------------------------------------- -sal_Bool ImplRegionBand::operator==( const ImplRegionBand& rRegionBand ) const +bool ImplRegionBand::operator==( const ImplRegionBand& rRegionBand ) const { ImplRegionBandSep* pOwnRectBandSep = mpFirstSep; ImplRegionBandSep* pSecondRectBandSep = rRegionBand.mpFirstSep; @@ -880,12 +912,12 @@ sal_Bool ImplRegionBand::operator==( const ImplRegionBand& rRegionBand ) const long nOwnXLeft = pOwnRectBandSep->mnXLeft; long nSecondXLeft = pSecondRectBandSep->mnXLeft; if ( nOwnXLeft != nSecondXLeft ) - return sal_False; + return false; long nOwnXRight = pOwnRectBandSep->mnXRight; long nSecondXRight = pSecondRectBandSep->mnXRight; if ( nOwnXRight != nSecondXRight ) - return sal_False; + return false; // get next separation from current band pOwnRectBandSep = pOwnRectBandSep->mpNextSep; @@ -896,9 +928,9 @@ sal_Bool ImplRegionBand::operator==( const ImplRegionBand& rRegionBand ) const // different number of separations? if ( pOwnRectBandSep || pSecondRectBandSep ) - return sal_False; + return false; - return sal_True; + return true; } // ----------------------------------------------------------------------- diff --git a/vcl/source/gdi/region.cxx b/vcl/source/gdi/region.cxx index ad33c97cb9a1..fd91eb69b615 100644 --- a/vcl/source/gdi/region.cxx +++ b/vcl/source/gdi/region.cxx @@ -18,15 +18,11 @@ */ #include <limits.h> - #include <tools/vcompat.hxx> #include <tools/stream.hxx> -#include <tools/debug.hxx> #include <tools/helpers.hxx> #include <vcl/region.hxx> -#include <vcl/regband.hxx> - -#include <region.h> +#include <regionband.hxx> #include <basegfx/matrix/b2dhommatrix.hxx> #include <basegfx/polygon/b2dpolypolygontools.hxx> @@ -36,2033 +32,1238 @@ #include <basegfx/range/b2drange.hxx> #include <basegfx/matrix/b2dhommatrixtools.hxx> -// ======================================================================= -// -// ImplRegionBand -// -// Die Klassen RegionBand/ImplRegionBand speichert Regionen in Form von -// Rechtecken ab. Die Region ist in Y-Richtung in Baendern unterteilt, die -// wiederum ein oder mehrere Rechtecke mit der Hoehe des Bandes enthalten. -// -// Leere Baender werden entfernt. -// -// Polygone und PolyPolygone werden ebenfalls in Rechtecke zerlegt und in -// der Baendern abgelegt. Hierzu werden alle Punkte der einzelnen Polygone -// mit dem Bresenham-Algorithmus berechnet und in die Baender eingetragen. -// Nach der vollstaendigen Berechung aller Kanten werden die entsprechenden -// Rechntecke berechnet - -// ======================================================================= - -static ImplRegionBase aImplNullRegion( 0 ); -static ImplRegionBase aImplEmptyRegion( 0 ); - -// ======================================================================= +////////////////////////////////////////////////////////////////////////////// DBG_NAME( Region ) DBG_NAMEEX( Polygon ) DBG_NAMEEX( PolyPolygon ) -namespace { +////////////////////////////////////////////////////////////////////////////// -/** Return <TRUE/> when the given polygon is rectiliner and oriented so that - all sides are either horizontal or vertical. -*/ -bool ImplIsPolygonRectilinear (const PolyPolygon& rPolyPoly) +namespace { - // Iterate over all polygons. - const sal_uInt16 nPolyCount = rPolyPoly.Count(); - for (sal_uInt16 nPoly = 0; nPoly < nPolyCount; ++nPoly) + /** Return <TRUE/> when the given polygon is rectiliner and oriented so that + all sides are either horizontal or vertical. + */ + bool ImplIsPolygonRectilinear (const PolyPolygon& rPolyPoly) { - const Polygon& aPoly = rPolyPoly.GetObject(nPoly); - - // Iterate over all edges of the current polygon. - const sal_uInt16 nSize = aPoly.GetSize(); - - if (nSize < 2) - continue; - Point aPoint (aPoly.GetPoint(0)); - const Point aLastPoint (aPoint); - for (sal_uInt16 nPoint = 1; nPoint < nSize; ++nPoint) - { - const Point aNextPoint (aPoly.GetPoint(nPoint)); - // When there is at least one edge that is neither vertical nor - // horizontal then the entire polygon is not rectilinear (and - // oriented along primary axes.) - if (aPoint.X() != aNextPoint.X() && aPoint.Y() != aNextPoint.Y()) - return false; - - aPoint = aNextPoint; - } - // Compare closing edge. - if (aLastPoint.X() != aPoint.X() && aLastPoint.Y() != aPoint.Y()) - return false; - } - return true; -} - - - -/** This function is similar to the ImplRegion::InsertBands() method. - It creates a minimal set of missing bands so that the entire vertical - interval from nTop to nBottom is covered by bands. -*/ -void ImplAddMissingBands ( - ImplRegion* pImplRegion, - const long nTop, - const long nBottom) -{ - // Iterate over already existing bands and add missing bands atop the - // first and between two bands. - ImplRegionBand* pPreviousBand = NULL; - ImplRegionBand* pBand = pImplRegion->ImplGetFirstRegionBand(); - long nCurrentTop (nTop); - while (pBand != NULL && nCurrentTop<nBottom) - { - if (nCurrentTop < pBand->mnYTop) + // Iterate over all polygons. + const sal_uInt16 nPolyCount = rPolyPoly.Count(); + for (sal_uInt16 nPoly = 0; nPoly < nPolyCount; ++nPoly) { - // Create new band above the current band. - ImplRegionBand* pAboveBand = new ImplRegionBand( - nCurrentTop, - ::std::min(nBottom,pBand->mnYTop-1)); - pImplRegion->InsertBand(pPreviousBand, pAboveBand); - } + const Polygon& aPoly = rPolyPoly.GetObject(nPoly); - // Adapt the top of the interval to prevent overlapping bands. - nCurrentTop = ::std::max(nTop, pBand->mnYBottom+1); + // Iterate over all edges of the current polygon. + const sal_uInt16 nSize = aPoly.GetSize(); - // Advance to next band. - pPreviousBand = pBand; - pBand = pBand->mpNextBand; - } - - // We still have to cover two cases: - // 1. The region does not yet contain any bands. - // 2. The intervall nTop->nBottom extends past the bottom most band. - if (nCurrentTop <= nBottom - && (pBand==NULL || nBottom>pBand->mnYBottom)) - { - // When there is no previous band then the new one will be the - // first. Otherwise the new band is inserted behind the last band. - pImplRegion->InsertBand( - pPreviousBand, - new ImplRegionBand( - nCurrentTop, - nBottom)); - } -} - - - -/** Convert a rectilinear polygon (that is oriented along the primary axes) - to a list of bands. For this special form of polygon we can use an - optimization that prevents the creation of one band per y value. - However, it still is possible that some temporary bands are created that - later can be optimized away. - @param rPolyPolygon - A set of zero, one, or more polygons, nested or not, that are - converted into a list of bands. - @return - A new ImplRegion object is returned that contains the bands that - represent the given poly-polygon. -*/ -ImplRegion* ImplRectilinearPolygonToBands (const PolyPolygon& rPolyPoly) -{ - OSL_ASSERT(ImplIsPolygonRectilinear (rPolyPoly)); - - // Create a new ImplRegion object as container of the bands. - ImplRegion* pImplRegion = new ImplRegion(); - long nLineId = 0L; - - // Iterate over all polygons. - const sal_uInt16 nPolyCount = rPolyPoly.Count(); - for (sal_uInt16 nPoly = 0; nPoly < nPolyCount; ++nPoly) - { - const Polygon& aPoly = rPolyPoly.GetObject(nPoly); - - // Iterate over all edges of the current polygon. - const sal_uInt16 nSize = aPoly.GetSize(); - if (nSize < 2) - continue; - // Avoid fetching every point twice (each point is the start point - // of one and the end point of another edge.) - Point aStart (aPoly.GetPoint(0)); - Point aEnd; - for (sal_uInt16 nPoint = 1; nPoint <= nSize; ++nPoint, aStart=aEnd) - { - // We take the implicit closing edge into account by mapping - // index nSize to 0. - aEnd = aPoly.GetPoint(nPoint%nSize); - if (aStart.Y() == aEnd.Y()) - { - // Horizontal lines are ignored. + if (nSize < 2) continue; - } - - // At this point the line has to be vertical. - OSL_ASSERT(aStart.X() == aEnd.X()); - - // Sort y-coordinates to simplify the algorithm and store the - // direction separately. The direction is calculated as it is - // in other places (but seems to be the wrong way.) - const long nTop (::std::min(aStart.Y(), aEnd.Y())); - const long nBottom (::std::max(aStart.Y(), aEnd.Y())); - const LineType eLineType (aStart.Y() > aEnd.Y() ? LINE_DESCENDING : LINE_ASCENDING); - - // Make sure that the current line is covered by bands. - ImplAddMissingBands(pImplRegion, nTop,nBottom); - - // Find top-most band that may contain nTop. - ImplRegionBand* pBand = pImplRegion->ImplGetFirstRegionBand(); - while (pBand!=NULL && pBand->mnYBottom < nTop) - pBand = pBand->mpNextBand; - ImplRegionBand* pTopBand = pBand; - // If necessary split the band at nTop so that nTop is contained - // in the lower band. - if (pBand!=NULL - // Prevent the current band from becoming 0 pixel high - && pBand->mnYTop<nTop - // this allows the lowest pixel of the band to be split off - && pBand->mnYBottom>=nTop - // do not split a band that is just one pixel high - && pBand->mnYTop<pBand->mnYBottom) - { - // Split the top band. - pTopBand = pBand->SplitBand(nTop); - } - - // Advance to band that may contain nBottom. - while (pBand!=NULL && pBand->mnYBottom < nBottom) - pBand = pBand->mpNextBand; - // The lowest band may have to be split at nBottom so that - // nBottom itself remains in the upper band. - if (pBand!=NULL - // allow the current band becoming 1 pixel high - && pBand->mnYTop<=nBottom - // prevent splitting off a band that is 0 pixel high - && pBand->mnYBottom>nBottom - // do not split a band that is just one pixel high - && pBand->mnYTop<pBand->mnYBottom) + Point aPoint (aPoly.GetPoint(0)); + const Point aLastPoint (aPoint); + for (sal_uInt16 nPoint = 1; nPoint < nSize; ++nPoint) { - // Split the bottom band. - pBand->SplitBand(nBottom+1); + const Point aNextPoint (aPoly.GetPoint(nPoint)); + // When there is at least one edge that is neither vertical nor + // horizontal then the entire polygon is not rectilinear (and + // oriented along primary axes.) + if (aPoint.X() != aNextPoint.X() && aPoint.Y() != aNextPoint.Y()) + return false; + + aPoint = aNextPoint; } - - // Note that we remember the top band (in pTopBand) but not the - // bottom band. The later can be determined by comparing y - // coordinates. - - // Add the x-value as point to all bands in the nTop->nBottom range. - for (pBand=pTopBand; pBand!=NULL&&pBand->mnYTop<=nBottom; pBand=pBand->mpNextBand) - pBand->InsertPoint(aStart.X(), nLineId++, sal_True, eLineType); + // Compare closing edge. + if (aLastPoint.X() != aPoint.X() && aLastPoint.Y() != aPoint.Y()) + return false; } + return true; } - return pImplRegion; -} - - - - -/** Convert a general polygon (one for which ImplIsPolygonRectilinear() - returns <FALSE/>) to bands. -*/ -ImplRegion* ImplGeneralPolygonToBands ( - const PolyPolygon& rPolyPoly, - const Rectangle& rPolygonBoundingBox) -{ - long nLineID = 0L; - - // initialisation and creation of Bands - ImplRegion* pImplRegion = new ImplRegion(); - pImplRegion->CreateBandRange( rPolygonBoundingBox.Top(), rPolygonBoundingBox.Bottom() ); - - // insert polygons - const sal_uInt16 nPolyCount = rPolyPoly.Count(); - for ( sal_uInt16 nPoly = 0; nPoly < nPolyCount; nPoly++ ) - { - // get reference to current polygon - const Polygon& aPoly = rPolyPoly.GetObject( nPoly ); - const sal_uInt16 nSize = aPoly.GetSize(); - - // not enough points ( <= 2 )? -> nothing to do! - if ( nSize <= 2 ) - continue; - - // band the polygon - for ( sal_uInt16 nPoint = 1; nPoint < nSize; nPoint++ ) - pImplRegion->InsertLine( aPoly.GetPoint(nPoint-1), aPoly.GetPoint(nPoint), nLineID++ ); - - // close polygon with line from first point to last point, if neccesary - const Point rLastPoint = aPoly.GetPoint(nSize-1); - const Point rFirstPoint = aPoly.GetPoint(0); - if ( rLastPoint != rFirstPoint ) - pImplRegion->InsertLine( rLastPoint, rFirstPoint, nLineID++ ); - } - - return pImplRegion; -} - - -} // end of anonymous namespace - - -// ----------------------------------------------------------------------- - -#ifdef DBG_UTIL -const char* ImplDbgTestRegion( const void* pObj ) -{ - Region* pRegion = (Region*)pObj; - ImplRegion* pImplRegion = pRegion->ImplGetImplRegion(); - - if ( aImplNullRegion.mnRefCount ) - return "Null-Region-RefCount modified"; - if ( aImplNullRegion.mnRectCount ) - return "Null-Region-RectCount modified"; - if ( aImplNullRegion.mpPolyPoly ) - return "Null-Region-PolyPoly modified"; - if ( aImplEmptyRegion.mnRefCount ) - return "Emptry-Region-RefCount modified"; - if ( aImplEmptyRegion.mnRectCount ) - return "Emptry-Region-RectCount modified"; - if ( aImplEmptyRegion.mpPolyPoly ) - return "Emptry-Region-PolyPoly modified"; - - if ( (pImplRegion != &aImplEmptyRegion) && (pImplRegion != &aImplNullRegion) ) - { - sal_uLong nCount = 0; - const ImplRegionBand* pBand = pImplRegion->ImplGetFirstRegionBand(); - while ( pBand ) + /** Convert a rectilinear polygon (that is oriented along the primary axes) + to a list of bands. For this special form of polygon we can use an + optimization that prevents the creation of one band per y value. + However, it still is possible that some temporary bands are created that + later can be optimized away. + @param rPolyPolygon + A set of zero, one, or more polygons, nested or not, that are + converted into a list of bands. + @return + A new RegionBand object is returned that contains the bands that + represent the given poly-polygon. + */ + RegionBand* ImplRectilinearPolygonToBands(const PolyPolygon& rPolyPoly) + { + OSL_ASSERT(ImplIsPolygonRectilinear (rPolyPoly)); + + // Create a new RegionBand object as container of the bands. + RegionBand* pRegionBand = new RegionBand(); + long nLineId = 0L; + + // Iterate over all polygons. + const sal_uInt16 nPolyCount = rPolyPoly.Count(); + for (sal_uInt16 nPoly = 0; nPoly < nPolyCount; ++nPoly) { - if ( pBand->mnYBottom < pBand->mnYTop ) - return "YBottom < YTop"; - if ( pBand->mpNextBand ) - { - if ( pBand->mnYBottom >= pBand->mpNextBand->mnYTop ) - return "overlapping bands in region"; - } - if ( pBand->mbTouched > 1 ) - return "Band-mbTouched overwrite"; + const Polygon& aPoly = rPolyPoly.GetObject(nPoly); - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) + // Iterate over all edges of the current polygon. + const sal_uInt16 nSize = aPoly.GetSize(); + if (nSize < 2) + continue; + // Avoid fetching every point twice (each point is the start point + // of one and the end point of another edge.) + Point aStart (aPoly.GetPoint(0)); + Point aEnd; + for (sal_uInt16 nPoint = 1; nPoint <= nSize; ++nPoint, aStart=aEnd) { - if ( pSep->mnXRight < pSep->mnXLeft ) - return "XLeft < XRight"; - if ( pSep->mpNextSep ) + // We take the implicit closing edge into account by mapping + // index nSize to 0. + aEnd = aPoly.GetPoint(nPoint%nSize); + if (aStart.Y() == aEnd.Y()) { - if ( pSep->mnXRight >= pSep->mpNextSep->mnXLeft ) - return "overlapping separations in region"; + // Horizontal lines are ignored. + continue; } - if ( pSep->mbRemoved > 1 ) - return "Sep-mbRemoved overwrite"; - nCount++; - pSep = pSep->mpNextSep; - } + // At this point the line has to be vertical. + OSL_ASSERT(aStart.X() == aEnd.X()); - pBand = pBand->mpNextBand; - } + // Sort y-coordinates to simplify the algorithm and store the + // direction seperately. The direction is calculated as it is + // in other places (but seems to be the wrong way.) + const long nTop (::std::min(aStart.Y(), aEnd.Y())); + const long nBottom (::std::max(aStart.Y(), aEnd.Y())); + const LineType eLineType (aStart.Y() > aEnd.Y() ? LINE_DESCENDING : LINE_ASCENDING); - if ( pImplRegion->mnRectCount != nCount ) - return "mnRetCount is not valid"; - } + // Make sure that the current line is covered by bands. + pRegionBand->ImplAddMissingBands(nTop,nBottom); - return NULL; -} - -#endif - -// ======================================================================= - -inline void Region::ImplPolyPolyRegionToBandRegion() -{ - if( mpImplRegion->mpPolyPoly || mpImplRegion->mpB2DPolyPoly ) - ImplPolyPolyRegionToBandRegionFunc(); -} - -// ======================================================================= - -ImplRegionBase::ImplRegionBase( int nRefCount ) -: mnRefCount( nRefCount ) -, mnRectCount( 0 ) -, mpPolyPoly( NULL ) -, mpB2DPolyPoly( NULL ) -{} - -// ------------------------------------------------------------------------ - -ImplRegion::ImplRegion() -{ - mpFirstBand = NULL; - mpLastCheckedBand = NULL; -} - -// ------------------------------------------------------------------------ - -ImplRegion::ImplRegion( const PolyPolygon& rPolyPoly ) -{ - mpFirstBand = NULL; - mpLastCheckedBand = NULL; - mpPolyPoly = new PolyPolygon( rPolyPoly ); -} - -// ------------------------------------------------------------------------ - -ImplRegion::ImplRegion( const basegfx::B2DPolyPolygon& rPolyPoly ) -{ - mpFirstBand = NULL; - mpLastCheckedBand = NULL; - mpB2DPolyPoly = new basegfx::B2DPolyPolygon( rPolyPoly ); -} - -// ----------------------------------------------------------------------- - -ImplRegion::ImplRegion( const ImplRegion& rImplRegion ) -: ImplRegionBase() -{ - mpFirstBand = NULL; - mpLastCheckedBand = NULL; - mnRectCount = rImplRegion.mnRectCount; - - if ( rImplRegion.mpPolyPoly ) - mpPolyPoly = new PolyPolygon( *rImplRegion.mpPolyPoly ); - else if( rImplRegion.mpB2DPolyPoly ) - mpB2DPolyPoly = new basegfx::B2DPolyPolygon( *rImplRegion.mpB2DPolyPoly ); - - // insert band(s) into the list - ImplRegionBand* pNewBand; - ImplRegionBand* pPrevBand = 0; - ImplRegionBand* pBand = rImplRegion.mpFirstBand; - while ( pBand ) - { - pNewBand = new ImplRegionBand( *pBand ); - - // first element? -> set as first into the list - if ( pBand == rImplRegion.mpFirstBand ) - mpFirstBand = pNewBand; - else - pPrevBand->mpNextBand = pNewBand; - - pPrevBand = pNewBand; - pBand = pBand->mpNextBand; - } -} - -// ----------------------------------------------------------------------- - -ImplRegion::~ImplRegion() -{ - DBG_ASSERT( (this != &aImplEmptyRegion) && (this != &aImplNullRegion), - "ImplRegion::~ImplRegion() - Empty oder NULL-Region" ); - - ImplRegionBand* pBand = mpFirstBand; - while ( pBand ) - { - ImplRegionBand* pTempBand = pBand->mpNextBand; - delete pBand; - pBand = pTempBand; - } -} - -// ----------------------------------------------------------------------- - -ImplRegionBase::~ImplRegionBase() -{ - delete mpPolyPoly; - delete mpB2DPolyPoly; -} - -// ----------------------------------------------------------------------- -// -// create complete range of bands in single steps - -void ImplRegion::CreateBandRange( long nYTop, long nYBottom ) -{ - // add top band - mpFirstBand = new ImplRegionBand( nYTop-1, nYTop-1 ); - - // begin first search from the first element - mpLastCheckedBand = mpFirstBand; - - ImplRegionBand* pBand = mpFirstBand; - for ( int i = nYTop; i <= nYBottom+1; i++ ) - { - // create new band - ImplRegionBand* pNewBand = new ImplRegionBand( i, i ); - pBand->mpNextBand = pNewBand; - if ( pBand != mpFirstBand ) - pNewBand->mpPrevBand = pBand; - - pBand = pBand->mpNextBand; - } -} - -// ----------------------------------------------------------------------- - -sal_Bool ImplRegion::InsertLine( const Point& rStartPt, const Point& rEndPt, - long nLineId ) -{ - long nX, nY; - - // lines consisting of a single point do not interest here - if ( rStartPt == rEndPt ) - return sal_True; + // Find top-most band that may contain nTop. + ImplRegionBand* pBand = pRegionBand->ImplGetFirstRegionBand(); + while (pBand!=NULL && pBand->mnYBottom < nTop) + pBand = pBand->mpNextBand; + ImplRegionBand* pTopBand = pBand; + // If necessary split the band at nTop so that nTop is contained + // in the lower band. + if (pBand!=NULL + // Prevent the current band from becoming 0 pixel high + && pBand->mnYTop<nTop + // this allows the lowest pixel of the band to be split off + && pBand->mnYBottom>=nTop + // do not split a band that is just one pixel high + && pBand->mnYTop<pBand->mnYBottom) + { + // Split the top band. + pTopBand = pBand->SplitBand(nTop); + } - LineType eLineType = (rStartPt.Y() > rEndPt.Y()) ? LINE_DESCENDING : LINE_ASCENDING; - if ( rStartPt.X() == rEndPt.X() ) - { - // vertical line - const long nEndY = rEndPt.Y(); + // Advance to band that may contain nBottom. + while (pBand!=NULL && pBand->mnYBottom < nBottom) + pBand = pBand->mpNextBand; + // The lowest band may have to be split at nBottom so that + // nBottom itself remains in the upper band. + if (pBand!=NULL + // allow the current band becoming 1 pixel high + && pBand->mnYTop<=nBottom + // prevent splitting off a band that is 0 pixel high + && pBand->mnYBottom>nBottom + // do not split a band that is just one pixel high + && pBand->mnYTop<pBand->mnYBottom) + { + // Split the bottom band. + pBand->SplitBand(nBottom+1); + } - nX = rStartPt.X(); - nY = rStartPt.Y(); + // Note that we remember the top band (in pTopBand) but not the + // bottom band. The later can be determined by comparing y + // coordinates. - if( nEndY > nY ) - { - for ( ; nY <= nEndY; nY++ ) - { - Point aNewPoint( nX, nY ); - InsertPoint( aNewPoint, nLineId, - (aNewPoint == rEndPt) || (aNewPoint == rStartPt), - eLineType ); - } - } - else - { - for ( ; nY >= nEndY; nY-- ) - { - Point aNewPoint( nX, nY ); - InsertPoint( aNewPoint, nLineId, - (aNewPoint == rEndPt) || (aNewPoint == rStartPt), - eLineType ); + // Add the x-value as point to all bands in the nTop->nBottom range. + for (pBand=pTopBand; pBand!=NULL&&pBand->mnYTop<=nBottom; pBand=pBand->mpNextBand) + pBand->InsertPoint(aStart.X(), nLineId++, true, eLineType); } } + + return pRegionBand; } - else if ( rStartPt.Y() != rEndPt.Y() ) + + /** Convert a general polygon (one for which ImplIsPolygonRectilinear() + returns <FALSE/>) to bands. + */ + RegionBand* ImplGeneralPolygonToBands(const PolyPolygon& rPolyPoly, const Rectangle& rPolygonBoundingBox) { - const long nDX = labs( rEndPt.X() - rStartPt.X() ); - const long nDY = labs( rEndPt.Y() - rStartPt.Y() ); - const long nStartX = rStartPt.X(); - const long nStartY = rStartPt.Y(); - const long nEndX = rEndPt.X(); - const long nEndY = rEndPt.Y(); - const long nXInc = ( nStartX < nEndX ) ? 1L : -1L; - const long nYInc = ( nStartY < nEndY ) ? 1L : -1L; + long nLineID = 0L; - if ( nDX >= nDY ) - { - const long nDYX = ( nDY - nDX ) << 1; - const long nDY2 = nDY << 1; - long nD = nDY2 - nDX; + // initialisation and creation of Bands + RegionBand* pRegionBand = new RegionBand(); + pRegionBand->CreateBandRange(rPolygonBoundingBox.Top(), rPolygonBoundingBox.Bottom()); - for ( nX = nStartX, nY = nStartY; nX != nEndX; nX += nXInc ) - { - InsertPoint( Point( nX, nY ), nLineId, nStartX == nX, eLineType ); + // insert polygons + const sal_uInt16 nPolyCount = rPolyPoly.Count(); - if ( nD < 0L ) - nD += nDY2; - else - nD += nDYX, nY += nYInc; - } - } - else + for ( sal_uInt16 nPoly = 0; nPoly < nPolyCount; nPoly++ ) { - const long nDYX = ( nDX - nDY ) << 1; - const long nDY2 = nDX << 1; - long nD = nDY2 - nDY; + // get reference to current polygon + const Polygon& aPoly = rPolyPoly.GetObject( nPoly ); + const sal_uInt16 nSize = aPoly.GetSize(); - for ( nX = nStartX, nY = nStartY; nY != nEndY; nY += nYInc ) - { - InsertPoint( Point( nX, nY ), nLineId, nStartY == nY, eLineType ); - - if ( nD < 0L ) - nD += nDY2; - else - nD += nDYX, nX += nXInc; - } - } - - // last point - InsertPoint( Point( nEndX, nEndY ), nLineId, sal_True, eLineType ); - } - - return sal_True; -} - -// ----------------------------------------------------------------------- -// -// search for appropriate place for the new point - -sal_Bool ImplRegion::InsertPoint( const Point &rPoint, long nLineID, - sal_Bool bEndPoint, LineType eLineType ) -{ - DBG_ASSERT( mpFirstBand != NULL, "ImplRegion::InsertPoint - no bands available!" ); - - if ( rPoint.Y() == mpLastCheckedBand->mnYTop ) - { - mpLastCheckedBand->InsertPoint( rPoint.X(), nLineID, bEndPoint, eLineType ); - return sal_True; - } + // not enough points ( <= 2 )? -> nothing to do! + if ( nSize <= 2 ) + continue; - if ( rPoint.Y() > mpLastCheckedBand->mnYTop ) - { - // Search ascending - while ( mpLastCheckedBand ) - { - // Insert point if possible - if ( rPoint.Y() == mpLastCheckedBand->mnYTop ) + // band the polygon + for ( sal_uInt16 nPoint = 1; nPoint < nSize; nPoint++ ) { - mpLastCheckedBand->InsertPoint( rPoint.X(), nLineID, bEndPoint, eLineType ); - return sal_True; + pRegionBand->InsertLine( aPoly.GetPoint(nPoint-1), aPoly.GetPoint(nPoint), nLineID++ ); } - mpLastCheckedBand = mpLastCheckedBand->mpNextBand; - } + // close polygon with line from first point to last point, if neccesary + const Point rLastPoint = aPoly.GetPoint(nSize-1); + const Point rFirstPoint = aPoly.GetPoint(0); - OSL_FAIL( "ImplRegion::InsertPoint reached the end of the list!" ); - } - else - { - // Search descending - while ( mpLastCheckedBand ) - { - // Insert point if possible - if ( rPoint.Y() == mpLastCheckedBand->mnYTop ) + if ( rLastPoint != rFirstPoint ) { - mpLastCheckedBand->InsertPoint( rPoint.X(), nLineID, bEndPoint, eLineType ); - return sal_True; + pRegionBand->InsertLine( rLastPoint, rFirstPoint, nLineID++ ); } - - mpLastCheckedBand = mpLastCheckedBand->mpPrevBand; } - OSL_FAIL( "ImplRegion::InsertPoint reached the beginning of the list!" ); + return pRegionBand; } +} // end of anonymous namespace - OSL_FAIL( "ImplRegion::InsertPoint point not inserted!" ); - - // reinitialize pointer (should never be reached!) - mpLastCheckedBand = mpFirstBand; - - return sal_False; -} - -// ----------------------------------------------------------------------- -// -// search for appropriate places for the new bands +////////////////////////////////////////////////////////////////////////////// -void ImplRegion::InsertBands( long nTop, long nBottom ) +bool Region::IsEmpty() const { - // region empty? -> set rectagle as first entry! - if ( !mpFirstBand ) - { - // add band with boundaries of the rectangle - mpFirstBand = new ImplRegionBand( nTop, nBottom ); - return; - } - - // find/insert bands for the boundaries of the rectangle - sal_Bool bTopBoundaryInserted = sal_False; - sal_Bool bTop2BoundaryInserted = sal_False; - sal_Bool bBottomBoundaryInserted = sal_False; - - // special case: top boundary is above the first band - ImplRegionBand* pNewBand; - if ( nTop < mpFirstBand->mnYTop ) - { - // create new band above the first in the list - pNewBand = new ImplRegionBand( nTop, mpFirstBand->mnYTop ); - if ( nBottom < mpFirstBand->mnYTop ) - pNewBand->mnYBottom = nBottom; - - // insert band into the list - pNewBand->mpNextBand = mpFirstBand; - mpFirstBand = pNewBand; - - bTopBoundaryInserted = sal_True; - } - - // insert band(s) into the list - ImplRegionBand* pBand = mpFirstBand; - while ( pBand ) - { - // Insert Bands if possible - if ( !bTopBoundaryInserted ) - bTopBoundaryInserted = InsertSingleBand( pBand, nTop - 1 ); - - if ( !bTop2BoundaryInserted ) - bTop2BoundaryInserted = InsertSingleBand( pBand, nTop ); - - if ( !bBottomBoundaryInserted && (nTop != nBottom) ) - bBottomBoundaryInserted = InsertSingleBand( pBand, nBottom ); - - // both boundaries inserted? -> nothing more to do - if ( bTopBoundaryInserted && bTop2BoundaryInserted && bBottomBoundaryInserted ) - break; - - // insert bands between two bands if necessary - if ( pBand->mpNextBand ) - { - if ( (pBand->mnYBottom + 1) < pBand->mpNextBand->mnYTop ) - { - // copy band with list and set new boundary - pNewBand = new ImplRegionBand( pBand->mnYBottom+1, - pBand->mpNextBand->mnYTop-1 ); - - // insert band into the list - pNewBand->mpNextBand = pBand->mpNextBand; - pBand->mpNextBand = pNewBand; - } - } - - pBand = pBand->mpNextBand; - } + return !mbIsNull && !mpB2DPolyPolygon.get() && !mpPolyPolygon.get() && !mpRegionBand.get(); } -// ----------------------------------------------------------------------- -// -// create new band and insert it into the list - -sal_Bool ImplRegion::InsertSingleBand( ImplRegionBand* pBand, - long nYBandPosition ) +bool Region::IsNull() const { - // boundary already included in band with height 1? -> nothing to do! - if ( (pBand->mnYTop == pBand->mnYBottom) && - (nYBandPosition == pBand->mnYTop) ) - return sal_True; - - // insert single height band on top? - ImplRegionBand* pNewBand; - if ( nYBandPosition == pBand->mnYTop ) - { - // copy band with list and set new boundary - pNewBand = new ImplRegionBand( *pBand ); - pNewBand->mnYTop = nYBandPosition+1; - - // insert band into the list - pNewBand->mpNextBand = pBand->mpNextBand; - pBand->mnYBottom = nYBandPosition; - pBand->mpNextBand = pNewBand; - - return sal_True; - } - - // top of new rectangle within the current band? -> insert new band and copy data - if ( (nYBandPosition > pBand->mnYTop) && - (nYBandPosition < pBand->mnYBottom) ) - { - // copy band with list and set new boundary - pNewBand = new ImplRegionBand( *pBand ); - pNewBand->mnYTop = nYBandPosition; - - // insert band into the list - pNewBand->mpNextBand = pBand->mpNextBand; - pBand->mnYBottom = nYBandPosition; - pBand->mpNextBand = pNewBand; - - // copy band with list and set new boundary - pNewBand = new ImplRegionBand( *pBand ); - pNewBand->mnYTop = nYBandPosition; - - // insert band into the list - pBand->mpNextBand->mnYTop = nYBandPosition+1; - - pNewBand->mpNextBand = pBand->mpNextBand; - pBand->mnYBottom = nYBandPosition - 1; - pBand->mpNextBand = pNewBand; - - return sal_True; - } - - // create new band behind the current in the list - if ( !pBand->mpNextBand ) - { - if ( nYBandPosition == pBand->mnYBottom ) - { - // copy band with list and set new boundary - pNewBand = new ImplRegionBand( *pBand ); - pNewBand->mnYTop = pBand->mnYBottom; - pNewBand->mnYBottom = nYBandPosition; - - pBand->mnYBottom = nYBandPosition-1; - - // append band to the list - pBand->mpNextBand = pNewBand; - return sal_True; - } - - if ( nYBandPosition > pBand->mnYBottom ) - { - // create new band - pNewBand = new ImplRegionBand( pBand->mnYBottom + 1, nYBandPosition ); - - // append band to the list - pBand->mpNextBand = pNewBand; - return sal_True; - } - } - - return sal_False; + return mbIsNull; } -// ------------------------------------------------------------------------ - -void ImplRegion::InsertBand (ImplRegionBand* pPreviousBand, ImplRegionBand* pBandToInsert) +RegionBand* ImplCreateRegionBandFromPolyPolygon(const PolyPolygon& rPolyPolygon) { - OSL_ASSERT(pBandToInsert!=NULL); + RegionBand* pRetval = 0; - if (pPreviousBand == NULL) + if(rPolyPolygon.Count()) { - // Insert band before all others. - if (mpFirstBand != NULL) - mpFirstBand->mpPrevBand = pBandToInsert; - pBandToInsert->mpNextBand = mpFirstBand; - mpFirstBand = pBandToInsert; - } - else - { - // Insert band directly after pPreviousBand. - pBandToInsert->mpNextBand = pPreviousBand->mpNextBand; - pPreviousBand->mpNextBand = pBandToInsert; - pBandToInsert->mpPrevBand = pPreviousBand; - } -} + // ensure to subdivide when bezier segemnts are used, it's going to + // be expanded to rectangles + PolyPolygon aPolyPolygon; -// ------------------------------------------------------------------------ + rPolyPolygon.AdaptiveSubdivide(aPolyPolygon); -void ImplRegion::Union( long nLeft, long nTop, long nRight, long nBottom ) -{ - DBG_ASSERT( nLeft <= nRight, "ImplRegion::Union() - nLeft > nRight" ); - DBG_ASSERT( nTop <= nBottom, "ImplRegion::Union() - nTop > nBottom" ); - - // process union - ImplRegionBand* pBand = mpFirstBand; - while ( pBand ) - { - if ( pBand->mnYTop >= nTop ) + if(aPolyPolygon.Count()) { - if ( pBand->mnYBottom <= nBottom ) - pBand->Union( nLeft, nRight ); - else + const Rectangle aRect(aPolyPolygon.GetBoundRect()); + + if(!aRect.IsEmpty()) { -#ifdef DBG_UTIL - long nCurY = pBand->mnYBottom; - pBand = pBand->mpNextBand; - while ( pBand ) + if(ImplIsPolygonRectilinear(aPolyPolygon)) { - if ( (pBand->mnYTop < nCurY) || (pBand->mnYBottom < nCurY) ) - { - OSL_FAIL( "ImplRegion::Union() - Bands not sorted!" ); - } - pBand = pBand->mpNextBand; + // For rectilinear polygons there is an optimized band conversion. + pRetval = ImplRectilinearPolygonToBands(aPolyPolygon); } -#endif - break; - } - } - - pBand = pBand->mpNextBand; - } -} - -// ----------------------------------------------------------------------- - -void ImplRegion::Exclude( long nLeft, long nTop, long nRight, long nBottom ) -{ - DBG_ASSERT( nLeft <= nRight, "ImplRegion::Exclude() - nLeft > nRight" ); - DBG_ASSERT( nTop <= nBottom, "ImplRegion::Exclude() - nTop > nBottom" ); - - // process exclude - ImplRegionBand* pBand = mpFirstBand; - while ( pBand ) - { - if ( pBand->mnYTop >= nTop ) - { - if ( pBand->mnYBottom <= nBottom ) - pBand->Exclude( nLeft, nRight ); - else - { -#ifdef DBG_UTIL - long nCurY = pBand->mnYBottom; - pBand = pBand->mpNextBand; - while ( pBand ) + else { - if ( (pBand->mnYTop < nCurY) || (pBand->mnYBottom < nCurY) ) - { - OSL_FAIL( "ImplRegion::Exclude() - Bands not sorted!" ); - } - pBand = pBand->mpNextBand; + pRetval = ImplGeneralPolygonToBands(aPolyPolygon, aRect); } -#endif - break; - } - } - pBand = pBand->mpNextBand; - } -} - -// ----------------------------------------------------------------------- - -void ImplRegion::XOr( long nLeft, long nTop, long nRight, long nBottom ) -{ - DBG_ASSERT( nLeft <= nRight, "ImplRegion::Exclude() - nLeft > nRight" ); - DBG_ASSERT( nTop <= nBottom, "ImplRegion::Exclude() - nTop > nBottom" ); - - // process xor - ImplRegionBand* pBand = mpFirstBand; - while ( pBand ) - { - if ( pBand->mnYTop >= nTop ) - { - if ( pBand->mnYBottom <= nBottom ) - pBand->XOr( nLeft, nRight ); - else - { -#ifdef DBG_UTIL - long nCurY = pBand->mnYBottom; - pBand = pBand->mpNextBand; - while ( pBand ) + // Convert points into seps. + if(pRetval) { - if ( (pBand->mnYTop < nCurY) || (pBand->mnYBottom < nCurY) ) + pRetval->processPoints(); + + // Optimize list of bands. Adjacent bands with identical lists + // of seps are joined. + if(!pRetval->OptimizeBandList()) { - OSL_FAIL( "ImplRegion::XOr() - Bands not sorted!" ); + delete pRetval; + pRetval = 0; } - pBand = pBand->mpNextBand; } -#endif - break; } } - - pBand = pBand->mpNextBand; } -} -// ----------------------------------------------------------------------- -// -// remove empty bands + return pRetval; +} -sal_Bool ImplRegion::OptimizeBandList() +PolyPolygon Region::ImplCreatePolyPolygonFromRegionBand() const { - DBG_ASSERT( (this != &aImplNullRegion) && (this != &aImplEmptyRegion), - "ImplRegion::OptimizeBandList() - Empty oder NULL-Region" ); - - mnRectCount = 0; + PolyPolygon aRetval; - ImplRegionBand* pPrevBand = 0; - ImplRegionBand* pBand = mpFirstBand; - while ( pBand ) + if(getRegionBand()) { - const sal_Bool bBTEqual = pBand->mpNextBand && - (pBand->mnYBottom == pBand->mpNextBand->mnYTop); + RectangleVector aRectangles; + GetRegionRectangles(aRectangles); - // no separation? -> remove! - if ( pBand->IsEmpty() || (bBTEqual && (pBand->mnYBottom == pBand->mnYTop)) ) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - // save pointer - ImplRegionBand* pOldBand = pBand; - - // previous element of the list - if ( pBand == mpFirstBand ) - mpFirstBand = pBand->mpNextBand; - else - pPrevBand->mpNextBand = pBand->mpNextBand; - - pBand = pBand->mpNextBand; - delete pOldBand; - } - else - { - // fixup - if ( bBTEqual ) - pBand->mnYBottom = pBand->mpNextBand->mnYTop-1; - - // this and next band with equal separations? -> combine! - if ( pBand->mpNextBand && - ((pBand->mnYBottom+1) == pBand->mpNextBand->mnYTop) && - (*pBand == *pBand->mpNextBand) ) - { - // expand current height - pBand->mnYBottom = pBand->mpNextBand->mnYBottom; - - // remove next band from list - ImplRegionBand* pDeletedBand = pBand->mpNextBand; - pBand->mpNextBand = pDeletedBand->mpNextBand; - delete pDeletedBand; - - // check band again! - } - else - { - // count rectangles within band - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) - { - mnRectCount++; - pSep = pSep->mpNextSep; - } - - pPrevBand = pBand; - pBand = pBand->mpNextBand; - } + aRetval.Insert(Polygon(*aRectIter)); } } - -#ifdef DBG_UTIL - pBand = mpFirstBand; - while ( pBand ) + else { - DBG_ASSERT( pBand->mpFirstSep != NULL, - "Exiting ImplRegion::OptimizeBandList(): empty band in region!" ); - - if ( pBand->mnYBottom < pBand->mnYTop ) - OSL_FAIL( "ImplRegion::OptimizeBandList(): YBottomBoundary < YTopBoundary" ); - - if ( pBand->mpNextBand ) - { - if ( pBand->mnYBottom >= pBand->mpNextBand->mnYTop ) - OSL_FAIL( "ImplRegion::OptimizeBandList(): overlapping bands in region!" ); - } - - pBand = pBand->mpNextBand; + OSL_ENSURE(false, "Called with no local RegionBand (!)"); } -#endif - - return (mnRectCount != 0); -} - -// ======================================================================= -void Region::ImplCopyData() -{ - mpImplRegion->mnRefCount--; - mpImplRegion = new ImplRegion( *mpImplRegion ); + return aRetval; } -// ======================================================================= - -Region::Region() +basegfx::B2DPolyPolygon Region::ImplCreateB2DPolyPolygonFromRegionBand() const { - DBG_CTOR( Region, ImplDbgTestRegion ); + PolyPolygon aPoly(ImplCreatePolyPolygonFromRegionBand()); - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + return aPoly.getB2DPolyPolygon(); } -// ----------------------------------------------------------------------- - -Region::Region( RegionType eType ) +Region::Region(bool bIsNull) +: mpB2DPolyPolygon(), + mpPolyPolygon(), + mpRegionBand(), + mbIsNull(bIsNull) { - DBG_CTOR( Region, ImplDbgTestRegion ); - DBG_ASSERT( (eType == REGION_NULL) || (eType == REGION_EMPTY), - "Region( RegionType ) - RegionType != EMPTY/NULL" ); - - if ( eType == REGION_NULL ) - mpImplRegion = (ImplRegion*)(&aImplNullRegion); - else - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); } -// ----------------------------------------------------------------------- - -Region::Region( const Rectangle& rRect ) +Region::Region(const Rectangle& rRect) +: mpB2DPolyPolygon(), + mpPolyPolygon(), + mpRegionBand(), + mbIsNull(false) { - DBG_CTOR( Region, ImplDbgTestRegion ); - - ImplCreateRectRegion( rRect ); + mpRegionBand.reset(rRect.IsEmpty() ? 0 : new RegionBand(rRect)); } -// ----------------------------------------------------------------------- - -Region::Region( const Polygon& rPolygon ) +Region::Region(const Polygon& rPolygon) +: mpB2DPolyPolygon(), + mpPolyPolygon(), + mpRegionBand(), + mbIsNull(false) { - DBG_CTOR( Region, ImplDbgTestRegion ); DBG_CHKOBJ( &rPolygon, Polygon, NULL ); - ImplCreatePolyPolyRegion( rPolygon ); + if(rPolygon.GetSize()) + { + ImplCreatePolyPolyRegion(rPolygon); + } } -// ----------------------------------------------------------------------- - -Region::Region( const PolyPolygon& rPolyPoly ) +Region::Region(const PolyPolygon& rPolyPoly) +: mpB2DPolyPolygon(), + mpPolyPolygon(), + mpRegionBand(), + mbIsNull(false) { - DBG_CTOR( Region, ImplDbgTestRegion ); DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); - ImplCreatePolyPolyRegion( rPolyPoly ); + if(rPolyPoly.Count()) + { + ImplCreatePolyPolyRegion(rPolyPoly); + } } -// ----------------------------------------------------------------------- - -Region::Region( const basegfx::B2DPolyPolygon& rPolyPoly ) +Region::Region(const basegfx::B2DPolyPolygon& rPolyPoly) +: mpB2DPolyPolygon(), + mpPolyPolygon(), + mpRegionBand(), + mbIsNull(false) { - DBG_CTOR( Region, ImplDbgTestRegion ); DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); - ImplCreatePolyPolyRegion( rPolyPoly ); + if(rPolyPoly.count()) + { + ImplCreatePolyPolyRegion(rPolyPoly); + } } -// ----------------------------------------------------------------------- - -Region::Region( const Region& rRegion ) +Region::Region(const Region& rRegion) +: mpB2DPolyPolygon(rRegion.mpB2DPolyPolygon), + mpPolyPolygon(rRegion.mpPolyPolygon), + mpRegionBand(rRegion.mpRegionBand), + mbIsNull(rRegion.mbIsNull) { - DBG_CTOR( Region, ImplDbgTestRegion ); - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - DBG_ASSERT( rRegion.mpImplRegion->mnRefCount < 0xFFFFFFFE, "Region: RefCount overflow" ); - - // copy pointer to instance of implementation - mpImplRegion = rRegion.mpImplRegion; - if ( mpImplRegion->mnRefCount ) - mpImplRegion->mnRefCount++; } -// ----------------------------------------------------------------------- - Region::~Region() { - DBG_DTOR( Region, ImplDbgTestRegion ); - - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) - { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - } } -// ----------------------------------------------------------------------- - -void Region::ImplCreateRectRegion( const Rectangle& rRect ) -{ - if ( rRect.IsEmpty() ) - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - else - { - // get justified rectangle - long nTop = std::min( rRect.Top(), rRect.Bottom() ); - long nBottom = std::max( rRect.Top(), rRect.Bottom() ); - long nLeft = std::min( rRect.Left(), rRect.Right() ); - long nRight = std::max( rRect.Left(), rRect.Right() ); - - // create instance of implementation class - mpImplRegion = new ImplRegion(); - - // add band with boundaries of the rectangle - mpImplRegion->mpFirstBand = new ImplRegionBand( nTop, nBottom ); - - // Set left and right boundaries of the band - mpImplRegion->mpFirstBand->Union( nLeft, nRight ); - mpImplRegion->mnRectCount = 1; - } -} - -// ----------------------------------------------------------------------- - void Region::ImplCreatePolyPolyRegion( const PolyPolygon& rPolyPoly ) { const sal_uInt16 nPolyCount = rPolyPoly.Count(); - if ( nPolyCount ) + + if(nPolyCount) { // polypolygon empty? -> empty region - const Rectangle aRect( rPolyPoly.GetBoundRect() ); + const Rectangle aRect(rPolyPoly.GetBoundRect()); - if ( !aRect.IsEmpty() ) + if(!aRect.IsEmpty()) { // width OR height == 1 ? => Rectangular region - if ( (aRect.GetWidth() == 1) - || (aRect.GetHeight() == 1) - || rPolyPoly.IsRect() ) + if((1 == aRect.GetWidth()) || (1 == aRect.GetHeight()) || rPolyPoly.IsRect()) { - ImplCreateRectRegion( aRect ); + mpRegionBand.reset(new RegionBand(aRect)); } else - mpImplRegion = new ImplRegion( rPolyPoly ); + { + mpPolyPolygon.reset(new PolyPolygon(rPolyPoly)); + } + + mbIsNull = false; } - else - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); } - else - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); } -// ----------------------------------------------------------------------- - void Region::ImplCreatePolyPolyRegion( const basegfx::B2DPolyPolygon& rPolyPoly ) { - if (rPolyPoly.count()==0 || rPolyPoly.getB2DRange().isEmpty()) - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - else - mpImplRegion = new ImplRegion( rPolyPoly ); -} - -// ----------------------------------------------------------------------- - -void Region::ImplPolyPolyRegionToBandRegionFunc() -{ - // ensure to subdivide when bezier segemnts are used, it's going to - // be expanded to rectangles - PolyPolygon aPolyPoly; - GetPolyPolygon().AdaptiveSubdivide(aPolyPoly); - - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - - if ( aPolyPoly.Count() ) + if(rPolyPoly.count() && !rPolyPoly.getB2DRange().isEmpty()) { - // polypolygon empty? -> empty region - const Rectangle aRect( aPolyPoly.GetBoundRect() ); - - if ( !aRect.IsEmpty() ) - { - if (ImplIsPolygonRectilinear(aPolyPoly)) - { - // For rectilinear polygons there is an optimized band conversion. - mpImplRegion = ImplRectilinearPolygonToBands(aPolyPoly); - } - else - { - mpImplRegion = ImplGeneralPolygonToBands(aPolyPoly, aRect); - } - - // Convert points into seps. - ImplRegionBand* pRegionBand = mpImplRegion->mpFirstBand; - while ( pRegionBand ) - { - // generate separations from the lines and process union - pRegionBand->ProcessPoints(); - pRegionBand = pRegionBand->mpNextBand; - } - - // Optimize list of bands. Adjacent bands with identical lists - // of seps are joined. - if ( !mpImplRegion->OptimizeBandList() ) - { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - } - } - else - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + mpB2DPolyPolygon.reset(new basegfx::B2DPolyPolygon(rPolyPoly)); + mbIsNull = false; } - else - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); } -// ----------------------------------------------------------------------- - void Region::Move( long nHorzMove, long nVertMove ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // no region data? -> nothing to do - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) + if(IsNull() || IsEmpty()) + { + // empty or null need no move return; + } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + if(!nHorzMove && !nVertMove) + { + // no move defined + return; + } - if ( mpImplRegion->mpPolyPoly ) - mpImplRegion->mpPolyPoly->Move( nHorzMove, nVertMove ); - else if( mpImplRegion->mpB2DPolyPoly ) + if(getB2DPolyPolygon()) { - mpImplRegion->mpB2DPolyPoly->transform(basegfx::tools::createTranslateB2DHomMatrix(nHorzMove, nVertMove)); + basegfx::B2DPolyPolygon aPoly(*getB2DPolyPolygon()); + + aPoly.transform(basegfx::tools::createTranslateB2DHomMatrix(nHorzMove, nVertMove)); + mpB2DPolyPolygon.reset(aPoly.count() ? new basegfx::B2DPolyPolygon(aPoly) : 0); + mpPolyPolygon.reset(); + mpRegionBand.reset(); } - else + else if(getPolyPolygon()) { - ImplRegionBand* pBand = mpImplRegion->mpFirstBand; - while ( pBand ) - { - // process the vertical move - if ( nVertMove != 0) - { - pBand->mnYTop = pBand->mnYTop + nVertMove; - pBand->mnYBottom = pBand->mnYBottom + nVertMove; - } + PolyPolygon aPoly(*getPolyPolygon()); - // process the horizontal move - if ( nHorzMove != 0) - pBand->MoveX( nHorzMove ); + aPoly.Move(nHorzMove, nVertMove); + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(aPoly.Count() ? new PolyPolygon(aPoly) : 0); + mpRegionBand.reset(); + } + else if(getRegionBand()) + { + RegionBand* pNew = new RegionBand(*getRegionBand()); - pBand = pBand->mpNextBand; - } + pNew->Move(nHorzMove, nVertMove); + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(); + mpRegionBand.reset(pNew); + } + else + { + OSL_ENSURE(false, "Region::Move error: impossible combination (!)"); } } -// ----------------------------------------------------------------------- - void Region::Scale( double fScaleX, double fScaleY ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // no region data? -> nothing to do - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) + if(IsNull() || IsEmpty()) + { + // empty or null need no scale return; + } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + if(basegfx::fTools::equalZero(fScaleX) && basegfx::fTools::equalZero(fScaleY)) + { + // no scale defined + return; + } - if ( mpImplRegion->mpPolyPoly ) - mpImplRegion->mpPolyPoly->Scale( fScaleX, fScaleY ); - else if( mpImplRegion->mpB2DPolyPoly ) + if(getB2DPolyPolygon()) { - mpImplRegion->mpB2DPolyPoly->transform(basegfx::tools::createScaleB2DHomMatrix(fScaleX, fScaleY)); + basegfx::B2DPolyPolygon aPoly(*getB2DPolyPolygon()); + + aPoly.transform(basegfx::tools::createScaleB2DHomMatrix(fScaleX, fScaleY)); + mpB2DPolyPolygon.reset(aPoly.count() ? new basegfx::B2DPolyPolygon(aPoly) : 0); + mpPolyPolygon.reset(); + mpRegionBand.reset(); } - else + else if(getPolyPolygon()) { - ImplRegionBand* pBand = mpImplRegion->mpFirstBand; - while ( pBand ) - { - // process the vertical move - if ( fScaleY != 0.0 ) - { - pBand->mnYTop = FRound( pBand->mnYTop * fScaleY ); - pBand->mnYBottom = FRound( pBand->mnYBottom * fScaleY ); - } + PolyPolygon aPoly(*getPolyPolygon()); - // process the horizontal move - if ( fScaleX != 0.0 ) - pBand->ScaleX( fScaleX ); + aPoly.Scale(fScaleX, fScaleY); + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(aPoly.Count() ? new PolyPolygon(aPoly) : 0); + mpRegionBand.reset(); + } + else if(getRegionBand()) + { + RegionBand* pNew = new RegionBand(*getRegionBand()); - pBand = pBand->mpNextBand; - } + pNew->Scale(fScaleX, fScaleY); + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(); + mpRegionBand.reset(pNew); + } + else + { + OSL_ENSURE(false, "Region::Scale error: impossible combination (!)"); } } -// ----------------------------------------------------------------------- - -void Region::Union( const Rectangle& rRect ) +bool Region::Union( const Rectangle& rRect ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(rRect.IsEmpty()) + { + // empty rectangle will not expand the existing union, nothing to do + return true; + } - // is rectangle empty? -> nothing to do - if ( rRect.IsEmpty() ) - return; + if(IsEmpty()) + { + // no local data, the union will be equal to source. Create using rectangle + *this = rRect; + return true; + } - if( HasPolyPolygon() ) + if(HasPolyPolygonOrB2DPolyPolygon()) { - // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); - aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); + // get this B2DPolyPolygon, solve on polygon base + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); - if( aThisPolyPoly.count() == 0 ) + aThisPolyPoly = basegfx::tools::prepareForPolygonOperation(aThisPolyPoly); + + if(!aThisPolyPoly.count()) { + // no local polygon, use the rectangle as new region *this = rRect; - return; + } + else + { + // get the other B2DPolyPolygon and use logical Or-Operation + const basegfx::B2DPolygon aRectPoly( + basegfx::tools::createPolygonFromRect( + basegfx::B2DRectangle( + rRect.Left(), + rRect.Top(), + rRect.Right(), + rRect.Bottom()))); + const basegfx::B2DPolyPolygon aClip( + basegfx::tools::solvePolygonOperationOr( + aThisPolyPoly, + basegfx::B2DPolyPolygon(aRectPoly))); + *this = Region(aClip); } - // get the other B2DPolyPolygon - basegfx::B2DPolygon aRectPoly( basegfx::tools::createPolygonFromRect( basegfx::B2DRectangle( rRect.Left(), rRect.Top(), rRect.Right(), rRect.Bottom() ) ) ); - basegfx::B2DPolyPolygon aOtherPolyPoly( aRectPoly ); - - basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationOr( aThisPolyPoly, aOtherPolyPoly ); - *this = Region( aClip ); - - return; + return true; } - ImplPolyPolyRegionToBandRegion(); + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); - // no instance data? -> create! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - mpImplRegion = new ImplRegion(); + if(!pCurrent) + { + // no region band, create using the rectangle + *this = rRect; + return true; + } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + RegionBand* pNew = new RegionBand(*pCurrent); // get justified rectangle - long nLeft = std::min( rRect.Left(), rRect.Right() ); - long nTop = std::min( rRect.Top(), rRect.Bottom() ); - long nRight = std::max( rRect.Left(), rRect.Right() ); - long nBottom = std::max( rRect.Top(), rRect.Bottom() ); + const long nLeft(std::min(rRect.Left(), rRect.Right())); + const long nTop(std::min(rRect.Top(), rRect.Bottom())); + const long nRight(std::max(rRect.Left(), rRect.Right())); + const long nBottom(std::max(rRect.Top(), rRect.Bottom())); // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( nTop, nBottom ); + pNew->InsertBands(nTop, nBottom); // process union - mpImplRegion->Union( nLeft, nTop, nRight, nBottom ); + pNew->Union(nLeft, nTop, nRight, nBottom); // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!pNew->OptimizeBandList()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + delete pNew; + pNew = 0; } -} -// ----------------------------------------------------------------------- + mpRegionBand.reset(pNew); + return true; +} -void Region::Intersect( const Rectangle& rRect ) +bool Region::Intersect( const Rectangle& rRect ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // is rectangle empty? -> nothing to do if ( rRect.IsEmpty() ) { - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) - { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - } - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - return; + // empty rectangle will create empty region + SetEmpty(); + return true; } - // #103137# Avoid banding for special cases - if ( mpImplRegion->mpPolyPoly ) + if(IsNull()) { - // #127431# make ImplRegion unique, if not already. - if( mpImplRegion->mnRefCount > 1 ) - { - mpImplRegion->mnRefCount--; - mpImplRegion = new ImplRegion( *mpImplRegion->mpPolyPoly ); - } - - // use the PolyPolygon::Clip method for rectangles, this is - // fairly simple (does not even use GPC) and saves us from - // unnecessary banding - mpImplRegion->mpPolyPoly->Clip( rRect ); + // null region (everything) intersect with rect will give rect + *this = rRect; + return true; + } - // The clipping above may lead to empty ClipRegion - if(!mpImplRegion->mpPolyPoly->Count()) - { - // react on empty ClipRegion; ImplRegion already is unique (see above) - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - } - return; + if(IsEmpty()) + { + // no content, cannot get more empty + return true; } - else if( mpImplRegion->mpB2DPolyPoly ) + + if(HasPolyPolygonOrB2DPolyPolygon()) { - // #127431# make ImplRegion unique, if not already. - if( mpImplRegion->mnRefCount > 1 ) + // if polygon data prefer double precision, the other will be lost (if buffered) + if(getB2DPolyPolygon()) { - mpImplRegion->mnRefCount--; - mpImplRegion = new ImplRegion( *mpImplRegion->mpB2DPolyPoly ); + const basegfx::B2DPolyPolygon aPoly( + basegfx::tools::clipPolyPolygonOnRange( + *getB2DPolyPolygon(), + basegfx::B2DRange( + rRect.Left(), + rRect.Top(), + rRect.Right() + 1, + rRect.Bottom() + 1), + true, + false)); + + mpB2DPolyPolygon.reset(aPoly.count() ? new basegfx::B2DPolyPolygon(aPoly) : 0); + mpPolyPolygon.reset(); + mpRegionBand.reset(); } + else // if(getPolyPolygon()) + { + PolyPolygon aPoly(*getPolyPolygon()); - *mpImplRegion->mpB2DPolyPoly = - basegfx::tools::clipPolyPolygonOnRange( - *mpImplRegion->mpB2DPolyPoly, - basegfx::B2DRange( - rRect.Left(), - rRect.Top(), - rRect.Right() + 1, - rRect.Bottom() + 1), - true, - false); + // use the PolyPolygon::Clip method for rectangles, this is + // fairly simple (does not even use GPC) and saves us from + // unnecessary banding + aPoly.Clip(rRect); - // The clipping above may lead to empty ClipRegion - if(!mpImplRegion->mpB2DPolyPoly->count()) - { - // react on empty ClipRegion; ImplRegion already is unique (see above) - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(aPoly.Count() ? new PolyPolygon(aPoly) : 0); + mpRegionBand.reset(); } - return; + return true; } - else - ImplPolyPolyRegionToBandRegion(); - // is region empty? -> nothing to do! - if ( mpImplRegion == &aImplEmptyRegion ) - return; - - // get justified rectangle - long nLeft = std::min( rRect.Left(), rRect.Right() ); - long nTop = std::min( rRect.Top(), rRect.Bottom() ); - long nRight = std::max( rRect.Left(), rRect.Right() ); - long nBottom = std::max( rRect.Top(), rRect.Bottom() ); + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); - // is own region NULL-region? -> copy data! - if ( mpImplRegion == &aImplNullRegion ) + if(!pCurrent) { - // create instance of implementation class - mpImplRegion = new ImplRegion(); - - // add band with boundaries of the rectangle - mpImplRegion->mpFirstBand = new ImplRegionBand( nTop, nBottom ); - - // Set left and right boundaries of the band - mpImplRegion->mpFirstBand->Union( nLeft, nRight ); - mpImplRegion->mnRectCount = 1; - - return; + // region is empty -> nothing to do! + return true; } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + RegionBand* pNew = new RegionBand(*pCurrent); - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( nTop, nBottom ); + // get justified rectangle + const long nLeft(std::min(rRect.Left(), rRect.Right())); + const long nTop(std::min(rRect.Top(), rRect.Bottom())); + const long nRight(std::max(rRect.Left(), rRect.Right())); + const long nBottom(std::max(rRect.Top(), rRect.Bottom())); - // process intersections - ImplRegionBand* pPrevBand = 0; - ImplRegionBand* pBand = mpImplRegion->mpFirstBand; - while ( pBand ) - { - // band within intersection boundary? -> process. otherwise remove - if ( (pBand->mnYTop >= nTop) && - (pBand->mnYBottom <= nBottom) ) - { - // process intersection - pBand->Intersect( nLeft, nRight ); + // insert bands if the boundaries are not allready in the list + pNew->InsertBands(nTop, nBottom); - pPrevBand = pBand; - pBand = pBand->mpNextBand; - } - else - { - ImplRegionBand* pOldBand = pBand; - if ( pBand == mpImplRegion->mpFirstBand ) - mpImplRegion->mpFirstBand = pBand->mpNextBand; - else - pPrevBand->mpNextBand = pBand->mpNextBand; - pBand = pBand->mpNextBand; - delete pOldBand; - } - } + // process intersect + pNew->Intersect(nLeft, nTop, nRight, nBottom); // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!pNew->OptimizeBandList()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + delete pNew; + pNew = 0; } -} -// ----------------------------------------------------------------------- + mpRegionBand.reset(pNew); + return true; +} -void Region::Exclude( const Rectangle& rRect ) +bool Region::Exclude( const Rectangle& rRect ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // is rectangle empty? -> nothing to do if ( rRect.IsEmpty() ) - return; + { + // excluding nothing will do no change + return true; + } + + if(IsEmpty()) + { + // cannot exclude from empty, done + return true; + } + + if(IsNull()) + { + // error; cannnot exclude from null region since this is not representable + // in the data + OSL_ENSURE(false, "Region::Exclude error: Cannot exclude from null region (!)"); + return true; + } - if( HasPolyPolygon() ) + if( HasPolyPolygonOrB2DPolyPolygon() ) { // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); - aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); + + aThisPolyPoly = basegfx::tools::prepareForPolygonOperation(aThisPolyPoly); - if( aThisPolyPoly.count() == 0 ) - return; + if(!aThisPolyPoly.count()) + { + // when local polygon is empty, nothing can be excluded + return true; + } // get the other B2DPolyPolygon - basegfx::B2DPolygon aRectPoly( basegfx::tools::createPolygonFromRect( basegfx::B2DRectangle( rRect.Left(), rRect.Top(), rRect.Right(), rRect.Bottom() ) ) ); - basegfx::B2DPolyPolygon aOtherPolyPoly( aRectPoly ); + const basegfx::B2DPolygon aRectPoly( + basegfx::tools::createPolygonFromRect( + basegfx::B2DRectangle(rRect.Left(), rRect.Top(), rRect.Right(), rRect.Bottom()))); + const basegfx::B2DPolyPolygon aOtherPolyPoly(aRectPoly); + const basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationDiff(aThisPolyPoly, aOtherPolyPoly); - basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationDiff( aThisPolyPoly, aOtherPolyPoly ); - *this = Region( aClip ); + *this = Region(aClip); - return; + return true; } - ImplPolyPolyRegionToBandRegion(); + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); - // no instance data? -> create! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return; + if(!pCurrent) + { + // empty? -> done! + return true; + } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + RegionBand* pNew = new RegionBand(*pCurrent); // get justified rectangle - long nLeft = std::min( rRect.Left(), rRect.Right() ); - long nTop = std::min( rRect.Top(), rRect.Bottom() ); - long nRight = std::max( rRect.Left(), rRect.Right() ); - long nBottom = std::max( rRect.Top(), rRect.Bottom() ); + const long nLeft(std::min(rRect.Left(), rRect.Right())); + const long nTop(std::min(rRect.Top(), rRect.Bottom())); + const long nRight(std::max(rRect.Left(), rRect.Right())); + const long nBottom(std::max(rRect.Top(), rRect.Bottom())); - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( nTop, nBottom ); + // insert bands if the boundaries are not allready in the list + pNew->InsertBands(nTop, nBottom); // process exclude - mpImplRegion->Exclude( nLeft, nTop, nRight, nBottom ); + pNew->Exclude(nLeft, nTop, nRight, nBottom); // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!pNew->OptimizeBandList()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + delete pNew; + pNew = 0; } -} -// ----------------------------------------------------------------------- + mpRegionBand.reset(pNew); + return true; +} -void Region::XOr( const Rectangle& rRect ) +bool Region::XOr( const Rectangle& rRect ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // is rectangle empty? -> nothing to do if ( rRect.IsEmpty() ) - return; + { + // empty rectangle will not change local content + return true; + } + + if(IsEmpty()) + { + // rRect will be the xored-form (local off, rect on) + *this = rRect; + return true; + } + + if(IsNull()) + { + // error; cannnot exclude from null region since this is not representable + // in the data + OSL_ENSURE(false, "Region::XOr error: Cannot XOr with null region (!)"); + return true; + } - if( HasPolyPolygon() ) + if( HasPolyPolygonOrB2DPolyPolygon() ) { // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); + aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); - if( aThisPolyPoly.count() == 0 ) + if(!aThisPolyPoly.count()) { + // no local content, XOr will be equal to rectangle *this = rRect; - return; + return true; } // get the other B2DPolyPolygon - basegfx::B2DPolygon aRectPoly( basegfx::tools::createPolygonFromRect( basegfx::B2DRectangle( rRect.Left(), rRect.Top(), rRect.Right(), rRect.Bottom() ) ) ); - basegfx::B2DPolyPolygon aOtherPolyPoly( aRectPoly ); + const basegfx::B2DPolygon aRectPoly( + basegfx::tools::createPolygonFromRect( + basegfx::B2DRectangle(rRect.Left(), rRect.Top(), rRect.Right(), rRect.Bottom()))); + const basegfx::B2DPolyPolygon aOtherPolyPoly(aRectPoly); + const basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationXor(aThisPolyPoly, aOtherPolyPoly); - basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationXor( aThisPolyPoly, aOtherPolyPoly ); - *this = Region( aClip ); + *this = Region(aClip); - return; + return true; } - ImplPolyPolyRegionToBandRegion(); + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); - // no instance data? -> create! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - mpImplRegion = new ImplRegion(); + if(!pCurrent) + { + // rRect will be the xored-form (local off, rect on) + *this = rRect; + return true; + } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + // only region band mode possibility left here or null/empty + RegionBand* pNew = new RegionBand(*getRegionBand()); // get justified rectangle - long nLeft = std::min( rRect.Left(), rRect.Right() ); - long nTop = std::min( rRect.Top(), rRect.Bottom() ); - long nRight = std::max( rRect.Left(), rRect.Right() ); - long nBottom = std::max( rRect.Top(), rRect.Bottom() ); + const long nLeft(std::min(rRect.Left(), rRect.Right())); + const long nTop(std::min(rRect.Top(), rRect.Bottom())); + const long nRight(std::max(rRect.Left(), rRect.Right())); + const long nBottom(std::max(rRect.Top(), rRect.Bottom())); - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( nTop, nBottom ); + // insert bands if the boundaries are not allready in the list + pNew->InsertBands(nTop, nBottom); // process xor - mpImplRegion->XOr( nLeft, nTop, nRight, nBottom ); + pNew->XOr(nLeft, nTop, nRight, nBottom); // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!pNew->OptimizeBandList()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + delete pNew; + pNew = 0; } + + mpRegionBand.reset(pNew); + return true; } -// ----------------------------------------------------------------------- -void Region::ImplUnionPolyPolygon( const Region& i_rRegion ) +bool Region::Union( const Region& rRegion ) { - // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); - aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); - - if( aThisPolyPoly.count() == 0 ) + if(rRegion.IsEmpty()) { - *this = i_rRegion; - return; + // no extension at all + return true; } - // get the other B2DPolyPolygon - basegfx::B2DPolyPolygon aOtherPolyPoly( const_cast<Region&>(i_rRegion).ConvertToB2DPolyPolygon() ); - aOtherPolyPoly = basegfx::tools::prepareForPolygonOperation( aOtherPolyPoly ); + if(rRegion.IsNull()) + { + // extending with null region -> null region + *this = Region(true); + return true; + } + if(IsEmpty()) + { + // local is empty, union will give source region + *this = rRegion; + return true; + } - basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationOr( aThisPolyPoly, aOtherPolyPoly ); + if(IsNull()) + { + // already fully expanded (is null region), cannot be extended + return true; + } - *this = Region( aClip ); -} + if( rRegion.HasPolyPolygonOrB2DPolyPolygon() || HasPolyPolygonOrB2DPolyPolygon() ) + { + // get this B2DPolyPolygon + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); -void Region::Union( const Region& rRegion ) -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + aThisPolyPoly = basegfx::tools::prepareForPolygonOperation(aThisPolyPoly); - if( rRegion.HasPolyPolygon() || HasPolyPolygon() ) - { - ImplUnionPolyPolygon( rRegion ); - return; - } + if(!aThisPolyPoly.count()) + { + // when no local content, union will be equal to rRegion + *this = rRegion; + return true; + } - ImplPolyPolyRegionToBandRegion(); - ((Region*)&rRegion)->ImplPolyPolyRegionToBandRegion(); + // get the other B2DPolyPolygon + basegfx::B2DPolyPolygon aOtherPolyPoly(rRegion.GetAsB2DPolyPolygon()); + aOtherPolyPoly = basegfx::tools::prepareForPolygonOperation(aOtherPolyPoly); - // is region empty or null? -> nothing to do - if ( (rRegion.mpImplRegion == &aImplEmptyRegion) || (rRegion.mpImplRegion == &aImplNullRegion) ) - return; + // use logical OR operation + basegfx::B2DPolyPolygon aClip(basegfx::tools::solvePolygonOperationOr(aThisPolyPoly, aOtherPolyPoly)); - // no instance data? -> create! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - mpImplRegion = new ImplRegion(); + *this = Region( aClip ); + return true; + } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); - // Alle Rechtecke aus der uebergebenen Region auf diese Region anwenden - ImplRegionBand* pBand = rRegion.mpImplRegion->mpFirstBand; - while ( pBand ) + if(!pCurrent) { - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( pBand->mnYTop, pBand->mnYBottom ); + // local is empty, union will give source region + *this = rRegion; + return true; + } - // process all elements of the list - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) - { - mpImplRegion->Union( pSep->mnXLeft, pBand->mnYTop, - pSep->mnXRight, pBand->mnYBottom ); - pSep = pSep->mpNextSep; - } + const RegionBand* pSource = rRegion.getRegionBand(); - pBand = pBand->mpNextBand; + if(!pSource) + { + // no extension at all + return true; } + // prepare source and target + RegionBand* pNew = new RegionBand(*pCurrent); + + // union with source + pNew->Union(*pSource); + // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!pNew->OptimizeBandList()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + delete pNew; + pNew = 0; } + + mpRegionBand.reset(pNew); + return true; } -// ----------------------------------------------------------------------- -void Region::ImplIntersectWithPolyPolygon( const Region& i_rRegion ) +bool Region::Intersect( const Region& rRegion ) { - // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); - if( aThisPolyPoly.count() == 0 ) + // same instance data? -> nothing to do! + if(getB2DPolyPolygon() && getB2DPolyPolygon() == rRegion.getB2DPolyPolygon()) { - *this = i_rRegion; - return; + return true; } - // get the other B2DPolyPolygon - basegfx::B2DPolyPolygon aOtherPolyPoly( const_cast<Region&>(i_rRegion).ConvertToB2DPolyPolygon() ); + if(getPolyPolygon() && getPolyPolygon() == rRegion.getPolyPolygon()) + { + return true; + } - basegfx::B2DPolyPolygon aClip = basegfx::tools::clipPolyPolygonOnPolyPolygon( aOtherPolyPoly, aThisPolyPoly, true, false ); - *this = Region( aClip ); -} + if(getRegionBand() && getRegionBand() == rRegion.getRegionBand()) + { + return true; + } -void Region::Intersect( const Region& rRegion ) -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(rRegion.IsNull()) + { + // source region is null-region, intersect will not change local region + return true; + } - // same instance data? -> nothing to do! - if ( mpImplRegion == rRegion.mpImplRegion ) - return; + if(IsNull()) + { + // when local region is null-region, intersect will be equal to source + *this = rRegion; + return true; + } - if( rRegion.HasPolyPolygon() || HasPolyPolygon() ) + if(rRegion.IsEmpty()) { - ImplIntersectWithPolyPolygon( rRegion ); - return; + // source region is empty, intersection will always be empty + SetEmpty(); + return true; } - ImplPolyPolyRegionToBandRegion(); - ((Region*)&rRegion)->ImplPolyPolyRegionToBandRegion(); + if(IsEmpty()) + { + // local region is empty, cannot get more emty than that. Nothing to do + return true; + } - if ( mpImplRegion == &aImplEmptyRegion ) - return; + if( rRegion.HasPolyPolygonOrB2DPolyPolygon() || HasPolyPolygonOrB2DPolyPolygon() ) + { + // get this B2DPolyPolygon + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); - // is region null? -> nothing to do - if ( rRegion.mpImplRegion == &aImplNullRegion ) - return; + if(!aThisPolyPoly.count()) + { + // local region is empty, cannot get more emty than that. Nothing to do + return true; + } - // is rectangle empty? -> nothing to do - if ( rRegion.mpImplRegion == &aImplEmptyRegion ) - { - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) + // get the other B2DPolyPolygon + basegfx::B2DPolyPolygon aOtherPolyPoly(rRegion.GetAsB2DPolyPolygon()); + + if(!aOtherPolyPoly.count()) { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; + // source region is empty, intersection will always be empty + SetEmpty(); + return true; } - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - return; + + const basegfx::B2DPolyPolygon aClip( + basegfx::tools::clipPolyPolygonOnPolyPolygon( + aOtherPolyPoly, + aThisPolyPoly, + true, + false)); + *this = Region( aClip ); + return true; } - // is own region NULL-region? -> copy data! - if ( mpImplRegion == &aImplNullRegion) + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); + + if(!pCurrent) { - mpImplRegion = rRegion.mpImplRegion; - rRegion.mpImplRegion->mnRefCount++; - return; + // local region is empty, cannot get more emty than that. Nothing to do + return true; } - // Wenn wir weniger Rechtecke haben, drehen wir den Intersect-Aufruf um - if ( mpImplRegion->mnRectCount+2 < rRegion.mpImplRegion->mnRectCount ) + const RegionBand* pSource = rRegion.getRegionBand(); + + if(!pSource) { + // source region is empty, intersection will always be empty + SetEmpty(); + return true; + } + + // both RegionBands exist and are not empty + if(pCurrent->getRectangleCount() + 2 < pSource->getRectangleCount()) + { + // when we have less rectangles, turn around the call Region aTempRegion = rRegion; aTempRegion.Intersect( *this ); *this = aTempRegion; } else { - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + // prepare new regionBand + RegionBand* pNew = pCurrent ? new RegionBand(*pCurrent) : new RegionBand(); - // mark all bands as untouched - ImplRegionBand* pBand = mpImplRegion->mpFirstBand; - while ( pBand ) - { - pBand->mbTouched = sal_False; - pBand = pBand->mpNextBand; - } + // intersect with source + pNew->Intersect(*pSource); - pBand = rRegion.mpImplRegion->mpFirstBand; - while ( pBand ) + // cleanup + if(!pNew->OptimizeBandList()) { - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( pBand->mnYTop, pBand->mnYBottom ); + delete pNew; + pNew = 0; + } - // process all elements of the list - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) - { - // left boundary? - if ( pSep == pBand->mpFirstSep ) - { - // process intersection and do not remove untouched bands - mpImplRegion->Exclude( LONG_MIN+1, pBand->mnYTop, - pSep->mnXLeft-1, pBand->mnYBottom ); - } + mpRegionBand.reset(pNew); + } - // right boundary? - if ( pSep->mpNextSep == NULL ) - { - // process intersection and do not remove untouched bands - mpImplRegion->Exclude( pSep->mnXRight+1, pBand->mnYTop, - LONG_MAX-1, pBand->mnYBottom ); - } - else - { - // process intersection and do not remove untouched bands - mpImplRegion->Exclude( pSep->mnXRight+1, pBand->mnYTop, - pSep->mpNextSep->mnXLeft-1, pBand->mnYBottom ); - } + return true; +} - pSep = pSep->mpNextSep; - } +bool Region::Exclude( const Region& rRegion ) +{ + if ( rRegion.IsEmpty() ) + { + // excluding nothing will do no change + return true; + } - pBand = pBand->mpNextBand; - } + if ( rRegion.IsNull() ) + { + // excluding everything will create empty region + SetEmpty(); + return true; + } - // remove all untouched bands if bands already left - ImplRegionBand* pPrevBand = 0; - pBand = mpImplRegion->mpFirstBand; - while ( pBand ) - { - if ( !pBand->mbTouched ) - { - // save pointer - ImplRegionBand* pOldBand = pBand; + if(IsEmpty()) + { + // cannot exclude from empty, done + return true; + } - // previous element of the list - if ( pBand == mpImplRegion->mpFirstBand ) - mpImplRegion->mpFirstBand = pBand->mpNextBand; - else - pPrevBand->mpNextBand = pBand->mpNextBand; + if(IsNull()) + { + // error; cannnot exclude from null region since this is not representable + // in the data + OSL_ENSURE(false, "Region::Exclude error: Cannot exclude from null region (!)"); + return true; + } - pBand = pBand->mpNextBand; - delete pOldBand; - } - else - { - pPrevBand = pBand; - pBand = pBand->mpNextBand; - } - } + if( rRegion.HasPolyPolygonOrB2DPolyPolygon() || HasPolyPolygonOrB2DPolyPolygon() ) + { + // get this B2DPolyPolygon + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); - // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!aThisPolyPoly.count()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + // cannot exclude from empty, done + return true; } - } -} -// ----------------------------------------------------------------------- -void Region::ImplExcludePolyPolygon( const Region& i_rRegion ) -{ - // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); - if( aThisPolyPoly.count() == 0 ) - return; - aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); + aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); - // get the other B2DPolyPolygon - basegfx::B2DPolyPolygon aOtherPolyPoly( const_cast<Region&>(i_rRegion).ConvertToB2DPolyPolygon() ); - aOtherPolyPoly = basegfx::tools::prepareForPolygonOperation( aOtherPolyPoly ); + // get the other B2DPolyPolygon + basegfx::B2DPolyPolygon aOtherPolyPoly(rRegion.GetAsB2DPolyPolygon()); + aOtherPolyPoly = basegfx::tools::prepareForPolygonOperation( aOtherPolyPoly ); - basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationDiff( aThisPolyPoly, aOtherPolyPoly ); - *this = Region( aClip ); -} + basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationDiff( aThisPolyPoly, aOtherPolyPoly ); + *this = Region( aClip ); + return true; + } -void Region::Exclude( const Region& rRegion ) -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); - if( rRegion.HasPolyPolygon() || HasPolyPolygon() ) + if(!pCurrent) { - ImplExcludePolyPolygon( rRegion ); - return; + // cannot exclude from empty, done + return true; } - ImplPolyPolyRegionToBandRegion(); - ((Region*)&rRegion)->ImplPolyPolyRegionToBandRegion(); - - // is region empty or null? -> nothing to do - if ( (rRegion.mpImplRegion == &aImplEmptyRegion) || (rRegion.mpImplRegion == &aImplNullRegion) ) - return; - - // no instance data? -> nothing to do - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return; - - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + const RegionBand* pSource = rRegion.getRegionBand(); - // Alle Rechtecke aus der uebergebenen Region auf diese Region anwenden - ImplRegionBand* pBand = rRegion.mpImplRegion->mpFirstBand; - while ( pBand ) + if(!pSource) { - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( pBand->mnYTop, pBand->mnYBottom ); + // excluding nothing will do no change + return true; + } - // process all elements of the list - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) - { - mpImplRegion->Exclude( pSep->mnXLeft, pBand->mnYTop, - pSep->mnXRight, pBand->mnYBottom ); - pSep = pSep->mpNextSep; - } + // prepare source and target + RegionBand* pNew = new RegionBand(*pCurrent); - // Wir optimieren schon in der Schleife, da wir davon - // ausgehen, das wir insgesammt weniger Baender ueberpruefen - // muessen - if ( !mpImplRegion->OptimizeBandList() ) - { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - break; - } + // union with source + const bool bSuccess(pNew->Exclude(*pSource)); - pBand = pBand->mpNextBand; + // cleanup + if(!bSuccess) + { + delete pNew; + pNew = 0; } + + mpRegionBand.reset(pNew); + return true; } -// ----------------------------------------------------------------------- -void Region::ImplXOrPolyPolygon( const Region& i_rRegion ) +bool Region::XOr( const Region& rRegion ) { - // get this B2DPolyPolygon - basegfx::B2DPolyPolygon aThisPolyPoly( ConvertToB2DPolyPolygon() ); - if( aThisPolyPoly.count() == 0 ) + if ( rRegion.IsEmpty() ) { - *this = i_rRegion; - return; + // empty region will not change local content + return true; } - aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); - // get the other B2DPolyPolygon - basegfx::B2DPolyPolygon aOtherPolyPoly( const_cast<Region&>(i_rRegion).ConvertToB2DPolyPolygon() ); - aOtherPolyPoly = basegfx::tools::prepareForPolygonOperation( aOtherPolyPoly ); - - basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationXor( aThisPolyPoly, aOtherPolyPoly ); - *this = Region( aClip ); -} + if ( rRegion.IsNull() ) + { + // error; cannnot exclude null region from local since this is not representable + // in the data + OSL_ENSURE(false, "Region::XOr error: Cannot XOr with null region (!)"); + return true; + } -void Region::XOr( const Region& rRegion ) -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(IsEmpty()) + { + // rRect will be the xored-form (local off, rect on) + *this = rRegion; + return true; + } - if( rRegion.HasPolyPolygon() || HasPolyPolygon() ) + if(IsNull()) { - ImplXOrPolyPolygon( rRegion ); - return; + // error; cannnot exclude from null region since this is not representable + // in the data + OSL_ENSURE(false, "Region::XOr error: Cannot XOr with null region (!)"); + return false; } - ImplPolyPolyRegionToBandRegion(); - ((Region*)&rRegion)->ImplPolyPolyRegionToBandRegion(); + if( rRegion.HasPolyPolygonOrB2DPolyPolygon() || HasPolyPolygonOrB2DPolyPolygon() ) + { + // get this B2DPolyPolygon + basegfx::B2DPolyPolygon aThisPolyPoly(GetAsB2DPolyPolygon()); - // is region empty or null? -> nothing to do - if ( (rRegion.mpImplRegion == &aImplEmptyRegion) || (rRegion.mpImplRegion == &aImplNullRegion) ) - return; + if(!aThisPolyPoly.count()) + { + // rRect will be the xored-form (local off, rect on) + *this = rRegion; + return true; + } + + aThisPolyPoly = basegfx::tools::prepareForPolygonOperation( aThisPolyPoly ); + + // get the other B2DPolyPolygon + basegfx::B2DPolyPolygon aOtherPolyPoly(rRegion.GetAsB2DPolyPolygon()); + aOtherPolyPoly = basegfx::tools::prepareForPolygonOperation( aOtherPolyPoly ); - // no own instance data? -> XOr = copy - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) + basegfx::B2DPolyPolygon aClip = basegfx::tools::solvePolygonOperationXor( aThisPolyPoly, aOtherPolyPoly ); + *this = Region( aClip ); + return true; + } + + // only region band mode possibility left here or null/empty + const RegionBand* pCurrent = getRegionBand(); + + if(!pCurrent) { + // rRect will be the xored-form (local off, rect on) *this = rRegion; - return; + return true; } - // no own instance data? -> make own copy! - if ( mpImplRegion->mnRefCount > 1 ) - ImplCopyData(); + const RegionBand* pSource = rRegion.getRegionBand(); - // Alle Rechtecke aus der uebergebenen Region auf diese Region anwenden - ImplRegionBand* pBand = rRegion.mpImplRegion->mpFirstBand; - while ( pBand ) + if(!pSource) { - // insert bands if the boundaries are not already in the list - mpImplRegion->InsertBands( pBand->mnYTop, pBand->mnYBottom ); + // empty region will not change local content + return true; + } - // process all elements of the list - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) - { - mpImplRegion->XOr( pSep->mnXLeft, pBand->mnYTop, - pSep->mnXRight, pBand->mnYBottom ); - pSep = pSep->mpNextSep; - } + // prepare source and target + RegionBand* pNew = new RegionBand(*pCurrent); - pBand = pBand->mpNextBand; - } + // union with source + pNew->XOr(*pSource); // cleanup - if ( !mpImplRegion->OptimizeBandList() ) + if(!pNew->OptimizeBandList()) { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + delete pNew; + pNew = 0; } -} -// ----------------------------------------------------------------------- + mpRegionBand.reset(pNew); + + return true; +} Rectangle Region::GetBoundRect() const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - Rectangle aRect; + if(IsEmpty()) + { + // no internal data? -> region is empty! + return Rectangle(); + } - // no internal data? -> region is empty! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return aRect; + if(IsNull()) + { + // error; null region has no BoundRect + // OSL_ENSURE(false, "Region::GetBoundRect error: null region has unlimitied bound rect, not representable (!)"); + return Rectangle(); + } - // PolyPolygon data im Imp structure? - if ( mpImplRegion->mpPolyPoly ) - return mpImplRegion->mpPolyPoly->GetBoundRect(); - if( mpImplRegion->mpB2DPolyPoly ) + // prefer double precision source + if(getB2DPolyPolygon()) { - const basegfx::B2DRange aRange(basegfx::tools::getRange(*mpImplRegion->mpB2DPolyPoly)); + const basegfx::B2DRange aRange(basegfx::tools::getRange(*getB2DPolyPolygon())); + if(aRange.isEmpty()) { // emulate PolyPolygon::GetBoundRect() when empty polygon @@ -2076,264 +1277,149 @@ Rectangle Region::GetBoundRect() const } } - // no band in the list? -> region is empty! - if ( !mpImplRegion->mpFirstBand ) - return aRect; - - // get the boundaries of the first band - long nYTop = mpImplRegion->mpFirstBand->mnYTop; - long nYBottom = mpImplRegion->mpFirstBand->mnYBottom; - long nXLeft = mpImplRegion->mpFirstBand->GetXLeftBoundary(); - long nXRight = mpImplRegion->mpFirstBand->GetXRightBoundary(); - - // look in the band list (don't test first band again!) - ImplRegionBand* pBand = mpImplRegion->mpFirstBand->mpNextBand; - while ( pBand ) + if(getPolyPolygon()) { - nYBottom = pBand->mnYBottom; - nXLeft = std::min( nXLeft, pBand->GetXLeftBoundary() ); - nXRight = std::max( nXRight, pBand->GetXRightBoundary() ); + return getPolyPolygon()->GetBoundRect(); + } - pBand = pBand->mpNextBand; + if(getRegionBand()) + { + return getRegionBand()->GetBoundRect(); } - // set rectangle - aRect = Rectangle( nXLeft, nYTop, nXRight, nYBottom ); - return aRect; + return Rectangle(); } -// ----------------------------------------------------------------------- - -sal_Bool Region::HasPolyPolygon() const +const PolyPolygon Region::GetAsPolyPolygon() const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - if( !mpImplRegion ) - return false; - if( mpImplRegion->mpPolyPoly ) - return true; - if( mpImplRegion->mpB2DPolyPoly ) - return true; - return false; -} - -// ----------------------------------------------------------------------- + if(getPolyPolygon()) + { + return *getPolyPolygon(); + } -PolyPolygon Region::GetPolyPolygon() const -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(getB2DPolyPolygon()) + { + // the polygon needs to be converted, buffer the down converion + const PolyPolygon aPolyPolgon(*getB2DPolyPolygon()); + const_cast< Region* >(this)->mpPolyPolygon.reset(new PolyPolygon(aPolyPolgon)); - PolyPolygon aRet; + return *getPolyPolygon(); + } - if( mpImplRegion->mpPolyPoly ) - aRet = *mpImplRegion->mpPolyPoly; - else if( mpImplRegion->mpB2DPolyPoly ) + if(getRegionBand()) { - // the polygon needs to be converted - aRet = PolyPolygon( *mpImplRegion->mpB2DPolyPoly ); - // TODO: cache the converted polygon? - // mpImplRegion->mpB2DPolyPoly = aRet; + // the BandRegion needs to be converted, buffer the converion + const PolyPolygon aPolyPolgon(ImplCreatePolyPolygonFromRegionBand()); + const_cast< Region* >(this)->mpPolyPolygon.reset(new PolyPolygon(aPolyPolgon)); + + return *getPolyPolygon(); } - return aRet; + return PolyPolygon(); } -// ----------------------------------------------------------------------- - -const basegfx::B2DPolyPolygon Region::GetB2DPolyPolygon() const +const basegfx::B2DPolyPolygon Region::GetAsB2DPolyPolygon() const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(getB2DPolyPolygon()) + { + return *getB2DPolyPolygon(); + } - basegfx::B2DPolyPolygon aRet; + if(getPolyPolygon()) + { + // the polygon needs to be converted, buffer the up conversion. This will be preferred from now. + const basegfx::B2DPolyPolygon aB2DPolyPolygon(getPolyPolygon()->getB2DPolyPolygon()); + const_cast< Region* >(this)->mpB2DPolyPolygon.reset(new basegfx::B2DPolyPolygon(aB2DPolyPolygon)); + + return *getB2DPolyPolygon(); + } - if( mpImplRegion->mpB2DPolyPoly ) - aRet = *mpImplRegion->mpB2DPolyPoly; - else if( mpImplRegion->mpPolyPoly ) + if(getRegionBand()) { - // the polygon needs to be converted - aRet = mpImplRegion->mpPolyPoly->getB2DPolyPolygon(); - // TODO: cache the converted polygon? - // mpImplRegion->mpB2DPolyPoly = aRet; + // the BandRegion needs to be converted, buffer the converion + const basegfx::B2DPolyPolygon aB2DPolyPolygon(ImplCreateB2DPolyPolygonFromRegionBand()); + const_cast< Region* >(this)->mpB2DPolyPolygon.reset(new basegfx::B2DPolyPolygon(aB2DPolyPolygon)); + + return *getB2DPolyPolygon(); } - return aRet; + return basegfx::B2DPolyPolygon(); } -// ----------------------------------------------------------------------- - -basegfx::B2DPolyPolygon Region::ConvertToB2DPolyPolygon() +const RegionBand* Region::GetAsRegionBand() const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - basegfx::B2DPolyPolygon aRet; - - if( HasPolyPolygon() ) - aRet = GetB2DPolyPolygon(); - else + if(!getRegionBand()) { - RegionHandle aHdl = BeginEnumRects(); - Rectangle aSubRect; - while( GetNextEnumRect( aHdl, aSubRect ) ) + if(getB2DPolyPolygon()) + { + // convert B2DPolyPolygon to RegionBand, buffer it and return it + const_cast< Region* >(this)->mpRegionBand.reset(ImplCreateRegionBandFromPolyPolygon(PolyPolygon(*getB2DPolyPolygon()))); + } + else if(getPolyPolygon()) { - basegfx::B2DPolygon aPoly( basegfx::tools::createPolygonFromRect( - basegfx::B2DRectangle( aSubRect.Left(), aSubRect.Top(), aSubRect.Right(), aSubRect.Bottom() ) ) ); - aRet.append( aPoly ); + // convert B2DPolyPolygon to RegionBand, buffer it and return it + const_cast< Region* >(this)->mpRegionBand.reset(ImplCreateRegionBandFromPolyPolygon(*getPolyPolygon())); } - EndEnumRects( aHdl ); } - return aRet; + return getRegionBand(); } -// ----------------------------------------------------------------------- - -bool Region::ImplGetFirstRect( ImplRegionInfo& rImplRegionInfo, - long& rX, long& rY, - long& rWidth, long& rHeight ) const -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - ((Region*)this)->ImplPolyPolyRegionToBandRegion(); - - // no internal data? -> region is empty! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return false; - - // no band in the list? -> region is empty! - if ( mpImplRegion->mpFirstBand == NULL ) - return false; - - // initialise pointer for first access - ImplRegionBand* pCurrRectBand = mpImplRegion->mpFirstBand; - ImplRegionBandSep* pCurrRectBandSep = pCurrRectBand->mpFirstSep; - - DBG_ASSERT( pCurrRectBandSep != NULL, "Erstes Band wurde nicht optimiert." ); - if ( !pCurrRectBandSep ) - return false; - - // get boundaries of current rectangle - rX = pCurrRectBandSep->mnXLeft; - rY = pCurrRectBand->mnYTop; - rWidth = pCurrRectBandSep->mnXRight - pCurrRectBandSep->mnXLeft + 1; - rHeight = pCurrRectBand->mnYBottom - pCurrRectBand->mnYTop + 1; - - // save pointers - rImplRegionInfo.mpVoidCurrRectBand = (void*)pCurrRectBand; - rImplRegionInfo.mpVoidCurrRectBandSep = (void*)pCurrRectBandSep; - - return true; -} - -// ----------------------------------------------------------------------- - -bool Region::ImplGetNextRect( ImplRegionInfo& rImplRegionInfo, - long& rX, long& rY, - long& rWidth, long& rHeight ) const +bool Region::IsInside( const Point& rPoint ) const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // no internal data? -> region is empty! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) + if(IsEmpty()) + { + // no point can be in empty region return false; + } - // get last pointers - ImplRegionBand* pCurrRectBand = (ImplRegionBand*)rImplRegionInfo.mpVoidCurrRectBand; - ImplRegionBandSep* pCurrRectBandSep = (ImplRegionBandSep*)rImplRegionInfo.mpVoidCurrRectBandSep; - - // get next separation from current band - pCurrRectBandSep = pCurrRectBandSep->mpNextSep; - - // no separation found? -> go to next band! - if ( !pCurrRectBandSep ) + if(IsNull()) { - // get next band - pCurrRectBand = pCurrRectBand->mpNextBand; - - // no band found? -> not further rectangles! - if( !pCurrRectBand ) - return false; - - // get first separation in current band - pCurrRectBandSep = pCurrRectBand->mpFirstSep; + // all points are inside null-region + return true; } - // get boundaries of current rectangle - rX = pCurrRectBandSep->mnXLeft; - rY = pCurrRectBand->mnYTop; - rWidth = pCurrRectBandSep->mnXRight - pCurrRectBandSep->mnXLeft + 1; - rHeight = pCurrRectBand->mnYBottom - pCurrRectBand->mnYTop + 1; + // Too expensive (?) + //if(mpImplRegion->getRegionPolyPoly()) + //{ + // return mpImplRegion->getRegionPolyPoly()->IsInside( rPoint ); + //} - // save new pointers - rImplRegionInfo.mpVoidCurrRectBand = (void*)pCurrRectBand; - rImplRegionInfo.mpVoidCurrRectBandSep = (void*)pCurrRectBandSep; + // ensure RegionBand existance + const_cast< Region* >(this)->GetAsRegionBand(); + const RegionBand* pRegionBand = getRegionBand(); - return true; -} - -// ----------------------------------------------------------------------- + if(pRegionBand) + { + return pRegionBand->IsInside(rPoint); + } -RegionType Region::GetType() const -{ - if ( mpImplRegion == &aImplEmptyRegion ) - return REGION_EMPTY; - else if ( mpImplRegion == &aImplNullRegion ) - return REGION_NULL; - else if ( mpImplRegion->mnRectCount == 1 ) - return REGION_RECTANGLE; - else - return REGION_COMPLEX; + return false; } -// ----------------------------------------------------------------------- - -sal_Bool Region::IsInside( const Point& rPoint ) const +bool Region::IsInside( const Rectangle& rRect ) const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // PolyPolygon data im Imp structure? - ((Region*)this)->ImplPolyPolyRegionToBandRegion(); - - // no instance data? -> not inside - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return sal_False; - - // search band list - ImplRegionBand* pBand = mpImplRegion->mpFirstBand; - while ( pBand ) + if(IsEmpty()) { - // is point within band? - if ( (pBand->mnYTop <= rPoint.Y()) && - (pBand->mnYBottom >= rPoint.Y()) ) - { - // is point within separation of the band? - if ( pBand->IsInside( rPoint.X() ) ) - return sal_True; - else - return sal_False; - } - - pBand = pBand->mpNextBand; + // no rectangle can be in empty region + return false; } - return sal_False; -} - -// ----------------------------------------------------------------------- - -sal_Bool Region::IsInside( const Rectangle& rRect ) const -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(IsNull()) + { + // rectangle always inside null-region + return true; + } - // is rectangle empty? -> not inside if ( rRect.IsEmpty() ) - return sal_False; - - // no instance data? -> not inside - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return sal_False; + { + // is rectangle empty? -> not inside + return false; + } // create region from rectangle and intersect own region Region aRegion(rRect); - aRegion.Exclude( *this ); + aRegion.Exclude(*this); // rectangle is inside if exclusion is empty return aRegion.IsEmpty(); @@ -2341,12 +1427,19 @@ sal_Bool Region::IsInside( const Rectangle& rRect ) const // ----------------------------------------------------------------------- -sal_Bool Region::IsOver( const Rectangle& rRect ) const +bool Region::IsOver( const Rectangle& rRect ) const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + if(IsEmpty()) + { + // nothing can be over something empty + return false; + } - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return sal_False; + if(IsNull()) + { + // everything is over null region + return true; + } // Can we optimize this ??? - is used in StarDraw for brushes pointers // Why we have no IsOver for Regions ??? @@ -2358,210 +1451,127 @@ sal_Bool Region::IsOver( const Rectangle& rRect ) const return !aRegion.IsEmpty(); } -// ----------------------------------------------------------------------- - void Region::SetNull() { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) - { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - } - - // set new type - mpImplRegion = (ImplRegion*)(&aImplNullRegion); + // reset all content + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(); + mpRegionBand.reset(); + mbIsNull = true; } -// ----------------------------------------------------------------------- - void Region::SetEmpty() { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) - { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - } - - // set new type - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); + // reset all content + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(); + mpRegionBand.reset(); + mbIsNull = false; } -// ----------------------------------------------------------------------- - Region& Region::operator=( const Region& rRegion ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - DBG_ASSERT( rRegion.mpImplRegion->mnRefCount < 0xFFFFFFFE, "Region: RefCount overflow" ); - - // Zuerst Referenzcounter erhoehen, damit man sich selbst zuweisen kann - // RefCount == 0 fuer statische Objekte - if ( rRegion.mpImplRegion->mnRefCount ) - rRegion.mpImplRegion->mnRefCount++; + // reset all content + mpB2DPolyPolygon = rRegion.mpB2DPolyPolygon; + mpPolyPolygon = rRegion.mpPolyPolygon; + mpRegionBand = rRegion.mpRegionBand; + mbIsNull = rRegion.mbIsNull; - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) - { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - } - - mpImplRegion = rRegion.mpImplRegion; return *this; } -// ----------------------------------------------------------------------- - Region& Region::operator=( const Rectangle& rRect ) { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + mpB2DPolyPolygon.reset(); + mpPolyPolygon.reset(); + mpRegionBand.reset(rRect.IsEmpty() ? 0 : new RegionBand(rRect)); + mbIsNull = false; - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) - { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; - } - - ImplCreateRectRegion( rRect ); return *this; } -// ----------------------------------------------------------------------- - -sal_Bool Region::operator==( const Region& rRegion ) const +bool Region::operator==( const Region& rRegion ) const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - - // reference to same object? -> equal! - if ( mpImplRegion == rRegion.mpImplRegion ) - return sal_True; - - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return sal_False; - - if ( (rRegion.mpImplRegion == &aImplEmptyRegion) || (rRegion.mpImplRegion == &aImplNullRegion) ) - return sal_False; - - if ( rRegion.mpImplRegion->mpPolyPoly && mpImplRegion->mpPolyPoly ) - return *rRegion.mpImplRegion->mpPolyPoly == *mpImplRegion->mpPolyPoly; - else + if(IsNull() && rRegion.IsNull()) { - ((Region*)this)->ImplPolyPolyRegionToBandRegion(); - ((Region*)&rRegion)->ImplPolyPolyRegionToBandRegion(); - - // Eine der beiden Regions kann jetzt Empty sein - if ( mpImplRegion == rRegion.mpImplRegion ) - return sal_True; - - if ( mpImplRegion == &aImplEmptyRegion ) - return sal_False; - - if ( rRegion.mpImplRegion == &aImplEmptyRegion ) - return sal_False; + // both are null region + return true; } - // initialise pointers - ImplRegionBand* pOwnRectBand = mpImplRegion->mpFirstBand; - ImplRegionBandSep* pOwnRectBandSep = pOwnRectBand->mpFirstSep; - ImplRegionBand* pSecondRectBand = rRegion.mpImplRegion->mpFirstBand; - ImplRegionBandSep* pSecondRectBandSep = pSecondRectBand->mpFirstSep; - while ( pOwnRectBandSep && pSecondRectBandSep ) + if(IsEmpty() && rRegion.IsEmpty()) { - // get boundaries of current rectangle - long nOwnXLeft = pOwnRectBandSep->mnXLeft; - long nSecondXLeft = pSecondRectBandSep->mnXLeft; - if ( nOwnXLeft != nSecondXLeft ) - return sal_False; - - long nOwnYTop = pOwnRectBand->mnYTop; - long nSecondYTop = pSecondRectBand->mnYTop; - if ( nOwnYTop != nSecondYTop ) - return sal_False; + // both are empty + return true; + } - long nOwnXRight = pOwnRectBandSep->mnXRight; - long nSecondXRight = pSecondRectBandSep->mnXRight; - if ( nOwnXRight != nSecondXRight ) - return sal_False; + if(getB2DPolyPolygon() && getB2DPolyPolygon() == rRegion.getB2DPolyPolygon()) + { + // same instance data? -> equal + return true; + } - long nOwnYBottom = pOwnRectBand->mnYBottom; - long nSecondYBottom = pSecondRectBand->mnYBottom; - if ( nOwnYBottom != nSecondYBottom ) - return sal_False; + if(getPolyPolygon() && getPolyPolygon() == rRegion.getPolyPolygon()) + { + // same instance data? -> equal + return true; + } - // get next separation from current band - pOwnRectBandSep = pOwnRectBandSep->mpNextSep; + if(getRegionBand() && getRegionBand() == rRegion.getRegionBand()) + { + // same instance data? -> equal + return true; + } - // no separation found? -> go to next band! - if ( !pOwnRectBandSep ) - { - // get next band - pOwnRectBand = pOwnRectBand->mpNextBand; + if(IsNull() || IsEmpty()) + { + return false; + } - // get first separation in current band - if( pOwnRectBand ) - pOwnRectBandSep = pOwnRectBand->mpFirstSep; - } + if(rRegion.IsNull() || rRegion.IsEmpty()) + { + return false; + } - // get next separation from current band - pSecondRectBandSep = pSecondRectBandSep->mpNextSep; + if(rRegion.getB2DPolyPolygon() || getB2DPolyPolygon()) + { + // one of both has a B2DPolyPolygon based region, ensure both have it + // by evtl. conversion + const_cast< Region* >(this)->GetAsB2DPolyPolygon(); + const_cast< Region& >(rRegion).GetAsB2DPolyPolygon(); - // no separation found? -> go to next band! - if ( !pSecondRectBandSep ) - { - // get next band - pSecondRectBand = pSecondRectBand->mpNextBand; + return *rRegion.getB2DPolyPolygon() == *getB2DPolyPolygon(); + } - // get first separation in current band - if( pSecondRectBand ) - pSecondRectBandSep = pSecondRectBand->mpFirstSep; - } + if(rRegion.getPolyPolygon() || getPolyPolygon()) + { + // one of both has a B2DPolyPolygon based region, ensure both have it + // by evtl. conversion + const_cast< Region* >(this)->GetAsPolyPolygon(); + const_cast< Region& >(rRegion).GetAsPolyPolygon(); - if ( pOwnRectBandSep && !pSecondRectBandSep ) - return sal_False; + return *rRegion.getPolyPolygon() == *getPolyPolygon(); + } - if ( !pOwnRectBandSep && pSecondRectBandSep ) - return sal_False; + // both are not empty or null (see above) and if content supported polygon + // data the comparison is already done. Only both on RegionBand base can be left, + // but better check + if(rRegion.getRegionBand() && getRegionBand()) + { + return *rRegion.getRegionBand() == *getRegionBand(); } - return sal_True; + // should not happen, but better deny equality + return false; } -// ----------------------------------------------------------------------- - -enum StreamEntryType { STREAMENTRY_BANDHEADER, STREAMENTRY_SEPARATION, STREAMENTRY_END }; - -SvStream& operator>>( SvStream& rIStrm, Region& rRegion ) +SvStream& operator>>(SvStream& rIStrm, Region& rRegion) { - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - - VersionCompat aCompat( rIStrm, STREAM_READ ); - sal_uInt16 nVersion; - sal_uInt16 nTmp16; + VersionCompat aCompat(rIStrm, STREAM_READ); + sal_uInt16 nVersion(0); + sal_uInt16 nTmp16(0); - // statische Object haben RefCount von 0 - if ( rRegion.mpImplRegion->mnRefCount ) - { - if ( rRegion.mpImplRegion->mnRefCount > 1 ) - rRegion.mpImplRegion->mnRefCount--; - else - delete rRegion.mpImplRegion; - } + // clear region to be loaded + rRegion.SetEmpty(); // get version of streamed region rIStrm >> nVersion; @@ -2569,479 +1579,172 @@ SvStream& operator>>( SvStream& rIStrm, Region& rRegion ) // get type of region rIStrm >> nTmp16; + enum RegionType { REGION_NULL, REGION_EMPTY, REGION_RECTANGLE, REGION_COMPLEX }; RegionType meStreamedType = (RegionType)nTmp16; - switch( meStreamedType ) + switch(meStreamedType) { case REGION_NULL: - rRegion.mpImplRegion = (ImplRegion*)&aImplNullRegion; - break; + { + rRegion.SetNull(); + break; + } case REGION_EMPTY: - rRegion.mpImplRegion = (ImplRegion*)&aImplEmptyRegion; - break; + { + rRegion.SetEmpty(); + break; + } default: { - // create instance of implementation class - rRegion.mpImplRegion = new ImplRegion(); - - // get header from first element - rIStrm >> nTmp16; - - // get all bands - rRegion.mpImplRegion->mnRectCount = 0; - ImplRegionBand* pCurrBand = NULL; - while ( (StreamEntryType)nTmp16 != STREAMENTRY_END ) - { - // insert new band or new separation? - if ( (StreamEntryType)nTmp16 == STREAMENTRY_BANDHEADER ) - { - //#fdo39428 SvStream no longer supports operator>>(long&) - sal_Int32 nYTop; - sal_Int32 nYBottom; - - rIStrm >> nYTop; - rIStrm >> nYBottom; - - // create band - ImplRegionBand* pNewBand = new ImplRegionBand( nYTop, nYBottom ); - - // first element? -> set as first into the list - if ( !pCurrBand ) - rRegion.mpImplRegion->mpFirstBand = pNewBand; - else - pCurrBand->mpNextBand = pNewBand; - - // save pointer for next creation - pCurrBand = pNewBand; - } - else - { - //#fdo39428 SvStream no longer supports operator>>(long&) - sal_Int32 nXLeft; - sal_Int32 nXRight; - - rIStrm >> nXLeft; - rIStrm >> nXRight; - - // add separation - if ( pCurrBand ) - { - pCurrBand->Union( nXLeft, nXRight ); - rRegion.mpImplRegion->mnRectCount++; - } - } - - if( rIStrm.IsEof() ) - { - OSL_FAIL( "premature end of region stream" ); - delete rRegion.mpImplRegion; - rRegion.mpImplRegion = (ImplRegion*)&aImplEmptyRegion; - return rIStrm; - } + RegionBand* pNewRegionBand = new RegionBand(); + pNewRegionBand->load(rIStrm); + rRegion.mpRegionBand.reset(pNewRegionBand); - // get next header - rIStrm >> nTmp16; - } - - if( aCompat.GetVersion() >= 2 ) + if(aCompat.GetVersion() >= 2) { - sal_Bool bHasPolyPolygon; + sal_Bool bHasPolyPolygon(sal_False); rIStrm >> bHasPolyPolygon; - if( bHasPolyPolygon ) + if(bHasPolyPolygon) { - delete rRegion.mpImplRegion->mpPolyPoly; - rRegion.mpImplRegion->mpPolyPoly = new PolyPolygon; - rIStrm >> *( rRegion.mpImplRegion->mpPolyPoly ); + PolyPolygon* pNewPoly = new PolyPolygon(); + rIStrm >> *pNewPoly; + rRegion.mpPolyPolygon.reset(pNewPoly); } } + + break; } - break; } return rIStrm; } -// ----------------------------------------------------------------------- - SvStream& operator<<( SvStream& rOStrm, const Region& rRegion ) { - DBG_CHKOBJ( &rRegion, Region, ImplDbgTestRegion ); - - sal_uInt16 nVersion = 2; - VersionCompat aCompat( rOStrm, STREAM_WRITE, nVersion ); - Region aTmpRegion( rRegion ); - - // use tmp region to avoid destruction of internal region (polypolygon) of rRegion - aTmpRegion.ImplPolyPolyRegionToBandRegion(); + const sal_uInt16 nVersion(2); + VersionCompat aCompat(rOStrm, STREAM_WRITE, nVersion); // put version rOStrm << nVersion; // put type - rOStrm << (sal_uInt16)aTmpRegion.GetType(); + enum RegionType { REGION_NULL, REGION_EMPTY, REGION_RECTANGLE, REGION_COMPLEX }; + RegionType aRegionType(REGION_COMPLEX); + bool bEmpty(rRegion.IsEmpty()); - // put all bands if not null or empty - if ( (aTmpRegion.mpImplRegion != &aImplEmptyRegion) && (aTmpRegion.mpImplRegion != &aImplNullRegion) ) + if(!bEmpty && rRegion.getB2DPolyPolygon() && 0 == rRegion.getB2DPolyPolygon()->count()) { - ImplRegionBand* pBand = aTmpRegion.mpImplRegion->mpFirstBand; - while ( pBand ) - { - // put boundaries - //#fdo39428 SvStream no longer supports operator<<(long) - rOStrm << (sal_uInt16) STREAMENTRY_BANDHEADER; - rOStrm << sal::static_int_cast<sal_Int32>(pBand->mnYTop); - rOStrm << sal::static_int_cast<sal_Int32>(pBand->mnYBottom); - - // put separations of current band - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while ( pSep ) - { - // put separation - //#fdo39428 SvStream no longer supports operator<<(long) - rOStrm << (sal_uInt16) STREAMENTRY_SEPARATION; - rOStrm << sal::static_int_cast<sal_Int32>(pSep->mnXLeft); - rOStrm << sal::static_int_cast<sal_Int32>(pSep->mnXRight); - - // next separation from current band - pSep = pSep->mpNextSep; - } - - pBand = pBand->mpNextBand; - } - - // put endmarker - rOStrm << (sal_uInt16) STREAMENTRY_END; - - // write polypolygon if available - const sal_Bool bHasPolyPolygon = rRegion.HasPolyPolygon(); - rOStrm << bHasPolyPolygon; - - if( bHasPolyPolygon ) - { - // #i105373# - PolyPolygon aNoCurvePolyPolygon; - rRegion.GetPolyPolygon().AdaptiveSubdivide(aNoCurvePolyPolygon); - - rOStrm << aNoCurvePolyPolygon; - } + OSL_ENSURE(false, "Region with empty B2DPolyPolygon, should not be created (!)"); + bEmpty = true; } - return rOStrm; -} - -// ----------------------------------------------------------------------- - -void Region::ImplBeginAddRect() -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - // statische Object haben RefCount von 0 - if ( mpImplRegion->mnRefCount ) + if(!bEmpty && rRegion.getPolyPolygon() && 0 == rRegion.getPolyPolygon()->Count()) { - if ( mpImplRegion->mnRefCount > 1 ) - mpImplRegion->mnRefCount--; - else - delete mpImplRegion; + OSL_ENSURE(false, "Region with empty PolyPolygon, should not be created (!)"); + bEmpty = true; } - // create fresh region - mpImplRegion = new ImplRegion(); -} - -// ----------------------------------------------------------------------- - -sal_Bool Region::ImplAddRect( const Rectangle& rRect ) -{ - // Hier kein CheckThis, da nicht alle Daten auf Stand - - if ( rRect.IsEmpty() ) - return sal_True; - - // get justified rectangle - long nTop; - long nBottom; - long nLeft; - long nRight; - if ( rRect.Top() <= rRect.Bottom() ) + if(bEmpty) { - nTop = rRect.Top(); - nBottom = rRect.Bottom(); + aRegionType = REGION_EMPTY; } - else + else if(rRegion.IsNull()) { - nTop = rRect.Bottom(); - nBottom = rRect.Top(); + aRegionType = REGION_NULL; } - if ( rRect.Left() <= rRect.Right() ) + else if(rRegion.getRegionBand() && rRegion.getRegionBand()->isSingleRectangle()) { - nLeft = rRect.Left(); - nRight = rRect.Right(); + aRegionType = REGION_RECTANGLE; } - else - { - nLeft = rRect.Right(); - nRight = rRect.Left(); - } - - if ( !mpImplRegion->mpLastCheckedBand ) - { - // create new band - mpImplRegion->mpLastCheckedBand = new ImplRegionBand( nTop, nBottom ); - - // set band as current - mpImplRegion->mpFirstBand = mpImplRegion->mpLastCheckedBand; - mpImplRegion->mpLastCheckedBand->Union( nLeft, nRight ); - } - else - { - DBG_ASSERT( nTop >= mpImplRegion->mpLastCheckedBand->mnYTop, - "Region::ImplAddRect() - nTopY < nLastTopY" ); - // new band? create it! - if ( (nTop != mpImplRegion->mpLastCheckedBand->mnYTop) || - (nBottom != mpImplRegion->mpLastCheckedBand->mnYBottom) ) - { - // create new band - ImplRegionBand* pNewRegionBand = new ImplRegionBand( nTop, nBottom ); + rOStrm << (sal_uInt16)aRegionType; - // append band to the end - mpImplRegion->mpLastCheckedBand->mpNextBand = pNewRegionBand; + // get RegionBand + const RegionBand* pRegionBand = rRegion.getRegionBand(); - // skip to the new band - mpImplRegion->mpLastCheckedBand = mpImplRegion->mpLastCheckedBand->mpNextBand; - } - - // Insert Sep - mpImplRegion->mpLastCheckedBand->Union( nLeft, nRight ); - } - - return sal_True; -} - -// ----------------------------------------------------------------------- - -void Region::ImplEndAddRect() -{ - // check if we are empty - if ( !mpImplRegion->mpFirstBand ) - { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - return; - } - - // check if we have something to optimize - if ( !mpImplRegion->mpFirstBand->mpNextBand ) - { - // update mpImplRegion->mnRectCount, because no OptimizeBandList is called - ImplRegionBandSep* pSep = mpImplRegion->mpFirstBand->mpFirstSep; - mpImplRegion->mnRectCount = 0; - while( pSep ) - { - mpImplRegion->mnRectCount++; - pSep = pSep->mpNextSep; - } - - // Erst hier testen, da hier die Daten wieder stimmen - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - return; - } - - // have to revert list? -> do it now! - if ( mpImplRegion->mpFirstBand->mnYTop > - mpImplRegion->mpFirstBand->mpNextBand->mnYTop ) + if(pRegionBand) { - ImplRegionBand * pNewFirstRegionBand; - - // initialize temp list with first element - pNewFirstRegionBand = mpImplRegion->mpFirstBand; - mpImplRegion->mpFirstBand = mpImplRegion->mpFirstBand->mpNextBand; - pNewFirstRegionBand->mpNextBand = NULL; - - // insert elements to the temp list - while ( mpImplRegion->mpFirstBand ) - { - ImplRegionBand * pSavedRegionBand = pNewFirstRegionBand; - pNewFirstRegionBand = mpImplRegion->mpFirstBand; - mpImplRegion->mpFirstBand = mpImplRegion->mpFirstBand->mpNextBand; - pNewFirstRegionBand->mpNextBand = pSavedRegionBand; - } - - // set temp list as new list - mpImplRegion->mpFirstBand = pNewFirstRegionBand; + pRegionBand->save(rOStrm); } - // cleanup - if ( !mpImplRegion->OptimizeBandList() ) - { - delete mpImplRegion; - mpImplRegion = (ImplRegion*)(&aImplEmptyRegion); - } - - // Erst hier testen, da hier die Daten wieder stimmen - DBG_CHKTHIS( Region, ImplDbgTestRegion ); -} - -// ----------------------------------------------------------------------- - -sal_uLong Region::GetRectCount() const -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - ((Region*)this)->ImplPolyPolyRegionToBandRegion(); - -#ifdef DBG_UTIL - sal_uLong nCount = 0; + // write polypolygon if available + const sal_Bool bHasPolyPolygon(rRegion.HasPolyPolygonOrB2DPolyPolygon()); + rOStrm << bHasPolyPolygon; - // all bands if not null or empty - if ( (mpImplRegion != &aImplEmptyRegion) && (mpImplRegion != &aImplNullRegion) ) + if(bHasPolyPolygon) { - ImplRegionBand* pBand = mpImplRegion->mpFirstBand; - while ( pBand ) - { - ImplRegionBandSep* pSep = pBand->mpFirstSep; - while( pSep ) - { - nCount++; - pSep = pSep->mpNextSep; - } - - pBand = pBand->mpNextBand; - } - } - - DBG_ASSERT( mpImplRegion->mnRectCount == nCount, "Region: invalid mnRectCount!" ); -#endif + // #i105373# + PolyPolygon aNoCurvePolyPolygon; + rRegion.GetAsPolyPolygon().AdaptiveSubdivide(aNoCurvePolyPolygon); - return mpImplRegion->mnRectCount; -} - -// ----------------------------------------------------------------------- - -RegionHandle Region::BeginEnumRects() -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - ImplPolyPolyRegionToBandRegion(); - - // no internal data? -> region is empty! - if ( (mpImplRegion == &aImplEmptyRegion) || (mpImplRegion == &aImplNullRegion) ) - return 0; - - // no band in the list? -> region is empty! - if ( mpImplRegion->mpFirstBand == NULL ) - { - DBG_ASSERT( mpImplRegion->mpFirstBand, "Region::BeginEnumRects() First Band is Empty!" ); - return 0; + rOStrm << aNoCurvePolyPolygon; } - ImplRegionHandle* pData = new ImplRegionHandle; - pData->mpRegion = new Region( *this ); - pData->mbFirst = sal_True; - - // save pointers - pData->mpCurrRectBand = pData->mpRegion->mpImplRegion->mpFirstBand; - pData->mpCurrRectBandSep = pData->mpCurrRectBand->mpFirstSep; - - return (RegionHandle)pData; + return rOStrm; } -// ----------------------------------------------------------------------- - -sal_Bool Region::GetEnumRects( RegionHandle pVoidData, Rectangle& rRect ) +void Region::GetRegionRectangles(RectangleVector& rTarget) const { - DBG_CHKTHIS( Region, ImplDbgTestRegion ); + // clear returnvalues + rTarget.clear(); - ImplRegionHandle* pData = (ImplRegionHandle*)pVoidData; - if ( !pData ) - return sal_False; + // ensure RegionBand existance + const_cast< Region* >(this)->GetAsRegionBand(); + const RegionBand* pRegionBand = getRegionBand(); - if ( pData->mbFirst ) - pData->mbFirst = sal_False; - else + if(pRegionBand) { - // get next separation from current band - pData->mpCurrRectBandSep = pData->mpCurrRectBandSep->mpNextSep; - - // no separation found? -> go to next band! - if ( !pData->mpCurrRectBandSep ) - { - // get next band - pData->mpCurrRectBand = pData->mpCurrRectBand->mpNextBand; - - // no band found? -> not further rectangles! - if ( !pData->mpCurrRectBand ) - return sal_False; - - // get first separation in current band - pData->mpCurrRectBandSep = pData->mpCurrRectBand->mpFirstSep; - } + pRegionBand->GetRegionRectangles(rTarget); } - - // get boundaries of current rectangle - rRect.Top() = pData->mpCurrRectBand->mnYTop; - rRect.Bottom() = pData->mpCurrRectBand->mnYBottom; - rRect.Left() = pData->mpCurrRectBandSep->mnXLeft; - rRect.Right() = pData->mpCurrRectBandSep->mnXRight; - return sal_True; } -// ----------------------------------------------------------------------- - -void Region::EndEnumRects( RegionHandle pVoidData ) -{ - DBG_CHKTHIS( Region, ImplDbgTestRegion ); - - ImplRegionHandle* pData = (ImplRegionHandle*)pVoidData; - if ( !pData ) - return; - - // cleanup - delete pData->mpRegion; - delete pData; -} - -// ----------------------------------------------------------------------- - static inline bool ImplPolygonRectTest( const Polygon& rPoly, Rectangle* pRectOut = NULL ) { bool bIsRect = false; const Point* pPoints = rPoly.GetConstPointAry(); sal_uInt16 nPoints = rPoly.GetSize(); + if( nPoints == 4 || (nPoints == 5 && pPoints[0] == pPoints[4]) ) { - long nX1 = pPoints[0].X(), nX2 = pPoints[2].X(), - nY1 = pPoints[0].Y(), nY2 = pPoints[2].Y(); - if( ( (pPoints[1].X() == nX1 && pPoints[3].X() == nX2) && - (pPoints[1].Y() == nY2 && pPoints[3].Y() == nY1) ) - || - ( (pPoints[1].X() == nX2 && pPoints[3].X() == nX1) && - (pPoints[1].Y() == nY1 && pPoints[3].Y() == nY2) ) ) + long nX1 = pPoints[0].X(), nX2 = pPoints[2].X(), nY1 = pPoints[0].Y(), nY2 = pPoints[2].Y(); + + if( ( (pPoints[1].X() == nX1 && pPoints[3].X() == nX2) && (pPoints[1].Y() == nY2 && pPoints[3].Y() == nY1) ) + || ( (pPoints[1].X() == nX2 && pPoints[3].X() == nX1) && (pPoints[1].Y() == nY1 && pPoints[3].Y() == nY2) ) ) { bIsRect = true; + if( pRectOut ) { long nSwap; + if( nX2 < nX1 ) { nSwap = nX2; nX2 = nX1; nX1 = nSwap; } + if( nY2 < nY1 ) { nSwap = nY2; nY2 = nY1; nY1 = nSwap; } + if( nX2 != nX1 ) + { nX2--; + } + if( nY2 != nY1 ) + { nY2--; + } + pRectOut->Left() = nX1; pRectOut->Right() = nX2; pRectOut->Top() = nY1; @@ -3049,6 +1752,7 @@ static inline bool ImplPolygonRectTest( const Polygon& rPoly, Rectangle* pRectOu } } } + return bIsRect; } @@ -3058,31 +1762,46 @@ Region Region::GetRegionFromPolyPolygon( const PolyPolygon& rPolyPoly ) // check if it's worth extracting the XOr'ing the Rectangles // empiricism shows that break even between XOr'ing rectangles separately - // and ImplPolyPolyRegionToBandRegion is at half rectangles/half polygons + // and ImplCreateRegionBandFromPolyPolygon is at half rectangles/half polygons int nPolygonRects = 0, nPolygonPolygons = 0; int nPolygons = rPolyPoly.Count(); for( sal_uInt16 i = 0; i < nPolygons; i++ ) { const Polygon& rPoly = rPolyPoly[i]; + if( ImplPolygonRectTest( rPoly ) ) + { nPolygonRects++; + } else + { nPolygonPolygons++; + } } + if( nPolygonPolygons > nPolygonRects ) + { return Region( rPolyPoly ); + } Region aResult; Rectangle aRect; + for( sal_uInt16 i = 0; i < nPolygons; i++ ) { const Polygon& rPoly = rPolyPoly[i]; + if( ImplPolygonRectTest( rPoly, &aRect ) ) + { aResult.XOr( aRect ); + } else + { aResult.XOr( Region(rPoly) ); + } } + return aResult; } diff --git a/vcl/source/gdi/regionband.cxx b/vcl/source/gdi/regionband.cxx new file mode 100644 index 000000000000..1a4d249c0a1d --- /dev/null +++ b/vcl/source/gdi/regionband.cxx @@ -0,0 +1,1377 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ +/* + * This file is part of the LibreOffice project. + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * This file incorporates work covered by the following license notice: + * + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed + * with this work for additional information regarding copyright + * ownership. The ASF licenses this file to you under the Apache + * License, Version 2.0 (the "License"); you may not use this file + * except in compliance with the License. You may obtain a copy of + * the License at http://www.apache.org/licenses/LICENSE-2.0 . + */ + +#include <tools/stream.hxx> +#include <tools/debug.hxx> +#include <regionband.hxx> + +////////////////////////////////////////////////////////////////////////////// + +DBG_NAME( RegionBand ) +DBG_NAMEEX( Polygon ) +DBG_NAMEEX( PolyPolygon ) + +////////////////////////////////////////////////////////////////////////////// + +RegionBand::RegionBand() +: mpFirstBand(0), + mpLastCheckedBand(0) +{ + DBG_CTOR(RegionBand, ImplDbgTestRegionBand); +} + +RegionBand::RegionBand(const RegionBand& rRef) +: mpFirstBand(0), + mpLastCheckedBand(0) +{ + *this = rRef; + DBG_CTOR(RegionBand, ImplDbgTestRegionBand); +} + +RegionBand& RegionBand::operator=(const RegionBand& rRef) +{ + ImplRegionBand* pPrevBand = 0; + ImplRegionBand* pBand = rRef.mpFirstBand; + + while(pBand) + { + ImplRegionBand* pNewBand = new ImplRegionBand(*pBand); + + // first element? -> set as first into the list + if(pBand == rRef.mpFirstBand) + { + mpFirstBand = pNewBand; + } + else + { + pPrevBand->mpNextBand = pNewBand; + } + + pPrevBand = pNewBand; + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + DBG_CHKOBJ(&rRef, RegionBand, ImplDbgTestRegionBand); + + return *this; +} + +RegionBand::RegionBand(const Rectangle& rRect) +: mpFirstBand(0), + mpLastCheckedBand(0) +{ + const long nTop(std::min(rRect.Top(), rRect.Bottom())); + const long nBottom(std::max(rRect.Top(), rRect.Bottom())); + const long nLeft(std::min(rRect.Left(), rRect.Right())); + const long nRight(std::max(rRect.Left(), rRect.Right())); + + // add band with boundaries of the rectangle + mpFirstBand = new ImplRegionBand(nTop, nBottom); + + // Set left and right boundaries of the band + mpFirstBand->Union(nLeft, nRight); + + DBG_CTOR(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::implReset() +{ + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + ImplRegionBand* pTempBand = pBand->mpNextBand; + delete pBand; + pBand = pTempBand; + } + + mpLastCheckedBand = 0; + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +RegionBand::~RegionBand() +{ + implReset(); + DBG_DTOR(RegionBand, ImplDbgTestRegionBand); +} + +bool RegionBand::operator==( const RegionBand& rRegionBand ) const +{ + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + DBG_CHKOBJ(&rRegionBand, RegionBand, ImplDbgTestRegionBand); + + // initialise pointers + ImplRegionBand* pOwnRectBand = mpFirstBand; + ImplRegionBandSep* pOwnRectBandSep = pOwnRectBand->mpFirstSep; + ImplRegionBand* pSecondRectBand = rRegionBand.mpFirstBand; + ImplRegionBandSep* pSecondRectBandSep = pSecondRectBand->mpFirstSep; + + while ( pOwnRectBandSep && pSecondRectBandSep ) + { + // get boundaries of current rectangle + long nOwnXLeft = pOwnRectBandSep->mnXLeft; + long nSecondXLeft = pSecondRectBandSep->mnXLeft; + + if ( nOwnXLeft != nSecondXLeft ) + { + return false; + } + + long nOwnYTop = pOwnRectBand->mnYTop; + long nSecondYTop = pSecondRectBand->mnYTop; + + if ( nOwnYTop != nSecondYTop ) + { + return false; + } + + long nOwnXRight = pOwnRectBandSep->mnXRight; + long nSecondXRight = pSecondRectBandSep->mnXRight; + + if ( nOwnXRight != nSecondXRight ) + { + return false; + } + + long nOwnYBottom = pOwnRectBand->mnYBottom; + long nSecondYBottom = pSecondRectBand->mnYBottom; + + if ( nOwnYBottom != nSecondYBottom ) + { + return false; + } + + // get next separation from current band + pOwnRectBandSep = pOwnRectBandSep->mpNextSep; + + // no separation found? -> go to next band! + if ( !pOwnRectBandSep ) + { + // get next band + pOwnRectBand = pOwnRectBand->mpNextBand; + + // get first separation in current band + if( pOwnRectBand ) + { + pOwnRectBandSep = pOwnRectBand->mpFirstSep; + } + } + + // get next separation from current band + pSecondRectBandSep = pSecondRectBandSep->mpNextSep; + + // no separation found? -> go to next band! + if ( !pSecondRectBandSep ) + { + // get next band + pSecondRectBand = pSecondRectBand->mpNextBand; + + // get first separation in current band + if( pSecondRectBand ) + { + pSecondRectBandSep = pSecondRectBand->mpFirstSep; + } + } + + if ( pOwnRectBandSep && !pSecondRectBandSep ) + { + return false; + } + + if ( !pOwnRectBandSep && pSecondRectBandSep ) + { + return false; + } + } + + return true; +} + +enum StreamEntryType { STREAMENTRY_BANDHEADER, STREAMENTRY_SEPARATION, STREAMENTRY_END }; + +void RegionBand::load(SvStream& rIStrm) +{ + // clear this nstance's data + implReset(); + + // get all bands + ImplRegionBand* pCurrBand = 0; + + // get header from first element + sal_uInt16 nTmp16(0); + rIStrm >> nTmp16; + + while(STREAMENTRY_END != (StreamEntryType)nTmp16) + { + // insert new band or new separation? + if(STREAMENTRY_BANDHEADER == (StreamEntryType)nTmp16) + { + long nYTop; + long nYBottom; + + rIStrm >> nYTop; + rIStrm >> nYBottom; + + // create band + ImplRegionBand* pNewBand = new ImplRegionBand( nYTop, nYBottom ); + + // first element? -> set as first into the list + if ( !pCurrBand ) + { + mpFirstBand = pNewBand; + } + else + { + pCurrBand->mpNextBand = pNewBand; + } + + // save pointer for next creation + pCurrBand = pNewBand; + } + else + { + long nXLeft; + long nXRight; + + rIStrm >> nXLeft; + rIStrm >> nXRight; + + // add separation + if ( pCurrBand ) + { + pCurrBand->Union( nXLeft, nXRight ); + } + } + + if( rIStrm.IsEof() ) + { + OSL_ENSURE(false, "premature end of region stream" ); + implReset(); + return; + } + + // get next header + rIStrm >> nTmp16; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::save(SvStream& rOStrm) const +{ + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + // put boundaries + rOStrm << (sal_uInt16)STREAMENTRY_BANDHEADER; + rOStrm << pBand->mnYTop; + rOStrm << pBand->mnYBottom; + + // put separations of current band + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while(pSep) + { + // put separation + rOStrm << (sal_uInt16)STREAMENTRY_SEPARATION; + rOStrm << pSep->mnXLeft; + rOStrm << pSep->mnXRight; + + // next separation from current band + pSep = pSep->mpNextSep; + } + + pBand = pBand->mpNextBand; + } + + // put endmarker + rOStrm << (sal_uInt16)STREAMENTRY_END; +} + +bool RegionBand::isSingleRectangle() const +{ + // just one band? + if(mpFirstBand && !mpFirstBand->mpNextBand) + { + // just one sep? + if(mpFirstBand->mpFirstSep && !mpFirstBand->mpFirstSep->mpNextSep) + { + return true; + } + } + + return false; +} + +void RegionBand::InsertBand(ImplRegionBand* pPreviousBand, ImplRegionBand* pBandToInsert) +{ + OSL_ASSERT(pBandToInsert!=NULL); + + if(!pPreviousBand) + { + // Insert band before all others. + if(mpFirstBand) + { + mpFirstBand->mpPrevBand = pBandToInsert; + } + + pBandToInsert->mpNextBand = mpFirstBand; + mpFirstBand = pBandToInsert; + } + else + { + // Insert band directly after pPreviousBand. + pBandToInsert->mpNextBand = pPreviousBand->mpNextBand; + pPreviousBand->mpNextBand = pBandToInsert; + pBandToInsert->mpPrevBand = pPreviousBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::processPoints() +{ + ImplRegionBand* pRegionBand = mpFirstBand; + + while(pRegionBand) + { + // generate separations from the lines and process union + pRegionBand->ProcessPoints(); + pRegionBand = pRegionBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +/** This function is similar to the RegionBand::InsertBands() method. + It creates a minimal set of missing bands so that the entire vertical + interval from nTop to nBottom is covered by bands. +*/ +void RegionBand::ImplAddMissingBands(const long nTop, const long nBottom) +{ + // Iterate over already existing bands and add missing bands atop the + // first and between two bands. + ImplRegionBand* pPreviousBand = NULL; + ImplRegionBand* pBand = ImplGetFirstRegionBand(); + long nCurrentTop (nTop); + + while (pBand != NULL && nCurrentTop<nBottom) + { + if (nCurrentTop < pBand->mnYTop) + { + // Create new band above the current band. + ImplRegionBand* pAboveBand = new ImplRegionBand( + nCurrentTop, + ::std::min(nBottom,pBand->mnYTop-1)); + InsertBand(pPreviousBand, pAboveBand); + } + + // Adapt the top of the interval to prevent overlapping bands. + nCurrentTop = ::std::max(nTop, pBand->mnYBottom+1); + + // Advance to next band. + pPreviousBand = pBand; + pBand = pBand->mpNextBand; + } + + // We still have to cover two cases: + // 1. The region does not yet contain any bands. + // 2. The intervall nTop->nBottom extends past the bottom most band. + if (nCurrentTop <= nBottom + && (pBand==NULL || nBottom>pBand->mnYBottom)) + { + // When there is no previous band then the new one will be the + // first. Otherwise the new band is inserted behind the last band. + InsertBand( + pPreviousBand, + new ImplRegionBand( + nCurrentTop, + nBottom)); + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::CreateBandRange(long nYTop, long nYBottom) +{ + // add top band + mpFirstBand = new ImplRegionBand( nYTop-1, nYTop-1 ); + + // begin first search from the first element + mpLastCheckedBand = mpFirstBand; + ImplRegionBand* pBand = mpFirstBand; + + for ( int i = nYTop; i <= nYBottom+1; i++ ) + { + // create new band + ImplRegionBand* pNewBand = new ImplRegionBand( i, i ); + pBand->mpNextBand = pNewBand; + + if ( pBand != mpFirstBand ) + { + pNewBand->mpPrevBand = pBand; + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +bool RegionBand::InsertLine(const Point& rStartPt, const Point& rEndPt, long nLineId) +{ + long nX, nY; + + // lines consisting of a single point do not interest here + if ( rStartPt == rEndPt ) + { + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + LineType eLineType = (rStartPt.Y() > rEndPt.Y()) ? LINE_DESCENDING : LINE_ASCENDING; + if ( rStartPt.X() == rEndPt.X() ) + { + // vertical line + const long nEndY = rEndPt.Y(); + + nX = rStartPt.X(); + nY = rStartPt.Y(); + + if( nEndY > nY ) + { + for ( ; nY <= nEndY; nY++ ) + { + Point aNewPoint( nX, nY ); + InsertPoint( aNewPoint, nLineId, + (aNewPoint == rEndPt) || (aNewPoint == rStartPt), + eLineType ); + } + } + else + { + for ( ; nY >= nEndY; nY-- ) + { + Point aNewPoint( nX, nY ); + InsertPoint( aNewPoint, nLineId, + (aNewPoint == rEndPt) || (aNewPoint == rStartPt), + eLineType ); + } + } + } + else if ( rStartPt.Y() != rEndPt.Y() ) + { + const long nDX = labs( rEndPt.X() - rStartPt.X() ); + const long nDY = labs( rEndPt.Y() - rStartPt.Y() ); + const long nStartX = rStartPt.X(); + const long nStartY = rStartPt.Y(); + const long nEndX = rEndPt.X(); + const long nEndY = rEndPt.Y(); + const long nXInc = ( nStartX < nEndX ) ? 1L : -1L; + const long nYInc = ( nStartY < nEndY ) ? 1L : -1L; + + if ( nDX >= nDY ) + { + const long nDYX = ( nDY - nDX ) << 1; + const long nDY2 = nDY << 1; + long nD = nDY2 - nDX; + + for ( nX = nStartX, nY = nStartY; nX != nEndX; nX += nXInc ) + { + InsertPoint( Point( nX, nY ), nLineId, nStartX == nX, eLineType ); + + if ( nD < 0L ) + nD += nDY2; + else + nD += nDYX, nY += nYInc; + } + } + else + { + const long nDYX = ( nDX - nDY ) << 1; + const long nDY2 = nDX << 1; + long nD = nDY2 - nDY; + + for ( nX = nStartX, nY = nStartY; nY != nEndY; nY += nYInc ) + { + InsertPoint( Point( nX, nY ), nLineId, nStartY == nY, eLineType ); + + if ( nD < 0L ) + nD += nDY2; + else + nD += nDYX, nX += nXInc; + } + } + + // last point + InsertPoint( Point( nEndX, nEndY ), nLineId, true, eLineType ); + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; +} + +bool RegionBand::InsertPoint(const Point &rPoint, long nLineID, bool bEndPoint, LineType eLineType) +{ + DBG_ASSERT( mpFirstBand != NULL, "RegionBand::InsertPoint - no bands available!" ); + + if ( rPoint.Y() == mpLastCheckedBand->mnYTop ) + { + mpLastCheckedBand->InsertPoint( rPoint.X(), nLineID, bEndPoint, eLineType ); + return true; + } + + if ( rPoint.Y() > mpLastCheckedBand->mnYTop ) + { + // Search ascending + while ( mpLastCheckedBand ) + { + // Insert point if possible + if ( rPoint.Y() == mpLastCheckedBand->mnYTop ) + { + mpLastCheckedBand->InsertPoint( rPoint.X(), nLineID, bEndPoint, eLineType ); + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + mpLastCheckedBand = mpLastCheckedBand->mpNextBand; + } + + OSL_ENSURE(false, "RegionBand::InsertPoint reached the end of the list!" ); + } + else + { + // Search descending + while ( mpLastCheckedBand ) + { + // Insert point if possible + if ( rPoint.Y() == mpLastCheckedBand->mnYTop ) + { + mpLastCheckedBand->InsertPoint( rPoint.X(), nLineID, bEndPoint, eLineType ); + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + mpLastCheckedBand = mpLastCheckedBand->mpPrevBand; + } + + OSL_ENSURE(false, "RegionBand::InsertPoint reached the beginning of the list!" ); + } + + OSL_ENSURE(false, "RegionBand::InsertPoint point not inserted!" ); + + // reinitialize pointer (should never be reached!) + mpLastCheckedBand = mpFirstBand; + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return false; +} + +bool RegionBand::OptimizeBandList() +{ + ImplRegionBand* pPrevBand = 0; + ImplRegionBand* pBand = mpFirstBand; + + while ( pBand ) + { + const bool bBTEqual = pBand->mpNextBand && (pBand->mnYBottom == pBand->mpNextBand->mnYTop); + + // no separation? -> remove! + if ( pBand->IsEmpty() || (bBTEqual && (pBand->mnYBottom == pBand->mnYTop)) ) + { + // save pointer + ImplRegionBand* pOldBand = pBand; + + // previous element of the list + if ( pBand == mpFirstBand ) + mpFirstBand = pBand->mpNextBand; + else + pPrevBand->mpNextBand = pBand->mpNextBand; + + pBand = pBand->mpNextBand; + delete pOldBand; + } + else + { + // fixup + if ( bBTEqual ) + pBand->mnYBottom = pBand->mpNextBand->mnYTop-1; + + // this and next band with equal separations? -> combine! + if ( pBand->mpNextBand && + ((pBand->mnYBottom+1) == pBand->mpNextBand->mnYTop) && + (*pBand == *pBand->mpNextBand) ) + { + // expand current height + pBand->mnYBottom = pBand->mpNextBand->mnYBottom; + + // remove next band from list + ImplRegionBand* pDeletedBand = pBand->mpNextBand; + pBand->mpNextBand = pDeletedBand->mpNextBand; + delete pDeletedBand; + + // check band again! + } + else + { + // count rectangles within band + ImplRegionBandSep* pSep = pBand->mpFirstSep; + while ( pSep ) + { + pSep = pSep->mpNextSep; + } + + pPrevBand = pBand; + pBand = pBand->mpNextBand; + } + } + } + +#ifdef DBG_UTIL + pBand = mpFirstBand; + while ( pBand ) + { + DBG_ASSERT( pBand->mpFirstSep != NULL, "Exiting RegionBand::OptimizeBandList(): empty band in region!" ); + + if ( pBand->mnYBottom < pBand->mnYTop ) + OSL_ENSURE(false, "RegionBand::OptimizeBandList(): YBottomBoundary < YTopBoundary" ); + + if ( pBand->mpNextBand ) + { + if ( pBand->mnYBottom >= pBand->mpNextBand->mnYTop ) + OSL_ENSURE(false, "RegionBand::OptimizeBandList(): overlapping bands in region!" ); + } + + pBand = pBand->mpNextBand; + } +#endif + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return (0 != mpFirstBand); +} + +void RegionBand::Move(long nHorzMove, long nVertMove) +{ + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + // process the vertical move + if(nVertMove) + { + pBand->mnYTop = pBand->mnYTop + nVertMove; + pBand->mnYBottom = pBand->mnYBottom + nVertMove; + } + + // process the horizontal move + if(nHorzMove) + { + pBand->MoveX(nHorzMove); + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::Scale(double fScaleX, double fScaleY) +{ + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + // process the vertical move + if(0.0 != fScaleY) + { + pBand->mnYTop = basegfx::fround(pBand->mnYTop * fScaleY); + pBand->mnYBottom = basegfx::fround(pBand->mnYBottom * fScaleY); + } + + // process the horizontal move + if(0.0 != fScaleX) + { + pBand->ScaleX(fScaleX); + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::InsertBands(long nTop, long nBottom) +{ + // region empty? -> set rectagle as first entry! + if ( !mpFirstBand ) + { + // add band with boundaries of the rectangle + mpFirstBand = new ImplRegionBand( nTop, nBottom ); + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return; + } + + // find/insert bands for the boundaries of the rectangle + bool bTopBoundaryInserted = false; + bool bTop2BoundaryInserted = false; + bool bBottomBoundaryInserted = false; + + // special case: top boundary is above the first band + ImplRegionBand* pNewBand; + + if ( nTop < mpFirstBand->mnYTop ) + { + // create new band above the first in the list + pNewBand = new ImplRegionBand( nTop, mpFirstBand->mnYTop ); + + if ( nBottom < mpFirstBand->mnYTop ) + { + pNewBand->mnYBottom = nBottom; + } + + // insert band into the list + pNewBand->mpNextBand = mpFirstBand; + mpFirstBand = pNewBand; + + bTopBoundaryInserted = true; + } + + // insert band(s) into the list + ImplRegionBand* pBand = mpFirstBand; + + while ( pBand ) + { + // Insert Bands if possible + if ( !bTopBoundaryInserted ) + { + bTopBoundaryInserted = InsertSingleBand( pBand, nTop - 1 ); + } + + if ( !bTop2BoundaryInserted ) + { + bTop2BoundaryInserted = InsertSingleBand( pBand, nTop ); + } + + if ( !bBottomBoundaryInserted && (nTop != nBottom) ) + { + bBottomBoundaryInserted = InsertSingleBand( pBand, nBottom ); + } + + // both boundaries inserted? -> nothing more to do + if ( bTopBoundaryInserted && bTop2BoundaryInserted && bBottomBoundaryInserted ) + { + break; + } + + // insert bands between two bands if neccessary + if ( pBand->mpNextBand ) + { + if ( (pBand->mnYBottom + 1) < pBand->mpNextBand->mnYTop ) + { + // copy band with list and set new boundary + pNewBand = new ImplRegionBand( pBand->mnYBottom+1, pBand->mpNextBand->mnYTop-1 ); + + // insert band into the list + pNewBand->mpNextBand = pBand->mpNextBand; + pBand->mpNextBand = pNewBand; + } + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +bool RegionBand::InsertSingleBand(ImplRegionBand* pBand, long nYBandPosition) +{ + // boundary already included in band with height 1? -> nothing to do! + if ( (pBand->mnYTop == pBand->mnYBottom) && (nYBandPosition == pBand->mnYTop) ) + { + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + // insert single height band on top? + ImplRegionBand* pNewBand; + + if ( nYBandPosition == pBand->mnYTop ) + { + // copy band with list and set new boundary + pNewBand = new ImplRegionBand( *pBand ); + pNewBand->mnYTop = nYBandPosition+1; + + // insert band into the list + pNewBand->mpNextBand = pBand->mpNextBand; + pBand->mnYBottom = nYBandPosition; + pBand->mpNextBand = pNewBand; + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + // top of new rectangle within the current band? -> insert new band and copy data + if ( (nYBandPosition > pBand->mnYTop) && (nYBandPosition < pBand->mnYBottom) ) + { + // copy band with list and set new boundary + pNewBand = new ImplRegionBand( *pBand ); + pNewBand->mnYTop = nYBandPosition; + + // insert band into the list + pNewBand->mpNextBand = pBand->mpNextBand; + pBand->mnYBottom = nYBandPosition; + pBand->mpNextBand = pNewBand; + + // copy band with list and set new boundary + pNewBand = new ImplRegionBand( *pBand ); + pNewBand->mnYTop = nYBandPosition; + + // insert band into the list + pBand->mpNextBand->mnYTop = nYBandPosition+1; + + pNewBand->mpNextBand = pBand->mpNextBand; + pBand->mnYBottom = nYBandPosition - 1; + pBand->mpNextBand = pNewBand; + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + // create new band behind the current in the list + if ( !pBand->mpNextBand ) + { + if ( nYBandPosition == pBand->mnYBottom ) + { + // copy band with list and set new boundary + pNewBand = new ImplRegionBand( *pBand ); + pNewBand->mnYTop = pBand->mnYBottom; + pNewBand->mnYBottom = nYBandPosition; + + pBand->mnYBottom = nYBandPosition-1; + + // append band to the list + pBand->mpNextBand = pNewBand; + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + + if ( nYBandPosition > pBand->mnYBottom ) + { + // create new band + pNewBand = new ImplRegionBand( pBand->mnYBottom + 1, nYBandPosition ); + + // append band to the list + pBand->mpNextBand = pNewBand; + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; + } + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return false; +} + +void RegionBand::Union(long nLeft, long nTop, long nRight, long nBottom) +{ + DBG_ASSERT( nLeft <= nRight, "RegionBand::Union() - nLeft > nRight" ); + DBG_ASSERT( nTop <= nBottom, "RegionBand::Union() - nTop > nBottom" ); + + // process union + ImplRegionBand* pBand = mpFirstBand; + while ( pBand ) + { + if ( pBand->mnYTop >= nTop ) + { + if ( pBand->mnYBottom <= nBottom ) + pBand->Union( nLeft, nRight ); + else + { +#ifdef DBG_UTIL + long nCurY = pBand->mnYBottom; + pBand = pBand->mpNextBand; + while ( pBand ) + { + if ( (pBand->mnYTop < nCurY) || (pBand->mnYBottom < nCurY) ) + { + OSL_ENSURE(false, "RegionBand::Union() - Bands not sorted!" ); + } + pBand = pBand->mpNextBand; + } +#endif + break; + } + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::Intersect(long nLeft, long nTop, long nRight, long nBottom) +{ + // process intersections + ImplRegionBand* pPrevBand = 0; + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + // band within intersection boundary? -> process. otherwise remove + if((pBand->mnYTop >= nTop) && (pBand->mnYBottom <= nBottom)) + { + // process intersection + pBand->Intersect(nLeft, nRight); + pPrevBand = pBand; + pBand = pBand->mpNextBand; + } + else + { + ImplRegionBand* pOldBand = pBand; + + if(pBand == mpFirstBand) + { + mpFirstBand = pBand->mpNextBand; + } + else + { + pPrevBand->mpNextBand = pBand->mpNextBand; + } + + pBand = pBand->mpNextBand; + delete pOldBand; + } + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::Union(const RegionBand& rSource) +{ + // apply all rectangles from rSource to this + ImplRegionBand* pBand = rSource.mpFirstBand; + + while ( pBand ) + { + // insert bands if the boundaries are not allready in the list + InsertBands(pBand->mnYTop, pBand->mnYBottom); + + // process all elements of the list + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while(pSep) + { + Union(pSep->mnXLeft, pBand->mnYTop, pSep->mnXRight, pBand->mnYBottom); + pSep = pSep->mpNextSep; + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::Exclude(long nLeft, long nTop, long nRight, long nBottom) +{ + DBG_ASSERT( nLeft <= nRight, "RegionBand::Exclude() - nLeft > nRight" ); + DBG_ASSERT( nTop <= nBottom, "RegionBand::Exclude() - nTop > nBottom" ); + + // process exclude + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + if(pBand->mnYTop >= nTop) + { + if(pBand->mnYBottom <= nBottom) + { + pBand->Exclude(nLeft, nRight); + } + else + { +#ifdef DBG_UTIL + long nCurY = pBand->mnYBottom; + pBand = pBand->mpNextBand; + + while(pBand) + { + if((pBand->mnYTop < nCurY) || (pBand->mnYBottom < nCurY)) + { + OSL_ENSURE(false, "RegionBand::Exclude() - Bands not sorted!" ); + } + + pBand = pBand->mpNextBand; + } +#endif + break; + } + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::XOr(long nLeft, long nTop, long nRight, long nBottom) +{ + DBG_ASSERT( nLeft <= nRight, "RegionBand::Exclude() - nLeft > nRight" ); + DBG_ASSERT( nTop <= nBottom, "RegionBand::Exclude() - nTop > nBottom" ); + + // process xor + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + if(pBand->mnYTop >= nTop) + { + if(pBand->mnYBottom <= nBottom) + { + pBand->XOr(nLeft, nRight); + } + else + { +#ifdef DBG_UTIL + long nCurY = pBand->mnYBottom; + pBand = pBand->mpNextBand; + + while(pBand) + { + if((pBand->mnYTop < nCurY) || (pBand->mnYBottom < nCurY)) + { + OSL_ENSURE(false, "RegionBand::XOr() - Bands not sorted!" ); + } + + pBand = pBand->mpNextBand; + } +#endif + break; + } + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +void RegionBand::Intersect(const RegionBand& rSource) +{ + // mark all bands as untouched + ImplRegionBand* pBand = mpFirstBand; + + while ( pBand ) + { + pBand->mbTouched = false; + pBand = pBand->mpNextBand; + } + + pBand = rSource.mpFirstBand; + + while ( pBand ) + { + // insert bands if the boundaries are not allready in the list + InsertBands( pBand->mnYTop, pBand->mnYBottom ); + + // process all elements of the list + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while ( pSep ) + { + // left boundary? + if ( pSep == pBand->mpFirstSep ) + { + // process intersection and do not remove untouched bands + Exclude( LONG_MIN+1, pBand->mnYTop, pSep->mnXLeft-1, pBand->mnYBottom ); + } + + // right boundary? + if ( pSep->mpNextSep == NULL ) + { + // process intersection and do not remove untouched bands + Exclude( pSep->mnXRight+1, pBand->mnYTop, LONG_MAX-1, pBand->mnYBottom ); + } + else + { + // process intersection and do not remove untouched bands + Exclude( pSep->mnXRight+1, pBand->mnYTop, pSep->mpNextSep->mnXLeft-1, pBand->mnYBottom ); + } + + pSep = pSep->mpNextSep; + } + + pBand = pBand->mpNextBand; + } + + // remove all untouched bands if bands allready left + ImplRegionBand* pPrevBand = 0; + pBand = mpFirstBand; + + while ( pBand ) + { + if ( !pBand->mbTouched ) + { + // save pointer + ImplRegionBand* pOldBand = pBand; + + // previous element of the list + if ( pBand == mpFirstBand ) + { + mpFirstBand = pBand->mpNextBand; + } + else + { + pPrevBand->mpNextBand = pBand->mpNextBand; + } + + pBand = pBand->mpNextBand; + delete pOldBand; + } + else + { + pPrevBand = pBand; + pBand = pBand->mpNextBand; + } + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); +} + +bool RegionBand::Exclude(const RegionBand& rSource) +{ + // Alle Rechtecke aus der uebergebenen Region auf diese Region anwenden + ImplRegionBand* pBand = rSource.mpFirstBand; + + while ( pBand ) + { + // insert bands if the boundaries are not allready in the list + InsertBands( pBand->mnYTop, pBand->mnYBottom ); + + // process all elements of the list + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while ( pSep ) + { + Exclude( pSep->mnXLeft, pBand->mnYTop, pSep->mnXRight, pBand->mnYBottom ); + pSep = pSep->mpNextSep; + } + + // to test less bands, already check in the loop + if ( !OptimizeBandList() ) + { + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return false; + } + + pBand = pBand->mpNextBand; + } + + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + return true; +} + +Rectangle RegionBand::GetBoundRect() const +{ + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + + // get the boundaries of the first band + long nYTop(mpFirstBand->mnYTop); + long nYBottom(mpFirstBand->mnYBottom); + long nXLeft(mpFirstBand->GetXLeftBoundary()); + long nXRight(mpFirstBand->GetXRightBoundary()); + + // look in the band list (don't test first band again!) + ImplRegionBand* pBand = mpFirstBand->mpNextBand; + + while ( pBand ) + { + nYBottom = pBand->mnYBottom; + nXLeft = std::min( nXLeft, pBand->GetXLeftBoundary() ); + nXRight = std::max( nXRight, pBand->GetXRightBoundary() ); + + pBand = pBand->mpNextBand; + } + + return Rectangle( nXLeft, nYTop, nXRight, nYBottom ); +} + +void RegionBand::XOr(const RegionBand& rSource) +{ + ImplRegionBand* pBand = rSource.mpFirstBand; + + while ( pBand ) + { + // insert bands if the boundaries are not allready in the list + InsertBands( pBand->mnYTop, pBand->mnYBottom ); + + // process all elements of the list + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while ( pSep ) + { + XOr( pSep->mnXLeft, pBand->mnYTop, pSep->mnXRight, pBand->mnYBottom ); + pSep = pSep->mpNextSep; + } + + pBand = pBand->mpNextBand; + } +} + +bool RegionBand::IsInside(const Point& rPoint) const +{ + DBG_CHKTHIS(RegionBand, ImplDbgTestRegionBand); + + // search band list + ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + // is point within band? + if((pBand->mnYTop <= rPoint.Y()) && (pBand->mnYBottom >= rPoint.Y())) + { + // is point within separation of the band? + if(pBand->IsInside(rPoint.X())) + { + return true; + } + else + { + return false; + } + } + + pBand = pBand->mpNextBand; + } + + return false; +} + +void RegionBand::GetRegionRectangles(RectangleVector& rTarget) const +{ + // clear result vector + rTarget.clear(); + ImplRegionBand* mpCurrRectBand = mpFirstBand; + Rectangle aRectangle; + + while(mpCurrRectBand) + { + ImplRegionBandSep* mpCurrRectBandSep = mpCurrRectBand->mpFirstSep; + + aRectangle.Top() = mpCurrRectBand->mnYTop; + aRectangle.Bottom() = mpCurrRectBand->mnYBottom; + + while(mpCurrRectBandSep) + { + aRectangle.Left() = mpCurrRectBandSep->mnXLeft; + aRectangle.Right() = mpCurrRectBandSep->mnXRight; + rTarget.push_back(aRectangle); + mpCurrRectBandSep = mpCurrRectBandSep->mpNextSep; + } + + mpCurrRectBand = mpCurrRectBand->mpNextBand; + } +} + +sal_uInt32 RegionBand::getRectangleCount() const +{ + sal_uInt32 nCount = 0; + const ImplRegionBand* pBand = mpFirstBand; + + while(pBand) + { + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while(pSep) + { + nCount++; + pSep = pSep->mpNextSep; + } + + pBand = pBand->mpNextBand; + } + + return 0; +} + +#ifdef DBG_UTIL +const char* ImplDbgTestRegionBand(const void* pObj) +{ + const RegionBand* pRegionBand = reinterpret_cast< const RegionBand* >(pObj); + + if(pRegionBand) + { + const ImplRegionBand* pBand = pRegionBand->ImplGetFirstRegionBand(); + + while(pBand) + { + if(pBand->mnYBottom < pBand->mnYTop) + { + return "YBottom < YTop"; + } + + if(pBand->mpNextBand) + { + if(pBand->mnYBottom >= pBand->mpNextBand->mnYTop) + { + return "overlapping bands in region"; + } + } + + if(pBand->mbTouched) + { + return "Band-mbTouched overwrite"; + } + + ImplRegionBandSep* pSep = pBand->mpFirstSep; + + while(pSep) + { + if(pSep->mnXRight < pSep->mnXLeft) + { + return "XLeft < XRight"; + } + + if(pSep->mpNextSep) + { + if(pSep->mnXRight >= pSep->mpNextSep->mnXLeft) + { + return "overlapping separations in region"; + } + } + + if ( pSep->mbRemoved ) + { + return "Sep-mbRemoved overwrite"; + } + + pSep = pSep->mpNextSep; + } + + pBand = pBand->mpNextBand; + } + } + + return 0; +} +#endif + +/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ diff --git a/vcl/source/gdi/salgdilayout.cxx b/vcl/source/gdi/salgdilayout.cxx index 01137b635ae2..529346a7768d 100644 --- a/vcl/source/gdi/salgdilayout.cxx +++ b/vcl/source/gdi/salgdilayout.cxx @@ -31,7 +31,6 @@ #include <vcl/unowrap.hxx> #include <window.h> -#include <region.h> #include <outdev.h> #include <sallayout.hxx> #include <salgdi.hxx> @@ -235,28 +234,38 @@ sal_Bool SalGraphics::mirror( sal_uInt32 nPoints, const SalPoint *pPtAry, SalPoi void SalGraphics::mirror( Region& rRgn, const OutputDevice *pOutDev, bool bBack ) const { - if( rRgn.HasPolyPolygon() ) + if( rRgn.HasPolyPolygonOrB2DPolyPolygon() ) { - basegfx::B2DPolyPolygon aPolyPoly( rRgn.ConvertToB2DPolyPolygon() ); - aPolyPoly = mirror( aPolyPoly, pOutDev, bBack ); - rRgn = Region( aPolyPoly ); + const basegfx::B2DPolyPolygon aPolyPoly(mirror(rRgn.GetAsB2DPolyPolygon(), pOutDev, bBack)); + + rRgn = Region(aPolyPoly); } else { - ImplRegionInfo aInfo; - bool bRegionRect; - Region aMirroredRegion; - long nX, nY, nWidth, nHeight; + RectangleVector aRectangles; + rRgn.GetRegionRectangles(aRectangles); + rRgn.SetEmpty(); - bRegionRect = rRgn.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); - while ( bRegionRect ) + for(RectangleVector::iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - Rectangle aRect( Point(nX, nY), Size(nWidth, nHeight) ); - mirror( aRect, pOutDev, bBack ); - aMirroredRegion.Union( aRect ); - bRegionRect = rRgn.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + mirror(*aRectIter, pOutDev, bBack); + rRgn.Union(*aRectIter); } - rRgn = aMirroredRegion; + + //ImplRegionInfo aInfo; + //bool bRegionRect; + //Region aMirroredRegion; + //long nX, nY, nWidth, nHeight; + // + //bRegionRect = rRgn.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); + //while ( bRegionRect ) + //{ + // Rectangle aRect( Point(nX, nY), Size(nWidth, nHeight) ); + // mirror( aRect, pOutDev, bBack ); + // aMirroredRegion.Union( aRect ); + // bRegionRect = rRgn.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + //} + //rRgn = aMirroredRegion; } } diff --git a/vcl/source/window/window.cxx b/vcl/source/window/window.cxx index 1c8e8908ecc9..b9d89429dade 100644 --- a/vcl/source/window/window.cxx +++ b/vcl/source/window/window.cxx @@ -68,7 +68,6 @@ #include "window.h" #include "toolbox.h" #include "outdev.h" -#include "region.h" #include "brdwin.hxx" #include "helpwin.hxx" #include "sallayout.hxx" @@ -152,8 +151,8 @@ ImplAccessibleInfos::~ImplAccessibleInfos() WindowImpl::WindowImpl( WindowType nType ) { maZoom = Fraction( 1, 1 ); - maWinRegion = Region( REGION_NULL ); - maWinClipRegion = Region( REGION_NULL ); + maWinRegion = Region(true); + maWinClipRegion = Region(true); mpWinData = NULL; // Extra Window Data, that we dont need for all windows mpOverlapData = NULL; // Overlap Data mpFrameData = NULL; // Frame Data @@ -1834,23 +1833,38 @@ sal_Bool Window::ImplSysObjClip( const Region* pOldRegion ) aRegion.Move( -mnOutOffX, -mnOutOffY ); // ClipRegion setzen/updaten - long nX; - long nY; - long nWidth; - long nHeight; - sal_uLong nRectCount; - ImplRegionInfo aInfo; - sal_Bool bRegionRect; - - nRectCount = aRegion.GetRectCount(); - mpWindowImpl->mpSysObj->BeginSetClipRegion( nRectCount ); - bRegionRect = aRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); - while ( bRegionRect ) + RectangleVector aRectangles; + aRegion.GetRegionRectangles(aRectangles); + mpWindowImpl->mpSysObj->BeginSetClipRegion(aRectangles.size()); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - mpWindowImpl->mpSysObj->UnionClipRegion( nX, nY, nWidth, nHeight ); - bRegionRect = aRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + mpWindowImpl->mpSysObj->UnionClipRegion( + aRectIter->Left(), + aRectIter->Top(), + aRectIter->GetWidth(), // orig nWidth was ((R - L) + 1), same as GetWidth does + aRectIter->GetHeight()); // same for height } + mpWindowImpl->mpSysObj->EndSetClipRegion(); + + //long nX; + //long nY; + //long nWidth; + //long nHeight; + //sal_uLong nRectCount; + //ImplRegionInfo aInfo; + //sal_Bool bRegionRect; + // + //nRectCount = aRegion.GetRectCount(); + //mpWindowImpl->mpSysObj->BeginSetClipRegion( nRectCount ); + //bRegionRect = aRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); + //while ( bRegionRect ) + //{ + // mpWindowImpl->mpSysObj->UnionClipRegion( nX, nY, nWidth, nHeight ); + // bRegionRect = aRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + //} + //mpWindowImpl->mpSysObj->EndSetClipRegion(); } } else @@ -2371,6 +2385,9 @@ void Window::ImplCalcOverlapRegion( const Rectangle& rSourceRect, Region& rRegio void Window::ImplCallPaint( const Region* pRegion, sal_uInt16 nPaintFlags ) { + Exception aException; + bool bExceptionCaught(false); + // call PrePaint. PrePaint may add to the invalidate region as well as // other parameters used below. PrePaint(); @@ -2464,7 +2481,19 @@ void Window::ImplCallPaint( const Region* pRegion, sal_uInt16 nPaintFlags ) if( mpWindowImpl->mbDrawSelectionBackground ) aSelectionRect = aPaintRect; - Paint( aPaintRect ); + // Paint can throw exceptions; to not have a situation where + // mpWindowImpl->mbInPaint keeps to be on true (and other + // settings, too) better catch here to avoid to go completely out of + // this method without executing the after-paint stuff + try + { + Paint( aPaintRect ); + } + catch(Exception& rException) + { + aException = rException; + bExceptionCaught = true; + } if ( mpWindowImpl->mpWinData ) { @@ -2504,6 +2533,11 @@ void Window::ImplCallPaint( const Region* pRegion, sal_uInt16 nPaintFlags ) DrawSelectionBackground( aSelectionRect, 3, sal_False, sal_True, sal_False ); delete pChildRegion; + + if(bExceptionCaught) + { + throw(aException); + } } // ----------------------------------------------------------------------- @@ -5865,7 +5899,7 @@ void Window::SetWindowRegionPixel() mpWindowImpl->mpBorderWindow->SetWindowRegionPixel(); else if( mpWindowImpl->mbFrame ) { - mpWindowImpl->maWinRegion = Region( REGION_NULL); + mpWindowImpl->maWinRegion = Region(true); mpWindowImpl->mbWinRegion = sal_False; mpWindowImpl->mpFrame->ResetClipRegion(); } @@ -5873,7 +5907,7 @@ void Window::SetWindowRegionPixel() { if ( mpWindowImpl->mbWinRegion ) { - mpWindowImpl->maWinRegion = Region( REGION_NULL ); + mpWindowImpl->maWinRegion = Region(true); mpWindowImpl->mbWinRegion = sal_False; ImplSetClipFlag(); @@ -5902,30 +5936,46 @@ void Window::SetWindowRegionPixel( const Region& rRegion ) mpWindowImpl->mpBorderWindow->SetWindowRegionPixel( rRegion ); else if( mpWindowImpl->mbFrame ) { - if( rRegion.GetType() != REGION_NULL ) + if( !rRegion.IsNull() ) { mpWindowImpl->maWinRegion = rRegion; mpWindowImpl->mbWinRegion = ! rRegion.IsEmpty(); + if( mpWindowImpl->mbWinRegion ) { // set/update ClipRegion - long nX; - long nY; - long nWidth; - long nHeight; - sal_uLong nRectCount; - ImplRegionInfo aInfo; - sal_Bool bRegionRect; - - nRectCount = mpWindowImpl->maWinRegion.GetRectCount(); - mpWindowImpl->mpFrame->BeginSetClipRegion( nRectCount ); - bRegionRect = mpWindowImpl->maWinRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); - while ( bRegionRect ) + RectangleVector aRectangles; + mpWindowImpl->maWinRegion.GetRegionRectangles(aRectangles); + mpWindowImpl->mpFrame->BeginSetClipRegion(aRectangles.size()); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - mpWindowImpl->mpFrame->UnionClipRegion( nX, nY, nWidth, nHeight ); - bRegionRect = mpWindowImpl->maWinRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + mpWindowImpl->mpFrame->UnionClipRegion( + aRectIter->Left(), + aRectIter->Top(), + aRectIter->GetWidth(), // orig nWidth was ((R - L) + 1), same as GetWidth does + aRectIter->GetHeight()); // same for height } + mpWindowImpl->mpFrame->EndSetClipRegion(); + + //long nX; + //long nY; + //long nWidth; + //long nHeight; + //sal_uLong nRectCount; + //ImplRegionInfo aInfo; + //sal_Bool bRegionRect; + // + //nRectCount = mpWindowImpl->maWinRegion.GetRectCount(); + //mpWindowImpl->mpFrame->BeginSetClipRegion( nRectCount ); + //bRegionRect = mpWindowImpl->maWinRegion.ImplGetFirstRect( aInfo, nX, nY, nWidth, nHeight ); + //while ( bRegionRect ) + //{ + // mpWindowImpl->mpFrame->UnionClipRegion( nX, nY, nWidth, nHeight ); + // bRegionRect = mpWindowImpl->maWinRegion.ImplGetNextRect( aInfo, nX, nY, nWidth, nHeight ); + //} + //mpWindowImpl->mpFrame->EndSetClipRegion(); } else SetWindowRegionPixel(); @@ -5935,11 +5985,11 @@ void Window::SetWindowRegionPixel( const Region& rRegion ) } else { - if ( rRegion.GetType() == REGION_NULL ) + if ( rRegion.IsNull() ) { if ( mpWindowImpl->mbWinRegion ) { - mpWindowImpl->maWinRegion = Region( REGION_NULL ); + mpWindowImpl->maWinRegion = Region(true); mpWindowImpl->mbWinRegion = sal_False; ImplSetClipFlag(); } @@ -6040,7 +6090,7 @@ Region Window::GetPaintRegion() const } else { - Region aPaintRegion( REGION_NULL ); + Region aPaintRegion(true); return aPaintRegion; } } @@ -6290,7 +6340,7 @@ void Window::Show( sal_Bool bVisible, sal_uInt16 nFlags ) if ( mpWindowImpl->mbReallyVisible ) { - Region aInvRegion( REGION_EMPTY ); + Region aInvRegion; sal_Bool bSaveBack = sal_False; if ( ImplIsOverlapWindow() && !mpWindowImpl->mbFrame ) diff --git a/vcl/unx/generic/gdi/salgdi.cxx b/vcl/unx/generic/gdi/salgdi.cxx index 2098457b9e1e..18ca15ece773 100644 --- a/vcl/unx/generic/gdi/salgdi.cxx +++ b/vcl/unx/generic/gdi/salgdi.cxx @@ -52,7 +52,6 @@ #include "generic/printergfx.hxx" #include "xrender_peer.hxx" -#include "region.h" #define STATIC_POINTS 64 @@ -559,24 +558,48 @@ bool X11SalGraphics::setClipRegion( const Region& i_rClip ) XDestroyRegion( mpClipRegion ); mpClipRegion = XCreateRegion(); - ImplRegionInfo aInfo; - long nX, nY, nW, nH; - bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); - while( bRegionRect ) + RectangleVector aRectangles; + i_rClip.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - if ( nW && nH ) + const long nW(aRectIter->GetWidth()); + + if(nW) { - XRectangle aRect; - aRect.x = (short)nX; - aRect.y = (short)nY; - aRect.width = (unsigned short)nW; - aRect.height = (unsigned short)nH; + const long nH(aRectIter->GetHeight()); + + if(nH) + { + XRectangle aRect; - XUnionRectWithRegion( &aRect, mpClipRegion, mpClipRegion ); + aRect.x = (short)aRectIter->Left(); + aRect.y = (short)aRectIter->Top(); + aRect.width = (unsigned short)nW; + aRect.height = (unsigned short)nH; + XUnionRectWithRegion(&aRect, mpClipRegion, mpClipRegion); + } } - bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); } + //ImplRegionInfo aInfo; + //long nX, nY, nW, nH; + //bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); + //while( bRegionRect ) + //{ + // if ( nW && nH ) + // { + // XRectangle aRect; + // aRect.x = (short)nX; + // aRect.y = (short)nY; + // aRect.width = (unsigned short)nW; + // aRect.height = (unsigned short)nH; + // + // XUnionRectWithRegion( &aRect, mpClipRegion, mpClipRegion ); + // } + // bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); + //} + // done, invalidate GCs bPenGC_ = sal_False; bFontGC_ = sal_False; diff --git a/vcl/unx/gtk/gdi/salnativewidgets-gtk.cxx b/vcl/unx/gtk/gdi/salnativewidgets-gtk.cxx index 0a0fe89fabad..8518ab32dcb5 100644 --- a/vcl/unx/gtk/gdi/salnativewidgets-gtk.cxx +++ b/vcl/unx/gtk/gdi/salnativewidgets-gtk.cxx @@ -49,7 +49,7 @@ sal_Bool GtkSalGraphics::bNeedPixmapPaint = sal_False; GtkSalGraphics::GtkSalGraphics( GtkSalFrame *pFrame, GtkWidget *pWindow ) : X11SalGraphics(), m_pWindow( pWindow ), - m_aClipRegion( REGION_NULL ) + m_aClipRegion(true) { Init( pFrame, GDK_WINDOW_XID( widget_get_window( pWindow ) ), SalX11Screen( gdk_x11_screen_get_screen_number( @@ -828,16 +828,29 @@ sal_Bool GtkSalGraphics::drawNativeControl( ControlType nType, } else { - RegionHandle aHdl = aClipRegion.BeginEnumRects(); - Rectangle aPaintRect; - while( aClipRegion.GetNextEnumRect( aHdl, aPaintRect ) ) + RectangleVector aRectangles; + aClipRegion.GetRegionRectangles(aRectangles); + + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - aPaintRect = aCtrlRect.GetIntersection( aPaintRect ); - if( aPaintRect.IsEmpty() ) + if(aRectIter->IsEmpty()) + { continue; - aClip.push_back( aPaintRect ); + } + + aClip.push_back(*aRectIter); } - aClipRegion.EndEnumRects( aHdl ); + + //RegionHandle aHdl = aClipRegion.BeginEnumRects(); + //Rectangle aPaintRect; + //while( aClipRegion.GetEnumRects( aHdl, aPaintRect ) ) + //{ + // aPaintRect = aCtrlRect.GetIntersection( aPaintRect ); + // if( aPaintRect.IsEmpty() ) + // continue; + // aClip.push_back( aPaintRect ); + //} + //aClipRegion.EndEnumRects( aHdl ); } if ( (nType==CTRL_PUSHBUTTON) && (nPart==PART_ENTIRE_CONTROL) ) diff --git a/vcl/unx/gtk3/gdi/gtk3salnativewidgets-gtk.cxx b/vcl/unx/gtk3/gdi/gtk3salnativewidgets-gtk.cxx index 8b9f2d20737e..cc33f0130bed 100644 --- a/vcl/unx/gtk3/gdi/gtk3salnativewidgets-gtk.cxx +++ b/vcl/unx/gtk3/gdi/gtk3salnativewidgets-gtk.cxx @@ -1450,7 +1450,8 @@ void GtkSalGraphics::getStyleContext(GtkStyleContext** style, GtkWidget* widget) GtkSalGraphics::GtkSalGraphics( GtkSalFrame *pFrame, GtkWidget *pWindow ) : SvpSalGraphics(), mpFrame( pFrame ), - mpWindow( pWindow ) + mpWindow( pWindow ), + m_aClipRegion(true) { if(style_loaded) return; diff --git a/vcl/win/source/gdi/salgdi.cxx b/vcl/win/source/gdi/salgdi.cxx index 7bba68e43354..59fde95c7054 100644 --- a/vcl/win/source/gdi/salgdi.cxx +++ b/vcl/win/source/gdi/salgdi.cxx @@ -35,8 +35,6 @@ #include <win/salgdi.h> #include <win/salframe.h> -#include <region.h> - // ======================================================================= // comment out to prevent use of beziers on GDI functions @@ -848,10 +846,9 @@ bool WinSalGraphics::setClipRegion( const Region& i_rClip ) mhRegion = 0; } - if( i_rClip.HasPolyPolygon() ) + if( i_rClip.HasPolyPolygonOrB2DPolyPolygon() ) { - // TODO: ConvertToB2DPolyPolygon actually is kind of const, just it does not advertise it in the header - basegfx::B2DPolyPolygon aPolyPolygon( const_cast<Region&>(i_rClip).ConvertToB2DPolyPolygon() ); + const basegfx::B2DPolyPolygon aPolyPolygon( i_rClip.GetAsB2DPolyPolygon() ); const sal_uInt32 nCount(aPolyPolygon.count()); if( nCount ) @@ -859,12 +856,15 @@ bool WinSalGraphics::setClipRegion( const Region& i_rClip ) std::vector< POINT > aPolyPoints; aPolyPoints.reserve( 1024 ); std::vector< INT > aPolyCounts( nCount, 0 ); + for(sal_uInt32 a(0); a < nCount; a++) { - basegfx::B2DPolygon aPoly( aPolyPolygon.getB2DPolygon(a) ); + basegfx::B2DPolygon aPoly(aPolyPolygon.getB2DPolygon(a)); + aPoly = basegfx::tools::adaptiveSubdivideByDistance( aPoly, 1 ); const sal_uInt32 nPoints = aPoly.count(); aPolyCounts[a] = nPoints; + for( sal_uInt32 b = 0; b < nPoints; b++ ) { basegfx::B2DPoint aPt( aPoly.getB2DPoint( b ) ); @@ -874,15 +874,18 @@ bool WinSalGraphics::setClipRegion( const Region& i_rClip ) aPolyPoints.push_back( aPOINT ); } } + mhRegion = CreatePolyPolygonRgn( &aPolyPoints[0], &aPolyCounts[0], nCount, ALTERNATE ); } } else { - sal_uLong nRectCount = i_rClip.GetRectCount(); + RectangleVector aRectangles; + i_rClip.GetRegionRectangles(aRectangles); - sal_uLong nRectBufSize = sizeof(RECT)*nRectCount; - if ( nRectCount < SAL_CLIPRECT_COUNT ) + //sal_uLong nRectCount = i_rClip.GetRectCount(); + sal_uLong nRectBufSize = sizeof(RECT)*aRectangles.size(); + if ( aRectangles.size() < SAL_CLIPRECT_COUNT ) { if ( !mpStdClipRgnData ) mpStdClipRgnData = (RGNDATA*)new BYTE[sizeof(RGNDATA)-1+(SAL_CLIPRECT_COUNT*sizeof(RECT))]; @@ -892,50 +895,58 @@ bool WinSalGraphics::setClipRegion( const Region& i_rClip ) mpClipRgnData = (RGNDATA*)new BYTE[sizeof(RGNDATA)-1+nRectBufSize]; mpClipRgnData->rdh.dwSize = sizeof( RGNDATAHEADER ); mpClipRgnData->rdh.iType = RDH_RECTANGLES; - mpClipRgnData->rdh.nCount = nRectCount; + mpClipRgnData->rdh.nCount = aRectangles.size(); mpClipRgnData->rdh.nRgnSize = nRectBufSize; RECT* pBoundRect = &(mpClipRgnData->rdh.rcBound); SetRectEmpty( pBoundRect ); RECT* pNextClipRect = (RECT*)(&(mpClipRgnData->Buffer)); bool bFirstClipRect = true; - ImplRegionInfo aInfo; - long nX, nY, nW, nH; - bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); - while( bRegionRect ) + for(RectangleVector::const_iterator aRectIter(aRectangles.begin()); aRectIter != aRectangles.end(); aRectIter++) { - if ( nW && nH ) + const long nW(aRectIter->GetWidth()); + const long nH(aRectIter->GetHeight()); + + if(nW && nH) { - long nRight = nX + nW; - long nBottom = nY + nH; + const long nRight(aRectIter->Left() + nW); + const long nBottom(aRectIter->Top() + nH); - if ( bFirstClipRect ) + if(bFirstClipRect) { - pBoundRect->left = nX; - pBoundRect->top = nY; - pBoundRect->right = nRight; - pBoundRect->bottom = nBottom; + pBoundRect->left = aRectIter->Left(); + pBoundRect->top = aRectIter->Top(); + pBoundRect->right = nRight; + pBoundRect->bottom = nBottom; bFirstClipRect = false; } else { - if ( nX < pBoundRect->left ) - pBoundRect->left = (int)nX; + if(aRectIter->Left() < pBoundRect->left) + { + pBoundRect->left = (int)aRectIter->Left(); + } - if ( nY < pBoundRect->top ) - pBoundRect->top = (int)nY; + if(aRectIter->Top() < pBoundRect->top) + { + pBoundRect->top = (int)aRectIter->Top(); + } - if ( nRight > pBoundRect->right ) + if(nRight > pBoundRect->right) + { pBoundRect->right = (int)nRight; + } - if ( nBottom > pBoundRect->bottom ) + if(nBottom > pBoundRect->bottom) + { pBoundRect->bottom = (int)nBottom; + } } - pNextClipRect->left = (int)nX; - pNextClipRect->top = (int)nY; - pNextClipRect->right = (int)nRight; - pNextClipRect->bottom = (int)nBottom; + pNextClipRect->left = (int)aRectIter->Left(); + pNextClipRect->top = (int)aRectIter->Top(); + pNextClipRect->right = (int)nRight; + pNextClipRect->bottom = (int)nBottom; pNextClipRect++; } else @@ -943,8 +954,55 @@ bool WinSalGraphics::setClipRegion( const Region& i_rClip ) mpClipRgnData->rdh.nCount--; mpClipRgnData->rdh.nRgnSize -= sizeof( RECT ); } - bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); } + + //ImplRegionInfo aInfo; + //long nX, nY, nW, nH; + //bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH ); + //while( bRegionRect ) + //{ + // if ( nW && nH ) + // { + // long nRight = nX + nW; + // long nBottom = nY + nH; + // + // if ( bFirstClipRect ) + // { + // pBoundRect->left = nX; + // pBoundRect->top = nY; + // pBoundRect->right = nRight; + // pBoundRect->bottom = nBottom; + // bFirstClipRect = false; + // } + // else + // { + // if ( nX < pBoundRect->left ) + // pBoundRect->left = (int)nX; + // + // if ( nY < pBoundRect->top ) + // pBoundRect->top = (int)nY; + // + // if ( nRight > pBoundRect->right ) + // pBoundRect->right = (int)nRight; + // + // if ( nBottom > pBoundRect->bottom ) + // pBoundRect->bottom = (int)nBottom; + // } + // + // pNextClipRect->left = (int)nX; + // pNextClipRect->top = (int)nY; + // pNextClipRect->right = (int)nRight; + // pNextClipRect->bottom = (int)nBottom; + // pNextClipRect++; + // } + // else + // { + // mpClipRgnData->rdh.nCount--; + // mpClipRgnData->rdh.nRgnSize -= sizeof( RECT ); + // } + // bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH ); + //} + // create clip region from ClipRgnData if ( mpClipRgnData->rdh.nCount == 1 ) { |