1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/dcclient.cpp 
   3 // Purpose:     wxWindowDCImpl implementation 
   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" 
  13 #include "wx/gtk/dcclient.h" 
  16     #include "wx/window.h" 
  18     #include "wx/dcmemory.h" 
  21     #include "wx/module.h" 
  24 #include "wx/fontutil.h" 
  26 #include "wx/gtk/private.h" 
  27 #include "wx/gtk/private/object.h" 
  29 //----------------------------------------------------------------------------- 
  31 //----------------------------------------------------------------------------- 
  33 #define XLOG2DEV(x)    LogicalToDeviceX(x) 
  34 #define XLOG2DEVREL(x) LogicalToDeviceXRel(x) 
  35 #define YLOG2DEV(y)    LogicalToDeviceY(y) 
  36 #define YLOG2DEVREL(y) LogicalToDeviceYRel(y) 
  38 #define USE_PAINT_REGION 1 
  40 //----------------------------------------------------------------------------- 
  42 //----------------------------------------------------------------------------- 
  51 static GdkPixmap
* hatches
[wxBRUSHSTYLE_LAST_HATCH 
- wxBRUSHSTYLE_FIRST_HATCH 
+ 1]; 
  53 //----------------------------------------------------------------------------- 
  55 //----------------------------------------------------------------------------- 
  57 static const double RAD2DEG  
= 180.0 / M_PI
; 
  59 // ---------------------------------------------------------------------------- 
  61 // ---------------------------------------------------------------------------- 
  63 static inline double dmax(double a
, double b
) { return a 
> b 
? a 
: b
; } 
  64 static inline double dmin(double a
, double b
) { return a 
< b 
? a 
: b
; } 
  66 static inline double DegToRad(double deg
) { return (deg 
* M_PI
) / 180.0; } 
  68 static GdkPixmap
* GetHatch(int style
) 
  70     wxASSERT(style 
>= wxBRUSHSTYLE_FIRST_HATCH 
&& style 
<= wxBRUSHSTYLE_LAST_HATCH
); 
  71     const int i 
= style 
- wxBRUSHSTYLE_FIRST_HATCH
; 
  72     if (hatches
[i
] == NULL
) 
  76         case wxBRUSHSTYLE_BDIAGONAL_HATCH
: 
  77             hatches
[i
] = gdk_bitmap_create_from_data(NULL
, bdiag_bits
, bdiag_width
, bdiag_height
); 
  79         case wxBRUSHSTYLE_CROSSDIAG_HATCH
: 
  80             hatches
[i
] = gdk_bitmap_create_from_data(NULL
, cdiag_bits
, cdiag_width
, cdiag_height
); 
  82         case wxBRUSHSTYLE_CROSS_HATCH
: 
  83             hatches
[i
] = gdk_bitmap_create_from_data(NULL
, cross_bits
, cross_width
, cross_height
); 
  85         case wxBRUSHSTYLE_FDIAGONAL_HATCH
: 
  86             hatches
[i
] = gdk_bitmap_create_from_data(NULL
, fdiag_bits
, fdiag_width
, fdiag_height
); 
  88         case wxBRUSHSTYLE_HORIZONTAL_HATCH
: 
  89             hatches
[i
] = gdk_bitmap_create_from_data(NULL
, horiz_bits
, horiz_width
, horiz_height
); 
  91         case wxBRUSHSTYLE_VERTICAL_HATCH
: 
  92             hatches
[i
] = gdk_bitmap_create_from_data(NULL
, verti_bits
, verti_width
, verti_height
); 
  99 //----------------------------------------------------------------------------- 
 100 // Implement Pool of Graphic contexts. Creating them takes too much time. 
 101 //----------------------------------------------------------------------------- 
 127 #define GC_POOL_ALLOC_SIZE 100 
 129 static int wxGCPoolSize 
= 0; 
 131 static wxGC 
*wxGCPool 
= NULL
; 
 133 static void wxInitGCPool() 
 135     // This really could wait until the first call to 
 136     // wxGetPoolGC, but we will make the first allocation 
 137     // now when other initialization is being performed. 
 139     // Set initial pool size. 
 140     wxGCPoolSize 
= GC_POOL_ALLOC_SIZE
; 
 142     // Allocate initial pool. 
 143     wxGCPool 
= (wxGC 
*)malloc(wxGCPoolSize 
* sizeof(wxGC
)); 
 144     if (wxGCPool 
== NULL
) 
 146         // If we cannot malloc, then fail with error 
 147         // when debug is enabled.  If debug is not enabled, 
 148         // the problem will eventually get caught 
 150         wxFAIL_MSG( wxT("Cannot allocate GC pool") ); 
 154     // Zero initial pool. 
 155     memset(wxGCPool
, 0, wxGCPoolSize 
* sizeof(wxGC
)); 
 158 static void wxCleanUpGCPool() 
 160     for (int i 
= 0; i 
< wxGCPoolSize
; i
++) 
 162         if (wxGCPool
[i
].m_gc
) 
 163             g_object_unref (wxGCPool
[i
].m_gc
); 
 171 static GdkGC
* wxGetPoolGC( GdkWindow 
*window
, wxPoolGCType type 
) 
 175     // Look for an available GC. 
 176     for (int i 
= 0; i 
< wxGCPoolSize
; i
++) 
 178         if (!wxGCPool
[i
].m_gc
) 
 180             wxGCPool
[i
].m_gc 
= gdk_gc_new( window 
); 
 181             gdk_gc_set_exposures( wxGCPool
[i
].m_gc
, FALSE 
); 
 182             wxGCPool
[i
].m_type 
= type
; 
 183             wxGCPool
[i
].m_used 
= false; 
 185         if ((!wxGCPool
[i
].m_used
) && (wxGCPool
[i
].m_type 
== type
)) 
 187             wxGCPool
[i
].m_used 
= true; 
 188             return wxGCPool
[i
].m_gc
; 
 192     // We did not find an available GC. 
 193     // We need to grow the GC pool. 
 194     pptr 
= (wxGC 
*)realloc(wxGCPool
, 
 195         (wxGCPoolSize 
+ GC_POOL_ALLOC_SIZE
)*sizeof(wxGC
)); 
 198         // Initialize newly allocated pool. 
 200         memset(&wxGCPool
[wxGCPoolSize
], 0, 
 201             GC_POOL_ALLOC_SIZE
*sizeof(wxGC
)); 
 203         // Initialize entry we will return. 
 204         wxGCPool
[wxGCPoolSize
].m_gc 
= gdk_gc_new( window 
); 
 205         gdk_gc_set_exposures( wxGCPool
[wxGCPoolSize
].m_gc
, FALSE 
); 
 206         wxGCPool
[wxGCPoolSize
].m_type 
= type
; 
 207         wxGCPool
[wxGCPoolSize
].m_used 
= true; 
 209         // Set new value of pool size. 
 210         wxGCPoolSize 
+= GC_POOL_ALLOC_SIZE
; 
 212         // Return newly allocated entry. 
 213         return wxGCPool
[wxGCPoolSize
-GC_POOL_ALLOC_SIZE
].m_gc
; 
 216     // The realloc failed.  Fall through to error. 
 217     wxFAIL_MSG( wxT("No GC available") ); 
 222 static void wxFreePoolGC( GdkGC 
*gc 
) 
 224     for (int i 
= 0; i 
< wxGCPoolSize
; i
++) 
 226         if (wxGCPool
[i
].m_gc 
== gc
) 
 228             wxGCPool
[i
].m_used 
= false; 
 233     wxFAIL_MSG( wxT("Wrong GC") ); 
 236 //----------------------------------------------------------------------------- 
 238 //----------------------------------------------------------------------------- 
 240 IMPLEMENT_ABSTRACT_CLASS(wxWindowDCImpl
, wxGTKDCImpl
) 
 242 wxWindowDCImpl::wxWindowDCImpl( wxDC 
*owner 
) : 
 251     m_isScreenDC 
= false; 
 257 wxWindowDCImpl::wxWindowDCImpl( wxDC 
*owner
, wxWindow 
*window 
) : 
 260     wxASSERT_MSG( window
, wxT("DC needs a window") ); 
 268     m_isScreenDC 
= false; 
 269     m_font 
= window
->GetFont(); 
 271     GtkWidget 
*widget 
= window
->m_wxwindow
; 
 273     // Some controls don't have m_wxwindow - like wxStaticBox, but the user 
 274     // code should still be able to create wxClientDCs for them 
 277         widget 
= window
->m_widget
; 
 279         wxCHECK_RET(widget
, "DC needs a widget"); 
 281         if (GTK_WIDGET_NO_WINDOW(widget
)) 
 282             SetDeviceLocalOrigin(widget
->allocation
.x
, widget
->allocation
.y
); 
 285     m_context 
= window
->GTKGetPangoDefaultContext(); 
 286     m_layout 
= pango_layout_new( m_context 
); 
 287     m_fontdesc 
= pango_font_description_copy( widget
->style
->font_desc 
); 
 289     m_gdkwindow 
= widget
->window
; 
 291     // Window not realized ? 
 294          // Don't report problems as per MSW. 
 300     m_cmap 
= gtk_widget_get_colormap(widget
); 
 304     /* this must be done after SetUpDC, bacause SetUpDC calls the 
 305        repective SetBrush, SetPen, SetBackground etc functions 
 306        to set up the DC. SetBackground call m_owner->SetBackground 
 307        and this might not be desired as the standard dc background 
 308        is white whereas a window might assume gray to be the 
 309        standard (as e.g. wxStatusBar) */ 
 313     if (m_window 
&& m_window
->m_wxwindow 
&& 
 314         (m_window
->GetLayoutDirection() == wxLayout_RightToLeft
)) 
 319         // origin in the upper right corner 
 320         m_deviceOriginX 
= m_window
->GetClientSize().x
; 
 324 wxWindowDCImpl::~wxWindowDCImpl() 
 329         g_object_unref (m_layout
); 
 331         pango_font_description_free( m_fontdesc 
); 
 334 void wxWindowDCImpl::SetUpDC( bool isMemDC 
) 
 338     wxASSERT_MSG( !m_penGC
, wxT("GCs already created") ); 
 342     if ((isMemDC
) && (GetSelectedBitmap().IsOk())) 
 344         if (GetSelectedBitmap().GetDepth() == 1) 
 346             m_penGC 
= wxGetPoolGC( m_gdkwindow
, wxPEN_MONO 
); 
 347             m_brushGC 
= wxGetPoolGC( m_gdkwindow
, wxBRUSH_MONO 
); 
 348             m_textGC 
= wxGetPoolGC( m_gdkwindow
, wxTEXT_MONO 
); 
 349             m_bgGC 
= wxGetPoolGC( m_gdkwindow
, wxBG_MONO 
); 
 358             m_penGC 
= wxGetPoolGC( m_gdkwindow
, wxPEN_SCREEN 
); 
 359             m_brushGC 
= wxGetPoolGC( m_gdkwindow
, wxBRUSH_SCREEN 
); 
 360             m_textGC 
