]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/ownerdrw.cpp
   1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/ownerdrw.cpp 
   3 // Purpose:     implementation of wxOwnerDrawn class 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // Licence:     wxWindows license 
  10 /////////////////////////////////////////////////////////////////////////////// 
  13 #pragma implementation 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  18 #include "wx/msw/private.h" 
  25   #include "wx/window.h" 
  26   #include "wx/msw/private.h" 
  28   #include "wx/bitmap.h" 
  29   #include "wx/dcmemory.h" 
  34 #include "wx/settings.h" 
  35 #include "wx/ownerdrw.h" 
  36 #include "wx/menuitem.h" 
  41 // ============================================================================ 
  42 // implementation of wxOwnerDrawn class 
  43 // ============================================================================ 
  47 wxOwnerDrawn::wxOwnerDrawn(const wxString
& str
, 
  48                            bool bCheckable
, bool WXUNUSED(bMenuItem
)) 
  51   m_bCheckable   
= bCheckable
; 
  52   m_bOwnerDrawn  
= FALSE
; 
  54   m_nMarginWidth 
= ms_nLastMarginWidth
; 
  56     m_font 
= * wxNORMAL_FONT
; 
  59 #if defined(__WXMSW__) && defined(__WIN32__) && defined(SM_CXMENUCHECK) 
  60   size_t wxOwnerDrawn::ms_nDefaultMarginWidth 
= GetSystemMetrics(SM_CXMENUCHECK
); 
  61 #else   // # what is the reasonable default? 
  62   size_t wxOwnerDrawn::ms_nDefaultMarginWidth 
= 15; 
  65 size_t wxOwnerDrawn::ms_nLastMarginWidth 
= ms_nDefaultMarginWidth
; 
  70 // get size of the item 
  71 bool wxOwnerDrawn::OnMeasureItem(size_t *pwidth
, size_t *pheight
) 
  75   wxString str 
= wxStripMenuCodes(m_strName
); 
  77   // # without this menu items look too tightly packed (at least under Windows) 
  78   str 
+= wxT('W'); // 'W' is typically the widest letter 
  81       dc
.SetFont(GetFont()); 
  83   dc
.GetTextExtent(str
, (long *)pwidth
, (long *)pheight
); 
  85   // JACS: items still look too tightly packed, so adding 2 pixels. 
  86   (*pheight
) = (*pheight
) + 2; 
  88   // Ray Gilbert's changes - Corrects the problem of a BMP 
  89   // being placed next to text in a menu item, and the BMP does 
  90   // not match the size expected by the system.  This will 
  91   // resize the space so the BMP will fit.  Without this, BMPs 
  92   // must be no larger or smaller than 16x16. 
  93   if (m_bmpChecked
.Ok()) 
  95       // Is BMP height larger then text height? 
  96       size_t adjustedHeight 
= m_bmpChecked
.GetHeight() + 
  97                               wxSystemSettings::GetMetric(wxSYS_EDGE_Y
); 
  98       if (*pheight 
< adjustedHeight
) 
  99           *pheight 
= adjustedHeight
; 
 101       // Does BMP encroach on default check menu position? 
 102       size_t adjustedWidth 
= m_bmpChecked
.GetWidth() + 
 103                              (wxSystemSettings::GetMetric(wxSYS_EDGE_X
) * 2); 
 104       if (ms_nDefaultMarginWidth 
< adjustedWidth
) 
 105           *pwidth 
+= adjustedWidth 
- ms_nDefaultMarginWidth
; 
 107       // Do we need to widen margin to fit BMP? 
 108       if ((size_t)GetMarginWidth() < adjustedWidth
) 
 109           SetMarginWidth(adjustedWidth
); 
 112   m_nHeight 
