]> git.saurik.com Git - wxWidgets.git/blob - src/osx/core/display.cpp
Fix crash in wxColour ctor from NSColor in wxOSX/Cocoa.
[wxWidgets.git] / src / osx / core / display.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/core/display.cpp
3 // Purpose: Mac implementation of wxDisplay class
4 // Author: Ryan Norton & Brian Victor
5 // Modified by: Royce Mitchell III, Vadim Zeitlin
6 // Created: 06/21/02
7 // RCS-ID: $Id$
8 // Copyright: (c) wxWidgets team
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 #include "wx/wxprec.h"
21
22 #ifdef __BORLANDC__
23 #pragma hdrstop
24 #endif
25
26 #if wxUSE_DISPLAY
27
28 #include "wx/display.h"
29
30 #ifndef WX_PRECOMP
31 #include "wx/dynarray.h"
32 #include "wx/log.h"
33 #include "wx/string.h"
34 #include "wx/gdicmn.h"
35 #endif
36
37 #include "wx/display_impl.h"
38 #include "wx/osx/private.h"
39
40 #if wxOSX_USE_COCOA_OR_CARBON
41
42 // ----------------------------------------------------------------------------
43 // display classes implementation
44 // ----------------------------------------------------------------------------
45
46 class wxDisplayImplMacOSX : public wxDisplayImpl
47 {
48 public:
49 wxDisplayImplMacOSX(unsigned n, CGDirectDisplayID id)
50 : wxDisplayImpl(n),
51 m_id(id)
52 {
53 }
54
55 virtual wxRect GetGeometry() const;
56 virtual wxRect GetClientArea() const;
57 virtual wxString GetName() const { return wxString(); }
58
59 virtual wxArrayVideoModes GetModes(const wxVideoMode& mode) const;
60 virtual wxVideoMode GetCurrentMode() const;
61 virtual bool ChangeMode(const wxVideoMode& mode);
62
63 virtual bool IsPrimary() const;
64
65 private:
66 CGDirectDisplayID m_id;
67
68 wxDECLARE_NO_COPY_CLASS(wxDisplayImplMacOSX);
69 };
70
71 class wxDisplayFactoryMacOSX : public wxDisplayFactory
72 {
73 public:
74 wxDisplayFactoryMacOSX() {}
75
76 virtual wxDisplayImpl *CreateDisplay(unsigned n);
77 virtual unsigned GetCount();
78 virtual int GetFromPoint(const wxPoint& pt);
79
80 protected:
81 wxDECLARE_NO_COPY_CLASS(wxDisplayFactoryMacOSX);
82 };
83
84 // ============================================================================
85 // wxDisplayFactoryMacOSX implementation
86 // ============================================================================
87
88 // gets all displays that are not mirror displays
89
90 static CGDisplayErr wxOSXGetDisplayList(CGDisplayCount maxDisplays,
91 CGDirectDisplayID *displays,
92 CGDisplayCount *displayCount)
93 {
94 CGDisplayErr error = kCGErrorSuccess;
95 CGDisplayCount onlineCount;
96
97 error = CGGetOnlineDisplayList(0,NULL,&onlineCount);
98 if ( error == kCGErrorSuccess )
99 {
100 *displayCount = 0;
101 if ( onlineCount > 0 )
102 {
103 CGDirectDisplayID *onlineDisplays = new CGDirectDisplayID[onlineCount];
104 error = CGGetOnlineDisplayList(onlineCount,onlineDisplays,&onlineCount);
105 if ( error == kCGErrorSuccess )
106 {
107 for ( CGDisplayCount i = 0; i < onlineCount; ++i )
108 {
109 if ( CGDisplayMirrorsDisplay(onlineDisplays[i]) != kCGNullDirectDisplay )
110 continue;
111
112 if ( displays == NULL )
113 *displayCount += 1;
114 else
115 {
116 if ( *displayCount < maxDisplays )
117 {
118 displays[*displayCount] = onlineDisplays[i];
119 *displayCount += 1;
120 }
121 }
122 }
123 }
124 delete[] onlineDisplays;
125 }
126
127 }
128 return error;
129 }
130
131 unsigned wxDisplayFactoryMacOSX::GetCount()
132 {
133 CGDisplayCount count;
134 CGDisplayErr err = wxOSXGetDisplayList(0, NULL, &count);
135
136 wxCHECK_MSG( err == CGDisplayNoErr, 0, "wxOSXGetDisplayList() failed" );
137
138 return count;
139 }
140
141 int wxDisplayFactoryMacOSX::GetFromPoint(const wxPoint& p)
142 {
143 CGPoint thePoint = {(float)p.x, (float)p.y};
144 CGDirectDisplayID theID;
145 CGDisplayCount theCount;
146 CGDisplayErr err = CGGetDisplaysWithPoint(thePoint, 1, &theID, &theCount);
147 wxASSERT(err == CGDisplayNoErr);
148
149 int nWhich = wxNOT_FOUND;
150
151 if (theCount)
152 {
153 theCount = GetCount();
154 CGDirectDisplayID* theIDs = new CGDirectDisplayID[theCount];
155 err = wxOSXGetDisplayList(theCount, theIDs, &theCount);
156 wxASSERT(err == CGDisplayNoErr);
157
158 for (nWhich = 0; nWhich < (int) theCount; ++nWhich)
159 {
160 if (theIDs[nWhich] == theID)
161 break;
162 }
163
164 delete [] theIDs;
165
166 if (nWhich == (int) theCount)
167 {
168 wxFAIL_MSG(wxT("Failed to find display in display list"));
169 nWhich = wxNOT_FOUND;
170 }
171 }
172
173 return nWhich;
174 }
175
176 wxDisplayImpl *wxDisplayFactoryMacOSX::CreateDisplay(unsigned n)
177 {
178 CGDisplayCount theCount = GetCount();
179 CGDirectDisplayID* theIDs = new CGDirectDisplayID[theCount];
180
181 CGDisplayErr err = wxOSXGetDisplayList(theCount, theIDs, &theCount);
182 wxCHECK_MSG( err == CGDisplayNoErr, NULL, "wxOSXGetDisplayList() failed" );
183
184 wxASSERT( n < theCount );
185
186 wxDisplayImplMacOSX *display = new wxDisplayImplMacOSX(n, theIDs[n]);
187
188 delete [] theIDs;
189
190 return display;
191 }
192
193 // ============================================================================
194 // wxDisplayImplMacOSX implementation
195 // ============================================================================
196
197 bool wxDisplayImplMacOSX::IsPrimary() const
198 {
199 return CGDisplayIsMain(m_id);
200 }
201
202 wxRect wxDisplayImplMacOSX::GetGeometry() const
203 {
204 CGRect theRect = CGDisplayBounds(m_id);
205 return wxRect( (int)theRect.origin.x,
206 (int)theRect.origin.y,
207 (int)theRect.size.width,
208 (int)theRect.size.height ); //floats
209 }
210
211 wxRect wxDisplayImplMacOSX::GetClientArea() const
212 {
213 // VZ: I don't know how to get client area for arbitrary display but
214 // wxGetClientDisplayRect() does work correctly for at least the main
215 // one (TODO: do it correctly for the other displays too)
216 if ( IsPrimary() )
217 return wxGetClientDisplayRect();
218
219 return wxDisplayImpl::GetClientArea();
220 }
221
222 static int wxCFDictKeyToInt( CFDictionaryRef desc, CFStringRef key )
223 {
224 CFNumberRef value = (CFNumberRef) CFDictionaryGetValue( desc, key );
225 if (value == NULL)
226 return 0;
227
228 int num = 0;
229 CFNumberGetValue( value, kCFNumberIntType, &num );
230
231 return num;
232 }
233
234 wxArrayVideoModes wxDisplayImplMacOSX::GetModes(const wxVideoMode& mode) const
235 {
236 wxArrayVideoModes resultModes;
237
238 CFArrayRef theArray = CGDisplayAvailableModes( m_id );
239
240 for (CFIndex i = 0; i < CFArrayGetCount(theArray); ++i)
241 {
242 CFDictionaryRef theValue = (CFDictionaryRef) CFArrayGetValueAtIndex( theArray, i );
243
244 wxVideoMode theMode(
245 wxCFDictKeyToInt( theValue, kCGDisplayWidth ),
246 wxCFDictKeyToInt( theValue, kCGDisplayHeight ),
247 wxCFDictKeyToInt( theValue, kCGDisplayBitsPerPixel ),
248 wxCFDictKeyToInt( theValue, kCGDisplayRefreshRate ));
249
250 if (theMode.Matches( mode ))
251 resultModes.Add( theMode );
252 }
253
254 return resultModes;
255 }
256
257 wxVideoMode wxDisplayImplMacOSX::GetCurrentMode() const
258 {
259 CFDictionaryRef theValue = CGDisplayCurrentMode( m_id );
260
261 return wxVideoMode(
262 wxCFDictKeyToInt( theValue, kCGDisplayWidth ),
263 wxCFDictKeyToInt( theValue, kCGDisplayHeight ),
264 wxCFDictKeyToInt( theValue, kCGDisplayBitsPerPixel ),
265 wxCFDictKeyToInt( theValue, kCGDisplayRefreshRate ));
266 }
267
268 bool wxDisplayImplMacOSX::ChangeMode( const wxVideoMode& mode )
269 {
270 // Changing to default mode (wxDefaultVideoMode) doesn't
271 // work because we don't have access to the system's 'scrn'
272 // resource which holds the user's mode which the system
273 // will return to after this app is done
274 boolean_t bExactMatch;
275 CFDictionaryRef theCGMode = CGDisplayBestModeForParametersAndRefreshRate(
276 m_id,
277 (size_t)mode.GetDepth(),
278 (size_t)mode.GetWidth(),
279 (size_t)mode.GetHeight(),
280 (double)mode.GetRefresh(),
281 &bExactMatch );
282
283 bool bOK = bExactMatch;
284
285 if (bOK)
286 bOK = CGDisplaySwitchToMode( m_id, theCGMode ) == CGDisplayNoErr;
287
288 return bOK;
289 }
290
291 // ============================================================================
292 // wxDisplay::CreateFactory()
293 // ============================================================================
294
295 /* static */ wxDisplayFactory *wxDisplay::CreateFactory()
296 {
297 return new wxDisplayFactoryMacOSX;
298 }
299
300 #else
301
302 /* static */ wxDisplayFactory *wxDisplay::CreateFactory()
303 {
304 return new wxDisplayFactorySingle;
305 }
306
307 #endif
308
309 #endif // wxUSE_DISPLAY