= wxGetPoolGC( m_gdkwindow
, wxTEXT_SCREEN 
); 
 361             m_bgGC 
= wxGetPoolGC( m_gdkwindow
, wxBG_SCREEN 
); 
 365             m_penGC 
= wxGetPoolGC( m_gdkwindow
, wxPEN_COLOUR 
); 
 366             m_brushGC 
= wxGetPoolGC( m_gdkwindow
, wxBRUSH_COLOUR 
); 
 367             m_textGC 
= wxGetPoolGC( m_gdkwindow
, wxTEXT_COLOUR 
); 
 368             m_bgGC 
= wxGetPoolGC( m_gdkwindow
, wxBG_COLOUR 
); 
 372     /* background colour */ 
 373     m_backgroundBrush 
= *wxWHITE_BRUSH
; 
 374     m_backgroundBrush
.GetColour().CalcPixel( m_cmap 
); 
 375     const GdkColor 
*bg_col 
= m_backgroundBrush
.GetColour().GetColor(); 
 378     m_textForegroundColour
.CalcPixel( m_cmap 
); 
 379     gdk_gc_set_foreground( m_textGC
, m_textForegroundColour
.GetColor() ); 
 381     m_textBackgroundColour
.CalcPixel( m_cmap 
); 
 382     gdk_gc_set_background( m_textGC
, m_textBackgroundColour
.GetColor() ); 
 384     gdk_gc_set_fill( m_textGC
, GDK_SOLID 
); 
 386     gdk_gc_set_colormap( m_textGC
, m_cmap 
); 
 389     m_pen
.GetColour().CalcPixel( m_cmap 
); 
 390     gdk_gc_set_foreground( m_penGC
, m_pen
.GetColour().GetColor() ); 
 391     gdk_gc_set_background( m_penGC
, bg_col 
); 
 393     gdk_gc_set_line_attributes( m_penGC
, 0, GDK_LINE_SOLID
, GDK_CAP_NOT_LAST
, GDK_JOIN_ROUND 
); 
 396     m_brush
.GetColour().CalcPixel( m_cmap 
); 
 397     gdk_gc_set_foreground( m_brushGC
, m_brush
.GetColour().GetColor() ); 
 398     gdk_gc_set_background( m_brushGC
, bg_col 
); 
 400     gdk_gc_set_fill( m_brushGC
, GDK_SOLID 
); 
 403     gdk_gc_set_background( m_bgGC
, bg_col 
); 
 404     gdk_gc_set_foreground( m_bgGC
, bg_col 
); 
 406     gdk_gc_set_fill( m_bgGC
, GDK_SOLID 
); 
 409     gdk_gc_set_function( m_textGC
, GDK_COPY 
); 
 410     gdk_gc_set_function( m_brushGC
, GDK_COPY 
); 
 411     gdk_gc_set_function( m_penGC
, GDK_COPY 
); 
 414     gdk_gc_set_clip_rectangle( m_penGC
, NULL 
); 
 415     gdk_gc_set_clip_rectangle( m_brushGC
, NULL 
); 
 416     gdk_gc_set_clip_rectangle( m_textGC
, NULL 
); 
 417     gdk_gc_set_clip_rectangle( m_bgGC
, NULL 
); 
 420 void wxWindowDCImpl::DoGetSize( int* width
, int* height 
) const 
 422     wxCHECK_RET( m_window
, wxT("GetSize() doesn't work without window") ); 
 424     m_window
->GetSize(width
, height
); 
 427 bool wxWindowDCImpl::DoFloodFill(wxCoord x
, wxCoord y
, 
 428                                  const wxColour
& col
, wxFloodFillStyle style
) 
 431     extern bool wxDoFloodFill(wxDC 
*dc
, wxCoord x
, wxCoord y
, 
 432                               const wxColour 
& col
, wxFloodFillStyle style
); 
 434     return wxDoFloodFill( GetOwner(), x
, y
, col
, style
); 
 445 bool wxWindowDCImpl::DoGetPixel( wxCoord x1
, wxCoord y1
, wxColour 
*col 
) const 
 447     GdkImage
* image 
= NULL
; 
 450         const int x 
= LogicalToDeviceX(x1
); 
 451         const int y 
= LogicalToDeviceY(y1
); 
 453         gdk_drawable_get_size(m_gdkwindow
, &rect
.width
, &rect
.height
); 
 454         if (rect
.Contains(x
, y
)) 
 455             image 
= gdk_drawable_get_image(m_gdkwindow
, x
, y
, 1, 1); 
 462     GdkColormap
* colormap 
= gdk_image_get_colormap(image
); 
 463     const unsigned pixel 
= gdk_image_get_pixel(image
, 0, 0); 
 464     if (colormap 
== NULL
) 
 465         *col 
= pixel 
? m_textForegroundColour 
: m_textBackgroundColour
; 
 469         gdk_colormap_query_color(colormap
, pixel
, &c
); 
 470         col
->Set(c
.red 
>> 8, c
.green 
>> 8, c
.blue 
>> 8); 
 472     g_object_unref(image
); 
 476 void wxWindowDCImpl::DoDrawLine( wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2 
) 
 478     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 480     if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 483             gdk_draw_line( m_gdkwindow
, m_penGC
, XLOG2DEV(x1
), YLOG2DEV(y1
), XLOG2DEV(x2
), YLOG2DEV(y2
) ); 
 485         CalcBoundingBox(x1
, y1
); 
 486         CalcBoundingBox(x2
, y2
); 
 490 void wxWindowDCImpl::DoCrossHair( wxCoord x
, wxCoord y 
) 
 492     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 494     if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 498         GetOwner()->GetSize( &w
, &h 
); 
 499         wxCoord xx 
= XLOG2DEV(x
); 
 500         wxCoord yy 
= YLOG2DEV(y
); 
 503             gdk_draw_line( m_gdkwindow
, m_penGC
, 0, yy
, XLOG2DEVREL(w
), yy 
); 
 504             gdk_draw_line( m_gdkwindow
, m_penGC
, xx
, 0, xx
, YLOG2DEVREL(h
) ); 
 509 void wxWindowDCImpl::DrawingSetup(GdkGC
*& gc
, bool& originChanged
) 
 512     GdkPixmap
* pixmap 
= NULL
; 
 513     const int style 
= m_brush
.GetStyle(); 
 515     if (style 
== wxBRUSHSTYLE_STIPPLE 
|| style 
== wxBRUSHSTYLE_STIPPLE_MASK_OPAQUE
) 
 517         const wxBitmap
* stipple 
= m_brush
.GetStipple(); 
 520             if (style 
== wxBRUSHSTYLE_STIPPLE
) 
 521                 pixmap 
= stipple
->GetPixmap(); 
 522             else if (stipple
->GetMask()) 
 524                 pixmap 
= stipple
->GetPixmap(); 
 529     else if (m_brush
.IsHatch()) 
 531         pixmap 
= GetHatch(style
); 
 539         gdk_drawable_get_size(pixmap
, &w
, &h
); 
 540         origin_x 
= m_deviceOriginX 
% w
; 
 541         origin_y 
= m_deviceOriginY 
% h
; 
 544     originChanged 
= origin_x 
|| origin_y
; 
 546         gdk_gc_set_ts_origin(gc
, origin_x
, origin_y
); 
 549 void wxWindowDCImpl::DoDrawArc( wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2
, 
 550                             wxCoord xc
, wxCoord yc 
) 
 552     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 554     wxCoord xx1 
= XLOG2DEV(x1
); 
 555     wxCoord yy1 
= YLOG2DEV(y1
); 
 556     wxCoord xx2 
= XLOG2DEV(x2
); 
 557     wxCoord yy2 
= YLOG2DEV(y2
); 
 558     wxCoord xxc 
= XLOG2DEV(xc
); 
 559     wxCoord yyc 
= YLOG2DEV(yc
); 
 560     double dx 
= xx1 
- xxc
; 
 561     double dy 
= yy1 
- yyc
; 
 562     double radius 
= sqrt((double)(dx
*dx
+dy
*dy
)); 
 563     wxCoord   r      
= (wxCoord
)radius
; 
 564     double radius1
, radius2
; 
 566     if (xx1 
== xx2 
&& yy1 
== yy2
) 
 571     else if ( wxIsNullDouble(radius
) ) 
 578         radius1 
= (xx1 
- xxc 
== 0) ? 
 579             (yy1 
- yyc 
< 0) ? 90.0 : -90.0 : 
 580             -atan2(double(yy1
-yyc
), double(xx1
-xxc
)) * RAD2DEG
; 
 581         radius2 
= (xx2 
- xxc 
== 0) ? 
 582             (yy2 
- yyc 
< 0) ? 90.0 : -90.0 : 
 583             -atan2(double(yy2
-yyc
), double(xx2
-xxc
)) * RAD2DEG
; 
 585     wxCoord alpha1 
= wxCoord(radius1 
* 64.0); 
 586     wxCoord alpha2 
= wxCoord((radius2 
- radius1
) * 64.0); 
 587     while (alpha2 
<= 0) alpha2 
+= 360*64; 
 588     while (alpha1 
> 360*64) alpha1 
-= 360*64; 
 592         if (m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) 
 596             DrawingSetup(gc
, originChanged
); 
 598             gdk_draw_arc(m_gdkwindow
, gc
, true, xxc
-r
, yyc
-r
, 2*r
, 2*r
, alpha1
, alpha2
); 
 601                 gdk_gc_set_ts_origin(gc
, 0, 0); 
 604         if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 606             gdk_draw_arc( m_gdkwindow
, m_penGC
, FALSE
, xxc
-r
, yyc
-r
, 2*r
,2*r
, alpha1
, alpha2 
); 
 608             if ((m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) && (alpha2 
- alpha1 
!= 360*64)) 
 610                 gdk_draw_line( m_gdkwindow
, m_penGC
, xx1
, yy1
, xxc
, yyc 
); 
 611                 gdk_draw_line( m_gdkwindow
, m_penGC
, xxc
, yyc
, xx2
, yy2 
); 
 616     CalcBoundingBox (x1
, y1
); 
 617     CalcBoundingBox (x2
, y2
); 
 620 void wxWindowDCImpl::DoDrawEllipticArc( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double sa
, double ea 
) 
 622     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 624     wxCoord xx 
= XLOG2DEV(x
); 
 625     wxCoord yy 
= YLOG2DEV(y
); 
 626     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 627     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 629     // CMB: handle -ve width and/or height 
 630     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 631     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 635         wxCoord start 
= wxCoord(sa 
* 64.0); 
 636         wxCoord end 
