1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:      src/msw/region.cpp 
   3 // Purpose:   wxRegion implementation using Win32 API 
   4 // Author:    Vadim Zeitlin 
   6 // Created:   Fri Oct 24 10:46:34 MET 1997 
   8 // Copyright: (c) 1997-2002 wxWidgets team 
   9 // Licence:   wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  27 #include "wx/region.h" 
  30     #include "wx/gdicmn.h" 
  33 #include "wx/msw/private.h" 
  35 IMPLEMENT_DYNAMIC_CLASS(wxRegion
, wxGDIObject
) 
  36 IMPLEMENT_DYNAMIC_CLASS(wxRegionIterator
, wxObject
) 
  38 // ---------------------------------------------------------------------------- 
  39 // wxRegionRefData implementation 
  40 // ---------------------------------------------------------------------------- 
  42 class WXDLLEXPORT wxRegionRefData 
: public wxGDIRefData
 
  50     wxRegionRefData(const wxRegionRefData
& data
) : wxGDIRefData() 
  52 #if defined(__WIN32__) && !defined(__WXMICROWIN__) && !defined(__WXWINCE__) 
  53         DWORD noBytes 
= ::GetRegionData(data
.m_region
, 0, NULL
); 
  54         RGNDATA 
*rgnData 
= (RGNDATA
*) new char[noBytes
]; 
  55         ::GetRegionData(data
.m_region
, noBytes
, rgnData
); 
  56         m_region 
= ::ExtCreateRegion(NULL
, noBytes
, rgnData
); 
  57         delete[] (char*) rgnData
; 
  60         ::GetRgnBox(data
.m_region
, &rect
); 
  61         m_region 
= ::CreateRectRgnIndirect(&rect
); 
  65     virtual ~wxRegionRefData() 
  67         ::DeleteObject(m_region
); 
  75 //  DECLARE_NO_COPY_CLASS(wxRegionRefData) 
  76 // because copy constructor is explicitly declared above; 
  77 // but no copy assignment operator is defined, so declare 
  78 // it private to prevent the compiler from defining it: 
  79     wxRegionRefData
& operator=(const wxRegionRefData
&); 
  82 #define M_REGION (((wxRegionRefData*)m_refData)->m_region) 
  83 #define M_REGION_OF(rgn) (((wxRegionRefData*)(rgn.m_refData))->m_region) 
  85 // ============================================================================ 
  86 // wxRegion implementation 
  87 // ============================================================================ 
  89 // ---------------------------------------------------------------------------- 
  91 // ---------------------------------------------------------------------------- 
  95     m_refData 
= (wxRegionRefData 
*)NULL
; 
  98 wxRegion::wxRegion(WXHRGN hRegion
) 
 100     m_refData 
= new wxRegionRefData
; 
 101     M_REGION 
= (HRGN
) hRegion
; 
 104 wxRegion::wxRegion(wxCoord x
, wxCoord y
, wxCoord w
, wxCoord h
) 
 106     m_refData 
= new wxRegionRefData
; 
 107     M_REGION 
= ::CreateRectRgn(x
, y
, x 
+ w
, y 
+ h
); 
 110 wxRegion::wxRegion(const wxPoint
& topLeft
, const wxPoint
& bottomRight
) 
 112     m_refData 
= new wxRegionRefData
; 
 113     M_REGION 
= ::CreateRectRgn(topLeft
.x
, topLeft
.y
, bottomRight
.x
, bottomRight
.y
); 
 116 wxRegion::wxRegion(const wxRect
& rect
) 
 118     m_refData 
= new wxRegionRefData
; 
 119     M_REGION 
= ::CreateRectRgn(rect
.x
, rect
.y
, rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height
); 
 122 wxRegion::wxRegion(size_t n
, const wxPoint 
*points
, int fillStyle
) 
 124 #if defined(__WXMICROWIN__) || defined(__WXWINCE__) 
 127     wxUnusedVar(fillStyle
); 
 131     m_refData 
= new wxRegionRefData
; 
 132     M_REGION 
= ::CreatePolygonRgn
 
 136                     fillStyle 
== wxODDEVEN_RULE 
? ALTERNATE 
: WINDING
 
 141 wxRegion::~wxRegion() 
 143     // m_refData unrefed in ~wxObject 
 146 wxObjectRefData 
*wxRegion::CreateRefData() const 
 148     return new wxRegionRefData
; 
 151 wxObjectRefData 
*wxRegion::CloneRefData(const wxObjectRefData 
*data
) const 
 153     return new wxRegionRefData(*(wxRegionRefData 
*)data
); 
 156 // ---------------------------------------------------------------------------- 
 157 // wxRegion operations 
 158 // ---------------------------------------------------------------------------- 
 160 // Clear current region 
 161 void wxRegion::Clear() 
 166 bool wxRegion::DoOffset(wxCoord x
, wxCoord y
) 
 168     wxCHECK_MSG( M_REGION
, false, _T("invalid wxRegion") ); 
 178     if ( ::OffsetRgn(GetHrgn(), x
, y
) == ERROR 
) 
 180         wxLogLastError(_T("OffsetRgn")); 
 188 // combine another region with this one 
 189 bool wxRegion::DoCombine(const wxRegion
& rgn
, wxRegionOp op
) 
 191     // we can't use the API functions if we don't have a valid region handle 
 194         // combining with an empty/invalid region works differently depending 
 205                 wxFAIL_MSG( _T("unknown region operation") ); 
 210                 // leave empty/invalid 
 214     else // we have a valid region 
 238                 wxFAIL_MSG( _T("unknown region operation") ); 
 246         if ( ::CombineRgn(M_REGION
, M_REGION
, M_REGION_OF(rgn
), mode
) == ERROR 
) 
 248             wxLogLastError(_T("CombineRgn")); 
 257 // ---------------------------------------------------------------------------- 
 258 // wxRegion bounding box 
 259 // ---------------------------------------------------------------------------- 
 261 // Outer bounds of region 
 262 bool wxRegion::DoGetBox(wxCoord
& x
, wxCoord
& y
, wxCoord
&w
, wxCoord 
&h
) const 
 267         ::GetRgnBox(M_REGION
, & rect
); 
 270         w 
