]>
git.saurik.com Git - wxWidgets.git/blob - src/gtk1/dcclient.cpp
   1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk1/dcclient.cpp 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling, Chris Breeze 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  14 #define XCopyPlane XCOPYPLANE 
  19     #include "wx/math.h" // for floating-point functions 
  21     #include "wx/module.h" 
  24 #include "wx/fontutil.h" 
  26 #include "wx/gtk1/win_gtk.h" 
  27 #include "wx/gtk1/dcclient.h" 
  28 #include "wx/gtk1/dcmemory.h" 
  32 #include <gdk/gdkprivate.h> 
  35 //----------------------------------------------------------------------------- 
  37 //----------------------------------------------------------------------------- 
  39 #define USE_PAINT_REGION 1 
  41 //----------------------------------------------------------------------------- 
  43 //----------------------------------------------------------------------------- 
  53 #define IS_15_PIX_HATCH(s) ((s)==wxCROSSDIAG_HATCH || (s)==wxHORIZONTAL_HATCH || (s)==wxVERTICAL_HATCH) 
  54 #define IS_16_PIX_HATCH(s) ((s)!=wxCROSSDIAG_HATCH && (s)!=wxHORIZONTAL_HATCH && (s)!=wxVERTICAL_HATCH) 
  57 static GdkPixmap  
*hatches
[num_hatches
]; 
  58 static GdkPixmap 
**hatch_bitmap 
= (GdkPixmap 
**) NULL
; 
  60 extern GtkWidget 
*wxGetRootWindow(); 
  62 //----------------------------------------------------------------------------- 
  64 //----------------------------------------------------------------------------- 
  66 const double RAD2DEG  
