]>
git.saurik.com Git - wxWidgets.git/blob - src/os2/ownerdrw.cpp
   1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/ownerdrw.cpp 
   3 // Purpose:     implementation of wxOwnerDrawn class 
   4 // Author:      David Webster 
   8 // Copyright:   (c) David Webster 
   9 // Licence:     wxWindows license 
  10 /////////////////////////////////////////////////////////////////////////////// 
  13 #pragma implementation 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  20   #include "wx/window.h" 
  21   #include "wx/msw/private.h" 
  23   #include "wx/bitmap.h" 
  24   #include "wx/dcmemory.h" 
  31 #include "wx/ownerdrw.h" 
  32 #include "wx/menuitem.h" 
  35 // ============================================================================ 
  36 // implementation of wxOwnerDrawn class 
  37 // ============================================================================ 
  43 wxOwnerDrawn::wxOwnerDrawn( 
  50     m_bCheckable   
= bCheckable
; 
  51     m_bOwnerDrawn  
= FALSE
; 
  53     m_nMarginWidth 
= ms_nLastMarginWidth
; 
  55         m_font 
= *wxNORMAL_FONT
; 
  56 } // end of wxOwnerDrawn::wxOwnerDrawn 
  58 size_t wxOwnerDrawn::ms_nDefaultMarginWidth 
= 15; 
  60 size_t wxOwnerDrawn::ms_nLastMarginWidth 
