Add an initial implementation of the new design for the display panel. The display previews and presentation mode are not yet fully implemented. https://bugzilla.gnome.org/show_bug.cgi?id=706115
493 lines
13 KiB
C
493 lines
13 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
|
|
*
|
|
* cc-rr-labeler.c - Utility to label monitors to identify them
|
|
* while they are being configured.
|
|
*
|
|
* Copyright 2008, Novell, Inc.
|
|
*
|
|
* This file is part of the Gnome Library.
|
|
*
|
|
* The Gnome Library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* The Gnome 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with the Gnome Library; see the file COPYING.LIB. If not,
|
|
* write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*
|
|
* Author: Federico Mena-Quintero <federico@novell.com>
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <glib/gi18n-lib.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
#include <X11/Xproto.h>
|
|
#include <X11/Xlib.h>
|
|
#include <X11/Xutil.h>
|
|
#include <X11/Xatom.h>
|
|
#include <gdk/gdkx.h>
|
|
#endif
|
|
|
|
#include "cc-rr-labeler.h"
|
|
#include "cc-display-panel.h"
|
|
|
|
struct _CcRRLabelerPrivate {
|
|
GnomeRRConfig *config;
|
|
|
|
int num_outputs;
|
|
|
|
GtkWidget **windows;
|
|
|
|
GdkScreen *screen;
|
|
gboolean has_event_filter;
|
|
#ifdef GDK_WINDOWING_X11
|
|
Atom workarea_atom;
|
|
#endif
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_CONFIG,
|
|
PROP_LAST
|
|
};
|
|
|
|
G_DEFINE_TYPE (CcRRLabeler, cc_rr_labeler, G_TYPE_OBJECT);
|
|
|
|
static void cc_rr_labeler_finalize (GObject *object);
|
|
static void setup_from_config (CcRRLabeler *labeler);
|
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
static GdkFilterReturn
|
|
screen_xevent_filter (GdkXEvent *xevent,
|
|
GdkEvent *event,
|
|
CcRRLabeler *labeler)
|
|
{
|
|
XEvent *xev;
|
|
|
|
xev = (XEvent *) xevent;
|
|
|
|
if (xev->type == PropertyNotify &&
|
|
xev->xproperty.atom == labeler->priv->workarea_atom) {
|
|
/* update label positions */
|
|
if (labeler->priv->windows != NULL) {
|
|
cc_rr_labeler_hide (labeler);
|
|
cc_rr_labeler_show (labeler);
|
|
}
|
|
}
|
|
|
|
return GDK_FILTER_CONTINUE;
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
cc_rr_labeler_init (CcRRLabeler *labeler)
|
|
{
|
|
GdkWindow *gdkwindow;
|
|
|
|
labeler->priv = G_TYPE_INSTANCE_GET_PRIVATE (labeler, GNOME_TYPE_RR_LABELER, CcRRLabelerPrivate);
|
|
|
|
labeler->priv->screen = gdk_screen_get_default ();
|
|
#ifdef GDK_WINDOWING_X11
|
|
if (GDK_IS_X11_DISPLAY (gdk_display_get_default ())) {
|
|
labeler->priv->workarea_atom = XInternAtom (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()),
|
|
"_NET_WORKAREA",
|
|
True);
|
|
|
|
/* code is not really designed to handle multiple screens so *shrug* */
|
|
gdkwindow = gdk_screen_get_root_window (labeler->priv->screen);
|
|
gdk_window_add_filter (gdkwindow, (GdkFilterFunc) screen_xevent_filter, labeler);
|
|
gdk_window_set_events (gdkwindow, gdk_window_get_events (gdkwindow) | GDK_PROPERTY_CHANGE_MASK);
|
|
|
|
labeler->priv->has_event_filter = TRUE;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
cc_rr_labeler_set_property (GObject *gobject, guint property_id, const GValue *value, GParamSpec *param_spec)
|
|
{
|
|
CcRRLabeler *self = CC_RR_LABELER (gobject);
|
|
|
|
switch (property_id) {
|
|
case PROP_CONFIG:
|
|
self->priv->config = GNOME_RR_CONFIG (g_value_dup_object (value));
|
|
return;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, property_id, param_spec);
|
|
}
|
|
}
|
|
|
|
static GObject *
|
|
cc_rr_labeler_constructor (GType type, guint n_construct_properties, GObjectConstructParam *construct_properties)
|
|
{
|
|
CcRRLabeler *self = (CcRRLabeler*) G_OBJECT_CLASS (cc_rr_labeler_parent_class)->constructor (type, n_construct_properties, construct_properties);
|
|
|
|
setup_from_config (self);
|
|
|
|
return (GObject*) self;
|
|
}
|
|
|
|
static void
|
|
cc_rr_labeler_class_init (CcRRLabelerClass *klass)
|
|
{
|
|
GObjectClass *object_class;
|
|
|
|
g_type_class_add_private (klass, sizeof (CcRRLabelerPrivate));
|
|
|
|
object_class = (GObjectClass *) klass;
|
|
|
|
object_class->set_property = cc_rr_labeler_set_property;
|
|
object_class->finalize = cc_rr_labeler_finalize;
|
|
object_class->constructor = cc_rr_labeler_constructor;
|
|
|
|
g_object_class_install_property (object_class, PROP_CONFIG, g_param_spec_object ("config",
|
|
"Configuration",
|
|
"RandR configuration to label",
|
|
GNOME_TYPE_RR_CONFIG,
|
|
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB));
|
|
}
|
|
|
|
static void
|
|
cc_rr_labeler_finalize (GObject *object)
|
|
{
|
|
CcRRLabeler *labeler;
|
|
GdkWindow *gdkwindow;
|
|
|
|
labeler = CC_RR_LABELER (object);
|
|
|
|
if (labeler->priv->has_event_filter) {
|
|
gdkwindow = gdk_screen_get_root_window (labeler->priv->screen);
|
|
gdk_window_remove_filter (gdkwindow, (GdkFilterFunc) screen_xevent_filter, labeler);
|
|
}
|
|
|
|
if (labeler->priv->config != NULL) {
|
|
g_object_unref (labeler->priv->config);
|
|
}
|
|
|
|
if (labeler->priv->windows != NULL) {
|
|
cc_rr_labeler_hide (labeler);
|
|
g_free (labeler->priv->windows);
|
|
}
|
|
|
|
G_OBJECT_CLASS (cc_rr_labeler_parent_class)->finalize (object);
|
|
}
|
|
|
|
static int
|
|
count_outputs (GnomeRRConfig *config)
|
|
{
|
|
int i;
|
|
GnomeRROutputInfo **outputs = gnome_rr_config_get_outputs (config);
|
|
|
|
for (i = 0; outputs[i] != NULL; i++)
|
|
;
|
|
|
|
return i;
|
|
}
|
|
|
|
static void
|
|
rounded_rectangle (cairo_t *cr,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height,
|
|
gint x_radius,
|
|
gint y_radius)
|
|
{
|
|
gint x1, x2;
|
|
gint y1, y2;
|
|
gint xr1, xr2;
|
|
gint yr1, yr2;
|
|
|
|
x1 = x;
|
|
x2 = x1 + width;
|
|
y1 = y;
|
|
y2 = y1 + height;
|
|
|
|
x_radius = MIN (x_radius, width / 2.0);
|
|
y_radius = MIN (y_radius, width / 2.0);
|
|
|
|
xr1 = x_radius;
|
|
xr2 = x_radius / 2.0;
|
|
yr1 = y_radius;
|
|
yr2 = y_radius / 2.0;
|
|
|
|
cairo_move_to (cr, x1 + xr1, y1);
|
|
cairo_line_to (cr, x2 - xr1, y1);
|
|
cairo_curve_to (cr, x2 - xr2, y1, x2, y1 + yr2, x2, y1 + yr1);
|
|
cairo_line_to (cr, x2, y2 - yr1);
|
|
cairo_curve_to (cr, x2, y2 - yr2, x2 - xr2, y2, x2 - xr1, y2);
|
|
cairo_line_to (cr, x1 + xr1, y2);
|
|
cairo_curve_to (cr, x1 + xr2, y2, x1, y2 - yr2, x1, y2 - yr1);
|
|
cairo_line_to (cr, x1, y1 + yr1);
|
|
cairo_curve_to (cr, x1, y1 + yr2, x1 + xr2, y1, x1 + xr1, y1);
|
|
cairo_close_path (cr);
|
|
}
|
|
|
|
#define LABEL_WINDOW_SIZE 80
|
|
#define LABEL_WINDOW_MARGIN 14
|
|
#define LABEL_WINDOW_EDGE_THICKNESS 1
|
|
#define LABEL_WINDOW_PADDING 12
|
|
/* Look for panel-corner in:
|
|
* http://git.gnome.org/browse/gnome-shell/tree/data/theme/gnome-shell.css
|
|
* to match the corner radius */
|
|
#define LABEL_CORNER_RADIUS 6 + LABEL_WINDOW_EDGE_THICKNESS
|
|
|
|
static void
|
|
label_draw_background_and_frame (GtkWidget *widget, cairo_t *cr, gboolean for_shape)
|
|
{
|
|
GdkRGBA shape_color = { 0, 0, 0, 1 };
|
|
GdkRGBA black = { 0, 0, 0, 0.75 };
|
|
GtkAllocation allocation;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
cairo_save (cr);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
|
|
|
|
/* edge outline */
|
|
if (for_shape)
|
|
gdk_cairo_set_source_rgba (cr, &shape_color);
|
|
else
|
|
cairo_set_source_rgba (cr, 0.75, 0.75, 0.75, 0.75);
|
|
|
|
rounded_rectangle (cr,
|
|
LABEL_WINDOW_EDGE_THICKNESS / 2.0,
|
|
LABEL_WINDOW_EDGE_THICKNESS / 2.0,
|
|
allocation.width - LABEL_WINDOW_EDGE_THICKNESS,
|
|
allocation.height - LABEL_WINDOW_EDGE_THICKNESS,
|
|
LABEL_CORNER_RADIUS, LABEL_CORNER_RADIUS);
|
|
cairo_set_line_width (cr, LABEL_WINDOW_EDGE_THICKNESS);
|
|
cairo_stroke (cr);
|
|
|
|
/* fill */
|
|
if (for_shape) {
|
|
gdk_cairo_set_source_rgba (cr, &shape_color);
|
|
} else {
|
|
gdk_cairo_set_source_rgba (cr, &black);
|
|
}
|
|
|
|
rounded_rectangle (cr,
|
|
LABEL_WINDOW_EDGE_THICKNESS,
|
|
LABEL_WINDOW_EDGE_THICKNESS,
|
|
allocation.width - LABEL_WINDOW_EDGE_THICKNESS * 2,
|
|
allocation.height - LABEL_WINDOW_EDGE_THICKNESS * 2,
|
|
LABEL_CORNER_RADIUS - LABEL_WINDOW_EDGE_THICKNESS / 2.0,
|
|
LABEL_CORNER_RADIUS - LABEL_WINDOW_EDGE_THICKNESS / 2.0);
|
|
cairo_fill (cr);
|
|
|
|
cairo_restore (cr);
|
|
}
|
|
|
|
static gboolean
|
|
label_window_draw_event_cb (GtkWidget *widget, cairo_t *cr, gpointer data)
|
|
{
|
|
/* clear any content */
|
|
cairo_save (cr);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
|
|
cairo_set_source_rgba (cr, 0, 0, 0, 0);
|
|
cairo_paint (cr);
|
|
cairo_restore (cr);
|
|
|
|
gtk_widget_shape_combine_region (widget, NULL);
|
|
label_draw_background_and_frame (widget, cr, FALSE);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
position_window (CcRRLabeler *labeler,
|
|
GtkWidget *window,
|
|
int x,
|
|
int y)
|
|
{
|
|
GdkRectangle workarea;
|
|
GdkRectangle monitor;
|
|
int monitor_num;
|
|
|
|
monitor_num = gdk_screen_get_monitor_at_point (labeler->priv->screen, x, y);
|
|
gdk_screen_get_monitor_workarea (labeler->priv->screen, monitor_num, &workarea);
|
|
gdk_screen_get_monitor_geometry (labeler->priv->screen,
|
|
monitor_num,
|
|
&monitor);
|
|
gdk_rectangle_intersect (&monitor, &workarea, &workarea);
|
|
|
|
gtk_window_move (GTK_WINDOW (window), workarea.x + LABEL_WINDOW_MARGIN,
|
|
workarea.y + LABEL_WINDOW_MARGIN);
|
|
}
|
|
|
|
static void
|
|
label_window_realize_cb (GtkWidget *widget)
|
|
{
|
|
cairo_region_t *region;
|
|
|
|
/* make the whole window ignore events */
|
|
region = cairo_region_create ();
|
|
gtk_widget_input_shape_combine_region (widget, region);
|
|
cairo_region_destroy (region);
|
|
|
|
gtk_widget_shape_combine_region (widget, NULL);
|
|
}
|
|
|
|
static void
|
|
label_window_composited_changed_cb (GtkWidget *widget, CcRRLabeler *labeler)
|
|
{
|
|
if (gtk_widget_get_realized (widget))
|
|
gtk_widget_shape_combine_region (widget, NULL);
|
|
}
|
|
|
|
static GtkWidget *
|
|
create_label_window (CcRRLabeler *labeler, GnomeRROutputInfo *output)
|
|
{
|
|
GtkWidget *window;
|
|
GtkWidget *widget;
|
|
char *str;
|
|
GdkRGBA white = { 1, 1, 1, 1 };
|
|
int x, y, display_num;
|
|
GdkScreen *screen;
|
|
GdkVisual *visual;
|
|
|
|
display_num = cc_display_panel_get_output_id (output);
|
|
|
|
if (display_num == 0)
|
|
return NULL;
|
|
|
|
window = gtk_window_new (GTK_WINDOW_POPUP);
|
|
gtk_window_set_type_hint (GTK_WINDOW (window), GDK_WINDOW_TYPE_HINT_TOOLTIP);
|
|
gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
|
|
gtk_widget_set_app_paintable (window, TRUE);
|
|
screen = gtk_widget_get_screen (window);
|
|
visual = gdk_screen_get_rgba_visual (screen);
|
|
|
|
if (visual != NULL)
|
|
gtk_widget_set_visual (window, visual);
|
|
|
|
gtk_widget_set_size_request (window, LABEL_WINDOW_SIZE,
|
|
LABEL_WINDOW_SIZE);
|
|
|
|
g_signal_connect (window, "draw",
|
|
G_CALLBACK (label_window_draw_event_cb), labeler);
|
|
g_signal_connect (window, "realize",
|
|
G_CALLBACK (label_window_realize_cb), labeler);
|
|
g_signal_connect (window, "composited-changed",
|
|
G_CALLBACK (label_window_composited_changed_cb), labeler);
|
|
|
|
str = g_strdup_printf ("<span size='xx-large' font-weight='bold'>%d</span>", display_num);
|
|
widget = gtk_label_new (NULL);
|
|
gtk_label_set_markup (GTK_LABEL (widget), str);
|
|
g_free (str);
|
|
|
|
/* Make the label explicitly white. We don't want it to follow the
|
|
* theme's colors, since the label is always shown against a black
|
|
* background. See bgo#556050
|
|
*/
|
|
gtk_widget_override_color (widget,
|
|
gtk_widget_get_state_flags (widget),
|
|
&white);
|
|
|
|
gtk_container_add (GTK_CONTAINER (window), widget);
|
|
|
|
/* Should we center this at the top edge of the monitor, instead of using the upper-left corner? */
|
|
gnome_rr_output_info_get_geometry (output, &x, &y, NULL, NULL);
|
|
position_window (labeler, window, x, y);
|
|
|
|
gtk_widget_show_all (window);
|
|
|
|
return window;
|
|
}
|
|
|
|
static void
|
|
setup_from_config (CcRRLabeler *labeler)
|
|
{
|
|
labeler->priv->num_outputs = count_outputs (labeler->priv->config);
|
|
|
|
cc_rr_labeler_show (labeler);
|
|
}
|
|
|
|
/**
|
|
* cc_rr_labeler_new:
|
|
* @config: Configuration of the screens to label
|
|
*
|
|
* Create a GUI element that will display colored labels on each connected monitor.
|
|
* This is useful when users are required to identify which monitor is which, e.g. for
|
|
* for configuring multiple monitors.
|
|
* The labels will be shown by default, use cc_rr_labeler_hide to hide them.
|
|
*
|
|
* Returns: A new #CcRRLabeler
|
|
*/
|
|
CcRRLabeler *
|
|
cc_rr_labeler_new (GnomeRRConfig *config)
|
|
{
|
|
g_return_val_if_fail (GNOME_IS_RR_CONFIG (config), NULL);
|
|
|
|
return g_object_new (GNOME_TYPE_RR_LABELER, "config", config, NULL);
|
|
}
|
|
|
|
/**
|
|
* cc_rr_labeler_show:
|
|
* @labeler: A #CcRRLabeler
|
|
*
|
|
* Show the labels.
|
|
*/
|
|
void
|
|
cc_rr_labeler_show (CcRRLabeler *labeler)
|
|
{
|
|
int i;
|
|
GnomeRROutputInfo **outputs;
|
|
|
|
g_return_if_fail (GNOME_IS_RR_LABELER (labeler));
|
|
|
|
if (labeler->priv->windows != NULL)
|
|
return;
|
|
|
|
if (gnome_rr_config_get_clone (labeler->priv->config))
|
|
return;
|
|
|
|
outputs = gnome_rr_config_get_outputs (labeler->priv->config);
|
|
|
|
labeler->priv->windows = g_new (GtkWidget *, labeler->priv->num_outputs);
|
|
|
|
for (i = 0; i < labeler->priv->num_outputs; i++) {
|
|
if (gnome_rr_output_info_is_active (outputs[i])) {
|
|
labeler->priv->windows[i] = create_label_window (labeler, outputs[i]);
|
|
} else
|
|
labeler->priv->windows[i] = NULL;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* cc_rr_labeler_hide:
|
|
* @labeler: A #CcRRLabeler
|
|
*
|
|
* Hide ouput labels.
|
|
*/
|
|
void
|
|
cc_rr_labeler_hide (CcRRLabeler *labeler)
|
|
{
|
|
int i;
|
|
CcRRLabelerPrivate *priv;
|
|
|
|
g_return_if_fail (GNOME_IS_RR_LABELER (labeler));
|
|
|
|
priv = labeler->priv;
|
|
|
|
if (priv->windows == NULL)
|
|
return;
|
|
|
|
for (i = 0; i < priv->num_outputs; i++)
|
|
if (priv->windows[i] != NULL) {
|
|
gtk_widget_destroy (priv->windows[i]);
|
|
priv->windows[i] = NULL;
|
|
}
|
|
g_free (priv->windows);
|
|
priv->windows = NULL;
|
|
}
|