= wxCoord((ea
-sa
) * 64.0); 
 638         if (m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) 
 642             DrawingSetup(gc
, originChanged
); 
 644             gdk_draw_arc(m_gdkwindow
, gc
, true, xx
, yy
, ww
, hh
, start
, end
); 
 647                 gdk_gc_set_ts_origin(gc
, 0, 0); 
 650         if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 651             gdk_draw_arc( m_gdkwindow
, m_penGC
, FALSE
, xx
, yy
, ww
, hh
, start
, end 
); 
 654     CalcBoundingBox (x
, y
); 
 655     CalcBoundingBox (x 
+ width
, y 
+ height
); 
 658 void wxWindowDCImpl::DoDrawPoint( wxCoord x
, wxCoord y 
) 
 660     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 662     if ((m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) && m_gdkwindow
) 
 663         gdk_draw_point( m_gdkwindow
, m_penGC
, XLOG2DEV(x
), YLOG2DEV(y
) ); 
 665     CalcBoundingBox (x
, y
); 
 668 void wxWindowDCImpl::DoDrawLines( int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset 
) 
 670     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 672     if (m_pen
.GetStyle() == wxPENSTYLE_TRANSPARENT
) return; 
 675     //Check, if scaling is necessary 
 677         xoffset 
!= 0 || yoffset 
!= 0 || XLOG2DEV(10) != 10 || YLOG2DEV(10) != 10; 
 679     // GdkPoint and wxPoint have the same memory layout, so we can cast one to the other 
 680     GdkPoint
* gpts 
= reinterpret_cast<GdkPoint
*>(points
); 
 683         gpts 
= new GdkPoint
[n
]; 
 685     for (int i 
= 0; i 
< n
; i
++) 
 689             gpts
[i
].x 
= XLOG2DEV(points
[i
].x 
+ xoffset
); 
 690             gpts
[i
].y 
= YLOG2DEV(points
[i
].y 
+ yoffset
); 
 692         CalcBoundingBox(points
[i
].x 
+ xoffset
, points
[i
].y 
+ yoffset
); 
 696         gdk_draw_lines( m_gdkwindow
, m_penGC
, gpts
, n
); 
 702 void wxWindowDCImpl::DoDrawPolygon( int n
, wxPoint points
[], 
 703                                     wxCoord xoffset
, wxCoord yoffset
, 
 704                                     wxPolygonFillMode 
WXUNUSED(fillStyle
) ) 
 706     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 710     //Check, if scaling is necessary 
 712         xoffset 
!= 0 || yoffset 
!= 0 || XLOG2DEV(10) != 10 || YLOG2DEV(10) != 10; 
 714     // GdkPoint and wxPoint have the same memory layout, so we can cast one to the other 
 715     GdkPoint
* gdkpoints 
= reinterpret_cast<GdkPoint
*>(points
); 
 718         gdkpoints 
= new GdkPoint
[n
]; 
 721     for (i 
= 0 ; i 
< n 
; i
++) 
 725             gdkpoints
[i
].x 
= XLOG2DEV(points
[i
].x 
+ xoffset
); 
 726             gdkpoints
[i
].y 
= YLOG2DEV(points
[i
].y 
+ yoffset
); 
 728         CalcBoundingBox(points
[i
].x 
+ xoffset
, points
[i
].y 
+ yoffset
); 
 733         if (m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) 
 737             DrawingSetup(gc
, originChanged
); 
 739             gdk_draw_polygon(m_gdkwindow
, gc
, true, gdkpoints
, n
); 
 742                 gdk_gc_set_ts_origin(gc
, 0, 0); 
 745         if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 748             for (i = 0 ; i < n ; i++) 
 750                 gdk_draw_line( m_gdkwindow, m_penGC, 
 753                                gdkpoints[(i+1)%n].x, 
 754                                gdkpoints[(i+1)%n].y); 
 757             gdk_draw_polygon( m_gdkwindow
, m_penGC
, FALSE
, gdkpoints
, n 
); 
 766 void wxWindowDCImpl::DoDrawRectangle( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height 
) 
 768     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 770     wxCoord xx 
= XLOG2DEV(x
); 
 771     wxCoord yy 
= YLOG2DEV(y
); 
 772     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 773     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 775     // CMB: draw nothing if transformed w or h is 0 
 776     if (ww 
== 0 || hh 
== 0) return; 
 778     // CMB: handle -ve width and/or height 
 779     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 780     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 784         if (m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) 
 788             DrawingSetup(gc
, originChanged
); 
 790             gdk_draw_rectangle(m_gdkwindow
, gc
, true, xx
, yy
, ww
, hh
); 
 793                 gdk_gc_set_ts_origin(gc
, 0, 0); 
 796         if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 799             if ((m_pen
.GetWidth() == 2) && (m_pen
.GetCap() == wxCAP_ROUND
) && 
 800                 (m_pen
.GetJoin() == wxJOIN_ROUND
) && (m_pen
.GetStyle() == wxPENSTYLE_SOLID
)) 
 802                 // Use 2 1-line rects instead 
 803                 gdk_gc_set_line_attributes( m_penGC
, 1, GDK_LINE_SOLID
, GDK_CAP_ROUND
, GDK_JOIN_ROUND 
); 
 808                     gdk_draw_rectangle( m_gdkwindow
, m_penGC
, FALSE
, xx
+1, yy
, ww
-2, hh
-2 ); 
 809                     gdk_draw_rectangle( m_gdkwindow
, m_penGC
, FALSE
, xx
, yy
-1, ww
, hh 
); 
 813                     gdk_draw_rectangle( m_gdkwindow
, m_penGC
, FALSE
, xx
, yy
, ww
-2, hh
-2 ); 
 814                     gdk_draw_rectangle( m_gdkwindow
, m_penGC
, FALSE
, xx
-1, yy
-1, ww
, hh 
); 
 818                 gdk_gc_set_line_attributes( m_penGC
, 2, GDK_LINE_SOLID
, GDK_CAP_ROUND
, GDK_JOIN_ROUND 
); 
 823                 // Just use X11 for other cases 
 824                 gdk_draw_rectangle( m_gdkwindow
, m_penGC
, FALSE
, xx
, yy
, ww
-1, hh
-1 ); 
 829     CalcBoundingBox( x
, y 
); 
 830     CalcBoundingBox( x 
+ width
, y 
+ height 
); 
 833 void wxWindowDCImpl::DoDrawRoundedRectangle( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double radius 
) 
 835     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 837     if (radius 
< 0.0) radius 
= - radius 
* ((width 
< height
) ? width 
: height
); 
 839     wxCoord xx 
= XLOG2DEV(x
); 
 840     wxCoord yy 
= YLOG2DEV(y
); 
 841     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 842     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 843     wxCoord rr 
= XLOG2DEVREL((wxCoord
)radius
); 
 845     // CMB: handle -ve width and/or height 
 846     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 847     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 849     // CMB: if radius is zero use DrawRectangle() instead to avoid 
 850     // X drawing errors with small radii 
 853         DoDrawRectangle( x
, y
, width
, height 
); 
 857     // CMB: draw nothing if transformed w or h is 0 
 858     if (ww 
== 0 || hh 
== 0) return; 
 860     // CMB: adjust size if outline is drawn otherwise the result is 
 861     // 1 pixel too wide and high 
 862     if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 870         // CMB: ensure dd is not larger than rectangle otherwise we 
 871         // get an hour glass shape 
 873         if (dd 
> ww
) dd 
= ww
; 
 874         if (dd 
> hh
) dd 
= hh
; 
 877         if (m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) 
 881             DrawingSetup(gc
, originChanged
); 
 883             gdk_draw_rectangle(m_gdkwindow
, gc
, true, xx
+rr
, yy
, ww
-dd
+1, hh
); 
 884             gdk_draw_rectangle(m_gdkwindow
, gc
, true, xx
, yy
+rr
, ww
, hh
-dd
+1); 
 885             gdk_draw_arc(m_gdkwindow
, gc
, true, xx
, yy
, dd
, dd
, 90*64, 90*64); 
 886             gdk_draw_arc(m_gdkwindow
, gc
, true, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64); 
 887             gdk_draw_arc(m_gdkwindow
, gc
, true, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64); 
 888             gdk_draw_arc(m_gdkwindow
, gc
, true, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64); 
 891                 gdk_gc_set_ts_origin(gc
, 0, 0); 
 894         if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 896             gdk_draw_line( m_gdkwindow
, m_penGC
, xx
+rr
+1, yy
, xx
+ww
-rr
, yy 
); 
 897             gdk_draw_line( m_gdkwindow
, m_penGC
, xx
+rr
+1, yy
+hh
, xx
+ww
-rr
, yy
+hh 
); 
 898             gdk_draw_line( m_gdkwindow
, m_penGC
, xx
, yy
+rr
+1, xx
, yy
+hh
-rr 
); 
 899             gdk_draw_line( m_gdkwindow
, m_penGC
, xx
+ww
, yy
+rr
+1, xx
+ww
, yy
+hh
-rr 
); 
 900             gdk_draw_arc( m_gdkwindow
, m_penGC
, FALSE
, xx
, yy
, dd
, dd
, 90*64, 90*64 ); 
 901             gdk_draw_arc( m_gdkwindow
, m_penGC
, FALSE
, xx
+ww
-dd
, yy
, dd
, dd
, 0, 90*64 ); 
 902             gdk_draw_arc( m_gdkwindow
, m_penGC
, FALSE
, xx
+ww
-dd
, yy
+hh
-dd
, dd
, dd
, 270*64, 90*64 ); 
 903             gdk_draw_arc( m_gdkwindow
, m_penGC
, FALSE
, xx
, yy
+hh
-dd
, dd
, dd
, 180*64, 90*64 ); 
 907     // this ignores the radius 
 908     CalcBoundingBox( x
, y 
); 
 909     CalcBoundingBox( x 
+ width
, y 
+ height 
); 
 912 void wxWindowDCImpl::DoDrawEllipse( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height 
) 
 914     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 916     wxCoord xx 
= XLOG2DEV(x
); 
 917     wxCoord yy 
= YLOG2DEV(y
); 
 918     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 919     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 921     // CMB: handle -ve width and/or height 
 922     if (ww 
< 0) { ww 
= -ww
; xx 
= xx 
- ww
; } 
 923     if (hh 
< 0) { hh 
= -hh
; yy 
= yy 
- hh
; } 
 927         if (m_brush
.GetStyle() != wxBRUSHSTYLE_TRANSPARENT
) 
 931             DrawingSetup(gc
, originChanged
); 
 933             // If the pen is transparent pen we increase the size 
 934             // for better compatibility with other platforms. 
 935             if (m_pen
.GetStyle() == wxPENSTYLE_TRANSPARENT
) 
 941             gdk_draw_arc(m_gdkwindow
, gc
, true, xx
, yy
, ww
, hh
, 0, 360*64); 
 944                 gdk_gc_set_ts_origin(gc
, 0, 0); 
 947         if (m_pen
.GetStyle() != wxPENSTYLE_TRANSPARENT
) 
 948             gdk_draw_arc( m_gdkwindow
, m_penGC
, false, xx
, yy
, ww
, hh
, 0, 360*64 ); 
 951     CalcBoundingBox( x
, y 
); 
 952     CalcBoundingBox( x 
+ width
, y 
+ height 
); 
 955 void wxWindowDCImpl::DoDrawIcon( const wxIcon 
&icon
, wxCoord x
, wxCoord y 
) 
 957     // VZ: egcs 1.0.3 refuses to compile this without cast, no idea why 
 958     DoDrawBitmap( (const wxBitmap
&)icon
, x
, y
, true ); 
 961 // scale a pixbuf, return new pixbuf, unref old one 
 963 Scale(GdkPixbuf
* pixbuf
, int dst_w
, int dst_h
, double sx
, double sy
) 
 965     GdkPixbuf
