]> git.saurik.com Git - wxWidgets.git/blob - src/msw/display.cpp
update digitalmars makefile
[wxWidgets.git] / src / msw / display.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: display.cpp
3 // Purpose: MSW Implementation of wxDisplay class
4 // Author: Royce Mitchell III
5 // Modified by: VZ (resolutions enumeration/change support, DirectDraw, ...)
6 // Created: 06/21/02
7 // RCS-ID: $Id$
8 // Copyright: (c) wxWindows team
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ===========================================================================
13 // declarations
14 // ===========================================================================
15
16 // ---------------------------------------------------------------------------
17 // headers
18 // ---------------------------------------------------------------------------
19
20 #ifdef __GNUG__
21 #pragma implementation "display.h"
22 #endif
23
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
26
27 #ifdef __BORLANDC__
28 #pragma hdrstop
29 #endif
30
31 #if wxUSE_DISPLAY
32
33 #ifndef WX_PRECOMP
34 #include "wx/dynarray.h"
35 #endif
36
37 #include "wx/dynload.h"
38
39 #include "wx/display.h"
40
41 // the following define is necessary to access the multi-monitor function
42 // declarations in a manner safe to use w/ Windows 95
43 #define COMPILE_MULTIMON_STUBS
44
45 // if you don't have multimon.h you can download the file from:
46 //
47 // http://www.microsoft.com/msj/0697/monitor/monitortextfigs.htm#fig4
48 //
49
50 #ifdef _MSC_VER
51 // as (m)any standard header(s), this one doesn't compile without warnings
52 // with VC++ 6 <sigh>
53 #pragma warning(disable:4706)
54 #endif
55
56 #include <multimon.h>
57
58 #ifdef _MSC_VER
59 #pragma warning(default:4706)
60 #endif
61
62 #include <ddraw.h>
63
64 // we don't want to link with ddraw.lib which contains the real
65 // IID_IDirectDraw2 definition
66 const GUID wxIID_IDirectDraw2 =
67 { 0xB3A6F3E0, 0x2B43, 0x11CF, { 0xA2,0xDE,0x00,0xAA,0x00,0xB9,0x33,0x56 } };
68
69 // ----------------------------------------------------------------------------
70 // macros
71 // ----------------------------------------------------------------------------
72
73 #ifdef _UNICODE
74 #define WINFUNC(x) _T(#x) L"W"
75 #else
76 #define WINFUNC(x) #x "A"
77 #endif
78
79 // ----------------------------------------------------------------------------
80 // typedefs for dynamically loaded Windows functions
81 // ----------------------------------------------------------------------------
82
83 typedef LONG (WINAPI *ChangeDisplaySettingsEx_t)(LPCTSTR lpszDeviceName,
84 LPDEVMODE lpDevMode,
85 HWND hwnd,
86 DWORD dwFlags,
87 LPVOID lParam);
88
89 typedef BOOL (PASCAL *DDEnumExCallback_t)(GUID *pGuid,
90 LPTSTR driverDescription,
91 LPTSTR driverName,
92 LPVOID lpContext,
93 HMONITOR hmon);
94
95 typedef HRESULT (WINAPI *DirectDrawEnumerateEx_t)(DDEnumExCallback_t lpCallback,
96 LPVOID lpContext,
97 DWORD dwFlags);
98
99 typedef HRESULT (WINAPI *DirectDrawCreate_t)(GUID *lpGUID,
100 LPDIRECTDRAW *lplpDD,
101 IUnknown *pUnkOuter);
102
103 // ----------------------------------------------------------------------------
104 // private classes
105 // ----------------------------------------------------------------------------
106
107 class wxDisplayInfo
108 {
109 public:
110 // handle of this monitor used by MonitorXXX() functions, never NULL
111 HMONITOR m_hmon;
112
113 // IDirectDraw object used to control this display, may be NULL
114 IDirectDraw2 *m_pDD2;
115
116 // DirectDraw GUID for this display, only valid when using DirectDraw
117 GUID m_guid;
118
119 // the entire area of this monitor in virtual screen coordinates
120 wxRect m_rect;
121
122 // the display device name for this monitor, empty initially and retrieved
123 // on demand by DoGetName()
124 wxString m_devName;
125
126 wxDisplayInfo() { m_hmon = NULL; m_pDD2 = NULL; }
127 ~wxDisplayInfo() { if ( m_pDD2 ) m_pDD2->Release(); }
128 };
129
130 WX_DECLARE_OBJARRAY(wxDisplayInfo, wxDisplayInfoArray);
131 #include "wx/arrimpl.cpp"
132 WX_DEFINE_OBJARRAY(wxDisplayInfoArray);
133
134 // this module is used to cleanup gs_displays array
135 class wxDisplayModule : public wxModule
136 {
137 public:
138 virtual bool OnInit() { return true; }
139 virtual void OnExit();
140
141 DECLARE_DYNAMIC_CLASS(wxDisplayModule)
142 };
143
144 IMPLEMENT_DYNAMIC_CLASS(wxDisplayModule, wxModule)
145
146 // ----------------------------------------------------------------------------
147 // globals
148 // ----------------------------------------------------------------------------
149
150 // do we use DirectX?
151 static bool gs_useDirectX = false;
152
153 // dynamically resolved DirectDrawCreate()
154 static DirectDrawCreate_t gs_DirectDrawCreate = NULL;
155
156 // this is not really MT-unsafe as wxDisplay is only going to be used from the
157 // main thread, i.e. we consider that it's a GUI class and so don't protect it
158 static wxDisplayInfoArray *gs_displays = NULL;
159
160 // ===========================================================================
161 // implementation
162 // ===========================================================================
163
164 // ----------------------------------------------------------------------------
165 // callbacks for monitor/modes enumeration stuff
166 // ----------------------------------------------------------------------------
167
168 static BOOL CALLBACK wxmswMonitorEnumProc (
169 HMONITOR hMonitor, // handle to display monitor
170 HDC hdcMonitor, // handle to monitor-appropriate device context (NULL)
171 LPRECT lprcMonitor, // pointer to monitor intersection rectangle
172 LPARAM dwData // data passed from EnumDisplayMonitors (unused)
173 )
174 {
175 wxDisplayInfo *info = new wxDisplayInfo();
176
177 // we need hMonitor to be able to map display id to it which is needed for
178 // MonitorXXX() functions, in particular MonitorFromPoint()
179 info->m_hmon = hMonitor;
180
181 // we also store the display geometry
182 info->m_rect.SetX ( lprcMonitor->left );
183 info->m_rect.SetY ( lprcMonitor->top );
184 info->m_rect.SetWidth ( lprcMonitor->right - lprcMonitor->left );
185 info->m_rect.SetHeight ( lprcMonitor->bottom - lprcMonitor->top );
186
187 // now add this monitor to the array
188 gs_displays->Add(info);
189
190 // continue the enumeration
191 return true;
192 }
193
194 BOOL PASCAL
195 wxDDEnumExCallback(GUID *pGuid,
196 LPTSTR driverDescription,
197 LPTSTR driverName,
198 LPVOID lpContext,
199 HMONITOR hmon)
200 {
201 if ( pGuid )
202 {
203 wxDisplayInfo *info = new wxDisplayInfo();
204
205 info->m_hmon = hmon;
206 info->m_guid = *pGuid;
207 info->m_devName = driverName;
208
209 gs_displays->Add(info);
210 }
211 //else: we're called for the primary monitor, skip it
212
213 // continue the enumeration
214 return true;
215 }
216
217 HRESULT WINAPI wxDDEnumModesCallback(LPDDSURFACEDESC lpDDSurfaceDesc,
218 LPVOID lpContext)
219 {
220 // we need at least the mode size
221 static const int FLAGS_REQUIRED = DDSD_HEIGHT | DDSD_WIDTH;
222 if ( (lpDDSurfaceDesc->dwFlags & FLAGS_REQUIRED) == FLAGS_REQUIRED )
223 {
224 wxArrayVideoModes * const modes = (wxArrayVideoModes *)lpContext;
225
226 modes->Add(wxVideoMode(lpDDSurfaceDesc->dwWidth,
227 lpDDSurfaceDesc->dwHeight,
228 lpDDSurfaceDesc->ddpfPixelFormat.dwRGBBitCount,
229 lpDDSurfaceDesc->dwRefreshRate));
230 }
231
232 // continue the enumeration
233 return DDENUMRET_OK;
234 }
235
236 // ----------------------------------------------------------------------------
237 // local functions
238 // ----------------------------------------------------------------------------
239
240 // initialize gs_displays using DirectX functions
241 static bool DoInitDirectX()
242 {
243 // suppress the errors if ddraw.dll is not found
244 wxLog::EnableLogging(false);
245
246 wxDynamicLibrary dllDX(_T("ddraw.dll"));
247
248 wxLog::EnableLogging(true);
249
250 if ( !dllDX.IsLoaded() )
251 return false;
252
253 DirectDrawEnumerateEx_t pDDEnumEx = (DirectDrawEnumerateEx_t)
254 dllDX.GetSymbol(WINFUNC(DirectDrawEnumerateEx));
255 if ( !pDDEnumEx )
256 return false;
257
258 // we'll also need DirectDrawCreate() later, resolve it right now
259 gs_DirectDrawCreate = (DirectDrawCreate_t)
260 dllDX.GetSymbol(_T("DirectDrawCreate"));
261 if ( !gs_DirectDrawCreate )
262 return false;
263
264 if ( (*pDDEnumEx)(wxDDEnumExCallback,
265 NULL,
266 DDENUM_ATTACHEDSECONDARYDEVICES) != DD_OK )
267 {
268 return false;
269 }
270
271 // ok, it seems like we're going to use DirectDraw and so we're going to
272 // need ddraw.dll all the time, don't unload it
273 dllDX.Detach();
274
275 return true;
276 }
277
278 // initialize gs_displays using the standard Windows functions
279 static void DoInitStdWindows()
280 {
281 // enumerate all displays
282 if ( !::EnumDisplayMonitors(NULL, NULL, wxmswMonitorEnumProc, 0) )
283 {
284 wxLogLastError(wxT("EnumDisplayMonitors"));
285
286 // TODO: still create at least one (valid) entry in gs_displays for the
287 // primary display!
288 }
289 }
290
291 // this function must be called before accessing gs_displays array as it
292 // creates and initializes it
293 static void InitDisplays()
294 {
295 if ( gs_displays )
296 return;
297
298 gs_displays = new wxDisplayInfoArray();
299
300 if ( gs_useDirectX && !DoInitDirectX() )
301 {
302 // either we were told not to try to use DirectX or fall back to std
303 // functions if DirectX method failed
304 gs_useDirectX = false;
305
306 DoInitStdWindows();
307 }
308 }
309
310 // convert a DEVMODE to our wxVideoMode
311 wxVideoMode ConvertToVideoMode(const DEVMODE& dm)
312 {
313 // note that dmDisplayFrequency may be 0 or 1 meaning "standard one" and
314 // although 0 is ok for us we don't want to return modes with 1hz refresh
315 return wxVideoMode(dm.dmPelsWidth,
316 dm.dmPelsHeight,
317 dm.dmBitsPerPel,
318 dm.dmDisplayFrequency > 1 ? dm.dmDisplayFrequency : 0);
319 }
320
321 // emulation of ChangeDisplaySettingsEx() for Win95
322 LONG WINAPI ChangeDisplaySettingsExForWin95(LPCTSTR WXUNUSED(lpszDeviceName),
323 LPDEVMODE lpDevMode,
324 HWND WXUNUSED(hwnd),
325 DWORD dwFlags,
326 LPVOID WXUNUSED(lParam))
327 {
328 return ::ChangeDisplaySettings(lpDevMode, dwFlags);
329 }
330
331 // ----------------------------------------------------------------------------
332 // wxDisplayModule
333 // ----------------------------------------------------------------------------
334
335 void wxDisplayModule::OnExit()
336 {
337 delete gs_displays;
338 }
339
340 // ---------------------------------------------------------------------------
341 // wxDisplay
342 // ---------------------------------------------------------------------------
343
344 /* static */
345 void wxDisplay::UseDirectX(bool useDX)
346 {
347 wxCHECK_RET( !gs_displays, _T("it is too late to call UseDirectX") );
348
349 gs_useDirectX = useDX;
350 }
351
352 // helper of GetFromPoint() and GetFromWindow()
353 static int DisplayFromHMONITOR(HMONITOR hmon)
354 {
355 if ( hmon )
356 {
357 const size_t count = wxDisplay::GetCount();
358
359 for ( size_t n = 0; n < count; n++ )
360 {
361 if ( hmon == (*gs_displays)[n].m_hmon )
362 return n;
363 }
364 }
365
366 return wxNOT_FOUND;
367 }
368
369 /* static */
370 size_t wxDisplayBase::GetCount()
371 {
372 InitDisplays();
373
374 // I'm not sure if they really always return the same thing and if this is
375 // not true I'd like to know in which situation does it happen
376 wxASSERT_MSG( gs_displays->GetCount() == (size_t)::GetSystemMetrics(SM_CMONITORS),
377 _T("So how many displays does this system have?") );
378
379 return gs_displays->GetCount();
380 }
381
382 /* static */
383 int wxDisplayBase::GetFromPoint ( const wxPoint& pt )
384 {
385 POINT pt2;
386 pt2.x = pt.x;
387 pt2.y = pt.y;
388
389 return DisplayFromHMONITOR(::MonitorFromPoint(pt2, MONITOR_DEFAULTTONULL));
390 }
391
392 /* static */
393 int wxDisplayBase::GetFromWindow(wxWindow *window)
394 {
395 return DisplayFromHMONITOR
396 (
397 ::MonitorFromWindow(GetHwndOf(window), MONITOR_DEFAULTTONULL)
398 );
399 }
400
401 // ----------------------------------------------------------------------------
402 // wxDisplay ctor/dtor
403 // ----------------------------------------------------------------------------
404
405 wxDisplay::wxDisplay ( size_t n )
406 : wxDisplayBase ( n )
407 {
408 // if we do this in ctor we won't have to call it from all the member
409 // functions
410 InitDisplays();
411
412 if ( gs_useDirectX )
413 {
414 wxDisplayInfo& dpyInfo = (*gs_displays)[n];
415
416 LPDIRECTDRAW2& pDD2 = dpyInfo.m_pDD2;
417 if ( !pDD2 )
418 {
419 if ( !gs_DirectDrawCreate )
420 {
421 // what to do??
422 return;
423 }
424
425 IDirectDraw *pDD;
426 HRESULT hr = (*gs_DirectDrawCreate)(&dpyInfo.m_guid, &pDD, NULL);
427
428 if ( FAILED(hr) || !pDD )
429 {
430 // what to do??
431 wxLogApiError(_T("DirectDrawCreate"), hr);
432 }
433 else // got IDirectDraw, we want IDirectDraw2
434 {
435 hr = pDD->QueryInterface(wxIID_IDirectDraw2, (void **)&pDD2);
436 if ( FAILED(hr) || !pDD2 )
437 {
438 wxLogApiError(_T("IDirectDraw::QueryInterface(IDD2)"), hr);
439 }
440
441 pDD->Release();
442 }
443 }
444 //else: DirectDraw object corresponding to our display already exists
445
446 // increment its ref count to account for Release() in dtor
447 //
448 // NB: pDD2 will be only really Release()d when gs_displays is
449 // destroyed which is ok as we don't want to recreate DD objects
450 // all the time
451 pDD2->AddRef();
452 }
453 }
454
455 wxDisplay::~wxDisplay()
456 {
457 wxDisplayInfo& dpyInfo = (*gs_displays)[m_index];
458
459 LPDIRECTDRAW2& pDD2 = dpyInfo.m_pDD2;
460 if ( pDD2 )
461 {
462 pDD2->Release();
463 }
464 }
465
466 // ----------------------------------------------------------------------------
467 // wxDisplay simple accessors
468 // ----------------------------------------------------------------------------
469
470 bool wxDisplay::IsOk() const
471 {
472 return m_index < GetCount() &&
473 (!gs_useDirectX || (*gs_displays)[m_index].m_pDD2);
474 }
475
476 wxRect wxDisplay::GetGeometry() const
477 {
478 wxDisplayInfo& dpyInfo = (*gs_displays)[m_index];
479 wxRect& rect = dpyInfo.m_rect;
480 if ( !rect.width )
481 {
482 MONITORINFO monInfo;
483 wxZeroMemory(monInfo);
484 monInfo.cbSize = sizeof(monInfo);
485
486 if ( !::GetMonitorInfo(dpyInfo.m_hmon, &monInfo) )
487 {
488 wxLogLastError(_T("GetMonitorInfo"));
489 }
490 else
491 {
492 wxCopyRECTToRect(monInfo.rcMonitor, rect);
493 }
494 }
495
496 return rect;
497 }
498
499 wxString wxDisplay::GetName() const
500 {
501 wxDisplayInfo& dpyInfo = (*gs_displays)[m_index];
502 if ( dpyInfo.m_devName.empty() )
503 {
504 MONITORINFOEX monInfo;
505 wxZeroMemory(monInfo);
506 monInfo.cbSize = sizeof(monInfo);
507
508 if ( !::GetMonitorInfo(dpyInfo.m_hmon, &monInfo) )
509 {
510 wxLogLastError(_T("GetMonitorInfo"));
511 }
512 else
513 {
514 dpyInfo.m_devName = monInfo.szDevice;
515 }
516 }
517
518 return dpyInfo.m_devName;
519 }
520
521 wxString wxDisplay::GetNameForEnumSettings() const
522 {
523 int major, minor;
524 const bool isWin95 = wxGetOsVersion(&major, &minor) == wxWIN95 &&
525 major == 4 && minor == 0;
526
527 // the first parameter of EnumDisplaySettings() must be NULL under Win95
528 // according to MSDN but GetMonitorInfo() stub in multimon.h still returns
529 // something even in this case, so we have to correct this manually
530 wxString name;
531 if ( !isWin95 )
532 name = GetName();
533
534 return name;
535 }
536
537 // ----------------------------------------------------------------------------
538 // video modes enumeration
539 // ----------------------------------------------------------------------------
540
541 wxArrayVideoModes
542 wxDisplay::DoGetModesDirectX(const wxVideoMode& modeMatch) const
543 {
544 wxArrayVideoModes modes;
545
546 IDirectDraw2 *pDD = (*gs_displays)[m_index].m_pDD2;
547
548 if ( pDD )
549 {
550 HRESULT hr = pDD->EnumDisplayModes
551 (
552 DDEDM_REFRESHRATES,
553 NULL, // all modes (TODO: use modeMatch!)
554 &modes, // callback parameter
555 wxDDEnumModesCallback
556 );
557
558 if ( FAILED(hr) )
559 {
560 wxLogApiError(_T("IDirectDraw::EnumDisplayModes"), hr);
561 }
562 }
563
564 return modes;
565 }
566
567 wxArrayVideoModes
568 wxDisplay::DoGetModesWindows(const wxVideoMode& modeMatch) const
569 {
570 wxArrayVideoModes modes;
571
572 const wxString name = GetNameForEnumSettings();
573
574 const wxChar * const deviceName = name.empty() ? NULL : name.c_str();
575
576 DEVMODE dm;
577 for ( int iModeNum = 0;
578 ::EnumDisplaySettings(deviceName, iModeNum, &dm);
579 iModeNum++ )
580 {
581 const wxVideoMode mode = ConvertToVideoMode(dm);
582 if ( mode.Matches(modeMatch) )
583 {
584 modes.Add(mode);
585 }
586 }
587
588 return modes;
589 }
590
591 wxArrayVideoModes wxDisplay::GetModes(const wxVideoMode& modeMatch) const
592 {
593 return gs_useDirectX ? DoGetModesDirectX(modeMatch)
594 : DoGetModesWindows(modeMatch);
595 }
596
597 wxVideoMode wxDisplay::GetCurrentMode() const
598 {
599 wxVideoMode mode;
600
601 const wxString name = GetNameForEnumSettings();
602
603 DEVMODE dm;
604 if ( !::EnumDisplaySettings(name.empty() ? NULL : name.c_str(),
605 ENUM_CURRENT_SETTINGS,
606 &dm) )
607 {
608 wxLogLastError(_T("EnumDisplaySettings(ENUM_CURRENT_SETTINGS)"));
609 }
610 else
611 {
612 mode = ConvertToVideoMode(dm);
613 }
614
615 return mode;
616 }
617
618 // ----------------------------------------------------------------------------
619 // video mode switching
620 // ----------------------------------------------------------------------------
621
622 bool wxDisplay::DoChangeModeDirectX(const wxVideoMode& mode)
623 {
624 IDirectDraw2 *pDD = (*gs_displays)[m_index].m_pDD2;
625 if ( !pDD )
626 return false;
627
628 wxWindow *winTop = wxTheApp->GetTopWindow();
629 wxCHECK_MSG( winTop, false, _T("top level window required for DirectX") );
630
631 HRESULT hr = pDD->SetCooperativeLevel
632 (
633 GetHwndOf(winTop),
634 DDSCL_EXCLUSIVE | DDSCL_FULLSCREEN
635 );
636 if ( FAILED(hr) )
637 {
638 wxLogApiError(_T("IDirectDraw::SetCooperativeLevel"), hr);
639
640 return false;
641 }
642
643 hr = pDD->SetDisplayMode(mode.w, mode.h, mode.bpp, mode.refresh, 0);
644 if ( FAILED(hr) )
645 {
646 wxLogApiError(_T("IDirectDraw::SetDisplayMode"), hr);
647
648 return false;
649 }
650
651
652 return true;
653 }
654
655 bool wxDisplay::DoChangeModeWindows(const wxVideoMode& mode)
656 {
657 // prepare ChangeDisplaySettingsEx() parameters
658 DEVMODE dm,
659 *pDevMode;
660 int flags;
661
662 if ( mode == wxDefaultVideoMode )
663 {
664 // reset the video mode to default
665 pDevMode = NULL;
666 flags = 0;
667 }
668 else // change to the given mode
669 {
670 wxCHECK_MSG( mode.w && mode.h, false,
671 _T("at least the width and height must be specified") );
672
673 wxZeroMemory(dm);
674 dm.dmSize = sizeof(dm);
675 dm.dmFields = DM_PELSWIDTH | DM_PELSHEIGHT;
676 dm.dmPelsWidth = mode.w;
677 dm.dmPelsHeight = mode.h;
678
679 if ( mode.bpp )
680 {
681 dm.dmFields |= DM_BITSPERPEL;
682 dm.dmBitsPerPel = mode.bpp;
683 }
684
685 if ( mode.refresh )
686 {
687 dm.dmFields |= DM_DISPLAYFREQUENCY;
688 dm.dmDisplayFrequency = mode.refresh;
689 }
690
691 pDevMode = &dm;
692
693 flags = CDS_FULLSCREEN;
694 }
695
696
697 // get pointer to the function dynamically
698 //
699 // we're only called from the main thread, so it's ok to use static
700 // variable
701 static ChangeDisplaySettingsEx_t pfnChangeDisplaySettingsEx = NULL;
702 if ( !pfnChangeDisplaySettingsEx )
703 {
704 wxDynamicLibrary dllUser32(_T("user32.dll"));
705 if ( dllUser32.IsLoaded() )
706 {
707 pfnChangeDisplaySettingsEx = (ChangeDisplaySettingsEx_t)
708 dllUser32.GetSymbol(WINFUNC(ChangeDisplaySettingsEx));
709 }
710 //else: huh, no user32.dll??
711
712 if ( !pfnChangeDisplaySettingsEx )
713 {
714 // we must be under Win95 and so there is no multiple monitors
715 // support anyhow
716 pfnChangeDisplaySettingsEx = ChangeDisplaySettingsExForWin95;
717 }
718 }
719
720 // do change the mode
721 switch ( pfnChangeDisplaySettingsEx
722 (
723 GetName(), // display name
724 pDevMode, // dev mode or NULL to reset
725 NULL, // reserved
726 flags,
727 NULL // pointer to video parameters (not used)
728 ) )
729 {
730 case DISP_CHANGE_SUCCESSFUL:
731 // ok
732 return true;
733
734 case DISP_CHANGE_BADMODE:
735 // don't complain about this, this is the only "expected" error
736 break;
737
738 default:
739 wxFAIL_MSG( _T("unexpected ChangeDisplaySettingsEx() return value") );
740 }
741
742 return false;
743 }
744
745 bool wxDisplay::ChangeMode(const wxVideoMode& mode)
746 {
747 return gs_useDirectX ? DoChangeModeDirectX(mode)
748 : DoChangeModeWindows(mode);
749 }
750
751 #endif // wxUSE_DISPLAY
752