]> git.saurik.com Git - wxWidgets.git/blob - src/msw/settings.cpp
Catch attempts to create a window with itself as parent.
[wxWidgets.git] / src / msw / settings.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/settings.cpp
3 // Purpose: wxSystemSettingsNative implementation for MSW
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 04/01/98
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
22
23 #ifdef __BORLANDC__
24 #pragma hdrstop
25 #endif
26
27 #include "wx/settings.h"
28
29 #ifndef WX_PRECOMP
30 #include "wx/utils.h"
31 #include "wx/gdicmn.h"
32 #include "wx/module.h"
33 #endif
34
35 #include "wx/msw/private.h"
36 #include "wx/msw/missing.h" // for SM_CXCURSOR, SM_CYCURSOR, SM_TABLETPC
37 #include "wx/msw/private/metrics.h"
38
39 #ifndef SPI_GETFLATMENU
40 #define SPI_GETFLATMENU 0x1022
41 #endif
42
43 #include "wx/fontutil.h"
44 #include "wx/fontenum.h"
45
46 // ----------------------------------------------------------------------------
47 // private classes
48 // ----------------------------------------------------------------------------
49
50 // the module which is used to clean up wxSystemSettingsNative data (this is a
51 // singleton class so it can't be done in the dtor)
52 class wxSystemSettingsModule : public wxModule
53 {
54 public:
55 virtual bool OnInit();
56 virtual void OnExit();
57
58 private:
59 DECLARE_DYNAMIC_CLASS(wxSystemSettingsModule)
60 };
61
62 // ----------------------------------------------------------------------------
63 // global data
64 // ----------------------------------------------------------------------------
65
66 // the font returned by GetFont(wxSYS_DEFAULT_GUI_FONT): it is created when
67 // GetFont() is called for the first time and deleted by wxSystemSettingsModule
68 static wxFont *gs_fontDefault = NULL;
69
70 // ============================================================================
71 // implementation
72 // ============================================================================
73
74 // TODO: see ::SystemParametersInfo for all sorts of Windows settings.
75 // Different args are required depending on the id. How does this differ
76 // from GetSystemMetric, and should it? Perhaps call it GetSystemParameter
77 // and pass an optional void* arg to get further info.
78 // Should also have SetSystemParameter.
79 // Also implement WM_WININICHANGE (NT) / WM_SETTINGCHANGE (Win95)
80
81 // ----------------------------------------------------------------------------
82 // wxSystemSettingsModule
83 // ----------------------------------------------------------------------------
84
85 IMPLEMENT_DYNAMIC_CLASS(wxSystemSettingsModule, wxModule)
86
87 bool wxSystemSettingsModule::OnInit()
88 {
89 return true;
90 }
91
92 void wxSystemSettingsModule::OnExit()
93 {
94 wxDELETE(gs_fontDefault);
95 }
96
97 // ----------------------------------------------------------------------------
98 // wxSystemSettingsNative
99 // ----------------------------------------------------------------------------
100
101 // ----------------------------------------------------------------------------
102 // colours
103 // ----------------------------------------------------------------------------
104
105 wxColour wxSystemSettingsNative::GetColour(wxSystemColour index)
106 {
107 // we use 0 as the default value just to avoid compiler warnings, as there
108 // is no invalid colour value we use hasCol as the real indicator of
109 // whether colSys was initialized or not
110 COLORREF colSys = 0;
111 bool hasCol = false;
112
113 // the default colours for the entries after BTNHIGHLIGHT
114 static const COLORREF s_defaultSysColors[] =
115 {
116 0x000000, // 3DDKSHADOW
117 0xdfdfdf, // 3DLIGHT
118 0x000000, // INFOTEXT
119 0xe1ffff, // INFOBK
120
121 0, // filler - no std colour with this index
122
123 // TODO: please fill in the standard values of those, I don't have them
124 0, // HOTLIGHT
125 0, // GRADIENTACTIVECAPTION
126 0, // GRADIENTINACTIVECAPTION
127 0, // MENU
128 0, // MENUBAR (unused)
129 };
130
131 if ( index == wxSYS_COLOUR_LISTBOXTEXT)
132 {
133 // there is no standard colour with this index, map to another one
134 index = wxSYS_COLOUR_WINDOWTEXT;
135 }
136 else if ( index == wxSYS_COLOUR_LISTBOXHIGHLIGHTTEXT)
137 {
138 // there is no standard colour with this index, map to another one
139 index = wxSYS_COLOUR_HIGHLIGHTTEXT;
140 }
141 else if ( index == wxSYS_COLOUR_LISTBOX )
142 {
143 // there is no standard colour with this index, map to another one
144 index = wxSYS_COLOUR_WINDOW;
145 }
146 else if ( index > wxSYS_COLOUR_BTNHIGHLIGHT )
147 {
148 // the indices before BTNHIGHLIGHT are understood by GetSysColor() in
149 // all Windows version, for the other ones we have to check
150 bool useDefault;
151
152 int verMaj, verMin;
153 wxGetOsVersion(&verMaj, &verMin);
154 if ( verMaj < 4 )
155 {
156 // NT 3.5
157 useDefault = true;
158 }
159 else if ( verMaj == 4 )
160 {
161 // Win95/NT 4.0
162 useDefault = index > wxSYS_COLOUR_INFOBK;
163 }
164 else if ( verMaj == 5 && verMin == 0 )
165 {
166 // Win98/Win2K
167 useDefault = index > wxSYS_COLOUR_GRADIENTINACTIVECAPTION;
168 }
169 else // >= 5.1
170 {
171 // 5.1 is Windows XP
172 useDefault = false;
173 // Determine if we are using flat menus, only then allow wxSYS_COLOUR_MENUBAR
174 if ( index == wxSYS_COLOUR_MENUBAR )
175 {
176 BOOL isFlat ;
177 if ( SystemParametersInfo( SPI_GETFLATMENU , 0 ,&isFlat, 0 ) )
178 {
179 if ( !isFlat )
180 index = wxSYS_COLOUR_MENU ;
181 }
182 }
183 }
184
185 if ( useDefault )
186 {
187 // special handling for MENUBAR colour: we use this in wxToolBar
188 // and wxStatusBar to have correct bg colour under Windows XP
189 // (which uses COLOR_MENUBAR for them) but they should still look
190 // correctly under previous Windows versions as well
191 if ( index == wxSYS_COLOUR_MENUBAR )
192 {
193 index = wxSYS_COLOUR_3DFACE;
194 }
195 else // replace with default colour
196 {
197 unsigned int n = index - wxSYS_COLOUR_BTNHIGHLIGHT;
198
199 wxASSERT_MSG( n < WXSIZEOF(s_defaultSysColors),
200 wxT("forgot tp update the default colours array") );
201
202 colSys = s_defaultSysColors[n];
203 hasCol = true;
204 }
205 }
206 }
207
208 if ( !hasCol )
209 {
210 #ifdef __WXWINCE__
211 colSys = ::GetSysColor(index|SYS_COLOR_INDEX_FLAG);
212 #else
213 colSys = ::GetSysColor(index);
214 #endif
215 }
216
217 wxColour ret = wxRGBToColour(colSys);
218 wxASSERT(ret.IsOk());
219 return ret;
220 }
221
222 // ----------------------------------------------------------------------------
223 // fonts
224 // ----------------------------------------------------------------------------
225
226 wxFont wxCreateFontFromStockObject(int index)
227 {
228 wxFont font;
229
230 HFONT hFont = (HFONT) ::GetStockObject(index);
231 if ( hFont )
232 {
233 LOGFONT lf;
234 if ( ::GetObject(hFont, sizeof(LOGFONT), &lf) != 0 )
235 {
236 wxNativeFontInfo info;
237 info.lf = lf;
238 // Under MicroWindows we pass the HFONT as well
239 // because it's hard to convert HFONT -> LOGFONT -> HFONT
240 // It's OK to delete stock objects, the delete will be ignored.
241 #ifdef __WXMICROWIN__
242 font.Create(info, (WXHFONT) hFont);
243 #else
244 font.Create(info);
245 #endif
246 }
247 else
248 {
249 wxFAIL_MSG( wxT("failed to get LOGFONT") );
250 }
251 }
252 else // GetStockObject() failed
253 {
254 wxFAIL_MSG( wxT("stock font not found") );
255 }
256
257 return font;
258 }
259
260 wxFont wxSystemSettingsNative::GetFont(wxSystemFont index)
261 {
262 #ifdef __WXWINCE__
263 // under CE only a single SYSTEM_FONT exists
264 index;
265
266 if ( !gs_fontDefault )
267 {
268 gs_fontDefault = new wxFont(wxCreateFontFromStockObject(SYSTEM_FONT));
269 }
270
271 wxASSERT(gs_fontDefault->IsOk() &&
272 wxFontEnumerator::IsValidFacename(gs_fontDefault->GetFaceName()));
273 return *gs_fontDefault;
274 #else // !__WXWINCE__
275 // wxWindow ctor calls GetFont(wxSYS_DEFAULT_GUI_FONT) so we're
276 // called fairly often -- this is why we cache this particular font
277 if ( index == wxSYS_DEFAULT_GUI_FONT )
278 {
279 if ( !gs_fontDefault )
280 {
281 // http://blogs.msdn.com/oldnewthing/archive/2005/07/07/436435.aspx
282 // explains why neither SYSTEM_FONT nor DEFAULT_GUI_FONT should be
283 // used here
284 //
285 // the message box font seems to be the one which should be used
286 // for most (simple) controls, e.g. buttons and such but other
287 // controls may prefer to use lfStatusFont or lfCaptionFont if it
288 // is more appropriate for them
289 wxNativeFontInfo info;
290 info.lf = wxMSWImpl::GetNonClientMetrics().lfMessageFont;
291 gs_fontDefault = new wxFont(info);
292 }
293
294 return *gs_fontDefault;
295 }
296
297 wxFont font = wxCreateFontFromStockObject(index);
298
299 wxASSERT(font.IsOk());
300
301 #if wxUSE_FONTENUM
302 wxASSERT(wxFontEnumerator::IsValidFacename(font.GetFaceName()));
303 #endif // wxUSE_FONTENUM
304
305 return font;
306 #endif // __WXWINCE__/!__WXWINCE__
307 }
308
309 // ----------------------------------------------------------------------------
310 // system metrics/features
311 // ----------------------------------------------------------------------------
312
313 // TODO: some of the "metrics" clearly should be features now that we have
314 // HasFeature()!
315
316 // the conversion table from wxSystemMetric enum to GetSystemMetrics() param
317 //
318 // if the constant is not defined, put -1 in the table to indicate that it is
319 // unknown
320 static const int gs_metricsMap[] =
321 {
322 -1, // wxSystemMetric enums start at 1, so give a dummy value for pos 0.
323 #if defined(__WIN32__) && !defined(__WXWINCE__)
324 SM_CMOUSEBUTTONS,
325 #else
326 -1,
327 #endif
328
329 SM_CXBORDER,
330 SM_CYBORDER,
331 #ifdef SM_CXCURSOR
332 SM_CXCURSOR,
333 SM_CYCURSOR,
334 #else
335 -1, -1,
336 #endif
337 SM_CXDOUBLECLK,
338 SM_CYDOUBLECLK,
339 #if defined(__WIN32__) && defined(SM_CXDRAG)
340 SM_CXDRAG,
341 SM_CYDRAG,
342 SM_CXEDGE,
343 SM_CYEDGE,
344 #else
345 -1, -1, -1, -1,
346 #endif
347 SM_CXHSCROLL,
348 SM_CYHSCROLL,
349 #ifdef SM_CXHTHUMB
350 SM_CXHTHUMB,
351 #else
352 -1,
353 #endif
354 SM_CXICON,
355 SM_CYICON,
356 SM_CXICONSPACING,
357 SM_CYICONSPACING,
358 #ifdef SM_CXHTHUMB
359 SM_CXMIN,
360 SM_CYMIN,
361 #else
362 -1, -1,
363 #endif
364 SM_CXSCREEN,
365 SM_CYSCREEN,
366
367 #if defined(__WIN32__) && defined(SM_CXSIZEFRAME)
368 SM_CXSIZEFRAME,
369 SM_CYSIZEFRAME,
370 SM_CXSMICON,
371 SM_CYSMICON,
372 #else
373 -1, -1, -1, -1,
374 #endif
375 SM_CYHSCROLL,
376 SM_CXHSCROLL,
377 SM_CXVSCROLL,
378 SM_CYVSCROLL,
379 #ifdef SM_CYVTHUMB
380 SM_CYVTHUMB,
381 #else
382 -1,
383 #endif
384 SM_CYCAPTION,
385 SM_CYMENU,
386 #if defined(__WIN32__) && defined(SM_NETWORK)
387 SM_NETWORK,
388 #else
389 -1,
390 #endif
391 #ifdef SM_PENWINDOWS
392 SM_PENWINDOWS,
393 #else
394 -1,
395 #endif
396 #if defined(__WIN32__) && defined(SM_SHOWSOUNDS)
397 SM_SHOWSOUNDS,
398 #else
399 -1,
400 #endif
401 // SM_SWAPBUTTON is not available under CE and it doesn't make sense to ask
402 // for it there
403 #ifdef SM_SWAPBUTTON
404 SM_SWAPBUTTON,
405 #else
406 -1,
407 #endif
408 -1 // wxSYS_DCLICK_MSEC - not available as system metric
409 };
410
411 // Get a system metric, e.g. scrollbar size
412 int wxSystemSettingsNative::GetMetric(wxSystemMetric index, wxWindow* WXUNUSED(win))
413 {
414 #ifdef __WXMICROWIN__
415 // TODO: probably use wxUniv themes functionality
416 return 0;
417 #else // !__WXMICROWIN__
418 wxCHECK_MSG( index > 0 && (size_t)index < WXSIZEOF(gs_metricsMap), 0,
419 wxT("invalid metric") );
420
421 if ( index == wxSYS_DCLICK_MSEC )
422 {
423 // This one is not a Win32 system metric
424 return ::GetDoubleClickTime();
425 }
426
427 int indexMSW = gs_metricsMap[index];
428 if ( indexMSW == -1 )
429 {
430 // not supported under current system
431 return -1;
432 }
433
434 int rc = ::GetSystemMetrics(indexMSW);
435 if ( index == wxSYS_NETWORK_PRESENT )
436 {
437 // only the last bit is significant according to the MSDN
438 rc &= 1;
439 }
440
441 return rc;
442 #endif // __WXMICROWIN__/!__WXMICROWIN__
443 }
444
445 bool wxSystemSettingsNative::HasFeature(wxSystemFeature index)
446 {
447 switch ( index )
448 {
449 case wxSYS_CAN_ICONIZE_FRAME:
450 case wxSYS_CAN_DRAW_FRAME_DECORATIONS:
451 return true;
452
453 case wxSYS_TABLET_PRESENT:
454 return ::GetSystemMetrics(SM_TABLETPC) != 0;
455
456 default:
457 wxFAIL_MSG( wxT("unknown system feature") );
458
459 return false;
460 }
461 }
462
463 // ----------------------------------------------------------------------------
464 // function from wx/msw/wrapcctl.h: there is really no other place for it...
465 // ----------------------------------------------------------------------------
466
467 #if wxUSE_LISTCTRL || wxUSE_TREECTRL
468
469 extern wxFont wxGetCCDefaultFont()
470 {
471 #ifndef __WXWINCE__
472 // under the systems enumerated below (anything released after Win98), the
473 // default font used for the common controls seems to be the desktop font
474 // which is also used for the icon titles and not the stock default GUI
475 // font
476 bool useIconFont;
477 int verMaj, verMin;
478 switch ( wxGetOsVersion(&verMaj, &verMin) )
479 {
480 case wxOS_WINDOWS_9X:
481 // 4.10 is Win98
482 useIconFont = verMaj == 4 && verMin >= 10;
483 break;
484
485 case wxOS_WINDOWS_NT:
486 // 5.0 is Win2k
487 useIconFont = verMaj >= 5;
488 break;
489
490 default:
491 useIconFont = false;
492 }
493
494 if ( useIconFont )
495 {
496 LOGFONT lf;
497 if ( ::SystemParametersInfo
498 (
499 SPI_GETICONTITLELOGFONT,
500 sizeof(lf),
501 &lf,
502 0
503 ) )
504 {
505 return wxFont(wxCreateFontFromLogFont(&lf));
506 }
507 else
508 {
509 wxLogLastError(wxT("SystemParametersInfo(SPI_GETICONTITLELOGFONT"));
510 }
511 }
512 #endif // __WXWINCE__
513
514 // fall back to the default font for the normal controls
515 return wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
516 }
517
518 #endif // wxUSE_LISTCTRL || wxUSE_TREECTRL