= 180.0 / M_PI
; 
  68 // ---------------------------------------------------------------------------- 
  70 // ---------------------------------------------------------------------------- 
  72 static inline double dmax(double a
, double b
) { return a 
> b 
? a 
: b
; } 
  73 static inline double dmin(double a
, double b
) { return a 
< b 
? a 
: b
; } 
  75 static inline double DegToRad(double deg
) { return (deg 
* M_PI
) / 180.0; } 
  77 //----------------------------------------------------------------------------- 
  78 // temporary implementation of the missing GDK function 
  79 //----------------------------------------------------------------------------- 
  81 #include "gdk/gdkprivate.h" 
  83 void gdk_wx_draw_bitmap(GdkDrawable  
*drawable
, 
  93     wxCHECK_RET( drawable
, _T("NULL drawable in gdk_wx_draw_bitmap") ); 
  94     wxCHECK_RET( src
, _T("NULL src in gdk_wx_draw_bitmap") ); 
  95     wxCHECK_RET( gc
, _T("NULL gc in gdk_wx_draw_bitmap") ); 
  97     GdkWindowPrivate 
*drawable_private
; 
  98     GdkWindowPrivate 
*src_private
; 
  99     GdkGCPrivate 
*gc_private
; 
 101     drawable_private 
= (GdkWindowPrivate
*) drawable
; 
 102     src_private 
= (GdkWindowPrivate
*) src
; 
 103     if (drawable_private
->destroyed 
|| src_private
->destroyed
) 
 106     gint src_width 
= src_private
->width
; 
 107     gint src_height 
= src_private
->height
; 
 109     gc_private 
= (GdkGCPrivate
*) gc
; 
 111     if (width 
== -1) width 
= src_width
; 
 112     if (height 
== -1) height 
= src_height
; 
 114     XCopyPlane( drawable_private
->xdisplay
, 
 115                 src_private
->xwindow
, 
 116                 drawable_private
->xwindow
, 
 124 //----------------------------------------------------------------------------- 
 125 // Implement Pool of Graphic contexts. Creating them takes too much time. 
 126 //----------------------------------------------------------------------------- 
 152 #define GC_POOL_ALLOC_SIZE 100 
 154 static int wxGCPoolSize 
= 0; 
 156 static wxGC 
*wxGCPool 
= NULL
; 
 158 static void wxInitGCPool() 
 160     // This really could wait until the first call to 
 161     // wxGetPoolGC, but we will make the first allocation 
 162     // now when other initialization is being performed. 
 164     // Set initial pool size. 
 165     wxGCPoolSize 
= GC_POOL_ALLOC_SIZE
; 
 167     // Allocate initial pool. 
 168     wxGCPool 
= (wxGC 
*)malloc(wxGCPoolSize 
* sizeof(wxGC
)); 
 169     if (wxGCPool 
== NULL
) 
 171         // If we cannot malloc, then fail with error 
 172         // when debug is enabled.  If debug is not enabled, 
 173         // the problem will eventually get caught 
 175         wxFAIL_MSG( wxT("Cannot allocate GC pool") ); 
 179     // Zero initial pool. 
 180     memset(wxGCPool
, 0, wxGCPoolSize 
* sizeof(wxGC
)); 
 183 static void wxCleanUpGCPool() 
 185     for (int i 
= 0; i 
< wxGCPoolSize
; i
++) 
 187         if (wxGCPool
[i
].m_gc
) 
 188             gdk_gc_unref( wxGCPool
[i
].m_gc 
); 
 196 static GdkGC
* wxGetPoolGC( GdkWindow 
*window
, wxPoolGCType type 
) 
 200     // Look for an available GC. 
 201     for (int i 
= 0; i 
< wxGCPoolSize
; i
++) 
 203         if (!wxGCPool
[i
].m_gc
) 
 205             wxGCPool
[i
].m_gc 
= gdk_gc_new( window 
); 
 206             gdk_gc_set_exposures( wxGCPool
[i
].m_gc
, FALSE 
); 
 207             wxGCPool
[i
].m_type 
= type
; 
 208             wxGCPool
[i
].m_used 
= false; 
 210         if ((!wxGCPool
[i
].m_used
) && (wxGCPool
[i
].m_type 
== type
)) 
 212             wxGCPool
[i
].m_used 
= true; 
 213             return wxGCPool
[i
].m_gc
; 
 217     // We did not find an available GC. 
 218     // We need to grow the GC pool. 
 219     pptr 
= (wxGC 
*)realloc(wxGCPool
, 
 220         (wxGCPoolSize 
+ GC_POOL_ALLOC_SIZE
)*sizeof(wxGC
)); 
 223         // Initialize newly allocated pool. 
 225         memset(&wxGCPool
[wxGCPoolSize
], 0, 
 226             GC_POOL_ALLOC_SIZE
*sizeof(wxGC
)); 
 228         // Initialize entry we will return. 
 229         wxGCPool
[wxGCPoolSize
].m_gc 
= gdk_gc_new( window 
); 
 230         gdk_gc_set_exposures( wxGCPool
[wxGCPoolSize
].m_gc
, FALSE 
); 
 231         wxGCPool
[wxGCPoolSize
].m_type 
= type
; 
 232         wxGCPool
[wxGCPoolSize
].m_used 
= true; 
 234         // Set new value of pool size. 
 235         wxGCPoolSize 
+= GC_POOL_ALLOC_SIZE
; 
 237         // Return newly allocated entry. 
 238         return wxGCPool
[wxGCPoolSize
-GC_POOL_ALLOC_SIZE
].m_gc
; 
 241     // The realloc failed.  Fall through to error. 
 242     wxFAIL_MSG( wxT("No GC available") ); 
 244     return (GdkGC
*) NULL
; 
 247 static void wxFreePoolGC( GdkGC 
*gc 
) 
 249     for (int i 
= 0; i 
< wxGCPoolSize
; i
++) 
 251         if (wxGCPool
[i
].m_gc 
== gc
) 
 253             wxGCPool
[i
].m_used 
= false; 
 258     wxFAIL_MSG( wxT("Wrong GC") ); 
 261 //----------------------------------------------------------------------------- 
 263 //----------------------------------------------------------------------------- 
 265 IMPLEMENT_ABSTRACT_CLASS(wxWindowDCImpl
, wxDC
) 
 267 wxWindowDCImpl::wxWindowDCImpl(wxDC 
*owner
) 
 270     m_penGC 
= (GdkGC 
*) NULL
; 
 271     m_brushGC 
= (GdkGC 
*) NULL
; 
 272     m_textGC 
= (GdkGC 
*) NULL
; 
 273     m_bgGC 
= (GdkGC 
*) NULL
; 
 274     m_cmap 
= (GdkColormap 
*) NULL
; 
 276     m_isScreenDC 
= false; 
 277     m_owner 
= (wxWindow 
*)NULL
; 
 280 wxWindowDCImpl::wxWindowDCImpl(wxDC 
*owner
, wxWindow 
*window
) 
 283     wxASSERT_MSG( window
, wxT("DC needs a window") ); 
 285     m_penGC 
= (GdkGC 
*) NULL
; 
 286     m_brushGC 
= (GdkGC 
*) NULL
; 
 287     m_textGC 
= (GdkGC 
*) NULL
; 
 288     m_bgGC 
= (GdkGC 
*) NULL
; 
 289     m_cmap 
= (GdkColormap 
*) NULL
; 
 290     m_owner 
= (wxWindow 
*)NULL
; 
 292     m_isScreenDC 
= false; 
 293     m_font 
= window
->GetFont(); 
 295     GtkWidget 
*widget 
= window
->m_wxwindow
; 
 297     // Some controls don't have m_wxwindow - like wxStaticBox, but the user 
 298     // code should still be able to create wxClientDCs for them, so we will 
 299     // use the parent window here then. 
 302         window 
= window
->GetParent(); 
 303         widget 
= window
->m_wxwindow
; 
 306     wxASSERT_MSG( widget
, wxT("DC needs a widget") ); 
 308     GtkPizza 
*pizza 
= GTK_PIZZA( widget 
); 
 309     m_window 
= pizza
->bin_window
; 
 311     // Window not realized ? 
 314          // Don't report problems as per MSW. 
 320     m_cmap 
= gtk_widget_get_colormap( widget 
? widget 
: window
->m_widget 
); 
 324     /* this must be done after SetUpDC, bacause SetUpDC calls the 
 325        repective SetBrush, SetPen, SetBackground etc functions 
 326        to set up the DC. SetBackground call m_owner->SetBackground 
 327        and this might not be desired as the standard dc background 
 328        is white whereas a window might assume gray to be the 
 329        standard (as e.g. wxStatusBar) */ 
 334 wxWindowDCImpl::~wxWindowDCImpl() 
 339 void wxWindowDCImpl::SetUpDC() 
 343     wxASSERT_MSG( !m_penGC
, wxT("GCs already created") ); 
 347         m_penGC 
= wxGetPoolGC( m_window
, wxPEN_SCREEN 
); 
 348         m_brushGC 
= wxGetPoolGC( m_window
, wxBRUSH_SCREEN 
); 
 349         m_textGC 
= wxGetPoolGC( m_window
, wxTEXT_SCREEN 
); 
 350         m_bgGC 
= wxGetPoolGC( m_window
, wxBG_SCREEN 
); 
 352     else if (m_isMemDC 
&& (((wxMemoryDCImpl
*)this)->m_selected
.GetDepth() == 1)) 
 354         m_penGC 
= wxGetPoolGC( m_window
, wxPEN_MONO 
); 
 355         m_brushGC 
= wxGetPoolGC( m_window
, wxBRUSH_MONO 
); 
 356         m_textGC 
= wxGetPoolGC( m_window
, wxTEXT_MONO 
); 
 357         m_bgGC 
= wxGetPoolGC( m_window
, wxBG_MONO 
); 
 361         m_penGC 
= wxGetPoolGC( m_window
, wxPEN_COLOUR 
); 
 362         m_brushGC 
= wxGetPoolGC( m_window
, wxBRUSH_COLOUR 
); 
 363         m_textGC 
= wxGetPoolGC( m_window
, wxTEXT_COLOUR 
); 
 364         m_bgGC 
= wxGetPoolGC( m_window
, wxBG_COLOUR 
); 
 367     /* background colour */ 
 368     m_backgroundBrush 
= *wxWHITE_BRUSH
; 
 369     m_backgroundBrush
.GetColour().CalcPixel( m_cmap 
); 
 370     GdkColor 
*bg_col 
= m_backgroundBrush
.GetColour().GetColor(); 
 373     m_textForegroundColour
.CalcPixel( m_cmap 
); 
 374     gdk_gc_set_foreground( m_textGC
, m_textForegroundColour
.GetColor() ); 
 376     m_textBackgroundColour
.CalcPixel( m_cmap 
); 
 377     gdk_gc_set_background( m_textGC
, m_textBackgroundColour
.GetColor() ); 
 379     gdk_gc_set_fill( m_textGC
, GDK_SOLID 
); 
 382     m_pen
.GetColour().CalcPixel( m_cmap 
); 
 383     gdk_gc_set_foreground( m_penGC
, m_pen
.GetColour().GetColor() ); 
 384     gdk_gc_set_background( m_penGC
, bg_col 
); 
 386     gdk_gc_set_line_attributes( m_penGC
, 0, GDK_LINE_SOLID
, GDK_CAP_NOT_LAST
, GDK_JOIN_ROUND 
); 
 389     m_brush
.GetColour().CalcPixel( m_cmap 
); 
 390     gdk_gc_set_foreground( m_brushGC
, m_brush
.GetColour().GetColor() ); 
 391     gdk_gc_set_background( m_brushGC
, bg_col 
); 
 393     gdk_gc_set_fill( m_brushGC
, GDK_SOLID 
); 
 396     gdk_gc_set_background( m_bgGC
, bg_col 
); 
 397     gdk_gc_set_foreground( m_bgGC
, bg_col 
); 
 399     gdk_gc_set_fill( m_bgGC
, GDK_SOLID 
); 
 402     gdk_gc_set_function( m_textGC
, GDK_COPY 
); 
 403     gdk_gc_set_function( m_brushGC
, GDK_COPY 
); 
 404     gdk_gc_set_function( m_penGC
, GDK_COPY 
); 
 407     gdk_gc_set_clip_rectangle( m_penGC
, (GdkRectangle 
*) NULL 
); 
 408     gdk_gc_set_clip_rectangle( m_brushGC
, (GdkRectangle 
*) NULL 
); 
 409     gdk_gc_set_clip_rectangle( m_textGC
, (GdkRectangle 
*) NULL 
); 
 410     gdk_gc_set_clip_rectangle( m_bgGC
, (GdkRectangle 
*) NULL 
); 
 414         hatch_bitmap    
= hatches
; 
 415         hatch_bitmap
[0] = gdk_bitmap_create_from_data( (GdkWindow 
*) NULL
, bdiag_bits
, bdiag_width
, bdiag_height 
); 
 416         hatch_bitmap
[1] = gdk_bitmap_create_from_data( (GdkWindow 
*) NULL
, cdiag_bits
, cdiag_width
, cdiag_height 
); 
 417         hatch_bitmap
[2] = gdk_bitmap_create_from_data( (GdkWindow 
*) NULL
, fdiag_bits
, fdiag_width
, fdiag_height 
); 
 418         hatch_bitmap
[3] = gdk_bitmap_create_from_data( (GdkWindow 
*) NULL
, cross_bits
, cross_width
, cross_height 
); 
 419         hatch_bitmap
[4] = gdk_bitmap_create_from_data( (GdkWindow 
*) NULL
, horiz_bits
, horiz_width
, horiz_height 
); 
 420         hatch_bitmap
[5] = gdk_bitmap_create_from_data( (GdkWindow 
*) NULL
, verti_bits
, verti_width
, verti_height 
); 
 424 void wxWindowDCImpl::DoGetSize( int* width
, int* height 
) const 
 426     wxCHECK_RET( m_owner
, _T("GetSize() doesn't work without window") ); 
 428     m_owner
->GetSize(width
, height
); 
 431 extern bool wxDoFloodFill(wxDC 
*dc
, wxCoord x
, wxCoord y
, 
 432                           const wxColour 
& col
, int style
); 
 434 bool wxWindowDCImpl::DoFloodFill(wxCoord x
, wxCoord y
, 
 435                              const wxColour
& col
, int style
) 
 437     return wxDoFloodFill(GetOwner(), x
, y
, col
, style
); 
 440 bool wxWindowDCImpl::DoGetPixel( wxCoord x1
, wxCoord y1
, wxColour 
*col 
) const 
 442     // Generic (and therefore rather inefficient) method. 
 443     // Could be improved. 
 445     wxBitmap 
bitmap(1, 1); 
 446     memdc
.SelectObject(bitmap
); 
 447     memdc
.Blit(0, 0, 1, 1, GetOwner(), x1
, y1
); 
 448     memdc
.SelectObject(wxNullBitmap
); 
 450     wxImage image 
= bitmap
.ConvertToImage(); 
 451     col
->Set(image
.GetRed(0, 0), image
.GetGreen(0, 0), image
.GetBlue(0, 0)); 
 455 void wxWindowDCImpl::DoDrawLine( wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2 
) 
 457     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 459     if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 462             gdk_draw_line( m_window
, m_penGC
, XLOG2DEV(x1
), YLOG2DEV(y1
), XLOG2DEV(x2
), YLOG2DEV(y2
) ); 
 464         CalcBoundingBox(x1
, y1
); 
 465         CalcBoundingBox(x2
, y2
); 
 469 void wxWindowDCImpl::DoCrossHair( wxCoord x
, wxCoord y 
) 
 471     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 473     if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 478         wxCoord xx 
= XLOG2DEV(x
); 
 479         wxCoord yy 
= YLOG2DEV(y
); 
 482             gdk_draw_line( m_window
, m_penGC
, 0, yy
, XLOG2DEVREL(w
), yy 
); 
 483             gdk_draw_line( m_window
, m_penGC
, xx
, 0, xx
, YLOG2DEVREL(h
) ); 
 488 void wxWindowDCImpl::DoDrawArc( wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2
, 
 489                             wxCoord xc
, wxCoord yc 
) 
 491     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 493     wxCoord xx1 
= XLOG2DEV(x1
); 
 494     wxCoord yy1 
= YLOG2DEV(y1
); 
 495     wxCoord xx2 
= XLOG2DEV(x2
); 
 496     wxCoord yy2 
= YLOG2DEV(y2
); 
 497     wxCoord xxc 
= XLOG2DEV(xc
); 
 498     wxCoord yyc 
= YLOG2DEV(yc
); 
 499     double dx 
= xx1 
- xxc
; 
 500     double dy 
= yy1 
- yyc
; 
 501     double radius 
= sqrt((double)(dx
*dx
+dy
*dy
)); 
 502     wxCoord   r      
= (wxCoord
)radius
; 
 503     double radius1
, radius2
; 
 505     if (xx1 
== xx2 
&& yy1 
== yy2
) 
 510     else if ( wxIsNullDouble(radius
) ) 
 517         radius1 
= (xx1 
- xxc 
== 0) ? 
 518             (yy1 
- yyc 
< 0) ? 90.0 : -90.0 : 
 519             -atan2(double(yy1
-yyc
), double(xx1
-xxc
)) * RAD2DEG
; 
 520         radius2 
= (xx2 
- xxc 
== 0) ? 
 521             (yy2 
- yyc 
< 0) ? 90.0 : -90.0 : 
 522             -atan2(double(yy2
-yyc
), double(xx2
-xxc
)) * RAD2DEG
; 
 524     wxCoord alpha1 
= wxCoord(radius1 
* 64.0); 
 525     wxCoord alpha2 
= wxCoord((radius2 
- radius1
) * 64.0); 
 526     while (alpha2 
<= 0) alpha2 
+= 360*64; 
 527     while (alpha1 
> 360*64) alpha1 
-= 360*64; 
 531         if (m_brush
.GetStyle() != wxTRANSPARENT
) 
 533             if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
 535                 gdk_gc_set_ts_origin( m_textGC
, 
 536                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 537                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 538                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 539                 gdk_gc_set_ts_origin( m_textGC
, 0, 0 ); 
 541             if (IS_15_PIX_HATCH(m_brush
.GetStyle())) 
 543                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 15, m_deviceOriginY 
% 15 ); 
 544                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 545                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 547             if (IS_16_PIX_HATCH(m_brush
.GetStyle())) 
 549                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 16, m_deviceOriginY 
% 16 ); 
 550                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 551                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 553             if (m_brush
.GetStyle() == wxSTIPPLE
) 
 555                 gdk_gc_set_ts_origin( m_brushGC
, 
 556                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 557                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 558                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 559                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 563                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 567         if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 569             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 571             gdk_draw_line( m_window
, m_penGC
, xx1
, yy1
, xxc
, yyc 
); 
 572             gdk_draw_line( m_window
, m_penGC
, xxc
, yyc
, xx2
, yy2 
); 
 576     CalcBoundingBox (x1
, y1
); 
 577     CalcBoundingBox (x2
, y2
); 
 580 void wxWindowDCImpl::DoDrawEllipticArc( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double sa
, double ea 
) 
 582     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 584     wxCoord xx 
= XLOG2DEV(x
); 
 585     wxCoord yy 
= YLOG2DEV(y
); 
 586     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 587     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 589     // CMB: handle -ve width and/or height 
 590     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 591     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 595         wxCoord start 
= wxCoord(sa 
* 64.0); 
 596         wxCoord end 
= wxCoord((ea
-sa
) * 64.0); 
 598         if (m_brush
.GetStyle() != wxTRANSPARENT
) 
 600             if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
 602                 gdk_gc_set_ts_origin( m_textGC
, 
 603                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 604                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 605                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xx
, yy
, ww
, hh
, start
, end 
); 
 606                 gdk_gc_set_ts_origin( m_textGC
, 0, 0 ); 
 608             if (IS_15_PIX_HATCH(m_brush
.GetStyle())) 
 610                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 15, m_deviceOriginY 
% 15 ); 
 611                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, start
, end 
); 
 612                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 614             if (IS_16_PIX_HATCH(m_brush
.GetStyle())) 
 616                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 16, m_deviceOriginY 