= ms_nDefaultMarginWidth
; 
  67 bool wxOwnerDrawn::OnMeasureItem( 
  74     vDC
.SetFont(GetFont()); 
  76     wxString                        sStr 
= wxStripMenuCodes(m_strName
); 
  78     vDC
.GetTextExtent( sStr
 
  83     (*pHeight
) = (*pHeight
) + 2; 
  84     m_nHeight 
= *pHeight
;        // remember height for use in OnDrawItem 
  86 } // end of wxOwnerDrawn::OnMeasureItem 
  89 bool wxOwnerDrawn::OnDrawItem( 
  97     // We do nothing on focus change 
  99     if (eAction 
== wxODFocusChanged 
) 
 103     // Select the font and draw the text 
 104     // --------------------------------- 
 108     HPS                             hPS
= rDC
.GetHPS(); 
 112     RECTL                           vRect 
= {rRect
.x 
+ 4, rRect
.y 
+ 1, rRect
.x 
+ (rRect
.width 
- 2), rRect
.y 
+ rRect
.height
}; 
 114     memset(&vCbnd
, 0, sizeof(CHARBUNDLE
)); 
 117     // Use default font if no font set 
 121         m_font
.RealizeResource(); 
 125         ::GpiSetCharSet(hPS
, LCID_DEFAULT
); 
 129     // Base on the status of the menu item pick the right colors 
 131     if (eStatus 
& wxODSelected
) 
 133         wxColour                        
vCol2("WHITE"); 
 134         vColBack
.Set( (unsigned char)0 
 137                     ); // no dark blue in color table 
 140     else if (eStatus 
& wxODDisabled
) 
 142         vRef 
= (ULONG
)::WinQuerySysColor( HWND_DESKTOP
 
 143                                          ,SYSCLR_MENU 
// Light gray 
 146         vColBack
.Set( GetRValue(vRef
) 
 150         vRef 
= (ULONG
)::WinQuerySysColor( HWND_DESKTOP
 
 151                                          ,SYSCLR_MENUDISABLEDTEXT 
// dark gray 
 154         vColText
.Set( GetRValue(vRef
) 
 162         // Fall back to default colors if none explicitly specified 
 164         vRef 
= ::WinQuerySysColor( HWND_DESKTOP
 
 165                                   ,SYSCLR_MENU  
// we are using gray for all our window backgrounds in wxWindows 
 168         vColBack
.Set( GetRValue(vRef
) 
 172         vRef 
= ::WinQuerySysColor( HWND_DESKTOP
 
 173                                   ,SYSCLR_WINDOWTEXT 
// Black 
 176         vColText
.Set( GetRValue(vRef
) 
 182     rDC
.SetTextBackground(vColBack
); 
 183     rDC
.SetTextForeground(vColText
); 
 184     rDC
.SetBackgroundMode(wxTRANSPARENT
); 
 185     vCbnd
.lColor     
= vColText
.GetPixel(); 
 186     vCbnd
.lBackColor 
= vColBack
.GetPixel(); 
 189                   ,CBB_BACK_COLOR 
| CBB_COLOR
 
 198     // Paint the background 
 200     ::WinFillRect(hPS
, &vRect
, vColBack
.GetPixel()); 
 203     // Determine where to draw and leave space for a check-mark. 
 205     int                             nX 
= rRect
.x 
+ GetMarginWidth(); 
 208     // Unfortunately, unlike Win32, PM has no owner drawn specific text 
 209     // drawing methods like ::DrawState that can cleanly handle accel 
 210     // pneumonics and deal, automatically, with various states, so we have 
 211     // to handle them ourselves. Notice Win32 can't handle \t in ownerdrawn 
 212     // strings either.  We cannot handle mneumonics either.  We display 
 213     // it, though, in hopes we can figure it out some day. 
 217     // Display main text and accel text separately to allign better 
 219     wxString                        sTgt 
= "\t"; 
 220     wxString                        sFullString 
= m_strName
; // need to save the original text 
 226     bool                            bFoundMneumonic 
= FALSE
; 
 227     bool                            bFoundAccel 
= FALSE
; 
 230     // Deal with the tab, extracting the Accel text 
 232     nIndex 
= sFullString
.Find(sTgt
.c_str()); 
 236         sAccel 
= sFullString
.Mid(nIndex 
+ 1); 
 237         sFullString
.Remove(nIndex
); 
 241     // Deal with the mneumonic character 
 244     nIndex 
= sFullString
.Find(sTgt
.c_str()); 
 247         wxString                    sTmp 
= sFullString
; 
 249         bFoundMneumonic 
= TRUE
; 
 251         rDC
.GetTextExtent( sTmp
 
 255         sTmp 
= sFullString
[nIndex 
+ 1]; 
 256         rDC
.GetTextExtent( sTmp
 
 260         sFullString
.Replace(sTgt
.c_str(), "", TRUE
); 
 264     // Draw the main item text sans the accel text 
 266     POINTL                      vPntStart 
= {nX
, rRect
.y 
+ 4}; 
 267     ::GpiCharStringAt( rDC
.GetHPS() 
 269                       ,sFullString
.length() 
 270                       ,(PCH
)sFullString
.c_str() 
 275         // Underline the mneumonic -- still won't work, but at least it "looks" right 
 278         POINTL                      vPntEnd 
= {nX 
+ nWidth 
+ nCharWidth 
- 3, rRect
.y 
+ 2}; //CharWidth is bit wide 
 280         vPntStart
.x 
= nX 
+ nWidth 
- 1; 
 281         vPntStart
.y 
= rRect
.y 
+ 2; // Make it look pretty! 
 282         vPen 
= wxPen(vColText
, 1, wxSOLID
); // Assuming we are always black 
 284         ::GpiMove(hPS
, &vPntStart
); 
 285         ::GpiLine(hPS
, &vPntEnd
); 
 289     // Now draw the accel text 
 296         rDC
.GetTextExtent( sAccel
 
 301         // Back off the starting position from the right edge 
 303         vPntStart
.x 
= rRect
.width 
- (nWidth 
+ 7); 
 304         vPntStart
.y 
= rRect
.y 
+ 4; 
 305         ::GpiCharStringAt( rDC
.GetHPS() 
 316     if (IsCheckable() && !m_bmpChecked
.Ok()) 
 318         if (eStatus 
& wxODChecked
) 
 321             HBITMAP                 hBmpCheck 
= ::WinGetSysBitmap(HWND_DESKTOP
, SBMP_MENUCHECK
); 
 323             vRect
.xLeft   
= rRect
.x
; 
 324             vRect
.xRight  
= rRect
.x 
+ GetMarginWidth(); 
 325             vRect
.yBottom 
= rRect
.y
; 
 326             vRect
.yTop    
= rRect
.y 
+ m_nHeight 
- 3; 
 328             ::WinDrawBitmap( hPS             
// PS for this menuitem 
 329                             ,hBmpCheck       
// system checkmark 
 330                             ,NULL            
// draw the whole bitmap 
 331                             ,(PPOINTL
)&vRect 
// destination -- bottom left corner of the menuitem area 
 334                             ,DBM_NORMAL      
// draw normal size 
 341         // For uncheckable item we use only the 'checked' bitmap 
 343         wxBitmap                    
vBmp(GetBitmap(IsCheckable() ? ((eStatus 
& wxODChecked
) != 0) : TRUE
)); 
 348             wxMemoryDC              
vDCMem(&rDC
); 
 349             wxMemoryDC
*             pOldDC 
= (wxMemoryDC
*)vBmp
.GetSelectedInto(); 
 353                 vBmp
.SetSelectedInto(NULL
); 
 355             vDCMem
.SelectObject(vBmp
); 
 360             int                     nBmpWidth 
= vBmp
.GetWidth(); 
 361             int                     nBmpHeight 
= vBmp
.GetHeight(); 
 364             // There should be enough space! 
 366             wxASSERT((nBmpWidth 
<= rRect
.width
) && (nBmpHeight 
<= rRect
.height
)); 
 368             int                     nHeightDiff 
= m_nHeight 
- nBmpHeight
; 
 370             rDC
.Blit( rRect
.x 
+ (GetMarginWidth() - nBmpWidth
) / 2 
 371                      ,rRect
.y 
+ nHeightDiff 
/ 2 
 381             if (eStatus 
& wxODSelected
) 
 383                 RECT                vRectBmp 
= { rRect
.x
 
 385                                                 ,rRect
.x 
+ GetMarginWidth() - 1 
 386                                                 ,rRect
.y 
+ m_nHeight 
- 1 
 388                 POINTL              vPnt1 
= {rRect
.x 
+ 1, rRect
.y 
+ 3}; // Leave a little background border 
 389                 POINTL              vPnt2 
= {rRect
.x 
+ GetMarginWidth(), rRect
.y 
+ m_nHeight 
- 3}; 
 393                 vLine
.lColor 
= vColBack
.GetPixel(); 
 400                 ::GpiMove(hPS
, &vPnt1
); 
 408             vBmp
.SetSelectedInto(NULL
); 
 412 } // end of wxOwnerDrawn::OnDrawItem 
 414 #endif //wxUSE_OWNER_DRAWN