]> git.saurik.com Git - wxWidgets.git/blame - src/gtk1/dcclient.cpp
MoreFilesX 1.0.1: A collection of useful high-level File Manager routines which
[wxWidgets.git] / src / gtk1 / dcclient.cpp
CommitLineData
c801d85f 1/////////////////////////////////////////////////////////////////////////////
e5131165 2// Name: gtk/dcclient.cpp
c801d85f
KB
3// Purpose:
4// Author: Robert Roebling
6f65e337 5// RCS-ID: $Id$
01111366 6// Copyright: (c) 1998 Robert Roebling, Markus Holzem, Chris Breeze
e5131165 7// Licence: wxWindows licence
c801d85f
KB
8/////////////////////////////////////////////////////////////////////////////
9
10#ifdef __GNUG__
11#pragma implementation "dcclient.h"
12#endif
13
d02af7bb
JJ
14#ifdef __VMS
15#define XCopyPlane XCOPYPLANE
16#endif
17
c801d85f 18#include "wx/dcclient.h"
6f65e337 19#include "wx/dcmemory.h"
4bc67cc5 20#include "wx/image.h"
e5131165 21#include "wx/module.h"
b1699cd3 22#include "wx/log.h"
db16cab4 23#include "wx/fontutil.h"
e5131165 24
ed673c6a 25#include "wx/gtk/win_gtk.h"
c801d85f 26
071a2d78 27#include <math.h> // for floating-point functions
e5131165 28
071a2d78 29#include <gdk/gdk.h>
993f97ee 30#include <gdk/gdkx.h>
8943b403 31#include <gdk/gdkprivate.h>
071a2d78 32#include <gtk/gtk.h>
83624f79 33
809934d2
RR
34//-----------------------------------------------------------------------------
35// local defines
36//-----------------------------------------------------------------------------
37
e1208c31 38#define USE_PAINT_REGION 1
809934d2 39
c801d85f
KB
40//-----------------------------------------------------------------------------
41// local data
42//-----------------------------------------------------------------------------
43
44#include "bdiag.xbm"
45#include "fdiag.xbm"
46#include "cdiag.xbm"
47#include "horiz.xbm"
48#include "verti.xbm"
49#include "cross.xbm"
50#define num_hatches 6
51
3ca6a5f0
BP
52#define IS_15_PIX_HATCH(s) ((s)==wxCROSSDIAG_HATCH || (s)==wxHORIZONTAL_HATCH || (s)==wxVERTICAL_HATCH)
53#define IS_16_PIX_HATCH(s) ((s)!=wxCROSSDIAG_HATCH && (s)!=wxHORIZONTAL_HATCH && (s)!=wxVERTICAL_HATCH)
54
55
c801d85f 56static GdkPixmap *hatches[num_hatches];
c67daf87 57static GdkPixmap **hatch_bitmap = (GdkPixmap **) NULL;
c801d85f 58
c2fa61e8 59extern GtkWidget *wxGetRootWindow();
3d2d8da1 60
c801d85f
KB
61//-----------------------------------------------------------------------------
62// constants
63//-----------------------------------------------------------------------------
64
95724b1a 65const double RAD2DEG = 180.0 / M_PI;
c801d85f 66
9a8c7620
VZ
67// ----------------------------------------------------------------------------
68// private functions
69// ----------------------------------------------------------------------------
70
71static inline double dmax(double a, double b) { return a > b ? a : b; }
72static inline double dmin(double a, double b) { return a < b ? a : b; }
73
74static inline double DegToRad(double deg) { return (deg * M_PI) / 180.0; }
75
6f65e337
JS
76//-----------------------------------------------------------------------------
77// temporary implementation of the missing GDK function
78//-----------------------------------------------------------------------------
b0e0d661 79
6f65e337 80#include "gdk/gdkprivate.h"
b0e0d661 81
1e6feb95
VZ
82void gdk_wx_draw_bitmap(GdkDrawable *drawable,
83 GdkGC *gc,
84 GdkDrawable *src,
85 gint xsrc,
86 gint ysrc,
87 gint xdest,
88 gint ydest,
89 gint width,
90 gint height)
6f65e337 91{
8943b403 92 gint src_width, src_height;
3419e8e7 93#ifndef __WXGTK20__
265898fd
RR
94 GdkWindowPrivate *drawable_private;
95 GdkWindowPrivate *src_private;
96 GdkGCPrivate *gc_private;
8943b403 97#endif
265898fd
RR
98
99 g_return_if_fail (drawable != NULL);
100 g_return_if_fail (src != NULL);
101 g_return_if_fail (gc != NULL);
102
8943b403
OK
103#ifdef __WXGTK20__
104 if (GDK_WINDOW_DESTROYED(drawable) || GDK_WINDOW_DESTROYED(src))
105 return;
106
107 gdk_drawable_get_size(src, &src_width, &src_height);
108#else
265898fd
RR
109 drawable_private = (GdkWindowPrivate*) drawable;
110 src_private = (GdkWindowPrivate*) src;
111 if (drawable_private->destroyed || src_private->destroyed)
112 return;
7d5af6fa 113
8943b403
OK
114 src_width = src_private->width;
115 src_height = src_private->height;
116
265898fd 117 gc_private = (GdkGCPrivate*) gc;
8943b403 118#endif
265898fd 119
8943b403
OK
120 if (width == -1) width = src_width;
121 if (height == -1) height = src_height;
265898fd 122
8943b403
OK
123#ifdef __WXGTK20__
124 XCopyPlane( GDK_WINDOW_XDISPLAY(drawable),
125 GDK_WINDOW_XID(src),
126 GDK_WINDOW_XID(drawable),
127 GDK_GC_XGC(gc),
128 xsrc, ysrc,
129 width, height,
130 xdest, ydest,
131 1 );
132#else
265898fd 133 XCopyPlane( drawable_private->xdisplay,
b0e0d661
VZ
134 src_private->xwindow,
135 drawable_private->xwindow,
136 gc_private->xgc,
137 xsrc, ysrc,
138 width, height,
139 xdest, ydest,
140 1 );
8943b403 141#endif
6f65e337
JS
142}
143
3d2d8da1
RR
144//-----------------------------------------------------------------------------
145// Implement Pool of Graphic contexts. Creating them takes too much time.
146//-----------------------------------------------------------------------------
147
e1208c31
RR
148#define GC_POOL_SIZE 200
149
0e09f76e
RR
150enum wxPoolGCType
151{
152 wxGC_ERROR = 0,
153 wxTEXT_MONO,
154 wxBG_MONO,
155 wxPEN_MONO,
156 wxBRUSH_MONO,
157 wxTEXT_COLOUR,
158 wxBG_COLOUR,
159 wxPEN_COLOUR,
f6bcfd97
BP
160 wxBRUSH_COLOUR,
161 wxTEXT_SCREEN,
162 wxBG_SCREEN,
163 wxPEN_SCREEN,
164 wxBRUSH_SCREEN
0e09f76e
RR
165};
166
3d2d8da1
RR
167struct wxGC
168{
0e09f76e
RR
169 GdkGC *m_gc;
170 wxPoolGCType m_type;
171 bool m_used;
3d2d8da1
RR
172};
173
696f36b7
JS
174#define GC_POOL_ALLOC_SIZE 100
175
176static int wxGCPoolSize = 0;
177
178static wxGC *wxGCPool = NULL;
3d2d8da1
RR
179
180static void wxInitGCPool()
181{
696f36b7
JS
182 // This really could wait until the first call to
183 // wxGetPoolGC, but we will make the first allocation
184 // now when other initialization is being performed.
185
186 // Set initial pool size.
187 wxGCPoolSize = GC_POOL_ALLOC_SIZE;
188
189 // Allocate initial pool.
190 wxGCPool = (wxGC *)malloc(wxGCPoolSize * sizeof(wxGC));
191 if (wxGCPool == NULL)
192 {
193 // If we cannot malloc, then fail with error
194 // when debug is enabled. If debug is not enabled,
195 // the problem will eventually get caught
196 // in wxGetPoolGC.
197 wxFAIL_MSG( wxT("Cannot allocate GC pool") );
198 return;
199 }
200
201 // Zero initial pool.
202 memset(wxGCPool, 0, wxGCPoolSize * sizeof(wxGC));
3d2d8da1
RR
203}
204
205static void wxCleanUpGCPool()
206{
f6116855 207 for (int i = 0; i < wxGCPoolSize; i++)
3d2d8da1
RR
208 {
209 if (wxGCPool[i].m_gc)
210 gdk_gc_unref( wxGCPool[i].m_gc );
211 }
696f36b7
JS
212
213 free(wxGCPool);
214 wxGCPool = NULL;
215 wxGCPoolSize = 0;
3d2d8da1
RR
216}
217
0e09f76e 218static GdkGC* wxGetPoolGC( GdkWindow *window, wxPoolGCType type )
3d2d8da1 219{
696f36b7
JS
220 wxGC *pptr;
221
222 // Look for an available GC.
223 for (int i = 0; i < wxGCPoolSize; i++)
3d2d8da1
RR
224 {
225 if (!wxGCPool[i].m_gc)
226 {
227 wxGCPool[i].m_gc = gdk_gc_new( window );
228 gdk_gc_set_exposures( wxGCPool[i].m_gc, FALSE );
0e09f76e 229 wxGCPool[i].m_type = type;
3d2d8da1
RR
230 wxGCPool[i].m_used = FALSE;
231 }
0e09f76e 232 if ((!wxGCPool[i].m_used) && (wxGCPool[i].m_type == type))
3d2d8da1
RR
233 {
234 wxGCPool[i].m_used = TRUE;
235 return wxGCPool[i].m_gc;
236 }
237 }
5f170f33 238
696f36b7
JS
239 // We did not find an available GC.
240 // We need to grow the GC pool.
241 pptr = (wxGC *)realloc(wxGCPool,
242 (wxGCPoolSize + GC_POOL_ALLOC_SIZE)*sizeof(wxGC));
243 if (pptr != NULL)
244 {
245 // Initialize newly allocated pool.
246 wxGCPool = pptr;
247 memset(&wxGCPool[wxGCPoolSize], 0,
248 GC_POOL_ALLOC_SIZE*sizeof(wxGC));
249
250 // Initialize entry we will return.
251 wxGCPool[wxGCPoolSize].m_gc = gdk_gc_new( window );
252 gdk_gc_set_exposures( wxGCPool[wxGCPoolSize].m_gc, FALSE );
253 wxGCPool[wxGCPoolSize].m_type = type;
254 wxGCPool[wxGCPoolSize].m_used = TRUE;
255
256 // Set new value of pool size.
257 wxGCPoolSize += GC_POOL_ALLOC_SIZE;
258
259 // Return newly allocated entry.
260 return wxGCPool[wxGCPoolSize-GC_POOL_ALLOC_SIZE].m_gc;
261 }
262
263 // The realloc failed. Fall through to error.
3d2d8da1 264 wxFAIL_MSG( wxT("No GC available") );
5f170f33 265
3d2d8da1
RR
266 return (GdkGC*) NULL;
267}
268
269static void wxFreePoolGC( GdkGC *gc )
270{
696f36b7 271 for (int i = 0; i < wxGCPoolSize; i++)
3d2d8da1
RR
272 {
273 if (wxGCPool[i].m_gc == gc)
274 {
275 wxGCPool[i].m_used = FALSE;
276 return;
277 }
278 }
5f170f33 279
3d2d8da1
RR
280 wxFAIL_MSG( wxT("Wrong GC") );
281}
282
c801d85f 283//-----------------------------------------------------------------------------
ec758a20 284// wxWindowDC
c801d85f
KB
285//-----------------------------------------------------------------------------
286
b0e0d661 287IMPLEMENT_DYNAMIC_CLASS(wxWindowDC, wxDC)
c801d85f 288
4bc67cc5 289wxWindowDC::wxWindowDC()
c801d85f 290{
265898fd
RR
291 m_penGC = (GdkGC *) NULL;
292 m_brushGC = (GdkGC *) NULL;
293 m_textGC = (GdkGC *) NULL;
294 m_bgGC = (GdkGC *) NULL;
295 m_cmap = (GdkColormap *) NULL;
296 m_isMemDC = FALSE;
e1208c31 297 m_isScreenDC = FALSE;
bbbbe360 298 m_owner = (wxWindow *)NULL;
8943b403
OK
299#ifdef __WXGTK20__
300 m_context = (PangoContext *)NULL;
301 m_fontdesc = (PangoFontDescription *)NULL;
302#endif
903f689b 303}
c801d85f 304
ec758a20 305wxWindowDC::wxWindowDC( wxWindow *window )
c801d85f 306{
3cd0b8c5
RR
307 wxASSERT_MSG( window, wxT("DC needs a window") );
308
265898fd
RR
309 m_penGC = (GdkGC *) NULL;
310 m_brushGC = (GdkGC *) NULL;
311 m_textGC = (GdkGC *) NULL;
312 m_bgGC = (GdkGC *) NULL;
313 m_cmap = (GdkColormap *) NULL;
bbbbe360 314 m_owner = (wxWindow *)NULL;
6db90681 315 m_isMemDC = FALSE;
e1208c31 316 m_isScreenDC = FALSE;
3b245d60 317 m_font = window->GetFont();
7d5af6fa 318
a2053b27 319 GtkWidget *widget = window->m_wxwindow;
7d5af6fa 320
b7f1f77f
VZ
321 // some controls don't have m_wxwindow - like wxStaticBox, but the user
322 // code should still be able to create wxClientDCs for them, so we will
323 // use the parent window here then
324 if ( !widget )
325 {
326 window = window->GetParent();
327 widget = window->m_wxwindow;
328 }
329
223d09f6 330 wxASSERT_MSG( widget, wxT("DC needs a widget") );
7d5af6fa 331
8943b403 332#ifdef __WXGTK20__
2b5f62a0 333 m_context = window->GtkGetPangoDefaultContext();
8943b403
OK
334 m_fontdesc = widget->style->font_desc;
335#endif
336
da048e3d
RR
337 GtkPizza *pizza = GTK_PIZZA( widget );
338 m_window = pizza->bin_window;
7d5af6fa 339
1e133b7d
RR
340 /* not realized ? */
341 if (!m_window)
342 {
6db90681 343 /* don't report problems */
7d5af6fa
VZ
344 m_ok = TRUE;
345
346 return;
1e133b7d 347 }
7d5af6fa 348
b7f1f77f 349 m_cmap = gtk_widget_get_colormap( widget ? widget : window->m_widget );
7d5af6fa 350
265898fd 351 SetUpDC();
bbbbe360
RR
352
353 /* this must be done after SetUpDC, bacause SetUpDC calls the
354 repective SetBrush, SetPen, SetBackground etc functions
355 to set up the DC. SetBackground call m_owner->SetBackground
356 and this might not be desired as the standard dc background
357 is white whereas a window might assume gray to be the
358 standard (as e.g. wxStatusBar) */
7d5af6fa 359
bbbbe360 360 m_owner = window;
903f689b 361}
c801d85f 362
4bc67cc5 363wxWindowDC::~wxWindowDC()
c801d85f 364{
265898fd 365 Destroy();
903f689b 366}
c801d85f 367
809934d2
RR
368void wxWindowDC::SetUpDC()
369{
370 m_ok = TRUE;
5f170f33 371
809934d2 372 wxASSERT_MSG( !m_penGC, wxT("GCs already created") );
5f170f33 373
f6bcfd97
BP
374 if (m_isScreenDC)
375 {
376 m_penGC = wxGetPoolGC( m_window, wxPEN_SCREEN );
377 m_brushGC = wxGetPoolGC( m_window, wxBRUSH_SCREEN );
378 m_textGC = wxGetPoolGC( m_window, wxTEXT_SCREEN );
379 m_bgGC = wxGetPoolGC( m_window, wxBG_SCREEN );
380 }
381 else
809934d2
RR
382 if (m_isMemDC && (((wxMemoryDC*)this)->m_selected.GetDepth() == 1))
383 {
384 m_penGC = wxGetPoolGC( m_window, wxPEN_MONO );
385 m_brushGC = wxGetPoolGC( m_window, wxBRUSH_MONO );
386 m_textGC = wxGetPoolGC( m_window, wxTEXT_MONO );
387 m_bgGC = wxGetPoolGC( m_window, wxBG_MONO );
388 }
389 else
390 {
391 m_penGC = wxGetPoolGC( m_window, wxPEN_COLOUR );
392 m_brushGC = wxGetPoolGC( m_window, wxBRUSH_COLOUR );
393 m_textGC = wxGetPoolGC( m_window, wxTEXT_COLOUR );
394 m_bgGC = wxGetPoolGC( m_window, wxBG_COLOUR );
395 }
396
397 /* background colour */
398 m_backgroundBrush = *wxWHITE_BRUSH;
399 m_backgroundBrush.GetColour().CalcPixel( m_cmap );
400 GdkColor *bg_col = m_backgroundBrush.GetColour().GetColor();
401
402 /* m_textGC */
403 m_textForegroundColour.CalcPixel( m_cmap );
404 gdk_gc_set_foreground( m_textGC, m_textForegroundColour.GetColor() );
405
406 m_textBackgroundColour.CalcPixel( m_cmap );
407 gdk_gc_set_background( m_textGC, m_textBackgroundColour.GetColor() );
408
409 gdk_gc_set_fill( m_textGC, GDK_SOLID );
410
411 /* m_penGC */
412 m_pen.GetColour().CalcPixel( m_cmap );
413 gdk_gc_set_foreground( m_penGC, m_pen.GetColour().GetColor() );
414 gdk_gc_set_background( m_penGC, bg_col );
5f170f33 415
809934d2 416 gdk_gc_set_line_attributes( m_penGC, 0, GDK_LINE_SOLID, GDK_CAP_NOT_LAST, GDK_JOIN_ROUND );
5f170f33 417
809934d2
RR
418 /* m_brushGC */
419 m_brush.GetColour().CalcPixel( m_cmap );
420 gdk_gc_set_foreground( m_brushGC, m_brush.GetColour().GetColor() );
421 gdk_gc_set_background( m_brushGC, bg_col );
5f170f33 422
809934d2 423 gdk_gc_set_fill( m_brushGC, GDK_SOLID );
5f170f33 424
809934d2
RR
425 /* m_bgGC */
426 gdk_gc_set_background( m_bgGC, bg_col );
427 gdk_gc_set_foreground( m_bgGC, bg_col );
428
429 gdk_gc_set_fill( m_bgGC, GDK_SOLID );
5f170f33 430
809934d2
RR
431 /* ROPs */
432 gdk_gc_set_function( m_textGC, GDK_COPY );
433 gdk_gc_set_function( m_brushGC, GDK_COPY );
434 gdk_gc_set_function( m_penGC, GDK_COPY );
5f170f33 435
809934d2
RR
436 /* clipping */
437 gdk_gc_set_clip_rectangle( m_penGC, (GdkRectangle *) NULL );
438 gdk_gc_set_clip_rectangle( m_brushGC, (GdkRectangle *) NULL );
439 gdk_gc_set_clip_rectangle( m_textGC, (GdkRectangle *) NULL );
440 gdk_gc_set_clip_rectangle( m_bgGC, (GdkRectangle *) NULL );
441
442 if (!hatch_bitmap)
443 {
444 hatch_bitmap = hatches;
445 hatch_bitmap[0] = gdk_bitmap_create_from_data( (GdkWindow *) NULL, bdiag_bits, bdiag_width, bdiag_height );
446 hatch_bitmap[1] = gdk_bitmap_create_from_data( (GdkWindow *) NULL, cdiag_bits, cdiag_width, cdiag_height );
447 hatch_bitmap[2] = gdk_bitmap_create_from_data( (GdkWindow *) NULL, fdiag_bits, fdiag_width, fdiag_height );
448 hatch_bitmap[3] = gdk_bitmap_create_from_data( (GdkWindow *) NULL, cross_bits, cross_width, cross_height );
449 hatch_bitmap[4] = gdk_bitmap_create_from_data( (GdkWindow *) NULL, horiz_bits, horiz_width, horiz_height );
450 hatch_bitmap[5] = gdk_bitmap_create_from_data( (GdkWindow *) NULL, verti_bits, verti_width, verti_height );
451 }
452}
453
376aa62a
VZ
454void wxWindowDC::DoGetSize( int* width, int* height ) const
455{
456 wxCHECK_RET( m_owner, _T("GetSize() doesn't work without window") );
457
458 m_owner->GetSize(width, height);
459}
460
387ebd3e 461extern bool wxDoFloodFill(wxDC *dc, wxCoord x, wxCoord y,
06cf7c08 462 const wxColour & col, int style);
b1699cd3 463
387ebd3e 464bool wxWindowDC::DoFloodFill(wxCoord x, wxCoord y,
06cf7c08
VS
465 const wxColour& col, int style)
466{
387ebd3e 467 return wxDoFloodFill(this, x, y, col, style);
903f689b 468}
c801d85f 469
dc1efb1d 470bool wxWindowDC::DoGetPixel( wxCoord x1, wxCoord y1, wxColour *col ) const
c801d85f 471{
dc1efb1d
JS
472 // Generic (and therefore rather inefficient) method.
473 // Could be improved.
474 wxMemoryDC memdc;
475 wxBitmap bitmap(1, 1);
476 memdc.SelectObject(bitmap);
477 memdc.Blit(0, 0, 1, 1, (wxDC*) this, x1, y1);
478 memdc.SelectObject(wxNullBitmap);
c2fa61e8 479
368d59f0 480 wxImage image = bitmap.ConvertToImage();
dc1efb1d
JS
481 col->Set(image.GetRed(0, 0), image.GetGreen(0, 0), image.GetBlue(0, 0));
482 return TRUE;
903f689b 483}
c801d85f 484
72cdf4c9 485void wxWindowDC::DoDrawLine( wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2 )
c801d85f 486{
223d09f6 487 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 488
265898fd
RR
489 if (m_pen.GetStyle() != wxTRANSPARENT)
490 {
6db90681
RR
491 if (m_window)
492 gdk_draw_line( m_window, m_penGC, XLOG2DEV(x1), YLOG2DEV(y1), XLOG2DEV(x2), YLOG2DEV(y2) );
7d5af6fa 493
ed880dd4
RR
494 CalcBoundingBox(x1, y1);
495 CalcBoundingBox(x2, y2);
265898fd 496 }
903f689b 497}
c801d85f 498
72cdf4c9 499void wxWindowDC::DoCrossHair( wxCoord x, wxCoord y )
c801d85f 500{
223d09f6 501 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 502
265898fd
RR
503 if (m_pen.GetStyle() != wxTRANSPARENT)
504 {
505 int w = 0;
506 int h = 0;
507 GetSize( &w, &h );
72cdf4c9
VZ
508 wxCoord xx = XLOG2DEV(x);
509 wxCoord yy = YLOG2DEV(y);
6db90681 510 if (m_window)
7d5af6fa 511 {
6db90681
RR
512 gdk_draw_line( m_window, m_penGC, 0, yy, XLOG2DEVREL(w), yy );
513 gdk_draw_line( m_window, m_penGC, xx, 0, xx, YLOG2DEVREL(h) );
7d5af6fa 514 }
265898fd 515 }
903f689b 516}
c801d85f 517
72cdf4c9
VZ
518void wxWindowDC::DoDrawArc( wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2,
519 wxCoord xc, wxCoord yc )
c801d85f 520{
223d09f6 521 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 522
72cdf4c9
VZ
523 wxCoord xx1 = XLOG2DEV(x1);
524 wxCoord yy1 = YLOG2DEV(y1);
525 wxCoord xx2 = XLOG2DEV(x2);
526 wxCoord yy2 = YLOG2DEV(y2);
527 wxCoord xxc = XLOG2DEV(xc);
528 wxCoord yyc = YLOG2DEV(yc);
7d5af6fa 529 double dx = xx1 - xxc;
265898fd 530 double dy = yy1 - yyc;
de87c353 531 double radius = sqrt((double)(dx*dx+dy*dy));
72cdf4c9 532 wxCoord r = (wxCoord)radius;
265898fd
RR
533 double radius1, radius2;
534
7d5af6fa 535 if (xx1 == xx2 && yy1 == yy2)
265898fd
RR
536 {
537 radius1 = 0.0;
538 radius2 = 360.0;
7d5af6fa
VZ
539 }
540 else
541 if (radius == 0.0)
265898fd
RR
542 {
543 radius1 = radius2 = 0.0;
7d5af6fa
VZ
544 }
545 else
265898fd
RR
546 {
547 radius1 = (xx1 - xxc == 0) ?
b0e0d661
VZ
548 (yy1 - yyc < 0) ? 90.0 : -90.0 :
549 -atan2(double(yy1-yyc), double(xx1-xxc)) * RAD2DEG;
265898fd 550 radius2 = (xx2 - xxc == 0) ?
b0e0d661
VZ
551 (yy2 - yyc < 0) ? 90.0 : -90.0 :
552 -atan2(double(yy2-yyc), double(xx2-xxc)) * RAD2DEG;
265898fd 553 }
72cdf4c9
VZ
554 wxCoord alpha1 = wxCoord(radius1 * 64.0);
555 wxCoord alpha2 = wxCoord((radius2 - radius1) * 64.0);
265898fd
RR
556 while (alpha2 <= 0) alpha2 += 360*64;
557 while (alpha1 > 360*64) alpha1 -= 360*64;
558
6db90681
RR
559 if (m_window)
560 {
561 if (m_brush.GetStyle() != wxTRANSPARENT)
e9f88d04
VZ
562 {
563 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
564 {
565 gdk_gc_set_ts_origin( m_textGC,
566 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
567 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
568 gdk_draw_arc( m_window, m_textGC, TRUE, xxc-r, yyc-r, 2*r,2*r, alpha1, alpha2 );
569 gdk_gc_set_ts_origin( m_textGC, 0, 0 );
570 } else
3ca6a5f0
BP
571 if (IS_15_PIX_HATCH(m_brush.GetStyle()))
572 {
573 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 15, m_deviceOriginY % 15 );
574 gdk_draw_arc( m_window, m_brushGC, TRUE, xxc-r, yyc-r, 2*r,2*r, alpha1, alpha2 );
575 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
576 } else
577 if (IS_16_PIX_HATCH(m_brush.GetStyle()))
578 {
579 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 16, m_deviceOriginY % 16 );
580 gdk_draw_arc( m_window, m_brushGC, TRUE, xxc-r, yyc-r, 2*r,2*r, alpha1, alpha2 );
581 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
582 } else
e9f88d04
VZ
583 if (m_brush.GetStyle() == wxSTIPPLE)
584 {
585 gdk_gc_set_ts_origin( m_brushGC,
586 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
587 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
588 gdk_draw_arc( m_window, m_brushGC, TRUE, xxc-r, yyc-r, 2*r,2*r, alpha1, alpha2 );
589 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
590 }
591 else
592 {
593 gdk_draw_arc( m_window, m_brushGC, TRUE, xxc-r, yyc-r, 2*r,2*r, alpha1, alpha2 );
594 }
595 }
7d5af6fa 596
6db90681 597 if (m_pen.GetStyle() != wxTRANSPARENT)
3d0c4d2e 598 {
6db90681 599 gdk_draw_arc( m_window, m_penGC, FALSE, xxc-r, yyc-r, 2*r,2*r, alpha1, alpha2 );
f469b27c 600
3d0c4d2e
RR
601 gdk_draw_line( m_window, m_penGC, xx1, yy1, xxc, yyc );
602 gdk_draw_line( m_window, m_penGC, xxc, yyc, xx2, yy2 );
603 }
6db90681 604 }
7d5af6fa 605
265898fd
RR
606 CalcBoundingBox (x1, y1);
607 CalcBoundingBox (x2, y2);
903f689b 608}
c801d85f 609
72cdf4c9 610void wxWindowDC::DoDrawEllipticArc( wxCoord x, wxCoord y, wxCoord width, wxCoord height, double sa, double ea )
c801d85f 611{
223d09f6 612 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 613
72cdf4c9
VZ
614 wxCoord xx = XLOG2DEV(x);
615 wxCoord yy = YLOG2DEV(y);
616 wxCoord ww = m_signX * XLOG2DEVREL(width);
617 wxCoord hh = m_signY * YLOG2DEVREL(height);
7d5af6fa 618
265898fd
RR
619 // CMB: handle -ve width and/or height
620 if (ww < 0) { ww = -ww; xx = xx - ww; }
621 if (hh < 0) { hh = -hh; yy = yy - hh; }
7d5af6fa 622
6db90681
RR
623 if (m_window)
624 {
72cdf4c9 625 wxCoord start = wxCoord(sa * 64.0);
3d0c4d2e 626 wxCoord end = wxCoord((ea-sa) * 64.0);
7d5af6fa 627
6db90681 628 if (m_brush.GetStyle() != wxTRANSPARENT)
e9f88d04
VZ
629 {
630 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
631 {
632 gdk_gc_set_ts_origin( m_textGC,
633 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
634 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
635 gdk_draw_arc( m_window, m_textGC, TRUE, xx, yy, ww, hh, start, end );
636 gdk_gc_set_ts_origin( m_textGC, 0, 0 );
637 } else
3ca6a5f0
BP
638 if (IS_15_PIX_HATCH(m_brush.GetStyle()))
639 {
640 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 15, m_deviceOriginY % 15 );
641 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, start, end );
642 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
643 } else
644 if (IS_16_PIX_HATCH(m_brush.GetStyle()))
645 {
646 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 16, m_deviceOriginY % 16 );
647 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, start, end );
648 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
649 } else
e9f88d04
VZ
650 if (m_brush.GetStyle() == wxSTIPPLE)
651 {
652 gdk_gc_set_ts_origin( m_brushGC,
653 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
654 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
655 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, start, end );
656 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
657 }
658 else
659 {
660 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, start, end );
661 }
662 }
7d5af6fa 663
6db90681
RR
664 if (m_pen.GetStyle() != wxTRANSPARENT)
665 gdk_draw_arc( m_window, m_penGC, FALSE, xx, yy, ww, hh, start, end );
666 }
7d5af6fa 667
265898fd
RR
668 CalcBoundingBox (x, y);
669 CalcBoundingBox (x + width, y + height);
903f689b 670}
c801d85f 671
72cdf4c9 672void wxWindowDC::DoDrawPoint( wxCoord x, wxCoord y )
c801d85f 673{
223d09f6 674 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 675
6db90681 676 if ((m_pen.GetStyle() != wxTRANSPARENT) && m_window)
265898fd 677 gdk_draw_point( m_window, m_penGC, XLOG2DEV(x), YLOG2DEV(y) );
7d5af6fa 678
265898fd 679 CalcBoundingBox (x, y);
903f689b 680}
c801d85f 681
72cdf4c9 682void wxWindowDC::DoDrawLines( int n, wxPoint points[], wxCoord xoffset, wxCoord yoffset )
c801d85f 683{
223d09f6 684 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 685
265898fd
RR
686 if (m_pen.GetStyle() == wxTRANSPARENT) return;
687 if (n <= 0) return;
7d5af6fa 688
265898fd 689 CalcBoundingBox( points[0].x + xoffset, points[0].y + yoffset );
7d5af6fa 690
265898fd
RR
691 for (int i = 0; i < n-1; i++)
692 {
72cdf4c9
VZ
693 wxCoord x1 = XLOG2DEV(points[i].x + xoffset);
694 wxCoord x2 = XLOG2DEV(points[i+1].x + xoffset);
695 wxCoord y1 = YLOG2DEV(points[i].y + yoffset); // oh, what a waste
696 wxCoord y2 = YLOG2DEV(points[i+1].y + yoffset);
7d5af6fa
VZ
697
698 if (m_window)
6db90681 699 gdk_draw_line( m_window, m_penGC, x1, y1, x2, y2 );
7d5af6fa 700
265898fd
RR
701 CalcBoundingBox( points[i+1].x + xoffset, points[i+1].y + yoffset );
702 }
903f689b 703}
c801d85f 704
72cdf4c9 705void wxWindowDC::DoDrawPolygon( int n, wxPoint points[], wxCoord xoffset, wxCoord yoffset, int WXUNUSED(fillStyle) )
cf7a7e13 706{
223d09f6 707 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 708
4bc67cc5 709 if (n <= 0) return;
7d5af6fa 710
265898fd
RR
711 GdkPoint *gdkpoints = new GdkPoint[n+1];
712 int i;
713 for (i = 0 ; i < n ; i++)
714 {
715 gdkpoints[i].x = XLOG2DEV(points[i].x + xoffset);
716 gdkpoints[i].y = YLOG2DEV(points[i].y + yoffset);
7d5af6fa 717
265898fd
RR
718 CalcBoundingBox( points[i].x + xoffset, points[i].y + yoffset );
719 }
7d5af6fa 720
de2d2cdc 721 if (m_window)
72174350 722 {
e1208c31 723 if (m_brush.GetStyle() != wxTRANSPARENT)
72174350 724 {
e1208c31
RR
725 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
726 {
5f170f33
VZ
727 gdk_gc_set_ts_origin( m_textGC,
728 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
e1208c31
RR
729 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
730 gdk_draw_polygon( m_window, m_textGC, TRUE, gdkpoints, n );
731 gdk_gc_set_ts_origin( m_textGC, 0, 0 );
732 } else
3ca6a5f0
BP
733 if (IS_15_PIX_HATCH(m_brush.GetStyle()))
734 {
735 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 15, m_deviceOriginY % 15 );
736 gdk_draw_polygon( m_window, m_brushGC, TRUE, gdkpoints, n );
737 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
738 } else
739 if (IS_16_PIX_HATCH(m_brush.GetStyle()))
740 {
741 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 16, m_deviceOriginY % 16 );
742 gdk_draw_polygon( m_window, m_brushGC, TRUE, gdkpoints, n );
743 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
744 } else
e1208c31
RR
745 if (m_brush.GetStyle() == wxSTIPPLE)
746 {
5f170f33
VZ
747 gdk_gc_set_ts_origin( m_brushGC,
748 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
e1208c31
RR
749 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
750 gdk_draw_polygon( m_window, m_brushGC, TRUE, gdkpoints, n );
751 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
752 }
753 else
754 {
755 gdk_draw_polygon( m_window, m_brushGC, TRUE, gdkpoints, n );
756 }
72174350 757 }
7d5af6fa 758
e1208c31 759 if (m_pen.GetStyle() != wxTRANSPARENT)
b0e0d661 760 {
e1208c31
RR
761 for (i = 0 ; i < n ; i++)
762 {
763 gdk_draw_line( m_window, m_penGC,
764 gdkpoints[i%n].x,
765 gdkpoints[i%n].y,
766 gdkpoints[(i+1)%n].x,
767 gdkpoints[(i+1)%n].y);
768 }
265898fd 769 }
6db90681 770 }
7d5af6fa 771
265898fd 772 delete[] gdkpoints;
903f689b 773}
c801d85f 774
72cdf4c9 775void wxWindowDC::DoDrawRectangle( wxCoord x, wxCoord y, wxCoord width, wxCoord height )
c801d85f 776{
223d09f6 777 wxCHECK_RET( Ok(), wxT("invalid window dc") );
c801d85f 778
72cdf4c9
VZ
779 wxCoord xx = XLOG2DEV(x);
780 wxCoord yy = YLOG2DEV(y);
781 wxCoord ww = m_signX * XLOG2DEVREL(width);
782 wxCoord hh = m_signY * YLOG2DEVREL(height);
7d5af6fa 783
265898fd
RR
784 // CMB: draw nothing if transformed w or h is 0
785 if (ww == 0 || hh == 0) return;
6f65e337 786
265898fd
RR
787 // CMB: handle -ve width and/or height
788 if (ww < 0) { ww = -ww; xx = xx - ww; }
789 if (hh < 0) { hh = -hh; yy = yy - hh; }
6f65e337 790
6db90681
RR
791 if (m_window)
792 {
e1208c31 793 if (m_brush.GetStyle() != wxTRANSPARENT)
72174350 794 {
e1208c31
RR
795 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
796 {
5f170f33
VZ
797 gdk_gc_set_ts_origin( m_textGC,
798 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
e1208c31
RR
799 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
800 gdk_draw_rectangle( m_window, m_textGC, TRUE, xx, yy, ww, hh );
801 gdk_gc_set_ts_origin( m_textGC, 0, 0 );
3ca6a5f0
BP
802 } else
803 if (IS_15_PIX_HATCH(m_brush.GetStyle()))
804 {
805 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 15, m_deviceOriginY % 15 );
806 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy, ww, hh );
807 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
808 } else
809 if (IS_16_PIX_HATCH(m_brush.GetStyle()))
810 {
811 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 16, m_deviceOriginY % 16 );
812 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy, ww, hh );
813 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
814 } else
815 if (m_brush.GetStyle() == wxSTIPPLE)
e1208c31 816 {
5f170f33
VZ
817 gdk_gc_set_ts_origin( m_brushGC,
818 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
e1208c31 819 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
72174350 820 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy, ww, hh );
e1208c31
RR
821 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
822 }
823 else
824 {
825 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy, ww, hh );
826 }
72174350 827 }
e1208c31
RR
828
829 if (m_pen.GetStyle() != wxTRANSPARENT)
830 gdk_draw_rectangle( m_window, m_penGC, FALSE, xx, yy, ww-1, hh-1 );
6db90681 831 }
7d5af6fa 832
265898fd
RR
833 CalcBoundingBox( x, y );
834 CalcBoundingBox( x + width, y + height );
903f689b 835}
c801d85f 836
72cdf4c9 837void wxWindowDC::DoDrawRoundedRectangle( wxCoord x, wxCoord y, wxCoord width, wxCoord height, double radius )
c801d85f 838{
223d09f6 839 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 840
265898fd 841 if (radius < 0.0) radius = - radius * ((width < height) ? width : height);
7d5af6fa 842
72cdf4c9
VZ
843 wxCoord xx = XLOG2DEV(x);
844 wxCoord yy = YLOG2DEV(y);
845 wxCoord ww = m_signX * XLOG2DEVREL(width);
846 wxCoord hh = m_signY * YLOG2DEVREL(height);
847 wxCoord rr = XLOG2DEVREL((wxCoord)radius);
265898fd
RR
848
849 // CMB: handle -ve width and/or height
850 if (ww < 0) { ww = -ww; xx = xx - ww; }
851 if (hh < 0) { hh = -hh; yy = yy - hh; }
852
853 // CMB: if radius is zero use DrawRectangle() instead to avoid
854 // X drawing errors with small radii
855 if (rr == 0)
856 {
857 DrawRectangle( x, y, width, height );
858 return;
859 }
860
861 // CMB: draw nothing if transformed w or h is 0
862 if (ww == 0 || hh == 0) return;
863
864 // CMB: adjust size if outline is drawn otherwise the result is
865 // 1 pixel too wide and high
866 if (m_pen.GetStyle() != wxTRANSPARENT)
867 {
868 ww--;
869 hh--;
870 }
871
6db90681 872 if (m_window)
265898fd 873 {
6db90681
RR
874 // CMB: ensure dd is not larger than rectangle otherwise we
875 // get an hour glass shape
72cdf4c9 876 wxCoord dd = 2 * rr;
6db90681
RR
877 if (dd > ww) dd = ww;
878 if (dd > hh) dd = hh;
879 rr = dd / 2;
880
881 if (m_brush.GetStyle() != wxTRANSPARENT)
882 {
a56fcaaf
RR
883 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
884 {
5f170f33
VZ
885 gdk_gc_set_ts_origin( m_textGC,
886 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
a56fcaaf
RR
887 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
888 gdk_draw_rectangle( m_window, m_textGC, TRUE, xx+rr, yy, ww-dd+1, hh );
889 gdk_draw_rectangle( m_window, m_textGC, TRUE, xx, yy+rr, ww, hh-dd+1 );
890 gdk_draw_arc( m_window, m_textGC, TRUE, xx, yy, dd, dd, 90*64, 90*64 );
891 gdk_draw_arc( m_window, m_textGC, TRUE, xx+ww-dd, yy, dd, dd, 0, 90*64 );
892 gdk_draw_arc( m_window, m_textGC, TRUE, xx+ww-dd, yy+hh-dd, dd, dd, 270*64, 90*64 );
893 gdk_draw_arc( m_window, m_textGC, TRUE, xx, yy+hh-dd, dd, dd, 180*64, 90*64 );
894 gdk_gc_set_ts_origin( m_textGC, 0, 0 );
3ca6a5f0
BP
895 } else
896 if (IS_15_PIX_HATCH(m_brush.GetStyle()))
897 {
898 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 15, m_deviceOriginY % 15 );
899 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx+rr, yy, ww-dd+1, hh );
900 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy+rr, ww, hh-dd+1 );
901 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, dd, dd, 90*64, 90*64 );
902 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy, dd, dd, 0, 90*64 );
903 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy+hh-dd, dd, dd, 270*64, 90*64 );
904 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy+hh-dd, dd, dd, 180*64, 90*64 );
905 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
906 } else
907 if (IS_16_PIX_HATCH(m_brush.GetStyle()))
908 {
909 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 16, m_deviceOriginY % 16 );
910 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx+rr, yy, ww-dd+1, hh );
911 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy+rr, ww, hh-dd+1 );
912 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, dd, dd, 90*64, 90*64 );
913 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy, dd, dd, 0, 90*64 );
914 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy+hh-dd, dd, dd, 270*64, 90*64 );
915 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy+hh-dd, dd, dd, 180*64, 90*64 );
916 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
917 } else
918 if (m_brush.GetStyle() == wxSTIPPLE)
a56fcaaf 919 {
5f170f33
VZ
920 gdk_gc_set_ts_origin( m_brushGC,
921 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
a56fcaaf
RR
922 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
923 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx+rr, yy, ww-dd+1, hh );
924 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy+rr, ww, hh-dd+1 );
925 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, dd, dd, 90*64, 90*64 );
926 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy, dd, dd, 0, 90*64 );
927 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy+hh-dd, dd, dd, 270*64, 90*64 );
928 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy+hh-dd, dd, dd, 180*64, 90*64 );
929 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
930 }
931 else
932 {
933 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx+rr, yy, ww-dd+1, hh );
934 gdk_draw_rectangle( m_window, m_brushGC, TRUE, xx, yy+rr, ww, hh-dd+1 );
935 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, dd, dd, 90*64, 90*64 );
936 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy, dd, dd, 0, 90*64 );
937 gdk_draw_arc( m_window, m_brushGC, TRUE, xx+ww-dd, yy+hh-dd, dd, dd, 270*64, 90*64 );
938 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy+hh-dd, dd, dd, 180*64, 90*64 );
939 }
6db90681 940 }
7d5af6fa 941
6db90681 942 if (m_pen.GetStyle() != wxTRANSPARENT)
7d5af6fa 943 {
3ca6a5f0
BP
944 gdk_draw_line( m_window, m_penGC, xx+rr+1, yy, xx+ww-rr, yy );
945 gdk_draw_line( m_window, m_penGC, xx+rr+1, yy+hh, xx+ww-rr, yy+hh );
946 gdk_draw_line( m_window, m_penGC, xx, yy+rr+1, xx, yy+hh-rr );
947 gdk_draw_line( m_window, m_penGC, xx+ww, yy+rr+1, xx+ww, yy+hh-rr );
6db90681
RR
948 gdk_draw_arc( m_window, m_penGC, FALSE, xx, yy, dd, dd, 90*64, 90*64 );
949 gdk_draw_arc( m_window, m_penGC, FALSE, xx+ww-dd, yy, dd, dd, 0, 90*64 );
950 gdk_draw_arc( m_window, m_penGC, FALSE, xx+ww-dd, yy+hh-dd, dd, dd, 270*64, 90*64 );
951 gdk_draw_arc( m_window, m_penGC, FALSE, xx, yy+hh-dd, dd, dd, 180*64, 90*64 );
7d5af6fa 952 }
265898fd 953 }
7d5af6fa 954
265898fd
RR
955 // this ignores the radius
956 CalcBoundingBox( x, y );
957 CalcBoundingBox( x + width, y + height );
903f689b 958}
c801d85f 959
72cdf4c9 960void wxWindowDC::DoDrawEllipse( wxCoord x, wxCoord y, wxCoord width, wxCoord height )
c801d85f 961{
223d09f6 962 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 963
72cdf4c9
VZ
964 wxCoord xx = XLOG2DEV(x);
965 wxCoord yy = YLOG2DEV(y);
966 wxCoord ww = m_signX * XLOG2DEVREL(width);
967 wxCoord hh = m_signY * YLOG2DEVREL(height);
6f65e337 968
265898fd
RR
969 // CMB: handle -ve width and/or height
970 if (ww < 0) { ww = -ww; xx = xx - ww; }
971 if (hh < 0) { hh = -hh; yy = yy - hh; }
7d5af6fa 972
6db90681
RR
973 if (m_window)
974 {
975 if (m_brush.GetStyle() != wxTRANSPARENT)
a56fcaaf
RR
976 {
977 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
978 {
5f170f33
VZ
979 gdk_gc_set_ts_origin( m_textGC,
980 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
a56fcaaf
RR
981 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
982 gdk_draw_arc( m_window, m_textGC, TRUE, xx, yy, ww, hh, 0, 360*64 );
983 gdk_gc_set_ts_origin( m_textGC, 0, 0 );
3ca6a5f0
BP
984 } else
985 if (IS_15_PIX_HATCH(m_brush.GetStyle()))
986 {
987 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 15, m_deviceOriginY % 15 );
988 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, 0, 360*64 );
989 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
990 } else
991 if (IS_16_PIX_HATCH(m_brush.GetStyle()))
992 {
993 gdk_gc_set_ts_origin( m_brushGC, m_deviceOriginX % 16, m_deviceOriginY % 16 );
994 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, 0, 360*64 );
995 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
996 } else
997 if (m_brush.GetStyle() == wxSTIPPLE)
a56fcaaf 998 {
5f170f33
VZ
999 gdk_gc_set_ts_origin( m_brushGC,
1000 m_deviceOriginX % m_brush.GetStipple()->GetWidth(),
a56fcaaf
RR
1001 m_deviceOriginY % m_brush.GetStipple()->GetHeight() );
1002 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, 0, 360*64 );
1003 gdk_gc_set_ts_origin( m_brushGC, 0, 0 );
1004 }
1005 else
1006 {
1007 gdk_draw_arc( m_window, m_brushGC, TRUE, xx, yy, ww, hh, 0, 360*64 );
1008 }
1009 }
7d5af6fa 1010
6db90681
RR
1011 if (m_pen.GetStyle() != wxTRANSPARENT)
1012 gdk_draw_arc( m_window, m_penGC, FALSE, xx, yy, ww, hh, 0, 360*64 );
1013 }
7d5af6fa 1014
44856297 1015 CalcBoundingBox( x, y );
265898fd 1016 CalcBoundingBox( x + width, y + height );
903f689b 1017}
c801d85f 1018
72cdf4c9 1019void wxWindowDC::DoDrawIcon( const wxIcon &icon, wxCoord x, wxCoord y )
c801d85f 1020{
b0e0d661
VZ
1021 // VZ: egcs 1.0.3 refuses to compile this without cast, no idea why
1022 DoDrawBitmap( (const wxBitmap&)icon, x, y, (bool)TRUE );
903f689b 1023}
c801d85f 1024
b0e0d661 1025void wxWindowDC::DoDrawBitmap( const wxBitmap &bitmap,
72cdf4c9 1026 wxCoord x, wxCoord y,
b0e0d661 1027 bool useMask )
4bc67cc5 1028{
223d09f6 1029 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 1030
223d09f6 1031 wxCHECK_RET( bitmap.Ok(), wxT("invalid bitmap") );
7d5af6fa 1032
cf214c35 1033 bool is_mono = (bitmap.GetBitmap() != NULL);
82ea63e6 1034
6e13c196 1035 /* scale/translate size and position */
265898fd
RR
1036 int xx = XLOG2DEV(x);
1037 int yy = YLOG2DEV(y);
7d5af6fa 1038
4bc67cc5
RR
1039 int w = bitmap.GetWidth();
1040 int h = bitmap.GetHeight();
7d5af6fa 1041
6453876e
RR
1042 CalcBoundingBox( x, y );
1043 CalcBoundingBox( x + w, y + h );
7d5af6fa 1044
6453876e 1045 if (!m_window) return;
7d5af6fa 1046
4bc67cc5
RR
1047 int ww = XLOG2DEVREL(w);
1048 int hh = YLOG2DEVREL(h);
7d5af6fa 1049
3d2d8da1 1050 /* compare to current clipping region */
e1208c31 1051 if (!m_currentClippingRegion.IsNull())
3d2d8da1
RR
1052 {
1053 wxRegion tmp( xx,yy,ww,hh );
1054 tmp.Intersect( m_currentClippingRegion );
1055 if (tmp.IsEmpty())
1056 return;
1057 }
5f170f33 1058
6e13c196 1059 /* scale bitmap if required */
4bc67cc5 1060 wxBitmap use_bitmap;
4bc67cc5 1061 if ((w != ww) || (h != hh))
265898fd 1062 {
368d59f0 1063 wxImage image = bitmap.ConvertToImage();
82ea63e6
RR
1064 image.Rescale( ww, hh );
1065 if (is_mono)
368d59f0 1066 use_bitmap = wxBitmap(image.ConvertToMono(255,255,255), 1);
82ea63e6 1067 else
368d59f0 1068 use_bitmap = wxBitmap(image);
265898fd 1069 }
4bc67cc5 1070 else
265898fd 1071 {
4bc67cc5 1072 use_bitmap = bitmap;
265898fd 1073 }
c7f62467 1074
6e13c196 1075 /* apply mask if any */
463c1fa1 1076 GdkBitmap *mask = (GdkBitmap *) NULL;
4bc67cc5 1077 if (use_bitmap.GetMask()) mask = use_bitmap.GetMask()->GetBitmap();
7d5af6fa 1078
3d2d8da1 1079 if (useMask && mask)
82ea63e6 1080 {
3d2d8da1 1081 GdkBitmap *new_mask = (GdkBitmap*) NULL;
8cce8940 1082#ifndef __WXGTK20__ // TODO fix crash
e1208c31 1083 if (!m_currentClippingRegion.IsNull())
3d2d8da1
RR
1084 {
1085 GdkColor col;
c2fa61e8 1086 new_mask = gdk_pixmap_new( wxGetRootWindow()->window, ww, hh, 1 );
3d2d8da1
RR
1087 GdkGC *gc = gdk_gc_new( new_mask );
1088 col.pixel = 0;
1089 gdk_gc_set_foreground( gc, &col );
1090 gdk_draw_rectangle( new_mask, gc, TRUE, 0, 0, ww, hh );
1091 col.pixel = 0;
1092 gdk_gc_set_background( gc, &col );
1093 col.pixel = 1;
1094 gdk_gc_set_foreground( gc, &col );
1095 gdk_gc_set_clip_region( gc, m_currentClippingRegion.GetRegion() );
1096 gdk_gc_set_clip_origin( gc, -xx, -yy );
1097 gdk_gc_set_fill( gc, GDK_OPAQUE_STIPPLED );
1098 gdk_gc_set_stipple( gc, mask );
1099 gdk_draw_rectangle( new_mask, gc, TRUE, 0, 0, ww, hh );
1100 gdk_gc_unref( gc );
1101 }
8cce8940 1102#endif
3d2d8da1
RR
1103 if (is_mono)
1104 {
1105 if (new_mask)
1106 gdk_gc_set_clip_mask( m_textGC, new_mask );
1107 else
1108 gdk_gc_set_clip_mask( m_textGC, mask );
1109 gdk_gc_set_clip_origin( m_textGC, xx, yy );
1110 }
1111 else
1112 {
1113 if (new_mask)
1114 gdk_gc_set_clip_mask( m_penGC, new_mask );
1115 else
1116 gdk_gc_set_clip_mask( m_penGC, mask );
1117 gdk_gc_set_clip_origin( m_penGC, xx, yy );
1118 }
8cce8940 1119
3d2d8da1
RR
1120 if (new_mask)
1121 gdk_bitmap_unref( new_mask );
b0e0d661 1122 }
7d5af6fa 1123
82ea63e6
RR
1124 /* Draw XPixmap or XBitmap, depending on what the wxBitmap contains. For
1125 drawing a mono-bitmap (XBitmap) we use the current text GC */
1126 if (is_mono)
f6bcfd97 1127 gdk_wx_draw_bitmap( m_window, m_textGC, use_bitmap.GetBitmap(), 0, 0, xx, yy, -1, -1 );
82ea63e6
RR
1128 else
1129 gdk_draw_pixmap( m_window, m_penGC, use_bitmap.GetPixmap(), 0, 0, xx, yy, -1, -1 );
72174350 1130
6e13c196 1131 /* remove mask again if any */
7d5af6fa 1132 if (useMask && mask)
463c1fa1 1133 {
82ea63e6
RR
1134 if (is_mono)
1135 {
1136 gdk_gc_set_clip_mask( m_textGC, (GdkBitmap *) NULL );
1137 gdk_gc_set_clip_origin( m_textGC, 0, 0 );
e1208c31 1138 if (!m_currentClippingRegion.IsNull())
3d2d8da1 1139 gdk_gc_set_clip_region( m_textGC, m_currentClippingRegion.GetRegion() );
82ea63e6
RR
1140 }
1141 else
1142 {
1143 gdk_gc_set_clip_mask( m_penGC, (GdkBitmap *) NULL );
1144 gdk_gc_set_clip_origin( m_penGC, 0, 0 );
e1208c31 1145 if (!m_currentClippingRegion.IsNull())
3d2d8da1 1146 gdk_gc_set_clip_region( m_penGC, m_currentClippingRegion.GetRegion() );
82ea63e6 1147 }
463c1fa1 1148 }
463c1fa1
RR
1149}
1150
1e6feb95
VZ
1151bool wxWindowDC::DoBlit( wxCoord xdest, wxCoord ydest,
1152 wxCoord width, wxCoord height,
1153 wxDC *source,
1154 wxCoord xsrc, wxCoord ysrc,
1155 int logical_func,
0cbff120
JS
1156 bool useMask,
1157 wxCoord xsrcMask, wxCoord ysrcMask )
c801d85f 1158{
6e13c196
RR
1159 /* this is the nth try to get this utterly useless function to
1160 work. it now completely ignores the scaling or translation
1161 of the source dc, but scales correctly on the target dc and
1162 knows about possible mask information in a memory dc. */
1163
223d09f6 1164 wxCHECK_MSG( Ok(), FALSE, wxT("invalid window dc") );
7d5af6fa 1165
223d09f6 1166 wxCHECK_MSG( source, FALSE, wxT("invalid source dc") );
7d5af6fa 1167
6db90681 1168 if (!m_window) return FALSE;
7d5af6fa 1169
06201c35 1170#if 1
1e6feb95 1171 // transform the source DC coords to the device ones
06201c35
RR
1172 xsrc = source->XLOG2DEV(xsrc);
1173 ysrc = source->YLOG2DEV(ysrc);
1e6feb95
VZ
1174#endif
1175
6e13c196
RR
1176 wxClientDC *srcDC = (wxClientDC*)source;
1177 wxMemoryDC *memDC = (wxMemoryDC*)source;
7d5af6fa 1178
6e13c196 1179 bool use_bitmap_method = FALSE;
82ea63e6 1180 bool is_mono = FALSE;
7d5af6fa 1181
0cbff120
JS
1182 /* TODO: use the mask origin when drawing transparently */
1183 if (xsrcMask == -1 && ysrcMask == -1)
1184 {
1185 xsrcMask = xsrc; ysrcMask = ysrc;
1186 }
1187
6e13c196
RR
1188 if (srcDC->m_isMemDC)
1189 {
b0e0d661 1190 if (!memDC->m_selected.Ok()) return FALSE;
7d5af6fa 1191
6e13c196 1192 /* we use the "XCopyArea" way to copy a memory dc into
b0e0d661
VZ
1193 y different window if the memory dc BOTH
1194 a) doesn't have any mask or its mask isn't used
1195 b) it is clipped
1196 c) is not 1-bit */
7d5af6fa 1197
f234c60c 1198 if (useMask && (memDC->m_selected.GetMask()))
b0e0d661
VZ
1199 {
1200 /* we HAVE TO use the direct way for memory dcs
1201 that have mask since the XCopyArea doesn't know
1202 about masks */
1203 use_bitmap_method = TRUE;
1204 }
1205 else if (memDC->m_selected.GetDepth() == 1)
1206 {
1207 /* we HAVE TO use the direct way for memory dcs
1208 that are bitmaps because XCopyArea doesn't cope
1209 with different bit depths */
82ea63e6 1210 is_mono = TRUE;
b0e0d661
VZ
1211 use_bitmap_method = TRUE;
1212 }
1213 else if ((xsrc == 0) && (ysrc == 0) &&
1214 (width == memDC->m_selected.GetWidth()) &&
1215 (height == memDC->m_selected.GetHeight()))
1216 {
7d5af6fa
VZ
1217 /* we SHOULD use the direct way if all of the bitmap
1218 in the memory dc is copied in which case XCopyArea
1219 wouldn't be able able to boost performace by reducing
b0e0d661
VZ
1220 the area to be scaled */
1221 use_bitmap_method = TRUE;
1222 }
1223 else
1224 {
1225 use_bitmap_method = FALSE;
1226 }
6e13c196 1227 }
7d5af6fa 1228
265898fd
RR
1229 CalcBoundingBox( xdest, ydest );
1230 CalcBoundingBox( xdest + width, ydest + height );
7d5af6fa 1231
3d2d8da1
RR
1232 /* scale/translate size and position */
1233 wxCoord xx = XLOG2DEV(xdest);
1234 wxCoord yy = YLOG2DEV(ydest);
1235
1236 wxCoord ww = XLOG2DEVREL(width);
1237 wxCoord hh = YLOG2DEVREL(height);
1238
1239 /* compare to current clipping region */
e1208c31 1240 if (!m_currentClippingRegion.IsNull())
3d2d8da1
RR
1241 {
1242 wxRegion tmp( xx,yy,ww,hh );
1243 tmp.Intersect( m_currentClippingRegion );
1244 if (tmp.IsEmpty())
1245 return TRUE;
1246 }
1247
4bc67cc5
RR
1248 int old_logical_func = m_logicalFunction;
1249 SetLogicalFunction( logical_func );
5f170f33 1250
6e13c196 1251 if (use_bitmap_method)
6f65e337 1252 {
6e13c196 1253 /* scale/translate bitmap size */
72cdf4c9
VZ
1254 wxCoord bm_width = memDC->m_selected.GetWidth();
1255 wxCoord bm_height = memDC->m_selected.GetHeight();
7d5af6fa 1256
72cdf4c9
VZ
1257 wxCoord bm_ww = XLOG2DEVREL( bm_width );
1258 wxCoord bm_hh = YLOG2DEVREL( bm_height );
7d5af6fa 1259
6e13c196 1260 /* scale bitmap if required */
6e13c196 1261 wxBitmap use_bitmap;
7d5af6fa 1262
6e13c196
RR
1263 if ((bm_width != bm_ww) || (bm_height != bm_hh))
1264 {
368d59f0 1265 wxImage image = memDC->m_selected.ConvertToImage();
b0e0d661 1266 image = image.Scale( bm_ww, bm_hh );
7d5af6fa 1267
82ea63e6 1268 if (is_mono)
368d59f0 1269 use_bitmap = wxBitmap(image.ConvertToMono(255,255,255), 1);
82ea63e6 1270 else
368d59f0 1271 use_bitmap = wxBitmap(image);
6e13c196
RR
1272 }
1273 else
e23d0e95 1274 {
6e13c196
RR
1275 use_bitmap = memDC->m_selected;
1276 }
7d5af6fa 1277
6e13c196 1278 /* apply mask if any */
6e13c196
RR
1279 GdkBitmap *mask = (GdkBitmap *) NULL;
1280 if (use_bitmap.GetMask()) mask = use_bitmap.GetMask()->GetBitmap();
7d5af6fa
VZ
1281
1282 if (useMask && mask)
6e13c196 1283 {
3d2d8da1 1284 GdkBitmap *new_mask = (GdkBitmap*) NULL;
8cce8940 1285#ifndef __WXGTK20__ // TODO fix crash
e1208c31 1286 if (!m_currentClippingRegion.IsNull())
3d2d8da1
RR
1287 {
1288 GdkColor col;
c2fa61e8 1289 new_mask = gdk_pixmap_new( wxGetRootWindow()->window, bm_ww, bm_hh, 1 );
3d2d8da1
RR
1290 GdkGC *gc = gdk_gc_new( new_mask );
1291 col.pixel = 0;
1292 gdk_gc_set_foreground( gc, &col );
1293 gdk_draw_rectangle( new_mask, gc, TRUE, 0, 0, bm_ww, bm_hh );
1294 col.pixel = 0;
1295 gdk_gc_set_background( gc, &col );
1296 col.pixel = 1;
1297 gdk_gc_set_foreground( gc, &col );
1298 gdk_gc_set_clip_region( gc, m_currentClippingRegion.GetRegion() );
1299 gdk_gc_set_clip_origin( gc, -xx, -yy );
1300 gdk_gc_set_fill( gc, GDK_OPAQUE_STIPPLED );
1301 gdk_gc_set_stipple( gc, mask );
1302 gdk_draw_rectangle( new_mask, gc, TRUE, 0, 0, bm_ww, bm_hh );
1303 gdk_gc_unref( gc );
1304 }
8cce8940 1305#endif
82ea63e6
RR
1306 if (is_mono)
1307 {
3d2d8da1
RR
1308 if (new_mask)
1309 gdk_gc_set_clip_mask( m_textGC, new_mask );
1310 else
1311 gdk_gc_set_clip_mask( m_textGC, mask );
82ea63e6
RR
1312 gdk_gc_set_clip_origin( m_textGC, xx, yy );
1313 }
1314 else
6e13c196 1315 {
3d2d8da1
RR
1316 if (new_mask)
1317 gdk_gc_set_clip_mask( m_penGC, new_mask );
1318 else
1319 gdk_gc_set_clip_mask( m_penGC, mask );
82ea63e6 1320 gdk_gc_set_clip_origin( m_penGC, xx, yy );
b0e0d661 1321 }
3d2d8da1
RR
1322 if (new_mask)
1323 gdk_bitmap_unref( new_mask );
6e13c196 1324 }
7d5af6fa 1325
82ea63e6
RR
1326 /* Draw XPixmap or XBitmap, depending on what the wxBitmap contains. For
1327 drawing a mono-bitmap (XBitmap) we use the current text GC */
5f170f33 1328
82ea63e6 1329 if (is_mono)
f6bcfd97 1330 gdk_wx_draw_bitmap( m_window, m_textGC, use_bitmap.GetBitmap(), xsrc, ysrc, xx, yy, ww, hh );
82ea63e6
RR
1331 else
1332 gdk_draw_pixmap( m_window, m_penGC, use_bitmap.GetPixmap(), xsrc, ysrc, xx, yy, ww, hh );
1333
6e13c196 1334 /* remove mask again if any */
7d5af6fa 1335 if (useMask && mask)
6e13c196 1336 {
82ea63e6
RR
1337 if (is_mono)
1338 {
1339 gdk_gc_set_clip_mask( m_textGC, (GdkBitmap *) NULL );
1340 gdk_gc_set_clip_origin( m_textGC, 0, 0 );
e1208c31 1341 if (!m_currentClippingRegion.IsNull())
3d2d8da1 1342 gdk_gc_set_clip_region( m_textGC, m_currentClippingRegion.GetRegion() );
82ea63e6
RR
1343 }
1344 else
1345 {
1346 gdk_gc_set_clip_mask( m_penGC, (GdkBitmap *) NULL );
1347 gdk_gc_set_clip_origin( m_penGC, 0, 0 );
e1208c31 1348 if (!m_currentClippingRegion.IsNull())
3d2d8da1 1349 gdk_gc_set_clip_region( m_penGC, m_currentClippingRegion.GetRegion() );
82ea63e6 1350 }
265898fd 1351 }
6f65e337 1352 }
6e13c196
RR
1353 else /* use_bitmap_method */
1354 {
6e13c196 1355 if ((width != ww) || (height != hh))
b0e0d661
VZ
1356 {
1357 /* draw source window into a bitmap as we cannot scale
1358 a window in contrast to a bitmap. this would actually
1359 work with memory dcs as well, but we'd lose the mask
1360 information and waste one step in this process since
1361 a memory already has a bitmap. all this is slightly
1362 inefficient as we could take an XImage directly from
1363 an X window, but we'd then also have to care that
1364 the window is not outside the screen (in which case
1365 we'd get a BadMatch or what not).
1366 Is a double XGetImage and combined XGetPixel and
1367 XPutPixel really faster? I'm not sure. look at wxXt
1368 for a different implementation of the same problem. */
7d5af6fa 1369
b0e0d661 1370 wxBitmap bitmap( width, height );
5f170f33 1371
a56fcaaf
RR
1372 /* copy including child window contents */
1373 gdk_gc_set_subwindow( m_penGC, GDK_INCLUDE_INFERIORS );
e1208c31 1374 gdk_window_copy_area( bitmap.GetPixmap(), m_penGC, 0, 0,
6e13c196 1375 srcDC->GetWindow(),
b0e0d661 1376 xsrc, ysrc, width, height );
a56fcaaf 1377 gdk_gc_set_subwindow( m_penGC, GDK_CLIP_BY_CHILDREN );
5f170f33 1378
b0e0d661 1379 /* scale image */
368d59f0 1380 wxImage image = bitmap.ConvertToImage();
b0e0d661 1381 image = image.Scale( ww, hh );
7d5af6fa 1382
b0e0d661 1383 /* convert to bitmap */
368d59f0 1384 bitmap = wxBitmap(image);
7d5af6fa 1385
b0e0d661 1386 /* draw scaled bitmap */
6e13c196 1387 gdk_draw_pixmap( m_window, m_penGC, bitmap.GetPixmap(), 0, 0, xx, yy, -1, -1 );
7d5af6fa 1388
b0e0d661
VZ
1389 }
1390 else
1391 {
993f97ee
RR
1392 /* No scaling and not a memory dc with a mask either */
1393
a56fcaaf
RR
1394 /* copy including child window contents */
1395 gdk_gc_set_subwindow( m_penGC, GDK_INCLUDE_INFERIORS );
6e13c196
RR
1396 gdk_window_copy_area( m_window, m_penGC, xx, yy,
1397 srcDC->GetWindow(),
1398 xsrc, ysrc, width, height );
a56fcaaf 1399 gdk_gc_set_subwindow( m_penGC, GDK_CLIP_BY_CHILDREN );
b0e0d661 1400 }
6e13c196 1401 }
c801d85f 1402
4bc67cc5
RR
1403 SetLogicalFunction( old_logical_func );
1404 return TRUE;
903f689b 1405}
c801d85f 1406
72cdf4c9 1407void wxWindowDC::DoDrawText( const wxString &text, wxCoord x, wxCoord y )
c801d85f 1408{
223d09f6 1409 wxCHECK_RET( Ok(), wxT("invalid window dc") );
18a2fa37 1410
6db90681 1411 if (!m_window) return;
2b5f62a0
VZ
1412
1413 if (text.empty()) return;
7d5af6fa 1414
2b5f62a0 1415#ifndef __WXGTK20__
265898fd 1416 GdkFont *font = m_font.GetInternalFont( m_scaleY );
6f65e337 1417
a1fffa9f 1418 wxCHECK_RET( font, wxT("invalid font") );
2b5f62a0 1419#endif
a1fffa9f 1420
2b5f62a0 1421#ifdef __WXGTK20__
8943b403
OK
1422 wxCHECK_RET( m_context, wxT("no Pango context") );
1423#endif
1424
265898fd
RR
1425 x = XLOG2DEV(x);
1426 y = YLOG2DEV(y);
18a2fa37 1427
2b5f62a0 1428#ifdef __WXGTK20__
9e691f46 1429 // TODO: the layout engine should be abstracted at a higher level!
8943b403
OK
1430 PangoLayout *layout = pango_layout_new(m_context);
1431 pango_layout_set_font_description(layout, m_fontdesc);
1432 {
fb3ed106 1433#if wxUSE_UNICODE
48d011c8
RR
1434 const wxCharBuffer data = wxConvUTF8.cWC2MB( text );
1435 pango_layout_set_text(layout, (const char*) data, strlen( (const char*) data ));
fb3ed106
RR
1436#else
1437 const wxWCharBuffer wdata = wxConvLocal.cMB2WC( text );
1438 const wxCharBuffer data = wxConvUTF8.cWC2MB( wdata );
1439 pango_layout_set_text(layout, (const char*) data, strlen( (const char*) data ));
1440#endif
8943b403 1441 }
2b5f62a0
VZ
1442
1443 // Measure layout.
1444 int w,h;
1445 pango_layout_get_pixel_size(layout, &w, &h);
1446 wxCoord width = w;
1447 wxCoord height = h;
1448
1449 // Draw layout.
8943b403 1450 gdk_draw_layout( m_window, m_textGC, x, y, layout );
2b5f62a0
VZ
1451
1452 g_object_unref( G_OBJECT( layout ) );
9e691f46 1453#else // GTK+ 1.x
8943b403
OK
1454 wxCoord width = gdk_string_width( font, text.mbc_str() );
1455 wxCoord height = font->ascent + font->descent;
2c37aade
VZ
1456
1457 if ( m_backgroundMode == wxSOLID )
265898fd 1458 {
265898fd
RR
1459 gdk_gc_set_foreground( m_textGC, m_textBackgroundColour.GetColor() );
1460 gdk_draw_rectangle( m_window, m_textGC, TRUE, x, y, width, height );
2c37aade 1461 gdk_gc_set_foreground( m_textGC, m_textForegroundColour.GetColor() );
265898fd 1462 }
93c5dd39 1463 gdk_draw_string( m_window, font, m_textGC, x, y + font->ascent, text.mbc_str() );
18a2fa37 1464
bbbbe360
RR
1465 /* CMB 17/7/98: simple underline: ignores scaling and underlying
1466 X font's XA_UNDERLINE_POSITION and XA_UNDERLINE_THICKNESS
1467 properties (see wxXt implementation) */
265898fd
RR
1468 if (m_font.GetUnderlined())
1469 {
13111b2a 1470 wxCoord ul_y = y + font->ascent;
265898fd
RR
1471 if (font->descent > 0) ul_y++;
1472 gdk_draw_line( m_window, m_textGC, x, ul_y, x + width, ul_y);
1473 }
48d011c8 1474#endif // GTK+ 2.0/1.x
7d5af6fa 1475
8943b403
OK
1476
1477 width = wxCoord(width / m_scaleX);
1478 height = wxCoord(height / m_scaleY);
1479 CalcBoundingBox (x + width, y + height);
265898fd 1480 CalcBoundingBox (x, y);
903f689b 1481}
c801d85f 1482
95724b1a
VZ
1483void wxWindowDC::DoDrawRotatedText( const wxString &text, wxCoord x, wxCoord y, double angle )
1484{
1485 if (angle == 0.0)
1486 {
1487 DrawText(text, x, y);
1488 return;
1489 }
1490
1491 wxCHECK_RET( Ok(), wxT("invalid window dc") );
1492
1493 if (!m_window) return;
1494
1495 GdkFont *font = m_font.GetInternalFont( m_scaleY );
1496
1497 wxCHECK_RET( font, wxT("invalid font") );
1498
9a8c7620 1499 // the size of the text
13111b2a
VZ
1500 wxCoord w = gdk_string_width( font, text.mbc_str() );
1501 wxCoord h = font->ascent + font->descent;
95724b1a 1502
9a8c7620
VZ
1503 // draw the string normally
1504 wxBitmap src(w, h);
95724b1a
VZ
1505 wxMemoryDC dc;
1506 dc.SelectObject(src);
1507 dc.SetFont(GetFont());
1508 dc.SetBackground(*wxWHITE_BRUSH);
1509 dc.SetBrush(*wxBLACK_BRUSH);
1510 dc.Clear();
1511 dc.DrawText(text, 0, 0);
9a8c7620 1512 dc.SelectObject(wxNullBitmap);
95724b1a
VZ
1513
1514 // Calculate the size of the rotated bounding box.
9a8c7620
VZ
1515 double rad = DegToRad(angle);
1516 double dx = cos(rad),
1517 dy = sin(rad);
1518
1519 // the rectngle vertices are counted clockwise with the first one being at
1520 // (0, 0) (or, rather, at (x, y))
1521 double x2 = w*dx,
1522 y2 = -w*dy; // y axis points to the bottom, hence minus
1523 double x4 = h*dy,
1524 y4 = h*dx;
1525 double x3 = x4 + x2,
1526 y3 = y4 + y2;
1527
72174350 1528 // calc max and min
9a8c7620
VZ
1529 wxCoord maxX = (wxCoord)(dmax(x2, dmax(x3, x4)) + 0.5),
1530 maxY = (wxCoord)(dmax(y2, dmax(y3, y4)) + 0.5),
1531 minX = (wxCoord)(dmin(x2, dmin(x3, x4)) - 0.5),
1532 minY = (wxCoord)(dmin(y2, dmin(y3, y4)) - 0.5);
1533
1534 // prepare to blit-with-rotate the bitmap to the DC
368d59f0 1535 wxImage image = src.ConvertToImage();
9a8c7620
VZ
1536
1537 GdkColor *colText = m_textForegroundColour.GetColor(),
1538 *colBack = m_textBackgroundColour.GetColor();
72174350 1539
9a8c7620
VZ
1540 bool textColSet = TRUE;
1541
1542 unsigned char *data = image.GetData();
1543
1544 // paint pixel by pixel
1545 for ( wxCoord srcX = 0; srcX < w; srcX++ )
95724b1a 1546 {
9a8c7620 1547 for ( wxCoord srcY = 0; srcY < h; srcY++ )
95724b1a 1548 {
9a8c7620 1549 // transform source coords to dest coords
f6bcfd97
BP
1550 double r = sqrt((double)srcX*srcX + srcY*srcY);
1551 double angleOrig = atan2((double)srcY, (double)srcX) - rad;
9a8c7620
VZ
1552 wxCoord dstX = (wxCoord)(r*cos(angleOrig) + 0.5),
1553 dstY = (wxCoord)(r*sin(angleOrig) + 0.5);
1554
1555 // black pixel?
1556 bool textPixel = data[(srcY*w + srcX)*3] == 0;
1557 if ( textPixel || (m_backgroundMode == wxSOLID) )
95724b1a 1558 {
9a8c7620
VZ
1559 // change colour if needed
1560 if ( textPixel != textColSet )
95724b1a 1561 {
9a8c7620
VZ
1562 gdk_gc_set_foreground( m_textGC, textPixel ? colText
1563 : colBack );
1564
1565 textColSet = textPixel;
95724b1a 1566 }
9a8c7620
VZ
1567
1568 // don't use DrawPoint() because it uses the current pen
1569 // colour, and we don't need it here
1570 gdk_draw_point( m_window, m_textGC,
a57a1fb7 1571 XLOG2DEV(x) + dstX, YLOG2DEV(y) + dstY );
95724b1a
VZ
1572 }
1573 }
1574 }
1575
9a8c7620
VZ
1576 // it would be better to draw with non underlined font and draw the line
1577 // manually here (it would be more straight...)
1578#if 0
1579 if ( m_font.GetUnderlined() )
1580 {
1581 gdk_draw_line( m_window, m_textGC,
1582 XLOG2DEV(x + x4), YLOG2DEV(y + y4 + font->descent),
1583 XLOG2DEV(x + x3), YLOG2DEV(y + y3 + font->descent));
1584 }
1585#endif // 0
1586
1587 // restore the font colour
1588 gdk_gc_set_foreground( m_textGC, colText );
1589
1590 // update the bounding box
1591 CalcBoundingBox(x + minX, y + minY);
1592 CalcBoundingBox(x + maxX, y + maxY);
95724b1a
VZ
1593}
1594
72cdf4c9
VZ
1595void wxWindowDC::DoGetTextExtent(const wxString &string,
1596 wxCoord *width, wxCoord *height,
1597 wxCoord *descent, wxCoord *externalLeading,
1598 wxFont *theFont) const
c801d85f 1599{
48d011c8
RR
1600 if (string.IsEmpty())
1601 {
1602 if (width) (*width) = 0;
1603 if (height) (*height) = 0;
1604 return;
1605 }
1606
1607#ifdef __WXGTK20__
2b5f62a0 1608 // Create layout and set font description
48d011c8 1609 PangoLayout *layout = pango_layout_new(m_context);
2b5f62a0
VZ
1610 if (theFont)
1611 pango_layout_set_font_description( layout, theFont->GetNativeFontInfo()->description );
1612 else
1613 pango_layout_set_font_description(layout, m_fontdesc);
1614
1615 // Set layout's text
fb3ed106 1616#if wxUSE_UNICODE
48d011c8
RR
1617 const wxCharBuffer data = wxConvUTF8.cWC2MB( string );
1618 pango_layout_set_text(layout, (const char*) data, strlen( (const char*) data ));
fb3ed106
RR
1619#else
1620 const wxWCharBuffer wdata = wxConvLocal.cMB2WC( string );
1621 const wxCharBuffer data = wxConvUTF8.cWC2MB( wdata );
1622 pango_layout_set_text(layout, (const char*) data, strlen( (const char*) data ));
1623#endif
48d011c8 1624
2b5f62a0
VZ
1625 // Measure text.
1626 int w,h;
1627 pango_layout_get_pixel_size(layout, &w, &h);
48d011c8 1628
2b5f62a0
VZ
1629 if (width) (*width) = (wxCoord) w;
1630 if (height) (*height) = (wxCoord) h;
48d011c8
RR
1631 if (descent)
1632 {
2b5f62a0 1633 // Do something about metrics here. TODO.
48d011c8
RR
1634 (*descent) = 0;
1635 }
1636 if (externalLeading) (*externalLeading) = 0; // ??
1637
1638 g_object_unref( G_OBJECT( layout ) );
1639#else
2b5f62a0
VZ
1640 wxFont fontToUse = m_font;
1641 if (theFont) fontToUse = *theFont;
1642
265898fd 1643 GdkFont *font = fontToUse.GetInternalFont( m_scaleY );
72cdf4c9
VZ
1644 if (width) (*width) = wxCoord(gdk_string_width( font, string.mbc_str() ) / m_scaleX);
1645 if (height) (*height) = wxCoord((font->ascent + font->descent) / m_scaleY);
1646 if (descent) (*descent) = wxCoord(font->descent / m_scaleY);
265898fd 1647 if (externalLeading) (*externalLeading) = 0; // ??
48d011c8 1648#endif
903f689b 1649}
c801d85f 1650
72cdf4c9 1651wxCoord wxWindowDC::GetCharWidth() const
c801d85f 1652{
2b5f62a0
VZ
1653#ifdef __WXGTK20__
1654 // There should be an easier way.
1655 PangoLayout *layout = pango_layout_new(m_context);
1656 pango_layout_set_font_description(layout, m_fontdesc);
1657 pango_layout_set_text(layout, "H", 1 );
1658 int w,h;
1659 pango_layout_get_pixel_size(layout, &w, &h);
1660 g_object_unref( G_OBJECT( layout ) );
1661 return w;
1662#else
265898fd 1663 GdkFont *font = m_font.GetInternalFont( m_scaleY );
58c837a4 1664 wxCHECK_MSG( font, -1, wxT("invalid font") );
7beba2fc 1665
72cdf4c9 1666 return wxCoord(gdk_string_width( font, "H" ) / m_scaleX);
2b5f62a0 1667#endif
903f689b 1668}
c801d85f 1669
72cdf4c9 1670wxCoord wxWindowDC::GetCharHeight() const
c801d85f 1671{
2b5f62a0
VZ
1672#ifdef __WXGTK20__
1673 // There should be an easier way.
1674 PangoLayout *layout = pango_layout_new(m_context);
1675 pango_layout_set_font_description(layout, m_fontdesc);
1676 pango_layout_set_text(layout, "H", 1 );
1677 int w,h;
1678 pango_layout_get_pixel_size(layout, &w, &h);
1679 g_object_unref( G_OBJECT( layout ) );
1680 return h;
1681#else
265898fd 1682 GdkFont *font = m_font.GetInternalFont( m_scaleY );
58c837a4 1683 wxCHECK_MSG( font, -1, wxT("invalid font") );
7beba2fc 1684
72cdf4c9 1685 return wxCoord((font->ascent + font->descent) / m_scaleY);
2b5f62a0 1686#endif
903f689b 1687}
c801d85f 1688
4bc67cc5 1689void wxWindowDC::Clear()
c801d85f 1690{
223d09f6 1691 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 1692
6db90681 1693 if (!m_window) return;
7d5af6fa 1694
f60e7fdd
VZ
1695 // VZ: the code below results in infinite recursion and crashes when
1696 // dc.Clear() is done from OnPaint() so I disable it for now.
1697 // I don't know what the correct fix is but Clear() surely should not
1698 // reenter OnPaint()!
1699#if 0
f234c60c
RR
1700 /* - we either are a memory dc or have a window as the
1701 owner. anything else shouldn't happen.
1702 - we don't use gdk_window_clear() as we don't set
1703 the window's background colour anymore. it is too
1704 much pain to keep the DC's and the window's back-
1705 ground colour in synch. */
7d5af6fa 1706
f234c60c 1707 if (m_owner)
265898fd 1708 {
f90566f5 1709 m_owner->Clear();
b0e0d661 1710 return;
265898fd 1711 }
f234c60c
RR
1712
1713 if (m_isMemDC)
265898fd
RR
1714 {
1715 int width,height;
1716 GetSize( &width, &height );
1717 gdk_draw_rectangle( m_window, m_bgGC, TRUE, 0, 0, width, height );
b0e0d661 1718 return;
265898fd 1719 }
f60e7fdd
VZ
1720#else // 1
1721 int width,height;
1722 GetSize( &width, &height );
1723 gdk_draw_rectangle( m_window, m_bgGC, TRUE, 0, 0, width, height );
1724#endif // 0/1
903f689b 1725}
c801d85f 1726
ec758a20 1727void wxWindowDC::SetFont( const wxFont &font )
c801d85f 1728{
4855a477
JS
1729 // It is common practice to set the font to wxNullFont, so
1730 // don't consider it to be an error
1731 // wxCHECK_RET( font.Ok(), _T("invalid font in wxWindowDC::SetFont") );
6a41b213 1732
265898fd 1733 m_font = font;
8943b403 1734#ifdef __WXGTK20__
77416abd 1735 if (m_font.Ok())
2b5f62a0 1736 {
77416abd
JS
1737 m_fontdesc = m_font.GetNativeFontInfo()->description;
1738
1739 if (m_owner)
1740 {
1741 if (m_font.GetNoAntiAliasing())
1742 m_context = m_owner->GtkGetPangoX11Context();
1743 else
1744 m_context = m_owner->GtkGetPangoDefaultContext();
1745 }
2b5f62a0 1746 }
8943b403 1747#endif
903f689b 1748}
c801d85f 1749
ec758a20 1750void wxWindowDC::SetPen( const wxPen &pen )
c801d85f 1751{
223d09f6 1752 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 1753
265898fd 1754 if (m_pen == pen) return;
7d5af6fa 1755
265898fd 1756 m_pen = pen;
7d5af6fa 1757
265898fd 1758 if (!m_pen.Ok()) return;
7d5af6fa 1759
6db90681 1760 if (!m_window) return;
7d5af6fa 1761
265898fd 1762 gint width = m_pen.GetWidth();
265898fd
RR
1763 if (width <= 0)
1764 {
112c5086 1765 // CMB: if width is non-zero scale it with the dc
265898fd
RR
1766 width = 1;
1767 }
1768 else
1769 {
1770 // X doesn't allow different width in x and y and so we take
1771 // the average
503f414e
VZ
1772 double w = 0.5 +
1773 ( fabs((double) XLOG2DEVREL(width)) +
1774 fabs((double) YLOG2DEVREL(width)) ) / 2.0;
265898fd
RR
1775 width = (int)w;
1776 }
7d5af6fa 1777
2eca425d
RL
1778 static const wxGTKDash dotted[] = {1, 1};
1779 static const wxGTKDash short_dashed[] = {2, 2};
1780 static const wxGTKDash wxCoord_dashed[] = {2, 4};
1781 static const wxGTKDash dotted_dashed[] = {3, 3, 1, 3};
7d5af6fa 1782
112c5086
RR
1783 // We express dash pattern in pen width unit, so we are
1784 // independent of zoom factor and so on...
1785 int req_nb_dash;
2eca425d 1786 const wxGTKDash *req_dash;
7d5af6fa 1787
265898fd
RR
1788 GdkLineStyle lineStyle = GDK_LINE_SOLID;
1789 switch (m_pen.GetStyle())
1790 {
112c5086 1791 case wxUSER_DASH:
7d5af6fa
VZ
1792 {
1793 lineStyle = GDK_LINE_ON_OFF_DASH;
112c5086 1794 req_nb_dash = m_pen.GetDashCount();
2eca425d 1795 req_dash = (wxGTKDash*)m_pen.GetDash();
112c5086 1796 break;
7d5af6fa
VZ
1797 }
1798 case wxDOT:
1799 {
1800 lineStyle = GDK_LINE_ON_OFF_DASH;
112c5086
RR
1801 req_nb_dash = 2;
1802 req_dash = dotted;
7d5af6fa
VZ
1803 break;
1804 }
1805 case wxLONG_DASH:
1806 {
1807 lineStyle = GDK_LINE_ON_OFF_DASH;
112c5086 1808 req_nb_dash = 2;
72cdf4c9 1809 req_dash = wxCoord_dashed;
7d5af6fa
VZ
1810 break;
1811 }
1812 case wxSHORT_DASH:
1813 {
1814 lineStyle = GDK_LINE_ON_OFF_DASH;
112c5086
RR
1815 req_nb_dash = 2;
1816 req_dash = short_dashed;
7d5af6fa
VZ
1817 break;
1818 }
1819 case wxDOT_DASH:
1820 {
1821// lineStyle = GDK_LINE_DOUBLE_DASH;
1822 lineStyle = GDK_LINE_ON_OFF_DASH;
112c5086
RR
1823 req_nb_dash = 4;
1824 req_dash = dotted_dashed;
7d5af6fa
VZ
1825 break;
1826 }
1827
1828 case wxTRANSPARENT:
72174350 1829 case wxSTIPPLE_MASK_OPAQUE:
7d5af6fa
VZ
1830 case wxSTIPPLE:
1831 case wxSOLID:
1832 default:
1833 {
1834 lineStyle = GDK_LINE_SOLID;
2eca425d 1835 req_dash = (wxGTKDash*)NULL;
112c5086 1836 req_nb_dash = 0;
7d5af6fa
VZ
1837 break;
1838 }
265898fd 1839 }
7d5af6fa 1840
2eca425d 1841#if (GTK_MINOR_VERSION > 0) || (GTK_MAJOR_VERSION > 1)
112c5086
RR
1842 if (req_dash && req_nb_dash)
1843 {
2eca425d 1844 wxGTKDash *real_req_dash = new wxGTKDash[req_nb_dash];
112c5086
RR
1845 if (real_req_dash)
1846 {
1847 for (int i = 0; i < req_nb_dash; i++)
1848 real_req_dash[i] = req_dash[i] * width;
2eca425d 1849 gdk_gc_set_dashes( m_penGC, 0, real_req_dash, req_nb_dash );
112c5086
RR
1850 delete[] real_req_dash;
1851 }
1852 else
1853 {
1854 // No Memory. We use non-scaled dash pattern...
2eca425d 1855 gdk_gc_set_dashes( m_penGC, 0, (wxGTKDash*)req_dash, req_nb_dash );
112c5086
RR
1856 }
1857 }
2b5f62a0 1858#endif // GTK+ > 1.0
7d5af6fa 1859
265898fd
RR
1860 GdkCapStyle capStyle = GDK_CAP_ROUND;
1861 switch (m_pen.GetCap())
1862 {
265898fd
RR
1863 case wxCAP_PROJECTING: { capStyle = GDK_CAP_PROJECTING; break; }
1864 case wxCAP_BUTT: { capStyle = GDK_CAP_BUTT; break; }
d4aa3a4b
RR
1865 case wxCAP_ROUND:
1866 default:
72174350 1867 {
d4aa3a4b
RR
1868 if (width <= 1)
1869 {
1870 width = 0;
1871 capStyle = GDK_CAP_NOT_LAST;
1872 }
1873 else
1874 {
1875 capStyle = GDK_CAP_ROUND;
1876 }
72174350 1877 break;
d4aa3a4b 1878 }
265898fd 1879 }
7d5af6fa 1880
265898fd
RR
1881 GdkJoinStyle joinStyle = GDK_JOIN_ROUND;
1882 switch (m_pen.GetJoin())
1883 {
1884 case wxJOIN_BEVEL: { joinStyle = GDK_JOIN_BEVEL; break; }
265898fd 1885 case wxJOIN_MITER: { joinStyle = GDK_JOIN_MITER; break; }
d4aa3a4b
RR
1886 case wxJOIN_ROUND:
1887 default: { joinStyle = GDK_JOIN_ROUND; break; }
265898fd 1888 }
7d5af6fa 1889
265898fd 1890 gdk_gc_set_line_attributes( m_penGC, width, lineStyle, capStyle, joinStyle );
7d5af6fa 1891
265898fd
RR
1892 m_pen.GetColour().CalcPixel( m_cmap );
1893 gdk_gc_set_foreground( m_penGC, m_pen.GetColour().GetColor() );
903f689b 1894}
c801d85f 1895
ec758a20 1896void wxWindowDC::SetBrush( const wxBrush &brush )
c801d85f 1897{
223d09f6 1898 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 1899
265898fd 1900 if (m_brush == brush) return;
7d5af6fa 1901
265898fd 1902 m_brush = brush;
7d5af6fa 1903
265898fd 1904 if (!m_brush.Ok()) return;
7d5af6fa 1905
6db90681 1906 if (!m_window) return;
7d5af6fa 1907
265898fd
RR
1908 m_brush.GetColour().CalcPixel( m_cmap );
1909 gdk_gc_set_foreground( m_brushGC, m_brush.GetColour().GetColor() );
7d5af6fa 1910
956dbab1 1911 gdk_gc_set_fill( m_brushGC, GDK_SOLID );
7d5af6fa 1912
4fcd73bd 1913 if ((m_brush.GetStyle() == wxSTIPPLE) && (m_brush.GetStipple()->Ok()))
265898fd 1914 {
4fcd73bd 1915 if (m_brush.GetStipple()->GetPixmap())
7d5af6fa 1916 {
956dbab1 1917 gdk_gc_set_fill( m_brushGC, GDK_TILED );
cd25b18c 1918 gdk_gc_set_tile( m_brushGC, m_brush.GetStipple()->GetPixmap() );
7d5af6fa 1919 }
b0e0d661 1920 else
7d5af6fa 1921 {
956dbab1 1922 gdk_gc_set_fill( m_brushGC, GDK_STIPPLED );
4fcd73bd 1923 gdk_gc_set_stipple( m_brushGC, m_brush.GetStipple()->GetBitmap() );
7d5af6fa 1924 }
265898fd 1925 }
7d5af6fa 1926
de2d2cdc
VZ
1927 if ((m_brush.GetStyle() == wxSTIPPLE_MASK_OPAQUE) && (m_brush.GetStipple()->GetMask()))
1928 {
e1208c31
RR
1929 gdk_gc_set_fill( m_textGC, GDK_OPAQUE_STIPPLED);
1930 gdk_gc_set_stipple( m_textGC, m_brush.GetStipple()->GetMask()->GetBitmap() );
de2d2cdc
VZ
1931 }
1932
265898fd
RR
1933 if (IS_HATCH(m_brush.GetStyle()))
1934 {
956dbab1 1935 gdk_gc_set_fill( m_brushGC, GDK_STIPPLED );
265898fd
RR
1936 int num = m_brush.GetStyle() - wxBDIAGONAL_HATCH;
1937 gdk_gc_set_stipple( m_brushGC, hatches[num] );
1938 }
903f689b 1939}
c801d85f 1940
ec758a20 1941void wxWindowDC::SetBackground( const wxBrush &brush )
46dc76ba 1942{
bbbbe360
RR
1943 /* CMB 21/7/98: Added SetBackground. Sets background brush
1944 * for Clear() and bg colour for shapes filled with cross-hatch brush */
7d5af6fa 1945
223d09f6 1946 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 1947
265898fd 1948 if (m_backgroundBrush == brush) return;
7d5af6fa 1949
265898fd 1950 m_backgroundBrush = brush;
7d5af6fa 1951
265898fd 1952 if (!m_backgroundBrush.Ok()) return;
7d5af6fa 1953
6db90681 1954 if (!m_window) return;
7d5af6fa 1955
265898fd
RR
1956 m_backgroundBrush.GetColour().CalcPixel( m_cmap );
1957 gdk_gc_set_background( m_brushGC, m_backgroundBrush.GetColour().GetColor() );
a802c3a1 1958 gdk_gc_set_background( m_penGC, m_backgroundBrush.GetColour().GetColor() );
031b2a7b 1959 gdk_gc_set_background( m_bgGC, m_backgroundBrush.GetColour().GetColor() );
265898fd 1960 gdk_gc_set_foreground( m_bgGC, m_backgroundBrush.GetColour().GetColor() );
3417c2cd
RR
1961
1962 gdk_gc_set_fill( m_bgGC, GDK_SOLID );
7d5af6fa 1963
cd25b18c 1964 if ((m_backgroundBrush.GetStyle() == wxSTIPPLE) && (m_backgroundBrush.GetStipple()->Ok()))
265898fd 1965 {
fd128b0c 1966 if (m_backgroundBrush.GetStipple()->GetPixmap())
7d5af6fa 1967 {
3417c2cd 1968 gdk_gc_set_fill( m_bgGC, GDK_TILED );
fd128b0c 1969 gdk_gc_set_tile( m_bgGC, m_backgroundBrush.GetStipple()->GetPixmap() );
7d5af6fa 1970 }
cd25b18c 1971 else
7d5af6fa 1972 {
3417c2cd 1973 gdk_gc_set_fill( m_bgGC, GDK_STIPPLED );
fd128b0c 1974 gdk_gc_set_stipple( m_bgGC, m_backgroundBrush.GetStipple()->GetBitmap() );
7d5af6fa 1975 }
265898fd 1976 }
7d5af6fa 1977
265898fd
RR
1978 if (IS_HATCH(m_backgroundBrush.GetStyle()))
1979 {
3417c2cd 1980 gdk_gc_set_fill( m_bgGC, GDK_STIPPLED );
265898fd
RR
1981 int num = m_backgroundBrush.GetStyle() - wxBDIAGONAL_HATCH;
1982 gdk_gc_set_stipple( m_bgGC, hatches[num] );
bbbbe360 1983 }
903f689b 1984}
46dc76ba 1985
ec758a20 1986void wxWindowDC::SetLogicalFunction( int function )
c801d85f 1987{
223d09f6 1988 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 1989
72174350
VZ
1990 if (m_logicalFunction == function)
1991 return;
1992
1993 // VZ: shouldn't this be a CHECK?
1994 if (!m_window)
1995 return;
01eaf507 1996
2b5f62a0 1997 GdkFunction mode;
265898fd
RR
1998 switch (function)
1999 {
3c679789
RR
2000 case wxXOR: mode = GDK_XOR; break;
2001 case wxINVERT: mode = GDK_INVERT; break;
2b5f62a0 2002#if (GTK_MINOR_VERSION > 0) || (GTK_MAJOR_VERSION > 1)
3c679789
RR
2003 case wxOR_REVERSE: mode = GDK_OR_REVERSE; break;
2004 case wxAND_REVERSE: mode = GDK_AND_REVERSE; break;
2005 case wxCLEAR: mode = GDK_CLEAR; break;
2006 case wxSET: mode = GDK_SET; break;
2007 case wxOR_INVERT: mode = GDK_OR_INVERT; break;
2008 case wxAND: mode = GDK_AND; break;
2009 case wxOR: mode = GDK_OR; break;
2010 case wxEQUIV: mode = GDK_EQUIV; break;
2011 case wxNAND: mode = GDK_NAND; break;
2012 case wxAND_INVERT: mode = GDK_AND_INVERT; break;
7d5af6fa
VZ
2013 case wxCOPY: mode = GDK_COPY; break;
2014 case wxNO_OP: mode = GDK_NOOP; break;
2015 case wxSRC_INVERT: mode = GDK_COPY_INVERT; break;
72174350 2016
2d52841d
RR
2017 // unsupported by GTK
2018 case wxNOR: mode = GDK_COPY; break;
2b5f62a0 2019#endif // GTK+ > 1.0
01eaf507 2020 default:
223d09f6 2021 wxFAIL_MSG( wxT("unsupported logical function") );
2b5f62a0 2022 mode = GDK_COPY;
265898fd 2023 }
7d5af6fa 2024
265898fd 2025 m_logicalFunction = function;
7d5af6fa 2026
265898fd
RR
2027 gdk_gc_set_function( m_penGC, mode );
2028 gdk_gc_set_function( m_brushGC, mode );
72174350
VZ
2029
2030 // to stay compatible with wxMSW, we don't apply ROPs to the text
3d2d8da1
RR
2031 // operations (i.e. DrawText/DrawRotatedText).
2032 // True, but mono-bitmaps use the m_textGC and they use ROPs as well.
2033 gdk_gc_set_function( m_textGC, mode );
903f689b 2034}
c801d85f 2035
ec758a20 2036void wxWindowDC::SetTextForeground( const wxColour &col )
c801d85f 2037{
223d09f6 2038 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 2039
71ec83d2
VZ
2040 // don't set m_textForegroundColour to an invalid colour as we'd crash
2041 // later then (we use m_textForegroundColour.GetColor() without checking
2042 // in a few places)
2043 if ( !col.Ok() || (m_textForegroundColour == col) )
2044 return;
7d5af6fa 2045
265898fd 2046 m_textForegroundColour = col;
7d5af6fa 2047
71ec83d2
VZ
2048 if ( m_window )
2049 {
2050 m_textForegroundColour.CalcPixel( m_cmap );
2051 gdk_gc_set_foreground( m_textGC, m_textForegroundColour.GetColor() );
2052 }
903f689b 2053}
c801d85f 2054
ec758a20 2055void wxWindowDC::SetTextBackground( const wxColour &col )
c801d85f 2056{
223d09f6 2057 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 2058
71ec83d2
VZ
2059 // same as above
2060 if ( !col.Ok() || (m_textBackgroundColour == col) )
2061 return;
7d5af6fa 2062
265898fd 2063 m_textBackgroundColour = col;
7d5af6fa 2064
71ec83d2
VZ
2065 if ( m_window )
2066 {
2067 m_textBackgroundColour.CalcPixel( m_cmap );
2068 gdk_gc_set_background( m_textGC, m_textBackgroundColour.GetColor() );
2069 }
903f689b 2070}
c801d85f 2071
ec758a20 2072void wxWindowDC::SetBackgroundMode( int mode )
c801d85f 2073{
223d09f6 2074 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 2075
265898fd 2076 m_backgroundMode = mode;
46dc76ba 2077
6db90681 2078 if (!m_window) return;
7d5af6fa 2079
265898fd
RR
2080 // CMB 21/7/98: fill style of cross-hatch brushes is affected by
2081 // transparent/solid background mode
7d5af6fa 2082
265898fd
RR
2083 if (m_brush.GetStyle() != wxSOLID && m_brush.GetStyle() != wxTRANSPARENT)
2084 {
2085 gdk_gc_set_fill( m_brushGC,
2086 (m_backgroundMode == wxTRANSPARENT) ? GDK_STIPPLED : GDK_OPAQUE_STIPPLED);
2087 }
903f689b 2088}
c801d85f 2089
ec758a20 2090void wxWindowDC::SetPalette( const wxPalette& WXUNUSED(palette) )
c801d85f 2091{
223d09f6 2092 wxFAIL_MSG( wxT("wxWindowDC::SetPalette not implemented") );
903f689b 2093}
c801d85f 2094
72cdf4c9 2095void wxWindowDC::DoSetClippingRegion( wxCoord x, wxCoord y, wxCoord width, wxCoord height )
c801d85f 2096{
223d09f6 2097 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 2098
6db90681 2099 if (!m_window) return;
7d5af6fa 2100
3d2d8da1 2101 wxRect rect;
265898fd
RR
2102 rect.x = XLOG2DEV(x);
2103 rect.y = YLOG2DEV(y);
2104 rect.width = XLOG2DEVREL(width);
2105 rect.height = YLOG2DEVREL(height);
5f170f33 2106
e1208c31 2107 if (!m_currentClippingRegion.IsNull())
993f97ee
RR
2108 m_currentClippingRegion.Intersect( rect );
2109 else
2110 m_currentClippingRegion.Union( rect );
5f170f33
VZ
2111
2112#if USE_PAINT_REGION
e1208c31 2113 if (!m_paintClippingRegion.IsNull())
3d2d8da1 2114 m_currentClippingRegion.Intersect( m_paintClippingRegion );
809934d2 2115#endif
3d2d8da1 2116
ee8dbe63
RL
2117 wxCoord xx, yy, ww, hh;
2118 m_currentClippingRegion.GetBox( xx, yy, ww, hh );
2119 wxDC::DoSetClippingRegion( xx, yy, ww, hh );
2120
3d2d8da1
RR
2121 gdk_gc_set_clip_region( m_penGC, m_currentClippingRegion.GetRegion() );
2122 gdk_gc_set_clip_region( m_brushGC, m_currentClippingRegion.GetRegion() );
2123 gdk_gc_set_clip_region( m_textGC, m_currentClippingRegion.GetRegion() );
2124 gdk_gc_set_clip_region( m_bgGC, m_currentClippingRegion.GetRegion() );
903f689b 2125}
c801d85f 2126
b0e0d661 2127void wxWindowDC::DoSetClippingRegionAsRegion( const wxRegion &region )
463c1fa1 2128{
223d09f6 2129 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 2130
463c1fa1
RR
2131 if (region.Empty())
2132 {
2133 DestroyClippingRegion();
2134 return;
2135 }
7d5af6fa 2136
6db90681 2137 if (!m_window) return;
5f170f33 2138
e1208c31 2139 if (!m_currentClippingRegion.IsNull())
993f97ee
RR
2140 m_currentClippingRegion.Intersect( region );
2141 else
2142 m_currentClippingRegion.Union( region );
5f170f33
VZ
2143
2144#if USE_PAINT_REGION
e1208c31 2145 if (!m_paintClippingRegion.IsNull())
3d2d8da1 2146 m_currentClippingRegion.Intersect( m_paintClippingRegion );
809934d2 2147#endif
3d2d8da1 2148
ee8dbe63
RL
2149 wxCoord xx, yy, ww, hh;
2150 m_currentClippingRegion.GetBox( xx, yy, ww, hh );
2151 wxDC::DoSetClippingRegion( xx, yy, ww, hh );
2152
3d2d8da1
RR
2153 gdk_gc_set_clip_region( m_penGC, m_currentClippingRegion.GetRegion() );
2154 gdk_gc_set_clip_region( m_brushGC, m_currentClippingRegion.GetRegion() );
2155 gdk_gc_set_clip_region( m_textGC, m_currentClippingRegion.GetRegion() );
2156 gdk_gc_set_clip_region( m_bgGC, m_currentClippingRegion.GetRegion() );
463c1fa1
RR
2157}
2158
4bc67cc5 2159void wxWindowDC::DestroyClippingRegion()
c801d85f 2160{
223d09f6 2161 wxCHECK_RET( Ok(), wxT("invalid window dc") );
7d5af6fa 2162
265898fd 2163 wxDC::DestroyClippingRegion();
7d5af6fa 2164
3d2d8da1 2165 m_currentClippingRegion.Clear();
5f170f33
VZ
2166
2167#if USE_PAINT_REGION
3d2d8da1
RR
2168 if (!m_paintClippingRegion.IsEmpty())
2169 m_currentClippingRegion.Union( m_paintClippingRegion );
993f97ee 2170#endif
3d2d8da1 2171
6db90681 2172 if (!m_window) return;
7d5af6fa 2173
3d2d8da1
RR
2174 if (m_currentClippingRegion.IsEmpty())
2175 {
2176 gdk_gc_set_clip_rectangle( m_penGC, (GdkRectangle *) NULL );
2177 gdk_gc_set_clip_rectangle( m_brushGC, (GdkRectangle *) NULL );
2178 gdk_gc_set_clip_rectangle( m_textGC, (GdkRectangle *) NULL );
2179 gdk_gc_set_clip_rectangle( m_bgGC, (GdkRectangle *) NULL );
2180 }
2181 else
2182 {
2183 gdk_gc_set_clip_region( m_penGC, m_currentClippingRegion.GetRegion() );
2184 gdk_gc_set_clip_region( m_brushGC, m_currentClippingRegion.GetRegion() );
2185 gdk_gc_set_clip_region( m_textGC, m_currentClippingRegion.GetRegion() );
2186 gdk_gc_set_clip_region( m_bgGC, m_currentClippingRegion.GetRegion() );
2187 }
903f689b 2188}
c801d85f 2189
4bc67cc5 2190void wxWindowDC::Destroy()
dbf858b5 2191{
3d2d8da1 2192 if (m_penGC) wxFreePoolGC( m_penGC );
265898fd 2193 m_penGC = (GdkGC*) NULL;
3d2d8da1 2194 if (m_brushGC) wxFreePoolGC( m_brushGC );
265898fd 2195 m_brushGC = (GdkGC*) NULL;
3d2d8da1 2196 if (m_textGC) wxFreePoolGC( m_textGC );
265898fd 2197 m_textGC = (GdkGC*) NULL;
3d2d8da1 2198 if (m_bgGC) wxFreePoolGC( m_bgGC );
265898fd 2199 m_bgGC = (GdkGC*) NULL;
dbf858b5
RR
2200}
2201
238d735d
RR
2202void wxWindowDC::ComputeScaleAndOrigin()
2203{
2204 /* CMB: copy scale to see if it changes */
2205 double origScaleX = m_scaleX;
2206 double origScaleY = m_scaleY;
2207
2208 wxDC::ComputeScaleAndOrigin();
2209
2210 /* CMB: if scale has changed call SetPen to recalulate the line width */
2211 if ((m_scaleX != origScaleX || m_scaleY != origScaleY) &&
2212 (m_pen.Ok()))
2213 {
2214 /* this is a bit artificial, but we need to force wxDC to think
2215 the pen has changed */
2216 wxPen pen = m_pen;
2217 m_pen = wxNullPen;
2218 SetPen( pen );
2219 }
2220}
2221
b0e0d661
VZ
2222// Resolution in pixels per logical inch
2223wxSize wxWindowDC::GetPPI() const
2224{
7a5e6267 2225 return wxSize( (int) (m_mm_to_pix_x * 25.4 + 0.5), (int) (m_mm_to_pix_y * 25.4 + 0.5));
b0e0d661
VZ
2226}
2227
2228int wxWindowDC::GetDepth() const
c801d85f 2229{
223d09f6 2230 wxFAIL_MSG(wxT("not implemented"));
b0e0d661
VZ
2231
2232 return -1;
903f689b 2233}
c801d85f 2234
ec758a20
RR
2235
2236//-----------------------------------------------------------------------------
2237// wxPaintDC
2238//-----------------------------------------------------------------------------
2239
f469b27c 2240IMPLEMENT_DYNAMIC_CLASS(wxPaintDC, wxClientDC)
ec758a20
RR
2241
2242wxPaintDC::wxPaintDC( wxWindow *win )
f469b27c 2243 : wxClientDC( win )
ec758a20 2244{
b6fa52db
RR
2245#if USE_PAINT_REGION
2246 if (!win->m_clipPaintRegion)
2247 return;
f469b27c 2248
e1208c31 2249 m_paintClippingRegion = win->GetUpdateRegion();
5f170f33
VZ
2250 GdkRegion *region = m_paintClippingRegion.GetRegion();
2251 if ( region )
2252 {
1e6feb95
VZ
2253 m_paintClippingRegion = win->GetUpdateRegion();
2254 GdkRegion *region = m_paintClippingRegion.GetRegion();
2255 if ( region )
2256 {
2257 m_currentClippingRegion.Union( m_paintClippingRegion );
5f170f33 2258
1e6feb95
VZ
2259 gdk_gc_set_clip_region( m_penGC, region );
2260 gdk_gc_set_clip_region( m_brushGC, region );
2261 gdk_gc_set_clip_region( m_textGC, region );
2262 gdk_gc_set_clip_region( m_bgGC, region );
2263 }
5f170f33 2264 }
1e6feb95 2265#endif // USE_PAINT_REGION
ec758a20
RR
2266}
2267
2268//-----------------------------------------------------------------------------
2269// wxClientDC
2270//-----------------------------------------------------------------------------
2271
f469b27c 2272IMPLEMENT_DYNAMIC_CLASS(wxClientDC, wxWindowDC)
ec758a20 2273
f469b27c
VZ
2274wxClientDC::wxClientDC( wxWindow *win )
2275 : wxWindowDC( win )
ec758a20 2276{
4f55a07f
VZ
2277 wxCHECK_RET( win, _T("NULL window in wxClientDC::wxClientDC") );
2278
1e6feb95
VZ
2279#ifdef __WXUNIVERSAL__
2280 wxPoint ptOrigin = win->GetClientAreaOrigin();
2281 SetDeviceOrigin(ptOrigin.x, ptOrigin.y);
2282 wxSize size = win->GetClientSize();
2283 SetClippingRegion(wxPoint(0, 0), size);
2284#endif // __WXUNIVERSAL__
ec758a20
RR
2285}
2286
4f55a07f
VZ
2287void wxClientDC::DoGetSize(int *width, int *height) const
2288{
2289 wxCHECK_RET( m_owner, _T("GetSize() doesn't work without window") );
2290
2291 m_owner->GetClientSize( width, height );
2292}
2293
3d2d8da1
RR
2294// ----------------------------------------------------------------------------
2295// wxDCModule
2296// ----------------------------------------------------------------------------
2297
2298class wxDCModule : public wxModule
2299{
2300public:
2301 bool OnInit();
2302 void OnExit();
2303
2304private:
2305 DECLARE_DYNAMIC_CLASS(wxDCModule)
2306};
2307
2308IMPLEMENT_DYNAMIC_CLASS(wxDCModule, wxModule)
2309
2310bool wxDCModule::OnInit()
2311{
2312 wxInitGCPool();
2313 return TRUE;
2314}
2315
2316void wxDCModule::OnExit()
2317{
2318 wxCleanUpGCPool();
2319}