/* gdkdisplay-quartz.c * * Copyright (C) 2005 Imendio AB * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ #include "config.h" #include #include #include #include "gdkprivate-quartz.h" #include "gdkquartzscreen.h" #include "gdkquartzwindow.h" #include "gdkquartzdisplay.h" #include "gdkquartzdevicemanager-core.h" #include "gdkscreen.h" #include "gdkmonitorprivate.h" #include "gdkdisplay-quartz.h" #include "gdkmonitor-quartz.h" static gint MONITORS_CHANGED = 0; static void display_reconfiguration_callback (CGDirectDisplayID display, CGDisplayChangeSummaryFlags flags, void *data); static GdkWindow * gdk_quartz_display_get_default_group (GdkDisplay *display) { /* X11-only. */ return NULL; } GdkDeviceManager * _gdk_device_manager_new (GdkDisplay *display) { return g_object_new (GDK_TYPE_QUARTZ_DEVICE_MANAGER_CORE, "display", display, NULL); } GdkDisplay * _gdk_quartz_display_open (const gchar *display_name) { if (_gdk_display != NULL) return NULL; /* Initialize application */ [NSApplication sharedApplication]; _gdk_display = g_object_new (gdk_quartz_display_get_type (), NULL); _gdk_display->device_manager = _gdk_device_manager_new (_gdk_display); _gdk_screen = g_object_new (gdk_quartz_screen_get_type (), NULL); _gdk_quartz_screen_init_visuals (_gdk_screen); _gdk_quartz_window_init_windowing (_gdk_display, _gdk_screen); _gdk_quartz_events_init (); #if 0 /* FIXME: Remove the #if 0 when we have these functions */ _gdk_quartz_dnd_init (); #endif g_signal_emit_by_name (_gdk_display, "opened"); return _gdk_display; } static const gchar * gdk_quartz_display_get_name (GdkDisplay *display) { static gchar *display_name = NULL; if (!display_name) { GDK_QUARTZ_ALLOC_POOL; display_name = g_strdup ([[[NSHost currentHost] name] UTF8String]); GDK_QUARTZ_RELEASE_POOL; } return display_name; } static GdkScreen * gdk_quartz_display_get_default_screen (GdkDisplay *display) { return _gdk_screen; } static void gdk_quartz_display_beep (GdkDisplay *display) { g_return_if_fail (GDK_IS_DISPLAY (display)); NSBeep(); } static void gdk_quartz_display_sync (GdkDisplay *display) { /* Not needed. */ } static void gdk_quartz_display_flush (GdkDisplay *display) { /* Not needed. */ } static gboolean gdk_quartz_display_supports_selection_notification (GdkDisplay *display) { g_return_val_if_fail (GDK_IS_DISPLAY (display), FALSE); /* X11-only. */ return FALSE; } static gboolean gdk_quartz_display_request_selection_notification (GdkDisplay *display, GdkAtom selection) { /* X11-only. */ return FALSE; } static gboolean gdk_quartz_display_supports_clipboard_persistence (GdkDisplay *display) { /* X11-only */ return FALSE; } static gboolean gdk_quartz_display_supports_shapes (GdkDisplay *display) { /* Not needed, nothing ever calls this.*/ return FALSE; } static gboolean gdk_quartz_display_supports_input_shapes (GdkDisplay *display) { /* Not needed, nothign ever calls this. */ return FALSE; } static void gdk_quartz_display_store_clipboard (GdkDisplay *display, GdkWindow *clipboard_window, guint32 time_, const GdkAtom *targets, gint n_targets) { /* MacOS persists pasteboard items automatically, no application * action is required. */ } static gboolean gdk_quartz_display_supports_composite (GdkDisplay *display) { /* X11-only. */ return FALSE; } static gulong gdk_quartz_display_get_next_serial (GdkDisplay *display) { /* X11-only. */ return 0; } static void gdk_quartz_display_notify_startup_complete (GdkDisplay *display, const gchar *startup_id) { /* This should call finishLaunching, but doing so causes Quartz to throw * "_createMenuRef called with existing principal MenuRef already" * " associated with menu". [NSApp finishLaunching]; */ } static void gdk_quartz_display_push_error_trap (GdkDisplay *display) { /* X11-only. */ } static gint gdk_quartz_display_pop_error_trap (GdkDisplay *display, gboolean ignore) { /* X11 only. */ return 0; } /* The display monitor list comprises all of the CGDisplays connected to the system, some of which may not be drawable either because they're asleep or are mirroring another monitor. The NSScreens array contains only the monitors that are currently drawable and we use the index of the screens array placing GdkNSViews, so we'll use the same for determining the number of monitors and indexing them. */ int get_active_displays (CGDirectDisplayID **screens) { unsigned int displays = 0; CGGetActiveDisplayList (0, NULL, &displays); if (screens) { *screens = g_new0 (CGDirectDisplayID, displays); CGGetActiveDisplayList (displays, *screens, &displays); } return displays; } static void configure_monitor (GdkMonitor *monitor) { GdkQuartzMonitor *quartz_monitor = GDK_QUARTZ_MONITOR (monitor); CGSize disp_size = CGDisplayScreenSize (quartz_monitor->id); gint width = (int)trunc (disp_size.width); gint height = (int)trunc (disp_size.height); CGRect disp_bounds = CGDisplayBounds (quartz_monitor->id); GdkRectangle disp_geometry = {(int)trunc (disp_bounds.origin.x), (int)trunc (disp_bounds.origin.y), (int)trunc (disp_bounds.size.width), (int)trunc (disp_bounds.size.height)}; CGDisplayModeRef mode = CGDisplayCopyDisplayMode (quartz_monitor->id); gint refresh_rate = (int)trunc (CGDisplayModeGetRefreshRate (mode)); monitor->width_mm = width; monitor->height_mm = height; monitor->geometry = disp_geometry; monitor->scale_factor = 1; monitor->refresh_rate = refresh_rate; monitor->subpixel_layout = GDK_SUBPIXEL_LAYOUT_UNKNOWN; CGDisplayModeRelease (mode); } static void display_reconfiguration_callback (CGDirectDisplayID cg_display, CGDisplayChangeSummaryFlags flags, void *data) { GdkQuartzDisplay *display = data; GdkQuartzMonitor *monitor; /* Ignore the begin configuration signal. */ if (flags & kCGDisplayBeginConfigurationFlag) return; if (flags & (kCGDisplayMovedFlag | kCGDisplayAddFlag | kCGDisplayEnabledFlag | kCGDisplaySetMainFlag | kCGDisplayDesktopShapeChangedFlag | kCGDisplayMirrorFlag | kCGDisplayUnMirrorFlag)) { monitor = g_hash_table_lookup (display->monitors, GINT_TO_POINTER (cg_display)); if (!monitor) { monitor = g_object_new (GDK_TYPE_QUARTZ_MONITOR, "display", display, NULL); monitor->id = cg_display; g_hash_table_insert (display->monitors, GINT_TO_POINTER (monitor->id), monitor); gdk_display_monitor_added (GDK_DISPLAY (display), GDK_MONITOR (monitor)); } configure_monitor (GDK_MONITOR (monitor)); } else if (flags & (kCGDisplayRemoveFlag | kCGDisplayDisabledFlag)) { GdkMonitor *monitor = g_hash_table_lookup (display->monitors, GINT_TO_POINTER (cg_display)); gdk_display_monitor_removed (GDK_DISPLAY (display), GDK_MONITOR (monitor)); g_hash_table_remove (display->monitors, GINT_TO_POINTER (cg_display)); } g_signal_emit (display, MONITORS_CHANGED, 0); } static int gdk_quartz_display_get_n_monitors (GdkDisplay *display) { return get_active_displays (NULL); } static GdkMonitor * gdk_quartz_display_get_monitor (GdkDisplay *display, int monitor_num) { GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (display); CGDirectDisplayID *screens = NULL; int count = get_active_displays (&screens); if (monitor_num >= 0 && monitor_num < count) return g_hash_table_lookup (quartz_display->monitors, GINT_TO_POINTER (screens[monitor_num])); return NULL; } static GdkMonitor * gdk_quartz_display_get_primary_monitor (GdkDisplay *display) { GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (display); CGDirectDisplayID primary_id = CGMainDisplayID (); return g_hash_table_lookup (quartz_display->monitors, GINT_TO_POINTER (primary_id)); } static GdkMonitor * gdk_quartz_display_get_monitor_at_window (GdkDisplay *display, GdkWindow *window) { GdkWindowImplQuartz *impl = GDK_WINDOW_IMPL_QUARTZ (window->impl); NSWindow *nswindow = impl->toplevel; NSScreen *screen = [nswindow screen]; CGDirectDisplayID id = [[[screen deviceDescription] objectForKey: @"NSScreenNumber"] unsignedIntValue]; return g_hash_table_lookup (GDK_QUARTZ_DISPLAY (display)->monitors, GINT_TO_POINTER (id)); } G_DEFINE_TYPE (GdkQuartzDisplay, gdk_quartz_display, GDK_TYPE_DISPLAY) static void gdk_quartz_display_init (GdkQuartzDisplay *display) { uint32_t max_displays = 0, disp; CGDirectDisplayID *displays; CGGetActiveDisplayList (0, NULL, &max_displays); display->monitors = g_hash_table_new_full (g_direct_hash, NULL, NULL, g_object_unref); displays = g_new0 (CGDirectDisplayID, max_displays); CGGetActiveDisplayList (max_displays, displays, &max_displays); for (disp = 0; disp < max_displays; ++disp) { GdkQuartzMonitor *monitor = g_object_new (GDK_TYPE_QUARTZ_MONITOR, "display", display, NULL); monitor->id = displays[disp]; g_hash_table_insert (display->monitors, GINT_TO_POINTER (monitor->id), monitor); configure_monitor (GDK_MONITOR (monitor)); } CGDisplayRegisterReconfigurationCallback (display_reconfiguration_callback, display); g_signal_emit (display, MONITORS_CHANGED, 0); } static void gdk_quartz_display_dispose (GObject *object) { GdkQuartzDisplay *display_quartz = GDK_QUARTZ_DISPLAY (object); g_hash_table_destroy (display_quartz->monitors); CGDisplayRemoveReconfigurationCallback (display_reconfiguration_callback, display_quartz); G_OBJECT_CLASS (gdk_quartz_display_parent_class)->dispose (object); } static void gdk_quartz_display_finalize (GObject *object) { G_OBJECT_CLASS (gdk_quartz_display_parent_class)->finalize (object); } static void gdk_quartz_display_class_init (GdkQuartzDisplayClass *class) { GObjectClass *object_class = G_OBJECT_CLASS (class); GdkDisplayClass *display_class = GDK_DISPLAY_CLASS (class); object_class->finalize = gdk_quartz_display_finalize; object_class->dispose = gdk_quartz_display_dispose; display_class->window_type = GDK_TYPE_QUARTZ_WINDOW; display_class->get_name = gdk_quartz_display_get_name; display_class->get_default_screen = gdk_quartz_display_get_default_screen; display_class->beep = gdk_quartz_display_beep; display_class->sync = gdk_quartz_display_sync; display_class->flush = gdk_quartz_display_flush; display_class->has_pending = _gdk_quartz_display_has_pending; display_class->queue_events = _gdk_quartz_display_queue_events; display_class->get_default_group = gdk_quartz_display_get_default_group; display_class->supports_selection_notification = gdk_quartz_display_supports_selection_notification; display_class->request_selection_notification = gdk_quartz_display_request_selection_notification; display_class->supports_shapes = gdk_quartz_display_supports_shapes; display_class->supports_input_shapes = gdk_quartz_display_supports_input_shapes; display_class->supports_composite = gdk_quartz_display_supports_composite; display_class->supports_cursor_alpha = _gdk_quartz_display_supports_cursor_alpha; display_class->supports_cursor_color = _gdk_quartz_display_supports_cursor_color; display_class->supports_clipboard_persistence = gdk_quartz_display_supports_clipboard_persistence; display_class->store_clipboard = gdk_quartz_display_store_clipboard; display_class->get_default_cursor_size = _gdk_quartz_display_get_default_cursor_size; display_class->get_maximal_cursor_size = _gdk_quartz_display_get_maximal_cursor_size; display_class->get_cursor_for_type = _gdk_quartz_display_get_cursor_for_type; display_class->get_cursor_for_name = _gdk_quartz_display_get_cursor_for_name; display_class->get_cursor_for_surface = _gdk_quartz_display_get_cursor_for_surface; /* display_class->get_app_launch_context = NULL; Has default. */ display_class->before_process_all_updates = _gdk_quartz_display_before_process_all_updates; display_class->after_process_all_updates = _gdk_quartz_display_after_process_all_updates; display_class->get_next_serial = gdk_quartz_display_get_next_serial; display_class->notify_startup_complete = gdk_quartz_display_notify_startup_complete; display_class->event_data_copy = _gdk_quartz_display_event_data_copy; display_class->event_data_free = _gdk_quartz_display_event_data_free; display_class->create_window_impl = _gdk_quartz_display_create_window_impl; display_class->get_keymap = _gdk_quartz_display_get_keymap; display_class->push_error_trap = gdk_quartz_display_push_error_trap; display_class->pop_error_trap = gdk_quartz_display_pop_error_trap; display_class->get_selection_owner = _gdk_quartz_display_get_selection_owner; display_class->set_selection_owner = _gdk_quartz_display_set_selection_owner; display_class->send_selection_notify = NULL; /* Ignore. X11 stuff removed in master. */ display_class->get_selection_property = _gdk_quartz_display_get_selection_property; display_class->convert_selection = _gdk_quartz_display_convert_selection; display_class->text_property_to_utf8_list = _gdk_quartz_display_text_property_to_utf8_list; display_class->utf8_to_string_target = _gdk_quartz_display_utf8_to_string_target; /* display_class->get_default_seat; The parent class default works fine. */ display_class->get_n_monitors = gdk_quartz_display_get_n_monitors; display_class->get_monitor = gdk_quartz_display_get_monitor; display_class->get_primary_monitor = gdk_quartz_display_get_primary_monitor; display_class->get_monitor_at_window = gdk_quartz_display_get_monitor_at_window; /** * GdkQuartzDisplay::monitors-changed: * @display: The object on which the signal is emitted * * The ::monitors-changed signal is emitted whenever the arrangement * of the monitors changes, either because of the addition or * removal of a monitor or because of some other configuration * change in System Preferences>Displays including a resolution * change or a position change. Note that enabling or disabling * mirroring will result in the addition or removal of the mirror * monitor(s). */ MONITORS_CHANGED = g_signal_new (g_intern_static_string ("monitors-changed"), G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL, NULL, NULL, G_TYPE_NONE, 0, NULL); ProcessSerialNumber psn = { 0, kCurrentProcess }; /* Make the current process a foreground application, i.e. an app * with a user interface, in case we're not running from a .app bundle */ TransformProcessType (&psn, kProcessTransformToForegroundApplication); }