% 16 ); 
 617                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, start
, end 
); 
 618                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 620             if (m_brush
.GetStyle() == wxSTIPPLE
) 
 622                 gdk_gc_set_ts_origin( m_brushGC
, 
 623                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 624                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 625                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, start
, end 
); 
 626                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 630                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, start
, end 
); 
 634         if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 635             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xx
, yy
, ww
, hh
, start
, end 
); 
 638     CalcBoundingBox (x
, y
); 
 639     CalcBoundingBox (x 
+ width
, y 
+ height
); 
 642 void wxWindowDCImpl::DoDrawPoint( wxCoord x
, wxCoord y 
) 
 644     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 646     if ((m_pen
.GetStyle() != wxTRANSPARENT
) && m_window
) 
 647         gdk_draw_point( m_window
, m_penGC
, XLOG2DEV(x
), YLOG2DEV(y
) ); 
 649     CalcBoundingBox (x
, y
); 
 652 void wxWindowDCImpl::DoDrawLines( int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset 
) 
 654     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 656     if (m_pen
.GetStyle() == wxTRANSPARENT
) return; 
 660     GdkPoint 
* const gpts 
= new GdkPoint
[n
]; 
 663         wxFAIL_MSG( wxT("Cannot allocate PolyLine") ); 
 667     for (int i 
= 0; i 
< n
; i
++) 
 669         wxCoord x 
= points
[i
].x 
+ xoffset
, 
 670                 y 
= points
[i
].y 
+ yoffset
; 
 672         CalcBoundingBox(x 
+ xoffset
, y 
+ yoffset
); 
 674         gpts
[i
].x 
= XLOG2DEV(x
); 
 675         gpts
[i
].y 
= YLOG2DEV(y
); 
 678     gdk_draw_lines( m_window
, m_penGC
, gpts
, n
); 
 683 void wxWindowDCImpl::DoDrawPolygon( int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
, int WXUNUSED(fillStyle
) ) 
 685     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 689     GdkPoint 
* const gpts 
= new GdkPoint
[n
]; 
 691     for (int i 
= 0 ; i 
< n 
; i
++) 
 693         wxCoord x 
= points
[i
].x 
+ xoffset
, 
 694                 y 
= points
[i
].y 
+ yoffset
; 
 696         CalcBoundingBox(x 
+ xoffset
, y 
+ yoffset
); 
 698         gpts
[i
].x 
= XLOG2DEV(x
); 
 699         gpts
[i
].y 
= YLOG2DEV(y
); 
 702     if (m_brush
.GetStyle() == wxSOLID
) 
 704         gdk_draw_polygon( m_window
, m_brushGC
, TRUE
, gpts
, n 
); 
 706     else if (m_brush
.GetStyle() != wxTRANSPARENT
) 
 708         if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
 710             gdk_gc_set_ts_origin( m_textGC
, 
 711                                   m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 712                                   m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 713             gdk_draw_polygon( m_window
, m_textGC
, TRUE
, gpts
, n 
); 
 714             gdk_gc_set_ts_origin( m_textGC
, 0, 0 ); 
 716         if (IS_15_PIX_HATCH(m_brush
.GetStyle())) 
 718             gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 15, m_deviceOriginY 
% 15 ); 
 719             gdk_draw_polygon( m_window
, m_brushGC
, TRUE
, gpts
, n 
); 
 720             gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 722         if (IS_16_PIX_HATCH(m_brush
.GetStyle())) 
 724             gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 16, m_deviceOriginY 
% 16 ); 
 725             gdk_draw_polygon( m_window
, m_brushGC
, TRUE
, gpts
, n 
); 
 726             gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 728         if (m_brush
.GetStyle() == wxSTIPPLE
) 
 730             gdk_gc_set_ts_origin( m_brushGC
, 
 731                                   m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 732                                   m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 733             gdk_draw_polygon( m_window
, m_brushGC
, TRUE
, gpts
, n 
); 
 734             gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 738             gdk_draw_polygon( m_window
, m_brushGC
, TRUE
, gpts
, n 
); 
 742     if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 744         gdk_draw_polygon( m_window
, m_penGC
, FALSE
, gpts
, n 
); 
 751 void wxWindowDCImpl::DoDrawRectangle( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height 
) 
 753     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 755     wxCoord xx 
= XLOG2DEV(x
); 
 756     wxCoord yy 
= YLOG2DEV(y
); 
 757     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 758     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 760     // CMB: draw nothing if transformed w or h is 0 
 761     if (ww 
== 0 || hh 
== 0) return; 
 763     // CMB: handle -ve width and/or height 
 764     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 765     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 769         if (m_brush
.GetStyle() != wxTRANSPARENT
) 
 771             if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
 773                 gdk_gc_set_ts_origin( m_textGC
, 
 774                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 775                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 776                 gdk_draw_rectangle( m_window
, m_textGC
, TRUE
, xx
, yy
, ww
, hh 
); 
 777                 gdk_gc_set_ts_origin( m_textGC
, 0, 0 ); 
 779             if (IS_15_PIX_HATCH(m_brush
.GetStyle())) 
 781                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 15, m_deviceOriginY 
% 15 ); 
 782                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh 
); 
 783                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 785             if (IS_16_PIX_HATCH(m_brush
.GetStyle())) 
 787                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 16, m_deviceOriginY 
% 16 ); 
 788                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh 
); 
 789                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 791             if (m_brush
.GetStyle() == wxSTIPPLE
) 
 793                 gdk_gc_set_ts_origin( m_brushGC
, 
 794                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 795                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 796                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh 
); 
 797                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 801                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh 
); 
 805         if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 806             gdk_draw_rectangle( m_window
, m_penGC
, FALSE
, xx
, yy
, ww
-1, hh
-1 ); 
 809     CalcBoundingBox( x
, y 
); 
 810     CalcBoundingBox( x 
+ width
, y 
+ height 
); 
 813 void wxWindowDCImpl::DoDrawRoundedRectangle( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double radius 
) 
 815     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 817     if (radius 
< 0.0) radius 
= - radius 
* ((width 
< height
) ? width 
: height
); 
 819     wxCoord xx 
= XLOG2DEV(x
); 
 820     wxCoord yy 
= YLOG2DEV(y
); 
 821     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 822     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 823     wxCoord rr 
= XLOG2DEVREL((wxCoord
)radius
); 
 825     // CMB: handle -ve width and/or height 
 826     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 827     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 829     // CMB: if radius is zero use DrawRectangle() instead to avoid 
 830     // X drawing errors with small radii 
 833         DoDrawRectangle( x
, y
, width
, height 
); 
 837     // CMB: draw nothing if transformed w or h is 0 
 838     if (ww 
== 0 || hh 
== 0) return; 
 840     // CMB: adjust size if outline is drawn otherwise the result is 
 841     // 1 pixel too wide and high 
 842     if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 850         // CMB: ensure dd is not larger than rectangle otherwise we 
 851         // get an hour glass shape 
 853         if (dd 
> ww
) dd 
= ww
; 
 854         if (dd 
> hh
) dd 
= hh
; 
 857         if (m_brush
.GetStyle() != wxTRANSPARENT
) 
 859             if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
 861                 gdk_gc_set_ts_origin( m_textGC
, 
 862                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 863                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 864                 gdk_draw_rectangle( m_window
, m_textGC
, TRUE
, xx
+rr
, yy
, ww
-dd
+1, hh 
); 
 865                 gdk_draw_rectangle( m_window
, m_textGC
, TRUE
, xx
, yy
+rr
, ww
, hh
-dd
+1 ); 
 866                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xx
, yy
, dd
, dd
, 90*64, 90*64 ); 
 867                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64 ); 
 868                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64 ); 
 869                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64 ); 
 870                 gdk_gc_set_ts_origin( m_textGC
, 0, 0 ); 
 872             if (IS_15_PIX_HATCH(m_brush
.GetStyle())) 
 874                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 15, m_deviceOriginY 
% 15 ); 
 875                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
+rr
, yy
, ww
-dd
+1, hh 
); 
 876                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
+rr
, ww
, hh
-dd
+1 ); 
 877                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, dd
, dd
, 90*64, 90*64 ); 
 878                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64 ); 
 879                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64 ); 
 880                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64 ); 
 881                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 883             if (IS_16_PIX_HATCH(m_brush
.GetStyle())) 
 885                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 16, m_deviceOriginY 
% 16 ); 
 886                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
+rr
, yy
, ww
-dd
+1, hh 
); 
 887                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
+rr
, ww
, hh
-dd
+1 ); 
 888                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, dd
