#include "wx/fontutil.h"
#include "wx/msw/private.h"
+#include "wx/msw/dc.h"
#ifndef SPI_GETKEYBOARDCUES
#define SPI_GETKEYBOARDCUES 0x100A
#endif
+#ifndef DSS_HIDEPREFIX
+#define DSS_HIDEPREFIX 0x0200
+#endif
+
class wxMSWSystemMenuFontModule : public wxModule
{
public:
virtual bool OnInit()
+ {
+ return true;
+ }
+
+ virtual void OnExit()
+ {
+ if ( ms_systemMenuFont )
+ {
+ delete ms_systemMenuFont;
+ ms_systemMenuFont = NULL;
+ }
+ }
+
+ static const wxFont& GetSystemMenuFont()
+ {
+ if ( !ms_systemMenuFont )
+ DoInitFont();
+
+ return *ms_systemMenuFont;
+ }
+
+ static int GetSystemMenuHeight()
+ {
+ if ( !ms_systemMenuHeight )
+ DoInitMetrics();
+
+ return ms_systemMenuHeight;
+ }
+
+ static bool AlwaysShowCues()
+ {
+ if ( !ms_systemMenuHeight )
+ DoInitMetrics();
+
+ return ms_alwaysShowCues;
+ }
+
+private:
+ static NONCLIENTMETRICS GetNCM()
{
WinStruct<NONCLIENTMETRICS> nm;
- ::SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, &nm, 0);
+ if ( !::SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, &nm, 0) )
+ {
+#if WINVER >= 0x0600
+ // a new field has been added to NONCLIENTMETRICS under Vista, so
+ // the call to SystemParametersInfo() fails if we use the struct
+ // size incorporating this new value on an older system -- retry
+ // without it
+ nm.cbSize -= sizeof(int);
+ if ( !::SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, &nm, 0) )
+#endif // WINVER >= 0x0600
+ {
+ // maybe we should initialize the struct with some defaults?
+ wxLogLastError(_T("SystemParametersInfo(SPI_GETNONCLIENTMETRICS)"));
+ }
+ }
- ms_systemMenuButtonWidth = nm.iMenuHeight;
+ return nm;
+ }
+ static void DoInitMetrics()
+ {
// iMenuHeight is the menu bar height and the menu items are less tall,
// although I don't know by how much -- below is the value for my system
- ms_systemMenuHeight = nm.iMenuHeight - 4;
+ ms_systemMenuHeight = GetNCM().iMenuHeight - 4;
- // create menu font
- wxNativeFontInfo info;
- memcpy(&info.lf, &nm.lfMenuFont, sizeof(LOGFONT));
- ms_systemMenuFont = new wxFont(info);
+ wxASSERT_MSG( ms_systemMenuHeight > 0,
+ "menu height should be positive" );
if ( ::SystemParametersInfo(SPI_GETKEYBOARDCUES, 0,
- &ms_showCues, 0) == 0 )
+ &ms_alwaysShowCues, 0) == 0 )
{
// if it's not supported, we must be on an old Windows version
// which always shows them
- ms_showCues = true;
+ ms_alwaysShowCues = true;
}
-
- return true;
}
- virtual void OnExit()
+ static void DoInitFont()
{
- delete ms_systemMenuFont;
- ms_systemMenuFont = NULL;
+ ms_systemMenuFont = new wxFont(wxNativeFontInfo(GetNCM().lfMenuFont));
}
static wxFont* ms_systemMenuFont;
- static int ms_systemMenuButtonWidth;
static int ms_systemMenuHeight;
- static bool ms_showCues;
+ static bool ms_alwaysShowCues;
+
-private:
DECLARE_DYNAMIC_CLASS(wxMSWSystemMenuFontModule)
};
// SystemParametersInfo() call.
wxFont* wxMSWSystemMenuFontModule::ms_systemMenuFont = NULL;
-int wxMSWSystemMenuFontModule::ms_systemMenuButtonWidth = 18;
-int wxMSWSystemMenuFontModule::ms_systemMenuHeight = 18;
-bool wxMSWSystemMenuFontModule::ms_showCues = true;
+int wxMSWSystemMenuFontModule::ms_systemMenuHeight = 0;
+bool wxMSWSystemMenuFontModule::ms_alwaysShowCues = false;
IMPLEMENT_DYNAMIC_CLASS(wxMSWSystemMenuFontModule, wxModule)
if ( !font.Ok() )
{
if ( IsMenuItem() )
- font = *wxMSWSystemMenuFontModule::ms_systemMenuFont;
+ font = wxMSWSystemMenuFontModule::GetSystemMenuFont();
if ( !font.Ok() )
font = *wxNORMAL_FONT;
*pwidth += 4;
// make sure that this item is at least as tall as the system menu height
- const size_t heightStd = wxMSWSystemMenuFontModule::ms_systemMenuHeight;
+ const size_t heightStd = wxMSWSystemMenuFontModule::GetSystemMenuHeight();
if ( *pheight < heightStd )
*pheight = heightStd;
SIZE sizeRect;
::GetTextExtentPoint32(hdc, strMenuText.c_str(), strMenuText.length(), &sizeRect);
- ::DrawState(hdc, NULL, NULL,
- (LPARAM)strMenuText.wx_str(),
- strMenuText.length(),
- xText, rc.y + (int) ((rc.GetHeight()-sizeRect.cy)/2.0), // centre text vertically
- rc.GetWidth()-margin, sizeRect.cy,
- DST_PREFIXTEXT |
- (((st & wxODDisabled) && !(st & wxODSelected)) ? DSS_DISABLED : 0) |
- (((st & wxODHidePrefix) && !wxMSWSystemMenuFontModule::ms_showCues) ? 512 : 0)); // 512 == DSS_HIDEPREFIX
+
+ int flags = DST_PREFIXTEXT;
+ if ( (st & wxODDisabled) && !(st & wxODSelected) )
+ flags |= DSS_DISABLED;
+
+ if ( (st & wxODHidePrefix) &&
+ !wxMSWSystemMenuFontModule::AlwaysShowCues() )
+ flags |= DSS_HIDEPREFIX;
+
+ ::DrawState
+ (
+ hdc,
+ NULL,
+ NULL,
+ (LPARAM)strMenuText.wx_str(),
+ strMenuText.length(),
+ xText,
+ rc.y + (rc.GetHeight() - sizeRect.cy + 1)/2, // centre vertically
+ rc.GetWidth() - margin,
+ sizeRect.cy,
+ flags
+ );
// ::SetTextAlign(hdc, TA_RIGHT) doesn't work with DSS_DISABLED or DSS_MONO
// as the last parameter in DrawState() (at least with Windows98). So we have