3 * Copyright (C) 2005 Imendio AB
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 * Boston, MA 02111-1307, USA.
23 #include "gdkscreen-quartz.h"
24 #include "gdkprivate-quartz.h"
27 static void gdk_screen_quartz_dispose (GObject *object);
28 static void gdk_screen_quartz_finalize (GObject *object);
29 static void gdk_screen_quartz_calculate_layout (GdkScreenQuartz *screen);
31 static void display_reconfiguration_callback (CGDirectDisplayID display,
32 CGDisplayChangeSummaryFlags flags,
35 G_DEFINE_TYPE (GdkScreenQuartz, _gdk_screen_quartz, GDK_TYPE_SCREEN);
38 _gdk_screen_quartz_class_init (GdkScreenQuartzClass *klass)
40 GObjectClass *object_class = G_OBJECT_CLASS (klass);
42 object_class->dispose = gdk_screen_quartz_dispose;
43 object_class->finalize = gdk_screen_quartz_finalize;
47 _gdk_screen_quartz_init (GdkScreenQuartz *screen_quartz)
49 GdkScreen *screen = GDK_SCREEN (screen_quartz);
51 gdk_screen_set_default_colormap (screen,
52 gdk_screen_get_system_colormap (screen));
54 gdk_screen_quartz_calculate_layout (screen_quartz);
56 CGDisplayRegisterReconfigurationCallback (display_reconfiguration_callback,
61 gdk_screen_quartz_dispose (GObject *object)
63 GdkScreenQuartz *screen = GDK_SCREEN_QUARTZ (object);
65 if (screen->default_colormap)
67 g_object_unref (screen->default_colormap);
68 screen->default_colormap = NULL;
71 if (screen->screen_changed_id)
73 g_source_remove (screen->screen_changed_id);
74 screen->screen_changed_id = 0;
77 CGDisplayRemoveReconfigurationCallback (display_reconfiguration_callback,
80 G_OBJECT_CLASS (_gdk_screen_quartz_parent_class)->dispose (object);
84 gdk_screen_quartz_screen_rects_free (GdkScreenQuartz *screen)
86 screen->n_screens = 0;
88 if (screen->screen_rects)
90 g_free (screen->screen_rects);
91 screen->screen_rects = NULL;
96 gdk_screen_quartz_finalize (GObject *object)
98 GdkScreenQuartz *screen = GDK_SCREEN_QUARTZ (object);
100 gdk_screen_quartz_screen_rects_free (screen);
105 gdk_screen_quartz_calculate_layout (GdkScreenQuartz *screen)
111 GDK_QUARTZ_ALLOC_POOL;
113 gdk_screen_quartz_screen_rects_free (screen);
115 array = [NSScreen screens];
117 /* FIXME: For now we only support screen layouts where the screens are laid
118 * out horizontally. Mac OS X also supports laying out the screens vertically
119 * and the screens having "non-standard" offsets from eachother. In the
120 * future we need a much more sophiscated algorithm to translate these
121 * layouts to GDK coordinate space and GDK screen layout.
126 for (i = 0; i < [array count]; i++)
128 NSRect rect = [[array objectAtIndex:i] frame];
130 screen->width += rect.size.width;
131 screen->height = MAX (screen->height, rect.size.height);
134 screen->n_screens = [array count];
135 screen->screen_rects = g_new0 (GdkRectangle, screen->n_screens);
137 /* Find the monitor with the largest height. All monitors should be
138 * offset to this one in the GDK screen space instead of offset to
139 * the screen with the menu bar.
141 largest_rect = [[array objectAtIndex:0] frame];
142 for (i = 1; i < [array count]; i++)
144 NSRect rect = [[array objectAtIndex:i] frame];
146 if (rect.size.height > largest_rect.size.height)
147 largest_rect = [[array objectAtIndex:i] frame];
150 for (i = 0; i < screen->n_screens; i++)
155 nsscreen = [array objectAtIndex:i];
156 rect = [nsscreen frame];
158 screen->screen_rects[i].x = rect.origin.x;
159 screen->screen_rects[i].width = rect.size.width;
160 screen->screen_rects[i].height = rect.size.height;
162 if (largest_rect.size.height - rect.size.height == 0)
163 screen->screen_rects[i].y = 0;
165 screen->screen_rects[i].y = largest_rect.size.height - rect.size.height + largest_rect.origin.y;
168 GDK_QUARTZ_RELEASE_POOL;
173 process_display_reconfiguration (GdkScreenQuartz *screen)
177 width = gdk_screen_get_width (GDK_SCREEN (screen));
178 height = gdk_screen_get_height (GDK_SCREEN (screen));
180 gdk_screen_quartz_calculate_layout (GDK_SCREEN_QUARTZ (screen));
182 if (width != gdk_screen_get_width (GDK_SCREEN (screen))
183 || height != gdk_screen_get_height (GDK_SCREEN (screen)))
184 g_signal_emit_by_name (_gdk_screen, "size-changed");
188 screen_changed_idle (gpointer data)
190 GdkScreenQuartz *screen = data;
192 process_display_reconfiguration (data);
194 screen->screen_changed_id = 0;
200 display_reconfiguration_callback (CGDirectDisplayID display,
201 CGDisplayChangeSummaryFlags flags,
204 GdkScreenQuartz *screen = userInfo;
206 if (flags & kCGDisplayBeginConfigurationFlag)
208 /* Ignore the begin configuration signal. */
210 /* FIXME: We can most probably use this flag to properly
211 * emit monitors-changed.
217 /* At this point Cocoa does not know about the new screen data
218 * yet, so we delay our refresh into an idle handler.
221 if (!screen->screen_changed_id)
222 screen->screen_changed_id = gdk_threads_add_idle (screen_changed_idle,
228 _gdk_screen_quartz_new (void)
230 return g_object_new (GDK_TYPE_SCREEN_QUARTZ, NULL);
234 gdk_screen_get_display (GdkScreen *screen)
236 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
243 gdk_screen_get_root_window (GdkScreen *screen)
245 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
251 gdk_screen_get_number (GdkScreen *screen)
253 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
259 _gdk_windowing_substitute_screen_number (const gchar *display_name,
262 if (screen_number != 0)
265 return g_strdup (display_name);
269 gdk_screen_get_default_colormap (GdkScreen *screen)
271 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
273 return GDK_SCREEN_QUARTZ (screen)->default_colormap;
277 gdk_screen_set_default_colormap (GdkScreen *screen,
278 GdkColormap *colormap)
280 GdkColormap *old_colormap;
282 g_return_if_fail (GDK_IS_SCREEN (screen));
283 g_return_if_fail (GDK_IS_COLORMAP (colormap));
285 old_colormap = GDK_SCREEN_QUARTZ (screen)->default_colormap;
287 GDK_SCREEN_QUARTZ (screen)->default_colormap = g_object_ref (colormap);
290 g_object_unref (old_colormap);
294 gdk_screen_get_width (GdkScreen *screen)
296 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
298 return GDK_SCREEN_QUARTZ (screen)->width;
302 gdk_screen_get_height (GdkScreen *screen)
304 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
306 return GDK_SCREEN_QUARTZ (screen)->height;
310 get_mm_from_pixels (NSScreen *screen, int pixels)
312 /* userSpaceScaleFactor is in "pixels per point",
313 * 72 is the number of points per inch,
314 * and 25.4 is the number of millimeters per inch.
316 #if MAC_OS_X_VERSION_MAX_ALLOWED > MAC_OS_X_VERSION_10_3
317 float dpi = [screen userSpaceScaleFactor] * 72.0;
319 float dpi = 96.0 / 72.0;
322 return (pixels / dpi) * 25.4;
326 get_nsscreen_for_monitor (gint monitor_num)
331 GDK_QUARTZ_ALLOC_POOL;
333 array = [NSScreen screens];
334 screen = [array objectAtIndex:monitor_num];
336 GDK_QUARTZ_RELEASE_POOL;
342 gdk_screen_get_width_mm (GdkScreen *screen)
344 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
346 return get_mm_from_pixels (get_nsscreen_for_monitor (0),
347 GDK_SCREEN_QUARTZ (screen)->width);
351 gdk_screen_get_height_mm (GdkScreen *screen)
353 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
355 return get_mm_from_pixels (get_nsscreen_for_monitor (0),
356 GDK_SCREEN_QUARTZ (screen)->height);
360 gdk_screen_get_n_monitors (GdkScreen *screen)
362 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
364 return GDK_SCREEN_QUARTZ (screen)->n_screens;
368 gdk_screen_get_monitor_width_mm (GdkScreen *screen,
371 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
372 g_return_val_if_fail (monitor_num < gdk_screen_get_n_monitors (screen), 0);
373 g_return_val_if_fail (monitor_num >= 0, 0);
375 return get_mm_from_pixels (get_nsscreen_for_monitor (monitor_num),
376 GDK_SCREEN_QUARTZ (screen)->screen_rects[monitor_num].width);
380 gdk_screen_get_monitor_height_mm (GdkScreen *screen,
383 g_return_val_if_fail (GDK_IS_SCREEN (screen), 0);
384 g_return_val_if_fail (monitor_num < gdk_screen_get_n_monitors (screen), 0);
385 g_return_val_if_fail (monitor_num >= 0, 0);
387 return get_mm_from_pixels (get_nsscreen_for_monitor (monitor_num),
388 GDK_SCREEN_QUARTZ (screen)->screen_rects[monitor_num].height);
392 gdk_screen_get_monitor_plug_name (GdkScreen *screen,
395 /* FIXME: Is there some useful name we could use here? */
400 gdk_screen_get_monitor_geometry (GdkScreen *screen,
404 g_return_if_fail (GDK_IS_SCREEN (screen));
405 g_return_if_fail (monitor_num < gdk_screen_get_n_monitors (screen));
406 g_return_if_fail (monitor_num >= 0);
408 *dest = GDK_SCREEN_QUARTZ (screen)->screen_rects[monitor_num];
412 gdk_screen_make_display_name (GdkScreen *screen)
414 return g_strdup (gdk_display_get_name (_gdk_display));
418 gdk_screen_get_active_window (GdkScreen *screen)
420 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
426 gdk_screen_get_window_stack (GdkScreen *screen)
428 g_return_val_if_fail (GDK_IS_SCREEN (screen), NULL);
434 gdk_screen_is_composited (GdkScreen *screen)
436 g_return_val_if_fail (GDK_IS_SCREEN (screen), FALSE);