These will be useful to keep track of changes across different places in the new UI. https://bugzilla.gnome.org/show_bug.cgi?id=785949
715 lines
21 KiB
C
715 lines
21 KiB
C
/*
|
|
* Copyright (C) 2016 Red Hat, Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
*/
|
|
|
|
#include "cc-display-config-rr.h"
|
|
|
|
#define GNOME_DESKTOP_USE_UNSTABLE_API
|
|
#include <libgnome-desktop/gnome-rr.h>
|
|
#include <libgnome-desktop/gnome-rr-config.h>
|
|
|
|
struct _CcDisplayModeRR
|
|
{
|
|
CcDisplayMode parent_instance;
|
|
|
|
GnomeRRMode *rr_mode;
|
|
};
|
|
|
|
G_DEFINE_TYPE (CcDisplayModeRR,
|
|
cc_display_mode_rr,
|
|
CC_TYPE_DISPLAY_MODE)
|
|
|
|
static void
|
|
cc_display_mode_rr_get_resolution (CcDisplayMode *pself,
|
|
int *w, int *h)
|
|
{
|
|
CcDisplayModeRR *self = CC_DISPLAY_MODE_RR (pself);
|
|
|
|
if (w)
|
|
*w = gnome_rr_mode_get_width (self->rr_mode);
|
|
if (h)
|
|
*h = gnome_rr_mode_get_height (self->rr_mode);
|
|
}
|
|
|
|
static double rr_supported_scales[] = { 1.0, 0.0 };
|
|
|
|
static const double *
|
|
cc_display_mode_rr_get_supported_scales (CcDisplayMode *pself)
|
|
{
|
|
return rr_supported_scales;
|
|
}
|
|
|
|
static double
|
|
cc_display_mode_rr_get_preferred_scale (CcDisplayMode *pself)
|
|
{
|
|
return rr_supported_scales[0];
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_mode_rr_is_interlaced (CcDisplayMode *pself)
|
|
{
|
|
CcDisplayModeRR *self = CC_DISPLAY_MODE_RR (pself);
|
|
|
|
return gnome_rr_mode_get_is_interlaced (self->rr_mode);
|
|
}
|
|
|
|
static int
|
|
cc_display_mode_rr_get_freq (CcDisplayMode *pself)
|
|
{
|
|
CcDisplayModeRR *self = CC_DISPLAY_MODE_RR (pself);
|
|
|
|
return gnome_rr_mode_get_freq (self->rr_mode);
|
|
}
|
|
|
|
static double
|
|
cc_display_mode_rr_get_freq_f (CcDisplayMode *pself)
|
|
{
|
|
CcDisplayModeRR *self = CC_DISPLAY_MODE_RR (pself);
|
|
|
|
return gnome_rr_mode_get_freq_f (self->rr_mode);
|
|
}
|
|
|
|
static void
|
|
cc_display_mode_rr_init (CcDisplayModeRR *self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
cc_display_mode_rr_finalize (GObject *object)
|
|
{
|
|
G_OBJECT_CLASS (cc_display_mode_rr_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
cc_display_mode_rr_class_init (CcDisplayModeRRClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
CcDisplayModeClass *parent_class = CC_DISPLAY_MODE_CLASS (klass);
|
|
|
|
gobject_class->finalize = cc_display_mode_rr_finalize;
|
|
|
|
parent_class->get_resolution = cc_display_mode_rr_get_resolution;
|
|
parent_class->get_supported_scales = cc_display_mode_rr_get_supported_scales;
|
|
parent_class->get_preferred_scale = cc_display_mode_rr_get_preferred_scale;
|
|
parent_class->is_interlaced = cc_display_mode_rr_is_interlaced;
|
|
parent_class->get_freq = cc_display_mode_rr_get_freq;
|
|
parent_class->get_freq_f = cc_display_mode_rr_get_freq_f;
|
|
}
|
|
|
|
static CcDisplayMode *
|
|
cc_display_mode_rr_new (GnomeRRMode *mode)
|
|
{
|
|
CcDisplayModeRR *self = g_object_new (CC_TYPE_DISPLAY_MODE_RR, NULL);
|
|
self->rr_mode = mode;
|
|
|
|
return CC_DISPLAY_MODE (self);
|
|
}
|
|
|
|
|
|
static const GnomeRRRotation rotation_map[] =
|
|
{
|
|
GNOME_RR_ROTATION_0,
|
|
GNOME_RR_ROTATION_90,
|
|
GNOME_RR_ROTATION_180,
|
|
GNOME_RR_ROTATION_270,
|
|
GNOME_RR_REFLECT_X | GNOME_RR_ROTATION_0,
|
|
GNOME_RR_REFLECT_X | GNOME_RR_ROTATION_90,
|
|
GNOME_RR_REFLECT_X | GNOME_RR_ROTATION_180,
|
|
GNOME_RR_REFLECT_X | GNOME_RR_ROTATION_270,
|
|
};
|
|
|
|
struct _CcDisplayMonitorRR
|
|
{
|
|
CcDisplayMonitor parent_instance;
|
|
CcDisplayConfigRR *config;
|
|
|
|
GnomeRROutput *output;
|
|
GnomeRROutputInfo *output_info;
|
|
|
|
GList *modes;
|
|
CcDisplayMode *current_mode;
|
|
CcDisplayMode *preferred_mode;
|
|
};
|
|
|
|
G_DEFINE_TYPE (CcDisplayMonitorRR,
|
|
cc_display_monitor_rr,
|
|
CC_TYPE_DISPLAY_MONITOR)
|
|
|
|
static void
|
|
cc_display_config_rr_set_primary (CcDisplayConfigRR *self,
|
|
CcDisplayMonitorRR *new_primary);
|
|
static void
|
|
cc_display_config_rr_unset_primary (CcDisplayConfigRR *self,
|
|
CcDisplayMonitorRR *old_primary);
|
|
|
|
static const char *
|
|
cc_display_monitor_rr_get_display_name (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_get_display_name (self->output_info);
|
|
}
|
|
|
|
static const char *
|
|
cc_display_monitor_rr_get_connector_name (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_get_name (self->output);
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_monitor_rr_is_builtin (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_is_builtin_display (self->output);
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_monitor_rr_is_primary (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_get_primary (self->output_info);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_primary (CcDisplayMonitor *pself,
|
|
gboolean primary)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
if (primary)
|
|
cc_display_config_rr_set_primary (self->config, self);
|
|
else
|
|
cc_display_config_rr_unset_primary (self->config, self);
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_monitor_rr_is_active (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_is_active (self->output_info);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_active (CcDisplayMonitor *pself,
|
|
gboolean active)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
gnome_rr_output_info_set_active (self->output_info, active);
|
|
g_signal_emit_by_name (self, "active");
|
|
}
|
|
|
|
static CcDisplayRotation
|
|
cc_display_monitor_rr_get_rotation (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
int i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (rotation_map); ++i)
|
|
if (rotation_map[i] == gnome_rr_output_info_get_rotation (self->output_info))
|
|
return i;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_rotation (CcDisplayMonitor *pself,
|
|
CcDisplayRotation rotation)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
gnome_rr_output_info_set_rotation (self->output_info,
|
|
rotation_map[rotation]);
|
|
g_signal_emit_by_name (self, "rotation");
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_monitor_rr_supports_rotation (CcDisplayMonitor *pself,
|
|
CcDisplayRotation rotation)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_supports_rotation (self->output_info,
|
|
rotation_map[rotation]);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_get_physical_size (CcDisplayMonitor *pself,
|
|
int *w, int *h)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_get_physical_size (self->output, w, h);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_get_geometry (CcDisplayMonitor *pself,
|
|
int *x, int *y, int *w, int *h)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_get_geometry (self->output_info, x, y, w, h);
|
|
}
|
|
|
|
static CcDisplayMode *
|
|
cc_display_monitor_rr_get_mode (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return self->current_mode;
|
|
}
|
|
|
|
static CcDisplayMode *
|
|
cc_display_monitor_rr_get_preferred_mode (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return self->preferred_mode;
|
|
}
|
|
|
|
static guint32
|
|
cc_display_monitor_rr_get_id (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_get_id (self->output);
|
|
}
|
|
|
|
static GList *
|
|
cc_display_monitor_rr_get_modes (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return self->modes;
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_monitor_rr_supports_underscanning (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_supports_underscanning (self->output);
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_monitor_rr_get_underscanning (CcDisplayMonitor *pself)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_get_underscanning (self->output_info);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_underscanning (CcDisplayMonitor *pself,
|
|
gboolean underscanning)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
|
|
return gnome_rr_output_info_set_underscanning (self->output_info, underscanning);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_mode (CcDisplayMonitor *pself,
|
|
CcDisplayMode *mode)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
int x, y, w, h;
|
|
|
|
gnome_rr_output_info_get_geometry (self->output_info, &x, &y, NULL, NULL);
|
|
cc_display_mode_get_resolution (mode, &w, &h);
|
|
gnome_rr_output_info_set_geometry (self->output_info, x, y, w, h);
|
|
|
|
gnome_rr_output_info_set_refresh_rate (self->output_info,
|
|
cc_display_mode_get_freq (mode));
|
|
g_signal_emit_by_name (self, "mode");
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_position (CcDisplayMonitor *pself,
|
|
int x, int y)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (pself);
|
|
int w, h;
|
|
|
|
gnome_rr_output_info_get_geometry (self->output_info, NULL, NULL, &w, &h);
|
|
gnome_rr_output_info_set_geometry (self->output_info, x, y, w, h);
|
|
}
|
|
|
|
static double
|
|
cc_display_monitor_rr_get_scale (CcDisplayMonitor *pself)
|
|
{
|
|
return rr_supported_scales[0];
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_set_scale (CcDisplayMonitor *pself,
|
|
double scale)
|
|
{
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_init (CcDisplayMonitorRR *self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_finalize (GObject *object)
|
|
{
|
|
CcDisplayMonitorRR *self = CC_DISPLAY_MONITOR_RR (object);
|
|
|
|
g_list_foreach (self->modes, (GFunc) g_object_unref, NULL);
|
|
g_clear_pointer (&self->modes, g_list_free);
|
|
|
|
G_OBJECT_CLASS (cc_display_monitor_rr_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
cc_display_monitor_rr_class_init (CcDisplayMonitorRRClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
CcDisplayMonitorClass *parent_class = CC_DISPLAY_MONITOR_CLASS (klass);
|
|
|
|
gobject_class->finalize = cc_display_monitor_rr_finalize;
|
|
|
|
parent_class->get_display_name = cc_display_monitor_rr_get_display_name;
|
|
parent_class->get_connector_name = cc_display_monitor_rr_get_connector_name;
|
|
parent_class->is_builtin = cc_display_monitor_rr_is_builtin;
|
|
parent_class->is_primary = cc_display_monitor_rr_is_primary;
|
|
parent_class->set_primary = cc_display_monitor_rr_set_primary;
|
|
parent_class->is_active = cc_display_monitor_rr_is_active;
|
|
parent_class->set_active = cc_display_monitor_rr_set_active;
|
|
parent_class->get_rotation = cc_display_monitor_rr_get_rotation;
|
|
parent_class->set_rotation = cc_display_monitor_rr_set_rotation;
|
|
parent_class->supports_rotation = cc_display_monitor_rr_supports_rotation;
|
|
parent_class->get_physical_size = cc_display_monitor_rr_get_physical_size;
|
|
parent_class->get_geometry = cc_display_monitor_rr_get_geometry;
|
|
parent_class->get_mode = cc_display_monitor_rr_get_mode;
|
|
parent_class->get_preferred_mode = cc_display_monitor_rr_get_preferred_mode;
|
|
parent_class->get_id = cc_display_monitor_rr_get_id;
|
|
parent_class->get_modes = cc_display_monitor_rr_get_modes;
|
|
parent_class->supports_underscanning = cc_display_monitor_rr_supports_underscanning;
|
|
parent_class->get_underscanning = cc_display_monitor_rr_get_underscanning;
|
|
parent_class->set_underscanning = cc_display_monitor_rr_set_underscanning;
|
|
parent_class->set_mode = cc_display_monitor_rr_set_mode;
|
|
parent_class->set_position = cc_display_monitor_rr_set_position;
|
|
parent_class->get_scale = cc_display_monitor_rr_get_scale;
|
|
parent_class->set_scale = cc_display_monitor_rr_set_scale;
|
|
}
|
|
|
|
static CcDisplayMonitor *
|
|
cc_display_monitor_rr_new (GnomeRROutput *output,
|
|
GnomeRROutputInfo *output_info,
|
|
CcDisplayConfigRR *config)
|
|
{
|
|
CcDisplayMonitorRR *self = g_object_new (CC_TYPE_DISPLAY_MONITOR_RR, NULL);
|
|
GnomeRRMode **modes = gnome_rr_output_list_modes (output);
|
|
GnomeRRMode *preferred_mode = gnome_rr_output_get_preferred_mode (output);
|
|
GnomeRRMode *current_mode = gnome_rr_output_get_current_mode (output);
|
|
gint i;
|
|
|
|
self->config = config;
|
|
self->output = output;
|
|
self->output_info = output_info;
|
|
|
|
for (i = 0; modes[i] != NULL; ++i)
|
|
{
|
|
CcDisplayMode *mode = cc_display_mode_rr_new (modes[i]);
|
|
self->modes = g_list_prepend (self->modes, mode);
|
|
|
|
if (current_mode &&
|
|
gnome_rr_mode_get_id (current_mode) == gnome_rr_mode_get_id (modes[i]))
|
|
self->current_mode = mode;
|
|
|
|
if (preferred_mode &&
|
|
gnome_rr_mode_get_id (preferred_mode) == gnome_rr_mode_get_id (modes[i]))
|
|
self->preferred_mode = mode;
|
|
}
|
|
|
|
return CC_DISPLAY_MONITOR (self);
|
|
}
|
|
|
|
struct _CcDisplayConfigRR
|
|
{
|
|
CcDisplayConfig parent_instance;
|
|
|
|
GnomeRRScreen *rr_screen;
|
|
GnomeRRConfig *rr_config;
|
|
|
|
GList *monitors;
|
|
CcDisplayMonitorRR *primary;
|
|
GList *clone_modes;
|
|
};
|
|
|
|
G_DEFINE_TYPE (CcDisplayConfigRR,
|
|
cc_display_config_rr,
|
|
CC_TYPE_DISPLAY_CONFIG)
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_GNOME_RR_SCREEN,
|
|
};
|
|
|
|
static void
|
|
cc_display_config_rr_set_primary (CcDisplayConfigRR *self,
|
|
CcDisplayMonitorRR *new_primary)
|
|
{
|
|
if (self->primary == new_primary)
|
|
return;
|
|
|
|
gnome_rr_output_info_set_primary (self->primary->output_info, FALSE);
|
|
g_signal_emit_by_name (self->primary, "primary");
|
|
|
|
self->primary = new_primary;
|
|
gnome_rr_output_info_set_primary (self->primary->output_info, TRUE);
|
|
g_signal_emit_by_name (self->primary, "primary");
|
|
|
|
g_signal_emit_by_name (self, "primary");
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_unset_primary (CcDisplayConfigRR *self,
|
|
CcDisplayMonitorRR *old_primary)
|
|
{
|
|
GList *l;
|
|
|
|
if (self->primary != old_primary)
|
|
return;
|
|
|
|
for (l = self->monitors; l != NULL; l = l->next)
|
|
{
|
|
CcDisplayMonitorRR *monitor = l->data;
|
|
if (gnome_rr_output_info_is_active (monitor->output_info) &&
|
|
monitor != old_primary)
|
|
{
|
|
cc_display_config_rr_set_primary (self, monitor);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static GList *
|
|
cc_display_config_rr_get_monitors (CcDisplayConfig *pself)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
|
|
return self->monitors;
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_config_rr_is_applicable (CcDisplayConfig *pself)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
|
|
return gnome_rr_config_applicable (self->rr_config, self->rr_screen, NULL);
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_config_rr_equal (CcDisplayConfig *pself,
|
|
CcDisplayConfig *pother)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
CcDisplayConfigRR *other = CC_DISPLAY_CONFIG_RR (pother);
|
|
gboolean same_modes, same_primary, same_cloning;
|
|
|
|
same_modes = gnome_rr_config_equal (self->rr_config, other->rr_config);
|
|
same_primary = (gnome_rr_output_get_id (self->primary->output) ==
|
|
gnome_rr_output_get_id (other->primary->output));
|
|
same_cloning = (gnome_rr_config_get_clone (self->rr_config) ==
|
|
gnome_rr_config_get_clone (other->rr_config));
|
|
return same_modes && same_primary && same_cloning;
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_config_rr_apply (CcDisplayConfig *pself,
|
|
GError **error)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
|
|
gnome_rr_config_sanitize (self->rr_config);
|
|
gnome_rr_config_ensure_primary (self->rr_config);
|
|
|
|
return gnome_rr_config_apply_persistent (self->rr_config, self->rr_screen, error);
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_config_rr_is_cloning (CcDisplayConfig *pself)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
|
|
return gnome_rr_config_get_clone (self->rr_config);
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_set_cloning (CcDisplayConfig *pself,
|
|
gboolean clone)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
|
|
return gnome_rr_config_set_clone (self->rr_config, clone);
|
|
}
|
|
|
|
static GList *
|
|
cc_display_config_rr_get_cloning_modes (CcDisplayConfig *pself)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (pself);
|
|
|
|
return self->clone_modes;
|
|
}
|
|
|
|
static gboolean
|
|
cc_display_config_rr_is_layout_logical (CcDisplayConfig *pself)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_init (CcDisplayConfigRR *self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_constructed (GObject *object)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (object);
|
|
GnomeRROutputInfo **output_infos;
|
|
GnomeRRMode **clone_modes;
|
|
gint i;
|
|
|
|
self->rr_config = gnome_rr_config_new_current (self->rr_screen, NULL);
|
|
gnome_rr_config_ensure_primary (self->rr_config);
|
|
|
|
output_infos = gnome_rr_config_get_outputs (self->rr_config);
|
|
|
|
for (i = 0; output_infos[i] != NULL; ++i)
|
|
{
|
|
GnomeRROutput *output;
|
|
CcDisplayMonitor *monitor;
|
|
|
|
if (!gnome_rr_output_info_is_primary_tile (output_infos[i]))
|
|
continue;
|
|
|
|
output = gnome_rr_screen_get_output_by_name (self->rr_screen,
|
|
gnome_rr_output_info_get_name (output_infos[i]));
|
|
|
|
monitor = cc_display_monitor_rr_new (output, output_infos[i], self);
|
|
self->monitors = g_list_prepend (self->monitors, monitor);
|
|
|
|
if (cc_display_monitor_is_primary (monitor))
|
|
self->primary = CC_DISPLAY_MONITOR_RR (monitor);
|
|
}
|
|
|
|
clone_modes = gnome_rr_screen_list_clone_modes (self->rr_screen);
|
|
|
|
for (i = 0; clone_modes[i] != NULL; ++i)
|
|
{
|
|
CcDisplayMode *mode = cc_display_mode_rr_new (clone_modes[i]);
|
|
self->clone_modes = g_list_prepend (self->clone_modes, mode);
|
|
}
|
|
|
|
G_OBJECT_CLASS (cc_display_config_rr_parent_class)->constructed (object);
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_GNOME_RR_SCREEN:
|
|
self->rr_screen = g_value_dup_object (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_GNOME_RR_SCREEN:
|
|
g_value_set_object (value, self->rr_screen);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_finalize (GObject *object)
|
|
{
|
|
CcDisplayConfigRR *self = CC_DISPLAY_CONFIG_RR (object);
|
|
|
|
g_clear_object (&self->rr_screen);
|
|
g_clear_object (&self->rr_config);
|
|
|
|
g_list_foreach (self->monitors, (GFunc) g_object_unref, NULL);
|
|
g_clear_pointer (&self->monitors, g_list_free);
|
|
|
|
g_list_foreach (self->clone_modes, (GFunc) g_object_unref, NULL);
|
|
g_clear_pointer (&self->clone_modes, g_list_free);
|
|
|
|
G_OBJECT_CLASS (cc_display_config_rr_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
cc_display_config_rr_class_init (CcDisplayConfigRRClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
CcDisplayConfigClass *parent_class = CC_DISPLAY_CONFIG_CLASS (klass);
|
|
GParamSpec *pspec;
|
|
|
|
gobject_class->constructed = cc_display_config_rr_constructed;
|
|
gobject_class->set_property = cc_display_config_rr_set_property;
|
|
gobject_class->get_property = cc_display_config_rr_get_property;
|
|
gobject_class->finalize = cc_display_config_rr_finalize;
|
|
|
|
parent_class->get_monitors = cc_display_config_rr_get_monitors;
|
|
parent_class->is_applicable = cc_display_config_rr_is_applicable;
|
|
parent_class->equal = cc_display_config_rr_equal;
|
|
parent_class->apply = cc_display_config_rr_apply;
|
|
parent_class->is_cloning = cc_display_config_rr_is_cloning;
|
|
parent_class->set_cloning = cc_display_config_rr_set_cloning;
|
|
parent_class->get_cloning_modes = cc_display_config_rr_get_cloning_modes;
|
|
parent_class->is_layout_logical = cc_display_config_rr_is_layout_logical;
|
|
|
|
pspec = g_param_spec_object ("gnome-rr-screen",
|
|
"GnomeRRScreen",
|
|
"GnomeRRScreen",
|
|
GNOME_TYPE_RR_SCREEN,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_STATIC_STRINGS |
|
|
G_PARAM_CONSTRUCT_ONLY);
|
|
g_object_class_install_property (gobject_class, PROP_GNOME_RR_SCREEN, pspec);
|
|
}
|