2006-01-12 00:28:25 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2006-01-12 00:27:48 +03:00
|
|
|
|
|
|
|
#include "nsScreenGtk.h"
|
|
|
|
|
2008-08-07 00:48:55 +04:00
|
|
|
#include <gdk/gdk.h>
|
|
|
|
#ifdef MOZ_X11
|
2006-01-12 00:27:50 +03:00
|
|
|
#include <gdk/gdkx.h>
|
2008-08-07 03:24:13 +04:00
|
|
|
#include <X11/Xatom.h>
|
2008-08-07 00:48:55 +04:00
|
|
|
#endif
|
|
|
|
#include <gtk/gtk.h>
|
2015-01-26 19:21:00 +03:00
|
|
|
#include <dlfcn.h>
|
2015-05-01 21:08:04 +03:00
|
|
|
#include "gfxPlatformGtk.h"
|
2006-01-12 00:27:50 +03:00
|
|
|
|
2014-07-14 21:22:26 +04:00
|
|
|
static uint32_t sScreenId = 0;
|
|
|
|
|
|
|
|
|
2006-01-12 00:27:48 +03:00
|
|
|
nsScreenGtk :: nsScreenGtk ( )
|
2008-07-14 07:20:30 +04:00
|
|
|
: mScreenNum(0),
|
2006-01-12 00:28:41 +03:00
|
|
|
mRect(0, 0, 0, 0),
|
2014-07-14 21:22:26 +04:00
|
|
|
mAvailRect(0, 0, 0, 0),
|
|
|
|
mId(++sScreenId)
|
2006-01-12 00:27:48 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
nsScreenGtk :: ~nsScreenGtk()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-14 21:22:26 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsScreenGtk :: GetId(uint32_t *aId)
|
|
|
|
{
|
|
|
|
*aId = mId;
|
|
|
|
return NS_OK;
|
|
|
|
} // GetId
|
|
|
|
|
|
|
|
|
2006-01-12 00:28:07 +03:00
|
|
|
NS_IMETHODIMP
|
2012-08-22 19:56:38 +04:00
|
|
|
nsScreenGtk :: GetRect(int32_t *outLeft, int32_t *outTop, int32_t *outWidth, int32_t *outHeight)
|
2006-01-12 00:27:48 +03:00
|
|
|
{
|
2006-01-12 00:28:41 +03:00
|
|
|
*outLeft = mRect.x;
|
|
|
|
*outTop = mRect.y;
|
|
|
|
*outWidth = mRect.width;
|
|
|
|
*outHeight = mRect.height;
|
2006-01-12 00:27:48 +03:00
|
|
|
|
2006-01-12 00:28:08 +03:00
|
|
|
return NS_OK;
|
|
|
|
|
2006-01-12 00:28:07 +03:00
|
|
|
} // GetRect
|
2006-01-12 00:27:48 +03:00
|
|
|
|
|
|
|
|
2006-01-12 00:28:07 +03:00
|
|
|
NS_IMETHODIMP
|
2012-08-22 19:56:38 +04:00
|
|
|
nsScreenGtk :: GetAvailRect(int32_t *outLeft, int32_t *outTop, int32_t *outWidth, int32_t *outHeight)
|
2006-01-12 00:27:48 +03:00
|
|
|
{
|
2006-01-12 00:28:41 +03:00
|
|
|
*outLeft = mAvailRect.x;
|
|
|
|
*outTop = mAvailRect.y;
|
|
|
|
*outWidth = mAvailRect.width;
|
|
|
|
*outHeight = mAvailRect.height;
|
2006-01-12 00:27:48 +03:00
|
|
|
|
2006-01-12 00:28:08 +03:00
|
|
|
return NS_OK;
|
|
|
|
|
2006-01-12 00:28:07 +03:00
|
|
|
} // GetAvailRect
|
2006-01-12 00:27:48 +03:00
|
|
|
|
2015-04-23 19:35:00 +03:00
|
|
|
gint
|
|
|
|
nsScreenGtk :: GetGtkMonitorScaleFactor()
|
|
|
|
{
|
|
|
|
#if (MOZ_WIDGET_GTK >= 3)
|
|
|
|
// Since GDK 3.10
|
|
|
|
static auto sGdkScreenGetMonitorScaleFactorPtr = (gint (*)(GdkScreen*, gint))
|
|
|
|
dlsym(RTLD_DEFAULT, "gdk_screen_get_monitor_scale_factor");
|
|
|
|
if (sGdkScreenGetMonitorScaleFactorPtr) {
|
|
|
|
// FIXME: In the future, we'll want to fix this for GTK on Wayland which
|
|
|
|
// supports a variable scale factor per display.
|
|
|
|
GdkScreen *screen = gdk_screen_get_default();
|
|
|
|
return sGdkScreenGetMonitorScaleFactorPtr(screen, 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-02-10 10:14:00 +03:00
|
|
|
double
|
|
|
|
nsScreenGtk :: GetDPIScale()
|
|
|
|
{
|
|
|
|
double dpiScale = nsIWidget::DefaultScaleOverride();
|
|
|
|
if (dpiScale <= 0.0) {
|
2015-06-26 11:19:00 +03:00
|
|
|
dpiScale = GetGtkMonitorScaleFactor() * gfxPlatformGtk::GetDPIScale();
|
2015-02-10 10:14:00 +03:00
|
|
|
}
|
|
|
|
return dpiScale;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsScreenGtk :: GetRectDisplayPix(int32_t *outLeft, int32_t *outTop, int32_t *outWidth, int32_t *outHeight)
|
|
|
|
{
|
|
|
|
int32_t left, top, width, height;
|
|
|
|
|
|
|
|
GetRect(&left, &top, &width, &height);
|
|
|
|
|
|
|
|
double scaleFactor = 1.0 / GetDPIScale();
|
|
|
|
*outLeft = NSToIntRound(left * scaleFactor);
|
|
|
|
*outTop = NSToIntRound(top * scaleFactor);
|
|
|
|
*outWidth = NSToIntRound(width * scaleFactor);
|
|
|
|
*outHeight = NSToIntRound(height * scaleFactor);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsScreenGtk :: GetAvailRectDisplayPix(int32_t *outLeft, int32_t *outTop, int32_t *outWidth, int32_t *outHeight)
|
|
|
|
{
|
|
|
|
int32_t left, top, width, height;
|
|
|
|
|
|
|
|
GetAvailRect(&left, &top, &width, &height);
|
|
|
|
|
|
|
|
double scaleFactor = 1.0 / GetDPIScale();
|
|
|
|
*outLeft = NSToIntRound(left * scaleFactor);
|
|
|
|
*outTop = NSToIntRound(top * scaleFactor);
|
|
|
|
*outWidth = NSToIntRound(width * scaleFactor);
|
|
|
|
*outHeight = NSToIntRound(height * scaleFactor);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2006-01-12 00:27:48 +03:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2012-08-22 19:56:38 +04:00
|
|
|
nsScreenGtk :: GetPixelDepth(int32_t *aPixelDepth)
|
2006-01-12 00:27:48 +03:00
|
|
|
{
|
2012-09-14 05:56:59 +04:00
|
|
|
GdkVisual * visual = gdk_screen_get_system_visual(gdk_screen_get_default());
|
|
|
|
*aPixelDepth = gdk_visual_get_depth(visual);
|
2006-01-12 00:27:50 +03:00
|
|
|
|
2006-01-12 00:27:48 +03:00
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
} // GetPixelDepth
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2012-08-22 19:56:38 +04:00
|
|
|
nsScreenGtk :: GetColorDepth(int32_t *aColorDepth)
|
2006-01-12 00:27:48 +03:00
|
|
|
{
|
2006-01-12 00:27:50 +03:00
|
|
|
return GetPixelDepth ( aColorDepth );
|
2006-01-12 00:27:48 +03:00
|
|
|
|
|
|
|
} // GetColorDepth
|
|
|
|
|
|
|
|
|
2006-01-12 00:28:41 +03:00
|
|
|
void
|
2008-07-14 07:20:30 +04:00
|
|
|
nsScreenGtk :: Init (GdkWindow *aRootWindow)
|
2006-01-12 00:28:41 +03:00
|
|
|
{
|
2015-04-23 19:35:00 +03:00
|
|
|
gint scale = nsScreenGtk::GetGtkMonitorScaleFactor();
|
|
|
|
gint width = gdk_screen_width()*scale;
|
|
|
|
gint height = gdk_screen_height()*scale;
|
2015-01-26 19:21:00 +03:00
|
|
|
|
2006-09-19 03:37:21 +04:00
|
|
|
// We listen for configure events on the root window to pick up
|
|
|
|
// changes to this rect. We could listen for "size_changed" signals
|
|
|
|
// on the default screen to do this, except that doesn't work with
|
|
|
|
// versions of GDK predating the GdkScreen object. See bug 256646.
|
2015-01-26 19:21:00 +03:00
|
|
|
mAvailRect = mRect = nsIntRect(0, 0, width, height);
|
2006-01-12 00:28:41 +03:00
|
|
|
|
2008-08-07 00:48:55 +04:00
|
|
|
#ifdef MOZ_X11
|
2006-01-12 00:28:41 +03:00
|
|
|
// We need to account for the taskbar, etc in the available rect.
|
|
|
|
// See http://freedesktop.org/Standards/wm-spec/index.html#id2767771
|
|
|
|
|
|
|
|
// XXX do we care about _NET_WM_STRUT_PARTIAL? That will
|
|
|
|
// add much more complexity to the code here (our screen
|
|
|
|
// could have a non-rectangular shape), but should
|
|
|
|
// lead to greater accuracy.
|
|
|
|
|
|
|
|
long *workareas;
|
|
|
|
GdkAtom type_returned;
|
|
|
|
int format_returned;
|
|
|
|
int length_returned;
|
|
|
|
|
2006-01-12 00:28:42 +03:00
|
|
|
GdkAtom cardinal_atom = gdk_x11_xatom_to_atom(XA_CARDINAL);
|
|
|
|
|
2006-01-12 00:28:45 +03:00
|
|
|
gdk_error_trap_push();
|
|
|
|
|
2006-01-12 00:28:47 +03:00
|
|
|
// gdk_property_get uses (length + 3) / 4, hence G_MAXLONG - 3 here.
|
2008-07-14 07:20:30 +04:00
|
|
|
if (!gdk_property_get(aRootWindow,
|
2006-01-12 00:28:41 +03:00
|
|
|
gdk_atom_intern ("_NET_WORKAREA", FALSE),
|
2006-01-12 00:28:42 +03:00
|
|
|
cardinal_atom,
|
2006-01-12 00:28:47 +03:00
|
|
|
0, G_MAXLONG - 3, FALSE,
|
2006-01-12 00:28:41 +03:00
|
|
|
&type_returned,
|
|
|
|
&format_returned,
|
|
|
|
&length_returned,
|
|
|
|
(guchar **) &workareas)) {
|
|
|
|
// This window manager doesn't support the freedesktop standard.
|
|
|
|
// Nothing we can do about it, so assume full screen size.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-01-12 00:28:45 +03:00
|
|
|
// Flush the X queue to catch errors now.
|
|
|
|
gdk_flush();
|
|
|
|
|
2006-01-12 00:28:41 +03:00
|
|
|
if (!gdk_error_trap_pop() &&
|
2006-01-12 00:28:43 +03:00
|
|
|
type_returned == cardinal_atom &&
|
2006-01-12 00:28:41 +03:00
|
|
|
length_returned && (length_returned % 4) == 0 &&
|
|
|
|
format_returned == 32) {
|
|
|
|
int num_items = length_returned / sizeof(long);
|
|
|
|
|
|
|
|
for (int i = 0; i < num_items; i += 4) {
|
2009-01-15 06:27:09 +03:00
|
|
|
nsIntRect workarea(workareas[i], workareas[i + 1],
|
|
|
|
workareas[i + 2], workareas[i + 3]);
|
2006-01-12 00:28:41 +03:00
|
|
|
if (!mRect.Contains(workarea)) {
|
2006-09-19 03:37:21 +04:00
|
|
|
// Note that we hit this when processing screen size changes,
|
|
|
|
// since we'll get the configure event before the toolbars have
|
|
|
|
// been moved. We'll end up cleaning this up when we get the
|
|
|
|
// change notification to the _NET_WORKAREA property. However,
|
|
|
|
// we still want to listen to both, so we'll handle changes
|
|
|
|
// properly for desktop environments that don't set the
|
|
|
|
// _NET_WORKAREA property.
|
2006-01-12 00:28:41 +03:00
|
|
|
NS_WARNING("Invalid bounds");
|
2006-01-12 00:28:44 +03:00
|
|
|
continue;
|
2006-01-12 00:28:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
mAvailRect.IntersectRect(mAvailRect, workarea);
|
|
|
|
}
|
|
|
|
}
|
2006-01-12 00:28:46 +03:00
|
|
|
g_free (workareas);
|
2008-08-07 00:48:55 +04:00
|
|
|
#endif
|
2006-01-12 00:28:41 +03:00
|
|
|
}
|
|
|
|
|
2008-08-07 00:48:55 +04:00
|
|
|
#ifdef MOZ_X11
|
2006-01-12 00:28:41 +03:00
|
|
|
void
|
|
|
|
nsScreenGtk :: Init (XineramaScreenInfo *aScreenInfo)
|
|
|
|
{
|
2009-01-15 06:27:09 +03:00
|
|
|
nsIntRect xineRect(aScreenInfo->x_org, aScreenInfo->y_org,
|
|
|
|
aScreenInfo->width, aScreenInfo->height);
|
2006-01-12 00:28:41 +03:00
|
|
|
|
|
|
|
mScreenNum = aScreenInfo->screen_number;
|
|
|
|
|
|
|
|
mAvailRect = mRect = xineRect;
|
|
|
|
}
|
2008-08-07 00:48:55 +04:00
|
|
|
#endif
|