1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "dc.h"
16 #include "wx/wxprec.h"
20 #if wxMAC_USE_CORE_GRAPHICS
23 #include "wx/mac/uma.h"
24 #include "wx/dcmemory.h"
25 #include "wx/dcprint.h"
26 #include "wx/region.h"
36 #include "wx/mac/private.h"
37 #include <ATSUnicode.h>
38 #include <TextCommon.h>
39 #include <TextEncodingConverter.h>
41 #include <CGContext.h>
43 #if !USE_SHARED_LIBRARY
44 IMPLEMENT_ABSTRACT_CLASS(wxDC
, wxObject
)
47 //-----------------------------------------------------------------------------
49 //-----------------------------------------------------------------------------
51 #if !defined( __DARWIN__ ) || defined(__MWERKS__)
53 const double M_PI
= 3.14159265358979 ;
56 const double RAD2DEG
= 180.0 / M_PI
;
57 const short kEmulatedMode
= -1 ;
58 const short kUnsupportedMode
= -2 ;
60 extern TECObjectRef s_TECNativeCToUnicode
;
63 // The text ctrl implementation still needs that for the non hiview implementation
65 wxMacWindowClipper::wxMacWindowClipper( const wxWindow
* win
) :
66 wxMacPortSaver( (GrafPtr
) GetWindowPort((WindowRef
) win
->MacGetTopLevelWindowRef()) )
68 m_newPort
=(GrafPtr
) GetWindowPort((WindowRef
) win
->MacGetTopLevelWindowRef()) ;
69 m_formerClip
= NewRgn() ;
70 m_newClip
= NewRgn() ;
71 GetClip( m_formerClip
) ;
76 win
->MacWindowToRootWindow( &x
,&y
) ;
77 // get area including focus rect
78 CopyRgn( (RgnHandle
) ((wxWindow
*)win
)->MacGetVisibleRegion(true).GetWXHRGN() , m_newClip
) ;
79 if ( !EmptyRgn( m_newClip
) )
80 OffsetRgn( m_newClip
, x
, y
) ;
82 SetClip( m_newClip
) ;
86 wxMacWindowClipper::~wxMacWindowClipper()
88 SetPort( m_newPort
) ;
89 SetClip( m_formerClip
) ;
90 DisposeRgn( m_newClip
) ;
91 DisposeRgn( m_formerClip
) ;
94 wxMacWindowStateSaver::wxMacWindowStateSaver( const wxWindow
* win
) :
95 wxMacWindowClipper( win
)
97 // the port is already set at this point
98 m_newPort
=(GrafPtr
) GetWindowPort((WindowRef
) win
->MacGetTopLevelWindowRef()) ;
99 GetThemeDrawingState( &m_themeDrawingState
) ;
102 wxMacWindowStateSaver::~wxMacWindowStateSaver()
104 SetPort( m_newPort
) ;
105 SetThemeDrawingState( m_themeDrawingState
, true ) ;
108 //-----------------------------------------------------------------------------
110 //-----------------------------------------------------------------------------
112 static inline double dmin(double a
, double b
) { return a
< b
? a
: b
; }
113 static inline double dmax(double a
, double b
) { return a
> b
? a
: b
; }
114 static inline double DegToRad(double deg
) { return (deg
* M_PI
) / 180.0; }
116 //-----------------------------------------------------------------------------
117 // device context implementation
119 // more and more of the dc functionality should be implemented by calling
120 // the appropricate wxMacCGContext, but we will have to do that step by step
121 // also coordinate conversions should be moved to native matrix ops
122 //-----------------------------------------------------------------------------
124 wxMacCGPath::wxMacCGPath()
126 m_path
= CGPathCreateMutable() ;
129 wxMacCGPath::~wxMacCGPath()
131 CGPathRelease( m_path
) ;
134 // Starts a new subpath at
135 void wxMacCGPath::MoveToPoint( wxCoord x1
, wxCoord y1
)
137 CGPathMoveToPoint( m_path
, NULL
, x1
, y1
) ;
140 void wxMacCGPath::AddLineToPoint( wxCoord x1
, wxCoord y1
)
142 CGPathAddLineToPoint( m_path
, NULL
, x1
, y1
) ;
145 void wxMacCGPath::AddRectangle( wxCoord x
, wxCoord y
, wxCoord w
, wxCoord h
)
147 CGRect cgRect
= { { x
, y
} , { w
, h
} } ;
148 CGPathAddRect( m_path
, NULL
, cgRect
) ;
151 void wxMacCGPath::AddCircle( wxCoord x
, wxCoord y
, wxCoord r
)
153 CGPathAddArc( m_path
, NULL
, x
, y
, r
, 0.0 , 2 * M_PI
, true ) ;
156 // closes the current subpath
157 void wxMacCGPath::CloseSubpath()
159 CGPathCloseSubpath( m_path
) ;
162 CGPathRef
wxMacCGPath::GetPath() const
168 wxMacCGContext::wxMacCGContext( CGrafPtr port
)
174 wxMacCGContext::wxMacCGContext( CGContextRef cgcontext
)
177 m_cgContext
= cgcontext
;
180 wxMacCGContext::wxMacCGContext()
186 wxMacCGContext::~wxMacCGContext()
189 QDEndCGContext( m_qdPort
, &m_cgContext
) ;
193 void wxMacCGContext::Clip( const wxRegion
®ion
)
195 // ClipCGContextToRegion ( m_cgContext, &bounds , (RgnHandle) dc->m_macCurrentClipRgn ) ;
198 void wxMacCGContext::StrokePath( const wxGraphicPath
*p
)
200 const wxMacCGPath
* path
= dynamic_cast< const wxMacCGPath
*>( p
) ;
201 CGContextBeginPath( m_cgContext
) ;
202 CGContextAddPath( m_cgContext
, path
->GetPath() ) ;
203 CGContextClosePath( m_cgContext
) ;
204 CGContextStrokePath( m_cgContext
) ;
207 void wxMacCGContext::DrawPath( const wxGraphicPath
*p
, int fillStyle
)
209 const wxMacCGPath
* path
= dynamic_cast< const wxMacCGPath
*>( p
) ;
210 CGPathDrawingMode mode
= m_mode
;
211 if ( fillStyle
== wxODDEVEN_RULE
)
213 if ( mode
== kCGPathFill
)
214 mode
= kCGPathEOFill
;
215 else if ( mode
== kCGPathFillStroke
)
216 mode
= kCGPathEOFillStroke
;
218 CGContextBeginPath( m_cgContext
) ;
219 CGContextAddPath( m_cgContext
, path
->GetPath() ) ;
220 CGContextClosePath( m_cgContext
) ;
221 CGContextDrawPath( m_cgContext
, mode
) ;
224 void wxMacCGContext::FillPath( const wxGraphicPath
*p
, const wxColor
&fillColor
, int fillStyle
)
226 const wxMacCGPath
* path
= dynamic_cast< const wxMacCGPath
*>( p
) ;
227 CGContextSaveGState( m_cgContext
) ;
229 RGBColor col
= MAC_WXCOLORREF( fillColor
.GetPixel() ) ;
230 CGContextSetRGBFillColor( m_cgContext
, col
.red
/ 65536.0 , col
.green
/ 65536.0 , col
.blue
/ 65536.0 , 1.0 ) ;
231 CGPathDrawingMode mode
= kCGPathFill
;
233 if ( fillStyle
== wxODDEVEN_RULE
)
234 mode
= kCGPathEOFill
;
236 CGContextBeginPath( m_cgContext
) ;
237 CGContextAddPath( m_cgContext
, path
->GetPath() ) ;
238 CGContextClosePath( m_cgContext
) ;
239 CGContextDrawPath( m_cgContext
, mode
) ;
241 CGContextRestoreGState( m_cgContext
) ;
244 wxGraphicPath
* wxMacCGContext::CreatePath()
246 CGContextRef cg
= GetNativeContext() ;
247 // make sure that we now have a real cgref, before doing
248 // anything with paths
249 return new wxMacCGPath() ;
252 // in case we only got a QDPort only create a cgref now
254 CGContextRef
wxMacCGContext::GetNativeContext()
256 if( m_cgContext
== NULL
)
259 GetPortBounds( (CGrafPtr
) m_qdPort
, &bounds
) ;
260 OSStatus status
= QDBeginCGContext( (CGrafPtr
) m_qdPort
, &m_cgContext
) ;
262 wxASSERT_MSG( status
== noErr
, wxT("Cannot nest wxDCs on the same window") ) ;
263 CGContextTranslateCTM( m_cgContext
, 0 , bounds
.bottom
- bounds
.top
) ;
264 CGContextScaleCTM( m_cgContext
, 1 , -1 ) ;
267 SetBrush( m_brush
) ;
272 void wxMacCGContext::SetNativeContext( CGContextRef cg
)
277 void wxMacCGContext::SetPen( const wxPen
&pen
)
280 if ( m_cgContext
== NULL
)
282 bool fill
= m_brush
.GetStyle() != wxTRANSPARENT
;
283 bool stroke
= pen
.GetStyle() != wxTRANSPARENT
;
286 // we can benchmark performance, should go into a setting later
287 CGContextSetShouldAntialias( m_cgContext
, false ) ;
292 m_mode
= kCGPathFill
; // just a default
296 m_mode
= kCGPathFill
;
300 RGBColor col
= MAC_WXCOLORREF( pen
.GetColour().GetPixel() ) ;
301 CGContextSetRGBStrokeColor( m_cgContext
, col
.red
/ 65536.0 , col
.green
/ 65536.0 , col
.blue
/ 65536.0 , 1.0 ) ;
304 switch( pen
.GetCap() )
307 cap
= kCGLineCapRound
;
309 case wxCAP_PROJECTING
:
310 cap
= kCGLineCapSquare
;
313 cap
= kCGLineCapButt
;
316 cap
= kCGLineCapButt
;
319 CGContextSetLineCap( m_cgContext
, cap
) ;
322 switch( pen
.GetJoin() )
325 join
= kCGLineJoinBevel
;
328 join
= kCGLineJoinMiter
;
331 join
= kCGLineJoinRound
;
334 join
= kCGLineJoinMiter
;
337 CGContextSetLineJoin( m_cgContext
, join
) ;
339 CGContextSetLineWidth( m_cgContext
, pen
.GetWidth() == 0 ? 0.1 : pen
.GetWidth() /* TODO * m_dc->m_scaleX */ ) ;
341 m_mode
= kCGPathStroke
;
343 const float *lengths
= NULL
;
344 float *userLengths
= NULL
;
346 const float dotted
[] = { 3 , 3 };
347 const float dashed
[] = { 19 , 9 };
348 const float short_dashed
[] = { 9 , 6 };
349 const float dotted_dashed
[] = { 9 , 6 , 3 , 3 };
351 switch( pen
.GetStyle() )
357 count
= WXSIZEOF(dotted
);
361 count
= WXSIZEOF(dashed
) ;
364 lengths
= short_dashed
;
365 count
= WXSIZEOF(short_dashed
) ;
368 lengths
= dotted_dashed
;
369 count
= WXSIZEOF(dotted_dashed
);
373 count
= pen
.GetDashes( &dashes
) ;
376 userLengths
= new float[count
] ;
377 for( int i
= 0 ; i
< count
; ++i
)
378 userLengths
[i
] = dashes
[i
] ;
380 lengths
= userLengths
;
386 CGContextSetLineDash( m_cgContext
, 0 , lengths
, count
) ;
387 delete[] userLengths
;
388 // we need to change the cap, otherwise everything overlaps
389 // and we get solid lines
391 CGContextSetLineCap( m_cgContext
, kCGLineCapButt
) ;
393 if ( fill
&& stroke
)
395 m_mode
= kCGPathFillStroke
;
400 void wxMacCGContext::SetBrush( const wxBrush
&brush
)
403 if ( m_cgContext
== NULL
)
406 bool fill
= brush
.GetStyle() != wxTRANSPARENT
;
407 bool stroke
= m_pen
.GetStyle() != wxTRANSPARENT
;
410 // we can benchmark performance, should go into a setting later
411 CGContextSetShouldAntialias( m_cgContext
, false ) ;
417 m_mode
= kCGPathFill
; // just a default
421 RGBColor col
= MAC_WXCOLORREF( brush
.GetColour().GetPixel() ) ;
422 CGContextSetRGBFillColor( m_cgContext
, col
.red
/ 65536.0 , col
.green
/ 65536.0 , col
.blue
/ 65536.0 , 1.0 ) ;
423 m_mode
= kCGPathFill
;
427 m_mode
= kCGPathStroke
;
429 if ( fill
&& stroke
)
431 m_mode
= kCGPathFillStroke
;
437 // snippets from Sketch Sample from Apple :
439 #define kGenericRGBProfilePathStr "/System/Library/ColorSync/Profiles/Generic RGB Profile.icc"
441 This function locates, opens, and returns the profile reference for the calibrated
442 Generic RGB color space. It is up to the caller to call CMCloseProfile when done
443 with the profile reference this function returns.
445 CMProfileRef
wxMacOpenGenericProfile(void)
447 static CMProfileRef cachedRGBProfileRef
= NULL
;
449 // we only create the profile reference once
450 if (cachedRGBProfileRef
== NULL
)
452 CMProfileLocation loc
;
454 loc
.locType
= cmPathBasedProfile
;
455 strcpy(loc
.u
.pathLoc
.path
, kGenericRGBProfilePathStr
);
457 verify_noerr( CMOpenProfile(&cachedRGBProfileRef
, &loc
) );
460 if (cachedRGBProfileRef
)
462 // clone the profile reference so that the caller has their own reference, not our cached one
463 CMCloneProfileRef(cachedRGBProfileRef
);
466 return cachedRGBProfileRef
;
470 Return the generic RGB color space. This is a 'get' function and the caller should
471 not release the returned value unless the caller retains it first. Usually callers
472 of this routine will immediately use the returned colorspace with CoreGraphics
473 so they typically do not need to retain it themselves.
475 This function creates the generic RGB color space once and hangs onto it so it can
476 return it whenever this function is called.
479 CGColorSpaceRef
wxMacGetGenericRGBColorSpace()
481 static CGColorSpaceRef genericRGBColorSpace
= NULL
;
483 if (genericRGBColorSpace
== NULL
)
485 CMProfileRef genericRGBProfile
= wxMacOpenGenericProfile();
487 if (genericRGBProfile
)
489 genericRGBColorSpace
= CGColorSpaceCreateWithPlatformColorSpace(genericRGBProfile
);
490 wxASSERT_MSG( genericRGBColorSpace
!= NULL
, wxT("couldn't create the generic RGB color space") ) ;
492 // we opened the profile so it is up to us to close it
493 CMCloseProfile(genericRGBProfile
);
496 return genericRGBColorSpace
;
499 void AddEllipticArcToPath(CGContextRef c
, CGPoint center
, float a
, float b
, float fromDegree
, float toDegree
)
501 CGContextSaveGState(c
);
502 CGContextTranslateCTM(c
, center
.x
, center
.y
);
503 CGContextScaleCTM(c
, a
, b
);
504 CGContextMoveToPoint(c
, 1, 0);
505 CGContextAddArc(c
, 0, 0, 1, DegToRad(fromDegree
), DegToRad(toDegree
), 0);
506 CGContextClosePath(c
);
507 CGContextRestoreGState(c
);
510 void AddRoundedRectToPath(CGContextRef c
, CGRect rect
, float ovalWidth
,
514 if (ovalWidth
== 0 || ovalHeight
== 0)
516 CGContextAddRect(c
, rect
);
519 CGContextSaveGState(c
);
520 CGContextTranslateCTM(c
, CGRectGetMinX(rect
), CGRectGetMinY(rect
));
521 CGContextScaleCTM(c
, ovalWidth
, ovalHeight
);
522 fw
= CGRectGetWidth(rect
) / ovalWidth
;
523 fh
= CGRectGetHeight(rect
) / ovalHeight
;
524 CGContextMoveToPoint(c
, fw
, fh
/2);
525 CGContextAddArcToPoint(c
, fw
, fh
, fw
/2, fh
, 1);
526 CGContextAddArcToPoint(c
, 0, fh
, 0, fh
/2, 1);
527 CGContextAddArcToPoint(c
, 0, 0, fw
/2, 0, 1);
528 CGContextAddArcToPoint(c
, fw
, 0, fw
, fh
/2, 1);
529 CGContextClosePath(c
);
530 CGContextRestoreGState(c
);
537 m_mm_to_pix_x
= mm2pt
;
538 m_mm_to_pix_y
= mm2pt
;
539 m_internalDeviceOriginX
= 0;
540 m_internalDeviceOriginY
= 0;
541 m_externalDeviceOriginX
= 0;
542 m_externalDeviceOriginY
= 0;
543 m_logicalScaleX
= 1.0;
544 m_logicalScaleY
= 1.0;
549 m_needComputeScaleX
= FALSE
;
550 m_needComputeScaleY
= FALSE
;
554 m_macLocalOrigin
.x
= m_macLocalOrigin
.y
= 0 ;
556 m_pen
= *wxBLACK_PEN
;
557 m_font
= *wxNORMAL_FONT
;
558 m_brush
= *wxWHITE_BRUSH
;
560 m_macATSUIStyle
= NULL
;
562 m_graphicContext
= NULL
;
567 if( m_macATSUIStyle
)
569 ::ATSUDisposeStyle((ATSUStyle
)m_macATSUIStyle
);
570 m_macATSUIStyle
= NULL
;
573 delete m_graphicContext
;
576 void wxDC::DoDrawBitmap( const wxBitmap
&bmp
, wxCoord x
, wxCoord y
, bool useMask
)
578 wxCHECK_RET( Ok(), wxT("invalid window dc") );
579 wxCHECK_RET( bmp
.Ok(), wxT("invalid bitmap") );
580 wxCoord xx
= XLOG2DEVMAC(x
);
581 wxCoord yy
= YLOG2DEVMAC(y
);
582 wxCoord w
= bmp
.GetWidth();
583 wxCoord h
= bmp
.GetHeight();
584 wxCoord ww
= XLOG2DEVREL(w
);
585 wxCoord hh
= YLOG2DEVREL(h
);
587 CGContextRef cg
= dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() ;
588 CGImageRef image
= (CGImageRef
)( bmp
.CGImageCreate() ) ;
589 HIRect r
= CGRectMake( xx
, yy
, ww
, hh
) ;
590 HIViewDrawCGImage( cg
, &r
, image
) ;
591 CGImageRelease( image
) ;
594 void wxDC::DoDrawIcon( const wxIcon
&icon
, wxCoord x
, wxCoord y
)
596 wxCHECK_RET(Ok(), wxT("Invalid dc wxDC::DoDrawIcon"));
597 wxCHECK_RET(icon
.Ok(), wxT("Invalid icon wxDC::DoDrawIcon"));
599 wxCoord xx
= XLOG2DEVMAC(x
);
600 wxCoord yy
= YLOG2DEVMAC(y
);
601 wxCoord w
= icon
.GetWidth();
602 wxCoord h
= icon
.GetHeight();
603 wxCoord ww
= XLOG2DEVREL(w
);
604 wxCoord hh
= YLOG2DEVREL(h
);
606 CGContextRef cg
= dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() ;
607 CGRect r
= CGRectMake( 00 , 00 , ww
, hh
) ;
608 CGContextSaveGState(cg
);
609 CGContextTranslateCTM(cg
, xx
, yy
+ hh
);
610 CGContextScaleCTM(cg
, 1, -1);
611 PlotIconRefInContext( cg
, &r
, kAlignNone
, kTransformNone
,
612 NULL
, kPlotIconRefNormalFlags
, MAC_WXHICON( icon
.GetHICON() ) ) ;
613 CGContextRestoreGState( cg
) ;
616 void wxDC::DoSetClippingRegion( wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
)
618 wxCHECK_RET(Ok(), wxT("wxDC::DoSetClippingRegion Invalid DC"));
619 wxCoord xx
, yy
, ww
, hh
;
622 ww
= XLOG2DEVREL(width
);
623 hh
= YLOG2DEVREL(height
);
624 // SetRectRgn( (RgnHandle) m_macCurrentClipRgn , xx , yy , xx + ww , yy + hh ) ;
625 // SectRgn( (RgnHandle) m_macCurrentClipRgn , (RgnHandle) m_macBoundaryClipRgn , (RgnHandle) m_macCurrentClipRgn ) ;
628 m_clipX1
= wxMax( m_clipX1
, xx
);
629 m_clipY1
= wxMax( m_clipY1
, yy
);
630 m_clipX2
= wxMin( m_clipX2
, (xx
+ ww
));
631 m_clipY2
= wxMin( m_clipY2
, (yy
+ hh
));
641 // TODO as soon as we don't reset the context for each operation anymore
642 // we have to update the context as well
645 void wxDC::DoSetClippingRegionAsRegion( const wxRegion
®ion
)
647 wxCHECK_RET( Ok(), wxT("invalid window dc") ) ;
650 DestroyClippingRegion();
654 region
.GetBox( x
, y
, w
, h
);
655 wxCoord xx
, yy
, ww
, hh
;
660 // if we have a scaling that we cannot map onto native regions
661 // we must use the box
662 if ( ww
!= w
|| hh
!= h
)
664 wxDC::DoSetClippingRegion( x
, y
, w
, h
);
669 CopyRgn( (RgnHandle) region.GetWXHRGN() , (RgnHandle) m_macCurrentClipRgn ) ;
670 if ( xx != x || yy != y )
672 OffsetRgn( (RgnHandle) m_macCurrentClipRgn , xx - x , yy - y ) ;
674 SectRgn( (RgnHandle) m_macCurrentClipRgn , (RgnHandle) m_macBoundaryClipRgn , (RgnHandle) m_macCurrentClipRgn ) ;
678 m_clipX1
= wxMax( m_clipX1
, xx
);
679 m_clipY1
= wxMax( m_clipY1
, yy
);
680 m_clipX2
= wxMin( m_clipX2
, (xx
+ ww
));
681 m_clipY2
= wxMin( m_clipY2
, (yy
+ hh
));
694 void wxDC::DestroyClippingRegion()
696 // CopyRgn( (RgnHandle) m_macBoundaryClipRgn , (RgnHandle) m_macCurrentClipRgn ) ;
700 void wxDC::DoGetSizeMM( int* width
, int* height
) const
705 *width
= long( double(w
) / (m_scaleX
*m_mm_to_pix_x
) );
706 *height
= long( double(h
) / (m_scaleY
*m_mm_to_pix_y
) );
709 void wxDC::SetTextForeground( const wxColour
&col
)
711 wxCHECK_RET(Ok(), wxT("Invalid DC"));
712 if ( col
!= m_textForegroundColour
)
714 m_textForegroundColour
= col
;
719 void wxDC::SetTextBackground( const wxColour
&col
)
721 wxCHECK_RET(Ok(), wxT("Invalid DC"));
722 m_textBackgroundColour
= col
;
725 void wxDC::SetMapMode( int mode
)
730 SetLogicalScale( twips2mm
*m_mm_to_pix_x
, twips2mm
*m_mm_to_pix_y
);
733 SetLogicalScale( pt2mm
*m_mm_to_pix_x
, pt2mm
*m_mm_to_pix_y
);
736 SetLogicalScale( m_mm_to_pix_x
, m_mm_to_pix_y
);
739 SetLogicalScale( m_mm_to_pix_x
/10.0, m_mm_to_pix_y
/10.0 );
743 SetLogicalScale( 1.0, 1.0 );
746 if (mode
!= wxMM_TEXT
)
748 m_needComputeScaleX
= TRUE
;
749 m_needComputeScaleY
= TRUE
;
753 void wxDC::SetUserScale( double x
, double y
)
755 // allow negative ? -> no
758 ComputeScaleAndOrigin();
761 void wxDC::SetLogicalScale( double x
, double y
)
766 ComputeScaleAndOrigin();
769 void wxDC::SetLogicalOrigin( wxCoord x
, wxCoord y
)
771 m_logicalOriginX
= x
* m_signX
; // is this still correct ?
772 m_logicalOriginY
= y
* m_signY
;
773 ComputeScaleAndOrigin();
776 void wxDC::SetDeviceOrigin( wxCoord x
, wxCoord y
)
778 m_externalDeviceOriginX
= x
;
779 m_externalDeviceOriginY
= y
;
780 ComputeScaleAndOrigin();
783 void wxDC::SetAxisOrientation( bool xLeftRight
, bool yBottomUp
)
785 m_signX
= (xLeftRight
? 1 : -1);
786 m_signY
= (yBottomUp
? -1 : 1);
787 ComputeScaleAndOrigin();
790 wxSize
wxDC::GetPPI() const
792 return wxSize(72, 72);
795 int wxDC::GetDepth() const
800 void wxDC::ComputeScaleAndOrigin()
802 // CMB: copy scale to see if it changes
803 double origScaleX
= m_scaleX
;
804 double origScaleY
= m_scaleY
;
805 m_scaleX
= m_logicalScaleX
* m_userScaleX
;
806 m_scaleY
= m_logicalScaleY
* m_userScaleY
;
807 m_deviceOriginX
= m_internalDeviceOriginX
+ m_externalDeviceOriginX
;
808 m_deviceOriginY
= m_internalDeviceOriginY
+ m_externalDeviceOriginY
;
809 // CMB: if scale has changed call SetPen to recalulate the line width
810 if (m_scaleX
!= origScaleX
|| m_scaleY
!= origScaleY
)
812 // this is a bit artificial, but we need to force wxDC to think
813 // the pen has changed
820 void wxDC::SetPalette( const wxPalette
& palette
)
824 void wxDC::SetBackgroundMode( int mode
)
826 m_backgroundMode
= mode
;
829 void wxDC::SetFont( const wxFont
&font
)
835 void wxDC::SetPen( const wxPen
&pen
)
840 if ( m_graphicContext
)
842 m_graphicContext
->SetPen( m_pen
) ;
846 void wxDC::SetBrush( const wxBrush
&brush
)
848 if (m_brush
== brush
)
851 if ( m_graphicContext
)
853 m_graphicContext
->SetBrush( m_brush
) ;
857 void wxDC::SetBackground( const wxBrush
&brush
)
859 if (m_backgroundBrush
== brush
)
861 m_backgroundBrush
= brush
;
862 if (!m_backgroundBrush
.Ok())
866 void wxDC::SetLogicalFunction( int function
)
868 if (m_logicalFunction
== function
)
870 m_logicalFunction
= function
;
873 extern bool wxDoFloodFill(wxDC
*dc
, wxCoord x
, wxCoord y
,
874 const wxColour
& col
, int style
);
876 bool wxDC::DoFloodFill(wxCoord x
, wxCoord y
,
877 const wxColour
& col
, int style
)
879 return wxDoFloodFill(this, x
, y
, col
, style
);
882 bool wxDC::DoGetPixel( wxCoord x
, wxCoord y
, wxColour
*col
) const
884 wxCHECK_MSG( Ok(), false, wxT("wxDC::DoGetPixel Invalid DC") );
885 wxFAIL_MSG( wxT("GetPixel not implemented on Core Graphics") ) ;
889 void wxDC::DoDrawLine( wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2
)
891 wxCHECK_RET(Ok(), wxT("Invalid DC"));
893 wxCoord xx1
= XLOG2DEVMAC(x1
) ;
894 wxCoord yy1
= YLOG2DEVMAC(y1
) ;
895 wxCoord xx2
= XLOG2DEVMAC(x2
) ;
896 wxCoord yy2
= YLOG2DEVMAC(y2
) ;
898 wxGraphicPath
* path
= m_graphicContext
->CreatePath() ;
899 path
->MoveToPoint( xx1
, yy1
) ;
900 path
->AddLineToPoint( xx2
, yy2
) ;
901 path
->CloseSubpath() ;
902 m_graphicContext
->StrokePath( path
) ;
905 CalcBoundingBox(x1
, y1
);
906 CalcBoundingBox(x2
, y2
);
909 void wxDC::DoCrossHair( wxCoord x
, wxCoord y
)
911 wxCHECK_RET( Ok(), wxT("wxDC::DoCrossHair Invalid window dc") );
917 wxCoord xx
= XLOG2DEVMAC(x
);
918 wxCoord yy
= YLOG2DEVMAC(y
);
920 wxGraphicPath
* path
= m_graphicContext
->CreatePath() ;
921 path
->MoveToPoint( XLOG2DEVMAC(0), yy
) ;
922 path
->AddLineToPoint( XLOG2DEVMAC(w
), yy
) ;
923 path
->CloseSubpath() ;
924 path
->MoveToPoint( xx
, YLOG2DEVMAC(0) ) ;
925 path
->AddLineToPoint( xx
, YLOG2DEVMAC(h
) ) ;
926 path
->CloseSubpath() ;
927 m_graphicContext
->StrokePath( path
) ;
930 CalcBoundingBox(x
, y
);
931 CalcBoundingBox(x
+w
, y
+h
);
935 * To draw arcs properly the angles need to be converted from the WX style:
936 * Angles start on the +ve X axis and go anti-clockwise (As you would draw on
937 * a normal axis on paper).
940 * Angles start on the +ve y axis and go clockwise.
943 static double wxConvertWXangleToMACangle(double angle
)
945 double newAngle
= 90 - angle
;
951 void wxDC::DoDrawArc( wxCoord x1
, wxCoord y1
,
952 wxCoord x2
, wxCoord y2
,
953 wxCoord xc
, wxCoord yc
)
955 wxCHECK_RET(Ok(), wxT("wxDC::DoDrawArc Invalid DC"));
956 wxCoord xx1
= XLOG2DEVMAC(x1
);
957 wxCoord yy1
= YLOG2DEVMAC(y1
);
958 wxCoord xx2
= XLOG2DEVMAC(x2
);
959 wxCoord yy2
= YLOG2DEVMAC(y2
);
960 wxCoord xxc
= XLOG2DEVMAC(xc
);
961 wxCoord yyc
= YLOG2DEVMAC(yc
);
962 double dx
= xx1
- xxc
;
963 double dy
= yy1
- yyc
;
964 double radius
= sqrt((double)(dx
*dx
+dy
*dy
));
965 wxCoord rad
= (wxCoord
)radius
;
966 double radius1
, radius2
;
967 if (xx1
== xx2
&& yy1
== yy2
)
972 else if (radius
== 0.0)
974 radius1
= radius2
= 0.0;
978 radius1
= (xx1
- xxc
== 0) ?
979 (yy1
- yyc
< 0) ? 90.0 : -90.0 :
980 -atan2(double(yy1
-yyc
), double(xx1
-xxc
)) * RAD2DEG
;
981 radius2
= (xx2
- xxc
== 0) ?
982 (yy2
- yyc
< 0) ? 90.0 : -90.0 :
983 -atan2(double(yy2
-yyc
), double(xx2
-xxc
)) * RAD2DEG
;
985 wxCoord alpha2
= wxCoord(radius2
- radius1
);
986 wxCoord alpha1
= wxCoord(wxConvertWXangleToMACangle(radius1
));
987 if( (xx1
> xx2
) || (yy1
> yy2
) ) {
990 Rect r
= { yyc
- rad
, xxc
- rad
, yyc
+ rad
, xxc
+ rad
};
991 wxMacCGContext
* mctx
= ((wxMacCGContext
*) m_graphicContext
) ;
992 CGContextRef ctx
= mctx
->GetNativeContext() ;
993 AddEllipticArcToPath( ctx
, CGPointMake( xxc
, yyc
) , rad
, rad
, 0 , 180 ) ;
994 CGContextDrawPath( ctx
, mctx
->GetDrawingMode() ) ;
997 void wxDC::DoDrawEllipticArc( wxCoord x
, wxCoord y
, wxCoord w
, wxCoord h
,
998 double sa
, double ea
)
1000 wxCHECK_RET(Ok(), wxT("wxDC::DoDrawEllepticArc Invalid DC"));
1002 double angle
= sa
- ea
; // Order important Mac in opposite direction to wx
1003 // we have to make sure that the filling is always counter-clockwise
1006 wxCoord xx
= XLOG2DEVMAC(x
);
1007 wxCoord yy
= YLOG2DEVMAC(y
);
1008 wxCoord ww
= m_signX
* XLOG2DEVREL(w
);
1009 wxCoord hh
= m_signY
* YLOG2DEVREL(h
);
1010 // handle -ve width and/or height
1011 if (ww
< 0) { ww
= -ww
; xx
= xx
- ww
; }
1012 if (hh
< 0) { hh
= -hh
; yy
= yy
- hh
; }
1013 sa
= wxConvertWXangleToMACangle(sa
);
1014 wxMacCGContext
* mctx
= ((wxMacCGContext
*) m_graphicContext
) ;
1015 CGContextRef ctx
= mctx
->GetNativeContext() ;
1016 AddEllipticArcToPath( ctx
, CGPointMake( xx
+ ww
/ 2 , yy
+ hh
/ 2 ) , ww
/ 2 , hh
/ 2 , sa
, angle
) ;
1017 CGContextDrawPath( ctx
, mctx
->GetDrawingMode() ) ;
1020 void wxDC::DoDrawPoint( wxCoord x
, wxCoord y
)
1022 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1023 DoDrawLine( x
, y
, x
+ 1 , y
+ 1 ) ;
1026 void wxDC::DoDrawLines(int n
, wxPoint points
[],
1027 wxCoord xoffset
, wxCoord yoffset
)
1029 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1031 wxCoord x1
, x2
, y1
, y2
;
1032 x1
= XLOG2DEVMAC(points
[0].x
+ xoffset
);
1033 y1
= YLOG2DEVMAC(points
[0].y
+ yoffset
);
1034 wxGraphicPath
* path
= m_graphicContext
->CreatePath() ;
1035 path
->MoveToPoint( x1
, y1
) ;
1036 for (int i
= 1; i
< n
; i
++)
1038 x2
= XLOG2DEVMAC(points
[i
].x
+ xoffset
);
1039 y2
= YLOG2DEVMAC(points
[i
].y
+ yoffset
);
1041 path
->AddLineToPoint( x2
, y2
) ;
1043 m_graphicContext
->StrokePath( path
) ;
1047 void wxDC::DoDrawPolygon(int n
, wxPoint points
[],
1048 wxCoord xoffset
, wxCoord yoffset
,
1051 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1052 wxCoord x1
, x2
, y1
, y2
;
1053 if ( n
== 0 || (m_brush
.GetStyle() == wxTRANSPARENT
&& m_pen
.GetStyle() == wxTRANSPARENT
) )
1056 x2
= x1
= XLOG2DEVMAC(points
[0].x
+ xoffset
);
1057 y2
= y1
= YLOG2DEVMAC(points
[0].y
+ yoffset
);
1059 wxGraphicPath
* path
= m_graphicContext
->CreatePath() ;
1060 path
->MoveToPoint( x1
, y1
) ;
1061 for (int i
= 1; i
< n
; i
++)
1063 x2
= XLOG2DEVMAC(points
[i
].x
+ xoffset
);
1064 y2
= YLOG2DEVMAC(points
[i
].y
+ yoffset
);
1066 path
->AddLineToPoint( x2
, y2
) ;
1068 if ( x1
!= x2
|| y1
!= y2
)
1070 path
->AddLineToPoint( x1
,y1
) ;
1072 path
->CloseSubpath() ;
1073 m_graphicContext
->DrawPath( path
, fillStyle
) ;
1077 void wxDC::DoDrawRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
)
1079 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1080 wxCoord xx
= XLOG2DEVMAC(x
);
1081 wxCoord yy
= YLOG2DEVMAC(y
);
1082 wxCoord ww
= m_signX
* XLOG2DEVREL(width
);
1083 wxCoord hh
= m_signY
* YLOG2DEVREL(height
);
1084 // CMB: draw nothing if transformed w or h is 0
1085 if (ww
== 0 || hh
== 0)
1087 // CMB: handle -ve width and/or height
1098 wxGraphicPath
* path
= m_graphicContext
->CreatePath() ;
1099 path
->AddRectangle(xx
,yy
, ww
, hh
) ;
1100 path
->CloseSubpath() ;
1101 m_graphicContext
->DrawPath( path
) ;
1105 void wxDC::DoDrawRoundedRectangle(wxCoord x
, wxCoord y
,
1106 wxCoord width
, wxCoord height
,
1109 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1111 radius
= - radius
* ((width
< height
) ? width
: height
);
1112 wxCoord xx
= XLOG2DEVMAC(x
);
1113 wxCoord yy
= YLOG2DEVMAC(y
);
1114 wxCoord ww
= m_signX
* XLOG2DEVREL(width
);
1115 wxCoord hh
= m_signY
* YLOG2DEVREL(height
);
1116 // CMB: draw nothing if transformed w or h is 0
1117 if (ww
== 0 || hh
== 0)
1119 // CMB: handle -ve width and/or height
1130 Rect rect
= { yy
, xx
, yy
+ hh
, xx
+ ww
} ;
1131 wxMacCGContext
* mctx
= ((wxMacCGContext
*) m_graphicContext
) ;
1132 CGContextRef ctx
= mctx
->GetNativeContext() ;
1133 AddRoundedRectToPath( ctx
, CGRectMake( xx
, yy
, ww
, hh
) , 16 ,16 ) ;
1134 CGContextDrawPath( ctx
, mctx
->GetDrawingMode() ) ;
1137 void wxDC::DoDrawEllipse(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
)
1139 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1140 wxCoord xx
= XLOG2DEVMAC(x
);
1141 wxCoord yy
= YLOG2DEVMAC(y
);
1142 wxCoord ww
= m_signX
* XLOG2DEVREL(width
);
1143 wxCoord hh
= m_signY
* YLOG2DEVREL(height
);
1144 // CMB: draw nothing if transformed w or h is 0
1145 if (ww
== 0 || hh
== 0)
1147 // CMB: handle -ve width and/or height
1159 wxMacCGContext
* mctx
= ((wxMacCGContext
*) m_graphicContext
) ;
1160 CGContextRef ctx
= mctx
->GetNativeContext() ;
1161 if ( width
== height
)
1163 CGContextBeginPath(ctx
);
1164 CGContextAddArc(ctx
,
1171 CGContextClosePath(ctx
);
1173 CGContextDrawPath( ctx
, kCGPathFillStroke
) ;
1177 AddEllipticArcToPath( ctx
, CGPointMake( xx
+ ww
/ 2 , yy
+ hh
/ 2 ) , ww
/ 2 , hh
/ 2 , 0 , 360) ;
1178 CGContextDrawPath( ctx
, mctx
->GetDrawingMode() ) ;
1182 bool wxDC::CanDrawBitmap(void) const
1187 bool wxDC::DoBlit(wxCoord xdest
, wxCoord ydest
, wxCoord width
, wxCoord height
,
1188 wxDC
*source
, wxCoord xsrc
, wxCoord ysrc
, int logical_func
, bool useMask
,
1189 wxCoord xsrcMask
, wxCoord ysrcMask
)
1191 wxCHECK_MSG(Ok(), false, wxT("wxDC::DoBlit Illegal dc"));
1192 wxCHECK_MSG(source
->Ok(), false, wxT("wxDC::DoBlit Illegal source DC"));
1193 if ( logical_func
== wxNO_OP
)
1195 if (xsrcMask
== -1 && ysrcMask
== -1)
1197 xsrcMask
= xsrc
; ysrcMask
= ysrc
;
1200 wxMemoryDC
* memdc
= dynamic_cast<wxMemoryDC
*>(source
) ;
1203 wxBitmap blit
= memdc
->GetSelectedObject() ;
1204 wxASSERT_MSG( blit
.Ok() , wxT("Invalid bitmap for blitting") ) ;
1206 wxCoord xxdest
= XLOG2DEVMAC(xdest
);
1207 wxCoord yydest
= YLOG2DEVMAC(ydest
);
1208 wxCoord ww
= XLOG2DEVREL(width
);
1209 wxCoord hh
= YLOG2DEVREL(height
);
1211 wxCoord bmpwidth
= blit
.GetWidth();
1212 wxCoord bmpheight
= blit
.GetHeight();
1214 if ( xsrc
!= 0 || ysrc
!= 0 || bmpwidth
!= width
|| bmpheight
!= height
)
1216 wxRect
subrect( xsrc
, ysrc
, width
, height
) ;
1217 blit
= blit
.GetSubBitmap( subrect
) ;
1220 CGContextRef cg
= dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() ;
1221 CGImageRef image
= (CGImageRef
)( blit
.CGImageCreate() ) ;
1222 HIRect r
= CGRectMake( xxdest
, yydest
, ww
, hh
) ;
1223 HIViewDrawCGImage( cg
, &r
, image
) ;
1224 CGImageRelease( image
) ;
1229 wxFAIL_MSG( wxT("Blitting is only supported from bitmap contexts") ) ;
1234 void wxDC::DoDrawRotatedText(const wxString
& str
, wxCoord x
, wxCoord y
,
1237 wxCHECK_RET( Ok(), wxT("wxDC::DoDrawRotatedText Invalid window dc") );
1239 if ( str
.Length() == 0 )
1242 wxCHECK_RET( m_macATSUIStyle
!= NULL
, wxT("No valid font set") ) ;
1244 OSStatus status
= noErr
;
1245 ATSUTextLayout atsuLayout
;
1246 UniCharCount chars
= str
.Length() ;
1247 UniChar
* ubuf
= NULL
;
1248 #if SIZEOF_WCHAR_T == 4
1249 wxMBConvUTF16BE converter
;
1251 size_t unicharlen
= converter
.WC2MB( NULL
, str
.wc_str() , 0 ) ;
1252 ubuf
= (UniChar
*) malloc( unicharlen
+ 2 ) ;
1253 converter
.WC2MB( (char*) ubuf
, str
.wc_str(), unicharlen
+ 2 ) ;
1255 const wxWCharBuffer wchar
= str
.wc_str( wxConvLocal
) ;
1256 size_t unicharlen
= converter
.WC2MB( NULL
, wchar
.data() , 0 ) ;
1257 ubuf
= (UniChar
*) malloc( unicharlen
+ 2 ) ;
1258 converter
.WC2MB( (char*) ubuf
, wchar
.data() , unicharlen
+ 2 ) ;
1260 chars
= unicharlen
/ 2 ;
1263 ubuf
= (UniChar
*) str
.wc_str() ;
1265 wxWCharBuffer wchar
= str
.wc_str( wxConvLocal
) ;
1266 chars
= wxWcslen( wchar
.data() ) ;
1267 ubuf
= (UniChar
*) wchar
.data() ;
1271 int drawX
= XLOG2DEVMAC(x
) ;
1272 int drawY
= YLOG2DEVMAC(y
) ;
1274 status
= ::ATSUCreateTextLayoutWithTextPtr( (UniCharArrayPtr
) ubuf
, 0 , chars
, chars
, 1 ,
1275 &chars
, (ATSUStyle
*) &m_macATSUIStyle
, &atsuLayout
) ;
1277 wxASSERT_MSG( status
== noErr
, wxT("couldn't create the layout of the rotated text") );
1278 int iAngle
= int( angle
);
1282 if ( abs(iAngle
) > 0 )
1284 Fixed atsuAngle
= IntToFixed( iAngle
) ;
1285 ATSUAttributeTag atsuTags
[] =
1287 kATSULineRotationTag
,
1289 ByteCount atsuSizes
[sizeof(atsuTags
)/sizeof(ATSUAttributeTag
)] =
1293 ATSUAttributeValuePtr atsuValues
[sizeof(atsuTags
)/sizeof(ATSUAttributeTag
)] =
1297 status
= ::ATSUSetLayoutControls(atsuLayout
, sizeof(atsuTags
)/sizeof(ATSUAttributeTag
),
1298 atsuTags
, atsuSizes
, atsuValues
) ;
1301 CGContextRef cgContext
= dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() ;
1302 ATSUAttributeTag atsuTags
[] =
1306 ByteCount atsuSizes
[sizeof(atsuTags
)/sizeof(ATSUAttributeTag
)] =
1308 sizeof( CGContextRef
) ,
1310 ATSUAttributeValuePtr atsuValues
[sizeof(atsuTags
)/sizeof(ATSUAttributeTag
)] =
1314 status
= ::ATSUSetLayoutControls(atsuLayout
, sizeof(atsuTags
)/sizeof(ATSUAttributeTag
),
1315 atsuTags
, atsuSizes
, atsuValues
) ;
1318 ATSUTextMeasurement textBefore
;
1319 ATSUTextMeasurement textAfter
;
1320 ATSUTextMeasurement ascent
;
1321 ATSUTextMeasurement descent
;
1323 status
= ::ATSUGetUnjustifiedBounds( atsuLayout
, kATSUFromTextBeginning
, kATSUToTextEnd
,
1324 &textBefore
, &textAfter
, &ascent
, &descent
);
1325 wxASSERT_MSG( status
== noErr
, wxT("couldn't measure the rotated text") );
1329 if ( m_backgroundMode
== wxSOLID
)
1331 wxGraphicPath
* path
= m_graphicContext
->CreatePath() ;
1335 path
->AddLineToPoint(
1336 drawX
+ sin(angle
/RAD2DEG
) * FixedToInt(ascent
+ descent
) ,
1337 drawY
+ cos(angle
/RAD2DEG
) * FixedToInt(ascent
+ descent
) ) ;
1338 path
->AddLineToPoint(
1339 drawX
+ sin(angle
/RAD2DEG
) * FixedToInt(ascent
+ descent
) + cos(angle
/RAD2DEG
) * FixedToInt(textAfter
) ,
1340 drawY
+ cos(angle
/RAD2DEG
) * FixedToInt(ascent
+ descent
) - sin(angle
/RAD2DEG
) * FixedToInt(textAfter
) ) ;
1341 path
->AddLineToPoint(
1342 drawX
+ cos(angle
/RAD2DEG
) * FixedToInt(textAfter
) ,
1343 drawY
- sin(angle
/RAD2DEG
) * FixedToInt(textAfter
) ) ;
1345 m_graphicContext
->FillPath( path
, m_textBackgroundColour
) ;
1349 drawX
+= (int)(sin(angle
/RAD2DEG
) * FixedToInt(ascent
));
1350 drawY
+= (int)(cos(angle
/RAD2DEG
) * FixedToInt(ascent
));
1352 status
= ::ATSUMeasureTextImage( atsuLayout
, kATSUFromTextBeginning
, kATSUToTextEnd
,
1353 IntToFixed(drawX
) , IntToFixed(drawY
) , &rect
);
1354 wxASSERT_MSG( status
== noErr
, wxT("couldn't measure the rotated text") );
1356 CGContextSaveGState(dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext());
1357 CGContextTranslateCTM(dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext(), drawX
, drawY
);
1358 CGContextScaleCTM(dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext(), 1, -1);
1359 status
= ::ATSUDrawText( atsuLayout
, kATSUFromTextBeginning
, kATSUToTextEnd
,
1360 IntToFixed(0) , IntToFixed(0) );
1361 wxASSERT_MSG( status
== noErr
, wxT("couldn't draw the rotated text") );
1362 CGContextRestoreGState( dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() ) ;
1364 CalcBoundingBox(XDEV2LOG(rect
.left
), YDEV2LOG(rect
.top
) );
1365 CalcBoundingBox(XDEV2LOG(rect
.right
), YDEV2LOG(rect
.bottom
) );
1367 ::ATSUDisposeTextLayout(atsuLayout
);
1368 #if SIZEOF_WCHAR_T == 4
1373 void wxDC::DoDrawText(const wxString
& strtext
, wxCoord x
, wxCoord y
)
1375 wxCHECK_RET(Ok(), wxT("wxDC::DoDrawText Invalid DC"));
1376 DoDrawRotatedText( strtext
, x
, y
, 0.0 ) ;
1379 bool wxDC::CanGetTextExtent() const
1381 wxCHECK_MSG(Ok(), false, wxT("Invalid DC"));
1385 void wxDC::DoGetTextExtent( const wxString
&str
, wxCoord
*width
, wxCoord
*height
,
1386 wxCoord
*descent
, wxCoord
*externalLeading
,
1387 wxFont
*theFont
) const
1389 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1390 wxFont formerFont
= m_font
;
1393 // work around the constness
1394 *((wxFont
*)(&m_font
)) = *theFont
;
1398 if ( str
.Length() == 0 )
1401 wxCHECK_RET( m_macATSUIStyle
!= NULL
, wxT("No valid font set") ) ;
1403 OSStatus status
= noErr
;
1404 ATSUTextLayout atsuLayout
;
1405 UniCharCount chars
= str
.Length() ;
1406 UniChar
* ubuf
= NULL
;
1407 #if SIZEOF_WCHAR_T == 4
1408 wxMBConvUTF16BE converter
;
1410 size_t unicharlen
= converter
.WC2MB( NULL
, str
.wc_str() , 0 ) ;
1411 ubuf
= (UniChar
*) malloc( unicharlen
+ 2 ) ;
1412 converter
.WC2MB( (char*) ubuf
, str
.wc_str(), unicharlen
+ 2 ) ;
1414 const wxWCharBuffer wchar
= str
.wc_str( wxConvLocal
) ;
1415 size_t unicharlen
= converter
.WC2MB( NULL
, wchar
.data() , 0 ) ;
1416 ubuf
= (UniChar
*) malloc( unicharlen
+ 2 ) ;
1417 converter
.WC2MB( (char*) ubuf
, wchar
.data() , unicharlen
+ 2 ) ;
1419 chars
= unicharlen
/ 2 ;
1422 ubuf
= (UniChar
*) str
.wc_str() ;
1424 wxWCharBuffer wchar
= str
.wc_str( wxConvLocal
) ;
1425 chars
= wxWcslen( wchar
.data() ) ;
1426 ubuf
= (UniChar
*) wchar
.data() ;
1431 status
= ::ATSUCreateTextLayoutWithTextPtr( (UniCharArrayPtr
) ubuf
, 0 , chars
, chars
, 1 ,
1432 &chars
, (ATSUStyle
*) &m_macATSUIStyle
, &atsuLayout
) ;
1434 wxASSERT_MSG( status
== noErr
, wxT("couldn't create the layout of the text") );
1436 ATSUTextMeasurement textBefore
;
1437 ATSUTextMeasurement textAfter
;
1438 ATSUTextMeasurement textAscent
;
1439 ATSUTextMeasurement textDescent
;
1441 status
= ::ATSUGetUnjustifiedBounds( atsuLayout
, kATSUFromTextBeginning
, kATSUToTextEnd
,
1442 &textBefore
, &textAfter
, &textAscent
, &textDescent
);
1445 *height
= YDEV2LOGREL( FixedToInt(textAscent
+ textDescent
) ) ;
1447 *descent
=YDEV2LOGREL( FixedToInt(textDescent
) );
1448 if ( externalLeading
)
1449 *externalLeading
= 0 ;
1451 *width
= XDEV2LOGREL( FixedToInt(textAfter
- textBefore
) ) ;
1453 ::ATSUDisposeTextLayout(atsuLayout
);
1454 #if SIZEOF_WCHAR_T == 4
1459 // work around the constness
1460 *((wxFont
*)(&m_font
)) = formerFont
;
1466 bool wxDC::DoGetPartialTextExtents(const wxString
& text
, wxArrayInt
& widths
) const
1468 wxCHECK_MSG(Ok(), false, wxT("Invalid DC"));
1471 widths
.Add(0, text
.Length());
1473 if (text
.Length() == 0)
1476 wxFAIL_MSG( wxT("Unimplemented for Core Graphics") ) ;
1481 wxCoord
wxDC::GetCharWidth(void) const
1484 DoGetTextExtent(wxT("g") , &width
, NULL
, NULL
, NULL
, NULL
) ;
1488 wxCoord
wxDC::GetCharHeight(void) const
1491 DoGetTextExtent(wxT("g") , NULL
, &height
, NULL
, NULL
, NULL
) ;
1495 void wxDC::Clear(void)
1497 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1499 if ( m_backgroundBrush
.Ok() && m_backgroundBrush
.GetStyle() != wxTRANSPARENT
)
1501 HIRect rect
= CGRectMake( -10000 , -10000 , 20000 , 20000 ) ;
1502 switch( m_backgroundBrush
.MacGetBrushKind() )
1504 case kwxMacBrushTheme
:
1508 case kwxMacBrushThemeBackground
:
1510 HIThemeBackgroundDrawInfo drawInfo
;
1511 drawInfo
.version
= 0 ;
1512 drawInfo
.state
= kThemeStateActive
;
1513 drawInfo
.kind
= m_backgroundBrush
.MacGetThemeBackground(NULL
) ;
1514 HIThemeDrawBackground( &rect
, &drawInfo
, dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() ,
1515 kHIThemeOrientationNormal
) ;
1519 case kwxMacBrushColour
:
1521 RGBColor col
= MAC_WXCOLORREF( m_backgroundBrush
.GetColour().GetPixel()) ;
1522 CGContextSetRGBFillColor( dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() , col
.red
/ 65536.0 , col
.green
/ 65536.0 , col
.blue
/ 65536.0 , 1.0 ) ;
1523 CGContextFillRect(dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext(), rect
);
1525 // reset to normal value
1526 col
= MAC_WXCOLORREF( GetBrush().GetColour().GetPixel() ) ;
1527 CGContextSetRGBFillColor( dynamic_cast<wxMacCGContext
*>(m_graphicContext
)->GetNativeContext() , col
.red
/ 65536.0 , col
.green
/ 65536.0 , col
.blue
/ 65536.0 , 1.0 ) ;
1534 void wxDC::MacInstallFont() const
1536 wxCHECK_RET(Ok(), wxT("Invalid DC"));
1538 if( m_macATSUIStyle
)
1540 ::ATSUDisposeStyle((ATSUStyle
)m_macATSUIStyle
);
1541 m_macATSUIStyle
= NULL
;
1544 OSStatus status
= noErr
;
1545 status
= ATSUCreateAndCopyStyle( (ATSUStyle
) m_font
.MacGetATSUStyle() , (ATSUStyle
*) &m_macATSUIStyle
) ;
1546 wxASSERT_MSG( status
== noErr
, wxT("couldn't set create ATSU style") ) ;
1548 Fixed atsuSize
= IntToFixed( int(m_scaleY
* m_font
.MacGetFontSize()) ) ;
1549 RGBColor atsuColor
= MAC_WXCOLORREF( m_textForegroundColour
.GetPixel() ) ;
1550 ATSUAttributeTag atsuTags
[] =
1555 ByteCount atsuSizes
[sizeof(atsuTags
)/sizeof(ATSUAttributeTag
)] =
1558 sizeof( RGBColor
) ,
1560 // Boolean kTrue = true ;
1561 // Boolean kFalse = false ;
1563 // ATSUVerticalCharacterType kHorizontal = kATSUStronglyHorizontal;
1564 ATSUAttributeValuePtr atsuValues
[sizeof(atsuTags
)/sizeof(ATSUAttributeTag
)] =
1569 status
= ::ATSUSetAttributes((ATSUStyle
)m_macATSUIStyle
, sizeof(atsuTags
)/sizeof(ATSUAttributeTag
) ,
1570 atsuTags
, atsuSizes
, atsuValues
);
1572 wxASSERT_MSG( status
== noErr
, wxT("couldn't Modify ATSU style") ) ;
1575 // ---------------------------------------------------------------------------
1576 // coordinates transformations
1577 // ---------------------------------------------------------------------------
1579 wxCoord
wxDCBase::DeviceToLogicalX(wxCoord x
) const
1581 return ((wxDC
*)this)->XDEV2LOG(x
);
1584 wxCoord
wxDCBase::DeviceToLogicalY(wxCoord y
) const
1586 return ((wxDC
*)this)->YDEV2LOG(y
);
1589 wxCoord
wxDCBase::DeviceToLogicalXRel(wxCoord x
) const
1591 return ((wxDC
*)this)->XDEV2LOGREL(x
);
1594 wxCoord
wxDCBase::DeviceToLogicalYRel(wxCoord y
) const
1596 return ((wxDC
*)this)->YDEV2LOGREL(y
);
1599 wxCoord
wxDCBase::LogicalToDeviceX(wxCoord x
) const
1601 return ((wxDC
*)this)->XLOG2DEV(x
);
1604 wxCoord
wxDCBase::LogicalToDeviceY(wxCoord y
) const
1606 return ((wxDC
*)this)->YLOG2DEV(y
);
1609 wxCoord
wxDCBase::LogicalToDeviceXRel(wxCoord x
) const
1611 return ((wxDC
*)this)->XLOG2DEVREL(x
);
1614 wxCoord
wxDCBase::LogicalToDeviceYRel(wxCoord y
) const
1616 return ((wxDC
*)this)->YLOG2DEVREL(y
);
1619 #endif // wxMAC_USE_CORE_GRAPHICS