3 * Copyright (C) 2005 Imendio AB
4 * Copyright (C) 2009 Kristian Rietveld <kris@gtk.org>
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the
18 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 * Boston, MA 02111-1307, USA.
24 #include "gdkscreen-quartz.h"
25 #include "gdkprivate-quartz.h"
28 static void gdk_screen_quartz_dispose (GObject *object);
29 static void gdk_screen_quartz_finalize (GObject *object);
30 static void gdk_screen_quartz_calculate_layout (GdkScreenQuartz *screen);
32 static void display_reconfiguration_callback (CGDirectDisplayID display,
33 CGDisplayChangeSummaryFlags flags,
36 G_DEFINE_TYPE (GdkScreenQuartz, _gdk_screen_quartz, GDK_TYPE_SCREEN);
39 _gdk_screen_quartz_class_init (GdkScreenQuartzClass *klass)
41 GObjectClass *object_class = G_OBJECT_CLASS (klass);
43 object_class->dispose = gdk_screen_quartz_dispose;
44 object_class->finalize = gdk_screen_quartz_finalize;
48 _gdk_screen_quartz_init (GdkScreenQuartz *screen_quartz)
50 GdkScreen *screen = GDK_SCREEN (screen_quartz);
53 gdk_screen_set_default_colormap (screen,
54 gdk_screen_get_system_colormap (screen));
56 nsscreen = [[NSScreen screens] objectAtIndex:0];
57 gdk_screen_set_resolution (screen,
58 72.0 * [nsscreen userSpaceScaleFactor]);
60 gdk_screen_quartz_calculate_layout (screen_quartz);
62 CGDisplayRegisterReconfigurationCallback (display_reconfiguration_callback,
65 screen_quartz->emit_monitors_changed = FALSE;
69 gdk_screen_quartz_dispose (GObject *object)
71 GdkScreenQuartz *screen = GDK_SCREEN_QUARTZ (object);
73 if (screen->default_colormap)
75 g_object_unref (screen->default_colormap);
76 screen->default_colormap = NULL;
79 if (screen->screen_changed_id)
81 g_source_remove (screen->screen_changed_id);
82 screen->screen_changed_id = 0;
85 CGDisplayRemoveReconfigurationCallback (display_reconfiguration_callback,
88 G_OBJECT_CLASS (_gdk_screen_quartz_parent_class)->dispose (object);
92 gdk_screen_quartz_screen_rects_free (GdkScreenQuartz *screen)
94 screen->n_screens = 0;
96 if (screen->screen_rects)
98 g_free (screen->screen_rects);
99 screen->screen_rects = NULL;
104 gdk_screen_quartz_finalize (GObject *object)
106 GdkScreenQuartz *screen = GDK_SCREEN_QUARTZ (object);
108 gdk_screen_quartz_screen_rects_free (screen);
113 gdk_screen_quartz_calculate_layout (GdkScreenQuartz *screen)
119 GDK_QUARTZ_ALLOC_POOL;
121 gdk_screen_quartz_screen_rects_free (screen);
123 array = [NSScreen screens];
131 /* We determine the minimum and maximum x and y coordinates
132 * covered by the monitors. From this we can deduce the width
133 * and height of the root screen.
135 for (i = 0; i < [array count]; i++)
137 NSRect rect = [[array objectAtIndex:i] frame];
139 screen->min_x = MIN (screen->min_x, rect.origin.x);
140 max_x = MAX (max_x, rect.origin.x + rect.size.width);
142 screen->min_y = MIN (screen->min_y, rect.origin.y);
143 max_y = MAX (max_y, rect.origin.y + rect.size.height);
146 screen->width = max_x - screen->min_x;
147 screen->height = max_y - screen->min_y;
149 screen->n_screens = [array count];
150 screen->screen_rects = g_new0 (GdkRectangle, screen->n_screens);
152 for (i = 0; i < screen->n_screens; i++)
157 nsscreen = [array objectAtIndex:i];
158 rect = [nsscreen frame];
160 screen->screen_rects[i].x = rect.origin.x - screen->min_x;
161 screen->screen_rects[i].y
162 = screen->height - (rect.origin.y + rect.size.height) + screen->min_y;
163 screen->screen_rects[i].width = rect.size.width;
164 screen->screen_rects[i].height = rect.size.height;
167 GDK_QUARTZ_RELEASE_POOL;
172 process_display_reconfiguration (GdkScreenQuartz *screen)
176 width = gdk_screen_get_width (GDK_SCREEN (screen));
177 height = gdk_screen_get_height (GDK_SCREEN (screen));
179 gdk_screen_quartz_calculate_layout (GDK_SCREEN_QUARTZ (screen));
181 _gdk_windowing_update_window_sizes (GDK_SCREEN (screen));
183 if (screen->emit_monitors_changed)
185 g_signal_emit_by_name (screen, "monitors-changed");
186 screen->emit_monitors_changed = FALSE;
189 if (width != gdk_screen_get_width (GDK_SCREEN (screen))
190 || height != gdk_screen_get_height (GDK_SCREEN (screen)))
191 g_signal_emit_by_name (screen, "size-changed");
195 screen_changed_idle (gpointer data)
197 GdkScreenQuartz *screen = data;
199 process_display_reconfiguration (data);
201 screen->screen_changed_id = 0;
207 display_reconfiguration_callback (CGDirectDisplayID display,
208 CGDisplayChangeSummaryFlags flags,
211 GdkScreenQuartz *screen = userInfo;
213 if (flags & kCGDisplayBeginConfigurationFlag)
215 /* Ignore the begin configuration signal. */
220 /* We save information about the changes, so we can emit
221 * ::monitors-changed when appropriate. This signal must be
222 * emitted when the number, size of position of one of the
225 if (flags & kCGDisplayMovedFlag
226 || flags & kCGDisplayAddFlag
227 || flags & kCGDisplayRemoveFlag
228 || flags & kCGDisplayEnabledFlag
229 || flags & kCGDisplayDisabledFlag)
230 screen->emit_monitors_changed = TRUE;
232 /* At this point Cocoa does not know about the new screen data
233 * yet, so we delay our refresh into an idle handler.
235 if (!screen->screen_changed_id)
236 screen->screen_changed_id = gdk_threads_add_idle (screen_changed_idle,
242 _gdk_screen_quartz_new (void)
244 return g_object_new (GDK_TYPE_SCREEN_QUARTZ, NULL);
248 gdk_screen_get_display (GdkScreen *screen)
250 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
257 gdk_screen_get_root_window (GdkScreen *screen)
259 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
265 gdk_screen_get_number (GdkScreen *screen)
267 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
273 _gdk_windowing_substitute_screen_number (const gchar *display_name,
276 if (screen_number != 0)
279 return g_strdup (display_name);
283 gdk_screen_get_default_colormap (GdkScreen *screen)
285 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
287 return GDK_SCREEN_QUARTZ (screen)->default_colormap;
291 gdk_screen_set_default_colormap (GdkScreen *screen,
292 GdkColormap *colormap)
294 GdkColormap *old_colormap;
296 g_return_if_fail (GDK_IS_SCREEN (screen));
297 g_return_if_fail (GDK_IS_COLORMAP (colormap));
299 old_colormap = GDK_SCREEN_QUARTZ (screen)->default_colormap;
301 GDK_SCREEN_QUARTZ (screen)->default_colormap = g_object_ref (colormap);
304 g_object_unref (old_colormap);
308 gdk_screen_get_width (GdkScreen *screen)
310 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
312 return GDK_SCREEN_QUARTZ (screen)->width;
316 gdk_screen_get_height (GdkScreen *screen)
318 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
320 return GDK_SCREEN_QUARTZ (screen)->height;
324 get_mm_from_pixels (NSScreen *screen, int pixels)
326 /* userSpaceScaleFactor is in "pixels per point",
327 * 72 is the number of points per inch,
328 * and 25.4 is the number of millimeters per inch.
330 #if MAC_OS_X_VERSION_MAX_ALLOWED > MAC_OS_X_VERSION_10_3
331 float dpi = [screen userSpaceScaleFactor] * 72.0;
333 float dpi = 96.0 / 72.0;
336 return (pixels / dpi) * 25.4;
340 get_nsscreen_for_monitor (gint monitor_num)
345 GDK_QUARTZ_ALLOC_POOL;
347 array = [NSScreen screens];
348 screen = [array objectAtIndex:monitor_num];
350 GDK_QUARTZ_RELEASE_POOL;
356 gdk_screen_get_width_mm (GdkScreen *screen)
358 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
360 return get_mm_from_pixels (get_nsscreen_for_monitor (0),
361 GDK_SCREEN_QUARTZ (screen)->width);
365 gdk_screen_get_height_mm (GdkScreen *screen)
367 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
369 return get_mm_from_pixels (get_nsscreen_for_monitor (0),
370 GDK_SCREEN_QUARTZ (screen)->height);
374 gdk_screen_get_n_monitors (GdkScreen *screen)
376 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
378 return GDK_SCREEN_QUARTZ (screen)->n_screens;
382 gdk_screen_get_monitor_width_mm (GdkScreen *screen,
385 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
386 g_return_val_if_fail (monitor_num < gdk_screen_get_n_monitors (screen), 0);
387 g_return_val_if_fail (monitor_num >= 0, 0);
389 return get_mm_from_pixels (get_nsscreen_for_monitor (monitor_num),
390 GDK_SCREEN_QUARTZ (screen)->screen_rects[monitor_num].width);
394 gdk_screen_get_monitor_height_mm (GdkScreen *screen,
397 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
398 g_return_val_if_fail (monitor_num < gdk_screen_get_n_monitors (screen), 0);
399 g_return_val_if_fail (monitor_num >= 0, 0);
401 return get_mm_from_pixels (get_nsscreen_for_monitor (monitor_num),
402 GDK_SCREEN_QUARTZ (screen)->screen_rects[monitor_num].height);
406 gdk_screen_get_monitor_plug_name (GdkScreen *screen,
409 /* FIXME: Is there some useful name we could use here? */
414 gdk_screen_get_monitor_geometry (GdkScreen *screen,
418 g_return_if_fail (GDK_IS_SCREEN (screen));
419 g_return_if_fail (monitor_num < gdk_screen_get_n_monitors (screen));
420 g_return_if_fail (monitor_num >= 0);
422 *dest = GDK_SCREEN_QUARTZ (screen)->screen_rects[monitor_num];
426 gdk_screen_make_display_name (GdkScreen *screen)
428 return g_strdup (gdk_display_get_name (_gdk_display));
432 gdk_screen_get_active_window (GdkScreen *screen)
434 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
440 gdk_screen_get_window_stack (GdkScreen *screen)
442 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
448 gdk_screen_is_composited (GdkScreen *screen)
450 g_return_val_if_fail (GDK_IS_SCREEN (screen), FALSE);