* pixbuf_scaled 
= gdk_pixbuf_new( 
 966         GDK_COLORSPACE_RGB
, gdk_pixbuf_get_has_alpha(pixbuf
), 8, dst_w
, dst_h
); 
 967     gdk_pixbuf_scale(pixbuf
, pixbuf_scaled
, 
 968         0, 0, dst_w
, dst_h
, 0, 0, sx
, sy
, GDK_INTERP_NEAREST
); 
 969     g_object_unref(pixbuf
); 
 970     return pixbuf_scaled
; 
 973 // scale part of a pixmap using pixbuf scaling, return pixbuf 
 975 Scale(GdkPixmap
* pixmap
, int x
, int y
, int w
, int h
, int dst_w
, int dst_h
, double sx
, double sy
) 
 977     GdkPixbuf
* pixbuf 
= gdk_pixbuf_get_from_drawable( 
 978         NULL
, pixmap
, NULL
, x
, y
, 0, 0, w
, h
); 
 979     return Scale(pixbuf
, dst_w
, dst_h
, sx
, sy
); 
 982 // scale part of a mask pixmap, return new mask, unref old one 
 984 ScaleMask(GdkPixmap
* mask
, int x
, int y
, int w
, int h
, int dst_w
, int dst_h
, double sx
, double sy
) 
 986     GdkPixbuf
* pixbuf 
= Scale(mask
, x
, y
, w
, h
, dst_w
, dst_h
, sx
, sy
); 
 988     // convert black and white pixbuf back to a mono pixmap 
 989     const unsigned out_rowstride 
= (dst_w 
+ 7) / 8; 
 990     const size_t data_size 
= out_rowstride 
* size_t(dst_h
); 
 991     char* data 
= new char[data_size
]; 
 993     const guchar
* row 
= gdk_pixbuf_get_pixels(pixbuf
); 
 994     const int rowstride 
= gdk_pixbuf_get_rowstride(pixbuf
); 
 995     memset(data
, 0, data_size
); 
 996     for (int j 
= 0; j 
< dst_h
; j
++, row 
+= rowstride
, out 
+= out_rowstride
) 
 998         const guchar