, dd
, 90*64, 90*64 ); 
 889                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64 ); 
 890                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64 ); 
 891                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64 ); 
 892                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 894             if (m_brush
.GetStyle() == wxSTIPPLE
) 
 896                 gdk_gc_set_ts_origin( m_brushGC
, 
 897                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 898                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 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 ); 
 909                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
+rr
, yy
, ww
-dd
+1, hh 
); 
 910                 gdk_draw_rectangle( m_window
, m_brushGC
, TRUE
, xx
, yy
+rr
, ww
, hh
-dd
+1 ); 
 911                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, dd
, dd
, 90*64, 90*64 ); 
 912                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64 ); 
 913                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64 ); 
 914                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64 ); 
 918         if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 920             gdk_draw_line( m_window
, m_penGC
, xx
+rr
+1, yy
, xx
+ww
-rr
, yy 
); 
 921             gdk_draw_line( m_window
, m_penGC
, xx
+rr
+1, yy
+hh
, xx
+ww
-rr
, yy
+hh 
); 
 922             gdk_draw_line( m_window
, m_penGC
, xx
, yy
+rr
+1, xx
, yy
+hh
-rr 
); 
 923             gdk_draw_line( m_window
, m_penGC
, xx
+ww
, yy
+rr
+1, xx
+ww
, yy
+hh
-rr 
); 
 924             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xx
, yy
, dd
, dd
, 90*64, 90*64 ); 
 925             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64 ); 
 926             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64 ); 
 927             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64 ); 
 931     // this ignores the radius 
 932     CalcBoundingBox( x
, y 
); 
 933     CalcBoundingBox( x 
+ width
, y 
+ height 
); 
 936 void wxWindowDCImpl::DoDrawEllipse( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height 
) 
 938     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 940     wxCoord xx 
= XLOG2DEV(x
); 
 941     wxCoord yy 