= rect
.right 
- rect
.left
; 
 271         h 
= rect
.bottom 
- rect
.top
; 
 284 bool wxRegion::IsEmpty() const 
 289     return (w 
== 0) && (h 
== 0); 
 292 bool wxRegion::DoIsEqual(const wxRegion
& region
) const 
 294     return ::EqualRgn(M_REGION
, M_REGION_OF(region
)) != 0; 
 297 // ---------------------------------------------------------------------------- 
 298 // wxRegion hit testing 
 299 // ---------------------------------------------------------------------------- 
 301 // Does the region contain the point (x,y)? 
 302 wxRegionContain 
wxRegion::DoContainsPoint(wxCoord x
, wxCoord y
) const 
 307     return ::PtInRegion(M_REGION
, (int) x
, (int) y
) ? wxInRegion 
: wxOutRegion
; 
 310 // Does the region contain the rectangle (x, y, w, h)? 
 311 wxRegionContain 
wxRegion::DoContainsRect(const wxRect
& rect
) const 
 317     wxCopyRectToRECT(rect
, rc
); 
 319     return ::RectInRegion(M_REGION
, &rc
) ? wxInRegion 
: wxOutRegion
; 
 322 // Get internal region handle 
 323 WXHRGN 
wxRegion::GetHRGN() const 
 325     return (WXHRGN
)(m_refData 
? M_REGION 
: 0); 
 328 // ============================================================================ 
 329 // wxRegionIterator implementation 
 330 // ============================================================================ 
 332 // ---------------------------------------------------------------------------- 
 333 // wxRegionIterator ctors/dtor 
 334 // ---------------------------------------------------------------------------- 
 336 void wxRegionIterator::Init() 
 344 wxRegionIterator::~wxRegionIterator() 
 349 // Initialize iterator for region 
 350 wxRegionIterator::wxRegionIterator(const wxRegion
& region
) 
 357 wxRegionIterator
& wxRegionIterator::operator=(const wxRegionIterator
& ri
) 
 361     m_current 
= ri
.m_current
; 
 362     m_numRects 
= ri
.m_numRects
; 
 365         m_rects 
= new wxRect
[m_numRects
]; 
 366         for ( long n 
= 0; n 
< m_numRects
; n
++ ) 
 367             m_rects
[n
] = ri
.m_rects
[n
]; 
 377 // ---------------------------------------------------------------------------- 
 378 // wxRegionIterator operations 
 379 // ---------------------------------------------------------------------------- 
 381 // Reset iterator for a new region. 
 382 void wxRegionIterator::Reset(const wxRegion
& region
) 
 394     if (m_region
.Empty()) 
 398         DWORD noBytes 
= ::GetRegionData(((wxRegionRefData
*)region
.m_refData
)->m_region
, 0, NULL
); 
 399         RGNDATA 
*rgnData 
= (RGNDATA
*) new char[noBytes
]; 
 400         ::GetRegionData(((wxRegionRefData
*)region
.m_refData
)->m_region
, noBytes
, rgnData
); 
 402         RGNDATAHEADER
* header 
= (RGNDATAHEADER
*) rgnData
; 
 404         m_rects 
= new wxRect
[header
->nCount
]; 
 406         RECT
* rect 
= (RECT
*) ((char*)rgnData 
+ sizeof(RGNDATAHEADER
)); 
 408         for (i 
= 0; i 
< header
->nCount
; i
++) 
 410             m_rects
[i
] = wxRect(rect
->left
, rect
->top
, 
 411                                  rect
->right 
- rect
->left
, rect
->bottom 
- rect
->top
); 
 412             rect 
++; // Advances pointer by sizeof(RECT) 
 415         m_numRects 
= header
->nCount
; 
 417         delete[] (char*) rgnData
; 
 421 wxRegionIterator
& wxRegionIterator::operator++() 
 423     if (m_current 
< m_numRects
) 
 429 wxRegionIterator 
wxRegionIterator::operator ++ (int) 
 431     wxRegionIterator tmp 
= *this; 
 432     if (m_current 
< m_numRects
) 
 438 // ---------------------------------------------------------------------------- 
 439 // wxRegionIterator accessors 
 440 // ---------------------------------------------------------------------------- 
 442 wxCoord 
wxRegionIterator::GetX() const 
 444     wxCHECK_MSG( m_current 
< m_numRects
, 0, _T("invalid wxRegionIterator") ); 
 446     return m_rects
[m_current
].x
; 
 449 wxCoord 
wxRegionIterator::GetY() const 
 451     wxCHECK_MSG( m_current 
< m_numRects
, 0, _T("invalid wxRegionIterator") ); 
 453     return m_rects
[m_current
].y
; 
 456 wxCoord 
wxRegionIterator::GetW() const 
 458     wxCHECK_MSG( m_current 
< m_numRects
, 0, _T("invalid wxRegionIterator") ); 
 460     return m_rects
[m_current
].width
; 
 463 wxCoord 
wxRegionIterator::GetH() const 
 465     wxCHECK_MSG( m_current 
< m_numRects
, 0, _T("invalid wxRegionIterator") ); 
 467     return m_rects
[m_current
].height
;