= *pheight
;                // remember height for use in OnDrawItem 
 117 // searching for this macro you'll find all the code where I'm using the native 
 118 // Win32 GDI functions and not wxWindows ones. Might help to whoever decides to 
 119 // port this code to X. (VZ) 
 121 // JACS: TODO. Why does a disabled but highlighted item still 
 122 // get drawn embossed? How can we tell DrawState that we don't want the 
 125 #if defined(__WIN32__) && !defined(__SC__) && !defined(__TWIN32__) 
 126 #define   O_DRAW_NATIVE_API     // comments below explain why I use it 
 130 bool wxOwnerDrawn::OnDrawItem(wxDC
& dc
, 
 135   // we do nothing on focus change 
 136   if ( act 
== wxODFocusChanged 
) 
 140   #define   ToRGB(col)  PALETTERGB(col.Red(), col.Green(), col.Blue()) 
 141   #define   UnRGB(col)  GetRValue(col), GetGValue(col), GetBValue(col) 
 145   DWORD colBack
, colText
; 
 146   if ( st 
& wxODSelected 
) { 
 147     colBack 
= GetSysColor(COLOR_HIGHLIGHT
); 
 148     colText 
= GetSysColor(COLOR_HIGHLIGHTTEXT
); 
 151     // fall back to default colors if none explicitly specified 
 152     colBack 
= m_colBack
.Ok() ? ToRGB(m_colBack
) : GetSysColor(COLOR_WINDOW
); 
 153     colText 
= m_colText
.Ok() ? ToRGB(m_colText
) : GetSysColor(COLOR_WINDOWTEXT
); 
 156   #ifdef  O_DRAW_NATIVE_API 
 157     #define  hdc           (HDC)dc.GetHDC() 
 158     COLORREF colOldText 
= ::SetTextColor(hdc
, colText
), 
 159              colOldBack 
= ::SetBkColor(hdc
, colBack
); 
 161     dc
.SetTextForeground(wxColor(UnRGB(colText
))); 
 162     dc
.SetTextBackground(wxColor(UnRGB(colBack
))); 
 165   // select the font and draw the text 
 166   // --------------------------------- 
 168   // determine where to draw and leave space for a check-mark. 
 169   int x 
= rc
.x 
+ GetMarginWidth(); 
 171   // using native API because it reckognizes '&' 
 172   #ifdef  O_DRAW_NATIVE_API 
 173     int nPrevMode 
= SetBkMode(hdc
, TRANSPARENT
); 
 174     HBRUSH hbr 
= CreateSolidBrush(colBack
), 
 175            hPrevBrush 
= (HBRUSH
)SelectObject(hdc
, hbr
); 
 177     RECT rectAll 
= { rc
.GetLeft(), rc
.GetTop(), rc
.GetRight(), rc
.GetBottom() }; 
 178     FillRect(hdc
, &rectAll
, hbr
); 
 182     // use default font if no font set 
 185       m_font
.RealizeResource(); 
 186       hfont 
= (HFONT
)m_font
.GetResourceHandle(); 
 189       hfont 
= (HFONT
)::GetStockObject(SYSTEM_FONT
); 
 192     HFONT hPrevFont 
= (HFONT
) ::SelectObject(hdc
, hfont
); 
 194         DrawState(hdc
, NULL
, NULL
, 
 195               (LPARAM
)m_strName
.c_str(), m_strName
.length(), 
 196               x
, rc
.y
, rc
.GetWidth(), rc
.GetHeight(), 
 197               DST_PREFIXTEXT 
| (st 
& wxODDisabled 
? DSS_DISABLED 
: 0)); 
 199     if ( !m_strAccel
.empty() ) 
 203         r
.left 
= rc
.GetLeft(); 
 204         r
.right 
= rc
.GetRight() - GetMarginWidth(); 
 205         r
.bottom 
= rc
.GetBottom(); 
 207         DrawText(hdc
, m_strAccel
, m_strAccel
.length(), &r
, 
 208                  DT_SINGLELINE 
| DT_RIGHT 
| DT_VCENTER
); 
 211     (void)SelectObject(hdc
, hPrevBrush
); 
 212     (void)SelectObject(hdc
, hPrevFont
); 
 213     (void)SetBkMode(hdc
, nPrevMode
); 
 215     dc
.SetFont(GetFont()); 
 216     dc
.DrawText(m_strName
, x
, rc
.y
); 
 217   #endif  //O_DRAW_NATIVE_API 
 221   if ( IsCheckable() && !m_bmpChecked
.Ok() ) { 
 222     if ( st 
& wxODChecked 
) { 
 223       // using native APIs for performance and simplicity 
 224 #ifdef  O_DRAW_NATIVE_API 
 225       // what goes on: DrawFrameControl creates a b/w mask, 
 226       // then we copy it to screen to have right colors 
 228         // first create a monochrome bitmap in a memory DC 
 229       HDC hdcMem 
= CreateCompatibleDC(hdc
); 
 230       HBITMAP hbmpCheck 
= CreateBitmap(GetMarginWidth(), m_nHeight
, 1, 1, 0); 
 231       SelectObject(hdcMem
, hbmpCheck
); 
 233         // then draw a check mark into it 
 234       RECT rect 
= { 0, 0, GetMarginWidth(), m_nHeight 
}; 
 238         DrawFrameControl(hdcMem
, &rect
, DFC_MENU
, DFCS_MENUCHECK
); 
 242         // finally copy it to screen DC and clean up 
 243       BitBlt(hdc
, rc
.x
, rc
.y
, GetMarginWidth(), m_nHeight
, 
 244              hdcMem
, 0, 0, SRCCOPY
); 
 247       DeleteObject(hbmpCheck
); 
 249         // #### to do: perhaps using Marlett font (create equiv. font under X) 
 250 //        wxFAIL("not implemented"); 
 251 #endif  //O_DRAW_NATIVE_API 
 255     // for uncheckable item we use only the 'checked' bitmap 
 256     wxBitmap 
bmp(GetBitmap(IsCheckable() ? ((st 
& wxODChecked
) != 0) : TRUE
)); 
 258       wxMemoryDC 
dcMem(&dc
); 
 259       dcMem
.SelectObject(bmp
); 
 262       int nBmpWidth 
= bmp
.GetWidth(), 
 263           nBmpHeight 
= bmp
.GetHeight(); 
 265       // there should be enough place! 
 266       wxASSERT((nBmpWidth 
<= rc
.GetWidth()) && (nBmpHeight 
<= rc
.GetHeight())); 
 268       int heightDiff 
= (m_nHeight 
- nBmpHeight
); 
 269 //      if (heightDiff = -1) 
 272       //MT: blit with mask enabled. 
 273       dc
.Blit(rc
.x 
+ (GetMarginWidth() - nBmpWidth
) / 2, 
 274               rc
.y 
+ heightDiff 
/ 2, 
 275               nBmpWidth
, nBmpHeight
, 
 276               &dcMem
, 0, 0, wxCOPY
, TRUE
); 
 278       if ( st 
& wxODSelected 
) { 
 279         #ifdef  O_DRAW_NATIVE_API 
 280           RECT rectBmp 
= { rc
.GetLeft(), rc
.GetTop(), 
 281                            rc
.GetLeft() + GetMarginWidth(), 
 282                            rc
.GetTop() + m_nHeight 
}; 
 283           SetBkColor(hdc
, colBack
); 
 284           DrawEdge(hdc
, &rectBmp
, EDGE_RAISED
, BF_SOFT 
| BF_RECT
); 
 286           // ## to write portable DrawEdge 
 287         #endif  //O_DRAW_NATIVE_API 
 292   #ifdef  O_DRAW_NATIVE_API 
 293     ::SetTextColor(hdc
, colOldText
); 
 294     ::SetBkColor(hdc
, colOldBack
); 
 297   #endif  //O_DRAW_NATIVE_API 
 303 #endif // wxUSE_OWNER_DRAWN