1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/pen.cpp
4 // Author: Julian Smart
5 // Modified by: Vadim Zeitlin: refactored wxPen code to wxPenRefData
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
30 #include "wx/bitmap.h"
34 #include "wx/msw/private.h"
36 #define M_PENDATA ((wxPenRefData*)m_refData)
38 // Win32 has ExtCreatePen() but WinCE doesn't
39 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
40 #define wxHAVE_EXT_CREATE_PEN
43 // ----------------------------------------------------------------------------
44 // wxPenRefData: contains information about an HPEN and its handle
45 // ----------------------------------------------------------------------------
47 class WXDLLEXPORT wxPenRefData
: public wxGDIRefData
54 wxPenRefData(const wxPenRefData
& data
);
55 wxPenRefData(const wxColour
& col
, int width
, wxPenStyle style
);
56 wxPenRefData(const wxBitmap
& stipple
, int width
);
57 virtual ~wxPenRefData();
59 bool operator==(const wxPenRefData
& data
) const
61 // we intentionally don't compare m_hPen fields here
62 return m_style
== data
.m_style
&&
63 m_width
== data
.m_width
&&
64 m_join
== data
.m_join
&&
65 m_cap
== data
.m_cap
&&
66 m_colour
== data
.m_colour
&&
67 (m_style
!= wxPENSTYLE_STIPPLE
|| m_stipple
.IsSameAs(data
.m_stipple
)) &&
68 (m_style
!= wxPENSTYLE_USER_DASH
||
69 (m_nbDash
== data
.m_nbDash
&&
70 memcmp(m_dash
, data
.m_dash
, m_nbDash
*sizeof(wxDash
)) == 0));
74 // accessors and setters
75 // ---------------------
77 wxColour
& GetColour() const { return const_cast<wxColour
&>(m_colour
); }
78 int GetWidth() const { return m_width
; }
79 wxPenStyle
GetStyle() const { return m_style
; }
80 wxPenJoin
GetJoin() const { return m_join
; }
81 wxPenCap
GetCap() const { return m_cap
; }
82 wxDash
* GetDash() const { return m_dash
; }
83 int GetDashCount() const { return m_nbDash
; }
84 wxBitmap
* GetStipple() const { return const_cast<wxBitmap
*>(&m_stipple
); }
86 void SetColour(const wxColour
& col
) { Free(); m_colour
= col
; }
87 void SetWidth(int width
) { Free(); m_width
= width
; }
88 void SetStyle(wxPenStyle style
) { Free(); m_style
= style
; }
89 void SetStipple(const wxBitmap
& stipple
)
93 m_style
= wxPENSTYLE_STIPPLE
;
97 void SetDashes(int nb_dashes
, const wxDash
*dash
)
101 m_nbDash
= nb_dashes
;
102 m_dash
= const_cast<wxDash
*>(dash
);
105 void SetJoin(wxPenJoin join
) { Free(); m_join
= join
; }
106 void SetCap(wxPenCap cap
) { Free(); m_cap
= cap
; }
112 // create the HPEN if we don't have it yet
115 // get the HPEN creating it on demand
116 WXHPEN
GetHPEN() const;
118 // return true if we have a valid HPEN
119 bool HasHPEN() const { return m_hPen
!= 0; }
121 // return true if we had a valid handle before, false otherwise
125 // initialize the fields which have reasonable default values
127 // doesn't initialize m_width and m_style which must be initialize in ctor
130 m_join
= wxJOIN_ROUND
;
147 wxDECLARE_NO_ASSIGN_CLASS(wxPenRefData
);
150 // ============================================================================
152 // ============================================================================
154 // ----------------------------------------------------------------------------
155 // wxPenRefData ctors/dtor
156 // ----------------------------------------------------------------------------
158 wxPenRefData::wxPenRefData()
162 m_style
= wxPENSTYLE_SOLID
;
166 wxPenRefData::wxPenRefData(const wxPenRefData
& data
)
169 m_style
= data
.m_style
;
170 m_width
= data
.m_width
;
171 m_join
= data
.m_join
;
173 m_nbDash
= data
.m_nbDash
;
174 m_dash
= data
.m_dash
;
175 m_colour
= data
.m_colour
;
179 wxPenRefData::wxPenRefData(const wxColour
& col
, int width
, wxPenStyle style
)
189 wxPenRefData::wxPenRefData(const wxBitmap
& stipple
, int width
)
193 m_style
= wxPENSTYLE_STIPPLE
;
199 wxPenRefData::~wxPenRefData()
202 ::DeleteObject(m_hPen
);
205 // ----------------------------------------------------------------------------
206 // wxPenRefData HPEN management
207 // ----------------------------------------------------------------------------
209 static int ConvertPenStyle(wxPenStyle style
)
213 case wxPENSTYLE_SHORT_DASH
:
214 case wxPENSTYLE_LONG_DASH
:
217 case wxPENSTYLE_TRANSPARENT
:
221 wxFAIL_MSG( _T("unknown pen style") );
224 #ifdef wxHAVE_EXT_CREATE_PEN
228 case wxPENSTYLE_DOT_DASH
:
231 case wxPENSTYLE_USER_DASH
:
234 case wxPENSTYLE_STIPPLE
:
235 case wxPENSTYLE_BDIAGONAL_HATCH
:
236 case wxPENSTYLE_CROSSDIAG_HATCH
:
237 case wxPENSTYLE_FDIAGONAL_HATCH
:
238 case wxPENSTYLE_CROSS_HATCH
:
239 case wxPENSTYLE_HORIZONTAL_HATCH
:
240 case wxPENSTYLE_VERTICAL_HATCH
:
241 case wxPENSTYLE_SOLID
:
242 #endif // wxHAVE_EXT_CREATE_PEN
248 #ifdef wxHAVE_EXT_CREATE_PEN
250 static int ConvertJoinStyle(wxPenJoin join
)
255 return PS_JOIN_BEVEL
;
258 return PS_JOIN_MITER
;
261 wxFAIL_MSG( _T("unknown pen join style") );
265 return PS_JOIN_ROUND
;
269 static int ConvertCapStyle(wxPenCap cap
)
273 case wxCAP_PROJECTING
:
274 return PS_ENDCAP_SQUARE
;
277 return PS_ENDCAP_FLAT
;
280 wxFAIL_MSG( _T("unknown pen cap style") );
284 return PS_ENDCAP_ROUND
;
288 #endif // wxHAVE_EXT_CREATE_PEN
290 bool wxPenRefData::Alloc()
295 if ( m_style
== wxPENSTYLE_TRANSPARENT
)
297 m_hPen
= (HPEN
)::GetStockObject(NULL_PEN
);
301 const COLORREF col
= m_colour
.GetPixel();
303 #ifdef wxHAVE_EXT_CREATE_PEN
304 // Only NT can display dashed or dotted lines with width > 1
305 static const int os
= wxGetOsVersion();
306 if ( os
!= wxOS_WINDOWS_NT
&&
307 (m_style
== wxPENSTYLE_DOT
||
308 m_style
== wxPENSTYLE_LONG_DASH
||
309 m_style
== wxPENSTYLE_SHORT_DASH
||
310 m_style
== wxPENSTYLE_DOT_DASH
||
311 m_style
== wxPENSTYLE_USER_DASH
) &&
317 // check if it's a standard kind of pen which can be created with just
319 if ( m_join
== wxJOIN_ROUND
&&
320 m_cap
== wxCAP_ROUND
&&
321 m_style
!= wxPENSTYLE_USER_DASH
&&
322 m_style
!= wxPENSTYLE_STIPPLE
&&
323 (m_width
<= 1 || m_style
== wxPENSTYLE_SOLID
) )
324 #endif // !wxHAVE_EXT_CREATE_PEN
326 m_hPen
= ::CreatePen(ConvertPenStyle(m_style
), m_width
, col
);
328 #ifdef wxHAVE_EXT_CREATE_PEN
329 else // need to use ExtCreatePen()
331 DWORD styleMSW
= PS_GEOMETRIC
|
332 ConvertPenStyle(m_style
) |
333 ConvertJoinStyle(m_join
) |
334 ConvertCapStyle(m_cap
);
339 case wxPENSTYLE_STIPPLE
:
340 lb
.lbStyle
= BS_PATTERN
;
341 lb
.lbHatch
= wxPtrToUInt(m_stipple
.GetHBITMAP());
344 case wxPENSTYLE_BDIAGONAL_HATCH
:
345 lb
.lbStyle
= BS_HATCHED
;
346 lb
.lbHatch
= HS_BDIAGONAL
;
349 case wxPENSTYLE_CROSSDIAG_HATCH
:
350 lb
.lbStyle
= BS_HATCHED
;
351 lb
.lbHatch
= HS_DIAGCROSS
;
354 case wxPENSTYLE_FDIAGONAL_HATCH
:
355 lb
.lbStyle
= BS_HATCHED
;
356 lb
.lbHatch
= HS_FDIAGONAL
;
359 case wxPENSTYLE_CROSS_HATCH
:
360 lb
.lbStyle
= BS_HATCHED
;
361 lb
.lbHatch
= HS_CROSS
;
364 case wxPENSTYLE_HORIZONTAL_HATCH
:
365 lb
.lbStyle
= BS_HATCHED
;
366 lb
.lbHatch
= HS_HORIZONTAL
;
369 case wxPENSTYLE_VERTICAL_HATCH
:
370 lb
.lbStyle
= BS_HATCHED
;
371 lb
.lbHatch
= HS_VERTICAL
;
375 lb
.lbStyle
= BS_SOLID
;
377 // this should be unnecessary (it's unused) but suppresses the
378 // Purify messages about uninitialized memory read
387 if ( m_style
== wxPENSTYLE_USER_DASH
&& m_nbDash
&& m_dash
)
389 dash
= new DWORD
[m_nbDash
];
390 int rw
= m_width
> 1 ? m_width
: 1;
391 for ( int i
= 0; i
< m_nbDash
; i
++ )
392 dash
[i
] = m_dash
[i
] * rw
;
399 m_hPen
= ::ExtCreatePen(styleMSW
, m_width
, &lb
, m_nbDash
, (LPDWORD
)dash
);
403 #endif // wxHAVE_EXT_CREATE_PEN
408 bool wxPenRefData::Free()
413 ::DeleteObject(m_hPen
);
419 WXHPEN
wxPenRefData::GetHPEN() const
422 const_cast<wxPenRefData
*>(this)->Alloc();
424 return (WXHPEN
)m_hPen
;
427 // ----------------------------------------------------------------------------
429 // ----------------------------------------------------------------------------
431 IMPLEMENT_DYNAMIC_CLASS(wxPen
, wxGDIObject
)
433 wxPen::wxPen(const wxColour
& col
, int width
, wxPenStyle style
)
435 m_refData
= new wxPenRefData(col
, width
, style
);
438 #if FUTURE_WXWIN_COMPATIBILITY_3_0
439 wxPen::wxPen(const wxColour
& colour
, int width
, int style
)
441 m_refData
= new wxPenRefData(colour
, width
, (wxPenStyle
)style
);
445 wxPen::wxPen(const wxBitmap
& stipple
, int width
)
447 m_refData
= new wxPenRefData(stipple
, width
);
450 bool wxPen::operator==(const wxPen
& pen
) const
453 penData
= static_cast<const wxPenRefData
*>(pen
.m_refData
);
455 // an invalid pen is only equal to another invalid pen
456 return m_refData
? penData
&& *M_PENDATA
== *penData
: !penData
;
459 bool wxPen::RealizeResource()
461 return M_PENDATA
&& M_PENDATA
->Alloc();
464 WXHANDLE
wxPen::GetResourceHandle() const
466 return M_PENDATA
? M_PENDATA
->GetHPEN() : 0;
469 bool wxPen::FreeResource(bool WXUNUSED(force
))
471 return M_PENDATA
&& M_PENDATA
->Free();
474 bool wxPen::IsFree() const
476 return M_PENDATA
&& !M_PENDATA
->HasHPEN();
479 wxGDIRefData
* wxPen::CreateGDIRefData() const
481 return new wxPenRefData
;
484 wxGDIRefData
* wxPen::CloneGDIRefData(const wxGDIRefData
* data
) const
486 return new wxPenRefData(*static_cast<const wxPenRefData
*>(data
));
489 void wxPen::SetColour(const wxColour
& col
)
493 M_PENDATA
->SetColour(col
);
496 void wxPen::SetColour(unsigned char r
, unsigned char g
, unsigned char b
)
498 SetColour(wxColour(r
, g
, b
));
501 void wxPen::SetWidth(int width
)
505 M_PENDATA
->SetWidth(width
);
508 void wxPen::SetStyle(wxPenStyle style
)
512 M_PENDATA
->SetStyle(style
);
515 void wxPen::SetStipple(const wxBitmap
& stipple
)
519 M_PENDATA
->SetStipple(stipple
);
522 void wxPen::SetDashes(int nb_dashes
, const wxDash
*dash
)
526 M_PENDATA
->SetDashes(nb_dashes
, dash
);
529 void wxPen::SetJoin(wxPenJoin join
)
533 M_PENDATA
->SetJoin(join
);
536 void wxPen::SetCap(wxPenCap cap
)
540 M_PENDATA
->SetCap(cap
);
543 wxColour
wxPen::GetColour() const
545 wxCHECK_MSG( Ok(), wxNullColour
, wxT("invalid pen") );
547 return M_PENDATA
->GetColour();
550 int wxPen::GetWidth() const
552 wxCHECK_MSG( Ok(), -1, wxT("invalid pen") );
554 return M_PENDATA
->GetWidth();
557 wxPenStyle
wxPen::GetStyle() const
559 wxCHECK_MSG( Ok(), wxPENSTYLE_INVALID
, wxT("invalid pen") );
561 return M_PENDATA
->GetStyle();
564 wxPenJoin
wxPen::GetJoin() const
566 wxCHECK_MSG( Ok(), wxJOIN_INVALID
, wxT("invalid pen") );
568 return M_PENDATA
->GetJoin();
571 wxPenCap
wxPen::GetCap() const
573 wxCHECK_MSG( Ok(), wxCAP_INVALID
, wxT("invalid pen") );
575 return M_PENDATA
->GetCap();
578 int wxPen::GetDashes(wxDash
** ptr
) const
580 wxCHECK_MSG( Ok(), -1, wxT("invalid pen") );
582 *ptr
= M_PENDATA
->GetDash();
583 return M_PENDATA
->GetDashCount();
586 wxDash
* wxPen::GetDash() const
588 wxCHECK_MSG( Ok(), NULL
, wxT("invalid pen") );
590 return m_refData
? M_PENDATA
->GetDash() : NULL
;
593 int wxPen::GetDashCount() const
595 wxCHECK_MSG( Ok(), -1, wxT("invalid pen") );
597 return m_refData
? M_PENDATA
->GetDashCount() : 0;
600 wxBitmap
* wxPen::GetStipple() const
602 wxCHECK_MSG( Ok(), NULL
, wxT("invalid pen") );
604 return m_refData
? M_PENDATA
->GetStipple() : NULL
;