= YLOG2DEV(y
); 
 942     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 943     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 945     // CMB: handle -ve width and/or height 
 946     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 947     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 951         if (m_brush
.GetStyle() != wxTRANSPARENT
) 
 953             if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
 955                 gdk_gc_set_ts_origin( m_textGC
, 
 956                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 957                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 958                 gdk_draw_arc( m_window
, m_textGC
, TRUE
, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 959                 gdk_gc_set_ts_origin( m_textGC
, 0, 0 ); 
 961             if (IS_15_PIX_HATCH(m_brush
.GetStyle())) 
 963                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 15, m_deviceOriginY 
% 15 ); 
 964                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 965                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 967             if (IS_16_PIX_HATCH(m_brush
.GetStyle())) 
 969                 gdk_gc_set_ts_origin( m_brushGC
, m_deviceOriginX 
% 16, m_deviceOriginY 
% 16 ); 
 970                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 971                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 973             if (m_brush
.GetStyle() == wxSTIPPLE
) 
 975                 gdk_gc_set_ts_origin( m_brushGC
, 
 976                                       m_deviceOriginX 
% m_brush
.GetStipple()->GetWidth(), 
 977                                       m_deviceOriginY 
% m_brush
.GetStipple()->GetHeight() ); 
 978                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 979                 gdk_gc_set_ts_origin( m_brushGC
, 0, 0 ); 
 983                 gdk_draw_arc( m_window
, m_brushGC
, TRUE
, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 987         if (m_pen
.GetStyle() != wxTRANSPARENT
) 
 988             gdk_draw_arc( m_window
, m_penGC
, FALSE
, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 991     CalcBoundingBox( x
, y 
); 
 992     CalcBoundingBox( x 
+ width
, y 
+ height 
); 
 995 void wxWindowDCImpl::DoDrawIcon( const wxIcon 
&icon
, wxCoord x
, wxCoord y 
) 
 997     // VZ: egcs 1.0.3 refuses to compile this without cast, no idea why 
 998     DoDrawBitmap( (const wxBitmap
&)icon
, x
, y
, true ); 
1001 void wxWindowDCImpl::DoDrawBitmap( const wxBitmap 
&bitmap
, 
1002                                wxCoord x
, wxCoord y
, 
1005     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1007     wxCHECK_RET( bitmap
.Ok(), wxT("invalid bitmap") ); 
1009     bool is_mono 
= (bitmap
.GetBitmap() != NULL
); 
1011     // scale/translate size and position 
1012     int xx 
= XLOG2DEV(x
); 
1013     int yy 
= YLOG2DEV(y
); 
1015     int w 
= bitmap
.GetWidth(); 
1016     int h 
= bitmap
.GetHeight(); 
1018     CalcBoundingBox( x
, y 
); 
1019     CalcBoundingBox( x 
+ w
, y 
+ h 
); 
1021     if (!m_window
) return; 
1023     int ww 
= XLOG2DEVREL(w
); 
1024     int hh 
= YLOG2DEVREL(h
); 
1026     // compare to current clipping region 
1027     if (!m_currentClippingRegion
.IsNull()) 
1029         wxRegion 
tmp( xx
,yy
,ww
,hh 
); 
1030         tmp
.Intersect( m_currentClippingRegion 
); 
1035     // scale bitmap if required 
1036     wxBitmap use_bitmap 
= bitmap
; 
1037     if ((w 
!= ww
) || (h 
!= hh
)) 
1038         use_bitmap 
= use_bitmap
.Rescale( 0, 0, ww
, hh
, ww
, hh 
); 
1040     // NB: We can't render pixbufs with GTK+ < 2.2, we need to use pixmaps code. 
1041     //     Pixbufs-based bitmaps with alpha channel don't have a mask, so we 
1042     //     have to call GetPixmap() here -- it converts the pixbuf into pixmap 
1043     //     and also creates the mask as a side-effect: 
1044     use_bitmap
.GetPixmap(); 
1046     // apply mask if any 
1047     GdkBitmap 
*mask 
= (GdkBitmap 
*) NULL
; 
1048     if (use_bitmap
.GetMask()) mask 
= use_bitmap
.GetMask()->GetBitmap(); 
1050     GdkBitmap 
*new_mask 
= (GdkBitmap
*) NULL
; 
1052     if (useMask 
&& mask
) 
1054         if (!m_currentClippingRegion
.IsNull()) 
1057             new_mask 
= gdk_pixmap_new( wxGetRootWindow()->window
, ww
, hh
, 1 ); 
1058             GdkGC 
*gc 
= gdk_gc_new( new_mask 
); 
1060             gdk_gc_set_foreground( gc
, &col 
); 
1061             gdk_draw_rectangle( new_mask
, gc
, TRUE
, 0, 0, ww
, hh 
); 
1063             gdk_gc_set_background( gc
, &col 
); 
1065             gdk_gc_set_foreground( gc
, &col 
); 
1066             gdk_gc_set_clip_region( gc
, m_currentClippingRegion
.GetRegion() ); 
1067             gdk_gc_set_clip_origin( gc
, -xx
, -yy 
); 
1068             gdk_gc_set_fill( gc
, GDK_OPAQUE_STIPPLED 
); 
1069             gdk_gc_set_stipple( gc
, mask 
); 
1070             gdk_draw_rectangle( new_mask
, gc
, TRUE
, 0, 0, ww
, hh 
); 
1077                 gdk_gc_set_clip_mask( m_textGC
, new_mask 
); 
1079                 gdk_gc_set_clip_mask( m_textGC
, mask 
); 
1080             gdk_gc_set_clip_origin( m_textGC
, xx
, yy 
); 
1085                 gdk_gc_set_clip_mask( m_penGC
, new_mask 
); 
1087                 gdk_gc_set_clip_mask( m_penGC
, mask 
); 
1088             gdk_gc_set_clip_origin( m_penGC
, xx
, yy 
); 
1092     // Draw XPixmap or XBitmap, depending on what the wxBitmap contains. For 
1093     // drawing a mono-bitmap (XBitmap) we use the current text GC 
1096         gdk_wx_draw_bitmap( m_window
, m_textGC
, use_bitmap
.GetBitmap(), 0, 0, xx
, yy
, -1, -1 ); 
1100         gdk_draw_pixmap(m_window
, m_penGC
, 
1101                         use_bitmap
.GetPixmap(), 
1102                         0, 0, xx
, yy
, -1, -1); 
1105     // remove mask again if any 
1106     if (useMask 
&& mask
) 
1110             gdk_gc_set_clip_mask( m_textGC
, (GdkBitmap 
*) NULL 
); 
1111             gdk_gc_set_clip_origin( m_textGC
, 0, 0 ); 
1112             if (!m_currentClippingRegion
.IsNull()) 
1113                 gdk_gc_set_clip_region( m_textGC
, m_currentClippingRegion
.GetRegion() ); 
1117             gdk_gc_set_clip_mask( m_penGC
, (GdkBitmap 
*) NULL 
); 
1118             gdk_gc_set_clip_origin( m_penGC
, 0, 0 ); 
1119             if (!m_currentClippingRegion
.IsNull()) 
1120                 gdk_gc_set_clip_region( m_penGC
, m_currentClippingRegion
.GetRegion() ); 
1125         gdk_bitmap_unref( new_mask 
); 
1128 bool wxWindowDCImpl::DoBlit( wxCoord xdest
, wxCoord ydest
, 
1129                              wxCoord width
, wxCoord height
, 
1131                              wxCoord xsrc
, wxCoord ysrc
, 
1134                              wxCoord xsrcMask
, wxCoord ysrcMask 
) 
1136     wxCHECK_MSG( IsOk(), false, wxT("invalid window dc") ); 
1138     wxCHECK_MSG( source
, false, wxT("invalid source dc") ); 
1140     if (!m_window
) return false; 
1142     // transform the source DC coords to the device ones 
1143     xsrc 
= source
->LogicalToDeviceX(xsrc
); 
1144     ysrc 
= source
->LogicalToDeviceY(ysrc
); 
1146     wxWindowDCImpl 
*srcDC 
= wxDynamicCast(source
->GetImpl(), wxWindowDCImpl
); 
1147     wxCHECK_MSG( srcDC
, false, "source must be a window DC" ); 
1149     // FIXME: this cast is not always valid, see the code using m_isMemDC 
1150     wxMemoryDCImpl 
*memDC 
= wx_static_cast(wxMemoryDCImpl 
*, srcDC
); 
1152     bool use_bitmap_method 
= false; 
1153     bool is_mono 
= false; 
1155     if (xsrcMask 
== -1 && ysrcMask 
== -1) 
1161     if (srcDC
->m_isMemDC
) 
1163         if (!memDC
->m_selected
.Ok()) return false; 
1165         is_mono 
= (memDC
->m_selected
.GetDepth() == 1); 
1167         // we use the "XCopyArea" way to copy a memory dc into 
1168         // a different window if the memory dc BOTH 
1169         // a) doesn't have any mask or its mask isn't used 
1173         if (useMask 
&& (memDC
->m_selected
.GetMask())) 
1175             // we HAVE TO use the direct way for memory dcs 
1176             // that have mask since the XCopyArea doesn't know 
1178             use_bitmap_method 
= true; 
1182             // we HAVE TO use the direct way for memory dcs 
1183             // that are bitmaps because XCopyArea doesn't cope 
1184             // with different bit depths 
1185             use_bitmap_method 
= true; 
1187         else if ((xsrc 
== 0) && (ysrc 
== 0) && 
1188                  (width 
== memDC
->m_selected
.GetWidth()) && 
1189                  (height 
== memDC
->m_selected
.GetHeight())) 
1191             // we SHOULD use the direct way if all of the bitmap 
1192             // in the memory dc is copied in which case XCopyArea 
1193             // wouldn't be able able to boost performace by reducing 
1194             // the area to be scaled 
1195             use_bitmap_method 
= true; 
1199             use_bitmap_method 
= false; 
1203     CalcBoundingBox( xdest
, ydest 
); 
1204     CalcBoundingBox( xdest 
+ width
, ydest 
+ height 
); 
1206     // scale/translate size and position 
1207     wxCoord xx 
= XLOG2DEV(xdest
); 
1208     wxCoord yy 
= YLOG2DEV(ydest
); 
1210     wxCoord ww 
= XLOG2DEVREL(width
); 
1211     wxCoord hh 
= YLOG2DEVREL(height
); 
1213     // compare to current clipping region 
1214     if (!m_currentClippingRegion
.IsNull()) 
1216         wxRegion 
tmp( xx
,yy
,ww
,hh 
); 
1217         tmp
.Intersect( m_currentClippingRegion 
); 
1222     int old_logical_func 
= m_logicalFunction
; 
1223     SetLogicalFunction( logical_func 
); 
1225     if (use_bitmap_method
) 
1227         // scale/translate bitmap size 
1228         wxCoord bm_width 
= memDC
->m_selected
.GetWidth(); 
1229         wxCoord bm_height 
= memDC
->m_selected
.GetHeight(); 
1231         // Get clip coords for the bitmap. If we don't 
1232         // use wxBitmap::Rescale(), which can clip the 
1233         // bitmap, these are the same as the original 
1240         // interpret userscale of src too 
1242         memDC
->GetUserScale(&xsc
,&ysc
); 
1243         bm_width 
= (int) (bm_width 
/ xsc
); 
1244         bm_height 
= (int) (bm_height 
/ ysc
); 
1246         wxCoord bm_ww 
= XLOG2DEVREL( bm_width 
); 
1247         wxCoord bm_hh 
= YLOG2DEVREL( bm_height 
); 
1249         // Scale bitmap if required 
1250         wxBitmap use_bitmap
; 
1251         if ((memDC
->m_selected
.GetWidth()!= bm_ww
) || ( memDC
->m_selected
.GetHeight()!= bm_hh
)) 
1253             // This indicates that the blitting code below will get 
1254             // a clipped bitmap and therefore needs to move the origin 
1256             wxRegion 
tmp( xx
,yy
,ww
,hh 
); 
1257             tmp
.Intersect( m_currentClippingRegion 
); 
1258             tmp
.GetBox(cx
,cy
,cw
,ch
); 
1260             // Scale and clipped bitmap 
1261             use_bitmap 
= memDC
->m_selected
.Rescale(cx
-xx
,cy
-yy
,cw
,ch
,bm_ww
,bm_hh
); 
1265             // Don't scale bitmap 
1266             use_bitmap 
= memDC
->m_selected
; 
1269         // apply mask if any 
1270         GdkBitmap 
*mask 
= (GdkBitmap 
*) NULL
; 
1271         if (use_bitmap
.GetMask()) mask 
= use_bitmap
.GetMask()->GetBitmap(); 
1273         GdkBitmap 
*new_mask 
= (GdkBitmap
*) NULL
; 
1275         if (useMask 
&& mask
) 
1277             if (!m_currentClippingRegion
.IsNull()) 
1280                 new_mask 
= gdk_pixmap_new( wxGetRootWindow()->window
, bm_ww
, bm_hh
, 1 ); 
1281                 GdkGC 
*gc 
= gdk_gc_new( new_mask 
); 
1283                 gdk_gc_set_foreground( gc
, &col 
); 
1284                 gdk_gc_set_ts_origin( gc
, -xsrcMask
, -ysrcMask
); 
1285                 gdk_draw_rectangle( new_mask
, gc
, TRUE
, 0, 0, bm_ww
, bm_hh 
); 
1287                 gdk_gc_set_background( gc
, &col 
); 
1289                 gdk_gc_set_foreground( gc
, &col 
); 
1290                 gdk_gc_set_clip_region( gc
, m_currentClippingRegion
.GetRegion() ); 
1291                 // was: gdk_gc_set_clip_origin( gc, -xx, -yy ); 
1292                 gdk_gc_set_clip_origin( gc
, -cx
, -cy 
); 
1293                 gdk_gc_set_fill( gc
, GDK_OPAQUE_STIPPLED 
); 
1294                 gdk_gc_set_stipple( gc
, mask 
); 
1295                 gdk_draw_rectangle( new_mask
, gc
, TRUE
, 0, 0, bm_ww
, bm_hh 
); 
1303                     gdk_gc_set_clip_mask( m_textGC
, new_mask 
); 
1304                     gdk_gc_set_clip_origin( m_textGC
, cx
, cy 
); 
1308                     gdk_gc_set_clip_mask( m_textGC
, mask 
); 
1309                     gdk_gc_set_clip_origin( m_textGC
, cx
-xsrcMask
, cy
-ysrcMask 
); 
1316                     gdk_gc_set_clip_mask( m_penGC
, new_mask 
); 
1317                     gdk_gc_set_clip_origin( m_penGC
, cx
, cy 
); 
1321                     gdk_gc_set_clip_mask( m_penGC
, mask 
); 
1322                     gdk_gc_set_clip_origin( m_penGC
, cx
-xsrcMask
, cy
-ysrcMask 
); 
1327         // Draw XPixmap or XBitmap, depending on what the wxBitmap contains. For 
1328         // drawing a mono-bitmap (XBitmap) we use the current text GC 
1332             // was: gdk_wx_draw_bitmap( m_window, m_textGC, use_bitmap.GetBitmap(), xsrc, ysrc, xx, yy, ww, hh ); 
1333             gdk_wx_draw_bitmap( m_window
, m_textGC
, use_bitmap
.GetBitmap(), xsrc
, ysrc
, cx
, cy
, cw
, ch 
); 
1337             // was: gdk_draw_pixmap( m_window, m_penGC, use_bitmap.GetPixmap(), xsrc, ysrc, xx, yy, ww, hh ); 
1338             gdk_draw_pixmap( m_window
, m_penGC
, use_bitmap
.GetPixmap(), xsrc
, ysrc
, cx
, cy
, cw
, ch 
); 
1341         // remove mask again if any 
1342         if (useMask 
&& mask
) 
1346                 gdk_gc_set_clip_mask( m_textGC
, (GdkBitmap 
*) NULL 
); 
1347                 gdk_gc_set_clip_origin( m_textGC
, 0, 0 ); 
1348                 if (!m_currentClippingRegion
.IsNull()) 
1349                     gdk_gc_set_clip_region( m_textGC
, m_currentClippingRegion
.GetRegion() ); 
1353                 gdk_gc_set_clip_mask( m_penGC
, (GdkBitmap 
*) NULL 
); 
1354                 gdk_gc_set_clip_origin( m_penGC
, 0, 0 ); 
1355                 if (!m_currentClippingRegion
.IsNull()) 
1356                     gdk_gc_set_clip_region( m_penGC
, m_currentClippingRegion
.GetRegion() ); 
1361             gdk_bitmap_unref( new_mask 
); 
1363     else // use_bitmap_method 
1365         if ((width 
!= ww
) || (height 
!= hh
)) 
1368             wxRegion 
tmp( xx
,yy
,ww
,hh 
); 
1369             tmp
.Intersect( m_currentClippingRegion 
); 
1370             wxCoord cx
,cy
,cw
,ch
; 
1371             tmp
.GetBox(cx
,cy
,cw
,ch
); 
1374             wxBitmap bitmap 
= memDC
->m_selected
.Rescale( cx
-xx
, cy
-yy
, cw
, ch
, ww
, hh 
); 
1376             // draw scaled bitmap 
1377             // was: gdk_draw_pixmap( m_window, m_penGC, bitmap.GetPixmap(), 0, 0, xx, yy, -1, -1 ); 
1378             gdk_draw_pixmap( m_window
, m_penGC
, bitmap
.GetPixmap(), 0, 0, cx
, cy
, -1, -1 ); 
1382             // No scaling and not a memory dc with a mask either 
1384             // copy including child window contents 
1385             gdk_gc_set_subwindow( m_penGC
, GDK_INCLUDE_INFERIORS 
); 
1386             gdk_window_copy_area( m_window
, m_penGC
, xx
, yy
, 
1388                                   xsrc
, ysrc
, width
, height 
); 
1389             gdk_gc_set_subwindow( m_penGC
, GDK_CLIP_BY_CHILDREN 
); 
1393     SetLogicalFunction( old_logical_func 
); 
1398 void wxWindowDCImpl::DoDrawText( const wxString 
&text
, wxCoord x
, wxCoord y 
) 
1400     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1402     if (!m_window
) return; 
1404     if (text
.empty()) return; 
1406     GdkFont 
*font 
= m_font
.GetInternalFont( m_scaleY 
); 
1408     wxCHECK_RET( font
, wxT("invalid font") ); 
1413     wxCoord width 
= gdk_string_width( font
, text
.mbc_str() ); 
1414     wxCoord height 
= font
->ascent 
+ font
->descent
; 
1416     if ( m_backgroundMode 
== wxSOLID 
) 
1418         gdk_gc_set_foreground( m_textGC
, m_textBackgroundColour
.GetColor() ); 
1419         gdk_draw_rectangle( m_window
, m_textGC
, TRUE
, x
, y
, width
, height 
); 
1420         gdk_gc_set_foreground( m_textGC
, m_textForegroundColour
.GetColor() ); 
1422     gdk_draw_string( m_window
, font
, m_textGC
, x
, y 
+ font
->ascent
, text
.mbc_str() ); 
1424     /* CMB 17/7/98: simple underline: ignores scaling and underlying 
1425        X font's XA_UNDERLINE_POSITION and XA_UNDERLINE_THICKNESS 
1426        properties (see wxXt implementation) */ 
1427     if (m_font
.GetUnderlined()) 
1429         wxCoord ul_y 
= y 
+ font
->ascent
; 
1430         if (font
->descent 
> 0) ul_y
++; 
1431         gdk_draw_line( m_window
, m_textGC
, x
, ul_y
, x 
+ width
, ul_y
); 
1434     width 
= wxCoord(width 
/ m_scaleX
); 
1435     height 
= wxCoord(height 
/ m_scaleY
); 
1436     CalcBoundingBox (x 
+ width
, y 
+ height
); 
1437     CalcBoundingBox (x
, y
); 
1441 // TODO: There is an example of rotating text with GTK2 that would probably be 
1442 // a better approach here: 
1443 //           http://www.daa.com.au/pipermail/pygtk/2003-April/005052.html 
1445 void wxWindowDCImpl::DoDrawRotatedText( const wxString 
&text
, wxCoord x
, wxCoord y
, double angle 
) 
1447     if ( wxIsNullDouble(angle
) ) 
1449         DoDrawText(text
, x
, y
); 
1453     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1455     if (!m_window
) return; 
1460     GdkFont 
*font 
= m_font
.GetInternalFont( m_scaleY 
); 
1462     wxCHECK_RET( font
, wxT("invalid font") ); 
1464     // the size of the text 
1465     w 
= gdk_string_width( font
, text
.mbc_str() ); 
1466     h 
= font
->ascent 
+ font
->descent
; 
1468     // draw the string normally 
1471     dc
.SelectObject(src
); 
1472     dc
.SetFont(GetFont()); 
1473     dc
.SetBackground(*wxBLACK_BRUSH
); 
1474     dc
.SetBrush(*wxBLACK_BRUSH
); 
1476     dc
.SetTextForeground( *wxWHITE 
); 
1477     dc
.DrawText(text
, 0, 0); 
1478     dc
.SelectObject(wxNullBitmap
); 
1480     // Calculate the size of the rotated bounding box. 
1481     double rad 
= DegToRad(angle
); 
1482     double dx 
= cos(rad
), 
1485     // the rectngle vertices are counted clockwise with the first one being at 
1486     // (0, 0) (or, rather, at (x, y)) 
1488            y2 
= -w
*dy
;      // y axis points to the bottom, hence minus 
1491     double x3 
= x4 
+ x2
, 
1495     wxCoord maxX 
= (wxCoord
)(dmax(x2
, dmax(x3
, x4
)) + 0.5), 
1496             maxY 
= (wxCoord
)(dmax(y2
, dmax(y3
, y4
)) + 0.5), 
1497             minX 
= (wxCoord
)(dmin(x2
, dmin(x3
, x4
)) - 0.5), 
1498             minY 
= (wxCoord
)(dmin(y2
, dmin(y3
, y4
)) - 0.5); 
1501     wxImage image 
= src
.ConvertToImage(); 
1503     image
.ConvertColourToAlpha( m_textForegroundColour
.Red(), 
1504                                 m_textForegroundColour
.Green(), 
1505                                 m_textForegroundColour
.Blue() ); 
1506     image 
= image
.Rotate( rad
, wxPoint(0,0) ); 
1508     int i_angle 
= (int) angle
; 
1509     i_angle 
= i_angle 
% 360; 
1513     if ((i_angle 
>= 90.0) && (i_angle 
< 270.0)) 
1514         xoffset 
= image
.GetWidth(); 
1516     if ((i_angle 
>= 0.0) && (i_angle 
< 180.0)) 
1517         yoffset 
= image
.GetHeight(); 
1519     if ((i_angle 
>= 0) && (i_angle 
< 90)) 
1520         yoffset 
-= (int)( cos(rad
)*h 
); 
1521     if ((i_angle 
>= 90) && (i_angle 
< 180)) 
1522         xoffset 
-= (int)( sin(rad
)*h 
); 
1523     if ((i_angle 
>= 180) && (i_angle 
< 270)) 
1524         yoffset 
-= (int)( cos(rad
)*h 
); 
1525     if ((i_angle 
>= 270) && (i_angle 
< 360)) 
1526         xoffset 
-= (int)( sin(rad
)*h 
); 
1528     int i_x 
= x 
- xoffset
; 
1529     int i_y 
= y 
- yoffset
; 
1532     DoDrawBitmap( src
, i_x
, i_y
, true ); 
1535     // it would be better to draw with non underlined font and draw the line 
1536     // manually here (it would be more straight...) 
1538     if ( m_font
.GetUnderlined() ) 
1540         gdk_draw_line( m_window
, m_textGC
, 
1541                        XLOG2DEV(x 
+ x4
), YLOG2DEV(y 
+ y4 
+ font
->descent
), 
1542                        XLOG2DEV(x 
+ x3
), YLOG2DEV(y 
+ y3 
+ font
->descent
)); 
1546     // update the bounding box 
1547     CalcBoundingBox(x 
+ minX
, y 
+ minY
); 
1548     CalcBoundingBox(x 
+ maxX
, y 
+ maxY
); 
1551 void wxWindowDCImpl::DoGetTextExtent(const wxString 
&string
, 
1552                                  wxCoord 
*width
, wxCoord 
*height
, 
1553                                  wxCoord 
*descent
, wxCoord 
*externalLeading
, 
1554                                  const wxFont 
*theFont
) const 
1562     if ( externalLeading 
) 
1563         *externalLeading 
= 0; 
1570     wxFont fontToUse 
= m_font
; 
1572         fontToUse 
= *theFont
; 
1574     GdkFont 
*font 
= fontToUse
.GetInternalFont( m_scaleY 
); 
1579         *width 
= wxCoord(gdk_string_width( font
, string
.mbc_str() ) / m_scaleX
); 
1581         *height 
= wxCoord((font
->ascent 
+ font
->descent
) / m_scaleY
); 
1583         *descent 
= wxCoord(font
->descent 
/ m_scaleY
); 
1586 wxCoord 
wxWindowDCImpl::GetCharWidth() const 
1588     GdkFont 
*font 
= m_font
.GetInternalFont( m_scaleY 
); 
1589     wxCHECK_MSG( font
, -1, wxT("invalid font") ); 
1591     return wxCoord(gdk_string_width( font
, "H" ) / m_scaleX
); 
1594 wxCoord 
wxWindowDCImpl::GetCharHeight() const 
1596     GdkFont 
*font 
= m_font
.GetInternalFont( m_scaleY 
); 
1597     wxCHECK_MSG( font
, -1, wxT("invalid font") ); 
1599     return wxCoord((font
->ascent 
+ font
->descent
) / m_scaleY
); 
1602 void wxWindowDCImpl::Clear() 
1604     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1606     if (!m_window
) return; 
1608     // VZ: the code below results in infinite recursion and crashes when 
1609     //     dc.Clear() is done from OnPaint() so I disable it for now. 
1610     //     I don't know what the correct fix is but Clear() surely should not 
1611     //     reenter OnPaint()! 
1613     /* - we either are a memory dc or have a window as the 
1614        owner. anything else shouldn't happen. 
1615        - we don't use gdk_window_clear() as we don't set 
1616        the window's background colour anymore. it is too 
1617        much pain to keep the DC's and the window's back- 
1618        ground colour in synch. */ 
1629         GetSize( &width
, &height 
); 
1630         gdk_draw_rectangle( m_window
, m_bgGC
, TRUE
, 0, 0, width
, height 
); 
1635     GetSize( &width
, &height 
); 
1636     gdk_draw_rectangle( m_window
, m_bgGC
, TRUE
, 0, 0, width
, height 
); 
1640 void wxWindowDCImpl::SetFont( const wxFont 
&font 
) 
1645 void wxWindowDCImpl::SetPen( const wxPen 
&pen 
) 
1647     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1649     if (m_pen 
== pen
) return; 
1653     if (!m_pen
.Ok()) return; 
1655     if (!m_window
) return; 
1657     gint width 
= m_pen
.GetWidth(); 
1660         // CMB: if width is non-zero scale it with the dc 
1665         // X doesn't allow different width in x and y and so we take 
1668                    ( fabs((double) XLOG2DEVREL(width
)) + 
1669                      fabs((double) YLOG2DEVREL(width
)) ) / 2.0; 
1673             // width can't be 0 or an internal GTK error occurs inside 
1674             // gdk_gc_set_dashes() below 
1679     static const wxGTKDash dotted
[] = {1, 1}; 
1680     static const wxGTKDash short_dashed
[] = {2, 2}; 
1681     static const wxGTKDash wxCoord_dashed
[] = {2, 4}; 
1682     static const wxGTKDash dotted_dashed
[] = {3, 3, 1, 3}; 
1684     // We express dash pattern in pen width unit, so we are 
1685     // independent of zoom factor and so on... 
1687     const wxGTKDash 
*req_dash
; 
1689     GdkLineStyle lineStyle 
= GDK_LINE_SOLID
; 
1690     switch (m_pen
.GetStyle()) 
1694             lineStyle 
= GDK_LINE_ON_OFF_DASH
; 
1695             req_nb_dash 
= m_pen
.GetDashCount(); 
1696             req_dash 
= (wxGTKDash
*)m_pen
.GetDash(); 
1701             lineStyle 
= GDK_LINE_ON_OFF_DASH
; 
1708             lineStyle 
= GDK_LINE_ON_OFF_DASH
; 
1710             req_dash 
= wxCoord_dashed
; 
1715             lineStyle 
= GDK_LINE_ON_OFF_DASH
; 
1717             req_dash 
= short_dashed
; 
1722 //            lineStyle = GDK_LINE_DOUBLE_DASH; 
1723             lineStyle 
= GDK_LINE_ON_OFF_DASH
; 
1725             req_dash 
= dotted_dashed
; 
1730         case wxSTIPPLE_MASK_OPAQUE
: 
1735             lineStyle 
= GDK_LINE_SOLID
; 
1736             req_dash 
= (wxGTKDash
*)NULL
; 
1742     if (req_dash 
&& req_nb_dash
) 
1744         wxGTKDash 
*real_req_dash 
= new wxGTKDash
[req_nb_dash
]; 
1747             for (int i 
= 0; i 
< req_nb_dash
; i
++) 
1748                 real_req_dash
[i
] = req_dash
[i
] * width
; 
1749             gdk_gc_set_dashes( m_penGC
, 0, real_req_dash
, req_nb_dash 
); 
1750             delete[] real_req_dash
; 
1754             // No Memory. We use non-scaled dash pattern... 
1755             gdk_gc_set_dashes( m_penGC
, 0, (wxGTKDash
*)req_dash
, req_nb_dash 
); 
1759     GdkCapStyle capStyle 
= GDK_CAP_ROUND
; 
1760     switch (m_pen
.GetCap()) 
1762         case wxCAP_PROJECTING
: { capStyle 
= GDK_CAP_PROJECTING
; break; } 
1763         case wxCAP_BUTT
:       { capStyle 
= GDK_CAP_BUTT
;       break; } 
1770                 capStyle 
= GDK_CAP_NOT_LAST
; 
1774                 capStyle 
= GDK_CAP_ROUND
; 
1780     GdkJoinStyle joinStyle 
= GDK_JOIN_ROUND
; 
1781     switch (m_pen
.GetJoin()) 
1783         case wxJOIN_BEVEL
: { joinStyle 
= GDK_JOIN_BEVEL
; break; } 
1784         case wxJOIN_MITER
: { joinStyle 
= GDK_JOIN_MITER
; break; } 
1786         default:           { joinStyle 
= GDK_JOIN_ROUND
; break; } 
1789     gdk_gc_set_line_attributes( m_penGC
, width
, lineStyle
, capStyle
, joinStyle 
); 
1791     m_pen
.GetColour().CalcPixel( m_cmap 
); 
1792     gdk_gc_set_foreground( m_penGC
, m_pen
.GetColour().GetColor() ); 
1795 void wxWindowDCImpl::SetBrush( const wxBrush 
&brush 
) 
1797     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1799     if (m_brush 
== brush
) return; 
1803     if (!m_brush
.Ok()) return; 
1805     if (!m_window
) return; 
1807     m_brush
.GetColour().CalcPixel( m_cmap 
); 
1808     gdk_gc_set_foreground( m_brushGC
, m_brush
.GetColour().GetColor() ); 
1810     gdk_gc_set_fill( m_brushGC
, GDK_SOLID 
); 
1812     if ((m_brush
.GetStyle() == wxSTIPPLE
) && (m_brush
.GetStipple()->Ok())) 
1814         if (m_brush
.GetStipple()->GetPixmap()) 
1816             gdk_gc_set_fill( m_brushGC
, GDK_TILED 
); 
1817             gdk_gc_set_tile( m_brushGC
, m_brush
.GetStipple()->GetPixmap() ); 
1821             gdk_gc_set_fill( m_brushGC
, GDK_STIPPLED 
); 
1822             gdk_gc_set_stipple( m_brushGC
, m_brush
.GetStipple()->GetBitmap() ); 
1826     if ((m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
1828         gdk_gc_set_fill( m_textGC
, GDK_OPAQUE_STIPPLED
); 
1829         gdk_gc_set_stipple( m_textGC
, m_brush
.GetStipple()->GetMask()->GetBitmap() ); 
1832     if (m_brush
.IsHatch()) 
1834         gdk_gc_set_fill( m_brushGC
, GDK_STIPPLED 
); 
1835         int num 
= m_brush
.GetStyle() - wxBDIAGONAL_HATCH
; 
1836         gdk_gc_set_stipple( m_brushGC
, hatches
[num
] ); 
1840 void wxWindowDCImpl::SetBackground( const wxBrush 
&brush 
) 
1842    /* CMB 21/7/98: Added SetBackground. Sets background brush 
1843     * for Clear() and bg colour for shapes filled with cross-hatch brush */ 
1845     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1847     if (m_backgroundBrush 
== brush
) return; 
1849     m_backgroundBrush 
= brush
; 
1851     if (!m_backgroundBrush
.Ok()) return; 
1853     if (!m_window
) return; 
1855     m_backgroundBrush
.GetColour().CalcPixel( m_cmap 
); 
1856     gdk_gc_set_background( m_brushGC
, m_backgroundBrush
.GetColour().GetColor() ); 
1857     gdk_gc_set_background( m_penGC
, m_backgroundBrush
.GetColour().GetColor() ); 
1858     gdk_gc_set_background( m_bgGC
, m_backgroundBrush
.GetColour().GetColor() ); 
1859     gdk_gc_set_foreground( m_bgGC
, m_backgroundBrush
.GetColour().GetColor() ); 
1861     gdk_gc_set_fill( m_bgGC
, GDK_SOLID 
); 
1863     if ((m_backgroundBrush
.GetStyle() == wxSTIPPLE
) && (m_backgroundBrush
.GetStipple()->Ok())) 
1865         if (m_backgroundBrush
.GetStipple()->GetPixmap()) 
1867             gdk_gc_set_fill( m_bgGC
, GDK_TILED 
); 
1868             gdk_gc_set_tile( m_bgGC
, m_backgroundBrush
.GetStipple()->GetPixmap() ); 
1872             gdk_gc_set_fill( m_bgGC
, GDK_STIPPLED 
); 
1873             gdk_gc_set_stipple( m_bgGC
, m_backgroundBrush
.GetStipple()->GetBitmap() ); 
1877     if (m_backgroundBrush
.IsHatch()) 
1879         gdk_gc_set_fill( m_bgGC
, GDK_STIPPLED 
); 
1880         int num 
= m_backgroundBrush
.GetStyle() - wxBDIAGONAL_HATCH
; 
1881         gdk_gc_set_stipple( m_bgGC
, hatches
[num
] ); 
1885 void wxWindowDCImpl::SetLogicalFunction( int function 
) 
1887     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1889     if (m_logicalFunction 
== function
) 
1892     // VZ: shouldn't this be a CHECK? 
1899         case wxXOR
:          mode 
= GDK_XOR
;           break; 
1900         case wxINVERT
:       mode 
= GDK_INVERT
;        break; 
1901         case wxOR_REVERSE
:   mode 
= GDK_OR_REVERSE
;    break; 
1902         case wxAND_REVERSE
:  mode 
= GDK_AND_REVERSE
;   break; 
1903         case wxCLEAR
:        mode 
= GDK_CLEAR
;         break; 
1904         case wxSET
:          mode 
= GDK_SET
;           break; 
1905         case wxOR_INVERT
:    mode 
= GDK_OR_INVERT
;     break; 
1906         case wxAND
:          mode 
= GDK_AND
;           break; 
1907         case wxOR
:           mode 
= GDK_OR
;            break; 
1908         case wxEQUIV
:        mode 
= GDK_EQUIV
;         break; 
1909         case wxNAND
:         mode 
= GDK_NAND
;          break; 
1910         case wxAND_INVERT
:   mode 
= GDK_AND_INVERT
;    break; 
1911         case wxCOPY
:         mode 
= GDK_COPY
;          break; 
1912         case wxNO_OP
:        mode 
= GDK_NOOP
;          break; 
1913         case wxSRC_INVERT
:   mode 
= GDK_COPY_INVERT
;   break; 
1915         // unsupported by GTK 
1916         case wxNOR
:          mode 
= GDK_COPY
;          break; 
1918            wxFAIL_MSG( wxT("unsupported logical function") ); 
1922     m_logicalFunction 
= function
; 
1924     gdk_gc_set_function( m_penGC
, mode 
); 
1925     gdk_gc_set_function( m_brushGC
, mode 
); 
1927     // to stay compatible with wxMSW, we don't apply ROPs to the text 
1928     // operations (i.e. DrawText/DrawRotatedText). 
1929     // True, but mono-bitmaps use the m_textGC and they use ROPs as well. 
1930     gdk_gc_set_function( m_textGC
, mode 
); 
1933 void wxWindowDCImpl::SetTextForeground( const wxColour 
&col 
) 
1935     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1937     // don't set m_textForegroundColour to an invalid colour as we'd crash 
1938     // later then (we use m_textForegroundColour.GetColor() without checking 
1940     if ( !col
.Ok() || (m_textForegroundColour 
== col
) ) 
1943     m_textForegroundColour 
= col
; 
1947         m_textForegroundColour
.CalcPixel( m_cmap 
); 
1948         gdk_gc_set_foreground( m_textGC
, m_textForegroundColour
.GetColor() ); 
1952 void wxWindowDCImpl::SetTextBackground( const wxColour 
&col 
) 
1954     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1957     if ( !col
.Ok() || (m_textBackgroundColour 
== col
) ) 
1960     m_textBackgroundColour 
= col
; 
1964         m_textBackgroundColour
.CalcPixel( m_cmap 
); 
1965         gdk_gc_set_background( m_textGC
, m_textBackgroundColour
.GetColor() ); 
1969 void wxWindowDCImpl::SetBackgroundMode( int mode 
) 
1971     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1973     m_backgroundMode 
= mode
; 
1975     if (!m_window
) return; 
1977     // CMB 21/7/98: fill style of cross-hatch brushes is affected by 
1978     // transparent/solid background mode 
1980     if (m_brush
.GetStyle() != wxSOLID 
&& m_brush
.GetStyle() != wxTRANSPARENT
) 
1982         gdk_gc_set_fill( m_brushGC
, 
1983           (m_backgroundMode 
== wxTRANSPARENT
) ? GDK_STIPPLED 
: GDK_OPAQUE_STIPPLED
); 
1987 void wxWindowDCImpl::SetPalette( const wxPalette
& WXUNUSED(palette
) ) 
1989     wxFAIL_MSG( wxT("wxWindowDCImpl::SetPalette not implemented") ); 
1992 void wxWindowDCImpl::DoSetClippingRegion( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height 
) 
1994     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1996     if (!m_window
) return; 
1999     rect
.x 
= XLOG2DEV(x
); 
2000     rect
.y 
= YLOG2DEV(y
); 
2001     rect
.width 
= XLOG2DEVREL(width
); 
2002     rect
.height 
= YLOG2DEVREL(height
); 
2004     if (!m_currentClippingRegion
.IsNull()) 
2005         m_currentClippingRegion
.Intersect( rect 
); 
2007         m_currentClippingRegion
.Union( rect 
); 
2009 #if USE_PAINT_REGION 
2010     if (!m_paintClippingRegion
.IsNull()) 
2011         m_currentClippingRegion
.Intersect( m_paintClippingRegion 
); 
2014     wxCoord xx
, yy
, ww
, hh
; 
2015     m_currentClippingRegion
.GetBox( xx
, yy
, ww
, hh 
); 
2016     wxGTKDCImpl::DoSetClippingRegion( xx
, yy
, ww
, hh 
); 
2018     gdk_gc_set_clip_region( m_penGC
, m_currentClippingRegion
.GetRegion() ); 
2019     gdk_gc_set_clip_region( m_brushGC
, m_currentClippingRegion
.GetRegion() ); 
2020     gdk_gc_set_clip_region( m_textGC
, m_currentClippingRegion
.GetRegion() ); 
2021     gdk_gc_set_clip_region( m_bgGC
, m_currentClippingRegion
.GetRegion() ); 
2024 void wxWindowDCImpl::DoSetDeviceClippingRegion( const wxRegion 
®ion  
) 
2026     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2030         DestroyClippingRegion(); 
2034     if (!m_window
) return; 
2036     if (!m_currentClippingRegion
.IsNull()) 
2037         m_currentClippingRegion
.Intersect( region 
); 
2039         m_currentClippingRegion
.Union( region 
); 
2041 #if USE_PAINT_REGION 
2042     if (!m_paintClippingRegion
.IsNull()) 
2043         m_currentClippingRegion
.Intersect( m_paintClippingRegion 
); 
2046     wxCoord xx
, yy
, ww
, hh
; 
2047     m_currentClippingRegion
.GetBox( xx
, yy
, ww
, hh 
); 
2048     wxGTKDCImpl::DoSetClippingRegion( xx
, yy
, ww
, hh 
); 
2050     gdk_gc_set_clip_region( m_penGC
, m_currentClippingRegion
.GetRegion() ); 
2051     gdk_gc_set_clip_region( m_brushGC
, m_currentClippingRegion
.GetRegion() ); 
2052     gdk_gc_set_clip_region( m_textGC
, m_currentClippingRegion
.GetRegion() ); 
2053     gdk_gc_set_clip_region( m_bgGC
, m_currentClippingRegion
.GetRegion() ); 
2056 void wxWindowDCImpl::DestroyClippingRegion() 
2058     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2060     wxGTKDCImpl::DestroyClippingRegion(); 
2062     m_currentClippingRegion
.Clear(); 
2064 #if USE_PAINT_REGION 
2065     if (!m_paintClippingRegion
.IsEmpty()) 
2066         m_currentClippingRegion
.Union( m_paintClippingRegion 
); 
2069     if (!m_window
) return; 
2071     if (m_currentClippingRegion
.IsEmpty()) 
2073         gdk_gc_set_clip_rectangle( m_penGC
, (GdkRectangle 
*) NULL 
); 
2074         gdk_gc_set_clip_rectangle( m_brushGC
, (GdkRectangle 
*) NULL 
); 
2075         gdk_gc_set_clip_rectangle( m_textGC
, (GdkRectangle 
*) NULL 
); 
2076         gdk_gc_set_clip_rectangle( m_bgGC
, (GdkRectangle 
*) NULL 
); 
2080         gdk_gc_set_clip_region( m_penGC
, m_currentClippingRegion
.GetRegion() ); 
2081         gdk_gc_set_clip_region( m_brushGC
, m_currentClippingRegion
.GetRegion() ); 
2082         gdk_gc_set_clip_region( m_textGC
, m_currentClippingRegion
.GetRegion() ); 
2083         gdk_gc_set_clip_region( m_bgGC
, m_currentClippingRegion
.GetRegion() ); 
2087 void wxWindowDCImpl::Destroy() 
2089     if (m_penGC
) wxFreePoolGC( m_penGC 
); 
2090     m_penGC 
= (GdkGC
*) NULL
; 
2091     if (m_brushGC
) wxFreePoolGC( m_brushGC 
); 
2092     m_brushGC 
= (GdkGC
*) NULL
; 
2093     if (m_textGC
) wxFreePoolGC( m_textGC 
); 
2094     m_textGC 
= (GdkGC
*) NULL
; 
2095     if (m_bgGC
) wxFreePoolGC( m_bgGC 
); 
2096     m_bgGC 
= (GdkGC
*) NULL
; 
2099 void wxWindowDCImpl::ComputeScaleAndOrigin() 
2101     const wxRealPoint 
origScale(m_scaleX
, m_scaleY
); 
2103     wxGTKDCImpl::ComputeScaleAndOrigin(); 
2105     // if scale has changed call SetPen to recalulate the line width 
2106     if ( wxRealPoint(m_scaleX
, m_scaleY
) != origScale 
&& m_pen
.Ok() ) 
2108         // this is a bit artificial, but we need to force wxDC to think the pen 
2116 // Resolution in pixels per logical inch 
2117 wxSize 
wxWindowDCImpl::GetPPI() const 
2119     return wxSize( (int) (m_mm_to_pix_x 
* 25.4 + 0.5), (int) (m_mm_to_pix_y 
* 25.4 + 0.5)); 
2122 int wxWindowDCImpl::GetDepth() const 
2124     wxFAIL_MSG(wxT("not implemented")); 
2130 //----------------------------------------------------------------------------- 
2132 //----------------------------------------------------------------------------- 
2134 IMPLEMENT_ABSTRACT_CLASS(wxPaintDCImpl
, wxClientDCImpl
) 
2136 // Limit the paint region to the window size. Sometimes 
2137 // the paint region is too big, and this risks X11 errors 
2138 static void wxLimitRegionToSize(wxRegion
& region
, const wxSize
& sz
) 
2140     wxRect originalRect 
= region
.GetBox(); 
2141     wxRect 
rect(originalRect
); 
2142     if (rect
.width 
+ rect
.x 
> sz
.x
) 
2143         rect
.width 
= sz
.x 
- rect
.x
; 
2144     if (rect
.height 
+ rect
.y 
> sz
.y
) 
2145         rect
.height 
= sz
.y 
- rect
.y
; 
2146     if (rect 
!= originalRect
) 
2148         region 
= wxRegion(rect
); 
2149         wxLogTrace(wxT("painting"), wxT("Limiting region from %d, %d, %d, %d to %d, %d, %d, %d\n"), 
2150                    originalRect
.x
, originalRect
.y
, originalRect
.width
, originalRect
.height
, 
2151                    rect
.x
, rect
.y
, rect
.width
, rect
.height
); 
2155 wxPaintDCImpl::wxPaintDCImpl(wxDC 
*owner
, wxWindow 
*win
) 
2156              : wxClientDCImpl(owner
, win
) 
2158 #if USE_PAINT_REGION 
2159     if (!win
->m_clipPaintRegion
) 
2162     wxSize sz 
= win
->GetSize(); 
2163     m_paintClippingRegion 
= win
->GetUpdateRegion(); 
2164     wxLimitRegionToSize(m_paintClippingRegion
, sz
); 
2166     GdkRegion 
*region 
= m_paintClippingRegion
.GetRegion(); 
2169         m_currentClippingRegion
.Union( m_paintClippingRegion 
); 
2170         wxLimitRegionToSize(m_currentClippingRegion
, sz
); 
2172         if (sz
.x 
<= 0 || sz
.y 
<= 0) 
2175         gdk_gc_set_clip_region( m_penGC
, region 
); 
2176         gdk_gc_set_clip_region( m_brushGC
, region 
); 
2177         gdk_gc_set_clip_region( m_textGC
, region 
); 
2178         gdk_gc_set_clip_region( m_bgGC
, region 
); 
2180 #endif // USE_PAINT_REGION 
2183 //----------------------------------------------------------------------------- 
2185 //----------------------------------------------------------------------------- 
2187 IMPLEMENT_ABSTRACT_CLASS(wxClientDCImpl
, wxWindowDCImpl
) 
2189 wxClientDCImpl::wxClientDCImpl(wxDC 
*owner
, wxWindow 
*win
) 
2190               : wxWindowDCImpl(owner
, win
) 
2192     wxCHECK_RET( win
, _T("NULL window in wxClientDCImpl::wxClientDCImpl") ); 
2194 #ifdef __WXUNIVERSAL__ 
2195     wxPoint ptOrigin 
= win
->GetClientAreaOrigin(); 
2196     SetDeviceOrigin(ptOrigin
.x
, ptOrigin
.y
); 
2197     wxSize size 
= win
->GetClientSize(); 
2198     SetClippingRegion(wxPoint(0, 0), size
); 
2199 #endif // __WXUNIVERSAL__ 
2202 void wxClientDCImpl::DoGetSize(int *width
, int *height
) const 
2204     wxCHECK_RET( m_owner
, _T("GetSize() doesn't work without window") ); 
2206     m_owner
->GetClientSize( width
, height 
); 
2209 // ---------------------------------------------------------------------------- 
2211 // ---------------------------------------------------------------------------- 
2213 class wxDCModule 
: public wxModule
 
2220     DECLARE_DYNAMIC_CLASS(wxDCModule
) 
2223 IMPLEMENT_DYNAMIC_CLASS(wxDCModule
, wxModule
) 
2225 bool wxDCModule::OnInit() 
2231 void wxDCModule::OnExit()