* in 
= row
; 
 999         for (int i 
= 0; i 
< dst_w
; i
++, in 
+= 3) 
1001                 out
[i 
>> 3] |= 1 << (i 
& 7); 
1003     g_object_unref(pixbuf
); 
1004     GdkPixmap
* pixmap 
= gdk_bitmap_create_from_data(mask
, data
, dst_w
, dst_h
); 
1006     g_object_unref(mask
); 
1010 // Make a new mask from part of a mask and a clip region. 
1011 // Return new mask, unref old one. 
1013 ClipMask(GdkPixmap
* mask
, GdkRegion
* clipRegion
, int x
, int y
, int dst_x
, int dst_y
, int w
, int h
) 
1015     GdkGCValues gcValues
; 
1016     gcValues
.foreground
.pixel 
= 0; 
1017     GdkGC
* gc 
= gdk_gc_new_with_values(mask
, &gcValues
, GDK_GC_FOREGROUND
); 
1018     GdkPixmap
* pixmap 
= gdk_pixmap_new(mask
, w
, h
, 1); 
1019     // clear new mask, so clipped areas will be masked 
1020     gdk_draw_rectangle(pixmap
, gc
, true, 0, 0, w
, h
); 
1021     gdk_gc_set_clip_region(gc
, clipRegion
); 
1022     gdk_gc_set_clip_origin(gc
, -dst_x
, -dst_y
); 
1023     // draw old mask onto new one, with clip 
1024     gdk_draw_drawable(pixmap
, gc
, mask
, x
, y
, 0, 0, w
, h
); 
1026     g_object_unref(mask
); 
1030 // make a color pixmap from part of a mono one, using text fg/bg colors 
1032 wxWindowDCImpl::MonoToColor(GdkPixmap
* monoPixmap
, int x
, int y
, int w
, int h
) const 
1034     GdkPixmap
* pixmap 
= gdk_pixmap_new(m_gdkwindow
, w
, h
, -1); 
1035     GdkGCValues gcValues
; 
1036     gcValues
.foreground
.pixel 
= m_textForegroundColour
.GetColor()->pixel
; 
1037     gcValues
.background
.pixel 
= m_textBackgroundColour
.GetColor()->pixel
; 
1038     gcValues
.stipple 
= monoPixmap
; 
1039     gcValues
.fill 
= GDK_OPAQUE_STIPPLED
; 
1040     gcValues
.ts_x_origin 
= -x
; 
1041     gcValues
.ts_y_origin 
= -y
; 
1042     GdkGC
* gc 
= gdk_gc_new_with_values(pixmap
, &gcValues
, GdkGCValuesMask( 
1043         GDK_GC_FOREGROUND 
| GDK_GC_BACKGROUND 
| GDK_GC_STIPPLE 
| GDK_GC_FILL 
| 
1044         GDK_GC_TS_X_ORIGIN 
| GDK_GC_TS_Y_ORIGIN
)); 
1045     gdk_draw_rectangle(pixmap
, gc
, true, 0, 0, w
, h
); 
1050 void wxWindowDCImpl::DoDrawBitmap( const wxBitmap 
&bitmap
, 
1051                                wxCoord x
, wxCoord y
, 
1054     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1055     wxCHECK_RET( bitmap
.IsOk(), wxT("invalid bitmap") ); 
1057     if (!m_gdkwindow
) return; 
1059     const int w 
= bitmap
.GetWidth(); 
1060     const int h 
= bitmap
.GetHeight(); 
1062     CalcBoundingBox(x
, y
); 
1063     CalcBoundingBox(x 
+ w
, y 
+ h
); 
1066     int xx 
= LogicalToDeviceX(x
); 
1067     const int yy 
= LogicalToDeviceY(y
); 
1068     const int ww 
= LogicalToDeviceXRel(w
); 
1069     const int hh 
= LogicalToDeviceYRel(h
); 
1071     if (m_window 
&& m_window
->GetLayoutDirection() == wxLayout_RightToLeft
) 
1074     GdkRegion
* const clipRegion 
= m_currentClippingRegion
.GetRegion(); 
1075     // determine clip region overlap 
1076     int overlap 
= wxInRegion
; 
1079         overlap 
= m_currentClippingRegion
.Contains(xx
, yy
, ww
, hh
); 
1080         if (overlap 
== wxOutRegion
) 
1084     const bool isScaled 
= ww 
!= w 
|| hh 
!= h
; 
1085     const bool hasAlpha 
= bitmap
.HasAlpha(); 
1086     GdkGC
* const use_gc 
= m_penGC
; 
1088     GdkPixmap
* mask 
= NULL
; 
1089     // mask does not work when drawing a pixbuf with alpha 
1090     if (useMask 
&& !hasAlpha
) 
1092         wxMask
* m 
= bitmap
.GetMask(); 
1094             mask 
= m
->GetBitmap(); 
1100             mask 
= ScaleMask(mask
, 0, 0, w
, h
, ww
, hh
, m_scaleX
, m_scaleY
); 
1101         if (overlap 
== wxPartRegion
) 
1103             // need a new mask that also masks the clipped area, 
1104             // because gc can't have both a mask and a clip region 
1105             mask 
= ClipMask(mask
, clipRegion
, 0, 0, xx
, yy
, ww
, hh
); 
1107         gdk_gc_set_clip_mask(use_gc
, mask
); 
1108         gdk_gc_set_clip_origin(use_gc
, xx
, yy
); 
1111     // determine whether to use pixmap or pixbuf 
1112     GdkPixmap
* pixmap 
= NULL
; 
1113     GdkPixbuf
* pixbuf 
= NULL
; 
1114     if (bitmap
.HasPixmap()) 
1115         pixmap 
= bitmap
.GetPixmap(); 
1116     if (pixmap 
&& gdk_drawable_get_depth(pixmap
) == 1) 
1118         // convert mono pixmap to color using text fg/bg colors 
1119         pixmap 
= MonoToColor(pixmap
, 0, 0, w
, h
); 
1121     else if (hasAlpha 
|| pixmap 
== NULL
) 
1124         pixbuf 
= bitmap
.GetPixbuf(); 
1125         g_object_ref(pixbuf
); 
1129         g_object_ref(pixmap
); 
1135             pixbuf 
= Scale(pixbuf
, ww
, hh
, m_scaleX
, m_scaleY
); 
1137             pixbuf 
= Scale(pixmap
, 0, 0, w
, h
, ww
, hh
, m_scaleX
, m_scaleY
); 
1142         gdk_draw_pixbuf(m_gdkwindow
, use_gc
, pixbuf
, 
1143             0, 0, xx
, yy
, ww
, hh
, GDK_RGB_DITHER_NORMAL
, 0, 0); 
1144         g_object_unref(pixbuf
); 
1148         gdk_draw_drawable(m_gdkwindow
, use_gc
, pixmap
, 0, 0, xx
, yy
, ww
, hh
); 
1152         g_object_unref(pixmap
); 
1155         g_object_unref(mask
); 
1156         gdk_gc_set_clip_region(use_gc
, clipRegion
); 
1160 bool wxWindowDCImpl::DoBlit( wxCoord xdest
, wxCoord ydest
, 
1161                          wxCoord width
, wxCoord height
, 
1163                          wxCoord xsrc
, wxCoord ysrc
, 
1164                          wxRasterOperationMode logical_func
, 
1166                          wxCoord xsrcMask
, wxCoord ysrcMask 
) 
1168     wxCHECK_MSG( IsOk(), false, wxT("invalid window dc") ); 
1169     wxCHECK_MSG( source
, false, wxT("invalid source dc") ); 
1171     if (!m_gdkwindow
) return false; 
1173     GdkDrawable
* srcDrawable 
= NULL
; 
1174     GdkPixmap
* mask 
= NULL
; 
1175     wxMemoryDC
* memDC 
= wxDynamicCast(source
, wxMemoryDC
); 
1178         const wxBitmap
& bitmap 
= memDC
->GetSelectedBitmap(); 
1181         srcDrawable 
= bitmap
.GetPixmap(); 
1184             wxMask
* m 
= bitmap
.GetMask(); 
1186                 mask 
= m
->GetBitmap(); 
1191         wxDCImpl
* impl 
= source
->GetImpl(); 
1192         wxWindowDCImpl
* gtk_impl 
= wxDynamicCast(impl
, wxWindowDCImpl
); 
1194             srcDrawable 
= gtk_impl
->GetGDKWindow(); 
1195         if (srcDrawable 
== NULL
) 
1199     CalcBoundingBox(xdest
, ydest
); 
1200     CalcBoundingBox(xdest 
+ width
, ydest 
+ height
); 
1202     // source device coords 
1203     int src_x 
= source
->LogicalToDeviceX(xsrc
); 
1204     int src_y 
= source
->LogicalToDeviceY(ysrc
); 
1205     int src_w 
= source
->LogicalToDeviceXRel(width
); 
1206     int src_h 
= source
->LogicalToDeviceYRel(height
); 
1208     // Clip source rect to source dc. 
1209     // Only necessary when scaling, to avoid GDK errors when 
1210     // converting to pixbuf, but no harm in always doing it. 
1211     // If source rect changes, it also changes the dest rect. 
1213     gdk_drawable_get_size(srcDrawable
, &clip
.width
, &clip
.height
); 
1214     clip
.Intersect(wxRect(src_x
, src_y
, src_w
, src_h
)); 
1215     if (src_w 
!= clip
.width 
|| src_h 
!= clip
.height
) 
1217         if (clip
.width 
== 0) 
1221         src_h 
= clip
.height
; 
1222         width  
= source
->DeviceToLogicalXRel(src_w
); 
1223         height 
= source
->DeviceToLogicalYRel(src_h
); 
1224         if (src_x 
!= clip
.x 
|| src_y 
!= clip
.y
) 
1226             xdest 
+= source
->DeviceToLogicalXRel(clip
.x 
- src_x
); 
1227             ydest 
+= source
->DeviceToLogicalYRel(clip
.y 
- src_y
); 
1233     // destination device coords 
1234     const int dst_x 
= LogicalToDeviceX(xdest
); 
1235     const int dst_y 
= LogicalToDeviceY(ydest
); 
1236     const int dst_w 
= LogicalToDeviceXRel(width
); 
1237     const int dst_h 
= LogicalToDeviceYRel(height
); 
1239     GdkRegion
* const clipRegion 
= m_currentClippingRegion
.GetRegion(); 
1240     // determine dest clip region overlap 
1241     int overlap 
= wxInRegion
; 
1244         overlap 
= m_currentClippingRegion
.Contains(dst_x
, dst_y
, dst_w
, dst_h
); 
1245         if (overlap 
== wxOutRegion
) 
1249     const bool isScaled 
= src_w 
!= dst_w 
|| src_h 
!= dst_h
; 
1254         // get source to dest scale 
1255         double usx
, usy
, lsx
, lsy
; 
1256         source
->GetUserScale(&usx
, &usy
); 
1257         source
->GetLogicalScale(&lsx
, &lsy
); 
1258         scale_x 
= m_scaleX 
/ (usx 
* lsx
); 
1259         scale_y 
= m_scaleY 
/ (usy 
* lsy
); 
1262     GdkGC
* const use_gc 
= m_penGC
; 
1267         int srcMask_x 
= src_x
; 
1268         int srcMask_y 
= src_y
; 
1269         if (xsrcMask 
!= -1 || ysrcMask 
!= -1) 
1271             srcMask_x 
= source
->LogicalToDeviceX(xsrcMask
); 
1272             srcMask_y 
= source
->LogicalToDeviceY(ysrcMask
); 
1276             mask 
= ScaleMask(mask
, srcMask_x
, srcMask_y
, 
1277                 src_w
, src_h
, dst_w
, dst_h
, scale_x
, scale_y
); 
1281         if (overlap 
== wxPartRegion
) 
1283             // need a new mask that also masks the clipped area, 
1284             // because gc can't have both a mask and a clip region 
1285             mask 
= ClipMask(mask
, clipRegion
, 
1286                 srcMask_x
, srcMask_y
, dst_x
, dst_y
, dst_w
, dst_h
); 
1290         gdk_gc_set_clip_mask(use_gc
, mask
); 
1291         gdk_gc_set_clip_origin(use_gc
, dst_x 
- srcMask_x
, dst_y 
- srcMask_y
); 
1294     GdkPixmap
* pixmap 
= NULL
; 
1295     if (gdk_drawable_get_depth(srcDrawable
) == 1) 
1297         // Convert mono pixmap to color using text fg/bg colors. 
1298         // Scaling/drawing is simpler if this is done first. 
1299         pixmap 
= MonoToColor(srcDrawable
, src_x
, src_y
, src_w
, src_h
); 
1300         srcDrawable 
= pixmap
; 
1305     const wxRasterOperationMode logical_func_save 
= m_logicalFunction
; 
1306     SetLogicalFunction(logical_func
); 
1308         gdk_gc_set_subwindow(use_gc
, GDK_INCLUDE_INFERIORS
); 
1312         GdkPixbuf
* pixbuf 
= Scale(srcDrawable
, 
1313             src_x
, src_y
, src_w
, src_h
, dst_w
, dst_h
, scale_x
, scale_y
); 
1314         gdk_draw_pixbuf(m_gdkwindow
, use_gc
, pixbuf
, 
1315             0, 0, dst_x
, dst_y
, dst_w
, dst_h
, GDK_RGB_DITHER_NONE
, 0, 0); 
1316         g_object_unref(pixbuf
); 
1320         gdk_draw_drawable(m_gdkwindow
, use_gc
, srcDrawable
, 
1321             src_x
, src_y
, dst_x
, dst_y
, dst_w
, dst_h
); 
1324     SetLogicalFunction(logical_func_save
); 
1326         gdk_gc_set_subwindow(use_gc
, GDK_CLIP_BY_CHILDREN
); 
1329         g_object_unref(pixmap
); 
1332         g_object_unref(mask
); 
1333         gdk_gc_set_clip_region(use_gc
, clipRegion
); 
1338 void wxWindowDCImpl::DoDrawText( const wxString 
&text
, wxCoord x
, wxCoord y 
) 
1340     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1342     if (!m_gdkwindow
) return; 
1344     if (text
.empty()) return; 
1349     wxCHECK_RET( m_context
, wxT("no Pango context") ); 
1350     wxCHECK_RET( m_layout
, wxT("no Pango layout") ); 
1351     wxCHECK_RET( m_fontdesc
, wxT("no Pango font description") ); 
1353     gdk_pango_context_set_colormap( m_context
, m_cmap 
);  // not needed in gtk+ >= 2.6 
1355     bool underlined 
= m_font
.IsOk() && m_font
.GetUnderlined(); 
1357     wxCharBuffer data 
= wxGTK_CONV(text
); 
1360     size_t datalen 
= strlen(data
); 
1362     // in Pango >= 1.16 the "underline of leading/trailing spaces" bug 
1363     // has been fixed and thus the hack implemented below should never be used 
1364     static bool pangoOk 
= !wx_pango_version_check(1, 16, 0); 
1366     bool needshack 
= underlined 
&& !pangoOk
; 
1370         // a PangoLayout which has leading/trailing spaces with underlined font 
1371         // is not correctly drawn by this pango version: Pango won't underline the spaces. 
1372         // This can be a problem; e.g. wxHTML rendering of underlined text relies on 
1373         // this behaviour. To workaround this problem, we use a special hack here 
1374         // suggested by pango maintainer Behdad Esfahbod: we prepend and append two 
1375         // empty space characters and give them a dummy colour attribute. 
1376         // This will force Pango to underline the leading/trailing spaces, too. 
1378         wxCharBuffer 
data_tmp(datalen 
+ 6); 
1379         // copy the leading U+200C ZERO WIDTH NON-JOINER encoded in UTF8 format 
1380         memcpy(data_tmp
.data(), "\342\200\214", 3); 
1381         // copy the user string 
1382         memcpy(data_tmp
.data() + 3, data
, datalen
); 
1383         // copy the trailing U+200C ZERO WIDTH NON-JOINER encoded in UTF8 format 
1384         memcpy(data_tmp
.data() + 3 + datalen
, "\342\200\214", 3); 
1390     pango_layout_set_text(m_layout
, data
, datalen
); 
1394         PangoAttrList 
*attrs 
= pango_attr_list_new(); 
1395         PangoAttribute 
*a 
= pango_attr_underline_new(PANGO_UNDERLINE_SINGLE
); 
1397         a
->end_index 
= datalen
; 
1398         pango_attr_list_insert(attrs
, a
); 
1402             // dummy colour for the leading space 
1403             a 
= pango_attr_foreground_new (0x0057, 0x52A9, 0xD614); 
1406             pango_attr_list_insert(attrs
, a
); 
1408             // dummy colour for the trailing space 
1409             a 
= pango_attr_foreground_new (0x0057, 0x52A9, 0xD614); 
1410             a
->start_index 
= datalen 
- 1; 
1411             a
->end_index 
= datalen
; 
1412             pango_attr_list_insert(attrs
, a
); 
1415         pango_layout_set_attributes(m_layout
, attrs
); 
1416         pango_attr_list_unref(attrs
); 
1420     const bool isScaled 
= fabs(m_scaleY 
- 1.0) > 0.00001; 
1423         // If there is a user or actually any scale applied to 
1424         // the device context, scale the font. 
1426         // scale font description 
1427         oldSize 
= pango_font_description_get_size(m_fontdesc
); 
1428         pango_font_description_set_size(m_fontdesc
, int(oldSize 
* m_scaleY
)); 
1430         // actually apply scaled font 
1431         pango_layout_set_font_description( m_layout
, m_fontdesc 
); 
1435     pango_layout_get_pixel_size(m_layout
, &w
, &h
); 
1439     if (m_window 
&& m_window
->GetLayoutDirection() == wxLayout_RightToLeft
) 
1442     const GdkColor
* bg_col 
= NULL
; 
1443     if (m_backgroundMode 
== wxBRUSHSTYLE_SOLID
) 
1444         bg_col 
= m_textBackgroundColour
.GetColor(); 
1446     gdk_draw_layout_with_colors(m_gdkwindow
, m_textGC
, x_rtl
, y
, m_layout
, NULL
, bg_col
); 
1450          // reset unscaled size 
1451          pango_font_description_set_size( m_fontdesc
, oldSize 
); 
1453          // actually apply unscaled font 
1454          pango_layout_set_font_description( m_layout
, m_fontdesc 
); 
1458         // undo underline attributes setting: 
1459         pango_layout_set_attributes(m_layout
, NULL
); 
1462     CalcBoundingBox(x 
+ int(w 
/ m_scaleX
), y 
+ int(h 
/ m_scaleY
)); 
1463     CalcBoundingBox(x
, y
); 
1466 // TODO: When GTK2.6 is required, merge DoDrawText and DoDrawRotatedText to 
1467 // avoid code duplication 
1468 void wxWindowDCImpl::DoDrawRotatedText( const wxString 
&text
, wxCoord x
, wxCoord y
, double angle 
) 
1470     if (!m_gdkwindow 
|| text
.empty()) 
1473     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1476     if (!gtk_check_version(2,6,0)) 
1481         pango_layout_set_text(m_layout
, wxGTK_CONV(text
), -1); 
1483         if (m_font
.GetUnderlined()) 
1485             PangoAttrList 
*attrs 
= pango_attr_list_new(); 
1486             PangoAttribute 
*a 
= pango_attr_underline_new(PANGO_UNDERLINE_SINGLE
); 
1487             pango_attr_list_insert(attrs
, a
); 
1488             pango_layout_set_attributes(m_layout
, attrs
); 
1489             pango_attr_list_unref(attrs
); 
1493         const bool isScaled 
= fabs(m_scaleY 
- 1.0) > 0.00001; 
1496             //TODO: when Pango >= 1.6 is required, use pango_matrix_scale() 
1497              // If there is a user or actually any scale applied to 
1498              // the device context, scale the font. 
1500              // scale font description 
1501             oldSize 
= pango_font_description_get_size(m_fontdesc
); 
1502             pango_font_description_set_size(m_fontdesc
, int(oldSize 
* m_scaleY
)); 
1504              // actually apply scaled font 
1505              pango_layout_set_font_description( m_layout
, m_fontdesc 
); 
1509         pango_layout_get_pixel_size(m_layout
, &w
, &h
); 
1511         const GdkColor
* bg_col 
= NULL
; 
1512         if (m_backgroundMode 
== wxBRUSHSTYLE_SOLID
) 
1513             bg_col 
= m_textBackgroundColour
.GetColor(); 
1516         PangoMatrix matrix 
= PANGO_MATRIX_INIT
; 
1517         pango_matrix_rotate (&matrix
, angle
); 
1518         pango_context_set_matrix (m_context
, &matrix
); 
1519         pango_layout_context_changed (m_layout
); 
1521         // To be compatible with MSW, the rotation axis must be in the old 
1523         // Calculate the vertices of the rotated rectangle containing the text, 
1524         // relative to the old top-left vertex. 
1525         // We could use the matrix for this, but it's simpler with trignonometry. 
1526         double rad 
= DegToRad(angle
); 
1527         // the rectangle vertices are counted clockwise with the first one 
1529         double x2 
= w 
* cos(rad
); 
1530         double y2 
= -w 
* sin(rad
);   // y axis points to the bottom, hence minus 
1531         double x4 
= h 
* sin(rad
); 
1532         double y4 
= h 
* cos(rad
); 
1533         double x3 
= x4 
+ x2
; 
1534         double y3 
= y4 
+ y2
; 
1535         // Then we calculate max and min of the rotated rectangle. 
1536         wxCoord maxX 
= (wxCoord
)(dmax(dmax(0, x2
), dmax(x3
, x4
)) + 0.5), 
1537                 maxY 
= (wxCoord
)(dmax(dmax(0, y2
), dmax(y3
, y4
)) + 0.5), 
1538                 minX 
= (wxCoord
)(dmin(dmin(0, x2
), dmin(x3
, x4
)) - 0.5), 
1539                 minY 
= (wxCoord
)(dmin(dmin(0, y2
), dmin(y3
, y4
)) - 0.5); 
1541         gdk_draw_layout_with_colors(m_gdkwindow
, m_textGC
, x
+minX
, y
+minY
, 
1542                                     m_layout
, NULL
, bg_col
); 
1544         if (m_font
.GetUnderlined()) 
1545             pango_layout_set_attributes(m_layout
, NULL
); 
1547         // clean up the transformation matrix 
1548         pango_context_set_matrix(m_context
, NULL
); 
1552              // reset unscaled size 
1553              pango_font_description_set_size( m_fontdesc
, oldSize 
); 
1555              // actually apply unscaled font 
1556              pango_layout_set_font_description( m_layout
, m_fontdesc 
); 
1559         CalcBoundingBox(x
+minX
, y
+minY
); 
1560         CalcBoundingBox(x
+maxX
, y
+maxY
); 
1563 #endif //__WXGTK26__ 
1566     if ( wxIsNullDouble(angle
) ) 
1568         DoDrawText(text
, x
, y
); 
1575     // TODO: implement later without GdkFont for GTK 2.0 
1576     DoGetTextExtent(text
, &w
, &h
, NULL
,NULL
, &m_font
); 
1578     // draw the string normally 
1581     dc
.SelectObject(src
); 
1582     dc
.SetFont(GetFont()); 
1583     dc
.SetBackground(*wxBLACK_BRUSH
); 
1584     dc
.SetBrush(*wxBLACK_BRUSH
); 
1586     dc
.SetTextForeground( *wxWHITE 
); 
1587     dc
.DrawText(text
, 0, 0); 
1588     dc
.SelectObject(wxNullBitmap
); 
1590     // Calculate the size of the rotated bounding box. 
1591     double rad 
= DegToRad(angle
); 
1592     double dx 
= cos(rad
), 
1595     // the rectngle vertices are counted clockwise with the first one being at 
1596     // (0, 0) (or, rather, at (x, y)) 
1598            y2 
= -w
*dy
;      // y axis points to the bottom, hence minus 
1601     double x3 
= x4 
+ x2
, 
1605     wxCoord maxX 
= (wxCoord
)(dmax(x2
, dmax(x3
, x4
)) + 0.5), 
1606             maxY 
= (wxCoord
)(dmax(y2
, dmax(y3
, y4
)) + 0.5), 
1607             minX 
= (wxCoord
)(dmin(x2
, dmin(x3
, x4
)) - 0.5), 
1608             minY 
= (wxCoord
)(dmin(y2
, dmin(y3
, y4
)) - 0.5); 
1611     wxImage image 
= src
.ConvertToImage(); 
1613     image
.ConvertColourToAlpha( m_textForegroundColour
.Red(), 
1614                                 m_textForegroundColour
.Green(), 
1615                                 m_textForegroundColour
.Blue() ); 
1616     image 
= image
.Rotate( rad
, wxPoint(0,0) ); 
1618     int i_angle 
= (int) angle
; 
1619     i_angle 
= i_angle 
% 360; 
1623     if ((i_angle 
>= 90.0) && (i_angle 
< 270.0)) 
1624         xoffset 
= image
.GetWidth(); 
1626     if ((i_angle 
>= 0.0) && (i_angle 
< 180.0)) 
1627         yoffset 
= image
.GetHeight(); 
1629     if ((i_angle 
>= 0) && (i_angle 
< 90)) 
1630         yoffset 
-= (int)( cos(rad
)*h 
); 
1631     if ((i_angle 
>= 90) && (i_angle 
< 180)) 
1632         xoffset 
-= (int)( sin(rad
)*h 
); 
1633     if ((i_angle 
>= 180) && (i_angle 
< 270)) 
1634         yoffset 
-= (int)( cos(rad
)*h 
); 
1635     if ((i_angle 
>= 270) && (i_angle 
< 360)) 
1636         xoffset 
-= (int)( sin(rad
)*h 
); 
1638     int i_x 
= x 
- xoffset
; 
1639     int i_y 
= y 
- yoffset
; 
1642     DoDrawBitmap( src
, i_x
, i_y
, true ); 
1645     // it would be better to draw with non underlined font and draw the line 
1646     // manually here (it would be more straight...) 
1648     if ( m_font
.GetUnderlined() ) 
1650         gdk_draw_line( m_gdkwindow
, m_textGC
, 
1651                        XLOG2DEV(x 
+ x4
), YLOG2DEV(y 
+ y4 
+ font
->descent
), 
1652                        XLOG2DEV(x 
+ x3
), YLOG2DEV(y 
+ y3 
+ font
->descent
)); 
1656     // update the bounding box 
1657     CalcBoundingBox(x 
+ minX
, y 
+ minY
); 
1658     CalcBoundingBox(x 
+ maxX
, y 
+ maxY
); 
1659 #else // !wxUSE_IMAGE 
1664 #endif // wxUSE_IMAGE/!wxUSE_IMAGE 
1668 void wxWindowDCImpl::DoGetTextExtent(const wxString 
&string
, 
1669                                  wxCoord 
*width
, wxCoord 
*height
, 
1670                                  wxCoord 
*descent
, wxCoord 
*externalLeading
, 
1671                                  const wxFont 
*theFont
) const 
1679     if ( externalLeading 
) 
1680         *externalLeading 
= 0; 
1685     // ensure that theFont is always non-NULL 
1686     if ( !theFont 
|| !theFont
->IsOk() ) 
1689     // and use it if it's valid 
1690     if ( theFont
->IsOk() ) 
1692         pango_layout_set_font_description
 
1695             theFont
->GetNativeFontInfo()->description
 
1699     // Set layout's text 
1700     const wxCharBuffer dataUTF8 
= wxGTK_CONV_FONT(string
, *theFont
); 
1703         // hardly ideal, but what else can we do if conversion failed? 
1707     pango_layout_set_text(m_layout
, dataUTF8
, -1); 
1710     pango_layout_get_pixel_size(m_layout
, width
, &h
); 
1713         PangoLayoutIter 
*iter 
= pango_layout_get_iter(m_layout
); 
1714         int baseline 
= pango_layout_iter_get_baseline(iter
); 
1715         pango_layout_iter_free(iter
); 
1716         *descent 
= h 
- PANGO_PIXELS(baseline
); 
1721     // Reset old font description 
1722     if (theFont
->IsOk()) 
1723         pango_layout_set_font_description( m_layout
, m_fontdesc 
); 
1727 bool wxWindowDCImpl::DoGetPartialTextExtents(const wxString
& text
, 
1728                                          wxArrayInt
& widths
) const 
1730     const size_t len 
= text
.length(); 
1737     // Set layout's text 
1738     const wxCharBuffer dataUTF8 
= wxGTK_CONV_FONT(text
, m_font
); 
1741         // hardly ideal, but what else can we do if conversion failed? 
1742         wxLogLastError(wxT("DoGetPartialTextExtents")); 
1746     pango_layout_set_text(m_layout
, dataUTF8
, -1); 
1748     // Calculate the position of each character based on the widths of 
1749     // the previous characters 
1751     // Code borrowed from Scintilla's PlatGTK 
1752     PangoLayoutIter 
*iter 
= pango_layout_get_iter(m_layout
); 
1754     pango_layout_iter_get_cluster_extents(iter
, NULL
, &pos
); 
1756     while (pango_layout_iter_next_cluster(iter
)) 
1758         pango_layout_iter_get_cluster_extents(iter
, NULL
, &pos
); 
1759         int position 
= PANGO_PIXELS(pos
.x
); 
1760         widths
[i
++] = position
; 
1763         widths
[i
++] = PANGO_PIXELS(pos
.x 
+ pos
.width
); 
1764     pango_layout_iter_free(iter
); 
1770 wxCoord 
wxWindowDCImpl::GetCharWidth() const 
1772     pango_layout_set_text( m_layout
, "H", 1 ); 
1774     pango_layout_get_pixel_size( m_layout
, &w
, NULL 
); 
1778 wxCoord 
wxWindowDCImpl::GetCharHeight() const 
1780     PangoFontMetrics 
*metrics 
= pango_context_get_metrics (m_context
, m_fontdesc
, pango_context_get_language(m_context
)); 
1781     wxCHECK_MSG( metrics
, -1, wxT("failed to get pango font metrics") ); 
1783     wxCoord h 
= PANGO_PIXELS (pango_font_metrics_get_descent (metrics
) + 
1784                               pango_font_metrics_get_ascent (metrics
)); 
1785     pango_font_metrics_unref (metrics
); 
1789 void wxWindowDCImpl::Clear() 
1791     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1793     if (!m_gdkwindow
) return; 
1796     DoGetSize( &width
, &height 
); 
1797     gdk_draw_rectangle( m_gdkwindow
, m_bgGC
, TRUE
, 0, 0, width
, height 
); 
1800 void wxWindowDCImpl::SetFont( const wxFont 
&font 
) 
1807             pango_font_description_free( m_fontdesc 
); 
1809         m_fontdesc 
= pango_font_description_copy( m_font
.GetNativeFontInfo()->description 
); 
1814             PangoContext 
*oldContext 
= m_context
; 
1816             m_context 
= m_window
->GTKGetPangoDefaultContext(); 
1818             // If we switch back/forth between different contexts 
1819             // we also have to create a new layout. I think so, 
1820             // at least, and it doesn't hurt to do it. 
1821             if (oldContext 
!= m_context
) 
1824                     g_object_unref (m_layout
); 
1826                 m_layout 
= pango_layout_new( m_context 
); 
1830         pango_layout_set_font_description( m_layout
, m_fontdesc 
); 
1834 void wxWindowDCImpl::SetPen( const wxPen 
&pen 
) 
1836     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1838     if (m_pen 
== pen
) return; 
1842     if (!m_pen
.IsOk()) return; 
1844     if (!m_gdkwindow
) return; 
1846     gint width 
= m_pen
.GetWidth(); 
1849         // CMB: if width is non-zero scale it with the dc 
1854         // X doesn't allow different width in x and y and so we take 
1857                    ( fabs((double) XLOG2DEVREL(width
)) + 
1858                      fabs((double) YLOG2DEVREL(width
)) ) / 2.0; 
1862             // width can't be 0 or an internal GTK error occurs inside 
1863             // gdk_gc_set_dashes() below 
1868     static const wxGTKDash dotted
[] = {1, 1}; 
1869     static const wxGTKDash short_dashed
[] = {2, 2}; 
1870     static const wxGTKDash wxCoord_dashed
[] = {2, 4}; 
1871     static const wxGTKDash dotted_dashed
[] = {3, 3, 1, 3}; 
1873     // We express dash pattern in pen width unit, so we are 
1874     // independent of zoom factor and so on... 
1876     const wxGTKDash 
*req_dash
; 
1878     GdkLineStyle lineStyle 
= GDK_LINE_ON_OFF_DASH
; 
1879     switch (m_pen
.GetStyle()) 
1881         case wxPENSTYLE_USER_DASH
: 
1882             req_nb_dash 
= m_pen
.GetDashCount(); 
1883             req_dash 
= (wxGTKDash
*)m_pen
.GetDash(); 
1885         case wxPENSTYLE_DOT
: 
1889         case wxPENSTYLE_LONG_DASH
: 
1891             req_dash 
= wxCoord_dashed
; 
1893         case wxPENSTYLE_SHORT_DASH
: 
1895             req_dash 
= short_dashed
; 
1897         case wxPENSTYLE_DOT_DASH
: 
1899             req_dash 
= dotted_dashed
; 
1902         case wxPENSTYLE_TRANSPARENT
: 
1903         case wxPENSTYLE_STIPPLE_MASK_OPAQUE
: 
1904         case wxPENSTYLE_STIPPLE
: 
1905         case wxPENSTYLE_SOLID
: 
1907             lineStyle 
= GDK_LINE_SOLID
; 
1913     if (req_dash 
&& req_nb_dash
) 
1915         wxGTKDash 
*real_req_dash 
= new wxGTKDash
[req_nb_dash
]; 
1918             for (int i 
= 0; i 
< req_nb_dash
; i
++) 
1919                 real_req_dash
[i
] = req_dash
[i
] * width
; 
1920             gdk_gc_set_dashes( m_penGC
, 0, real_req_dash
, req_nb_dash 
); 
1921             delete[] real_req_dash
; 
1925             // No Memory. We use non-scaled dash pattern... 
1926             gdk_gc_set_dashes( m_penGC
, 0, (wxGTKDash
*)req_dash
, req_nb_dash 
); 
1930     GdkCapStyle capStyle 
= GDK_CAP_ROUND
; 
1931     switch (m_pen
.GetCap()) 
1933         case wxCAP_PROJECTING
: { capStyle 
= GDK_CAP_PROJECTING
; break; } 
1934         case wxCAP_BUTT
:       { capStyle 
= GDK_CAP_BUTT
;       break; } 
1940                 capStyle 
= GDK_CAP_NOT_LAST
; 
1945     GdkJoinStyle joinStyle 
= GDK_JOIN_ROUND
; 
1946     switch (m_pen
.GetJoin()) 
1948         case wxJOIN_BEVEL
: { joinStyle 
= GDK_JOIN_BEVEL
; break; } 
1949         case wxJOIN_MITER
: { joinStyle 
= GDK_JOIN_MITER
; break; } 
1951         default:           { joinStyle 
= GDK_JOIN_ROUND
; break; } 
1954     gdk_gc_set_line_attributes( m_penGC
, width
, lineStyle
, capStyle
, joinStyle 
); 
1956     m_pen
.GetColour().CalcPixel( m_cmap 
); 
1957     gdk_gc_set_foreground( m_penGC
, m_pen
.GetColour().GetColor() ); 
1960 void wxWindowDCImpl::SetBrush( const wxBrush 
&brush 
) 
1962     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
1964     if (m_brush 
== brush
) return; 
1968     if (!m_brush
.IsOk()) return; 
1970     if (!m_gdkwindow
) return; 
1972     m_brush
.GetColour().CalcPixel( m_cmap 
); 
1973     gdk_gc_set_foreground( m_brushGC
, m_brush
.GetColour().GetColor() ); 
1975     gdk_gc_set_fill( m_brushGC
, GDK_SOLID 
); 
1977     if ((m_brush
.GetStyle() == wxBRUSHSTYLE_STIPPLE
) && (m_brush
.GetStipple()->IsOk())) 
1979         if (m_brush
.GetStipple()->GetDepth() != 1) 
1981             gdk_gc_set_fill( m_brushGC
, GDK_TILED 
); 
1982             gdk_gc_set_tile( m_brushGC
, m_brush
.GetStipple()->GetPixmap() ); 
1986             gdk_gc_set_fill( m_brushGC
, GDK_STIPPLED 
); 
1987             gdk_gc_set_stipple( m_brushGC
, m_brush
.GetStipple()->GetPixmap() ); 
1991     if ((m_brush
.GetStyle() == wxBRUSHSTYLE_STIPPLE_MASK_OPAQUE
) && (m_brush
.GetStipple()->GetMask())) 
1993         gdk_gc_set_fill( m_textGC
, GDK_OPAQUE_STIPPLED
); 
1994         gdk_gc_set_stipple( m_textGC
, m_brush
.GetStipple()->GetMask()->GetBitmap() ); 
1997     if (m_brush
.IsHatch()) 
1999         gdk_gc_set_fill( m_brushGC
, GDK_STIPPLED 
); 
2000         gdk_gc_set_stipple(m_brushGC
, GetHatch(m_brush
.GetStyle())); 
2004 void wxWindowDCImpl::SetBackground( const wxBrush 
&brush 
) 
2006    /* CMB 21/7/98: Added SetBackground. Sets background brush 
2007     * for Clear() and bg colour for shapes filled with cross-hatch brush */ 
2009     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2011     if (m_backgroundBrush 
== brush
) return; 
2013     m_backgroundBrush 
= brush
; 
2015     if (!m_backgroundBrush
.IsOk()) return; 
2017     if (!m_gdkwindow
) return; 
2019     wxColor color 
= m_backgroundBrush
.GetColour(); 
2020     color
.CalcPixel(m_cmap
); 
2021     const GdkColor
* gdkColor 
= color
.GetColor(); 
2022     gdk_gc_set_background(m_brushGC
, gdkColor
); 
2023     gdk_gc_set_background(m_penGC
,   gdkColor
); 
2024     gdk_gc_set_background(m_bgGC
,    gdkColor
); 
2025     gdk_gc_set_foreground(m_bgGC
,    gdkColor
); 
2028     gdk_gc_set_fill( m_bgGC
, GDK_SOLID 
); 
2030     if (m_backgroundBrush
.GetStyle() == wxBRUSHSTYLE_STIPPLE
) 
2032         const wxBitmap
* stipple 
= m_backgroundBrush
.GetStipple(); 
2033         if (stipple
->IsOk()) 
2035             if (stipple
->GetDepth() != 1) 
2037                 gdk_gc_set_fill(m_bgGC
, GDK_TILED
); 
2038                 gdk_gc_set_tile(m_bgGC
, stipple
->GetPixmap()); 
2042                 gdk_gc_set_fill(m_bgGC
, GDK_STIPPLED
); 
2043                 gdk_gc_set_stipple(m_bgGC
, stipple
->GetPixmap()); 
2047     else if (m_backgroundBrush
.IsHatch()) 
2049         gdk_gc_set_fill( m_bgGC
, GDK_STIPPLED 
); 
2050         gdk_gc_set_stipple(m_bgGC
, GetHatch(m_backgroundBrush
.GetStyle())); 
2054 void wxWindowDCImpl::SetLogicalFunction( wxRasterOperationMode function 
) 
2056     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2058     if (m_logicalFunction 
== function
) 
2061     // VZ: shouldn't this be a CHECK? 
2068         case wxXOR
:          mode 
= GDK_XOR
;           break; 
2069         case wxINVERT
:       mode 
= GDK_INVERT
;        break; 
2070         case wxOR_REVERSE
:   mode 
= GDK_OR_REVERSE
;    break; 
2071         case wxAND_REVERSE
:  mode 
= GDK_AND_REVERSE
;   break; 
2072         case wxCLEAR
:        mode 
= GDK_CLEAR
;         break; 
2073         case wxSET
:          mode 
= GDK_SET
;           break; 
2074         case wxOR_INVERT
:    mode 
= GDK_OR_INVERT
;     break; 
2075         case wxAND
:          mode 
= GDK_AND
;           break; 
2076         case wxOR
:           mode 
= GDK_OR
;            break; 
2077         case wxEQUIV
:        mode 
= GDK_EQUIV
;         break; 
2078         case wxNAND
:         mode 
= GDK_NAND
;          break; 
2079         case wxAND_INVERT
:   mode 
= GDK_AND_INVERT
;    break; 
2080         case wxCOPY
:         mode 
= GDK_COPY
;          break; 
2081         case wxNO_OP
:        mode 
= GDK_NOOP
;          break; 
2082         case wxSRC_INVERT
:   mode 
= GDK_COPY_INVERT
;   break; 
2083         case wxNOR
:          mode 
= GDK_NOR
;           break; 
2085             wxFAIL_MSG("unknown mode"); 
2089     m_logicalFunction 
= function
; 
2091     gdk_gc_set_function( m_penGC
, mode 
); 
2092     gdk_gc_set_function( m_brushGC
, mode 
); 
2094     // to stay compatible with wxMSW, we don't apply ROPs to the text 
2095     // operations (i.e. DrawText/DrawRotatedText). 
2096     // True, but mono-bitmaps use the m_textGC and they use ROPs as well. 
2097     gdk_gc_set_function( m_textGC
, mode 
); 
2100 void wxWindowDCImpl::SetTextForeground( const wxColour 
&col 
) 
2102     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2104     // don't set m_textForegroundColour to an invalid colour as we'd crash 
2105     // later then (we use m_textForegroundColour.GetColor() without checking 
2107     if ( !col
.IsOk() || (m_textForegroundColour 
== col
) ) 
2110     m_textForegroundColour 
= col
; 
2114         m_textForegroundColour
.CalcPixel( m_cmap 
); 
2115         gdk_gc_set_foreground( m_textGC
, m_textForegroundColour
.GetColor() ); 
2119 void wxWindowDCImpl::SetTextBackground( const wxColour 
&col 
) 
2121     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2124     if ( !col
.IsOk() || (m_textBackgroundColour 
== col
) ) 
2127     m_textBackgroundColour 
= col
; 
2131         m_textBackgroundColour
.CalcPixel( m_cmap 
); 
2132         gdk_gc_set_background( m_textGC
, m_textBackgroundColour
.GetColor() ); 
2136 void wxWindowDCImpl::SetBackgroundMode( int mode 
) 
2138     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2140     m_backgroundMode 
= mode
; 
2143 void wxWindowDCImpl::SetPalette( const wxPalette
& WXUNUSED(palette
) ) 
2145     wxFAIL_MSG( wxT("wxWindowDCImpl::SetPalette not implemented") ); 
2148 void wxWindowDCImpl::DoSetClippingRegion( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height 
) 
2150     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2152     if (!m_gdkwindow
) return; 
2155     rect
.x 
= XLOG2DEV(x
); 
2156     rect
.y 
= YLOG2DEV(y
); 
2157     rect
.width 
= XLOG2DEVREL(width
); 
2158     rect
.height 
= YLOG2DEVREL(height
); 
2160     if (m_window 
&& m_window
->m_wxwindow 
&& 
2161         (m_window
->GetLayoutDirection() == wxLayout_RightToLeft
)) 
2163         rect
.x 
-= rect
.width
; 
2166     DoSetDeviceClippingRegion(wxRegion(rect
)); 
2169 void wxWindowDCImpl::DoSetDeviceClippingRegion( const wxRegion 
®ion  
) 
2171     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2175         DestroyClippingRegion(); 
2179     if (!m_gdkwindow
) return; 
2181     if (!m_currentClippingRegion
.IsNull()) 
2182         m_currentClippingRegion
.Intersect( region 
); 
2184         m_currentClippingRegion
.Union( region 
); 
2186 #if USE_PAINT_REGION 
2187     if (!m_paintClippingRegion
.IsNull()) 
2188         m_currentClippingRegion
.Intersect( m_paintClippingRegion 
); 
2191     wxCoord xx
, yy
, ww
, hh
; 
2192     m_currentClippingRegion
.GetBox( xx
, yy
, ww
, hh 
); 
2193     wxGTKDCImpl::DoSetClippingRegion( xx
, yy
, ww
, hh 
); 
2195     GdkRegion
* gdkRegion 
= m_currentClippingRegion
.GetRegion(); 
2196     gdk_gc_set_clip_region(m_penGC
,   gdkRegion
); 
2197     gdk_gc_set_clip_region(m_brushGC
, gdkRegion
); 
2198     gdk_gc_set_clip_region(m_textGC
,  gdkRegion
); 
2199     gdk_gc_set_clip_region(m_bgGC
,    gdkRegion
); 
2202 void wxWindowDCImpl::DestroyClippingRegion() 
2204     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
2206     wxDCImpl::DestroyClippingRegion(); 
2208     m_currentClippingRegion
.Clear(); 
2210 #if USE_PAINT_REGION 
2211     if (!m_paintClippingRegion
.IsEmpty()) 
2212         m_currentClippingRegion
.Union( m_paintClippingRegion 
); 
2215     if (!m_gdkwindow
) return; 
2217     GdkRegion
* gdkRegion 
= NULL
; 
2218     if (!m_currentClippingRegion
.IsEmpty()) 
2219         gdkRegion 
= m_currentClippingRegion
.GetRegion(); 
2221     gdk_gc_set_clip_region(m_penGC
,   gdkRegion
); 
2222     gdk_gc_set_clip_region(m_brushGC
, gdkRegion
); 
2223     gdk_gc_set_clip_region(m_textGC
,  gdkRegion
); 
2224     gdk_gc_set_clip_region(m_bgGC
,    gdkRegion
); 
2227 void wxWindowDCImpl::Destroy() 
2229     if (m_penGC
) wxFreePoolGC( m_penGC 
); 
2231     if (m_brushGC
) wxFreePoolGC( m_brushGC 
); 
2233     if (m_textGC
) wxFreePoolGC( m_textGC 
); 
2235     if (m_bgGC
) wxFreePoolGC( m_bgGC 
); 
2239 void wxWindowDCImpl::SetDeviceOrigin( wxCoord x
, wxCoord y 
) 
2241     m_deviceOriginX 
= x
; 
2242     m_deviceOriginY 
= y
; 
2244     ComputeScaleAndOrigin(); 
2247 void wxWindowDCImpl::SetAxisOrientation( bool xLeftRight
, bool yBottomUp 
) 
2249     m_signX 
= (xLeftRight 
?  1 : -1); 
2250     m_signY 
= (yBottomUp  
? -1 :  1); 
2252     if (m_window 
&& m_window
->m_wxwindow 
&& 
2253         (m_window
->GetLayoutDirection() == wxLayout_RightToLeft
)) 
2256     ComputeScaleAndOrigin(); 
2259 void wxWindowDCImpl::ComputeScaleAndOrigin() 
2261     const wxRealPoint 
origScale(m_scaleX
, m_scaleY
); 
2263     wxDCImpl::ComputeScaleAndOrigin(); 
2265     // if scale has changed call SetPen to recalulate the line width 
2266     if ( wxRealPoint(m_scaleX
, m_scaleY
) != origScale 
&& m_pen
.IsOk() ) 
2268         // this is a bit artificial, but we need to force wxDC to think the pen 
2276 // Resolution in pixels per logical inch 
2277 wxSize 
wxWindowDCImpl::GetPPI() const 
2279     return wxSize( (int) (m_mm_to_pix_x 
* 25.4 + 0.5), (int) (m_mm_to_pix_y 
* 25.4 + 0.5)); 
2282 int wxWindowDCImpl::GetDepth() const 
2284     return gdk_drawable_get_depth(m_gdkwindow
); 
2288 //----------------------------------------------------------------------------- 
2290 //----------------------------------------------------------------------------- 
2292 IMPLEMENT_ABSTRACT_CLASS(wxClientDCImpl
, wxWindowDCImpl
) 
2294 wxClientDCImpl::wxClientDCImpl( wxDC 
*owner 
) 
2295           : wxWindowDCImpl( owner 
) 
2299 wxClientDCImpl::wxClientDCImpl( wxDC 
*owner
, wxWindow 
*win 
) 
2300           : wxWindowDCImpl( owner
, win 
) 
2302     wxCHECK_RET( win
, wxT("NULL window in wxClientDCImpl::wxClientDC") ); 
2304 #ifdef __WXUNIVERSAL__ 
2305     wxPoint ptOrigin 
= win
->GetClientAreaOrigin(); 
2306     SetDeviceOrigin(ptOrigin
.x
, ptOrigin
.y
); 
2307     wxSize size 
= win
->GetClientSize(); 
2308     DoSetClippingRegion(0, 0, size
.x
, size
.y
); 
2313 void wxClientDCImpl::DoGetSize(int *width
, int *height
) const 
2315     wxCHECK_RET( m_window
, wxT("GetSize() doesn't work without window") ); 
2317     m_window
->GetClientSize( width
, height 
); 
2320 //----------------------------------------------------------------------------- 
2322 //----------------------------------------------------------------------------- 
2324 IMPLEMENT_ABSTRACT_CLASS(wxPaintDCImpl
, wxClientDCImpl
) 
2326 // Limit the paint region to the window size. Sometimes 
2327 // the paint region is too big, and this risks X11 errors 
2328 static void wxLimitRegionToSize(wxRegion
& region
, const wxSize
& sz
) 
2330     wxRect originalRect 
= region
.GetBox(); 
2331     wxRect 
rect(originalRect
); 
2332     if (rect
.width 
+ rect
.x 
> sz
.x
) 
2333         rect
.width 
= sz
.x 
- rect
.x
; 
2334     if (rect
.height 
+ rect
.y 
> sz
.y
) 
2335         rect
.height 
= sz
.y 
- rect
.y
; 
2336     if (rect 
!= originalRect
) 
2338         region 
= wxRegion(rect
); 
2339         wxLogTrace(wxT("painting"), wxT("Limiting region from %d, %d, %d, %d to %d, %d, %d, %d\n"), 
2340                    originalRect
.x
, originalRect
.y
, originalRect
.width
, originalRect
.height
, 
2341                    rect
.x
, rect
.y
, rect
.width
, rect
.height
); 
2345 wxPaintDCImpl::wxPaintDCImpl( wxDC 
*owner 
) 
2346          : wxClientDCImpl( owner 
) 
2350 wxPaintDCImpl::wxPaintDCImpl( wxDC 
*owner
, wxWindow 
*win 
) 
2351          : wxClientDCImpl( owner
, win 
) 
2353 #if USE_PAINT_REGION 
2354     if (!win
->m_clipPaintRegion
) 
2357     wxSize sz 
= win
->GetSize(); 
2358     m_paintClippingRegion 
= win
->m_nativeUpdateRegion
; 
2359     wxLimitRegionToSize(m_paintClippingRegion
, sz
); 
2361     GdkRegion 
*region 
= m_paintClippingRegion
.GetRegion(); 
2364         m_currentClippingRegion
.Union( m_paintClippingRegion 
); 
2365         wxLimitRegionToSize(m_currentClippingRegion
, sz
); 
2367         if (sz
.x 
<= 0 || sz
.y 
<= 0) 
2370         gdk_gc_set_clip_region( m_penGC
, region 
); 
2371         gdk_gc_set_clip_region( m_brushGC
, region 
); 
2372         gdk_gc_set_clip_region( m_textGC
, region 
); 
2373         gdk_gc_set_clip_region( m_bgGC
, region 
); 
2378 // ---------------------------------------------------------------------------- 
2380 // ---------------------------------------------------------------------------- 
2382 class wxDCModule 
: public wxModule
 
2389     DECLARE_DYNAMIC_CLASS(wxDCModule
) 
2392 IMPLEMENT_DYNAMIC_CLASS(wxDCModule
, wxModule
) 
2394 bool wxDCModule::OnInit() 
2400 void wxDCModule::OnExit() 
2404     for (int i 
= wxBRUSHSTYLE_LAST_HATCH 
- wxBRUSHSTYLE_FIRST_HATCH
; i
--; ) 
2407             g_object_unref(hatches
[i
]);