2005-11-22 10:03:32 +00:00
|
|
|
/* 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
|
2012-02-27 13:01:10 +00:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
2005-11-22 10:03:32 +00:00
|
|
|
*/
|
|
|
|
|
2008-06-22 14:28:52 +00:00
|
|
|
#include "config.h"
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-12-23 11:59:26 +00:00
|
|
|
#include <gdk/gdk.h>
|
|
|
|
#include <gdk/gdkdisplayprivate.h>
|
2018-12-03 12:22:19 +00:00
|
|
|
#include <gdk/gdkmonitorprivate.h>
|
2010-12-23 11:59:26 +00:00
|
|
|
|
2005-11-22 10:03:32 +00:00
|
|
|
#include "gdkprivate-quartz.h"
|
2010-12-23 11:59:26 +00:00
|
|
|
#include "gdkquartzscreen.h"
|
|
|
|
#include "gdkquartzwindow.h"
|
|
|
|
#include "gdkquartzdisplay.h"
|
|
|
|
#include "gdkquartzdevicemanager-core.h"
|
2017-03-10 13:48:36 +00:00
|
|
|
#include "gdkscreen.h"
|
|
|
|
#include "gdkmonitorprivate.h"
|
|
|
|
#include "gdkdisplay-quartz.h"
|
2018-11-21 12:32:41 +00:00
|
|
|
#include "gdkmonitor-quartz.h"
|
2014-11-16 00:04:58 +00:00
|
|
|
#include "gdkglcontext-quartz.h"
|
2010-12-23 11:59:26 +00:00
|
|
|
|
2019-02-23 22:35:42 +00:00
|
|
|
/* Note about coordinates: There are three coordinate systems at play:
|
|
|
|
*
|
|
|
|
* 1. Core Graphics starts at the origin at the upper right of the
|
|
|
|
* main window (the one with the menu bar when you look at arrangement
|
|
|
|
* in System Preferences>Displays) and increases down and to the
|
|
|
|
* right; up and to the left are negative values of y and x
|
|
|
|
* respectively.
|
|
|
|
*
|
|
|
|
* 2. AppKit (functions beginning with "NS" for NextStep) coordinates
|
|
|
|
* also have their origin at the main window, but it's the *lower*
|
|
|
|
* left corner and coordinates increase up and to the
|
|
|
|
* right. Coordinates below or left of the origin are negative.
|
|
|
|
*
|
|
|
|
* 3. Gdk coordinates origin is at the upper left corner of the
|
|
|
|
* imaginary rectangle enclosing all monitors and like Core Graphics
|
|
|
|
* increase down and to the right. There are no negative coordinates.
|
|
|
|
*
|
|
|
|
* We need to deal with all three because AppKit's NSScreen array is
|
|
|
|
* recomputed with new pointers whenever the monitor arrangement
|
|
|
|
* changes so we can't cache the references it provides. CoreGraphics
|
|
|
|
* screen IDs are constant between reboots so those are what we use to
|
|
|
|
* map GdkMonitors and screens, but the sizes and origins must be
|
|
|
|
* converted to Gdk coordinates to make sense to Gdk and we must
|
|
|
|
* frequently convert between Gdk and AppKit coordinates when
|
|
|
|
* determining the drawable area of a monitor and placing windows and
|
|
|
|
* views (the latter containing our cairo surfaces for drawing on).
|
|
|
|
*/
|
|
|
|
|
2018-11-21 12:32:41 +00:00
|
|
|
static gint MONITORS_CHANGED = 0;
|
|
|
|
|
|
|
|
static void display_reconfiguration_callback (CGDirectDisplayID display,
|
|
|
|
CGDisplayChangeSummaryFlags flags,
|
|
|
|
void *data);
|
2010-12-23 11:59:26 +00:00
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static GdkWindow *
|
|
|
|
gdk_quartz_display_get_default_group (GdkDisplay *display)
|
2006-03-27 10:47:29 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* X11-only. */
|
2006-03-27 10:47:29 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
GdkDeviceManager *
|
|
|
|
_gdk_device_manager_new (GdkDisplay *display)
|
|
|
|
{
|
2010-12-22 16:30:29 +00:00
|
|
|
return g_object_new (GDK_TYPE_QUARTZ_DEVICE_MANAGER_CORE,
|
2010-05-25 22:38:44 +00:00
|
|
|
"display", display,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2005-11-22 10:03:32 +00:00
|
|
|
GdkDisplay *
|
2010-12-17 16:38:02 +00:00
|
|
|
_gdk_quartz_display_open (const gchar *display_name)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
|
|
|
if (_gdk_display != NULL)
|
|
|
|
return NULL;
|
|
|
|
|
2007-05-21 19:40:43 +00:00
|
|
|
/* Initialize application */
|
|
|
|
[NSApplication sharedApplication];
|
|
|
|
|
2010-12-23 12:39:07 +00:00
|
|
|
_gdk_display = g_object_new (gdk_quartz_display_get_type (), NULL);
|
2010-05-25 22:38:44 +00:00
|
|
|
_gdk_display->device_manager = _gdk_device_manager_new (_gdk_display);
|
2009-10-24 09:48:39 +00:00
|
|
|
|
2010-12-23 12:43:29 +00:00
|
|
|
_gdk_screen = g_object_new (gdk_quartz_screen_get_type (), NULL);
|
2010-12-22 13:45:35 +00:00
|
|
|
_gdk_quartz_screen_init_visuals (_gdk_screen);
|
2010-12-17 16:05:53 +00:00
|
|
|
|
2010-12-23 13:08:40 +00:00
|
|
|
_gdk_quartz_window_init_windowing (_gdk_display, _gdk_screen);
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-12-17 16:38:02 +00:00
|
|
|
_gdk_quartz_events_init ();
|
2010-05-25 22:38:44 +00:00
|
|
|
|
2005-11-22 10:03:32 +00:00
|
|
|
#if 0
|
|
|
|
/* FIXME: Remove the #if 0 when we have these functions */
|
2010-12-17 15:53:00 +00:00
|
|
|
_gdk_quartz_dnd_init ();
|
2005-11-22 10:03:32 +00:00
|
|
|
#endif
|
|
|
|
|
2013-04-17 22:00:00 +00:00
|
|
|
g_signal_emit_by_name (_gdk_display, "opened");
|
|
|
|
|
2005-11-22 10:03:32 +00:00
|
|
|
return _gdk_display;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static const gchar *
|
|
|
|
gdk_quartz_display_get_name (GdkDisplay *display)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
2006-08-24 19:41:40 +00:00
|
|
|
static gchar *display_name = NULL;
|
|
|
|
|
2007-03-10 21:04:05 +00:00
|
|
|
if (!display_name)
|
|
|
|
{
|
|
|
|
GDK_QUARTZ_ALLOC_POOL;
|
|
|
|
display_name = g_strdup ([[[NSHost currentHost] name] UTF8String]);
|
|
|
|
GDK_QUARTZ_RELEASE_POOL;
|
|
|
|
}
|
2006-08-24 19:41:40 +00:00
|
|
|
|
|
|
|
return display_name;
|
2005-11-22 10:03:32 +00:00
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static GdkScreen *
|
|
|
|
gdk_quartz_display_get_default_screen (GdkDisplay *display)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
|
|
|
return _gdk_screen;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static void
|
|
|
|
gdk_quartz_display_beep (GdkDisplay *display)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
|
|
|
g_return_if_fail (GDK_IS_DISPLAY (display));
|
|
|
|
|
|
|
|
NSBeep();
|
|
|
|
}
|
|
|
|
|
2010-12-23 13:08:40 +00:00
|
|
|
static void
|
|
|
|
gdk_quartz_display_sync (GdkDisplay *display)
|
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* Not needed. */
|
2010-12-23 13:08:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gdk_quartz_display_flush (GdkDisplay *display)
|
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* Not needed. */
|
2010-12-23 13:08:40 +00:00
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static gboolean
|
|
|
|
gdk_quartz_display_supports_selection_notification (GdkDisplay *display)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
|
|
|
g_return_val_if_fail (GDK_IS_DISPLAY (display), FALSE);
|
2018-11-25 02:42:34 +00:00
|
|
|
/* X11-only. */
|
2005-11-22 10:03:32 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static gboolean
|
|
|
|
gdk_quartz_display_request_selection_notification (GdkDisplay *display,
|
|
|
|
GdkAtom selection)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* X11-only. */
|
2005-11-22 10:03:32 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static gboolean
|
|
|
|
gdk_quartz_display_supports_clipboard_persistence (GdkDisplay *display)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* X11-only */
|
2005-11-22 10:03:32 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static gboolean
|
|
|
|
gdk_quartz_display_supports_shapes (GdkDisplay *display)
|
2006-02-20 11:57:12 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* Not needed, nothing ever calls this.*/
|
2006-02-20 11:57:12 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static gboolean
|
|
|
|
gdk_quartz_display_supports_input_shapes (GdkDisplay *display)
|
2006-02-20 11:57:12 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* Not needed, nothign ever calls this. */
|
2006-02-20 11:57:12 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static void
|
|
|
|
gdk_quartz_display_store_clipboard (GdkDisplay *display,
|
|
|
|
GdkWindow *clipboard_window,
|
|
|
|
guint32 time_,
|
|
|
|
const GdkAtom *targets,
|
|
|
|
gint n_targets)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* MacOS persists pasteboard items automatically, no application
|
|
|
|
* action is required.
|
|
|
|
*/
|
2005-11-22 10:03:32 +00:00
|
|
|
}
|
2007-06-01 12:16:12 +00:00
|
|
|
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
static gboolean
|
|
|
|
gdk_quartz_display_supports_composite (GdkDisplay *display)
|
2007-06-01 12:16:12 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* X11-only. */
|
2007-06-01 12:16:12 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2009-01-24 09:03:25 +00:00
|
|
|
|
2010-12-23 12:39:07 +00:00
|
|
|
static gulong
|
|
|
|
gdk_quartz_display_get_next_serial (GdkDisplay *display)
|
2009-01-24 09:03:25 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* X11-only. */
|
2009-01-24 09:03:25 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2010-12-22 02:13:02 +00:00
|
|
|
|
2010-12-23 12:09:01 +00:00
|
|
|
static void
|
2010-12-23 12:39:07 +00:00
|
|
|
gdk_quartz_display_notify_startup_complete (GdkDisplay *display,
|
|
|
|
const gchar *startup_id)
|
2010-12-23 12:09:01 +00:00
|
|
|
{
|
2018-11-25 02:42:34 +00:00
|
|
|
/* 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;
|
2010-12-23 12:09:01 +00:00
|
|
|
}
|
|
|
|
|
2018-12-03 12:22:19 +00:00
|
|
|
/* 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.
|
|
|
|
*/
|
|
|
|
|
2018-11-21 12:32:41 +00:00
|
|
|
int
|
2019-02-23 22:35:42 +00:00
|
|
|
get_active_displays (CGDirectDisplayID **displays)
|
2017-03-10 13:48:36 +00:00
|
|
|
{
|
2019-02-23 22:35:42 +00:00
|
|
|
unsigned int n_displays = 0;
|
2017-03-10 13:48:36 +00:00
|
|
|
|
2019-02-23 22:35:42 +00:00
|
|
|
CGGetActiveDisplayList (0, NULL, &n_displays);
|
|
|
|
if (displays)
|
2018-11-21 12:32:41 +00:00
|
|
|
{
|
2019-02-23 22:35:42 +00:00
|
|
|
*displays = g_new0 (CGDirectDisplayID, n_displays);
|
|
|
|
CGGetActiveDisplayList (n_displays, *displays, &n_displays);
|
2018-11-21 12:32:41 +00:00
|
|
|
}
|
2017-03-10 13:48:36 +00:00
|
|
|
|
2019-02-23 22:35:42 +00:00
|
|
|
return n_displays;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline GdkRectangle
|
|
|
|
cgrect_to_gdkrect (CGRect cgrect)
|
|
|
|
{
|
|
|
|
GdkRectangle gdkrect = {(int)trunc (cgrect.origin.x),
|
|
|
|
(int)trunc (cgrect.origin.y),
|
|
|
|
(int)trunc (cgrect.size.width),
|
|
|
|
(int)trunc (cgrect.size.height)};
|
|
|
|
return gdkrect;
|
2018-11-21 12:32:41 +00:00
|
|
|
}
|
2018-12-03 12:22:19 +00:00
|
|
|
|
2018-11-21 12:32:41 +00:00
|
|
|
static void
|
2019-02-23 22:35:42 +00:00
|
|
|
configure_monitor (GdkMonitor *monitor,
|
|
|
|
GdkQuartzDisplay *display)
|
2018-11-21 12:32:41 +00:00
|
|
|
{
|
|
|
|
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);
|
2019-02-23 22:35:42 +00:00
|
|
|
CGRect main_bounds = CGDisplayBounds (CGMainDisplayID());
|
|
|
|
/* Change origin to Gdk coordinates. */
|
|
|
|
disp_bounds.origin.x = disp_bounds.origin.x + display->geometry.origin.x;
|
|
|
|
disp_bounds.origin.y =
|
|
|
|
display->geometry.origin.y - main_bounds.size.height + disp_bounds.origin.y;
|
|
|
|
GdkRectangle disp_geometry = cgrect_to_gdkrect (disp_bounds);
|
2018-11-21 12:32:41 +00:00
|
|
|
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;
|
2018-12-18 10:33:16 +00:00
|
|
|
#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1080
|
2019-01-06 04:02:26 +00:00
|
|
|
if (mode && gdk_quartz_osx_version () >= GDK_OSX_MOUNTAIN_LION)
|
|
|
|
{
|
2018-12-18 10:33:16 +00:00
|
|
|
monitor->scale_factor = CGDisplayModeGetPixelWidth (mode) / CGDisplayModeGetWidth (mode);
|
2019-01-06 04:02:26 +00:00
|
|
|
CGDisplayModeRelease (mode);
|
|
|
|
}
|
2018-12-18 10:33:16 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
monitor->scale_factor = 1;
|
2018-11-21 12:32:41 +00:00
|
|
|
monitor->refresh_rate = refresh_rate;
|
|
|
|
monitor->subpixel_layout = GDK_SUBPIXEL_LAYOUT_UNKNOWN;
|
|
|
|
}
|
2018-12-03 12:22:19 +00:00
|
|
|
|
2019-02-23 22:35:42 +00:00
|
|
|
static void
|
|
|
|
display_rect (GdkQuartzDisplay *display)
|
|
|
|
{
|
2019-03-11 23:31:58 +00:00
|
|
|
uint32_t disp, n_displays = 0;
|
2019-02-23 22:35:42 +00:00
|
|
|
float min_x = 0.0, max_x = 0.0, min_y = 0.0, max_y = 0.0;
|
|
|
|
float min_x_mm = 0.0, max_x_mm = 0.0, min_y_mm = 0.0, max_y_mm = 0.0;
|
|
|
|
float main_height;
|
|
|
|
CGDirectDisplayID *displays;
|
|
|
|
|
2019-03-11 23:31:58 +00:00
|
|
|
n_displays = get_active_displays (&displays);
|
|
|
|
for (disp = 0; disp < n_displays; ++disp)
|
2019-02-23 22:35:42 +00:00
|
|
|
{
|
|
|
|
CGRect bounds = CGDisplayBounds (displays[disp]);
|
|
|
|
CGSize disp_size = CGDisplayScreenSize (displays[disp]);
|
|
|
|
float x_scale = disp_size.width / bounds.size.width;
|
|
|
|
float y_scale = disp_size.height / bounds.size.height;
|
|
|
|
if (disp == 0)
|
|
|
|
main_height = bounds.size.height;
|
|
|
|
min_x = MIN (min_x, bounds.origin.x);
|
|
|
|
min_y = MIN (min_y, bounds.origin.y);
|
|
|
|
|
|
|
|
max_x = MAX (max_x, bounds.origin.x + bounds.size.width);
|
|
|
|
max_y = MAX (max_y, bounds.origin.y + bounds.size.height);
|
|
|
|
min_x_mm = MIN (min_x_mm, bounds.origin.x / x_scale);
|
|
|
|
min_y_mm = MIN (min_y_mm, main_height - (bounds.size.height + bounds.origin.y) / y_scale);
|
|
|
|
max_x_mm = MAX (max_x_mm, (bounds.origin.x + bounds.size.width) / x_scale);
|
|
|
|
max_y_mm = MAX (max_y_mm, (bounds.origin.y + bounds.size.height) / y_scale);
|
|
|
|
|
|
|
|
}
|
|
|
|
g_free (displays);
|
|
|
|
/* Adjusts the origin to AppKit coordinates. */
|
|
|
|
display->geometry = NSMakeRect (-min_x, main_height - min_y,
|
|
|
|
max_x - min_x, max_y - min_y);
|
|
|
|
display->size = NSMakeSize (max_x_mm - min_x_mm, max_y_mm - min_y_mm);
|
|
|
|
}
|
|
|
|
|
2019-02-24 13:38:24 +00:00
|
|
|
static gboolean
|
|
|
|
same_monitor (gconstpointer a, gconstpointer b)
|
|
|
|
{
|
|
|
|
GdkQuartzMonitor *mon_a = GDK_QUARTZ_MONITOR (a);
|
|
|
|
CGDirectDisplayID disp_id = (CGDirectDisplayID)GPOINTER_TO_INT (b);
|
|
|
|
if (!mon_a)
|
|
|
|
return FALSE;
|
|
|
|
return mon_a->id == disp_id;
|
|
|
|
}
|
|
|
|
|
2018-11-21 12:32:41 +00:00
|
|
|
static void
|
|
|
|
display_reconfiguration_callback (CGDirectDisplayID cg_display,
|
|
|
|
CGDisplayChangeSummaryFlags flags,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
GdkQuartzDisplay *display = data;
|
|
|
|
|
|
|
|
/* Ignore the begin configuration signal. */
|
|
|
|
if (flags & kCGDisplayBeginConfigurationFlag)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (flags & (kCGDisplayMovedFlag | kCGDisplayAddFlag | kCGDisplayEnabledFlag |
|
2019-07-14 22:28:34 +00:00
|
|
|
kCGDisplaySetMainFlag | kCGDisplayMirrorFlag |
|
|
|
|
kCGDisplayUnMirrorFlag))
|
2018-11-21 12:32:41 +00:00
|
|
|
{
|
2019-02-24 13:38:24 +00:00
|
|
|
GdkQuartzMonitor *monitor = NULL;
|
|
|
|
guint index;
|
|
|
|
|
|
|
|
if (!g_ptr_array_find_with_equal_func (display->monitors,
|
|
|
|
GINT_TO_POINTER (cg_display),
|
|
|
|
same_monitor,
|
|
|
|
&index))
|
2018-11-21 12:32:41 +00:00
|
|
|
{
|
|
|
|
monitor = g_object_new (GDK_TYPE_QUARTZ_MONITOR,
|
|
|
|
"display", display, NULL);
|
|
|
|
monitor->id = cg_display;
|
2019-02-24 13:38:24 +00:00
|
|
|
g_ptr_array_add (display->monitors, monitor);
|
2018-11-21 12:32:41 +00:00
|
|
|
gdk_display_monitor_added (GDK_DISPLAY (display),
|
|
|
|
GDK_MONITOR (monitor));
|
|
|
|
}
|
2019-02-24 13:38:24 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
monitor = g_ptr_array_index (display->monitors, index);
|
|
|
|
}
|
|
|
|
|
2019-02-23 22:35:42 +00:00
|
|
|
display_rect (display);
|
|
|
|
configure_monitor (GDK_MONITOR (monitor), display);
|
2018-11-21 12:32:41 +00:00
|
|
|
}
|
|
|
|
else if (flags & (kCGDisplayRemoveFlag | kCGDisplayDisabledFlag))
|
|
|
|
{
|
2019-02-24 13:38:24 +00:00
|
|
|
guint index;
|
|
|
|
|
|
|
|
if (g_ptr_array_find_with_equal_func (display->monitors,
|
|
|
|
GINT_TO_POINTER (cg_display),
|
|
|
|
same_monitor,
|
|
|
|
&index))
|
|
|
|
{
|
|
|
|
GdkQuartzMonitor *monitor = g_ptr_array_index (display->monitors,
|
|
|
|
index);
|
|
|
|
gdk_display_monitor_removed (GDK_DISPLAY (display),
|
|
|
|
GDK_MONITOR (monitor));
|
|
|
|
g_ptr_array_remove_fast (display->monitors, monitor);
|
|
|
|
}
|
2018-11-21 12:32:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_signal_emit (display, MONITORS_CHANGED, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
gdk_quartz_display_get_n_monitors (GdkDisplay *display)
|
|
|
|
{
|
2019-02-24 13:38:24 +00:00
|
|
|
GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (display);
|
|
|
|
return quartz_display->monitors->len;
|
2018-12-03 12:22:19 +00:00
|
|
|
}
|
2017-03-10 13:48:36 +00:00
|
|
|
|
|
|
|
static GdkMonitor *
|
|
|
|
gdk_quartz_display_get_monitor (GdkDisplay *display,
|
|
|
|
int monitor_num)
|
|
|
|
{
|
|
|
|
GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (display);
|
2019-03-11 23:31:58 +00:00
|
|
|
int n_displays = gdk_quartz_display_get_n_monitors (display);
|
2018-12-03 12:22:19 +00:00
|
|
|
|
2019-03-11 23:31:58 +00:00
|
|
|
if (monitor_num >= 0 && monitor_num < n_displays)
|
2019-02-24 13:38:24 +00:00
|
|
|
return g_ptr_array_index (quartz_display->monitors, monitor_num);
|
|
|
|
|
|
|
|
return NULL;
|
2017-03-10 13:48:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static GdkMonitor *
|
|
|
|
gdk_quartz_display_get_primary_monitor (GdkDisplay *display)
|
|
|
|
{
|
|
|
|
GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (display);
|
2018-12-03 12:22:19 +00:00
|
|
|
CGDirectDisplayID primary_id = CGMainDisplayID ();
|
2019-02-24 13:38:24 +00:00
|
|
|
GdkMonitor *monitor = NULL;
|
|
|
|
guint index;
|
2017-03-10 13:48:36 +00:00
|
|
|
|
2019-02-24 13:38:24 +00:00
|
|
|
if (g_ptr_array_find_with_equal_func (quartz_display->monitors,
|
|
|
|
GINT_TO_POINTER (primary_id),
|
|
|
|
same_monitor, &index))
|
|
|
|
monitor = g_ptr_array_index (quartz_display->monitors, index);
|
|
|
|
|
|
|
|
return monitor;
|
2017-03-10 13:48:36 +00:00
|
|
|
}
|
2010-12-23 12:09:01 +00:00
|
|
|
|
2018-11-25 02:42:34 +00:00
|
|
|
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];
|
2019-02-24 13:38:24 +00:00
|
|
|
GdkMonitor *monitor = NULL;
|
2019-02-23 22:35:42 +00:00
|
|
|
if (screen)
|
|
|
|
{
|
2019-02-24 13:38:24 +00:00
|
|
|
GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (display);
|
|
|
|
guint index;
|
|
|
|
CGDirectDisplayID disp_id =
|
|
|
|
[[[screen deviceDescription]
|
|
|
|
objectForKey: @"NSScreenNumber"] unsignedIntValue];
|
|
|
|
if (g_ptr_array_find_with_equal_func (quartz_display->monitors,
|
|
|
|
GINT_TO_POINTER (disp_id),
|
|
|
|
same_monitor, &index))
|
|
|
|
monitor = g_ptr_array_index (quartz_display->monitors, index);
|
2019-02-23 22:35:42 +00:00
|
|
|
}
|
2019-02-24 13:38:24 +00:00
|
|
|
if (!monitor)
|
|
|
|
{
|
2019-03-21 23:01:51 +00:00
|
|
|
GdkRectangle rect = cgrect_to_gdkrect (NSRectToCGRect ([nswindow frame]));
|
2019-02-24 13:38:24 +00:00
|
|
|
monitor = gdk_display_get_monitor_at_point (display,
|
|
|
|
rect.x + rect.width/2,
|
|
|
|
rect.y + rect.height /2);
|
|
|
|
}
|
|
|
|
return monitor;
|
2018-11-25 02:42:34 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 12:39:07 +00:00
|
|
|
G_DEFINE_TYPE (GdkQuartzDisplay, gdk_quartz_display, GDK_TYPE_DISPLAY)
|
2010-12-22 02:13:02 +00:00
|
|
|
|
|
|
|
static void
|
2010-12-23 12:39:07 +00:00
|
|
|
gdk_quartz_display_init (GdkQuartzDisplay *display)
|
2010-12-22 02:13:02 +00:00
|
|
|
{
|
2019-03-11 23:31:58 +00:00
|
|
|
uint32_t n_displays = 0, disp;
|
2018-12-03 12:22:19 +00:00
|
|
|
CGDirectDisplayID *displays;
|
2018-11-21 12:32:41 +00:00
|
|
|
|
2019-02-23 22:35:42 +00:00
|
|
|
display_rect(display); /* Initialize the overall display coordinates. */
|
2019-03-11 23:31:58 +00:00
|
|
|
n_displays = get_active_displays (&displays);
|
|
|
|
display->monitors = g_ptr_array_new_full (n_displays, g_object_unref);
|
|
|
|
for (disp = 0; disp < n_displays; ++disp)
|
2018-12-03 12:22:19 +00:00
|
|
|
{
|
2018-11-21 12:32:41 +00:00
|
|
|
GdkQuartzMonitor *monitor = g_object_new (GDK_TYPE_QUARTZ_MONITOR,
|
2018-12-03 12:22:19 +00:00
|
|
|
"display", display, NULL);
|
2018-11-21 12:32:41 +00:00
|
|
|
monitor->id = displays[disp];
|
2019-02-24 13:38:24 +00:00
|
|
|
g_ptr_array_add (display->monitors, monitor);
|
2019-02-23 22:35:42 +00:00
|
|
|
configure_monitor (GDK_MONITOR (monitor), display);
|
2018-12-03 12:22:19 +00:00
|
|
|
}
|
2019-02-23 22:35:42 +00:00
|
|
|
g_free (displays);
|
2018-11-21 12:32:41 +00:00
|
|
|
CGDisplayRegisterReconfigurationCallback (display_reconfiguration_callback,
|
|
|
|
display);
|
2019-02-23 22:35:42 +00:00
|
|
|
/* So that monitors changed will keep display->geometry syncronized. */
|
2018-11-21 12:32:41 +00:00
|
|
|
g_signal_emit (display, MONITORS_CHANGED, 0);
|
2010-12-17 16:38:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-12-23 12:39:07 +00:00
|
|
|
gdk_quartz_display_dispose (GObject *object)
|
2010-12-17 16:38:02 +00:00
|
|
|
{
|
2019-02-24 13:38:24 +00:00
|
|
|
GdkQuartzDisplay *quartz_display = GDK_QUARTZ_DISPLAY (object);
|
2010-12-22 15:05:12 +00:00
|
|
|
|
2019-02-24 13:38:24 +00:00
|
|
|
g_ptr_array_free (quartz_display->monitors, TRUE);
|
2018-11-21 12:32:41 +00:00
|
|
|
CGDisplayRemoveReconfigurationCallback (display_reconfiguration_callback,
|
2019-02-24 13:38:24 +00:00
|
|
|
quartz_display);
|
2017-03-10 13:48:36 +00:00
|
|
|
|
2010-12-23 12:39:07 +00:00
|
|
|
G_OBJECT_CLASS (gdk_quartz_display_parent_class)->dispose (object);
|
2010-12-22 02:13:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-12-23 12:39:07 +00:00
|
|
|
gdk_quartz_display_finalize (GObject *object)
|
2010-12-22 02:13:02 +00:00
|
|
|
{
|
2010-12-23 12:39:07 +00:00
|
|
|
G_OBJECT_CLASS (gdk_quartz_display_parent_class)->finalize (object);
|
2010-12-22 02:13:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-12-23 12:39:07 +00:00
|
|
|
gdk_quartz_display_class_init (GdkQuartzDisplayClass *class)
|
2010-12-22 02:13:02 +00:00
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
GdkDisplayClass *display_class = GDK_DISPLAY_CLASS (class);
|
|
|
|
|
2010-12-23 12:39:07 +00:00
|
|
|
object_class->finalize = gdk_quartz_display_finalize;
|
2011-05-22 10:29:37 +00:00
|
|
|
object_class->dispose = gdk_quartz_display_dispose;
|
2010-12-22 02:13:02 +00:00
|
|
|
|
2010-12-22 15:56:58 +00:00
|
|
|
display_class->window_type = GDK_TYPE_QUARTZ_WINDOW;
|
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
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;
|
2010-12-23 13:08:40 +00:00
|
|
|
display_class->sync = gdk_quartz_display_sync;
|
|
|
|
display_class->flush = gdk_quartz_display_flush;
|
2010-12-17 16:46:22 +00:00
|
|
|
display_class->has_pending = _gdk_quartz_display_has_pending;
|
2018-11-17 00:46:21 +00:00
|
|
|
display_class->queue_events = _gdk_quartz_display_queue_events;
|
2010-12-22 02:13:02 +00:00
|
|
|
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;
|
2018-11-17 00:46:21 +00:00
|
|
|
|
2010-12-22 02:13:02 +00:00
|
|
|
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;
|
2018-11-17 00:46:21 +00:00
|
|
|
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;
|
2010-12-17 17:01:54 +00:00
|
|
|
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;
|
2013-08-11 17:43:35 +00:00
|
|
|
display_class->get_cursor_for_surface = _gdk_quartz_display_get_cursor_for_surface;
|
2010-12-22 16:06:13 +00:00
|
|
|
|
2018-11-25 02:42:34 +00:00
|
|
|
/* display_class->get_app_launch_context = NULL; Has default. */
|
2010-12-22 16:06:13 +00:00
|
|
|
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;
|
2010-12-23 12:39:07 +00:00
|
|
|
display_class->get_next_serial = gdk_quartz_display_get_next_serial;
|
|
|
|
display_class->notify_startup_complete = gdk_quartz_display_notify_startup_complete;
|
2010-12-17 17:25:47 +00:00
|
|
|
display_class->event_data_copy = _gdk_quartz_display_event_data_copy;
|
|
|
|
display_class->event_data_free = _gdk_quartz_display_event_data_free;
|
2010-12-17 17:34:40 +00:00
|
|
|
display_class->create_window_impl = _gdk_quartz_display_create_window_impl;
|
2010-12-17 17:46:46 +00:00
|
|
|
display_class->get_keymap = _gdk_quartz_display_get_keymap;
|
2018-11-25 02:42:34 +00:00
|
|
|
display_class->push_error_trap = gdk_quartz_display_push_error_trap;
|
|
|
|
display_class->pop_error_trap = gdk_quartz_display_pop_error_trap;
|
2018-11-17 00:46:21 +00:00
|
|
|
|
2010-12-17 18:25:04 +00:00
|
|
|
display_class->get_selection_owner = _gdk_quartz_display_get_selection_owner;
|
|
|
|
display_class->set_selection_owner = _gdk_quartz_display_set_selection_owner;
|
2018-11-25 02:42:34 +00:00
|
|
|
display_class->send_selection_notify = NULL; /* Ignore. X11 stuff removed in master. */
|
2010-12-17 18:29:54 +00:00
|
|
|
display_class->get_selection_property = _gdk_quartz_display_get_selection_property;
|
|
|
|
display_class->convert_selection = _gdk_quartz_display_convert_selection;
|
2010-12-17 18:34:04 +00:00
|
|
|
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;
|
2018-11-17 00:46:21 +00:00
|
|
|
|
2018-11-21 12:32:41 +00:00
|
|
|
/* display_class->get_default_seat; The parent class default works fine. */
|
2018-11-17 00:46:21 +00:00
|
|
|
|
2017-03-10 13:48:36 +00:00
|
|
|
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;
|
2018-11-25 02:42:34 +00:00
|
|
|
display_class->get_monitor_at_window = gdk_quartz_display_get_monitor_at_window;
|
2014-11-16 00:04:58 +00:00
|
|
|
display_class->make_gl_context_current = gdk_quartz_display_make_gl_context_current;
|
2013-05-02 13:43:09 +00:00
|
|
|
|
2018-11-21 12:32:41 +00:00
|
|
|
/**
|
|
|
|
* 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);
|
|
|
|
|
2013-05-02 13:43:09 +00:00
|
|
|
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);
|
2010-12-22 02:13:02 +00:00
|
|
|
}
|