Commit | Line | Data |
---|---|---|
c801d85f | 1 | ///////////////////////////////////////////////////////////////////////////// |
e4db172a | 2 | // Name: src/gtk/region.cpp |
c801d85f KB |
3 | // Purpose: |
4 | // Author: Robert Roebling | |
9fe4c99c | 5 | // Modified: VZ at 05.10.00: use AllocExclusive(), comparison fixed |
f96aa4d9 RR |
6 | // Id: $Id$ |
7 | // Copyright: (c) 1998 Robert Roebling | |
65571936 | 8 | // Licence: wxWindows licence |
c801d85f KB |
9 | ///////////////////////////////////////////////////////////////////////////// |
10 | ||
1e6feb95 VZ |
11 | // ============================================================================ |
12 | // declarations | |
13 | // ============================================================================ | |
14 | ||
1e6feb95 VZ |
15 | // ---------------------------------------------------------------------------- |
16 | // headers | |
17 | // ---------------------------------------------------------------------------- | |
18 | ||
14f355c2 VS |
19 | // For compilers that support precompilation, includes "wx.h". |
20 | #include "wx/wxprec.h" | |
21 | ||
b3a44e05 WS |
22 | #include "wx/region.h" |
23 | ||
e4db172a WS |
24 | #ifndef WX_PRECOMP |
25 | #include "wx/log.h" | |
26 | #endif | |
27 | ||
9e691f46 | 28 | #include "wx/gtk/private.h" |
1e6feb95 | 29 | |
57f2b902 | 30 | |
1e6feb95 VZ |
31 | // ---------------------------------------------------------------------------- |
32 | // wxRegionRefData: private class containing the information about the region | |
33 | // ---------------------------------------------------------------------------- | |
c801d85f | 34 | |
8f884a0d | 35 | class wxRegionRefData : public wxGDIRefData |
c801d85f | 36 | { |
864e8bd0 | 37 | public: |
48850fa7 RR |
38 | wxRegionRefData() |
39 | { | |
40 | m_region = NULL; | |
41 | } | |
57f2b902 | 42 | |
48850fa7 | 43 | wxRegionRefData(const wxRegionRefData& refData) |
8f884a0d | 44 | : wxGDIRefData() |
48850fa7 | 45 | { |
48850fa7 | 46 | m_region = gdk_region_copy(refData.m_region); |
48850fa7 | 47 | } |
57f2b902 | 48 | |
d3c7fc99 | 49 | virtual ~wxRegionRefData() |
48850fa7 RR |
50 | { |
51 | if (m_region) | |
52 | gdk_region_destroy( m_region ); | |
53 | } | |
5fc7ede9 | 54 | |
c801d85f KB |
55 | GdkRegion *m_region; |
56 | }; | |
57 | ||
1e6feb95 VZ |
58 | // ---------------------------------------------------------------------------- |
59 | // macros | |
60 | // ---------------------------------------------------------------------------- | |
61 | ||
62 | #define M_REGIONDATA ((wxRegionRefData *)m_refData) | |
63 | #define M_REGIONDATA_OF(rgn) ((wxRegionRefData *)(rgn.m_refData)) | |
64 | ||
d84afea9 GD |
65 | IMPLEMENT_DYNAMIC_CLASS(wxRegion, wxGDIObject) |
66 | IMPLEMENT_DYNAMIC_CLASS(wxRegionIterator,wxObject) | |
1e6feb95 | 67 | |
1e6feb95 VZ |
68 | // ---------------------------------------------------------------------------- |
69 | // wxRegion construction | |
70 | // ---------------------------------------------------------------------------- | |
c801d85f KB |
71 | |
72 | #define M_REGIONDATA ((wxRegionRefData *)m_refData) | |
73 | ||
9fe4c99c | 74 | void wxRegion::InitRect(wxCoord x, wxCoord y, wxCoord w, wxCoord h) |
c801d85f | 75 | { |
bbe0af5b RR |
76 | GdkRectangle rect; |
77 | rect.x = x; | |
78 | rect.y = y; | |
79 | rect.width = w; | |
80 | rect.height = h; | |
57f2b902 | 81 | |
48850fa7 | 82 | m_refData = new wxRegionRefData(); |
57f2b902 | 83 | |
9e691f46 | 84 | M_REGIONDATA->m_region = gdk_region_rectangle( &rect ); |
ff7b1510 | 85 | } |
c801d85f | 86 | |
b15ed747 RR |
87 | wxRegion::wxRegion( GdkRegion *region ) |
88 | { | |
89 | m_refData = new wxRegionRefData(); | |
90 | M_REGIONDATA->m_region = gdk_region_copy( region ); | |
91 | } | |
92 | ||
03647350 | 93 | wxRegion::wxRegion( size_t n, const wxPoint *points, |
89efaf2b | 94 | wxPolygonFillMode fillStyle ) |
5549e9f7 VZ |
95 | { |
96 | GdkPoint *gdkpoints = new GdkPoint[n]; | |
97 | for ( size_t i = 0 ; i < n ; i++ ) | |
98 | { | |
99 | gdkpoints[i].x = points[i].x; | |
100 | gdkpoints[i].y = points[i].y; | |
101 | } | |
102 | ||
103 | m_refData = new wxRegionRefData(); | |
104 | ||
105 | GdkRegion* reg = gdk_region_polygon | |
106 | ( | |
107 | gdkpoints, | |
108 | n, | |
109 | fillStyle == wxWINDING_RULE ? GDK_WINDING_RULE | |
110 | : GDK_EVEN_ODD_RULE | |
111 | ); | |
112 | ||
113 | M_REGIONDATA->m_region = reg; | |
114 | ||
115 | delete [] gdkpoints; | |
116 | } | |
117 | ||
1e6feb95 | 118 | wxRegion::~wxRegion() |
c801d85f | 119 | { |
e0f0b197 | 120 | // m_refData unrefed in ~wxObject |
ff7b1510 | 121 | } |
c801d85f | 122 | |
8f884a0d | 123 | wxGDIRefData *wxRegion::CreateGDIRefData() const |
e0f0b197 RR |
124 | { |
125 | return new wxRegionRefData; | |
126 | } | |
127 | ||
8f884a0d | 128 | wxGDIRefData *wxRegion::CloneGDIRefData(const wxGDIRefData *data) const |
e0f0b197 RR |
129 | { |
130 | return new wxRegionRefData(*(wxRegionRefData *)data); | |
131 | } | |
c89f5c02 | 132 | |
9fe4c99c VZ |
133 | // ---------------------------------------------------------------------------- |
134 | // wxRegion comparison | |
135 | // ---------------------------------------------------------------------------- | |
136 | ||
8a16d737 | 137 | bool wxRegion::DoIsEqual(const wxRegion& region) const |
c801d85f | 138 | { |
1e6feb95 VZ |
139 | return gdk_region_equal(M_REGIONDATA->m_region, |
140 | M_REGIONDATA_OF(region)->m_region); | |
ff7b1510 | 141 | } |
c801d85f | 142 | |
1e6feb95 VZ |
143 | // ---------------------------------------------------------------------------- |
144 | // wxRegion operations | |
145 | // ---------------------------------------------------------------------------- | |
146 | ||
bf57d1ad | 147 | void wxRegion::Clear() |
c801d85f | 148 | { |
bbe0af5b | 149 | UnRef(); |
ff7b1510 | 150 | } |
c801d85f | 151 | |
8a16d737 | 152 | bool wxRegion::DoUnionWithRect(const wxRect& r) |
c801d85f | 153 | { |
57351df0 VZ |
154 | // workaround for a strange GTK/X11 bug: taking union with an empty |
155 | // rectangle results in an empty region which is definitely not what we | |
156 | // want | |
8a16d737 | 157 | if ( r.IsEmpty() ) |
b3a44e05 | 158 | return true; |
57351df0 | 159 | |
2b5f62a0 | 160 | if ( !m_refData ) |
e1208c31 | 161 | { |
8a16d737 | 162 | InitRect(r.x, r.y, r.width, r.height); |
e1208c31 RR |
163 | } |
164 | else | |
165 | { | |
9fe4c99c | 166 | AllocExclusive(); |
1e6feb95 | 167 | |
2b5f62a0 | 168 | GdkRectangle rect; |
8a16d737 VZ |
169 | rect.x = r.x; |
170 | rect.y = r.y; | |
171 | rect.width = r.width; | |
172 | rect.height = r.height; | |
2b5f62a0 | 173 | |
b7c2d6ff | 174 | gdk_region_union_with_rect( M_REGIONDATA->m_region, &rect ); |
e1208c31 | 175 | } |
1e6feb95 | 176 | |
b3a44e05 | 177 | return true; |
ff7b1510 | 178 | } |
c801d85f | 179 | |
8a16d737 | 180 | bool wxRegion::DoUnionWithRegion( const wxRegion& region ) |
c801d85f | 181 | { |
a1b806b9 | 182 | wxCHECK_MSG( region.IsOk(), false, wxT("invalid region") ); |
e1208c31 | 183 | |
48850fa7 RR |
184 | if (!m_refData) |
185 | { | |
186 | m_refData = new wxRegionRefData(); | |
187 | M_REGIONDATA->m_region = gdk_region_new(); | |
188 | } | |
189 | else | |
190 | { | |
191 | AllocExclusive(); | |
192 | } | |
e1208c31 | 193 | |
b7c2d6ff | 194 | gdk_region_union( M_REGIONDATA->m_region, region.GetRegion() ); |
5fc7ede9 | 195 | |
b3a44e05 | 196 | return true; |
ff7b1510 | 197 | } |
c801d85f | 198 | |
8a16d737 | 199 | bool wxRegion::DoIntersect( const wxRegion& region ) |
c801d85f | 200 | { |
a1b806b9 | 201 | wxCHECK_MSG( region.IsOk(), false, wxT("invalid region") ); |
1e6feb95 | 202 | |
e1208c31 RR |
203 | if (!m_refData) |
204 | { | |
2b5f62a0 | 205 | // intersecting with invalid region doesn't make sense |
b3a44e05 | 206 | return false; |
1e6feb95 VZ |
207 | } |
208 | ||
2b5f62a0 VZ |
209 | AllocExclusive(); |
210 | ||
48850fa7 | 211 | gdk_region_intersect( M_REGIONDATA->m_region, region.GetRegion() ); |
9fe4c99c | 212 | |
b3a44e05 | 213 | return true; |
ff7b1510 | 214 | } |
c801d85f | 215 | |
8a16d737 | 216 | bool wxRegion::DoSubtract( const wxRegion& region ) |
c801d85f | 217 | { |
a1b806b9 | 218 | wxCHECK_MSG( region.IsOk(), false, wxT("invalid region") ); |
e1208c31 RR |
219 | |
220 | if (!m_refData) | |
221 | { | |
2b5f62a0 | 222 | // subtracting from an invalid region doesn't make sense |
b3a44e05 | 223 | return false; |
48850fa7 | 224 | } |
1e6feb95 | 225 | |
2b5f62a0 VZ |
226 | AllocExclusive(); |
227 | ||
b7c2d6ff | 228 | gdk_region_subtract( M_REGIONDATA->m_region, region.GetRegion() ); |
1e6feb95 | 229 | |
b3a44e05 | 230 | return true; |
ff7b1510 | 231 | } |
c801d85f | 232 | |
8a16d737 | 233 | bool wxRegion::DoXor( const wxRegion& region ) |
c801d85f | 234 | { |
a1b806b9 | 235 | wxCHECK_MSG( region.IsOk(), false, wxT("invalid region") ); |
e1208c31 RR |
236 | |
237 | if (!m_refData) | |
238 | { | |
b3a44e05 | 239 | return false; |
1e6feb95 | 240 | } |
e1208c31 | 241 | |
2b5f62a0 VZ |
242 | AllocExclusive(); |
243 | ||
b7c2d6ff | 244 | gdk_region_xor( M_REGIONDATA->m_region, region.GetRegion() ); |
5fc7ede9 | 245 | |
b3a44e05 | 246 | return true; |
ff7b1510 | 247 | } |
c801d85f | 248 | |
8a16d737 | 249 | bool wxRegion::DoOffset( wxCoord x, wxCoord y ) |
2b5f62a0 VZ |
250 | { |
251 | if (!m_refData) | |
b3a44e05 | 252 | return false; |
2b5f62a0 VZ |
253 | |
254 | AllocExclusive(); | |
255 | ||
256 | gdk_region_offset( M_REGIONDATA->m_region, x, y ); | |
257 | ||
b3a44e05 | 258 | return true; |
2b5f62a0 VZ |
259 | } |
260 | ||
1e6feb95 VZ |
261 | // ---------------------------------------------------------------------------- |
262 | // wxRegion tests | |
263 | // ---------------------------------------------------------------------------- | |
264 | ||
8a16d737 | 265 | bool wxRegion::DoGetBox( wxCoord &x, wxCoord &y, wxCoord &w, wxCoord &h ) const |
c801d85f | 266 | { |
1e6feb95 VZ |
267 | if ( m_refData ) |
268 | { | |
269 | GdkRectangle rect; | |
270 | gdk_region_get_clipbox( M_REGIONDATA->m_region, &rect ); | |
271 | x = rect.x; | |
272 | y = rect.y; | |
273 | w = rect.width; | |
274 | h = rect.height; | |
8a16d737 VZ |
275 | |
276 | return true; | |
1e6feb95 VZ |
277 | } |
278 | else | |
279 | { | |
280 | x = 0; | |
281 | y = 0; | |
282 | w = -1; | |
283 | h = -1; | |
c801d85f | 284 | |
8a16d737 VZ |
285 | return false; |
286 | } | |
ff7b1510 | 287 | } |
c801d85f | 288 | |
8a16d737 | 289 | bool wxRegion::IsEmpty() const |
c801d85f | 290 | { |
e1208c31 | 291 | if (!m_refData) |
b3a44e05 | 292 | return true; |
e1208c31 | 293 | |
bbe0af5b | 294 | return gdk_region_empty( M_REGIONDATA->m_region ); |
ff7b1510 | 295 | } |
c801d85f | 296 | |
8a16d737 | 297 | wxRegionContain wxRegion::DoContainsPoint( wxCoord x, wxCoord y ) const |
c801d85f | 298 | { |
e1208c31 RR |
299 | if (!m_refData) |
300 | return wxOutRegion; | |
301 | ||
bbe0af5b RR |
302 | if (gdk_region_point_in( M_REGIONDATA->m_region, x, y )) |
303 | return wxInRegion; | |
304 | else | |
305 | return wxOutRegion; | |
ff7b1510 | 306 | } |
c801d85f | 307 | |
8a16d737 | 308 | wxRegionContain wxRegion::DoContainsRect(const wxRect& r) const |
c801d85f | 309 | { |
e1208c31 RR |
310 | if (!m_refData) |
311 | return wxOutRegion; | |
312 | ||
bbe0af5b | 313 | GdkRectangle rect; |
8a16d737 VZ |
314 | rect.x = r.x; |
315 | rect.y = r.y; | |
316 | rect.width = r.width; | |
317 | rect.height = r.height; | |
bbe0af5b RR |
318 | GdkOverlapType res = gdk_region_rect_in( M_REGIONDATA->m_region, &rect ); |
319 | switch (res) | |
320 | { | |
321 | case GDK_OVERLAP_RECTANGLE_IN: return wxInRegion; | |
322 | case GDK_OVERLAP_RECTANGLE_OUT: return wxOutRegion; | |
323 | case GDK_OVERLAP_RECTANGLE_PART: return wxPartRegion; | |
324 | } | |
325 | return wxOutRegion; | |
ff7b1510 | 326 | } |
c801d85f | 327 | |
bf57d1ad | 328 | GdkRegion *wxRegion::GetRegion() const |
c801d85f | 329 | { |
e1208c31 | 330 | if (!m_refData) |
d3b9f782 | 331 | return NULL; |
e1208c31 | 332 | |
bbe0af5b | 333 | return M_REGIONDATA->m_region; |
ff7b1510 | 334 | } |
c801d85f | 335 | |
1e6feb95 | 336 | // ---------------------------------------------------------------------------- |
3d0c4d2e | 337 | // wxRegionIterator |
1e6feb95 | 338 | // ---------------------------------------------------------------------------- |
8429bec1 | 339 | |
55ccdb93 | 340 | wxRegionIterator::wxRegionIterator() |
3d0c4d2e | 341 | { |
55ccdb93 VZ |
342 | Init(); |
343 | Reset(); | |
344 | } | |
3d0c4d2e | 345 | |
55ccdb93 VZ |
346 | wxRegionIterator::wxRegionIterator( const wxRegion& region ) |
347 | { | |
348 | Init(); | |
349 | Reset(region); | |
350 | } | |
3d0c4d2e | 351 | |
55ccdb93 | 352 | void wxRegionIterator::Init() |
3d0c4d2e | 353 | { |
55ccdb93 VZ |
354 | m_rects = NULL; |
355 | m_numRects = 0; | |
3d0c4d2e RR |
356 | } |
357 | ||
55ccdb93 | 358 | wxRegionIterator::~wxRegionIterator() |
3d0c4d2e | 359 | { |
55ccdb93 VZ |
360 | wxDELETEA(m_rects); |
361 | } | |
48850fa7 | 362 | |
55ccdb93 VZ |
363 | void wxRegionIterator::CreateRects( const wxRegion& region ) |
364 | { | |
365 | wxDELETEA(m_rects); | |
366 | m_numRects = 0; | |
57f2b902 | 367 | |
48850fa7 | 368 | GdkRegion *gdkregion = region.GetRegion(); |
3f0fb1d4 VZ |
369 | if (!gdkregion) |
370 | return; | |
57f2b902 | 371 | |
9e691f46 | 372 | GdkRectangle *gdkrects = NULL; |
48850fa7 | 373 | gint numRects = 0; |
9e691f46 | 374 | gdk_region_get_rectangles( gdkregion, &gdkrects, &numRects ); |
57f2b902 | 375 | |
48850fa7 RR |
376 | m_numRects = numRects; |
377 | if (numRects) | |
378 | { | |
379 | m_rects = new wxRect[m_numRects]; | |
380 | for (size_t i=0; i < m_numRects; ++i) | |
3d0c4d2e | 381 | { |
48850fa7 RR |
382 | GdkRectangle &gr = gdkrects[i]; |
383 | wxRect &wr = m_rects[i]; | |
384 | wr.x = gr.x; | |
385 | wr.y = gr.y; | |
386 | wr.width = gr.width; | |
387 | wr.height = gr.height; | |
3d0c4d2e | 388 | } |
3d0c4d2e | 389 | } |
9e691f46 | 390 | g_free( gdkrects ); |
3d0c4d2e RR |
391 | } |
392 | ||
3d0c4d2e RR |
393 | void wxRegionIterator::Reset( const wxRegion& region ) |
394 | { | |
395 | m_region = region; | |
55ccdb93 | 396 | CreateRects(region); |
3d0c4d2e RR |
397 | Reset(); |
398 | } | |
399 | ||
400 | bool wxRegionIterator::HaveRects() const | |
401 | { | |
55ccdb93 | 402 | return m_current < m_numRects; |
3d0c4d2e RR |
403 | } |
404 | ||
2b5f62a0 | 405 | wxRegionIterator& wxRegionIterator::operator ++ () |
3d0c4d2e | 406 | { |
2b5f62a0 VZ |
407 | if (HaveRects()) |
408 | ++m_current; | |
3d0c4d2e | 409 | |
2b5f62a0 | 410 | return *this; |
3d0c4d2e RR |
411 | } |
412 | ||
2b5f62a0 | 413 | wxRegionIterator wxRegionIterator::operator ++ (int) |
3d0c4d2e | 414 | { |
2b5f62a0 | 415 | wxRegionIterator tmp = *this; |
55ccdb93 | 416 | |
2b5f62a0 VZ |
417 | if (HaveRects()) |
418 | ++m_current; | |
419 | ||
420 | return tmp; | |
3d0c4d2e RR |
421 | } |
422 | ||
423 | wxCoord wxRegionIterator::GetX() const | |
424 | { | |
9a83f860 | 425 | wxCHECK_MSG( HaveRects(), 0, wxT("invalid wxRegionIterator") ); |
2b5f62a0 | 426 | |
55ccdb93 | 427 | return m_rects[m_current].x; |
3d0c4d2e RR |
428 | } |
429 | ||
430 | wxCoord wxRegionIterator::GetY() const | |
431 | { | |
9a83f860 | 432 | wxCHECK_MSG( HaveRects(), 0, wxT("invalid wxRegionIterator") ); |
2b5f62a0 | 433 | |
55ccdb93 | 434 | return m_rects[m_current].y; |
3d0c4d2e RR |
435 | } |
436 | ||
437 | wxCoord wxRegionIterator::GetW() const | |
438 | { | |
9a83f860 | 439 | wxCHECK_MSG( HaveRects(), 0, wxT("invalid wxRegionIterator") ); |
2b5f62a0 | 440 | |
55ccdb93 | 441 | return m_rects[m_current].width; |
3d0c4d2e RR |
442 | } |
443 | ||
444 | wxCoord wxRegionIterator::GetH() const | |
445 | { | |
9a83f860 | 446 | wxCHECK_MSG( HaveRects(), 0, wxT("invalid wxRegionIterator") ); |
2b5f62a0 | 447 | |
55ccdb93 | 448 | return m_rects[m_current].height; |
3d0c4d2e RR |
449 | } |
450 | ||
1e6feb95 VZ |
451 | wxRect wxRegionIterator::GetRect() const |
452 | { | |
453 | wxRect r; | |
3379ed37 | 454 | if( HaveRects() ) |
55ccdb93 | 455 | r = m_rects[m_current]; |
1e6feb95 VZ |
456 | |
457 | return r; | |
458 | } | |
ac7d3dd1 RR |
459 | |
460 | wxRegionIterator& wxRegionIterator::operator=(const wxRegionIterator& ri) | |
461 | { | |
462 | wxDELETEA(m_rects); | |
f4322df6 | 463 | |
ac7d3dd1 RR |
464 | m_current = ri.m_current; |
465 | m_numRects = ri.m_numRects; | |
466 | if ( m_numRects ) | |
467 | { | |
468 | m_rects = new wxRect[m_numRects]; | |
de3cfa6e | 469 | for ( unsigned int n = 0; n < m_numRects; n++ ) |
ac7d3dd1 RR |
470 | m_rects[n] = ri.m_rects[n]; |
471 | } | |
472 | else | |
473 | { | |
474 | m_rects = NULL; | |
475 | } | |
476 | ||
477 | return *this; | |
478 | } |